/* * Copyright (c) 2010-2015 Wind River Systems, Inc. * * SPDX-License-Identifier: Apache-2.0 */ /** * @file * @brief Crt0 module for the IA-32 boards * * This module contains the initial code executed by the Zephyr Kernel ELF image * after having been loaded into RAM. * * Note that most addresses (functions and variables) must be in physical * address space. Depending on page table setup, they may or may not be * available in virtual address space after loading of page table. */ #include #include #include #include #include #include #include /* exports (private APIs) */ GTEXT(__start) /* externs */ GTEXT(z_prep_c) GTEXT(z_bss_zero) GTEXT(z_data_copy) GDATA(_idt_base_address) GDATA(z_interrupt_stacks) GDATA(z_x86_idt) #ifndef CONFIG_GDT_DYNAMIC GDATA(_gdt) #endif #if defined(CONFIG_X86_SSE) GDATA(_sse_mxcsr_default_value) #endif #if defined(CONFIG_THREAD_LOCAL_STORAGE) GTEXT(z_x86_early_tls_update_gdt) #endif GDATA(x86_cpu_boot_arg) .macro install_page_tables #ifdef CONFIG_X86_MMU /* Enable paging. If virtual memory is enabled, the instruction pointer * is currently at a physical address. There is an identity mapping * for all RAM, plus a virtual mapping of RAM starting at * CONFIG_KERNEL_VM_BASE using the same paging structures. * * Until we enable these page tables, only physical memory addresses * work. */ movl $K_MEM_PHYS_ADDR(z_x86_kernel_ptables), %eax movl %eax, %cr3 #ifdef CONFIG_X86_PAE /* Enable PAE */ movl %cr4, %eax orl $CR4_PAE, %eax movl %eax, %cr4 /* IA32_EFER NXE bit set */ movl $0xC0000080, %ecx rdmsr orl $0x800, %eax wrmsr #else /* Enable Page Size Extensions (allowing 4MB pages). * This is ignored if PAE is enabled so no need to do * this above in PAE code. */ movl %cr4, %eax orl $CR4_PSE, %eax movl %eax, %cr4 #endif /* CONFIG_X86_PAE */ /* Enable paging (CR0.PG, bit 31) / write protect (CR0.WP, bit 16) */ movl %cr0, %eax orl $(CR0_PG | CR0_WP), %eax movl %eax, %cr0 #ifdef K_MEM_IS_VM_KERNEL /* Jump to a virtual address, which works because the identity and * virtual mappings both are to the same physical address. */ ljmp $CODE_SEG, $vm_enter vm_enter: /* We are now executing in virtual memory. We'll un-map the identity * mappings later once we are in the C domain */ #endif /* K_MEM_IS_VM_KERNEL */ #endif /* CONFIG_X86_MMU */ .endm SECTION_FUNC(BOOT_TEXT, __start) #include "../common.S" /* Enable write-back caching by clearing the NW and CD bits */ movl %cr0, %eax andl $0x9fffffff, %eax movl %eax, %cr0 /* * Ensure interrupts are disabled. Interrupts are enabled when * the first context switch occurs. */ cli /* * Although the bootloader sets up an Interrupt Descriptor Table (IDT) * and a Global Descriptor Table (GDT), the specification encourages * booted operating systems to setup their own IDT and GDT. */ #if CONFIG_SET_GDT /* load 32-bit operand size GDT */ lgdt K_MEM_PHYS_ADDR(_gdt_rom) /* If we set our own GDT, update the segment registers as well. */ movw $DATA_SEG, %ax /* data segment selector (entry = 3) */ movw %ax, %ds /* set DS */ movw %ax, %es /* set ES */ movw %ax, %ss /* set SS */ xorw %ax, %ax /* AX = 0 */ movw %ax, %fs /* Zero FS */ movw %ax, %gs /* Zero GS */ ljmp $CODE_SEG, $K_MEM_PHYS_ADDR(__csSet) /* set CS = 0x08 */ __csSet: #endif /* CONFIG_SET_GDT */ #if !defined(CONFIG_FPU) /* * Force an #NM exception for floating point instructions * since FP support hasn't been configured */ movl %cr0, %eax /* move CR0 to EAX */ orl $0x2e, %eax /* CR0[NE+TS+EM+MP]=1 */ movl %eax, %cr0 /* move EAX to CR0 */ #else /* * Permit use of x87 FPU instructions * * Note that all floating point exceptions are masked by default, * and that _no_ handler for x87 FPU exceptions (#MF) is provided. */ movl %cr0, %eax /* move CR0 to EAX */ orl $0x22, %eax /* CR0[NE+MP]=1 */ andl $~0xc, %eax /* CR0[TS+EM]=0 */ movl %eax, %cr0 /* move EAX to CR0 */ fninit /* set x87 FPU to its default state */ #if defined(CONFIG_X86_SSE) /* * Permit use of SSE instructions * * Note that all SSE exceptions are masked by default, * and that _no_ handler for SSE exceptions (#XM) is provided. */ movl %cr4, %eax /* move CR4 to EAX */ orl $0x200, %eax /* CR4[OSFXSR] = 1 */ andl $~0x400, %eax /* CR4[OSXMMEXCPT] = 0 */ movl %eax, %cr4 /* move EAX to CR4 */ /* initialize SSE control/status reg */ ldmxcsr K_MEM_PHYS_ADDR(_sse_mxcsr_default_value) #endif /* CONFIG_X86_SSE */ #endif /* !CONFIG_FPU */ /* * Set the stack pointer to the area used for the interrupt stack. * Note this stack is used during the execution of __start() and * z_cstart() until the multi-tasking kernel is initialized. The * dual-purposing of this area of memory is safe since * interrupts are disabled until the first context switch. * * kernel/init.c enforces that the z_interrupt_stacks pointer and * the ISR stack size are some multiple of ARCH_STACK_PTR_ALIGN, which * is at least 4. */ #ifdef CONFIG_INIT_STACKS movl $0xAAAAAAAA, %eax leal K_MEM_PHYS_ADDR(z_interrupt_stacks), %edi #ifdef CONFIG_X86_STACK_PROTECTION addl $4096, %edi #endif stack_size_dwords = (CONFIG_ISR_STACK_SIZE / 4) movl $stack_size_dwords, %ecx rep stosl #endif movl $K_MEM_PHYS_ADDR(z_interrupt_stacks), %esp #ifdef CONFIG_X86_STACK_PROTECTION /* In this configuration, all stacks, including IRQ stack, are declared * with a 4K non-present guard page preceding the stack buffer */ addl $(CONFIG_ISR_STACK_SIZE + 4096), %esp #else addl $CONFIG_ISR_STACK_SIZE, %esp #endif #ifdef CONFIG_XIP /* Copy data from flash to RAM. * * This is a must is CONFIG_GDT_DYNAMIC is enabled, * as _gdt needs to be in RAM. */ call z_data_copy #endif /* Note that installing page tables must be done after * z_data_copy() as the page tables are being copied into * RAM there. */ install_page_tables #ifdef CONFIG_GDT_DYNAMIC /* activate RAM-based Global Descriptor Table (GDT) */ lgdt %ds:_gdt #endif #if defined(CONFIG_X86_ENABLE_TSS) mov $MAIN_TSS, %ax ltr %ax #endif #ifdef K_MEM_IS_VM_KERNEL /* Need to reset the stack to virtual address after * page table is loaded. */ movl $z_interrupt_stacks, %esp #ifdef CONFIG_X86_STACK_PROTECTION addl $(CONFIG_ISR_STACK_SIZE + 4096), %esp #else addl $CONFIG_ISR_STACK_SIZE, %esp #endif #endif /* K_MEM_IS_VM_KERNEL */ #ifdef CONFIG_THREAD_LOCAL_STORAGE pushl %esp call z_x86_early_tls_update_gdt popl %esp #endif /* Clear BSS */ #ifdef CONFIG_LINKER_USE_BOOT_SECTION call z_bss_zero_boot #endif #ifdef CONFIG_LINKER_USE_PINNED_SECTION call z_bss_zero_pinned #endif #ifdef CONFIG_LINKER_GENERIC_SECTIONS_PRESENT_AT_BOOT /* Don't clear BSS if the section is not present * in memory at boot. Or else it would cause page * faults. Zeroing BSS will be done later once the * paging mechanism has been initialized. */ call z_bss_zero #endif /* load 32-bit operand size IDT */ lidt z_x86_idt movl $x86_cpu_boot_arg, %ebp /* Boot type to multiboot, ebx content will help to mitigate */ movl $MULTIBOOT_BOOT_TYPE, \ __x86_boot_arg_t_boot_type_OFFSET(%ebp) /* pointer to multiboot info, or NULL */ movl %ebx, __x86_boot_arg_t_arg_OFFSET(%ebp) pushl $x86_cpu_boot_arg call z_prep_c /* enter kernel; never returns */ #if defined(CONFIG_X86_SSE) /* SSE control & status register initial value */ _sse_mxcsr_default_value: .long 0x1f80 /* all SSE exceptions clear & masked */ #endif /* CONFIG_X86_SSE */ /* Interrupt Descriptor Table (IDT) definition */ z_x86_idt: .word (CONFIG_IDT_NUM_VECTORS * 8) - 1 /* limit: size of IDT-1 */ /* * Physical start address = 0. When executing natively, this * will be placed at the same location as the interrupt vector table * setup by the BIOS (or GRUB?). */ /* IDT table start address */ .long _idt_base_address #ifdef CONFIG_SET_GDT /* * The following 3 GDT entries implement the so-called "basic * flat model", i.e. a single code segment descriptor and a single * data segment descriptor, giving the kernel access to a continuous, * unsegmented address space. Both segment descriptors map the entire * linear address space (i.e. 0 to 4 GB-1), thus the segmentation * mechanism will never generate "out of limit memory reference" * exceptions even if physical memory does not reside at the referenced * address. * * The 'A' (accessed) bit in the type field is set for all the * data/code segment descriptors to accommodate placing these entries * in ROM, to prevent the processor from freaking out when it tries * and fails to set it. */ SECTION_VAR(PINNED_RODATA, _gdt_rom) #ifndef CONFIG_GDT_DYNAMIC _gdt: #endif /* GDT should be aligned on 8-byte boundary for best processor * performance, see Section 3.5.1 of IA architecture SW developer * manual, Vol 3. */ .balign 8 /* Entry 0 (selector=0x0000): The "NULL descriptor". The CPU never * actually looks at this entry, so we stuff 6-byte the pseudo * descriptor here */ /* Limit on GDT */ .word K_MEM_PHYS_ADDR(_gdt_rom_end) - K_MEM_PHYS_ADDR(_gdt_rom) - 1 /* table address: _gdt_rom */ .long K_MEM_PHYS_ADDR(_gdt_rom) .word 0x0000 /* Entry 1 (selector=0x0008): Code descriptor: DPL0 */ .word 0xffff /* limit: xffff */ .word 0x0000 /* base : xxxx0000 */ .byte 0x00 /* base : xx00xxxx */ .byte 0x9b /* Accessed, Code e/r, Present, DPL0 */ .byte 0xcf /* limit: fxxxx, Page Gra, 32bit */ .byte 0x00 /* base : 00xxxxxx */ /* Entry 2 (selector=0x0010): Data descriptor: DPL0 */ .word 0xffff /* limit: xffff */ .word 0x0000 /* base : xxxx0000 */ .byte 0x00 /* base : xx00xxxx */ .byte 0x93 /* Accessed, Data r/w, Present, DPL0 */ .byte 0xcf /* limit: fxxxx, Page Gra, 32bit */ .byte 0x00 /* base : 00xxxxxx */ _gdt_rom_end: #endif