lwIP(Lightweight IP)是一個為嵌入式系統(tǒng)設(shè)計的輕量級TCP/IP協(xié)議棧。它旨在為資源受限的環(huán)境提供完整的網(wǎng)絡(luò)協(xié)議功能,同時保持低內(nèi)存使用和代碼大小。由于其模塊化的設(shè)計,開發(fā)者可以根據(jù)需要選擇包含或排除特定功能,以滿足特定應(yīng)用的資源要求。
Xilinx的lwIP是基于開源lwIP TCP/IP協(xié)議棧的一個適應(yīng)版本,專門為Xilinx的硬件平臺,如Zynq-7000和MicroBlaze,進行了優(yōu)化和集成。Xilinx為其硬件平臺提供了lwIP的庫,使得開發(fā)者可以輕松地在其FPGA和SoC設(shè)計中實現(xiàn)網(wǎng)絡(luò)通信功能。
以lwip TCP Perf Client為例,這是一個fpga作為TCP Client,像TCP Server發(fā)送批量數(shù)據(jù),并測試傳輸性能的例程。

TCP參數(shù)
先看幾個TCP相關(guān)的參數(shù)
TCP_CONN_PORT表示TCP的端口號,在Server中,需要指定該端口號,如果發(fā)現(xiàn)tcp一直不通,但ping是可以通的,多半原因是這個端口被占用了;
TCP_SERVER_IP_ADDRESS表示TCP Server的IP地址

FPGA的IP地址是在main.c里面指定的:

如果TCP Server使用網(wǎng)絡(luò)調(diào)試助手接收數(shù)據(jù),設(shè)置如下:(需要注意,本地端口號應(yīng)該是5001,跟代碼中匹配)

