Quellcodebibliothek Statistik Leitseite products/Sources/formale Sprachen/C/Linux/drivers/pinctrl/   (Open Source Betriebssystem Version 6.17.9©)  Datei vom 24.10.2025 mit Größe 132 kB image not shown  

Quelle  pinctrl-rockchip.c   Sprache: C

 
// SPDX-License-Identifier: GPL-2.0-only
/*
 * Pinctrl driver for Rockchip SoCs
 * Copyright (c) 2020-2024 Rockchip Electronics Co., Ltd.
 * Copyright (c) 2013 MundoReader S.L.
 * Author: Heiko Stuebner <heiko@sntech.de>
 *
 * With some ideas taken from pinctrl-samsung:
 * Copyright (c) 2012 Samsung Electronics Co., Ltd.
 * http://www.samsung.com
 * Copyright (c) 2012 Linaro Ltd
 * https://www.linaro.org
 *
 * and pinctrl-at91:
 * Copyright (C) 2011-2012 Jean-Christophe PLAGNIOL-VILLARD <plagnioj@jcrosoft.com>
 */


#include <linux/init.h>
#include <linux/module.h>
#include <linux/platform_device.h>
#include <linux/io.h>
#include <linux/bitops.h>
#include <linux/gpio/driver.h>
#include <linux/of.h>
#include <linux/of_platform.h>
#include <linux/pinctrl/machine.h>
#include <linux/pinctrl/pinconf.h>
#include <linux/pinctrl/pinctrl.h>
#include <linux/pinctrl/pinmux.h>
#include <linux/pinctrl/pinconf-generic.h>
#include <linux/irqchip/chained_irq.h>
#include <linux/clk.h>
#include <linux/regmap.h>
#include <linux/mfd/syscon.h>
#include <linux/string_helpers.h>

#include <dt-bindings/pinctrl/rockchip.h>

#include "core.h"
#include "pinconf.h"
#include "pinctrl-rockchip.h"

/*
 * Generate a bitmask for setting a value (v) with a write mask bit in hiword
 * register 31:16 area.
 */

#define WRITE_MASK_VAL(h, l, v) \
 (GENMASK(((h) + 16), ((l) + 16)) | (((v) << (l)) & GENMASK((h), (l))))

/*
 * Encode variants of iomux registers into a type variable
 */

#define IOMUX_GPIO_ONLY  BIT(0)
#define IOMUX_WIDTH_4BIT BIT(1)
#define IOMUX_SOURCE_PMU BIT(2)
#define IOMUX_UNROUTED  BIT(3)
#define IOMUX_WIDTH_3BIT BIT(4)
#define IOMUX_WIDTH_2BIT BIT(5)
#define IOMUX_L_SOURCE_PMU BIT(6)

#define PIN_BANK(id, pins, label)   \
 {      \
  .bank_num = id,   \
  .nr_pins = pins,   \
  .name  = label,  \
  .iomux  = {   \
   { .offset = -1 },  \
   { .offset = -1 },  \
   { .offset = -1 },  \
   { .offset = -1 },  \
  },     \
 }

#define PIN_BANK_IOMUX_FLAGS(id, pins, label, iom0, iom1, iom2, iom3) \
 {        \
  .bank_num = id,     \
  .nr_pins = pins,     \
  .name  = label,    \
  .iomux  = {     \
   { .type = iom0, .offset = -1 },   \
   { .type = iom1, .offset = -1 },   \
   { .type = iom2, .offset = -1 },   \
   { .type = iom3, .offset = -1 },   \
  },       \
 }

#define PIN_BANK_IOMUX_FLAGS_OFFSET_PULL_FLAGS(id, pins, label, iom0, \
            iom1, iom2, iom3, \
            offset0, offset1, \
            offset2, offset3, pull0, \
            pull1, pull2, pull3) \
 {        \
  .bank_num = id,     \
  .nr_pins = pins,     \
  .name  = label,    \
  .iomux  = {     \
   { .type = iom0, .offset = offset0 },  \
   { .type = iom1, .offset = offset1 },  \
   { .type = iom2, .offset = offset2 },  \
   { .type = iom3, .offset = offset3 },  \
  },       \
  .pull_type[0] = pull0,     \
  .pull_type[1] = pull1,     \
  .pull_type[2] = pull2,     \
  .pull_type[3] = pull3,     \
 }

#define PIN_BANK_DRV_FLAGS(id, pins, label, type0, type1, type2, type3) \
 {        \
  .bank_num = id,     \
  .nr_pins = pins,     \
  .name  = label,    \
  .iomux  = {     \
   { .offset = -1 },    \
   { .offset = -1 },    \
   { .offset = -1 },    \
   { .offset = -1 },    \
  },       \
  .drv  = {     \
   { .drv_type = type0, .offset = -1 },  \
   { .drv_type = type1, .offset = -1 },  \
   { .drv_type = type2, .offset = -1 },  \
   { .drv_type = type3, .offset = -1 },  \
  },       \
 }

#define PIN_BANK_IOMUX_FLAGS_PULL_FLAGS(id, pins, label, iom0, iom1, \
     iom2, iom3, pull0, pull1, \
     pull2, pull3)   \
 {        \
  .bank_num = id,     \
  .nr_pins = pins,     \
  .name  = label,    \
  .iomux  = {     \
   { .type = iom0, .offset = -1 },   \
   { .type = iom1, .offset = -1 },   \
   { .type = iom2, .offset = -1 },   \
   { .type = iom3, .offset = -1 },   \
  },       \
  .pull_type[0] = pull0,     \
  .pull_type[1] = pull1,     \
  .pull_type[2] = pull2,     \
  .pull_type[3] = pull3,     \
 }

#define PIN_BANK_DRV_FLAGS_PULL_FLAGS(id, pins, label, drv0, drv1, \
          drv2, drv3, pull0, pull1,  \
          pull2, pull3)   \
 {        \
  .bank_num = id,     \
  .nr_pins = pins,     \
  .name  = label,    \
  .iomux  = {     \
   { .offset = -1 },    \
   { .offset = -1 },    \
   { .offset = -1 },    \
   { .offset = -1 },    \
  },       \
  .drv  = {     \
   { .drv_type = drv0, .offset = -1 },  \
   { .drv_type = drv1, .offset = -1 },  \
   { .drv_type = drv2, .offset = -1 },  \
   { .drv_type = drv3, .offset = -1 },  \
  },       \
  .pull_type[0] = pull0,     \
  .pull_type[1] = pull1,     \
  .pull_type[2] = pull2,     \
  .pull_type[3] = pull3,     \
 }

#define PIN_BANK_IOMUX_FLAGS_OFFSET(id, pins, label, iom0, iom1, iom2, \
        iom3, offset0, offset1, offset2, \
        offset3)    \
 {        \
  .bank_num = id,     \
  .nr_pins = pins,     \
  .name  = label,    \
  .iomux  = {     \
   { .type = iom0, .offset = offset0 },  \
   { .type = iom1, .offset = offset1 },  \
   { .type = iom2, .offset = offset2 },  \
   { .type = iom3, .offset = offset3 },  \
  },       \
 }

#define PIN_BANK_IOMUX_DRV_FLAGS_OFFSET(id, pins, label, iom0, iom1, \
     iom2, iom3, drv0, drv1, drv2, \
     drv3, offset0, offset1,  \
     offset2, offset3)  \
 {        \
  .bank_num = id,     \
  .nr_pins = pins,     \
  .name  = label,    \
  .iomux  = {     \
   { .type = iom0, .offset = -1 },   \
   { .type = iom1, .offset = -1 },   \
   { .type = iom2, .offset = -1 },   \
   { .type = iom3, .offset = -1 },   \
  },       \
  .drv  = {     \
   { .drv_type = drv0, .offset = offset0 }, \
   { .drv_type = drv1, .offset = offset1 }, \
   { .drv_type = drv2, .offset = offset2 }, \
   { .drv_type = drv3, .offset = offset3 }, \
  },       \
 }

#define PIN_BANK_IOMUX_FLAGS_DRV_FLAGS_OFFSET_PULL_FLAGS(id, pins, \
           label, iom0, iom1, iom2,  \
           iom3, drv0, drv1, drv2,   \
           drv3, offset0, offset1,   \
           offset2, offset3, pull0,  \
           pull1, pull2, pull3) \
 {        \
  .bank_num = id,     \
  .nr_pins = pins,     \
  .name  = label,    \
  .iomux  = {     \
   { .type = iom0, .offset = -1 },   \
   { .type = iom1, .offset = -1 },   \
   { .type = iom2, .offset = -1 },   \
   { .type = iom3, .offset = -1 },   \
  },       \
  .drv  = {     \
   { .drv_type = drv0, .offset = offset0 }, \
   { .drv_type = drv1, .offset = offset1 }, \
   { .drv_type = drv2, .offset = offset2 }, \
   { .drv_type = drv3, .offset = offset3 }, \
  },       \
  .pull_type[0] = pull0,     \
  .pull_type[1] = pull1,     \
  .pull_type[2] = pull2,     \
  .pull_type[3] = pull3,     \
 }

#define PIN_BANK_MUX_ROUTE_FLAGS(ID, PIN, FUNC, REG, VAL, FLAG)  \
 {        \
  .bank_num = ID,     \
  .pin  = PIN,     \
  .func  = FUNC,     \
  .route_offset = REG,     \
  .route_val = VAL,     \
  .route_location = FLAG,     \
 }

#define RK_MUXROUTE_SAME(ID, PIN, FUNC, REG, VAL) \
 PIN_BANK_MUX_ROUTE_FLAGS(ID, PIN, FUNC, REG, VAL, ROCKCHIP_ROUTE_SAME)

#define RK_MUXROUTE_GRF(ID, PIN, FUNC, REG, VAL) \
 PIN_BANK_MUX_ROUTE_FLAGS(ID, PIN, FUNC, REG, VAL, ROCKCHIP_ROUTE_GRF)

#define RK_MUXROUTE_PMU(ID, PIN, FUNC, REG, VAL) \
 PIN_BANK_MUX_ROUTE_FLAGS(ID, PIN, FUNC, REG, VAL, ROCKCHIP_ROUTE_PMU)

#define RK3588_PIN_BANK_FLAGS(ID, PIN, LABEL, M, P)   \
 PIN_BANK_IOMUX_FLAGS_PULL_FLAGS(ID, PIN, LABEL, M, M, M, M, P, P, P, P)

static struct regmap_config rockchip_regmap_config = {
 .reg_bits = 32,
 .val_bits = 32,
 .reg_stride = 4,
};

static inline const struct rockchip_pin_group *pinctrl_name_to_group(
     const struct rockchip_pinctrl *info,
     const char *name)
{
 int i;

 for (i = 0; i < info->ngroups; i++) {
  if (!strcmp(info->groups[i].name, name))
   return &info->groups[i];
 }

 return NULL;
}

/*
 * given a pin number that is local to a pin controller, find out the pin bank
 * and the register base of the pin bank.
 */

static struct rockchip_pin_bank *pin_to_bank(struct rockchip_pinctrl *info,
        unsigned pin)
{
 struct rockchip_pin_bank *b = info->ctrl->pin_banks;

 while (pin >= (b->pin_base + b->nr_pins))
  b++;

 return b;
}

static struct rockchip_pin_bank *bank_num_to_bank(
     struct rockchip_pinctrl *info,
     unsigned num)
{
 struct rockchip_pin_bank *b = info->ctrl->pin_banks;
 int i;

 for (i = 0; i < info->ctrl->nr_banks; i++, b++) {
  if (b->bank_num == num)
   return b;
 }

 return ERR_PTR(-EINVAL);
}

/*
 * Pinctrl_ops handling
 */


static int rockchip_get_groups_count(struct pinctrl_dev *pctldev)
{
 struct rockchip_pinctrl *info = pinctrl_dev_get_drvdata(pctldev);

 return info->ngroups;
}

static const char *rockchip_get_group_name(struct pinctrl_dev *pctldev,
       unsigned selector)
{
 struct rockchip_pinctrl *info = pinctrl_dev_get_drvdata(pctldev);

 return info->groups[selector].name;
}

static int rockchip_get_group_pins(struct pinctrl_dev *pctldev,
          unsigned selector, const unsigned **pins,
          unsigned *npins)
{
 struct rockchip_pinctrl *info = pinctrl_dev_get_drvdata(pctldev);

 if (selector >= info->ngroups)
  return -EINVAL;

 *pins = info->groups[selector].pins;
 *npins = info->groups[selector].npins;

 return 0;
}

static int rockchip_dt_node_to_map(struct pinctrl_dev *pctldev,
     struct device_node *np,
     struct pinctrl_map **map, unsigned *num_maps)
{
 struct rockchip_pinctrl *info = pinctrl_dev_get_drvdata(pctldev);
 const struct rockchip_pin_group *grp;
 struct device *dev = info->dev;
 struct pinctrl_map *new_map;
 struct device_node *parent;
 int map_num = 1;
 int i;

 /*
 * first find the group of this node and check if we need to create
 * config maps for pins
 */

 grp = pinctrl_name_to_group(info, np->name);
 if (!grp) {
  dev_err(dev, "unable to find group for node %pOFn\n", np);
  return -EINVAL;
 }

 map_num += grp->npins;

 new_map = kcalloc(map_num, sizeof(*new_map), GFP_KERNEL);
 if (!new_map)
  return -ENOMEM;

 *map = new_map;
 *num_maps = map_num;

