日韩性视频-久久久蜜桃-www中文字幕-在线中文字幕av-亚洲欧美一区二区三区四区-撸久久-香蕉视频一区-久久无码精品丰满人妻-国产高潮av-激情福利社-日韩av网址大全-国产精品久久999-日本五十路在线-性欧美在线-久久99精品波多结衣一区-男女午夜免费视频-黑人极品ⅴideos精品欧美棵-人人妻人人澡人人爽精品欧美一区-日韩一区在线看-欧美a级在线免费观看

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 运维知识 > linux >内容正文

linux

linux内核中的GPIO系统之(2):pin control subsystem

發布時間:2025/4/16 linux 57 豆豆
生活随笔 收集整理的這篇文章主要介紹了 linux内核中的GPIO系统之(2):pin control subsystem 小編覺得挺不錯的,現在分享給大家,幫大家做個參考.

一、前言

在linux2.6內核上工作的嵌入式軟件工程師在pin control上都會遇到這樣的狀況:

(1)啟動一個新的項目后,需要根據硬件平臺的設定進行pin control相關的編碼。例如:在bootloader中建立一個大的table,描述各個引腳的配置和缺省狀態。此外,由于SOC的引腳是可以復用的,因此在各個具體的driver中,也可能會對引腳進行的配置。這些工作都是比較繁瑣的工作,需要極大的耐心和細致度。

(2)發現某個driver不能正常工作,辛辛苦苦debug后發現僅僅是因為其他的driver在初始化的過程中修改了引腳的配置,導致自己的driver無法正常工作

(3)即便是主CPU是一樣的項目,但是由于外設的不同,我們也不能使用一個kernel image,而是必須要修改代碼(這些代碼主要是board-specific startup code)

(4)代碼不是非常的整潔,cut-and-pasted代碼滿天飛,linux中的冗余代碼太多

作為一個嵌入式軟件工程師,項目做多了,接觸的CPU就多了,摔的跤就多了,之后自然會去思考,我們是否可以解決上面的問題呢?此外,對于基于ARM core那些SOC,雖然表面上看起來各個SOC各不相同,但是在pin control上還有很多相同的內容的,是否可以把它抽取出來,進行進一步的抽象呢?新版本中的內核(本文以3.14版本內核為例)提出了pin control subsystem來解決這些問題。

?

二、pin control subsystem的文件列表

1、源文件列表

我們整理linux/drivers/pinctrl目錄下pin control subsystem的源文件列表如下:

文件名描述
core.c core.hpin control subsystem的core driver
pinctrl-utils.c pinctrl-utils.hpin control subsystem的一些utility接口函數
pinmux.c pinmux.hpin control subsystem的core driver(pin muxing部分的代碼,也稱為pinmux driver)
pinconf.c pinconf.hpin control subsystem的core driver(pin config部分的代碼,也稱為pin config driver)
devicetree.c devicetree.hpin control subsystem的device tree代碼
pinctrl-xxxx.c各種pin controller的low level driver。

在pin controller driver文檔中 ,我們以2416的pin controller為例,描述了一個具體的low level的driver,這個driver涉及的文件包括pinctrl-samsung.c,pinctrl-samsung.h和pinctrl-s3c24xx.c。

2、和其他內核模塊接口頭文件

很多內核的其他模塊需要用到pin control subsystem的服務,這些頭文件就定義了pin control subsystem的外部接口以及相關的數據結構。我們整理linux/include/linux/pinctrl目錄下pin control subsystem的外部接口頭文件列表如下:

文件名描述
consumer.h其他的driver要使用pin control subsystem的下列接口:?
a、設置引腳復用功能?
b、配置引腳的電氣特性?
這時候需要include這個頭文件
devinfo.h這是for linux內核的驅動模型模塊(driver model)使用的接口。struct device中包括了一個struct dev_pin_info??? *pins的成員,這個成員描述了該設備的引腳的初始狀態信息,在probe之前,driver model中的core driver在調用driver的probe函數之前會先設定pin state
machine.h和machine模塊的接口。

