kernel/0019-feat-support-mpq8785-to-set-npu-voltage.patch
2025-01-02 22:30:57 -05:00

1339 lines
39 KiB
Diff

From 89f052a47872cabdb36d1faae4566e4bcb4183f1 Mon Sep 17 00:00:00 2001
From: yangwei1 <yangwei1@eswincomputing.com>
Date: Thu, 23 May 2024 15:15:28 +0800
Subject: [PATCH 019/222] feat:support mpq8785 to set npu voltage
Changelogs:
---
arch/riscv/boot/dts/eswin/eic7700-evb-a2.dts | 13 +-
arch/riscv/boot/dts/eswin/eic7700-evb.dts | 11 +-
arch/riscv/configs/win2030_defconfig | 1 +
drivers/regulator/Kconfig | 6 +
drivers/regulator/Makefile | 2 +-
drivers/regulator/mpq8785.c | 1217 ++++++++++++++++++
6 files changed, 1239 insertions(+), 11 deletions(-)
create mode 100644 drivers/regulator/mpq8785.c
diff --git a/arch/riscv/boot/dts/eswin/eic7700-evb-a2.dts b/arch/riscv/boot/dts/eswin/eic7700-evb-a2.dts
index 00154660ab7c..cddfc166e851 100644
--- a/arch/riscv/boot/dts/eswin/eic7700-evb-a2.dts
+++ b/arch/riscv/boot/dts/eswin/eic7700-evb-a2.dts
@@ -724,18 +724,19 @@ d0_codec1_endpoint: endpoint {
&d0_i2c1 {
/* mpq8785 */
status = "okay";
- eswin,syscfg = <&d0_sys_con 0x3C0 15>;
- iic_hold_time = <0x40>;
mpq8785@10 {
compatible = "mps,mpq8785";
reg = <0x10>;
+ eswin,regulator_default-microvolt=<1000000>;
+ eswin,regulator_label = "supply vdd1", "npu vdd1", "npu current1", "npu temperature1";
regulators{
npu_vcc1:npu_svcc{
regulator-name="NPU_SVCC";
- regulator-min-microvolt=<100000>;
- regulator-max-microvolt=<1600000>;
- regulator-min-microamp=<50000000>;
- regulator-max-microamp=<90000000>;
+ regulator-min-microvolt=<700000>;
+ regulator-max-microvolt=<1100000>;
+ regulator-min-microamp=<20000000>;
+ regulator-max-microamp=<40000000>;
+ regulator-ov-protection-microvolt=<1100000>;
regulator-always-on;
};
};
diff --git a/arch/riscv/boot/dts/eswin/eic7700-evb.dts b/arch/riscv/boot/dts/eswin/eic7700-evb.dts
index 7a9883a89dc3..34bc531d58f9 100644
--- a/arch/riscv/boot/dts/eswin/eic7700-evb.dts
+++ b/arch/riscv/boot/dts/eswin/eic7700-evb.dts
@@ -760,13 +760,16 @@ &d0_aon_i2c1 {
mpq8785@10 {
compatible = "mps,mpq8785";
reg = <0x10>;
+ eswin,regulator_default-microvolt=<1000000>;
+ eswin,regulator_label = "supply vdd1", "npu vdd1", "npu current1", "npu temperature1";
regulators{
npu_vcc1:npu_svcc{
regulator-name="NPU_SVCC";
- regulator-min-microvolt=<100000>;
- regulator-max-microvolt=<1600000>;
- regulator-min-microamp=<50000000>;
- regulator-max-microamp=<90000000>;
+ regulator-min-microvolt=<700000>;
+ regulator-max-microvolt=<1100000>;
+ regulator-min-microamp=<20000000>;
+ regulator-max-microamp=<40000000>;
+ regulator-ov-protection-microvolt=<1100000>;
regulator-always-on;
};
};
diff --git a/arch/riscv/configs/win2030_defconfig b/arch/riscv/configs/win2030_defconfig
index 5d25e67b935d..7f8075d825bc 100644
--- a/arch/riscv/configs/win2030_defconfig
+++ b/arch/riscv/configs/win2030_defconfig
@@ -150,6 +150,7 @@ CONFIG_SENSORS_INA2XX=y
CONFIG_WATCHDOG=y
CONFIG_DW_WATCHDOG=y
CONFIG_REGULATOR=y
+CONFIG_REGULATOR_MPQ8785=y
# CONFIG_MEDIA_CEC_SUPPORT is not set
CONFIG_MEDIA_SUPPORT=y
CONFIG_V4L_PLATFORM_DRIVERS=y
diff --git a/drivers/regulator/Kconfig b/drivers/regulator/Kconfig
index 965d4f0c18a6..512ca2e53f1b 100644
--- a/drivers/regulator/Kconfig
+++ b/drivers/regulator/Kconfig
@@ -795,6 +795,12 @@ config REGULATOR_MPQ7920
This driver supports the control of different power rails of device
through regulator interface.
+config REGULATOR_MPQ8785
+ tristate "mps,mpq8785"
+ help
+ This driver provides support for the voltage regulators on the
+ eswin evb.
+
config REGULATOR_MT6311
tristate "MediaTek MT6311 PMIC"
depends on I2C
diff --git a/drivers/regulator/Makefile b/drivers/regulator/Makefile
index 23074714a81a..a0b29a60d7f8 100644
--- a/drivers/regulator/Makefile
+++ b/drivers/regulator/Makefile
@@ -195,5 +195,5 @@ obj-$(CONFIG_REGULATOR_WM831X) += wm831x-ldo.o
obj-$(CONFIG_REGULATOR_WM8350) += wm8350-regulator.o
obj-$(CONFIG_REGULATOR_WM8400) += wm8400-regulator.o
obj-$(CONFIG_REGULATOR_WM8994) += wm8994-regulator.o
-
+obj-$(CONFIG_REGULATOR_MPQ8785) += mpq8785.o
ccflags-$(CONFIG_REGULATOR_DEBUG) += -DDEBUG
diff --git a/drivers/regulator/mpq8785.c b/drivers/regulator/mpq8785.c
new file mode 100644
index 000000000000..60c48ae47d31
--- /dev/null
+++ b/drivers/regulator/mpq8785.c
@@ -0,0 +1,1217 @@
+
+// SPDX-License-Identifier: GPL-2.0
+/*
+ * eswin Specific Glue layer
+ *
+ * Copyright 2024, Beijing ESWIN Computing Technology Co., Ltd.. All rights
+ * reserved. SPDX-License-Identifier: GPL-2.0
+ *
+ * This program is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation, version 2.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ * GNU General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program. If not, see <https://www.gnu.org/licenses/>.
+ *
+ * Authors: Yang Wei <yangwei1@eswincomputing.com>
+ */
+
+#include <linux/cdev.h>
+#include <linux/device.h>
+#include <linux/err.h>
+#include <linux/fs.h>
+#include <linux/gpio/consumer.h>
+#include <linux/hwmon-sysfs.h>
+#include <linux/hwmon.h>
+#include <linux/i2c.h>
+#include <linux/init.h>
+#include <linux/interrupt.h>
+#include <linux/irq.h>
+#include <linux/kernel.h>
+#include <linux/module.h>
+#include <linux/regulator/driver.h>
+#include <linux/regulator/machine.h>
+#include <linux/regulator/of_regulator.h>
+#include <linux/slab.h>
+#include <linux/sysfs.h>
+
+#define MPQ8785_CMD_PAGE 0x0
+#define MPQ8785_CMD_OPERATION 0x1
+#define MPQ8785_CMD_ON_OFF_CONFIG 0x2
+#define MPQ8785_CMD_CLEAR_FAULT 0x3
+#define MPQ8785_CMD_CLEAR_LAST_FAULT 0x8
+#define MPQ8785_CMD_LAST_FAULT_RESTORE 0xc
+#define MPQ8785_CMD_WRITE_PROTECTION 0x10
+#define MPQ8785_CMD_STORE_ALL 0x15
+#define MPQ8785_CMD_RESTORE_ALL 0x16
+#define MPQ8785_CMD_CAPABILITY 0x19
+#define MPQ8785_CMD_PMBUS_PS_NUM 0x1C
+#define MPQ8785_CMD_VOUT_MODE 0x20
+#define MPQ8785_CMD_VOUT_COMMAND 0x21
+#define MPQ8785_CMD_VOUT_MAX 0x24
+#define MPQ8785_CMD_VOUT_MARGIN_HIGH 0x25
+#define MPQ8785_CMD_VOUT_MARGIN_LOW 0x26
+#define MPQ8785_CMD_VOUT_SCALE_LOOP 0x29
+#define MPQ8785_CMD_VOUT_MIN 0x2B
+#define MPQ8785_CMD_COEFFICIENT 0x30
+#define MPQ8785_CMD_VIN_ON 0x35
+#define MPQ8785_CMD_VIN_OFF 0x36
+#define MPQ8785_CMD_IOUT_CAL_GAIN 0x38
+#define MPQ8785_CMD_IOUT_CAL_OFFSET 0x39
+#define MPQ8785_CMD_IOUT_OC_FAULT_LIMIT 0x46
+#define MPQ8785_CMD_IOUT_OC_WARN_LIMIT 0x4A
+#define MPQ8785_CMD_VBOOT_SET_FOR_XOh_ADDR 0x4D
+#define MPQ8785_CMD_OT_FAULT_LIMIT 0x4F
+#define MPQ8785_CMD_OT_WARN_LIMIT 0x51
+#define MPQ8785_CMD_VIN_OV_FAULT_LIMIT 0x55
+#define MPQ8785_CMD_VIN_OV_WARN_LIMIT 0x57
+#define MPQ8785_CMD_VBOOT_SET_FOR_X4h_ADDR 0x5E
+#define MPQ8785_CMD_VBOOT_SET_FOR_X8h_ADDR 0x5F
+#define MPQ8785_CMD_TON_DELAY 0x60
+#define MPQ8785_CMD_TON_RISE 0x61
+#define MPQ8785_CMD_TOFF_DELAY 0x64
+#define MPQ8785_CMD_TOFF_FALL 0x65
+#define MPQ8785_CMD_VBOOT_SET_FOR_XEh_ADDR 0x6A
+#define MPQ8785_CMD_STATUS_WORD 0x79
+#define MPQ8785_CMD_STATUS_VOUT 0x7A
+#define MPQ8785_CMD_STATUS_IOUT 0x7B
+#define MPQ8785_CMD_STATUS_INPUT 0x7C
+#define MPQ8785_CMD_STATUS_TEMPERATURE 0x7D
+#define MPQ8785_CMD_STATUS_CML 0x7E
+#define MPQ8785_CMD_REV_ID 0x80
+#define MPQ8785_CMD_READ_VIN 0x88
+#define MPQ8785_CMD_READ_VOUT 0x8B
+#define MPQ8785_CMD_READ_IOUT 0x8C
+#define MPQ8785_CMD_READ_TEMPERATURE 0x8D
+#define MPQ8785_CMD_PMBUS_REV_CONST 0x98
+#define MPQ8785_CMD_MFR_ID 0x99
+#define MPQ8785_CMD_MFR_REVISION 0x9B
+#define MPQ8785_CMD_MFR_CONFIG_ID 0xC0
+#define MPQ8785_CMD_MFR_CONFIG_CODE_REV 0xC1
+#define MPQ8785_CMD_MFR_PRODUCT_REV_USER 0xC2
+#define MPQ8785_CMD_MFR_SILICON_REV 0xC3
+#define MPQ8785_CMD_MFR_APS_LEVEL 0xC5
+#define MPQ8785_CMD_MFR_CONFIG_A 0xD0
+#define MPQ8785_CMD_MFR_FS_CFG 0xD1
+#define MPQ8785_CMD_MFR_ADDR_PMBUS 0xD2
+#define MPQ8785_CMD_MFR_VOUT_RATE 0xD3
+#define MPQ8785_CMD_MFR_PWM_TIME_CFG 0xD4
+#define MPQ8785_CMD_MFR_PWM_TIME_CFG2 0xD5
+#define MPQ8785_CMD_MFR_PHASE_BLANK_TIME 0xD6
+#define MPQ8785_CMD_MFR_PHASE_SLOPE_BLANK_TIME 0xD7
+#define MPQ8785_CMD_MFR_SLOPE_BLANK_TIME 0xD8
+#define MPQ8785_CMD_MFR_BLANK_TIME_LV 0xD9
+#define MPQ8785_CMD_MFR_SLOPE_CNT_DCM 0xDA
+#define MPQ8785_CMD_MFR_SLOPE_SR_DCM 0xDB
+#define MPQ8785_CMD_MFR_SW_BLOCK_LIMIT 0xDC
+#define MPQ8785_CMD_MFR_VCOMP 0xDD
+#define MPQ8785_CMD_MFR_DROOP_CFG 0xDE
+#define MPQ8785_CMD_MFR_CONFIG_B 0xDF
+#define MPQ8785_CMD_MFR_DC_LOOP_CTRL 0xE0
+#define MPQ8785_CMD_MFR_CB_LOOP_CTRL 0xE1
+#define MPQ8785_CMD_MFR_FS_LOOP_CTRL 0xE2
+#define MPQ8785_CMD_MFR_VIN_CFG 0xE3
+#define MPQ8785_CMD_MFR_VIN_SCALE 0xE4
+#define MPQ8785_CMD_MFR_TEMP_TUNE 0xE5
+#define MPQ8785_CMD_MFR_PROTECT_CFG 0xE6
+#define MPQ8785_CMD_MFR_PROTECT_LEVEL 0xE7
+#define MPQ8785_CMD_MFR_PRT_DELAY 0xE8
+#define MPQ8785_CMD_SMBALERT_MASK 0xE9
+#define MPQ8785_CMD_MFR_NOCP_OCP_SET 0xEA
+#define MPQ8785_CMD_MFR_LEVEL_SEL2 0xEB
+#define MPQ8785_CMD_MFR_PG_CFG 0xEC
+#define MPQ8785_CMD_MFR_PS_CTRL 0xED
+#define MPQ8785_CMD_MFR_PMBUS_LOCK 0xEE
+#define MPQ8785_CMD_MFR_SET_SYNC_CFG 0xEF
+#define MPQ8785_CMD_MFR_SLAVE_PROTECT 0xF0
+#define MPQ8785_CMD_MFR_CTRL 0xF1
+#define MPQ8785_CMD_MFR_AUTO_SLOPE_CFG 0xF2
+#define MPQ8785_CMD_MFR_SLOPE_DELTA_LIMIT 0xF3
+#define MPQ8785_CMD_MFR_RETRY_TIMES 0xF4
+#define MPQ8785_CMD_MFR_CFG_EXT 0xF5
+#define MPQ8785_CMD_MFR_CDROOP_SET 0xF6
+#define MPQ8785_CMD_MFR_CFG_BACKUP 0xF7
+#define MPQ8785_CMD_CHECK_SUM_FUNC 0xF8
+#define MPQ8785_CMD_PROTECTION 0xFA
+#define MPQ8785_CMD_PROTECTION_LAST 0xFB
+#define MPQ8785_CMD_MFR_VBOOT_CFG 0xFC
+#define MPQ8785_CMD_CLEAR_NVM_FAULT 0xFE
+
+#define MPQ8785_PMBUS_EXTRA_READ_FLAG (1 << 7)
+#define MPQ8785_LABEL_CNT 4
+
+struct MPQ8785_DRIVER_DATA
+{
+ u32 volt_format;
+ u32 volt_numerator;
+ struct regulator_dev *rdev;
+ struct regulator_desc *dev_desc;
+ struct i2c_client *client;
+ struct mutex config_lock;
+ char mpq8785_label[MPQ8785_LABEL_CNT][16];
+};
+
+#define MPQ8785_MASK_OPERATION_ENABLE 0X80
+#define MPQ8785_MASK_VIN_OV_FAULT 0x7f
+#define MPQ8785_MASK_VOUT_LIMIT 0xFFF
+#define MPQ8785_MASK_VOUT_VALUE 0xFFF
+#define MPQ8785_MASK_IOUT_LIMIT 0x3FF
+#define MPQ8785_MASK_TOUT_LIMIT 0xFF
+#define MPQ8785_MASK_SW_FREQ_FREQ 0x1FF
+
+// Voltage LSB= {125/64, 125/80, 125/80,
+// 125/80}; 1.953125mV/1.5625mV/1.5625mV/1.5625mV
+#define MPQ8785_VOLT_DENOMINATOR 125
+#define MPQ8785_VOLT_DENOMINATOR_HALF (MPQ8785_VOLT_DENOMINATOR >> 1)
+// Voltage_in LSB=200mV
+#define MPQ8785_VOLT_IN_LSB 200
+#define MPQ8785_VOLT_IN_LSB_HALF (MPQ8785_VOLT_IN_LSB >> 1)
+
+// sense_Voltage_in LSB=25mV
+#define MPQ8785_VOLTE_IN_SENSE_LSB 25
+
+/* current*2 LSB=125mA */
+#define MPQ8785_CURRENT_LSB_DENOMINATOR 125
+#define MPQ8785_CURRENT_LSB_DENOMINATOR_HALF (MPQ8785_CURRENT_LSB_DENOMINATOR >> 1)
+#define MPQ8785_CURRENT_LSB_NUMERATOR_BIT 1 // 2
+
+#define MPQ8785_REGULATOT_CURRENT_LSB 1000000 /*1uA*/
+#define MPQ8785_CURRENT_OUT_LSB 1000 /*1mA*/
+
+/*mini sw frequency is 300kHz, frequency lsb = 10kHz */
+#define MPQ8785_FREQUENCY_LSB 10
+#define MPQ8785_FREQUENCY_BASE_MINI 300 /* 300kHz=30*10kHz */
+#define MPQ8785_FREQUENCY_BASE_MAX 2000
+
+static u32 garr_volt_numerator[] = {64, 80, 80, 80};
+static char garr_bool_string[][2] = {"N", "Y"};
+static char garr_speed_string[][8] = {"100kHz", "400kHz", "1MHz", "resv"};
+
+static struct of_regulator_match mpq8785_matches[] = {
+ {
+ .name = "npu_svcc",
+ },
+};
+
+static inline u32 mpq8785_volt2reg(u32 vlot, u32 volt_numerator)
+{
+ u32 value = 0;
+
+ value = (vlot * volt_numerator + MPQ8785_VOLT_DENOMINATOR_HALF) /
+ MPQ8785_VOLT_DENOMINATOR;
+ return value;
+}
+
+static inline u32 mpq8785_reg2volt(u32 value, u32 volt_numerator)
+{
+ u32 vlot = 0;
+
+ vlot = value * MPQ8785_VOLT_DENOMINATOR / volt_numerator;
+
+ return vlot;
+}
+
+static inline s32 mpq8785_str2ul(const char *buf, u32 *value)
+{
+ unsigned long cache = 0;
+ int ret = 0;
+
+ if (NULL == strstr(buf, "0x"))
+ {
+ ret = kstrtoul(buf, 10, &cache);
+ }
+ else
+ {
+ ret = kstrtoul(buf, 16, &cache);
+ }
+ *value = cache;
+
+ return ret;
+}
+
+static u8 mpq8785_read_byte(struct MPQ8785_DRIVER_DATA *data, u8 command)
+{
+ int ret = 0;
+ mutex_lock(&data->config_lock);
+ ret = i2c_smbus_read_byte_data(data->client, command);
+ mutex_unlock(&data->config_lock);
+ if (ret < 0)
+ {
+ dev_err(&data->client->dev, "get command:0x%x value error:%d\n", command,
+ ret);
+ return 0xff;
+ }
+ return (u8)ret;
+}
+
+static s32 mpq8785_write_byte(struct MPQ8785_DRIVER_DATA *data, u8 command,
+ u8 val)
+{
+ int ret = 0;
+ mutex_lock(&data->config_lock);
+ ret = i2c_smbus_write_byte_data(data->client, command, val);
+ mutex_unlock(&data->config_lock);
+ if (ret < 0)
+ {
+ dev_err(&data->client->dev, "set command:0x%x value:0x%x error:%d\n",
+ command, val, ret);
+ }
+ return ret;
+}
+
+static s32 mpq8785_update_byte(struct MPQ8785_DRIVER_DATA *data, u8 command,
+ u8 mask, u8 val)
+{
+ u8 old_value = 0;
+ u8 new_value = 0;
+ if (0 != (~mask & val))
+ {
+ dev_err(&data->client->dev, "command:0x%x,input:0x%x outrange mask:0x%x\n",
+ command, val, mask);
+ return -EINVAL;
+ }
+ old_value = mpq8785_read_byte(data, command);
+ new_value = ~mask & old_value;
+ new_value = new_value | val;
+ return mpq8785_write_byte(data, command, new_value);
+}
+
+static u16 mpq8785_read_word(struct MPQ8785_DRIVER_DATA *data, u8 command)
+{
+ int ret = 0;
+ mutex_lock(&data->config_lock);
+ ret = i2c_smbus_read_word_data(data->client, command);
+ mutex_unlock(&data->config_lock);
+ if (ret < 0)
+ {
+ dev_err(&data->client->dev, "get command:0x%x value error:%d\n", command,
+ ret);
+ return 0xffff;
+ }
+ return (u16)ret;
+}
+
+static u16 mpq8785_read_mask_word(struct MPQ8785_DRIVER_DATA *data, u8 command,
+ u16 mask)
+{
+ u16 ret = mpq8785_read_word(data, command);
+ return (ret & mask);
+}
+
+static s32 mpq8785_write_word(struct MPQ8785_DRIVER_DATA *data, u8 command,
+ u16 val)
+{
+ int ret = 0;
+ mutex_lock(&data->config_lock);
+ ret = i2c_smbus_write_word_data(data->client, command, val);
+ mutex_unlock(&data->config_lock);
+ if (ret < 0)
+ {
+ dev_err(&data->client->dev, "set command:0x%x value:0x%x error:%d\n",
+ command, val, ret);
+ }
+ return ret;
+}
+
+static s32 mpq8785_update_word(struct MPQ8785_DRIVER_DATA *data, u8 command,
+ u16 mask, u16 val)
+{
+ u16 old_value = 0;
+ u16 new_value = 0;
+ if (0 != (~mask & val))
+ {
+ dev_err(&data->client->dev, "command:0x%x,input:0x%x outrange mask:0x%x\n",
+ command, val, mask);
+ return -EINVAL;
+ }
+ old_value = mpq8785_read_word(data, command);
+ new_value = ~mask & old_value;
+ new_value = new_value | val;
+ return mpq8785_write_word(data, command, new_value);
+}
+
+static int mpq8785_read_block(struct MPQ8785_DRIVER_DATA *data, u8 command,
+ u8 *buf, u8 len, bool extra_len)
+{
+ int ret = 0;
+ int num = 0;
+
+ if (extra_len)
+ {
+ len++;
+ }
+ mutex_lock(&data->config_lock);
+ ret = i2c_smbus_read_i2c_block_data(data->client, command, len, buf);
+ mutex_unlock(&data->config_lock);
+ if (ret < 0)
+ {
+ dev_err(&data->client->dev, "get command:0x%x value error:%d\n", command,
+ ret);
+ return -EIO;
+ }
+ if (extra_len)
+ {
+ for (num = 0; num < (len - 1); num++)
+ {
+ buf[num] = buf[num + 1];
+ }
+ buf[num] = 0;
+ }
+ return 0;
+}
+
+static int mpq8785_get_enable(struct MPQ8785_DRIVER_DATA *data)
+{
+ u8 cache = 0;
+
+ cache = mpq8785_read_byte(data, MPQ8785_CMD_OPERATION);
+
+ return ((cache >> 7) & 0x1);
+}
+
+static const struct hwmon_channel_info *mpq8785_info[] = {
+ HWMON_CHANNEL_INFO(in, HWMON_I_INPUT | HWMON_I_MAX | HWMON_I_MAX_ALARM | HWMON_I_LABEL,
+ HWMON_I_INPUT | HWMON_I_ENABLE | HWMON_I_MAX | HWMON_I_MAX_ALARM | HWMON_I_LABEL),
+ HWMON_CHANNEL_INFO(curr, HWMON_C_INPUT | HWMON_C_CRIT | HWMON_C_CRIT_ALARM | HWMON_C_LABEL),
+ HWMON_CHANNEL_INFO(temp, HWMON_T_INPUT | HWMON_T_CRIT | HWMON_T_CRIT_ALARM | HWMON_T_LABEL),
+ NULL};
+
+static umode_t mpq8785_is_visible(const void *_data,
+ enum hwmon_sensor_types type, u32 attr,
+ int channel)
+{
+ switch (type)
+ {
+ case hwmon_in:
+ switch (attr)
+ {
+ case hwmon_in_input:
+ case hwmon_in_label:
+ case hwmon_in_max_alarm:
+ return 0444;
+ case hwmon_in_enable:
+ case hwmon_in_min:
+ case hwmon_in_max:
+ return 0644;
+ }
+
+ break;
+ case hwmon_curr:
+ switch (attr)
+ {
+ case hwmon_curr_input:
+ case hwmon_curr_crit_alarm:
+ case hwmon_curr_label:
+ return 0444;
+ case hwmon_curr_crit:
+ case hwmon_curr_max:
+ return 0644;
+ }
+ break;
+ case hwmon_temp:
+ switch (attr)
+ {
+ case hwmon_temp_input:
+ case hwmon_temp_label:
+ case hwmon_temp_crit_alarm:
+ return 0444;
+ case hwmon_temp_max:
+ case hwmon_temp_crit:
+ return 0644;
+ }
+ break;
+
+ default:
+ break;
+ }
+ return 0;
+}
+
+static int mpq8785_read(struct device *dev, enum hwmon_sensor_types type,
+ u32 attr, int channel, long *val)
+{
+ struct i2c_client *client = to_i2c_client(dev);
+ struct MPQ8785_DRIVER_DATA *data = i2c_get_clientdata(client);
+ u32 get_value = 0;
+
+ switch (type)
+ {
+ case hwmon_in:
+ switch (attr)
+ {
+ case hwmon_in_input:
+ if (channel == 0)
+ {
+ get_value = mpq8785_read_word(data, MPQ8785_CMD_READ_VIN);
+ *val = get_value * MPQ8785_VOLTE_IN_SENSE_LSB;
+ }
+ else if (channel == 1)
+ {
+ get_value = mpq8785_read_word(data, MPQ8785_CMD_READ_VOUT);
+ *val = mpq8785_reg2volt(get_value, data->volt_numerator);
+ }
+ else
+ {
+ dev_err(dev, "not support channel%d\n", channel);
+ }
+ break;
+ case hwmon_in_max_alarm:
+ get_value = mpq8785_read_word(data, MPQ8785_CMD_STATUS_WORD);
+ if (channel == 0)
+ {
+ *val = ((get_value >> 13) & 0x1);
+ }
+ else if (channel == 1)
+ {
+ *val = ((get_value >> 15) & 0x1);
+ }
+ else
+ {
+ dev_err(dev, "not support channel%d\n", channel);
+ }
+ break;
+ case hwmon_in_enable:
+ *val = mpq8785_get_enable(data);
+ break;
+ case hwmon_in_min:
+ get_value = mpq8785_read_mask_word(data, MPQ8785_CMD_VOUT_MIN,
+ MPQ8785_MASK_VOUT_LIMIT);
+ *val = mpq8785_reg2volt(get_value, data->volt_numerator);
+ break;
+ case hwmon_in_max:
+ if (channel == 0)
+ {
+ get_value =
+ mpq8785_read_mask_word(data, MPQ8785_CMD_VIN_OV_FAULT_LIMIT,
+ MPQ8785_MASK_VIN_OV_FAULT);
+ *val = get_value * MPQ8785_VOLT_IN_LSB;
+ }
+ else if (channel == 1)
+ {
+ get_value = mpq8785_read_mask_word(data, MPQ8785_CMD_VOUT_MAX,
+ MPQ8785_MASK_VOUT_LIMIT);
+ *val = mpq8785_reg2volt(get_value, data->volt_numerator);
+ }
+ else
+ {
+ dev_err(dev, "not support channel%d\n", channel);
+ }
+ break;
+ }
+
+ break;
+ case hwmon_curr:
+ switch (attr)
+ {
+ case hwmon_curr_input:
+ get_value = mpq8785_read_word(data, MPQ8785_CMD_READ_IOUT);
+ *val = (get_value * MPQ8785_CURRENT_LSB_DENOMINATOR) >>
+ MPQ8785_CURRENT_LSB_NUMERATOR_BIT;
+ break;
+ case hwmon_curr_crit_alarm:
+ get_value = mpq8785_read_word(data, MPQ8785_CMD_STATUS_WORD);
+ if (channel == 0)
+ {
+ *val = ((get_value >> 4) & 0x1);
+ }
+ break;
+ case hwmon_curr_crit:
+ *val = mpq8785_read_mask_word(data, MPQ8785_CMD_IOUT_OC_FAULT_LIMIT,
+ MPQ8785_MASK_IOUT_LIMIT);
+
+ break;
+ case hwmon_curr_max:
+ *val = mpq8785_read_mask_word(data, MPQ8785_CMD_IOUT_OC_WARN_LIMIT,
+ MPQ8785_MASK_IOUT_LIMIT);
+
+ break;
+ }
+ break;
+ case hwmon_temp:
+ switch (attr)
+ {
+ case hwmon_temp_input:
+ *val = mpq8785_read_byte(data, MPQ8785_CMD_READ_TEMPERATURE);
+ break;
+ case hwmon_temp_crit_alarm:
+ get_value = mpq8785_read_word(data, MPQ8785_CMD_STATUS_WORD);
+ if (channel == 0)
+ {
+ *val = ((get_value >> 2) & 0x1);
+ }
+ break;
+ case hwmon_temp_max:
+ *val = mpq8785_read_mask_word(data, MPQ8785_CMD_OT_WARN_LIMIT,
+ MPQ8785_MASK_TOUT_LIMIT);
+
+ break;
+ case hwmon_temp_crit:
+ *val = mpq8785_read_mask_word(data, MPQ8785_CMD_OT_FAULT_LIMIT,
+ MPQ8785_MASK_TOUT_LIMIT);
+
+ break;
+ }
+ break;
+
+ default:
+ break;
+ }
+ return 0;
+}
+
+static int mpq8785_read_string(struct device *dev,
+ enum hwmon_sensor_types type,
+ u32 attr, int channel, const char **str)
+{
+ struct i2c_client *client = to_i2c_client(dev);
+ struct MPQ8785_DRIVER_DATA *data = i2c_get_clientdata(client);
+ switch (type)
+ {
+ case hwmon_in:
+ switch (attr)
+ {
+ case hwmon_in_label:
+ if (channel == 0)
+ {
+ *str = data->mpq8785_label[0];
+ }
+ else if (channel == 1)
+ {
+ *str = data->mpq8785_label[1];
+ }
+ else
+ {
+ dev_err(dev, "not support channel%d\n", channel);
+ }
+ break;
+ }
+ break;
+ case hwmon_curr:
+ switch (attr)
+ {
+ case hwmon_curr_label:
+ *str = data->mpq8785_label[2];
+ break;
+ }
+ break;
+ case hwmon_temp:
+ switch (attr)
+ {
+
+ case hwmon_temp_label:
+ *str = data->mpq8785_label[3];
+ break;
+ }
+ break;
+
+ default:
+ break;
+ }
+ return 0;
+}
+
+static int mpq8785_write(struct device *dev, enum hwmon_sensor_types type,
+ u32 attr, int channel, long val)
+{
+ struct i2c_client *client = to_i2c_client(dev);
+ struct MPQ8785_DRIVER_DATA *data = i2c_get_clientdata(client);
+ u16 new_value = 0;
+
+ int ret = 0;
+ switch (type)
+ {
+ case hwmon_in:
+ switch (attr)
+ {
+ case hwmon_in_enable:
+ mpq8785_update_byte(data, MPQ8785_CMD_OPERATION,
+ MPQ8785_MASK_OPERATION_ENABLE, (u8)(val << 7));
+ break;
+ case hwmon_in_min:
+ new_value = mpq8785_volt2reg(val, data->volt_numerator);
+ ret = mpq8785_update_word(data, MPQ8785_CMD_VOUT_MIN,
+ MPQ8785_MASK_VOUT_LIMIT, new_value);
+ break;
+ case hwmon_in_max:
+ if (channel == 0)
+ {
+ new_value = (MPQ8785_VOLT_IN_LSB_HALF + val) / MPQ8785_VOLT_IN_LSB;
+ ret = mpq8785_update_word(data, MPQ8785_CMD_VIN_OV_FAULT_LIMIT,
+ MPQ8785_MASK_VIN_OV_FAULT, new_value);
+ }
+ else if (channel == 1)
+ {
+ new_value = mpq8785_volt2reg(val, data->volt_numerator);
+ ret = mpq8785_update_word(data, MPQ8785_CMD_VOUT_MAX,
+ MPQ8785_MASK_VOUT_LIMIT, new_value);
+ }
+ else
+ {
+ dev_err(dev, "not support channel%d\n", channel);
+ }
+ break;
+ }
+
+ break;
+ case hwmon_curr:
+ switch (attr)
+ {
+ case hwmon_curr_crit:
+ ret = mpq8785_update_word(data, MPQ8785_CMD_IOUT_OC_FAULT_LIMIT,
+ MPQ8785_MASK_IOUT_LIMIT, (u16)val);
+ break;
+ case hwmon_curr_max:
+ ret = mpq8785_update_word(data, MPQ8785_CMD_IOUT_OC_WARN_LIMIT,
+ MPQ8785_MASK_IOUT_LIMIT, (u16)val);
+ break;
+ }
+ break;
+ case hwmon_temp:
+ switch (attr)
+ {
+ case hwmon_temp_max:
+ ret = mpq8785_update_word(data, MPQ8785_CMD_OT_WARN_LIMIT,
+ MPQ8785_MASK_TOUT_LIMIT, (u16)val);
+ break;
+ case hwmon_temp_crit:
+ ret = mpq8785_update_word(data, MPQ8785_CMD_OT_FAULT_LIMIT,
+ MPQ8785_MASK_TOUT_LIMIT, (u16)val);
+ break;
+ }
+ break;
+
+ default:
+ break;
+ }
+ return ret;
+}
+static const struct hwmon_ops pac193x_hwmon_ops = {
+ .is_visible = mpq8785_is_visible,
+ .read = mpq8785_read,
+ .write = mpq8785_write,
+ .read_string = mpq8785_read_string,
+};
+
+static struct hwmon_chip_info mpq8785_chip_info = {
+ .ops = &pac193x_hwmon_ops,
+ .info = mpq8785_info,
+
+};
+
+static ssize_t mpq8785_status_show(struct device *d,
+ struct device_attribute *attr, char *buf)
+{
+ struct i2c_client *client = to_i2c_client(d);
+ struct MPQ8785_DRIVER_DATA *data = i2c_get_clientdata(client);
+ u32 value = mpq8785_read_word(data, MPQ8785_CMD_STATUS_WORD);
+
+ return sysfs_emit(
+ buf,
+ "An output voltage fault or warning has occurred:%s\n"
+ "An output current or output power fault or warning has occurred:%s\n"
+ "An input voltage, input current, or input power fault or warning has "
+ "occurred:%s\n"
+ "Power Good:%s\n"
+ "Watch Dog overflow fault:%s\n"
+ "PMBus BUSY:%s\n"
+ "power OFF:%s\n"
+ "An output over-voltage fault has occurred:%s\n"
+ "An output over-current fault has occurred:%s\n"
+ "An input under-voltage fault has occurred:%s\n"
+ "A temperature fault or warning has occurred:%s\n"
+ "A communications, memory or logic fault has occurred:%s\n"
+ "DrMOS fault:%s\n",
+ garr_bool_string[(value >> 15) & 0x1],
+ garr_bool_string[(value >> 14) & 0x1],
+ garr_bool_string[(value >> 13) & 0x1],
+ garr_bool_string[(value >> 11) & 0x1],
+ garr_bool_string[(value >> 8) & 0x1],
+ garr_bool_string[(value >> 7) & 0x1],
+ garr_bool_string[(value >> 6) & 0x1],
+ garr_bool_string[(value >> 5) & 0x1],
+ garr_bool_string[(value >> 4) & 0x1],
+ garr_bool_string[(value >> 3) & 0x1],
+ garr_bool_string[(value >> 2) & 0x1],
+ garr_bool_string[(value >> 1) & 0x1],
+ garr_bool_string[value & 0x1]);
+}
+DEVICE_ATTR(mpq8785_status, S_IRUGO, mpq8785_status_show, NULL);
+
+static ssize_t mpq8785_cap_version_show(struct device *d,
+ struct device_attribute *attr,
+ char *buf)
+{
+ struct i2c_client *client = to_i2c_client(d);
+ struct MPQ8785_DRIVER_DATA *data = i2c_get_clientdata(client);
+ u8 cap_value = mpq8785_read_byte(data, MPQ8785_CMD_CAPABILITY);
+ u8 rev_id = mpq8785_read_byte(data, MPQ8785_CMD_REV_ID);
+ u8 pmbus_rev = mpq8785_read_byte(data, MPQ8785_CMD_PMBUS_REV_CONST);
+ u32 mfr_id = 0;
+ u8 mfr_rev = mpq8785_read_byte(data, MPQ8785_CMD_MFR_REVISION);
+ mpq8785_read_block(data, MPQ8785_CMD_MFR_ID, (u8 *)&mfr_id, 3, true);
+ return sysfs_emit(buf,
+ "PEC_SUPPORT:%s\n"
+ "MAX_BUS_SPEED:%s\n"
+ "SMBALERT_SUPPORT:%s\n"
+ "AVSBUS_SUPPORT:%s\n"
+ "Silicon revision number:%x\n"
+ "pmbus_revision:1.%c\n"
+ "mfr_id:%c%c%c\n"
+ "mfr_revision:%x\n",
+ garr_bool_string[(cap_value >> 7) & 0x1],
+ garr_speed_string[(cap_value >> 5) & 0x3],
+ garr_bool_string[(cap_value >> 4) & 0x1],
+ garr_bool_string[(cap_value >> 2) & 0x1], rev_id,
+ pmbus_rev & 0xff, (mfr_id >> 16) & 0xff,
+ (mfr_id >> 8) & 0xff, mfr_id & 0xff, mfr_rev & 0xff);
+}
+DEVICE_ATTR(mpq8785_cap_verison, S_IRUGO, mpq8785_cap_version_show, NULL);
+
+static u32 mpq8785_get_vout(struct MPQ8785_DRIVER_DATA *data)
+{
+ u32 get_value = mpq8785_read_mask_word(data, MPQ8785_CMD_VOUT_COMMAND,
+ MPQ8785_MASK_VOUT_VALUE);
+
+ return mpq8785_reg2volt(get_value, data->volt_numerator);
+}
+
+static s32 mpq8785_set_vout(struct MPQ8785_DRIVER_DATA *data, u32 volt_mv)
+{
+ u16 new_value = mpq8785_volt2reg(volt_mv, data->volt_numerator);
+ return mpq8785_update_word(data, MPQ8785_CMD_VOUT_COMMAND,
+ MPQ8785_MASK_VOUT_VALUE, (new_value));
+}
+
+static ssize_t mpq8785_vout_show(struct device *d,
+ struct device_attribute *attr, char *buf)
+{
+ struct i2c_client *client = to_i2c_client(d);
+ struct MPQ8785_DRIVER_DATA *data = i2c_get_clientdata(client);
+
+ return sysfs_emit(buf, "%u", mpq8785_get_vout(data));
+}
+static ssize_t mpq8785_vout_store(struct device *d,
+ struct device_attribute *attr,
+ const char *buf, size_t count)
+{
+ struct i2c_client *client = to_i2c_client(d);
+ struct MPQ8785_DRIVER_DATA *data = i2c_get_clientdata(client);
+ u32 volt_value = 0;
+ int ret = 0;
+ ret = mpq8785_str2ul(buf, &volt_value);
+
+ if (ret)
+ {
+ return ret;
+ }
+ ret = mpq8785_set_vout(data, volt_value);
+ if (0 != ret)
+ {
+ return ret;
+ }
+ return count;
+}
+DEVICE_ATTR(mpq8785_vout, 0600, mpq8785_vout_show, mpq8785_vout_store);
+
+static ssize_t mpq8785_sw_freq_show(struct device *d,
+ struct device_attribute *attr, char *buf)
+{
+ struct i2c_client *client = to_i2c_client(d);
+ struct MPQ8785_DRIVER_DATA *data = i2c_get_clientdata(client);
+ u32 get_value = mpq8785_read_mask_word(data, MPQ8785_CMD_MFR_FS_CFG,
+ MPQ8785_MASK_SW_FREQ_FREQ);
+
+ return sysfs_emit(
+ buf, "%u",
+ get_value * MPQ8785_FREQUENCY_LSB + MPQ8785_FREQUENCY_BASE_MINI);
+}
+static ssize_t mpq8785_sw_freq_store(struct device *d,
+ struct device_attribute *attr,
+ const char *buf, size_t count)
+{
+ struct i2c_client *client = to_i2c_client(d);
+ struct MPQ8785_DRIVER_DATA *data = i2c_get_clientdata(client);
+ u32 new_value = 0;
+ int ret = 0;
+
+ ret = mpq8785_str2ul(buf, &new_value);
+
+ if (ret)
+ {
+ return ret;
+ }
+ if ((new_value < MPQ8785_FREQUENCY_BASE_MINI) ||
+ (new_value > MPQ8785_FREQUENCY_BASE_MAX))
+ {
+ return -EINVAL;
+ }
+ ret = mpq8785_update_word(
+ data, MPQ8785_CMD_MFR_FS_CFG, MPQ8785_MASK_SW_FREQ_FREQ,
+ ((new_value - MPQ8785_FREQUENCY_BASE_MINI) / MPQ8785_FREQUENCY_LSB));
+ return count;
+}
+DEVICE_ATTR(mpq8785_sw_freq, 0600, mpq8785_sw_freq_show, mpq8785_sw_freq_store);
+
+static struct attribute *mp8785_attrs[] = {
+ &dev_attr_mpq8785_status.attr, &dev_attr_mpq8785_cap_verison.attr,
+ &dev_attr_mpq8785_vout.attr, &dev_attr_mpq8785_sw_freq.attr, NULL};
+
+ATTRIBUTE_GROUPS(mp8785);
+
+static struct linear_range mpq8785_ext_ranges[] = {
+ /* REGULATOR_LINEAR_RANGE(700000, 0, 100, 15625), //=1.953125mV*1000*8 */
+ REGULATOR_LINEAR_RANGE(600000, 0, 320, 3125), /* =1.5625mV*1000*2 */
+};
+
+/**
+ * mpq8785_set_voltage_sel - set_voltage_sel for users
+ *
+ * @rdev: regulator to operate on
+ * @sel: Selector to set
+ */
+static s32 mpq8785_set_voltage_sel(struct regulator_dev *rdev,
+ unsigned selector)
+{
+ struct device *dev = &rdev->dev;
+ struct i2c_client *client = to_i2c_client(rdev->dev.parent);
+ struct MPQ8785_DRIVER_DATA *data = i2c_get_clientdata(client);
+ u32 new_value = 0;
+
+ if (selector > mpq8785_ext_ranges->max_sel)
+ {
+ dev_err(dev, "selector:%u out of rang 0~%u\n", selector,
+ mpq8785_ext_ranges->max_sel);
+ return -EINVAL;
+ }
+
+ new_value = mpq8785_ext_ranges->min + mpq8785_ext_ranges->step * selector;
+
+ dev_dbg(dev, "%s_volt:%duV,selector:%u,step:%u,min:%u\n", __FUNCTION__,
+ new_value, selector, mpq8785_ext_ranges->step,
+ mpq8785_ext_ranges->min);
+
+ mpq8785_set_vout(data, new_value / 1000);
+
+ return 0;
+}
+
+/**
+ * mpq8785_get_voltage_sel - get_voltage_sel for users
+ *
+ * @rdev: regulator to operate on
+ */
+static s32 mpq8785_get_voltage_sel(struct regulator_dev *rdev)
+{
+ s32 index = 0;
+ struct device *dev = &rdev->dev;
+ struct i2c_client *client = to_i2c_client(rdev->dev.parent);
+ struct MPQ8785_DRIVER_DATA *data = i2c_get_clientdata(client);
+ u32 volt_value = 0;
+ u32 diff_volt = 0;
+
+ volt_value = mpq8785_get_vout(data);
+ volt_value *= 1000;
+
+ if (volt_value >= mpq8785_ext_ranges->min)
+ {
+ diff_volt = volt_value - mpq8785_ext_ranges->min;
+ }
+ else
+ {
+ diff_volt = 0;
+ }
+ dev_dbg(dev, "%s_diff_volt:%duV,volt:%u,min:%u\n", __FUNCTION__, diff_volt,
+ volt_value, mpq8785_ext_ranges->min);
+ index = DIV_ROUND_CLOSEST(diff_volt, mpq8785_ext_ranges->step);
+ if (index > mpq8785_ext_ranges->max_sel)
+ {
+ dev_err(dev, "volt:%duV out legal range\n", volt_value);
+ }
+
+ dev_dbg(dev, "%s_diff_volt:%duV,step:%d,index:%d\n", __FUNCTION__, diff_volt,
+ mpq8785_ext_ranges->step, index);
+ return index;
+}
+/**
+ * mpq8785_set_current_limit- set_current_limit for users
+ * @rdev: regulator to operate on
+ * @min_uA: Lower bound for current limit
+ * @max_uA: Upper bound for current limit
+ */
+static s32 mpq8785_set_current_limit(struct regulator_dev *rdev, s32 min_uA,
+ s32 max_uA)
+{
+ struct i2c_client *client = to_i2c_client(rdev->dev.parent);
+ struct MPQ8785_DRIVER_DATA *data = i2c_get_clientdata(client);
+ u16 new_value = 0;
+
+ new_value = max_uA / MPQ8785_REGULATOT_CURRENT_LSB;
+ mpq8785_update_word(data, MPQ8785_CMD_IOUT_OC_FAULT_LIMIT,
+ MPQ8785_MASK_IOUT_LIMIT, new_value);
+ dev_dbg(&rdev->dev, "mpq8785_set_current_limit,min_uA:%d,max_uA:%d,now_A:%d\n", min_uA,
+ max_uA, new_value);
+ return 0;
+}
+
+static s32 mpq8785_get_current_limit(struct regulator_dev *rdev)
+{
+ struct i2c_client *client = to_i2c_client(rdev->dev.parent);
+ struct MPQ8785_DRIVER_DATA *data = i2c_get_clientdata(client);
+ u32 get_value = mpq8785_read_mask_word(data, MPQ8785_CMD_IOUT_OC_FAULT_LIMIT,
+ MPQ8785_MASK_IOUT_LIMIT);
+ get_value = get_value * MPQ8785_REGULATOT_CURRENT_LSB;
+ dev_dbg(&rdev->dev, "mpq8785_get_current_limit_%duA\n", get_value);
+ return get_value;
+}
+
+static s32 mpq8785_get_error_flags(struct regulator_dev *rdev, u32 *flags)
+{
+ struct i2c_client *client = to_i2c_client(rdev->dev.parent);
+ struct MPQ8785_DRIVER_DATA *data = i2c_get_clientdata(client);
+
+ *flags = mpq8785_read_word(data, MPQ8785_CMD_STATUS_WORD);
+
+ dev_dbg(&rdev->dev, "mpq8785_get_error_flags_%u\n", *flags);
+ return 0;
+}
+
+int mpq8785_regulator_enable(struct regulator_dev *rdev)
+{
+ struct i2c_client *client = to_i2c_client(rdev->dev.parent);
+ struct MPQ8785_DRIVER_DATA *data = i2c_get_clientdata(client);
+ dev_dbg(&rdev->dev, "%s.%d\n", __FUNCTION__, __LINE__);
+ return mpq8785_update_byte(data, MPQ8785_CMD_OPERATION,
+ MPQ8785_MASK_OPERATION_ENABLE,
+ MPQ8785_MASK_OPERATION_ENABLE);
+}
+
+int mpq8785_regulator_disable(struct regulator_dev *rdev)
+{
+ struct i2c_client *client = to_i2c_client(rdev->dev.parent);
+ struct MPQ8785_DRIVER_DATA *data = i2c_get_clientdata(client);
+ dev_dbg(&rdev->dev, "%s.%d\n", __FUNCTION__, __LINE__);
+ return mpq8785_update_byte(data, MPQ8785_CMD_OPERATION,
+ MPQ8785_MASK_OPERATION_ENABLE, 0);
+}
+int mpq8785_regulator_is_enabled(struct regulator_dev *rdev)
+{
+ struct i2c_client *client = to_i2c_client(rdev->dev.parent);
+ struct MPQ8785_DRIVER_DATA *data = i2c_get_clientdata(client);
+ dev_dbg(&rdev->dev, "%s.%d\n", __FUNCTION__, __LINE__);
+ return mpq8785_get_enable(data);
+}
+
+static struct regulator_ops mpq8785_core_ops = {
+
+ .list_voltage = regulator_list_voltage_linear_range,
+ .map_voltage = regulator_map_voltage_linear_range,
+
+ /* get/set regulator voltage */
+ /* Only one of each(set_voltage&&set_voltage_sel) should be implemented */
+ /* .set_voltage = mpq8785_set_voltage, */
+ .set_voltage_sel = mpq8785_set_voltage_sel,
+
+ /* Only one of each(get_voltage&&get_voltage_sel) should be implemented */
+ /* .get_voltage=mpq8785_get_voltage, */
+ .get_voltage_sel = mpq8785_get_voltage_sel,
+
+ /* get/set regulator current */
+ .set_current_limit = mpq8785_set_current_limit,
+ .get_current_limit = mpq8785_get_current_limit,
+
+ /* enable/disable regulator */
+ .enable = mpq8785_regulator_enable,
+ .disable = mpq8785_regulator_disable,
+ .is_enabled = mpq8785_regulator_is_enabled,
+
+ .get_error_flags = mpq8785_get_error_flags,
+
+};
+static struct regulator_desc mpq8785_regulator_desc = {
+ .name = "NPUVDD",
+ .type = REGULATOR_VOLTAGE,
+ .n_voltages = 321,
+ .ops = &mpq8785_core_ops,
+ .linear_ranges = mpq8785_ext_ranges,
+ .n_linear_ranges = ARRAY_SIZE(mpq8785_ext_ranges),
+ .owner = THIS_MODULE,
+};
+
+static s32 mpq8785_init_data(struct MPQ8785_DRIVER_DATA *data,
+ const struct regulation_constraints *constraints, u32 default_voltage)
+{
+ u8 value = 0;
+ s32 ret = 0;
+ u16 new_value = 0;
+ struct device *dev = &data->client->dev;
+
+ /*set voltage format to VID to get better step for regulator*/
+ mpq8785_write_byte(data, MPQ8785_CMD_VOUT_MODE, 0x20);
+ value = mpq8785_read_byte(data, MPQ8785_CMD_VOUT_MODE); /*format*/
+ data->volt_format = (value >> 5) & 0x3;
+ data->volt_numerator = garr_volt_numerator[data->volt_format];
+
+ dev_info(dev,
+ "min_uV:%d,max_uV:%d,uV_offset:%d,min_uA:%d,max_uA:%d,"
+ "over_voltage_limits:%d,%d,%d\n",
+ constraints->min_uV, constraints->max_uV, constraints->uV_offset,
+ constraints->min_uA, constraints->max_uA,
+ constraints->over_voltage_limits.err,
+ constraints->over_voltage_limits.prot,
+ constraints->over_voltage_limits.warn);
+ mpq8785_ext_ranges->min = constraints->min_uV;
+ mpq8785_ext_ranges->min_sel = 0;
+ mpq8785_ext_ranges->max_sel = (constraints->max_uV - constraints->min_uV) / mpq8785_ext_ranges->step + 1;
+ mpq8785_regulator_desc.n_voltages = mpq8785_ext_ranges->max_sel;
+ new_value = mpq8785_volt2reg(constraints->over_voltage_limits.prot / 1000,
+ data->volt_numerator);
+ ret = mpq8785_update_word(data, MPQ8785_CMD_VOUT_MAX, MPQ8785_MASK_VOUT_LIMIT,
+ new_value);
+ if (ret < 0)
+ {
+ dev_err(dev, "set vout limit error\n");
+ return ret;
+ }
+
+ mpq8785_set_vout(data, default_voltage / 1000);
+ return ret;
+}
+
+static s32 mpq8785_probe(struct i2c_client *client)
+{
+ struct MPQ8785_DRIVER_DATA *data = NULL;
+ s32 ret = 0;
+ s32 regulator_cnt = 0;
+ u32 default_voltage = 0;
+ struct device *hwmon_dev;
+ struct regulator_config config = {};
+ struct device *dev = &client->dev;
+ struct device_node *np, *parent;
+ const char *output_names[4];
+
+ if (!i2c_check_functionality(client->adapter, I2C_FUNC_SMBUS_BYTE_DATA))
+ {
+ dev_err(dev, "not support smbus\n");
+ return -EIO;
+ }
+ data = devm_kzalloc(dev, sizeof(struct MPQ8785_DRIVER_DATA), GFP_KERNEL);
+ if (!data)
+ return -ENOMEM;
+
+ mutex_init(&data->config_lock);
+ data->client = client;
+ i2c_set_clientdata(client, data);
+ /* Get the device (PMIC) node */
+ np = of_node_get(dev->of_node);
+ if (!np)
+ return -EINVAL;
+
+ /* Get 'regulators' subnode */
+ parent = of_get_child_by_name(np, "regulators");
+ if (!parent)
+ {
+ dev_err(dev, "regulators node not found\n");
+ return -EINVAL;
+ }
+ ret = of_property_read_u32(np, "eswin,regulator_default-microvolt", &default_voltage);
+ if (ret)
+ {
+ default_voltage = 900000;
+ }
+ of_property_read_string_array(np, "eswin,regulator_label", output_names, 4);
+ if (NULL != output_names[0])
+ {
+ strcpy(data->mpq8785_label[0], output_names[0]);
+ }
+ if (NULL != output_names[1])
+ {
+ strcpy(data->mpq8785_label[1], output_names[1]);
+ }
+ if (NULL != output_names[2])
+ {
+ strcpy(data->mpq8785_label[2], output_names[2]);
+ }
+ if (NULL != output_names[3])
+ {
+ strcpy(data->mpq8785_label[3], output_names[3]);
+ }
+
+ dev_dbg(dev, "default_voltage:%u,%s,%s,%s,%s\n", default_voltage, data->mpq8785_label[0],
+ data->mpq8785_label[1], data->mpq8785_label[2], data->mpq8785_label[3]);
+ /* fill isl6271a_matches array */
+ regulator_cnt = of_regulator_match(dev, parent, mpq8785_matches, ARRAY_SIZE(mpq8785_matches));
+ of_node_put(parent);
+ if (regulator_cnt != 1)
+ {
+ dev_err(dev, "Error parsing regulator init data: %d\n", regulator_cnt);
+ return regulator_cnt;
+ }
+
+ /* Fetched from device tree */
+ config.init_data = mpq8785_matches[0].init_data;
+ config.dev = dev;
+ config.of_node = mpq8785_matches[0].of_node;
+ /* config.ena_gpio = -EINVAL; */
+ ret = mpq8785_init_data(data, &config.init_data->constraints, default_voltage);
+ if (0 != ret)
+ {
+ dev_err(dev, "init mpq8785 error\n");
+ return -EIO;
+ }
+ data->rdev = devm_regulator_register(dev, &mpq8785_regulator_desc, &config);
+ if (IS_ERR(data->rdev))
+ {
+ dev_err(dev, "failed to register %s\n", mpq8785_regulator_desc.name);
+ }
+ hwmon_dev = devm_hwmon_device_register_with_info(
+ dev, client->name, data, &mpq8785_chip_info, mp8785_groups);
+ if (IS_ERR(hwmon_dev))
+ return PTR_ERR(hwmon_dev);
+
+ dev_dbg(dev, "mpq8785_probe\n");
+
+ return 0;
+}
+
+static void mpq8785_remove(struct i2c_client *client)
+{
+ dev_dbg(&client->dev, "mpq8785_remove\n");
+}
+
+static s32 mpq8785_detect(struct i2c_client *client,
+ struct i2c_board_info *info)
+{
+ dev_dbg(&client->dev, "mpq8785_detect\n");
+ return 0;
+}
+
+static const struct i2c_device_id mpq8785_id[] = {{"mpq8785", 0}, {}};
+MODULE_DEVICE_TABLE(i2c, mpq8785_id);
+
+/* Addresses to scan */
+static const unsigned short normal_i2c[] = {0x2c, 0x2d, 0x2e, 0x60,
+ I2C_CLIENT_END};
+
+static struct i2c_driver mpq8785_driver = {
+ .class = I2C_CLASS_HWMON,
+ .driver =
+ {
+ .name = "mpq8785",
+ },
+ .probe = mpq8785_probe,
+ .remove = mpq8785_remove,
+ .id_table = mpq8785_id,
+ .detect = mpq8785_detect,
+ .address_list = normal_i2c,
+};
+
+module_i2c_driver(mpq8785_driver);
+
+MODULE_AUTHOR("Yang Wei <yangwei1@eswincomputing.com>");
+MODULE_DESCRIPTION("mpq8785 driver");
+MODULE_LICENSE("GPL");
\ No newline at end of file
--
2.47.0