mpparse.c

来自「LINUX 2.6.17.4的源码」· C语言 代码 · 共 1,165 行 · 第 1/3 页

C
1,165
字号
void __init find_smp_config (void){	unsigned int address;	/*	 * FIXME: Linux assumes you have 640K of base ram..	 * this continues the error...	 *	 * 1) Scan the bottom 1K for a signature	 * 2) Scan the top 1K of base RAM	 * 3) Scan the 64K of bios	 */	if (smp_scan_config(0x0,0x400) ||		smp_scan_config(639*0x400,0x400) ||			smp_scan_config(0xF0000,0x10000))		return;	/*	 * If it is an SMP machine we should know now, unless the	 * configuration is in an EISA/MCA bus machine with an	 * extended bios data area.	 *	 * there is a real-mode segmented pointer pointing to the	 * 4K EBDA area at 0x40E, calculate and scan it here.	 *	 * NOTE! There are Linux loaders that will corrupt the EBDA	 * area, and as such this kind of SMP config may be less	 * trustworthy, simply because the SMP table may have been	 * stomped on during early boot. These loaders are buggy and	 * should be fixed.	 *	 * MP1.4 SPEC states to only scan first 1K of 4K EBDA.	 */	address = get_bios_ebda();	if (address)		smp_scan_config(address, 0x400);}int es7000_plat;/* --------------------------------------------------------------------------                            ACPI-based MP Configuration   -------------------------------------------------------------------------- */#ifdef CONFIG_ACPIvoid __init mp_register_lapic_address (	u64			address){	mp_lapic_addr = (unsigned long) address;	set_fixmap_nocache(FIX_APIC_BASE, mp_lapic_addr);	if (boot_cpu_physical_apicid == -1U)		boot_cpu_physical_apicid = GET_APIC_ID(apic_read(APIC_ID));	Dprintk("Boot CPU = %d\n", boot_cpu_physical_apicid);}void __devinit mp_register_lapic (	u8			id, 	u8			enabled){	struct mpc_config_processor processor;	int			boot_cpu = 0;		if (MAX_APICS - id <= 0) {		printk(KERN_WARNING "Processor #%d invalid (max %d)\n",			id, MAX_APICS);		return;	}	if (id == boot_cpu_physical_apicid)		boot_cpu = 1;	processor.mpc_type = MP_PROCESSOR;	processor.mpc_apicid = id;	processor.mpc_apicver = GET_APIC_VERSION(apic_read(APIC_LVR));	processor.mpc_cpuflag = (enabled ? CPU_ENABLED : 0);	processor.mpc_cpuflag |= (boot_cpu ? CPU_BOOTPROCESSOR : 0);	processor.mpc_cpufeature = (boot_cpu_data.x86 << 8) | 		(boot_cpu_data.x86_model << 4) | boot_cpu_data.x86_mask;	processor.mpc_featureflag = boot_cpu_data.x86_capability[0];	processor.mpc_reserved[0] = 0;	processor.mpc_reserved[1] = 0;	MP_processor_info(&processor);}#ifdef	CONFIG_X86_IO_APIC#define MP_ISA_BUS		0#define MP_MAX_IOAPIC_PIN	127static struct mp_ioapic_routing {	int			apic_id;	int			gsi_base;	int			gsi_end;	u32			pin_programmed[4];} mp_ioapic_routing[MAX_IO_APICS];static int mp_find_ioapic (	int			gsi){	int			i = 0;	/* Find the IOAPIC that manages this GSI. */	for (i = 0; i < nr_ioapics; i++) {		if ((gsi >= mp_ioapic_routing[i].gsi_base)			&& (gsi <= mp_ioapic_routing[i].gsi_end))			return i;	}	printk(KERN_ERR "ERROR: Unable to locate IOAPIC for GSI %d\n", gsi);	return -1;}	void __init mp_register_ioapic (	u8			id, 	u32			address,	u32			gsi_base){	int			idx = 0;	int			tmpid;	if (nr_ioapics >= MAX_IO_APICS) {		printk(KERN_ERR "ERROR: Max # of I/O APICs (%d) exceeded "			"(found %d)\n", MAX_IO_APICS, nr_ioapics);		panic("Recompile kernel with bigger MAX_IO_APICS!\n");	}	if (!address) {		printk(KERN_ERR "WARNING: Bogus (zero) I/O APIC address"			" found in MADT table, skipping!\n");		return;	}	idx = nr_ioapics++;	mp_ioapics[idx].mpc_type = MP_IOAPIC;	mp_ioapics[idx].mpc_flags = MPC_APIC_USABLE;	mp_ioapics[idx].mpc_apicaddr = address;	set_fixmap_nocache(FIX_IO_APIC_BASE_0 + idx, address);	if ((boot_cpu_data.x86_vendor == X86_VENDOR_INTEL)		&& !APIC_XAPIC(apic_version[boot_cpu_physical_apicid]))		tmpid = io_apic_get_unique_id(idx, id);	else		tmpid = id;	if (tmpid == -1) {		nr_ioapics--;		return;	}	mp_ioapics[idx].mpc_apicid = tmpid;	mp_ioapics[idx].mpc_apicver = io_apic_get_version(idx);		/* 	 * Build basic GSI lookup table to facilitate gsi->io_apic lookups	 * and to prevent reprogramming of IOAPIC pins (PCI GSIs).	 */	mp_ioapic_routing[idx].apic_id = mp_ioapics[idx].mpc_apicid;	mp_ioapic_routing[idx].gsi_base = gsi_base;	mp_ioapic_routing[idx].gsi_end = gsi_base + 		io_apic_get_redir_entries(idx);	printk("IOAPIC[%d]: apic_id %d, version %d, address 0x%lx, "		"GSI %d-%d\n", idx, mp_ioapics[idx].mpc_apicid, 		mp_ioapics[idx].mpc_apicver, mp_ioapics[idx].mpc_apicaddr,		mp_ioapic_routing[idx].gsi_base,		mp_ioapic_routing[idx].gsi_end);	return;}void __init mp_override_legacy_irq (	u8			bus_irq,	u8			polarity, 	u8			trigger, 	u32			gsi){	struct mpc_config_intsrc intsrc;	int			ioapic = -1;	int			pin = -1;	/* 	 * Convert 'gsi' to 'ioapic.pin'.	 */	ioapic = mp_find_ioapic(gsi);	if (ioapic < 0)		return;	pin = gsi - mp_ioapic_routing[ioapic].gsi_base;	/*	 * TBD: This check is for faulty timer entries, where the override	 *      erroneously sets the trigger to level, resulting in a HUGE 	 *      increase of timer interrupts!	 */	if ((bus_irq == 0) && (trigger == 3))		trigger = 1;	intsrc.mpc_type = MP_INTSRC;	intsrc.mpc_irqtype = mp_INT;	intsrc.mpc_irqflag = (trigger << 2) | polarity;	intsrc.mpc_srcbus = MP_ISA_BUS;	intsrc.mpc_srcbusirq = bus_irq;				       /* IRQ */	intsrc.mpc_dstapic = mp_ioapics[ioapic].mpc_apicid;	   /* APIC ID */	intsrc.mpc_dstirq = pin;				    /* INTIN# */	Dprintk("Int: type %d, pol %d, trig %d, bus %d, irq %d, %d-%d\n",		intsrc.mpc_irqtype, intsrc.mpc_irqflag & 3, 		(intsrc.mpc_irqflag >> 2) & 3, intsrc.mpc_srcbus, 		intsrc.mpc_srcbusirq, intsrc.mpc_dstapic, intsrc.mpc_dstirq);	mp_irqs[mp_irq_entries] = intsrc;	if (++mp_irq_entries == MAX_IRQ_SOURCES)		panic("Max # of irq sources exceeded!\n");	return;}void __init mp_config_acpi_legacy_irqs (void){	struct mpc_config_intsrc intsrc;	int			i = 0;	int			ioapic = -1;	/* 	 * Fabricate the legacy ISA bus (bus #31).	 */	mp_bus_id_to_type[MP_ISA_BUS] = MP_BUS_ISA;	Dprintk("Bus #%d is ISA\n", MP_ISA_BUS);	/*	 * Older generations of ES7000 have no legacy identity mappings	 */	if (es7000_plat == 1)		return;	/* 	 * Locate the IOAPIC that manages the ISA IRQs (0-15). 	 */	ioapic = mp_find_ioapic(0);	if (ioapic < 0)		return;	intsrc.mpc_type = MP_INTSRC;	intsrc.mpc_irqflag = 0;					/* Conforming */	intsrc.mpc_srcbus = MP_ISA_BUS;	intsrc.mpc_dstapic = mp_ioapics[ioapic].mpc_apicid;	/* 	 * Use the default configuration for the IRQs 0-15.  Unless	 * overriden by (MADT) interrupt source override entries.	 */	for (i = 0; i < 16; i++) {		int idx;		for (idx = 0; idx < mp_irq_entries; idx++) {			struct mpc_config_intsrc *irq = mp_irqs + idx;			/* Do we already have a mapping for this ISA IRQ? */			if (irq->mpc_srcbus == MP_ISA_BUS && irq->mpc_srcbusirq == i)				break;			/* Do we already have a mapping for this IOAPIC pin */			if ((irq->mpc_dstapic == intsrc.mpc_dstapic) &&				(irq->mpc_dstirq == i))				break;		}		if (idx != mp_irq_entries) {			printk(KERN_DEBUG "ACPI: IRQ%d used by override.\n", i);			continue;			/* IRQ already used */		}		intsrc.mpc_irqtype = mp_INT;		intsrc.mpc_srcbusirq = i;		   /* Identity mapped */		intsrc.mpc_dstirq = i;		Dprintk("Int: type %d, pol %d, trig %d, bus %d, irq %d, "			"%d-%d\n", intsrc.mpc_irqtype, intsrc.mpc_irqflag & 3, 			(intsrc.mpc_irqflag >> 2) & 3, intsrc.mpc_srcbus, 			intsrc.mpc_srcbusirq, intsrc.mpc_dstapic, 			intsrc.mpc_dstirq);		mp_irqs[mp_irq_entries] = intsrc;		if (++mp_irq_entries == MAX_IRQ_SOURCES)			panic("Max # of irq sources exceeded!\n");	}}#define MAX_GSI_NUM	4096int mp_register_gsi (u32 gsi, int triggering, int polarity){	int			ioapic = -1;	int			ioapic_pin = 0;	int			idx, bit = 0;	static int		pci_irq = 16;	/*	 * Mapping between Global System Interrups, which	 * represent all possible interrupts, and IRQs	 * assigned to actual devices.	 */	static int		gsi_to_irq[MAX_GSI_NUM];	/* Don't set up the ACPI SCI because it's already set up */	if (acpi_fadt.sci_int == gsi)		return gsi;	ioapic = mp_find_ioapic(gsi);	if (ioapic < 0) {		printk(KERN_WARNING "No IOAPIC for GSI %u\n", gsi);		return gsi;	}	ioapic_pin = gsi - mp_ioapic_routing[ioapic].gsi_base;	if (ioapic_renumber_irq)		gsi = ioapic_renumber_irq(ioapic, gsi);	/* 	 * Avoid pin reprogramming.  PRTs typically include entries  	 * with redundant pin->gsi mappings (but unique PCI devices);	 * we only program the IOAPIC on the first.	 */	bit = ioapic_pin % 32;	idx = (ioapic_pin < 32) ? 0 : (ioapic_pin / 32);	if (idx > 3) {		printk(KERN_ERR "Invalid reference to IOAPIC pin "			"%d-%d\n", mp_ioapic_routing[ioapic].apic_id, 			ioapic_pin);		return gsi;	}	if ((1<<bit) & mp_ioapic_routing[ioapic].pin_programmed[idx]) {		Dprintk(KERN_DEBUG "Pin %d-%d already programmed\n",			mp_ioapic_routing[ioapic].apic_id, ioapic_pin);		return gsi_to_irq[gsi];	}	mp_ioapic_routing[ioapic].pin_programmed[idx] |= (1<<bit);	if (triggering == ACPI_LEVEL_SENSITIVE) {		/*		 * For PCI devices assign IRQs in order, avoiding gaps		 * due to unused I/O APIC pins.		 */		int irq = gsi;		if (gsi < MAX_GSI_NUM) {			/*			 * Retain the VIA chipset work-around (gsi > 15), but			 * avoid a problem where the 8254 timer (IRQ0) is setup			 * via an override (so it's not on pin 0 of the ioapic),			 * and at the same time, the pin 0 interrupt is a PCI			 * type.  The gsi > 15 test could cause these two pins			 * to be shared as IRQ0, and they are not shareable.			 * So test for this condition, and if necessary, avoid			 * the pin collision.			 */			if (gsi > 15 || (gsi == 0 && !timer_uses_ioapic_pin_0))				gsi = pci_irq++;			/*			 * Don't assign IRQ used by ACPI SCI			 */			if (gsi == acpi_fadt.sci_int)				gsi = pci_irq++;			gsi_to_irq[irq] = gsi;		} else {			printk(KERN_ERR "GSI %u is too high\n", gsi);			return gsi;		}	}	io_apic_set_pci_routing(ioapic, ioapic_pin, gsi,		    triggering == ACPI_EDGE_SENSITIVE ? 0 : 1,		    polarity == ACPI_ACTIVE_HIGH ? 0 : 1);	return gsi;}#endif /* CONFIG_X86_IO_APIC */#endif /* CONFIG_ACPI */

⌨️ 快捷键说明

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