应用实战精解系列(二十):RVB2601之YoC系统

简介: 应用实战精解系列(二十):RVB2601之YoC系统

编辑语:

芯片开放社区(OCC)面向广大开发者推出应用实战系列内容,通过分享开发者实战开发案例,总结应用开发经验,梳理开发中的常见问题及解决方案,为后续参与的开发者提供更多参考与借鉴。


关于YoC系统,我们已经介绍了其KV组件的使用,并成功移植到了RVB2601开发板。本期内容,我们将继续分享YoC的实战应用,带大家了解如何通过使用其AOS的事件标志组,避免定时器回调和w800接收回调相互阻塞的情形。

01 AOS事件标志组

停滞了一段时间,继续RVB2601创意应用的开发工作。本人的创意项目需要RVB2601连接TCP Server——这个Server是建立在LoRa网关设备上的,所以基本就是用RVB2601去做LoRa网关的无线式上位机。


工作流程大体是:RVB2601向网关发送轮询帧,网关转发给LoRa终端,终端则采集传感器数据并通过网关回传到RVB2601。因此,RVB2601既要WiFi联网,又要开启TCP应用,而且还需要接收TCP数据。


好在板子的SDK中有接收回调的注册接口,就是之前帖子已经提到过的w800_packet_input_cb_register(),回调的原型为:

typedef void (*net_data_input_cb_t)(int linkid, void *data, size_t len, char remote_ip[16], uint16_t remote_ports);


不过案例的业务逻辑设计了这样一种需求:板子每隔固定周期会发出轮询帧,且发出后等待回复数据一定时间(默认10s),如果超时则再发轮询帧,最多尝试3次(也是默认,可以调整),这个需求通过定时器回调来实现的。好巧不巧上一阶段的工作主要完成NTP功能,自以为对C time库了解的不错,所以本人设计这样“有坑版”的代码写法:

//w800 input callback function
void aita_w800in_cb(int linkid, void *data, size_t len, char remote_ip[16], uint16_t remote_ports) {
    uint8_t* d;
    d = (uint8_t *)data;
    printf("linkid: %d\n", linkid);
    printf("len: %d\n", len);
    printf("remote_ip: %s\n", remote_ip);
    printf("remote_port: %d\n", remote_ports);
    printf("data:%s\n", d);
    if(remote_ports == GW_PORT) {
    isTimeout = 0;
    memset(USR220resbuff, 0, USR220RES_LEN);
    for(int i=0; i<len; i++) {
    USR220resbuff[i] = d[i];
    }
    }
}
void aita_PollingFunc(void) {
    int i = 0;
    int cyc = 1;
    time_t nowtime, endtime;
    //connect to USR220 TCP Server
    int ret = -1;    
    ret = w800_connect_remote(0, NET_TYPE_TCP_CLIENT, GW_IP, GW_PORT);
    if(ret < 0) return;
    for(i=0; i<endian_count; i++) {
    aita_BuildCommandPack(i); //build polling pack
    cyc = repeat;             //repeat times when timeout
    isOneEndianDone = 0;
    while(cyc--) {
    w800_send_data(comm_pack, COMMPACK_LEN, 3000);
    isTimeout = 1;
    endtime = time(NULL) + timeout;
    while((nowtime=time(NULL)) < endtime) {
    if(!isTimeout) {
    cyc = 0;
    isOneEndianDone = 1;
    break;
    }
    }
    printf("usr220res: %s, len: %d\n", USR220resbuff, sizeof(USR220resbuff));
        }
    if(!isOneEndianDone) {
    printf("%s disconnected\n", EID[i]);
    }
    }
    //断开TCP 连接
    w800_close(0);
}


上述代码的逻辑也很简单,就是基于程序查询思想,发出一帧开始不断通过time(NULL)来获取本地时间,如果时间过去timeout秒,就表示超时了,而w800接收回调会将超时标志isTimeOut清零,帮助结束程序查询的循环——while((nowtime=time(NULL)) < endtime)。


