Android 从上层到底层-----kernel层
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层的更多相关文章
- Android 从上层到底层-----app层
CPU:RK3288 系统:Android 5.1 功能:上层 app 控制 led 亮灭 开发板:Firefly RK3288 MainActivity.java package com.aaron ...
- Android 从上层到底层-----jni层
CPU:RK3288 系统:Android 5.1 功能:上层 app 控制 led 亮灭 开发板:Firefly RK3288 led_jni.h path:hardware/rockchip/fi ...
- Android 从上层到底层-----hal层
CPU:RK3288 系统:Android 5.1 功能:上层 app 控制 led 亮灭 开发板:Firefly RK3288 led_hal.c path:hardware/rockchip/fi ...
- Linux、Android系统调用从上层到底层的调用路径浅析
参考: https://blog.csdn.net/liuhangtiant/article/details/85149369 http://blog.sina.com.cn/s/blog_79433 ...
- Android HAL层与Linux Kernel层驱动开发简介
近日稍微对Android中的驱动开发做了一些简要的了解. HAL:Hardware Abstract Layer 硬件抽象层,由于Linux Kernel需要遵循GPL开源协议,硬件厂商为了保护自己硬 ...
- Linux/Android——input系统之 kernel层 与 frameworks层交互 (五)【转】
本文转载自:http://blog.csdn.net/jscese/article/details/42291149 之前的四篇博文记录的都是linux中的input体系相关的东西,最底层以我调试的u ...
- Android Camera 通过V4L2与kernel driver的完整交互过程
http://blog.chinaunix.net/uid-26215986-id-3552456.html 原文地址:Android Camera 通过V4L2与kernel driver的完整交互 ...
- Android源码剖析之Framework层升级版(窗口、系统启动)
本文来自http://blog.csdn.net/liuxian13183/ ,引用必须注明出处! 看本篇文章之前,建议先查看: Android源码剖析之Framework层基础版 前面讲了frame ...
- 创建Android守护进程(底层服务)【转】
本文转载自:https://blog.csdn.net/myfriend0/article/details/80016739 创建Android守护进程(底层服务) 前言 Android底层服务,即运 ...
随机推荐
- 常用的SpringMVC注解
1.@RequestMapping() 访问链接编写: 例如: (1).请求方法: 访问链接: (2).请求参数和请求头: 访问链接: 2.@PathVariable 例如: 访问链接: 结果显示: ...
- NetHogs 实时检测网络流量 转
有很多适用于Linux系统的开源网络监视工具.比如说,你可以用命令iftop来检查带宽使用情况. netstat用来查看接口统计报告,还有top监控系统当前运行进程.但是如果你想要找一个能够按进程实时 ...
- Where do I belong
先给数组排序,然后找到指定的值在数组的位置,最后返回位置对应的索引. 举例:where([1,2,3,4], 1.5) 应该返回 1.因为1.5插入到数组[1,2,3,4]后变成[1,1.5,2,3, ...
- 030——VUE中鼠标语义修饰符
<!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8&quo ...
- jquery动态创建元素 div元素随垂直滚动条位置变化置顶显示
刚打开页面效果 拖动滑动条之后效果 页面代码 <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" & ...
- 【51nod-1009】数字1的数量
给定一个十进制正整数N,写下从1开始,到N的所有正数,计算出其中出现所有1的个数. 例如:n = 12,包含了5个1.1,10,12共包含3个1,11包含2个1,总共5个1. Input 输入N( ...
- LeetCode OJ:Partition List(分割链表)
Given a linked list and a value x, partition it such that all nodes less than x come before nodes gr ...
- 在Vim中使用gtags
之前一直使用vim+ctags+cscope来弄c的代码,最近看同事使用gtags,觉得在搜索方面要高级很多,网上大多都是emacs+gtags的资料,而vim的则比较少,这里搞通了之后,做个记录. ...
- jauery-layer弹出框的使用
一布局: <div id="detailLayer"> <div class="box-header"> <div class=& ...
- 跟着小程学微服务-Mock自动化系统的原理及实现
一.前言 在之前的文章 http://blog.csdn.net/u013970991/article/details/54862772 中已经介绍了"自动化Mock系统0.9版本" ...