/* * Copyright (C) 2015, Intel Corporation. All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * 1. Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * 2. Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in the * documentation and/or other materials provided with the distribution. * * 3. Neither the name of the copyright holder nor the names of its * contributors may be used to endorse or promote products derived * from this software without specific prior written permission. * * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE * COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED * OF THE POSSIBILITY OF SUCH DAMAGE. */ #include <stdint.h> #define NUM_DESC 3 #define GDT_IDX_NULL 0 #define GDT_IDX_CODE 1 #define GDT_IDX_DATA 2 /* All code in the x86 port of Contiki runs at ring (privilege) level 0 */ #define PRIV_LVL 0 /* Compute GDT selector from descriptor index and requested privilege level */ #define GDT_SEL(IDX, RPL) (((IDX) << 3) | (RPL)) #define GDT_SEL_NULL GDT_SEL(GDT_IDX_NULL, 0) #define GDT_SEL_CODE GDT_SEL(GDT_IDX_CODE, PRIV_LVL) #define GDT_SEL_DATA GDT_SEL(GDT_IDX_DATA, PRIV_LVL) /* Each define here is for a specific flag in the descriptor. Refer to Intel * Combined Manual (Intel 64 and IA-32 Architectures Software Developer's * Manual), Vol. 3, Section 3.4.5 for a description of each flag. */ #define SEG_DESCTYPE(x) ((x) << 0x04) /* Descriptor type (0 for system, 1 for code/data) */ #define SEG_PRES(x) ((x) << 0x07) /* Present */ #define SEG_SAVL(x) ((x) << 0x0C) /* Available for system use */ #define SEG_LONG(x) ((x) << 0x0D) /* Long mode */ #define SEG_SIZE(x) ((x) << 0x0E) /* Size (0 for 16-bit, 1 for 32) */ #define SEG_GRAN(x) ((x) << 0x0F) /* Granularity (0 for 1B - 1MB, 1 for 4KB - 4GB) */ #define SEG_PRIV(x) (((x) & 0x03) << 0x05) /* Set privilege level (0 - 3) */ #define SEG_DATA_RDWR 0x02 /* Read/Write */ #define SEG_CODE_EXRD 0x0A /* Execute/Read */ #define GDT_CODE_PL0 SEG_DESCTYPE(1) | SEG_PRES(1) | SEG_SAVL(0) | \ SEG_LONG(0) | SEG_SIZE(1) | SEG_GRAN(1) | \ SEG_PRIV(0) | SEG_CODE_EXRD #define GDT_DATA_PL0 SEG_DESCTYPE(1) | SEG_PRES(1) | SEG_SAVL(0) | \ SEG_LONG(0) | SEG_SIZE(1) | SEG_GRAN(1) | \ SEG_PRIV(0) | SEG_DATA_RDWR typedef struct gdtr { uint16_t limit; uint32_t base; } __attribute__((packed)) gdtr_t; typedef uint64_t segment_desc_t; /* From Intel Combined Manual, Vol. 3 , Section 3.5.1: The base addresses of * the GDT should be aligned on an eight-byte boundary to yield the best * processor performance. */ static segment_desc_t gdt[NUM_DESC] __attribute__ ((aligned (8))); static void set_descriptor(unsigned int index, uint32_t base, uint32_t limit, uint16_t flag) { segment_desc_t descriptor; if (index >= NUM_DESC) return; /* Create the high 32 bit segment */ descriptor = limit & 0x000F0000; /* set limit bits 19:16 */ descriptor |= (flag << 8) & 0x00F0FF00; /* set type, p, dpl, s, g, d/b, l and avl fields */ descriptor |= (base >> 16) & 0x000000FF; /* set base bits 23:16 */ descriptor |= base & 0xFF000000; /* set base bits 31:24 */ /* Shift by 32 to allow for low part of segment */ descriptor <<= 32; /* Create the low 32 bit segment */ descriptor |= base << 16; /* set base bits 15:0 */ descriptor |= limit & 0x0000FFFF; /* set limit bits 15:0 */ /* Save descriptor into gdt */ gdt[index] = descriptor; } /* This function initializes the Global Offset Table. For simplicity, the * memory is organized following the flat model. Thus, memory appears to * Contiki as a single continuous address space. Code, data, and stack * are all contained in this address space (so called linear address space). */ void gdt_init(void) { gdtr_t gdtr; /* Initialize gdtr structure */ gdtr.limit = sizeof(segment_desc_t) * NUM_DESC - 1; gdtr.base = (uint32_t) &gdt; /* Initialize descriptors */ set_descriptor(GDT_IDX_NULL, 0, 0, 0); set_descriptor(GDT_IDX_CODE, 0, 0x0FFFFF, GDT_CODE_PL0); set_descriptor(GDT_IDX_DATA, 0, 0x0FFFFF, GDT_DATA_PL0); /* Load GDTR register and update segment registers. * * CS register cannot be changed directly. For that reason, we do a far jump. */ __asm__ ("lgdt %[_gdtr_]\n\t" "jmp %[_cs_], $1f\n\t" "1:\n\t" "mov %[_ds_], %%ds\n\t" "mov %[_ds_], %%ss\n\t" "mov %[_ds_], %%es\n\t" "mov %[_ds_], %%fs\n\t" "mov %[_ds_], %%gs\n\t" : : [_gdtr_] "m" (gdtr), [_cs_] "i" (GDT_SEL_CODE), [_ds_] "r" (GDT_SEL_DATA) ); }