Blame view

drivers/mfd/rn5t618.c 4.48 KB
9bb9e29c7   Beniamino Galvani   mfd: Add Ricoh RN...
1
2
3
4
  /*
   * MFD core driver for Ricoh RN5T618 PMIC
   *
   * Copyright (C) 2014 Beniamino Galvani <b.galvani@gmail.com>
a99ab50db   Stefan Agner   mfd: rn5t618: Add...
5
   * Copyright (C) 2016 Toradex AG
9bb9e29c7   Beniamino Galvani   mfd: Add Ricoh RN...
6
7
8
9
10
11
12
13
   *
   * This program is free software; you can redistribute it and/or
   * modify it under the terms of the GNU General Public License
   * version 2 as published by the Free Software Foundation.
   *
   * You should have received a copy of the GNU General Public License
   * along with this program. If not, see <http://www.gnu.org/licenses/>.
   */
a370f60a5   Stefan Agner   mfd: rn5t618: Reg...
14
  #include <linux/delay.h>
9bb9e29c7   Beniamino Galvani   mfd: Add Ricoh RN...
15
16
17
18
  #include <linux/i2c.h>
  #include <linux/mfd/core.h>
  #include <linux/mfd/rn5t618.h>
  #include <linux/module.h>
a99ab50db   Stefan Agner   mfd: rn5t618: Add...
19
  #include <linux/of_device.h>
a370f60a5   Stefan Agner   mfd: rn5t618: Reg...
20
  #include <linux/reboot.h>
9bb9e29c7   Beniamino Galvani   mfd: Add Ricoh RN...
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
  #include <linux/regmap.h>
  
  static const struct mfd_cell rn5t618_cells[] = {
  	{ .name = "rn5t618-regulator" },
  	{ .name = "rn5t618-wdt" },
  };
  
  static bool rn5t618_volatile_reg(struct device *dev, unsigned int reg)
  {
  	switch (reg) {
  	case RN5T618_WATCHDOGCNT:
  	case RN5T618_DCIRQ:
  	case RN5T618_ILIMDATAH ... RN5T618_AIN0DATAL:
  	case RN5T618_IR_ADC1 ... RN5T618_IR_ADC3:
  	case RN5T618_IR_GPR:
  	case RN5T618_IR_GPF:
  	case RN5T618_MON_IOIN:
  	case RN5T618_INTMON:
  		return true;
  	default:
  		return false;
  	}
  }
  
  static const struct regmap_config rn5t618_regmap_config = {
  	.reg_bits	= 8,
  	.val_bits	= 8,
  	.volatile_reg	= rn5t618_volatile_reg,
  	.max_register	= RN5T618_MAX_REG,
  	.cache_type	= REGCACHE_RBTREE,
  };
  
  static struct rn5t618 *rn5t618_pm_power_off;
a370f60a5   Stefan Agner   mfd: rn5t618: Reg...
54
  static struct notifier_block rn5t618_restart_handler;
9bb9e29c7   Beniamino Galvani   mfd: Add Ricoh RN...
55

a370f60a5   Stefan Agner   mfd: rn5t618: Reg...
56
  static void rn5t618_trigger_poweroff_sequence(bool repower)
9bb9e29c7   Beniamino Galvani   mfd: Add Ricoh RN...
57
58
59
  {
  	/* disable automatic repower-on */
  	regmap_update_bits(rn5t618_pm_power_off->regmap, RN5T618_REPCNT,
a370f60a5   Stefan Agner   mfd: rn5t618: Reg...
60
61
  			   RN5T618_REPCNT_REPWRON,
  			   repower ? RN5T618_REPCNT_REPWRON : 0);
9bb9e29c7   Beniamino Galvani   mfd: Add Ricoh RN...
62
63
64
65
  	/* start power-off sequence */
  	regmap_update_bits(rn5t618_pm_power_off->regmap, RN5T618_SLPCNT,
  			   RN5T618_SLPCNT_SWPWROFF, RN5T618_SLPCNT_SWPWROFF);
  }
a370f60a5   Stefan Agner   mfd: rn5t618: Reg...
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
  static void rn5t618_power_off(void)
  {
  	rn5t618_trigger_poweroff_sequence(false);
  }
  
  static int rn5t618_restart(struct notifier_block *this,
  			    unsigned long mode, void *cmd)
  {
  	rn5t618_trigger_poweroff_sequence(true);
  
  	/*
  	 * Re-power factor detection on PMIC side is not instant. 1ms
  	 * proved to be enough time until reset takes effect.
  	 */
  	mdelay(1);
  
  	return NOTIFY_DONE;
  }
a99ab50db   Stefan Agner   mfd: rn5t618: Add...
84
85
86
87
88
89
  static const struct of_device_id rn5t618_of_match[] = {
  	{ .compatible = "ricoh,rn5t567", .data = (void *)RN5T567 },
  	{ .compatible = "ricoh,rn5t618", .data = (void *)RN5T618 },
  	{ }
  };
  MODULE_DEVICE_TABLE(of, rn5t618_of_match);
