pinctrl: mediatek: add advanced pull related support to pinctrl-mtk-common-v2.c
authorSean Wang <sean.wang@mediatek.com>
Sat, 8 Sep 2018 11:07:25 +0000 (19:07 +0800)
committerLinus Walleij <linus.walleij@linaro.org>
Tue, 18 Sep 2018 21:52:57 +0000 (14:52 -0700)
There are some specific pins (i.e. MMC/SD) need specific registers to
turn on/off the 10K & 50k(75K) resistors when pull up/down.

Therefore, this patch adds the custom prarmeters so that the user could
control it through device tree.

Signed-off-by: Ryder.Lee <ryder.lee@mediatek.com>
Signed-off-by: Sean Wang <sean.wang@mediatek.com>
Signed-off-by: Linus Walleij <linus.walleij@linaro.org>
drivers/pinctrl/mediatek/pinctrl-moore.c
drivers/pinctrl/mediatek/pinctrl-mtk-common-v2.c
drivers/pinctrl/mediatek/pinctrl-mtk-common-v2.h

index d69f024c416d44ed4bb10762b2495ab033ca8be3..4009329597274d097e7eda7b020351e495ba3970 100644 (file)
 /* Custom pinconf parameters */
 #define MTK_PIN_CONFIG_TDSEL   (PIN_CONFIG_END + 1)
 #define MTK_PIN_CONFIG_RDSEL   (PIN_CONFIG_END + 2)
+#define MTK_PIN_CONFIG_PU_ADV  (PIN_CONFIG_END + 3)
+#define MTK_PIN_CONFIG_PD_ADV  (PIN_CONFIG_END + 4)
 
 static const struct pinconf_generic_params mtk_custom_bindings[] = {
        {"mediatek,tdsel",      MTK_PIN_CONFIG_TDSEL,           0},
        {"mediatek,rdsel",      MTK_PIN_CONFIG_RDSEL,           0},
+       {"mediatek,pull-up-adv", MTK_PIN_CONFIG_PU_ADV,         1},
+       {"mediatek,pull-down-adv", MTK_PIN_CONFIG_PD_ADV,       1},
 };
 
 #ifdef CONFIG_DEBUG_FS
 static const struct pin_config_item mtk_conf_items[] = {
        PCONFDUMP(MTK_PIN_CONFIG_TDSEL, "tdsel", NULL, true),
        PCONFDUMP(MTK_PIN_CONFIG_RDSEL, "rdsel", NULL, true),
+       PCONFDUMP(MTK_PIN_CONFIG_PU_ADV, "pu-adv", NULL, true),
+       PCONFDUMP(MTK_PIN_CONFIG_PD_ADV, "pd-adv", NULL, true),
 };
 #endif
 
