ARMv7-A: A little more logic and a few more fixes for Cortex-A kernel build

This commit is contained in:
Gregory Nutt 2014-08-31 07:15:46 -06:00
parent fc7bd31e07
commit 67a54fd52e
5 changed files with 157 additions and 28 deletions

View File

@ -82,6 +82,10 @@ CMN_CSRCS += arm_allocpage.c arm_checkmapping.c arm_pginitialize.c
CMN_CSRCS += arm_va2pte.c CMN_CSRCS += arm_va2pte.c
endif endif
ifeq ($(CONFIG_BUILD_KERNEL),y)
CMN_CSRCS += up_task_start.c up_pthread_start.c up_stackframe.c
endif
ifeq ($(CONFIG_ARCH_ADDRENV),y) ifeq ($(CONFIG_ARCH_ADDRENV),y)
CMN_CSRCS += arm_addrenv.c CMN_CSRCS += arm_addrenv.c
endif endif

View File

@ -48,8 +48,8 @@
#include <arch/irq.h> #include <arch/irq.h>
#include <nuttx/sched.h> #include <nuttx/sched.h>
#include "arm.h"
#include "svcall.h" #include "svcall.h"
#include "up_internal.h"
/**************************************************************************** /****************************************************************************
* Pre-processor Definitions * Pre-processor Definitions
@ -242,21 +242,21 @@ uint32_t *arm_syscall(uint32_t *regs)
#ifdef CONFIG_BUILD_KERNEL #ifdef CONFIG_BUILD_KERNEL
case SYS_task_start: case SYS_task_start:
{ {
/* Set up to return to the user-space task start-up function in /* Set up to return to the user-space _start function in
* unprivileged mode. * unprivileged mode. We need:
*
* R0 = argc
* R1 = argv
* PC = taskentry
* CSPR = user mode
*/ */
regs[REG_PC] = (uint32_t)USERSPACE->task_startup; regs[REG_PC] = regs[REG_R1];
regval = regs[REG_CPSR] & ~PSR_MODE_MASK; regs[REG_R0] = regs[REG_R2];
regs[REG_CPSR] = regval | PSR_MODE_USR; regs[REG_R1] = regs[REG_R3];
/* Change the parameter ordering to match the expectation of struct cpsr = regs[REG_CPSR] & ~PSR_MODE_MASK;
* userpace_s task_startup: regs[REG_CPSR] = cpsr | PSR_MODE_USR;
*/
regs[REG_R0] = regs[REG_R1]; /* Task entry */
regs[REG_R1] = regs[REG_R2]; /* argc */
regs[REG_R2] = regs[REG_R3]; /* argv */
} }
break; break;
#endif #endif
@ -276,19 +276,19 @@ uint32_t *arm_syscall(uint32_t *regs)
case SYS_pthread_start: case SYS_pthread_start:
{ {
/* Set up to return to the user-space pthread start-up function in /* Set up to return to the user-space pthread start-up function in
* unprivileged mode. * unprivileged mode. We need:
*
* R0 = arg
* PC = entrypt
* CSPR = user mode
*/ */
regs[REG_PC] = (uint32_t)USERSPACE->pthread_startup;
regval = regs[REG_CPSR] & ~PSR_MODE_MASK;
regs[REG_CPSR] = regval | PSR_MODE_USR;
/* Change the parameter ordering to match the expectation of struct regs[REG_PC] = regs[REG_R1];
* userpace_s pthread_startup: regs[REG_R0] = regs[REG_R2];
*/
regs[REG_R0] = regs[REG_R1]; /* pthread entry */ cpsr = regs[REG_CPSR] & ~PSR_MODE_MASK;
regs[REG_R1] = regs[REG_R2]; /* arg */ regs[REG_CPSR] = cpsr | PSR_MODE_USR;
} }
break; break;
#endif #endif
@ -310,6 +310,8 @@ uint32_t *arm_syscall(uint32_t *regs)
#if defined(CONFIG_BUILD_KERNEL) && !defined(CONFIG_DISABLE_SIGNALS) #if defined(CONFIG_BUILD_KERNEL) && !defined(CONFIG_DISABLE_SIGNALS)
case SYS_signal_handler: case SYS_signal_handler:
{ {
#warning Missing logic
#if 0
struct tcb_s *rtcb = sched_self(); struct tcb_s *rtcb = sched_self();
/* Remember the caller's return address */ /* Remember the caller's return address */
@ -322,8 +324,8 @@ uint32_t *arm_syscall(uint32_t *regs)
*/ */
regs[REG_PC] = (uint32_t)USERSPACE->signal_handler; regs[REG_PC] = (uint32_t)USERSPACE->signal_handler;
regval = regs[REG_CPSR] & ~PSR_MODE_MASK; cpsr = regs[REG_CPSR] & ~PSR_MODE_MASK;
regs[REG_CPSR] = regval | PSR_MODE_USR; regs[REG_CPSR] = cpsr | PSR_MODE_USR;
/* Change the parameter ordering to match the expectation of struct /* Change the parameter ordering to match the expectation of struct
* userpace_s signal_handler. * userpace_s signal_handler.
@ -338,6 +340,7 @@ uint32_t *arm_syscall(uint32_t *regs)
*/ */
regs[REG_R3] = *(uint32_t*)(regs[REG_SP+4]); regs[REG_R3] = *(uint32_t*)(regs[REG_SP+4]);
#endif
} }
break; break;
#endif #endif
@ -354,6 +357,8 @@ uint32_t *arm_syscall(uint32_t *regs)
#if defined(CONFIG_BUILD_KERNEL) && !defined(CONFIG_DISABLE_SIGNALS) #if defined(CONFIG_BUILD_KERNEL) && !defined(CONFIG_DISABLE_SIGNALS)
case SYS_signal_handler_return: case SYS_signal_handler_return:
{ {
#warning Missing logic
#if 0
struct tcb_s *rtcb = sched_self(); struct tcb_s *rtcb = sched_self();
/* Set up to return to the kernel-mode signal dispatching logic. */ /* Set up to return to the kernel-mode signal dispatching logic. */
@ -361,9 +366,10 @@ uint32_t *arm_syscall(uint32_t *regs)
DEBUGASSERT(rtcb->xcp.sigreturn != 0); DEBUGASSERT(rtcb->xcp.sigreturn != 0);
regs[REG_PC] = rtcb->xcp.sigreturn; regs[REG_PC] = rtcb->xcp.sigreturn;
regval = regs[REG_CPSR] & ~PSR_MODE_MASK; cpsr = regs[REG_CPSR] & ~PSR_MODE_MASK;
regs[REG_CPSR] = regval | PSR_MODE_SVC; regs[REG_CPSR] = cpsr | PSR_MODE_SVC;
rtcb->xcp.sigreturn = 0; rtcb->xcp.sigreturn = 0;
#endif
} }
break; break;
#endif #endif
@ -399,8 +405,8 @@ uint32_t *arm_syscall(uint32_t *regs)
regs[REG_PC] = (uint32_t)dispatch_syscall; regs[REG_PC] = (uint32_t)dispatch_syscall;
#ifdef CONFIG_BUILD_KERNEL #ifdef CONFIG_BUILD_KERNEL
regval = regs[REG_CPSR] & ~PSR_MODE_MASK; cpsr = regs[REG_CPSR] & ~PSR_MODE_MASK;
regs[REG_CPSR] = regval | PSR_MODE_SVC; regs[REG_CPSR] = cpsr | PSR_MODE_SVC;
#endif #endif
/* Offset R0 to account for the reserved values */ /* Offset R0 to account for the reserved values */

