概述
arp_solicit用来发送ARP请求,首先会根据ARP_ANNOUNCE参数来选取源地址,然后判断是否达到内核发送次数上限,未达到则调用内核arp_send_dst函数发送,如果达到上限,则继续判断是否达到应用程序请求发送次数上限,未达到则通知应用程序发送ARP请求;
源码分析
1 /* 2 发送arp请求 3 ARP请求的目的邻居项 4 缓存在该邻居项中的待发送报文,来获取该skb的源IP地址 5 */ 6 static void arp_solicit(struct neighbour *neigh, struct sk_buff *skb) 7 { 8 __be32 saddr = 0; 9 u8 dst_ha[MAX_ADDR_LEN], *dst_hw = NULL; 10 struct net_device *dev = neigh->dev; 11 __be32 target = *(__be32 *)neigh->primary_key; 12 int probes = atomic_read(&neigh->probes); 13 struct in_device *in_dev; 14 struct dst_entry *dst = NULL; 15 16 rcu_read_lock(); 17 18 /* 获取IP配置块 */ 19 in_dev = __in_dev_get_rcu(dev); 20 if (!in_dev) { 21 rcu_read_unlock(); 22 return; 23 } 24 25 /* 选取源地址 */ 26 switch (IN_DEV_ARP_ANNOUNCE(in_dev)) { 27 default: 28 case 0: /* By default announce any local IP */ 29 /* 使用任意本地地址 */ 30 if (skb && inet_addr_type_dev_table(dev_net(dev), dev, 31 ip_hdr(skb)->saddr) == RTN_LOCAL) 32 saddr = ip_hdr(skb)->saddr; 33 break; 34 case 1: /* Restrict announcements of saddr in same subnet */ 35 /* 在同一子网 */ 36 if (!skb) 37 break; 38 /* 获取skb的源ip */ 39 saddr = ip_hdr(skb)->saddr; 40 /* 源地址是本地接口地址 */ 41 if (inet_addr_type_dev_table(dev_net(dev), dev, 42 saddr) == RTN_LOCAL) { 43 /* saddr should be known to target */ 44 /* 判断是否在同一子网 */ 45 if (inet_addr_onlink(in_dev, target, saddr)) 46 break; 47 } 48 saddr = 0; 49 break; 50 case 2: /* Avoid secondary IPs, get a primary/preferred one */ 51 /* 禁止从IP,获取主IP */ 52 break; 53 } 54 rcu_read_unlock(); 55 56 /* 源地址为空,选择一个地址 */ 57 if (!saddr) 58 saddr = inet_select_addr(dev, target, RT_SCOPE_LINK); 59 60 /* 计算是否使用完单播发送次数 */ 61 probes -= NEIGH_VAR(neigh->parms, UCAST_PROBES); 62 63 /* 未使用完 */ 64 if (probes < 0) { 65 if (!(neigh->nud_state & NUD_VALID)) 66 pr_debug("trying to ucast probe in NUD_INVALID "); 67 /* 目的硬件地址 */ 68 neigh_ha_snapshot(dst_ha, neigh, dev); 69 dst_hw = dst_ha; 70 } 71 /* 使用完 */ 72 else { 73 /* 计算是否使用完应用程序ARPD发送次数 */ 74 probes -= NEIGH_VAR(neigh->parms, APP_PROBES); 75 76 /* 未使用完 */ 77 if (probes < 0) { 78 /* 应用程序ARPD请求 */ 79 neigh_app_ns(neigh); 80 return; 81 } 82 } 83 84 /* 获取目的路由缓存 */ 85 if (skb && !(dev->priv_flags & IFF_XMIT_DST_RELEASE)) 86 dst = skb_dst(skb); 87 88 /* 发送arp请求包 */ 89 arp_send_dst(ARPOP_REQUEST, ETH_P_ARP, target, dev, saddr, 90 dst_hw, dev->dev_addr, NULL, dst); 91 }