资料介绍
														 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次下载 | 免费
 
	                电子发烧友App
	            
	        
	        
          
        
        
	                    
                        
                        
                        
                        
                        

创作
发文章
发帖  
提问  
发资料
发视频
上传资料赚积分
           
            
            
                
            
评论