105
arch/arm/src/armv7-a/crt0.c Normal file
View File

@ -0,0 +1,105 @@
/****************************************************************************
* arch/arm/src/armv7-a/crt0.c
*
* Copyright (C) 2014 Gregory Nutt. All rights reserved.
* Author: Gregory Nutt <gnutt@nuttx.org>
*
* 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 NuttX 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.
*
****************************************************************************/
/****************************************************************************
* Included Files
****************************************************************************/
#include <nuttx/config.h>
#include <sys/types.h>
#ifdef CONFIG_BUILD_KERNEL
/****************************************************************************
* Pre-processor Definitions
****************************************************************************/
/****************************************************************************
* Private Data
****************************************************************************/
/****************************************************************************
* Public Function Prototypes
****************************************************************************/
extern main_t main;
/****************************************************************************
* Private Functions
****************************************************************************/
/****************************************************************************
* Public Functions
****************************************************************************/
/****************************************************************************
* Name: _start
*
* Description:
* This function is the low level entry point into the main thread of
* execution of a task. It receives initial control when the task is
* started and calls main entry point of the newly started task.
*
* Input Parameters:
* argc - The number of parameters being passed.
* argv - The parameters being passed. These lie in kernel-space memory
* and will have to be reallocated in user-space memory.
*
* Returned Value:
* This function should not return. It should call the user-mode start-up
* main() function. If that function returns, this function will call
* exit.
*
****************************************************************************/
void _start(int argc, FAR char *argv[])
{
int ret;
/* Call C++ constructors */
/* Setup so that C++ destructors called on task exit */
/* REVISIT: Missing logic */
/* Call the main() entry point passing argc and argv. */
ret = main(argc, argc);
/* Call exit() if/when the main() returns */
exit(ret);
}
#endif /* CONFIG_BUILD_KERNEL */

