pinctrl: gemini: Support drive strength setting
authorLinus Walleij <linus.walleij@linaro.org>
Sat, 2 Dec 2017 11:23:09 +0000 (12:23 +0100)
committerLinus Walleij <linus.walleij@linaro.org>
Thu, 7 Dec 2017 08:59:26 +0000 (09:59 +0100)
The Gemini pin controller can set drive strength for a few
select groups of pins (not individually). Implement this
for GMAC0 and 1 (ethernet ports), IDE and PCI.

Cc: devicetree@vger.kernel.org
Reviewed-by: Rob Herring <robh@kernel.org>
Signed-off-by: Linus Walleij <linus.walleij@linaro.org>
Documentation/devicetree/bindings/pinctrl/cortina,gemini-pinctrl.txt
drivers/pinctrl/pinctrl-gemini.c

index d857b67fab72a61fcf85b8eed6fd389883d4c097..4346ff2dd8e680353557f4861671f096cbec6890 100644 (file)
@@ -17,6 +17,9 @@ and generic pin config nodes.
 
 Supported configurations:
 - skew-delay is supported on the Ethernet pins
+- drive-strength with 4, 8, 12 or 16 mA as argument is supported for
+  entire groups on the groups "idegrp", "gmii_gmac0_grp", "gmii_gmac1_grp"
+  and "pcigrp".
 
 Example:
 
