source: trunk/hal/x86_64/core/hal_init.c @ 165

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

put the per-cpu structures into one structure

File size: 11.2 KB
RevLine 
[29]1/*
2 * hal_init.c - C initialization procedure for x86.
3 *
4 * Copyright (c) 2017 Maxime Villard
5 *
6 * This file is part of ALMOS-MKH.
7 *
8 * ALMOS-MKH is free software; you can redistribute it and/or modify it
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 *
12 * ALMOS-MKH is distributed in the hope that it will be useful, but
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
18 * along with ALMOS-MKH.; if not, write to the Free Software Foundation,
19 * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
20 */
21
22#include <hal_types.h>
23#include <hal_boot.h>
[32]24#include <hal_multiboot.h>
[29]25#include <hal_segmentation.h>
[45]26#include <hal_acpi.h>
[82]27#include <hal_apic.h>
[35]28#include <hal_internal.h>
[72]29#include <hal_remote.h>
[99]30#include <hal_irqmask.h>
[29]31
32#include <memcpy.h>
33#include <thread.h>
34#include <string.h>
35#include <process.h>
36#include <printk.h>
37#include <vmm.h>
38#include <core.h>
39#include <cluster.h>
[70]40#include <chdev.h>
[29]41
[70]42#include <boot_info.h>
43
[81]44void kernel_init(boot_info_t *info);
45
[29]46static void gdt_create();
47static void idt_create();
48void cpu_attach();
49
[44]50size_t mytest __in_kdata = 0;
[32]51
52struct multiboot_info mb_info __in_kdata;
53char mb_loader_name[PAGE_SIZE] __in_kdata;
54uint8_t mb_mmap[PAGE_SIZE] __in_kdata;
55
[29]56/* -------------------------------------------------------------------------- */
57
[32]58static void
59dump_memmap()
60{
61        size_t mmap_length = mb_info.mi_mmap_length;
62        uint8_t *mmap_addr = (uint8_t *)&mb_mmap;
63        size_t i;
64
65        if (!(mb_info.mi_flags & MULTIBOOT_INFO_HAS_MMAP))
[46]66                x86_panic("No mmap");
[32]67
68        i = 0;
69        while (i < mmap_length) {
70                struct multiboot_mmap *mm;
71
72                mm = (struct multiboot_mmap *)(mmap_addr + i);
73
74                x86_printf("-> [%Z, %Z] %s\n", mm->mm_base_addr,
75                    mm->mm_base_addr + mm->mm_length,
76                    (mm->mm_type == 1) ? "ram" : "rsv" );
77
78                i += mm->mm_size + 4;
79        }
80}
81
[135]82/* -------------------------------------------------------------------------- */
83
84static void init_bootinfo_icu(boot_device_t *dev)
85{
[137]86        extern uint32_t hwi_baseidx;
87        extern uint32_t wti_baseidx;
88        extern uint32_t pti_baseidx;
89        extern size_t ioapic_pins;
90
[135]91        memset(dev, 0, sizeof(boot_device_t));
92
[137]93        dev->base = 0;
[135]94        dev->type = (DEV_FUNC_ICU << 16) | IMPL_ICU_XCU;
95        dev->channels = 1;
[137]96
[138]97#if NOTYET
[137]98        /*
99         * Give 20% of the pins to HWI, 80% to WTI.
100         */
101        dev->param0 = (ioapic_pins * 20) / 100;  /* hwi_nr */
102        dev->param1 = ioapic_pins - dev->param0; /* wti_nr */
[138]103#else
104        dev->param0 = 1; /* hwi_nr */
105        dev->param1 = 1; /* wti_nr */
106#endif
[137]107
108        /*
109         * We always set 1 for pti_nr. On x86, timer interrupts are handled by
110         * LAPIC, which is per-cpu and not global. Therefore, we always have one
111         * timer for each CPU, and its IRQ number is faked to 0.
112         */
113        dev->param2 = 1; /* pti_nr */
114
[135]115        dev->param3 = 0;
116
[137]117        /* Set the base idx for the XCU driver */
118        hwi_baseidx = 0;
119        wti_baseidx = dev->param0;
120        pti_baseidx = 0xFFFFFFFF;
121
[135]122#ifdef NOTYET
123    uint32_t    irqs;                 /*! number of input IRQs                              */
124    boot_irq_t  irq[32];              /*! array of input IRQS (PIC and ICU only)            */
125#endif
126}
127
[119]128static size_t init_bootinfo_pages_nr()
129{
130        size_t mmap_length = mb_info.mi_mmap_length;
131        uint8_t *mmap_addr = (uint8_t *)&mb_mmap;
132        paddr_t maxpa, pa;
133        size_t i;
134
135        i = 0;
136        maxpa = 0;
137        while (i < mmap_length) {
138                struct multiboot_mmap *mm;
139
140                mm = (struct multiboot_mmap *)(mmap_addr + i);
141
142                if (mm->mm_type == 1) {
143                        pa = mm->mm_base_addr + mm->mm_length;
144                        if (pa > maxpa)
145                                maxpa = pa;
146                }
147
148                i += mm->mm_size + 4;
149        }
150
151        return (maxpa / PAGE_SIZE);
152}
153
[116]154static size_t init_bootinfo_rsvd(boot_rsvd_t *rsvd)
155{
156        size_t mmap_length = mb_info.mi_mmap_length;
157        uint8_t *mmap_addr = (uint8_t *)&mb_mmap;
158        size_t i, rsvd_nr;
159
[137]160        memset(rsvd, 0, sizeof(boot_rsvd_t) * CONFIG_PPM_MAX_RSVD);
[116]161
162        i = 0, rsvd_nr = 0;
163        while (i < mmap_length) {
164                struct multiboot_mmap *mm;
165
166                mm = (struct multiboot_mmap *)(mmap_addr + i);
167
[119]168                if (mm->mm_type != 1) {
169                        rsvd[rsvd_nr].first_page =
170                            rounddown(mm->mm_base_addr, PAGE_SIZE) / PAGE_SIZE;
171                        rsvd[rsvd_nr].npages =
172                            roundup(mm->mm_length, PAGE_SIZE) / PAGE_SIZE;
173                        rsvd_nr++;
174                        if (rsvd_nr == CONFIG_PPM_MAX_RSVD)
175                                x86_panic("too many memory holes");
176                }
[116]177
178                i += mm->mm_size + 4;
179        }
180
181        return rsvd_nr;
182}
183
[70]184static void init_bootinfo_core(boot_core_t *core)
185{
186        memset(core, 0, sizeof(boot_core_t));
187
188        core->gid = hal_lapic_gid();
189        core->lid = 0;
190        core->cxy = 0;
191}
192
193static void init_bootinfo_txt(boot_device_t *dev)
194{
195        memset(dev, 0, sizeof(boot_device_t));
196
197        dev->base = 0xB8000;
198        dev->type = (DEV_FUNC_TXT << 16) | IMPL_TXT_X86;
199        dev->channels = 1;
200        dev->param0 = 0;
201        dev->param1 = 0;
202        dev->param2 = 0;
203        dev->param3 = 0;
204
205#ifdef NOTYET
206    uint32_t    irqs;    /*! number of input IRQs                    */
207    boot_irq_t  irq[32]; /*! array of input IRQS (PIC and ICU only)  */
208#endif
209}
210
211static void init_bootinfo(boot_info_t *info)
212{
[116]213        size_t offset;
[114]214
[70]215        extern uint64_t __kernel_data_start;
216        extern uint64_t __kernel_end;
217
218        memset(info, 0, sizeof(boot_info_t));
219
220        info->signature = 0;
221
222        info->paddr_width = 0;
223        info->x_width = 1;
224        info->y_width = 1;
225        info->x_size = 1;
226        info->y_size = 1;
227        info->io_cxy = 0;
228
229        info->ext_dev_nr = 1;
230        init_bootinfo_txt(&info->ext_dev[0]);
231
232        info->cxy = 0;
233        info->cores_nr = 1;
234        init_bootinfo_core(&info->core[0]);
235
[137]236        info->rsvd_nr = init_bootinfo_rsvd((boot_rsvd_t *)&info->rsvd);
[70]237
[135]238        init_bootinfo_icu(&info->dev_icu);
239        /* TODO: dev_mmc */
240        /* TODO: dev_dma */
241
[116]242        offset = hal_gpt_bootstrap_uniformize();
243        info->pages_offset = offset / PAGE_SIZE;
[119]244        info->pages_nr = init_bootinfo_pages_nr();
[70]245
246        info->kernel_code_start = (intptr_t)(KERNTEXTOFF - KERNBASE);
247        info->kernel_code_end = (intptr_t)(&__kernel_data_start - KERNBASE) - 1;
248        info->kernel_data_start = (intptr_t)(&__kernel_data_start - KERNBASE);
249        info->kernel_code_end = (intptr_t)(&__kernel_end - KERNBASE) - 1;
250}
251
[29]252void init_x86_64(paddr_t firstpa)
253{
[70]254        boot_info_t btinfo;
255
[154]256        /* Initialize the serial port */
257        hal_com_init_early();
258
[29]259        x86_printf("[+] init_x86_64 called\n");
260
261        /* Create the global structures */
262        gdt_create();
263        idt_create();
264
265        /* Attach cpu0 */
[162]266        cpu_attach(0);
[29]267        x86_printf("[+] cpu_attach called\n");
268
[47]269        x86_printf("[+] bootloader: '%s'\n", mb_loader_name);
[32]270
271        dump_memmap();
272        x86_printf("[+] dump finished\n");
273
[35]274        hal_gpt_init(firstpa);
275        x86_printf("[+] hal_gpt_init called\n");
[29]276
[35]277        hal_acpi_init();
278        x86_printf("[+] hal_acpi_init called\n");
279
[45]280        hal_gpt_bootstrap_reset();
281        x86_printf("[+] hal_gpt_bootstrap_reset called\n");
282
[82]283        hal_apic_init();
284        x86_printf("[+] hal_apic_init called\n");
[45]285
[46]286        hal_tls_init_cpu0();
287        x86_printf("[+] hal_tls_init_cpu0 called\n");
288
[44]289        x86_printf("-> mytest = %z\n", mytest);
[94]290        void *hoho = &init_x86_64;
[74]291        xptr_t myptr = XPTR(0, &mytest);
[35]292
[94]293        hal_remote_spt(myptr, hoho);
294        x86_printf("-> mytest = %Z\n", hal_remote_lpt(myptr));
295
[99]296        init_bootinfo(&btinfo);
[94]297
[99]298        reg_t dummy;
299        hal_enable_irq(&dummy);
300
[152]301        while (1);
302
[70]303        kernel_init(&btinfo);
[99]304
[70]305        x86_printf("[+] kernel_init called\n");
306
[99]307        while (1);
308
[82]309//      void x86_stop();
310//      x86_stop();
[29]311}
312
313/* -------------------------------------------------------------------------- */
314
[165]315/* x86-specific per-cluster structures */
[29]316uint8_t gdtstore[PAGE_SIZE] __in_kdata;
317uint8_t idtstore[PAGE_SIZE] __in_kdata;
318
[165]319/* x86-specific per-cpu structures */
320typedef struct {
321        struct tss tss;
322        uint8_t intr_stack[STKSIZE];
323        uint8_t dbfl_stack[STKSIZE];
324        uint8_t nmfl_stack[STKSIZE];
325} percpu_archdata_t;
326percpu_archdata_t cpudata[CONFIG_MAX_LOCAL_CORES] __in_kdata;
327
[29]328static void
329setregion(struct region_descriptor *rd, void *base, uint16_t limit)
330{
331        rd->rd_limit = limit;
332        rd->rd_base = (uint64_t)base;
333}
334
335/* -------------------------------------------------------------------------- */
336
337static void
338gdt_set_memseg(struct gdt_memseg *sd, void *base, size_t limit,
339        int type, int dpl, int gran, int is64)
340{
341        sd->sd_lolimit = (unsigned)limit;
342        sd->sd_lobase = (unsigned long)base;
343        sd->sd_type = type;
344        sd->sd_dpl = dpl;
345        sd->sd_p = 1;
346        sd->sd_hilimit = (unsigned)limit >> 16;
347        sd->sd_avl = 0;
348        sd->sd_long = is64;
349        sd->sd_def32 = 0;
350        sd->sd_gran = gran;
351        sd->sd_hibase = (unsigned long)base >> 24;
352}
353
354static void
355gdt_set_sysseg(struct gdt_sysseg *sd, void *base, size_t limit,
356        int type, int dpl, int gran)
357{
358        memset(sd, 0, sizeof *sd);
359        sd->sd_lolimit = (unsigned)limit;
360        sd->sd_lobase = (uint64_t)base;
361        sd->sd_type = type;
362        sd->sd_dpl = dpl;
363        sd->sd_p = 1;
364        sd->sd_hilimit = (unsigned)limit >> 16;
365        sd->sd_gran = gran;
366        sd->sd_hibase = (uint64_t)base >> 24;
367}
368
369static void gdt_create()
370{
371        memset(&gdtstore, 0, PAGE_SIZE);
372
373        /* Flat segments */
374        gdt_set_memseg(GDT_ADDR_MEM(gdtstore, GDT_KCODE_SEL), 0,
375            0xfffff, SDT_MEMERA, SEL_KPL, 1, 1);
376        gdt_set_memseg(GDT_ADDR_MEM(gdtstore, GDT_KDATA_SEL), 0,
377            0xfffff, SDT_MEMRWA, SEL_KPL, 1, 1);
378        gdt_set_memseg(GDT_ADDR_MEM(gdtstore, GDT_UCODE_SEL), 0,
379            0xfffff, SDT_MEMERA, SEL_UPL, 1, 1);
380        gdt_set_memseg(GDT_ADDR_MEM(gdtstore, GDT_UDATA_SEL), 0,
381            0xfffff, SDT_MEMRWA, SEL_UPL, 1, 1);
382}
383
384void cpu_load_gdt()
385{
386        struct region_descriptor region;
387        setregion(&region, &gdtstore, PAGE_SIZE - 1);
388        lgdt(&region);
389}
390
391/* -------------------------------------------------------------------------- */
392
393static void
394idt_set_seg(struct idt_seg *seg, void *func, int ist, int type, int dpl, int sel)
395{
396        seg->gd_looffset = (uint64_t)func & 0xffff;
397        seg->gd_selector = sel;
398        seg->gd_ist = ist;
399        seg->gd_type = type;
400        seg->gd_dpl = dpl;
401        seg->gd_p = 1;
402        seg->gd_hioffset = (uint64_t)func >> 16;
403        seg->gd_zero = 0;
404        seg->gd_xx1 = 0;
405        seg->gd_xx2 = 0;
406        seg->gd_xx3 = 0;
407}
408
409static void idt_create()
410{
[80]411        extern uint64_t x86_traps[], x86_intrs[], x86_rsvd;
[29]412        struct idt_seg *idt;
413        size_t i;
414
415        idt = (struct idt_seg *)&idtstore;
[45]416
[80]417        /* First, put a dead entry */
418        for (i = 0; i < NIDT; i++) {
419                idt_set_seg(&idt[i], (void *)&x86_rsvd, 0,
420                    SDT_SYS386IGT, SEL_KPL, GDT_FIXED_SEL(GDT_KCODE_SEL, SEL_KPL));
421        }
422
[45]423        /* General exceptions */
424        for (i = CPUVEC_MIN; i < CPUVEC_MAX; i++) {
425                idt_set_seg(&idt[i], (void *)x86_traps[i - CPUVEC_MIN], 0,
426                    SDT_SYS386IGT, SEL_KPL, GDT_FIXED_SEL(GDT_KCODE_SEL, SEL_KPL));
[29]427        }
[45]428
[138]429        /* Dynamically configured interrupts */
430        for (i = DYNVEC_MIN; i < DYNVEC_MAX; i++) {
431                idt_set_seg(&idt[i], (void *)x86_intrs[i - DYNVEC_MIN], 0,
[45]432                    SDT_SYS386IGT, SEL_KPL, GDT_FIXED_SEL(GDT_KCODE_SEL, SEL_KPL));
433        }
[29]434}
435
436void cpu_load_idt()
437{
438        struct region_descriptor region;
439        setregion(&region, &idtstore, PAGE_SIZE - 1);
440        lidt(&region);
441}
442
443/* -------------------------------------------------------------------------- */
444
[164]445int tss_alloc(struct tss *tss, size_t lid)
[29]446{
447        int slot;
448
[164]449        slot = GDT_CPUTSS_SEL + lid;
[29]450
451        gdt_set_sysseg(GDT_ADDR_SYS(gdtstore, slot), tss,
452            sizeof(*tss) - 1, SDT_SYS386TSS, SEL_KPL, 0);
453
454        return GDT_DYNAM_SEL(slot, SEL_KPL);
455}
456
[162]457void cpu_create_tss(size_t lid)
[29]458{
[165]459        percpu_archdata_t *data = &cpudata[lid];
460        struct tss *tss = &data->tss;
[29]461        int sel;
462
463        /* Create the tss */
464        memset(tss, 0, sizeof(*tss));
[162]465
466        /* tss->tss_rsp0 */
[165]467        tss->tss_ist[0] = (uint64_t)data->intr_stack[lid] + STKSIZE;
468        tss->tss_ist[1] = (uint64_t)data->dbfl_stack[lid] + STKSIZE;
469        tss->tss_ist[2] = (uint64_t)data->nmfl_stack[lid] + STKSIZE;
[29]470        tss->tss_iobase = IOMAP_INVALOFF << 16;
[164]471        sel = tss_alloc(tss, lid);
[29]472
473        /* Load it */
474        ltr(sel);
475}
476
477/* -------------------------------------------------------------------------- */
478
[162]479void cpu_attach(size_t lid)
[29]480{
481        cpu_load_gdt();
482        cpu_load_idt();
[162]483        cpu_create_tss(lid);
[29]484}
485
Note: See TracBrowser for help on using the repository browser.