os/kernelhwsrv/kernel/eka/include/nkernsmp/arm/nk_plat.h
author sl
Tue, 10 Jun 2014 14:32:02 +0200
changeset 1 260cb5ec6c19
permissions -rw-r--r--
Update contrib.
sl@0
     1
// Copyright (c) 2007-2009 Nokia Corporation and/or its subsidiary(-ies).
sl@0
     2
// All rights reserved.
sl@0
     3
// This component and the accompanying materials are made available
sl@0
     4
// under the terms of the License "Eclipse Public License v1.0"
sl@0
     5
// which accompanies this distribution, and is available
sl@0
     6
// at the URL "http://www.eclipse.org/legal/epl-v10.html".
sl@0
     7
//
sl@0
     8
// Initial Contributors:
sl@0
     9
// Nokia Corporation - initial contribution.
sl@0
    10
//
sl@0
    11
// Contributors:
sl@0
    12
//
sl@0
    13
// Description:
sl@0
    14
// e32\include\nkernsmp\arm\nk_plat.h
sl@0
    15
// 
sl@0
    16
// WARNING: This file contains some APIs which are internal and are subject
sl@0
    17
//          to change without notice. Such APIs should therefore not be used
sl@0
    18
//          outside the Kernel and Hardware Services package.
sl@0
    19
//
sl@0
    20
sl@0
    21
/**
sl@0
    22
 @file
sl@0
    23
 @internalComponent
sl@0
    24
*/
sl@0
    25
sl@0
    26
#ifndef __NK_ARM_H__
sl@0
    27
#define __NK_ARM_H__
sl@0
    28
#include <nk_cpu.h>
sl@0
    29
sl@0
    30
// These macros are intended for Symbian use only.
sl@0
    31
// It may not be possible to build the kernel if any of these macros are undefined
sl@0
    32
//#define __SCHEDULER_MACHINE_CODED__
sl@0
    33
//#define __DFC_MACHINE_CODED__
sl@0
    34
//#define __MSTIM_MACHINE_CODED__
sl@0
    35
#define __PRI_LIST_MACHINE_CODED__
sl@0
    36
#define __FAST_SEM_MACHINE_CODED__
sl@0
    37
#define __FAST_MUTEX_MACHINE_CODED__
sl@0
    38
#define __NTHREAD_WAITSTATE_MACHINE_CODED__
sl@0
    39
sl@0
    40
// TSubScheduler member data
sl@0
    41
#define	i_ScuAddr			iExtras[4]		// Address of SCU (also in TScheduler)
sl@0
    42
#define	i_GicDistAddr		iExtras[5]		// Address of GIC Distributor (also in TScheduler)
sl@0
    43
#define	i_GicCpuIfcAddr		iExtras[6]		// Address of GIC CPU Interface (also in TScheduler)
sl@0
    44
#define	i_LocalTimerAddr	iExtras[7]		// Address of local timer registers (also in TScheduler)
sl@0
    45
#define	i_IrqCount			iExtras[8]		// count of interrupts handled
sl@0
    46
#define	i_IrqNestCount		iExtras[9]		// IRQ nest count for this CPU (starts at -1)
sl@0
    47
#define	i_ExcInfo			iExtras[10]		// pointer to exception info for crash debugger
sl@0
    48
#define	i_CrashState		iExtras[11]		// 0=normal, 1=this CPU faulted, 2=this CPU has received an NMI and halted
sl@0
    49
#define	i_AbtStackTop		iExtras[12]		// Top of ABT stack for this CPU, also used to point to SFullArmRegSet
sl@0
    50
#define	i_UndStackTop		iExtras[13]		// Top of UND stack for this CPU
sl@0
    51
#define	i_FiqStackTop		iExtras[14]		// Top of FIQ stack for this CPU
sl@0
    52
#define	i_IrqStackTop		iExtras[15]		// Top of IRQ stack for this CPU
sl@0
    53
