osl.c

来自「linux 内核源代码」· C语言 代码 · 共 1,313 行 · 第 1/3 页

C
1,313
字号
/* *  acpi_osl.c - OS-dependent functions ($Revision: 83 $) * *  Copyright (C) 2000       Andrew Henroid *  Copyright (C) 2001, 2002 Andy Grover <andrew.grover@intel.com> *  Copyright (C) 2001, 2002 Paul Diefenbaugh <paul.s.diefenbaugh@intel.com> * * ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ * *  This program is free software; you can redistribute it and/or modify *  it under the terms of the GNU General Public License as published by *  the Free Software Foundation; either version 2 of the License, or *  (at your option) any later version. * *  This program is distributed in the hope that it will be useful, *  but WITHOUT ANY WARRANTY; without even the implied warranty of *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the *  GNU General Public License for more details. * *  You should have received a copy of the GNU General Public License *  along with this program; if not, write to the Free Software *  Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA * * ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ * */#include <linux/module.h>#include <linux/kernel.h>#include <linux/slab.h>#include <linux/mm.h>#include <linux/pci.h>#include <linux/interrupt.h>#include <linux/kmod.h>#include <linux/delay.h>#include <linux/dmi.h>#include <linux/workqueue.h>#include <linux/nmi.h>#include <linux/acpi.h>#include <acpi/acpi.h>#include <asm/io.h>#include <acpi/acpi_bus.h>#include <acpi/processor.h>#include <asm/uaccess.h>#include <linux/efi.h>#define _COMPONENT		ACPI_OS_SERVICESACPI_MODULE_NAME("osl");#define PREFIX		"ACPI: "struct acpi_os_dpc {	acpi_osd_exec_callback function;	void *context;	struct work_struct work;};#ifdef CONFIG_ACPI_CUSTOM_DSDT#include CONFIG_ACPI_CUSTOM_DSDT_FILE#endif#ifdef ENABLE_DEBUGGER#include <linux/kdb.h>/* stuff for debugger support */int acpi_in_debugger;EXPORT_SYMBOL(acpi_in_debugger);extern char line_buf[80];#endif				/*ENABLE_DEBUGGER */static unsigned int acpi_irq_irq;static acpi_osd_handler acpi_irq_handler;static void *acpi_irq_context;static struct workqueue_struct *kacpid_wq;static struct workqueue_struct *kacpi_notify_wq;#define	OSI_STRING_LENGTH_MAX 64	/* arbitrary */static char osi_additional_string[OSI_STRING_LENGTH_MAX];/* * "Ode to _OSI(Linux)" * * osi_linux -- Control response to BIOS _OSI(Linux) query. * * As Linux evolves, the features that it supports change. * So an OSI string such as "Linux" is not specific enough * to be useful across multiple versions of Linux.  It * doesn't identify any particular feature, interface, * or even any particular version of Linux... * * Unfortunately, Linux-2.6.22 and earlier responded "yes" * to a BIOS _OSI(Linux) query.  When * a reference mobile BIOS started using it, its use * started to spread to many vendor platforms. * As it is not supportable, we need to halt that spread. * * Today, most BIOS references to _OSI(Linux) are noise -- * they have no functional effect and are just dead code * carried over from the reference BIOS. * * The next most common case is that _OSI(Linux) harms Linux, * usually by causing the BIOS to follow paths that are * not tested during Windows validation. * * Finally, there is a short list of platforms * where OSI(Linux) benefits Linux. * * In Linux-2.6.23, OSI(Linux) is first disabled by default. * DMI is used to disable the dmesg warning about OSI(Linux) * on platforms where it is known to have no effect. * But a dmesg warning remains for systems where * we do not know if OSI(Linux) is good or bad for the system. * DMI is also used to enable OSI(Linux) for the machines * that are known to need it. * * BIOS writers should NOT query _OSI(Linux) on future systems. * It will be ignored by default, and to get Linux to * not ignore it will require a kernel source update to * add a DMI entry, or a boot-time "acpi_osi=Linux" invocation. */#define OSI_LINUX_ENABLE 0static struct osi_linux {	unsigned int	enable:1;	unsigned int	dmi:1;	unsigned int	cmdline:1;	unsigned int	known:1;} osi_linux = { OSI_LINUX_ENABLE, 0, 0, 0};static void __init acpi_request_region (struct acpi_generic_address *addr,	unsigned int length, char *desc){	struct resource *res;	if (!addr->address || !length)		return;	if (addr->space_id == ACPI_ADR_SPACE_SYSTEM_IO)		res = request_region(addr->address, length, desc);	else if (addr->space_id == ACPI_ADR_SPACE_SYSTEM_MEMORY)		res = request_mem_region(addr->address, length, desc);}static int __init acpi_reserve_resources(void){	acpi_request_region(&acpi_gbl_FADT.xpm1a_event_block, acpi_gbl_FADT.pm1_event_length,		"ACPI PM1a_EVT_BLK");	acpi_request_region(&acpi_gbl_FADT.xpm1b_event_block, acpi_gbl_FADT.pm1_event_length,		"ACPI PM1b_EVT_BLK");	acpi_request_region(&acpi_gbl_FADT.xpm1a_control_block, acpi_gbl_FADT.pm1_control_length,		"ACPI PM1a_CNT_BLK");	acpi_request_region(&acpi_gbl_FADT.xpm1b_control_block, acpi_gbl_FADT.pm1_control_length,		"ACPI PM1b_CNT_BLK");	if (acpi_gbl_FADT.pm_timer_length == 4)		acpi_request_region(&acpi_gbl_FADT.xpm_timer_block, 4, "ACPI PM_TMR");	acpi_request_region(&acpi_gbl_FADT.xpm2_control_block, acpi_gbl_FADT.pm2_control_length,		"ACPI PM2_CNT_BLK");	/* Length of GPE blocks must be a non-negative multiple of 2 */	if (!(acpi_gbl_FADT.gpe0_block_length & 0x1))		acpi_request_region(&acpi_gbl_FADT.xgpe0_block,			       acpi_gbl_FADT.gpe0_block_length, "ACPI GPE0_BLK");	if (!(acpi_gbl_FADT.gpe1_block_length & 0x1))		acpi_request_region(&acpi_gbl_FADT.xgpe1_block,			       acpi_gbl_FADT.gpe1_block_length, "ACPI GPE1_BLK");	return 0;}device_initcall(acpi_reserve_resources);acpi_status __init acpi_os_initialize(void){	return AE_OK;}acpi_status acpi_os_initialize1(void){	/*	 * Initialize PCI configuration space access, as we'll need to access	 * it while walking the namespace (bus 0 and root bridges w/ _BBNs).	 */	if (!raw_pci_ops) {		printk(KERN_ERR PREFIX		       "Access to PCI configuration space unavailable\n");		return AE_NULL_ENTRY;	}	kacpid_wq = create_singlethread_workqueue("kacpid");	kacpi_notify_wq = create_singlethread_workqueue("kacpi_notify");	BUG_ON(!kacpid_wq);	BUG_ON(!kacpi_notify_wq);	return AE_OK;}acpi_status acpi_os_terminate(void){	if (acpi_irq_handler) {		acpi_os_remove_interrupt_handler(acpi_irq_irq,						 acpi_irq_handler);	}	destroy_workqueue(kacpid_wq);	destroy_workqueue(kacpi_notify_wq);	return AE_OK;}void acpi_os_printf(const char *fmt, ...){	va_list args;	va_start(args, fmt);	acpi_os_vprintf(fmt, args);	va_end(args);}EXPORT_SYMBOL(acpi_os_printf);void acpi_os_vprintf(const char *fmt, va_list args){	static char buffer[512];	vsprintf(buffer, fmt, args);#ifdef ENABLE_DEBUGGER	if (acpi_in_debugger) {		kdb_printf("%s", buffer);	} else {		printk("%s", buffer);	}#else	printk("%s", buffer);#endif}acpi_physical_address __init acpi_os_get_root_pointer(void){	if (efi_enabled) {		if (efi.acpi20 != EFI_INVALID_TABLE_ADDR)			return efi.acpi20;		else if (efi.acpi != EFI_INVALID_TABLE_ADDR)			return efi.acpi;		else {			printk(KERN_ERR PREFIX			       "System description tables not found\n");			return 0;		}	} else		return acpi_find_rsdp();}void __iomem *acpi_os_map_memory(acpi_physical_address phys, acpi_size size){	if (phys > ULONG_MAX) {		printk(KERN_ERR PREFIX "Cannot map memory that high\n");		return NULL;	}	if (acpi_gbl_permanent_mmap)		/*		* ioremap checks to ensure this is in reserved space		*/		return ioremap((unsigned long)phys, size);	else		return __acpi_map_table((unsigned long)phys, size);}EXPORT_SYMBOL_GPL(acpi_os_map_memory);void acpi_os_unmap_memory(void __iomem * virt, acpi_size size){	if (acpi_gbl_permanent_mmap) {		iounmap(virt);	}}EXPORT_SYMBOL_GPL(acpi_os_unmap_memory);#ifdef ACPI_FUTURE_USAGEacpi_statusacpi_os_get_physical_address(void *virt, acpi_physical_address * phys){	if (!phys || !virt)		return AE_BAD_PARAMETER;	*phys = virt_to_phys(virt);	return AE_OK;}#endif#define ACPI_MAX_OVERRIDE_LEN 100static char acpi_os_name[ACPI_MAX_OVERRIDE_LEN];acpi_statusacpi_os_predefined_override(const struct acpi_predefined_names *init_val,			    acpi_string * new_val){	if (!init_val || !new_val)		return AE_BAD_PARAMETER;	*new_val = NULL;	if (!memcmp(init_val->name, "_OS_", 4) && strlen(acpi_os_name)) {		printk(KERN_INFO PREFIX "Overriding _OS definition to '%s'\n",		       acpi_os_name);		*new_val = acpi_os_name;	}	return AE_OK;}acpi_statusacpi_os_table_override(struct acpi_table_header * existing_table,		       struct acpi_table_header ** new_table){	if (!existing_table || !new_table)		return AE_BAD_PARAMETER;#ifdef CONFIG_ACPI_CUSTOM_DSDT	if (strncmp(existing_table->signature, "DSDT", 4) == 0)		*new_table = (struct acpi_table_header *)AmlCode;	else		*new_table = NULL;#else	*new_table = NULL;#endif	return AE_OK;}static irqreturn_t acpi_irq(int irq, void *dev_id){	return (*acpi_irq_handler) (acpi_irq_context) ? IRQ_HANDLED : IRQ_NONE;}acpi_statusacpi_os_install_interrupt_handler(u32 gsi, acpi_osd_handler handler,				  void *context){	unsigned int irq;	/*	 * Ignore the GSI from the core, and use the value in our copy of the	 * FADT. It may not be the same if an interrupt source override exists	 * for the SCI.	 */	gsi = acpi_gbl_FADT.sci_interrupt;	if (acpi_gsi_to_irq(gsi, &irq) < 0) {		printk(KERN_ERR PREFIX "SCI (ACPI GSI %d) not registered\n",		       gsi);		return AE_OK;	}	acpi_irq_handler = handler;	acpi_irq_context = context;	if (request_irq(irq, acpi_irq, IRQF_SHARED, "acpi", acpi_irq)) {		printk(KERN_ERR PREFIX "SCI (IRQ%d) allocation failed\n", irq);		return AE_NOT_ACQUIRED;	}	acpi_irq_irq = irq;	return AE_OK;}acpi_status acpi_os_remove_interrupt_handler(u32 irq, acpi_osd_handler handler){	if (irq) {		free_irq(irq, acpi_irq);		acpi_irq_handler = NULL;		acpi_irq_irq = 0;	}	return AE_OK;}/* * Running in interpreter thread context, safe to sleep */void acpi_os_sleep(acpi_integer ms){	schedule_timeout_interruptible(msecs_to_jiffies(ms));}EXPORT_SYMBOL(acpi_os_sleep);void acpi_os_stall(u32 us){	while (us) {		u32 delay = 1000;		if (delay > us)			delay = us;		udelay(delay);		touch_nmi_watchdog();		us -= delay;	}}EXPORT_SYMBOL(acpi_os_stall);/* * Support ACPI 3.0 AML Timer operand * Returns 64-bit free-running, monotonically increasing timer * with 100ns granularity */u64 acpi_os_get_timer(void){	static u64 t;#ifdef	CONFIG_HPET	/* TBD: use HPET if available */#endif#ifdef	CONFIG_X86_PM_TIMER	/* TBD: default to PM timer if HPET was not available */#endif	if (!t)		printk(KERN_ERR PREFIX "acpi_os_get_timer() TBD\n");	return ++t;}acpi_status acpi_os_read_port(acpi_io_address port, u32 * value, u32 width){	u32 dummy;	if (!value)		value = &dummy;	*value = 0;	if (width <= 8) {		*(u8 *) value = inb(port);	} else if (width <= 16) {		*(u16 *) value = inw(port);	} else if (width <= 32) {

⌨️ 快捷键说明

复制代码Ctrl + C
搜索代码Ctrl + F
全屏模式F11
增大字号Ctrl + =
减小字号Ctrl + -
显示快捷键?