| /* |
| * Pinctrl driver for the Toumaz Xenif TZ1090 SoC |
| * |
| * Copyright (c) 2013, Imagination Technologies Ltd. |
| * |
| * Derived from Tegra code: |
| * Copyright (c) 2011-2012, NVIDIA CORPORATION. All rights reserved. |
| * |
| * Derived from code: |
| * Copyright (C) 2010 Google, Inc. |
| * Copyright (C) 2010 NVIDIA Corporation |
| * Copyright (C) 2009-2011 ST-Ericsson AB |
| * |
| * This program is free software; you can redistribute it and/or modify it |
| * under the terms and conditions of the GNU General Public License, |
| * version 2, as published by the Free Software Foundation. |
| * |
| * This program is distributed in the hope it will be useful, but WITHOUT |
| * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or |
| * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for |
| * more details. |
| */ |
| |
| #include <linux/bitops.h> |
| #include <linux/io.h> |
| #include <linux/module.h> |
| #include <linux/of.h> |
| #include <linux/platform_device.h> |
| #include <linux/pinctrl/machine.h> |
| #include <linux/pinctrl/pinconf-generic.h> |
| #include <linux/pinctrl/pinctrl.h> |
| #include <linux/pinctrl/pinmux.h> |
| #include <linux/slab.h> |
| #include <linux/spinlock.h> |
| |
| /* |
| * The registers may be shared with other threads/cores, so we need to use the |
| * metag global lock2 for atomicity. |
| */ |
| #include <asm/global_lock.h> |
| |
| #include "core.h" |
| #include "pinconf.h" |
| |
| /* Register offsets from bank base address */ |
| #define REG_PINCTRL_SELECT 0x10 |
| #define REG_PINCTRL_SCHMITT 0x90 |
| #define REG_PINCTRL_PU_PD 0xa0 |
| #define REG_PINCTRL_SR 0xc0 |
| #define REG_PINCTRL_DR 0xd0 |
| #define REG_PINCTRL_IF_CTL 0xe0 |
| |
| /* REG_PINCTRL_PU_PD field values */ |
| #define REG_PU_PD_TRISTATE 0 |
| #define REG_PU_PD_UP 1 |
| #define REG_PU_PD_DOWN 2 |
| #define REG_PU_PD_REPEATER 3 |
| |
| /* REG_PINCTRL_DR field values */ |
| #define REG_DR_2mA 0 |
| #define REG_DR_4mA 1 |
| #define REG_DR_8mA 2 |
| #define REG_DR_12mA 3 |
| |
| /** |
| * struct tz1090_function - TZ1090 pinctrl mux function |
| * @name: The name of the function, exported to pinctrl core. |
| * @groups: An array of pin groups that may select this function. |
| * @ngroups: The number of entries in @groups. |
| */ |
| struct tz1090_function { |
| const char *name; |
| const char * const *groups; |
| unsigned int ngroups; |
| }; |
| |
| /** |
| * struct tz1090_muxdesc - TZ1090 individual mux description |
| * @funcs: Function for each mux value. |
| * @reg: Mux register offset. 0 if unsupported. |
| * @bit: Mux register bit. 0 if unsupported. |
| * @width: Mux field width. 0 if unsupported. |
| * |
| * A representation of a group of signals (possibly just one signal) in the |
| * TZ1090 which can be muxed to a set of functions or sub muxes. |
| */ |
| struct tz1090_muxdesc { |
| int funcs[5]; |
| u16 reg; |
| u8 bit; |
| u8 width; |
| }; |
| |
| /** |
| * struct tz1090_pingroup - TZ1090 pin group |
| * @name: Name of pin group. |
| * @pins: Array of pin numbers in this pin group. |
| * @npins: Number of pins in this pin group. |
| * @mux: Top level mux. |
| * @drv: Drive control supported, 0 if unsupported. |
| * This means Schmitt, Slew, and Drive strength. |
| * @slw_bit: Slew register bit. 0 if unsupported. |
| * The same bit is used for Schmitt, and Drive (*2). |
| * @func: Currently muxed function. |
| * @func_count: Number of pins using current mux function. |
| * |
| * A representation of a group of pins (possibly just one pin) in the TZ1090 |
| * pin controller. Each group allows some parameter or parameters to be |
| * configured. The most common is mux function selection. |
| */ |
| struct tz1090_pingroup { |
| const char *name; |
| const unsigned int *pins; |
| unsigned int npins; |
| struct tz1090_muxdesc mux; |
| |
| bool drv; |
| u8 slw_bit; |
| |
| int func; |
| unsigned int func_count; |
| }; |
| |
| /* |
| * Most pins affected by the pinmux can also be GPIOs. Define these first. |
| * These must match how the GPIO driver names/numbers its pins. |
| */ |
| |
| enum tz1090_pin { |
| /* GPIO pins */ |
| TZ1090_PIN_SDIO_CLK, |
| TZ1090_PIN_SDIO_CMD, |
| TZ1090_PIN_SDIO_D0, |
| TZ1090_PIN_SDIO_D1, |
| TZ1090_PIN_SDIO_D2, |
| TZ1090_PIN_SDIO_D3, |
| TZ1090_PIN_SDH_CD, |
| TZ1090_PIN_SDH_WP, |
| TZ1090_PIN_SPI0_MCLK, |
| TZ1090_PIN_SPI0_CS0, |
| TZ1090_PIN_SPI0_CS1, |
| TZ1090_PIN_SPI0_CS2, |
| TZ1090_PIN_SPI0_DOUT, |
| TZ1090_PIN_SPI0_DIN, |
| TZ1090_PIN_SPI1_MCLK, |
| TZ1090_PIN_SPI1_CS0, |
| TZ1090_PIN_SPI1_CS1, |
| TZ1090_PIN_SPI1_CS2, |
| TZ1090_PIN_SPI1_DOUT, |
| TZ1090_PIN_SPI1_DIN, |
| TZ1090_PIN_UART0_RXD, |
| TZ1090_PIN_UART0_TXD, |
| TZ1090_PIN_UART0_CTS, |
| TZ1090_PIN_UART0_RTS, |
| TZ1090_PIN_UART1_RXD, |
| TZ1090_PIN_UART1_TXD, |
| TZ1090_PIN_SCB0_SDAT, |
| TZ1090_PIN_SCB0_SCLK, |
| TZ1090_PIN_SCB1_SDAT, |
| TZ1090_PIN_SCB1_SCLK, |
| TZ1090_PIN_SCB2_SDAT, |
| TZ1090_PIN_SCB2_SCLK, |
| TZ1090_PIN_I2S_MCLK, |
| TZ1090_PIN_I2S_BCLK_OUT, |
| TZ1090_PIN_I2S_LRCLK_OUT, |
| TZ1090_PIN_I2S_DOUT0, |
| TZ1090_PIN_I2S_DOUT1, |
| TZ1090_PIN_I2S_DOUT2, |
| TZ1090_PIN_I2S_DIN, |
| TZ1090_PIN_PDM_A, |
| TZ1090_PIN_PDM_B, |
| TZ1090_PIN_PDM_C, |
| TZ1090_PIN_PDM_D, |
| TZ1090_PIN_TFT_RED0, |
| TZ1090_PIN_TFT_RED1, |
| TZ1090_PIN_TFT_RED2, |
| TZ1090_PIN_TFT_RED3, |
| TZ1090_PIN_TFT_RED4, |
| TZ1090_PIN_TFT_RED5, |
| TZ1090_PIN_TFT_RED6, |
| TZ1090_PIN_TFT_RED7, |
| TZ1090_PIN_TFT_GREEN0, |
| TZ1090_PIN_TFT_GREEN1, |
| TZ1090_PIN_TFT_GREEN2, |
| TZ1090_PIN_TFT_GREEN3, |
| TZ1090_PIN_TFT_GREEN4, |
| TZ1090_PIN_TFT_GREEN5, |
| TZ1090_PIN_TFT_GREEN6, |
| TZ1090_PIN_TFT_GREEN7, |
| TZ1090_PIN_TFT_BLUE0, |
| TZ1090_PIN_TFT_BLUE1, |
| TZ1090_PIN_TFT_BLUE2, |
| TZ1090_PIN_TFT_BLUE3, |
| TZ1090_PIN_TFT_BLUE4, |
| TZ1090_PIN_TFT_BLUE5, |
| TZ1090_PIN_TFT_BLUE6, |
| TZ1090_PIN_TFT_BLUE7, |
| TZ1090_PIN_TFT_VDDEN_GD, |
| TZ1090_PIN_TFT_PANELCLK, |
| TZ1090_PIN_TFT_BLANK_LS, |
| TZ1090_PIN_TFT_VSYNC_NS, |
| TZ1090_PIN_TFT_HSYNC_NR, |
| TZ1090_PIN_TFT_VD12ACB, |
| TZ1090_PIN_TFT_PWRSAVE, |
| TZ1090_PIN_TX_ON, |
| TZ1090_PIN_RX_ON, |
| TZ1090_PIN_PLL_ON, |
| TZ1090_PIN_PA_ON, |
| TZ1090_PIN_RX_HP, |
| TZ1090_PIN_GAIN0, |
| TZ1090_PIN_GAIN1, |
| TZ1090_PIN_GAIN2, |
| TZ1090_PIN_GAIN3, |
| TZ1090_PIN_GAIN4, |
| TZ1090_PIN_GAIN5, |
| TZ1090_PIN_GAIN6, |
| TZ1090_PIN_GAIN7, |
| TZ1090_PIN_ANT_SEL0, |
| TZ1090_PIN_ANT_SEL1, |
| TZ1090_PIN_SDH_CLK_IN, |
| |
| /* Non-GPIO pins */ |
| TZ1090_PIN_TCK, |
| TZ1090_PIN_TRST, |
| TZ1090_PIN_TDI, |
| TZ1090_PIN_TDO, |
| TZ1090_PIN_TMS, |
| TZ1090_PIN_CLK_OUT0, |
| TZ1090_PIN_CLK_OUT1, |
| |
| NUM_GPIOS = TZ1090_PIN_TCK, |
| }; |
| |
| /* Pin names */ |
| |
| static const struct pinctrl_pin_desc tz1090_pins[] = { |
| /* GPIO pins */ |
| PINCTRL_PIN(TZ1090_PIN_SDIO_CLK, "sdio_clk"), |
| PINCTRL_PIN(TZ1090_PIN_SDIO_CMD, "sdio_cmd"), |
| PINCTRL_PIN(TZ1090_PIN_SDIO_D0, "sdio_d0"), |
| PINCTRL_PIN(TZ1090_PIN_SDIO_D1, "sdio_d1"), |
| PINCTRL_PIN(TZ1090_PIN_SDIO_D2, "sdio_d2"), |
| PINCTRL_PIN(TZ1090_PIN_SDIO_D3, "sdio_d3"), |
| PINCTRL_PIN(TZ1090_PIN_SDH_CD, "sdh_cd"), |
| PINCTRL_PIN(TZ1090_PIN_SDH_WP, "sdh_wp"), |
| PINCTRL_PIN(TZ1090_PIN_SPI0_MCLK, "spi0_mclk"), |
| PINCTRL_PIN(TZ1090_PIN_SPI0_CS0, "spi0_cs0"), |
| PINCTRL_PIN(TZ1090_PIN_SPI0_CS1, "spi0_cs1"), |
| PINCTRL_PIN(TZ1090_PIN_SPI0_CS2, "spi0_cs2"), |
| PINCTRL_PIN(TZ1090_PIN_SPI0_DOUT, "spi0_dout"), |
| PINCTRL_PIN(TZ1090_PIN_SPI0_DIN, "spi0_din"), |
| PINCTRL_PIN(TZ1090_PIN_SPI1_MCLK, "spi1_mclk"), |
| PINCTRL_PIN(TZ1090_PIN_SPI1_CS0, "spi1_cs0"), |
| PINCTRL_PIN(TZ1090_PIN_SPI1_CS1, "spi1_cs1"), |
| PINCTRL_PIN(TZ1090_PIN_SPI1_CS2, "spi1_cs2"), |
| PINCTRL_PIN(TZ1090_PIN_SPI1_DOUT, "spi1_dout"), |
| PINCTRL_PIN(TZ1090_PIN_SPI1_DIN, "spi1_din"), |
| PINCTRL_PIN(TZ1090_PIN_UART0_RXD, "uart0_rxd"), |
| PINCTRL_PIN(TZ1090_PIN_UART0_TXD, "uart0_txd"), |
| PINCTRL_PIN(TZ1090_PIN_UART0_CTS, "uart0_cts"), |
| PINCTRL_PIN(TZ1090_PIN_UART0_RTS, "uart0_rts"), |
| PINCTRL_PIN(TZ1090_PIN_UART1_RXD, "uart1_rxd"), |
| PINCTRL_PIN(TZ1090_PIN_UART1_TXD, "uart1_txd"), |
| PINCTRL_PIN(TZ1090_PIN_SCB0_SDAT, "scb0_sdat"), |
| PINCTRL_PIN(TZ1090_PIN_SCB0_SCLK, "scb0_sclk"), |
| PINCTRL_PIN(TZ1090_PIN_SCB1_SDAT, "scb1_sdat"), |
| PINCTRL_PIN(TZ1090_PIN_SCB1_SCLK, "scb1_sclk"), |
| PINCTRL_PIN(TZ1090_PIN_SCB2_SDAT, "scb2_sdat"), |
| PINCTRL_PIN(TZ1090_PIN_SCB2_SCLK, "scb2_sclk"), |
| PINCTRL_PIN(TZ1090_PIN_I2S_MCLK, "i2s_mclk"), |
| PINCTRL_PIN(TZ1090_PIN_I2S_BCLK_OUT, "i2s_bclk_out"), |
| PINCTRL_PIN(TZ1090_PIN_I2S_LRCLK_OUT, "i2s_lrclk_out"), |
| PINCTRL_PIN(TZ1090_PIN_I2S_DOUT0, "i2s_dout0"), |
| PINCTRL_PIN(TZ1090_PIN_I2S_DOUT1, "i2s_dout1"), |
| PINCTRL_PIN(TZ1090_PIN_I2S_DOUT2, "i2s_dout2"), |
| PINCTRL_PIN(TZ1090_PIN_I2S_DIN, "i2s_din"), |
| PINCTRL_PIN(TZ1090_PIN_PDM_A, "pdm_a"), |
| PINCTRL_PIN(TZ1090_PIN_PDM_B, "pdm_b"), |
| PINCTRL_PIN(TZ1090_PIN_PDM_C, "pdm_c"), |
| PINCTRL_PIN(TZ1090_PIN_PDM_D, "pdm_d"), |
| PINCTRL_PIN(TZ1090_PIN_TFT_RED0, "tft_red0"), |
| PINCTRL_PIN(TZ1090_PIN_TFT_RED1, "tft_red1"), |
| PINCTRL_PIN(TZ1090_PIN_TFT_RED2, "tft_red2"), |
| PINCTRL_PIN(TZ1090_PIN_TFT_RED3, "tft_red3"), |
| PINCTRL_PIN(TZ1090_PIN_TFT_RED4, "tft_red4"), |
| PINCTRL_PIN(TZ1090_PIN_TFT_RED5, "tft_red5"), |
| PINCTRL_PIN(TZ1090_PIN_TFT_RED6, "tft_red6"), |
| PINCTRL_PIN(TZ1090_PIN_TFT_RED7, "tft_red7"), |
| PINCTRL_PIN(TZ1090_PIN_TFT_GREEN0, "tft_green0"), |
| PINCTRL_PIN(TZ1090_PIN_TFT_GREEN1, "tft_green1"), |
| PINCTRL_PIN(TZ1090_PIN_TFT_GREEN2, "tft_green2"), |
| PINCTRL_PIN(TZ1090_PIN_TFT_GREEN3, "tft_green3"), |
| PINCTRL_PIN(TZ1090_PIN_TFT_GREEN4, "tft_green4"), |
| PINCTRL_PIN(TZ1090_PIN_TFT_GREEN5, "tft_green5"), |
| PINCTRL_PIN(TZ1090_PIN_TFT_GREEN6, "tft_green6"), |
| PINCTRL_PIN(TZ1090_PIN_TFT_GREEN7, "tft_green7"), |
| PINCTRL_PIN(TZ1090_PIN_TFT_BLUE0, "tft_blue0"), |
| PINCTRL_PIN(TZ1090_PIN_TFT_BLUE1, "tft_blue1"), |
| PINCTRL_PIN(TZ1090_PIN_TFT_BLUE2, "tft_blue2"), |
| PINCTRL_PIN(TZ1090_PIN_TFT_BLUE3, "tft_blue3"), |
| PINCTRL_PIN(TZ1090_PIN_TFT_BLUE4, "tft_blue4"), |
| PINCTRL_PIN(TZ1090_PIN_TFT_BLUE5, "tft_blue5"), |
| PINCTRL_PIN(TZ1090_PIN_TFT_BLUE6, "tft_blue6"), |
| PINCTRL_PIN(TZ1090_PIN_TFT_BLUE7, "tft_blue7"), |
| PINCTRL_PIN(TZ1090_PIN_TFT_VDDEN_GD, "tft_vdden_gd"), |
| PINCTRL_PIN(TZ1090_PIN_TFT_PANELCLK, "tft_panelclk"), |
| PINCTRL_PIN(TZ1090_PIN_TFT_BLANK_LS, "tft_blank_ls"), |
| PINCTRL_PIN(TZ1090_PIN_TFT_VSYNC_NS, "tft_vsync_ns"), |
| PINCTRL_PIN(TZ1090_PIN_TFT_HSYNC_NR, "tft_hsync_nr"), |
| PINCTRL_PIN(TZ1090_PIN_TFT_VD12ACB, "tft_vd12acb"), |
| PINCTRL_PIN(TZ1090_PIN_TFT_PWRSAVE, "tft_pwrsave"), |
| PINCTRL_PIN(TZ1090_PIN_TX_ON, "tx_on"), |
| PINCTRL_PIN(TZ1090_PIN_RX_ON, "rx_on"), |
| PINCTRL_PIN(TZ1090_PIN_PLL_ON, "pll_on"), |
| PINCTRL_PIN(TZ1090_PIN_PA_ON, "pa_on"), |
| PINCTRL_PIN(TZ1090_PIN_RX_HP, "rx_hp"), |
| PINCTRL_PIN(TZ1090_PIN_GAIN0, "gain0"), |
| PINCTRL_PIN(TZ1090_PIN_GAIN1, "gain1"), |
| PINCTRL_PIN(TZ1090_PIN_GAIN2, "gain2"), |
| PINCTRL_PIN(TZ1090_PIN_GAIN3, "gain3"), |
| PINCTRL_PIN(TZ1090_PIN_GAIN4, "gain4"), |
| PINCTRL_PIN(TZ1090_PIN_GAIN5, "gain5"), |
| PINCTRL_PIN(TZ1090_PIN_GAIN6, "gain6"), |
| PINCTRL_PIN(TZ1090_PIN_GAIN7, "gain7"), |
| PINCTRL_PIN(TZ1090_PIN_ANT_SEL0, "ant_sel0"), |
| PINCTRL_PIN(TZ1090_PIN_ANT_SEL1, "ant_sel1"), |
| PINCTRL_PIN(TZ1090_PIN_SDH_CLK_IN, "sdh_clk_in"), |
| |
| /* Non-GPIO pins */ |
| PINCTRL_PIN(TZ1090_PIN_TCK, "tck"), |
| PINCTRL_PIN(TZ1090_PIN_TRST, "trst"), |
| PINCTRL_PIN(TZ1090_PIN_TDI, "tdi"), |
| PINCTRL_PIN(TZ1090_PIN_TDO, "tdo"), |
| PINCTRL_PIN(TZ1090_PIN_TMS, "tms"), |
| PINCTRL_PIN(TZ1090_PIN_CLK_OUT0, "clk_out0"), |
| PINCTRL_PIN(TZ1090_PIN_CLK_OUT1, "clk_out1"), |
| }; |
| |
| /* Pins in each pin group */ |
| |
| static const unsigned int spi1_cs2_pins[] = { |
| TZ1090_PIN_SPI1_CS2, |
| }; |
| |
| static const unsigned int pdm_d_pins[] = { |
| TZ1090_PIN_PDM_D, |
| }; |
| |
| static const unsigned int tft_pins[] = { |
| TZ1090_PIN_TFT_RED0, |
| TZ1090_PIN_TFT_RED1, |
| TZ1090_PIN_TFT_RED2, |
| TZ1090_PIN_TFT_RED3, |
| TZ1090_PIN_TFT_RED4, |
| TZ1090_PIN_TFT_RED5, |
| TZ1090_PIN_TFT_RED6, |
| TZ1090_PIN_TFT_RED7, |
| TZ1090_PIN_TFT_GREEN0, |
| TZ1090_PIN_TFT_GREEN1, |
| TZ1090_PIN_TFT_GREEN2, |
| TZ1090_PIN_TFT_GREEN3, |
| TZ1090_PIN_TFT_GREEN4, |
| TZ1090_PIN_TFT_GREEN5, |
| TZ1090_PIN_TFT_GREEN6, |
| TZ1090_PIN_TFT_GREEN7, |
| TZ1090_PIN_TFT_BLUE0, |
| TZ1090_PIN_TFT_BLUE1, |
| TZ1090_PIN_TFT_BLUE2, |
| TZ1090_PIN_TFT_BLUE3, |
| TZ1090_PIN_TFT_BLUE4, |
| TZ1090_PIN_TFT_BLUE5, |
| TZ1090_PIN_TFT_BLUE6, |
| TZ1090_PIN_TFT_BLUE7, |
| TZ1090_PIN_TFT_VDDEN_GD, |
| TZ1090_PIN_TFT_PANELCLK, |
| TZ1090_PIN_TFT_BLANK_LS, |
| TZ1090_PIN_TFT_VSYNC_NS, |
| TZ1090_PIN_TFT_HSYNC_NR, |
| TZ1090_PIN_TFT_VD12ACB, |
| TZ1090_PIN_TFT_PWRSAVE, |
| }; |
| |
| static const unsigned int afe_pins[] = { |
| TZ1090_PIN_TX_ON, |
| TZ1090_PIN_RX_ON, |
| TZ1090_PIN_PLL_ON, |
| TZ1090_PIN_PA_ON, |
| TZ1090_PIN_RX_HP, |
| TZ1090_PIN_ANT_SEL0, |
| TZ1090_PIN_ANT_SEL1, |
| TZ1090_PIN_GAIN0, |
| TZ1090_PIN_GAIN1, |
| TZ1090_PIN_GAIN2, |
| TZ1090_PIN_GAIN3, |
| TZ1090_PIN_GAIN4, |
| TZ1090_PIN_GAIN5, |
| TZ1090_PIN_GAIN6, |
| TZ1090_PIN_GAIN7, |
| }; |
| |
| static const unsigned int sdio_pins[] = { |
| TZ1090_PIN_SDIO_CLK, |
| TZ1090_PIN_SDIO_CMD, |
| TZ1090_PIN_SDIO_D0, |
| TZ1090_PIN_SDIO_D1, |
| TZ1090_PIN_SDIO_D2, |
| TZ1090_PIN_SDIO_D3, |
| }; |
| |
| static const unsigned int sdh_pins[] = { |
| TZ1090_PIN_SDH_CD, |
| TZ1090_PIN_SDH_WP, |
| TZ1090_PIN_SDH_CLK_IN, |
| }; |
| |
| static const unsigned int spi0_pins[] = { |
| TZ1090_PIN_SPI0_MCLK, |
| TZ1090_PIN_SPI0_CS0, |
| TZ1090_PIN_SPI0_CS1, |
| TZ1090_PIN_SPI0_CS2, |
| TZ1090_PIN_SPI0_DOUT, |
| TZ1090_PIN_SPI0_DIN, |
| }; |
| |
| static const unsigned int spi1_pins[] = { |
| TZ1090_PIN_SPI1_MCLK, |
| TZ1090_PIN_SPI1_CS0, |
| TZ1090_PIN_SPI1_CS1, |
| TZ1090_PIN_SPI1_CS2, |
| TZ1090_PIN_SPI1_DOUT, |
| TZ1090_PIN_SPI1_DIN, |
| }; |
| |
| static const unsigned int uart0_pins[] = { |
| TZ1090_PIN_UART0_RTS, |
| TZ1090_PIN_UART0_CTS, |
| TZ1090_PIN_UART0_TXD, |
| TZ1090_PIN_UART0_RXD, |
| }; |
| |
| static const unsigned int uart1_pins[] = { |
| TZ1090_PIN_UART1_TXD, |
| TZ1090_PIN_UART1_RXD, |
| }; |
| |
| static const unsigned int uart_pins[] = { |
| TZ1090_PIN_UART1_TXD, |
| TZ1090_PIN_UART1_RXD, |
| TZ1090_PIN_UART0_RTS, |
| TZ1090_PIN_UART0_CTS, |
| TZ1090_PIN_UART0_TXD, |
| TZ1090_PIN_UART0_RXD, |
| }; |
| |
| static const unsigned int scb0_pins[] = { |
| TZ1090_PIN_SCB0_SDAT, |
| TZ1090_PIN_SCB0_SCLK, |
| }; |
| |
| static const unsigned int scb1_pins[] = { |
| TZ1090_PIN_SCB1_SDAT, |
| TZ1090_PIN_SCB1_SCLK, |
| }; |
| |
| static const unsigned int scb2_pins[] = { |
| TZ1090_PIN_SCB2_SDAT, |
| TZ1090_PIN_SCB2_SCLK, |
| }; |
| |
| static const unsigned int i2s_pins[] = { |
| TZ1090_PIN_I2S_MCLK, |
| TZ1090_PIN_I2S_BCLK_OUT, |
| TZ1090_PIN_I2S_LRCLK_OUT, |
| TZ1090_PIN_I2S_DOUT0, |
| TZ1090_PIN_I2S_DOUT1, |
| TZ1090_PIN_I2S_DOUT2, |
| TZ1090_PIN_I2S_DIN, |
| }; |
| |
| static const unsigned int jtag_pins[] = { |
| TZ1090_PIN_TCK, |
| TZ1090_PIN_TRST, |
| TZ1090_PIN_TDI, |
| TZ1090_PIN_TDO, |
| TZ1090_PIN_TMS, |
| }; |
| |
| /* Pins in each drive pin group */ |
| |
| static const unsigned int drive_sdio_pins[] = { |
| TZ1090_PIN_SDIO_CLK, |
| TZ1090_PIN_SDIO_CMD, |
| TZ1090_PIN_SDIO_D0, |
| TZ1090_PIN_SDIO_D1, |
| TZ1090_PIN_SDIO_D2, |
| TZ1090_PIN_SDIO_D3, |
| TZ1090_PIN_SDH_WP, |
| TZ1090_PIN_SDH_CD, |
| TZ1090_PIN_SDH_CLK_IN, |
| }; |
| |
| static const unsigned int drive_i2s_pins[] = { |
| TZ1090_PIN_CLK_OUT1, |
| TZ1090_PIN_I2S_DIN, |
| TZ1090_PIN_I2S_DOUT0, |
| TZ1090_PIN_I2S_DOUT1, |
| TZ1090_PIN_I2S_DOUT2, |
| TZ1090_PIN_I2S_LRCLK_OUT, |
| TZ1090_PIN_I2S_BCLK_OUT, |
| TZ1090_PIN_I2S_MCLK, |
| }; |
| |
| static const unsigned int drive_scb0_pins[] = { |
| TZ1090_PIN_SCB0_SCLK, |
| TZ1090_PIN_SCB0_SDAT, |
| TZ1090_PIN_PDM_D, |
| TZ1090_PIN_PDM_C, |
| }; |
| |
| static const unsigned int drive_pdm_pins[] = { |
| TZ1090_PIN_CLK_OUT0, |
| TZ1090_PIN_PDM_B, |
| TZ1090_PIN_PDM_A, |
| }; |
| |
| /* Pin groups each function can be muxed to */ |
| |
| /* |
| * The magic "perip" function allows otherwise non-muxing pins to be enabled in |
| * peripheral mode. |
| */ |
| static const char * const perip_groups[] = { |
| /* non-muxing convenient gpio pingroups */ |
| "uart", |
| "uart0", |
| "uart1", |
| "spi0", |
| "spi1", |
| "scb0", |
| "scb1", |
| "scb2", |
| "i2s", |
| /* individual pins not part of a pin mux group */ |
| "spi0_mclk", |
| "spi0_cs0", |
| "spi0_cs1", |
| "spi0_cs2", |
| "spi0_dout", |
| "spi0_din", |
| "spi1_mclk", |
| "spi1_cs0", |
| "spi1_cs1", |
| "spi1_dout", |
| "spi1_din", |
| "uart0_rxd", |
| "uart0_txd", |
| "uart0_cts", |
| "uart0_rts", |
| "uart1_rxd", |
| "uart1_txd", |
| "scb0_sdat", |
| "scb0_sclk", |
| "scb1_sdat", |
| "scb1_sclk", |
| "scb2_sdat", |
| "scb2_sclk", |
| "i2s_mclk", |
| "i2s_bclk_out", |
| "i2s_lrclk_out", |
| "i2s_dout0", |
| "i2s_dout1", |
| "i2s_dout2", |
| "i2s_din", |
| "pdm_a", |
| "pdm_b", |
| "pdm_c", |
| }; |
| |
| static const char * const sdh_sdio_groups[] = { |
| "sdh", |
| "sdio", |
| /* sdh pins */ |
| "sdh_cd", |
| "sdh_wp", |
| "sdh_clk_in", |
| /* sdio pins */ |
| "sdio_clk", |
| "sdio_cmd", |
| "sdio_d0", |
| "sdio_d1", |
| "sdio_d2", |
| "sdio_d3", |
| }; |
| |
| static const char * const spi1_cs2_groups[] = { |
| "spi1_cs2", |
| }; |
| |
| static const char * const pdm_dac_groups[] = { |
| "pdm_d", |
| }; |
| |
| static const char * const usb_vbus_groups[] = { |
| "spi1_cs2", |
| "pdm_d", |
| }; |
| |
| static const char * const afe_groups[] = { |
| "afe", |
| /* afe pins */ |
| "tx_on", |
| "rx_on", |
| "pll_on", |
| "pa_on", |
| "rx_hp", |
| "ant_sel0", |
| "ant_sel1", |
| "gain0", |
| "gain1", |
| "gain2", |
| "gain3", |
| "gain4", |
| "gain5", |
| "gain6", |
| "gain7", |
| }; |
| |
| static const char * const tft_groups[] = { |
| "tft", |
| /* tft pins */ |
| "tft_red0", |
| "tft_red1", |
| "tft_red2", |
| "tft_red3", |
| "tft_red4", |
| "tft_red5", |
| "tft_red6", |
| "tft_red7", |
| "tft_green0", |
| "tft_green1", |
| "tft_green2", |
| "tft_green3", |
| "tft_green4", |
| "tft_green5", |
| "tft_green6", |
| "tft_green7", |
| "tft_blue0", |
| "tft_blue1", |
| "tft_blue2", |
| "tft_blue3", |
| "tft_blue4", |
| "tft_blue5", |
| "tft_blue6", |
| "tft_blue7", |
| "tft_vdden_gd", |
| "tft_panelclk", |
| "tft_blank_ls", |
| "tft_vsync_ns", |
| "tft_hsync_nr", |
| "tft_vd12acb", |
| "tft_pwrsave", |
| }; |
| |
| /* Mux functions that can be used by a mux */ |
| |
| enum tz1090_mux { |
| /* internal placeholder */ |
| TZ1090_MUX_NA = -1, |
| /* magic per-non-muxing-GPIO-pin peripheral mode mux */ |
| TZ1090_MUX_PERIP, |
| /* SDH/SDIO mux */ |
| TZ1090_MUX_SDH, |
| TZ1090_MUX_SDIO, |
| /* USB_VBUS muxes */ |
| TZ1090_MUX_SPI1_CS2, |
| TZ1090_MUX_PDM_DAC, |
| TZ1090_MUX_USB_VBUS, |
| /* AFE mux */ |
| TZ1090_MUX_AFE, |
| TZ1090_MUX_TS_OUT_0, |
| /* EXT_DAC mux */ |
| TZ1090_MUX_DAC, |
| TZ1090_MUX_NOT_IQADC_STB, |
| TZ1090_MUX_IQDAC_STB, |
| /* TFT mux */ |
| TZ1090_MUX_TFT, |
| TZ1090_MUX_EXT_DAC, |
| TZ1090_MUX_TS_OUT_1, |
| TZ1090_MUX_LCD_TRACE, |
| TZ1090_MUX_PHY_RINGOSC, |
| }; |
| |
| #define FUNCTION(mux, fname, group) \ |
| [(TZ1090_MUX_ ## mux)] = { \ |
| .name = #fname, \ |
| .groups = group##_groups, \ |
| .ngroups = ARRAY_SIZE(group##_groups), \ |
| } |
| /* For intermediate functions with submuxes */ |
| #define NULL_FUNCTION(mux, fname) \ |
| [(TZ1090_MUX_ ## mux)] = { \ |
| .name = #fname, \ |
| } |
| |
| /* Must correlate with enum tz1090_mux */ |
| static const struct tz1090_function tz1090_functions[] = { |
| /* FUNCTION function name pingroups */ |
| FUNCTION(PERIP, perip, perip), |
| FUNCTION(SDH, sdh, sdh_sdio), |
| FUNCTION(SDIO, sdio, sdh_sdio), |
| FUNCTION(SPI1_CS2, spi1_cs2, spi1_cs2), |
| FUNCTION(PDM_DAC, pdm_dac, pdm_dac), |
| FUNCTION(USB_VBUS, usb_vbus, usb_vbus), |
| FUNCTION(AFE, afe, afe), |
| FUNCTION(TS_OUT_0, ts_out_0, afe), |
| FUNCTION(DAC, ext_dac, tft), |
| FUNCTION(NOT_IQADC_STB, not_iqadc_stb, tft), |
| FUNCTION(IQDAC_STB, iqdac_stb, tft), |
| FUNCTION(TFT, tft, tft), |
| NULL_FUNCTION(EXT_DAC, _ext_dac), |
| FUNCTION(TS_OUT_1, ts_out_1, tft), |
| FUNCTION(LCD_TRACE, lcd_trace, tft), |
| FUNCTION(PHY_RINGOSC, phy_ringosc, tft), |
| }; |
| |
| /* Sub muxes */ |
| |
| /** |
| * MUX() - Initialise a mux description. |
| * @f0: Function 0 (TZ1090_MUX_ is prepended, NA for none) |
| * @f1: Function 1 (TZ1090_MUX_ is prepended, NA for none) |
| * @f2: Function 2 (TZ1090_MUX_ is prepended, NA for none) |
| * @f3: Function 3 (TZ1090_MUX_ is prepended, NA for none) |
| * @f4: Function 4 (TZ1090_MUX_ is prepended, NA for none) |
| * @mux_r: Mux register (REG_PINCTRL_ is prepended) |
| * @mux_b: Bit number in register that the mux field begins |
| * @mux_w: Width of mux field in register |
| */ |
| #define MUX(f0, f1, f2, f3, f4, mux_r, mux_b, mux_w) \ |
| { \ |
| .funcs = { \ |
| TZ1090_MUX_ ## f0, \ |
| TZ1090_MUX_ ## f1, \ |
| TZ1090_MUX_ ## f2, \ |
| TZ1090_MUX_ ## f3, \ |
| TZ1090_MUX_ ## f4, \ |
| }, \ |
| .reg = (REG_PINCTRL_ ## mux_r), \ |
| .bit = (mux_b), \ |
| .width = (mux_w), \ |
| } |
| |
| /** |
| * DEFINE_SUBMUX() - Defines a submux description separate from a pin group. |
| * @mux: Mux name (_submux is appended) |
| * @f0: Function 0 (TZ1090_MUX_ is prepended, NA for none) |
| * @f1: Function 1 (TZ1090_MUX_ is prepended, NA for none) |
| * @f2: Function 2 (TZ1090_MUX_ is prepended, NA for none) |
| * @f3: Function 3 (TZ1090_MUX_ is prepended, NA for none) |
| * @f4: Function 4 (TZ1090_MUX_ is prepended, NA for none) |
| * @mux_r: Mux register (REG_PINCTRL_ is prepended) |
| * @mux_b: Bit number in register that the mux field begins |
| * @mux_w: Width of mux field in register |
| * |
| * A sub mux is a nested mux that can be bound to a magic function number used |
| * by another mux description. For example value 4 of the top level mux might |
| * correspond to a function which has a submux pointed to in tz1090_submux[]. |
| * The outer mux can then take on any function in the top level mux or the |
| * submux, and if a submux function is chosen both muxes are updated to route |
| * the signal from the submux. |
| * |
| * The submux can be defined with DEFINE_SUBMUX and pointed to from |
| * tz1090_submux[] using SUBMUX. |
| */ |
| #define DEFINE_SUBMUX(mux, f0, f1, f2, f3, f4, mux_r, mux_b, mux_w) \ |
| static struct tz1090_muxdesc mux ## _submux = \ |
| MUX(f0, f1, f2, f3, f4, mux_r, mux_b, mux_w) |
| |
| /** |
| * SUBMUX() - Link a submux to a function number. |
| * @f: Function name (TZ1090_MUX_ is prepended) |
| * @submux: Submux name (_submux is appended) |
| * |
| * For use in tz1090_submux[] initialisation to link an intermediate function |
| * number to a particular submux description. It indicates that when the |
| * function is chosen the signal is connected to the submux. |
| */ |
| #define SUBMUX(f, submux) [(TZ1090_MUX_ ## f)] = &(submux ## _submux) |
| |
| /** |
| * MUX_PG() - Initialise a pin group with mux control |
| * @pg_name: Pin group name (stringified, _pins appended to get pins array) |
| * @f0: Function 0 (TZ1090_MUX_ is prepended, NA for none) |
| * @f1: Function 1 (TZ1090_MUX_ is prepended, NA for none) |
| * @f2: Function 2 (TZ1090_MUX_ is prepended, NA for none) |
| * @f3: Function 3 (TZ1090_MUX_ is prepended, NA for none) |
| * @f4: Function 4 (TZ1090_MUX_ is prepended, NA for none) |
| * @mux_r: Mux register (REG_PINCTRL_ is prepended) |
| * @mux_b: Bit number in register that the mux field begins |
| * @mux_w: Width of mux field in register |
| */ |
| #define MUX_PG(pg_name, f0, f1, f2, f3, f4, \ |
| mux_r, mux_b, mux_w) \ |
| { \ |
| .name = #pg_name, \ |
| .pins = pg_name##_pins, \ |
| .npins = ARRAY_SIZE(pg_name##_pins), \ |
| .mux = MUX(f0, f1, f2, f3, f4, \ |
| mux_r, mux_b, mux_w), \ |
| } |
| |
| /** |
| * SIMPLE_PG() - Initialise a simple convenience pin group |
| * @pg_name: Pin group name (stringified, _pins appended to get pins array) |
| * |
| * A simple pin group is simply used for binding pins together so they can be |
| * referred to by a single name instead of having to list every pin |
| * individually. |
| */ |
| #define SIMPLE_PG(pg_name) \ |
| { \ |
| .name = #pg_name, \ |
| .pins = pg_name##_pins, \ |
| .npins = ARRAY_SIZE(pg_name##_pins), \ |
| } |
| |
| /** |
| * DRV_PG() - Initialise a pin group with drive control |
| * @pg_name: Pin group name (stringified, _pins appended to get pins array) |
| * @slw_b: Slew register bit. |
| * The same bit is used for Schmitt, and Drive (*2). |
| */ |
| #define DRV_PG(pg_name, slw_b) \ |
| { \ |
| .name = #pg_name, \ |
| .pins = pg_name##_pins, \ |
| .npins = ARRAY_SIZE(pg_name##_pins), \ |
| .drv = true, \ |
| .slw_bit = (slw_b), \ |
| } |
| |
| /* |
| * Define main muxing pin groups |
| */ |
| |
| /* submuxes */ |
| |
| /* name f0, f1, f2, f3, f4, mux r/b/w */ |
| DEFINE_SUBMUX(ext_dac, DAC, NOT_IQADC_STB, IQDAC_STB, NA, NA, IF_CTL, 6, 2); |
| |
| /* bind submuxes to internal functions */ |
| static struct tz1090_muxdesc *tz1090_submux[] = { |
| SUBMUX(EXT_DAC, ext_dac), |
| }; |
| |
| /* |
| * These are the pin mux groups. Pin muxing can be enabled and disabled for each |
| * pin individually so these groups are internal. The mapping of pins to pin mux |
| * group is below (tz1090_mux_pins). |
| */ |
| static struct tz1090_pingroup tz1090_mux_groups[] = { |
| /* Muxing pin groups */ |
| /* pg_name, f0, f1, f2, f3, f4, mux r/b/w */ |
| MUX_PG(sdh, SDH, SDIO, NA, NA, NA, IF_CTL, 20, 2), |
| MUX_PG(sdio, SDIO, SDH, NA, NA, NA, IF_CTL, 16, 2), |
| MUX_PG(spi1_cs2, SPI1_CS2, USB_VBUS, NA, NA, NA, IF_CTL, 10, 2), |
| MUX_PG(pdm_d, PDM_DAC, USB_VBUS, NA, NA, NA, IF_CTL, 8, 2), |
| MUX_PG(afe, AFE, TS_OUT_0, NA, NA, NA, IF_CTL, 4, 2), |
| MUX_PG(tft, TFT, EXT_DAC, TS_OUT_1, LCD_TRACE, PHY_RINGOSC, IF_CTL, 0, 3), |
| }; |
| |
| /* |
| * This is the mapping from GPIO pins to pin mux groups in tz1090_mux_groups[]. |
| * Pins which aren't muxable to multiple peripherals are set to |
| * TZ1090_MUX_GROUP_MAX to enable the "perip" function to enable/disable |
| * peripheral control of the pin. |
| * |
| * This array is initialised in tz1090_init_mux_pins(). |
| */ |
| static u8 tz1090_mux_pins[NUM_GPIOS]; |
| |
| /* TZ1090_MUX_GROUP_MAX is used in tz1090_mux_pins[] for non-muxing pins */ |
| #define TZ1090_MUX_GROUP_MAX ARRAY_SIZE(tz1090_mux_groups) |
| |
| /** |
| * tz1090_init_mux_pins() - Initialise GPIO pin to mux group mapping. |
| * |
| * Initialises the tz1090_mux_pins[] array to be the inverse of the pin lists in |
| * each pin mux group in tz1090_mux_groups[]. |
| * |
| * It is assumed that no pin mux groups overlap (share pins). |
| */ |
| static void __init tz1090_init_mux_pins(void) |
| { |
| unsigned int g, p; |
| const struct tz1090_pingroup *grp; |
| const unsigned int *pin; |
| |
| for (p = 0; p < NUM_GPIOS; ++p) |
| tz1090_mux_pins[p] = TZ1090_MUX_GROUP_MAX; |
| |
| grp = tz1090_mux_groups; |
| for (g = 0, grp = tz1090_mux_groups; |
| g < ARRAY_SIZE(tz1090_mux_groups); ++g, ++grp) |
| for (pin = grp->pins, p = 0; p < grp->npins; ++p, ++pin) |
| tz1090_mux_pins[*pin] = g; |
| } |
| |
| /* |
| * These are the externally visible pin groups. Some of them allow group control |
| * of drive configuration. Some are just simple convenience pingroups. All the |
| * internal pin mux groups in tz1090_mux_groups[] are mirrored here with the |
| * same pins. |
| * Pseudo pin groups follow in the group numbers after this array for each GPIO |
| * pin. Any group used for muxing must have all pins belonging to the same pin |
| * mux group. |
| */ |
| static struct tz1090_pingroup tz1090_groups[] = { |
| /* Pin groups with drive control (with no out of place pins) */ |
| /* pg_name, slw/schmitt/drv b */ |
| DRV_PG(jtag, 11 /* 11, 22 */), |
| DRV_PG(tft, 10 /* 10, 20 */), |
| DRV_PG(scb2, 9 /* 9, 18 */), |
| DRV_PG(spi0, 7 /* 7, 14 */), |
| DRV_PG(uart, 5 /* 5, 10 */), |
| DRV_PG(scb1, 4 /* 4, 8 */), |
| DRV_PG(spi1, 3 /* 3, 6 */), |
| DRV_PG(afe, 0 /* 0, 0 */), |
| |
| /* |
| * Drive specific pin groups (with odd combinations of pins which makes |
| * the pin group naming somewhat arbitrary) |
| */ |
| /* pg_name, slw/schmitt/drv b */ |
| DRV_PG(drive_sdio, 8 /* 8, 16 */), /* sdio_* + sdh_* */ |
| DRV_PG(drive_i2s, 6 /* 6, 12 */), /* i2s_* + clk_out1 */ |
| DRV_PG(drive_scb0, 2 /* 2, 4 */), /* scb0_* + pdm_{c,d} */ |
| DRV_PG(drive_pdm, 1 /* 1, 2 */), /* pdm_{a,b} + clk_out0 */ |
| |
| /* Convenience pin groups */ |
| /* pg_name */ |
| SIMPLE_PG(uart0), |
| SIMPLE_PG(uart1), |
| SIMPLE_PG(scb0), |
| SIMPLE_PG(i2s), |
| SIMPLE_PG(sdh), |
| SIMPLE_PG(sdio), |
| |
| /* pseudo-pingroups for each GPIO pin follow */ |
| }; |
| |
| /** |
| * struct tz1090_pmx - Private pinctrl data |
| * @dev: Platform device |
| * @pctl: Pin control device |
| * @regs: Register region |
| * @lock: Lock protecting coherency of pin_en, gpio_en, and SELECT regs |
| * @pin_en: Pins that have been enabled (32 pins packed into each element) |
| * @gpio_en: GPIOs that have been enabled (32 pins packed into each element) |
| */ |
| struct tz1090_pmx { |
| struct device *dev; |
| struct pinctrl_dev *pctl; |
| void __iomem *regs; |
| spinlock_t lock; |
| u32 pin_en[3]; |
| u32 gpio_en[3]; |
| }; |
| |
| static inline u32 pmx_read(struct tz1090_pmx *pmx, u32 reg) |
| { |
| return ioread32(pmx->regs + reg); |
| } |
| |
| static inline void pmx_write(struct tz1090_pmx *pmx, u32 val, u32 reg) |
| { |
| iowrite32(val, pmx->regs + reg); |
| } |
| |
| /* |
| * Pin control operations |
| */ |
| |
| /* each GPIO pin has it's own pseudo pingroup containing only itself */ |
| |
| static int tz1090_pinctrl_get_groups_count(struct pinctrl_dev *pctldev) |
| { |
| return ARRAY_SIZE(tz1090_groups) + NUM_GPIOS; |
| } |
| |
| static const char *tz1090_pinctrl_get_group_name(struct pinctrl_dev *pctldev, |
| unsigned int group) |
| { |
| if (group < ARRAY_SIZE(tz1090_groups)) { |
| /* normal pingroup */ |
| return tz1090_groups[group].name; |
| } else { |
| /* individual gpio pin pseudo-pingroup */ |
| unsigned int pin = group - ARRAY_SIZE(tz1090_groups); |
| return tz1090_pins[pin].name; |
| } |
| } |
| |
| static int tz1090_pinctrl_get_group_pins(struct pinctrl_dev *pctldev, |
| unsigned int group, |
| const unsigned int **pins, |
| unsigned int *num_pins) |
| { |
| if (group < ARRAY_SIZE(tz1090_groups)) { |
| /* normal pingroup */ |
| *pins = tz1090_groups[group].pins; |
| *num_pins = tz1090_groups[group].npins; |
| } else { |
| /* individual gpio pin pseudo-pingroup */ |
| unsigned int pin = group - ARRAY_SIZE(tz1090_groups); |
| *pins = &tz1090_pins[pin].number; |
| *num_pins = 1; |
| } |
| |
| return 0; |
| } |
| |
| #ifdef CONFIG_DEBUG_FS |
| static void tz1090_pinctrl_pin_dbg_show(struct pinctrl_dev *pctldev, |
| struct seq_file *s, |
| unsigned int offset) |
| { |
| seq_printf(s, " %s", dev_name(pctldev->dev)); |
| } |
| #endif |
| |
| static int reserve_map(struct device *dev, struct pinctrl_map **map, |
| unsigned int *reserved_maps, unsigned int *num_maps, |
| unsigned int reserve) |
| { |
| unsigned int old_num = *reserved_maps; |
| unsigned int new_num = *num_maps + reserve; |
| struct pinctrl_map *new_map; |
| |
| if (old_num >= new_num) |
| return 0; |
| |
| new_map = krealloc(*map, sizeof(*new_map) * new_num, GFP_KERNEL); |
| if (!new_map) { |
| dev_err(dev, "krealloc(map) failed\n"); |
| return -ENOMEM; |
| } |
| |
| memset(new_map + old_num, 0, (new_num - old_num) * sizeof(*new_map)); |
| |
| *map = new_map; |
| *reserved_maps = new_num; |
| |
| return 0; |
| } |
| |
| static int add_map_mux(struct pinctrl_map **map, unsigned int *reserved_maps, |
| unsigned int *num_maps, const char *group, |
| const char *function) |
| { |
| if (WARN_ON(*num_maps == *reserved_maps)) |
| return -ENOSPC; |
| |
| (*map)[*num_maps].type = PIN_MAP_TYPE_MUX_GROUP; |
| (*map)[*num_maps].data.mux.group = group; |
| (*map)[*num_maps].data.mux.function = function; |
| (*num_maps)++; |
| |
| return 0; |
| } |
| |
| static int add_map_configs(struct device *dev, |
| struct pinctrl_map **map, |
| unsigned int *reserved_maps, unsigned int *num_maps, |
| const char *group, unsigned long *configs, |
| unsigned int num_configs) |
| { |
| unsigned long *dup_configs; |
| |
| if (WARN_ON(*num_maps == *reserved_maps)) |
| return -ENOSPC; |
| |
| dup_configs = kmemdup(configs, num_configs * sizeof(*dup_configs), |
| GFP_KERNEL); |
| if (!dup_configs) { |
| dev_err(dev, "kmemdup(configs) failed\n"); |
| return -ENOMEM; |
| } |
| |
| (*map)[*num_maps].type = PIN_MAP_TYPE_CONFIGS_GROUP; |
| (*map)[*num_maps].data.configs.group_or_pin = group; |
| (*map)[*num_maps].data.configs.configs = dup_configs; |
| (*map)[*num_maps].data.configs.num_configs = num_configs; |
| (*num_maps)++; |
| |
| return 0; |
| } |
| |
| static void tz1090_pinctrl_dt_free_map(struct pinctrl_dev *pctldev, |
| struct pinctrl_map *map, |
| unsigned int num_maps) |
| { |
| int i; |
| |
| for (i = 0; i < num_maps; i++) |
| if (map[i].type == PIN_MAP_TYPE_CONFIGS_GROUP) |
| kfree(map[i].data.configs.configs); |
| |
| kfree(map); |
| } |
| |
| static int tz1090_pinctrl_dt_subnode_to_map(struct device *dev, |
| struct device_node *np, |
| struct pinctrl_map **map, |
| unsigned int *reserved_maps, |
| unsigned int *num_maps) |
| { |
| int ret; |
| const char *function; |
| unsigned long *configs = NULL; |
| unsigned int num_configs = 0; |
| unsigned int reserve; |
| struct property *prop; |
| const char *group; |
| |
| ret = of_property_read_string(np, "tz1090,function", &function); |
| if (ret < 0) { |
| /* EINVAL=missing, which is fine since it's optional */ |
| if (ret != -EINVAL) |
| dev_err(dev, "could not parse property function\n"); |
| function = NULL; |
| } |
| |
| ret = pinconf_generic_parse_dt_config(np, &configs, &num_configs); |
| if (ret) |
| return ret; |
| |
| reserve = 0; |
| if (function != NULL) |
| reserve++; |
| if (num_configs) |
| reserve++; |
| ret = of_property_count_strings(np, "tz1090,pins"); |
| if (ret < 0) { |
| dev_err(dev, "could not parse property pins\n"); |
| goto exit; |
| } |
| reserve *= ret; |
| |
| ret = reserve_map(dev, map, reserved_maps, num_maps, reserve); |
| if (ret < 0) |
| goto exit; |
| |
| of_property_for_each_string(np, "tz1090,pins", prop, group) { |
| if (function) { |
| ret = add_map_mux(map, reserved_maps, num_maps, |
| group, function); |
| if (ret < 0) |
| goto exit; |
| } |
| |
| if (num_configs) { |
| ret = add_map_configs(dev, map, reserved_maps, |
| num_maps, group, configs, |
| num_configs); |
| if (ret < 0) |
| goto exit; |
| } |
| } |
| |
| ret = 0; |
| |
| exit: |
| kfree(configs); |
| return ret; |
| } |
| |
| static int tz1090_pinctrl_dt_node_to_map(struct pinctrl_dev *pctldev, |
| struct device_node *np_config, |
| struct pinctrl_map **map, |
| unsigned int *num_maps) |
| { |
| unsigned int reserved_maps; |
| struct device_node *np; |
| int ret; |
| |
| reserved_maps = 0; |
| *map = NULL; |
| *num_maps = 0; |
| |
| for_each_child_of_node(np_config, np) { |
| ret = tz1090_pinctrl_dt_subnode_to_map(pctldev->dev, np, map, |
| &reserved_maps, |
| num_maps); |
| if (ret < 0) { |
| tz1090_pinctrl_dt_free_map(pctldev, *map, *num_maps); |
| return ret; |
| } |
| } |
| |
| return 0; |
| } |
| |
| static struct pinctrl_ops tz1090_pinctrl_ops = { |
| .get_groups_count = tz1090_pinctrl_get_groups_count, |
| .get_group_name = tz1090_pinctrl_get_group_name, |
| .get_group_pins = tz1090_pinctrl_get_group_pins, |
| #ifdef CONFIG_DEBUG_FS |
| .pin_dbg_show = tz1090_pinctrl_pin_dbg_show, |
| #endif |
| .dt_node_to_map = tz1090_pinctrl_dt_node_to_map, |
| .dt_free_map = tz1090_pinctrl_dt_free_map, |
| }; |
| |
| /* |
| * Pin mux operations |
| */ |
| |
| static int tz1090_pinctrl_get_funcs_count(struct pinctrl_dev *pctldev) |
| { |
| return ARRAY_SIZE(tz1090_functions); |
| } |
| |
| static const char *tz1090_pinctrl_get_func_name(struct pinctrl_dev *pctldev, |
| unsigned int function) |
| { |
| return tz1090_functions[function].name; |
| } |
| |
| static int tz1090_pinctrl_get_func_groups(struct pinctrl_dev *pctldev, |
| unsigned int function, |
| const char * const **groups, |
| unsigned int * const num_groups) |
| { |
| /* pingroup functions */ |
| *groups = tz1090_functions[function].groups; |
| *num_groups = tz1090_functions[function].ngroups; |
| return 0; |
| } |
| |
| /** |
| * tz1090_pinctrl_select() - update bit in SELECT register |
| * @pmx: Pinmux data |
| * @pin: Pin number (must be within GPIO range) |
| */ |
| static void tz1090_pinctrl_select(struct tz1090_pmx *pmx, |
| unsigned int pin) |
| { |
| u32 reg, reg_shift, select, val; |
| unsigned int pmx_index, pmx_shift; |
| unsigned long flags; |
| |
| /* uses base 32 instead of base 30 */ |
| pmx_index = pin >> 5; |
| pmx_shift = pin & 0x1f; |
| |
| /* select = !perip || gpio */ |
| select = ((~pmx->pin_en[pmx_index] | |
| pmx->gpio_en[pmx_index]) >> pmx_shift) & 1; |
| |
| /* find register and bit offset (base 30) */ |
| reg = REG_PINCTRL_SELECT + 4*(pin / 30); |
| reg_shift = pin % 30; |
| |
| /* modify gpio select bit */ |
| __global_lock2(flags); |
| val = pmx_read(pmx, reg); |
| val &= ~BIT(reg_shift); |
| val |= select << reg_shift; |
| pmx_write(pmx, val, reg); |
| __global_unlock2(flags); |
| } |
| |
| /** |
| * tz1090_pinctrl_gpio_select() - enable/disable GPIO usage for a pin |
| * @pmx: Pinmux data |
| * @pin: Pin number |
| * @gpio_select: true to enable pin as GPIO, |
| * false to leave control to whatever function is enabled |
| * |
| * Records that GPIO usage is enabled/disabled so that enabling a function |
| * doesn't override the SELECT register bit. |
| */ |
| static void tz1090_pinctrl_gpio_select(struct tz1090_pmx *pmx, |
| unsigned int pin, |
| bool gpio_select) |
| { |
| unsigned int index, shift; |
| u32 gpio_en; |
| |
| if (pin >= NUM_GPIOS) |
| return; |
| |
| /* uses base 32 instead of base 30 */ |
| index = pin >> 5; |
| shift = pin & 0x1f; |
| |
| spin_lock(&pmx->lock); |
| |
| /* keep a record whether gpio is selected */ |
| gpio_en = pmx->gpio_en[index]; |
| gpio_en &= ~BIT(shift); |
| if (gpio_select) |
| gpio_en |= BIT(shift); |
| pmx->gpio_en[index] = gpio_en; |
| |
| /* update the select bit */ |
| tz1090_pinctrl_select(pmx, pin); |
| |
| spin_unlock(&pmx->lock); |
| } |
| |
| /** |
| * tz1090_pinctrl_perip_select() - enable/disable peripheral interface for a pin |
| * @pmx: Pinmux data |
| * @pin: Pin number |
| * @perip_select: true to enable peripheral interface when not GPIO, |
| * false to leave pin in GPIO mode |
| * |
| * Records that peripheral usage is enabled/disabled so that SELECT register can |
| * be set appropriately when GPIO is disabled. |
| */ |
| static void tz1090_pinctrl_perip_select(struct tz1090_pmx *pmx, |
| unsigned int pin, |
| bool perip_select) |
| { |
| unsigned int index, shift; |
| u32 pin_en; |
| |
| if (pin >= NUM_GPIOS) |
| return; |
| |
| /* uses base 32 instead of base 30 */ |
| index = pin >> 5; |
| shift = pin & 0x1f; |
| |
| spin_lock(&pmx->lock); |
| |
| /* keep a record whether peripheral is selected */ |
| pin_en = pmx->pin_en[index]; |
| pin_en &= ~BIT(shift); |
| if (perip_select) |
| pin_en |= BIT(shift); |
| pmx->pin_en[index] = pin_en; |
| |
| /* update the select bit */ |
| tz1090_pinctrl_select(pmx, pin); |
| |
| spin_unlock(&pmx->lock); |
| } |
| |
| /** |
| * tz1090_pinctrl_enable_mux() - Switch a pin mux group to a function. |
| * @pmx: Pinmux data |
| * @desc: Pinmux description |
| * @function: Function to switch to |
| * |
| * Enable a particular function on a pin mux group. Since pin mux descriptions |
| * are nested this function is recursive. |
| */ |
| static int tz1090_pinctrl_enable_mux(struct tz1090_pmx *pmx, |
| const struct tz1090_muxdesc *desc, |
| unsigned int function) |
| { |
| const int *fit; |
| unsigned long flags; |
| int mux; |
| unsigned int func, ret; |
| u32 reg, mask; |
| |
| /* find the mux value for this function, searching recursively */ |
| for (mux = 0, fit = desc->funcs; |
| mux < ARRAY_SIZE(desc->funcs); ++mux, ++fit) { |
| func = *fit; |
| if (func == function) |
| goto found_mux; |
| |
| /* maybe it's a sub-mux */ |
| if (func < ARRAY_SIZE(tz1090_submux) && tz1090_submux[func]) { |
| ret = tz1090_pinctrl_enable_mux(pmx, |
| tz1090_submux[func], |
| function); |
| if (!ret) |
| goto found_mux; |
| } |
| } |
| |
| return -EINVAL; |
| found_mux: |
| |
| /* Set up the mux */ |
| if (desc->width) { |
| mask = (BIT(desc->width) - 1) << desc->bit; |
| __global_lock2(flags); |
| reg = pmx_read(pmx, desc->reg); |
| reg &= ~mask; |
| reg |= (mux << desc->bit) & mask; |
| pmx_write(pmx, reg, desc->reg); |
| __global_unlock2(flags); |
| } |
| |
| return 0; |
| } |
| |
| /** |
| * tz1090_pinctrl_enable() - Enable a function on a pin group. |
| * @pctldev: Pin control data |
| * @function: Function index to enable |
| * @group: Group index to enable |
| * |
| * Enable a particular function on a group of pins. The per GPIO pin pseudo pin |
| * groups can be used (in which case the pin will be enabled in peripheral mode |
| * and if it belongs to a pin mux group the mux will be switched if it isn't |
| * already in use. Some convenience pin groups can also be used in which case |
| * the effect is the same as enabling the function on each individual pin in the |
| * group. |
| */ |
| static int tz1090_pinctrl_set_mux(struct pinctrl_dev *pctldev, |
| unsigned int function, unsigned int group) |
| { |
| struct tz1090_pmx *pmx = pinctrl_dev_get_drvdata(pctldev); |
| struct tz1090_pingroup *grp; |
| int ret; |
| unsigned int pin_num, mux_group, i, npins; |
| const unsigned int *pins; |
| |
| /* group of pins? */ |
| if (group < ARRAY_SIZE(tz1090_groups)) { |
| grp = &tz1090_groups[group]; |
| npins = grp->npins; |
| pins = grp->pins; |
| /* |
| * All pins in the group must belong to the same mux group, |
| * which allows us to just use the mux group of the first pin. |
| * By explicitly listing permitted pingroups for each function |
| * the pinmux core should ensure this is always the case. |
| */ |
| } else { |
| pin_num = group - ARRAY_SIZE(tz1090_groups); |
| npins = 1; |
| pins = &pin_num; |
| } |
| mux_group = tz1090_mux_pins[*pins]; |
| |
| /* no mux group, but can still be individually muxed to peripheral */ |
| if (mux_group >= TZ1090_MUX_GROUP_MAX) { |
| if (function == TZ1090_MUX_PERIP) |
| goto mux_pins; |
| return -EINVAL; |
| } |
| |
| /* mux group already set to a different function? */ |
| grp = &tz1090_mux_groups[mux_group]; |
| if (grp->func_count && grp->func != function) { |
| dev_err(pctldev->dev, |
| "%s: can't mux pin(s) to '%s', group already muxed to '%s'\n", |
| __func__, tz1090_functions[function].name, |
| tz1090_functions[grp->func].name); |
| return -EBUSY; |
| } |
| |
| dev_dbg(pctldev->dev, "%s: muxing %u pin(s) in '%s' to '%s'\n", |
| __func__, npins, grp->name, tz1090_functions[function].name); |
| |
| /* if first pin in mux group to be enabled, enable the group mux */ |
| if (!grp->func_count) { |
| grp->func = function; |
| ret = tz1090_pinctrl_enable_mux(pmx, &grp->mux, function); |
| if (ret) |
| return ret; |
| } |
| /* add pins to ref count and mux individually to peripheral */ |
| grp->func_count += npins; |
| mux_pins: |
| for (i = 0; i < npins; ++i) |
| tz1090_pinctrl_perip_select(pmx, pins[i], true); |
| |
| return 0; |
| } |
| |
| /** |
| * tz1090_pinctrl_gpio_request_enable() - Put pin in GPIO mode. |
| * @pctldev: Pin control data |
| * @range: GPIO range |
| * @pin: Pin number |
| * |
| * Puts a particular pin into GPIO mode, disabling peripheral control until it's |
| * disabled again. |
| */ |
| static int tz1090_pinctrl_gpio_request_enable(struct pinctrl_dev *pctldev, |
| struct pinctrl_gpio_range *range, |
| unsigned int pin) |
| { |
| struct tz1090_pmx *pmx = pinctrl_dev_get_drvdata(pctldev); |
| tz1090_pinctrl_gpio_select(pmx, pin, true); |
| return 0; |
| } |
| |
| /** |
| * tz1090_pinctrl_gpio_disable_free() - Take pin out of GPIO mode. |
| * @pctldev: Pin control data |
| * @range: GPIO range |
| * @pin: Pin number |
| * |
| * Take a particular pin out of GPIO mode. If the pin is enabled for a |
| * peripheral it will return to peripheral mode. |
| */ |
| static void tz1090_pinctrl_gpio_disable_free(struct pinctrl_dev *pctldev, |
| struct pinctrl_gpio_range *range, |
| unsigned int pin) |
| { |
| struct tz1090_pmx *pmx = pinctrl_dev_get_drvdata(pctldev); |
| tz1090_pinctrl_gpio_select(pmx, pin, false); |
| } |
| |
| static struct pinmux_ops tz1090_pinmux_ops = { |
| .get_functions_count = tz1090_pinctrl_get_funcs_count, |
| .get_function_name = tz1090_pinctrl_get_func_name, |
| .get_function_groups = tz1090_pinctrl_get_func_groups, |
| .set_mux = tz1090_pinctrl_set_mux, |
| .gpio_request_enable = tz1090_pinctrl_gpio_request_enable, |
| .gpio_disable_free = tz1090_pinctrl_gpio_disable_free, |
| }; |
| |
| /* |
| * Pin config operations |
| */ |
| |
| struct tz1090_pinconf_pullup { |
| unsigned char index; |
| unsigned char shift; |
| }; |
| |
| /* The mapping of pin to pull up/down register index and shift */ |
| static struct tz1090_pinconf_pullup tz1090_pinconf_pullup[] = { |
| {5, 22}, /* 0 - TZ1090_PIN_SDIO_CLK */ |
| {0, 14}, /* 1 - TZ1090_PIN_SDIO_CMD */ |
| {0, 6}, /* 2 - TZ1090_PIN_SDIO_D0 */ |
| {0, 8}, /* 3 - TZ1090_PIN_SDIO_D1 */ |
| {0, 10}, /* 4 - TZ1090_PIN_SDIO_D2 */ |
| {0, 12}, /* 5 - TZ1090_PIN_SDIO_D3 */ |
| {0, 2}, /* 6 - TZ1090_PIN_SDH_CD */ |
| {0, 4}, /* 7 - TZ1090_PIN_SDH_WP */ |
| {0, 16}, /* 8 - TZ1090_PIN_SPI0_MCLK */ |
| {0, 18}, /* 9 - TZ1090_PIN_SPI0_CS0 */ |
| {0, 20}, /* 10 - TZ1090_PIN_SPI0_CS1 */ |
| {0, 22}, /* 11 - TZ1090_PIN_SPI0_CS2 */ |
| {0, 24}, /* 12 - TZ1090_PIN_SPI0_DOUT */ |
| {0, 26}, /* 13 - TZ1090_PIN_SPI0_DIN */ |
| {0, 28}, /* 14 - TZ1090_PIN_SPI1_MCLK */ |
| {0, 30}, /* 15 - TZ1090_PIN_SPI1_CS0 */ |
| {1, 0}, /* 16 - TZ1090_PIN_SPI1_CS1 */ |
| {1, 2}, /* 17 - TZ1090_PIN_SPI1_CS2 */ |
| {1, 4}, /* 18 - TZ1090_PIN_SPI1_DOUT */ |
| {1, 6}, /* 19 - TZ1090_PIN_SPI1_DIN */ |
| {1, 8}, /* 20 - TZ1090_PIN_UART0_RXD */ |
| {1, 10}, /* 21 - TZ1090_PIN_UART0_TXD */ |
| {1, 12}, /* 22 - TZ1090_PIN_UART0_CTS */ |
| {1, 14}, /* 23 - TZ1090_PIN_UART0_RTS */ |
| {1, 16}, /* 24 - TZ1090_PIN_UART1_RXD */ |
| {1, 18}, /* 25 - TZ1090_PIN_UART1_TXD */ |
| {1, 20}, /* 26 - TZ1090_PIN_SCB0_SDAT */ |
| {1, 22}, /* 27 - TZ1090_PIN_SCB0_SCLK */ |
| {1, 24}, /* 28 - TZ1090_PIN_SCB1_SDAT */ |
| {1, 26}, /* 29 - TZ1090_PIN_SCB1_SCLK */ |
| |
| {1, 28}, /* 30 - TZ1090_PIN_SCB2_SDAT */ |
| {1, 30}, /* 31 - TZ1090_PIN_SCB2_SCLK */ |
| {2, 0}, /* 32 - TZ1090_PIN_I2S_MCLK */ |
| {2, 2}, /* 33 - TZ1090_PIN_I2S_BCLK_OUT */ |
| {2, 4}, /* 34 - TZ1090_PIN_I2S_LRCLK_OUT */ |
| {2, 6}, /* 35 - TZ1090_PIN_I2S_DOUT0 */ |
| {2, 8}, /* 36 - TZ1090_PIN_I2S_DOUT1 */ |
| {2, 10}, /* 37 - TZ1090_PIN_I2S_DOUT2 */ |
| {2, 12}, /* 38 - TZ1090_PIN_I2S_DIN */ |
| {4, 12}, /* 39 - TZ1090_PIN_PDM_A */ |
| {4, 14}, /* 40 - TZ1090_PIN_PDM_B */ |
| {4, 18}, /* 41 - TZ1090_PIN_PDM_C */ |
| {4, 20}, /* 42 - TZ1090_PIN_PDM_D */ |
| {2, 14}, /* 43 - TZ1090_PIN_TFT_RED0 */ |
| {2, 16}, /* 44 - TZ1090_PIN_TFT_RED1 */ |
| {2, 18}, /* 45 - TZ1090_PIN_TFT_RED2 */ |
| {2, 20}, /* 46 - TZ1090_PIN_TFT_RED3 */ |
| {2, 22}, /* 47 - TZ1090_PIN_TFT_RED4 */ |
| {2, 24}, /* 48 - TZ1090_PIN_TFT_RED5 */ |
| {2, 26}, /* 49 - TZ1090_PIN_TFT_RED6 */ |
| {2, 28}, /* 50 - TZ1090_PIN_TFT_RED7 */ |
| {2, 30}, /* 51 - TZ1090_PIN_TFT_GREEN0 */ |
| {3, 0}, /* 52 - TZ1090_PIN_TFT_GREEN1 */ |
| {3, 2}, /* 53 - TZ1090_PIN_TFT_GREEN2 */ |
| {3, 4}, /* 54 - TZ1090_PIN_TFT_GREEN3 */ |
| {3, 6}, /* 55 - TZ1090_PIN_TFT_GREEN4 */ |
| {3, 8}, /* 56 - TZ1090_PIN_TFT_GREEN5 */ |
| {3, 10}, /* 57 - TZ1090_PIN_TFT_GREEN6 */ |
| {3, 12}, /* 58 - TZ1090_PIN_TFT_GREEN7 */ |
| {3, 14}, /* 59 - TZ1090_PIN_TFT_BLUE0 */ |
| |
| {3, 16}, /* 60 - TZ1090_PIN_TFT_BLUE1 */ |
| {3, 18}, /* 61 - TZ1090_PIN_TFT_BLUE2 */ |
| {3, 20}, /* 62 - TZ1090_PIN_TFT_BLUE3 */ |
| {3, 22}, /* 63 - TZ1090_PIN_TFT_BLUE4 */ |
| {3, 24}, /* 64 - TZ1090_PIN_TFT_BLUE5 */ |
| {3, 26}, /* 65 - TZ1090_PIN_TFT_BLUE6 */ |
| {3, 28}, /* 66 - TZ1090_PIN_TFT_BLUE7 */ |
| {3, 30}, /* 67 - TZ1090_PIN_TFT_VDDEN_GD */ |
| {4, 0}, /* 68 - TZ1090_PIN_TFT_PANELCLK */ |
| {4, 2}, /* 69 - TZ1090_PIN_TFT_BLANK_LS */ |
| {4, 4}, /* 70 - TZ1090_PIN_TFT_VSYNC_NS */ |
| {4, 6}, /* 71 - TZ1090_PIN_TFT_HSYNC_NR */ |
| {4, 8}, /* 72 - TZ1090_PIN_TFT_VD12ACB */ |
| {4, 10}, /* 73 - TZ1090_PIN_TFT_PWRSAVE */ |
| {4, 24}, /* 74 - TZ1090_PIN_TX_ON */ |
| {4, 26}, /* 75 - TZ1090_PIN_RX_ON */ |
| {4, 28}, /* 76 - TZ1090_PIN_PLL_ON */ |
| {4, 30}, /* 77 - TZ1090_PIN_PA_ON */ |
| {5, 0}, /* 78 - TZ1090_PIN_RX_HP */ |
| {5, 6}, /* 79 - TZ1090_PIN_GAIN0 */ |
| {5, 8}, /* 80 - TZ1090_PIN_GAIN1 */ |
| {5, 10}, /* 81 - TZ1090_PIN_GAIN2 */ |
| {5, 12}, /* 82 - TZ1090_PIN_GAIN3 */ |
| {5, 14}, /* 83 - TZ1090_PIN_GAIN4 */ |
| {5, 16}, /* 84 - TZ1090_PIN_GAIN5 */ |
| {5, 18}, /* 85 - TZ1090_PIN_GAIN6 */ |
| {5, 20}, /* 86 - TZ1090_PIN_GAIN7 */ |
| {5, 2}, /* 87 - TZ1090_PIN_ANT_SEL0 */ |
| {5, 4}, /* 88 - TZ1090_PIN_ANT_SEL1 */ |
| {0, 0}, /* 89 - TZ1090_PIN_SDH_CLK_IN */ |
| |
| {5, 24}, /* 90 - TZ1090_PIN_TCK */ |
| {5, 26}, /* 91 - TZ1090_PIN_TRST */ |
| {5, 28}, /* 92 - TZ1090_PIN_TDI */ |
| {5, 30}, /* 93 - TZ1090_PIN_TDO */ |
| {6, 0}, /* 94 - TZ1090_PIN_TMS */ |
| {4, 16}, /* 95 - TZ1090_PIN_CLK_OUT0 */ |
| {4, 22}, /* 96 - TZ1090_PIN_CLK_OUT1 */ |
| }; |
| |
| static int tz1090_pinconf_reg(struct pinctrl_dev *pctldev, |
| unsigned int pin, |
| enum pin_config_param param, |
| bool report_err, |
| u32 *reg, u32 *width, u32 *mask, u32 *shift, |
| u32 *val) |
| { |
| struct tz1090_pinconf_pullup *pu; |
| |
| /* All supported pins have controllable input bias */ |
| switch (param) { |
| case PIN_CONFIG_BIAS_DISABLE: |
| case PIN_CONFIG_BIAS_HIGH_IMPEDANCE: |
| *val = REG_PU_PD_TRISTATE; |
| break; |
| case PIN_CONFIG_BIAS_PULL_UP: |
| *val = REG_PU_PD_UP; |
| break; |
| case PIN_CONFIG_BIAS_PULL_DOWN: |
| *val = REG_PU_PD_DOWN; |
| break; |
| case PIN_CONFIG_BIAS_BUS_HOLD: |
| *val = REG_PU_PD_REPEATER; |
| break; |
| default: |
| return -ENOTSUPP; |
| }; |
| |
| /* Only input bias parameters supported */ |
| pu = &tz1090_pinconf_pullup[pin]; |
| *reg = REG_PINCTRL_PU_PD + 4*pu->index; |
| *shift = pu->shift; |
| *width = 2; |
| |
| /* Calculate field information */ |
| *mask = (BIT(*width) - 1) << *shift; |
| |
| return 0; |
| } |
| |
| static int tz1090_pinconf_get(struct pinctrl_dev *pctldev, |
| unsigned int pin, unsigned long *config) |
| { |
| struct tz1090_pmx *pmx = pinctrl_dev_get_drvdata(pctldev); |
| enum pin_config_param param = pinconf_to_config_param(*config); |
| int ret; |
| u32 reg, width, mask, shift, val, tmp, arg; |
| |
| /* Get register information */ |
| ret = tz1090_pinconf_reg(pctldev, pin, param, true, |
| ®, &width, &mask, &shift, &val); |
| if (ret < 0) |
| return ret; |
| |
| /* Extract field from register */ |
| tmp = pmx_read(pmx, reg); |
| arg = ((tmp & mask) >> shift) == val; |
| |
| /* Config not active */ |
| if (!arg) |
| return -EINVAL; |
| |
| /* And pack config */ |
| *config = pinconf_to_config_packed(param, arg); |
| |
| return 0; |
| } |
| |
| static int tz1090_pinconf_set(struct pinctrl_dev *pctldev, |
| unsigned int pin, unsigned long *configs, |
| unsigned num_configs) |
| { |
| struct tz1090_pmx *pmx = pinctrl_dev_get_drvdata(pctldev); |
| enum pin_config_param param; |
| unsigned int arg; |
| int ret; |
| u32 reg, width, mask, shift, val, tmp; |
| unsigned long flags; |
| int i; |
| |
| for (i = 0; i < num_configs; i++) { |
| param = pinconf_to_config_param(configs[i]); |
| arg = pinconf_to_config_argument(configs[i]); |
| |
| dev_dbg(pctldev->dev, "%s(pin=%s, config=%#lx)\n", |
| __func__, tz1090_pins[pin].name, configs[i]); |
| |
| /* Get register information */ |
| ret = tz1090_pinconf_reg(pctldev, pin, param, true, |
| ®, &width, &mask, &shift, &val); |
| if (ret < 0) |
| return ret; |
| |
| /* Unpack argument and range check it */ |
| if (arg > 1) { |
| dev_dbg(pctldev->dev, "%s: arg %u out of range\n", |
| __func__, arg); |
| return -EINVAL; |
| } |
| |
| /* Write register field */ |
| __global_lock2(flags); |
| tmp = pmx_read(pmx, reg); |
| tmp &= ~mask; |
| if (arg) |
| tmp |= val << shift; |
| pmx_write(pmx, tmp, reg); |
| __global_unlock2(flags); |
| } /* for each config */ |
| |
| return 0; |
| } |
| |
| static const int tz1090_boolean_map[] = { |
| [0] = -EINVAL, |
| [1] = 1, |
| }; |
| |
| static const int tz1090_dr_map[] = { |
| [REG_DR_2mA] = 2, |
| [REG_DR_4mA] = 4, |
| [REG_DR_8mA] = 8, |
| [REG_DR_12mA] = 12, |
| }; |
| |
| static int tz1090_pinconf_group_reg(struct pinctrl_dev *pctldev, |
| const struct tz1090_pingroup *g, |
| enum pin_config_param param, |
| bool report_err, |
| u32 *reg, u32 *width, u32 *mask, u32 *shift, |
| const int **map) |
| { |
| /* Drive configuration applies in groups, but not to all groups. */ |
| if (!g->drv) { |
| if (report_err) |
| dev_dbg(pctldev->dev, |
| "%s: group %s has no drive control\n", |
| __func__, g->name); |
| return -ENOTSUPP; |
| } |
| |
| /* Find information about drive parameter's register */ |
| switch (param) { |
| case PIN_CONFIG_INPUT_SCHMITT_ENABLE: |
| *reg = REG_PINCTRL_SCHMITT; |
| *width = 1; |
| *map = tz1090_boolean_map; |
| break; |
| case PIN_CONFIG_DRIVE_STRENGTH: |
| *reg = REG_PINCTRL_DR; |
| *width = 2; |
| *map = tz1090_dr_map; |
| break; |
| default: |
| return -ENOTSUPP; |
| }; |
| |
| /* Calculate field information */ |
| *shift = g->slw_bit * *width; |
| *mask = (BIT(*width) - 1) << *shift; |
| |
| return 0; |
| } |
| |
| static int tz1090_pinconf_group_get(struct pinctrl_dev *pctldev, |
| unsigned int group, |
| unsigned long *config) |
| { |
| struct tz1090_pmx *pmx = pinctrl_dev_get_drvdata(pctldev); |
| const struct tz1090_pingroup *g; |
| enum pin_config_param param = pinconf_to_config_param(*config); |
| int ret, arg; |
| unsigned int pin; |
| u32 reg, width, mask, shift, val; |
| const int *map; |
| |
| if (group >= ARRAY_SIZE(tz1090_groups)) { |
| pin = group - ARRAY_SIZE(tz1090_groups); |
| return tz1090_pinconf_get(pctldev, pin, config); |
| } |
| |
| g = &tz1090_groups[group]; |
| if (g->npins == 1) { |
| pin = g->pins[0]; |
| ret = tz1090_pinconf_get(pctldev, pin, config); |
| if (ret != -ENOTSUPP) |
| return ret; |
| } |
| |
| /* Get register information */ |
| ret = tz1090_pinconf_group_reg(pctldev, g, param, true, |
| ®, &width, &mask, &shift, &map); |
| if (ret < 0) |
| return ret; |
| |
| /* Extract field from register */ |
| val = pmx_read(pmx, reg); |
| arg = map[(val & mask) >> shift]; |
| if (arg < 0) |
| return arg; |
| |
| /* And pack config */ |
| *config = pinconf_to_config_packed(param, arg); |
| |
| return 0; |
| } |
| |
| static int tz1090_pinconf_group_set(struct pinctrl_dev *pctldev, |
| unsigned int group, unsigned long *configs, |
| unsigned num_configs) |
| { |
| struct tz1090_pmx *pmx = pinctrl_dev_get_drvdata(pctldev); |
| const struct tz1090_pingroup *g; |
| enum pin_config_param param; |
| unsigned int arg, pin, i; |
| const unsigned int *pit; |
| int ret; |
| u32 reg, width, mask, shift, val; |
| unsigned long flags; |
| const int *map; |
| int j; |
| |
| if (group >= ARRAY_SIZE(tz1090_groups)) { |
| pin = group - ARRAY_SIZE(tz1090_groups); |
| return tz1090_pinconf_set(pctldev, pin, configs, num_configs); |
| } |
| |
| g = &tz1090_groups[group]; |
| if (g->npins == 1) { |
| pin = g->pins[0]; |
| ret = tz1090_pinconf_set(pctldev, pin, configs, num_configs); |
| if (ret != -ENOTSUPP) |
| return ret; |
| } |
| |
| for (j = 0; j < num_configs; j++) { |
| param = pinconf_to_config_param(configs[j]); |
| |
| dev_dbg(pctldev->dev, "%s(group=%s, config=%#lx)\n", |
| __func__, g->name, configs[j]); |
| |
| /* Get register information */ |
| ret = tz1090_pinconf_group_reg(pctldev, g, param, true, ®, |
| &width, &mask, &shift, &map); |
| if (ret < 0) { |
| /* |
| * Maybe we're trying to set a per-pin configuration |
| * of a group, so do the pins one by one. This is |
| * mainly as a convenience. |
| */ |
| for (i = 0, pit = g->pins; i < g->npins; ++i, ++pit) { |
| ret = tz1090_pinconf_set(pctldev, *pit, configs, |
| num_configs); |
| if (ret) |
| return ret; |
| } |
| return 0; |
| } |
| |
| /* Unpack argument and map it to register value */ |
| arg = pinconf_to_config_argument(configs[j]); |
| for (i = 0; i < BIT(width); ++i) { |
| if (map[i] == arg || (map[i] == -EINVAL && !arg)) { |
| /* Write register field */ |
| __global_lock2(flags); |
| val = pmx_read(pmx, reg); |
| val &= ~mask; |
| val |= i << shift; |
| pmx_write(pmx, val, reg); |
| __global_unlock2(flags); |
| goto next_config; |
| } |
| } |
| |
| dev_dbg(pctldev->dev, "%s: arg %u not supported\n", |
| __func__, arg); |
| return -EINVAL; |
| |
| next_config: |
| ; |
| } /* for each config */ |
| |
| return 0; |
| } |
| |
| static struct pinconf_ops tz1090_pinconf_ops = { |
| .is_generic = true, |
| .pin_config_get = tz1090_pinconf_get, |
| .pin_config_set = tz1090_pinconf_set, |
| .pin_config_group_get = tz1090_pinconf_group_get, |
| .pin_config_group_set = tz1090_pinconf_group_set, |
| .pin_config_config_dbg_show = pinconf_generic_dump_config, |
| }; |
| |
| /* |
| * Pin control driver setup |
| */ |
| |
| static struct pinctrl_desc tz1090_pinctrl_desc = { |
| .pctlops = &tz1090_pinctrl_ops, |
| .pmxops = &tz1090_pinmux_ops, |
| .confops = &tz1090_pinconf_ops, |
| .owner = THIS_MODULE, |
| }; |
| |
| static int tz1090_pinctrl_probe(struct platform_device *pdev) |
| { |
| struct tz1090_pmx *pmx; |
| struct resource *res; |
| |
| pmx = devm_kzalloc(&pdev->dev, sizeof(*pmx), GFP_KERNEL); |
| if (!pmx) { |
| dev_err(&pdev->dev, "Can't alloc tz1090_pmx\n"); |
| return -ENOMEM; |
| } |
| pmx->dev = &pdev->dev; |
| spin_lock_init(&pmx->lock); |
| |
| tz1090_pinctrl_desc.name = dev_name(&pdev->dev); |
| tz1090_pinctrl_desc.pins = tz1090_pins; |
| tz1090_pinctrl_desc.npins = ARRAY_SIZE(tz1090_pins); |
| |
| res = platform_get_resource(pdev, IORESOURCE_MEM, 0); |
| pmx->regs = devm_ioremap_resource(&pdev->dev, res); |
| if (IS_ERR(pmx->regs)) |
| return PTR_ERR(pmx->regs); |
| |
| pmx->pctl = pinctrl_register(&tz1090_pinctrl_desc, &pdev->dev, pmx); |
| if (!pmx->pctl) { |
| dev_err(&pdev->dev, "Couldn't register pinctrl driver\n"); |
| return -ENODEV; |
| } |
| |
| platform_set_drvdata(pdev, pmx); |
| |
| dev_info(&pdev->dev, "TZ1090 pinctrl driver initialised\n"); |
| |
| return 0; |
| } |
| |
| static int tz1090_pinctrl_remove(struct platform_device *pdev) |
| { |
| struct tz1090_pmx *pmx = platform_get_drvdata(pdev); |
| |
| pinctrl_unregister(pmx->pctl); |
| |
| return 0; |
| } |
| |
| static struct of_device_id tz1090_pinctrl_of_match[] = { |
| { .compatible = "img,tz1090-pinctrl", }, |
| { }, |
| }; |
| |
| static struct platform_driver tz1090_pinctrl_driver = { |
| .driver = { |
| .name = "tz1090-pinctrl", |
| .of_match_table = tz1090_pinctrl_of_match, |
| }, |
| .probe = tz1090_pinctrl_probe, |
| .remove = tz1090_pinctrl_remove, |
| }; |
| |
| static int __init tz1090_pinctrl_init(void) |
| { |
| tz1090_init_mux_pins(); |
| return platform_driver_register(&tz1090_pinctrl_driver); |
| } |
| arch_initcall(tz1090_pinctrl_init); |
| |
| static void __exit tz1090_pinctrl_exit(void) |
| { |
| platform_driver_unregister(&tz1090_pinctrl_driver); |
| } |
| module_exit(tz1090_pinctrl_exit); |
| |
| MODULE_AUTHOR("Imagination Technologies Ltd."); |
| MODULE_DESCRIPTION("Toumaz Xenif TZ1090 pinctrl driver"); |
| MODULE_LICENSE("GPL v2"); |
| MODULE_DEVICE_TABLE(of, tz1090_pinctrl_of_match); |