#define	i_TimerMultF		iExtras[16]		// Timer frequency / Max Timer frequency * 2^32
sl@0
    54
#define	i_TimerMultI		iExtras[17]		// Max Timer frequency / Timer frequency * 2^24
sl@0
    55
#define	i_CpuMult			iExtras[18]		// CPU frequency / Max CPU frequency * 2^32
sl@0
    56
#define	i_LastTimerSet		iExtras[20]		// Value last written to local timer counter
sl@0
    57
#define	i_TimestampError	iExtras[21]		// Current error in the timestamp
sl@0
    58
#define	i_MaxCorrection		iExtras[22]		// Maximum correction to timestamp in one go
sl@0
    59
#define	i_TimerGap			iExtras[23]		// Timestamp ticks taken to read and write local timer counter
sl@0
    60
sl@0
    61
#define	i_Regs				iExtras[12]		// Alias for i_AbtStackTop
sl@0
    62
sl@0
    63
// TScheduler member data
sl@0
    64
#define	i_TimerMax			iExtras[16]		// Maximum per-CPU timer frequency (after prescaling)
sl@0
    65
sl@0
    66
sl@0
    67
#define	RESCHED_IPI_VECTOR			0x00
sl@0
    68
#define	GENERIC_IPI_VECTOR			0x01
sl@0
    69
#define	TRANSFERRED_IRQ_VECTOR		0x02
sl@0
    70
#define	CRASH_IPI_VECTOR			0x03	// would really like this to be a FIQ
sl@0
    71
#define	BOOT_IPI_VECTOR				0x04	// used during boot to handshake with APs
sl@0
    72
#define RESERVED_IPI_VECTOR_1		0x05	// reserved for future kernel functionality
sl@0
    73
#define RESERVED_IPI_VECTOR_2		0x06	// reserved for future kernel functionality
sl@0
    74
#define RESERVED_IPI_VECTOR_3		0x07	// reserved for future kernel functionality
sl@0
    75
sl@0
    76
#if defined(__CPU_ARM11MP__)
sl@0
    77
#define	TIMESLICE_VECTOR			0x1D	// vector 29 is per-CPU timer interrupt
sl@0
    78
											// vector 30 is per-CPU Watchdog timer when not in watchdog mode
sl@0
    79
											// vector 31 is external nIRQ local interrupt pin
sl@0
    80
#elif defined(__CPU_CORTEX_A9__)
sl@0
    81
#define	TIMESLICE_VECTOR			0x1D	// vector 29 is per-CPU timer interrupt
sl@0
    82
											// vector 30 is per-CPU Watchdog timer when not in watchdog mode
sl@0
    83
#else
sl@0
    84
#error	TIMESLICE_VECTOR not defined
sl@0
    85
#endif
sl@0
    86
sl@0
    87
sl@0
    88
//extern "C" TSubScheduler* SubSchedulerLookupTable[256];		// look up subscheduler from APIC ID
sl@0
    89
sl@0
    90
const TUint32 KNThreadContextFlagThumbBit0=1;
sl@0
    91
sl@0
    92
/** Registers saved by the scheduler
sl@0
    93
sl@0
    94
Let's just have the same stack layout for all CPUs shall we?
sl@0
    95
TEEHBR, FpExc may not be used but leave space on the stack for them.
sl@0
    96
sl@0
    97
@internalComponent
sl@0
    98
*/
sl@0
    99
struct SThreadReschedStack
sl@0
   100
	{
sl@0
   101
	TUint32	iFpExc;			// VFP enable
sl@0
   102
	TUint32	iCar;			// coprocessor access register
sl@0
   103
	TUint32	iTEEHBR;		// Thumb2-EE Handler Base
sl@0
   104
	TUint32	iRWROTID;		// User RO Thread ID
sl@0
   105
	TUint32	iRWRWTID;		// User RW Thread ID
sl@0
   106
	TUint32	iDacr;			// domain access control
sl@0
   107
	TUint32	iSpare;
sl@0
   108
	TUint32	iSpsrSvc;
sl@0
   109
	TUint32 iSPRschdFlg;	// Stack pointer plus flag indicating reschedule occurred
sl@0
   110
	TUint32	iR15;			// return address from Reschedule()
sl@0
   111
	};