逻辑看似完美,但是在下忽略了这是在实时系统的平台上,而且代码跑在定时器回调中,而网络接收回调抢不到CPU权,所以查询本地时间的循环会阻塞w800接收回调的出现。


当时,为了验证自己的想法,干脆将超时查询改为aos_msleep(timeout*1000),果然接收回调出现了,但是每一次轮询都要固定耗时timeout秒钟显然是不行的。因此,想到了引入事件机制,因为只是简单的条件获取,所以选用了事件标志组。YoC的事件标志组可以查看链接:


https://yoc.docs.t-head.cn/yocbook/Chapter3-AliOS/%E6%93%8D%E4%BD%9C%E7%B3%BB%E7%BB%9F%E6%8E%A5%E5%8F%A3/%E4%BA%8B%E4%BB%B6.html?spm=a2cl5.25411629.0.0.2348180foY12EN


首先定义事件标志组变量和初始化函数,初始化函数会在案例的board_yoc_init()中调用。下述代码中,标志定义为宏EVENT_FLAG_3,选用这个纯粹是从文档抄来也没必要改名字。

#define EVENT_FLAG_3 0x000000F0
aos_event_t usr220res_event;
void aita_InitEvent(void) {
    int ret = -1;
    ret = aos_event_new(&usr220res_event, 0);
    if (ret != 0) {
        printf("event create failed\n");
        return;
    }    
}


接着就是修改aita_PollingFunc()函数,它会在定时器回调中调用。

void aita_PollingFunc(void) {
    int i = 0;
    int cyc = 1;
    //connect to USR220(LoRa Gateway) TCP Server
    int ret = -1;    
    ret = w800_connect_remote(0, NET_TYPE_TCP_CLIENT, GW_IP, GW_PORT);
    if(ret < 0) return;
    //once polling request all endians
    for(i=0; i<endian_count; i++) {
    aita_BuildCommandPack(i);//build req pack in USR220 format 
    cyc = repeat;
    isOneEndianDone = 0;
    while(cyc--) {
    w800_send_data(comm_pack, COMMPACK_LEN, 3000);
    isTimeout = 1;
    uint32_t actl_flags;
    //wait for flag(0x000000F0) for timeout-seconds, if get the flag then clear the bit
    aos_event_get(&usr220res_event, EVENT_FLAG_3, AOS_EVENT_OR_CLEAR, &actl_flags, timeout*1000);
    if(!isTimeout) {
    cyc = 0;
    isOneEndianDone = 1;
    printf("usr220res: %s, len: %d\n", USR220resbuff, sizeof(USR220resbuff));
    }    
        }
    if(!isOneEndianDone) {
    printf("%s disconnected\n", EID[i]);
    }
    }
    //once polling done & disconnect to tcp server
    w800_close(0);
}


w800接收回调当然就是在收到回复数据包后,发出事件标志EVENT_FLAG_3。

void aita_w800in_cb(int linkid, void *data, size_t len, char remote_ip[16], uint16_t remote_ports) {
    uint8_t* d;
    d = (uint8_t *)data;
    printf("linkid: %d\n", linkid);
    printf("len: %d\n", len);
    printf("remote_ip: %s\n", remote_ip);
    printf("remote_port: %d\n", remote_ports);
    printf("data:%s\n", d);
    if(remote_ports == GW_PORT) {
    isTimeout = 0;
    memset(USR220resbuff, 0, USR220RES_LEN);
    for(int i=0; i<len; i++) {
    USR220resbuff[i] = d[i];
    }
    aos_event_set(&usr220res_event, EVENT_FLAG_3, AOS_EVENT_OR);    
    }
}