 /* create mux map */
 parent = of_get_parent(np);
 if (!parent) {
  kfree(new_map);
  return -EINVAL;
 }
 new_map[0].type = PIN_MAP_TYPE_MUX_GROUP;
 new_map[0].data.mux.function = parent->name;
 new_map[0].data.mux.group = np->name;
 of_node_put(parent);

 /* create config map */
 new_map++;
 for (i = 0; i < grp->npins; i++) {
  new_map[i].type = PIN_MAP_TYPE_CONFIGS_PIN;
  new_map[i].data.configs.group_or_pin =
    pin_get_name(pctldev, grp->pins[i]);
  new_map[i].data.configs.configs = grp->data[i].configs;
  new_map[i].data.configs.num_configs = grp->data[i].nconfigs;
 }

 dev_dbg(dev, "maps: function %s group %s num %d\n",
  (*map)->data.mux.function, (*map)->data.mux.group, map_num);

 return 0;
}

static void rockchip_dt_free_map(struct pinctrl_dev *pctldev,
        struct pinctrl_map *map, unsigned num_maps)
{
 kfree(map);
}

static const struct pinctrl_ops rockchip_pctrl_ops = {
 .get_groups_count = rockchip_get_groups_count,
 .get_group_name  = rockchip_get_group_name,
 .get_group_pins  = rockchip_get_group_pins,
 .dt_node_to_map  = rockchip_dt_node_to_map,
 .dt_free_map  = rockchip_dt_free_map,
};

/*
 * Hardware access
 */


static struct rockchip_mux_recalced_data rv1108_mux_recalced_data[] = {
 {
  .num = 1,
  .pin = 0,
  .reg = 0x418,
  .bit = 0,
  .mask = 0x3
 }, {
  .num = 1,
  .pin = 1,
  .reg = 0x418,
  .bit = 2,
  .mask = 0x3
 }, {
  .num = 1,
  .pin = 2,
  .reg = 0x418,
  .bit = 4,
  .mask = 0x3
 }, {
  .num = 1,
  .pin = 3,
  .reg = 0x418,
  .bit = 6,
  .mask = 0x3
 }, {
  .num = 1,
  .pin = 4,
  .reg = 0x418,
  .bit = 8,
  .mask = 0x3
 }, {
  .num = 1,
  .pin = 5,
  .reg = 0x418,
  .bit = 10,
  .mask = 0x3
 }, {
  .num = 1,
  .pin = 6,
  .reg = 0x418,
  .bit = 12,
  .mask = 0x3
 }, {
  .num = 1,
  .pin = 7,
  .reg = 0x418,
  .bit = 14,
  .mask = 0x3
 }, {
  .num = 1,
  .pin = 8,
  .reg = 0x41c,
  .bit = 0,
  .mask = 0x3
 }, {
  .num = 1,
  .pin = 9,
  .reg = 0x41c,
  .bit = 2,
  .mask = 0x3
 },
};

static struct rockchip_mux_recalced_data rv1126_mux_recalced_data[] = {
 {
  .num = 0,
  .pin = 20,
  .reg = 0x10000,
  .bit = 0,
  .mask = 0xf
 },
 {
  .num = 0,
  .pin = 21,
  .reg = 0x10000,
  .bit = 4,
  .mask = 0xf
 },
 {
  .num = 0,
  .pin = 22,
  .reg = 0x10000,
  .bit = 8,
  .mask = 0xf
 },
 {
  .num = 0,
  .pin = 23,
  .reg = 0x10000,
  .bit = 12,
  .mask = 0xf
 },
};

static  struct rockchip_mux_recalced_data rk3128_mux_recalced_data[] = {
 {
  .num = 2,
  .pin = 20,
  .reg = 0xe8,
  .bit = 0,
  .mask = 0x7
 }, {
  .num = 2,
  .pin = 21,
  .reg = 0xe8,
  .bit = 4,
  .mask = 0x7
 }, {
  .num = 2,
  .pin = 22,
  .reg = 0xe8,
  .bit = 8,
  .mask = 0x7
 }, {
  .num = 2,
  .pin = 23,
  .reg = 0xe8,
  .bit = 12,
  .mask = 0x7
 }, {
  .num = 2,
  .pin = 24,
  .reg = 0xd4,
  .bit = 12,
  .mask = 0x7
 },
};

static struct rockchip_mux_recalced_data rk3308_mux_recalced_data[] = {
 {
  /* gpio1b6_sel */
  .num = 1,
  .pin = 14,
  .reg = 0x28,
  .bit = 12,
  .mask = 0xf
 }, {
  /* gpio1b7_sel */
  .num = 1,
  .pin = 15,
  .reg = 0x2c,
  .bit = 0,
  .mask = 0x3
 }, {
  /* gpio1c2_sel */
  .num = 1,
  .pin = 18,
  .reg = 0x30,
  .bit = 4,
  .mask = 0xf
 }, {
  /* gpio1c3_sel */
  .num = 1,
  .pin = 19,
  .reg = 0x30,
  .bit = 8,
  .mask = 0xf
 }, {
  /* gpio1c4_sel */
  .num = 1,
  .pin = 20,
  .reg = 0x30,
  .bit = 12,
  .mask = 0xf
 }, {
  /* gpio1c5_sel */
  .num = 1,
  .pin = 21,
  .reg = 0x34,
  .bit = 0,
  .mask = 0xf
 }, {
  /* gpio1c6_sel */
  .num = 1,
  .pin = 22,
  .reg = 0x34,
  .bit = 4,
  .mask = 0xf
 }, {
  /* gpio1c7_sel */
  .num = 1,
  .pin = 23,
  .reg = 0x34,
  .bit = 8,
  .mask = 0xf
 }, {
  /* gpio2a2_sel */
  .num = 2,
  .pin = 2,
  .reg = 0x40,
  .bit = 4,
  .mask = 0x3
 }, {
  /* gpio2a3_sel */
  .num = 2,
  .pin = 3,
  .reg = 0x40,
  .bit = 6,
  .mask = 0x3
 }, {
  /* gpio2c0_sel */
  .num = 2,
  .pin = 16,
  .reg = 0x50,
  .bit = 0,
  .mask = 0x3
 }, {
  /* gpio3b2_sel */
  .num = 3,
  .pin = 10,
  .reg = 0x68,
  .bit = 4,
  .mask = 0x3
 }, {
  /* gpio3b3_sel */
  .num = 3,
  .pin = 11,
  .reg = 0x68,
  .bit = 6,
  .mask = 0x3
 }, {
  /* gpio3b4_sel */
  .num = 3,
  .pin = 12,
  .reg = 0x68,
  .bit = 8,
  .mask = 0xf
 }, {
  /* gpio3b5_sel */
  .num = 3,
  .pin = 13,
  .reg = 0x68,
  .bit = 12,
  .mask = 0xf
 },
};

static struct rockchip_mux_recalced_data rk3328_mux_recalced_data[] = {
 {
  /* gpio2_b7_sel */
  .num = 2,
  .pin = 15,
  .reg = 0x28,
  .bit = 0,
  .mask = 0x7
 }, {
  /* gpio2_c7_sel */
  .num = 2,
  .pin = 23,
  .reg = 0x30,
  .bit = 14,
  .mask = 0x3
 }, {
  /* gpio3_b1_sel */
  .num = 3,
  .pin = 9,
  .reg = 0x44,
  .bit = 2,
  .mask = 0x3
 }, {
  /* gpio3_b2_sel */
  .num = 3,
  .pin = 10,
  .reg = 0x44,
  .bit = 4,
  .mask = 0x3
 }, {
  /* gpio3_b3_sel */
  .num = 3,
  .pin = 11,
  .reg = 0x44,
  .bit = 6,
  .mask = 0x3
 }, {
  /* gpio3_b4_sel */
  .num = 3,
  .pin = 12,
  .reg = 0x44,
  .bit = 8,
  .mask = 0x3
 }, {
  /* gpio3_b5_sel */
  .num = 3,
  .pin = 13,
  .reg = 0x44,
  .bit = 10,
  .mask = 0x3
 }, {
  /* gpio3_b6_sel */
  .num = 3,
  .pin = 14,
  .reg = 0x44,
  .bit = 12,
  .mask = 0x3
 }, {
  /* gpio3_b7_sel */
  .num = 3,
  .pin = 15,
  .reg = 0x44,
  .bit = 14,
  .mask = 0x3
 },
};

static void rockchip_get_recalced_mux(struct rockchip_pin_bank *bank, int pin,
          int *reg, u8 *bit, int *mask)
{
 struct rockchip_pinctrl *info = bank->drvdata;
 struct rockchip_pin_ctrl *ctrl = info->ctrl;
 struct rockchip_mux_recalced_data *data;
 int i;

 for (i = 0; i < ctrl->niomux_recalced; i++) {
  data = &ctrl->iomux_recalced[i];
  if (data->num == bank->bank_num &&
      data->pin == pin)
   break;
 }

 if (i >= ctrl->niomux_recalced)
  return;

 *reg = data->reg;
 *mask = data->mask;
 *bit = data->bit;
}

static struct rockchip_mux_route_data px30_mux_route_data[] = {
 RK_MUXROUTE_SAME(2, RK_PB4, 1, 0x184, BIT(16 + 7)), /* cif-d0m0 */
 RK_MUXROUTE_SAME(3, RK_PA1, 3, 0x184, BIT(16 + 7) | BIT(7)), /* cif-d0m1 */
 RK_MUXROUTE_SAME(2, RK_PB6, 1, 0x184, BIT(16 + 7)), /* cif-d1m0 */
 RK_MUXROUTE_SAME(3, RK_PA2, 3, 0x184, BIT(16 + 7) | BIT(7)), /* cif-d1m1 */
 RK_MUXROUTE_SAME(2, RK_PA0, 1, 0x184, BIT(16 + 7)), /* cif-d2m0 */
 RK_MUXROUTE_SAME(3, RK_PA3, 3, 0x184, BIT(16 + 7) | BIT(7)), /* cif-d2m1 */
 RK_MUXROUTE_SAME(2, RK_PA1, 1, 0x184, BIT(16 + 7)), /* cif-d3m0 */
 RK_MUXROUTE_SAME(3, RK_PA5, 3, 0x184, BIT(16 + 7) | BIT(7)), /* cif-d3m1 */
 RK_MUXROUTE_SAME(2, RK_PA2, 1, 0x184, BIT(16 + 7)), /* cif-d4m0 */
 RK_MUXROUTE_SAME(3, RK_PA7, 3, 0x184, BIT(16 + 7) | BIT(7)), /* cif-d4m1 */
 RK_MUXROUTE_SAME(2, RK_PA3, 1, 0x184, BIT(16 + 7)), /* cif-d5m0 */
 RK_MUXROUTE_SAME(3, RK_PB0, 3, 0x184, BIT(16 + 7) | BIT(7)), /* cif-d5m1 */
 RK_MUXROUTE_SAME(2, RK_PA4, 1, 0x184, BIT(16 + 7)), /* cif-d6m0 */
 RK_MUXROUTE_SAME(3, RK_PB1, 3, 0x184, BIT(16 + 7) | BIT(7)), /* cif-d6m1 */
 RK_MUXROUTE_SAME(2, RK_PA5, 1, 0x184, BIT(16 + 7)), /* cif-d7m0 */
 RK_MUXROUTE_SAME(3, RK_PB4, 3, 0x184, BIT(16 + 7) | BIT(7)), /* cif-d7m1 */
 RK_MUXROUTE_SAME(2, RK_PA6, 1, 0x184, BIT(16 + 7)), /* cif-d8m0 */
 RK_MUXROUTE_SAME(3, RK_PB6, 3, 0x184, BIT(16 + 7) | BIT(7)), /* cif-d8m1 */
 RK_MUXROUTE_SAME(2, RK_PA7, 1, 0x184, BIT(16 + 7)), /* cif-d9m0 */
 RK_MUXROUTE_SAME(3, RK_PB7, 3, 0x184, BIT(16 + 7) | BIT(7)), /* cif-d9m1 */
 RK_MUXROUTE_SAME(2, RK_PB7, 1, 0x184, BIT(16 + 7)), /* cif-d10m0 */
 RK_MUXROUTE_SAME(3, RK_PC6, 3, 0x184, BIT(16 + 7) | BIT(7)), /* cif-d10m1 */
 RK_MUXROUTE_SAME(2, RK_PC0, 1, 0x184, BIT(16 + 7)), /* cif-d11m0 */
 RK_MUXROUTE_SAME(3, RK_PC7, 3, 0x184, BIT(16 + 7) | BIT(7)), /* cif-d11m1 */
 RK_MUXROUTE_SAME(2, RK_PB0, 1, 0x184, BIT(16 + 7)), /* cif-vsyncm0 */
 RK_MUXROUTE_SAME(3, RK_PD1, 3, 0x184, BIT(16 + 7) | BIT(7)), /* cif-vsyncm1 */
 RK_MUXROUTE_SAME(2, RK_PB1, 1, 0x184, BIT(16 + 7)), /* cif-hrefm0 */
 RK_MUXROUTE_SAME(3, RK_PD2, 3, 0x184, BIT(16 + 7) | BIT(7)), /* cif-hrefm1 */
 RK_MUXROUTE_SAME(2, RK_PB2, 1, 0x184, BIT(16 + 7)), /* cif-clkinm0 */
 RK_MUXROUTE_SAME(3, RK_PD3, 3, 0x184, BIT(16 + 7) | BIT(7)), /* cif-clkinm1 */
 RK_MUXROUTE_SAME(2, RK_PB3, 1, 0x184, BIT(16 + 7)), /* cif-clkoutm0 */
 RK_MUXROUTE_SAME(3, RK_PD0, 3, 0x184, BIT(16 + 7) | BIT(7)), /* cif-clkoutm1 */
 RK_MUXROUTE_SAME(3, RK_PC6, 2, 0x184, BIT(16 + 8)), /* pdm-m0 */
 RK_MUXROUTE_SAME(2, RK_PC6, 1, 0x184, BIT(16 + 8) | BIT(8)), /* pdm-m1 */
 RK_MUXROUTE_SAME(3, RK_PD3, 2, 0x184, BIT(16 + 8)), /* pdm-sdi0m0 */
 RK_MUXROUTE_SAME(2, RK_PC5, 2, 0x184, BIT(16 + 8) | BIT(8)), /* pdm-sdi0m1 */
 RK_MUXROUTE_SAME(1, RK_PD3, 2, 0x184, BIT(16 + 10)), /* uart2-rxm0 */
 RK_MUXROUTE_SAME(2, RK_PB6, 2, 0x184, BIT(16 + 10) | BIT(10)), /* uart2-rxm1 */
 RK_MUXROUTE_SAME(1, RK_PD2, 2, 0x184, BIT(16 + 10)), /* uart2-txm0 */
 RK_MUXROUTE_SAME(2, RK_PB4, 2, 0x184, BIT(16 + 10) | BIT(10)), /* uart2-txm1 */
 RK_MUXROUTE_SAME(0, RK_PC1, 2, 0x184, BIT(16 + 9)), /* uart3-rxm0 */
 RK_MUXROUTE_SAME(1, RK_PB7, 2, 0x184, BIT(16 + 9) | BIT(9)), /* uart3-rxm1 */
 RK_MUXROUTE_SAME(0, RK_PC0, 2, 0x184, BIT(16 + 9)), /* uart3-txm0 */
 RK_MUXROUTE_SAME(1, RK_PB6, 2, 0x184, BIT(16 + 9) | BIT(9)), /* uart3-txm1 */
 RK_MUXROUTE_SAME(0, RK_PC2, 2, 0x184, BIT(16 + 9)), /* uart3-ctsm0 */
 RK_MUXROUTE_SAME(1, RK_PB4, 2, 0x184, BIT(16 + 9) | BIT(9)), /* uart3-ctsm1 */
 RK_MUXROUTE_SAME(0, RK_PC3, 2, 0x184, BIT(16 + 9)), /* uart3-rtsm0 */
 RK_MUXROUTE_SAME(1, RK_PB5, 2, 0x184, BIT(16 + 9) | BIT(9)), /* uart3-rtsm1 */
};