main函數(shù)
main函數(shù)的內(nèi)容如下:
intmain(void)
{
structnetif*netif;
/*themacaddressoftheboard.thisshouldbeuniqueperboard*/
unsignedcharmac_ethernet_address[]={
0x00,0x0a,0x35,0x00,0x01,0x02};
netif=&server_netif;
#ifdefined(__arm__)&&!defined(ARMR5)
#ifXPAR_GIGE_PCS_PMA_SGMII_CORE_PRESENT==1||
XPAR_GIGE_PCS_PMA_1000BASEX_CORE_PRESENT==1
ProgramSi5324();
ProgramSfpPhy();
#endif
#endif
/*DefinethisboardspecificmacroinorderperformPHYreset
*onZCU102
*/
#ifdefXPS_BOARD_ZCU102
IicPhyReset();
#endif
init_platform();
xil_printf("
");
xil_printf("-----lwIPRAWModeTCPClientApplication-----
");
/*initializelwIP*/
lwip_init();
/*Addnetworkinterfacetothenetif_list,andsetitasdefault*/
if(!xemac_add(netif,NULL,NULL,NULL,mac_ethernet_address,
PLATFORM_EMAC_BASEADDR)){
xil_printf("ErroraddingN/Winterface
");
return-1;
}
netif_set_default(netif);
/*nowenableinterrupts*/
platform_enable_interrupts();
/*specifythatthenetworkifisup*/
netif_set_up(netif);
assign_default_ip(&(netif->ip_addr),&(netif->netmask),&(netif->gw));
print_ip_settings(&(netif->ip_addr),&(netif->netmask),&(netif->gw));
xil_printf("
");
/*printappheader*/
print_app_header();
/*starttheapplication*/
start_application();
xil_printf("
");
while(1){
if(TcpFastTmrFlag){
tcp_fasttmr();
TcpFastTmrFlag=0;
}
if(TcpSlowTmrFlag){
tcp_slowtmr();
TcpSlowTmrFlag=0;
}
xemacif_input(netif);
transfer_data();
}
/*neverreached*/
cleanup_platform();
return0;
}
在main函數(shù)中,首先就是定義各種網(wǎng)口接口相關(guān)的變量,并定義了MAC地址。
netif
這個netif的指針,需要多關(guān)注一下。
在lwIP中,netif(網(wǎng)絡(luò)接口)是一個核心的結(jié)構(gòu)體,它代表了一個網(wǎng)絡(luò)接口,例如以太網(wǎng)接口、Wi-Fi接口等。netif結(jié)構(gòu)體用于定義和管理這些接口,使lwIP可以在多個接口上運行并進行路由決策。
具體來說,netif結(jié)構(gòu)體包括了以下幾個主要的部分:
硬件地址:例如MAC地址。
IP地址、子網(wǎng)掩碼和網(wǎng)關(guān):這些用于IP層的路由和地址決策。
狀態(tài)標志:表示接口的狀態(tài),例如是否激活、是否為默認接口等。
輸入和輸出函數(shù)指針:這些函數(shù)用于處理從該接口接收到的數(shù)據(jù)包或向該接口發(fā)送數(shù)據(jù)包。
其他驅(qū)動特定的數(shù)據(jù):例如用于DMA的描述符、緩沖區(qū)等。
當你在lwIP中添加一個新的網(wǎng)絡(luò)接口時,你通常會初始化一個netif結(jié)構(gòu)體并使用netif_add()函數(shù)將其添加到lwIP的接口列表中。這樣,lwIP就可以開始在該接口上接收和發(fā)送數(shù)據(jù)包了。
簡而言之,netif是lwIP中用于表示和管理網(wǎng)絡(luò)接口的關(guān)鍵結(jié)構(gòu)體。
init_platform
在init_platform()函數(shù)中,初始化定時器和中斷。

接下來就是lwip的初始化,這三個初始化都是在platform的庫里面寫好的,直接調(diào)用就行。
xemac_add
后面xemac_add的原型如下,可以簡單理解為設(shè)置網(wǎng)口的mac地址,此處沒有設(shè)置IP的信息,可以看到傳進去的參數(shù)都是NULL。
structnetif* xemac_add(structnetif*netif, ip_addr_t*ipaddr,ip_addr_t*netmask,ip_addr_t*gw, unsignedchar*mac_ethernet_address, UINTPTRmac_baseaddr)
netif_set_default
netif_set_default函數(shù)在lwIP中用于設(shè)置默認的網(wǎng)絡(luò)接口。在一個系統(tǒng)中可能存在多個網(wǎng)絡(luò)接口,但通常只有一個被視為默認接口。當lwIP需要發(fā)送數(shù)據(jù)包,但不知道應(yīng)該通過哪個接口發(fā)送時,它會選擇默認接口。
函數(shù)原型如下:
/**
*@ingroupnetif
*Setanetworkinterfaceasthedefaultnetworkinterface
*(usedtooutputallpacketsforwhichnospecificrouteisfound)
*
*@paramnetifthedefaultnetworkinterface
*/
void
netif_set_default(structnetif*netif)
{
LWIP_ASSERT_CORE_LOCKED();
if(netif==NULL){
/*removedefaultroute*/
mib2_remove_route_ip4(1,netif);
}else{
/*installdefaultroute*/
mib2_add_route_ip4(1,netif);
}
netif_default=netif;
LWIP_DEBUGF(NETIF_DEBUG,("netif:settingdefaultinterface%c%c
",
netif?netif->name[0]:''',netif?netif->name[1]:'''));
}
其中,netif是你希望設(shè)置為默認的網(wǎng)絡(luò)接口的指針。
當你調(diào)用這個函數(shù)時,傳入的netif結(jié)構(gòu)體會被設(shè)置為默認網(wǎng)絡(luò)接口。這意味著,除非有特定的路由決策指示其他接口,否則所有的出站數(shù)據(jù)包都會通過這個接口發(fā)送。
例如,如果你有一個以太網(wǎng)接口和一個Wi-Fi接口,并且你希望所有的通信默認通過Wi-Fi接口進行,那么你會在初始化Wi-Fi接口后調(diào)用netif_set_default函數(shù),并傳入Wi-Fi接口的netif結(jié)構(gòu)體指針。
這個函數(shù)對于確保正確的網(wǎng)絡(luò)通信行為非常重要,特別是在存在多個網(wǎng)絡(luò)接口的系統(tǒng)中。
platform_enable_interrupts
這個函數(shù)就很容易理解了,就是使能中斷,函數(shù)原型如下:
voidplatform_enable_interrupts()
{
/*
*Enablenon-criticalexceptions.
*/
Xil_ExceptionEnableMask(XIL_EXCEPTION_IRQ);
XScuTimer_EnableInterrupt(&TimerInstance);
XScuTimer_Start(&TimerInstance);
return;
}
netif_set_up
netif_set_up函數(shù)在lwIP中用于激活一個網(wǎng)絡(luò)接口。當你初始化一個網(wǎng)絡(luò)接口并準備好開始接收和發(fā)送數(shù)據(jù)時,你需要調(diào)用這個函數(shù)來標記該接口為"up"狀態(tài)。
函數(shù)原型如下:
void
netif_set_up(structnetif*netif)
{
LWIP_ASSERT_CORE_LOCKED();
LWIP_ERROR("netif_set_up:invalidnetif",netif!=NULL,return);
if(!(netif->flags&NETIF_FLAG_UP)){
netif_set_flags(netif,NETIF_FLAG_UP);
MIB2_COPY_SYSUPTIME_TO(&netif->ts);
NETIF_STATUS_CALLBACK(netif);
#ifLWIP_NETIF_EXT_STATUS_CALLBACK
{
netif_ext_callback_args_targs;
args.status_changed.state=1;
netif_invoke_ext_callback(netif,LWIP_NSC_STATUS_CHANGED,&args);
}
#endif
netif_issue_reports(netif,NETIF_REPORT_TYPE_IPV4|NETIF_REPORT_TYPE_IPV6);
#ifLWIP_IPV6
nd6_restart_netif(netif);
#endif/*LWIP_IPV6*/
}
}
其中,netif是你希望激活的網(wǎng)絡(luò)接口的指針。
當你調(diào)用netif_set_up函數(shù)時,它會執(zhí)行以下操作:
設(shè)置netif結(jié)構(gòu)體中的flags字段,標記該接口為"up"狀態(tài)。
如果配置了lwIP的相關(guān)回調(diào),例如NETIF_STATUS_CALLBACK,那么這些回調(diào)函數(shù)也會被觸發(fā),通知應(yīng)用程序該接口的狀態(tài)已經(jīng)改變。
通常,在你完成網(wǎng)絡(luò)接口的硬件初始化、分配了必要的資源,并確信接口已經(jīng)準備好進行通信后,你會調(diào)用netif_set_up函數(shù)。這樣,lwIP就知道它可以開始在該接口上接收和發(fā)送數(shù)據(jù)包了。
相反地,如果你需要將一個接口標記為"down"狀態(tài),例如在接口遇到錯誤或需要進行維護時,你可以調(diào)用netif_set_down函數(shù)。這會告訴lwIP停止在該接口上的通信,直到接口再次被設(shè)置為"up"狀態(tài)。
assign_default_ip
從名字也可以看到出來,就是設(shè)置ip地址、Netmask和gate way
函數(shù)原型也非常直觀,不做過多解釋了
staticvoidassign_default_ip(ip_addr_t*ip,ip_addr_t*mask,ip_addr_t*gw)
{
interr;
xil_printf("ConfiguringdefaultIP%s
",DEFAULT_IP_ADDRESS);
err=inet_aton(DEFAULT_IP_ADDRESS,ip);
if(!err)
xil_printf("InvaliddefaultIPaddress:%d
",err);
err=inet_aton(DEFAULT_IP_MASK,mask);
if(!err)
xil_printf("InvaliddefaultIPMASK:%d
",err);
err=inet_aton(DEFAULT_GW_ADDRESS,gw);
if(!err)
xil_printf("Invaliddefaultgatewayaddress:%d
",err);
}
start_application
start_application函數(shù)是一個啟動網(wǎng)絡(luò)應(yīng)用的函數(shù)。在很多l(xiāng)wIP的示例應(yīng)用中,這個函數(shù)被用來初始化和啟動特定的網(wǎng)絡(luò)應(yīng)用,例如啟動一個HTTP服務(wù)器、TCP客戶端、UDP回聲服務(wù)等。具體的功能和行為取決于應(yīng)用的需求和設(shè)計。這個函數(shù)可能會初始化所需的網(wǎng)絡(luò)資源,設(shè)置回調(diào)函數(shù),并開始監(jiān)聽網(wǎng)絡(luò)事件。
初始化變量:函數(shù)開始時,初始化了一些變量,如err用于錯誤處理,pcb代表TCP控制塊,remote_addr用于存儲遠程服務(wù)器的IP地址,以及一個循環(huán)計數(shù)器i。
設(shè)置遠程服務(wù)器的IP地址:
如果啟用了IPv6(LWIP_IPV6==1),則使用inet6_aton函數(shù)將TCP_SERVER_IPV6_ADDRESS字符串轉(zhuǎn)換為IPv6地址格式并存儲在remote_addr中。
如果未啟用IPv6,則使用inet_aton函數(shù)將TCP_SERVER_IP_ADDRESS字符串轉(zhuǎn)換為IPv4地址格式。
檢查IP地址的有效性:如果IP地址轉(zhuǎn)換失敗,函數(shù)會打印錯誤消息并返回。
創(chuàng)建TCP控制塊(PCB):使用tcp_new_ip_type函數(shù)為客戶端創(chuàng)建一個新的TCP控制塊。
連接到遠程服務(wù)器:使用tcp_connect函數(shù)嘗試連接到遠程服務(wù)器的指定IP地址和端口TCP_CONN_PORT。如果連接成功,tcp_client_connected回調(diào)函數(shù)將被注冊,以便在連接建立后進行處理。
錯誤處理:如果在上述步驟中出現(xiàn)任何錯誤,函數(shù)會打印相應(yīng)的錯誤消息并關(guān)閉TCP連接。
初始化發(fā)送緩沖區(qū):為send_buf緩沖區(qū)填充數(shù)據(jù),數(shù)據(jù)內(nèi)容是0到9的數(shù)字字符。
總的來說,start_application函數(shù)的主要目的是初始化一個TCP客戶端,嘗試連接到指定的遠程服務(wù)器,并準備發(fā)送數(shù)據(jù)。
函數(shù)原型如下:
voidstart_application(void)
{
err_terr;
structtcp_pcb*pcb;
ip_addr_tremote_addr;
u32_ti;
#ifLWIP_IPV6==1
remote_addr.type=IPADDR_TYPE_V6;
err=inet6_aton(TCP_SERVER_IPV6_ADDRESS,&remote_addr);
#else
err=inet_aton(TCP_SERVER_IP_ADDRESS,&remote_addr);
#endif/*LWIP_IPV6*/
if(!err){
xil_printf("InvalidServerIPaddress:%d
",err);
return;
}
/*CreateClientPCB*/
pcb=tcp_new_ip_type(IPADDR_TYPE_ANY);
if(!pcb){
xil_printf("ErrorinPCBcreation.outofmemory
");
return;
}
err=tcp_connect(pcb,&remote_addr,TCP_CONN_PORT,
tcp_client_connected);
if(err){
xil_printf("Errorontcp_connect:%d
",err);
tcp_client_close(pcb);
return;
}
client.client_id=0;
/*initializedatabufferbeingsentwithsameasusediniperf*/
for(i=0;i
tcp_fasttmr和tcp_slowtmr
在lwip的TCP視線中,快速定時器(tcp_fasttmr)和慢速定時器(tcp_slowtmr)都是為了TCP連接的維護而存在的,但它們關(guān)注的方面和執(zhí)行頻率是不同的。
運行頻率:
快速定時器:通常每250毫秒被調(diào)用一次(這是默認值,但可以配置)。
慢速定時器:通常每500毫秒被調(diào)用一次(這也是默認值,但同樣可以配置)。
關(guān)注的方面:
連接的生命周期管理:例如,關(guān)閉那些已經(jīng)結(jié)束但還沒有完全關(guān)閉的連接。
持續(xù)活動檢測:例如,檢查長時間沒有活動的連接,并可能發(fā)送探測數(shù)據(jù)段來檢查對方是否仍然活躍。
超時管理:管理那些因為長時間沒有響應(yīng)而需要關(guān)閉的連接。
擁塞控制:調(diào)整窗口大小和其他與流量控制相關(guān)的參數(shù)。
重傳管理:如果一個數(shù)據(jù)段沒有得到確認,它會被重新發(fā)送??焖俣〞r器負責(zé)處理這些重傳。
延遲確認:TCP不會立刻確認每一個接收到的數(shù)據(jù)段,而是稍作延遲,以期待有數(shù)據(jù)可以與確認一同發(fā)送,從而減少網(wǎng)絡(luò)的數(shù)據(jù)包數(shù)量。快速定時器可以觸發(fā)這些延遲確認的發(fā)送。
快速定時器 (tcp_fasttmr)
主要關(guān)注:
慢速定時器 (tcp_slowtmr)
主要關(guān)注:
簡而言之,快速定時器主要關(guān)注與數(shù)據(jù)傳輸直接相關(guān)的事務(wù),如重傳和確認,而慢速定時器則更多地關(guān)注連接的維護、超時和流控制。
tcp_write
tcp_write 函數(shù)用于將數(shù)據(jù)排入到一個TCP連接的發(fā)送隊列。它是應(yīng)用程序與 lwIP TCP層之間的一個關(guān)鍵接口,允許應(yīng)用程序發(fā)送數(shù)據(jù)到其TCP連接。
以下是關(guān)于 tcp_write 函數(shù)的一些關(guān)鍵點:
非阻塞:與某些TCP/IP實現(xiàn)不同,tcp_write 是非阻塞的。這意味著,如果當前沒有足夠的可用緩沖區(qū)來容納你想發(fā)送的數(shù)據(jù),函數(shù)將不會阻塞,而是返回一個錯誤。
排隊,不是直接發(fā)送:當你調(diào)用 tcp_write 時,你實際上是將數(shù)據(jù)放入發(fā)送隊列,而不是立即發(fā)送數(shù)據(jù)。真正的數(shù)據(jù)傳輸將在后續(xù)的 lwIP 處理中進行,這可能涉及與其他TCP機制的交互,如擁塞控制。
參數(shù):該函數(shù)通常接受以下參數(shù):
pcb:代表TCP連接的控制塊。
data:指向要發(fā)送數(shù)據(jù)的指針。
len:要發(fā)送的數(shù)據(jù)的長度。
flags:與數(shù)據(jù)發(fā)送相關(guān)的標志。例如,TCP_WRITE_FLAG_COPY 表示應(yīng)從應(yīng)用程序的數(shù)據(jù)緩沖區(qū)復(fù)制數(shù)據(jù)(而不是直接引用)。
確認機制:使用 tcp_write 發(fā)送的數(shù)據(jù)將在對方確認收到之后才從發(fā)送隊列中移除。這意味著,即使你已經(jīng)調(diào)用了 tcp_write,你也需要確保你的應(yīng)用程序繼續(xù)處理(例如,通過調(diào)用 tcp_output 或等待 lwIP 的主循環(huán))來確保數(shù)據(jù)被實際發(fā)送和確認。
合適的調(diào)用時間:為了避免不必要的網(wǎng)絡(luò)擁塞和效率低下,建議在連接建立后或在接收到數(shù)據(jù)或發(fā)送緩沖區(qū)有可用空間時(通過相關(guān)的TCP回調(diào)函數(shù))再調(diào)用 tcp_write。
tcp_write 是 lwIP 的TCP API的一部分,與其他函數(shù)(如 tcp_connect, tcp_listen, tcp_close 等)一起,提供了完整的TCP功能。在使用它時,重要的是要理解其工作原理,以及與其他TCP操作的交互方式。
審核編輯:劉清
-
FPGA
+關(guān)注
關(guān)注
1650文章
22217瀏覽量
628026 -
嵌入式系統(tǒng)
+關(guān)注
關(guān)注
41文章
3702瀏覽量
132810 -
SoC設(shè)計
+關(guān)注
關(guān)注
1文章
150瀏覽量
19445 -
TCP
+關(guān)注
關(guān)注
8文章
1416瀏覽量
82783 -
LwIP協(xié)議棧
+關(guān)注
關(guān)注
0文章
19瀏覽量
7831
原文標題:lwip代碼分析
文章出處:【微信號:傅里葉的貓,微信公眾號:傅里葉的貓】歡迎添加關(guān)注!文章轉(zhuǎn)載請注明出處。
發(fā)布評論請先 登錄
基于MM32F5270的Ethernet實現(xiàn)LwIP協(xié)議棧移植
LwIP協(xié)議棧的設(shè)計與實現(xiàn)資料分享!
嵌入式TCPIP協(xié)議棧LWIP的內(nèi)部結(jié)構(gòu)
Lwip協(xié)議棧的設(shè)計方案
lwip協(xié)議中文版
LwIP協(xié)議詳解
TCPIP協(xié)議棧的實現(xiàn)lwip
lwip協(xié)議棧源碼詳解說明
LWIP協(xié)議棧中Raw TCP中使用
使用LwIP協(xié)議棧淺析實戰(zhàn)分析(i.MX RT)

lwip協(xié)議棧代碼分析
評論