kernel/eka/nkernsmp/x86/vectors.cpp
author William Roberts <williamr@symbian.org>
Mon, 21 Dec 2009 16:15:43 +0000
changeset 3 9947e075979d
parent 0 a41df078684a
child 43 c1f20ce4abcf
permissions -rw-r--r--
Merge improved comments

// Copyright (c) 2006-2009 Nokia Corporation and/or its subsidiary(-ies).
// All rights reserved.
// This component and the accompanying materials are made available
// under the terms of the License "Eclipse Public License v1.0"
// which accompanies this distribution, and is available
// at the URL "http://www.eclipse.org/legal/epl-v10.html".
//
// Initial Contributors:
// Nokia Corporation - initial contribution.
//
// Contributors:
//
// Description:
// e32\nkernsmp\x86\vectors.cpp
// 
//

#include <x86.h>
#include <apic.h>
#include <nk_irq.h>
#include "vectors.h"

TUint32 __tr();
void __ltr(TInt);

extern "C" void _irqdebug(TUint a)
	{
	if (a>=0x30)
		return;
	__KTRACE_OPT(KSCHED2,DEBUGPRINT("!%02x",a));
	}

extern "C" void IrqStartTrace(TUint32 aVector)
	{
	__ACQUIRE_BTRACE_LOCK();
	BTraceData.iHandler(BTRACE_HEADER(8,BTrace::ECpuUsage,BTrace::EIrqStart),0,0,aVector,0,0,0,0);
	__RELEASE_BTRACE_LOCK();
	}

extern "C" void IrqEndTrace()
	{
	__ACQUIRE_BTRACE_LOCK();
	BTraceData.iHandler(BTRACE_HEADER(4,BTrace::ECpuUsage,BTrace::EIrqEnd),0,0,0,0,0,0,0);
	__RELEASE_BTRACE_LOCK();
	}


/** Register the global IRQ handler
	Called by the base port at boot time to bind the top level IRQ dispatcher
	to the X86 common IRQ handler. Should not be called at any other time.

	The handler specified will be called with IRQs disabled. ESP will point
	to the top of the interrupt stack. On entry to the handler EAX will point
	to a block of saved registers, as follows:

	[EAX+00h] = saved ESI
	[EAX+04h] = saved EDI
	[EAX+08h] = saved EDX
	[EAX+0Ch] = saved ECX
	[EAX+10h] = saved EAX
	[EAX+14h] = saved ES
	[EAX+18h] = saved DS
	[EAX+1Ch] = interrupt vector number
	[EAX+20h] = return EIP
	[EAX+24h] = return CS
	[EAX+28h] = return EFLAGS
	[EAX+2Ch] = return ESP if interrupt occurred while CPL>0
	[EAX+30h] = return SS if interrupt occurred while CPL>0

	The handler should preserve all registers other than EAX, ECX, EDX
	and should return using a standard RET instruction.

	@param	aHandler The address of the top level IRQ dispatcher routine
 */
EXPORT_C void X86::SetIrqHandler(TLinAddr aHandler)
	{
	X86_IrqHandler = aHandler;
	}


/** Return the address immediately after the end of the interrupt stack.

	@return Interrupt Stack Base + Interrupt Stack Size
 */
EXPORT_C TLinAddr X86::IrqStackTop(TInt aCpu)
	{
	TLinAddr a = 0;
	if (aCpu>=0 && aCpu<KMaxCpus)
		a = TLinAddr(TheSubSchedulers[aCpu].i_IrqStackTop);
	else
		{
		TInt irq = NKern::DisableAllInterrupts();
		a = TLinAddr(SubScheduler().i_IrqStackTop);
		NKern::RestoreInterrupts(irq);
		}
	return a;
	}

void SetTrapGate(SX86Des* aEntry, PFV aHandler, TInt aDPL)
	{
	aEntry->iLow=(KRing0CS<<16)|(TUint32(aHandler)&0xffff);
	aEntry->iHigh=(TUint32(aHandler)&0xffff0000) | 0x8f00 | (aDPL<<13);
	}

void SetInterruptGate(SX86Des* aEntry, PFV aHandler, TInt aDPL)
	{
	aEntry->iLow=(KRing0CS<<16)|(TUint32(aHandler)&0xffff);
	aEntry->iHigh=(TUint32(aHandler)&0xffff0000) | 0x8e00 | (aDPL<<13);
	}

void SetTssDescriptor(SX86Des* aEntry, TX86Tss* aTss)
	{
	TUint addr3=TUint(aTss)>>24;
	TUint addr2=(TUint(aTss)>>16)&0xff;
	TUint addr01=TUint(aTss)&0xffff;
	aEntry->iLow=(addr01<<16)|(sizeof(TX86Tss)-1);
	aEntry->iHigh=(addr3<<24)|0x00108900|addr2;
	}

void X86::Init1Interrupts()
//
// Initialise the interrupt and exception vector handlers.
//
	{
//	TheIrqHandler=0;	// done by placing TheIrqHandler, TheFiqHandler in .bss
	__KTRACE_OPT(KBOOT,DEBUGPRINT(">X86::Init1Interrupts()"));

	TCpuPages& cp=X86::CpuPage();
	memclr(cp.iIdt, KIdtSize*sizeof(SX86Des));
	TInt i;
	for (i=0; i<(TInt)(sizeof(TheExcVectors)/sizeof(PFV)); i++)
		{
		if (i==0x03 || i==0x20 || i==0x21)
			SetInterruptGate(cp.iIdt+i, TheExcVectors[i], 3);
		else if (i<0x20 && i!=0x02)
			SetInterruptGate(cp.iIdt+i, TheExcVectors[i], 0);
		if (i==0x02 || i>=0x27)
			SetInterruptGate(cp.iIdt+i, TheExcVectors[i], 0);
		}
	for (i=0; i<KMaxCpus; ++i)
		{
		SCpuData& cd = cp.iCpuData[i];
		memclr(&cd, sizeof(cd) - sizeof(cd.iIrqStack));
		memset(cd.iIrqStack, 0xab+i*2, sizeof(cd.iIrqStack));

		TUint32 esp = (TUint32)cd.iIrqStack + sizeof(cd.iIrqStack);
		cd.iTss.iCR3 = get_cr3();
		cd.iTss.iSs0 = KRing0DS;
		cd.iTss.iEsp0 = esp;
		SetTssDescriptor(&cp.iGdt[5+i], &cd.iTss);

		TSubScheduler& ss = TheSubSchedulers[i];
		ss.i_IrqNestCount = (TAny*)(-1);
		ss.i_IrqStackTop = (TAny*)esp;
		ss.i_Tss = &cd.iTss;
		}

	X86::DefaultCR0 = get_cr0();
	NIrq::HwInit1();

	__ltr(TSS_SELECTOR(0));
	__lidt(cp.iIdt, KIdtSize);
	__KTRACE_OPT(KBOOT,DEBUGPRINT("<X86::Init1Interrupts()"));
	}


extern "C" void ExcFault(TAny*);
void __X86ExcFault(TAny* aInfo)
	{
	ExcFault(aInfo);
	}