Book a Demo!
CoCalc Logo Icon
StoreFeaturesDocsShareSupportNewsAboutPoliciesSign UpSign In
folium-app
GitHub Repository: folium-app/Folium
Path: blob/a-new-beginning/SharedDependencies/Sources/libslirp/bootp.c
2 views
1
/* SPDX-License-Identifier: MIT */
2
/*
3
* QEMU BOOTP/DHCP server
4
*
5
* Copyright (c) 2004 Fabrice Bellard
6
*
7
* Permission is hereby granted, free of charge, to any person obtaining a copy
8
* of this software and associated documentation files (the "Software"), to deal
9
* in the Software without restriction, including without limitation the rights
10
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
11
* copies of the Software, and to permit persons to whom the Software is
12
* furnished to do so, subject to the following conditions:
13
*
14
* The above copyright notice and this permission notice shall be included in
15
* all copies or substantial portions of the Software.
16
*
17
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
18
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
19
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
20
* THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
21
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
22
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
23
* THE SOFTWARE.
24
*/
25
#include "slirp.h"
26
27
#if defined(_WIN32)
28
/* Windows ntohl() returns an u_long value.
29
* Add a type cast to match the format strings. */
30
#define ntohl(n) ((uint32_t)ntohl(n))
31
#endif
32
33
/* XXX: only DHCP is supported */
34
35
#define LEASE_TIME (24 * 3600)
36
37
#define UEFI_HTTP_VENDOR_CLASS_ID "HTTPClient"
38
39
static const uint8_t rfc1533_cookie[] = { RFC1533_COOKIE };
40
41
#define DPRINTF(...) DEBUG_RAW_CALL(__VA_ARGS__)
42
43
static BOOTPClient *get_new_addr(Slirp *slirp, struct in_addr *paddr,
44
const uint8_t *macaddr)
45
{
46
BOOTPClient *bc;
47
int i;
48
49
for (i = 0; i < NB_BOOTP_CLIENTS; i++) {
50
bc = &slirp->bootp_clients[i];
51
if (!bc->allocated || !memcmp(macaddr, bc->macaddr, 6))
52
goto found;
53
}
54
return NULL;
55
found:
56
bc = &slirp->bootp_clients[i];
57
bc->allocated = 1;
58
paddr->s_addr = slirp->vdhcp_startaddr.s_addr + htonl(i);
59
return bc;
60
}
61
62
static BOOTPClient *request_addr(Slirp *slirp, const struct in_addr *paddr,
63
const uint8_t *macaddr)
64
{
65
uint32_t req_addr = ntohl(paddr->s_addr);
66
uint32_t dhcp_addr = ntohl(slirp->vdhcp_startaddr.s_addr);
67
BOOTPClient *bc;
68
69
if (req_addr >= dhcp_addr && req_addr < (dhcp_addr + NB_BOOTP_CLIENTS)) {
70
bc = &slirp->bootp_clients[req_addr - dhcp_addr];
71
if (!bc->allocated || !memcmp(macaddr, bc->macaddr, 6)) {
72
bc->allocated = 1;
73
return bc;
74
}
75
}
76
return NULL;
77
}
78
79
static BOOTPClient *find_addr(Slirp *slirp, struct in_addr *paddr,
80
const uint8_t *macaddr)
81
{
82
BOOTPClient *bc;
83
int i;
84
85
for (i = 0; i < NB_BOOTP_CLIENTS; i++) {
86
if (!memcmp(macaddr, slirp->bootp_clients[i].macaddr, 6))
87
goto found;
88
}
89
return NULL;
90
found:
91
bc = &slirp->bootp_clients[i];
92
bc->allocated = 1;
93
paddr->s_addr = slirp->vdhcp_startaddr.s_addr + htonl(i);
94
return bc;
95
}
96
97
static void dhcp_decode(const struct bootp_t *bp,
98
const uint8_t *bp_end,
99
int *pmsg_type,
100
struct in_addr *preq_addr)
101
{
102
const uint8_t *p;
103
int len, tag;
104
105
*pmsg_type = 0;
106
preq_addr->s_addr = htonl(0L);
107
108
p = bp->bp_vend;
109
if (memcmp(p, rfc1533_cookie, 4) != 0)
110
return;
111
p += 4;
112
while (p < bp_end) {
113
tag = p[0];
114
if (tag == RFC1533_PAD) {
115
p++;
116
} else if (tag == RFC1533_END) {
117
break;
118
} else {
119
p++;
120
if (p >= bp_end)
121
break;
122
len = *p++;
123
if (p + len > bp_end) {
124
break;
125
}
126
DPRINTF("dhcp: tag=%d len=%d\n", tag, len);
127
128
switch (tag) {
129
case RFC2132_MSG_TYPE:
130
if (len >= 1)
131
*pmsg_type = p[0];
132
break;
133
case RFC2132_REQ_ADDR:
134
if (len >= 4) {
135
memcpy(&(preq_addr->s_addr), p, 4);
136
}
137
break;
138
default:
139
break;
140
}
141
p += len;
142
}
143
}
144
if (*pmsg_type == DHCPREQUEST && preq_addr->s_addr == htonl(0L) &&
145
bp->bp_ciaddr.s_addr) {
146
memcpy(&(preq_addr->s_addr), &bp->bp_ciaddr, 4);
147
}
148
}
149
150
static void bootp_reply(Slirp *slirp,
151
const struct bootp_t *bp,
152
const uint8_t *bp_end)
153
{
154
BOOTPClient *bc = NULL;
155
struct mbuf *m;
156
struct bootp_t *rbp;
157
struct sockaddr_in saddr, daddr;
158
struct in_addr preq_addr;
159
int dhcp_msg_type, val;
160
uint8_t *q;
161
uint8_t *end;
162
uint8_t client_ethaddr[ETH_ALEN];
163
164
/* extract exact DHCP msg type */
165
dhcp_decode(bp, bp_end, &dhcp_msg_type, &preq_addr);
166
DPRINTF("bootp packet op=%d msgtype=%d", bp->bp_op, dhcp_msg_type);
167
if (preq_addr.s_addr != htonl(0L))
168
DPRINTF(" req_addr=%08" PRIx32 "\n", ntohl(preq_addr.s_addr));
169
else {
170
DPRINTF("\n");
171
}
172
173
if (dhcp_msg_type == 0)
174
dhcp_msg_type = DHCPREQUEST; /* Force reply for old BOOTP clients */
175
176
if (dhcp_msg_type != DHCPDISCOVER && dhcp_msg_type != DHCPREQUEST)
177
return;
178
179
/* Get client's hardware address from bootp request */
180
memcpy(client_ethaddr, bp->bp_hwaddr, ETH_ALEN);
181
182
m = m_get(slirp);
183
if (!m) {
184
return;
185
}
186
m->m_data += IF_MAXLINKHDR;
187
m_inc(m, sizeof(struct bootp_t) + DHCP_OPT_LEN);
188
rbp = (struct bootp_t *)m->m_data;
189
m->m_data += sizeof(struct udpiphdr);
190
memset(rbp, 0, sizeof(struct bootp_t) + DHCP_OPT_LEN);
191
192
if (dhcp_msg_type == DHCPDISCOVER) {
193
if (preq_addr.s_addr != htonl(0L)) {
194
bc = request_addr(slirp, &preq_addr, client_ethaddr);
195
if (bc) {
196
daddr.sin_addr = preq_addr;
197
}
198
}
199
if (!bc) {
200
new_addr:
201
bc = get_new_addr(slirp, &daddr.sin_addr, client_ethaddr);
202
if (!bc) {
203
DPRINTF("no address left\n");
204
return;
205
}
206
}
207
memcpy(bc->macaddr, client_ethaddr, ETH_ALEN);
208
} else if (preq_addr.s_addr != htonl(0L)) {
209
bc = request_addr(slirp, &preq_addr, client_ethaddr);
210
if (bc) {
211
daddr.sin_addr = preq_addr;
212
memcpy(bc->macaddr, client_ethaddr, ETH_ALEN);
213
} else {
214
/* DHCPNAKs should be sent to broadcast */
215
daddr.sin_addr.s_addr = 0xffffffff;
216
}
217
} else {
218
bc = find_addr(slirp, &daddr.sin_addr, bp->bp_hwaddr);
219
if (!bc) {
220
/* if never assigned, behaves as if it was already
221
assigned (windows fix because it remembers its address) */
222
goto new_addr;
223
}
224
}
225
226
/* Update ARP table for this IP address */
227
arp_table_add(slirp, daddr.sin_addr.s_addr, client_ethaddr);
228
229
saddr.sin_addr = slirp->vhost_addr;
230
saddr.sin_port = htons(BOOTP_SERVER);
231
232
daddr.sin_port = htons(BOOTP_CLIENT);
233
234
rbp->bp_op = BOOTP_REPLY;
235
rbp->bp_xid = bp->bp_xid;
236
rbp->bp_htype = 1;
237
rbp->bp_hlen = 6;
238
memcpy(rbp->bp_hwaddr, bp->bp_hwaddr, ETH_ALEN);
239
240
rbp->bp_yiaddr = daddr.sin_addr; /* Client IP address */
241
rbp->bp_siaddr = saddr.sin_addr; /* Server IP address */
242
243
q = rbp->bp_vend;
244
end = rbp->bp_vend + DHCP_OPT_LEN;
245
memcpy(q, rfc1533_cookie, 4);
246
q += 4;
247
248
if (bc) {
249
DPRINTF("%s addr=%08" PRIx32 "\n",
250
(dhcp_msg_type == DHCPDISCOVER) ? "offered" : "ack'ed",
251
ntohl(daddr.sin_addr.s_addr));
252
253
if (dhcp_msg_type == DHCPDISCOVER) {
254
*q++ = RFC2132_MSG_TYPE;
255
*q++ = 1;
256
*q++ = DHCPOFFER;
257
} else /* DHCPREQUEST */ {
258
*q++ = RFC2132_MSG_TYPE;
259
*q++ = 1;
260
*q++ = DHCPACK;
261
}
262
263
if (slirp->bootp_filename) {
264
g_assert(strlen(slirp->bootp_filename) < sizeof(rbp->bp_file));
265
strcpy(rbp->bp_file, slirp->bootp_filename);
266
}
267
268
*q++ = RFC2132_SRV_ID;
269
*q++ = 4;
270
memcpy(q, &saddr.sin_addr, 4);
271
q += 4;
272
273
*q++ = RFC1533_NETMASK;
274
*q++ = 4;
275
memcpy(q, &slirp->vnetwork_mask, 4);
276
q += 4;
277
278
if (!slirp->restricted) {
279
*q++ = RFC1533_GATEWAY;
280
*q++ = 4;
281
memcpy(q, &saddr.sin_addr, 4);
282
q += 4;
283
284
*q++ = RFC1533_DNS;
285
*q++ = 4;
286
memcpy(q, &slirp->vnameserver_addr, 4);
287
q += 4;
288
}
289
290
*q++ = RFC2132_LEASE_TIME;
291
*q++ = 4;
292
val = htonl(LEASE_TIME);
293
memcpy(q, &val, 4);
294
q += 4;
295
296
if (*slirp->client_hostname) {
297
val = strlen(slirp->client_hostname);
298
if (q + val + 2 >= end) {
299
g_warning("DHCP packet size exceeded, "
300
"omitting host name option.");
301
} else {
302
*q++ = RFC1533_HOSTNAME;
303
*q++ = val;
304
memcpy(q, slirp->client_hostname, val);
305
q += val;
306
}
307
}
308
309
if (slirp->vdomainname) {
310
val = strlen(slirp->vdomainname);
311
if (q + val + 2 >= end) {
312
g_warning("DHCP packet size exceeded, "
313
"omitting domain name option.");
314
} else {
315
*q++ = RFC1533_DOMAINNAME;
316
*q++ = val;
317
memcpy(q, slirp->vdomainname, val);
318
q += val;
319
}
320
}
321
322
if (slirp->tftp_server_name) {
323
val = strlen(slirp->tftp_server_name);
324
if (q + val + 2 >= end) {
325
g_warning("DHCP packet size exceeded, "
326
"omitting tftp-server-name option.");
327
} else {
328
*q++ = RFC2132_TFTP_SERVER_NAME;
329
*q++ = val;
330
memcpy(q, slirp->tftp_server_name, val);
331
q += val;
332
}
333
}
334
335
if (slirp->vdnssearch) {
336
val = slirp->vdnssearch_len;
337
if (q + val >= end) {
338
g_warning("DHCP packet size exceeded, "
339
"omitting domain-search option.");
340
} else {
341
memcpy(q, slirp->vdnssearch, val);
342
q += val;
343
}
344
}
345
346
/* this allows to support UEFI HTTP boot: according to the UEFI
347
specification, DHCP server must send vendor class identifier option
348
set to "HTTPClient" string, when responding to DHCP requests as part
349
of the UEFI HTTP boot
350
351
we assume that, if the bootfile parameter was configured as an http
352
URL, the user intends to perform UEFI HTTP boot, so send this option
353
automatically */
354
if (slirp->bootp_filename && g_str_has_prefix(slirp->bootp_filename, "http://")) {
355
val = strlen(UEFI_HTTP_VENDOR_CLASS_ID);
356
if (q + val + 2 >= end) {
357
g_warning("DHCP packet size exceeded, "
358
"omitting vendor class id option.");
359
} else {
360
*q++ = RFC2132_VENDOR_CLASS_ID;
361
*q++ = val;
362
memcpy(q, UEFI_HTTP_VENDOR_CLASS_ID, val);
363
q += val;
364
}
365
}
366
} else {
367
static const char nak_msg[] = "requested address not available";
368
369
DPRINTF("nak'ed addr=%08" PRIx32 "\n", ntohl(preq_addr.s_addr));
370
371
*q++ = RFC2132_MSG_TYPE;
372
*q++ = 1;
373
*q++ = DHCPNAK;
374
375
*q++ = RFC2132_MESSAGE;
376
*q++ = sizeof(nak_msg) - 1;
377
memcpy(q, nak_msg, sizeof(nak_msg) - 1);
378
q += sizeof(nak_msg) - 1;
379
}
380
assert(q < end);
381
*q++ = RFC1533_END;
382
383
daddr.sin_addr.s_addr = 0xffffffffu;
384
385
assert(q <= end);
386
387
m->m_len = sizeof(struct bootp_t) + (end - rbp->bp_vend) - sizeof(struct ip) - sizeof(struct udphdr);
388
udp_output(NULL, m, &saddr, &daddr, IPTOS_LOWDELAY);
389
}
390
391
void bootp_input(struct mbuf *m)
392
{
393
struct bootp_t *bp = mtod_check(m, sizeof(struct bootp_t));
394
395
if (!m->slirp->disable_dhcp && bp && bp->bp_op == BOOTP_REQUEST) {
396
bootp_reply(m->slirp, bp, m_end(m));
397
}
398
}
399
400