如何使用W7100A實現Telnet服務器(三)

在上兩篇博文(如何使用W7100A實現Telnet服務器(一)如何使用W7100A實現Telnet服務器(二))里我們簡單地介紹一下Telnet,第三章將會演示Telnet函數的具體功能,第四章將涉及到代碼的分析。這篇文檔中所有的示例代碼都是基於Keil uVision3。

第一篇在這裡:http://blog.iwiznet.cn/?p=1193

第二篇在這裡:http://blog.iwiznet.cn/?p=1196

下面我們介紹第三部分(主要內容是第四章的餘下的代碼分析部分):

4.3 tel_input()函數

tel_input()函數用來處理Telnet終端機內的輸入命令。具體每一個命令以及處理方法請參考表3.1。

void tel_input(SOCKET s)

{

uint8 xdata c;

while(1){

if((getSn_RX_RSR(s)) == 0)  break;     /*如果沒有接收到的數據,斷開*/

if(recv(s, &c, 1) == 0) break;     /*如果接收到的數據為0,斷開*/

if(user_state == LOGOUT)  break;     /*如果用戶的聲明是LOGOUT, 斷開*/

if(c != IAC){         /*如果接收到的數據不是控制字符*/

data_buf[buf_index++] = c;    /*保存接收到的數據到data_buf*/

putchar(c);

if(user_state == LOGOUT)  break;

if(user_state != PASSWORD){

       sprintf(buf, “%c”, c);

       send(s, buf, strlen(buf));

}

if(c == ‘\n’){     /*如果接收到一個\n’ ASCII 代碼*/

       if(buf_index > 1){

         if(data_buf[buf_index-2] == ‘\r’)  data_buf[buf_index-2] = ‘\0’;

         else  data_buf[buf_index-1] = ‘\0’;

         proc_command(s);      /* 處理接收到的數據*/

         if(user_state == LOGIN) {

           sprintf(buf, “W7100>”);

           send(s, buf, strlen(buf));

         }

       }

       else{

         sprintf(buf, “W7100>”);

         send(s, buf, strlen(buf));

       }

       buf_index = 0;

}

continue;

}

if(recv(s, &c, 1) == 0)  break;

switch(c){            /*如果接收到一個IAC字符*/

case WILL:

if(recv(s, &c, 1) == 0)  break;

willopt(s, c);      /*調用willopt()處理WILL命令*/

break;

case WONT:

if(recv(s, &c, 1) == 0)  break;

wontopt(s, c);      /*調用wontopt()來處理WONT命令*/

break;

case DO:

if(recv(s, &c, 1) == 0)  break;

doopt(s, c);      /*調用doopt()處理DO命令*/

break;

case DONT:

if(recv(s, &c, 1) == 0)  break;

dontopt(c);      /*調用dontopt()處理DON’T命令*/

break;

case IAC:

break;

}

break;

}

return;

}

程序4.3parseMSG()函數

 

4.4 proc_command()函數

proc_command()函數處理在tel_input()函數中輸入的命令。它定義了HELP、GET LED、LED0 ON/OFF和LED2 ON/OFF命令。對於未定義的命令它顯示“BAD COMMAND”。

void proc_command(SOCKET s)

