CPU:RK3288

系统:Android 5.1

功能:上层 app 控制 led 亮灭

开发板:Firefly RK3288

1、在dts文件中增加 led 设备

path:kernel/arch/arm/boot/dts/firefly-rk3288.dts

firefly-led{
//匹配内容
compatible = "firefly,led";
//led引脚及有效电平
led-work = <&gpio8 GPIO_A2 GPIO_ACTIVE_LOW>;
status = "okay";
};

2、led 驱动

path:kernel/drivers/firefly/leds.c

#include <linux/init.h>
#include <linux/kernel.h>
#include <linux/uaccess.h>
#include <linux/proc_fs.h>
#include <linux/module.h>
#include <linux/slab.h>
#include <linux/gpio.h>
#include <linux/miscdevice.h>
#include <linux/of_gpio.h>
#include <linux/of_platform.h>
#include <linux/platform_device.h>
#include <linux/of.h>
#include <linux/of_gpio.h> #define DEBUG #ifdef DEBUG
#define LED_DBG(fmt,arg...) do{\
printk("<LED-DEBUG> [%d]"fmt"\n", __LINE__, ##arg);\
}while()
#else
#define LED_DBG(fmt,arg...)
#endif struct led_data {
int work_pin; //led引脚
int work_status; //led状态
int work_enable; //led使能电平
}; struct led_data *led_info; static int led_drv_open(struct inode *inode, struct file *filp)
{
LED_DBG("--%s()--", __func__);
return ;
} static int led_drv_release(struct inode *inode, struct file *filp)
{
LED_DBG("--%s()--", __func__);
return ;
} static ssize_t led_drv_read(struct file *filp, char __user *buf, size_t len, loff_t *pos)
{
int ret; LED_DBG("--%s()--", __func__);
ret = copy_to_user(buf, &(led_info->work_status), len);
if(ret > )
{
LED_DBG("copy to user failed.");
return ret;
} return len;
} static ssize_t led_drv_write(struct file *filp, const char __user *buf, size_t len, loff_t *pos)
{
int on;
int ret; LED_DBG("--%s()--", __func__);
ret = copy_from_user(&on, buf, len);
if(ret > )
{
LED_DBG("copy from user failed.");
return ret;
} gpio_direction_output(led_info->work_pin, !on);
led_info->work_status = on; return ret;
} //上层没添加此函数api
static long led_drv_ioctl(struct file *filp, unsigned int cmd, unsigned long arg)
{
LED_DBG("--%s()--", __func__);
return ;
} struct file_operations led_fops = {
.owner = THIS_MODULE,
.open = led_drv_open,
.release = led_drv_release,
.write = led_drv_write,
.read = led_drv_read,
.unlocked_ioctl = led_drv_ioctl,
}; //注册为杂项设备
struct miscdevice led_dev =
{
.minor = MISC_DYNAMIC_MINOR,
.fops = &led_fops,
.name = "firefly_leds",
}; static int led_drv_probe(struct platform_device *pdev)
{
int ret = -;
enum of_gpio_flags work_flags;
struct device_node *led_node = pdev->dev.of_node; LED_DBG("--%s()--", __func__); led_info = kzalloc(sizeof(struct led_data), GFP_KERNEL);
if(led_info == NULL)
{
LED_DBG("Alloc GFP_KERNEL memory failed.");
return -ENOMEM;
} //从dts中获取led的引脚,和有效电平
led_info->work_pin = of_get_named_gpio_flags(led_node, "led-work", , &work_flags);
if(!gpio_is_valid(led_info->work_pin))
{
LED_DBG("Work pin is invaild.");
ret = -ENODEV;
goto dev_fail;
} led_info->work_enable = (work_flags == OF_GPIO_ACTIVE_LOW) ? : ; ret = gpio_request(led_info->work_pin, "work_led");
if(ret < )
{
LED_DBG("Work led request failed.");
goto dev_fail;
} gpio_direction_output(led_info->work_pin, !(led_info->work_enable));
led_info->work_status = ; misc_register(&led_dev); return ; request_fail:
gpio_free(led_info->work_pin);
dev_fail:
kfree(led_info); return ret;
} static int led_drv_remove(struct platform_device *pdev)
{
misc_deregister(&led_dev);
LED_DBG("--%s()--", __func__);
gpio_free(led_info->work_pin);
kfree(led_info);
} int led_drv_suspend(struct device *dev)
{
LED_DBG("--%s()--", __func__);
return ;
} int led_drv_resume(struct device *dev)
{
LED_DBG("--%s()--", __func__);
return ;
} //suspend和resume暂时没调试成功
const struct dev_pm_ops led_pm_ops = {
.suspend = led_drv_suspend,
.resume = led_drv_resume,
}; static const struct of_device_id of_gpio_leds_match[] = {
{.compatible = "firefly,led", },
{},
}; static struct platform_driver led_drv = {
.probe = led_drv_probe,
.remove = led_drv_remove,
.driver = {
.name = "leds",
.owner = THIS_MODULE,
.pm = &led_pm_ops,
.of_match_table = of_match_ptr(of_gpio_leds_match),
},
}; module_platform_driver(led_drv); MODULE_LICENSE("GPL");
MODULE_AUTHOR("AaronLee");

3、Makefile

path:kernel/drivers/firefly/Makefile

obj-$(CONFIG_FIREFLY_RK3288_LEDS)                += leds.o

4.Kconfig

path:kernel/drivers/firefly/Kconfig

