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

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

3天內不再提示

LwIP的帶操作系統基本移植

CHANBAEK ? 來源:木南創(chuàng)智 ? 作者:尹家軍 ? 2022-12-14 15:35 ? 次閱讀

現在,TCP/IP協議的應用無處不在。隨著物聯網的火爆,嵌入式領域使用TCP/IP協議進行通訊也越來越廣泛。在我們的相關產品中,也都有應用,所以我們結合應用實際對相關應用作相應的總結。

1 、技術準備

我們采用的開發(fā)平臺是STM32F407和LwIP協議棧。在開始之前,我們需要做必要的準備工作。

首先要獲得LwIP的源碼,在網上有很多,不同版本及不同平臺的都有,不過我們還是建議直接從官方網站獲得。

其次,需要硬件平臺,我們采用了STM32F407ZG+DM9161的網絡接口方式,這并不是必須的,其他硬件平臺也是一樣的。

最后,因為我們后面要在操作系統下移植,采用的操作系統是FreeRTOS,所以還需下載FreeRTOS的源碼。

2 、 LwIP****簡要說明

LwIP是一款免費的TCP/IP協議棧,但它的功能趨勢十分完備。LwIP 具有三種應用編程接口 (API):

  • Raw API :為原始的 LwIP API。它通過事件回調機制進行應用開發(fā)。該 API 提供了最好的性能和優(yōu)化的代碼長度,但增加了應用開發(fā)的復雜性。
  • Netconn API :為高層有序 API,需要實時操作系統 (RTOS)的支持 (提供進程間通訊的方法)。 Netconn API 支持多線程工作。
  • BSD Socket API :類似 Berkeley 的套接字 API (開發(fā)于 Netconn API 之上) 。

對于以上三種接口,前一種只需要裸機即可調用,后兩種需要操作系統才能調用。所以據此LwIP存在兩種移植方式:一是,只移植內核,此時應用程序的編寫只能基于RAW/Callback API進行。二是,移植內核和上層API,此時應用程序編寫可以使用3種API,即:RAW/Callback API、Sequential API和Socket API。

3 LwIP****的帶操作系統基本移植

帶操作系統的移植首先是建立在無操作系統移植基礎之上的。在無操作系統移植時,定義的數據類型和宏都是有效的,只需要對lwipopts.h配置文件做簡單修改,并根據sys_arch.txt移植說明文件編寫sys_arch.c和sys_arch.h兩個文件以實現操作系統模擬層就可以了。

操作系統模擬層的功能再以為協議棧提供郵箱、信號量、互斥量等機制,用以保證內核與上層API的通訊。這些操作系統模擬層函數均在sys.h中已經聲明,我們一般在sys_arch.c文件中完成其定義。所以,我們很清楚,帶操作系統的移植就是在無操作系統的基礎上添加操作系統模擬層。在接下來我們就看看操作系統模擬層的編寫。

在操作系統已經正確移植的基礎上,我們根據sys_arch.txt移植說明文件的描述,還需要移植的宏定義及函數等如下:

名稱 屬性 功能
sys_mbox_t 數據類型 指針類型,指向系統郵箱
sys_sem_t 數據類型 指針類型,指向系統信號量
sys_mutex_t 數據類型 指針類型,指向系統互斥量
sys_thread_t 數據類型 系統任務標識
SYS_MBOX_NULL 郵箱指針指向的空值
SYS_SEM_NULL 信號量指針指向的空值
sys_init 函數 初始化系統模擬層
sys_sem_new 函數 生成一個信號量
sys_sem_free 函數 刪除一個信號量
sys_sem_signal 函數 釋放一個信號量
sys_arch_sem_wait 函數 等待一個信號量
sys_sem_valid 函數 判斷一個信號量是否有效
sys_sem_set_invalid 函數 將一個信號量置為無效
sys_mutex_new 函數 生成一個新的互斥量
sys_mutex_free 函數 刪除一個互斥量
sys_mutex_lock 函數 鎖住一個互斥量
sys_mutex_unlock 函數 解鎖一個互斥量
sys_mutex_valid 函數 判斷一個互斥量是否有效
sys_mutex_set_invalid 函數 將一個互斥量置為無效
sys_mbox_new 函數 新建一個郵箱
sys_mbox_free 函數 刪除一個郵箱
sys_mbox_post 函數 向郵箱投遞消息,阻塞
sys_mbox_trypost 函數 嘗試向郵箱投遞消息,不阻塞
sys_arch_mbox_fetch 函數 從郵箱獲取消息,阻塞
sys_arch_mbox_tryfetch 函數 嘗試從郵箱獲取消息,不阻塞
sys_mbox_valid 函數 判斷一個郵箱是否有效
sys_mbox_set_invalid 函數 將一個郵箱設置為無效
sys_thread_new 函數 創(chuàng)建新進程
sys_arch_protect 函數 臨界區(qū)保護
sys_arch_unprotect 函數 退出臨界區(qū)保護

