在线观看www成人影院-在线观看www日本免费网站-在线观看www视频-在线观看操-欧美18在线-欧美1级

0
  • 聊天消息
  • 系統(tǒng)消息
  • 評論與回復
登錄后你可以
  • 下載海量資料
  • 學習在線課程
  • 觀看技術(shù)視頻
  • 寫文章/發(fā)帖/加入社區(qū)
會員中心
創(chuàng)作中心

完善資料讓更多小伙伴認識你,還能領(lǐng)取20積分哦,立即完善>

3天內(nèi)不再提示

LwIP中的ARP實現(xiàn)是什么

汽車電子技術(shù) ? 來源:物聯(lián)網(wǎng)IoT開發(fā) ? 作者: 杰杰mcu ? 2023-02-14 10:12 ? 次閱讀

前言

從前面的文章,我們知道,ARP協(xié)議的核心是ARP緩存表,而ARP協(xié)議的實質(zhì)就是對緩存表項(entry)的建立、更新、查詢等操作。

那么,LwIP中是是怎么實現(xiàn)ARP協(xié)議的呢?

ARP緩存表的數(shù)據(jù)結(jié)構(gòu)

LwIP使用一個arp_table數(shù)組描述ARP緩存表,數(shù)組的內(nèi)容是表項的內(nèi)容,每個表項都必須記錄一對IP地址與MAC地址的映射關(guān)系,此外還有一些基本的信息,如表項的狀態(tài)、生命周期(生存時間)以及對應網(wǎng)卡的基本信息,LwIP使用一個etharp_entry結(jié)構(gòu)體對表項進行描述。

而且LwIP預先定義了緩存表的大小,ARP_TABLE_SIZE默認為10,也就是最大能存放10個表項,由于這個表很小,LwIP對表的操作直接采用遍歷方式,遍歷每個表項并且更改其中的內(nèi)容。

static struct etharp_entry arp_table[ARP_TABLE_SIZE];

struct etharp_q_entry 
{
  struct etharp_q_entry *next;
  struct pbuf *p;
};

struct etharp_entry 
{
#if ARP_QUEUEING
  /** 指向此ARP表項上掛起的數(shù)據(jù)包隊列的指針. */
  struct etharp_q_entry *q;
#else /* ARP_QUEUEING */
  /** 指向此ARP表項上的單個掛起數(shù)據(jù)包的指針. */
  struct pbuf *q;
#endif 
  ip4_addr_t ipaddr;    //記錄目標IP地址
  struct netif *netif;    //對應網(wǎng)卡信息
  struct eth_addr ethaddr;    //記錄與目標IP地址對應的MAC地址
  u16_t ctime;    //生存時間
  u8_t state;    //表項的狀態(tài)
};

因為APR協(xié)議在沒找到MAC地址的時候是不會發(fā)送數(shù)據(jù)的,因此這些數(shù)據(jù)會暫時存儲在ARP表項中,因此LwIP實現(xiàn)了ARP表項掛載數(shù)據(jù)的結(jié)構(gòu),etharp_q_entry指向的是數(shù)據(jù)包緩存隊列,etharp_q_entry是一個結(jié)構(gòu)體,LwIP為了方便管理pbuf數(shù)據(jù)包,直接再一次封裝這個結(jié)構(gòu)體,讓數(shù)據(jù)包能形成隊列的形式,其實簡單理解為數(shù)據(jù)包就行了。而q指向的就是一個pbuf數(shù)據(jù)包。

圖片

ARP表項的pbuf

圖片

ARP表項的pbuf隊列

除此之外,ARP表項還有很重要的信息,那就是IP地址 MAC地址,狀態(tài)、生存時間等信息。

而對于ARP表項的狀態(tài),LwIP還枚舉了多種不同的狀態(tài):

/** ARP states */
enum etharp_state {
  ETHARP_STATE_EMPTY = 0,
  ETHARP_STATE_PENDING,
  ETHARP_STATE_STABLE,
  ETHARP_STATE_STABLE_REREQUESTING_1,
  ETHARP_STATE_STABLE_REREQUESTING_2
#if ETHARP_SUPPORT_STATIC_ENTRIES
  , ETHARP_STATE_STATIC
#endif /* ETHARP_SUPPORT_STATIC_ENTRIES */
};

ARP緩存表在初始化的時候,所有的表項都會被初始化為ETHARP_STATE_EMPTY,也就是空狀態(tài),表示這些表項能被使用,在需要添加表項的時候,LwIP內(nèi)核就會遍歷ARP緩存表,找到合適的表項,進行添加。如果ARP表項處于ETHARP_STATE_PENDING狀態(tài),表示ARP已經(jīng)發(fā)出了一個ARP請求包,但是還未收到目標IP地址主機的應答,處于這個狀態(tài)的緩存表項是有等待時間的,它通過宏定義ARP_MAXPENDING指定,默認為5秒鐘,如果從發(fā)出ARP請求包后的5秒內(nèi)還沒收到應答,那么該表項又會被刪除;而如果收到應答后,ARP就會更新緩存表的信息,記錄目標IP地址與目標MAC地址的映射關(guān)系并且開始記錄表項的生存時間,同時該表項的狀態(tài)會變成ETHARP_STATE_STABLE狀態(tài)。當要發(fā)送數(shù)據(jù)包的時候,而此時表項為ETHARP_STATE_PENDING狀態(tài),那么這些數(shù)據(jù)包就會暫時被掛載到表項的數(shù)據(jù)包緩沖隊列上,直到表項的狀態(tài)為ETHARP_STATE_STABLE,才進行發(fā)送數(shù)據(jù)包。對于狀態(tài)為ETHARP_STATE_STABLE的表項,這些表項代表著ARP記錄了IP地址與MAC地址的映射關(guān)系,能隨意通過IP地址進行數(shù)據(jù)的發(fā)送,但是這些表項是具有生存時間的,通過宏定義ARP_MAXAGE指定,默認為5分鐘,在這些時間,LwIP會不斷維護這些緩存表以保持緩存表的有效。當表項是ETHARP_STATE_STABLE的時候又發(fā)送一個ARP請求包,那么表項狀態(tài)會暫時被設(shè)置為ETHARP_STATE_STABLE_REREQUESTING_1,然后被設(shè)置為ETHARP_STATE_STABLE_REREQUESTING_2狀態(tài),這些是一個過渡狀態(tài),當收到ARP應答后,表項又會被設(shè)置為ETHARP_STATE_STABLE,這樣子能保持表項的有效。

所以ARP緩存表是一個動態(tài)更新的過程,為什么要動態(tài)更新呢?因為以太網(wǎng)的物理性質(zhì)并不能保證數(shù)據(jù)傳輸?shù)氖强煽康摹R蕴W(wǎng)發(fā)送數(shù)據(jù)并不會知道對方是否已經(jīng)介紹成功,而兩臺主機的物理線路不可能一直保持有效暢通,那么如果不是動態(tài)更新的話,主機就不會知道另一臺主機是否在工作中,這樣子發(fā)出去的數(shù)據(jù)是沒有意義的。

