测试环境:  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. 如何在win下编译thunderbird

    最近突然想研究一下thunderbird的实现,于是在WIN2K3下对其进行了系列的编译,特将编译的一些心得与大家共享.其实编译过程已经非常简单了,本文以VC8 ( VISUAL STUDIO 200 ...

  2. Yii modules中layout文件的调用

    在YII中,如果我们使用了modules区分了前后台,那么在不同的modules中需要使用各自的layout文件,在使用中发现经常会调用不到modules中的layout,下面介绍一下如何才能正确的调 ...

  3. RMA Sales Order – Stuck with “Awaiting Return Disposition”

    RMA Sales Order – Stuck with "Awaiting Return Disposition"                     Action : (P ...

  4. 英语之路 zt

    各位为英语而郁闷的兄弟姐妹们: 自从考完GRE和TOEFL以后,心有所感,本想写点心得,但是因为太懒没写成.今日风雨如晦,心中又有所感,于是一舒笔墨,写下我学英语的方法.俺知道有很多兄弟姐妹们和曾经的 ...

  5. HDU-2562 奇偶位互换

    http://acm.hdu.edu.cn/showproblem.php?pid=2562 奇偶位互换 Time Limit: 3000/1000 MS (Java/Others)    Memor ...

  6. ManagerDay-2

    新工作开始了两个星期 基本没有什么产出,主要还是适应新岗位和学东西.作为一个由高级开发转初级PM的人,要学要接触的还有太多. 公司给我安排了一个刚刚起步的项目,可能也是我从业三年接触到的最大的一个项目 ...

  7. Linux Mono Asp.net 部署方案

    1.Jexus 国内的 官网:http://www.jexus.org 2.Apache 官网:http://mono-project.com/Mod_mono 3.Nginx 官网:http://m ...

  8. bzoj 1007 [HNOI2008]水平可见直线(单调栈)

    1007: [HNOI2008]水平可见直线 Time Limit: 1 Sec  Memory Limit: 162 MBSubmit: 5120  Solved: 1899[Submit][Sta ...

  9. 使用DNSAgent拦截特定域名

    开发程序时,为方便测试,需要把本来发往abc.com的数据发到本地. 最简单的方法是直接在程序中修改,把abc.com修改为需要的地址. 但这样提交代码时,容易把调试地址给提交到服务器. 或是嵌入式设 ...

  10. 【转】shell 教程——07 Shell特殊变量:Shell $0, $#, $*, $@, $?, $$和命令行参数

    前面已经讲到,变量名只能包含数字.字母和下划线,因为某些包含其他字符的变量有特殊含义,这样的变量被称为特殊变量. 例如,$ 表示当前Shell进程的ID,即pid,看下面的代码: $echo $$ 运 ...