-
Notifications
You must be signed in to change notification settings - Fork 3
/
Copy pathicmp.c
119 lines (104 loc) · 3.99 KB
/
icmp.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
#include <kernel.h>
uint16_t icmp_calculate_checksum(void* packet, size_t len)
{
int array_size = len / 2;
uint16_t * array = (uint16_t*)packet; // XXX: Alignment!
uint32_t sum = 0;
for(int i = 0; i < array_size; i++) {
sum += htons(array[i]);
}
uint32_t carry = sum >> 16;
sum = sum & 0x0000ffff;
sum = sum + carry;
uint16_t ret = ~sum;
return htons(ret);
}
void icmp_send(uint32_t destination, void* icmp, uint16_t size)
{
ip_send_packet((uint8_t*)&destination, icmp, size, PROTOCOL_ICMP);
}
void icmp_send_echo(uint32_t destination, uint16_t id, uint16_t seq)
{
icmp_echo_packet_t echo = { .type = ICMP_ECHO, .code = 0, .checksum = 0, .id = htons(id), .seq = htons(seq) };
echo.checksum = icmp_calculate_checksum(&echo, (uint16_t)sizeof(icmp_echo_packet_t));
icmp_send(destination, &echo, (uint16_t)sizeof(icmp_echo_packet_t));
}
void icmp_handle_echo_packet([[maybe_unused]] ip_packet_t* encap_packet, [[maybe_unused]] icmp_echo_packet_t* packet, [[maybe_unused]] size_t len)
{
/* TODO: Send reply here */
}
void icmp_handle_echo_reply_packet([[maybe_unused]] ip_packet_t* encap_packet, icmp_echo_packet_t* packet, [[maybe_unused]] size_t len)
{
char ip[14];
get_ip_str(ip, encap_packet->src_ip);
dprintf("ECHO echo ECHO! Got ICMP reply FROM %s, seq=%d id=%d\n", ip, ntohs(packet->seq), ntohs(packet->id));
}
void icmp_handle_destination_unreachable_packet([[maybe_unused]] ip_packet_t* encap_packet, [[maybe_unused]] icmp_packet_t* packet, [[maybe_unused]] size_t len)
{
}
void icmp_handle_source_quench_packet([[maybe_unused]] ip_packet_t* encap_packet, [[maybe_unused]] icmp_packet_t* packet, [[maybe_unused]] size_t len)
{
}
void icmp_handle_redirect_packet([[maybe_unused]] ip_packet_t* encap_packet, [[maybe_unused]] icmp_packet_t* packet, [[maybe_unused]] size_t len)
{
}
void icmp_handle_time_exceeded_packet([[maybe_unused]] ip_packet_t* encap_packet, [[maybe_unused]] icmp_packet_t* packet, [[maybe_unused]] size_t len)
{
}
void icmp_handle_parameter_problem_packet([[maybe_unused]] ip_packet_t* encap_packet, [[maybe_unused]] icmp_packet_t* packet, [[maybe_unused]] size_t len)
{
}
void icmp_handle_timestamp_packet([[maybe_unused]] ip_packet_t* encap_packet, [[maybe_unused]] icmp_packet_t* packet, [[maybe_unused]] size_t len)
{
}
void icmp_handle_timestamp_reply_packet([[maybe_unused]] ip_packet_t* encap_packet, [[maybe_unused]] icmp_packet_t* packet, [[maybe_unused]] size_t len)
{
}
void icmp_handle_information_request_packet([[maybe_unused]] ip_packet_t* encap_packet, [[maybe_unused]] icmp_packet_t* packet, [[maybe_unused]] size_t len)
{
}
void icmp_handle_information_reply_packet([[maybe_unused]] ip_packet_t* encap_packet, [[maybe_unused]] icmp_packet_t* packet, [[maybe_unused]] size_t len)
{
}
void icmp_handle_packet([[maybe_unused]] ip_packet_t* encap_packet, icmp_packet_t* packet, size_t len)
{
void* p = packet;
switch (packet->type) {
case ICMP_ECHO_REPLY:
icmp_handle_echo_reply_packet(encap_packet, p, len);
break;
case ICMP_DESTINATION_UNREACHABLE:
icmp_handle_destination_unreachable_packet(encap_packet, p, len);
break;
case ICMP_SOURCE_QUENCH:
icmp_handle_source_quench_packet(encap_packet, p, len);
break;
case ICMP_REDIRECT:
icmp_handle_redirect_packet(encap_packet, p, len);
break;
case ICMP_ECHO:
icmp_handle_echo_packet(encap_packet, p, len);
break;
case ICMP_TIME_EXCEEDED:
icmp_handle_time_exceeded_packet(encap_packet, p, len);
break;
case ICMP_PARAMETER_PROBLEM:
icmp_handle_parameter_problem_packet(encap_packet, p, len);
break;
case ICMP_TIMESTAMP:
icmp_handle_timestamp_packet(encap_packet, p, len);
break;
case ICMP_TIMESTAMP_REPLY:
icmp_handle_timestamp_reply_packet(encap_packet, p, len);
break;
case ICMP_INFORMATION_REQUEST:
icmp_handle_information_request_packet(encap_packet, p, len);
break;
case ICMP_INFORMATION_REPLY:
icmp_handle_information_reply_packet(encap_packet, p, len);
break;
default:
dprintf("*** WARN *** Unknown icmp type %d on inbound packet\n", packet->type);
break;
}
}