commit
						4160f2138d
					
				
											
												Binary file not shown.
											
										
									
								| @ -0,0 +1,3 @@ | |||||||
|  | Test 0:PASS | ||||||
|  | Test 1:PASS | ||||||
|  | Test 2:PASS | ||||||
| @ -0,0 +1,5 @@ | |||||||
|  | TEST=xics | ||||||
|  |  | ||||||
|  | include ../Makefile.test | ||||||
|  |  | ||||||
|  | xics.o : xics.h | ||||||
| @ -0,0 +1,186 @@ | |||||||
|  | /* Copyright 2013-2014 IBM Corp. | ||||||
|  |  * | ||||||
|  |  * Licensed under the Apache License, Version 2.0 (the "License"); | ||||||
|  |  * you may not use this file except in compliance with the License. | ||||||
|  |  * You may obtain a copy of the License at | ||||||
|  |  * | ||||||
|  |  * 	http://www.apache.org/licenses/LICENSE-2.0 | ||||||
|  |  * | ||||||
|  |  * Unless required by applicable law or agreed to in writing, software | ||||||
|  |  * distributed under the License is distributed on an "AS IS" BASIS, | ||||||
|  |  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or | ||||||
|  |  * implied. | ||||||
|  |  * See the License for the specific language governing permissions and | ||||||
|  |  * limitations under the License. | ||||||
|  |  */ | ||||||
|  |  | ||||||
|  | #define STACK_TOP 0x4000 | ||||||
|  |  | ||||||
|  | /* Load an immediate 64-bit value into a register */ | ||||||
|  | #define LOAD_IMM64(r, e)			\ | ||||||
|  | 	lis     r,(e)@highest;			\ | ||||||
|  | 	ori     r,r,(e)@higher;			\ | ||||||
|  | 	rldicr  r,r, 32, 31;			\ | ||||||
|  | 	oris    r,r, (e)@h;			\ | ||||||
|  | 	ori     r,r, (e)@l; | ||||||
|  |  | ||||||
|  | 	.section ".head","ax" | ||||||
|  |  | ||||||
|  | 	/* Microwatt currently enters in LE mode at 0x0 */ | ||||||
|  | 	. = 0 | ||||||
|  | .global _start | ||||||
|  | _start: | ||||||
|  | 	LOAD_IMM64(%r12, 0x000000000ffffff) | ||||||
|  | 	mtdec	%r12 | ||||||
|  | 	LOAD_IMM64(%r12, 0x9000000000008003) | ||||||
|  | 	mtmsrd	%r12	// EE on | ||||||
|  | 	/* setup stack */ | ||||||
|  | 	LOAD_IMM64(%r1, STACK_TOP - 0x100) | ||||||
|  | 	LOAD_IMM64(%r12, main) | ||||||
|  | 	mtctr	%r12 | ||||||
|  | 	bctrl | ||||||
|  | 	attn // terminate on exit | ||||||
|  | 	b . | ||||||
|  |  | ||||||
|  | #define EXCEPTION(nr)		\ | ||||||
|  | 	.= nr			;\ | ||||||
|  | 	b	. | ||||||
|  |  | ||||||
|  | 	/* More exception stubs */ | ||||||
|  | 	EXCEPTION(0x300) | ||||||
|  | 	EXCEPTION(0x380) | ||||||
|  | 	EXCEPTION(0x400) | ||||||
|  | 	EXCEPTION(0x480) | ||||||
|  | 	. = 0x500 | ||||||
|  | 	b	__isr | ||||||
|  |  | ||||||
|  | 	EXCEPTION(0x600) | ||||||
|  | 	EXCEPTION(0x700) | ||||||
|  | 	EXCEPTION(0x800) | ||||||
|  | 	EXCEPTION(0x900) | ||||||
|  | 	EXCEPTION(0x980) | ||||||
|  | 	EXCEPTION(0xa00) | ||||||
|  | 	EXCEPTION(0xb00) | ||||||
|  | 	EXCEPTION(0xc00) | ||||||
|  | 	EXCEPTION(0xd00) | ||||||
|  |  | ||||||
|  | //  ISR data | ||||||
|  |  | ||||||
|  | #define REDZONE_SIZE    (512) | ||||||
|  | #define REG_SAVE_SIZE	((32 + 5)*8) | ||||||
|  | #define STACK_FRAME_C_MINIMAL   64 | ||||||
|  |  | ||||||
|  | #define SAVE_NIA	(32*8) | ||||||
|  | #define SAVE_LR		(33*8) | ||||||
|  | #define SAVE_CTR	(34*8) | ||||||
|  | #define SAVE_CR		(35*8) | ||||||
|  | #define SAVE_SRR1	(36*8) | ||||||
|  |  | ||||||
|  | __isr: | ||||||
|  | /* | ||||||
|  |  * Assume where we are coming from has a stack and can save there. | ||||||
|  |  * We save the full register set. Since we are calling out to C, we | ||||||
|  |  * could just save the ABI volatile registers | ||||||
|  |  */ | ||||||
|  | 	stdu	%r1,-(REG_SAVE_SIZE+REDZONE_SIZE)(%r1) | ||||||
|  | 	std	%r0,   1*8(%r1) | ||||||
|  | //	std	%r1,   1*8(%r1) | ||||||
|  | 	std	%r2,   2*8(%r1) | ||||||
|  | 	std	%r3,   3*8(%r1) | ||||||
|  | 	std	%r4,   4*8(%r1) | ||||||
|  | 	std	%r5,   5*8(%r1) | ||||||
|  | 	std	%r6,   6*8(%r1) | ||||||
|  | 	std	%r7,   7*8(%r1) | ||||||
|  | 	std	%r8,   8*8(%r1) | ||||||
|  | 	std	%r9,   9*8(%r1) | ||||||
|  | 	std	%r10, 10*8(%r1) | ||||||
|  | 	std	%r11, 11*8(%r1) | ||||||
|  | 	std	%r12, 12*8(%r1) | ||||||
|  | 	std	%r13, 13*8(%r1) | ||||||
|  | 	std	%r14, 14*8(%r1) | ||||||
|  | 	std	%r15, 15*8(%r1) | ||||||
|  | 	std	%r16, 16*8(%r1) | ||||||
|  | 	std	%r17, 17*8(%r1) | ||||||
|  | 	std	%r18, 18*8(%r1) | ||||||
|  | 	std	%r19, 19*8(%r1) | ||||||
|  | 	std	%r20, 20*8(%r1) | ||||||
|  | 	std	%r21, 21*8(%r1) | ||||||
|  | 	std	%r22, 22*8(%r1) | ||||||
|  | 	std	%r23, 23*8(%r1) | ||||||
|  | 	std	%r24, 24*8(%r1) | ||||||
|  | 	std	%r25, 25*8(%r1) | ||||||
|  | 	std	%r26, 26*8(%r1) | ||||||
|  | 	std	%r27, 27*8(%r1) | ||||||
|  | 	std	%r28, 28*8(%r1) | ||||||
|  | 	std	%r29, 29*8(%r1) | ||||||
|  | 	std	%r30, 30*8(%r1) | ||||||
|  | 	std	%r31, 31*8(%r1) | ||||||
|  | 	mfsrr0	%r0 | ||||||
|  | 	std	%r0,  SAVE_NIA*8(%r1) | ||||||
|  | 	mflr	%r0 | ||||||
|  | 	std	%r0,  SAVE_LR*8(%r1) | ||||||
|  | 	mfctr	%r0 | ||||||
|  | 	std	%r0,  SAVE_CTR*8(%r1) | ||||||
|  | 	mfcr	%r0 | ||||||
|  | 	std	%r0,  SAVE_CR*8(%r1) | ||||||
|  | 	mfsrr1	%r0 | ||||||
|  | 	std	%r0,  SAVE_SRR1*8(%r1) | ||||||
|  |  | ||||||
|  | 	stdu	%r1,-STACK_FRAME_C_MINIMAL(%r1) | ||||||
|  | 	LOAD_IMM64(%r3, isr) | ||||||
|  | 	mtctr	%r3, | ||||||
|  | 	bctrl | ||||||
|  | 	nop | ||||||
|  | 	ld	%r1, 0(%r1) | ||||||
|  |  | ||||||
|  | 	ld	%r0,   1*8(%r1) | ||||||
|  | //	ld	%r1,   1*8(%r1) // do this at rfid | ||||||
|  | 	ld	%r2,   2*8(%r1) | ||||||
|  | //	ld	%r3,   3*8(%r1) // do this at rfid | ||||||
|  | 	ld	%r4,   4*8(%r1) | ||||||
|  | 	ld	%r5,   5*8(%r1) | ||||||
|  | 	ld	%r6,   6*8(%r1) | ||||||
|  | 	ld	%r7,   7*8(%r1) | ||||||
|  | 	ld	%r8,   8*8(%r1) | ||||||
|  | 	ld	%r9,   9*8(%r1) | ||||||
|  | 	ld	%r10, 10*8(%r1) | ||||||
|  | 	ld	%r11, 11*8(%r1) | ||||||
|  | 	ld	%r12, 12*8(%r1) | ||||||
|  | 	ld	%r13, 13*8(%r1) | ||||||
|  | 	ld	%r14, 14*8(%r1) | ||||||
|  | 	ld	%r15, 15*8(%r1) | ||||||
|  | 	ld	%r16, 16*8(%r1) | ||||||
|  | 	ld	%r17, 17*8(%r1) | ||||||
|  | 	ld	%r18, 18*8(%r1) | ||||||
|  | 	ld	%r19, 19*8(%r1) | ||||||
|  | 	ld	%r20, 20*8(%r1) | ||||||
|  | 	ld	%r21, 21*8(%r1) | ||||||
|  | 	ld	%r22, 22*8(%r1) | ||||||
|  | 	ld	%r23, 23*8(%r1) | ||||||
|  | 	ld	%r24, 24*8(%r1) | ||||||
|  | 	ld	%r25, 25*8(%r1) | ||||||
|  | 	ld	%r26, 26*8(%r1) | ||||||
|  | 	ld	%r27, 27*8(%r1) | ||||||
|  | 	ld	%r28, 28*8(%r1) | ||||||
|  | 	ld	%r29, 29*8(%r1) | ||||||
|  | 	ld	%r30, 30*8(%r1) | ||||||
|  | 	ld	%r31, 31*8(%r1) | ||||||
|  |  | ||||||
|  | 	ld	%r3, SAVE_LR*8(%r1) | ||||||
|  | 	mtlr	%r3 | ||||||
|  | 	ld	%r3, SAVE_CTR*8(%r1) | ||||||
|  | 	mtctr	%r3 | ||||||
|  | 	ld	%r3, SAVE_CR*8(%r1) | ||||||
|  | 	mtcr	%r3 | ||||||
|  | 	ld	%r3, SAVE_SRR1*8(%r1) | ||||||
|  | 	mtsrr1	%r3 | ||||||
|  | 	ld	%r3, SAVE_NIA*8(%r1) | ||||||
|  | 	mtsrr0	%r3 | ||||||
|  |  | ||||||
|  | 	/* restore %r3 */ | ||||||
|  | 	ld	%r3, 3*8(%r1) | ||||||
|  |  | ||||||
|  | 	/* do final fixup r1 */ | ||||||
|  | 	ld	%r1, 0*8(%r1) | ||||||
|  |  | ||||||
|  | 	rfid | ||||||
| @ -0,0 +1,13 @@ | |||||||
|  | SECTIONS | ||||||
|  | { | ||||||
|  | 	_start = .; | ||||||
|  | 	. = 0; | ||||||
|  | 	.head : { | ||||||
|  | 		KEEP(*(.head)) | ||||||
|  | 	} | ||||||
|  | 	. = 0x1000; | ||||||
|  | 	.text : { *(.text) } | ||||||
|  | 	. = 0x3000; | ||||||
|  | 	.data : { *(.data) } | ||||||
|  | 	.bss : { *(.bss) } | ||||||
|  | } | ||||||
| @ -0,0 +1,262 @@ | |||||||
|  | #include <stddef.h> | ||||||
|  | #include <stdint.h> | ||||||
|  | #include <stdbool.h> | ||||||
|  | #include <limits.h> | ||||||
|  |  | ||||||
|  | #include "console.h" | ||||||
|  | #include "xics.h" | ||||||
|  |  | ||||||
|  | #undef DEBUG | ||||||
|  | //#define DEBUG 1 | ||||||
|  |  | ||||||
|  | void print_number(unsigned int i) // only for i = 0-999 | ||||||
|  | { | ||||||
|  | 	unsigned int j, k, m; | ||||||
|  | 	bool zeros = false; | ||||||
|  |  | ||||||
|  | 	k = 1000000000; | ||||||
|  |  | ||||||
|  | 	for (m = 0; m < 10 ; m++) { | ||||||
|  | 		j = i/k; | ||||||
|  | 		if (m == 9) zeros = true; | ||||||
|  | 		if (zeros || (j != 0)) { | ||||||
|  | 		    putchar(48 + j); | ||||||
|  | 		    zeros = true; | ||||||
|  | 		} | ||||||
|  | 		i = i % k; | ||||||
|  | 		k = k / 10; | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #ifdef DEBUG | ||||||
|  | #define DEBUG_STR "\r\nDEBUG: " | ||||||
|  | void debug_print(int i) | ||||||
|  | { | ||||||
|  | 	putstr(DEBUG_STR, strlen(DEBUG_STR)); | ||||||
|  | 	print_number(i); | ||||||
|  | 	putstr("\r\n", 2); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #define debug_putstr(a, b) putstr(a,b) | ||||||
|  | #else | ||||||
|  | #define debug_putstr(a, b) | ||||||
|  | #define debug_print(i) | ||||||
|  | #endif | ||||||
|  |  | ||||||
|  | #define ASSERT_FAIL "() ASSERT_FAILURE!\r\n " | ||||||
|  | #define assert(cond)	\ | ||||||
|  | 	if (!(cond))  { \ | ||||||
|  | 		putstr(__FILE__, strlen(__FILE__)); \ | ||||||
|  | 		putstr(":", 1);	    \ | ||||||
|  | 		print_number(__LINE__);	\ | ||||||
|  | 		putstr(":", 1);	    \ | ||||||
|  | 		putstr(__FUNCTION__, strlen(__FUNCTION__));\ | ||||||
|  | 		putstr(ASSERT_FAIL, strlen(ASSERT_FAIL)); \ | ||||||
|  | 		__asm__ ("attn"); \ | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  |  | ||||||
|  | volatile uint64_t isrs_run; | ||||||
|  |  | ||||||
|  | #define ISR_IPI      0x0000000000000001 | ||||||
|  | #define ISR_UART     0x0000000000000002 | ||||||
|  | #define ISR_SPURIOUS 0x0000000000000004 | ||||||
|  |  | ||||||
|  | #define IPI "IPI\r\n" | ||||||
|  | void ipi_isr(void) { | ||||||
|  | 	debug_putstr(IPI, strlen(IPI)); | ||||||
|  |  | ||||||
|  | 	isrs_run |= ISR_IPI; | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | #define UART "UART\r\n" | ||||||
|  | void uart_isr(void) { | ||||||
|  | 	debug_putstr(UART, strlen(UART)); | ||||||
|  |  | ||||||
|  | 	potato_uart_irq_dis(); // disable interrupt to ack it | ||||||
|  |  | ||||||
|  | 	isrs_run |= ISR_UART; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // The hardware doesn't support this but it's part of XICS so add it. | ||||||
|  | #define SPURIOUS "SPURIOUS\r\n" | ||||||
|  | void spurious_isr(void) { | ||||||
|  | 	debug_putstr(SPURIOUS, strlen(SPURIOUS)); | ||||||
|  |  | ||||||
|  | 	isrs_run |= ISR_SPURIOUS; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | struct isr_op { | ||||||
|  | 	void (*func)(void); | ||||||
|  | 	int source_id; | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | struct isr_op isr_table[] = { | ||||||
|  | 	{ .func = ipi_isr,  .source_id = 2 }, | ||||||
|  | 	{ .func = uart_isr, .source_id = 16 }, | ||||||
|  | 	{ .func = spurious_isr,  .source_id = 0 }, | ||||||
|  | 	{ .func = NULL, .source_id = 0 } | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | bool ipi_running; | ||||||
|  |  | ||||||
|  | #define ISR "ISR XISR=" | ||||||
|  | void isr(void) | ||||||
|  | { | ||||||
|  | 	struct isr_op *op; | ||||||
|  | 	uint32_t xirr; | ||||||
|  |  | ||||||
|  | 	assert(!ipi_running); // check we aren't reentrant | ||||||
|  | 	ipi_running = true; | ||||||
|  |  | ||||||
|  | 	xirr = xics_read32(XICS_XIRR); // read hardware irq source | ||||||
|  |  | ||||||
|  | #ifdef DEBUG | ||||||
|  | 	putstr(ISR, strlen(ISR)); | ||||||
|  | 	print_number(xirr & 0xff); | ||||||
|  | 	putstr("\r\n", 2); | ||||||
|  | #endif | ||||||
|  |  | ||||||
|  | 	op = isr_table; | ||||||
|  | 	while (1) { | ||||||
|  | 		assert(op->func); // didn't find isr | ||||||
|  | 		if (op->source_id == (xirr & 0x00ffffff)) { | ||||||
|  | 		    op->func(); | ||||||
|  | 		    break; | ||||||
|  | 		} | ||||||
|  | 		op++; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	xics_write32(XICS_XIRR, xirr); // EOI | ||||||
|  |  | ||||||
|  | 	ipi_running = false; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /*****************************************/ | ||||||
|  |  | ||||||
|  | int xics_test_0(void) | ||||||
|  | { | ||||||
|  | 	// setup | ||||||
|  | 	xics_write8(XICS_XIRR, 0x00); // mask all interrupts | ||||||
|  | 	isrs_run = 0; | ||||||
|  |  | ||||||
|  | 	xics_write8(XICS_XIRR, 0x00); // mask all interrupts | ||||||
|  |  | ||||||
|  | 	// trigger two interrupts | ||||||
|  | 	potato_uart_irq_en(); // cause 0x500 interrupt | ||||||
|  | 	xics_write8(XICS_MFRR, 0x05); // cause 0x500 interrupt | ||||||
|  |  | ||||||
|  | 	// still masked, so shouldn't happen yet | ||||||
|  | 	assert(isrs_run == 0); | ||||||
|  |  | ||||||
|  | 	// unmask IPI only | ||||||
|  | 	xics_write8(XICS_XIRR, 0x40); | ||||||
|  | 	assert(isrs_run == ISR_IPI); | ||||||
|  |  | ||||||
|  | 	// unmask UART | ||||||
|  | 	xics_write8(XICS_XIRR, 0xc0); | ||||||
|  | 	assert(isrs_run == (ISR_IPI | ISR_UART)); | ||||||
|  |  | ||||||
|  | 	// cleanup | ||||||
|  | 	xics_write8(XICS_XIRR, 0x00); // mask all interrupts | ||||||
|  | 	isrs_run = 0; | ||||||
|  |  | ||||||
|  | 	return 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | int xics_test_1(void) | ||||||
|  | { | ||||||
|  | 	// setup | ||||||
|  | 	xics_write8(XICS_XIRR, 0x00); // mask all interrupts | ||||||
|  | 	isrs_run = 0; | ||||||
|  |  | ||||||
|  | 	xics_write8(XICS_XIRR, 0xff); // allow all interrupts | ||||||
|  |  | ||||||
|  | 	// should be none pending | ||||||
|  | 	assert(isrs_run == 0); | ||||||
|  |  | ||||||
|  | 	// trigger both | ||||||
|  | 	potato_uart_irq_en(); // cause 0x500 interrupt | ||||||
|  | 	xics_write8(XICS_MFRR, 0x05); // cause 0x500 interrupt | ||||||
|  |  | ||||||
|  | 	assert(isrs_run == (ISR_IPI | ISR_UART)); | ||||||
|  |  | ||||||
|  | 	// cleanup | ||||||
|  | 	xics_write8(XICS_XIRR, 0x00); // mask all interrupts | ||||||
|  | 	isrs_run = 0; | ||||||
|  |  | ||||||
|  | 	return 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | void mtmsrd(uint64_t val) | ||||||
|  | { | ||||||
|  | 	__asm__ volatile("mtmsrd %0" : : "r" (val)); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | int xics_test_2(void) | ||||||
|  | { | ||||||
|  | 	// setup | ||||||
|  | 	xics_write8(XICS_XIRR, 0x00); // mask all interrupts | ||||||
|  | 	isrs_run = 0; | ||||||
|  |  | ||||||
|  | 	// trigger interrupts with MSR[EE]=0 and show they are not run | ||||||
|  | 	mtmsrd(0x9000000000000003); // EE off | ||||||
|  |  | ||||||
|  | 	xics_write8(XICS_XIRR, 0xff); // allow all interrupts | ||||||
|  |  | ||||||
|  | 	// trigger an IPI | ||||||
|  | 	xics_write8(XICS_MFRR, 0x05); // cause 0x500 interrupt | ||||||
|  |  | ||||||
|  | 	assert(isrs_run == 0); | ||||||
|  |  | ||||||
|  | 	mtmsrd(0x9000000000008003); // EE on | ||||||
|  | 	assert(isrs_run == ISR_IPI); | ||||||
|  |  | ||||||
|  | 	// cleanup | ||||||
|  | 	xics_write8(XICS_XIRR, 0x00); // mask all interrupts | ||||||
|  | 	isrs_run = 0; | ||||||
|  |  | ||||||
|  | 	return 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #define TEST "Test " | ||||||
|  | #define PASS "PASS\r\n" | ||||||
|  | #define FAIL "FAIL\r\n" | ||||||
|  |  | ||||||
|  | int (*tests[])(void) = { | ||||||
|  | 	xics_test_0, | ||||||
|  | 	xics_test_1, | ||||||
|  | 	xics_test_2, | ||||||
|  | 	NULL | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | int main(void) | ||||||
|  | { | ||||||
|  | 	int fail = 0; | ||||||
|  | 	int i = 0; | ||||||
|  | 	int (*t)(void); | ||||||
|  |  | ||||||
|  | 	potato_uart_init(); | ||||||
|  | 	ipi_running = false; | ||||||
|  |  | ||||||
|  | 	/* run the tests */ | ||||||
|  | 	while (1) { | ||||||
|  | 		t = tests[i]; | ||||||
|  | 		if (!t) | ||||||
|  | 			break; | ||||||
|  |  | ||||||
|  | 		putstr(TEST, strlen(TEST)); | ||||||
|  | 		print_number(i); | ||||||
|  | 		putstr(": ", 1); | ||||||
|  | 		if (t() != 0) { | ||||||
|  | 			fail = 1; | ||||||
|  | 			putstr(FAIL, strlen(FAIL)); | ||||||
|  | 		} else | ||||||
|  | 			putstr(PASS, strlen(PASS)); | ||||||
|  |  | ||||||
|  | 		i++; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return fail; | ||||||
|  | } | ||||||
| @ -0,0 +1,36 @@ | |||||||
|  | #include <stdint.h> | ||||||
|  |  | ||||||
|  | #define XICS_BASE 0xc0004000 | ||||||
|  |  | ||||||
|  | static uint64_t xics_base = XICS_BASE; | ||||||
|  |  | ||||||
|  | #define XICS_XIRR_POLL 0x0 | ||||||
|  | #define XICS_XIRR      0x4 | ||||||
|  | #define XICS_RESV      0x8 | ||||||
|  | #define XICS_MFRR      0xC | ||||||
|  |  | ||||||
|  | uint8_t xics_read8(int offset) | ||||||
|  | { | ||||||
|  | 	uint32_t val; | ||||||
|  |  | ||||||
|  | 	__asm__ volatile("lbzcix %0,%1,%2" : "=r" (val) : "b" (xics_base), "r" (offset)); | ||||||
|  | 	return val; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | void xics_write8(int offset, uint8_t val) | ||||||
|  | { | ||||||
|  | 	__asm__ volatile("stbcix %0,%1,%2" : : "r" (val), "b" (xics_base), "r" (offset)); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | uint32_t xics_read32(int offset) | ||||||
|  | { | ||||||
|  | 	uint32_t val; | ||||||
|  |  | ||||||
|  | 	__asm__ volatile("lwzcix %0,%1,%2" : "=r" (val) : "b" (xics_base), "r" (offset)); | ||||||
|  | 	return val; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | void xics_write32(int offset, uint32_t val) | ||||||
|  | { | ||||||
|  | 	__asm__ volatile("stwcix %0,%1,%2" : : "r" (val), "b" (xics_base), "r" (offset)); | ||||||
|  | } | ||||||
| @ -0,0 +1,207 @@ | |||||||
|  | -- | ||||||
|  | -- This is a simple XICS compliant interrupt controller.  This is a | ||||||
|  | -- Presenter (ICP) and Source (ICS) in a single unit with no routing | ||||||
|  | -- layer. | ||||||
|  | -- | ||||||
|  | -- The sources have a fixed IRQ priority set by HW_PRIORITY. The | ||||||
|  | -- source id starts at 16 for int_level_in(0) and go up from | ||||||
|  | -- there (ie int_level_in(1) is source id 17). | ||||||
|  | -- | ||||||
|  | -- The presentation layer will pick an interupt that is more | ||||||
|  | -- favourable than the current CPPR and present it via the XISR and | ||||||
|  | -- send an interrpt to the processor (via e_out). This may not be the | ||||||
|  | -- highest priority interrupt currently presented (which is allowed | ||||||
|  | -- via XICS) | ||||||
|  | -- | ||||||
|  |  | ||||||
|  | library ieee; | ||||||
|  | use ieee.std_logic_1164.all; | ||||||
|  | use ieee.numeric_std.all; | ||||||
|  |  | ||||||
|  | library work; | ||||||
|  | use work.common.all; | ||||||
|  | use work.wishbone_types.all; | ||||||
|  |  | ||||||
|  | entity xics is | ||||||
|  |     generic ( | ||||||
|  |         LEVEL_NUM : positive := 16 | ||||||
|  |         ); | ||||||
|  |     port ( | ||||||
|  |         clk          : in std_logic; | ||||||
|  |         rst          : in std_logic; | ||||||
|  |  | ||||||
|  |         wb_in   : in wishbone_master_out; | ||||||
|  |         wb_out  : out wishbone_slave_out; | ||||||
|  |  | ||||||
|  | 	int_level_in : in std_ulogic_vector(LEVEL_NUM - 1 downto 0); | ||||||
|  |  | ||||||
|  | 	e_out : out XicsToExecute1Type | ||||||
|  |         ); | ||||||
|  | end xics; | ||||||
|  |  | ||||||
|  | architecture behaviour of xics is | ||||||
|  |     type reg_internal_t is record | ||||||
|  | 	xisr : std_ulogic_vector(23 downto 0); | ||||||
|  | 	cppr : std_ulogic_vector(7 downto 0); | ||||||
|  | 	pending_priority : std_ulogic_vector(7 downto 0); | ||||||
|  | 	mfrr : std_ulogic_vector(7 downto 0); | ||||||
|  | 	mfrr_pending : std_ulogic; | ||||||
|  | 	irq : std_ulogic; | ||||||
|  | 	wb_rd_data : wishbone_data_type; | ||||||
|  | 	wb_ack : std_ulogic; | ||||||
|  |     end record; | ||||||
|  |     constant reg_internal_init : reg_internal_t := | ||||||
|  | 	(wb_ack => '0', | ||||||
|  | 	 mfrr_pending => '0', | ||||||
|  | 	 mfrr => x"00", -- mask everything on reset | ||||||
|  | 	 irq => '0', | ||||||
|  | 	 others => (others => '0')); | ||||||
|  |  | ||||||
|  |     signal r, r_next : reg_internal_t; | ||||||
|  |  | ||||||
|  |     -- hardwire the hardware IRQ priority | ||||||
|  |     constant HW_PRIORITY : std_ulogic_vector(7 downto 0) := x"80"; | ||||||
|  |  | ||||||
|  |     -- 32 bit offsets for each presentation | ||||||
|  |     constant XIRR_POLL : std_ulogic_vector(31 downto 0) := x"00000000"; | ||||||
|  |     constant XIRR      : std_ulogic_vector(31 downto 0) := x"00000004"; | ||||||
|  |     constant RESV0     : std_ulogic_vector(31 downto 0) := x"00000008"; | ||||||
|  |     constant MFRR      : std_ulogic_vector(31 downto 0) := x"0000000c"; | ||||||
|  |  | ||||||
|  | begin | ||||||
|  |  | ||||||
|  |     regs : process(clk) | ||||||
|  |     begin | ||||||
|  | 	if rising_edge(clk) then | ||||||
|  | 	    r <= r_next; | ||||||
|  | 	end if; | ||||||
|  |     end process; | ||||||
|  |  | ||||||
|  |     wb_out.dat <= r.wb_rd_data; | ||||||
|  |     wb_out.ack <= r.wb_ack; | ||||||
|  |     wb_out.stall <= '0'; -- never stall wishbone | ||||||
|  |     e_out.irq <= r.irq; | ||||||
|  |  | ||||||
|  |     comb : process(all) | ||||||
|  | 	variable v : reg_internal_t; | ||||||
|  | 	variable xirr_accept_rd : std_ulogic; | ||||||
|  | 	variable irq_eoi : std_ulogic; | ||||||
|  |     begin | ||||||
|  | 	v := r; | ||||||
|  |  | ||||||
|  | 	v.wb_ack := '0'; | ||||||
|  |  | ||||||
|  | 	xirr_accept_rd := '0'; | ||||||
|  | 	irq_eoi := '0'; | ||||||
|  |  | ||||||
|  | 	if wb_in.cyc = '1' and wb_in.stb = '1' then | ||||||
|  | 	    -- wishbone addresses we get are 64 bit alligned, so we | ||||||
|  | 	    -- need to use the sel bits to get 32 bit chunks. | ||||||
|  | 	    v.wb_ack := '1'; -- always ack | ||||||
|  | 	    if wb_in.we = '1' then -- write | ||||||
|  | 		-- writes to both XIRR are the same | ||||||
|  | 		if wb_in.adr = XIRR_POLL then | ||||||
|  | 		    report "XICS XIRR_POLL/XIRR write"; | ||||||
|  | 		    if wb_in.sel = x"0f" then -- 4 bytes | ||||||
|  | 			v.cppr := wb_in.dat(31 downto 24); | ||||||
|  | 		    elsif wb_in.sel = x"f0"  then -- 4 byte | ||||||
|  | 			v.cppr := wb_in.dat(63 downto 56); | ||||||
|  | 			irq_eoi := '1'; | ||||||
|  | 		    elsif wb_in.sel = x"01"  then -- 1 byte | ||||||
|  | 			v.cppr := wb_in.dat(7 downto 0); | ||||||
|  | 		    elsif wb_in.sel = x"10"  then -- 1 byte | ||||||
|  | 			v.cppr := wb_in.dat(39 downto 32); | ||||||
|  | 		    end if; | ||||||
|  |  | ||||||
|  | 		elsif wb_in.adr = RESV0 then | ||||||
|  | 		    report "XICS MFRR write"; | ||||||
|  | 		    if wb_in.sel = x"f0" then -- 4 bytes | ||||||
|  | 			v.mfrr_pending := '1'; | ||||||
|  | 			v.mfrr := wb_in.dat(63 downto 56); | ||||||
|  | 		    elsif wb_in.sel = x"10" then -- 1 byte | ||||||
|  | 			v.mfrr_pending := '1'; | ||||||
|  | 			v.mfrr := wb_in.dat(39 downto 32); | ||||||
|  | 		    end if; | ||||||
|  |  | ||||||
|  | 		end if; | ||||||
|  |  | ||||||
|  | 	    else -- read | ||||||
|  | 		v.wb_rd_data := (others => '0'); | ||||||
|  |  | ||||||
|  | 		if wb_in.adr = XIRR_POLL then | ||||||
|  | 		    report "XICS XIRR_POLL/XIRR read"; | ||||||
|  | 		    if wb_in.sel = x"0f" then | ||||||
|  | 			v.wb_rd_data(23 downto  0) := r.xisr; | ||||||
|  | 			v.wb_rd_data(31 downto 24) := r.cppr; | ||||||
|  | 		    elsif wb_in.sel = x"f0" then | ||||||
|  | 			v.wb_rd_data(55 downto 32) := r.xisr; | ||||||
|  | 			v.wb_rd_data(63 downto 56) := r.cppr; | ||||||
|  | 			xirr_accept_rd := '1'; | ||||||
|  | 		    elsif wb_in.sel = x"01" then | ||||||
|  | 			v.wb_rd_data(7 downto  0) := r.cppr; | ||||||
|  | 		    elsif wb_in.sel = x"10" then | ||||||
|  | 			v.wb_rd_data(39 downto 32) := r.cppr; | ||||||
|  | 		    end if; | ||||||
|  |  | ||||||
|  | 		elsif wb_in.adr = RESV0 then | ||||||
|  | 		    report "XICS MFRR read"; | ||||||
|  | 		    if wb_in.sel = x"f0" then -- 4 bytes | ||||||
|  | 			v.wb_rd_data(63 downto 56) := r.mfrr; | ||||||
|  | 		    elsif wb_in.sel = x"10" then -- 1 byte | ||||||
|  | 			v.wb_rd_data( 7 downto  0) := r.mfrr; | ||||||
|  | 		    end if; | ||||||
|  | 		end if; | ||||||
|  | 	    end if; | ||||||
|  | 	end if; | ||||||
|  |  | ||||||
|  | 	-- generate interrupt | ||||||
|  | 	if r.irq = '0' then | ||||||
|  | 	    -- Here we just present any interrupt that's valid and | ||||||
|  | 	    -- below cppr. For ordering, we ignore hardware | ||||||
|  | 	    -- priorities. | ||||||
|  | 	    if unsigned(HW_PRIORITY) < unsigned(r.cppr) then -- | ||||||
|  | 		-- lower HW sources are higher priority | ||||||
|  | 		for i in LEVEL_NUM - 1 downto 0 loop | ||||||
|  | 		    if int_level_in(i) = '1' then | ||||||
|  | 			v.irq := '1'; | ||||||
|  | 			v.xisr := std_ulogic_vector(to_unsigned(16 + i, 24)); | ||||||
|  | 			v.pending_priority := HW_PRIORITY; -- hardware HW IRQs | ||||||
|  | 		    end if; | ||||||
|  | 		end loop; | ||||||
|  | 	    end if; | ||||||
|  |  | ||||||
|  | 	    -- Do mfrr as a higher priority so mfrr_pending is cleared | ||||||
|  | 	    if unsigned(r.mfrr) < unsigned(r.cppr) then -- | ||||||
|  | 		report "XICS: MFRR INTERRUPT"; | ||||||
|  | 		-- IPI | ||||||
|  | 		if r.mfrr_pending = '1' then | ||||||
|  | 		    v.irq := '1'; | ||||||
|  | 		    v.xisr := x"000002"; -- special XICS MFRR IRQ source number | ||||||
|  | 		    v.pending_priority := r.mfrr; | ||||||
|  | 		    v.mfrr_pending := '0'; | ||||||
|  | 		end if; | ||||||
|  | 	    end if; | ||||||
|  | 	end if; | ||||||
|  |  | ||||||
|  | 	-- Accept the interrupt | ||||||
|  | 	if xirr_accept_rd = '1' then | ||||||
|  | 	    report "XICS: ACCEPT" & | ||||||
|  | 		" cppr:" &  to_hstring(r.cppr) & | ||||||
|  | 		" xisr:" & to_hstring(r.xisr) & | ||||||
|  | 		" mfrr:" & to_hstring(r.mfrr); | ||||||
|  | 	    v.cppr := r.pending_priority; | ||||||
|  | 	end if; | ||||||
|  |  | ||||||
|  | 	if irq_eoi = '1' then | ||||||
|  | 	    v.irq := '0'; | ||||||
|  | 	end if; | ||||||
|  |  | ||||||
|  | 	if rst = '1' then | ||||||
|  | 	    v := reg_internal_init; | ||||||
|  | 	end if; | ||||||
|  |  | ||||||
|  | 	r_next <= v; | ||||||
|  |  | ||||||
|  |     end process; | ||||||
|  |  | ||||||
|  | end architecture behaviour; | ||||||
					Loading…
					
					
				
		Reference in New Issue
	
	 Anton Blanchard
						Anton Blanchard