?

3、Low level pin controller driver接口

我們整理linux/include/linux/pinctrl目錄下pin control subsystem提供給底層specific pin controller driver的頭文件列表如下:

文件名描述
pinconf-generic.h這個接口主要是提供給各種pin controller driver使用的,不是外部接口。
pinconf.hpin configuration 接口
pinctrl-state.hpin control state狀態定義
pinmux.hpin mux function接口

?

三、pin control subsystem的軟件框架圖

1、功能和接口概述

一般而言,學習復雜的軟件組件或者軟件模塊是一個痛苦的過程。我們可以把我們要學習的那個軟件block看成一個黑盒子,不論里面有多么復雜,第一步總是先了解其功能和外部接口特性。如果你愿意,你可以不去看其內部實現,先自己思考其內部邏輯,并形成若干問題,然后帶著這些問題去看代碼,往往事半功倍。

(1)、功能規格。pin control subsystem的主要功能包括:

(A)管理系統中所有可以控制的pin。在系統初始化的時候,枚舉所有可以控制的pin,并標識這些pin。

(B)管理這些pin的復用(Multiplexing)。對于SOC而言,其引腳除了配置成普通GPIO之外,若干個引腳還可以組成一個pin group,形成特定的功能。例如pin number是{ 0, 8, 16, 24 }這四個引腳組合形成一個pin group,提供SPI的功能。pin control subsystem要管理所有的pin group。

(C)配置這些pin的特性。例如配置該引腳上的pull-up/down電阻,配置drive strength等

(2)接口規格。linux內核的某個軟件組件必須放回到linux系統中才容易探討它的接口以及在系統中的位置,因此,在本章的第二節會基于系統block上描述各個pin control subsystem和其他內核模塊的接口。

(3)內部邏輯。要研究一個subsystem的內部邏輯,首先要打開黑盒子,細分模塊,然后針對每一個模塊進行功能分析、外部接口分析、內部邏輯分析。如果模塊還是比較大,難于掌握,那么就繼續細分,拆成子模塊,重復上面的分析過程。在本章的第三節中,我們打開pin control subsystem的黑盒子進行進一步的分析。

2、pin control subsystem在和其他linux內核模塊的接口關系圖如下圖所示:

pin control subsystem會向系統中的其他driver提供接口以便進行該driver的pin config和pin mux的設定,這部分的接口在第四章描述。理想的狀態是GPIO controll driver也只是象UART,SPI這樣driver一樣和pin control subsystem進行交互,但是,實際上由于各種源由(后文詳述),pin control subsystem和GPIO subsystem必須有交互,這部分的接口在第五章描述。第六章描述了Driver model和pin control subsystem的接口,第七章描述了為Pin control subsystem提供database支持的Device Tree和Machine driver的接口。

?

3、pin control subsystem內部block diagram

起始理解了接口部分內容,閱讀和解析pin control subsystem的內部邏輯已經很簡單,本文就不再分析了。

?

四、pin control subsystem向其他driver提供的接口

當你準備撰寫一個普通的linux driver(例如串口驅動)的時候,你期望pin control subsystem提供的接口是什么樣子的?簡單,當然最好是簡單的,最最好是沒有接口,當然這是可能的,具體請參考第六章的接口。

1、概述

普通driver調用pin control subsystem的主要目標是:

(1)設定該設備的功能復用。設定設備的功能復用需要了解兩個概念,一個是function,另外一個pin group。function是功能抽象,對應一個HW邏輯block,例如SPI0。雖然給定了具體的gunction name,我們并不能確定其使用的pins的情況。例如:為了設計靈活,芯片內部的SPI0的功能可能引出到pin group { A8, A7, A6, A5 },也可能引出到另外一個pin group{ G4, G3, G2, G1 },但毫無疑問,這兩個pin group不能同時active,畢竟芯片內部的SPI0的邏輯功能電路只有一個。 因此,只有給出function selector(所謂selector就是一個ID或者index)以及function的pin group selector才能進行function mux的設定。