static struct rockchip_mux_route_data rv1126_mux_route_data[] = {
 RK_MUXROUTE_GRF(3, RK_PD2, 1, 0x10260, WRITE_MASK_VAL(0, 0, 0)), /* I2S0_MCLK_M0 */
 RK_MUXROUTE_GRF(3, RK_PB0, 3, 0x10260, WRITE_MASK_VAL(0, 0, 1)), /* I2S0_MCLK_M1 */

 RK_MUXROUTE_GRF(0, RK_PD4, 4, 0x10260, WRITE_MASK_VAL(3, 2, 0)), /* I2S1_MCLK_M0 */
 RK_MUXROUTE_GRF(1, RK_PD5, 2, 0x10260, WRITE_MASK_VAL(3, 2, 1)), /* I2S1_MCLK_M1 */
 RK_MUXROUTE_GRF(2, RK_PC7, 6, 0x10260, WRITE_MASK_VAL(3, 2, 2)), /* I2S1_MCLK_M2 */

 RK_MUXROUTE_GRF(1, RK_PD0, 1, 0x10260, WRITE_MASK_VAL(4, 4, 0)), /* I2S2_MCLK_M0 */
 RK_MUXROUTE_GRF(2, RK_PB3, 2, 0x10260, WRITE_MASK_VAL(4, 4, 1)), /* I2S2_MCLK_M1 */

 RK_MUXROUTE_GRF(3, RK_PD4, 2, 0x10260, WRITE_MASK_VAL(12, 12, 0)), /* PDM_CLK0_M0 */
 RK_MUXROUTE_GRF(3, RK_PC0, 3, 0x10260, WRITE_MASK_VAL(12, 12, 1)), /* PDM_CLK0_M1 */

 RK_MUXROUTE_GRF(3, RK_PC6, 1, 0x10264, WRITE_MASK_VAL(0, 0, 0)), /* CIF_CLKOUT_M0 */
 RK_MUXROUTE_GRF(2, RK_PD1, 3, 0x10264, WRITE_MASK_VAL(0, 0, 1)), /* CIF_CLKOUT_M1 */

 RK_MUXROUTE_GRF(3, RK_PA4, 5, 0x10264, WRITE_MASK_VAL(5, 4, 0)), /* I2C3_SCL_M0 */
 RK_MUXROUTE_GRF(2, RK_PD4, 7, 0x10264, WRITE_MASK_VAL(5, 4, 1)), /* I2C3_SCL_M1 */
 RK_MUXROUTE_GRF(1, RK_PD6, 3, 0x10264, WRITE_MASK_VAL(5, 4, 2)), /* I2C3_SCL_M2 */

 RK_MUXROUTE_GRF(3, RK_PA0, 7, 0x10264, WRITE_MASK_VAL(6, 6, 0)), /* I2C4_SCL_M0 */
 RK_MUXROUTE_GRF(4, RK_PA0, 4, 0x10264, WRITE_MASK_VAL(6, 6, 1)), /* I2C4_SCL_M1 */

 RK_MUXROUTE_GRF(2, RK_PA5, 7, 0x10264, WRITE_MASK_VAL(9, 8, 0)), /* I2C5_SCL_M0 */
 RK_MUXROUTE_GRF(3, RK_PB0, 5, 0x10264, WRITE_MASK_VAL(9, 8, 1)), /* I2C5_SCL_M1 */
 RK_MUXROUTE_GRF(1, RK_PD0, 4, 0x10264, WRITE_MASK_VAL(9, 8, 2)), /* I2C5_SCL_M2 */

 RK_MUXROUTE_GRF(3, RK_PC0, 5, 0x10264, WRITE_MASK_VAL(11, 10, 0)), /* SPI1_CLK_M0 */
 RK_MUXROUTE_GRF(1, RK_PC6, 3, 0x10264, WRITE_MASK_VAL(11, 10, 1)), /* SPI1_CLK_M1 */
 RK_MUXROUTE_GRF(2, RK_PD5, 6, 0x10264, WRITE_MASK_VAL(11, 10, 2)), /* SPI1_CLK_M2 */

 RK_MUXROUTE_GRF(3, RK_PC0, 2, 0x10264, WRITE_MASK_VAL(12, 12, 0)), /* RGMII_CLK_M0 */
 RK_MUXROUTE_GRF(2, RK_PB7, 2, 0x10264, WRITE_MASK_VAL(12, 12, 1)), /* RGMII_CLK_M1 */

 RK_MUXROUTE_GRF(3, RK_PA1, 3, 0x10264, WRITE_MASK_VAL(13, 13, 0)), /* CAN_TXD_M0 */
 RK_MUXROUTE_GRF(3, RK_PA7, 5, 0x10264, WRITE_MASK_VAL(13, 13, 1)), /* CAN_TXD_M1 */

 RK_MUXROUTE_GRF(3, RK_PA4, 6, 0x10268, WRITE_MASK_VAL(0, 0, 0)), /* PWM8_M0 */
 RK_MUXROUTE_GRF(2, RK_PD7, 5, 0x10268, WRITE_MASK_VAL(0, 0, 1)), /* PWM8_M1 */

 RK_MUXROUTE_GRF(3, RK_PA5, 6, 0x10268, WRITE_MASK_VAL(2, 2, 0)), /* PWM9_M0 */
 RK_MUXROUTE_GRF(2, RK_PD6, 5, 0x10268, WRITE_MASK_VAL(2, 2, 1)), /* PWM9_M1 */

 RK_MUXROUTE_GRF(3, RK_PA6, 6, 0x10268, WRITE_MASK_VAL(4, 4, 0)), /* PWM10_M0 */
 RK_MUXROUTE_GRF(2, RK_PD5, 5, 0x10268, WRITE_MASK_VAL(4, 4, 1)), /* PWM10_M1 */

 RK_MUXROUTE_GRF(3, RK_PA7, 6, 0x10268, WRITE_MASK_VAL(6, 6, 0)), /* PWM11_IR_M0 */
 RK_MUXROUTE_GRF(3, RK_PA1, 5, 0x10268, WRITE_MASK_VAL(6, 6, 1)), /* PWM11_IR_M1 */

 RK_MUXROUTE_GRF(1, RK_PA5, 3, 0x10268, WRITE_MASK_VAL(8, 8, 0)), /* UART2_TX_M0 */
 RK_MUXROUTE_GRF(3, RK_PA2, 1, 0x10268, WRITE_MASK_VAL(8, 8, 1)), /* UART2_TX_M1 */

 RK_MUXROUTE_GRF(3, RK_PC6, 3, 0x10268, WRITE_MASK_VAL(11, 10, 0)), /* UART3_TX_M0 */
 RK_MUXROUTE_GRF(1, RK_PA7, 2, 0x10268, WRITE_MASK_VAL(11, 10, 1)), /* UART3_TX_M1 */
 RK_MUXROUTE_GRF(3, RK_PA0, 4, 0x10268, WRITE_MASK_VAL(11, 10, 2)), /* UART3_TX_M2 */

 RK_MUXROUTE_GRF(3, RK_PA4, 4, 0x10268, WRITE_MASK_VAL(13, 12, 0)), /* UART4_TX_M0 */
 RK_MUXROUTE_GRF(2, RK_PA6, 4, 0x10268, WRITE_MASK_VAL(13, 12, 1)), /* UART4_TX_M1 */
 RK_MUXROUTE_GRF(1, RK_PD5, 3, 0x10268, WRITE_MASK_VAL(13, 12, 2)), /* UART4_TX_M2 */

 RK_MUXROUTE_GRF(3, RK_PA6, 4, 0x10268, WRITE_MASK_VAL(15, 14, 0)), /* UART5_TX_M0 */
 RK_MUXROUTE_GRF(2, RK_PB0, 4, 0x10268, WRITE_MASK_VAL(15, 14, 1)), /* UART5_TX_M1 */
 RK_MUXROUTE_GRF(2, RK_PA0, 3, 0x10268, WRITE_MASK_VAL(15, 14, 2)), /* UART5_TX_M2 */

 RK_MUXROUTE_PMU(0, RK_PB6, 3, 0x0114, WRITE_MASK_VAL(0, 0, 0)), /* PWM0_M0 */
 RK_MUXROUTE_PMU(2, RK_PB3, 5, 0x0114, WRITE_MASK_VAL(0, 0, 1)), /* PWM0_M1 */

 RK_MUXROUTE_PMU(0, RK_PB7, 3, 0x0114, WRITE_MASK_VAL(2, 2, 0)), /* PWM1_M0 */
 RK_MUXROUTE_PMU(2, RK_PB2, 5, 0x0114, WRITE_MASK_VAL(2, 2, 1)), /* PWM1_M1 */

 RK_MUXROUTE_PMU(0, RK_PC0, 3, 0x0114, WRITE_MASK_VAL(4, 4, 0)), /* PWM2_M0 */
 RK_MUXROUTE_PMU(2, RK_PB1, 5, 0x0114, WRITE_MASK_VAL(4, 4, 1)), /* PWM2_M1 */

 RK_MUXROUTE_PMU(0, RK_PC1, 3, 0x0114, WRITE_MASK_VAL(6, 6, 0)), /* PWM3_IR_M0 */
 RK_MUXROUTE_PMU(2, RK_PB0, 5, 0x0114, WRITE_MASK_VAL(6, 6, 1)), /* PWM3_IR_M1 */

 RK_MUXROUTE_PMU(0, RK_PC2, 3, 0x0114, WRITE_MASK_VAL(8, 8, 0)), /* PWM4_M0 */
 RK_MUXROUTE_PMU(2, RK_PA7, 5, 0x0114, WRITE_MASK_VAL(8, 8, 1)), /* PWM4_M1 */

 RK_MUXROUTE_PMU(0, RK_PC3, 3, 0x0114, WRITE_MASK_VAL(10, 10, 0)), /* PWM5_M0 */
 RK_MUXROUTE_PMU(2, RK_PA6, 5, 0x0114, WRITE_MASK_VAL(10, 10, 1)), /* PWM5_M1 */

 RK_MUXROUTE_PMU(0, RK_PB2, 3, 0x0114, WRITE_MASK_VAL(12, 12, 0)), /* PWM6_M0 */
 RK_MUXROUTE_PMU(2, RK_PD4, 5, 0x0114, WRITE_MASK_VAL(12, 12, 1)), /* PWM6_M1 */

 RK_MUXROUTE_PMU(0, RK_PB1, 3, 0x0114, WRITE_MASK_VAL(14, 14, 0)), /* PWM7_IR_M0 */
 RK_MUXROUTE_PMU(3, RK_PA0, 5, 0x0114, WRITE_MASK_VAL(14, 14, 1)), /* PWM7_IR_M1 */

