summaryrefslogtreecommitdiffstats
path: root/cpukit/score/cpu/m32c/context_init.c
blob: 713f8fa1613bf6099b01ba41af90d39f5971116e (plain) (blame)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
/*
 *  COPYRIGHT (c) 1989-2008.
 *  On-Line Applications Research Corporation (OAR).
 *
 *  The license and distribution terms for this file may be
 *  found in the file LICENSE in this distribution or at
 *  http://www.rtems.com/license/LICENSE.
 */

#ifdef HAVE_CONFIG_H
#include "config.h"
#endif

#include <stdint.h>
#include <rtems/system.h>

typedef struct {
  uint16_t sbLow;
  uint16_t sbHigh;       /* push/pop sb */
  uint16_t flg;          /* push/pop flg */
  uint32_t a1;           /* pushm */
  uint32_t a0;
  uint32_t r0r2;
  uint32_t r1r3;
  uint16_t frameLow;     /* exitd */
  uint16_t frameHigh;
  uint16_t startLow;
  uint16_t startHigh;
  uint16_t zero;
} Starting_Frame;

#if defined(__r8c_cpu__)
  #warning "_get_sb: not implemented on R8C"
  #define _get_sb( _sb )
#else
  #define _get_sb( _sb ) \
    __asm__ volatile( "stc sb, %0" : "=r" (_sb))
#endif

void _CPU_Context_Initialize(
  Context_Control  *the_context,
  uint32_t         *stack_base,
  size_t            size,
  uint32_t          new_level,
  void             *entry_point,
  bool              is_fp
)
{
  void *stackEnd = stack_base;
  register uint32_t sb;
  Starting_Frame *frame;

  _get_sb( sb );
  stackEnd += size;

  frame = (Starting_Frame *)stackEnd;
  frame--;

  frame->zero = 0;
  frame->sbLow = ((uint32_t)sb) & 0xffff;
  frame->sbHigh = ((uint32_t)sb >> 16) & 0xffff;
  frame->flg = 0x80;        /* User stack */
  if ( !new_level )         /* interrupt level 0 --> enabled */
    frame->flg |= 0x40;
  frame->a0 = 0x01020304;
  frame->a1 =0xa1a2a3a4;
  frame->r0r2 = 0;
  frame->r1r3 = 0;
#if defined(__r8c_cpu__)
  #warning "not implemented on R8C"
#else
  frame->frameLow  = (uint16_t) (((uint32_t)frame)  & 0xffff);
  frame->frameHigh = (uint16_t) (((uint32_t)frame >> 16) & 0xffff);
  frame->startLow  = (uint16_t) (((uint32_t)entry_point) & 0xffff);
  frame->startHigh = (uint16_t) (((uint32_t)entry_point >> 16) & 0xffff);
#endif
  the_context->sp = (uintptr_t)frame;
  the_context->fb = (uintptr_t)&frame->frameLow;
}