2013-07-24 21:41:41

by Hanumant Singh

[permalink] [raw]
Subject: [PATCH] pinctrl: msm: Add support for MSM TLMM pinmux

Add a new device tree enabled pinctrl driver for
Qualcomm MSM SoC's. This driver provides an extensible
framework to interface all MSM's that use a TLMM pinmux,
with the pinctrl subsytem.

This driver is split into two parts: the pinctrl interface
and the TLMM version specific implementation. The pinctrl
interface parses the device tree and registers with the pinctrl
subsytem. The TLMM version specific implementation supports
pin configuration/register programming for the different
pin types present on a given TLMM pinmux version.

Add support only for TLMM version 3 pinmux right now,
as well as, only two of the different pin types present on the
TLMM v3 pinmux.
Pintype 1: General purpose pins.
Pintype 2: SDC pins.

Change-Id: I065d874cd2c6fd002d5b3cb4b355445bb6912bf4
---
.../devicetree/bindings/pinctrl/msm-pinctrl.txt | 184 ++++++
drivers/pinctrl/Kconfig | 10 +
drivers/pinctrl/Makefile | 2 +
drivers/pinctrl/pinctrl-msm-tlmm-v3.c | 330 ++++++++++
drivers/pinctrl/pinctrl-msm.c | 724 +++++++++++++++++++++
drivers/pinctrl/pinctrl-msm.h | 97 +++
6 files changed, 1347 insertions(+)
create mode 100644 Documentation/devicetree/bindings/pinctrl/msm-pinctrl.txt
create mode 100644 drivers/pinctrl/pinctrl-msm-tlmm-v3.c
create mode 100644 drivers/pinctrl/pinctrl-msm.c
create mode 100644 drivers/pinctrl/pinctrl-msm.h

diff --git a/Documentation/devicetree/bindings/pinctrl/msm-pinctrl.txt b/Documentation/devicetree/bindings/pinctrl/msm-pinctrl.txt
new file mode 100644
index 0000000..0f17a94
--- /dev/null
+++ b/Documentation/devicetree/bindings/pinctrl/msm-pinctrl.txt
@@ -0,0 +1,184 @@
+MSM TLMM pinmux controller
+
+Qualcomm MSM integrates a GPIO and Pin mux/config hardware, (TOP Level Mode
+Multiplexper in short TLMM). It controls the input/output settings on the
+available pads/pins and also provides ability to multiplex and configure the
+output of various on-chip controllers onto these pads. The pins are also of
+different types, encapsulating different functions and having differing register
+semantics.
+
+Required Properties:
+- compatible: should be one of the following.
+ - "qcom,msm-tlmm-v3": for MSM with TLMM version 3.
+
+- reg: Base address of the pin controller hardware module and length of
+ the address space it occupies.
+
+- Pin types as child nodes: Pin types supported by a particular controller
+ instance are represented as child nodes of the controller node. Each
+ pin type node must contain following properties:
+
+Required Properties
+ - qcom,pin-type-*: identifies the pin type. Pin types supported are
+ qcom,pin-type-gp (General purpose)
+ qcom,pin-type-sdc (SDC)
+ - qcom,pin-cells: number of cells in the pin type specifier.
+ - qcom,num-pins: number of pins of given type present on the MSM.
+
+- Pin groups as child nodes: The pin mux (selecting pin function
+ mode) and pin config (pull up/down, driver strength, direction) settings are
+ represented as child nodes of the pin-controller node. There is no limit on
+ the count of these child nodes.
+
+ Required Properties
+ -qcom,pins: phandle specifying pin type and a pin number.
+ -qcom,num-grp-pins: number of pins in the group.
+ -label: name to to identify the pin group.
+
+ Optional Properties
+ -qcom,pin-func: function setting for the pin group.
+
+ The child node should contain a list of pin(s) on which a particular pin
+ function selection or pin configuration (or both) have to applied. This
+ list of pins is specified using the property name "qcom,pins". There
+ should be atleast one pin specfied for this property and there is no upper
+ limit on the count of pins that can be specified. The pins are specified
+ using the pintype phandle and the pin number within that pintype.
+
+ The pin function selection that should be applied on the pins listed in the
+ child node is specified using the "qcom,pin-func" property. The value
+ of this property that should be applied to each of the pins listed in the
+ "qcom,pins" property, should be picked from the hardware manual of the SoC.
+ This property is optional in the child node if no specific function
+ selection is desired for the pins listed in the child node or if the pin is
+ to be used for bit bang.
+
+ The pin group node must additionally have a pin configuration node as its own
+ child node. There can be more then one such configuration node for a pin group
+ node. There can be one or more configurations within the configuration
+ node. These configurations are applied to all pins mentoned above using the
+ "qcom,pins" property. These configurations are specific to the pintype of the
+ pins.
+ For the pin configuration properties supported by general purpose pins as well
+ as SDC pins lookup Documentation/devicetree/bindings/pinctrl-bindings.txt
+
+ The values specified by these config properties should be derived from the
+ hardware manual and these values are programmed as-is into the pin config
+ register of the pin-controller.
+
+ NOTE: A pin group node should be formed for all pins that are going to have
+ the same function and configuration settings. If a subset of pins to be used
+ by a client require different function or configuration settings or both
+ then they should be modelled as a separate pin group node to be used by
+ the client.
+
+ The client nodes that require a particular pin function selection and/or
+ pin configuration should use the bindings listed in the "pinctrl-bindings txt"
+ file.
+
+Example 1: A pin-controller node with pin types
+
+ pinctrl@fd5110000 {
+ compatible = "qcom,msm-tlmm-v3";
+ reg = <0xfd5110000 0x4000>;
+
+ /* General purpose pin type */
+ gp: gp {
+ qcom,pin-type-gp;
+ qcom,num-pins = <117>;
+ #qcom,pin-cells = <1>;
+ };
+ };
+
+Example 2: Spi pin entries within the pincontroller node
+ pinctrl@fd511000 {
+ ....
+ ..
+ pmx-spi-bus {
+ /*
+ * MOSI, MISO and CLK lines
+ * all sharing same function and config
+ * settings for each configuration node.
+ */
+ qcom,pins = <&gp 0>, <&gp 1>, <&gp 3>;
+ qcom,num-grp-pins = <3>;
+ qcom,pin-func = <1>;
+ label = "spi-bus";
+
+ /* Active configuration of bus pins */
+ spi-bus-active: spi-bus-active {
+ /*
+ * Property names as specified in
+ * pinctrl-bindings.txt
+ */
+ drive-strength = <8>; /* 8 MA */
+ bias-disable; /* No PULL */
+ };
+ /* Sleep configuration of bus pins */
+ spi-bus-sleep: spi-bus-sleep {
+ /*
+ * Property values as specified in HW
+ * manual.
+ */
+ drive-strength = <2>; /* 2 MA */
+ bias-disable;
+ };
+ };
+
+ pmx-spi-cs {
+ /*
+ * Chip select for SPI
+ * different config
+ * settings as compared to bus pins.
+ */
+ qcom,pins = <&Gp 2>;
+ qcom,num-grp-pins = <1>;
+ qcom,pin-func = <1>;
+ label = "spi-cs"
+
+ /* Active configuration of cs pin */
+ spi-cs-active: spi-cs-active {
+ /*
+ * Property names as specified in
+ * pinctrl-bindings.txt
+ */
+ drive-strength = <4>; /* 4 MA */
+ bias-disable; /* No PULL */
+ };
+ /* Sleep configuration of cs pin */
+ spi-bus-sleep: spi-bus-sleep {
+ /*
+ * Property values as specified in HW
+ * manual.
+ */
+ drive-strength = <2>; /* 2 MA */
+ bias-disable = <0>; /* No PULL */
+ };
+ };
+ };
+
+Example 3: A SPI client node that supports 'active' and 'sleep' states.
+
+ spi_0: spi@f9923000 { /* BLSP1 QUP1 */
+ compatible = "qcom,spi-qup-v2";
+ #address-cells = <1>;
+ #size-cells = <0>;
+ reg-names = "spi_physical", "spi_bam_physical";
+ reg = <0xf9923000 0x1000>,
+ <0xf9904000 0xF000>;
+ interrupt-names = "spi_irq", "spi_bam_irq";
+ interrupts = <0 95 0>, <0 238 0>;
+ spi-max-frequency = <19200000>;
+
+ /* pins used by spi controllers */
+ pinctrl-names = "default", "sleep";
+ pinctrl-0 = <&spi-bus-active &spi-cs-active>;
+ pinctrl-1 = <&spi-bus-sleep &spi-cs-sleep>;
+
+ qcom,infinite-mode = <0>;
+ qcom,use-bam;
+ qcom,ver-reg-exists;
+ qcom,bam-consumer-pipe-index = <12>;
+ qcom,bam-producer-pipe-index = <13>;
+ };
+
diff --git a/drivers/pinctrl/Kconfig b/drivers/pinctrl/Kconfig
index 34f51d2..480cb26 100644
--- a/drivers/pinctrl/Kconfig
+++ b/drivers/pinctrl/Kconfig
@@ -133,6 +133,16 @@ config PINCTRL_IMX28
bool
select PINCTRL_MXS