index 10f8f3a35fe1caf2513f4268a21ab099cdcc1483..5e789753f94551b87605857e7cefe2a7d1d92d59 100644 (file)
@@ -67,6 +67,9 @@ struct gemini_pmx {
  *     elements in .pins so we can iterate over that array
  * @mask: bits to clear to enable this when doing pin muxing
  * @value: bits to set to enable this when doing pin muxing
+ * @driving_mask: bitmask for the IO Pad driving register for this
+ *     group, if it supports altering the driving strength of
+ *     its lines.
  */
 struct gemini_pin_group {
        const char *name;
@@ -74,12 +77,14 @@ struct gemini_pin_group {
        const unsigned int num_pins;
        u32 mask;
        u32 value;
+       u32 driving_mask;
 };
 
 /* Some straight-forward control registers */
 #define GLOBAL_WORD_ID         0x00
 #define GLOBAL_STATUS          0x04
 #define GLOBAL_STATUS_FLPIN    BIT(20)
+#define GLOBAL_IODRIVE         0x10
 #define GLOBAL_GMAC_CTRL_SKEW  0x1c
 #define GLOBAL_GMAC0_DATA_SKEW 0x20
 #define GLOBAL_GMAC1_DATA_SKEW 0x24
@@ -738,6 +743,7 @@ static const struct gemini_pin_group gemini_3512_pin_groups[] = {
                /* Conflict with all flash usage */
                .value = IDE_PADS_ENABLE | NAND_PADS_DISABLE |
                        PFLASH_PADS_DISABLE | SFLASH_PADS_DISABLE,
+               .driving_mask = GENMASK(21, 20),
        },
        {
                .name = "satagrp",
@@ -753,6 +759,7 @@ static const struct gemini_pin_group gemini_3512_pin_groups[] = {
                .name = "gmii_gmac0_grp",
                .pins = gmii_gmac0_3512_pins,
                .num_pins = ARRAY_SIZE(gmii_gmac0_3512_pins),
+               .driving_mask = GENMASK(17, 16),
        },
        {
                .name = "gmii_gmac1_grp",
@@ -760,6 +767,7 @@ static const struct gemini_pin_group gemini_3512_pin_groups[] = {
                .num_pins = ARRAY_SIZE(gmii_gmac1_3512_pins),
                /* Bring out RGMII on the GMAC1 pins */
                .value = GEMINI_GMAC_IOSEL_GMAC0_GMAC1_RGMII,
+               .driving_mask = GENMASK(19, 18),
        },
        {
                .name = "pcigrp",
@@ -767,6 +775,7 @@ static const struct gemini_pin_group gemini_3512_pin_groups[] = {
                .num_pins = ARRAY_SIZE(pci_3512_pins),
                /* Conflict only with GPIO2 */
                .value = PCI_PADS_ENABLE | PCI_CLK_PAD_ENABLE,
+               .driving_mask = GENMASK(23, 22),
        },
        {
                .name = "lpcgrp",
@@ -1671,6 +1680,7 @@ static const struct gemini_pin_group gemini_3516_pin_groups[] = {
                /* Conflict with all flash usage */
                .value = IDE_PADS_ENABLE | NAND_PADS_DISABLE |
                        PFLASH_PADS_DISABLE | SFLASH_PADS_DISABLE,
+               .driving_mask = GENMASK(21, 20),
        },
        {
                .name = "satagrp",
@@ -1686,6 +1696,7 @@ static const struct gemini_pin_group gemini_3516_pin_groups[] = {
                .name = "gmii_gmac0_grp",
                .pins = gmii_gmac0_3516_pins,
                .num_pins = ARRAY_SIZE(gmii_gmac0_3516_pins),
+               .driving_mask = GENMASK(17, 16),
        },
        {
                .name = "gmii_gmac1_grp",
@@ -1693,6 +1704,7 @@ static const struct gemini_pin_group gemini_3516_pin_groups[] = {
                .num_pins = ARRAY_SIZE(gmii_gmac1_3516_pins),
                /* Bring out RGMII on the GMAC1 pins */
                .value = GEMINI_GMAC_IOSEL_GMAC0_GMAC1_RGMII,
+               .driving_mask = GENMASK(19, 18),
        },
        {
                .name = "pcigrp",
@@ -1700,6 +1712,7 @@ static const struct gemini_pin_group gemini_3516_pin_groups[] = {
                .num_pins = ARRAY_SIZE(pci_3516_pins),
                /* Conflict only with GPIO2 */
                .value = PCI_PADS_ENABLE | PCI_CLK_PAD_ENABLE,
+               .driving_mask = GENMASK(23, 22),
        },
        {
                .name = "lpcgrp",
@@ -2394,9 +2407,77 @@ static int gemini_pinconf_set(struct pinctrl_dev *pctldev, unsigned int pin,
        return ret;
 }
 
+static int gemini_pinconf_group_set(struct pinctrl_dev *pctldev,
+                                   unsigned selector,
+                                   unsigned long *configs,
+                                   unsigned num_configs)
+{
+       struct gemini_pmx *pmx = pinctrl_dev_get_drvdata(pctldev);
+       const struct gemini_pin_group *grp = NULL;
+       enum pin_config_param param;
+       u32 arg;
+       u32 val;
+       int i;
+
+       if (pmx->is_3512)
+               grp = &gemini_3512_pin_groups[selector];
+       if (pmx->is_3516)
+               grp = &gemini_3516_pin_groups[selector];
+
+       /* First figure out if this group supports configs */
+       if (!grp->driving_mask) {
+               dev_err(pmx->dev, "pin config group \"%s\" does "
+                       "not support drive strength setting\n",
+                       grp->name);
+               return -EINVAL;
+       }
+
+       for (i = 0; i < num_configs; i++) {
+               param = pinconf_to_config_param(configs[i]);
+               arg = pinconf_to_config_argument(configs[i]);
+
+               switch (param) {
+               case PIN_CONFIG_DRIVE_STRENGTH:
+                       switch (arg) {
+                       case 4:
+                               val = 0;
+                               break;
+                       case 8:
+                               val = 1;
+                               break;
+                       case 12:
+                               val = 2;
+                               break;
+                       case 16:
+                               val = 3;
+                               break;
+                       default:
+                               dev_err(pmx->dev,
+                                       "invalid drive strength %d mA\n",
+                                       arg);
+                               return -ENOTSUPP;
+                       }
+                       val <<= (ffs(grp->driving_mask) - 1);
+                       regmap_update_bits(pmx->map, GLOBAL_IODRIVE,
+                                          grp->driving_mask,
+                                          val);
+                       dev_info(pmx->dev,
+                                "set group %s to %d mA drive strength mask %08x val %08x\n",
+                                grp->name, arg, grp->driving_mask, val);
+                       break;
+               default:
+                       dev_err(pmx->dev, "invalid config param %04x\n", param);
+                       return -ENOTSUPP;
+               }
+       }
+
+       return 0;
+}
+
 static const struct pinconf_ops gemini_pinconf_ops = {
        .pin_config_get = gemini_pinconf_get,
        .pin_config_set = gemini_pinconf_set,
+       .pin_config_group_set = gemini_pinconf_group_set,
        .is_generic = true,
 };