(2)設定該device對應的那些pin的電氣特性。

此外,由于電源管理的要求,某個device可能處于某個電源管理狀態,例如idle或者sleep,這時候,屬于該device的所有的pin就會需要處于另外的狀態。綜合上述的需求,我們把定義了pin control state的概念,也就是說設備可能處于非常多的狀態中的一個,device driver可以切換設備處于的狀態。為了方便管理pin control state,我們又提出了一個pin control state holder的概念,用來管理一個設備的所有的pin control狀態。因此普通driver調用pin control subsystem的接口從邏輯上將主要是:

(1)獲取pin control state holder的句柄

(2)設定pin control狀態

(3)釋放pin control state holder的句柄

pin control state holder的定義如下:

struct pinctrl {?
??? struct list_head node;--系統中的所有device的pin control state holder被掛入到了一個全局鏈表中?
??? struct device *dev;---該pin control state holder對應的device?
??? struct list_head states;----該設備的所有的狀態被掛入到這個鏈表中?
??? struct pinctrl_state *state;---當前的pin control state?
??? struct list_head dt_maps;----mapping table?
??? struct kref users;------reference count?
};

系統中的每一個需要和pin control subsystem進行交互的設備在進行設定之前都需要首先獲取這個句柄。而屬于該設備的所有的狀態都是掛入到一個鏈表中,鏈表頭就是pin control state holder的states成員,一個state的定義如下:

struct pinctrl_state {?
??? struct list_head node;---掛入鏈表頭的節點?
??? const char *name;-----該state的名字?
??? struct list_head settings;---屬于該狀態的所有的settings?
};

一個pin state包含若干個setting,所有的settings被掛入一個鏈表中,鏈表頭就是pin state中的settings成員,定義如下:

struct pinctrl_setting {?
??? struct list_head node;?
??? enum pinctrl_map_type type;?
??? struct pinctrl_dev *pctldev;?
??? const char *dev_name;?
??? union {?
??????? struct pinctrl_setting_mux mux;?
??????? struct pinctrl_setting_configs configs;?
??? } data;?
};

當driver設定一個pin state的時候,pin control subsystem內部會遍歷該state的settings鏈表,將一個一個的setting進行設定。這些settings有各種類型,定義如下:

enum pinctrl_map_type {?
??? PIN_MAP_TYPE_INVALID,?
??? PIN_MAP_TYPE_DUMMY_STATE,?
??? PIN_MAP_TYPE_MUX_GROUP,---功能復用的setting?
??? PIN_MAP_TYPE_CONFIGS_PIN,----設定單一一個pin的電氣特性?
??? PIN_MAP_TYPE_CONFIGS_GROUP,----設定單pin group的電氣特性?
};

有pin mux相關的設定(PIN_MAP_TYPE_MUX_GROUP),定義如下:

struct pinctrl_setting_mux {?
??? unsigned group;--------該setting所對應的group selector?
??? unsigned func;---------該setting所對應的function selector?
};

有了function selector以及屬于該functiong的roup selector就可以進行該device和pin mux相關的設定了。設定電氣特性的settings定義如下:

struct pinctrl_map_configs {?
??? const char *group_or_pin;----該pin或者pin group的名字?
??? unsigned long *configs;----要設定的值的列表。這個值被用來寫入HW?
??? unsigned num_configs;----列表中值的個數?
};

?

2、具體的接口

(1)devm_pinctrl_get和pinctrl_get。devm_pinctrl_get是Resource managed版本的pinctrl_get,核心還是pinctrl_get函數。這兩個接口都是獲取設備(設備模型中的struct device)的pin control state holder(struct pinctrl)。pin control state holder不是靜態定義的,一般在第一次調用該函數的時候會動態創建。創建一個pin control state holder是一個大工程,我們分析一下這段代碼:

