mirror of
https://github.com/kelvinlawson/atomthreads.git
synced 2026-01-24 16:53:14 +01:00
Compare commits
179 Commits
release1.2
...
navaro
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
0e2d899e41 | ||
|
|
20d8da0eb6 | ||
|
|
65edc4ab09 | ||
|
|
8ecb72c755 | ||
|
|
eeefac9a40 | ||
|
|
0222cd9ff7 | ||
|
|
0aa74332d4 | ||
|
|
3db5731941 | ||
|
|
dda5aedb57 | ||
|
|
4e7c9c4c52 | ||
|
|
5ca7a2c7a5 | ||
|
|
63debdf613 | ||
|
|
02d14edd9e | ||
|
|
033ff4fd8d | ||
|
|
144434ac22 | ||
|
|
e192dc4e35 | ||
|
|
3d61d86a0a | ||
|
|
21411e439f | ||
|
|
2925d865a4 | ||
|
|
36e856020b | ||
|
|
07931272ae | ||
|
|
92cac339c9 | ||
|
|
3ec86aca58 | ||
|
|
e5882cb0e6 | ||
|
|
84e2e7aecf | ||
|
|
ba5255ec0b | ||
|
|
0f175c5da4 | ||
|
|
3e41694c37 | ||
|
|
30183b1b9f | ||
|
|
8af57c8e8f | ||
|
|
ada2ed3fac | ||
|
|
1e4f7892a4 | ||
|
|
0027134452 | ||
|
|
1b63c32848 | ||
|
|
2bd4163252 | ||
|
|
b6339e65ca | ||
|
|
554c772f95 | ||
|
|
dac253ceac | ||
|
|
05f391631c | ||
|
|
8e6270f30e | ||
|
|
06684c94ce | ||
|
|
5e56576b2d | ||
|
|
d83e5dbe00 | ||
|
|
b19004817c | ||
|
|
356685005a | ||
|
|
9d18dafd72 | ||
|
|
8cdd707b41 | ||
|
|
d9f1f8389f | ||
|
|
8d62300d41 | ||
|
|
648d35dbe6 | ||
|
|
6f34e08e2f | ||
|
|
c016f54ad0 | ||
|
|
9ebb4d14b4 | ||
|
|
5995495350 | ||
|
|
4c669225c6 | ||
|
|
b8afea38ba | ||
|
|
047c741e21 | ||
|
|
f6945acf5b | ||
|
|
650f5e2ac3 | ||
|
|
daa748332b | ||
|
|
c84c7d3222 | ||
|
|
6d8aee9e65 | ||
|
|
af6671019f | ||
|
|
f5de0d76ae | ||
|
|
cd3edd6726 | ||
|
|
c009a05421 | ||
|
|
0d07f0274d | ||
|
|
28e7382732 | ||
|
|
dfb831fa63 | ||
|
|
5181143343 | ||
|
|
34e989424f | ||
|
|
43df30809c | ||
|
|
6ce391f581 | ||
|
|
7ce220e42f | ||
|
|
b47b9697a5 | ||
|
|
97a15799b4 | ||
|
|
a2646ec695 | ||
|
|
12dbe5ebe9 | ||
|
|
1eb5c7bd41 | ||
|
|
113f0a8690 | ||
|
|
37d6d553bf | ||
|
|
5cc1e9831d | ||
|
|
4a84c4cccf | ||
|
|
22a5af0eae | ||
|
|
b812419f4a | ||
|
|
b298144baa | ||
|
|
49f27eff8d | ||
|
|
a44a38190b | ||
|
|
511187c7bd | ||
|
|
8de14626d0 | ||
|
|
743ce54031 | ||
|
|
6af5014e9f | ||
|
|
48fe6bc12a | ||
|
|
19536e1fc7 | ||
|
|
f9a16861b9 | ||
|
|
fc53574c12 | ||
|
|
1ca423d8f6 | ||
|
|
0ec7a4629e | ||
|
|
d64a524522 | ||
|
|
a96a1afbc7 | ||
|
|
2e075ad413 | ||
|
|
112928594c | ||
|
|
f36452df66 | ||
|
|
1239eb1827 | ||
|
|
a88b9b542e | ||
|
|
3468a0d479 | ||
|
|
d97eac380e | ||
|
|
32fc47ef89 | ||
|
|
a4a35ddf86 | ||
|
|
0ee1eb55c5 | ||
|
|
2df02ed0ad | ||
|
|
b213428c24 | ||
|
|
c031dcf641 | ||
|
|
ac0643a959 | ||
|
|
7bc40f9d62 | ||
|
|
fae8dd7516 | ||
|
|
711ce0e469 | ||
|
|
b9931b4c38 | ||
|
|
546990788d | ||
|
|
8dfd1f4c0f | ||
|
|
9fe564e674 | ||
|
|
c7bc5cf396 | ||
|
|
4fbbe465ee | ||
|
|
fe6232c981 | ||
|
|
9a4dac8c76 | ||
|
|
d06cbdc16a | ||
|
|
c0e238ffb2 | ||
|
|
9237406b12 | ||
|
|
d5a8c186b0 | ||
|
|
f686c6527a | ||
|
|
1e80052c98 | ||
|
|
dca450ffb8 | ||
|
|
4ba6c9d9a7 | ||
|
|
05bd1987ea | ||
|
|
271eba687c | ||
|
|
4e2b83c36f | ||
|
|
83841d2673 | ||
|
|
4b3c5e4ae3 | ||
|
|
1780000e10 | ||
|
|
a630a8945d | ||
|
|
c471814b3c | ||
|
|
02660d0c80 | ||
|
|
af84aa9848 | ||
|
|
00d4e3f2af | ||
|
|
b8e805e91a | ||
|
|
fa125d0883 | ||
|
|
f7dde300d4 | ||
|
|
8bb70d8a90 | ||
|
|
a87d40688b | ||
|
|
789975075e | ||
|
|
be6e60bf2f | ||
|
|
89f8775011 | ||
|
|
8b1739e8f4 | ||
|
|
c091aabec0 | ||
|
|
8688d0eee0 | ||
|
|
ac40279132 | ||
|
|
daf07b5946 | ||
|
|
10b93017a3 | ||
|
|
2e99ef35d5 | ||
|
|
b0d5a83689 | ||
|
|
202cbcf2df | ||
|
|
a3dec21707 | ||
|
|
8a880d2716 | ||
|
|
e11d17147d | ||
|
|
b9e7e9cc63 | ||
|
|
916dc11a0c | ||
|
|
b4bc5a9c53 | ||
|
|
e218dbb7be | ||
|
|
545a6932a4 | ||
|
|
912c83f74e | ||
|
|
e34c5a0ffa | ||
|
|
9e4da88b1d | ||
|
|
d25a49edeb | ||
|
|
c9ae8200db | ||
|
|
61e6ca0b10 | ||
|
|
b79c5c7f44 | ||
|
|
2ec7fd2d8c | ||
|
|
eafe07cea9 | ||
|
|
3ac858b8e0 |
2
README
2
README
@@ -1,7 +1,7 @@
|
||||
---------------------------------------------------------------------------
|
||||
|
||||
Library: Atomthreads
|
||||
Author: Kelvin Lawson <kelvinl@users.sf.net>
|
||||
Author: Kelvin Lawson <info@atomthreads.com>
|
||||
Website: http://atomthreads.com
|
||||
License: BSD Revised
|
||||
|
||||
|
||||
@@ -1,11 +1,11 @@
|
||||
---------------------------------------------------------------------------
|
||||
|
||||
Library: Atomthreads
|
||||
Author: Kelvin Lawson <kelvinl@users.sf.net>
|
||||
Website: http://atomthreads.com
|
||||
License: BSD Revised
|
||||
|
||||
---------------------------------------------------------------------------
|
||||
---------------------------------------------------------------------------
|
||||
|
||||
Library: Atomthreads
|
||||
Author: Kelvin Lawson <info@atomthreads.com>
|
||||
Website: http://atomthreads.com
|
||||
License: BSD Revised
|
||||
|
||||
---------------------------------------------------------------------------
|
||||
|
||||
KERNEL SOURCES
|
||||
|
||||
@@ -21,12 +21,12 @@ Each module source file contains detailed documentation including an
|
||||
introduction to usage of the module and full descriptions of each API.
|
||||
Refer to the sources for further documentation.
|
||||
|
||||
---------------------------------------------------------------------------
|
||||
---------------------------------------------------------------------------
|
||||
|
||||
BUILDING THE KERNEL
|
||||
|
||||
The kernel is built from the architecture port folder. Build instructions
|
||||
are included in the README file for each port.
|
||||
|
||||
---------------------------------------------------------------------------
|
||||
---------------------------------------------------------------------------
|
||||
|
||||
|
||||
@@ -30,6 +30,10 @@
|
||||
#ifndef __ATOM_H
|
||||
#define __ATOM_H
|
||||
|
||||
#ifdef __cplusplus
|
||||
extern "C" {
|
||||
#endif
|
||||
|
||||
#include "atomtimer.h"
|
||||
#include "atomport.h"
|
||||
|
||||
@@ -40,8 +44,10 @@ struct atom_tcb;
|
||||
|
||||
typedef struct atom_tcb
|
||||
{
|
||||
/* Thread's current stack pointer. When a thread is scheduled
|
||||
* out the architecture port can save*/
|
||||
/*
|
||||
* Thread's current stack pointer. When a thread is scheduled
|
||||
* out the architecture port can save its stack pointer here.
|
||||
*/
|
||||
POINTER sp_save_ptr;
|
||||
|
||||
/* Thread priority (0-255) */
|
||||
@@ -62,10 +68,14 @@ typedef struct atom_tcb
|
||||
|
||||
/* Details used if thread stack-checking is required */
|
||||
#ifdef ATOM_STACK_CHECKING
|
||||
POINTER stack_top; /* Pointer to top of stack allocation */
|
||||
POINTER stack_bottom; /* Pointer to bottom of stack allocation */
|
||||
uint32_t stack_size; /* Size of stack allocation in bytes */
|
||||
#endif
|
||||
|
||||
#ifdef ATOM_TLS
|
||||
ATOM_TLS /* Thread Local Storage */
|
||||
#endif
|
||||
|
||||
} ATOM_TCB;
|
||||
|
||||
|
||||
@@ -98,7 +108,7 @@ extern uint8_t atomOSStarted;
|
||||
|
||||
|
||||
/* Function prototypes */
|
||||
extern uint8_t atomOSInit (void *idle_thread_stack_top, uint32_t stack_size);
|
||||
extern uint8_t atomOSInit (void *idle_thread_stack_bottom, uint32_t idle_thread_stack_size, uint8_t idle_thread_stack_check);
|
||||
extern void atomOSStart (void);
|
||||
|
||||
extern void atomSched (uint8_t timer_tick);
|
||||
@@ -113,7 +123,7 @@ extern ATOM_TCB *tcbDequeuePriority (ATOM_TCB **tcb_queue_ptr, uint8_t priority)
|
||||
|
||||
extern ATOM_TCB *atomCurrentContext (void);
|
||||
|
||||
extern uint8_t atomThreadCreate (ATOM_TCB *tcb_ptr, uint8_t priority, void (*entry_point)(uint32_t), uint32_t entry_param, void *stack_top, uint32_t stack_size);
|
||||
extern uint8_t atomThreadCreate (ATOM_TCB *tcb_ptr, uint8_t priority, void (*entry_point)(uint32_t), uint32_t entry_param, void *stack_bottom, uint32_t stack_size, uint8_t stack_check);
|
||||
extern uint8_t atomThreadStackCheck (ATOM_TCB *tcb_ptr, uint32_t *used_bytes, uint32_t *free_bytes);
|
||||
|
||||
extern void archContextSwitch (ATOM_TCB *old_tcb_ptr, ATOM_TCB *new_tcb_ptr);
|
||||
@@ -122,5 +132,8 @@ extern void archFirstThreadRestore(ATOM_TCB *new_tcb_ptr);
|
||||
|
||||
extern void atomTimerTick (void);
|
||||
|
||||
#ifdef __cplusplus
|
||||
}
|
||||
#endif
|
||||
|
||||
#endif /* __ATOM_H */
|
||||
|
||||
@@ -28,7 +28,7 @@
|
||||
*/
|
||||
|
||||
|
||||
/**
|
||||
/**
|
||||
* \file
|
||||
* Kernel library.
|
||||
*
|
||||
@@ -114,7 +114,7 @@
|
||||
* run since the thread was created. The context-save area is formatted in
|
||||
* exactly the same manner.
|
||||
*
|
||||
*
|
||||
*
|
||||
* \b Functions contained in this module:\n
|
||||
*
|
||||
* \b Application-callable initialisation functions: \n
|
||||
@@ -144,7 +144,6 @@
|
||||
*/
|
||||
|
||||
|
||||
#include <stddef.h>
|
||||
#include "atom.h"
|
||||
|
||||
|
||||
@@ -370,25 +369,31 @@ static void atomThreadSwitch(ATOM_TCB *old_tcb, ATOM_TCB *new_tcb)
|
||||
* new thread may be scheduled in before the function returns.
|
||||
*
|
||||
* Optionally prefills the thread stack with a known value to enable stack
|
||||
* usage checking (if the ATOM_STACK_CHECKING macro is defined).
|
||||
* usage checking (if the ATOM_STACK_CHECKING macro is defined and
|
||||
* stack_check parameter is set to TRUE).
|
||||
*
|
||||
* @param[in] tcb_ptr Pointer to the thread's TCB storage
|
||||
* @param[in] priority Priority of the thread (0 to 255)
|
||||
* @param[in] entry_point Thread entry point
|
||||
* @param[in] entry_param Parameter passed to thread entry point
|
||||
* @param[in] stack_top Top of the stack area
|
||||
* @param[in] stack_bottom Bottom of the stack area
|
||||
* @param[in] stack_size Size of the stack area in bytes
|
||||
* @param[in] stack_check TRUE to enable stack checking for this thread
|
||||
*
|
||||
* @retval ATOM_OK Success
|
||||
* @retval ATOM_ERR_PARAM Bad parameters
|
||||
* @retval ATOM_ERR_QUEUE Error putting the thread on the ready queue
|
||||
*/
|
||||
uint8_t atomThreadCreate (ATOM_TCB *tcb_ptr, uint8_t priority, void (*entry_point)(uint32_t), uint32_t entry_param, void *stack_top, uint32_t stack_size)
|
||||
uint8_t atomThreadCreate (ATOM_TCB *tcb_ptr, uint8_t priority, void (*entry_point)(uint32_t), uint32_t entry_param, void *stack_bottom, uint32_t stack_size, uint8_t stack_check)
|
||||
{
|
||||
CRITICAL_STORE;
|
||||
uint8_t status;
|
||||
uint8_t *stack_top;
|
||||
#ifdef ATOM_STACK_CHECKING
|
||||
int32_t count;
|
||||
#endif
|
||||
|
||||
if ((tcb_ptr == NULL) || (entry_point == NULL) || (stack_top == NULL)
|
||||
if ((tcb_ptr == NULL) || (entry_point == NULL) || (stack_bottom == NULL)
|
||||
|| (stack_size == 0))
|
||||
{
|
||||
/* Bad parameters */
|
||||
@@ -412,6 +417,13 @@ uint8_t atomThreadCreate (ATOM_TCB *tcb_ptr, uint8_t priority, void (*entry_poin
|
||||
tcb_ptr->entry_point = entry_point;
|
||||
tcb_ptr->entry_param = entry_param;
|
||||
|
||||
/**
|
||||
* Calculate a pointer to the topmost stack entry, suitably aligned
|
||||
* for the architecture. This may discard the top few bytes if the
|
||||
* stack size is not a multiple of the stack entry/alignment size.
|
||||
*/
|
||||
stack_top = (uint8_t *)stack_bottom + (stack_size & ~(STACK_ALIGN_SIZE - 1)) - STACK_ALIGN_SIZE;
|
||||
|
||||
/**
|
||||
* Additional processing only required if stack-checking is
|
||||
* enabled. Incurs a slight overhead on each thread creation
|
||||
@@ -419,25 +431,29 @@ uint8_t atomThreadCreate (ATOM_TCB *tcb_ptr, uint8_t priority, void (*entry_poin
|
||||
* compiled out if not desired.
|
||||
*/
|
||||
#ifdef ATOM_STACK_CHECKING
|
||||
|
||||
/* Store the stack details for use by the stack-check function */
|
||||
tcb_ptr->stack_top = stack_top;
|
||||
tcb_ptr->stack_size = stack_size;
|
||||
|
||||
/**
|
||||
* Prefill the stack with a known value. This is used later in
|
||||
* calls to atomThreadStackCheck() to get an indication of how
|
||||
* much stack has been used during runtime.
|
||||
*/
|
||||
while (stack_size > 0)
|
||||
/* Set up stack-checking if enabled for this thread */
|
||||
if (stack_check)
|
||||
{
|
||||
/* Initialise all stack bytes from bottom up to 0x5A */
|
||||
*((uint8_t *)stack_top - (stack_size - 1)) = STACK_CHECK_BYTE;
|
||||
stack_size--;
|
||||
/* Store the stack details for use by the stack-check function */
|
||||
tcb_ptr->stack_bottom = stack_bottom;
|
||||
tcb_ptr->stack_size = stack_size;
|
||||
|
||||
/**
|
||||
* Prefill the stack with a known value. This is used later in
|
||||
* calls to atomThreadStackCheck() to get an indication of how
|
||||
* much stack has been used during runtime.
|
||||
*/
|
||||
count = (int32_t)stack_size;
|
||||
while (count > 0)
|
||||
{
|
||||
/* Initialise all stack bytes from top down to 0x5A */
|
||||
*((uint8_t *)stack_bottom + (count - 1)) = STACK_CHECK_BYTE;
|
||||
count--;
|
||||
}
|
||||
}
|
||||
#else
|
||||
/* Avoid compiler warnings due to unused stack_size variable */
|
||||
stack_size = stack_size;
|
||||
/* Avoid compiler warning due to unused parameter */
|
||||
stack_check = stack_check;
|
||||
#endif
|
||||
|
||||
/**
|
||||
@@ -501,7 +517,7 @@ uint8_t atomThreadCreate (ATOM_TCB *tcb_ptr, uint8_t priority, void (*entry_poin
|
||||
* have strayed outside of the allowable stack area while leaving some of
|
||||
* the known-value bytes unmodified. This simple method cannot trap stack
|
||||
* usage outside of the thread's allocated stack, for which you could use
|
||||
* additional guard areas (still limited in scope) or compiler/CPU/MMU
|
||||
* additional guard areas (still limited in scope) or compiler/CPU/MMU
|
||||
* features.
|
||||
*
|
||||
* The function takes a thread's TCB and returns both the number of stack
|
||||
@@ -529,10 +545,10 @@ uint8_t atomThreadStackCheck (ATOM_TCB *tcb_ptr, uint32_t *used_bytes, uint32_t
|
||||
else
|
||||
{
|
||||
/**
|
||||
* Starting at the far end, count the unmodified areas until a
|
||||
* Starting at the bottom end, count the unmodified areas until a
|
||||
* modified byte is found.
|
||||
*/
|
||||
stack_ptr = (uint8_t *)tcb_ptr->stack_top - (tcb_ptr->stack_size - 1);
|
||||
stack_ptr = (uint8_t *)tcb_ptr->stack_bottom;
|
||||
for (i = 0; i < tcb_ptr->stack_size; i++)
|
||||
{
|
||||
/* Loop until a modified byte is found */
|
||||
@@ -648,13 +664,14 @@ ATOM_TCB *atomCurrentContext (void)
|
||||
* operating system facilities are being initialised. They are normally
|
||||
* enabled by the archFirstThreadRestore() routine in the architecture port.
|
||||
*
|
||||
* @param[in] idle_thread_stack_top Ptr to top of stack area for idle thread
|
||||
* @param[in] idle_thread_stack_bottom Ptr to bottom of stack for idle thread
|
||||
* @param[in] idle_thread_stack_size Size of idle thread stack in bytes
|
||||
* @param[in] idle_thread_stack_check TRUE if stack checking required on idle thread
|
||||
*
|
||||
* @retval ATOM_OK Success
|
||||
* @retval ATOM_ERROR Initialisation error
|
||||
*/
|
||||
uint8_t atomOSInit (void *idle_thread_stack_top, uint32_t idle_thread_stack_size)
|
||||
uint8_t atomOSInit (void *idle_thread_stack_bottom, uint32_t idle_thread_stack_size, uint8_t idle_thread_stack_check)
|
||||
{
|
||||
uint8_t status;
|
||||
|
||||
@@ -668,8 +685,9 @@ uint8_t atomOSInit (void *idle_thread_stack_top, uint32_t idle_thread_stack_size
|
||||
IDLE_THREAD_PRIORITY,
|
||||
atomIdleThread,
|
||||
0,
|
||||
idle_thread_stack_top,
|
||||
idle_thread_stack_size);
|
||||
idle_thread_stack_bottom,
|
||||
idle_thread_stack_size,
|
||||
idle_thread_stack_check);
|
||||
|
||||
/* Return status */
|
||||
return (status);
|
||||
|
||||
@@ -101,7 +101,6 @@
|
||||
*/
|
||||
|
||||
|
||||
#include <stdio.h>
|
||||
#include "atom.h"
|
||||
#include "atommutex.h"
|
||||
#include "atomtimer.h"
|
||||
|
||||
@@ -29,6 +29,10 @@
|
||||
#ifndef __ATOM_MUTEX_H
|
||||
#define __ATOM_MUTEX_H
|
||||
|
||||
#ifdef __cplusplus
|
||||
extern "C" {
|
||||
#endif
|
||||
|
||||
typedef struct atom_mutex
|
||||
{
|
||||
ATOM_TCB * suspQ; /* Queue of threads suspended on this mutex */
|
||||
@@ -41,4 +45,8 @@ extern uint8_t atomMutexDelete (ATOM_MUTEX *mutex);
|
||||
extern uint8_t atomMutexGet (ATOM_MUTEX *mutex, int32_t timeout);
|
||||
extern uint8_t atomMutexPut (ATOM_MUTEX *mutex);
|
||||
|
||||
#ifdef __cplusplus
|
||||
}
|
||||
#endif
|
||||
|
||||
#endif /* __ATOM_MUTEX_H */
|
||||
|
||||
@@ -34,6 +34,15 @@
|
||||
/* Required number of system ticks per second (normally 100 for 10ms tick) */
|
||||
#define SYSTEM_TICKS_PER_SEC 100
|
||||
|
||||
/**
|
||||
* Definition of NULL.
|
||||
* If stddef.h is available on the platform it is simplest to include it
|
||||
* from this header, otherwise define below.
|
||||
*/
|
||||
#define NULL ((void *)(0))
|
||||
|
||||
/* Size of each stack entry / stack alignment size (e.g. 8 bits) */
|
||||
#define STACK_ALIGN_SIZE sizeof(unsigned char)
|
||||
|
||||
/**
|
||||
* Architecture-specific types.
|
||||
@@ -51,7 +60,12 @@
|
||||
#define POINTER void *
|
||||
|
||||
|
||||
/* Critical region protection */
|
||||
/**
|
||||
* Critical region protection: this should disable interrupts
|
||||
* to protect OS data structures during modification. It must
|
||||
* allow nested calls, which means that interrupts should only
|
||||
* be re-enabled when the outer CRITICAL_END() is reached.
|
||||
*/
|
||||
#define CRITICAL_STORE uint8_t sreg
|
||||
#define CRITICAL_START() sreg = SREG; cli();
|
||||
#define CRITICAL_END() SREG = sreg
|
||||
|
||||
@@ -91,7 +91,6 @@
|
||||
*/
|
||||
|
||||
|
||||
#include <stdio.h>
|
||||
#include <string.h>
|
||||
|
||||
#include "atom.h"
|
||||
|
||||
@@ -29,6 +29,10 @@
|
||||
#ifndef __ATOM_QUEUE_H
|
||||
#define __ATOM_QUEUE_H
|
||||
|
||||
#ifdef __cplusplus
|
||||
extern "C" {
|
||||
#endif
|
||||
|
||||
typedef struct atom_queue
|
||||
{
|
||||
ATOM_TCB * putSuspQ; /* Queue of threads waiting to send */
|
||||
@@ -46,4 +50,8 @@ extern uint8_t atomQueueDelete (ATOM_QUEUE *qptr);
|
||||
extern uint8_t atomQueueGet (ATOM_QUEUE *qptr, int32_t timeout, uint8_t *msgptr);
|
||||
extern uint8_t atomQueuePut (ATOM_QUEUE *qptr, int32_t timeout, uint8_t *msgptr);
|
||||
|
||||
#ifdef __cplusplus
|
||||
}
|
||||
#endif
|
||||
|
||||
#endif /* __ATOM_QUEUE_H */
|
||||
|
||||
@@ -88,7 +88,6 @@
|
||||
*/
|
||||
|
||||
|
||||
#include <stdio.h>
|
||||
#include "atom.h"
|
||||
#include "atomsem.h"
|
||||
#include "atomtimer.h"
|
||||
|
||||
@@ -30,6 +30,10 @@
|
||||
#ifndef __ATOM_SEM_H
|
||||
#define __ATOM_SEM_H
|
||||
|
||||
#ifdef __cplusplus
|
||||
extern "C" {
|
||||
#endif
|
||||
|
||||
typedef struct atom_sem
|
||||
{
|
||||
ATOM_TCB * suspQ; /* Queue of threads suspended on this semaphore */
|
||||
@@ -42,4 +46,8 @@ extern uint8_t atomSemGet (ATOM_SEM *sem, int32_t timeout);
|
||||
extern uint8_t atomSemPut (ATOM_SEM *sem);
|
||||
extern uint8_t atomSemResetCount (ATOM_SEM *sem, uint8_t count);
|
||||
|
||||
#ifdef __cplusplus
|
||||
}
|
||||
#endif
|
||||
|
||||
#endif /* __ATOM_SEM_H */
|
||||
|
||||
@@ -67,7 +67,6 @@
|
||||
*/
|
||||
|
||||
|
||||
#include <stdio.h>
|
||||
#include "atom.h"
|
||||
|
||||
|
||||
|
||||
@@ -30,6 +30,10 @@
|
||||
#ifndef __ATOM_TIMER_H
|
||||
#define __ATOM_TIMER_H
|
||||
|
||||
#ifdef __cplusplus
|
||||
extern "C" {
|
||||
#endif
|
||||
|
||||
#include "atomport.h"
|
||||
|
||||
|
||||
@@ -58,4 +62,8 @@ extern uint8_t atomTimerDelay (uint32_t ticks);
|
||||
extern uint32_t atomTimeGet (void);
|
||||
extern void atomTimeSet (uint32_t new_time);
|
||||
|
||||
#ifdef __cplusplus
|
||||
}
|
||||
#endif
|
||||
|
||||
#endif /* __ATOM_TIMER_H */
|
||||
|
||||
Binary file not shown.
1035
platforms/lpc17xx/CMSISv2p00_LPC17xx/inc/LPC17xx.h
Normal file
1035
platforms/lpc17xx/CMSISv2p00_LPC17xx/inc/LPC17xx.h
Normal file
File diff suppressed because it is too large
Load Diff
1236
platforms/lpc17xx/CMSISv2p00_LPC17xx/inc/core_cm3.h
Normal file
1236
platforms/lpc17xx/CMSISv2p00_LPC17xx/inc/core_cm3.h
Normal file
File diff suppressed because it is too large
Load Diff
844
platforms/lpc17xx/CMSISv2p00_LPC17xx/inc/core_cmFunc.h
Normal file
844
platforms/lpc17xx/CMSISv2p00_LPC17xx/inc/core_cmFunc.h
Normal file
@@ -0,0 +1,844 @@
|
||||
/**************************************************************************//**
|
||||
* @file core_cmFunc.h
|
||||
* @brief CMSIS Cortex-M Core Function Access Header File
|
||||
* @version V2.01
|
||||
* @date 06. December 2010
|
||||
*
|
||||
* @note
|
||||
* Copyright (C) 2009-2010 ARM Limited. All rights reserved.
|
||||
*
|
||||
* @par
|
||||
* ARM Limited (ARM) is supplying this software for use with Cortex-M
|
||||
* processor based microcontrollers. This file can be freely distributed
|
||||
* within development tools that are supporting such ARM based processors.
|
||||
*
|
||||
* @par
|
||||
* THIS SOFTWARE IS PROVIDED "AS IS". NO WARRANTIES, WHETHER EXPRESS, IMPLIED
|
||||
* OR STATUTORY, INCLUDING, BUT NOT LIMITED TO, IMPLIED WARRANTIES OF
|
||||
* MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE APPLY TO THIS SOFTWARE.
|
||||
* ARM SHALL NOT, IN ANY CIRCUMSTANCES, BE LIABLE FOR SPECIAL, INCIDENTAL, OR
|
||||
* CONSEQUENTIAL DAMAGES, FOR ANY REASON WHATSOEVER.
|
||||
*
|
||||
******************************************************************************/
|
||||
|
||||
#ifndef __CORE_CMFUNC_H__
|
||||
#define __CORE_CMFUNC_H__
|
||||
|
||||
/* ########################### Core Function Access ########################### */
|
||||
/** \ingroup CMSIS_Core_FunctionInterface
|
||||
\defgroup CMSIS_Core_RegAccFunctions CMSIS Core Register Access Functions
|
||||
@{
|
||||
*/
|
||||
|
||||
#if defined ( __CC_ARM ) /*------------------ RealView Compiler ----------------*/
|
||||
/* ARM armcc specific functions */
|
||||
|
||||
/* intrinsic void __enable_irq(); */
|
||||
/* intrinsic void __disable_irq(); */
|
||||
|
||||
/** \brief Get Control Register
|
||||
|
||||
This function returns the content of the Control Register.
|
||||
|
||||
\return Control Register value
|
||||
*/
|
||||
#if (__ARMCC_VERSION < 400000)
|
||||
extern uint32_t __get_CONTROL(void);
|
||||
#else /* (__ARMCC_VERSION >= 400000) */
|
||||
static __INLINE uint32_t __get_CONTROL(void)
|
||||
{
|
||||
register uint32_t __regControl __ASM("control");
|
||||
return(__regControl);
|
||||
}
|
||||
#endif /* __ARMCC_VERSION */
|
||||
|
||||
|
||||
/** \brief Set Control Register
|
||||
|
||||
This function writes the given value to the Control Register.
|
||||
|
||||
\param [in] control Control Register value to set
|
||||
*/
|
||||
#if (__ARMCC_VERSION < 400000)
|
||||
extern void __set_CONTROL(uint32_t control);
|
||||
#else /* (__ARMCC_VERSION >= 400000) */
|
||||
static __INLINE void __set_CONTROL(uint32_t control)
|
||||
{
|
||||
register uint32_t __regControl __ASM("control");
|
||||
__regControl = control;
|
||||
}
|
||||
#endif /* __ARMCC_VERSION */
|
||||
|
||||
|
||||
/** \brief Get ISPR Register
|
||||
|
||||
This function returns the content of the ISPR Register.
|
||||
|
||||
\return ISPR Register value
|
||||
*/
|
||||
#if (__ARMCC_VERSION < 400000)
|
||||
extern uint32_t __get_IPSR(void);
|
||||
#else /* (__ARMCC_VERSION >= 400000) */
|
||||
static __INLINE uint32_t __get_IPSR(void)
|
||||
{
|
||||
register uint32_t __regIPSR __ASM("ipsr");
|
||||
return(__regIPSR);
|
||||
}
|
||||
#endif /* __ARMCC_VERSION */
|
||||
|
||||
|
||||
/** \brief Get APSR Register
|
||||
|
||||
This function returns the content of the APSR Register.
|
||||
|
||||
\return APSR Register value
|
||||
*/
|
||||
#if (__ARMCC_VERSION < 400000)
|
||||
extern uint32_t __get_APSR(void);
|
||||
#else /* (__ARMCC_VERSION >= 400000) */
|
||||
static __INLINE uint32_t __get_APSR(void)
|
||||
{
|
||||
register uint32_t __regAPSR __ASM("apsr");
|
||||
return(__regAPSR);
|
||||
}
|
||||
#endif /* __ARMCC_VERSION */
|
||||
|
||||
|
||||
/** \brief Get xPSR Register
|
||||
|
||||
This function returns the content of the xPSR Register.
|
||||
|
||||
\return xPSR Register value
|
||||
*/
|
||||
#if (__ARMCC_VERSION < 400000)
|
||||
extern uint32_t __get_xPSR(void);
|
||||
#else /* (__ARMCC_VERSION >= 400000) */
|
||||
static __INLINE uint32_t __get_xPSR(void)
|
||||
{
|
||||
register uint32_t __regXPSR __ASM("xpsr");
|
||||
return(__regXPSR);
|
||||
}
|
||||
#endif /* __ARMCC_VERSION */
|
||||
|
||||
|
||||
/** \brief Get Process Stack Pointer
|
||||
|
||||
This function returns the current value of the Process Stack Pointer (PSP).
|
||||
|
||||
\return PSP Register value
|
||||
*/
|
||||
#if (__ARMCC_VERSION < 400000)
|
||||
extern uint32_t __get_PSP(void);
|
||||
#else /* (__ARMCC_VERSION >= 400000) */
|
||||
static __INLINE uint32_t __get_PSP(void)
|
||||
{
|
||||
register uint32_t __regProcessStackPointer __ASM("psp");
|
||||
return(__regProcessStackPointer);
|
||||
}
|
||||
#endif /* __ARMCC_VERSION */
|
||||
|
||||
|
||||
/** \brief Set Process Stack Pointer
|
||||
|
||||
This function assigns the given value to the Process Stack Pointer (PSP).
|
||||
|
||||
\param [in] topOfProcStack Process Stack Pointer value to set
|
||||
*/
|
||||
#if (__ARMCC_VERSION < 400000)
|
||||
extern void __set_PSP(uint32_t topOfProcStack);
|
||||
#else /* (__ARMCC_VERSION >= 400000) */
|
||||
static __INLINE void __set_PSP(uint32_t topOfProcStack)
|
||||
{
|
||||
register uint32_t __regProcessStackPointer __ASM("psp");
|
||||
__regProcessStackPointer = topOfProcStack;
|
||||
}
|
||||
#endif /* __ARMCC_VERSION */
|
||||
|
||||
|
||||
/** \brief Get Main Stack Pointer
|
||||
|
||||
This function returns the current value of the Main Stack Pointer (MSP).
|
||||
|
||||
\return MSP Register value
|
||||
*/
|
||||
#if (__ARMCC_VERSION < 400000)
|
||||
extern uint32_t __get_MSP(void);
|
||||
#else /* (__ARMCC_VERSION >= 400000) */
|
||||
static __INLINE uint32_t __get_MSP(void)
|
||||
{
|
||||
register uint32_t __regMainStackPointer __ASM("msp");
|
||||
return(__regMainStackPointer);
|
||||
}
|
||||
#endif /* __ARMCC_VERSION */
|
||||
|
||||
|
||||
/** \brief Set Main Stack Pointer
|
||||
|
||||
This function assigns the given value to the Main Stack Pointer (MSP).
|
||||
|
||||
\param [in] topOfMainStack Main Stack Pointer value to set
|
||||
*/
|
||||
#if (__ARMCC_VERSION < 400000)
|
||||
extern void __set_MSP(uint32_t topOfMainStack);
|
||||
#else /* (__ARMCC_VERSION >= 400000) */
|
||||
static __INLINE void __set_MSP(uint32_t topOfMainStack)
|
||||
{
|
||||
register uint32_t __regMainStackPointer __ASM("msp");
|
||||
__regMainStackPointer = topOfMainStack;
|
||||
}
|
||||
#endif /* __ARMCC_VERSION */
|
||||
|
||||
|
||||
/** \brief Get Priority Mask
|
||||
|
||||
This function returns the current state of the priority mask bit from the Priority Mask Register.
|
||||
|
||||
\return Priority Mask value
|
||||
*/
|
||||
#if (__ARMCC_VERSION < 400000)
|
||||
extern uint32_t __get_PRIMASK(void);
|
||||
#else /* (__ARMCC_VERSION >= 400000) */
|
||||
static __INLINE uint32_t __get_PRIMASK(void)
|
||||
{
|
||||
register uint32_t __regPriMask __ASM("primask");
|
||||
return(__regPriMask);
|
||||
}
|
||||
#endif /* __ARMCC_VERSION */
|
||||
|
||||
|
||||
/** \brief Set Priority Mask
|
||||
|
||||
This function assigns the given value to the Priority Mask Register.
|
||||
|
||||
\param [in] priMask Priority Mask
|
||||
*/
|
||||
#if (__ARMCC_VERSION < 400000)
|
||||
extern void __set_PRIMASK(uint32_t priMask);
|
||||
#else /* (__ARMCC_VERSION >= 400000) */
|
||||
static __INLINE void __set_PRIMASK(uint32_t priMask)
|
||||
{
|
||||
register uint32_t __regPriMask __ASM("primask");
|
||||
__regPriMask = (priMask);
|
||||
}
|
||||
#endif /* __ARMCC_VERSION */
|
||||
|
||||
|
||||
#if (__CORTEX_M >= 0x03)
|
||||
|
||||
/** \brief Enable FIQ
|
||||
|
||||
This function enables FIQ interrupts by clearing the F-bit in the CPSR.
|
||||
Can only be executed in Privileged modes.
|
||||
*/
|
||||
#define __enable_fault_irq __enable_fiq
|
||||
|
||||
|
||||
/** \brief Disable FIQ
|
||||
|
||||
This function disables FIQ interrupts by setting the F-bit in the CPSR.
|
||||
Can only be executed in Privileged modes.
|
||||
*/
|
||||
#define __disable_fault_irq __disable_fiq
|
||||
|
||||
|
||||
/** \brief Get Base Priority
|
||||
|
||||
This function returns the current value of the Base Priority register.
|
||||
|
||||
\return Base Priority register value
|
||||
*/
|
||||
#if (__ARMCC_VERSION < 400000)
|
||||
extern uint32_t __get_BASEPRI(void);
|
||||
#else /* (__ARMCC_VERSION >= 400000) */
|
||||
static __INLINE uint32_t __get_BASEPRI(void)
|
||||
{
|
||||
register uint32_t __regBasePri __ASM("basepri");
|
||||
return(__regBasePri);
|
||||
}
|
||||
#endif /* __ARMCC_VERSION */
|
||||
|
||||
|
||||
/** \brief Set Base Priority
|
||||
|
||||
This function assigns the given value to the Base Priority register.
|
||||
|
||||
\param [in] basePri Base Priority value to set
|
||||
*/
|
||||
#if (__ARMCC_VERSION < 400000)
|
||||
extern void __set_BASEPRI(uint32_t basePri);
|
||||
#else /* (__ARMCC_VERSION >= 400000) */
|
||||
static __INLINE void __set_BASEPRI(uint32_t basePri)
|
||||
{
|
||||
register uint32_t __regBasePri __ASM("basepri");
|
||||
__regBasePri = (basePri & 0xff);
|
||||
}
|
||||
#endif /* __ARMCC_VERSION */
|
||||
|
||||
|
||||
/** \brief Get Fault Mask
|
||||
|
||||
This function returns the current value of the Fault Mask register.
|
||||
|
||||
\return Fault Mask register value
|
||||
*/
|
||||
#if (__ARMCC_VERSION < 400000)
|
||||
extern uint32_t __get_FAULTMASK(void);
|
||||
#else /* (__ARMCC_VERSION >= 400000) */
|
||||
static __INLINE uint32_t __get_FAULTMASK(void)
|
||||
{
|
||||
register uint32_t __regFaultMask __ASM("faultmask");
|
||||
return(__regFaultMask);
|
||||
}
|
||||
#endif /* __ARMCC_VERSION */
|
||||
|
||||
|
||||
/** \brief Set Fault Mask
|
||||
|
||||
This function assigns the given value to the Fault Mask register.
|
||||
|
||||
\param [in] faultMask Fault Mask value to set
|
||||
*/
|
||||
#if (__ARMCC_VERSION < 400000)
|
||||
extern void __set_FAULTMASK(uint32_t faultMask);
|
||||
#else /* (__ARMCC_VERSION >= 400000) */
|
||||
static __INLINE void __set_FAULTMASK(uint32_t faultMask)
|
||||
{
|
||||
register uint32_t __regFaultMask __ASM("faultmask");
|
||||
__regFaultMask = (faultMask & 1);
|
||||
}
|
||||
#endif /* __ARMCC_VERSION */
|
||||
|
||||
#endif /* (__CORTEX_M >= 0x03) */
|
||||
|
||||
|
||||
#if (__CORTEX_M == 0x04)
|
||||
|
||||
/** \brief Get FPSCR
|
||||
|
||||
This function returns the current value of the Floating Point Status/Control register.
|
||||
|
||||
\return Floating Point Status/Control register value
|
||||
*/
|
||||
static __INLINE uint32_t __get_FPSCR(void)
|
||||
{
|
||||
#if (__FPU_PRESENT == 1)
|
||||
register uint32_t __regfpscr __ASM("fpscr");
|
||||
return(__regfpscr);
|
||||
#else
|
||||
return(0);
|
||||
#endif
|
||||
}
|
||||
|
||||
|
||||
/** \brief Set FPSCR
|
||||
|
||||
This function assigns the given value to the Floating Point Status/Control register.
|
||||
|
||||
\param [in] fpscr Floating Point Status/Control value to set
|
||||
*/
|
||||
static __INLINE void __set_FPSCR(uint32_t fpscr)
|
||||
{
|
||||
#if (__FPU_PRESENT == 1)
|
||||
register uint32_t __regfpscr __ASM("fpscr");
|
||||
__regfpscr = (fpscr);
|
||||
#endif
|
||||
}
|
||||
|
||||
#endif /* (__CORTEX_M == 0x04) */
|
||||
|
||||
|
||||
#elif (defined (__ICCARM__)) /*---------------- ICC Compiler ---------------------*/
|
||||
/* IAR iccarm specific functions */
|
||||
|
||||
#if defined (__ICCARM__)
|
||||
#include <intrinsics.h> /* IAR Intrinsics */
|
||||
#endif
|
||||
|
||||
#pragma diag_suppress=Pe940
|
||||
|
||||
/** \brief Enable IRQ Interrupts
|
||||
|
||||
This function enables IRQ interrupts by clearing the I-bit in the CPSR.
|
||||
Can only be executed in Privileged modes.
|
||||
*/
|
||||
#define __enable_irq __enable_interrupt
|
||||
|
||||
|
||||
/** \brief Disable IRQ Interrupts
|
||||
|
||||
This function disables IRQ interrupts by setting the I-bit in the CPSR.
|
||||
Can only be executed in Privileged modes.
|
||||
*/
|
||||
#define __disable_irq __disable_interrupt
|
||||
|
||||
|
||||
/* intrinsic unsigned long __get_CONTROL( void ); (see intrinsic.h) */
|
||||
/* intrinsic void __set_CONTROL( unsigned long ); (see intrinsic.h) */
|
||||
|
||||
|
||||
/** \brief Get ISPR Register
|
||||
|
||||
This function returns the content of the ISPR Register.
|
||||
|
||||
\return ISPR Register value
|
||||
*/
|
||||
static uint32_t __get_IPSR(void)
|
||||
{
|
||||
__ASM("mrs r0, ipsr");
|
||||
}
|
||||
|
||||
|
||||
/** \brief Get APSR Register
|
||||
|
||||
This function returns the content of the APSR Register.
|
||||
|
||||
\return APSR Register value
|
||||
*/
|
||||
static uint32_t __get_APSR(void)
|
||||
{
|
||||
__ASM("mrs r0, apsr");
|
||||
}
|
||||
|
||||
|
||||
/** \brief Get xPSR Register
|
||||
|
||||
This function returns the content of the xPSR Register.
|
||||
|
||||
\return xPSR Register value
|
||||
*/
|
||||
static uint32_t __get_xPSR(void)
|
||||
{
|
||||
__ASM("mrs r0, psr"); // assembler does not know "xpsr"
|
||||
}
|
||||
|
||||
|
||||
/** \brief Get Process Stack Pointer
|
||||
|
||||
This function returns the current value of the Process Stack Pointer (PSP).
|
||||
|
||||
\return PSP Register value
|
||||
*/
|
||||
static uint32_t __get_PSP(void)
|
||||
{
|
||||
__ASM("mrs r0, psp");
|
||||
}
|
||||
|
||||
|
||||
/** \brief Set Process Stack Pointer
|
||||
|
||||
This function assigns the given value to the Process Stack Pointer (PSP).
|
||||
|
||||
\param [in] topOfProcStack Process Stack Pointer value to set
|
||||
*/
|
||||
static void __set_PSP(uint32_t topOfProcStack)
|
||||
{
|
||||
__ASM("msr psp, r0");
|
||||
}
|
||||
|
||||
|
||||
/** \brief Get Main Stack Pointer
|
||||
|
||||
This function returns the current value of the Main Stack Pointer (MSP).
|
||||
|
||||
\return MSP Register value
|
||||
*/
|
||||
static uint32_t __get_MSP(void)
|
||||
{
|
||||
__ASM("mrs r0, msp");
|
||||
}
|
||||
|
||||
|
||||
/** \brief Set Main Stack Pointer
|
||||
|
||||
This function assigns the given value to the Main Stack Pointer (MSP).
|
||||
|
||||
\param [in] topOfMainStack Main Stack Pointer value to set
|
||||
*/
|
||||
static void __set_MSP(uint32_t topOfMainStack)
|
||||
{
|
||||
__ASM("msr msp, r0");
|
||||
}
|
||||
|
||||
|
||||
/* intrinsic unsigned long __get_PRIMASK( void ); (see intrinsic.h) */
|
||||
/* intrinsic void __set_PRIMASK( unsigned long ); (see intrinsic.h) */
|
||||
|
||||
|
||||
#if (__CORTEX_M >= 0x03)
|
||||
|
||||
/** \brief Enable FIQ
|
||||
|
||||
This function enables FIQ interrupts by clearing the F-bit in the CPSR.
|
||||
Can only be executed in Privileged modes.
|
||||
*/
|
||||
static __INLINE void __enable_fault_irq(void)
|
||||
{
|
||||
__ASM ("cpsie f");
|
||||
}
|
||||
|
||||
|
||||
/** \brief Disable FIQ
|
||||
|
||||
This function disables FIQ interrupts by setting the F-bit in the CPSR.
|
||||
Can only be executed in Privileged modes.
|
||||
*/
|
||||
static __INLINE void __disable_fault_irq(void)
|
||||
{
|
||||
__ASM ("cpsid f");
|
||||
}
|
||||
|
||||
|
||||
/* intrinsic unsigned long __get_BASEPRI( void ); (see intrinsic.h) */
|
||||
/* intrinsic void __set_BASEPRI( unsigned long ); (see intrinsic.h) */
|
||||
/* intrinsic unsigned long __get_FAULTMASK( void ); (see intrinsic.h) */
|
||||
/* intrinsic void __set_FAULTMASK(unsigned long); (see intrinsic.h) */
|
||||
|
||||
#endif /* (__CORTEX_M >= 0x03) */
|
||||
|
||||
|
||||
#if (__CORTEX_M == 0x04)
|
||||
|
||||
/** \brief Get FPSCR
|
||||
|
||||
This function returns the current value of the Floating Point Status/Control register.
|
||||
|
||||
\return Floating Point Status/Control register value
|
||||
*/
|
||||
static uint32_t __get_FPSCR(void)
|
||||
{
|
||||
#if (__FPU_PRESENT == 1)
|
||||
__ASM("vmrs r0, fpscr");
|
||||
#else
|
||||
return(0);
|
||||
#endif
|
||||
}
|
||||
|
||||
|
||||
/** \brief Set FPSCR
|
||||
|
||||
This function assigns the given value to the Floating Point Status/Control register.
|
||||
|
||||
\param [in] fpscr Floating Point Status/Control value to set
|
||||
*/
|
||||
static void __set_FPSCR(uint32_t fpscr)
|
||||
{
|
||||
#if (__FPU_PRESENT == 1)
|
||||
__ASM("vmsr fpscr, r0");
|
||||
#endif
|
||||
}
|
||||
|
||||
#endif /* (__CORTEX_M == 0x04) */
|
||||
|
||||
#pragma diag_default=Pe940
|
||||
|
||||
|
||||
#elif (defined (__GNUC__)) /*------------------ GNU Compiler ---------------------*/
|
||||
/* GNU gcc specific functions */
|
||||
|
||||
/** \brief Enable IRQ Interrupts
|
||||
|
||||
This function enables IRQ interrupts by clearing the I-bit in the CPSR.
|
||||
Can only be executed in Privileged modes.
|
||||
*/
|
||||
__attribute__( ( always_inline ) ) static __INLINE void __enable_irq(void)
|
||||
{
|
||||
__ASM volatile ("cpsie i");
|
||||
}
|
||||
|
||||
|
||||
/** \brief Disable IRQ Interrupts
|
||||
|
||||
This function disables IRQ interrupts by setting the I-bit in the CPSR.
|
||||
Can only be executed in Privileged modes.
|
||||
*/
|
||||
__attribute__( ( always_inline ) ) static __INLINE void __disable_irq(void)
|
||||
{
|
||||
__ASM volatile ("cpsid i");
|
||||
}
|
||||
|
||||
|
||||
/** \brief Get Control Register
|
||||
|
||||
This function returns the content of the Control Register.
|
||||
|
||||
\return Control Register value
|
||||
*/
|
||||
__attribute__( ( always_inline ) ) static __INLINE uint32_t __get_CONTROL(void)
|
||||
{
|
||||
uint32_t result;
|
||||
|
||||
__ASM volatile ("MRS %0, control" : "=r" (result) );
|
||||
return(result);
|
||||
}
|
||||
|
||||
|
||||
/** \brief Set Control Register
|
||||
|
||||
This function writes the given value to the Control Register.
|
||||
|
||||
\param [in] control Control Register value to set
|
||||
*/
|
||||
__attribute__( ( always_inline ) ) static __INLINE void __set_CONTROL(uint32_t control)
|
||||
{
|
||||
__ASM volatile ("MSR control, %0" : : "r" (control) );
|
||||
}
|
||||
|
||||
|
||||
/** \brief Get ISPR Register
|
||||
|
||||
This function returns the content of the ISPR Register.
|
||||
|
||||
\return ISPR Register value
|
||||
*/
|
||||
__attribute__( ( always_inline ) ) static __INLINE uint32_t __get_IPSR(void)
|
||||
{
|
||||
uint32_t result;
|
||||
|
||||
__ASM volatile ("MRS %0, ipsr" : "=r" (result) );
|
||||
return(result);
|
||||
}
|
||||
|
||||
|
||||
/** \brief Get APSR Register
|
||||
|
||||
This function returns the content of the APSR Register.
|
||||
|
||||
\return APSR Register value
|
||||
*/
|
||||
__attribute__( ( always_inline ) ) static __INLINE uint32_t __get_APSR(void)
|
||||
{
|
||||
uint32_t result;
|
||||
|
||||
__ASM volatile ("MRS %0, apsr" : "=r" (result) );
|
||||
return(result);
|
||||
}
|
||||
|
||||
|
||||
/** \brief Get xPSR Register
|
||||
|
||||
This function returns the content of the xPSR Register.
|
||||
|
||||
\return xPSR Register value
|
||||
*/
|
||||
__attribute__( ( always_inline ) ) static __INLINE uint32_t __get_xPSR(void)
|
||||
{
|
||||
uint32_t result;
|
||||
|
||||
__ASM volatile ("MRS %0, xpsr" : "=r" (result) );
|
||||
return(result);
|
||||
}
|
||||
|
||||
|
||||
/** \brief Get Process Stack Pointer
|
||||
|
||||
This function returns the current value of the Process Stack Pointer (PSP).
|
||||
|
||||
\return PSP Register value
|
||||
*/
|
||||
__attribute__( ( always_inline ) ) static __INLINE uint32_t __get_PSP(void)
|
||||
{
|
||||
register uint32_t result;
|
||||
|
||||
__ASM volatile ("MRS %0, psp\n" : "=r" (result) );
|
||||
return(result);
|
||||
}
|
||||
|
||||
|
||||
/** \brief Set Process Stack Pointer
|
||||
|
||||
This function assigns the given value to the Process Stack Pointer (PSP).
|
||||
|
||||
\param [in] topOfProcStack Process Stack Pointer value to set
|
||||
*/
|
||||
__attribute__( ( always_inline ) ) static __INLINE void __set_PSP(uint32_t topOfProcStack)
|
||||
{
|
||||
__ASM volatile ("MSR psp, %0\n" : : "r" (topOfProcStack) );
|
||||
}
|
||||
|
||||
|
||||
/** \brief Get Main Stack Pointer
|
||||
|
||||
This function returns the current value of the Main Stack Pointer (MSP).
|
||||
|
||||
\return MSP Register value
|
||||
*/
|
||||
__attribute__( ( always_inline ) ) static __INLINE uint32_t __get_MSP(void)
|
||||
{
|
||||
register uint32_t result;
|
||||
|
||||
__ASM volatile ("MRS %0, msp\n" : "=r" (result) );
|
||||
return(result);
|
||||
}
|
||||
|
||||
|
||||
/** \brief Set Main Stack Pointer
|
||||
|
||||
This function assigns the given value to the Main Stack Pointer (MSP).
|
||||
|
||||
\param [in] topOfMainStack Main Stack Pointer value to set
|
||||
*/
|
||||
__attribute__( ( always_inline ) ) static __INLINE void __set_MSP(uint32_t topOfMainStack)
|
||||
{
|
||||
__ASM volatile ("MSR msp, %0\n" : : "r" (topOfMainStack) );
|
||||
}
|
||||
|
||||
|
||||
/** \brief Get Priority Mask
|
||||
|
||||
This function returns the current state of the priority mask bit from the Priority Mask Register.
|
||||
|
||||
\return Priority Mask value
|
||||
*/
|
||||
__attribute__( ( always_inline ) ) static __INLINE uint32_t __get_PRIMASK(void)
|
||||
{
|
||||
uint32_t result;
|
||||
|
||||
__ASM volatile ("MRS %0, primask" : "=r" (result) );
|
||||
return(result);
|
||||
}
|
||||
|
||||
|
||||
/** \brief Set Priority Mask
|
||||
|
||||
This function assigns the given value to the Priority Mask Register.
|
||||
|
||||
\param [in] priMask Priority Mask
|
||||
*/
|
||||
__attribute__( ( always_inline ) ) static __INLINE void __set_PRIMASK(uint32_t priMask)
|
||||
{
|
||||
__ASM volatile ("MSR primask, %0" : : "r" (priMask) );
|
||||
}
|
||||
|
||||
|
||||
#if (__CORTEX_M >= 0x03)
|
||||
|
||||
/** \brief Enable FIQ
|
||||
|
||||
This function enables FIQ interrupts by clearing the F-bit in the CPSR.
|
||||
Can only be executed in Privileged modes.
|
||||
*/
|
||||
__attribute__( ( always_inline ) ) static __INLINE void __enable_fault_irq(void)
|
||||
{
|
||||
__ASM volatile ("cpsie f");
|
||||
}
|
||||
|
||||
|
||||
/** \brief Disable FIQ
|
||||
|
||||
This function disables FIQ interrupts by setting the F-bit in the CPSR.
|
||||
Can only be executed in Privileged modes.
|
||||
*/
|
||||
__attribute__( ( always_inline ) ) static __INLINE void __disable_fault_irq(void)
|
||||
{
|
||||
__ASM volatile ("cpsid f");
|
||||
}
|
||||
|
||||
|
||||
/** \brief Get Base Priority
|
||||
|
||||
This function returns the current value of the Base Priority register.
|
||||
|
||||
\return Base Priority register value
|
||||
*/
|
||||
__attribute__( ( always_inline ) ) static __INLINE uint32_t __get_BASEPRI(void)
|
||||
{
|
||||
uint32_t result;
|
||||
|
||||
__ASM volatile ("MRS %0, basepri_max" : "=r" (result) );
|
||||
return(result);
|
||||
}
|
||||
|
||||
|
||||
/** \brief Set Base Priority
|
||||
|
||||
This function assigns the given value to the Base Priority register.
|
||||
|
||||
\param [in] basePri Base Priority value to set
|
||||
*/
|
||||
__attribute__( ( always_inline ) ) static __INLINE void __set_BASEPRI(uint32_t value)
|
||||
{
|
||||
__ASM volatile ("MSR basepri, %0" : : "r" (value) );
|
||||
}
|
||||
|
||||
|
||||
/** \brief Get Fault Mask
|
||||
|
||||
This function returns the current value of the Fault Mask register.
|
||||
|
||||
\return Fault Mask register value
|
||||
*/
|
||||
__attribute__( ( always_inline ) ) static __INLINE uint32_t __get_FAULTMASK(void)
|
||||
{
|
||||
uint32_t result;
|
||||
|
||||
__ASM volatile ("MRS %0, faultmask" : "=r" (result) );
|
||||
return(result);
|
||||
}
|
||||
|
||||
|
||||
/** \brief Set Fault Mask
|
||||
|
||||
This function assigns the given value to the Fault Mask register.
|
||||
|
||||
\param [in] faultMask Fault Mask value to set
|
||||
*/
|
||||
__attribute__( ( always_inline ) ) static __INLINE void __set_FAULTMASK(uint32_t faultMask)
|
||||
{
|
||||
__ASM volatile ("MSR faultmask, %0" : : "r" (faultMask) );
|
||||
}
|
||||
|
||||
#endif /* (__CORTEX_M >= 0x03) */
|
||||
|
||||
|
||||
#if (__CORTEX_M == 0x04)
|
||||
|
||||
/** \brief Get FPSCR
|
||||
|
||||
This function returns the current value of the Floating Point Status/Control register.
|
||||
|
||||
\return Floating Point Status/Control register value
|
||||
*/
|
||||
__attribute__( ( always_inline ) ) static __INLINE uint32_t __get_FPSCR(void)
|
||||
{
|
||||
#if (__FPU_PRESENT == 1)
|
||||
uint32_t result;
|
||||
|
||||
__ASM volatile ("MRS %0, fpscr" : "=r" (result) );
|
||||
return(result);
|
||||
#else
|
||||
return(0);
|
||||
#endif
|
||||
}
|
||||
|
||||
|
||||
/** \brief Set FPSCR
|
||||
|
||||
This function assigns the given value to the Floating Point Status/Control register.
|
||||
|
||||
\param [in] fpscr Floating Point Status/Control value to set
|
||||
*/
|
||||
__attribute__( ( always_inline ) ) static __INLINE void __set_FPSCR(uint32_t fpscr)
|
||||
{
|
||||
#if (__FPU_PRESENT == 1)
|
||||
__ASM volatile ("MSR fpscr, %0" : : "r" (fpscr) );
|
||||
#endif
|
||||
}
|
||||
|
||||
#endif /* (__CORTEX_M == 0x04) */
|
||||
|
||||
|
||||
#elif (defined (__TASKING__)) /*--------------- TASKING Compiler -----------------*/
|
||||
/* TASKING carm specific functions */
|
||||
|
||||
/*
|
||||
* The CMSIS functions have been implemented as intrinsics in the compiler.
|
||||
* Please use "carm -?i" to get an up to date list of all instrinsics,
|
||||
* Including the CMSIS ones.
|
||||
*/
|
||||
|
||||
#endif
|
||||
|
||||
/*@} end of CMSIS_Core_RegAccFunctions */
|
||||
|
||||
|
||||
#endif /* __CORE_CMFUNC_H__ */
|
||||
775
platforms/lpc17xx/CMSISv2p00_LPC17xx/inc/core_cmInstr.h
Normal file
775
platforms/lpc17xx/CMSISv2p00_LPC17xx/inc/core_cmInstr.h
Normal file
@@ -0,0 +1,775 @@
|
||||
/**************************************************************************//**
|
||||
* @file core_cmInstr.h
|
||||
* @brief CMSIS Cortex-M Core Instruction Access Header File
|
||||
* @version V2.01
|
||||
* @date 06. December 2010
|
||||
*
|
||||
* @note
|
||||
* Copyright (C) 2009-2010 ARM Limited. All rights reserved.
|
||||
*
|
||||
* @par
|
||||
* ARM Limited (ARM) is supplying this software for use with Cortex-M
|
||||
* processor based microcontrollers. This file can be freely distributed
|
||||
* within development tools that are supporting such ARM based processors.
|
||||
*
|
||||
* @par
|
||||
* THIS SOFTWARE IS PROVIDED "AS IS". NO WARRANTIES, WHETHER EXPRESS, IMPLIED
|
||||
* OR STATUTORY, INCLUDING, BUT NOT LIMITED TO, IMPLIED WARRANTIES OF
|
||||
* MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE APPLY TO THIS SOFTWARE.
|
||||
* ARM SHALL NOT, IN ANY CIRCUMSTANCES, BE LIABLE FOR SPECIAL, INCIDENTAL, OR
|
||||
* CONSEQUENTIAL DAMAGES, FOR ANY REASON WHATSOEVER.
|
||||
*
|
||||
******************************************************************************/
|
||||
|
||||
#ifndef __CORE_CMINSTR_H__
|
||||
#define __CORE_CMINSTR_H__
|
||||
|
||||
|
||||
/* ########################## Core Instruction Access ######################### */
|
||||
/** \defgroup CMSIS_Core_InstructionInterface CMSIS Core Instruction Interface
|
||||
Access to dedicated instructions
|
||||
@{
|
||||
*/
|
||||
|
||||
#if defined ( __CC_ARM ) /*------------------ RealView Compiler ----------------*/
|
||||
/* ARM armcc specific functions */
|
||||
|
||||
/** \brief No Operation
|
||||
|
||||
No Operation does nothing. This instruction can be used for code alignment purposes.
|
||||
*/
|
||||
#define __NOP __nop
|
||||
|
||||
|
||||
/** \brief Wait For Interrupt
|
||||
|
||||
Wait For Interrupt is a hint instruction that suspends execution
|
||||
until one of a number of events occurs.
|
||||
*/
|
||||
#define __WFI __wfi
|
||||
|
||||
|
||||
/** \brief Wait For Event
|
||||
|
||||
Wait For Event is a hint instruction that permits the processor to enter
|
||||
a low-power state until one of a number of events occurs.
|
||||
*/
|
||||
#define __WFE __wfe
|
||||
|
||||
|
||||
/** \brief Send Event
|
||||
|
||||
Send Event is a hint instruction. It causes an event to be signaled to the CPU.
|
||||
*/
|
||||
#define __SEV __sev
|
||||
|
||||
|
||||
/** \brief Instruction Synchronization Barrier
|
||||
|
||||
Instruction Synchronization Barrier flushes the pipeline in the processor,
|
||||
so that all instructions following the ISB are fetched from cache or
|
||||
memory, after the instruction has been completed.
|
||||
*/
|
||||
#define __ISB() __isb(0xF)
|
||||
|
||||
|
||||
/** \brief Data Synchronization Barrier
|
||||
|
||||
This function acts as a special kind of Data Memory Barrier.
|
||||
It completes when all explicit memory accesses before this instruction complete.
|
||||
*/
|
||||
#define __DSB() __dsb(0xF)
|
||||
|
||||
|
||||
/** \brief Data Memory Barrier
|
||||
|
||||
This function ensures the apparent order of the explicit memory operations before
|
||||
and after the instruction, without ensuring their completion.
|
||||
*/
|
||||
#define __DMB() __dmb(0xF)
|
||||
|
||||
|
||||
/** \brief Reverse byte order (32 bit)
|
||||
|
||||
This function reverses the byte order in integer value.
|
||||
|
||||
\param [in] value Value to reverse
|
||||
\return Reversed value
|
||||
*/
|
||||
#define __REV __rev
|
||||
|
||||
|
||||
/** \brief Reverse byte order (16 bit)
|
||||
|
||||
This function reverses the byte order in two unsigned short values.
|
||||
|
||||
\param [in] value Value to reverse
|
||||
\return Reversed value
|
||||
*/
|
||||
#if (__ARMCC_VERSION < 400677)
|
||||
extern uint32_t __REV16(uint32_t value);
|
||||
#else /* (__ARMCC_VERSION >= 400677) */
|
||||
static __INLINE __ASM uint32_t __REV16(uint32_t value)
|
||||
{
|
||||
rev16 r0, r0
|
||||
bx lr
|
||||
}
|
||||
#endif /* __ARMCC_VERSION */
|
||||
|
||||
|
||||
/** \brief Reverse byte order in signed short value
|
||||
|
||||
This function reverses the byte order in a signed short value with sign extension to integer.
|
||||
|
||||
\param [in] value Value to reverse
|
||||
\return Reversed value
|
||||
*/
|
||||
#if (__ARMCC_VERSION < 400677)
|
||||
extern int32_t __REVSH(int32_t value);
|
||||
#else /* (__ARMCC_VERSION >= 400677) */
|
||||
static __INLINE __ASM int32_t __REVSH(int32_t value)
|
||||
{
|
||||
revsh r0, r0
|
||||
bx lr
|
||||
}
|
||||
#endif /* __ARMCC_VERSION */
|
||||
|
||||
|
||||
#if (__CORTEX_M >= 0x03)
|
||||
|
||||
/** \brief Reverse bit order of value
|
||||
|
||||
This function reverses the bit order of the given value.
|
||||
|
||||
\param [in] value Value to reverse
|
||||
\return Reversed value
|
||||
*/
|
||||
#define __RBIT __rbit
|
||||
|
||||
|
||||
/** \brief LDR Exclusive (8 bit)
|
||||
|
||||
This function performs a exclusive LDR command for 8 bit value.
|
||||
|
||||
\param [in] ptr Pointer to data
|
||||
\return value of type uint8_t at (*ptr)
|
||||
*/
|
||||
#define __LDREXB(ptr) ((uint8_t ) __ldrex(ptr))
|
||||
|
||||
|
||||
/** \brief LDR Exclusive (16 bit)
|
||||
|
||||
This function performs a exclusive LDR command for 16 bit values.
|
||||
|
||||
\param [in] ptr Pointer to data
|
||||
\return value of type uint16_t at (*ptr)
|
||||
*/
|
||||
#define __LDREXH(ptr) ((uint16_t) __ldrex(ptr))
|
||||
|
||||
|
||||
/** \brief LDR Exclusive (32 bit)
|
||||
|
||||
This function performs a exclusive LDR command for 32 bit values.
|
||||
|
||||
\param [in] ptr Pointer to data
|
||||
\return value of type uint32_t at (*ptr)
|
||||
*/
|
||||
#define __LDREXW(ptr) ((uint32_t ) __ldrex(ptr))
|
||||
|
||||
|
||||
/** \brief STR Exclusive (8 bit)
|
||||
|
||||
This function performs a exclusive STR command for 8 bit values.
|
||||
|
||||
\param [in] value Value to store
|
||||
\param [in] ptr Pointer to location
|
||||
\return 0 Function succeeded
|
||||
\return 1 Function failed
|
||||
*/
|
||||
#define __STREXB(value, ptr) __strex(value, ptr)
|
||||
|
||||
|
||||
/** \brief STR Exclusive (16 bit)
|
||||
|
||||
This function performs a exclusive STR command for 16 bit values.
|
||||
|
||||
\param [in] value Value to store
|
||||
\param [in] ptr Pointer to location
|
||||
\return 0 Function succeeded
|
||||
\return 1 Function failed
|
||||
*/
|
||||
#define __STREXH(value, ptr) __strex(value, ptr)
|
||||
|
||||
|
||||
/** \brief STR Exclusive (32 bit)
|
||||
|
||||
This function performs a exclusive STR command for 32 bit values.
|
||||
|
||||
\param [in] value Value to store
|
||||
\param [in] ptr Pointer to location
|
||||
\return 0 Function succeeded
|
||||
\return 1 Function failed
|
||||
*/
|
||||
#define __STREXW(value, ptr) __strex(value, ptr)
|
||||
|
||||
|
||||
/** \brief Remove the exclusive lock
|
||||
|
||||
This function removes the exclusive lock which is created by LDREX.
|
||||
|
||||
*/
|
||||
#if (__ARMCC_VERSION < 400000)
|
||||
extern void __CLREX(void);
|
||||
#else /* (__ARMCC_VERSION >= 400000) */
|
||||
#define __CLREX __clrex
|
||||
#endif /* __ARMCC_VERSION */
|
||||
|
||||
|
||||
/** \brief Signed Saturate
|
||||
|
||||
This function saturates a signed value.
|
||||
|
||||
\param [in] value Value to be saturated
|
||||
\param [in] sat Bit position to saturate to (1..32)
|
||||
\return Saturated value
|
||||
*/
|
||||
#define __SSAT __ssat
|
||||
|
||||
|
||||
/** \brief Unsigned Saturate
|
||||
|
||||
This function saturates an unsigned value.
|
||||
|
||||
\param [in] value Value to be saturated
|
||||
\param [in] sat Bit position to saturate to (0..31)
|
||||
\return Saturated value
|
||||
*/
|
||||
#define __USAT __usat
|
||||
|
||||
|
||||
/** \brief Count leading zeros
|
||||
|
||||
This function counts the number of leading zeros of a data value.
|
||||
|
||||
\param [in] value Value to count the leading zeros
|
||||
\return number of leading zeros in value
|
||||
*/
|
||||
#define __CLZ __clz
|
||||
|
||||
#endif /* (__CORTEX_M >= 0x03) */
|
||||
|
||||
|
||||
|
||||
#elif (defined (__ICCARM__)) /*---------------- ICC Compiler ---------------------*/
|
||||
/* IAR iccarm specific functions */
|
||||
|
||||
#include <intrinsics.h> /* IAR Intrinsics */
|
||||
|
||||
#pragma diag_suppress=Pe940
|
||||
|
||||
/** \brief No Operation
|
||||
|
||||
No Operation does nothing. This instruction can be used for code alignment purposes.
|
||||
*/
|
||||
#define __NOP __no_operation
|
||||
|
||||
|
||||
/** \brief Wait For Interrupt
|
||||
|
||||
Wait For Interrupt is a hint instruction that suspends execution
|
||||
until one of a number of events occurs.
|
||||
*/
|
||||
static __INLINE void __WFI(void)
|
||||
{
|
||||
__ASM ("wfi");
|
||||
}
|
||||
|
||||
|
||||
/** \brief Wait For Event
|
||||
|
||||
Wait For Event is a hint instruction that permits the processor to enter
|
||||
a low-power state until one of a number of events occurs.
|
||||
*/
|
||||
static __INLINE void __WFE(void)
|
||||
{
|
||||
__ASM ("wfe");
|
||||
}
|
||||
|
||||
|
||||
/** \brief Send Event
|
||||
|
||||
Send Event is a hint instruction. It causes an event to be signaled to the CPU.
|
||||
*/
|
||||
static __INLINE void __SEV(void)
|
||||
{
|
||||
__ASM ("sev");
|
||||
}
|
||||
|
||||
|
||||
/* intrinsic void __ISB(void) (see intrinsics.h) */
|
||||
/* intrinsic void __DSB(void) (see intrinsics.h) */
|
||||
/* intrinsic void __DMB(void) (see intrinsics.h) */
|
||||
/* intrinsic uint32_t __REV(uint32_t value) (see intrinsics.h) */
|
||||
/* intrinsic __SSAT (see intrinsics.h) */
|
||||
/* intrinsic __USAT (see intrinsics.h) */
|
||||
|
||||
|
||||
/** \brief Reverse byte order (16 bit)
|
||||
|
||||
This function reverses the byte order in two unsigned short values.
|
||||
|
||||
\param [in] value Value to reverse
|
||||
\return Reversed value
|
||||
*/
|
||||
static uint32_t __REV16(uint32_t value)
|
||||
{
|
||||
__ASM("rev16 r0, r0");
|
||||
}
|
||||
|
||||
|
||||
/* intrinsic uint32_t __REVSH(uint32_t value) (see intrinsics.h */
|
||||
|
||||
|
||||
#if (__CORTEX_M >= 0x03)
|
||||
|
||||
/** \brief Reverse bit order of value
|
||||
|
||||
This function reverses the bit order of the given value.
|
||||
|
||||
\param [in] value Value to reverse
|
||||
\return Reversed value
|
||||
*/
|
||||
static uint32_t __RBIT(uint32_t value)
|
||||
{
|
||||
__ASM("rbit r0, r0");
|
||||
}
|
||||
|
||||
|
||||
/** \brief LDR Exclusive (8 bit)
|
||||
|
||||
This function performs a exclusive LDR command for 8 bit value.
|
||||
|
||||
\param [in] ptr Pointer to data
|
||||
\return value of type uint8_t at (*ptr)
|
||||
*/
|
||||
static uint8_t __LDREXB(volatile uint8_t *addr)
|
||||
{
|
||||
__ASM("ldrexb r0, [r0]");
|
||||
}
|
||||
|
||||
|
||||
/** \brief LDR Exclusive (16 bit)
|
||||
|
||||
This function performs a exclusive LDR command for 16 bit values.
|
||||
|
||||
\param [in] ptr Pointer to data
|
||||
\return value of type uint16_t at (*ptr)
|
||||
*/
|
||||
static uint16_t __LDREXH(volatile uint16_t *addr)
|
||||
{
|
||||
__ASM("ldrexh r0, [r0]");
|
||||
}
|
||||
|
||||
|
||||
/** \brief LDR Exclusive (32 bit)
|
||||
|
||||
This function performs a exclusive LDR command for 32 bit values.
|
||||
|
||||
\param [in] ptr Pointer to data
|
||||
\return value of type uint32_t at (*ptr)
|
||||
*/
|
||||
/* intrinsic unsigned long __LDREX(unsigned long *) (see intrinsics.h) */
|
||||
static uint32_t __LDREXW(volatile uint32_t *addr)
|
||||
{
|
||||
__ASM("ldrex r0, [r0]");
|
||||
}
|
||||
|
||||
|
||||
/** \brief STR Exclusive (8 bit)
|
||||
|
||||
This function performs a exclusive STR command for 8 bit values.
|
||||
|
||||
\param [in] value Value to store
|
||||
\param [in] ptr Pointer to location
|
||||
\return 0 Function succeeded
|
||||
\return 1 Function failed
|
||||
*/
|
||||
static uint32_t __STREXB(uint8_t value, volatile uint8_t *addr)
|
||||
{
|
||||
__ASM("strexb r0, r0, [r1]");
|
||||
}
|
||||
|
||||
|
||||
/** \brief STR Exclusive (16 bit)
|
||||
|
||||
This function performs a exclusive STR command for 16 bit values.
|
||||
|
||||
\param [in] value Value to store
|
||||
\param [in] ptr Pointer to location
|
||||
\return 0 Function succeeded
|
||||
\return 1 Function failed
|
||||
*/
|
||||
static uint32_t __STREXH(uint16_t value, volatile uint16_t *addr)
|
||||
{
|
||||
__ASM("strexh r0, r0, [r1]");
|
||||
}
|
||||
|
||||
|
||||
/** \brief STR Exclusive (32 bit)
|
||||
|
||||
This function performs a exclusive STR command for 32 bit values.
|
||||
|
||||
\param [in] value Value to store
|
||||
\param [in] ptr Pointer to location
|
||||
\return 0 Function succeeded
|
||||
\return 1 Function failed
|
||||
*/
|
||||
/* intrinsic unsigned long __STREX(unsigned long, unsigned long) (see intrinsics.h )*/
|
||||
static uint32_t __STREXW(uint32_t value, volatile uint32_t *addr)
|
||||
{
|
||||
__ASM("strex r0, r0, [r1]");
|
||||
}
|
||||
|
||||
|
||||
/** \brief Remove the exclusive lock
|
||||
|
||||
This function removes the exclusive lock which is created by LDREX.
|
||||
|
||||
*/
|
||||
static __INLINE void __CLREX(void)
|
||||
{
|
||||
__ASM ("clrex");
|
||||
}
|
||||
|
||||
/* intrinsic unsigned char __CLZ( unsigned long ) (see intrinsics.h) */
|
||||
|
||||
#endif /* (__CORTEX_M >= 0x03) */
|
||||
|
||||
#pragma diag_default=Pe940
|
||||
|
||||
|
||||
|
||||
#elif (defined (__GNUC__)) /*------------------ GNU Compiler ---------------------*/
|
||||
/* GNU gcc specific functions */
|
||||
|
||||
/** \brief No Operation
|
||||
|
||||
No Operation does nothing. This instruction can be used for code alignment purposes.
|
||||
*/
|
||||
__attribute__( ( always_inline ) ) static __INLINE void __NOP(void)
|
||||
{
|
||||
__ASM volatile ("nop");
|
||||
}
|
||||
|
||||
|
||||
/** \brief Wait For Interrupt
|
||||
|
||||
Wait For Interrupt is a hint instruction that suspends execution
|
||||
until one of a number of events occurs.
|
||||
*/
|
||||
__attribute__( ( always_inline ) ) static __INLINE void __WFI(void)
|
||||
{
|
||||
__ASM volatile ("wfi");
|
||||
}
|
||||
|
||||
|
||||
/** \brief Wait For Event
|
||||
|
||||
Wait For Event is a hint instruction that permits the processor to enter
|
||||
a low-power state until one of a number of events occurs.
|
||||
*/
|
||||
__attribute__( ( always_inline ) ) static __INLINE void __WFE(void)
|
||||
{
|
||||
__ASM volatile ("wfe");
|
||||
}
|
||||
|
||||
|
||||
/** \brief Send Event
|
||||
|
||||
Send Event is a hint instruction. It causes an event to be signaled to the CPU.
|
||||
*/
|
||||
__attribute__( ( always_inline ) ) static __INLINE void __SEV(void)
|
||||
{
|
||||
__ASM volatile ("sev");
|
||||
}
|
||||
|
||||
|
||||
/** \brief Instruction Synchronization Barrier
|
||||
|
||||
Instruction Synchronization Barrier flushes the pipeline in the processor,
|
||||
so that all instructions following the ISB are fetched from cache or
|
||||
memory, after the instruction has been completed.
|
||||
*/
|
||||
__attribute__( ( always_inline ) ) static __INLINE void __ISB(void)
|
||||
{
|
||||
__ASM volatile ("isb");
|
||||
}
|
||||
|
||||
|
||||
/** \brief Data Synchronization Barrier
|
||||
|
||||
This function acts as a special kind of Data Memory Barrier.
|
||||
It completes when all explicit memory accesses before this instruction complete.
|
||||
*/
|
||||
__attribute__( ( always_inline ) ) static __INLINE void __DSB(void)
|
||||
{
|
||||
__ASM volatile ("dsb");
|
||||
}
|
||||
|
||||
|
||||
/** \brief Data Memory Barrier
|
||||
|
||||
This function ensures the apparent order of the explicit memory operations before
|
||||
and after the instruction, without ensuring their completion.
|
||||
*/
|
||||
__attribute__( ( always_inline ) ) static __INLINE void __DMB(void)
|
||||
{
|
||||
__ASM volatile ("dmb");
|
||||
}
|
||||
|
||||
|
||||
/** \brief Reverse byte order (32 bit)
|
||||
|
||||
This function reverses the byte order in integer value.
|
||||
|
||||
\param [in] value Value to reverse
|
||||
\return Reversed value
|
||||
*/
|
||||
__attribute__( ( always_inline ) ) static __INLINE uint32_t __REV(uint32_t value)
|
||||
{
|
||||
uint32_t result;
|
||||
|
||||
__ASM volatile ("rev %0, %1" : "=r" (result) : "r" (value) );
|
||||
return(result);
|
||||
}
|
||||
|
||||
|
||||
/** \brief Reverse byte order (16 bit)
|
||||
|
||||
This function reverses the byte order in two unsigned short values.
|
||||
|
||||
\param [in] value Value to reverse
|
||||
\return Reversed value
|
||||
*/
|
||||
__attribute__( ( always_inline ) ) static __INLINE uint32_t __REV16(uint32_t value)
|
||||
{
|
||||
uint32_t result;
|
||||
|
||||
__ASM volatile ("rev16 %0, %1" : "=r" (result) : "r" (value) );
|
||||
return(result);
|
||||
}
|
||||
|
||||
|
||||
/** \brief Reverse byte order in signed short value
|
||||
|
||||
This function reverses the byte order in a signed short value with sign extension to integer.
|
||||
|
||||
\param [in] value Value to reverse
|
||||
\return Reversed value
|
||||
*/
|
||||
__attribute__( ( always_inline ) ) static __INLINE int32_t __REVSH(int32_t value)
|
||||
{
|
||||
uint32_t result;
|
||||
|
||||
__ASM volatile ("revsh %0, %1" : "=r" (result) : "r" (value) );
|
||||
return(result);
|
||||
}
|
||||
|
||||
|
||||
#if (__CORTEX_M >= 0x03)
|
||||
|
||||
/** \brief Reverse bit order of value
|
||||
|
||||
This function reverses the bit order of the given value.
|
||||
|
||||
\param [in] value Value to reverse
|
||||
\return Reversed value
|
||||
*/
|
||||
__attribute__( ( always_inline ) ) static __INLINE uint32_t __RBIT(uint32_t value)
|
||||
{
|
||||
uint32_t result;
|
||||
|
||||
__ASM volatile ("rbit %0, %1" : "=r" (result) : "r" (value) );
|
||||
return(result);
|
||||
}
|
||||
|
||||
|
||||
/** \brief LDR Exclusive (8 bit)
|
||||
|
||||
This function performs a exclusive LDR command for 8 bit value.
|
||||
|
||||
\param [in] ptr Pointer to data
|
||||
\return value of type uint8_t at (*ptr)
|
||||
*/
|
||||
__attribute__( ( always_inline ) ) static __INLINE uint8_t __LDREXB(volatile uint8_t *addr)
|
||||
{
|
||||
uint8_t result;
|
||||
|
||||
__ASM volatile ("ldrexb %0, [%1]" : "=r" (result) : "r" (addr) );
|
||||
return(result);
|
||||
}
|
||||
|
||||
|
||||
/** \brief LDR Exclusive (16 bit)
|
||||
|
||||
This function performs a exclusive LDR command for 16 bit values.
|
||||
|
||||
\param [in] ptr Pointer to data
|
||||
\return value of type uint16_t at (*ptr)
|
||||
*/
|
||||
__attribute__( ( always_inline ) ) static __INLINE uint16_t __LDREXH(volatile uint16_t *addr)
|
||||
{
|
||||
uint16_t result;
|
||||
|
||||
__ASM volatile ("ldrexh %0, [%1]" : "=r" (result) : "r" (addr) );
|
||||
return(result);
|
||||
}
|
||||
|
||||
|
||||
/** \brief LDR Exclusive (32 bit)
|
||||
|
||||
This function performs a exclusive LDR command for 32 bit values.
|
||||
|
||||
\param [in] ptr Pointer to data
|
||||
\return value of type uint32_t at (*ptr)
|
||||
*/
|
||||
__attribute__( ( always_inline ) ) static __INLINE uint32_t __LDREXW(volatile uint32_t *addr)
|
||||
{
|
||||
uint32_t result;
|
||||
|
||||
__ASM volatile ("ldrex %0, [%1]" : "=r" (result) : "r" (addr) );
|
||||
return(result);
|
||||
}
|
||||
|
||||
|
||||
/** \brief STR Exclusive (8 bit)
|
||||
|
||||
This function performs a exclusive STR command for 8 bit values.
|
||||
|
||||
\param [in] value Value to store
|
||||
\param [in] ptr Pointer to location
|
||||
\return 0 Function succeeded
|
||||
\return 1 Function failed
|
||||
*/
|
||||
__attribute__( ( always_inline ) ) static __INLINE uint32_t __STREXB(uint8_t value, volatile uint8_t *addr)
|
||||
{
|
||||
uint32_t result;
|
||||
|
||||
__ASM volatile ("strexb %0, %2, [%1]" : "=r" (result) : "r" (addr), "r" (value) );
|
||||
return(result);
|
||||
}
|
||||
|
||||
|
||||
/** \brief STR Exclusive (16 bit)
|
||||
|
||||
This function performs a exclusive STR command for 16 bit values.
|
||||
|
||||
\param [in] value Value to store
|
||||
\param [in] ptr Pointer to location
|
||||
\return 0 Function succeeded
|
||||
\return 1 Function failed
|
||||
*/
|
||||
__attribute__( ( always_inline ) ) static __INLINE uint32_t __STREXH(uint16_t value, volatile uint16_t *addr)
|
||||
{
|
||||
uint32_t result;
|
||||
|
||||
__ASM volatile ("strexh %0, %2, [%1]" : "=r" (result) : "r" (addr), "r" (value) );
|
||||
return(result);
|
||||
}
|
||||
|
||||
|
||||
/** \brief STR Exclusive (32 bit)
|
||||
|
||||
This function performs a exclusive STR command for 32 bit values.
|
||||
|
||||
\param [in] value Value to store
|
||||
\param [in] ptr Pointer to location
|
||||
\return 0 Function succeeded
|
||||
\return 1 Function failed
|
||||
*/
|
||||
__attribute__( ( always_inline ) ) static __INLINE uint32_t __STREXW(uint32_t value, volatile uint32_t *addr)
|
||||
{
|
||||
uint32_t result;
|
||||
|
||||
__ASM volatile ("strex %0, %2, [%1]" : "=r" (result) : "r" (addr), "r" (value) );
|
||||
return(result);
|
||||
}
|
||||
|
||||
|
||||
/** \brief Remove the exclusive lock
|
||||
|
||||
This function removes the exclusive lock which is created by LDREX.
|
||||
|
||||
*/
|
||||
__attribute__( ( always_inline ) ) static __INLINE void __CLREX(void)
|
||||
{
|
||||
__ASM volatile ("clrex");
|
||||
}
|
||||
|
||||
|
||||
/** \brief Signed Saturate
|
||||
|
||||
This function saturates a signed value.
|
||||
|
||||
\param [in] value Value to be saturated
|
||||
\param [in] sat Bit position to saturate to (1..32)
|
||||
\return Saturated value
|
||||
*/
|
||||
#define __SSAT(ARG1,ARG2) \
|
||||
({ \
|
||||
uint32_t __RES, __ARG1 = (ARG1); \
|
||||
__ASM ("ssat %0, %1, %2" : "=r" (__RES) : "I" (ARG2), "r" (__ARG1) ); \
|
||||
__RES; \
|
||||
})
|
||||
|
||||
|
||||
/** \brief Unsigned Saturate
|
||||
|
||||
This function saturates an unsigned value.
|
||||
|
||||
\param [in] value Value to be saturated
|
||||
\param [in] sat Bit position to saturate to (0..31)
|
||||
\return Saturated value
|
||||
*/
|
||||
#define __USAT(ARG1,ARG2) \
|
||||
({ \
|
||||
uint32_t __RES, __ARG1 = (ARG1); \
|
||||
__ASM ("usat %0, %1, %2" : "=r" (__RES) : "I" (ARG2), "r" (__ARG1) ); \
|
||||
__RES; \
|
||||
})
|
||||
|
||||
|
||||
/** \brief Count leading zeros
|
||||
|
||||
This function counts the number of leading zeros of a data value.
|
||||
|
||||
\param [in] value Value to count the leading zeros
|
||||
\return number of leading zeros in value
|
||||
*/
|
||||
__attribute__( ( always_inline ) ) static __INLINE uint8_t __CLZ(uint32_t value)
|
||||
{
|
||||
uint8_t result;
|
||||
|
||||
__ASM volatile ("clz %0, %1" : "=r" (result) : "r" (value) );
|
||||
return(result);
|
||||
}
|
||||
|
||||
#endif /* (__CORTEX_M >= 0x03) */
|
||||
|
||||
|
||||
|
||||
|
||||
#elif (defined (__TASKING__)) /*--------------- TASKING Compiler -----------------*/
|
||||
/* TASKING carm specific functions */
|
||||
|
||||
/*
|
||||
* The CMSIS functions have been implemented as intrinsics in the compiler.
|
||||
* Please use "carm -?i" to get an up to date list of all instrinsics,
|
||||
* Including the CMSIS ones.
|
||||
*/
|
||||
|
||||
#endif
|
||||
|
||||
/*@}*/ /* end of group CMSIS_Core_InstructionInterface */
|
||||
|
||||
#endif /* __CORE_CMINSTR_H__ */
|
||||
64
platforms/lpc17xx/CMSISv2p00_LPC17xx/inc/system_LPC17xx.h
Normal file
64
platforms/lpc17xx/CMSISv2p00_LPC17xx/inc/system_LPC17xx.h
Normal file
@@ -0,0 +1,64 @@
|
||||
/**************************************************************************//**
|
||||
* @file system_LPC17xx.h
|
||||
* @brief CMSIS Cortex-M3 Device Peripheral Access Layer Header File
|
||||
* for the NXP LPC17xx Device Series
|
||||
* @version V1.02
|
||||
* @date 08. September 2009
|
||||
*
|
||||
* @note
|
||||
* Copyright (C) 2009 ARM Limited. All rights reserved.
|
||||
*
|
||||
* @par
|
||||
* ARM Limited (ARM) is supplying this software for use with Cortex-M
|
||||
* processor based microcontrollers. This file can be freely distributed
|
||||
* within development tools that are supporting such ARM based processors.
|
||||
*
|
||||
* @par
|
||||
* THIS SOFTWARE IS PROVIDED "AS IS". NO WARRANTIES, WHETHER EXPRESS, IMPLIED
|
||||
* OR STATUTORY, INCLUDING, BUT NOT LIMITED TO, IMPLIED WARRANTIES OF
|
||||
* MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE APPLY TO THIS SOFTWARE.
|
||||
* ARM SHALL NOT, IN ANY CIRCUMSTANCES, BE LIABLE FOR SPECIAL, INCIDENTAL, OR
|
||||
* CONSEQUENTIAL DAMAGES, FOR ANY REASON WHATSOEVER.
|
||||
*
|
||||
******************************************************************************/
|
||||
|
||||
|
||||
#ifndef __SYSTEM_LPC17xx_H
|
||||
#define __SYSTEM_LPC17xx_H
|
||||
|
||||
#ifdef __cplusplus
|
||||
extern "C" {
|
||||
#endif
|
||||
|
||||
#include <stdint.h>
|
||||
|
||||
extern uint32_t SystemCoreClock; /*!< System Clock Frequency (Core Clock) */
|
||||
|
||||
|
||||
/**
|
||||
* Initialize the system
|
||||
*
|
||||
* @param none
|
||||
* @return none
|
||||
*
|
||||
* @brief Setup the microcontroller system.
|
||||
* Initialize the System and update the SystemCoreClock variable.
|
||||
*/
|
||||
extern void SystemInit (void);
|
||||
|
||||
/**
|
||||
* Update SystemCoreClock variable
|
||||
*
|
||||
* @param none
|
||||
* @return none
|
||||
*
|
||||
* @brief Updates the SystemCoreClock with current core Clock
|
||||
* retrieved from cpu registers.
|
||||
*/
|
||||
extern void SystemCoreClockUpdate (void);
|
||||
|
||||
#ifdef __cplusplus
|
||||
}
|
||||
#endif
|
||||
|
||||
#endif /* __SYSTEM_LPC17xx_H */
|
||||
339
platforms/lpc17xx/CMSISv2p00_LPC17xx/src/core_cm3.c
Normal file
339
platforms/lpc17xx/CMSISv2p00_LPC17xx/src/core_cm3.c
Normal file
@@ -0,0 +1,339 @@
|
||||
/**************************************************************************//**
|
||||
* @file core_cm3.c
|
||||
* @brief CMSIS Cortex-M3 Core Peripheral Access Layer Source File
|
||||
* @version V2.00
|
||||
* @date 13. September 2010
|
||||
*
|
||||
* @note
|
||||
* Copyright (C) 2009-2010 ARM Limited. All rights reserved.
|
||||
*
|
||||
* @par
|
||||
* ARM Limited (ARM) is supplying this software for use with Cortex-M
|
||||
* processor based microcontrollers. This file can be freely distributed
|
||||
* within development tools that are supporting such ARM based processors.
|
||||
*
|
||||
* @par
|
||||
* THIS SOFTWARE IS PROVIDED "AS IS". NO WARRANTIES, WHETHER EXPRESS, IMPLIED
|
||||
* OR STATUTORY, INCLUDING, BUT NOT LIMITED TO, IMPLIED WARRANTIES OF
|
||||
* MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE APPLY TO THIS SOFTWARE.
|
||||
* ARM SHALL NOT, IN ANY CIRCUMSTANCES, BE LIABLE FOR SPECIAL, INCIDENTAL, OR
|
||||
* CONSEQUENTIAL DAMAGES, FOR ANY REASON WHATSOEVER.
|
||||
*
|
||||
******************************************************************************/
|
||||
|
||||
#include <stdint.h>
|
||||
|
||||
/* define compiler specific symbols */
|
||||
#if defined ( __CC_ARM )
|
||||
#define __ASM __asm /*!< asm keyword for ARM Compiler */
|
||||
#define __INLINE __inline /*!< inline keyword for ARM Compiler */
|
||||
|
||||
#elif defined ( __ICCARM__ )
|
||||
#define __ASM __asm /*!< asm keyword for IAR Compiler */
|
||||
#define __INLINE inline /*!< inline keyword for IAR Compiler. Only avaiable in High optimization mode! */
|
||||
|
||||
#elif defined ( __GNUC__ )
|
||||
#define __ASM __asm /*!< asm keyword for GNU Compiler */
|
||||
#define __INLINE inline /*!< inline keyword for GNU Compiler */
|
||||
|
||||
#elif defined ( __TASKING__ )
|
||||
#define __ASM __asm /*!< asm keyword for TASKING Compiler */
|
||||
#define __INLINE inline /*!< inline keyword for TASKING Compiler */
|
||||
|
||||
#endif
|
||||
|
||||
|
||||
/* ########################## Core Instruction Access ######################### */
|
||||
|
||||
#if defined ( __CC_ARM ) /*------------------ RealView Compiler ----------------*/
|
||||
|
||||
/** \brief Reverse byte order (16 bit)
|
||||
|
||||
This function reverses the byte order in two unsigned short values.
|
||||
|
||||
\param [in] value Value to reverse
|
||||
\return Reversed value
|
||||
*/
|
||||
#if (__ARMCC_VERSION < 400677)
|
||||
__ASM uint32_t __REV16(uint32_t value)
|
||||
{
|
||||
rev16 r0, r0
|
||||
bx lr
|
||||
}
|
||||
#endif /* __ARMCC_VERSION */
|
||||
|
||||
|
||||
/** \brief Reverse byte order in signed short value
|
||||
|
||||
This function reverses the byte order in a signed short value with sign extension to integer.
|
||||
|
||||
\param [in] value Value to reverse
|
||||
\return Reversed value
|
||||
*/
|
||||
#if (__ARMCC_VERSION < 400677)
|
||||
__ASM int32_t __REVSH(int32_t value)
|
||||
{
|
||||
revsh r0, r0
|
||||
bx lr
|
||||
}
|
||||
#endif /* __ARMCC_VERSION */
|
||||
|
||||
|
||||
/** \brief Remove the exclusive lock
|
||||
|
||||
This function removes the exclusive lock which is created by LDREX.
|
||||
|
||||
*/
|
||||
#if (__ARMCC_VERSION < 400000)
|
||||
__ASM void __CLREX(void)
|
||||
{
|
||||
clrex
|
||||
}
|
||||
#endif /* __ARMCC_VERSION */
|
||||
|
||||
|
||||
#elif (defined (__ICCARM__)) /*---------------- ICC Compiler ---------------------*/
|
||||
/* obsolete */
|
||||
#elif (defined (__GNUC__)) /*------------------ GNU Compiler ---------------------*/
|
||||
/* obsolete */
|
||||
#elif (defined (__TASKING__)) /*--------------- TASKING Compiler -----------------*/
|
||||
/* obsolete */
|
||||
#endif
|
||||
|
||||
|
||||
/* ########################### Core Function Access ########################### */
|
||||
|
||||
#if defined ( __CC_ARM ) /*------------------ RealView Compiler ----------------*/
|
||||
|
||||
/** \brief Get Control Register
|
||||
|
||||
This function returns the content of the Control Register.
|
||||
|
||||
\return Control Register value
|
||||
*/
|
||||
#if (__ARMCC_VERSION < 400000)
|
||||
__ASM uint32_t __get_CONTROL(void)
|
||||
{
|
||||
mrs r0, control
|
||||
bx lr
|
||||
}
|
||||
#endif /* __ARMCC_VERSION */
|
||||
|
||||
|
||||
/** \brief Set Control Register
|
||||
|
||||
This function writes the given value to the Control Register.
|
||||
|
||||
\param [in] control Control Register value to set
|
||||
*/
|
||||
#if (__ARMCC_VERSION < 400000)
|
||||
__ASM void __set_CONTROL(uint32_t control)
|
||||
{
|
||||
msr control, r0
|
||||
bx lr
|
||||
}
|
||||
#endif /* __ARMCC_VERSION */
|
||||
|
||||
|
||||
/** \brief Get ISPR Register
|
||||
|
||||
This function returns the content of the ISPR Register.
|
||||
|
||||
\return ISPR Register value
|
||||
*/
|
||||
#if (__ARMCC_VERSION < 400000)
|
||||
__ASM uint32_t __get_IPSR(void)
|
||||
{
|
||||
mrs r0, ipsr
|
||||
bx lr
|
||||
}
|
||||
#endif /* __ARMCC_VERSION */
|
||||
|
||||
|
||||
/** \brief Get APSR Register
|
||||
|
||||
This function returns the content of the APSR Register.
|
||||
|
||||
\return APSR Register value
|
||||
*/
|
||||
#if (__ARMCC_VERSION < 400000)
|
||||
__ASM uint32_t __get_APSR(void)
|
||||
{
|
||||
mrs r0, apsr
|
||||
bx lr
|
||||
}
|
||||
#endif /* __ARMCC_VERSION */
|
||||
|
||||
|
||||
/** \brief Get xPSR Register
|
||||
|
||||
This function returns the content of the xPSR Register.
|
||||
|
||||
\return xPSR Register value
|
||||
*/
|
||||
#if (__ARMCC_VERSION < 400000)
|
||||
__ASM uint32_t __get_xPSR(void)
|
||||
{
|
||||
mrs r0, xpsr
|
||||
bx lr
|
||||
}
|
||||
#endif /* __ARMCC_VERSION */
|
||||
|
||||
|
||||
/** \brief Get Process Stack Pointer
|
||||
|
||||
This function returns the current value of the Process Stack Pointer (PSP).
|
||||
|
||||
\return PSP Register value
|
||||
*/
|
||||
#if (__ARMCC_VERSION < 400000)
|
||||
__ASM uint32_t __get_PSP(void)
|
||||
{
|
||||
mrs r0, psp
|
||||
bx lr
|
||||
}
|
||||
#endif /* __ARMCC_VERSION */
|
||||
|
||||
|
||||
/** \brief Set Process Stack Pointer
|
||||
|
||||
This function assigns the given value to the Process Stack Pointer (PSP).
|
||||
|
||||
\param [in] topOfProcStack Process Stack Pointer value to set
|
||||
*/
|
||||
#if (__ARMCC_VERSION < 400000)
|
||||
__ASM void __set_PSP(uint32_t topOfProcStack)
|
||||
{
|
||||
msr psp, r0
|
||||
bx lr
|
||||
}
|
||||
#endif /* __ARMCC_VERSION */
|
||||
|
||||
|
||||
/** \brief Get Main Stack Pointer
|
||||
|
||||
This function returns the current value of the Main Stack Pointer (MSP).
|
||||
|
||||
\return MSP Register value
|
||||
*/
|
||||
#if (__ARMCC_VERSION < 400000)
|
||||
__ASM uint32_t __get_MSP(void)
|
||||
{
|
||||
mrs r0, msp
|
||||
bx lr
|
||||
}
|
||||
#endif /* __ARMCC_VERSION */
|
||||
|
||||
|
||||
/** \brief Set Main Stack Pointer
|
||||
|
||||
This function assigns the given value to the Main Stack Pointer (MSP).
|
||||
|
||||
\param [in] topOfMainStack Main Stack Pointer value to set
|
||||
*/
|
||||
#if (__ARMCC_VERSION < 400000)
|
||||
__ASM void __set_MSP(uint32_t mainStackPointer)
|
||||
{
|
||||
msr msp, r0
|
||||
bx lr
|
||||
}
|
||||
#endif /* __ARMCC_VERSION */
|
||||
|
||||
|
||||
/** \brief Get Base Priority
|
||||
|
||||
This function returns the current value of the Base Priority register.
|
||||
|
||||
\return Base Priority register value
|
||||
*/
|
||||
#if (__ARMCC_VERSION < 400000)
|
||||
__ASM uint32_t __get_BASEPRI(void)
|
||||
{
|
||||
mrs r0, basepri
|
||||
bx lr
|
||||
}
|
||||
#endif /* __ARMCC_VERSION */
|
||||
|
||||
|
||||
/** \brief Set Base Priority
|
||||
|
||||
This function assigns the given value to the Base Priority register.
|
||||
|
||||
\param [in] basePri Base Priority value to set
|
||||
*/
|
||||
#if (__ARMCC_VERSION < 400000)
|
||||
__ASM void __set_BASEPRI(uint32_t basePri)
|
||||
{
|
||||
msr basepri, r0
|
||||
bx lr
|
||||
}
|
||||
#endif /* __ARMCC_VERSION */
|
||||
|
||||
/** \brief Get Priority Mask
|
||||
|
||||
This function returns the current state of the priority mask bit from the Priority Mask Register.
|
||||
|
||||
\return Priority Mask value
|
||||
*/
|
||||
#if (__ARMCC_VERSION < 400000)
|
||||
__ASM uint32_t __get_PRIMASK(void)
|
||||
{
|
||||
mrs r0, primask
|
||||
bx lr
|
||||
}
|
||||
#endif /* __ARMCC_VERSION */
|
||||
|
||||
|
||||
/** \brief Set Priority Mask
|
||||
|
||||
This function assigns the given value to the Priority Mask Register.
|
||||
|
||||
\param [in] priMask Priority Mask
|
||||
*/
|
||||
#if (__ARMCC_VERSION < 400000)
|
||||
__ASM void __set_PRIMASK(uint32_t priMask)
|
||||
{
|
||||
msr primask, r0
|
||||
bx lr
|
||||
}
|
||||
#endif /* __ARMCC_VERSION */
|
||||
|
||||
|
||||
/** \brief Get Fault Mask
|
||||
|
||||
This function returns the current value of the Fault Mask Register.
|
||||
|
||||
\return Fault Mask value
|
||||
*/
|
||||
#if (__ARMCC_VERSION < 400000)
|
||||
__ASM uint32_t __get_FAULTMASK(void)
|
||||
{
|
||||
mrs r0, faultmask
|
||||
bx lr
|
||||
}
|
||||
#endif /* __ARMCC_VERSION */
|
||||
|
||||
|
||||
/** \brief Set the Fault Mask
|
||||
|
||||
This function assigns the given value to the Fault Mask Register.
|
||||
|
||||
\param [in] faultMask Fault Mask value value to set
|
||||
*/
|
||||
#if (__ARMCC_VERSION < 400000)
|
||||
__ASM void __set_FAULTMASK(uint32_t faultMask)
|
||||
{
|
||||
msr faultmask, r0
|
||||
bx lr
|
||||
}
|
||||
#endif /* __ARMCC_VERSION */
|
||||
|
||||
|
||||
|
||||
#elif (defined (__ICCARM__)) /*---------------- ICC Compiler ---------------------*/
|
||||
/* obsolete */
|
||||
#elif (defined (__GNUC__)) /*------------------ GNU Compiler ---------------------*/
|
||||
/* obsolete */
|
||||
#elif (defined (__TASKING__)) /*--------------- TASKING Compiler -----------------*/
|
||||
/* obsolete */
|
||||
#endif
|
||||
532
platforms/lpc17xx/CMSISv2p00_LPC17xx/src/system_LPC17xx.c
Normal file
532
platforms/lpc17xx/CMSISv2p00_LPC17xx/src/system_LPC17xx.c
Normal file
@@ -0,0 +1,532 @@
|
||||
/**************************************************************************//**
|
||||
* @file system_LPC17xx.c
|
||||
* @brief CMSIS Cortex-M3 Device Peripheral Access Layer Source File
|
||||
* for the NXP LPC17xx Device Series
|
||||
* @version V1.08
|
||||
* @date 12. May 2010
|
||||
*
|
||||
* @note
|
||||
* Copyright (C) 2009 ARM Limited. All rights reserved.
|
||||
*
|
||||
* @par
|
||||
* ARM Limited (ARM) is supplying this software for use with Cortex-M
|
||||
* processor based microcontrollers. This file can be freely distributed
|
||||
* within development tools that are supporting such ARM based processors.
|
||||
*
|
||||
* @par
|
||||
* THIS SOFTWARE IS PROVIDED "AS IS". NO WARRANTIES, WHETHER EXPRESS, IMPLIED
|
||||
* OR STATUTORY, INCLUDING, BUT NOT LIMITED TO, IMPLIED WARRANTIES OF
|
||||
* MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE APPLY TO THIS SOFTWARE.
|
||||
* ARM SHALL NOT, IN ANY CIRCUMSTANCES, BE LIABLE FOR SPECIAL, INCIDENTAL, OR
|
||||
* CONSEQUENTIAL DAMAGES, FOR ANY REASON WHATSOEVER.
|
||||
*
|
||||
******************************************************************************/
|
||||
|
||||
|
||||
#include <stdint.h>
|
||||
#include "LPC17xx.h"
|
||||
|
||||
/*
|
||||
//-------- <<< Use Configuration Wizard in Context Menu >>> ------------------
|
||||
*/
|
||||
|
||||
/*--------------------- Clock Configuration ----------------------------------
|
||||
//
|
||||
// <e> Clock Configuration
|
||||
// <h> System Controls and Status Register (SCS)
|
||||
// <o1.4> OSCRANGE: Main Oscillator Range Select
|
||||
// <0=> 1 MHz to 20 MHz
|
||||
// <1=> 15 MHz to 24 MHz
|
||||
// <e1.5> OSCEN: Main Oscillator Enable
|
||||
// </e>
|
||||
// </h>
|
||||
//
|
||||
// <h> Clock Source Select Register (CLKSRCSEL)
|
||||
// <o2.0..1> CLKSRC: PLL Clock Source Selection
|
||||
// <0=> Internal RC oscillator
|
||||
// <1=> Main oscillator
|
||||
// <2=> RTC oscillator
|
||||
// </h>
|
||||
//
|
||||
// <e3> PLL0 Configuration (Main PLL)
|
||||
// <h> PLL0 Configuration Register (PLL0CFG)
|
||||
// <i> F_cco0 = (2 * M * F_in) / N
|
||||
// <i> F_in must be in the range of 32 kHz to 50 MHz
|
||||
// <i> F_cco0 must be in the range of 275 MHz to 550 MHz
|
||||
// <o4.0..14> MSEL: PLL Multiplier Selection
|
||||
// <6-32768><#-1>
|
||||
// <i> M Value
|
||||
// <o4.16..23> NSEL: PLL Divider Selection
|
||||
// <1-256><#-1>
|
||||
// <i> N Value
|
||||
// </h>
|
||||
// </e>
|
||||
//
|
||||
// <e5> PLL1 Configuration (USB PLL)
|
||||
// <h> PLL1 Configuration Register (PLL1CFG)
|
||||
// <i> F_usb = M * F_osc or F_usb = F_cco1 / (2 * P)
|
||||
// <i> F_cco1 = F_osc * M * 2 * P
|
||||
// <i> F_cco1 must be in the range of 156 MHz to 320 MHz
|
||||
// <o6.0..4> MSEL: PLL Multiplier Selection
|
||||
// <1-32><#-1>
|
||||
// <i> M Value (for USB maximum value is 4)
|
||||
// <o6.5..6> PSEL: PLL Divider Selection
|
||||
// <0=> 1
|
||||
// <1=> 2
|
||||
// <2=> 4
|
||||
// <3=> 8
|
||||
// <i> P Value
|
||||
// </h>
|
||||
// </e>
|
||||
//
|
||||
// <h> CPU Clock Configuration Register (CCLKCFG)
|
||||
// <o7.0..7> CCLKSEL: Divide Value for CPU Clock from PLL0
|
||||
// <1-256><#-1>
|
||||
// </h>
|
||||
//
|
||||
// <h> USB Clock Configuration Register (USBCLKCFG)
|
||||
// <o8.0..3> USBSEL: Divide Value for USB Clock from PLL0
|
||||
// <0-15>
|
||||
// <i> Divide is USBSEL + 1
|
||||
// </h>
|
||||
//
|
||||
// <h> Peripheral Clock Selection Register 0 (PCLKSEL0)
|
||||
// <o9.0..1> PCLK_WDT: Peripheral Clock Selection for WDT
|
||||
// <0=> Pclk = Cclk / 4
|
||||
// <1=> Pclk = Cclk
|
||||
// <2=> Pclk = Cclk / 2
|
||||
// <3=> Pclk = Hclk / 8
|
||||
// <o9.2..3> PCLK_TIMER0: Peripheral Clock Selection for TIMER0
|
||||
// <0=> Pclk = Cclk / 4
|
||||
// <1=> Pclk = Cclk
|
||||
// <2=> Pclk = Cclk / 2
|
||||
// <3=> Pclk = Hclk / 8
|
||||
// <o9.4..5> PCLK_TIMER1: Peripheral Clock Selection for TIMER1
|
||||
// <0=> Pclk = Cclk / 4
|
||||
// <1=> Pclk = Cclk
|
||||
// <2=> Pclk = Cclk / 2
|
||||
// <3=> Pclk = Hclk / 8
|
||||
// <o9.6..7> PCLK_UART0: Peripheral Clock Selection for UART0
|
||||
// <0=> Pclk = Cclk / 4
|
||||
// <1=> Pclk = Cclk
|
||||
// <2=> Pclk = Cclk / 2
|
||||
// <3=> Pclk = Hclk / 8
|
||||
// <o9.8..9> PCLK_UART1: Peripheral Clock Selection for UART1
|
||||
// <0=> Pclk = Cclk / 4
|
||||
// <1=> Pclk = Cclk
|
||||
// <2=> Pclk = Cclk / 2
|
||||
// <3=> Pclk = Hclk / 8
|
||||
// <o9.12..13> PCLK_PWM1: Peripheral Clock Selection for PWM1
|
||||
// <0=> Pclk = Cclk / 4
|
||||
// <1=> Pclk = Cclk
|
||||
// <2=> Pclk = Cclk / 2
|
||||
// <3=> Pclk = Hclk / 8
|
||||
// <o9.14..15> PCLK_I2C0: Peripheral Clock Selection for I2C0
|
||||
// <0=> Pclk = Cclk / 4
|
||||
// <1=> Pclk = Cclk
|
||||
// <2=> Pclk = Cclk / 2
|
||||
// <3=> Pclk = Hclk / 8
|
||||
// <o9.16..17> PCLK_SPI: Peripheral Clock Selection for SPI
|
||||
// <0=> Pclk = Cclk / 4
|
||||
// <1=> Pclk = Cclk
|
||||
// <2=> Pclk = Cclk / 2
|
||||
// <3=> Pclk = Hclk / 8
|
||||
// <o9.20..21> PCLK_SSP1: Peripheral Clock Selection for SSP1
|
||||
// <0=> Pclk = Cclk / 4
|
||||
// <1=> Pclk = Cclk
|
||||
// <2=> Pclk = Cclk / 2
|
||||
// <3=> Pclk = Hclk / 8
|
||||
// <o9.22..23> PCLK_DAC: Peripheral Clock Selection for DAC
|
||||
// <0=> Pclk = Cclk / 4
|
||||
// <1=> Pclk = Cclk
|
||||
// <2=> Pclk = Cclk / 2
|
||||
// <3=> Pclk = Hclk / 8
|
||||
// <o9.24..25> PCLK_ADC: Peripheral Clock Selection for ADC
|
||||
// <0=> Pclk = Cclk / 4
|
||||
// <1=> Pclk = Cclk
|
||||
// <2=> Pclk = Cclk / 2
|
||||
// <3=> Pclk = Hclk / 8
|
||||
// <o9.26..27> PCLK_CAN1: Peripheral Clock Selection for CAN1
|
||||
// <0=> Pclk = Cclk / 4
|
||||
// <1=> Pclk = Cclk
|
||||
// <2=> Pclk = Cclk / 2
|
||||
// <3=> Pclk = Hclk / 6
|
||||
// <o9.28..29> PCLK_CAN2: Peripheral Clock Selection for CAN2
|
||||
// <0=> Pclk = Cclk / 4
|
||||
// <1=> Pclk = Cclk
|
||||
// <2=> Pclk = Cclk / 2
|
||||
// <3=> Pclk = Hclk / 6
|
||||
// <o9.30..31> PCLK_ACF: Peripheral Clock Selection for ACF
|
||||
// <0=> Pclk = Cclk / 4
|
||||
// <1=> Pclk = Cclk
|
||||
// <2=> Pclk = Cclk / 2
|
||||
// <3=> Pclk = Hclk / 6
|
||||
// </h>
|
||||
//
|
||||
// <h> Peripheral Clock Selection Register 1 (PCLKSEL1)
|
||||
// <o10.0..1> PCLK_QEI: Peripheral Clock Selection for the Quadrature Encoder Interface
|
||||
// <0=> Pclk = Cclk / 4
|
||||
// <1=> Pclk = Cclk
|
||||
// <2=> Pclk = Cclk / 2
|
||||
// <3=> Pclk = Hclk / 8
|
||||
// <o10.2..3> PCLK_GPIO: Peripheral Clock Selection for GPIOs
|
||||
// <0=> Pclk = Cclk / 4
|
||||
// <1=> Pclk = Cclk
|
||||
// <2=> Pclk = Cclk / 2
|
||||
// <3=> Pclk = Hclk / 8
|
||||
// <o10.4..5> PCLK_PCB: Peripheral Clock Selection for the Pin Connect Block
|
||||
// <0=> Pclk = Cclk / 4
|
||||
// <1=> Pclk = Cclk
|
||||
// <2=> Pclk = Cclk / 2
|
||||
// <3=> Pclk = Hclk / 8
|
||||
// <o10.6..7> PCLK_I2C1: Peripheral Clock Selection for I2C1
|
||||
// <0=> Pclk = Cclk / 4
|
||||
// <1=> Pclk = Cclk
|
||||
// <2=> Pclk = Cclk / 2
|
||||
// <3=> Pclk = Hclk / 8
|
||||
// <o10.10..11> PCLK_SSP0: Peripheral Clock Selection for SSP0
|
||||
// <0=> Pclk = Cclk / 4
|
||||
// <1=> Pclk = Cclk
|
||||
// <2=> Pclk = Cclk / 2
|
||||
// <3=> Pclk = Hclk / 8
|
||||
// <o10.12..13> PCLK_TIMER2: Peripheral Clock Selection for TIMER2
|
||||
// <0=> Pclk = Cclk / 4
|
||||
// <1=> Pclk = Cclk
|
||||
// <2=> Pclk = Cclk / 2
|
||||
// <3=> Pclk = Hclk / 8
|
||||
// <o10.14..15> PCLK_TIMER3: Peripheral Clock Selection for TIMER3
|
||||
// <0=> Pclk = Cclk / 4
|
||||
// <1=> Pclk = Cclk
|
||||
// <2=> Pclk = Cclk / 2
|
||||
// <3=> Pclk = Hclk / 8
|
||||
// <o10.16..17> PCLK_UART2: Peripheral Clock Selection for UART2
|
||||
// <0=> Pclk = Cclk / 4
|
||||
// <1=> Pclk = Cclk
|
||||
// <2=> Pclk = Cclk / 2
|
||||
// <3=> Pclk = Hclk / 8
|
||||
// <o10.18..19> PCLK_UART3: Peripheral Clock Selection for UART3
|
||||
// <0=> Pclk = Cclk / 4
|
||||
// <1=> Pclk = Cclk
|
||||
// <2=> Pclk = Cclk / 2
|
||||
// <3=> Pclk = Hclk / 8
|
||||
// <o10.20..21> PCLK_I2C2: Peripheral Clock Selection for I2C2
|
||||
// <0=> Pclk = Cclk / 4
|
||||
// <1=> Pclk = Cclk
|
||||
// <2=> Pclk = Cclk / 2
|
||||
// <3=> Pclk = Hclk / 8
|
||||
// <o10.22..23> PCLK_I2S: Peripheral Clock Selection for I2S
|
||||
// <0=> Pclk = Cclk / 4
|
||||
// <1=> Pclk = Cclk
|
||||
// <2=> Pclk = Cclk / 2
|
||||
// <3=> Pclk = Hclk / 8
|
||||
// <o10.26..27> PCLK_RIT: Peripheral Clock Selection for the Repetitive Interrupt Timer
|
||||
// <0=> Pclk = Cclk / 4
|
||||
// <1=> Pclk = Cclk
|
||||
// <2=> Pclk = Cclk / 2
|
||||
// <3=> Pclk = Hclk / 8
|
||||
// <o10.28..29> PCLK_SYSCON: Peripheral Clock Selection for the System Control Block
|
||||
// <0=> Pclk = Cclk / 4
|
||||
// <1=> Pclk = Cclk
|
||||
// <2=> Pclk = Cclk / 2
|
||||
// <3=> Pclk = Hclk / 8
|
||||
// <o10.30..31> PCLK_MC: Peripheral Clock Selection for the Motor Control PWM
|
||||
// <0=> Pclk = Cclk / 4
|
||||
// <1=> Pclk = Cclk
|
||||
// <2=> Pclk = Cclk / 2
|
||||
// <3=> Pclk = Hclk / 8
|
||||
// </h>
|
||||
//
|
||||
// <h> Power Control for Peripherals Register (PCONP)
|
||||
// <o11.1> PCTIM0: Timer/Counter 0 power/clock enable
|
||||
// <o11.2> PCTIM1: Timer/Counter 1 power/clock enable
|
||||
// <o11.3> PCUART0: UART 0 power/clock enable
|
||||
// <o11.4> PCUART1: UART 1 power/clock enable
|
||||
// <o11.6> PCPWM1: PWM 1 power/clock enable
|
||||
// <o11.7> PCI2C0: I2C interface 0 power/clock enable
|
||||
// <o11.8> PCSPI: SPI interface power/clock enable
|
||||
// <o11.9> PCRTC: RTC power/clock enable
|
||||
// <o11.10> PCSSP1: SSP interface 1 power/clock enable
|
||||
// <o11.12> PCAD: A/D converter power/clock enable
|
||||
// <o11.13> PCCAN1: CAN controller 1 power/clock enable
|
||||
// <o11.14> PCCAN2: CAN controller 2 power/clock enable
|
||||
// <o11.15> PCGPIO: GPIOs power/clock enable
|
||||
// <o11.16> PCRIT: Repetitive interrupt timer power/clock enable
|
||||
// <o11.17> PCMC: Motor control PWM power/clock enable
|
||||
// <o11.18> PCQEI: Quadrature encoder interface power/clock enable
|
||||
// <o11.19> PCI2C1: I2C interface 1 power/clock enable
|
||||
// <o11.21> PCSSP0: SSP interface 0 power/clock enable
|
||||
// <o11.22> PCTIM2: Timer 2 power/clock enable
|
||||
// <o11.23> PCTIM3: Timer 3 power/clock enable
|
||||
// <o11.24> PCUART2: UART 2 power/clock enable
|
||||
// <o11.25> PCUART3: UART 3 power/clock enable
|
||||
// <o11.26> PCI2C2: I2C interface 2 power/clock enable
|
||||
// <o11.27> PCI2S: I2S interface power/clock enable
|
||||
// <o11.29> PCGPDMA: GP DMA function power/clock enable
|
||||
// <o11.30> PCENET: Ethernet block power/clock enable
|
||||
// <o11.31> PCUSB: USB interface power/clock enable
|
||||
// </h>
|
||||
//
|
||||
// <h> Clock Output Configuration Register (CLKOUTCFG)
|
||||
// <o12.0..3> CLKOUTSEL: Selects clock source for CLKOUT
|
||||
// <0=> CPU clock
|
||||
// <1=> Main oscillator
|
||||
// <2=> Internal RC oscillator
|
||||
// <3=> USB clock
|
||||
// <4=> RTC oscillator
|
||||
// <o12.4..7> CLKOUTDIV: Selects clock divider for CLKOUT
|
||||
// <1-16><#-1>
|
||||
// <o12.8> CLKOUT_EN: CLKOUT enable control
|
||||
// </h>
|
||||
//
|
||||
// </e>
|
||||
*/
|
||||
#define CLOCK_SETUP 1
|
||||
#define SCS_Val 0x00000020
|
||||
#define CLKSRCSEL_Val 0x00000001
|
||||
#define PLL0_SETUP 1
|
||||
#define PLL0CFG_Val 0x00050063
|
||||
#define PLL1_SETUP 1
|
||||
#define PLL1CFG_Val 0x00000023
|
||||
#define CCLKCFG_Val 0x00000003
|
||||
#define USBCLKCFG_Val 0x00000000
|
||||
#define PCLKSEL0_Val 0x00000000
|
||||
#define PCLKSEL1_Val 0x00000000
|
||||
#define PCONP_Val 0x042887DE
|
||||
#define CLKOUTCFG_Val 0x00000000
|
||||
|
||||
|
||||
/*--------------------- Flash Accelerator Configuration ----------------------
|
||||
//
|
||||
// <e> Flash Accelerator Configuration
|
||||
// <o1.12..15> FLASHTIM: Flash Access Time
|
||||
// <0=> 1 CPU clock (for CPU clock up to 20 MHz)
|
||||
// <1=> 2 CPU clocks (for CPU clock up to 40 MHz)
|
||||
// <2=> 3 CPU clocks (for CPU clock up to 60 MHz)
|
||||
// <3=> 4 CPU clocks (for CPU clock up to 80 MHz)
|
||||
// <4=> 5 CPU clocks (for CPU clock up to 100 MHz)
|
||||
// <5=> 6 CPU clocks (for any CPU clock)
|
||||
// </e>
|
||||
*/
|
||||
#define FLASH_SETUP 0
|
||||
#define FLASHCFG_Val 0x00004000
|
||||
|
||||
/*
|
||||
//-------- <<< end of configuration section >>> ------------------------------
|
||||
*/
|
||||
|
||||
/*----------------------------------------------------------------------------
|
||||
Check the register settings
|
||||
*----------------------------------------------------------------------------*/
|
||||
#define CHECK_RANGE(val, min, max) ((val < min) || (val > max))
|
||||
#define CHECK_RSVD(val, mask) (val & mask)
|
||||
|
||||
/* Clock Configuration -------------------------------------------------------*/
|
||||
#if (CHECK_RSVD((SCS_Val), ~0x00000030))
|
||||
#error "SCS: Invalid values of reserved bits!"
|
||||
#endif
|
||||
|
||||
#if (CHECK_RANGE((CLKSRCSEL_Val), 0, 2))
|
||||
#error "CLKSRCSEL: Value out of range!"
|
||||
#endif
|
||||
|
||||
#if (CHECK_RSVD((PLL0CFG_Val), ~0x00FF7FFF))
|
||||
#error "PLL0CFG: Invalid values of reserved bits!"
|
||||
#endif
|
||||
|
||||
#if (CHECK_RSVD((PLL1CFG_Val), ~0x0000007F))
|
||||
#error "PLL1CFG: Invalid values of reserved bits!"
|
||||
#endif
|
||||
|
||||
#if (PLL0_SETUP) /* if PLL0 is used */
|
||||
#if (CCLKCFG_Val < 2) /* CCLKSEL must be greater then 1 */
|
||||
#error "CCLKCFG: CCLKSEL must be greater then 1 if PLL0 is used!"
|
||||
#endif
|
||||
#endif
|
||||
|
||||
#if (CHECK_RANGE((CCLKCFG_Val), 2, 255))
|
||||
#error "CCLKCFG: Value out of range!"
|
||||
#endif
|
||||
|
||||
#if (CHECK_RSVD((USBCLKCFG_Val), ~0x0000000F))
|
||||
#error "USBCLKCFG: Invalid values of reserved bits!"
|
||||
#endif
|
||||
|
||||
#if (CHECK_RSVD((PCLKSEL0_Val), 0x000C0C00))
|
||||
#error "PCLKSEL0: Invalid values of reserved bits!"
|
||||
#endif
|
||||
|
||||
#if (CHECK_RSVD((PCLKSEL1_Val), 0x03000300))
|
||||
#error "PCLKSEL1: Invalid values of reserved bits!"
|
||||
#endif
|
||||
|
||||
#if (CHECK_RSVD((PCONP_Val), 0x10100821))
|
||||
#error "PCONP: Invalid values of reserved bits!"
|
||||
#endif
|
||||
|
||||
#if (CHECK_RSVD((CLKOUTCFG_Val), ~0x000001FF))
|
||||
#error "CLKOUTCFG: Invalid values of reserved bits!"
|
||||
#endif
|
||||
|
||||
/* Flash Accelerator Configuration -------------------------------------------*/
|
||||
#if (CHECK_RSVD((FLASHCFG_Val), ~0x0000F000))
|
||||
#error "FLASHCFG: Invalid values of reserved bits!"
|
||||
#endif
|
||||
|
||||
|
||||
/*----------------------------------------------------------------------------
|
||||
DEFINES
|
||||
*----------------------------------------------------------------------------*/
|
||||
|
||||
/*----------------------------------------------------------------------------
|
||||
Define clocks
|
||||
*----------------------------------------------------------------------------*/
|
||||
#define XTAL (12000000UL) /* Oscillator frequency */
|
||||
#define OSC_CLK ( XTAL) /* Main oscillator frequency */
|
||||
#define RTC_CLK ( 32000UL) /* RTC oscillator frequency */
|
||||
#define IRC_OSC ( 4000000UL) /* Internal RC oscillator frequency */
|
||||
|
||||
|
||||
/* F_cco0 = (2 * M * F_in) / N */
|
||||
#define __M (((PLL0CFG_Val ) & 0x7FFF) + 1)
|
||||
#define __N (((PLL0CFG_Val >> 16) & 0x00FF) + 1)
|
||||
#define __FCCO(__F_IN) ((2/*ULL*/ * __M * __F_IN) / __N)
|
||||
#define __CCLK_DIV (((CCLKCFG_Val ) & 0x00FF) + 1)
|
||||
|
||||
/* Determine core clock frequency according to settings */
|
||||
#if (PLL0_SETUP)
|
||||
#if ((CLKSRCSEL_Val & 0x03) == 1)
|
||||
#define __CORE_CLK (__FCCO(OSC_CLK) / __CCLK_DIV)
|
||||
#elif ((CLKSRCSEL_Val & 0x03) == 2)
|
||||
#define __CORE_CLK (__FCCO(RTC_CLK) / __CCLK_DIV)
|
||||
#else
|
||||
#define __CORE_CLK (__FCCO(IRC_OSC) / __CCLK_DIV)
|
||||
#endif
|
||||
#else
|
||||
#if ((CLKSRCSEL_Val & 0x03) == 1)
|
||||
#define __CORE_CLK (OSC_CLK / __CCLK_DIV)
|
||||
#elif ((CLKSRCSEL_Val & 0x03) == 2)
|
||||
#define __CORE_CLK (RTC_CLK / __CCLK_DIV)
|
||||
#else
|
||||
#define __CORE_CLK (IRC_OSC / __CCLK_DIV)
|
||||
#endif
|
||||
#endif
|
||||
|
||||
|
||||
/*----------------------------------------------------------------------------
|
||||
Clock Variable definitions
|
||||
*----------------------------------------------------------------------------*/
|
||||
uint32_t SystemCoreClock = __CORE_CLK;/*!< System Clock Frequency (Core Clock)*/
|
||||
|
||||
|
||||
/*----------------------------------------------------------------------------
|
||||
Clock functions
|
||||
*----------------------------------------------------------------------------*/
|
||||
void SystemCoreClockUpdate (void) /* Get Core Clock Frequency */
|
||||
{
|
||||
/* Determine clock frequency according to clock register values */
|
||||
if (((LPC_SC->PLL0STAT >> 24) & 3) == 3) { /* If PLL0 enabled and connected */
|
||||
switch (LPC_SC->CLKSRCSEL & 0x03) {
|
||||
case 0: /* Int. RC oscillator => PLL0 */
|
||||
case 3: /* Reserved, default to Int. RC */
|
||||
SystemCoreClock = (IRC_OSC *
|
||||
((2/*ULL*/ * ((LPC_SC->PLL0STAT & 0x7FFF) + 1))) /
|
||||
(((LPC_SC->PLL0STAT >> 16) & 0xFF) + 1) /
|
||||
((LPC_SC->CCLKCFG & 0xFF)+ 1));
|
||||
break;
|
||||
case 1: /* Main oscillator => PLL0 */
|
||||
SystemCoreClock = (OSC_CLK *
|
||||
((2/*ULL*/ * ((LPC_SC->PLL0STAT & 0x7FFF) + 1))) /
|
||||
(((LPC_SC->PLL0STAT >> 16) & 0xFF) + 1) /
|
||||
((LPC_SC->CCLKCFG & 0xFF)+ 1));
|
||||
break;
|
||||
case 2: /* RTC oscillator => PLL0 */
|
||||
SystemCoreClock = (RTC_CLK *
|
||||
((2/*ULL*/ * ((LPC_SC->PLL0STAT & 0x7FFF) + 1))) /
|
||||
(((LPC_SC->PLL0STAT >> 16) & 0xFF) + 1) /
|
||||
((LPC_SC->CCLKCFG & 0xFF)+ 1));
|
||||
break;
|
||||
}
|
||||
} else {
|
||||
switch (LPC_SC->CLKSRCSEL & 0x03) {
|
||||
case 0: /* Int. RC oscillator => PLL0 */
|
||||
case 3: /* Reserved, default to Int. RC */
|
||||
SystemCoreClock = IRC_OSC / ((LPC_SC->CCLKCFG & 0xFF)+ 1);
|
||||
break;
|
||||
case 1: /* Main oscillator => PLL0 */
|
||||
SystemCoreClock = OSC_CLK / ((LPC_SC->CCLKCFG & 0xFF)+ 1);
|
||||
break;
|
||||
case 2: /* RTC oscillator => PLL0 */
|
||||
SystemCoreClock = RTC_CLK / ((LPC_SC->CCLKCFG & 0xFF)+ 1);
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
/**
|
||||
* Initialize the system
|
||||
*
|
||||
* @param none
|
||||
* @return none
|
||||
*
|
||||
* @brief Setup the microcontroller system.
|
||||
* Initialize the System.
|
||||
*/
|
||||
void SystemInit (void)
|
||||
{
|
||||
#if (CLOCK_SETUP) /* Clock Setup */
|
||||
LPC_SC->SCS = SCS_Val;
|
||||
if (SCS_Val & (1 << 5)) { /* If Main Oscillator is enabled */
|
||||
while ((LPC_SC->SCS & (1<<6)) == 0);/* Wait for Oscillator to be ready */
|
||||
}
|
||||
|
||||
LPC_SC->CCLKCFG = CCLKCFG_Val; /* Setup Clock Divider */
|
||||
|
||||
LPC_SC->PCLKSEL0 = PCLKSEL0_Val; /* Peripheral Clock Selection */
|
||||
LPC_SC->PCLKSEL1 = PCLKSEL1_Val;
|
||||
|
||||
LPC_SC->CLKSRCSEL = CLKSRCSEL_Val; /* Select Clock Source for PLL0 */
|
||||
|
||||
#if (PLL0_SETUP)
|
||||
LPC_SC->PLL0CFG = PLL0CFG_Val; /* configure PLL0 */
|
||||
LPC_SC->PLL0FEED = 0xAA;
|
||||
LPC_SC->PLL0FEED = 0x55;
|
||||
|
||||
LPC_SC->PLL0CON = 0x01; /* PLL0 Enable */
|
||||
LPC_SC->PLL0FEED = 0xAA;
|
||||
LPC_SC->PLL0FEED = 0x55;
|
||||
while (!(LPC_SC->PLL0STAT & (1<<26)));/* Wait for PLOCK0 */
|
||||
|
||||
LPC_SC->PLL0CON = 0x03; /* PLL0 Enable & Connect */
|
||||
LPC_SC->PLL0FEED = 0xAA;
|
||||
LPC_SC->PLL0FEED = 0x55;
|
||||
while (!(LPC_SC->PLL0STAT & ((1<<25) | (1<<24))));/* Wait for PLLC0_STAT & PLLE0_STAT */
|
||||
#endif
|
||||
|
||||
#if (PLL1_SETUP)
|
||||
LPC_SC->PLL1CFG = PLL1CFG_Val;
|
||||
LPC_SC->PLL1FEED = 0xAA;
|
||||
LPC_SC->PLL1FEED = 0x55;
|
||||
|
||||
LPC_SC->PLL1CON = 0x01; /* PLL1 Enable */
|
||||
LPC_SC->PLL1FEED = 0xAA;
|
||||
LPC_SC->PLL1FEED = 0x55;
|
||||
while (!(LPC_SC->PLL1STAT & (1<<10)));/* Wait for PLOCK1 */
|
||||
|
||||
LPC_SC->PLL1CON = 0x03; /* PLL1 Enable & Connect */
|
||||
LPC_SC->PLL1FEED = 0xAA;
|
||||
LPC_SC->PLL1FEED = 0x55;
|
||||
while (!(LPC_SC->PLL1STAT & ((1<< 9) | (1<< 8))));/* Wait for PLLC1_STAT & PLLE1_STAT */
|
||||
#else
|
||||
LPC_SC->USBCLKCFG = USBCLKCFG_Val; /* Setup USB Clock Divider */
|
||||
#endif
|
||||
|
||||
LPC_SC->PCONP = PCONP_Val; /* Power Control for Peripherals */
|
||||
|
||||
LPC_SC->CLKOUTCFG = CLKOUTCFG_Val; /* Clock Output Configuration */
|
||||
#endif
|
||||
|
||||
#if (FLASH_SETUP == 1) /* Flash Accelerator Setup */
|
||||
LPC_SC->FLASHCFG = (LPC_SC->FLASHCFG & ~0x0000F000) | FLASHCFG_Val;
|
||||
#endif
|
||||
}
|
||||
94
platforms/lpc17xx/Makefile
Normal file
94
platforms/lpc17xx/Makefile
Normal file
@@ -0,0 +1,94 @@
|
||||
ifeq ($(TARGET_NAME),)
|
||||
TARGET_NAME=boot
|
||||
endif
|
||||
ifeq ($(ATOMTHREADS),)
|
||||
ATOMTHREADS = $(shell pwd)/../../
|
||||
endif
|
||||
ifeq ($(TEST_NAME),)
|
||||
TEST_NAME = kern1
|
||||
endif
|
||||
|
||||
CC = arm-none-eabi-gcc
|
||||
LN = arm-none-eabi-gcc
|
||||
AS = arm-none-eabi-gcc
|
||||
|
||||
CFLAGS := $(CFLAGS) -O3 -Os -g3 -Wall -c -mcpu=cortex-m3 -mthumb
|
||||
AFLAGS := $(AFLAGS) -O3 -Os -g3 -Wall -c -fmessage-length=0 -fno-builtin -ffunction-sections -fdata-sections -mcpu=cortex-m3 -mthumb
|
||||
LFLAGS := $(LFLAGS) -O3 -Os -Wall -mcpu=cortex-m3 -mthumb -Wl,-Map=system.map -Tsystem.ld
|
||||
|
||||
CDEFS := $(CDEFS) -DATOMTHREADS_TEST='"$(TEST_NAME)"' -DBOARD_MBED_LP1768
|
||||
ADEFS := $(ADEFS) -D__thumb2__ -DARM_RDI_MONITOR
|
||||
|
||||
LLIBS := $(LLIBS)
|
||||
|
||||
|
||||
SRCS := $(SRCS) \
|
||||
./CMSISv2p00_LPC17xx/src/core_cm3.c \
|
||||
./CMSISv2p00_LPC17xx/src/system_LPC17xx.c \
|
||||
./drivers/lpc17xx_uart.c \
|
||||
startup.c \
|
||||
modules.c \
|
||||
$(ATOMTHREADS)/tests/$(TEST_NAME).c \
|
||||
main.c \
|
||||
|
||||
|
||||
|
||||
ASMS := $(ASMS) \
|
||||
|
||||
|
||||
INCLUDES := $(INCLUDES) \
|
||||
-I$(ATOMTHREADS)/platforms/lpc17xx/CMSISv2p00_LPC17xx/inc \
|
||||
-I$(ATOMTHREADS)/platforms/lpc17xx \
|
||||
-I$(ATOMTHREADS)
|
||||
|
||||
include $(ATOMTHREADS)/ports/cortex_m/Makefile
|
||||
|
||||
OBJS = $(SRCS:.c=.o) $(ASMS:.S=.o)
|
||||
|
||||
include ../rules.mk
|
||||
|
||||
run_test: clean all
|
||||
cp boot.bin bin/$(TEST_NAME).bin
|
||||
|
||||
all_tests:
|
||||
echo "Starting atomthreads test suite"
|
||||
make run_test "TEST_NAME=mutex1"
|
||||
make run_test "TEST_NAME=mutex2"
|
||||
make run_test "TEST_NAME=mutex3"
|
||||
make run_test "TEST_NAME=mutex4"
|
||||
make run_test "TEST_NAME=mutex5"
|
||||
make run_test "TEST_NAME=mutex6"
|
||||
make run_test "TEST_NAME=mutex7"
|
||||
make run_test "TEST_NAME=mutex8"
|
||||
make run_test "TEST_NAME=mutex9"
|
||||
make run_test "TEST_NAME=kern1"
|
||||
make run_test "TEST_NAME=kern2"
|
||||
make run_test "TEST_NAME=kern3"
|
||||
make run_test "TEST_NAME=kern4"
|
||||
make run_test "TEST_NAME=timer1"
|
||||
make run_test "TEST_NAME=timer2"
|
||||
make run_test "TEST_NAME=timer3"
|
||||
make run_test "TEST_NAME=timer4"
|
||||
make run_test "TEST_NAME=timer5"
|
||||
make run_test "TEST_NAME=timer6"
|
||||
make run_test "TEST_NAME=timer7"
|
||||
make run_test "TEST_NAME=queue1"
|
||||
make run_test "TEST_NAME=queue2"
|
||||
make run_test "TEST_NAME=queue3"
|
||||
make run_test "TEST_NAME=queue4"
|
||||
make run_test "TEST_NAME=queue5"
|
||||
make run_test "TEST_NAME=queue6"
|
||||
make run_test "TEST_NAME=queue7"
|
||||
make run_test "TEST_NAME=queue8"
|
||||
make run_test "TEST_NAME=queue9"
|
||||
make run_test "TEST_NAME=sem1"
|
||||
make run_test "TEST_NAME=sem2"
|
||||
make run_test "TEST_NAME=sem3"
|
||||
make run_test "TEST_NAME=sem4"
|
||||
make run_test "TEST_NAME=sem5"
|
||||
make run_test "TEST_NAME=sem6"
|
||||
make run_test "TEST_NAME=sem7"
|
||||
make run_test "TEST_NAME=sem8"
|
||||
make run_test "TEST_NAME=sem9"
|
||||
|
||||
|
||||
24
platforms/lpc17xx/README
Normal file
24
platforms/lpc17xx/README
Normal file
@@ -0,0 +1,24 @@
|
||||
---------------------------------------------------------------------------
|
||||
|
||||
Library: Atomthreads NXP LPC17xx Platform.
|
||||
Author: Natie van Rooyen <natie@navaro.nl>
|
||||
License: BSD Revised
|
||||
|
||||
---------------------------------------------------------------------------
|
||||
|
||||
NXP LPC17xx Platform
|
||||
|
||||
The "lpc17xx" platform contains sources for building the Atomthreads test
|
||||
suite for the NXP LPC17xx microcontroller.
|
||||
|
||||
The build was tested on the "mbed NXP LPC1768" board (http://www.mbed.org)
|
||||
but it should work on any LPC17xx development board where UART0 can be used
|
||||
to monitor the output of the test.
|
||||
|
||||
The NXP LPC17xx microcontrollers use the ARM Cortex M3 processor core. The
|
||||
source code in this example uses the ARM CMSIS Cortex-M Access Library V2.01
|
||||
to initialize the platform and Newlib as the runtime library. Also it uses a
|
||||
driver provided by NXP for the UART. The CMSIS library and the UART driver
|
||||
are provided as source with the sample and Newlib is expected to be installed
|
||||
together with the GNU ARM tool chain.
|
||||
|
||||
1035
platforms/lpc17xx/drivers/lpc17xx.h
Normal file
1035
platforms/lpc17xx/drivers/lpc17xx.h
Normal file
File diff suppressed because it is too large
Load Diff
179
platforms/lpc17xx/drivers/lpc17xx_uart.c
Normal file
179
platforms/lpc17xx/drivers/lpc17xx_uart.c
Normal file
@@ -0,0 +1,179 @@
|
||||
/**************************************************************************//**
|
||||
* @file lpc17xx_uart.c
|
||||
* @brief Drivers for UART peripheral in lpc17xx.
|
||||
* @version 1.0
|
||||
* @date 18. Nov. 2010
|
||||
*
|
||||
* @note
|
||||
* Copyright (C) 2010 NXP Semiconductors(NXP). All rights reserved.
|
||||
*
|
||||
* @par
|
||||
* Software that is described herein is for illustrative purposes only
|
||||
* which provides customers with programming information regarding the
|
||||
* products. This software is supplied "AS IS" without any warranties.
|
||||
* NXP Semiconductors assumes no responsibility or liability for the
|
||||
* use of the software, conveys no license or title under any patent,
|
||||
* copyright, or mask work right to the product. NXP Semiconductors
|
||||
* reserves the right to make changes in the software without
|
||||
* notification. NXP Semiconductors also make no representation or
|
||||
* warranty that such application will be suitable for the specified
|
||||
* use without further testing or modification.
|
||||
******************************************************************************/
|
||||
|
||||
#include <stdarg.h>
|
||||
#include <stdio.h>
|
||||
#include "lpc17xx_uart.h"
|
||||
#include "lpc17xx.h"
|
||||
|
||||
|
||||
/**
|
||||
* @brief Initializes the UART0.
|
||||
*
|
||||
* @param baudrate: Specifies the baud rate
|
||||
* @retval None
|
||||
*/
|
||||
void LPC17xx_UART_Init(uint32_t baudrate)
|
||||
{
|
||||
uint32_t Fdiv;
|
||||
uint32_t pclkdiv, pclk;
|
||||
|
||||
/***/
|
||||
LPC_PINCON->PINSEL0 &= ~0x000000F0;
|
||||
|
||||
LPC_PINCON->PINSEL0 |= 0x00000050; /* RxD0 and TxD0 */
|
||||
|
||||
/* PCLK_UART0=CCLK/2 */
|
||||
//**LPC_SC->PCLKSEL1 &= ~(3<<6); /* PCLK_UART0 = CCLK/4 (18MHz) */
|
||||
//**LPC_SC->PCLKSEL1 |= (2<<6); /* PCLK_UART0 = CCLK/2 (36MHz) */
|
||||
//**pclk = SystemCoreClock/2;
|
||||
|
||||
/* By default, the PCLKSELx value is zero, thus, the PCLK for
|
||||
all the peripherals is 1/4 of the SystemFrequency. */
|
||||
/* Bit 6~7 is for UART0 */
|
||||
pclkdiv = (LPC_SC->PCLKSEL0 >> 6) & 0x03;
|
||||
switch ( pclkdiv )
|
||||
{
|
||||
case 0x00:
|
||||
default:
|
||||
pclk = SystemCoreClock/4;
|
||||
break;
|
||||
case 0x01:
|
||||
pclk = SystemCoreClock;
|
||||
break;
|
||||
case 0x02:
|
||||
pclk = SystemCoreClock/2;
|
||||
break;
|
||||
case 0x03:
|
||||
pclk = SystemCoreClock/8;
|
||||
break;
|
||||
}
|
||||
|
||||
LPC_UART0->LCR = 0x83; /* 8 bits, no Parity, 1 Stop bit */
|
||||
Fdiv = ( pclk / 16 ) / baudrate ; /*baud rate */
|
||||
LPC_UART0->DLM = Fdiv / 256;
|
||||
LPC_UART0->DLL = Fdiv % 256;
|
||||
LPC_UART0->LCR = 0x03; /* DLAB = 0 */
|
||||
LPC_UART0->FCR = 0x07; /* Enable and reset TX and RX FIFO. */
|
||||
}
|
||||
|
||||
/**
|
||||
* @brief Write one character to UART0.
|
||||
*
|
||||
* @param ch: Character to be written
|
||||
* @retval None
|
||||
*/
|
||||
void LPC17xx_UART_PutChar (uint8_t ch)
|
||||
{
|
||||
while (!(LPC_UART0->LSR & 0x20));
|
||||
|
||||
LPC_UART0->THR = ch;
|
||||
}
|
||||
|
||||
/**
|
||||
* @brief Read one character from UART0 (blocking read).
|
||||
*
|
||||
* @param None
|
||||
* @retval Received character
|
||||
*/
|
||||
uint8_t LPC17xx_UART_GetChar (void)
|
||||
{
|
||||
while (!(LPC_UART0->LSR & 0x01));
|
||||
return (LPC_UART0->RBR);
|
||||
}
|
||||
|
||||
/**
|
||||
* @brief Read one character from UART0 (non blocking read).
|
||||
*
|
||||
* @param None
|
||||
* @retval Received character
|
||||
*/
|
||||
uint8_t LPC17xx_UART_GetChar_nb (void)
|
||||
{
|
||||
if (LPC_UART0->LSR & 0x01)
|
||||
return (LPC_UART0->RBR);
|
||||
else
|
||||
return 0;
|
||||
}
|
||||
|
||||
/**
|
||||
* @brief Write a string to UART0.
|
||||
*
|
||||
* @param str: NULL-terminated char string to be written
|
||||
* @retval None
|
||||
*/
|
||||
void LPC17xx_UART_PutString (uint8_t *str)
|
||||
{
|
||||
/* usage: LPC1700_UART_Printf("xxx\n\r");*/
|
||||
#if 1
|
||||
while (*str != 0)
|
||||
{
|
||||
LPC17xx_UART_PutChar(*str++);
|
||||
}
|
||||
|
||||
#else
|
||||
/* usage: LPC1700_UART_Printf("xxx\n");*/
|
||||
while ((*str) != 0) {
|
||||
if (*str == '\n') {
|
||||
LPC17xx_UART_PutChar(*str++);
|
||||
LPC17xx_UART_PutChar('\r');
|
||||
} else {
|
||||
LPC17xx_UART_PutChar(*str++);
|
||||
}
|
||||
}
|
||||
#endif
|
||||
}
|
||||
|
||||
/**
|
||||
* @brief Write a buffer to UART0.
|
||||
*
|
||||
* @param buffer: buffer to be written
|
||||
* @retval None
|
||||
*/
|
||||
void LPC17xx_UART_WriteBuffer (uint8_t *buffer, uint32_t len)
|
||||
{
|
||||
while (len-- != 0) {
|
||||
LPC17xx_UART_PutChar(*buffer++);
|
||||
}
|
||||
|
||||
}
|
||||
/**
|
||||
* @brief Print formatted string. This function takes variable length arguments.
|
||||
*
|
||||
* @param format
|
||||
* @param ...
|
||||
* @retval None
|
||||
*
|
||||
* Note: using library functions "vsprintf" will increase the RO size by about 6KB
|
||||
*/
|
||||
//void LPC17xx_UART_Printf (const uint8_t *format, ...)
|
||||
//{
|
||||
// static uint8_t buffer[40 + 1];
|
||||
// va_list vArgs;
|
||||
//
|
||||
// va_start(vArgs, format);
|
||||
// vsprintf((char *)buffer, (char const *)format, vArgs);
|
||||
// va_end(vArgs);
|
||||
// LPC17xx_UART_PutString((uint8_t *) buffer);
|
||||
//}
|
||||
|
||||
/* --------------------------------- End Of File ------------------------------ */
|
||||
37
platforms/lpc17xx/drivers/lpc17xx_uart.h
Normal file
37
platforms/lpc17xx/drivers/lpc17xx_uart.h
Normal file
@@ -0,0 +1,37 @@
|
||||
/**************************************************************************//**
|
||||
* @file lpc17xx_uart.h
|
||||
* @brief Header file for lpc17xx_uart.c.
|
||||
* @version 1.0
|
||||
* @date 18. Nov. 2010
|
||||
*
|
||||
* @note
|
||||
* Copyright (C) 2010 NXP Semiconductors(NXP). All rights reserved.
|
||||
*
|
||||
* @par
|
||||
* Software that is described herein is for illustrative purposes only
|
||||
* which provides customers with programming information regarding the
|
||||
* products. This software is supplied "AS IS" without any warranties.
|
||||
* NXP Semiconductors assumes no responsibility or liability for the
|
||||
* use of the software, conveys no license or title under any patent,
|
||||
* copyright, or mask work right to the product. NXP Semiconductors
|
||||
* reserves the right to make changes in the software without
|
||||
* notification. NXP Semiconductors also make no representation or
|
||||
* warranty that such application will be suitable for the specified
|
||||
* use without further testing or modification.
|
||||
******************************************************************************/
|
||||
|
||||
#ifndef __LPC17xx_UART_H_
|
||||
#define __LPC17xx_UART_H_
|
||||
|
||||
#include <stdint.h>
|
||||
|
||||
/* external functions */
|
||||
void LPC17xx_UART_PutChar (uint8_t);
|
||||
uint8_t LPC17xx_UART_GetChar (void);
|
||||
void LPC17xx_UART_Init(uint32_t baudrate);
|
||||
void LPC17xx_UART_PutString (uint8_t *str) ;
|
||||
void LPC17xx_UART_WriteBuffer (uint8_t *buffer, uint32_t len) ;
|
||||
|
||||
#endif // __LPC17xx_UART_H_
|
||||
|
||||
/* --------------------------------- End Of File ------------------------------ */
|
||||
122
platforms/lpc17xx/main.c
Normal file
122
platforms/lpc17xx/main.c
Normal file
@@ -0,0 +1,122 @@
|
||||
/*
|
||||
* Copyright (c) 2012, Natie van Rooyen. 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 "LPC17xx.h"
|
||||
#include "drivers/lpc17xx_uart.h"
|
||||
#include "modules.h"
|
||||
#include "atom.h"
|
||||
#include "tests/atomtests.h"
|
||||
|
||||
|
||||
|
||||
#ifndef ATOMTHREADS_TEST
|
||||
#define ATOMTHREADS_TEST "kern1"
|
||||
#endif
|
||||
|
||||
// for mbed board
|
||||
#define MBED_LED1_GPIO (1 << 18)
|
||||
#define MBED_LED2_GPIO (1 << 20)
|
||||
#define MBED_LED3_GPIO (1 << 21)
|
||||
#define MBED_LED4_GPIO (1 << 23)
|
||||
|
||||
#define MBED_LED_GET(led) (LPC_GPIO1->FIOSET & led)
|
||||
#define MBED_LED_SET(led, on) { if (on) LPC_GPIO1->FIOSET = led ; else LPC_GPIO1->FIOCLR = led ; }
|
||||
#define MBED_LED_TOGGLE(led) MBED_LED_SET(led, !MBED_LED_GET(led))
|
||||
#define MBED_LED_COUNT(count) MBED_LED_SET(MBED_LED1_GPIO, count & 1) ; MBED_LED_SET(MBED_LED2_GPIO, count & 2) ; \
|
||||
MBED_LED_SET(MBED_LED3_GPIO, count & 4) ; MBED_LED_SET(MBED_LED4_GPIO, count & 8) ;
|
||||
|
||||
|
||||
#define TEST_STACK_BYTE_SIZE 512
|
||||
#define IDLE_STACK_BYTE_SIZE 128
|
||||
|
||||
static unsigned char test_stack[TEST_STACK_BYTE_SIZE] ;
|
||||
static unsigned char idle_stack[IDLE_STACK_BYTE_SIZE] ;
|
||||
ATOM_TCB test_tcb ;
|
||||
|
||||
|
||||
/**
|
||||
* \b test_thread
|
||||
*
|
||||
* Function calling the test function of the Atomthreads test suite.
|
||||
*
|
||||
*/
|
||||
void
|
||||
test_thread (uint32_t param)
|
||||
{
|
||||
uint32_t failures ;
|
||||
CRITICAL_STORE ;
|
||||
|
||||
failures = test_start () ;
|
||||
|
||||
atomTimerDelay (10) ;
|
||||
CRITICAL_START() ;
|
||||
dbg_format_msg ("%s %s\r\n", ATOMTHREADS_TEST, failures ? "FAIL" : "PASS") ;
|
||||
CRITICAL_END() ;
|
||||
|
||||
while(1) {
|
||||
#ifdef BOARD_MBED_LP1768
|
||||
MBED_LED_TOGGLE(MBED_LED1_GPIO) ;
|
||||
#endif
|
||||
atomTimerDelay (65) ;
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
/**
|
||||
* \b main
|
||||
*
|
||||
* Initialize atomthreads and start a test_thread to run the Atomthreads test suite.
|
||||
*
|
||||
*/
|
||||
int
|
||||
main(void)
|
||||
{
|
||||
|
||||
#ifdef BOARD_MBED_LP1768
|
||||
LPC_GPIO1->FIODIR |= MBED_LED1_GPIO | MBED_LED2_GPIO | MBED_LED3_GPIO | MBED_LED4_GPIO ;
|
||||
MBED_LED_SET(MBED_LED1_GPIO | MBED_LED2_GPIO | MBED_LED3_GPIO | MBED_LED4_GPIO, 1);
|
||||
#endif
|
||||
|
||||
dbg_format_msg ("\r\nLPC17xx SystemCoreClock = %d\r\n",SystemCoreClock) ;
|
||||
|
||||
//atomthreads_stress_test (36) ;
|
||||
|
||||
dbg_format_msg ("Atomthreads starting %s... \r\n", ATOMTHREADS_TEST) ;
|
||||
|
||||
atomOSInit(&idle_stack[0], IDLE_STACK_BYTE_SIZE, TRUE) ;
|
||||
atomThreadCreate ((ATOM_TCB *)&test_tcb, TEST_THREAD_PRIO, test_thread, 0,
|
||||
&test_stack[0], TEST_STACK_BYTE_SIZE, TRUE);
|
||||
atomOSStart() ;
|
||||
|
||||
while(1) ;
|
||||
|
||||
return 0 ;
|
||||
}
|
||||
|
||||
179
platforms/lpc17xx/modules.c
Normal file
179
platforms/lpc17xx/modules.c
Normal file
@@ -0,0 +1,179 @@
|
||||
/*
|
||||
* Copyright (c) 2012, Natie van Rooyen. 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 "modules.h"
|
||||
#include <stdio.h>
|
||||
#include <stdarg.h>
|
||||
#include "atomport_private.h"
|
||||
#include "atom.h"
|
||||
#include "atomport.h"
|
||||
|
||||
|
||||
|
||||
/**
|
||||
* \b dbg_format_msg
|
||||
*
|
||||
* Same as printf.
|
||||
*
|
||||
*/
|
||||
void
|
||||
dbg_format_msg (char *format, ...)
|
||||
{
|
||||
va_list args;
|
||||
static char msg[256] ;
|
||||
//CRITICAL_STORE ;
|
||||
|
||||
va_start (args, format) ;
|
||||
//CRITICAL_START() ;
|
||||
|
||||
vsniprintf ((char*)msg, 256, (char*)format, args) ;
|
||||
LPC17xx_UART_PutString (msg) ;
|
||||
//CRITICAL_END() ;
|
||||
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* \b low_level_init
|
||||
*
|
||||
* Initializes the PIC and start the system timer tick intrerupt.
|
||||
*
|
||||
*/
|
||||
int
|
||||
low_level_init (void)
|
||||
{
|
||||
SystemInit () ;
|
||||
SystemCoreClockUpdate ();
|
||||
//contextInit () ;
|
||||
NVIC_SetPriority (PendSV_IRQn, 0xFF) ;
|
||||
LPC17xx_UART_Init (115200) ;
|
||||
SysTick_Config (1000000) ;
|
||||
|
||||
return 0 ;
|
||||
}
|
||||
|
||||
|
||||
|
||||
/**
|
||||
* \b __context_preempt_handler
|
||||
*
|
||||
* System timer tic interupt handler.
|
||||
*
|
||||
*/
|
||||
void
|
||||
__context_tick_handler (void)
|
||||
{
|
||||
|
||||
/* Call the interrupt enter routine */
|
||||
atomIntEnter();
|
||||
|
||||
/* Call the OS system tick handler */
|
||||
atomTimerTick();
|
||||
|
||||
/* Call the interrupt exit routine */
|
||||
atomIntExit(TRUE);
|
||||
|
||||
}
|
||||
|
||||
/**
|
||||
* \b dbg_mem_dump_40
|
||||
*
|
||||
* Dumps size bytes of memory from data.
|
||||
*
|
||||
*/
|
||||
void dbg_mem_dump_40 (unsigned int* data, int size)
|
||||
{
|
||||
int j ;
|
||||
|
||||
dbg_format_msg ("Dump %d bytes at %.8X:\r\n",size * 4, (unsigned int)data) ;
|
||||
data = (unsigned int*)((unsigned int)data & ~0x3) ;
|
||||
for (j=0; j<size-3; j+=4) {
|
||||
dbg_format_msg (" :%.8X: %.8X %.8X %.8X %.8X\r\n", (unsigned int)&data[j], data[j+0], data[j+1], data[j+2], data[j+3]) ;
|
||||
}
|
||||
|
||||
if (size-j == 3) {
|
||||
dbg_format_msg (" :%.8X: %.8X %.8X %.8X\r\n", (unsigned int)&data[j], data[j+0], data[j+1], data[j+2]) ;
|
||||
} else if (size-j == 2) {
|
||||
dbg_format_msg (" :%.8X: %.8X %.8X\r\n", (unsigned int)&data[j], data[j+0], data[j+1]) ;
|
||||
} else if (size-j == 1) {
|
||||
dbg_format_msg (" :%.8X: %.8X\r\n", (unsigned int)&data[j], data[j+0]) ;
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
/**
|
||||
* \b dbg_fault_handler
|
||||
*
|
||||
* Prints cortex m exception debug information.
|
||||
*
|
||||
*/
|
||||
void
|
||||
dbg_fault_handler (unsigned int * hardfault_args)
|
||||
{
|
||||
unsigned int stacked_r0;
|
||||
unsigned int stacked_r1;
|
||||
unsigned int stacked_r2;
|
||||
unsigned int stacked_r3;
|
||||
unsigned int stacked_r12;
|
||||
unsigned int stacked_lr;
|
||||
unsigned int stacked_pc;
|
||||
unsigned int stacked_psr;
|
||||
|
||||
stacked_r0 = ((unsigned long) hardfault_args[0]);
|
||||
stacked_r1 = ((unsigned long) hardfault_args[1]);
|
||||
stacked_r2 = ((unsigned long) hardfault_args[2]);
|
||||
stacked_r3 = ((unsigned long) hardfault_args[3]);
|
||||
|
||||
stacked_r12 = ((unsigned long) hardfault_args[4]);
|
||||
stacked_lr = ((unsigned long) hardfault_args[5]);
|
||||
stacked_pc = ((unsigned long) hardfault_args[6]);
|
||||
stacked_psr = ((unsigned long) hardfault_args[7]);
|
||||
|
||||
dbg_format_msg ("\r\n\r\n[Hard fault handler - all numbers in hex]\r\n");
|
||||
dbg_format_msg ("SP = 0x%x\r\n", hardfault_args);
|
||||
dbg_format_msg ("R0 = 0x%x\r\n", stacked_r0);
|
||||
dbg_format_msg ("R1 = 0x%x\r\n", stacked_r1);
|
||||
dbg_format_msg ("R2 = 0x%x\r\n", stacked_r2);
|
||||
dbg_format_msg ("R3 = 0x%x\r\n", stacked_r3);
|
||||
dbg_format_msg ("R12 = 0x%x\r\n", stacked_r12);
|
||||
dbg_format_msg ("LR [R14] = 0x%x subroutine call return address\r\n", stacked_lr);
|
||||
dbg_format_msg ("PC [R15] = 0x%x program counter\r\n", stacked_pc);
|
||||
dbg_format_msg ("PSR = 0x%x\r\n", stacked_psr);
|
||||
//printf ("BFAR = 0x%x\r\n", (*((volatile unsigned long *)(0xE000ED38))));
|
||||
//printf ("CFSR = 0x%x\r\n", (*((volatile unsigned long *)(0xE000ED28))));
|
||||
//printf ("HFSR = 0x%x\r\n", (*((volatile unsigned long *)(0xE000ED2C))));
|
||||
//printf ("DFSR = 0x%x\r\n", (*((volatile unsigned long *)(0xE000ED30))));
|
||||
//printf ("AFSR = 0x%x\r\n", (*((volatile unsigned long *)(0xE000ED3C))));
|
||||
// printf ("SCB_SHCSR = %x\n", SCB->SHCSR);
|
||||
|
||||
dbg_mem_dump_40 (hardfault_args, 0x40) ;
|
||||
|
||||
while (1);
|
||||
|
||||
}
|
||||
|
||||
46
platforms/lpc17xx/modules.h
Normal file
46
platforms/lpc17xx/modules.h
Normal file
@@ -0,0 +1,46 @@
|
||||
/*
|
||||
* Copyright (c) 2012, Natie van Rooyen. 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 __MODULES_H__
|
||||
#define __MODULES_H__
|
||||
|
||||
/*
|
||||
* Module definitions to use with the Stellaris LM3S6965 Microcontroller
|
||||
*/
|
||||
|
||||
|
||||
#include "LPC17xx.h"
|
||||
#include "drivers/lpc17xx_uart.h"
|
||||
|
||||
|
||||
/* Function prototypes */
|
||||
extern int low_level_init (void) ;
|
||||
extern void dbg_format_msg (char *format, ...) ;
|
||||
extern void dbg_fault_handler (unsigned int * hardfault_args) ;
|
||||
|
||||
#endif /* __MODULES_H__ */
|
||||
297
platforms/lpc17xx/startup.c
Normal file
297
platforms/lpc17xx/startup.c
Normal file
@@ -0,0 +1,297 @@
|
||||
/**************************************************************************//**
|
||||
* @file startup.c
|
||||
* @brief
|
||||
* @version
|
||||
* @date
|
||||
*
|
||||
* @note
|
||||
* Copyright (C) 2010 NXP Semiconductors(NXP). All rights reserved.
|
||||
*
|
||||
* @par
|
||||
* Software that is described herein is for illustrative purposes only
|
||||
* which provides customers with programming information regarding the
|
||||
* products. This software is supplied "AS IS" without any warranties.
|
||||
* NXP Semiconductors assumes no responsibility or liability for the
|
||||
* use of the software, conveys no license or title under any patent,
|
||||
* copyright, or mask work right to the product. NXP Semiconductors
|
||||
* reserves the right to make changes in the software without
|
||||
* notification. NXP Semiconductors also make no representation or
|
||||
* warranty that such application will be suitable for the specified
|
||||
* use without further testing or modification.
|
||||
******************************************************************************/
|
||||
#define WEAK __attribute__ ((weak))
|
||||
#define ALIAS(f) __attribute__ ((weak, alias (#f)))
|
||||
|
||||
|
||||
#include "system_LPC17xx.h"
|
||||
#include "atomport_private.h"
|
||||
|
||||
|
||||
|
||||
|
||||
void ResetISR(void);
|
||||
WEAK void NMI_Handler(void);
|
||||
WEAK void HardFault_Handler(void);
|
||||
WEAK void MemManage_Handler(void);
|
||||
WEAK void BusFault_Handler(void);
|
||||
WEAK void UsageFault_Handler(void);
|
||||
WEAK void SVC_Handler(void);
|
||||
WEAK void DebugMon_Handler(void);
|
||||
WEAK void PendSV_Handler(void);
|
||||
WEAK void SysTick_Handler(void);
|
||||
WEAK void IntDefault_Handler(void);
|
||||
|
||||
|
||||
//*****************************************************************************
|
||||
//
|
||||
// The entry point for the application.
|
||||
// __main() is the entry point for Redlib based applications
|
||||
// main() is the entry point for Newlib based applications
|
||||
//
|
||||
//*****************************************************************************
|
||||
|
||||
extern int main(void);
|
||||
extern void low_level_init(void);
|
||||
|
||||
//*****************************************************************************
|
||||
//
|
||||
// External declaration for the pointer to the stack top from the Linker Script
|
||||
//
|
||||
//*****************************************************************************
|
||||
extern void _vStackTop(void);
|
||||
|
||||
|
||||
//*****************************************************************************
|
||||
//
|
||||
// The vector table.
|
||||
// This relies on the linker script to place at correct location in memory.
|
||||
//
|
||||
//*****************************************************************************
|
||||
extern void (* const g_pfnVectors[])(void);
|
||||
__attribute__ ((section(".isr_vector")))
|
||||
void (* const g_pfnVectors[])(void) = {
|
||||
// Core Level - CM3
|
||||
&_vStackTop, // The initial stack pointer
|
||||
ResetISR, // The reset handler
|
||||
NMI_Handler, // The NMI handler
|
||||
HardFault_Handler, // The hard fault handler
|
||||
MemManage_Handler, // The MPU fault handler
|
||||
BusFault_Handler, // The bus fault handler
|
||||
UsageFault_Handler, // The usage fault handler
|
||||
0, // Reserved
|
||||
0, // Reserved
|
||||
0, // Reserved
|
||||
0, // Reserved
|
||||
SVC_Handler, // SVCall handler
|
||||
DebugMon_Handler, // Debug monitor handler
|
||||
0, // Reserved
|
||||
archPendSVHandler, // The PendSV handler
|
||||
archTickHandler /*SysTick_Handler*/, // The SysTick handler
|
||||
|
||||
// Chip Level - LPC17
|
||||
IntDefault_Handler, // 16, 0x40 - WDT
|
||||
IntDefault_Handler, // 17, 0x44 - TIMER0
|
||||
IntDefault_Handler, // 18, 0x48 - TIMER1
|
||||
IntDefault_Handler, // 19, 0x4c - TIMER2
|
||||
IntDefault_Handler, // 20, 0x50 - TIMER3
|
||||
IntDefault_Handler, // 21, 0x54 - UART0
|
||||
IntDefault_Handler, // 22, 0x58 - UART1
|
||||
IntDefault_Handler, // 23, 0x5c - UART2
|
||||
IntDefault_Handler, // 24, 0x60 - UART3
|
||||
IntDefault_Handler, // 25, 0x64 - PWM1
|
||||
IntDefault_Handler, // 26, 0x68 - I2C0
|
||||
IntDefault_Handler, // 27, 0x6c - I2C1
|
||||
IntDefault_Handler, // 28, 0x70 - I2C2
|
||||
IntDefault_Handler, // 29, 0x74 - SPI
|
||||
IntDefault_Handler, // 30, 0x78 - SSP0
|
||||
IntDefault_Handler, // 31, 0x7c - SSP1
|
||||
IntDefault_Handler, // 32, 0x80 - PLL0 (Main PLL)
|
||||
IntDefault_Handler, // 33, 0x84 - RTC
|
||||
IntDefault_Handler, // 34, 0x88 - EINT0
|
||||
IntDefault_Handler, // 35, 0x8c - EINT1
|
||||
IntDefault_Handler, // 36, 0x90 - EINT2
|
||||
IntDefault_Handler, // 37, 0x94 - EINT3
|
||||
IntDefault_Handler, // 38, 0x98 - ADC
|
||||
IntDefault_Handler, // 39, 0x9c - BOD
|
||||
IntDefault_Handler, // 40, 0xA0 - USB
|
||||
IntDefault_Handler, // 41, 0xa4 - CAN
|
||||
IntDefault_Handler, // 42, 0xa8 - GP DMA
|
||||
IntDefault_Handler, // 43, 0xac - I2S
|
||||
IntDefault_Handler, // 44, 0xb0 - Ethernet
|
||||
IntDefault_Handler, // 45, 0xb4 - RITINT
|
||||
IntDefault_Handler, // 46, 0xb8 - Motor Control PWM
|
||||
IntDefault_Handler, // 47, 0xbc - Quadrature Encoder
|
||||
IntDefault_Handler, // 48, 0xc0 - PLL1 (USB PLL)
|
||||
IntDefault_Handler, // 49, 0xc4 - USB Activity interrupt to wakeup
|
||||
IntDefault_Handler, // 50, 0xc8 - CAN Activity interrupt to wakeup
|
||||
};
|
||||
|
||||
//*****************************************************************************
|
||||
// Functions to carry out the initialization of RW and BSS data sections. These
|
||||
// are written as separate functions rather than being inlined within the
|
||||
// ResetISR() function in order to cope with MCUs with multiple banks of
|
||||
// memory.
|
||||
//*****************************************************************************
|
||||
__attribute__ ((section(".after_vectors")))
|
||||
void data_init(unsigned int romstart, unsigned int start, unsigned int len) {
|
||||
unsigned int *pulDest = (unsigned int*) start;
|
||||
unsigned int *pulSrc = (unsigned int*) romstart;
|
||||
unsigned int loop;
|
||||
for (loop = 0; loop < len; loop = loop + 4)
|
||||
*pulDest++ = *pulSrc++;
|
||||
}
|
||||
|
||||
__attribute__ ((section(".after_vectors")))
|
||||
void bss_init(unsigned int start, unsigned int len) {
|
||||
unsigned int *pulDest = (unsigned int*) start;
|
||||
unsigned int loop;
|
||||
for (loop = 0; loop < len; loop = loop + 4)
|
||||
*pulDest++ = 0;
|
||||
}
|
||||
|
||||
|
||||
//*****************************************************************************
|
||||
// The following symbols are constructs generated by the linker, indicating
|
||||
// the location of various points in the "Global Section Table". This table is
|
||||
// created by the linker via the Code Red managed linker script mechanism. It
|
||||
// contains the load address, execution address and length of each RW data
|
||||
// section and the execution and length of each BSS (zero initialized) section.
|
||||
//*****************************************************************************
|
||||
extern unsigned int __data_section_table;
|
||||
extern unsigned int __data_section_table_end;
|
||||
extern unsigned int __bss_section_table;
|
||||
extern unsigned int __bss_section_table_end;
|
||||
|
||||
|
||||
//*****************************************************************************
|
||||
// Reset entry point for your code.
|
||||
// Sets up a simple runtime environment and initializes the C/C++
|
||||
// library.
|
||||
//*****************************************************************************
|
||||
__attribute__ ((section(".after_vectors")))
|
||||
void
|
||||
ResetISR(void) {
|
||||
|
||||
//
|
||||
// Copy the data sections from flash to SRAM.
|
||||
//
|
||||
unsigned int LoadAddr, ExeAddr, SectionLen;
|
||||
unsigned int *SectionTableAddr;
|
||||
|
||||
// Load base address of Global Section Table
|
||||
SectionTableAddr = &__data_section_table;
|
||||
|
||||
// Copy the data sections from flash to SRAM.
|
||||
while (SectionTableAddr < &__data_section_table_end) {
|
||||
LoadAddr = *SectionTableAddr++;
|
||||
ExeAddr = *SectionTableAddr++;
|
||||
SectionLen = *SectionTableAddr++;
|
||||
data_init(LoadAddr, ExeAddr, SectionLen);
|
||||
}
|
||||
// At this point, SectionTableAddr = &__bss_section_table;
|
||||
// Zero fill the bss segment
|
||||
while (SectionTableAddr < &__bss_section_table_end) {
|
||||
ExeAddr = *SectionTableAddr++;
|
||||
SectionLen = *SectionTableAddr++;
|
||||
bss_init(ExeAddr, SectionLen);
|
||||
}
|
||||
|
||||
|
||||
|
||||
low_level_init();
|
||||
|
||||
|
||||
|
||||
|
||||
main();
|
||||
|
||||
|
||||
//
|
||||
// main() shouldn't return, but if it does, we'll just enter an infinite loop
|
||||
//
|
||||
while (1) ;
|
||||
}
|
||||
|
||||
//*****************************************************************************
|
||||
// Default exception handlers. Override the ones here by defining your own
|
||||
// handler routines in your application code.
|
||||
//*****************************************************************************
|
||||
__attribute__ ((section(".after_vectors"))) __attribute__( ( naked ) )
|
||||
void NMI_Handler(void)
|
||||
{
|
||||
while(1) ;
|
||||
|
||||
}
|
||||
|
||||
__attribute__ ((section(".after_vectors"))) __attribute__( ( naked ) )
|
||||
void HardFault_Handler(void)
|
||||
{
|
||||
|
||||
__asm volatile
|
||||
(
|
||||
" tst lr, #4 \n"
|
||||
" ite eq \n"
|
||||
" mrseq r0, msp \n"
|
||||
" mrsne r0, psp \n"
|
||||
" b dbg_fault_handler \n"
|
||||
);
|
||||
|
||||
while(1) ;
|
||||
|
||||
}
|
||||
|
||||
__attribute__ ((section(".after_vectors"))) __attribute__( ( naked ) )
|
||||
void MemManage_Handler(void)
|
||||
{
|
||||
while(1) ;
|
||||
|
||||
}
|
||||
|
||||
__attribute__ ((section(".after_vectors"))) __attribute__( ( naked ) )
|
||||
void BusFault_Handler(void)
|
||||
{
|
||||
while(1) ;
|
||||
|
||||
}
|
||||
|
||||
__attribute__ ((section(".after_vectors"))) __attribute__( ( naked ) )
|
||||
void UsageFault_Handler(void)
|
||||
{
|
||||
while(1) ;
|
||||
|
||||
}
|
||||
|
||||
__attribute__ ((section(".after_vectors"))) __attribute__( ( naked ) )
|
||||
void SVC_Handler(void)
|
||||
{
|
||||
while(1) ;
|
||||
|
||||
}
|
||||
|
||||
__attribute__ ((section(".after_vectors"))) __attribute__( ( naked ) )
|
||||
void DebugMon_Handler(void)
|
||||
{
|
||||
while(1) ;
|
||||
|
||||
}
|
||||
|
||||
__attribute__ ((section(".after_vectors"))) __attribute__( ( naked ) )
|
||||
void PendSV_Handler(void)
|
||||
{
|
||||
while(1) ;
|
||||
|
||||
}
|
||||
|
||||
__attribute__ ((section(".after_vectors"))) __attribute__( ( naked ) )
|
||||
void SysTick_Handler(void)
|
||||
{
|
||||
while(1) ;
|
||||
|
||||
}
|
||||
|
||||
__attribute__ ((section(".after_vectors"))) __attribute__( ( naked ) )
|
||||
void IntDefault_Handler(void)
|
||||
{
|
||||
while(1) ;
|
||||
|
||||
}
|
||||
148
platforms/lpc17xx/system.ld
Normal file
148
platforms/lpc17xx/system.ld
Normal file
@@ -0,0 +1,148 @@
|
||||
/**************************************************************************//**
|
||||
* @file system.ld
|
||||
* @brief
|
||||
* @version
|
||||
* @date
|
||||
*
|
||||
* @note
|
||||
* Copyright (C) 2010 NXP Semiconductors(NXP). All rights reserved.
|
||||
*
|
||||
* @par
|
||||
* Software that is described herein is for illustrative purposes only
|
||||
* which provides customers with programming information regarding the
|
||||
* products. This software is supplied "AS IS" without any warranties.
|
||||
* NXP Semiconductors assumes no responsibility or liability for the
|
||||
* use of the software, conveys no license or title under any patent,
|
||||
* copyright, or mask work right to the product. NXP Semiconductors
|
||||
* reserves the right to make changes in the software without
|
||||
* notification. NXP Semiconductors also make no representation or
|
||||
* warranty that such application will be suitable for the specified
|
||||
* use without further testing or modification.
|
||||
******************************************************************************/
|
||||
|
||||
MEMORY
|
||||
{
|
||||
/* Define each memory region */
|
||||
MFlash512 (rx) : ORIGIN = 0x0, LENGTH = 0x80000 /* 512k */
|
||||
RamLoc32 (rwx) : ORIGIN = 0x10000000, LENGTH = 0x8000 /* 32k */
|
||||
RamAHB32 (rwx) : ORIGIN = 0x2007c000, LENGTH = 0x8000 /* 32k */
|
||||
|
||||
}
|
||||
/* Define a symbol for the top of each memory region */
|
||||
__top_MFlash512 = 0x0 + 0x80000;
|
||||
__top_RamLoc32 = 0x10000000 + 0x8000;
|
||||
__top_RamAHB32 = 0x2007c000 + 0x8000;
|
||||
|
||||
|
||||
|
||||
ENTRY(ResetISR)
|
||||
|
||||
SECTIONS
|
||||
{
|
||||
|
||||
/* MAIN TEXT SECTION */
|
||||
.text : ALIGN(4)
|
||||
{
|
||||
FILL(0xff)
|
||||
KEEP(*(.isr_vector))
|
||||
|
||||
/* Global Section Table */
|
||||
. = ALIGN(4) ;
|
||||
__section_table_start = .;
|
||||
__data_section_table = .;
|
||||
LONG(LOADADDR(.data));
|
||||
LONG( ADDR(.data)) ;
|
||||
LONG( SIZEOF(.data));
|
||||
LONG(LOADADDR(.data_RAM2));
|
||||
LONG( ADDR(.data_RAM2)) ;
|
||||
LONG( SIZEOF(.data_RAM2));
|
||||
__data_section_table_end = .;
|
||||
__bss_section_table = .;
|
||||
LONG( ADDR(.bss));
|
||||
LONG( SIZEOF(.bss));
|
||||
LONG( ADDR(.bss_RAM2));
|
||||
LONG( SIZEOF(.bss_RAM2));
|
||||
__bss_section_table_end = .;
|
||||
__section_table_end = . ;
|
||||
/* End of Global Section Table */
|
||||
|
||||
|
||||
*(.after_vectors*)
|
||||
|
||||
*(.text*)
|
||||
*(.rodata .rodata.*)
|
||||
. = ALIGN(4);
|
||||
|
||||
} > MFlash512
|
||||
|
||||
/*
|
||||
* for exception handling/unwind - some Newlib functions (in common
|
||||
* with C++ and STDC++) use this.
|
||||
* Use KEEP so not discarded with --gc-sections
|
||||
*/
|
||||
.ARM.extab : ALIGN(4)
|
||||
{
|
||||
KEEP(*(.ARM.extab* .gnu.linkonce.armextab.*))
|
||||
} > MFlash512
|
||||
__exidx_start = .;
|
||||
|
||||
.ARM.exidx : ALIGN(4)
|
||||
{
|
||||
KEEP(*(.ARM.exidx* .gnu.linkonce.armexidx.*))
|
||||
} > MFlash512
|
||||
__exidx_end = .;
|
||||
|
||||
_etext = .;
|
||||
|
||||
|
||||
.data_RAM2 : ALIGN(4)
|
||||
{
|
||||
FILL(0xff)
|
||||
*(.data.$RAM2*)
|
||||
*(.data.$RamAHB32*)
|
||||
. = ALIGN(4) ;
|
||||
} > RamAHB32 AT>MFlash512
|
||||
|
||||
/* MAIN DATA SECTION */
|
||||
|
||||
.uninit_RESERVED : ALIGN(4)
|
||||
{
|
||||
KEEP(*(.bss.$RESERVED*))
|
||||
. = ALIGN(4) ;
|
||||
_end_uninit_RESERVED = .;
|
||||
} > RamLoc32
|
||||
|
||||
.data : ALIGN(4)
|
||||
{
|
||||
FILL(0xff)
|
||||
_data = .;
|
||||
*(vtable)
|
||||
*(.data*)
|
||||
. = ALIGN(4) ;
|
||||
_edata = .;
|
||||
} > RamLoc32 AT>MFlash512
|
||||
|
||||
|
||||
.bss_RAM2 : ALIGN(4)
|
||||
{
|
||||
*(.bss.$RAM2*)
|
||||
*(.bss.$RamAHB32*)
|
||||
. = ALIGN(4) ;
|
||||
} > RamAHB32
|
||||
|
||||
/* MAIN BSS SECTION */
|
||||
.bss : ALIGN(4)
|
||||
{
|
||||
__bss_start__ = . ;
|
||||
_bss = .;
|
||||
*(.bss*)
|
||||
*(COMMON)
|
||||
. = ALIGN(4) ;
|
||||
__bss_end__ = . ;
|
||||
_ebss = .;
|
||||
PROVIDE(end = .);
|
||||
} > RamLoc32
|
||||
|
||||
PROVIDE(_pvHeapStart = .);
|
||||
PROVIDE(_vStackTop = __top_RamLoc32 - 0);
|
||||
}
|
||||
96
platforms/qemu_lm3s/Makefile
Normal file
96
platforms/qemu_lm3s/Makefile
Normal file
@@ -0,0 +1,96 @@
|
||||
ifeq ($(TARGET_NAME),)
|
||||
TARGET_NAME=boot
|
||||
endif
|
||||
ifeq ($(ATOMTHREADS),)
|
||||
ATOMTHREADS = $(shell pwd)/../../
|
||||
endif
|
||||
ifeq ($(TEST_NAME),)
|
||||
TEST_NAME = kern1
|
||||
endif
|
||||
|
||||
|
||||
|
||||
CC = arm-none-eabi-gcc
|
||||
LN = arm-none-eabi-gcc
|
||||
AS = arm-none-eabi-gcc
|
||||
|
||||
CFLAGS := $(CFLAGS) -Wall -g -c -mcpu=cortex-m3 -mthumb -mthumb-interwork -ffreestanding
|
||||
AFLAGS := $(AFLAGS) -Wall -g -c -mcpu=cortex-m3 -mthumb -mthumb-interwork -ffreestanding
|
||||
LFLAGS := $(LFLAGS) -Wall -mcpu=cortex-m3 -mthumb -Wl,-Map=system.map -Tsystem.ld
|
||||
|
||||
CDEFS := $(CDEFS) -DATOMTHREADS_TEST='"$(TEST_NAME)"' -DPLATFORM_QEMU_LM3S_HACK
|
||||
ADEFS := $(ADEFS) -D__thumb2__ -DARM_RDI_MONITOR -DPLATFORM_QEMU_LM3S_HACK
|
||||
|
||||
LLIBS := $(LLIBS)
|
||||
|
||||
|
||||
SRCS := $(SRCS) \
|
||||
modules.c \
|
||||
main.c \
|
||||
$(ATOMTHREADS)/tests/$(TEST_NAME).c \
|
||||
|
||||
|
||||
ASMS := $(ASMS) \
|
||||
startup.S \
|
||||
|
||||
INCLUDES := $(INCLUDES) \
|
||||
-I$(ATOMTHREADS)
|
||||
|
||||
include $(ATOMTHREADS)/ports/cortex_m/Makefile
|
||||
|
||||
OBJS = $(SRCS:.c=.o) $(ASMS:.S=.o)
|
||||
|
||||
include ../rules.mk
|
||||
|
||||
|
||||
run_test: clean all
|
||||
echo "START TEST $(TEST_NAME)"
|
||||
qemu-system-arm -M lm3s6965evb -kernel boot.elf -semihosting >> atomthreads_test.out
|
||||
|
||||
all_tests:
|
||||
echo "Starting atomthreads test suite" > atomthreads_test.out
|
||||
make run_test "TEST_NAME=mutex1"
|
||||
make run_test "TEST_NAME=mutex2"
|
||||
make run_test "TEST_NAME=mutex3"
|
||||
make run_test "TEST_NAME=mutex5"
|
||||
make run_test "TEST_NAME=mutex6"
|
||||
make run_test "TEST_NAME=mutex7"
|
||||
make run_test "TEST_NAME=mutex8"
|
||||
make run_test "TEST_NAME=mutex9"
|
||||
make run_test "TEST_NAME=kern1"
|
||||
make run_test "TEST_NAME=kern2"
|
||||
make run_test "TEST_NAME=kern3"
|
||||
make run_test "TEST_NAME=kern4"
|
||||
make run_test "TEST_NAME=timer1"
|
||||
make run_test "TEST_NAME=timer2"
|
||||
make run_test "TEST_NAME=timer3"
|
||||
make run_test "TEST_NAME=timer5"
|
||||
make run_test "TEST_NAME=timer6"
|
||||
make run_test "TEST_NAME=timer7"
|
||||
make run_test "TEST_NAME=queue1"
|
||||
make run_test "TEST_NAME=queue2"
|
||||
make run_test "TEST_NAME=queue3"
|
||||
make run_test "TEST_NAME=queue4"
|
||||
make run_test "TEST_NAME=queue5"
|
||||
make run_test "TEST_NAME=queue6"
|
||||
make run_test "TEST_NAME=queue7"
|
||||
make run_test "TEST_NAME=queue8"
|
||||
make run_test "TEST_NAME=queue9"
|
||||
make run_test "TEST_NAME=sem1"
|
||||
make run_test "TEST_NAME=sem2"
|
||||
make run_test "TEST_NAME=sem3"
|
||||
make run_test "TEST_NAME=sem5"
|
||||
make run_test "TEST_NAME=sem6"
|
||||
make run_test "TEST_NAME=sem7"
|
||||
make run_test "TEST_NAME=sem8"
|
||||
make run_test "TEST_NAME=sem9"
|
||||
|
||||
fail_tests:
|
||||
make run_test "TEST_NAME=mutex4"
|
||||
make run_test "TEST_NAME=timer4"
|
||||
make run_test "TEST_NAME=sem4"
|
||||
|
||||
|
||||
run_last:
|
||||
qemu-system-arm -M lm3s6965evb -kernel boot.elf -semihosting
|
||||
|
||||
37
platforms/qemu_lm3s/README
Normal file
37
platforms/qemu_lm3s/README
Normal file
@@ -0,0 +1,37 @@
|
||||
---------------------------------------------------------------------------
|
||||
|
||||
Library: Atomthreads QEMU Stellaris LM3S6965 Platform.
|
||||
Author: Natie van Rooyen <natie@navaro.nl>
|
||||
License: BSD Revised
|
||||
|
||||
---------------------------------------------------------------------------
|
||||
|
||||
QEMU Stellaris LM3S6965 Platform
|
||||
|
||||
The "qemu_lm3s" platform contains sources for building a sample Atomthreads
|
||||
application for the Stellaris LM3S6965 platform.
|
||||
|
||||
|
||||
ISSUES:
|
||||
|
||||
There seems to be several problems for the QEMU Cortex M3 processor. The
|
||||
platform and port contains specific hacks to make it work on the QEMU 1.2.0
|
||||
release it was tested on. Also see the latest patches for QEMU.
|
||||
|
||||
Fixes implemented for the QEMU 1.2.0 release:
|
||||
|
||||
1. Install the patch http://patchwork.ozlabs.org/patch/180315/
|
||||
2. Use the PLATFORM_QEMU_LM3S_HACK define in the Makefile:
|
||||
|
||||
- Disabling interrupts on the processor does not work (verified).
|
||||
- Disabling interrupts of the Cortex M Sys Tick Interrupt does not
|
||||
work (verified).
|
||||
- NVIC Interrupt priorities not implemented correctly (not verified).
|
||||
|
||||
Because of the problems with the Sys Tick Interrupt the The Stellaris
|
||||
General-Purpose Timer Module (GPTM) was used to generate the system timer
|
||||
tick.
|
||||
|
||||
BUGS:
|
||||
mutex4 testcase fails.
|
||||
sem4 testcase fails.
|
||||
88
platforms/qemu_lm3s/main.c
Normal file
88
platforms/qemu_lm3s/main.c
Normal file
@@ -0,0 +1,88 @@
|
||||
/*
|
||||
* Copyright (c) 2012, Natie van Rooyen. 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 "modules.h"
|
||||
#include "atom.h"
|
||||
#include "tests/atomtests.h"
|
||||
|
||||
#ifndef ATOMTHREADS_TEST
|
||||
#define ATOMTHREADS_TEST "kern1"
|
||||
#endif
|
||||
|
||||
#define TEST_STACK_BYTE_SIZE 1024
|
||||
#define IDLE_STACK_BYTE_SIZE 512
|
||||
|
||||
static unsigned char test_stack[TEST_STACK_BYTE_SIZE] ;
|
||||
static unsigned char idle_stack[IDLE_STACK_BYTE_SIZE] ;
|
||||
ATOM_TCB test_tcb ;
|
||||
|
||||
|
||||
/**
|
||||
* \b test_thread
|
||||
*
|
||||
* Function calling the test function of the Atomthreads test suite.
|
||||
*
|
||||
*/
|
||||
void
|
||||
test_thread (uint32_t param)
|
||||
{
|
||||
uint32_t failures ;
|
||||
CRITICAL_STORE ;
|
||||
|
||||
failures = test_start () ;
|
||||
|
||||
atomTimerDelay (10) ;
|
||||
CRITICAL_START() ;
|
||||
printf ("%s %s\r\n", ATOMTHREADS_TEST, failures ? "FAIL" : "PASS") ;
|
||||
exit (failures) ;
|
||||
CRITICAL_END() ;
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* \b main
|
||||
*
|
||||
* Initialize atomthreads and start a test_thread to run the Atomthreads test suite.
|
||||
*
|
||||
*/
|
||||
int
|
||||
main (void)
|
||||
{
|
||||
int i = 0 ;
|
||||
|
||||
uint32_t failures ;
|
||||
printf ("Atomthreads starting %s... \r\n", ATOMTHREADS_TEST) ;
|
||||
|
||||
atomOSInit(&idle_stack[0], IDLE_STACK_BYTE_SIZE, TRUE) ;
|
||||
atomThreadCreate ((ATOM_TCB *)&test_tcb, TEST_THREAD_PRIO, test_thread, 0, &test_stack[0], TEST_STACK_BYTE_SIZE, TRUE);
|
||||
atomOSStart() ;
|
||||
|
||||
return 0 ;
|
||||
}
|
||||
|
||||
164
platforms/qemu_lm3s/modules.c
Normal file
164
platforms/qemu_lm3s/modules.c
Normal file
@@ -0,0 +1,164 @@
|
||||
/*
|
||||
* Copyright (c) 2012, Natie van Rooyen. 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 "modules.h"
|
||||
#include <stdio.h>
|
||||
#include <stdarg.h>
|
||||
#include "atomport_private.h"
|
||||
#include "atom.h"
|
||||
#include "atomport.h"
|
||||
#include "types.h"
|
||||
|
||||
SYSTICK_T* const board_systick = (SYSTICK_T*) BOARD_BASE_ADDRESS_SYSTICK ;
|
||||
NVIC_T* const board_nvic = (NVIC_T*) BOARD_BASE_ADDRESS_NVIC ;
|
||||
SCB_T * const board_scb = (SCB_T*) BOARD_BASE_ADDRESS_SCB ;
|
||||
GPTM_TIMER_T * const board_gptm0 = (GPTM_TIMER_T*) BOARD_BASE_ADDRESS_GPTIMER0 ;
|
||||
|
||||
|
||||
/**
|
||||
* \b dbg_format_msg
|
||||
*
|
||||
* Same as printf.
|
||||
*
|
||||
*/
|
||||
void
|
||||
dbg_format_msg (char *format, ...)
|
||||
{
|
||||
va_list args;
|
||||
static char msg[256] ;
|
||||
CRITICAL_STORE ;
|
||||
|
||||
va_start (args, format) ;
|
||||
CRITICAL_START() ;
|
||||
vsnprintf ((char*)msg, 256, (char*)format, args) ;
|
||||
printf (msg) ;
|
||||
CRITICAL_END() ;
|
||||
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* \b low_level_init
|
||||
*
|
||||
* Initializes the PIC and start the system timer tick intrerupt.
|
||||
*
|
||||
*/
|
||||
int
|
||||
low_level_init (void)
|
||||
{
|
||||
contextInit () ;
|
||||
|
||||
//board_systick->STRELOAD = 0x010000 ;
|
||||
//board_systick->STCTRL = NVIC_STCTRL_CLK |
|
||||
// NVIC_STCTRL_INTEN |
|
||||
// NVIC_STCTRL_ENABLE ;
|
||||
|
||||
board_gptm0->CTL &= ~GPTM_TIMER_CTL_TAEN ;
|
||||
board_gptm0->CFG = 0 ;
|
||||
board_gptm0->TAMR = GPTM_TIMER_TMR_TMR_PERIODIC ;
|
||||
board_gptm0->TAILR = 0x10000 ;
|
||||
board_gptm0->IMR |= GPTM_TIMER_INT_TATOIM ;
|
||||
board_gptm0->CTL |= GPTM_TIMER_CTL_TAEN ;
|
||||
|
||||
// board_nvic->ISER[0] = 0x80000 ;
|
||||
|
||||
return 0 ;
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* \b __context_tick_handler
|
||||
*
|
||||
* System timer tic interupt handler.
|
||||
*
|
||||
*/
|
||||
void
|
||||
__context_tick_handler (void)
|
||||
{
|
||||
|
||||
atomIntEnter();
|
||||
|
||||
/* Call the OS system tick handler */
|
||||
atomTimerTick();
|
||||
|
||||
board_gptm0->ICR |= GPTM_TIMER_INT_TATOIM ;
|
||||
|
||||
/* Call the interrupt exit routine */
|
||||
atomIntExit(TRUE);
|
||||
|
||||
}
|
||||
|
||||
/**
|
||||
* \b dbg_hard_fault_handler_c
|
||||
*
|
||||
* Dumps the registers pushed on the stack after a fault.
|
||||
*
|
||||
*/
|
||||
void
|
||||
dbg_hard_fault_handler_c (unsigned int * hardfault_args)
|
||||
{
|
||||
unsigned int stacked_r0;
|
||||
unsigned int stacked_r1;
|
||||
unsigned int stacked_r2;
|
||||
unsigned int stacked_r3;
|
||||
unsigned int stacked_r12;
|
||||
unsigned int stacked_lr;
|
||||
unsigned int stacked_pc;
|
||||
unsigned int stacked_psr;
|
||||
|
||||
stacked_r0 = ((unsigned long) hardfault_args[0]);
|
||||
stacked_r1 = ((unsigned long) hardfault_args[1]);
|
||||
stacked_r2 = ((unsigned long) hardfault_args[2]);
|
||||
stacked_r3 = ((unsigned long) hardfault_args[3]);
|
||||
|
||||
stacked_r12 = ((unsigned long) hardfault_args[4]);
|
||||
stacked_lr = ((unsigned long) hardfault_args[5]);
|
||||
stacked_pc = ((unsigned long) hardfault_args[6]);
|
||||
stacked_psr = ((unsigned long) hardfault_args[7]);
|
||||
|
||||
printf ("\r\n\r\n[Hard fault handler - all numbers in hex]\r\n");
|
||||
printf ("SP = 0x%x\r\n", hardfault_args);
|
||||
printf ("R0 = 0x%x\r\n", stacked_r0);
|
||||
printf ("R1 = 0x%x\r\n", stacked_r1);
|
||||
printf ("R2 = 0x%x\r\n", stacked_r2);
|
||||
printf ("R3 = 0x%x\r\n", stacked_r3);
|
||||
printf ("R12 = 0x%x\r\n", stacked_r12);
|
||||
printf ("LR [R14] = 0x%x subroutine call return address\r\n", stacked_lr);
|
||||
printf ("PC [R15] = 0x%x program counter\r\n", stacked_pc);
|
||||
printf ("PSR = 0x%x\r\n", stacked_psr);
|
||||
//printf ("BFAR = 0x%x\r\n", (*((volatile unsigned long *)(0xE000ED38))));
|
||||
//printf ("CFSR = 0x%x\r\n", (*((volatile unsigned long *)(0xE000ED28))));
|
||||
//printf ("HFSR = 0x%x\r\n", (*((volatile unsigned long *)(0xE000ED2C))));
|
||||
//printf ("DFSR = 0x%x\r\n", (*((volatile unsigned long *)(0xE000ED30))));
|
||||
//printf ("AFSR = 0x%x\r\n", (*((volatile unsigned long *)(0xE000ED3C))));
|
||||
// printf ("SCB_SHCSR = %x\n", SCB->SHCSR);
|
||||
|
||||
while (1);
|
||||
|
||||
}
|
||||
|
||||
208
platforms/qemu_lm3s/modules.h
Normal file
208
platforms/qemu_lm3s/modules.h
Normal file
@@ -0,0 +1,208 @@
|
||||
/*
|
||||
* Copyright (c) 2012, Natie van Rooyen. 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 __MODULES_H__
|
||||
#define __MODULES_H__
|
||||
|
||||
/*
|
||||
* Module definitions to use with the Stellaris LM3S6965 Microcontroller
|
||||
*/
|
||||
|
||||
#include "atomport.h"
|
||||
|
||||
|
||||
// *****************************************************************************
|
||||
// The Stellaris General-Purpose Timer Module (GPTM)
|
||||
// *****************************************************************************
|
||||
typedef struct GPTM_TIMER_S {
|
||||
|
||||
// offset read/write reset Description
|
||||
__IO uint32_t CFG ; // 0x000 R/W 0x00000000 GPTM Configuration 345
|
||||
__IO uint32_t TAMR ; // 0x004 R/W 0x00000000 GPTM TimerA Mode 346
|
||||
__IO uint32_t TBMR ; // 0x008 R/W 0x00000000 GPTM TimerB Mode 348
|
||||
__IO uint32_t CTL ; // 0x00C R/W 0x00000000 GPTM Control 350
|
||||
uint32_t Reserved[2] ; // 0x010
|
||||
__IO uint32_t IMR ; // 0x018 R/W 0x00000000 GPTM Interrupt Mask 353
|
||||
__I uint32_t RIS ; // 0x01C RO 0x00000000 GPTM Raw Interrupt Status 355
|
||||
__I uint32_t MIS ; // 0x020 RO 0x00000000 GPTM Masked Interrupt Status 356
|
||||
__O uint32_t ICR ; // 0x024 W1C 0x00000000 GPTM Interrupt Clear 357
|
||||
__IO uint32_t TAILR ; // 0x028 R/W 0xFFFFFFFF GPTM TimerA Interval Load 359
|
||||
__IO uint32_t TBILR ; // 0x02C R/W 0x0000FFFF GPTM TimerB Interval Load 360
|
||||
__IO uint32_t TAMATCHR ; // 0x030 R/W 0xFFFFFFFF GPTM TimerA Match 361
|
||||
__IO uint32_t TBMATCHR ; // 0x034 R/W 0x0000FFFF GPTM TimerB Match 362
|
||||
__IO uint32_t TAPR ; // 0x038 R/W 0x00000000 GPTM TimerA Prescale 363
|
||||
__IO uint32_t TBPR ; // 0x03C R/W 0x00000000 GPTM TimerB Prescale 364
|
||||
__IO uint32_t TAPMR ; // 0x040 R/W 0x00000000 GPTM TimerA Prescale Match 365
|
||||
__IO uint32_t TBPMR ; // 0x044 R/W 0x00000000 GPTM TimerB Prescale Match 366
|
||||
__I uint32_t TAR ; // 0x048 RO 0xFFFFFFFF GPTM TimerA 367
|
||||
__I uint32_t TBR ; // 0x04C RO 0x0000FFFF GPTM TimerB 368
|
||||
|
||||
} GPTM_TIMER_T, *PGPTM_TIMER_T ;
|
||||
|
||||
// -------- GPTM_TIMER_CFG : (CFG Offset: 0x00) This register configures the global operation of the GPTM module --------
|
||||
#define GPTM_TIMER_CFG_MASK ((unsigned int)0x07 << 0) //
|
||||
#define GPTM_TIMER_CFG_32BIT ((unsigned int)0x00 << 0) // 32-bit timer configuration
|
||||
#define GPTM_TIMER_CFG_32BIT_RT ((unsigned int)0x01 << 0) // 32-bit real-time clock (RTC) counter configuration
|
||||
// -------- GPTM_TIMER_TAMR : (TAMR Offset: 0x04) This register configures the GPTM based on the configuration selected in the GPTMCFG register --------
|
||||
// -------- GPTM_TIMER_TBMR : (TBMR Offset: 0x08) This register configures the GPTM based on the configuration selected in the GPTMCFG register --------
|
||||
#define GPTM_TIMER_TMR_TAMS ((unsigned int)0x01 << 3) // GPTM TimerA Alternate Mode Select. 0 Capture mode is enabled. 1 PWM mode is enabled
|
||||
#define GPTM_TIMER_TMR_TCMR ((unsigned int)0x01 << 2) // GPTM TimerA Capture Mode. 0 Edge-Count mode. 1 Edge-Time mode.
|
||||
#define GPTM_TIMER_TMR_TMR_ONE_SHOT ((unsigned int)0x01 << 0) // One-Shot Timer mode
|
||||
#define GPTM_TIMER_TMR_TMR_PERIODIC ((unsigned int)0x02 << 0) // Periodic Timer mode
|
||||
#define GPTM_TIMER_TMR_TMR_CAPTURE ((unsigned int)0x03 << 0) // Capture mode
|
||||
// -------- GPTM_TIMER_CTL : (CTL Offset: 0x0C) This register is used alongside the GPTMCFG and GMTMTnMR registers to fine-tune the timer configuration --------
|
||||
#define GPTM_TIMER_CTL_TBPWML ((unsigned int)0x01 << 14) // GPTM TimerB PWM Output Level. 0 Output is unaffected. 1 Output is inverted.
|
||||
#define GPTM_TIMER_CTL_TBOTE ((unsigned int)0x01 << 13) // GPTM TimerB Output Trigger Enable. 0 The output TimerB ADC trigger is disabled. 1 The output TimerB ADC trigger is enabled.
|
||||
#define GPTM_TIMER_CTL_TBEVENT_MASK ((unsigned int)0x03 << 10) // GPTM TimerB Event Mode
|
||||
#define GPTM_TIMER_CTL_TBEVENT_PE ((unsigned int)0x00 << 10) // Positive edge
|
||||
#define GPTM_TIMER_CTL_TBEVENT_NE ((unsigned int)0x01 << 10) // Negative edge
|
||||
#define GPTM_TIMER_CTL_TBEVENT ((unsigned int)0x03 << 10) // Both edges
|
||||
#define GPTM_TIMER_CTL_TBSTALL ((unsigned int)0x01 << 9) // GPTM Timer B Stall Enable. 0 Timer B continues counting while the processor is halted by the debugger
|
||||
#define GPTM_TIMER_CTL_TBEN ((unsigned int)0x01 << 8) // GPTM TimerB Enable
|
||||
// --------
|
||||
#define GPTM_TIMER_CTL_TAPWML ((unsigned int)0x01 << 6) // GPTM TimerA PWM Output Level. 0 Output is unaffected. 1 Output is inverted.
|
||||
#define GPTM_TIMER_CTL_TAOTE ((unsigned int)0x01 << 5) // GPTM TimerA Output Trigger Enable. 0 The output TimerB ADC trigger is disabled. 1 The output TimerB ADC trigger is enabled.
|
||||
#define GPTM_TIMER_CTL_RTCEN ((unsigned int)0x01 << 4) // GPTM RTC Enable
|
||||
#define GPTM_TIMER_CTL_TAEVENT_MASK ((unsigned int)0x03 << 2) // GPTM TimerA Event Mode
|
||||
#define GPTM_TIMER_CTL_TAEVENT_PE ((unsigned int)0x00 << 2) // Positive edge
|
||||
#define GPTM_TIMER_CTL_TAEVENT_NE ((unsigned int)0x01 << 2) // Negative edge
|
||||
#define GPTM_TIMER_CTL_TAEVENT ((unsigned int)0x03 << 2) // Both edges
|
||||
#define GPTM_TIMER_CTL_TASTALL ((unsigned int)0x01 << 1) // GPTM Timer A Stall Enable. 0 Timer B continues counting while the processor is halted by the debugger
|
||||
#define GPTM_TIMER_CTL_TAEN ((unsigned int)0x01 << 0) // GPTM TimerA Enable
|
||||
// -------- GPTM_TIMER_IMR : (IMR Offset: 0x18) This register allows software to enable/disable GPTM controller-level interrupts. --------
|
||||
// -------- GPTM_TIMER_RIS : (RIS Offset: 0x1C) This register shows the state of the GPTM's internal interrupt signal. --------
|
||||
// -------- GPTM_TIMER_MIS : (MIS Offset: 0x20) This register show the state of the GPTM's controller-level interrupt. --------
|
||||
// -------- GPTM_TIMER_ICR : (ICR Offset: 0x24) This register is used to clear the status bits in the GPTMRIS and GPTMMIS registers. --------
|
||||
#define GPTM_TIMER_INT_CBEIM ((unsigned int)0x01 << 10) // GPTM CaptureB Event Interrupt Mask
|
||||
#define GPTM_TIMER_INT_CBMIM ((unsigned int)0x01 << 9) // GPTM CaptureB Match Interrupt Mask
|
||||
#define GPTM_TIMER_INT_TBTOIM ((unsigned int)0x01 << 8) // GPTM TimerB Time-Out Interrupt Mask
|
||||
// --------
|
||||
#define GPTM_TIMER_INT_RTCIM ((unsigned int)0x01 << 3) // GPTM RTC Interrupt Mask
|
||||
#define GPTM_TIMER_INT_CAEIM ((unsigned int)0x01 << 2) // GPTM CaptureA Event Interrupt Mask
|
||||
#define GPTM_TIMER_INT_CAMIM ((unsigned int)0x01 << 1) // GPTM CaptureA Match Interrupt Mask
|
||||
#define GPTM_TIMER_INT_TATOIM ((unsigned int)0x01 << 0) // GPTM TimerA Time-Out Interrupt Mask
|
||||
|
||||
|
||||
|
||||
// *****************************************************************************
|
||||
// Cortex M System Timer (SysTick)
|
||||
// *****************************************************************************
|
||||
typedef struct SYSTICK_S {
|
||||
|
||||
uint32_t Res0[1] ; // 0xE000E000
|
||||
__IO uint32_t ICT ; // 0xE000E004
|
||||
uint32_t Res1[2] ; // 0xE000E008
|
||||
__IO uint32_t STCTRL ; // 0xE000E010
|
||||
__IO uint32_t STRELOAD ; // 0xE000E014
|
||||
__IO uint32_t STCURRENT; // 0xE000E018
|
||||
__IO uint32_t STCALIB ; // 0xE000E01C
|
||||
uint32_t Res2[56] ; // 0xE000E020
|
||||
|
||||
} SYSTICK_T, *PSYSTICK_T ;
|
||||
|
||||
// -------- SYSTICK_STCTRL : (STCTRL Offset: 0xE000E010) SysTick Control and Status Register --------
|
||||
#define SYSTICK_STCTRL_COUNT ((unsigned int)0x1 << 16) // 0 - The SysTick timer has not counted to 0 since the last time this bit was read.
|
||||
#define SYSTICK_STCTRL_CLK ((unsigned int)0x1 << 2) // 1 - System clock
|
||||
#define SYSTICK_STCTRL_INTEN ((unsigned int)0x1 << 1) // 1 - An interrupt is generated to the NVIC when SysTick counts to 0.
|
||||
#define SYSTICK_STCTRL_ENABLE ((unsigned int)0x1 << 1) // Enables SysTick to operate in a multi-shot way.
|
||||
// -------- SYSTICK_STRELOAD : (STRELOAD Offset: 0xE000E014) Reload Value --------
|
||||
#define SYSTICK_STRELOAD_MASK ((unsigned int)0xFFFFFF << 0) // IRQ mask
|
||||
// -------- SYSTICK_STCURRENT : (STCURRENT Offset: 0xE000E018) SysTick Current Value Register --------
|
||||
|
||||
|
||||
// *****************************************************************************
|
||||
// Cortex M Nested Vectored Interrupt Controller
|
||||
// *****************************************************************************
|
||||
typedef struct NVIC_S {
|
||||
|
||||
__IO uint32_t ISER[2] ; // 0xE000E100
|
||||
uint32_t Res3[30] ; // 0xE000E120
|
||||
__IO uint32_t ICER[2] ; // 0xE000E180
|
||||
uint32_t Res4[30] ; // 0xE000E1A0
|
||||
__IO uint32_t ISPR[2] ; // 0xE000E200
|
||||
uint32_t Res5[30] ; // 0xE000E220
|
||||
__IO uint32_t ICPR[2] ; // 0xE000E280
|
||||
uint32_t Res6[30] ; // 0xE000E2A0
|
||||
__IO uint32_t IABR[2] ; // 0xE000E300
|
||||
uint32_t Res7[64] ; // 0xE000E320
|
||||
__IO uint32_t IPR[2] ; // 0xE000E400
|
||||
// uint32_t Res7[515] ; // 0xE000E4F4
|
||||
|
||||
} NVIC_T, *PNVIC_T ;
|
||||
|
||||
#define NVIC_EXCEPTION_RESET 1
|
||||
#define NVIC_EXCEPTION_NMI 2
|
||||
#define NVIC_EXCEPTION_HARD_FAULT 3
|
||||
#define NVIC_EXCEPTION_MEM_MANAGEMENT 4
|
||||
#define NVIC_EXCEPTION_BUS_FAULT 5
|
||||
#define NVIC_EXCEPTION_USAGE_FAULT 6
|
||||
#define NVIC_EXCEPTION_SVCALL 11
|
||||
#define NVIC_EXCEPTION_DEBUG_MON 12
|
||||
#define NVIC_EXCEPTION_PEND_SV 14
|
||||
#define NVIC_EXCEPTION_SYS_TICK 15
|
||||
|
||||
// *****************************************************************************
|
||||
// System Control Block (SCB) Registers
|
||||
// *****************************************************************************
|
||||
typedef struct SCB_S {
|
||||
|
||||
__IO uint32_t CPUID ; // 0xE000ED00
|
||||
__IO uint32_t ICSR ; // 0xE000ED04
|
||||
__IO uint32_t VTOR ; // 0xE000ED08
|
||||
__IO uint32_t AIRCR ; // 0xE000ED0C
|
||||
__IO uint32_t SCR ; // 0xE000ED10
|
||||
__IO uint32_t CCR ; // 0xE000ED14
|
||||
|
||||
__IO uint32_t SYS_PRIO[3] ; // 0xE000ED18
|
||||
__IO uint32_t SYSHNDCTRL ; // 0xE000ED24
|
||||
//__IO uint32_t FAULTSTAT ; // 0xE000ED28
|
||||
//__IO uint32_t HFAULTSTAT ; // 0xE000ED2C
|
||||
|
||||
} SCB_T, *PSCB_T ;
|
||||
|
||||
|
||||
/* module definitions */
|
||||
#define BOARD_BASE_ADDRESS_SYSTICK 0xE000E000
|
||||
#define BOARD_BASE_ADDRESS_NVIC 0xE000E100
|
||||
#define BOARD_BASE_ADDRESS_SCB 0xE000ED00
|
||||
#define BOARD_BASE_ADDRESS_GPTIMER0 0x40030000
|
||||
|
||||
extern SYSTICK_T* const board_systick ;
|
||||
extern NVIC_T* const board_nvic ;
|
||||
extern SCB_T* const board_scb ;
|
||||
extern GPTM_TIMER_T* const board_gptm0 ;
|
||||
|
||||
|
||||
/* Function prototypes */
|
||||
extern int low_level_init (void) ;
|
||||
extern void dbg_format_msg (char *format, ...) ;
|
||||
extern void dbg_hard_fault_handler_c (unsigned int * hardfault_args) ;
|
||||
|
||||
#define DBG_MESSAGE(fmt_str) { dbg_format_msg fmt_str ; }
|
||||
|
||||
#endif /* __MODULES_H__ */
|
||||
215
platforms/qemu_lm3s/startup.S
Normal file
215
platforms/qemu_lm3s/startup.S
Normal file
@@ -0,0 +1,215 @@
|
||||
/*
|
||||
Copyright (c) 2012, Natie van Rooyen. 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.
|
||||
*/
|
||||
|
||||
/* lm3s Startup Script */
|
||||
|
||||
.section .vectors,"x",%progbits
|
||||
.syntax unified
|
||||
.thumb
|
||||
|
||||
|
||||
.global __interrupt_vector_table
|
||||
.extern archTickHandler
|
||||
.extern archPendSVHandler
|
||||
.extern dbg_hard_fault_handler_c
|
||||
|
||||
/**
|
||||
* \b __interrupt_vector_table
|
||||
*
|
||||
*/
|
||||
__interrupt_vector_table:
|
||||
.long __c_stack_top__
|
||||
.long reset_Handler
|
||||
.long fault_Handler
|
||||
.long fault_Handler
|
||||
.long fault_Handler
|
||||
.long fault_Handler
|
||||
.long fault_Handler
|
||||
.long 0
|
||||
.long 0
|
||||
.long 0
|
||||
.long 0
|
||||
.long sys_Handler
|
||||
.long sys_Handler
|
||||
.long 0
|
||||
.long archPendSVHandler
|
||||
.long sys_Handler
|
||||
|
||||
/* External interrupts */
|
||||
.long default_Handler // GPIO Port A
|
||||
.long default_Handler // GPIO Port B
|
||||
.long default_Handler // GPIO Port C
|
||||
.long default_Handler // GPIO Port D
|
||||
.long default_Handler // GPIO Port E
|
||||
.long default_Handler // UART0 Rx and Tx
|
||||
.long default_Handler // UART1 Rx and Tx
|
||||
.long default_Handler // SSI0 Rx and Tx
|
||||
.long default_Handler // I2C0 Master and Slave
|
||||
.long default_Handler // PWM Fault
|
||||
.long default_Handler // PWM Generator 0
|
||||
.long default_Handler // PWM Generator 1
|
||||
.long default_Handler // PWM Generator 2
|
||||
.long default_Handler // Quadrature Encoder 0
|
||||
.long default_Handler // ADC Sequence 0
|
||||
.long default_Handler // ADC Sequence 1
|
||||
.long default_Handler // ADC Sequence 2
|
||||
.long default_Handler // ADC Sequence 3
|
||||
.long default_Handler // Watchdog timer
|
||||
.long archTickHandler // Timer 0 subtimer A
|
||||
.long default_Handler // Timer 0 subtimer B
|
||||
.long default_Handler // Timer 1 subtimer A
|
||||
.long default_Handler // Timer 1 subtimer B
|
||||
.long default_Handler // Timer 2 subtimer A
|
||||
.long default_Handler // Timer 2 subtimer B
|
||||
.long default_Handler // Analog Comparator 0
|
||||
.long default_Handler // Analog Comparator 1
|
||||
.long default_Handler // Analog Comparator 2
|
||||
.long default_Handler // System Control (PLL, OSC, BO)
|
||||
.long default_Handler // FLASH Control
|
||||
.long default_Handler // GPIO Port F
|
||||
.long default_Handler // GPIO Port G
|
||||
.long default_Handler // GPIO Port H
|
||||
.long default_Handler // UART2 Rx and Tx
|
||||
.long default_Handler // SSI1 Rx and Tx
|
||||
.long default_Handler // Timer 3 subtimer A
|
||||
.long default_Handler // Timer 3 subtimer B
|
||||
.long default_Handler // I2C1 Master and Slave
|
||||
.long default_Handler // Quadrature Encoder 1
|
||||
.long default_Handler // CAN0
|
||||
.long default_Handler // CAN1
|
||||
.long default_Handler // CAN2
|
||||
.long default_Handler // Ethernet
|
||||
.long default_Handler // Hibernate
|
||||
|
||||
/**
|
||||
* \b sys_Handler
|
||||
*
|
||||
* @return None
|
||||
*/
|
||||
.thumb_func
|
||||
sys_Handler:
|
||||
B .
|
||||
|
||||
/**
|
||||
* \b default_Handler
|
||||
*
|
||||
* @return None
|
||||
*/
|
||||
.thumb_func
|
||||
default_Handler:
|
||||
B .
|
||||
|
||||
/**
|
||||
* \b fault_Handler
|
||||
*
|
||||
* @return None
|
||||
*/
|
||||
.thumb_func
|
||||
fault_Handler:
|
||||
tst lr, #4
|
||||
ite eq
|
||||
mrseq r0, MSP
|
||||
mrsne r0, PSP
|
||||
b dbg_hard_fault_handler_c
|
||||
|
||||
|
||||
.section .startup,"x",%progbits
|
||||
.syntax unified
|
||||
.thumb
|
||||
|
||||
.global reset_Handler
|
||||
.extern initialise_monitor_handles
|
||||
.extern low_level_init
|
||||
.extern main
|
||||
|
||||
|
||||
/**
|
||||
* \b reset_Handler
|
||||
*
|
||||
*
|
||||
*
|
||||
* @return None
|
||||
*/
|
||||
.thumb_func
|
||||
reset_Handler:
|
||||
|
||||
/*
|
||||
* Initialize the data and bss sections.
|
||||
*/
|
||||
init_data:
|
||||
ldr r0, .ETEXT
|
||||
ldr r1, .DATA
|
||||
ldr r2, .EDATA
|
||||
sub r2, r2, r1
|
||||
cmp r2, #0
|
||||
beq init_bss
|
||||
init_data_copy:
|
||||
ldrb r4, [r0], #1
|
||||
strb r4, [r1], #1
|
||||
subs r2, r2, #1
|
||||
bne init_data_copy
|
||||
init_bss:
|
||||
mov r0, #0
|
||||
ldr r1, = .BSS
|
||||
ldr r2, = .EBSS
|
||||
sub r2, r2, r1
|
||||
cmp r2, #0
|
||||
beq init_done
|
||||
init_bss_zero:
|
||||
strb r0, [r1], #1
|
||||
subs r2, r2, #1
|
||||
bne init_bss_zero
|
||||
init_done:
|
||||
|
||||
/*
|
||||
* The following call initializes the function pointers for stdio etc.
|
||||
* These are used by the semihosting interface.
|
||||
*
|
||||
* This function is implemented in newlib.
|
||||
*/
|
||||
bl initialise_monitor_handles
|
||||
|
||||
/*
|
||||
* Platform specific low level initialization.
|
||||
*/
|
||||
bl low_level_init
|
||||
|
||||
/*
|
||||
* Call the application's entry point.
|
||||
*/
|
||||
bl main
|
||||
|
||||
|
||||
.BSS: .long _bss
|
||||
.EBSS: .long _ebss
|
||||
.ETEXT: .long _etext
|
||||
.DATA: .long _data
|
||||
.EDATA: .long _edata
|
||||
|
||||
.end
|
||||
72
platforms/qemu_lm3s/system.ld
Normal file
72
platforms/qemu_lm3s/system.ld
Normal file
@@ -0,0 +1,72 @@
|
||||
|
||||
|
||||
MEMORY
|
||||
{
|
||||
FLASH (rx) : ORIGIN = 0x00000000, LENGTH = 0x00040000
|
||||
SRAM (rwx) : ORIGIN = 0x20000000, LENGTH = 0x00010000
|
||||
}
|
||||
|
||||
SECTIONS
|
||||
{
|
||||
_vRamTop = 0x20000000 + 0x00010000;
|
||||
.text :
|
||||
{
|
||||
KEEP(*(.vectors))
|
||||
*(.startup)
|
||||
*(.text*)
|
||||
*(.rodata*)
|
||||
} > FLASH
|
||||
|
||||
/*
|
||||
* for exception handling/unwind - some Newlib functions (in common with
|
||||
* C++ and STDC++) use this.
|
||||
*/
|
||||
|
||||
.ARM.extab :
|
||||
{
|
||||
*(.ARM.extab* .gnu.linkonce.armextab.*)
|
||||
} > FLASH
|
||||
|
||||
__exidx_start = .;
|
||||
.ARM.exidx :
|
||||
{
|
||||
*(.ARM.exidx* .gnu.linkonce.armexidx.*)
|
||||
} > FLASH
|
||||
__exidx_end = .;
|
||||
|
||||
_etext = .;
|
||||
|
||||
.data : AT (__exidx_end)
|
||||
{
|
||||
_data = .;
|
||||
*(vtable)
|
||||
*(.data*)
|
||||
_edata = .;
|
||||
} > SRAM
|
||||
|
||||
/* zero initialized data */
|
||||
|
||||
.bss :
|
||||
{
|
||||
__bss_start__ = . ;
|
||||
_bss = .;
|
||||
*(.bss*)
|
||||
*(COMMON)
|
||||
__bss_end__ = . ;
|
||||
_ebss = .;
|
||||
} > SRAM
|
||||
|
||||
/* Where we put the heap with cr_clib */
|
||||
|
||||
.cr_heap :
|
||||
{
|
||||
end = .;
|
||||
_pvHeapStart = .;
|
||||
} > SRAM
|
||||
|
||||
_vStackTop = _vRamTop - 16;
|
||||
.stack _vStackTop :
|
||||
{
|
||||
__c_stack_top__ = . ;
|
||||
}
|
||||
}
|
||||
62
platforms/rules.mk
Normal file
62
platforms/rules.mk
Normal file
@@ -0,0 +1,62 @@
|
||||
#
|
||||
# The following part of the makefile is generic; it can be used to
|
||||
# build any executable just by changing the definitions above and by
|
||||
# deleting dependencies appended to the file from 'make depend'
|
||||
#
|
||||
.SUFFIXES: .asm .elf .hex .lst .o .S .s .c .cpp
|
||||
.PHONY: depend clean
|
||||
|
||||
dump:
|
||||
@echo "Target: "
|
||||
@echo $(TARGET_NAME)
|
||||
@echo "Source files: "
|
||||
@echo $(SRCS)
|
||||
@echo $(ASMS)
|
||||
@echo "Object files: "
|
||||
@echo $(OBJS)
|
||||
|
||||
all: target
|
||||
|
||||
target: $(OBJS)
|
||||
$(LN) $(LFLAGS) $(LIBFLAGS) $(OBJS) $(LLIBS) -o $(TARGET_NAME).elf
|
||||
@echo $(TARGET_NAME).elf was compiled
|
||||
arm-none-eabi-objcopy -O binary $(TARGET_NAME).elf $(TARGET_NAME).bin
|
||||
arm-none-eabi-objdump -dxS $(TARGET_NAME).elf > $(TARGET_NAME).out
|
||||
|
||||
clean:
|
||||
rm -f $(OBJS)
|
||||
|
||||
|
||||
# this is a suffix replacement rule for building .o's from .c's
|
||||
# it uses automatic variables $<: the name of the prerequisite of
|
||||
# the rule(a .c file) and $@: the name of the target of the rule (a .o file)
|
||||
# (see the gnu make manual section about automatic variables)
|
||||
|
||||
.c.o:
|
||||
$(CC) $(CDEFS) $(CFLAGS) $(INCLUDES) -c $< -o $@
|
||||
|
||||
.cpp.o:
|
||||
$(CC) $(CDEFS) $(CFLAGS) $(INCLUDES) -c $< -o $@
|
||||
|
||||
.S.o:
|
||||
$(AS) $(ADEFS) $(AFLAGS) $(INCLUDES) -c $< -o $@
|
||||
|
||||
.s.o:
|
||||
$(AS) $(ADEFS) $(AFLAGS) $(INCLUDES) -c $< -o $@
|
||||
|
||||
DEPFILE=.depends
|
||||
DEPTOKEN='\# MAKEDEPENDS'
|
||||
DEPFLAGS=-Y -f $(DEPFILE) -s $(DEPTOKEN) -p $(OUTDIR)/
|
||||
|
||||
|
||||
depend:
|
||||
rm -f $(DEPFILE)
|
||||
make $(DEPFILE)
|
||||
|
||||
$(DEPFILE):
|
||||
@echo $(DEPTOKEN) > $(DEPFILE)
|
||||
makedepend $(DEPFLAGS) -- $(CFLAGS) -- $(SRCS) >&/dev/null
|
||||
|
||||
# put this file in the last line of your Makefile
|
||||
sinclude $(DEPFILE)
|
||||
|
||||
1161
ports/arm/Doxyfile
Normal file
1161
ports/arm/Doxyfile
Normal file
File diff suppressed because it is too large
Load Diff
107
ports/arm/README
Normal file
107
ports/arm/README
Normal file
@@ -0,0 +1,107 @@
|
||||
---------------------------------------------------------------------------
|
||||
|
||||
Library: Atomthreads ARM Port
|
||||
Author: Natie van Rooyen <natie@navaro.nl>
|
||||
License: BSD Revised
|
||||
|
||||
---------------------------------------------------------------------------
|
||||
|
||||
ARM PORT
|
||||
|
||||
This folder contains a port of the Atomthreads real time kernel for the
|
||||
ARM processor architecture. This port was tested on the ARMv5 and ARMv7
|
||||
architectures.
|
||||
|
||||
The same common/core port can be used on a wide variety of ARM devices
|
||||
but platform-specific code has been separated out into multiple "platform"
|
||||
(BSP) folders. This allows the common ARM code to be shared among several
|
||||
different ARM platforms and boards. For example, different ARM devices and
|
||||
platforms might use different interrupt controllers, timer subsystems and
|
||||
UARTs but share the same core OS context-switching routines etc.
|
||||
|
||||
An example platform is in the "platforms/qemu_integratorcp" folder. The
|
||||
platform-specific folders such as this contain the Makefile to build the
|
||||
project for that platform, so you may wish to head straight there if you
|
||||
wish to quickly get started building and running Atomthreads on ARM. The
|
||||
qemu_integratorcp platform is designed for the Integrator/CP platform with
|
||||
ARM926EJ-S processor, and can be run within QEMU for quick evaluation of
|
||||
Atomthreads without real hardware.
|
||||
|
||||
|
||||
---------------------------------------------------------------------------
|
||||
|
||||
FILES IN THE COMMON ARM PORT FOLDER
|
||||
|
||||
* tests-main.c: Contains a sample Atomthreads application starting at
|
||||
main() that initialises the operating system and runs the automated test
|
||||
suite applications. You will normally make your own main() function
|
||||
suitable for your application, possibly using this as a basis.
|
||||
* atomport-asm.s: Contains the main assembler code that forms the portion
|
||||
of the core ARM architecture port that must be implemented in assembler
|
||||
(e.g. register save/restore for thread context-switching).
|
||||
* atomport.c: Contains the main C code that forms the portion of the core
|
||||
ARM architecture port that can be implemented in C (e.g. prefilling the
|
||||
stack context for new threads).
|
||||
* syscalls.c: Contains the open/close/read/write/heap-management
|
||||
functions typically required if you want to do anything with stdio
|
||||
(e.g. printf() calls etc). This is a very simple implementation that
|
||||
always writes to the UART regardless of which file is "opened". Use of
|
||||
printf() and friends with GCC toolchains typically requires a heap, and
|
||||
thus a heap is supported in this file via the _sbrk() function. Your
|
||||
linker script should specify where the heap starts and stops using "end"
|
||||
and "heap_top" definitions respectively. Note that in QEMU environments
|
||||
this may not be required as some "semi-hosted" toolchains implement
|
||||
these functions and the UART driver for you. In that case these
|
||||
functions will be left out of the build because they are defined with
|
||||
weak linkage.
|
||||
|
||||
|
||||
---------------------------------------------------------------------------
|
||||
|
||||
PORTING TO NEW ARM PLATFORMS
|
||||
|
||||
To port Atomthreads to your ARM platform you must provide the following
|
||||
functionality in your platform folder (in all cases example filenames are
|
||||
from the qemu_integratorcp sample platform):
|
||||
|
||||
* Startup code: see Reset_Handler in startup.s. Typically this will be (at
|
||||
least in the first few instructions) some assembly code, and will set up
|
||||
the initial stack pointer, perform any copying of segments from flash to
|
||||
RAM, zero the BSS section etc, before branching to the main application
|
||||
function (e.g. main()). At some point during initialisation the timer
|
||||
tick interrupt required by the OS should be started (100 ticks per
|
||||
second) and this might be done here in the very early startup code. Note
|
||||
that some compiler toolchains will provide a portion of the C startup
|
||||
code e.g. the function _mainCRTStartup() provided by some GCC
|
||||
toolchains.
|
||||
* Interrupt vector table: see __interrupt_vector_table in startup.s.
|
||||
Typically this will contain at least an entry for the startup/reset
|
||||
code, and an entry for the hardware IRQ handler. In order to share
|
||||
common code amongst all platforms, the hardware IRQ handler
|
||||
(archIRQHandler()) is actually implemented in the common ARM port
|
||||
folder but calls back out to a dispatcher function in your platform
|
||||
folder to determine the source of the interrupt and handle it
|
||||
appropriately. Your platform folder should contain this dispatcher
|
||||
function (__interrupt_dispatcher). It must support at least the timer
|
||||
interrupt service routine (atomTimerTick()) required by the OS. The
|
||||
dispatcher also handles other hardware interrupt sources; it determines
|
||||
the source of the IRQ and calls out to the appropriate ISR for that
|
||||
interrupt.
|
||||
* Linker script: Here you should specify the location of the interrupt
|
||||
vector table within RAM, location of code/text segment etc. The
|
||||
Atomthreads ARM port does not dictate particular section names or
|
||||
layout unless your toolchain does not provide a suitable syscalls.c and
|
||||
you wish to use heap. In that case you will (at least initially) be
|
||||
using the heap implementation _sbrk() in syscalls.c in the common ARM
|
||||
port which expects "heap_top" and "end" (heap_base) to be defined by the
|
||||
linker script. These names can be easily changed in ports/arm/syscalls.c
|
||||
if necessary.
|
||||
* UART driver: You should provide at least a UART write routine If you
|
||||
would like to see debug statements, for example to see the results of
|
||||
running the automated test suite. See uart.c for a simple example. Note
|
||||
that for QEMU targets some semihosted toolchains will implement this for
|
||||
you, in which case you won't need either ports/arm/syscalls.c or a UART
|
||||
driver.
|
||||
|
||||
|
||||
---------------------------------------------------------------------------
|
||||
210
ports/arm/atomport-asm.s
Normal file
210
ports/arm/atomport-asm.s
Normal file
@@ -0,0 +1,210 @@
|
||||
/*
|
||||
Copyright (c) 2012, Natie van Rooyen. 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.
|
||||
*/
|
||||
|
||||
|
||||
|
||||
/**/
|
||||
|
||||
|
||||
|
||||
.global archIRQHandler
|
||||
.global contextEnterCritical
|
||||
.global contextExitCritical
|
||||
.global contextEnableInterrupts
|
||||
.global archContextSwitch
|
||||
.global archFirstThreadRestore
|
||||
|
||||
|
||||
.extern __interrupt_dispatcher
|
||||
|
||||
/**/
|
||||
.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 */
|
||||
|
||||
|
||||
.text
|
||||
.code 32
|
||||
|
||||
|
||||
/*
|
||||
* \b archContextSwitch
|
||||
*
|
||||
* Architecture-specific context switch routine.
|
||||
*
|
||||
* Note that interrupts are always locked out when this routine is
|
||||
* called. For cooperative switches, the scheduler will have entered
|
||||
* a critical region. For preemptions (called from an ISR), the
|
||||
* ISR will have disabled interrupts on entry.
|
||||
*
|
||||
* @param[in] old_tcb_ptr Pointer to the thread being scheduled out
|
||||
* @param[in] new_tcb_ptr Pointer to the thread being scheduled in
|
||||
*
|
||||
* @return None
|
||||
*
|
||||
* void archContextSwitch (ATOM_TCB *old_tcb_ptr, ATOM_TCB *new_tcb_ptr)
|
||||
*/
|
||||
archContextSwitch:
|
||||
STMFD sp!, {r4 - r11, lr} /* Save registers */
|
||||
|
||||
STR sp, [r0] /* Save old SP in old_tcb_ptr->sp_save_ptr (first TCB element) */
|
||||
LDR r1, [r1] /* Load new SP from new_tcb_ptr->sp_save_ptr (first TCB element) */
|
||||
MOV sp, r1
|
||||
|
||||
LDMFD sp!, {r4 - r11, pc} /* Load new registers */
|
||||
|
||||
|
||||
/**
|
||||
* \b archFirstThreadRestore
|
||||
*
|
||||
* Architecture-specific function to restore and start the first thread.
|
||||
* This is called by atomOSStart() when the OS is starting.
|
||||
*
|
||||
* This function will be largely similar to the latter half of
|
||||
* archContextSwitch(). Its job is to restore the context for the
|
||||
* first thread, and finally enable interrupts (although we actually
|
||||
* enable interrupts in thread_shell() for new threads in this port
|
||||
* rather than doing it explicitly here).
|
||||
*
|
||||
* It expects to see the context saved in the same way as if the
|
||||
* thread has been previously scheduled out, and had its context
|
||||
* saved. That is, archThreadContextInit() will have been called
|
||||
* first (via atomThreadCreate()) to create a "fake" context save
|
||||
* area, containing the relevant register-save values for a thread
|
||||
* restore.
|
||||
*
|
||||
* Note that you can create more than one thread before starting
|
||||
* the OS - only one thread is restored using this function, so
|
||||
* all other threads are actually restored by archContextSwitch().
|
||||
* This is another reminder that the initial context set up by
|
||||
* archThreadContextInit() must look the same whether restored by
|
||||
* archFirstThreadRestore() or archContextSwitch().
|
||||
*
|
||||
* @param[in] new_tcb_ptr Pointer to the thread being scheduled in
|
||||
*
|
||||
* @return None
|
||||
*
|
||||
* void archFirstThreadRestore (ATOM_TCB *new_tcb_ptr)
|
||||
*/
|
||||
archFirstThreadRestore:
|
||||
LDR r0, [r0] /* Get SP (sp_save_ptr is conveniently first element of TCB) */
|
||||
MOV sp, r0 /* Load new stack pointer */
|
||||
LDMFD sp!, {r4 - r11, pc} /* Load new registers */
|
||||
|
||||
|
||||
/**
|
||||
* \b contextEnableInterrupts
|
||||
*
|
||||
* Enables interrupts on the processor
|
||||
*
|
||||
* @return None
|
||||
*/
|
||||
contextEnableInterrupts:
|
||||
MRS r0, CPSR
|
||||
MOV r1, #I_BIT
|
||||
BIC r0, r0, r1
|
||||
MSR CPSR_c, r0
|
||||
BX lr
|
||||
|
||||
|
||||
/**
|
||||
* \b contextExitCritical
|
||||
*
|
||||
* Exit critical section (restores interrupt posture)
|
||||
*
|
||||
* @param[in] r0 Interrupt Posture
|
||||
*
|
||||
* @return None
|
||||
*/
|
||||
contextExitCritical:
|
||||
MSR CPSR_cxsf, r0
|
||||
BX lr
|
||||
|
||||
|
||||
/**
|
||||
* \b contextEnterCritical
|
||||
*
|
||||
* Enter critical section (disables interrupts)
|
||||
*
|
||||
* @return Current interrupt posture
|
||||
*/
|
||||
contextEnterCritical:
|
||||
MRS r0, CPSR
|
||||
ORR r1, r0, #I_BIT
|
||||
MSR CPSR_cxsf, r1
|
||||
BX lr
|
||||
|
||||
|
||||
/**
|
||||
* \b archIRQHandler
|
||||
*
|
||||
* IRQ entry point.
|
||||
*
|
||||
* Save the process/thread context onto its own stack before calling __interrupt_dispatcher().
|
||||
* __interrupt_dispatcher() might switch stacks. On return the same context is popped from the
|
||||
* stack and control is returned to the process.
|
||||
*
|
||||
* @return None
|
||||
*/
|
||||
archIRQHandler:
|
||||
|
||||
MSR cpsr_c, #(SVC_MODE | I_BIT) /* Save current process context in process stack */
|
||||
STMFD sp!, {r0 - r3, ip, lr}
|
||||
|
||||
MSR cpsr_c, #(IRQ_MODE | I_BIT) /* Save lr_irq and spsr_irq in process stack */
|
||||
SUB lr, lr, #4
|
||||
MOV r1, lr
|
||||
MRS r2, spsr
|
||||
MSR cpsr_c, #(SVC_MODE | I_BIT)
|
||||
STMFD sp!, {r1, r2}
|
||||
|
||||
BL __interrupt_dispatcher /* Dispatch the interrupt to platform folder for
|
||||
the timer tick interrupt or a simular function
|
||||
for other interrupts. Some of those IRQs may
|
||||
call Atomthreads kernel routines and cause a
|
||||
thread switch. */
|
||||
|
||||
LDMFD sp!, {r1, r2} /* Restore lr_irq and spsr_irq from process stack */
|
||||
MSR cpsr_c, #(IRQ_MODE | I_BIT)
|
||||
STMFD sp!, {r1}
|
||||
MSR spsr_cxsf, r2
|
||||
|
||||
MSR cpsr_c, #(SVC_MODE | I_BIT) /* Restore process regs */
|
||||
LDMFD sp!, {r0 - r3, ip, lr}
|
||||
|
||||
MSR cpsr_c, #(IRQ_MODE | I_BIT) /* Exit from IRQ */
|
||||
LDMFD sp!, {pc}^
|
||||
|
||||
40
ports/arm/atomport-private.h
Normal file
40
ports/arm/atomport-private.h
Normal file
@@ -0,0 +1,40 @@
|
||||
/*
|
||||
* Copyright (c) 2012, Natie van Rooyen. 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_PORT_PRIVATE_H
|
||||
#define __ATOM_PORT_PRIVATE_H
|
||||
|
||||
|
||||
/* Function prototypes */
|
||||
extern void archIRQHandler (void);
|
||||
|
||||
/* Platform-specific interrupt dispatcher called on receipt of IRQ */
|
||||
extern void __interrupt_dispatcher (void);
|
||||
|
||||
#endif /* __ATOM_PORT_PRIVATE_H */
|
||||
59
ports/arm/atomport-tests.h
Normal file
59
ports/arm/atomport-tests.h
Normal file
@@ -0,0 +1,59 @@
|
||||
/*
|
||||
* Copyright (c) 2012, Natie van Rooyen. 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_PORT_TESTS_H
|
||||
#define __ATOM_PORT_TESTS_H
|
||||
|
||||
/* Include Atomthreads kernel API */
|
||||
#include "atom.h"
|
||||
|
||||
/* Include printf for ATOMLOG() */
|
||||
#include <stdio.h>
|
||||
|
||||
|
||||
/* Logger macro for viewing test results */
|
||||
#define ATOMLOG printf
|
||||
|
||||
/*
|
||||
* String location macro: for platforms which need to place strings in
|
||||
* alternative locations, e.g. on avr-gcc strings can be placed in
|
||||
* program space, saving SRAM. On most platforms this can expand to
|
||||
* empty.
|
||||
*/
|
||||
#define _STR(x) x
|
||||
|
||||
/* Default thread stack size (in bytes). Allow plenty for printf(). */
|
||||
#define TEST_THREAD_STACK_SIZE 4096
|
||||
|
||||
/* Uncomment to enable logging of stack usage to UART */
|
||||
/* #define TESTS_LOG_STACK_USAGE */
|
||||
|
||||
|
||||
#endif /* __ATOM_PORT_TESTS_H */
|
||||
|
||||
157
ports/arm/atomport.c
Normal file
157
ports/arm/atomport.c
Normal file
@@ -0,0 +1,157 @@
|
||||
/*
|
||||
* Copyright (c) 2013, Natie van Rooyen. 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 "atom.h"
|
||||
#include "atomport.h"
|
||||
|
||||
|
||||
/** Functions defined in atomport_s.s */
|
||||
extern void contextEnableInterrupts (void);
|
||||
|
||||
|
||||
/** Forward declarations */
|
||||
static void thread_shell (void);
|
||||
|
||||
|
||||
/**
|
||||
* \b thread_shell
|
||||
*
|
||||
* Shell routine which is used to call all thread entry points.
|
||||
*
|
||||
* This routine is called whenever a new thread is starting, and
|
||||
* provides a simple wrapper around the thread entry point that
|
||||
* allows us to carry out any actions we want to do on thread's
|
||||
* first starting up, or returning after completion.
|
||||
*
|
||||
* We mainly just want to make sure interrupts are enabled when a
|
||||
* thread is run for the first time. This can be done via stack
|
||||
* restores when threads are first run, but it's handy to have this
|
||||
* wrapper anyway to run some common code if threads run to
|
||||
* completion.
|
||||
*
|
||||
* A thread shell is also handy for providing port users with a place
|
||||
* to do any other initialisation that must be done for each thread
|
||||
* (e.g. opening stdio files etc).
|
||||
*
|
||||
* @return None
|
||||
*/
|
||||
static void thread_shell (void)
|
||||
{
|
||||
ATOM_TCB *curr_tcb;
|
||||
|
||||
/* Get the TCB of the thread being started */
|
||||
curr_tcb = atomCurrentContext();
|
||||
|
||||
/**
|
||||
* Enable interrupts - these will not be enabled when a thread
|
||||
* is first restored.
|
||||
*/
|
||||
contextEnableInterrupts ();
|
||||
|
||||
/* Call the thread entry point */
|
||||
if (curr_tcb && curr_tcb->entry_point)
|
||||
{
|
||||
curr_tcb->entry_point(curr_tcb->entry_param);
|
||||
}
|
||||
|
||||
/* Thread has run to completion: remove it from the ready list */
|
||||
curr_tcb->suspended = TRUE;
|
||||
atomSched (FALSE);
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* \b archThreadContextInit
|
||||
*
|
||||
* Architecture-specific thread context initialisation routine.
|
||||
*
|
||||
* This function must set up a thread's context ready for restoring
|
||||
* and running the thread via archFirstThreadRestore() or
|
||||
* archContextSwitch().
|
||||
*
|
||||
* The layout required to fill the correct register values is
|
||||
* described in archContextSwitch(). Note that not all registers
|
||||
* are restored by archContextSwitch() and archFirstThreadRestore()
|
||||
* as this port takes advantage of the fact that not all registers
|
||||
* must be stored by ARM gcc C subroutines. This means that we don't
|
||||
* provide start values for those registers, as they are "don't cares".
|
||||
*
|
||||
* @param[in] tcb_ptr Pointer to the TCB of the thread being created
|
||||
* @param[in] stack_top Pointer to the top of the new thread's stack
|
||||
* @param[in] entry_point Pointer to the thread entry point function
|
||||
* @param[in] entry_param Parameter to be passed to the thread entry point
|
||||
*
|
||||
* @return None
|
||||
*/
|
||||
void archThreadContextInit (ATOM_TCB *tcb_ptr, void *stack_top, void (*entry_point)(uint32_t), uint32_t entry_param)
|
||||
{
|
||||
uint32_t *stack_ptr;
|
||||
|
||||
/** Start at stack top */
|
||||
tcb_ptr->sp_save_ptr = stack_ptr = stack_top;
|
||||
|
||||
/**
|
||||
* After restoring all of the context registers, the thread restore
|
||||
* routines will return to the address of the calling routine on the
|
||||
* stack. In this case (the first time a thread is run) we "return"
|
||||
* to the entry point for the thread. That is, we store the thread
|
||||
* entry point in the place that return will look for the return
|
||||
* address: the stack.
|
||||
*
|
||||
* Note that we are using the thread_shell() routine to start all
|
||||
* threads, so we actually store the address of thread_shell()
|
||||
* here. Other ports may store the real thread entry point here
|
||||
* and call it directly from the thread restore routines.
|
||||
*
|
||||
* Because we are filling the stack from top to bottom, this goes
|
||||
* on the stack first (at the top).
|
||||
*/
|
||||
*stack_ptr-- = (uint32_t)thread_shell;
|
||||
|
||||
/**
|
||||
* Store starting register values for R4-R11
|
||||
*/
|
||||
*stack_ptr-- = (uint32_t) 0x00001111; /* R11 */
|
||||
*stack_ptr-- = (uint32_t) 0x00001010; /* R10 */
|
||||
*stack_ptr-- = (uint32_t) 0x00000909; /* R9 */
|
||||
*stack_ptr-- = (uint32_t) 0x00000808; /* R8 */
|
||||
*stack_ptr-- = (uint32_t) 0x00000707; /* R7 */
|
||||
*stack_ptr-- = (uint32_t) 0x00000606; /* R6 */
|
||||
*stack_ptr-- = (uint32_t) 0x00000505; /* R5 */
|
||||
*stack_ptr = (uint32_t) 0x00000404; /* R4 */
|
||||
|
||||
/**
|
||||
* All thread context has now been initialised. Save the current
|
||||
* stack pointer to the thread's TCB so it knows where to start
|
||||
* looking when the thread is started.
|
||||
*/
|
||||
tcb_ptr->sp_save_ptr = stack_ptr;
|
||||
|
||||
}
|
||||
|
||||
76
ports/arm/atomport.h
Normal file
76
ports/arm/atomport.h
Normal file
@@ -0,0 +1,76 @@
|
||||
/*
|
||||
* Copyright (c) 2012, Natie van Rooyen. 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_PORT_H
|
||||
#define __ATOM_PORT_H
|
||||
|
||||
/* Portable uint8_t and friends available from stdint.h on this platform */
|
||||
#include <stdint.h>
|
||||
|
||||
/* Definition of NULL is available from stddef.h on this platform */
|
||||
#include <stddef.h>
|
||||
|
||||
/* Required number of system ticks per second (normally 100 for 10ms tick) */
|
||||
#define SYSTEM_TICKS_PER_SEC 100
|
||||
|
||||
|
||||
/* Size of each stack entry / stack alignment size (32 bits on this platform) */
|
||||
#define STACK_ALIGN_SIZE sizeof(uint32_t)
|
||||
|
||||
/**
|
||||
* Architecture-specific types.
|
||||
* Most of these are available from stdint.h on this platform, which is
|
||||
* included above.
|
||||
*/
|
||||
#define POINTER void *
|
||||
|
||||
/* *
|
||||
*
|
||||
* Functions defined in atomport_arm.asm
|
||||
*
|
||||
*/
|
||||
extern uint32_t contextEnterCritical (void) ;
|
||||
extern void contextExitCritical (uint32_t posture) ;
|
||||
|
||||
|
||||
/**
|
||||
* Critical region protection: this should disable interrupts
|
||||
* to protect OS data structures during modification. It must
|
||||
* allow nested calls, which means that interrupts should only
|
||||
* be re-enabled when the outer CRITICAL_END() is reached.
|
||||
*/
|
||||
#define CRITICAL_STORE uint32_t __atom_critical
|
||||
#define CRITICAL_START() __atom_critical = contextEnterCritical()
|
||||
#define CRITICAL_END() contextExitCritical(__atom_critical)
|
||||
|
||||
/* Uncomment to enable stack-checking */
|
||||
/* #define ATOM_STACK_CHECKING */
|
||||
|
||||
|
||||
#endif /* __ATOM_PORT_H */
|
||||
1161
ports/arm/platforms/qemu_integratorcp/Doxyfile
Normal file
1161
ports/arm/platforms/qemu_integratorcp/Doxyfile
Normal file
File diff suppressed because it is too large
Load Diff
129
ports/arm/platforms/qemu_integratorcp/Makefile
Normal file
129
ports/arm/platforms/qemu_integratorcp/Makefile
Normal file
@@ -0,0 +1,129 @@
|
||||
############
|
||||
# Settings #
|
||||
############
|
||||
|
||||
# Build all test applications:
|
||||
# make
|
||||
#
|
||||
# Run all tests within QEMU
|
||||
# make qemutests
|
||||
|
||||
# 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
|
||||
QEMU=qemu-system-arm
|
||||
|
||||
# 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 = modules.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
|
||||
|
||||
# Run tests within simavr simulator
|
||||
phony_qemu_elfs = $(addsuffix .sim, $(TEST_ELFS))
|
||||
qemutests: $(phony_qemu_elfs)
|
||||
.PHONY: qemutests $(phony_qemu_elfs)
|
||||
$(phony_qemu_elfs):
|
||||
./run_test.exp $(QEMU) $(BUILD_DIR)/$(basename $@)
|
||||
223
ports/arm/platforms/qemu_integratorcp/README
Normal file
223
ports/arm/platforms/qemu_integratorcp/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.
|
||||
|
||||
|
||||
---------------------------------------------------------------------------
|
||||
|
||||
163
ports/arm/platforms/qemu_integratorcp/modules.c
Normal file
163
ports/arm/platforms/qemu_integratorcp/modules.c
Normal file
@@ -0,0 +1,163 @@
|
||||
/*
|
||||
* Copyright (c) 2012, Natie van Rooyen. 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 "modules.h"
|
||||
#include <stdio.h>
|
||||
#include <stdarg.h>
|
||||
#include "atomport.h"
|
||||
#include "atomport-private.h"
|
||||
#include "atom.h"
|
||||
#include "atomport.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);
|
||||
|
||||
/** Board-specific registers */
|
||||
ICP_TIMER_T * const board_timer_0 = (ICP_TIMER_T*)BOARD_BASE_ADDRESS_TIMER_0;
|
||||
ICP_PIC_T * const board_pic = (ICP_PIC_T*)BOARD_BASE_ADDRESS_PIC;
|
||||
|
||||
/** TIMER0 clock speed (Hz) */
|
||||
#define TIMER0_CLOCK_SPEED 40000000
|
||||
|
||||
|
||||
/**
|
||||
* \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)
|
||||
{
|
||||
|
||||
board_pic->IRQ_ENABLECLR = ICP_PIC_IRQ_TIMERINT0 ;
|
||||
board_timer_0->INTCLR = 1 ;
|
||||
board_pic->IRQ_ENABLESET |= ICP_PIC_IRQ_TIMERINT0 ;
|
||||
|
||||
/* Set the timer to go off 100 times per second (input clock speed is 40MHz) */
|
||||
board_timer_0->LOAD = TIMER0_CLOCK_SPEED / SYSTEM_TICKS_PER_SEC ;
|
||||
board_timer_0->BGLOAD = TIMER0_CLOCK_SPEED / SYSTEM_TICKS_PER_SEC ;
|
||||
board_timer_0->CONTROL = ICP_TIMER_CONTROL_ENABLE |
|
||||
ICP_TIMER_CONTROL_MODE |
|
||||
ICP_TIMER_CONTROL_IE |
|
||||
ICP_TIMER_CONTROL_TIMER_SIZE ;
|
||||
|
||||
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)
|
||||
{
|
||||
unsigned int status;
|
||||
|
||||
/* Read STATUS register to determine the source of the interrupt */
|
||||
status = board_pic->IRQ_STATUS;
|
||||
|
||||
/* Timer tick interrupt (call Atomthreads timer tick ISR) */
|
||||
if (status | ICP_PIC_IRQ_TIMERINT0)
|
||||
{
|
||||
/*
|
||||
* 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 */
|
||||
board_timer_0->INTCLR = 0x1;
|
||||
|
||||
/* 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");
|
||||
}
|
||||
|
||||
126
ports/arm/platforms/qemu_integratorcp/modules.h
Normal file
126
ports/arm/platforms/qemu_integratorcp/modules.h
Normal file
@@ -0,0 +1,126 @@
|
||||
/*
|
||||
* Copyright (c) 2012, Natie van Rooyen. 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 __MODULES_H__
|
||||
#define __MODULES_H__
|
||||
|
||||
/*
|
||||
* Module definitions to use with the ARM Integrator/CP (ARM926EJ-S)
|
||||
*/
|
||||
|
||||
#include "atomport.h"
|
||||
|
||||
|
||||
/* IO definitions (access restrictions to peripheral registers) */
|
||||
#define __I volatile /*!< defines 'read only' permissions */
|
||||
#define __O volatile /*!< defines 'write only' permissions */
|
||||
#define __IO volatile /*!< defines 'read / write' permissions */
|
||||
|
||||
|
||||
// *****************************************************************************
|
||||
// INTEGRATORCP TIMER
|
||||
// *****************************************************************************
|
||||
typedef struct ICP_TIMER_S {
|
||||
// offset read/write word size reset Description
|
||||
__IO uint32_t LOAD ; // 0x0000 Read/write 32 0x00000000 Load value for Timer
|
||||
__I uint32_t VALUE ; // 0x0004 Read 32 0xFFFFFFFF The current value for Timer
|
||||
__IO uint8_t CONTROL ; // 0x0008 Read/write 8 0x20 Timer control register
|
||||
__O uint32_t INTCLR ; // 0x000C Write - - Timer interrupt clear
|
||||
__I uint32_t RIS ; // 0x0010 Read 1 0x0 Timer raw interrupt status
|
||||
__I uint32_t MIS ; // 0x0014 Read 1 0x0 Timer masked interrupt status
|
||||
__IO uint32_t BGLOAD ; // 0x0018 Read/write 32 0x00000000 Background load value for Timer
|
||||
|
||||
} ICP_TIMER_T, *PICP_TIMER_T ;
|
||||
|
||||
// -------- ICP_TIMER_LOAD : (LOAD Offset: 0x00) Load value for Timer --------
|
||||
// -------- ICP_TIMER_VALUE : (LOAD Offset: 0x04) The current value for Timer --------
|
||||
// -------- ICP_TIMER_CONTROL : (CONTROL Offset: 0x04) Timer control register --------
|
||||
#define ICP_TIMER_CONTROL_MASK ((unsigned int)0x0F << 0) // Timer control mask
|
||||
#define ICP_TIMER_CONTROL_ENABLE ((unsigned int)0x01 << 7) // Timer enable: 0 = disabled 1 = enabled.
|
||||
#define ICP_TIMER_CONTROL_MODE ((unsigned int)0x01 << 6) // Timer mode: 0 = free running, counts once and then wraps to 0xFFFF 1 = periodic, reloads from load register at the end of each count..
|
||||
#define ICP_TIMER_CONTROL_IE ((unsigned int)0x01 << 5) // Interrupt enable.
|
||||
#define ICP_TIMER_CONTROL_R ((unsigned int)0x01 << 4) // Unused, always write as 0s.
|
||||
#define ICP_TIMER_CONTROL_PRESCALE_MASK ((unsigned int)0x03 << 2) // Prescale divisor
|
||||
#define ICP_TIMER_CONTROL_PRESCALE_NONE ((unsigned int)0x00 << 2) //
|
||||
#define ICP_TIMER_CONTROL_PRESCALE_16 ((unsigned int)0x01 << 2) //
|
||||
#define ICP_TIMER_CONTROL_PRESCALE_256 ((unsigned int)0x02 << 2) //
|
||||
#define ICP_TIMER_CONTROL_TIMER_SIZE ((unsigned int)0x01 << 1) // Selects 16/32 bit counter operation: 0 = 16-bit counter (default) 1 = 32-bit counter For 16-bit mode, write the high 16 bits of the 32-bit value as 0.
|
||||
#define ICP_TIMER_CONTROL_ONE_SHOT ((unsigned int)0x01 << 0) // Selects one-shot or wrapping counter mode: 0 = wrapping mode (default) 1 = one-shot mode
|
||||
// -------- ICP_TIMER_INTCLR : (INTCLR Offset: 0x0C) Timer interrupt clear --------
|
||||
// -------- ICP_TIMER_RIS : (RIS Offset: 0x10) Timer raw interrupt status --------
|
||||
// -------- ICP_TIMER_MIS : (MIS Offset: 0x14) Timer masked interrupt status --------
|
||||
#define ICP_TIMER_INT ((unsigned int)0x01 << 0) // Interrupt
|
||||
// -------- ICP_TIMER_BGLOAD : (BGLOAD Offset: 0x18) Timer masked interrupt status --------
|
||||
|
||||
|
||||
// *****************************************************************************
|
||||
// INTEGRATORCP PIC
|
||||
// *****************************************************************************
|
||||
typedef struct ICP_PIC_S {
|
||||
// offset read/write word size reset Description
|
||||
__I uint32_t IRQ_STATUS ; // 0x0000 Read 22 IRQ gated interrupt status
|
||||
__I uint32_t IRQ_RAWSTAT ; // 0x0004 Read 22 IRQ raw interrupt status
|
||||
__IO uint32_t IRQ_ENABLESET ; // 0x0008 Read/write 22 IRQ enable set
|
||||
__O uint32_t IRQ_ENABLECLR ; // 0x000C Write 22 IRQ enable clear
|
||||
__IO uint32_t INT_SOFTSET ; // 0x0010 Read/write 16 Software interrupt set
|
||||
__O uint32_t INT_SOFTCLR ; // 0x0014 Write 16 Software interrupt clear
|
||||
uint32_t RESERVED[2] ; // 0x0018
|
||||
__I uint32_t FIQ_STATUS ; // 0x0020 Read 22 FIQ gated interrupt status
|
||||
__I uint32_t FIQ_RAWSTAT ; // 0x0024 Read 22 FIQ raw interrupt status
|
||||
__IO uint32_t FIQ_ENABLESET ; // 0x0028 Read/write 22 FIQ enable set
|
||||
__O uint32_t FIQ_ENABLECLR ; // 0x002C Write-only 22 FIQ enable clear
|
||||
|
||||
} ICP_PIC_T, *PICP_PIC_T ;
|
||||
|
||||
// -------- ICP_PIC_IRQ_STATUS : (IRQ_STATUS Offset: 0x00) IRQ gated interrupt status --------
|
||||
// -------- ICP_PIC_IRQ_RAWSTAT : (IRQ_RAWSTAT Offset: 0x04) IRQ raw interrupt status --------
|
||||
// -------- ICP_PIC_IRQ_ENABLESET : (IRQ_ENABLESET Offset: 0x08) IRQ enable set --------
|
||||
// -------- ICP_PIC_IRQ_ENABLECLR : (IRQ_ENABLECLR Offset: 0x0C) IRQ enable clear --------
|
||||
#define ICP_PIC_IRQ_MASK ((unsigned int)0x3FFFFF << 0) // IRQ mask
|
||||
#define ICP_PIC_IRQ_TIMERINT2 ((unsigned int)0x01 << 7) // TIMERINT2 Counter-timer 2 interrupt
|
||||
#define ICP_PIC_IRQ_TIMERINT1 ((unsigned int)0x01 << 6) // TIMERINT1 Counter-timer 1 interrupt
|
||||
#define ICP_PIC_IRQ_TIMERINT0 ((unsigned int)0x01 << 5) // TIMERINT0 Counter-timer 0 interrupt
|
||||
#define ICP_PIC_IRQ_SOFTINT ((unsigned int)0x01 << 0) // OFTINT Software interrupt
|
||||
// -------- ICP_PIC_INT_SOFTSET : (INT_SOFTSET Offset: 0x10) Software interrupt set --------
|
||||
// -------- ICP_PIC_INT_SOFTCLR : (INT_SOFTCLR Offset: 0x14) Software interrupt clear --------
|
||||
|
||||
|
||||
|
||||
/* module definitions */
|
||||
#define BOARD_BASE_ADDRESS_TIMER_0 0x13000000
|
||||
#define BOARD_BASE_ADDRESS_PIC 0x14000000
|
||||
|
||||
extern ICP_TIMER_T* const board_timer_0 ;
|
||||
extern ICP_PIC_T* const board_pic ;
|
||||
|
||||
|
||||
/* Function prototypes */
|
||||
extern int low_level_init (void) ;
|
||||
|
||||
|
||||
#endif /* __MODULES_H__ */
|
||||
39
ports/arm/platforms/qemu_integratorcp/run_test.exp
Executable file
39
ports/arm/platforms/qemu_integratorcp/run_test.exp
Executable file
@@ -0,0 +1,39 @@
|
||||
#!/usr/bin/env expect
|
||||
|
||||
# Expect script to run an automated test within QEMU and check for successful
|
||||
# completion.
|
||||
#
|
||||
# Arguments: <path_to_qemu> <test_elf_file>
|
||||
#
|
||||
# Returns 0 on successful test run within QEMU, 1 on failure
|
||||
|
||||
|
||||
# Start the test
|
||||
spawn [lindex $argv 0] -M integratorcp -semihosting -nographic -kernel [lindex $argv 1]
|
||||
|
||||
# Expect to see the test starting within 10 seconds
|
||||
set timeout 10
|
||||
|
||||
# Wait for the test to start ("Go")
|
||||
expect {
|
||||
"Go\r\n" {
|
||||
puts "Test started"
|
||||
|
||||
# The test could take up to 3 minutes to complete once started
|
||||
set timeout 180
|
||||
|
||||
# Now expect to see "Pass" or "Fail" within 3 minutes
|
||||
expect {
|
||||
"Pass\r\n" { puts "Test passed"; exit 0 }
|
||||
"Fail\r\n" { puts "Test failed"; exit 1 }
|
||||
timeout { puts "Test timed out without completing"; exit 1 }
|
||||
}
|
||||
}
|
||||
|
||||
timeout {
|
||||
# Didn't receive "Go" within 10 seconds
|
||||
puts "Test failed to start ('Go' not seen)"
|
||||
exit 1
|
||||
}
|
||||
}
|
||||
|
||||
54
ports/arm/platforms/qemu_integratorcp/startup.s
Normal file
54
ports/arm/platforms/qemu_integratorcp/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/qemu_integratorcp/system.ld
Normal file
89
ports/arm/platforms/qemu_integratorcp/system.ld
Normal file
@@ -0,0 +1,89 @@
|
||||
|
||||
ENTRY(__interrupt_vector_table)
|
||||
|
||||
|
||||
MEMORY
|
||||
{
|
||||
flash (rx) : ORIGIN = 0x00000000, LENGTH = 0x00020000
|
||||
sram (rwx) : ORIGIN = 0x00020000, LENGTH = 0x00020000
|
||||
}
|
||||
|
||||
|
||||
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;
|
||||
255
ports/arm/platforms/qemu_integratorcp/uart.c
Normal file
255
ports/arm/platforms/qemu_integratorcp/uart.c
Normal file
@@ -0,0 +1,255 @@
|
||||
/*
|
||||
* 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 "uart.h"
|
||||
|
||||
|
||||
/* Constants */
|
||||
|
||||
/** UART0 registers base address */
|
||||
#define UART0_ADDR 0x16000000
|
||||
|
||||
/** 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 (atomMutexGet(&uart_mutex, 0) == ATOM_OK)
|
||||
{
|
||||
/* Wait for not-empty */
|
||||
while(UART_FR(UART0_ADDR) & UART_FR_RXFE)
|
||||
;
|
||||
|
||||
/* Read first byte */
|
||||
*ptr++ = UART_DR(UART0_ADDR);
|
||||
|
||||
/* Loop over remaining bytes until empty */
|
||||
for (todo = 1; todo < len; todo++)
|
||||
{
|
||||
/* Quit if receive FIFO empty */
|
||||
if(UART_FR(UART0_ADDR) & UART_FR_RXFE)
|
||||
{
|
||||
break;
|
||||
}
|
||||
|
||||
/* Read next byte */
|
||||
*ptr++ = UART_DR(UART0_ADDR);
|
||||
}
|
||||
|
||||
/* Return mutex access */
|
||||
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 (atomMutexGet(&uart_mutex, 0) == ATOM_OK)
|
||||
{
|
||||
/* Loop through all bytes to write */
|
||||
for (todo = 0; todo < len; todo++)
|
||||
{
|
||||
/* Wait for empty */
|
||||
while(UART_FR(UART0_ADDR) & UART_FR_TXFF)
|
||||
;
|
||||
|
||||
/* Write byte to UART */
|
||||
UART_DR(UART0_ADDR) = *ptr++;
|
||||
}
|
||||
|
||||
/* Return mutex access */
|
||||
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(UART0_ADDR) & UART_FR_TXFF)
|
||||
;
|
||||
|
||||
/* Write byte to UART */
|
||||
UART_DR(UART0_ADDR) = *ptr++;
|
||||
}
|
||||
}
|
||||
|
||||
/* Loop forever */
|
||||
while (1)
|
||||
;
|
||||
|
||||
}
|
||||
38
ports/arm/platforms/qemu_integratorcp/uart.h
Normal file
38
ports/arm/platforms/qemu_integratorcp/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 */
|
||||
211
ports/arm/syscalls.c
Normal file
211
ports/arm/syscalls.c
Normal file
@@ -0,0 +1,211 @@
|
||||
/*
|
||||
* 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
|
||||
* Syscalls implementation for stdio and heap management.
|
||||
*
|
||||
*
|
||||
* Simple implementation of syscalls.c for ARM compiler toolchains built
|
||||
* without newlib. Allows usage of printf() and friends as well as heap
|
||||
* allocation.
|
||||
*
|
||||
* NOTE: Platform/BSP must implement uart_read() and uart_write().
|
||||
*
|
||||
* NOTE: Platform/BSP linker script must define "end" and "heap_top" which are
|
||||
* the heap base and top respectively.
|
||||
*
|
||||
* No file table is implemented. All file read/write operations are carried
|
||||
* out on the UART driver, regardless of file descriptor.
|
||||
*
|
||||
* Mostly based on code from http://balau82.wordpress.com
|
||||
*
|
||||
*/
|
||||
|
||||
#include <sys/stat.h>
|
||||
#include "uart.h"
|
||||
|
||||
|
||||
/**
|
||||
* Define all functions as weak so that the functions in this file will
|
||||
* only be used if the compiler toolchain doesn't already provide them.
|
||||
*/
|
||||
extern int _close(int file) __attribute__((weak));
|
||||
extern int _fstat(int file, struct stat *st) __attribute__((weak));
|
||||
extern int _isatty(int file) __attribute__((weak));
|
||||
extern int _lseek(int file, int ptr, int dir) __attribute__((weak));
|
||||
extern int _open(const char *name, int flags, int mode) __attribute__((weak));
|
||||
extern int _read(int file, char *ptr, int len) __attribute__((weak));
|
||||
extern caddr_t _sbrk(int incr) __attribute__((weak));
|
||||
extern int _write(int file, char *ptr, int len) __attribute__((weak));
|
||||
|
||||
|
||||
/**
|
||||
* \b _close
|
||||
*
|
||||
* Simple stub implementation with no file table. All parameters ignored.
|
||||
*
|
||||
*/
|
||||
int _close(int file)
|
||||
{
|
||||
return -1;
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* \b _fstat
|
||||
*
|
||||
* Simple stub implementation. Always return character device.
|
||||
*
|
||||
*/
|
||||
int _fstat(int file, struct stat *st)
|
||||
{
|
||||
/* Only UART supported, always return character-oriented device file */
|
||||
st->st_mode = S_IFCHR;
|
||||
return 0;
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* \b _isatty
|
||||
*
|
||||
* Simple stub implementation. Only UART supported so TTY always true.
|
||||
*
|
||||
*/
|
||||
int _isatty(int file)
|
||||
{
|
||||
return 1;
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* \b _lseek
|
||||
*
|
||||
* Simple stub implementation. All parameters ignored.
|
||||
*
|
||||
*/
|
||||
int _lseek(int file, int ptr, int dir)
|
||||
{
|
||||
return 0;
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* \b _open
|
||||
*
|
||||
* Simple stub implementation with no file table. All parameters ignored.
|
||||
*
|
||||
*/
|
||||
int _open(const char *name, int flags, int mode)
|
||||
{
|
||||
return -1;
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* \b _read
|
||||
*
|
||||
* Simple read file implementation. Ignores file descriptor parameter
|
||||
* and always reads from the UART driver.
|
||||
*
|
||||
* @param[in] file File descriptor (parameter ignored)
|
||||
* @param[in] ptr Pointer to receive buffer
|
||||
* @param[in] len Max bytes to read
|
||||
*
|
||||
* @retval Number of bytes read
|
||||
*/
|
||||
int _read(int file, char *ptr, int len)
|
||||
{
|
||||
/* Read from the UART driver, regardless of file descriptor */
|
||||
return (uart_read (ptr, len));
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* \b _write
|
||||
*
|
||||
* Simple write file implementation. Ignores file descriptor parameter
|
||||
* and always writes to the UART driver.
|
||||
*
|
||||
* @param[in] file File descriptor (parameter ignored)
|
||||
* @param[in] ptr Pointer to write buffer
|
||||
* @param[in] len Number of bytes to write
|
||||
*
|
||||
* @retval Number of bytes written
|
||||
*/
|
||||
int _write(int file, char *ptr, int len)
|
||||
{
|
||||
/* Write to the UART driver, regardless of file descriptor */
|
||||
return (uart_write (ptr, len));
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* \b _sbrk
|
||||
*
|
||||
* Simple heap implementation.
|
||||
*
|
||||
* The platform/BSP must define "end" and "heap_top" which are the heap
|
||||
* base and top respectively.
|
||||
*
|
||||
* @param[in] incr Chunk size
|
||||
*
|
||||
* @retval Pointer to allocated chunk start
|
||||
*/
|
||||
caddr_t _sbrk(int incr)
|
||||
{
|
||||
extern char end; /* Defined by the linker */
|
||||
extern char heap_top; /* Defined by the linker */
|
||||
static char *heap_end = 0;
|
||||
char *prev_heap_end;
|
||||
|
||||
/* First time in, initialise heap base using definition from linker script */
|
||||
if (heap_end == 0)
|
||||
{
|
||||
heap_end = &end;
|
||||
}
|
||||
|
||||
/* Save the previous heap base */
|
||||
prev_heap_end = heap_end;
|
||||
|
||||
/* Check we have not passed the heap top */
|
||||
if (heap_end + incr > &heap_top)
|
||||
{
|
||||
/* Heap top reached, failed to allocate */
|
||||
return (caddr_t)0;
|
||||
}
|
||||
|
||||
/* New heap base */
|
||||
heap_end += incr;
|
||||
|
||||
/* Return pointer to previous base (where our allocation starts) */
|
||||
return (caddr_t)prev_heap_end;
|
||||
}
|
||||
|
||||
203
ports/arm/tests-main.c
Normal file
203
ports/arm/tests-main.c
Normal file
@@ -0,0 +1,203 @@
|
||||
/*
|
||||
* 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 "atom.h"
|
||||
#include "atomport-private.h"
|
||||
#include "atomtests.h"
|
||||
#include "atomtimer.h"
|
||||
|
||||
|
||||
/* Constants */
|
||||
|
||||
/*
|
||||
* Idle thread stack size
|
||||
*
|
||||
* This needs to be large enough to handle any interrupt handlers
|
||||
* and callbacks called by interrupt handlers (e.g. user-created
|
||||
* timer callbacks) as well as the saving of all context when
|
||||
* switching away from this thread.
|
||||
*/
|
||||
#define IDLE_STACK_SIZE_BYTES 512
|
||||
|
||||
|
||||
/*
|
||||
* Main thread stack size
|
||||
*
|
||||
* Note that this is not a required OS kernel thread - you will replace
|
||||
* this with your own application thread.
|
||||
*
|
||||
* In this case the Main thread is responsible for calling out to the
|
||||
* test routines. Once a test routine has finished, the test status is
|
||||
* printed out on the UART and the thread remains running in a loop.
|
||||
*
|
||||
* The Main thread stack generally needs to be larger than the idle
|
||||
* thread stack, as not only does it need to store interrupt handler
|
||||
* stack saves and context switch saves, but the application main thread
|
||||
* will generally be carrying out more nested function calls and require
|
||||
* stack for application code local variables etc.
|
||||
*
|
||||
* 1KB might be adequate but if using printf() then at least 2KB would be
|
||||
* prudent otherwise the stdio functions otherwise stack overruns are
|
||||
* likely. Nearly 2KB was seen to be used on the toolchain used for
|
||||
* development.
|
||||
*/
|
||||
#define MAIN_STACK_SIZE_BYTES 4096
|
||||
|
||||
|
||||
/* Local data */
|
||||
|
||||
/* Application threads' TCBs */
|
||||
static ATOM_TCB main_tcb;
|
||||
|
||||
/* Main thread's stack area */
|
||||
static uint8_t main_thread_stack[MAIN_STACK_SIZE_BYTES];
|
||||
|
||||
/* Idle thread's stack area */
|
||||
static uint8_t idle_thread_stack[IDLE_STACK_SIZE_BYTES];
|
||||
|
||||
|
||||
/* Forward declarations */
|
||||
static void main_thread_func (uint32_t data);
|
||||
|
||||
|
||||
/**
|
||||
* \b main
|
||||
*
|
||||
* Program entry point.
|
||||
*
|
||||
* Creates an application thread and starts the OS.
|
||||
*/
|
||||
|
||||
int main ( void )
|
||||
{
|
||||
int8_t status;
|
||||
|
||||
/**
|
||||
* Note: to protect OS structures and data during initialisation,
|
||||
* interrupts must remain disabled until the first thread
|
||||
* has been restored. They are reenabled at the very end of
|
||||
* the first thread restore, at which point it is safe for a
|
||||
* reschedule to take place.
|
||||
*/
|
||||
|
||||
/**
|
||||
* Initialise the OS before creating our threads.
|
||||
*/
|
||||
status = atomOSInit(&idle_thread_stack[0], IDLE_STACK_SIZE_BYTES, TRUE);
|
||||
if (status == ATOM_OK)
|
||||
{
|
||||
/* Create an application thread */
|
||||
status = atomThreadCreate(&main_tcb,
|
||||
TEST_THREAD_PRIO, main_thread_func, 0,
|
||||
&main_thread_stack[0],
|
||||
MAIN_STACK_SIZE_BYTES,
|
||||
TRUE);
|
||||
if (status == ATOM_OK)
|
||||
{
|
||||
/**
|
||||
* First application thread successfully created. It is
|
||||
* now possible to start the OS. Execution will not return
|
||||
* from atomOSStart(), which will restore the context of
|
||||
* our application thread and start executing it.
|
||||
*
|
||||
* Note that interrupts are still disabled at this point.
|
||||
* They will be enabled as we restore and execute our first
|
||||
* thread in archFirstThreadRestore().
|
||||
*/
|
||||
atomOSStart();
|
||||
}
|
||||
}
|
||||
|
||||
while (1)
|
||||
;
|
||||
|
||||
/* There was an error starting the OS if we reach here */
|
||||
return (0);
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* \b main_thread_func
|
||||
*
|
||||
* Entry point for main application thread.
|
||||
*
|
||||
* This is the first thread that will be executed when the OS is started.
|
||||
*
|
||||
* @param[in] data Unused (optional thread entry parameter)
|
||||
*
|
||||
* @return None
|
||||
*/
|
||||
static void main_thread_func (uint32_t data)
|
||||
{
|
||||
uint32_t test_status;
|
||||
|
||||
/* Put a message out on the UART */
|
||||
printf ("Go\n");
|
||||
|
||||
/* Start test. All tests use the same start API. */
|
||||
test_status = test_start();
|
||||
|
||||
/* Check main thread stack usage (if enabled) */
|
||||
#ifdef ATOM_STACK_CHECKING
|
||||
if (test_status == 0)
|
||||
{
|
||||
uint32_t used_bytes, free_bytes;
|
||||
|
||||
/* Check idle thread stack usage */
|
||||
if (atomThreadStackCheck (&main_tcb, &used_bytes, &free_bytes) == ATOM_OK)
|
||||
{
|
||||
/* Check the thread did not use up to the end of stack */
|
||||
if (free_bytes == 0)
|
||||
{
|
||||
printf ("Main stack overflow\n");
|
||||
test_status++;
|
||||
}
|
||||
|
||||
/* Log the stack usage */
|
||||
#ifdef TESTS_LOG_STACK_USAGE
|
||||
printf ("MainUse:%d\n", (int)used_bytes);
|
||||
#endif
|
||||
}
|
||||
|
||||
}
|
||||
#endif
|
||||
|
||||
/* Log final status */
|
||||
if (test_status == 0)
|
||||
{
|
||||
printf ("Pass\n");
|
||||
}
|
||||
else
|
||||
{
|
||||
printf ("Fail(%d)\n", (int)test_status);
|
||||
}
|
||||
|
||||
}
|
||||
1161
ports/armv7a/Doxyfile
Normal file
1161
ports/armv7a/Doxyfile
Normal file
File diff suppressed because it is too large
Load Diff
149
ports/armv7a/Makefile
Normal file
149
ports/armv7a/Makefile
Normal file
@@ -0,0 +1,149 @@
|
||||
##########################################
|
||||
# Toplevel makefile for all ARM7a boards #
|
||||
##########################################
|
||||
|
||||
# Build directory
|
||||
ifdef O
|
||||
build_dir=$(shell readlink -f $(O))
|
||||
else
|
||||
build_dir=$(CURDIR)/build
|
||||
endif
|
||||
|
||||
# Source directory
|
||||
src_dir=$(CURDIR)
|
||||
|
||||
# Configuration
|
||||
ifndef CPU
|
||||
CPU=cortex-a8
|
||||
endif
|
||||
ifndef BOARD
|
||||
BOARD=pb-a8
|
||||
endif
|
||||
CC=$(CROSS_COMPILE)gcc
|
||||
OBJCOPY=$(CROSS_COMPILE)objcopy
|
||||
# Enable stack-checking. WARNING: the full automated test suite currently
|
||||
# requires a little over 1KB RAM with stack-checking enabled. If you are
|
||||
# using a device with 1KB internal SRAM and no external SRAM then you
|
||||
# must disable stack-checking to run all of the automated tests.
|
||||
#STACK_CHECK=true
|
||||
|
||||
# Location of atomthreads sources
|
||||
board_dir=$(src_dir)/$(BOARD)
|
||||
kernel_dir=$(src_dir)/../../kernel
|
||||
tests_dir=$(src_dir)/../../tests
|
||||
|
||||
# Check if verbosity is ON for build process
|
||||
VERBOSE_DEFAULT := 0
|
||||
CMD_PREFIX_DEFAULT := @
|
||||
ifdef VERBOSE
|
||||
ifeq ("$(origin VERBOSE)", "command line")
|
||||
VB := $(VERBOSE)
|
||||
else
|
||||
VB := $(VERBOSE_DEFAULT)
|
||||
endif
|
||||
else
|
||||
VB := $(VERBOSE_DEFAULT)
|
||||
endif
|
||||
ifeq ($(VB), 1)
|
||||
V :=
|
||||
else
|
||||
V := $(CMD_PREFIX_DEFAULT)
|
||||
endif
|
||||
|
||||
# object files
|
||||
objs = arm_irq.o
|
||||
objs += arm_main.o
|
||||
objs += atomport.o
|
||||
objs += arm_entry.o
|
||||
objs += atomport-asm.o
|
||||
|
||||
# include board makefile for board specific objects
|
||||
-include $(board_dir)/Makefile
|
||||
|
||||
# library object files
|
||||
objs += printk.o
|
||||
objs += string.o
|
||||
objs += vsprintf.o
|
||||
|
||||
# Kernel object files
|
||||
objs += atomkernel.o
|
||||
objs += atomsem.o
|
||||
objs += atommutex.o
|
||||
objs += atomtimer.o
|
||||
objs += atomqueue.o
|
||||
|
||||
# Collection of built objects (excluding test applications)
|
||||
build_objs = $(foreach obj,$(objs),$(build_dir)/$(obj))
|
||||
|
||||
# Target application filenames .elf for each test object
|
||||
tobjs = $(notdir $(patsubst %.c,%.o,$(wildcard $(tests_dir)/*.c)))
|
||||
telfs = $(patsubst %.o,%.elf,$(tobjs))
|
||||
tbins = $(patsubst %.o,%.bin,$(tobjs))
|
||||
build_tobjs = $(foreach tobj,$(tobjs),$(build_dir)/$(tobj))
|
||||
build_telfs = $(foreach telf,$(telfs),$(build_dir)/$(telf))
|
||||
build_tbins = $(foreach tbin,$(tbins),$(build_dir)/$(tbin))
|
||||
|
||||
# GCC flags
|
||||
CFLAGS= -g \
|
||||
-Wall \
|
||||
-Werror \
|
||||
-mcpu=$(CPU) \
|
||||
-nostdinc \
|
||||
-nostdlib \
|
||||
-nodefaultlibs \
|
||||
-fno-builtin \
|
||||
-I$(src_dir) \
|
||||
-I$(board_dir) \
|
||||
-I$(kernel_dir) \
|
||||
-I$(tests_dir)
|
||||
|
||||
# Enable stack-checking (disable if not required)
|
||||
ifeq ($(STACK_CHECK),true)
|
||||
CFLAGS += -DATOM_STACK_CHECKING
|
||||
endif
|
||||
|
||||
# All
|
||||
.PHONY: all
|
||||
all: $(build_tbins) $(build_telfs) $(build_tobjs) $(build_objs) Makefile
|
||||
|
||||
$(build_dir)/%.bin: $(build_dir)/%.elf
|
||||
$(V)mkdir -p `dirname $@`
|
||||
$(if $(V), @echo " (OBJCOPY) $(subst $(build_dir)/,,$@)")
|
||||
$(V)$(OBJCOPY) -O binary $< $@
|
||||
|
||||
$(build_dir)/%.elf: $(build_dir)/%.o $(build_objs) $(board_dir)/linker.ld
|
||||
$(V)mkdir -p `dirname $@`
|
||||
$(if $(V), @echo " (ELF) $(subst $(build_dir)/,,$@)")
|
||||
$(V)$(CC) $(CFLAGS) $(build_objs) $< -static-libgcc -lgcc -Wl -T $(board_dir)/linker.ld -o $@
|
||||
|
||||
$(build_dir)/%.o: $(src_dir)/%.S
|
||||
$(V)mkdir -p `dirname $@`
|
||||
$(if $(V), @echo " (AS) $(subst $(build_dir)/,,$@)")
|
||||
$(V)$(CC) $(CFLAGS) -D__ASSEMBLY__ -I`dirname $<` -c $< -o $@
|
||||
|
||||
$(build_dir)/%.o: $(src_dir)/%.c
|
||||
$(V)mkdir -p `dirname $@`
|
||||
$(if $(V), @echo " (CC) $(subst $(build_dir)/,,$@)")
|
||||
$(V)$(CC) $(CFLAGS) $(CFLAGS) -I`dirname $<` -c $< -o $@
|
||||
|
||||
$(build_dir)/%.o: $(kernel_dir)/%.c
|
||||
$(V)mkdir -p `dirname $@`
|
||||
$(if $(V), @echo " (CC) $(subst $(build_dir)/,,$@)")
|
||||
$(V)$(CC) $(CFLAGS) $(CFLAGS) -I`dirname $<` -c $< -o $@
|
||||
|
||||
$(build_dir)/%.o: $(tests_dir)/%.c
|
||||
$(V)mkdir -p `dirname $@`
|
||||
$(if $(V), @echo " (CC) $(subst $(build_dir)/,,$@)")
|
||||
$(V)$(CC) $(CFLAGS) $(CFLAGS) -I`dirname $<` -c $< -o $@
|
||||
|
||||
# Clean
|
||||
.PHONY: clean
|
||||
clean:
|
||||
rm -rf doxygen-kernel
|
||||
rm -rf doxygen-armv7a
|
||||
rm -rf $(build_dir)
|
||||
# Docs
|
||||
.PHONY: doxygen
|
||||
doxygen:
|
||||
doxygen $(kernel_dir)/Doxyfile
|
||||
doxygen ./Doxyfile
|
||||
14
ports/armv7a/README
Normal file
14
ports/armv7a/README
Normal file
@@ -0,0 +1,14 @@
|
||||
CROSS_COMPILE=/opt/arm-2009q3/bin/arm-none-eabi-
|
||||
export CROSS_COMPILE
|
||||
|
||||
qemu-system-arm -M realview-pb-a8 -serial stdio -kernel build/kern2.elf
|
||||
|
||||
Following are the steps to add new board <abc> under armv7a port:
|
||||
1. Create a directory under ports/armv7a with name <abc>
|
||||
2. Add linker.ld in ports/armv7a/<abc> (similar to the one in ports/armv7a/pb-a8)
|
||||
3. Add arm_pic.h, arm_timer.h, and arm_uart.h in ports/armv7a/<abc> (similar to the one in ports/armv7a/pb-a8)
|
||||
4. Add .c files in ports/armv7a/<abc> to implement the board specific functions expected in arm_pic.h, arm_timer.h, and arm_uart.h
|
||||
5. Add Makefile in ports/armv7a/<abc> (similar to the one in ports/armv7a/pb-a8) to build board specific objects corresponding to the .c files added in step 4.
|
||||
5. To build atomthreads for <abc> board use following command "make BOARD=<abc>" with ports/armv7a as current directory.
|
||||
|
||||
Port contributed by Anup Patel (http://brainfault.blogspot.com).
|
||||
123
ports/armv7a/arm_asm_macro.h
Normal file
123
ports/armv7a/arm_asm_macro.h
Normal file
@@ -0,0 +1,123 @@
|
||||
/*
|
||||
* Copyright (c) 2011, Anup Patel. 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 __ARM_ASM_MACRO_H__
|
||||
#define __ARM_ASM_MACRO_H__
|
||||
|
||||
#include <arm_defines.h>
|
||||
|
||||
#ifdef __ASSEMBLY__
|
||||
|
||||
.macro SET_CURRENT_FLAGS flags, treg
|
||||
mrs \treg, cpsr
|
||||
orr \treg, \treg, #(\flags)
|
||||
msr cpsr, \treg
|
||||
.endm
|
||||
|
||||
.macro SET_CURRENT_MODE mode
|
||||
cps #(\mode)
|
||||
.endm
|
||||
|
||||
.macro SET_CURRENT_STACK new_stack
|
||||
ldr sp, \new_stack
|
||||
.endm
|
||||
|
||||
.macro START_EXCEPTION_HANDLER irqname, lroffset
|
||||
.align 5
|
||||
\irqname:
|
||||
sub lr, lr, #\lroffset
|
||||
.endm
|
||||
|
||||
/* Save User Registers */
|
||||
.macro PUSH_USER_REGS
|
||||
str lr, [sp, #-4]!; /* Push the return address */
|
||||
sub sp, sp, #(4*15); /* Adjust the stack pointer */
|
||||
stmia sp, {r0-r12}; /* Push user mode registers */
|
||||
add r0, sp, #(4*13); /* Adjust the stack pointer */
|
||||
stmia r0, {r13-r14}^; /* Push user mode registers */
|
||||
mov r0, r0; /* NOP for previous inst */
|
||||
mrs r0, spsr_all; /* Put the SPSR on the stack */
|
||||
str r0, [sp, #-4]!
|
||||
.endm
|
||||
|
||||
/* If came from priviledged mode then push banked registers */
|
||||
.macro PUSH_BANKED_REGS skip_lable
|
||||
mov r4, r0
|
||||
and r0, r0, #CPSR_MODE_MASK
|
||||
cmp r0, #CPSR_MODE_USER
|
||||
beq \skip_lable
|
||||
add r1, sp, #(4*14)
|
||||
mrs r5, cpsr
|
||||
orr r4, r4, #(CPSR_IRQ_DISABLED | CPSR_FIQ_DISABLED)
|
||||
msr cpsr, r4
|
||||
str sp, [r1, #0]
|
||||
str lr, [r1, #4]
|
||||
msr cpsr, r5
|
||||
\skip_lable:
|
||||
.endm
|
||||
|
||||
/* Call C function to handle exception */
|
||||
.macro CALL_EXCEPTION_CFUNC cfunc
|
||||
mov r0, sp
|
||||
bl \cfunc
|
||||
.endm
|
||||
|
||||
/* If going back to priviledged mode then pull banked registers */
|
||||
.macro PULL_BANKED_REGS skip_lable
|
||||
ldr r0, [sp, #0]
|
||||
mov r4, r0
|
||||
and r0, r0, #CPSR_MODE_MASK
|
||||
cmp r0, #CPSR_MODE_USER
|
||||
beq \skip_lable
|
||||
add r1, sp, #(4*14)
|
||||
mrs r5, cpsr
|
||||
orr r4, r4, #(CPSR_IRQ_DISABLED | CPSR_FIQ_DISABLED)
|
||||
msr cpsr, r4
|
||||
ldr sp, [r1, #0]
|
||||
ldr lr, [r1, #4]
|
||||
msr cpsr, r5
|
||||
\skip_lable:
|
||||
.endm
|
||||
|
||||
/* Restore User Registers */
|
||||
.macro PULL_USER_REGS
|
||||
ldr r0, [sp], #0x0004; /* Get SPSR from stack */
|
||||
msr spsr_all, r0;
|
||||
ldmia sp, {r0-r14}^; /* Restore registers (user) */
|
||||
mov r0, r0; /* NOP for previous isnt */
|
||||
add sp, sp, #(4*15); /* Adjust the stack pointer */
|
||||
ldr lr, [sp], #0x0004 /* Pull return address */
|
||||
.endm
|
||||
|
||||
.macro END_EXCEPTION_HANDLER
|
||||
movs pc, lr
|
||||
.endm
|
||||
|
||||
#endif
|
||||
|
||||
#endif
|
||||
68
ports/armv7a/arm_defines.h
Normal file
68
ports/armv7a/arm_defines.h
Normal file
@@ -0,0 +1,68 @@
|
||||
/*
|
||||
* Copyright (c) 2010, Atomthreads Project. 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 __ARM_DEFINES_H_
|
||||
#define __ARM_DEFINES_H_
|
||||
|
||||
#define CPSR_VALIDBITS_MASK 0xFF0FFFFF
|
||||
#define CPSR_USERBITS_MASK 0xFFFFFC00
|
||||
#define CPSR_USERBITS_SHIFT 10
|
||||
#define CPSR_PRIVBITS_MASK 0x000003FF
|
||||
#define CPSR_PRIVBITS_SHIFT 0
|
||||
#define CPSR_MODE_MASK 0x0000001f
|
||||
#define CPSR_MODE_USER 0x00000010
|
||||
#define CPSR_MODE_FIQ 0x00000011
|
||||
#define CPSR_MODE_IRQ 0x00000012
|
||||
#define CPSR_MODE_SUPERVISOR 0x00000013
|
||||
#define CPSR_MODE_MONITOR 0x00000016
|
||||
#define CPSR_MODE_ABORT 0x00000017
|
||||
#define CPSR_MODE_UNDEFINED 0x0000001b
|
||||
#define CPSR_MODE_SYSTEM 0x0000001f
|
||||
#define CPSR_THUMB_ENABLED (1 << 5)
|
||||
#define CPSR_FIQ_DISABLED (1 << 6)
|
||||
#define CPSR_IRQ_DISABLED (1 << 7)
|
||||
#define CPSR_ASYNC_ABORT_DISABLED (1 << 8)
|
||||
#define CPSR_BE_ENABLED (1 << 9)
|
||||
#define CPSR_IT2_MASK 0x0000FC00
|
||||
#define CPSR_IT2_SHIFT 10
|
||||
#define CPSR_GE_MASK 0x000F0000
|
||||
#define CPSR_GE_SHIFT 16
|
||||
#define CPSR_JAZZLE_ENABLED (1 << 24)
|
||||
#define CPSR_IT1_MASK 0x06000000
|
||||
#define CPSR_IT1_SHIFT 25
|
||||
#define CPSR_COND_OVERFLOW_MASK (1 << 28)
|
||||
#define CPSR_COND_OVERFLOW_SHIFT 28
|
||||
#define CPSR_COND_CARRY_MASK (1 << 29)
|
||||
#define CPSR_COND_CARRY_SHIFT 29
|
||||
#define CPSR_COND_ZERO_MASK (1 << 30)
|
||||
#define CPSR_COND_ZERO_SHIFT 30
|
||||
#define CPSR_COND_NEGATIVE_MASK (1 << 31)
|
||||
#define CPSR_COND_NEGATIVE_SHIFT 31
|
||||
|
||||
#endif /* __ARM_DEFINES_H_ */
|
||||
151
ports/armv7a/arm_entry.S
Normal file
151
ports/armv7a/arm_entry.S
Normal file
@@ -0,0 +1,151 @@
|
||||
/*
|
||||
* Copyright (c) 2011, Anup Patel. 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 <arm_asm_macro.h>
|
||||
|
||||
.section .expvect, "ax", %progbits
|
||||
.globl _start_vect
|
||||
_start_vect:
|
||||
ldr pc, __reset
|
||||
ldr pc, __undefined_instruction
|
||||
ldr pc, __software_interrupt
|
||||
ldr pc, __prefetch_abort
|
||||
ldr pc, __data_abort
|
||||
ldr pc, __not_used
|
||||
ldr pc, __irq
|
||||
ldr pc, __fiq
|
||||
__reset:
|
||||
.word _reset
|
||||
__undefined_instruction:
|
||||
.word _undefined_instruction
|
||||
__software_interrupt:
|
||||
.word _software_interrupt
|
||||
__prefetch_abort:
|
||||
.word _prefetch_abort
|
||||
__data_abort:
|
||||
.word _data_abort
|
||||
__not_used:
|
||||
.word _not_used
|
||||
__irq:
|
||||
.word _irq
|
||||
__fiq:
|
||||
.word _fiq
|
||||
.global _end_vect
|
||||
_end_vect:
|
||||
|
||||
__initial_stack_end:
|
||||
.word _initial_stack_end
|
||||
|
||||
.globl _reset
|
||||
_reset:
|
||||
/* Clear a register for temporary usage */
|
||||
mov r8, #0
|
||||
/* Disable IRQ & FIQ */
|
||||
cpsid if
|
||||
/* Set Supervisor Mode Stack */
|
||||
SET_CURRENT_MODE CPSR_MODE_SUPERVISOR
|
||||
SET_CURRENT_STACK __initial_stack_end
|
||||
/* Set Undefined Mode Stack */
|
||||
SET_CURRENT_MODE CPSR_MODE_UNDEFINED
|
||||
SET_CURRENT_STACK __initial_stack_end
|
||||
/* Set Abort Mode Stack */
|
||||
SET_CURRENT_MODE CPSR_MODE_ABORT
|
||||
SET_CURRENT_STACK __initial_stack_end
|
||||
/* Set IRQ Mode Stack */
|
||||
SET_CURRENT_MODE CPSR_MODE_IRQ
|
||||
SET_CURRENT_STACK __initial_stack_end
|
||||
/* Set FIQ Mode Stack */
|
||||
SET_CURRENT_MODE CPSR_MODE_FIQ
|
||||
SET_CURRENT_STACK __initial_stack_end
|
||||
/* Set System Mode Stack */
|
||||
SET_CURRENT_MODE CPSR_MODE_SYSTEM
|
||||
SET_CURRENT_STACK __initial_stack_end
|
||||
/* Set to Supervisor Mode */
|
||||
SET_CURRENT_MODE CPSR_MODE_SUPERVISOR
|
||||
/* Call main function */
|
||||
bl main
|
||||
/* We should never reach here */
|
||||
b .
|
||||
|
||||
START_EXCEPTION_HANDLER _undefined_instruction, 4
|
||||
PUSH_USER_REGS
|
||||
PUSH_BANKED_REGS _undefined_instruction_bankpush_skip
|
||||
CALL_EXCEPTION_CFUNC do_undefined_instruction
|
||||
PULL_BANKED_REGS _undefined_instruction_bankpull_skip
|
||||
PULL_USER_REGS
|
||||
END_EXCEPTION_HANDLER
|
||||
|
||||
START_EXCEPTION_HANDLER _software_interrupt, 4
|
||||
PUSH_USER_REGS
|
||||
PUSH_BANKED_REGS _software_interrupt_bankpush_skip
|
||||
CALL_EXCEPTION_CFUNC do_software_interrupt
|
||||
PULL_BANKED_REGS _software_interrupt_bankpull_skip
|
||||
PULL_USER_REGS
|
||||
END_EXCEPTION_HANDLER
|
||||
|
||||
START_EXCEPTION_HANDLER _prefetch_abort, 4
|
||||
PUSH_USER_REGS
|
||||
PUSH_BANKED_REGS _prefetch_abort_bankpush_skip
|
||||
CALL_EXCEPTION_CFUNC do_prefetch_abort
|
||||
PULL_BANKED_REGS _prefetch_abort_bankpull_skip
|
||||
PULL_USER_REGS
|
||||
END_EXCEPTION_HANDLER
|
||||
|
||||
START_EXCEPTION_HANDLER _data_abort, 8
|
||||
PUSH_USER_REGS
|
||||
PUSH_BANKED_REGS _data_abort_bankpush_skip
|
||||
CALL_EXCEPTION_CFUNC do_data_abort
|
||||
PULL_BANKED_REGS _data_abort_bankpull_skip
|
||||
PULL_USER_REGS
|
||||
END_EXCEPTION_HANDLER
|
||||
|
||||
START_EXCEPTION_HANDLER _not_used, 4
|
||||
PUSH_USER_REGS
|
||||
PUSH_BANKED_REGS _not_used_bankpush_skip
|
||||
CALL_EXCEPTION_CFUNC do_not_used
|
||||
PULL_BANKED_REGS _not_used_bankpull_skip
|
||||
PULL_USER_REGS
|
||||
END_EXCEPTION_HANDLER
|
||||
|
||||
START_EXCEPTION_HANDLER _irq, 4
|
||||
PUSH_USER_REGS
|
||||
PUSH_BANKED_REGS _irq_bankpush_skip
|
||||
CALL_EXCEPTION_CFUNC do_irq
|
||||
PULL_BANKED_REGS _irq_bankpull_skip
|
||||
PULL_USER_REGS
|
||||
END_EXCEPTION_HANDLER
|
||||
|
||||
START_EXCEPTION_HANDLER _fiq, 4
|
||||
PUSH_USER_REGS
|
||||
PUSH_BANKED_REGS _fiq_bankpush_skip
|
||||
CALL_EXCEPTION_CFUNC do_fiq
|
||||
PULL_BANKED_REGS _fiq_bankpull_skip
|
||||
PULL_USER_REGS
|
||||
END_EXCEPTION_HANDLER
|
||||
|
||||
45
ports/armv7a/arm_io.h
Normal file
45
ports/armv7a/arm_io.h
Normal file
@@ -0,0 +1,45 @@
|
||||
/*
|
||||
* Copyright (c) 2011, Anup Patel. 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 __ARM_IO_H_
|
||||
#define __ARM_IO_H_
|
||||
|
||||
#include <atomport.h>
|
||||
|
||||
static inline uint32_t arm_readl(void * addr)
|
||||
{
|
||||
return *((uint32_t *)addr);
|
||||
}
|
||||
|
||||
static inline void arm_writel(uint32_t data, void * addr)
|
||||
{
|
||||
*((uint32_t *)addr) = data;
|
||||
}
|
||||
|
||||
#endif /* __ARM_IO_H_ */
|
||||
201
ports/armv7a/arm_irq.c
Normal file
201
ports/armv7a/arm_irq.c
Normal file
@@ -0,0 +1,201 @@
|
||||
/*
|
||||
* Copyright (c) 2011, Anup Patel. 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 <atom.h>
|
||||
#include <arm_config.h>
|
||||
#include <arm_pic.h>
|
||||
#include <arm_irq.h>
|
||||
|
||||
arm_irq_handler_t irq_hndls[NR_IRQS_PBA8];
|
||||
|
||||
void do_undefined_instruction(pt_regs_t *regs)
|
||||
{
|
||||
/* Call the interrupt entry routine */
|
||||
atomIntEnter();
|
||||
|
||||
/* Call the interrupt exit routine */
|
||||
atomIntExit(TRUE);
|
||||
}
|
||||
|
||||
void do_software_interrupt(pt_regs_t *regs)
|
||||
{
|
||||
/* Call the interrupt entry routine */
|
||||
atomIntEnter();
|
||||
|
||||
/* Call the interrupt exit routine */
|
||||
atomIntExit(TRUE);
|
||||
}
|
||||
|
||||
void do_prefetch_abort(pt_regs_t *regs)
|
||||
{
|
||||
/* Call the interrupt entry routine */
|
||||
atomIntEnter();
|
||||
|
||||
/* Call the interrupt exit routine */
|
||||
atomIntExit(TRUE);
|
||||
}
|
||||
|
||||
void do_data_abort(pt_regs_t *regs)
|
||||
{
|
||||
/* Call the interrupt entry routine */
|
||||
atomIntEnter();
|
||||
|
||||
/* Call the interrupt exit routine */
|
||||
atomIntExit(TRUE);
|
||||
}
|
||||
|
||||
void do_not_used(pt_regs_t *regs)
|
||||
{
|
||||
/* Call the interrupt entry routine */
|
||||
atomIntEnter();
|
||||
|
||||
/* Call the interrupt exit routine */
|
||||
atomIntExit(TRUE);
|
||||
}
|
||||
|
||||
void do_irq(pt_regs_t *uregs)
|
||||
{
|
||||
int rc = 0;
|
||||
int irq = arm_pic_active_irq();
|
||||
|
||||
/* Call the interrupt entry routine */
|
||||
atomIntEnter();
|
||||
|
||||
if (-1 < irq) {
|
||||
if (irq_hndls[irq]) {
|
||||
rc = irq_hndls[irq](irq, uregs);
|
||||
if (rc) {
|
||||
while (1);
|
||||
}
|
||||
}
|
||||
rc = arm_pic_ack_irq(irq);
|
||||
if (rc) {
|
||||
while (1);
|
||||
}
|
||||
}
|
||||
|
||||
/* Call the interrupt exit routine */
|
||||
atomIntExit(TRUE);
|
||||
}
|
||||
|
||||
void do_fiq(pt_regs_t *uregs)
|
||||
{
|
||||
/* Call the interrupt entry routine */
|
||||
atomIntEnter();
|
||||
|
||||
/* Call the interrupt exit routine */
|
||||
atomIntExit(TRUE);
|
||||
}
|
||||
|
||||
void arm_irq_init(void)
|
||||
{
|
||||
extern uint32_t _start_vect[];
|
||||
uint32_t *vectors = (uint32_t *)NULL;
|
||||
uint32_t *vectors_data = vectors + CPU_IRQ_NR;
|
||||
int vec;
|
||||
|
||||
/*
|
||||
* Loop through the vectors we're taking over, and copy the
|
||||
* vector's insn and data word.
|
||||
*/
|
||||
for (vec = 0; vec < CPU_IRQ_NR; vec++) {
|
||||
vectors[vec] = _start_vect[vec];
|
||||
vectors_data[vec] = _start_vect[vec+CPU_IRQ_NR];
|
||||
}
|
||||
|
||||
/*
|
||||
* Check if verctors are set properly
|
||||
*/
|
||||
for (vec = 0; vec < CPU_IRQ_NR; vec++) {
|
||||
if ((vectors[vec] != _start_vect[vec]) ||
|
||||
(vectors_data[vec] != _start_vect[vec+CPU_IRQ_NR])) {
|
||||
/* Hang */
|
||||
while(1);
|
||||
}
|
||||
}
|
||||
|
||||
/*
|
||||
* Reset irq handlers
|
||||
*/
|
||||
for (vec = 0; vec < NR_IRQS_PBA8; vec++) {
|
||||
irq_hndls[vec] = NULL;
|
||||
}
|
||||
|
||||
/*
|
||||
* Initialize Generic Interrupt Controller
|
||||
*/
|
||||
vec = arm_pic_init();
|
||||
if (vec) {
|
||||
while(1);
|
||||
}
|
||||
}
|
||||
|
||||
void arm_irq_register(uint32_t irq, arm_irq_handler_t hndl)
|
||||
{
|
||||
int rc = 0;
|
||||
if (irq < NR_IRQS_PBA8) {
|
||||
irq_hndls[irq] = hndl;
|
||||
if (irq_hndls[irq]) {
|
||||
rc = arm_pic_unmask(irq);
|
||||
if (rc) {
|
||||
while (1);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
void arm_irq_enable(void)
|
||||
{
|
||||
__asm( "cpsie if" );
|
||||
}
|
||||
|
||||
void arm_irq_disable(void)
|
||||
{
|
||||
__asm( "cpsid if" );
|
||||
}
|
||||
|
||||
irq_flags_t arm_irq_save(void)
|
||||
{
|
||||
unsigned long retval;
|
||||
|
||||
asm volatile (" mrs %0, cpsr\n\t" " cpsid i" /* Syntax CPSID <iflags> {, #<p_mode>}
|
||||
* Note: This instruction is supported
|
||||
* from ARM6 and above
|
||||
*/
|
||||
:"=r" (retval)::"memory", "cc");
|
||||
|
||||
return retval;
|
||||
}
|
||||
|
||||
void arm_irq_restore(irq_flags_t flags)
|
||||
{
|
||||
asm volatile (" msr cpsr_c, %0"::"r" (flags)
|
||||
:"memory", "cc");
|
||||
}
|
||||
|
||||
58
ports/armv7a/arm_irq.h
Normal file
58
ports/armv7a/arm_irq.h
Normal file
@@ -0,0 +1,58 @@
|
||||
/*
|
||||
* Copyright (c) 2011, Anup Patel. 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 __ARM_IRQ_H
|
||||
#define __ARM_IRQ_H
|
||||
|
||||
#include "atomport.h"
|
||||
#include "atomport-private.h"
|
||||
#include "arm_defines.h"
|
||||
|
||||
typedef int (*arm_irq_handler_t) (uint32_t irq_no, pt_regs_t * regs);
|
||||
|
||||
#define CPU_IRQ_NR 8
|
||||
|
||||
/** IRQ Numbers */
|
||||
#define ARM_RESET_IRQ 0
|
||||
#define ARM_UNDEF_INST_IRQ 1
|
||||
#define ARM_SOFT_IRQ 2
|
||||
#define ARM_PREFETCH_ABORT_IRQ 3
|
||||
#define ARM_DATA_ABORT_IRQ 4
|
||||
#define ARM_NOT_USED_IRQ 5
|
||||
#define ARM_EXTERNAL_IRQ 6
|
||||
#define ARM_EXTERNAL_FIQ 7
|
||||
|
||||
void arm_irq_init(void);
|
||||
void arm_irq_register(uint32_t irq_no, arm_irq_handler_t hndl);
|
||||
void arm_irq_enable(void);
|
||||
void arm_irq_disable(void);
|
||||
irq_flags_t arm_irq_save(void);
|
||||
void arm_irq_restore(irq_flags_t flags);
|
||||
|
||||
#endif /* __ARM_IRQ_H */
|
||||
233
ports/armv7a/arm_main.c
Normal file
233
ports/armv7a/arm_main.c
Normal file
@@ -0,0 +1,233 @@
|
||||
/*
|
||||
* Copyright (c) 2011, Anup Patel. 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 "atom.h"
|
||||
#include "atomport.h"
|
||||
#include "atomtests.h"
|
||||
#include "atomtimer.h"
|
||||
#include "system.h"
|
||||
#include "arm_irq.h"
|
||||
#include "arm_timer.h"
|
||||
#include "arm_uart.h"
|
||||
|
||||
/* Constants */
|
||||
|
||||
/*
|
||||
* Idle thread stack size
|
||||
*
|
||||
* This needs to be large enough to handle any interrupt handlers
|
||||
* and callbacks called by interrupt handlers (e.g. user-created
|
||||
* timer callbacks) as well as the saving of all context when
|
||||
* switching away from this thread.
|
||||
*
|
||||
* In this case, the idle stack is allocated on the BSS via the
|
||||
* idle_thread_stack[] byte array.
|
||||
*/
|
||||
#define IDLE_STACK_SIZE_BYTES 8192
|
||||
|
||||
|
||||
/*
|
||||
* Main thread stack size
|
||||
*
|
||||
* Note that this is not a required OS kernel thread - you will replace
|
||||
* this with your own application thread.
|
||||
*
|
||||
* In this case the Main thread is responsible for calling out to the
|
||||
* test routines. Once a test routine has finished, the test status is
|
||||
* printed out on the UART and the thread remains running in a loop
|
||||
* flashing a LED.
|
||||
*
|
||||
* The Main thread stack generally needs to be larger than the idle
|
||||
* thread stack, as not only does it need to store interrupt handler
|
||||
* stack saves and context switch saves, but the application main thread
|
||||
* will generally be carrying out more nested function calls and require
|
||||
* stack for application code local variables etc.
|
||||
*
|
||||
* With all OS tests implemented to date on the AVR, the Main thread
|
||||
* stack has not exceeded 198 bytes. To allow all tests to run we set
|
||||
* a minimum main thread stack size of 204 bytes. This may increase in
|
||||
* future as the codebase changes but for the time being is enough to
|
||||
* cope with all of the automated tests.
|
||||
*/
|
||||
#define MAIN_STACK_SIZE_BYTES 8192
|
||||
|
||||
|
||||
/*
|
||||
* Startup code stack
|
||||
*
|
||||
* Some stack space is required at initial startup for running the main()
|
||||
* routine. This stack space is only temporarily required at first bootup
|
||||
* and is no longer required as soon as the OS is started. By default
|
||||
* GCC sets this to the top of RAM (RAMEND) and it grows down from there.
|
||||
* Because we only need this temporarily, though, it would be wasteful to
|
||||
* set aside a region at the top of RAM which is not used during runtime.
|
||||
*
|
||||
* What we do here is to reuse part of the idle thread's stack during
|
||||
* initial startup. As soon as we enter the main() routine we move the
|
||||
* stack pointer to half-way down the idle thread's stack. This is used
|
||||
* temporarily while calls are made to atomOSInit(), atomThreadCreate()
|
||||
* and atomOSStart(). Once the OS is started this stack area is no
|
||||
* longer required, and can be used for its original purpose (for the
|
||||
* idle thread's stack).
|
||||
*
|
||||
* This does mean, however, that we cannot monitor the stack usage of the
|
||||
* idle thread. Stack usage is monitored by prefilling the stack with a
|
||||
* known value, and we are obliterating some of that prefilled area by
|
||||
* using it as our startup stack, so we cannot use the stack-checking API
|
||||
* to get a true picture of idle thread stack usage. If you wish to
|
||||
* monitor idle thread stack usage for your applications then you are
|
||||
* free to use a different region for the startup stack (e.g. set aside
|
||||
* an area permanently, or place it somewhere you know you can reuse
|
||||
* later in the application). For the time being, this method gives us a
|
||||
* simple way of reducing the memory consumption without having to add
|
||||
* any special AVR-specific considerations to the automated test
|
||||
* applications.
|
||||
*
|
||||
* This optimisation was required to allow some of the larger automated
|
||||
* test modules to run on devices with 1KB of RAM. You should avoid doing
|
||||
* this if you can afford to set aside 64 bytes or so, or if you are
|
||||
* writing your own applications in which you have further control over
|
||||
* where data is located.
|
||||
*/
|
||||
|
||||
|
||||
/* Local data */
|
||||
|
||||
/* Application threads' TCBs */
|
||||
static ATOM_TCB main_tcb;
|
||||
|
||||
/* Main thread's stack area */
|
||||
static uint8_t main_thread_stack[MAIN_STACK_SIZE_BYTES];
|
||||
|
||||
/* Idle thread's stack area */
|
||||
static uint8_t idle_thread_stack[IDLE_STACK_SIZE_BYTES];
|
||||
|
||||
/* Forward declarations */
|
||||
static void main_thread_func (uint32_t data);
|
||||
|
||||
/**
|
||||
* \b main
|
||||
*
|
||||
* Program entry point.
|
||||
*
|
||||
* Sets up the AVR hardware resources (system tick timer interrupt) necessary
|
||||
* for the OS to be started. Creates an application thread and starts the OS.
|
||||
*/
|
||||
|
||||
int main ( void )
|
||||
{
|
||||
int8_t status;
|
||||
|
||||
/**
|
||||
* Note: to protect OS structures and data during initialisation,
|
||||
* interrupts must remain disabled until the first thread
|
||||
* has been restored. They are reenabled at the very end of
|
||||
* the first thread restore, at which point it is safe for a
|
||||
* reschedule to take place.
|
||||
*/
|
||||
|
||||
/**
|
||||
* Initialise the OS before creating our threads.
|
||||
*
|
||||
* Note that we tell the OS that the idle stack is half its actual
|
||||
* size. This prevents it prefilling the bottom half with known
|
||||
* values for stack-checkig purposes, which we cannot allow because
|
||||
* we are temporarily using it for our own stack. The remainder will
|
||||
* still be available once the OS is started, this only prevents the
|
||||
* OS from prefilling it.
|
||||
*
|
||||
* If you are not reusing the idle thread's stack during startup then
|
||||
* you should pass in the correct size here.
|
||||
*/
|
||||
status = atomOSInit(&idle_thread_stack[0],
|
||||
IDLE_STACK_SIZE_BYTES, 0);
|
||||
if (status == ATOM_OK)
|
||||
{
|
||||
arm_irq_init();
|
||||
|
||||
arm_timer_init(SYSTEM_TICKS_PER_SEC);
|
||||
|
||||
arm_uart_init();
|
||||
|
||||
/* Create an application thread */
|
||||
status = atomThreadCreate(&main_tcb,
|
||||
TEST_THREAD_PRIO, main_thread_func, 0,
|
||||
&main_thread_stack[0],
|
||||
MAIN_STACK_SIZE_BYTES, 0);
|
||||
if (status == ATOM_OK)
|
||||
{
|
||||
arm_timer_enable();
|
||||
|
||||
/**
|
||||
* First application thread successfully created. It is
|
||||
* now possible to start the OS. Execution will not return
|
||||
* from atomOSStart(), which will restore the context of
|
||||
* our application thread and start executing it.
|
||||
*
|
||||
* Note that interrupts are still disabled at this point.
|
||||
* They will be enabled as we restore and execute our first
|
||||
* thread in archFirstThreadRestore().
|
||||
*/
|
||||
atomOSStart();
|
||||
}
|
||||
}
|
||||
|
||||
while (1)
|
||||
;
|
||||
|
||||
/* There was an error starting the OS if we reach here */
|
||||
return (0);
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* \b main_thread_func
|
||||
*
|
||||
* Entry point for main application thread.
|
||||
*
|
||||
* This is the first thread that will be executed when the OS is started.
|
||||
*
|
||||
* @param[in] data Unused (optional thread entry parameter)
|
||||
*
|
||||
* @return None
|
||||
*/
|
||||
static void main_thread_func (uint32_t data)
|
||||
{
|
||||
/* Put a message out on the UART */
|
||||
printk("Test Started ... ");
|
||||
if (test_start() != 0) {
|
||||
printk("FAILED!\n");
|
||||
} else {
|
||||
printk("SUCCESS!\n");
|
||||
}
|
||||
printk("Reset your board !!!!!");
|
||||
/* Test finished so just hang !!! */
|
||||
while (1)
|
||||
;
|
||||
}
|
||||
73
ports/armv7a/atomport-asm.S
Normal file
73
ports/armv7a/atomport-asm.S
Normal file
@@ -0,0 +1,73 @@
|
||||
/*
|
||||
* Copyright (c) 2011, Anup Patel for Atomthreads Project.
|
||||
* 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 <arm_asm_macro.h>
|
||||
|
||||
.section .text
|
||||
|
||||
/**
|
||||
* int archSetJump(pt_regs_t *regs)
|
||||
*/
|
||||
.globl archSetJump
|
||||
archSetJump:
|
||||
add r0, r0, #(4 * 16)
|
||||
str lr, [r0]
|
||||
sub r0, r0, #(4 * 14)
|
||||
stm r0, {r1-r14}
|
||||
mov r0, r0 /* NOP */
|
||||
str r2, [r1]
|
||||
mov r2, #0
|
||||
sub r0, r0, #4
|
||||
str r2, [r0]
|
||||
mrs r2, cpsr_all
|
||||
sub r0, r0, #4
|
||||
str r2, [r0]
|
||||
ldr r2, [r1]
|
||||
mov r0, #1
|
||||
bx lr
|
||||
|
||||
/**
|
||||
* void archLongJump(pt_regs_t *regs)
|
||||
*/
|
||||
.globl archLongJump
|
||||
archLongJump:
|
||||
mrs r1, cpsr_all
|
||||
SET_CURRENT_MODE CPSR_MODE_IRQ
|
||||
mov sp, r0
|
||||
SET_CURRENT_MODE CPSR_MODE_FIQ
|
||||
mov sp, r0
|
||||
msr cpsr_all, r1
|
||||
ldr r1, [r0], #4 /* Get CPSR from stack */
|
||||
orr r2, r1, #(CPSR_IRQ_DISABLED | CPSR_FIQ_DISABLED)
|
||||
msr cpsr_all, r2
|
||||
msr spsr_all, r1
|
||||
ldm r0, {r0-r15}^
|
||||
mov r0, r0 /* NOP */
|
||||
|
||||
54
ports/armv7a/atomport-private.h
Normal file
54
ports/armv7a/atomport-private.h
Normal file
@@ -0,0 +1,54 @@
|
||||
/*
|
||||
* Copyright (c) 2011, Atomthreads Project. 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 __ATOMPORT_PRIVATE_H_
|
||||
#define __ATOMPORT_PRIVATE_H_
|
||||
|
||||
typedef unsigned int irq_flags_t;
|
||||
typedef unsigned int virtual_addr_t;
|
||||
typedef unsigned int virtual_size_t;
|
||||
typedef unsigned int physical_addr_t;
|
||||
typedef unsigned int physical_size_t;
|
||||
typedef unsigned int clock_freq_t;
|
||||
typedef unsigned long long jiffies_t;
|
||||
|
||||
struct pt_regs {
|
||||
uint32_t cpsr; // Current Program Status
|
||||
uint32_t gpr[13]; // R0 - R12
|
||||
uint32_t sp;
|
||||
uint32_t lr;
|
||||
uint32_t pc;
|
||||
} __attribute ((packed)) ;
|
||||
typedef struct pt_regs pt_regs_t;
|
||||
|
||||
/* Function prototypes */
|
||||
extern int archSetJump(pt_regs_t *regs, uint32_t *tmp);
|
||||
extern void archLongJump(pt_regs_t *regs);
|
||||
|
||||
#endif /* __ATOMPORT_PRIVATE_H_ */
|
||||
50
ports/armv7a/atomport-tests.h
Normal file
50
ports/armv7a/atomport-tests.h
Normal file
@@ -0,0 +1,50 @@
|
||||
/*
|
||||
* Copyright (c) 2011, Anup Patel. 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_PORT_TESTS_H
|
||||
#define __ATOM_PORT_TESTS_H
|
||||
|
||||
/* Include Atomthreads kernel API */
|
||||
#include "atom.h"
|
||||
|
||||
/* Prerequisite include for ATOMLOG() macro (via printf) */
|
||||
#include "printk.h"
|
||||
|
||||
/* Logger macro for viewing test results */
|
||||
#define ATOMLOG printk
|
||||
#define _STR
|
||||
|
||||
/* Default thread stack size (in bytes) */
|
||||
#define TEST_THREAD_STACK_SIZE 8192
|
||||
|
||||
/* Uncomment to enable logging of stack usage to UART */
|
||||
/* #define TESTS_LOG_STACK_USAGE */
|
||||
|
||||
#endif /* __ATOM_PORT_TESTS_H */
|
||||
|
||||
112
ports/armv7a/atomport.c
Normal file
112
ports/armv7a/atomport.c
Normal file
@@ -0,0 +1,112 @@
|
||||
/*
|
||||
* Copyright (c) 2011, Anup Patel for Atomthreads Project.
|
||||
* 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 "atom.h"
|
||||
#include "atomport.h"
|
||||
#include "atomport-private.h"
|
||||
#include "string.h"
|
||||
#include "arm_defines.h"
|
||||
|
||||
/**
|
||||
* This function initialises each thread's stack during creation, before the
|
||||
* thread is first run. New threads are scheduled in using the same
|
||||
* context-switch function used for threads which were previously scheduled
|
||||
* out, therefore this function should set up a stack context which looks
|
||||
* much like a thread which has been scheduled out and had its context saved.
|
||||
* We fill part of the stack with those registers which are involved in the
|
||||
* context switch, including appropriate stack or register contents to cause
|
||||
* the thread to branch to its entry point function when it is scheduled in.
|
||||
*
|
||||
* Interrupts should also be enabled whenever a thread is restored, hence
|
||||
* ports may wish to explicitly include the interrupt-enable register here
|
||||
* which will be restored when the thread is scheduled in. Other methods
|
||||
* can be used to enable interrupts, however, without explicitly storing
|
||||
* it in the thread's context.
|
||||
*/
|
||||
void archThreadContextInit (ATOM_TCB *tcb_ptr, void *stack_top,
|
||||
void (*entry_point)(UINT32),
|
||||
UINT32 entry_param)
|
||||
{
|
||||
int i;
|
||||
pt_regs_t *regs = (pt_regs_t *)((uint32_t)stack_top - sizeof(pt_regs_t));
|
||||
|
||||
tcb_ptr->sp_save_ptr = stack_top;
|
||||
regs->cpsr = CPSR_COND_ZERO_MASK |
|
||||
CPSR_ASYNC_ABORT_DISABLED | CPSR_MODE_SUPERVISOR;
|
||||
regs->gpr[0] = entry_param;
|
||||
for (i = 1; i < 13; i++) {
|
||||
regs->gpr[i] = 0x0;
|
||||
}
|
||||
regs->sp = (uint32_t)stack_top - sizeof(pt_regs_t) - 1024;
|
||||
regs->lr = (uint32_t)entry_point;
|
||||
regs->pc = (uint32_t)entry_point;
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* archFirstThreadRestore(ATOM_TCB *new_tcb)
|
||||
*
|
||||
* This function is responsible for restoring and starting the first
|
||||
* thread the OS runs. It expects to find the thread context exactly
|
||||
* as it would be if a context save had previously taken place on it.
|
||||
* The only real difference between this and the archContextSwitch()
|
||||
* routine is that there is no previous thread for which context must
|
||||
* be saved.
|
||||
*
|
||||
* The final action this function must do is to restore interrupts.
|
||||
*/
|
||||
void archFirstThreadRestore(ATOM_TCB *new_tcb)
|
||||
{
|
||||
pt_regs_t *regs = (pt_regs_t *)((uint32_t)new_tcb->sp_save_ptr
|
||||
- sizeof(pt_regs_t));
|
||||
archLongJump(regs);
|
||||
}
|
||||
|
||||
/**
|
||||
* Function that performs the contextSwitch. Whether its a voluntary release
|
||||
* of CPU by thread or a pre-emption, under both conditions this function is
|
||||
* called. The signature is as follows:
|
||||
*
|
||||
* archContextSwitch(ATOM_TCB *old_tcb, ATOM_TCB *new_tcb)
|
||||
*/
|
||||
void archContextSwitch(ATOM_TCB *old_tcb, ATOM_TCB *new_tcb)
|
||||
{
|
||||
uint32_t tmp = 0x0, lr = 0x0;
|
||||
pt_regs_t *old_regs = (pt_regs_t *)((uint32_t)old_tcb->sp_save_ptr
|
||||
- sizeof(pt_regs_t));
|
||||
pt_regs_t *new_regs = (pt_regs_t *)((uint32_t)new_tcb->sp_save_ptr
|
||||
- sizeof(pt_regs_t));
|
||||
asm volatile (" mov %0, lr\n\t" :"=r"(lr):);
|
||||
if (archSetJump(old_regs, &tmp)) {
|
||||
old_regs->lr = lr;
|
||||
archLongJump(new_regs);
|
||||
}
|
||||
}
|
||||
|
||||
78
ports/armv7a/atomport.h
Normal file
78
ports/armv7a/atomport.h
Normal file
@@ -0,0 +1,78 @@
|
||||
/*
|
||||
* Copyright (c) 2011, Anup Patel for Atomthreads Project.
|
||||
* 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_PORT_H
|
||||
#define __ATOM_PORT_H
|
||||
|
||||
|
||||
/* Required number of system ticks per second (normally 100 for 10ms tick) */
|
||||
#define SYSTEM_TICKS_PER_SEC 1000
|
||||
|
||||
/**
|
||||
* Definition of NULL. stddef.h not available on this platform.
|
||||
*/
|
||||
#define NULL ((void *)(0))
|
||||
|
||||
/* Size of each stack entry / stack alignment size (32 bits on ARMv7A) */
|
||||
#define STACK_ALIGN_SIZE sizeof(uint32_t)
|
||||
|
||||
/**
|
||||
* Architecture-specific types.
|
||||
* Provide stdint.h style types.
|
||||
*/
|
||||
#define uint8_t unsigned char
|
||||
#define uint16_t unsigned short
|
||||
#define uint32_t unsigned int
|
||||
#define uint64_t unsigned long long
|
||||
#define int8_t signed char
|
||||
#define int16_t signed short
|
||||
#define int32_t signed int
|
||||
#define int64_t long long
|
||||
#define size_t unsigned long
|
||||
#define POINTER void *
|
||||
#define UINT32 uint32_t
|
||||
|
||||
/**
|
||||
* Critical region protection: this should disable interrupts
|
||||
* to protect OS data structures during modification. It must
|
||||
* allow nested calls, which means that interrupts should only
|
||||
* be re-enabled when the outer CRITICAL_END() is reached.
|
||||
*/
|
||||
#include "arm_irq.h"
|
||||
#include "atomport-private.h"
|
||||
#define CRITICAL_STORE irq_flags_t status_flags
|
||||
#define CRITICAL_START() status_flags = arm_irq_save();
|
||||
#define CRITICAL_END() arm_irq_restore(status_flags);
|
||||
|
||||
/* Uncomment to enable stack-checking */
|
||||
/* #define ATOM_STACK_CHECKING */
|
||||
|
||||
|
||||
#endif /* __ATOM_PORT_H */
|
||||
9
ports/armv7a/pb-a8/Makefile
Normal file
9
ports/armv7a/pb-a8/Makefile
Normal file
@@ -0,0 +1,9 @@
|
||||
##########################
|
||||
# Board specific objects #
|
||||
##########################
|
||||
|
||||
# board object files
|
||||
objs += pb-a8/arm_pic.o
|
||||
objs += pb-a8/arm_timer.o
|
||||
objs += pb-a8/arm_uart.o
|
||||
|
||||
147
ports/armv7a/pb-a8/arm_config.h
Normal file
147
ports/armv7a/pb-a8/arm_config.h
Normal file
@@ -0,0 +1,147 @@
|
||||
/**
|
||||
* Copyright (c) 2011 Anup Patel.
|
||||
* All rights reserved.
|
||||
*
|
||||
* This program is free software; you can redistribute it and/or modify
|
||||
* it under the terms of the GNU General Public License as published by
|
||||
* the Free Software Foundation; either version 2, or (at your option)
|
||||
* any later version.
|
||||
*
|
||||
* This program 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 General Public License for more details.
|
||||
*
|
||||
* You should have received a copy of the GNU General Public License
|
||||
* along with this program; if not, write to the Free Software
|
||||
* Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
|
||||
*
|
||||
* @file arm_config.h
|
||||
* @version 1.0
|
||||
* @author Anup Patel (anup@brainfault.org)
|
||||
* @brief ARM Platform Configuration Header
|
||||
*/
|
||||
#ifndef _ARM_CONFIG_H__
|
||||
#define _ARM_CONFIG_H__
|
||||
|
||||
/*
|
||||
* Peripheral addresses
|
||||
*/
|
||||
#define REALVIEW_PBA8_UART0_BASE 0x10009000 /* UART 0 */
|
||||
#define REALVIEW_PBA8_UART1_BASE 0x1000A000 /* UART 1 */
|
||||
#define REALVIEW_PBA8_UART2_BASE 0x1000B000 /* UART 2 */
|
||||
#define REALVIEW_PBA8_UART3_BASE 0x1000C000 /* UART 3 */
|
||||
#define REALVIEW_PBA8_SSP_BASE 0x1000D000 /* Synchronous Serial Port */
|
||||
#define REALVIEW_PBA8_WATCHDOG0_BASE 0x1000F000 /* Watchdog 0 */
|
||||
#define REALVIEW_PBA8_WATCHDOG_BASE 0x10010000 /* watchdog interface */
|
||||
#define REALVIEW_PBA8_TIMER0_1_BASE 0x10011000 /* Timer 0 and 1 */
|
||||
#define REALVIEW_PBA8_TIMER2_3_BASE 0x10012000 /* Timer 2 and 3 */
|
||||
#define REALVIEW_PBA8_GPIO0_BASE 0x10013000 /* GPIO port 0 */
|
||||
#define REALVIEW_PBA8_RTC_BASE 0x10017000 /* Real Time Clock */
|
||||
#define REALVIEW_PBA8_TIMER4_5_BASE 0x10018000 /* Timer 4/5 */
|
||||
#define REALVIEW_PBA8_TIMER6_7_BASE 0x10019000 /* Timer 6/7 */
|
||||
#define REALVIEW_PBA8_SCTL_BASE 0x1001A000 /* System Controller */
|
||||
#define REALVIEW_PBA8_CLCD_BASE 0x10020000 /* CLCD */
|
||||
#define REALVIEW_PBA8_ONB_SRAM_BASE 0x10060000 /* On-board SRAM */
|
||||
#define REALVIEW_PBA8_DMC_BASE 0x100E0000 /* DMC configuration */
|
||||
#define REALVIEW_PBA8_SMC_BASE 0x100E1000 /* SMC configuration */
|
||||
#define REALVIEW_PBA8_CAN_BASE 0x100E2000 /* CAN bus */
|
||||
#define REALVIEW_PBA8_GIC_CPU_BASE 0x1E000000 /* Generic interrupt controller CPU interface */
|
||||
#define REALVIEW_PBA8_FLASH0_BASE 0x40000000
|
||||
#define REALVIEW_PBA8_FLASH0_SIZE SZ_64M
|
||||
#define REALVIEW_PBA8_FLASH1_BASE 0x44000000
|
||||
#define REALVIEW_PBA8_FLASH1_SIZE SZ_64M
|
||||
#define REALVIEW_PBA8_ETH_BASE 0x4E000000 /* Ethernet */
|
||||
#define REALVIEW_PBA8_USB_BASE 0x4F000000 /* USB */
|
||||
#define REALVIEW_PBA8_GIC_DIST_BASE 0x1E001000 /* Generic interrupt controller distributor */
|
||||
#define REALVIEW_PBA8_LT_BASE 0xC0000000 /* Logic Tile expansion */
|
||||
#define REALVIEW_PBA8_SDRAM6_BASE 0x70000000 /* SDRAM bank 6 256MB */
|
||||
#define REALVIEW_PBA8_SDRAM7_BASE 0x80000000 /* SDRAM bank 7 256MB */
|
||||
|
||||
#define REALVIEW_PBA8_SYS_PLD_CTRL1 0x74
|
||||
|
||||
/*
|
||||
* PBA8 PCI regions
|
||||
*/
|
||||
#define REALVIEW_PBA8_PCI_BASE 0x90040000 /* PCI-X Unit base */
|
||||
#define REALVIEW_PBA8_PCI_IO_BASE 0x90050000 /* IO Region on AHB */
|
||||
#define REALVIEW_PBA8_PCI_MEM_BASE 0xA0000000 /* MEM Region on AHB */
|
||||
|
||||
#define REALVIEW_PBA8_PCI_BASE_SIZE 0x10000 /* 16 Kb */
|
||||
#define REALVIEW_PBA8_PCI_IO_SIZE 0x1000 /* 4 Kb */
|
||||
#define REALVIEW_PBA8_PCI_MEM_SIZE 0x20000000 /* 512 MB */
|
||||
|
||||
/*
|
||||
* Irqs
|
||||
*/
|
||||
#define IRQ_PBA8_GIC_START 32
|
||||
|
||||
/* L220
|
||||
#define IRQ_PBA8_L220_EVENT (IRQ_PBA8_GIC_START + 29)
|
||||
#define IRQ_PBA8_L220_SLAVE (IRQ_PBA8_GIC_START + 30)
|
||||
#define IRQ_PBA8_L220_DECODE (IRQ_PBA8_GIC_START + 31)
|
||||
*/
|
||||
|
||||
/*
|
||||
* PB-A8 on-board gic irq sources
|
||||
*/
|
||||
#define IRQ_PBA8_WATCHDOG (IRQ_PBA8_GIC_START + 0) /* Watchdog timer */
|
||||
#define IRQ_PBA8_SOFT (IRQ_PBA8_GIC_START + 1) /* Software interrupt */
|
||||
#define IRQ_PBA8_COMMRx (IRQ_PBA8_GIC_START + 2) /* Debug Comm Rx interrupt */
|
||||
#define IRQ_PBA8_COMMTx (IRQ_PBA8_GIC_START + 3) /* Debug Comm Tx interrupt */
|
||||
#define IRQ_PBA8_TIMER0_1 (IRQ_PBA8_GIC_START + 4) /* Timer 0/1 (default timer) */
|
||||
#define IRQ_PBA8_TIMER2_3 (IRQ_PBA8_GIC_START + 5) /* Timer 2/3 */
|
||||
#define IRQ_PBA8_GPIO0 (IRQ_PBA8_GIC_START + 6) /* GPIO 0 */
|
||||
#define IRQ_PBA8_GPIO1 (IRQ_PBA8_GIC_START + 7) /* GPIO 1 */
|
||||
#define IRQ_PBA8_GPIO2 (IRQ_PBA8_GIC_START + 8) /* GPIO 2 */
|
||||
/* 9 reserved */
|
||||
#define IRQ_PBA8_RTC (IRQ_PBA8_GIC_START + 10) /* Real Time Clock */
|
||||
#define IRQ_PBA8_SSP (IRQ_PBA8_GIC_START + 11) /* Synchronous Serial Port */
|
||||
#define IRQ_PBA8_UART0 (IRQ_PBA8_GIC_START + 12) /* UART 0 on development chip */
|
||||
#define IRQ_PBA8_UART1 (IRQ_PBA8_GIC_START + 13) /* UART 1 on development chip */
|
||||
#define IRQ_PBA8_UART2 (IRQ_PBA8_GIC_START + 14) /* UART 2 on development chip */
|
||||
#define IRQ_PBA8_UART3 (IRQ_PBA8_GIC_START + 15) /* UART 3 on development chip */
|
||||
#define IRQ_PBA8_SCI (IRQ_PBA8_GIC_START + 16) /* Smart Card Interface */
|
||||
#define IRQ_PBA8_MMCI0A (IRQ_PBA8_GIC_START + 17) /* Multimedia Card 0A */
|
||||
#define IRQ_PBA8_MMCI0B (IRQ_PBA8_GIC_START + 18) /* Multimedia Card 0B */
|
||||
#define IRQ_PBA8_AACI (IRQ_PBA8_GIC_START + 19) /* Audio Codec */
|
||||
#define IRQ_PBA8_KMI0 (IRQ_PBA8_GIC_START + 20) /* Keyboard/Mouse port 0 */
|
||||
#define IRQ_PBA8_KMI1 (IRQ_PBA8_GIC_START + 21) /* Keyboard/Mouse port 1 */
|
||||
#define IRQ_PBA8_CHARLCD (IRQ_PBA8_GIC_START + 22) /* Character LCD */
|
||||
#define IRQ_PBA8_CLCD (IRQ_PBA8_GIC_START + 23) /* CLCD controller */
|
||||
#define IRQ_PBA8_DMAC (IRQ_PBA8_GIC_START + 24) /* DMA controller */
|
||||
#define IRQ_PBA8_PWRFAIL (IRQ_PBA8_GIC_START + 25) /* Power failure */
|
||||
#define IRQ_PBA8_PISMO (IRQ_PBA8_GIC_START + 26) /* PISMO interface */
|
||||
#define IRQ_PBA8_DoC (IRQ_PBA8_GIC_START + 27) /* Disk on Chip memory controller */
|
||||
#define IRQ_PBA8_ETH (IRQ_PBA8_GIC_START + 28) /* Ethernet controller */
|
||||
#define IRQ_PBA8_USB (IRQ_PBA8_GIC_START + 29) /* USB controller */
|
||||
#define IRQ_PBA8_TSPEN (IRQ_PBA8_GIC_START + 30) /* Touchscreen pen */
|
||||
#define IRQ_PBA8_TSKPAD (IRQ_PBA8_GIC_START + 31) /* Touchscreen keypad */
|
||||
|
||||
/* ... */
|
||||
#define IRQ_PBA8_PCI0 (IRQ_PBA8_GIC_START + 50)
|
||||
#define IRQ_PBA8_PCI1 (IRQ_PBA8_GIC_START + 51)
|
||||
#define IRQ_PBA8_PCI2 (IRQ_PBA8_GIC_START + 52)
|
||||
#define IRQ_PBA8_PCI3 (IRQ_PBA8_GIC_START + 53)
|
||||
|
||||
#define IRQ_PBA8_SMC -1
|
||||
#define IRQ_PBA8_SCTL -1
|
||||
|
||||
#define NR_GIC_PBA8 1
|
||||
|
||||
/*
|
||||
* Only define NR_IRQS if less than NR_IRQS_PBA8
|
||||
*/
|
||||
#define NR_IRQS_PBA8 (IRQ_PBA8_GIC_START + 64)
|
||||
|
||||
#if !defined(ARM_GIC_NR_IRQS) || (ARM_GIC_NR_IRQS < NR_IRQS_PBA8)
|
||||
#undef ARM_GIC_NR_IRQS
|
||||
#define ARM_GIC_NR_IRQS NR_IRQS_PBA8
|
||||
#endif
|
||||
|
||||
#if !defined(ARM_GIC_MAX_NR) || (REALVIEW_GIC_MAX_NR < NR_GIC_PBA8)
|
||||
#undef ARM_GIC_MAX_NR
|
||||
#define ARM_GIC_MAX_NR NR_GIC_PBA8
|
||||
#endif
|
||||
|
||||
#endif
|
||||
247
ports/armv7a/pb-a8/arm_pic.c
Normal file
247
ports/armv7a/pb-a8/arm_pic.c
Normal file
@@ -0,0 +1,247 @@
|
||||
/*
|
||||
* Copyright (c) 2011, Anup Patel. 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 <arm_config.h>
|
||||
#include <arm_io.h>
|
||||
#include <arm_pic.h>
|
||||
|
||||
#define max(a,b) ((a) < (b) ? (b) : (a))
|
||||
|
||||
struct gic_chip_data {
|
||||
uint32_t irq_offset;
|
||||
virtual_addr_t dist_base;
|
||||
virtual_addr_t cpu_base;
|
||||
};
|
||||
|
||||
static struct gic_chip_data gic_data[ARM_GIC_MAX_NR];
|
||||
|
||||
static inline void arm_gic_write(uint32_t val, virtual_addr_t addr)
|
||||
{
|
||||
arm_writel(val, (void *)(addr));
|
||||
}
|
||||
|
||||
static inline uint32_t arm_gic_read(virtual_addr_t addr)
|
||||
{
|
||||
return arm_readl((void *)(addr));
|
||||
}
|
||||
|
||||
int arm_gic_active_irq(uint32_t gic_nr)
|
||||
{
|
||||
int ret = -1;
|
||||
|
||||
if (ARM_GIC_MAX_NR <= gic_nr) {
|
||||
return -1;
|
||||
}
|
||||
|
||||
ret = arm_gic_read(gic_data[gic_nr].cpu_base +
|
||||
GIC_CPU_INTACK) & 0x3FF;
|
||||
ret += gic_data[gic_nr].irq_offset;
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
int arm_gic_ack_irq(uint32_t gic_nr, uint32_t irq)
|
||||
{
|
||||
uint32_t mask = 1 << (irq % 32);
|
||||
uint32_t gic_irq;
|
||||
|
||||
if (ARM_GIC_MAX_NR <= gic_nr) {
|
||||
return -1;
|
||||
}
|
||||
|
||||
if (irq < gic_data[gic_nr].irq_offset) {
|
||||
return -1;
|
||||
}
|
||||
|
||||
gic_irq = irq - gic_data[gic_nr].irq_offset;
|
||||
|
||||
arm_gic_write(mask, gic_data[gic_nr].dist_base +
|
||||
GIC_DIST_ENABLE_CLEAR + (gic_irq / 32) * 4);
|
||||
arm_gic_write(gic_irq, gic_data[gic_nr].cpu_base + GIC_CPU_EOI);
|
||||
arm_gic_write(mask, gic_data[gic_nr].dist_base +
|
||||
GIC_DIST_ENABLE_SET + (gic_irq / 32) * 4);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
int arm_gic_mask(uint32_t gic_nr, uint32_t irq)
|
||||
{
|
||||
uint32_t mask = 1 << (irq % 32);
|
||||
uint32_t gic_irq;
|
||||
|
||||
if (ARM_GIC_MAX_NR <= gic_nr) {
|
||||
return -1;
|
||||
}
|
||||
|
||||
if (irq < gic_data[gic_nr].irq_offset) {
|
||||
return -1;
|
||||
}
|
||||
|
||||
gic_irq = irq - gic_data[gic_nr].irq_offset;
|
||||
|
||||
arm_gic_write(mask, gic_data[gic_nr].dist_base +
|
||||
GIC_DIST_ENABLE_CLEAR + (gic_irq / 32) * 4);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
int arm_gic_unmask(uint32_t gic_nr, uint32_t irq)
|
||||
{
|
||||
uint32_t mask = 1 << (irq % 32);
|
||||
uint32_t gic_irq;
|
||||
|
||||
if (ARM_GIC_MAX_NR <= gic_nr) {
|
||||
return -1;
|
||||
}
|
||||
|
||||
if (irq < gic_data[gic_nr].irq_offset) {
|
||||
return -1;
|
||||
}
|
||||
|
||||
gic_irq = irq - gic_data[gic_nr].irq_offset;
|
||||
|
||||
arm_gic_write(mask, gic_data[gic_nr].dist_base +
|
||||
GIC_DIST_ENABLE_SET + (gic_irq / 32) * 4);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
int arm_gic_dist_init(uint32_t gic_nr, virtual_addr_t base, uint32_t irq_start)
|
||||
{
|
||||
unsigned int max_irq, i;
|
||||
uint32_t cpumask = 1 << 0; /*smp_processor_id(); */
|
||||
|
||||
if (ARM_GIC_MAX_NR <= gic_nr) {
|
||||
return -1;
|
||||
}
|
||||
|
||||
cpumask |= cpumask << 8;
|
||||
cpumask |= cpumask << 16;
|
||||
|
||||
gic_data[gic_nr].dist_base = base;
|
||||
gic_data[gic_nr].irq_offset = (irq_start - 1) & ~31;
|
||||
|
||||
arm_gic_write(0, base + GIC_DIST_CTRL);
|
||||
|
||||
/*
|
||||
* Find out how many interrupts are supported.
|
||||
*/
|
||||
max_irq = arm_gic_read(base + GIC_DIST_CTR) & 0x1f;
|
||||
max_irq = (max_irq + 1) * 32;
|
||||
|
||||
/*
|
||||
* The GIC only supports up to 1020 interrupt sources.
|
||||
* Limit this to either the architected maximum, or the
|
||||
* platform maximum.
|
||||
*/
|
||||
if (max_irq > max(1020, ARM_GIC_NR_IRQS))
|
||||
max_irq = max(1020, ARM_GIC_NR_IRQS);
|
||||
|
||||
/*
|
||||
* Set all global interrupts to be level triggered, active low.
|
||||
*/
|
||||
for (i = 32; i < max_irq; i += 16)
|
||||
arm_gic_write(0, base + GIC_DIST_CONFIG + i * 4 / 16);
|
||||
|
||||
/*
|
||||
* Set all global interrupts to this CPU only.
|
||||
*/
|
||||
for (i = 32; i < max_irq; i += 4)
|
||||
arm_gic_write(cpumask, base + GIC_DIST_TARGET + i * 4 / 4);
|
||||
|
||||
/*
|
||||
* Set priority on all interrupts.
|
||||
*/
|
||||
for (i = 0; i < max_irq; i += 4)
|
||||
arm_gic_write(0xa0a0a0a0, base + GIC_DIST_PRI + i * 4 / 4);
|
||||
|
||||
/*
|
||||
* Disable all interrupts.
|
||||
*/
|
||||
for (i = 0; i < max_irq; i += 32)
|
||||
arm_gic_write(0xffffffff,
|
||||
base + GIC_DIST_ENABLE_CLEAR + i * 4 / 32);
|
||||
|
||||
arm_gic_write(1, base + GIC_DIST_CTRL);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
int arm_gic_cpu_init(uint32_t gic_nr, virtual_addr_t base)
|
||||
{
|
||||
if (ARM_GIC_MAX_NR <= gic_nr) {
|
||||
return -1;
|
||||
}
|
||||
|
||||
gic_data[gic_nr].cpu_base = base;
|
||||
|
||||
arm_gic_write(0xf0, base + GIC_CPU_PRIMASK);
|
||||
arm_gic_write(1, base + GIC_CPU_CTRL);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
int arm_pic_active_irq(void)
|
||||
{
|
||||
return arm_gic_active_irq(0);
|
||||
}
|
||||
|
||||
int arm_pic_ack_irq(uint32_t irq)
|
||||
{
|
||||
return arm_gic_ack_irq(0, irq);
|
||||
}
|
||||
|
||||
int arm_pic_mask(uint32_t irq)
|
||||
{
|
||||
return arm_gic_mask(0, irq);
|
||||
}
|
||||
|
||||
int arm_pic_unmask(uint32_t irq)
|
||||
{
|
||||
return arm_gic_unmask(0, irq);
|
||||
}
|
||||
|
||||
int arm_pic_init(void)
|
||||
{
|
||||
int rc = 0;
|
||||
|
||||
rc = arm_gic_dist_init(0, REALVIEW_PBA8_GIC_DIST_BASE,
|
||||
IRQ_PBA8_GIC_START);
|
||||
if (rc) {
|
||||
return rc;
|
||||
}
|
||||
rc = arm_gic_cpu_init(0, REALVIEW_PBA8_GIC_CPU_BASE);
|
||||
if (rc) {
|
||||
while(1);
|
||||
}
|
||||
|
||||
return rc;
|
||||
}
|
||||
|
||||
|
||||
61
ports/armv7a/pb-a8/arm_pic.h
Normal file
61
ports/armv7a/pb-a8/arm_pic.h
Normal file
@@ -0,0 +1,61 @@
|
||||
/*
|
||||
* Copyright (c) 2011, Anup Patel. 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 _ARM_GIC_H__
|
||||
#define _ARM_GIC_H__
|
||||
|
||||
#include <atomport.h>
|
||||
#include <arm_plat.h>
|
||||
|
||||
#define GIC_CPU_CTRL 0x00
|
||||
#define GIC_CPU_PRIMASK 0x04
|
||||
#define GIC_CPU_BINPOINT 0x08
|
||||
#define GIC_CPU_INTACK 0x0c
|
||||
#define GIC_CPU_EOI 0x10
|
||||
#define GIC_CPU_RUNNINGPRI 0x14
|
||||
#define GIC_CPU_HIGHPRI 0x18
|
||||
|
||||
#define GIC_DIST_CTRL 0x000
|
||||
#define GIC_DIST_CTR 0x004
|
||||
#define GIC_DIST_ENABLE_SET 0x100
|
||||
#define GIC_DIST_ENABLE_CLEAR 0x180
|
||||
#define GIC_DIST_PENDING_SET 0x200
|
||||
#define GIC_DIST_PENDING_CLEAR 0x280
|
||||
#define GIC_DIST_ACTIVE_BIT 0x300
|
||||
#define GIC_DIST_PRI 0x400
|
||||
#define GIC_DIST_TARGET 0x800
|
||||
#define GIC_DIST_CONFIG 0xc00
|
||||
#define GIC_DIST_SOFTINT 0xf00
|
||||
|
||||
int arm_pic_active_irq(void);
|
||||
int arm_pic_ack_irq(uint32_t irq);
|
||||
int arm_pic_mask(uint32_t irq);
|
||||
int arm_pic_unmask(uint32_t irq);
|
||||
int arm_pic_init(void);
|
||||
|
||||
#endif
|
||||
287
ports/armv7a/pb-a8/arm_plat.h
Normal file
287
ports/armv7a/pb-a8/arm_plat.h
Normal file
@@ -0,0 +1,287 @@
|
||||
/*
|
||||
* Copyright (c) 2011, Anup Patel. 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 _ARM_PLAT_H__
|
||||
#define _ARM_PLAT_H__
|
||||
|
||||
/*
|
||||
* Memory definitions
|
||||
*/
|
||||
#define REALVIEW_BOOT_ROM_LO 0x30000000 /* DoC Base (64Mb)... */
|
||||
#define REALVIEW_BOOT_ROM_HI 0x30000000
|
||||
#define REALVIEW_BOOT_ROM_BASE REALVIEW_BOOT_ROM_HI /* Normal position */
|
||||
#define REALVIEW_BOOT_ROM_SIZE SZ_64M
|
||||
|
||||
#define REALVIEW_SSRAM_BASE /* REALVIEW_SSMC_BASE ? */
|
||||
#define REALVIEW_SSRAM_SIZE SZ_2M
|
||||
|
||||
/*
|
||||
* SDRAM
|
||||
*/
|
||||
#define REALVIEW_SDRAM_BASE 0x00000000
|
||||
|
||||
/*
|
||||
* Logic expansion modules
|
||||
*
|
||||
*/
|
||||
|
||||
/* ------------------------------------------------------------------------
|
||||
* RealView Registers
|
||||
* ------------------------------------------------------------------------
|
||||
*
|
||||
*/
|
||||
#define REALVIEW_SYS_ID_OFFSET 0x00
|
||||
#define REALVIEW_SYS_SW_OFFSET 0x04
|
||||
#define REALVIEW_SYS_LED_OFFSET 0x08
|
||||
#define REALVIEW_SYS_OSC0_OFFSET 0x0C
|
||||
|
||||
#define REALVIEW_SYS_OSC1_OFFSET 0x10
|
||||
#define REALVIEW_SYS_OSC2_OFFSET 0x14
|
||||
#define REALVIEW_SYS_OSC3_OFFSET 0x18
|
||||
#define REALVIEW_SYS_OSC4_OFFSET 0x1C /* OSC1 for RealView/AB */
|
||||
|
||||
#define REALVIEW_SYS_LOCK_OFFSET 0x20
|
||||
#define REALVIEW_SYS_100HZ_OFFSET 0x24
|
||||
#define REALVIEW_SYS_CFGDATA1_OFFSET 0x28
|
||||
#define REALVIEW_SYS_CFGDATA2_OFFSET 0x2C
|
||||
#define REALVIEW_SYS_FLAGS_OFFSET 0x30
|
||||
#define REALVIEW_SYS_FLAGSSET_OFFSET 0x30
|
||||
#define REALVIEW_SYS_FLAGSCLR_OFFSET 0x34
|
||||
#define REALVIEW_SYS_NVFLAGS_OFFSET 0x38
|
||||
#define REALVIEW_SYS_NVFLAGSSET_OFFSET 0x38
|
||||
#define REALVIEW_SYS_NVFLAGSCLR_OFFSET 0x3C
|
||||
#define REALVIEW_SYS_RESETCTL_OFFSET 0x40
|
||||
#define REALVIEW_SYS_PCICTL_OFFSET 0x44
|
||||
#define REALVIEW_SYS_MCI_OFFSET 0x48
|
||||
#define REALVIEW_SYS_FLASH_OFFSET 0x4C
|
||||
#define REALVIEW_SYS_CLCD_OFFSET 0x50
|
||||
#define REALVIEW_SYS_CLCDSER_OFFSET 0x54
|
||||
#define REALVIEW_SYS_BOOTCS_OFFSET 0x58
|
||||
#define REALVIEW_SYS_24MHz_OFFSET 0x5C
|
||||
#define REALVIEW_SYS_MISC_OFFSET 0x60
|
||||
#define REALVIEW_SYS_IOSEL_OFFSET 0x70
|
||||
#define REALVIEW_SYS_PROCID_OFFSET 0x84
|
||||
#define REALVIEW_SYS_TEST_OSC0_OFFSET 0xC0
|
||||
#define REALVIEW_SYS_TEST_OSC1_OFFSET 0xC4
|
||||
#define REALVIEW_SYS_TEST_OSC2_OFFSET 0xC8
|
||||
#define REALVIEW_SYS_TEST_OSC3_OFFSET 0xCC
|
||||
#define REALVIEW_SYS_TEST_OSC4_OFFSET 0xD0
|
||||
|
||||
#define REALVIEW_SYS_BASE 0x10000000
|
||||
#define REALVIEW_SYS_ID (REALVIEW_SYS_BASE + REALVIEW_SYS_ID_OFFSET)
|
||||
#define REALVIEW_SYS_SW (REALVIEW_SYS_BASE + REALVIEW_SYS_SW_OFFSET)
|
||||
#define REALVIEW_SYS_LED (REALVIEW_SYS_BASE + REALVIEW_SYS_LED_OFFSET)
|
||||
#define REALVIEW_SYS_OSC0 (REALVIEW_SYS_BASE + REALVIEW_SYS_OSC0_OFFSET)
|
||||
#define REALVIEW_SYS_OSC1 (REALVIEW_SYS_BASE + REALVIEW_SYS_OSC1_OFFSET)
|
||||
|
||||
#define REALVIEW_SYS_LOCK (REALVIEW_SYS_BASE + REALVIEW_SYS_LOCK_OFFSET)
|
||||
#define REALVIEW_SYS_100HZ (REALVIEW_SYS_BASE + REALVIEW_SYS_100HZ_OFFSET)
|
||||
#define REALVIEW_SYS_CFGDATA1 (REALVIEW_SYS_BASE + REALVIEW_SYS_CFGDATA1_OFFSET)
|
||||
#define REALVIEW_SYS_CFGDATA2 (REALVIEW_SYS_BASE + REALVIEW_SYS_CFGDATA2_OFFSET)
|
||||
#define REALVIEW_SYS_FLAGS (REALVIEW_SYS_BASE + REALVIEW_SYS_FLAGS_OFFSET)
|
||||
#define REALVIEW_SYS_FLAGSSET (REALVIEW_SYS_BASE + REALVIEW_SYS_FLAGSSET_OFFSET)
|
||||
#define REALVIEW_SYS_FLAGSCLR (REALVIEW_SYS_BASE + REALVIEW_SYS_FLAGSCLR_OFFSET)
|
||||
#define REALVIEW_SYS_NVFLAGS (REALVIEW_SYS_BASE + REALVIEW_SYS_NVFLAGS_OFFSET)
|
||||
#define REALVIEW_SYS_NVFLAGSSET (REALVIEW_SYS_BASE + REALVIEW_SYS_NVFLAGSSET_OFFSET)
|
||||
#define REALVIEW_SYS_NVFLAGSCLR (REALVIEW_SYS_BASE + REALVIEW_SYS_NVFLAGSCLR_OFFSET)
|
||||
#define REALVIEW_SYS_RESETCTL (REALVIEW_SYS_BASE + REALVIEW_SYS_RESETCTL_OFFSET)
|
||||
#define REALVIEW_SYS_PCICTL (REALVIEW_SYS_BASE + REALVIEW_SYS_PCICTL_OFFSET)
|
||||
#define REALVIEW_SYS_MCI (REALVIEW_SYS_BASE + REALVIEW_SYS_MCI_OFFSET)
|
||||
#define REALVIEW_SYS_FLASH (REALVIEW_SYS_BASE + REALVIEW_SYS_FLASH_OFFSET)
|
||||
#define REALVIEW_SYS_CLCD (REALVIEW_SYS_BASE + REALVIEW_SYS_CLCD_OFFSET)
|
||||
#define REALVIEW_SYS_CLCDSER (REALVIEW_SYS_BASE + REALVIEW_SYS_CLCDSER_OFFSET)
|
||||
#define REALVIEW_SYS_BOOTCS (REALVIEW_SYS_BASE + REALVIEW_SYS_BOOTCS_OFFSET)
|
||||
#define REALVIEW_SYS_24MHz (REALVIEW_SYS_BASE + REALVIEW_SYS_24MHz_OFFSET)
|
||||
#define REALVIEW_SYS_MISC (REALVIEW_SYS_BASE + REALVIEW_SYS_MISC_OFFSET)
|
||||
#define REALVIEW_SYS_IOSEL (REALVIEW_SYS_BASE + REALVIEW_SYS_IOSEL_OFFSET)
|
||||
#define REALVIEW_SYS_PROCID (REALVIEW_SYS_BASE + REALVIEW_SYS_PROCID_OFFSET)
|
||||
#define REALVIEW_SYS_TEST_OSC0 (REALVIEW_SYS_BASE + REALVIEW_SYS_TEST_OSC0_OFFSET)
|
||||
#define REALVIEW_SYS_TEST_OSC1 (REALVIEW_SYS_BASE + REALVIEW_SYS_TEST_OSC1_OFFSET)
|
||||
#define REALVIEW_SYS_TEST_OSC2 (REALVIEW_SYS_BASE + REALVIEW_SYS_TEST_OSC2_OFFSET)
|
||||
#define REALVIEW_SYS_TEST_OSC3 (REALVIEW_SYS_BASE + REALVIEW_SYS_TEST_OSC3_OFFSET)
|
||||
#define REALVIEW_SYS_TEST_OSC4 (REALVIEW_SYS_BASE + REALVIEW_SYS_TEST_OSC4_OFFSET)
|
||||
|
||||
/*
|
||||
* Values for REALVIEW_SYS_RESET_CTRL
|
||||
*/
|
||||
#define REALVIEW_SYS_CTRL_RESET_CONFIGCLR 0x01
|
||||
#define REALVIEW_SYS_CTRL_RESET_CONFIGINIT 0x02
|
||||
#define REALVIEW_SYS_CTRL_RESET_DLLRESET 0x03
|
||||
#define REALVIEW_SYS_CTRL_RESET_PLLRESET 0x04
|
||||
#define REALVIEW_SYS_CTRL_RESET_POR 0x05
|
||||
#define REALVIEW_SYS_CTRL_RESET_DoC 0x06
|
||||
|
||||
#define REALVIEW_SYS_CTRL_LED (1 << 0)
|
||||
|
||||
/* ------------------------------------------------------------------------
|
||||
* RealView control registers
|
||||
* ------------------------------------------------------------------------
|
||||
*/
|
||||
|
||||
/*
|
||||
* REALVIEW_IDFIELD
|
||||
*
|
||||
* 31:24 = manufacturer (0x41 = ARM)
|
||||
* 23:16 = architecture (0x08 = AHB system bus, ASB processor bus)
|
||||
* 15:12 = FPGA (0x3 = XVC600 or XVC600E)
|
||||
* 11:4 = build value
|
||||
* 3:0 = revision number (0x1 = rev B (AHB))
|
||||
*/
|
||||
|
||||
/*
|
||||
* REALVIEW_SYS_LOCK
|
||||
* control access to SYS_OSCx, SYS_CFGDATAx, SYS_RESETCTL,
|
||||
* SYS_CLD, SYS_BOOTCS
|
||||
*/
|
||||
#define REALVIEW_SYS_LOCK_LOCKED (1 << 16)
|
||||
#define REALVIEW_SYS_LOCKVAL_MASK 0xFFFF /* write 0xA05F to enable write access */
|
||||
|
||||
/*
|
||||
* REALVIEW_SYS_FLASH
|
||||
*/
|
||||
#define REALVIEW_FLASHPROG_FLVPPEN (1 << 0) /* Enable writing to flash */
|
||||
|
||||
/*
|
||||
* REALVIEW_INTREG
|
||||
* - used to acknowledge and control MMCI and UART interrupts
|
||||
*/
|
||||
#define REALVIEW_INTREG_WPROT 0x00 /* MMC protection status (no interrupt generated) */
|
||||
#define REALVIEW_INTREG_RI0 0x01 /* Ring indicator UART0 is asserted, */
|
||||
#define REALVIEW_INTREG_CARDIN 0x08 /* MMCI card in detect */
|
||||
/* write 1 to acknowledge and clear */
|
||||
#define REALVIEW_INTREG_RI1 0x02 /* Ring indicator UART1 is asserted, */
|
||||
#define REALVIEW_INTREG_CARDINSERT 0x03 /* Signal insertion of MMC card */
|
||||
|
||||
/*
|
||||
* RealView common peripheral addresses
|
||||
*/
|
||||
#define REALVIEW_SCTL_BASE 0x10001000 /* System controller */
|
||||
#define REALVIEW_I2C_BASE 0x10002000 /* I2C control */
|
||||
#define REALVIEW_AACI_BASE 0x10004000 /* Audio */
|
||||
#define REALVIEW_MMCI0_BASE 0x10005000 /* MMC interface */
|
||||
#define REALVIEW_KMI0_BASE 0x10006000 /* KMI interface */
|
||||
#define REALVIEW_KMI1_BASE 0x10007000 /* KMI 2nd interface */
|
||||
#define REALVIEW_CHAR_LCD_BASE 0x10008000 /* Character LCD */
|
||||
#define REALVIEW_SCI_BASE 0x1000E000 /* Smart card controller */
|
||||
#define REALVIEW_GPIO1_BASE 0x10014000 /* GPIO port 1 */
|
||||
#define REALVIEW_GPIO2_BASE 0x10015000 /* GPIO port 2 */
|
||||
#define REALVIEW_DMC_BASE 0x10018000 /* DMC configuration */
|
||||
#define REALVIEW_DMAC_BASE 0x10030000 /* DMA controller */
|
||||
|
||||
/* PCI space */
|
||||
#define REALVIEW_PCI_BASE 0x41000000 /* PCI Interface */
|
||||
#define REALVIEW_PCI_CFG_BASE 0x42000000
|
||||
#define REALVIEW_PCI_MEM_BASE0 0x44000000
|
||||
#define REALVIEW_PCI_MEM_BASE1 0x50000000
|
||||
#define REALVIEW_PCI_MEM_BASE2 0x60000000
|
||||
/* Sizes of above maps */
|
||||
#define REALVIEW_PCI_BASE_SIZE 0x01000000
|
||||
#define REALVIEW_PCI_CFG_BASE_SIZE 0x02000000
|
||||
#define REALVIEW_PCI_MEM_BASE0_SIZE 0x0c000000 /* 32Mb */
|
||||
#define REALVIEW_PCI_MEM_BASE1_SIZE 0x10000000 /* 256Mb */
|
||||
#define REALVIEW_PCI_MEM_BASE2_SIZE 0x10000000 /* 256Mb */
|
||||
|
||||
#define REALVIEW_SDRAM67_BASE 0x70000000 /* SDRAM banks 6 and 7 */
|
||||
#define REALVIEW_LT_BASE 0x80000000 /* Logic Tile expansion */
|
||||
|
||||
/*
|
||||
* CompactFlash
|
||||
*/
|
||||
#define REALVIEW_CF_BASE 0x18000000 /* CompactFlash */
|
||||
#define REALVIEW_CF_MEM_BASE 0x18003000 /* SMC for CompactFlash */
|
||||
|
||||
/*
|
||||
* Disk on Chip
|
||||
*/
|
||||
#define REALVIEW_DOC_BASE 0x2C000000
|
||||
#define REALVIEW_DOC_SIZE (16 << 20)
|
||||
#define REALVIEW_DOC_PAGE_SIZE 512
|
||||
#define REALVIEW_DOC_TOTAL_PAGES (DOC_SIZE / PAGE_SIZE)
|
||||
|
||||
#define ERASE_UNIT_PAGES 32
|
||||
#define START_PAGE 0x80
|
||||
|
||||
/*
|
||||
* LED settings, bits [7:0]
|
||||
*/
|
||||
#define REALVIEW_SYS_LED0 (1 << 0)
|
||||
#define REALVIEW_SYS_LED1 (1 << 1)
|
||||
#define REALVIEW_SYS_LED2 (1 << 2)
|
||||
#define REALVIEW_SYS_LED3 (1 << 3)
|
||||
#define REALVIEW_SYS_LED4 (1 << 4)
|
||||
#define REALVIEW_SYS_LED5 (1 << 5)
|
||||
#define REALVIEW_SYS_LED6 (1 << 6)
|
||||
#define REALVIEW_SYS_LED7 (1 << 7)
|
||||
|
||||
#define ALL_LEDS 0xFF
|
||||
|
||||
#define LED_BANK REALVIEW_SYS_LED
|
||||
|
||||
/*
|
||||
* Control registers
|
||||
*/
|
||||
#define REALVIEW_IDFIELD_OFFSET 0x0 /* RealView build information */
|
||||
#define REALVIEW_FLASHPROG_OFFSET 0x4 /* Flash devices */
|
||||
#define REALVIEW_INTREG_OFFSET 0x8 /* Interrupt control */
|
||||
#define REALVIEW_DECODE_OFFSET 0xC /* Fitted logic modules */
|
||||
|
||||
/*
|
||||
* Clean base - dummy
|
||||
*
|
||||
*/
|
||||
#define CLEAN_BASE REALVIEW_BOOT_ROM_HI
|
||||
|
||||
/*
|
||||
* System controller bit assignment
|
||||
*/
|
||||
#define REALVIEW_REFCLK 0
|
||||
#define REALVIEW_TIMCLK 1
|
||||
|
||||
#define REALVIEW_TIMER1_EnSel 15
|
||||
#define REALVIEW_TIMER2_EnSel 17
|
||||
#define REALVIEW_TIMER3_EnSel 19
|
||||
#define REALVIEW_TIMER4_EnSel 21
|
||||
|
||||
#define MAX_TIMER 2
|
||||
#define MAX_PERIOD 699050
|
||||
#define TICKS_PER_uSEC 1
|
||||
|
||||
/*
|
||||
* These are useconds NOT ticks.
|
||||
*
|
||||
*/
|
||||
#define mSEC_1 1000
|
||||
#define mSEC_5 (mSEC_1 * 5)
|
||||
#define mSEC_10 (mSEC_1 * 10)
|
||||
#define mSEC_25 (mSEC_1 * 25)
|
||||
#define SEC_1 (mSEC_1 * 1000)
|
||||
|
||||
#define REALVIEW_CSR_BASE 0x10000000
|
||||
#define REALVIEW_CSR_SIZE 0x10000000
|
||||
|
||||
#endif
|
||||
97
ports/armv7a/pb-a8/arm_timer.c
Normal file
97
ports/armv7a/pb-a8/arm_timer.c
Normal file
@@ -0,0 +1,97 @@
|
||||
/*
|
||||
* Copyright (c) 2011, Anup Patel. 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 <atom.h>
|
||||
#include <atomport.h>
|
||||
#include <arm_io.h>
|
||||
#include <arm_irq.h>
|
||||
#include <arm_config.h>
|
||||
#include <arm_plat.h>
|
||||
#include <arm_timer.h>
|
||||
|
||||
unsigned long long jiffies;
|
||||
|
||||
void arm_timer_enable(void)
|
||||
{
|
||||
uint32_t ctrl;
|
||||
|
||||
ctrl = arm_readl((void *)(REALVIEW_PBA8_TIMER0_1_BASE + TIMER_CTRL));
|
||||
ctrl |= TIMER_CTRL_ENABLE;
|
||||
arm_writel(ctrl, (void *)(REALVIEW_PBA8_TIMER0_1_BASE + TIMER_CTRL));
|
||||
}
|
||||
|
||||
void arm_timer_disable(void)
|
||||
{
|
||||
uint32_t ctrl;
|
||||
|
||||
ctrl = arm_readl((void *)(REALVIEW_PBA8_TIMER0_1_BASE + TIMER_CTRL));
|
||||
ctrl &= ~TIMER_CTRL_ENABLE;
|
||||
arm_writel(ctrl, (void *)(REALVIEW_PBA8_TIMER0_1_BASE + TIMER_CTRL));
|
||||
}
|
||||
|
||||
void arm_timer_clearirq(void)
|
||||
{
|
||||
arm_writel(1, (void *)(REALVIEW_PBA8_TIMER0_1_BASE + TIMER_INTCLR));
|
||||
}
|
||||
|
||||
int arm_timer_irqhndl(uint32_t irq_no, pt_regs_t * regs)
|
||||
{
|
||||
/* Call the OS system tick handler */
|
||||
atomTimerTick();
|
||||
|
||||
arm_timer_clearirq();
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
int arm_timer_init(uint32_t ticks_per_sec)
|
||||
{
|
||||
uint32_t val;
|
||||
|
||||
/*
|
||||
* set clock frequency:
|
||||
* REALVIEW_TIMCLK is 1MHz
|
||||
*/
|
||||
val = arm_readl((void *)REALVIEW_SCTL_BASE) | (REALVIEW_TIMCLK << 0x1);
|
||||
arm_writel(val, (void *)REALVIEW_SCTL_BASE);
|
||||
|
||||
/* Register interrupt handler */
|
||||
arm_irq_register(IRQ_PBA8_TIMER0_1, &arm_timer_irqhndl);
|
||||
|
||||
val = arm_readl((void *)(REALVIEW_PBA8_TIMER0_1_BASE + TIMER_CTRL));
|
||||
val &= ~TIMER_CTRL_ENABLE;
|
||||
val |= (TIMER_CTRL_32BIT | TIMER_CTRL_PERIODIC | TIMER_CTRL_IE);
|
||||
arm_writel(val, (void *)(REALVIEW_PBA8_TIMER0_1_BASE + TIMER_CTRL));
|
||||
arm_writel((1000000 / ticks_per_sec),
|
||||
(void *)(REALVIEW_PBA8_TIMER0_1_BASE + TIMER_LOAD));
|
||||
arm_writel((1000000 / ticks_per_sec),
|
||||
(void *)(REALVIEW_PBA8_TIMER0_1_BASE + TIMER_VALUE));
|
||||
|
||||
return 0;
|
||||
}
|
||||
56
ports/armv7a/pb-a8/arm_timer.h
Normal file
56
ports/armv7a/pb-a8/arm_timer.h
Normal file
@@ -0,0 +1,56 @@
|
||||
/*
|
||||
* Copyright (c) 2011, Anup Patel. 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 __ARM_TIMER_H
|
||||
#define __ARM_TIMER_H
|
||||
|
||||
#include <atomport.h>
|
||||
|
||||
#define TIMER_LOAD 0x00
|
||||
#define TIMER_VALUE 0x04
|
||||
#define TIMER_CTRL 0x08
|
||||
#define TIMER_CTRL_ONESHOT (1 << 0)
|
||||
#define TIMER_CTRL_32BIT (1 << 1)
|
||||
#define TIMER_CTRL_DIV1 (0 << 2)
|
||||
#define TIMER_CTRL_DIV16 (1 << 2)
|
||||
#define TIMER_CTRL_DIV256 (2 << 2)
|
||||
#define TIMER_CTRL_IE (1 << 5) /* Interrupt Enable (versatile only) */
|
||||
#define TIMER_CTRL_PERIODIC (1 << 6)
|
||||
#define TIMER_CTRL_ENABLE (1 << 7)
|
||||
|
||||
#define TIMER_INTCLR 0x0c
|
||||
#define TIMER_RIS 0x10
|
||||
#define TIMER_MIS 0x14
|
||||
#define TIMER_BGLOAD 0x18
|
||||
|
||||
void arm_timer_enable(void);
|
||||
void arm_timer_disable(void);
|
||||
void arm_timer_clearirq(void);
|
||||
int arm_timer_init(uint32_t ticks_per_sec);
|
||||
|
||||
#endif /* __ARM_TIMER_H */
|
||||
112
ports/armv7a/pb-a8/arm_uart.c
Normal file
112
ports/armv7a/pb-a8/arm_uart.c
Normal file
@@ -0,0 +1,112 @@
|
||||
/*
|
||||
* Copyright (c) 2011, Anup Patel. 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 <arm_io.h>
|
||||
#include <arm_uart.h>
|
||||
|
||||
void arm_uart_putc(uint8_t ch)
|
||||
{
|
||||
unsigned int base = 0x10009000;
|
||||
if(ch=='\n') {
|
||||
/* Wait until there is space in the FIFO */
|
||||
while (arm_readl((void*)(base + UART_PL01x_FR)) & UART_PL01x_FR_TXFF);
|
||||
|
||||
/* Send the character */
|
||||
arm_writel('\r', (void*)(base + UART_PL01x_DR));
|
||||
}
|
||||
|
||||
/* Wait until there is space in the FIFO */
|
||||
while (arm_readl((void*)(base + UART_PL01x_FR)) & UART_PL01x_FR_TXFF);
|
||||
|
||||
/* Send the character */
|
||||
arm_writel(ch, (void*)(base + UART_PL01x_DR));
|
||||
}
|
||||
|
||||
uint8_t arm_uart_getc(void)
|
||||
{
|
||||
unsigned int base = 0x10009000;
|
||||
uint8_t data;
|
||||
|
||||
/* Wait until there is data in the FIFO */
|
||||
while (arm_readl((void*)(base + UART_PL01x_FR)) & UART_PL01x_FR_RXFE);
|
||||
|
||||
data = arm_readl((void*)(base + UART_PL01x_DR));
|
||||
|
||||
/* Check for an error flag */
|
||||
if (data & 0xFFFFFF00) {
|
||||
/* Clear the error */
|
||||
arm_writel(0xFFFFFFFF, (void*)(base + UART_PL01x_ECR));
|
||||
return -1;
|
||||
}
|
||||
|
||||
return data;
|
||||
}
|
||||
|
||||
void arm_uart_init(void)
|
||||
{
|
||||
unsigned int base = 0x10009000;
|
||||
unsigned int baudrate = 115200;
|
||||
unsigned int input_clock = 24000000;
|
||||
unsigned int divider;
|
||||
unsigned int temp;
|
||||
unsigned int remainder;
|
||||
unsigned int fraction;
|
||||
|
||||
/* First, disable everything */
|
||||
arm_writel(0x0, (void*)(base + UART_PL011_CR));
|
||||
|
||||
/*
|
||||
* Set baud rate
|
||||
*
|
||||
* IBRD = UART_CLK / (16 * BAUD_RATE)
|
||||
* FBRD = RND((64 * MOD(UART_CLK,(16 * BAUD_RATE)))
|
||||
* / (16 * BAUD_RATE))
|
||||
*/
|
||||
temp = 16 * baudrate;
|
||||
divider = input_clock / temp;
|
||||
remainder = input_clock % temp;
|
||||
temp = (8 * remainder) / baudrate;
|
||||
fraction = (temp >> 1) + (temp & 1);
|
||||
|
||||
arm_writel(divider, (void*)(base + UART_PL011_IBRD));
|
||||
arm_writel(fraction, (void*)(base + UART_PL011_FBRD));
|
||||
|
||||
/* Set the UART to be 8 bits, 1 stop bit,
|
||||
* no parity, fifo enabled
|
||||
*/
|
||||
arm_writel((UART_PL011_LCRH_WLEN_8 | UART_PL011_LCRH_FEN),
|
||||
(void*)(base + UART_PL011_LCRH));
|
||||
|
||||
/* Finally, enable the UART */
|
||||
arm_writel((UART_PL011_CR_UARTEN |
|
||||
UART_PL011_CR_TXE |
|
||||
UART_PL011_CR_RXE),
|
||||
(void*)(base + UART_PL011_CR));
|
||||
}
|
||||
|
||||
110
ports/armv7a/pb-a8/arm_uart.h
Normal file
110
ports/armv7a/pb-a8/arm_uart.h
Normal file
@@ -0,0 +1,110 @@
|
||||
/*
|
||||
* Copyright (c) 2011, Anup Patel. 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 __ARM_UART_H_
|
||||
#define __ARM_UART_H_
|
||||
|
||||
#include <atomport.h>
|
||||
|
||||
/*
|
||||
* ARM PrimeCell UART's (PL010 & PL011)
|
||||
* ------------------------------------
|
||||
*
|
||||
* Definitions common to both PL010 & PL011
|
||||
*
|
||||
*/
|
||||
#define UART_PL01x_DR 0x00 /* Data read or written from the interface. */
|
||||
#define UART_PL01x_RSR 0x04 /* Receive status register (Read). */
|
||||
#define UART_PL01x_ECR 0x04 /* Error clear register (Write). */
|
||||
#define UART_PL01x_FR 0x18 /* Flag register (Read only). */
|
||||
|
||||
#define UART_PL01x_RSR_OE 0x08
|
||||
#define UART_PL01x_RSR_BE 0x04
|
||||
#define UART_PL01x_RSR_PE 0x02
|
||||
#define UART_PL01x_RSR_FE 0x01
|
||||
|
||||
#define UART_PL01x_FR_TXFE 0x80
|
||||
#define UART_PL01x_FR_RXFF 0x40
|
||||
#define UART_PL01x_FR_TXFF 0x20
|
||||
#define UART_PL01x_FR_RXFE 0x10
|
||||
#define UART_PL01x_FR_BUSY 0x08
|
||||
#define UART_PL01x_FR_TMSK (UART_PL01x_FR_TXFF + UART_PL01x_FR_BUSY)
|
||||
|
||||
/*
|
||||
* PL011 definitions
|
||||
*
|
||||
*/
|
||||
#define UART_PL011_IBRD 0x24
|
||||
#define UART_PL011_FBRD 0x28
|
||||
#define UART_PL011_LCRH 0x2C
|
||||
#define UART_PL011_CR 0x30
|
||||
#define UART_PL011_IMSC 0x38
|
||||
#define UART_PL011_PERIPH_ID0 0xFE0
|
||||
|
||||
#define UART_PL011_LCRH_SPS (1 << 7)
|
||||
#define UART_PL011_LCRH_WLEN_8 (3 << 5)
|
||||
#define UART_PL011_LCRH_WLEN_7 (2 << 5)
|
||||
#define UART_PL011_LCRH_WLEN_6 (1 << 5)
|
||||
#define UART_PL011_LCRH_WLEN_5 (0 << 5)
|
||||
#define UART_PL011_LCRH_FEN (1 << 4)
|
||||
#define UART_PL011_LCRH_STP2 (1 << 3)
|
||||
#define UART_PL011_LCRH_EPS (1 << 2)
|
||||
#define UART_PL011_LCRH_PEN (1 << 1)
|
||||
#define UART_PL011_LCRH_BRK (1 << 0)
|
||||
|
||||
#define UART_PL011_CR_CTSEN (1 << 15)
|
||||
#define UART_PL011_CR_RTSEN (1 << 14)
|
||||
#define UART_PL011_CR_OUT2 (1 << 13)
|
||||
#define UART_PL011_CR_OUT1 (1 << 12)
|
||||
#define UART_PL011_CR_RTS (1 << 11)
|
||||
#define UART_PL011_CR_DTR (1 << 10)
|
||||
#define UART_PL011_CR_RXE (1 << 9)
|
||||
#define UART_PL011_CR_TXE (1 << 8)
|
||||
#define UART_PL011_CR_LPE (1 << 7)
|
||||
#define UART_PL011_CR_IIRLP (1 << 2)
|
||||
#define UART_PL011_CR_SIREN (1 << 1)
|
||||
#define UART_PL011_CR_UARTEN (1 << 0)
|
||||
|
||||
#define UART_PL011_IMSC_OEIM (1 << 10)
|
||||
#define UART_PL011_IMSC_BEIM (1 << 9)
|
||||
#define UART_PL011_IMSC_PEIM (1 << 8)
|
||||
#define UART_PL011_IMSC_FEIM (1 << 7)
|
||||
#define UART_PL011_IMSC_RTIM (1 << 6)
|
||||
#define UART_PL011_IMSC_TXIM (1 << 5)
|
||||
#define UART_PL011_IMSC_RXIM (1 << 4)
|
||||
#define UART_PL011_IMSC_DSRMIM (1 << 3)
|
||||
#define UART_PL011_IMSC_DCDMIM (1 << 2)
|
||||
#define UART_PL011_IMSC_CTSMIM (1 << 1)
|
||||
#define UART_PL011_IMSC_RIMIM (1 << 0)
|
||||
|
||||
uint8_t arm_uart_getc(void);
|
||||
void arm_uart_putc(uint8_t ch);
|
||||
void arm_uart_init(void);
|
||||
|
||||
#endif /* __ARM_UART_H_ */
|
||||
74
ports/armv7a/pb-a8/linker.ld
Executable file
74
ports/armv7a/pb-a8/linker.ld
Executable file
@@ -0,0 +1,74 @@
|
||||
/*
|
||||
* Copyright (c) 2011, Anup Patel. 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.
|
||||
*/
|
||||
|
||||
OUTPUT_FORMAT("elf32-littlearm", "elf32-littlearm", "elf32-littlearm")
|
||||
OUTPUT_ARCH("arm")
|
||||
ENTRY(_start_vect)
|
||||
|
||||
SECTIONS
|
||||
{
|
||||
. = 0x100000;
|
||||
|
||||
.text :
|
||||
{
|
||||
*(.expvect)
|
||||
*(.text)
|
||||
. = ALIGN(4);
|
||||
_etext = .;
|
||||
}
|
||||
|
||||
.data :
|
||||
{
|
||||
*(.data)
|
||||
. = ALIGN(4);
|
||||
_edata = .;
|
||||
}
|
||||
|
||||
.bss :
|
||||
{
|
||||
*(.bss)
|
||||
. = ALIGN(4);
|
||||
_ebss = .;
|
||||
}
|
||||
|
||||
.rodata :
|
||||
{
|
||||
*(.rodata .rodata.*)
|
||||
. = ALIGN(4);
|
||||
_erodata = .;
|
||||
}
|
||||
|
||||
.initial_stack :
|
||||
{
|
||||
PROVIDE(_initial_stack_start = .);
|
||||
. = . + 4096;
|
||||
. = ALIGN(4);
|
||||
PROVIDE(_initial_stack_end = .);
|
||||
}
|
||||
}
|
||||
60
ports/armv7a/printk.c
Normal file
60
ports/armv7a/printk.c
Normal file
@@ -0,0 +1,60 @@
|
||||
/*
|
||||
* Copyright (c) Himanshu Chauhan 2009-11.
|
||||
* 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 Himanshu Chauhan 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 OWNER 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 <stdarg.h>
|
||||
#include "system.h"
|
||||
#include "atomport.h"
|
||||
#include "printk.h"
|
||||
#include "arm_uart.h"
|
||||
|
||||
static int8_t buf[2048];
|
||||
|
||||
/* Uses the above routine to output a string... */
|
||||
void puts(const uint8_t *text)
|
||||
{
|
||||
int32_t i;
|
||||
|
||||
for (i = 0; i < strlen((const int8_t *)text); i++) {
|
||||
arm_uart_putc(text[i]);
|
||||
}
|
||||
}
|
||||
|
||||
void printk(const char *format, ...)
|
||||
{
|
||||
va_list args;
|
||||
int i;
|
||||
|
||||
va_start(args, format);
|
||||
i = vsprintf(buf, (const int8_t *)format, args);
|
||||
va_end(args);
|
||||
|
||||
puts((const uint8_t *)buf);
|
||||
}
|
||||
|
||||
42
ports/armv7a/printk.h
Normal file
42
ports/armv7a/printk.h
Normal file
@@ -0,0 +1,42 @@
|
||||
/*
|
||||
* This file is part of Freax kernel.
|
||||
*
|
||||
* Copyright (c) Himanshu Chauhan 2009-10.
|
||||
* 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 Himanshu Chauhan 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 OWNER 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 _PRINTK_H
|
||||
#define _PRINTK_H
|
||||
|
||||
#include "atomport.h"
|
||||
|
||||
extern void putch (uint8_t ch);
|
||||
extern void puts (const uint8_t *text);
|
||||
extern void printk (const char*format, ...);
|
||||
|
||||
#endif
|
||||
28
ports/armv7a/stdarg.h
Executable file
28
ports/armv7a/stdarg.h
Executable file
@@ -0,0 +1,28 @@
|
||||
#ifndef _STDARG_H
|
||||
#define _STDARG_H
|
||||
|
||||
typedef char *va_list;
|
||||
|
||||
/* Amount of space required in an argument list for an arg of type TYPE.
|
||||
TYPE may alternatively be an expression whose type is used. */
|
||||
|
||||
#define __va_rounded_size(TYPE) \
|
||||
(((sizeof (TYPE) + sizeof (int) - 1) / sizeof (int)) * sizeof (int))
|
||||
|
||||
#ifndef __sparc__
|
||||
#define va_start(AP, LASTARG) \
|
||||
(AP = ((char *) &(LASTARG) + __va_rounded_size (LASTARG)))
|
||||
#else
|
||||
#define va_start(AP, LASTARG) \
|
||||
(__builtin_saveregs (), \
|
||||
AP = ((char *) &(LASTARG) + __va_rounded_size (LASTARG)))
|
||||
#endif
|
||||
|
||||
void va_end (va_list); /* Defined in gnulib */
|
||||
#define va_end(AP)
|
||||
|
||||
#define va_arg(AP, TYPE) \
|
||||
(AP += __va_rounded_size (TYPE), \
|
||||
*((TYPE *) (AP - __va_rounded_size (TYPE))))
|
||||
|
||||
#endif /* _STDARG_H */
|
||||
61
ports/armv7a/string.c
Normal file
61
ports/armv7a/string.c
Normal file
@@ -0,0 +1,61 @@
|
||||
/*
|
||||
* Copyright (c) Himanshu Chauhan 2009-11.
|
||||
* 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 Himanshu Chauhan 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 OWNER 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 <atomport.h>
|
||||
#include <system.h>
|
||||
|
||||
void *memcpy(void *dest, const void *src, size_t count)
|
||||
{
|
||||
const int8_t *sp = (const int8_t *)src;
|
||||
int8_t *dp = (int8_t *)dest;
|
||||
for(; count != 0; count--) *dp++ = *sp++;
|
||||
return dest;
|
||||
}
|
||||
|
||||
void *memset(void *dest, int8_t val, size_t count)
|
||||
{
|
||||
int8_t *temp = (int8_t *)dest;
|
||||
for( ; count != 0; count--) *temp++ = val;
|
||||
return dest;
|
||||
}
|
||||
|
||||
uint16_t *memsetw(uint16_t *dest, uint16_t val, size_t count)
|
||||
{
|
||||
uint16_t *temp = (uint16_t *)dest;
|
||||
for( ; count != 0; count--) *temp++ = val;
|
||||
return dest;
|
||||
}
|
||||
|
||||
size_t strlen(const int8_t *str)
|
||||
{
|
||||
size_t retval;
|
||||
for(retval = 0; *str != '\0'; str++) retval++;
|
||||
return retval;
|
||||
}
|
||||
42
ports/armv7a/string.h
Normal file
42
ports/armv7a/string.h
Normal file
@@ -0,0 +1,42 @@
|
||||
/*
|
||||
* Copyright (c) Himanshu Chauhan 2009-11.
|
||||
* 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 Himanshu Chauhan 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 OWNER 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 __STRING_H
|
||||
#define __STRING_H
|
||||
|
||||
#include <atomport.h>
|
||||
|
||||
void *memcpy(void *dest, const void *src, size_t count);
|
||||
void *memset(void *dest, int8_t val, size_t count);
|
||||
uint16_t *memsetw(uint16_t *dest, uint16_t val, size_t count);
|
||||
size_t strlen(const int8_t *str);
|
||||
|
||||
#endif /* __STRING_H */
|
||||
|
||||
52
ports/armv7a/system.h
Normal file
52
ports/armv7a/system.h
Normal file
@@ -0,0 +1,52 @@
|
||||
/*
|
||||
* Copyright (c) Himanshu Chauhan 2009-11.
|
||||
* 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 Himanshu Chauhan 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 OWNER 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 _SYSTEM_H
|
||||
#define _SYSTEM_H
|
||||
|
||||
#include <atomport.h>
|
||||
#include <stdarg.h>
|
||||
|
||||
extern const uint8_t *kernel_name;
|
||||
extern const uint8_t *kernel_version;
|
||||
extern const uint8_t *kernel_bdate;
|
||||
extern const uint8_t *kernel_btime;
|
||||
|
||||
extern void *memcpy (void *dest, const void *src, size_t count);
|
||||
extern void *memset (void *dest, int8_t val, size_t count);
|
||||
extern uint16_t *memsetw (uint16_t *dest, uint16_t val, size_t count);
|
||||
extern size_t strlen (const int8_t *str);
|
||||
extern int vsprintf (int8_t *buf, const int8_t *fmt, va_list args);
|
||||
extern void init_console (void);
|
||||
extern int32_t arch_init (void);
|
||||
extern uint8_t ioreadb (void *addr);
|
||||
extern void iowriteb (void *addr, uint8_t data);
|
||||
|
||||
#endif /* _SYSTEM_H */
|
||||
244
ports/armv7a/vsprintf.c
Normal file
244
ports/armv7a/vsprintf.c
Normal file
@@ -0,0 +1,244 @@
|
||||
/* vsprintf.c -- Lars Wirzenius & Linus Torvalds. */
|
||||
/*
|
||||
* Wirzenius wrote this portably, Torvalds fucked it up :-)
|
||||
* and Himanshu Fucked it up further :))
|
||||
*/
|
||||
|
||||
#include <stdarg.h>
|
||||
#include "system.h"
|
||||
#include "atomport.h"
|
||||
|
||||
/* we use this so that we can do without the ctype library */
|
||||
#define is_digit(c) ((c) >= '0' && (c) <= '9')
|
||||
|
||||
static int skip_atoi(const int8_t **s)
|
||||
{
|
||||
int i=0;
|
||||
|
||||
while (is_digit(**s))
|
||||
i = i*10 + *((*s)++) - '0';
|
||||
return i;
|
||||
}
|
||||
|
||||
#define ZEROPAD 1 /* pad with zero */
|
||||
#define SIGN 2 /* unsigned/signed long */
|
||||
#define PLUS 4 /* show plus */
|
||||
#define SPACE 8 /* space if plus */
|
||||
#define LEFT 16 /* left justified */
|
||||
#define SPECIAL 32 /* 0x */
|
||||
#define SMALL 64 /* use 'abcdef' instead of 'ABCDEF' */
|
||||
|
||||
/*#define do_div(n,base) ({ \
|
||||
int __res; \
|
||||
__asm__("divl %4":"=a" (n),"=d" (__res):"0" (n),"1" (0),"r" (base)); \
|
||||
__res; })*/
|
||||
|
||||
static uint32_t do_div (int32_t *n, int32_t base)
|
||||
{
|
||||
uint32_t remainder = *n % base;
|
||||
*n /= base;
|
||||
return remainder;
|
||||
}
|
||||
|
||||
static int8_t * number(int8_t * str, int num, int32_t base,
|
||||
int32_t size, int32_t precision, int32_t type)
|
||||
{
|
||||
int8_t c,sign,tmp[36];
|
||||
const int8_t *digits=(const int8_t *)"0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZ";
|
||||
int32_t i;
|
||||
|
||||
if (type & SMALL) digits = (const int8_t *)"0123456789abcdefghijklmnopqrstuvwxyz";
|
||||
if (type & LEFT) type &= ~ZEROPAD;
|
||||
if (base < 2 || base > 36)
|
||||
return 0;
|
||||
c = (type & ZEROPAD) ? '0' : ' ' ;
|
||||
if (type & SIGN && num < 0) {
|
||||
sign = '-';
|
||||
num = -num;
|
||||
} else
|
||||
sign = (type & PLUS) ? '+' : ((type & SPACE) ? ' ' : 0);
|
||||
if (sign) size--;
|
||||
|
||||
if (type & SPECIAL) {
|
||||
if (base == 16) {
|
||||
size -= 2;
|
||||
} else if (base == 8) {
|
||||
size--;
|
||||
}
|
||||
}
|
||||
|
||||
i = 0;
|
||||
if (num == 0)
|
||||
tmp[i++] = '0';
|
||||
else while (num != 0)
|
||||
tmp[i++] = digits[do_div(&num,base)];
|
||||
if (i > precision) precision = i;
|
||||
size -= precision;
|
||||
if (!(type & (ZEROPAD + LEFT)))
|
||||
while(size-- > 0)
|
||||
*str++ = ' ';
|
||||
if (sign)
|
||||
*str++ = sign;
|
||||
if (type & SPECIAL) {
|
||||
if (base == 8) {
|
||||
*str++ = '0';
|
||||
} else if (base == 16) {
|
||||
*str++ = '0';
|
||||
*str++ = digits[33];
|
||||
}
|
||||
}
|
||||
|
||||
if (!(type & LEFT))
|
||||
while(size-- > 0)
|
||||
*str++ = c;
|
||||
while(i < precision--)
|
||||
*str++ = '0';
|
||||
while(i-- > 0)
|
||||
*str++ = tmp[i];
|
||||
while(size-- > 0)
|
||||
*str++ = ' ';
|
||||
return str;
|
||||
}
|
||||
|
||||
int32_t vsprintf (int8_t *buf, const int8_t *fmt, va_list args)
|
||||
{
|
||||
int32_t len;
|
||||
int32_t i;
|
||||
int8_t * str;
|
||||
int8_t *s;
|
||||
int32_t *ip;
|
||||
|
||||
int32_t flags; /* flags to number() */
|
||||
|
||||
int32_t field_width; /* width of output field */
|
||||
int32_t precision; /* min. # of digits for integers; max
|
||||
number of chars for from string */
|
||||
int32_t qualifier; /* 'h', 'l', or 'L' for integer fields */
|
||||
|
||||
for (str=buf ; *fmt ; ++fmt) {
|
||||
if (*fmt != '%') {
|
||||
*str++ = *fmt;
|
||||
continue;
|
||||
}
|
||||
|
||||
/* process flags */
|
||||
flags = 0;
|
||||
repeat:
|
||||
++fmt; /* this also skips first '%' */
|
||||
switch (*fmt) {
|
||||
case '-': flags |= LEFT; goto repeat;
|
||||
case '+': flags |= PLUS; goto repeat;
|
||||
case ' ': flags |= SPACE; goto repeat;
|
||||
case '#': flags |= SPECIAL; goto repeat;
|
||||
case '0': flags |= ZEROPAD; goto repeat;
|
||||
}
|
||||
|
||||
/* get field width */
|
||||
field_width = -1;
|
||||
if (is_digit(*fmt))
|
||||
field_width = skip_atoi(&fmt);
|
||||
else if (*fmt == '*') {
|
||||
/* it's the next argument */
|
||||
field_width = va_arg(args, int);
|
||||
if (field_width < 0) {
|
||||
field_width = -field_width;
|
||||
flags |= LEFT;
|
||||
}
|
||||
}
|
||||
|
||||
/* get the precision */
|
||||
precision = -1;
|
||||
if (*fmt == '.') {
|
||||
++fmt;
|
||||
if (is_digit(*fmt))
|
||||
precision = skip_atoi(&fmt);
|
||||
else if (*fmt == '*') {
|
||||
/* it's the next argument */
|
||||
precision = va_arg(args, int);
|
||||
}
|
||||
if (precision < 0)
|
||||
precision = 0;
|
||||
}
|
||||
|
||||
/* get the conversion qualifier */
|
||||
qualifier = -1;
|
||||
if (*fmt == 'h' || *fmt == 'l' || *fmt == 'L') {
|
||||
qualifier = *fmt;
|
||||
++fmt;
|
||||
}
|
||||
|
||||
switch (*fmt) {
|
||||
case 'c':
|
||||
if (!(flags & LEFT))
|
||||
while (--field_width > 0)
|
||||
*str++ = ' ';
|
||||
*str++ = (unsigned char) va_arg(args, int);
|
||||
while (--field_width > 0)
|
||||
*str++ = ' ';
|
||||
break;
|
||||
|
||||
case 's':
|
||||
s = va_arg(args, int8_t *);
|
||||
len = strlen(s);
|
||||
if (precision < 0)
|
||||
precision = len;
|
||||
else if (len > precision)
|
||||
len = precision;
|
||||
|
||||
if (!(flags & LEFT))
|
||||
while (len < field_width--)
|
||||
*str++ = ' ';
|
||||
for (i = 0; i < len; ++i)
|
||||
*str++ = *s++;
|
||||
while (len < field_width--)
|
||||
*str++ = ' ';
|
||||
break;
|
||||
|
||||
case 'o':
|
||||
str = number(str, va_arg(args, unsigned long), 8,
|
||||
field_width, precision, flags);
|
||||
break;
|
||||
|
||||
case 'p':
|
||||
if (field_width == -1) {
|
||||
field_width = 8;
|
||||
flags |= ZEROPAD;
|
||||
}
|
||||
str = number(str,
|
||||
(unsigned long) va_arg(args, void *), 16,
|
||||
field_width, precision, flags);
|
||||
break;
|
||||
|
||||
case 'x':
|
||||
flags |= SMALL;
|
||||
case 'X':
|
||||
str = number(str, va_arg(args, unsigned long), 16,
|
||||
field_width, precision, flags);
|
||||
break;
|
||||
|
||||
case 'd':
|
||||
case 'i':
|
||||
flags |= SIGN;
|
||||
case 'u':
|
||||
str = number(str, va_arg(args, unsigned long), 10,
|
||||
field_width, precision, flags);
|
||||
break;
|
||||
|
||||
case 'n':
|
||||
ip = va_arg(args, int32_t *);
|
||||
*ip = (str - buf);
|
||||
break;
|
||||
|
||||
default:
|
||||
if (*fmt != '%')
|
||||
*str++ = '%';
|
||||
if (*fmt)
|
||||
*str++ = *fmt;
|
||||
else
|
||||
--fmt;
|
||||
break;
|
||||
}
|
||||
}
|
||||
*str = '\0';
|
||||
return str-buf;
|
||||
}
|
||||
40
ports/atomvm/README
Normal file
40
ports/atomvm/README
Normal file
@@ -0,0 +1,40 @@
|
||||
---------------------------------------------------------------------------
|
||||
|
||||
Library: Atomvn
|
||||
Author: Natie van Rooyen <natie@navaro.nl>
|
||||
License: BSD Revised
|
||||
|
||||
---------------------------------------------------------------------------
|
||||
|
||||
Atomvm is a tiny virtual machine that runs on Windows and can be debugged
|
||||
from an IDE like Microsoft Visual C++ Express. The primary purpose of this
|
||||
virtual machine is for the evaluation of Real Time Operating Systems like
|
||||
Atomthreads and the development and testing of programs for it on a Windows
|
||||
machine.
|
||||
|
||||
Atomvm makes use of the Windows API functions GetThreadContext() and
|
||||
SetThreadContext() to create multiple virtual contexts or threads inside a
|
||||
single Windows thread. Atomvm also simulates interrupts with an interrupt
|
||||
mask accessible from the Atomvm threads. External events can be queued as
|
||||
interrupts to Atomvm, for example a timer loop generating system timer tick
|
||||
interrupts for a Real Time Operating System ported to Atomvm.
|
||||
|
||||
---------------------------------------------------------------------------
|
||||
|
||||
BUILDING THE SOURCE
|
||||
|
||||
To test this project, just add all the files from the "atomthreads/kernel"
|
||||
directory and the "atomthreads/ports/atomvm" directory as well as the test
|
||||
program "atomthreads/ports/atomvm/test/main.c" to your project. Add both the
|
||||
before mentioned directories to the include paths of your project and compile.
|
||||
|
||||
Atomvm was designed for multi core systems but also runs fine on any single
|
||||
core system.
|
||||
|
||||
---------------------------------------------------------------------------
|
||||
|
||||
RUNNING THE TESTS
|
||||
|
||||
The test, main.c, is intentioned to stress the virtual machine.
|
||||
|
||||
|
||||
36
ports/atomvm/README.txt
Normal file
36
ports/atomvm/README.txt
Normal file
@@ -0,0 +1,36 @@
|
||||
---------------------------------------------------------------------------
|
||||
|
||||
Library: Atomvn
|
||||
Author: Natie van Rooyen <natie@navaro.nl>
|
||||
License: BSD Revised
|
||||
|
||||
---------------------------------------------------------------------------
|
||||
|
||||
Atomvm is a tiny virtual machine that can run on Windows inside an IDE with a
|
||||
debugger like Microsoft Visual C++ Express. The primary purpose of this virtual
|
||||
machine is for the evaluation of Real Time Operating Systems (like atomthreads)
|
||||
and the development and testing of modules for this Real Time Operating System
|
||||
in a user friendly environment.
|
||||
|
||||
---------------------------------------------------------------------------
|
||||
|
||||
BUILDING THE SOURCE
|
||||
|
||||
To test this project, just add all the files from the "atomthreads/kernel"
|
||||
directory and the "atomthreads/ports/atomvm" directory as well as the test
|
||||
program "atomthreads/ports/atomvm/test/main.c" to your project. Add both the
|
||||
before mentioned directories to the include paths of your project and compile.
|
||||
|
||||
Atomvm was designed for multi core systems but also runs fine on any single
|
||||
core system.
|
||||
|
||||
---------------------------------------------------------------------------
|
||||
|
||||
RUNNING THE TESTS
|
||||
|
||||
The test, main.c, is intentioned to stress the virtual machine as opposed to
|
||||
testing the Real Time Operating System. However, this test can also run the
|
||||
unit tests of atomthreads by using the preprocessor directive "UNIT_TESTS" and
|
||||
linking in the desired unit test into the project.
|
||||
|
||||
|
||||
57
ports/atomvm/atomport-tests.h
Normal file
57
ports/atomvm/atomport-tests.h
Normal file
@@ -0,0 +1,57 @@
|
||||
/*
|
||||
* Copyright (c) 2010, 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 __ATOMPORT_TEST_H__
|
||||
#define __ATOMPORT_TEST_H__
|
||||
|
||||
/* Include Atomthreads kernel API */
|
||||
#include "atom.h"
|
||||
|
||||
/* Prerequisite include for ATOMLOG() macro (via printf) */
|
||||
#include <stdio.h>
|
||||
|
||||
/* Logger macro for viewing test results */
|
||||
#define ATOMLOG printf
|
||||
|
||||
/*
|
||||
* String location macro: for platforms which need to place strings in
|
||||
* alternative locations, e.g. on avr-gcc strings can be placed in
|
||||
* program space, saving SRAM. On most platforms this can expand to
|
||||
* empty.
|
||||
*/
|
||||
#define _STR(x) x
|
||||
|
||||
/* Default thread stack size (in bytes) */
|
||||
#define TEST_THREAD_STACK_SIZE 0x4000
|
||||
|
||||
/* Uncomment to enable logging of stack usage to UART */
|
||||
/* #define TESTS_LOG_STACK_USAGE */
|
||||
|
||||
|
||||
#endif /* __ATOMPORT_TEST_H__ */
|
||||
159
ports/atomvm/atomport.c
Normal file
159
ports/atomvm/atomport.c
Normal file
@@ -0,0 +1,159 @@
|
||||
/*
|
||||
* Copyright (c) 2010, 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 "atom.h"
|
||||
#include "atomport.h"
|
||||
#include "atomvm.h"
|
||||
#include "windows.h"
|
||||
|
||||
/** Forward declarations */
|
||||
DWORD WINAPI cntrl_thread_proc (LPVOID lpParameter) ;
|
||||
|
||||
/* Global data */
|
||||
HATOMVM the_atomvm ;
|
||||
|
||||
/* Local data */
|
||||
static HANDLE cntrl_thread ;
|
||||
|
||||
|
||||
/**
|
||||
* \b atomvmRun
|
||||
*
|
||||
* Starts the atom vm. atomvmRun creates a thread from where the atomvmCtrlRun function
|
||||
* will be called. atomvmCtrlRun never returns and this thread becomes the controll
|
||||
* thread of the vm.
|
||||
*
|
||||
*/
|
||||
void
|
||||
atomvmRun (void)
|
||||
{
|
||||
atomvmCtrlCreate (&the_atomvm) ;
|
||||
cntrl_thread = CreateThread (NULL, 0, cntrl_thread_proc, (uint32_t*)the_atomvm, CREATE_SUSPENDED, NULL) ;
|
||||
ResumeThread (cntrl_thread) ;
|
||||
}
|
||||
|
||||
DWORD WINAPI
|
||||
cntrl_thread_proc (LPVOID lpParameter)
|
||||
{
|
||||
atomvmCtrlRun ((HATOMVM)lpParameter, 0) ;
|
||||
return 0 ;
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* \b thread_shell
|
||||
*
|
||||
* Documented in atomThreads.
|
||||
*
|
||||
*/
|
||||
void
|
||||
thread_shell (uint32_t arg)
|
||||
{
|
||||
ATOM_TCB *curr_tcb;
|
||||
|
||||
/* Get the TCB of the thread being started */
|
||||
curr_tcb = atomCurrentContext();
|
||||
|
||||
/**
|
||||
* Enable interrupts - these will not be enabled when a thread
|
||||
* is first restored.
|
||||
*/
|
||||
// sei();
|
||||
//atomvmExitCritical () ;
|
||||
atomvmInterruptMask (0) ;
|
||||
|
||||
/* Call the thread entry point */
|
||||
if (curr_tcb && curr_tcb->entry_point)
|
||||
{
|
||||
curr_tcb->entry_point(curr_tcb->entry_param);
|
||||
}
|
||||
|
||||
/* Not reached - threads should never return from the entry point */
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* \b archThreadContextInit
|
||||
*
|
||||
* Documented in atomThreads.
|
||||
*
|
||||
*/
|
||||
void
|
||||
archThreadContextInit (ATOM_TCB *tcb_ptr, void *stack_top, void (*entry_point)(uint32_t), uint32_t entry_param)
|
||||
{
|
||||
tcb_ptr->sp_save_ptr = stack_top;
|
||||
tcb_ptr->entry_param = entry_param ;
|
||||
tcb_ptr->entry_point = entry_point ;
|
||||
|
||||
tcb_ptr->context = atomvmContextCreate (1) ;
|
||||
atomvmContextInit (tcb_ptr->context, (unsigned int *)stack_top, thread_shell, entry_param, 0) ;
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* \b archFirstThreadRestore
|
||||
*
|
||||
* Documented in atomThreads.
|
||||
*
|
||||
*/
|
||||
void
|
||||
archFirstThreadRestore(ATOM_TCB * p_sp_new)
|
||||
{
|
||||
atomvmContextSwitch (0, p_sp_new->context) ;
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* \b archContextSwitch
|
||||
*
|
||||
* Documented in atomThreads.
|
||||
*
|
||||
*/
|
||||
void
|
||||
archContextSwitch (ATOM_TCB * p_sp_old, ATOM_TCB * p_sp_new)
|
||||
{
|
||||
atomvmContextSwitch (p_sp_old->context, p_sp_new->context) ;
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* \b archTimerTickIrqHandler
|
||||
*
|
||||
* System timer tick interrupt handler.
|
||||
*
|
||||
*/
|
||||
void archTimerTickIrqHandler (void)
|
||||
{
|
||||
atomIntEnter();
|
||||
|
||||
/* Call the OS system tick handler */
|
||||
atomTimerTick();
|
||||
|
||||
/* Call the interrupt exit routine */
|
||||
atomIntExit(TRUE);
|
||||
}
|
||||
62
ports/atomvm/atomport.h
Normal file
62
ports/atomvm/atomport.h
Normal file
@@ -0,0 +1,62 @@
|
||||
/*
|
||||
* Copyright (c) 2012, Natie van Rooyen. 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_PORT_H__
|
||||
#define __ATOM_PORT_H__
|
||||
|
||||
#include "atomvm.h"
|
||||
|
||||
#define SYSTEM_TICKS_PER_SEC 100
|
||||
/* Size of each stack entry / stack alignment size (e.g. 32 bits) */
|
||||
#define STACK_ALIGN_SIZE sizeof(unsigned int)
|
||||
|
||||
/**
|
||||
* Architecture-specific types.
|
||||
* Most of these are available from stdint.h on this platform, which is
|
||||
* included above.
|
||||
*/
|
||||
#define POINTER void *
|
||||
#define ATOM_TLS HATOMVM_CONTEXT context ;
|
||||
|
||||
|
||||
/* Critical region protection */
|
||||
#define CRITICAL_STORE unsigned int __atom_int_mask
|
||||
#define CRITICAL_START() __atom_int_mask = atomvmInterruptMask(1)
|
||||
#define CRITICAL_END() atomvmInterruptMask(__atom_int_mask)
|
||||
|
||||
#define ATOM_TLS HATOMVM_CONTEXT context ;
|
||||
|
||||
/* Function prototypes */
|
||||
extern void atomvmRun (void) ;
|
||||
extern void archTimerTickIrqHandler (void) ;
|
||||
|
||||
/* The instance of the atomvm for this port */
|
||||
extern HATOMVM the_atomvm ;
|
||||
|
||||
#endif /* __ATOM_PORT_H__ */
|
||||
63
ports/atomvm/atomuser.h
Normal file
63
ports/atomvm/atomuser.h
Normal file
@@ -0,0 +1,63 @@
|
||||
/*
|
||||
* Copyright (c) 2010,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.
|
||||
*/
|
||||
#if 1
|
||||
#ifndef __ATOM_USER_H__
|
||||
#define __ATOM_USER_H__
|
||||
|
||||
|
||||
/* Portable uint8_t and friends not available from stdint.h on this platform */
|
||||
#include <windows.h>
|
||||
|
||||
|
||||
#define SYSTEM_MEMALIGN sizeof (unsigned int)
|
||||
|
||||
|
||||
typedef unsigned int uintptr_t ;
|
||||
typedef int intptr_t ;
|
||||
typedef unsigned int uint32_t ;
|
||||
typedef unsigned short uint16_t ;
|
||||
typedef unsigned char uint8_t ;
|
||||
typedef int int32_t ;
|
||||
typedef short int16_t ;
|
||||
typedef char int8_t ;
|
||||
|
||||
|
||||
/**
|
||||
* Architecture-specific types.
|
||||
* Most of these are available from stdint.h on this platform, which is
|
||||
* included above.
|
||||
*/
|
||||
#define POINTER void *
|
||||
|
||||
|
||||
#define ATOM_TLS HATOMVM_CONTEXT context ;
|
||||
|
||||
|
||||
#endif /* __ATOM_USER_H__ */
|
||||
#endif
|
||||
852
ports/atomvm/atomvm.c
Normal file
852
ports/atomvm/atomvm.c
Normal file
@@ -0,0 +1,852 @@
|
||||
/*
|
||||
* Copyright (c) 2010, Natie van Rooyen. 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
|
||||
* Atom Virtual Machine.
|
||||
*
|
||||
*
|
||||
* This module implements the virtual machine.
|
||||
*
|
||||
*
|
||||
* \b Functions contained in this module:\n
|
||||
*
|
||||
* \b Function prototypes used for controlling the atom virtual machine: \n
|
||||
*
|
||||
* \li atomvmCtrlCreate(): .
|
||||
* \li atomvmCtrlRun(): .
|
||||
* \li atomvmCtrlIntRequest(): .
|
||||
* \li atomvmCtrlClose(): .
|
||||
*
|
||||
* \b Function prototypes for use by the atom virtual machine: \n
|
||||
*
|
||||
* \li atomvmInterruptMask(): .
|
||||
* \li atomvmContextCreate(): .
|
||||
* \li atomvmContextSwitch(): .
|
||||
* \li atomvmContextDesrtroy(): .
|
||||
* \li atomvmWriteThreadId(): .
|
||||
* \li atomvmReadThreadId(): .
|
||||
* \li atomvmInterruptWait(): .
|
||||
* \li atomvmGetVmId(): .
|
||||
*
|
||||
* \b Function prototypes to be implemted in the atom virtual machine: \n
|
||||
*
|
||||
* \li __atomvmReset(): .
|
||||
* \li __atomvmClose(): .
|
||||
*
|
||||
*/
|
||||
|
||||
#include "atomvm.h"
|
||||
#include <string.h>
|
||||
#include <windows.h>
|
||||
|
||||
|
||||
|
||||
#define CONTEXT_VM (CONTEXT_INTEGER | CONTEXT_CONTROL | CONTEXT_SEGMENTS)
|
||||
|
||||
/* Data types */
|
||||
|
||||
/* Forward declarations */
|
||||
typedef struct ATOMVM_S * PATOMVM ;
|
||||
typedef struct ATOMVM_CALLBACK_S * PATOMVM_CALLBACK ;
|
||||
typedef struct ATOMVM_CONTEXT_S * PATOMVM_CONTEXT ;
|
||||
|
||||
typedef uint32_t (*ATOMVM_CALLBACK_F) (PATOMVM, PATOMVM_CALLBACK) ;
|
||||
|
||||
typedef struct ATOMVM_CALLBACK_S {
|
||||
|
||||
/* Address of callback function */
|
||||
volatile ATOMVM_CALLBACK_F callback ;
|
||||
|
||||
/* Synchronization lock, the virtual machine will be suspended during
|
||||
the callback. Regular WIN32 synchronization methods cant be used
|
||||
because SuspendThread() is used on the vm thread. */
|
||||
volatile uint32_t lock ;
|
||||
|
||||
/* Result of the call */
|
||||
volatile uint32_t result ;
|
||||
|
||||
} ATOMVM_CALLBACK, *PATOMVM_CALLBACK ;
|
||||
|
||||
|
||||
/* ATOMVM_CALLBACK_CONTEXT is the parameter for a ATOMVM_CALLBACK_F call
|
||||
that take as parameter a pointer to a ATOMVM_CONTEXT to operate on */
|
||||
typedef struct ATOMVM_CALLBACK_CONTEXT_S {
|
||||
|
||||
ATOMVM_CALLBACK callback ;
|
||||
|
||||
/* Context the callback function will operate on */
|
||||
volatile PATOMVM_CONTEXT pcontext ;
|
||||
|
||||
} ATOMVM_CALLBACK_CONTEXT, *PATOMVM_CALLBACK_CONTEXT ;
|
||||
|
||||
|
||||
/* ATOMVM_CALLBACK_CONTEXT_SWITCH is the parameter for a ATOMVM_CALLBACK_F call
|
||||
that take as parameter a pointer to a ATOMVM_CONTEXT to operate on */
|
||||
typedef struct ATOMVM_CALLBACK_CONTEXT_SWITCH_S {
|
||||
|
||||
ATOMVM_CALLBACK callback ;
|
||||
|
||||
/* Context the callback function will operate on */
|
||||
volatile PATOMVM_CONTEXT p_old_context ;
|
||||
volatile PATOMVM_CONTEXT p_new_context ;
|
||||
|
||||
} ATOMVM_CALLBACK_CONTEXT_SWITCH, *PATOMVM_CALLBACK_CONTEXT_SWITCH ;
|
||||
|
||||
/* ATOMVM_CALLBACK_INT_REQUEST is the parameter for a ATOMVM_CALLBACK_F call
|
||||
that take as parameter a pointer to to the function that will be called in
|
||||
an interrupt context */
|
||||
typedef struct ATOMVM_CALLBACK_INT_REQUEST_S {
|
||||
|
||||
ATOMVM_CALLBACK callback ;
|
||||
|
||||
/* Function pointer the callback will call */
|
||||
void (*isr) (void) ;
|
||||
|
||||
} ATOMVM_CALLBACK_INT_REQUEST, *PATOMVM_CALLBACK_INT_REQUEST ;
|
||||
|
||||
/* ATOMVM_CONTEXT saves the state of a context created by
|
||||
atomvmContextCreate() and sheduled by atomvmContextSwitch(). */
|
||||
typedef struct ATOMVM_CONTEXT_S {
|
||||
|
||||
/* A virtual machine thread context. These are saved and restored
|
||||
during context initialization and context switches */
|
||||
CONTEXT context ;
|
||||
|
||||
/* When entering a critical section the interrupt_mask is
|
||||
set for the context. Interrupts will only occur while
|
||||
the interrupt_mask is zero. */
|
||||
volatile uint32_t interrupt_mask ;
|
||||
uint32_t thread_id ;
|
||||
|
||||
} ATOMVM_CONTEXT, *PATOMVM_CONTEXT ;
|
||||
|
||||
/* ATOMVM defines the state of an instance to an atomvm. It is created
|
||||
by a call to atomvmCtrlCreate(). */
|
||||
typedef struct ATOMVM_S {
|
||||
|
||||
uint32_t atomvm_id ;
|
||||
|
||||
/* Thread the virtual machine will run in */
|
||||
HANDLE vm_thread ;
|
||||
|
||||
/* Handles to events and mutexes used for synchronization */
|
||||
HANDLE atomvm_call ;
|
||||
HANDLE atomvm_int ;
|
||||
HANDLE atomvm_int_complete ;
|
||||
HANDLE atomvm_close ;
|
||||
|
||||
/* next ISR */
|
||||
volatile void (*isr)(void) ;
|
||||
/* True if in an ISR */
|
||||
volatile uint32_t status_isr ;
|
||||
|
||||
/* The current context that was scheduled by a call
|
||||
to atomvmContextSwitch() */
|
||||
PATOMVM_CONTEXT current_context ;
|
||||
|
||||
/* Service call address, synchronization lock, parameters
|
||||
and, return value for the current service call */
|
||||
PATOMVM_CALLBACK service_call ;
|
||||
|
||||
/* Context for startup, before any context was scheduled */
|
||||
ATOMVM_CONTEXT atom_init_context ;
|
||||
|
||||
} ATOMVM, *PATOMVM ;
|
||||
|
||||
|
||||
/* Global declarations */
|
||||
volatile uint32_t g_atomvm_id = 0 ;
|
||||
volatile DWORD g_atomvm_tls_idx ;
|
||||
|
||||
|
||||
/* Forward declaration for the atom virtual machine thread */
|
||||
static DWORD WINAPI vm_thread (LPVOID lpParameter) ;
|
||||
|
||||
|
||||
/**
|
||||
* \ingroup atomvm
|
||||
* \b atomvmCtrlCreate
|
||||
*
|
||||
* This is an atomvm controll function used by a controlling thread.
|
||||
*
|
||||
* Initializes the virtual machine.
|
||||
*
|
||||
* @param[out] atomvm Handle to the virtual machine to create.
|
||||
*
|
||||
* @return Zero on failure.
|
||||
*/
|
||||
uint32_t
|
||||
atomvmCtrlCreate (HATOMVM *atomvm)
|
||||
{
|
||||
PATOMVM patomvm = 0 ;
|
||||
|
||||
patomvm = (PATOMVM) malloc (sizeof(struct ATOMVM_S)) ;
|
||||
|
||||
if (patomvm) {
|
||||
|
||||
memset (patomvm, 0, sizeof(struct ATOMVM_S)) ;
|
||||
|
||||
patomvm->atomvm_id = InterlockedIncrement(&g_atomvm_id) - 1 ;
|
||||
|
||||
if (patomvm->atomvm_id == 0) {
|
||||
g_atomvm_tls_idx = TlsAlloc () ;
|
||||
}
|
||||
|
||||
patomvm->atomvm_call = CreateEvent (NULL, TRUE, FALSE, 0) ;
|
||||
patomvm->atomvm_int = CreateEvent (NULL, TRUE, FALSE, 0) ;
|
||||
patomvm->atomvm_int_complete = CreateEvent (NULL, FALSE, TRUE, 0) ;
|
||||
patomvm->atomvm_close = CreateEvent (NULL, TRUE, FALSE, 0) ;
|
||||
|
||||
ATOMVM_ASSERT(patomvm->atomvm_call && patomvm->atomvm_int && patomvm->atomvm_int_complete &&
|
||||
patomvm->atomvm_close, _T("ResumeThread failed")) ;
|
||||
|
||||
patomvm->vm_thread = CreateThread (NULL, 0, vm_thread, (void*)patomvm, CREATE_SUSPENDED, NULL) ;
|
||||
|
||||
ATOMVM_ASSERT(patomvm->vm_thread, _T("CreateThread failed")) ;
|
||||
|
||||
patomvm->atom_init_context.interrupt_mask = 1 ;
|
||||
patomvm->current_context = &patomvm->atom_init_context ;
|
||||
|
||||
*atomvm = (HATOMVM)patomvm ;
|
||||
|
||||
}
|
||||
|
||||
return patomvm != 0 ;
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* \ingroup atomvm
|
||||
* \b atomvmCtrlRun
|
||||
*
|
||||
* After a call to atomvmCtrlCreate this function start the atom virtual machine.
|
||||
* The calling thread will be used to manage interrupts and service calls in
|
||||
* the virtual machine. This function will not return untill atomvmCtrlClose
|
||||
* is called.
|
||||
*
|
||||
* @param[in] atomvm Handle to the virtual machine created by atomvmCtrlCreate.
|
||||
* @param[in] flags not used.
|
||||
*
|
||||
* @return None
|
||||
*/
|
||||
void
|
||||
atomvmCtrlRun (HATOMVM atomvm, uint32_t flags)
|
||||
{
|
||||
PATOMVM patomvm = (PATOMVM) atomvm ;
|
||||
HANDLE wait[3] ;
|
||||
uint32_t res ;
|
||||
uint32_t wait_object ;
|
||||
PATOMVM_CALLBACK service_call ;
|
||||
#if defined DEBUG || defined _DEBUG
|
||||
BOOL tls_res =
|
||||
#endif
|
||||
TlsSetValue (g_atomvm_tls_idx, (void*) atomvm) ;
|
||||
ATOMVM_ASSERT(tls_res, _T("TlsSetValue failed")) ;
|
||||
|
||||
ResumeThread (patomvm->vm_thread) ;
|
||||
|
||||
wait[0] = patomvm->atomvm_call ;
|
||||
wait[1] = patomvm->atomvm_int ;
|
||||
wait[2] = patomvm->atomvm_close ;
|
||||
|
||||
for(;;) {
|
||||
|
||||
wait_object = WaitForMultipleObjects (3, wait,FALSE,INFINITE) ;
|
||||
|
||||
if (wait_object == WAIT_OBJECT_0) {
|
||||
|
||||
service_call = patomvm->service_call ;
|
||||
while (!service_call->lock) {
|
||||
SwitchToThread () ;
|
||||
}
|
||||
|
||||
while ((res = SuspendThread (patomvm->vm_thread)) == (DWORD)-1) ;
|
||||
ATOMVM_ASSERT(res == 0 , _T("SuspendThread failed")) ;
|
||||
#if (_WIN32_WINNT >= 0x0600)
|
||||
/*
|
||||
This is used for multi processor machines to ensure the thread
|
||||
is stopped before executing the next instruction. Set
|
||||
_WIN32_WINNT < 0x0600 if you are running Windows XP */
|
||||
FlushProcessWriteBuffers ();
|
||||
#endif
|
||||
InterlockedExchange (&service_call->result, service_call->callback (patomvm, service_call)) ;
|
||||
InterlockedExchange (&service_call->lock, 0) ;
|
||||
ResetEvent (patomvm->atomvm_call) ;
|
||||
res = ResumeThread (patomvm->vm_thread) ;
|
||||
ATOMVM_ASSERT(res == 1 , _T("ResumeThread failed")) ;
|
||||
|
||||
}
|
||||
|
||||
else if (wait_object == WAIT_OBJECT_0 + 1) {
|
||||
|
||||
if (patomvm->current_context->interrupt_mask == 0) {
|
||||
|
||||
while ((res = SuspendThread (patomvm->vm_thread)) == (DWORD)-1) ;
|
||||
ATOMVM_ASSERT(res == 0 , _T("SuspendThread failed")) ;
|
||||
#if (_WIN32_WINNT >= 0x0600)
|
||||
/*
|
||||
This is used for multi processor machines to ensure the thread
|
||||
is stopped before executing the next instruction. Set
|
||||
_WIN32_WINNT < 0x0600 if you are running Windows XP */
|
||||
FlushProcessWriteBuffers ();
|
||||
#endif
|
||||
if (patomvm->current_context->interrupt_mask == 0) {
|
||||
|
||||
patomvm->status_isr++ ;
|
||||
patomvm->isr () ;
|
||||
patomvm->status_isr-- ;
|
||||
|
||||
res = ResumeThread (patomvm->vm_thread) ;
|
||||
ATOMVM_ASSERT(res == 1 , _T("ResumeThread failed")) ;
|
||||
|
||||
ResetEvent (patomvm->atomvm_int) ;
|
||||
InterlockedExchange ((volatile uint32_t*)&patomvm->isr, 0) ;
|
||||
SetEvent (patomvm->atomvm_int_complete) ;
|
||||
|
||||
} else {
|
||||
|
||||
res = ResumeThread (patomvm->vm_thread) ;
|
||||
ATOMVM_ASSERT(res == 1 , _T("ResumeThread failed")) ;
|
||||
SwitchToThread () ;
|
||||
|
||||
}
|
||||
|
||||
} else {
|
||||
|
||||
SwitchToThread () ;
|
||||
|
||||
}
|
||||
|
||||
} else if (wait_object == WAIT_OBJECT_0 + 2) {
|
||||
|
||||
break ;
|
||||
|
||||
} else {
|
||||
|
||||
ATOMVM_ASSERT(res == 1 , _T("WaitForMultipleObjects failed")) ;
|
||||
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* \ingroup atomvm
|
||||
* \b atomvmCtrlClose
|
||||
*
|
||||
* This is an atomvm controll function used by a controlling thread
|
||||
* and must not be called from the atom virtual machine.
|
||||
*
|
||||
* Closes the virtual machine and release all memory and handles created
|
||||
* in atomvmCtrlCreate.
|
||||
*
|
||||
* ToDo: more testing.
|
||||
*
|
||||
* @param[in] atomvm Handle to the virtual machine created by atomvmCtrlCreate.
|
||||
*
|
||||
* @return None
|
||||
*/
|
||||
void
|
||||
atomvmCtrlClose (HATOMVM atomvm)
|
||||
{
|
||||
PATOMVM patomvm = (PATOMVM) atomvm ;
|
||||
DWORD code ;
|
||||
|
||||
__atomvmClose () ;
|
||||
|
||||
SetEvent (patomvm->atomvm_close) ;
|
||||
do {
|
||||
SwitchToThread () ;
|
||||
GetExitCodeThread (patomvm->vm_thread, &code) ;
|
||||
} while (code == STILL_ACTIVE) ;
|
||||
|
||||
CloseHandle (patomvm->atomvm_call) ;
|
||||
CloseHandle (patomvm->atomvm_int) ;
|
||||
CloseHandle (patomvm->atomvm_int_complete) ;
|
||||
CloseHandle (patomvm->atomvm_close) ;
|
||||
CloseHandle (patomvm->vm_thread) ;
|
||||
|
||||
// TlsFree (g_atomvm_tls_idx) ;
|
||||
|
||||
free (atomvm) ;
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* \b invokeCallback
|
||||
*
|
||||
* Invokes callback functions in the context of the controll thread as
|
||||
* requested from the virtual machine. In case this callback came from inside,
|
||||
* an isr it is already in the conrtext of the controll thread and the callback
|
||||
* routine is called directly.
|
||||
*
|
||||
* The atom virtual machine thread is suspended during the callback.
|
||||
*
|
||||
* @param[in] patomvm Pointer to the virtual machine created by atomvmCtrlCreate.
|
||||
* @param[in] callback Callback function.
|
||||
* @param[in/out] context Context the function will operate on.
|
||||
*
|
||||
* @return Zero on failure, try to call GetLastError().
|
||||
*/
|
||||
uint32_t
|
||||
invokeCallback (PATOMVM patomvm, ATOMVM_CALLBACK_F callback, PATOMVM_CALLBACK service_call)
|
||||
{
|
||||
uint32_t res ;
|
||||
|
||||
if (patomvm->status_isr == 0) {
|
||||
|
||||
service_call->lock = 0 ;
|
||||
service_call->callback = callback ;
|
||||
patomvm->service_call = service_call ;
|
||||
|
||||
SetEvent (patomvm->atomvm_call) ;
|
||||
InterlockedIncrement (&service_call->lock) ;
|
||||
while (service_call->lock != 0) ;
|
||||
res = service_call->result ;
|
||||
|
||||
} else {
|
||||
|
||||
res = callback (patomvm, service_call) ;
|
||||
|
||||
}
|
||||
|
||||
return res ;
|
||||
}
|
||||
|
||||
|
||||
/*
|
||||
* \b getAtomvm
|
||||
*
|
||||
* Get the atomvm instance for the calling thread
|
||||
*
|
||||
* @return atomvm instance
|
||||
*/
|
||||
__inline PATOMVM
|
||||
getAtomvm (void)
|
||||
{
|
||||
PATOMVM patomvm = (PATOMVM) TlsGetValue (g_atomvm_tls_idx) ;
|
||||
|
||||
ATOMVM_ASSERT(patomvm , _T("TlsGetValue failed")) ;
|
||||
|
||||
return patomvm ;
|
||||
}
|
||||
|
||||
/**
|
||||
* \ingroup atomvm
|
||||
* \b atomvmInterruptMask
|
||||
*
|
||||
* This function is to be used by the atom virtual machine.
|
||||
*
|
||||
* This function will mask interrupts for the current atomvm context.
|
||||
*
|
||||
* @param[in] mask zero enables interrupts any other value masks interrupts.
|
||||
*
|
||||
* @return Interrupt mask before the function call.
|
||||
*/
|
||||
int32_t
|
||||
atomvmInterruptMask (uint32_t mask)
|
||||
{
|
||||
PATOMVM patomvm = getAtomvm () ;
|
||||
int32_t interrupts = 0;
|
||||
|
||||
if (patomvm->status_isr == 0) {
|
||||
interrupts = InterlockedExchange (&patomvm->current_context->interrupt_mask, mask) ;
|
||||
}
|
||||
|
||||
return interrupts ;
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* \ingroup atomvm
|
||||
* \b atomvmCtrlIntRequest
|
||||
*
|
||||
* This is an atomvm controll function used by external threads
|
||||
* and must not be called from the atom virtual machine.
|
||||
*
|
||||
* This function requests an interrupt service routine to be called in the
|
||||
* context of the atom virtual machine.
|
||||
*
|
||||
* The call will return immediately after the interrupt was scheduled.
|
||||
* The call will block while a previously scheduled interrupt is in progress.
|
||||
*
|
||||
* @param[in] atomvm Handle to the virtual machine created by atomvmCtrlCreate.
|
||||
* @param[in] isr The address of the interrupt service routine.
|
||||
*
|
||||
* @return None
|
||||
*/
|
||||
void
|
||||
atomvmCtrlIntRequest (HATOMVM atomvm, void (*isr) (void))
|
||||
{
|
||||
PATOMVM patomvm = (PATOMVM) atomvm ;
|
||||
|
||||
WaitForSingleObject (patomvm->atomvm_int_complete, INFINITE) ;
|
||||
while (InterlockedCompareExchange ((volatile uint32_t *)&patomvm->isr, (uint32_t)isr, 0) != 0) {
|
||||
SwitchToThread() ;
|
||||
}
|
||||
SetEvent (patomvm->atomvm_int) ;
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* \b callbackContextCreate
|
||||
*
|
||||
* This function is invoked from the controll thread after a call to atomvmContextCreate.
|
||||
*
|
||||
* The atom virtual machine is suspended while this function is called.
|
||||
*
|
||||
* @param[in] patomvm Pointer to the virtual machine created by atomvmCtrlCreate.
|
||||
* @param[out] context Context to be initialized.
|
||||
*
|
||||
* @return Zero on failure, try to call GetLastError().
|
||||
*/
|
||||
uint32_t
|
||||
callbackContextCreate (PATOMVM patomvm, PATOMVM_CALLBACK callback)
|
||||
{
|
||||
PATOMVM_CALLBACK_CONTEXT context_switch = (PATOMVM_CALLBACK_CONTEXT)callback;
|
||||
CONTEXT * pcontext = &context_switch->pcontext->context ;
|
||||
|
||||
pcontext->ContextFlags = CONTEXT_VM ;
|
||||
|
||||
return GetThreadContext (patomvm->vm_thread, pcontext) ;
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* \ingroup atomvm
|
||||
* \b atomvmContextCreate
|
||||
*
|
||||
* This function is to be used by the atom virtual machine.
|
||||
*
|
||||
* This function creates a atomvm thread context that can be scheduled
|
||||
* by atomvmContextSwitch.
|
||||
*
|
||||
* @param[in] interrupt_mask initial interrupt mask of the thread.
|
||||
*
|
||||
* @return Handle to the context of the thread created.
|
||||
*/
|
||||
HATOMVM_CONTEXT
|
||||
atomvmContextCreate (uint32_t interrupt_mask)
|
||||
{
|
||||
uint32_t res ;
|
||||
PATOMVM patomvm = getAtomvm () ;
|
||||
PATOMVM_CONTEXT new_context = (PATOMVM_CONTEXT)malloc (sizeof(ATOMVM_CONTEXT)) ;
|
||||
CONTEXT* pcontext = &new_context->context ;
|
||||
ATOMVM_CALLBACK_CONTEXT context_init ;
|
||||
|
||||
context_init.pcontext = new_context ;
|
||||
new_context->interrupt_mask = interrupt_mask ;
|
||||
new_context->thread_id = (uint32_t) -1 ;
|
||||
res = invokeCallback (patomvm, callbackContextCreate, (PATOMVM_CALLBACK)&context_init) ;
|
||||
|
||||
if (res) {
|
||||
return (HATOMVM_CONTEXT)new_context ;
|
||||
} else {
|
||||
free (new_context) ;
|
||||
}
|
||||
|
||||
return 0 ;
|
||||
}
|
||||
|
||||
/**
|
||||
* \ingroup atomvm
|
||||
* \b atomvmContextInit
|
||||
*
|
||||
* This function is to be used by the atom virtual machine.
|
||||
*
|
||||
* This function initialize a atomvm thread context that can be scheduled
|
||||
* by atomvmContextSwitch.
|
||||
*
|
||||
* @param[out] context Handle to the context of the thread that are allocated
|
||||
* by the caller.
|
||||
* @param[in] stack Stack top.
|
||||
* @param[in] entry Entry point of the thread.
|
||||
* @param[in] arg argument passed on the stack as first parameter.
|
||||
* @param[in] exit exit function to return to.
|
||||
* @param[in] status status for exit function.
|
||||
*
|
||||
* @return Zero on failure, try to call GetLastError().
|
||||
*/
|
||||
uint32_t
|
||||
atomvmContextInit (HATOMVM_CONTEXT context, uint32_t* stack, void (*entry)(uint32_t), uint32_t arg, void (*exit)(uint32_t))
|
||||
{
|
||||
uint32_t res = 0 ;
|
||||
PATOMVM_CONTEXT new_context = (PATOMVM_CONTEXT)context ;
|
||||
CONTEXT* pcontext = &new_context->context ;
|
||||
|
||||
*stack-- = arg;
|
||||
*stack = (uint32_t)exit ;
|
||||
|
||||
pcontext->Ebp = (uint32_t)stack ;
|
||||
pcontext->Esp = (uint32_t)stack ;
|
||||
pcontext->Eip = (uint32_t)entry ;
|
||||
|
||||
return res ;
|
||||
}
|
||||
|
||||
/**
|
||||
* \b callbackContextSwitch
|
||||
*
|
||||
* This function is invoked from the controll thread after a call to atomvmContextSwitch.
|
||||
*
|
||||
* The atom virtual machine is suspended while this function is called.
|
||||
*
|
||||
* @param[in] patomvm Pointer to the virtual machine created by atomvmCtrlCreate.
|
||||
* @param[out] context Context to be scheduled.
|
||||
*
|
||||
* @return Zero on failure, try to call GetLastError().
|
||||
*/
|
||||
uint32_t
|
||||
callbackContextSwitch (PATOMVM patomvm, PATOMVM_CALLBACK callback)
|
||||
{
|
||||
uint32_t res1 = 1 ;
|
||||
uint32_t res2 ;
|
||||
PATOMVM_CALLBACK_CONTEXT_SWITCH context_switch = (PATOMVM_CALLBACK_CONTEXT_SWITCH)callback ;
|
||||
|
||||
if (context_switch->p_old_context) {
|
||||
res1 = GetThreadContext (patomvm->vm_thread, &context_switch->p_old_context->context) ;
|
||||
ATOMVM_ASSERT(res1 , _T("GetThreadContext failed")) ;
|
||||
}
|
||||
|
||||
patomvm->current_context = context_switch->p_new_context ;
|
||||
res2 = SetThreadContext (patomvm->vm_thread, &context_switch->p_new_context->context) ;
|
||||
ATOMVM_ASSERT(res2 , _T("SetThreadContext failed")) ;
|
||||
|
||||
return res1 & res2 ;
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* \ingroup atomvm
|
||||
* \b atomvmContextSwitch
|
||||
*
|
||||
* This function is to be used by the atom virtual machine.
|
||||
*
|
||||
* This function schedules a thread for the context created by atomvmContextCreate.
|
||||
*
|
||||
* @param[in] new_context The context to schedule.
|
||||
*
|
||||
* @return Zero on failure, try to call GetLastError().
|
||||
*/
|
||||
uint32_t
|
||||
atomvmContextSwitch (HATOMVM_CONTEXT old_context, HATOMVM_CONTEXT new_context)
|
||||
{
|
||||
PATOMVM patomvm = getAtomvm () ;
|
||||
ATOMVM_CALLBACK_CONTEXT_SWITCH context_switch ;
|
||||
|
||||
context_switch.p_old_context = (PATOMVM_CONTEXT) old_context ;
|
||||
context_switch.p_new_context = (PATOMVM_CONTEXT) new_context ;
|
||||
|
||||
return invokeCallback (patomvm, callbackContextSwitch, (PATOMVM_CALLBACK)&context_switch) ;
|
||||
}
|
||||
|
||||
/**
|
||||
* \ingroup atomvm
|
||||
* \b atomvmContextDesrtroy
|
||||
*
|
||||
* This function is to be used by the atom virtual machine.
|
||||
*
|
||||
* This functiondestroyes a atomvm context created by atomvmContextCreate.
|
||||
*
|
||||
* @param[in] context The context to destroy.
|
||||
*
|
||||
* @return None
|
||||
*/
|
||||
void
|
||||
atomvmContextDesrtroy (HATOMVM_CONTEXT context)
|
||||
{
|
||||
PATOMVM patomvm = getAtomvm () ;
|
||||
|
||||
ATOMVM_ASSERT(patomvm->current_context != (PATOMVM_CONTEXT)context,
|
||||
_T("atomvmContextDesrtroy failed")) ;
|
||||
|
||||
free((void*)context) ;
|
||||
}
|
||||
|
||||
/**
|
||||
* \ingroup atomvm
|
||||
* \b atomvmWriteThreadId
|
||||
*
|
||||
* Write a thread ID.
|
||||
*
|
||||
* Write a thread ID for the current context.
|
||||
*
|
||||
* @param[in] thread_id thread_id.
|
||||
*
|
||||
* @return None
|
||||
*/
|
||||
void
|
||||
atomvmWriteThreadId (uint32_t thread_id)
|
||||
{
|
||||
PATOMVM patomvm = getAtomvm () ;
|
||||
|
||||
patomvm->current_context->thread_id = thread_id ;
|
||||
}
|
||||
|
||||
/**
|
||||
* \ingroup atomvm
|
||||
* \b atomvmReadThreadId
|
||||
*
|
||||
* Write a thread ID.
|
||||
*
|
||||
* Read a thread ID for the current context.
|
||||
*
|
||||
* @return thread_id
|
||||
*/
|
||||
uint32_t
|
||||
atomvmReadThreadId (void)
|
||||
{
|
||||
PATOMVM patomvm = getAtomvm () ;
|
||||
|
||||
return patomvm->current_context->thread_id ;
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* \ingroup atomvm
|
||||
* \b atomvmGetVmId
|
||||
*
|
||||
* Returns an identifier for the virtual machine. This is zero for the first
|
||||
* virtual machine created with atomvmCtrlCreate(), 1 for the second and so on.
|
||||
*
|
||||
* @return The atom vm ID
|
||||
*/
|
||||
uint32_t
|
||||
atomvmGetVmId (void)
|
||||
{
|
||||
PATOMVM patomvm = getAtomvm () ;
|
||||
|
||||
return patomvm->atomvm_id ;
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* \b callbackInterruptWait
|
||||
*
|
||||
* This function is invoked from the controll thread after a call to atomvmInterruptWait().
|
||||
*
|
||||
* The atom virtual machine is suspended while this function is called.
|
||||
*
|
||||
* @param[in] patomvm Pointer to the virtual machine created by atomvmCtrlCreate.
|
||||
* @param[out] callback Callback parameter.
|
||||
*
|
||||
* @return Zero on failure, try to call GetLastError().
|
||||
*/
|
||||
uint32_t
|
||||
callbackIntWait (PATOMVM patomvm, PATOMVM_CALLBACK callback)
|
||||
{
|
||||
WaitForSingleObject (patomvm->atomvm_int_complete, INFINITE) ;
|
||||
return WaitForSingleObject (patomvm->atomvm_int, INFINITE) == WAIT_OBJECT_0 ;
|
||||
}
|
||||
|
||||
/**
|
||||
* \ingroup atomvm
|
||||
* \b atomvmInterruptWait
|
||||
*
|
||||
* This function is to be used by the atom virtual machine.
|
||||
*
|
||||
* This function if for synchronization between multiple
|
||||
* atom vms.
|
||||
*
|
||||
*
|
||||
* @return void.
|
||||
*/
|
||||
void
|
||||
atomvmIntWait (void)
|
||||
{
|
||||
PATOMVM patomvm = getAtomvm () ;
|
||||
ATOMVM_CALLBACK callback ;
|
||||
|
||||
invokeCallback (patomvm, callbackIntWait, (PATOMVM_CALLBACK)&callback) ;
|
||||
}
|
||||
|
||||
/**
|
||||
* \b callbackIntRequest
|
||||
*
|
||||
* This function is invoked from the controll thread after a call to atomvmIntRequest().
|
||||
*
|
||||
* The atom virtual machine is suspended while this function is called.
|
||||
*
|
||||
* @param[in] patomvm Pointer to the virtual machine created by atomvmCtrlCreate.
|
||||
* @param[in] callback Callback parameter.
|
||||
*
|
||||
* @return Zero on failure, try to call GetLastError().
|
||||
*/
|
||||
uint32_t
|
||||
callbackIntRequest (PATOMVM patomvm, PATOMVM_CALLBACK callback)
|
||||
{
|
||||
PATOMVM_CALLBACK_INT_REQUEST int_request = (PATOMVM_CALLBACK_INT_REQUEST)callback ;
|
||||
|
||||
int_request->isr () ;
|
||||
return 1 ;
|
||||
}
|
||||
|
||||
/**
|
||||
* \ingroup atomvm
|
||||
* \b atomvmIntRequest
|
||||
*
|
||||
* This function is to be used by the atom virtual machine.
|
||||
*
|
||||
* @param[in] isr Function that will be called from the controll thread.
|
||||
*
|
||||
* @return void.
|
||||
*/
|
||||
void
|
||||
atomvmIntRequest (void (*isr) (void))
|
||||
{
|
||||
PATOMVM patomvm = getAtomvm () ;
|
||||
ATOMVM_CALLBACK_INT_REQUEST callback ;
|
||||
|
||||
callback.isr = isr ;
|
||||
invokeCallback (patomvm, callbackIntRequest, (PATOMVM_CALLBACK)&callback) ;
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* \b vm_thread
|
||||
*
|
||||
* Windows thread in which the atom virtual machine will execute.
|
||||
*
|
||||
* __atomvmReset() runs the virtual machie and should only return after
|
||||
* __atomvmClose() was called.
|
||||
*
|
||||
* @return None.
|
||||
*/
|
||||
DWORD WINAPI
|
||||
vm_thread (LPVOID lpParameter)
|
||||
{
|
||||
BOOL res = TlsSetValue (g_atomvm_tls_idx, lpParameter) ;
|
||||
|
||||
ATOMVM_ASSERT(res, _T("TlsSetValue failed")) ;
|
||||
__atomvmReset () ;
|
||||
return 0 ;
|
||||
}
|
||||
121
ports/atomvm/atomvm.h
Normal file
121
ports/atomvm/atomvm.h
Normal file
@@ -0,0 +1,121 @@
|
||||
/*
|
||||
* Copyright (c) 2010, Natie van Rooyen. 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.
|
||||
*/
|
||||
|
||||
|
||||
|
||||
/** \mainpage \ref atomvm
|
||||
* \defgroup atomvm Atomvm API
|
||||
*
|
||||
* @authors Natie van Rooyen
|
||||
*
|
||||
* @section intro Introduction
|
||||
* Atomvm is a tiny virtual machine that runs on Windows and can be debugged
|
||||
* from an IDE like Microsoft Visual C++ Express. The primary purpose of this
|
||||
* virtual machine is for the evaluation of Real Time Operating Systems (like
|
||||
* Atomthreads) and the development and testing of modules for this Real Time
|
||||
* Operating System on a Windows machine.
|
||||
*
|
||||
* Atomvm makes use of the Windows API functions GetThreadContext() and
|
||||
* SetThreadContext() to create multiple virtual contexts or threads inside a
|
||||
* single Windows thread. Atomvm also simulates interrupts with an interrupt
|
||||
* mask accessible from the Atomvm threads. External events can be queued as
|
||||
* interrupts to Atomvm, for example a timer loop generating system timer tick
|
||||
* interrupts for a Real Time Operating System ported to Atomvm.
|
||||
*
|
||||
* */
|
||||
|
||||
#ifndef __ATOMVM_H__
|
||||
#define __ATOMVM_H__
|
||||
|
||||
#include <crtdbg.h>
|
||||
#include "types.h"
|
||||
|
||||
|
||||
#if defined _DEBUG || defined DEBUG
|
||||
#define ATOMVM_ASSERT(x, msg) _ASSERT((x))
|
||||
#else
|
||||
#define ATOMVM_ASSERT(x, msg)
|
||||
#endif
|
||||
|
||||
|
||||
#define ATOMVM_MAX_VM 8
|
||||
|
||||
/* Forward declarations */
|
||||
|
||||
/* This is an opaque handle to an instance of an atomvm created
|
||||
by a call to atomvmCtrlCreate() */
|
||||
typedef struct ATOMVM* HATOMVM ;
|
||||
|
||||
/* This is an opaque handle to an atomvm context created
|
||||
by a call to atomvmContextCreate() */
|
||||
typedef struct ATOMVM_CONTEXT* HATOMVM_CONTEXT ;
|
||||
|
||||
|
||||
/* Function prototypes used for controlling the atom virtual machine */
|
||||
extern uint32_t atomvmCtrlCreate (HATOMVM* atomvm) ;
|
||||
extern void atomvmCtrlRun (HATOMVM atomvm, uint32_t flags) ;
|
||||
extern void atomvmCtrlIntRequest (HATOMVM atomvm, void (*isr) (void)) ;
|
||||
extern void atomvmCtrlClose (HATOMVM atomvm) ;
|
||||
|
||||
/* Function prototypes for use by the atom virtual machine from within the
|
||||
call to __atomvmReset(). */
|
||||
extern int32_t atomvmInterruptMask (uint32_t mask) ;
|
||||
extern HATOMVM_CONTEXT atomvmContextCreate (uint32_t interrupt_mask) ;
|
||||
extern uint32_t atomvmContextInit (HATOMVM_CONTEXT context, uint32_t* stack,
|
||||
void (*entry)(uint32_t), uint32_t arg, void (*exit)(uint32_t)) ;
|
||||
extern uint32_t atomvmContextSwitch (HATOMVM_CONTEXT old_context, HATOMVM_CONTEXT new_context) ;
|
||||
extern void atomvmContextDesrtroy (HATOMVM_CONTEXT context) ;
|
||||
extern void atomvmWriteThreadId (uint32_t thread_id) ;
|
||||
extern uint32_t atomvmReadThreadId (void) ;
|
||||
extern void atomvmIntWait (void) ;
|
||||
extern void atomvmIntRequest (void (*isr) (void)) ;
|
||||
extern uint32_t atomvmGetVmId (void) ;
|
||||
|
||||
|
||||
/**
|
||||
* \ingroup atomvm
|
||||
* \b __atomvmReset
|
||||
*
|
||||
* Function prototype to be implemented as entry point for the atom virtual machine.
|
||||
*
|
||||
* @return void.
|
||||
*/
|
||||
extern void __atomvmReset (void) ;
|
||||
|
||||
/**
|
||||
* \ingroup atomvm
|
||||
* \b __atomvmClose
|
||||
*
|
||||
* Function prototype to be implemted in the atom virtual machine
|
||||
*
|
||||
* @return void.
|
||||
*/
|
||||
extern void __atomvmClose (void) ;
|
||||
|
||||
#endif /* __ATOMVM_H__ */
|
||||
270
ports/atomvm/msvc/main.c
Normal file
270
ports/atomvm/msvc/main.c
Normal file
@@ -0,0 +1,270 @@
|
||||
#include <stdio.h>
|
||||
#include <windows.h>
|
||||
|
||||
#include "atom.h"
|
||||
#include "atomvm.h"
|
||||
#include "atomport.h"
|
||||
|
||||
|
||||
// #define UNIT_TESTS
|
||||
#ifdef UNIT_TESTS
|
||||
extern uint32_t test_start (void) ;
|
||||
#endif
|
||||
|
||||
#define TEST_THREADS 47
|
||||
|
||||
#define TEST_STACK_BYTE_SIZE 0x10000
|
||||
#define IDLE_STACK_BYTE_SIZE 0x10000
|
||||
#define MONITOR_STACK_BYTE_SIZE 0x10000
|
||||
|
||||
|
||||
|
||||
static unsigned char idle_stack[IDLE_STACK_BYTE_SIZE] ;
|
||||
static unsigned char monitor_stack[MONITOR_STACK_BYTE_SIZE] ;
|
||||
static unsigned char test_stack[TEST_THREADS+1][TEST_STACK_BYTE_SIZE] ;
|
||||
static unsigned int test_counter[TEST_THREADS+1] = {0} ;
|
||||
|
||||
static unsigned int test2_counter = 0 ;
|
||||
static unsigned int test3_counter = 0 ;
|
||||
static unsigned int test_isr_count = 0 ;
|
||||
|
||||
|
||||
static unsigned char test2_stack[TEST_STACK_BYTE_SIZE] ;
|
||||
static unsigned char test3_stack[TEST_STACK_BYTE_SIZE] ;
|
||||
static unsigned char test_idle_stack[TEST_STACK_BYTE_SIZE] ;
|
||||
|
||||
|
||||
static uint8_t test_prio[60] = {
|
||||
001,010,100,200,250, 200,200,200,200,200,
|
||||
150,150,150,150,150, 250,250,250,250,250,
|
||||
101,102,103,104,105, 202,204,206,208,210,
|
||||
150,150,150,150,150, 250,250,250,250,250,
|
||||
121,122,123,124,125, 061,063,065,067,061,
|
||||
150,150,150,150,150, 250,250,250,250,250
|
||||
};
|
||||
|
||||
static uint32_t test_interv[60] = {
|
||||
001,001,001,001,001, 002,003,004,005,006,
|
||||
015,015,015,015,015, 025,024,023,022,021,
|
||||
905,005,005,005,805, 050,051,052,053,054,
|
||||
015,015,015,015,015, 025,024,023,022,021,
|
||||
030,030,030,030,030, 070,071,072,073,474,
|
||||
005,006,007,007,001, 001,001,003,003,005
|
||||
};
|
||||
|
||||
|
||||
ATOM_TCB test_tcb[TEST_THREADS+1] ;
|
||||
ATOM_TCB monitor_tcb ;
|
||||
ATOM_TCB test2_tcb ;
|
||||
ATOM_TCB test3_tcb ;
|
||||
ATOM_TCB test_idle_tcb ;
|
||||
|
||||
DWORD WINAPI isr_thread_proc (LPVOID lpParameter) ;
|
||||
static HANDLE isr_thread_1 ;
|
||||
static HANDLE isr_thread_2 ;
|
||||
static HANDLE isr_thread_3 ;
|
||||
static HANDLE isr_thread_4 ;
|
||||
|
||||
void
|
||||
ipi_sr()
|
||||
{
|
||||
printf("ipi\r\n") ;
|
||||
}
|
||||
|
||||
void
|
||||
monitor_thread (uint32_t parm)
|
||||
{
|
||||
CRITICAL_STORE;
|
||||
int i ;
|
||||
int c = 0 ;
|
||||
ATOM_TCB *tcb ;
|
||||
static unsigned int idle_1 = 0, idle_2 = 0, int_count = 0 ;
|
||||
unsigned int delta_idle_1 , delta_idle_2 , delta_int_count ;
|
||||
|
||||
tcb = atomCurrentContext() ;
|
||||
|
||||
for (;;)
|
||||
{
|
||||
|
||||
CRITICAL_START();
|
||||
|
||||
printf("Monitor # %04d (%08d)\n", c++, atomTimeGet()) ;
|
||||
printf("-------------------------\n") ;
|
||||
|
||||
for (i=0; i<TEST_THREADS/3; i++) {
|
||||
printf("Thr %.2d cnt %08d\t",i,test_counter[i]);
|
||||
printf("Thr %.2d cnt %08d\t",i+TEST_THREADS/3,test_counter[i+TEST_THREADS/3]);
|
||||
printf("Thr %.2d cnt %08d\n",i+TEST_THREADS*2/3,test_counter[i+TEST_THREADS*2/3]);
|
||||
}
|
||||
|
||||
delta_idle_1 = test2_counter - idle_1 ;
|
||||
delta_idle_2 = test3_counter - idle_2 ;
|
||||
delta_int_count = test_isr_count - int_count ;
|
||||
printf("\nIdle Threadd 1 Counter = %d %d %d\nIdle Theadrd 2 Counter = %d %d %d\nInterrupt Counter = %d %d %d",
|
||||
test2_counter, delta_idle_1, (unsigned int)(test2_counter / c),
|
||||
test3_counter, delta_idle_2, (unsigned int)(test3_counter / c),
|
||||
test_isr_count, delta_int_count, (unsigned int)(test_isr_count / c));
|
||||
printf ("\n\n") ;
|
||||
idle_1 = test2_counter ;
|
||||
idle_2 = test3_counter ;
|
||||
int_count = test_isr_count ;
|
||||
CRITICAL_END();
|
||||
//for (i=0; i<100;i++) {
|
||||
// atomvmInterruptWait () ;
|
||||
//}
|
||||
atomTimerDelay (450) ;
|
||||
//atomvmScheduleIpi (atomvmGetVmId(), (uint32_t) ipi_sr) ;
|
||||
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
void
|
||||
test_thread (uint32_t parm)
|
||||
{
|
||||
CRITICAL_STORE;
|
||||
|
||||
for (;;) {
|
||||
|
||||
atomTimerDelay (test_interv[parm]) ;
|
||||
CRITICAL_START();
|
||||
test_counter[parm]++ ;
|
||||
CRITICAL_END();
|
||||
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
void
|
||||
test2_thread (uint32_t parm)
|
||||
{
|
||||
CRITICAL_STORE;
|
||||
|
||||
for (;;) {
|
||||
|
||||
CRITICAL_START();
|
||||
test2_counter++ ;
|
||||
CRITICAL_END();
|
||||
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
void test3_thread(uint32_t parm)
|
||||
{
|
||||
CRITICAL_STORE;
|
||||
|
||||
for (;;) {
|
||||
|
||||
CRITICAL_START();
|
||||
test3_counter++ ;
|
||||
CRITICAL_END();
|
||||
|
||||
}
|
||||
}
|
||||
|
||||
#ifdef UNIT_TESTS
|
||||
void unit_test_thread(uint32_t parm)
|
||||
{
|
||||
unsigned int failures ;
|
||||
|
||||
failures = test_start () ;
|
||||
printf ("test_start %d failures\n", failures) ;
|
||||
|
||||
while(1) {
|
||||
|
||||
atomTimerDelay (100);
|
||||
}
|
||||
|
||||
}
|
||||
#endif
|
||||
|
||||
void
|
||||
__atomvmReset ()
|
||||
{
|
||||
unsigned int i ;
|
||||
|
||||
atomOSInit(&idle_stack[0], IDLE_STACK_BYTE_SIZE, 1) ;
|
||||
#ifndef UNIT_TESTS
|
||||
|
||||
for (i=0; i< TEST_THREADS;i++) {
|
||||
atomThreadCreate ((ATOM_TCB *)&test_tcb[i], test_prio[i], test_thread, i, &test_stack[i][0], TEST_STACK_BYTE_SIZE, 1);
|
||||
}
|
||||
|
||||
atomThreadCreate ((ATOM_TCB *)&monitor_tcb, 50, monitor_thread, 0, &monitor_stack[0], MONITOR_STACK_BYTE_SIZE, 1);
|
||||
|
||||
atomThreadCreate ((ATOM_TCB *)&test2_tcb, 253, test2_thread, 0, &test2_stack[0], TEST_STACK_BYTE_SIZE, 1);
|
||||
atomThreadCreate ((ATOM_TCB *)&test3_tcb, 253, test3_thread, 0, &test3_stack[0], TEST_STACK_BYTE_SIZE, 1);
|
||||
#else
|
||||
|
||||
atomThreadCreate ((ATOM_TCB *)&test2_tcb, 16, unit_test_thread, 0, &test2_stack[], TEST_STACK_BYTE_SIZE, 1);
|
||||
|
||||
#endif
|
||||
|
||||
atomOSStart() ;
|
||||
}
|
||||
|
||||
void
|
||||
__atomvmClose ()
|
||||
{
|
||||
|
||||
}
|
||||
|
||||
|
||||
|
||||
void
|
||||
test_isr (void)
|
||||
{
|
||||
static int i = 0 ;
|
||||
test_isr_count++ ;
|
||||
if (i++==25) {
|
||||
//Sleep(3) ;
|
||||
i = 0;
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
|
||||
DWORD WINAPI
|
||||
isr_thread_proc (LPVOID lpParameter)
|
||||
{
|
||||
int i = 0 ;
|
||||
int x ;
|
||||
int y = rand() % 100 ;
|
||||
while (1) {
|
||||
atomvmCtrlIntRequest (the_atomvm, test_isr) ;
|
||||
if (i++==y) {
|
||||
x = rand() % 50 ;
|
||||
Sleep (x) ;
|
||||
y = rand() % 100 ;
|
||||
i = 0 ;
|
||||
}
|
||||
}
|
||||
|
||||
return 0 ;
|
||||
}
|
||||
|
||||
|
||||
void
|
||||
main ()
|
||||
{
|
||||
atomvmRun () ;
|
||||
|
||||
#ifndef UNIT_TESTS
|
||||
isr_thread_1 = CreateThread (NULL, 0, isr_thread_proc, 0, CREATE_SUSPENDED, NULL) ;
|
||||
isr_thread_2 = CreateThread (NULL, 0, isr_thread_proc, 0, CREATE_SUSPENDED, NULL) ;
|
||||
isr_thread_3 = CreateThread (NULL, 0, isr_thread_proc, 0, CREATE_SUSPENDED, NULL) ;
|
||||
isr_thread_4 = CreateThread (NULL, 0, isr_thread_proc, 0, CREATE_SUSPENDED, NULL) ;
|
||||
|
||||
ResumeThread (isr_thread_1) ;
|
||||
ResumeThread (isr_thread_2) ;
|
||||
ResumeThread (isr_thread_3) ;
|
||||
ResumeThread (isr_thread_4) ;
|
||||
#endif
|
||||
|
||||
while (1) {
|
||||
Sleep(1) ;
|
||||
atomvmCtrlIntRequest (the_atomvm, archTimerTickIrqHandler) ;
|
||||
}
|
||||
|
||||
}
|
||||
20
ports/atomvm/msvc/msvc.sln
Normal file
20
ports/atomvm/msvc/msvc.sln
Normal file
@@ -0,0 +1,20 @@
|
||||
|
||||
Microsoft Visual Studio Solution File, Format Version 11.00
|
||||
# Visual C++ Express 2010
|
||||
Project("{8BC9CEB8-8B4A-11D0-8D11-00A0C91BC942}") = "msvc", "msvc.vcxproj", "{77095EB1-4988-4A04-8751-69C63C7C541B}"
|
||||
EndProject
|
||||
Global
|
||||
GlobalSection(SolutionConfigurationPlatforms) = preSolution
|
||||
Debug|Win32 = Debug|Win32
|
||||
Release|Win32 = Release|Win32
|
||||
EndGlobalSection
|
||||
GlobalSection(ProjectConfigurationPlatforms) = postSolution
|
||||
{77095EB1-4988-4A04-8751-69C63C7C541B}.Debug|Win32.ActiveCfg = Debug|Win32
|
||||
{77095EB1-4988-4A04-8751-69C63C7C541B}.Debug|Win32.Build.0 = Debug|Win32
|
||||
{77095EB1-4988-4A04-8751-69C63C7C541B}.Release|Win32.ActiveCfg = Release|Win32
|
||||
{77095EB1-4988-4A04-8751-69C63C7C541B}.Release|Win32.Build.0 = Release|Win32
|
||||
EndGlobalSection
|
||||
GlobalSection(SolutionProperties) = preSolution
|
||||
HideSolutionNode = FALSE
|
||||
EndGlobalSection
|
||||
EndGlobal
|
||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user