navienginebsp/naviengine_assp/interrupts.cpp
changeset 0 5de814552237
equal deleted inserted replaced
-1:000000000000 0:5de814552237
       
     1 /*
       
     2 * Copyright (c) 2008-2010 Nokia Corporation and/or its subsidiary(-ies).
       
     3 * All rights reserved.
       
     4 * This component and the accompanying materials are made available
       
     5 * under the terms of "Eclipse Public License v1.0"
       
     6 * which accompanies this distribution, and is available
       
     7 * at the URL "http://www.eclipse.org/legal/epl-v10.html".
       
     8 *
       
     9 * Initial Contributors:
       
    10 * Nokia Corporation - initial contribution.
       
    11 *
       
    12 * Contributors:
       
    13 *
       
    14 * Description:  
       
    15 * naviengine_assp\interrupts.cpp
       
    16 * NE1_TBVariant ASSP interrupt control and dispatch
       
    17 *
       
    18 */
       
    19 
       
    20 
       
    21 
       
    22 #include <naviengine_priv.h>
       
    23 
       
    24 const TUint8 IntIsEdge[96] =
       
    25 	{
       
    26 	0, 1, 0, 0,		0, 0, 0, 0,		0, 0, 0, 0,		0, 0, 0, 1,
       
    27 	0, 0, 0, 0,		0, 0, 0, 0,		0, 0, 0, 0,		0, 0, 0, 0,
       
    28 	0, 0, 0, 0,		0, 0, 0, 1,		1, 0, 0, 0,		1, 1, 1, 1,
       
    29 	1, 1, 1, 1,		1, 1, 0, 0,		0, 0, 0, 0,		0, 0, 0, 0,
       
    30 	0, 1, 1, 0,		0, 0, 0, 0,		0, 0, 0, 0,		0, 0, 0, 0,
       
    31 	0, 0, 0, 0,		0, 0, 0, 0,		0, 0, 0, 0,		0, 0, 0, 0
       
    32 	};
       
    33 
       
    34 #ifdef __SMP__
       
    35 #include <arm_gic.h>
       
    36 
       
    37 #ifndef __STANDALONE_NANOKERNEL__
       
    38 #undef	DEBUGPRINT
       
    39 #define	DEBUGPRINT	Kern::Printf
       
    40 #endif
       
    41 
       
    42 TUint32 NaviEngineInterrupt::IrqDispatch(TUint32 aVector)
       
    43 	{
       
    44 	if (aVector<32 || aVector>127)
       
    45 		{
       
    46 		GicCpuIfc& g = *(GicCpuIfc*)KHwBaseIntIf;
       
    47 		g.iEoi = aVector;
       
    48 		*(TInt*)0xdeaddead = 0;
       
    49 		return aVector;
       
    50 		}
       
    51 	NKern::Interrupt(aVector - 32);
       
    52 	return aVector;
       
    53 	}
       
    54 
       
    55 void NaviEngineInterrupt::DisableAndClearAll()
       
    56 	{
       
    57 	// NKern does all the GIC stuff
       
    58 	}
       
    59 
       
    60 void NaviEngineInterrupt::Init1()
       
    61 	{
       
    62 	__KTRACE_OPT(KBOOT, DEBUGPRINT(">Interrupt_Init1()"));
       
    63 	DisableAndClearAll();
       
    64 	Arm::SetIrqHandler((TLinAddr)&IrqDispatch);
       
    65 	Arm::SetFiqHandler((TLinAddr)&FiqDispatch);
       
    66 	TInt i;
       
    67 	for (i=32; i<128; ++i)
       
    68 		{
       
    69 		TBool edge = IntIsEdge[i-32];
       
    70 		TUint32 flags = 0;
       
    71 		if (i>=36 && i<42)
       
    72 			flags |= NKern::EIrqInit_Count;	// timers count all interrupts
       
    73 		if (edge)
       
    74 			flags |= NKern::EIrqInit_RisingEdge;
       
    75 		else
       
    76 			flags |= NKern::EIrqInit_LevelHigh;
       
    77 		TInt r = NKern::InterruptInit(i-32, flags, i, i);
       
    78 		__KTRACE_OPT(KBOOT, DEBUGPRINT("InterruptInit %d(%02x) -> %d", i-32, i, r));
       
    79 		__NK_ASSERT_ALWAYS(r==KErrNone);
       
    80 		}
       
    81 
       
    82 	__KTRACE_OPT(KBOOT, DEBUGPRINT("<Interrupt_Init1()"));
       
    83 	}
       
    84 
       
    85 //extern "C" void Interrupt_Init2AP()
       
    86 //	{
       
    87 //	}
       
    88 
       
    89 void NaviEngineInterrupt::Init3()
       
    90 	{
       
    91 	//
       
    92 	// Any further initialisation of the Hardware Interrupt Controller
       
    93 	// Note This is not called at the moment. Should be deleted.
       
    94 	}
       
    95 
       
    96 EXPORT_C TInt Interrupt::Bind(TInt aId, TIsr aIsr, TAny* aPtr)
       
    97 	{
       
    98 	__KTRACE_OPT(KEXTENSION,Kern::Printf("Interrupt::Bind id=%d func=%08x ptr=%08x", aId, aIsr, aPtr));
       
    99 	TUint id = (TUint)aId;
       
   100 	if (id<32 || id>=(TUint)KNumNaviEngineInts)
       
   101 		return KErrArgument;
       
   102 	TUint32 flags = 0;
       
   103 	if (id>=36 && id<=41)
       
   104 		{
       
   105 		flags |= NKern::EIrqBind_Count;
       
   106 		}
       
   107 	return NKern::InterruptBind(id-32, aIsr, aPtr, flags, 0);
       
   108 	}
       
   109 
       
   110 EXPORT_C TInt Interrupt::Unbind(TInt aId)
       
   111 	{
       
   112 	__KTRACE_OPT(KEXTENSION,Kern::Printf("Interrupt::Unbind id=%08x",aId));
       
   113 	if (aId >= 0x10000)
       
   114 		return NKern::InterruptUnbind(aId);			// aId is a handle not a specific ID
       
   115 	if (TUint(aId) < TUint(KNumNaviEngineMaxInts))
       
   116 		return NKern::InterruptUnbind(aId - 32);	// Vector number to nanokernel interrupt number
       
   117 	return KErrArgument;
       
   118 	}
       
   119 
       
   120 EXPORT_C TInt Interrupt::Enable(TInt aId)
       
   121 	{
       
   122 	__KTRACE_OPT(KEXTENSION,Kern::Printf("Interrupt::Enable id=%08x",aId));
       
   123 	if (aId >= 0x10000)
       
   124 		return NKern::InterruptEnable(aId);			// aId is a handle not a specific ID
       
   125 	if (TUint(aId) < TUint(KNumNaviEngineMaxInts))
       
   126 		return NKern::InterruptEnable(aId - 32);	// Vector number to nanokernel interrupt number
       
   127 	return KErrArgument;
       
   128 	}
       
   129 
       
   130 EXPORT_C TInt Interrupt::Disable(TInt aId)
       
   131 	{
       
   132 	__KTRACE_OPT(KEXTENSION,Kern::Printf("Interrupt::Disable id=%08x",aId));
       
   133 	if (aId >= 0x10000)
       
   134 		return NKern::InterruptDisable(aId);		// aId is a handle not a specific ID
       
   135 	if (TUint(aId) < TUint(KNumNaviEngineMaxInts))
       
   136 		return NKern::InterruptDisable(aId - 32);	// Vector number to nanokernel interrupt number
       
   137 	return KErrArgument;
       
   138 	}
       
   139 
       
   140 EXPORT_C TInt Interrupt::Clear(TInt aId)
       
   141 	{
       
   142 	__KTRACE_OPT(KEXTENSION,Kern::Printf("Interrupt::Clear id=%08x",aId));
       
   143 	if (aId >= 0x10000)
       
   144 		return NKern::InterruptClear(aId);			// aId is a handle not a specific ID
       
   145 	if (TUint(aId) < TUint(KNumNaviEngineMaxInts))
       
   146 		return NKern::InterruptClear(aId - 32);		// Vector number to nanokernel interrupt number
       
   147 	return KErrArgument;
       
   148 	}
       
   149 
       
   150 EXPORT_C TInt Interrupt::SetPriority(TInt /*anId*/, TInt /*aPriority*/)
       
   151 	{
       
   152 	return KErrNotSupported;
       
   153 	}
       
   154 
       
   155 #else
       
   156 SInterruptHandler NaviEngineInterrupt::Handlers[KNumNaviEngineInts];
       
   157 
       
   158 void NaviEngineInterrupt::DisableAndClearAll()
       
   159 	{
       
   160     //Disable all interrupts
       
   161     for(TUint i=0; i<=((KNumNaviEngineInts-1)/32); i++)
       
   162     	AsspRegister::Write32(KHoGidIntDisableBase+i*4, 0xffffffff);
       
   163     //Clear all interrupts
       
   164     for(TUint i=0; i<=((KNumNaviEngineInts-1)/32); i++)
       
   165     	AsspRegister::Write32(KHoGidPendClearBase+i*4, 0xffffffff);
       
   166 
       
   167 	}
       
   168 
       
   169 void NaviEngineInterrupt::Init1()
       
   170 	{
       
   171 	//
       
   172 	// need to hook the ARM IRQ and FIQ handlers as early as possible and disable and clear all interrupt sources
       
   173 	//
       
   174 	__KTRACE_OPT(KBOOT,Kern::Printf("NaviEngineInterrupt::Init1()"));
       
   175 	TInt i;
       
   176 	for (i=0; i<KNumNaviEngineInts; i++)
       
   177 		{
       
   178 		Handlers[i].iPtr=(TAny*)i;
       
   179 		Handlers[i].iIsr=Spurious;
       
   180 		}
       
   181 	DisableAndClearAll();
       
   182 	Arm::SetIrqHandler((TLinAddr)NaviEngineInterrupt::IrqDispatch);
       
   183 	Arm::SetFiqHandler((TLinAddr)NaviEngineInterrupt::FiqDispatch);
       
   184 
       
   185     // All interrupts are N-N (LSB = 1)
       
   186     // Both edge (MSB = 1) and level triggered (MSB = 0)
       
   187 	// are present
       
   188 	for (TInt i=0; i<KNumNaviEngineInts; ++i)
       
   189 		{
       
   190 		TUint8 isEdge = 0x0;
       
   191 		if(i>=32 && i<128)
       
   192 			{
       
   193 			isEdge = IntIsEdge[i-32];
       
   194 			}
       
   195 		const TUint8 isNN = 0x1;
       
   196 		const TUint8 irqConfig = (isEdge << 1) | isNN;
       
   197 
       
   198 		const TInt byteOffset = i/4;
       
   199 		const TInt shiftForIrq = i%4;
       
   200 		TUint8 setMask = irqConfig << shiftForIrq;
       
   201 		AsspRegister::Modify8(KHoGidConfigBase + byteOffset, 0, setMask);
       
   202 		}
       
   203 
       
   204 	// Set interrupts starting from 32 to this CPU only
       
   205     for(TUint i=0; i<=((KNumNaviEngineInts-1)/4); i++)
       
   206     	AsspRegister::Write32(KHoGidTargetBase+i*4, 0x0f0f0f0f);
       
   207 
       
   208 	// Set priority on all interrupts
       
   209     for(TUint i=0; i<=((KNumNaviEngineInts-1)/4); i++)
       
   210     	AsspRegister::Write32(KHoGidPriorityBase+i*4, 0xA0A0A0A0);
       
   211 
       
   212     // Set binary point. No-preemption, all bits used
       
   213     AsspRegister::Write32(KHwCIIBinPoint, 7);
       
   214 
       
   215     // Set priority mask
       
   216     AsspRegister::Write32(KHwCIIPrioMask, 0xF0);
       
   217 
       
   218     // Enable Distributor and CPU Interface for IRQ
       
   219     AsspRegister::Write32(KHoGidControl, 1);
       
   220     AsspRegister::Write32(KHwCIIControl, 1);
       
   221 	}
       
   222 
       
   223 void NaviEngineInterrupt::Init3()
       
   224 	{
       
   225 	//
       
   226 	// Any further initialisation of the Hardware Interrupt Controller
       
   227 	// Note This is not called at the moment. Should be deleted.
       
   228 	}
       
   229 
       
   230 void NaviEngineInterrupt::Spurious(TAny* anId)
       
   231 	{
       
   232 	Kern::Fault("SpuriousInt", (TInt)anId);
       
   233 	}
       
   234 
       
   235 //
       
   236 // The APIs below assume ther is a second level Interrupt controller located at Variant level which handles
       
   237 // interrupts generated by hardware at that level.
       
   238 //
       
   239 
       
   240 EXPORT_C TInt Interrupt::Bind(TInt anId, TIsr anIsr, TAny* aPtr)
       
   241 	{
       
   242 	__KTRACE_OPT(KEXTENSION,Kern::Printf("Interrupt::Bind id=%d func=%08x ptr=%08x",anId,anIsr,aPtr));
       
   243 	TInt r = anId;
       
   244 	// if ID indicates a chained interrupt, call variant...
       
   245 //	if (anId<0 && ((((TUint)anId)>>16)&0x7fff)<(TUint)KNumNaviEngineInts)
       
   246 //		r=NaviEngineAssp::Variant->InterruptBind(anId,anIsr,aPtr);
       
   247 	__NK_ASSERT_ALWAYS(anId>=0);
       
   248 	if ((TUint)anId >= (TUint)KNumNaviEngineInts)
       
   249 		r=KErrArgument;
       
   250 	else
       
   251 		{
       
   252 		SInterruptHandler& h=NaviEngineInterrupt::Handlers[anId];
       
   253 		TInt irq=NKern::DisableAllInterrupts();
       
   254 		if (h.iIsr != NaviEngineInterrupt::Spurious)
       
   255 			r=KErrInUse;
       
   256 		else
       
   257 			{
       
   258 			h.iPtr=aPtr;
       
   259 			h.iIsr=anIsr;
       
   260 			}
       
   261 		NKern::RestoreInterrupts(irq);
       
   262 		}
       
   263 	return r;
       
   264 	}
       
   265 
       
   266 EXPORT_C TInt Interrupt::Unbind(TInt anId)
       
   267 	{
       
   268 	__KTRACE_OPT(KEXTENSION,Kern::Printf("Interrupt::Unbind id=%d",anId));
       
   269 	TInt r=KErrNone;
       
   270 	// if ID indicates a chained interrupt, call variant...
       
   271 	if (anId<0 && ((((TUint)anId)>>16)&0x7fff)<(TUint)KNumNaviEngineInts)
       
   272 		r=NaviEngineAssp::Variant->InterruptUnbind(anId);
       
   273 	else if ((TUint)anId >= (TUint)KNumNaviEngineInts)
       
   274 		r=KErrArgument;
       
   275 	else
       
   276 		{
       
   277 		SInterruptHandler& h=NaviEngineInterrupt::Handlers[anId];
       
   278 		TInt irq=NKern::DisableAllInterrupts();
       
   279 		if (h.iIsr == NaviEngineInterrupt::Spurious)
       
   280 			r=KErrGeneral;
       
   281 		else
       
   282 			{
       
   283 			h.iPtr=(TAny*)anId;
       
   284 			h.iIsr=NaviEngineInterrupt::Spurious;
       
   285 			Disable(anId);
       
   286 			}
       
   287 		NKern::RestoreInterrupts(irq);
       
   288 		}
       
   289 	return r;
       
   290 	}
       
   291 
       
   292 EXPORT_C TInt Interrupt::Enable(TInt anId)
       
   293 	{
       
   294 	__KTRACE_OPT(KEXTENSION,Kern::Printf("Interrupt::Enable id=%d",anId));
       
   295 	TInt r=KErrNone;
       
   296 	// if ID indicates a chained interrupt, call variant...
       
   297 	if (anId<0 && ((((TUint)anId)>>16)&0x7fff)<(TUint)KNumNaviEngineInts)
       
   298 		r=NaviEngineAssp::Variant->InterruptEnable(anId);
       
   299 	else if ((TUint)anId>=(TUint)KNumNaviEngineInts)
       
   300 		r=KErrArgument;
       
   301 	else if (NaviEngineInterrupt::Handlers[anId].iIsr==NaviEngineInterrupt::Spurious)
       
   302 		r=KErrNotReady;
       
   303 	else
       
   304 		{
       
   305 		TUint reg = anId/32;
       
   306 		TUint mask = 1 << (anId - 32*reg);
       
   307 	    AsspRegister::Write32(KHoGidIntEnableBase+reg*4, mask);
       
   308 		}
       
   309 	return r;
       
   310 	}
       
   311 
       
   312 EXPORT_C TInt Interrupt::Disable(TInt anId)
       
   313 	{
       
   314 	__KTRACE_OPT(KEXTENSION,Kern::Printf("Interrupt::Disable id=%d",anId));
       
   315 	TInt r=KErrNone;
       
   316 	// if ID indicates a chained interrupt, call variant...
       
   317 	if (anId<0 && ((((TUint)anId)>>16)&0x7fff)<(TUint)KNumNaviEngineInts)
       
   318 		r=NaviEngineAssp::Variant->InterruptDisable(anId);
       
   319 	else if ((TUint)anId>=(TUint)KNumNaviEngineInts)
       
   320 		r=KErrArgument;
       
   321 	else
       
   322 		{
       
   323 		TUint reg = anId/32;
       
   324 		TUint mask = 1 << (anId - 32*reg);
       
   325 	    AsspRegister::Write32(KHoGidIntDisableBase+reg*4, mask);
       
   326 		}
       
   327 	return r;
       
   328 	}
       
   329 
       
   330 EXPORT_C TInt Interrupt::Clear(TInt anId)
       
   331 	{
       
   332 	__KTRACE_OPT(KEXTENSION,Kern::Printf("Interrupt::Clear id=%d",anId));
       
   333 	TInt r=KErrNone;
       
   334 	// if ID indicates a chained interrupt, call variant...
       
   335 	if (anId<0 && ((((TUint)anId)>>16)&0x7fff)<(TUint)KNumNaviEngineInts)
       
   336 		r=NaviEngineAssp::Variant->InterruptClear(anId);
       
   337 	else if ((TUint)anId>=(TUint)KNumNaviEngineInts)
       
   338 		r=KErrArgument;
       
   339 	else
       
   340 		{
       
   341 		TUint reg = anId/32;
       
   342 		TUint mask = 1 << (anId - 32*reg);
       
   343 	    AsspRegister::Write32(KHoGidPendClearBase+reg*4, mask);
       
   344 		}
       
   345 	return r;
       
   346 	}
       
   347 
       
   348 EXPORT_C TInt Interrupt::SetPriority(TInt /*anId*/, TInt /*aPriority*/)
       
   349 	{
       
   350 	return KErrNotSupported;
       
   351 	}
       
   352 #endif