13 void ICMP_Initialise();
14 void ICMP_GetPacket(tInterface *Interface, void *Address, int Length, void *Buffer);
18 tInterface *Interface;
20 } gICMP_PingSlots[PING_SLOTS];
24 * \fn void ICMP_Initialise()
25 * \brief Initialise the ICMP Layer
27 void ICMP_Initialise()
29 IPv4_RegisterCallback(IP4PROT_ICMP, ICMP_GetPacket);
33 * \fn void ICMP_GetPacket(tInterface *Interface, void *Address, int Length, void *Buffer)
34 * \brief Handles a packet from the IP Layer
36 void ICMP_GetPacket(tInterface *Interface, void *Address, int Length, void *Buffer)
38 tICMPHeader *hdr = Buffer;
40 //Log_Debug("ICMPv4", "Length = %i", Length);
41 Log_Debug("ICMPv4", "hdr->Type, hdr->Code = %i, %i", hdr->Type, hdr->Code);
42 //Log_Debug("ICMPv4", "hdr->Checksum = 0x%x", ntohs(hdr->Checksum));
43 Log_Debug("ICMPv4", "hdr->ID = 0x%x", ntohs(hdr->ID));
44 Log_Debug("ICMPv4", "hdr->Sequence = 0x%x", ntohs(hdr->Sequence));
51 Log_Warning("ICMPv4", "Code == %i for ICMP Echo Reply, should be 0", hdr->Code);
54 if(hdr->ID != (Uint16)~hdr->Sequence) {
55 Log_Warning("ICMPv4", "ID and Sequence values do not match");
58 gICMP_PingSlots[hdr->ID].bArrived = 1;
61 // -- 3: Destination Unreachable
62 case ICMP_UNREACHABLE:
65 case 3: // Port Unreachable
66 Log_Debug("ICMPv4", "Destination Unreachable (Port Unreachable)");
69 Log_Debug("ICMPv4", "Destination Unreachable (Code %i)", hdr->Code);
72 // IPv4_Unreachable( Interface, hdr->Code, htons(hdr->Length)-sizeof(tICMPHeader), hdr->Data );
78 Log_Warning("ICMPv4", "Code == %i for ICMP Echo Request, should be 0", hdr->Code);
81 //Log_Debug("ICMPv4", "Replying");
82 hdr->Type = ICMP_ECHOREPLY;
84 hdr->Checksum = htons( IPv4_Checksum( (Uint16*)hdr, Length/2 ) );
85 //Log_Debug("ICMPv4", "Checksum = 0x%04x", hdr->Checksum);
87 tIPStackBuffer *buffer = IPStack_Buffer_CreateBuffer(1 + IPV4_BUFFERS);
88 IPStack_Buffer_AppendSubBuffer(buffer, Length, 0, hdr, NULL, NULL);
89 IPv4_SendPacket(Interface, *(tIPv4*)Address, 1, ntohs(hdr->Sequence), buffer);
90 IPStack_Buffer_DestroyBuffer(buffer);
99 * \brief Sends ICMP Echo and waits for the reply
100 * \note Times out after \a Interface->TimeoutDelay has elapsed
102 int ICMP_Ping(tInterface *Interface, tIPv4 Addr)
106 char buf[32] = "\x8\0\0\0\0\0\0\0Acess2 I"
108 tICMPHeader *hdr = (void*)buf;
113 for(i=0;i<PING_SLOTS;i++)
115 if(gICMP_PingSlots[i].Interface == NULL) break;
117 if( i < PING_SLOTS ) break;
120 gICMP_PingSlots[i].Interface = Interface;
121 gICMP_PingSlots[i].bArrived = 0;
124 hdr->Checksum = htons( IPv4_Checksum((Uint16*)buf, sizeof(buf)) );
128 tIPStackBuffer *buffer = IPStack_Buffer_CreateBuffer(1 + IPV4_BUFFERS);
129 IPStack_Buffer_AppendSubBuffer(buffer, sizeof(buf), 0, buf, NULL, NULL);
130 IPv4_SendPacket(Interface, Addr, 1, i, buffer);
131 IPStack_Buffer_DestroyBuffer(buffer);
133 end = ts + Interface->TimeoutDelay;
134 while( !gICMP_PingSlots[i].bArrived && now() < end) Threads_Yield();
139 return (int)( now() - ts );