aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorJean Delvare <khali@linux-fr.org>2012-10-05 16:23:51 -0400
committerJean Delvare <khali@endymion.delvare>2012-10-05 16:23:51 -0400
commiteee543e8248150e8fb833943c71f40c7b1724600 (patch)
tree56efec8384b2bca4cf926583b52a7cf6136fa76f
parent5f3d2f2e1a63679cf1c4a4210f2f1cc2f335bef6 (diff)
i2c-mux: Add support for device auto-detection
Let I2C bus segments behind multiplexers have a class. This allows for device auto-detection on these segments. As long as parent segments don't share the same class, it should be fine. I implemented support in drivers i2c-mux-gpio and i2c-mux-pca954x. I left i2c-mux-pca9541 and i2c-mux-pinctrl alone for the moment as I don't know if this feature makes sense for the use cases of these drivers. Signed-off-by: Jean Delvare <khali@linux-fr.org> Cc: Peter Korsgaard <peter.korsgaard@barco.com> Cc: David Daney <david.daney@cavium.com> Cc: Michael Lawnick <ml.lawnick@gmx.de> Cc: Rodolfo Giometti <giometti@linux.it>
-rw-r--r--drivers/i2c/i2c-mux.c22
-rw-r--r--drivers/i2c/muxes/i2c-mux-gpio.c4
-rw-r--r--drivers/i2c/muxes/i2c-mux-pca9541.c2
-rw-r--r--drivers/i2c/muxes/i2c-mux-pca954x.c10
-rw-r--r--drivers/i2c/muxes/i2c-mux-pinctrl.c2
-rw-r--r--include/linux/i2c-mux-gpio.h2
-rw-r--r--include/linux/i2c-mux.h1
-rw-r--r--include/linux/i2c/pca954x.h1
8 files changed, 37 insertions, 7 deletions
diff --git a/drivers/i2c/i2c-mux.c b/drivers/i2c/i2c-mux.c
index 1038c381aea..d94e0ce7827 100644
--- a/drivers/i2c/i2c-mux.c
+++ b/drivers/i2c/i2c-mux.c
@@ -88,9 +88,23 @@ static u32 i2c_mux_functionality(struct i2c_adapter *adap)
88 return parent->algo->functionality(parent); 88 return parent->algo->functionality(parent);
89} 89}
90 90
91/* Return all parent classes, merged */
92static unsigned int i2c_mux_parent_classes(struct i2c_adapter *parent)
93{
94 unsigned int class = 0;
95
96 do {
97 class |= parent->class;
98 parent = i2c_parent_is_i2c_adapter(parent);
99 } while (parent);
100
101 return class;
102}
103
91struct i2c_adapter *i2c_add_mux_adapter(struct i2c_adapter *parent, 104struct i2c_adapter *i2c_add_mux_adapter(struct i2c_adapter *parent,
92 struct device *mux_dev, 105 struct device *mux_dev,
93 void *mux_priv, u32 force_nr, u32 chan_id, 106 void *mux_priv, u32 force_nr, u32 chan_id,
107 unsigned int class,
94 int (*select) (struct i2c_adapter *, 108 int (*select) (struct i2c_adapter *,
95 void *, u32), 109 void *, u32),
96 int (*deselect) (struct i2c_adapter *, 110 int (*deselect) (struct i2c_adapter *,
@@ -127,6 +141,14 @@ struct i2c_adapter *i2c_add_mux_adapter(struct i2c_adapter *parent,
127 priv->adap.algo_data = priv; 141 priv->adap.algo_data = priv;
128 priv->adap.dev.parent = &parent->dev; 142 priv->adap.dev.parent = &parent->dev;
129 143
144 /* Sanity check on class */
145 if (i2c_mux_parent_classes(parent) & class)
146 dev_err(&parent->dev,
147 "Segment %d behind mux can't share classes with ancestors\n",
148 chan_id);
149 else
150 priv->adap.class = class;
151
130 /* 152 /*
131 * Try to populate the mux adapter's of_node, expands to 153 * Try to populate the mux adapter's of_node, expands to
132 * nothing if !CONFIG_OF. 154 * nothing if !CONFIG_OF.
diff --git a/drivers/i2c/muxes/i2c-mux-gpio.c b/drivers/i2c/muxes/i2c-mux-gpio.c
index 68b1f8ec343..56889e00c76 100644
--- a/drivers/i2c/muxes/i2c-mux-gpio.c
+++ b/drivers/i2c/muxes/i2c-mux-gpio.c
@@ -104,8 +104,10 @@ static int __devinit i2c_mux_gpio_probe(struct platform_device *pdev)
104 104
105 for (i = 0; i < pdata->n_values; i++) { 105 for (i = 0; i < pdata->n_values; i++) {
106 u32 nr = pdata->base_nr ? (pdata->base_nr + i) : 0; 106 u32 nr = pdata->base_nr ? (pdata->base_nr + i) : 0;
107 unsigned int class = pdata->classes ? pdata->classes[i] : 0;
107 108
108 mux->adap[i] = i2c_add_mux_adapter(parent, &pdev->dev, mux, nr, i, 109 mux->adap[i] = i2c_add_mux_adapter(parent, &pdev->dev, mux, nr,
110 i, class,
109 i2c_mux_gpio_select, deselect); 111 i2c_mux_gpio_select, deselect);
110 if (!mux->adap[i]) { 112 if (!mux->adap[i]) {
111 ret = -ENODEV; 113 ret = -ENODEV;
diff --git a/drivers/i2c/muxes/i2c-mux-pca9541.c b/drivers/i2c/muxes/i2c-mux-pca9541.c
index f8f72f39e0b..f3b8f9a6a89 100644
--- a/drivers/i2c/muxes/i2c-mux-pca9541.c
+++ b/drivers/i2c/muxes/i2c-mux-pca9541.c
@@ -354,7 +354,7 @@ static int pca9541_probe(struct i2c_client *client,
354 if (pdata) 354 if (pdata)
355 force = pdata->modes[0].adap_id; 355 force = pdata->modes[0].adap_id;
356 data->mux_adap = i2c_add_mux_adapter(adap, &client->dev, client, 356 data->mux_adap = i2c_add_mux_adapter(adap, &client->dev, client,
357 force, 0, 357 force, 0, 0,
358 pca9541_select_chan, 358 pca9541_select_chan,
359 pca9541_release_chan); 359 pca9541_release_chan);
360 360
diff --git a/drivers/i2c/muxes/i2c-mux-pca954x.c b/drivers/i2c/muxes/i2c-mux-pca954x.c
index f2dfe0d8fcc..8e4387235b6 100644
--- a/drivers/i2c/muxes/i2c-mux-pca954x.c
+++ b/drivers/i2c/muxes/i2c-mux-pca954x.c
@@ -186,7 +186,7 @@ static int pca954x_probe(struct i2c_client *client,
186{ 186{
187 struct i2c_adapter *adap = to_i2c_adapter(client->dev.parent); 187 struct i2c_adapter *adap = to_i2c_adapter(client->dev.parent);
188 struct pca954x_platform_data *pdata = client->dev.platform_data; 188 struct pca954x_platform_data *pdata = client->dev.platform_data;
189 int num, force; 189 int num, force, class;
190 struct pca954x *data; 190 struct pca954x *data;
191 int ret = -ENODEV; 191 int ret = -ENODEV;
192 192
@@ -216,18 +216,20 @@ static int pca954x_probe(struct i2c_client *client,
216 /* Now create an adapter for each channel */ 216 /* Now create an adapter for each channel */
217 for (num = 0; num < chips[data->type].nchans; num++) { 217 for (num = 0; num < chips[data->type].nchans; num++) {
218 force = 0; /* dynamic adap number */ 218 force = 0; /* dynamic adap number */
219 class = 0; /* no class by default */
219 if (pdata) { 220 if (pdata) {
220 if (num < pdata->num_modes) 221 if (num < pdata->num_modes) {
221 /* force static number */ 222 /* force static number */
222 force = pdata->modes[num].adap_id; 223 force = pdata->modes[num].adap_id;
223 else 224 class = pdata->modes[num].class;
225 } else
224 /* discard unconfigured channels */ 226 /* discard unconfigured channels */
225 break; 227 break;
226 } 228 }
227 229
228 data->virt_adaps[num] = 230 data->virt_adaps[num] =
229 i2c_add_mux_adapter(adap, &client->dev, client, 231 i2c_add_mux_adapter(adap, &client->dev, client,
230 force, num, pca954x_select_chan, 232 force, num, class, pca954x_select_chan,
231 (pdata && pdata->modes[num].deselect_on_exit) 233 (pdata && pdata->modes[num].deselect_on_exit)
232 ? pca954x_deselect_mux : NULL); 234 ? pca954x_deselect_mux : NULL);
233 235
diff --git a/drivers/i2c/muxes/i2c-mux-pinctrl.c b/drivers/i2c/muxes/i2c-mux-pinctrl.c
index 46a66976347..5f097f309b9 100644
--- a/drivers/i2c/muxes/i2c-mux-pinctrl.c
+++ b/drivers/i2c/muxes/i2c-mux-pinctrl.c
@@ -221,7 +221,7 @@ static int __devinit i2c_mux_pinctrl_probe(struct platform_device *pdev)
221 (mux->pdata->base_bus_num + i) : 0; 221 (mux->pdata->base_bus_num + i) : 0;
222 222
223 mux->busses[i] = i2c_add_mux_adapter(mux->parent, &pdev->dev, 223 mux->busses[i] = i2c_add_mux_adapter(mux->parent, &pdev->dev,
224 mux, bus, i, 224 mux, bus, i, 0,
225 i2c_mux_pinctrl_select, 225 i2c_mux_pinctrl_select,
226 deselect); 226 deselect);
227 if (!mux->busses[i]) { 227 if (!mux->busses[i]) {
diff --git a/include/linux/i2c-mux-gpio.h b/include/linux/i2c-mux-gpio.h
index a36343a37eb..4ea1cc7392b 100644
--- a/include/linux/i2c-mux-gpio.h
+++ b/include/linux/i2c-mux-gpio.h
@@ -21,6 +21,7 @@
21 * @values: Array of bitmasks of GPIO settings (low/high) for each 21 * @values: Array of bitmasks of GPIO settings (low/high) for each
22 * position 22 * position
23 * @n_values: Number of multiplexer positions (busses to instantiate) 23 * @n_values: Number of multiplexer positions (busses to instantiate)
24 * @classes: Optional I2C auto-detection classes
24 * @gpios: Array of GPIO numbers used to control MUX 25 * @gpios: Array of GPIO numbers used to control MUX
25 * @n_gpios: Number of GPIOs used to control MUX 26 * @n_gpios: Number of GPIOs used to control MUX
26 * @idle: Bitmask to write to MUX when idle or GPIO_I2CMUX_NO_IDLE if not used 27 * @idle: Bitmask to write to MUX when idle or GPIO_I2CMUX_NO_IDLE if not used
@@ -30,6 +31,7 @@ struct i2c_mux_gpio_platform_data {
30 int base_nr; 31 int base_nr;
31 const unsigned *values; 32 const unsigned *values;
32 int n_values; 33 int n_values;
34 const unsigned *classes;
33 const unsigned *gpios; 35 const unsigned *gpios;
34 int n_gpios; 36 int n_gpios;
35 unsigned idle; 37 unsigned idle;
diff --git a/include/linux/i2c-mux.h b/include/linux/i2c-mux.h
index c7908383001..40cb05a97b4 100644
--- a/include/linux/i2c-mux.h
+++ b/include/linux/i2c-mux.h
@@ -36,6 +36,7 @@
36struct i2c_adapter *i2c_add_mux_adapter(struct i2c_adapter *parent, 36struct i2c_adapter *i2c_add_mux_adapter(struct i2c_adapter *parent,
37 struct device *mux_dev, 37 struct device *mux_dev,
38 void *mux_priv, u32 force_nr, u32 chan_id, 38 void *mux_priv, u32 force_nr, u32 chan_id,
39 unsigned int class,
39 int (*select) (struct i2c_adapter *, 40 int (*select) (struct i2c_adapter *,
40 void *mux_dev, u32 chan_id), 41 void *mux_dev, u32 chan_id),
41 int (*deselect) (struct i2c_adapter *, 42 int (*deselect) (struct i2c_adapter *,
diff --git a/include/linux/i2c/pca954x.h b/include/linux/i2c/pca954x.h
index 28f1f8d5ab1..1712677d590 100644
--- a/include/linux/i2c/pca954x.h
+++ b/include/linux/i2c/pca954x.h
@@ -36,6 +36,7 @@
36struct pca954x_platform_mode { 36struct pca954x_platform_mode {
37 int adap_id; 37 int adap_id;
38 unsigned int deselect_on_exit:1; 38 unsigned int deselect_on_exit:1;
39 unsigned int class;
39}; 40};
40 41
41/* Per mux/switch data, used with i2c_register_board_info */ 42/* Per mux/switch data, used with i2c_register_board_info */