 RK_MUXROUTE_PMU(0, RK_PB0, 1, 0x0118, WRITE_MASK_VAL(1, 0, 0)), /* SPI0_CLK_M0 */
 RK_MUXROUTE_PMU(2, RK_PA1, 1, 0x0118, WRITE_MASK_VAL(1, 0, 1)), /* SPI0_CLK_M1 */
 RK_MUXROUTE_PMU(2, RK_PB2, 6, 0x0118, WRITE_MASK_VAL(1, 0, 2)), /* SPI0_CLK_M2 */

 RK_MUXROUTE_PMU(0, RK_PB6, 2, 0x0118, WRITE_MASK_VAL(2, 2, 0)), /* UART1_TX_M0 */
 RK_MUXROUTE_PMU(1, RK_PD0, 5, 0x0118, WRITE_MASK_VAL(2, 2, 1)), /* UART1_TX_M1 */
};

static struct rockchip_mux_route_data rk3128_mux_route_data[] = {
 RK_MUXROUTE_SAME(1, RK_PB2, 1, 0x144, BIT(16 + 3) | BIT(16 + 4)), /* spi-0 */
 RK_MUXROUTE_SAME(1, RK_PD3, 3, 0x144, BIT(16 + 3) | BIT(16 + 4) | BIT(3)), /* spi-1 */
 RK_MUXROUTE_SAME(0, RK_PB5, 2, 0x144, BIT(16 + 3) | BIT(16 + 4) | BIT(4)), /* spi-2 */
 RK_MUXROUTE_SAME(1, RK_PA5, 1, 0x144, BIT(16 + 5)), /* i2s-0 */
 RK_MUXROUTE_SAME(0, RK_PB6, 1, 0x144, BIT(16 + 5) | BIT(5)), /* i2s-1 */
 RK_MUXROUTE_SAME(1, RK_PC6, 2, 0x144, BIT(16 + 6)), /* emmc-0 */
 RK_MUXROUTE_SAME(2, RK_PA4, 2, 0x144, BIT(16 + 6) | BIT(6)), /* emmc-1 */
};

static struct rockchip_mux_route_data rk3188_mux_route_data[] = {
 RK_MUXROUTE_SAME(0, RK_PD0, 1, 0xa0, BIT(16 + 11)), /* non-iomuxed emmc/flash pins on flash-dqs */
 RK_MUXROUTE_SAME(0, RK_PD0, 2, 0xa0, BIT(16 + 11) | BIT(11)), /* non-iomuxed emmc/flash pins on emmc-clk */
};

static struct rockchip_mux_route_data rk3228_mux_route_data[] = {
 RK_MUXROUTE_SAME(0, RK_PD2, 1, 0x50, BIT(16)), /* pwm0-0 */
 RK_MUXROUTE_SAME(3, RK_PC5, 1, 0x50, BIT(16) | BIT(0)), /* pwm0-1 */
 RK_MUXROUTE_SAME(0, RK_PD3, 1, 0x50, BIT(16 + 1)), /* pwm1-0 */
 RK_MUXROUTE_SAME(0, RK_PD6, 2, 0x50, BIT(16 + 1) | BIT(1)), /* pwm1-1 */
 RK_MUXROUTE_SAME(0, RK_PD4, 1, 0x50, BIT(16 + 2)), /* pwm2-0 */
 RK_MUXROUTE_SAME(1, RK_PB4, 2, 0x50, BIT(16 + 2) | BIT(2)), /* pwm2-1 */
 RK_MUXROUTE_SAME(3, RK_PD2, 1, 0x50, BIT(16 + 3)), /* pwm3-0 */
 RK_MUXROUTE_SAME(1, RK_PB3, 2, 0x50, BIT(16 + 3) | BIT(3)), /* pwm3-1 */
 RK_MUXROUTE_SAME(1, RK_PA1, 1, 0x50, BIT(16 + 4)), /* sdio-0_d0 */
 RK_MUXROUTE_SAME(3, RK_PA2, 1, 0x50, BIT(16 + 4) | BIT(4)), /* sdio-1_d0 */
 RK_MUXROUTE_SAME(0, RK_PB5, 2, 0x50, BIT(16 + 5)), /* spi-0_rx */
 RK_MUXROUTE_SAME(2, RK_PA0, 2, 0x50, BIT(16 + 5) | BIT(5)), /* spi-1_rx */
 RK_MUXROUTE_SAME(1, RK_PC6, 2, 0x50, BIT(16 + 7)), /* emmc-0_cmd */
 RK_MUXROUTE_SAME(2, RK_PA4, 2, 0x50, BIT(16 + 7) | BIT(7)), /* emmc-1_cmd */
 RK_MUXROUTE_SAME(1, RK_PC3, 2, 0x50, BIT(16 + 8)), /* uart2-0_rx */
 RK_MUXROUTE_SAME(1, RK_PB2, 2, 0x50, BIT(16 + 8) | BIT(8)), /* uart2-1_rx */
 RK_MUXROUTE_SAME(1, RK_PB2, 1, 0x50, BIT(16 + 11)), /* uart1-0_rx */
 RK_MUXROUTE_SAME(3, RK_PB5, 1, 0x50, BIT(16 + 11) | BIT(11)), /* uart1-1_rx */
};

static struct rockchip_mux_route_data rk3288_mux_route_data[] = {
 RK_MUXROUTE_SAME(7, RK_PC0, 2, 0x264, BIT(16 + 12) | BIT(12)), /* edphdmi_cecinoutt1 */
 RK_MUXROUTE_SAME(7, RK_PC7, 4, 0x264, BIT(16 + 12)), /* edphdmi_cecinout */
};

static struct rockchip_mux_route_data rk3308_mux_route_data[] = {
 RK_MUXROUTE_SAME(0, RK_PC3, 1, 0x314, BIT(16 + 0) | BIT(0)), /* rtc_clk */
 RK_MUXROUTE_SAME(1, RK_PC6, 2, 0x314, BIT(16 + 2) | BIT(16 + 3)), /* uart2_rxm0 */
 RK_MUXROUTE_SAME(4, RK_PD2, 2, 0x314, BIT(16 + 2) | BIT(16 + 3) | BIT(2)), /* uart2_rxm1 */
 RK_MUXROUTE_SAME(0, RK_PB7, 2, 0x314, BIT(16 + 4)), /* i2c3_sdam0 */
 RK_MUXROUTE_SAME(3, RK_PB4, 2, 0x314, BIT(16 + 4) | BIT(4)), /* i2c3_sdam1 */
 RK_MUXROUTE_SAME(1, RK_PA3, 2, 0x308, BIT(16 + 3)), /* i2s-8ch-1-sclktxm0 */
 RK_MUXROUTE_SAME(1, RK_PA4, 2, 0x308, BIT(16 + 3)), /* i2s-8ch-1-sclkrxm0 */
 RK_MUXROUTE_SAME(1, RK_PB5, 2, 0x308, BIT(16 + 3) | BIT(3)), /* i2s-8ch-1-sclktxm1 */
 RK_MUXROUTE_SAME(1, RK_PB6, 2, 0x308, BIT(16 + 3) | BIT(3)), /* i2s-8ch-1-sclkrxm1 */
 RK_MUXROUTE_SAME(1, RK_PA4, 3, 0x308, BIT(16 + 12) | BIT(16 + 13)), /* pdm-clkm0 */
 RK_MUXROUTE_SAME(1, RK_PB6, 4, 0x308, BIT(16 + 12) | BIT(16 + 13) | BIT(12)), /* pdm-clkm1 */
 RK_MUXROUTE_SAME(2, RK_PA6, 2, 0x308, BIT(16 + 12) | BIT(16 + 13) | BIT(13)), /* pdm-clkm2 */
 RK_MUXROUTE_SAME(2, RK_PA4, 3, 0x600, BIT(16 + 2) | BIT(2)), /* pdm-clkm-m2 */
};

static struct rockchip_mux_route_data rk3328_mux_route_data[] = {
 RK_MUXROUTE_SAME(1, RK_PA1, 2, 0x50, BIT(16) | BIT(16 + 1)), /* uart2dbg_rxm0 */
 RK_MUXROUTE_SAME(2, RK_PA1, 1, 0x50, BIT(16) | BIT(16 + 1) | BIT(0)), /* uart2dbg_rxm1 */
 RK_MUXROUTE_SAME(1, RK_PB3, 2, 0x50, BIT(16 + 2) | BIT(2)), /* gmac-m1_rxd0 */
 RK_MUXROUTE_SAME(1, RK_PB6, 2, 0x50, BIT(16 + 10) | BIT(10)), /* gmac-m1-optimized_rxd3 */
 RK_MUXROUTE_SAME(2, RK_PC3, 2, 0x50, BIT(16 + 3)), /* pdm_sdi0m0 */
 RK_MUXROUTE_SAME(1, RK_PC7, 3, 0x50, BIT(16 + 3) | BIT(3)), /* pdm_sdi0m1 */
 RK_MUXROUTE_SAME(3, RK_PA2, 4, 0x50, BIT(16 + 4) | BIT(16 + 5) | BIT(5)), /* spi_rxdm2 */
 RK_MUXROUTE_SAME(1, RK_PD0, 1, 0x50, BIT(16 + 6)), /* i2s2_sdim0 */
 RK_MUXROUTE_SAME(3, RK_PA2, 6, 0x50, BIT(16 + 6) | BIT(6)), /* i2s2_sdim1 */
 RK_MUXROUTE_SAME(2, RK_PC6, 3, 0x50, BIT(16 + 7) | BIT(7)), /* card_iom1 */
 RK_MUXROUTE_SAME(2, RK_PC0, 3, 0x50, BIT(16 + 8) | BIT(8)), /* tsp_d5m1 */
 RK_MUXROUTE_SAME(2, RK_PC0, 4, 0x50, BIT(16 + 9) | BIT(9)), /* cif_data5m1 */
};

static struct rockchip_mux_route_data rk3399_mux_route_data[] = {
 RK_MUXROUTE_SAME(4, RK_PB0, 2, 0xe21c, BIT(16 + 10) | BIT(16 + 11)), /* uart2dbga_rx */
 RK_MUXROUTE_SAME(4, RK_PC0, 2, 0xe21c, BIT(16 + 10) | BIT(16 + 11) | BIT(10)), /* uart2dbgb_rx */
 RK_MUXROUTE_SAME(4, RK_PC3, 1, 0xe21c, BIT(16 + 10) | BIT(16 + 11) | BIT(11)), /* uart2dbgc_rx */
 RK_MUXROUTE_SAME(2, RK_PD2, 2, 0xe21c, BIT(16 + 14)), /* pcie_clkreqn */
 RK_MUXROUTE_SAME(4, RK_PD0, 1, 0xe21c, BIT(16 + 14) | BIT(14)), /* pcie_clkreqnb */
};

