Hey Neil, On Mon Jun 30, 2025 at 5:04 PM WEST, Neil Armstrong wrote: > The current qcom_pmic_gpio driver is too limited and doesn't > support state tracking for all pins like the Linux driver. > > Adding full pinconf support would require adding the state > and it's much simpler to restart from scratch with a new > driver based on the Linux one adapted to the U-Boot GPIO > and Pinctrl APIs. > > For now only the PMICs I've been able to validate are > added in the compatible list but we should be able to > add the entire list from the Linux driver. > > There's a few difference from the Linux driver: > - no IRQ support > - uses the U-Boot GPIO flags that maps very well > - uses the gpio-ranges to get the pins count > - no debugfs but prints the pin state via pinmux callback > > It uses the same CONFIG entry as the old one, since > the ultimate goal is to migrate entirely on this new > driver once we verify it doesn't break the older > platforms. > > Tested-by: Alexey Minnekhanov <[email protected]> > Signed-off-by: Neil Armstrong <[email protected]>
LGTM Reviewed-by: Rui Miguel Silva <[email protected]> Cheers, Rui > --- > drivers/gpio/Makefile | 2 +- > drivers/gpio/qcom_spmi_gpio.c | 1034 > +++++++++++++++++++++++++++++++++++++++++ > 2 files changed, 1035 insertions(+), 1 deletion(-) > > diff --git a/drivers/gpio/Makefile b/drivers/gpio/Makefile > index > d64c14db5cfd3e5e303285025188e68b24bb9448..73fd5a7b1bd408b972080c8a97052a5847c8966d > 100644 > --- a/drivers/gpio/Makefile > +++ b/drivers/gpio/Makefile > @@ -65,7 +65,7 @@ obj-$(CONFIG_OCTEON_GPIO) += octeon_gpio.o > obj-$(CONFIG_MVEBU_GPIO) += mvebu_gpio.o > obj-$(CONFIG_MSM_GPIO) += msm_gpio.o > obj-$(CONFIG_$(PHASE_)PCF8575_GPIO) += pcf8575_gpio.o > -obj-$(CONFIG_$(PHASE_)QCOM_PMIC_GPIO) += qcom_pmic_gpio.o > +obj-$(CONFIG_$(PHASE_)QCOM_PMIC_GPIO) += qcom_pmic_gpio.o > qcom_spmi_gpio.o > obj-$(CONFIG_MT7620_GPIO) += mt7620_gpio.o > obj-$(CONFIG_MT7621_GPIO) += mt7621_gpio.o > obj-$(CONFIG_MSCC_SGPIO) += mscc_sgpio.o > diff --git a/drivers/gpio/qcom_spmi_gpio.c b/drivers/gpio/qcom_spmi_gpio.c > new file mode 100644 > index > 0000000000000000000000000000000000000000..2bb0f0d10c32ff55d3fd5c61b9bc2999878bfea6 > --- /dev/null > +++ b/drivers/gpio/qcom_spmi_gpio.c > @@ -0,0 +1,1034 @@ > +// SPDX-License-Identifier: GPL-2.0+ > +/* > + * Qualcomm generic pmic gpio driver > + * > + * Based on the original qcom_spmi_pmic_gpio driver from: > + * Copyright (c) 2015 Mateusz Kulikowski <[email protected]> > + * > + * Updated from the Linux pinctrl-spmi-gpio driver from: > + * Copyright (c) 2012-2014, 2016-2021 The Linux Foundation. All rights > reserved. > + * Copyright (c) 2021-2022 Qualcomm Innovation Center, Inc. All rights > reserved. > + */ > + > +#include <dm.h> > +#include <dm/device-internal.h> > +#include <dm/lists.h> > +#include <dm/pinctrl.h> > +#include <dm/device_compat.h> > +#include <log.h> > +#include <power/pmic.h> > +#include <spmi/spmi.h> > +#include <asm/io.h> > +#include <stdlib.h> > +#include <asm/gpio.h> > +#include <linux/bitops.h> > +#include <dt-bindings/pinctrl/qcom,pmic-gpio.h> > + > +#define PMIC_MAX_GPIOS 36 > + > +#define PMIC_GPIO_ADDRESS_RANGE 0x100 > + > +/* type and subtype registers base address offsets */ > +#define PMIC_GPIO_REG_TYPE 0x4 > +#define PMIC_GPIO_REG_SUBTYPE 0x5 > + > +/* GPIO peripheral type and subtype out_values */ > +#define PMIC_GPIO_TYPE 0x10 > +#define PMIC_GPIO_SUBTYPE_GPIO_4CH 0x1 > +#define PMIC_GPIO_SUBTYPE_GPIOC_4CH 0x5 > +#define PMIC_GPIO_SUBTYPE_GPIO_8CH 0x9 > +#define PMIC_GPIO_SUBTYPE_GPIOC_8CH 0xd > +#define PMIC_GPIO_SUBTYPE_GPIO_LV 0x10 > +#define PMIC_GPIO_SUBTYPE_GPIO_MV 0x11 > +#define PMIC_GPIO_SUBTYPE_GPIO_LV_VIN2 0x12 > +#define PMIC_GPIO_SUBTYPE_GPIO_MV_VIN3 0x13 > + > +#define PMIC_MPP_REG_RT_STS 0x10 > +#define PMIC_MPP_REG_RT_STS_VAL_MASK 0x1 > + > +/* control register base address offsets */ > +#define PMIC_GPIO_REG_MODE_CTL 0x40 > +#define PMIC_GPIO_REG_DIG_VIN_CTL 0x41 > +#define PMIC_GPIO_REG_DIG_PULL_CTL 0x42 > +#define PMIC_GPIO_REG_LV_MV_DIG_OUT_SOURCE_CTL 0x44 > + #define PMIC_GPIO_REG_DIG_IN_CTL 0x43 > +#define PMIC_GPIO_REG_DIG_OUT_CTL 0x45 > +#define PMIC_GPIO_REG_EN_CTL 0x46 > + #define PMIC_GPIO_REG_LV_MV_ANA_PASS_THRU_SEL 0x4A > + > +/* PMIC_GPIO_REG_MODE_CTL */ > +#define PMIC_GPIO_REG_MODE_VALUE_SHIFT 0x1 > +#define PMIC_GPIO_REG_MODE_FUNCTION_SHIFT 1 > +#define PMIC_GPIO_REG_MODE_FUNCTION_MASK 0x7 > +#define PMIC_GPIO_REG_MODE_DIR_SHIFT 4 > +#define PMIC_GPIO_REG_MODE_DIR_MASK 0x7 > + > +#define PMIC_GPIO_MODE_DIGITAL_INPUT 0 > +#define PMIC_GPIO_MODE_DIGITAL_OUTPUT 1 > +#define PMIC_GPIO_MODE_DIGITAL_INPUT_OUTPUT 2 > +#define PMIC_GPIO_MODE_ANALOG_PASS_THRU 3 > +#define PMIC_GPIO_REG_LV_MV_MODE_DIR_MASK 0x3 > + > +/* PMIC_GPIO_REG_DIG_VIN_CTL */ > +#define PMIC_GPIO_REG_VIN_SHIFT 0 > +#define PMIC_GPIO_REG_VIN_MASK 0x7 > + > +/* PMIC_GPIO_REG_DIG_PULL_CTL */ > +#define PMIC_GPIO_REG_PULL_SHIFT 0 > +#define PMIC_GPIO_REG_PULL_MASK 0x7 > + > +#define PMIC_GPIO_PULL_DOWN 4 > +#define PMIC_GPIO_PULL_DISABLE 5 > + > +/* PMIC_GPIO_REG_LV_MV_DIG_OUT_SOURCE_CTL for LV/MV */ > +#define PMIC_GPIO_LV_MV_OUTPUT_INVERT 0x80 > +#define PMIC_GPIO_LV_MV_OUTPUT_INVERT_SHIFT 7 > +#define PMIC_GPIO_LV_MV_OUTPUT_SOURCE_SEL_MASK 0xF > + > +/* PMIC_GPIO_REG_DIG_IN_CTL */ > +#define PMIC_GPIO_LV_MV_DIG_IN_DTEST_EN 0x80 > +#define PMIC_GPIO_LV_MV_DIG_IN_DTEST_SEL_MASK 0x7 > +#define PMIC_GPIO_DIG_IN_DTEST_SEL_MASK 0xf > + > +/* PMIC_GPIO_REG_DIG_OUT_CTL */ > +#define PMIC_GPIO_REG_OUT_STRENGTH_SHIFT 0 > +#define PMIC_GPIO_REG_OUT_STRENGTH_MASK 0x3 > +#define PMIC_GPIO_REG_OUT_TYPE_SHIFT 4 > +#define PMIC_GPIO_REG_OUT_TYPE_MASK 0x3 > + > +/* > + * Output type - indicates pin should be configured as push-pull, > + * open drain or open source. > + */ > +#define PMIC_GPIO_OUT_BUF_CMOS 0 > +#define PMIC_GPIO_OUT_BUF_OPEN_DRAIN_NMOS 1 > +#define PMIC_GPIO_OUT_BUF_OPEN_DRAIN_PMOS 2 > + > +#define PMIC_GPIO_OUT_STRENGTH_LOW 1 > +#define PMIC_GPIO_OUT_STRENGTH_HIGH 3 > + > +/* PMIC_GPIO_REG_EN_CTL */ > +#define PMIC_GPIO_REG_MASTER_EN_SHIFT 7 > + > +#define PMIC_GPIO_PHYSICAL_OFFSET 1 > + > +/* PMIC_GPIO_REG_LV_MV_ANA_PASS_THRU_SEL */ > +#define PMIC_GPIO_LV_MV_ANA_MUX_SEL_MASK 0x3 > + > +/* The index of each function in spmi_pmic_gpio_functions[] array */ > +enum spmi_pmic_gpio_func_index { > + PMIC_GPIO_FUNC_INDEX_NORMAL, > + PMIC_GPIO_FUNC_INDEX_PAIRED, > + PMIC_GPIO_FUNC_INDEX_FUNC1, > + PMIC_GPIO_FUNC_INDEX_FUNC2, > + PMIC_GPIO_FUNC_INDEX_FUNC3, > + PMIC_GPIO_FUNC_INDEX_FUNC4, > + PMIC_GPIO_FUNC_INDEX_DTEST1, > + PMIC_GPIO_FUNC_INDEX_DTEST2, > + PMIC_GPIO_FUNC_INDEX_DTEST3, > + PMIC_GPIO_FUNC_INDEX_DTEST4, > +}; > + > +static const char *const spmi_pmic_gpio_functions[] = { > + [PMIC_GPIO_FUNC_INDEX_NORMAL] = PMIC_GPIO_FUNC_NORMAL, > + [PMIC_GPIO_FUNC_INDEX_PAIRED] = PMIC_GPIO_FUNC_PAIRED, > + [PMIC_GPIO_FUNC_INDEX_FUNC1] = PMIC_GPIO_FUNC_FUNC1, > + [PMIC_GPIO_FUNC_INDEX_FUNC2] = PMIC_GPIO_FUNC_FUNC2, > + [PMIC_GPIO_FUNC_INDEX_FUNC3] = PMIC_GPIO_FUNC_FUNC3, > + [PMIC_GPIO_FUNC_INDEX_FUNC4] = PMIC_GPIO_FUNC_FUNC4, > + [PMIC_GPIO_FUNC_INDEX_DTEST1] = PMIC_GPIO_FUNC_DTEST1, > + [PMIC_GPIO_FUNC_INDEX_DTEST2] = PMIC_GPIO_FUNC_DTEST2, > + [PMIC_GPIO_FUNC_INDEX_DTEST3] = PMIC_GPIO_FUNC_DTEST3, > + [PMIC_GPIO_FUNC_INDEX_DTEST4] = PMIC_GPIO_FUNC_DTEST4, > +}; > + > +/** > + * struct spmi_pmic_gpio_pad - keep current GPIO settings > + * @base: Address base in SPMI device. > + * @is_enabled: Set to false when GPIO should be put in high Z state. > + * @out_value: Cached pin output value > + * @have_buffer: Set to true if GPIO output could be configured in push-pull, > + * open-drain or open-source mode. > + * @output_enabled: Set to true if GPIO output logic is enabled. > + * @input_enabled: Set to true if GPIO input buffer logic is enabled. > + * @analog_pass: Set to true if GPIO is in analog-pass-through mode. > + * @lv_mv_type: Set to true if GPIO subtype is GPIO_LV(0x10) or > GPIO_MV(0x11). > + * @num_sources: Number of power-sources supported by this GPIO. > + * @power_source: Current power-source used. > + * @buffer_type: Push-pull, open-drain or open-source. > + * @pullup: Constant current which flow trough GPIO output buffer. > + * @strength: No, Low, Medium, High > + * @function: See spmi_pmic_gpio_functions[] > + * @atest: the ATEST selection for GPIO analog-pass-through mode > + * @dtest_buffer: the DTEST buffer selection for digital input mode. > + */ > +struct spmi_pmic_gpio_pad { > + u16 base; > + bool is_enabled; > + bool out_value; > + bool have_buffer; > + bool output_enabled; > + bool input_enabled; > + bool analog_pass; > + bool lv_mv_type; > + unsigned int num_sources; > + unsigned int power_source; > + unsigned int buffer_type; > + unsigned int pullup; > + unsigned int strength; > + unsigned int function; > + unsigned int atest; > + unsigned int dtest_buffer; > +}; > + > +struct qcom_spmi_pmic_gpio_data { > + struct udevice *dev; > + u32 pid; /* Peripheral ID on SPMI bus */ > + struct udevice *pmic; /* Reference to pmic device for read/write */ > + u32 pin_count; > + struct spmi_pmic_gpio_pad *pads; > +}; > + > +static int qcom_spmi_pmic_pinctrl_pinconf_set(struct udevice *dev, unsigned > int selector, > + unsigned int param, unsigned int > arg); > + > +static int spmi_pmic_gpio_read(struct qcom_spmi_pmic_gpio_data *plat, > + struct spmi_pmic_gpio_pad *pad, > + unsigned int addr) > +{ > + return pmic_reg_read(plat->pmic, pad->base + addr); > +} > + > +static int spmi_pmic_gpio_write(struct qcom_spmi_pmic_gpio_data *plat, > + struct spmi_pmic_gpio_pad *pad, > + unsigned int addr, unsigned int val) > +{ > + return pmic_reg_write(plat->pmic, pad->base + addr, val); > +} > + > +static void spmi_pmic_gpio_get_state(struct qcom_spmi_pmic_gpio_data *plat, > + struct spmi_pmic_gpio_pad *pad, > + char *buf, int size) > +{ > + int ret, val, function, cnt; > + > + static const char *const biases[] = { > + "pull-up 30uA", "pull-up 1.5uA", "pull-up 31.5uA", > + "pull-up 1.5uA + 30uA boost", "pull-down 10uA", "no pull" > + }; > + static const char *const buffer_types[] = { > + "push-pull", "open-drain", "open-source" > + }; > + static const char *const strengths[] = { > + "no", "low", "medium", "high" > + }; > + > + val = spmi_pmic_gpio_read(plat, pad, PMIC_GPIO_REG_EN_CTL); > + > + if (val < 0 || !(val >> PMIC_GPIO_REG_MASTER_EN_SHIFT)) { > + cnt = snprintf(buf, size, "disabled"); > + } else { > + if (pad->input_enabled) { > + ret = spmi_pmic_gpio_read(plat, pad, > PMIC_MPP_REG_RT_STS); > + if (ret < 0) > + return; > + > + ret &= PMIC_MPP_REG_RT_STS_VAL_MASK; > + pad->out_value = ret; > + } > + /* > + * For the non-LV/MV subtypes only 2 special functions are > + * available, offsetting the dtest function values by 2. > + */ > + function = pad->function; > + if (!pad->lv_mv_type && > + pad->function >= PMIC_GPIO_FUNC_INDEX_FUNC3) > + function += PMIC_GPIO_FUNC_INDEX_DTEST1 - > + PMIC_GPIO_FUNC_INDEX_FUNC3; > + > + if (pad->analog_pass) > + cnt = snprintf(buf, size, "analog-pass"); > + else > + cnt = snprintf(buf, size, "%-4s", > + pad->output_enabled ? "out" : "in"); > + buf += cnt; > + size -= cnt; > + > + snprintf(buf, size, " %-4s %-7s vin-%d %-27s %-10s %-7s > atest-%d dtest-%d", > + pad->out_value ? "high" : "low", > + spmi_pmic_gpio_functions[function], > + pad->power_source, > + biases[pad->pullup], > + buffer_types[pad->buffer_type], > + strengths[pad->strength], > + pad->atest, > + pad->dtest_buffer); > + } > +} > + > +static int qcom_spmi_pmic_gpio_set(struct qcom_spmi_pmic_gpio_data *plat, > + struct spmi_pmic_gpio_pad *pad) > +{ > + unsigned int val; > + int ret; > + > + val = pad->power_source << PMIC_GPIO_REG_VIN_SHIFT; > + > + ret = spmi_pmic_gpio_write(plat, pad, PMIC_GPIO_REG_DIG_VIN_CTL, val); > + if (ret < 0) > + return ret; > + > + val = pad->pullup << PMIC_GPIO_REG_PULL_SHIFT; > + > + ret = spmi_pmic_gpio_write(plat, pad, PMIC_GPIO_REG_DIG_PULL_CTL, val); > + if (ret < 0) > + return ret; > + > + val = pad->buffer_type << PMIC_GPIO_REG_OUT_TYPE_SHIFT; > + val |= pad->strength << PMIC_GPIO_REG_OUT_STRENGTH_SHIFT; > + > + ret = spmi_pmic_gpio_write(plat, pad, PMIC_GPIO_REG_DIG_OUT_CTL, val); > + if (ret < 0) > + return ret; > + > + if (pad->dtest_buffer == 0) { > + val = 0; > + } else { > + if (pad->lv_mv_type) { > + val = pad->dtest_buffer - 1; > + val |= PMIC_GPIO_LV_MV_DIG_IN_DTEST_EN; > + } else { > + val = BIT(pad->dtest_buffer - 1); > + } > + } > + ret = spmi_pmic_gpio_write(plat, pad, PMIC_GPIO_REG_DIG_IN_CTL, val); > + if (ret < 0) > + return ret; > + > + if (pad->analog_pass) > + val = PMIC_GPIO_MODE_ANALOG_PASS_THRU; > + else if (pad->output_enabled && pad->input_enabled) > + val = PMIC_GPIO_MODE_DIGITAL_INPUT_OUTPUT; > + else if (pad->output_enabled) > + val = PMIC_GPIO_MODE_DIGITAL_OUTPUT; > + else > + val = PMIC_GPIO_MODE_DIGITAL_INPUT; > + > + if (pad->lv_mv_type) { > + ret = spmi_pmic_gpio_write(plat, pad, > + PMIC_GPIO_REG_MODE_CTL, val); > + if (ret < 0) > + return ret; > + > + val = pad->atest - 1; > + ret = spmi_pmic_gpio_write(plat, pad, > + > PMIC_GPIO_REG_LV_MV_ANA_PASS_THRU_SEL, > + val); > + if (ret < 0) > + return ret; > + > + val = pad->out_value > + << PMIC_GPIO_LV_MV_OUTPUT_INVERT_SHIFT; > + val |= pad->function > + & PMIC_GPIO_LV_MV_OUTPUT_SOURCE_SEL_MASK; > + ret = spmi_pmic_gpio_write(plat, pad, > + > PMIC_GPIO_REG_LV_MV_DIG_OUT_SOURCE_CTL, > + val); > + if (ret < 0) > + return ret; > + } else { > + val = val << PMIC_GPIO_REG_MODE_DIR_SHIFT; > + val |= pad->function << PMIC_GPIO_REG_MODE_FUNCTION_SHIFT; > + val |= pad->out_value & PMIC_GPIO_REG_MODE_VALUE_SHIFT; > + > + ret = spmi_pmic_gpio_write(plat, pad, PMIC_GPIO_REG_MODE_CTL, > val); > + if (ret < 0) > + return ret; > + } > + > + val = pad->is_enabled << PMIC_GPIO_REG_MASTER_EN_SHIFT; > + > + ret = spmi_pmic_gpio_write(plat, pad, PMIC_GPIO_REG_EN_CTL, val); > + if (ret) > + return ret; > + > + return 0; > +} > + > +static int spmi_pmic_gpio_populate(struct qcom_spmi_pmic_gpio_data *plat, > + struct spmi_pmic_gpio_pad *pad) > +{ > + int type, subtype, val, dir; > + > + type = spmi_pmic_gpio_read(plat, pad, PMIC_GPIO_REG_TYPE); > + if (type < 0) > + return type; > + > + if (type != PMIC_GPIO_TYPE) { > + dev_err(plat->dev, "incorrect block type 0x%x at 0x%x\n", > + type, pad->base); > + return -ENODEV; > + } > + > + subtype = spmi_pmic_gpio_read(plat, pad, PMIC_GPIO_REG_SUBTYPE); > + if (subtype < 0) > + return subtype; > + > + switch (subtype) { > + case PMIC_GPIO_SUBTYPE_GPIO_4CH: > + pad->have_buffer = true; > + fallthrough; > + case PMIC_GPIO_SUBTYPE_GPIOC_4CH: > + pad->num_sources = 4; > + break; > + case PMIC_GPIO_SUBTYPE_GPIO_8CH: > + pad->have_buffer = true; > + fallthrough; > + case PMIC_GPIO_SUBTYPE_GPIOC_8CH: > + pad->num_sources = 8; > + break; > + case PMIC_GPIO_SUBTYPE_GPIO_LV: > + pad->num_sources = 1; > + pad->have_buffer = true; > + pad->lv_mv_type = true; > + break; > + case PMIC_GPIO_SUBTYPE_GPIO_MV: > + pad->num_sources = 2; > + pad->have_buffer = true; > + pad->lv_mv_type = true; > + break; > + case PMIC_GPIO_SUBTYPE_GPIO_LV_VIN2: > + pad->num_sources = 2; > + pad->have_buffer = true; > + pad->lv_mv_type = true; > + break; > + case PMIC_GPIO_SUBTYPE_GPIO_MV_VIN3: > + pad->num_sources = 3; > + pad->have_buffer = true; > + pad->lv_mv_type = true; > + break; > + default: > + dev_err(plat->dev, "unknown GPIO type 0x%x\n", subtype); > + return -ENODEV; > + } > + > + if (pad->lv_mv_type) { > + val = spmi_pmic_gpio_read(plat, pad, > + > PMIC_GPIO_REG_LV_MV_DIG_OUT_SOURCE_CTL); > + if (val < 0) > + return val; > + > + pad->out_value = !!(val & PMIC_GPIO_LV_MV_OUTPUT_INVERT); > + pad->function = val & PMIC_GPIO_LV_MV_OUTPUT_SOURCE_SEL_MASK; > + > + val = spmi_pmic_gpio_read(plat, pad, PMIC_GPIO_REG_MODE_CTL); > + if (val < 0) > + return val; > + > + dir = val & PMIC_GPIO_REG_LV_MV_MODE_DIR_MASK; > + } else { > + val = spmi_pmic_gpio_read(plat, pad, PMIC_GPIO_REG_MODE_CTL); > + if (val < 0) > + return val; > + > + pad->out_value = val & PMIC_GPIO_REG_MODE_VALUE_SHIFT; > + > + dir = val >> PMIC_GPIO_REG_MODE_DIR_SHIFT; > + dir &= PMIC_GPIO_REG_MODE_DIR_MASK; > + pad->function = val >> PMIC_GPIO_REG_MODE_FUNCTION_SHIFT; > + pad->function &= PMIC_GPIO_REG_MODE_FUNCTION_MASK; > + } > + > + switch (dir) { > + case PMIC_GPIO_MODE_DIGITAL_INPUT: > + pad->input_enabled = true; > + pad->output_enabled = false; > + break; > + case PMIC_GPIO_MODE_DIGITAL_OUTPUT: > + pad->input_enabled = false; > + pad->output_enabled = true; > + break; > + case PMIC_GPIO_MODE_DIGITAL_INPUT_OUTPUT: > + pad->input_enabled = true; > + pad->output_enabled = true; > + break; > + case PMIC_GPIO_MODE_ANALOG_PASS_THRU: > + if (!pad->lv_mv_type) > + return -ENODEV; > + pad->analog_pass = true; > + break; > + default: > + dev_err(plat->dev, "unknown GPIO direction\n"); > + return -ENODEV; > + } > + > + val = spmi_pmic_gpio_read(plat, pad, PMIC_GPIO_REG_DIG_VIN_CTL); > + if (val < 0) > + return val; > + > + pad->power_source = val >> PMIC_GPIO_REG_VIN_SHIFT; > + pad->power_source &= PMIC_GPIO_REG_VIN_MASK; > + > + val = spmi_pmic_gpio_read(plat, pad, PMIC_GPIO_REG_DIG_PULL_CTL); > + if (val < 0) > + return val; > + > + pad->pullup = val >> PMIC_GPIO_REG_PULL_SHIFT; > + pad->pullup &= PMIC_GPIO_REG_PULL_MASK; > + > + val = spmi_pmic_gpio_read(plat, pad, PMIC_GPIO_REG_DIG_IN_CTL); > + if (val < 0) > + return val; > + > + if (pad->lv_mv_type && (val & PMIC_GPIO_LV_MV_DIG_IN_DTEST_EN)) > + pad->dtest_buffer = > + (val & PMIC_GPIO_LV_MV_DIG_IN_DTEST_SEL_MASK) + 1; > + else if (!pad->lv_mv_type) > + pad->dtest_buffer = ffs(val); > + else > + pad->dtest_buffer = 0; > + > + val = spmi_pmic_gpio_read(plat, pad, PMIC_GPIO_REG_DIG_OUT_CTL); > + if (val < 0) > + return val; > + > + pad->strength = val >> PMIC_GPIO_REG_OUT_STRENGTH_SHIFT; > + pad->strength &= PMIC_GPIO_REG_OUT_STRENGTH_MASK; > + > + pad->buffer_type = val >> PMIC_GPIO_REG_OUT_TYPE_SHIFT; > + pad->buffer_type &= PMIC_GPIO_REG_OUT_TYPE_MASK; > + > + if (pad->lv_mv_type) { > + val = spmi_pmic_gpio_read(plat, pad, > + > PMIC_GPIO_REG_LV_MV_ANA_PASS_THRU_SEL); > + if (val < 0) > + return val; > + > + pad->atest = (val & PMIC_GPIO_LV_MV_ANA_MUX_SEL_MASK) + 1; > + } > + > + /* Pin could be disabled with PIN_CONFIG_BIAS_HIGH_IMPEDANCE */ > + pad->is_enabled = true; > + return 0; > +} > + > +static int qcom_spmi_pmic_gpio_set_flags(struct udevice *dev, unsigned int > offset, ulong flags) > +{ > + struct qcom_spmi_pmic_gpio_data *plat = dev_get_plat(dev); > + struct spmi_pmic_gpio_pad *pad; > + > + if (offset >= plat->pin_count) > + return -EINVAL; > + > + pad = &plat->pads[offset]; > + > + pad->input_enabled = flags & GPIOD_IS_IN; > + pad->output_enabled = flags & GPIOD_IS_OUT; > + > + if (pad->output_enabled) { > + pad->out_value = flags & GPIOD_IS_OUT_ACTIVE; > + > + if ((flags & GPIOD_OPEN_DRAIN) && pad->have_buffer) > + pad->buffer_type = PMIC_GPIO_OUT_BUF_OPEN_DRAIN_NMOS; > + else if ((flags & GPIOD_OPEN_SOURCE) && pad->have_buffer) > + pad->buffer_type = PMIC_GPIO_OUT_BUF_OPEN_DRAIN_PMOS; > + else > + pad->buffer_type = PMIC_GPIO_OUT_BUF_CMOS; > + } > + > + if (flags & GPIOD_PULL_UP) > + pad->pullup = PMIC_GPIO_PULL_UP_30; > + else if (flags & GPIOD_PULL_DOWN) > + pad->pullup = PMIC_GPIO_PULL_DOWN; > + > + return qcom_spmi_pmic_gpio_set(plat, pad); > +} > + > +static int qcom_spmi_pmic_gpio_get_flags(struct udevice *dev, unsigned int > offset, > + ulong *flagsp) > +{ > + struct qcom_spmi_pmic_gpio_data *plat = dev_get_plat(dev); > + struct spmi_pmic_gpio_pad *pad; > + ulong flags = 0; > + > + if (offset >= plat->pin_count) > + return -EINVAL; > + > + pad = &plat->pads[offset]; > + > + if (pad->input_enabled) > + flags |= GPIOD_IS_IN; > + > + if (pad->output_enabled) { > + flags |= GPIOD_IS_OUT; > + > + if (pad->out_value) > + flags |= GPIOD_IS_OUT_ACTIVE; > + > + switch (pad->buffer_type) { > + case PMIC_GPIO_OUT_BUF_OPEN_DRAIN_NMOS: > + flags |= GPIOD_OPEN_DRAIN; > + break; > + case PMIC_GPIO_OUT_BUF_OPEN_DRAIN_PMOS: > + flags |= GPIOD_OPEN_SOURCE; > + break; > + } > + } > + > + if (pad->pullup == PMIC_GPIO_PULL_DOWN) > + flags |= GPIOD_PULL_DOWN; > + else if (pad->pullup != PMIC_GPIO_PULL_DISABLE) > + flags |= GPIOD_PULL_UP; > + > + if (pad->analog_pass) > + flags |= GPIOD_IS_AF; > + > + *flagsp = flags; > + > + return 0; > +} > + > +static int qcom_spmi_pmic_gpio_get_value(struct udevice *dev, unsigned int > offset) > +{ > + struct qcom_spmi_pmic_gpio_data *plat = dev_get_plat(dev); > + struct spmi_pmic_gpio_pad *pad; > + int ret; > + > + if (offset >= plat->pin_count) > + return -EINVAL; > + > + pad = &plat->pads[offset]; > + > + if (!pad->is_enabled) > + return -EINVAL; > + > + if (pad->input_enabled) { > + ret = spmi_pmic_gpio_read(plat, pad, PMIC_MPP_REG_RT_STS); > + if (ret < 0) > + return ret; > + > + pad->out_value = ret & PMIC_MPP_REG_RT_STS_VAL_MASK; > + } > + > + return !!pad->out_value; > +} > + > +static int qcom_spmi_pmic_gpio_get_function(struct udevice *dev, unsigned > int offset) > +{ > + struct qcom_spmi_pmic_gpio_data *plat = dev_get_plat(dev); > + struct spmi_pmic_gpio_pad *pad; > + int val; > + > + if (offset >= plat->pin_count) > + return GPIOF_UNKNOWN; > + > + pad = &plat->pads[offset]; > + > + val = spmi_pmic_gpio_read(plat, pad, PMIC_GPIO_REG_EN_CTL); > + if (!(val >> PMIC_GPIO_REG_MASTER_EN_SHIFT)) > + return GPIOF_UNKNOWN; > + else if (pad->analog_pass) > + return GPIOF_FUNC; > + else if (pad->output_enabled) > + return GPIOF_OUTPUT; > + > + return GPIOF_INPUT; > +} > + > +static int qcom_spmi_pmic_gpio_xlate(struct udevice *dev, struct gpio_desc > *desc, > + struct ofnode_phandle_args *args) > +{ > + struct gpio_dev_priv *uc_priv = dev_get_uclass_priv(dev); > + > + if (args->args_count < 1) > + return -EINVAL; > + > + /* GPIOs in DT are 1-based */ > + desc->offset = args->args[0] - 1; > + if (desc->offset >= uc_priv->gpio_count) > + return -EINVAL; > + > + if (args->args_count < 2) > + return 0; > + > + desc->flags = gpio_flags_xlate(args->args[1]); > + > + return 0; > +} > + > +static const struct dm_gpio_ops qcom_spmi_pmic_gpio_ops = { > + .set_flags = qcom_spmi_pmic_gpio_set_flags, > + .get_flags = qcom_spmi_pmic_gpio_get_flags, > + .get_value = qcom_spmi_pmic_gpio_get_value, > + .get_function = qcom_spmi_pmic_gpio_get_function, > + .xlate = qcom_spmi_pmic_gpio_xlate, > +}; > + > +static int qcom_spmi_pmic_gpio_bind(struct udevice *dev) > +{ > + struct qcom_spmi_pmic_gpio_data *plat = dev_get_plat(dev); > + struct udevice *child; > + struct driver *drv; > + int ret; > + > + drv = lists_driver_lookup_name("qcom_spmi_pmic_pinctrl"); > + if (!drv) { > + log_warning("Cannot find driver '%s'\n", > "qcom_spmi_pmic_pinctrl"); > + return -ENOENT; > + } > + > + /* Bind the GPIO driver as a child of the PMIC. */ > + ret = device_bind_with_driver_data(dev, drv, > + dev->name, > + 0, dev_ofnode(dev), &child); > + if (ret) > + return log_msg_ret("bind", ret); > + > + dev_set_plat(child, plat); > + > + return 0; > +} > + > +static int qcom_spmi_pmic_gpio_probe(struct udevice *dev) > +{ > + struct gpio_dev_priv *uc_priv = dev_get_uclass_priv(dev); > + struct qcom_spmi_pmic_gpio_data *plat = dev_get_plat(dev); > + struct ofnode_phandle_args args; > + int i, ret; > + u64 pid; > + > + plat->dev = dev; > + plat->pmic = dev->parent; > + > + pid = dev_read_addr(dev); > + if (pid == FDT_ADDR_T_NONE) > + return log_msg_ret("bad address", -EINVAL); > + > + plat->pid = pid; > + > + /* > + * Parse basic GPIO count specified via the gpio-ranges property > + * as specified in upstream devicetrees > + */ > + ret = ofnode_parse_phandle_with_args(dev_ofnode(dev), "gpio-ranges", > + NULL, 3, 0, &args); > + if (ret) > + return log_msg_ret("gpio-ranges", ret); > + > + plat->pin_count = min_t(u32, args.args[2], PMIC_MAX_GPIOS); > + > + uc_priv->gpio_count = plat->pin_count; > + > + uc_priv->bank_name = strchr(dev_read_string(dev, "compatible"), ','); > + if (uc_priv->bank_name) > + uc_priv->bank_name += 1; /* skip the , */ > + else > + uc_priv->bank_name = dev->name; > + > + plat->pads = calloc(plat->pin_count, sizeof(struct spmi_pmic_gpio_pad)); > + if (!plat->pads) > + return -ENOMEM; > + > + for (i = 0; i < plat->pin_count; ++i) { > + struct spmi_pmic_gpio_pad *pad = &plat->pads[i]; > + > + pad->base = plat->pid + i * PMIC_GPIO_ADDRESS_RANGE; > + > + ret = spmi_pmic_gpio_populate(plat, pad); > + if (ret < 0) > + return ret; > + } > + > + return 0; > +} > + > +static const struct udevice_id qcom_spmi_pmic_gpio_ids[] = { > + { .compatible = "qcom,pm8550b-gpio" }, > + { .compatible = "qcom,pm8550ve-gpio" }, > + { .compatible = "qcom,pm8550vs-gpio" }, > + { .compatible = "qcom,pmk8550-gpio" }, > + { .compatible = "qcom,pmr735d-gpio" }, > + { } > +}; > + > +U_BOOT_DRIVER(qcom_spmi_pmic_gpio) = { > + .name = "qcom_spmi_pmic_gpio", > + .id = UCLASS_GPIO, > + .of_match = qcom_spmi_pmic_gpio_ids, > + .bind = qcom_spmi_pmic_gpio_bind, > + .probe = qcom_spmi_pmic_gpio_probe, > + .ops = &qcom_spmi_pmic_gpio_ops, > + .plat_auto = sizeof(struct qcom_spmi_pmic_gpio_data), > + .flags = DM_FLAG_ALLOC_PDATA, > +}; > + > +/* Qualcomm specific pin configurations */ > +#define PMIC_GPIO_CONF_PULL_UP (PIN_CONFIG_END + 1) > +#define PMIC_GPIO_CONF_STRENGTH (PIN_CONFIG_END + 2) > +#define PMIC_GPIO_CONF_ATEST (PIN_CONFIG_END + 3) > +#define PMIC_GPIO_CONF_ANALOG_PASS (PIN_CONFIG_END + 4) > +#define PMIC_GPIO_CONF_DTEST_BUFFER (PIN_CONFIG_END + 5) > + > +static const struct pinconf_param qcom_spmi_pmic_pinctrl_conf_params[] = { > + { "drive-push-pull", PIN_CONFIG_DRIVE_PUSH_PULL, 0 }, > + { "drive-open-drain", PIN_CONFIG_DRIVE_OPEN_DRAIN, 0 }, > + { "drive-open-source", PIN_CONFIG_DRIVE_OPEN_SOURCE, 0 }, > + { "bias-disable", PIN_CONFIG_BIAS_DISABLE, 0 }, > + { "bias-pull-up", PIN_CONFIG_BIAS_PULL_UP, PMIC_GPIO_PULL_UP_30 }, > + { "bias-pull-down", PIN_CONFIG_BIAS_PULL_UP, 0 }, > + { "bias-high-impedance", PIN_CONFIG_BIAS_HIGH_IMPEDANCE, 0 }, > + { "power-source", PIN_CONFIG_POWER_SOURCE, 0 }, > + { "input-disable", PIN_CONFIG_INPUT_ENABLE, 0 }, > + { "input-enable", PIN_CONFIG_INPUT_ENABLE, 1 }, > + { "output-disable", PIN_CONFIG_OUTPUT_ENABLE, 0 }, > + { "output-enable", PIN_CONFIG_OUTPUT_ENABLE, 1 }, > + { "output-high", PIN_CONFIG_OUTPUT, 1 }, > + { "output-low", PIN_CONFIG_OUTPUT, 0 }, > + { "qcom,pull-up-strength", PMIC_GPIO_CONF_PULL_UP, 0}, > + { "qcom,drive-strength", PMIC_GPIO_CONF_STRENGTH, 0}, > + { "qcom,atest", PMIC_GPIO_CONF_ATEST, 0}, > + { "qcom,analog-pass", PMIC_GPIO_CONF_ANALOG_PASS, 0}, > + { "qcom,dtest-buffer", PMIC_GPIO_CONF_DTEST_BUFFER, 0}, > +}; > + > +static int qcom_spmi_pmic_pinctrl_get_pins_count(struct udevice *dev) > +{ > + struct qcom_spmi_pmic_gpio_data *plat = dev_get_plat(dev); > + > + return plat->pin_count; > +} > + > +static const char *qcom_spmi_pmic_pinctrl_get_pin_name(struct udevice *dev, > + unsigned int selector) > +{ > + static char name[8]; > + > + /* DT indexes from 1 */ > + snprintf(name, sizeof(name), "gpio%u", selector + 1); > + > + return name; > +} > + > +static int qcom_spmi_pmic_pinctrl_get_pin_muxing(struct udevice *dev, > + unsigned int selector, > + char *buf, int size) > +{ > + struct qcom_spmi_pmic_gpio_data *plat = dev_get_plat(dev); > + struct spmi_pmic_gpio_pad *pad; > + > + if (selector >= plat->pin_count) > + return -EINVAL; > + > + pad = &plat->pads[selector]; > + > + spmi_pmic_gpio_get_state(plat, pad, buf, size); > + > + return 0; > +} > + > +static int qcom_spmi_pmic_pinctrl_pinconf_set(struct udevice *dev, unsigned > int selector, > + unsigned int param, unsigned int > arg) > +{ > + struct qcom_spmi_pmic_gpio_data *plat = dev_get_plat(dev); > + struct spmi_pmic_gpio_pad *pad; > + > + if (selector >= plat->pin_count) > + return -EINVAL; > + > + pad = &plat->pads[selector]; > + pad->is_enabled = true; > + > + switch (param) { > + case PIN_CONFIG_DRIVE_PUSH_PULL: > + pad->buffer_type = PMIC_GPIO_OUT_BUF_CMOS; > + break; > + case PIN_CONFIG_DRIVE_OPEN_DRAIN: > + if (!pad->have_buffer) > + return -EINVAL; > + pad->buffer_type = PMIC_GPIO_OUT_BUF_OPEN_DRAIN_NMOS; > + break; > + case PIN_CONFIG_DRIVE_OPEN_SOURCE: > + if (!pad->have_buffer) > + return -EINVAL; > + pad->buffer_type = PMIC_GPIO_OUT_BUF_OPEN_DRAIN_PMOS; > + break; > + case PIN_CONFIG_BIAS_DISABLE: > + pad->pullup = PMIC_GPIO_PULL_DISABLE; > + break; > + case PIN_CONFIG_BIAS_PULL_UP: > + pad->pullup = PMIC_GPIO_PULL_UP_30; > + break; > + case PIN_CONFIG_BIAS_PULL_DOWN: > + if (arg) > + pad->pullup = PMIC_GPIO_PULL_DOWN; > + else > + pad->pullup = PMIC_GPIO_PULL_DISABLE; > + break; > + case PIN_CONFIG_BIAS_HIGH_IMPEDANCE: > + pad->is_enabled = false; > + break; > + case PIN_CONFIG_POWER_SOURCE: > + if (arg >= pad->num_sources) > + return -EINVAL; > + pad->power_source = arg; > + break; > + case PIN_CONFIG_INPUT_ENABLE: > + pad->input_enabled = arg ? true : false; > + break; > + case PIN_CONFIG_OUTPUT_ENABLE: > + pad->output_enabled = arg ? true : false; > + break; > + case PIN_CONFIG_OUTPUT: > + pad->output_enabled = true; > + pad->out_value = arg; > + break; > + case PMIC_GPIO_CONF_PULL_UP: > + if (arg > PMIC_GPIO_PULL_UP_1P5_30) > + return -EINVAL; > + pad->pullup = arg; > + break; > + case PMIC_GPIO_CONF_STRENGTH: > + if (arg > PMIC_GPIO_STRENGTH_LOW) > + return -EINVAL; > + switch (arg) { > + case PMIC_GPIO_STRENGTH_HIGH: > + pad->strength = PMIC_GPIO_OUT_STRENGTH_HIGH; > + break; > + case PMIC_GPIO_STRENGTH_LOW: > + pad->strength = PMIC_GPIO_OUT_STRENGTH_LOW; > + break; > + default: > + pad->strength = arg; > + break; > + } > + break; > + case PMIC_GPIO_CONF_ATEST: > + if (!pad->lv_mv_type || arg > 4) > + return -EINVAL; > + pad->atest = arg; > + break; > + case PMIC_GPIO_CONF_ANALOG_PASS: > + if (!pad->lv_mv_type) > + return -EINVAL; > + pad->analog_pass = true; > + break; > + case PMIC_GPIO_CONF_DTEST_BUFFER: > + if (arg > 4) > + return -EINVAL; > + pad->dtest_buffer = arg; > + break; > + default: > + return -EINVAL; > + } > + > + return qcom_spmi_pmic_gpio_set(plat, pad); > +} > + > +static const char *qcom_spmi_pmic_pinctrl_get_function_name(struct udevice > *dev, > + unsigned int > selector) > +{ > + if (selector >= ARRAY_SIZE(spmi_pmic_gpio_functions)) > + return NULL; > + > + return spmi_pmic_gpio_functions[selector]; > +} > + > +static int qcom_spmi_pmic_pinctrl_get_functions_count(struct udevice *dev) > +{ > + return ARRAY_SIZE(spmi_pmic_gpio_functions); > +} > + > +static int qcom_spmi_pmic_pinctrl_pinmux_set_mux(struct udevice *dev, > unsigned int selector, > + unsigned int function) > +{ > + struct qcom_spmi_pmic_gpio_data *plat = dev_get_plat(dev); > + struct spmi_pmic_gpio_pad *pad; > + unsigned int val; > + int ret; > + > + if (selector >= plat->pin_count) > + return -EINVAL; > + > + pad = &plat->pads[selector]; > + > + /* > + * Non-LV/MV subtypes only support 2 special functions, > + * offsetting the dtestx function values by 2 > + */ > + if (!pad->lv_mv_type) { > + if (function == PMIC_GPIO_FUNC_INDEX_FUNC3 || > + function == PMIC_GPIO_FUNC_INDEX_FUNC4) { > + pr_err("LV/MV subtype doesn't have func3/func4\n"); > + return -EINVAL; > + } > + if (function >= PMIC_GPIO_FUNC_INDEX_DTEST1) > + function -= (PMIC_GPIO_FUNC_INDEX_DTEST1 - > + PMIC_GPIO_FUNC_INDEX_FUNC3); > + } > + > + pad->function = function; > + > + if (pad->analog_pass) > + val = PMIC_GPIO_MODE_ANALOG_PASS_THRU; > + else if (pad->output_enabled && pad->input_enabled) > + val = PMIC_GPIO_MODE_DIGITAL_INPUT_OUTPUT; > + else if (pad->output_enabled) > + val = PMIC_GPIO_MODE_DIGITAL_OUTPUT; > + else > + val = PMIC_GPIO_MODE_DIGITAL_INPUT; > + > + if (pad->lv_mv_type) { > + ret = spmi_pmic_gpio_write(plat, pad, > + PMIC_GPIO_REG_MODE_CTL, val); > + if (ret < 0) > + return ret; > + > + val = pad->atest - 1; > + ret = spmi_pmic_gpio_write(plat, pad, > + > PMIC_GPIO_REG_LV_MV_ANA_PASS_THRU_SEL, > + val); > + if (ret < 0) > + return ret; > + > + val = pad->out_value > + << PMIC_GPIO_LV_MV_OUTPUT_INVERT_SHIFT; > + val |= pad->function > + & PMIC_GPIO_LV_MV_OUTPUT_SOURCE_SEL_MASK; > + ret = spmi_pmic_gpio_write(plat, pad, > + > PMIC_GPIO_REG_LV_MV_DIG_OUT_SOURCE_CTL, > + val); > + if (ret < 0) > + return ret; > + } else { > + val = val << PMIC_GPIO_REG_MODE_DIR_SHIFT; > + val |= pad->function << PMIC_GPIO_REG_MODE_FUNCTION_SHIFT; > + val |= pad->out_value & PMIC_GPIO_REG_MODE_VALUE_SHIFT; > + > + ret = spmi_pmic_gpio_write(plat, pad, PMIC_GPIO_REG_MODE_CTL, > val); > + if (ret < 0) > + return ret; > + } > + > + val = pad->is_enabled << PMIC_GPIO_REG_MASTER_EN_SHIFT; > + > + return spmi_pmic_gpio_write(plat, pad, PMIC_GPIO_REG_EN_CTL, val); > +} > + > +struct pinctrl_ops qcom_spmi_pmic_pinctrl_ops = { > + .get_pins_count = qcom_spmi_pmic_pinctrl_get_pins_count, > + .get_pin_name = qcom_spmi_pmic_pinctrl_get_pin_name, > + .set_state = pinctrl_generic_set_state, > + .pinconf_num_params = ARRAY_SIZE(qcom_spmi_pmic_pinctrl_conf_params), > + .pinconf_params = qcom_spmi_pmic_pinctrl_conf_params, > + .pinconf_set = qcom_spmi_pmic_pinctrl_pinconf_set, > + .get_function_name = qcom_spmi_pmic_pinctrl_get_function_name, > + .get_functions_count = qcom_spmi_pmic_pinctrl_get_functions_count, > + .pinmux_set = qcom_spmi_pmic_pinctrl_pinmux_set_mux, > + .get_pin_muxing = qcom_spmi_pmic_pinctrl_get_pin_muxing, > +}; > + > +U_BOOT_DRIVER(qcom_spmi_pmic_pinctrl) = { > + .name = "qcom_spmi_pmic_pinctrl", > + .id = UCLASS_PINCTRL, > + .ops = &qcom_spmi_pmic_pinctrl_ops, > +}; > > -- > 2.34.1

