// an IP packet header (comes after an Ethernet) struct ip { uint8 ip_vhl; // version << 5 | header length >> 2 uint8 ip_tos; // type of service uint16 ip_len; // total length uint16 ip_id; // identification uint16 ip_off; // fragment offset field uint8 ip_ttl; // time to live uint8 ip_p; // protocol,上层协议是什么 uint8 ip_sum; // checksum uint8 ip_src, ip_dst; }
struct udp { uint16 sport; // src port uint16 dport; // dst port uint16 ulen; // length, including udp header, not including ip header uint16 sum // checksum }
int e1000_transmit(struct mbuf *m) { // // Your code here. // // the mbuf contains an ethernet frame; program it into // the TX descriptor ring so that the e1000 sends it. Stash // a pointer so that it can be freed after sending. // // ask the E1000 for the TX ring index at which // it's expecting the next packet, by reading the E1000_TDT control register. acquire(&e1000_lock); uint32 index = regs[E1000_TDT]; // the E1000 hasn't finished the corresponding previous transmission request, so return an error. if (!(tx_ring[index].status & E1000_TXD_STAT_DD)) { release(&e1000_lock); return-1; } // free the last mbuf that was transmitted from that descriptor (if there was one). if (tx_mbufs[index]) { mbuffree(tx_mbufs[index]); } // stash away a pointer to the mbuf for later freeing. tx_mbufs[index] = m; memset(&tx_ring[index],0,sizeof(tx_ring[index])); // fill in the descriptor. m->head points to the packet's content in memory, and m->len is the packet length. tx_ring[index].addr = (uint64)m->head; tx_ring[index].length = m->len; // set the necessary cmd flags. tx_ring[index].cmd = E1000_TXD_CMD_RS | E1000_TXD_CMD_EOP; // update the ring position by adding one to E1000_TDT modulo TX_RING_SIZE. regs[E1000_TDT] = (index + 1) % TX_RING_SIZE; release(&e1000_lock); return0; }