当前位置:首页 > 嵌入式 > 嵌入式案例Show
[导读]为了更好的理解安卓的层次关系,本文在RK3399的安卓系统上增加LED灯的外设,并使用APP打开关闭LED灯。以这样一个最简单的实例,来演示从上层到底层的调用过程。首先从最底层的kernel层开始。 一、驱动开发 Kernel层就是要将LED硬件接入到系统,完成驱动的开发

为了更好的理解安卓的层次关系,本文在RK3399的安卓系统上增加LED灯的外设,并使用APP打开关闭LED灯。以这样一个最简单的实例,来演示从上层到底层的调用过程。首先从最底层的kernel层开始。


一、驱动开发

Kernel层就是要将LED硬件接入到系统,完成驱动的开发。Linux下的驱动是使用C语言进行开发的,可分为三类设备类型:字符设备,块设备,网络设备。每种类型的驱动都有他自有的驱动框架,学习驱动开发就是要熟悉各种驱动架构,并根据实际需求在框架内添加内容。LED的驱动我们选择最简单的杂项字符类设备驱动即可。

从原理图中可以得到两个GPIO:GPIO1_C7和GPIO1_D0,驱动三极管来使得LED灯亮灭。

图:led灯原理图


1)设备树文件(kernel/arch/arm64/boot/dts/rockchip/rk3399-nanopi4-common.dtsi)

test-leds{compatible = "test,leds";led1-work = <&gpio1 23 GPIO_ACTIVE_LOW>;led2-work = <&gpio1 24 GPIO_ACTIVE_LOW>;status = "okay";};

2)  驱动文件(kernel/drivers/gpio/gpio-testled.c)

#include <linux/errno.h>#include <linux/kernel.h>#include <linux/module.h>#include <linux/slab.h>#include <linux/input.h>#include <linux/init.h>#include <linux/serio.h>#include <linux/delay.h>#include <linux/clk.h>#include <linux/gpio.h>#include <linux/platform_device.h>#include <linux/miscdevice.h>#include <linux/of.h>#include <linux/of_device.h>#include <linux/of_address.h>#include <linux/of_platform.h>#include <linux/of_gpio.h>#include <asm/io.h>#include <asm/uaccess.h>#include <linux/cdev.h>
MODULE_AUTHOR("embeddedtech");MODULE_LICENSE("Dual BSD/GPL");
#define LEDCTRL_MAGIC 'k'#define LED1CTRL_ON_CMD _IO (LEDCTRL_MAGIC, 1)#define LED1CTRL_OFF_CMD _IO (LEDCTRL_MAGIC, 2)#define LED2CTRL_ON_CMD _IO (LEDCTRL_MAGIC, 3)#define LED2CTRL_OFF_CMD _IO (LEDCTRL_MAGIC, 4)
struct led_data {int led1_pin; //led1引脚int led2_pin; //led2引脚};

struct led_data led_info;/** Open the device; in fact, there's nothing to do here.*/int testled_open (struct inode *inode, struct file *filp){return 0;}

ssize_t testled_read(struct file *file, char __user *buff, size_t count, loff_t *offp){return 0;}

ssize_t testled_write(struct file *file, const char __user *buff, size_t count, loff_t *offp){return 0;}

static long testled_compat_ioctl(struct file *file, unsigned int cmd, unsigned long arg){ switch (cmd) { case LED1CTRL_ON_CMD: { gpio_direction_output(led_info.led1_pin,1); break; } case LED1CTRL_OFF_CMD: { gpio_direction_output(led_info.led1_pin,0); break; } case LED2CTRL_ON_CMD: { gpio_direction_output(led_info.led2_pin,1); break;} case LED2CTRL_OFF_CMD: {gpio_direction_output(led_info.led2_pin,0); break;} default: { printk("testled compat Default %d\n",cmd); break;}}return 0;}

long testled_ioctl(struct file *file, unsigned int cmd, unsigned long arg){switch (cmd) {case LED1CTRL_ON_CMD: {gpio_direction_output(led_info.led1_pin,1);break;}case LED1CTRL_OFF_CMD: {gpio_direction_output(led_info.led1_pin,0);break;}case LED2CTRL_ON_CMD: {gpio_direction_output(led_info.led2_pin,1);break;}case LED2CTRL_OFF_CMD: {gpio_direction_output(led_info.led2_pin,0);break;}default: {printk("testled Default %d\n",cmd);break;}}return 0;

}

static int testled_release(struct inode *node, struct file *file){return 0;}

/** Our various sub-devices.*//* Device 0 uses remap_pfn_range */static struct file_operations testled_remap_ops = {.owner = THIS_MODULE,.open = testled_open,.release = testled_release,.read = testled_read,.write = testled_write,.unlocked_ioctl = testled_ioctl,.compat_ioctl = testled_compat_ioctl,};

static struct miscdevice testled_misc = {.minor = MISC_DYNAMIC_MINOR,.name = "test-led",.fops = &testled_remap_ops,};/** Module housekeeping.*/

static int testled_probe(struct platform_device *pdev){int ret;struct device_node *led_node = pdev->dev.of_node;//enum of_gpio_flags work_flags;

ret = misc_register(&testled_misc);if(ret < 0){pr_err("testled_misc_register fails!!!\n");return ret;}

led_info.led1_pin = of_get_named_gpio(led_node, "led1-work", 0);if(!gpio_is_valid(led_info.led1_pin)){pr_err("led1 pin invaild: %d",led_info.led1_pin);ret = -ENODEV;goto probe_fail;}

ret = gpio_request(led_info.led1_pin, "led1_pin");if(ret < 0){pr_err("led1 pin request failed.");goto probe_fail;}

led_info.led2_pin = of_get_named_gpio(led_node, "led2-work", 0);if(!gpio_is_valid(led_info.led2_pin)){pr_err("led2 pin invaild: %d",led_info.led2_pin);ret = -ENODEV;goto probe_fail;}

ret = gpio_request(led_info.led2_pin, "led2_pin");if(ret < 0){pr_err("led2 pin request failed.");goto probe_fail;}
printk("led1 pin is: %d,led2 pin is: %d",led_info.led1_pin,led_info.led2_pin);return 0;

probe_fail:

return ret;}

static int testled_remove(struct platform_device *pdev){pr_err("testled_misc_remove!!!\n");misc_deregister(&testled_misc);return 0;}

static struct of_device_id testled_table[] = {{ .compatible = "test,leds",}, //Compatible node must match dts{ },};

static struct platform_driver testled_driver = {.probe = testled_probe,.remove = testled_remove,.driver = {.name = "test_leds",.owner = THIS_MODULE,.of_match_table = testled_table,}};

static int testled_init(void){int ret;ret = platform_driver_register(&testled_driver);if(ret < 0)pr_err("platform_register_driver fails!!!\n");pr_err("platform_register_driver succeeds :ret=%d!!!\n",ret);return ret;}



static void testled_cleanup(void){platform_driver_unregister(&testled_driver);}

module_init(testled_init);module_exit(testled_cleanup);

3)Makefile文件(kernel/drivers/gpio/Makefile)

obj-$(CONFIG_GPIO_TEST_LED) += gpio-testled.o

4) Kconfig文件 (kernel/drivers/gpio/Kconfig)

