当前位置:网站首页>LED的C语言应用程序
LED的C语言应用程序
2022-08-04 19:36:00 【随遇而安的dandelion】
引言
在本文中,用C语言编写一个LED灯的应用程序,对文章《基于HDF的LED驱动程序开发》(1)、(2)中开发的LED灯的驱动程序进行测试。
另外,在编写LED灯的应用程序时,我们会用到很多由HDF框架提供的API。为了便于查阅这这些API的用法,建议大家在阅读本文的同时,打开文章《HDF驱动框架的API》(1)、(2)、(3)。在这几篇文章中汇集了本文所用到的API。当然,你也可以直接去阅读这些API的源码和官方说明文档。
本文参考了小熊派BearPi-HM_Micro_Small开发板的部分教程。
https://gitee.com/bearpi/bearpi-hm_micro_small
在阅读本文之前,确保已阅读了《小熊派BearPi-HM_Micro_Small之Hello_World》。
一、编写源程序
首先,在路径applications/BearPi/BearPi-HM_Micro/samples/dandelion
下新建一个文件夹:my_led_app
。
然后,在文件夹my_led_app
中新建一个文件:my_led_app.c
,接下来我们在这个文件中编写源程序。
这个应用程序没有用户图形界面,只能以命令行的形式运行:./my_led 参数
,命令参数有三种取值:0、1、2,分别控制LED关闭、打开和状态切换。
在源程序my_led_app.c
中,除了一些头文件和宏定义之外,主要包括以下两个函数:
函数 | 描述 |
---|---|
int main(int argc, char **argv) | 应用程序的主函数 |
static int LedWriteRead(struct HdfIoService *serv, uint8_t sendData, uint8_t *recvData) | 向驱动程序发送数据,然后接收驱动程序的响应。 |
1.1 头文件、宏定义
#include <fcntl.h>
#include <sys/stat.h>
#include <sys/ioctl.h>
#include <unistd.h>
#include <stdio.h>
#include "hdf_log.h"
#include "hdf_sbuf.h"
#include "hdf_io_service_if.h"
#define LED_WRITE_READ 1 //应用程序下发给LED驱动的命令字。1:写;0:读。
#define LED_SERVICE "hdf_led" //LED驱动对外提供的服务的名称,与驱动配置中的名称一致。
1.2 主函数
int main(int argc, char **argv)
{
int i;
// 打印输出在终端中输入的命令行
for (i=0; i < argc; i++)
{
printf("\r\nArgument %d is %s.\r\n", i, argv[i]);
}
// 获取驱动提供的服务
struct HdfIoService *serv = HdfIoServiceBind(LED_SERVICE);
if (serv == NULL)
{
printf("fail to get service %s!\r\n", LED_SERVICE);
return HDF_FAILURE;
}
// 向驱动发送LED的控制数据,从驱动接收LED当前的状态
uint8_t replyData = 0;
LedWriteRead(serv, atoi(argv[1]), &replyData);
printf("\r\nGet reply is: %d\r\n", replyData);
// 回收资源
HdfIoServiceRecycle(serv);
// 退出程序
printf("exit");
return HDF_SUCCESS;
}
在main
函数中,使用到了HDF提供的接口函数:HdfIoServiceBind
、HdfIoServiceRecycle
,以及结构体HdfIoService
。
1.3 LedWriteRead函数
LedWriteRead
函数在主函数main
中被调用,用于与内核态的驱动程序交换数据。第1个参数serv
:指向驱动服务对象(HdfIoService
结构体)的指针;第2个参数sendData
:向驱动程序下发的数据;第3个参数recvData
:指向一个变量的指针,该变量用于存放从驱动函数收到的应答数据。
static int LedWriteRead(struct HdfIoService *serv, uint8_t sendData, uint8_t *recvData)
{
int ret = HDF_SUCCESS;
// 创建一个缓冲区,用于向驱动发送数据
struct HdfSBuf *data = HdfSBufObtainDefaultSize();
if (data == NULL)
{
printf("fail to obtain sbuf data!\r\n");
return HDF_FAILURE;
}
// 创建一个缓冲区,用于从驱动接收响应
struct HdfSBuf *reply = HdfSBufObtainDefaultSize();
if (reply == NULL)
{
printf("fail to obtain sbuf reply!\r\n");
ret = HDF_DEV_ERR_NO_MEMORY;
goto out;
}
// 把要下发给驱动的数据写入缓冲区
if (!HdfSbufWriteUint8(data, sendData))
{
printf("fail to write sbuf!\r\n");
ret = HDF_FAILURE;
goto out;
}
// 调用应用态的Dispatch函数与驱动交互
ret = serv->dispatcher->Dispatch(&serv->object, LED_WRITE_READ, data, reply);
if (ret != HDF_SUCCESS)
{
printf("fail to send service call!\r\n");
goto out;
}
// 从缓冲区中读取驱动的响应数据
if (!HdfSbufReadUint8(reply, recvData))
{
printf("fail to get service call reply!\r\n");
ret = HDF_ERR_INVALID_OBJECT;
goto out;
}
out:
// 回收缓冲区
HdfSBufRecycle(data);
HdfSBufRecycle(reply);
return ret;
}
在main
函数中,使用到了HDF提供的接口函数:HdfSBufObtainDefaultSize
、HdfSbufWriteUint8
、HdfSbufReadUint8
、HdfSBufRecycle
,以及结构体HdfSBuf
、HdfIoService
。
二、编写/修改编译脚本
1、在文件夹my_led_app
下新建一个文件:BUILD.gn
,在这个文件中编写源程序my_led_app.c
的编译脚本:
import("//build/lite/config/component/lite_component.gni")
HDF_FRAMEWORKS = "//drivers/framework"
executable("my_led_exe") {
output_name = "my_led"
sources = [
"my_led_app.c",
]
include_dirs = [
"$HDF_FRAMEWORKS/ability/sbuf/include",
"$HDF_FRAMEWORKS/core/shared/include",
"$HDF_FRAMEWORKS/core/host/include",
"$HDF_FRAMEWORKS/core/master/include",
"$HDF_FRAMEWORKS/include/core",
"$HDF_FRAMEWORKS/include/utils",
"$HDF_FRAMEWORKS/utils/include",
"$HDF_FRAMEWORKS/include/osal",
"//drivers/adapter/uhdf/posix/include",
"//third_party/bounds_checking_function/include",
"//base/hiviewdfx/hilog_lite/interfaces/native/innerkits",
]
deps = [
"//base/hiviewdfx/hilog_lite/frameworks/featured:hilog_shared",
"//drivers/adapter/uhdf/manager:hdf_core",
"//drivers/adapter/uhdf/posix:hdf_posix_osal",
]
}
executable("my_led_exe")
用于声明一个名为my_led_exe
的、类型为executable
的目标(target),指示编译构建子系统将源码my_led_app.c
编译成名为my_led
的可执行文件。
output_name
用于指定编译生成的可执行文件的名称。
sources
列表中是编译构建所需要的源码文件。
include_dirs
列表中是源码文件中引用的头文件的路径。
deps
列表中是编译构建时需要依赖的其他库。
2、修改文件夹dandelion
下的编译脚本文件:BUILD.gn
。
import("//build/lite/config/component/lite_component.gni")
lite_component("dandelion_app") {
features = [
"hello_world:hello_world_exe",
"my_led_app:my_led_app",
]
}
lite_component
是在build/lite/config/component/lite_component.gni
中定义的一个template,用于将上面那个名为my_led_exe
的target纳入到名为my_led_app
的编译构建目标里面。
三、编译、烧写
参考《小熊派BearPi-HM_Micro_Small之Hello_World》第五、六部分。
四、测试
待系统启动完毕后,在串口终端中依次执行以下命令:
./bin/my_led 1
./bin/my_led 0
./bin/my_led 2
将分别控制小熊派BearPi-HM_Micro_Small开发板上的LED点亮、熄灭、状态切换,串口终端终端输出入下图所示:
边栏推荐
- 译文推荐|Apache Pulsar 隔离系列(四):单集群隔离策略
- The list of Kubernetes - watch mechanism
- [Latest Information] 2 new regions will announce the registration time for the soft exam in the second half of 2022
- 宏定义小方法
- 正则表达式未完
- Use "green computing" technology to promote sustainable development of computing power
- 【Attention演变史】翻译模型seq2seq (第二弹)
- 切换node版本和切换npm源工具
- T+Cloud:构建新型生意社交网络和营销关系的“智公司”
- 密码学系列之:PEM和PKCS7,PKCS8,PKCS12
猜你喜欢
Openharmony code framework (2) the person that
[Sql刷题篇] 查询信息数据--Day1
机器学习之支持向量机实例,线性核函数 多项式核函数 RBF高斯核函数 sigmoid核函数
迪赛智慧数——其他图表(主题河流图):近年居民消费、储蓄、投资意愿
openharmony代码框架初识(2)
使用.NET简单实现一个Redis的高性能克隆版(二)
Dragoma(DMA)元宇宙系统开发
[Sql brush topic] Query information data--Day1
openharmony初体验(1)
Query the published version records of the APP Store
随机推荐
一些性能测试的要点
致-.-- -..- -
电脑一键重装系统后连不上远程了?教你设置的方法
如何给MySQL添加自定义语法 ?
[Awards for Essays] Autumn recruitment special training to create your exclusive product experience
Highlights of some performance tests
Zip4j使用
really time ntp服务启动命令
MMDetection 使用示例:从入门到出门
Switch node version and switch npm source tool
入门:人脸专集1 | 级联卷积神经网络用于人脸检测(文末福利)
完善的交叉编译环境记录 peta 生成的shell 脚本
SIGIR 2022 | 邻域建模Graph-Masked Transformer,显著提高CTR预测性能
JS手写JSON.stringify() (面试)
【ASP.NET Core】 中间件
Notepad++更改显示背景
按需视觉识别:愿景和初步方案
正畸MIA微种植体支抗技术中国10周年交流会在沈举办
SAP UI5 视图控制器 View Controller 的生命周期方法 - Lifecycle methods
宏定义小方法