從上表中我們可以發(fā)現,這些變量和函數主要是面向信號量、互斥量及郵箱,包括新建、刪除、釋放、獲取等各類操作,我們需要根據操作系統的規(guī)定來實現這些函數,我們在這里使用的FreeRTOS,所以我根據FreeRTOS對信號量、互斥量及郵箱的操作來實現這些函數。我們列舉郵箱的各操作函數實現如下:

1 /*創(chuàng)建一個空的郵箱。*/
  2 err_t sys_mbox_new(sys_mbox_t *mbox, int size)
  3 {
  4   osMessageQDef(QUEUE, size, void *);
  5  
  6   *mbox = osMessageCreate(osMessageQ(QUEUE), NULL);
  7  
  8 #if SYS_STATS
  9       ++lwip_stats.sys.mbox.used;
 10       if (lwip_stats.sys.mbox.max < lwip_stats.sys.mbox.used) {
 11          lwip_stats.sys.mbox.max = lwip_stats.sys.mbox.used;
 12          }
 13 #endif /* SYS_STATS */
 14  if (*mbox == NULL)
 15   return ERR_MEM;
 16  
 17  return ERR_OK;
 18 }
 19  
 20 /*重新分配一個郵箱。如果郵箱被釋放時,郵箱中仍有消息,在lwIP中這是出現編碼錯誤的指示,并通知開發(fā)人員。*/
 21 void sys_mbox_free(sys_mbox_t *mbox)
 22 {
 23        if( osMessageWaiting(*mbox) )
 24        {
 25               portNOP();
 26 #if SYS_STATS
 27            lwip_stats.sys.mbox.err++;
 28 #endif /* SYS_STATS */
 29        }
 30  
 31        osMessageDelete(*mbox);
 32  
 33 #if SYS_STATS
 34      --lwip_stats.sys.mbox.used;
 35 #endif /* SYS_STATS */
 36 }
 37  
 38 /*發(fā)送消息到郵箱*/
 39 void sys_mbox_post(sys_mbox_t *mbox, void *data)
 40 {
 41   while(osMessagePut(*mbox, (uint32_t)data, osWaitForever) != osOK);
 42 }
 43  
 44 /*嘗試將消息發(fā)送到郵箱*/
 45 err_t sys_mbox_trypost(sys_mbox_t *mbox, void *msg)
 46 {
 47     err_t result;
 48  
 49    if ( osMessagePut(*mbox, (uint32_t)msg, 0) == osOK)
 50    {
 51       result = ERR_OK;
 52    }
 53    else {
 54       result = ERR_MEM;
 55                     
 56 #if SYS_STATS
 57       lwip_stats.sys.mbox.err++;
 58 #endif /* SYS_STATS */
 59                     
 60    }
 61  
 62    return result;
 63 }
 64  
 65 /*阻塞進程從郵箱獲取消息*/
 66 u32_t sys_arch_mbox_fetch(sys_mbox_t *mbox, void **msg, u32_t timeout)
 67 {
 68   osEvent event;
 69   uint32_t starttime = osKernelSysTick();;
 70  
 71   if(timeout != 0)
 72   {
 73     event = osMessageGet (*mbox, timeout);
 74    
 75     if(event.status == osEventMessage)
 76     {
 77       *msg = (void *)event.value.v;
 78       return (osKernelSysTick() - starttime);
 79     }
 80     else
 81     {
 82       return SYS_ARCH_TIMEOUT;
 83     }
 84   }
 85   else
 86   {
 87     event = osMessageGet (*mbox, osWaitForever);
 88     *msg = (void *)event.value.v;
 89     return (osKernelSysTick() - starttime);
 90   }
 91 }
 92  
 93 /*嘗試從郵箱獲取消息*/
 94 u32_t sys_arch_mbox_tryfetch(sys_mbox_t *mbox, void **msg)
 95 {
 96   osEvent event;
 97  
 98   event = osMessageGet (*mbox, 0);
 99  
100   if(event.status == osEventMessage)
101   {
102     *msg = (void *)event.value.v;
103     return ERR_OK;
104   }
105   else
106   {
107     return SYS_MBOX_EMPTY;
108   }
109 }
110  
111 /*判斷一個郵箱是否有效*/
112 int sys_mbox_valid(sys_mbox_t *mbox)         
113 {     
114   if (*mbox == SYS_MBOX_NULL)
115     return 0;
116   else
117     return 1;
118 }
119  
120 /*設置一個郵箱無效*/                                             
121 void sys_mbox_set_invalid(sys_mbox_t *mbox)  
122 {                                            
123   *mbox = SYS_MBOX_NULL;                     
124 }                                             
125 
126 //  創(chuàng)建一個新的信號量。而 "count"參數指示該信號量的初始狀態(tài)
127 err_t sys_sem_new(sys_sem_t *sem, u8_t count)
128 {
129   osSemaphoreDef(SEM);
130  
131   *sem = osSemaphoreCreate (osSemaphore(SEM), 1);
132       
133   if(*sem == NULL)
134   {
135 #if SYS_STATS
136       ++lwip_stats.sys.sem.err;
137 #endif /* SYS_STATS */ 
138               return ERR_MEM;
139   }
140       
141   if(count == 0)  // Means it can't be taken
142   {
143     osSemaphoreWait(*sem,0);
144   }
145  
146 #if SYS_STATS
147        ++lwip_stats.sys.sem.used;
148       if (lwip_stats.sys.sem.max < lwip_stats.sys.sem.used) {
149               lwip_stats.sys.sem.max = lwip_stats.sys.sem.used;
150        }
151 #endif /* SYS_STATS */
152              
153        return ERR_OK;
154 }

