日韩av黄I国产麻豆传媒I国产91av视频在线观看I日韩一区二区三区在线看I美女国产在线I麻豆视频国产在线观看I成人黄色短片

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 编程资源 > 编程问答 >内容正文

编程问答

UIP协议栈移植到u-boot详解

發(fā)布時間:2023/12/20 编程问答 44 豆豆
生活随笔 收集整理的這篇文章主要介紹了 UIP协议栈移植到u-boot详解 小編覺得挺不錯的,現(xiàn)在分享給大家,幫大家做個參考.


???????? Author: 楊正? date2014.11.5? Emaily2012ww@gmail.com?QQ: 1209758756

1、uip簡介

?????? Uip網絡是一個簡單好用的嵌入式協(xié)議棧,易于移植且消耗的內存空間較少,應用于很多嵌入式產品。uIP 協(xié)議棧去掉了完整的TCP/IP系統(tǒng)中不常用的功能,簡化了通訊流程,只保留了網絡通信必須使用的協(xié)議,設計重點放在了IP/TCP/ICMP/UDP/ARP這些網絡層和傳輸層的協(xié)議上,因此保證了其代碼的通用性和結構的穩(wěn)定性。由于uIP協(xié)議棧專門為嵌入式系統(tǒng)而設計,因此還具有以下優(yōu)越功能:

(1)代碼非常少,其協(xié)議棧代碼不到6K,方便閱讀與移植。

(2)占用的內存數極少,RAM占用僅幾百字節(jié)。

(3)它的硬件處理層、協(xié)議棧層和應用層共用一個全局緩存區(qū),不存在數據拷貝,而且發(fā)送和接收都是依靠這個緩存區(qū),極大的節(jié)省了空間和時間。

(4)支持多個主動連接和被動連接并發(fā)。

(5)它的源代碼中提供一套實例程序:web 客戶端,web 服務器,Telnet 服務器,電子郵件發(fā)送程序(SMTP 客戶端),DNS 主機名解析程序等。

(6)在數據的處理上采用輪循機制,不需要操作系統(tǒng)的支持。由于 uIP 對資源的需求少而且移植容易,大部分的8位微控制器都使用過uIP協(xié)議棧, 而且很多著名的嵌入式產品和項目(如衛(wèi)星,Cisco 路由器,無線傳感器網絡)中均在使用uIP 協(xié)議棧。

下面就開始進入移植過程。

2、uip移植

2.1 uip架構

在移植uip之前先簡單看一下uip的大致框架,uIP相當于一個代碼庫,通過一系列的函數實現(xiàn)與底層硬件和高層應用程序之間的通訊,對于整個系統(tǒng)來說它內部的協(xié)議組是透明的,從而增加了協(xié)議的通用性。uIP協(xié)議棧與系統(tǒng)底層和高層應用之間的關系如下:




uIP 協(xié)議棧主要提供了三個函數供系統(tǒng)底層調用。即uip_init(), uip_input()和uip_periodic()。其與應用程序的主要接口是UIP_APPCALL( )。

uip_init()是系統(tǒng)初始化時調用的,主要用于初始化協(xié)議棧的偵聽端口和默認所有連接是關閉的。當網卡驅動收到一個輸入包時,將其放入全局緩沖區(qū) uip_buf 中,包的大小由全局變量uip_len 約束。同時將調用uip_input()函數,這個函數將會根據包首部的協(xié)議處理這個包并在需要時調用應用程序。當uip_input()返回時,一個輸出包同樣放在全局緩沖區(qū)uip_buf 里,并把大小賦給uip_len。若uip_len 是0,則說明沒有包要發(fā)送;否則調用底層系統(tǒng)的發(fā)包函數就會將包發(fā)送到網絡上。uIP周期計時用于驅動所有的uIP內部時鐘事件:當周期計時激發(fā),每一個TCP連接都會調用uIP函數uip_periodic()。類似于uip_input()函數,uip_periodic()函數返回時,輸出的IP 包要放到uip_buf 中,供底層系統(tǒng)查詢uip_len 的大小并發(fā)送。由于TCP/IP 的應用場景很多,所以應用程序作為單獨的模塊由用戶實現(xiàn)。uIP 協(xié)議棧提供一系列接口函數供用戶程序調用,其中大部分函數是作為C的宏命令實現(xiàn)的,主要是為了速度、代碼大小、堆棧和效率的使用。用戶需要將應用層入口程序作為接口提供給uIP協(xié)議棧,并將這個函數定義為UIP_APPCALL()。這樣以來,uIP在接受到底層傳來的數據包后,在需要送到上層應用程序處理的地方,調用UIP_APPCALL(),在不用修改協(xié)議棧的情況下可以適配不同的應用程序。

?

2.2 UIP移植過程

2.2.1 拷貝UIP-0.9到u-boot-2010.06/net目錄

以u-boot_sources_for_tp-link_AR9331_by_pepe2k為藍本進行移植,將該藍本里面的uip-0.9這個目錄復制到我的u-boot 的u-boot-2010.06/net目錄下,這個uip-0.9里面有這個一些文件:

ap121.h? ctype.h?fsdata.c?? tapdev.c??? uip_arch.h?uip_arp.h?uip.h

ar7240.h??????? flash.h?fsdata.h? httpd.c? main.c???tapdev.h??? cmd_confdefs.h? fs.c????fs.h?????httpd.h? Makefile? uip_arch.c?uip_arp.c??uip.c?????uipopt.h

當然不是所有的文件都用到。拷貝過來以后要修改makefile不然編譯會通不過的,我的makefile修改如下:

CC=g++

CFLAGS=-Wall -O2-fpack-struct -DDUMP=0

?

CFLAGS += -I../../include

?

all: uip

?

uip: uip.o uip_arch.otapdev.o httpd.o main.o fs.o uip_arp.o

#uip: uip.o uip_arch.otapdev.o main.o fs.o uip_arp.o

??? $(CC) $(CFLAGS) $(LDFLAGS) $^ -o $@

?

%.o: %.c

??? $(CC) $(CFLAGS) -c? $^ -o $@

?

clean:

rm-f *.o *~ *core uip


接下來就進入u-boot-2010.06/net目錄下的net.c文件,其實整個移植過程都是在這個文件里面打轉。

?

2.2.2 在u-boot-2010.06/net/net.c里面添加函數

net.c里面添加NetReceiveHttpd()函數:

void NetReceiveHttpd(volatileuchar * inpkt, int len) {

??? memcpy(uip_buf, (const void *) inpkt, len);

??? uip_len = len;

?

#ifdef ET_DEBUG???? //debug?by yangzheng

??? DBG("NetReceiveHttpd buf->type =%04X\n", ntohs(BUF->type));

#endif

??? if (BUF->type == htons(UIP_ETHTYPE_IP)){

#ifdef ET_DEBUG???? //debug?by yangzheng

??? DBG("buf type isUIP_ETHTYPE_IP\n");

#endif

??????? uip_arp_ipin();?? //處理傳入的ip

??????? DBG("file=%s, func=%s,line=%d\n", __FILE__, __FUNCTION__, __LINE__ );

??????? DBG("uip_len2=%d\n",uip_len);

??????? uip_input();? //從上往下封裝包的函數,這個函數會調用UIP_APPCALL()

??????? DBG("file=%s, func=%s,line=%d\n", __FILE__, __FUNCTION__, __LINE__ );

??????? DBG("uip_len2=%d\n",uip_len);

??????? if (uip_len > 0) {

??????????? DBG("ipin->uip bufferinside data\n");

??????????? DBG("file=%s, func=%s,line=%d\n", __FILE__, __FUNCTION__, __LINE__ );

??????????? uip_arp_out();?? //arp請求發(fā)送函數

??????????? NetSendHttpd();?? //調用網卡驅動的發(fā)送函數

??????? }

??? } else if (BUF->type ==htons(UIP_ETHTYPE_ARP)) {

#ifdef ET_DEBUG???? //debug?by yangzheng

??? DBG("buf type isUIP_ETHTYPE_ARP\n");

#endif

uip_arp_arpin();? ?//處理arp應答

??????? DBG("uip_len3=%d\n",uip_len);

??????? if (uip_len > 0) {

??????????? DBG("arpin->uip bufferinside data\n");

??????????? DBG("file=%s, func=%s,line=%d\n", __FILE__, __FUNCTION__, __LINE__ );

??????????? NetSendHttpd();

??????? }

??? }

}

在net.c的NetReceive()函數里面還要添加如下代碼,調用NetReceiveHttpd()函數:

?

if(webfailsafe_is_running)

{

NetReceiveHttpd(inpkt,len);? //這個函數上面已經說明

return;

}

?

?

?

?

?

這里有一個函數需要說明一下,因為我在這里徘徊了很久。ARP請求發(fā)送函數:

void uip_arp_out(void)(在uip-0.9目錄的uip_arp.c里面定義)。

*==================================================================
*
為傳出的IP包添加以太網頭并看是否需要發(fā)送ARP請求.?
*
此函數應該在發(fā)送IP包時調用,它會檢查IP包的目的IP地址,看看以太網應該使用什么目的MAC地址.
*
如果目的IP地址是在局域網中(IP地址與子網掩碼的與邏輯決定),函數就會從ARP緩存表中查找有
*
無對應項.若有,就取對應的MAC地址,加上以太網頭,并返回,否則uip_buf[]中的數據包會被替換成一個
*
目的IP在址的ARP請求.原來的IP包會被簡單的仍掉,此函數假設高層協(xié)議(TCP)會最終重傳扔掉的包.
*
如果目標IP地址并非一個局域網IP,則會使用默認路由的IP地址.
* uip_len.
函數返回時,uip_buf[]中已經有了一個包,其長度由uip_len指定.

*===================================================================

void?uip_arp_out(void)