+config PINCTRL_MSM
+ depends on OF
+ bool
+ select PINMUX
+ select GENERIC_PINCONF
+
+config PINCTRL_MSM_TLMM_V3
+ bool
+ select PINCTRL_MSM
+
config PINCTRL_NOMADIK
bool "Nomadik pin controller driver"
depends on ARCH_U8500 || ARCH_NOMADIK
diff --git a/drivers/pinctrl/Makefile b/drivers/pinctrl/Makefile
index f82cc5b..3cf8fba 100644
--- a/drivers/pinctrl/Makefile
+++ b/drivers/pinctrl/Makefile
@@ -27,6 +27,8 @@ obj-$(CONFIG_PINCTRL_MMP2) += pinctrl-mmp2.o
obj-$(CONFIG_PINCTRL_MXS) += pinctrl-mxs.o
obj-$(CONFIG_PINCTRL_IMX23) += pinctrl-imx23.o
obj-$(CONFIG_PINCTRL_IMX28) += pinctrl-imx28.o
+obj-$(CONFIG_PINCTRL_MSM) += pinctrl-msm.o
+obj-$(CONFIG_PINCTRL_MSM_TLMM_V3) += pinctrl-msm-tlmm-v3.o
obj-$(CONFIG_PINCTRL_NOMADIK) += pinctrl-nomadik.o
obj-$(CONFIG_PINCTRL_STN8815) += pinctrl-nomadik-stn8815.o
obj-$(CONFIG_PINCTRL_DB8500) += pinctrl-nomadik-db8500.o
diff --git a/drivers/pinctrl/pinctrl-msm-tlmm-v3.c b/drivers/pinctrl/pinctrl-msm-tlmm-v3.c
new file mode 100644
index 0000000..47b50f8
--- /dev/null
+++ b/drivers/pinctrl/pinctrl-msm-tlmm-v3.c
@@ -0,0 +1,330 @@
+/* Copyright (c) 2013, The Linux Foundation. All rights reserved.
+ *
+ * This program is free software; you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License version 2 and
+ * only version 2 as published by the Free Software Foundation.
+ *
+ * This program is distributed in the hope that 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/err.h>
+#include <linux/io.h>
+#include <linux/pinctrl/pinconf-generic.h>
+#include "pinctrl-msm.h"
+
+/* config translations */
+#define drv_str_to_rval(drv) ((drv >> 1) - 1)
+#define rval_to_drv_str(val) ((val + 1) << 1)
+#define dir_to_inout_val(dir) (dir << 1)
+#define inout_val_to_dir(val) (val >> 1)
+#define rval_to_pull(val) ((val > 2) ? 1 : val)
+#define TLMMV3_NO_PULL 0
+#define TLMMV3_PULL_DOWN 1
+#define TLMMV3_PULL_UP 3
+/* GP PIN TYPE REG MASKS */
+#define TLMMV3_GP_DRV_SHFT 6
+#define TLMMV3_GP_DRV_MASK 0x7
+#define TLMMV3_GP_PULL_SHFT 0
+#define TLMMV3_GP_PULL_MASK 0x3
+#define TLMMV3_GP_DIR_SHFT 9
+#define TLMMV3_GP_DIR_MASK 1
+#define TLMMV3_GP_FUNC_SHFT 2
+#define TLMMV3_GP_FUNC_MASK 0xF
+/* SDC1 PIN TYPE REG MASKS */
+#define TLMMV3_SDC1_CLK_DRV_SHFT 6
+#define TLMMV3_SDC1_CLK_DRV_MASK 0x7
+#define TLMMV3_SDC1_DATA_DRV_SHFT 0
+#define TLMMV3_SDC1_DATA_DRV_MASK 0x7
+#define TLMMV3_SDC1_CMD_DRV_SHFT 3
+#define TLMMV3_SDC1_CMD_DRV_MASK 0x7
+#define TLMMV3_SDC1_CLK_PULL_SHFT 13
+#define TLMMV3_SDC1_CLK_PULL_MASK 0x3
+#define TLMMV3_SDC1_DATA_PULL_SHFT 9
+#define TLMMV3_SDC1_DATA_PULL_MASK 0x3
+#define TLMMV3_SDC1_CMD_PULL_SHFT 11
+#define TLMMV3_SDC1_CMD_PULL_MASK 0x3
+/* SDC2 PIN TYPE REG MASKS */
+#define TLMMV3_SDC2_CLK_DRV_SHFT 6
+#define TLMMV3_SDC2_CLK_DRV_MASK 0x7
+#define TLMMV3_SDC2_DATA_DRV_SHFT 0
+#define TLMMV3_SDC2_DATA_DRV_MASK 0x7
+#define TLMMV3_SDC2_CMD_DRV_SHFT 3
+#define TLMMV3_SDC2_CMD_DRV_MASK 0x7
+#define TLMMV3_SDC2_CLK_PULL_SHFT 14
+#define TLMMV3_SDC2_CLK_PULL_MASK 0x3
+#define TLMMV3_SDC2_DATA_PULL_SHFT 9
+#define TLMMV3_SDC2_DATA_PULL_MASK 0x3
+#define TLMMV3_SDC2_CMD_PULL_SHFT 11
+#define TLMMV3_SDC2_CMD_PULL_MASK 0x3
+
+#define TLMMV3_GP_INOUT_BIT 1
+#define TLMMV3_GP_OUT BIT(TLMMV3_GP_INOUT_BIT)
+#define TLMMV3_GP_IN 0
+
+/* SDC Pin type register offsets */
+#define TLMMV3_SDC_OFFSET 0x2044
+#define TLMMV3_SDC1_CFG(base) (base)
+#define TLMMV3_SDC2_CFG(base) (TLMMV3_SDC1_CFG(base) + 0x4)
+
+/* GP pin type register offsets */
+#define TLMMV3_GP_CFG(base, pin) (base + 0x1000 + 0x10 * (pin))
+#define TLMMV3_GP_INOUT(base, pin) (base + 0x1004 + 0x10 * (pin))
+
+struct msm_sdc_regs {
+ unsigned int offset;
+ unsigned long pull_mask;
+ unsigned long pull_shft;
+ unsigned long drv_mask;
+ unsigned long drv_shft;
+};
+
+static struct msm_sdc_regs sdc_regs[] = {
+ /* SDC1 CLK */
+ {
+ .offset = 0,
+ .pull_mask = TLMMV3_SDC1_CLK_PULL_MASK,
+ .pull_shft = TLMMV3_SDC1_CLK_PULL_SHFT,
+ .drv_mask = TLMMV3_SDC1_CLK_DRV_MASK,
+ .drv_shft = TLMMV3_SDC1_CLK_DRV_SHFT,
+ },
+ /* SDC1 CMD */
+ {
+ .offset = 0,
+ .pull_mask = TLMMV3_SDC1_CMD_PULL_MASK,
+ .pull_shft = TLMMV3_SDC1_CMD_PULL_SHFT,
+ .drv_mask = TLMMV3_SDC1_CMD_DRV_MASK,
+ .drv_shft = TLMMV3_SDC1_CMD_DRV_SHFT,
+ },
+ /* SDC1 DATA */
+ {
+ .offset = 0,
+ .pull_mask = TLMMV3_SDC1_DATA_PULL_MASK,
+ .pull_shft = TLMMV3_SDC1_DATA_PULL_SHFT,
+ .drv_mask = TLMMV3_SDC1_DATA_DRV_MASK,
+ .drv_shft = TLMMV3_SDC1_DATA_DRV_SHFT,
+ },
+ /* SDC2 CLK */
+ {
+ .offset = 0x4,
+ .pull_mask = TLMMV3_SDC2_CLK_PULL_MASK,
+ .pull_shft = TLMMV3_SDC2_CLK_PULL_SHFT,
+ .drv_mask = TLMMV3_SDC2_CLK_DRV_MASK,
+ .drv_shft = TLMMV3_SDC2_CLK_DRV_SHFT,
+ },
+ /* SDC2 CMD */
+ {
+ .offset = 0x4,
+ .pull_mask = TLMMV3_SDC2_CMD_PULL_MASK,
+ .pull_shft = TLMMV3_SDC2_CMD_PULL_SHFT,
+ .drv_mask = TLMMV3_SDC2_CMD_DRV_MASK,
+ .drv_shft = TLMMV3_SDC2_CMD_DRV_SHFT,
+ },
+ /* SDC2 DATA */
+ {
+ .offset = 0x4,
+ .pull_mask = TLMMV3_SDC2_DATA_PULL_MASK,
+ .pull_shft = TLMMV3_SDC2_DATA_PULL_SHFT,
+ .drv_mask = TLMMV3_SDC2_DATA_DRV_MASK,
+ .drv_shft = TLMMV3_SDC2_DATA_DRV_SHFT,
+ },
+};
+
+static int msm_tlmm_v3_sdc_cfg(uint pin_no, unsigned long *config,
+ void __iomem *reg_base,
+ bool write)
+{
+ unsigned int val, id, data;
+ u32 mask, shft;
+ void __iomem *cfg_reg;
+
+ cfg_reg = reg_base + sdc_regs[pin_no].offset;
+ id = pinconf_to_config_param(*config);
+ val = readl_relaxed(cfg_reg);
+ /* Get mask and shft values for this config type */
+ switch (id) {
+ case PIN_CONFIG_BIAS_DISABLE:
+ mask = sdc_regs[pin_no].pull_mask;
+ shft = sdc_regs[pin_no].pull_shft;
+ data = TLMMV3_NO_PULL;
+ if (!write) {
+ val >>= shft;
+ val &= mask;
+ data = rval_to_pull(val);
+ }
+ break;
+ case PIN_CONFIG_BIAS_PULL_DOWN:
+ mask = sdc_regs[pin_no].pull_mask;
+ shft = sdc_regs[pin_no].pull_shft;
+ data = TLMMV3_PULL_DOWN;
+ if (!write) {
+ val >>= shft;
+ val &= mask;
+ data = rval_to_pull(val);
+ }
+ break;
+ case PIN_CONFIG_BIAS_PULL_UP:
+ mask = sdc_regs[pin_no].pull_mask;
+ shft = sdc_regs[pin_no].pull_shft;
+ data = TLMMV3_PULL_UP;
+ if (!write) {
+ val >>= shft;
+ val &= mask;
+ data = rval_to_pull(val);
+ }
+ break;
+ case PIN_CONFIG_DRIVE_STRENGTH:
+ mask = sdc_regs[pin_no].drv_mask;
+ shft = sdc_regs[pin_no].drv_shft;
+ if (write) {
+ data = pinconf_to_config_argument(*config);
+ data = drv_str_to_rval(data);
+ } else {
+ val >>= shft;
+ val &= mask;
+ data = rval_to_drv_str(val);
+ }
+ break;
+ default:
+ return -EINVAL;
+ };
+
+ if (write) {
+ val &= ~(mask << shft);
+ val |= (data << shft);
+ writel_relaxed(val, cfg_reg);
+ } else
+ *config = pinconf_to_config_packed(id, data);
+ return 0;
+}
+
+static void msm_tlmm_v3_sdc_set_reg_base(void __iomem **ptype_base,
+ void __iomem *tlmm_base)
+{
+ *ptype_base = tlmm_base + TLMMV3_SDC_OFFSET;
+}
+
+static int msm_tlmm_v3_gp_cfg(uint pin_no, unsigned long *config,
+ void *reg_base, bool write)
+{
+ unsigned int val, id, data, inout_val;
+ u32 mask = 0, shft = 0;
+ void __iomem *inout_reg = NULL;
+ void __iomem *cfg_reg = TLMMV3_GP_CFG(reg_base, pin_no);
+
+ id = pinconf_to_config_param(*config);
+ val = readl_relaxed(cfg_reg);
+ /* Get mask and shft values for this config type */
+ switch (id) {
+ case PIN_CONFIG_BIAS_DISABLE:
+ mask = TLMMV3_GP_PULL_MASK;
+ shft = TLMMV3_GP_PULL_SHFT;
+ data = TLMMV3_NO_PULL;
+ if (!write) {
+ val >>= shft;
+ val &= mask;
+ data = rval_to_pull(val);
+ }
+ break;
+ case PIN_CONFIG_BIAS_PULL_DOWN:
+ mask = TLMMV3_GP_PULL_MASK;
+ shft = TLMMV3_GP_PULL_SHFT;
+ data = TLMMV3_PULL_DOWN;
+ if (!write) {
+ val >>= shft;
+ val &= mask;
+ data = rval_to_pull(val);
+ }
+ break;
+ case PIN_CONFIG_BIAS_PULL_UP:
+ mask = TLMMV3_GP_PULL_MASK;
+ shft = TLMMV3_GP_PULL_SHFT;
+ data = TLMMV3_PULL_UP;
+ if (!write) {
+ val >>= shft;
+ val &= mask;
+ data = rval_to_pull(val);
+ }
+ break;
+ case PIN_CONFIG_DRIVE_STRENGTH:
+ mask = TLMMV3_GP_DRV_MASK;
+ shft = TLMMV3_GP_DRV_SHFT;
+ if (write) {
+ data = pinconf_to_config_argument(*config);
+ data = drv_str_to_rval(data);
+ } else {
+ val >>= shft;
+ val &= mask;
+ data = rval_to_drv_str(val);
+ }
+ break;
+ case PIN_CONFIG_OUTPUT:
+ mask = TLMMV3_GP_DIR_MASK;
+ shft = TLMMV3_GP_DIR_SHFT;
+ inout_reg = TLMMV3_GP_INOUT(reg_base, pin_no);
+ if (write) {
+ data = pinconf_to_config_argument(*config);
+ inout_val = dir_to_inout_val(data);
+ writel_relaxed(inout_val, inout_reg);
+ data = (mask << shft);
+ } else {
+ inout_val = readl_relaxed(inout_reg);
+ data = inout_val_to_dir(inout_val);
+ }
+ break;
+ default:
+ return -EINVAL;
+ };
+
+ if (write) {
+ val &= ~(mask << shft);
+ val |= (data << shft);
+ writel_relaxed(val, cfg_reg);
+ } else
+ *config = pinconf_to_config_packed(id, data);
+ return 0;
+}
+
+static void msm_tlmm_v3_gp_fn(uint pin_no, u32 func, void *reg_base,
+ bool enable)
+{
+ unsigned int val;
+ void __iomem *cfg_reg = TLMMV3_GP_CFG(reg_base, pin_no);
+ val = readl_relaxed(cfg_reg);
+ val &= ~(TLMMV3_GP_FUNC_MASK << TLMMV3_GP_FUNC_SHFT);
+ if (enable)
+ val |= (func << TLMMV3_GP_FUNC_SHFT);
+ writel_relaxed(val, cfg_reg);
+}
+
+static void msm_tlmm_v3_gp_set_reg_base(void __iomem **ptype_base,
+ void __iomem *tlmm_base)
+{
+ *ptype_base = tlmm_base;
+}
+
+static struct msm_pintype_info tlmm_v3_pininfo[] = {
+ {
+ .prg_cfg = msm_tlmm_v3_gp_cfg,
+ .prg_func = msm_tlmm_v3_gp_fn,
+ .set_reg_base = msm_tlmm_v3_gp_set_reg_base,
+ .reg_data = NULL,
+ .prop_name = "qcom,pin-type-gp",
+ .name = "gp",
+ },
+ {
+ .prg_cfg = msm_tlmm_v3_sdc_cfg,
+ .set_reg_base = msm_tlmm_v3_sdc_set_reg_base,
+ .reg_data = NULL,
+ .prop_name = "qcom,pin-type-sdc",
+ .name = "sdc",
+ }
+};
+
+struct msm_tlmm tlmm_v3_pintypes = {
+ .num_entries = ARRAY_SIZE(tlmm_v3_pininfo),
+ .pintype_info = tlmm_v3_pininfo,
+};
+
diff --git a/drivers/pinctrl/pinctrl-msm.c b/drivers/pinctrl/pinctrl-msm.c
new file mode 100644
index 0000000..11671b49
--- /dev/null
+++ b/drivers/pinctrl/pinctrl-msm.c
@@ -0,0 +1,724 @@
+/* Copyright (c) 2013, The Linux Foundation. All rights reserved.
+ *
+ * This program is free software; you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License version 2 and
+ * only version 2 as published by the Free Software Foundation.
+ *
+ * This program is distributed in the hope that 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/err.h>
+#include <linux/irqdomain.h>
+#include <linux/io.h>
+#include <linux/module.h>
+#include <linux/of.h>
+#include <linux/pinctrl/pinctrl.h>
+#include <linux/pinctrl/pinmux.h>
+#include <linux/pinctrl/machine.h>
+#include <linux/platform_device.h>
+#include <linux/slab.h>
+#include "core.h"
+#include "pinconf.h"
+#include "pinctrl-msm.h"
+
+/**
+ * struct msm_pinctrl_dd: represents the pinctrol driver data.
+ * @base: virtual base of TLMM.
+ * @irq: interrupt number for TLMM summary interrupt.
+ * @num_pins: Number of total pins present on TLMM.
+ * @msm_pindesc: list of descriptors for each pin.
+ * @num_pintypes: number of pintypes on TLMM.
+ * @msm_pintype: points to the representation of all pin types supported.
+ * @pctl: pin controller instance managed by the driver.
+ * @pctl_dev: pin controller descriptor registered with the pinctrl subsystem.
+ * @pin_grps: list of pin groups available to the driver.
+ * @num_grps: number of groups.
+ * @pmx_funcs:list of pin functions available to the driver
+ * @num_funcs: number of functions.
+ * @dev: pin contol device.
+ */
+struct msm_pinctrl_dd {
+ void __iomem *base;
+ int irq;
+ unsigned int num_pins;
+ struct msm_pindesc *msm_pindesc;
+ unsigned int num_pintypes;
+ struct msm_pintype_info *msm_pintype;
+ struct pinctrl_desc pctl;
+ struct pinctrl_dev *pctl_dev;
+ struct msm_pin_grps *pin_grps;
+ unsigned int num_grps;
+ struct msm_pmx_funcs *pmx_funcs;
+ unsigned int num_funcs;
+ struct device *dev;
+};
+
+static int msm_pmx_functions_count(struct pinctrl_dev *pctldev)
+{
+ struct msm_pinctrl_dd *dd;
+
+ dd = pinctrl_dev_get_drvdata(pctldev);
+ return dd->num_funcs;
+}
+
+static const char *msm_pmx_get_fname(struct pinctrl_dev *pctldev,
+ unsigned selector)
+{
+ struct msm_pinctrl_dd *dd;
+
+ dd = pinctrl_dev_get_drvdata(pctldev);
+ return dd->pmx_funcs[selector].name;
+}
+
+static int msm_pmx_get_groups(struct pinctrl_dev *pctldev,
+ unsigned selector, const char * const **groups,
+ unsigned * const num_groups)
+{
+ struct msm_pinctrl_dd *dd;
+
+ dd = pinctrl_dev_get_drvdata(pctldev);
+ *groups = dd->pmx_funcs[selector].gps;
+ *num_groups = dd->pmx_funcs[selector].num_grps;
+ return 0;
+}
+
+static void msm_pmx_prg_fn(struct pinctrl_dev *pctldev, unsigned selector,
+ unsigned group, bool enable)
+{
+ struct msm_pinctrl_dd *dd;
+ const unsigned int *pins;
+ struct msm_pindesc *pindesc;
+ struct msm_pintype_info *pintype;
+ unsigned int pin, cnt, func;
+
+ dd = pinctrl_dev_get_drvdata(pctldev);
+ pins = dd->pin_grps[group].pins;
+ pindesc = dd->msm_pindesc;
+
+ /*
+ * for each pin in the pin group selected, program the correspoding
+ * pin function number in the config register.
+ */
+ for (cnt = 0; cnt < dd->pin_grps[group].num_pins; cnt++) {
+ pin = pins[cnt];
+ /* Obtain the pin type for given pin */
+ pintype = pindesc[pin].pin_info;
+ /* Obtain the pin number within the pin type */
+ pin = pin - pintype->pin_start;
+ func = dd->pin_grps[group].func;
+ /* program the function value for the given pin type */
+ pintype->prg_func(pin, func, pintype->reg_data, enable);
+ }
+}
+
+static int msm_pmx_enable(struct pinctrl_dev *pctldev, unsigned selector,
+ unsigned group)
+{
+ msm_pmx_prg_fn(pctldev, selector, group, true);
+ return 0;
+}
+
+static void msm_pmx_disable(struct pinctrl_dev *pctldev,
+ unsigned selector, unsigned group)
+{
+ msm_pmx_prg_fn(pctldev, selector, group, false);
+}
+
+static struct pinmux_ops msm_pmxops = {
+ .get_functions_count = msm_pmx_functions_count,
+ .get_function_name = msm_pmx_get_fname,
+ .get_function_groups = msm_pmx_get_groups,
+ .enable = msm_pmx_enable,
+ .disable = msm_pmx_disable,
+};
+
+static int msm_pconf_prg(struct pinctrl_dev *pctldev, unsigned int pin,
+ unsigned long *config, bool rw)
+{
+ struct msm_pinctrl_dd *dd;
+ struct msm_pindesc *pindesc;
+ struct msm_pintype_info *pintype;
+
+ dd = pinctrl_dev_get_drvdata(pctldev);
+ pindesc = dd->msm_pindesc;
+ /* Get pin type for given pin */
+ pintype = pindesc[pin].pin_info;
+ /* Get pin offset from the pintype start pin number */
+ pin = pin - pintype->pin_start;
+ /* Program the config value for pin type */
+ return pintype->prg_cfg(pin, config, pintype->reg_data, rw);
+}
+
+static int msm_pconf_set(struct pinctrl_dev *pctldev, unsigned int pin,
+ unsigned long config)
+{
+ return msm_pconf_prg(pctldev, pin, &config, true);
+}
+
+static int msm_pconf_get(struct pinctrl_dev *pctldev, unsigned int pin,
+ unsigned long *config)
+{
+ return msm_pconf_prg(pctldev, pin, config, false);
+}
+
+static int msm_pconf_group_set(struct pinctrl_dev *pctldev,
+ unsigned group, unsigned long config)
+{
+ struct msm_pinctrl_dd *dd;
+ const unsigned int *pins;
+ unsigned int cnt;
+
+ dd = pinctrl_dev_get_drvdata(pctldev);
+ pins = dd->pin_grps[group].pins;
+
+ for (cnt = 0; cnt < dd->pin_grps[group].num_pins; cnt++)
+ msm_pconf_set(pctldev, pins[cnt], config);
+
+ return 0;
+}
+
+static int msm_pconf_group_get(struct pinctrl_dev *pctldev,
+ unsigned int group, unsigned long *config)
+{
+ struct msm_pinctrl_dd *dd;
+ const unsigned int *pins;
+
+ dd = pinctrl_dev_get_drvdata(pctldev);
+ pins = dd->pin_grps[group].pins;
+ msm_pconf_get(pctldev, pins[0], config);
+ return 0;
+}
+
+static struct pinconf_ops msm_pconfops = {
+ .pin_config_get = msm_pconf_get,
+ .pin_config_set = msm_pconf_set,
+ .pin_config_group_get = msm_pconf_group_get,
+ .pin_config_group_set = msm_pconf_group_set,
+};
+
+static int msm_get_grps_count(struct pinctrl_dev *pctldev)
+{
+ struct msm_pinctrl_dd *dd;
+
+ dd = pinctrl_dev_get_drvdata(pctldev);
+ return dd->num_grps;
+}
+
+static const char *msm_get_grps_name(struct pinctrl_dev *pctldev,
+ unsigned selector)
+{
+ struct msm_pinctrl_dd *dd;
+
+ dd = pinctrl_dev_get_drvdata(pctldev);
+ return dd->pin_grps[selector].name;
+}
+
+static int msm_get_grps_pins(struct pinctrl_dev *pctldev,
+ unsigned selector, const unsigned **pins, unsigned *num_pins)
+{
+ struct msm_pinctrl_dd *dd;
+
+ dd = pinctrl_dev_get_drvdata(pctldev);
+ *pins = dd->pin_grps[selector].pins;
+ *num_pins = dd->pin_grps[selector].num_pins;
+ return 0;
+}
+
+static struct msm_pintype_info *msm_pgrp_to_pintype(struct device_node *nd,
+ struct msm_pinctrl_dd *dd)
+{
+ struct device_node *ptype_nd;
+ struct msm_pintype_info *pinfo = NULL;
+ int idx = 0;
+
+ /*Extract pin type node from parent node */
+ ptype_nd = of_parse_phandle(nd, "qcom,pins", 0);
+ /* find the pin type info for this pin type node */
+ for (idx = 0; idx < dd->num_pintypes; idx++) {
+ pinfo = &dd->msm_pintype[idx];
+ if (ptype_nd == pinfo->node) {
+ of_node_put(ptype_nd);
+ break;
+ }
+ }
+ return pinfo;
+}
+
+/* create pinctrl_map entries by parsing device tree nodes */
+static int msm_dt_node_to_map(struct pinctrl_dev *pctldev,
+ struct device_node *cfg_np, struct pinctrl_map **maps,
+ unsigned *nmaps)
+{
+ struct msm_pinctrl_dd *dd;
+ struct device_node *parent;
+ struct msm_pindesc *pindesc;
+ struct msm_pintype_info *pinfo;
+ struct pinctrl_map *map;
+ const char *grp_name;
+ char *fn_name;
+ u32 val;
+ unsigned long *cfg;
+ int cfg_cnt = 0, map_cnt = 0, func_cnt = 0, ret = 0;
+
+ dd = pinctrl_dev_get_drvdata(pctldev);
+ pindesc = dd->msm_pindesc;
+ /* get parent node of config node */
+ parent = of_get_parent(cfg_np);
+ /*
+ * parent node contains pin grouping
+ * get pin type from pin grouping
+ */
+ pinfo = msm_pgrp_to_pintype(parent, dd);
+ /* check if there is a function associated with the parent pin group */
+ if (of_find_property(parent, "qcom,pin-func", NULL))
+ func_cnt++;
+ /* get pin configs */
+ ret = pinconf_generic_parse_dt_config(cfg_np, &cfg, &cfg_cnt);
+ if (ret) {
+ dev_err(dd->dev, "properties incorrect\n");
+ return ret;
+ }
+
+ map_cnt = cfg_cnt + func_cnt;
+
+ /* Allocate memory for pin-map entries */
+ map = kzalloc(sizeof(*map) * map_cnt, GFP_KERNEL);
+ if (!map)
+ return -ENOMEM;
+ *nmaps = 0;
+
+ /* Get group name from node */
+ of_property_read_string(parent, "label", &grp_name);
+ /* create the config map entry */
+ map[*nmaps].data.configs.group_or_pin = grp_name;
+ map[*nmaps].data.configs.configs = cfg;
+ map[*nmaps].data.configs.num_configs = cfg_cnt;
+ map[*nmaps].type = PIN_MAP_TYPE_CONFIGS_GROUP;
+ *nmaps += 1;
+
+ /* If there is no function specified in device tree return */
+ if (func_cnt == 0) {
+ *maps = map;
+ goto no_func;
+ }
+ /* Get function mapping */
+ of_property_read_u32(parent, "qcom,pin-func", &val);
+ fn_name = kzalloc(strlen(grp_name) + strlen("-func"),
+ GFP_KERNEL);
+ if (!fn_name) {
+ ret = -ENOMEM;
+ goto func_err;
+ }
+ snprintf(fn_name, strlen(grp_name) + strlen("-func") + 1, "%s%s",
+ grp_name, "-func");
+ map[*nmaps].data.mux.group = grp_name;
+ map[*nmaps].data.mux.function = fn_name;
+ map[*nmaps].type = PIN_MAP_TYPE_MUX_GROUP;
+ *nmaps += 1;
+ *maps = map;
+ of_node_put(parent);
+ return 0;
+
+func_err:
+ kfree(cfg);
+ kfree(map);
+no_func:
+ of_node_put(parent);
+ return ret;
+}
+
+/* free the memory allocated to hold the pin-map table */
+static void msm_dt_free_map(struct pinctrl_dev *pctldev,
+ struct pinctrl_map *map, unsigned num_maps)
+{
+ int idx;
+
+ for (idx = 0; idx < num_maps; idx++) {
+ if (map[idx].type == PIN_MAP_TYPE_CONFIGS_GROUP)
+ kfree(map[idx].data.configs.configs);
+ else if (map->type == PIN_MAP_TYPE_MUX_GROUP)
+ kfree(map[idx].data.mux.function);
+ };
+
+ kfree(map);
+}
+
+static struct pinctrl_ops msm_pctrlops = {
+ .get_groups_count = msm_get_grps_count,
+ .get_group_name = msm_get_grps_name,
+ .get_group_pins = msm_get_grps_pins,
+ .dt_node_to_map = msm_dt_node_to_map,
+ .dt_free_map = msm_dt_free_map,
+};
+
+static int msm_of_get_pin(struct device_node *np, int index,
+ struct msm_pinctrl_dd *dd, uint *pin)
+{
+ struct of_phandle_args pargs;
+ struct msm_pintype_info *pinfo;
+ int num_pintypes;
+ int ret, i;
+
+ ret = of_parse_phandle_with_args(np, "qcom,pins", "#qcom,pin-cells",
+ index, &pargs);
+ if (ret)
+ return ret;
+ pinfo = dd->msm_pintype;
+ num_pintypes = dd->num_pintypes;
+ for (i = 0; i < num_pintypes; i++) {
+ /* Find the matching pin type node */
+ if (pargs.np != pinfo->node)
+ continue;
+ /* Check if arg specified is in valid range for pin type */
+ if (pargs.args[0] > pinfo->num_pins) {
+ ret = -EINVAL;
+ dev_err(dd->dev, "Invalid pin number for type %s\n",
+ pinfo->name);
+ goto out;
+ }
+ /*
+ * Pin number = index within pin type + start of pin numbers
+ * for this pin type
+ */
+ *pin = pargs.args[0] + pinfo->pin_start;
+ }
+out:
+ of_node_put(pargs.np);
+ return ret;
+}
+
+static int msm_pinctrl_dt_parse_pins(struct device_node *dev_node,
+ struct msm_pinctrl_dd *dd)
+{
+ struct device *dev;
+ struct device_node *pgrp_np;
+ struct msm_pin_grps *pin_grps, *curr_grp;
+ struct msm_pmx_funcs *pmx_funcs, *curr_func;
+ char *func_name;
+ const char *grp_name;
+ int ret, i, grp_index = 0, func_index = 0;
+ uint pin = 0, *pins, num_grps = 0, num_pins = 0, len = 0;
+ uint num_funcs = 0;
+ u32 func = 0;
+
+ dev = dd->dev;
+ for_each_child_of_node(dev_node, pgrp_np) {
+ if (!of_find_property(pgrp_np, "qcom,pins", NULL))
+ continue;
+ if (of_find_property(pgrp_np, "qcom,pin-func", NULL))
+ num_funcs++;
+ num_grps++;
+ }
+
+ pin_grps = (struct msm_pin_grps *)devm_kzalloc(dd->dev,
+ sizeof(*pin_grps) * num_grps,
+ GFP_KERNEL);
+ if (!pin_grps) {
+ dev_err(dev, "Failed to allocate grp desc\n");
+ return -ENOMEM;
+ }
+ pmx_funcs = (struct msm_pmx_funcs *)devm_kzalloc(dd->dev,
+ sizeof(*pmx_funcs) * num_funcs,
+ GFP_KERNEL);
+ if (!pmx_funcs) {
+ dev_err(dev, "Failed to allocate grp desc\n");
+ return -ENOMEM;
+ }
+ /*
+ * Iterate over all child nodes, and for nodes containing pin lists
+ * populate corresponding pin group, and if provided, corresponding
+ * function
+ */
+ for_each_child_of_node(dev_node, pgrp_np) {
+ if (!of_find_property(pgrp_np, "qcom,pins", NULL))
+ continue;
+ curr_grp = pin_grps + grp_index;
+ /* Get group name from label*/
+ ret = of_property_read_string(pgrp_np, "label", &grp_name);
+ if (ret) {
+ dev_err(dev, "Unable to allocate group name\n");
+ return ret;
+ }
+ num_pins = of_count_phandle_with_args(pgrp_np,
+ "qcom,pins",
+ "qcom,pin-cells");
+ if (IS_ERR_VALUE(num_pins)) {
+ dev_err(dev, "pin count not specified for groups %s\n",
+ grp_name);
+ return ret;
+ }
+ pins = devm_kzalloc(dd->dev, sizeof(unsigned int) * num_pins,
+ GFP_KERNEL);
+ if (!pins) {
+ dev_err(dev, "Unable to allocte pins for %s\n",
+ grp_name);
+ return -ENOMEM;
+ }
+ for (i = 0; i < num_pins; i++) {
+ ret = msm_of_get_pin(pgrp_np, i, dd, &pin);
+ if (ret) {
+ dev_err(dev, "Pin grp %s does not have pins\n",
+ grp_name);
+ return ret;
+ }
+ pins[i] = pin;
+ }
+ curr_grp->pins = pins;
+ curr_grp->num_pins = num_pins;
+ curr_grp->name = grp_name;
+ grp_index++;
+ /* Check if func specified */
+ if (!of_find_property(pgrp_np, "qcom,pin-func", NULL))
+ continue;
+ curr_func = pmx_funcs + func_index;
+ len = strlen(grp_name) + strlen("-func") + 1;
+ func_name = devm_kzalloc(dev, len, GFP_KERNEL);
+ if (!func_name) {
+ dev_err(dev, "Cannot allocate func name for grp %s",
+ grp_name);
+ return -ENOMEM;
+ }
+ snprintf(func_name, len, "%s%s", grp_name, "-func");
+ curr_func->name = func_name;
+ curr_func->gps = devm_kzalloc(dev, sizeof(char *), GFP_KERNEL);
+ if (!curr_func->gps) {
+ dev_err(dev, "failed to alloc memory for group list ");
+ return -ENOMEM;
+ }
+ of_property_read_u32(pgrp_np, "qcom,pin-func", &func);
+ curr_grp->func = func;
+ curr_func->gps[0] = grp_name;
+ curr_func->num_grps = 1;
+ func_index++;
+ }
+ dd->pin_grps = pin_grps;
+ dd->num_grps = num_grps;
+ dd->pmx_funcs = pmx_funcs;
+ dd->num_funcs = num_funcs;
+ return 0;
+}
+
+static void msm_populate_pindesc(struct msm_pintype_info *pinfo,
+ struct msm_pindesc *msm_pindesc)
+{
+ int i;
+ struct msm_pindesc *pindesc;
+
+ for (i = 0; i < pinfo->num_pins; i++) {
+ pindesc = &msm_pindesc[i + pinfo->pin_start];
+ pindesc->pin_info = pinfo;
+ snprintf(pindesc->name, sizeof(pindesc->name),
+ "%s-%d", pinfo->name, i);
+ }
+}
+
+static int msm_pinctrl_dt_parse_pintype(struct device_node *dev_node,
+ struct msm_pinctrl_dd *dd)
+{
+ struct device_node *pt_node;
+ struct msm_pindesc *msm_pindesc;
+ struct msm_pintype_info *pintype, *pinfo;
+ void __iomem **ptype_base;
+ u32 num_pins, pinfo_entries, curr_pins;
+ int i, ret;
+ uint total_pins = 0;
+
+ pinfo = dd->msm_pintype;
+ pinfo_entries = dd->num_pintypes;
+ curr_pins = 0;
+
+ for_each_child_of_node(dev_node, pt_node) {
+ for (i = 0; i < pinfo_entries; i++) {
+ pintype = &pinfo[i];
+ /* Check if node is pintype node */
+ if (!of_find_property(pt_node, pinfo->prop_name, NULL))
+ continue;
+ of_node_get(pt_node);
+ pintype->node = pt_node;
+ /* determine number of pins of given pin type */
+ ret = of_property_read_u32(pt_node, "qcom,num-pins",
+ &num_pins);
+ if (ret) {
+ dev_err(dd->dev, "num pins not specified\n");
+ return ret;
+ }
+ /* determine pin number range for given pin type */
+ pintype->num_pins = num_pins;
+ pintype->pin_start = curr_pins;
+ pintype->pin_end = curr_pins + num_pins;
+ ptype_base = &pintype->reg_data;
+ pintype->set_reg_base(ptype_base, dd->base);
+ total_pins += num_pins;
+ curr_pins += num_pins;
+ }
+ }
+ dd->msm_pindesc = devm_kzalloc(dd->dev,
+ sizeof(struct msm_pindesc) *
+ total_pins, GFP_KERNEL);
+ if (!dd->msm_pindesc) {
+ dev_err(dd->dev, "Unable to allocate msm pindesc");
+ goto alloc_fail;
+ }
+
+ dd->num_pins = total_pins;
+ msm_pindesc = dd->msm_pindesc;
+ /*
+ * Populate pin descriptor based on each pin type present in Device
+ * tree and supported by the driver
+ */
+ for (i = 0; i < pinfo_entries; i++) {
+ pintype = &pinfo[i];
+ /* If entry not in device tree, skip */
+ if (!pintype->node)
+ continue;
+ msm_populate_pindesc(pintype, msm_pindesc);
+ }
+ return 0;
+alloc_fail:
+ for (i = 0; i < pinfo_entries; i++) {
+ pintype = &pinfo[i];
+ if (pintype->node)
+ of_node_put(pintype->node);
+ }
+ return -ENOMEM;
+}
+
+static const struct of_device_id msm_pinctrl_dt_match[] = {
+ { .compatible = "qcom,msm-tlmm-v3",
+ .data = &tlmm_v3_pintypes, },
+ {},
+};
+MODULE_DEVICE_TABLE(of, msm_pinctrl_dt_match);
+
+static void msm_pinctrl_cleanup_dd(struct msm_pinctrl_dd *dd)
+{
+ int i;
+ struct msm_pintype_info *pintype;
+
+ pintype = dd->msm_pintype;
+ for (i = 0; i < dd->num_pintypes; i++) {
+ if (pintype->node)
+ of_node_put(dd->msm_pintype[i].node);
+ }
+}
+
+static int msm_pinctrl_get_drvdata(struct msm_pinctrl_dd *dd,
+ struct platform_device *pdev)
+{
+ const struct of_device_id *match;
+ const struct msm_tlmm *tlmm_info;
+ int ret;
+ struct device_node *node = pdev->dev.of_node;
+
+ match = of_match_node(msm_pinctrl_dt_match, node);
+ if (IS_ERR(match))
+ return PTR_ERR(match);
+ tlmm_info = match->data;
+ dd->msm_pintype = tlmm_info->pintype_info;
+ dd->num_pintypes = tlmm_info->num_entries;
+ ret = msm_pinctrl_dt_parse_pintype(node, dd);
+ if (ret)
+ goto out;
+
+ ret = msm_pinctrl_dt_parse_pins(node, dd);
+ if (ret)
+ msm_pinctrl_cleanup_dd(dd);
+out:
+ return ret;
+}
+
+static int msm_register_pinctrl(struct msm_pinctrl_dd *dd)
+{
+ int i;
+ struct pinctrl_pin_desc *pindesc;
+ struct pinctrl_desc *ctrl_desc = &dd->pctl;
+
+ ctrl_desc->name = "msm-pinctrl";
+ ctrl_desc->owner = THIS_MODULE;
+ ctrl_desc->pmxops = &msm_pmxops;
+ ctrl_desc->confops = &msm_pconfops;
+ ctrl_desc->pctlops = &msm_pctrlops;
+
+ pindesc = devm_kzalloc(dd->dev, sizeof(*pindesc) * dd->num_pins,
+ GFP_KERNEL);
+ if (!pindesc) {
+ dev_err(dd->dev, "Failed to allocate pinctrl pin desc\n");
+ return -ENOMEM;
+ }
+
+ for (i = 0; i < dd->num_pins; i++) {
+ pindesc[i].number = i;
+ pindesc[i].name = dd->msm_pindesc[i].name;
+ }
+ ctrl_desc->pins = pindesc;
+ ctrl_desc->npins = dd->num_pins;
+ dd->pctl_dev = pinctrl_register(ctrl_desc, dd->dev, dd);
+ if (!dd->pctl_dev) {
+ dev_err(dd->dev, "could not register pinctrl driver\n");
+ return -EINVAL;
+ }
+ return 0;
+}
+
+static int msm_pinctrl_probe(struct platform_device *pdev)
+{
+ struct msm_pinctrl_dd *dd;
+ struct device *dev = &pdev->dev;
+ struct resource *res;
+ int ret;
+
+ dd = devm_kzalloc(dev, sizeof(*dd), GFP_KERNEL);
+ if (!dd) {
+ dev_err(dev, "Alloction failed for driver data\n");
+ return -ENOMEM;
+ }
+
+ res = platform_get_resource(pdev, IORESOURCE_MEM, 0);
+ dd->base = devm_ioremap_resource(&pdev->dev, res);
+ if (IS_ERR(dd->base))
+ return PTR_ERR(dd->base);
+ res = platform_get_resource(pdev, IORESOURCE_IRQ, 0);
+ if (res)
+ dd->irq = res->start;
+ dd->dev = dev;
+ ret = msm_pinctrl_get_drvdata(dd, pdev);
+ if (ret) {
+ dev_err(&pdev->dev, "driver data not available\n");
+ return ret;
+ }
+ ret = msm_register_pinctrl(dd);
+ if (ret) {
+ msm_pinctrl_cleanup_dd(dd);
+ return ret;
+ }
+ platform_set_drvdata(pdev, dd);
+ return 0;
+}
+
+static struct platform_driver msm_pinctrl_driver = {
+ .probe = msm_pinctrl_probe,
+ .driver = {
+ .name = "msm-pinctrl",
+ .owner = THIS_MODULE,
+ .of_match_table = of_match_ptr(msm_pinctrl_dt_match),
+ },
+};
+
+static int __init msm_pinctrl_drv_register(void)
+{
+ return platform_driver_register(&msm_pinctrl_driver);
+}
+postcore_initcall(msm_pinctrl_drv_register);
+
+static void __exit msm_pinctrl_drv_unregister(void)
+{
+ platform_driver_unregister(&msm_pinctrl_driver);
+}
+module_exit(msm_pinctrl_drv_unregister);
+
+MODULE_AUTHOR("Hanumant Singh <[email protected]>");
+MODULE_LICENSE("GPL v2");
+
diff --git a/drivers/pinctrl/pinctrl-msm.h b/drivers/pinctrl/pinctrl-msm.h
new file mode 100644
index 0000000..fee159d
--- /dev/null
+++ b/drivers/pinctrl/pinctrl-msm.h
@@ -0,0 +1,97 @@
+/* Copyright (c) 2013, The Linux Foundation. All rights reserved.
+ *
+ * This program is free software; you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License version 2 and
+ * only version 2 as published by the Free Software Foundation.
+ *
+ * This program is distributed in the hope that 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.
+ */
+#ifndef __PINCTRL_MSM_H__
+#define __PINCTRL_MSM_H__
+
+#include <linux/pinctrl/pinctrl.h>
+#include <linux/pinctrl/pinmux.h>
+#include <linux/pinctrl/pinconf.h>
+#include <linux/pinctrl/machine.h>
+
+/**
+ * struct msm_pin_group: group of pins having the same pinmux function.
+ * @name: name of the pin group.
+ * @pins: the pins included in this group.
+ * @num_pins: number of pins included in this group.
+ * @func: the function number to be programmed when selected.
+ */
+struct msm_pin_grps {
+ const char *name;
+ unsigned int *pins;
+ unsigned num_pins;
+ u32 func;
+};
+
+/**
+ * struct msm_pmx_funcs: represent a pin function.
+ * @name: name of the pin function.
+ * @gps: one or more names of pin groups that provide this function.
+ * @num_grps: number of groups included in @groups.
+ */
+struct msm_pmx_funcs {
+ const char *name;
+ const char **gps;
+ unsigned num_grps;
+};
+
+/**
+ * struct msm_pintype_info: represent a pin type supported by the TLMM.
+ * @prg_cfg: helper to program a given config for a pintype.
+ * @prg_func: helper to program a given func for a pintype.
+ * @set_reg_base: helper to set the register base address for a pintype.
+ * @reg_data: register base for a pintype.
+ * @prop_name: DT property name for a pintype.
+ * @name: name of pintype.
+ * @num_pins: number of pins of given pintype.
+ * @pin_start: starting pin number for the given pintype within pinctroller.
+ * @pin_end: ending pin number for the given pintype within pinctroller.
+ * @node: device node for the pintype.
+ */
+struct msm_pintype_info {
+ int (*prg_cfg)(uint pin_no, unsigned long *config, void *reg_data,
+ bool rw);
+ void (*prg_func)(uint pin_no, u32 func, void *reg_data, bool enable);
+ void (*set_reg_base)(void __iomem **ptype_base,
+ void __iomem *tlmm_base);
+ void __iomem *reg_data;
+ const char *prop_name;
+ const char *name;
+ u32 num_pins;
+ int pin_start;
+ int pin_end;
+ struct device_node *node;
+};
+
+/**
+ * struct msm_tlmm: represents all the TLMM pintypes for a given TLMM version.
+ * @num_entries: number of pintypes.
+ * @pintype_info: descriptor for the pintypes. One for each present.
+ */
+struct msm_tlmm {
+ const uint num_entries;
+ struct msm_pintype_info *pintype_info;
+};
+
+/**
+ * struct msm_pindesc: descriptor for all pins maintained by pinctrl driver
+ * @pin_info: pintype for a given pin.
+ * @name: name of the pin.
+ */
+struct msm_pindesc {
+ struct msm_pintype_info *pin_info;
+ char name[20];
+};
+
+/* TLMM version specific data */
+extern struct msm_tlmm tlmm_v3_pintypes;
+#endif
+