此外還有一些函數也是協議棧需要的函數,特別是sys_thread_new函數,不但協議棧在初始化是需要用到,在后續(xù)我們實現各類基于LwIP的應用時也需要用到,其實現如下:

1 sys_thread_t sys_thread_new(const char *name, lwip_thread_fn thread , void *arg, int stacksize, int prio)
 2 {
 3   const osThreadDef_t os_thread_def = { (char *)name, (os_pthread)thread, (osPriority)prio, 0, stacksize};
 4   return osThreadCreate(&os_thread_def, arg);
 5 }
 6 osThreadId osThreadCreate (const osThreadDef_t *thread_def, void *argument)
 7 {
 8   TaskHandle_t handle;
 9  
10 #if( configSUPPORT_STATIC_ALLOCATION == 1 ) &&  ( configSUPPORT_DYNAMIC_ALLOCATION == 1 )
11   if((thread_def->buffer != NULL) && (thread_def->controlblock != NULL)) {
12     handle = xTaskCreateStatic((TaskFunction_t)thread_def->pthread,(const portCHAR *)thread_def->name,
13               thread_def->stacksize, argument, makeFreeRtosPriority(thread_def->tpriority),
14               thread_def->buffer, thread_def->controlblock);
15   }
16   else {
17     if (xTaskCreate((TaskFunction_t)thread_def->pthread,(const portCHAR *)thread_def->name,
18               thread_def->stacksize, argument, makeFreeRtosPriority(thread_def->tpriority),
19               &handle) != pdPASS)  {
20       return NULL;
21     }
22   }
23 #elif( configSUPPORT_STATIC_ALLOCATION == 1 )
24  
25     handle = xTaskCreateStatic((TaskFunction_t)thread_def->pthread,(const portCHAR *)thread_def->name,
26               thread_def->stacksize, argument, makeFreeRtosPriority(thread_def->tpriority),
27               thread_def->buffer, thread_def->controlblock);
28 #else
29   if (xTaskCreate((TaskFunction_t)thread_def->pthread,(const portCHAR *)thread_def->name,
30                    thread_def->stacksize, argument, makeFreeRtosPriority(thread_def->tpriority),
31                    &handle) != pdPASS)  {
32     return NULL;
33   }    
34 #endif
35  
36   return handle;
37 }

至此,基于FreeRTOS操作系統的LwIP移植結算完成了,我們編譯下載就可以對其進行驗證。

4 、結論

前面已經移植了基于操作系統的LwIP,那怎么知道我們的移植是否成功呢?接下來我們對它進行必要的驗證。

首先我們查看目標板在網絡上的配置是否正確。我們打開命令行窗口,運行ipconfig命令,查看MAC地址和IP地址配置:

我們配置的MAC地址00:08:E1:00:00:00和IP地址192.168.2.110顯示正常。接下來我們采用ping命令測試網絡鏈接:

上圖顯示網絡連接正常,經此測試,說明我們的LwIP在有操作系統情況下移植正常。

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

    關注

    12

    文章

    9285

    瀏覽量

    85845
  • 操作系統
    +關注

    關注

    37

    文章

    6875

    瀏覽量

    123577
  • TCP
    TCP
    +關注

    關注

    8

    文章

    1377

    瀏覽量

    79183
  • LwIP
    +關注

    關注

    2

    文章

    88

    瀏覽量

    27292