sl@0
   112
sl@0
   113
/** Registers saved on any exception, interrupt or system call
sl@0
   114
sl@0
   115
@internalComponent
sl@0
   116
*/
sl@0
   117
struct SThreadExcStack
sl@0
   118
	{
sl@0
   119
	enum	TType
sl@0
   120
		{
sl@0
   121
		EPrefetch	=0,		// prefetch abort
sl@0
   122
		EData		=1,		// data abort
sl@0
   123
		EUndef		=2,		// undefined instruction
sl@0
   124
		EIrq		=3,		// IRQ interrupt
sl@0
   125
		EFiq		=4,		// FIQ interrupt
sl@0
   126
		ESvc		=5,		// SWI
sl@0
   127
		EInit		=6,		// Thread has never run
sl@0
   128
		EStub		=7,		// Stub indicating parameter block still on stack
sl@0
   129
		};
sl@0
   130
sl@0
   131
	TUint32	iR0;
sl@0
   132
	TUint32	iR1;
sl@0
   133
	TUint32	iR2;
sl@0
   134
	TUint32	iR3;
sl@0
   135
	TUint32	iR4;
sl@0
   136
	TUint32	iR5;
sl@0
   137
	TUint32	iR6;
sl@0
   138
	TUint32	iR7;
sl@0
   139
	TUint32	iR8;
sl@0
   140
	TUint32	iR9;
sl@0
   141
	TUint32	iR10;
sl@0
   142
	TUint32	iR11;
sl@0
   143
	TUint32	iR12;
sl@0
   144
	TUint32	iR13usr;		// always user mode R13
sl@0
   145
	TUint32	iR14usr;		// always user mode R14
sl@0
   146
	TUint32	iExcCode;
sl@0
   147
	TUint32	iR15;			// return address
sl@0
   148
	TUint32	iCPSR;			// return CPSR
sl@0
   149
	};
sl@0
   150
sl@0
   151
/**
sl@0
   152
@internalComponent
sl@0
   153
*/
sl@0
   154
struct SThreadStackStub
sl@0
   155
	{
sl@0
   156
	TLinAddr iPBlock;		// pointer to parameter block
sl@0
   157
	TUint32	iExcCode;		// always EStub
sl@0
   158
	TUint32	iR15;			// unused
sl@0
   159
	TUint32	iCPSR;			// unused
sl@0
   160
	};
sl@0
   161
sl@0
   162
/**
sl@0
   163
@internalComponent
sl@0
   164
*/
sl@0
   165
struct SThreadInitStack
sl@0
   166
	{
sl@0
   167
	SThreadReschedStack		iR;
sl@0
   168
	SThreadExcStack			iX;
sl@0
   169
	};
sl@0
   170
sl@0
   171
sl@0
   172
/**
sl@0
   173
@internalComponent
sl@0
   174
*/
sl@0
   175
struct SThreadIrqStack
sl@0
   176
	{
sl@0
   177
	SThreadReschedStack		iR;
sl@0
   178
	TUint32					iUMGSave;	// User memory guard state (if active)
sl@0
   179
	TUint32					iR14svc;
sl@0
   180
	SThreadExcStack			iX;
sl@0
   181
	};
sl@0
   182
sl@0
   183
sl@0
   184
class TArmContextElement;
sl@0
   185
class TArmRegSet;
sl@0
   186
sl@0
   187
/** ARM-specific part of the nano-thread abstraction.
sl@0
   188
	@internalComponent
sl@0
   189
 */
