diff options
author | Mark Brown <broonie@opensource.wolfsonmicro.com> | 2009-07-27 09:45:56 -0400 |
---|---|---|
committer | Samuel Ortiz <sameo@linux.intel.com> | 2009-09-17 03:46:59 -0400 |
commit | 6704e5171ba9053ba173bcd807c7392d2076bdb4 (patch) | |
tree | 84f91270bdf1bfbefe2e059dec395d473cbd72d0 /drivers/mfd | |
parent | 63aed85e3535b4603798184cc941e49de386d354 (diff) |
mfd: Add basic WM831x OTP support
The WM831x series of devices use OTP (One Time Programmable, a type
of PROM) to store system configuration. At run time this data is
visible via registers.
Currently the only explicitly supported feature is that the unique
ID provided by every WM831x device is exported to user space via
sysfs. Other configuration data may be read by system-specific
code in the pre_init() and post_init() platform data operations.
Signed-off-by: Mark Brown <broonie@opensource.wolfsonmicro.com>
Signed-off-by: Samuel Ortiz <sameo@linux.intel.com>
Diffstat (limited to 'drivers/mfd')
-rw-r--r-- | drivers/mfd/wm831x-core.c | 4 | ||||
-rw-r--r-- | drivers/mfd/wm831x-otp.c | 83 |
2 files changed, 87 insertions, 0 deletions
diff --git a/drivers/mfd/wm831x-core.c b/drivers/mfd/wm831x-core.c index bc40ea315cc0..33eaea2f988e 100644 --- a/drivers/mfd/wm831x-core.c +++ b/drivers/mfd/wm831x-core.c | |||
@@ -23,6 +23,7 @@ | |||
23 | #include <linux/mfd/wm831x/pdata.h> | 23 | #include <linux/mfd/wm831x/pdata.h> |
24 | #include <linux/mfd/wm831x/irq.h> | 24 | #include <linux/mfd/wm831x/irq.h> |
25 | #include <linux/mfd/wm831x/auxadc.h> | 25 | #include <linux/mfd/wm831x/auxadc.h> |
26 | #include <linux/mfd/wm831x/otp.h> | ||
26 | 27 | ||
27 | enum wm831x_parent { | 28 | enum wm831x_parent { |
28 | WM8310 = 0, | 29 | WM8310 = 0, |
@@ -1340,6 +1341,8 @@ static int wm831x_device_init(struct wm831x *wm831x, unsigned long id, int irq) | |||
1340 | ret); | 1341 | ret); |
1341 | } | 1342 | } |
1342 | 1343 | ||
1344 | wm831x_otp_init(wm831x); | ||
1345 | |||
1343 | if (pdata && pdata->post_init) { | 1346 | if (pdata && pdata->post_init) { |
1344 | ret = pdata->post_init(wm831x); | 1347 | ret = pdata->post_init(wm831x); |
1345 | if (ret != 0) { | 1348 | if (ret != 0) { |
@@ -1360,6 +1363,7 @@ err: | |||
1360 | 1363 | ||
1361 | static void wm831x_device_exit(struct wm831x *wm831x) | 1364 | static void wm831x_device_exit(struct wm831x *wm831x) |
1362 | { | 1365 | { |
1366 | wm831x_otp_exit(wm831x); | ||
1363 | mfd_remove_devices(wm831x->dev); | 1367 | mfd_remove_devices(wm831x->dev); |
1364 | wm831x_irq_exit(wm831x); | 1368 | wm831x_irq_exit(wm831x); |
1365 | kfree(wm831x); | 1369 | kfree(wm831x); |
diff --git a/drivers/mfd/wm831x-otp.c b/drivers/mfd/wm831x-otp.c new file mode 100644 index 000000000000..f742745ff354 --- /dev/null +++ b/drivers/mfd/wm831x-otp.c | |||
@@ -0,0 +1,83 @@ | |||
1 | /* | ||
2 | * wm831x-otp.c -- OTP for Wolfson WM831x PMICs | ||
3 | * | ||
4 | * Copyright 2009 Wolfson Microelectronics PLC. | ||
5 | * | ||
6 | * Author: Mark Brown <broonie@opensource.wolfsonmicro.com> | ||
7 | * | ||
8 | * This program is free software; you can redistribute it and/or modify it | ||
9 | * under the terms of the GNU General Public License as published by the | ||
10 | * Free Software Foundation; either version 2 of the License, or (at your | ||
11 | * option) any later version. | ||
12 | * | ||
13 | */ | ||
14 | |||
15 | #include <linux/kernel.h> | ||
16 | #include <linux/module.h> | ||
17 | #include <linux/i2c.h> | ||
18 | #include <linux/bcd.h> | ||
19 | #include <linux/delay.h> | ||
20 | #include <linux/mfd/core.h> | ||
21 | |||
22 | #include <linux/mfd/wm831x/core.h> | ||
23 | #include <linux/mfd/wm831x/otp.h> | ||
24 | |||
25 | /* In bytes */ | ||
26 | #define WM831X_UNIQUE_ID_LEN 16 | ||
27 | |||
28 | /* Read the unique ID from the chip into id */ | ||
29 | static int wm831x_unique_id_read(struct wm831x *wm831x, char *id) | ||
30 | { | ||
31 | int i, val; | ||
32 | |||
33 | for (i = 0; i < WM831X_UNIQUE_ID_LEN / 2; i++) { | ||
34 | val = wm831x_reg_read(wm831x, WM831X_UNIQUE_ID_1 + i); | ||
35 | if (val < 0) | ||
36 | return val; | ||
37 | |||
38 | id[i * 2] = (val >> 8) & 0xff; | ||
39 | id[(i * 2) + 1] = val & 0xff; | ||
40 | } | ||
41 | |||
42 | return 0; | ||
43 | } | ||
44 | |||
45 | static ssize_t wm831x_unique_id_show(struct device *dev, | ||
46 | struct device_attribute *attr, char *buf) | ||
47 | { | ||
48 | struct wm831x *wm831x = dev_get_drvdata(dev); | ||
49 | int i, rval; | ||
50 | char id[WM831X_UNIQUE_ID_LEN]; | ||
51 | ssize_t ret = 0; | ||
52 | |||
53 | rval = wm831x_unique_id_read(wm831x, id); | ||
54 | if (rval < 0) | ||
55 | return 0; | ||
56 | |||
57 | for (i = 0; i < WM831X_UNIQUE_ID_LEN; i++) | ||
58 | ret += sprintf(&buf[ret], "%02x", buf[i]); | ||
59 | |||
60 | ret += sprintf(&buf[ret], "\n"); | ||
61 | |||
62 | return ret; | ||
63 | } | ||
64 | |||
65 | static DEVICE_ATTR(unique_id, 0444, wm831x_unique_id_show, NULL); | ||
66 | |||
67 | int wm831x_otp_init(struct wm831x *wm831x) | ||
68 | { | ||
69 | int ret; | ||
70 | |||
71 | ret = device_create_file(wm831x->dev, &dev_attr_unique_id); | ||
72 | if (ret != 0) | ||
73 | dev_err(wm831x->dev, "Unique ID attribute not created: %d\n", | ||
74 | ret); | ||
75 | |||
76 | return ret; | ||
77 | } | ||
78 | |||
79 | void wm831x_otp_exit(struct wm831x *wm831x) | ||
80 | { | ||
81 | device_remove_file(wm831x->dev, &dev_attr_unique_id); | ||
82 | } | ||
83 | |||