0
  • 聊天消息
  • 系統(tǒng)消息
  • 評(píng)論與回復(fù)
登錄后你可以
  • 下載海量資料
  • 學(xué)習(xí)在線(xiàn)課程
  • 觀(guān)看技術(shù)視頻
  • 寫(xiě)文章/發(fā)帖/加入社區(qū)
會(huì)員中心
創(chuàng)作中心

完善資料讓更多小伙伴認(rèn)識(shí)你,還能領(lǐng)取20積分哦,立即完善>

3天內(nèi)不再提示

Firefly-RK3128主板GPIO引腳介紹

firefly ? 來(lái)源:firefly ? 作者:firefly ? 2019-11-28 17:15 ? 次閱讀
GPIO 使用
簡(jiǎn)介

GPIO, 全稱(chēng) General-Purpose Input/Output(通用輸入輸出),是一種軟件運(yùn)行期間能夠動(dòng)態(tài)配置和控制的通用引腳。 Firefly-RK3128 有 4 組 GPIO bank: GPIO0,GPIO1, GPIO2, GPIO3。每組又以 A0~A7, B0~B7, C0~C7, D0~D7 作為編號(hào)區(qū)分。 每個(gè) GPIO 口除了通用輸入輸出功能外,還可能有其它復(fù)用功能,例如 GPIO1_C2,可以復(fù)用成以下功能之一:

每個(gè) GPIO 口的驅(qū)動(dòng)電流、上下拉和重置后的初始狀態(tài)都不盡相同,詳細(xì)情況請(qǐng)參考《RK3128 規(guī)格書(shū)》中的 “RK3128 function IO description” 一章。

Firefly-RK3128 的 GPIO 驅(qū)動(dòng)是在以下 pinctrl 文件中實(shí)現(xiàn)的:

kernel/drivers/pinctrl/pinctrl-rockchip.c

其核心是填充 GPIO bank 的方法和參數(shù),并調(diào)用 gpiochip_add 注冊(cè)到內(nèi)核中。

使用

開(kāi)發(fā)板有兩個(gè)電源 LED 燈是 GPIO 口控制的,分別是:

從電路圖上看,GPIO 口輸出低電平時(shí)燈亮,高電平時(shí)燈滅。

輸入輸出

下面以電源 LED 燈的驅(qū)動(dòng)為例,講述如何在內(nèi)核編寫(xiě)代碼控制 GPIO 口的輸出。

首先需要在 rk3128-firerpime.dts 中增加驅(qū)動(dòng)的資源描述:

firefly-led{ compatible = "firefly,led"; led-work = <&gpio1 GPIO_C6 GPIO_ACTIVE_LOW>; led-power = <&gpio1 GPIO_C7 GPIO_ACTIVE_LOW>; status = "okay"; };

這里定義了兩顆 LED 燈的 GPIO 設(shè)置:

led-work GPIO1_C6 GPIO_ACTIVE_LOW led-power GPIO1_C7 GPIO_ACTIVE_LOW

GPIO_ACTIVE_LOW 表示低電平有效(燈亮),如果是高電平有效,需要替換為 GPIO_ACTIVE_HIGH 。 之后在驅(qū)動(dòng)程序中加入對(duì) GPIO 口的申請(qǐng)和控制則可:

#ifdef CONFIG_OF #include #include #endif static int firefly_led_probe(struct platform_device *pdev) { int ret = -1; int gpio, flag; struct device_node *led_node = pdev->dev.of_node; gpio = of_get_named_gpio_flags(led_node, "led-power", 0, &flag); if (!gpio_is_valid(gpio)){ printk("invalid led-power: %d\n",gpio); return -1; } if (gpio_request(gpio, "led_power")) { printk("gpio %d request failed!\n",gpio); return ret; } led_info.power_gpio = gpio; led_info.power_enable_value = (flag == OF_GPIO_ACTIVE_LOW) ? 0 : 1; gpio_direction_output(led_info.power_gpio, !(led_info.power_enable_value)); ... on_error:gpio_free(gpio); }

