0
  • 聊天消息
  • 系统消息
  • 评论与回复
登录后你可以
  • 下载海量资料
  • 学习在线课程
  • 观看技术视频
  • 写文章/发帖/加入社区
会员中心
创作中心

完善资料让更多小伙伴认识你,还能领取20积分哦,立即完善>

3天内不再提示

RT-Thread使用webserver(lwip协议栈自带httpd )

冬至子 来源:YZRD 作者:YZRD 2023-10-12 14:47 次阅读

参考正点原子的 网络实验10 NETCONN_WEBserver实验和《lwIP开发指南》。

开发环境:野火的stm32f407,rt-thread studio版本是版本: 2.2.6,stm32f4的资源包为0.2.2,rt-thread版本为4.0.3。

以RT-Thread中Lan8720和lwip协议栈的使用文章创建的工程为基础。

httpd(The Apache HTTP Server)的官方网址。

rtthread工程中新建文件夹webserver,存放webserver相关文件。

对工程进行编译,正常通过。

需要修改的代码,过程如下:

rt-threadcomponentsnetlwip-2.0.2srcincludelwipappshttpd_opts.h 文件中的宏定义

LWIP_HTTPD_CGI 默认为0,改为1
LWIP_HTTPD_SSI 默认为0,改为1
HTTPD_USE_CUSTOM_FSDATA默认为0
LWIP_HTTPD_DYNAMIC_FILE_READ默认为0,改为1
将 rt-threadcomponentsnetlwip-2.0.2srcappshttpd文件夹 添加构建
httpd文件夹下的fsdata.c 排除构建。

在主函数中增加如下代码

extern void httpd_init(void);
httpd_init();
while (count++)
{
LOG_D("Hello RT-Thread!");
rt_thread_mdelay(10000);
}

编译正常,下载到开发板,效果如图1:

1.jpg

如何使用自己的网页呢?修改如下:
将rt-threadcomponentsnetlwip-2.0.2srcappshttpd文件夹下的fsdata.c替换成自己的fsdata.c,(使用makefsdata.exe这个软件自动生成即可)。

在webserve文件夹下创建httpd_cgi_ssi.c文件(CGI和SSI句柄函数)
参考原子的代码,修改如下:

