162306a36Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0-only
262306a36Sopenharmony_ci/*
362306a36Sopenharmony_ci * mt65xx pinctrl driver based on Allwinner A1X pinctrl driver.
462306a36Sopenharmony_ci * Copyright (c) 2014 MediaTek Inc.
562306a36Sopenharmony_ci * Author: Hongzhou.Yang <hongzhou.yang@mediatek.com>
662306a36Sopenharmony_ci */
762306a36Sopenharmony_ci
862306a36Sopenharmony_ci#include <linux/io.h>
962306a36Sopenharmony_ci#include <linux/gpio/driver.h>
1062306a36Sopenharmony_ci#include <linux/of.h>
1162306a36Sopenharmony_ci#include <linux/of_irq.h>
1262306a36Sopenharmony_ci#include <linux/pinctrl/consumer.h>
1362306a36Sopenharmony_ci#include <linux/pinctrl/machine.h>
1462306a36Sopenharmony_ci#include <linux/pinctrl/pinconf.h>
1562306a36Sopenharmony_ci#include <linux/pinctrl/pinconf-generic.h>
1662306a36Sopenharmony_ci#include <linux/pinctrl/pinctrl.h>
1762306a36Sopenharmony_ci#include <linux/pinctrl/pinmux.h>
1862306a36Sopenharmony_ci#include <linux/platform_device.h>
1962306a36Sopenharmony_ci#include <linux/slab.h>
2062306a36Sopenharmony_ci#include <linux/bitops.h>
2162306a36Sopenharmony_ci#include <linux/regmap.h>
2262306a36Sopenharmony_ci#include <linux/mfd/syscon.h>
2362306a36Sopenharmony_ci#include <linux/delay.h>
2462306a36Sopenharmony_ci#include <linux/interrupt.h>
2562306a36Sopenharmony_ci#include <linux/pm.h>
2662306a36Sopenharmony_ci#include <dt-bindings/pinctrl/mt65xx.h>
2762306a36Sopenharmony_ci
2862306a36Sopenharmony_ci#include "../core.h"
2962306a36Sopenharmony_ci#include "../pinconf.h"
3062306a36Sopenharmony_ci#include "../pinctrl-utils.h"
3162306a36Sopenharmony_ci#include "mtk-eint.h"
3262306a36Sopenharmony_ci#include "pinctrl-mtk-common.h"
3362306a36Sopenharmony_ci
3462306a36Sopenharmony_ci#define GPIO_MODE_BITS        3
3562306a36Sopenharmony_ci#define GPIO_MODE_PREFIX "GPIO"
3662306a36Sopenharmony_ci
3762306a36Sopenharmony_cistatic const char * const mtk_gpio_functions[] = {
3862306a36Sopenharmony_ci	"func0", "func1", "func2", "func3",
3962306a36Sopenharmony_ci	"func4", "func5", "func6", "func7",
4062306a36Sopenharmony_ci	"func8", "func9", "func10", "func11",
4162306a36Sopenharmony_ci	"func12", "func13", "func14", "func15",
4262306a36Sopenharmony_ci};
4362306a36Sopenharmony_ci
4462306a36Sopenharmony_ci/*
4562306a36Sopenharmony_ci * There are two base address for pull related configuration
4662306a36Sopenharmony_ci * in mt8135, and different GPIO pins use different base address.
4762306a36Sopenharmony_ci * When pin number greater than type1_start and less than type1_end,
4862306a36Sopenharmony_ci * should use the second base address.
4962306a36Sopenharmony_ci */
5062306a36Sopenharmony_cistatic struct regmap *mtk_get_regmap(struct mtk_pinctrl *pctl,
5162306a36Sopenharmony_ci		unsigned long pin)
5262306a36Sopenharmony_ci{
5362306a36Sopenharmony_ci	if (pin >= pctl->devdata->type1_start && pin < pctl->devdata->type1_end)
5462306a36Sopenharmony_ci		return pctl->regmap2;
5562306a36Sopenharmony_ci	return pctl->regmap1;
5662306a36Sopenharmony_ci}
5762306a36Sopenharmony_ci
5862306a36Sopenharmony_cistatic unsigned int mtk_get_port(struct mtk_pinctrl *pctl, unsigned long pin)
5962306a36Sopenharmony_ci{
6062306a36Sopenharmony_ci	/* Different SoC has different mask and port shift. */
6162306a36Sopenharmony_ci	return ((pin >> pctl->devdata->mode_shf) & pctl->devdata->port_mask)
6262306a36Sopenharmony_ci			<< pctl->devdata->port_shf;
6362306a36Sopenharmony_ci}
6462306a36Sopenharmony_ci
6562306a36Sopenharmony_cistatic int mtk_pmx_gpio_set_direction(struct pinctrl_dev *pctldev,
6662306a36Sopenharmony_ci			struct pinctrl_gpio_range *range, unsigned offset,
6762306a36Sopenharmony_ci			bool input)
6862306a36Sopenharmony_ci{
6962306a36Sopenharmony_ci	unsigned int reg_addr;
7062306a36Sopenharmony_ci	unsigned int bit;
7162306a36Sopenharmony_ci	struct mtk_pinctrl *pctl = pinctrl_dev_get_drvdata(pctldev);
7262306a36Sopenharmony_ci
7362306a36Sopenharmony_ci	reg_addr = mtk_get_port(pctl, offset) + pctl->devdata->dir_offset;
7462306a36Sopenharmony_ci	bit = BIT(offset & pctl->devdata->mode_mask);
7562306a36Sopenharmony_ci
7662306a36Sopenharmony_ci	if (pctl->devdata->spec_dir_set)
7762306a36Sopenharmony_ci		pctl->devdata->spec_dir_set(&reg_addr, offset);
7862306a36Sopenharmony_ci
7962306a36Sopenharmony_ci	if (input)
8062306a36Sopenharmony_ci		/* Different SoC has different alignment offset. */
8162306a36Sopenharmony_ci		reg_addr = CLR_ADDR(reg_addr, pctl);
8262306a36Sopenharmony_ci	else
8362306a36Sopenharmony_ci		reg_addr = SET_ADDR(reg_addr, pctl);
8462306a36Sopenharmony_ci
8562306a36Sopenharmony_ci	regmap_write(mtk_get_regmap(pctl, offset), reg_addr, bit);
8662306a36Sopenharmony_ci	return 0;
8762306a36Sopenharmony_ci}
8862306a36Sopenharmony_ci
8962306a36Sopenharmony_cistatic void mtk_gpio_set(struct gpio_chip *chip, unsigned offset, int value)
9062306a36Sopenharmony_ci{
9162306a36Sopenharmony_ci	unsigned int reg_addr;
9262306a36Sopenharmony_ci	unsigned int bit;
9362306a36Sopenharmony_ci	struct mtk_pinctrl *pctl = gpiochip_get_data(chip);
9462306a36Sopenharmony_ci
9562306a36Sopenharmony_ci	reg_addr = mtk_get_port(pctl, offset) + pctl->devdata->dout_offset;
9662306a36Sopenharmony_ci	bit = BIT(offset & pctl->devdata->mode_mask);
9762306a36Sopenharmony_ci
9862306a36Sopenharmony_ci	if (value)
9962306a36Sopenharmony_ci		reg_addr = SET_ADDR(reg_addr, pctl);
10062306a36Sopenharmony_ci	else
10162306a36Sopenharmony_ci		reg_addr = CLR_ADDR(reg_addr, pctl);
10262306a36Sopenharmony_ci
10362306a36Sopenharmony_ci	regmap_write(mtk_get_regmap(pctl, offset), reg_addr, bit);
10462306a36Sopenharmony_ci}
10562306a36Sopenharmony_ci
10662306a36Sopenharmony_cistatic int mtk_pconf_set_ies_smt(struct mtk_pinctrl *pctl, unsigned pin,
10762306a36Sopenharmony_ci		int value, enum pin_config_param arg)
10862306a36Sopenharmony_ci{
10962306a36Sopenharmony_ci	unsigned int reg_addr, offset;
11062306a36Sopenharmony_ci	unsigned int bit;
11162306a36Sopenharmony_ci
11262306a36Sopenharmony_ci	/**
11362306a36Sopenharmony_ci	 * Due to some soc are not support ies/smt config, add this special
11462306a36Sopenharmony_ci	 * control to handle it.
11562306a36Sopenharmony_ci	 */
11662306a36Sopenharmony_ci	if (!pctl->devdata->spec_ies_smt_set &&
11762306a36Sopenharmony_ci		pctl->devdata->ies_offset == MTK_PINCTRL_NOT_SUPPORT &&
11862306a36Sopenharmony_ci			arg == PIN_CONFIG_INPUT_ENABLE)
11962306a36Sopenharmony_ci		return -EINVAL;
12062306a36Sopenharmony_ci
12162306a36Sopenharmony_ci	if (!pctl->devdata->spec_ies_smt_set &&
12262306a36Sopenharmony_ci		pctl->devdata->smt_offset == MTK_PINCTRL_NOT_SUPPORT &&
12362306a36Sopenharmony_ci			arg == PIN_CONFIG_INPUT_SCHMITT_ENABLE)
12462306a36Sopenharmony_ci		return -EINVAL;
12562306a36Sopenharmony_ci
12662306a36Sopenharmony_ci	/*
12762306a36Sopenharmony_ci	 * Due to some pins are irregular, their input enable and smt
12862306a36Sopenharmony_ci	 * control register are discontinuous, so we need this special handle.
12962306a36Sopenharmony_ci	 */
13062306a36Sopenharmony_ci	if (pctl->devdata->spec_ies_smt_set) {
13162306a36Sopenharmony_ci		return pctl->devdata->spec_ies_smt_set(mtk_get_regmap(pctl, pin),
13262306a36Sopenharmony_ci			pctl->devdata, pin, value, arg);
13362306a36Sopenharmony_ci	}
13462306a36Sopenharmony_ci
13562306a36Sopenharmony_ci	if (arg == PIN_CONFIG_INPUT_ENABLE)
13662306a36Sopenharmony_ci		offset = pctl->devdata->ies_offset;
13762306a36Sopenharmony_ci	else
13862306a36Sopenharmony_ci		offset = pctl->devdata->smt_offset;
13962306a36Sopenharmony_ci
14062306a36Sopenharmony_ci	bit = BIT(offset & pctl->devdata->mode_mask);
14162306a36Sopenharmony_ci
14262306a36Sopenharmony_ci	if (value)
14362306a36Sopenharmony_ci		reg_addr = SET_ADDR(mtk_get_port(pctl, pin) + offset, pctl);
14462306a36Sopenharmony_ci	else
14562306a36Sopenharmony_ci		reg_addr = CLR_ADDR(mtk_get_port(pctl, pin) + offset, pctl);
14662306a36Sopenharmony_ci
14762306a36Sopenharmony_ci	regmap_write(mtk_get_regmap(pctl, pin), reg_addr, bit);
14862306a36Sopenharmony_ci	return 0;
14962306a36Sopenharmony_ci}
15062306a36Sopenharmony_ci
15162306a36Sopenharmony_ciint mtk_pconf_spec_set_ies_smt_range(struct regmap *regmap,
15262306a36Sopenharmony_ci		const struct mtk_pinctrl_devdata *devdata,
15362306a36Sopenharmony_ci		unsigned int pin, int value, enum pin_config_param arg)
15462306a36Sopenharmony_ci{
15562306a36Sopenharmony_ci	const struct mtk_pin_ies_smt_set *ies_smt_infos = NULL;
15662306a36Sopenharmony_ci	unsigned int i, info_num, reg_addr, bit;
15762306a36Sopenharmony_ci
15862306a36Sopenharmony_ci	switch (arg) {
15962306a36Sopenharmony_ci	case PIN_CONFIG_INPUT_ENABLE:
16062306a36Sopenharmony_ci		ies_smt_infos = devdata->spec_ies;
16162306a36Sopenharmony_ci		info_num = devdata->n_spec_ies;
16262306a36Sopenharmony_ci		break;
16362306a36Sopenharmony_ci	case PIN_CONFIG_INPUT_SCHMITT_ENABLE:
16462306a36Sopenharmony_ci		ies_smt_infos = devdata->spec_smt;
16562306a36Sopenharmony_ci		info_num = devdata->n_spec_smt;
16662306a36Sopenharmony_ci		break;
16762306a36Sopenharmony_ci	default:
16862306a36Sopenharmony_ci		break;
16962306a36Sopenharmony_ci	}
17062306a36Sopenharmony_ci
17162306a36Sopenharmony_ci	if (!ies_smt_infos)
17262306a36Sopenharmony_ci		return -EINVAL;
17362306a36Sopenharmony_ci
17462306a36Sopenharmony_ci	for (i = 0; i < info_num; i++) {
17562306a36Sopenharmony_ci		if (pin >= ies_smt_infos[i].start &&
17662306a36Sopenharmony_ci				pin <= ies_smt_infos[i].end) {
17762306a36Sopenharmony_ci			break;
17862306a36Sopenharmony_ci		}
17962306a36Sopenharmony_ci	}
18062306a36Sopenharmony_ci
18162306a36Sopenharmony_ci	if (i == info_num)
18262306a36Sopenharmony_ci		return -EINVAL;
18362306a36Sopenharmony_ci
18462306a36Sopenharmony_ci	if (value)
18562306a36Sopenharmony_ci		reg_addr = ies_smt_infos[i].offset + devdata->port_align;
18662306a36Sopenharmony_ci	else
18762306a36Sopenharmony_ci		reg_addr = ies_smt_infos[i].offset + (devdata->port_align << 1);
18862306a36Sopenharmony_ci
18962306a36Sopenharmony_ci	bit = BIT(ies_smt_infos[i].bit);
19062306a36Sopenharmony_ci	regmap_write(regmap, reg_addr, bit);
19162306a36Sopenharmony_ci	return 0;
19262306a36Sopenharmony_ci}
19362306a36Sopenharmony_ci
19462306a36Sopenharmony_cistatic const struct mtk_pin_drv_grp *mtk_find_pin_drv_grp_by_pin(
19562306a36Sopenharmony_ci		struct mtk_pinctrl *pctl,  unsigned long pin) {
19662306a36Sopenharmony_ci	int i;
19762306a36Sopenharmony_ci
19862306a36Sopenharmony_ci	for (i = 0; i < pctl->devdata->n_pin_drv_grps; i++) {
19962306a36Sopenharmony_ci		const struct mtk_pin_drv_grp *pin_drv =
20062306a36Sopenharmony_ci				pctl->devdata->pin_drv_grp + i;
20162306a36Sopenharmony_ci		if (pin == pin_drv->pin)
20262306a36Sopenharmony_ci			return pin_drv;
20362306a36Sopenharmony_ci	}
20462306a36Sopenharmony_ci
20562306a36Sopenharmony_ci	return NULL;
20662306a36Sopenharmony_ci}
20762306a36Sopenharmony_ci
20862306a36Sopenharmony_cistatic int mtk_pconf_set_driving(struct mtk_pinctrl *pctl,
20962306a36Sopenharmony_ci		unsigned int pin, unsigned char driving)
21062306a36Sopenharmony_ci{
21162306a36Sopenharmony_ci	const struct mtk_pin_drv_grp *pin_drv;
21262306a36Sopenharmony_ci	unsigned int val;
21362306a36Sopenharmony_ci	unsigned int bits, mask, shift;
21462306a36Sopenharmony_ci	const struct mtk_drv_group_desc *drv_grp;
21562306a36Sopenharmony_ci
21662306a36Sopenharmony_ci	if (pin >= pctl->devdata->npins)
21762306a36Sopenharmony_ci		return -EINVAL;
21862306a36Sopenharmony_ci
21962306a36Sopenharmony_ci	pin_drv = mtk_find_pin_drv_grp_by_pin(pctl, pin);
22062306a36Sopenharmony_ci	if (!pin_drv || pin_drv->grp > pctl->devdata->n_grp_cls)
22162306a36Sopenharmony_ci		return -EINVAL;
22262306a36Sopenharmony_ci
22362306a36Sopenharmony_ci	drv_grp = pctl->devdata->grp_desc + pin_drv->grp;
22462306a36Sopenharmony_ci	if (driving >= drv_grp->min_drv && driving <= drv_grp->max_drv
22562306a36Sopenharmony_ci		&& !(driving % drv_grp->step)) {
22662306a36Sopenharmony_ci		val = driving / drv_grp->step - 1;
22762306a36Sopenharmony_ci		bits = drv_grp->high_bit - drv_grp->low_bit + 1;
22862306a36Sopenharmony_ci		mask = BIT(bits) - 1;
22962306a36Sopenharmony_ci		shift = pin_drv->bit + drv_grp->low_bit;
23062306a36Sopenharmony_ci		mask <<= shift;
23162306a36Sopenharmony_ci		val <<= shift;
23262306a36Sopenharmony_ci		return regmap_update_bits(mtk_get_regmap(pctl, pin),
23362306a36Sopenharmony_ci				pin_drv->offset, mask, val);
23462306a36Sopenharmony_ci	}
23562306a36Sopenharmony_ci
23662306a36Sopenharmony_ci	return -EINVAL;
23762306a36Sopenharmony_ci}
23862306a36Sopenharmony_ci
23962306a36Sopenharmony_ciint mtk_pctrl_spec_pull_set_samereg(struct regmap *regmap,
24062306a36Sopenharmony_ci		const struct mtk_pinctrl_devdata *devdata,
24162306a36Sopenharmony_ci		unsigned int pin, bool isup, unsigned int r1r0)
24262306a36Sopenharmony_ci{
24362306a36Sopenharmony_ci	unsigned int i;
24462306a36Sopenharmony_ci	unsigned int reg_pupd, reg_set, reg_rst;
24562306a36Sopenharmony_ci	unsigned int bit_pupd, bit_r0, bit_r1;
24662306a36Sopenharmony_ci	const struct mtk_pin_spec_pupd_set_samereg *spec_pupd_pin;
24762306a36Sopenharmony_ci	bool find = false;
24862306a36Sopenharmony_ci
24962306a36Sopenharmony_ci	if (!devdata->spec_pupd)
25062306a36Sopenharmony_ci		return -EINVAL;
25162306a36Sopenharmony_ci
25262306a36Sopenharmony_ci	for (i = 0; i < devdata->n_spec_pupd; i++) {
25362306a36Sopenharmony_ci		if (pin == devdata->spec_pupd[i].pin) {
25462306a36Sopenharmony_ci			find = true;
25562306a36Sopenharmony_ci			break;
25662306a36Sopenharmony_ci		}
25762306a36Sopenharmony_ci	}
25862306a36Sopenharmony_ci
25962306a36Sopenharmony_ci	if (!find)
26062306a36Sopenharmony_ci		return -EINVAL;
26162306a36Sopenharmony_ci
26262306a36Sopenharmony_ci	spec_pupd_pin = devdata->spec_pupd + i;
26362306a36Sopenharmony_ci	reg_set = spec_pupd_pin->offset + devdata->port_align;
26462306a36Sopenharmony_ci	reg_rst = spec_pupd_pin->offset + (devdata->port_align << 1);
26562306a36Sopenharmony_ci
26662306a36Sopenharmony_ci	if (isup)
26762306a36Sopenharmony_ci		reg_pupd = reg_rst;
26862306a36Sopenharmony_ci	else
26962306a36Sopenharmony_ci		reg_pupd = reg_set;
27062306a36Sopenharmony_ci
27162306a36Sopenharmony_ci	bit_pupd = BIT(spec_pupd_pin->pupd_bit);
27262306a36Sopenharmony_ci	regmap_write(regmap, reg_pupd, bit_pupd);
27362306a36Sopenharmony_ci
27462306a36Sopenharmony_ci	bit_r0 = BIT(spec_pupd_pin->r0_bit);
27562306a36Sopenharmony_ci	bit_r1 = BIT(spec_pupd_pin->r1_bit);
27662306a36Sopenharmony_ci
27762306a36Sopenharmony_ci	switch (r1r0) {
27862306a36Sopenharmony_ci	case MTK_PUPD_SET_R1R0_00:
27962306a36Sopenharmony_ci		regmap_write(regmap, reg_rst, bit_r0);
28062306a36Sopenharmony_ci		regmap_write(regmap, reg_rst, bit_r1);
28162306a36Sopenharmony_ci		break;
28262306a36Sopenharmony_ci	case MTK_PUPD_SET_R1R0_01:
28362306a36Sopenharmony_ci		regmap_write(regmap, reg_set, bit_r0);
28462306a36Sopenharmony_ci		regmap_write(regmap, reg_rst, bit_r1);
28562306a36Sopenharmony_ci		break;
28662306a36Sopenharmony_ci	case MTK_PUPD_SET_R1R0_10:
28762306a36Sopenharmony_ci		regmap_write(regmap, reg_rst, bit_r0);
28862306a36Sopenharmony_ci		regmap_write(regmap, reg_set, bit_r1);
28962306a36Sopenharmony_ci		break;
29062306a36Sopenharmony_ci	case MTK_PUPD_SET_R1R0_11:
29162306a36Sopenharmony_ci		regmap_write(regmap, reg_set, bit_r0);
29262306a36Sopenharmony_ci		regmap_write(regmap, reg_set, bit_r1);
29362306a36Sopenharmony_ci		break;
29462306a36Sopenharmony_ci	default:
29562306a36Sopenharmony_ci		return -EINVAL;
29662306a36Sopenharmony_ci	}
29762306a36Sopenharmony_ci
29862306a36Sopenharmony_ci	return 0;
29962306a36Sopenharmony_ci}
30062306a36Sopenharmony_ci
30162306a36Sopenharmony_cistatic int mtk_pconf_set_pull_select(struct mtk_pinctrl *pctl,
30262306a36Sopenharmony_ci		unsigned int pin, bool enable, bool isup, unsigned int arg)
30362306a36Sopenharmony_ci{
30462306a36Sopenharmony_ci	unsigned int bit;
30562306a36Sopenharmony_ci	unsigned int reg_pullen, reg_pullsel, r1r0;
30662306a36Sopenharmony_ci	int ret;
30762306a36Sopenharmony_ci
30862306a36Sopenharmony_ci	/* Some pins' pull setting are very different,
30962306a36Sopenharmony_ci	 * they have separate pull up/down bit, R0 and R1
31062306a36Sopenharmony_ci	 * resistor bit, so we need this special handle.
31162306a36Sopenharmony_ci	 */
31262306a36Sopenharmony_ci	if (pctl->devdata->spec_pull_set) {
31362306a36Sopenharmony_ci		/* For special pins, bias-disable is set by R1R0,
31462306a36Sopenharmony_ci		 * the parameter should be "MTK_PUPD_SET_R1R0_00".
31562306a36Sopenharmony_ci		 */
31662306a36Sopenharmony_ci		r1r0 = enable ? arg : MTK_PUPD_SET_R1R0_00;
31762306a36Sopenharmony_ci		ret = pctl->devdata->spec_pull_set(mtk_get_regmap(pctl, pin),
31862306a36Sopenharmony_ci						   pctl->devdata, pin, isup,
31962306a36Sopenharmony_ci						   r1r0);
32062306a36Sopenharmony_ci		if (!ret)
32162306a36Sopenharmony_ci			return 0;
32262306a36Sopenharmony_ci	}
32362306a36Sopenharmony_ci
32462306a36Sopenharmony_ci	/* For generic pull config, default arg value should be 0 or 1. */
32562306a36Sopenharmony_ci	if (arg != 0 && arg != 1) {
32662306a36Sopenharmony_ci		dev_err(pctl->dev, "invalid pull-up argument %d on pin %d .\n",
32762306a36Sopenharmony_ci			arg, pin);
32862306a36Sopenharmony_ci		return -EINVAL;
32962306a36Sopenharmony_ci	}
33062306a36Sopenharmony_ci
33162306a36Sopenharmony_ci	if (pctl->devdata->mt8365_set_clr_mode) {
33262306a36Sopenharmony_ci		bit = pin & pctl->devdata->mode_mask;
33362306a36Sopenharmony_ci		reg_pullen = mtk_get_port(pctl, pin) +
33462306a36Sopenharmony_ci			pctl->devdata->pullen_offset;
33562306a36Sopenharmony_ci		reg_pullsel = mtk_get_port(pctl, pin) +
33662306a36Sopenharmony_ci			pctl->devdata->pullsel_offset;
33762306a36Sopenharmony_ci		ret = pctl->devdata->mt8365_set_clr_mode(mtk_get_regmap(pctl, pin),
33862306a36Sopenharmony_ci			bit, reg_pullen, reg_pullsel,
33962306a36Sopenharmony_ci			enable, isup);
34062306a36Sopenharmony_ci		if (ret)
34162306a36Sopenharmony_ci			return -EINVAL;
34262306a36Sopenharmony_ci
34362306a36Sopenharmony_ci		return 0;
34462306a36Sopenharmony_ci	}
34562306a36Sopenharmony_ci
34662306a36Sopenharmony_ci	bit = BIT(pin & pctl->devdata->mode_mask);
34762306a36Sopenharmony_ci	if (enable)
34862306a36Sopenharmony_ci		reg_pullen = SET_ADDR(mtk_get_port(pctl, pin) +
34962306a36Sopenharmony_ci			pctl->devdata->pullen_offset, pctl);
35062306a36Sopenharmony_ci	else
35162306a36Sopenharmony_ci		reg_pullen = CLR_ADDR(mtk_get_port(pctl, pin) +
35262306a36Sopenharmony_ci			pctl->devdata->pullen_offset, pctl);
35362306a36Sopenharmony_ci
35462306a36Sopenharmony_ci	if (isup)
35562306a36Sopenharmony_ci		reg_pullsel = SET_ADDR(mtk_get_port(pctl, pin) +
35662306a36Sopenharmony_ci			pctl->devdata->pullsel_offset, pctl);
35762306a36Sopenharmony_ci	else
35862306a36Sopenharmony_ci		reg_pullsel = CLR_ADDR(mtk_get_port(pctl, pin) +
35962306a36Sopenharmony_ci			pctl->devdata->pullsel_offset, pctl);
36062306a36Sopenharmony_ci
36162306a36Sopenharmony_ci	regmap_write(mtk_get_regmap(pctl, pin), reg_pullen, bit);
36262306a36Sopenharmony_ci	regmap_write(mtk_get_regmap(pctl, pin), reg_pullsel, bit);
36362306a36Sopenharmony_ci	return 0;
36462306a36Sopenharmony_ci}
36562306a36Sopenharmony_ci
36662306a36Sopenharmony_cistatic int mtk_pconf_parse_conf(struct pinctrl_dev *pctldev,
36762306a36Sopenharmony_ci		unsigned int pin, enum pin_config_param param,
36862306a36Sopenharmony_ci		enum pin_config_param arg)
36962306a36Sopenharmony_ci{
37062306a36Sopenharmony_ci	int ret = 0;
37162306a36Sopenharmony_ci	struct mtk_pinctrl *pctl = pinctrl_dev_get_drvdata(pctldev);
37262306a36Sopenharmony_ci
37362306a36Sopenharmony_ci	switch (param) {
37462306a36Sopenharmony_ci	case PIN_CONFIG_BIAS_DISABLE:
37562306a36Sopenharmony_ci		ret = mtk_pconf_set_pull_select(pctl, pin, false, false, arg);
37662306a36Sopenharmony_ci		break;
37762306a36Sopenharmony_ci	case PIN_CONFIG_BIAS_PULL_UP:
37862306a36Sopenharmony_ci		ret = mtk_pconf_set_pull_select(pctl, pin, true, true, arg);
37962306a36Sopenharmony_ci		break;
38062306a36Sopenharmony_ci	case PIN_CONFIG_BIAS_PULL_DOWN:
38162306a36Sopenharmony_ci		ret = mtk_pconf_set_pull_select(pctl, pin, true, false, arg);
38262306a36Sopenharmony_ci		break;
38362306a36Sopenharmony_ci	case PIN_CONFIG_INPUT_ENABLE:
38462306a36Sopenharmony_ci		mtk_pmx_gpio_set_direction(pctldev, NULL, pin, true);
38562306a36Sopenharmony_ci		ret = mtk_pconf_set_ies_smt(pctl, pin, arg, param);
38662306a36Sopenharmony_ci		break;
38762306a36Sopenharmony_ci	case PIN_CONFIG_OUTPUT:
38862306a36Sopenharmony_ci		mtk_gpio_set(pctl->chip, pin, arg);
38962306a36Sopenharmony_ci		ret = mtk_pmx_gpio_set_direction(pctldev, NULL, pin, false);
39062306a36Sopenharmony_ci		break;
39162306a36Sopenharmony_ci	case PIN_CONFIG_INPUT_SCHMITT_ENABLE:
39262306a36Sopenharmony_ci		mtk_pmx_gpio_set_direction(pctldev, NULL, pin, true);
39362306a36Sopenharmony_ci		ret = mtk_pconf_set_ies_smt(pctl, pin, arg, param);
39462306a36Sopenharmony_ci		break;
39562306a36Sopenharmony_ci	case PIN_CONFIG_DRIVE_STRENGTH:
39662306a36Sopenharmony_ci		ret = mtk_pconf_set_driving(pctl, pin, arg);
39762306a36Sopenharmony_ci		break;
39862306a36Sopenharmony_ci	default:
39962306a36Sopenharmony_ci		ret = -EINVAL;
40062306a36Sopenharmony_ci	}
40162306a36Sopenharmony_ci
40262306a36Sopenharmony_ci	return ret;
40362306a36Sopenharmony_ci}
40462306a36Sopenharmony_ci
40562306a36Sopenharmony_cistatic int mtk_pconf_group_get(struct pinctrl_dev *pctldev,
40662306a36Sopenharmony_ci				 unsigned group,
40762306a36Sopenharmony_ci				 unsigned long *config)
40862306a36Sopenharmony_ci{
40962306a36Sopenharmony_ci	struct mtk_pinctrl *pctl = pinctrl_dev_get_drvdata(pctldev);
41062306a36Sopenharmony_ci
41162306a36Sopenharmony_ci	*config = pctl->groups[group].config;
41262306a36Sopenharmony_ci
41362306a36Sopenharmony_ci	return 0;
41462306a36Sopenharmony_ci}
41562306a36Sopenharmony_ci
41662306a36Sopenharmony_cistatic int mtk_pconf_group_set(struct pinctrl_dev *pctldev, unsigned group,
41762306a36Sopenharmony_ci				 unsigned long *configs, unsigned num_configs)
41862306a36Sopenharmony_ci{
41962306a36Sopenharmony_ci	struct mtk_pinctrl *pctl = pinctrl_dev_get_drvdata(pctldev);
42062306a36Sopenharmony_ci	struct mtk_pinctrl_group *g = &pctl->groups[group];
42162306a36Sopenharmony_ci	int i, ret;
42262306a36Sopenharmony_ci
42362306a36Sopenharmony_ci	for (i = 0; i < num_configs; i++) {
42462306a36Sopenharmony_ci		ret = mtk_pconf_parse_conf(pctldev, g->pin,
42562306a36Sopenharmony_ci			pinconf_to_config_param(configs[i]),
42662306a36Sopenharmony_ci			pinconf_to_config_argument(configs[i]));
42762306a36Sopenharmony_ci		if (ret < 0)
42862306a36Sopenharmony_ci			return ret;
42962306a36Sopenharmony_ci
43062306a36Sopenharmony_ci		g->config = configs[i];
43162306a36Sopenharmony_ci	}
43262306a36Sopenharmony_ci
43362306a36Sopenharmony_ci	return 0;
43462306a36Sopenharmony_ci}
43562306a36Sopenharmony_ci
43662306a36Sopenharmony_cistatic const struct pinconf_ops mtk_pconf_ops = {
43762306a36Sopenharmony_ci	.pin_config_group_get	= mtk_pconf_group_get,
43862306a36Sopenharmony_ci	.pin_config_group_set	= mtk_pconf_group_set,
43962306a36Sopenharmony_ci};
44062306a36Sopenharmony_ci
44162306a36Sopenharmony_cistatic struct mtk_pinctrl_group *
44262306a36Sopenharmony_cimtk_pctrl_find_group_by_pin(struct mtk_pinctrl *pctl, u32 pin)
44362306a36Sopenharmony_ci{
44462306a36Sopenharmony_ci	int i;
44562306a36Sopenharmony_ci
44662306a36Sopenharmony_ci	for (i = 0; i < pctl->ngroups; i++) {
44762306a36Sopenharmony_ci		struct mtk_pinctrl_group *grp = pctl->groups + i;
44862306a36Sopenharmony_ci
44962306a36Sopenharmony_ci		if (grp->pin == pin)
45062306a36Sopenharmony_ci			return grp;
45162306a36Sopenharmony_ci	}
45262306a36Sopenharmony_ci
45362306a36Sopenharmony_ci	return NULL;
45462306a36Sopenharmony_ci}
45562306a36Sopenharmony_ci
45662306a36Sopenharmony_cistatic const struct mtk_desc_function *mtk_pctrl_find_function_by_pin(
45762306a36Sopenharmony_ci		struct mtk_pinctrl *pctl, u32 pin_num, u32 fnum)
45862306a36Sopenharmony_ci{
45962306a36Sopenharmony_ci	const struct mtk_desc_pin *pin = pctl->devdata->pins + pin_num;
46062306a36Sopenharmony_ci	const struct mtk_desc_function *func = pin->functions;
46162306a36Sopenharmony_ci
46262306a36Sopenharmony_ci	while (func && func->name) {
46362306a36Sopenharmony_ci		if (func->muxval == fnum)
46462306a36Sopenharmony_ci			return func;
46562306a36Sopenharmony_ci		func++;
46662306a36Sopenharmony_ci	}
46762306a36Sopenharmony_ci
46862306a36Sopenharmony_ci	return NULL;
46962306a36Sopenharmony_ci}
47062306a36Sopenharmony_ci
47162306a36Sopenharmony_cistatic bool mtk_pctrl_is_function_valid(struct mtk_pinctrl *pctl,
47262306a36Sopenharmony_ci		u32 pin_num, u32 fnum)
47362306a36Sopenharmony_ci{
47462306a36Sopenharmony_ci	int i;
47562306a36Sopenharmony_ci
47662306a36Sopenharmony_ci	for (i = 0; i < pctl->devdata->npins; i++) {
47762306a36Sopenharmony_ci		const struct mtk_desc_pin *pin = pctl->devdata->pins + i;
47862306a36Sopenharmony_ci
47962306a36Sopenharmony_ci		if (pin->pin.number == pin_num) {
48062306a36Sopenharmony_ci			const struct mtk_desc_function *func =
48162306a36Sopenharmony_ci					pin->functions;
48262306a36Sopenharmony_ci
48362306a36Sopenharmony_ci			while (func && func->name) {
48462306a36Sopenharmony_ci				if (func->muxval == fnum)
48562306a36Sopenharmony_ci					return true;
48662306a36Sopenharmony_ci				func++;
48762306a36Sopenharmony_ci			}
48862306a36Sopenharmony_ci
48962306a36Sopenharmony_ci			break;
49062306a36Sopenharmony_ci		}
49162306a36Sopenharmony_ci	}
49262306a36Sopenharmony_ci
49362306a36Sopenharmony_ci	return false;
49462306a36Sopenharmony_ci}
49562306a36Sopenharmony_ci
49662306a36Sopenharmony_cistatic int mtk_pctrl_dt_node_to_map_func(struct mtk_pinctrl *pctl,
49762306a36Sopenharmony_ci		u32 pin, u32 fnum, struct mtk_pinctrl_group *grp,
49862306a36Sopenharmony_ci		struct pinctrl_map **map, unsigned *reserved_maps,
49962306a36Sopenharmony_ci		unsigned *num_maps)
50062306a36Sopenharmony_ci{
50162306a36Sopenharmony_ci	bool ret;
50262306a36Sopenharmony_ci
50362306a36Sopenharmony_ci	if (*num_maps == *reserved_maps)
50462306a36Sopenharmony_ci		return -ENOSPC;
50562306a36Sopenharmony_ci
50662306a36Sopenharmony_ci	(*map)[*num_maps].type = PIN_MAP_TYPE_MUX_GROUP;
50762306a36Sopenharmony_ci	(*map)[*num_maps].data.mux.group = grp->name;
50862306a36Sopenharmony_ci
50962306a36Sopenharmony_ci	ret = mtk_pctrl_is_function_valid(pctl, pin, fnum);
51062306a36Sopenharmony_ci	if (!ret) {
51162306a36Sopenharmony_ci		dev_err(pctl->dev, "invalid function %d on pin %d .\n",
51262306a36Sopenharmony_ci				fnum, pin);
51362306a36Sopenharmony_ci		return -EINVAL;
51462306a36Sopenharmony_ci	}
51562306a36Sopenharmony_ci
51662306a36Sopenharmony_ci	(*map)[*num_maps].data.mux.function = mtk_gpio_functions[fnum];
51762306a36Sopenharmony_ci	(*num_maps)++;
51862306a36Sopenharmony_ci
51962306a36Sopenharmony_ci	return 0;
52062306a36Sopenharmony_ci}
52162306a36Sopenharmony_ci
52262306a36Sopenharmony_cistatic int mtk_pctrl_dt_subnode_to_map(struct pinctrl_dev *pctldev,
52362306a36Sopenharmony_ci				      struct device_node *node,
52462306a36Sopenharmony_ci				      struct pinctrl_map **map,
52562306a36Sopenharmony_ci				      unsigned *reserved_maps,
52662306a36Sopenharmony_ci				      unsigned *num_maps)
52762306a36Sopenharmony_ci{
52862306a36Sopenharmony_ci	struct property *pins;
52962306a36Sopenharmony_ci	u32 pinfunc, pin, func;
53062306a36Sopenharmony_ci	int num_pins, num_funcs, maps_per_pin;
53162306a36Sopenharmony_ci	unsigned long *configs;
53262306a36Sopenharmony_ci	unsigned int num_configs;
53362306a36Sopenharmony_ci	bool has_config = false;
53462306a36Sopenharmony_ci	int i, err;
53562306a36Sopenharmony_ci	unsigned reserve = 0;
53662306a36Sopenharmony_ci	struct mtk_pinctrl_group *grp;
53762306a36Sopenharmony_ci	struct mtk_pinctrl *pctl = pinctrl_dev_get_drvdata(pctldev);
53862306a36Sopenharmony_ci
53962306a36Sopenharmony_ci	pins = of_find_property(node, "pinmux", NULL);
54062306a36Sopenharmony_ci	if (!pins) {
54162306a36Sopenharmony_ci		dev_err(pctl->dev, "missing pins property in node %pOFn .\n",
54262306a36Sopenharmony_ci				node);
54362306a36Sopenharmony_ci		return -EINVAL;
54462306a36Sopenharmony_ci	}
54562306a36Sopenharmony_ci
54662306a36Sopenharmony_ci	err = pinconf_generic_parse_dt_config(node, pctldev, &configs,
54762306a36Sopenharmony_ci		&num_configs);
54862306a36Sopenharmony_ci	if (err)
54962306a36Sopenharmony_ci		return err;
55062306a36Sopenharmony_ci
55162306a36Sopenharmony_ci	if (num_configs)
55262306a36Sopenharmony_ci		has_config = true;
55362306a36Sopenharmony_ci
55462306a36Sopenharmony_ci	num_pins = pins->length / sizeof(u32);
55562306a36Sopenharmony_ci	num_funcs = num_pins;
55662306a36Sopenharmony_ci	maps_per_pin = 0;
55762306a36Sopenharmony_ci	if (num_funcs)
55862306a36Sopenharmony_ci		maps_per_pin++;
55962306a36Sopenharmony_ci	if (has_config && num_pins >= 1)
56062306a36Sopenharmony_ci		maps_per_pin++;
56162306a36Sopenharmony_ci
56262306a36Sopenharmony_ci	if (!num_pins || !maps_per_pin) {
56362306a36Sopenharmony_ci		err = -EINVAL;
56462306a36Sopenharmony_ci		goto exit;
56562306a36Sopenharmony_ci	}
56662306a36Sopenharmony_ci
56762306a36Sopenharmony_ci	reserve = num_pins * maps_per_pin;
56862306a36Sopenharmony_ci
56962306a36Sopenharmony_ci	err = pinctrl_utils_reserve_map(pctldev, map,
57062306a36Sopenharmony_ci			reserved_maps, num_maps, reserve);
57162306a36Sopenharmony_ci	if (err < 0)
57262306a36Sopenharmony_ci		goto exit;
57362306a36Sopenharmony_ci
57462306a36Sopenharmony_ci	for (i = 0; i < num_pins; i++) {
57562306a36Sopenharmony_ci		err = of_property_read_u32_index(node, "pinmux",
57662306a36Sopenharmony_ci				i, &pinfunc);
57762306a36Sopenharmony_ci		if (err)
57862306a36Sopenharmony_ci			goto exit;
57962306a36Sopenharmony_ci
58062306a36Sopenharmony_ci		pin = MTK_GET_PIN_NO(pinfunc);
58162306a36Sopenharmony_ci		func = MTK_GET_PIN_FUNC(pinfunc);
58262306a36Sopenharmony_ci
58362306a36Sopenharmony_ci		if (pin >= pctl->devdata->npins ||
58462306a36Sopenharmony_ci				func >= ARRAY_SIZE(mtk_gpio_functions)) {
58562306a36Sopenharmony_ci			dev_err(pctl->dev, "invalid pins value.\n");
58662306a36Sopenharmony_ci			err = -EINVAL;
58762306a36Sopenharmony_ci			goto exit;
58862306a36Sopenharmony_ci		}
58962306a36Sopenharmony_ci
59062306a36Sopenharmony_ci		grp = mtk_pctrl_find_group_by_pin(pctl, pin);
59162306a36Sopenharmony_ci		if (!grp) {
59262306a36Sopenharmony_ci			dev_err(pctl->dev, "unable to match pin %d to group\n",
59362306a36Sopenharmony_ci					pin);
59462306a36Sopenharmony_ci			err = -EINVAL;
59562306a36Sopenharmony_ci			goto exit;
59662306a36Sopenharmony_ci		}
59762306a36Sopenharmony_ci
59862306a36Sopenharmony_ci		err = mtk_pctrl_dt_node_to_map_func(pctl, pin, func, grp, map,
59962306a36Sopenharmony_ci				reserved_maps, num_maps);
60062306a36Sopenharmony_ci		if (err < 0)
60162306a36Sopenharmony_ci			goto exit;
60262306a36Sopenharmony_ci
60362306a36Sopenharmony_ci		if (has_config) {
60462306a36Sopenharmony_ci			err = pinctrl_utils_add_map_configs(pctldev, map,
60562306a36Sopenharmony_ci					reserved_maps, num_maps, grp->name,
60662306a36Sopenharmony_ci					configs, num_configs,
60762306a36Sopenharmony_ci					PIN_MAP_TYPE_CONFIGS_GROUP);
60862306a36Sopenharmony_ci			if (err < 0)
60962306a36Sopenharmony_ci				goto exit;
61062306a36Sopenharmony_ci		}
61162306a36Sopenharmony_ci	}
61262306a36Sopenharmony_ci
61362306a36Sopenharmony_ci	err = 0;
61462306a36Sopenharmony_ci
61562306a36Sopenharmony_ciexit:
61662306a36Sopenharmony_ci	kfree(configs);
61762306a36Sopenharmony_ci	return err;
61862306a36Sopenharmony_ci}
61962306a36Sopenharmony_ci
62062306a36Sopenharmony_cistatic int mtk_pctrl_dt_node_to_map(struct pinctrl_dev *pctldev,
62162306a36Sopenharmony_ci				 struct device_node *np_config,
62262306a36Sopenharmony_ci				 struct pinctrl_map **map, unsigned *num_maps)
62362306a36Sopenharmony_ci{
62462306a36Sopenharmony_ci	struct device_node *np;
62562306a36Sopenharmony_ci	unsigned reserved_maps;
62662306a36Sopenharmony_ci	int ret;
62762306a36Sopenharmony_ci
62862306a36Sopenharmony_ci	*map = NULL;
62962306a36Sopenharmony_ci	*num_maps = 0;
63062306a36Sopenharmony_ci	reserved_maps = 0;
63162306a36Sopenharmony_ci
63262306a36Sopenharmony_ci	for_each_child_of_node(np_config, np) {
63362306a36Sopenharmony_ci		ret = mtk_pctrl_dt_subnode_to_map(pctldev, np, map,
63462306a36Sopenharmony_ci				&reserved_maps, num_maps);
63562306a36Sopenharmony_ci		if (ret < 0) {
63662306a36Sopenharmony_ci			pinctrl_utils_free_map(pctldev, *map, *num_maps);
63762306a36Sopenharmony_ci			of_node_put(np);
63862306a36Sopenharmony_ci			return ret;
63962306a36Sopenharmony_ci		}
64062306a36Sopenharmony_ci	}
64162306a36Sopenharmony_ci
64262306a36Sopenharmony_ci	return 0;
64362306a36Sopenharmony_ci}
64462306a36Sopenharmony_ci
64562306a36Sopenharmony_cistatic int mtk_pctrl_get_groups_count(struct pinctrl_dev *pctldev)
64662306a36Sopenharmony_ci{
64762306a36Sopenharmony_ci	struct mtk_pinctrl *pctl = pinctrl_dev_get_drvdata(pctldev);
64862306a36Sopenharmony_ci
64962306a36Sopenharmony_ci	return pctl->ngroups;
65062306a36Sopenharmony_ci}
65162306a36Sopenharmony_ci
65262306a36Sopenharmony_cistatic const char *mtk_pctrl_get_group_name(struct pinctrl_dev *pctldev,
65362306a36Sopenharmony_ci					      unsigned group)
65462306a36Sopenharmony_ci{
65562306a36Sopenharmony_ci	struct mtk_pinctrl *pctl = pinctrl_dev_get_drvdata(pctldev);
65662306a36Sopenharmony_ci
65762306a36Sopenharmony_ci	return pctl->groups[group].name;
65862306a36Sopenharmony_ci}
65962306a36Sopenharmony_ci
66062306a36Sopenharmony_cistatic int mtk_pctrl_get_group_pins(struct pinctrl_dev *pctldev,
66162306a36Sopenharmony_ci				      unsigned group,
66262306a36Sopenharmony_ci				      const unsigned **pins,
66362306a36Sopenharmony_ci				      unsigned *num_pins)
66462306a36Sopenharmony_ci{
66562306a36Sopenharmony_ci	struct mtk_pinctrl *pctl = pinctrl_dev_get_drvdata(pctldev);
66662306a36Sopenharmony_ci
66762306a36Sopenharmony_ci	*pins = (unsigned *)&pctl->groups[group].pin;
66862306a36Sopenharmony_ci	*num_pins = 1;
66962306a36Sopenharmony_ci
67062306a36Sopenharmony_ci	return 0;
67162306a36Sopenharmony_ci}
67262306a36Sopenharmony_ci
67362306a36Sopenharmony_cistatic const struct pinctrl_ops mtk_pctrl_ops = {
67462306a36Sopenharmony_ci	.dt_node_to_map		= mtk_pctrl_dt_node_to_map,
67562306a36Sopenharmony_ci	.dt_free_map		= pinctrl_utils_free_map,
67662306a36Sopenharmony_ci	.get_groups_count	= mtk_pctrl_get_groups_count,
67762306a36Sopenharmony_ci	.get_group_name		= mtk_pctrl_get_group_name,
67862306a36Sopenharmony_ci	.get_group_pins		= mtk_pctrl_get_group_pins,
67962306a36Sopenharmony_ci};
68062306a36Sopenharmony_ci
68162306a36Sopenharmony_cistatic int mtk_pmx_get_funcs_cnt(struct pinctrl_dev *pctldev)
68262306a36Sopenharmony_ci{
68362306a36Sopenharmony_ci	return ARRAY_SIZE(mtk_gpio_functions);
68462306a36Sopenharmony_ci}
68562306a36Sopenharmony_ci
68662306a36Sopenharmony_cistatic const char *mtk_pmx_get_func_name(struct pinctrl_dev *pctldev,
68762306a36Sopenharmony_ci					   unsigned selector)
68862306a36Sopenharmony_ci{
68962306a36Sopenharmony_ci	return mtk_gpio_functions[selector];
69062306a36Sopenharmony_ci}
69162306a36Sopenharmony_ci
69262306a36Sopenharmony_cistatic int mtk_pmx_get_func_groups(struct pinctrl_dev *pctldev,
69362306a36Sopenharmony_ci				     unsigned function,
69462306a36Sopenharmony_ci				     const char * const **groups,
69562306a36Sopenharmony_ci				     unsigned * const num_groups)
69662306a36Sopenharmony_ci{
69762306a36Sopenharmony_ci	struct mtk_pinctrl *pctl = pinctrl_dev_get_drvdata(pctldev);
69862306a36Sopenharmony_ci
69962306a36Sopenharmony_ci	*groups = pctl->grp_names;
70062306a36Sopenharmony_ci	*num_groups = pctl->ngroups;
70162306a36Sopenharmony_ci
70262306a36Sopenharmony_ci	return 0;
70362306a36Sopenharmony_ci}
70462306a36Sopenharmony_ci
70562306a36Sopenharmony_cistatic int mtk_pmx_set_mode(struct pinctrl_dev *pctldev,
70662306a36Sopenharmony_ci		unsigned long pin, unsigned long mode)
70762306a36Sopenharmony_ci{
70862306a36Sopenharmony_ci	unsigned int reg_addr;
70962306a36Sopenharmony_ci	unsigned char bit;
71062306a36Sopenharmony_ci	unsigned int val;
71162306a36Sopenharmony_ci	unsigned int mask = (1L << GPIO_MODE_BITS) - 1;
71262306a36Sopenharmony_ci	struct mtk_pinctrl *pctl = pinctrl_dev_get_drvdata(pctldev);
71362306a36Sopenharmony_ci
71462306a36Sopenharmony_ci	if (pctl->devdata->spec_pinmux_set)
71562306a36Sopenharmony_ci		pctl->devdata->spec_pinmux_set(mtk_get_regmap(pctl, pin),
71662306a36Sopenharmony_ci					pin, mode);
71762306a36Sopenharmony_ci
71862306a36Sopenharmony_ci	reg_addr = ((pin / pctl->devdata->mode_per_reg) << pctl->devdata->port_shf)
71962306a36Sopenharmony_ci			+ pctl->devdata->pinmux_offset;
72062306a36Sopenharmony_ci
72162306a36Sopenharmony_ci	mode &= mask;
72262306a36Sopenharmony_ci	bit = pin % pctl->devdata->mode_per_reg;
72362306a36Sopenharmony_ci	mask <<= (GPIO_MODE_BITS * bit);
72462306a36Sopenharmony_ci	val = (mode << (GPIO_MODE_BITS * bit));
72562306a36Sopenharmony_ci	return regmap_update_bits(mtk_get_regmap(pctl, pin),
72662306a36Sopenharmony_ci			reg_addr, mask, val);
72762306a36Sopenharmony_ci}
72862306a36Sopenharmony_ci
72962306a36Sopenharmony_cistatic const struct mtk_desc_pin *
73062306a36Sopenharmony_cimtk_find_pin_by_eint_num(struct mtk_pinctrl *pctl, unsigned int eint_num)
73162306a36Sopenharmony_ci{
73262306a36Sopenharmony_ci	int i;
73362306a36Sopenharmony_ci	const struct mtk_desc_pin *pin;
73462306a36Sopenharmony_ci
73562306a36Sopenharmony_ci	for (i = 0; i < pctl->devdata->npins; i++) {
73662306a36Sopenharmony_ci		pin = pctl->devdata->pins + i;
73762306a36Sopenharmony_ci		if (pin->eint.eintnum == eint_num)
73862306a36Sopenharmony_ci			return pin;
73962306a36Sopenharmony_ci	}
74062306a36Sopenharmony_ci
74162306a36Sopenharmony_ci	return NULL;
74262306a36Sopenharmony_ci}
74362306a36Sopenharmony_ci
74462306a36Sopenharmony_cistatic int mtk_pmx_set_mux(struct pinctrl_dev *pctldev,
74562306a36Sopenharmony_ci			    unsigned function,
74662306a36Sopenharmony_ci			    unsigned group)
74762306a36Sopenharmony_ci{
74862306a36Sopenharmony_ci	bool ret;
74962306a36Sopenharmony_ci	const struct mtk_desc_function *desc;
75062306a36Sopenharmony_ci	struct mtk_pinctrl *pctl = pinctrl_dev_get_drvdata(pctldev);
75162306a36Sopenharmony_ci	struct mtk_pinctrl_group *g = pctl->groups + group;
75262306a36Sopenharmony_ci
75362306a36Sopenharmony_ci	ret = mtk_pctrl_is_function_valid(pctl, g->pin, function);
75462306a36Sopenharmony_ci	if (!ret) {
75562306a36Sopenharmony_ci		dev_err(pctl->dev, "invalid function %d on group %d .\n",
75662306a36Sopenharmony_ci				function, group);
75762306a36Sopenharmony_ci		return -EINVAL;
75862306a36Sopenharmony_ci	}
75962306a36Sopenharmony_ci
76062306a36Sopenharmony_ci	desc = mtk_pctrl_find_function_by_pin(pctl, g->pin, function);
76162306a36Sopenharmony_ci	if (!desc)
76262306a36Sopenharmony_ci		return -EINVAL;
76362306a36Sopenharmony_ci	mtk_pmx_set_mode(pctldev, g->pin, desc->muxval);
76462306a36Sopenharmony_ci	return 0;
76562306a36Sopenharmony_ci}
76662306a36Sopenharmony_ci
76762306a36Sopenharmony_cistatic int mtk_pmx_find_gpio_mode(struct mtk_pinctrl *pctl,
76862306a36Sopenharmony_ci				unsigned offset)
76962306a36Sopenharmony_ci{
77062306a36Sopenharmony_ci	const struct mtk_desc_pin *pin = pctl->devdata->pins + offset;
77162306a36Sopenharmony_ci	const struct mtk_desc_function *func = pin->functions;
77262306a36Sopenharmony_ci
77362306a36Sopenharmony_ci	while (func && func->name) {
77462306a36Sopenharmony_ci		if (!strncmp(func->name, GPIO_MODE_PREFIX,
77562306a36Sopenharmony_ci			sizeof(GPIO_MODE_PREFIX)-1))
77662306a36Sopenharmony_ci			return func->muxval;
77762306a36Sopenharmony_ci		func++;
77862306a36Sopenharmony_ci	}
77962306a36Sopenharmony_ci	return -EINVAL;
78062306a36Sopenharmony_ci}
78162306a36Sopenharmony_ci
78262306a36Sopenharmony_cistatic int mtk_pmx_gpio_request_enable(struct pinctrl_dev *pctldev,
78362306a36Sopenharmony_ci				    struct pinctrl_gpio_range *range,
78462306a36Sopenharmony_ci				    unsigned offset)
78562306a36Sopenharmony_ci{
78662306a36Sopenharmony_ci	int muxval;
78762306a36Sopenharmony_ci	struct mtk_pinctrl *pctl = pinctrl_dev_get_drvdata(pctldev);
78862306a36Sopenharmony_ci
78962306a36Sopenharmony_ci	muxval = mtk_pmx_find_gpio_mode(pctl, offset);
79062306a36Sopenharmony_ci
79162306a36Sopenharmony_ci	if (muxval < 0) {
79262306a36Sopenharmony_ci		dev_err(pctl->dev, "invalid gpio pin %d.\n", offset);
79362306a36Sopenharmony_ci		return -EINVAL;
79462306a36Sopenharmony_ci	}
79562306a36Sopenharmony_ci
79662306a36Sopenharmony_ci	mtk_pmx_set_mode(pctldev, offset, muxval);
79762306a36Sopenharmony_ci	mtk_pconf_set_ies_smt(pctl, offset, 1, PIN_CONFIG_INPUT_ENABLE);
79862306a36Sopenharmony_ci
79962306a36Sopenharmony_ci	return 0;
80062306a36Sopenharmony_ci}
80162306a36Sopenharmony_ci
80262306a36Sopenharmony_cistatic const struct pinmux_ops mtk_pmx_ops = {
80362306a36Sopenharmony_ci	.get_functions_count	= mtk_pmx_get_funcs_cnt,
80462306a36Sopenharmony_ci	.get_function_name	= mtk_pmx_get_func_name,
80562306a36Sopenharmony_ci	.get_function_groups	= mtk_pmx_get_func_groups,
80662306a36Sopenharmony_ci	.set_mux		= mtk_pmx_set_mux,
80762306a36Sopenharmony_ci	.gpio_set_direction	= mtk_pmx_gpio_set_direction,
80862306a36Sopenharmony_ci	.gpio_request_enable	= mtk_pmx_gpio_request_enable,
80962306a36Sopenharmony_ci};
81062306a36Sopenharmony_ci
81162306a36Sopenharmony_cistatic int mtk_gpio_direction_input(struct gpio_chip *chip,
81262306a36Sopenharmony_ci					unsigned offset)
81362306a36Sopenharmony_ci{
81462306a36Sopenharmony_ci	return pinctrl_gpio_direction_input(chip->base + offset);
81562306a36Sopenharmony_ci}
81662306a36Sopenharmony_ci
81762306a36Sopenharmony_cistatic int mtk_gpio_direction_output(struct gpio_chip *chip,
81862306a36Sopenharmony_ci					unsigned offset, int value)
81962306a36Sopenharmony_ci{
82062306a36Sopenharmony_ci	mtk_gpio_set(chip, offset, value);
82162306a36Sopenharmony_ci	return pinctrl_gpio_direction_output(chip->base + offset);
82262306a36Sopenharmony_ci}
82362306a36Sopenharmony_ci
82462306a36Sopenharmony_cistatic int mtk_gpio_get_direction(struct gpio_chip *chip, unsigned offset)
82562306a36Sopenharmony_ci{
82662306a36Sopenharmony_ci	unsigned int reg_addr;
82762306a36Sopenharmony_ci	unsigned int bit;
82862306a36Sopenharmony_ci	unsigned int read_val = 0;
82962306a36Sopenharmony_ci
83062306a36Sopenharmony_ci	struct mtk_pinctrl *pctl = gpiochip_get_data(chip);
83162306a36Sopenharmony_ci
83262306a36Sopenharmony_ci	reg_addr =  mtk_get_port(pctl, offset) + pctl->devdata->dir_offset;
83362306a36Sopenharmony_ci	bit = BIT(offset & pctl->devdata->mode_mask);
83462306a36Sopenharmony_ci
83562306a36Sopenharmony_ci	if (pctl->devdata->spec_dir_set)
83662306a36Sopenharmony_ci		pctl->devdata->spec_dir_set(&reg_addr, offset);
83762306a36Sopenharmony_ci
83862306a36Sopenharmony_ci	regmap_read(pctl->regmap1, reg_addr, &read_val);
83962306a36Sopenharmony_ci	if (read_val & bit)
84062306a36Sopenharmony_ci		return GPIO_LINE_DIRECTION_OUT;
84162306a36Sopenharmony_ci
84262306a36Sopenharmony_ci	return GPIO_LINE_DIRECTION_IN;
84362306a36Sopenharmony_ci}
84462306a36Sopenharmony_ci
84562306a36Sopenharmony_cistatic int mtk_gpio_get(struct gpio_chip *chip, unsigned offset)
84662306a36Sopenharmony_ci{
84762306a36Sopenharmony_ci	unsigned int reg_addr;
84862306a36Sopenharmony_ci	unsigned int bit;
84962306a36Sopenharmony_ci	unsigned int read_val = 0;
85062306a36Sopenharmony_ci	struct mtk_pinctrl *pctl = gpiochip_get_data(chip);
85162306a36Sopenharmony_ci
85262306a36Sopenharmony_ci	reg_addr = mtk_get_port(pctl, offset) +
85362306a36Sopenharmony_ci		pctl->devdata->din_offset;
85462306a36Sopenharmony_ci
85562306a36Sopenharmony_ci	bit = BIT(offset & pctl->devdata->mode_mask);
85662306a36Sopenharmony_ci	regmap_read(pctl->regmap1, reg_addr, &read_val);
85762306a36Sopenharmony_ci	return !!(read_val & bit);
85862306a36Sopenharmony_ci}
85962306a36Sopenharmony_ci
86062306a36Sopenharmony_cistatic int mtk_gpio_to_irq(struct gpio_chip *chip, unsigned offset)
86162306a36Sopenharmony_ci{
86262306a36Sopenharmony_ci	struct mtk_pinctrl *pctl = gpiochip_get_data(chip);
86362306a36Sopenharmony_ci	const struct mtk_desc_pin *pin;
86462306a36Sopenharmony_ci	unsigned long eint_n;
86562306a36Sopenharmony_ci
86662306a36Sopenharmony_ci	pin = pctl->devdata->pins + offset;
86762306a36Sopenharmony_ci	if (pin->eint.eintnum == NO_EINT_SUPPORT)
86862306a36Sopenharmony_ci		return -EINVAL;
86962306a36Sopenharmony_ci
87062306a36Sopenharmony_ci	eint_n = pin->eint.eintnum;
87162306a36Sopenharmony_ci
87262306a36Sopenharmony_ci	return mtk_eint_find_irq(pctl->eint, eint_n);
87362306a36Sopenharmony_ci}
87462306a36Sopenharmony_ci
87562306a36Sopenharmony_cistatic int mtk_gpio_set_config(struct gpio_chip *chip, unsigned offset,
87662306a36Sopenharmony_ci			       unsigned long config)
87762306a36Sopenharmony_ci{
87862306a36Sopenharmony_ci	struct mtk_pinctrl *pctl = gpiochip_get_data(chip);
87962306a36Sopenharmony_ci	const struct mtk_desc_pin *pin;
88062306a36Sopenharmony_ci	unsigned long eint_n;
88162306a36Sopenharmony_ci	u32 debounce;
88262306a36Sopenharmony_ci
88362306a36Sopenharmony_ci	if (pinconf_to_config_param(config) != PIN_CONFIG_INPUT_DEBOUNCE)
88462306a36Sopenharmony_ci		return -ENOTSUPP;
88562306a36Sopenharmony_ci
88662306a36Sopenharmony_ci	pin = pctl->devdata->pins + offset;
88762306a36Sopenharmony_ci	if (pin->eint.eintnum == NO_EINT_SUPPORT)
88862306a36Sopenharmony_ci		return -EINVAL;
88962306a36Sopenharmony_ci
89062306a36Sopenharmony_ci	debounce = pinconf_to_config_argument(config);
89162306a36Sopenharmony_ci	eint_n = pin->eint.eintnum;
89262306a36Sopenharmony_ci
89362306a36Sopenharmony_ci	return mtk_eint_set_debounce(pctl->eint, eint_n, debounce);
89462306a36Sopenharmony_ci}
89562306a36Sopenharmony_ci
89662306a36Sopenharmony_cistatic const struct gpio_chip mtk_gpio_chip = {
89762306a36Sopenharmony_ci	.owner			= THIS_MODULE,
89862306a36Sopenharmony_ci	.request		= gpiochip_generic_request,
89962306a36Sopenharmony_ci	.free			= gpiochip_generic_free,
90062306a36Sopenharmony_ci	.get_direction		= mtk_gpio_get_direction,
90162306a36Sopenharmony_ci	.direction_input	= mtk_gpio_direction_input,
90262306a36Sopenharmony_ci	.direction_output	= mtk_gpio_direction_output,
90362306a36Sopenharmony_ci	.get			= mtk_gpio_get,
90462306a36Sopenharmony_ci	.set			= mtk_gpio_set,
90562306a36Sopenharmony_ci	.to_irq			= mtk_gpio_to_irq,
90662306a36Sopenharmony_ci	.set_config		= mtk_gpio_set_config,
90762306a36Sopenharmony_ci};
90862306a36Sopenharmony_ci
90962306a36Sopenharmony_cistatic int mtk_eint_suspend(struct device *device)
91062306a36Sopenharmony_ci{
91162306a36Sopenharmony_ci	struct mtk_pinctrl *pctl = dev_get_drvdata(device);
91262306a36Sopenharmony_ci
91362306a36Sopenharmony_ci	return mtk_eint_do_suspend(pctl->eint);
91462306a36Sopenharmony_ci}
91562306a36Sopenharmony_ci
91662306a36Sopenharmony_cistatic int mtk_eint_resume(struct device *device)
91762306a36Sopenharmony_ci{
91862306a36Sopenharmony_ci	struct mtk_pinctrl *pctl = dev_get_drvdata(device);
91962306a36Sopenharmony_ci
92062306a36Sopenharmony_ci	return mtk_eint_do_resume(pctl->eint);
92162306a36Sopenharmony_ci}
92262306a36Sopenharmony_ci
92362306a36Sopenharmony_ciconst struct dev_pm_ops mtk_eint_pm_ops = {
92462306a36Sopenharmony_ci	.suspend_noirq = mtk_eint_suspend,
92562306a36Sopenharmony_ci	.resume_noirq = mtk_eint_resume,
92662306a36Sopenharmony_ci};
92762306a36Sopenharmony_ci
92862306a36Sopenharmony_cistatic int mtk_pctrl_build_state(struct platform_device *pdev)
92962306a36Sopenharmony_ci{
93062306a36Sopenharmony_ci	struct mtk_pinctrl *pctl = platform_get_drvdata(pdev);
93162306a36Sopenharmony_ci	int i;
93262306a36Sopenharmony_ci
93362306a36Sopenharmony_ci	pctl->ngroups = pctl->devdata->npins;
93462306a36Sopenharmony_ci
93562306a36Sopenharmony_ci	/* Allocate groups */
93662306a36Sopenharmony_ci	pctl->groups = devm_kcalloc(&pdev->dev, pctl->ngroups,
93762306a36Sopenharmony_ci				    sizeof(*pctl->groups), GFP_KERNEL);
93862306a36Sopenharmony_ci	if (!pctl->groups)
93962306a36Sopenharmony_ci		return -ENOMEM;
94062306a36Sopenharmony_ci
94162306a36Sopenharmony_ci	/* We assume that one pin is one group, use pin name as group name. */
94262306a36Sopenharmony_ci	pctl->grp_names = devm_kcalloc(&pdev->dev, pctl->ngroups,
94362306a36Sopenharmony_ci				       sizeof(*pctl->grp_names), GFP_KERNEL);
94462306a36Sopenharmony_ci	if (!pctl->grp_names)
94562306a36Sopenharmony_ci		return -ENOMEM;
94662306a36Sopenharmony_ci
94762306a36Sopenharmony_ci	for (i = 0; i < pctl->devdata->npins; i++) {
94862306a36Sopenharmony_ci		const struct mtk_desc_pin *pin = pctl->devdata->pins + i;
94962306a36Sopenharmony_ci		struct mtk_pinctrl_group *group = pctl->groups + i;
95062306a36Sopenharmony_ci
95162306a36Sopenharmony_ci		group->name = pin->pin.name;
95262306a36Sopenharmony_ci		group->pin = pin->pin.number;
95362306a36Sopenharmony_ci
95462306a36Sopenharmony_ci		pctl->grp_names[i] = pin->pin.name;
95562306a36Sopenharmony_ci	}
95662306a36Sopenharmony_ci
95762306a36Sopenharmony_ci	return 0;
95862306a36Sopenharmony_ci}
95962306a36Sopenharmony_ci
96062306a36Sopenharmony_cistatic int
96162306a36Sopenharmony_cimtk_xt_get_gpio_n(void *data, unsigned long eint_n, unsigned int *gpio_n,
96262306a36Sopenharmony_ci		  struct gpio_chip **gpio_chip)
96362306a36Sopenharmony_ci{
96462306a36Sopenharmony_ci	struct mtk_pinctrl *pctl = (struct mtk_pinctrl *)data;
96562306a36Sopenharmony_ci	const struct mtk_desc_pin *pin;
96662306a36Sopenharmony_ci
96762306a36Sopenharmony_ci	pin = mtk_find_pin_by_eint_num(pctl, eint_n);
96862306a36Sopenharmony_ci	if (!pin)
96962306a36Sopenharmony_ci		return -EINVAL;
97062306a36Sopenharmony_ci
97162306a36Sopenharmony_ci	*gpio_chip = pctl->chip;
97262306a36Sopenharmony_ci	*gpio_n = pin->pin.number;
97362306a36Sopenharmony_ci
97462306a36Sopenharmony_ci	return 0;
97562306a36Sopenharmony_ci}
97662306a36Sopenharmony_ci
97762306a36Sopenharmony_cistatic int mtk_xt_get_gpio_state(void *data, unsigned long eint_n)
97862306a36Sopenharmony_ci{
97962306a36Sopenharmony_ci	struct mtk_pinctrl *pctl = (struct mtk_pinctrl *)data;
98062306a36Sopenharmony_ci	const struct mtk_desc_pin *pin;
98162306a36Sopenharmony_ci
98262306a36Sopenharmony_ci	pin = mtk_find_pin_by_eint_num(pctl, eint_n);
98362306a36Sopenharmony_ci	if (!pin)
98462306a36Sopenharmony_ci		return -EINVAL;
98562306a36Sopenharmony_ci
98662306a36Sopenharmony_ci	return mtk_gpio_get(pctl->chip, pin->pin.number);
98762306a36Sopenharmony_ci}
98862306a36Sopenharmony_ci
98962306a36Sopenharmony_cistatic int mtk_xt_set_gpio_as_eint(void *data, unsigned long eint_n)
99062306a36Sopenharmony_ci{
99162306a36Sopenharmony_ci	struct mtk_pinctrl *pctl = (struct mtk_pinctrl *)data;
99262306a36Sopenharmony_ci	const struct mtk_desc_pin *pin;
99362306a36Sopenharmony_ci
99462306a36Sopenharmony_ci	pin = mtk_find_pin_by_eint_num(pctl, eint_n);
99562306a36Sopenharmony_ci	if (!pin)
99662306a36Sopenharmony_ci		return -EINVAL;
99762306a36Sopenharmony_ci
99862306a36Sopenharmony_ci	/* set mux to INT mode */
99962306a36Sopenharmony_ci	mtk_pmx_set_mode(pctl->pctl_dev, pin->pin.number, pin->eint.eintmux);
100062306a36Sopenharmony_ci	/* set gpio direction to input */
100162306a36Sopenharmony_ci	mtk_pmx_gpio_set_direction(pctl->pctl_dev, NULL, pin->pin.number,
100262306a36Sopenharmony_ci				   true);
100362306a36Sopenharmony_ci	/* set input-enable */
100462306a36Sopenharmony_ci	mtk_pconf_set_ies_smt(pctl, pin->pin.number, 1,
100562306a36Sopenharmony_ci			      PIN_CONFIG_INPUT_ENABLE);
100662306a36Sopenharmony_ci
100762306a36Sopenharmony_ci	return 0;
100862306a36Sopenharmony_ci}
100962306a36Sopenharmony_ci
101062306a36Sopenharmony_cistatic const struct mtk_eint_xt mtk_eint_xt = {
101162306a36Sopenharmony_ci	.get_gpio_n = mtk_xt_get_gpio_n,
101262306a36Sopenharmony_ci	.get_gpio_state = mtk_xt_get_gpio_state,
101362306a36Sopenharmony_ci	.set_gpio_as_eint = mtk_xt_set_gpio_as_eint,
101462306a36Sopenharmony_ci};
101562306a36Sopenharmony_ci
101662306a36Sopenharmony_cistatic int mtk_eint_init(struct mtk_pinctrl *pctl, struct platform_device *pdev)
101762306a36Sopenharmony_ci{
101862306a36Sopenharmony_ci	struct device_node *np = pdev->dev.of_node;
101962306a36Sopenharmony_ci
102062306a36Sopenharmony_ci	if (!of_property_read_bool(np, "interrupt-controller"))
102162306a36Sopenharmony_ci		return -ENODEV;
102262306a36Sopenharmony_ci
102362306a36Sopenharmony_ci	pctl->eint = devm_kzalloc(pctl->dev, sizeof(*pctl->eint), GFP_KERNEL);
102462306a36Sopenharmony_ci	if (!pctl->eint)
102562306a36Sopenharmony_ci		return -ENOMEM;
102662306a36Sopenharmony_ci
102762306a36Sopenharmony_ci	pctl->eint->base = devm_platform_ioremap_resource(pdev, 0);
102862306a36Sopenharmony_ci	if (IS_ERR(pctl->eint->base))
102962306a36Sopenharmony_ci		return PTR_ERR(pctl->eint->base);
103062306a36Sopenharmony_ci
103162306a36Sopenharmony_ci	pctl->eint->irq = irq_of_parse_and_map(np, 0);
103262306a36Sopenharmony_ci	if (!pctl->eint->irq)
103362306a36Sopenharmony_ci		return -EINVAL;
103462306a36Sopenharmony_ci
103562306a36Sopenharmony_ci	pctl->eint->dev = &pdev->dev;
103662306a36Sopenharmony_ci	/*
103762306a36Sopenharmony_ci	 * If pctl->eint->regs == NULL, it would fall back into using a generic
103862306a36Sopenharmony_ci	 * register map in mtk_eint_do_init calls.
103962306a36Sopenharmony_ci	 */
104062306a36Sopenharmony_ci	pctl->eint->regs = pctl->devdata->eint_regs;
104162306a36Sopenharmony_ci	pctl->eint->hw = &pctl->devdata->eint_hw;
104262306a36Sopenharmony_ci	pctl->eint->pctl = pctl;
104362306a36Sopenharmony_ci	pctl->eint->gpio_xlate = &mtk_eint_xt;
104462306a36Sopenharmony_ci
104562306a36Sopenharmony_ci	return mtk_eint_do_init(pctl->eint);
104662306a36Sopenharmony_ci}
104762306a36Sopenharmony_ci
104862306a36Sopenharmony_ci/* This is used as a common probe function */
104962306a36Sopenharmony_ciint mtk_pctrl_init(struct platform_device *pdev,
105062306a36Sopenharmony_ci		const struct mtk_pinctrl_devdata *data,
105162306a36Sopenharmony_ci		struct regmap *regmap)
105262306a36Sopenharmony_ci{
105362306a36Sopenharmony_ci	struct device *dev = &pdev->dev;
105462306a36Sopenharmony_ci	struct pinctrl_pin_desc *pins;
105562306a36Sopenharmony_ci	struct mtk_pinctrl *pctl;
105662306a36Sopenharmony_ci	struct device_node *np = pdev->dev.of_node, *node;
105762306a36Sopenharmony_ci	int ret, i;
105862306a36Sopenharmony_ci
105962306a36Sopenharmony_ci	pctl = devm_kzalloc(&pdev->dev, sizeof(*pctl), GFP_KERNEL);
106062306a36Sopenharmony_ci	if (!pctl)
106162306a36Sopenharmony_ci		return -ENOMEM;
106262306a36Sopenharmony_ci
106362306a36Sopenharmony_ci	platform_set_drvdata(pdev, pctl);
106462306a36Sopenharmony_ci
106562306a36Sopenharmony_ci	node = of_parse_phandle(np, "mediatek,pctl-regmap", 0);
106662306a36Sopenharmony_ci	if (node) {
106762306a36Sopenharmony_ci		pctl->regmap1 = syscon_node_to_regmap(node);
106862306a36Sopenharmony_ci		of_node_put(node);
106962306a36Sopenharmony_ci		if (IS_ERR(pctl->regmap1))
107062306a36Sopenharmony_ci			return PTR_ERR(pctl->regmap1);
107162306a36Sopenharmony_ci	} else if (regmap) {
107262306a36Sopenharmony_ci		pctl->regmap1  = regmap;
107362306a36Sopenharmony_ci	} else {
107462306a36Sopenharmony_ci		return dev_err_probe(dev, -EINVAL, "Cannot find pinctrl regmap.\n");
107562306a36Sopenharmony_ci	}
107662306a36Sopenharmony_ci
107762306a36Sopenharmony_ci	/* Only 8135 has two base addr, other SoCs have only one. */
107862306a36Sopenharmony_ci	node = of_parse_phandle(np, "mediatek,pctl-regmap", 1);
107962306a36Sopenharmony_ci	if (node) {
108062306a36Sopenharmony_ci		pctl->regmap2 = syscon_node_to_regmap(node);
108162306a36Sopenharmony_ci		of_node_put(node);
108262306a36Sopenharmony_ci		if (IS_ERR(pctl->regmap2))
108362306a36Sopenharmony_ci			return PTR_ERR(pctl->regmap2);
108462306a36Sopenharmony_ci	}
108562306a36Sopenharmony_ci
108662306a36Sopenharmony_ci	pctl->devdata = data;
108762306a36Sopenharmony_ci	ret = mtk_pctrl_build_state(pdev);
108862306a36Sopenharmony_ci	if (ret)
108962306a36Sopenharmony_ci		return dev_err_probe(dev, ret, "build state failed\n");
109062306a36Sopenharmony_ci
109162306a36Sopenharmony_ci	pins = devm_kcalloc(&pdev->dev, pctl->devdata->npins, sizeof(*pins),
109262306a36Sopenharmony_ci			    GFP_KERNEL);
109362306a36Sopenharmony_ci	if (!pins)
109462306a36Sopenharmony_ci		return -ENOMEM;
109562306a36Sopenharmony_ci
109662306a36Sopenharmony_ci	for (i = 0; i < pctl->devdata->npins; i++)
109762306a36Sopenharmony_ci		pins[i] = pctl->devdata->pins[i].pin;
109862306a36Sopenharmony_ci
109962306a36Sopenharmony_ci	pctl->pctl_desc.name = dev_name(&pdev->dev);
110062306a36Sopenharmony_ci	pctl->pctl_desc.owner = THIS_MODULE;
110162306a36Sopenharmony_ci	pctl->pctl_desc.pins = pins;
110262306a36Sopenharmony_ci	pctl->pctl_desc.npins = pctl->devdata->npins;
110362306a36Sopenharmony_ci	pctl->pctl_desc.confops = &mtk_pconf_ops;
110462306a36Sopenharmony_ci	pctl->pctl_desc.pctlops = &mtk_pctrl_ops;
110562306a36Sopenharmony_ci	pctl->pctl_desc.pmxops = &mtk_pmx_ops;
110662306a36Sopenharmony_ci	pctl->dev = &pdev->dev;
110762306a36Sopenharmony_ci
110862306a36Sopenharmony_ci	pctl->pctl_dev = devm_pinctrl_register(&pdev->dev, &pctl->pctl_desc,
110962306a36Sopenharmony_ci					       pctl);
111062306a36Sopenharmony_ci	if (IS_ERR(pctl->pctl_dev))
111162306a36Sopenharmony_ci		return dev_err_probe(dev, PTR_ERR(pctl->pctl_dev),
111262306a36Sopenharmony_ci				     "Couldn't register pinctrl driver\n");
111362306a36Sopenharmony_ci
111462306a36Sopenharmony_ci	pctl->chip = devm_kzalloc(&pdev->dev, sizeof(*pctl->chip), GFP_KERNEL);
111562306a36Sopenharmony_ci	if (!pctl->chip)
111662306a36Sopenharmony_ci		return -ENOMEM;
111762306a36Sopenharmony_ci
111862306a36Sopenharmony_ci	*pctl->chip = mtk_gpio_chip;
111962306a36Sopenharmony_ci	pctl->chip->ngpio = pctl->devdata->npins;
112062306a36Sopenharmony_ci	pctl->chip->label = dev_name(&pdev->dev);
112162306a36Sopenharmony_ci	pctl->chip->parent = &pdev->dev;
112262306a36Sopenharmony_ci	pctl->chip->base = -1;
112362306a36Sopenharmony_ci
112462306a36Sopenharmony_ci	ret = gpiochip_add_data(pctl->chip, pctl);
112562306a36Sopenharmony_ci	if (ret)
112662306a36Sopenharmony_ci		return -EINVAL;
112762306a36Sopenharmony_ci
112862306a36Sopenharmony_ci	/* Register the GPIO to pin mappings. */
112962306a36Sopenharmony_ci	ret = gpiochip_add_pin_range(pctl->chip, dev_name(&pdev->dev),
113062306a36Sopenharmony_ci			0, 0, pctl->devdata->npins);
113162306a36Sopenharmony_ci	if (ret) {
113262306a36Sopenharmony_ci		ret = -EINVAL;
113362306a36Sopenharmony_ci		goto chip_error;
113462306a36Sopenharmony_ci	}
113562306a36Sopenharmony_ci
113662306a36Sopenharmony_ci	ret = mtk_eint_init(pctl, pdev);
113762306a36Sopenharmony_ci	if (ret)
113862306a36Sopenharmony_ci		goto chip_error;
113962306a36Sopenharmony_ci
114062306a36Sopenharmony_ci	return 0;
114162306a36Sopenharmony_ci
114262306a36Sopenharmony_cichip_error:
114362306a36Sopenharmony_ci	gpiochip_remove(pctl->chip);
114462306a36Sopenharmony_ci	return ret;
114562306a36Sopenharmony_ci}
114662306a36Sopenharmony_ci
114762306a36Sopenharmony_ciint mtk_pctrl_common_probe(struct platform_device *pdev)
114862306a36Sopenharmony_ci{
114962306a36Sopenharmony_ci	struct device *dev = &pdev->dev;
115062306a36Sopenharmony_ci	const struct mtk_pinctrl_devdata *data = device_get_match_data(dev);
115162306a36Sopenharmony_ci
115262306a36Sopenharmony_ci	if (!data)
115362306a36Sopenharmony_ci		return -ENODEV;
115462306a36Sopenharmony_ci
115562306a36Sopenharmony_ci	return mtk_pctrl_init(pdev, data, NULL);
115662306a36Sopenharmony_ci}
1157