diff options
Diffstat (limited to 'drivers/of')
-rw-r--r-- | drivers/of/Kconfig | 6 | ||||
-rw-r--r-- | drivers/of/Makefile | 1 | ||||
-rw-r--r-- | drivers/of/base.c | 88 | ||||
-rw-r--r-- | drivers/of/of_i2c.c | 64 | ||||
-rw-r--r-- | drivers/of/of_spi.c | 93 |
5 files changed, 191 insertions, 61 deletions
diff --git a/drivers/of/Kconfig b/drivers/of/Kconfig index 1d7ec3129349..f821dbc952a4 100644 --- a/drivers/of/Kconfig +++ b/drivers/of/Kconfig | |||
@@ -13,3 +13,9 @@ config OF_I2C | |||
13 | depends on PPC_OF && I2C | 13 | depends on PPC_OF && I2C |
14 | help | 14 | help |
15 | OpenFirmware I2C accessors | 15 | OpenFirmware I2C accessors |
16 | |||
17 | config OF_SPI | ||
18 | def_tristate SPI | ||
19 | depends on OF && PPC_OF && SPI | ||
20 | help | ||
21 | OpenFirmware SPI accessors | ||
diff --git a/drivers/of/Makefile b/drivers/of/Makefile index 548772e871fd..4c3c6f8e36f5 100644 --- a/drivers/of/Makefile +++ b/drivers/of/Makefile | |||
@@ -2,3 +2,4 @@ obj-y = base.o | |||
2 | obj-$(CONFIG_OF_DEVICE) += device.o platform.o | 2 | obj-$(CONFIG_OF_DEVICE) += device.o platform.o |
3 | obj-$(CONFIG_OF_GPIO) += gpio.o | 3 | obj-$(CONFIG_OF_GPIO) += gpio.o |
4 | obj-$(CONFIG_OF_I2C) += of_i2c.o | 4 | obj-$(CONFIG_OF_I2C) += of_i2c.o |
5 | obj-$(CONFIG_OF_SPI) += of_spi.o | ||
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 | |||
diff --git a/drivers/of/of_i2c.c b/drivers/of/of_i2c.c index 344e1b03dd8b..6a98dc8aa30b 100644 --- a/drivers/of/of_i2c.c +++ b/drivers/of/of_i2c.c | |||
@@ -16,62 +16,6 @@ | |||
16 | #include <linux/of_i2c.h> | 16 | #include <linux/of_i2c.h> |
17 | #include <linux/module.h> | 17 | #include <linux/module.h> |
18 | 18 | ||
19 | struct i2c_driver_device { | ||
20 | char *of_device; | ||
21 | char *i2c_type; | ||
22 | }; | ||
23 | |||
24 | static struct i2c_driver_device i2c_devices[] = { | ||
25 | }; | ||
26 | |||
27 | static int of_find_i2c_driver(struct device_node *node, | ||
28 | struct i2c_board_info *info) | ||
29 | { | ||
30 | int i, cplen; | ||
31 | const char *compatible; | ||
32 | const char *p; | ||
33 | |||
34 | /* 1. search for exception list entry */ | ||
35 | for (i = 0; i < ARRAY_SIZE(i2c_devices); i++) { | ||
36 | if (!of_device_is_compatible(node, i2c_devices[i].of_device)) | ||
37 | continue; | ||
38 | if (strlcpy(info->type, i2c_devices[i].i2c_type, | ||
39 | I2C_NAME_SIZE) >= I2C_NAME_SIZE) | ||
40 | return -ENOMEM; | ||
41 | |||
42 | return 0; | ||
43 | } | ||
44 | |||
45 | compatible = of_get_property(node, "compatible", &cplen); | ||
46 | if (!compatible) | ||
47 | return -ENODEV; | ||
48 | |||
49 | /* 2. search for linux,<i2c-type> entry */ | ||
50 | p = compatible; | ||
51 | while (cplen > 0) { | ||
52 | if (!strncmp(p, "linux,", 6)) { | ||
53 | p += 6; | ||
54 | if (strlcpy(info->type, p, | ||
55 | I2C_NAME_SIZE) >= I2C_NAME_SIZE) | ||
56 | return -ENOMEM; | ||
57 | return 0; | ||
58 | } | ||
59 | |||
60 | i = strlen(p) + 1; | ||
61 | p += i; | ||
62 | cplen -= i; | ||
63 | } | ||
64 | |||
65 | /* 3. take fist compatible entry and strip manufacturer */ | ||
66 | p = strchr(compatible, ','); | ||
67 | if (!p) | ||
68 | return -ENODEV; | ||
69 | p++; | ||
70 | if (strlcpy(info->type, p, I2C_NAME_SIZE) >= I2C_NAME_SIZE) | ||
71 | return -ENOMEM; | ||
72 | return 0; | ||
73 | } | ||
74 | |||
75 | void of_register_i2c_devices(struct i2c_adapter *adap, | 19 | void of_register_i2c_devices(struct i2c_adapter *adap, |
76 | struct device_node *adap_node) | 20 | struct device_node *adap_node) |
77 | { | 21 | { |
@@ -83,6 +27,9 @@ void of_register_i2c_devices(struct i2c_adapter *adap, | |||
83 | const u32 *addr; | 27 | const u32 *addr; |
84 | int len; | 28 | int len; |
85 | 29 | ||
30 | if (of_modalias_node(node, info.type, sizeof(info.type)) < 0) | ||
31 | continue; | ||
32 | |||
86 | addr = of_get_property(node, "reg", &len); | 33 | addr = of_get_property(node, "reg", &len); |
87 | if (!addr || len < sizeof(int) || *addr > (1 << 10) - 1) { | 34 | if (!addr || len < sizeof(int) || *addr > (1 << 10) - 1) { |
88 | printk(KERN_ERR | 35 | printk(KERN_ERR |
@@ -92,11 +39,6 @@ void of_register_i2c_devices(struct i2c_adapter *adap, | |||
92 | 39 | ||
93 | info.irq = irq_of_parse_and_map(node, 0); | 40 | info.irq = irq_of_parse_and_map(node, 0); |
94 | 41 | ||
95 | if (of_find_i2c_driver(node, &info) < 0) { | ||
96 | irq_dispose_mapping(info.irq); | ||
97 | continue; | ||
98 | } | ||
99 | |||
100 | info.addr = *addr; | 42 | info.addr = *addr; |
101 | 43 | ||
102 | request_module(info.type); | 44 | request_module(info.type); |
diff --git a/drivers/of/of_spi.c b/drivers/of/of_spi.c new file mode 100644 index 000000000000..b01eec026f68 --- /dev/null +++ b/drivers/of/of_spi.c | |||
@@ -0,0 +1,93 @@ | |||
1 | /* | ||
2 | * SPI OF support routines | ||
3 | * Copyright (C) 2008 Secret Lab Technologies Ltd. | ||
4 | * | ||
5 | * Support routines for deriving SPI device attachments from the device | ||
6 | * tree. | ||
7 | */ | ||
8 | |||
9 | #include <linux/of.h> | ||
10 | #include <linux/device.h> | ||
11 | #include <linux/spi/spi.h> | ||
12 | #include <linux/of_spi.h> | ||
13 | |||
14 | /** | ||
15 | * of_register_spi_devices - Register child devices onto the SPI bus | ||
16 | * @master: Pointer to spi_master device | ||
17 | * @np: parent node of SPI device nodes | ||
18 | * | ||
19 | * Registers an spi_device for each child node of 'np' which has a 'reg' | ||
20 | * property. | ||
21 | */ | ||
22 | void of_register_spi_devices(struct spi_master *master, struct device_node *np) | ||
23 | { | ||
24 | struct spi_device *spi; | ||
25 | struct device_node *nc; | ||
26 | const u32 *prop; | ||
27 | int rc; | ||
28 | int len; | ||
29 | |||
30 | for_each_child_of_node(np, nc) { | ||
31 | /* Alloc an spi_device */ | ||
32 | spi = spi_alloc_device(master); | ||
33 | if (!spi) { | ||
34 | dev_err(&master->dev, "spi_device alloc error for %s\n", | ||
35 | nc->full_name); | ||
36 | spi_dev_put(spi); | ||
37 | continue; | ||
38 | } | ||
39 | |||
40 | /* Select device driver */ | ||
41 | if (of_modalias_node(nc, spi->modalias, | ||
42 | sizeof(spi->modalias)) < 0) { | ||
43 | dev_err(&master->dev, "cannot find modalias for %s\n", | ||
44 | nc->full_name); | ||
45 | spi_dev_put(spi); | ||
46 | continue; | ||
47 | } | ||
48 | |||
49 | /* Device address */ | ||
50 | prop = of_get_property(nc, "reg", &len); | ||
51 | if (!prop || len < sizeof(*prop)) { | ||
52 | dev_err(&master->dev, "%s has no 'reg' property\n", | ||
53 | nc->full_name); | ||
54 | spi_dev_put(spi); | ||
55 | continue; | ||
56 | } | ||
57 | spi->chip_select = *prop; | ||
58 | |||
59 | /* Mode (clock phase/polarity/etc.) */ | ||
60 | if (of_find_property(nc, "spi-cpha", NULL)) | ||
61 | spi->mode |= SPI_CPHA; | ||
62 | if (of_find_property(nc, "spi-cpol", NULL)) | ||
63 | spi->mode |= SPI_CPOL; | ||
64 | |||
65 | /* Device speed */ | ||
66 | prop = of_get_property(nc, "spi-max-frequency", &len); | ||
67 | if (!prop || len < sizeof(*prop)) { | ||
68 | dev_err(&master->dev, "%s has no 'spi-max-frequency' property\n", | ||
69 | nc->full_name); | ||
70 | spi_dev_put(spi); | ||
71 | continue; | ||
72 | } | ||
73 | spi->max_speed_hz = *prop; | ||
74 | |||
75 | /* IRQ */ | ||
76 | spi->irq = irq_of_parse_and_map(nc, 0); | ||
77 | |||
78 | /* Store a pointer to the node in the device structure */ | ||
79 | of_node_get(nc); | ||
80 | spi->dev.archdata.of_node = nc; | ||
81 | |||
82 | /* Register the new device */ | ||
83 | request_module(spi->modalias); | ||
84 | rc = spi_add_device(spi); | ||
85 | if (rc) { | ||
86 | dev_err(&master->dev, "spi_device register error %s\n", | ||
87 | nc->full_name); | ||
88 | spi_dev_put(spi); | ||
89 | } | ||
90 | |||
91 | } | ||
92 | } | ||
93 | EXPORT_SYMBOL(of_register_spi_devices); | ||