原始版本
This commit is contained in:
13
RT_Thread/libcpu/ti-dsp/c28x/SConscript
Normal file
13
RT_Thread/libcpu/ti-dsp/c28x/SConscript
Normal file
@ -0,0 +1,13 @@
|
||||
# RT-Thread building script for component
|
||||
|
||||
from building import *
|
||||
|
||||
Import('rtconfig')
|
||||
|
||||
cwd = GetCurrentDir()
|
||||
src = Glob('*.c') + Glob('*.cpp') + Glob('*.s')
|
||||
CPPPATH = [cwd]
|
||||
|
||||
group = DefineGroup('libcpu', src, depend = [''], CPPPATH = CPPPATH)
|
||||
|
||||
Return('group')
|
||||
360
RT_Thread/libcpu/ti-dsp/c28x/context.s
Normal file
360
RT_Thread/libcpu/ti-dsp/c28x/context.s
Normal file
@ -0,0 +1,360 @@
|
||||
;
|
||||
; Copyright (c) 2006-2022, RT-Thread Development Team
|
||||
;
|
||||
; SPDX-License-Identifier: Apache-2.0
|
||||
;
|
||||
; Change Logs:
|
||||
; Date Author Notes
|
||||
; 2018-09-01 xuzhuoyi the first version.
|
||||
; 2019-06-17 zhaoxiaowei fix bugs of old c28x interrupt api.
|
||||
; 2019-07-03 zhaoxiaowei add _rt_hw_calc_csb function to support __rt_ffs.
|
||||
; 2019-12-05 xiaolifan add support for hardware fpu32
|
||||
; 2022-06-21 guyunjie trim pendsv (RTOSINT_Handler)
|
||||
; 2022-08-24 guyunjie fix bugs in context switching
|
||||
; 2022-10-15 guyunjie add zero-latency interrupt
|
||||
|
||||
.ref rt_interrupt_to_thread
|
||||
.ref rt_interrupt_from_thread
|
||||
.ref rt_thread_switch_interrupt_flag
|
||||
|
||||
.def rtosint_handler
|
||||
.def rt_hw_get_st0
|
||||
.def rt_hw_get_st1
|
||||
.def rt_hw_calc_csb
|
||||
.def rt_hw_context_switch_interrupt
|
||||
.def rt_hw_context_switch
|
||||
.def rt_hw_context_switch_to
|
||||
.def rt_hw_interrupt_thread_switch
|
||||
.def rt_hw_interrupt_disable
|
||||
.def rt_hw_interrupt_enable
|
||||
|
||||
;importing settings from compiler and config
|
||||
.cdecls C,NOLIST
|
||||
%{
|
||||
#include <rtconfig.h>
|
||||
|
||||
#ifdef __TMS320C28XX_FPU32__
|
||||
#define __FPU32__ 1
|
||||
#else
|
||||
#define __FPU32__ 0
|
||||
#endif
|
||||
|
||||
#ifdef __TMS320C28XX_FPU64__
|
||||
#define __FPU64__ 1
|
||||
#else
|
||||
#define __FPU64__ 0
|
||||
#endif
|
||||
|
||||
#ifdef __TMS320C28XX_VCRC__
|
||||
#define __VCRC__ 1
|
||||
#else
|
||||
#define __VCRC__ 0
|
||||
#endif
|
||||
|
||||
#ifdef RT_USING_ZERO_LATENCY
|
||||
#define ZERO_LATENCY 1
|
||||
#ifndef ZERO_LATENCY_INT_MASK
|
||||
#error ZERO_LATENCY_INT_MASK must be defined for zero latency interrupt
|
||||
#elif ZERO_LATENCY_INT_MASK & 0x8000
|
||||
#error RTOS bit (0x8000) must not be set in ZERO_LATENCY_INT_MASK
|
||||
#endif
|
||||
#else
|
||||
#define ZERO_LATENCY 0
|
||||
#endif
|
||||
%}
|
||||
|
||||
|
||||
.text
|
||||
.newblock
|
||||
|
||||
;
|
||||
; rt_base_t rt_hw_interrupt_disable();
|
||||
;
|
||||
.asmfunc
|
||||
rt_hw_interrupt_disable:
|
||||
.if ZERO_LATENCY
|
||||
MOV AL, IER
|
||||
AND IER, #ZERO_LATENCY_INT_MASK
|
||||
.else
|
||||
PUSH ST1
|
||||
SETC INTM
|
||||
POP AL
|
||||
.endif
|
||||
MOV AH, #0
|
||||
LRETR
|
||||
.endasmfunc
|
||||
|
||||
;
|
||||
; void rt_hw_interrupt_enable(rt_base_t level);
|
||||
;
|
||||
.asmfunc
|
||||
rt_hw_interrupt_enable:
|
||||
.if ZERO_LATENCY
|
||||
MOV IER, AL
|
||||
.else
|
||||
PUSH AL
|
||||
POP ST1
|
||||
.endif
|
||||
LRETR
|
||||
.endasmfunc
|
||||
|
||||
;
|
||||
; void rt_hw_context_switch(rt_uint32 from, rt_uint32 to);
|
||||
; ACC --> from
|
||||
; SP[4] --> to
|
||||
;
|
||||
.asmfunc
|
||||
rt_hw_context_switch_interrupt:
|
||||
|
||||
; ACC, XAR4-7 are "save on call" following TI C28x C/C++ compiler convention
|
||||
; and therefore can be used in a function without being saved on stack first
|
||||
; (the compiler has already saved it before the call).
|
||||
; Reference: TMS320C28x Optimizing CC++ Compiler
|
||||
; note this convention is only applicable to normal functions not to isrs
|
||||
|
||||
MOVL XAR6, ACC
|
||||
MOVL XAR4, *-SP[4]
|
||||
; set rt_thread_switch_interrupt_flag to 1
|
||||
MOVL XAR5, #rt_thread_switch_interrupt_flag
|
||||
MOVL ACC, *XAR5
|
||||
BF reswitch2, NEQ ; ACC!=0
|
||||
MOVB ACC, #1
|
||||
MOVL *XAR5, ACC
|
||||
|
||||
MOVL XAR5, #rt_interrupt_from_thread ; set rt_interrupt_from_thread
|
||||
MOVL *XAR5, XAR6
|
||||
|
||||
reswitch2:
|
||||
MOVL XAR5, #rt_interrupt_to_thread ; set rt_interrupt_to_thread
|
||||
MOVL *XAR5, XAR4
|
||||
LRETR
|
||||
.endasmfunc
|
||||
|
||||
;
|
||||
; void rt_hw_context_switch(rt_uint32 from, rt_uint32 to);
|
||||
; ACC --> from
|
||||
; SP[4] --> to
|
||||
;
|
||||
.asmfunc
|
||||
rt_hw_context_switch:
|
||||
|
||||
MOVL XAR6, ACC
|
||||
MOVL XAR4, *-SP[4]
|
||||
; set rt_thread_switch_interrupt_flag to 1
|
||||
MOVL XAR5, #rt_thread_switch_interrupt_flag
|
||||
MOVL ACC, *XAR5
|
||||
BF reswitch1, NEQ ; ACC!=0
|
||||
MOVB ACC, #1
|
||||
MOVL *XAR5, ACC
|
||||
|
||||
MOVL XAR5, #rt_interrupt_from_thread ; set rt_interrupt_from_thread
|
||||
MOVL *XAR5, XAR6
|
||||
|
||||
reswitch1:
|
||||
MOVL XAR5, #rt_interrupt_to_thread ; set rt_interrupt_to_thread
|
||||
MOVL *XAR5, XAR4
|
||||
OR IFR, #0x8000
|
||||
LRETR
|
||||
.endasmfunc
|
||||
;
|
||||
; * void rt_hw_context_switch_to(rt_uint32 to);
|
||||
; * ACC --> to
|
||||
;
|
||||
.asmfunc
|
||||
rt_hw_context_switch_to:
|
||||
; get to thread
|
||||
MOVL XAR5, #rt_interrupt_to_thread
|
||||
MOVL *XAR5, ACC
|
||||
|
||||
; set from thread to 0
|
||||
MOVL XAR5, #rt_interrupt_from_thread
|
||||
MOVL XAR4, #0
|
||||
MOVL *XAR5, XAR4
|
||||
|
||||
; set interrupt flag to 1
|
||||
MOVL XAR5, #rt_thread_switch_interrupt_flag
|
||||
MOVL XAR4, #1
|
||||
MOVL *XAR5, XAR4
|
||||
|
||||
; trigger rtos interrupt
|
||||
OR IFR, #0x8000
|
||||
OR IER, #0x8000
|
||||
CLRC INTM
|
||||
|
||||
; never reach here!
|
||||
.endasmfunc
|
||||
|
||||
.asmfunc
|
||||
rtosint_handler:
|
||||
|
||||
.if ZERO_LATENCY
|
||||
; mask out non-critical interrupts and enable global interrupt
|
||||
; so rtosint_handler won't block critical interrupts
|
||||
AND IER, #ZERO_LATENCY_INT_MASK
|
||||
CLRC INTM
|
||||
.endif
|
||||
|
||||
MOVL ACC, *-SP[4]
|
||||
MOV AR0, AL ; save original IER
|
||||
|
||||
PUSH AR1H:AR0H
|
||||
PUSH XAR2
|
||||
|
||||
; get rt_thread_switch_interrupt_flag
|
||||
MOVL XAR1, #rt_thread_switch_interrupt_flag
|
||||
MOVL ACC, *XAR1
|
||||
BF rtosint_exit, EQ ; rtos_int already handled
|
||||
|
||||
; clear rt_thread_switch_interrupt_flag to 0
|
||||
MOVL XAR2, #0
|
||||
MOVL *XAR1, XAR2
|
||||
|
||||
MOVL XAR1, #rt_interrupt_from_thread
|
||||
MOVL ACC, *XAR1
|
||||
BF switch_to_thread, EQ ; skip register save at the first time
|
||||
|
||||
PUSH XAR3
|
||||
PUSH XAR4
|
||||
PUSH XAR5
|
||||
PUSH XAR6
|
||||
PUSH XAR7
|
||||
PUSH XT
|
||||
PUSH RPC
|
||||
|
||||
.if __FPU32__
|
||||
PUSH RB
|
||||
MOV32 *SP++, STF
|
||||
MOV32 *SP++, R0H
|
||||
MOV32 *SP++, R1H
|
||||
MOV32 *SP++, R2H
|
||||
MOV32 *SP++, R3H
|
||||
MOV32 *SP++, R4H
|
||||
MOV32 *SP++, R5H
|
||||
MOV32 *SP++, R6H
|
||||
MOV32 *SP++, R7H
|
||||
.endif
|
||||
|
||||
.if __FPU64__
|
||||
MOV32 *SP++, R0L
|
||||
MOV32 *SP++, R1L
|
||||
MOV32 *SP++, R2L
|
||||
MOV32 *SP++, R3L
|
||||
MOV32 *SP++, R4L
|
||||
MOV32 *SP++, R5L
|
||||
MOV32 *SP++, R6L
|
||||
MOV32 *SP++, R7L
|
||||
.endif
|
||||
|
||||
.if __VCRC__
|
||||
VMOV32 *SP++, VCRC
|
||||
VMOV32 *SP++, VSTATUS
|
||||
VMOV32 *SP++, VCRCPOLY
|
||||
VMOV32 *SP++, VCRCSIZE
|
||||
.endif
|
||||
|
||||
MOVL ACC, *XAR1
|
||||
MOVL XAR1, ACC
|
||||
MOVZ AR2, @SP ; get from thread stack pointer
|
||||
MOVL *XAR1, XAR2 ; update from thread stack pointer
|
||||
|
||||
switch_to_thread:
|
||||
MOVL XAR1, #rt_interrupt_to_thread
|
||||
MOVL ACC, *XAR1
|
||||
MOVL XAR1, ACC
|
||||
MOVL ACC, *XAR1
|
||||
MOV @SP, AL ; load thread stack pointer
|
||||
|
||||
.if __VCRC__
|
||||
VMOV32 VCRCSIZE, *--SP
|
||||
VMOV32 VCRCPOLY, *--SP
|
||||
VMOV32 VSTATUS, *--SP
|
||||
VMOV32 VCRC, *--SP
|
||||
.endif
|
||||
|
||||
.if __FPU64__
|
||||
MOV32 R7L, *--SP
|
||||
MOV32 R6L, *--SP
|
||||
MOV32 R5L, *--SP
|
||||
MOV32 R4L, *--SP
|
||||
MOV32 R3L, *--SP
|
||||
MOV32 R2L, *--SP
|
||||
MOV32 R1L, *--SP
|
||||
MOV32 R0L, *--SP
|
||||
.endif
|
||||
|
||||
.if __FPU32__
|
||||
MOV32 R7H, *--SP
|
||||
MOV32 R6H, *--SP
|
||||
MOV32 R5H, *--SP
|
||||
MOV32 R4H, *--SP
|
||||
MOV32 R3H, *--SP
|
||||
MOV32 R2H, *--SP
|
||||
MOV32 R1H, *--SP
|
||||
MOV32 R0H, *--SP
|
||||
MOV32 STF, *--SP
|
||||
POP RB
|
||||
.endif
|
||||
|
||||
POP RPC
|
||||
POP XT
|
||||
POP XAR7
|
||||
POP XAR6
|
||||
POP XAR5
|
||||
POP XAR4
|
||||
POP XAR3
|
||||
|
||||
rtosint_exit:
|
||||
; do not restore interrupt here: to be restored according to the
|
||||
; switched-to context during IRET (automaticlly by hardware)
|
||||
|
||||
POP XAR2
|
||||
POP AR1H:AR0H
|
||||
|
||||
MOVL ACC , *-SP[4]
|
||||
MOV AL, AR0
|
||||
MOVL *-SP[4], ACC
|
||||
|
||||
IRET
|
||||
.endasmfunc
|
||||
|
||||
.asmfunc
|
||||
rt_hw_get_st0:
|
||||
PUSH ST0
|
||||
POP AL
|
||||
LRETR
|
||||
.endasmfunc
|
||||
|
||||
.asmfunc
|
||||
rt_hw_get_st1:
|
||||
PUSH ST1
|
||||
POP AL
|
||||
LRETR
|
||||
.endasmfunc
|
||||
|
||||
; C28x do not have a build-in "__ffs" func in its C compiler.
|
||||
; We can use the "Count Sign Bits" (CSB) instruction to make one.
|
||||
; CSB will return the number of 0's minus 1 above the highest set bit.
|
||||
; The count is placed in T. For example:
|
||||
; ACC T maxbit
|
||||
; 0x00000001 30 0
|
||||
; 0x00000010 26 4
|
||||
; 0x000001FF 22 8
|
||||
; 0x000001F0 22 8
|
||||
.asmfunc
|
||||
rt_hw_calc_csb:
|
||||
MOV AH, #0
|
||||
CSB ACC ; T = no. of sign bits - 1
|
||||
MOVU ACC, T ; ACC = no. of sign bits - 1
|
||||
SUBB ACC, #30 ; ACC = ACC - 30
|
||||
ABS ACC ; ACC = |ACC|
|
||||
LRETR
|
||||
.endasmfunc
|
||||
|
||||
; compatible with old version
|
||||
.asmfunc
|
||||
rt_hw_interrupt_thread_switch:
|
||||
LRETR
|
||||
NOP
|
||||
.endasmfunc
|
||||
|
||||
.end
|
||||
188
RT_Thread/libcpu/ti-dsp/c28x/cpuport.c
Normal file
188
RT_Thread/libcpu/ti-dsp/c28x/cpuport.c
Normal file
@ -0,0 +1,188 @@
|
||||
/*
|
||||
* Copyright (c) 2006-2022, RT-Thread Development Team
|
||||
*
|
||||
* SPDX-License-Identifier: Apache-2.0
|
||||
*
|
||||
* Change Logs:
|
||||
* Date Author Notes
|
||||
* 2018-09-01 xuzhuoyi the first version.
|
||||
* 2019-07-03 zhaoxiaowei add support for __rt_ffs.
|
||||
* 2019-12-05 xiaolifan add support for hardware fpu32
|
||||
* 2022-10-17 guyunjie add support for hardware fpu64 and vcrc
|
||||
*/
|
||||
|
||||
#include <rthw.h>
|
||||
|
||||
#define DBG_TAG "cpu.ti.c28x"
|
||||
#define DBG_LVL DBG_INFO
|
||||
#include <rtdbg.h>
|
||||
|
||||
extern volatile rt_atomic_t rt_interrupt_nest;
|
||||
|
||||
/* exception and interrupt handler table */
|
||||
rt_uint32_t rt_interrupt_from_thread;
|
||||
rt_uint32_t rt_interrupt_to_thread;
|
||||
rt_uint32_t rt_thread_switch_interrupt_flag;
|
||||
/* exception hook */
|
||||
static rt_err_t (*rt_exception_hook)(void *context) = RT_NULL;
|
||||
|
||||
extern rt_uint16_t rt_hw_get_st0(void);
|
||||
extern rt_uint16_t rt_hw_get_st1(void);
|
||||
extern int rt_hw_calc_csb(int value);
|
||||
|
||||
|
||||
struct exception_stack_frame
|
||||
{
|
||||
rt_uint32_t t_st0;
|
||||
rt_uint32_t acc;
|
||||
rt_uint32_t p;
|
||||
rt_uint32_t ar1_ar0;
|
||||
rt_uint32_t dp_st1;
|
||||
rt_uint32_t dbgstat_ier;
|
||||
rt_uint32_t return_address;
|
||||
};
|
||||
|
||||
struct stack_frame
|
||||
{
|
||||
struct exception_stack_frame exception_stack_frame;
|
||||
|
||||
/* r4 ~ r11 register */
|
||||
rt_uint16_t ar0h;
|
||||
rt_uint16_t ar1h;
|
||||
rt_uint32_t xar2;
|
||||
rt_uint32_t xar3;
|
||||
rt_uint32_t xar4;
|
||||
rt_uint32_t xar5;
|
||||
rt_uint32_t xar6;
|
||||
rt_uint32_t xar7;
|
||||
rt_uint32_t xt;
|
||||
rt_uint32_t rpc;
|
||||
|
||||
#ifdef __TMS320C28XX_FPU32__
|
||||
rt_uint32_t rb;
|
||||
rt_uint32_t stf;
|
||||
rt_uint32_t r0h;
|
||||
rt_uint32_t r1h;
|
||||
rt_uint32_t r2h;
|
||||
rt_uint32_t r3h;
|
||||
rt_uint32_t r4h;
|
||||
rt_uint32_t r5h;
|
||||
rt_uint32_t r6h;
|
||||
rt_uint32_t r7h;
|
||||
#endif
|
||||
|
||||
#ifdef __TMS320C28XX_FPU64__
|
||||
rt_uint32_t r0l;
|
||||
rt_uint32_t r1l;
|
||||
rt_uint32_t r2l;
|
||||
rt_uint32_t r3l;
|
||||
rt_uint32_t r4l;
|
||||
rt_uint32_t r5l;
|
||||
rt_uint32_t r6l;
|
||||
rt_uint32_t r7l;
|
||||
#endif
|
||||
|
||||
#ifdef __TMS320C28XX_VCRC__
|
||||
rt_uint32_t vcrc;
|
||||
rt_uint32_t vstatus;
|
||||
rt_uint32_t vcrcpoly;
|
||||
rt_uint32_t vcrcsize;
|
||||
#endif
|
||||
|
||||
};
|
||||
|
||||
rt_uint8_t *rt_hw_stack_init(void *tentry,
|
||||
void *parameter,
|
||||
rt_uint8_t *stack_addr,
|
||||
void *texit)
|
||||
{
|
||||
struct stack_frame *stack_frame;
|
||||
rt_uint8_t *stk;
|
||||
unsigned long i;
|
||||
|
||||
stk = stack_addr;
|
||||
stk = (rt_uint8_t *)RT_ALIGN((rt_uint32_t)stk, 2);
|
||||
stk += 1; /*to work around the stack alignment*/
|
||||
|
||||
stack_frame = (struct stack_frame *)stk;
|
||||
|
||||
/* zero all registers */
|
||||
for (i = 0; i < sizeof(struct stack_frame) / sizeof(rt_uint32_t); i ++)
|
||||
{
|
||||
((rt_uint32_t *)stack_frame)[i] = 0;
|
||||
}
|
||||
|
||||
/* configure special registers*/
|
||||
stack_frame->exception_stack_frame.dp_st1 = 0x00000A08;
|
||||
stack_frame->xar4 = (rt_uint32_t)parameter;
|
||||
stack_frame->exception_stack_frame.return_address = (rt_uint32_t)tentry;
|
||||
stack_frame->rpc = (rt_uint32_t)texit;
|
||||
|
||||
#ifdef __TMS320C28XX_FPU32__
|
||||
stack_frame->stf = 0x00000200;
|
||||
stack_frame->rb = 0;
|
||||
#endif
|
||||
|
||||
/* return task's current stack address */
|
||||
return stk + sizeof(struct stack_frame);
|
||||
}
|
||||
|
||||
/**
|
||||
* This function set the hook, which is invoked on fault exception handling.
|
||||
*
|
||||
* @param exception_handle the exception handling hook function.
|
||||
*/
|
||||
void rt_hw_exception_install(rt_err_t (*exception_handle)(void *context))
|
||||
{
|
||||
rt_exception_hook = exception_handle;
|
||||
}
|
||||
|
||||
|
||||
struct exception_info
|
||||
{
|
||||
rt_uint32_t exc_return;
|
||||
struct stack_frame stack_frame;
|
||||
};
|
||||
|
||||
#ifdef RT_USING_CPU_FFS
|
||||
/*
|
||||
* This function called rt_hw_calc_csb to finds the first bit set in value.
|
||||
* rt_hw_calc_csb is a native assembly program that use "CSB" instruction in C28x.
|
||||
* When you use this function, remember that "int" is only 16-bit in C28x's C compiler.
|
||||
* If value is a number bigger that 0xFFFF, trouble may be caused.
|
||||
* Maybe change "int __rt_ffs(int value)" to "rt_int32_t __rt_ffs(rt_int32_t value)" will be better.
|
||||
*/
|
||||
int __rt_ffs(int value)
|
||||
{
|
||||
return rt_hw_calc_csb(value);
|
||||
}
|
||||
#endif
|
||||
|
||||
void rt_interrupt_enter(void)
|
||||
{
|
||||
rt_base_t level;
|
||||
|
||||
__asm(" EINT");
|
||||
level = rt_hw_interrupt_disable();
|
||||
rt_interrupt_nest ++;
|
||||
RT_OBJECT_HOOK_CALL(rt_interrupt_enter_hook,());
|
||||
rt_hw_interrupt_enable(level);
|
||||
|
||||
LOG_D("irq has come..., irq current nest:%d",
|
||||
(rt_int32_t)rt_interrupt_nest);
|
||||
}
|
||||
|
||||
void rt_interrupt_leave(void)
|
||||
{
|
||||
LOG_D("irq is going to leave, irq current nest:%d",
|
||||
(rt_int32_t)rt_interrupt_nest);
|
||||
|
||||
rt_hw_interrupt_disable();
|
||||
RT_OBJECT_HOOK_CALL(rt_interrupt_leave_hook,());
|
||||
rt_interrupt_nest --;
|
||||
if(rt_thread_switch_interrupt_flag && !rt_interrupt_nest)
|
||||
{
|
||||
__asm(" OR IFR, #0x8000"); /* trigger rtos int */
|
||||
}
|
||||
/* rt_hw_interrupt_enable auto done by hardware on IRET */
|
||||
}
|
||||
Reference in New Issue
Block a user