9bb9e29c7   Beniamino Galvani   mfd: Add Ricoh RN...
90
91
92
  static int rn5t618_i2c_probe(struct i2c_client *i2c,
  			     const struct i2c_device_id *id)
  {
a99ab50db   Stefan Agner   mfd: rn5t618: Add...
93
  	const struct of_device_id *of_id;
9bb9e29c7   Beniamino Galvani   mfd: Add Ricoh RN...
94
95
  	struct rn5t618 *priv;
  	int ret;
a99ab50db   Stefan Agner   mfd: rn5t618: Add...
96
97
98
99
100
101
  	of_id = of_match_device(rn5t618_of_match, &i2c->dev);
  	if (!of_id) {
  		dev_err(&i2c->dev, "Failed to find matching DT ID
  ");
  		return -EINVAL;
  	}
9bb9e29c7   Beniamino Galvani   mfd: Add Ricoh RN...
102
103
104
105
106
  	priv = devm_kzalloc(&i2c->dev, sizeof(*priv), GFP_KERNEL);
  	if (!priv)
  		return -ENOMEM;
  
  	i2c_set_clientdata(i2c, priv);
a99ab50db   Stefan Agner   mfd: rn5t618: Add...
107
  	priv->variant = (long)of_id->data;
9bb9e29c7   Beniamino Galvani   mfd: Add Ricoh RN...
108
109
110
111
112
113
114
115
  
  	priv->regmap = devm_regmap_init_i2c(i2c, &rn5t618_regmap_config);
  	if (IS_ERR(priv->regmap)) {
  		ret = PTR_ERR(priv->regmap);
  		dev_err(&i2c->dev, "regmap init failed: %d
  ", ret);
  		return ret;
  	}
f41206c9f   Laxman Dewangan   mfd: rn5t618: Use...
116
117
  	ret = devm_mfd_add_devices(&i2c->dev, -1, rn5t618_cells,
  				   ARRAY_SIZE(rn5t618_cells), NULL, 0, NULL);
9bb9e29c7   Beniamino Galvani   mfd: Add Ricoh RN...
118
119
120
121
122
  	if (ret) {
  		dev_err(&i2c->dev, "failed to add sub-devices: %d
  ", ret);
  		return ret;
  	}
a370f60a5   Stefan Agner   mfd: rn5t618: Reg...
123
  	rn5t618_pm_power_off = priv;
fd65ae4cb   Stefan Agner   mfd: rn5t618: Reg...
124
  	if (of_device_is_system_power_controller(i2c->dev.of_node)) {
a370f60a5   Stefan Agner   mfd: rn5t618: Reg...
125
  		if (!pm_power_off)
fd65ae4cb   Stefan Agner   mfd: rn5t618: Reg...
126
  			pm_power_off = rn5t618_power_off;
a370f60a5   Stefan Agner   mfd: rn5t618: Reg...
127
  		else
fd65ae4cb   Stefan Agner   mfd: rn5t618: Reg...
128
129
  			dev_warn(&i2c->dev, "Poweroff callback already assigned
  ");
a370f60a5   Stefan Agner   mfd: rn5t618: Reg...
130
131
132
133
134
135
136
137
138
139
  	}
  
  	rn5t618_restart_handler.notifier_call = rn5t618_restart;
  	rn5t618_restart_handler.priority = 192;
  
  	ret = register_restart_handler(&rn5t618_restart_handler);
  	if (ret) {
  		dev_err(&i2c->dev, "cannot register restart handler, %d
  ", ret);
  		return ret;
9bb9e29c7   Beniamino Galvani   mfd: Add Ricoh RN...
140
141
142
143
144
145
146
147
148
149
150
151
152
  	}
  
  	return 0;
  }
  
  static int rn5t618_i2c_remove(struct i2c_client *i2c)
  {
  	struct rn5t618 *priv = i2c_get_clientdata(i2c);
  
  	if (priv == rn5t618_pm_power_off) {
  		rn5t618_pm_power_off = NULL;
  		pm_power_off = NULL;
  	}
9bb9e29c7   Beniamino Galvani   mfd: Add Ricoh RN...
153
154
  	return 0;
  }
9bb9e29c7   Beniamino Galvani   mfd: Add Ricoh RN...
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
  static const struct i2c_device_id rn5t618_i2c_id[] = {
  	{ }
  };
  MODULE_DEVICE_TABLE(i2c, rn5t618_i2c_id);
  
  static struct i2c_driver rn5t618_i2c_driver = {
  	.driver = {
  		.name = "rn5t618",
  		.of_match_table = of_match_ptr(rn5t618_of_match),
  	},
  	.probe = rn5t618_i2c_probe,
  	.remove = rn5t618_i2c_remove,
  	.id_table = rn5t618_i2c_id,
  };
  
  module_i2c_driver(rn5t618_i2c_driver);
  
  MODULE_AUTHOR("Beniamino Galvani <b.galvani@gmail.com>");
a99ab50db   Stefan Agner   mfd: rn5t618: Add...
173
  MODULE_DESCRIPTION("Ricoh RN5T567/618 MFD driver");
9bb9e29c7   Beniamino Galvani   mfd: Add Ricoh RN...
174
  MODULE_LICENSE("GPL v2");