sl@0
   190
class NThread : public NThreadBase
sl@0
   191
	{
sl@0
   192
public:
sl@0
   193
	TInt Create(SNThreadCreateInfo& aInfo, TBool aInitial);
sl@0
   194
	inline void Stillborn()
sl@0
   195
		{}
sl@0
   196
sl@0
   197
	/** Value indicating what event caused thread to enter privileged mode.
sl@0
   198
		@publishedPartner
sl@0
   199
		@released
sl@0
   200
	 */
sl@0
   201
	enum TUserContextType
sl@0
   202
		{
sl@0
   203
		EContextNone=0,             /**< Thread has no user context */
sl@0
   204
		EContextException=1,		/**< Hardware exception while in user mode */
sl@0
   205
		EContextUndefined,			
sl@0
   206
		EContextUserInterrupt,		/**< Preempted by interrupt taken in user mode */
sl@0
   207
		EContextUserInterruptDied,  /**< Killed while preempted by interrupt taken in user mode */	// NOT USED
sl@0
   208
		EContextSvsrInterrupt1,     /**< Preempted by interrupt taken in executive call handler */
sl@0
   209
		EContextSvsrInterrupt1Died, /**< Killed while preempted by interrupt taken in executive call handler */	// NOT USED
sl@0
   210
		EContextSvsrInterrupt2,     /**< Preempted by interrupt taken in executive call handler */	// NOT USED
sl@0
   211
		EContextSvsrInterrupt2Died, /**< Killed while preempted by interrupt taken in executive call handler */	// NOT USED
sl@0
   212
		EContextWFAR,               /**< Blocked on User::WaitForAnyRequest() */
sl@0
   213
		EContextWFARDied,           /**< Killed while blocked on User::WaitForAnyRequest() */		// NOT USED
sl@0
   214
		EContextExec,				/**< Slow executive call */
sl@0
   215
		EContextKernel,				/**< Kernel side context (for kernel threads) */
sl@0
   216
		EContextKernel1,			/**< Kernel side context (for kernel threads) (NKern::Unlock, NKern::PreemptionPoint) */
sl@0
   217
		EContextKernel2,			/**< Kernel side context (for kernel threads) (NKern::FSWait, NKern::WaitForAnyRequest) */
sl@0
   218
		EContextKernel3,			/**< Kernel side context (for kernel threads) (Interrupt) */
sl@0
   219
		EContextKernel4,			/**< Kernel side context (for kernel threads) (Exec::WaitForAnyRequest) */
sl@0
   220
		};
sl@0
   221
sl@0
   222
	IMPORT_C static const TArmContextElement* const* UserContextTables();
sl@0
   223
	IMPORT_C TUserContextType UserContextType();
sl@0
   224
	void GetUserContext(TArmRegSet& aContext, TUint32& aAvailRegistersMask);
sl@0
   225
	void SetUserContext(const TArmRegSet& aContext, TUint32& aRegMask);
sl@0
   226
	void GetSystemContext(TArmRegSet& aContext, TUint32& aAvailRegistersMask);
sl@0
   227
sl@0
   228
	TUint32 Dacr();
sl@0
   229
	void SetDacr(TUint32 aDacr);
sl@0
   230
	TUint32 ModifyDacr(TUint32 aClearMask, TUint32 aSetMask);
sl@0
   231
sl@0
   232
	void SetCar(TUint32 aDacr);
sl@0
   233
	IMPORT_C TUint32 Car();
sl@0
   234
	IMPORT_C TUint32 ModifyCar(TUint32 aClearMask, TUint32 aSetMask);
sl@0
   235
sl@0
   236
#ifdef __CPU_HAS_VFP
sl@0
   237
	void SetFpExc(TUint32 aDacr);
sl@0
   238
#endif
sl@0
   239
	IMPORT_C TUint32 FpExc();
sl@0
   240
	IMPORT_C TUint32 ModifyFpExc(TUint32 aClearMask, TUint32 aSetMask);
sl@0
   241
sl@0
   242
	void CompleteContextSave();
sl@0
   243
	};
