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

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

3天内不再提示

一个通用嵌入式驱动层的代码实现

工程师进阶笔记 来源:工程师进阶笔记 作者:工程师进阶笔记 2022-11-16 11:31 次阅读

一、前言

STM32为例,打开网络上下载的例程或者是购买开发板自带的例程,都会发现应用层中会有stm32f10x.h或者stm32f10x_gpio.h,这些文件严格来时属于硬件层的,如果软件层出现这些文件会显得很乱。

使用过Linux的童鞋们肯定知道linux系统无法直接操作硬件层,打开linux或者rt_thread代码会发现代码中都会有device的源文件,没错,这就是驱动层。

二、实现原理

原理就是将硬件操作的接口全都放到驱动链表上,在驱动层实现device的open、read、write等操作。当然这样做也有弊端,就是驱动find的时候需要遍历一遍驱动链表,这样会增加代码运行时间。

三、代码实现

国际惯例,写代码先写头文件。rt_thread中使用的是双向链表,为了简单在这我只用单向链表。有兴趣的可以自行研究rt_thread

头文件接口:

本次只实现如下接口,device_open 和device_close等剩下的接口可以自行研究。这样就可以在应用层中只调用如下接口可实现:

/*
驱动注册
*/
int cola_device_register(cola_device_t *dev);


/*
驱动查找
*/
cola_device_t*cola_device_find(constchar*name);


/*
驱动读
*/
int cola_device_read(cola_device_t *dev, int pos, void *buffer, int size);


/*
驱动写
*/
int cola_device_write(cola_device_t *dev, int pos, const void *buffer, int size);


/*
驱动控制
*/
int cola_device_ctrl(cola_device_t *dev, int cmd, void *arg);;

头文件cola_device.h:

#ifndef_COLA_DEVICE_H_
#define _COLA_DEVICE_H_


enumLED_state
{
  LED_OFF,
  LED_ON,
  LED_TOGGLE,
};


typedefstructcola_devicecola_device_t;
structcola_device_ops
{
  int(*init)(cola_device_t*dev);
  int(*open)(cola_device_t*dev,intoflag);
  int(*close)(cola_device_t*dev);
  int(*read)(cola_device_t*dev,intpos,void*buffer,intsize);
  int(*write)(cola_device_t*dev,intpos,constvoid*buffer,intsize);
  int(*control)(cola_device_t*dev,intcmd,void*args);
};


structcola_device
{
  constchar*name;
  structcola_device_ops*dops;
  structcola_device*next;
};


/*
驱动注册
*/
int cola_device_register(cola_device_t *dev);


/*
驱动查找
*/
cola_device_t *cola_device_find(const char *name);


/*
驱动读
*/
int cola_device_read(cola_device_t *dev, int pos, void *buffer, int size);


/*
驱动写
*/
int cola_device_write(cola_device_t *dev, int pos, const void *buffer, int size);


/*
驱动控制
*/
int cola_device_ctrl(cola_device_t *dev, int cmd, void *arg);


#endif

源文件cola_device.c:

#include"cola_device.h"
#include
#include 


struct cola_device *device_list = NULL;


/*
查找任务是否存在
*/
static bool cola_device_is_exists( cola_device_t *dev )
{
  cola_device_t*cur=device_list;
  while(cur!=NULL)
  {
    if(strcmp(cur->name,dev->name)==0)
{
returntrue;
}
cur=cur->next;
}
returnfalse;
}


static int device_list_inster(cola_device_t *dev)
{
  cola_device_t *cur = device_list;
  
  if(NULL==device_list)
  {
    device_list=dev;
    dev->next=NULL;
  }
  else
  {
    while(NULL!=cur->next)
    {
        cur=cur->next;
    }
    
    cur->next=dev;
    dev->next=NULL;
  }
  return1;
}


/*
驱动注册
*/
int cola_device_register(cola_device_t *dev)
{
  if((NULL==dev)||(cola_device_is_exists(dev)))
  {
    return0;
  }
  if((NULL==dev->name)||(NULL==dev->dops))
  {
    return0;
  }
  returndevice_list_inster(dev);
}


/*
驱动查找
*/
cola_device_t*cola_device_find(constchar*name)
{
  cola_device_t*cur=device_list;
  while(cur!=NULL)
  {
    if(strcmp(cur->name,name)==0)
    {
      returncur;
    }
    cur=cur->next;
  }
  returnNULL;
}