--
Sent by an employee of the Qualcomm Innovation Center, Inc.
The Qualcomm Innovation Center, Inc. is a member of the Code Aurora Forum.
--


2013-07-29 16:37:38

by Linus Walleij

[permalink] [raw]
Subject: Re: [PATCH] pinctrl: msm: Add support for MSM TLMM pinmux

On Wed, Jul 24, 2013 at 11:41 PM, Hanumant Singh
<[email protected]> wrote:

> Add a new device tree enabled pinctrl driver for
> Qualcomm MSM SoC's. This driver provides an extensible
> framework to interface all MSM's that use a TLMM pinmux,
> with the pinctrl subsytem.
>
> This driver is split into two parts: the pinctrl interface
> and the TLMM version specific implementation. The pinctrl
> interface parses the device tree and registers with the pinctrl
> subsytem. The TLMM version specific implementation supports
> pin configuration/register programming for the different
> pin types present on a given TLMM pinmux version.
>
> Add support only for TLMM version 3 pinmux right now,
> as well as, only two of the different pin types present on the
> TLMM v3 pinmux.
> Pintype 1: General purpose pins.
> Pintype 2: SDC pins.

I guess this is the v4 patch set?

Please include a small changelog so I can keep track of
things...


> Change-Id: I065d874cd2c6fd002d5b3cb4b355445bb6912bf4

This thing is not interesting to the kernel community.

> diff --git a/Documentation/devicetree/bindings/pinctrl/msm-pinctrl.txt b/Documentation/devicetree/bindings/pinctrl/msm-pinctrl.txt
> new file mode 100644
> index 0000000..0f17a94
> --- /dev/null
> +++ b/Documentation/devicetree/bindings/pinctrl/msm-pinctrl.txt

This needs to be broken out and sent for separate review on
the [email protected] mailing list.

> +Required Properties
> + - qcom,pin-type-*: identifies the pin type. Pin types supported are
> + qcom,pin-type-gp (General purpose)
> + qcom,pin-type-sdc (SDC)

I am wondering if it it would not be simpler to split this
driver in two, one for the GP pins and one for the SDC pins.
Having this tag on each and every pin seems *very*
awkward.

If you have two drivers and two device tree nodes, one for
GP pins and one for SDC pins, just separated by different
compatible strings and different memory regs, things will
look simpler and be easier to maintain I think.

Unless there is some strong cross-dependency between
GP and SDC please explore this approach.

> +- Pin groups as child nodes: The pin mux (selecting pin function
> + mode) and pin config (pull up/down, driver strength, direction) settings are
> + represented as child nodes of the pin-controller node. There is no limit on
> + the count of these child nodes.
> +
> + Required Properties
> + -qcom,pins: phandle specifying pin type and a pin number.
> + -qcom,num-grp-pins: number of pins in the group.
> + -label: name to to identify the pin group.
> +
> + Optional Properties
> + -qcom,pin-func: function setting for the pin group.

After some scratching my head I realize that you are trying
to reimplement parts of pinctrl-single.c.

I.e. you try to put all the definitions of groups and functions
into the devicetree instead of having this in the driver
as tables.

If this is what you want to do you should use pinctrl-single.c.
It might be possible to use pinctrl-single.c only for the
GP pins.

But if there is something complex about the hardware that
make pinctrl-single.c not fit the bill I advice you to encode
the lists of groups and functions into the driver instead.

Also, split this in a per-soc manner (compare the other
drivers) so you can support plugging one file per SoC
for the different MSM chips using this pin controller.

Yours,
Linus Walleij

2013-07-29 17:32:54

by Stephen Warren

[permalink] [raw]
Subject: Re: [PATCH] pinctrl: msm: Add support for MSM TLMM pinmux

On 07/29/2013 10:37 AM, Linus Walleij wrote:
> On Wed, Jul 24, 2013 at 11:41 PM, Hanumant Singh
> <[email protected]> wrote:
>
>> Add a new device tree enabled pinctrl driver for
>> Qualcomm MSM SoC's. This driver provides an extensible
>> framework to interface all MSM's that use a TLMM pinmux,
>> with the pinctrl subsytem.
>>
>> This driver is split into two parts: the pinctrl interface
>> and the TLMM version specific implementation. The pinctrl
>> interface parses the device tree and registers with the pinctrl
>> subsytem. The TLMM version specific implementation supports
>> pin configuration/register programming for the different
>> pin types present on a given TLMM pinmux version.
>>
>> Add support only for TLMM version 3 pinmux right now,
>> as well as, only two of the different pin types present on the
>> TLMM v3 pinmux.
>> Pintype 1: General purpose pins.
>> Pintype 2: SDC pins.
>
> I guess this is the v4 patch set?
>
> Please include a small changelog so I can keep track of
> things...
>
>
>> Change-Id: I065d874cd2c6fd002d5b3cb4b355445bb6912bf4
>
> This thing is not interesting to the kernel community.
>
>> diff --git a/Documentation/devicetree/bindings/pinctrl/msm-pinctrl.txt b/Documentation/devicetree/bindings/pinctrl/msm-pinctrl.txt
>> new file mode 100644
>> index 0000000..0f17a94
>> --- /dev/null
>> +++ b/Documentation/devicetree/bindings/pinctrl/msm-pinctrl.txt
>
> This needs to be broken out and sent for separate review on
> the [email protected] mailing list.

It certainly should be reviewed there, although I don't think the
binding .txt file needs to be a separate patch. At present, it's quite
typical to include the binding doc with the first driver that implements
it, although that may change in the future; we'll see.

2013-07-29 23:39:21

by Bjorn Andersson

[permalink] [raw]
Subject: Re: [PATCH] pinctrl: msm: Add support for MSM TLMM pinmux

On Wed, Jul 24, 2013 at 1:41 PM, Hanumant Singh <[email protected]> wrote:
> Add a new device tree enabled pinctrl driver for
> Qualcomm MSM SoC's. This driver provides an extensible
> framework to interface all MSM's that use a TLMM pinmux,
> with the pinctrl subsytem.

Thanks! I was about to write an implementation for v2 myself when I
found your patch. I hacked a little bit on it and got it to work on my
8960 based board.

>
> This driver is split into two parts: the pinctrl interface
> and the TLMM version specific implementation. The pinctrl
> interface parses the device tree and registers with the pinctrl
> subsytem. The TLMM version specific implementation supports
> pin configuration/register programming for the different
> pin types present on a given TLMM pinmux version.
>
> Add support only for TLMM version 3 pinmux right now,
> as well as, only two of the different pin types present on the
> TLMM v3 pinmux.
> Pintype 1: General purpose pins.
> Pintype 2: SDC pins.

It seems that the gp pins of TLMM v3 have the same layout and offsets
as the once in v2. Unless I'm missing something I think this patch
should be structured (and have appropriate naming) in a way that we
can support both of them.

As a general note on the patch, the pins and pin groups are defined by
the soc, I'm therefore not convinced that these should be configured
from the devicetree. It's at least not how it's done in other
platforms.


After talking to Linus a little bit I concluded that the way the
design idea behind pinmux is that you have pins and you have
functions;
* pins are your 117 gp-pins and with some imagination your sd pins, so
that's fine.
* the function is some functionality provided by the hard
Then you can set up muxes between these two. Therefore I think your
use of the word "function" in the patch is what normally is called a
mux. It might be worth sorting this out, to make it easier to maintain
this code down the road.

@Linus, can you please confirm my understanding of the design?

>
> Change-Id: I065d874cd2c6fd002d5b3cb4b355445bb6912bf4
> ---
> .../devicetree/bindings/pinctrl/msm-pinctrl.txt | 184 ++++++
> drivers/pinctrl/Kconfig | 10 +
> drivers/pinctrl/Makefile | 2 +
> drivers/pinctrl/pinctrl-msm-tlmm-v3.c | 330 ++++++++++
> drivers/pinctrl/pinctrl-msm.c | 724 +++++++++++++++++++++
> drivers/pinctrl/pinctrl-msm.h | 97 +++
> 6 files changed, 1347 insertions(+)
> create mode 100644 Documentation/devicetree/bindings/pinctrl/msm-pinctrl.txt
> create mode 100644 drivers/pinctrl/pinctrl-msm-tlmm-v3.c
> create mode 100644 drivers/pinctrl/pinctrl-msm.c
> create mode 100644 drivers/pinctrl/pinctrl-msm.h
>
> diff --git a/Documentation/devicetree/bindings/pinctrl/msm-pinctrl.txt b/Documentation/devicetree/bindings/pinctrl/msm-pinctrl.txt
> new file mode 100644
> index 0000000..0f17a94
> --- /dev/null
> +++ b/Documentation/devicetree/bindings/pinctrl/msm-pinctrl.txt
> @@ -0,0 +1,184 @@
> +MSM TLMM pinmux controller
> +
> +Qualcomm MSM integrates a GPIO and Pin mux/config hardware, (TOP Level Mode
> +Multiplexper in short TLMM). It controls the input/output settings on the
> +available pads/pins and also provides ability to multiplex and configure the
> +output of various on-chip controllers onto these pads. The pins are also of
> +different types, encapsulating different functions and having differing register
> +semantics.
> +
> +Required Properties:
> +- compatible: should be one of the following.
> + - "qcom,msm-tlmm-v3": for MSM with TLMM version 3.
> +
> +- reg: Base address of the pin controller hardware module and length of
> + the address space it occupies.
> +
> +- Pin types as child nodes: Pin types supported by a particular controller
> + instance are represented as child nodes of the controller node. Each
> + pin type node must contain following properties:
> +
> +Required Properties
> + - qcom,pin-type-*: identifies the pin type. Pin types supported are
> + qcom,pin-type-gp (General purpose)
> + qcom,pin-type-sdc (SDC)
> + - qcom,pin-cells: number of cells in the pin type specifier.
> + - qcom,num-pins: number of pins of given type present on the MSM.
> +
> +- Pin groups as child nodes: The pin mux (selecting pin function
> + mode) and pin config (pull up/down, driver strength, direction) settings are
> + represented as child nodes of the pin-controller node. There is no limit on
> + the count of these child nodes.
> +
> + Required Properties
> + -qcom,pins: phandle specifying pin type and a pin number.
> + -qcom,num-grp-pins: number of pins in the group.
> + -label: name to to identify the pin group.
> +
> + Optional Properties
> + -qcom,pin-func: function setting for the pin group.
> +
> + The child node should contain a list of pin(s) on which a particular pin
> + function selection or pin configuration (or both) have to applied. This
> + list of pins is specified using the property name "qcom,pins". There
> + should be atleast one pin specfied for this property and there is no upper
> + limit on the count of pins that can be specified. The pins are specified
> + using the pintype phandle and the pin number within that pintype.
> +
> + The pin function selection that should be applied on the pins listed in the
> + child node is specified using the "qcom,pin-func" property. The value
> + of this property that should be applied to each of the pins listed in the
> + "qcom,pins" property, should be picked from the hardware manual of the SoC.
> + This property is optional in the child node if no specific function
> + selection is desired for the pins listed in the child node or if the pin is
> + to be used for bit bang.
> +
> + The pin group node must additionally have a pin configuration node as its own
> + child node. There can be more then one such configuration node for a pin group
> + node. There can be one or more configurations within the configuration
> + node. These configurations are applied to all pins mentoned above using the
> + "qcom,pins" property. These configurations are specific to the pintype of the
> + pins.
> + For the pin configuration properties supported by general purpose pins as well
> + as SDC pins lookup Documentation/devicetree/bindings/pinctrl-bindings.txt
> +
> + The values specified by these config properties should be derived from the
> + hardware manual and these values are programmed as-is into the pin config
> + register of the pin-controller.
> +
> + NOTE: A pin group node should be formed for all pins that are going to have
> + the same function and configuration settings. If a subset of pins to be used
> + by a client require different function or configuration settings or both
> + then they should be modelled as a separate pin group node to be used by
> + the client.
> +
> + The client nodes that require a particular pin function selection and/or
> + pin configuration should use the bindings listed in the "pinctrl-bindings txt"
> + file.
> +
> +Example 1: A pin-controller node with pin types
> +
> + pinctrl@fd5110000 {
> + compatible = "qcom,msm-tlmm-v3";
> + reg = <0xfd5110000 0x4000>;
> +
> + /* General purpose pin type */
> + gp: gp {
> + qcom,pin-type-gp;
> + qcom,num-pins = <117>;
> + #qcom,pin-cells = <1>;
> + };
> + };
> +
> +Example 2: Spi pin entries within the pincontroller node
> + pinctrl@fd511000 {
> + ....
> + ..
> + pmx-spi-bus {
> + /*
> + * MOSI, MISO and CLK lines
> + * all sharing same function and config
> + * settings for each configuration node.
> + */
> + qcom,pins = <&gp 0>, <&gp 1>, <&gp 3>;
> + qcom,num-grp-pins = <3>;

qcom,num-grp-pins is not used by the code, please remove it.

> + qcom,pin-func = <1>;
> + label = "spi-bus";
> +
> + /* Active configuration of bus pins */
> + spi-bus-active: spi-bus-active {
> + /*
> + * Property names as specified in
> + * pinctrl-bindings.txt
> + */
> + drive-strength = <8>; /* 8 MA */
> + bias-disable; /* No PULL */
> + };
> + /* Sleep configuration of bus pins */
> + spi-bus-sleep: spi-bus-sleep {
> + /*
> + * Property values as specified in HW
> + * manual.
> + */
> + drive-strength = <2>; /* 2 MA */
> + bias-disable;
> + };
> + };
> +
> + pmx-spi-cs {
> + /*
> + * Chip select for SPI
> + * different config
> + * settings as compared to bus pins.
> + */
> + qcom,pins = <&Gp 2>;

Make Gp lower case.

> + qcom,num-grp-pins = <1>;
> + qcom,pin-func = <1>;
> + label = "spi-cs"
> +
> + /* Active configuration of cs pin */
> + spi-cs-active: spi-cs-active {
> + /*
> + * Property names as specified in
> + * pinctrl-bindings.txt
> + */
> + drive-strength = <4>; /* 4 MA */
> + bias-disable; /* No PULL */
> + };
> + /* Sleep configuration of cs pin */
> + spi-bus-sleep: spi-bus-sleep {
> + /*
> + * Property values as specified in HW
> + * manual.
> + */
> + drive-strength = <2>; /* 2 MA */
> + bias-disable = <0>; /* No PULL */
> + };
> + };
> + };
> +
> +Example 3: A SPI client node that supports 'active' and 'sleep' states.
> +
> + spi_0: spi@f9923000 { /* BLSP1 QUP1 */
> + compatible = "qcom,spi-qup-v2";
> + #address-cells = <1>;
> + #size-cells = <0>;
> + reg-names = "spi_physical", "spi_bam_physical";
> + reg = <0xf9923000 0x1000>,
> + <0xf9904000 0xF000>;
> + interrupt-names = "spi_irq", "spi_bam_irq";
> + interrupts = <0 95 0>, <0 238 0>;
> + spi-max-frequency = <19200000>;
> +
> + /* pins used by spi controllers */
> + pinctrl-names = "default", "sleep";
> + pinctrl-0 = <&spi-bus-active &spi-cs-active>;
> + pinctrl-1 = <&spi-bus-sleep &spi-cs-sleep>;
> +
> + qcom,infinite-mode = <0>;
> + qcom,use-bam;
> + qcom,ver-reg-exists;
> + qcom,bam-consumer-pipe-index = <12>;
> + qcom,bam-producer-pipe-index = <13>;

Please fix the indentation here.

> + };
> +
> diff --git a/drivers/pinctrl/Kconfig b/drivers/pinctrl/Kconfig
> index 34f51d2..480cb26 100644
> --- a/drivers/pinctrl/Kconfig
> +++ b/drivers/pinctrl/Kconfig
> @@ -133,6 +133,16 @@ config PINCTRL_IMX28
> bool
> select PINCTRL_MXS
>
> +config PINCTRL_MSM
> + depends on OF
> + bool
> + select PINMUX
> + select GENERIC_PINCONF
> +
> +config PINCTRL_MSM_TLMM_V3
> + bool
> + select PINCTRL_MSM

PINCTRL_MSM does selects only to compile the "common" parts, but this
does not build. Adding e.g MSM_TLMM_V2 here as well (as I did first)
just made me have to select 2 things to make one thing compile.

My suggestion is that you remove the V3 config and just put it all
under PINCTRL_MSM

> +
> config PINCTRL_NOMADIK
> bool "Nomadik pin controller driver"
> depends on ARCH_U8500 || ARCH_NOMADIK
> diff --git a/drivers/pinctrl/Makefile b/drivers/pinctrl/Makefile
> index f82cc5b..3cf8fba 100644
> --- a/drivers/pinctrl/Makefile
> +++ b/drivers/pinctrl/Makefile
> @@ -27,6 +27,8 @@ obj-$(CONFIG_PINCTRL_MMP2) += pinctrl-mmp2.o
> obj-$(CONFIG_PINCTRL_MXS) += pinctrl-mxs.o
> obj-$(CONFIG_PINCTRL_IMX23) += pinctrl-imx23.o
> obj-$(CONFIG_PINCTRL_IMX28) += pinctrl-imx28.o
> +obj-$(CONFIG_PINCTRL_MSM) += pinctrl-msm.o
> +obj-$(CONFIG_PINCTRL_MSM_TLMM_V3) += pinctrl-msm-tlmm-v3.o
> obj-$(CONFIG_PINCTRL_NOMADIK) += pinctrl-nomadik.o
> obj-$(CONFIG_PINCTRL_STN8815) += pinctrl-nomadik-stn8815.o
> obj-$(CONFIG_PINCTRL_DB8500) += pinctrl-nomadik-db8500.o
> diff --git a/drivers/pinctrl/pinctrl-msm-tlmm-v3.c b/drivers/pinctrl/pinctrl-msm-tlmm-v3.c
> new file mode 100644
> index 0000000..47b50f8
> --- /dev/null
> +++ b/drivers/pinctrl/pinctrl-msm-tlmm-v3.c
> @@ -0,0 +1,330 @@
> +/* Copyright (c) 2013, The Linux Foundation. All rights reserved.
> + *
> + * This program is free software; you can redistribute it and/or modify
> + * it under the terms of the GNU General Public License version 2 and
> + * only version 2 as published by the Free Software Foundation.
> + *
> + * This program is distributed in the hope that 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/err.h>
> +#include <linux/io.h>
> +#include <linux/pinctrl/pinconf-generic.h>
> +#include "pinctrl-msm.h"
> +
> +/* config translations */
> +#define drv_str_to_rval(drv) ((drv >> 1) - 1)
> +#define rval_to_drv_str(val) ((val + 1) << 1)
> +#define dir_to_inout_val(dir) (dir << 1)
> +#define inout_val_to_dir(val) (val >> 1)
> +#define rval_to_pull(val) ((val > 2) ? 1 : val)
> +#define TLMMV3_NO_PULL 0
> +#define TLMMV3_PULL_DOWN 1
> +#define TLMMV3_PULL_UP 3
> +/* GP PIN TYPE REG MASKS */
> +#define TLMMV3_GP_DRV_SHFT 6
> +#define TLMMV3_GP_DRV_MASK 0x7
> +#define TLMMV3_GP_PULL_SHFT 0
> +#define TLMMV3_GP_PULL_MASK 0x3
> +#define TLMMV3_GP_DIR_SHFT 9
> +#define TLMMV3_GP_DIR_MASK 1
> +#define TLMMV3_GP_FUNC_SHFT 2
> +#define TLMMV3_GP_FUNC_MASK 0xF
> +/* SDC1 PIN TYPE REG MASKS */
> +#define TLMMV3_SDC1_CLK_DRV_SHFT 6
> +#define TLMMV3_SDC1_CLK_DRV_MASK 0x7
> +#define TLMMV3_SDC1_DATA_DRV_SHFT 0
> +#define TLMMV3_SDC1_DATA_DRV_MASK 0x7
> +#define TLMMV3_SDC1_CMD_DRV_SHFT 3
> +#define TLMMV3_SDC1_CMD_DRV_MASK 0x7
> +#define TLMMV3_SDC1_CLK_PULL_SHFT 13
> +#define TLMMV3_SDC1_CLK_PULL_MASK 0x3
> +#define TLMMV3_SDC1_DATA_PULL_SHFT 9
> +#define TLMMV3_SDC1_DATA_PULL_MASK 0x3
> +#define TLMMV3_SDC1_CMD_PULL_SHFT 11
> +#define TLMMV3_SDC1_CMD_PULL_MASK 0x3
> +/* SDC2 PIN TYPE REG MASKS */
> +#define TLMMV3_SDC2_CLK_DRV_SHFT 6
> +#define TLMMV3_SDC2_CLK_DRV_MASK 0x7
> +#define TLMMV3_SDC2_DATA_DRV_SHFT 0
> +#define TLMMV3_SDC2_DATA_DRV_MASK 0x7
> +#define TLMMV3_SDC2_CMD_DRV_SHFT 3
> +#define TLMMV3_SDC2_CMD_DRV_MASK 0x7
> +#define TLMMV3_SDC2_CLK_PULL_SHFT 14
> +#define TLMMV3_SDC2_CLK_PULL_MASK 0x3
> +#define TLMMV3_SDC2_DATA_PULL_SHFT 9
> +#define TLMMV3_SDC2_DATA_PULL_MASK 0x3
> +#define TLMMV3_SDC2_CMD_PULL_SHFT 11
> +#define TLMMV3_SDC2_CMD_PULL_MASK 0x3
> +
> +#define TLMMV3_GP_INOUT_BIT 1
> +#define TLMMV3_GP_OUT BIT(TLMMV3_GP_INOUT_BIT)
> +#define TLMMV3_GP_IN 0

