forked from renbaoke/h3c
-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathh3c.c
executable file
·357 lines (301 loc) · 9.54 KB
/
h3c.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
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
/*
* h3c.c
*
* Copyright 2015 BK <[email protected]>
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 2 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston,
* MA 02110-1301, USA.
*
*
*/
#include "h3c.h"
#include "md5/md5.h"
#define send_pkt ((struct packet *)send_buf)
#ifdef AF_LINK
#define sdl(p) ((struct sockaddr_dl *)(p))
#define recv_pkt ((struct packet *)(recv_buf + \
((struct bpf_hdr *)recv_buf)->bh_hdrlen))
#else
#define recv_pkt ((struct packet *)recv_buf)
#endif
#define eap_type(p) ((unsigned char *)(p) + sizeof(struct packet))
#define eap_data(p) (eap_type(p) + TYPE_LEN)
#define eap_id_info(p) eap_data(p)
#define eap_id_username(p) ((eap_id_info(p)) + sizeof(VERSION_INFO))
#define eap_md5_length(p) eap_data(p)
#define eap_md5_data(p) ((eap_md5_length(p)) + MD5_LEN_LEN)
#define eap_md5_username(p) ((eap_md5_data(p)) + MD5_LEN)
#define eap_h3c_length(p) eap_data(p)
#define eap_h3c_password(p) ((eap_h3c_length(p)) + H3C_LEN_LEN)
#define eap_h3c_username(p) (eap_h3c_password(p) + password_length)
static int sockfd;
static char username[USR_LEN];
static char password[PWD_LEN];
static unsigned char send_buf[BUF_LEN];
static unsigned char recv_buf[BUF_LEN];
#ifdef AF_LINK
static struct bpf_insn insns[] = {
BPF_STMT(BPF_LD+BPF_H+BPF_ABS, 12),
BPF_JUMP(BPF_JMP+BPF_JEQ+BPF_K, ETH_P_PAE, 0, 1),
BPF_STMT(BPF_RET+BPF_K, (u_int)-1),
BPF_STMT(BPF_RET+BPF_K, 0)
};
static struct bpf_program filter = {
sizeof(insns) / sizeof(insns[0]),
insns
};
#else
static struct sockaddr_ll addr;
#endif /* AF_LINK */
static inline void set_eapol_header(unsigned char type, unsigned short length) {
send_pkt->eapol_header.version = EAPOL_VERSION;
send_pkt->eapol_header.type = type;
send_pkt->eapol_header.length = length;
}
static inline void set_eap_header(unsigned char code, unsigned char id,
unsigned short length) {
send_pkt->eap_header.code = code;
send_pkt->eap_header.id = id;
send_pkt->eap_header.length = length;
}
static int sendout(int length) {
#ifdef AF_LINK
if (write(sockfd, send_buf, length) == -1)
return SEND_ERR;
else
return SUCCESS;
#else
if (sendto(sockfd, send_buf, length, 0, (struct sockaddr*) &addr,
sizeof(addr)) == -1)
return SEND_ERR;
else
return SUCCESS;
#endif /* AF_LINK */
}
static int recvin(int length) {
#ifdef AF_LINK
if (read(sockfd, recv_buf, length) == -1)
return RECV_ERR;
else
return SUCCESS;
#else
socklen_t len;
len = sizeof(addr);
if (recvfrom(sockfd, recv_buf, length, 0, (struct sockaddr *) &addr, &len)
== -1)
return RECV_ERR;
else
return SUCCESS;
#endif /* AF_LINK */
}
static int send_id(unsigned char packet_id) {
int username_length = strlen(username);
unsigned short len = htons(
sizeof(struct eap) + TYPE_LEN + sizeof(VERSION_INFO)
+ username_length);
set_eapol_header(EAPOL_EAPPACKET, len);
set_eap_header(EAP_RESPONSE, packet_id, len);
*eap_type(send_pkt) = EAP_TYPE_ID;
memcpy(eap_id_info(send_pkt), VERSION_INFO, sizeof(VERSION_INFO));
memcpy(eap_id_username(send_pkt), username, username_length);
return sendout(
sizeof(struct packet) + TYPE_LEN + sizeof(VERSION_INFO)
+ username_length);
}
static void get_md5_digest(unsigned char *digest, unsigned char packet_id, char *passwd, unsigned char *md5data) {
unsigned char msgbuf[128]; // msgbuf = packet_id + passwd + md5data
unsigned short msglen;
unsigned short passlen;
passlen = strlen(passwd);
msglen = 1 + passlen + 16;
msgbuf[0] = packet_id;
memcpy(msgbuf + 1, passwd, passlen);
memcpy(msgbuf + 1 + passlen, md5data, 16);
// calculate MD5 digest
md5_state_t state;
md5_init(&state);
md5_append(&state, (const md5_byte_t *)msgbuf, msglen);
md5_finish(&state, digest);
}
static int send_md5(unsigned char packet_id, unsigned char *md5data, char md5_method) {
int username_length = strlen(username);
unsigned char md5[MD5_LEN];
unsigned short len = htons(sizeof(struct eap) + TYPE_LEN +
MD5_LEN_LEN + MD5_LEN + username_length);
memset(md5, 0, MD5_LEN);
// choose md5 method
if(md5_method == MD5_XOR) {
//using XOR
memcpy(md5, password, MD5_LEN);
int i;
for (i = 0; i < MD5_LEN; i++)
md5[i] ^= md5data[i];
} else if(md5_method == MD5_MD5) {
//using MD5
get_md5_digest(md5, packet_id, password, md5data);
}
set_eapol_header(EAPOL_EAPPACKET, len);
set_eap_header(EAP_RESPONSE, packet_id, len);
*eap_type(send_pkt) = EAP_TYPE_MD5;
*eap_md5_length(send_pkt) = MD5_LEN;
memcpy(eap_md5_data(send_pkt), md5, MD5_LEN);
memcpy(eap_md5_username(send_pkt), username, username_length);
return sendout(sizeof(struct packet) + TYPE_LEN + MD5_LEN_LEN +
MD5_LEN + username_length);
}
static int send_h3c(unsigned char packet_id) {
int username_length = strlen(username);
int password_length = strlen(password);
unsigned short len = htons(
sizeof(struct eap) + 1 + 1 + password_length + username_length);
set_eapol_header(EAPOL_EAPPACKET, len);
set_eap_header(EAP_RESPONSE, packet_id, len);
*eap_type(send_pkt) = EAP_TYPE_H3C;
*eap_h3c_length(send_pkt) = (unsigned char) password_length;
memcpy(eap_h3c_password(send_pkt), password, password_length);
memcpy(eap_h3c_username(send_pkt), username, username_length);
return sendout(
sizeof(struct packet) + TYPE_LEN + H3C_LEN_LEN + password_length
+ username_length);
}
int h3c_init(char *_interface) {
struct ifreq ifr;
/* Set destination mac address. */
memcpy(send_pkt->eth_header.ether_dhost, PAE_GROUP_ADDR, ETH_ALEN);
/* Set ethernet type. */
send_pkt->eth_header.ether_type = htons(ETH_P_PAE);
strcpy(ifr.ifr_name, _interface);
#ifdef AF_LINK
struct ifaddrs *ifhead, *ifa;
char device[] = "/dev/bpf0";
int n = 0;
do {
sockfd = open(device, O_RDWR);
}while ((sockfd == -1) && (errno == EBUSY) && (device[8]++ != '9'));
if (sockfd == -1)
return BPF_OPEN_ERR;
n = BUF_LEN;
if (ioctl(sockfd, BIOCSBLEN, &n) == -1)
return BPF_SET_BUF_LEN_ERR;
if (ioctl(sockfd, BIOCSETIF, &ifr) == -1)
return BPF_SET_IF_ERR;
if (ioctl(sockfd, BIOCSETF, &filter) == -1)
return BPF_SET_FILTER_ERR;
n = 1;
if (ioctl(sockfd, BIOCIMMEDIATE, &n) == -1)
return BPF_SET_IMMEDIATE_ERR;
#ifdef __NetBSD__
n = 0;
if (ioctl(sockfd, BIOCSSEESENT, &n) == -1)
return BPF_SET_DIRECTION_ERR;
#elif __FreeBSD__
n = BPF_D_IN;
if (ioctl(sockfd, BIOCSDIRECTION, &n) == -1)
return BPF_SET_DIRECTION_ERR;
#elif __OpenBSD__
n = BPF_DIRECTION_OUT;
if (ioctl(sockfd, BIOCSDIRFILT, &n) == -1)
return BPF_SET_DIRECTION_ERR;
#endif
#else
if ((sockfd = socket(AF_PACKET, SOCK_RAW, htons(ETH_P_PAE))) == -1)
return SOCKET_OPEN_ERR;
if (ioctl(sockfd, SIOCGIFINDEX, &ifr) == -1)
return SOCKET_SET_IF_ERR;
else
addr.sll_ifindex = ifr.ifr_ifindex;
if (ioctl(sockfd, SIOCGIFHWADDR, &ifr) == -1)
return SOCKET_GET_HWADDR_ERR;
/* Set source mac address. */
memcpy(send_pkt->eth_header.ether_shost, ifr.ifr_hwaddr.sa_data, ETH_ALEN);
#endif /* AF_LINK */
return SUCCESS;
}
int h3c_set_username(char *_username) {
int username_length = strlen(_username);
if (username_length > USR_LEN - 1)
return USR_TOO_LONG;
strcpy(username, _username);
return SUCCESS;
}
int h3c_set_password(char *_password) {
int password_length = strlen(_password);
if (password_length > PWD_LEN - 1)
return PWD_TOO_LONG;
strcpy(password, _password);
return SUCCESS;
}
int h3c_start() {
set_eapol_header(EAPOL_START, 0);
return sendout(sizeof(struct ether_header) + sizeof(struct eapol));
}
int h3c_logoff() {
set_eapol_header(EAPOL_LOGOFF, 0);
return sendout(sizeof(struct ether_header) + sizeof(struct eapol));
}
int h3c_response(int (*success_callback)(void), int (*failure_callback)(void),
int (*unkown_eapol_callback)(void), int (*unkown_eap_callback)(void),
int (*got_response_callback)(void), char md5_method) {
if (recvin(BUF_LEN) == RECV_ERR)
return RECV_ERR;
if (recv_pkt->eapol_header.type != EAPOL_EAPPACKET) {
/* Got unknown eapol type. */
if (unkown_eapol_callback != NULL)
return unkown_eapol_callback();
else
return EAPOL_UNHANDLED;
}
if (recv_pkt->eap_header.code == EAP_SUCCESS) {
/* Got success. */
if (success_callback != NULL)
return success_callback();
else
return SUCCESS_UNHANDLED;
} else if (recv_pkt->eap_header.code == EAP_FAILURE) {
/* Got failure. */
if (failure_callback != NULL)
return failure_callback();
else
return FAILURE_UNHANDLED;
} else if (recv_pkt->eap_header.code == EAP_REQUEST)
/*
* Got request.
* Response according to request type.
*/
if (*eap_type(recv_pkt) == EAP_TYPE_ID)
return send_id(recv_pkt->eap_header.id);
else if (*eap_type(recv_pkt) == EAP_TYPE_MD5)
return send_md5(recv_pkt->eap_header.id, eap_md5_data(recv_pkt), md5_method);
else if (*eap_type(recv_pkt) == EAP_TYPE_H3C)
return send_h3c(recv_pkt->eap_header.id);
else
return SUCCESS;
else if (recv_pkt->eap_header.code == EAP_RESPONSE) {
/* Got response. */
if (got_response_callback != NULL)
return got_response_callback();
else
return RESPONSE_UNHANDLED;
} else {
/* Got unkown eap type. */
if (unkown_eap_callback != NULL)
return unkown_eap_callback();
else
return EAP_UNHANDLED;
}
}
void h3c_clean() {
close(sockfd);
}