msm: pil: Reorg registration API

Instead of having pil device driver authors register struct
pil_device pointers, have them register a set of ops along with
the important parts of struct pil_device. This allows us to
separate the platform_device from the pil_device so that future
device driver authors can move away from the platform bus. Plus,
this allows us to hide the details of struct pil_device outside
of peripheral-loader.c.

Change-Id: I0945c0bc2447207270efcd3fb935437c449edc63
Signed-off-by: Stephen Boyd <sboyd@codeaurora.org>
diff --git a/arch/arm/mach-msm/peripheral-loader.c b/arch/arm/mach-msm/peripheral-loader.c
index dc3b26f..672f332 100644
--- a/arch/arm/mach-msm/peripheral-loader.c
+++ b/arch/arm/mach-msm/peripheral-loader.c
@@ -12,13 +12,14 @@
 
 #include <linux/module.h>
 #include <linux/string.h>
-#include <linux/platform_device.h>
+#include <linux/device.h>
 #include <linux/firmware.h>
 #include <linux/io.h>
 #include <linux/debugfs.h>
 #include <linux/elf.h>
 #include <linux/mutex.h>
 #include <linux/memblock.h>
+#include <linux/slab.h>
 
 #include <mach/socinfo.h>
 
@@ -27,6 +28,13 @@
 
 #include "peripheral-loader.h"
 
+struct pil_device {
+	struct pil_desc *desc;
+	int count;
+	struct mutex lock;
+	struct list_head list;
+};
+
 static DEFINE_MUTEX(pil_list_lock);
 static LIST_HEAD(pil_list);
 
@@ -35,7 +43,7 @@
 	struct pil_device *dev;
 
 	list_for_each_entry(dev, &pil_list, list)
-		if (!strcmp(dev->name, str))
+		if (!strcmp(dev->desc->name, str))
 			return dev;
 	return NULL;
 }
@@ -65,24 +73,24 @@
 	const u8 *data;
 
 	if (memblock_is_region_memory(phdr->p_paddr, phdr->p_memsz)) {
-		dev_err(&pil->pdev.dev, "Kernel memory would be overwritten");
+		dev_err(pil->desc->dev, "Kernel memory would be overwritten");
 		return -EPERM;
 	}
 
 	if (phdr->p_filesz) {
-		snprintf(fw_name, ARRAY_SIZE(fw_name), "%s.b%02d", pil->name,
-				num);
-		ret = request_firmware(&fw, fw_name, &pil->pdev.dev);
+		snprintf(fw_name, ARRAY_SIZE(fw_name), "%s.b%02d",
+				pil->desc->name, num);
+		ret = request_firmware(&fw, fw_name, pil->desc->dev);
 		if (ret) {
-			dev_err(&pil->pdev.dev, "Failed to locate blob %s\n",
+			dev_err(pil->desc->dev, "Failed to locate blob %s\n",
 					fw_name);
 			return ret;
 		}
 
 		if (fw->size != phdr->p_filesz) {
-			dev_err(&pil->pdev.dev,
-					"Blob size %u doesn't match %u\n",
-					fw->size, phdr->p_filesz);
+			dev_err(pil->desc->dev,
+				"Blob size %u doesn't match %u\n", fw->size,
+				phdr->p_filesz);
 			ret = -EPERM;
 			goto release_fw;
 		}
@@ -99,7 +107,7 @@
 		size = min_t(size_t, IOMAP_SIZE, count);
 		buf = ioremap(paddr, size);
 		if (!buf) {
-			dev_err(&pil->pdev.dev, "Failed to map memory\n");
+			dev_err(pil->desc->dev, "Failed to map memory\n");
 			ret = -ENOMEM;
 			goto release_fw;
 		}
@@ -120,7 +128,7 @@
 		size = min_t(size_t, IOMAP_SIZE, count);
 		buf = ioremap(paddr, size);
 		if (!buf) {
-			dev_err(&pil->pdev.dev, "Failed to map memory\n");
+			dev_err(pil->desc->dev, "Failed to map memory\n");
 			ret = -ENOMEM;
 			goto release_fw;
 		}
@@ -131,9 +139,10 @@
 		paddr += size;
 	}
 
-	ret = pil->ops->verify_blob(pil, phdr->p_paddr, phdr->p_memsz);
+	ret = pil->desc->ops->verify_blob(pil->desc, phdr->p_paddr,
+					  phdr->p_memsz);
 	if (ret)
-		dev_err(&pil->pdev.dev, "Blob %u failed verification\n", num);
+		dev_err(pil->desc->dev, "Blob %u failed verification\n", num);
 
 release_fw:
 	release_firmware(fw);
@@ -155,41 +164,41 @@
 	const struct elf32_phdr *phdr;
 	const struct firmware *fw;
 
-	snprintf(fw_name, sizeof(fw_name), "%s.mdt", pil->name);
-	ret = request_firmware(&fw, fw_name, &pil->pdev.dev);
+	snprintf(fw_name, sizeof(fw_name), "%s.mdt", pil->desc->name);
+	ret = request_firmware(&fw, fw_name, pil->desc->dev);
 	if (ret) {
-		dev_err(&pil->pdev.dev, "Failed to locate %s\n", fw_name);
+		dev_err(pil->desc->dev, "Failed to locate %s\n", fw_name);
 		goto out;
 	}
 
 	if (fw->size < sizeof(*ehdr)) {
-		dev_err(&pil->pdev.dev, "Not big enough to be an elf header\n");
+		dev_err(pil->desc->dev, "Not big enough to be an elf header\n");
 		ret = -EIO;
 		goto release_fw;
 	}
 
 	ehdr = (struct elf32_hdr *)fw->data;
 	if (memcmp(ehdr->e_ident, ELFMAG, SELFMAG)) {
-		dev_err(&pil->pdev.dev, "Not an elf header\n");
+		dev_err(pil->desc->dev, "Not an elf header\n");
 		ret = -EIO;
 		goto release_fw;
 	}
 
 	if (ehdr->e_phnum == 0) {
-		dev_err(&pil->pdev.dev, "No loadable segments\n");
+		dev_err(pil->desc->dev, "No loadable segments\n");
 		ret = -EIO;
 		goto release_fw;
 	}
 	if (sizeof(struct elf32_phdr) * ehdr->e_phnum +
 	    sizeof(struct elf32_hdr) > fw->size) {
-		dev_err(&pil->pdev.dev, "Program headers not within mdt\n");
+		dev_err(pil->desc->dev, "Program headers not within mdt\n");
 		ret = -EIO;
 		goto release_fw;
 	}
 
-	ret = pil->ops->init_image(pil, fw->data, fw->size);
+	ret = pil->desc->ops->init_image(pil->desc, fw->data, fw->size);
 	if (ret) {
-		dev_err(&pil->pdev.dev, "Invalid firmware metadata\n");
+		dev_err(pil->desc->dev, "Invalid firmware metadata\n");
 		goto release_fw;
 	}
 
@@ -200,15 +209,15 @@
 
 		ret = load_segment(phdr, i, pil);
 		if (ret) {
-			dev_err(&pil->pdev.dev, "Failed to load segment %d\n",
+			dev_err(pil->desc->dev, "Failed to load segment %d\n",
 					i);
 			goto release_fw;
 		}
 	}
 
-	ret = pil->ops->auth_and_reset(pil);
+	ret = pil->desc->ops->auth_and_reset(pil->desc);
 	if (ret) {
-		dev_err(&pil->pdev.dev, "Failed to bring out of reset\n");
+		dev_err(pil->desc->dev, "Failed to bring out of reset\n");
 		goto release_fw;
 	}
 
@@ -242,9 +251,9 @@
 	if (!pil)
 		return ERR_PTR(-ENODEV);
 
-	pil_d = find_peripheral(pil->depends_on);
+	pil_d = find_peripheral(pil->desc->depends_on);
 	if (pil_d) {
-		void *p = pil_get(pil_d->name);
+		void *p = pil_get(pil_d->desc->name);
 		if (IS_ERR(p))
 			return p;
 	}
@@ -290,11 +299,11 @@
 	if (pil->count)
 		pil->count--;
 	if (pil->count == 0)
-		pil->ops->shutdown(pil);
+		pil->desc->ops->shutdown(pil->desc);
 unlock:
 	mutex_unlock(&pil->lock);
 
-	pil_d = find_peripheral(pil->depends_on);
+	pil_d = find_peripheral(pil->desc->depends_on);
 	if (pil_d)
 		pil_put(pil_d);
 }
