#ifndef _KERNEL_ARCH_x86_APIC_H
#define _KERNEL_ARCH_x86_APIC_H
#include <boot/kernel_args.h>
#include <SupportDefs.h>
#define APIC_ENABLE 0x100
#define APIC_FOCUS (~(1 << 9))
#define APIC_SIV (0xff)
#define APIC_ID 0x020
#define APIC_VERSION 0x030
#define APIC_TASK_PRIORITY 0x080
#define APIC_ARBITRATION_PRIORITY 0x090
#define APIC_PROCESSOR_PRIORITY 0x0a0
#define APIC_EOI 0x0b0
#define APIC_LOGICAL_DEST 0x0d0
#define APIC_DEST_FORMAT 0x0e0
#define APIC_SPURIOUS_INTR_VECTOR 0x0f0
#define APIC_ERROR_STATUS 0x280
#define APIC_INTR_COMMAND_1 0x300
#define APIC_INTR_COMMAND_2 0x310
#define APIC_LVT_TIMER 0x320
#define APIC_LVT_THERMAL_SENSOR 0x330
#define APIC_LVT_PERFMON_COUNTERS 0x340
#define APIC_LVT_LINT0 0x350
#define APIC_LVT_LINT1 0x360
#define APIC_LVT_ERROR 0x370
#define APIC_INITIAL_TIMER_COUNT 0x380
#define APIC_CURRENT_TIMER_COUNT 0x390
#define APIC_TIMER_DIVIDE_CONFIG 0x3e0
#define APIC_DELIVERY_MODE_FIXED 0
#define APIC_DELIVERY_MODE_LOWESTPRI (1 << 8)
#define APIC_DELIVERY_MODE_SMI (2 << 8)
#define APIC_DELIVERY_MODE_NMI (4 << 8)
#define APIC_DELIVERY_MODE_INIT (5 << 8)
#define APIC_DELIVERY_MODE_STARTUP (6 << 8)
#define APIC_DELIVERY_MODE_ExtINT (7 << 8)
#define APIC_DELIVERY_STATUS (1 << 12)
#define APIC_TRIGGER_MODE_LEVEL (1 << 15)
#define APIC_INTR_COMMAND_1_MASK 0xfff32000
#define APIC_INTR_COMMAND_2_MASK 0x00ffffff
#define APIC_INTR_COMMAND_1_DEST_MODE_PHYSICAL 0
#define APIC_INTR_COMMAND_1_DEST_MODE_LOGICAL (1 << 11)
#define APIC_INTR_COMMAND_1_ASSERT (1 << 14)
#define APIC_INTR_COMMAND_1_DEST_FIELD 0
#define APIC_INTR_COMMAND_1_DEST_SELF (1 << 18)
#define APIC_INTR_COMMAND_1_DEST_ALL (2 << 18)
#define APIC_INTR_COMMAND_1_DEST_ALL_BUT_SELF (3 << 18)
#define APIC_LVT_MASKED (1 << 16)
#define APIC_LVT_TIMER_MASK 0xfffcef00
#define APIC_LVT_LINT_MASK 0xfffe0800
#define APIC_LVT_LINT_INPUT_POLARITY (1 << 13)
#define APIC_TIMER_DIVIDE_CONFIG_1 0x0b
#define APIC_TIMER_DIVIDE_CONFIG_2 0x00
#define APIC_TIMER_DIVIDE_CONFIG_4 0x01
#define APIC_TIMER_DIVIDE_CONFIG_8 0x02
#define APIC_TIMER_DIVIDE_CONFIG_16 0x03
#define APIC_TIMER_DIVIDE_CONFIG_32 0x08
#define APIC_TIMER_DIVIDE_CONFIG_64 0x09
#define APIC_TIMER_DIVIDE_CONFIG_128 0x0a
#if !_BOOT_MODE
bool apic_available();
bool x2apic_available();
uint32 apic_local_id();
uint32 apic_local_version();
uint32 apic_task_priority();
void apic_set_task_priority(uint32 config);
void apic_end_of_interrupt();
void apic_disable_local_ints();
uint32 apic_spurious_intr_vector();
void apic_set_spurious_intr_vector(uint32 config);
void apic_set_interrupt_command(uint32 destination, uint32 mode);
bool apic_interrupt_delivered(void);
uint32 apic_lvt_timer();
void apic_set_lvt_timer(uint32 config);
uint32 apic_lvt_error();
void apic_set_lvt_error(uint32 config);
uint32 apic_lvt_initial_timer_count();
void apic_set_lvt_initial_timer_count(uint32 config);
uint32 apic_lvt_current_timer_count();
uint32 apic_lvt_timer_divide_config();
void apic_set_lvt_timer_divide_config(uint32 config);
status_t apic_init(kernel_args *args);
status_t apic_per_cpu_init(kernel_args *args, int32 cpu);
#endif
#endif