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

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

Explain a bit.

File size: 3.8 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>
[404]27#include <printk.h>
[75]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);
[196]96}
[75]97
[208]98void pic_apic_enable_irq(lid_t lid, xptr_t src_chdev_xp)
[196]99{
[208]100        chdev_t *src_chdev = (chdev_t *)src_chdev_xp;
[203]101        uint32_t func = src_chdev->func;
102        uint32_t channel = src_chdev->channel;
103        uint32_t irq_id;
104
105        /* get external IRQ index */
106        if (func == DEV_FUNC_IOC)
107                irq_id = iopic_input.ioc[channel];
108        else if (func == DEV_FUNC_TXT)
109                irq_id = iopic_input.txt[channel];
110        else
111                assert(false, __FUNCTION__, "unsupported device\n");
112
113        /* enable the line */
114        hal_ioapic_enable_irq(irq_id);
[75]115}
116
[208]117void pic_apic_disable_irq(lid_t lid, xptr_t src_chdev_xp)
[75]118{
[208]119        chdev_t *src_chdev = (chdev_t *)src_chdev_xp;
[203]120        uint32_t func = src_chdev->func;
121        uint32_t channel = src_chdev->channel;
122        uint32_t irq_id;
123
124        /* get external IRQ index */
125        if (func == DEV_FUNC_IOC)
126                irq_id = iopic_input.ioc[channel];
127        else if (func == DEV_FUNC_TXT)
128                irq_id = iopic_input.txt[channel];
129        else
130                assert(false, __FUNCTION__, "unsupported device\n");
131
132        /* disable the line */
133        hal_ioapic_disable_irq(irq_id);
[196]134}
[75]135
[208]136void pic_apic_enable_timer(uint32_t period)
[196]137{
[404]138        /*
139         * TODO: right now, we don't do anything, because the LAPIC is already
140         * calibrated and launched earlier, mostly for debugging purposes. But it
141         * would be good to launch it here for real.
142         */
[75]143}
144
[280]145void pic_apic_enable_ipi()
146{
[404]147        /* TODO */
148//      x86_panic((char *)__func__);
[280]149}
150
[208]151void pic_apic_send_ipi(cxy_t cxy, lid_t lid)
[75]152{
[196]153        x86_panic((char *)__func__);
[75]154}
Note: See TracBrowser for help on using the repository browser.