相关文章
|
6天前
|
人工智能 定位技术 SEO
我学 GEO 第 15 天:终于知道AI GEO该如何做?
我是暴走的莉莉酱,边旅行边研究AI GEO的数字游民。专注普通人如何提升“AI可见度”——让AI在回答用户问题时准确识别、理解并推荐你。不讲玄学,只做可测、可调、可持续的GEO实践。
421 125
|
8天前
|
机器学习/深度学习 人工智能 调度
🐴 HappyHorse 1.1 现已上线阿里云百炼!快来查收模型使用指南,现在调用享 6 折~
HappyHorse 1.1 是新一代视频生成大模型,全面升级动态表现力、角色一致性、指令遵循、视觉质感与音画协同能力。支持I2V/T2V/R2V三类生成,适配短剧、电商广告、品牌营销等场景,提供高质、流畅、可控的AI视频生产力。
713 5
🐴 HappyHorse 1.1 现已上线阿里云百炼!快来查收模型使用指南,现在调用享 6 折~
|
6天前
|
缓存 人工智能 运维
阿里云618百炼大模型Qwen3.7-Max功能、免费试用、订阅计费、配置接入详解
Qwen3.7-MAX是阿里云百炼平台推出的通义千问3.7系列旗舰大语言模型,专为智能体时代复杂任务打造,依托阿里云全域算力与自研技术,在逻辑推理、长文本处理、代码工程、长周期自主执行等领域达到行业顶尖水平。2026年618期间,该模型推出多重免费试用权益、按量计费5折、订阅套餐优惠等专属福利,覆盖个人开发者、团队与企业全场景需求,以下从核心功能、免费试用、订阅计费、配置接入四方面展开详细解析。
416 123
|
4天前
|
人工智能 自然语言处理 API
阿里云Token Plan团队版解析:功能、三档套餐与省钱订阅指南
阿里云百炼平台推出的Token Plan团队版,是面向企业与团队的AI大模型订阅服务,以Credits为统一计量单位,整合文本与图像生成模型,提供团队管理、数据安全、多工具兼容等核心能力,解决团队零散订阅AI服务的管理混乱、成本失控、数据安全等痛点。本文将从核心定位、套餐详情、计费规则、团队管理、工具兼容、便宜订阅技巧等方面,全面解析Token Plan团队版,帮助企业与团队高效、低成本地使用AI服务。
309 108
|
5天前
|
存储 人工智能 数据可视化
别再手动复制 Skill 了:多 Agent 时代的 Skill 管理方案
多 Agent 场景下 Skill 的统一管理与同步。
261 123
|
19天前
|
缓存 测试技术 API
Qwen 3.7 Plus 与 Max 实测:性价比与多模态能力差异解析(2026)
2026 年 6 月 1 日,阿里悄无声息地发布了 Qwen 3.7 Plus,距 Qwen 3.7 Max 上线刚好 11 天。同样的 1M 上下文,同样的 35 小时自治上限。但价格才是头条:Plus 是 0.40/M输入,Max是 2.50/M——便宜约 6 倍——并且还能看图、看视频。Vision Arena 上 Plus 已经排到 #16。所以这周真正值得讨论的问题不是”要不要为视觉能力买单”,而是”Max 凭什么用 6 倍价格换来 2 个百分点的 benchmark 领先”。
|
12天前
|
缓存 人工智能 运维
GLM 5.2自托管全流程实战:硬件选型、vLLM/SGLang部署与成本盈亏测算
2026年智谱发布GLM 5.2超大混合专家模型,区别于以往仅开放API的闭源大模型,该模型权重以MIT开源协议对外发布,企业与开发者可完整下载、本地审计、私有化部署,实现数据不出环境、自定义微调、自主调度推理资源。GLM 5.2拥有753B总参数,原生支持百万级上下文窗口,在代码生成、长文档推理、数学逻辑等多项基准测试中对标国际顶尖商用模型,是首款可完整自托管的前沿代码向大模型。
939 0
|
13天前
|
Linux 程序员 数据格式
【2026最新】Notepad++下载、安装和使用一篇搞定(附中文版安装包)
Notepad++ 是一款免费开源、轻量高效的 Windows 文本编辑器,支持 C/Python/HTML 等 80+ 语言语法高亮、代码折叠、正则替换、编码转换及插件扩展,专为程序员与文本处理用户打造,完美替代系统记事本。(239字)