7c15b5f9b9f00830e269f98c6d74c5d27f15f568
[people/dverkamp/gpxe.git] / src / usr / dhcpmgmt.c
1 /*
2  * Copyright (C) 2007 Michael Brown <mbrown@fensystems.co.uk>.
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 #include <string.h>
20 #include <stdio.h>
21 #include <byteswap.h>
22 #include <gpxe/in.h>
23 #include <gpxe/ip.h>
24 #include <gpxe/dhcp.h>
25 #include <gpxe/async.h>
26 #include <gpxe/netdevice.h>
27 #include <gpxe/dns.h>
28 #include <usr/ifmgmt.h>
29 #include <usr/dhcpmgmt.h>
30
31 /** @file
32  *
33  * DHCP management
34  *
35  */
36
37 /* Avoid dragging in dns.o */
38 struct in_addr nameserver;
39
40 /* Avoid dragging in syslog.o */
41 struct in_addr syslogserver;
42
43 /**
44  * Configure network device via DHCP
45  *
46  * @v netdev            Network device
47  * @ret rc              Return status code
48  */
49 int dhcp ( struct net_device *netdev ) {
50         static struct dhcp_option_block *dhcp_options = NULL;
51         struct dhcp_session dhcp;
52         struct in_addr address = { 0 };
53         struct in_addr netmask = { 0 };
54         struct in_addr gateway = { INADDR_NONE };
55         struct async async;
56         int rc;
57
58         /* Check we can open the interface first */
59         if ( ( rc = ifopen ( netdev ) ) != 0 )
60                 return rc;
61
62         /* Free up any previously-acquired options */
63         if ( dhcp_options ) {
64                 unregister_dhcp_options ( dhcp_options );
65                 free_dhcp_options ( dhcp_options );
66                 dhcp_options = NULL;
67         }
68
69         /* Clear any existing routing table entry */
70         del_ipv4_address ( netdev );
71
72         /* Issue DHCP request */
73         printf ( "DHCP (%s %s)...", netdev->name, netdev_hwaddr ( netdev ) );
74         memset ( &dhcp, 0, sizeof ( dhcp ) );
75         dhcp.netdev = netdev;
76         if ( ( rc = async_block ( &async,
77                                   start_dhcp ( &dhcp, &async ) ) ) != 0 ) {
78                 printf ( "failed (%s)\n", strerror ( rc ) );
79                 return rc;
80         }
81         printf ( "done\n" );
82
83         /* Store and register options */
84         dhcp_options = dhcp.options;
85         register_dhcp_options ( dhcp_options );
86
87         /* Retrieve IP address configuration */
88         find_dhcp_ipv4_option ( dhcp_options, DHCP_EB_YIADDR, &address );
89         find_dhcp_ipv4_option ( dhcp_options, DHCP_SUBNET_MASK, &netmask );
90         find_dhcp_ipv4_option ( dhcp_options, DHCP_ROUTERS, &gateway );
91
92         /* Set up new IP address configuration */
93         if ( ( rc = add_ipv4_address ( netdev, address, netmask,
94                                        gateway ) ) != 0 ) {
95                 printf ( "Could not configure %s with DHCP results: %s\n",
96                          netdev->name, strerror ( rc ) );
97                 return rc;
98         }
99
100         /* Retrieve other DHCP options that we care about */
101         find_dhcp_ipv4_option ( dhcp_options, DHCP_DNS_SERVERS,
102                                 &nameserver );
103         find_dhcp_ipv4_option ( dhcp_options, DHCP_LOG_SERVERS,
104                                 &syslogserver );
105
106         return 0;
107 }