View File

@ -84,6 +84,10 @@ CMN_CSRCS += arm_allocpage.c arm_checkmapping.c arm_pginitialize.c
CMN_CSRCS += arm_va2pte.c CMN_CSRCS += arm_va2pte.c
endif endif
ifeq ($(CONFIG_BUILD_KERNEL),y)
CMN_CSRCS += up_task_start.c up_pthread_start.c up_stackframe.c
endif
ifeq ($(CONFIG_ARCH_ADDRENV),y) ifeq ($(CONFIG_ARCH_ADDRENV),y)
CMN_CSRCS += arm_addrenv.c CMN_CSRCS += arm_addrenv.c
endif endif

View File

@ -261,14 +261,20 @@ config USER_INITPATH
config INIT_SYMTAB config INIT_SYMTAB
string "Symbol table" string "Symbol table"
default "NULL" default "NULL"
depends on !BUILD_PROTECTED && !BUILD_KERNEL
---help--- ---help---
The name of othe global array that holds the exported symbol table. The name of othe global array that holds the exported symbol table.
The special string "NULL" may be provided if there is no symbol The special string "NULL" may be provided if there is no symbol
table. Quotation marks will be stripped when config.h is generated. table. Quotation marks will be stripped when config.h is generated.
NOTE: This setting cannot be used in protected or kernel builds.
Any kernel mode symbols tables would not be usable for resolving
symbols in user mode executables.
config INIT_NEXPORTS config INIT_NEXPORTS
string "Symbol table size" string "Symbol table size"
default "0" default "0"
depends on !BUILD_PROTECTED && !BUILD_KERNEL
---help--- ---help---
The size of the symbol table. NOTE that is is logically a numeric The size of the symbol table. NOTE that is is logically a numeric
value but is represent by a string. That allows you to put value but is represent by a string. That allows you to put
@ -276,6 +282,10 @@ config INIT_NEXPORTS
symbol table size. Quotation marks will be stripped when config.h symbol table size. Quotation marks will be stripped when config.h
is generated. is generated.
NOTE: This setting cannot be used in protected or kernel builds.
Any kernel mode symbols tables would not be usable for resolving
symbols in user mode executables.
endif # INIT_FILEPATH endif # INIT_FILEPATH
config RR_INTERVAL config RR_INTERVAL