学习了RT-Thread的内核也有一段时间了,由于各种各样的琐事自己没有去做一个综合应用示例,刚最近有点时间,做了一个对接OneNET的历程,采用的是OneNET的EDP协议,关于OneNET的EDP协议可以点击跳转至OneNET EDP协议讲解与应用这篇博客看一下,这篇博客会比较详细的介绍对接EDP协议的思路。本篇博客会根据着重的将RT-Thread的应用思路,在本应用示例也只是用来软件定时器、线程管理、信号量而已,因为我用的是潘多拉开发板进行实验,所以AP6181驱动就直接用RT-Thread的,自己也只是在这个基础上做了应用。
使用AP6181 WiFi模组对接OneNET应用示例:STM32+AP6181(SDIO WiFi模组)对接OneNET,实现温湿度定时上报,控制上报周期、控制设备LED灯、蜂鸣器以及电机,掉线自动连接、WiFi自动连接。整个工程代码点击跳转至GitHub获取。
在移植时,主要也就是修改几个地方,代码如下:
#define ONENET_EDP_PRODUCT_ID "190254" /* 产品id */
#define ONENET_EDP_PRODUCT_APIKEY "cWPICK6PDU6cOHP=T0SqMcXWRc4=" /* api key */
#define ONENET_EDP_DEVICE_ID "504890772" /* 设备id */
#define ONENET_EDP_DEVICE_AUTHKEY "edp20181122" /* 设备鉴权信息 */
需要修改的为产品ID、APIKEY、设备ID、设备鉴权信息,将这些信息修改为你自己在OneNET开发者中心的产品信息或设备信息。
1、开机后,RT-Thread完成一系列的初始化后进入main线程,首先进入main函数会进行执行onenet_edp_sample();
函数进行信号量创建、软件定时器创建以及线程创建:
rt_err_t onenet_edp_sample(void)
{
/* 连接OneNET信号量 */
link_onenet_sem = rt_sem_create("link_onenet_sem", 0, RT_IPC_FLAG_FIFO);
if(link_onenet_sem == RT_NULL)
{
rt_kprintf("[EDP]link_onenet_sem create failed\r\n");
return RT_ERROR;
}
/* 发送心跳信号量 */
send_heart_sem = rt_sem_create("send_heart_sem", 0, RT_IPC_FLAG_FIFO);
if(send_heart_sem == RT_NULL)
{
rt_kprintf("[EDP]send_heart_sem create failed\r\n");
return RT_ERROR;
}
/* 用于获取温湿度和发送温湿度信号量 */
humi_temp_sem = rt_sem_create("humi_temp_sem", 1, RT_IPC_FLAG_FIFO);
if(humi_temp_sem == RT_NULL)
{
rt_kprintf("[EDP]humi_temp_sem create failed\r\n");
return RT_ERROR;
}
/* 用于定时触发上报温湿度数据信号量 */
send_humi_temp_sem = rt_sem_create("send_humi_temp_sem", 0, RT_IPC_FLAG_FIFO);
if(send_humi_temp_sem == RT_NULL)
{
rt_kprintf("[EDP]send_humi_temp_sem create failed\r\n");
return RT_ERROR;
}
/* 软件定时器,周期执行,120秒 */
send_heart_timer = rt_timer_create("send_heart_timer", send_heart_timer_callback, RT_NULL, 120000, RT_TIMER_FLAG_SOFT_TIMER | RT_TIMER_FLAG_PERIODIC);
if(send_heart_timer == RT_NULL)
{
rt_kprintf("[EDP]send_heart_timer create failed\r\n");
return RT_ERROR;
}
/* 软件定时器,周期执行,300秒 */
send_humi_temp_timer = rt_timer_create("send_humi_temp_timer", send_humi_temp_timer_callback, RT_NULL, 30000, RT_TIMER_FLAG_SOFT_TIMER | RT_TIMER_FLAG_PERIODIC);
if(send_humi_temp_timer == RT_NULL)
{
rt_kprintf("[EDP]send_humi_temp_timer create failed\r\n");
return RT_ERROR;
}
/* 连接设备线程 */
link_dev_thread = rt_thread_create("link_dev_thread", onenet_edp_link_device_thread, RT_NULL, 2048, 4, 200);
if(link_dev_thread == RT_NULL)
{
rt_kprintf("[EDP]link_dev_thread create failed\r\n");
return RT_ERROR;
}
rt_thread_startup(link_dev_thread);
/* 发送心跳线程 */
send_heart_thread = rt_thread_create("send_heart_thread", onenet_edp_send_heart_thread, RT_NULL, 1024, 4, 50);
if(send_heart_thread == RT_NULL)
{
rt_kprintf("[EDP]send_heart_thread create failed\r\n");
return RT_ERROR;
}
rt_thread_startup(send_heart_thread);
/* 平台下发命令或结果处理线程 */
recv_data_pro_thread = rt_thread_create("recv_data_pro_thread",onenet_edp_receive_process_thread, RT_NULL, 4096, 4, 100);
if(recv_data_pro_thread == RT_NULL)
{
rt_kprintf("[EDP]recv_data_pro_thread create failed\r\n");
return RT_ERROR;
}
rt_thread_startup(recv_data_pro_thread);
/* 获取温湿度数据线程 */
get_humi_temp_thread = rt_thread_create("get_humi_temp_thread", onenet_edp_get_humi_temp_thread, RT_NULL, 4096, 4, 50);
if(get_humi_temp_thread == RT_NULL)
{
rt_kprintf("[EDP]get_humi_temp_thread create failed\r\n");
return RT_ERROR;
}
rt_thread_startup(get_humi_temp_thread);
/* 上报温湿度数据到平台线程 */
send_humi_temp_thread = rt_thread_create("send_humi_temp_thread", onenet_edp_send_humi_temp_thread, RT_NULL, 4096, 4, 100);
if(send_humi_temp_thread == RT_NULL)
{
rt_kprintf("[EDP]send_humi_temp_thread create failed\r\n");
return RT_ERROR;
}
rt_thread_startup(send_humi_temp_thread);
return RT_EOK;
}
2、完成信号量创建、软件定时器的线程创建后,会进行注册网络就绪事件回调、网络断开事件回调、以及连接网络、开启自动连接网络:
/* 注册网络就绪回调、 */
rt_wlan_register_event_handler(RT_WLAN_EVT_READY, wlan_ready_handler, RT_NULL);
/* 注册网络断开事件回调 */
rt_wlan_register_event_handler(RT_WLAN_EVT_STA_DISCONNECTED, wlan_station_disconnect_handler, RT_NULL);
/* 连接路由器 */
rt_wlan_connect(WLAN_SSID, WLAN_PASSWORD);
rt_thread_mdelay(1000);
/* 开启自动连接 */
wlan_autoconnect_init();
rt_wlan_config_autoreconnect(RT_TRUE);
3、网络就绪回调函数用于释放连接OneNET信号量,告诉系统已经有网络了,可以去连接OneNET了:
void wlan_ready_handler(int event, struct rt_wlan_buff *buff, void *parameter)
{
rt_kprintf("wlan ready\r\n");
if(link_onenet_sem != RT_NULL)
{
rt_sem_release(link_onenet_sem);
}
}
而另一方面,连接OneNET线程会移植挂起等待连接OneNET的信号量,指导获取到这个信号量才开始去连接OneNET:
/**************************************************************
函数名称:onenet_edp_link_device_thread
函数功能:连接OneNET平台设备线程
输入参数:parameter:线程入口参数
返 回 值:无
备 注:WiFi注册上网了后连接平台,连接成功则启动定时器发送心跳
**************************************************************/
void onenet_edp_link_device_thread(void *parameter)
{
rt_err_t result = RT_ERROR;
while(1)
{
result = rt_sem_take(link_onenet_sem, RT_WAITING_FOREVER); /* 等待时间:一直等 */
if(RT_EOK == result)
{
#ifdef USING_DEVICEID_APIKEY_LINK
onenet_edp_link_device(ONENET_EDP_SRV_ADDR, ONENET_EDP_SRV_PORT, ONENET_EDP_DEVICE_ID, ONENET_EDP_PRODUCT_APIKEY);
#else
onenet_edp_link_device(ONENET_EDP_SRV_ADDR, ONENET_EDP_SRV_PORT, ONENET_EDP_PRODUCT_ID, ONENET_EDP_DEVICE_AUTHKEY);
#endif
}
rt_thread_mdelay(1);
rt_thread_yield();/* 放弃剩余时间片,进行一次线程切换 */
}
}
1、连接上OneNET之后,处理平台下发命令或结果的线程会得到连接成功返回结果并进行处理,启动软件定时器进行发送心跳和上报数据:
case 0:
{
rt_kprintf("Tips: 连接成功\r\n");
if(send_heart_timer != RT_NULL)
{
rt_kprintf("[EDP]start send_heart_timer\r\n");
rt_timer_start(send_heart_timer);/* 启动软件定时器定时发送心跳 */
}
if(send_humi_temp_timer != RT_NULL)
{
rt_kprintf("[EDP]start send_humi_temp_timer\r\n");
rt_timer_start(send_humi_temp_timer);/* 启动软件定时器定时上报数据 */
}
break;
}
2、软件定时器回调函数用于释放发送心跳信号量:
/**************************************************************
函数名称:send_heart_timer_callback
函数功能:软件定时器回调函数,用于定时发送信号量来触发发送心跳包
输入参数:parameter:入口参数
返 回 值:无
备 注:无
**************************************************************/
void send_heart_timer_callback(void *parameter)
{
if(send_heart_timer != RT_NULL)
{
rt_sem_release(send_heart_sem);
}
}
3、发送心跳线程获取到信号量之后执行发送心跳操作:
/**************************************************************
函数名称:onenet_edp_send_heart_thread
函数功能:发送心跳线程
输入参数:parameter:线程入口参数
返 回 值:无
备 注:无
**************************************************************/
void onenet_edp_send_heart_thread(void *parameter)
{
rt_err_t result = RT_ERROR;
while(1)
{
result = rt_sem_take(send_heart_sem, RT_WAITING_FOREVER); /* 等待时间:一直等 */
if(RT_EOK == result)
{
onenet_edp_send_heart();
}
rt_thread_mdelay(1);
rt_thread_yield();/* 放弃剩余时间片,进行一次线程切换 */
}
}
4、同理,上报数据也是等待信号量,一旦获取到就上报数据,为保证数据上报的同步,获取数据与上报数据用来二值信号量,实现代码:
/**************************************************************
函数名称:onenet_edp_get_humi_temp_thread
函数功能:获取温湿度数据线程
输入参数:parameter:线程入口参数
返 回 值:无
备 注:无
**************************************************************/
void onenet_edp_get_humi_temp_thread(void *parameter)
{
rt_err_t result = RT_ERROR;
while(1)
{
result = rt_sem_take(send_humi_temp_sem, RT_WAITING_FOREVER); /* 等待时间:一直等 */
if(RT_EOK == result)
{
result = rt_sem_take(humi_temp_sem, RT_WAITING_FOREVER); /* 等待时间:一直等 */
if(RT_EOK == result)
{
aht10_read_data(&g_temperature, &g_humidity);
}
rt_sem_release(humi_temp_sem); /* 释放信号量 */
}
rt_thread_mdelay(1);
rt_thread_yield();/* 放弃剩余时间片,进行一次线程切换 */
}
}
/**************************************************************
函数名称:onenet_edp_send_humi_temp_thread
函数功能:发送温湿度数据线程
输入参数:parameter:线程入口参数
返 回 值:无
备 注:无
**************************************************************/
void onenet_edp_send_humi_temp_thread(void *parameter)
{
rt_err_t result = RT_ERROR;
while(1)
{
result = rt_sem_take(send_humi_temp_sem, RT_WAITING_FOREVER); /* 等待时间:一直等 */
if(RT_EOK == result)
{
result = rt_sem_take(humi_temp_sem, RT_WAITING_FOREVER); /* 等待时间:一直等 */
if(RT_EOK == result)
{
/* 上传温湿度数据到平台 */
if(RT_EOK == onenet_edp_send_data(FORMAT_TYPE3, ONENET_EDP_DEVICE_ID, ONENET_EDP_PRODUCT_APIKEY, humi_temp_data_stream, humi_temp_data_stream_cnt))
{
rt_kprintf("Humi_Temp Send success\n");
}
else
{
rt_kprintf("Humi_Temp Status Send failed\n");
}
}
rt_sem_release(humi_temp_sem); /* 释放信号量 */
}
rt_thread_mdelay(1);
rt_thread_yield();/* 放弃剩余时间片,进行一次线程切换 */
}
}
1、如果在有网络的情况下设备突然断开,那么就需要重新连接OneNET了,首先会停止所有定时器、接着重新发送连接OneNET信号量,处理如下:
case DISCONNECT:
{
rt_kprintf("WARN:连接断开,准备重连\r\n");
if(send_heart_timer != RT_NULL)
{
rt_kprintf("[EDP]stop send_heart_timer\r\n");
rt_timer_stop(send_heart_timer);/* 关闭软件定时器 */
}
if(send_humi_temp_timer != RT_NULL)
{
rt_kprintf("[EDP]stop send_humi_temp_timer\r\n");
rt_timer_stop(send_humi_temp_timer);/* 关闭软件定时器 */
}
if(link_onenet_sem != RT_NULL)
{
rt_sem_release(link_onenet_sem);
}
break;
}
2、如果是网络断开,那么就需要停止软件定时器,处理如下:
void wlan_station_disconnect_handler(int event, struct rt_wlan_buff *buff, void *parameter)
{
rt_kprintf("disconnect from the network!\n");
if(send_heart_timer != RT_NULL)
{
rt_kprintf("[EDP]stop send_heart_timer\r\n");
rt_timer_stop(send_heart_timer);/* 关闭软件定时器 */
}
if(send_humi_temp_timer != RT_NULL)
{
rt_kprintf("[EDP]stop send_humi_temp_timer\r\n");
rt_timer_stop(send_humi_temp_timer);/* 关闭软件定时器 */
}
}
1、在命令处理里面,主要做了控制LED、BEEP、电机以及控制上报数据周期,其中控制数据上报周期用到了rt_timer_control
函数,处理如下:
/**************************************************************
函数名称:onenet_edp_command_callback
函数功能:平台下发命令回调函数
输入参数:req:下发的命令
返 回 值:无
备 注:无
**************************************************************/
void onenet_edp_command_callback(char *req)
{
char *time_ptr = NULL;
unsigned int current_time = 0;
rt_pin_mode(PIN_LED_R, PIN_MODE_OUTPUT);
rt_pin_mode(PIN_MOTOR_A, PIN_MODE_OUTPUT);
rt_pin_mode(PIN_MOTOR_B, PIN_MODE_OUTPUT);
rt_pin_mode(PIN_BEEP, PIN_MODE_OUTPUT);
led_ctrl(0);
beep_ctrl(0);
motor_ctrl(0);
/* LED */
if(strstr((const char*)req, "LED"))
{
if(strcmp("LED:1", req) == 0)
{
g_led_status = 1;
led_ctrl(1);
/* 上传LED状态数据到平台 */
onenet_edp_send_data(FORMAT_TYPE3, ONENET_EDP_DEVICE_ID, ONENET_EDP_PRODUCT_APIKEY, led_status_data_stream, led_status_data_stream_cnt);
}
else if(strcmp("LED:0", req) == 0)
{
g_led_status = 0;
led_ctrl(0);
/* 上传LED状态数据到平台 */
onenet_edp_send_data(FORMAT_TYPE3, ONENET_EDP_DEVICE_ID, ONENET_EDP_PRODUCT_APIKEY, led_status_data_stream, led_status_data_stream_cnt);
}
else
{
}
}
/* 蜂鸣器 */
else if(strstr((const char*)req, "BEEP"))
{
if(strcmp("BEEP:1", req) == 0)
{
g_beep_status = 1;
beep_ctrl(1);
/* 上传BEEP状态数据到平台 */
onenet_edp_send_data(FORMAT_TYPE3, ONENET_EDP_DEVICE_ID, ONENET_EDP_PRODUCT_APIKEY, beep_status_data_stream, beep_status_data_stream_cnt);
}
else if(strcmp("BEEP:0", req) == 0)
{
g_beep_status = 0;
beep_ctrl(0);
/* 上传BEEP状态数据到平台 */
onenet_edp_send_data(FORMAT_TYPE3, ONENET_EDP_DEVICE_ID, ONENET_EDP_PRODUCT_APIKEY, beep_status_data_stream, beep_status_data_stream_cnt);
}
else
{
}
}
/* 电机 */
else if(strstr((const char*)req, "MOTOR"))
{
if(strcmp("MOTOR:1", req) == 0)
{
g_motor_status = 1;
motor_ctrl(1);
/* 上传BEEP状态数据到平台 */
onenet_edp_send_data(FORMAT_TYPE3, ONENET_EDP_DEVICE_ID, ONENET_EDP_PRODUCT_APIKEY, motor_status_data_stream, motor_status_data_stream_cnt);
}
else if(strcmp("MOTOR:0", req) == 0)
{
g_motor_status = 0;
motor_ctrl(0);
/* 上传BEEP状态数据到平台 */
onenet_edp_send_data(FORMAT_TYPE3, ONENET_EDP_DEVICE_ID, ONENET_EDP_PRODUCT_APIKEY, motor_status_data_stream, motor_status_data_stream_cnt);
}
else
{
}
}
else if(strstr((const char*)req, "TIME"))
{
time_ptr = req + 5;
g_data_time = (RT_TICK_PER_SECOND * atoi(time_ptr));/* 单位:秒 */
if(g_data_time > 0)
{
rt_kprintf("g_data_time:%d\r\n", g_data_time);
if(RT_EOK == rt_timer_control(send_humi_temp_timer, RT_TIMER_CTRL_SET_TIME, &g_data_time))
{
rt_kprintf("修改定时上传温湿度周期成功,当前周期值为:%d\r\n", g_data_time);
}
else
{
rt_kprintf("修改定时上传温湿度周期失败\r\n");
}
}
if(RT_EOK == rt_timer_control(send_humi_temp_timer, RT_TIMER_CTRL_GET_TIME, ¤t_time))
{
rt_kprintf("current_time:%d\r\n", current_time);
}
/* 上传获取数据周期值到平台 */
onenet_edp_send_data(FORMAT_TYPE3, ONENET_EDP_DEVICE_ID, ONENET_EDP_PRODUCT_APIKEY, data_time_data_stream, data_time_data_stream_cnt);
}
else
{
rt_kprintf("receive other commond\r\n");
}
}
1、连接成功
2、发送心跳成功
3、上报数据成功
4、接收命令执行操作以及上报状态成功:
1、信息如下:
\ | /
- RT - Thread Operating System
/ | \ 4.0.0 build Mar 15 2019
2006 - 2018 Copyright by rt-thread team
lwIP-2.0.2 initialized!
[SFUD] Find a Winbond flash chip. Size is 16777216 bytes.
[SFUD] w25q128 flash device is initialize success.
msh />[I/FAL] RT-Thread Flash Abstraction Layer (V0.2.0) initialize success.
[I/OTA] RT-Thread OTA package(V0.1.3) initialize success.
[I/OTA] Verify ‘wifi_image’ partition(fw ver: 1.0, timestamp: 1529386280) success.
[I/WICED] wifi initialize done. wiced version 3.3.1
[I/WLAN.dev] wlan init success
[I/WLAN.lwip] eth device init ok name:w0
join ssid:MY_WiFi
[I/WLAN.mgnt] wifi connect success ssid:MY_WiFi
[Flash] EasyFlash V3.2.1 is initialize success.
[Flash] You can get the latest version on https://github.com/armink/EasyFlash .
wlan ready
socket create success!!!, socket_id:0
[I/WLAN.lwip] Got IP address : 192.168.43.5
socket connect success!!!
[EDP]EDP_PacketConnect send success
socket receive data
Tips: 连接成功
[EDP]start send_heart_timer
[EDP]start send_humi_temp_timer
[EDP]send heart
socket receive data
cmdid: 00a8d84c-d740-586c-94d1-4e6ac32dc883, req: LED:1, req_len: 5
Send 35 Bytes
socket receive data
Tips: HeartBeat OK
socket receive data
socket receive data
Tips: Send Ok
[EDP]send heart
socket receive data
Tips: HeartBeat OK
socket receive data
cmdid: 4b3ffb43-4f0c-50f3-9370-1ec9179658d8, req: LED:0, req_len: 5
Send 35 Bytes
socket receive data
socket receive data
Tips: Send Ok
socket receive data
cmdid: 3c039820-9487-5415-948d-f209550672e5, req: LED:1, req_len: 5
Send 35 Bytes
socket receive data
socket receive data
Tips: Send Ok
socket receive data
cmdid: e8f37c5b-d136-5ff5-95f2-40a530d9d553, req: LED:0, req_len: 5
Send 35 Bytes
socket receive data
socket receive data
Tips: Send Ok
socket receive data
cmdid: f38d85db-2b5b-51a9-94c7-6fda84de0ce1, req: LED:1, req_len: 5
Send 35 Bytes
socket receive data
socket receive data
Tips: Send Ok
socket receive data
cmdid: f2d9c7cc-0b02-594d-84b6-2b8e344a621f, req: LED:0, req_len: 5
Send 35 Bytes
socket receive data
socket receive data
Tips: Send Ok
[EDP]send heart
socket receive data
Tips: HeartBeat OK
[EDP]send heart
socket receive data
Tips: HeartBeat OK
[EDP]send heart
socket receive data
Tips: HeartBeat OK
socket receive data
cmdid: 4fc4d0de-078e-5ce5-bae1-cc4fdff23ec8, req: LED:1, req_len: 5
Send 35 Bytes
socket receive data
socket receive data
Tips: Send Ok
socket receive data
cmdid: 5f56d146-0313-5814-9c62-f5836221aebd, req: LED:0, req_len: 5
Send 35 Bytes
socket receive data
socket receive data
Tips: Send Ok
[EDP]send heart
Send 65 Bytes
Humi_Temp Send success
socket receive data
Tips: HeartBeat OK
socket receive data
socket receive data
Tips: Send Ok
[EDP]send heart
socket receive data
Tips: HeartBeat OK
[EDP]send heart
socket receive data
Tips: HeartBeat OK
[EDP]send heart
socket receive data
Tips: HeartBeat OK
[EDP]send heart
socket receive data
Tips: HeartBeat OK
[EDP]send heart
socket receive data
Tips: HeartBeat OK
[EDP]send heart
Send 65 Bytes
Humi_Temp Send success
socket receive data
Tips: HeartBeat OK
socket receive data
socket receive data
Tips: Send Ok
[EDP]send heart
socket receive data
Tips: HeartBeat OK
socket receive data
cmdid: 6235c513-bee0-5a46-a5ef-9282377f352a, req: MOTOR:1, req_len: 7
Send 37 Bytes
socket receive data
socket receive data
Tips: Send Ok
socket receive data
cmdid: fcbb4818-eeb7-5cac-b122-41b6329d2c99, req: MOTOR:0, req_len: 7
Send 37 Bytes
socket receive data
socket receive data
Tips: Send Ok