config GPIO_TEST_LEDbool "select test leds"default y

5)驱动配置文件(kernel/arch/arm64/configs/nanopi4_nougat_defconfig)

CONFIG_GPIO_TEST_LED=y

6)修改节点权限(device/rockchip/common/ueventd.rockchip.rc)

/dev/test_leds 0666 system system
设备驱动文件在与 dts匹配的关键字是 compatible 属性。即比较驱动文件中 of_device_id 结构体元素的 .compatible 成员变量和 dts 文件中 node 中 compatible 属性两个字符串,匹配成功之后会进行probe,驱动就被加载进内核了。Dts文件可以放入一些数据、属性等,驱动文件加载是可以来读取,这样如果更改硬件管脚等就不需要改动驱动文件,只改设备树文件即可,使得驱动移植和维护更加方便。
Makefile文件、Kconfig文件和驱动配置文件使得gpio-testled.c的驱动可以被编译进内核,简易调试也可以只在Makefile文件添加 obj-y += gpio-testled.o就行。
驱动加载成功的标志是在设备的dev/目录下生成了test-led的设备文件节点,后面的上层就是以此文件来调用led驱动的。


二、驱动测试

驱动完成后要测试一下驱动对外的接口函数是否正常,可以写一个简易的测试代码。
  1. 测试代码gpioleds_test.c

#include <stdio.h>#include <stdlib.h>#include <unistd.h>#include <fcntl.h>#include <linux/ioctl.h>
#define LEDCTRL_MAGIC 'k'#define LED1CTRL_ON_CMD _IO (LEDCTRL_MAGIC, 1)#define LED1CTRL_OFF_CMD _IO (LEDCTRL_MAGIC, 2)#define LED2CTRL_ON_CMD _IO (LEDCTRL_MAGIC, 3)#define LED2CTRL_OFF_CMD _IO (LEDCTRL_MAGIC, 4)

int main(){int i = 0;int dev_fd;dev_fd = open("/dev/test-led",O_RDWR | O_NONBLOCK);if ( dev_fd == -1 ) {printf("Cann't open file /dev/test-led\n");exit(1);}printf("Led1 on\n");ioctl (dev_fd, LED1CTRL_ON_CMD,0);getchar();ioctl (dev_fd, LED1CTRL_OFF_CMD,0);printf("led1 off\n");
printf("Led2 on\n");ioctl (dev_fd, LED2CTRL_ON_CMD,0);getchar();ioctl (dev_fd, LED2CTRL_OFF_CMD,0);printf("led2 off\n");
close(dev_fd);return 0;}