{

uint8 i;

char **cmdp, *cp;

char *help = {“HELP: Show all available commands\r\n\GET LED: Show all LED status\                  \r\nLED0 ON/OFF: Turn ON/OFF the LED0\r\nLED1 ON/OFF: Turn ON/OFF the LED1\

         \r\nLED2 ON/OFF: Turn ON/OFF the LED2\r\nEXIT: Exit from W7100 Telnet server\r\n”};

       /*Translate the first word to lower case*/

for(cp = data_buf; *cp !=’\0′; cp++){

*cp = tolower(*cp);     /* 將大寫字母翻譯成小寫字母*/

}

if(user_state == USERNAME){

strcpy(user_name, data_buf);

sprintf(buf, “Please insert your PW: “);

send(s, buf, strlen(buf));

user_state = PASSWORD;

return;

}

else if(user_state == PASSWORD){

strcpy(user_password, data_buf);

sprintf(buf, “\r\nSuccessfully connected!!\r\nImplemented Command: \

HELP, GET LED, LED0 ON/OFF, LED1 ON/OFF, LED2 ON/OFF, EXIT\r\n”);

send(s, buf, strlen(buf));

user_state = LOGIN;

return;

}     /*查找表中的輸入命令; 如果不表內,返回語法錯誤*/

for(cmdp = commands; *cmdp != NULL; cmdp++){

if(strncmp(*cmdp, data_buf, strlen(*cmdp)) == 0)     break;

}

if(*cmdp == NULL){

printf(“NULL command\r\n”);

sprintf(buf, “BAD command\r\n”);

send(s, buf, strlen(buf));

return;

}

switch(cmdp – commands){

case HELP_CMD:         /* 處理HELP命令*/

printf(“HELP_CMD\r\n”);

sprintf(buf, help);

send(s, buf, strlen(buf));

break;

case GET_LED_CMD:        /*處理GET LED 命令*/

printf(“GET_LED_CMD\r\n”);

for(i = 0 ; i < 3 ; i++){

       sprintf(buf, “LED%bd is %s\r\n”, i, ((P0 >> (i+3)) & 0x01) ? “OFF” : “ON”);

       send(s, buf, strlen(buf));

}

break;

case LED0_ON_CMD:        /*處理LED0 ON命令*/

printf(“LED0_ON_CMD\r\n”);

sprintf(buf, “Turn ON the LED0\r\n”);

send(s, buf, strlen(buf));

P0_3 = 0;       /* 將GPIO 0_3設置為0, 與低有效LED連接*/

break;

case LED1_ON_CMD:/*處理LED1 ON命令*/

printf(“LED1_ON_CMD\r\n”);

sprintf(buf, “Turn ON the LED1\r\n”);

send(s, buf, strlen(buf));

P0_4 = 0;       /*將GPIO 0_4設置為0, 與低有效LED連接*/

break;

case LED2_ON_CMD:        /*處理LED2 ON命令*/

printf(“LED2_ON_CMD\r\n”);

sprintf(buf, “Turn ON the LED2\r\n”);

send(s, buf, strlen(buf));

P0_5 = 0;        /* 將GPIO 0_5設置為0,與低有效LED連接*/

break;

case LED0_OFF_CMD:        /* 處理LED0 OFF命令*/

printf(“LED0_OFF_CMD\r\n”);

sprintf(buf, “Turn OFF the LED0\r\n”);

send(s, buf, strlen(buf));

P0_3 = 1;        /*將GPIO 0_3設置為1,與低有效LED連接*/

break;

case LED1_OFF_CMD:        /* 處理LED1 OFF命令*/

printf(“LED1_OFF_CMD\r\n”);

sprintf(buf, “Turn OFF the LED1\r\n”);

send(s, buf, strlen(buf));

P0_4 = 1;        /* 將GPIO 0_4設置為1,與低有效LED連接*/

break;

case LED2_OFF_CMD:        /* 處理LED2 OFF命令*/

printf(“LED2_OFF_CMD\r\n”);

sprintf(buf, “Turn OFF the LED2\r\n”);

send(s, buf, strlen(buf));

P0_5 = 1;        /* 將GPIO 0_5設置為,與低有效LED連接*/

break;

case EXIT_CMD:        /*處理EXIT 命令*/

printf(“EXIT command\r\n”);

sprintf(buf, “EXIT command\r\n Good Bye~~\r\n Logout from W7100 TELNET”);

send(s, buf, strlen(buf));

close(s);

user_state = LOGOUT;

break;

default:

break;

}

}

程序4.4proc_command()函數

 