{

??? ?struct arp_entry?*tabptr=0;

?

ipaddr[0] =IPBUF->destipaddr[0];

??? ipaddr[1] = IPBUF->destipaddr[1];

???DBG("****************ipaddr[0]=%x******************ipaddr[1]=%x\n",ipaddr[0], ???ipaddr[1]);

??? DBG("file=%s, func=%s,line=%d\n", __FILE__, __FUNCTION__, __LINE__ );

?????

? for(i = 0; i < UIP_ARPTAB_SIZE; ++i) {

??? tabptr = &arp_table[i];

??? if(ipaddr[0] == tabptr->ipaddr[0]&&

?????? ipaddr[1] == tabptr->ipaddr[1])

????? break;

? }

?

?????????if(i?==?UIP_ARPTAB_SIZE)

?? ? ???{

????? ? ???/*?如果遍歷到頭沒找到,將原IP包替換為ARP請求并返回?*/

?????????????

??? memset(BUF->ethhdr.dest.addr, 0xff, 6);

??? memset(BUF->dhwaddr.addr, 0x00, 6);

??? memcpy(BUF->ethhdr.src.addr,uip_ethaddr.addr, 6);

??? memcpy(BUF->shwaddr.addr,uip_ethaddr.addr, 6);

???

??? BUF->dipaddr[0] = ipaddr[0];

??? BUF->dipaddr[1] = ipaddr[1];

??? BUF->sipaddr[0] = uip_hostaddr[0];

??? BUF->sipaddr[1] = uip_hostaddr[1];

??? BUF->opcode = HTONS(ARP_REQUEST); /* ARPrequest. */

??? BUF->hwtype = HTONS(ARP_HWTYPE_ETH);

??? BUF->protocol = HTONS(UIP_ETHTYPE_IP);

??? BUF->hwlen = 6;

??? BUF->protolen = 4;

??? BUF->ethhdr.type = HTONS(UIP_ETHTYPE_ARP);

??? DBG("file=%s, func=%s,line=%d\n", __FILE__, __FUNCTION__, __LINE__ );

?

??? uip_appdata = &uip_buf[40 +UIP_LLH_LEN];

???

??? uip_len = sizeof(struct arp_hdr);

??? return;

? }

?

? /* Build an ethernet header. */

? memcpy(IPBUF->ethhdr.dest.addr,tabptr->ethaddr.addr, 6);

? memcpy(IPBUF->ethhdr.src.addr,uip_ethaddr.addr, 6);

?

? IPBUF->ethhdr.type =HTONS(UIP_ETHTYPE_IP);

?

? uip_len += sizeof(struct uip_eth_hdr);

}

?

接下來看arp應答函數:uip_arp_arpin()在uip-0.9目錄的uip_arp.c里面定義。

uip_arp_arpin()函數主要是處理ARP應答。這個函數是在設備接收到ARP包時,由驅動程序調用的.如果收到是ARP包是一個對本地主機上次發(fā)送的ARP請求的應答,那么就從包中取得自己想要的主機的MAC地址,加入自己的ARP緩存表中.如果收到是一個ARP請求,那就把自己的MAC地址打包成一個ARP應答,發(fā)送給請求的主機。

/*-----------------------------------------------------------------------------------*/

/**

?* ARP processing for incoming ARP packets.

?*

?* This function should be called by the devicedriver when an ARP

?* packet has been received. The function willact differently

?* depending on the ARP packet type: if it is areply for a request

?* that we previously sent out, the ARP cachewill be filled in with

?* the values from the ARP reply. If theincoming ARP packet is an ARP

?* request for our IP address, an ARP replypacket is created and put

?* into the uip_buf[] buffer.

?*

?* When the function returns, the value of theglobal variable uip_len

?* indicates whether the device driver shouldsend out a packet or

?* not. If uip_len is zero, no packet should besent. If uip_len is

?* non-zero, it contains the length of theoutbound packet that is

?* present in the uip_buf[] buffer.

?*

?* This function expects an ARP packet with aprepended Ethernet

?* header in the uip_buf[] buffer, and thelength of the packet in the

?* global variable uip_len.

?*/

/*-----------------------------------------------------------------------------------*/

void

uip_arp_arpin(void)

{

? if(uip_len < sizeof(struct arp_hdr)) {

??? uip_len = 0;

??? return;

? }

?

? uip_len = 0;

?

? int ar = HTONS(ARP_REQUEST);

?

? case HTONS(ARP_REQUEST):? //arp請求:1arp應答:2

??? /* ARP request. If it asked for ouraddress, we send out a

?????? reply. */

??? if(BUF->dipaddr[0] == uip_hostaddr[0]&&

?????? BUF->dipaddr[1] == uip_hostaddr[1]) {

????? /* The reply opcode is 2. */

????? BUF->opcode = HTONS(2); ?//回應的操作碼是2

?????

????? //將收到的arp包的發(fā)送端以太網地址變?yōu)槟康囊蕴W地址

????? memcpy(BUF->dhwaddr.addr,BUF->shwaddr.addr, 6);

????? //將自己的以太網地址賦值給arp包的發(fā)送端以太網地址

????? memcpy(BUF->shwaddr.addr,uip_ethaddr.addr, 6);

????? memcpy(BUF->ethhdr.src.addr,uip_ethaddr.addr, 6);

????? memcpy(BUF->ethhdr.dest.addr,BUF->dhwaddr.addr, 6);

?????

????? BUF->dipaddr[0] = BUF->sipaddr[0];

????? BUF->dipaddr[1] = BUF->sipaddr[1];

?

????? BUF->sipaddr[0] = uip_hostaddr[0];

????? BUF->sipaddr[1] = uip_hostaddr[1];

?

?

????? BUF->ethhdr.type =HTONS(UIP_ETHTYPE_ARP);?????

????? uip_len = sizeof(struct arp_hdr);

?????

??? }?????

??? break;

? case HTONS(ARP_REPLY):

??? /* ARP reply. We insert or update the ARPtable if it was meant

?????? for us. */

??? if(BUF->dipaddr[0] == uip_hostaddr[0]&&

?????? BUF->dipaddr[1] == uip_hostaddr[1]) {

?

???? ??uip_arp_update(BUF->sipaddr,&BUF->shwaddr);

??? }

??? break;

? }

?

? return;

}

?

net.c里面還要添加如下函數:

#define BUF???????????? ((struct uip_eth_hdr*)&uip_buf[0])

voidNetSendHttpd(void) {

??? volatile uchar *tmpbuf = NetTxPacket;

??? int i;

?

??? for (i = 0; i < 40 + UIP_LLH_LEN; i++) {

??????? tmpbuf[i] = uip_buf[i];

??????? //printf("uip_buf[%d]=%d\n",i, uip_buf[i]);

??? }

?

??? for (; i < uip_len; i++) {

??????? tmpbuf[i] = uip_appdata[i - 40 -UIP_LLH_LEN];

??? }

??? eth_send(NetTxPacket, uip_len);? //這里就是調用uboot里面網卡?驅動的發(fā)送函數

??? //NetSendPacket(NetTxPacket, uip_len);

}

?

voidHttpdHandler(void) {??? //

??? int i;

?

??? for (i = 0; i < UIP_CONNS; i++) {

??????? uip_periodic(i);

?

??????? if (uip_len > 0) {

??????????? uip_arp_out();

??????????? NetSendHttpd();

??????? }

??? }

?

??? // TODO: check this

??? if (++arptimer == 20) {

??????? uip_arp_timer();

??????? arptimer = 0;

??? }

}

?

// start http daemon

void HttpdStart(void){

??? DBG("file=%s, func=%s,line=%d\n", __FILE__, __FUNCTION__, __LINE__);

??? uip_init();??? ?//下面講到

??? httpd_init();

}

?

?

uip協(xié)議結合網卡芯片組成嵌入式網卡,硬件提供能力,uip提供策略。由上往下逐步封裝數據,如:

應用層----------傳輸層------------網絡層---------------數據鏈路層---------物理層

應用數據------>tcp封裝頭------>IP封裝頭-----------MAC封裝+尾部------->發(fā)送

?

任何事物都需要一個初始化的過程,uip協(xié)議棧也不例外,uip協(xié)議通過uip_init()來初始化。uip_init()函數里主要的工作是:

1.? uip_state結構體全部清零。

2.? 初始化用于TCP連接的uip_conn結構體,將連接狀態(tài)置為close

3.? 設置用于TCP連接的端口號lastport=1024

4.? 如果定義了UDP,同樣也要初始化。

/*-----------------------------------------------------------------------------------*/

void

uip_init(void)

{

? for(c = 0; c < UIP_LISTENPORTS; ++c) {

??? uip_listenports[c] = 0;

? }

? for(c = 0; c < UIP_CONNS; ++c) {

??? uip_conns[c].tcpstateflags = CLOSED;

? }

#if UIP_ACTIVE_OPEN

? lastport = 1024;

#endif /*UIP_ACTIVE_OPEN */

?

#if UIP_UDP

? for(c = 0; c < UIP_UDP_CONNS; ++c) {

??? uip_udp_conns[c].lport = 0;

? }

#endif /* UIP_UDP */

?

? /* IPv4 initialization. */

#if UIP_FIXEDADDR == 0

? uip_hostaddr[0] = uip_hostaddr[1] = 0;

#endif /*UIP_FIXEDADDR */

}

/*-----------------------------------------------------------------------------------*/

?

?

net.c里面還要添加NetLoopHttpd(void)函數,這個函數前半部分可以模仿net.c里面的NetLoop()函數。紅色部分是字節(jié)序的轉換,在移植過程當中,字節(jié)序轉換也是花了很長時間。