These three defines are not used.

> +
> +/* SDC Pin type register offsets */
> +#define TLMMV3_SDC_OFFSET 0x2044
> +#define TLMMV3_SDC1_CFG(base) (base)
> +#define TLMMV3_SDC2_CFG(base) (TLMMV3_SDC1_CFG(base) + 0x4)
> +
> +/* GP pin type register offsets */
> +#define TLMMV3_GP_CFG(base, pin) (base + 0x1000 + 0x10 * (pin))
> +#define TLMMV3_GP_INOUT(base, pin) (base + 0x1004 + 0x10 * (pin))
> +
> +struct msm_sdc_regs {
> + unsigned int offset;
> + unsigned long pull_mask;
> + unsigned long pull_shft;
> + unsigned long drv_mask;
> + unsigned long drv_shft;
> +};
> +
> +static struct msm_sdc_regs sdc_regs[] = {
> + /* SDC1 CLK */
> + {
> + .offset = 0,
> + .pull_mask = TLMMV3_SDC1_CLK_PULL_MASK,
> + .pull_shft = TLMMV3_SDC1_CLK_PULL_SHFT,
> + .drv_mask = TLMMV3_SDC1_CLK_DRV_MASK,
> + .drv_shft = TLMMV3_SDC1_CLK_DRV_SHFT,
> + },
> + /* SDC1 CMD */
> + {
> + .offset = 0,
> + .pull_mask = TLMMV3_SDC1_CMD_PULL_MASK,
> + .pull_shft = TLMMV3_SDC1_CMD_PULL_SHFT,
> + .drv_mask = TLMMV3_SDC1_CMD_DRV_MASK,
> + .drv_shft = TLMMV3_SDC1_CMD_DRV_SHFT,
> + },
> + /* SDC1 DATA */
> + {
> + .offset = 0,
> + .pull_mask = TLMMV3_SDC1_DATA_PULL_MASK,
> + .pull_shft = TLMMV3_SDC1_DATA_PULL_SHFT,
> + .drv_mask = TLMMV3_SDC1_DATA_DRV_MASK,
> + .drv_shft = TLMMV3_SDC1_DATA_DRV_SHFT,
> + },
> + /* SDC2 CLK */
> + {
> + .offset = 0x4,
> + .pull_mask = TLMMV3_SDC2_CLK_PULL_MASK,
> + .pull_shft = TLMMV3_SDC2_CLK_PULL_SHFT,
> + .drv_mask = TLMMV3_SDC2_CLK_DRV_MASK,
> + .drv_shft = TLMMV3_SDC2_CLK_DRV_SHFT,
> + },
> + /* SDC2 CMD */
> + {
> + .offset = 0x4,
> + .pull_mask = TLMMV3_SDC2_CMD_PULL_MASK,
> + .pull_shft = TLMMV3_SDC2_CMD_PULL_SHFT,
> + .drv_mask = TLMMV3_SDC2_CMD_DRV_MASK,
> + .drv_shft = TLMMV3_SDC2_CMD_DRV_SHFT,
> + },
> + /* SDC2 DATA */
> + {
> + .offset = 0x4,
> + .pull_mask = TLMMV3_SDC2_DATA_PULL_MASK,
> + .pull_shft = TLMMV3_SDC2_DATA_PULL_SHFT,
> + .drv_mask = TLMMV3_SDC2_DATA_DRV_MASK,
> + .drv_shft = TLMMV3_SDC2_DATA_DRV_SHFT,
> + },
> +};
> +
> +static int msm_tlmm_v3_sdc_cfg(uint pin_no, unsigned long *config,
> + void __iomem *reg_base,
> + bool write)

I strongly recommend you split this function into a get_config and
set_config. It took me plenty of time to get my head around what
you're doing here.

I would also suggest that you do:

val = readl()
switch () {
fix val in various ways
}
writel(val)

Instead of maintaining mask and shift throughout the function.

> +{
> + unsigned int val, id, data;
> + u32 mask, shft;
> + void __iomem *cfg_reg;
> +
> + cfg_reg = reg_base + sdc_regs[pin_no].offset;
> + id = pinconf_to_config_param(*config);
> + val = readl_relaxed(cfg_reg);
> + /* Get mask and shft values for this config type */
> + switch (id) {
> + case PIN_CONFIG_BIAS_DISABLE:
> + mask = sdc_regs[pin_no].pull_mask;
> + shft = sdc_regs[pin_no].pull_shft;
> + data = TLMMV3_NO_PULL;
> + if (!write) {
> + val >>= shft;
> + val &= mask;
> + data = rval_to_pull(val);
> + }
> + break;
> + case PIN_CONFIG_BIAS_PULL_DOWN:
> + mask = sdc_regs[pin_no].pull_mask;
> + shft = sdc_regs[pin_no].pull_shft;
> + data = TLMMV3_PULL_DOWN;
> + if (!write) {
> + val >>= shft;
> + val &= mask;
> + data = rval_to_pull(val);
> + }
> + break;
> + case PIN_CONFIG_BIAS_PULL_UP:
> + mask = sdc_regs[pin_no].pull_mask;
> + shft = sdc_regs[pin_no].pull_shft;
> + data = TLMMV3_PULL_UP;
> + if (!write) {
> + val >>= shft;
> + val &= mask;
> + data = rval_to_pull(val);
> + }
> + break;
> + case PIN_CONFIG_DRIVE_STRENGTH:
> + mask = sdc_regs[pin_no].drv_mask;
> + shft = sdc_regs[pin_no].drv_shft;
> + if (write) {
> + data = pinconf_to_config_argument(*config);
> + data = drv_str_to_rval(data);
> + } else {
> + val >>= shft;
> + val &= mask;
> + data = rval_to_drv_str(val);
> + }
> + break;
> + default:
> + return -EINVAL;
> + };
> +
> + if (write) {
> + val &= ~(mask << shft);
> + val |= (data << shft);
> + writel_relaxed(val, cfg_reg);
> + } else
> + *config = pinconf_to_config_packed(id, data);
> + return 0;
> +}
> +
> +static void msm_tlmm_v3_sdc_set_reg_base(void __iomem **ptype_base,
> + void __iomem *tlmm_base)
> +{
> + *ptype_base = tlmm_base + TLMMV3_SDC_OFFSET;
> +}

This function is used to do "pin type based base offset shifting", if
you instead moved the calculation into the msm_sdc_regs above you
could just skip it from your interface and pass the blocks reg_base to
allt these functions.

> +
> +static int msm_tlmm_v3_gp_cfg(uint pin_no, unsigned long *config,
> + void *reg_base, bool write)

As with the sdc config function I think this is cluttered. Better
duplicate some of it to make two nice functions that one can easily
follow.