#include
#include "lwip/tcp.h"
#include
#include
#include
int LED1=0;
int BEEP=0;
#define NUM_CONFIG_CGI_URIS (sizeof(ppcURLs) / sizeof(tCGI))
#define NUM_CONFIG_SSI_TAGS (sizeof(ppcTAGs) / sizeof(char *))
//extern short Get_Temprate(void);
//extern void RTC_Get_Time(u8 *hour,u8 *min,u8 *sec,u8 *ampm);
//extern void RTC_Get_Date(u8 *year,u8 *month,u8 *date,u8 *week);
//控制LED的CGI handler
const char* LEDS_CGI_Handler(int iIndex, int iNumParams, char *pcParam[], char *pcValue[]);
const char* BEEP_CGI_Handler(int iIndex,int iNumParams,char *pcParam[],char *pcValue[]);
static const char *ppcTAGs[]= //SSI的Tag
{
"t", //ADC
"w", //温度值
"h", //时间
"y" //日期
};
static const tCGI ppcURLs[]= //cgi程序
{
{"/leds.cgi",LEDS_CGI_Handler},
{"/beep.cgi",BEEP_CGI_Handler},
};
//当web客户端请求浏览器的时候,使用此函数被CGI handler调用
static int FindCGIParameter(const char *pcToFind,char *pcParam[],int iNumParams)
{
int iLoop;
for(iLoop = 0;iLoop < iNumParams;iLoop ++ )
{
if(strcmp(pcToFind,pcParam[iLoop]) == 0)
{
return (iLoop); //返回 iLOOP
}
}
return (-1);
}
//SSIHandlerÖ中adc处理函数
void ADC_Handler(char *pcInsert)
{
char Digit1=0, Digit2=0, Digit3=0, Digit4=0;
static uint32_t ADCVal = 0;
//ADCVal = Get_Adc_Average(5,10);//ADC1_CH5的电压值
ADCVal+=10;
ADCVal=ADCVal%3000;
//转换为 ADCVval * 0.8mv
ADCVal = (uint32_t)(ADCVal * 0.8);
Digit1= ADCVal/1000;
Digit2= (ADCVal-(Digit1*1000))/100 ;
Digit3= (ADCVal-((Digit1*1000)+(Digit2*100)))/10;
Digit4= ADCVal -((Digit1*1000)+(Digit2*100)+ (Digit3*10));
/* 准备添加到html中的数据 */
*pcInsert = (char)(Digit1+0x30);
*(pcInsert + 1) = (char)(Digit2+0x30);
*(pcInsert + 2) = (char)(Digit3+0x30);
*(pcInsert + 3) = (char)(Digit4+0x30);
}
//SSIHandler中需要用到的内部处理温度传感器
void Temperate_Handler(char *pcInsert)
{
char Digit1=0, Digit2=0, Digit3=0, Digit4=0,Digit5=0;
static short Temperate = 0;
//Temperate = Get_Temprate();
Temperate+=1.3;
Digit1 = Temperate / 10000;
Digit2 = (Temperate % 10000)/1000;
Digit3 = (Temperate % 1000)/100 ;
Digit4 = (Temperate % 100)/10;
Digit5 = Temperate % 10;
/* 准备添加到html中的数据 */
*pcInsert = (char)(Digit1+0x30);
*(pcInsert+1) = (char)(Digit2+0x30);
*(pcInsert+2) = (char)(Digit3+0x30);
*(pcInsert+3) = '.';
*(pcInsert+4) = (char)(Digit4+0x30);
*(pcInsert+5) = (char)(Digit5+0x30);
}
//SSIHandler中需要用到的处理RTC日时间的函数
void RTCTime_Handler(char *pcInsert)
{
static uint8_t hour,min,sec,ampm;
hour++;
min++;
sec++;
ampm++;
//RTC_Get_Time(&hour,&min,&sec,&m);
/* 准备添加到html中的数据 */
*pcInsert = (char)((hour/10) + 0x30);
*(pcInsert+1) = (char)((hour%10) + 0x30);
*(pcInsert+2) = ':';
*(pcInsert+3) = (char)((min/10) + 0x30);
*(pcInsert+4) = (char)((min%10) + 0x30);
*(pcInsert+5) = ':';
*(pcInsert+6) = (char)((sec/10) + 0x30);
*(pcInsert+7) = (char)((sec%10) + 0x30);
}
//SSIHandler中需要用到的处理RTC日期的函数
void RTCdate_Handler(char *pcInsert)
{
static uint8_t year,month,date,week;
//RTC_Get_Date(&year,&month,&date,&week);
year++;
month++;
date++;
week++;
/* 准备添加到html中的数据 */
*pcInsert = '2';
*(pcInsert+1) = '0';
*(pcInsert+2) = (char)((year/10) + 0x30);
*(pcInsert+3) = (char)((year%10) + 0x30);
*(pcInsert+4) = '-';
*(pcInsert+5) = (char)((month/10) + 0x30);
*(pcInsert+6) = (char)((month%10) + 0x30);
*(pcInsert+7) = '-';
*(pcInsert+8) = (char)((date/10) + 0x30);
*(pcInsert+9) = (char)((date%10) + 0x30);
*(pcInsert+10) = ' ';
*(pcInsert+11) = 'w';
*(pcInsert+12) = 'e';
*(pcInsert+13) = 'e';
*(pcInsert+14) = 'k';
*(pcInsert+15) = ':';
*(pcInsert+16) = (char)(week + 0x30);
}
//SSI的 Handler 句柄
static u16_t SSIHandler(int iIndex,char *pcInsert,int iInsertLen)
{
switch(iIndex)
{
case 0:
ADC_Handler(pcInsert);
break;
case 1:
Temperate_Handler(pcInsert);
break;
case 2:
RTCTime_Handler(pcInsert);
break;
case 3:
RTCdate_Handler(pcInsert);
break;
}
return strlen(pcInsert);
}
//CGI LED控制句柄
const char* LEDS_CGI_Handler(int iIndex, int iNumParams, char *pcParam[], char *pcValue[])
{
uint8_t i=0; //注意根据自己的GET的参数的多少来选择i值范围
iIndex = FindCGIParameter("LED1",pcParam,iNumParams); //找到LED的索引
//只有一个CGI句柄 iIndex=0
if (iIndex != -1)
{
LED1=1;
for (i=0; i {
if (strcmp(pcParam[i] , "LED1")==0) //检查参数"led"
{
if(strcmp(pcValue[i], "LED1ON") ==0)
{
LED1=0;
rt_kprintf("LED1ONn");
}
else if(strcmp(pcValue[i],"LED1OFF") == 0)
{
LED1=1;
rt_kprintf("LED1OFFn");
}
}
}
}
if(LED1 == 0 && BEEP == 0) return "/STM32F407LED_ON_BEEP_OFF.shtml"; //
else if(LED1 == 0 && BEEP == 1) return "/STM32F407LED_ON_BEEP_ON.shtml"; //
else if(LED1 == 1 && BEEP == 1) return "/STM32F407LED_OFF_BEEP_ON.shtml"; //
else return "/STM32F407LED_OFF_BEEP_OFF.shtml"; //
}
//BEEP的CGI控制句柄
const char *BEEP_CGI_Handler(int iIndex,int iNumParams,char *pcParam[],char *pcValue[])
{
uint8_t i=0;
iIndex = FindCGIParameter("BEEP",pcParam,iNumParams); //找到BEEP的索引号
if(iIndex != -1) //找到BEEP的索引号
{
BEEP=0; //
for(i = 0;i < iNumParams;i++)
{
if(strcmp(pcParam[i],"BEEP") == 0) //
{
if(strcmp(pcValue[i],"BEEPON") == 0) //
{
BEEP = 1;
rt_kprintf("BEEPONn");
}
else if(strcmp(pcValue[i],"BEEPOFF") == 0) //
{
BEEP = 0;
rt_kprintf("BEEPOFFn");
}
}
}
}
if(LED1 == 0 && BEEP == 0) return "/STM32F407LED_ON_BEEP_OFF.shtml"; //
else if(LED1 == 0 && BEEP == 1) return "/STM32F407LED_ON_BEEP_ON.shtml"; //
else if(LED1 == 1 && BEEP == 1) return "/STM32F407LED_OFF_BEEP_ON.shtml"; //
else return "/STM32F407LED_OFF_BEEP_OFF.shtml"; //
}
//SSI句柄初始化
void httpd_ssi_init(void)
{
//配置内部温度传感器的SSI句柄
http_set_ssi_handler(SSIHandler,ppcTAGs,NUM_CONFIG_SSI_TAGS);
}
//CGI句柄初始化
void httpd_cgi_init(void)
{
//配置CGI句柄 LEDs control CGI) */
http_set_cgi_handlers(ppcURLs, NUM_CONFIG_CGI_URIS);
}

主函数中代码如下:

extern void httpd_ssi_init(void);
extern void httpd_cgi_init(void);
extern void httpd_init(void);
httpd_cgi_init();
httpd_ssi_init();
httpd_init();

重新编译,下载到开发板,在浏览器输入开发板ip地址,查看效果。效果如图2:

1.jpg

LWIP HTTP 协议中默认只支持GET方法 但是一般提交表单时都用POST方法 而LWIPPOST方案需要自己实现 不过LWIP已经需要实现的函数申明在httpd.h中了。

声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。 举报投诉
  • LwIP协议栈
    +关注

    关注

    0

    文章

    19

    浏览量

    7338
  • RT-Thread
    +关注

    关注

    31

    文章

    1271

    浏览量

    39902
收藏 人收藏

    评论

    相关推荐

    如何使用RT-Thread Studio添加以太网驱动和lwIP协议

    用户快速开发自己的网络应用。 本文将基于正点原子 stm32f407-atk-explorer 开发板主要介绍如何使用 RT-Thread Studio 来添加以太网驱动和 lwIP 协议
    发表于 03-29 06:29

    如何去处理RT-Thread/LwIP TCP发送速度偏低的情况呢

    RT-Thread使用LwIP做为TCP/IP协议LwIP包含了非常多的选项,不同的选项进行不同功能的剪裁,部分选项
    发表于 08-10 11:48

    SC0085 AT32在RT-Thread上运行LwIP协议

    示例目的演示在RT-Thread上运行LwIP协议,并搭建一个UDP echo server支持型号列表:支持型号AT32F407xxAT32F437xx主要使用外设列表:主要使用外
    发表于 08-23 20:40

    基于RT-Thread实现的Agile Modbus协议

    基于 RT-Thread 实现的支持 Modbus 固件升级的 Bootloader:HPM6750_Boot  特性  支持 rtu 及 tcp 协议,使用纯 C 开发,不涉及任何硬件接口,可在任何形式的硬件上
    发表于 10-08 15:04

    RT-Thread移植使用webserver的流程

    ,stm32f4的资源包为0.2.2,rt-thread版本为4.0.3。以RT-Thread中Lan8720和lwip协议的使用文章创建
    发表于 02-22 15:10

    RT-Thread使用webserver的步骤

    sTM32F407,rt-thead studio版本版本:2.2.6 .3。以RT-Thread中Lan8720和lwip协议的使用文章
    发表于 02-24 14:47

    RT-Thread编程指南

    RT-Thread编程指南——RT-Thread开发组(2015-03-31)。RT-Thread做为国内有较大影响力的开源实时操作系统,本文是RT-Thread实时操作系统的编程指南
    发表于 11-26 16:06 0次下载

    RT-Thread全球技术大会:在RT-Thread中使用帧来调试程序

    百问网科技CTO韦东山,在RT-Thread全球技术大会大会中,以在RT-Thread中使用帧来调试程序为主题进行了介绍。
    的头像 发表于 05-28 09:33 1802次阅读
    <b class='flag-5'>RT-Thread</b>全球技术大会:在<b class='flag-5'>RT-Thread</b>中使用<b class='flag-5'>栈</b>帧来调试程序

    RT-Thread Studio如何测试Ethernet连网

    RT-Thread 是一款开源实时操作系统,包括 RT-Thread 内核、FinSH 命令行工具、设备文件系统、TCP/IP 协议RT-Thr
    的头像 发表于 05-31 11:17 2767次阅读
    <b class='flag-5'>RT-Thread</b> Studio如何测试Ethernet连网

    RT-Thread文档_RT-Thread 简介

    RT-Thread文档_RT-Thread 简介
    发表于 02-22 18:22 5次下载
    <b class='flag-5'>RT-Thread</b>文档_<b class='flag-5'>RT-Thread</b> 简介

    RT-Thread文档_RT-Thread SMP 介绍与移植

    RT-Thread文档_RT-Thread SMP 介绍与移植
    发表于 02-22 18:31 9次下载
    <b class='flag-5'>RT-Thread</b>文档_<b class='flag-5'>RT-Thread</b> SMP 介绍与移植

    RT-Thread移植使用webserver (lwip+httpd)

    开发环境:野火的stm32f407,rt-thread studio版本是版本: 2.2.6,stm32f4的资源包为0.2.2,rt-thread版本为4.0.3。
    的头像 发表于 10-12 12:49 1258次阅读
    <b class='flag-5'>RT-Thread</b>移植使用<b class='flag-5'>webserver</b> (<b class='flag-5'>lwip+httpd</b>)

    RT-Thread在Lan8720a和 lwip基础上移植ntp流程

    开发环境:野火的stm32f407,rt-thread studio版本是版本: 2.2.6,stm32f4的资源包为0.2.2。以RT-Thread中Lan8720和lwip协议
    的头像 发表于 10-12 16:59 1779次阅读
    <b class='flag-5'>RT-Thread</b>在Lan8720a和 <b class='flag-5'>lwip</b>基础上移植ntp流程

    RT-Thread中Lan8720和lwip协议的使用

    开发环境:野火的stm32f407,rt-thread studio版本是版本: 2.2.6,stm32f4的资源包为0.2.2,Agile Modbus软件包版本为v1.1.2。新建选择芯片工程。
    的头像 发表于 10-13 10:19 1316次阅读
    <b class='flag-5'>RT-Thread</b>中Lan8720和<b class='flag-5'>lwip</b><b class='flag-5'>协议</b><b class='flag-5'>栈</b>的使用

    RT-Thread中mymqtt软件包的使用方法

    在上一篇文章 RT-Thread中Lan8720和lwip协议的使用的工程基础上添加mymqtt软件包。 使能mqtt example和mqtt test,保存,等待下载更新软件包
    的头像 发表于 10-13 10:44 919次阅读
    <b class='flag-5'>RT-Thread</b>中mymqtt软件包的使用方法