比如兩臺主機A和B,一開始兩臺主機都是處于連接狀態(tài),能正常進行通信,但是某個時刻主機B斷開了,但是主機A不會知道主機B是否正常運行,因為以太網(wǎng)不會提示主機B已經(jīng)斷開,那么主機A會一直按照MAC地址發(fā)送數(shù)據(jù),而此時在物理鏈路層就已經(jīng)是不通的,那么這些數(shù)據(jù)是沒有意義的,而如果ARP動態(tài)更新的話,主機A就會發(fā)出ARP請求包,如果得不到主機B的回應,則說明無法與主機B進行通信,那么就會刪除ARP表項,就無法進行通信。

ARP緩存表的超時處理

ARP表項的生存時間是5分鐘,而ARP請求的等待時間是5秒鐘,當這些時間到達后,就會更新ARP表項,如果在物理鏈路層無法連通則會刪除表項。這就需要ARP層有一個超時處理函數(shù)對ARP進行管理,這些操作都是根據(jù)ARP表項的ctime字段進行的,它記錄著對應表項的生存時間,而超時處理函數(shù)是etharp_tmr(),它是一個周期性的超時處理函數(shù),每隔1秒就調(diào)用一次,當ctime的值大于指定的時間,就會刪除對應的表項。

LwIP中實現(xiàn)的函數(shù)是:etharp_tmr(void)。

由于LwIP的ARP表是比較小的,LwIP采用直接遍歷ARP緩存表,更新ARP表的內(nèi)容,而當表項的時間大于表項的生存時間(5分鐘),或者表項狀態(tài)是ETHARP_STATE_PENDING處于等待目標主機回應ARP請求包,并且等待的時間超過ARP_MAXPENDING(5秒),那么LwIP就認為這些表項是無效了,就調(diào)用etharp_free_entry()函數(shù)刪除表項。

void
etharp_tmr(void)
{
  int i;

  LWIP_DEBUGF(ETHARP_DEBUG, ("etharp_timer\\n"));

  /* 遍歷ARP表,從ARP表中刪除過期的表項 */
  for (i = 0; i < ARP_TABLE_SIZE; ++i) {
    u8_t state = arp_table[i].state;
    if (state != ETHARP_STATE_EMPTY
#if ETHARP_SUPPORT_STATIC_ENTRIES
        && (state != ETHARP_STATE_STATIC)
#endif /* ETHARP_SUPPORT_STATIC_ENTRIES */
       ) {
      arp_table[i].ctime++;

       /* 等待表項穩(wěn)定或者表項已經(jīng)過期*/
      if ((arp_table[i].ctime >= ARP_MAXAGE) ||
          ((arp_table[i].state == ETHARP_STATE_PENDING)  &&
           (arp_table[i].ctime >= ARP_MAXPENDING))) 
      {
        LWIP_DEBUGF(ETHARP_DEBUG, ("etharp_timer: expired %s entry %d.\\n",
                                   arp_table[i].state >= ETHARP_STATE_STABLE ? "stable" : "pending", i));
        /* clean up entries that have just been expired */
        etharp_free_entry(i);
      } 
      else if (arp_table[i].state == ETHARP_STATE_STABLE_REREQUESTING_1) 
      {
        /* 過渡階段 */
        arp_table[i].state = ETHARP_STATE_STABLE_REREQUESTING_2;
      } 
      else if (arp_table[i].state == ETHARP_STATE_STABLE_REREQUESTING_2) 
      {
        /* 進入ETHARP_STATE_STABLE狀態(tài) */
        arp_table[i].state = ETHARP_STATE_STABLE;
      } 
      else if (arp_table[i].state == ETHARP_STATE_PENDING) 
      {
        /*仍然掛起,重新發(fā)送ARP請求 */
        etharp_request(arp_table[i].netif, &arp_table[i].ipaddr);
      }
    }
  }
}

發(fā)送ARP請求包

發(fā)送ARP請求包的時候,需要填充已知的目標IP地址、源MAC地址、源IP地址等,并且需要該ARP包進行廣播出去,所以以太網(wǎng)首部的目標MAC地址為FF-FF-FF-FF-FF-FF

LwIP先調(diào)用etharp_request()函數(shù)進行發(fā)送ARP請求包,在etharp_request()函數(shù)中會調(diào)用etharp_request_dst()函數(shù)進行發(fā)送,此時指定的目標MAC地址是ethbroadcast,而在etharp_request_dst()函數(shù)中會調(diào)用etharp_raw()進行發(fā)送ARP請求包,層層調(diào)用,并且每層的參數(shù)都是越來越多的,這樣子封裝對于上層程序來說更加好處理,在etharp_raw()函數(shù)中,會對ARP數(shù)據(jù)包進行封裝,然后再封裝到以太網(wǎng)數(shù)據(jù)幀中,最終調(diào)用以太網(wǎng)底層發(fā)送函數(shù)進行將以太網(wǎng)數(shù)據(jù)幀發(fā)送出去。

LwIP的實現(xiàn)函數(shù)是etharp_raw()

/* --------------------------------------------------------------------------------------------- */
err_t
etharp_request(struct netif *netif, const ip4_addr_t *ipaddr)
{
  return etharp_request_dst(netif, ipaddr, ebroadcast);
}
/* --------------------------------------------------------------------------------------------- */

const struct eth_addr ethbroadcast = {{0xff, 0xff, 0xff, 0xff, 0xff, 0xff}};    //FF-FF-FF-FF-FF-FF
const struct eth_addr ethzero = {{0, 0, 0, 0, 0, 0}};    //00-00-00-00-00-00

static err_t
etharp_request_dst(struct netif *netif, const ip4_addr_t *ipaddr, const struct eth_addr *hw_dst_addr)
{
  return etharp_raw(netif, 
                    (struct eth_addr *)netif->hwaddr,
                    hw_dst_addr,
                    (struct eth_addr *)netif->hwaddr,
                    netif_ip4_addr(netif),
                    ezero,
                    ipaddr, 
                    ARP_REQUEST);
}
/* --------------------------------------------------------------------------------------------- */

