網絡子係統50_ip協議選項填充
//調用路徑ip_queue_xmit->ip_options_build //函數主要任務: // 1.非分片ip報文,向ip報頭填充ip選項,ip選項在創建socket時設置 // 2.分片ip報文,將record route選項,time stamp選項設置為NOP // daddr,ip接收方的地址 1.1 void ip_options_build(struct sk_buff * skb, struct ip_options * opt, u32 daddr, struct rtable *rt, int is_frag) { unsigned char * iph = skb->nh.raw; memcpy(&(IPCB(skb)->opt), opt, sizeof(struct ip_options));//將socket中提供的ip選項拷貝到skb->cb中 memcpy(iph+sizeof(struct iphdr), opt->__data, opt->optlen);//複製ip選項到ip報頭後 opt = &(IPCB(skb)->opt); opt->is_data = 0; if (opt->srr)//將目標地址拷貝到源路由選項列表中的最後一個位置上 memcpy(iph+opt->srr+iph[opt->srr+1]-4, &daddr, 4); if (!is_frag) {//在ip_queue_xmit->ip_options_build調用路徑上,is_frag為0 if (opt->rr_needaddr)//record route選項 ip_rt_get_source(iph+opt->rr+iph[opt->rr+2]-5, rt);//獲取路由的首選源地址填充到選項ptr所指的位置 if (opt->ts_needaddr)//time stamp選項 ip_rt_get_source(iph+opt->ts+iph[opt->ts+2]-9, rt); if (opt->ts_needtime) { struct timeval tv; __u32 midtime; do_gettimeofday(&tv);//通過do_gettimeofday獲取係統時間,填充當天的ms midtime = htonl((tv.tv_sec % 86400) * 1000 + tv.tv_usec / 1000); memcpy(iph+opt->ts+iph[opt->ts+2]-5, &midtime, 4); } return; } //ip分片中的選項 if (opt->rr) { memset(iph+opt->rr, IPOPT_NOP, iph[opt->rr+1]);//設置record route為NOP opt->rr = 0; opt->rr_needaddr = 0; } if (opt->ts) { memset(iph+opt->ts, IPOPT_NOP, iph[opt->ts+1]);//設置time stamp為NOP opt->ts = 0; opt->ts_needaddr = opt->ts_needtime = 0; } }
最後更新:2017-04-03 14:53:40