diff options
Diffstat (limited to 'drivers/power/pmic/pmic_qcom.c')
-rw-r--r-- | drivers/power/pmic/pmic_qcom.c | 86 |
1 files changed, 86 insertions, 0 deletions
diff --git a/drivers/power/pmic/pmic_qcom.c b/drivers/power/pmic/pmic_qcom.c new file mode 100644 index 00000000000..ad8daf43f06 --- /dev/null +++ b/drivers/power/pmic/pmic_qcom.c @@ -0,0 +1,86 @@ +// SPDX-License-Identifier: GPL-2.0+ +/* + * Qualcomm generic pmic driver + * + * (C) Copyright 2015 Mateusz Kulikowski <mateusz.kulikowski@gmail.com> + */ +#include <common.h> +#include <dm.h> +#include <power/pmic.h> +#include <spmi/spmi.h> + +#define PID_SHIFT 8 +#define PID_MASK (0xFF << PID_SHIFT) +#define REG_MASK 0xFF + +struct pmic_qcom_priv { + uint32_t usid; /* Slave ID on SPMI bus */ +}; + +static int pmic_qcom_reg_count(struct udevice *dev) +{ + return 0xFFFF; +} + +static int pmic_qcom_write(struct udevice *dev, uint reg, const uint8_t *buff, + int len) +{ + struct pmic_qcom_priv *priv = dev_get_priv(dev); + + if (len != 1) + return -EINVAL; + + return spmi_reg_write(dev->parent, priv->usid, + (reg & PID_MASK) >> PID_SHIFT, reg & REG_MASK, + *buff); +} + +static int pmic_qcom_read(struct udevice *dev, uint reg, uint8_t *buff, int len) +{ + struct pmic_qcom_priv *priv = dev_get_priv(dev); + int val; + + if (len != 1) + return -EINVAL; + + val = spmi_reg_read(dev->parent, priv->usid, + (reg & PID_MASK) >> PID_SHIFT, reg & REG_MASK); + + if (val < 0) + return val; + *buff = val; + return 0; +} + +static struct dm_pmic_ops pmic_qcom_ops = { + .reg_count = pmic_qcom_reg_count, + .read = pmic_qcom_read, + .write = pmic_qcom_write, +}; + +static const struct udevice_id pmic_qcom_ids[] = { + { .compatible = "qcom,spmi-pmic" }, + { } +}; + +static int pmic_qcom_probe(struct udevice *dev) +{ + struct pmic_qcom_priv *priv = dev_get_priv(dev); + + priv->usid = dev_read_addr(dev); + + if (priv->usid == FDT_ADDR_T_NONE) + return -EINVAL; + + return 0; +} + +U_BOOT_DRIVER(pmic_qcom) = { + .name = "pmic_qcom", + .id = UCLASS_PMIC, + .of_match = pmic_qcom_ids, + .bind = dm_scan_fdt_dev, + .probe = pmic_qcom_probe, + .ops = &pmic_qcom_ops, + .priv_auto = sizeof(struct pmic_qcom_priv), +}; |