* @param netif                用于發(fā)送ARP數(shù)據(jù)包的lwip網(wǎng)絡接口
* @param ethsrc_addr        以太網(wǎng)頭的源MAC地址
* @param ethdst_addr        以太網(wǎng)頭的目標MAC地址
* @param hwsrc_addr         ARP協(xié)議頭的源MAC地址
* @param ipsrc_addr         ARP協(xié)議頭的源IP地址
* @param hwdst_addr         ARP協(xié)議頭的目標MAC地址
* @param ipdst_addr         ARP協(xié)議頭的目標IP地址
* @param opcode                ARP數(shù)據(jù)包的類型
* @return ERR_OK            如果已發(fā)送ARP數(shù)據(jù)包
* 如果無法分配ARP數(shù)據(jù)包,則為ERR_MEM
static err_t
etharp_raw(struct netif *netif, const struct eth_addr *ethsrc_addr,
           const struct eth_addr *ethdst_addr,
           const struct eth_addr *hwsrc_addr, const ip4_addr_t *ipsrc_addr,
           const struct eth_addr *hwdst_addr, const ip4_addr_t *ipdst_addr,
           const u16_t opcode)
{
  struct pbuf *p;
  err_t result = ERR_OK;
  struct etharp_hdr *hdr;

  LWIP_ASSERT("netif != NULL", netif != NULL);

  /* 申請ARP報文的內(nèi)存空間 */
  p = pbuf_alloc(PBUF_LINK, SIZEOF_ETHARP_HDR, PBUF_RAM);

  if (p == NULL) {
    LWIP_DEBUGF(ETHARP_DEBUG | LWIP_DBG_TRACE | LWIP_DBG_LEVEL_SERIOUS,
                ("etharp_raw: could not allocate pbuf for ARP request.\\n"));
    ETHARP_STATS_INC(etharp.memerr);
    return ERR_MEM;
  }
  LWIP_ASSERT("check that first pbuf can hold struct etharp_hdr",
              (p->len >= SIZEOF_ETHARP_HDR));

  hdr = (struct etharp_hdr *)p->payload;
  LWIP_DEBUGF(ETHARP_DEBUG | LWIP_DBG_TRACE, ("etharp_raw: sending raw ARP packet.\\n"));
  hdr->opcode = lwip_htons(opcode);

  LWIP_ASSERT("netif->hwaddr_len must be the same as ETH_HWADDR_LEN for etharp!",
              (netif->hwaddr_len == ETH_HWADDR_LEN));

  /* 填寫源MAC地址與目標MAC地址 */
  SMEMCPY(&hdr->shwaddr, hwsrc_addr, ETH_HWADDR_LEN);
  SMEMCPY(&hdr->dhwaddr, hwdst_addr, ETH_HWADDR_LEN);

  /* 以太網(wǎng)首部源ip地址與目標ip地址 */
  IPADDR_WORDALIGNED_COPY_FROM_IP4_ADDR_T(&hdr->sipaddr, ipsrc_addr);
  IPADDR_WORDALIGNED_COPY_FROM_IP4_ADDR_T(&hdr->dipaddr, ipdst_addr);

  //填寫ARP首部硬件類型與協(xié)議類型
  hdr->hwtype = PP_HTONS(LWIP_IANA_HWTYPE_ETHERNET);
  hdr->proto = PP_HTONS(ETHTYPE_IP);

  /* 填寫ARP數(shù)據(jù)包硬件地址長度與協(xié)議地址長度 */
  hdr->hwlen = ETH_HWADDR_LEN;
  hdr->protolen = sizeof(ip4_addr_t);

  /* 發(fā)送請求包 */
#if LWIP_AUTOIP
  if (ip4_addr_islinklocal(ipsrc_addr)) 
  {
    ethernet_output(netif, p, ethsrc_addr, ebroadcast, ETHTYPE_ARP);
  }
  else
#endif /* LWIP_AUTOIP */
  {
    ethernet_output(netif, p, ethsrc_addr, ethdst_addr, ETHTYPE_ARP);
  }

  ETHARP_STATS_INC(etharp.xmit);
  /* 釋放內(nèi)存 */
  pbuf_free(p);
  p = NULL;

  return result;
}

ARP數(shù)據(jù)包處理

以太網(wǎng)是有自己獨立的尋址方式(MAC地址),而對于TCP/IP的上層協(xié)議(如TCP協(xié)議、IP協(xié)議),它們是以IP地址作為網(wǎng)絡的標識,如果沒有IP地址則無法進行收發(fā)數(shù)據(jù)。當數(shù)據(jù)通過網(wǎng)卡中接收回來的時候,LwIP內(nèi)核就需要將數(shù)據(jù)進行分解,如果是IP數(shù)據(jù)報則遞交給IP協(xié)議去處理,如果是ARP數(shù)據(jù)包則交由ARP協(xié)議去處理。

真正讓LwIP內(nèi)核去處理接收到的數(shù)據(jù)包是ethernet_input()函數(shù)。代碼太多了,簡單截取部分代碼。

err_t
ethernet_input(struct pbuf *p, struct netif *netif)
{
  struct eth_hdr *ethhdr;
  u16_t type;

  LWIP_ASSERT_CORE_LOCKED();

  //校驗數(shù)據(jù)長度
  if (p->len <= SIZEOF_ETH_HDR) {
    ETHARP_STATS_INC(etharp.proterr);
    ETHARP_STATS_INC(etharp.drop);
    MIB2_STATS_NETIF_INC(netif, ifinerrors);
    goto free_and_return;
  }

  if (p->if_idx == NETIF_NO_INDEX) {
    p->if_idx = netif_get_index(netif);
  }

  /* ethhdr指針指向以太網(wǎng)幀頭部,并且強制轉(zhuǎn)換成eth_hdr結(jié)構(gòu) */
  ethhdr = (struct eth_hdr *)p->payload;

  //獲取類型
  type = ethhdr->type;

  if (ethhdr->dest.addr[0] & 1) 
  {
    /*  這可能是多播或廣播數(shù)據(jù)包,如果目標IP地址的第一個字節(jié)的bit0是1,
        那么有可能是多播或者是廣播數(shù)據(jù)包,所以,還需要進行判斷,
        如果是多播的,就將pbuf標記為鏈路層多播。 */
    if (ethhdr->dest.addr[0] == LL_IP4_MULTICAST_ADDR_0) {
      if ((ethhdr->dest.addr[1] == LL_IP4_MULTICAST_ADDR_1) &&
          (ethhdr->dest.addr[2] == LL_IP4_MULTICAST_ADDR_2)) 
      {
       /* 將pbuf標記為鏈路層多播 */
        p->flags |= PBUF_FLAG_LLMCAST;
      }
    }
    else if (eth_addr_cmp(ehdr->dest, ebroadcast)) 
    {
      /* 將pbuf標記為鏈路層廣播 */
      p->flags |= PBUF_FLAG_LLBCAST;
    }
  }

  switch (type) {
     /* 如果是IP數(shù)據(jù)報 */
    case PP_HTONS(ETHTYPE_IP):
      if (!(netif->flags & NETIF_FLAG_ETHARP)) {
        goto free_and_return;
      }
      /* 去掉太網(wǎng)首部 */
      if (pbuf_remove_header(p, next_hdr_offset)) 
      {
        goto free_and_return;
      } 
      else
      {
        /* 遞交到IP層處理 */
        ip4_input(p, netif);
      }
      break;

    //對于是ARP包
    case PP_HTONS(ETHTYPE_ARP):
      if (!(netif->flags & NETIF_FLAG_ETHARP)) 
      {
        goto free_and_return;
      }
      /* 去掉太網(wǎng)首部 */
      if (pbuf_remove_header(p, next_hdr_offset)) 
      {
        ETHARP_STATS_INC(etharp.lenerr);
        ETHARP_STATS_INC(etharp.drop);
        goto free_and_return;
      } 
      else 
      {
        /* 傳遞到ARP協(xié)議處理 */
        etharp_input(p, netif);
      }
      break;

    default:
      ETHARP_STATS_INC(etharp.proterr);
      ETHARP_STATS_INC(etharp.drop);
      MIB2_STATS_NETIF_INC(netif, ifinunknownprotos);
      goto free_and_return;
  }

  return ERR_OK;

free_and_return:
  pbuf_free(p);
  return ERR_OK;
}

