[ipv6] Modify "ipv6" command to set an IPv6 address on an interface
[people/meteger/gpxe.git] / src / usr / ip6mgmt.c
1 /*
2  * Copyright (C) 2011 Matthew Iselin <matthew@theiselins.net>.
3  *
4  * This program is free software; you can redistribute it and/or
5  * modify it under the terms of the GNU General Public License as
6  * published by the Free Software Foundation; either version 2 of the
7  * License, or any later version.
8  *
9  * This program is distributed in the hope that it will be useful, but
10  * WITHOUT ANY WARRANTY; without even the implied warranty of
11  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
12  * General Public License for more details.
13  *
14  * You should have received a copy of the GNU General Public License
15  * along with this program; if not, write to the Free Software
16  * Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
17  */
18
19 FILE_LICENCE ( GPL2_OR_LATER );
20
21 #include <string.h>
22 #include <stdio.h>
23 #include <errno.h>
24 #include <gpxe/netdevice.h>
25 #include <gpxe/in.h>
26 #include <gpxe/ip6.h>
27 #include <gpxe/monojob.h>
28 #include <gpxe/process.h>
29 #include <usr/ifmgmt.h>
30 #include <usr/ip6mgmt.h>
31
32 #define LINK_WAIT_MS    15000
33
34 /* Maximum length of the link-layer address we'll insert as an EUI-64. */
35 #define AUTOCONF_LL_MAX 6
36
37 int ip6_autoconf ( struct net_device *netdev ) {
38         struct in6_addr ip6addr, ip6zero;
39         size_t ll_size;
40         int rc;
41
42         /* Check we can open the interface first */
43         if ( ( rc = ifopen ( netdev ) ) != 0 )
44                 return rc;
45
46         /* Wait for link-up */
47         if ( ( rc = iflinkwait ( netdev, LINK_WAIT_MS ) ) != 0 )
48                 return rc;
49         
50         /* Create the host ID part of the IPv6 address from the Link-Layer
51          * address on the netdevice. */
52         memset ( &ip6addr, 0, sizeof (struct in6_addr) );
53         memset ( &ip6zero, 0, sizeof (struct in6_addr) );
54         
55         ll_size = netdev->ll_protocol->ll_addr_len;
56         if ( ll_size < 6 ) {
57                 memcpy ( ip6addr.s6_addr + (8 - ll_size), netdev->ll_addr, ll_size );
58         } else {
59                 /* Create an EUI-64 identifier. */
60                 memcpy( ip6addr.s6_addr + 8, netdev->ll_addr, 3 );
61                 memcpy( ip6addr.s6_addr + 8 + 5, netdev->ll_addr + 3, 3 );
62                 ip6addr.s6_addr[11] = 0xFF;
63                 ip6addr.s6_addr[12] = 0xFE;
64                 
65                 /* Designate that this is in fact an EUI-64. */
66                 ip6addr.s6_addr[8] |= 0x2;
67         }
68         
69         /* Fill in the link-local prefix. */
70         ip6addr.s6_addr[0] = 0xFE;
71         ip6addr.s6_addr[1] = 0x80;
72         
73         /* TODO: send a few neighbour solicits on this address before we take
74          * it (once NDP is implemented). */
75         
76         DBG( "ipv6 autoconfig address is %s\n", inet6_ntoa(ip6addr) );
77         
78         /* Add as a route. */
79         add_ipv6_address ( netdev, ip6addr, 10, ip6addr, ip6zero );
80         
81         return 0;
82 }
83