> +{
> + unsigned int val, id, data, inout_val;
> + u32 mask = 0, shft = 0;
> + void __iomem *inout_reg = NULL;
> + void __iomem *cfg_reg = TLMMV3_GP_CFG(reg_base, pin_no);
> +
> + id = pinconf_to_config_param(*config);
> + val = readl_relaxed(cfg_reg);
> + /* Get mask and shft values for this config type */
> + switch (id) {
> + case PIN_CONFIG_BIAS_DISABLE:
> + mask = TLMMV3_GP_PULL_MASK;
> + shft = TLMMV3_GP_PULL_SHFT;
> + data = TLMMV3_NO_PULL;
> + if (!write) {
> + val >>= shft;
> + val &= mask;
> + data = rval_to_pull(val);

I assume this should return a "boolean" whether or not this mode is
selected; if so you should return 1 here iff (val & 0x3) == 3. I
assume the same goes for the sdc config function?

@Linus, could you please confirm this interpretation of the get_config
for pull config.

> + }
> + break;
> + case PIN_CONFIG_BIAS_PULL_DOWN:
> + mask = TLMMV3_GP_PULL_MASK;
> + shft = TLMMV3_GP_PULL_SHFT;
> + data = TLMMV3_PULL_DOWN;
> + if (!write) {
> + val >>= shft;
> + val &= mask;
> + data = rval_to_pull(val);
> + }
> + break;
> + case PIN_CONFIG_BIAS_PULL_UP:
> + mask = TLMMV3_GP_PULL_MASK;
> + shft = TLMMV3_GP_PULL_SHFT;
> + data = TLMMV3_PULL_UP;
> + if (!write) {
> + val >>= shft;
> + val &= mask;
> + data = rval_to_pull(val);
> + }
> + break;
> + case PIN_CONFIG_DRIVE_STRENGTH:
> + mask = TLMMV3_GP_DRV_MASK;
> + shft = TLMMV3_GP_DRV_SHFT;
> + if (write) {
> + data = pinconf_to_config_argument(*config);
> + data = drv_str_to_rval(data);
> + } else {
> + val >>= shft;
> + val &= mask;
> + data = rval_to_drv_str(val);
> + }
> + break;
> + case PIN_CONFIG_OUTPUT:
> + mask = TLMMV3_GP_DIR_MASK;
> + shft = TLMMV3_GP_DIR_SHFT;
> + inout_reg = TLMMV3_GP_INOUT(reg_base, pin_no);
> + if (write) {
> + data = pinconf_to_config_argument(*config);
> + inout_val = dir_to_inout_val(data);
> + writel_relaxed(inout_val, inout_reg);
> + data = (mask << shft);
> + } else {
> + inout_val = readl_relaxed(inout_reg);
> + data = inout_val_to_dir(inout_val);
> + }
> + break;
> + default:
> + return -EINVAL;
> + };
> +
> + if (write) {
> + val &= ~(mask << shft);
> + val |= (data << shft);
> + writel_relaxed(val, cfg_reg);
> + } else
> + *config = pinconf_to_config_packed(id, data);
> + return 0;
> +}
> +
> +static void msm_tlmm_v3_gp_fn(uint pin_no, u32 func, void *reg_base,
> + bool enable)
> +{
> + unsigned int val;
> + void __iomem *cfg_reg = TLMMV3_GP_CFG(reg_base, pin_no);
> + val = readl_relaxed(cfg_reg);
> + val &= ~(TLMMV3_GP_FUNC_MASK << TLMMV3_GP_FUNC_SHFT);
> + if (enable)
> + val |= (func << TLMMV3_GP_FUNC_SHFT);
> + writel_relaxed(val, cfg_reg);
> +}
> +
> +static void msm_tlmm_v3_gp_set_reg_base(void __iomem **ptype_base,
> + void __iomem *tlmm_base)
> +{
> + *ptype_base = tlmm_base;
> +}
> +
> +static struct msm_pintype_info tlmm_v3_pininfo[] = {
> + {
> + .prg_cfg = msm_tlmm_v3_gp_cfg,
> + .prg_func = msm_tlmm_v3_gp_fn,

Please add a separate function for getting the config, as it would
simplify both sides of the interface.
Please rename these functions ".get_config", ".set_config" and ".set_function".

> + .set_reg_base = msm_tlmm_v3_gp_set_reg_base,

Please remove this from the interface, better let the sdc definition
include the offset needed.

> + .reg_data = NULL,

By removing the set_reg_base, you don't need a per pin-type definition
of the register base.

> + .prop_name = "qcom,pin-type-gp",
> + .name = "gp",
> + },
> + {
> + .prg_cfg = msm_tlmm_v3_sdc_cfg,
> + .set_reg_base = msm_tlmm_v3_sdc_set_reg_base,
> + .reg_data = NULL,
> + .prop_name = "qcom,pin-type-sdc",
> + .name = "sdc",
> + }
> +};
> +
> +struct msm_tlmm tlmm_v3_pintypes = {
> + .num_entries = ARRAY_SIZE(tlmm_v3_pininfo),
> + .pintype_info = tlmm_v3_pininfo,
> +};
> +
> diff --git a/drivers/pinctrl/pinctrl-msm.c b/drivers/pinctrl/pinctrl-msm.c
> new file mode 100644
> index 0000000..11671b49
> --- /dev/null
> +++ b/drivers/pinctrl/pinctrl-msm.c
> @@ -0,0 +1,724 @@
> +/* Copyright (c) 2013, The Linux Foundation. All rights reserved.
> + *
> + * This program is free software; you can redistribute it and/or modify
> + * it under the terms of the GNU General Public License version 2 and
> + * only version 2 as published by the Free Software Foundation.
> + *
> + * This program is distributed in the hope that 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/err.h>
> +#include <linux/irqdomain.h>
> +#include <linux/io.h>
> +#include <linux/module.h>
> +#include <linux/of.h>
> +#include <linux/pinctrl/pinctrl.h>
> +#include <linux/pinctrl/pinmux.h>
> +#include <linux/pinctrl/machine.h>
> +#include <linux/platform_device.h>
> +#include <linux/slab.h>
> +#include "core.h"
> +#include "pinconf.h"
> +#include "pinctrl-msm.h"
> +
> +/**
> + * struct msm_pinctrl_dd: represents the pinctrol driver data.
> + * @base: virtual base of TLMM.
> + * @irq: interrupt number for TLMM summary interrupt.
> + * @num_pins: Number of total pins present on TLMM.
> + * @msm_pindesc: list of descriptors for each pin.
> + * @num_pintypes: number of pintypes on TLMM.
> + * @msm_pintype: points to the representation of all pin types supported.
> + * @pctl: pin controller instance managed by the driver.
> + * @pctl_dev: pin controller descriptor registered with the pinctrl subsystem.
> + * @pin_grps: list of pin groups available to the driver.
> + * @num_grps: number of groups.
> + * @pmx_funcs:list of pin functions available to the driver
> + * @num_funcs: number of functions.
> + * @dev: pin contol device.
> + */
> +struct msm_pinctrl_dd {
> + void __iomem *base;
> + int irq;
> + unsigned int num_pins;
> + struct msm_pindesc *msm_pindesc;
> + unsigned int num_pintypes;
> + struct msm_pintype_info *msm_pintype;
> + struct pinctrl_desc pctl;
> + struct pinctrl_dev *pctl_dev;
> + struct msm_pin_grps *pin_grps;
> + unsigned int num_grps;
> + struct msm_pmx_funcs *pmx_funcs;
> + unsigned int num_funcs;
> + struct device *dev;
> +};
> +
> +static int msm_pmx_functions_count(struct pinctrl_dev *pctldev)
> +{
> + struct msm_pinctrl_dd *dd;
> +
> + dd = pinctrl_dev_get_drvdata(pctldev);
> + return dd->num_funcs;
> +}
> +
> +static const char *msm_pmx_get_fname(struct pinctrl_dev *pctldev,
> + unsigned selector)
> +{
> + struct msm_pinctrl_dd *dd;
> +
> + dd = pinctrl_dev_get_drvdata(pctldev);
> + return dd->pmx_funcs[selector].name;
> +}
> +
> +static int msm_pmx_get_groups(struct pinctrl_dev *pctldev,
> + unsigned selector, const char * const **groups,
> + unsigned * const num_groups)
> +{
> + struct msm_pinctrl_dd *dd;
> +
> + dd = pinctrl_dev_get_drvdata(pctldev);
> + *groups = dd->pmx_funcs[selector].gps;
> + *num_groups = dd->pmx_funcs[selector].num_grps;
> + return 0;
> +}
> +
> +static void msm_pmx_prg_fn(struct pinctrl_dev *pctldev, unsigned selector,
> + unsigned group, bool enable)
> +{
> + struct msm_pinctrl_dd *dd;
> + const unsigned int *pins;
> + struct msm_pindesc *pindesc;
> + struct msm_pintype_info *pintype;
> + unsigned int pin, cnt, func;
> +
> + dd = pinctrl_dev_get_drvdata(pctldev);
> + pins = dd->pin_grps[group].pins;
> + pindesc = dd->msm_pindesc;
> +
> + /*
> + * for each pin in the pin group selected, program the correspoding
> + * pin function number in the config register.
> + */
> + for (cnt = 0; cnt < dd->pin_grps[group].num_pins; cnt++) {
> + pin = pins[cnt];
> + /* Obtain the pin type for given pin */
> + pintype = pindesc[pin].pin_info;
> + /* Obtain the pin number within the pin type */
> + pin = pin - pintype->pin_start;
> + func = dd->pin_grps[group].func;
> + /* program the function value for the given pin type */
> + pintype->prg_func(pin, func, pintype->reg_data, enable);
> + }
> +}
> +
> +static int msm_pmx_enable(struct pinctrl_dev *pctldev, unsigned selector,
> + unsigned group)
> +{
> + msm_pmx_prg_fn(pctldev, selector, group, true);
> + return 0;
> +}
> +
> +static void msm_pmx_disable(struct pinctrl_dev *pctldev,
> + unsigned selector, unsigned group)
> +{
> + msm_pmx_prg_fn(pctldev, selector, group, false);
> +}
> +
> +static struct pinmux_ops msm_pmxops = {
> + .get_functions_count = msm_pmx_functions_count,
> + .get_function_name = msm_pmx_get_fname,
> + .get_function_groups = msm_pmx_get_groups,
> + .enable = msm_pmx_enable,
> + .disable = msm_pmx_disable,
> +};
> +
> +static int msm_pconf_prg(struct pinctrl_dev *pctldev, unsigned int pin,
> + unsigned long *config, bool rw)
> +{
> + struct msm_pinctrl_dd *dd;
> + struct msm_pindesc *pindesc;
> + struct msm_pintype_info *pintype;
> +
> + dd = pinctrl_dev_get_drvdata(pctldev);
> + pindesc = dd->msm_pindesc;
> + /* Get pin type for given pin */
> + pintype = pindesc[pin].pin_info;
> + /* Get pin offset from the pintype start pin number */
> + pin = pin - pintype->pin_start;
> + /* Program the config value for pin type */
> + return pintype->prg_cfg(pin, config, pintype->reg_data, rw);
> +}
> +
> +static int msm_pconf_set(struct pinctrl_dev *pctldev, unsigned int pin,
> + unsigned long config)
> +{
> + return msm_pconf_prg(pctldev, pin, &config, true);
> +}
> +
> +static int msm_pconf_get(struct pinctrl_dev *pctldev, unsigned int pin,
> + unsigned long *config)
> +{
> + return msm_pconf_prg(pctldev, pin, config, false);
> +}
> +
> +static int msm_pconf_group_set(struct pinctrl_dev *pctldev,
> + unsigned group, unsigned long config)
> +{
> + struct msm_pinctrl_dd *dd;
> + const unsigned int *pins;
> + unsigned int cnt;
> +
> + dd = pinctrl_dev_get_drvdata(pctldev);
> + pins = dd->pin_grps[group].pins;
> +
> + for (cnt = 0; cnt < dd->pin_grps[group].num_pins; cnt++)
> + msm_pconf_set(pctldev, pins[cnt], config);
> +
> + return 0;
> +}
> +
> +static int msm_pconf_group_get(struct pinctrl_dev *pctldev,
> + unsigned int group, unsigned long *config)
> +{
> + struct msm_pinctrl_dd *dd;
> + const unsigned int *pins;
> +
> + dd = pinctrl_dev_get_drvdata(pctldev);
> + pins = dd->pin_grps[group].pins;
> + msm_pconf_get(pctldev, pins[0], config);
> + return 0;
> +}
> +
> +static struct pinconf_ops msm_pconfops = {
> + .pin_config_get = msm_pconf_get,
> + .pin_config_set = msm_pconf_set,
> + .pin_config_group_get = msm_pconf_group_get,
> + .pin_config_group_set = msm_pconf_group_set,
> +};
> +
> +static int msm_get_grps_count(struct pinctrl_dev *pctldev)
> +{
> + struct msm_pinctrl_dd *dd;
> +
> + dd = pinctrl_dev_get_drvdata(pctldev);
> + return dd->num_grps;
> +}
> +
> +static const char *msm_get_grps_name(struct pinctrl_dev *pctldev,
> + unsigned selector)
> +{
> + struct msm_pinctrl_dd *dd;
> +
> + dd = pinctrl_dev_get_drvdata(pctldev);
> + return dd->pin_grps[selector].name;
> +}
> +
> +static int msm_get_grps_pins(struct pinctrl_dev *pctldev,
> + unsigned selector, const unsigned **pins, unsigned *num_pins)
> +{
> + struct msm_pinctrl_dd *dd;
> +
> + dd = pinctrl_dev_get_drvdata(pctldev);
> + *pins = dd->pin_grps[selector].pins;
> + *num_pins = dd->pin_grps[selector].num_pins;
> + return 0;
> +}
> +
> +static struct msm_pintype_info *msm_pgrp_to_pintype(struct device_node *nd,
> + struct msm_pinctrl_dd *dd)
> +{
> + struct device_node *ptype_nd;
> + struct msm_pintype_info *pinfo = NULL;
> + int idx = 0;
> +
> + /*Extract pin type node from parent node */
> + ptype_nd = of_parse_phandle(nd, "qcom,pins", 0);
> + /* find the pin type info for this pin type node */
> + for (idx = 0; idx < dd->num_pintypes; idx++) {
> + pinfo = &dd->msm_pintype[idx];
> + if (ptype_nd == pinfo->node) {
> + of_node_put(ptype_nd);
> + break;
> + }
> + }
> + return pinfo;
> +}
> +
> +/* create pinctrl_map entries by parsing device tree nodes */
> +static int msm_dt_node_to_map(struct pinctrl_dev *pctldev,
> + struct device_node *cfg_np, struct pinctrl_map **maps,
> + unsigned *nmaps)
> +{
> + struct msm_pinctrl_dd *dd;
> + struct device_node *parent;
> + struct msm_pindesc *pindesc;
> + struct msm_pintype_info *pinfo;
> + struct pinctrl_map *map;
> + const char *grp_name;
> + char *fn_name;
> + u32 val;
> + unsigned long *cfg;
> + int cfg_cnt = 0, map_cnt = 0, func_cnt = 0, ret = 0;
> +
> + dd = pinctrl_dev_get_drvdata(pctldev);
> + pindesc = dd->msm_pindesc;
> + /* get parent node of config node */
> + parent = of_get_parent(cfg_np);
> + /*
> + * parent node contains pin grouping
> + * get pin type from pin grouping
> + */
> + pinfo = msm_pgrp_to_pintype(parent, dd);
> + /* check if there is a function associated with the parent pin group */
> + if (of_find_property(parent, "qcom,pin-func", NULL))
> + func_cnt++;
> + /* get pin configs */
> + ret = pinconf_generic_parse_dt_config(cfg_np, &cfg, &cfg_cnt);
> + if (ret) {
> + dev_err(dd->dev, "properties incorrect\n");
> + return ret;
> + }
> +
> + map_cnt = cfg_cnt + func_cnt;
> +
> + /* Allocate memory for pin-map entries */
> + map = kzalloc(sizeof(*map) * map_cnt, GFP_KERNEL);
> + if (!map)
> + return -ENOMEM;
> + *nmaps = 0;
> +
> + /* Get group name from node */
> + of_property_read_string(parent, "label", &grp_name);
> + /* create the config map entry */
> + map[*nmaps].data.configs.group_or_pin = grp_name;
> + map[*nmaps].data.configs.configs = cfg;
> + map[*nmaps].data.configs.num_configs = cfg_cnt;
> + map[*nmaps].type = PIN_MAP_TYPE_CONFIGS_GROUP;
> + *nmaps += 1;
> +
> + /* If there is no function specified in device tree return */
> + if (func_cnt == 0) {
> + *maps = map;
> + goto no_func;
> + }
> + /* Get function mapping */
> + of_property_read_u32(parent, "qcom,pin-func", &val);
> + fn_name = kzalloc(strlen(grp_name) + strlen("-func"),
> + GFP_KERNEL);
> + if (!fn_name) {
> + ret = -ENOMEM;
> + goto func_err;
> + }
> + snprintf(fn_name, strlen(grp_name) + strlen("-func") + 1, "%s%s",
> + grp_name, "-func");

Store the size of this in a variable to be used both in the allocation
and in the snprintf. That way you don't have to line wrap here.
Also make sure you allocate room for the nul byte (that you have in
your snprintf).

And move the "-func" into the format string ("%s-func").

> + map[*nmaps].data.mux.group = grp_name;
> + map[*nmaps].data.mux.function = fn_name;
> + map[*nmaps].type = PIN_MAP_TYPE_MUX_GROUP;
> + *nmaps += 1;
> + *maps = map;
> + of_node_put(parent);
> + return 0;
> +
> +func_err:
> + kfree(cfg);
> + kfree(map);
> +no_func:
> + of_node_put(parent);
> + return ret;
> +}
> +
> +/* free the memory allocated to hold the pin-map table */
> +static void msm_dt_free_map(struct pinctrl_dev *pctldev,
> + struct pinctrl_map *map, unsigned num_maps)
> +{
> + int idx;
> +
> + for (idx = 0; idx < num_maps; idx++) {
> + if (map[idx].type == PIN_MAP_TYPE_CONFIGS_GROUP)
> + kfree(map[idx].data.configs.configs);
> + else if (map->type == PIN_MAP_TYPE_MUX_GROUP)
> + kfree(map[idx].data.mux.function);
> + };
> +
> + kfree(map);
> +}
> +
> +static struct pinctrl_ops msm_pctrlops = {
> + .get_groups_count = msm_get_grps_count,
> + .get_group_name = msm_get_grps_name,
> + .get_group_pins = msm_get_grps_pins,
> + .dt_node_to_map = msm_dt_node_to_map,
> + .dt_free_map = msm_dt_free_map,
> +};
> +
> +static int msm_of_get_pin(struct device_node *np, int index,
> + struct msm_pinctrl_dd *dd, uint *pin)
> +{
> + struct of_phandle_args pargs;
> + struct msm_pintype_info *pinfo;
> + int num_pintypes;
> + int ret, i;
> +
> + ret = of_parse_phandle_with_args(np, "qcom,pins", "#qcom,pin-cells",
> + index, &pargs);
> + if (ret)
> + return ret;
> + pinfo = dd->msm_pintype;
> + num_pintypes = dd->num_pintypes;
> + for (i = 0; i < num_pintypes; i++) {
> + /* Find the matching pin type node */
> + if (pargs.np != pinfo->node)
> + continue;
> + /* Check if arg specified is in valid range for pin type */
> + if (pargs.args[0] > pinfo->num_pins) {
> + ret = -EINVAL;
> + dev_err(dd->dev, "Invalid pin number for type %s\n",
> + pinfo->name);
> + goto out;
> + }
> + /*
> + * Pin number = index within pin type + start of pin numbers
> + * for this pin type
> + */
> + *pin = pargs.args[0] + pinfo->pin_start;
> + }
> +out:
> + of_node_put(pargs.np);
> + return ret;
> +}
> +
> +static int msm_pinctrl_dt_parse_pins(struct device_node *dev_node,
> + struct msm_pinctrl_dd *dd)
> +{
> + struct device *dev;
> + struct device_node *pgrp_np;
> + struct msm_pin_grps *pin_grps, *curr_grp;
> + struct msm_pmx_funcs *pmx_funcs, *curr_func;
> + char *func_name;
> + const char *grp_name;
> + int ret, i, grp_index = 0, func_index = 0;
> + uint pin = 0, *pins, num_grps = 0, num_pins = 0, len = 0;
> + uint num_funcs = 0;
> + u32 func = 0;
> +
> + dev = dd->dev;
> + for_each_child_of_node(dev_node, pgrp_np) {
> + if (!of_find_property(pgrp_np, "qcom,pins", NULL))
> + continue;
> + if (of_find_property(pgrp_np, "qcom,pin-func", NULL))
> + num_funcs++;
> + num_grps++;
> + }
> +
> + pin_grps = (struct msm_pin_grps *)devm_kzalloc(dd->dev,
> + sizeof(*pin_grps) * num_grps,
> + GFP_KERNEL);
> + if (!pin_grps) {
> + dev_err(dev, "Failed to allocate grp desc\n");
> + return -ENOMEM;
> + }
> + pmx_funcs = (struct msm_pmx_funcs *)devm_kzalloc(dd->dev,
> + sizeof(*pmx_funcs) * num_funcs,
> + GFP_KERNEL);
> + if (!pmx_funcs) {
> + dev_err(dev, "Failed to allocate grp desc\n");
> + return -ENOMEM;
> + }
> + /*
> + * Iterate over all child nodes, and for nodes containing pin lists
> + * populate corresponding pin group, and if provided, corresponding
> + * function
> + */
> + for_each_child_of_node(dev_node, pgrp_np) {
> + if (!of_find_property(pgrp_np, "qcom,pins", NULL))
> + continue;
> + curr_grp = pin_grps + grp_index;
> + /* Get group name from label*/
> + ret = of_property_read_string(pgrp_np, "label", &grp_name);
> + if (ret) {
> + dev_err(dev, "Unable to allocate group name\n");
> + return ret;
> + }
> + num_pins = of_count_phandle_with_args(pgrp_np,
> + "qcom,pins",
> + "qcom,pin-cells");

This doesn't work, the last parameter needs to be "#qcom,pin-cells"
based on your example device tree.

> + if (IS_ERR_VALUE(num_pins)) {
> + dev_err(dev, "pin count not specified for groups %s\n",
> + grp_name);
> + return ret;
> + }
> + pins = devm_kzalloc(dd->dev, sizeof(unsigned int) * num_pins,
> + GFP_KERNEL);
> + if (!pins) {
> + dev_err(dev, "Unable to allocte pins for %s\n",
> + grp_name);
> + return -ENOMEM;
> + }
> + for (i = 0; i < num_pins; i++) {
> + ret = msm_of_get_pin(pgrp_np, i, dd, &pin);
> + if (ret) {
> + dev_err(dev, "Pin grp %s does not have pins\n",
> + grp_name);
> + return ret;
> + }
> + pins[i] = pin;
> + }
> + curr_grp->pins = pins;
> + curr_grp->num_pins = num_pins;
> + curr_grp->name = grp_name;
> + grp_index++;
> + /* Check if func specified */
> + if (!of_find_property(pgrp_np, "qcom,pin-func", NULL))
> + continue;
> + curr_func = pmx_funcs + func_index;
> + len = strlen(grp_name) + strlen("-func") + 1;
> + func_name = devm_kzalloc(dev, len, GFP_KERNEL);
> + if (!func_name) {
> + dev_err(dev, "Cannot allocate func name for grp %s",
> + grp_name);
> + return -ENOMEM;
> + }
> + snprintf(func_name, len, "%s%s", grp_name, "-func");

Move "-func" into the format.

> + curr_func->name = func_name;
> + curr_func->gps = devm_kzalloc(dev, sizeof(char *), GFP_KERNEL);
> + if (!curr_func->gps) {
> + dev_err(dev, "failed to alloc memory for group list ");
> + return -ENOMEM;
> + }
> + of_property_read_u32(pgrp_np, "qcom,pin-func", &func);
> + curr_grp->func = func;
> + curr_func->gps[0] = grp_name;
> + curr_func->num_grps = 1;

This is the only place you store something in gps and the only place
you read the value is in msm_pmux_get_groups(). As you're in control
of the msm_pmx_funcs struct I suggest that you remove this allocation
and just store the group_name in a char * pointer and drops the
num_grps. Then in msm_pmux_get_groups you assign

*groups = &dd->pmx_funcs[selector].gps;
*num_groups = 1;

And preferably you rename gps to group_name, or something else that's readable.

> + func_index++;
> + }
> + dd->pin_grps = pin_grps;
> + dd->num_grps = num_grps;
> + dd->pmx_funcs = pmx_funcs;
> + dd->num_funcs = num_funcs;
> + return 0;
> +}
> +
> +static void msm_populate_pindesc(struct msm_pintype_info *pinfo,
> + struct msm_pindesc *msm_pindesc)
> +{
> + int i;
> + struct msm_pindesc *pindesc;
> +
> + for (i = 0; i < pinfo->num_pins; i++) {
> + pindesc = &msm_pindesc[i + pinfo->pin_start];
> + pindesc->pin_info = pinfo;
> + snprintf(pindesc->name, sizeof(pindesc->name),
> + "%s-%d", pinfo->name, i);
> + }
> +}
> +
> +static int msm_pinctrl_dt_parse_pintype(struct device_node *dev_node,
> + struct msm_pinctrl_dd *dd)
> +{
> + struct device_node *pt_node;
> + struct msm_pindesc *msm_pindesc;
> + struct msm_pintype_info *pintype, *pinfo;
> + void __iomem **ptype_base;
> + u32 num_pins, pinfo_entries, curr_pins;
> + int i, ret;
> + uint total_pins = 0;
> +
> + pinfo = dd->msm_pintype;
> + pinfo_entries = dd->num_pintypes;
> + curr_pins = 0;
> +
> + for_each_child_of_node(dev_node, pt_node) {
> + for (i = 0; i < pinfo_entries; i++) {
> + pintype = &pinfo[i];
> + /* Check if node is pintype node */
> + if (!of_find_property(pt_node, pinfo->prop_name, NULL))
> + continue;
> + of_node_get(pt_node);
> + pintype->node = pt_node;
> + /* determine number of pins of given pin type */
> + ret = of_property_read_u32(pt_node, "qcom,num-pins",
> + &num_pins);
> + if (ret) {

The alloc_fail at the bottom is taking care of putting any of_nodes
you got here, in the event of an allocation failure below. But here
you simply return. Please roll back your of_node_gets.

> + dev_err(dd->dev, "num pins not specified\n");
> + return ret;
> + }
> + /* determine pin number range for given pin type */
> + pintype->num_pins = num_pins;
> + pintype->pin_start = curr_pins;
> + pintype->pin_end = curr_pins + num_pins;

How does this work with having multiple pin types defined at the same time.

As far as I can see the order of the definition of the pin types
matters because of this.
It also gives an idea that you could have multiple gp/sdc groups, but
then the addressing will be way confusing.

Compare what actual pin <&gp 0> refers to in these two cases:

gp: gp { qcom,num-pins = <10>; }
sdc: sdc { qcom,num-pins = <10>; }

sdc: sdc { qcom,num-pins = <10>; }
gp: gp { qcom,num-pins = <10>; }


I suggest that you drop the pin_start and only store num_pins in the
pintype struct.

> + ptype_base = &pintype->reg_data;
> + pintype->set_reg_base(ptype_base, dd->base);

Please refer to my comment on this in the v3.c file regarding this object model.

> + total_pins += num_pins;
> + curr_pins += num_pins;
> + }
> + }
> + dd->msm_pindesc = devm_kzalloc(dd->dev,
> + sizeof(struct msm_pindesc) *
> + total_pins, GFP_KERNEL);
> + if (!dd->msm_pindesc) {
> + dev_err(dd->dev, "Unable to allocate msm pindesc");
> + goto alloc_fail;
> + }
> +
> + dd->num_pins = total_pins;
> + msm_pindesc = dd->msm_pindesc;
> + /*
> + * Populate pin descriptor based on each pin type present in Device
> + * tree and supported by the driver
> + */
> + for (i = 0; i < pinfo_entries; i++) {
> + pintype = &pinfo[i];
> + /* If entry not in device tree, skip */
> + if (!pintype->node)
> + continue;
> + msm_populate_pindesc(pintype, msm_pindesc);
> + }
> + return 0;
> +alloc_fail:
> + for (i = 0; i < pinfo_entries; i++) {
> + pintype = &pinfo[i];
> + if (pintype->node)
> + of_node_put(pintype->node);
> + }
> + return -ENOMEM;
> +}
> +
> +static const struct of_device_id msm_pinctrl_dt_match[] = {
> + { .compatible = "qcom,msm-tlmm-v3",
> + .data = &tlmm_v3_pintypes, },
> + {},
> +};
> +MODULE_DEVICE_TABLE(of, msm_pinctrl_dt_match);
> +
> +static void msm_pinctrl_cleanup_dd(struct msm_pinctrl_dd *dd)
> +{
> + int i;
> + struct msm_pintype_info *pintype;
> +
> + pintype = dd->msm_pintype;
> + for (i = 0; i < dd->num_pintypes; i++) {
> + if (pintype->node)
> + of_node_put(dd->msm_pintype[i].node);
> + }
> +}
> +
> +static int msm_pinctrl_get_drvdata(struct msm_pinctrl_dd *dd,
> + struct platform_device *pdev)
> +{
> + const struct of_device_id *match;
> + const struct msm_tlmm *tlmm_info;
> + int ret;
> + struct device_node *node = pdev->dev.of_node;
> +
> + match = of_match_node(msm_pinctrl_dt_match, node);
> + if (IS_ERR(match))
> + return PTR_ERR(match);
> + tlmm_info = match->data;
> + dd->msm_pintype = tlmm_info->pintype_info;
> + dd->num_pintypes = tlmm_info->num_entries;
> + ret = msm_pinctrl_dt_parse_pintype(node, dd);
> + if (ret)
> + goto out;
> +
> + ret = msm_pinctrl_dt_parse_pins(node, dd);
> + if (ret)
> + msm_pinctrl_cleanup_dd(dd);
> +out:
> + return ret;
> +}
> +
> +static int msm_register_pinctrl(struct msm_pinctrl_dd *dd)
> +{
> + int i;
> + struct pinctrl_pin_desc *pindesc;
> + struct pinctrl_desc *ctrl_desc = &dd->pctl;
> +
> + ctrl_desc->name = "msm-pinctrl";
> + ctrl_desc->owner = THIS_MODULE;
> + ctrl_desc->pmxops = &msm_pmxops;
> + ctrl_desc->confops = &msm_pconfops;
> + ctrl_desc->pctlops = &msm_pctrlops;
> +
> + pindesc = devm_kzalloc(dd->dev, sizeof(*pindesc) * dd->num_pins,
> + GFP_KERNEL);
> + if (!pindesc) {
> + dev_err(dd->dev, "Failed to allocate pinctrl pin desc\n");
> + return -ENOMEM;
> + }
> +
> + for (i = 0; i < dd->num_pins; i++) {
> + pindesc[i].number = i;
> + pindesc[i].name = dd->msm_pindesc[i].name;
> + }
> + ctrl_desc->pins = pindesc;
> + ctrl_desc->npins = dd->num_pins;
> + dd->pctl_dev = pinctrl_register(ctrl_desc, dd->dev, dd);
> + if (!dd->pctl_dev) {
> + dev_err(dd->dev, "could not register pinctrl driver\n");
> + return -EINVAL;
> + }
> + return 0;
> +}
> +
> +static int msm_pinctrl_probe(struct platform_device *pdev)
> +{
> + struct msm_pinctrl_dd *dd;
> + struct device *dev = &pdev->dev;
> + struct resource *res;
> + int ret;
> +
> + dd = devm_kzalloc(dev, sizeof(*dd), GFP_KERNEL);
> + if (!dd) {
> + dev_err(dev, "Alloction failed for driver data\n");
> + return -ENOMEM;
> + }
> +
> + res = platform_get_resource(pdev, IORESOURCE_MEM, 0);
> + dd->base = devm_ioremap_resource(&pdev->dev, res);
> + if (IS_ERR(dd->base))
> + return PTR_ERR(dd->base);
> + res = platform_get_resource(pdev, IORESOURCE_IRQ, 0);
> + if (res)
> + dd->irq = res->start;
> + dd->dev = dev;
> + ret = msm_pinctrl_get_drvdata(dd, pdev);
> + if (ret) {
> + dev_err(&pdev->dev, "driver data not available\n");
> + return ret;
> + }
> + ret = msm_register_pinctrl(dd);
> + if (ret) {
> + msm_pinctrl_cleanup_dd(dd);
> + return ret;
> + }
> + platform_set_drvdata(pdev, dd);
> + return 0;
> +}
> +
> +static struct platform_driver msm_pinctrl_driver = {
> + .probe = msm_pinctrl_probe,
> + .driver = {
> + .name = "msm-pinctrl",
> + .owner = THIS_MODULE,
> + .of_match_table = of_match_ptr(msm_pinctrl_dt_match),
> + },
> +};
> +
> +static int __init msm_pinctrl_drv_register(void)
> +{
> + return platform_driver_register(&msm_pinctrl_driver);
> +}
> +postcore_initcall(msm_pinctrl_drv_register);
> +
> +static void __exit msm_pinctrl_drv_unregister(void)
> +{
> + platform_driver_unregister(&msm_pinctrl_driver);
> +}
> +module_exit(msm_pinctrl_drv_unregister);
> +
> +MODULE_AUTHOR("Hanumant Singh <[email protected]>");
> +MODULE_LICENSE("GPL v2");
> +
> diff --git a/drivers/pinctrl/pinctrl-msm.h b/drivers/pinctrl/pinctrl-msm.h
> new file mode 100644
> index 0000000..fee159d
> --- /dev/null
> +++ b/drivers/pinctrl/pinctrl-msm.h
> @@ -0,0 +1,97 @@
> +/* Copyright (c) 2013, The Linux Foundation. All rights reserved.
> + *
> + * This program is free software; you can redistribute it and/or modify
> + * it under the terms of the GNU General Public License version 2 and
> + * only version 2 as published by the Free Software Foundation.
> + *
> + * This program is distributed in the hope that 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.
> + */
> +#ifndef __PINCTRL_MSM_H__
> +#define __PINCTRL_MSM_H__
> +
> +#include <linux/pinctrl/pinctrl.h>
> +#include <linux/pinctrl/pinmux.h>
> +#include <linux/pinctrl/pinconf.h>
> +#include <linux/pinctrl/machine.h>
> +
> +/**
> + * struct msm_pin_group: group of pins having the same pinmux function.
> + * @name: name of the pin group.
> + * @pins: the pins included in this group.
> + * @num_pins: number of pins included in this group.
> + * @func: the function number to be programmed when selected.
> + */
> +struct msm_pin_grps {

I prefer the name you use in the comment; msm_pin_group over the
shortened version.

> + const char *name;
> + unsigned int *pins;
> + unsigned num_pins;
> + u32 func;
> +};
> +
> +/**
> + * struct msm_pmx_funcs: represent a pin function.
> + * @name: name of the pin function.
> + * @gps: one or more names of pin groups that provide this function.
> + * @num_grps: number of groups included in @groups.
> + */
> +struct msm_pmx_funcs {
> + const char *name;
> + const char **gps;
> + unsigned num_grps;

Please spell this out as "groups" and "num_groups". Also, as argued
before, with the current implementation there's no need for ** and
numb_groups. Because they are always 1 element in an array and 1.

> +};
> +
> +/**
> + * struct msm_pintype_info: represent a pin type supported by the TLMM.
> + * @prg_cfg: helper to program a given config for a pintype.
> + * @prg_func: helper to program a given func for a pintype.
> + * @set_reg_base: helper to set the register base address for a pintype.
> + * @reg_data: register base for a pintype.
> + * @prop_name: DT property name for a pintype.
> + * @name: name of pintype.
> + * @num_pins: number of pins of given pintype.
> + * @pin_start: starting pin number for the given pintype within pinctroller.
> + * @pin_end: ending pin number for the given pintype within pinctroller.
> + * @node: device node for the pintype.
> + */
> +struct msm_pintype_info {
> + int (*prg_cfg)(uint pin_no, unsigned long *config, void *reg_data,
> + bool rw);

Please split this and name it get_config and set_config.

> + void (*prg_func)(uint pin_no, u32 func, void *reg_data, bool enable);

Please rename this set_function.

> + void (*set_reg_base)(void __iomem **ptype_base,
> + void __iomem *tlmm_base);

Please remove this.

> + void __iomem *reg_data;

Please just reference the reg_base of the msm_pinctrl_dd.

> + const char *prop_name;
> + const char *name;
> + u32 num_pins;
> + int pin_start;
> + int pin_end;

If pin_start ever is anything else than 0 I think things will be
mighty confusing. Please see my comment on this above.

> + struct device_node *node;
> +};
> +
> +/**
> + * struct msm_tlmm: represents all the TLMM pintypes for a given TLMM version.
> + * @num_entries: number of pintypes.
> + * @pintype_info: descriptor for the pintypes. One for each present.
> + */
> +struct msm_tlmm {
> + const uint num_entries;
> + struct msm_pintype_info *pintype_info;
> +};
> +
> +/**
> + * struct msm_pindesc: descriptor for all pins maintained by pinctrl driver
> + * @pin_info: pintype for a given pin.
> + * @name: name of the pin.
> + */
> +struct msm_pindesc {
> + struct msm_pintype_info *pin_info;
> + char name[20];
> +};
> +
> +/* TLMM version specific data */
> +extern struct msm_tlmm tlmm_v3_pintypes;
> +#endif
> +
>
> --
> Sent by an employee of the Qualcomm Innovation Center, Inc.
> The Qualcomm Innovation Center, Inc. is a member of the Code Aurora Forum.
> --
> --
> To unsubscribe from this list: send the line "unsubscribe linux-kernel" in
> the body of a message to [email protected]
> More majordomo info at http://vger.kernel.org/majordomo-info.html
> Please read the FAQ at http://www.tux.org/lkml/

And again, thanks for writing this driver. I look forward to have it
mainlined! (with support for my 8960 board ;))

Regards,
Bjorn

2013-07-30 21:10:40

by Hanumant Singh

[permalink] [raw]
Subject: Re: [PATCH] pinctrl: msm: Add support for MSM TLMM pinmux

On 7/29/2013 9:37 AM, Linus Walleij wrote:
> On Wed, Jul 24, 2013 at 11:41 PM, Hanumant Singh
> <[email protected]> wrote:
>
>> Add a new device tree enabled pinctrl driver for
>> Qualcomm MSM SoC's. This driver provides an extensible
>> framework to interface all MSM's that use a TLMM pinmux,
>> with the pinctrl subsytem.
>>
>> This driver is split into two parts: the pinctrl interface
>> and the TLMM version specific implementation. The pinctrl
>> interface parses the device tree and registers with the pinctrl
>> subsytem. The TLMM version specific implementation supports
>> pin configuration/register programming for the different
>> pin types present on a given TLMM pinmux version.
>>
>> Add support only for TLMM version 3 pinmux right now,
>> as well as, only two of the different pin types present on the
>> TLMM v3 pinmux.
>> Pintype 1: General purpose pins.
>> Pintype 2: SDC pins.
>
> I guess this is the v4 patch set?
>
> Please include a small changelog so I can keep track of
> things...
>
>
>> Change-Id: I065d874cd2c6fd002d5b3cb4b355445bb6912bf4
>
> This thing is not interesting to the kernel community.
>
>> diff --git a/Documentation/devicetree/bindings/pinctrl/msm-pinctrl.txt b/Documentation/devicetree/bindings/pinctrl/msm-pinctrl.txt
>> new file mode 100644
>> index 0000000..0f17a94
>> --- /dev/null
>> +++ b/Documentation/devicetree/bindings/pinctrl/msm-pinctrl.txt
>
> This needs to be broken out and sent for separate review on
> the [email protected] mailing list.
>
>> +Required Properties
>> + - qcom,pin-type-*: identifies the pin type. Pin types supported are
>> + qcom,pin-type-gp (General purpose)
>> + qcom,pin-type-sdc (SDC)
>
> I am wondering if it it would not be simpler to split this
> driver in two, one for the GP pins and one for the SDC pins.
> Having this tag on each and every pin seems *very*
> awkward.
>
> If you have two drivers and two device tree nodes, one for
> GP pins and one for SDC pins, just separated by different
> compatible strings and different memory regs, things will
> look simpler and be easier to maintain I think.
>
> Unless there is some strong cross-dependency between
> GP and SDC please explore this approach.

Actually we have had this discussion.
There are more pin types on a TLMM then just these 2.
The registers are interleaved.
They are no separated at a clean page boundary.
Also there is mode register that governs the function mode of the
"unifunction"/single function pins like SDC. This register is again
interspersed in the address map.
Having it the way it is right now, will help me add clean support for
other pintypes on a given TLMM version.

>
>> +- Pin groups as child nodes: The pin mux (selecting pin function
>> + mode) and pin config (pull up/down, driver strength, direction) settings are
>> + represented as child nodes of the pin-controller node. There is no limit on
>> + the count of these child nodes.
>> +
>> + Required Properties
>> + -qcom,pins: phandle specifying pin type and a pin number.
>> + -qcom,num-grp-pins: number of pins in the group.
>> + -label: name to to identify the pin group.
>> +
>> + Optional Properties
>> + -qcom,pin-func: function setting for the pin group.
>
> After some scratching my head I realize that you are trying
> to reimplement parts of pinctrl-single.c.
>
> I.e. you try to put all the definitions of groups and functions
> into the devicetree instead of having this in the driver
> as tables.
>
> If this is what you want to do you should use pinctrl-single.c.
> It might be possible to use pinctrl-single.c only for the
> GP pins.
I would prefer not to split the driver into one driver for its own
pintype for above reasons.
Also there isn't just one register per pin for gp pins, there are more
registers.
For example to configure direction, I have to touch additional registers
then just the config register.
Also for future TLMM version, we are very likely to move away from
single config register.
>
> But if there is something complex about the hardware that
> make pinctrl-single.c not fit the bill I advice you to encode
> the lists of groups and functions into the driver instead.
>
> Also, split this in a per-soc manner (compare the other
> drivers) so you can support plugging one file per SoC
> for the different MSM chips using this pin controller.
>
We actually have the same TLMM pinmux used by several socs of a family.
The number of pins on each soc may vary.
Also a given soc gets used in a number of boards.
The device tree for a given soc is split into the different boards that
its in ie the boards inherit a common soc.dtsi but have separate dts.
The boards for the same soc may use different pin groups for
accomplishing a function, since we have multiple i2c, spi uart etc
peripheral instances on a soc. A different instance of each of the above
peripherals, can be used in different boards, utilizing different
or subset of same pin groups.
Thus I would need to have multiple C files for one soc, based on the
boards that it goes into.

Thanks
Hanumant


--
Sent by an employee of the Qualcomm Innovation Center, Inc.
The Qualcomm Innovation Center, Inc. is a member of the Code Aurora Forum.
--

2013-07-30 21:22:43

by Stephen Warren

[permalink] [raw]
Subject: Re: [PATCH] pinctrl: msm: Add support for MSM TLMM pinmux

On 07/30/2013 03:10 PM, hanumant wrote:
...
> We actually have the same TLMM pinmux used by several socs of a family.
> The number of pins on each soc may vary.
> Also a given soc gets used in a number of boards.
> The device tree for a given soc is split into the different boards that
> its in ie the boards inherit a common soc.dtsi but have separate dts.
> The boards for the same soc may use different pin groups for
> accomplishing a function, since we have multiple i2c, spi uart etc
> peripheral instances on a soc. A different instance of each of the above
> peripherals, can be used in different boards, utilizing different
> or subset of same pin groups.
> Thus I would need to have multiple C files for one soc, based on the
> boards that it goes into.

The pinctrl driver should be exposing the raw capabilities of the HW.
All the board-specific configuration should be expressed in DT. So, the
driver shouldn't have to know anything about different boards at
compile-time.

