4f8cebb6d065fecd81f1cfb9a8c6de7ab3023c50
[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/icmp6.h>
28 #include <gpxe/monojob.h>
29 #include <gpxe/process.h>
30 #include <usr/ifmgmt.h>
31 #include <usr/ip6mgmt.h>
32
33 #define LINK_WAIT_MS    15000
34
35 /* Maximum length of the link-layer address we'll insert as an EUI-64. */
36 #define AUTOCONF_LL_MAX 6
37
38 int ip6_autoconf ( struct net_device *netdev ) {
39         struct in6_addr ip6addr, ip6zero;
40         size_t ll_size;
41         int rc;
42
43         /* Check we can open the interface first */
44         if ( ( rc = ifopen ( netdev ) ) != 0 )
45                 return rc;
46
47         /* Wait for link-up */
48         if ( ( rc = iflinkwait ( netdev, LINK_WAIT_MS ) ) != 0 )
49                 return rc;
50         
51         /* Create the host ID part of the IPv6 address from the Link-Layer
52          * address on the netdevice. */
53         memset ( &ip6addr, 0, sizeof (struct in6_addr) );
54         memset ( &ip6zero, 0, sizeof (struct in6_addr) );
55         
56         ll_size = netdev->ll_protocol->ll_addr_len;
57         if ( ll_size < 6 ) {
58                 memcpy ( ip6addr.s6_addr + (8 - ll_size), netdev->ll_addr, ll_size );
59         } else {
60                 /* Create an EUI-64 identifier. */
61                 memcpy( ip6addr.s6_addr + 8, netdev->ll_addr, 3 );
62                 memcpy( ip6addr.s6_addr + 8 + 5, netdev->ll_addr + 3, 3 );
63                 ip6addr.s6_addr[11] = 0xFF;
64                 ip6addr.s6_addr[12] = 0xFE;
65                 
66                 /* Designate that this is in fact an EUI-64. */
67                 ip6addr.s6_addr[8] |= 0x2;
68         }
69         
70         /* Fill in the link-local prefix. */
71         ip6addr.s6_addr[0] = 0xFE;
72         ip6addr.s6_addr[1] = 0x80;
73         
74         /* TODO: send a few neighbour solicits on this address before we take
75          * it (once NDP is implemented). */
76         
77         DBG( "ipv6 autoconfig address is %s\n", inet6_ntoa(ip6addr) );
78         
79         /* Add as a route. */
80         add_ipv6_address ( netdev, ip6addr, 10, ip6addr, ip6zero );
81         
82         /* Solicit routers on the network. */
83         icmp6_send_rsolicit ( netdev );
84         
85         return 0;
86 }
87