static struct pinctrl *create_pinctrl(struct device *dev)?
{

?? 分配pin control state holder占用的內存并初始化?
??? p = kzalloc(sizeof(*p), GFP_KERNEL);?
??? p->dev = dev;?
??? INIT_LIST_HEAD(&p->states);?
??? INIT_LIST_HEAD(&p->dt_maps);

mapping table這個database的建立也是動態的,當第一次調用pin control state holder的get函數的時候,就會通過調用pinctrl_dt_to_map來建立該device需要的mapping entry。具體請參考第七章。

??? ret = pinctrl_dt_to_map(p);

??? devname = dev_name(dev);

??? mutex_lock(&pinctrl_maps_mutex);?
??? for_each_maps(maps_node, i, map) {?
??????? /* Map must be for this device */?
??????? if (strcmp(map->dev_name, devname))?
??????????? continue;

??????? ret = add_setting(p, map);----分析一個mapping entry,把這個setting的代碼加入到holder中

??? }?
??? mutex_unlock(&pinctrl_maps_mutex);

??? kref_init(&p->users);

??? /* 把這個新增加的pin control state holder加入到全局鏈表中 */?
??? mutex_lock(&pinctrl_list_mutex);?
??? list_add_tail(&p->node, &pinctrl_list);?
??? mutex_unlock(&pinctrl_list_mutex);

??? return p;?
}

(2)devm_pinctrl_put和pinctrl_put。是(1)接口中的逆函數。devm_pinctrl_get和pinctrl_get獲取句柄的時候申請了很多資源,在devm_pinctrl_put和pinctrl_put可以釋放。需要注意的是多次調用get函數不會重復分配資源,只會reference count加一,在put中referrenct count減一,當count==0的時候才釋放該device的pin control state holder持有的所有資源。

(3)pinctrl_lookup_state。根據state name在pin control state holder找到對應的pin control state。具體的state是各個device自己定義的,不過pin control subsystem自己定義了一些標準的pin control state,定義在pinctrl-state.h文件中:

#define PINCTRL_STATE_DEFAULT "default"?
#define PINCTRL_STATE_IDLE "idle"?
#define PINCTRL_STATE_SLEEP "sleep"

(4)pinctrl_select_state。設定一個具體的pin control state接口。

?

五、和GPIO subsystem交互

1、為何pin control subsystem要和GPIO subsystem交互?

作為軟件工程師,我們期望的硬件設計應該如下圖所示:

GPIO的HW block應該和其他功能復用的block是對等關系的,它們共同輸入到一個復用器block,這個block的寄存器控制哪一個功能電路目前是active的。pin configuration是全局的,不論哪種功能是active的,都可以針對pin進行電氣特性的設定。這樣的架構下,上圖中紅色邊框的三個block是完全獨立的HW block,其控制寄存器在SOC datasheet中應該是分成三個章節描述,同時,這些block的寄存器應該分別處于不同的地址區間。

對于軟件工程師,我們可以讓pin control subsystem和GPIO subsystem完全獨立,各自進行初始化,各自映射自己的寄存器地址空間,對于pin control subsystem而言,GPIO和其他的HW block沒有什么不同,都是使用自己提供服務的一個軟件模塊而已。然而實際上SOC的設計并非總是向軟件工程師期望的那樣,有的SOC的設計框架圖如下:

這時候,GPIO block是alway active的,而紅色邊框的三個block是緊密的捆綁在一起,它們的寄存器占據了一個memory range(datasheet中用一個章節描述這三個block)。這時候,對于軟件工程師來說就有些糾結了,本來不屬于我的GPIO控制也被迫要參與進來。這時候,硬件寄存器的控制都是pin controller來處理,GPIO相關的操作都要經過pin controller driver,這時候,pin controller driver要作為GPIO driver的back-end出現。

2、具體的接口形態

(1)pinctrl_request_gpio。該接口主要用來申請GPIO。GPIO也是一種資源,使用前應該request,使用完畢后釋放。具體的代碼如下:

