diff options
author | Jean Delvare <khali@linux-fr.org> | 2012-10-05 16:23:51 -0400 |
---|---|---|
committer | Jean Delvare <khali@endymion.delvare> | 2012-10-05 16:23:51 -0400 |
commit | eee543e8248150e8fb833943c71f40c7b1724600 (patch) | |
tree | 56efec8384b2bca4cf926583b52a7cf6136fa76f | |
parent | 5f3d2f2e1a63679cf1c4a4210f2f1cc2f335bef6 (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.c | 22 | ||||
-rw-r--r-- | drivers/i2c/muxes/i2c-mux-gpio.c | 4 | ||||
-rw-r--r-- | drivers/i2c/muxes/i2c-mux-pca9541.c | 2 | ||||
-rw-r--r-- | drivers/i2c/muxes/i2c-mux-pca954x.c | 10 | ||||
-rw-r--r-- | drivers/i2c/muxes/i2c-mux-pinctrl.c | 2 | ||||
-rw-r--r-- | include/linux/i2c-mux-gpio.h | 2 | ||||
-rw-r--r-- | include/linux/i2c-mux.h | 1 | ||||
-rw-r--r-- | include/linux/i2c/pca954x.h | 1 |
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 */ | ||
92 | static 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 | |||
91 | struct i2c_adapter *i2c_add_mux_adapter(struct i2c_adapter *parent, | 104 | struct 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 @@ | |||
36 | struct i2c_adapter *i2c_add_mux_adapter(struct i2c_adapter *parent, | 36 | struct 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 @@ | |||
36 | struct pca954x_platform_mode { | 36 | struct 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 */ |