当前位置:首页 > 单片机 > 单片机
[导读] 在上两篇博文(如何使用W7100A实现Telnet服务器(一)、如何使用W7100A实现Telnet服务器(二))里我们简单地介绍一下Telnet,第三章将会演示Telnet函数的具体功能,第四章将涉及到代码的分析。这篇文档

在上两篇博文(如何使用W7100A实现Telnet服务器(一)、如何使用W7100A实现Telnet服务器(二))里我们简单地介绍一下Telnet,第三章将会演示Telnet函数的具体功能,第四章将涉及到代码的分析。这篇文档中所有的示例代码都是基于Keil uVision3。

第一篇在这里:http://blog.csdn.net/wiznet2012/article/details/7721006

第二篇在这里:http://blog.csdn.net/wiznet2012/article/details/7728408

下面我们介绍第三部分(主要内容是第四章的余下的代码分析部分):

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] = '';

else data_buf[buf_index-1] = '';

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 commandsrnGET LED: Show all LED status rnLED0 ON/OFF: Turn ON/OFF the LED0rnLED1 ON/OFF: Turn ON/OFF the LED1

rnLED2 ON/OFF: Turn ON/OFF the LED2rnEXIT: Exit from W7100 Telnet serverrn"};

/*Translate the first word to lower case*/

for(cp = data_buf; *cp !=''; 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, "rnSuccessfully connected!!rnImplemented Command:

HELP, GET LED, LED0 ON/OFF, LED1 ON/OFF, LED2 ON/OFF, EXITrn");

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 commandrn");

sprintf(buf, "BAD commandrn");

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

return;

}

switch(cmdp - commands){

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

printf("HELP_CMDrn");

sprintf(buf, help);

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

break;

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

printf("GET_LED_CMDrn");

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

sprintf(buf, "LED%bd is %srn", i, ((P0 >> (i+3)) & 0x01) ? "OFF" : "ON");

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

}

break;

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

printf("LED0_ON_CMDrn");

sprintf(buf, "Turn ON the LED0rn");

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

P0_3 = 0; /* 将GPIO 0_3设置为0, 与低有效LED连接*/

break;

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

printf("LED1_ON_CMDrn");

sprintf(buf, "Turn ON the LED1rn");

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

P0_4 = 0; /*将GPIO 0_4设置为0, 与低有效LED连接*/

break;

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

printf("LED2_ON_CMDrn");

sprintf(buf, "Turn ON the LED2rn");

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

P0_5 = 0; /* 将GPIO 0_5设置为0,与低有效LED连接*/

break;

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

printf("LED0_OFF_CMDrn");

sprintf(buf, "Turn OFF the LED0rn");

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

P0_3 = 1; /*将GPIO 0_3设置为1,与低有效LED连接*/

break;

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

printf("LED1_OFF_CMDrn");

sprintf(buf, "Turn OFF the LED1rn");

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

P0_4 = 1; /* 将GPIO 0_4设置为1,与低有效LED连接*/

break;

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

printf("LED2_OFF_CMDrn");

sprintf(buf, "Turn OFF the LED2rn");

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

P0_5 = 1; /* 将GPIO 0_5设置为,与低有效LED连接*/

break;

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

printf("EXIT commandrn");

sprintf(buf, "EXIT commandrn Good Bye~~rn 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("%srn", tel_options[opt]);

else printf("%urn", 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("%srn", tel_options[opt]);

else printf("%urn", 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("%srn", tel_options[opt]);

else printf("%urn", 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!!rnPlease 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("%srn", tel_options[opt]);

else printf("%urn", 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()函数


本站声明: 本文章由作者或相关机构授权发布,目的在于传递更多信息,并不代表本站赞同其观点,本站亦不保证或承诺内容真实性等。需要转载请联系该专栏作者,如若文章内容侵犯您的权益,请及时联系本站删除。
换一批
延伸阅读

Sept. 8, 2025 ---- 根据TrendForce集邦咨询最新调查,2025年第二季NVIDIA(英伟达) Blackwell平台规模化出货,以及北美CSP业者持续扩大布局General Server(通用型...

关键字: SSD DDR4 服务器

9 月 5 日,一则关于英伟达的商业动态引发行业关注。这家 AI 芯片巨头斥资 15 亿美元,从人工智能小型云服务提供商 Lambda 手中,租用了搭载自家 GPU 芯片的服务器。

关键字: 英伟达 GPU 服务器 AI芯片

往期发布了基于小华HC32F334数字电源控制器的两路交错无桥图腾柱TCM PFC参考设计,TCM PFC以其全输入范围下软开关的优势越来越受到服务器电源以及通信电源的青睐。同时,两路交错无桥图腾柱CCM PFC因其EM...

关键字: 数字电源控制器 服务器 滤波器

Aug. 21, 2025 ---- 根据TrendForce集邦咨询最新液冷产业研究,随着NVIDIA GB200 NVL72机柜式服务器于2025年放量出货,云端业者加速升级AI数据中心架构,促使液冷技术从早期试点迈...

关键字: AI 数据中心 服务器

服务器作为企业信息化建设的核心基础设施,其供电系统的可靠性直接关系到业务连续性与数据安全。在数据中心场景中,电源故障是导致服务器宕机的主要原因之一,而内置电源架构的冗余设计与并联均流技术,正是解决这一问题的关键技术路径。...

关键字: 服务器 内置电源

北京——2025年8月15日,亚马逊云科技日前宣布,Amazon DocumentDB Serverless已正式可用,这是Amazon DocumentDB(兼容MongoDB)的一种全新配置,能够根据应用程序需求自动...

关键字: 数据库 服务器

在快速发展的AI领域,性能至关重要——而这不仅限于计算性能。现代数据中心里,连接GPU、交换机和服务器的网络基础设施承受着巨大的压力。随着AI模型扩展到数千亿个参数,行业关注的焦点正转向AI训练性能中最为关键但又经常被忽...

关键字: AI GPU 交换机 服务器

在企业数据中心的稳定运行中,UPS(不间断电源)扮演着至关重要的角色,它如同数据安全的第一道防线,在供电突发状况时为服务器争取宝贵的停机准备时间。然而,当 UPS 供电异常导致多台服务器突然关机,进而引发数据丢失时,不仅...

关键字: 数据安全 服务器 自动检测

采用分离式架构,充分利用主机 CPU 和 PCIe® 基础设施,克服传统存储瓶颈

关键字: CPU 数据中心 服务器

这些1 A和2 A器件采用小尺寸SlimSMA HV (DO-221AC)封装,提供了低电容电荷和3.2 mm的较大最小爬电距离

关键字: SiC 肖特基二极管 服务器
关闭