static struct rockchip_mux_route_data rk3568_mux_route_data[] = {
 RK_MUXROUTE_PMU(0, RK_PB7, 1, 0x0110, WRITE_MASK_VAL(1, 0, 0)), /* PWM0 IO mux M0 */
 RK_MUXROUTE_PMU(0, RK_PC7, 2, 0x0110, WRITE_MASK_VAL(1, 0, 1)), /* PWM0 IO mux M1 */
 RK_MUXROUTE_PMU(0, RK_PC0, 1, 0x0110, WRITE_MASK_VAL(3, 2, 0)), /* PWM1 IO mux M0 */
 RK_MUXROUTE_PMU(0, RK_PB5, 4, 0x0110, WRITE_MASK_VAL(3, 2, 1)), /* PWM1 IO mux M1 */
 RK_MUXROUTE_PMU(0, RK_PC1, 1, 0x0110, WRITE_MASK_VAL(5, 4, 0)), /* PWM2 IO mux M0 */
 RK_MUXROUTE_PMU(0, RK_PB6, 4, 0x0110, WRITE_MASK_VAL(5, 4, 1)), /* PWM2 IO mux M1 */
 RK_MUXROUTE_GRF(0, RK_PB3, 2, 0x0300, WRITE_MASK_VAL(0, 0, 0)), /* CAN0 IO mux M0 */
 RK_MUXROUTE_GRF(2, RK_PA1, 4, 0x0300, WRITE_MASK_VAL(0, 0, 1)), /* CAN0 IO mux M1 */
 RK_MUXROUTE_GRF(1, RK_PA1, 3, 0x0300, WRITE_MASK_VAL(2, 2, 0)), /* CAN1 IO mux M0 */
 RK_MUXROUTE_GRF(4, RK_PC3, 3, 0x0300, WRITE_MASK_VAL(2, 2, 1)), /* CAN1 IO mux M1 */
 RK_MUXROUTE_GRF(4, RK_PB5, 3, 0x0300, WRITE_MASK_VAL(4, 4, 0)), /* CAN2 IO mux M0 */
 RK_MUXROUTE_GRF(2, RK_PB2, 4, 0x0300, WRITE_MASK_VAL(4, 4, 1)), /* CAN2 IO mux M1 */
 RK_MUXROUTE_GRF(4, RK_PC4, 1, 0x0300, WRITE_MASK_VAL(6, 6, 0)), /* HPDIN IO mux M0 */
 RK_MUXROUTE_GRF(0, RK_PC2, 2, 0x0300, WRITE_MASK_VAL(6, 6, 1)), /* HPDIN IO mux M1 */
 RK_MUXROUTE_GRF(3, RK_PB1, 3, 0x0300, WRITE_MASK_VAL(8, 8, 0)), /* GMAC1 IO mux M0 */
 RK_MUXROUTE_GRF(4, RK_PA7, 3, 0x0300, WRITE_MASK_VAL(8, 8, 1)), /* GMAC1 IO mux M1 */
 RK_MUXROUTE_GRF(4, RK_PD1, 1, 0x0300, WRITE_MASK_VAL(10, 10, 0)), /* HDMITX IO mux M0 */
 RK_MUXROUTE_GRF(0, RK_PC7, 1, 0x0300, WRITE_MASK_VAL(10, 10, 1)), /* HDMITX IO mux M1 */
 RK_MUXROUTE_GRF(0, RK_PB6, 1, 0x0300, WRITE_MASK_VAL(14, 14, 0)), /* I2C2 IO mux M0 */
 RK_MUXROUTE_GRF(4, RK_PB4, 1, 0x0300, WRITE_MASK_VAL(14, 14, 1)), /* I2C2 IO mux M1 */
 RK_MUXROUTE_GRF(1, RK_PA0, 1, 0x0304, WRITE_MASK_VAL(0, 0, 0)), /* I2C3 IO mux M0 */
 RK_MUXROUTE_GRF(3, RK_PB6, 4, 0x0304, WRITE_MASK_VAL(0, 0, 1)), /* I2C3 IO mux M1 */
 RK_MUXROUTE_GRF(4, RK_PB2, 1, 0x0304, WRITE_MASK_VAL(2, 2, 0)), /* I2C4 IO mux M0 */
 RK_MUXROUTE_GRF(2, RK_PB1, 2, 0x0304, WRITE_MASK_VAL(2, 2, 1)), /* I2C4 IO mux M1 */
 RK_MUXROUTE_GRF(3, RK_PB4, 4, 0x0304, WRITE_MASK_VAL(4, 4, 0)), /* I2C5 IO mux M0 */
 RK_MUXROUTE_GRF(4, RK_PD0, 2, 0x0304, WRITE_MASK_VAL(4, 4, 1)), /* I2C5 IO mux M1 */
 RK_MUXROUTE_GRF(3, RK_PB1, 5, 0x0304, WRITE_MASK_VAL(14, 14, 0)), /* PWM8 IO mux M0 */
 RK_MUXROUTE_GRF(1, RK_PD5, 4, 0x0304, WRITE_MASK_VAL(14, 14, 1)), /* PWM8 IO mux M1 */
 RK_MUXROUTE_GRF(3, RK_PB2, 5, 0x0308, WRITE_MASK_VAL(0, 0, 0)), /* PWM9 IO mux M0 */
 RK_MUXROUTE_GRF(1, RK_PD6, 4, 0x0308, WRITE_MASK_VAL(0, 0, 1)), /* PWM9 IO mux M1 */
 RK_MUXROUTE_GRF(3, RK_PB5, 5, 0x0308, WRITE_MASK_VAL(2, 2, 0)), /* PWM10 IO mux M0 */
 RK_MUXROUTE_GRF(2, RK_PA1, 2, 0x0308, WRITE_MASK_VAL(2, 2, 1)), /* PWM10 IO mux M1 */
 RK_MUXROUTE_GRF(3, RK_PB6, 5, 0x0308, WRITE_MASK_VAL(4, 4, 0)), /* PWM11 IO mux M0 */
 RK_MUXROUTE_GRF(4, RK_PC0, 3, 0x0308, WRITE_MASK_VAL(4, 4, 1)), /* PWM11 IO mux M1 */
 RK_MUXROUTE_GRF(3, RK_PB7, 2, 0x0308, WRITE_MASK_VAL(6, 6, 0)), /* PWM12 IO mux M0 */
 RK_MUXROUTE_GRF(4, RK_PC5, 1, 0x0308, WRITE_MASK_VAL(6, 6, 1)), /* PWM12 IO mux M1 */
 RK_MUXROUTE_GRF(3, RK_PC0, 2, 0x0308, WRITE_MASK_VAL(8, 8, 0)), /* PWM13 IO mux M0 */
 RK_MUXROUTE_GRF(4, RK_PC6, 1, 0x0308, WRITE_MASK_VAL(8, 8, 1)), /* PWM13 IO mux M1 */
 RK_MUXROUTE_GRF(3, RK_PC4, 1, 0x0308, WRITE_MASK_VAL(10, 10, 0)), /* PWM14 IO mux M0 */
 RK_MUXROUTE_GRF(4, RK_PC2, 1, 0x0308, WRITE_MASK_VAL(10, 10, 1)), /* PWM14 IO mux M1 */
 RK_MUXROUTE_GRF(3, RK_PC5, 1, 0x0308, WRITE_MASK_VAL(12, 12, 0)), /* PWM15 IO mux M0 */
 RK_MUXROUTE_GRF(4, RK_PC3, 1, 0x0308, WRITE_MASK_VAL(12, 12, 1)), /* PWM15 IO mux M1 */
 RK_MUXROUTE_GRF(3, RK_PD2, 3, 0x0308, WRITE_MASK_VAL(14, 14, 0)), /* SDMMC2 IO mux M0 */
 RK_MUXROUTE_GRF(3, RK_PA5, 5, 0x0308, WRITE_MASK_VAL(14, 14, 1)), /* SDMMC2 IO mux M1 */
 RK_MUXROUTE_GRF(0, RK_PB5, 2, 0x030c, WRITE_MASK_VAL(0, 0, 0)), /* SPI0 IO mux M0 */
 RK_MUXROUTE_GRF(2, RK_PD3, 3, 0x030c, WRITE_MASK_VAL(0, 0, 1)), /* SPI0 IO mux M1 */
 RK_MUXROUTE_GRF(2, RK_PB5, 3, 0x030c, WRITE_MASK_VAL(2, 2, 0)), /* SPI1 IO mux M0 */
 RK_MUXROUTE_GRF(3, RK_PC3, 3, 0x030c, WRITE_MASK_VAL(2, 2, 1)), /* SPI1 IO mux M1 */
 RK_MUXROUTE_GRF(2, RK_PC1, 4, 0x030c, WRITE_MASK_VAL(4, 4, 0)), /* SPI2 IO mux M0 */
 RK_MUXROUTE_GRF(3, RK_PA0, 3, 0x030c, WRITE_MASK_VAL(4, 4, 1)), /* SPI2 IO mux M1 */
 RK_MUXROUTE_GRF(4, RK_PB3, 4, 0x030c, WRITE_MASK_VAL(6, 6, 0)), /* SPI3 IO mux M0 */
 RK_MUXROUTE_GRF(4, RK_PC2, 2, 0x030c, WRITE_MASK_VAL(6, 6, 1)), /* SPI3 IO mux M1 */
 RK_MUXROUTE_GRF(2, RK_PB4, 2, 0x030c, WRITE_MASK_VAL(8, 8, 0)), /* UART1 IO mux M0 */
 RK_MUXROUTE_GRF(3, RK_PD6, 4, 0x030c, WRITE_MASK_VAL(8, 8, 1)), /* UART1 IO mux M1 */
 RK_MUXROUTE_GRF(0, RK_PD1, 1, 0x030c, WRITE_MASK_VAL(10, 10, 0)), /* UART2 IO mux M0 */
 RK_MUXROUTE_GRF(1, RK_PD5, 2, 0x030c, WRITE_MASK_VAL(10, 10, 1)), /* UART2 IO mux M1 */
 RK_MUXROUTE_GRF(1, RK_PA1, 2, 0x030c, WRITE_MASK_VAL(12, 12, 0)), /* UART3 IO mux M0 */
 RK_MUXROUTE_GRF(3, RK_PB7, 4, 0x030c, WRITE_MASK_VAL(12, 12, 1)), /* UART3 IO mux M1 */
 RK_MUXROUTE_GRF(1, RK_PA6, 2, 0x030c, WRITE_MASK_VAL(14, 14, 0)), /* UART4 IO mux M0 */
 RK_MUXROUTE_GRF(3, RK_PB2, 4, 0x030c, WRITE_MASK_VAL(14, 14, 1)), /* UART4 IO mux M1 */
 RK_MUXROUTE_GRF(2, RK_PA2, 3, 0x0310, WRITE_MASK_VAL(0, 0, 0)), /* UART5 IO mux M0 */
 RK_MUXROUTE_GRF(3, RK_PC2, 4, 0x0310, WRITE_MASK_VAL(0, 0, 1)), /* UART5 IO mux M1 */
 RK_MUXROUTE_GRF(2, RK_PA4, 3, 0x0310, WRITE_MASK_VAL(2, 2, 0)), /* UART6 IO mux M0 */
 RK_MUXROUTE_GRF(1, RK_PD5, 3, 0x0310, WRITE_MASK_VAL(2, 2, 1)), /* UART6 IO mux M1 */
 RK_MUXROUTE_GRF(2, RK_PA6, 3, 0x0310, WRITE_MASK_VAL(5, 4, 0)), /* UART7 IO mux M0 */
 RK_MUXROUTE_GRF(3, RK_PC4, 4, 0x0310, WRITE_MASK_VAL(5, 4, 1)), /* UART7 IO mux M1 */
 RK_MUXROUTE_GRF(4, RK_PA2, 4, 0x0310, WRITE_MASK_VAL(5, 4, 2)), /* UART7 IO mux M2 */
 RK_MUXROUTE_GRF(2, RK_PC5, 3, 0x0310, WRITE_MASK_VAL(6, 6, 0)), /* UART8 IO mux M0 */
 RK_MUXROUTE_GRF(2, RK_PD7, 4, 0x0310, WRITE_MASK_VAL(6, 6, 1)), /* UART8 IO mux M1 */
 RK_MUXROUTE_GRF(2, RK_PB0, 3, 0x0310, WRITE_MASK_VAL(9, 8, 0)), /* UART9 IO mux M0 */
 RK_MUXROUTE_GRF(4, RK_PC5, 4, 0x0310, WRITE_MASK_VAL(9, 8, 1)), /* UART9 IO mux M1 */
 RK_MUXROUTE_GRF(4, RK_PA4, 4, 0x0310, WRITE_MASK_VAL(9, 8, 2)), /* UART9 IO mux M2 */
 RK_MUXROUTE_GRF(1, RK_PA2, 1, 0x0310, WRITE_MASK_VAL(11, 10, 0)), /* I2S1 IO mux M0 */
 RK_MUXROUTE_GRF(3, RK_PC6, 4, 0x0310, WRITE_MASK_VAL(11, 10, 1)), /* I2S1 IO mux M1 */
 RK_MUXROUTE_GRF(2, RK_PD0, 5, 0x0310, WRITE_MASK_VAL(11, 10, 2)), /* I2S1 IO mux M2 */
 RK_MUXROUTE_GRF(2, RK_PC1, 1, 0x0310, WRITE_MASK_VAL(12, 12, 0)), /* I2S2 IO mux M0 */
 RK_MUXROUTE_GRF(4, RK_PB6, 5, 0x0310, WRITE_MASK_VAL(12, 12, 1)), /* I2S2 IO mux M1 */
 RK_MUXROUTE_GRF(3, RK_PA2, 4, 0x0310, WRITE_MASK_VAL(14, 14, 0)), /* I2S3 IO mux M0 */
 RK_MUXROUTE_GRF(4, RK_PC2, 5, 0x0310, WRITE_MASK_VAL(14, 14, 1)), /* I2S3 IO mux M1 */
 RK_MUXROUTE_GRF(1, RK_PA4, 3, 0x0314, WRITE_MASK_VAL(1, 0, 0)), /* PDM IO mux M0 */
 RK_MUXROUTE_GRF(1, RK_PA6, 3, 0x0314, WRITE_MASK_VAL(1, 0, 0)), /* PDM IO mux M0 */
 RK_MUXROUTE_GRF(3, RK_PD6, 5, 0x0314, WRITE_MASK_VAL(1, 0, 1)), /* PDM IO mux M1 */
 RK_MUXROUTE_GRF(4, RK_PA0, 4, 0x0314, WRITE_MASK_VAL(1, 0, 1)), /* PDM IO mux M1 */
 RK_MUXROUTE_GRF(3, RK_PC4, 5, 0x0314, WRITE_MASK_VAL(1, 0, 2)), /* PDM IO mux M2 */
 RK_MUXROUTE_GRF(0, RK_PA5, 3, 0x0314, WRITE_MASK_VAL(3, 2, 0)), /* PCIE20 IO mux M0 */
 RK_MUXROUTE_GRF(2, RK_PD0, 4, 0x0314, WRITE_MASK_VAL(3, 2, 1)), /* PCIE20 IO mux M1 */
 RK_MUXROUTE_GRF(1, RK_PB0, 4, 0x0314, WRITE_MASK_VAL(3, 2, 2)), /* PCIE20 IO mux M2 */
 RK_MUXROUTE_GRF(0, RK_PA4, 3, 0x0314, WRITE_MASK_VAL(5, 4, 0)), /* PCIE30X1 IO mux M0 */
 RK_MUXROUTE_GRF(2, RK_PD2, 4, 0x0314, WRITE_MASK_VAL(5, 4, 1)), /* PCIE30X1 IO mux M1 */
 RK_MUXROUTE_GRF(1, RK_PA5, 4, 0x0314, WRITE_MASK_VAL(5, 4, 2)), /* PCIE30X1 IO mux M2 */
 RK_MUXROUTE_GRF(0, RK_PA6, 2, 0x0314, WRITE_MASK_VAL(7, 6, 0)), /* PCIE30X2 IO mux M0 */
 RK_MUXROUTE_GRF(2, RK_PD4, 4, 0x0314, WRITE_MASK_VAL(7, 6, 1)), /* PCIE30X2 IO mux M1 */
 RK_MUXROUTE_GRF(4, RK_PC2, 4, 0x0314, WRITE_MASK_VAL(7, 6, 2)), /* PCIE30X2 IO mux M2 */
};