sl@0
   244
sl@0
   245
sl@0
   246
struct SArmInterruptInfo
sl@0
   247
	{
sl@0
   248
	TLinAddr iIrqHandler;
sl@0
   249
	TLinAddr iFiqHandler;
sl@0
   250
	SCpuIdleHandler iCpuIdleHandler;
sl@0
   251
	};
sl@0
   252
sl@0
   253
extern "C" SArmInterruptInfo ArmInterruptInfo;
sl@0
   254
sl@0
   255
#if defined(__ARMCC__)
sl@0
   256
#ifndef __CIA__
sl@0
   257
inline void mb()
sl@0
   258
	{
sl@0
   259
	TUint32 reg = 0;
sl@0
   260
	asm("mcr p15, 0, reg, c7, c10, 5 ");
sl@0
   261
	}
sl@0
   262
sl@0
   263
inline void arm_dsb()
sl@0
   264
	{
sl@0
   265
	TUint32 reg = 0;
sl@0
   266
	asm("mcr p15, 0, reg, c7, c10, 4 ");
sl@0
   267
	}
sl@0
   268
sl@0
   269
inline void arm_isb()
sl@0
   270
	{
sl@0
   271
	TUint32 reg = 0;
sl@0
   272
	asm("mcr p15, 0, reg, c7, c5, 4 ");
sl@0
   273
	}
sl@0
   274
#endif
sl@0
   275
#elif defined(__GNUC__) || defined(__GCC32__)
sl@0
   276
#define	mb()	\
sl@0
   277
	do	{	\
sl@0
   278
		TUint32 reg = 0;	\
sl@0
   279
		__asm__ __volatile__("mcr p15, 0, %0, c7, c10, 5" : : "r"(reg) : "memory");	\
sl@0
   280
		} while(0)
sl@0
   281
sl@0
   282
#define	arm_dsb()	\
sl@0
   283
	do	{	\
sl@0
   284
		TUint32 reg = 0;	\
sl@0
   285
		__asm__ __volatile__("mcr p15, 0, %0, c7, c10, 4" : : "r"(reg) : "memory");	\
sl@0
   286
		} while(0)
sl@0
   287
sl@0
   288
#define	arm_isb()	\
sl@0
   289
	do	{	\
sl@0
   290
		TUint32 reg = 0;	\
sl@0
   291
		__asm__ __volatile__("mcr p15, 0, %0, c7, c5, 4" : : "r"(reg) : "memory");	\
sl@0
   292
		} while(0)
sl@0
   293
#else
sl@0
   294
#error Unknown ARM compiler
sl@0
   295
#endif
sl@0
   296
sl@0
   297
#define	smp_mb()	mb()
sl@0
   298
#define	wmb()		mb()
sl@0
   299
#define	smp_wmb()	mb()
sl@0
   300
sl@0
   301
#ifdef	__IN_KERNEL__
sl@0
   302
struct ArmScu;
sl@0
   303
struct GicDistributor;
sl@0
   304
struct GicCpuIfc;
sl@0
   305
struct ArmLocalTimer;
sl@0
   306
#define	SCU			(*(ArmScu*)TheScheduler.i_ScuAddr)
sl@0
   307
#define	GIC_DIST	(*(GicDistributor*)TheScheduler.i_GicDistAddr)
sl@0
   308
#define	GIC_CPU_IFC	(*(GicCpuIfc*)TheScheduler.i_GicCpuIfcAddr)
sl@0
   309
#define	LOCAL_TIMER	(*(ArmLocalTimer*)TheScheduler.i_LocalTimerAddr)
sl@0
   310
#endif
sl@0
   311
sl@0
   312
sl@0
   313
// End of file
sl@0
   314
#endif