source: trunk/hal/x86_64/drivers/pic_apic.c @ 218

Last change on this file since 218 was 208, checked in by max@…, 7 years ago

sync

File size: 3.6 KB
RevLine 
[75]1/*
[197]2 * pic_apic.c - APIC PIC driver implementation
[75]3 *
[197]4 * Copyright (c) 2017 Maxime Villard
[75]5 *
6 * This file is part of ALMOS-MKH.
7 *
[197]8 * ALMOS-MKH is free software; you can redistribute it and/or modify it
[75]9 * under the terms of the GNU General Public License as published by
10 * the Free Software Foundation; version 2.0 of the License.
11 *
[197]12 * ALMOS-MKH is distributed in the hope that it will be useful, but
[75]13 * WITHOUT ANY WARRANTY; without even the implied warranty of
14 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
15 * General Public License for more details.
16 *
17 * You should have received a copy of the GNU General Public License
[197]18 * along with ALMOS-MKH; if not, write to the Free Software Foundation,
[75]19 * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
20 */
21
22#include <hal_types.h>
23#include <chdev.h>
[197]24#include <pic_apic.h>
[75]25#include <errno.h>
26#include <string.h>
27#include <vfs.h>
28
[203]29#include <hal_internal.h>
30#include <hal_segmentation.h>
31#include <hal_apic.h>
32
33extern iopic_input_t iopic_input;
34
35static void
36idt_set_seg(struct idt_seg *seg, void *func, int ist, int type, int dpl, int sel)
[75]37{
[203]38        seg->gd_looffset = (uint64_t)func & 0xffff;
39        seg->gd_selector = sel;
40        seg->gd_ist = ist;
41        seg->gd_type = type;
42        seg->gd_dpl = dpl;
43        seg->gd_p = 1;
44        seg->gd_hioffset = (uint64_t)func >> 16;
45        seg->gd_zero = 0;
46        seg->gd_xx1 = 0;
47        seg->gd_xx2 = 0;
48        seg->gd_xx3 = 0;
[196]49}
[75]50
[203]51/* -------------------------------------------------------------------------- */
52
53extern uint8_t *idtstore;
54
55void pic_apic_init(chdev_t *pic)
56{
57        /* XXX APIC already initialized earlier */
58}
59
[208]60void pic_apic_extend_init(uint32_t *xcu_base)
[196]61{
62        x86_panic((char *)__func__);
[75]63}
64
[203]65void pic_apic_bind_irq(lid_t lid, chdev_t *src_chdev)
[75]66{
[203]67        struct idt_seg seg, *idt;
68        void *isr;
69        int slot;
70
71        /* get the source chdev functional type, channel, and direction */
72        uint32_t func    = src_chdev->func;
73        uint32_t channel = src_chdev->channel;
74
75        /* get external IRQ index */
76        uint32_t irq_id;
77        if (func == DEV_FUNC_IOC)
78                irq_id = iopic_input.ioc[channel];
79        else if (func == DEV_FUNC_TXT)
80                irq_id = iopic_input.txt[channel];
81        else
82                assert(false, __FUNCTION__, "unsupported device\n");
83
84        /* get the ISR pointer */
85        isr = src_chdev->isr;
86
87        /* create the IDT entry, and register it */
88        idt_set_seg(&seg, isr, 0, SDT_SYS386IGT, SEL_KPL,
89            GDT_FIXED_SEL(GDT_KCODE_SEL, SEL_KPL));
90        slot = idt_slot_alloc();
91        idt = (struct idt_seg *)&idtstore;
92        memcpy(&idt[slot], &seg, sizeof(struct idt_seg));
93
94        /* Bind the IRQ line in IOAPIC */
95        hal_ioapic_bind_irq(irq_id, slot, lid);
96
97x86_printf("-> allocated %z\n", (uint64_t)slot);
[196]98}
[75]99
[208]100void pic_apic_enable_irq(lid_t lid, xptr_t src_chdev_xp)
[196]101{
[208]102        chdev_t *src_chdev = (chdev_t *)src_chdev_xp;
[203]103        uint32_t func = src_chdev->func;
104        uint32_t channel = src_chdev->channel;
105        uint32_t irq_id;
106
107        /* get external IRQ index */
108        if (func == DEV_FUNC_IOC)
109                irq_id = iopic_input.ioc[channel];
110        else if (func == DEV_FUNC_TXT)
111                irq_id = iopic_input.txt[channel];
112        else
113                assert(false, __FUNCTION__, "unsupported device\n");
114
115        /* enable the line */
116        hal_ioapic_enable_irq(irq_id);
[75]117}
118
[208]119void pic_apic_disable_irq(lid_t lid, xptr_t src_chdev_xp)
[75]120{
[208]121        chdev_t *src_chdev = (chdev_t *)src_chdev_xp;
[203]122        uint32_t func = src_chdev->func;
123        uint32_t channel = src_chdev->channel;
124        uint32_t irq_id;
125
126        /* get external IRQ index */
127        if (func == DEV_FUNC_IOC)
128                irq_id = iopic_input.ioc[channel];
129        else if (func == DEV_FUNC_TXT)
130                irq_id = iopic_input.txt[channel];
131        else
132                assert(false, __FUNCTION__, "unsupported device\n");
133
134        /* disable the line */
135        hal_ioapic_disable_irq(irq_id);
[196]136}
[75]137
[208]138void pic_apic_enable_timer(uint32_t period)
[196]139{
140        x86_panic((char *)__func__);
[75]141}
142
[208]143void pic_apic_send_ipi(cxy_t cxy, lid_t lid)
[75]144{
[196]145        x86_panic((char *)__func__);
[75]146}
Note: See TracBrowser for help on using the repository browser.