色偷偷偷久久伊人大杳蕉,色爽交视频免费观看,欧美扒开腿做爽爽爽a片,欧美孕交alscan巨交xxx,日日碰狠狠躁久久躁蜜桃

書白的個人空間 http://m.54549.cn/space-uid-134862.html [收藏] [復制] [RSS]

博客

學習嵌入式-迅為4412開發(fā)板的GPIO是怎么操作的?

已有 1710 次閱讀2017-4-11 10:30 |個人分類:4412| 嵌入式開發(fā)板, 迅為開發(fā)板, 4412開發(fā)板

Exynos4412所有的GPIO都有固定的地址,為了方便操作這些GPIO,Linux內(nèi)核在gpio-exynos4.h里面定義了一些GPIO的宏

詳情了解:http://topeetboard.com 更多了解:https://arm-board.taobao.com

例如: 
#define EXYNOS4_GPA0(_nr) (EXYNOS4_GPIO_A0_START + (_nr)) 
#define EXYNOS4_GPA1(_nr) (EXYNOS4_GPIO_A1_START + (_nr)) 
#define EXYNOS4_GPB(_nr) (EXYNOS4_GPIO_B_START + (_nr)) 
..................................... 
#define EXYNOS4_GPY5(_nr) (EXYNOS4_GPIO_Y5_START + (_nr)) 
#define EXYNOS4_GPY6(_nr) (EXYNOS4_GPIO_Y6_START + (_nr)) 
#define EXYNOS4_GPZ(_nr) (EXYNOS4_GPIO_Z_START + (_nr)) 
這些宏就是把每個GPIO的地址做了一下封裝,它的好處就是方便我們使用并且根據(jù)宏的名字就能直觀的知道是在操作哪個GPIO。 
Linux內(nèi)核中關(guān)于GPIO的驅(qū)動在driver/gpio/gpio-exynos4.c文件里面,在這個文件中GPIO驅(qū)動初始化入口函數(shù)是exynos4_gpiolib_init,因為這個文件同時也支持4210的GPIO,所以開始先初始化了一些通用的GPIO(4412和4210都有的GPIO),代碼如下: 
chip = exynos4_gpio_common_4bit; 
nr_chips = ARRAY_SIZE(exynos4_gpio_common_4bit); 
for (i = 0; i < nr_chips; i++, chip++) { 
if (chip->config == NULL) 
chip->config = &gpio_cfg; 
if (chip->base == NULL) 
pr_err("No allocation of base address for [common gpio]"); 

samsung_gpiolib_add_4bit_chips(exynos4_gpio_common_4bit, nr_chips); 
變量exynos4_gpio_common_4bit是一個數(shù)組,定義了一些通用的GPIO,nr_chips是記錄的exynos4_gpio_common_4bit數(shù)組里面元素個數(shù)。 
首先使用for循環(huán)遍歷exynos4_gpio_common_4bit所有的元素,為每個元素的config結(jié)構(gòu)賦值: 
if (chip->config == NULL) 
chip->config = &gpio_cfg; 
gpio_cfg是類型為s3c_gpio_cfg的結(jié)構(gòu)體,這個結(jié)構(gòu)體的定義如下: 
struct s3c_gpio_cfg { 
unsigned int cfg_eint; 
s3c_gpio_pull_t (*get_pull)(struct s3c_gpio_chip *chip, unsigned offs); 
int (*set_pull)(struct s3c_gpio_chip *chip, unsigned offs, 
s3c_gpio_pull_t pull); 
unsigned (*get_config)(struct s3c_gpio_chip *chip, unsigned offs); 
int (*set_config)(struct s3c_gpio_chip *chip, unsigned offs, 
unsigned config); 
}; 
通過上面的代碼我們可以看到這個結(jié)構(gòu)體里主要是一些函數(shù)指針,get_pull是獲取GPIO的上拉狀態(tài),set_pull是設置GPIO上拉或下拉的,set_config是設置GPIO的工作模式,例如:輸出/輸入/其他功能。下面我們來看看gpio_cfg變量的定義,如下: 
static struct s3c_gpio_cfg gpio_cfg = { 
.set_config = s3c_gpio_setcfg_s3c64xx_4bit, 
.set_pull = s3c_gpio_setpull_exynos4, 
.get_pull = s3c_gpio_getpull_exynos4, 
}; 
通過上面的代碼,可以看到分別對gpio_cfg結(jié)構(gòu)的三個函數(shù)指針賦值,這三個函數(shù)的定義是在gpio-config.c里面實現(xiàn)的,這個文件在內(nèi)核源碼arch/arm/plat-samsung目錄下,這三個函數(shù)的作用就是根據(jù)傳進來的參數(shù),配置GPIO相應的寄存器,從而實現(xiàn)對GPIO的操作。 
然后我們回到gpio-exynos4.c,接著看下面的代碼,完成了exynos4_gpio_common_4bit中每個元素的config結(jié)構(gòu)賦值后,接著會調(diào)用函數(shù)samsung_gpiolib_add_4bit_chips(exynos4_gpio_common_4bit, 
nr_chips)來向系統(tǒng)注冊GPIO結(jié)構(gòu)體。代碼如下: 
void __init samsung_gpiolib_add_4bit_chips(struct s3c_gpio_chip *chip, 
int nr_chips) 

for (; nr_chips > 0; nr_chips--, chip++) { 
samsung_gpiolib_add_4bit(chip); 
s3c_gpiolib_add(chip); 


上面的代碼主要有兩個函數(shù)組成分別是samsung_gpiolib_add_4bit(chip)和s3c_gpiolib_add(chip),首先我們來看下samsung_gpiolib_add_4bit(chip)函數(shù)的實現(xiàn): 
void __init samsung_gpiolib_add_4bit(struct s3c_gpio_chip *chip) 

chip->chip.direction_input = samsung_gpiolib_4bit_input; 
chip->chip.direction_output = samsung_gpiolib_4bit_output; 
chip->pm = __gpio_pm(&s3c_gpio_pm_4bit); 

這個函數(shù)也是為函數(shù)指針賦值,direction_input是設置GPIO為輸入模式,direction_output是設置GPIO為輸出。 
s3c_gpiolib_add(chip)函數(shù)主要作用是給一些函數(shù)指針賦值,然后根據(jù)傳進來的參數(shù)把對應的GPIO的信息保存到gpio_desc結(jié)構(gòu)里,gpio_desc是內(nèi)核里面定義的一個全局變量,用來保存每個GPIO的信息。至此GPIO的驅(qū)動初始化就完成了,其他它主要完成的功能就是為每個GPIO的結(jié)構(gòu)體里面的函數(shù)指針賦值,最后把每個GPIO結(jié)構(gòu)信息保存到全局變量gpio_desc里面。
上面已經(jīng)完成了一些通用的GPIO驅(qū)動的初始化,我們在回到gpio-exynos4.c,下面是根據(jù)CPU的型號初始化CPU特定的GPIO了,代碼如下: 
/* Only 4210 GPIO part */ 
if (soc_is_exynos4210()) { 
chip = exynos4210_gpio_4bit; 
nr_chips = ARRAY_SIZE(exynos4210_gpio_4bit); 
for (i = 0; i < nr_chips; i++, chip++) { 
if (chip->config == NULL) 
chip->config = &gpio_cfg; 
if (chip->base == NULL) 
pr_err("No allocation of base address [4210 gpio]"); 

samsung_gpiolib_add_4bit_chips(exynos4210_gpio_4bit, nr_chips); 
} else { 
/* Only 4212/4412 GPIO part */ 
chip = exynos4212_gpio_4bit; 
nr_chips = ARRAY_SIZE(exynos4212_gpio_4bit); 
for (i = 0; i < nr_chips; i++, chip++) { 
if (chip->config == NULL) 
chip->config = &gpio_cfg; 
if (chip->base == NULL) 
pr_err("No allocation of base address [4212 gpio]"); 

samsung_gpiolib_add_4bit_chips(exynos4212_gpio_4bit, nr_chips); 

通過看上面的代碼,初始化過程與前面介紹的初始化通用GPIO原理是一樣的,這里我們不在詳細介紹。對所有GPIO的初始化完成以后內(nèi)核中的其他驅(qū)動模塊就可以方便的使用我們注冊到gpio_desc里面的GPIO了。內(nèi)核提供了幾個全局函數(shù)來操作這些GPIO: 
int gpio_request(unsigned gpio, const char *label) 
void gpio_free(unsigned gpio) 
int s3c_gpio_setpull(unsigned int pin, s3c_gpio_pull_t pull) 
int s3c_gpio_cfgpin(unsigned int pin, unsigned int config) 
int gpio_direction_input(unsigned gpio) 
int gpio_direction_output(unsigned gpio, int value) 
gpio_request函數(shù)是申請GPIO操作,根據(jù)傳遞進來的參數(shù)gpio,會去全局變量gpio_desc里面找到對應的GPIO結(jié)構(gòu),判斷desc的標志位flag有沒有被設置FLAG_REQUESTED,如果有設置說明其他地方在使用這個GPIO,程序返回-EBUSY錯誤,如果沒有設置就設置flags的標記為FLAG_REQUESTED。 
gpio_free函數(shù)是釋放GPIO操作,根據(jù)傳遞進來的參數(shù),在gpio_desc全局變量找到對應的GPIO結(jié)構(gòu),清除掉desc的flag標志變量的FLAG_REQUESTED位。 
s3c_gpio_setpull函數(shù)是設置GPIO的上拉或下拉的,變量pull的取值范圍如下定義: 
#define S3C_GPIO_PULL_NONE ((__force s3c_gpio_pull_t)0x00) 
#define S3C_GPIO_PULL_DOWN ((__force s3c_gpio_pull_t)0x01) 
#define S3C_GPIO_PULL_UP ((__force s3c_gpio_pull_t)0x02) 
S3C_GPIO_PULL_NONE是懸空 
S3C_GPIO_PULL_DOWN是下拉 
S3C_GPIO_PULL_UP是上拉 
s3c_gpio_cfgpin函數(shù)是設置GPIO的功能:輸入/輸出/其他功能,第二個參數(shù)config取值范圍如下: 
#define S3C_GPIO_INPUT (S3C_GPIO_SPECIAL(0)) 
#define S3C_GPIO_OUTPUT (S3C_GPIO_SPECIAL(1)) 
#define S3C_GPIO_SFN(x) (S3C_GPIO_SPECIAL(x)) 
S3C_GPIO_INPUT是輸入模式,S3C_GPIO_OUTPUT是輸出模式,S3C_GPIO_SFN(x)是其他模式,例如中斷模式等。 
gpio_direction_input函數(shù)設置GPIO是輸入功能。 
gpio_direction_output設置GPIO輸出,第二個參數(shù)value取值0或1,0代表輸出低電平,1代表輸出高電平。 
下面我們來看幾個GPIO操作的例子: 
if (gpio_request(EXYNOS4_GPX3(3), "MPU6050 INT")) 
printk(KERN_WARNING "MPU6050 INT(GPX3.3) Port request error!!!\n"); 
else{ 
s3c_gpio_setpull(EXYNOS4_GPX3(3), S3C_GPIO_PULL_NONE); 
s3c_gpio_cfgpin(EXYNOS4_GPX3(3), S3C_GPIO_SFN(0)); 
gpio_direction_input(EXYNOS4_GPX3(3)); 
gpio_free(EXYNOS4_GPX3(3)); 

上面的代碼是設置GPIO引腳GPX3_3為輸入模式,懸空。 
err = gpio_request_one(EXYNOS4_GPX0(0), GPIOF_IN, "mcp251x_INT"); 
if (err) { 
printk(KERN_ERR "failed to request mcp251x_INT\n"); 
return -1; 

s3c_gpio_cfgpin(EXYNOS4_GPX0(0), S3C_GPIO_SFN(0xf)); 
s3c_gpio_setpull(EXYNOS4_GPX0(0), S3C_GPIO_PULL_NONE); 
gpio_free(EXYNOS4_GPX0(0)); 
上面的代碼設置GPIO引腳GPX0_0為中斷模式。 
if(gpio_request(EXYNOS4_GPK1(0), "GPK1_0")) 

printk(KERN_ERR "failed to request GPK1_0 for " 
"USI control\n"); 
return err; 

gpio_direction_output(EXYNOS4_GPK1(0), 1); 
s3c_gpio_cfgpin(EXYNOS4_GPK1(0), S3C_GPIO_OUTPUT); 
gpio_free(EXYNOS4_GPK1(0)); 
上面的代碼設置GPIO引腳GPK1_0為輸出模式,并且輸出高電平。 
iTOP-4412的GPIO驅(qū)動就介紹到這里,大家有興趣的話可以去內(nèi)核里面詳細的查看一下整個驅(qū)動的詳細實現(xiàn)。

關(guān)于我們  -  服務條款  -  使用指南  -  站點地圖  -  友情鏈接  -  聯(lián)系我們
電子工程網(wǎng) © 版權(quán)所有   京ICP備16069177號 | 京公網(wǎng)安備11010502021702
返回頂部