int NetLoopHttpd(void){

?????? bd_t *bd = gd->bd;

?????? unsigned short int ip[2];

?????? unsigned char ethinit_attempt = 0;

?????? struct uip_eth_addr eaddr;

?

#ifdef CONFIG_NET_MULTI

?????? NetRestarted = 0;

?????? NetDevExists = 0;

#endif

?

?????? /* XXX problem with bss workaround */

?????? NetArpWaitPacketMAC = NULL;

?????? NetArpWaitTxPacket = NULL;

?????? NetArpWaitPacketIP = 0;

?????? NetArpWaitReplyIP = 0;

?????? NetArpWaitTxPacket = NULL;

?????? NetTxPacket = NULL;

?

?????? if (!NetTxPacket) {

????????????? int i;

????????????? // Setup packet buffers, alignedcorrectly.

????????????? NetTxPacket = &PktBuf[0] +(PKTALIGN - 1);

????????????? NetTxPacket -= (ulong) NetTxPacket% PKTALIGN;

?

????????????? for (i = 0; i < PKTBUFSRX; i++){

???????????????????? NetRxPackets[i] =NetTxPacket + (i + 1) * PKTSIZE_ALIGN;

????????????? }

?????? }

?

?????? if (!NetArpWaitTxPacket) {

????????????? NetArpWaitTxPacket =&NetArpWaitPacketBuf[0] + (PKTALIGN - 1);

????????????? NetArpWaitTxPacket -= (ulong)NetArpWaitTxPacket % PKTALIGN;

????????????? NetArpWaitTxPacketSize = 0;

?????? }

?

?????? // restart label

?????? restart:

?

?????? eth_halt();

?

#ifdef CONFIG_NET_MULTI

?????? eth_set_current();

#endif

?

??? int ret = eth_init(bd);

??? DBG("eth_init = %d\n", ret);

?????? while(ethinit_attempt < 10){

????????????? if(!eth_init(bd)){???? //eth_init->!eth_init???? add by yangzheng

??????????? DBG("file=%s, func=%s,line=%d\n", __FILE__, __FUNCTION__, __LINE__);

???????????????????? ethinit_attempt = 0;

???????????????????? break;

????????????? } else {

???????????????????? ethinit_attempt++;

???????????????????? eth_halt();

???????????????????? milisecdelay(1000);

????????????? }

?????? }

?

?????? if (ethinit_attempt > 0) {

????????????? eth_halt();

????????????? printf("## Error: couldn'tinitialize eth (cable disconnected?)!\n\n");

????????????? return (-1);

?????? }

?

?????? // get MAC address

#ifdefCONFIG_NET_MULTI

?????? memcpy(NetOurEther,eth_get_dev()->enetaddr, 6);

#else

?????? eth_getenv_enetaddr("ethaddr",NetOurEther);

#endif

?

??? //獲取物理地址

?????? eaddr.addr[0] = NetOurEther[0];

?????? eaddr.addr[1] = NetOurEther[1];

?????? eaddr.addr[2] = NetOurEther[2];

?????? eaddr.addr[3] = NetOurEther[3];

?????? eaddr.addr[4] = NetOurEther[4];

?????? eaddr.addr[5] = NetOurEther[5];

??? DBG("file=%s, func=%s,line=%d\n", __FILE__, __FUNCTION__, __LINE__);

??? DBG("%x:%x:%x:%x:%x:%x\n",eaddr.addr[0], eaddr.addr[1], eaddr.addr[2], eaddr.addr[3], eaddr.addr[4],eaddr.addr[5]);

?

?????? // set MAC address

?????? uip_setethaddr(eaddr);

?

?????? // set ip and other addresses

?????? // TODO: do we need this with uIP stack?

?????? NetCopyIP(&NetOurIP,&bd->bi_ip_addr);

?????? NetOurGatewayIP = getenv_IPaddr("gatewayip");

?????? NetOurSubnetMask =getenv_IPaddr("netmask");

?????? NetOurVLAN =getenv_VLAN("vlan");

?????? NetOurNativeVLAN =getenv_VLAN("nvlan");

?

?????? // start server...

//???? printf("HTTP server is starting at IP:%ld.%ld.%ld.%ld\n", (bd->bi_ip_addr & 0xff000000) >> 24,(bd->bi_ip_addr & 0x00ff0000) >> 16, (bd->bi_ip_addr &0x0000ff00) >> 8, (bd->bi_ip_addr & 0x000000ff));

//?? printf("HTTP server is starting at IP:%ld.%ld.%ld.%ld\n", (bd->bi_ip_addr & 0x000000ff),(bd->bi_ip_addr & 0x0000ff00) >> 8, (bd->bi_ip_addr & 0x00ff0000)>> 16, (bd->bi_ip_addr & 0xff000000) >> 24);

?

??? IPaddr_t x =ntohl(bd->bi_ip_addr); //add ?yangzheng

??? char tmp[22];

??? ip_to_string(bd->bi_ip_addr,tmp);

??? printf("HTTP server is starting at IP: %s\n",tmp);

?

??? DBG("file=%s, func=%s, line=%d\n",__FILE__, __FUNCTION__, __LINE__);

?????? HttpdStart();? //這里就會初始化uiphttp

??? DBG("file=%s, func=%s,line=%d\n", __FILE__, __FUNCTION__, __LINE__);

?

?????? // set local host ip address

?????? //ip[1] = ((bd->bi_ip_addr &0xFFFF0000) >> 16);

?????? //ip[0] = (bd->bi_ip_addr &0x0000FFFF);

?????? ip[0] = htons(((x& 0xFFFF0000) >> 16));?? //dbgyangzheng

?????? ip[1] = htons((x & 0x0000FFFF));

?

?????? uip_sethostaddr(ip);

?

?????? // set network mask (255.255.255.0 ->local network)

?????? ip[0] = ((0xFFFFFF00 & 0xFFFF0000)>> 16);

?????? ip[1] = (0xFFFFFF00 & 0x0000FFFF);

?????? //ip[0] = htons(0xFFFF); //dbg yangzheng

?????? //ip[1] = htons(0xFF00);

?

?????? uip_setnetmask(ip);

??? ip[0] = 0xFFFF;? //dbg yangzheng

??? ip[1] = 0xFFFF;

??? uip_setdraddr(ip);

?

?????? // should we also set default router ipaddress?

?????? //uip_setdraddr();

?

?????? // show current progress of the process

?????? do_http_progress(WEBFAILSAFE_PROGRESS_START);

?

?????? webfailsafe_is_running = 1;

?

?????? // infinite loop

?????? for (;;) {

?

????????????? // TODO: ??

????????????? WATCHDOG_RESET();

?

????????????? /*

????????????? ?*??? Checkthe ethernet for a new packet.

????????????? ?*??? Theethernet receive routine will process it.

????????????? ?*/

????????????? if (eth_rx() > 0) {

???????????????????? HttpdHandler();

????????????? }

?

????????????? // if CTRL+C was pressed ->return!

????????????? if (ctrlc()) {

???????????????????? eth_halt();

???????????????????? printf("\nWeb failsafemode aborted!\n\n");

???????????????????? return (-1);

????????????? }

?

????????????? // until upload is not completed,get back to the start of the loop

????????????? if(!webfailsafe_ready_for_upgrade) continue;

?

????????????? // stop eth interface

????????????? eth_halt();

?

????????????? // show progress

?????? ???do_http_progress(WEBFAILSAFE_PROGRESS_UPLOAD_READY);

?

????????????? // try to make upgrade!

// try to makeupgrade!

??????? if ( !do_upgrade());??? //這個函數后面會重點說明

??????? {

??????????? DBG("file=%s, func=%s,line=%d\n", __FILE__, __FUNCTION__, __LINE__);

???????????do_http_progress(WEBFAILSAFE_PROGRESS_UPGRADE_READY);

??????????? do_reset(NULL, 0, 0, NULL);? //這個函數是在uboot里面的

??????????? return 0;

??????? }

#if 0

????????????? if(do_http_upgrade(NetBootFileXferSize, webfailsafe_upgrade_type) >= 0) {

???????????????????? do_http_progress(WEBFAILSAFE_PROGRESS_UPGRADE_READY);

???????????????????? udelay(1000 * 10);

???????????????????? do_reset(0, 0, 0, 0);

???????????????????? return 0;

????????????? }

#endif

????????????? break;

?????? }

?

?????? webfailsafe_is_running = 0;

?????? webfailsafe_ready_for_upgrade = 0;

?????? webfailsafe_upgrade_type =WEBFAILSAFE_UPGRADE_TYPE_FIRMWARE;

?

?????? NetBootFileXferSize = 0;

??? DBG("file=%s, func=%s,line=%d\n", __FILE__, __FUNCTION__, __LINE__);

?

?????? do_http_progress(WEBFAILSAFE_PROGRESS_UPGRADE_FAILED);

?

?????? // go to restart

?????? goto restart;

?

?????? return -1;

}

?

?

?

下面就說一下do_upgrade()函數。我移植uip協(xié)議棧的目的就是通過這個協(xié)議來升級uboot,內核,文件系統(tǒng)。這個函數就是將通過網頁上傳到buffer的數據即bin文件寫到Flash里去,那么具體寫的函數可以參考u-boot-2010.06/common/cmd_sf.c里面的函數。(這里解釋一下,上傳的文件時存放在一個buffer里面,而不是直接寫到Flash里面,所以需要用下面這個函數來寫到Flash)我也是參考uboot里面cmd_sf.c文件,添加了如下代碼:

int do_upgrade (void)

{

???DBG("file=%s, func=%s, line=%d\n", __FILE__, __FUNCTION__,__LINE__);

???printf("+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++\n");

???printf("+zmodo_upgrade - upgrade kenel ,rootfilesystem,andAPP\n");

???printf("+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++\n");

??? int argc =3;

??? char*argv[10];

?

??? argv[0] ="sf";

???argv[1]="probe";

??? argv[2]="0";

???do_spi_flash_probe(argc - 1, argv + 1);

?

??? argc = 4;

??? argv[0] ="sf";

??? argv[1] ="erase";

??? argv[2] ="0";

??? //argv[3]= "80000";

??? argv[3] ="1000000"; ??//所燒錄文件的大小,我的是燒錄的是整個燒片文件16M

???do_spi_flash_erase(argc - 1, argv + 1);

?

??? argc = 5;

??? argv[0] ="sf";

??? argv[1] ="write";

??? argv[2] ="82000000";

??? argv[3] ="0";

??? //argv[4]= "80000";

??? argv[4] ="1000000";

???do_spi_flash_read_write(argc - 1, argv + 1);

?

??? return 0;

}???????

?

下面三個函數我完全是從u-boot-2010.06/common/cmd_sf.c文件里面拷貝過來的,僅供參考:

static int do_spi_flash_probe(int argc, char*argv[])

