• ARP 之 发送请求arp_solicit


    概述

    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 }
  • 相关阅读:
    交换排序:冒泡排序vs快速排序
    SSO
    MVC源码分析
    python_正则表达式概述
    (爬虫向)python_json学习笔记
    Pycharm Debug调试心得
    HTML学习二_HTML常用的行级标签,常用实体字符及表单标签
    吴恩达机器学习笔记3-代价函数II(cost function)
    吴恩达机器学习笔记2-代价函数I(cost function)
    Python面向对象1:类与对象
  • 原文地址:https://www.cnblogs.com/wanpengcoder/p/11755414.html
Copyright © 2020-2023  润新知