62fc195d0f
This patch refactors the GDT initialization code in more of a self-documenting style.
148 lines
5.5 KiB
C
148 lines
5.5 KiB
C
/*
|
|
* 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)
|
|
);
|
|
}
|