/
main.c
175 lines (136 loc) · 4.36 KB
/
main.c
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
#include <stdio.h>
#include <stdlib.h>
#include <assert.h>
#include "../libuv/include/uv.h"
#define ERROR(msg, code) do { \
fprintf(stderr, "%s: [%s: %s]\n", msg, uv_err_name((code)), uv_strerror((code))); \
assert(0); \
} while(0);
uv_loop_t *loop;
uv_udp_t send_socket;
uv_udp_t recv_socket;
uv_udp_send_t send_req;
uv_buf_t* make_discover_msg(uv_udp_send_t *req) {
uv_buf_t *buf = malloc(sizeof(uv_buf_t));
int buf_size = 256;
buf->base = malloc(sizeof(char*) * buf_size);
buf->len = buf_size;
memset(buf->base, 0, buf->len);
// BOOTREQUEST
buf->base[0] = 0x1;
// HTYPE ethernet
buf->base[1] = 0x1;
// HLEN
buf->base[2] = 0x6;
// HOPS
buf->base[3] = 0x0;
// XID 4 bytes
buf->base[4] = (unsigned int) random();
// SECS
buf->base[8] = 0x0;
// FLAGS
buf->base[10] = 0x80;
// CIADDR 12-15 is all zeros
// YIADDR 16-19 is all zeros
// SIADDR 20-23 is all zeros
// GIADDR 24-27 is all zeros
// CHADDR 28-43 is the MAC address, use your own
buf->base[28] = 0xe4;
buf->base[29] = 0xce;
buf->base[30] = 0x8f;
buf->base[31] = 0x13;
buf->base[32] = 0xf6;
buf->base[33] = 0xd4;
// SNAME 64 bytes zero
// FILE 128 bytes zero
// OPTIONS
// - magic cookie
buf->base[236] = 99;
buf->base[237] = 130;
buf->base[238] = 83;
buf->base[239] = 99;
// DHCP Message type
buf->base[240] = 53;
buf->base[241] = 1;
buf->base[242] = 1; // DHCPDISCOVER
// DHCP Parameter request list
buf->base[243] = 55;
buf->base[244] = 4;
buf->base[245] = 1;
buf->base[246] = 3;
buf->base[247] = 15;
buf->base[248] = 6;
return buf;
}
void print_dhcp_info(const uv_buf_t *buf) {
unsigned int *as_integer = (unsigned int*)buf->base;
unsigned int ipbin = ntohl(as_integer[4]);
unsigned char ip[4] = {0};
for (int i = 0; i < 4; i++) ip[i] = (ipbin >> i*8) & 0xff;
fprintf(stderr, "Offered IP %d.%d.%d.%d\n", ip[3], ip[2], ip[1], ip[0]);
}
void recv_cb(uv_udp_t *req, ssize_t nread, const uv_buf_t *buf, const struct sockaddr *addr, unsigned int flags) {
if (nread < 0) {
// there seems to be no way to get an error code here (none of the udp tests do)
fprintf(stderr, "recv error unexpected\n");
uv_close((uv_handle_t*) req, NULL);
free(buf->base);
return;
}
char sender[17] = { 0 };
uv_ip4_name((struct sockaddr_in*) addr, sender, 16);
fprintf(stderr, "recv from %s\n", sender);
print_dhcp_info(buf);
free(buf->base);
uv_udp_recv_stop(req);
}
void send_cb(uv_udp_send_t *req, int status) {
if (status) fprintf(stderr, "async send %s %s\n", uv_err_name(status), uv_strerror(status));
}
void alloc_cb(uv_handle_t *handle, size_t size, uv_buf_t *buf) {
buf->base = malloc(size);
buf->len = size;
assert(buf->base != NULL);
}
void start_recv() {
int r;
struct sockaddr_in recv_addr;
r = uv_ip4_addr("0.0.0.0", 68, &recv_addr);
if (r) ERROR("ip4_addr", r);
// bind
r = uv_udp_bind(&recv_socket, (const struct sockaddr*)&recv_addr, 0 /*unused*/);
if (r) ERROR("udp bind", r);
// start
r = uv_udp_recv_start(&recv_socket, alloc_cb, recv_cb);
if (r) ERROR("recv start", r);
}
// more info: http://beej.us/guide/bgnet/output/html/multipage/advanced.html#broadcast
void set_broadcast() {
int r;
struct sockaddr_in broadcast_addr;
// The IP address 0.0.0.0 is used to bind to all interfaces.
// Port 0 means that the OS randomly assigns a port.
r = uv_ip4_addr("0.0.0.0", 0, &broadcast_addr);
if (r) ERROR("ip4_addr", r);
// init, bind, broadcast
uv_udp_init(loop, &send_socket);
uv_udp_bind(&send_socket, (const struct sockaddr*)&broadcast_addr, 0);
uv_udp_set_broadcast(&send_socket, 1);
}
void send_msg(const uv_buf_t *discover_msg ) {
int r;
struct sockaddr_in send_addr;
// The IP address 255.255.255.255 is a broadcast address meaning that packets will be sent to all interfaces on the subnet.
r = uv_ip4_addr("255.255.255.255", 67, &send_addr);
if (r) ERROR("ip4_addr", r);
uv_udp_send(&send_req, &send_socket, discover_msg, 1 /*nbufs*/, (const struct sockaddr*) &send_addr, send_cb);
}
int main() {
loop = uv_default_loop();
uv_udp_init(loop, &recv_socket);
start_recv();
set_broadcast();
const uv_buf_t *discover_msg = make_discover_msg(&send_req);
send_msg(discover_msg);
return uv_run(loop, UV_RUN_DEFAULT);
}