收藏 人收藏

    評論

    相關推薦

    誰有LWIP入門例子?最好是stm32F103不帶操作系統的!

    誰有LWIP入門例子?最好是stm32F103不帶操作系統的!自己移植沒有成功郁悶吶
    發(fā)表于 05-25 19:26

    請問帶操作系統比沒有操作系統的程序的好處在哪?

    帶操作系統比沒有操作系統的程序的好處在哪?
    發(fā)表于 06-02 23:52

    請問ucGUI+STM32F103不帶操作系統刷DEMO源碼分享嗎?

    求ucGUI+STM32F103的不帶操作系統刷DEMO 源碼,mini板+寄存器版本的最好,謝謝大俠了,找了好久就是找不到啊 自己移植一直刷不了demo,不知道什么原因 請大俠們不吝賜教啊 ,謝謝了
    發(fā)表于 08-06 04:36

    請問STM32F407+Lwip一定要移植操作系統嗎?

    現在想做網絡產品的開發(fā),需要做TCP服務器,多個TCP客戶端連接進來,進行數據傳輸。STM32F407+Lwip一定要移植操作系統嗎?有無操作系統哪種好?不知道有
    發(fā)表于 08-16 03:25

    沒有操作系統真的不能使用LwIP套接字和域名解析嗎?

    裸機移植LwIP協議棧,整體感覺不錯但是在使用套接字(Socket)功能和域名解析(lwip_gethostbyname)時發(fā)現編譯通不過,說是要啟用操作系統。如果不用
    發(fā)表于 08-22 22:25

    lwip帶freertos操作系統移植代碼分享!

    本人在stm32f407 探索者板子上實驗過,根據原子哥lwip帶UCOS2移植的這一節(jié)視頻,做了lwip帶FreeRTOS的移植,代碼結構跟原子哥源代碼結構一樣,容易理解。代碼如下
    發(fā)表于 10-18 01:45

    LWIP操作系統移植不能識別jlink

    用原子的 網絡實驗1 LWIP操作系統移植無法識別jlink其他基礎例程,UCOS等例程都可以,就這一個程序無法識別,換過電腦,換jlink都一樣,配置也都一樣,請問原子哥和各位大神有什么高招。
    發(fā)表于 03-11 04:36

    如何利用STM32CubeMX移植LWIP到STM32F429開發(fā)板中

    STM32CubeMX+LAN8720+LWIP帶操作系統實現網絡通訊使用STM32CubeMX可以非常方便的將LWIP移植到工程中,本文就是介紹如何利用STM32CubeMX
    發(fā)表于 08-11 08:27

    為什么要做無操作系統LWIP

    操作系統LWIP有何優(yōu)點?為什么要做無操作系統LWIP呢?
    發(fā)表于 10-29 08:11

    如何將LwIP協議棧移植到μC/OS-II實時操作系統上去呢

    LwIP協議是什么?什么是μC/OS-II實時操作系統呢?如何將LwIP協議棧移植到μC/OS-II實時操作系統上去呢?
    發(fā)表于 11-05 08:44

    LwIP|無操作系統

    LwIP操作系統下的實驗 本文詳細講述了LwIP在無操作系統支持環(huán)境下的API函數介紹及編程應用。首先,介紹了RAW API的特點及優(yōu)缺點,然后逐個介紹了
    發(fā)表于 04-07 16:39 ?110次下載

    LwIP操作系統下的實驗

    本文詳細講述了LwIP在無操作系統支持環(huán)境下的API函數介紹及編程應用。首先,介紹了RAW API的特點及優(yōu)缺點,然后逐個介紹了LwIP提供的所有的RAW API函數,最后通過實例的形式介紹了
    發(fā)表于 07-19 15:33 ?107次下載

    NXPl788上lwip的無操作系統移植,基于Embest開發(fā)板

    NXPl788上lwip的無操作系統移植,基于Embest開發(fā)板
    發(fā)表于 03-26 15:59 ?85次下載

    lwip移植說明及心得

    lwip是一套用于嵌入式系統的開放源代碼TCP/IP協議棧。Lwip既可以移植操作系統上,又可以在無
    發(fā)表于 12-11 16:06 ?2.1w次閱讀
    <b class='flag-5'>lwip</b><b class='flag-5'>移植</b>說明及心得

    裸機開發(fā)和帶操作系統開發(fā)的區(qū)別

    、RT-Thread 、eCos和Linux等。2.區(qū)別馬克思主義認為,事物之間是相互聯系和相互區(qū)別的。帶操作系統開發(fā)由于操作系統具有并發(fā)性,所以可以支持多個任務運行,可以從本質上認為它是裸機開發(fā)效...
    發(fā)表于 12-09 12:51 ?24次下載
    裸機開發(fā)和<b class='flag-5'>帶操作系統</b>開發(fā)的區(qū)別