2013-07-31 00:01:07

by Hanumant Singh

[permalink] [raw]
Subject: Re: [PATCH] pinctrl: msm: Add support for MSM TLMM pinmux

On 7/30/2013 2:22 PM, Stephen Warren wrote:
> On 07/30/2013 03:10 PM, hanumant wrote:
> ...
>> We actually have the same TLMM pinmux used by several socs of a family.
>> The number of pins on each soc may vary.
>> Also a given soc gets used in a number of boards.
>> The device tree for a given soc is split into the different boards that
>> its in ie the boards inherit a common soc.dtsi but have separate dts.
>> The boards for the same soc may use different pin groups for
>> accomplishing a function, since we have multiple i2c, spi uart etc
>> peripheral instances on a soc. A different instance of each of the above
>> peripherals, can be used in different boards, utilizing different
>> or subset of same pin groups.
>> Thus I would need to have multiple C files for one soc, based on the
>> boards that it goes into.
>
> The pinctrl driver should be exposing the raw capabilities of the HW.
> All the board-specific configuration should be expressed in DT. So, the
> driver shouldn't have to know anything about different boards at
> compile-time.
>
I agree, so I wanted to keep the pin grouping information in DT, we
already have a board based differentiation of dts files in DT, for the
same soc.

Thanks
Hanumant

--
Sent by an employee of the Qualcomm Innovation Center, Inc.
The Qualcomm Innovation Center, Inc. is a member of the Code Aurora Forum.
--

2013-07-31 00:08:15

by Stephen Warren

[permalink] [raw]
Subject: Re: [PATCH] pinctrl: msm: Add support for MSM TLMM pinmux

On 07/30/2013 06:01 PM, Hanumant Singh wrote:
> On 7/30/2013 2:22 PM, Stephen Warren wrote:
>> On 07/30/2013 03:10 PM, hanumant wrote:
>> ...
>>> We actually have the same TLMM pinmux used by several socs of a family.
>>> The number of pins on each soc may vary.
>>> Also a given soc gets used in a number of boards.
>>> The device tree for a given soc is split into the different boards that
>>> its in ie the boards inherit a common soc.dtsi but have separate dts.
>>> The boards for the same soc may use different pin groups for
>>> accomplishing a function, since we have multiple i2c, spi uart etc
>>> peripheral instances on a soc. A different instance of each of the above
>>> peripherals, can be used in different boards, utilizing different
>>> or subset of same pin groups.
>>> Thus I would need to have multiple C files for one soc, based on the
>>> boards that it goes into.
>>
>> The pinctrl driver should be exposing the raw capabilities of the HW.
>> All the board-specific configuration should be expressed in DT. So, the
>> driver shouldn't have to know anything about different boards at
>> compile-time.
>>
> I agree, so I wanted to keep the pin grouping information in DT, we
> already have a board based differentiation of dts files in DT, for the
> same soc.

That's the opposite of what I was saying. Pin groups are a feature of
the SoC design, not the board.

2013-07-31 00:13:07

by Hanumant Singh

[permalink] [raw]
Subject: Re: [PATCH] pinctrl: msm: Add support for MSM TLMM pinmux

On 7/30/2013 5:08 PM, Stephen Warren wrote:
> On 07/30/2013 06:01 PM, Hanumant Singh wrote:
>> On 7/30/2013 2:22 PM, Stephen Warren wrote:
>>> On 07/30/2013 03:10 PM, hanumant wrote:
>>> ...
>>>> We actually have the same TLMM pinmux used by several socs of a family.
>>>> The number of pins on each soc may vary.
>>>> Also a given soc gets used in a number of boards.
>>>> The device tree for a given soc is split into the different boards that
>>>> its in ie the boards inherit a common soc.dtsi but have separate dts.
>>>> The boards for the same soc may use different pin groups for
>>>> accomplishing a function, since we have multiple i2c, spi uart etc
>>>> peripheral instances on a soc. A different instance of each of the above
>>>> peripherals, can be used in different boards, utilizing different
>>>> or subset of same pin groups.
>>>> Thus I would need to have multiple C files for one soc, based on the
>>>> boards that it goes into.
>>>
>>> The pinctrl driver should be exposing the raw capabilities of the HW.
>>> All the board-specific configuration should be expressed in DT. So, the
>>> driver shouldn't have to know anything about different boards at
>>> compile-time.
>>>
>> I agree, so I wanted to keep the pin grouping information in DT, we
>> already have a board based differentiation of dts files in DT, for the
>> same soc.
>
> That's the opposite of what I was saying. Pin groups are a feature of
> the SoC design, not the board.
>
Sorry I guess I wasn't clear.
Right now I have a soc-pinctrl.dtsi containing pin groupings.
This will be "inherited" by soc-boardtype.dts.
The pinctrl client device nodes in soc-boardtype.dts will point to pin
groupings in soc-pinctrl.dtsi that are valid for that particular boardtype.
Is this a valid design?

Thanks
Hanumant
that are valid for that


--
Sent by an employee of the Qualcomm Innovation Center, Inc.
The Qualcomm Innovation Center, Inc. is a member of the Code Aurora Forum.
--

2013-07-31 03:59:45

by Stephen Warren

[permalink] [raw]
Subject: Re: [PATCH] pinctrl: msm: Add support for MSM TLMM pinmux

On 07/30/2013 06:13 PM, Hanumant Singh wrote:
> On 7/30/2013 5:08 PM, Stephen Warren wrote:
>> On 07/30/2013 06:01 PM, Hanumant Singh wrote:
>>> On 7/30/2013 2:22 PM, Stephen Warren wrote:
>>>> On 07/30/2013 03:10 PM, hanumant wrote:
>>>> ...
>>>>> We actually have the same TLMM pinmux used by several socs of a
>>>>> family.
>>>>> The number of pins on each soc may vary.
>>>>> Also a given soc gets used in a number of boards.
>>>>> The device tree for a given soc is split into the different boards
>>>>> that
>>>>> its in ie the boards inherit a common soc.dtsi but have separate dts.
>>>>> The boards for the same soc may use different pin groups for
>>>>> accomplishing a function, since we have multiple i2c, spi uart etc
>>>>> peripheral instances on a soc. A different instance of each of the
>>>>> above
>>>>> peripherals, can be used in different boards, utilizing different
>>>>> or subset of same pin groups.
>>>>> Thus I would need to have multiple C files for one soc, based on the
>>>>> boards that it goes into.
>>>>
>>>> The pinctrl driver should be exposing the raw capabilities of the HW.
>>>> All the board-specific configuration should be expressed in DT. So, the
>>>> driver shouldn't have to know anything about different boards at
>>>> compile-time.
>>>>
>>> I agree, so I wanted to keep the pin grouping information in DT, we
>>> already have a board based differentiation of dts files in DT, for the
>>> same soc.
>>
>> That's the opposite of what I was saying. Pin groups are a feature of
>> the SoC design, not the board.
>>
> Sorry I guess I wasn't clear.
> Right now I have a soc-pinctrl.dtsi containing pin groupings.
> This will be "inherited" by soc-boardtype.dts.
> The pinctrl client device nodes in soc-boardtype.dts will point to pin
> groupings in soc-pinctrl.dtsi that are valid for that particular boardtype.
> Is this a valid design?

OK, so you have two types of child node inside the pinctrl DT node; some
define the pin groups the SoC has (in soc.dtsi) and some define pinctrl
states that reference the pin group nodes and are referenced by the
client nodes.

That's probably fine. However, I'd still question putting the pin group
nodes in DT at all; I'm not convinced it's better than just putting
those into the driver itself. You end up with the same data tables after
parsing the DT anyway.

2013-07-31 19:46:48

by Hanumant Singh

[permalink] [raw]
Subject: Re: [PATCH] pinctrl: msm: Add support for MSM TLMM pinmux

On 7/30/2013 8:59 PM, Stephen Warren wrote:
> On 07/30/2013 06:13 PM, Hanumant Singh wrote:
>> On 7/30/2013 5:08 PM, Stephen Warren wrote:
>>> On 07/30/2013 06:01 PM, Hanumant Singh wrote:
>>>> On 7/30/2013 2:22 PM, Stephen Warren wrote:
>>>>> On 07/30/2013 03:10 PM, hanumant wrote:
>>>>> ...
>>>>>> We actually have the same TLMM pinmux used by several socs of a
>>>>>> family.
>>>>>> The number of pins on each soc may vary.
>>>>>> Also a given soc gets used in a number of boards.
>>>>>> The device tree for a given soc is split into the different boards
>>>>>> that
>>>>>> its in ie the boards inherit a common soc.dtsi but have separate dts.
>>>>>> The boards for the same soc may use different pin groups for
>>>>>> accomplishing a function, since we have multiple i2c, spi uart etc
>>>>>> peripheral instances on a soc. A different instance of each of the
>>>>>> above
>>>>>> peripherals, can be used in different boards, utilizing different
>>>>>> or subset of same pin groups.
>>>>>> Thus I would need to have multiple C files for one soc, based on the
>>>>>> boards that it goes into.
>>>>>
>>>>> The pinctrl driver should be exposing the raw capabilities of the HW.
>>>>> All the board-specific configuration should be expressed in DT. So, the
>>>>> driver shouldn't have to know anything about different boards at
>>>>> compile-time.
>>>>>
>>>> I agree, so I wanted to keep the pin grouping information in DT, we
>>>> already have a board based differentiation of dts files in DT, for the
>>>> same soc.
>>>
>>> That's the opposite of what I was saying. Pin groups are a feature of
>>> the SoC design, not the board.
>>>
>> Sorry I guess I wasn't clear.
>> Right now I have a soc-pinctrl.dtsi containing pin groupings.
>> This will be "inherited" by soc-boardtype.dts.
>> The pinctrl client device nodes in soc-boardtype.dts will point to pin
>> groupings in soc-pinctrl.dtsi that are valid for that particular boardtype.
>> Is this a valid design?
>
> OK, so you have two types of child node inside the pinctrl DT node; some
> define the pin groups the SoC has (in soc.dtsi) and some define pinctrl
> states that reference the pin group nodes and are referenced by the
> client nodes.
>
> That's probably fine. However, I'd still question putting the pin group
> nodes in DT at all; I'm not convinced it's better than just putting
> those into the driver itself. You end up with the same data tables after
> parsing the DT anyway.
>

Any feedback for the rest of the patch?

Thanks
Hanumant

--
Sent by an employee of the Qualcomm Innovation Center, Inc.
The Qualcomm Innovation Center, Inc. is a member of the Code Aurora Forum.
--

2013-07-31 21:06:18

by Stephen Warren

[permalink] [raw]
Subject: Re: [PATCH] pinctrl: msm: Add support for MSM TLMM pinmux

On 07/31/2013 01:46 PM, Hanumant Singh wrote:
> On 7/30/2013 8:59 PM, Stephen Warren wrote:
>> On 07/30/2013 06:13 PM, Hanumant Singh wrote:
>>> On 7/30/2013 5:08 PM, Stephen Warren wrote:
>>>> On 07/30/2013 06:01 PM, Hanumant Singh wrote:
>>>>> On 7/30/2013 2:22 PM, Stephen Warren wrote:
>>>>>> On 07/30/2013 03:10 PM, hanumant wrote:
>>>>>> ...
>>>>>>> We actually have the same TLMM pinmux used by several socs of a
>>>>>>> family.
>>>>>>> The number of pins on each soc may vary.
>>>>>>> Also a given soc gets used in a number of boards.
>>>>>>> The device tree for a given soc is split into the different boards
>>>>>>> that
>>>>>>> its in ie the boards inherit a common soc.dtsi but have separate
>>>>>>> dts.
>>>>>>> The boards for the same soc may use different pin groups for
>>>>>>> accomplishing a function, since we have multiple i2c, spi uart etc
>>>>>>> peripheral instances on a soc. A different instance of each of the
>>>>>>> above
>>>>>>> peripherals, can be used in different boards, utilizing different
>>>>>>> or subset of same pin groups.
>>>>>>> Thus I would need to have multiple C files for one soc, based on the
>>>>>>> boards that it goes into.
>>>>>>
>>>>>> The pinctrl driver should be exposing the raw capabilities of the HW.
>>>>>> All the board-specific configuration should be expressed in DT.
>>>>>> So, the
>>>>>> driver shouldn't have to know anything about different boards at
>>>>>> compile-time.
>>>>>>
>>>>> I agree, so I wanted to keep the pin grouping information in DT, we
>>>>> already have a board based differentiation of dts files in DT, for the
>>>>> same soc.
>>>>
>>>> That's the opposite of what I was saying. Pin groups are a feature of
>>>> the SoC design, not the board.
>>>>
>>> Sorry I guess I wasn't clear.
>>> Right now I have a soc-pinctrl.dtsi containing pin groupings.
>>> This will be "inherited" by soc-boardtype.dts.
>>> The pinctrl client device nodes in soc-boardtype.dts will point to pin
>>> groupings in soc-pinctrl.dtsi that are valid for that particular
>>> boardtype.
>>> Is this a valid design?
>>
>> OK, so you have two types of child node inside the pinctrl DT node; some
>> define the pin groups the SoC has (in soc.dtsi) and some define pinctrl
>> states that reference the pin group nodes and are referenced by the
>> client nodes.
>>
>> That's probably fine. However, I'd still question putting the pin group
>> nodes in DT at all; I'm not convinced it's better than just putting
>> those into the driver itself. You end up with the same data tables after
>> parsing the DT anyway.
>>
>
> Any feedback for the rest of the patch?

I'm certainly waiting for this aspect of the patch to be resolved; I
think it will impact the rest of the patch so much that it's not worth
reviewing until we decide on where to represent the pin groups (some DT
parsing could would be removed if we put the pin group definitions into
the driver, hence wouldn't need to be reviewed, and likewise there's be
some new tables to review).

2013-08-01 00:17:59

by Hanumant Singh

[permalink] [raw]
Subject: Re: [PATCH] pinctrl: msm: Add support for MSM TLMM pinmux

On 7/31/2013 2:06 PM, Stephen Warren wrote:
> On 07/31/2013 01:46 PM, Hanumant Singh wrote:
>> On 7/30/2013 8:59 PM, Stephen Warren wrote:
>>> On 07/30/2013 06:13 PM, Hanumant Singh wrote:
>>>> On 7/30/2013 5:08 PM, Stephen Warren wrote:
>>>>> On 07/30/2013 06:01 PM, Hanumant Singh wrote:
>>>>>> On 7/30/2013 2:22 PM, Stephen Warren wrote:
>>>>>>> On 07/30/2013 03:10 PM, hanumant wrote:
>>>>>>> ...
>>>>>>>> We actually have the same TLMM pinmux used by several socs of a
>>>>>>>> family.
>>>>>>>> The number of pins on each soc may vary.
>>>>>>>> Also a given soc gets used in a number of boards.
>>>>>>>> The device tree for a given soc is split into the different boards
>>>>>>>> that
>>>>>>>> its in ie the boards inherit a common soc.dtsi but have separate
>>>>>>>> dts.
>>>>>>>> The boards for the same soc may use different pin groups for
>>>>>>>> accomplishing a function, since we have multiple i2c, spi uart etc
>>>>>>>> peripheral instances on a soc. A different instance of each of the
>>>>>>>> above
>>>>>>>> peripherals, can be used in different boards, utilizing different
>>>>>>>> or subset of same pin groups.
>>>>>>>> Thus I would need to have multiple C files for one soc, based on the
>>>>>>>> boards that it goes into.
>>>>>>>
>>>>>>> The pinctrl driver should be exposing the raw capabilities of the HW.
>>>>>>> All the board-specific configuration should be expressed in DT.
>>>>>>> So, the
>>>>>>> driver shouldn't have to know anything about different boards at
>>>>>>> compile-time.
>>>>>>>
>>>>>> I agree, so I wanted to keep the pin grouping information in DT, we
>>>>>> already have a board based differentiation of dts files in DT, for the
>>>>>> same soc.
>>>>>
>>>>> That's the opposite of what I was saying. Pin groups are a feature of
>>>>> the SoC design, not the board.
>>>>>
>>>> Sorry I guess I wasn't clear.
>>>> Right now I have a soc-pinctrl.dtsi containing pin groupings.
>>>> This will be "inherited" by soc-boardtype.dts.
>>>> The pinctrl client device nodes in soc-boardtype.dts will point to pin
>>>> groupings in soc-pinctrl.dtsi that are valid for that particular
>>>> boardtype.
>>>> Is this a valid design?
>>>
>>> OK, so you have two types of child node inside the pinctrl DT node; some
>>> define the pin groups the SoC has (in soc.dtsi) and some define pinctrl
>>> states that reference the pin group nodes and are referenced by the
>>> client nodes.
>>>
>>> That's probably fine. However, I'd still question putting the pin group
>>> nodes in DT at all; I'm not convinced it's better than just putting
>>> those into the driver itself. You end up with the same data tables after
>>> parsing the DT anyway.
>>>
>>
>> Any feedback for the rest of the patch?
>
> I'm certainly waiting for this aspect of the patch to be resolved; I
> think it will impact the rest of the patch so much that it's not worth
> reviewing until we decide on where to represent the pin groups (some DT
> parsing could would be removed if we put the pin group definitions into
> the driver, hence wouldn't need to be reviewed, and likewise there's be
> some new tables to review).
>

I am trying to look at examples of what you are suggesting.
I was looking at the exynos implementation, and just from a brief glance
it seems like there too the pin grouping is being specified in the
device tree, using what looks like labels of the pins.
The labels are matched to group structures in soc specific files?

By having the pin groupings in DT I am able to reuse the driver without
any SOC based code bloat.
As I mentioned earlier, we have entire families of SOCs using the same
TLMM hardware.
Its not a guarantee that for a given TLMM version,
the pin groupings on that hardware are the same for every SOC that its
in. Its infact most likely that I wont be able to use the pin groupings
from one SOC to the next even if they both use the same TLMM.
It will very quickly lead to a bloat of
pinctrl-<msm_soc>.c (containing the pin groupings replicated for each soc)
which use TLMM version specific register programming implementation
pinctrl-tlmm-<version>.c
and the DT parsing and interface to framework (which remains unchanged).
pinctrl-msm.c.

Thanks
Hanumant

--
Sent by an employee of the Qualcomm Innovation Center, Inc.
The Qualcomm Innovation Center, Inc. is a member of the Code Aurora Forum.
--

2013-08-06 23:45:21

by Hanumant Singh

[permalink] [raw]
Subject: Re: [PATCH] pinctrl: msm: Add support for MSM TLMM pinmux

On 7/31/2013 5:17 PM, Hanumant Singh wrote:
> On 7/31/2013 2:06 PM, Stephen Warren wrote:
>> On 07/31/2013 01:46 PM, Hanumant Singh wrote:
>>> On 7/30/2013 8:59 PM, Stephen Warren wrote:
>>>> On 07/30/2013 06:13 PM, Hanumant Singh wrote:
>>>>> On 7/30/2013 5:08 PM, Stephen Warren wrote:
>>>>>> On 07/30/2013 06:01 PM, Hanumant Singh wrote:
>>>>>>> On 7/30/2013 2:22 PM, Stephen Warren wrote:
>>>>>>>> On 07/30/2013 03:10 PM, hanumant wrote:
>>>>>>>> ...
>>>>>>>>> We actually have the same TLMM pinmux used by several socs of a
>>>>>>>>> family.
>>>>>>>>> The number of pins on each soc may vary.
>>>>>>>>> Also a given soc gets used in a number of boards.
>>>>>>>>> The device tree for a given soc is split into the different boards
>>>>>>>>> that
>>>>>>>>> its in ie the boards inherit a common soc.dtsi but have separate
>>>>>>>>> dts.
>>>>>>>>> The boards for the same soc may use different pin groups for
>>>>>>>>> accomplishing a function, since we have multiple i2c, spi uart etc
>>>>>>>>> peripheral instances on a soc. A different instance of each of the
>>>>>>>>> above
>>>>>>>>> peripherals, can be used in different boards, utilizing different
>>>>>>>>> or subset of same pin groups.
>>>>>>>>> Thus I would need to have multiple C files for one soc, based
>>>>>>>>> on the
>>>>>>>>> boards that it goes into.
>>>>>>>>
>>>>>>>> The pinctrl driver should be exposing the raw capabilities of
>>>>>>>> the HW.
>>>>>>>> All the board-specific configuration should be expressed in DT.
>>>>>>>> So, the
>>>>>>>> driver shouldn't have to know anything about different boards at
>>>>>>>> compile-time.
>>>>>>>>
>>>>>>> I agree, so I wanted to keep the pin grouping information in DT, we
>>>>>>> already have a board based differentiation of dts files in DT,
>>>>>>> for the
>>>>>>> same soc.
>>>>>>
>>>>>> That's the opposite of what I was saying. Pin groups are a feature of
>>>>>> the SoC design, not the board.
>>>>>>
>>>>> Sorry I guess I wasn't clear.
>>>>> Right now I have a soc-pinctrl.dtsi containing pin groupings.
>>>>> This will be "inherited" by soc-boardtype.dts.
>>>>> The pinctrl client device nodes in soc-boardtype.dts will point to pin
>>>>> groupings in soc-pinctrl.dtsi that are valid for that particular
>>>>> boardtype.
>>>>> Is this a valid design?
>>>>
>>>> OK, so you have two types of child node inside the pinctrl DT node;
>>>> some
>>>> define the pin groups the SoC has (in soc.dtsi) and some define pinctrl
>>>> states that reference the pin group nodes and are referenced by the
>>>> client nodes.
>>>>
>>>> That's probably fine. However, I'd still question putting the pin group
>>>> nodes in DT at all; I'm not convinced it's better than just putting
>>>> those into the driver itself. You end up with the same data tables
>>>> after
>>>> parsing the DT anyway.
>>>>
>>>
>>> Any feedback for the rest of the patch?
>>
>> I'm certainly waiting for this aspect of the patch to be resolved; I
>> think it will impact the rest of the patch so much that it's not worth
>> reviewing until we decide on where to represent the pin groups (some DT
>> parsing could would be removed if we put the pin group definitions into
>> the driver, hence wouldn't need to be reviewed, and likewise there's be
>> some new tables to review).
>>
>
> I am trying to look at examples of what you are suggesting.
> I was looking at the exynos implementation, and just from a brief glance
> it seems like there too the pin grouping is being specified in the
> device tree, using what looks like labels of the pins.
> The labels are matched to group structures in soc specific files?
>
> By having the pin groupings in DT I am able to reuse the driver without
> any SOC based code bloat.
> As I mentioned earlier, we have entire families of SOCs using the same
> TLMM hardware.
> Its not a guarantee that for a given TLMM version,
> the pin groupings on that hardware are the same for every SOC that its
> in. Its infact most likely that I wont be able to use the pin groupings
> from one SOC to the next even if they both use the same TLMM.
> It will very quickly lead to a bloat of
> pinctrl-<msm_soc>.c (containing the pin groupings replicated for each soc)
> which use TLMM version specific register programming implementation
> pinctrl-tlmm-<version>.c
> and the DT parsing and interface to framework (which remains unchanged).
> pinctrl-msm.c.
>
> Thanks
> Hanumant
>

Any comments on this?

Thanks
Hanumant


--
Sent by an employee of the Qualcomm Innovation Center, Inc.
The Qualcomm Innovation Center, Inc. is a member of the Code Aurora Forum.
--

2013-08-07 16:00:20

by Stephen Warren

[permalink] [raw]
Subject: Re: [PATCH] pinctrl: msm: Add support for MSM TLMM pinmux