static bool rockchip_get_mux_route(struct rockchip_pin_bank *bank, int pin,
       int mux, u32 *loc, u32 *reg, u32 *value)
{
 struct rockchip_pinctrl *info = bank->drvdata;
 struct rockchip_pin_ctrl *ctrl = info->ctrl;
 struct rockchip_mux_route_data *data;
 int i;

 for (i = 0; i < ctrl->niomux_routes; i++) {
  data = &ctrl->iomux_routes[i];
  if ((data->bank_num == bank->bank_num) &&
      (data->pin == pin) && (data->func == mux))
   break;
 }

 if (i >= ctrl->niomux_routes)
  return false;

 *loc = data->route_location;
 *reg = data->route_offset;
 *value = data->route_val;

 return true;
}

static int rockchip_get_mux(struct rockchip_pin_bank *bank, int pin)
{
 struct rockchip_pinctrl *info = bank->drvdata;
 struct rockchip_pin_ctrl *ctrl = info->ctrl;
 int iomux_num = (pin / 8);
 struct regmap *regmap;
 unsigned int val;
 int reg, ret, mask, mux_type;
 u8 bit;

 if (iomux_num > 3)
  return -EINVAL;

 if (bank->iomux[iomux_num].type & IOMUX_UNROUTED) {
  dev_err(info->dev, "pin %d is unrouted\n", pin);
  return -EINVAL;
 }

 if (bank->iomux[iomux_num].type & IOMUX_GPIO_ONLY)
  return RK_FUNC_GPIO;

 if (bank->iomux[iomux_num].type & IOMUX_SOURCE_PMU)
  regmap = info->regmap_pmu;
 else if (bank->iomux[iomux_num].type & IOMUX_L_SOURCE_PMU)
  regmap = (pin % 8 < 4) ? info->regmap_pmu : info->regmap_base;
 else
  regmap = info->regmap_base;

 /* get basic quadrupel of mux registers and the correct reg inside */
 mux_type = bank->iomux[iomux_num].type;
 reg = bank->iomux[iomux_num].offset;
 if (mux_type & IOMUX_WIDTH_4BIT) {
  if ((pin % 8) >= 4)
   reg += 0x4;
  bit = (pin % 4) * 4;
  mask = 0xf;
 } else if (mux_type & IOMUX_WIDTH_3BIT) {
  if ((pin % 8) >= 5)
   reg += 0x4;
  bit = (pin % 8 % 5) * 3;
  mask = 0x7;
 } else {
  bit = (pin % 8) * 2;
  mask = 0x3;
 }

 if (bank->recalced_mask & BIT(pin))
  rockchip_get_recalced_mux(bank, pin, ®, &bit, &mask);

 if (ctrl->type == RK3576) {
  if ((bank->bank_num == 0) && (pin >= RK_PB4) && (pin <= RK_PB7))
   reg += 0x1ff4; /* GPIO0_IOC_GPIO0B_IOMUX_SEL_H */
 }

 if (ctrl->type == RK3588) {
  if (bank->bank_num == 0) {
   if ((pin >= RK_PB4) && (pin <= RK_PD7)) {
    u32 reg0 = 0;

    reg0 = reg + 0x4000 - 0xC; /* PMU2_IOC_BASE */
    ret = regmap_read(regmap, reg0, &val);
    if (ret)
     return ret;

    if (!(val & BIT(8)))
     return ((val >> bit) & mask);

    reg = reg + 0x8000; /* BUS_IOC_BASE */
    regmap = info->regmap_base;
   }
  } else if (bank->bank_num > 0) {
   reg += 0x8000; /* BUS_IOC_BASE */
  }
 }

 ret = regmap_read(regmap, reg, &val);
 if (ret)
  return ret;

 return ((val >> bit) & mask);
}

static int rockchip_verify_mux(struct rockchip_pin_bank *bank,
          int pin, int mux)
{
 struct rockchip_pinctrl *info = bank->drvdata;
 struct device *dev = info->dev;
 int iomux_num = (pin / 8);

 if (iomux_num > 3)
  return -EINVAL;

 if (bank->iomux[iomux_num].type & IOMUX_UNROUTED) {
  dev_err(dev, "pin %d is unrouted\n", pin);
  return -EINVAL;
 }

 if (bank->iomux[iomux_num].type & IOMUX_GPIO_ONLY) {
  if (mux != RK_FUNC_GPIO) {
   dev_err(dev, "pin %d only supports a gpio mux\n", pin);
   return -ENOTSUPP;
  }
 }

 return 0;
}

/*
 * Set a new mux function for a pin.
 *
 * The register is divided into the upper and lower 16 bit. When changing
 * a value, the previous register value is not read and changed. Instead
 * it seems the changed bits are marked in the upper 16 bit, while the
 * changed value gets set in the same offset in the lower 16 bit.
 * All pin settings seem to be 2 bit wide in both the upper and lower
 * parts.
 * @bank: pin bank to change
 * @pin: pin to change
 * @mux: new mux function to set
 */

static int rockchip_set_mux(struct rockchip_pin_bank *bank, int pin, int mux)
{
 struct rockchip_pinctrl *info = bank->drvdata;
 struct rockchip_pin_ctrl *ctrl = info->ctrl;
 struct device *dev = info->dev;
 int iomux_num = (pin / 8);
 struct regmap *regmap;
 int reg, ret, mask, mux_type;
 u8 bit;
 u32 data, rmask, route_location, route_reg, route_val;

 ret = rockchip_verify_mux(bank, pin, mux);
 if (ret < 0)
  return ret;

 if (bank->iomux[iomux_num].type & IOMUX_GPIO_ONLY)
  return 0;

 dev_dbg(dev, "setting mux of GPIO%d-%d to %d\n", bank->bank_num, pin, mux);

 if (bank->iomux[iomux_num].type & IOMUX_SOURCE_PMU)
  regmap = info->regmap_pmu;
 else if (bank->iomux[iomux_num].type & IOMUX_L_SOURCE_PMU)
  regmap = (pin % 8 < 4) ? info->regmap_pmu : info->regmap_base;
 else
  regmap = info->regmap_base;

 /* get basic quadrupel of mux registers and the correct reg inside */
 mux_type = bank->iomux[iomux_num].type;
 reg = bank->iomux[iomux_num].offset;
 if (mux_type & IOMUX_WIDTH_4BIT) {
  if ((pin % 8) >= 4)
   reg += 0x4;
  bit = (pin % 4) * 4;
  mask = 0xf;
 } else if (mux_type & IOMUX_WIDTH_3BIT) {
  if ((pin % 8) >= 5)
   reg += 0x4;
  bit = (pin % 8 % 5) * 3;
  mask = 0x7;
 } else {
  bit = (pin % 8) * 2;
  mask = 0x3;
 }

 if (bank->recalced_mask & BIT(pin))
  rockchip_get_recalced_mux(bank, pin, ®, &bit, &mask);

 if (ctrl->type == RK3576) {
  if ((bank->bank_num == 0) && (pin >= RK_PB4) && (pin <= RK_PB7))
   reg += 0x1ff4; /* GPIO0_IOC_GPIO0B_IOMUX_SEL_H */
 }

 if (ctrl->type == RK3588) {
  if (bank->bank_num == 0) {
   if ((pin >= RK_PB4) && (pin <= RK_PD7)) {
    if (mux < 8) {
     reg += 0x4000 - 0xC; /* PMU2_IOC_BASE */
     data = (mask << (bit + 16));
     rmask = data | (data >> 16);
     data |= (mux & mask) << bit;
     ret = regmap_update_bits(regmap, reg, rmask, data);
    } else {
     u32 reg0 = 0;

     reg0 = reg + 0x4000 - 0xC; /* PMU2_IOC_BASE */
     data = (mask << (bit + 16));
     rmask = data | (data >> 16);
     data |= 8 << bit;
     ret = regmap_update_bits(regmap, reg0, rmask, data);

     reg0 = reg + 0x8000; /* BUS_IOC_BASE */
     data = (mask << (bit + 16));
     rmask = data | (data >> 16);
     data |= mux << bit;
     regmap = info->regmap_base;
     ret |= regmap_update_bits(regmap, reg0, rmask, data);
    }
   } else {
    data = (mask << (bit + 16));
    rmask = data | (data >> 16);
    data |= (mux & mask) << bit;
    ret = regmap_update_bits(regmap, reg, rmask, data);
   }
   return ret;
  } else if (bank->bank_num > 0) {
   reg += 0x8000; /* BUS_IOC_BASE */
  }
 }

 if (mux > mask)
  return -EINVAL;

 if (bank->route_mask & BIT(pin)) {
  if (rockchip_get_mux_route(bank, pin, mux, &route_location,
        &route_reg, &route_val)) {
   struct regmap *route_regmap = regmap;

   /* handle special locations */
   switch (route_location) {
   case ROCKCHIP_ROUTE_PMU:
    route_regmap = info->regmap_pmu;
    break;
   case ROCKCHIP_ROUTE_GRF:
    route_regmap = info->regmap_base;
    break;
   }

   ret = regmap_write(route_regmap, route_reg, route_val);
   if (ret)
    return ret;
  }
 }

 data = (mask << (bit + 16));
 rmask = data | (data >> 16);
 data |= (mux & mask) << bit;
 ret = regmap_update_bits(regmap, reg, rmask, data);

 return ret;
}

#define PX30_PULL_PMU_OFFSET  0x10
#define PX30_PULL_GRF_OFFSET  0x60
#define PX30_PULL_BITS_PER_PIN  2
#define PX30_PULL_PINS_PER_REG  8
#define PX30_PULL_BANK_STRIDE  16

static int px30_calc_pull_reg_and_bit(struct rockchip_pin_bank *bank,
          int pin_num, struct regmap **regmap,
          int *reg, u8 *bit)
{
 struct rockchip_pinctrl *info = bank->drvdata;

 /* The first 32 pins of the first bank are located in PMU */
 if (bank->bank_num == 0) {
  *regmap = info->regmap_pmu;
  *reg = PX30_PULL_PMU_OFFSET;
 } else {
  *regmap = info->regmap_base;
  *reg = PX30_PULL_GRF_OFFSET;

  /* correct the offset, as we're starting with the 2nd bank */
  *reg -= 0x10;
  *reg += bank->bank_num * PX30_PULL_BANK_STRIDE;
 }

 *reg += ((pin_num / PX30_PULL_PINS_PER_REG) * 4);
 *bit = (pin_num % PX30_PULL_PINS_PER_REG);
 *bit *= PX30_PULL_BITS_PER_PIN;

 return 0;
}

#define PX30_DRV_PMU_OFFSET  0x20
#define PX30_DRV_GRF_OFFSET  0xf0
#define PX30_DRV_BITS_PER_PIN  2
#define PX30_DRV_PINS_PER_REG  8
#define PX30_DRV_BANK_STRIDE  16

static int px30_calc_drv_reg_and_bit(struct rockchip_pin_bank *bank,
         int pin_num, struct regmap **regmap,
         int *reg, u8 *bit)
{
 struct rockchip_pinctrl *info = bank->drvdata;

 /* The first 32 pins of the first bank are located in PMU */
 if (bank->bank_num == 0) {
  *regmap = info->regmap_pmu;
  *reg = PX30_DRV_PMU_OFFSET;
 } else {
  *regmap = info->regmap_base;
  *reg = PX30_DRV_GRF_OFFSET;

  /* correct the offset, as we're starting with the 2nd bank */
  *reg -= 0x10;
  *reg += bank->bank_num * PX30_DRV_BANK_STRIDE;
 }

 *reg += ((pin_num / PX30_DRV_PINS_PER_REG) * 4);
 *bit = (pin_num % PX30_DRV_PINS_PER_REG);
 *bit *= PX30_DRV_BITS_PER_PIN;

 return 0;
}

#define PX30_SCHMITT_PMU_OFFSET   0x38
#define PX30_SCHMITT_GRF_OFFSET   0xc0
#define PX30_SCHMITT_PINS_PER_PMU_REG  16
#define PX30_SCHMITT_BANK_STRIDE  16
#define PX30_SCHMITT_PINS_PER_GRF_REG  8

static int px30_calc_schmitt_reg_and_bit(struct rockchip_pin_bank *bank,
      int pin_num,
      struct regmap **regmap,
      int *reg, u8 *bit)
{
 struct rockchip_pinctrl *info = bank->drvdata;
 int pins_per_reg;

 if (bank->bank_num == 0) {
  *regmap = info->regmap_pmu;
  *reg = PX30_SCHMITT_PMU_OFFSET;
  pins_per_reg = PX30_SCHMITT_PINS_PER_PMU_REG;
 } else {
  *regmap = info->regmap_base;
  *reg = PX30_SCHMITT_GRF_OFFSET;
  pins_per_reg = PX30_SCHMITT_PINS_PER_GRF_REG;
  *reg += (bank->bank_num  - 1) * PX30_SCHMITT_BANK_STRIDE;
 }

 *reg += ((pin_num / pins_per_reg) * 4);
 *bit = pin_num % pins_per_reg;

 return 0;
}

#define RV1108_PULL_PMU_OFFSET  0x10
#define RV1108_PULL_OFFSET  0x110
#define RV1108_PULL_PINS_PER_REG 8
#define RV1108_PULL_BITS_PER_PIN 2
#define RV1108_PULL_BANK_STRIDE  16

