找到目标函数在arch/arm/mach-s5pv210/gpiolib.c ----> s5pv210_gpiolib_init
smdkc110_map_io();
s5pv210_gpiolib_init(); 这个函数就是gpiolib初始化的函数
S5PV210有很多个IO口(160个左右),这些IO口首先被分成N个端口(port group),然后每个端口中又包含了M个IO口。
比如GPA0是一个端口,里面包含了8个IO口,一般记作:GPA0_0(或GPA0.0)、GPA0_1。
内核中为每个GPIO分配了一个编号,编号是一个数字(譬如一共有160个IO时编号就可以从1到160连续分布),编号可以让程序很方便的去识别每一个GPIO。
在终端中显示如下:
gpiochip112(112 对应base, 即当前端口GPA1的基地址)。端口分GPA0,GPA1,GPB
(1) s5pv210_gpio_4bit是一个结构体数组,数组中包含了很多个struct s3c_gpio_chip类型的变量。
// kernel/arch/arm/mach-s5pv210/include/mach
#define S5PV210_GPIO_NEXT(__gpio) \
((__gpio##_START) + (__gpio##_NR) + CONFIG_S3C_GPIO_SPACE + 1)
#define S5PV210_GPIO_NEXT(__gpio) \
((S5PV210_GPIO_A0_START) + (S5PV210_GPI
O_A0_NR) + CONFIG_S3C_GPIO_SPACE + 1)
root@wwj:~/driver/kernel# vi .config
CONFIG_S3C_GPIO_SPACE=0
#define S5PV210_GPIO_NEXT(__gpio) \
((__gpio##_START) + (__gpio##_NR) + CONFIG_S3C_GPIO_SPACE + 1)
= 0 + 8 + 0 + 1 = 9 (9+4+0+1=14)
封装的一个GPIO端口的所有信息的chip结构体变量
挂接到内核gpiolib模块定义的一个gpio_desc数组中的某一个格子中(和前面两个不一样,这个不是三星工程师写的,这个是内核开发者写的; 驱动就是内核开发者写一部分,厂商驱动开发工程师写一部分)。经过分析,发现内部其实并没有做gpiolib的注册工作,而是还在做填充,填充的是每一个GPIO被设置成输入模式/输出模式的操作方法。
函数接口 | 函数的作用 |
---|---|
gpiochip_add() | 是框架开出来的接口,给厂商驱动工程师用,用于向内核注册我们的gpiolib |
gpio_request() | 是框架开出来的接口,给使用gpiolib来编写自己的驱动的驱动工程师用的,驱动中要想使用某一个gpio,就必须先调用gpio_request接口来向内核的gpiolib部分申请,得到允许后才可以去使用这个gpio |
gpio_free() | 对应gpio_request,用来释放申请后用完了的gpio |
gpio_request_one()/gpio_request_array() | 这两个是gpio_request的变种 |
gpiochip_is_requested() | 接口用来判断某一个gpio是否已经被申请了 |
gpio_direction_input()/gpio_direction_output() | 接口用来设置GPIO为输入/输出模式,注意该函数内部实际并没有对硬件进行操作,只是通过chip结构体变量的函数指针调用了将来SoC厂商的驱动工程师写的真正的操作硬件实现gpio设置成输出模式的那个函数 |
以上的接口属于一类,这些都是给写其他驱动并且用到了gpiolib的人使用的。剩下的还有另外一类函数,这类函数是gpiolib内部自己的一些功能实现的代码。
#include // module_init module_exit
#include // __init __exit
#include
#include
#include
#include
#include
#include
#include
#define GPIO_LED1 S5PV210_GPJ0(3)
#define GPIO_LED2 S5PV210_GPJ0(4)
#define GPIO_LED3 S5PV210_GPJ0(5)
#define X210_LED_OFF 1 // X210中LED是正极接电源,负极节GPIO
#define X210_LED_ON 0 // 所以1是灭,0是亮
static struct led_classdev mydev1; // 定义结构体变量
static struct led_classdev mydev2; // 定义结构体变量
static struct led_classdev mydev3; // 定义结构体变量
// 这个函数就是要去完成具体的硬件读写任务的
static void s5pv210_led1_set(struct led_classdev *led_cdev,
enum led_brightness value)
{
printk(KERN_INFO "s5pv210_led1_set\n");
// 在这里根据用户设置的值来操作硬件
// 用户设置的值就是value
if (value == LED_OFF)
{
gpio_set_value(GPIO_LED1, X210_LED_OFF);
}
else
{
gpio_set_value(GPIO_LED1, X210_LED_ON);
}
}
static void s5pv210_led2_set(struct led_classdev *led_cdev,
enum led_brightness value)
{
printk(KERN_INFO "s5pv2102_led_set\n");
if (value == LED_OFF)
{
gpio_set_value(GPIO_LED2, X210_LED_OFF);
}
else
{
gpio_set_value(GPIO_LED2, X210_LED_ON);
}
}
static void s5pv210_led3_set(struct led_classdev *led_cdev,
enum led_brightness value)
{
printk(KERN_INFO "s5pv210_led3_set\n");
if (value == LED_OFF)
{
gpio_set_value(GPIO_LED3, X210_LED_OFF);
}
else
{
gpio_set_value(GPIO_LED3, X210_LED_ON);
}
}
static int __init s5pv210_led_init(void)
{
// 用户insmod安装驱动模块时会调用该函数
// 该函数的主要任务就是去使用led驱动框架提供的设备注册函数来注册一个设备
int ret = -1;
// 在这里去申请驱动用到的各种资源,当前驱动中就是GPIO资源
if (gpio_request(GPIO_LED1, "led1_gpj0.3"))
{
printk(KERN_ERR "gpio_request failed\n");
}
else
{
// 设置为输出模式,并且默认输出1让LED灯灭
gpio_direction_output(GPIO_LED1, 1);
}
// led1
mydev1.name = "led1";
mydev1.brightness = 0;
mydev1.brightness_set = s5pv210_led1_set;
ret = led_classdev_register(NULL, &mydev1);
if (ret < 0) {
printk(KERN_ERR "led_classdev_register failed\n");
return ret;
}
// led2
mydev2.name = "led2";
mydev2.brightness = 0;
mydev2.brightness_set = s5pv210_led2_set;
ret = led_classdev_register(NULL, &mydev2);
if (ret < 0) {
printk(KERN_ERR "led_classdev_register failed\n");
return ret;
}
// led3
mydev3.name = "led3";
mydev3.brightness = 0;
mydev3.brightness_set = s5pv210_led3_set;
ret = led_classdev_register(NULL, &mydev3);
if (ret < 0) {
printk(KERN_ERR "led_classdev_register failed\n");
return ret;
}
return 0;
}
static void __exit s5pv210_led_exit(void)
{
led_classdev_unregister(&mydev1);
led_classdev_unregister(&mydev2);
led_classdev_unregister(&mydev3);
gpio_free(GPIO_LED1);
}
module_init(s5pv210_led_init);
module_exit(s5pv210_led_exit);
// MODULE_xxx这种宏作用是用来添加模块描述信息
MODULE_LICENSE("GPL"); // 描述模块的许可证
MODULE_AUTHOR("aston <[email protected]>"); // 描述模块的作者
MODULE_DESCRIPTION("s5pv210 led driver"); // 描述模块的介绍信息
MODULE_ALIAS("s5pv210_led"); // 描述模块的别名信息
linux中查看gpio使用情况的方法 |
内核中提供了虚拟文件系统debugfs,里面有一个gpio文件,提供了gpio的使用信息。
使用方法:mount -t debugfs debugfs /tmp,然后cat /tmp/gpio即可得到gpio的所有信息,使用完后umount /tmp卸载掉debugfs