{

?????? unsignedint bus = 0;

?????? unsignedint cs;

?????? unsignedint speed = CONFIG_SF_DEFAULT_SPEED;

?????? unsignedint mode = CONFIG_SF_DEFAULT_MODE;

?????? char*endp;

?????? structspi_flash *new;

?

?????? if (argc< 2)

????????????? gotousage;

?

?????? cs =simple_strtoul(argv[1], &endp, 0);

?????? if(*argv[1] == 0 || (*endp != 0 && *endp != ':'))

????????????? gotousage;

?????? if(*endp == ':') {

????????????? if(endp[1] == 0)

???????????????????? gotousage;

?

????????????? bus= cs;

????????????? cs= simple_strtoul(endp + 1, &endp, 0);

????????????? if(*endp != 0)

???????????????????? gotousage;

?????? }

?

?????? if (argc>= 3) {

????????????? speed= simple_strtoul(argv[2], &endp, 0);

????????????? if(*argv[2] == 0 || *endp != 0)

???????????????????? gotousage;

?????? }

?????? if (argc>= 4) {

????????????? mode= simple_strtoul(argv[3], &endp, 16);

????????????? if(*argv[3] == 0 || *endp != 0)

???????????????????? gotousage;

?????? }

?

?????? new =spi_flash_probe(bus, cs, speed, mode);

?????? if(!new) {

????????????? printf("Failedto initialize SPI flash at %u:%u\n", bus, cs);

????????????? return1;

?????? }

?

?????? if(flash)

????????????? spi_flash_free(flash);

?????? flash =new;

?

?????? printf("%uKiB %s at %u:%u is now current device\n",

???????????????????? flash->size>> 10, flash->name, bus, cs);

?

?????? return0;

?

usage:

?????? puts("Usage:sf probe [bus:]cs [hz] [mode]\n");

?????? return1;

}

?

static int do_spi_flash_read_write(int argc, char*argv[])

{

?????? unsignedlong addr;

?????? unsignedlong offset;

?????? unsignedlong len;

?????? void*buf;

?????? char*endp;

?????? int ret;

?????? structmtd_info_ex *spiflash_info = get_spiflash_info();

?

?????? if (argc< 4)

????????????? gotousage;

?

?????? addr =simple_strtoul(argv[1], &endp, 16);

?????? if(*argv[1] == 0 || *endp != 0)

????????????? gotousage;

?????? offset =simple_strtoul(argv[2], &endp, 16);

?????? if(*argv[2] == 0 || *endp != 0)

????????????? gotousage;

?????? len =simple_strtoul(argv[3], &endp, 16);

?????? if(*argv[3] == 0 || *endp != 0)

????????????? gotousage;

????????????? if(offset + len >

????????????? spiflash_info->chipsize* spiflash_info->numchips) {

???????????????????? printf(

???????????????????? "ERROR:read/write area is out of range!\n\n");

???????????????????????????????????????????????? return-1;

????????????? }

?

?????? buf =map_physmem(addr, len, MAP_WRBACK);

?????? if(!buf) {

????????????? puts("Failedto map physical memory\n");

????????????? return1;

?????? }

?

?????? if(strcmp(argv[0], "read") == 0)

????????????? ret= spi_flash_read(flash, offset, len, buf);

?????? else {

????????????? unsignedlong write_start, write_len, write_step;

????????????? intpercent_complete = -1;

????????????? char*pbuf = buf;

?

????????????? write_start= offset;

????????????? write_len?? = len;

????????????? write_step? = spiflash_info->erasesize;

?

????????????? while(len > 0) {

???????????????????? if(len < write_step)

??????????????????????????? write_step= len;

?

???????????????????? ret= spi_flash_write(flash, offset, write_step, pbuf);

???????????????????? if(ret)

??????????????????????????? break;

?

???????????????????? offset+= write_step;

???????????????????? pbuf?? += write_step;

???????????????????? len??? -= write_step;

?

???????????????????? do{

??????????????????????????? unsignedlong long n = (unsigned long long)

?????????????????????????????????? (offset- write_start) * 100;

??????????????????????????? intpercent;

?

??????????????????????????? do_div(n,write_len);

??????????????????????????? percent= (int)n;

?

??????????????????????????? /*output progress message only at whole percent

??????????????????????????? ?* steps to reduce the number of messages

??????????????????????????? ?* printed on (slow) serial consoles

??????????????????????????? ?*/

??????????????????????????? if(percent != percent_complete) {

?????????????????????????????????? percent_complete= percent;

?

?????????????????????????????????? printf("\rWritingat 0x%lx -- %3d%% "

????????????????????????????????????????? "complete.",offset, percent);

??????????????????????????? }

???????????????????? }while (0);

????????????? }

?????? }

?????? puts("\n");

?

?????? unmap_physmem(buf,len);

?

?????? if (ret){

????????????? printf("SPIflash %s failed\n", argv[0]);

????????????? return1;

?????? }

?

?????? return0;

?

usage:

?????? printf("Usage:sf %s addr offset len\n", argv[0]);

?????? return1;

}

?

static int do_spi_flash_erase(int argc, char*argv[])

{

?????? unsignedlong offset;

?????? unsignedlong len;

?????? char*endp;

?????? int ret;

?????? structmtd_info_ex *spiflash_info = get_spiflash_info();

?????? unsignedlong erase_start, erase_len, erase_step;

?????? intpercent_complete = -1;

?

?????? if (argc< 3)

????????????? gotousage;

?

?????? offset =simple_strtoul(argv[1], &endp, 16);

?????? if(*argv[1] == 0 || *endp != 0)

????????????? gotousage;

?????? len =simple_strtoul(argv[2], &endp, 16);

?????? if(*argv[2] == 0 || *endp != 0)

????????????? gotousage;

?

?????? if(offset + len > spiflash_info->chipsize * spiflash_info->numchips) {

????????????? printf("ERROR:erase area is out of range!\n\n");

????????????? return1;

?????? }

?

?????? if(offset & (spiflash_info->erasesize-1)) {

????????????? printf("ERROR:erase start address is not block aligned!\n\n");

????????????? return1;

?????? }

?

?????? if (len& (spiflash_info->erasesize-1)) {

????????????? printf("ERROR:erase length is not block aligned!\n\n");

????????????? return1;

?????? }

?

?????? erase_start= offset;

?????? erase_len?? = len;

?????? erase_step? = spiflash_info->erasesize;

?

?????? while(len > 0) {

????????????? if(len < erase_step)

???????????????????? erase_step= len;

?

????????????? ret= spi_flash_erase(flash, offset, erase_step);

????????????? if(ret) {

???????????????????? printf("SPIflash %s failed\n", argv[0]);

???????????????????? return1;

????????????? }

?

????????????? len-= erase_step;

????????????? offset+= erase_step;

?

????????????? do{

???????????????????? unsignedlong long n = (unsigned long long)

??????????????????????????? (offset- erase_start) * 100;

???????????????????? intpercent;

?

???????????????????? do_div(n,erase_len);

???????????????????? percent= (int)n;

?

???????????????????? /*output progress message only at whole percent

???????????????????? ?* steps to reduce the number of messagesprinted

???????????????????? ?* on (slow) serial consoles

???????????????????? ?*/

???????????????????? if(percent != percent_complete) {

??????????????????????????? percent_complete= percent;

?

??????????????????????????? printf("\rErasingat 0x%lx -- %3d%% complete.",

????????????????????????????????????????? offset,percent);

???????????????????? }

????????????? }while (0);

?????? }

?????? puts("\n");

?

?????? return0;

?

usage:

?????? puts("Usage:sf erase offset len\n");

?????? return1;

}

?

當數據寫到Flash以后,需要重啟單板,所以在net.c里面調用uboot的重啟函數,do_reset(),這個函數在grub模式下執(zhí)行reset的時候就會被調用,現(xiàn)在把它添加到net.c里面,當上傳的文件寫到Flash之后就會自動重啟單板:

if ( !do_upgrade() );

{???

???? DBG("file=%s, func=%s,line=%d\n", __FILE__, __FUNCTION__, __LINE__);??????do_http_progress(WEBFAILSAFE_PROGRESS_UPGRADE_READY);

???? do_reset (NULL, 0, 0, NULL);

???? return 0;

}???

?

上面do_http_progress()這個函數只是打印文件加載成功或失敗的信息:

int do_http_progress(const int state) {

?

?????? /*toggle LED's here */

?????? switch(state) {

????????????? caseWEBFAILSAFE_PROGRESS_START:

???????????????????? printf("HTTPserver is ready!\n\n");

???????????????????? break;

?

????????????? caseWEBFAILSAFE_PROGRESS_TIMEOUT:

???????????????????? //printf("Waitingfor request...\n");

???????????????????? break;

?

????????????? caseWEBFAILSAFE_PROGRESS_UPLOAD_READY:

???????????????????? printf("HTTPupload is done! Upgrading...\n");

???????????????????? break;

?

????????????? caseWEBFAILSAFE_PROGRESS_UPGRADE_READY:

???????????????????? printf("HTTPugrade is done! Rebooting...\n\n");

???????????????????? break;

?

????????????? caseWEBFAILSAFE_PROGRESS_UPGRADE_FAILED:

???????????????????? printf("##Error: HTTP ugrade failed!\n\n");

???????????????????? //wait 1 sec

???????????????????? milisecdelay(1000);

?

???????????????????? break;

?????? }

?????? return0;

}

?

到這里在net.c里面要添加的函數基本上就完成了,當然還有個別的宏定義,全局變量等一些定義的話,在編譯的時候參考出錯信息來修改添加就可以了。例如我在文件開頭添加了如下信息:

/*--------------------------add byyangzheng---------------------*/

#include <common.h>

#include <spi_flash.h>

?

#include <asm/io.h>

#include <linux/mtd/mtd.h>

?

#ifndef CONFIG_SF_DEFAULT_SPEED

# define CONFIG_SF_DEFAULT_SPEED? 1000000

#endif

#ifndef CONFIG_SF_DEFAULT_MODE

# define CONFIG_SF_DEFAULT_MODE????????? SPI_MODE_3

#endif

?

#define WEBFAILSAFE_UPGRADE_TYPE_FIRMWARE?????????? 0

#define WEBFAILSAFE_PROGRESS_START????????????????? 0

#define WEBFAILSAFE_PROGRESS_TIMEOUT??????????????? 1

#define WEBFAILSAFE_PROGRESS_UPLOAD_READY?????????? 2

#define WEBFAILSAFE_PROGRESS_UPGRADE_FAILED???????? 4

#define WEBFAILSAFE_PROGRESS_UPGRADE_READY????????? 3

#define ET_DEBUG

?

#define milisecdelay(_x)???? udelay((_x) * 1000)

?

#if 0

#define DBG(x...)?printf(x)

#else

#define DBG(x...)??do { } while (0)

#endif

?

static int arptimer = 0;

int??webfailsafe_is_running = 0;

int??webfailsafe_ready_for_upgrade = 0;

int??webfailsafe_upgrade_type = WEBFAILSAFE_UPGRADE_TYPE_FIRMWARE;

unsigned char *webfailsafe_data_pointer = NULL;

extern int do_reset (cmd_tbl_t *cmdtp, int flag, intargc, char *argv[]);

