xref: /linux/drivers/mfd/88pm886.c (revision a1ff5a7d78a036d6c2178ee5acd6ba4946243800)
1*860f8e3bSKarel Balej // SPDX-License-Identifier: GPL-2.0-only
2*860f8e3bSKarel Balej #include <linux/i2c.h>
3*860f8e3bSKarel Balej #include <linux/mfd/core.h>
4*860f8e3bSKarel Balej #include <linux/module.h>
5*860f8e3bSKarel Balej #include <linux/notifier.h>
6*860f8e3bSKarel Balej #include <linux/of.h>
7*860f8e3bSKarel Balej #include <linux/platform_device.h>
8*860f8e3bSKarel Balej #include <linux/reboot.h>
9*860f8e3bSKarel Balej #include <linux/regmap.h>
10*860f8e3bSKarel Balej 
11*860f8e3bSKarel Balej #include <linux/mfd/88pm886.h>
12*860f8e3bSKarel Balej 
13*860f8e3bSKarel Balej static const struct regmap_config pm886_regmap_config = {
14*860f8e3bSKarel Balej 	.reg_bits = 8,
15*860f8e3bSKarel Balej 	.val_bits = 8,
16*860f8e3bSKarel Balej 	.max_register = PM886_REG_RTC_SPARE6,
17*860f8e3bSKarel Balej };
18*860f8e3bSKarel Balej 
19*860f8e3bSKarel Balej static struct regmap_irq pm886_regmap_irqs[] = {
20*860f8e3bSKarel Balej 	REGMAP_IRQ_REG(PM886_IRQ_ONKEY, 0, PM886_INT_ENA1_ONKEY),
21*860f8e3bSKarel Balej };
22*860f8e3bSKarel Balej 
23*860f8e3bSKarel Balej static struct regmap_irq_chip pm886_regmap_irq_chip = {
24*860f8e3bSKarel Balej 	.name = "88pm886",
25*860f8e3bSKarel Balej 	.irqs = pm886_regmap_irqs,
26*860f8e3bSKarel Balej 	.num_irqs = ARRAY_SIZE(pm886_regmap_irqs),
27*860f8e3bSKarel Balej 	.num_regs = 4,
28*860f8e3bSKarel Balej 	.status_base = PM886_REG_INT_STATUS1,
29*860f8e3bSKarel Balej 	.ack_base = PM886_REG_INT_STATUS1,
30*860f8e3bSKarel Balej 	.unmask_base = PM886_REG_INT_ENA_1,
31*860f8e3bSKarel Balej };
32*860f8e3bSKarel Balej 
33*860f8e3bSKarel Balej static struct resource pm886_onkey_resources[] = {
34*860f8e3bSKarel Balej 	DEFINE_RES_IRQ_NAMED(PM886_IRQ_ONKEY, "88pm886-onkey"),
35*860f8e3bSKarel Balej };
36*860f8e3bSKarel Balej 
37*860f8e3bSKarel Balej static struct mfd_cell pm886_devs[] = {
38*860f8e3bSKarel Balej 	MFD_CELL_RES("88pm886-onkey", pm886_onkey_resources),
39*860f8e3bSKarel Balej 	MFD_CELL_NAME("88pm886-regulator"),
40*860f8e3bSKarel Balej };
41*860f8e3bSKarel Balej 
pm886_power_off_handler(struct sys_off_data * sys_off_data)42*860f8e3bSKarel Balej static int pm886_power_off_handler(struct sys_off_data *sys_off_data)
43*860f8e3bSKarel Balej {
44*860f8e3bSKarel Balej 	struct pm886_chip *chip = sys_off_data->cb_data;
45*860f8e3bSKarel Balej 	struct regmap *regmap = chip->regmap;
46*860f8e3bSKarel Balej 	struct device *dev = &chip->client->dev;
47*860f8e3bSKarel Balej 	int err;
48*860f8e3bSKarel Balej 
49*860f8e3bSKarel Balej 	err = regmap_update_bits(regmap, PM886_REG_MISC_CONFIG1, PM886_SW_PDOWN, PM886_SW_PDOWN);
50*860f8e3bSKarel Balej 	if (err) {
51*860f8e3bSKarel Balej 		dev_err(dev, "Failed to power off the device: %d\n", err);
52*860f8e3bSKarel Balej 		return NOTIFY_BAD;
53*860f8e3bSKarel Balej 	}
54*860f8e3bSKarel Balej 	return NOTIFY_DONE;
55*860f8e3bSKarel Balej }
56*860f8e3bSKarel Balej 
pm886_setup_irq(struct pm886_chip * chip,struct regmap_irq_chip_data ** irq_data)57*860f8e3bSKarel Balej static int pm886_setup_irq(struct pm886_chip *chip,
58*860f8e3bSKarel Balej 		struct regmap_irq_chip_data **irq_data)
59*860f8e3bSKarel Balej {
60*860f8e3bSKarel Balej 	struct regmap *regmap = chip->regmap;
61*860f8e3bSKarel Balej 	struct device *dev = &chip->client->dev;
62*860f8e3bSKarel Balej 	int err;
63*860f8e3bSKarel Balej 
64*860f8e3bSKarel Balej 	/* Set interrupt clearing mode to clear on write. */
65*860f8e3bSKarel Balej 	err = regmap_update_bits(regmap, PM886_REG_MISC_CONFIG2,
66*860f8e3bSKarel Balej 			PM886_INT_INV | PM886_INT_CLEAR | PM886_INT_MASK_MODE,
67*860f8e3bSKarel Balej 			PM886_INT_WC);
68*860f8e3bSKarel Balej 	if (err) {
69*860f8e3bSKarel Balej 		dev_err(dev, "Failed to set interrupt clearing mode: %d\n", err);
70*860f8e3bSKarel Balej 		return err;
71*860f8e3bSKarel Balej 	}
72*860f8e3bSKarel Balej 
73*860f8e3bSKarel Balej 	err = devm_regmap_add_irq_chip(dev, regmap, chip->client->irq,
74*860f8e3bSKarel Balej 					IRQF_ONESHOT, 0, &pm886_regmap_irq_chip,
75*860f8e3bSKarel Balej 					irq_data);
76*860f8e3bSKarel Balej 	if (err) {
77*860f8e3bSKarel Balej 		dev_err(dev, "Failed to request IRQ: %d\n", err);
78*860f8e3bSKarel Balej 		return err;
79*860f8e3bSKarel Balej 	}
80*860f8e3bSKarel Balej 
81*860f8e3bSKarel Balej 	return 0;
82*860f8e3bSKarel Balej }
83*860f8e3bSKarel Balej 
pm886_probe(struct i2c_client * client)84*860f8e3bSKarel Balej static int pm886_probe(struct i2c_client *client)
85*860f8e3bSKarel Balej {
86*860f8e3bSKarel Balej 	struct regmap_irq_chip_data *irq_data;
87*860f8e3bSKarel Balej 	struct device *dev = &client->dev;
88*860f8e3bSKarel Balej 	struct pm886_chip *chip;
89*860f8e3bSKarel Balej 	struct regmap *regmap;
90*860f8e3bSKarel Balej 	unsigned int chip_id;
91*860f8e3bSKarel Balej 	int err;
92*860f8e3bSKarel Balej 
93*860f8e3bSKarel Balej 	chip = devm_kzalloc(dev, sizeof(*chip), GFP_KERNEL);
94*860f8e3bSKarel Balej 	if (!chip)
95*860f8e3bSKarel Balej 		return -ENOMEM;
96*860f8e3bSKarel Balej 
97*860f8e3bSKarel Balej 	chip->client = client;
98*860f8e3bSKarel Balej 	chip->chip_id = (uintptr_t)device_get_match_data(dev);
99*860f8e3bSKarel Balej 	i2c_set_clientdata(client, chip);
100*860f8e3bSKarel Balej 
101*860f8e3bSKarel Balej 	regmap = devm_regmap_init_i2c(client, &pm886_regmap_config);
102*860f8e3bSKarel Balej 	if (IS_ERR(regmap))
103*860f8e3bSKarel Balej 		return dev_err_probe(dev, PTR_ERR(regmap), "Failed to initialize regmap\n");
104*860f8e3bSKarel Balej 	chip->regmap = regmap;
105*860f8e3bSKarel Balej 
106*860f8e3bSKarel Balej 	err = regmap_read(regmap, PM886_REG_ID, &chip_id);
107*860f8e3bSKarel Balej 	if (err)
108*860f8e3bSKarel Balej 		return dev_err_probe(dev, err, "Failed to read chip ID\n");
109*860f8e3bSKarel Balej 
110*860f8e3bSKarel Balej 	if (chip->chip_id != chip_id)
111*860f8e3bSKarel Balej 		return dev_err_probe(dev, -EINVAL, "Unsupported chip: 0x%x\n", chip_id);
112*860f8e3bSKarel Balej 
113*860f8e3bSKarel Balej 	err = pm886_setup_irq(chip, &irq_data);
114*860f8e3bSKarel Balej 	if (err)
115*860f8e3bSKarel Balej 		return err;
116*860f8e3bSKarel Balej 
117*860f8e3bSKarel Balej 	err = devm_mfd_add_devices(dev, PLATFORM_DEVID_NONE, pm886_devs, ARRAY_SIZE(pm886_devs),
118*860f8e3bSKarel Balej 				NULL, 0, regmap_irq_get_domain(irq_data));
119*860f8e3bSKarel Balej 	if (err)
120*860f8e3bSKarel Balej 		return dev_err_probe(dev, err, "Failed to add devices\n");
121*860f8e3bSKarel Balej 
122*860f8e3bSKarel Balej 	err = devm_register_power_off_handler(dev, pm886_power_off_handler, chip);
123*860f8e3bSKarel Balej 	if (err)
124*860f8e3bSKarel Balej 		return dev_err_probe(dev, err, "Failed to register power off handler\n");
125*860f8e3bSKarel Balej 
126*860f8e3bSKarel Balej 	device_init_wakeup(dev, device_property_read_bool(dev, "wakeup-source"));
127*860f8e3bSKarel Balej 
128*860f8e3bSKarel Balej 	return 0;
129*860f8e3bSKarel Balej }
130*860f8e3bSKarel Balej 
131*860f8e3bSKarel Balej static const struct of_device_id pm886_of_match[] = {
132*860f8e3bSKarel Balej 	{ .compatible = "marvell,88pm886-a1", .data = (void *)PM886_A1_CHIP_ID },
133*860f8e3bSKarel Balej 	{ }
134*860f8e3bSKarel Balej };
135*860f8e3bSKarel Balej MODULE_DEVICE_TABLE(of, pm886_of_match);
136*860f8e3bSKarel Balej 
137*860f8e3bSKarel Balej static struct i2c_driver pm886_i2c_driver = {
138*860f8e3bSKarel Balej 	.driver = {
139*860f8e3bSKarel Balej 		.name = "88pm886",
140*860f8e3bSKarel Balej 		.of_match_table = pm886_of_match,
141*860f8e3bSKarel Balej 	},
142*860f8e3bSKarel Balej 	.probe = pm886_probe,
143*860f8e3bSKarel Balej };
144*860f8e3bSKarel Balej module_i2c_driver(pm886_i2c_driver);
145*860f8e3bSKarel Balej 
146*860f8e3bSKarel Balej MODULE_DESCRIPTION("Marvell 88PM886 PMIC driver");
147*860f8e3bSKarel Balej MODULE_AUTHOR("Karel Balej <balejk@matfyz.cz>");
148*860f8e3bSKarel Balej MODULE_LICENSE("GPL");
149