blob: 187dff96356bd1b79557c6274a11c9747061dcde [file] [log] [blame]
Linus Torvalds1da177e2005-04-16 15:20:36 -07001/*
2 * acpi_osl.c - OS-dependent functions ($Revision: 83 $)
3 *
4 * Copyright (C) 2000 Andrew Henroid
5 * Copyright (C) 2001, 2002 Andy Grover <andrew.grover@intel.com>
6 * Copyright (C) 2001, 2002 Paul Diefenbaugh <paul.s.diefenbaugh@intel.com>
Matthew Wilcoxf1241c82008-03-14 13:43:13 -04007 * Copyright (c) 2008 Intel Corporation
8 * Author: Matthew Wilcox <willy@linux.intel.com>
Linus Torvalds1da177e2005-04-16 15:20:36 -07009 *
10 * ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
11 *
12 * This program is free software; you can redistribute it and/or modify
13 * it under the terms of the GNU General Public License as published by
14 * the Free Software Foundation; either version 2 of the License, or
15 * (at your option) any later version.
16 *
17 * This program is distributed in the hope that it will be useful,
18 * but WITHOUT ANY WARRANTY; without even the implied warranty of
19 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
20 * GNU General Public License for more details.
21 *
22 * You should have received a copy of the GNU General Public License
23 * along with this program; if not, write to the Free Software
24 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
25 *
26 * ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
27 *
28 */
29
Linus Torvalds1da177e2005-04-16 15:20:36 -070030#include <linux/module.h>
31#include <linux/kernel.h>
32#include <linux/slab.h>
33#include <linux/mm.h>
34#include <linux/pci.h>
Linus Torvalds1da177e2005-04-16 15:20:36 -070035#include <linux/interrupt.h>
36#include <linux/kmod.h>
37#include <linux/delay.h>
38#include <linux/workqueue.h>
39#include <linux/nmi.h>
Alexey Starikovskiyad718602007-02-02 19:48:19 +030040#include <linux/acpi.h>
Rafael J. Wysocki2d6d9fd2011-01-19 22:27:14 +010041#include <linux/acpi_io.h>
Linus Torvalds1da177e2005-04-16 15:20:36 -070042#include <linux/efi.h>
Thomas Renningerdf92e692008-02-04 23:31:22 -080043#include <linux/ioport.h>
44#include <linux/list.h>
Matthew Wilcoxf1241c82008-03-14 13:43:13 -040045#include <linux/jiffies.h>
46#include <linux/semaphore.h>
47
48#include <asm/io.h>
49#include <asm/uaccess.h>
50
51#include <acpi/acpi.h>
52#include <acpi/acpi_bus.h>
53#include <acpi/processor.h>
Linus Torvalds1da177e2005-04-16 15:20:36 -070054
Linus Torvalds1da177e2005-04-16 15:20:36 -070055#define _COMPONENT ACPI_OS_SERVICES
Len Brownf52fd662007-02-12 22:42:12 -050056ACPI_MODULE_NAME("osl");
Linus Torvalds1da177e2005-04-16 15:20:36 -070057#define PREFIX "ACPI: "
Len Brown4be44fc2005-08-05 00:44:28 -040058struct acpi_os_dpc {
59 acpi_osd_exec_callback function;
60 void *context;
David Howells65f27f32006-11-22 14:55:48 +000061 struct work_struct work;
Bjorn Helgaas9ac61852009-08-31 22:32:10 +000062 int wait;
Linus Torvalds1da177e2005-04-16 15:20:36 -070063};
64
65#ifdef CONFIG_ACPI_CUSTOM_DSDT
66#include CONFIG_ACPI_CUSTOM_DSDT_FILE
67#endif
68
69#ifdef ENABLE_DEBUGGER
70#include <linux/kdb.h>
71
72/* stuff for debugger support */
73int acpi_in_debugger;
74EXPORT_SYMBOL(acpi_in_debugger);
75
76extern char line_buf[80];
Len Brown4be44fc2005-08-05 00:44:28 -040077#endif /*ENABLE_DEBUGGER */
Linus Torvalds1da177e2005-04-16 15:20:36 -070078
Linus Torvalds1da177e2005-04-16 15:20:36 -070079static acpi_osd_handler acpi_irq_handler;
80static void *acpi_irq_context;
81static struct workqueue_struct *kacpid_wq;
Alexey Starikovskiy88db5e12007-05-09 23:31:03 -040082static struct workqueue_struct *kacpi_notify_wq;
Zhang Ruic02256b2009-06-23 10:20:29 +080083static struct workqueue_struct *kacpi_hotplug_wq;
Linus Torvalds1da177e2005-04-16 15:20:36 -070084
Thomas Renningerdf92e692008-02-04 23:31:22 -080085struct acpi_res_list {
86 resource_size_t start;
87 resource_size_t end;
88 acpi_adr_space_type resource_type; /* IO port, System memory, ...*/
89 char name[5]; /* only can have a length of 4 chars, make use of this
90 one instead of res->name, no need to kalloc then */
91 struct list_head resource_list;
Lin Minga5fe1a02009-08-13 10:43:27 +080092 int count;
Thomas Renningerdf92e692008-02-04 23:31:22 -080093};
94
95static LIST_HEAD(resource_list_head);
96static DEFINE_SPINLOCK(acpi_res_lock);
97
Myron Stowe620242a2010-10-21 14:23:53 -060098/*
99 * This list of permanent mappings is for memory that may be accessed from
100 * interrupt context, where we can't do the ioremap().
101 */
102struct acpi_ioremap {
103 struct list_head list;
104 void __iomem *virt;
105 acpi_physical_address phys;
106 acpi_size size;
Myron Stowe4a3cba52010-10-21 14:24:14 -0600107 struct kref ref;
Myron Stowe620242a2010-10-21 14:23:53 -0600108};
109
110static LIST_HEAD(acpi_ioremaps);
111static DEFINE_SPINLOCK(acpi_ioremap_lock);
112
Lin Mingb0ed7a92010-08-06 09:35:51 +0800113static void __init acpi_osi_setup_late(void);
Len Brownae00d812007-05-29 18:43:33 -0400114
Len Brownd4b7dc42008-01-23 20:50:56 -0500115/*
Len Browna6e08872008-11-08 01:21:10 -0500116 * The story of _OSI(Linux)
Len Brownd4b7dc42008-01-23 20:50:56 -0500117 *
Len Browna6e08872008-11-08 01:21:10 -0500118 * From pre-history through Linux-2.6.22,
119 * Linux responded TRUE upon a BIOS OSI(Linux) query.
Len Brownd4b7dc42008-01-23 20:50:56 -0500120 *
Len Browna6e08872008-11-08 01:21:10 -0500121 * Unfortunately, reference BIOS writers got wind of this
122 * and put OSI(Linux) in their example code, quickly exposing
123 * this string as ill-conceived and opening the door to
124 * an un-bounded number of BIOS incompatibilities.
Len Brownd4b7dc42008-01-23 20:50:56 -0500125 *
Len Browna6e08872008-11-08 01:21:10 -0500126 * For example, OSI(Linux) was used on resume to re-POST a
127 * video card on one system, because Linux at that time
128 * could not do a speedy restore in its native driver.
129 * But then upon gaining quick native restore capability,
130 * Linux has no way to tell the BIOS to skip the time-consuming
131 * POST -- putting Linux at a permanent performance disadvantage.
132 * On another system, the BIOS writer used OSI(Linux)
133 * to infer native OS support for IPMI! On other systems,
134 * OSI(Linux) simply got in the way of Linux claiming to
135 * be compatible with other operating systems, exposing
136 * BIOS issues such as skipped device initialization.
Len Brownd4b7dc42008-01-23 20:50:56 -0500137 *
Len Browna6e08872008-11-08 01:21:10 -0500138 * So "Linux" turned out to be a really poor chose of
139 * OSI string, and from Linux-2.6.23 onward we respond FALSE.
Len Brownd4b7dc42008-01-23 20:50:56 -0500140 *
141 * BIOS writers should NOT query _OSI(Linux) on future systems.
Len Browna6e08872008-11-08 01:21:10 -0500142 * Linux will complain on the console when it sees it, and return FALSE.
143 * To get Linux to return TRUE for your system will require
144 * a kernel source update to add a DMI entry,
145 * or boot with "acpi_osi=Linux"
Len Brownd4b7dc42008-01-23 20:50:56 -0500146 */
Len Brownd4b7dc42008-01-23 20:50:56 -0500147
Adrian Bunk1d15d842008-01-29 00:10:15 +0200148static struct osi_linux {
Len Brownd4b7dc42008-01-23 20:50:56 -0500149 unsigned int enable:1;
150 unsigned int dmi:1;
151 unsigned int cmdline:1;
Lin Mingd90aa922010-12-09 16:50:52 +0800152} osi_linux = {0, 0, 0};
Len Brownf5076542007-05-30 00:10:38 -0400153
Lin Mingb0ed7a92010-08-06 09:35:51 +0800154static u32 acpi_osi_handler(acpi_string interface, u32 supported)
155{
156 if (!strcmp("Linux", interface)) {
157
Len Brown3af283e2010-10-15 21:38:57 -0400158 printk(KERN_NOTICE FW_BUG PREFIX
Lin Mingb0ed7a92010-08-06 09:35:51 +0800159 "BIOS _OSI(Linux) query %s%s\n",
160 osi_linux.enable ? "honored" : "ignored",
161 osi_linux.cmdline ? " via cmdline" :
162 osi_linux.dmi ? " via DMI" : "");
163 }
164
165 return supported;
166}
167
Bjorn Helgaas9a47cdb2007-01-18 16:42:55 -0700168static void __init acpi_request_region (struct acpi_generic_address *addr,
169 unsigned int length, char *desc)
170{
Bjorn Helgaas9a47cdb2007-01-18 16:42:55 -0700171 if (!addr->address || !length)
172 return;
173
Andi Kleencfa806f2010-07-20 15:18:36 -0700174 /* Resources are never freed */
Len Browneee3c852007-02-03 01:38:16 -0500175 if (addr->space_id == ACPI_ADR_SPACE_SYSTEM_IO)
Andi Kleencfa806f2010-07-20 15:18:36 -0700176 request_region(addr->address, length, desc);
Len Browneee3c852007-02-03 01:38:16 -0500177 else if (addr->space_id == ACPI_ADR_SPACE_SYSTEM_MEMORY)
Andi Kleencfa806f2010-07-20 15:18:36 -0700178 request_mem_region(addr->address, length, desc);
Bjorn Helgaas9a47cdb2007-01-18 16:42:55 -0700179}
180
181static int __init acpi_reserve_resources(void)
182{
Len Browneee3c852007-02-03 01:38:16 -0500183 acpi_request_region(&acpi_gbl_FADT.xpm1a_event_block, acpi_gbl_FADT.pm1_event_length,
Bjorn Helgaas9a47cdb2007-01-18 16:42:55 -0700184 "ACPI PM1a_EVT_BLK");
185
Len Browneee3c852007-02-03 01:38:16 -0500186 acpi_request_region(&acpi_gbl_FADT.xpm1b_event_block, acpi_gbl_FADT.pm1_event_length,
Bjorn Helgaas9a47cdb2007-01-18 16:42:55 -0700187 "ACPI PM1b_EVT_BLK");
188
Len Browneee3c852007-02-03 01:38:16 -0500189 acpi_request_region(&acpi_gbl_FADT.xpm1a_control_block, acpi_gbl_FADT.pm1_control_length,
Bjorn Helgaas9a47cdb2007-01-18 16:42:55 -0700190 "ACPI PM1a_CNT_BLK");
191
Len Browneee3c852007-02-03 01:38:16 -0500192 acpi_request_region(&acpi_gbl_FADT.xpm1b_control_block, acpi_gbl_FADT.pm1_control_length,
Bjorn Helgaas9a47cdb2007-01-18 16:42:55 -0700193 "ACPI PM1b_CNT_BLK");
194
Len Browneee3c852007-02-03 01:38:16 -0500195 if (acpi_gbl_FADT.pm_timer_length == 4)
196 acpi_request_region(&acpi_gbl_FADT.xpm_timer_block, 4, "ACPI PM_TMR");
Bjorn Helgaas9a47cdb2007-01-18 16:42:55 -0700197
Len Browneee3c852007-02-03 01:38:16 -0500198 acpi_request_region(&acpi_gbl_FADT.xpm2_control_block, acpi_gbl_FADT.pm2_control_length,
Bjorn Helgaas9a47cdb2007-01-18 16:42:55 -0700199 "ACPI PM2_CNT_BLK");
200
201 /* Length of GPE blocks must be a non-negative multiple of 2 */
202
Len Browneee3c852007-02-03 01:38:16 -0500203 if (!(acpi_gbl_FADT.gpe0_block_length & 0x1))
204 acpi_request_region(&acpi_gbl_FADT.xgpe0_block,
205 acpi_gbl_FADT.gpe0_block_length, "ACPI GPE0_BLK");
Bjorn Helgaas9a47cdb2007-01-18 16:42:55 -0700206
Len Browneee3c852007-02-03 01:38:16 -0500207 if (!(acpi_gbl_FADT.gpe1_block_length & 0x1))
208 acpi_request_region(&acpi_gbl_FADT.xgpe1_block,
209 acpi_gbl_FADT.gpe1_block_length, "ACPI GPE1_BLK");
Bjorn Helgaas9a47cdb2007-01-18 16:42:55 -0700210
211 return 0;
212}
213device_initcall(acpi_reserve_resources);
214
Len Brown4be44fc2005-08-05 00:44:28 -0400215void acpi_os_printf(const char *fmt, ...)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700216{
217 va_list args;
218 va_start(args, fmt);
219 acpi_os_vprintf(fmt, args);
220 va_end(args);
221}
Len Brown4be44fc2005-08-05 00:44:28 -0400222
Len Brown4be44fc2005-08-05 00:44:28 -0400223void acpi_os_vprintf(const char *fmt, va_list args)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700224{
225 static char buffer[512];
Len Brown4be44fc2005-08-05 00:44:28 -0400226
Linus Torvalds1da177e2005-04-16 15:20:36 -0700227 vsprintf(buffer, fmt, args);
228
229#ifdef ENABLE_DEBUGGER
230 if (acpi_in_debugger) {
231 kdb_printf("%s", buffer);
232 } else {
Frank Seidel4d939152009-02-04 17:03:07 +0100233 printk(KERN_CONT "%s", buffer);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700234 }
235#else
Frank Seidel4d939152009-02-04 17:03:07 +0100236 printk(KERN_CONT "%s", buffer);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700237#endif
238}
239
Alexey Starikovskiyad718602007-02-02 19:48:19 +0300240acpi_physical_address __init acpi_os_get_root_pointer(void)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700241{
242 if (efi_enabled) {
Bjorn Helgaasb2c99e32006-03-26 01:37:08 -0800243 if (efi.acpi20 != EFI_INVALID_TABLE_ADDR)
Alexey Starikovskiyad718602007-02-02 19:48:19 +0300244 return efi.acpi20;
Bjorn Helgaasb2c99e32006-03-26 01:37:08 -0800245 else if (efi.acpi != EFI_INVALID_TABLE_ADDR)
Alexey Starikovskiyad718602007-02-02 19:48:19 +0300246 return efi.acpi;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700247 else {
Len Brown4be44fc2005-08-05 00:44:28 -0400248 printk(KERN_ERR PREFIX
249 "System description tables not found\n");
Alexey Starikovskiyad718602007-02-02 19:48:19 +0300250 return 0;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700251 }
Len Brown239665a2007-11-23 20:08:02 -0500252 } else {
253 acpi_physical_address pa = 0;
254
255 acpi_find_root_pointer(&pa);
256 return pa;
257 }
Linus Torvalds1da177e2005-04-16 15:20:36 -0700258}
259
Myron Stowe78cdb3e2010-10-21 14:24:09 -0600260/* Must be called with 'acpi_ioremap_lock' or RCU read lock held. */
Myron Stowe4a3cba52010-10-21 14:24:14 -0600261static struct acpi_ioremap *
262acpi_map_lookup(acpi_physical_address phys, acpi_size size)
Myron Stowe620242a2010-10-21 14:23:53 -0600263{
264 struct acpi_ioremap *map;
265
Myron Stowe78cdb3e2010-10-21 14:24:09 -0600266 list_for_each_entry_rcu(map, &acpi_ioremaps, list)
Myron Stowe620242a2010-10-21 14:23:53 -0600267 if (map->phys <= phys &&
268 phys + size <= map->phys + map->size)
Myron Stowe4a3cba52010-10-21 14:24:14 -0600269 return map;
270
271 return NULL;
272}
273
274/* Must be called with 'acpi_ioremap_lock' or RCU read lock held. */
275static void __iomem *
276acpi_map_vaddr_lookup(acpi_physical_address phys, unsigned int size)
277{
278 struct acpi_ioremap *map;
279
280 map = acpi_map_lookup(phys, size);
281 if (map)
282 return map->virt + (phys - map->phys);
Myron Stowe620242a2010-10-21 14:23:53 -0600283
284 return NULL;
285}
286
Myron Stowe78cdb3e2010-10-21 14:24:09 -0600287/* Must be called with 'acpi_ioremap_lock' or RCU read lock held. */
Myron Stowe620242a2010-10-21 14:23:53 -0600288static struct acpi_ioremap *
289acpi_map_lookup_virt(void __iomem *virt, acpi_size size)
290{
291 struct acpi_ioremap *map;
292
Myron Stowe78cdb3e2010-10-21 14:24:09 -0600293 list_for_each_entry_rcu(map, &acpi_ioremaps, list)
Myron Stowe4a3cba52010-10-21 14:24:14 -0600294 if (map->virt <= virt &&
295 virt + size <= map->virt + map->size)
Myron Stowe620242a2010-10-21 14:23:53 -0600296 return map;
297
298 return NULL;
299}
300
Jan Beulich2fdf0742007-12-13 08:33:59 +0000301void __iomem *__init_refok
302acpi_os_map_memory(acpi_physical_address phys, acpi_size size)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700303{
Myron Stowe4a3cba52010-10-21 14:24:14 -0600304 struct acpi_ioremap *map, *tmp_map;
Rafael J. Wysocki2d6d9fd2011-01-19 22:27:14 +0100305 unsigned long flags;
Myron Stowe620242a2010-10-21 14:23:53 -0600306 void __iomem *virt;
Rafael J. Wysocki2d6d9fd2011-01-19 22:27:14 +0100307 acpi_physical_address pg_off;
308 acpi_size pg_sz;
Myron Stowe620242a2010-10-21 14:23:53 -0600309
Bjorn Helgaas9f4fd612006-03-26 01:37:10 -0800310 if (phys > ULONG_MAX) {
311 printk(KERN_ERR PREFIX "Cannot map memory that high\n");
Randy Dunlap70c08462007-02-13 16:11:36 -0800312 return NULL;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700313 }
Myron Stowe620242a2010-10-21 14:23:53 -0600314
315 if (!acpi_gbl_permanent_mmap)
Alexey Starikovskiyad718602007-02-02 19:48:19 +0300316 return __acpi_map_table((unsigned long)phys, size);
Myron Stowe620242a2010-10-21 14:23:53 -0600317
318 map = kzalloc(sizeof(*map), GFP_KERNEL);
319 if (!map)
320 return NULL;
321
Myron Stowe4a3cba52010-10-21 14:24:14 -0600322 pg_off = round_down(phys, PAGE_SIZE);
323 pg_sz = round_up(phys + size, PAGE_SIZE) - pg_off;
Rafael J. Wysocki2d6d9fd2011-01-19 22:27:14 +0100324 virt = acpi_os_ioremap(pg_off, pg_sz);
Myron Stowe620242a2010-10-21 14:23:53 -0600325 if (!virt) {
326 kfree(map);
327 return NULL;
328 }
329
330 INIT_LIST_HEAD(&map->list);
331 map->virt = virt;
Myron Stowe4a3cba52010-10-21 14:24:14 -0600332 map->phys = pg_off;
333 map->size = pg_sz;
334 kref_init(&map->ref);
Myron Stowe620242a2010-10-21 14:23:53 -0600335
336 spin_lock_irqsave(&acpi_ioremap_lock, flags);
Myron Stowe4a3cba52010-10-21 14:24:14 -0600337 /* Check if page has already been mapped. */
338 tmp_map = acpi_map_lookup(phys, size);
339 if (tmp_map) {
340 kref_get(&tmp_map->ref);
341 spin_unlock_irqrestore(&acpi_ioremap_lock, flags);
342 iounmap(map->virt);
343 kfree(map);
344 return tmp_map->virt + (phys - tmp_map->phys);
345 }
Myron Stowe78cdb3e2010-10-21 14:24:09 -0600346 list_add_tail_rcu(&map->list, &acpi_ioremaps);
Myron Stowe620242a2010-10-21 14:23:53 -0600347 spin_unlock_irqrestore(&acpi_ioremap_lock, flags);
348
Myron Stowe4a3cba52010-10-21 14:24:14 -0600349 return map->virt + (phys - map->phys);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700350}
Kylene Jo Hall55a82ab2006-01-08 01:03:15 -0800351EXPORT_SYMBOL_GPL(acpi_os_map_memory);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700352
Myron Stowe4a3cba52010-10-21 14:24:14 -0600353static void acpi_kref_del_iomap(struct kref *ref)
354{
355 struct acpi_ioremap *map;
356
357 map = container_of(ref, struct acpi_ioremap, ref);
358 list_del_rcu(&map->list);
359}
360
Jeremy Fitzhardinge0d3a9cf2009-02-22 14:58:56 -0800361void __ref acpi_os_unmap_memory(void __iomem *virt, acpi_size size)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700362{
Myron Stowe620242a2010-10-21 14:23:53 -0600363 struct acpi_ioremap *map;
364 unsigned long flags;
Myron Stowe4a3cba52010-10-21 14:24:14 -0600365 int del;
Myron Stowe620242a2010-10-21 14:23:53 -0600366
367 if (!acpi_gbl_permanent_mmap) {
Yinghai Lu7d972772009-02-07 15:39:41 -0800368 __acpi_unmap_table(virt, size);
Myron Stowe620242a2010-10-21 14:23:53 -0600369 return;
370 }
371
372 spin_lock_irqsave(&acpi_ioremap_lock, flags);
373 map = acpi_map_lookup_virt(virt, size);
374 if (!map) {
375 spin_unlock_irqrestore(&acpi_ioremap_lock, flags);
376 printk(KERN_ERR PREFIX "%s: bad address %p\n", __func__, virt);
377 dump_stack();
378 return;
379 }
380
Myron Stowe4a3cba52010-10-21 14:24:14 -0600381 del = kref_put(&map->ref, acpi_kref_del_iomap);
Myron Stowe620242a2010-10-21 14:23:53 -0600382 spin_unlock_irqrestore(&acpi_ioremap_lock, flags);
383
Myron Stowe4a3cba52010-10-21 14:24:14 -0600384 if (!del)
385 return;
386
Myron Stowe78cdb3e2010-10-21 14:24:09 -0600387 synchronize_rcu();
Myron Stowe620242a2010-10-21 14:23:53 -0600388 iounmap(map->virt);
389 kfree(map);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700390}
Kylene Jo Hall55a82ab2006-01-08 01:03:15 -0800391EXPORT_SYMBOL_GPL(acpi_os_unmap_memory);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700392
Jeremy Fitzhardinge0d3a9cf2009-02-22 14:58:56 -0800393void __init early_acpi_os_unmap_memory(void __iomem *virt, acpi_size size)
Yinghai Lu7d972772009-02-07 15:39:41 -0800394{
395 if (!acpi_gbl_permanent_mmap)
396 __acpi_unmap_table(virt, size);
397}
398
Myron Stowe29718522010-10-21 14:23:59 -0600399int acpi_os_map_generic_address(struct acpi_generic_address *addr)
400{
401 void __iomem *virt;
402
403 if (addr->space_id != ACPI_ADR_SPACE_SYSTEM_MEMORY)
404 return 0;
405
406 if (!addr->address || !addr->bit_width)
407 return -EINVAL;
408
409 virt = acpi_os_map_memory(addr->address, addr->bit_width / 8);
410 if (!virt)
411 return -EIO;
412
413 return 0;
414}
415EXPORT_SYMBOL_GPL(acpi_os_map_generic_address);
416
417void acpi_os_unmap_generic_address(struct acpi_generic_address *addr)
418{
419 void __iomem *virt;
420 unsigned long flags;
421 acpi_size size = addr->bit_width / 8;
422
423 if (addr->space_id != ACPI_ADR_SPACE_SYSTEM_MEMORY)
424 return;
425
426 if (!addr->address || !addr->bit_width)
427 return;
428
429 spin_lock_irqsave(&acpi_ioremap_lock, flags);
430 virt = acpi_map_vaddr_lookup(addr->address, size);
431 spin_unlock_irqrestore(&acpi_ioremap_lock, flags);
432
433 acpi_os_unmap_memory(virt, size);
434}
435EXPORT_SYMBOL_GPL(acpi_os_unmap_generic_address);
436
Linus Torvalds1da177e2005-04-16 15:20:36 -0700437#ifdef ACPI_FUTURE_USAGE
438acpi_status
Len Brown4be44fc2005-08-05 00:44:28 -0400439acpi_os_get_physical_address(void *virt, acpi_physical_address * phys)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700440{
Len Brown4be44fc2005-08-05 00:44:28 -0400441 if (!phys || !virt)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700442 return AE_BAD_PARAMETER;
443
444 *phys = virt_to_phys(virt);
445
446 return AE_OK;
447}
448#endif
449
450#define ACPI_MAX_OVERRIDE_LEN 100
451
452static char acpi_os_name[ACPI_MAX_OVERRIDE_LEN];
453
454acpi_status
Len Brown4be44fc2005-08-05 00:44:28 -0400455acpi_os_predefined_override(const struct acpi_predefined_names *init_val,
456 acpi_string * new_val)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700457{
458 if (!init_val || !new_val)
459 return AE_BAD_PARAMETER;
460
461 *new_val = NULL;
Len Brown4be44fc2005-08-05 00:44:28 -0400462 if (!memcmp(init_val->name, "_OS_", 4) && strlen(acpi_os_name)) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700463 printk(KERN_INFO PREFIX "Overriding _OS definition to '%s'\n",
Len Brown4be44fc2005-08-05 00:44:28 -0400464 acpi_os_name);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700465 *new_val = acpi_os_name;
466 }
467
468 return AE_OK;
469}
470
471acpi_status
Len Brown4be44fc2005-08-05 00:44:28 -0400472acpi_os_table_override(struct acpi_table_header * existing_table,
473 struct acpi_table_header ** new_table)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700474{
475 if (!existing_table || !new_table)
476 return AE_BAD_PARAMETER;
477
Markus Gaugusch71fc47a2008-02-05 00:04:06 +0100478 *new_table = NULL;
479
Linus Torvalds1da177e2005-04-16 15:20:36 -0700480#ifdef CONFIG_ACPI_CUSTOM_DSDT
481 if (strncmp(existing_table->signature, "DSDT", 4) == 0)
Len Brown4be44fc2005-08-05 00:44:28 -0400482 *new_table = (struct acpi_table_header *)AmlCode;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700483#endif
Éric Piel6ed31e92008-02-05 00:04:50 +0100484 if (*new_table != NULL) {
485 printk(KERN_WARNING PREFIX "Override [%4.4s-%8.8s], "
486 "this is unsafe: tainting kernel\n",
487 existing_table->signature,
488 existing_table->oem_table_id);
489 add_taint(TAINT_OVERRIDDEN_ACPI_TABLE);
490 }
Linus Torvalds1da177e2005-04-16 15:20:36 -0700491 return AE_OK;
492}
493
David Howells7d12e782006-10-05 14:55:46 +0100494static irqreturn_t acpi_irq(int irq, void *dev_id)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700495{
Len Brown5229e872008-02-06 01:26:55 -0500496 u32 handled;
497
498 handled = (*acpi_irq_handler) (acpi_irq_context);
499
500 if (handled) {
501 acpi_irq_handled++;
502 return IRQ_HANDLED;
Len Brown88bea182009-04-21 00:35:47 -0400503 } else {
504 acpi_irq_not_handled++;
Len Brown5229e872008-02-06 01:26:55 -0500505 return IRQ_NONE;
Len Brown88bea182009-04-21 00:35:47 -0400506 }
Linus Torvalds1da177e2005-04-16 15:20:36 -0700507}
508
509acpi_status
Len Brown4be44fc2005-08-05 00:44:28 -0400510acpi_os_install_interrupt_handler(u32 gsi, acpi_osd_handler handler,
511 void *context)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700512{
513 unsigned int irq;
514
Len Brown5229e872008-02-06 01:26:55 -0500515 acpi_irq_stats_init();
516
Linus Torvalds1da177e2005-04-16 15:20:36 -0700517 /*
Rafael J. Wysocki23fe3632011-02-08 23:48:16 +0100518 * ACPI interrupts different from the SCI in our copy of the FADT are
519 * not supported.
Linus Torvalds1da177e2005-04-16 15:20:36 -0700520 */
Rafael J. Wysocki23fe3632011-02-08 23:48:16 +0100521 if (gsi != acpi_gbl_FADT.sci_interrupt)
522 return AE_BAD_PARAMETER;
523
524 if (acpi_irq_handler)
525 return AE_ALREADY_ACQUIRED;
526
Linus Torvalds1da177e2005-04-16 15:20:36 -0700527 if (acpi_gsi_to_irq(gsi, &irq) < 0) {
528 printk(KERN_ERR PREFIX "SCI (ACPI GSI %d) not registered\n",
529 gsi);
530 return AE_OK;
531 }
532
533 acpi_irq_handler = handler;
534 acpi_irq_context = context;
Thomas Gleixnerdace1452006-07-01 19:29:38 -0700535 if (request_irq(irq, acpi_irq, IRQF_SHARED, "acpi", acpi_irq)) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700536 printk(KERN_ERR PREFIX "SCI (IRQ%d) allocation failed\n", irq);
Rafael J. Wysocki23fe3632011-02-08 23:48:16 +0100537 acpi_irq_handler = NULL;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700538 return AE_NOT_ACQUIRED;
539 }
Linus Torvalds1da177e2005-04-16 15:20:36 -0700540
541 return AE_OK;
542}
543
Len Brown4be44fc2005-08-05 00:44:28 -0400544acpi_status acpi_os_remove_interrupt_handler(u32 irq, acpi_osd_handler handler)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700545{
Rafael J. Wysocki23fe3632011-02-08 23:48:16 +0100546 if (irq != acpi_gbl_FADT.sci_interrupt)
547 return AE_BAD_PARAMETER;
548
549 free_irq(irq, acpi_irq);
550 acpi_irq_handler = NULL;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700551
552 return AE_OK;
553}
554
555/*
556 * Running in interpreter thread context, safe to sleep
557 */
558
Lin Ming439913f2010-01-28 10:53:19 +0800559void acpi_os_sleep(u64 ms)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700560{
Nishanth Aravamudan01a527e2005-11-07 01:01:14 -0800561 schedule_timeout_interruptible(msecs_to_jiffies(ms));
Linus Torvalds1da177e2005-04-16 15:20:36 -0700562}
Len Brown4be44fc2005-08-05 00:44:28 -0400563
Len Brown4be44fc2005-08-05 00:44:28 -0400564void acpi_os_stall(u32 us)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700565{
566 while (us) {
567 u32 delay = 1000;
568
569 if (delay > us)
570 delay = us;
571 udelay(delay);
572 touch_nmi_watchdog();
573 us -= delay;
574 }
575}
Len Brown4be44fc2005-08-05 00:44:28 -0400576
Linus Torvalds1da177e2005-04-16 15:20:36 -0700577/*
578 * Support ACPI 3.0 AML Timer operand
579 * Returns 64-bit free-running, monotonically increasing timer
580 * with 100ns granularity
581 */
Len Brown4be44fc2005-08-05 00:44:28 -0400582u64 acpi_os_get_timer(void)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700583{
584 static u64 t;
585
586#ifdef CONFIG_HPET
587 /* TBD: use HPET if available */
588#endif
589
590#ifdef CONFIG_X86_PM_TIMER
591 /* TBD: default to PM timer if HPET was not available */
592#endif
593 if (!t)
594 printk(KERN_ERR PREFIX "acpi_os_get_timer() TBD\n");
595
596 return ++t;
597}
598
Len Brown4be44fc2005-08-05 00:44:28 -0400599acpi_status acpi_os_read_port(acpi_io_address port, u32 * value, u32 width)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700600{
601 u32 dummy;
602
603 if (!value)
604 value = &dummy;
605
Zhao Yakui49fbabf2007-11-15 17:01:06 +0800606 *value = 0;
607 if (width <= 8) {
Len Brown4be44fc2005-08-05 00:44:28 -0400608 *(u8 *) value = inb(port);
Zhao Yakui49fbabf2007-11-15 17:01:06 +0800609 } else if (width <= 16) {
Len Brown4be44fc2005-08-05 00:44:28 -0400610 *(u16 *) value = inw(port);
Zhao Yakui49fbabf2007-11-15 17:01:06 +0800611 } else if (width <= 32) {
Len Brown4be44fc2005-08-05 00:44:28 -0400612 *(u32 *) value = inl(port);
Zhao Yakui49fbabf2007-11-15 17:01:06 +0800613 } else {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700614 BUG();
615 }
616
617 return AE_OK;
618}
Len Brown4be44fc2005-08-05 00:44:28 -0400619
Linus Torvalds1da177e2005-04-16 15:20:36 -0700620EXPORT_SYMBOL(acpi_os_read_port);
621
Len Brown4be44fc2005-08-05 00:44:28 -0400622acpi_status acpi_os_write_port(acpi_io_address port, u32 value, u32 width)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700623{
Zhao Yakui49fbabf2007-11-15 17:01:06 +0800624 if (width <= 8) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700625 outb(value, port);
Zhao Yakui49fbabf2007-11-15 17:01:06 +0800626 } else if (width <= 16) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700627 outw(value, port);
Zhao Yakui49fbabf2007-11-15 17:01:06 +0800628 } else if (width <= 32) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700629 outl(value, port);
Zhao Yakui49fbabf2007-11-15 17:01:06 +0800630 } else {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700631 BUG();
632 }
633
634 return AE_OK;
635}
Len Brown4be44fc2005-08-05 00:44:28 -0400636
Linus Torvalds1da177e2005-04-16 15:20:36 -0700637EXPORT_SYMBOL(acpi_os_write_port);
638
639acpi_status
Len Brown4be44fc2005-08-05 00:44:28 -0400640acpi_os_read_memory(acpi_physical_address phys_addr, u32 * value, u32 width)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700641{
Len Brown4be44fc2005-08-05 00:44:28 -0400642 void __iomem *virt_addr;
Rafael J. Wysocki884b8212011-02-08 23:37:16 +0100643 unsigned int size = width / 8;
644 bool unmap = false;
645 u32 dummy;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700646
Myron Stowe78cdb3e2010-10-21 14:24:09 -0600647 rcu_read_lock();
Myron Stowe620242a2010-10-21 14:23:53 -0600648 virt_addr = acpi_map_vaddr_lookup(phys_addr, size);
Myron Stowe620242a2010-10-21 14:23:53 -0600649 if (!virt_addr) {
Rafael J. Wysocki884b8212011-02-08 23:37:16 +0100650 rcu_read_unlock();
Rafael J. Wysocki2d6d9fd2011-01-19 22:27:14 +0100651 virt_addr = acpi_os_ioremap(phys_addr, size);
Rafael J. Wysocki884b8212011-02-08 23:37:16 +0100652 if (!virt_addr)
653 return AE_BAD_ADDRESS;
654 unmap = true;
Myron Stowe620242a2010-10-21 14:23:53 -0600655 }
Rafael J. Wysocki884b8212011-02-08 23:37:16 +0100656
Linus Torvalds1da177e2005-04-16 15:20:36 -0700657 if (!value)
658 value = &dummy;
659
660 switch (width) {
661 case 8:
Len Brown4be44fc2005-08-05 00:44:28 -0400662 *(u8 *) value = readb(virt_addr);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700663 break;
664 case 16:
Len Brown4be44fc2005-08-05 00:44:28 -0400665 *(u16 *) value = readw(virt_addr);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700666 break;
667 case 32:
Len Brown4be44fc2005-08-05 00:44:28 -0400668 *(u32 *) value = readl(virt_addr);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700669 break;
670 default:
671 BUG();
672 }
673
Myron Stowe620242a2010-10-21 14:23:53 -0600674 if (unmap)
675 iounmap(virt_addr);
Rafael J. Wysocki884b8212011-02-08 23:37:16 +0100676 else
677 rcu_read_unlock();
Linus Torvalds1da177e2005-04-16 15:20:36 -0700678
679 return AE_OK;
680}
681
682acpi_status
Len Brown4be44fc2005-08-05 00:44:28 -0400683acpi_os_write_memory(acpi_physical_address phys_addr, u32 value, u32 width)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700684{
Len Brown4be44fc2005-08-05 00:44:28 -0400685 void __iomem *virt_addr;
Rafael J. Wysocki884b8212011-02-08 23:37:16 +0100686 unsigned int size = width / 8;
687 bool unmap = false;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700688
Myron Stowe78cdb3e2010-10-21 14:24:09 -0600689 rcu_read_lock();
Myron Stowe620242a2010-10-21 14:23:53 -0600690 virt_addr = acpi_map_vaddr_lookup(phys_addr, size);
Myron Stowe620242a2010-10-21 14:23:53 -0600691 if (!virt_addr) {
Rafael J. Wysocki884b8212011-02-08 23:37:16 +0100692 rcu_read_unlock();
Rafael J. Wysocki2d6d9fd2011-01-19 22:27:14 +0100693 virt_addr = acpi_os_ioremap(phys_addr, size);
Rafael J. Wysocki884b8212011-02-08 23:37:16 +0100694 if (!virt_addr)
695 return AE_BAD_ADDRESS;
696 unmap = true;
Myron Stowe620242a2010-10-21 14:23:53 -0600697 }
Linus Torvalds1da177e2005-04-16 15:20:36 -0700698
699 switch (width) {
700 case 8:
701 writeb(value, virt_addr);
702 break;
703 case 16:
704 writew(value, virt_addr);
705 break;
706 case 32:
707 writel(value, virt_addr);
708 break;
709 default:
710 BUG();
711 }
712
Myron Stowe620242a2010-10-21 14:23:53 -0600713 if (unmap)
714 iounmap(virt_addr);
Rafael J. Wysocki884b8212011-02-08 23:37:16 +0100715 else
716 rcu_read_unlock();
Linus Torvalds1da177e2005-04-16 15:20:36 -0700717
718 return AE_OK;
719}
720
Linus Torvalds1da177e2005-04-16 15:20:36 -0700721acpi_status
Len Brown4be44fc2005-08-05 00:44:28 -0400722acpi_os_read_pci_configuration(struct acpi_pci_id * pci_id, u32 reg,
Bob Moorec5f02312010-08-06 08:57:53 +0800723 u64 *value, u32 width)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700724{
725 int result, size;
Bob Moorec5f02312010-08-06 08:57:53 +0800726 u32 value32;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700727
728 if (!value)
729 return AE_BAD_PARAMETER;
730
731 switch (width) {
732 case 8:
733 size = 1;
734 break;
735 case 16:
736 size = 2;
737 break;
738 case 32:
739 size = 4;
740 break;
741 default:
742 return AE_ERROR;
743 }
744
Matthew Wilcoxb6ce0682008-02-10 09:45:28 -0500745 result = raw_pci_read(pci_id->segment, pci_id->bus,
746 PCI_DEVFN(pci_id->device, pci_id->function),
Bob Moorec5f02312010-08-06 08:57:53 +0800747 reg, size, &value32);
748 *value = value32;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700749
750 return (result ? AE_ERROR : AE_OK);
751}
Len Brown4be44fc2005-08-05 00:44:28 -0400752
Linus Torvalds1da177e2005-04-16 15:20:36 -0700753acpi_status
Len Brown4be44fc2005-08-05 00:44:28 -0400754acpi_os_write_pci_configuration(struct acpi_pci_id * pci_id, u32 reg,
Lin Ming439913f2010-01-28 10:53:19 +0800755 u64 value, u32 width)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700756{
757 int result, size;
758
759 switch (width) {
760 case 8:
761 size = 1;
762 break;
763 case 16:
764 size = 2;
765 break;
766 case 32:
767 size = 4;
768 break;
769 default:
770 return AE_ERROR;
771 }
772
Matthew Wilcoxb6ce0682008-02-10 09:45:28 -0500773 result = raw_pci_write(pci_id->segment, pci_id->bus,
774 PCI_DEVFN(pci_id->device, pci_id->function),
775 reg, size, value);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700776
777 return (result ? AE_ERROR : AE_OK);
778}
779
David Howells65f27f32006-11-22 14:55:48 +0000780static void acpi_os_execute_deferred(struct work_struct *work)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700781{
David Howells65f27f32006-11-22 14:55:48 +0000782 struct acpi_os_dpc *dpc = container_of(work, struct acpi_os_dpc, work);
Alexey Starikovskiy88db5e12007-05-09 23:31:03 -0400783
Bjorn Helgaas9ac61852009-08-31 22:32:10 +0000784 if (dpc->wait)
785 acpi_os_wait_events_complete(NULL);
Zhang Rui19cd8472008-08-28 10:05:06 +0800786
787 dpc->function(dpc->context);
788 kfree(dpc);
Zhang Rui19cd8472008-08-28 10:05:06 +0800789}
790
Alexey Starikovskiyb8d35192006-05-05 03:23:00 -0400791/*******************************************************************************
792 *
793 * FUNCTION: acpi_os_execute
794 *
795 * PARAMETERS: Type - Type of the callback
796 * Function - Function to be executed
797 * Context - Function parameters
798 *
799 * RETURN: Status
800 *
801 * DESCRIPTION: Depending on type, either queues function for deferred execution or
802 * immediately executes function on a separate thread.
803 *
804 ******************************************************************************/
805
Zhang Rui19cd8472008-08-28 10:05:06 +0800806static acpi_status __acpi_os_execute(acpi_execute_type type,
807 acpi_osd_exec_callback function, void *context, int hp)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700808{
Len Brown4be44fc2005-08-05 00:44:28 -0400809 acpi_status status = AE_OK;
810 struct acpi_os_dpc *dpc;
Alexey Starikovskiy17bc54e2007-11-13 13:05:45 +0300811 struct workqueue_struct *queue;
Zhang Rui19cd8472008-08-28 10:05:06 +0800812 int ret;
Len Brown72945b22006-07-12 22:46:42 -0400813 ACPI_DEBUG_PRINT((ACPI_DB_EXEC,
814 "Scheduling function [%p(%p)] for deferred execution.\n",
815 function, context));
Linus Torvalds1da177e2005-04-16 15:20:36 -0700816
Linus Torvalds1da177e2005-04-16 15:20:36 -0700817 /*
818 * Allocate/initialize DPC structure. Note that this memory will be
David Howells65f27f32006-11-22 14:55:48 +0000819 * freed by the callee. The kernel handles the work_struct list in a
Linus Torvalds1da177e2005-04-16 15:20:36 -0700820 * way that allows us to also free its memory inside the callee.
821 * Because we may want to schedule several tasks with different
822 * parameters we can't use the approach some kernel code uses of
David Howells65f27f32006-11-22 14:55:48 +0000823 * having a static work_struct.
Linus Torvalds1da177e2005-04-16 15:20:36 -0700824 */
Len Brown72945b22006-07-12 22:46:42 -0400825
David Howells65f27f32006-11-22 14:55:48 +0000826 dpc = kmalloc(sizeof(struct acpi_os_dpc), GFP_ATOMIC);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700827 if (!dpc)
Lin Ming889c78b2008-12-31 09:23:57 +0800828 return AE_NO_MEMORY;
Linus Torvaldsb976fe12006-11-17 19:31:09 -0800829
Linus Torvalds1da177e2005-04-16 15:20:36 -0700830 dpc->function = function;
831 dpc->context = context;
Linus Torvaldsb976fe12006-11-17 19:31:09 -0800832
Zhang Ruic02256b2009-06-23 10:20:29 +0800833 /*
834 * We can't run hotplug code in keventd_wq/kacpid_wq/kacpid_notify_wq
835 * because the hotplug code may call driver .remove() functions,
836 * which invoke flush_scheduled_work/acpi_os_wait_events_complete
837 * to flush these workqueues.
838 */
839 queue = hp ? kacpi_hotplug_wq :
840 (type == OSL_NOTIFY_HANDLER ? kacpi_notify_wq : kacpid_wq);
Bjorn Helgaas9ac61852009-08-31 22:32:10 +0000841 dpc->wait = hp ? 1 : 0;
Zhang Ruibc736752010-03-22 15:48:54 +0800842
843 if (queue == kacpi_hotplug_wq)
844 INIT_WORK(&dpc->work, acpi_os_execute_deferred);
845 else if (queue == kacpi_notify_wq)
846 INIT_WORK(&dpc->work, acpi_os_execute_deferred);
847 else
848 INIT_WORK(&dpc->work, acpi_os_execute_deferred);
849
Tejun Heo8fec62b2010-06-29 10:07:09 +0200850 /*
851 * On some machines, a software-initiated SMI causes corruption unless
852 * the SMI runs on CPU 0. An SMI can be initiated by any AML, but
853 * typically it's done in GPE-related methods that are run via
854 * workqueues, so we can avoid the known corruption cases by always
855 * queueing on CPU 0.
856 */
857 ret = queue_work_on(0, queue, &dpc->work);
Zhang Rui19cd8472008-08-28 10:05:06 +0800858
859 if (!ret) {
Lin Ming55ac9a02008-09-28 14:51:56 +0800860 printk(KERN_ERR PREFIX
861 "Call to queue_work() failed.\n");
Alexey Starikovskiy17bc54e2007-11-13 13:05:45 +0300862 status = AE_ERROR;
863 kfree(dpc);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700864 }
Lin Ming889c78b2008-12-31 09:23:57 +0800865 return status;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700866}
Len Brown4be44fc2005-08-05 00:44:28 -0400867
Zhang Rui19cd8472008-08-28 10:05:06 +0800868acpi_status acpi_os_execute(acpi_execute_type type,
869 acpi_osd_exec_callback function, void *context)
870{
871 return __acpi_os_execute(type, function, context, 0);
872}
Alexey Starikovskiyb8d35192006-05-05 03:23:00 -0400873EXPORT_SYMBOL(acpi_os_execute);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700874
Zhang Rui19cd8472008-08-28 10:05:06 +0800875acpi_status acpi_os_hotplug_execute(acpi_osd_exec_callback function,
876 void *context)
877{
878 return __acpi_os_execute(0, function, context, 1);
879}
880
Len Brown4be44fc2005-08-05 00:44:28 -0400881void acpi_os_wait_events_complete(void *context)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700882{
883 flush_workqueue(kacpid_wq);
Zhang Rui2f67a062008-04-29 02:34:42 -0400884 flush_workqueue(kacpi_notify_wq);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700885}
Len Brown4be44fc2005-08-05 00:44:28 -0400886
Linus Torvalds1da177e2005-04-16 15:20:36 -0700887EXPORT_SYMBOL(acpi_os_wait_events_complete);
888
889/*
Linus Torvalds1da177e2005-04-16 15:20:36 -0700890 * Deallocate the memory for a spinlock.
891 */
Bob Moore967440e2006-06-23 17:04:00 -0400892void acpi_os_delete_lock(acpi_spinlock handle)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700893{
Patrick Mocheld550d982006-06-27 00:41:40 -0400894 return;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700895}
896
Linus Torvalds1da177e2005-04-16 15:20:36 -0700897acpi_status
Len Brown4be44fc2005-08-05 00:44:28 -0400898acpi_os_create_semaphore(u32 max_units, u32 initial_units, acpi_handle * handle)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700899{
Len Brown4be44fc2005-08-05 00:44:28 -0400900 struct semaphore *sem = NULL;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700901
Linus Torvalds1da177e2005-04-16 15:20:36 -0700902 sem = acpi_os_allocate(sizeof(struct semaphore));
903 if (!sem)
Patrick Mocheld550d982006-06-27 00:41:40 -0400904 return AE_NO_MEMORY;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700905 memset(sem, 0, sizeof(struct semaphore));
906
907 sema_init(sem, initial_units);
908
Len Brown4be44fc2005-08-05 00:44:28 -0400909 *handle = (acpi_handle *) sem;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700910
Len Brown4be44fc2005-08-05 00:44:28 -0400911 ACPI_DEBUG_PRINT((ACPI_DB_MUTEX, "Creating semaphore[%p|%d].\n",
912 *handle, initial_units));
Linus Torvalds1da177e2005-04-16 15:20:36 -0700913
Patrick Mocheld550d982006-06-27 00:41:40 -0400914 return AE_OK;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700915}
Linus Torvalds1da177e2005-04-16 15:20:36 -0700916
Linus Torvalds1da177e2005-04-16 15:20:36 -0700917/*
918 * TODO: A better way to delete semaphores? Linux doesn't have a
919 * 'delete_semaphore()' function -- may result in an invalid
920 * pointer dereference for non-synchronized consumers. Should
921 * we at least check for blocked threads and signal/cancel them?
922 */
923
Len Brown4be44fc2005-08-05 00:44:28 -0400924acpi_status acpi_os_delete_semaphore(acpi_handle handle)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700925{
Len Brown4be44fc2005-08-05 00:44:28 -0400926 struct semaphore *sem = (struct semaphore *)handle;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700927
Linus Torvalds1da177e2005-04-16 15:20:36 -0700928 if (!sem)
Patrick Mocheld550d982006-06-27 00:41:40 -0400929 return AE_BAD_PARAMETER;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700930
Len Brown4be44fc2005-08-05 00:44:28 -0400931 ACPI_DEBUG_PRINT((ACPI_DB_MUTEX, "Deleting semaphore[%p].\n", handle));
Linus Torvalds1da177e2005-04-16 15:20:36 -0700932
Matthew Wilcoxf1241c82008-03-14 13:43:13 -0400933 BUG_ON(!list_empty(&sem->wait_list));
Len Brown02438d82006-06-30 03:19:10 -0400934 kfree(sem);
Len Brown4be44fc2005-08-05 00:44:28 -0400935 sem = NULL;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700936
Patrick Mocheld550d982006-06-27 00:41:40 -0400937 return AE_OK;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700938}
Linus Torvalds1da177e2005-04-16 15:20:36 -0700939
Linus Torvalds1da177e2005-04-16 15:20:36 -0700940/*
Linus Torvalds1da177e2005-04-16 15:20:36 -0700941 * TODO: Support for units > 1?
942 */
Len Brown4be44fc2005-08-05 00:44:28 -0400943acpi_status acpi_os_wait_semaphore(acpi_handle handle, u32 units, u16 timeout)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700944{
Len Brown4be44fc2005-08-05 00:44:28 -0400945 acpi_status status = AE_OK;
946 struct semaphore *sem = (struct semaphore *)handle;
Matthew Wilcoxf1241c82008-03-14 13:43:13 -0400947 long jiffies;
Len Brown4be44fc2005-08-05 00:44:28 -0400948 int ret = 0;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700949
Linus Torvalds1da177e2005-04-16 15:20:36 -0700950 if (!sem || (units < 1))
Patrick Mocheld550d982006-06-27 00:41:40 -0400951 return AE_BAD_PARAMETER;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700952
953 if (units > 1)
Patrick Mocheld550d982006-06-27 00:41:40 -0400954 return AE_SUPPORT;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700955
Len Brown4be44fc2005-08-05 00:44:28 -0400956 ACPI_DEBUG_PRINT((ACPI_DB_MUTEX, "Waiting for semaphore[%p|%d|%d]\n",
957 handle, units, timeout));
Linus Torvalds1da177e2005-04-16 15:20:36 -0700958
Matthew Wilcoxf1241c82008-03-14 13:43:13 -0400959 if (timeout == ACPI_WAIT_FOREVER)
960 jiffies = MAX_SCHEDULE_TIMEOUT;
961 else
962 jiffies = msecs_to_jiffies(timeout);
963
964 ret = down_timeout(sem, jiffies);
965 if (ret)
966 status = AE_TIME;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700967
968 if (ACPI_FAILURE(status)) {
Bjorn Helgaas9e7e2c02006-04-27 05:25:00 -0400969 ACPI_DEBUG_PRINT((ACPI_DB_MUTEX,
Thomas Renningera6fc6722006-06-26 23:58:43 -0400970 "Failed to acquire semaphore[%p|%d|%d], %s",
Len Brown4be44fc2005-08-05 00:44:28 -0400971 handle, units, timeout,
972 acpi_format_exception(status)));
973 } else {
974 ACPI_DEBUG_PRINT((ACPI_DB_MUTEX,
Thomas Renningera6fc6722006-06-26 23:58:43 -0400975 "Acquired semaphore[%p|%d|%d]", handle,
Len Brown4be44fc2005-08-05 00:44:28 -0400976 units, timeout));
Linus Torvalds1da177e2005-04-16 15:20:36 -0700977 }
978
Patrick Mocheld550d982006-06-27 00:41:40 -0400979 return status;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700980}
Linus Torvalds1da177e2005-04-16 15:20:36 -0700981
Linus Torvalds1da177e2005-04-16 15:20:36 -0700982/*
983 * TODO: Support for units > 1?
984 */
Len Brown4be44fc2005-08-05 00:44:28 -0400985acpi_status acpi_os_signal_semaphore(acpi_handle handle, u32 units)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700986{
Len Brown4be44fc2005-08-05 00:44:28 -0400987 struct semaphore *sem = (struct semaphore *)handle;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700988
Linus Torvalds1da177e2005-04-16 15:20:36 -0700989 if (!sem || (units < 1))
Patrick Mocheld550d982006-06-27 00:41:40 -0400990 return AE_BAD_PARAMETER;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700991
992 if (units > 1)
Patrick Mocheld550d982006-06-27 00:41:40 -0400993 return AE_SUPPORT;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700994
Len Brown4be44fc2005-08-05 00:44:28 -0400995 ACPI_DEBUG_PRINT((ACPI_DB_MUTEX, "Signaling semaphore[%p|%d]\n", handle,
996 units));
Linus Torvalds1da177e2005-04-16 15:20:36 -0700997
998 up(sem);
999
Patrick Mocheld550d982006-06-27 00:41:40 -04001000 return AE_OK;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001001}
Len Brown4be44fc2005-08-05 00:44:28 -04001002
Linus Torvalds1da177e2005-04-16 15:20:36 -07001003#ifdef ACPI_FUTURE_USAGE
Len Brown4be44fc2005-08-05 00:44:28 -04001004u32 acpi_os_get_line(char *buffer)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001005{
1006
1007#ifdef ENABLE_DEBUGGER
1008 if (acpi_in_debugger) {
1009 u32 chars;
1010
1011 kdb_read(buffer, sizeof(line_buf));
1012
1013 /* remove the CR kdb includes */
1014 chars = strlen(buffer) - 1;
1015 buffer[chars] = '\0';
1016 }
1017#endif
1018
1019 return 0;
1020}
Len Brown4be44fc2005-08-05 00:44:28 -04001021#endif /* ACPI_FUTURE_USAGE */
Linus Torvalds1da177e2005-04-16 15:20:36 -07001022
Len Brown4be44fc2005-08-05 00:44:28 -04001023acpi_status acpi_os_signal(u32 function, void *info)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001024{
Len Brown4be44fc2005-08-05 00:44:28 -04001025 switch (function) {
Linus Torvalds1da177e2005-04-16 15:20:36 -07001026 case ACPI_SIGNAL_FATAL:
1027 printk(KERN_ERR PREFIX "Fatal opcode executed\n");
1028 break;
1029 case ACPI_SIGNAL_BREAKPOINT:
1030 /*
1031 * AML Breakpoint
1032 * ACPI spec. says to treat it as a NOP unless
1033 * you are debugging. So if/when we integrate
1034 * AML debugger into the kernel debugger its
1035 * hook will go here. But until then it is
1036 * not useful to print anything on breakpoints.
1037 */
1038 break;
1039 default:
1040 break;
1041 }
1042
1043 return AE_OK;
1044}
Len Brown4be44fc2005-08-05 00:44:28 -04001045
Len Brown4be44fc2005-08-05 00:44:28 -04001046static int __init acpi_os_name_setup(char *str)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001047{
1048 char *p = acpi_os_name;
Len Brown4be44fc2005-08-05 00:44:28 -04001049 int count = ACPI_MAX_OVERRIDE_LEN - 1;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001050
1051 if (!str || !*str)
1052 return 0;
1053
1054 for (; count-- && str && *str; str++) {
1055 if (isalnum(*str) || *str == ' ' || *str == ':')
1056 *p++ = *str;
1057 else if (*str == '\'' || *str == '"')
1058 continue;
1059 else
1060 break;
1061 }
1062 *p = 0;
1063
1064 return 1;
Len Brown4be44fc2005-08-05 00:44:28 -04001065
Linus Torvalds1da177e2005-04-16 15:20:36 -07001066}
1067
1068__setup("acpi_os_name=", acpi_os_name_setup);
1069
Lin Ming12d32062010-12-09 16:51:06 +08001070#define OSI_STRING_LENGTH_MAX 64 /* arbitrary */
1071#define OSI_STRING_ENTRIES_MAX 16 /* arbitrary */
1072
1073struct osi_setup_entry {
1074 char string[OSI_STRING_LENGTH_MAX];
1075 bool enable;
1076};
1077
1078static struct osi_setup_entry __initdata osi_setup_entries[OSI_STRING_ENTRIES_MAX];
1079
Lin Mingd90aa922010-12-09 16:50:52 +08001080void __init acpi_osi_setup(char *str)
1081{
Lin Ming12d32062010-12-09 16:51:06 +08001082 struct osi_setup_entry *osi;
1083 bool enable = true;
1084 int i;
1085
Lin Mingd90aa922010-12-09 16:50:52 +08001086 if (!acpi_gbl_create_osi_method)
1087 return;
1088
1089 if (str == NULL || *str == '\0') {
1090 printk(KERN_INFO PREFIX "_OSI method disabled\n");
1091 acpi_gbl_create_osi_method = FALSE;
Lin Ming12d32062010-12-09 16:51:06 +08001092 return;
1093 }
1094
1095 if (*str == '!') {
1096 str++;
1097 enable = false;
1098 }
1099
1100 for (i = 0; i < OSI_STRING_ENTRIES_MAX; i++) {
1101 osi = &osi_setup_entries[i];
1102 if (!strcmp(osi->string, str)) {
1103 osi->enable = enable;
1104 break;
1105 } else if (osi->string[0] == '\0') {
1106 osi->enable = enable;
1107 strncpy(osi->string, str, OSI_STRING_LENGTH_MAX);
1108 break;
1109 }
1110 }
Lin Mingd90aa922010-12-09 16:50:52 +08001111}
1112
Len Brownd4b7dc42008-01-23 20:50:56 -05001113static void __init set_osi_linux(unsigned int enable)
1114{
Lin Mingd90aa922010-12-09 16:50:52 +08001115 if (osi_linux.enable != enable)
Len Brownd4b7dc42008-01-23 20:50:56 -05001116 osi_linux.enable = enable;
Lin Mingb0ed7a92010-08-06 09:35:51 +08001117
1118 if (osi_linux.enable)
1119 acpi_osi_setup("Linux");
1120 else
1121 acpi_osi_setup("!Linux");
1122
Len Brownd4b7dc42008-01-23 20:50:56 -05001123 return;
1124}
Len Brownf5076542007-05-30 00:10:38 -04001125
Len Brownd4b7dc42008-01-23 20:50:56 -05001126static void __init acpi_cmdline_osi_linux(unsigned int enable)
1127{
Lin Mingd90aa922010-12-09 16:50:52 +08001128 osi_linux.cmdline = 1; /* cmdline set the default and override DMI */
1129 osi_linux.dmi = 0;
Len Brownd4b7dc42008-01-23 20:50:56 -05001130 set_osi_linux(enable);
Len Brownf5076542007-05-30 00:10:38 -04001131
Len Brownd4b7dc42008-01-23 20:50:56 -05001132 return;
1133}
1134
1135void __init acpi_dmi_osi_linux(int enable, const struct dmi_system_id *d)
1136{
Len Brownd4b7dc42008-01-23 20:50:56 -05001137 printk(KERN_NOTICE PREFIX "DMI detected: %s\n", d->ident);
1138
1139 if (enable == -1)
1140 return;
1141
Lin Mingd90aa922010-12-09 16:50:52 +08001142 osi_linux.dmi = 1; /* DMI knows that this box asks OSI(Linux) */
Len Brownd4b7dc42008-01-23 20:50:56 -05001143 set_osi_linux(enable);
1144
Len Brownf5076542007-05-30 00:10:38 -04001145 return;
1146}
1147
Linus Torvalds1da177e2005-04-16 15:20:36 -07001148/*
Len Brownae00d812007-05-29 18:43:33 -04001149 * Modify the list of "OS Interfaces" reported to BIOS via _OSI
1150 *
Linus Torvalds1da177e2005-04-16 15:20:36 -07001151 * empty string disables _OSI
Len Brownae00d812007-05-29 18:43:33 -04001152 * string starting with '!' disables that string
1153 * otherwise string is added to list, augmenting built-in strings
Linus Torvalds1da177e2005-04-16 15:20:36 -07001154 */
Lin Mingb0ed7a92010-08-06 09:35:51 +08001155static void __init acpi_osi_setup_late(void)
1156{
Lin Ming12d32062010-12-09 16:51:06 +08001157 struct osi_setup_entry *osi;
1158 char *str;
1159 int i;
Lin Mingd90aa922010-12-09 16:50:52 +08001160 acpi_status status;
Lin Mingb0ed7a92010-08-06 09:35:51 +08001161
Lin Ming12d32062010-12-09 16:51:06 +08001162 for (i = 0; i < OSI_STRING_ENTRIES_MAX; i++) {
1163 osi = &osi_setup_entries[i];
1164 str = osi->string;
Lin Mingb0ed7a92010-08-06 09:35:51 +08001165
Lin Ming12d32062010-12-09 16:51:06 +08001166 if (*str == '\0')
1167 break;
1168 if (osi->enable) {
1169 status = acpi_install_interface(str);
Lin Mingd90aa922010-12-09 16:50:52 +08001170
Lin Ming12d32062010-12-09 16:51:06 +08001171 if (ACPI_SUCCESS(status))
1172 printk(KERN_INFO PREFIX "Added _OSI(%s)\n", str);
1173 } else {
1174 status = acpi_remove_interface(str);
Lin Mingd90aa922010-12-09 16:50:52 +08001175
Lin Ming12d32062010-12-09 16:51:06 +08001176 if (ACPI_SUCCESS(status))
1177 printk(KERN_INFO PREFIX "Deleted _OSI(%s)\n", str);
1178 }
Lin Mingb0ed7a92010-08-06 09:35:51 +08001179 }
1180}
1181
Lin Mingd90aa922010-12-09 16:50:52 +08001182static int __init osi_setup(char *str)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001183{
Lin Mingd90aa922010-12-09 16:50:52 +08001184 if (str && !strcmp("Linux", str))
1185 acpi_cmdline_osi_linux(1);
1186 else if (str && !strcmp("!Linux", str))
1187 acpi_cmdline_osi_linux(0);
1188 else
1189 acpi_osi_setup(str);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001190
1191 return 1;
1192}
1193
Lin Mingd90aa922010-12-09 16:50:52 +08001194__setup("acpi_osi=", osi_setup);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001195
1196/* enable serialization to combat AE_ALREADY_EXISTS errors */
Len Brown4be44fc2005-08-05 00:44:28 -04001197static int __init acpi_serialize_setup(char *str)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001198{
1199 printk(KERN_INFO PREFIX "serialize enabled\n");
1200
1201 acpi_gbl_all_methods_serialized = TRUE;
1202
1203 return 1;
1204}
1205
1206__setup("acpi_serialize", acpi_serialize_setup);
1207
Thomas Renningerdf92e692008-02-04 23:31:22 -08001208/* Check of resource interference between native drivers and ACPI
1209 * OperationRegions (SystemIO and System Memory only).
1210 * IO ports and memory declared in ACPI might be used by the ACPI subsystem
1211 * in arbitrary AML code and can interfere with legacy drivers.
1212 * acpi_enforce_resources= can be set to:
1213 *
Luca Tettamanti7e905602009-03-30 00:01:27 +02001214 * - strict (default) (2)
Thomas Renningerdf92e692008-02-04 23:31:22 -08001215 * -> further driver trying to access the resources will not load
Luca Tettamanti7e905602009-03-30 00:01:27 +02001216 * - lax (1)
Thomas Renningerdf92e692008-02-04 23:31:22 -08001217 * -> further driver trying to access the resources will load, but you
1218 * get a system message that something might go wrong...
1219 *
1220 * - no (0)
1221 * -> ACPI Operation Region resources will not be registered
1222 *
1223 */
1224#define ENFORCE_RESOURCES_STRICT 2
1225#define ENFORCE_RESOURCES_LAX 1
1226#define ENFORCE_RESOURCES_NO 0
1227
Luca Tettamanti7e905602009-03-30 00:01:27 +02001228static unsigned int acpi_enforce_resources = ENFORCE_RESOURCES_STRICT;
Thomas Renningerdf92e692008-02-04 23:31:22 -08001229
1230static int __init acpi_enforce_resources_setup(char *str)
1231{
1232 if (str == NULL || *str == '\0')
1233 return 0;
1234
1235 if (!strcmp("strict", str))
1236 acpi_enforce_resources = ENFORCE_RESOURCES_STRICT;
1237 else if (!strcmp("lax", str))
1238 acpi_enforce_resources = ENFORCE_RESOURCES_LAX;
1239 else if (!strcmp("no", str))
1240 acpi_enforce_resources = ENFORCE_RESOURCES_NO;
1241
1242 return 1;
1243}
1244
1245__setup("acpi_enforce_resources=", acpi_enforce_resources_setup);
1246
1247/* Check for resource conflicts between ACPI OperationRegions and native
1248 * drivers */
Jean Delvare876fba42009-11-11 15:22:15 +01001249int acpi_check_resource_conflict(const struct resource *res)
Thomas Renningerdf92e692008-02-04 23:31:22 -08001250{
1251 struct acpi_res_list *res_list_elem;
Thomas Renninger106d1a02010-12-20 12:11:45 +01001252 int ioport = 0, clash = 0;
Thomas Renningerdf92e692008-02-04 23:31:22 -08001253
1254 if (acpi_enforce_resources == ENFORCE_RESOURCES_NO)
1255 return 0;
1256 if (!(res->flags & IORESOURCE_IO) && !(res->flags & IORESOURCE_MEM))
1257 return 0;
1258
1259 ioport = res->flags & IORESOURCE_IO;
1260
1261 spin_lock(&acpi_res_lock);
1262 list_for_each_entry(res_list_elem, &resource_list_head,
1263 resource_list) {
1264 if (ioport && (res_list_elem->resource_type
1265 != ACPI_ADR_SPACE_SYSTEM_IO))
1266 continue;
1267 if (!ioport && (res_list_elem->resource_type
1268 != ACPI_ADR_SPACE_SYSTEM_MEMORY))
1269 continue;
1270
1271 if (res->end < res_list_elem->start
1272 || res_list_elem->end < res->start)
1273 continue;
1274 clash = 1;
1275 break;
1276 }
1277 spin_unlock(&acpi_res_lock);
1278
1279 if (clash) {
1280 if (acpi_enforce_resources != ENFORCE_RESOURCES_NO) {
Chase Douglas1638bca2010-03-22 15:08:09 -04001281 printk(KERN_WARNING "ACPI: resource %s %pR"
Thomas Renninger106d1a02010-12-20 12:11:45 +01001282 " conflicts with ACPI region %s "
1283 "[%s 0x%zx-0x%zx]\n",
Chase Douglas1638bca2010-03-22 15:08:09 -04001284 res->name, res, res_list_elem->name,
Thomas Renninger106d1a02010-12-20 12:11:45 +01001285 (res_list_elem->resource_type ==
1286 ACPI_ADR_SPACE_SYSTEM_IO) ? "io" : "mem",
1287 (size_t) res_list_elem->start,
1288 (size_t) res_list_elem->end);
Jean Delvare14f03342009-09-08 15:31:46 +02001289 if (acpi_enforce_resources == ENFORCE_RESOURCES_LAX)
1290 printk(KERN_NOTICE "ACPI: This conflict may"
1291 " cause random problems and system"
1292 " instability\n");
1293 printk(KERN_INFO "ACPI: If an ACPI driver is available"
1294 " for this device, you should use it instead of"
1295 " the native driver\n");
Thomas Renningerdf92e692008-02-04 23:31:22 -08001296 }
1297 if (acpi_enforce_resources == ENFORCE_RESOURCES_STRICT)
1298 return -EBUSY;
1299 }
1300 return 0;
1301}
Thomas Renninger443dea72008-02-04 23:31:23 -08001302EXPORT_SYMBOL(acpi_check_resource_conflict);
Thomas Renningerdf92e692008-02-04 23:31:22 -08001303
1304int acpi_check_region(resource_size_t start, resource_size_t n,
1305 const char *name)
1306{
1307 struct resource res = {
1308 .start = start,
1309 .end = start + n - 1,
1310 .name = name,
1311 .flags = IORESOURCE_IO,
1312 };
1313
1314 return acpi_check_resource_conflict(&res);
1315}
1316EXPORT_SYMBOL(acpi_check_region);
1317
Linus Torvalds1da177e2005-04-16 15:20:36 -07001318/*
Jean Delvare70dd6be2010-05-27 19:58:37 +02001319 * Let drivers know whether the resource checks are effective
1320 */
1321int acpi_resources_are_enforced(void)
1322{
1323 return acpi_enforce_resources == ENFORCE_RESOURCES_STRICT;
1324}
1325EXPORT_SYMBOL(acpi_resources_are_enforced);
1326
1327/*
Robert Moore73459f72005-06-24 00:00:00 -04001328 * Acquire a spinlock.
1329 *
1330 * handle is a pointer to the spinlock_t.
Robert Moore73459f72005-06-24 00:00:00 -04001331 */
1332
Bob Moore967440e2006-06-23 17:04:00 -04001333acpi_cpu_flags acpi_os_acquire_lock(acpi_spinlock lockp)
Robert Moore73459f72005-06-24 00:00:00 -04001334{
Bob Mooreb8e4d892006-01-27 16:43:00 -05001335 acpi_cpu_flags flags;
Bob Moore967440e2006-06-23 17:04:00 -04001336 spin_lock_irqsave(lockp, flags);
Robert Moore73459f72005-06-24 00:00:00 -04001337 return flags;
1338}
1339
1340/*
1341 * Release a spinlock. See above.
1342 */
1343
Bob Moore967440e2006-06-23 17:04:00 -04001344void acpi_os_release_lock(acpi_spinlock lockp, acpi_cpu_flags flags)
Robert Moore73459f72005-06-24 00:00:00 -04001345{
Bob Moore967440e2006-06-23 17:04:00 -04001346 spin_unlock_irqrestore(lockp, flags);
Robert Moore73459f72005-06-24 00:00:00 -04001347}
1348
Robert Moore73459f72005-06-24 00:00:00 -04001349#ifndef ACPI_USE_LOCAL_CACHE
1350
1351/*******************************************************************************
1352 *
1353 * FUNCTION: acpi_os_create_cache
1354 *
Bob Mooreb229cf92006-04-21 17:15:00 -04001355 * PARAMETERS: name - Ascii name for the cache
1356 * size - Size of each cached object
1357 * depth - Maximum depth of the cache (in objects) <ignored>
1358 * cache - Where the new cache object is returned
Robert Moore73459f72005-06-24 00:00:00 -04001359 *
Bob Mooreb229cf92006-04-21 17:15:00 -04001360 * RETURN: status
Robert Moore73459f72005-06-24 00:00:00 -04001361 *
1362 * DESCRIPTION: Create a cache object
1363 *
1364 ******************************************************************************/
1365
1366acpi_status
Len Brown4be44fc2005-08-05 00:44:28 -04001367acpi_os_create_cache(char *name, u16 size, u16 depth, acpi_cache_t ** cache)
Robert Moore73459f72005-06-24 00:00:00 -04001368{
Paul Mundt20c2df82007-07-20 10:11:58 +09001369 *cache = kmem_cache_create(name, size, 0, 0, NULL);
Adrian Bunka6fdbf92006-12-19 12:56:13 -08001370 if (*cache == NULL)
Bob Mooreb229cf92006-04-21 17:15:00 -04001371 return AE_ERROR;
1372 else
1373 return AE_OK;
Robert Moore73459f72005-06-24 00:00:00 -04001374}
1375
1376/*******************************************************************************
1377 *
1378 * FUNCTION: acpi_os_purge_cache
1379 *
1380 * PARAMETERS: Cache - Handle to cache object
1381 *
1382 * RETURN: Status
1383 *
1384 * DESCRIPTION: Free all objects within the requested cache.
1385 *
1386 ******************************************************************************/
1387
Len Brown4be44fc2005-08-05 00:44:28 -04001388acpi_status acpi_os_purge_cache(acpi_cache_t * cache)
Robert Moore73459f72005-06-24 00:00:00 -04001389{
Jan Engelhardt50dd0962006-10-01 00:28:50 +02001390 kmem_cache_shrink(cache);
Len Brown4be44fc2005-08-05 00:44:28 -04001391 return (AE_OK);
Robert Moore73459f72005-06-24 00:00:00 -04001392}
1393
1394/*******************************************************************************
1395 *
1396 * FUNCTION: acpi_os_delete_cache
1397 *
1398 * PARAMETERS: Cache - Handle to cache object
1399 *
1400 * RETURN: Status
1401 *
1402 * DESCRIPTION: Free all objects within the requested cache and delete the
1403 * cache object.
1404 *
1405 ******************************************************************************/
1406
Len Brown4be44fc2005-08-05 00:44:28 -04001407acpi_status acpi_os_delete_cache(acpi_cache_t * cache)
Robert Moore73459f72005-06-24 00:00:00 -04001408{
Alexey Dobriyan1a1d92c2006-09-27 01:49:40 -07001409 kmem_cache_destroy(cache);
Len Brown4be44fc2005-08-05 00:44:28 -04001410 return (AE_OK);
Robert Moore73459f72005-06-24 00:00:00 -04001411}
1412
1413/*******************************************************************************
1414 *
1415 * FUNCTION: acpi_os_release_object
1416 *
1417 * PARAMETERS: Cache - Handle to cache object
1418 * Object - The object to be released
1419 *
1420 * RETURN: None
1421 *
1422 * DESCRIPTION: Release an object to the specified cache. If cache is full,
1423 * the object is deleted.
1424 *
1425 ******************************************************************************/
1426
Len Brown4be44fc2005-08-05 00:44:28 -04001427acpi_status acpi_os_release_object(acpi_cache_t * cache, void *object)
Robert Moore73459f72005-06-24 00:00:00 -04001428{
Len Brown4be44fc2005-08-05 00:44:28 -04001429 kmem_cache_free(cache, object);
1430 return (AE_OK);
Robert Moore73459f72005-06-24 00:00:00 -04001431}
1432
Lin Minga5fe1a02009-08-13 10:43:27 +08001433static inline int acpi_res_list_add(struct acpi_res_list *res)
1434{
1435 struct acpi_res_list *res_list_elem;
1436
1437 list_for_each_entry(res_list_elem, &resource_list_head,
1438 resource_list) {
1439
1440 if (res->resource_type == res_list_elem->resource_type &&
1441 res->start == res_list_elem->start &&
1442 res->end == res_list_elem->end) {
1443
1444 /*
1445 * The Region(addr,len) already exist in the list,
1446 * just increase the count
1447 */
1448
1449 res_list_elem->count++;
1450 return 0;
1451 }
1452 }
1453
1454 res->count = 1;
1455 list_add(&res->resource_list, &resource_list_head);
1456 return 1;
1457}
1458
1459static inline void acpi_res_list_del(struct acpi_res_list *res)
1460{
1461 struct acpi_res_list *res_list_elem;
1462
1463 list_for_each_entry(res_list_elem, &resource_list_head,
1464 resource_list) {
1465
1466 if (res->resource_type == res_list_elem->resource_type &&
1467 res->start == res_list_elem->start &&
1468 res->end == res_list_elem->end) {
1469
1470 /*
1471 * If the res count is decreased to 0,
1472 * remove and free it
1473 */
1474
1475 if (--res_list_elem->count == 0) {
1476 list_del(&res_list_elem->resource_list);
1477 kfree(res_list_elem);
1478 }
1479 return;
1480 }
1481 }
1482}
1483
1484acpi_status
1485acpi_os_invalidate_address(
1486 u8 space_id,
1487 acpi_physical_address address,
1488 acpi_size length)
1489{
1490 struct acpi_res_list res;
1491
1492 switch (space_id) {
1493 case ACPI_ADR_SPACE_SYSTEM_IO:
1494 case ACPI_ADR_SPACE_SYSTEM_MEMORY:
Thomas Weber88393162010-03-16 11:47:56 +01001495 /* Only interference checks against SystemIO and SystemMemory
Lin Minga5fe1a02009-08-13 10:43:27 +08001496 are needed */
1497 res.start = address;
1498 res.end = address + length - 1;
1499 res.resource_type = space_id;
1500 spin_lock(&acpi_res_lock);
1501 acpi_res_list_del(&res);
1502 spin_unlock(&acpi_res_lock);
1503 break;
1504 case ACPI_ADR_SPACE_PCI_CONFIG:
1505 case ACPI_ADR_SPACE_EC:
1506 case ACPI_ADR_SPACE_SMBUS:
1507 case ACPI_ADR_SPACE_CMOS:
1508 case ACPI_ADR_SPACE_PCI_BAR_TARGET:
1509 case ACPI_ADR_SPACE_DATA_TABLE:
1510 case ACPI_ADR_SPACE_FIXED_HARDWARE:
1511 break;
1512 }
1513 return AE_OK;
1514}
1515
Bob Mooreb229cf92006-04-21 17:15:00 -04001516/******************************************************************************
1517 *
1518 * FUNCTION: acpi_os_validate_address
1519 *
1520 * PARAMETERS: space_id - ACPI space ID
1521 * address - Physical address
1522 * length - Address length
1523 *
1524 * RETURN: AE_OK if address/length is valid for the space_id. Otherwise,
1525 * should return AE_AML_ILLEGAL_ADDRESS.
1526 *
1527 * DESCRIPTION: Validate a system address via the host OS. Used to validate
1528 * the addresses accessed by AML operation regions.
1529 *
1530 *****************************************************************************/
1531
1532acpi_status
1533acpi_os_validate_address (
1534 u8 space_id,
1535 acpi_physical_address address,
Thomas Renningerdf92e692008-02-04 23:31:22 -08001536 acpi_size length,
1537 char *name)
Bob Mooreb229cf92006-04-21 17:15:00 -04001538{
Thomas Renningerdf92e692008-02-04 23:31:22 -08001539 struct acpi_res_list *res;
Lin Minga5fe1a02009-08-13 10:43:27 +08001540 int added;
Thomas Renningerdf92e692008-02-04 23:31:22 -08001541 if (acpi_enforce_resources == ENFORCE_RESOURCES_NO)
1542 return AE_OK;
Bob Mooreb229cf92006-04-21 17:15:00 -04001543
Thomas Renningerdf92e692008-02-04 23:31:22 -08001544 switch (space_id) {
1545 case ACPI_ADR_SPACE_SYSTEM_IO:
1546 case ACPI_ADR_SPACE_SYSTEM_MEMORY:
Thomas Weber88393162010-03-16 11:47:56 +01001547 /* Only interference checks against SystemIO and SystemMemory
Thomas Renningerdf92e692008-02-04 23:31:22 -08001548 are needed */
1549 res = kzalloc(sizeof(struct acpi_res_list), GFP_KERNEL);
1550 if (!res)
1551 return AE_OK;
1552 /* ACPI names are fixed to 4 bytes, still better use strlcpy */
1553 strlcpy(res->name, name, 5);
1554 res->start = address;
1555 res->end = address + length - 1;
1556 res->resource_type = space_id;
1557 spin_lock(&acpi_res_lock);
Lin Minga5fe1a02009-08-13 10:43:27 +08001558 added = acpi_res_list_add(res);
Thomas Renningerdf92e692008-02-04 23:31:22 -08001559 spin_unlock(&acpi_res_lock);
Lin Minga5fe1a02009-08-13 10:43:27 +08001560 pr_debug("%s %s resource: start: 0x%llx, end: 0x%llx, "
1561 "name: %s\n", added ? "Added" : "Already exist",
1562 (space_id == ACPI_ADR_SPACE_SYSTEM_IO)
Thomas Renningerdf92e692008-02-04 23:31:22 -08001563 ? "SystemIO" : "System Memory",
1564 (unsigned long long)res->start,
1565 (unsigned long long)res->end,
1566 res->name);
Lin Minga5fe1a02009-08-13 10:43:27 +08001567 if (!added)
1568 kfree(res);
Thomas Renningerdf92e692008-02-04 23:31:22 -08001569 break;
1570 case ACPI_ADR_SPACE_PCI_CONFIG:
1571 case ACPI_ADR_SPACE_EC:
1572 case ACPI_ADR_SPACE_SMBUS:
1573 case ACPI_ADR_SPACE_CMOS:
1574 case ACPI_ADR_SPACE_PCI_BAR_TARGET:
1575 case ACPI_ADR_SPACE_DATA_TABLE:
1576 case ACPI_ADR_SPACE_FIXED_HARDWARE:
1577 break;
1578 }
1579 return AE_OK;
Bob Mooreb229cf92006-04-21 17:15:00 -04001580}
Robert Moore73459f72005-06-24 00:00:00 -04001581#endif
Myron Stowed362eda2010-10-21 14:24:04 -06001582
1583acpi_status __init acpi_os_initialize(void)
1584{
1585 acpi_os_map_generic_address(&acpi_gbl_FADT.xpm1a_event_block);
1586 acpi_os_map_generic_address(&acpi_gbl_FADT.xpm1b_event_block);
1587 acpi_os_map_generic_address(&acpi_gbl_FADT.xgpe0_block);
1588 acpi_os_map_generic_address(&acpi_gbl_FADT.xgpe1_block);
1589
1590 return AE_OK;
1591}
1592
Zhang Rui32d47ee2010-12-08 10:40:36 +08001593acpi_status __init acpi_os_initialize1(void)
Myron Stowed362eda2010-10-21 14:24:04 -06001594{
1595 kacpid_wq = create_workqueue("kacpid");
1596 kacpi_notify_wq = create_workqueue("kacpi_notify");
1597 kacpi_hotplug_wq = create_workqueue("kacpi_hotplug");
1598 BUG_ON(!kacpid_wq);
1599 BUG_ON(!kacpi_notify_wq);
1600 BUG_ON(!kacpi_hotplug_wq);
Len Brown1bd64d42010-10-26 14:50:56 -04001601 acpi_install_interface_handler(acpi_osi_handler);
1602 acpi_osi_setup_late();
Myron Stowed362eda2010-10-21 14:24:04 -06001603 return AE_OK;
1604}
1605
1606acpi_status acpi_os_terminate(void)
1607{
1608 if (acpi_irq_handler) {
Rafael J. Wysocki23fe3632011-02-08 23:48:16 +01001609 acpi_os_remove_interrupt_handler(acpi_gbl_FADT.sci_interrupt,
Myron Stowed362eda2010-10-21 14:24:04 -06001610 acpi_irq_handler);
1611 }
1612
1613 acpi_os_unmap_generic_address(&acpi_gbl_FADT.xgpe1_block);
1614 acpi_os_unmap_generic_address(&acpi_gbl_FADT.xgpe0_block);
1615 acpi_os_unmap_generic_address(&acpi_gbl_FADT.xpm1b_event_block);
1616 acpi_os_unmap_generic_address(&acpi_gbl_FADT.xpm1a_event_block);
1617
1618 destroy_workqueue(kacpid_wq);
1619 destroy_workqueue(kacpi_notify_wq);
1620 destroy_workqueue(kacpi_hotplug_wq);
1621
1622 return AE_OK;
1623}