static struct spi_flash *flash;

/*---------------------------------------------------------------*/

?

小結:uip 的移植大部分時間就是用在net.c里面,net.c里面需要修改的東西很多,尤其是字節(jié)序的問題也是讓我花了很長時間去轉換,總之net.c需要修改的大致就這么多,可能還有個別地方沒講到,但是那都是小問題,這篇文檔已經說明了移植uip協(xié)議的整體思路,那么沒有提及的個別小問題就靠自己分析了,這樣做移植才能有更多收獲。下面還有一點需要完善,我需要系統(tǒng)啟動時,在找不到內核的情況下,自動調用httpd服務程序,然后通過web來升級我的系統(tǒng),方法如下。

?

2.2.3 在uboot里面添加httpd命令

?????? 首先要知道,uboot里面的那些命令是在哪兒定義的,比如,printenv, tftp, help等。這些命令是在u-boot-2010.06/common的.c文件里面定義的。可以模仿其中的一個命令來添加我們的httpd命令,我的代碼添加如下:

[kernel@localhost common]$vim cmd_httpd.c

/*********************************************************************************

?*?????Copyright:? (C) 2014 YangZheng<yz2012ww@gmail.com>?

?*?????????????????All rights reserved.

?*

?*??????Filename:? cmd_httpd.c

?*???Description:? This file

?*????????????????

?*???????Version:? 1.0.0(10/09/2014~)

?*????????Author:? Yang Zheng<yz2012ww@gmail.com>

?*?????ChangeLog:? 1, Release initialversion on "10/09/2014 03:48:07 PM"

?*????????????????

?********************************************************************************/

?

#include<common.h>

#include<command.h>

#include <net.h>

?

extern intNetLoopHttpd(void);

int do_httpd(cmd_tbl_t *cmdtp, int flag, int argc, char *argv[])

{

??? printf("file=%s, func=%s,line=%d\n", __FILE__, __FUNCTION__, __LINE__);

??? return NetLoopHttpd();? //這里就會調用net.c里面的NetLoopHttpd()函數

}

U_BOOT_CMD(httpd, 1,1, do_httpd, "start www server for firmware recovery\n", NULL);??????????

?

?

然后再添加一個功能,那就是上面提到的,系統(tǒng)啟動找不到內核是就調用httpd服務程序。在u-boot-2010.06/common/cmd_bootm.c的bootm_start()函數里面添加NetLoopHttpd():

extern int NetLoopHttpd(void);

static int bootm_start(cmd_tbl_t *cmdtp, int flag,int argc, char *argv[])

