测试环境:  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驱动编程模型的更多相关文章

  1. at91sam9x5 linux 4.1.0下使能蜂鸣器驱动

    测试环境:  CPU: AT91SAM9X35      Linux: Atmel提供的linux-at91-linux4sam_5.3 (Linux-4.1.0) 转载请注明: 凌云物网智科嵌入式实 ...

  2. 【Linux】CentOS7.0下安装JDK环境

    写在前面:此次试验是在CentOS7上面安装的,亲测成功. 所需工具:JDK1.8安装包,xftp 具体步骤: 1,首先使用xftp连接到自己的虚拟机,然后查看是否有"/usr/java/j ...

  3. Windows平台VC++ 6.0 下的网络编程学习 - 简单的测试winsock.h头文件

    最近学习数据结构和算法学得有点累了(貌似也没那么累...)...找了本网络编程翻了翻当做打一个小基础吧,打算一边继续学习数据结构一边也看看网络编程相关的... 简单的第一次尝试,就大致梳理一下看书+自 ...

  4. SpringCloud的应用发布(二)vmvare+linux,Centos7.0下发布应用

    一.运行环境 1.jdk下载安装 地址:http://www.oracle.com/technetwork/java/javase/downloads/index.html 检查是否有老版本jdk 如 ...

  5. 【Tomcat】CentOS7.0下安装多个Tomcat及其配置

    安装前所需环境 在开始安装Tomcat之前,需要安装环境JDK,并配置JAVA环境.如果不知道如何配置,可参考这篇博客:[Linux]CentOS7.0下安装JDK环境 Tomcat安装 Tomcat ...

  6. 梦织未来Windows驱动编程 第03课 驱动的编程规范

    最近根据梦织未来论坛的驱动教程学习了一下Windows下的驱动编程,做个笔记备忘.这是第03课<驱动的编程规范>. 驱动部分包括基本的驱动卸载函数.驱动打开关闭读取写入操作最简单的分发例程 ...

  7. linux下无线鼠标驱动执行流程

    操作系统: debian 7.4(linux 3.2.54) 硬件: 一个无线鼠标.一个有线鼠标.usb集线器. 从淘宝上花了15块钱买了个无线鼠标,很好奇它的驱动程序是如何执行的. 首先将usb集线 ...

  8. 很好的linux下GPIO驱动详解文章

    原文地址  http://blog.csdn.net/llxmedici/article/details/6282372 打算跟着友善之臂的<mini2440 linux移植开发指南>来做 ...

  9. linux下I2C驱动架构全面分析【转】

    本文转载自:http://blog.csdn.net/wangpengqi/article/details/17711165 I2C 概述 I2C是philips提出的外设总线. I2C只有两条线,一 ...

随机推荐

  1. 批量建立EXCHANGE邮件帐号建立三部曲

    第一步:从AD里导出用户名(可以基于OU),将输出的CSV的DN列删除,并去除可能的测试及其它用途用户名. csvde -f users-gz.csv -d "ou=MKT gz,dc=xm ...

  2. 有关TCP和UDP 粘包 消息保护边界

    http://www.cnblogs.com/lancidie/archive/2013/10/28/3392428.html 在socket网络程序中,TCP和UDP分别是面向连接和非面向连接的.因 ...

  3. Java 编程下 Eclipse 如何设置单行代码显示的最大宽度

    Eclipse 下一行代码的默认宽度是 80 , 稍长一点的一行代码就会自动换行,代码可读性较差,我们可以自己在 Eclipse 对代码宽度进行设置. 设置路径为:[Window]→[Preferen ...

  4. 【转】Android源代码编译命令m/mm/mmm/make分析--不错

    原文网址:http://blog.csdn.net/luoshengyang/article/details/19023609 在前文中,我们分析了Android编译环境的初始化过程.Android编 ...

  5. (转载)按行合并两个sql的查询结果

    (转载)http://blog.csdn.net/wxwstrue/article/details/6784774 Union all join 是平行合并 为水平连接 Union all 是垂直合并 ...

  6. [PeterDLax著泛函分析习题参考解答]第4章 Hahn-Bananch 定理的应用

    1. 证明: 若在 4.1 节中取 $S=\sed{\mbox{正整数}}$, $Y$ 是收敛数列构成的空间, $\ell$ 由 (14) 式定义, 则由 (4) 给出的 $p$ 和由 (11) 定义 ...

  7. Count Color POJ--2777

    Count Color Time Limit: 1000MS   Memory Limit: 65536K Total Submissions: 32217   Accepted: 9681 Desc ...

  8. DHU-1241 Oil Deposits

    Oil Deposits Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others)Tota ...

  9. arcGis引入Dll报无法嵌入互操作类型错误解决方法

    arcGis引入Dll报“无法嵌入互操作类型"ESRI.ARCGIS.Geometry.PointClass".请改用通用接口."   解决方法:设置引用DLL的“嵌入互 ...

  10. 理解wait notify的好例子

    import java.util.concurrent.TimeUnit; public class Example2 { /** * @param args */ public static voi ...