static int rv1108_calc_pull_reg_and_bit(struct rockchip_pin_bank *bank,
     int pin_num, struct regmap **regmap,
     int *reg, u8 *bit)
{
 struct rockchip_pinctrl *info = bank->drvdata;

 /* The first 24 pins of the first bank are located in PMU */
 if (bank->bank_num == 0) {
  *regmap = info->regmap_pmu;
  *reg = RV1108_PULL_PMU_OFFSET;
 } else {
  *reg = RV1108_PULL_OFFSET;
  *regmap = info->regmap_base;
  /* correct the offset, as we're starting with the 2nd bank */
  *reg -= 0x10;
  *reg += bank->bank_num * RV1108_PULL_BANK_STRIDE;
 }

 *reg += ((pin_num / RV1108_PULL_PINS_PER_REG) * 4);
 *bit = (pin_num % RV1108_PULL_PINS_PER_REG);
 *bit *= RV1108_PULL_BITS_PER_PIN;

 return 0;
}

#define RV1108_DRV_PMU_OFFSET  0x20
#define RV1108_DRV_GRF_OFFSET  0x210
#define RV1108_DRV_BITS_PER_PIN  2
#define RV1108_DRV_PINS_PER_REG  8
#define RV1108_DRV_BANK_STRIDE  16

static int rv1108_calc_drv_reg_and_bit(struct rockchip_pin_bank *bank,
           int pin_num, struct regmap **regmap,
           int *reg, u8 *bit)
{
 struct rockchip_pinctrl *info = bank->drvdata;

 /* The first 24 pins of the first bank are located in PMU */
 if (bank->bank_num == 0) {
  *regmap = info->regmap_pmu;
  *reg = RV1108_DRV_PMU_OFFSET;
 } else {
  *regmap = info->regmap_base;
  *reg = RV1108_DRV_GRF_OFFSET;

  /* correct the offset, as we're starting with the 2nd bank */
  *reg -= 0x10;
  *reg += bank->bank_num * RV1108_DRV_BANK_STRIDE;
 }

 *reg += ((pin_num / RV1108_DRV_PINS_PER_REG) * 4);
 *bit = pin_num % RV1108_DRV_PINS_PER_REG;
 *bit *= RV1108_DRV_BITS_PER_PIN;

 return 0;
}

#define RV1108_SCHMITT_PMU_OFFSET  0x30
#define RV1108_SCHMITT_GRF_OFFSET  0x388
#define RV1108_SCHMITT_BANK_STRIDE  8
#define RV1108_SCHMITT_PINS_PER_GRF_REG  16
#define RV1108_SCHMITT_PINS_PER_PMU_REG  8

static int rv1108_calc_schmitt_reg_and_bit(struct rockchip_pin_bank *bank,
        int pin_num,
        struct regmap **regmap,
        int *reg, u8 *bit)
{
 struct rockchip_pinctrl *info = bank->drvdata;
 int pins_per_reg;

 if (bank->bank_num == 0) {
  *regmap = info->regmap_pmu;
  *reg = RV1108_SCHMITT_PMU_OFFSET;
  pins_per_reg = RV1108_SCHMITT_PINS_PER_PMU_REG;
 } else {
  *regmap = info->regmap_base;
  *reg = RV1108_SCHMITT_GRF_OFFSET;
  pins_per_reg = RV1108_SCHMITT_PINS_PER_GRF_REG;
  *reg += (bank->bank_num  - 1) * RV1108_SCHMITT_BANK_STRIDE;
 }
 *reg += ((pin_num / pins_per_reg) * 4);
 *bit = pin_num % pins_per_reg;

 return 0;
}

#define RV1126_PULL_PMU_OFFSET  0x40
#define RV1126_PULL_GRF_GPIO1A0_OFFSET 0x10108
#define RV1126_PULL_PINS_PER_REG 8
#define RV1126_PULL_BITS_PER_PIN 2
#define RV1126_PULL_BANK_STRIDE  16
#define RV1126_GPIO_C4_D7(p)  (p >= 20 && p <= 31) /* GPIO0_C4 ~ GPIO0_D7 */

static int rv1126_calc_pull_reg_and_bit(struct rockchip_pin_bank *bank,
     int pin_num, struct regmap **regmap,
     int *reg, u8 *bit)
{
 struct rockchip_pinctrl *info = bank->drvdata;

 /* The first 24 pins of the first bank are located in PMU */
 if (bank->bank_num == 0) {
  if (RV1126_GPIO_C4_D7(pin_num)) {
   *regmap = info->regmap_base;
   *reg = RV1126_PULL_GRF_GPIO1A0_OFFSET;
   *reg -= (((31 - pin_num) / RV1126_PULL_PINS_PER_REG + 1) * 4);
   *bit = pin_num % RV1126_PULL_PINS_PER_REG;
   *bit *= RV1126_PULL_BITS_PER_PIN;
   return 0;
  }
  *regmap = info->regmap_pmu;
  *reg = RV1126_PULL_PMU_OFFSET;
 } else {
  *reg = RV1126_PULL_GRF_GPIO1A0_OFFSET;
  *regmap = info->regmap_base;
  *reg += (bank->bank_num - 1) * RV1126_PULL_BANK_STRIDE;
 }

 *reg += ((pin_num / RV1126_PULL_PINS_PER_REG) * 4);
 *bit = (pin_num % RV1126_PULL_PINS_PER_REG);
 *bit *= RV1126_PULL_BITS_PER_PIN;

 return 0;
}

#define RV1126_DRV_PMU_OFFSET  0x20
#define RV1126_DRV_GRF_GPIO1A0_OFFSET 0x10090
#define RV1126_DRV_BITS_PER_PIN  4
#define RV1126_DRV_PINS_PER_REG  4
#define RV1126_DRV_BANK_STRIDE  32

static int rv1126_calc_drv_reg_and_bit(struct rockchip_pin_bank *bank,
           int pin_num, struct regmap **regmap,
           int *reg, u8 *bit)
{
 struct rockchip_pinctrl *info = bank->drvdata;

 /* The first 24 pins of the first bank are located in PMU */
 if (bank->bank_num == 0) {
  if (RV1126_GPIO_C4_D7(pin_num)) {
   *regmap = info->regmap_base;
   *reg = RV1126_DRV_GRF_GPIO1A0_OFFSET;
   *reg -= (((31 - pin_num) / RV1126_DRV_PINS_PER_REG + 1) * 4);
   *reg -= 0x4;
   *bit = pin_num % RV1126_DRV_PINS_PER_REG;
   *bit *= RV1126_DRV_BITS_PER_PIN;
   return 0;
  }
  *regmap = info->regmap_pmu;
  *reg = RV1126_DRV_PMU_OFFSET;
 } else {
  *regmap = info->regmap_base;
  *reg = RV1126_DRV_GRF_GPIO1A0_OFFSET;
  *reg += (bank->bank_num - 1) * RV1126_DRV_BANK_STRIDE;
 }

 *reg += ((pin_num / RV1126_DRV_PINS_PER_REG) * 4);
 *bit = pin_num % RV1126_DRV_PINS_PER_REG;
 *bit *= RV1126_DRV_BITS_PER_PIN;

 return 0;
}

#define RV1126_SCHMITT_PMU_OFFSET  0x60
#define RV1126_SCHMITT_GRF_GPIO1A0_OFFSET 0x10188
#define RV1126_SCHMITT_BANK_STRIDE  16
#define RV1126_SCHMITT_PINS_PER_GRF_REG  8
#define RV1126_SCHMITT_PINS_PER_PMU_REG  8

static int rv1126_calc_schmitt_reg_and_bit(struct rockchip_pin_bank *bank,
        int pin_num,
        struct regmap **regmap,
        int *reg, u8 *bit)
{
 struct rockchip_pinctrl *info = bank->drvdata;
 int pins_per_reg;

 if (bank->bank_num == 0) {
  if (RV1126_GPIO_C4_D7(pin_num)) {
   *regmap = info->regmap_base;
   *reg = RV1126_SCHMITT_GRF_GPIO1A0_OFFSET;
   *reg -= (((31 - pin_num) / RV1126_SCHMITT_PINS_PER_GRF_REG + 1) * 4);
   *bit = pin_num % RV1126_SCHMITT_PINS_PER_GRF_REG;
   return 0;
  }
  *regmap = info->regmap_pmu;
  *reg = RV1126_SCHMITT_PMU_OFFSET;
  pins_per_reg = RV1126_SCHMITT_PINS_PER_PMU_REG;
 } else {
  *regmap = info->regmap_base;
  *reg = RV1126_SCHMITT_GRF_GPIO1A0_OFFSET;
  pins_per_reg = RV1126_SCHMITT_PINS_PER_GRF_REG;
  *reg += (bank->bank_num - 1) * RV1126_SCHMITT_BANK_STRIDE;
 }
 *reg += ((pin_num / pins_per_reg) * 4);
 *bit = pin_num % pins_per_reg;

 return 0;
}

#define RK3308_SCHMITT_PINS_PER_REG  8
#define RK3308_SCHMITT_BANK_STRIDE  16
#define RK3308_SCHMITT_GRF_OFFSET  0x1a0

static int rk3308_calc_schmitt_reg_and_bit(struct rockchip_pin_bank *bank,
        int pin_num, struct regmap **regmap,
        int *reg, u8 *bit)
{
 struct rockchip_pinctrl *info = bank->drvdata;

 *regmap = info->regmap_base;
 *reg = RK3308_SCHMITT_GRF_OFFSET;

 *reg += bank->bank_num * RK3308_SCHMITT_BANK_STRIDE;
 *reg += ((pin_num / RK3308_SCHMITT_PINS_PER_REG) * 4);
 *bit = pin_num % RK3308_SCHMITT_PINS_PER_REG;

 return 0;
}

#define RK2928_PULL_OFFSET  0x118
#define RK2928_PULL_PINS_PER_REG 16
#define RK2928_PULL_BANK_STRIDE  8

static int rk2928_calc_pull_reg_and_bit(struct rockchip_pin_bank *bank,
     int pin_num, struct regmap **regmap,
     int *reg, u8 *bit)
{
 struct rockchip_pinctrl *info = bank->drvdata;

 *regmap = info->regmap_base;
 *reg = RK2928_PULL_OFFSET;
 *reg += bank->bank_num * RK2928_PULL_BANK_STRIDE;
 *reg += (pin_num / RK2928_PULL_PINS_PER_REG) * 4;

 *bit = pin_num % RK2928_PULL_PINS_PER_REG;

 return 0;
};

#define RK3128_PULL_OFFSET 0x118

static int rk3128_calc_pull_reg_and_bit(struct rockchip_pin_bank *bank,
     int pin_num, struct regmap **regmap,
     int *reg, u8 *bit)
{
 struct rockchip_pinctrl *info = bank->drvdata;

 *regmap = info->regmap_base;
 *reg = RK3128_PULL_OFFSET;
 *reg += bank->bank_num * RK2928_PULL_BANK_STRIDE;
 *reg += ((pin_num / RK2928_PULL_PINS_PER_REG) * 4);

 *bit = pin_num % RK2928_PULL_PINS_PER_REG;

 return 0;
}

#define RK3188_PULL_OFFSET  0x164
#define RK3188_PULL_BITS_PER_PIN 2
#define RK3188_PULL_PINS_PER_REG 8
#define RK3188_PULL_BANK_STRIDE  16
#define RK3188_PULL_PMU_OFFSET  0x64

static int rk3188_calc_pull_reg_and_bit(struct rockchip_pin_bank *bank,
     int pin_num, struct regmap **regmap,
     int *reg, u8 *bit)
{
 struct rockchip_pinctrl *info = bank->drvdata;

 /* The first 12 pins of the first bank are located elsewhere */
 if (bank->bank_num == 0 && pin_num < 12) {
  *regmap = info->regmap_pmu ? info->regmap_pmu
        : bank->regmap_pull;
  *reg = info->regmap_pmu ? RK3188_PULL_PMU_OFFSET : 0;
  *reg += ((pin_num / RK3188_PULL_PINS_PER_REG) * 4);
  *bit = pin_num % RK3188_PULL_PINS_PER_REG;
  *bit *= RK3188_PULL_BITS_PER_PIN;
 } else {
  *regmap = info->regmap_pull ? info->regmap_pull
         : info->regmap_base;
  *reg = info->regmap_pull ? 0 : RK3188_PULL_OFFSET;

  /* correct the offset, as it is the 2nd pull register */
  *reg -= 4;
  *reg += bank->bank_num * RK3188_PULL_BANK_STRIDE;
  *reg += ((pin_num / RK3188_PULL_PINS_PER_REG) * 4);

  /*
 * The bits in these registers have an inverse ordering
 * with the lowest pin being in bits 15:14 and the highest
 * pin in bits 1:0
 */

  *bit = 7 - (pin_num % RK3188_PULL_PINS_PER_REG);
  *bit *= RK3188_PULL_BITS_PER_PIN;
 }

 return 0;
}

