资料介绍
11.3 GPIO驱动程序实例
11.3.1 GPIO工作原理
FS2410开发板的S3C2410处理器具有117个多功能通用I/O(GPIO)端口管脚,包括GPIO 8个端口组,分别为GPA(23个输出端口)、GPB(11个输入/输出端口)、GPC(16个输入/输出端口)、GPD(16个输入/输出端口)、GPE(16个输入/输出端口)、GPF(8个输入/输出端口)、GPH(11个输入/输出端口)。根据各种系统设计的需求,通过软件方法可以将这些端口配置成具有相应功能(例如:外部中断或数据总线)的端口。
为了控制这些端口,S3C2410处理器为每个端口组分别提供几种相应的控制寄存器。其中最常用的有端口配置寄存器(GPACON ~ GPHCON)和端口数据寄存器(GPADAT ~ GPHDAT)。因为大部分I/O管脚可以提供多种功能,通过配置寄存器(PnCON)设定每个管脚用于何种目的。数据寄存器的每位将对应于某个管脚上的输入或输出。所以通过对数据寄存器(PnDAT)的位读写,可以进行对每个端口的输入或输出。
在此主要以发光二极管(LED)和蜂鸣器为例,讨论GPIO设备的驱动程序。它们的硬件驱动电路的原理图如图11.4所示。
图11.4 LED(左)和蜂鸣器(右)的驱动电路原理图
在图11.4中,可知使用S3C2410处理器的通用I/O口GPF4、GPF5、GPF6和GPF7分别直接驱动LED D12、D11、D10以及D9,而使用GPB0端口驱动蜂鸣器。4个LED分别在对应端口(GPF4~GPF7)为低电平时发亮,而蜂鸣器在GPB0为高电平时发声。这5个端口的数据流方向均为输出。
在表11.15中,详细描述了GPF的主要控制寄存器。GPB的相关寄存器的描述与此类似,具体可以参考S3C2410处理器数据手册。
表11.15 GPF端口(GPF0-GPF7)的主要控制寄存器
寄存器地址R/W功能初始值
GPFCON0x56000050R/W配置GPF端口组0x0
GPFDAT0x56000054R/WGPF端口的数据寄存器未定义
GPFUP0x56000058R/WGPF端口的取消上拉寄存器0x0
GPFCON位描述
GPF7[15:14]00 = 输入 01 = 输出 10 = EINT7 11 = 保留
GPF6[13:12]00 = 输入 01 = 输出 10 = EINT6 11 = 保留
GPF5[11:10]00 = 输入 01 = 输出 10 = EINT5 11 = 保留
GPF4[9:8]00 = 输入 01 = 输出 10 = EINT4 11 = 保留
GPF3[7:6]00 = 输入 01 = 输出 10 = EINT3 11 = 保留
GPF2[5:4]00 = 输入 01 = 输出 10 = EINT2 11 = 保留
GPF1[3:2]00 = 输入 01 = 输出 10 = EINT1 11 = 保留
GPF0[1:0]00 = 输入 01 = 输出 10 = EINT0 11 = 保留
GPFDAT位描述
GPF[7:0][7:0]每位对应于相应的端口,若端口用于输入,则可以通过相应的位读取数据;若端口用于输出,则可以通过相应的位输出数据;若端口用于其他功能,则其值无法确定。
GPFUP位描述
GPF[7:0][7:0]0:向相应端口管脚赋予上拉(pull-up)功能
1:取消上拉功能
为了驱动LED和蜂鸣器,首先通过端口配置寄存器将5个相应寄存器配置为输出模式。然后通过对端口数据寄存器的写操作,实现对每个GPIO设备的控制(发亮或发声)。在下一个小节中介绍的驱动程序中,s3c2410_gpio_cfgpin()函数和s3c2410_gpio_pullup()函数将进行对某个端口的配置,而s3c2410_gpio_setpin()函数实现向数据寄存器的某个端口的输出。
11.3.2 GPIO驱动程序
GPIO驱动程序代码如下所示:
/* gpio_drv.h */
#ifndef FS2410_GPIO_SET_H
#define FS2410_GPIO_SET_H
#include 《linux/ioctl.h》
#define GPIO_DEVICE_NAME “gpio”
#define GPIO_DEVICE_FILENAME “/dev/gpio”
#define LED_NUM 4
#define GPIO_IOCTL_MAGIC ‘G’
#define LED_D09_SWT _IOW(GPIO_IOCTL_MAGIC, 0, unsigned int)
#define LED_D10_SWT _IOW(GPIO_IOCTL_MAGIC, 1, unsigned int)
#define LED_D11_SWT _IOW(GPIO_IOCTL_MAGIC, 2, unsigned int)
#define LED_D12_SWT _IOW(GPIO_IOCTL_MAGIC, 3, unsigned int)
#define BEEP_SWT _IOW(GPIO_IOCTL_MAGIC, 4, unsigned int)
#define LED_SWT_ON 0
#define LED_SWT_OFF 1
#define BEEP_SWT_ON 1
#define BEEP_SWT_OFF 0
#endif /* FS2410_GPIO_SET_H */
/* gpio_drv.c */
#include 《linux/config.h》
#include 《linux/module.h》
#include 《linux/moduleparam.h》
#include 《linux/init.h》
#include 《linux/kernel.h》 /* printk() */
#include 《linux/slab.h》 /* kmalloc() */
#include 《linux/fs.h》 /* everything.。. */
#include 《linux/errno.h》 /* error codes */
#include 《linux/types.h》 /* size_t */
#include 《linux/mm.h》
#include 《linux/kdev_t.h》
#include 《linux/cdev.h》
#include 《linux/delay.h》
#include 《linux/device.h》
#include 《asm/io.h》
#include 《asm/uaccess.h》
#include 《asm/arch-s3c2410/regs-gpio.h》
#include “gpio_drv.h”
static int major = 0; /* 采用字符设备号的动态分配 */
module_param(major, int, 0); /* 以参数的方式可以指定设备的主设备号*/
void s3c2410_gpio_cfgpin(unsigned int pin, unsigned int function)
{ /* 对某个管脚进行配置(输入/输出/其他功能)*/
unsigned long base = S3C2410_GPIO_BASE(pin); /* 获得端口的组基地址*/
unsigned long shift = 1;
unsigned long mask = 0x03; /* 通常用配置寄存器的两位表示一个端口*/
unsigned long con;
unsigned long flags;
if (pin 《 S3C2410_GPIO_BANKB)
{
shift = 0;
mask = 0x01; /* 在GPA端口中用配置寄存器的一位表示一个端口*/
}
mask 《《= (S3C2410_GPIO_OFFSET(pin) 《《 shift);
local_irq_save(flags); /* 保存现场,保证下面一段是原子操作 */
con = __raw_readl(base + 0x00);
con &= ~mask;
con |= function;
__raw_writel(con, base + 0x00); /* 向配置寄存器写入新配置数据 */
local_irq_restore(flags); /* 恢复现场 */
}
void s3c2410_gpio_pullup(unsigned int pin, unsigned int to)
{ /* 配置上拉功能 */
unsigned long base = S3C2410_GPIO_BASE(pin); /* 获得端口的组基地址*/
unsigned long offs = S3C2410_GPIO_OFFSET(pin);/* 获得端口的组内偏移地址 */
unsigned long flags;
unsigned long up;
if (pin 《 S3C2410_GPIO_BANKB)
{
return;
}
local_irq_save(flags);
up = __raw_readl(base + 0x08);
up &= ~(1 《《 offs);
up |= to 《《 offs;
__raw_writel(up, base + 0x08); /* 向上拉功能寄存器写入新配置数据*/
local_irq_restore(flags);
}
void s3c2410_gpio_setpin(unsigned int pin, unsigned int to)
{ /* 向某个管脚进行输出 */
unsigned long base = S3C2410_GPIO_BASE(pin);
unsigned long offs = S3C2410_GPIO_OFFSET(pin);
unsigned long flags;
unsigned long dat;
local_irq_save(flags);
dat = __raw_readl(base + 0x04);
dat &= ~(1 《《 offs);
dat |= to 《《 offs;
__raw_writel(dat, base + 0x04); /* 向数据寄存器写入新数据*/
11.3.1 GPIO工作原理
FS2410开发板的S3C2410处理器具有117个多功能通用I/O(GPIO)端口管脚,包括GPIO 8个端口组,分别为GPA(23个输出端口)、GPB(11个输入/输出端口)、GPC(16个输入/输出端口)、GPD(16个输入/输出端口)、GPE(16个输入/输出端口)、GPF(8个输入/输出端口)、GPH(11个输入/输出端口)。根据各种系统设计的需求,通过软件方法可以将这些端口配置成具有相应功能(例如:外部中断或数据总线)的端口。
为了控制这些端口,S3C2410处理器为每个端口组分别提供几种相应的控制寄存器。其中最常用的有端口配置寄存器(GPACON ~ GPHCON)和端口数据寄存器(GPADAT ~ GPHDAT)。因为大部分I/O管脚可以提供多种功能,通过配置寄存器(PnCON)设定每个管脚用于何种目的。数据寄存器的每位将对应于某个管脚上的输入或输出。所以通过对数据寄存器(PnDAT)的位读写,可以进行对每个端口的输入或输出。
在此主要以发光二极管(LED)和蜂鸣器为例,讨论GPIO设备的驱动程序。它们的硬件驱动电路的原理图如图11.4所示。
图11.4 LED(左)和蜂鸣器(右)的驱动电路原理图
在图11.4中,可知使用S3C2410处理器的通用I/O口GPF4、GPF5、GPF6和GPF7分别直接驱动LED D12、D11、D10以及D9,而使用GPB0端口驱动蜂鸣器。4个LED分别在对应端口(GPF4~GPF7)为低电平时发亮,而蜂鸣器在GPB0为高电平时发声。这5个端口的数据流方向均为输出。
在表11.15中,详细描述了GPF的主要控制寄存器。GPB的相关寄存器的描述与此类似,具体可以参考S3C2410处理器数据手册。
表11.15 GPF端口(GPF0-GPF7)的主要控制寄存器
寄存器地址R/W功能初始值
GPFCON0x56000050R/W配置GPF端口组0x0
GPFDAT0x56000054R/WGPF端口的数据寄存器未定义
GPFUP0x56000058R/WGPF端口的取消上拉寄存器0x0
GPFCON位描述
GPF7[15:14]00 = 输入 01 = 输出 10 = EINT7 11 = 保留
GPF6[13:12]00 = 输入 01 = 输出 10 = EINT6 11 = 保留
GPF5[11:10]00 = 输入 01 = 输出 10 = EINT5 11 = 保留
GPF4[9:8]00 = 输入 01 = 输出 10 = EINT4 11 = 保留
GPF3[7:6]00 = 输入 01 = 输出 10 = EINT3 11 = 保留
GPF2[5:4]00 = 输入 01 = 输出 10 = EINT2 11 = 保留
GPF1[3:2]00 = 输入 01 = 输出 10 = EINT1 11 = 保留
GPF0[1:0]00 = 输入 01 = 输出 10 = EINT0 11 = 保留
GPFDAT位描述
GPF[7:0][7:0]每位对应于相应的端口,若端口用于输入,则可以通过相应的位读取数据;若端口用于输出,则可以通过相应的位输出数据;若端口用于其他功能,则其值无法确定。
GPFUP位描述
GPF[7:0][7:0]0:向相应端口管脚赋予上拉(pull-up)功能
1:取消上拉功能
为了驱动LED和蜂鸣器,首先通过端口配置寄存器将5个相应寄存器配置为输出模式。然后通过对端口数据寄存器的写操作,实现对每个GPIO设备的控制(发亮或发声)。在下一个小节中介绍的驱动程序中,s3c2410_gpio_cfgpin()函数和s3c2410_gpio_pullup()函数将进行对某个端口的配置,而s3c2410_gpio_setpin()函数实现向数据寄存器的某个端口的输出。
11.3.2 GPIO驱动程序
GPIO驱动程序代码如下所示:
/* gpio_drv.h */
#ifndef FS2410_GPIO_SET_H
#define FS2410_GPIO_SET_H
#include 《linux/ioctl.h》
#define GPIO_DEVICE_NAME “gpio”
#define GPIO_DEVICE_FILENAME “/dev/gpio”
#define LED_NUM 4
#define GPIO_IOCTL_MAGIC ‘G’
#define LED_D09_SWT _IOW(GPIO_IOCTL_MAGIC, 0, unsigned int)
#define LED_D10_SWT _IOW(GPIO_IOCTL_MAGIC, 1, unsigned int)
#define LED_D11_SWT _IOW(GPIO_IOCTL_MAGIC, 2, unsigned int)
#define LED_D12_SWT _IOW(GPIO_IOCTL_MAGIC, 3, unsigned int)
#define BEEP_SWT _IOW(GPIO_IOCTL_MAGIC, 4, unsigned int)
#define LED_SWT_ON 0
#define LED_SWT_OFF 1
#define BEEP_SWT_ON 1
#define BEEP_SWT_OFF 0
#endif /* FS2410_GPIO_SET_H */
/* gpio_drv.c */
#include 《linux/config.h》
#include 《linux/module.h》
#include 《linux/moduleparam.h》
#include 《linux/init.h》
#include 《linux/kernel.h》 /* printk() */
#include 《linux/slab.h》 /* kmalloc() */
#include 《linux/fs.h》 /* everything.。. */
#include 《linux/errno.h》 /* error codes */
#include 《linux/types.h》 /* size_t */
#include 《linux/mm.h》
#include 《linux/kdev_t.h》
#include 《linux/cdev.h》
#include 《linux/delay.h》
#include 《linux/device.h》
#include 《asm/io.h》
#include 《asm/uaccess.h》
#include 《asm/arch-s3c2410/regs-gpio.h》
#include “gpio_drv.h”
static int major = 0; /* 采用字符设备号的动态分配 */
module_param(major, int, 0); /* 以参数的方式可以指定设备的主设备号*/
void s3c2410_gpio_cfgpin(unsigned int pin, unsigned int function)
{ /* 对某个管脚进行配置(输入/输出/其他功能)*/
unsigned long base = S3C2410_GPIO_BASE(pin); /* 获得端口的组基地址*/
unsigned long shift = 1;
unsigned long mask = 0x03; /* 通常用配置寄存器的两位表示一个端口*/
unsigned long con;
unsigned long flags;
if (pin 《 S3C2410_GPIO_BANKB)
{
shift = 0;
mask = 0x01; /* 在GPA端口中用配置寄存器的一位表示一个端口*/
}
mask 《《= (S3C2410_GPIO_OFFSET(pin) 《《 shift);
local_irq_save(flags); /* 保存现场,保证下面一段是原子操作 */
con = __raw_readl(base + 0x00);
con &= ~mask;
con |= function;
__raw_writel(con, base + 0x00); /* 向配置寄存器写入新配置数据 */
local_irq_restore(flags); /* 恢复现场 */
}
void s3c2410_gpio_pullup(unsigned int pin, unsigned int to)
{ /* 配置上拉功能 */
unsigned long base = S3C2410_GPIO_BASE(pin); /* 获得端口的组基地址*/
unsigned long offs = S3C2410_GPIO_OFFSET(pin);/* 获得端口的组内偏移地址 */
unsigned long flags;
unsigned long up;
if (pin 《 S3C2410_GPIO_BANKB)
{
return;
}
local_irq_save(flags);
up = __raw_readl(base + 0x08);
up &= ~(1 《《 offs);
up |= to 《《 offs;
__raw_writel(up, base + 0x08); /* 向上拉功能寄存器写入新配置数据*/
local_irq_restore(flags);
}
void s3c2410_gpio_setpin(unsigned int pin, unsigned int to)
{ /* 向某个管脚进行输出 */
unsigned long base = S3C2410_GPIO_BASE(pin);
unsigned long offs = S3C2410_GPIO_OFFSET(pin);
unsigned long flags;
unsigned long dat;
local_irq_save(flags);
dat = __raw_readl(base + 0x04);
dat &= ~(1 《《 offs);
dat |= to 《《 offs;
__raw_writel(dat, base + 0x04); /* 向数据寄存器写入新数据*/
下载该资料的人也在下载
下载该资料的人还在阅读
更多 >
- 乐化液晶驱动程序下载 7次下载
- STM32的ADC驱动程序
- USB驱动程序
- ADP5589输入键盘和GPIO Linux驱动程序
- ADP5588 GPIO Linux驱动程序
- ADP5588输入键盘和GPIO Linux驱动程序
- Linux的LEDS GPIO驱动程序免费下载 3次下载
- AM3553X的GPIO驱动的设计和硬件的控制驱动程序 20次下载
- 嵌入式Linux设备按键驱动程序实例分析 2次下载
- 第9章 Linux驱动程序设计 3次下载
- 基于BF533的Linux网络驱动程序
- 基于嵌入式Linux的步进电机驱动程序设计
- Windows CE下GPIO驱动程序的设计与应用
- 嵌入式Linux网络驱动程序的开发及实现原理
- acer aspire 5570驱动程序下载
- CAN分析仪的驱动程序如何正确安装 964次阅读
- 怎么编写Framebuffer驱动程序 402次阅读
- 基于OpenHarmony编写GPIO平台驱动和应用程序 733次阅读
- 了解和使用无操作系统和平台驱动程序 1066次阅读
- DS18B20的C语言驱动程序 5346次阅读
- 米尔科技LINUX设备驱动程序教程 1965次阅读
- 简要分析Thread的通用GPIO设备驱动 1413次阅读
- 浅谈电脑驱动程序的工作原理 详解电脑驱动程序意义 2.9w次阅读
- 可动态安装的Linux设备驱动程序 957次阅读
- 8255A驱动程序 3192次阅读
- 8155驱动程序 3057次阅读
- 深入了解USB驱动之总线驱动程序 8685次阅读
- 树莓派上MAX7219的字符驱动程序编写 6901次阅读
- Xilinx设备的驱动程序 7967次阅读
- PCI驱动程序开发实例 6701次阅读
下载排行
本周
- 1TC358743XBG评估板参考手册
- 1.36 MB | 330次下载 | 免费
- 2开关电源基础知识
- 5.73 MB | 6次下载 | 免费
- 3100W短波放大电路图
- 0.05 MB | 4次下载 | 3 积分
- 4嵌入式linux-聊天程序设计
- 0.60 MB | 3次下载 | 免费
- 5基于FPGA的光纤通信系统的设计与实现
- 0.61 MB | 2次下载 | 免费
- 6基于FPGA的C8051F单片机开发板设计
- 0.70 MB | 2次下载 | 免费
- 751单片机窗帘控制器仿真程序
- 1.93 MB | 2次下载 | 免费
- 8基于51单片机的RGB调色灯程序仿真
- 0.86 MB | 2次下载 | 免费
本月
- 1OrCAD10.5下载OrCAD10.5中文版软件
- 0.00 MB | 234315次下载 | 免费
- 2555集成电路应用800例(新编版)
- 0.00 MB | 33564次下载 | 免费
- 3接口电路图大全
- 未知 | 30323次下载 | 免费
- 4开关电源设计实例指南
- 未知 | 21548次下载 | 免费
- 5电气工程师手册免费下载(新编第二版pdf电子书)
- 0.00 MB | 15349次下载 | 免费
- 6数字电路基础pdf(下载)
- 未知 | 13750次下载 | 免费
- 7电子制作实例集锦 下载
- 未知 | 8113次下载 | 免费
- 8《LED驱动电路设计》 温德尔著
- 0.00 MB | 6653次下载 | 免费
总榜
- 1matlab软件下载入口
- 未知 | 935054次下载 | 免费
- 2protel99se软件下载(可英文版转中文版)
- 78.1 MB | 537796次下载 | 免费
- 3MATLAB 7.1 下载 (含软件介绍)
- 未知 | 420026次下载 | 免费
- 4OrCAD10.5下载OrCAD10.5中文版软件
- 0.00 MB | 234315次下载 | 免费
- 5Altium DXP2002下载入口
- 未知 | 233046次下载 | 免费
- 6电路仿真软件multisim 10.0免费下载
- 340992 | 191185次下载 | 免费
- 7十天学会AVR单片机与C语言视频教程 下载
- 158M | 183278次下载 | 免费
- 8proe5.0野火版下载(中文版免费下载)
- 未知 | 138040次下载 | 免费
评论
查看更多