4.5 willopt(), wontopt(), doopt()和dontopt()函數

willopt(), wontopt(), doopt()和dontopt()函數是用於協商Telnet選項的命令。它們需要socket s和選項作為輸入參數。更多關於每個命令和選項的詳細信息,請參閱表2.1和2.2。

void willopt(SOCKET s, int opt)

{

int ack;

printf(“Recv: will”);

if(opt <= NOPTIONS)  printf(“%s\r\n”, tel_options[opt]);

else  printf(“%u\r\n”, opt);

switch(opt){

case TN_TRANSMIT_BINARY:

case TN_ECHO:

case TN_SUPPRESS_GA:

ack = DO;     /*如果接收到‘WILL’並且它包含TN_SUPPRESS_GA選項,發送‘DO’ */

break;

default:

ack = DONT;  /* 拒絕其它未定義的命令*/

}

sendIAC(s, ack, opt);

}

void wontopt(SOCKET s, int opt)

{

printf(“recv: wont”);

if (opt <= NOPTIONS)  printf(“%s\r\n”, tel_options[opt]);

else  printf(“%u\r\n”, opt);

switch(opt){

case TN_TRANSMIT_BINARY:

case TN_ECHO:

case TN_SUPPRESS_GA:    /*如果接收到WONT帶有TN_SUPPRESS_GA選項的命令*/

if (remote[opt] == 0){

remote[opt] = 1;     /*設置TN_SUPPRESS_GA選項*/

sendIAC(s, DONT, opt);  /* 發送帶有TN_SUPPRESS_GA 選項的DONT命令*/

}

break;

}

}

void doopt(SOCKET s, int opt)

{

printf(“recv: do “);

if (opt <= NOPTIONS)  printf(“%s\r\n”, tel_options[opt]);

else  printf(“%u\r\n”, opt);

switch(opt){

case TN_SUPPRESS_GA:      /*如果接收到帶有TN_SUPPRESS_GA 選項的DO命令*/

sendIAC(s, WILL, opt);     /* 發送帶有TN_SUPPRESS_GA選項的WILL命令*/

break;

case TN_ECHO:    /* 如果接收帶有TN_ECHO選項的DO命令*/

sprintf(buf, “WELCOME TO THE W7100 TELNET SERVER!!\r\nPlease insert your ID: “);

send(s, buf, strlen(buf));

break;

default:

sendIAC(s, WONT, opt);

}

}

void dontopt(int opt)

{

printf(“recv: dont “);

if (opt <= NOPTIONS) printf(“%s\r\n”, tel_options[opt]);

else printf(“%u\r\n”, opt);

switch(opt){

case TN_TRANSMIT_BINARY:

case TN_ECHO:

case TN_SUPPRESS_GA:    /*如果接收帶有TN_SUPPRESS_GA 選項的DON’T命令*/

if (remote[opt] == 0) remote[opt] = 1;  /*設置TN_SUPPRESS_GA選型*/

break;

}

}

程序4.5willopt(), wontopt(), doopt() and dontopt()函數

 

更多有關W7100的博文請看這裡:

W7100例程學習之ADC應用http://blog.iwiznet.cn/?p=691

用W7100,做自己的HTTP服務器 http://blog.iwiznet.cn/?p=684

W7100學習筆記 -W7100部分固件(firmware)解析http://blog.iwiznet.cn/?p=677

什麼是API函數以及如何使用W7100A進行初始化(TCPIPCore)?(二)http://blog.iwiznet.cn/?p=628

 

更多有關W7100A的博文請看這裡:

如何使用W7100A實現網絡字符LCD  http://blog.iwiznet.cn/?p=168

單片機以太網控制芯片— iMCU W7100A http://blog.iwiznet.cn/?p=229

如何使用W7100A模擬I2C總線 http://blog.iwiznet.cn/?p=304

如何實現W7100A程序燒錄 http://blog.iwiznet.cn/?p=161