@@ -168,6 +174,19 @@ static int mtk_pinconf_get(struct pinctrl_dev *pctldev,
 
                ret = val;
 
+               break;
+       case MTK_PIN_CONFIG_PU_ADV:
+       case MTK_PIN_CONFIG_PD_ADV:
+               if (hw->soc->adv_pull_get) {
+                       bool pullup;
+
+                       pullup = param == MTK_PIN_CONFIG_PU_ADV;
+                       err = hw->soc->adv_pull_get(hw, desc, pullup, &ret);
+                       if (err)
+                               return err;
+               } else {
+                       return -ENOTSUPP;
+               }
                break;
        default:
                return -ENOTSUPP;
@@ -282,6 +301,20 @@ static int mtk_pinconf_set(struct pinctrl_dev *pctldev, unsigned int pin,
                        if (err)
                                goto err;
                        break;
+               case MTK_PIN_CONFIG_PU_ADV:
+               case MTK_PIN_CONFIG_PD_ADV:
+                       if (hw->soc->adv_pull_set) {
+                               bool pullup;
+
+                               pullup = param == MTK_PIN_CONFIG_PU_ADV;
+                               err = hw->soc->adv_pull_set(hw, desc, pullup,
+                                                           arg);
+                               if (err)
+                                       return err;
+                       } else {
+                               return -ENOTSUPP;
+                       }
+                       break;
                default:
                        err = -ENOTSUPP;
                }
index 67c95ef85ba984cd5670faa1f69070aca3fec750..e66bf49a8c4c9ff0fafa3cdb4053e5d897dd63f6 100644 (file)
@@ -358,3 +358,58 @@ int mtk_pinconf_drive_get_rev1(struct mtk_pinctrl *hw,
 
        return 0;
 }
+
+int mtk_pinconf_adv_pull_set(struct mtk_pinctrl *hw,
+                            const struct mtk_pin_desc *desc, bool pullup,
+                            u32 arg)
+{
+       int err;
+
+       /* 10K off & 50K (75K) off, when (R0, R1) = (0, 0);
+        * 10K off & 50K (75K) on, when (R0, R1) = (0, 1);
+        * 10K on & 50K (75K) off, when (R0, R1) = (1, 0);
+        * 10K on & 50K (75K) on, when (R0, R1) = (1, 1)
+        */
+       err = mtk_hw_set_value(hw, desc->number, PINCTRL_PIN_REG_R0, arg & 1);
+       if (err)
+               return 0;
+
+       err = mtk_hw_set_value(hw, desc->number, PINCTRL_PIN_REG_R1,
+                              !!(arg & 2));
+       if (err)
+               return 0;
+
+       arg = pullup ? 0 : 1;
+
+       err = mtk_hw_set_value(hw, desc->number, PINCTRL_PIN_REG_PUPD, arg);
+
+       return err;
+}
+
+int mtk_pinconf_adv_pull_get(struct mtk_pinctrl *hw,
+                            const struct mtk_pin_desc *desc, bool pullup,
+                            u32 *val)
+{
+       u32 t, t2;
+       int err;
+
+       err = mtk_hw_get_value(hw, desc->number, PINCTRL_PIN_REG_PUPD, &t);
+       if (err)
+               return err;
+
+       /* t == 0 supposes PULLUP for the customized PULL setup */
+       if (pullup ^ !t)
+               return -EINVAL;
+
+       err = mtk_hw_get_value(hw, desc->number, PINCTRL_PIN_REG_R0, &t);
+       if (err)
+               return err;
+
+       err = mtk_hw_get_value(hw, desc->number, PINCTRL_PIN_REG_R1, &t2);
+       if (err)
+               return err;
+
+       *val = (t | t2 << 1) & 0x7;
+
+       return 0;
+}
index 3a55ef70e01b00e5f1915e04ffe10023973a8f5e..ce364a1a28a61785a6edbe3f976b089d04f8ee73 100644 (file)
@@ -51,6 +51,9 @@ enum {
        PINCTRL_PIN_REG_TDSEL,
        PINCTRL_PIN_REG_RDSEL,
        PINCTRL_PIN_REG_DRV,
+       PINCTRL_PIN_REG_PUPD,
+       PINCTRL_PIN_REG_R0,
+       PINCTRL_PIN_REG_R1,
        PINCTRL_PIN_REG_MAX,
 };
 
@@ -163,6 +166,13 @@ struct mtk_pin_soc {
                         const struct mtk_pin_desc *desc, u32 arg);
        int (*drive_get)(struct mtk_pinctrl *hw,
                         const struct mtk_pin_desc *desc, int *val);
+
+       int (*adv_pull_set)(struct mtk_pinctrl *hw,
+                           const struct mtk_pin_desc *desc, bool pullup,
+                           u32 arg);
+       int (*adv_pull_get)(struct mtk_pinctrl *hw,
+                           const struct mtk_pin_desc *desc, bool pullup,
+                           u32 *val);
 };
 
 struct mtk_pinctrl {
@@ -199,4 +209,11 @@ int mtk_pinconf_drive_set_rev1(struct mtk_pinctrl *hw,
 int mtk_pinconf_drive_get_rev1(struct mtk_pinctrl *hw,
                               const struct mtk_pin_desc *desc, int *val);
 
+int mtk_pinconf_adv_pull_set(struct mtk_pinctrl *hw,
+                            const struct mtk_pin_desc *desc, bool pullup,
+                            u32 arg);
+int mtk_pinconf_adv_pull_get(struct mtk_pinctrl *hw,
+                            const struct mtk_pin_desc *desc, bool pullup,
+                            u32 *val);
+
 #endif /* __PINCTRL_MTK_COMMON_V2_H */