162306a36Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0-or-later
262306a36Sopenharmony_ci/*
362306a36Sopenharmony_ci * Hardware monitoring driver for ZL6100 and compatibles
462306a36Sopenharmony_ci *
562306a36Sopenharmony_ci * Copyright (c) 2011 Ericsson AB.
662306a36Sopenharmony_ci * Copyright (c) 2012 Guenter Roeck
762306a36Sopenharmony_ci */
862306a36Sopenharmony_ci
962306a36Sopenharmony_ci#include <linux/bitops.h>
1062306a36Sopenharmony_ci#include <linux/kernel.h>
1162306a36Sopenharmony_ci#include <linux/module.h>
1262306a36Sopenharmony_ci#include <linux/init.h>
1362306a36Sopenharmony_ci#include <linux/err.h>
1462306a36Sopenharmony_ci#include <linux/slab.h>
1562306a36Sopenharmony_ci#include <linux/i2c.h>
1662306a36Sopenharmony_ci#include <linux/ktime.h>
1762306a36Sopenharmony_ci#include <linux/delay.h>
1862306a36Sopenharmony_ci#include "pmbus.h"
1962306a36Sopenharmony_ci
2062306a36Sopenharmony_cienum chips { zl2004, zl2005, zl2006, zl2008, zl2105, zl2106, zl6100, zl6105,
2162306a36Sopenharmony_ci	     zl8802, zl9101, zl9117, zls1003, zls4009 };
2262306a36Sopenharmony_ci
2362306a36Sopenharmony_cistruct zl6100_data {
2462306a36Sopenharmony_ci	int id;
2562306a36Sopenharmony_ci	ktime_t access;		/* chip access time */
2662306a36Sopenharmony_ci	int delay;		/* Delay between chip accesses in uS */
2762306a36Sopenharmony_ci	struct pmbus_driver_info info;
2862306a36Sopenharmony_ci};
2962306a36Sopenharmony_ci
3062306a36Sopenharmony_ci#define to_zl6100_data(x)  container_of(x, struct zl6100_data, info)
3162306a36Sopenharmony_ci
3262306a36Sopenharmony_ci#define ZL6100_MFR_CONFIG		0xd0
3362306a36Sopenharmony_ci#define ZL6100_DEVICE_ID		0xe4
3462306a36Sopenharmony_ci
3562306a36Sopenharmony_ci#define ZL6100_MFR_XTEMP_ENABLE		BIT(7)
3662306a36Sopenharmony_ci
3762306a36Sopenharmony_ci#define ZL8802_MFR_USER_GLOBAL_CONFIG	0xe9
3862306a36Sopenharmony_ci#define ZL8802_MFR_TMON_ENABLE		BIT(12)
3962306a36Sopenharmony_ci#define ZL8802_MFR_USER_CONFIG		0xd1
4062306a36Sopenharmony_ci#define ZL8802_MFR_XTEMP_ENABLE_2	BIT(1)
4162306a36Sopenharmony_ci#define ZL8802_MFR_DDC_CONFIG		0xd3
4262306a36Sopenharmony_ci#define ZL8802_MFR_PHASES_MASK		0x0007
4362306a36Sopenharmony_ci
4462306a36Sopenharmony_ci#define MFR_VMON_OV_FAULT_LIMIT		0xf5
4562306a36Sopenharmony_ci#define MFR_VMON_UV_FAULT_LIMIT		0xf6
4662306a36Sopenharmony_ci#define MFR_READ_VMON			0xf7
4762306a36Sopenharmony_ci
4862306a36Sopenharmony_ci#define VMON_UV_WARNING			BIT(5)
4962306a36Sopenharmony_ci#define VMON_OV_WARNING			BIT(4)
5062306a36Sopenharmony_ci#define VMON_UV_FAULT			BIT(1)
5162306a36Sopenharmony_ci#define VMON_OV_FAULT			BIT(0)
5262306a36Sopenharmony_ci
5362306a36Sopenharmony_ci#define ZL6100_WAIT_TIME		1000	/* uS	*/
5462306a36Sopenharmony_ci
5562306a36Sopenharmony_cistatic ushort delay = ZL6100_WAIT_TIME;
5662306a36Sopenharmony_cimodule_param(delay, ushort, 0644);
5762306a36Sopenharmony_ciMODULE_PARM_DESC(delay, "Delay between chip accesses in uS");
5862306a36Sopenharmony_ci
5962306a36Sopenharmony_ci/* Convert linear sensor value to milli-units */
6062306a36Sopenharmony_cistatic long zl6100_l2d(s16 l)
6162306a36Sopenharmony_ci{
6262306a36Sopenharmony_ci	s16 exponent;
6362306a36Sopenharmony_ci	s32 mantissa;
6462306a36Sopenharmony_ci	long val;
6562306a36Sopenharmony_ci
6662306a36Sopenharmony_ci	exponent = l >> 11;
6762306a36Sopenharmony_ci	mantissa = ((s16)((l & 0x7ff) << 5)) >> 5;
6862306a36Sopenharmony_ci
6962306a36Sopenharmony_ci	val = mantissa;
7062306a36Sopenharmony_ci
7162306a36Sopenharmony_ci	/* scale result to milli-units */
7262306a36Sopenharmony_ci	val = val * 1000L;
7362306a36Sopenharmony_ci
7462306a36Sopenharmony_ci	if (exponent >= 0)
7562306a36Sopenharmony_ci		val <<= exponent;
7662306a36Sopenharmony_ci	else
7762306a36Sopenharmony_ci		val >>= -exponent;
7862306a36Sopenharmony_ci
7962306a36Sopenharmony_ci	return val;
8062306a36Sopenharmony_ci}
8162306a36Sopenharmony_ci
8262306a36Sopenharmony_ci#define MAX_MANTISSA	(1023 * 1000)
8362306a36Sopenharmony_ci#define MIN_MANTISSA	(511 * 1000)
8462306a36Sopenharmony_ci
8562306a36Sopenharmony_cistatic u16 zl6100_d2l(long val)
8662306a36Sopenharmony_ci{
8762306a36Sopenharmony_ci	s16 exponent = 0, mantissa;
8862306a36Sopenharmony_ci	bool negative = false;
8962306a36Sopenharmony_ci
9062306a36Sopenharmony_ci	/* simple case */
9162306a36Sopenharmony_ci	if (val == 0)
9262306a36Sopenharmony_ci		return 0;
9362306a36Sopenharmony_ci
9462306a36Sopenharmony_ci	if (val < 0) {
9562306a36Sopenharmony_ci		negative = true;
9662306a36Sopenharmony_ci		val = -val;
9762306a36Sopenharmony_ci	}
9862306a36Sopenharmony_ci
9962306a36Sopenharmony_ci	/* Reduce large mantissa until it fits into 10 bit */
10062306a36Sopenharmony_ci	while (val >= MAX_MANTISSA && exponent < 15) {
10162306a36Sopenharmony_ci		exponent++;
10262306a36Sopenharmony_ci		val >>= 1;
10362306a36Sopenharmony_ci	}
10462306a36Sopenharmony_ci	/* Increase small mantissa to improve precision */
10562306a36Sopenharmony_ci	while (val < MIN_MANTISSA && exponent > -15) {
10662306a36Sopenharmony_ci		exponent--;
10762306a36Sopenharmony_ci		val <<= 1;
10862306a36Sopenharmony_ci	}
10962306a36Sopenharmony_ci
11062306a36Sopenharmony_ci	/* Convert mantissa from milli-units to units */
11162306a36Sopenharmony_ci	mantissa = DIV_ROUND_CLOSEST(val, 1000);
11262306a36Sopenharmony_ci
11362306a36Sopenharmony_ci	/* Ensure that resulting number is within range */
11462306a36Sopenharmony_ci	if (mantissa > 0x3ff)
11562306a36Sopenharmony_ci		mantissa = 0x3ff;
11662306a36Sopenharmony_ci
11762306a36Sopenharmony_ci	/* restore sign */
11862306a36Sopenharmony_ci	if (negative)
11962306a36Sopenharmony_ci		mantissa = -mantissa;
12062306a36Sopenharmony_ci
12162306a36Sopenharmony_ci	/* Convert to 5 bit exponent, 11 bit mantissa */
12262306a36Sopenharmony_ci	return (mantissa & 0x7ff) | ((exponent << 11) & 0xf800);
12362306a36Sopenharmony_ci}
12462306a36Sopenharmony_ci
12562306a36Sopenharmony_ci/* Some chips need a delay between accesses */
12662306a36Sopenharmony_cistatic inline void zl6100_wait(const struct zl6100_data *data)
12762306a36Sopenharmony_ci{
12862306a36Sopenharmony_ci	if (data->delay) {
12962306a36Sopenharmony_ci		s64 delta = ktime_us_delta(ktime_get(), data->access);
13062306a36Sopenharmony_ci		if (delta < data->delay)
13162306a36Sopenharmony_ci			udelay(data->delay - delta);
13262306a36Sopenharmony_ci	}
13362306a36Sopenharmony_ci}
13462306a36Sopenharmony_ci
13562306a36Sopenharmony_cistatic int zl6100_read_word_data(struct i2c_client *client, int page,
13662306a36Sopenharmony_ci				 int phase, int reg)
13762306a36Sopenharmony_ci{
13862306a36Sopenharmony_ci	const struct pmbus_driver_info *info = pmbus_get_driver_info(client);
13962306a36Sopenharmony_ci	struct zl6100_data *data = to_zl6100_data(info);
14062306a36Sopenharmony_ci	int ret, vreg;
14162306a36Sopenharmony_ci
14262306a36Sopenharmony_ci	if (page >= info->pages)
14362306a36Sopenharmony_ci		return -ENXIO;
14462306a36Sopenharmony_ci
14562306a36Sopenharmony_ci	if (data->id == zl2005) {
14662306a36Sopenharmony_ci		/*
14762306a36Sopenharmony_ci		 * Limit register detection is not reliable on ZL2005.
14862306a36Sopenharmony_ci		 * Make sure registers are not erroneously detected.
14962306a36Sopenharmony_ci		 */
15062306a36Sopenharmony_ci		switch (reg) {
15162306a36Sopenharmony_ci		case PMBUS_VOUT_OV_WARN_LIMIT:
15262306a36Sopenharmony_ci		case PMBUS_VOUT_UV_WARN_LIMIT:
15362306a36Sopenharmony_ci		case PMBUS_IOUT_OC_WARN_LIMIT:
15462306a36Sopenharmony_ci			return -ENXIO;
15562306a36Sopenharmony_ci		}
15662306a36Sopenharmony_ci	}
15762306a36Sopenharmony_ci
15862306a36Sopenharmony_ci	switch (reg) {
15962306a36Sopenharmony_ci	case PMBUS_VIRT_READ_VMON:
16062306a36Sopenharmony_ci		vreg = MFR_READ_VMON;
16162306a36Sopenharmony_ci		break;
16262306a36Sopenharmony_ci	case PMBUS_VIRT_VMON_OV_WARN_LIMIT:
16362306a36Sopenharmony_ci	case PMBUS_VIRT_VMON_OV_FAULT_LIMIT:
16462306a36Sopenharmony_ci		vreg = MFR_VMON_OV_FAULT_LIMIT;
16562306a36Sopenharmony_ci		break;
16662306a36Sopenharmony_ci	case PMBUS_VIRT_VMON_UV_WARN_LIMIT:
16762306a36Sopenharmony_ci	case PMBUS_VIRT_VMON_UV_FAULT_LIMIT:
16862306a36Sopenharmony_ci		vreg = MFR_VMON_UV_FAULT_LIMIT;
16962306a36Sopenharmony_ci		break;
17062306a36Sopenharmony_ci	default:
17162306a36Sopenharmony_ci		if (reg >= PMBUS_VIRT_BASE)
17262306a36Sopenharmony_ci			return -ENXIO;
17362306a36Sopenharmony_ci		vreg = reg;
17462306a36Sopenharmony_ci		break;
17562306a36Sopenharmony_ci	}
17662306a36Sopenharmony_ci
17762306a36Sopenharmony_ci	zl6100_wait(data);
17862306a36Sopenharmony_ci	ret = pmbus_read_word_data(client, page, phase, vreg);
17962306a36Sopenharmony_ci	data->access = ktime_get();
18062306a36Sopenharmony_ci	if (ret < 0)
18162306a36Sopenharmony_ci		return ret;
18262306a36Sopenharmony_ci
18362306a36Sopenharmony_ci	switch (reg) {
18462306a36Sopenharmony_ci	case PMBUS_VIRT_VMON_OV_WARN_LIMIT:
18562306a36Sopenharmony_ci		ret = zl6100_d2l(DIV_ROUND_CLOSEST(zl6100_l2d(ret) * 9, 10));
18662306a36Sopenharmony_ci		break;
18762306a36Sopenharmony_ci	case PMBUS_VIRT_VMON_UV_WARN_LIMIT:
18862306a36Sopenharmony_ci		ret = zl6100_d2l(DIV_ROUND_CLOSEST(zl6100_l2d(ret) * 11, 10));
18962306a36Sopenharmony_ci		break;
19062306a36Sopenharmony_ci	}
19162306a36Sopenharmony_ci
19262306a36Sopenharmony_ci	return ret;
19362306a36Sopenharmony_ci}
19462306a36Sopenharmony_ci
19562306a36Sopenharmony_cistatic int zl6100_read_byte_data(struct i2c_client *client, int page, int reg)
19662306a36Sopenharmony_ci{
19762306a36Sopenharmony_ci	const struct pmbus_driver_info *info = pmbus_get_driver_info(client);
19862306a36Sopenharmony_ci	struct zl6100_data *data = to_zl6100_data(info);
19962306a36Sopenharmony_ci	int ret, status;
20062306a36Sopenharmony_ci
20162306a36Sopenharmony_ci	if (page >= info->pages)
20262306a36Sopenharmony_ci		return -ENXIO;
20362306a36Sopenharmony_ci
20462306a36Sopenharmony_ci	zl6100_wait(data);
20562306a36Sopenharmony_ci
20662306a36Sopenharmony_ci	switch (reg) {
20762306a36Sopenharmony_ci	case PMBUS_VIRT_STATUS_VMON:
20862306a36Sopenharmony_ci		ret = pmbus_read_byte_data(client, 0,
20962306a36Sopenharmony_ci					   PMBUS_STATUS_MFR_SPECIFIC);
21062306a36Sopenharmony_ci		if (ret < 0)
21162306a36Sopenharmony_ci			break;
21262306a36Sopenharmony_ci
21362306a36Sopenharmony_ci		status = 0;
21462306a36Sopenharmony_ci		if (ret & VMON_UV_WARNING)
21562306a36Sopenharmony_ci			status |= PB_VOLTAGE_UV_WARNING;
21662306a36Sopenharmony_ci		if (ret & VMON_OV_WARNING)
21762306a36Sopenharmony_ci			status |= PB_VOLTAGE_OV_WARNING;
21862306a36Sopenharmony_ci		if (ret & VMON_UV_FAULT)
21962306a36Sopenharmony_ci			status |= PB_VOLTAGE_UV_FAULT;
22062306a36Sopenharmony_ci		if (ret & VMON_OV_FAULT)
22162306a36Sopenharmony_ci			status |= PB_VOLTAGE_OV_FAULT;
22262306a36Sopenharmony_ci		ret = status;
22362306a36Sopenharmony_ci		break;
22462306a36Sopenharmony_ci	default:
22562306a36Sopenharmony_ci		ret = pmbus_read_byte_data(client, page, reg);
22662306a36Sopenharmony_ci		break;
22762306a36Sopenharmony_ci	}
22862306a36Sopenharmony_ci	data->access = ktime_get();
22962306a36Sopenharmony_ci
23062306a36Sopenharmony_ci	return ret;
23162306a36Sopenharmony_ci}
23262306a36Sopenharmony_ci
23362306a36Sopenharmony_cistatic int zl6100_write_word_data(struct i2c_client *client, int page, int reg,
23462306a36Sopenharmony_ci				  u16 word)
23562306a36Sopenharmony_ci{
23662306a36Sopenharmony_ci	const struct pmbus_driver_info *info = pmbus_get_driver_info(client);
23762306a36Sopenharmony_ci	struct zl6100_data *data = to_zl6100_data(info);
23862306a36Sopenharmony_ci	int ret, vreg;
23962306a36Sopenharmony_ci
24062306a36Sopenharmony_ci	if (page >= info->pages)
24162306a36Sopenharmony_ci		return -ENXIO;
24262306a36Sopenharmony_ci
24362306a36Sopenharmony_ci	switch (reg) {
24462306a36Sopenharmony_ci	case PMBUS_VIRT_VMON_OV_WARN_LIMIT:
24562306a36Sopenharmony_ci		word = zl6100_d2l(DIV_ROUND_CLOSEST(zl6100_l2d(word) * 10, 9));
24662306a36Sopenharmony_ci		vreg = MFR_VMON_OV_FAULT_LIMIT;
24762306a36Sopenharmony_ci		pmbus_clear_cache(client);
24862306a36Sopenharmony_ci		break;
24962306a36Sopenharmony_ci	case PMBUS_VIRT_VMON_OV_FAULT_LIMIT:
25062306a36Sopenharmony_ci		vreg = MFR_VMON_OV_FAULT_LIMIT;
25162306a36Sopenharmony_ci		pmbus_clear_cache(client);
25262306a36Sopenharmony_ci		break;
25362306a36Sopenharmony_ci	case PMBUS_VIRT_VMON_UV_WARN_LIMIT:
25462306a36Sopenharmony_ci		word = zl6100_d2l(DIV_ROUND_CLOSEST(zl6100_l2d(word) * 10, 11));
25562306a36Sopenharmony_ci		vreg = MFR_VMON_UV_FAULT_LIMIT;
25662306a36Sopenharmony_ci		pmbus_clear_cache(client);
25762306a36Sopenharmony_ci		break;
25862306a36Sopenharmony_ci	case PMBUS_VIRT_VMON_UV_FAULT_LIMIT:
25962306a36Sopenharmony_ci		vreg = MFR_VMON_UV_FAULT_LIMIT;
26062306a36Sopenharmony_ci		pmbus_clear_cache(client);
26162306a36Sopenharmony_ci		break;
26262306a36Sopenharmony_ci	default:
26362306a36Sopenharmony_ci		if (reg >= PMBUS_VIRT_BASE)
26462306a36Sopenharmony_ci			return -ENXIO;
26562306a36Sopenharmony_ci		vreg = reg;
26662306a36Sopenharmony_ci	}
26762306a36Sopenharmony_ci
26862306a36Sopenharmony_ci	zl6100_wait(data);
26962306a36Sopenharmony_ci	ret = pmbus_write_word_data(client, page, vreg, word);
27062306a36Sopenharmony_ci	data->access = ktime_get();
27162306a36Sopenharmony_ci
27262306a36Sopenharmony_ci	return ret;
27362306a36Sopenharmony_ci}
27462306a36Sopenharmony_ci
27562306a36Sopenharmony_cistatic int zl6100_write_byte(struct i2c_client *client, int page, u8 value)
27662306a36Sopenharmony_ci{
27762306a36Sopenharmony_ci	const struct pmbus_driver_info *info = pmbus_get_driver_info(client);
27862306a36Sopenharmony_ci	struct zl6100_data *data = to_zl6100_data(info);
27962306a36Sopenharmony_ci	int ret;
28062306a36Sopenharmony_ci
28162306a36Sopenharmony_ci	if (page >= info->pages)
28262306a36Sopenharmony_ci		return -ENXIO;
28362306a36Sopenharmony_ci
28462306a36Sopenharmony_ci	zl6100_wait(data);
28562306a36Sopenharmony_ci	ret = pmbus_write_byte(client, page, value);
28662306a36Sopenharmony_ci	data->access = ktime_get();
28762306a36Sopenharmony_ci
28862306a36Sopenharmony_ci	return ret;
28962306a36Sopenharmony_ci}
29062306a36Sopenharmony_ci
29162306a36Sopenharmony_cistatic const struct i2c_device_id zl6100_id[] = {
29262306a36Sopenharmony_ci	{"bmr450", zl2005},
29362306a36Sopenharmony_ci	{"bmr451", zl2005},
29462306a36Sopenharmony_ci	{"bmr462", zl2008},
29562306a36Sopenharmony_ci	{"bmr463", zl2008},
29662306a36Sopenharmony_ci	{"bmr464", zl2008},
29762306a36Sopenharmony_ci	{"bmr465", zls4009},
29862306a36Sopenharmony_ci	{"bmr466", zls1003},
29962306a36Sopenharmony_ci	{"bmr467", zls4009},
30062306a36Sopenharmony_ci	{"bmr469", zl8802},
30162306a36Sopenharmony_ci	{"zl2004", zl2004},
30262306a36Sopenharmony_ci	{"zl2005", zl2005},
30362306a36Sopenharmony_ci	{"zl2006", zl2006},
30462306a36Sopenharmony_ci	{"zl2008", zl2008},
30562306a36Sopenharmony_ci	{"zl2105", zl2105},
30662306a36Sopenharmony_ci	{"zl2106", zl2106},
30762306a36Sopenharmony_ci	{"zl6100", zl6100},
30862306a36Sopenharmony_ci	{"zl6105", zl6105},
30962306a36Sopenharmony_ci	{"zl8802", zl8802},
31062306a36Sopenharmony_ci	{"zl9101", zl9101},
31162306a36Sopenharmony_ci	{"zl9117", zl9117},
31262306a36Sopenharmony_ci	{"zls1003", zls1003},
31362306a36Sopenharmony_ci	{"zls4009", zls4009},
31462306a36Sopenharmony_ci	{ }
31562306a36Sopenharmony_ci};
31662306a36Sopenharmony_ciMODULE_DEVICE_TABLE(i2c, zl6100_id);
31762306a36Sopenharmony_ci
31862306a36Sopenharmony_cistatic int zl6100_probe(struct i2c_client *client)
31962306a36Sopenharmony_ci{
32062306a36Sopenharmony_ci	int ret, i;
32162306a36Sopenharmony_ci	struct zl6100_data *data;
32262306a36Sopenharmony_ci	struct pmbus_driver_info *info;
32362306a36Sopenharmony_ci	u8 device_id[I2C_SMBUS_BLOCK_MAX + 1];
32462306a36Sopenharmony_ci	const struct i2c_device_id *mid;
32562306a36Sopenharmony_ci
32662306a36Sopenharmony_ci	if (!i2c_check_functionality(client->adapter,
32762306a36Sopenharmony_ci				     I2C_FUNC_SMBUS_READ_WORD_DATA
32862306a36Sopenharmony_ci				     | I2C_FUNC_SMBUS_READ_BLOCK_DATA))
32962306a36Sopenharmony_ci		return -ENODEV;
33062306a36Sopenharmony_ci
33162306a36Sopenharmony_ci	ret = i2c_smbus_read_block_data(client, ZL6100_DEVICE_ID,
33262306a36Sopenharmony_ci					device_id);
33362306a36Sopenharmony_ci	if (ret < 0) {
33462306a36Sopenharmony_ci		dev_err(&client->dev, "Failed to read device ID\n");
33562306a36Sopenharmony_ci		return ret;
33662306a36Sopenharmony_ci	}
33762306a36Sopenharmony_ci	device_id[ret] = '\0';
33862306a36Sopenharmony_ci	dev_info(&client->dev, "Device ID %s\n", device_id);
33962306a36Sopenharmony_ci
34062306a36Sopenharmony_ci	mid = NULL;
34162306a36Sopenharmony_ci	for (mid = zl6100_id; mid->name[0]; mid++) {
34262306a36Sopenharmony_ci		if (!strncasecmp(mid->name, device_id, strlen(mid->name)))
34362306a36Sopenharmony_ci			break;
34462306a36Sopenharmony_ci	}
34562306a36Sopenharmony_ci	if (!mid->name[0]) {
34662306a36Sopenharmony_ci		dev_err(&client->dev, "Unsupported device\n");
34762306a36Sopenharmony_ci		return -ENODEV;
34862306a36Sopenharmony_ci	}
34962306a36Sopenharmony_ci	if (strcmp(client->name, mid->name) != 0)
35062306a36Sopenharmony_ci		dev_notice(&client->dev,
35162306a36Sopenharmony_ci			   "Device mismatch: Configured %s, detected %s\n",
35262306a36Sopenharmony_ci			   client->name, mid->name);
35362306a36Sopenharmony_ci
35462306a36Sopenharmony_ci	data = devm_kzalloc(&client->dev, sizeof(struct zl6100_data),
35562306a36Sopenharmony_ci			    GFP_KERNEL);
35662306a36Sopenharmony_ci	if (!data)
35762306a36Sopenharmony_ci		return -ENOMEM;
35862306a36Sopenharmony_ci
35962306a36Sopenharmony_ci	data->id = mid->driver_data;
36062306a36Sopenharmony_ci
36162306a36Sopenharmony_ci	/*
36262306a36Sopenharmony_ci	 * According to information from the chip vendor, all currently
36362306a36Sopenharmony_ci	 * supported chips are known to require a wait time between I2C
36462306a36Sopenharmony_ci	 * accesses.
36562306a36Sopenharmony_ci	 */
36662306a36Sopenharmony_ci	data->delay = delay;
36762306a36Sopenharmony_ci
36862306a36Sopenharmony_ci	/*
36962306a36Sopenharmony_ci	 * Since there was a direct I2C device access above, wait before
37062306a36Sopenharmony_ci	 * accessing the chip again.
37162306a36Sopenharmony_ci	 */
37262306a36Sopenharmony_ci	data->access = ktime_get();
37362306a36Sopenharmony_ci	zl6100_wait(data);
37462306a36Sopenharmony_ci
37562306a36Sopenharmony_ci	info = &data->info;
37662306a36Sopenharmony_ci
37762306a36Sopenharmony_ci	info->pages = 1;
37862306a36Sopenharmony_ci	info->func[0] = PMBUS_HAVE_VIN | PMBUS_HAVE_STATUS_INPUT
37962306a36Sopenharmony_ci	  | PMBUS_HAVE_VOUT | PMBUS_HAVE_STATUS_VOUT
38062306a36Sopenharmony_ci	  | PMBUS_HAVE_IOUT | PMBUS_HAVE_STATUS_IOUT
38162306a36Sopenharmony_ci	  | PMBUS_HAVE_TEMP | PMBUS_HAVE_STATUS_TEMP;
38262306a36Sopenharmony_ci
38362306a36Sopenharmony_ci	/*
38462306a36Sopenharmony_ci	 * ZL2004, ZL8802, ZL9101M, ZL9117M and ZLS4009 support monitoring
38562306a36Sopenharmony_ci	 * an extra voltage (VMON for ZL2004, ZL8802 and ZLS4009,
38662306a36Sopenharmony_ci	 * VDRV for ZL9101M and ZL9117M). Report it as vmon.
38762306a36Sopenharmony_ci	 */
38862306a36Sopenharmony_ci	if (data->id == zl2004 || data->id == zl8802 || data->id == zl9101 ||
38962306a36Sopenharmony_ci	    data->id == zl9117 || data->id == zls4009)
39062306a36Sopenharmony_ci		info->func[0] |= PMBUS_HAVE_VMON | PMBUS_HAVE_STATUS_VMON;
39162306a36Sopenharmony_ci
39262306a36Sopenharmony_ci	/*
39362306a36Sopenharmony_ci	 * ZL8802 has two outputs that can be used either independently or in
39462306a36Sopenharmony_ci	 * a current sharing configuration. The driver uses the DDC_CONFIG
39562306a36Sopenharmony_ci	 * register to check if the module is running with independent or
39662306a36Sopenharmony_ci	 * shared outputs. If the module is in shared output mode, only one
39762306a36Sopenharmony_ci	 * output voltage will be reported.
39862306a36Sopenharmony_ci	 */
39962306a36Sopenharmony_ci	if (data->id == zl8802) {
40062306a36Sopenharmony_ci		info->pages = 2;
40162306a36Sopenharmony_ci		info->func[0] |= PMBUS_HAVE_IIN;
40262306a36Sopenharmony_ci
40362306a36Sopenharmony_ci		ret = i2c_smbus_read_word_data(client, ZL8802_MFR_DDC_CONFIG);
40462306a36Sopenharmony_ci		if (ret < 0)
40562306a36Sopenharmony_ci			return ret;
40662306a36Sopenharmony_ci
40762306a36Sopenharmony_ci		data->access = ktime_get();
40862306a36Sopenharmony_ci		zl6100_wait(data);
40962306a36Sopenharmony_ci
41062306a36Sopenharmony_ci		if (ret & ZL8802_MFR_PHASES_MASK)
41162306a36Sopenharmony_ci			info->func[1] |= PMBUS_HAVE_IOUT | PMBUS_HAVE_STATUS_IOUT;
41262306a36Sopenharmony_ci		else
41362306a36Sopenharmony_ci			info->func[1] = PMBUS_HAVE_VOUT | PMBUS_HAVE_STATUS_VOUT
41462306a36Sopenharmony_ci				| PMBUS_HAVE_IOUT | PMBUS_HAVE_STATUS_IOUT;
41562306a36Sopenharmony_ci
41662306a36Sopenharmony_ci		for (i = 0; i < 2; i++) {
41762306a36Sopenharmony_ci			ret = i2c_smbus_write_byte_data(client, PMBUS_PAGE, i);
41862306a36Sopenharmony_ci			if (ret < 0)
41962306a36Sopenharmony_ci				return ret;
42062306a36Sopenharmony_ci
42162306a36Sopenharmony_ci			data->access = ktime_get();
42262306a36Sopenharmony_ci			zl6100_wait(data);
42362306a36Sopenharmony_ci
42462306a36Sopenharmony_ci			ret = i2c_smbus_read_word_data(client, ZL8802_MFR_USER_CONFIG);
42562306a36Sopenharmony_ci			if (ret < 0)
42662306a36Sopenharmony_ci				return ret;
42762306a36Sopenharmony_ci
42862306a36Sopenharmony_ci			if (ret & ZL8802_MFR_XTEMP_ENABLE_2)
42962306a36Sopenharmony_ci				info->func[i] |= PMBUS_HAVE_TEMP2;
43062306a36Sopenharmony_ci
43162306a36Sopenharmony_ci			data->access = ktime_get();
43262306a36Sopenharmony_ci			zl6100_wait(data);
43362306a36Sopenharmony_ci		}
43462306a36Sopenharmony_ci		ret = i2c_smbus_read_word_data(client, ZL8802_MFR_USER_GLOBAL_CONFIG);
43562306a36Sopenharmony_ci		if (ret < 0)
43662306a36Sopenharmony_ci			return ret;
43762306a36Sopenharmony_ci
43862306a36Sopenharmony_ci		if (ret & ZL8802_MFR_TMON_ENABLE)
43962306a36Sopenharmony_ci			info->func[0] |= PMBUS_HAVE_TEMP3;
44062306a36Sopenharmony_ci	} else {
44162306a36Sopenharmony_ci		ret = i2c_smbus_read_word_data(client, ZL6100_MFR_CONFIG);
44262306a36Sopenharmony_ci		if (ret < 0)
44362306a36Sopenharmony_ci			return ret;
44462306a36Sopenharmony_ci
44562306a36Sopenharmony_ci		if (ret & ZL6100_MFR_XTEMP_ENABLE)
44662306a36Sopenharmony_ci			info->func[0] |= PMBUS_HAVE_TEMP2;
44762306a36Sopenharmony_ci	}
44862306a36Sopenharmony_ci
44962306a36Sopenharmony_ci	data->access = ktime_get();
45062306a36Sopenharmony_ci	zl6100_wait(data);
45162306a36Sopenharmony_ci
45262306a36Sopenharmony_ci	info->read_word_data = zl6100_read_word_data;
45362306a36Sopenharmony_ci	info->read_byte_data = zl6100_read_byte_data;
45462306a36Sopenharmony_ci	info->write_word_data = zl6100_write_word_data;
45562306a36Sopenharmony_ci	info->write_byte = zl6100_write_byte;
45662306a36Sopenharmony_ci
45762306a36Sopenharmony_ci	return pmbus_do_probe(client, info);
45862306a36Sopenharmony_ci}
45962306a36Sopenharmony_ci
46062306a36Sopenharmony_cistatic struct i2c_driver zl6100_driver = {
46162306a36Sopenharmony_ci	.driver = {
46262306a36Sopenharmony_ci		   .name = "zl6100",
46362306a36Sopenharmony_ci		   },
46462306a36Sopenharmony_ci	.probe = zl6100_probe,
46562306a36Sopenharmony_ci	.id_table = zl6100_id,
46662306a36Sopenharmony_ci};
46762306a36Sopenharmony_ci
46862306a36Sopenharmony_cimodule_i2c_driver(zl6100_driver);
46962306a36Sopenharmony_ci
47062306a36Sopenharmony_ciMODULE_AUTHOR("Guenter Roeck");
47162306a36Sopenharmony_ciMODULE_DESCRIPTION("PMBus driver for ZL6100 and compatibles");
47262306a36Sopenharmony_ciMODULE_LICENSE("GPL");
47362306a36Sopenharmony_ciMODULE_IMPORT_NS(PMBUS);
474