blob: 6d1a5f03d1764228038d6927af5da9979daf1f6f [file] [log] [blame]
Matt Wagantall39f90cb2012-02-08 14:09:11 -08001/* Copyright (c) 2010-2012, Code Aurora Forum. All rights reserved.
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002 *
3 * This program is free software; you can redistribute it and/or modify
4 * it under the terms of the GNU General Public License version 2 and
5 * only version 2 as published by the Free Software Foundation.
6 *
7 * This program is distributed in the hope that it will be useful,
8 * but WITHOUT ANY WARRANTY; without even the implied warranty of
9 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
10 * GNU General Public License for more details.
11 */
12
13#include <linux/module.h>
14#include <linux/string.h>
Stephen Boyd3f4da322011-08-30 01:03:23 -070015#include <linux/device.h>
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070016#include <linux/firmware.h>
17#include <linux/io.h>
18#include <linux/debugfs.h>
19#include <linux/elf.h>
20#include <linux/mutex.h>
21#include <linux/memblock.h>
Stephen Boyd3f4da322011-08-30 01:03:23 -070022#include <linux/slab.h>
Stephen Boyd6d67d252011-09-27 11:50:05 -070023#include <linux/atomic.h>
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070024
25#include <asm/uaccess.h>
26#include <asm/setup.h>
Stephen Boyd6d67d252011-09-27 11:50:05 -070027#include <mach/peripheral-loader.h>
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070028
29#include "peripheral-loader.h"
30
Stephen Boyd3f4da322011-08-30 01:03:23 -070031struct pil_device {
32 struct pil_desc *desc;
33 int count;
34 struct mutex lock;
Stephen Boyd6d67d252011-09-27 11:50:05 -070035 struct device dev;
36 struct module *owner;
37#ifdef CONFIG_DEBUG_FS
38 struct dentry *dentry;
39#endif
Stephen Boyd3f4da322011-08-30 01:03:23 -070040};
41
Stephen Boyd6d67d252011-09-27 11:50:05 -070042#define to_pil_device(d) container_of(d, struct pil_device, dev)
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070043
Stephen Boyd6d67d252011-09-27 11:50:05 -070044struct bus_type pil_bus_type = {
45 .name = "pil",
46};
47
48static int __find_peripheral(struct device *dev, void *data)
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070049{
Stephen Boyd6d67d252011-09-27 11:50:05 -070050 struct pil_device *pdev = to_pil_device(dev);
51 return !strncmp(pdev->desc->name, data, INT_MAX);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070052}
53
54static struct pil_device *find_peripheral(const char *str)
55{
Stephen Boyd6d67d252011-09-27 11:50:05 -070056 struct device *dev;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070057
58 if (!str)
59 return NULL;
60
Stephen Boyd6d67d252011-09-27 11:50:05 -070061 dev = bus_find_device(&pil_bus_type, NULL, (void *)str,
62 __find_peripheral);
63 return dev ? to_pil_device(dev) : NULL;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070064}
65
66#define IOMAP_SIZE SZ_4M
67
68static int load_segment(const struct elf32_phdr *phdr, unsigned num,
69 struct pil_device *pil)
70{
71 int ret, count, paddr;
72 char fw_name[30];
73 const struct firmware *fw = NULL;
74 const u8 *data;
75
76 if (memblock_is_region_memory(phdr->p_paddr, phdr->p_memsz)) {
Stephen Boyd6d67d252011-09-27 11:50:05 -070077 dev_err(&pil->dev, "Kernel memory would be overwritten");
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070078 return -EPERM;
79 }
80
81 if (phdr->p_filesz) {
Stephen Boyd3f4da322011-08-30 01:03:23 -070082 snprintf(fw_name, ARRAY_SIZE(fw_name), "%s.b%02d",
83 pil->desc->name, num);
Stephen Boyd6d67d252011-09-27 11:50:05 -070084 ret = request_firmware(&fw, fw_name, &pil->dev);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070085 if (ret) {
Stephen Boyd6d67d252011-09-27 11:50:05 -070086 dev_err(&pil->dev, "Failed to locate blob %s\n",
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070087 fw_name);
88 return ret;
89 }
90
91 if (fw->size != phdr->p_filesz) {
Stephen Boyd6d67d252011-09-27 11:50:05 -070092 dev_err(&pil->dev, "Blob size %u doesn't match %u\n",
93 fw->size, phdr->p_filesz);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070094 ret = -EPERM;
95 goto release_fw;
96 }
97 }
98
99 /* Load the segment into memory */
100 count = phdr->p_filesz;
101 paddr = phdr->p_paddr;
102 data = fw ? fw->data : NULL;
103 while (count > 0) {
104 int size;
105 u8 __iomem *buf;
106
107 size = min_t(size_t, IOMAP_SIZE, count);
108 buf = ioremap(paddr, size);
109 if (!buf) {
Stephen Boyd6d67d252011-09-27 11:50:05 -0700110 dev_err(&pil->dev, "Failed to map memory\n");
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700111 ret = -ENOMEM;
112 goto release_fw;
113 }
114 memcpy(buf, data, size);
115 iounmap(buf);
116
117 count -= size;
118 paddr += size;
119 data += size;
120 }
121
122 /* Zero out trailing memory */
123 count = phdr->p_memsz - phdr->p_filesz;
124 while (count > 0) {
125 int size;
126 u8 __iomem *buf;
127
128 size = min_t(size_t, IOMAP_SIZE, count);
129 buf = ioremap(paddr, size);
130 if (!buf) {
Stephen Boyd6d67d252011-09-27 11:50:05 -0700131 dev_err(&pil->dev, "Failed to map memory\n");
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700132 ret = -ENOMEM;
133 goto release_fw;
134 }
135 memset(buf, 0, size);
136 iounmap(buf);
137
138 count -= size;
139 paddr += size;
140 }
141
Stephen Boyd3f4da322011-08-30 01:03:23 -0700142 ret = pil->desc->ops->verify_blob(pil->desc, phdr->p_paddr,
143 phdr->p_memsz);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700144 if (ret)
Stephen Boyd6d67d252011-09-27 11:50:05 -0700145 dev_err(&pil->dev, "Blob %u failed verification\n", num);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700146
147release_fw:
148 release_firmware(fw);
149 return ret;
150}
151
152#define segment_is_hash(flag) (((flag) & (0x7 << 24)) == (0x2 << 24))
153
154static int segment_is_loadable(const struct elf32_phdr *p)
155{
156 return (p->p_type & PT_LOAD) && !segment_is_hash(p->p_flags);
157}
158
159static int load_image(struct pil_device *pil)
160{
161 int i, ret;
162 char fw_name[30];
163 struct elf32_hdr *ehdr;
164 const struct elf32_phdr *phdr;
165 const struct firmware *fw;
166
Stephen Boyd3f4da322011-08-30 01:03:23 -0700167 snprintf(fw_name, sizeof(fw_name), "%s.mdt", pil->desc->name);
Stephen Boyd6d67d252011-09-27 11:50:05 -0700168 ret = request_firmware(&fw, fw_name, &pil->dev);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700169 if (ret) {
Stephen Boyd6d67d252011-09-27 11:50:05 -0700170 dev_err(&pil->dev, "Failed to locate %s\n", fw_name);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700171 goto out;
172 }
173
174 if (fw->size < sizeof(*ehdr)) {
Stephen Boyd6d67d252011-09-27 11:50:05 -0700175 dev_err(&pil->dev, "Not big enough to be an elf header\n");
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700176 ret = -EIO;
177 goto release_fw;
178 }
179
180 ehdr = (struct elf32_hdr *)fw->data;
181 if (memcmp(ehdr->e_ident, ELFMAG, SELFMAG)) {
Stephen Boyd6d67d252011-09-27 11:50:05 -0700182 dev_err(&pil->dev, "Not an elf header\n");
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700183 ret = -EIO;
184 goto release_fw;
185 }
186
187 if (ehdr->e_phnum == 0) {
Stephen Boyd6d67d252011-09-27 11:50:05 -0700188 dev_err(&pil->dev, "No loadable segments\n");
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700189 ret = -EIO;
190 goto release_fw;
191 }
Stephen Boyd96a9f902011-07-18 18:43:00 -0700192 if (sizeof(struct elf32_phdr) * ehdr->e_phnum +
193 sizeof(struct elf32_hdr) > fw->size) {
Stephen Boyd6d67d252011-09-27 11:50:05 -0700194 dev_err(&pil->dev, "Program headers not within mdt\n");
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700195 ret = -EIO;
196 goto release_fw;
197 }
198
Stephen Boyd3f4da322011-08-30 01:03:23 -0700199 ret = pil->desc->ops->init_image(pil->desc, fw->data, fw->size);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700200 if (ret) {
Stephen Boyd6d67d252011-09-27 11:50:05 -0700201 dev_err(&pil->dev, "Invalid firmware metadata\n");
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700202 goto release_fw;
203 }
204
Stephen Boydc9753e12011-07-13 17:58:48 -0700205 phdr = (const struct elf32_phdr *)(fw->data + sizeof(struct elf32_hdr));
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700206 for (i = 0; i < ehdr->e_phnum; i++, phdr++) {
207 if (!segment_is_loadable(phdr))
208 continue;
209
210 ret = load_segment(phdr, i, pil);
211 if (ret) {
Stephen Boyd6d67d252011-09-27 11:50:05 -0700212 dev_err(&pil->dev, "Failed to load segment %d\n",
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700213 i);
214 goto release_fw;
215 }
216 }
217
Stephen Boyd3f4da322011-08-30 01:03:23 -0700218 ret = pil->desc->ops->auth_and_reset(pil->desc);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700219 if (ret) {
Stephen Boyd6d67d252011-09-27 11:50:05 -0700220 dev_err(&pil->dev, "Failed to bring out of reset\n");
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700221 goto release_fw;
222 }
Stephen Boyd6d67d252011-09-27 11:50:05 -0700223 dev_info(&pil->dev, "brought %s out of reset\n", pil->desc->name);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700224
225release_fw:
226 release_firmware(fw);
227out:
228 return ret;
229}
230
231/**
232 * pil_get() - Load a peripheral into memory and take it out of reset
233 * @name: pointer to a string containing the name of the peripheral to load
234 *
235 * This function returns a pointer if it succeeds. If an error occurs an
236 * ERR_PTR is returned.
237 *
238 * If PIL is not enabled in the kernel, the value %NULL will be returned.
239 */
240void *pil_get(const char *name)
241{
242 int ret;
243 struct pil_device *pil;
244 struct pil_device *pil_d;
245 void *retval;
246
Stephen Boyd6d67d252011-09-27 11:50:05 -0700247 if (!name)
248 return NULL;
249
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700250 pil = retval = find_peripheral(name);
251 if (!pil)
252 return ERR_PTR(-ENODEV);
Stephen Boyd6d67d252011-09-27 11:50:05 -0700253 if (!try_module_get(pil->owner)) {
254 put_device(&pil->dev);
255 return ERR_PTR(-ENODEV);
256 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700257
Stephen Boyd6d67d252011-09-27 11:50:05 -0700258 pil_d = pil_get(pil->desc->depends_on);
259 if (IS_ERR(pil_d)) {
260 retval = pil_d;
261 goto err_depends;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700262 }
263
264 mutex_lock(&pil->lock);
Stephen Boyd6d67d252011-09-27 11:50:05 -0700265 if (!pil->count++) {
266 ret = load_image(pil);
267 if (ret) {
268 retval = ERR_PTR(ret);
269 goto err_load;
270 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700271 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700272 mutex_unlock(&pil->lock);
Stephen Boyd6d67d252011-09-27 11:50:05 -0700273out:
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700274 return retval;
Stephen Boyd6d67d252011-09-27 11:50:05 -0700275err_load:
276 mutex_unlock(&pil->lock);
277 pil_put(pil_d);
278err_depends:
279 put_device(&pil->dev);
280 module_put(pil->owner);
281 goto out;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700282}
283EXPORT_SYMBOL(pil_get);
284
285/**
286 * pil_put() - Inform PIL the peripheral no longer needs to be active
287 * @peripheral_handle: pointer from a previous call to pil_get()
288 *
289 * This doesn't imply that a peripheral is shutdown or in reset since another
290 * driver could be using the peripheral.
291 */
292void pil_put(void *peripheral_handle)
293{
Stephen Boyd6d67d252011-09-27 11:50:05 -0700294 struct pil_device *pil_d, *pil = peripheral_handle;
295
296 if (IS_ERR_OR_NULL(pil))
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700297 return;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700298
299 mutex_lock(&pil->lock);
Stephen Boyd6d67d252011-09-27 11:50:05 -0700300 if (WARN(!pil->count, "%s: Reference count mismatch\n", __func__))
301 goto err_out;
302 if (!--pil->count)
Stephen Boyd3f4da322011-08-30 01:03:23 -0700303 pil->desc->ops->shutdown(pil->desc);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700304 mutex_unlock(&pil->lock);
305
Stephen Boyd3f4da322011-08-30 01:03:23 -0700306 pil_d = find_peripheral(pil->desc->depends_on);
Stephen Boyd6d67d252011-09-27 11:50:05 -0700307 module_put(pil->owner);
308 if (pil_d) {
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700309 pil_put(pil_d);
Stephen Boyd6d67d252011-09-27 11:50:05 -0700310 put_device(&pil_d->dev);
311 }
312 put_device(&pil->dev);
313 return;
314err_out:
315 mutex_unlock(&pil->lock);
316 return;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700317}
318EXPORT_SYMBOL(pil_put);
319
320void pil_force_shutdown(const char *name)
321{
322 struct pil_device *pil;
323
324 pil = find_peripheral(name);
325 if (!pil)
326 return;
327
328 mutex_lock(&pil->lock);
329 if (!WARN(!pil->count, "%s: Reference count mismatch\n", __func__))
Stephen Boyd3f4da322011-08-30 01:03:23 -0700330 pil->desc->ops->shutdown(pil->desc);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700331 mutex_unlock(&pil->lock);
Stephen Boyd6d67d252011-09-27 11:50:05 -0700332 put_device(&pil->dev);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700333}
334EXPORT_SYMBOL(pil_force_shutdown);
335
336int pil_force_boot(const char *name)
337{
338 int ret = -EINVAL;
339 struct pil_device *pil;
340
341 pil = find_peripheral(name);
342 if (!pil)
343 return -EINVAL;
344
345 mutex_lock(&pil->lock);
346 if (!WARN(!pil->count, "%s: Reference count mismatch\n", __func__))
347 ret = load_image(pil);
348 mutex_unlock(&pil->lock);
Stephen Boyd6d67d252011-09-27 11:50:05 -0700349 put_device(&pil->dev);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700350
351 return ret;
352}
353EXPORT_SYMBOL(pil_force_boot);
354
355#ifdef CONFIG_DEBUG_FS
Stephen Boyd6d67d252011-09-27 11:50:05 -0700356static int msm_pil_debugfs_open(struct inode *inode, struct file *filp)
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700357{
358 filp->private_data = inode->i_private;
359 return 0;
360}
361
362static ssize_t msm_pil_debugfs_read(struct file *filp, char __user *ubuf,
363 size_t cnt, loff_t *ppos)
364{
365 int r;
366 char buf[40];
367 struct pil_device *pil = filp->private_data;
368
369 mutex_lock(&pil->lock);
370 r = snprintf(buf, sizeof(buf), "%d\n", pil->count);
371 mutex_unlock(&pil->lock);
372 return simple_read_from_buffer(ubuf, cnt, ppos, buf, r);
373}
374
375static ssize_t msm_pil_debugfs_write(struct file *filp,
376 const char __user *ubuf, size_t cnt, loff_t *ppos)
377{
378 struct pil_device *pil = filp->private_data;
379 char buf[4];
380
381 if (cnt > sizeof(buf))
382 return -EINVAL;
383
384 if (copy_from_user(&buf, ubuf, cnt))
385 return -EFAULT;
386
387 if (!strncmp(buf, "get", 3)) {
Stephen Boyd3f4da322011-08-30 01:03:23 -0700388 if (IS_ERR(pil_get(pil->desc->name)))
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700389 return -EIO;
390 } else if (!strncmp(buf, "put", 3))
391 pil_put(pil);
392 else
393 return -EINVAL;
394
395 return cnt;
396}
397
398static const struct file_operations msm_pil_debugfs_fops = {
399 .open = msm_pil_debugfs_open,
400 .read = msm_pil_debugfs_read,
401 .write = msm_pil_debugfs_write,
402};
403
404static struct dentry *pil_base_dir;
405
Stephen Boyd6d67d252011-09-27 11:50:05 -0700406static int __init msm_pil_debugfs_init(void)
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700407{
408 pil_base_dir = debugfs_create_dir("pil", NULL);
409 if (!pil_base_dir) {
410 pil_base_dir = NULL;
411 return -ENOMEM;
412 }
413
414 return 0;
415}
Stephen Boyd6d67d252011-09-27 11:50:05 -0700416
417static void __exit msm_pil_debugfs_exit(void)
418{
419 debugfs_remove_recursive(pil_base_dir);
420}
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700421
422static int msm_pil_debugfs_add(struct pil_device *pil)
423{
424 if (!pil_base_dir)
425 return -ENOMEM;
426
Stephen Boyd6d67d252011-09-27 11:50:05 -0700427 pil->dentry = debugfs_create_file(pil->desc->name, S_IRUGO | S_IWUSR,
428 pil_base_dir, pil, &msm_pil_debugfs_fops);
429 return !pil->dentry ? -ENOMEM : 0;
430}
431
432static void msm_pil_debugfs_remove(struct pil_device *pil)
433{
434 debugfs_remove(pil->dentry);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700435}
436#else
Stephen Boyd6d67d252011-09-27 11:50:05 -0700437static int __init msm_pil_debugfs_init(void) { return 0; };
438static void __exit msm_pil_debugfs_exit(void) { return 0; };
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700439static int msm_pil_debugfs_add(struct pil_device *pil) { return 0; }
Stephen Boyd6d67d252011-09-27 11:50:05 -0700440static void msm_pil_debugfs_remove(struct pil_device *pil) { }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700441#endif
442
Stephen Boyd6d67d252011-09-27 11:50:05 -0700443static int __msm_pil_shutdown(struct device *dev, void *data)
444{
445 struct pil_device *pil = to_pil_device(dev);
446 pil->desc->ops->shutdown(pil->desc);
447 return 0;
448}
449
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700450static int msm_pil_shutdown_at_boot(void)
451{
Stephen Boyd6d67d252011-09-27 11:50:05 -0700452 return bus_for_each_dev(&pil_bus_type, NULL, NULL, __msm_pil_shutdown);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700453}
454late_initcall(msm_pil_shutdown_at_boot);
455
Stephen Boyd6d67d252011-09-27 11:50:05 -0700456static void pil_device_release(struct device *dev)
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700457{
Stephen Boyd6d67d252011-09-27 11:50:05 -0700458 struct pil_device *pil = to_pil_device(dev);
459 mutex_destroy(&pil->lock);
460 kfree(pil);
461}
462
463struct pil_device *msm_pil_register(struct pil_desc *desc)
464{
465 int err;
466 static atomic_t pil_count = ATOMIC_INIT(-1);
Stephen Boyd3f4da322011-08-30 01:03:23 -0700467 struct pil_device *pil = kzalloc(sizeof(*pil), GFP_KERNEL);
Stephen Boyd6d67d252011-09-27 11:50:05 -0700468
Stephen Boyd3f4da322011-08-30 01:03:23 -0700469 if (!pil)
Stephen Boyd6d67d252011-09-27 11:50:05 -0700470 return ERR_PTR(-ENOMEM);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700471
472 mutex_init(&pil->lock);
Stephen Boyd3f4da322011-08-30 01:03:23 -0700473 pil->desc = desc;
Stephen Boyd6d67d252011-09-27 11:50:05 -0700474 pil->owner = desc->owner;
475 pil->dev.parent = desc->dev;
476 pil->dev.bus = &pil_bus_type;
477 pil->dev.release = pil_device_release;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700478
Stephen Boyd6d67d252011-09-27 11:50:05 -0700479 dev_set_name(&pil->dev, "pil%d", atomic_inc_return(&pil_count));
480 err = device_register(&pil->dev);
481 if (err) {
482 put_device(&pil->dev);
483 mutex_destroy(&pil->lock);
484 kfree(pil);
485 return ERR_PTR(err);
486 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700487
Stephen Boyd6d67d252011-09-27 11:50:05 -0700488 err = msm_pil_debugfs_add(pil);
489 if (err) {
490 device_unregister(&pil->dev);
491 return ERR_PTR(err);
492 }
493
494 return pil;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700495}
Stephen Boyd3f4da322011-08-30 01:03:23 -0700496EXPORT_SYMBOL(msm_pil_register);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700497
Stephen Boyd6d67d252011-09-27 11:50:05 -0700498void msm_pil_unregister(struct pil_device *pil)
499{
500 if (IS_ERR_OR_NULL(pil))
501 return;
502
503 if (get_device(&pil->dev)) {
504 mutex_lock(&pil->lock);
505 WARN_ON(pil->count);
506 msm_pil_debugfs_remove(pil);
507 device_unregister(&pil->dev);
508 mutex_unlock(&pil->lock);
509 put_device(&pil->dev);
510 }
511}
512EXPORT_SYMBOL(msm_pil_unregister);
513
514static int __init msm_pil_init(void)
515{
516 int ret = msm_pil_debugfs_init();
517 if (ret)
518 return ret;
519 return bus_register(&pil_bus_type);
520}
521subsys_initcall(msm_pil_init);
522
523static void __exit msm_pil_exit(void)
524{
525 bus_unregister(&pil_bus_type);
526 msm_pil_debugfs_exit();
527}
528module_exit(msm_pil_exit);
529
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700530MODULE_LICENSE("GPL v2");
531MODULE_DESCRIPTION("Load peripheral images and bring peripherals out of reset");