2)Android.mkLOCAL_PATH:= $(call my-dir)include $(CLEAR_VARS)

LOCAL_SRC_FILES:= \gpioleds_test.c

LOCAL_SHARED_LIBRARIES := \liblog \libc \libutils \libcrypto \libhardware \

LOCAL_MODULE := gpioleds_test

LOCAL_MODULE_TAGS :=

LOCAL_MODULE_PATH := $(LOCAL_PATH)

LOCAL_CFLAGS +=-D_FORTIFY_SOURCE=0

LOCAL_CFLAGS += -U_FORTIFY_SOURCE -D_FORTIFY_SOURCE=0

include $(BUILD_EXECUTABLE

3)编译成可执行文件

在安卓目录external/下新建gpioleds_test目录,将gpioleds_test.c和Android.mk拷贝进目录,编译此目录。
mmm external/gpioleds_test/

完成在目录下生成二进制文件gpioleds_test。拷贝进安卓设备。


4)测试

二进制文件拷贝进安卓设备后,赋予777权限,然后运行。
Chmod 777 /data/user/gpioleds_test./ data/user/gpioleds_test


扫码关注我们

看更多嵌入式案例



喜欢本篇内容请给我们点个再看

免责声明:本文内容由21ic获得授权后发布,版权归原作者所有,本平台仅提供信息存储服务。文章仅代表作者个人观点,不代表本平台立场,如有问题,请联系我们,谢谢!

本站声明: 本文章由作者或相关机构授权发布,目的在于传递更多信息,并不代表本站赞同其观点,本站亦不保证或承诺内容真实性等。需要转载请联系该专栏作者,如若文章内容侵犯您的权益,请及时联系本站删除。
换一批
延伸阅读

LED驱动电源的输入包括高压工频交流(即市电)、低压直流、高压直流、低压高频交流(如电子变压器的输出)等。

关键字: 驱动电源

在工业自动化蓬勃发展的当下,工业电机作为核心动力设备,其驱动电源的性能直接关系到整个系统的稳定性和可靠性。其中,反电动势抑制与过流保护是驱动电源设计中至关重要的两个环节,集成化方案的设计成为提升电机驱动性能的关键。

关键字: 工业电机 驱动电源

LED 驱动电源作为 LED 照明系统的 “心脏”,其稳定性直接决定了整个照明设备的使用寿命。然而,在实际应用中,LED 驱动电源易损坏的问题却十分常见,不仅增加了维护成本,还影响了用户体验。要解决这一问题,需从设计、生...

关键字: 驱动电源 照明系统 散热

根据LED驱动电源的公式,电感内电流波动大小和电感值成反比,输出纹波和输出电容值成反比。所以加大电感值和输出电容值可以减小纹波。

关键字: LED 设计 驱动电源

电动汽车(EV)作为新能源汽车的重要代表,正逐渐成为全球汽车产业的重要发展方向。电动汽车的核心技术之一是电机驱动控制系统,而绝缘栅双极型晶体管(IGBT)作为电机驱动系统中的关键元件,其性能直接影响到电动汽车的动力性能和...

关键字: 电动汽车 新能源 驱动电源

在现代城市建设中,街道及停车场照明作为基础设施的重要组成部分,其质量和效率直接关系到城市的公共安全、居民生活质量和能源利用效率。随着科技的进步,高亮度白光发光二极管(LED)因其独特的优势逐渐取代传统光源,成为大功率区域...

关键字: 发光二极管 驱动电源 LED

LED通用照明设计工程师会遇到许多挑战,如功率密度、功率因数校正(PFC)、空间受限和可靠性等。

关键字: LED 驱动电源 功率因数校正

在LED照明技术日益普及的今天,LED驱动电源的电磁干扰(EMI)问题成为了一个不可忽视的挑战。电磁干扰不仅会影响LED灯具的正常工作,还可能对周围电子设备造成不利影响,甚至引发系统故障。因此,采取有效的硬件措施来解决L...

关键字: LED照明技术 电磁干扰 驱动电源

开关电源具有效率高的特性,而且开关电源的变压器体积比串联稳压型电源的要小得多,电源电路比较整洁,整机重量也有所下降,所以,现在的LED驱动电源

关键字: LED 驱动电源 开关电源

LED驱动电源是把电源供应转换为特定的电压电流以驱动LED发光的电压转换器,通常情况下:LED驱动电源的输入包括高压工频交流(即市电)、低压直流、高压直流、低压高频交流(如电子变压器的输出)等。

关键字: LED 隧道灯 驱动电源
关闭