/*
驱动读
*/
int cola_device_read(cola_device_t *dev, int pos, void *buffer, int size)
{
  if(dev)
  {
    if(dev->dops->read)
    {
      returndev->dops->read(dev,pos,buffer,size);
    }
  }
  return0;
}


/*
驱动写
*/
int cola_device_write(cola_device_t *dev, int pos, const void *buffer, int size)
{
  if(dev)
  {
    if(dev->dops->write)
    {
      returndev->dops->write(dev,pos,buffer,size);
    }
  }
  return0;
}


/*
驱动控制
*/
int cola_device_ctrl(cola_device_t *dev, int cmd, void *arg)
{
  if(dev)
  {
    if(dev->dops->control)
    {
      returndev->dops->control(dev,cmd,arg);
    }
  }
  return0;
}

硬件注册方式:以LED为例,初始化接口voidled_register(void),需要在初始化中调用。

#include"stm32f0xx.h"
#include"led.h"
#include"cola_device.h"
#definePORT_GREEN_LEDGPIOC
#define PIN_GREENLED GPIO_Pin_13             


/*LED亮、灭、变化*/
#defineLED_GREEN_OFF(PORT_GREEN_LED->BSRR=PIN_GREENLED)
#defineLED_GREEN_ON(PORT_GREEN_LED->BRR=PIN_GREENLED)
#define LED_GREEN_TOGGLE (PORT_GREEN_LED->ODR ^= PIN_GREENLED)


static cola_device_t led_dev;


static void led_gpio_init(void)
{
  GPIO_InitTypeDefGPIO_InitStructure;
  RCC_AHBPeriphClockCmd(RCC_AHBPeriph_GPIOC,ENABLE);
  GPIO_InitStructure.GPIO_Pin=PIN_GREENLED;
  GPIO_InitStructure.GPIO_Mode=GPIO_Mode_OUT;
  GPIO_InitStructure.GPIO_Speed=GPIO_Speed_50MHz;
  GPIO_InitStructure.GPIO_OType=GPIO_OType_PP;
  GPIO_InitStructure.GPIO_PuPd=GPIO_PuPd_NOPULL;
  GPIO_Init(PORT_GREEN_LED,&GPIO_InitStructure);
  LED_GREEN_OFF;
}


static int led_ctrl(cola_device_t *dev, int cmd, void *args)
{
  if(LED_TOGGLE==cmd)
  {
    LED_GREEN_TOGGLE;
  }
  else{
  }
  return1;
}


staticstructcola_device_opsops=
{
  .control=led_ctrl,
};


void led_register(void)
{
  led_gpio_init();
  led_dev.dops=&ops;
  led_dev.name="led";
  cola_device_register(&led_dev);
}

应用层app代码:

#include
#include"app.h"
#include"config.h"
#include"cola_device.h"
#include "cola_os.h"


statictask_ttimer_500ms;
static cola_device_t *app_led_dev;


//led每500ms状态改变一次
static void timer_500ms_cb(uint32_t event)
{
  cola_device_ctrl(app_led_dev,LED_TOGGLE,0);
}


void app_init(void)
{
  app_led_dev=cola_device_find("led");
  assert(app_led_dev);
  cola_timer_create(&timer_500ms,timer_500ms_cb);
  cola_timer_start(&timer_500ms,TIMER_ALWAYS,500);
}

这样 app.c 文件中就不需要调用 led.h 头文件了,rtt 就是这样实现的。

四、总结

这样就可以实现软硬件分层了,是不是非常好用!

审核编辑:郭婷

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

    关注

    5006

    文章

    18440

    浏览量

    292116
  • 代码
    +关注

    关注

    30

    文章

    4586

    浏览量

    67214

原文标题:一个通用的嵌入式驱动层,基本都是这么玩的

文章出处:【微信号:工程师进阶笔记,微信公众号:工程师进阶笔记】欢迎添加关注!文章转载请注明出处。

