]> git.ipfire.org Git - thirdparty/kernel/stable.git/commitdiff
pinctrl: meson: support amlogic A9 SoC
authorXianwei Zhao <xianwei.zhao@amlogic.com>
Thu, 7 May 2026 08:21:07 +0000 (08:21 +0000)
committerLinus Walleij <linusw@kernel.org>
Fri, 29 May 2026 20:30:02 +0000 (22:30 +0200)
In Amlogic A9 SoC, subordinate bank reuse other master bank is
not from bit0, and subordinate bank reuse multi master banks.

This submission implements this situation.

Signed-off-by: Xianwei Zhao <xianwei.zhao@amlogic.com>
Signed-off-by: Linus Walleij <linusw@kernel.org>
drivers/pinctrl/meson/pinctrl-amlogic-a4.c

index 4c88ea1de9092bf4c4f81c845364bfab3c0ebc24..88c5408f51cfe98e951437b9d7103f252b97b4de 100644 (file)
@@ -55,14 +55,18 @@ struct aml_pio_control {
  * partial bank(subordinate) pins mux config use other bank(main) mux registgers
  * m_bank_id:  the main bank which pin_id from 0, but register bit not from bit 0
  * m_bit_offs: bit offset the main bank mux register
+ * s_bit_offs: start bit that subordinate bank use mux register
  * sid:         start pin_id of subordinate bank
  * eid:         end pin_id of subordinate bank
+ * next:       subordinate bank reused multiple other bank groups.
  */
 struct multi_mux {
        unsigned int m_bank_id;
        unsigned int m_bit_offs;
+       unsigned int s_bit_offs;
        unsigned int sid;
        unsigned int eid;
+       const struct multi_mux *next;
 };
 
 struct aml_pctl_data {
@@ -124,12 +128,51 @@ static const char *aml_bank_name[31] = {
 "GPIOCC", "TEST_N", "ANALOG"
 };
 
+static const struct multi_mux multi_mux_a9[] = {
+       {
+               .m_bank_id = AMLOGIC_GPIO_C,
+               .m_bit_offs = 4,
+               .s_bit_offs = 0,
+               .sid = (AMLOGIC_GPIO_D << 8) + 16,
+               .eid = (AMLOGIC_GPIO_D << 8) + 16,
+               .next = &multi_mux_a9[1],
+       }, {
+               .m_bank_id = AMLOGIC_GPIO_AO,
+               .m_bit_offs = 0,
+               .s_bit_offs = 52,
+               .sid = (AMLOGIC_GPIO_D << 8) + 17,
+               .eid = (AMLOGIC_GPIO_D << 8) + 17,
+               .next = NULL,
+       }, {
+               .m_bank_id = AMLOGIC_GPIO_A,
+               .m_bit_offs = 0,
+               .s_bit_offs = 80,
+               .sid = (AMLOGIC_GPIO_Y << 8) + 8,
+               .eid = (AMLOGIC_GPIO_Y << 8) + 9,
+               .next = NULL,
+       }, {
+               .m_bank_id = AMLOGIC_GPIO_CC,
+               .m_bit_offs = 24,
+               .s_bit_offs = 0,
+               .sid = (AMLOGIC_GPIO_X << 8) + 16,
+               .eid = (AMLOGIC_GPIO_X << 8) + 17,
+               .next = NULL,
+       },
+};
+
+static const struct aml_pctl_data a9_priv_data = {
+       .number = ARRAY_SIZE(multi_mux_a9),
+       .p_mux = multi_mux_a9,
+};
+
 static const struct multi_mux multi_mux_s7[] = {
        {
                .m_bank_id = AMLOGIC_GPIO_CC,
                .m_bit_offs = 24,
+               .s_bit_offs = 0,
                .sid = (AMLOGIC_GPIO_X << 8) + 16,
                .eid = (AMLOGIC_GPIO_X << 8) + 19,
+               .next = NULL,
        },
 };
 
@@ -142,13 +185,17 @@ static const struct multi_mux multi_mux_s6[] = {
        {
                .m_bank_id = AMLOGIC_GPIO_CC,
                .m_bit_offs = 24,
+               .s_bit_offs = 0,
                .sid = (AMLOGIC_GPIO_X << 8) + 16,
                .eid = (AMLOGIC_GPIO_X << 8) + 19,
+               .next = NULL,
        }, {
                .m_bank_id = AMLOGIC_GPIO_F,
                .m_bit_offs = 4,
+               .s_bit_offs = 0,
                .sid = (AMLOGIC_GPIO_D << 8) + 6,
                .eid = (AMLOGIC_GPIO_D << 8) + 6,
+               .next = NULL,
        },
 };
 
@@ -177,31 +224,34 @@ static int aml_pctl_set_function(struct aml_pinctrl *info,
        struct aml_gpio_bank *bank = gpio_chip_to_bank(range->gc);
        unsigned int shift;
        int reg;
-       int i;
+       int i, loop_count;
        unsigned int offset = bank->mux_bit_offs;
        const struct multi_mux *p_mux;
 
        /* peculiar mux reg set */
-       if (bank->p_mux) {
-               p_mux = bank->p_mux;
+       loop_count = 10;
+       p_mux = bank->p_mux;
+       while (p_mux && loop_count) {
                if (pin_id >= p_mux->sid && pin_id <= p_mux->eid) {
                        bank = NULL;
                        for (i = 0; i < info->nbanks; i++) {
                                if (info->banks[i].bank_id == p_mux->m_bank_id) {
                                        bank = &info->banks[i];
-                                               break;
+                                       break;
                                }
                        }
 
                        if (!bank || !bank->reg_mux)
                                return -EINVAL;
 
-                       shift = (pin_id - p_mux->sid) << 2;
+                       shift = ((pin_id - p_mux->sid) << 2) + p_mux->s_bit_offs;
                        reg = (shift / 32) * 4;
                        offset = shift % 32;
                        return regmap_update_bits(bank->reg_mux, reg,
                                        0xf << offset, (func & 0xf) << offset);
                }
+               p_mux = p_mux->next;
+               loop_count--;
        }
 
        /* normal mux reg set */
@@ -1158,6 +1208,7 @@ static int aml_pctl_probe(struct platform_device *pdev)
 
 static const struct of_device_id aml_pctl_of_match[] = {
        { .compatible = "amlogic,pinctrl-a4", },
+       { .compatible = "amlogic,pinctrl-a9", .data = &a9_priv_data, },
        { .compatible = "amlogic,pinctrl-s7", .data = &s7_priv_data, },
        { .compatible = "amlogic,pinctrl-s6", .data = &s6_priv_data, },
        { /* sentinel */ }