mirror of
https://github.com/kelvinlawson/atomthreads.git
synced 2026-02-05 14:33:15 +01:00
Add work-in-progress DM36X port.
This commit is contained in:
1161
ports/arm/platforms/dm36x/Doxyfile
Normal file
1161
ports/arm/platforms/dm36x/Doxyfile
Normal file
File diff suppressed because it is too large
Load Diff
119
ports/arm/platforms/dm36x/Makefile
Normal file
119
ports/arm/platforms/dm36x/Makefile
Normal file
@@ -0,0 +1,119 @@
|
||||
############
|
||||
# Settings #
|
||||
############
|
||||
|
||||
# Build all test applications:
|
||||
# make
|
||||
|
||||
# Location of build tools and atomthreads sources
|
||||
KERNEL_DIR=../../../../kernel
|
||||
TESTS_DIR=../../../../tests
|
||||
PORT_DIR=../..
|
||||
CC=arm-none-eabi-gcc
|
||||
OBJCOPY=arm-none-eabi-objcopy
|
||||
|
||||
# Enable stack-checking.
|
||||
#STACK_CHECK=true
|
||||
|
||||
# Test programs: Log stack usage to UART (if STACK_CHECK is enabled)
|
||||
#TESTS_LOG_STACK=true
|
||||
|
||||
# Directory for built objects
|
||||
BUILD_DIR=build
|
||||
|
||||
# Platform-specific object files
|
||||
PLATFORM_OBJECTS = atomport-private.o uart.o
|
||||
PLATFORM_ASM_OBJECTS = startup.o
|
||||
|
||||
# Port-specific object files
|
||||
PORT_OBJECTS = atomport.o tests-main.o syscalls.o
|
||||
PORT_ASM_OBJECTS = atomport-asm.o
|
||||
|
||||
# Kernel object files
|
||||
KERNEL_OBJECTS = atomkernel.o atomsem.o atommutex.o atomtimer.o atomqueue.o
|
||||
|
||||
# Collection of built objects (excluding test applications)
|
||||
ALL_OBJECTS = $(PLATFORM_OBJECTS) $(PLATFORM_ASM_OBJECTS) $(PORT_OBJECTS) $(PORT_ASM_OBJECTS) $(KERNEL_OBJECTS)
|
||||
BUILT_OBJECTS = $(patsubst %,$(BUILD_DIR)/%,$(ALL_OBJECTS))
|
||||
|
||||
# Test object files (dealt with separately as only one per application build)
|
||||
TEST_OBJECTS = $(notdir $(patsubst %.c,%.o,$(wildcard $(TESTS_DIR)/*.c)))
|
||||
|
||||
# Target application filenames for each test object
|
||||
TEST_ELFS = $(patsubst %.o,%.elf,$(TEST_OBJECTS))
|
||||
|
||||
# Search build/output directory for dependencies
|
||||
vpath %.o ./$(BUILD_DIR)
|
||||
vpath %.elf ./$(BUILD_DIR)
|
||||
|
||||
# GCC flags
|
||||
CFLAGS=-g -c -mcpu=arm926ej-s -ffreestanding -Wall -Werror
|
||||
AFLAGS=$(CFLAGS) -x assembler-with-cpp
|
||||
LFLAGS=-mcpu=arm926ej-s -Tsystem.ld -Wall
|
||||
|
||||
# Enable stack-checking options (disable if not required)
|
||||
ifeq ($(STACK_CHECK),true)
|
||||
CFLAGS += -DATOM_STACK_CHECKING
|
||||
endif
|
||||
ifeq ($(TESTS_LOG_STACK),true)
|
||||
CFLAGS += -DTESTS_LOG_STACK_USAGE
|
||||
endif
|
||||
|
||||
|
||||
#################
|
||||
# Build targets #
|
||||
#################
|
||||
|
||||
# All tests
|
||||
all: $(BUILD_DIR) $(TEST_ELFS) Makefile
|
||||
|
||||
# Make build/output directory
|
||||
$(BUILD_DIR):
|
||||
mkdir $(BUILD_DIR)
|
||||
|
||||
# Test ELF files (one application build for each test)
|
||||
$(TEST_ELFS): %.elf: %.o $(ALL_OBJECTS)
|
||||
$(CC) $(LFLAGS) $(BUILD_DIR)/$(notdir $<) $(BUILT_OBJECTS) --output $(BUILD_DIR)/$@ -Wl,-Map,$(BUILD_DIR)/$(basename $@).map
|
||||
|
||||
# Kernel objects builder
|
||||
$(KERNEL_OBJECTS): %.o: $(KERNEL_DIR)/%.c
|
||||
$(CC) -c $(CFLAGS) -I. -I$(PORT_DIR) $< -o $(BUILD_DIR)/$(notdir $@)
|
||||
|
||||
# Test objects builder
|
||||
$(TEST_OBJECTS): %.o: $(TESTS_DIR)/%.c
|
||||
$(CC) -c $(CFLAGS) -I. -I$(PORT_DIR) -I$(KERNEL_DIR) $< -o $(BUILD_DIR)/$(notdir $@)
|
||||
|
||||
# Platform C objects builder
|
||||
$(PLATFORM_OBJECTS): %.o: ./%.c
|
||||
$(CC) -c $(CFLAGS) -I. -I$(PORT_DIR) -I$(KERNEL_DIR) -I$(TESTS_DIR) $< -o $(BUILD_DIR)/$(notdir $@)
|
||||
|
||||
# Platform asm objects builder
|
||||
$(PLATFORM_ASM_OBJECTS): %.o: ./%.s
|
||||
$(CC) -c $(AFLAGS) -I. -I$(PORT_DIR) -I$(KERNEL_DIR) $< -o $(BUILD_DIR)/$(notdir $@)
|
||||
|
||||
# Port C objects builder
|
||||
$(PORT_OBJECTS): %.o: $(PORT_DIR)/%.c
|
||||
$(CC) -c $(CFLAGS) -I. -I$(PORT_DIR) -I$(KERNEL_DIR) -I$(TESTS_DIR) $< -o $(BUILD_DIR)/$(notdir $@)
|
||||
|
||||
# Port asm objects builder
|
||||
$(PORT_ASM_OBJECTS): %.o: $(PORT_DIR)/%.s
|
||||
$(CC) -c $(AFLAGS) -I. -I$(PORT_DIR) -I$(KERNEL_DIR) $< -o $(BUILD_DIR)/$(notdir $@)
|
||||
|
||||
# .lst file builder
|
||||
%.lst: %.c
|
||||
$(CC) $(CFLAGS) -I. -I$(PORT_DIR) -I$(KERNEL_DIR) -I$(TESTS_DIR) -Wa,-al $< > $@
|
||||
|
||||
# Clean
|
||||
clean:
|
||||
rm -f *.o *.elf *.map *.lst
|
||||
rm -rf doxygen-kernel
|
||||
rm -rf doxygen-arm
|
||||
rm -rf doxygen-platform
|
||||
rm -rf build
|
||||
|
||||
# Generate Doxygen documentation
|
||||
doxygen:
|
||||
doxygen $(KERNEL_DIR)/Doxyfile
|
||||
doxygen ../../Doxyfile
|
||||
doxygen ./Doxyfile
|
||||
|
||||
223
ports/arm/platforms/dm36x/README
Normal file
223
ports/arm/platforms/dm36x/README
Normal file
@@ -0,0 +1,223 @@
|
||||
---------------------------------------------------------------------------
|
||||
|
||||
Library: Atomthreads QEMU ARM Integrator/CP (ARM926EJ-S) Platform.
|
||||
Author: Natie van Rooyen <natie@navaro.nl>
|
||||
Website: http://atomthreads.com
|
||||
License: BSD Revised
|
||||
|
||||
---------------------------------------------------------------------------
|
||||
|
||||
QEMU ARM Integrator/CP (ARM926EJ-S) Platform
|
||||
|
||||
The "qemu_integratorcp" platform folder contains sources for building a
|
||||
sample Atomthreads application for the ARM Integrator/CP (ARM926EJ-S)
|
||||
platform running under QEMU.
|
||||
|
||||
All of the cross-platform kernel code is contained in the top-level
|
||||
'kernel' folder, while ports to specific CPU architectures are contained in
|
||||
the 'ports' folder tree. To support multiple ARM boards/platforms using a
|
||||
single common ARM architecture port, the ARM port contains 'platform'
|
||||
sub-folders in which the board/platform-specific code is situated. This
|
||||
allows the sharing of common ARM port code between many different ARM
|
||||
boards with different interrupt controllers, UARTs etc but which all reuse
|
||||
the same common core ARM context-switching code.
|
||||
|
||||
This platform contains a few key platform-specific files:
|
||||
|
||||
* startup.s: Interrupt vector table and basic startup assembly code
|
||||
* modules.c: Low level initialisation for this platform
|
||||
* uart.c: Simple UART implementation for debug purposes
|
||||
|
||||
The common ARM architecture port that is used across all platforms contains
|
||||
the basic code for thread-switching on all ARM platforms:
|
||||
|
||||
* atomport.c: Those functions which can be written in C
|
||||
* atomport-asm.s: The main register save/restore assembler routines
|
||||
|
||||
Each Atomthreads port requires also a header file which describes various
|
||||
architecture-specific details such as appropriate types for 8-bit, 16-bit
|
||||
etc variables, the port's system tick frequency, and macros for performing
|
||||
interrupt lockouts / critical sections:
|
||||
|
||||
* atomport.h: Port-specific header required by the kernel for each port
|
||||
|
||||
A couple of additional source files are also included in the common ARM port:
|
||||
|
||||
* tests-main.c: Main application file (used for launching automated tests)
|
||||
* syscalls.c: Simple implementation of open/close/read/write for stdio
|
||||
|
||||
Atomthreads includes a suite of automated tests which prove the key OS
|
||||
functionality, and can be used with any architecture ports. This port
|
||||
provides an easy mechanism for building and quickly running the test suite
|
||||
using QEMU to prove the OS without requiring any target hardware.
|
||||
|
||||
This platform folder has been tested on a variety of GCC ARM toolchains,
|
||||
including hosted and non-hosted.
|
||||
|
||||
|
||||
---------------------------------------------------------------------------
|
||||
|
||||
GCC TOOLCHAIN
|
||||
|
||||
The port works out-of-the-box with the GCC tools (for building) and QEMU
|
||||
(for simulation). It can be built on any OS for which GCC is available, and
|
||||
was tested using the CodeSourcery toolchain (2009q3 non-Linux but others
|
||||
should be supported) and self-built toolchains such as hosted toolchains
|
||||
built by build_arm_toolchain.sh (see http://navaro.nl for details). Note
|
||||
that the Makefile for this platform assumes that your GCC binary is named
|
||||
"arm-none-eabi-gcc".
|
||||
|
||||
Currently we assume that the toolchain will provide some header files like
|
||||
stdint.h. Not all toolchains will include this, in which case you simply
|
||||
need to add definitions for int32_t and friends in atomport.h, in place of
|
||||
the include declaration for stdint.h.
|
||||
|
||||
Some toolchains provide newlib syscalls.c which implement stdio
|
||||
functionality via open, close, read, write. Hosted toolchains will
|
||||
automatically provide versions of these which work with QEMU, and no UART
|
||||
driver will be needed to view the stdio printf()s etc. If these are not
|
||||
provided by by the compiler toolchain then backup implementations are
|
||||
implemented within the common ARM port folder (see ports/arm/syscalls.c)
|
||||
and a UART driver is implemented here in uart.c.
|
||||
|
||||
Similarly some toolchains provide startup assembly code via
|
||||
_mainCRTStartup(). If this is not provided by the toolchain then a backup
|
||||
version is used within modules.c.
|
||||
|
||||
|
||||
---------------------------------------------------------------------------
|
||||
|
||||
OTHER PREREQUISITES
|
||||
|
||||
QEMU is used for simulation of the target and versions 0.14.1, 1.2.0 &
|
||||
1.4.0 were tested.
|
||||
|
||||
Running the entire automated test suite in one command via "make qemutests"
|
||||
also requires the "expect" program.
|
||||
|
||||
|
||||
---------------------------------------------------------------------------
|
||||
|
||||
BUILDING THE SOURCE
|
||||
|
||||
A Makefile is provided for building the kernel, port, platform and
|
||||
automated tests. Make sure the ARM GCC toolchain is in the path
|
||||
(e.g. "PATH=$PATH:/opt/arm-2009q3/bin && export path") as well as QEMU and
|
||||
carry out the full build using the following:
|
||||
|
||||
* make all
|
||||
|
||||
All objects are built into the 'build' folder under
|
||||
ports/arm/platforms/qemu_integrator_cp. The build process builds separate
|
||||
target applications for each automated test, and appropriate ELF files can
|
||||
be found in the build folder ready for running on the target or within
|
||||
QEMU. Each test is built and run as a separate application.
|
||||
|
||||
|
||||
All built objects etc can be cleaned using:
|
||||
|
||||
* make clean
|
||||
|
||||
|
||||
The Atomthreads sources are documented using Doxygen markup. You can build
|
||||
both the kernel and port documentation from this folder using:
|
||||
|
||||
* make doxygen
|
||||
|
||||
|
||||
---------------------------------------------------------------------------
|
||||
|
||||
Integrator/CP SPECIFICS
|
||||
|
||||
The test applications make use of the Integrator's UART to print out
|
||||
pass/fail indications and other information. For this you should connect a
|
||||
serial debug cable to the board or when running in QEMU you will see the
|
||||
UART messages on the console when running the test applications.
|
||||
|
||||
|
||||
---------------------------------------------------------------------------
|
||||
|
||||
AUTOMATED TESTS
|
||||
|
||||
Atomthreads contains a set of generic kernel tests which can be run on any
|
||||
port to prove that all core functionality is working on your target.
|
||||
|
||||
The full set of tests can be found in the top-level 'tests' folder. Each of
|
||||
these tests is built as an independent application in the 'build' folder.
|
||||
|
||||
These can be run on the target or within QEMU using the instructions below.
|
||||
|
||||
To view the test results, connect a serial debug cable to your target
|
||||
platform or view the console if using QEMU. On starting, the test
|
||||
applications print out "Go" on the UART. Once the test is complete they
|
||||
will print out "Pass" or "Fail", along with other information if the test
|
||||
failed.
|
||||
|
||||
Most of the tests complete within a few seconds, but some (particularly
|
||||
the stress tests) can take longer, so be patient.
|
||||
|
||||
The full suite of tests endeavours to exercise as much of the kernel code
|
||||
as possible, and can be used for quick confirmation of core OS
|
||||
functionality if you ever need to make a change to the kernel or port.
|
||||
|
||||
The test application main() is contained in tests-main.c. This initialises
|
||||
the OS, creates a main thread, and calls out to the test modules.
|
||||
|
||||
|
||||
---------------------------------------------------------------------------
|
||||
|
||||
RUNNING TESTS WITHIN THE QEMU SIMULATOR
|
||||
|
||||
It is possible to run the full automated test suite in a simulator without
|
||||
programming the test applications into real hardware. This is very useful
|
||||
for quick verification of the entire test suite after making any software
|
||||
changes, and is much faster than downloading each test application to a
|
||||
real target.
|
||||
|
||||
A single command runs every single test application, and automatically
|
||||
parses the (simulated) UART output to verify that each test case passes.
|
||||
|
||||
This requires two applications on your development PC: expect and QEMU.
|
||||
|
||||
To run all tests in one command, type:
|
||||
|
||||
* make qemutests
|
||||
|
||||
This will run every single test application within the simulator and quit
|
||||
immediately if any one test fails.
|
||||
|
||||
The ability to run these automated tests in one command (and without real
|
||||
hardware) allows you to easily include the OS test suite in your nightly
|
||||
build or continous integration system and quickly find out if any of your
|
||||
local changes have caused any of the operating system tests to fail.
|
||||
|
||||
|
||||
---------------------------------------------------------------------------
|
||||
|
||||
DEBUGGING WITH QEMU
|
||||
|
||||
You may also use QEMU in combination with GDB to debug your built
|
||||
applications. To do this you should start QEMU with "-S -s" options e.g.:
|
||||
|
||||
* qemu-system-arm -M integratorcp -semihosting -nographic -kernel app.elf
|
||||
|
||||
You can now start GDB and attach to the target:
|
||||
|
||||
* arm-none-eabi-gdb
|
||||
* file app.elf
|
||||
* target remote localhost:1234
|
||||
|
||||
|
||||
---------------------------------------------------------------------------
|
||||
|
||||
WRITING APPLICATIONS
|
||||
|
||||
The easiest way to start a new application which utilises the Atomthreads
|
||||
scheduler is to base your main application startup on tests-main.c. This
|
||||
initialises the OS and calls out to the test module entry functions. You
|
||||
can generally simply replace the call to the test modules by a call to your
|
||||
own application startup code.
|
||||
|
||||
|
||||
---------------------------------------------------------------------------
|
||||
|
||||
172
ports/arm/platforms/dm36x/atomport-private.c
Normal file
172
ports/arm/platforms/dm36x/atomport-private.c
Normal file
@@ -0,0 +1,172 @@
|
||||
/*
|
||||
* Copyright (c) 2013, Kelvin Lawson. 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. No personal names or organizations' names associated with the
|
||||
* Atomthreads project may be used to endorse or promote products
|
||||
* derived from this software without specific prior written permission.
|
||||
*
|
||||
* THIS SOFTWARE IS PROVIDED BY THE ATOMTHREADS PROJECT 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 PROJECT 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 <stdio.h>
|
||||
#include <stdarg.h>
|
||||
#include "atomport.h"
|
||||
#include "atomport-private.h"
|
||||
#include "atom.h"
|
||||
#include "atomport.h"
|
||||
#include "dm36x-io.h"
|
||||
#include "uart.h"
|
||||
|
||||
|
||||
/** Imports required by C startup code */
|
||||
extern unsigned long _end_text, _start_data, _end_data, _start_bss, _end_bss;
|
||||
extern int main(void);
|
||||
|
||||
|
||||
/** Timer input clock speed: 24MHz */
|
||||
#define TIMER_CLK 24000000
|
||||
|
||||
|
||||
/** Register access macros */
|
||||
#define TIMER0_REG(offset) *(uint32_t *)(DM36X_TIMER0_BASE + offset)
|
||||
#define INTC_REG(offset) *(uint32_t *)(DM36X_INTC_BASE + offset)
|
||||
|
||||
|
||||
/**
|
||||
* \b _mainCRTStartup
|
||||
*
|
||||
* C startup code for environments without a suitable built-in one.
|
||||
* May be provided by the compiler toolchain in some cases.
|
||||
*
|
||||
*/
|
||||
extern void _mainCRTStartup (void) __attribute__((weak));
|
||||
void _mainCRTStartup(void)
|
||||
{
|
||||
unsigned long *src;
|
||||
#ifdef ROM
|
||||
unsigned long *dst;
|
||||
#endif
|
||||
|
||||
#ifdef ROM
|
||||
// Running from ROM: copy data section to RAM
|
||||
src = &_end_text;
|
||||
dst = &_start_data;
|
||||
while(dst < &_end_data)
|
||||
*(dst++) = *(src++);
|
||||
#endif
|
||||
|
||||
// Clear BSS
|
||||
src = &_start_bss;
|
||||
while(src < &_end_bss)
|
||||
*(src++) = 0;
|
||||
|
||||
// Jump to main application entry point
|
||||
main();
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* \b low_level_init
|
||||
*
|
||||
* Initializes the PIC and starts the system timer tick interrupt.
|
||||
*
|
||||
*/
|
||||
int
|
||||
low_level_init (void)
|
||||
{
|
||||
/* Initialise TIMER0 registers for interrupt 100 times per second */
|
||||
|
||||
/* Reset & disable all TIMER0 timers */
|
||||
TIMER0_REG(DM36X_TIMER_INTCTL_STAT) = 0; /* Disable interrupts */
|
||||
TIMER0_REG(DM36X_TIMER_TCR) = 0; /* Disable all TIMER0 timers */
|
||||
TIMER0_REG(DM36X_TIMER_TGCR) = 0; /* Put all TIMER0 timers in reset */
|
||||
TIMER0_REG(DM36X_TIMER_TIM12) = 0; /* Clear Timer 1:2 */
|
||||
|
||||
/* Set up Timer 1:2 in 32-bit unchained mode */
|
||||
TIMER0_REG(DM36X_TIMER_TGCR) = (1 << 2); /* Select 32-bit unchained mode (TIMMODE) */
|
||||
TIMER0_REG(DM36X_TIMER_TGCR) |= (1 << 0); /* Remove Timer 1:2 from reset (TIM12RS) */
|
||||
TIMER0_REG(DM36X_TIMER_PRD12) = (TIMER_CLK / SYSTEM_TICKS_PER_SEC) - 1; /* Set period to 100 ticks per second (PRD12) */
|
||||
TIMER0_REG(DM36X_TIMER_TCR) |= (0 << 8); /* Select external clock source for Timer 1:2 (CLKSRC12) */
|
||||
|
||||
/* Enable interrupts */
|
||||
TIMER0_REG(DM36X_TIMER_INTCTL_STAT) = (1 << 1) | (1 << 0); /* Enable/ack Compare/Match interrupt for Timer 1:2 */
|
||||
|
||||
/* Enable timer */
|
||||
TIMER0_REG(DM36X_TIMER_TCR) |= (2 << 6); /* Enable Timer 1:2 continuous (ENAMODE12) */
|
||||
|
||||
return 0 ;
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* \b __interrupt_dispatcher
|
||||
*
|
||||
* Interrupt dispatcher: determines the source of the IRQ and calls
|
||||
* the appropriate ISR.
|
||||
*
|
||||
* Currently only the OS system tick ISR is implemented.
|
||||
*
|
||||
* Note that any ISRs which call Atomthreads OS routines that can
|
||||
* cause rescheduling of threads must be surrounded by calls to
|
||||
* atomIntEnter() and atomIntExit().
|
||||
*
|
||||
*/
|
||||
void
|
||||
__interrupt_dispatcher (void)
|
||||
{
|
||||
uint32_t vector;
|
||||
|
||||
/* Read IRQENTRY register to determine the source of the interrupt */
|
||||
vector = (INTC_REG(DM36X_INTC_IRQENTRY) / 4) - 1;
|
||||
|
||||
/* TIMER0:12 tick interrupt (call Atomthreads timer tick ISR) */
|
||||
if (vector == DM36X_INTC_VEC_TINT0)
|
||||
{
|
||||
/*
|
||||
* Let the Atomthreads kernel know we're about to enter an OS-aware
|
||||
* interrupt handler which could cause scheduling of threads.
|
||||
*/
|
||||
atomIntEnter();
|
||||
|
||||
/* Call the OS system tick handler */
|
||||
atomTimerTick();
|
||||
|
||||
/* Ack the interrupt */
|
||||
INTC_REG((vector >= 32) ? DM36X_INTC_IRQ1 : DM36X_INTC_IRQ0) = (1 << (vector >= 32) ? (vector - 32) : vector);
|
||||
|
||||
/* Call the interrupt exit routine */
|
||||
atomIntExit(TRUE);
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* \b null_handler
|
||||
*
|
||||
* Handler to catch interrupts at uninitialised vectors.
|
||||
*
|
||||
*/
|
||||
void null_handler (void)
|
||||
{
|
||||
uart_write_halt ("Unhandled interrupt\n");
|
||||
}
|
||||
|
||||
70
ports/arm/platforms/dm36x/dm36x-io.h
Normal file
70
ports/arm/platforms/dm36x/dm36x-io.h
Normal file
@@ -0,0 +1,70 @@
|
||||
/*
|
||||
* Copyright (c) 2013, Kelvin Lawson. 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. No personal names or organizations' names associated with the
|
||||
* Atomthreads project may be used to endorse or promote products
|
||||
* derived from this software without specific prior written permission.
|
||||
*
|
||||
* THIS SOFTWARE IS PROVIDED BY THE ATOMTHREADS PROJECT 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 PROJECT 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.
|
||||
*/
|
||||
|
||||
#ifndef __DM36X_IO_H__
|
||||
#define __DM36X_IO_H__
|
||||
|
||||
#include "atomport.h"
|
||||
|
||||
|
||||
/*
|
||||
* IO Addresses for use with DM36x
|
||||
*/
|
||||
|
||||
#define DM36X_TIMER0_BASE 0x01C21400 /* TIMER0 */
|
||||
#define DM36X_TIMER_PID12 0x00
|
||||
#define DM36X_TIMER_EMUMGT 0x04
|
||||
#define DM36X_TIMER_TIM12 0x10
|
||||
#define DM36X_TIMER_TIM34 0x14
|
||||
#define DM36X_TIMER_PRD12 0x18
|
||||
#define DM36X_TIMER_PRD34 0x1C
|
||||
#define DM36X_TIMER_TCR 0x20
|
||||
#define DM36X_TIMER_TGCR 0x24
|
||||
#define DM36X_TIMER_WDTCR 0x28
|
||||
#define DM36X_TIMER_REL12 0x34
|
||||
#define DM36X_TIMER_REL34 0x38
|
||||
#define DM36X_TIMER_CAP12 0x3C
|
||||
#define DM36X_TIMER_CAP34 0x40
|
||||
#define DM36X_TIMER_INTCTL_STAT 0x44
|
||||
|
||||
#define DM36X_INTC_BASE 0x01C48000 /* Interrupt controller */
|
||||
#define DM36X_INTC_IRQ0 0x08
|
||||
#define DM36X_INTC_IRQ1 0x0C
|
||||
#define DM36X_INTC_IRQENTRY 0x10
|
||||
#define DM36X_INTC_VEC_TINT0 32
|
||||
|
||||
#define DM36X_UART1_BASE 0x01D06000 /* UART1 */
|
||||
|
||||
|
||||
|
||||
/* Function prototypes */
|
||||
extern int low_level_init (void) ;
|
||||
|
||||
|
||||
#endif /* __DM36X_IO_H__ */
|
||||
54
ports/arm/platforms/dm36x/startup.s
Normal file
54
ports/arm/platforms/dm36x/startup.s
Normal file
@@ -0,0 +1,54 @@
|
||||
.section .vectors, "x"
|
||||
|
||||
.global __interrupt_vector_table
|
||||
.extern __irq_stack_top__
|
||||
.extern __fiq_stack_top__
|
||||
.extern __svc_stack_top__
|
||||
|
||||
|
||||
.equ USR_MODE, 0x10
|
||||
.equ FIQ_MODE, 0x11
|
||||
.equ IRQ_MODE, 0x12
|
||||
.equ SVC_MODE, 0x13
|
||||
.equ ABT_MODE, 0x17
|
||||
.equ UND_MODE, 0x1B
|
||||
.equ SYS_MODE, 0x1F
|
||||
|
||||
.equ I_BIT, 0x80 /* when I bit is set, IRQ is disabled */
|
||||
.equ F_BIT, 0x40 /* when F bit is set, FIQ is disabled */
|
||||
|
||||
|
||||
__interrupt_vector_table:
|
||||
|
||||
B Reset_Handler /* Reset */
|
||||
B Null_Handler /* Undefined */
|
||||
B Null_Handler /* SWI */
|
||||
B Null_Handler /* Prefetch Abort */
|
||||
B Null_Handler /* Data Abort */
|
||||
B Null_Handler /* reserved */
|
||||
B IRQ_Handler /* IRQ */
|
||||
B Null_Handler /* FIQ */
|
||||
|
||||
|
||||
Reset_Handler:
|
||||
|
||||
MSR CPSR_c,#(IRQ_MODE | I_BIT | F_BIT)
|
||||
LDR sp,=__irq_stack_top__ /* set the IRQ stack pointer */
|
||||
MSR CPSR_c,#(FIQ_MODE | I_BIT | F_BIT)
|
||||
LDR sp,=__fiq_stack_top__ /* set the FIQ stack pointer */
|
||||
MSR CPSR_c,#(SVC_MODE | I_BIT | F_BIT)
|
||||
LDR sp,=__svc_stack_top__ /* set the SVC stack pointer */
|
||||
|
||||
BL low_level_init
|
||||
BL _mainCRTStartup
|
||||
|
||||
|
||||
B .
|
||||
|
||||
|
||||
IRQ_Handler:
|
||||
B archIRQHandler
|
||||
|
||||
|
||||
Null_Handler:
|
||||
B null_handler
|
||||
89
ports/arm/platforms/dm36x/system.ld
Normal file
89
ports/arm/platforms/dm36x/system.ld
Normal file
@@ -0,0 +1,89 @@
|
||||
|
||||
ENTRY(__interrupt_vector_table)
|
||||
|
||||
|
||||
MEMORY
|
||||
{
|
||||
flash (rx) : ORIGIN = 0x00000000, LENGTH = 0x00020000
|
||||
sram (rwx) : ORIGIN = 0x80000000, LENGTH = 0x10000000
|
||||
}
|
||||
|
||||
|
||||
EXTERN(__interrupt_vector_table);
|
||||
|
||||
|
||||
C_STACK_SIZE = 512;
|
||||
IRQ_STACK_SIZE = 256;
|
||||
FIQ_STACK_SIZE = 256;
|
||||
SVC_STACK_SIZE = 512;
|
||||
ABT_STACK_SIZE = 256;
|
||||
UND_STACK_SIZE = 256;
|
||||
|
||||
SECTIONS
|
||||
{
|
||||
|
||||
|
||||
.text :
|
||||
{
|
||||
*(.vectors)
|
||||
/* Startup assembly */
|
||||
*(.startup)
|
||||
*(.init)
|
||||
|
||||
/* Rest of the code (C) */
|
||||
*(.text)
|
||||
*(.rodata)
|
||||
*(.rodata*)
|
||||
|
||||
_end_text = .;
|
||||
} >flash
|
||||
|
||||
.data :
|
||||
{
|
||||
_start_data = .;
|
||||
*(.data)
|
||||
_end_data = .;
|
||||
} >sram
|
||||
|
||||
.bss :
|
||||
{
|
||||
_start_bss = .;
|
||||
__bss_start__ = . ;
|
||||
*(.bss)
|
||||
} >sram
|
||||
|
||||
. = ALIGN(4);
|
||||
_end_bss = .;
|
||||
__bss_end__ = . ;
|
||||
|
||||
. = ALIGN(256);
|
||||
|
||||
.stack : {
|
||||
__stack_start__ = . ;
|
||||
. += IRQ_STACK_SIZE;
|
||||
. = ALIGN (4);
|
||||
__irq_stack_top__ = . ;
|
||||
. += FIQ_STACK_SIZE;
|
||||
. = ALIGN (4);
|
||||
__fiq_stack_top__ = . ;
|
||||
. += SVC_STACK_SIZE;
|
||||
. = ALIGN (4);
|
||||
__svc_stack_top__ = . ;
|
||||
. += ABT_STACK_SIZE;
|
||||
. = ALIGN (4);
|
||||
__abt_stack_top__ = . ;
|
||||
. += UND_STACK_SIZE;
|
||||
. = ALIGN (4);
|
||||
__und_stack_top__ = . ;
|
||||
. += C_STACK_SIZE;
|
||||
. = ALIGN (4);
|
||||
__c_stack_top__ = . ;
|
||||
__stack_end__ = .;
|
||||
} >sram
|
||||
|
||||
}
|
||||
__end__ = .;
|
||||
_end = .;
|
||||
PROVIDE(end = .);
|
||||
|
||||
heap_top = ORIGIN(sram) + LENGTH(sram) - 4;
|
||||
259
ports/arm/platforms/dm36x/uart.c
Normal file
259
ports/arm/platforms/dm36x/uart.c
Normal file
@@ -0,0 +1,259 @@
|
||||
/*
|
||||
* Copyright (c) 2013, Kelvin Lawson. 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. No personal names or organizations' names associated with the
|
||||
* Atomthreads project may be used to endorse or promote products
|
||||
* derived from this software without specific prior written permission.
|
||||
*
|
||||
* THIS SOFTWARE IS PROVIDED BY THE ATOMTHREADS PROJECT 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 PROJECT 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.
|
||||
*/
|
||||
|
||||
|
||||
/**
|
||||
* \file
|
||||
* Simple polled UART implementation for non-hosted compiler toolchains.
|
||||
*
|
||||
*
|
||||
* This is only required for non-hosted toolchains which don't implement
|
||||
* stdout automatically for use within QEMU.
|
||||
*/
|
||||
|
||||
#include "atom.h"
|
||||
#include "atommutex.h"
|
||||
#include "atomport.h"
|
||||
#include "dm36x-io.h"
|
||||
#include "uart.h"
|
||||
|
||||
|
||||
/* Constants */
|
||||
|
||||
/** FR Register bits */
|
||||
#define UART_FR_RXFE 0x10
|
||||
#define UART_FR_TXFF 0x20
|
||||
|
||||
/** UART register access macros */
|
||||
#define UART_DR(baseaddr) (*(unsigned int *)(baseaddr))
|
||||
#define UART_FR(baseaddr) (*(((unsigned int *)(baseaddr))+6))
|
||||
|
||||
|
||||
/* Local data */
|
||||
|
||||
/*
|
||||
* Semaphore for single-threaded access to UART device
|
||||
*/
|
||||
static ATOM_MUTEX uart_mutex;
|
||||
|
||||
/*
|
||||
* Initialised flag
|
||||
*/
|
||||
static int initialised = FALSE;
|
||||
|
||||
|
||||
/* Forward declarations */
|
||||
static int uart_init (void);
|
||||
|
||||
|
||||
/**
|
||||
* \b uart_init
|
||||
*
|
||||
* Initialisation of UART driver. Creates a mutex that enforces
|
||||
* single-threaded access to the UART. We poll register bits
|
||||
* to check when space is available, which would not otherwise
|
||||
* be thread-safe.
|
||||
*
|
||||
* @retval ATOM_OK Success
|
||||
* @retval ATOM_ERROR Failed to create mutex
|
||||
*/
|
||||
static int uart_init (void)
|
||||
{
|
||||
int status;
|
||||
|
||||
/* Check we are not already initialised */
|
||||
if (initialised == FALSE)
|
||||
{
|
||||
/* Create a mutex for single-threaded UART access */
|
||||
if (atomMutexCreate (&uart_mutex) != ATOM_OK)
|
||||
{
|
||||
/* Mutex creation failed */
|
||||
status = ATOM_ERROR;
|
||||
}
|
||||
else
|
||||
{
|
||||
/* Success */
|
||||
initialised = TRUE;
|
||||
status = ATOM_OK;
|
||||
}
|
||||
}
|
||||
|
||||
/* Finished */
|
||||
return (status);
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* \b uart_read
|
||||
*
|
||||
* Simple polled UART read.
|
||||
*
|
||||
* @param[in] ptr Pointer to receive buffer
|
||||
* @param[in] len Max bytes to read
|
||||
*
|
||||
* @retval Number of bytes read
|
||||
*
|
||||
*/
|
||||
int uart_read (char *ptr, int len)
|
||||
{
|
||||
int todo = 0;
|
||||
|
||||
/* Check we are initialised */
|
||||
if (initialised == FALSE)
|
||||
{
|
||||
uart_init();
|
||||
}
|
||||
|
||||
/* Check parameters */
|
||||
if ((ptr == NULL) || (len == 0))
|
||||
{
|
||||
return 0;
|
||||
}
|
||||
|
||||
/* Block thread on private access to the UART */
|
||||
if (atomOSStarted && atomMutexGet(&uart_mutex, 0) == ATOM_OK)
|
||||
{
|
||||
/* Wait for not-empty */
|
||||
while(UART_FR(DM36X_UART1_BASE) & UART_FR_RXFE)
|
||||
;
|
||||
|
||||
/* Read first byte */
|
||||
*ptr++ = UART_DR(DM36X_UART1_BASE);
|
||||
|
||||
/* Loop over remaining bytes until empty */
|
||||
for (todo = 1; todo < len; todo++)
|
||||
{
|
||||
/* Quit if receive FIFO empty */
|
||||
if(UART_FR(DM36X_UART1_BASE) & UART_FR_RXFE)
|
||||
{
|
||||
break;
|
||||
}
|
||||
|
||||
/* Read next byte */
|
||||
*ptr++ = UART_DR(DM36X_UART1_BASE);
|
||||
}
|
||||
|
||||
/* Return mutex access */
|
||||
if (atomOSStarted)
|
||||
{
|
||||
atomMutexPut(&uart_mutex);
|
||||
}
|
||||
}
|
||||
|
||||
/* Return number of bytes read */
|
||||
return todo;
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* \b uart_write
|
||||
*
|
||||
* Simple polled UART write.
|
||||
*
|
||||
* @param[in] ptr Pointer to write buffer
|
||||
* @param[in] len Number of bytes to write
|
||||
*
|
||||
* @retval Number of bytes written
|
||||
*/
|
||||
int uart_write (const char *ptr, int len)
|
||||
{
|
||||
int todo;
|
||||
|
||||
/* Check we are initialised */
|
||||
if (initialised == FALSE)
|
||||
{
|
||||
uart_init();
|
||||
}
|
||||
|
||||
/* Check parameters */
|
||||
if ((ptr == NULL) || (len == 0))
|
||||
{
|
||||
return 0;
|
||||
}
|
||||
|
||||
/* Block thread on private access to the UART */
|
||||
if (atomOSStarted && atomMutexGet(&uart_mutex, 0) == ATOM_OK)
|
||||
{
|
||||
/* Loop through all bytes to write */
|
||||
for (todo = 0; todo < len; todo++)
|
||||
{
|
||||
/* Wait for empty */
|
||||
while(UART_FR(DM36X_UART1_BASE) & UART_FR_TXFF)
|
||||
;
|
||||
|
||||
/* Write byte to UART */
|
||||
UART_DR(DM36X_UART1_BASE) = *ptr++;
|
||||
}
|
||||
|
||||
/* Return mutex access */
|
||||
if (atomOSStarted)
|
||||
{
|
||||
atomMutexPut(&uart_mutex);
|
||||
}
|
||||
}
|
||||
|
||||
/* Return bytes-written count */
|
||||
return len;
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* \b uart_write_halt
|
||||
*
|
||||
* Simple polled UART write for handling critical failures
|
||||
* by printing out a message on the UART and looping forever.
|
||||
* Can be called from interrupt (unlike the standard
|
||||
* uart_write()) but is not thread-safe because it cannot
|
||||
* take the thread-safety mutex, and hence is only useful for
|
||||
* a last-resort catastrophic debug message.
|
||||
*
|
||||
* @param[in] ptr Pointer to write string
|
||||
*/
|
||||
void uart_write_halt (const char *ptr)
|
||||
{
|
||||
/* Check parameters */
|
||||
if (ptr != NULL)
|
||||
{
|
||||
/* Loop through all bytes until NULL terminator encountered */
|
||||
while (*ptr != '\0')
|
||||
{
|
||||
/* Wait for empty */
|
||||
while(UART_FR(DM36X_UART1_BASE) & UART_FR_TXFF)
|
||||
;
|
||||
|
||||
/* Write byte to UART */
|
||||
UART_DR(DM36X_UART1_BASE) = *ptr++;
|
||||
}
|
||||
}
|
||||
|
||||
/* Loop forever */
|
||||
while (1)
|
||||
;
|
||||
|
||||
}
|
||||
38
ports/arm/platforms/dm36x/uart.h
Normal file
38
ports/arm/platforms/dm36x/uart.h
Normal file
@@ -0,0 +1,38 @@
|
||||
/*
|
||||
* Copyright (c) 2013, Kelvin Lawson. 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. No personal names or organizations' names associated with the
|
||||
* Atomthreads project may be used to endorse or promote products
|
||||
* derived from this software without specific prior written permission.
|
||||
*
|
||||
* THIS SOFTWARE IS PROVIDED BY THE ATOMTHREADS PROJECT 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 PROJECT 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.
|
||||
*/
|
||||
|
||||
#ifndef __ATOM_UART_H
|
||||
#define __ATOM_UART_H
|
||||
|
||||
/* UART driver APIs */
|
||||
extern int uart_read (char *ptr, int len);
|
||||
extern int uart_write (const char *ptr, int len);
|
||||
extern void uart_write_halt (const char *ptr);
|
||||
|
||||
#endif /* __ATOM_UART_H */
|
||||
Reference in New Issue
Block a user