2014-01-16 07:39:33 +01:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2014 René Kijewski <rene.kijewski@fu-berlin.de>
|
|
|
|
*
|
|
|
|
* This library is free software; you can redistribute it and/or
|
|
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
|
|
* License as published by the Free Software Foundation; either
|
|
|
|
* version 2.1 of the License, or (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This library is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
|
|
* Lesser General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
|
|
* License along with this library; if not, write to the Free Software
|
|
|
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Basic definitions and forwards for x86 boards.
|
|
|
|
*
|
|
|
|
* @defgroup x86 i586
|
|
|
|
* @ingroup cpu
|
|
|
|
* @{
|
|
|
|
* @file
|
|
|
|
* @author René Kijewski <rene.kijewski@fu-berlin.de>
|
|
|
|
*/
|
|
|
|
|
2015-03-11 06:35:17 +01:00
|
|
|
#ifndef CPU_X86_CPU_H_
|
|
|
|
#define CPU_X86_CPU_H_
|
2014-01-16 07:39:33 +01:00
|
|
|
|
2016-03-09 01:27:23 +01:00
|
|
|
#include "kernel_defines.h"
|
2014-01-16 07:39:33 +01:00
|
|
|
#include "irq.h"
|
|
|
|
#include "ucontext.h"
|
2015-05-22 14:31:23 +02:00
|
|
|
#include "cpu_conf.h"
|
2014-01-16 07:39:33 +01:00
|
|
|
|
|
|
|
#include <stdbool.h>
|
|
|
|
#include <stdint.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
|
2014-10-13 10:53:20 +02:00
|
|
|
#ifdef __cplusplus
|
|
|
|
extern "C" {
|
|
|
|
#endif
|
|
|
|
|
2015-04-28 22:57:56 +02:00
|
|
|
/**
|
2015-04-26 09:20:59 +02:00
|
|
|
* @brief x86 has architecture specific atomic_cas in x86_atomic.c
|
2015-05-26 16:45:34 +02:00
|
|
|
* @{
|
2015-04-28 22:57:56 +02:00
|
|
|
*/
|
2015-03-11 06:35:17 +01:00
|
|
|
#define ARCH_HAS_ATOMIC_COMPARE_AND_SWAP 1
|
2015-05-26 16:45:34 +02:00
|
|
|
/** @} */
|
2015-04-28 22:57:56 +02:00
|
|
|
|
2014-01-16 07:39:33 +01:00
|
|
|
/**
|
|
|
|
* @brief Disable interrupts and halt forever.
|
|
|
|
*
|
|
|
|
* This function is the last resort in case of an unrecoverable error.
|
|
|
|
* No message will be printed, no nothing.
|
|
|
|
*/
|
|
|
|
static inline void __attribute__((always_inline, noreturn)) x86_hlt(void)
|
|
|
|
{
|
|
|
|
while (1) {
|
2016-03-14 14:28:00 +01:00
|
|
|
__asm__ volatile ("cli");
|
|
|
|
__asm__ volatile ("hlt");
|
2014-01-16 07:39:33 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Initialize subsystems and run kernel.
|
|
|
|
*
|
|
|
|
* Called by the board specific startup code.
|
2014-11-30 21:39:53 +01:00
|
|
|
* @li The .bss has to be cleared before.
|
|
|
|
* @li The stack has to be set up, probably somewhere in the low memory.
|
2014-11-30 23:18:56 +01:00
|
|
|
* @li The A20 line has to be activated, because all the code is beyond 1MB.
|
2014-11-30 21:39:53 +01:00
|
|
|
* @li Paging must be disabled.
|
|
|
|
* @li The SS, DS, and CS must span the whole 4GB of RAM.
|
|
|
|
* @li 32 bit protected mode must be entered.
|
|
|
|
* @li Interrupts must be disabled.
|
2014-01-16 07:39:33 +01:00
|
|
|
*/
|
|
|
|
void x86_startup(void) NORETURN;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Setup board specific hardware and such.
|
|
|
|
*
|
|
|
|
* Called by x86_startup() after all generic subsystems are activated,
|
|
|
|
* and before kernel_init() gets called.
|
|
|
|
* Interrupts are disabled before, and are expected to be disabled after calling this function.
|
|
|
|
* Interrupts may be enabled during the course of this function call.
|
|
|
|
*
|
|
|
|
* Probably most of the board specific initialization should be done in auto_init():
|
2014-11-30 21:39:53 +01:00
|
|
|
* @li You must not spawn thread_create() in this function.
|
|
|
|
* @li The hwtimer is not set up properly at this point of executation.
|
2014-01-16 07:39:33 +01:00
|
|
|
*/
|
|
|
|
void x86_init_board(void);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Returns a range of usable physical memory.
|
|
|
|
* @param[out] start Start address of the physical memory region.
|
|
|
|
* @param[out] len Total length of the physical memory region.
|
|
|
|
* One page are 0x1000 bytes.
|
|
|
|
* @param cnt Reentrant pointer. Must be zero on first call.
|
|
|
|
* The value is entirely opaque.
|
|
|
|
* @returns true iff there was a memory region set in start and len.
|
|
|
|
* false iff there are no more memory regions.
|
|
|
|
* The function must not get called again in the latter case.
|
|
|
|
*
|
|
|
|
* Called by x86_startup() before x86_init_board() gets called.
|
|
|
|
*
|
|
|
|
* The memory management is one of the earliest subsystems to be set up.
|
|
|
|
* The uart is already configured and usable.
|
|
|
|
* The interrupt handling is already set up and you may call x86_interrupt_handler_set().
|
|
|
|
*
|
|
|
|
* The function may return unaligned memory.
|
|
|
|
* In this case the begin of the region gets rounded up the the next page,
|
|
|
|
* and the end gets rounded down.
|
|
|
|
* The function may supply low memory regions, which will be ignored.
|
|
|
|
* The regions are expected to contain memory that lies inside the elf sections.
|
|
|
|
*/
|
|
|
|
bool x86_get_memory_region(uint64_t *start, uint64_t *len, unsigned long *cnt);
|
|
|
|
|
2015-09-04 16:51:08 +02:00
|
|
|
/**
|
|
|
|
* @brief Prints the last instruction's address
|
|
|
|
*/
|
|
|
|
static inline void cpu_print_last_instruction(void)
|
|
|
|
{
|
2015-09-04 16:50:25 +02:00
|
|
|
void *p;
|
|
|
|
__asm__("1: mov 1b, %0" : "=r" (p));
|
|
|
|
printf("%p\n", p);
|
2015-09-04 16:51:08 +02:00
|
|
|
}
|
|
|
|
|
2014-10-13 10:53:20 +02:00
|
|
|
#ifdef __cplusplus
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2014-01-16 07:39:33 +01:00
|
|
|
#endif
|
|
|
|
|
|
|
|
/** @} */
|