{

??? void??????? *os_hdr;

??? int???? ret;

?

??? memset((void *)&images, 0, sizeof (images));

???images.verify = getenv_yesno ("verify");

?

???bootm_start_lmb();

?

??? /* getkernel image header, start address and length */

??? os_hdr =boot_get_kernel (cmdtp, flag, argc, argv,

???????????&images, &images.os.image_start, &images.os.image_len);

?? ?if (images.os.image_len == 0) {

??????? puts("ERROR: can't get kernel image!\n");?? //找不到內核時會打印這一句信息

??????? puts ("#############start httpdserver!###############\n");

??????? NetLoopHttpd();

??????? return1;

…………

?

2.2.4 展示結果

?????? 終于到了看結果的時候了,有點激動。。。。。

首先在單板上運行httpd服務,在windows的DOS下,使用命令ping命令:


圖1 ?運行httpd


圖2 ping單板

?

使用web升級系統(tǒng),在瀏覽器地址欄輸入單板ip,就會看到如下圖所示:



圖3 ?web升級界面

點擊選擇文件->升級就開始上傳文件,上傳完以后就會自動寫到Flash了,這里就不再演示,不然我又得再次升級。。。

?

3、總結

3.1 uip協(xié)議關鍵庫函數的功能及使用方法

?????? 應用程序必須作為C函數去實現(xiàn),uIP在任何一個事件發(fā)生時調用UIP_APPCALL()。表 1 列出了可能的事件和每個事件的對應測試函數,測試函數用于區(qū)別不同的事件。函數是作為C宏命令實現(xiàn)的,將會是零或非零值。注意,某些函數可以在互相連接時發(fā)生(即新數據可以在數據確應的同時到達)。

表 1? uIP應用事件和對應的測試參數

一個數據包到達,確定先前發(fā)送到數據

uip_acked()

應用程序的新數據包已到達

uip_newdata()

一個遠程主機連接到監(jiān)聽端口

uip_connected()

一個到達遠程主機的連接建立成功

uip_connected()

計時時間滿重發(fā)

uip_rexmit()

計時時間滿周期性輪詢

uip_poll()

遠程主機關閉連接

uip_closed()

遠程主機中斷連接

uip_aborted()

由于太多重傳,連接中斷

uip_timedout()

?

當應用程序調用時,uIP設置全局變量uip_conn去指向當前連接的uip_conn結構,這可以用于區(qū)別不同的服務。一個典型的應用是檢查uip_conn->lport (當地TCP端口號)去決定哪個服務連接應該提供。例如,如果值uip_conn->lport等于80,應用程序可以決定啟動一個HTTP服務;若值是23,則是啟動TELNET服務。

?

3.1.1 接收數據

如果uIP測試函數uip_newdata()的值為1,則遠程連接的主機有發(fā)送新數據,uip_appdata指針指向實際數據,數據的大小通過uIP函數uip_datalen()獲得。在數據不是被緩沖后,應用程序必須立刻啟動。

?

3.1.2 發(fā)送數據

應用程序通過使用uIP函數uip_send()發(fā)送數據。uip_send()函數采用兩個參數:一個指針指向發(fā)送數據和數據的長度。如果應用程序為了產生要發(fā)送的實際數據需要RAM空間,包緩存(通過uip_appdata指針指向)可以用于這方面。在一個時間里應用程序只能在連接中發(fā)送一塊數據,所以不可以在每個應用程序啟用中調用uip_send()超過一次,只有上一次調用的數據將會發(fā)出后才可以。注意,調用uip_send()后會改變某些全局變量,在應用函數返回前它不能被調用。

?

3.1.3 重發(fā)數據

若數據在網絡中丟失,則應用程序必須重新發(fā)數據。無論是數據收到還是沒有收到,uIP都保并通知應用程序什么時候察覺出數據丟失了。若測試函數uip_rexmit()為真,則應用程序要重持跟蹤,發(fā)上一次發(fā)出的數據。重發(fā)就好像原來那樣發(fā)送,也就是通過uip_send()發(fā)送。

?

3.1.4 關閉連接

應用程序通過調用uip_close()關閉當前連接,這會導致連接干凈地關閉。為了指出致命的錯誤,應用程序可以通過中止連接和調用uip_abort()函數完成這項工作。若連接已經被遠端關閉,則測試函數uip_closed()為真,應用程序接著可以做一些必要的清理工作。

?

?

3.1.5 報告出錯

有兩個致命的錯誤可以發(fā)生在連接中:連接由遠程主機中止和連接多次重發(fā)上一數據而被中止。uIP通過調用函數報告這些問題,應用程序使用兩個測試函數uip_aborted()和uip_timedout() 去測試這些錯誤情況。

?

?

3.1.6 輪詢

當連接空閑時,uIP周期性地輪詢應用程序,應用程序使用測試函數uip_poll()去檢查它是否被輪詢過。

?

3.1.7 監(jiān)聽端口

uIP維持一個監(jiān)聽TCP端口列表,通過uip_listen()函數,一個新的監(jiān)聽端口被打開。當一個連接請求在一個監(jiān)聽端口到達,uIP產生一個新的連接。若一個新連接產生,則應用程序被調用,測試函數uip_connected()為真。

?

3.1.8 打開連接

作為uIP的0.6版,在uIP里面通過使用uip_connect()函數打開一個新連接。這個函數打開一個新連接到指定的IP地址和端口,返回一個新連接的指針到uip_conn結構。若是沒有空余的連接槽,則函數返回空值。為了方便,函數uip_ipaddr()可以用于將IP地址打包進兩個單元16位數組里,通過uIP去代表IP地址。

接下來用兩個例子說明。第一個例子展示了怎樣打開一個連接去遠端TCP端口8080。若沒有足夠的TCP連接插槽去允許一個新連接打開,則uip_connect()函數返回NULL并通過uip_abort()中止當前連接。第二個例子展示怎樣打開一個新連接去指定的IP地址。

例1:打開一個連接去遠端TCP端口8080。

void connect_example1_app(void)

{

if(uip_connect(uip_conn->ripaddr,8080) == NULL)

?{

uip_abort();

}

}

例2:打開一個連接去當前連接的遠端的端口8080。

void connect_example2(void)

?{

u16_t? ipaddr[2];

uip_ipaddr(ipaddr,192,168,0,1);

uip_connect(ipaddr,8080);

}

?

3.1.8 數據流控制

uIP通過函數uip_stop()和函數uip_restart()提供對存取TCP數據流的控制途徑。假如一個應用程序下載數據到一個慢速設備,例如磁盤驅動器。當磁盤驅動器的作業(yè)隊列滿時,應用程序不會準備從服務器接收更多的數據,直到隊列排出空位。函數uip_stop()可以用于維護流控制和停止遠程主機發(fā)送數據。當應用程序準備好接收更多數據時,可用函數uip_restart()告知遠程終端再次發(fā)送數據。函數uip_stopped()可以用于檢查當前連接是否停止。

?

?

3.1.9 UIP函數總結

表2 包含了所有uIP提供的函數

表2? uIP 函數總結

系統(tǒng)接口

?

uip_init()

uip_input()

uip_periodic()

初始化uIP

處理輸入包

處理周期計時事件

應用程序接口

?

uip_listen()

uip_connect()

uip_send()

uip_datalen()

uip_close()

uip_abort()

uip_stop()

uip_stopped()

uip_restart()

開始監(jiān)聽端口

連接到遠程主機

在當前連接發(fā)送數據

輸入數據的大小

關閉當前連接

中止當前連接

停止當前連接

查找連接是否停止

重新啟動當前連接

測試函數

?

uip_newdata()

uip_acked()

uip_connected()

uip_closed()

uip_aborted()

uip_timeout()

uip_rexmit

uip_poll()

遠程主機已經發(fā)出數據

確定發(fā)出的數據

當前連接剛連上

當前連接剛關閉

當前連接剛中止

當前連接剛超時

數據重發(fā)

應用程序循環(huán)運行

其它

?

uip_mss()

uip_ipaddr()

htons(),ntohs()

獲得當前連接的最大段的大小

將IP地址結構打包

在主機和網絡之間轉換字節(jié)次序

?

上面的這些都是引用網上別人博客里面的內容,總結的也很全面。

?

?

?

?

?

?

?

總結

以上是生活随笔為你收集整理的UIP协议栈移植到u-boot详解的全部內容,希望文章能夠幫你解決所遇到的問題。

如果覺得生活随笔網站內容還不錯,歡迎將生活随笔推薦給好友。

欧美小视频在线观看 | 91最新地址永久入口 | www.xxx.性狂虐| 日本免费一二三区 | 中文字幕在线观看资源 | 亚洲精品美女久久17c | 久草在线在线视频 | 欧美最猛性xxxxx亚洲精品 | av短片在线 | 九九久 | 韩日三级在线 | 久久综合精品国产一区二区三区 | 日本在线观看一区二区三区 | www.国产高清| 国产精品一区二区免费视频 | 国产一区在线免费观看视频 | 激情五月五月婷婷 | 国产特级毛片aaaaaaa高清 | 久久国产福利 | 狠狠色丁香婷婷综合最新地址 | 黄色成人在线观看 | 黄色影院在线免费观看 | 婷婷色中文字幕 | 日韩久久片 | 特级黄色片免费看 | 欧美韩国在线 | 国产二区av | 97精品一区 | 91成人精品在线 | 久久久国产一区 | 欧美午夜精品久久久久 | 成av人电影 | 久久精品视频一 | www在线观看国产 | 在线观看资源 | 99久久99精品| 日本久久影视 | 91亚州| 成人在线播放网站 | www.com操| 激情五月在线视频 | 久保带人 | 国产中文字幕一区 | 国产字幕在线看 | 狠狠gao| 精品欧美在线视频 | 成人中文字幕+乱码+中文字幕 | 日韩亚洲欧美中文字幕 | 草在线 | 一级一片免费观看 | 久久精品欧美日韩精品 | 97超碰在线人人 | 亚洲精品在线视频播放 | 久草视频精品 | 成人午夜影院在线观看 | 97免费在线观看视频 | 成人免费观看在线视频 | 最新不卡av| 夜色资源网 | 五月婷婷综合网 | 麻豆传媒在线免费看 | a级国产乱理论片在线观看 特级毛片在线观看 | 少妇bbw揉bbb欧美 | 久久综合色综合88 | 精品一区二区亚洲 | 久久香蕉国产精品麻豆粉嫩av | 国产又黄又猛又粗 | 久久久久免费 | 国产一性一爱一乱一交 | 久久免费一级片 | 日韩aⅴ视频 | 亚洲精品免费观看视频 | 免费观看xxxx9999片 | 伊人久久国产精品 | 中文字幕第一 | 日韩视频在线不卡 | 日韩专区视频 | 成人久久18免费网站 | 久久久久久久久久久高潮一区二区 | 啪啪资源 | 欧美亚洲精品在线观看 | 国产 在线观看 | 日韩欧美国产精品 | 911精品美国片911久久久 | 欧美在线91| 久久人人精品 | 中文字幕免费高清在线观看 | 久久99久久99精品免观看粉嫩 | 天天干天天射天天插 | 日韩mv欧美mv国产精品 | 在线免费观看羞羞视频 | 五月婷婷在线视频观看 | 国产日韩欧美视频 | 日韩毛片久久久 | 狠色狠色综合久久 | 黄色av免费| av丁香 | 日日夜夜天天综合 | 国产精品福利在线播放 | 少妇性xxx| 欧美日韩在线观看一区二区三区 | 日韩高清av| 五月天综合色激情 | 成人免费视频网站在线观看 | 国产精品一区二区三区视频免费 | 久久综合五月 | 亚州国产精品 | 国产清纯在线 | 五月婷亚洲 | 在线导航福利 | 欧美日韩亚洲精品在线 | 干干日日| 超碰官网 | 视频一区亚洲 | 国产福利小视频在线 | 在线免费黄色片 | 国产在线精 | 久久国产精品99久久久久久进口 | 最近中文字幕视频完整版 | 综合色影院 | 亚洲精品在线观看视频 | 国产无遮挡猛进猛出免费软件 | 欧美最猛性xxxxx(亚洲精品) | 国产高清av免费在线观看 | 欧美黄在线 | 欧美激情第八页 | 在线视频18在线视频4k | 一区二区三区四区五区在线视频 | 黄色国产高清 | 国产破处在线播放 | 日本天天色 | 亚洲国产精品第一区二区 | 久久精品91久久久久久再现 | 久青草视频在线观看 | 久久精品一区二区三 | 右手影院亚洲欧美 | 性色av免费看 | 国产中文在线视频 | 午夜资源站 | 亚洲综合色视频 | 日本久久影视 | 日韩大片免费在线观看 | 色综合天天色综合 | 香蕉在线视频观看 | 91视频啊啊啊 | 最新av在线网址 | 色5月婷婷 | 天天操人| 国产精品11 | 日韩在线大片 | 中文字幕在线观看播放 | 成人国产精品免费观看 | 久热这里有精品 | 亚洲区精品 | 91成人精品一区在线播放69 | 国产一区二区在线免费视频 | 国产婷婷一区二区 | 国产夫妻av在线 | 久久久视屏 | 中文字幕在线观看免费 | 在线观看视频你懂 | 成人动态视频 | 免费三级在线 | 免费av 在线 | 一区二区三区在线免费 | 久久y| 成人中文字幕在线 | 日韩av伦理片 | 丁香婷婷综合激情五月色 | 特及黄色片 | 又黄又爽又无遮挡的视频 | 91综合视频在线观看 | 人人草在线视频 | 国产精品美女久久 | 国产福利精品一区二区 | 精品国产成人av | 精品国产一区二区三区蜜臀 | 草久视频在线观看 | 国产成人在线播放 | 成人a免费看 | 国产高清在线视频 | 91看片看淫黄大片 | 国产无吗一区二区三区在线欢 | 在线观看亚洲 | 国产电影一区二区三区四区 | 综合色在线观看 | 亚洲精品久久久蜜桃直播 | 久久国产精品久久久 | 五月亚洲| 精品久久久久久久久久久院品网 | 亚洲精品午夜国产va久久成人 | 日日爽天天 | 三级黄色在线 | 中文字幕首页 | 黄色国产在线观看 | 高清精品久久 | 最近中文字幕视频完整版 | 久久视频在线 | 天天躁日日躁狠狠 | 亚洲第一区精品 | 91麻豆精品国产91久久久无限制版 | 日韩中文字幕第一页 | av电影免费在线播放 | 国内丰满少妇猛烈精品播放 | 久久专区 | 免费美女av| 国产在线一卡 | 久草在线资源观看 | 玖玖玖精品 | 综合色爱 | 日韩欧美在线免费观看 | 伊人资源站 | 国产福利网站 | 在线亚洲播放 | 久久精品视频在线观看 | 精品免费久久久久 | 国产成人一区二区啪在线观看 | 国产又黄又爽又猛视频日本 | 久久精品一区二区三区四区 | 国产黄色一级大片 | 国产精品成人一区二区三区吃奶 | 超碰官网 | 亚洲精品高清视频 | 亚洲永久国产精品 | 色婷婷狠狠五月综合天色拍 | 毛片一区二区 | 国产精品久久久久久久久毛片 | 亚洲天天草| 亚洲精品久久久蜜桃直播 | 国产在线 一区二区三区 | 天天干.com| 91在线观看黄 | 视频精品一区二区三区 | 精品亚洲va在线va天堂资源站 | 黄影院 | 亚洲欧美国产精品18p | 日韩av一区在线观看 | 波多野结衣最新 | 午夜视频黄 | 日韩欧美网址 | 国产又粗又猛又黄 | 成人91免费视频 | 九九热在线精品视频 | 国产在线观看91 | 深爱婷婷 | 91av久久| 成人午夜免费剧场 | 精品一区精品二区 | 精品久久久久免费极品大片 | 日韩久久在线 | 综合伊人久久 | 日韩毛片在线免费观看 | 97超碰影视 | 日韩1页| 探花视频免费在线观看 | 夜添久久精品亚洲国产精品 | 狠狠的干 | 亚洲精品理论片 | 福利一区在线 | 在线看欧美 | 二区三区中文字幕 | 日韩二区在线播放 | 一区二区三区高清在线观看 | 国产亚洲精品中文字幕 | av解说在线 | 中文字幕观看在线 | 国产精品永久免费在线 | 成人在线你懂得 | 欧美日韩二区在线 | avav99| 国内精品久久久久久久久久清纯 | 91精品国产综合久久婷婷香蕉 | 国产精品初高中精品久久 | 99精品免费久久久久久久久日本 | 国产伦精品一区二区三区四区视频 | 人人添人人澡人人澡人人人爽 | 国产欧美精品一区二区三区 | 中文字幕一区二区三 | 色婷婷激情电影 | 免费视频黄 | 亚洲精品99| 成人av在线观 | 欧美激情操 | 色婷婷www | 免费观看十分钟 | 麻豆视传媒官网免费观看 | 91九色免费视频 | 三级av在线免费观看 | 欧美性久久久 | 亚洲综合导航 | 国产精品久久久久久欧美 | 国产精彩在线视频 | 蜜桃av人人夜夜澡人人爽 | 久久久久久99精品 | 久久五月天婷婷 | 黄污网| 美女天天操 | 香蕉在线观看视频 | 国产91免费在线观看 | 成人在线一区二区三区 | av 一区二区三区 | 亚洲国产成人久久综合 | 国产精品九九九 | 天天天天天天天操 | 国产香蕉在线 | 1024在线看片 | 一区二区三区福利 | 久99久精品视频免费观看 | 黄污在线看 | 中文字幕日韩伦理 | 久久激情久久 | 一区二区三区影院 | 婷婷综合亚洲 | 欧美a级片免费看 | 国产精品国产三级国产专区53 | 色综合久久88色综合天天免费 | 欧美日韩18 | 一区二区av| 99 精品 在线 | 欧美日韩国产精品一区二区三区 | 黄色h在线观看 | 探花系列在线 | 天天干天天干天天干天天干天天干天天干 | 色婷婷激情四射 | 久久久精品电影 | 成 人 免费 黄 色 视频 | 久久久电影网站 | 国产伦精品一区二区三区在线 | 日本中文字幕在线电影 | 欧美日韩1区2区 | 1000部18岁以下禁看视频 | 日韩欧美在线一区二区 | 久久伦理电影 | 久久一区国产 | 毛片1000部免费看 | 一区二区三区四区五区在线 | 国产精品久久久久影院日本 | 在线观看日韩一区 | 在线亚洲免费视频 | 丁香综合av| 免费的黄色的网站 | 日韩免费av片 | 中文字幕成人在线 | 国产亚洲免费观看 | 成人av在线直播 | 超碰免费97 | 精品视频97 | 91精品黄色 | www免费 | 亚州av成人| 美女福利视频 | 国内偷拍精品视频 | 伊人婷婷在线 | av观看在线观看 | 在线香蕉视频 | 国产拍揄自揄精品视频麻豆 | 欧美在线视频一区二区 | 日本中文字幕在线 | 少妇精69xxtheporn | 婷婷色影院 | 久久久久久久亚洲精品 | 国产裸体永久免费视频网站 | 精品自拍网 | 久久精品视频在线免费观看 | 91麻豆精品91久久久久同性 | 国产一区在线免费 | 一区二区成人国产精品 | 国产群p | 欧洲精品码一区二区三区免费看 | av片中文 | 国产99久久99热这里精品5 | 在线视频 成人 | 亚洲最大成人免费网站 | 97国产一区 | 欧美性另类| 国产在线精品一区 | 国产成人精品一区在线 | 国产在线观看免费观看 | 日韩高清毛片 | 欧美一二三专区 | 天天操天天草 | 97小视频 | 欧美二区视频 | 丁香婷婷综合网 | www.888.av| 999超碰 | 久久视屏网 | 亚洲一区尤物 | 久草爱| 91国内在线 | 视频一区在线播放 | 中文字幕在线观看资源 | 国产无吗一区二区三区在线欢 | 在线观看视频h | 久久99亚洲精品 | 91av手机在线 | 在线播放日韩av | 久久成人免费 | 国产亚洲综合精品 | 99热这里有 | 亚洲午夜激情网 | 亚洲人久久久 | 亚洲精品毛片一级91精品 | 手机在线中文字幕 | 亚洲美女在线一区 | 成人免费看黄 | 亚洲黄色在线播放 | 天海冀一区二区三区 | 五月天色综合 | 亚洲精品tv | 精品国产色 | 国产精品系列在线 | 免费观看成年人视频 | 99re6热在线精品视频 | 九九热只有这里有精品 | 成人久久网| 欧美在线观看视频免费 | 国产123区在线观看 国产精品麻豆91 | 天天看天天干天天操 | 狠狠躁夜夜躁人人爽超碰97香蕉 | 日日夜夜操操操操 | 91毛片在线观看 | 久久er99热精品一区二区 | 久久国产精品一区二区三区 | 午夜视频在线瓜伦 | 久久午夜剧场 | 日韩精品最新在线观看 | 麻豆精品视频在线观看免费 | 国产精品激情在线观看 | 久草在线资源观看 | 波多野结衣电影一区二区三区 | 国产黄免费在线观看 | 欧美日韩高清一区 | 最新中文在线视频 | 国产一区二区电影在线观看 | 国产视频资源 | 日韩欧美综合精品 | 国产美女视频免费 | 狠狠插狠狠操 | 国产成人精品区 | 天天爽天天做 | 亚洲成人频道 | 精品国产一区二区三区四区在线观看 | 欧美成人区 | 成人黄色小视频 | 97精品一区二区三区 | 玖玖爱在线观看 | 波多野结衣在线中文字幕 | 国产在线播放观看 | 免费看污网站 | 亚洲欧美视频网站 | 国产精品久久久久久一区二区三区 | 亚洲精品乱码久久久久久蜜桃不爽 | 国产系列 在线观看 | 国产小视频你懂的 | 日韩在线观看第一页 | 国产精品美女视频 | 国产午夜麻豆影院在线观看 | 日韩激情小视频 | 97自拍超碰 | 欧美做受高潮 | 久久久精品国产一区二区 | 国产欧美精品在线观看 | 日本中文字幕在线 | 很黄很色很污的网站 | 国产在线播放一区二区三区 | 亚洲欧美在线综合 | 日韩欧美国产精品 | 日韩成人在线免费观看 | 91精品一 | 亚洲国产精品一区二区久久hs | 国内视频在线 | 精品一二区 | 在线观看精品黄av片免费 | 在线观看av中文字幕 | 欧美最爽乱淫视频播放 | 91伊人久久大香线蕉蜜芽人口 | 欧美日韩在线看 | 美女精品国产 | 欧美成人手机版 | 精品欧美一区二区精品久久 | 国产高清久久久 | 在线视频精品 | 青草视频免费观看 | 国产精品久久久久久一二三四五 | 亚洲 欧美 成人 | 国产欧美久久久精品影院 | 日韩激情影院 | 欧美性受极品xxxx喷水 | 国产五十路毛片 | 91一区在线观看 | 在线观看免费av网 | 久章草在线 | aaa亚洲精品一二三区 | 国产日产精品一区二区三区四区的观看方式 | 亚洲码国产日韩欧美高潮在线播放 | 欧美日韩国产亚洲乱码字幕 | 亚洲精品高清在线观看 | 日韩在线观看精品 | 五月婷婷免费 | 色资源二区在线视频 | se婷婷 | 成年人在线电影 | 成人欧美一区二区三区在线观看 | 中文字幕首页 | 97碰在线 | 国产精品视频免费看 | 国产精品福利午夜在线观看 | 欧美精品三级在线观看 | 日韩免费av在线 | 国产精品久久久久婷婷 | 久久夜视频 | 最新中文字幕在线观看视频 | 久久精品三 | 久久国产露脸精品国产 | 黄色小说视频网站 | 久久久精品国产免费观看一区二区 | 超碰在线最新地址 | 天堂黄色片 | 黄色免费在线看 | 欧洲亚洲精品 | 国产精品视频内 | 天天干com | 欧美性猛片 | 99免在线观看免费视频高清 | 国产精品18久久久久久久 | 午夜久久| 中文不卡视频 | 在线草| 久草在线久草在线2 | 国产精品免费大片视频 | 欧美韩日视频 | 深爱激情站 | 国产精品免费不 | 精品成人在线 | 午夜少妇 | 在线91播放| www91在线观看 | 亚洲,国产成人av | 国产精品久久久久久久av大片 | 天天人人 | 一级一片免费视频 | 久久最新 | 亚洲 中文 在线 精品 | 精品视频免费播放 | 97人人网 | 国产一级h| 日日爽| 亚洲精品国偷拍自产在线观看蜜桃 | 亚洲国产午夜精品 | 婷婷视频在线播放 | 综合激情伊人 | av天天草| 久久草草热国产精品直播 | 免费视频在线观看网站 | 一区在线免费观看 | 婷婷久久亚洲 | 91在线网址| 亚洲精品三级 | 97av在线视频免费播放 | 亚洲欧洲日韩在线观看 | 中文字幕在线观看第三页 | 中文字幕一区在线 | 久久国产成人午夜av影院宅 | 免费看短 | 日韩在线观看av | 在线99| 国产99免费视频 | 狠狠狠狠狠干 | 日韩高清毛片 | 色九九视频 | 五月丁色| 欧美韩国日本在线观看 | 婷婷激情综合网 | 精品久久综合 | 国产黄色一级大片 | 欧美做受69| 日韩资源在线 | 在线观看视频色 | 亚洲理论在线观看电影 | 久草线| 911免费视频 | 国产一级在线免费观看 | 国产精品欧美日韩在线观看 | 99久久日韩精品免费热麻豆美女 | 91av片 | 午夜狠狠操 | 亚洲精品一区二区三区新线路 | 99人久久精品视频最新地址 | 国产三级视频在线 | 美女网站色在线观看 | 久久免费视频网站 | 亚洲精品视频中文字幕 | 国产美腿白丝袜足在线av | 黄视频色网站 | 国产69精品久久久久久久久久 | 国产美女永久免费 | 日本特黄特色aaa大片免费 | 久久免费片 | 国产成人精品免费在线观看 | 久久曰视频 | 久久免费福利视频 | 99精品在线视频观看 | 国产精品久久久久久久久蜜臀 | 综合久久久 | 国产亚洲精品成人av久久ww | 国产成人精品久久二区二区 | 日日操天天射 | 久久久影院一区二区三区 | 亚洲成人一二三 | 亚洲精品午夜久久久久久久久久久 | 91av99| 黄色国产大片 | 91av在线播放视频 | 很黄很色很污的网站 | 国产成人av电影在线 | 开心色插 | 一级黄色大片在线观看 | 精品国精品自拍自在线 | 日韩在线首页 | 国产一级久久久 | 九九热只有精品 | 日韩免费看片 | 99综合电影在线视频 | 亚洲精品视频免费在线观看 | 国产成人a亚洲精品 | 69国产精品视频 | 波多野结衣在线观看一区二区三区 | 欧美a级在线 | 午夜久久电影网 | 亚洲欧洲视频 | 亚洲欧美国产精品18p | 国产成人av综合色 | 五月婷婷色综合 | 97成人在线 | 国产第一页在线播放 | 国产成人精品一二三区 | 99热精品久久 | 国产精品涩涩屋www在线观看 | 日韩手机在线观看 | 久久久视频在线 | 永久免费看av| 国产群p| 成人av.com| 手机av观看 | 亚洲视频电影在线 | 一区二区三区高清在线观看 | 久久99精品久久久久婷婷 | 国产主播大尺度精品福利免费 | 韩国精品福利一区二区三区 | www.精选视频.com | 51久久成人国产精品麻豆 | 天天射综合网站 | 欧美福利网站 | 日韩精品资源 | 国产成人精品免费在线观看 | 久久久高清免费视频 | 日韩高清在线一区二区三区 | 亚洲每日更新 | 国产中文字幕网 | 69国产精品视频免费观看 | 欧美日本不卡视频 | 亚洲最大成人网4388xx | 二区三区精品 | 国产精品永久久久久久久久久 | 久久综合九色 | 99精品久久久 | 美女视频网 | 免费人成在线观看网站 | 免费看成人片 | 免费av 在线 | 成人网在线免费视频 | 狠狠躁夜夜a产精品视频 | 久久99精品久久久久久三级 | 手机av在线免费观看 | 婷婷丁香色 | 免费看黄网站在线 | 欧美日韩在线播放一区 | 色99在线| 中文在线免费视频 | 免费看国产一级片 | 午夜久久视频 | 国产香蕉97碰碰久久人人 | 国产精品免费久久 | 久久国产午夜精品理论片最新版本 | 天天综合网天天综合色 | 久久热亚洲 | 国产无套视频 | a视频在线播放 | 国产在线观看,日本 | 国产精久久久久久妇女av | 中文字幕视频免费观看 | 欧美日产在线观看 | 男女靠逼app | 黄色高清视频在线观看 | 亚洲深夜影院 | 欧美一区二区三区免费观看 | 日韩一区二区三区高清免费看看 | 久久精品一区 | 99re亚洲国产精品 | 夜夜夜夜猛噜噜噜噜噜初音未来 | 色多多视频在线 | 美女激情影院 | 夜夜躁日日躁狠狠躁 | 四虎影视成人永久免费观看视频 | 特黄特黄的视频 | 色婷婷天天干 | 97超在线视频 | 欧美极品少妇xxxxⅹ欧美极品少妇xxxx亚洲精品 | 99久久久国产精品免费99 | 国产又粗又硬又爽的视频 | 国产专区一 | 手机在线欧美 | 成人午夜网 | 欧美aa在线 | 亚洲国产字幕 | 欧美一区二区在线刺激视频 | 欧美亚洲成人免费 | 日韩精品一区二区三区水蜜桃 | 丁香婷婷激情啪啪 | 综合色狠狠 | 99久久99久久精品免费 | 国偷自产视频一区二区久 | 亚洲 成人 欧美 | 国产精品久久久久久久久久不蜜月 | 在线免费色视频 | 国内精品在线看 | 免费看黄在线观看 | 国产成人av免费在线观看 | 精品久久精品久久 | 亚洲成人av一区 | 国产精品黄色影片导航在线观看 | 2023亚洲精品国偷拍自产在线 | 91亚洲夫妻| 日韩最新中文字幕 | 99精品免费久久久久久久久 | 麻豆一区在线观看 | 日本中文字幕电影在线免费观看 | 黄色av大片 | 狠狠色狠狠色 | 毛片网站在线观看 | 天天综合网 天天 | 国产91精品久久久久久 | 久草视频中文 | 亚洲作爱视频 | 探花国产在线 | 中文字幕av免费在线观看 | 日本精品一区二区三区在线播放视频 | 国产小视频在线免费观看视频 | 国产精品久久久久久爽爽爽 | 久久丁香 | 国产精品乱码久久久 | 欧美怡红院 | 久av电影| 国产99久久久精品视频 | 国产精品1区 | 91精品国产99久久久久久久 | 国产精品视频免费观看 | 黄色毛片一级片 | 最新久久免费视频 | 四虎在线免费观看 | 日韩激情在线 | 国产91精品久久久久久 | 久久看看 | av在线永久免费观看 | 99在线播放 | 在线观看视频一区二区三区 | 18pao国产成视频永久免费 | 久久精品99精品国产香蕉 | 成人avav| 欧美嫩草影院 | 99免费精品视频 | 久久久久免费精品国产小说色大师 | 欧美性极品xxxx做受 | www.五月天婷婷| 亚洲传媒在线 | 色吊丝av中文字幕 | 免费网站观看www在线观看 | 午夜久久久久久久久 | 精品国内自产拍在线观看视频 | 97看片 | 色综合久久中文字幕综合网 | 亚洲国产中文字幕 | 日韩欧美视频 | 狠狠干夜夜操天天爽 | 精品av网站| 欧美性生活一级片 | 91福利视频免费 | 日韩色综合网 | 免费一级片观看 | 激情综合网五月 | 久久久综合精品 | 色综合久久中文字幕综合网 | 色综合天 | 91大神在线看 | 亚洲爱爱视频 | 天天做日日做天天爽视频免费 | 亚洲色综合 | 欧美激情综合色综合啪啪五月 | 日韩欧美在线高清 | 国产精品区在线观看 | 精品一区av | 日韩二区精品 | 免费福利在线播放 | 99九九热只有国产精品 | 欧美日韩亚洲在线 | 久久a v电影| 一个色综合网站 | 国产精品一区二区果冻传媒 | 欧美一级视频一区 | 日韩三级在线观看 | 亚洲精品tv | 探花视频免费在线观看 | 免费性网站 | 五月婷在线| 亚洲污视频 | 国产91全国探花系列在线播放 | av天天干| 日本三级香港三级人妇99 | 成人性生交大片免费看中文网站 | 一区二区三区日韩精品 | 精品xxx| 精品一区二区电影 | 亚洲国产精品激情在线观看 | 久久亚洲热 | 日韩精品免费一区二区三区 | 国内久久看 | 中文字幕资源网在线观看 | 日韩视频在线播放 | 久久久久久国产一区二区三区 | 日本黄色a级大片 | 超碰在线9| 欧美性直播 | 日韩在线视 | 久久精品综合网 | 日韩一二三区不卡 | 欧美成人亚洲 | 91麻豆免费看 | 欧美孕妇与黑人孕交 | 国产精久久 | 日韩天天干 | 精品久久久久久综合日本 | 精品国产1区 | 久久国产视屏 | 99精品国产视频 | 极品久久久久久久 | 超碰成人免费电影 | 97国产在线观看 | 中文字幕在线观看三区 | 日韩欧美精品在线观看 | 精品一区二区三区久久 | 日韩在线视频一区二区三区 | 99在线精品免费视频九九视 | 精品在线不卡 | 国产流白浆高潮在线观看 | 97超碰精品 | 中文字幕精品一区二区三区电影 | 精品久久久久久久久亚洲 | 日韩一级片网址 | 99久久婷婷国产一区二区三区 | www.精选视频.com | 午夜免费福利视频 | 99视频在线精品国自产拍免费观看 | 国产精品扒开做爽爽的视频 | 日韩久久视频 | 亚洲高清视频在线 | 日韩伦理片一区二区三区 | 中文乱幕日产无线码1区 | 91中文视频 | 在线日本看片免费人成视久网 | 天堂网一区二区三区 | 国产精品一区二区三区在线播放 | 在线免费观看的av网站 | www.色午夜,com | 91豆花在线观看 | 国产精品不卡在线 | 日韩电影在线一区二区 | 欧美精品亚州精品 | 精品久久视频 | 91天天操| 999亚洲国产996395 | 91日韩在线| 国产涩图| av免费网站| 亚洲成人黄色av | 玖玖爱国产在线 | 人人射人人澡 | 亚洲综合小说 | 成人在线观看网址 | 精品免费国产一区二区三区四区 | 亚洲国产精品一区二区久久,亚洲午夜 | 国产一级免费在线观看 | 在线观看视频日韩 | 久久久久欧美精品 | 久久人人爽人人片 | 狠狠色综合网站久久久久久久 | 婷婷射五月 | 日韩在线观看网址 | 国产精品尤物视频 | 成人啪啪18免费游戏链接 | 免费视频a| 精品久久久久一区二区国产 | 欧美日韩中文视频 | 婷婷国产v亚洲v欧美久久 | 国产在线免费 | 国产麻豆精品在线观看 | 成人全视频免费观看在线看 | 黄色免费网战 | 中文字幕精品三区 | 国产一级免费在线 | 永久免费精品视频网站 | 国产精品视频app | 精品在线小视频 | 久久男人中文字幕资源站 | 中文永久字幕 | 日本三级在线观看中文字 | 在线免费视频你懂的 | 久久人人爽人人爽人人片 | 丝袜+亚洲+另类+欧美+变态 | 国产精品人人做人人爽人人添 | 中文字幕在线看视频国产 | 久久久久久久久毛片精品 | 亚州精品天堂中文字幕 | 在线免费观看一区二区三区 | 天堂av免费观看 | 成 人 免费 黄 色 视频 | 国产成人av电影在线 | 九九热在线视频免费观看 | 久久中国精品 | 国产黄色成人av | www久久com | 人人澡人人爽欧一区 | 亚洲欧美精品一区二区 | 亚洲年轻女教师毛茸茸 | 国产在线久草 | 91麻豆精品91久久久久同性 | 狠狠狠干狠狠 | 日本性xxx| 欧美色操 | 日本aaaa级毛片在线看 | 国产精品 中文在线 | 精品久久久久久久久中文字幕 | 一区二区三区免费在线播放 | 中文字幕专区高清在线观看 | 最近中文字幕免费观看 | 亚洲国产色一区 | 久久久久黄 | 免费网站看v片在线a | 91成人天堂久久成人 | 精品一区二区三区久久 | 国产精品久久久久久久久久了 | 狠狠干网 | 久久国产精品久久精品国产演员表 | a级国产毛片 | 天天射天天操天天 | 视频一区在线免费观看 | 成人午夜电影免费在线观看 | 特黄特色特刺激视频免费播放 | 久久久久高清毛片一级 | 国产精品第10页 | 97视频网站| 在线看的毛片 | 久久精品国产精品亚洲精品 | 狠狠操狠狠干天天操 | 成人免费视频网站在线观看 | 色av男人的天堂免费在线 | 欧美精品在线观看免费 | 色狠狠操 | 在线免费黄色 | 婷婷色中文字幕 | 国产成人99av超碰超爽 | 亚洲欧美一区二区三区孕妇写真 | 国产精品久久久久久久久久妇女 | 亚洲精品国产精品国自产 | 国产精品白浆视频 | 久久综合一本 | 波多在线视频 | 在线观看免费高清视频大全追剧 | 黄a在线观看 | 日韩在线观看不卡 | 麻豆传媒视频在线播放 | 天天操天天爽天天干 | 97视频在线观看视频免费视频 | 亚洲另类视频在线 | 天天操天天干天天操天天干 | 久青草电影| 久草视频资源 | 激情五月六月婷婷 | 日本视频久久久 | 日韩r级电影在线观看 | 黄色软件视频大全免费下载 | 午夜视频色| 国产视频91在线 | 中文字幕精品视频 | 欧美精品资源 | 天天久久综合 | 丁香婷婷深情五月亚洲 | 日韩二级毛片 |