1 /*
2 * ACPI support for platform bus type.
3 *
4 * Copyright (C) 2012, Intel Corporation
5 * Authors: Mika Westerberg <mika.westerberg@linux.intel.com>
6 * Mathias Nyman <mathias.nyman@linux.intel.com>
7 * Rafael J. Wysocki <rafael.j.wysocki@intel.com>
8 *
9 * This program is free software; you can redistribute it and/or modify
10 * it under the terms of the GNU General Public License version 2 as
11 * published by the Free Software Foundation.
12 */
13
14 #include <linux/acpi.h>
15 #include <linux/device.h>
16 #include <linux/err.h>
17 #include <linux/kernel.h>
18 #include <linux/module.h>
19 #include <linux/dma-mapping.h>
20 #include <linux/platform_device.h>
21
22 #include "internal.h"
23
24 ACPI_MODULE_NAME("platform");
25
26 /*
27 * The following ACPI IDs are known to be suitable for representing as
28 * platform devices.
29 */
30 static const struct acpi_device_id acpi_platform_device_ids[] = {
31
32 { "PNP0D40" },
33
34 /* Android-specific virtual hardware (goldfish devices)
35 *
36 * Note 1: The sole purpose of adding the following ACPI IDs is to
37 * ensure that these goldfish devices are enumerated to the kernel as
38 * platform devices.
39 *
40 * Note 2: Once the Android emulator has adopted Linux kernel 3.16 or
41 * newer (it still uses 3.10 at the time of writing), these changes will
42 * become obsolete (in fact, this acpi_platform_device_ids array will
43 * disappear), since the new kernel will recognize devices enumerated
44 * via ACPI as platform devices by default.
45 *
46 * Note 3: See external/qemu-android/hw/i386/acpi-dsdt-goldfish.dsl in
47 * the Android source tree for the original definitions of these ACPI
48 * IDs in ASL.
49 */
50 { "GFSH0001" }, /* goldfish battery */
51 { "GFSH0002" }, /* goldfish events */
52 { "GFSH0003" }, /* android pipe */
53 { "GFSH0004" }, /* goldfish framebuffer */
54 { "GFSH0005" }, /* goldfish audio */
55 { "GFSH0006" }, /* goldfish sync */
56 { "GFSH0007" }, /* goldfish rtc */
57 { "GFSH0008" }, /* goldfish rotary */
58
59 { }
60 };
61
62 /**
63 * acpi_create_platform_device - Create platform device for ACPI device node
64 * @adev: ACPI device node to create a platform device for.
65 * @id: ACPI device ID used to match @adev.
66 *
67 * Check if the given @adev can be represented as a platform device and, if
68 * that's the case, create and register a platform device, populate its common
69 * resources and returns a pointer to it. Otherwise, return %NULL.
70 *
71 * Name of the platform device will be the same as @adev's.
72 */
acpi_create_platform_device(struct acpi_device * adev,const struct acpi_device_id * id)73 int acpi_create_platform_device(struct acpi_device *adev,
74 const struct acpi_device_id *id)
75 {
76 struct platform_device *pdev = NULL;
77 struct acpi_device *acpi_parent;
78 struct platform_device_info pdevinfo;
79 struct resource_list_entry *rentry;
80 struct list_head resource_list;
81 struct resource *resources;
82 int count;
83
84 /* If the ACPI node already has a physical device attached, skip it. */
85 if (adev->physical_node_count)
86 return 0;
87
88 INIT_LIST_HEAD(&resource_list);
89 count = acpi_dev_get_resources(adev, &resource_list, NULL, NULL);
90 if (count <= 0)
91 return 0;
92
93 resources = kmalloc(count * sizeof(struct resource), GFP_KERNEL);
94 if (!resources) {
95 dev_err(&adev->dev, "No memory for resources\n");
96 acpi_dev_free_resource_list(&resource_list);
97 return -ENOMEM;
98 }
99 count = 0;
100 list_for_each_entry(rentry, &resource_list, node)
101 resources[count++] = rentry->res;
102
103 acpi_dev_free_resource_list(&resource_list);
104
105 memset(&pdevinfo, 0, sizeof(pdevinfo));
106 /*
107 * If the ACPI node has a parent and that parent has a physical device
108 * attached to it, that physical device should be the parent of the
109 * platform device we are about to create.
110 */
111 pdevinfo.parent = NULL;
112 acpi_parent = adev->parent;
113 if (acpi_parent) {
114 struct acpi_device_physical_node *entry;
115 struct list_head *list;
116
117 mutex_lock(&acpi_parent->physical_node_lock);
118 list = &acpi_parent->physical_node_list;
119 if (!list_empty(list)) {
120 entry = list_first_entry(list,
121 struct acpi_device_physical_node,
122 node);
123 pdevinfo.parent = entry->dev;
124 }
125 mutex_unlock(&acpi_parent->physical_node_lock);
126 }
127 pdevinfo.name = dev_name(&adev->dev);
128 pdevinfo.id = -1;
129 pdevinfo.res = resources;
130 pdevinfo.num_res = count;
131 pdevinfo.acpi_node.handle = adev->handle;
132 pdevinfo.dma_mask = DMA_BIT_MASK(32);
133 pdev = platform_device_register_full(&pdevinfo);
134 if (IS_ERR(pdev)) {
135 dev_err(&adev->dev, "platform device creation failed: %ld\n",
136 PTR_ERR(pdev));
137 pdev = NULL;
138 } else {
139 dev_dbg(&adev->dev, "created platform device %s\n",
140 dev_name(&pdev->dev));
141 }
142
143 kfree(resources);
144 return 1;
145 }
146
147 static struct acpi_scan_handler platform_handler = {
148 .ids = acpi_platform_device_ids,
149 .attach = acpi_create_platform_device,
150 };
151
acpi_platform_init(void)152 void __init acpi_platform_init(void)
153 {
154 acpi_scan_add_handler(&platform_handler);
155 }
156