1 /*
2 * Copyright (C) 2008 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 FILE_LICENCE ( GPL2_OR_LATER );
20
21 #include <stdint.h>
22 #include <stdlib.h>
23 #include <stdio.h>
24 #include <errno.h>
25 #include <string.h>
26 #include <gpxe/settings.h>
27 #include <gpxe/netdevice.h>
28 #include <gpxe/dhcppkt.h>
29 #include <gpxe/fakedhcp.h>
30
31 /** @file
32 *
33 * Fake DHCP packets
34 *
35 */
36
37 /**
38 * Copy settings to DHCP packet
39 *
40 * @v dest Destination DHCP packet
41 * @v source Source settings block
42 * @v encapsulator Encapsulating setting tag number, or zero
43 * @ret rc Return status code
44 */
copy_encap_settings(struct dhcp_packet * dest,struct settings * source,unsigned int encapsulator)45 static int copy_encap_settings ( struct dhcp_packet *dest,
46 struct settings *source,
47 unsigned int encapsulator ) {
48 struct setting setting = { .name = "" };
49 unsigned int subtag;
50 unsigned int tag;
51 int len;
52 int check_len;
53 int rc;
54
55 for ( subtag = DHCP_MIN_OPTION; subtag <= DHCP_MAX_OPTION; subtag++ ) {
56 tag = DHCP_ENCAP_OPT ( encapsulator, subtag );
57 switch ( tag ) {
58 case DHCP_EB_ENCAP:
59 case DHCP_VENDOR_ENCAP:
60 /* Process encapsulated settings */
61 if ( ( rc = copy_encap_settings ( dest, source,
62 tag ) ) != 0 )
63 return rc;
64 break;
65 default:
66 /* Copy setting, if present */
67 setting.tag = tag;
68 len = fetch_setting_len ( source, &setting );
69 if ( len < 0 )
70 break;
71 {
72 char buf[len];
73
74 check_len = fetch_setting ( source, &setting,
75 buf, sizeof (buf));
76 assert ( check_len == len );
77 if ( ( rc = dhcppkt_store ( dest, tag, buf,
78 sizeof(buf) )) !=0)
79 return rc;
80 }
81 break;
82 }
83 }
84
85 return 0;
86 }
87
88 /**
89 * Copy settings to DHCP packet
90 *
91 * @v dest Destination DHCP packet
92 * @v source Source settings block
93 * @ret rc Return status code
94 */
copy_settings(struct dhcp_packet * dest,struct settings * source)95 static int copy_settings ( struct dhcp_packet *dest,
96 struct settings *source ) {
97 return copy_encap_settings ( dest, source, 0 );
98 }
99
100 /**
101 * Create fake DHCPDISCOVER packet
102 *
103 * @v netdev Network device
104 * @v data Buffer for DHCP packet
105 * @v max_len Size of DHCP packet buffer
106 * @ret rc Return status code
107 *
108 * Used by external code.
109 */
create_fakedhcpdiscover(struct net_device * netdev,void * data,size_t max_len)110 int create_fakedhcpdiscover ( struct net_device *netdev,
111 void *data, size_t max_len ) {
112 struct dhcp_packet dhcppkt;
113 struct in_addr ciaddr = { 0 };
114 int rc;
115
116 if ( ( rc = dhcp_create_request ( &dhcppkt, netdev, DHCPDISCOVER,
117 ciaddr, data, max_len ) ) != 0 ) {
118 DBG ( "Could not create DHCPDISCOVER: %s\n",
119 strerror ( rc ) );
120 return rc;
121 }
122
123 return 0;
124 }
125
126 /**
127 * Create fake DHCPACK packet
128 *
129 * @v netdev Network device
130 * @v data Buffer for DHCP packet
131 * @v max_len Size of DHCP packet buffer
132 * @ret rc Return status code
133 *
134 * Used by external code.
135 */
create_fakedhcpack(struct net_device * netdev,void * data,size_t max_len)136 int create_fakedhcpack ( struct net_device *netdev,
137 void *data, size_t max_len ) {
138 struct dhcp_packet dhcppkt;
139 int rc;
140
141 /* Create base DHCPACK packet */
142 if ( ( rc = dhcp_create_packet ( &dhcppkt, netdev, DHCPACK, NULL, 0,
143 data, max_len ) ) != 0 ) {
144 DBG ( "Could not create DHCPACK: %s\n", strerror ( rc ) );
145 return rc;
146 }
147
148 /* Merge in globally-scoped settings, then netdev-specific
149 * settings. Do it in this order so that netdev-specific
150 * settings take precedence regardless of stated priorities.
151 */
152 if ( ( rc = copy_settings ( &dhcppkt, NULL ) ) != 0 ) {
153 DBG ( "Could not set DHCPACK global settings: %s\n",
154 strerror ( rc ) );
155 return rc;
156 }
157 if ( ( rc = copy_settings ( &dhcppkt,
158 netdev_settings ( netdev ) ) ) != 0 ) {
159 DBG ( "Could not set DHCPACK netdev settings: %s\n",
160 strerror ( rc ) );
161 return rc;
162 }
163
164 return 0;
165 }
166
167 /**
168 * Create fake PXE Boot Server ACK packet
169 *
170 * @v netdev Network device
171 * @v data Buffer for DHCP packet
172 * @v max_len Size of DHCP packet buffer
173 * @ret rc Return status code
174 *
175 * Used by external code.
176 */
create_fakepxebsack(struct net_device * netdev,void * data,size_t max_len)177 int create_fakepxebsack ( struct net_device *netdev,
178 void *data, size_t max_len ) {
179 struct dhcp_packet dhcppkt;
180 struct settings *proxy_settings;
181 struct settings *pxebs_settings;
182 int rc;
183
184 /* Identify available settings */
185 proxy_settings = find_settings ( PROXYDHCP_SETTINGS_NAME );
186 pxebs_settings = find_settings ( PXEBS_SETTINGS_NAME );
187 if ( ( ! proxy_settings ) && ( ! pxebs_settings ) ) {
188 /* No PXE boot server; return the regular DHCPACK */
189 return create_fakedhcpack ( netdev, data, max_len );
190 }
191
192 /* Create base DHCPACK packet */
193 if ( ( rc = dhcp_create_packet ( &dhcppkt, netdev, DHCPACK, NULL, 0,
194 data, max_len ) ) != 0 ) {
195 DBG ( "Could not create PXE BS ACK: %s\n",
196 strerror ( rc ) );
197 return rc;
198 }
199
200 /* Merge in ProxyDHCP options */
201 if ( proxy_settings &&
202 ( ( rc = copy_settings ( &dhcppkt, proxy_settings ) ) != 0 ) ) {
203 DBG ( "Could not copy ProxyDHCP settings: %s\n",
204 strerror ( rc ) );
205 return rc;
206 }
207
208 /* Merge in BootServerDHCP options, if present */
209 if ( pxebs_settings &&
210 ( ( rc = copy_settings ( &dhcppkt, pxebs_settings ) ) != 0 ) ) {
211 DBG ( "Could not copy PXE BS settings: %s\n",
212 strerror ( rc ) );
213 return rc;
214 }
215
216 return 0;
217 }
218