ARP數(shù)據(jù)包的處理

重點來了,我們主要是講解對收到的ARP數(shù)據(jù)包處理

ARP數(shù)據(jù)包的處理函數(shù)為etharp_input(),在這里它完成兩個任務:

如果收到的是ARP應答包,說明本機之前發(fā)出的ARP請求包有了回應,就根據(jù)應答包更新自身的ARP緩存表;

如果收到的是ARP請求包,如果包中的目標IP地址與主機IP地址匹配,除了記錄原主機的IP與MAC地址,更新自身的ARP表外,還要向源主機發(fā)送一個ARP應答包。但是如果如果包中目標IP地址與主機IP地址不匹配,則盡可能記錄源主機的IP與MAC地址,更新自身的ARP表,并丟棄該請求包,為什么說是盡可能呢,因為主機的ARP緩存表是有限的,不可能記錄太多的ARP表項,所以在有空閑的表項時才記錄,如果沒有空閑的表項,ARP覺得它自己已經(jīng)盡力了,也記不住那么多表項。

void
etharp_input(struct pbuf *p, struct netif *netif)
{
  struct etharp_hdr *hdr;
  /* these are aligned properly, whereas the ARP header fields might not be */
  ip4_addr_t sipaddr, dipaddr;
  u8_t for_us;

  LWIP_ASSERT_CORE_LOCKED();

  LWIP_ERROR("netif != NULL", (netif != NULL), return;);

  hdr = (struct etharp_hdr *)p->payload;

  /* 判斷ARP包的合法性,判斷ARP包的合法性,已經(jīng)類型是否為以太網(wǎng)、硬件地址長度是否為ETH_HWADDR_LEN、
     協(xié)議地址長度是否為sizeof(ip4_addr_t)以及協(xié)議是否為ARP協(xié)議,如果都滿足則表示ARP包合法。 */
  if ((hdr->hwtype != PP_HTONS(LWIP_IANA_HWTYPE_ETHERNET)) ||
      (hdr->hwlen != ETH_HWADDR_LEN) ||
      (hdr->protolen != sizeof(ip4_addr_t)) ||
      (hdr->proto != PP_HTONS(ETHTYPE_IP)))  {
    ETHARP_STATS_INC(etharp.proterr);
    ETHARP_STATS_INC(etharp.drop);
    pbuf_free(p);
    return;
  }
  ETHARP_STATS_INC(etharp.recv);


  //拷貝源IP地址與目標IP地址
  IPADDR_WORDALIGNED_COPY_TO_IP4_ADDR_T(&sipaddr, &hdr->sipaddr);
  IPADDR_WORDALIGNED_COPY_TO_IP4_ADDR_T(&dipaddr, &hdr->dipaddr);

  /* 看看主機網(wǎng)卡是否配置了IP地址 */
  if (ip4_addr_isany_val(*netif_ip4_addr(netif))) {
    for_us = 0;
  } 
  else 
  {
     /* 判斷ARP數(shù)據(jù)包的目標IP地址與主機IP地址是否一樣 */
    for_us = (u8_t)ip4_addr_cmp(&dipaddr, netif_ip4_addr(netif));
  }

  /* 更新ARP緩存表項 */
  etharp_update_arp_entry(netif, &sipaddr, &(hdr->shwaddr),
                          for_us ? ETHARP_FLAG_TRY_HARD : ETHARP_FLAG_FIND_ONLY);

 /* 更新完畢,根據(jù)包的類型處理 */
  switch (hdr->opcode) 
  {
    /* ARP請求包 */
    case PP_HTONS(ARP_REQUEST):
      if (for_us) {
         /* 是請求自己的,那就要做出應答 */
        etharp_raw(netif,
                   (struct eth_addr *)netif->hwaddr, &hdr->shwaddr,
                   (struct eth_addr *)netif->hwaddr, netif_ip4_addr(netif),
                   &hdr->shwaddr, &sipaddr,
                   ARP_REPLY);

      } 
       /* 不是給自己的,如果不是給自己的,原因有兩種,一種是網(wǎng)卡自身尚未配置IP地址,這樣子就只打印相關(guān)調(diào)試信息。
          另一種是ARP包中的目標IP地址與主機IP地址不符合,也不用做出回應,直接丟棄即可,并輸出相關(guān)調(diào)試信息*/
      else if (ip4_addr_isany_val(*netif_ip4_addr(netif))) 
      {
        LWIP_DEBUGF(ETHARP_DEBUG | LWIP_DBG_TRACE, ("etharp_input: we are unconfigured, ARP request ignored.\\n"));
      }
      else
      {
        LWIP_DEBUGF(ETHARP_DEBUG | LWIP_DBG_TRACE, ("etharp_input: ARP request was not for us.\\n"));
      }
      break;

    /* 對于ARP應答包 不用處理,前面已經(jīng)更新ARP表項了*/
    case PP_HTONS(ARP_REPLY):
      break;
    default:
      ETHARP_STATS_INC(etharp.err);
      break;
  }
  pbuf_free(p);
}

更新ARP表項

etharp_update_arp_entry()函數(shù)是用于更新ARP緩存表的,它會在收到一個ARP數(shù)據(jù)包的時候被調(diào)用,它會先查找一個ARP表項,如果沒有找到這個ARP表項的記錄,就會去新建一個ARP表項,然后重置ARP表項的參數(shù)(狀態(tài)、網(wǎng)卡。IP地址與對應的MAC地址以及生存時間等),然后檢測ARP表項中是否掛載數(shù)據(jù)包,如果有就將這些數(shù)據(jù)包發(fā)送出去。

表項的更新方式,動態(tài)表項有兩種方式,分別為ETHARP_FLAG_TRY_HARD和ETHARP_FLAG_FIND_ONLY。前者表示無論如何都要創(chuàng)建一個表項,如果ARP緩存表中沒有空間了,那就需要回收較老的表項,將他們刪除,然后建立新的表項。而如果是后者,就讓內(nèi)核盡量更新表項,如果ARP緩存表中沒有空間了,那么也無能為力,實在是添加不了新的表項。

