diff options
author | Grant Likely <grant.likely@secretlab.ca> | 2008-07-25 22:25:13 -0400 |
---|---|---|
committer | Grant Likely <grant.likely@secretlab.ca> | 2008-07-25 22:25:13 -0400 |
commit | 3f07af494dfa6de43137dae430431c9fbf929c0c (patch) | |
tree | ee204d7cb204fbe287f2626fdeb8926adc7d4889 /drivers/of/base.c | |
parent | 5047887caf1806f31652210df27fb62a7c43f27d (diff) |
of: adapt of_find_i2c_driver() to be usable by SPI also
SPI has a similar problem as I2C in that it needs to determine an
appropriate modalias value for each device node. This patch adapts
the of_i2c of_find_i2c_driver() function to be usable by of_spi also.
Signed-off-by: Grant Likely <grant.likely@secretlab.ca>
Diffstat (limited to 'drivers/of/base.c')
-rw-r--r-- | drivers/of/base.c | 88 |
1 files changed, 88 insertions, 0 deletions
diff --git a/drivers/of/base.c b/drivers/of/base.c index 23ffb7c0caf2..ad8ac1a8af28 100644 --- a/drivers/of/base.c +++ b/drivers/of/base.c | |||
@@ -385,3 +385,91 @@ struct device_node *of_find_matching_node(struct device_node *from, | |||
385 | return np; | 385 | return np; |
386 | } | 386 | } |
387 | EXPORT_SYMBOL(of_find_matching_node); | 387 | EXPORT_SYMBOL(of_find_matching_node); |
388 | |||
389 | /** | ||
390 | * of_modalias_table: Table of explicit compatible ==> modalias mappings | ||
391 | * | ||
392 | * This table allows particulare compatible property values to be mapped | ||
393 | * to modalias strings. This is useful for busses which do not directly | ||
394 | * understand the OF device tree but are populated based on data contained | ||
395 | * within the device tree. SPI and I2C are the two current users of this | ||
396 | * table. | ||
397 | * | ||
398 | * In most cases, devices do not need to be listed in this table because | ||
399 | * the modalias value can be derived directly from the compatible table. | ||
400 | * However, if for any reason a value cannot be derived, then this table | ||
401 | * provides a method to override the implicit derivation. | ||
402 | * | ||
403 | * At the moment, a single table is used for all bus types because it is | ||
404 | * assumed that the data size is small and that the compatible values | ||
405 | * should already be distinct enough to differentiate between SPI, I2C | ||
406 | * and other devices. | ||
407 | */ | ||
408 | struct of_modalias_table { | ||
409 | char *of_device; | ||
410 | char *modalias; | ||
411 | }; | ||
412 | static struct of_modalias_table of_modalias_table[] = { | ||
413 | /* Empty for now; add entries as needed */ | ||
414 | }; | ||
415 | |||
416 | /** | ||
417 | * of_modalias_node - Lookup appropriate modalias for a device node | ||
418 | * @node: pointer to a device tree node | ||
419 | * @modalias: Pointer to buffer that modalias value will be copied into | ||
420 | * @len: Length of modalias value | ||
421 | * | ||
422 | * Based on the value of the compatible property, this routine will determine | ||
423 | * an appropriate modalias value for a particular device tree node. Three | ||
424 | * separate methods are used to derive a modalias value. | ||
425 | * | ||
426 | * First method is to lookup the compatible value in of_modalias_table. | ||
427 | * Second is to look for a "linux,<modalias>" entry in the compatible list | ||
428 | * and used that for modalias. Third is to strip off the manufacturer | ||
429 | * prefix from the first compatible entry and use the remainder as modalias | ||
430 | * | ||
431 | * This routine returns 0 on success | ||
432 | */ | ||
433 | int of_modalias_node(struct device_node *node, char *modalias, int len) | ||
434 | { | ||
435 | int i, cplen; | ||
436 | const char *compatible; | ||
437 | const char *p; | ||
438 | |||
439 | /* 1. search for exception list entry */ | ||
440 | for (i = 0; i < ARRAY_SIZE(of_modalias_table); i++) { | ||
441 | compatible = of_modalias_table[i].of_device; | ||
442 | if (!of_device_is_compatible(node, compatible)) | ||
443 | continue; | ||
444 | strlcpy(modalias, of_modalias_table[i].modalias, len); | ||
445 | return 0; | ||
446 | } | ||
447 | |||
448 | compatible = of_get_property(node, "compatible", &cplen); | ||
449 | if (!compatible) | ||
450 | return -ENODEV; | ||
451 | |||
452 | /* 2. search for linux,<modalias> entry */ | ||
453 | p = compatible; | ||
454 | while (cplen > 0) { | ||
455 | if (!strncmp(p, "linux,", 6)) { | ||
456 | p += 6; | ||
457 | strlcpy(modalias, p, len); | ||
458 | return 0; | ||
459 | } | ||
460 | |||
461 | i = strlen(p) + 1; | ||
462 | p += i; | ||
463 | cplen -= i; | ||
464 | } | ||
465 | |||
466 | /* 3. take first compatible entry and strip manufacturer */ | ||
467 | p = strchr(compatible, ','); | ||
468 | if (!p) | ||
469 | return -ENODEV; | ||
470 | p++; | ||
471 | strlcpy(modalias, p, len); | ||
472 | return 0; | ||
473 | } | ||
474 | EXPORT_SYMBOL_GPL(of_modalias_node); | ||
475 | |||