#include #include #include #include #include #include #include #include //#define ETHERNET_DEBUG //#define IPV4_DEBUG static void handle_arp(const EthernetFrameHeader&, int frame_size); static void handle_ipv4(const EthernetFrameHeader&, int frame_size); static void handle_icmp(const EthernetFrameHeader&, int frame_size); Lockable>& arp_table() { static Lockable>* the; if (!the) the = new Lockable>; return *the; } void NetworkTask_main() { auto* e1000_ptr = E1000NetworkAdapter::the(); ASSERT(e1000_ptr); auto& e1000 = *e1000_ptr; e1000.set_ipv4_address(IPv4Address(192, 168, 5, 2)); kprintf("NetworkTask: Enter main loop.\n"); for (;;) { auto packet = e1000.dequeue_packet(); if (packet.is_null()) { sleep(100); continue; } if (packet.size() < (int)(sizeof(EthernetFrameHeader))) { kprintf("NetworkTask: Packet is too small to be an Ethernet packet! (%d)\n", packet.size()); continue; } auto& eth = *(const EthernetFrameHeader*)packet.pointer(); #ifdef ETHERNET_DEBUG kprintf("NetworkTask: From %s to %s, ether_type=%w, packet_length=%u\n", eth.source().to_string().characters(), eth.destination().to_string().characters(), eth.ether_type(), packet.size() ); #endif switch (eth.ether_type()) { case EtherType::ARP: handle_arp(eth, packet.size()); break; case EtherType::IPv4: handle_ipv4(eth, packet.size()); break; } } } void handle_arp(const EthernetFrameHeader& eth, int frame_size) { constexpr int minimum_arp_frame_size = sizeof(EthernetFrameHeader) + sizeof(ARPPacket); if (frame_size < minimum_arp_frame_size) { kprintf("handle_arp: Frame too small (%d, need %d)\n", frame_size, minimum_arp_frame_size); return; } auto& packet = *static_cast(eth.payload()); if (packet.hardware_type() != 1 || packet.hardware_address_length() != sizeof(MACAddress)) { kprintf("handle_arp: Hardware type not ethernet (%w, len=%u)\n", packet.hardware_type(), packet.hardware_address_length() ); return; } if (packet.protocol_type() != EtherType::IPv4 || packet.protocol_address_length() != sizeof(IPv4Address)) { kprintf("handle_arp: Protocol type not IPv4 (%w, len=%u)\n", packet.hardware_type(), packet.protocol_address_length() ); return; } #ifdef ARP_DEBUG kprintf("handle_arp: operation=%w, sender=%s/%s, target=%s/%s\n", packet.operation(), packet.sender_hardware_address().to_string().characters(), packet.sender_protocol_address().to_string().characters(), packet.target_hardware_address().to_string().characters(), packet.target_protocol_address().to_string().characters() ); #endif // FIXME: Get the adapter through some kind of lookup by IPv4 address. auto& e1000 = *E1000NetworkAdapter::the(); if (packet.operation() == ARPOperation::Request) { // Who has this IP address? if (e1000.ipv4_address() == packet.target_protocol_address()) { // We do! kprintf("handle_arp: Responding to ARP request for my IPv4 address (%s)\n", e1000.ipv4_address().to_string().characters()); ARPPacket response; response.set_operation(ARPOperation::Response); response.set_target_hardware_address(packet.sender_hardware_address()); response.set_target_protocol_address(packet.sender_protocol_address()); response.set_sender_hardware_address(e1000.mac_address()); response.set_sender_protocol_address(e1000.ipv4_address()); e1000.send(packet.sender_hardware_address(), response); } return; } if (packet.operation() == ARPOperation::Response) { // Someone has this IPv4 address. I guess we can try to remember that. // FIXME: Protect against ARP spamming. // FIXME: Support static ARP table entries. LOCKER(arp_table().lock()); arp_table().resource().set(packet.sender_protocol_address(), packet.sender_hardware_address()); kprintf("ARP table (%d entries):\n", arp_table().resource().size()); for (auto& it : arp_table().resource()) { kprintf("%s :: %s\n", it.value.to_string().characters(), it.key.to_string().characters()); } } } void handle_ipv4(const EthernetFrameHeader& eth, int frame_size) { constexpr int minimum_ipv4_frame_size = sizeof(EthernetFrameHeader) + sizeof(IPv4Packet); if (frame_size < minimum_ipv4_frame_size) { kprintf("handle_ipv4: Frame too small (%d, need %d)\n", frame_size, minimum_ipv4_frame_size); return; } auto& packet = *static_cast(eth.payload()); #ifdef IPV4_DEBUG kprintf("handle_ipv4: source=%s, target=%s\n", packet.source().to_string().characters(), packet.destination().to_string().characters() ); #endif switch (packet.protocol()) { case IPv4Protocol::ICMP: return handle_icmp(eth, frame_size); default: kprintf("handle_ipv4: Unhandled protocol %u\n", packet.protocol()); break; } } void handle_icmp(const EthernetFrameHeader& eth, int frame_size) { (void)frame_size; auto& ipv4_packet = *static_cast(eth.payload()); auto& icmp_header = *static_cast(ipv4_packet.payload()); #ifdef ICMP_DEBUG kprintf("handle_icmp: source=%s, destination=%d type=%b, code=%b\n", ipv4_packet.source().to_string().characters(), ipv4_packet.destination().to_string().characters(), icmp_header.type(), icmp_header.code() ); #endif auto& e1000 = *E1000NetworkAdapter::the(); if (ipv4_packet.destination() == e1000.ipv4_address()) { if (icmp_header.type() == ICMPType::EchoRequest) { auto& request = reinterpret_cast(icmp_header); kprintf("handle_icmp: EchoRequest from %s: id=%u, seq=%u\n", ipv4_packet.source().to_string().characters(), (word)request.identifier, (word)request.sequence_number ); byte* response_buffer = (byte*)kmalloc(ipv4_packet.length()); memset(response_buffer, 0, ipv4_packet.length()); struct [[gnu::packed]] EchoResponse { IPv4Packet ipv4; ICMPEchoPacket icmp_echo; }; auto& response = *(EchoResponse*)response_buffer; response.ipv4.set_version(4); response.ipv4.set_internet_header_length(5); response.ipv4.set_source(e1000.ipv4_address()); response.ipv4.set_destination(ipv4_packet.source()); response.ipv4.set_protocol(IPv4Protocol::ICMP); response.ipv4.set_length(ipv4_packet.length()); response.ipv4.set_ident(1); response.ipv4.set_ttl(64); response.ipv4.set_checksum(response.ipv4.compute_checksum()); response.icmp_echo.header.set_type(ICMPType::EchoReply); response.icmp_echo.header.set_code(0); response.icmp_echo.identifier = request.identifier; response.icmp_echo.sequence_number = request.sequence_number; size_t icmp_packet_length = ipv4_packet.length() - sizeof(IPv4Packet); size_t icmp_payload_length = ipv4_packet.length() - sizeof(EchoResponse); memcpy(response.icmp_echo.payload(), request.payload(), icmp_payload_length); response.icmp_echo.header.set_checksum(internet_checksum(&response.icmp_echo, icmp_packet_length)); e1000.send_ipv4(eth.source(), response_buffer, ipv4_packet.length()); kfree(response_buffer); } } }