⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 apic.c

📁 xen虚拟机源代码安装包
💻 C
📖 第 1 页 / 共 3 页
字号:
/* *      based on linux-2.6.17.13/arch/i386/kernel/apic.c * *  Local APIC handling, local APIC timers * *  (c) 1999, 2000 Ingo Molnar <mingo@redhat.com> * *  Fixes *  Maciej W. Rozycki   :   Bits for genuine 82489DX APICs; *                  thanks to Eric Gilmore *                  and Rolf G. Tews *                  for testing these extensively. *    Maciej W. Rozycki :   Various updates and fixes. *    Mikael Pettersson :   Power Management for UP-APIC. *    Pavel Machek and *    Mikael Pettersson    :    PM converted to driver model. */#include <xen/config.h>#include <xen/perfc.h>#include <xen/errno.h>#include <xen/init.h>#include <xen/mm.h>#include <xen/sched.h>#include <xen/irq.h>#include <xen/delay.h>#include <xen/smp.h>#include <xen/softirq.h>#include <asm/mc146818rtc.h>#include <asm/msr.h>#include <asm/atomic.h>#include <asm/mpspec.h>#include <asm/flushtlb.h>#include <asm/hardirq.h>#include <asm/apic.h>#include <asm/io_apic.h>#include <mach_apic.h>#include <io_ports.h>/* * Knob to control our willingness to enable the local APIC. */int enable_local_apic __initdata = 0; /* -1=force-disable, +1=force-enable *//* * Debug level */int apic_verbosity;int x2apic_enabled __read_mostly = 0;static void apic_pm_activate(void);int modern_apic(void){    unsigned int lvr, version;    /* AMD systems use old APIC versions, so check the CPU */    if (boot_cpu_data.x86_vendor == X86_VENDOR_AMD &&        boot_cpu_data.x86 >= 0xf)        return 1;    lvr = apic_read(APIC_LVR);    version = GET_APIC_VERSION(lvr);    return version >= 0x14;}/* * 'what should we do if we get a hw irq event on an illegal vector'. * each architecture has to answer this themselves. */void ack_bad_irq(unsigned int irq){    printk("unexpected IRQ trap at vector %02x\n", irq);    /*     * Currently unexpected vectors happen only on SMP and APIC.     * We _must_ ack these because every local APIC has only N     * irq slots per priority level, and a 'hanging, unacked' IRQ     * holds up an irq slot - in excessive cases (when multiple     * unexpected vectors occur) that might lock up the APIC     * completely.     * But only ack when the APIC is enabled -AK     */    if (cpu_has_apic)        ack_APIC_irq();}void __init apic_intr_init(void){#ifdef CONFIG_SMP    smp_intr_init();#endif    /* self generated IPI for local APIC timer */    set_intr_gate(LOCAL_TIMER_VECTOR, apic_timer_interrupt);    /* IPI vectors for APIC spurious and error interrupts */    set_intr_gate(SPURIOUS_APIC_VECTOR, spurious_interrupt);    set_intr_gate(ERROR_APIC_VECTOR, error_interrupt);    /* Performance Counters Interrupt */    set_intr_gate(PMU_APIC_VECTOR, pmu_apic_interrupt);    /* thermal monitor LVT interrupt */#ifdef CONFIG_X86_MCE_P4THERMAL    set_intr_gate(THERMAL_APIC_VECTOR, thermal_interrupt);#endif}/* Using APIC to generate smp_local_timer_interrupt? */int using_apic_timer = 0;static int enabled_via_apicbase;void enable_NMI_through_LVT0 (void * dummy){    unsigned int v, ver;    ver = apic_read(APIC_LVR);    ver = GET_APIC_VERSION(ver);    v = APIC_DM_NMI;			/* unmask and set to NMI */    if (!APIC_INTEGRATED(ver))		/* 82489DX */        v |= APIC_LVT_LEVEL_TRIGGER;    apic_write_around(APIC_LVT0, v);}int get_physical_broadcast(void){    if (modern_apic())        return 0xff;    else        return 0xf;}int get_maxlvt(void){    unsigned int v, ver, maxlvt;    v = apic_read(APIC_LVR);    ver = GET_APIC_VERSION(v);    /* 82489DXs do not report # of LVT entries. */    maxlvt = APIC_INTEGRATED(ver) ? GET_APIC_MAXLVT(v) : 2;    return maxlvt;}void clear_local_APIC(void){    int maxlvt;    unsigned long v;    maxlvt = get_maxlvt();    /*     * Masking an LVT entry on a P6 can trigger a local APIC error     * if the vector is zero. Mask LVTERR first to prevent this.     */    if (maxlvt >= 3) {        v = ERROR_APIC_VECTOR; /* any non-zero vector will do */        apic_write_around(APIC_LVTERR, v | APIC_LVT_MASKED);    }    /*     * Careful: we have to set masks only first to deassert     * any level-triggered sources.     */    v = apic_read(APIC_LVTT);    apic_write_around(APIC_LVTT, v | APIC_LVT_MASKED);    v = apic_read(APIC_LVT0);    apic_write_around(APIC_LVT0, v | APIC_LVT_MASKED);    v = apic_read(APIC_LVT1);    apic_write_around(APIC_LVT1, v | APIC_LVT_MASKED);    if (maxlvt >= 4) {        v = apic_read(APIC_LVTPC);        apic_write_around(APIC_LVTPC, v | APIC_LVT_MASKED);    }/* lets not touch this if we didn't frob it */#ifdef CONFIG_X86_MCE_P4THERMAL    if (maxlvt >= 5) {        v = apic_read(APIC_LVTTHMR);        apic_write_around(APIC_LVTTHMR, v | APIC_LVT_MASKED);    }#endif    /*     * Clean APIC state for other OSs:     */    apic_write_around(APIC_LVTT, APIC_LVT_MASKED);    apic_write_around(APIC_LVT0, APIC_LVT_MASKED);    apic_write_around(APIC_LVT1, APIC_LVT_MASKED);    if (maxlvt >= 3)        apic_write_around(APIC_LVTERR, APIC_LVT_MASKED);    if (maxlvt >= 4)        apic_write_around(APIC_LVTPC, APIC_LVT_MASKED);#ifdef CONFIG_X86_MCE_P4THERMAL    if (maxlvt >= 5)        apic_write_around(APIC_LVTTHMR, APIC_LVT_MASKED);#endif    v = GET_APIC_VERSION(apic_read(APIC_LVR));    if (APIC_INTEGRATED(v)) {  /* !82489DX */        if (maxlvt > 3)        /* Due to Pentium errata 3AP and 11AP. */            apic_write(APIC_ESR, 0);        apic_read(APIC_ESR);    }}void __init connect_bsp_APIC(void){    if (pic_mode) {        /*         * Do not trust the local APIC being empty at bootup.         */        clear_local_APIC();        /*         * PIC mode, enable APIC mode in the IMCR, i.e.         * connect BSP's local APIC to INT and NMI lines.         */        apic_printk(APIC_VERBOSE, "leaving PIC mode, "                    "enabling APIC mode.\n");        outb(0x70, 0x22);        outb(0x01, 0x23);    }    enable_apic_mode();}void disconnect_bsp_APIC(int virt_wire_setup){    if (pic_mode) {        /*         * Put the board back into PIC mode (has an effect         * only on certain older boards).  Note that APIC         * interrupts, including IPIs, won't work beyond         * this point!  The only exception are INIT IPIs.         */        apic_printk(APIC_VERBOSE, "disabling APIC mode, "                    "entering PIC mode.\n");        outb(0x70, 0x22);        outb(0x00, 0x23);    }    else {        /* Go back to Virtual Wire compatibility mode */        unsigned long value;        /* For the spurious interrupt use vector F, and enable it */        value = apic_read(APIC_SPIV);        value &= ~APIC_VECTOR_MASK;        value |= APIC_SPIV_APIC_ENABLED;        value |= 0xf;        apic_write_around(APIC_SPIV, value);        if (!virt_wire_setup) {            /* For LVT0 make it edge triggered, active high, external and enabled */            value = apic_read(APIC_LVT0);            value &= ~(APIC_MODE_MASK | APIC_SEND_PENDING |                       APIC_INPUT_POLARITY | APIC_LVT_REMOTE_IRR |                       APIC_LVT_LEVEL_TRIGGER | APIC_LVT_MASKED );            value |= APIC_LVT_REMOTE_IRR | APIC_SEND_PENDING;            value = SET_APIC_DELIVERY_MODE(value, APIC_MODE_EXTINT);            apic_write_around(APIC_LVT0, value);        }        else {            /* Disable LVT0 */            apic_write_around(APIC_LVT0, APIC_LVT_MASKED);        }        /* For LVT1 make it edge triggered, active high, nmi and enabled */        value = apic_read(APIC_LVT1);        value &= ~(            APIC_MODE_MASK | APIC_SEND_PENDING |            APIC_INPUT_POLARITY | APIC_LVT_REMOTE_IRR |            APIC_LVT_LEVEL_TRIGGER | APIC_LVT_MASKED);        value |= APIC_LVT_REMOTE_IRR | APIC_SEND_PENDING;        value = SET_APIC_DELIVERY_MODE(value, APIC_MODE_NMI);        apic_write_around(APIC_LVT1, value);    }}void disable_local_APIC(void){    unsigned long value;    clear_local_APIC();    /*     * Disable APIC (implies clearing of registers     * for 82489DX!).     */    value = apic_read(APIC_SPIV);    value &= ~APIC_SPIV_APIC_ENABLED;    apic_write_around(APIC_SPIV, value);    if (enabled_via_apicbase) {        unsigned int l, h;        rdmsr(MSR_IA32_APICBASE, l, h);        l &= ~MSR_IA32_APICBASE_ENABLE;        wrmsr(MSR_IA32_APICBASE, l, h);    }}/* * This is to verify that we're looking at a real local APIC. * Check these against your board if the CPUs aren't getting * started for no apparent reason. */int __init verify_local_APIC(void){    unsigned int reg0, reg1;    /*     * The version register is read-only in a real APIC.     */    reg0 = apic_read(APIC_LVR);    apic_printk(APIC_DEBUG, "Getting VERSION: %x\n", reg0);    /* We don't try writing LVR in x2APIC mode since that incurs #GP. */    if ( !x2apic_enabled )        apic_write(APIC_LVR, reg0 ^ APIC_LVR_MASK);    reg1 = apic_read(APIC_LVR);    apic_printk(APIC_DEBUG, "Getting VERSION: %x\n", reg1);    /*     * The two version reads above should print the same     * numbers.  If the second one is different, then we     * poke at a non-APIC.     */    if (reg1 != reg0)        return 0;    /*     * Check if the version looks reasonably.     */    reg1 = GET_APIC_VERSION(reg0);    if (reg1 == 0x00 || reg1 == 0xff)        return 0;    reg1 = get_maxlvt();    if (reg1 < 0x02 || reg1 == 0xff)        return 0;    /*     * The ID register is read/write in a real APIC.     */    reg0 = apic_read(APIC_ID);    apic_printk(APIC_DEBUG, "Getting ID: %x\n", reg0);    /*     * The next two are just to see if we have sane values.     * They're only really relevant if we're in Virtual Wire     * compatibility mode, but most boxes are anymore.     */    reg0 = apic_read(APIC_LVT0);    apic_printk(APIC_DEBUG, "Getting LVT0: %x\n", reg0);    reg1 = apic_read(APIC_LVT1);    apic_printk(APIC_DEBUG, "Getting LVT1: %x\n", reg1);    return 1;}void __init sync_Arb_IDs(void){    /* Unsupported on P4 - see Intel Dev. Manual Vol. 3, Ch. 8.6.1       And not needed on AMD */    if (modern_apic())        return;    /*     * Wait for idle.     */    apic_wait_icr_idle();    apic_printk(APIC_DEBUG, "Synchronizing Arb IDs.\n");    apic_write_around(APIC_ICR, APIC_DEST_ALLINC | APIC_INT_LEVELTRIG                      | APIC_DM_INIT);}extern void __error_in_apic_c (void);/* * An initial setup of the virtual wire mode. */void __init init_bsp_APIC(void){    unsigned long value, ver;    /*     * Don't do the setup now if we have a SMP BIOS as the     * through-I/O-APIC virtual wire mode might be active.     */    if (smp_found_config || !cpu_has_apic)        return;    value = apic_read(APIC_LVR);    ver = GET_APIC_VERSION(value);        /*     * Do not trust the local APIC being empty at bootup.     */    clear_local_APIC();        /*     * Enable APIC.     */    value = apic_read(APIC_SPIV);    value &= ~APIC_VECTOR_MASK;    value |= APIC_SPIV_APIC_ENABLED;        /* This bit is reserved on P4/Xeon and should be cleared */    if ((boot_cpu_data.x86_vendor == X86_VENDOR_INTEL) && (boot_cpu_data.x86 == 15))        value &= ~APIC_SPIV_FOCUS_DISABLED;    else        value |= APIC_SPIV_FOCUS_DISABLED;    value |= SPURIOUS_APIC_VECTOR;    apic_write_around(APIC_SPIV, value);    /*     * Set up the virtual wire mode.     */    apic_write_around(APIC_LVT0, APIC_DM_EXTINT);    value = APIC_DM_NMI;    if (!APIC_INTEGRATED(ver))              /* 82489DX */        value |= APIC_LVT_LEVEL_TRIGGER;    apic_write_around(APIC_LVT1, value);}void __devinit setup_local_APIC(void){    unsigned long oldvalue, value, ver, maxlvt;    int i, j;    /* Pound the ESR really hard over the head with a big hammer - mbligh */    if (esr_disable) {        apic_write(APIC_ESR, 0);        apic_write(APIC_ESR, 0);        apic_write(APIC_ESR, 0);        apic_write(APIC_ESR, 0);    }    value = apic_read(APIC_LVR);    ver = GET_APIC_VERSION(value);    if ((SPURIOUS_APIC_VECTOR & 0x0f) != 0x0f)        __error_in_apic_c();    /*     * Double-check whether this APIC is really registered.     */

⌨️ 快捷键说明

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