Contributors: 12
Author Tokens Token Proportion Commits Commit Proportion
Andre Draszik 436 46.28% 10 31.25%
Mark Brown 213 22.61% 2 6.25%
Sangbeom Kim 158 16.77% 1 3.12%
Krzysztof Kozlowski 44 4.67% 9 28.12%
Chanwoo Choi 35 3.72% 3 9.38%
Sachin Kamat 27 2.87% 1 3.12%
Thomas Abraham 7 0.74% 1 3.12%
Amit Daniel Kachhap 7 0.74% 1 3.12%
Yadwinder Singh Brar 6 0.64% 1 3.12%
Dzmitry Sankouski 5 0.53% 1 3.12%
Paul Cercueil 3 0.32% 1 3.12%
Axel Lin 1 0.11% 1 3.12%
Total 942 32


// SPDX-License-Identifier: GPL-2.0+
/*
 * Copyright 2012 Samsung Electronics Co., Ltd
 *                http://www.samsung.com
 * Copyright 2025 Linaro Ltd.
 *
 * Samsung SxM I2C driver
 */

#include <linux/dev_printk.h>
#include <linux/err.h>
#include <linux/i2c.h>
#include <linux/mfd/samsung/core.h>
#include <linux/mfd/samsung/s2mpa01.h>
#include <linux/mfd/samsung/s2mps11.h>
#include <linux/mfd/samsung/s2mps13.h>
#include <linux/mfd/samsung/s2mps14.h>
#include <linux/mfd/samsung/s2mps15.h>
#include <linux/mfd/samsung/s2mpu02.h>
#include <linux/mfd/samsung/s5m8767.h>
#include <linux/mod_devicetable.h>
#include <linux/module.h>
#include <linux/pm.h>
#include <linux/property.h>
#include <linux/regmap.h>
#include "sec-core.h"

struct sec_pmic_i2c_platform_data {
	const struct regmap_config *regmap_cfg;
	int device_type;
};

static bool s2mpa01_volatile(struct device *dev, unsigned int reg)
{
	switch (reg) {
	case S2MPA01_REG_INT1M:
	case S2MPA01_REG_INT2M:
	case S2MPA01_REG_INT3M:
		return false;
	default:
		return true;
	}
}

static bool s2mps11_volatile(struct device *dev, unsigned int reg)
{
	switch (reg) {
	case S2MPS11_REG_INT1M:
	case S2MPS11_REG_INT2M:
	case S2MPS11_REG_INT3M:
		return false;
	default:
		return true;
	}
}

static bool s2mpu02_volatile(struct device *dev, unsigned int reg)
{
	switch (reg) {
	case S2MPU02_REG_INT1M:
	case S2MPU02_REG_INT2M:
	case S2MPU02_REG_INT3M:
		return false;
	default:
		return true;
	}
}

static const struct regmap_config s2dos05_regmap_config = {
	.reg_bits = 8,
	.val_bits = 8,
};

static const struct regmap_config s2mpa01_regmap_config = {
	.reg_bits = 8,
	.val_bits = 8,

	.max_register = S2MPA01_REG_LDO_OVCB4,
	.volatile_reg = s2mpa01_volatile,
	.cache_type = REGCACHE_FLAT,
};

static const struct regmap_config s2mps11_regmap_config = {
	.reg_bits = 8,
	.val_bits = 8,

	.max_register = S2MPS11_REG_L38CTRL,
	.volatile_reg = s2mps11_volatile,
	.cache_type = REGCACHE_FLAT,
};

static const struct regmap_config s2mps13_regmap_config = {
	.reg_bits = 8,
	.val_bits = 8,

	.max_register = S2MPS13_REG_LDODSCH5,
	.volatile_reg = s2mps11_volatile,
	.cache_type = REGCACHE_FLAT,
};

static const struct regmap_config s2mps14_regmap_config = {
	.reg_bits = 8,
	.val_bits = 8,

	.max_register = S2MPS14_REG_LDODSCH3,
	.volatile_reg = s2mps11_volatile,
	.cache_type = REGCACHE_FLAT,
};

static const struct regmap_config s2mps15_regmap_config = {
	.reg_bits = 8,
	.val_bits = 8,

	.max_register = S2MPS15_REG_LDODSCH4,
	.volatile_reg = s2mps11_volatile,
	.cache_type = REGCACHE_FLAT,
};

static const struct regmap_config s2mpu02_regmap_config = {
	.reg_bits = 8,
	.val_bits = 8,

	.max_register = S2MPU02_REG_DVSDATA,
	.volatile_reg = s2mpu02_volatile,
	.cache_type = REGCACHE_FLAT,
};

