at91sam9x5 linux 4.1.0下dts驱动编程模型
测试环境: CPU: AT91SAM9X35 Linux: Atmel提供的linux-at91-linux4sam_5.3 (Linux-4.1.0)
转载请注明: 凌云物网智科嵌入式实验室: http://iot-yun.com/ 郭文学<guowenxue@gmail.com>
下面的这个驱动文件at91_keyled.c在Atmel提供的linux-at91-linux4sam_5.3下实现了按键控制LED的亮灭过程,通过这个简单的驱动描述了基于DTS的驱动开发模型以及Linux内核里的GPIO相关的操作函数。
/*********************************************************************************
* Copyright: (C) 2016 Guo Wenxue<guowenxue@gmail.com>
* All rights reserved.
*
* Filename: at91_keyled.c
* Description: This is a sample driver for GPIO operation with DTS linux on at91,
* which willl turn led on when a button pressed.
*
* Version: 1.0.0(2016-6-29~)
* Author: Guo Wenxue <guowenxue@gmail.com>
* ChangeLog: 1, Release initial version on "Wed Jun 29 12:00:44 CST 2016"
*
*
* DTS Changes:
* add keyleds support in arch/arm/boot/dts/at91sam9x5cm.dtsi
*
* keyleds{
* compatible = "key-leds";
* gpios = <&pioB 18 GPIO_ACTIVE_LOW priv->pin_key=of_get_gpio(pdev->dev.of_node, 0);
* &pioB 16 GPIO_ACTIVE_LOW>; priv->pin_key=of_get_gpio(pdev->dev.of_node, 1);
* status = "okay";
* }
*
* 1wire_cm {
* ... ...
* ... ...
* }
*
********************************************************************************/ #include <linux/module.h>
#include <linux/moduleparam.h>
#include <linux/platform_device.h> #include <linux/of.h>
#include <linux/of_device.h>
#include <linux/of_gpio.h>
#include <linux/delay.h>
#include <linux/gpio.h>
#include <linux/interrupt.h> typedef struct keyled_priv_s
{
int pin_key;
int pin_led;
int led_status;
} keyled_priv_t; /*--- end of struct keyled_priv_s ---*/ static const struct of_device_id of_key_leds_match[] = {
{ .compatible = "key-leds", },
{},
};
MODULE_DEVICE_TABLE(of, of_key_leds_match); static irqreturn_t key_detect_interrupt(int irq, void *dev_id)
{
keyled_priv_t *priv = (keyled_priv_t *)dev_id; priv->led_status ^= ;
gpio_set_value(priv->pin_led, priv->led_status); return IRQ_HANDLED;
} static int at91_keyled_probe(struct platform_device *pdev)
{
int res;
keyled_priv_t *priv; printk(KERN_INFO "at91_keyled driver probe\n"); if( != of_gpio_count(pdev->dev.of_node) )
{
printk(KERN_ERR "keyled pins definition in dts invalid\n");
return -EINVAL;
} priv = devm_kzalloc(&pdev->dev, sizeof(*priv), GFP_KERNEL);
if(!priv)
return -ENOMEM; platform_set_drvdata(pdev, priv); priv->pin_key=of_get_gpio(pdev->dev.of_node, );
priv->pin_led=of_get_gpio(pdev->dev.of_node, ); if( gpio_is_valid(priv->pin_key) )
{
if( (res=devm_gpio_request(&pdev->dev, priv->pin_key, "keyled_key")) < )
{
dev_err(&pdev->dev, "can't request key gpio %d\n", priv->pin_key);
return res;
}
dev_info(&pdev->dev, "request key gpio %d ok\n", priv->pin_key); if( (res=gpio_direction_input(priv->pin_key)) < )
{
dev_err(&pdev->dev, "can't request input direction key gpio %d\n", priv->pin_key);
return res;
}
dev_info(&pdev->dev, "request input direction key gpio %d ok\n", priv->pin_key); printk(KERN_INFO "Key gpio current status: %d\n", gpio_get_value(priv->pin_key)); res = request_irq( gpio_to_irq(priv->pin_key), key_detect_interrupt, IRQF_TRIGGER_FALLING, "keyled", priv);
if( res )
{
dev_err(&pdev->dev, "can't request IRQ<%d> for key gpio %d\n", gpio_to_irq(priv->pin_key), priv->pin_key);
return -EBUSY;
}
dev_info(&pdev->dev, "request IRQ<%d> for key gpio %d ok\n", gpio_to_irq(priv->pin_key), priv->pin_key);
} if( gpio_is_valid(priv->pin_led) )
{
if( (res=devm_gpio_request(&pdev->dev, priv->pin_led, "keyled_led")) < )
{
dev_err(&pdev->dev, "can't request key gpio %d\n", priv->pin_led);
return res;
} if( (res=gpio_direction_output(priv->pin_led, )) < )
{
dev_err(&pdev->dev, "can't request output direction key gpio %d\n", priv->pin_led);
return res;
}
} return ;
} static int at91_keyled_remove(struct platform_device *pdev)
{
keyled_priv_t *priv = platform_get_drvdata(pdev); printk(KERN_INFO "at91_keyled driver remove\n"); devm_gpio_free(&pdev->dev, priv->pin_led);
devm_gpio_free(&pdev->dev, priv->pin_key); free_irq(gpio_to_irq(priv->pin_key), priv); devm_kfree(&pdev->dev, priv); return ;
} static struct platform_driver at91_keyled_driver = {
.probe = at91_keyled_probe,
.remove = at91_keyled_remove,
.driver = {
.name = "key-leds",
.of_match_table = of_key_leds_match,
},
}; module_platform_driver(at91_keyled_driver); MODULE_AUTHOR("guowenxue <guowenxue@gmail.com>");
MODULE_DESCRIPTION("AT91 Linux DTS GPIO driver for Key and LED");
MODULE_LICENSE("GPL");
MODULE_ALIAS("platform:key-leds");
at91sam9x5 linux 4.1.0下dts驱动编程模型的更多相关文章
- at91sam9x5 linux 4.1.0下使能蜂鸣器驱动
测试环境: CPU: AT91SAM9X35 Linux: Atmel提供的linux-at91-linux4sam_5.3 (Linux-4.1.0) 转载请注明: 凌云物网智科嵌入式实 ...
- 【Linux】CentOS7.0下安装JDK环境
写在前面:此次试验是在CentOS7上面安装的,亲测成功. 所需工具:JDK1.8安装包,xftp 具体步骤: 1,首先使用xftp连接到自己的虚拟机,然后查看是否有"/usr/java/j ...
- Windows平台VC++ 6.0 下的网络编程学习 - 简单的测试winsock.h头文件
最近学习数据结构和算法学得有点累了(貌似也没那么累...)...找了本网络编程翻了翻当做打一个小基础吧,打算一边继续学习数据结构一边也看看网络编程相关的... 简单的第一次尝试,就大致梳理一下看书+自 ...
- SpringCloud的应用发布(二)vmvare+linux,Centos7.0下发布应用
一.运行环境 1.jdk下载安装 地址:http://www.oracle.com/technetwork/java/javase/downloads/index.html 检查是否有老版本jdk 如 ...
- 【Tomcat】CentOS7.0下安装多个Tomcat及其配置
安装前所需环境 在开始安装Tomcat之前,需要安装环境JDK,并配置JAVA环境.如果不知道如何配置,可参考这篇博客:[Linux]CentOS7.0下安装JDK环境 Tomcat安装 Tomcat ...
- 梦织未来Windows驱动编程 第03课 驱动的编程规范
最近根据梦织未来论坛的驱动教程学习了一下Windows下的驱动编程,做个笔记备忘.这是第03课<驱动的编程规范>. 驱动部分包括基本的驱动卸载函数.驱动打开关闭读取写入操作最简单的分发例程 ...
- linux下无线鼠标驱动执行流程
操作系统: debian 7.4(linux 3.2.54) 硬件: 一个无线鼠标.一个有线鼠标.usb集线器. 从淘宝上花了15块钱买了个无线鼠标,很好奇它的驱动程序是如何执行的. 首先将usb集线 ...
- 很好的linux下GPIO驱动详解文章
原文地址 http://blog.csdn.net/llxmedici/article/details/6282372 打算跟着友善之臂的<mini2440 linux移植开发指南>来做 ...
- linux下I2C驱动架构全面分析【转】
本文转载自:http://blog.csdn.net/wangpengqi/article/details/17711165 I2C 概述 I2C是philips提出的外设总线. I2C只有两条线,一 ...
随机推荐
- Spring MVC注解冲突
SpringMVC+MyBatis - 7 Spring自动扫描注解类的冲突问题 http://www.blogjava.net/crazycy/archive/2014/07/12/415738.h ...
- 【网络流24题】No.9 方格取数问题 (二分图点权最大独立集)
[题意] 在一个有 m*n 个方格的棋盘中, 每个方格中有一个正整数. 现要从方格中取数, 使任意 2 个数所在方格没有公共边,且取出的数的总和最大.试设计一个满足要求的取数算法. 输入文件示例inp ...
- EMMC 简要介绍
一直想写一篇关于EMMC的文章,但是因为之前弄了很多PPT,所以一直提不起兴趣,索性直接把之前的一个介绍EMMC的PPT贴出来给大家看看,有什么问题可以直接跟帖,我会第一时间进行解答,谢谢
- 单片机 C 语言模块化编程
好的开始是成功的一半 通过上一章的学习,我想你已经掌握了如何在程序中释放CPU了.希望能够继续坚持下去.一个良好的开始是成功的一半.我们今天所做的一切都是为了在单片机编程上做的更好. 在谈论今天的主题 ...
- 启动redis出现Creating Server TCP listening socket *:6379: bind: No such file or directory
E:\redis>redis-server.exe redis.windows.conf [8564] 10 Oct 20:00:36.745 # Creating Server TCP lis ...
- C#语言的几个层次
接到一位前不久C#培训学员的来信,这位学员虽然以前功底欠缺,但学习劲头很足,在培训中成长很快.即便基本吃透<.NET框架(修订版)>还嫌不够过瘾,一心要成为高手中的高手.来信的目的是希望我 ...
- 关于O(n)算法
首先要明确一点,当数据规模达到百万时需用O(n)算法 如何实现O(n)算法,其实是对原有算法的一种改进 后者说是 原有算法+一点小性质=O(n)算法 下面我将举几个例子来说明这一点: 1.后缀数组中h ...
- BZOJ1345: [Baltic2007]序列问题Sequence
1345: [Baltic2007]序列问题Sequence Time Limit: 5 Sec Memory Limit: 162 MBSubmit: 556 Solved: 274[Submi ...
- ExtJS练手
Ext.onReady(function () { //给grid添加数据源 var store = new Ext.data.JsonStore({ root: 'rows', totalPrope ...
- log4j的使用方法
1.Log4j是什么? Log4j可以帮助调试(有时候debug是发挥不了作 用的)和分析 2.Log4j的概念 Log4j中有三个主要的组件,它们分别是 Logger.Appender和Layout ...