aboutsummaryrefslogtreecommitdiffstats
path: root/drivers/i2c
diff options
context:
space:
mode:
authorLinus Walleij <linus.walleij@linaro.org>2013-05-13 16:18:21 -0400
committerWolfram Sang <wsa@the-dreams.de>2013-06-12 14:39:07 -0400
commitc80f52847c50109ca248c22efbf71ff10553dca4 (patch)
treeeec8391af161d42f47df5e50cb10e1d3f8376d1e /drivers/i2c
parentc2c64954723b9d365f35f36c0cd089e740bb0a0a (diff)
i2c: core: make it possible to match a pure device tree driver
This tries to address an issue found when writing an MFD driver for the Nomadik STw481x PMICs: as the platform is using device tree exclusively I want to specify the driver matching like this: static const struct of_device_id stw481x_match[] = { { .compatible = "st,stw4810", }, { .compatible = "st,stw4811", }, {}, }; static struct i2c_driver stw481x_driver = { .driver = { .name = "stw481x", .of_match_table = stw481x_match, }, .probe = stw481x_probe, .remove = stw481x_remove, }; However that turns out not to be possible: the I2C probe code is written so that the probe() call is always passed a match from i2c_match_id() using non-devicetree matches. This is probably why most devices using device tree for I2C clients currently will pass no .of_match_table *at all* but instead just use .id_table from struct i2c_driver to match the device. As you realize that means that the whole idea with compatible strings is discarded, and that is why we find strange device tree I2C device compatible strings like "product" instead of "vendor,product" as you could expect. Let's figure out how to fix this before the mess spreads. This patch will allow probeing devices with only an of_match_table as per above, and will pass NULL as the second argument to the probe() function. If the driver wants to deduce secondary info from the struct of_device_id .data field, it has to call of_match_device() on its own match table in the probe function device tree probe path. If drivers define both an .of_match_table *AND* a i2c_driver .id_table, the .of_match_table will take precedence, just as is done in the i2c_device_match() function in i2c-core.c. I2C devices probed from device tree should subsequently be fixed to handle the case where of_match_table() is used (I think none of them do that today), and platforms should fix their device trees to use compatible strings for I2C devices instead of setting the name to Linux device driver names as is done in multiple cases today. Cc: Rob Herring <rob.herring@calxeda.com> Cc: Grant Likely <grant.likely@linaro.org> Signed-off-by: Linus Walleij <linus.walleij@linaro.org> Signed-off-by: Wolfram Sang <wsa@the-dreams.de>
Diffstat (limited to 'drivers/i2c')
-rw-r--r--drivers/i2c/i2c-core.c9
1 files changed, 7 insertions, 2 deletions
diff --git a/drivers/i2c/i2c-core.c b/drivers/i2c/i2c-core.c
index 48e31ed69dbf..93fc5bfdb9b4 100644
--- a/drivers/i2c/i2c-core.c
+++ b/drivers/i2c/i2c-core.c
@@ -240,7 +240,7 @@ static int i2c_device_probe(struct device *dev)
240 return 0; 240 return 0;
241 241
242 driver = to_i2c_driver(dev->driver); 242 driver = to_i2c_driver(dev->driver);
243 if (!driver->probe || !driver->id_table) 243 if (!driver->probe || (!driver->id_table && !dev->driver->of_match_table))
244 return -ENODEV; 244 return -ENODEV;
245 client->driver = driver; 245 client->driver = driver;
246 if (!device_can_wakeup(&client->dev)) 246 if (!device_can_wakeup(&client->dev))
@@ -248,7 +248,12 @@ static int i2c_device_probe(struct device *dev)
248 client->flags & I2C_CLIENT_WAKE); 248 client->flags & I2C_CLIENT_WAKE);
249 dev_dbg(dev, "probe\n"); 249 dev_dbg(dev, "probe\n");
250 250
251 status = driver->probe(client, i2c_match_id(driver->id_table, client)); 251 if (of_match_device(dev->driver->of_match_table, dev))
252 /* Device tree matching */
253 status = driver->probe(client, NULL);
254 else
255 /* Fall back to matching the id_table */
256 status = driver->probe(client, i2c_match_id(driver->id_table, client));
252 if (status) { 257 if (status) {
253 client->driver = NULL; 258 client->driver = NULL;
254 i2c_set_clientdata(client, NULL); 259 i2c_set_clientdata(client, NULL);