@@ -310,7 +319,7 @@
 
 	mutex_lock(&pil->lock);
 	if (!WARN(!pil->count, "%s: Reference count mismatch\n", __func__))
-		pil->ops->shutdown(pil);
+		pil->desc->ops->shutdown(pil->desc);
 	mutex_unlock(&pil->lock);
 }
 EXPORT_SYMBOL(pil_force_shutdown);
@@ -366,7 +375,7 @@
 		return -EFAULT;
 
 	if (!strncmp(buf, "get", 3)) {
-		if (IS_ERR(pil_get(pil->name)))
+		if (IS_ERR(pil_get(pil->desc->name)))
 			return -EIO;
 	} else if (!strncmp(buf, "put", 3))
 		pil_put(pil);
@@ -401,8 +410,8 @@
 	if (!pil_base_dir)
 		return -ENOMEM;
 
-	if (!debugfs_create_file(pil->name, S_IRUGO | S_IWUSR, pil_base_dir,
-				pil, &msm_pil_debugfs_fops))
+	if (!debugfs_create_file(pil->desc->name, S_IRUGO | S_IWUSR,
+				pil_base_dir, pil, &msm_pil_debugfs_fops))
 		return -ENOMEM;
 	return 0;
 }
@@ -416,29 +425,30 @@
 
 	mutex_lock(&pil_list_lock);
 	list_for_each_entry(pil, &pil_list, list)
-		pil->ops->shutdown(pil);
+		pil->desc->ops->shutdown(pil->desc);
 	mutex_unlock(&pil_list_lock);
 
 	return 0;
 }
 late_initcall(msm_pil_shutdown_at_boot);
 
-int msm_pil_add_device(struct pil_device *pil)
+int msm_pil_register(struct pil_desc *desc)
 {
-	int ret;
-	ret = platform_device_register(&pil->pdev);
-	if (ret)
-		return ret;
+	struct pil_device *pil = kzalloc(sizeof(*pil), GFP_KERNEL);
+	if (!pil)
+		return -ENOMEM;
 
 	mutex_init(&pil->lock);
+	INIT_LIST_HEAD(&pil->list);
+	pil->desc = desc;
 
 	mutex_lock(&pil_list_lock);
 	list_add(&pil->list, &pil_list);
 	mutex_unlock(&pil_list_lock);
 
-	msm_pil_debugfs_add(pil);
-	return 0;
+	return msm_pil_debugfs_add(pil);
 }
+EXPORT_SYMBOL(msm_pil_register);
 
 MODULE_LICENSE("GPL v2");
 MODULE_DESCRIPTION("Load peripheral images and bring peripherals out of reset");