On 08/06/2013 05:45 PM, Hanumant Singh wrote:
> On 7/31/2013 5:17 PM, Hanumant Singh wrote:
>> On 7/31/2013 2:06 PM, Stephen Warren wrote:
>>> On 07/31/2013 01:46 PM, Hanumant Singh wrote:
>>>> On 7/30/2013 8:59 PM, Stephen Warren wrote:
>>>>> On 07/30/2013 06:13 PM, Hanumant Singh wrote:
>>>>>> On 7/30/2013 5:08 PM, Stephen Warren wrote:
>>>>>>> On 07/30/2013 06:01 PM, Hanumant Singh wrote:
>>>>>>>> On 7/30/2013 2:22 PM, Stephen Warren wrote:
>>>>>>>>> On 07/30/2013 03:10 PM, hanumant wrote:
>>>>>>>>> ...
>>>>>>>>>> We actually have the same TLMM pinmux used by several socs of a
>>>>>>>>>> family.
>>>>>>>>>> The number of pins on each soc may vary.
>>>>>>>>>> Also a given soc gets used in a number of boards.
>>>>>>>>>> The device tree for a given soc is split into the different
>>>>>>>>>> boards
>>>>>>>>>> that
>>>>>>>>>> its in ie the boards inherit a common soc.dtsi but have separate
>>>>>>>>>> dts.
>>>>>>>>>> The boards for the same soc may use different pin groups for
>>>>>>>>>> accomplishing a function, since we have multiple i2c, spi uart
>>>>>>>>>> etc
>>>>>>>>>> peripheral instances on a soc. A different instance of each of
>>>>>>>>>> the
>>>>>>>>>> above
>>>>>>>>>> peripherals, can be used in different boards, utilizing different
>>>>>>>>>> or subset of same pin groups.
>>>>>>>>>> Thus I would need to have multiple C files for one soc, based
>>>>>>>>>> on the
>>>>>>>>>> boards that it goes into.
>>>>>>>>>
>>>>>>>>> The pinctrl driver should be exposing the raw capabilities of
>>>>>>>>> the HW.
>>>>>>>>> All the board-specific configuration should be expressed in DT.
>>>>>>>>> So, the
>>>>>>>>> driver shouldn't have to know anything about different boards at
>>>>>>>>> compile-time.
>>>>>>>>>
>>>>>>>> I agree, so I wanted to keep the pin grouping information in DT, we
>>>>>>>> already have a board based differentiation of dts files in DT,
>>>>>>>> for the
>>>>>>>> same soc.
>>>>>>>
>>>>>>> That's the opposite of what I was saying. Pin groups are a
>>>>>>> feature of
>>>>>>> the SoC design, not the board.
>>>>>>>
>>>>>> Sorry I guess I wasn't clear.
>>>>>> Right now I have a soc-pinctrl.dtsi containing pin groupings.
>>>>>> This will be "inherited" by soc-boardtype.dts.
>>>>>> The pinctrl client device nodes in soc-boardtype.dts will point to
>>>>>> pin
>>>>>> groupings in soc-pinctrl.dtsi that are valid for that particular
>>>>>> boardtype.
>>>>>> Is this a valid design?
>>>>>
>>>>> OK, so you have two types of child node inside the pinctrl DT node;
>>>>> some
>>>>> define the pin groups the SoC has (in soc.dtsi) and some define
>>>>> pinctrl
>>>>> states that reference the pin group nodes and are referenced by the
>>>>> client nodes.
>>>>>
>>>>> That's probably fine. However, I'd still question putting the pin
>>>>> group
>>>>> nodes in DT at all; I'm not convinced it's better than just putting
>>>>> those into the driver itself. You end up with the same data tables
>>>>> after
>>>>> parsing the DT anyway.
>>>>>
>>>>
>>>> Any feedback for the rest of the patch?
>>>
>>> I'm certainly waiting for this aspect of the patch to be resolved; I
>>> think it will impact the rest of the patch so much that it's not worth
>>> reviewing until we decide on where to represent the pin groups (some DT
>>> parsing could would be removed if we put the pin group definitions into
>>> the driver, hence wouldn't need to be reviewed, and likewise there's be
>>> some new tables to review).
>>>
>>
>> I am trying to look at examples of what you are suggesting.
>> I was looking at the exynos implementation, and just from a brief glance
>> it seems like there too the pin grouping is being specified in the
>> device tree, using what looks like labels of the pins.
>> The labels are matched to group structures in soc specific files?
>>
>> By having the pin groupings in DT I am able to reuse the driver without
>> any SOC based code bloat.
>> As I mentioned earlier, we have entire families of SOCs using the same
>> TLMM hardware.
>> Its not a guarantee that for a given TLMM version,
>> the pin groupings on that hardware are the same for every SOC that its
>> in. Its infact most likely that I wont be able to use the pin groupings
>> from one SOC to the next even if they both use the same TLMM.
>> It will very quickly lead to a bloat of
>> pinctrl-<msm_soc>.c (containing the pin groupings replicated for each
>> soc)
>> which use TLMM version specific register programming implementation
>> pinctrl-tlmm-<version>.c
>> and the DT parsing and interface to framework (which remains unchanged).
>> pinctrl-msm.c.
>>
>> Thanks
>> Hanumant
>>
>
> Any comments on this?

No. As I said, I personally want to see all the pingroups defined in the
pinctrl driver. But, if someone else acks/... the patches without it, I
probably won't nack it.

2013-08-07 18:07:24

by Hanumant Singh

[permalink] [raw]
Subject: Re: [PATCH] pinctrl: msm: Add support for MSM TLMM pinmux

On 7/29/2013 4:39 PM, Bjorn Andersson wrote:
> On Wed, Jul 24, 2013 at 1:41 PM, Hanumant Singh <[email protected]> wrote:
>> Add a new device tree enabled pinctrl driver for
>> Qualcomm MSM SoC's. This driver provides an extensible
>> framework to interface all MSM's that use a TLMM pinmux,
>> with the pinctrl subsytem.
>
> Thanks! I was about to write an implementation for v2 myself when I
> found your patch. I hacked a little bit on it and got it to work on my
> 8960 based board.
>
Sorry for the delayed response. Right now the focus is to add support
for the pinmux on the snapdragon 800 soc which uses the V3.
>>
>> This driver is split into two parts: the pinctrl interface
>> and the TLMM version specific implementation. The pinctrl
>> interface parses the device tree and registers with the pinctrl
>> subsytem. The TLMM version specific implementation supports
>> pin configuration/register programming for the different
>> pin types present on a given TLMM pinmux version.
>>
>> Add support only for TLMM version 3 pinmux right now,
>> as well as, only two of the different pin types present on the
>> TLMM v3 pinmux.
>> Pintype 1: General purpose pins.
>> Pintype 2: SDC pins.
>
> It seems that the gp pins of TLMM v3 have the same layout and offsets
> as the once in v2. Unless I'm missing something I think this patch
> should be structured (and have appropriate naming) in a way that we
> can support both of them.
>
No, Actually there are significant differences in the v2 and v3 register
offsets and even hardware capabilities.
We are currently trying to add support for our snapdragon 800
soc which uses V3.
> As a general note on the patch, the pins and pin groups are defined by
> the soc, I'm therefore not convinced that these should be configured
> from the devicetree. It's at least not how it's done in other
> platforms.
>

I have been having a email discussion with Stephen Warren on this.

> After talking to Linus a little bit I concluded that the way the
> design idea behind pinmux is that you have pins and you have
> functions;
> * pins are your 117 gp-pins and with some imagination your sd pins, so
> that's fine.
> * the function is some functionality provided by the hard
> Then you can set up muxes between these two. Therefore I think your
> use of the word "function" in the patch is what normally is called a
> mux. It might be worth sorting this out, to make it easier to maintain
> this code down the road.
>
> @Linus, can you please confirm my understanding of the design?
>
>>

I am not sure that the use of the term func is misleading. You can look
at the samsung-pinctrl.txt documentation for a similar use of the term.


>> +Example 2: Spi pin entries within the pincontroller node
>> + pinctrl@fd511000 {
>> + ....
>> + ..
>> + pmx-spi-bus {
>> + /*
>> + * MOSI, MISO and CLK lines
>> + * all sharing same function and config
>> + * settings for each configuration node.
>> + */
>> + qcom,pins = <&gp 0>, <&gp 1>, <&gp 3>;
>> + qcom,num-grp-pins = <3>;
>
> qcom,num-grp-pins is not used by the code, please remove it.

Will do
>
>> + qcom,pin-func = <1>;
>> + label = "spi-bus";
>> +
>> + /* Active configuration of bus pins */
>> + spi-bus-active: spi-bus-active {
>> + /*
>> + * Property names as specified in
>> + * pinctrl-bindings.txt
>> + */
>> + drive-strength = <8>; /* 8 MA */
>> + bias-disable; /* No PULL */
>> + };
>> + /* Sleep configuration of bus pins */
>> + spi-bus-sleep: spi-bus-sleep {
>> + /*
>> + * Property values as specified in HW
>> + * manual.
>> + */
>> + drive-strength = <2>; /* 2 MA */
>> + bias-disable;
>> + };
>> + };
>> +
>> + pmx-spi-cs {
>> + /*
>> + * Chip select for SPI
>> + * different config
>> + * settings as compared to bus pins.
>> + */
>> + qcom,pins = <&Gp 2>;
>
> Make Gp lower case.
>
Will do


>> +Example 3: A SPI client node that supports 'active' and 'sleep' states.
>> +
>> + spi_0: spi@f9923000 { /* BLSP1 QUP1 */
>> + compatible = "qcom,spi-qup-v2";
>> + #address-cells = <1>;
>> + #size-cells = <0>;
>> + reg-names = "spi_physical", "spi_bam_physical";
>> + reg = <0xf9923000 0x1000>,
>> + <0xf9904000 0xF000>;
>> + interrupt-names = "spi_irq", "spi_bam_irq";
>> + interrupts = <0 95 0>, <0 238 0>;
>> + spi-max-frequency = <19200000>;
>> +
>> + /* pins used by spi controllers */
>> + pinctrl-names = "default", "sleep";
>> + pinctrl-0 = <&spi-bus-active &spi-cs-active>;
>> + pinctrl-1 = <&spi-bus-sleep &spi-cs-sleep>;
>> +
>> + qcom,infinite-mode = <0>;
>> + qcom,use-bam;
>> + qcom,ver-reg-exists;
>> + qcom,bam-consumer-pipe-index = <12>;
>> + qcom,bam-producer-pipe-index = <13>;
>
> Please fix the indentation here.
>

Will do
>> + };
>> +
>> diff --git a/drivers/pinctrl/Kconfig b/drivers/pinctrl/Kconfig
>> index 34f51d2..480cb26 100644
>> --- a/drivers/pinctrl/Kconfig
>> +++ b/drivers/pinctrl/Kconfig
>> @@ -133,6 +133,16 @@ config PINCTRL_IMX28
>> bool
>> select PINCTRL_MXS
>>
>> +config PINCTRL_MSM
>> + depends on OF
>> + bool
>> + select PINMUX
>> + select GENERIC_PINCONF
>> +
>> +config PINCTRL_MSM_TLMM_V3
>> + bool
>> + select PINCTRL_MSM
>
> PINCTRL_MSM does selects only to compile the "common" parts, but this
> does not build. Adding e.g MSM_TLMM_V2 here as well (as I did first)
> just made me have to select 2 things to make one thing compile.
>
> My suggestion is that you remove the V3 config and just put it all
> under PINCTRL_MSM
>

I need to distinguish the device tree interface and parsing
from the TLMM version.
I would probably go in the reverse direction and keep only the V3 config
and remove the generic PINCTRL_MSM?
>> +
>> +
>> +/* SDC Pin type register offsets */
>> +#define TLMMV3_SDC_OFFSET 0x2044
>> +#define TLMMV3_SDC1_CFG(base) (base)
>> +#define TLMMV3_SDC2_CFG(base) (TLMMV3_SDC1_CFG(base) + 0x4)
>> +
>> +/* GP pin type register offsets */
>> +#define TLMMV3_GP_CFG(base, pin) (base + 0x1000 + 0x10 * (pin))
>> +#define TLMMV3_GP_INOUT(base, pin) (base + 0x1004 + 0x10 * (pin))
>> +
>> +struct msm_sdc_regs {
>> + unsigned int offset;
>> + unsigned long pull_mask;
>> + unsigned long pull_shft;
>> + unsigned long drv_mask;
>> + unsigned long drv_shft;
>> +};
>> +
>> +static struct msm_sdc_regs sdc_regs[] = {
>> + /* SDC1 CLK */
>> + {
>> + .offset = 0,
>> + .pull_mask = TLMMV3_SDC1_CLK_PULL_MASK,
>> + .pull_shft = TLMMV3_SDC1_CLK_PULL_SHFT,
>> + .drv_mask = TLMMV3_SDC1_CLK_DRV_MASK,
>> + .drv_shft = TLMMV3_SDC1_CLK_DRV_SHFT,
>> + },
>> + /* SDC1 CMD */
>> + {
>> + .offset = 0,
>> + .pull_mask = TLMMV3_SDC1_CMD_PULL_MASK,
>> + .pull_shft = TLMMV3_SDC1_CMD_PULL_SHFT,
>> + .drv_mask = TLMMV3_SDC1_CMD_DRV_MASK,
>> + .drv_shft = TLMMV3_SDC1_CMD_DRV_SHFT,
>> + },
>> + /* SDC1 DATA */
>> + {
>> + .offset = 0,
>> + .pull_mask = TLMMV3_SDC1_DATA_PULL_MASK,
>> + .pull_shft = TLMMV3_SDC1_DATA_PULL_SHFT,
>> + .drv_mask = TLMMV3_SDC1_DATA_DRV_MASK,
>> + .drv_shft = TLMMV3_SDC1_DATA_DRV_SHFT,
>> + },
>> + /* SDC2 CLK */
>> + {
>> + .offset = 0x4,
>> + .pull_mask = TLMMV3_SDC2_CLK_PULL_MASK,
>> + .pull_shft = TLMMV3_SDC2_CLK_PULL_SHFT,
>> + .drv_mask = TLMMV3_SDC2_CLK_DRV_MASK,
>> + .drv_shft = TLMMV3_SDC2_CLK_DRV_SHFT,
>> + },
>> + /* SDC2 CMD */
>> + {
>> + .offset = 0x4,
>> + .pull_mask = TLMMV3_SDC2_CMD_PULL_MASK,
>> + .pull_shft = TLMMV3_SDC2_CMD_PULL_SHFT,
>> + .drv_mask = TLMMV3_SDC2_CMD_DRV_MASK,
>> + .drv_shft = TLMMV3_SDC2_CMD_DRV_SHFT,
>> + },
>> + /* SDC2 DATA */
>> + {
>> + .offset = 0x4,
>> + .pull_mask = TLMMV3_SDC2_DATA_PULL_MASK,
>> + .pull_shft = TLMMV3_SDC2_DATA_PULL_SHFT,
>> + .drv_mask = TLMMV3_SDC2_DATA_DRV_MASK,
>> + .drv_shft = TLMMV3_SDC2_DATA_DRV_SHFT,
>> + },
>> +};
>> +
>> +static int msm_tlmm_v3_sdc_cfg(uint pin_no, unsigned long *config,
>> + void __iomem *reg_base,
>> + bool write)
>
> I strongly recommend you split this function into a get_config and
> set_config. It took me plenty of time to get my head around what
> you're doing here.
>
> I would also suggest that you do:
>
> val = readl()
> switch () {
> fix val in various ways
> }
> writel(val)
>
> Instead of maintaining mask and shift throughout the function.
>

I thought the bool write would suffice to indicate a bidirectional
function. I can perhaps use a better term for the boolean if readability
is a concern.
There are more then 2 pin types on the TLMM, which we will be
adding support for. Adding a set and get for each will lead to a lot
of functions that look similar.
Right now the samsung implementation too uses a common function for set
and get.

>> +{
>> + unsigned int val, id, data;
>> + u32 mask, shft;
>> + void __iomem *cfg_reg;
>> +
>> + cfg_reg = reg_base + sdc_regs[pin_no].offset;
>> + id = pinconf_to_config_param(*config);
>> + val = readl_relaxed(cfg_reg);
>> + /* Get mask and shft values for this config type */
>> + switch (id) {
>> + case PIN_CONFIG_BIAS_DISABLE:
>> + mask = sdc_regs[pin_no].pull_mask;
>> + shft = sdc_regs[pin_no].pull_shft;
>> + data = TLMMV3_NO_PULL;
>> + if (!write) {
>> + val >>= shft;
>> + val &= mask;
>> + data = rval_to_pull(val);
>> + }
>> + break;
>> + case PIN_CONFIG_BIAS_PULL_DOWN:
>> + mask = sdc_regs[pin_no].pull_mask;
>> + shft = sdc_regs[pin_no].pull_shft;
>> + data = TLMMV3_PULL_DOWN;
>> + if (!write) {
>> + val >>= shft;
>> + val &= mask;
>> + data = rval_to_pull(val);
>> + }
>> + break;
>> + case PIN_CONFIG_BIAS_PULL_UP:
>> + mask = sdc_regs[pin_no].pull_mask;
>> + shft = sdc_regs[pin_no].pull_shft;
>> + data = TLMMV3_PULL_UP;
>> + if (!write) {
>> + val >>= shft;
>> + val &= mask;
>> + data = rval_to_pull(val);
>> + }
>> + break;
>> + case PIN_CONFIG_DRIVE_STRENGTH:
>> + mask = sdc_regs[pin_no].drv_mask;
>> + shft = sdc_regs[pin_no].drv_shft;
>> + if (write) {
>> + data = pinconf_to_config_argument(*config);
>> + data = drv_str_to_rval(data);
>> + } else {
>> + val >>= shft;
>> + val &= mask;
>> + data = rval_to_drv_str(val);
>> + }
>> + break;
>> + default:
>> + return -EINVAL;
>> + };
>> +
>> + if (write) {
>> + val &= ~(mask << shft);
>> + val |= (data << shft);
>> + writel_relaxed(val, cfg_reg);
>> + } else
>> + *config = pinconf_to_config_packed(id, data);
>> + return 0;
>> +}
>> +
>> +static void msm_tlmm_v3_sdc_set_reg_base(void __iomem **ptype_base,
>> + void __iomem *tlmm_base)
>> +{
>> + *ptype_base = tlmm_base + TLMMV3_SDC_OFFSET;
>> +}
>
> This function is used to do "pin type based base offset shifting", if
> you instead moved the calculation into the msm_sdc_regs above you
> could just skip it from your interface and pass the blocks reg_base to
> allt these functions.
>
The TLMM base is going to be maintained by the upper layer.
I need the pintype base to be stored with the pintype struct,
because future implementation of irq chip/gpio chip supported
by a given pin type, will not pass through the upper layer, but
rather arrive in this file directly via gpiolib.

>> +{
>> + unsigned int val, id, data, inout_val;
>> + u32 mask = 0, shft = 0;
>> + void __iomem *inout_reg = NULL;
>> + void __iomem *cfg_reg = TLMMV3_GP_CFG(reg_base, pin_no);
>> +
>> + id = pinconf_to_config_param(*config);
>> + val = readl_relaxed(cfg_reg);
>> + /* Get mask and shft values for this config type */
>> + switch (id) {
>> + case PIN_CONFIG_BIAS_DISABLE:
>> + mask = TLMMV3_GP_PULL_MASK;
>> + shft = TLMMV3_GP_PULL_SHFT;
>> + data = TLMMV3_NO_PULL;
>> + if (!write) {
>> + val >>= shft;
>> + val &= mask;
>> + data = rval_to_pull(val);
>
> I assume this should return a "boolean" whether or not this mode is
> selected; if so you should return 1 here iff (val & 0x3) == 3. I
> assume the same goes for the sdc config function?
>
> @Linus, could you please confirm this interpretation of the get_config
> for pull config.
>

Right now I use the default values associated with these config params
to program corresponding register values.
If its a BIAS_PULL_DOWN it still needs to return 1.
Only if its BIAS_DISABLE it should return 0.
The function will do that.

>> +static void msm_tlmm_v3_gp_set_reg_base(void __iomem **ptype_base,
>> + void __iomem *tlmm_base)
>> +{
>> + *ptype_base = tlmm_base;
>> +}
>> +
>> +static struct msm_pintype_info tlmm_v3_pininfo[] = {
>> + {
>> + .prg_cfg = msm_tlmm_v3_gp_cfg,
>> + .prg_func = msm_tlmm_v3_gp_fn,
>
> Please add a separate function for getting the config, as it would
> simplify both sides of the interface.
> Please rename these functions ".get_config", ".set_config" and ".set_function".
>
>> + .set_reg_base = msm_tlmm_v3_gp_set_reg_base,
>
explanation earlier.
> Please remove this from the interface, better let the sdc definition
> include the offset needed.
>
>> + .reg_data = NULL,
>
> By removing the set_reg_base, you don't need a per pin-type definition
> of the register base.
>
same as above.
>> + .prop_name = "qcom,pin-type-gp",
>> + .name = "gp",
>> + },
>> + {
>> + .prg_cfg = msm_tlmm_v3_sdc_cfg,
>> + .set_reg_base = msm_tlmm_v3_sdc_set_reg_base,
>> + .reg_data = NULL,
>> + .prop_name = "qcom,pin-type-sdc",
>> + .name = "sdc",
>> + }
>> +};
>> +
>> +struct msm_tlmm tlmm_v3_pintypes = {
>> + .num_entries = ARRAY_SIZE(tlmm_v3_pininfo),
>> + .pintype_info = tlmm_v3_pininfo,
>> +};
>> +
>> + /* Get function mapping */
>> + of_property_read_u32(parent, "qcom,pin-func", &val);
>> + fn_name = kzalloc(strlen(grp_name) + strlen("-func"),
>> + GFP_KERNEL);
>> + if (!fn_name) {
>> + ret = -ENOMEM;
>> + goto func_err;
>> + }
>> + snprintf(fn_name, strlen(grp_name) + strlen("-func") + 1, "%s%s",
>> + grp_name, "-func");
>
> Store the size of this in a variable to be used both in the allocation
> and in the snprintf. That way you don't have to line wrap here.
> Also make sure you allocate room for the nul byte (that you have in
> your snprintf).
>
> And move the "-func" into the format string ("%s-func").
>

Will do.

>> + }
>> + snprintf(func_name, len, "%s%s", grp_name, "-func");
>
> Move "-func" into the format.
>
will do

>> + curr_func->name = func_name;
>> + curr_func->gps = devm_kzalloc(dev, sizeof(char *), GFP_KERNEL);
>> + if (!curr_func->gps) {
>> + dev_err(dev, "failed to alloc memory for group list ");
>> + return -ENOMEM;
>> + }
>> + of_property_read_u32(pgrp_np, "qcom,pin-func", &func);
>> + curr_grp->func = func;
>> + curr_func->gps[0] = grp_name;
>> + curr_func->num_grps = 1;
>
> This is the only place you store something in gps and the only place
> you read the value is in msm_pmux_get_groups(). As you're in control
> of the msm_pmx_funcs struct I suggest that you remove this allocation
> and just store the group_name in a char * pointer and drops the
> num_grps. Then in msm_pmux_get_groups you assign
>
> *groups = &dd->pmx_funcs[selector].gps;
> *num_groups = 1;
>
> And preferably you rename gps to group_name, or something else that's readable.
>

Will do.
>> + func_index++;
>> + }
>> + dd->pin_grps = pin_grps;
>> + dd->num_grps = num_grps;
>> + dd->pmx_funcs = pmx_funcs;
>> + dd->num_funcs = num_funcs;
>> + return 0;
>> +}
>> +
>> +static void msm_populate_pindesc(struct msm_pintype_info *pinfo,
>> + struct msm_pindesc *msm_pindesc)
>> +{
>> + int i;
>> + struct msm_pindesc *pindesc;
>> +
>> + for (i = 0; i < pinfo->num_pins; i++) {
>> + pindesc = &msm_pindesc[i + pinfo->pin_start];
>> + pindesc->pin_info = pinfo;
>> + snprintf(pindesc->name, sizeof(pindesc->name),
>> + "%s-%d", pinfo->name, i);
>> + }
>> +}
>> +
>> +static int msm_pinctrl_dt_parse_pintype(struct device_node *dev_node,
>> + struct msm_pinctrl_dd *dd)
>> +{
>> + struct device_node *pt_node;
>> + struct msm_pindesc *msm_pindesc;
>> + struct msm_pintype_info *pintype, *pinfo;
>> + void __iomem **ptype_base;
>> + u32 num_pins, pinfo_entries, curr_pins;
>> + int i, ret;
>> + uint total_pins = 0;
>> +
>> + pinfo = dd->msm_pintype;
>> + pinfo_entries = dd->num_pintypes;
>> + curr_pins = 0;
>> +
>> + for_each_child_of_node(dev_node, pt_node) {
>> + for (i = 0; i < pinfo_entries; i++) {
>> + pintype = &pinfo[i];
>> + /* Check if node is pintype node */
>> + if (!of_find_property(pt_node, pinfo->prop_name, NULL))
>> + continue;
>> + of_node_get(pt_node);
>> + pintype->node = pt_node;
>> + /* determine number of pins of given pin type */
>> + ret = of_property_read_u32(pt_node, "qcom,num-pins",
>> + &num_pins);
>> + if (ret) {
>
> The alloc_fail at the bottom is taking care of putting any of_nodes
> you got here, in the event of an allocation failure below. But here
> you simply return. Please roll back your of_node_gets.
>

Will do.
>> + dev_err(dd->dev, "num pins not specified\n");
>> + return ret;
>> + }
>> + /* determine pin number range for given pin type */
>> + pintype->num_pins = num_pins;
>> + pintype->pin_start = curr_pins;
>> + pintype->pin_end = curr_pins + num_pins;
>
> How does this work with having multiple pin types defined at the same time.
>
> As far as I can see the order of the definition of the pin types
> matters because of this.
> It also gives an idea that you could have multiple gp/sdc groups, but
> then the addressing will be way confusing.
>
> Compare what actual pin <&gp 0> refers to in these two cases:
>
> gp: gp { qcom,num-pins = <10>; }
> sdc: sdc { qcom,num-pins = <10>; }
>
> sdc: sdc { qcom,num-pins = <10>; }
> gp: gp { qcom,num-pins = <10>; }
>
>
> I suggest that you drop the pin_start and only store num_pins in the
> pintype struct.
>

The framework is going to give me pin numbers which need to map to a
particular pin number for a pin type (not a group). The pin_start helps
me achieve this translation. In the above case the framework will know
of a total of 20 pins. By knowing where the gp pin type starts in the
above 20, i can map a pin number to it or any other pin type.

> And again, thanks for writing this driver. I look forward to have it
> mainlined! (with support for my 8960 board ;))
>

As I said earlier, I will be adding support for v3. You can start work
in parallel on v2, if you would like, and once this is mainlined, we can
review and push in your v2 effort as well?


Thanks
Hanumant

--
Sent by an employee of the Qualcomm Innovation Center, Inc.
The Qualcomm Innovation Center, Inc. is a member of the Code Aurora Forum.
--

2013-08-14 19:16:21

by Linus Walleij

[permalink] [raw]
Subject: Re: [PATCH] pinctrl: msm: Add support for MSM TLMM pinmux

On Wed, Aug 7, 2013 at 6:00 PM, Stephen Warren <[email protected]> wrote:
> On 08/06/2013 05:45 PM, Hanumant Singh wrote:

>> Any comments on this?
>
> No. As I said, I personally want to see all the pingroups defined in the
> pinctrl driver. But, if someone else acks/... the patches without it, I
> probably won't nack it.

I agree with Stephen that I want to see the groups defined in a
(per-SoC) in-kernel driver, not as DT data.

Yours,
Linus Walleij

2013-08-14 19:29:11

by Linus Walleij

[permalink] [raw]
Subject: Re: [PATCH] pinctrl: msm: Add support for MSM TLMM pinmux

On Tue, Jul 30, 2013 at 1:39 AM, Bjorn Andersson <[email protected]> wrote:
> On Wed, Jul 24, 2013 at 1:41 PM, Hanumant Singh <[email protected]> wrote:

> As a general note on the patch, the pins and pin groups are defined by
> the soc, I'm therefore not convinced that these should be configured
> from the devicetree. It's at least not how it's done in other
> platforms.

Now we are three people sayin the same thing so it should
be a good hint to the implementer :-)

> After talking to Linus a little bit I concluded that the way the
> design idea behind pinmux is that you have pins and you have
> functions;
> * pins are your 117 gp-pins and with some imagination your sd pins, so
> that's fine.
> * the function is some functionality provided by the hard
> Then you can set up muxes between these two. Therefore I think your
> use of the word "function" in the patch is what normally is called a
> mux. It might be worth sorting this out, to make it easier to maintain
> this code down the road.
>
> @Linus, can you please confirm my understanding of the design?

Yes, we have pins in groups which are just group = pin [a, b, ... , n]
and we have functions which is e.g. "I2C block 1". To push a function
out on the pins we connect the function with a group of pins.

For a function such as a UART, we connect 2 or 4 pins ... those will
provide TX+RX or maybe TX+RX+CTS+RTS if we want to be fancy.

So the mapping table combines this group with this function and
that becomes a setting such as "default" ...

You only have to do it a few times and then you get the hang
of it :-)

I wish we could have come up with something simpler than
this but it appears the world is complex, this was the common
denominator of all the worlds pinmuxes: they enable a group
of pins for a certain function (an IP core).

>> +static int msm_tlmm_v3_sdc_cfg(uint pin_no, unsigned long *config,
>> + void __iomem *reg_base,
>> + bool write)
>
> I strongly recommend you split this function into a get_config and
> set_config. It took me plenty of time to get my head around what
> you're doing here.
>
> I would also suggest that you do:
>
> val = readl()
> switch () {
> fix val in various ways
> }
> writel(val)
>
> Instead of maintaining mask and shift throughout the function.

Agree. And if you want to have performance as well:
readl_relaxed() ... writel() is preferred. (Not that it matters
much uless in a critical path.)

>> + /* Get mask and shft values for this config type */
>> + switch (id) {
>> + case PIN_CONFIG_BIAS_DISABLE:
>> + mask = TLMMV3_GP_PULL_MASK;
>> + shft = TLMMV3_GP_PULL_SHFT;
>> + data = TLMMV3_NO_PULL;
>> + if (!write) {
>> + val >>= shft;
>> + val &= mask;
>> + data = rval_to_pull(val);
>
> I assume this should return a "boolean" whether or not this mode is
> selected; if so you should return 1 here iff (val & 0x3) == 3. I
> assume the same goes for the sdc config function?
>
> @Linus, could you please confirm this interpretation of the get_config
> for pull config.

On bias disable the argument is ignored so it doesn't matter, it
should just be set to zero.

Actually the data should be more carefully handled for each
config option I think.

Yours,
Linus Walleij

2013-08-15 17:44:08

by Hanumant Singh

[permalink] [raw]
Subject: Re: [PATCH] pinctrl: msm: Add support for MSM TLMM pinmux

On 8/14/2013 12:29 PM, Linus Walleij wrote:
> On Tue, Jul 30, 2013 at 1:39 AM, Bjorn Andersson <[email protected]> wrote:
>> On Wed, Jul 24, 2013 at 1:41 PM, Hanumant Singh <[email protected]> wrote:
>
>> As a general note on the patch, the pins and pin groups are defined by
>> the soc, I'm therefore not convinced that these should be configured
>> from the devicetree. It's at least not how it's done in other
>> platforms.
>
> Now we are three people sayin the same thing so it should
> be a good hint to the implementer :-)
>

Ok i can switch to using pin groups defined in per soc files.
But in our case we have one soc going into different types of boards.
(atleast 3). In each of the boards the same external devices end up
using different pins. For ex camera on board 1 uses different pin group
then the same camera on board 2. Both having the same SOC.
So in this case the design would be to have all possible pin groups
for different boards enumerated in the same soc-pinctrl.c file?

Also in this implementation I will have.
1) pinctrl-msm.c => DT parsing and interface to framework.
2) pinctrl-msm-tlmm<version>.c => Register programming and pin types
supported by a particular TLMM pinmux version.
3) pinctrl-<soc>.c => All the pins/pin groups supported by a given SOC.
As I mentioned we will have a bloat of these, since we have entire
families of SOC using a given TLMM version but with unique pin groupings.