#define RK3288_PULL_OFFSET  0x140
static int rk3288_calc_pull_reg_and_bit(struct rockchip_pin_bank *bank,
     int pin_num, struct regmap **regmap,
     int *reg, u8 *bit)
{
 struct rockchip_pinctrl *info = bank->drvdata;

 /* The first 24 pins of the first bank are located in PMU */
 if (bank->bank_num == 0) {
  *regmap = info->regmap_pmu;
  *reg = RK3188_PULL_PMU_OFFSET;

  *reg += ((pin_num / RK3188_PULL_PINS_PER_REG) * 4);
  *bit = pin_num % RK3188_PULL_PINS_PER_REG;
  *bit *= RK3188_PULL_BITS_PER_PIN;
 } else {
  *regmap = info->regmap_base;
  *reg = RK3288_PULL_OFFSET;

  /* correct the offset, as we're starting with the 2nd bank */
  *reg -= 0x10;
  *reg += bank->bank_num * RK3188_PULL_BANK_STRIDE;
  *reg += ((pin_num / RK3188_PULL_PINS_PER_REG) * 4);

  *bit = (pin_num % RK3188_PULL_PINS_PER_REG);
  *bit *= RK3188_PULL_BITS_PER_PIN;
 }

 return 0;
}

#define RK3288_DRV_PMU_OFFSET  0x70
#define RK3288_DRV_GRF_OFFSET  0x1c0
#define RK3288_DRV_BITS_PER_PIN  2
#define RK3288_DRV_PINS_PER_REG  8
#define RK3288_DRV_BANK_STRIDE  16

static int rk3288_calc_drv_reg_and_bit(struct rockchip_pin_bank *bank,
           int pin_num, struct regmap **regmap,
           int *reg, u8 *bit)
{
 struct rockchip_pinctrl *info = bank->drvdata;

 /* The first 24 pins of the first bank are located in PMU */
 if (bank->bank_num == 0) {
  *regmap = info->regmap_pmu;
  *reg = RK3288_DRV_PMU_OFFSET;

  *reg += ((pin_num / RK3288_DRV_PINS_PER_REG) * 4);
  *bit = pin_num % RK3288_DRV_PINS_PER_REG;
  *bit *= RK3288_DRV_BITS_PER_PIN;
 } else {
  *regmap = info->regmap_base;
  *reg = RK3288_DRV_GRF_OFFSET;

  /* correct the offset, as we're starting with the 2nd bank */
  *reg -= 0x10;
  *reg += bank->bank_num * RK3288_DRV_BANK_STRIDE;
  *reg += ((pin_num / RK3288_DRV_PINS_PER_REG) * 4);

  *bit = (pin_num % RK3288_DRV_PINS_PER_REG);
  *bit *= RK3288_DRV_BITS_PER_PIN;
 }

 return 0;
}

#define RK3228_PULL_OFFSET  0x100

static int rk3228_calc_pull_reg_and_bit(struct rockchip_pin_bank *bank,
     int pin_num, struct regmap **regmap,
     int *reg, u8 *bit)
{
 struct rockchip_pinctrl *info = bank->drvdata;

 *regmap = info->regmap_base;
 *reg = RK3228_PULL_OFFSET;
 *reg += bank->bank_num * RK3188_PULL_BANK_STRIDE;
 *reg += ((pin_num / RK3188_PULL_PINS_PER_REG) * 4);

 *bit = (pin_num % RK3188_PULL_PINS_PER_REG);
 *bit *= RK3188_PULL_BITS_PER_PIN;

 return 0;
}

#define RK3228_DRV_GRF_OFFSET  0x200

static int rk3228_calc_drv_reg_and_bit(struct rockchip_pin_bank *bank,
           int pin_num, struct regmap **regmap,
           int *reg, u8 *bit)
{
 struct rockchip_pinctrl *info = bank->drvdata;

 *regmap = info->regmap_base;
 *reg = RK3228_DRV_GRF_OFFSET;
 *reg += bank->bank_num * RK3288_DRV_BANK_STRIDE;
 *reg += ((pin_num / RK3288_DRV_PINS_PER_REG) * 4);

 *bit = (pin_num % RK3288_DRV_PINS_PER_REG);
 *bit *= RK3288_DRV_BITS_PER_PIN;

 return 0;
}

#define RK3308_PULL_OFFSET  0xa0

static int rk3308_calc_pull_reg_and_bit(struct rockchip_pin_bank *bank,
     int pin_num, struct regmap **regmap,
     int *reg, u8 *bit)
{
 struct rockchip_pinctrl *info = bank->drvdata;

 *regmap = info->regmap_base;
 *reg = RK3308_PULL_OFFSET;
 *reg += bank->bank_num * RK3188_PULL_BANK_STRIDE;
 *reg += ((pin_num / RK3188_PULL_PINS_PER_REG) * 4);

 *bit = (pin_num % RK3188_PULL_PINS_PER_REG);
 *bit *= RK3188_PULL_BITS_PER_PIN;

 return 0;
}

#define RK3308_DRV_GRF_OFFSET  0x100

static int rk3308_calc_drv_reg_and_bit(struct rockchip_pin_bank *bank,
           int pin_num, struct regmap **regmap,
           int *reg, u8 *bit)
{
 struct rockchip_pinctrl *info = bank->drvdata;

 *regmap = info->regmap_base;
 *reg = RK3308_DRV_GRF_OFFSET;
 *reg += bank->bank_num * RK3288_DRV_BANK_STRIDE;
 *reg += ((pin_num / RK3288_DRV_PINS_PER_REG) * 4);

 *bit = (pin_num % RK3288_DRV_PINS_PER_REG);
 *bit *= RK3288_DRV_BITS_PER_PIN;

 return 0;
}

#define RK3368_PULL_GRF_OFFSET  0x100
#define RK3368_PULL_PMU_OFFSET  0x10

static int rk3368_calc_pull_reg_and_bit(struct rockchip_pin_bank *bank,
     int pin_num, struct regmap **regmap,
     int *reg, u8 *bit)
{
 struct rockchip_pinctrl *info = bank->drvdata;

 /* The first 32 pins of the first bank are located in PMU */
 if (bank->bank_num == 0) {
  *regmap = info->regmap_pmu;
  *reg = RK3368_PULL_PMU_OFFSET;

  *reg += ((pin_num / RK3188_PULL_PINS_PER_REG) * 4);
  *bit = pin_num % RK3188_PULL_PINS_PER_REG;
  *bit *= RK3188_PULL_BITS_PER_PIN;
 } else {
  *regmap = info->regmap_base;
  *reg = RK3368_PULL_GRF_OFFSET;

  /* correct the offset, as we're starting with the 2nd bank */
  *reg -= 0x10;
  *reg += bank->bank_num * RK3188_PULL_BANK_STRIDE;
  *reg += ((pin_num / RK3188_PULL_PINS_PER_REG) * 4);

  *bit = (pin_num % RK3188_PULL_PINS_PER_REG);
  *bit *= RK3188_PULL_BITS_PER_PIN;
 }

 return 0;
}

#define RK3368_DRV_PMU_OFFSET  0x20
#define RK3368_DRV_GRF_OFFSET  0x200

static int rk3368_calc_drv_reg_and_bit(struct rockchip_pin_bank *bank,
           int pin_num, struct regmap **regmap,
           int *reg, u8 *bit)
{
 struct rockchip_pinctrl *info = bank->drvdata;

 /* The first 32 pins of the first bank are located in PMU */
 if (bank->bank_num == 0) {
  *regmap = info->regmap_pmu;
  *reg = RK3368_DRV_PMU_OFFSET;

  *reg += ((pin_num / RK3288_DRV_PINS_PER_REG) * 4);
  *bit = pin_num % RK3288_DRV_PINS_PER_REG;
  *bit *= RK3288_DRV_BITS_PER_PIN;
 } else {
  *regmap = info->regmap_base;
  *reg = RK3368_DRV_GRF_OFFSET;

  /* correct the offset, as we're starting with the 2nd bank */
  *reg -= 0x10;
  *reg += bank->bank_num * RK3288_DRV_BANK_STRIDE;
  *reg += ((pin_num / RK3288_DRV_PINS_PER_REG) * 4);

  *bit = (pin_num % RK3288_DRV_PINS_PER_REG);
  *bit *= RK3288_DRV_BITS_PER_PIN;
 }

 return 0;
}

#define RK3399_PULL_GRF_OFFSET  0xe040
#define RK3399_PULL_PMU_OFFSET  0x40
#define RK3399_DRV_3BITS_PER_PIN 3

static int rk3399_calc_pull_reg_and_bit(struct rockchip_pin_bank *bank,
     int pin_num, struct regmap **regmap,
     int *reg, u8 *bit)
{
 struct rockchip_pinctrl *info = bank->drvdata;

 /* The bank0:16 and bank1:32 pins are located in PMU */
 if ((bank->bank_num == 0) || (bank->bank_num == 1)) {
  *regmap = info->regmap_pmu;
  *reg = RK3399_PULL_PMU_OFFSET;

  *reg += bank->bank_num * RK3188_PULL_BANK_STRIDE;

  *reg += ((pin_num / RK3188_PULL_PINS_PER_REG) * 4);
  *bit = pin_num % RK3188_PULL_PINS_PER_REG;
  *bit *= RK3188_PULL_BITS_PER_PIN;
 } else {
  *regmap = info->regmap_base;
  *reg = RK3399_PULL_GRF_OFFSET;

  /* correct the offset, as we're starting with the 3rd bank */
  *reg -= 0x20;
  *reg += bank->bank_num * RK3188_PULL_BANK_STRIDE;
  *reg += ((pin_num / RK3188_PULL_PINS_PER_REG) * 4);

  *bit = (pin_num % RK3188_PULL_PINS_PER_REG);
  *bit *= RK3188_PULL_BITS_PER_PIN;
 }

 return 0;
}

static int rk3399_calc_drv_reg_and_bit(struct rockchip_pin_bank *bank,
           int pin_num, struct regmap **regmap,
           int *reg, u8 *bit)
{
 struct rockchip_pinctrl *info = bank->drvdata;
 int drv_num = (pin_num / 8);

 /*  The bank0:16 and bank1:32 pins are located in PMU */
 if ((bank->bank_num == 0) || (bank->bank_num == 1))
  *regmap = info->regmap_pmu;
 else
  *regmap = info->regmap_base;

 *reg = bank->drv[drv_num].offset;
 if ((bank->drv[drv_num].drv_type == DRV_TYPE_IO_1V8_3V0_AUTO) ||
     (bank->drv[drv_num].drv_type == DRV_TYPE_IO_3V3_ONLY))
  *bit = (pin_num % 8) * 3;
 else
  *bit = (pin_num % 8) * 2;

 return 0;
}

#define RK3528_DRV_BITS_PER_PIN  8
#define RK3528_DRV_PINS_PER_REG  2
#define RK3528_DRV_GPIO0_OFFSET  0x100
#define RK3528_DRV_GPIO1_OFFSET  0x20120
#define RK3528_DRV_GPIO2_OFFSET  0x30160
#define RK3528_DRV_GPIO3_OFFSET  0x20190
#define RK3528_DRV_GPIO4_OFFSET  0x101C0

static int rk3528_calc_drv_reg_and_bit(struct rockchip_pin_bank *bank,
           int pin_num, struct regmap **regmap,
           int *reg, u8 *bit)
{
 struct rockchip_pinctrl *info = bank->drvdata;

 *regmap = info->regmap_base;

 if (bank->bank_num == 0)
  *reg = RK3528_DRV_GPIO0_OFFSET;
 else if (bank->bank_num == 1)
  *reg = RK3528_DRV_GPIO1_OFFSET;
 else if (bank->bank_num == 2)
  *reg = RK3528_DRV_GPIO2_OFFSET;
 else if (bank->bank_num == 3)
  *reg = RK3528_DRV_GPIO3_OFFSET;
 else if (bank->bank_num == 4)
  *reg = RK3528_DRV_GPIO4_OFFSET;
 else
  dev_err(info->dev, "unsupported bank_num %d\n", bank->bank_num);

 *reg += ((pin_num / RK3528_DRV_PINS_PER_REG) * 4);
 *bit = pin_num % RK3528_DRV_PINS_PER_REG;
 *bit *= RK3528_DRV_BITS_PER_PIN;

 return 0;
}

#define RK3528_PULL_BITS_PER_PIN  2
#define RK3528_PULL_PINS_PER_REG  8
#define RK3528_PULL_GPIO0_OFFSET  0x200
#define RK3528_PULL_GPIO1_OFFSET  0x20210
#define RK3528_PULL_GPIO2_OFFSET  0x30220
#define RK3528_PULL_GPIO3_OFFSET  0x20230
#define RK3528_PULL_GPIO4_OFFSET  0x10240

static int rk3528_calc_pull_reg_and_bit(struct rockchip_pin_bank *bank,
     int pin_num, struct regmap **regmap,
     int *reg, u8 *bit)
{
 struct rockchip_pinctrl *info = bank->drvdata;

 *regmap = info->regmap_base;

 if (bank->bank_num == 0)
  *reg = RK3528_PULL_GPIO0_OFFSET;
 else if (bank->bank_num == 1)
  *reg = RK3528_PULL_GPIO1_OFFSET;
--> --------------------

--> maximum size reached

--> --------------------

Messung V0.5
C=94 H=94 G=93

¤ Dauer der Verarbeitung: 0.21 Sekunden  (vorverarbeitet)  ¤

*© Formatika GbR, Deutschland






Wurzel

Suchen

Beweissystem der NASA

Beweissystem Isabelle

NIST Cobol Testsuite

Cephes Mathematical Library

Wiener Entwicklungsmethode

Haftungshinweis

Die Informationen auf dieser Webseite wurden nach bestem Wissen sorgfältig zusammengestellt. Es wird jedoch weder Vollständigkeit, noch Richtigkeit, noch Qualität der bereit gestellten Informationen zugesichert.

Bemerkung:

Die farbliche Syntaxdarstellung und die Messung sind noch experimentell.