2015-04-11 01:44:37 +02:00
|
|
|
/*
|
|
|
|
* Copyright (c) 1984-2008, 2011-2015 Wind River Systems, Inc.
|
|
|
|
*
|
2015-10-06 18:00:37 +02:00
|
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
* you may not use this file except in compliance with the License.
|
|
|
|
* You may obtain a copy of the License at
|
2015-04-11 01:44:37 +02:00
|
|
|
*
|
2015-10-06 18:00:37 +02:00
|
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
2015-04-11 01:44:37 +02:00
|
|
|
*
|
2015-10-06 18:00:37 +02:00
|
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
* See the License for the specific language governing permissions and
|
|
|
|
* limitations under the License.
|
2015-04-11 01:44:37 +02:00
|
|
|
*/
|
|
|
|
|
2015-12-04 16:09:39 +01:00
|
|
|
/**
|
|
|
|
* @file
|
|
|
|
* @brief LoApicIntr.c - Intel Pentium[234] Local APIC/xAPIC driver
|
|
|
|
*
|
2015-10-20 18:42:33 +02:00
|
|
|
* This module is a driver for the local APIC/xAPIC (Advanced Programmable
|
|
|
|
* Interrupt Controller) in P6 (PentiumPro, II, III) family processors
|
|
|
|
* and P7 (Pentium4) family processors. The local APIC/xAPIC is included
|
|
|
|
* in selected P6 (PentiumPro, II, III) and P7 (Pentium4) family processors.
|
|
|
|
* Beginning with the P6 family processors, the presence or absence of an
|
|
|
|
* on-chip local APIC can be detected using the CPUID instruction. When the
|
|
|
|
* CPUID instruction is executed, bit 9 of the feature flags returned in the
|
|
|
|
* EDX register indicates the presence (set) or absence (clear) of an on-chip
|
|
|
|
* local APIC.
|
|
|
|
*
|
|
|
|
* The local APIC performs two main functions for the processor:
|
|
|
|
* - It processes local external interrupts that the processor receives at its
|
|
|
|
* interrupt pins and local internal interrupts that software generates.
|
|
|
|
* - It communicates with an external IO APIC
|
|
|
|
* chip. The external IO APIC receives external interrupt events from
|
|
|
|
* peripheral and direct them to the local APIC. The IO APIC is
|
|
|
|
* part of Intel's system chip set.
|
|
|
|
* The local APIC controls the dispatching of interrupts (to its associated
|
|
|
|
* processor) that it receives either locally or from the IO APIC. It provides
|
|
|
|
* facilities for queuing, nesting and masking of interrupts. It handles the
|
|
|
|
* interrupt delivery protocol with its local processor and accesses to APIC
|
|
|
|
* registers.
|
|
|
|
* A timer on the local APIC allows local generation of interrupts, and
|
|
|
|
* local interrupt pins permit local reception of processor-specific interrupts.
|
|
|
|
* The local APIC can be disabled and used in conjunction with a standard 8259A
|
|
|
|
* style interrupt controller. Disabling the local APIC can be done in hardware
|
|
|
|
* for the Pentium processors or in software for the P6 and P7 (Pentium4) family
|
|
|
|
* processors.
|
|
|
|
*
|
|
|
|
* The local APIC in the Pentium4 processors (called the xAPIC) is an extension
|
|
|
|
* of the local APIC found in the P6 family processors. The primary difference
|
|
|
|
* between the APIC architecture and xAPIC architecture is that with Pentium4
|
|
|
|
* processors, the local xAPICs and IO xAPIC communicate with one another
|
|
|
|
* through the processors system bus; whereas, with the P6 family processors,
|
|
|
|
* communication between the local APICs and the IO APIC is handled through a
|
|
|
|
* dedicated 3-wire APIC bus. Also, some of the architectural features of the
|
|
|
|
* local APIC have been extended and/or modified in the local xAPIC.
|
|
|
|
*
|
|
|
|
* This driver contains three routines for use. They are:
|
|
|
|
* _loapic_init() initializes the Local APIC for the interrupt mode chosen.
|
|
|
|
* _loapic_enable()/disable() enables / disables the Local APIC.
|
|
|
|
*
|
|
|
|
* Local APIC is used in the Virtual Wire Mode: delivery mode ExtINT.
|
|
|
|
*
|
|
|
|
* Virtual Wire Mode is one of three interrupt modes defined by the MP
|
|
|
|
* specification. In this mode, interrupts are generated by the 8259A
|
|
|
|
* equivalent PICs (if present) and delivered to the Boot Strap Processor by
|
|
|
|
* the local APIC that is programmed to act as a "virtual Wire"; that
|
|
|
|
* is, the local APIC is logically indistinguishable from a hardware
|
|
|
|
* connection. This is a uniprocessor compatibility mode.
|
|
|
|
*
|
|
|
|
* The local and IO APICs support interrupts in the range of 32 to 255.
|
|
|
|
* Interrupt priority is implied by its vector, according to the following
|
|
|
|
* relationship: "priority = vector / 16".
|
|
|
|
* Here the quotient is rounded down to the nearest integer value to determine
|
|
|
|
* the priority, with 1 being the lowest and 15 is the highest. Because vectors
|
|
|
|
* 0 through 31 are reserved for exclusive use by the processor, the priority of
|
|
|
|
* user defined interrupts range from 2 to 15. A value of 15 in the Interrupt
|
|
|
|
* Class field of the Task Priority Register (TPR) will mask off all interrupts,
|
|
|
|
* which require interrupt service.
|
|
|
|
* The P6 family processor's local APIC includes an in-service entry and a
|
|
|
|
* holding entry for each priority level. To avoid losing interrupts, software
|
|
|
|
* should allocate no more than 2 interrupt vectors per priority. The Pentium4
|
|
|
|
* processor expands this support of all acceptance of two interrupts per vector
|
|
|
|
* rather than per priority level.
|
2015-10-20 18:42:33 +02:00
|
|
|
*
|
2015-10-20 18:42:33 +02:00
|
|
|
* INCLUDE FILES: loapic.h
|
2015-07-01 23:22:39 +02:00
|
|
|
*/
|
2015-04-11 01:44:37 +02:00
|
|
|
|
|
|
|
#include <nanokernel.h>
|
2015-05-28 19:56:47 +02:00
|
|
|
#include <arch/cpu.h>
|
2015-04-11 01:44:37 +02:00
|
|
|
#include <stdint.h>
|
2016-05-07 06:55:51 +02:00
|
|
|
#include <string.h>
|
2016-07-12 20:42:18 +02:00
|
|
|
#include <misc/__assert.h>
|
2015-04-11 01:44:37 +02:00
|
|
|
|
|
|
|
#include "board.h"
|
|
|
|
#include <toolchain.h>
|
|
|
|
#include <sections.h>
|
|
|
|
#include <drivers/loapic.h> /* public API declarations */
|
2015-12-08 20:17:56 +01:00
|
|
|
#include <init.h>
|
2016-05-07 06:55:51 +02:00
|
|
|
#include <drivers/sysapic.h>
|
2015-04-11 01:44:37 +02:00
|
|
|
|
|
|
|
/* IA32_APIC_BASE MSR Bits */
|
|
|
|
|
|
|
|
#define LOAPIC_BASE_MASK 0xfffff000 /* LO APIC Base Addr mask */
|
|
|
|
#define LOAPIC_GLOBAL_ENABLE 0x00000800 /* LO APIC Global Enable */
|
|
|
|
|
|
|
|
/* Local APIC ID Register Bits */
|
|
|
|
|
|
|
|
#define LOAPIC_ID_MASK 0x0f000000 /* LO APIC ID mask */
|
|
|
|
|
|
|
|
/* Local APIC Version Register Bits */
|
|
|
|
|
|
|
|
#define LOAPIC_VERSION_MASK 0x000000ff /* LO APIC Version mask */
|
|
|
|
#define LOAPIC_MAXLVT_MASK 0x00ff0000 /* LO APIC Max LVT mask */
|
|
|
|
#define LOAPIC_PENTIUM4 0x00000014 /* LO APIC in Pentium4 */
|
|
|
|
#define LOAPIC_LVT_PENTIUM4 5 /* LO APIC LVT - Pentium4 */
|
|
|
|
#define LOAPIC_LVT_P6 4 /* LO APIC LVT - P6 */
|
|
|
|
#define LOAPIC_LVT_P5 3 /* LO APIC LVT - P5 */
|
|
|
|
|
|
|
|
/* Local APIC Vector Table Bits */
|
|
|
|
|
|
|
|
#define LOAPIC_VECTOR 0x000000ff /* vectorNo */
|
|
|
|
#define LOAPIC_MODE 0x00000700 /* delivery mode */
|
|
|
|
#define LOAPIC_FIXED 0x00000000 /* delivery mode: FIXED */
|
|
|
|
#define LOAPIC_SMI 0x00000200 /* delivery mode: SMI */
|
|
|
|
#define LOAPIC_NMI 0x00000400 /* delivery mode: NMI */
|
|
|
|
#define LOAPIC_EXT 0x00000700 /* delivery mode: ExtINT */
|
|
|
|
#define LOAPIC_IDLE 0x00000000 /* delivery status: Idle */
|
|
|
|
#define LOAPIC_PEND 0x00001000 /* delivery status: Pend */
|
|
|
|
#define LOAPIC_HIGH 0x00000000 /* polarity: High */
|
|
|
|
#define LOAPIC_LOW 0x00002000 /* polarity: Low */
|
|
|
|
#define LOAPIC_REMOTE 0x00004000 /* remote IRR */
|
|
|
|
#define LOAPIC_EDGE 0x00000000 /* trigger mode: Edge */
|
|
|
|
#define LOAPIC_LEVEL 0x00008000 /* trigger mode: Level */
|
|
|
|
|
|
|
|
/* Local APIC Spurious-Interrupt Register Bits */
|
|
|
|
|
|
|
|
#define LOAPIC_ENABLE 0x100 /* APIC Enabled */
|
|
|
|
#define LOAPIC_FOCUS_DISABLE 0x200 /* Focus Processor Checking */
|
|
|
|
|
|
|
|
/* Local Vector's lock-unlock macro used in loApicIntLock/Unlock */
|
|
|
|
|
|
|
|
#define LOCKED_TIMER 0x01
|
|
|
|
#define LOCKED_PMC 0x02
|
|
|
|
#define LOCKED_LINT0 0x04
|
|
|
|
#define LOCKED_LINT1 0x08
|
|
|
|
#define LOCKED_ERROR 0x10
|
|
|
|
#define LOCKED_THERMAL 0x20
|
|
|
|
|
|
|
|
/* Interrupt Command Register: delivery mode and status */
|
|
|
|
|
|
|
|
#define MODE_FIXED 0x0 /* delivery mode: Fixed */
|
|
|
|
#define MODE_LOWEST 0x1 /* delivery mode: Lowest */
|
|
|
|
#define MODE_SMI 0x2 /* delivery mode: SMI */
|
|
|
|
#define MODE_NMI 0x4 /* delivery mode: NMI */
|
|
|
|
#define MODE_INIT 0x5 /* delivery mode: INIT */
|
|
|
|
#define MODE_STARTUP 0x6 /* delivery mode: StartUp */
|
|
|
|
#define STATUS_PEND 0x1000 /* delivery status: Pend */
|
|
|
|
|
|
|
|
/* MP Configuration Table Entries */
|
|
|
|
|
|
|
|
#define MP_ENTRY_CPU 0 /* Entry Type: CPU */
|
|
|
|
#define MP_ENTRY_BUS 1 /* Entry Type: BUS */
|
|
|
|
#define MP_ENTRY_IOAPIC 2 /* Entry Type: IO APIC */
|
|
|
|
#define MP_ENTRY_IOINTERRUPT 3 /* Entry Type: IO INT */
|
|
|
|
#define MP_ENTRY_LOINTERRUPT 4 /* Entry Type: LO INT */
|
|
|
|
|
|
|
|
/* Extended MP Configuration Table Entries */
|
|
|
|
|
|
|
|
#define EXT_MP_ENTRY_SASM 128 /* Entry Type: System Address Space Map */
|
|
|
|
#define EXT_MP_ENTRY_BHD 129 /* Entry Type: Bus Hierarchy Descriptor */
|
|
|
|
#define EXT_MP_ENTRY_CBASM 130 /* Entry Type: Comp Address Space Modifier */
|
|
|
|
|
|
|
|
/* MP Configuration Table CPU Flags */
|
|
|
|
|
|
|
|
#define MP_CPU_FLAGS_BP 0x02
|
|
|
|
|
|
|
|
/* IMCR related bits */
|
|
|
|
|
|
|
|
#define IMCR_ADRS 0x22 /* IMCR addr reg */
|
|
|
|
#define IMCR_DATA 0x23 /* IMCR data reg */
|
|
|
|
#define IMCR_REG_SEL 0x70 /* IMCR reg select */
|
|
|
|
#define IMCR_IOAPIC_ON 0x01 /* IMCR IOAPIC route enable */
|
|
|
|
#define IMCR_IOAPIC_OFF 0x00 /* IMCR IOAPIC route disable */
|
|
|
|
|
2015-12-09 23:53:41 +01:00
|
|
|
#if CONFIG_LOAPIC_SPURIOUS_VECTOR_ID == -1
|
|
|
|
#define LOAPIC_SPURIOUS_VECTOR_ID (CONFIG_IDT_NUM_VECTORS - 1)
|
|
|
|
#else
|
|
|
|
#define LOAPIC_SPURIOUS_VECTOR_ID CONFIG_LOAPIC_SPURIOUS_VECTOR_ID
|
|
|
|
#endif
|
2015-04-11 01:44:37 +02:00
|
|
|
|
2016-05-07 06:55:51 +02:00
|
|
|
#define LOPIC_SSPND_BITS_PER_IRQ 1 /* Just the one for enable disable*/
|
|
|
|
#define LOPIC_SUSPEND_BITS_REQD (ROUND_UP((LOAPIC_IRQ_COUNT * LOPIC_SSPND_BITS_PER_IRQ), 32))
|
|
|
|
#ifdef CONFIG_DEVICE_POWER_MANAGEMENT
|
|
|
|
#include <power.h>
|
|
|
|
uint32_t loapic_suspend_buf[LOPIC_SUSPEND_BITS_REQD / 32] = {0};
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
2015-07-01 23:22:39 +02:00
|
|
|
/**
|
|
|
|
*
|
2015-07-01 23:51:40 +02:00
|
|
|
* @brief Initialize the Local APIC or xAPIC
|
2015-07-01 23:22:39 +02:00
|
|
|
*
|
|
|
|
* This routine initializes Local APIC or xAPIC.
|
|
|
|
*
|
2015-07-01 23:29:04 +02:00
|
|
|
* @return N/A
|
2015-07-01 23:22:39 +02:00
|
|
|
*
|
|
|
|
*/
|
2015-04-11 01:44:37 +02:00
|
|
|
|
2016-08-02 21:05:08 +02:00
|
|
|
static int _loapic_init(struct device *unused)
|
2015-04-11 01:44:37 +02:00
|
|
|
{
|
2015-09-15 18:17:38 +02:00
|
|
|
ARG_UNUSED(unused);
|
2015-04-11 01:44:37 +02:00
|
|
|
int32_t loApicMaxLvt; /* local APIC Max LVT */
|
|
|
|
|
|
|
|
/* enable the Local APIC */
|
2016-08-02 21:05:08 +02:00
|
|
|
sys_write32(sys_read32(CONFIG_LOAPIC_BASE_ADDRESS + LOAPIC_SVR)
|
|
|
|
| LOAPIC_ENABLE, CONFIG_LOAPIC_BASE_ADDRESS + LOAPIC_SVR);
|
2015-04-11 01:44:37 +02:00
|
|
|
|
2015-07-28 20:39:12 +02:00
|
|
|
loApicMaxLvt = (*(volatile int *)(CONFIG_LOAPIC_BASE_ADDRESS + LOAPIC_VER) &
|
|
|
|
LOAPIC_MAXLVT_MASK) >> 16;
|
2015-04-11 01:44:37 +02:00
|
|
|
|
|
|
|
/* reset the DFR, TPR, TIMER_CONFIG, and TIMER_ICR */
|
|
|
|
|
2015-07-28 20:39:12 +02:00
|
|
|
*(volatile int *)(CONFIG_LOAPIC_BASE_ADDRESS + LOAPIC_DFR) =
|
|
|
|
(int)0xffffffff;
|
|
|
|
*(volatile int *)(CONFIG_LOAPIC_BASE_ADDRESS + LOAPIC_TPR) = (int)0x0;
|
|
|
|
*(volatile int *) (CONFIG_LOAPIC_BASE_ADDRESS + LOAPIC_TIMER_CONFIG) =
|
|
|
|
(int)0x0;
|
|
|
|
*(volatile int *)(CONFIG_LOAPIC_BASE_ADDRESS + LOAPIC_TIMER_ICR) = (int)0x0;
|
2015-04-11 01:44:37 +02:00
|
|
|
|
|
|
|
/* program Local Vector Table for the Virtual Wire Mode */
|
|
|
|
|
|
|
|
/* set LINT0: extInt, high-polarity, edge-trigger, not-masked */
|
|
|
|
|
2015-07-28 20:39:12 +02:00
|
|
|
*(volatile int *)(CONFIG_LOAPIC_BASE_ADDRESS + LOAPIC_LINT0) =
|
|
|
|
(*(volatile int *)(CONFIG_LOAPIC_BASE_ADDRESS + LOAPIC_LINT0) &
|
2015-04-22 19:41:07 +02:00
|
|
|
~(LOAPIC_MODE | LOAPIC_LOW | LOAPIC_LEVEL | LOAPIC_LVT_MASKED)) |
|
2015-04-11 01:44:37 +02:00
|
|
|
(LOAPIC_EXT | LOAPIC_HIGH | LOAPIC_EDGE);
|
|
|
|
|
|
|
|
/* set LINT1: NMI, high-polarity, edge-trigger, not-masked */
|
|
|
|
|
2015-07-28 20:39:12 +02:00
|
|
|
*(volatile int *)(CONFIG_LOAPIC_BASE_ADDRESS + LOAPIC_LINT1) =
|
|
|
|
(*(volatile int *)(CONFIG_LOAPIC_BASE_ADDRESS + LOAPIC_LINT1) &
|
2015-04-22 19:41:07 +02:00
|
|
|
~(LOAPIC_MODE | LOAPIC_LOW | LOAPIC_LEVEL | LOAPIC_LVT_MASKED)) |
|
2015-04-11 01:44:37 +02:00
|
|
|
(LOAPIC_NMI | LOAPIC_HIGH | LOAPIC_EDGE);
|
|
|
|
|
|
|
|
/* lock the Local APIC interrupts */
|
|
|
|
|
2015-07-28 20:39:12 +02:00
|
|
|
*(volatile int *)(CONFIG_LOAPIC_BASE_ADDRESS + LOAPIC_TIMER) =
|
|
|
|
LOAPIC_LVT_MASKED;
|
|
|
|
*(volatile int *)(CONFIG_LOAPIC_BASE_ADDRESS + LOAPIC_ERROR) =
|
|
|
|
LOAPIC_LVT_MASKED;
|
2015-04-11 01:44:37 +02:00
|
|
|
|
|
|
|
if (loApicMaxLvt >= LOAPIC_LVT_P6)
|
2015-07-28 20:39:12 +02:00
|
|
|
*(volatile int *) (CONFIG_LOAPIC_BASE_ADDRESS + LOAPIC_PMC) =
|
|
|
|
LOAPIC_LVT_MASKED;
|
2015-04-11 01:44:37 +02:00
|
|
|
|
|
|
|
if (loApicMaxLvt >= LOAPIC_LVT_PENTIUM4)
|
2015-07-28 20:39:12 +02:00
|
|
|
*(volatile int *)(CONFIG_LOAPIC_BASE_ADDRESS + LOAPIC_THERMAL) =
|
2015-04-22 19:41:07 +02:00
|
|
|
LOAPIC_LVT_MASKED;
|
2015-04-11 01:44:37 +02:00
|
|
|
|
2015-12-09 23:53:41 +01:00
|
|
|
#if CONFIG_LOAPIC_SPURIOUS_VECTOR
|
|
|
|
*(volatile int *)(CONFIG_LOAPIC_BASE_ADDRESS + LOAPIC_SVR) =
|
2016-08-02 21:05:08 +02:00
|
|
|
(*(volatile int *)(CONFIG_LOAPIC_BASE_ADDRESS + LOAPIC_SVR)
|
|
|
|
& 0xFFFFFF00)
|
2015-12-09 23:53:41 +01:00
|
|
|
| (LOAPIC_SPURIOUS_VECTOR_ID & 0xFF);
|
|
|
|
#endif
|
|
|
|
|
2015-04-11 01:44:37 +02:00
|
|
|
/* discard a pending interrupt if any */
|
2016-08-02 21:05:08 +02:00
|
|
|
#if CONFIG_EOI_FORWARDING_BUG
|
|
|
|
_lakemont_eoi();
|
|
|
|
#else
|
|
|
|
*(volatile int *)(CONFIG_LOAPIC_BASE_ADDRESS + LOAPIC_EOI) = 0;
|
|
|
|
#endif
|
2015-04-11 01:44:37 +02:00
|
|
|
|
2016-08-02 21:05:08 +02:00
|
|
|
return 0;
|
2015-04-11 01:44:37 +02:00
|
|
|
}
|
|
|
|
|
2015-07-01 23:22:39 +02:00
|
|
|
/**
|
|
|
|
*
|
2015-07-01 23:51:40 +02:00
|
|
|
* @brief Set the vector field in the specified RTE
|
2015-07-01 23:22:39 +02:00
|
|
|
*
|
2016-08-02 21:05:08 +02:00
|
|
|
* This routine is utilized by the interrupt controller's
|
|
|
|
* _interrupt_vector_allocate() routine (which exists to support the
|
|
|
|
* irq_connect_dynamic() API). Once a vector has been allocated, this routine
|
|
|
|
* is invoked to update the LVT entry associated with <irq> with the vector.
|
2015-07-01 23:22:39 +02:00
|
|
|
*
|
2015-07-01 23:29:04 +02:00
|
|
|
* @return N/A
|
2015-07-01 23:22:39 +02:00
|
|
|
*/
|
2015-04-11 01:44:37 +02:00
|
|
|
|
2015-07-27 17:02:41 +02:00
|
|
|
void _loapic_int_vec_set(unsigned int irq, /* IRQ number of the interrupt */
|
|
|
|
unsigned int vector /* vector to copy into the LVT */
|
2015-04-11 01:44:37 +02:00
|
|
|
)
|
|
|
|
{
|
|
|
|
volatile int *pLvt; /* pointer to local vector table */
|
|
|
|
int32_t oldLevel; /* previous interrupt lock level */
|
|
|
|
|
|
|
|
/*
|
|
|
|
* The following mappings are used:
|
|
|
|
*
|
|
|
|
* IRQ0 -> LOAPIC_TIMER
|
|
|
|
* IRQ1 -> LOAPIC_THERMAL
|
|
|
|
* IRQ2 -> LOAPIC_PMC
|
|
|
|
* IRQ3 -> LOAPIC_LINT0
|
|
|
|
* IRQ4 -> LOAPIC_LINT1
|
|
|
|
* IRQ5 -> LOAPIC_ERROR
|
|
|
|
*
|
|
|
|
* It's assumed that LVTs are spaced by 0x10 bytes
|
|
|
|
*/
|
|
|
|
|
2015-07-28 20:39:12 +02:00
|
|
|
pLvt = (volatile int *)
|
|
|
|
(CONFIG_LOAPIC_BASE_ADDRESS + LOAPIC_TIMER + (irq * 0x10));
|
2015-04-11 01:44:37 +02:00
|
|
|
|
|
|
|
/* update the 'vector' bits in the LVT */
|
|
|
|
|
|
|
|
oldLevel = irq_lock();
|
|
|
|
*pLvt = (*pLvt & ~LOAPIC_VECTOR) | vector;
|
|
|
|
irq_unlock(oldLevel);
|
|
|
|
}
|
|
|
|
|
2015-07-01 23:22:39 +02:00
|
|
|
/**
|
|
|
|
*
|
2015-07-01 23:51:40 +02:00
|
|
|
* @brief Enable an individual LOAPIC interrupt (IRQ)
|
2015-07-01 23:22:39 +02:00
|
|
|
*
|
2015-10-20 18:42:33 +02:00
|
|
|
* @param irq the IRQ number of the interrupt
|
|
|
|
*
|
2015-07-01 23:22:39 +02:00
|
|
|
* This routine clears the interrupt mask bit in the LVT for the specified IRQ
|
|
|
|
*
|
2015-07-01 23:29:04 +02:00
|
|
|
* @return N/A
|
2015-07-01 23:22:39 +02:00
|
|
|
*/
|
2015-04-11 01:44:37 +02:00
|
|
|
|
2015-10-20 18:42:33 +02:00
|
|
|
void _loapic_irq_enable(unsigned int irq)
|
2015-04-11 01:44:37 +02:00
|
|
|
{
|
|
|
|
volatile int *pLvt; /* pointer to local vector table */
|
|
|
|
int32_t oldLevel; /* previous interrupt lock level */
|
|
|
|
|
|
|
|
/*
|
|
|
|
* See the comments in _LoApicLvtVecSet() regarding IRQ to LVT mappings
|
|
|
|
* and ths assumption concerning LVT spacing.
|
|
|
|
*/
|
|
|
|
|
2015-07-28 20:39:12 +02:00
|
|
|
pLvt = (volatile int *)
|
|
|
|
(CONFIG_LOAPIC_BASE_ADDRESS + LOAPIC_TIMER + (irq * 0x10));
|
2015-04-11 01:44:37 +02:00
|
|
|
|
|
|
|
/* clear the mask bit in the LVT */
|
|
|
|
|
|
|
|
oldLevel = irq_lock();
|
2015-04-22 19:41:07 +02:00
|
|
|
*pLvt = *pLvt & ~LOAPIC_LVT_MASKED;
|
2015-04-11 01:44:37 +02:00
|
|
|
irq_unlock(oldLevel);
|
|
|
|
}
|
|
|
|
|
2015-07-01 23:22:39 +02:00
|
|
|
/**
|
|
|
|
*
|
2015-07-01 23:51:40 +02:00
|
|
|
* @brief Disable an individual LOAPIC interrupt (IRQ)
|
2015-07-01 23:22:39 +02:00
|
|
|
*
|
2015-10-20 18:42:33 +02:00
|
|
|
* @param irq the IRQ number of the interrupt
|
|
|
|
*
|
2015-07-01 23:22:39 +02:00
|
|
|
* This routine clears the interrupt mask bit in the LVT for the specified IRQ
|
|
|
|
*
|
2015-07-01 23:29:04 +02:00
|
|
|
* @return N/A
|
2015-07-01 23:22:39 +02:00
|
|
|
*/
|
2015-04-11 01:44:37 +02:00
|
|
|
|
2015-10-20 18:42:33 +02:00
|
|
|
void _loapic_irq_disable(unsigned int irq)
|
2015-04-11 01:44:37 +02:00
|
|
|
{
|
|
|
|
volatile int *pLvt; /* pointer to local vector table */
|
|
|
|
int32_t oldLevel; /* previous interrupt lock level */
|
|
|
|
|
|
|
|
/*
|
|
|
|
* See the comments in _LoApicLvtVecSet() regarding IRQ to LVT mappings
|
|
|
|
* and ths assumption concerning LVT spacing.
|
|
|
|
*/
|
|
|
|
|
2015-07-28 20:39:12 +02:00
|
|
|
pLvt = (volatile int *)
|
|
|
|
(CONFIG_LOAPIC_BASE_ADDRESS + LOAPIC_TIMER + (irq * 0x10));
|
2015-04-11 01:44:37 +02:00
|
|
|
|
|
|
|
/* set the mask bit in the LVT */
|
|
|
|
|
|
|
|
oldLevel = irq_lock();
|
2015-04-22 19:41:07 +02:00
|
|
|
*pLvt = *pLvt | LOAPIC_LVT_MASKED;
|
2015-04-11 01:44:37 +02:00
|
|
|
irq_unlock(oldLevel);
|
|
|
|
}
|
2015-08-22 02:10:32 +02:00
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Find the currently executing interrupt vector, if any
|
|
|
|
*
|
|
|
|
* This routine finds the vector of the interrupt that is being processed.
|
|
|
|
* The ISR (In-Service Register) register contain the vectors of the interrupts
|
|
|
|
* in service. And the higher vector is the indentification of the interrupt
|
|
|
|
* being currently processed.
|
|
|
|
*
|
2016-07-12 20:42:18 +02:00
|
|
|
* This function must be called with interrupts locked in interrupt context.
|
|
|
|
*
|
2015-08-22 02:10:32 +02:00
|
|
|
* ISR registers' offsets:
|
|
|
|
* --------------------
|
|
|
|
* | Offset | bits |
|
|
|
|
* --------------------
|
|
|
|
* | 0100H | 0:31 |
|
|
|
|
* | 0110H | 32:63 |
|
|
|
|
* | 0120H | 64:95 |
|
|
|
|
* | 0130H | 96:127 |
|
|
|
|
* | 0140H | 128:159 |
|
|
|
|
* | 0150H | 160:191 |
|
|
|
|
* | 0160H | 192:223 |
|
|
|
|
* | 0170H | 224:255 |
|
|
|
|
* --------------------
|
|
|
|
*
|
2016-09-08 20:01:23 +02:00
|
|
|
* @return The vector of the interrupt that is currently being processed, or -1
|
|
|
|
* if no IRQ is being serviced.
|
2015-08-22 02:10:32 +02:00
|
|
|
*/
|
2016-08-02 21:05:08 +02:00
|
|
|
int __irq_controller_isr_vector_get(void)
|
2015-08-22 02:10:32 +02:00
|
|
|
{
|
2016-07-12 20:42:18 +02:00
|
|
|
int pReg, block;
|
|
|
|
|
2016-09-08 20:01:23 +02:00
|
|
|
/* Block 0 bits never lit up as these are all exception or reserved
|
|
|
|
* vectors
|
|
|
|
*/
|
|
|
|
for (block = 7; likely(block > 0); block--) {
|
2016-07-12 20:42:18 +02:00
|
|
|
pReg = sys_read32(CONFIG_LOAPIC_BASE_ADDRESS + LOAPIC_ISR +
|
|
|
|
(block * 0x10));
|
|
|
|
if (pReg) {
|
|
|
|
return (block * 32) + (find_msb_set(pReg) - 1);
|
2015-08-22 02:10:32 +02:00
|
|
|
}
|
|
|
|
|
2016-07-12 20:42:18 +02:00
|
|
|
}
|
2016-09-08 20:01:23 +02:00
|
|
|
return -1;
|
2015-08-22 02:10:32 +02:00
|
|
|
}
|
2015-12-08 20:17:56 +01:00
|
|
|
|
2016-05-07 06:55:51 +02:00
|
|
|
#ifdef CONFIG_DEVICE_POWER_MANAGEMENT
|
|
|
|
static int loapic_suspend(struct device *port, int pm_policy)
|
|
|
|
{
|
|
|
|
volatile int *pLvt; /* pointer to local vector table */
|
|
|
|
int loapic_irq;
|
|
|
|
|
|
|
|
ARG_UNUSED(port);
|
|
|
|
|
|
|
|
if (pm_policy != SYS_PM_DEEP_SLEEP) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
memset(loapic_suspend_buf, 0, (LOPIC_SUSPEND_BITS_REQD >> 3));
|
|
|
|
|
|
|
|
for (loapic_irq = 0; loapic_irq < LOAPIC_IRQ_COUNT; loapic_irq++) {
|
|
|
|
|
|
|
|
if (_irq_to_interrupt_vector[LOAPIC_IRQ_BASE + loapic_irq]) {
|
|
|
|
|
|
|
|
/* Since vector numbers are already present in RAM/ROM,
|
|
|
|
* We save only the mask bits here.
|
|
|
|
*/
|
|
|
|
pLvt = (volatile int *)
|
|
|
|
(CONFIG_LOAPIC_BASE_ADDRESS + LOAPIC_TIMER +
|
|
|
|
(loapic_irq * 0x10));
|
|
|
|
|
|
|
|
if ((*pLvt & LOAPIC_LVT_MASKED) == 0) {
|
|
|
|
sys_bitfield_set_bit((mem_addr_t)loapic_suspend_buf,
|
|
|
|
loapic_irq);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2016-08-02 21:05:08 +02:00
|
|
|
static int loapic_resume(struct device *port, int pm_policy)
|
2016-05-07 06:55:51 +02:00
|
|
|
{
|
|
|
|
int loapic_irq;
|
|
|
|
|
|
|
|
ARG_UNUSED(port);
|
|
|
|
|
|
|
|
if (pm_policy != SYS_PM_DEEP_SLEEP) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Assuming all loapic device registers lose their state, the call to
|
|
|
|
* _loapic_init(), should bring all the registers to a sane state.
|
|
|
|
*/
|
|
|
|
_loapic_init(NULL);
|
|
|
|
|
|
|
|
for (loapic_irq = 0; loapic_irq < LOAPIC_IRQ_COUNT; loapic_irq++) {
|
|
|
|
|
|
|
|
if (_irq_to_interrupt_vector[LOAPIC_IRQ_BASE + loapic_irq]) {
|
|
|
|
/* Configure vector and enable the required ones*/
|
|
|
|
_loapic_int_vec_set(loapic_irq,
|
|
|
|
_irq_to_interrupt_vector[LOAPIC_IRQ_BASE + loapic_irq]);
|
|
|
|
|
|
|
|
if (sys_bitfield_test_bit((mem_addr_t) loapic_suspend_buf,
|
|
|
|
loapic_irq)) {
|
|
|
|
_loapic_irq_enable(loapic_irq);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
struct device_pm_ops loapic_pm_ops = {
|
|
|
|
.suspend = loapic_suspend,
|
|
|
|
.resume = loapic_resume
|
|
|
|
};
|
|
|
|
SYS_INIT_PM("loapic", _loapic_init, &loapic_pm_ops, PRIMARY,
|
|
|
|
CONFIG_KERNEL_INIT_PRIORITY_DEFAULT);
|
|
|
|
#else
|
2016-01-28 21:16:31 +01:00
|
|
|
SYS_INIT(_loapic_init, PRIMARY, CONFIG_KERNEL_INIT_PRIORITY_DEFAULT);
|
2016-05-07 06:55:51 +02:00
|
|
|
#endif /* CONFIG_DEVICE_POWER_MANAGEMENT */
|
|
|
|
|
2015-12-09 23:53:41 +01:00
|
|
|
|
|
|
|
#if CONFIG_LOAPIC_SPURIOUS_VECTOR
|
|
|
|
extern void _loapic_spurious_handler(void);
|
|
|
|
|
|
|
|
NANO_CPU_INT_REGISTER(_loapic_spurious_handler, NANO_SOFT_IRQ,
|
|
|
|
LOAPIC_SPURIOUS_VECTOR_ID >> 4,
|
|
|
|
LOAPIC_SPURIOUS_VECTOR_ID, 0);
|
|
|
|
#endif
|