前面重點對Client的創(chuàng)建方式及使用方式進行了介紹,本節(jié)通過Server實驗對TCP通信過程進行一次介紹。
在TCP/IP協(xié)議中,傳輸層及以下層的機制是由內(nèi)核提供的。應(yīng)用層由用戶提供,應(yīng)用層程序?qū)νㄐ艛?shù)據(jù)進行解析處理,傳輸層及以下層處理通信的細節(jié)(將數(shù)據(jù)從一端傳入另外一端)。應(yīng)用層數(shù)據(jù)通過協(xié)議棧發(fā)送到網(wǎng)絡(luò)上時,每層協(xié)議都要增加一個數(shù)據(jù)部首(header),進行一次封裝。其中不同的協(xié)議層對數(shù)據(jù)包有不同的稱謂,在傳輸層叫段(segment),在網(wǎng)絡(luò)層叫做數(shù)據(jù)報(datagram),在鏈路層稱為幀(frame)。
在通信過程中,發(fā)送端執(zhí)行以下動作:首先程序進行編碼,確定通信的建立連接、發(fā)送數(shù)據(jù)的時間。接著建立TCP連接,TCP根據(jù)應(yīng)用指示負責(zé)建立連接、發(fā)送數(shù)據(jù)及斷開連接。TCP首部包括源端口號和目標(biāo)端口號、序號和校驗和,加完首部后數(shù)據(jù)包繼續(xù)往下傳遞到IP層,IP層加上IP首部包括地址等信息用于尋址操作,之后將數(shù)據(jù)繼續(xù)往下傳遞附加數(shù)據(jù)鏈路層首部。最后發(fā)送時的分組數(shù)據(jù)包會加上以太網(wǎng)包尾(用于循環(huán)冗余校驗)。
主機端:收到數(shù)據(jù)包后會在以太網(wǎng)數(shù)據(jù)包中找到MAC地址,判斷是否為自己的數(shù)據(jù)包,如果不是則丟棄。如果是傳遞給IP層處理,以此類推,不斷往上傳遞到TCP層。在TCP層通過校驗和判斷數(shù)據(jù)是否損壞,然后檢查是否按序號接收數(shù)據(jù),最后檢查端口號。處理完成這一切后數(shù)據(jù)包繼續(xù)往上層發(fā)送,即應(yīng)用層。如果出現(xiàn)主機空間已滿等情況,主機則會發(fā)送“處理異常”通知發(fā)送端。
實驗使用MB-039開發(fā)板,在工程中使用LwIP+FreeRTOS,實驗展示如何制作一個TCP Server,并收發(fā)數(shù)據(jù),實驗使用到的硬件如下:
如圖是MB-039(完整原理圖可以通過MM32官網(wǎng)下載)的ETH部分。
各個信號引腳對應(yīng)如下:
在進行Server實驗前,我們先了解需要使用到的API:
1)netconn_bind ()
2)netconn_listen ()
3)netconn_accept ()
以下分API展開介紹:
01、netconn_bind ()
從源碼中可以看出其主要功能:為conn(服務(wù)器端)綁定地址與端口號。
err_t netconn_bind(struct netconn* conn, const ip_addr_t* addr, u16_t port) { API_MSG_VAR_DECLARE(msg); err_t err; LWIP_ERROR("netconn_bind: invalid conn", (conn != NULL), return ERR_ARG;); #if LWIP_IPV4 /* Don't propagate NULL pointer (IP_ADDR_ANY alias) to subsequent functions */ if (addr == NULL) { addr = IP4_ADDR_ANY; } #endif /* LWIP_IPV4 */ #if LWIP_IPV4 LWIP_IPV6 if ((netconn_get_ipv6only(conn) == 0) ip_addr_cmp(addr, IP6_ADDR_ANY)) { addr = IP_ANY_TYPE; } #endif /* LWIP_IPV4 LWIP_IPV6 */ API_MSG_VAR_ALLOC(msg); API_MSG_VAR_REF(msg).conn = conn; API_MSG_VAR_REF(msg).msg.bc.ipaddr = API_MSG_VAR_REF(addr); API_MSG_VAR_REF(msg).msg.bc.port = port; err = netconn_apimsg(lwip_netconn_do_bind, API_MSG_VAR_REF(msg)); API_MSG_VAR_FREE(msg); return err; }
02、netconn_listen ()
netconn_listen指向的函數(shù)是:netconn_listen_with_backlog,作用:使服務(wù)器進入監(jiān)聽狀態(tài),等待遠端的連接請求。
err_t netconn_listen_with_backlog(struct netconn* conn, u8_t backlog) { #if LWIP_TCP API_MSG_VAR_DECLARE(msg); err_t err; /* This does no harm. If TCP_LISTEN_BACKLOG is off, backlog is unused. */ LWIP_UNUSED_ARG(backlog); LWIP_ERROR("netconn_listen: invalid conn", (conn != NULL), return ERR_ARG;); API_MSG_VAR_ALLOC(msg); API_MSG_VAR_REF(msg).conn = conn; #if TCP_LISTEN_BACKLOG API_MSG_VAR_REF(msg).msg.lb.backlog = backlog; #endif /* TCP_LISTEN_BACKLOG */ err = netconn_apimsg(lwip_netconn_do_listen, API_MSG_VAR_REF(msg)); API_MSG_VAR_FREE(msg); return err; #else /* LWIP_TCP */ LWIP_UNUSED_ARG(conn); LWIP_UNUSED_ARG(backlog); return ERR_ARG; #endif /* LWIP_TCP */ }
03、netconn_accept ()
netconn_accept(代碼較長,這里不進行粘貼)用于TCP服務(wù)器中,等待著遠端主機的連接請求,并且建立一個新的TCP連接,在調(diào)用這個函數(shù)之前需要通過調(diào)用 listen()函數(shù)讓服務(wù)器進入監(jiān)聽狀態(tài)。accept()函數(shù)的調(diào)用會阻塞應(yīng)用線程直至與遠程主機建立TCP連接。參數(shù)addr是一個返回結(jié)果參數(shù),它的值由accept()函數(shù)設(shè)置,其實就是遠程主機的地址與端口號等信息,當(dāng)新的連接已經(jīng)建立后,遠端主機的信息將保存在連接句柄中,能夠標(biāo)識連接對象。
了解了以上3個API,我們開始創(chuàng)建Server工程:
static void server(void* thread_param) { struct netconn* conn, *newconn; err_t err; LWIP_UNUSED_ARG(arg); #if LWIP_IPV6 conn = netconn_new(NETCONN_TCP_IPV6); netconn_bind(conn, IP6_ADDR_ANY, LOCAL_PORT); #else /* LWIP_IPV6 */ conn = netconn_new(NETCONN_TCP); //① netconn_bind(conn, IP_ADDR_ANY, LOCAL_PORT); //② #endif /* LWIP_IPV6 */ LWIP_ERROR("tcpecho: invalid conn", (conn != NULL), return;); printf("The local port number is%dnn", LOCAL_PORT); netconn_listen(conn); //③ while (1) { err = netconn_accept(conn, newconn); //④ if (err == ERR_OK) { struct netbuf* buf; void* data; u16_t len; while ((err = netconn_recv(newconn, buf)) == ERR_OK) { //⑤ do { netbuf_data(buf, data, len); err = netconn_write(newconn, data, len, NETCONN_COPY); //⑥ } while (netbuf_next(buf) >= 0); netbuf_delete(buf); //⑦ } netconn_close(newconn); //⑧ netconn_delete(newconn); //⑨ } } }
1、申請一個連接結(jié)構(gòu),指定參數(shù)是NETCONN_TCP,即TCP連接
2、綁定本地的IP地址與端口號
3、使TCP服務(wù)器進入監(jiān)聽狀態(tài)
4、處理客戶端的連接請求,當(dāng)只有當(dāng)有客戶端發(fā)送連接請求的時候才會處理,否則將進入阻塞態(tài),而客戶端的信息保存在newconn連接結(jié)構(gòu)中
5、接收數(shù)據(jù),并裝填進buf
6、對接收的數(shù)據(jù)進行轉(zhuǎn)發(fā)(指定為不拷貝方式NETCONN_COPY)
7、釋放數(shù)據(jù)空間
8、主動關(guān)閉客戶端的連接
9、釋放newconn空間
到這里已經(jīng)完成了工程的創(chuàng)建,看一下PC的IP地址,設(shè)備需要處于同一網(wǎng)段,以方便測試。
打開命令行窗口輸入:ipconfig
PC的地址為:192.168.105.34,在sys_arch.h文件中對DEST_IP_ADDR0 、DEST_IP_ADDR1、DEST_IP_ADDR2、DEST_IP_ADDR3進行修改,DEST_PORT 隨意修改。
#define LOCAL_PORT 2021 #define IP_ADDR0 192 #define IP_ADDR1 168 #define IP_ADDR2 105 #define IP_ADDR3 21
將程序下載入開發(fā)板中,使用NetAssist進行如下設(shè)置:
1)協(xié)議設(shè)置,此時設(shè)備為Server,則PC為Client
2)設(shè)置遠程主機地址(即設(shè)備地址)
3)端口號
點擊連接,若提示連接失敗,則Ping一下開發(fā)板地址,可以正常Ping通則檢查端口號;如果無法Ping通則需要對工程進行檢查。
任意輸入字符進行發(fā)送。
通過上圖可以觀察到發(fā)送成功,并且設(shè)備返回數(shù)據(jù)與發(fā)送數(shù)據(jù)一致,表明實驗成功。實驗程序請登錄我們的官網(wǎng)下載MM32F3270 SDK,工程路徑如下:~MM32F3270_Lib_Samples_V0.90Demo_appEthernet_DemoETH_RTOSFreertos_Server。
來源:靈動MM32MCU
審核編輯:湯梓紅
-
以太網(wǎng)
+關(guān)注
關(guān)注
40文章
5590瀏覽量
174966 -
通信
+關(guān)注
關(guān)注
18文章
6182瀏覽量
137418 -
Server
+關(guān)注
關(guān)注
0文章
94瀏覽量
24561 -
TCP
+關(guān)注
關(guān)注
8文章
1398瀏覽量
80455
發(fā)布評論請先 登錄
靈動微課堂 (第182講) | 基于MM32F3270 以太網(wǎng) Client_Socket使用
靈動微課堂 (第185講) | 基于MM32F3270 以太網(wǎng) Client使用
靈動微課堂 (第186講) | 基于MM32F3270 以太網(wǎng) Server使用
MM32F3270系列32位MCU的特點有哪些
靈動微電子MM32F3270系列MCU的特點介紹
基于MM32F3270 以太網(wǎng) Server使用

基于MM32F3270 以太網(wǎng) Client使用

評論