Skip to content

Commit

Permalink
pinctrl: sunxi: Support I/O bias voltage setting on A80
Browse files Browse the repository at this point in the history
The A80 SoC has configuration registers for I/O bias voltage. Incorrect
settings would make the affected peripherals inoperable in some cases,
such as Ethernet RGMII signals biased at 2.5V with the settings still
at 3.3V. However low speed signals such as MDIO on the same group of
pins seem to be unaffected.

Previously there was no way to know what the actual voltage used was,
short of hard-coding a value in the device tree. With the new pin bank
regulator supply support in place, the driver can now query the
regulator for its voltage, and if it's valid (as opposed to being the
dummy regulator), set the bias voltage setting accordingly.

Add a quirk to denote the presence of the configuration registers, and
a function to set the correct setting based on the voltage read back
from the regulator.

This is only done when the regulator is first acquired and enabled.
While it would be nice to have a notifier on the regulator so that when
the voltage changes, the driver can update the setting, in practice no
board currently supports dynamic changing of the I/O voltages.

Signed-off-by: Chen-Yu Tsai <wens@csie.org>
Acked-by: Maxime Ripard <maxime.ripard@bootlin.com>
Signed-off-by: Linus Walleij <linus.walleij@linaro.org>
  • Loading branch information
Chen-Yu Tsai authored and Linus Walleij committed Feb 11, 2019
1 parent e653721 commit 402bfb3
Show file tree
Hide file tree
Showing 4 changed files with 55 additions and 0 deletions.
1 change: 1 addition & 0 deletions drivers/pinctrl/sunxi/pinctrl-sun9i-a80-r.c
Original file line number Diff line number Diff line change
Expand Up @@ -153,6 +153,7 @@ static const struct sunxi_pinctrl_desc sun9i_a80_r_pinctrl_data = {
.pin_base = PL_BASE,
.irq_banks = 2,
.disable_strict_mode = true,
.has_io_bias_cfg = true,
};

static int sun9i_a80_r_pinctrl_probe(struct platform_device *pdev)
Expand Down
1 change: 1 addition & 0 deletions drivers/pinctrl/sunxi/pinctrl-sun9i-a80.c
Original file line number Diff line number Diff line change
Expand Up @@ -722,6 +722,7 @@ static const struct sunxi_pinctrl_desc sun9i_a80_pinctrl_data = {
.npins = ARRAY_SIZE(sun9i_a80_pins),
.irq_banks = 5,
.disable_strict_mode = true,
.has_io_bias_cfg = true,
};

static int sun9i_a80_pinctrl_probe(struct platform_device *pdev)
Expand Down
41 changes: 41 additions & 0 deletions drivers/pinctrl/sunxi/pinctrl-sunxi.c
Original file line number Diff line number Diff line change
Expand Up @@ -603,6 +603,45 @@ static const struct pinconf_ops sunxi_pconf_ops = {
.pin_config_group_set = sunxi_pconf_group_set,
};

static int sunxi_pinctrl_set_io_bias_cfg(struct sunxi_pinctrl *pctl,
unsigned pin,
struct regulator *supply)
{
u32 val, reg;
int uV;

if (!pctl->desc->has_io_bias_cfg)
return 0;

uV = regulator_get_voltage(supply);
if (uV < 0)
return uV;

/* Might be dummy regulator with no voltage set */
if (uV == 0)
return 0;

/* Configured value must be equal or greater to actual voltage */
if (uV <= 1800000)
val = 0x0; /* 1.8V */
else if (uV <= 2500000)
val = 0x6; /* 2.5V */
else if (uV <= 2800000)
val = 0x9; /* 2.8V */
else if (uV <= 3000000)
val = 0xA; /* 3.0V */
else
val = 0xD; /* 3.3V */

pin -= pctl->desc->pin_base;

reg = readl(pctl->membase + sunxi_grp_config_reg(pin));
reg &= ~IO_BIAS_MASK;
writel(reg | val, pctl->membase + sunxi_grp_config_reg(pin));

return 0;
}

static int sunxi_pmx_get_funcs_cnt(struct pinctrl_dev *pctldev)
{
struct sunxi_pinctrl *pctl = pinctrl_dev_get_drvdata(pctldev);
Expand Down Expand Up @@ -725,6 +764,8 @@ static int sunxi_pmx_request(struct pinctrl_dev *pctldev, unsigned offset)
goto out;
}

sunxi_pinctrl_set_io_bias_cfg(pctl, offset, reg);

s_reg->regulator = reg;
refcount_set(&s_reg->refcount, 1);

Expand Down
12 changes: 12 additions & 0 deletions drivers/pinctrl/sunxi/pinctrl-sunxi.h
Original file line number Diff line number Diff line change
Expand Up @@ -79,6 +79,10 @@
#define IRQ_LEVEL_LOW 0x03
#define IRQ_EDGE_BOTH 0x04

#define GRP_CFG_REG 0x300

#define IO_BIAS_MASK GENMASK(3, 0)

#define SUN4I_FUNC_INPUT 0
#define SUN4I_FUNC_IRQ 6

Expand Down Expand Up @@ -113,6 +117,7 @@ struct sunxi_pinctrl_desc {
const unsigned int *irq_bank_map;
bool irq_read_needs_mux;
bool disable_strict_mode;
bool has_io_bias_cfg;
};

struct sunxi_pinctrl_function {
Expand Down Expand Up @@ -338,6 +343,13 @@ static inline u32 sunxi_irq_status_offset(u16 irq)
return irq_num * IRQ_STATUS_IRQ_BITS;
}

static inline u32 sunxi_grp_config_reg(u16 pin)
{
u8 bank = pin / PINS_PER_BANK;

return GRP_CFG_REG + bank * 0x4;
}

int sunxi_pinctrl_init_with_variant(struct platform_device *pdev,
const struct sunxi_pinctrl_desc *desc,
unsigned long variant);
Expand Down

0 comments on commit 402bfb3

Please sign in to comment.