int pinctrl_request_gpio(unsigned gpio)----這里傳入的是GPIO 的ID?
{?
??? struct pinctrl_dev *pctldev;?
??? struct pinctrl_gpio_range *range;?
??? int ret;?
??? int pin;

??? ret = pinctrl_get_device_gpio_range(gpio, &pctldev, &range);---A?
??? if (ret) {?
??????? if (pinctrl_ready_for_gpio_range(gpio))?
??????????? ret = 0;?
??????? return ret;?
??? }

??? mutex_lock(&pctldev->mutex);??
??? pin = gpio_to_pin(range, gpio); ---將GPIO ID轉換成pin ID

??? ret = pinmux_request_gpio(pctldev, range, pin, gpio); ------B

??? mutex_unlock(&pctldev->mutex);

??? return ret;?
}

毫無疑問,申請GPIO資源本應該是GPIO subsystem的責任,但是由于上一節描述的源由,pin control subsystem提供了這樣一個接口函數供GPIO driver使用(其他的內核driver不應該調用,它們應該使用GPIO subsystem提供的接口)。多么丑陋的代碼,作為pin control subsystem,除了維護pin space中的ID,還要維護GPIO 的ID以及pin ID和GPIO ID的關系。

A:根據GPIO ID找到該ID對應的pin control device(struct pinctrl_dev)和GPIO rang(pinctrl_gpio_range)。在core driver中,每個low level的pin controller device都被映射成一個struct pinctrl_dev,并形成鏈表,鏈表頭就是pinctrldev_list。由于實際的硬件設計(例如GPIO block被分成若干個GPIO 的bank,每個bank就對應一個HW GPIO Controller Block),一個pin control device要管理的GPIO ID是分成區域的,每個區域用struct pinctrl_gpio_range來抽象,在low level 的pin controller初始化的時候(具體參考samsung_pinctrl_register的代碼),會調用pinctrl_add_gpio_range將每個GPIO bank表示的gpio range掛入到pin control device的range list中(gpio_ranges成員)。pinctrl_gpio_range 的定義如下:

struct pinctrl_gpio_range {?
??? struct list_head node;?
??? const char *name;?
??? unsigned int id;-----------GPIO chip ID?
??? unsigned int base;------該range中的起始GPIO IDD?
??? unsigned int pin_base;---在線性映射的情況下,這是起始的pin base?
??? unsigned const *pins;---在非線性映射的時候,這是table是pin到GPIO的lookup table?
??? unsigned int npins;----這個range有多少個GPIO引腳?
??? struct gpio_chip *gc;------每個GPIO bank都是一個gpio chip,對應一個GPIO range?
};

pin ID和GPIO ID有兩種映射關系,一種是線性映射(這時候pin_base有效),也就是說,對于這個GPIO range,GPIO base ID是a,pin ID base是b,那么a<--->b,a+1<--->b+1,a+2<--->b+2,以此類推。對于非線性映射(pin_base無效,pins是有效的),我們需要建立一個lookup table,以GPIO ID為索引,可以找到對于的pin ID。

B:這里主要是進行復用功能的設定,畢竟GPIO也是引腳的一個特定的功能。pinmux_request_gpio函數的作用主要有兩個,一個是在core driver中標記該pin已經用作GPIO了,這樣,如果有模塊后續request該資源,那么core driver可以拒絕不合理的要求。第二步就是調用底層pin controller driver的callback函數,進行底層寄存器相關的操作。

(2)pinctrl_free_gpio。有申請就有釋放,這是pinctrl_request_gpio的逆函數

(3)pinctrl_gpio_direction_input和pinctrl_gpio_direction_output。為已經指定為GPIO功能的引腳設定方向,輸入或者輸出。代碼很簡單,不再贅述。

?

六、和驅動模型的接口

前文已經表述過,最好是讓統一設備驅動模型(Driver model)來處理pin 的各種設定。與其自己寫代碼調用devm_pinctrl_get、pinctrl_lookup_state、pinctrl_select_state等pin control subsystem的接口函數,為了不讓linux內核自己的框架處理呢。本章將分析具體的代碼,這些代碼實例對自己driver調用pin control subsystem的接口函數來設定本device的pin control的相關設定也是有指導意義的。 linux kernel中的驅動模型提供了driver和device的綁定機制,一旦匹配會調用probe函數如下:

static int really_probe(struct device *dev, struct device_driver *drv)?
{?
??? ……?
??? ret = pinctrl_bind_pins(dev); ---對該device涉及的pin進行pin control相關設定?
??? ……

??? if (dev->bus->probe) {------下面是真正的probe過程?
??????? ret = dev->bus->probe(dev);?
??????? if (ret)?
??????????? goto probe_failed;?
??? } else if (drv->probe) {?
??????? ret = drv->probe(dev);?
??????? if (ret)?
??????????? goto probe_failed;?
??? }

……

}

pinctrl_bind_pins的代碼如下:

int pinctrl_bind_pins(struct device *dev)?
{?
??? int ret;

??? dev->pins = devm_kzalloc(dev, sizeof(*(dev->pins)), GFP_KERNEL);---(1)

??? dev->pins->p = devm_pinctrl_get(dev);-----------------(2)

??? dev->pins->default_state = pinctrl_lookup_state(dev->pins->p, -------(3)?
??????????????????? PINCTRL_STATE_DEFAULT);

??? ret = pinctrl_select_state(dev->pins->p, dev->pins->default_state); -----(4)


??? dev->pins->sleep_state = pinctrl_lookup_state(dev->pins->p, ------(3)?
??????????????????? PINCTRL_STATE_SLEEP);

??? dev->pins->idle_state = pinctrl_lookup_state(dev->pins->p, -------(3)?
??????????????????? PINCTRL_STATE_IDLE);

??? return 0;?
}

(1)struct device數據結構有一個pins的成員,它描述了和該設備相關的pin control的信息,定義如下:

struct dev_pin_info {?
??? struct pinctrl *p;------------該device對應的pin control state holder?
??? struct pinctrl_state *default_state;----缺省狀態?
??? struct pinctrl_state *sleep_state;-----電源管理相關的狀態?
??? struct pinctrl_state *idle_state;-----電源管理相關的狀態?
};

(2)調用devm_pinctrl_get獲取該device對應的 pin control state holder句柄。

(3)搜索default state,sleep state,idle state并記錄在本device中

(3)將該設備設定為pin default state

?

七、和device tree或者machine driver相關的接口

1、概述

device tree或者machine driver這兩個模塊主要是為 pin control subsystem提供pin mapping database的支持。這個database的每個entry用下面的數據結構表示:

struct pinctrl_map {?
??? const char *dev_name;---使用這個mapping entry的設備名?
??? const char *name;------該名字表示了該mapping entry?
??? enum pinctrl_map_type type;---這個entry的mapping type?
??? const char *ctrl_dev_name; -----pin controller這個設備的名字?
??? union {?
??????? struct pinctrl_map_mux mux;?
??????? struct pinctrl_map_configs configs;?
??? } data;?
};

?

2、通過machine driver靜態定義的數據來建立pin mapping database

machine driver定義一個巨大的mapping table,描述,然后在machine初始化的時候,調用pinctrl_register_mappings將該table注冊到pin control subsystem中。

3、通過device tree來建立pin mapping database

pin mapping信息定義在dts中,主要包括兩個部分,一個是定義在各個具體的device node中,另外一處是定義在pin controller的device node中。

一個典型的device tree中的外設node定義如下(建議先看看pin controller driver的第二章關于dts的描述):

device-node-name {??
??????? 定義該device自己的屬性??

??????? pinctrl-names = "sleep", "default";?
??????? pinctrl-0 =?;?
??????? pinctrl-1 =?;?????????
??? };

對普通device的dts分析在函數pinctrl_dt_to_map中,代碼如下:

int pinctrl_dt_to_map(struct pinctrl *p)?
{???
??? of_node_get(np);??
??? for (state = 0; ; state++) {-------------------(1)?
??????? /* Retrieve the pinctrl-* property */?
??????? propname = kasprintf(GFP_KERNEL, "pinctrl-%d", state);?
??????? prop = of_find_property(np, propname, &size);??
??????? kfree(propname);?
??????? if (!prop)?
??????????? break;?
??????? list = prop->value;?
??????? size /= sizeof(*list); --------------(2)

??????? /* Determine whether pinctrl-names property names the state */?
??????? ret = of_property_read_string_index(np, "pinctrl-names", ------(3)?
??????????????????????????? state, &statename);??

??????? if (ret < 0) {?
??????????? /* strlen("pinctrl-") == 8 */?
??????????? statename = prop->name + 8; -------------(4)?
??????? }

??????? /* For every referenced pin configuration node in it */?
??????? for (config = 0; config < size; config++) { -----------(5)?
??????????? phandle = be32_to_cpup(list++);

??????????? /* Look up the pin configuration node */?
??????????? np_config = of_find_node_by_phandle(phandle); ------(6)

??????????? /* Parse the node */?
??????????? ret = dt_to_map_one_config(p, statename, np_config); ----(7)?
??????????? of_node_put(np_config);?
??????????? if (ret < 0)?
??????????????? goto err;?
??????? }

??????? /* No entries in DT? Generate a dummy state table entry */?
??????? if (!size) {?
??????????? ret = dt_remember_dummy_state(p, statename); -------(8)?
??????????? if (ret < 0)?
??????????????? goto err;?
??????? }?
??? }

??? return 0;

err:?
??? pinctrl_dt_free_maps(p);?
??? return ret;?
}

(1)pinctrl-0 pinctrl-1 pinctrl-2……表示了該設備的一個個的狀態,這里我們定義了兩個pinctrl-0和pinctrl-1分別對應sleep和default狀態。這里每次循環分析一個pin state。

(2)代碼執行到這里,size和list分別保存了該pin state中所涉及pin configuration phandle的數目以及phandle的列表

(3)讀取從pinctrl-names屬性中獲取state name

(4)如果沒有定義pinctrl-names屬性,那么我們將pinctrl-0 pinctrl-1 pinctrl-2……中的那個ID取出來作為state name

(5)遍歷一個pin state中的pin configuration list,這里的pin configuration實際應該是pin controler device node中的sub node,用phandle標識。

(6)用phandle作為索引,在device tree中找他該phandle表示的那個pin configuration

(7)分析一個pin configuration,具體下面會仔細分析

(8)如果該設備沒有定義pin configuration,那么也要創建一個dummy的pin state。

這里我們已經進入對pin controller node下面的子節點的分析過程了。分析一個pin configuration的代碼如下:

static int dt_to_map_one_config(struct pinctrl *p, const char *statename,?
??????????????? struct device_node *np_config)?
{?
??? struct device_node *np_pctldev;?
??? struct pinctrl_dev *pctldev;?
??? const struct pinctrl_ops *ops;?
??? int ret;?
??? struct pinctrl_map *map;?
??? unsigned num_maps;

??? /* Find the pin controller containing np_config */?
??? np_pctldev = of_node_get(np_config);?
??? for (;;) {?
??????? np_pctldev = of_get_next_parent(np_pctldev);-------(1)?
??????? if (!np_pctldev || of_node_is_root(np_pctldev)) {?
??????????? of_node_put(np_pctldev);?
??????????? return -EPROBE_DEFER;?
??????? }?
??????? pctldev = get_pinctrl_dev_from_of_node(np_pctldev);-----(2)?
??????? if (pctldev)?
??????????? break;------------------------(3)?
??????? /* Do not defer probing of hogs (circular loop) */?
??????? if (np_pctldev == p->dev->of_node) {?
??????????? of_node_put(np_pctldev);?
??????????? return -ENODEV;?
??????? }?
??? }?
??? of_node_put(np_pctldev);

??? /*?
???? * Call pinctrl driver to parse device tree node, and?
???? * generate mapping table entries?
???? */?
??? ops = pctldev->desc->pctlops;?
??? ret = ops->dt_node_to_map(pctldev, np_config, &map, &num_maps);----(4)?
??? if (ret < 0)?
??????? return ret;

??? /* Stash the mapping table chunk away for later use */?
??? return dt_remember_or_free_map(p, statename, pctldev, map, num_maps);----(5)?
}