menu "Firefly Device Options"

config FIREFLY_RK3288_LEDS
bool "select firefly rk3288 leds"
default y endmenu

5、defconfig文件中增加led

path:kernel/arch/arm/configs/firefly-rk3288_defconfig

CONFIG_FIREFLY_RK3288_LEDS=y

6、修改led节点的权限

path:device/rockchip/common/ueventd.rockchip.rc

/dev/firefly_leds            system        system

Android 从上层到底层-----hal层

Android 从上层到底层-----jni层

Android 从上层到底层-----app层

Android 从上层到底层-----kernel层的更多相关文章

  1. Android 从上层到底层-----app层

    CPU:RK3288 系统:Android 5.1 功能:上层 app 控制 led 亮灭 开发板:Firefly RK3288 MainActivity.java package com.aaron ...

  2. Android 从上层到底层-----jni层

    CPU:RK3288 系统:Android 5.1 功能:上层 app 控制 led 亮灭 开发板:Firefly RK3288 led_jni.h path:hardware/rockchip/fi ...

  3. Android 从上层到底层-----hal层

    CPU:RK3288 系统:Android 5.1 功能:上层 app 控制 led 亮灭 开发板:Firefly RK3288 led_hal.c path:hardware/rockchip/fi ...

  4. Linux、Android系统调用从上层到底层的调用路径浅析

    参考: https://blog.csdn.net/liuhangtiant/article/details/85149369 http://blog.sina.com.cn/s/blog_79433 ...

  5. Android HAL层与Linux Kernel层驱动开发简介

    近日稍微对Android中的驱动开发做了一些简要的了解. HAL:Hardware Abstract Layer 硬件抽象层,由于Linux Kernel需要遵循GPL开源协议,硬件厂商为了保护自己硬 ...

  6. Linux/Android——input系统之 kernel层 与 frameworks层交互 (五)【转】

    本文转载自:http://blog.csdn.net/jscese/article/details/42291149 之前的四篇博文记录的都是linux中的input体系相关的东西,最底层以我调试的u ...

  7. Android Camera 通过V4L2与kernel driver的完整交互过程

    http://blog.chinaunix.net/uid-26215986-id-3552456.html 原文地址:Android Camera 通过V4L2与kernel driver的完整交互 ...

  8. Android源码剖析之Framework层升级版(窗口、系统启动)

    本文来自http://blog.csdn.net/liuxian13183/ ,引用必须注明出处! 看本篇文章之前,建议先查看: Android源码剖析之Framework层基础版 前面讲了frame ...

  9. 创建Android守护进程(底层服务)【转】

    本文转载自:https://blog.csdn.net/myfriend0/article/details/80016739 创建Android守护进程(底层服务) 前言 Android底层服务,即运 ...

随机推荐

  1. 012——数组(十二) 数组range array_count_values array_pad array_product

    <?php /*range () 生成包含指定范围的数组,第三个参数是步长值 */ /*$arr=range(1,10,2); print_r($arr);//输出:Array ( [0] =& ...

  2. 【zzuli-1923】表达式求值

    题目描述 假设表达式定义为:1. 一个十进制的正整数 X 是一个表达式.2. 如果 X 和 Y 是 表达式,则 X+Y, X*Y 也是表达式; *优先级高于+.3. 如果 X 和 Y 是 表达式,则 ...

  3. log4cpp

    body, table{font-family: 微软雅黑; font-size: 13.5pt} table{border-collapse: collapse; border: solid gra ...

  4. C与C++基础知识补遗

    本随笔用来记载项目开发中遇到的以前没掌握的C/C++基础知识 void * buffer; 无类型指针,可以指向任何类型数据.ANSI标准规定无类型指针不能进行算法,而GNU规定无类型指针算法操作与c ...

  5. lcx端口转发 linux版

    一.端口转发使用 1.攻击机上执行以下命令 ./lcx -p1 -p2 -m 在本地监听3389端口,并将发送到本机3389端口的数据传递到本机2222端口 2.跳板机上执行以下命令 ./lcx -h ...

  6. 20165202 实验一 Java开发环境的熟悉

    一.实验内容及步骤 (一)使用JDK编译.运行简单的Java程序 建立"自己学号exp1"的目录 mkdir 20165202exp1 进入目录 cd 20165202exp1 在 ...

  7. 如何让PPT播放时仅电脑显示备注页,而投影仪不显示

    完全可以!第一步:在电脑上右键点击桌面选择属性,进入显示属性选着设置,点击2号屏(前提已连接投影仪或第2显示器),并且在“将WINDOS桌面扩展到改监视器上”(这个关键)前面打钩,且自主选择分辨率,应 ...

  8. Kotlin Reference (八) Classes and Objects

    most from reference 类 Kotlin的类的声明使用关键字class class Invoice { } 类声明由类名.类头(指定其类型参数,构造函数等)和类体组成,由大括号括起来. ...

  9. 关于Objective-C 2.0 的垃圾收集

      Objective-C 2.0最大的增强可能就是垃圾收集了(Garbage Collection).与“垃圾收集”对应的是传统的引用计数(Reference Count)内存管理形式. 使用了垃圾 ...

  10. 【pluginShare】依赖JQuery的弹出层封装

    目前正在做的项目,老大说打算一点点做成bootstrap那样,然后开源... 所以,趁现在不忙,先封装一些小插件,然后慢慢完善,扩大,开源... 因此博客也多了一个新的模块------分享 第一弹:a ...