162306a36Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0-only
262306a36Sopenharmony_ci/*
362306a36Sopenharmony_ci * Ampere Altra Family SMPro core driver
462306a36Sopenharmony_ci * Copyright (c) 2022, Ampere Computing LLC
562306a36Sopenharmony_ci */
662306a36Sopenharmony_ci
762306a36Sopenharmony_ci#include <linux/i2c.h>
862306a36Sopenharmony_ci#include <linux/kernel.h>
962306a36Sopenharmony_ci#include <linux/mfd/core.h>
1062306a36Sopenharmony_ci#include <linux/module.h>
1162306a36Sopenharmony_ci#include <linux/of_platform.h>
1262306a36Sopenharmony_ci#include <linux/regmap.h>
1362306a36Sopenharmony_ci
1462306a36Sopenharmony_ci/* Identification Registers */
1562306a36Sopenharmony_ci#define MANUFACTURER_ID_REG     0x02
1662306a36Sopenharmony_ci#define AMPERE_MANUFACTURER_ID  0xCD3A
1762306a36Sopenharmony_ci
1862306a36Sopenharmony_ci#define CORE_CE_ERR_DATA        0x82
1962306a36Sopenharmony_ci#define CORE_UE_ERR_DATA        0x85
2062306a36Sopenharmony_ci#define MEM_CE_ERR_DATA         0x92
2162306a36Sopenharmony_ci#define MEM_UE_ERR_DATA         0x95
2262306a36Sopenharmony_ci#define PCIE_CE_ERR_DATA        0xC2
2362306a36Sopenharmony_ci#define PCIE_UE_ERR_DATA        0xC5
2462306a36Sopenharmony_ci#define OTHER_CE_ERR_DATA       0xD2
2562306a36Sopenharmony_ci#define OTHER_UE_ERR_DATA       0xDA
2662306a36Sopenharmony_ci
2762306a36Sopenharmony_cistatic int smpro_core_write(void *context, const void *data, size_t count)
2862306a36Sopenharmony_ci{
2962306a36Sopenharmony_ci	struct device *dev = context;
3062306a36Sopenharmony_ci	struct i2c_client *i2c = to_i2c_client(dev);
3162306a36Sopenharmony_ci	int ret;
3262306a36Sopenharmony_ci
3362306a36Sopenharmony_ci	ret = i2c_master_send(i2c, data, count);
3462306a36Sopenharmony_ci	if (unlikely(ret != count))
3562306a36Sopenharmony_ci		return (ret < 0) ? ret : -EIO;
3662306a36Sopenharmony_ci
3762306a36Sopenharmony_ci	return 0;
3862306a36Sopenharmony_ci}
3962306a36Sopenharmony_ci
4062306a36Sopenharmony_cistatic int smpro_core_read(void *context, const void *reg, size_t reg_size,
4162306a36Sopenharmony_ci			   void *val, size_t val_size)
4262306a36Sopenharmony_ci{
4362306a36Sopenharmony_ci	struct device *dev = context;
4462306a36Sopenharmony_ci	struct i2c_client *i2c = to_i2c_client(dev);
4562306a36Sopenharmony_ci	struct i2c_msg xfer[2];
4662306a36Sopenharmony_ci	unsigned char buf[2];
4762306a36Sopenharmony_ci	int ret;
4862306a36Sopenharmony_ci
4962306a36Sopenharmony_ci	xfer[0].addr = i2c->addr;
5062306a36Sopenharmony_ci	xfer[0].flags = 0;
5162306a36Sopenharmony_ci
5262306a36Sopenharmony_ci	buf[0] = *(u8 *)reg;
5362306a36Sopenharmony_ci	buf[1] = val_size;
5462306a36Sopenharmony_ci	xfer[0].len = 2;
5562306a36Sopenharmony_ci	xfer[0].buf = buf;
5662306a36Sopenharmony_ci
5762306a36Sopenharmony_ci	xfer[1].addr = i2c->addr;
5862306a36Sopenharmony_ci	xfer[1].flags = I2C_M_RD;
5962306a36Sopenharmony_ci	xfer[1].len = val_size;
6062306a36Sopenharmony_ci	xfer[1].buf = val;
6162306a36Sopenharmony_ci
6262306a36Sopenharmony_ci	ret = i2c_transfer(i2c->adapter, xfer, 2);
6362306a36Sopenharmony_ci	if (unlikely(ret != 2))
6462306a36Sopenharmony_ci		return (ret < 0) ? ret : -EIO;
6562306a36Sopenharmony_ci
6662306a36Sopenharmony_ci	return 0;
6762306a36Sopenharmony_ci}
6862306a36Sopenharmony_ci
6962306a36Sopenharmony_cistatic const struct regmap_bus smpro_regmap_bus = {
7062306a36Sopenharmony_ci	.read = smpro_core_read,
7162306a36Sopenharmony_ci	.write = smpro_core_write,
7262306a36Sopenharmony_ci	.val_format_endian_default = REGMAP_ENDIAN_BIG,
7362306a36Sopenharmony_ci};
7462306a36Sopenharmony_ci
7562306a36Sopenharmony_cistatic bool smpro_core_readable_noinc_reg(struct device *dev, unsigned int reg)
7662306a36Sopenharmony_ci{
7762306a36Sopenharmony_ci	return  (reg == CORE_CE_ERR_DATA || reg == CORE_UE_ERR_DATA ||
7862306a36Sopenharmony_ci		 reg == MEM_CE_ERR_DATA || reg == MEM_UE_ERR_DATA ||
7962306a36Sopenharmony_ci		 reg == PCIE_CE_ERR_DATA || reg == PCIE_UE_ERR_DATA ||
8062306a36Sopenharmony_ci		 reg == OTHER_CE_ERR_DATA || reg == OTHER_UE_ERR_DATA);
8162306a36Sopenharmony_ci}
8262306a36Sopenharmony_ci
8362306a36Sopenharmony_cistatic const struct regmap_config smpro_regmap_config = {
8462306a36Sopenharmony_ci	.reg_bits = 8,
8562306a36Sopenharmony_ci	.val_bits = 16,
8662306a36Sopenharmony_ci	.readable_noinc_reg = smpro_core_readable_noinc_reg,
8762306a36Sopenharmony_ci};
8862306a36Sopenharmony_ci
8962306a36Sopenharmony_cistatic const struct mfd_cell smpro_devs[] = {
9062306a36Sopenharmony_ci	MFD_CELL_NAME("smpro-hwmon"),
9162306a36Sopenharmony_ci	MFD_CELL_NAME("smpro-errmon"),
9262306a36Sopenharmony_ci	MFD_CELL_NAME("smpro-misc"),
9362306a36Sopenharmony_ci};
9462306a36Sopenharmony_ci
9562306a36Sopenharmony_cistatic int smpro_core_probe(struct i2c_client *i2c)
9662306a36Sopenharmony_ci{
9762306a36Sopenharmony_ci	const struct regmap_config *config;
9862306a36Sopenharmony_ci	struct regmap *regmap;
9962306a36Sopenharmony_ci	unsigned int val;
10062306a36Sopenharmony_ci	int ret;
10162306a36Sopenharmony_ci
10262306a36Sopenharmony_ci	config = device_get_match_data(&i2c->dev);
10362306a36Sopenharmony_ci	if (!config)
10462306a36Sopenharmony_ci		return -EINVAL;
10562306a36Sopenharmony_ci
10662306a36Sopenharmony_ci	regmap = devm_regmap_init(&i2c->dev, &smpro_regmap_bus, &i2c->dev, config);
10762306a36Sopenharmony_ci	if (IS_ERR(regmap))
10862306a36Sopenharmony_ci		return PTR_ERR(regmap);
10962306a36Sopenharmony_ci
11062306a36Sopenharmony_ci	ret = regmap_read(regmap, MANUFACTURER_ID_REG, &val);
11162306a36Sopenharmony_ci	if (ret)
11262306a36Sopenharmony_ci		return ret;
11362306a36Sopenharmony_ci
11462306a36Sopenharmony_ci	if (val != AMPERE_MANUFACTURER_ID)
11562306a36Sopenharmony_ci		return -ENODEV;
11662306a36Sopenharmony_ci
11762306a36Sopenharmony_ci	return devm_mfd_add_devices(&i2c->dev, PLATFORM_DEVID_AUTO,
11862306a36Sopenharmony_ci				    smpro_devs, ARRAY_SIZE(smpro_devs), NULL, 0, NULL);
11962306a36Sopenharmony_ci}
12062306a36Sopenharmony_ci
12162306a36Sopenharmony_cistatic const struct of_device_id smpro_core_of_match[] = {
12262306a36Sopenharmony_ci	{ .compatible = "ampere,smpro", .data = &smpro_regmap_config },
12362306a36Sopenharmony_ci	{}
12462306a36Sopenharmony_ci};
12562306a36Sopenharmony_ciMODULE_DEVICE_TABLE(of, smpro_core_of_match);
12662306a36Sopenharmony_ci
12762306a36Sopenharmony_cistatic struct i2c_driver smpro_core_driver = {
12862306a36Sopenharmony_ci	.probe = smpro_core_probe,
12962306a36Sopenharmony_ci	.driver = {
13062306a36Sopenharmony_ci		.name = "smpro-core",
13162306a36Sopenharmony_ci		.of_match_table = smpro_core_of_match,
13262306a36Sopenharmony_ci	},
13362306a36Sopenharmony_ci};
13462306a36Sopenharmony_cimodule_i2c_driver(smpro_core_driver);
13562306a36Sopenharmony_ci
13662306a36Sopenharmony_ciMODULE_AUTHOR("Quan Nguyen <quan@os.amperecomputing.com>");
13762306a36Sopenharmony_ciMODULE_DESCRIPTION("SMPRO CORE - I2C driver");
13862306a36Sopenharmony_ciMODULE_LICENSE("GPL");
139