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

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

start moving the APIC code into the PIC driver

File size: 13.6 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>
[166]29#include <hal_register.h>
30
[72]31#include <hal_remote.h>
[99]32#include <hal_irqmask.h>
[29]33
34#include <memcpy.h>
35#include <thread.h>
36#include <string.h>
37#include <process.h>
38#include <printk.h>
39#include <vmm.h>
40#include <core.h>
41#include <cluster.h>
[70]42#include <chdev.h>
[29]43
[70]44#include <boot_info.h>
45
[81]46void kernel_init(boot_info_t *info);
47
[29]48static void gdt_create();
49static void idt_create();
[168]50void cpu_tls_init(size_t lid);
[166]51void cpu_identify();
[29]52void cpu_attach();
53
[44]54size_t mytest __in_kdata = 0;
[32]55
56struct multiboot_info mb_info __in_kdata;
57char mb_loader_name[PAGE_SIZE] __in_kdata;
58uint8_t mb_mmap[PAGE_SIZE] __in_kdata;
59
[29]60/* -------------------------------------------------------------------------- */
61
[32]62static void
63dump_memmap()
64{
65        size_t mmap_length = mb_info.mi_mmap_length;
66        uint8_t *mmap_addr = (uint8_t *)&mb_mmap;
67        size_t i;
68
69        if (!(mb_info.mi_flags & MULTIBOOT_INFO_HAS_MMAP))
[46]70                x86_panic("No mmap");
[32]71
72        i = 0;
73        while (i < mmap_length) {
74                struct multiboot_mmap *mm;
75
76                mm = (struct multiboot_mmap *)(mmap_addr + i);
77
78                x86_printf("-> [%Z, %Z] %s\n", mm->mm_base_addr,
79                    mm->mm_base_addr + mm->mm_length,
80                    (mm->mm_type == 1) ? "ram" : "rsv" );
81
82                i += mm->mm_size + 4;
83        }
84}
85
[135]86/* -------------------------------------------------------------------------- */
87
[119]88static size_t init_bootinfo_pages_nr()
89{
90        size_t mmap_length = mb_info.mi_mmap_length;
91        uint8_t *mmap_addr = (uint8_t *)&mb_mmap;
92        paddr_t maxpa, pa;
93        size_t i;
94
95        i = 0;
96        maxpa = 0;
97        while (i < mmap_length) {
98                struct multiboot_mmap *mm;
99
100                mm = (struct multiboot_mmap *)(mmap_addr + i);
101
102                if (mm->mm_type == 1) {
103                        pa = mm->mm_base_addr + mm->mm_length;
104                        if (pa > maxpa)
105                                maxpa = pa;
106                }
107
108                i += mm->mm_size + 4;
109        }
110
111        return (maxpa / PAGE_SIZE);
112}
113
[116]114static size_t init_bootinfo_rsvd(boot_rsvd_t *rsvd)
115{
116        size_t mmap_length = mb_info.mi_mmap_length;
117        uint8_t *mmap_addr = (uint8_t *)&mb_mmap;
118        size_t i, rsvd_nr;
119
[137]120        memset(rsvd, 0, sizeof(boot_rsvd_t) * CONFIG_PPM_MAX_RSVD);
[116]121
122        i = 0, rsvd_nr = 0;
123        while (i < mmap_length) {
124                struct multiboot_mmap *mm;
125
126                mm = (struct multiboot_mmap *)(mmap_addr + i);
127
[119]128                if (mm->mm_type != 1) {
129                        rsvd[rsvd_nr].first_page =
130                            rounddown(mm->mm_base_addr, PAGE_SIZE) / PAGE_SIZE;
131                        rsvd[rsvd_nr].npages =
132                            roundup(mm->mm_length, PAGE_SIZE) / PAGE_SIZE;
133                        rsvd_nr++;
134                        if (rsvd_nr == CONFIG_PPM_MAX_RSVD)
135                                x86_panic("too many memory holes");
136                }
[116]137
138                i += mm->mm_size + 4;
139        }
140
141        return rsvd_nr;
142}
143
[70]144static void init_bootinfo_core(boot_core_t *core)
145{
146        memset(core, 0, sizeof(boot_core_t));
147
148        core->gid = hal_lapic_gid();
149        core->lid = 0;
150        core->cxy = 0;
151}
152
[195]153static void init_bootinfo_ioc(boot_device_t *dev)
154{
155        memset(dev, 0, sizeof(boot_device_t));
156
157        dev->base = 0;
158        dev->type = (DEV_FUNC_IOC << 16) | IMPL_IOC_BDV;
159        dev->channels = 1;
160}
161
[192]162static void init_bootinfo_pic(boot_device_t *dev)
[70]163{
164        memset(dev, 0, sizeof(boot_device_t));
165
[192]166        dev->base = 0;
167        dev->type = (DEV_FUNC_PIC << 16) | IMPL_PIC_SCL;
[70]168        dev->channels = 1;
169        dev->param0 = 0;
170        dev->param1 = 0;
171        dev->param2 = 0;
172        dev->param3 = 0;
173
[202]174        dev->irqs = 16;
[192]175
176        /* COM1 */
[202]177        dev->irq[IRQ_COM1].dev_type = (DEV_FUNC_TXT << 16) | IMPL_TXT_TTY;
178        dev->irq[IRQ_COM1].channel = 0;
179        dev->irq[IRQ_COM1].is_rx = 0;
180        dev->irq[IRQ_COM1].valid = 1;
181
182        /* ATA */
183        dev->irq[IRQ_ATA0].dev_type = (DEV_FUNC_IOC << 16) | IMPL_IOC_BDV;
184        dev->irq[IRQ_ATA0].channel = 0;
185        dev->irq[IRQ_ATA0].is_rx = 0;
186        dev->irq[IRQ_ATA0].valid = 1;
[70]187}
188
[192]189static void init_bootinfo_txt(boot_device_t *dev)
190{
191        memset(dev, 0, sizeof(boot_device_t));
192
193        dev->base = 0;
194        dev->type = (DEV_FUNC_TXT << 16) | IMPL_TXT_TTY;
195        dev->channels = 1;
196        dev->param0 = 0;
197        dev->param1 = 0;
198        dev->param2 = 0;
199        dev->param3 = 0;
200}
201
[70]202static void init_bootinfo(boot_info_t *info)
203{
[116]204        size_t offset;
[114]205
[70]206        extern uint64_t __kernel_data_start;
207        extern uint64_t __kernel_end;
208
209        memset(info, 0, sizeof(boot_info_t));
210
211        info->signature = 0;
212
213        info->paddr_width = 0;
214        info->x_width = 1;
215        info->y_width = 1;
216        info->x_size = 1;
217        info->y_size = 1;
218        info->io_cxy = 0;
219
[195]220        info->ext_dev_nr = 3;
[70]221        init_bootinfo_txt(&info->ext_dev[0]);
[192]222        init_bootinfo_pic(&info->ext_dev[1]);
[195]223        init_bootinfo_ioc(&info->ext_dev[2]);
[70]224
225        info->cxy = 0;
226        info->cores_nr = 1;
227        init_bootinfo_core(&info->core[0]);
228
[137]229        info->rsvd_nr = init_bootinfo_rsvd((boot_rsvd_t *)&info->rsvd);
[70]230
[192]231        /* TODO: dev_icu */
[135]232        /* TODO: dev_mmc */
233        /* TODO: dev_dma */
234
[116]235        offset = hal_gpt_bootstrap_uniformize();
236        info->pages_offset = offset / PAGE_SIZE;
[119]237        info->pages_nr = init_bootinfo_pages_nr();
[70]238
239        info->kernel_code_start = (intptr_t)(KERNTEXTOFF - KERNBASE);
240        info->kernel_code_end = (intptr_t)(&__kernel_data_start - KERNBASE) - 1;
241        info->kernel_data_start = (intptr_t)(&__kernel_data_start - KERNBASE);
242        info->kernel_code_end = (intptr_t)(&__kernel_end - KERNBASE) - 1;
243}
244
[166]245/* -------------------------------------------------------------------------- */
246
[199]247static void apic_map()
248{
249        extern vaddr_t lapic_va, ioapic_va;
250        extern paddr_t lapic_pa, ioapic_pa;
251
252        lapic_va = hal_gpt_bootstrap_valloc(1); // XXX: should be shared
253        hal_gpt_enter(lapic_va, lapic_pa, PG_V|PG_KW|PG_NX|PG_N);
254
255        ioapic_va = hal_gpt_bootstrap_valloc(1); // XXX: should be shared
256        hal_gpt_enter(ioapic_va, ioapic_pa, PG_V|PG_KW|PG_NX|PG_N);
257}
258
[29]259void init_x86_64(paddr_t firstpa)
260{
[70]261        boot_info_t btinfo;
262
[154]263        /* Initialize the serial port */
264        hal_com_init_early();
265
[29]266        x86_printf("[+] init_x86_64 called\n");
267
268        /* Create the global structures */
269        gdt_create();
270        idt_create();
271
[166]272        /* Identify the features of the cpu */
273        cpu_identify();
274
[29]275        /* Attach cpu0 */
[162]276        cpu_attach(0);
[29]277        x86_printf("[+] cpu_attach called\n");
278
[47]279        x86_printf("[+] bootloader: '%s'\n", mb_loader_name);
[32]280
281        dump_memmap();
282        x86_printf("[+] dump finished\n");
283
[35]284        hal_gpt_init(firstpa);
285        x86_printf("[+] hal_gpt_init called\n");
[29]286
[35]287        hal_acpi_init();
288        x86_printf("[+] hal_acpi_init called\n");
289
[45]290        hal_gpt_bootstrap_reset();
291        x86_printf("[+] hal_gpt_bootstrap_reset called\n");
292
[199]293        apic_map();
294        x86_printf("[+] apic_map called\n");
295
[82]296        hal_apic_init();
297        x86_printf("[+] hal_apic_init called\n");
[45]298
[168]299        cpu_tls_init(0);
300        x86_printf("[+] cput_tls_init called\n");
[46]301
[44]302        x86_printf("-> mytest = %z\n", mytest);
[94]303        void *hoho = &init_x86_64;
[74]304        xptr_t myptr = XPTR(0, &mytest);
[35]305
[94]306        hal_remote_spt(myptr, hoho);
307        x86_printf("-> mytest = %Z\n", hal_remote_lpt(myptr));
308
[99]309        init_bootinfo(&btinfo);
[94]310
[99]311        reg_t dummy;
312        hal_enable_irq(&dummy);
313
[70]314        kernel_init(&btinfo);
[99]315
[70]316        x86_printf("[+] kernel_init called\n");
[192]317/*
318        void *ptr;
[70]319
[192]320        khm_t *khm = &LOCAL_CLUSTER->khm;
321        ptr = khm_alloc(khm, 10);
322        memset(ptr, 0, 10);
323        khm_free(ptr);
324
325
326        kcm_t *kcm = &LOCAL_CLUSTER->kcm;
327        ptr = kcm_alloc(kcm);
328        memset(ptr, 0, 1);
329        kcm_free(ptr);
330
331        ptr = ppm_alloc_pages(1);
332        ppm_free_pages(ptr);
333*/
[99]334        while (1);
335
[82]336//      void x86_stop();
337//      x86_stop();
[29]338}
339
340/* -------------------------------------------------------------------------- */
341
[165]342/* x86-specific per-cluster structures */
[29]343uint8_t gdtstore[PAGE_SIZE] __in_kdata;
344uint8_t idtstore[PAGE_SIZE] __in_kdata;
345
[165]346/* x86-specific per-cpu structures */
347typedef struct {
348        struct tss tss;
[168]349        struct tls tls;
[165]350        uint8_t intr_stack[STKSIZE];
351        uint8_t dbfl_stack[STKSIZE];
352        uint8_t nmfl_stack[STKSIZE];
353} percpu_archdata_t;
354percpu_archdata_t cpudata[CONFIG_MAX_LOCAL_CORES] __in_kdata;
355
[29]356static void
357setregion(struct region_descriptor *rd, void *base, uint16_t limit)
358{
359        rd->rd_limit = limit;
360        rd->rd_base = (uint64_t)base;
361}
362
363/* -------------------------------------------------------------------------- */
364
365static void
366gdt_set_memseg(struct gdt_memseg *sd, void *base, size_t limit,
367        int type, int dpl, int gran, int is64)
368{
369        sd->sd_lolimit = (unsigned)limit;
370        sd->sd_lobase = (unsigned long)base;
371        sd->sd_type = type;
372        sd->sd_dpl = dpl;
373        sd->sd_p = 1;
374        sd->sd_hilimit = (unsigned)limit >> 16;
375        sd->sd_avl = 0;
376        sd->sd_long = is64;
377        sd->sd_def32 = 0;
378        sd->sd_gran = gran;
379        sd->sd_hibase = (unsigned long)base >> 24;
380}
381
382static void
383gdt_set_sysseg(struct gdt_sysseg *sd, void *base, size_t limit,
384        int type, int dpl, int gran)
385{
386        memset(sd, 0, sizeof *sd);
387        sd->sd_lolimit = (unsigned)limit;
388        sd->sd_lobase = (uint64_t)base;
389        sd->sd_type = type;
390        sd->sd_dpl = dpl;
391        sd->sd_p = 1;
392        sd->sd_hilimit = (unsigned)limit >> 16;
393        sd->sd_gran = gran;
394        sd->sd_hibase = (uint64_t)base >> 24;
395}
396
397static void gdt_create()
398{
399        memset(&gdtstore, 0, PAGE_SIZE);
400
401        /* Flat segments */
402        gdt_set_memseg(GDT_ADDR_MEM(gdtstore, GDT_KCODE_SEL), 0,
403            0xfffff, SDT_MEMERA, SEL_KPL, 1, 1);
404        gdt_set_memseg(GDT_ADDR_MEM(gdtstore, GDT_KDATA_SEL), 0,
405            0xfffff, SDT_MEMRWA, SEL_KPL, 1, 1);
406        gdt_set_memseg(GDT_ADDR_MEM(gdtstore, GDT_UCODE_SEL), 0,
407            0xfffff, SDT_MEMERA, SEL_UPL, 1, 1);
408        gdt_set_memseg(GDT_ADDR_MEM(gdtstore, GDT_UDATA_SEL), 0,
409            0xfffff, SDT_MEMRWA, SEL_UPL, 1, 1);
410}
411
412void cpu_load_gdt()
413{
414        struct region_descriptor region;
415        setregion(&region, &gdtstore, PAGE_SIZE - 1);
416        lgdt(&region);
417}
418
419/* -------------------------------------------------------------------------- */
420
[203]421struct {
422        bool_t busy[256];
423} idt_bitmap __in_kdata;
424
425int idt_slot_alloc()
426{
427        size_t i;
428
429        for (i = 0; i < 256; i++) {
430                if (!idt_bitmap.busy[i])
431                        break;
432        }
433        if (i == 256) {
434                return -1;
435        }
436
437        idt_bitmap.busy[i] = true;
438        return (int)i;
439}
440
441void idt_slot_free(int slot)
442{
443        idt_bitmap.busy[slot] = false;
444}
445
[29]446static void
447idt_set_seg(struct idt_seg *seg, void *func, int ist, int type, int dpl, int sel)
448{
449        seg->gd_looffset = (uint64_t)func & 0xffff;
450        seg->gd_selector = sel;
451        seg->gd_ist = ist;
452        seg->gd_type = type;
453        seg->gd_dpl = dpl;
454        seg->gd_p = 1;
455        seg->gd_hioffset = (uint64_t)func >> 16;
456        seg->gd_zero = 0;
457        seg->gd_xx1 = 0;
458        seg->gd_xx2 = 0;
459        seg->gd_xx3 = 0;
460}
461
462static void idt_create()
463{
[80]464        extern uint64_t x86_traps[], x86_intrs[], x86_rsvd;
[29]465        struct idt_seg *idt;
466        size_t i;
467
[203]468        memset(&idt_bitmap, 0, sizeof(idt_bitmap));
[29]469        idt = (struct idt_seg *)&idtstore;
[45]470
[80]471        /* First, put a dead entry */
472        for (i = 0; i < NIDT; i++) {
473                idt_set_seg(&idt[i], (void *)&x86_rsvd, 0,
474                    SDT_SYS386IGT, SEL_KPL, GDT_FIXED_SEL(GDT_KCODE_SEL, SEL_KPL));
475        }
476
[45]477        /* General exceptions */
478        for (i = CPUVEC_MIN; i < CPUVEC_MAX; i++) {
479                idt_set_seg(&idt[i], (void *)x86_traps[i - CPUVEC_MIN], 0,
480                    SDT_SYS386IGT, SEL_KPL, GDT_FIXED_SEL(GDT_KCODE_SEL, SEL_KPL));
[203]481                idt_bitmap.busy[i] = true;
[29]482        }
[45]483
[138]484        /* Dynamically configured interrupts */
485        for (i = DYNVEC_MIN; i < DYNVEC_MAX; i++) {
486                idt_set_seg(&idt[i], (void *)x86_intrs[i - DYNVEC_MIN], 0,
[45]487                    SDT_SYS386IGT, SEL_KPL, GDT_FIXED_SEL(GDT_KCODE_SEL, SEL_KPL));
[203]488                idt_bitmap.busy[i] = true;
[45]489        }
[29]490}
491
492void cpu_load_idt()
493{
494        struct region_descriptor region;
495        setregion(&region, &idtstore, PAGE_SIZE - 1);
496        lidt(&region);
497}
498
499/* -------------------------------------------------------------------------- */
500
[164]501int tss_alloc(struct tss *tss, size_t lid)
[29]502{
503        int slot;
504
[164]505        slot = GDT_CPUTSS_SEL + lid;
[29]506
507        gdt_set_sysseg(GDT_ADDR_SYS(gdtstore, slot), tss,
508            sizeof(*tss) - 1, SDT_SYS386TSS, SEL_KPL, 0);
509
510        return GDT_DYNAM_SEL(slot, SEL_KPL);
511}
512
[162]513void cpu_create_tss(size_t lid)
[29]514{
[165]515        percpu_archdata_t *data = &cpudata[lid];
516        struct tss *tss = &data->tss;
[29]517        int sel;
518
519        /* Create the tss */
520        memset(tss, 0, sizeof(*tss));
[162]521
522        /* tss->tss_rsp0 */
[165]523        tss->tss_ist[0] = (uint64_t)data->intr_stack[lid] + STKSIZE;
524        tss->tss_ist[1] = (uint64_t)data->dbfl_stack[lid] + STKSIZE;
525        tss->tss_ist[2] = (uint64_t)data->nmfl_stack[lid] + STKSIZE;
[29]526        tss->tss_iobase = IOMAP_INVALOFF << 16;
[164]527        sel = tss_alloc(tss, lid);
[29]528
529        /* Load it */
530        ltr(sel);
531}
532
533/* -------------------------------------------------------------------------- */
534
[168]535void cpu_tls_init(size_t lid)
536{
537        percpu_archdata_t *data = &cpudata[lid];
538        tls_t *cputls = &data->tls;
539
540        memset(cputls, 0, sizeof(tls_t));
541
542        cputls->tls_self = cputls;
543        cputls->tls_gid = hal_lapic_gid();
544        cputls->tls_lid = lid;
545
546        wrmsr(MSR_FSBASE, 0);
547        wrmsr(MSR_GSBASE, (uint64_t)cputls);
548        wrmsr(MSR_KERNELGSBASE, 0);
549}
550
551/* -------------------------------------------------------------------------- */
552
[166]553uint64_t cpu_features[4] __in_kdata;
554
555void cpu_identify()
556{
557        /*
558         * desc[0] = eax
559         * desc[1] = ebx
560         * desc[2] = ecx
561         * desc[3] = edx
562         */
563        uint32_t desc[4];
564        char vendor[13];
565        size_t lvl;
566
567        /*
568         * Get information from the standard cpuid leafs
569         */
570        cpuid(0, 0, (uint32_t *)&desc);
571
572        lvl = (uint64_t)desc[0];
573        x86_printf("-> cpuid standard level: %z\n", lvl);
574
575        memcpy(vendor + 0, &desc[1], sizeof(uint32_t));
576        memcpy(vendor + 8, &desc[2], sizeof(uint32_t));
577        memcpy(vendor + 4, &desc[3], sizeof(uint32_t));
578        vendor[12] = '\0';
579        x86_printf("-> CPU vendor: '%s'\n", vendor);
580
581        if (lvl >= 1) {
582                cpuid(1, 0, (uint32_t *)&desc);
583                cpu_features[0] = desc[3];
584                cpu_features[1] = desc[2];
585        }
586
587        /*
588         * Get information from the extended cpuid leafs
589         */
590        cpuid(0x80000000, 0, desc);
591
592        lvl = (uint64_t)desc[0];
593        x86_printf("-> cpuid extended level: %Z\n", lvl);
594}
595
596/* -------------------------------------------------------------------------- */
597
[162]598void cpu_attach(size_t lid)
[29]599{
[168]600        /* Per-cluster structures */
[29]601        cpu_load_gdt();
602        cpu_load_idt();
[168]603
604        /* Per-cpu structures */
[162]605        cpu_create_tss(lid);
[166]606
607        if (cpu_features[0] & CPUID_PSE) {
608                lcr4(rcr4() | CR4_PSE);
609                tlbflushg();
610        } else {
611                /*
612                 * amd64 supports PSE by default, if it's not here we have a
613                 * problem
614                 */
615                x86_panic("PSE not supported");
616        }
[29]617}
618
Note: See TracBrowser for help on using the repository browser.