收藏 人收藏

    评论

    相关推荐

    嵌入式系统软硬件基础知识大全

    嵌入式系统是现代科技发展的重要分支,广泛应用于工业控制、消费电子、医疗设备、汽车电子等领域。本文试图全面解析嵌入式系统的软基础知识,以期让读者对这
    发表于 05-09 14:12

    如何成为嵌入式C语言高手?

    如何成为嵌入式C语言高手? 嵌入式系统是当今科技领域的核心,而C语言则是嵌入式系统开发中最常用的编程语言之。成为
    发表于 04-07 16:03

    如何成为嵌入式C语言高手?

    如何成为嵌入式C语言高手? 嵌入式系统是当今科技领域的核心,而C语言则是嵌入式系统开发中最常用的编程语言之。成为
    发表于 03-25 14:12

    嵌入式系统发展前景?

    设备、健康监测等领域有着广泛的应用前景。随着人们对健康的重视程度不断提高,嵌入式系统将更为深入地应用于医疗设备和健康护理中,实现个人健康管理的智能化和实时化。 汽车电子和自动驾驶是嵌入式系统的另
    发表于 02-22 14:09

    嵌入式软件开发应该掌握哪些知识?

    嵌入式软件开发应该掌握哪些知识? 、 什么是嵌入式软件? 嵌入式软件通常是指嵌入式系统中运行的软件,
    发表于 02-19 11:23

    嵌入式学习步骤

    嵌入式行业是涉及广泛领域的行业,嵌入式、物联网、人工智能、智能与科学、电子信息工程、通信工程、自动化工程、测控、计算机科学等专业在嵌入式
    发表于 02-02 15:24

    嵌入式硬件和软件哪个好?

    方案,要求理解嵌入式系统架构,有定的C语言基础,熟悉ARM、protel设计软件,有四板开发经验。 成为优秀的嵌入式硬件开发工程师需具备以下技能:由需求分析至总体方案、详细设计的规
    发表于 12-05 15:17

    什么是嵌入式Linux?

    什么是嵌入式Linux? 对于很多电气、电信、通信专业的同学来说,对口专业就业方向主要有软、硬件两方向。无论是对于学生还是就业而言,软硬件的开发学习,嵌入式物联网在近年来无疑是
    发表于 10-11 13:47

    从事嵌入式工作有哪些优势?

    。 二是工作流程相对固定。搞嵌入式系统的公司,都有自己的产品计划,按自己的节奏行事,所开发的产品通常是通用的,不会因客户的不同而修改,产品型号开发完了,往往还有较长
    发表于 10-08 15:05

    嵌入式Linux应用开发的完全手册

    是怎样和硬件发生作用的。 同样,对于想从硬件岗位转到软件岗位的人,对于想从传统单片机(比如51单片机)编程进·步学习“有操作系统的”嵌入式编程的人,需要找到学习的切入点:先掌握各
    发表于 09-25 07:12

    基于树莓派的嵌入式Linux开发教学

    + 虚拟机环境 + 交叉编译 + 驱动编程 + 应用编程 ”开发链可以深入理解和控制嵌入式 Linux 系统,但因其复杂性而更适合需要定制 Linux 内核与驱动的精英开发、精英教
    发表于 09-21 07:09

    嵌入式新手应该怎么学?

    的图形界面操作都要转化为命令传输给硬件的。 4、Linux设备驱动的架构要了解,结合原理图和Linux设备驱动相关书籍把每代码分析清楚。 上面只是
    发表于 09-08 10:22

    EmbeddedButton嵌入式按键驱动设计实现

    EmbeddedButton是一个轻量级简单易用的嵌入式按键驱动模块,可无限拓展按键,支持多连击、长按、短按长按等多种按键事件;该模块通过异步回调方式来简化程序结构,根据几个简单原则完成了整个代码逻辑的支撑。
    的头像 发表于 08-28 15:47 766次阅读
    EmbeddedButton<b class='flag-5'>嵌入式</b>按键<b class='flag-5'>驱动</b>设计<b class='flag-5'>实现</b>

    如何创建事件驱动嵌入图像

    本指南介绍如何编写事件驱动嵌入式系统代码。 本指南是系列相关指南中的第三本: •构建您的第一个嵌入式
    发表于 08-08 07:22

    教你一招——嵌入式C通用延时驱动的编写方法

    嵌入式C通用延时驱动编程中,免不了要用到软件延时。这一般通过循环语句实现。通过控制循环语句的循环次数,便可获得多种不同的延时时间。为了便于使用和提高程序
    的头像 发表于 07-05 10:27 644次阅读