static const struct regmap_config s2mpu05_regmap_config = {
	.reg_bits = 8,
	.val_bits = 8,
};

static const struct regmap_config s5m8767_regmap_config = {
	.reg_bits = 8,
	.val_bits = 8,

	.max_register = S5M8767_REG_LDO28CTRL,
	.volatile_reg = s2mps11_volatile,
	.cache_type = REGCACHE_FLAT,
};

static int sec_pmic_i2c_probe(struct i2c_client *client)
{
	const struct sec_pmic_i2c_platform_data *pdata;
	struct regmap *regmap_pmic;

	pdata = device_get_match_data(&client->dev);
	if (!pdata)
		return dev_err_probe(&client->dev, -ENODEV,
				     "Unsupported device type\n");

	regmap_pmic = devm_regmap_init_i2c(client, pdata->regmap_cfg);
	if (IS_ERR(regmap_pmic))
		return dev_err_probe(&client->dev, PTR_ERR(regmap_pmic),
				     "regmap init failed\n");

	return sec_pmic_probe(&client->dev, pdata->device_type, client->irq,
			      regmap_pmic, client);
}

static void sec_pmic_i2c_shutdown(struct i2c_client *i2c)
{
	sec_pmic_shutdown(&i2c->dev);
}

static const struct sec_pmic_i2c_platform_data s2dos05_data = {
	.regmap_cfg = &s2dos05_regmap_config,
	.device_type = S2DOS05
};

static const struct sec_pmic_i2c_platform_data s2mpa01_data = {
	.regmap_cfg = &s2mpa01_regmap_config,
	.device_type = S2MPA01,
};

static const struct sec_pmic_i2c_platform_data s2mps11_data = {
	.regmap_cfg = &s2mps11_regmap_config,
	.device_type = S2MPS11X,
};

static const struct sec_pmic_i2c_platform_data s2mps13_data = {
	.regmap_cfg = &s2mps13_regmap_config,
	.device_type = S2MPS13X,
};

static const struct sec_pmic_i2c_platform_data s2mps14_data = {
	.regmap_cfg = &s2mps14_regmap_config,
	.device_type = S2MPS14X,
};

static const struct sec_pmic_i2c_platform_data s2mps15_data = {
	.regmap_cfg = &s2mps15_regmap_config,
	.device_type = S2MPS15X,
};

static const struct sec_pmic_i2c_platform_data s2mpu02_data = {
	.regmap_cfg = &s2mpu02_regmap_config,
	.device_type = S2MPU02,
};

static const struct sec_pmic_i2c_platform_data s2mpu05_data = {
	.regmap_cfg = &s2mpu05_regmap_config,
	.device_type = S2MPU05,
};

static const struct sec_pmic_i2c_platform_data s5m8767_data = {
	.regmap_cfg = &s5m8767_regmap_config,
	.device_type = S5M8767X,
};

static const struct of_device_id sec_pmic_i2c_of_match[] = {
	{ .compatible = "samsung,s2dos05", .data = &s2dos05_data, },
	{ .compatible = "samsung,s2mpa01-pmic", .data = &s2mpa01_data, },
	{ .compatible = "samsung,s2mps11-pmic", .data = &s2mps11_data, },
	{ .compatible = "samsung,s2mps13-pmic", .data = &s2mps13_data, },
	{ .compatible = "samsung,s2mps14-pmic", .data = &s2mps14_data, },
	{ .compatible = "samsung,s2mps15-pmic", .data = &s2mps15_data, },
	{ .compatible = "samsung,s2mpu02-pmic", .data = &s2mpu02_data, },
	{ .compatible = "samsung,s2mpu05-pmic", .data = &s2mpu05_data, },
	{ .compatible = "samsung,s5m8767-pmic", .data = &s5m8767_data, },
	{ },
};
MODULE_DEVICE_TABLE(of, sec_pmic_i2c_of_match);

static struct i2c_driver sec_pmic_i2c_driver = {
	.driver = {
		.name = "sec-pmic-i2c",
		.pm = pm_sleep_ptr(&sec_pmic_pm_ops),
		.of_match_table = sec_pmic_i2c_of_match,
	},
	.probe = sec_pmic_i2c_probe,
	.shutdown = sec_pmic_i2c_shutdown,
};
module_i2c_driver(sec_pmic_i2c_driver);

MODULE_AUTHOR("Sangbeom Kim <sbkim73@samsung.com>");
MODULE_AUTHOR("André Draszik <andre.draszik@linaro.org>");
MODULE_DESCRIPTION("I2C driver for the Samsung S5M");
MODULE_LICENSE("GPL");