(1)首先找到該pin configuration node對應的parent node(也就是pin controler對應的node),如果找不到或者是root node,則進入出錯處理。

(2)獲取pin control class device

(3)一旦找到pin control class device則跳出for循環

(4)調用底層的callback函數處理pin configuration node。這也是合理的,畢竟很多的pin controller bindings是需要自己解析的。

(5)將該pin configuration node的mapping entry信息注冊到系統中

?

八、core driver和low level pin controller driver的接口規格

pin controller描述符。每一個特定的pin controller都用一個struct pinctrl_desc來抽象,具體如下:

struct pinctrl_desc {?
??? const char *name;?
??? struct pinctrl_pin_desc const *pins;?
??? unsigned int npins;?
??? const struct pinctrl_ops *pctlops;?
??? const struct pinmux_ops *pmxops;?
??? const struct pinconf_ops *confops;?
??? struct module *owner;?
};

pin controller描述符需要描述它可以控制多少個pin(成員npins),每一個pin的信息為何?(成員pins)。這兩個成員就確定了一個pin controller所能控制的引腳的信息。

pin controller描述符中包括了三類操作函數:pctlops是一些全局的控制函數,pmxops是復用引腳相關的操作函數,confops操作函數是用來配置引腳的特性(例如:pull-up/down)。struct pinctrl_ops中各個callback函數的具體的解釋如下:

callback函數描述
get_groups_count該pin controller支持多少個pin group。pin group的定義可以參考本文關于pin controller的功能規格中的描述。注意不要把pin group和IO port的硬件分組搞混了。例如:S3C2416有138個I/O 端口,分成11組,分別是gpa~gpl,這個組并不叫pin group,而是叫做pin bank。pin group是和特定功能(例如SPI、I2C)相關的一組pin。
get_group_name給定一個selector(index),獲取指定pin group的name
get_group_pins給定一個selector(index),獲取該pin group中pin的信息(該pin group包括多少個pin,每個pin的ID是什么)
pin_dbg_showdebug fs的callback接口
dt_node_to_map分析一個pin configuration node并把分析的結果保存成mapping table entry,每一個entry表示一個setting(一個功能復用設定,或者電氣特性設定)
dt_free_map上面函數的逆函數

復用引腳相關的操作函數的具體解釋如下:

call back函數描述
requestpin control core進行具體的復用設定之前需要調用該函數,主要是用來請底層的driver判斷某個引腳的復用設定是否是OK的。
free是request的逆函數。調用request函數請求占用了某些pin的資源,調用free可以釋放這些資源
get_functions_count就是返回pin controller支持的function的數目
get_function_name給定一個selector(index),獲取指定function的name
get_function_groups給定一個selector(index),獲取指定function的pin groups信息
enableenable一個function。當然要給出function selector和pin group的selector
disableenable的逆函數
gpio_request_enablerequest并且enable一個單獨的gpio pin
gpio_disable_freegpio_request_enable的逆函數
gpio_set_direction設定GPIO方向的回調函數

配置引腳的特性的struct pinconf_ops數據結構的各個成員定義如下:

call back函數描述
pin_config_get給定一個pin ID以及config type ID,獲取該引腳上指定type的配置。
pin_config_set設定一個指定pin的配置
pin_config_group_get以pin group為單位,獲取pin上的配置信息
pin_config_group_set以pin group為單位,設定pin group的特性配置
pin_config_dbg_parse_modifydebug接口
pin_config_dbg_showdebug接口
pin_config_group_dbg_showdebug接口
pin_config_config_dbg_showdebug接口

?

原創文章,轉發請注明出處。蝸窩科技。http://www.wowotech.net/linux_kenrel/pin-control-subsystem.html

總結

以上是生活随笔為你收集整理的linux内核中的GPIO系统之(2):pin control subsystem的全部內容,希望文章能夠幫你解決所遇到的問題。

如果覺得生活随笔網站內容還不錯,歡迎將生活随笔推薦給好友。