of_get_named_gpio_flags 從設(shè)備樹(shù)中讀取 led-power 的 GPIO 配置編號(hào)和標(biāo)志,gpio_is_valid 判斷該 GPIO 編號(hào)是否有效,gpio_request 則申請(qǐng)占用該 GPIO。如果初始化過(guò)程出錯(cuò),需要調(diào)用 gpio_free 來(lái)釋放之前申請(qǐng)過(guò)且成功的 GPIO 。

調(diào)用 gpio_direction_output 就可以設(shè)置輸出高還是低電平,因?yàn)槭?GPIO_ACTIVE_LOW ,如果要燈亮,需要寫(xiě)入 0 。

實(shí)際中如果要讀出 GPIO,需要先設(shè)置成輸入模式,然后再讀取值:

int val;gpio_direction_input(your_gpio);val = gpio_get_value(your_gpio);

下面是常用的 GPIO API 定義:

#include #include enum of_gpio_flags { OF_GPIO_ACTIVE_LOW = 0x1, }; int of_get_named_gpio_flags(struct device_node *np, const char *propname, int index, enum of_gpio_flags *flags); int gpio_is_valid(int gpio); int gpio_request(unsigned gpio, const char *label); void gpio_free(unsigned gpio); int gpio_direction_input(int gpio); int gpio_direction_output(int gpio, int v);
復(fù)用

如何定義 GPIO 有哪些功能可以復(fù)用,在運(yùn)行時(shí)又如何切換功能呢?以 I2C1 為例作簡(jiǎn)單的介紹。查規(guī)格表可知,I2C1_SDA 與 I2C1_SCL 的功能定義如下:

在 /kernel/arch/arm/boot/dts/rk312x.dtsi 里有:

i2c1: i2c@20056000 { compatible = "rockchip,rk30-i2c"; reg = <0x20056000 0x1000>; interrupts = <GIC_SPI 25 IRQ_TYPE_LEVEL_HIGH>;#address-cells = <1>;#size-cells = <0>; pinctrl-names = "default", "gpio"; pinctrl-0 = <&i2c1_sda &i2c1_scl>; pinctrl-1 = <&i2c1_gpio>; gpios = <&gpio0 GPIO_A3 GPIO_ACTIVE_LOW>, <&gpio0 GPIO_A2 GPIO_ACTIVE_LOW>; clocks = <&clk_gates8 5>; rockchip,check-idle = <1>; status = "disabled"; };

此處,跟復(fù)用控制相關(guān)的是 pinctrl- 開(kāi)頭的屬性:

  • pinctrl-names 定義了狀態(tài)名稱(chēng)列表: default (i2c 功能) 和 gpio 兩種狀態(tài)。
  • pinctrl-0 定義了狀態(tài) 0 (即 default)時(shí)需要設(shè)置的 pinctrl: i2c1_sda 和 i2c1_scl
  • pinctrl-1 定義了狀態(tài) 1 (即 gpio)時(shí)需要設(shè)置的 pinctrl: i2c1_gpio

這些 pinctrl 在 /kernel/arch/arm/boot/dts/rk312x-pinctrl.dtsi 中定義:

/ { pinctrl: pinctrl@20008000 { compatible = "rockchip,rk312x-pinctrl";... gpio0_i2c1 { i2c1_sda:i2c1-sda { rockchip,pins = <I2C1_SDA>; rockchip,pull = <VALUE_PULL_DEFAULT>; }; i2c1_scl:i2c1-scl { rockchip,pins = <I2C1_SCL>; rockchip,pull = <VALUE_PULL_DEFAULT>; }; i2c1_gpio: i2c1-gpio { rockchip,pins = <FUNC_TO_GPIO(I2C1_SDA)>, <FUNC_TO_GPIO(I2C1_SCL)>; rockchip,pull = <VALUE_PULL_DEFAULT>;};}; ... } }

I2C1_SDA, I2C1_SCL 的定義在 /kernel/arch/arm/boot/dts/include/dt-bindings/pinctrl/rockchip-rk312x.h 中:

#define GPIO0_A3 0x0a30#define I2C1_SDA 0x0a31#define MMC1_CMD 0x0a32 #define GPIO0_A2 0x0a20#define I2C1_SCL 0x0a21

FUN_TO_GPIO 的定義在 /kernel/arch/arm/boot/dts/include/dt-bindings/pinctrl/rockchip.h 中:

#define FUNC_TO_GPIO(m) ((m) & 0xfff0)

也就是說(shuō) FUNC_TO_GPIO(I2C1_SDA) == GPIO0_A3, FUNC_TO_GPIO(I2C1_SCL) == GPIO7_A2 。 像 0x0a31 這樣的值是有編碼規(guī)則的:

0 a3 1 | | `- func | `---- offset `------ bank

0x0a31 就表示 GPIO0_A3 func1, 即 I2C1_SDA 。

在復(fù)用時(shí),如果選擇了 “default” (即 i2c 功能),系統(tǒng)會(huì)應(yīng)用 i2c1_sda 和 i2c1_scl 這兩個(gè) pinctrl,最終得將 GPIO0_A3 和 GPIO0_A2 兩個(gè)針腳切換成對(duì)應(yīng)的 i2c 功能;而如果選擇了 “gpio” ,系統(tǒng)會(huì)應(yīng)用 i2c1_gpio 這個(gè) pinctrl,將 GPIO0_A3 和 GPIO0_A2 兩個(gè)針腳還原為 GPIO 功能。我們看看 i2c 的驅(qū)動(dòng)程序 /kernel/drivers/i2c/busses/i2c-rockchip.c 是如何切換復(fù)用功能的:

static int rockchip_i2c_probe(struct platform_device *pdev){ struct rockchip_i2c *i2c = NULL; struct resource *res; struct device_node *np = pdev->dev.of_node; int ret; // ... i2c->sda_gpio = of_get_gpio(np, 0); if (!gpio_is_valid(i2c->sda_gpio)) { dev_err(&pdev->dev, "sda gpio is invalid\n"); return -EINVAL; } ret = devm_gpio_request(&pdev->dev, i2c->sda_gpio, dev_name(&i2c->adap.dev)); if (ret) { dev_err(&pdev->dev, "failed to request sda gpio\n"); return ret; } i2c->scl_gpio = of_get_gpio(np, 1); if (!gpio_is_valid(i2c->scl_gpio)) { dev_err(&pdev->dev, "scl gpio is invalid\n"); return -EINVAL; } ret = devm_gpio_request(&pdev->dev, i2c->scl_gpio, dev_name(&i2c->adap.dev)); if (ret) { dev_err(&pdev->dev, "failed to request scl gpio\n"); return ret; } i2c->gpio_state = pinctrl_lookup_state(i2c->dev->pins->p, "gpio"); if (IS_ERR(i2c->gpio_state)) { dev_err(&pdev->dev, "no gpio pinctrl state\n"); return PTR_ERR(i2c->gpio_state); } pinctrl_select_state(i2c->dev->pins->p, i2c->gpio_state); gpio_direction_input(i2c->sda_gpio); gpio_direction_input(i2c->scl_gpio); pinctrl_select_state(i2c->dev->pins->p, i2c->dev->pins->default_state); // ... }

首先是調(diào)用 of_get_gpio 取出設(shè)備樹(shù)中 i2c1 結(jié)點(diǎn)的 gpios 屬于所定義的兩個(gè) gpio:

gpios = <&gpio0 GPIO_A3 GPIO_ACTIVE_LOW>, <&gpio0 GPIO_A2 GPIO_ACTIVE_LOW>;

然后是調(diào)用 devm_gpio_request 來(lái)申請(qǐng) gpio,接著是調(diào)用 pinctrl_lookup_state 來(lái)查找 “gpio” 狀態(tài),而默認(rèn)狀態(tài) “default” 已經(jīng)由框架保存到 i2c->dev-pins->default_state 中了。最后調(diào)用 pinctrl_select_state 來(lái)選擇是 “default” 還是 “gpio” 功能。 下面是常用的復(fù)用 API 定義:

#include struct device { //... #ifdef CONFIG_PINCTRL struct dev_pin_info *pins;#endif//...}; struct dev_pin_info {struct pinctrl *p; struct pinctrl_state *default_state; #ifdef CONFIG_PMstruct pinctrl_state *sleep_state; struct pinctrl_state *idle_state;#endif}; struct pinctrl_state * pinctrl_lookup_state(struct pinctrl *p, const char *name); int pinctrl_select_state(struct pinctrl *p, struct pinctrl_state *s);

聲明:本文內(nèi)容及配圖由入駐作者撰寫(xiě)或者入駐合作網(wǎng)站授權(quán)轉(zhuǎn)載。文章觀(guān)點(diǎn)僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場(chǎng)。文章及其配圖僅供工程師學(xué)習(xí)之用,如有內(nèi)容侵權(quán)或者其他違規(guī)問(wèn)題,請(qǐng)聯(lián)系本站處理。 舉報(bào)投訴
  • Linux
    +關(guān)注

    關(guān)注

    87

    文章

    11212

    瀏覽量

    208721
  • 嵌入式主板
    +關(guān)注

    關(guān)注

    7

    文章

    6084

    瀏覽量

    35154
  • Firefly
    +關(guān)注

    關(guān)注

    2

    文章

    538

    瀏覽量

    6963
收藏 人收藏

    評(píng)論

    相關(guān)推薦

    如何解決firefly rk3128 linux內(nèi)核適配問(wèn)題?

    如何解決firefly rk3128 linux內(nèi)核適配問(wèn)題?
    發(fā)表于 03-04 06:17

    Firefly-RK3128開(kāi)發(fā)板上手教程(上)

    1、Firefly-RK3128開(kāi)發(fā)板入手指南配件Firefly-RK3128 的標(biāo)準(zhǔn)套裝包含以下配件:Core-3128J 核心板一塊Firefly-RK3128 基板一塊WiFi
    發(fā)表于 08-12 17:32

    Firefly-RK3128開(kāi)發(fā)板上手教程(下)

    1、如何將主機(jī)上的固件文件通過(guò)雙公頭USB數(shù)據(jù)線(xiàn)燒錄到開(kāi)發(fā)板的閃存中準(zhǔn)備工作Firefly-RK3128 開(kāi)發(fā)板固件主機(jī)良好的 Micro USB OTG 線(xiàn)固件文件一般有兩種:?jiǎn)蝹€(gè)統(tǒng)一固件
    發(fā)表于 08-12 17:51

    Firefly-RK3128 MIPI DSI使用步驟說(shuō)明

    screen-type屬性:顯示屏類(lèi)型,Firefly-RK3128只支持單通道MIPI屏,配置成SCREEN_MIPI即可。lvds-format屬性:無(wú)關(guān)選項(xiàng)。out-face屬性:配置顏色,可為OUT_P888
    發(fā)表于 08-22 17:12

    firefly RK3128開(kāi)發(fā)板介紹

    Firefly-RK3128 采用Cortex-A7架構(gòu)四核1.3GHz處理器,集成Mali-400MP2 GPU, 擁有優(yōu)秀的運(yùn)算與圖形處理能力;板載千兆以太網(wǎng)口、2.4GHz Wi-Fi和藍(lán)牙4.0
    的頭像 發(fā)表于 11-14 11:27 ?4444次閱讀
    <b class='flag-5'>firefly</b> <b class='flag-5'>RK3128</b>開(kāi)發(fā)板<b class='flag-5'>介紹</b>

    Firefly-RK3128開(kāi)發(fā)板UART接口介紹

    refly-RK3128 開(kāi)發(fā)板內(nèi)置 3 路 UART,分別為 uart0,uart1,uart2。uart0 用于藍(lán)牙數(shù)據(jù)傳輸,如果要使用 uart0,必須關(guān)掉藍(lán)牙,才可以使用擴(kuò)展槽上的 UART0 針腳。
    的頭像 發(fā)表于 11-28 16:36 ?3258次閱讀
    <b class='flag-5'>Firefly-RK3128</b>開(kāi)發(fā)板UART接口<b class='flag-5'>介紹</b>

    Firefly-RK3128主板PWM輸出介紹

    Firefly-RK3128 開(kāi)發(fā)板上有 4 路 PWM 輸出,分別為 PWM0 ~ PWM3
    的頭像 發(fā)表于 11-28 16:56 ?2312次閱讀
    <b class='flag-5'>Firefly-RK3128</b><b class='flag-5'>主板</b>PWM輸出<b class='flag-5'>介紹</b>

    Firefly-RK3128主板LED子系統(tǒng)介紹

    Firefly-RK3128 開(kāi)發(fā)板上有 2 個(gè) LED 燈
    的頭像 發(fā)表于 11-28 17:01 ?2691次閱讀
    <b class='flag-5'>Firefly-RK3128</b><b class='flag-5'>主板</b>LED子系統(tǒng)<b class='flag-5'>介紹</b>

    Firefly-RK3128主板I2C控制器

    Firefly-RK3128 開(kāi)發(fā)板上有 4 個(gè)片上 I2C 控制器。本文主要描述如何在該開(kāi)發(fā)板上配置 I2C。
    的頭像 發(fā)表于 11-29 08:47 ?1417次閱讀
    <b class='flag-5'>Firefly-RK3128</b><b class='flag-5'>主板</b>I2C控制器

    Firefly-RK3128開(kāi)發(fā)板攝像頭的介紹

    Firefly-RK3128 開(kāi)發(fā)板上有 CIF 接口,支持 CIF 攝像頭。
    的頭像 發(fā)表于 11-28 16:48 ?3692次閱讀
    <b class='flag-5'>Firefly-RK3128</b>開(kāi)發(fā)板攝像頭的<b class='flag-5'>介紹</b>

    Firefly-RK3128主板ADC簡(jiǎn)介

    Firefly-RK3128 開(kāi)發(fā)板有一個(gè) 3 通道(0/1/2)、10 比特精度的 SAR ADC (Successive Approximation Register,逐次逼近寄存器),
    的頭像 發(fā)表于 11-28 17:17 ?3464次閱讀
    <b class='flag-5'>Firefly-RK3128</b><b class='flag-5'>主板</b>ADC簡(jiǎn)介

    Firefly-RK3128主板接口定義介紹

    Firefly-RK3128 開(kāi)發(fā)板提供了豐富的接口,主要包括:HDMI、音頻數(shù)字光纖、以太網(wǎng)、電源接口、復(fù)位按鍵、電源鍵、音頻輸入輸出、硅麥、串口調(diào)試接口、紅外接收、OTG接口、TF卡槽、USB Host1~4、
    的頭像 發(fā)表于 11-29 08:45 ?3397次閱讀
    <b class='flag-5'>Firefly-RK3128</b><b class='flag-5'>主板</b>接口定義<b class='flag-5'>介紹</b>

    FireflyRK3128主板外殼介紹

    外殼 Firefly-RK3128 CAD圖 核心板 底板 Firefly-RK3128 外殼亞克力CAD圖
    的頭像 發(fā)表于 11-29 09:19 ?1205次閱讀
    FireflyRK<b class='flag-5'>3128</b><b class='flag-5'>主板</b>外殼<b class='flag-5'>介紹</b>

    Firefly-RK3128主板啟動(dòng)模式簡(jiǎn)介

    Firefly-RK3128 有靈活的啟動(dòng)方式。
    的頭像 發(fā)表于 11-29 09:27 ?5120次閱讀

    Firefly-RK3128主板串口調(diào)試

    Firefly-RK3128 開(kāi)發(fā)板的調(diào)試串口與 TF 卡接口有信號(hào)引腳復(fù)用,因此無(wú)法同時(shí)使用
    的頭像 發(fā)表于 11-29 09:47 ?3448次閱讀
    <b class='flag-5'>Firefly-RK3128</b><b class='flag-5'>主板</b>串口調(diào)試