>>> + /* Get mask and shft values for this config type */
>>> + switch (id) {
>>> + case PIN_CONFIG_BIAS_DISABLE:
>>> + mask = TLMMV3_GP_PULL_MASK;
>>> + shft = TLMMV3_GP_PULL_SHFT;
>>> + data = TLMMV3_NO_PULL;
>>> + if (!write) {
>>> + val >>= shft;
>>> + val &= mask;
>>> + data = rval_to_pull(val);
>>
>> I assume this should return a "boolean" whether or not this mode is
>> selected; if so you should return 1 here iff (val & 0x3) == 3. I
>> assume the same goes for the sdc config function?
>>
>> @Linus, could you please confirm this interpretation of the get_config
>> for pull config.
>
> On bias disable the argument is ignored so it doesn't matter, it
> should just be set to zero.
>

I don't override the default values, since resistor values are not
configurable. I only care about which config option is chosen to program
it as pull up/down or disable.

> Actually the data should be more carefully handled for each
> config option I think.
>
Not sure I follow. Based on my use mentioned above, what do you suggest
for the read? Should I return default config value, which is what I am
doing ?

> Yours,
> Linus Walleij
>

Thanks
Hanumant
--
Sent by an employee of the Qualcomm Innovation Center, Inc.
The Qualcomm Innovation Center, Inc. is a member of the Code Aurora Forum.
--

2013-08-15 20:47:02

by Linus Walleij

[permalink] [raw]
Subject: Re: [PATCH] pinctrl: msm: Add support for MSM TLMM pinmux

On Thu, Aug 15, 2013 at 7:44 PM, Hanumant Singh <[email protected]> wrote:

> Ok i can switch to using pin groups defined in per soc files.
> But in our case we have one soc going into different types of boards.
> (atleast 3). In each of the boards the same external devices end up using
> different pins. For ex camera on board 1 uses different pin group
> then the same camera on board 2. Both having the same SOC.
> So in this case the design would be to have all possible pin groups
> for different boards enumerated in the same soc-pinctrl.c file?

Sorry I don't get this at all.

What pin groups and functions that exist on a SoC is what you put into
a SoC driver. Because this is a hardware characteristic.

How these are combined on a board into different states is what you put
into the device tree. (Or platform data.)

> Also in this implementation I will have.
> 1) pinctrl-msm.c => DT parsing and interface to framework.
> 2) pinctrl-msm-tlmm<version>.c => Register programming and pin types
> supported by a particular TLMM pinmux version.
> 3) pinctrl-<soc>.c => All the pins/pin groups supported by a given SOC.

Seems OK.

> As I
> mentioned we will have a bloat of these, since we have entire families of
> SOC using a given TLMM version but with unique pin groupings.

Bring 'em on. But is that really different groups you are talking about,
and not just combinations of groups with functions for a certain board
as I describe above?

If you have many SoC subdrivers, consider creating a subdir as some
drivers already have.

> I don't override the default values, since resistor values are not
> configurable. I only care about which config option is chosen to program it
> as pull up/down or disable.

That sounds correct.

>> Actually the data should be more carefully handled for each
>> config option I think.
>>
> Not sure I follow. Based on my use mentioned above, what do you suggest for
> the read? Should I return default config value, which is what I am doing ?

Here:

+ switch (id) {
+ case PIN_CONFIG_BIAS_DISABLE:
+ mask = TLMMV3_GP_PULL_MASK;
+ shft = TLMMV3_GP_PULL_SHFT;
+ data = TLMMV3_NO_PULL;

data should just be zero. (Maybe TLMMV3_NO_PULL is
zero? But anyway...)

+ if (!write) {
+ val >>= shft;
+ val &= mask;
+ data = rval_to_pull(val);

Dito.

Because it has no meaning in the framework.

Yours,
Linus Walleij

2013-08-15 21:28:42

by Hanumant Singh

[permalink] [raw]
Subject: Re: [PATCH] pinctrl: msm: Add support for MSM TLMM pinmux

On 8/15/2013 1:47 PM, Linus Walleij wrote:
> On Thu, Aug 15, 2013 at 7:44 PM, Hanumant Singh <[email protected]> wrote:
>
>> Ok i can switch to using pin groups defined in per soc files.
>> But in our case we have one soc going into different types of boards.
>> (atleast 3). In each of the boards the same external devices end up using
>> different pins. For ex camera on board 1 uses different pin group
>> then the same camera on board 2. Both having the same SOC.
>> So in this case the design would be to have all possible pin groups
>> for different boards enumerated in the same soc-pinctrl.c file?
>
> Sorry I don't get this at all.
>
> What pin groups and functions that exist on a SoC is what you put into
> a SoC driver. Because this is a hardware characteristic.
>
> How these are combined on a board into different states is what you put
> into the device tree. (Or platform data.)
>

For example lets say for a given SOC A it goes into boards 1, 2, and 3.
Each of the boards has a display panel. The display panel uses two pins
1) a reset pin 2) an interrupt pin.

In the combination of SOC A + board 1
- Display panel reset = pin no 5.
- Display panel interrupt = pin no 9.

In combination of SOC A + board 2
- Display panel reset = pin no 4.
- Display panel interrupt = pin no 9.

In combination of SOC A + board 3
- Display panel reset = pin no 7.
- Display panel interrupt = pin no 2.

The pin groupings to be used by the display panel can be {5,9}, {4,9},
or {7,2}
These different pin groups and their function setting will be present in
soc-pinctrl.c. The function setting is the same on all 3 cases.

The DT entry will correspond to the different states of these pins for
the different boards.

Is this understanding correct?


>> Also in this implementation I will have.
>> 1) pinctrl-msm.c => DT parsing and interface to framework.
>> 2) pinctrl-msm-tlmm<version>.c => Register programming and pin types
>> supported by a particular TLMM pinmux version.
>> 3) pinctrl-<soc>.c => All the pins/pin groups supported by a given SOC.
>
> Seems OK.
>
>> As I
>> mentioned we will have a bloat of these, since we have entire families of
>> SOC using a given TLMM version but with unique pin groupings.
>
> Bring 'em on. But is that really different groups you are talking about,
> and not just combinations of groups with functions for a certain board
> as I describe above?
>
> If you have many SoC subdrivers, consider creating a subdir as some
> drivers already have.

Actually the SOC files, as I see it, will only contain the different pin
groupings and the function setting for a given soc. The real driver
implementation will be in 1) and 2) (the device being the TLMM pinmux
version 3). I will currently refrain from creating a special msm
directory. Maybe that can be a step 2) once we start adding more SOC's?
I will be starting the patch with msm8974 SOC only.

Thanks
Hanumant


--
Sent by an employee of the Qualcomm Innovation Center, Inc.
The Qualcomm Innovation Center, Inc. is a member of the Code Aurora Forum.
--

2013-08-15 21:50:15

by Josh Cartwright

[permalink] [raw]
Subject: Re: [PATCH] pinctrl: msm: Add support for MSM TLMM pinmux

On Thu, Aug 15, 2013 at 10:44:03AM -0700, Hanumant Singh wrote:
> On 8/14/2013 12:29 PM, Linus Walleij wrote:
> >On Tue, Jul 30, 2013 at 1:39 AM, Bjorn Andersson <[email protected]> wrote:
> >>On Wed, Jul 24, 2013 at 1:41 PM, Hanumant Singh <[email protected]> wrote:
> >
> >>As a general note on the patch, the pins and pin groups are defined by
> >>the soc, I'm therefore not convinced that these should be configured
> >>from the devicetree. It's at least not how it's done in other
> >>platforms.
> >
> >Now we are three people sayin the same thing so it should
> >be a good hint to the implementer :-)
> >
>
> Ok i can switch to using pin groups defined in per soc files.
> But in our case we have one soc going into different types of boards.
> (atleast 3). In each of the boards the same external devices end up using
> different pins. For ex camera on board 1 uses different pin group
> then the same camera on board 2. Both having the same SOC.
> So in this case the design would be to have all possible pin groups
> for different boards enumerated in the same soc-pinctrl.c file?
>
> Also in this implementation I will have.
> 1) pinctrl-msm.c => DT parsing and interface to framework.
> 2) pinctrl-msm-tlmm<version>.c => Register programming and pin types
> supported by a particular TLMM pinmux version.

It isn't clear to me what you are trying to separate out between 1) and
2). Seems like there should only be pinctrl-msm-tlmm<version>.c.

--
Qualcomm Innovation Center, Inc. is a member of Code Aurora Forum,
hosted by The Linux Foundation

2013-08-15 21:58:34

by Hanumant Singh

[permalink] [raw]
Subject: Re: [PATCH] pinctrl: msm: Add support for MSM TLMM pinmux

On 8/15/2013 2:50 PM, Josh Cartwright wrote:
> On Thu, Aug 15, 2013 at 10:44:03AM -0700, Hanumant Singh wrote:
>> On 8/14/2013 12:29 PM, Linus Walleij wrote:
>>> On Tue, Jul 30, 2013 at 1:39 AM, Bjorn Andersson <[email protected]> wrote:
>>>> On Wed, Jul 24, 2013 at 1:41 PM, Hanumant Singh <[email protected]> wrote:
>>>
>>>> As a general note on the patch, the pins and pin groups are defined by
>>>> the soc, I'm therefore not convinced that these should be configured
>>> >from the devicetree. It's at least not how it's done in other
>>>> platforms.
>>>
>>> Now we are three people sayin the same thing so it should
>>> be a good hint to the implementer :-)
>>>
>>
>> Ok i can switch to using pin groups defined in per soc files.
>> But in our case we have one soc going into different types of boards.
>> (atleast 3). In each of the boards the same external devices end up using
>> different pins. For ex camera on board 1 uses different pin group
>> then the same camera on board 2. Both having the same SOC.
>> So in this case the design would be to have all possible pin groups
>> for different boards enumerated in the same soc-pinctrl.c file?
>>
>> Also in this implementation I will have.
>> 1) pinctrl-msm.c => DT parsing and interface to framework.
>> 2) pinctrl-msm-tlmm<version>.c => Register programming and pin types
>> supported by a particular TLMM pinmux version.
>
> It isn't clear to me what you are trying to separate out between 1) and
> 2). Seems like there should only be pinctrl-msm-tlmm<version>.c.
>
The idea is that we should be able to plug in the next TLMM version 4,
with the pintypes that it supports and its register programming
semantics without any changes to the DT parsing or interface to the
framework. The DT information describes the states of the pingroups.
How we handle them with respect to the pinctrl framework, does not
change with different versions of the TLMM.

Thanks
Hanumant

--
Sent by an employee of the Qualcomm Innovation Center, Inc.
The Qualcomm Innovation Center, Inc. is a member of the Code Aurora Forum.
--

2013-08-15 23:10:25

by Josh Cartwright

[permalink] [raw]
Subject: Re: [PATCH] pinctrl: msm: Add support for MSM TLMM pinmux

On Thu, Aug 15, 2013 at 02:58:31PM -0700, Hanumant Singh wrote:
> On 8/15/2013 2:50 PM, Josh Cartwright wrote:
> >On Thu, Aug 15, 2013 at 10:44:03AM -0700, Hanumant Singh wrote:
> >>On 8/14/2013 12:29 PM, Linus Walleij wrote:
> >>>On Tue, Jul 30, 2013 at 1:39 AM, Bjorn Andersson <[email protected]> wrote:
> >>>>On Wed, Jul 24, 2013 at 1:41 PM, Hanumant Singh <[email protected]> wrote:
[..]
> >>
> >>Ok i can switch to using pin groups defined in per soc files.
> >>But in our case we have one soc going into different types of boards.
> >>(atleast 3). In each of the boards the same external devices end up using
> >>different pins. For ex camera on board 1 uses different pin group
> >>then the same camera on board 2. Both having the same SOC.
> >>So in this case the design would be to have all possible pin groups
> >>for different boards enumerated in the same soc-pinctrl.c file?
> >>
> >>Also in this implementation I will have.
> >>1) pinctrl-msm.c => DT parsing and interface to framework.
> >>2) pinctrl-msm-tlmm<version>.c => Register programming and pin types
> >>supported by a particular TLMM pinmux version.
> >
> >It isn't clear to me what you are trying to separate out between 1) and
> >2). Seems like there should only be pinctrl-msm-tlmm<version>.c.
> >
> The idea is that we should be able to plug in the next TLMM version 4, with
> the pintypes that it supports and its register programming semantics without
> any changes to the DT parsing or interface to the framework. The DT
> information describes the states of the pingroups.
> How we handle them with respect to the pinctrl framework, does not change
> with different versions of the TLMM.

Okay, although I think that might be a bit too much indirection too
soon.

I'd rather the strategy be to have a straightforward driver supporting
TLMMv3 only (with MSM8974 support), and only worry about introducing a
generic TLMM parsing layer when we actually have a need for it.

If nothing else, I'd at least suggest that you rename pinctrl-msm.c to
pinctrl-msm-tlmm.c, if it's going to be housing the TLMM-generic DT
parsing code.

--
Qualcomm Innovation Center, Inc. is a member of Code Aurora Forum,
hosted by The Linux Foundation

2013-08-15 23:14:08

by Hanumant Singh

[permalink] [raw]
Subject: Re: [PATCH] pinctrl: msm: Add support for MSM TLMM pinmux

On 8/15/2013 4:10 PM, Josh Cartwright wrote:
> On Thu, Aug 15, 2013 at 02:58:31PM -0700, Hanumant Singh wrote:
>> On 8/15/2013 2:50 PM, Josh Cartwright wrote:
>>> On Thu, Aug 15, 2013 at 10:44:03AM -0700, Hanumant Singh wrote:
>>>> On 8/14/2013 12:29 PM, Linus Walleij wrote:
>>>>> On Tue, Jul 30, 2013 at 1:39 AM, Bjorn Andersson <[email protected]> wrote:
>>>>>> On Wed, Jul 24, 2013 at 1:41 PM, Hanumant Singh <[email protected]> wrote:
> [..]
>>>>
>>>> Ok i can switch to using pin groups defined in per soc files.
>>>> But in our case we have one soc going into different types of boards.
>>>> (atleast 3). In each of the boards the same external devices end up using
>>>> different pins. For ex camera on board 1 uses different pin group
>>>> then the same camera on board 2. Both having the same SOC.
>>>> So in this case the design would be to have all possible pin groups
>>>> for different boards enumerated in the same soc-pinctrl.c file?
>>>>
>>>> Also in this implementation I will have.
>>>> 1) pinctrl-msm.c => DT parsing and interface to framework.
>>>> 2) pinctrl-msm-tlmm<version>.c => Register programming and pin types
>>>> supported by a particular TLMM pinmux version.
>>>
>>> It isn't clear to me what you are trying to separate out between 1) and
>>> 2). Seems like there should only be pinctrl-msm-tlmm<version>.c.
>>>
>> The idea is that we should be able to plug in the next TLMM version 4, with
>> the pintypes that it supports and its register programming semantics without
>> any changes to the DT parsing or interface to the framework. The DT
>> information describes the states of the pingroups.
>> How we handle them with respect to the pinctrl framework, does not change
>> with different versions of the TLMM.
>
> Okay, although I think that might be a bit too much indirection too
> soon.
>
> I'd rather the strategy be to have a straightforward driver supporting
> TLMMv3 only (with MSM8974 support), and only worry about introducing a
> generic TLMM parsing layer when we actually have a need for it.
>
> If nothing else, I'd at least suggest that you rename pinctrl-msm.c to
> pinctrl-msm-tlmm.c, if it's going to be housing the TLMM-generic DT
> parsing code.
>

Actually we already have a request from Bjorn to support TLMMv2 which
has different pin types and register semantics.

Thanks
Hanumant

--
Sent by an employee of the Qualcomm Innovation Center, Inc.
The Qualcomm Innovation Center, Inc. is a member of the Code Aurora Forum.
--

2013-08-28 08:32:10

by Linus Walleij

[permalink] [raw]
Subject: Re: [PATCH] pinctrl: msm: Add support for MSM TLMM pinmux

On Thu, Aug 15, 2013 at 11:28 PM, Hanumant Singh
<[email protected]> wrote:
> On 8/15/2013 1:47 PM, Linus Walleij wrote:

>> What pin groups and functions that exist on a SoC is what you put into
>> a SoC driver. Because this is a hardware characteristic.
>>
>> How these are combined on a board into different states is what you put
>> into the device tree. (Or platform data.)
>
> For example lets say for a given SOC A it goes into boards 1, 2, and 3.
> Each of the boards has a display panel. The display panel uses two pins
> 1) a reset pin 2) an interrupt pin.

That sounds very close to two GPIO pins actually but I guess they
are not configured as "GPIO" in the data sheet sense when you do
this?

> In the combination of SOC A + board 1
> - Display panel reset = pin no 5.
> - Display panel interrupt = pin no 9.
>
> In combination of SOC A + board 2
> - Display panel reset = pin no 4.
> - Display panel interrupt = pin no 9.
>
> In combination of SOC A + board 3
> - Display panel reset = pin no 7.
> - Display panel interrupt = pin no 2.
>
> The pin groupings to be used by the display panel can be {5,9}, {4,9}, or
> {7,2}
> These different pin groups and their function setting will be present in
> soc-pinctrl.c. The function setting is the same on all 3 cases.

The last part is correct, the function setting is the same on all
three cases :-)

Define one group per pin in this case, example something
like this (using the examples in Documentation/pinctrl.txt
as a base):

static const unsigned pan_res_pos_a[] = { 4 };
static const unsigned pan_res_pos_b[] = { 5 };
static const unsigned pan_res_pos_c[] = { 7 };
static const unsigned pan_irq_pos_a[] = { 2 };
static const unsigned pan_irq_pos_b[] = { 9 };

Next define the function and applicable groups:

static const struct foo_group foo_groups[] = {
{
.name = "pan_res_pos_a_grp",
.pins = pan_res_pos_a_pins,
.num_pins = ARRAY_SIZE(pan_res_pos_a_pins),
},
{
.name = "pan_res_pos_b_grp",
.pins = pan_res_pos_b_pins,
.num_pins = ARRAY_SIZE(pan_res_pos_b_pins),
},
{
.name = "pan_res_pos_c_grp",
.pins = pan_res_pos_c_pins,
.num_pins = ARRAY_SIZE(pan_res_pos_c_pins),
},
{
.name = "pan_irq_pos_a_grp",
.pins = pan_irq_pos_a_pins,
.num_pins = ARRAY_SIZE(pan_irq_pos_a_pins),
},
{
.name = "pan_irq_pos_b_grp",
.pins = pan_irq_pos_b_pins,
.num_pins = ARRAY_SIZE(pan_irq_pos_b_pins),
},
};

static const char * const panel_groups[] = { "pan_res_pos_a_grp",
"pan_res_pos_b_grp", "pan_res_pos_c_grp",
"pan_irq_pos_a_grp", "pan_irq_pos_b_grp"
};

static const struct foo_pmx_func foo_functions[] = {
{
.name = "panel",
.groups = panel_groups,
.num_groups = ARRAY_SIZE(panel_groups),
},
....
};


Then in your DTS you combine the function "panel" with
the two groups you want, i.e. for your examples:

1: "panel" + "pan_res_pos_b_grp" + "pan_irq_pos_b_grp"
2: "panel" + "pan_res_pos_a_grp" + "pan_irq_pos_b_grp"
3: "panel" + "pan_res_pos_c_grp" + "pan_irq_pos_a_grp"

So in each case the function "panel" is combined with two
groups enabling the correct pins for a certain state, e.g.
the "default" state. Each line above corresponds to a
state.

> The DT entry will correspond to the different states of these pins for the
> different boards.
>
> Is this understanding correct?

See above.

>> Bring 'em on. But is that really different groups you are talking about,
>> and not just combinations of groups with functions for a certain board
>> as I describe above?
>>
>> If you have many SoC subdrivers, consider creating a subdir as some
>> drivers already have.
>
> Actually the SOC files, as I see it, will only contain the different pin
> groupings and the function setting for a given soc.

No settings. The SoC files will contain the available groups and
functions. How these are combined is defined by the device tree
(or even platform data). The pinctrl SoC files define what is
*possible* and the configuration from DTS or platform data
define what to do with the possibilities.

> 3). I will currently refrain from creating a special msm directory.
> Maybe that can be a step 2) once we start adding more SOC's?
> I will be starting the patch with msm8974 SOC only.

Sure any order you prefer, let's get a base support in that
looks fine and then we can start refining it.

Yours,
Linus Walleij