static err_t
etharp_update_arp_entry(struct netif *netif, const ip4_addr_t *ipaddr, struct eth_addr *ethaddr, u8_t flags)
{
  s16_t i;
  if (ip4_addr_isany(ipaddr) ||
      ip4_addr_isbroadcast(ipaddr, netif) ||
      ip4_addr_ismulticast(ipaddr)) {
    return ERR_ARG;
  }

 /* 查找或者創(chuàng)建ARP表項,并且返回索引值 */
  i = etharp_find_entry(ipaddr, flags, netif);

  /* 如果索引值不合法,更新ARP表項失敗 */
  if (i < 0) {
    return (err_t)i;
  }

  /* 設(shè)置表項狀態(tài)為ETHARP_STATE_STABLE */
  arp_table[i].state = ETHARP_STATE_STABLE;

  /* 記錄網(wǎng)卡 */
  arp_table[i].netif = netif;

  /* 插入ARP索引樹 */
  mib2_add_arp_entry(netif, &arp_table[i].ipaddr);

  LWIP_DEBUGF(ETHARP_DEBUG | LWIP_DBG_TRACE, ("etharp_update_arp_entry: updating stable entry %"S16_F"\\n", i));

  /* 更新緩存表中的MAC地址 */
  SMEMCPY(&arp_table[i].ethaddr, ethaddr, ETH_HWADDR_LEN);

  /* 重置生存時間 */
  arp_table[i].ctime = 0;

  /* 如果表項上與未發(fā)送的數(shù)據(jù)包,那就將這些數(shù)據(jù)包發(fā)送出去 */
#if ARP_QUEUEING        //使用隊列方式
  while (arp_table[i].q != NULL) 
  {
    struct pbuf *p;
    /* 定義q指向ARP表項中的數(shù)據(jù)包緩存隊列 */
    struct etharp_q_entry *q = arp_table[i].q;

    /* 指向下一個數(shù)據(jù)包節(jié)點 */
    arp_table[i].q = q->next;

    /* 獲取pbuf數(shù)據(jù)包 */
    p = q->p;

    /* 釋放MEMP_ARP_QUEUE類型的內(nèi)存塊 */
    memp_free(MEMP_ARP_QUEUE, q);
#else 
  if (arp_table[i].q != NULL) {
    struct pbuf *p = arp_table[i].q;
    arp_table[i].q = NULL;
#endif 
     /* 發(fā)送緩存隊列的數(shù)據(jù)包 */
    ethernet_output(netif, p, (struct eth_addr *)(netif->hwaddr), ethaddr, ETHTYPE_IP);
    /* free the queued IP packet */
    pbuf_free(p);
  }
  return ERR_OK;
}

圖片

ARP數(shù)據(jù)包處理流程

ARP數(shù)據(jù)包發(fā)送

我們知道一個數(shù)據(jù)包從底層傳遞進來的流程是怎么樣的,如果是ARP數(shù)據(jù)包就會給ARP去處理,如果是IP數(shù)據(jù)報就使用ip4_input()函數(shù)傳遞到上層,這些處理在后面的章節(jié)講解。那么如果上層協(xié)議想要發(fā)送數(shù)據(jù),也肯定需要經(jīng)過ARP協(xié)議將IP地址映射為MAC地址才能完成發(fā)送操作,IP數(shù)據(jù)報通過ip4_output()函數(shù)將上層數(shù)據(jù)包傳遞到ARP協(xié)議處理,關(guān)于IP協(xié)議是怎么樣傳遞的我們暫且不說,那么ARP通過etharp_output()函數(shù)接收到IP數(shù)據(jù)報后,就會進行發(fā)送,ARP會先從數(shù)據(jù)包中進行分析,看看這個IP數(shù)據(jù)報是單播數(shù)據(jù)包還是多播或者是廣播數(shù)據(jù)包,然后進行不同的處理:

對于多播或者是廣播數(shù)據(jù)包,這種處理就很簡單,直接將數(shù)據(jù)包丟給網(wǎng)卡就行了(調(diào)用ethernet_output()函數(shù))。

對于單播包的處理稍微麻煩一點,ARP協(xié)議需要根據(jù)IP地址找到對應的MAC地址,然后才能正確發(fā)送,如果找不到MAC地址的話,還要延遲發(fā)送數(shù)據(jù)包,ARP協(xié)議首先會創(chuàng)建一個ARP表項,然后將數(shù)據(jù)包掛到ARP表項對應的緩存隊列上,與此同時會發(fā)出一個ARP請求包,等待目標主機的回應后再發(fā)送IP數(shù)據(jù)報。

此處需要注意的是,對于PBUFF_ERF、PBUF_POOL、PBUF_RAM類型的數(shù)據(jù)包是不允許直接掛到ARP表項對應的緩存隊列上的,因為此時內(nèi)核需要等待目標主機的ARP應答,而這段時間里,這些數(shù)據(jù)有可能會被上層改動,這是不允許的,所以LwIP需要將這些pbuf數(shù)據(jù)包拷貝到新的空間,等待發(fā)送。

etharp_output()函數(shù)被IP層的ip4_output()函數(shù)調(diào)用,IP層傳遞一個數(shù)據(jù)包到ARP中,etharp_output()會根據(jù)數(shù)據(jù)包的目標IP地址選擇不同的處理。

err_t
etharp_output(struct netif *netif, struct pbuf *q, const ip4_addr_t *ipaddr)
{
  const struct eth_addr *dest;
  struct eth_addr mcastaddr;
  const ip4_addr_t *dst_addr = ipaddr;

  if (ip4_addr_isbroadcast(ipaddr, netif)) 
  {
    /* 如果是廣播數(shù)據(jù)包,目標MAC地址設(shè)置為FF-FF-FF-FF-FF-FF-FF */
    dest = (const struct eth_addr *)ebroadcast;
  } 
  else if (ip4_addr_ismulticast(ipaddr)) 
  {
    /* 如果是多播數(shù)據(jù)包,目標MAC地址設(shè)置為多播地址:01-00-5E-XX-XX-XX */
    mcastaddr.addr[0] = LL_IP4_MULTICAST_ADDR_0;
    mcastaddr.addr[1] = LL_IP4_MULTICAST_ADDR_1;
    mcastaddr.addr[2] = LL_IP4_MULTICAST_ADDR_2;
    mcastaddr.addr[3] = ip4_addr2(ipaddr) & 0x7f;
    mcastaddr.addr[4] = ip4_addr3(ipaddr);
    mcastaddr.addr[5] = ip4_addr4(ipaddr);
    /* destination Ethernet address is multicast */
    dest = &mcastaddr;
  } 
  else 
  {
    /* 如果是單播目標地IP地址 */
    netif_addr_idx_t i;
    /* 判斷目標IP地址是否與主機處于同一子網(wǎng)上,
       如果不是,則修改IP地址,發(fā)向網(wǎng)關(guān),請求網(wǎng)關(guān)轉(zhuǎn)發(fā),
       則需要修改IP地址,IP地址為網(wǎng)關(guān)的IP地址,目的是為了讓網(wǎng)關(guān)進行轉(zhuǎn)發(fā)。*/
    if (!ip4_addr_netcmp(ipaddr, netif_ip4_addr(netif), netif_ip4_netmask(netif)) &&
        !ip4_addr_islinklocal(ipaddr))
    {
#if LWIP_AUTOIP
      struct ip_hdr *iphdr = LWIP_ALIGNMENT_CAST(struct ip_hdr *, q->payload);
      if (!ip4_addr_islinklocal(&iphdr->src))
#endif 
      {
#ifdef LWIP_HOOK_ETHARP_GET_GW
        dst_addr = LWIP_HOOK_ETHARP_GET_GW(netif, ipaddr);
        if (dst_addr == NULL)
#endif 
        {
           /* 判斷一下網(wǎng)關(guān)地址是否有效 */
          if (!ip4_addr_isany_val(*netif_ip4_gw(netif)))
          {
            /* 發(fā)送到默認網(wǎng)關(guān),讓網(wǎng)關(guān)進行轉(zhuǎn)發(fā) */
            dst_addr = netif_ip4_gw(netif);

          }
          else
          {
            /* 沒有默認網(wǎng)關(guān)可用,返回錯誤 */
            return ERR_RTE;
          }
        }
      }
    }

    /* 遍歷ARP緩存表 */
    for (i = 0; i < ARP_TABLE_SIZE; i++)
    {
      if ((arp_table[i].state >= ETHARP_STATE_STABLE) &&
#if ETHARP_TABLE_MATCH_NETIF
          (arp_table[i].netif == netif) &&
#endif
          (ip4_addr_cmp(dst_addr, &arp_table[i].ipaddr))) 
      {
        /* 如果找到目標IP地址對應的表項,直接發(fā)送 */
        ETHARP_SET_ADDRHINT(netif, i);
        return etharp_output_to_arp_index(netif, q, i);
      }
    }
    /* 如果沒有找到與目標IP地址對應的ARP表項 */
    return etharp_query(netif, dst_addr, q);
  }

  /* 而對于多播、廣播數(shù)據(jù)包,直接能得到對應的MAC地址,可以進行發(fā)送*/
  return ethernet_output(netif, q, (struct eth_addr *)(netif->hwaddr), dest, ETHTYPE_IP);
}

在上一個函數(shù)中,會調(diào)用etharp_output_to_arp_index()這個函數(shù),因為是ARP找到了IP地址與MAC地址對應的表項,從而能直接進行發(fā)送,除此之外,ARP還需要更新ARP表項,我們知道,LwIP中的ARP表項生存時間是5分鐘(300秒),那么在APP表項的生存時間即將到來的時候,ARP需要更新表項,為什么要在發(fā)送數(shù)據(jù)的時候更新呢?因為如果不發(fā)送數(shù)據(jù),那就沒必要更新ARP表項,這樣子表項在生存時間到來的時候就會被系統(tǒng)刪除,回收ARP表項空間,而一直使用的ARP表項需要是誰更新,更新的方式也有兩種:

如果ARP表項還差15秒就過期了,LwIP會通過廣播的方式發(fā)送一個ARP請求包,試圖得到主機的回應。

而如果ARP表項還差30秒就過期了,那么LwIP會通過單播的方式向目標主機發(fā)送一個請求包并試圖得到回應。

在這種情況下發(fā)送ARP請求包的時候,表項的狀態(tài)會由ETHARP_STATE_STABLE變成ETHARP_STATE_STABLE_REREQUESTING_1,如果目標主機回應了,那就更新ARP緩存表中的表項。

當然,如果還沒那么快到期的話,那就直接調(diào)用ethernet_output()函數(shù)將數(shù)據(jù)包傳遞給網(wǎng)卡進行發(fā)送。

#define ARP_MAXAGE                      300

/* 即將到期的時間 */
#define ARP_AGE_REREQUEST_USED_UNICAST   (ARP_MAXAGE - 30)
#define ARP_AGE_REREQUEST_USED_BROADCAST (ARP_MAXAGE - 15)

static err_t
etharp_output_to_arp_index(struct netif *netif, struct pbuf *q, netif_addr_idx_t arp_idx)
{
  /* 如果arp表項即將過期:LwIP會發(fā)送一個ARP請求包,但只有當它的狀態(tài)是ETHARP_STATE_STABLE才能請求*/
  if (arp_table[arp_idx].state == ETHARP_STATE_STABLE) 
  {
    /* 還差15秒到期 */
    if (arp_table[arp_idx].ctime >= ARP_AGE_REREQUEST_USED_BROADCAST) 
    {
      /* 使用廣播方式發(fā)出請求包 */
      if (etharp_request(netif, &arp_table[arp_idx].ipaddr) == ERR_OK) 
      {
        arp_table[arp_idx].state = ETHARP_STATE_STABLE_REREQUESTING_1;
      }
    } 
    /* 還差30秒到期 */
    else if (arp_table[arp_idx].ctime >= ARP_AGE_REREQUEST_USED_UNICAST) 
    {
      /* 發(fā)出單播請求(持續(xù)15秒),以防止不必要的廣播 */
      if (etharp_request_dst(netif, &arp_table[arp_idx].ipaddr, &arp_table[arp_idx].ethaddr) == ERR_OK) 
      {
        arp_table[arp_idx].state = ETHARP_STATE_STABLE_REREQUESTING_1;
      }
    }
  }

  return ethernet_output(netif, q, (struct eth_addr *)(netif->hwaddr), &arp_table[arp_idx].ethaddr, ETHTYPE_IP);
}

而如果在ARP緩存表中沒有找到目標IP地址對應的表項,LwIP就會調(diào)用etharp_query()函數(shù),那么ARP協(xié)議就會創(chuàng)建一個表項,這也是ARP協(xié)議的核心處理,對于剛創(chuàng)建的表項,它在初始化網(wǎng)卡信息后會被設(shè)置為ETHARP_STATE_PENDING狀態(tài),與此同時一個ARP請求包將被廣播出去,這個時候的表項是無法發(fā)送數(shù)據(jù)的,只有等待到目標主機回應了一個ARP應答包才能發(fā)送數(shù)據(jù),那么這些數(shù)據(jù)在這段時間中將被掛到表項的等待隊列上,在ARP表項處于ETHARP_STATE_STABLE狀態(tài)完成數(shù)據(jù)的發(fā)送。

函數(shù)的處理邏輯是很清晰的,首先調(diào)用etharp_find_entry()函數(shù)在ARP緩存表中查找表項,如果沒有找到就嘗試創(chuàng)建表項并且返回表項的索引,當然ARP緩存表中可能存在表項,可能為新創(chuàng)建的表項(ETHARP_STATE_EMPTY),也可能為ETHARP_STATE_PENDING或者ETHARP_STATE_STABLE狀態(tài)。如果是新創(chuàng)建的表項,那么表項肯定沒有其他信息,LwIP就會初始化一些信息,如網(wǎng)卡,然后就將表項設(shè)置為ETHARP_STATE_PENDING狀態(tài)。

掛載的這些數(shù)據(jù)在等待到目標主機產(chǎn)生ARP應答的時候會發(fā)送出去,此時的發(fā)送就是延時了,所以在沒有ARP表項的時候,發(fā)送數(shù)據(jù)會產(chǎn)生延時,在指定等待ARP應答時間內(nèi)如果等不到目標主機的應答,那么這個表項將被系統(tǒng)回收,同時數(shù)據(jù)也無法發(fā)送出去。

err_t
etharp_query(struct netif *netif, const ip4_addr_t *ipaddr, struct pbuf *q)
{
  struct eth_addr *srcaddr = (struct eth_addr *)netif->hwaddr;
  err_t result = ERR_MEM;
  int is_new_entry = 0;
  s16_t i_err;
  netif_addr_idx_t i;

  /* 檢是否為單播地址 */
  if (ip4_addr_isbroadcast(ipaddr, netif) ||
      ip4_addr_ismulticast(ipaddr) ||
      ip4_addr_isany(ipaddr)) {
    return ERR_ARG;
  }

  /* 在ARP緩存中查找表項,如果沒有則嘗試創(chuàng)建表項 */
  i_err = etharp_find_entry(ipaddr, ETHARP_FLAG_TRY_HARD, netif);

  /* 沒有發(fā)現(xiàn)表項或者沒有創(chuàng)建表項成功 */
  if (i_err < 0) {
    LWIP_DEBUGF(ETHARP_DEBUG | LWIP_DBG_TRACE, ("etharp_query: could not create ARP entry\\n"));
    if (q) {
      LWIP_DEBUGF(ETHARP_DEBUG | LWIP_DBG_TRACE, ("etharp_query: packet dropped\\n"));
      ETHARP_STATS_INC(etharp.memerr);
    }
    return (err_t)i_err;
  }
  LWIP_ASSERT("type overflow", (size_t)i_err < NETIF_ADDR_IDX_MAX);

  //找到對應的表項或者創(chuàng)建表項成功
  i = (netif_addr_idx_t)i_err;

 /* 將新表項標記為待處理 */
  if (arp_table[i].state == ETHARP_STATE_EMPTY)
  {
    is_new_entry = 1;
    //設(shè)置表項的狀態(tài)
    arp_table[i].state = ETHARP_STATE_PENDING;

    /* 記錄網(wǎng)卡 */
    arp_table[i].netif = netif;
  }

  /* 是否有新的表項 */
  if (is_new_entry || (q == NULL)) 
  {
    /* 發(fā)送ARP請求包*/
    result = etharp_request(netif, ipaddr);
    if (result != ERR_OK)
    {

    }
    if (q == NULL) {
      return result;
    }
  }

  LWIP_ASSERT("q != NULL", q != NULL);
  /* 表項狀態(tài)是否穩(wěn)定 */
  if (arp_table[i].state >= ETHARP_STATE_STABLE)
  {
    ETHARP_SET_ADDRHINT(netif, i);

    /* 發(fā)送數(shù)據(jù)包 */
    result = ethernet_output(netif, q, srcaddr, &(arp_table[i].ethaddr), ETHTYPE_IP);


  } 
  else if (arp_table[i].state == ETHARP_STATE_PENDING)
  {
    /* 如果表項是ETHARP_STATE_PENDING狀態(tài) */
    /* 將給數(shù)據(jù)包'q'排隊 */
    struct pbuf *p;
    int copy_needed = 0;

    /* 如果q包含必須拷貝的pbuf,請將整個鏈復制到一個新的PBUF_RAM */
    p = q;
    while (p) 
    {
      LWIP_ASSERT("no packet queues allowed!", (p->len != p->tot_len) || (p->next == 0));
      if (PBUF_NEEDS_COPY(p)) 
      {
        //需要拷貝
        copy_needed = 1;
        break;
      }
      p = p->next;
    }
    if (copy_needed) 
    {
      /* 將整個數(shù)據(jù)包復制到新的pbuf中 */
      p = pbuf_clone(PBUF_LINK, PBUF_RAM, q);
    }
    else 
    {
      /* 引用舊的pbuf就足夠了 */
      p = q;
      pbuf_ref(p);
    }
    /* packet could be taken over? */
    if (p != NULL) {

#if ARP_QUEUEING        /* 如果使用隊列 */
      struct etharp_q_entry *new_entry;

      /* 分配一個新的arp隊列表項 */
      new_entry = (struct etharp_q_entry *)memp_malloc(MEMP_ARP_QUEUE);

      if (new_entry != NULL) 
      {
        unsigned int qlen = 0;
        new_entry->next = 0;
        new_entry->p = p;
        if (arp_table[i].q != NULL) 
        {
          /* 隊列已經(jīng)存在,將新數(shù)據(jù)包插入隊列后面 */
          struct etharp_q_entry *r;
          r = arp_table[i].q;
          qlen++;
          while (r->next != NULL)
          {
            r = r->next;
            qlen++;
          }
          r->next = new_entry;
        }
        else
        {
          /* 隊列不存在,數(shù)據(jù)包就是隊列的第一個節(jié)點 */
          arp_table[i].q = new_entry;
        }
#if ARP_QUEUE_LEN
        if (qlen >= ARP_QUEUE_LEN) {
          struct etharp_q_entry *old;
          old = arp_table[i].q;
          arp_table[i].q = arp_table[i].q->next;
          pbuf_free(old->p);
          memp_free(MEMP_ARP_QUEUE, old);
        }
#endif

        result = ERR_OK;
      }
      else 
      {
        /* 申請內(nèi)存失敗 */
        pbuf_free(p);
        result = ERR_MEM;
      }
#else 
      /* 如果只是掛載單個數(shù)據(jù)包,那么始終只為每個ARP請求排隊一個數(shù)據(jù)包,就需要釋放先前排隊的數(shù)據(jù)包*/
      if (arp_table[i].q != NULL)
      {
        pbuf_free(arp_table[i].q);
      }
      arp_table[i].q = p;
      result = ERR_OK;
#endif
    } else {
      ETHARP_STATS_INC(etharp.memerr);
      result = ERR_MEM;
    }
  }
  return result;
}

圖片

ARP發(fā)送流程

總得來說,整個ARP的工作流程是很清晰的。

聲明:本文內(nèi)容及配圖由入駐作者撰寫或者入駐合作網(wǎng)站授權(quán)轉(zhuǎn)載。文章觀點僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場。文章及其配圖僅供工程師學習之用,如有內(nèi)容侵權(quán)或者其他違規(guī)問題,請聯(lián)系本站處理。 舉報投訴
  • MAC地址
    +關(guān)注

    關(guān)注

    1

    文章

    51

    瀏覽量

    11773
  • ARP
    ARP
    +關(guān)注

    關(guān)注

    0

    文章

    50

    瀏覽量

    14766
  • ip地址
    +關(guān)注

    關(guān)注

    0

    文章

    303

    瀏覽量

    17090
  • LwIP
    +關(guān)注

    關(guān)注

    2

    文章

    87

    瀏覽量

    27263
收藏 人收藏

    評論

    相關(guān)推薦

    基于DWC_ether_qos的以太網(wǎng)驅(qū)動開發(fā)-LWIPARP模塊介紹

    TCP/IP通訊第一步需要先調(diào)通ARP,否則TCP/IP包都不知道MAC地址要發(fā)給誰。這一篇來基于LWIPARP實現(xiàn)進行相關(guān)的分析。
    的頭像 發(fā)表于 09-18 09:34 ?1985次閱讀
    基于DWC_ether_qos的以太網(wǎng)驅(qū)動開發(fā)-<b class='flag-5'>LWIP</b>的<b class='flag-5'>ARP</b>模塊介紹

    如何手動往esp32 arp列表添加自定義的arp綁定信息?

    我目前需要手動往esp32 arp列表添加自定義的arp綁定信息,而非arp請求獲得,因為另一方設(shè)備無法完成arp應答 如IP為:192.
    發(fā)表于 06-07 08:14

    淺談如何防治ARP病毒

    淺談如何防治ARP病毒近期, 現(xiàn)一種新的“ARP欺騙”木馬病毒在互聯(lián)網(wǎng)上迅速擴散.主要表現(xiàn)為用戶頻繁斷網(wǎng)、IE瀏覽器頻繁出錯以及一些常用軟件出現(xiàn)故障等問題。Arp病毒在局域網(wǎng)感染較多
    發(fā)表于 10-10 15:24

    請問戰(zhàn)艦LWIP移植是怎么實現(xiàn)內(nèi)存管理的?

    如題,最近在移植LWIP,參考原子戰(zhàn)艦V3,由于我的系統(tǒng)沒實現(xiàn)內(nèi)存管理,因此,涉及到malloc的函數(shù)我全部使用全局數(shù)據(jù)區(qū)來開辟空間(暫時先這么粗略地實現(xiàn)),但對內(nèi)存池的內(nèi)存分配我實現(xiàn)是
    發(fā)表于 09-02 04:36

    LwIP是什么意思

    是:用少量的資源消耗實現(xiàn)一個較為完整的TCP/IP協(xié)議棧,其中“完整”主要指的是TCP協(xié)議的完整性,實現(xiàn)的重點是在保持TCP協(xié)議主要功能的基礎(chǔ)上減少對RAM 的占用。此外LwIP既可以移植到操作系統(tǒng)上運行
    發(fā)表于 08-24 06:24

    掌握LwIPARP實現(xiàn)原理與作用

    ARP request,ARP response;即一個 ARP 查詢報文,一個 ARP 回復報文。學習目標:掌握 ARP 報文的作用。掌
    發(fā)表于 06-21 11:44

    ARP報文及其在Lwip實現(xiàn)

    1、ARP報文及其在Lwip實現(xiàn)  對于網(wǎng)絡世界來說,有 IP 地址就代表了身份。不過在我們常用的網(wǎng)絡拓撲類型,IP 地址并不能準確表達我們的身份。在 ipv4
    發(fā)表于 10-19 11:55

    STM32移植LWIP問題

    出這個 Assertion \"arp_table[i].q == NULL\" failed atline 369 in ..\\\\LWIP\\\\lwip
    發(fā)表于 07-30 14:31

    ARP協(xié)議攻擊及其解決方案

    由于ARP協(xié)議的設(shè)計缺陷,使得ARP協(xié)議在使用的過程存在著盜用IP地址和ARP欺騙等嚴重的安全問題。本文分析ARP攻擊的基本原理,并提出相
    發(fā)表于 06-11 10:17 ?16次下載

    地址解析協(xié)議(ARP),地址解析協(xié)議(ARP)是什么意思

    地址解析協(xié)議(ARP),地址解析協(xié)議(ARP)是什么意思 地址解析協(xié)議 (ARP) “地址解析協(xié)議 (ARP)”是所需的 TCP/IP 標準,在RFC826“地址解析協(xié)
    發(fā)表于 04-06 09:07 ?2115次閱讀

    TCPIP協(xié)議棧的實現(xiàn)lwip

    TCPIP協(xié)議棧的實現(xiàn)lwip方便初學者剛開始接觸lwip,有個大概的了解與認識。
    發(fā)表于 03-14 15:40 ?13次下載

    ARP是什么意思?ARP是什么協(xié)議?ARP協(xié)議用于什么地方

    ARP是什么意思?ARP是什么協(xié)議?ARP協(xié)議用于什么地方 ARP是什么意思?ARP是什么協(xié)議?ARP
    發(fā)表于 11-10 18:00 ?1.9w次閱讀

    arp攻擊原理_arp攻擊怎么解決

    相信絕大數(shù)的用戶對ARP病毒都不會陌生,如果本機遭受到ARP攻擊,電腦數(shù)據(jù)就會向指定地址傳送,一般最為明顯的現(xiàn)象就是電腦無故出現(xiàn)斷網(wǎng)的情況,并且網(wǎng)絡時連時斷,會成為擾亂局域網(wǎng)其它電腦上網(wǎng)的罪魁禍首。那么
    的頭像 發(fā)表于 01-11 16:12 ?3w次閱讀
    <b class='flag-5'>arp</b>攻擊原理_<b class='flag-5'>arp</b>攻擊怎么解決

    靜態(tài) ARP 表項的潛在問題

    在計算機網(wǎng)絡,地址解析協(xié)議(ARP)將 IP 地址轉(zhuǎn)換為物理地址(MAC 地址),以便在局域網(wǎng)實現(xiàn)數(shù)據(jù)的傳輸。靜態(tài) ARP 表項的設(shè)置為
    的頭像 發(fā)表于 07-29 11:51 ?406次閱讀
    靜態(tài) <b class='flag-5'>ARP</b> 表項的潛在問題

    利用LWIP 2.2實現(xiàn)以太網(wǎng)的DHCP功能

    最近學習了LWIP,了解到目前LWIP的版本已經(jīng)更新到了2.2版本。LWIP 2.2相較于之前的版本,在協(xié)議支持、性能、安全性等方面都有了顯著的改進,我將在本帖探討如何利用
    的頭像 發(fā)表于 11-26 14:37 ?1060次閱讀
    利用<b class='flag-5'>LWIP</b> 2.2<b class='flag-5'>實現(xiàn)</b>以太網(wǎng)的DHCP功能
    主站蜘蛛池模板: 国产精品视频你懂的| 国产美女特级嫩嫩嫩bbb| 亚洲伊人久久网| 一卡二卡卡四卡无人区中文| 在线你懂的网址| 夜夜穞狠狠穞| a色在线| 午夜视频国产| 日本黄色免费网址| 91精品久久国产青草| 综合视频网| 特黄一级大片| 夜夜操网| 天堂自拍| 操的网站| 亚洲日本一区二区三区| 国产做爰一区二区| 91亚洲国产成人久久精品网站| 精品久久久久久久久久| 202z国产高清日本在线播放| 91大神视频在线播放| h版欧美一区二区三区四区| 天堂成人网| 五月婷婷色| 欧美性天天影视| 成人在线网| bt种子在线搜索| 天天操天天爽天天射| 天天摸天天碰成人免费视频| 午夜在线视频免费观看| 爱我免费视频观看在线www| 天天拍夜夜爽| 香蕉成人999视频| 婷婷色九月综合激情丁香| 午夜性爽视频男人的天堂在线| 伊人99在线| 欧美黄色一级视频| 女人张开腿让男人做爽爽| 欧美专区一区二区三区| 国产大乳美女挤奶视频| 天堂资源最新版在线www|