diff options
Diffstat (limited to 'arch/arc/lib')
| -rw-r--r-- | arch/arc/lib/Makefile | 8 | ||||
| -rw-r--r-- | arch/arc/lib/_millicodethunk.S | 226 | ||||
| -rw-r--r-- | arch/arc/lib/cache.c | 168 | ||||
| -rw-r--r-- | arch/arc/lib/cpu.c | 47 | ||||
| -rw-r--r-- | arch/arc/lib/interrupts.c | 143 | ||||
| -rw-r--r-- | arch/arc/lib/libgcc2.c | 161 | ||||
| -rw-r--r-- | arch/arc/lib/libgcc2.h | 132 | ||||
| -rw-r--r-- | arch/arc/lib/memcmp.S | 2 | ||||
| -rw-r--r-- | arch/arc/lib/relocate.c | 19 | ||||
| -rw-r--r-- | arch/arc/lib/reset.c | 19 | ||||
| -rw-r--r-- | arch/arc/lib/sections.c | 2 | ||||
| -rw-r--r-- | arch/arc/lib/timer.c | 24 | 
12 files changed, 938 insertions, 13 deletions
| diff --git a/arch/arc/lib/Makefile b/arch/arc/lib/Makefile index 7675f855d5a..b8028c91e11 100644 --- a/arch/arc/lib/Makefile +++ b/arch/arc/lib/Makefile @@ -4,6 +4,9 @@  # SPDX-License-Identifier:	GPL-2.0+  # +obj-y += cache.o +obj-y += cpu.o +obj-y += interrupts.o  obj-y += sections.o  obj-y += relocate.o  obj-y += strchr-700.o @@ -13,4 +16,9 @@ obj-y += strlen.o  obj-y += memcmp.o  obj-y += memcpy-700.o  obj-y += memset.o +obj-y += reset.o +obj-y += timer.o +  obj-$(CONFIG_CMD_BOOTM) += bootm.o + +lib-$(CONFIG_USE_PRIVATE_LIBGCC) += _millicodethunk.o libgcc2.o diff --git a/arch/arc/lib/_millicodethunk.S b/arch/arc/lib/_millicodethunk.S new file mode 100644 index 00000000000..b3324167d47 --- /dev/null +++ b/arch/arc/lib/_millicodethunk.S @@ -0,0 +1,226 @@ +/* + * Copyright (C) 1995, 1997, 2007-2013 Free Software Foundation, Inc. + * + * SPDX-License-Identifier:	GPL-2.0+ + */ + + /* ANSI concatenation macros.  */ + + #define CONCAT1(a, b) CONCAT2(a, b) + #define CONCAT2(a, b) a ## b + + /* Use the right prefix for global labels.  */ + + #define SYM(x) CONCAT1 (__USER_LABEL_PREFIX__, x) + +#ifndef WORKING_ASSEMBLER +#define abs_l abs +#define asl_l asl +#define mov_l mov +#endif + +#define FUNC(X)         .type SYM(X),@function +#define HIDDEN_FUNC(X)	FUNC(X)` .hidden X +#define ENDFUNC0(X)     .Lfe_##X: .size X,.Lfe_##X-X +#define ENDFUNC(X)      ENDFUNC0(X) + +	.section .text +	.align 4 +	.global SYM(__st_r13_to_r15) +	.global SYM(__st_r13_to_r16) +	.global SYM(__st_r13_to_r17) +	.global SYM(__st_r13_to_r18) +	.global SYM(__st_r13_to_r19) +	.global SYM(__st_r13_to_r20) +	.global SYM(__st_r13_to_r21) +	.global SYM(__st_r13_to_r22) +	.global SYM(__st_r13_to_r23) +	.global SYM(__st_r13_to_r24) +	.global SYM(__st_r13_to_r25) +	HIDDEN_FUNC(__st_r13_to_r15) +	HIDDEN_FUNC(__st_r13_to_r16) +	HIDDEN_FUNC(__st_r13_to_r17) +	HIDDEN_FUNC(__st_r13_to_r18) +	HIDDEN_FUNC(__st_r13_to_r19) +	HIDDEN_FUNC(__st_r13_to_r20) +	HIDDEN_FUNC(__st_r13_to_r21) +	HIDDEN_FUNC(__st_r13_to_r22) +	HIDDEN_FUNC(__st_r13_to_r23) +	HIDDEN_FUNC(__st_r13_to_r24) +	HIDDEN_FUNC(__st_r13_to_r25) +	.align 4 +SYM(__st_r13_to_r25): +	st r25, [sp,48] +SYM(__st_r13_to_r24): +	st r24, [sp,44] +SYM(__st_r13_to_r23): +	st r23, [sp,40] +SYM(__st_r13_to_r22): +	st r22, [sp,36] +SYM(__st_r13_to_r21): +	st r21, [sp,32] +SYM(__st_r13_to_r20): +	st r20, [sp,28] +SYM(__st_r13_to_r19): +	st r19, [sp,24] +SYM(__st_r13_to_r18): +	st r18, [sp,20] +SYM(__st_r13_to_r17): +	st r17, [sp,16] +SYM(__st_r13_to_r16): +	st r16, [sp,12] +SYM(__st_r13_to_r15): +#ifdef __ARC700__ +	st r15, [sp,8] ; minimum function size to avoid stall: 6 bytes. +#else +	st_s r15, [sp,8] +#endif +	st_s r14, [sp,4] +	j_s.d [%blink] +	st_s r13, [sp,0] +	ENDFUNC(__st_r13_to_r15) +	ENDFUNC(__st_r13_to_r16) +	ENDFUNC(__st_r13_to_r17) +	ENDFUNC(__st_r13_to_r18) +	ENDFUNC(__st_r13_to_r19) +	ENDFUNC(__st_r13_to_r20) +	ENDFUNC(__st_r13_to_r21) +	ENDFUNC(__st_r13_to_r22) +	ENDFUNC(__st_r13_to_r23) +	ENDFUNC(__st_r13_to_r24) +	ENDFUNC(__st_r13_to_r25) + +	.section .text +	.align 4 +;	================================== +;	the loads + +	.global SYM(__ld_r13_to_r15) +	.global SYM(__ld_r13_to_r16) +	.global SYM(__ld_r13_to_r17) +	.global SYM(__ld_r13_to_r18) +	.global SYM(__ld_r13_to_r19) +	.global SYM(__ld_r13_to_r20) +	.global SYM(__ld_r13_to_r21) +	.global SYM(__ld_r13_to_r22) +	.global SYM(__ld_r13_to_r23) +	.global SYM(__ld_r13_to_r24) +	.global SYM(__ld_r13_to_r25) +	HIDDEN_FUNC(__ld_r13_to_r15) +	HIDDEN_FUNC(__ld_r13_to_r16) +	HIDDEN_FUNC(__ld_r13_to_r17) +	HIDDEN_FUNC(__ld_r13_to_r18) +	HIDDEN_FUNC(__ld_r13_to_r19) +	HIDDEN_FUNC(__ld_r13_to_r20) +	HIDDEN_FUNC(__ld_r13_to_r21) +	HIDDEN_FUNC(__ld_r13_to_r22) +	HIDDEN_FUNC(__ld_r13_to_r23) +	HIDDEN_FUNC(__ld_r13_to_r24) +	HIDDEN_FUNC(__ld_r13_to_r25) +SYM(__ld_r13_to_r25): +	ld r25, [sp,48] +SYM(__ld_r13_to_r24): +	ld r24, [sp,44] +SYM(__ld_r13_to_r23): +	ld r23, [sp,40] +SYM(__ld_r13_to_r22): +	ld r22, [sp,36] +SYM(__ld_r13_to_r21): +	ld r21, [sp,32] +SYM(__ld_r13_to_r20): +	ld r20, [sp,28] +SYM(__ld_r13_to_r19): +	ld r19, [sp,24] +SYM(__ld_r13_to_r18): +	ld r18, [sp,20] +SYM(__ld_r13_to_r17): +	ld r17, [sp,16] +SYM(__ld_r13_to_r16): +	ld r16, [sp,12] +SYM(__ld_r13_to_r15): +#ifdef __ARC700__ +	ld r15, [sp,8] ; minimum function size to avoid stall: 6 bytes. +#else +	ld_s r15, [sp,8] +#endif +	ld_s r14, [sp,4] +	j_s.d [%blink] +	ld_s r13, [sp,0] +	ENDFUNC(__ld_r13_to_r15) +	ENDFUNC(__ld_r13_to_r16) +	ENDFUNC(__ld_r13_to_r17) +	ENDFUNC(__ld_r13_to_r18) +	ENDFUNC(__ld_r13_to_r19) +	ENDFUNC(__ld_r13_to_r20) +	ENDFUNC(__ld_r13_to_r21) +	ENDFUNC(__ld_r13_to_r22) +	ENDFUNC(__ld_r13_to_r23) +	ENDFUNC(__ld_r13_to_r24) +	ENDFUNC(__ld_r13_to_r25) + +	.global SYM(__ld_r13_to_r14_ret) +	.global SYM(__ld_r13_to_r15_ret) +	.global SYM(__ld_r13_to_r16_ret) +	.global SYM(__ld_r13_to_r17_ret) +	.global SYM(__ld_r13_to_r18_ret) +	.global SYM(__ld_r13_to_r19_ret) +	.global SYM(__ld_r13_to_r20_ret) +	.global SYM(__ld_r13_to_r21_ret) +	.global SYM(__ld_r13_to_r22_ret) +	.global SYM(__ld_r13_to_r23_ret) +	.global SYM(__ld_r13_to_r24_ret) +	.global SYM(__ld_r13_to_r25_ret) +	HIDDEN_FUNC(__ld_r13_to_r14_ret) +	HIDDEN_FUNC(__ld_r13_to_r15_ret) +	HIDDEN_FUNC(__ld_r13_to_r16_ret) +	HIDDEN_FUNC(__ld_r13_to_r17_ret) +	HIDDEN_FUNC(__ld_r13_to_r18_ret) +	HIDDEN_FUNC(__ld_r13_to_r19_ret) +	HIDDEN_FUNC(__ld_r13_to_r20_ret) +	HIDDEN_FUNC(__ld_r13_to_r21_ret) +	HIDDEN_FUNC(__ld_r13_to_r22_ret) +	HIDDEN_FUNC(__ld_r13_to_r23_ret) +	HIDDEN_FUNC(__ld_r13_to_r24_ret) +	HIDDEN_FUNC(__ld_r13_to_r25_ret) +	.section .text +	.align 4 +SYM(__ld_r13_to_r25_ret): +	ld r25, [sp,48] +SYM(__ld_r13_to_r24_ret): +	ld r24, [sp,44] +SYM(__ld_r13_to_r23_ret): +	ld r23, [sp,40] +SYM(__ld_r13_to_r22_ret): +	ld r22, [sp,36] +SYM(__ld_r13_to_r21_ret): +	ld r21, [sp,32] +SYM(__ld_r13_to_r20_ret): +	ld r20, [sp,28] +SYM(__ld_r13_to_r19_ret): +	ld r19, [sp,24] +SYM(__ld_r13_to_r18_ret): +	ld r18, [sp,20] +SYM(__ld_r13_to_r17_ret): +	ld r17, [sp,16] +SYM(__ld_r13_to_r16_ret): +	ld r16, [sp,12] +SYM(__ld_r13_to_r15_ret): +	ld r15, [sp,8] +SYM(__ld_r13_to_r14_ret): +	ld blink,[sp,r12] +	ld_s r14, [sp,4] +	ld.ab r13, [sp,r12] +	j_s.d [%blink] +	add_s sp,sp,4 +	ENDFUNC(__ld_r13_to_r14_ret) +	ENDFUNC(__ld_r13_to_r15_ret) +	ENDFUNC(__ld_r13_to_r16_ret) +	ENDFUNC(__ld_r13_to_r17_ret) +	ENDFUNC(__ld_r13_to_r18_ret) +	ENDFUNC(__ld_r13_to_r19_ret) +	ENDFUNC(__ld_r13_to_r20_ret) +	ENDFUNC(__ld_r13_to_r21_ret) +	ENDFUNC(__ld_r13_to_r22_ret) +	ENDFUNC(__ld_r13_to_r23_ret) +	ENDFUNC(__ld_r13_to_r24_ret) +	ENDFUNC(__ld_r13_to_r25_ret) diff --git a/arch/arc/lib/cache.c b/arch/arc/lib/cache.c new file mode 100644 index 00000000000..a2277231ba1 --- /dev/null +++ b/arch/arc/lib/cache.c @@ -0,0 +1,168 @@ +/* + * Copyright (C) 2013-2014 Synopsys, Inc. All rights reserved. + * + * SPDX-License-Identifier:	GPL-2.0+ + */ + +#include <config.h> +#include <asm/arcregs.h> +#include <asm/cache.h> + +/* Bit values in IC_CTRL */ +#define IC_CTRL_CACHE_DISABLE	(1 << 0) + +/* Bit values in DC_CTRL */ +#define DC_CTRL_CACHE_DISABLE	(1 << 0) +#define DC_CTRL_INV_MODE_FLUSH	(1 << 6) +#define DC_CTRL_FLUSH_STATUS	(1 << 8) +#define CACHE_VER_NUM_MASK	0xF + +int icache_status(void) +{ +	/* If no cache in CPU exit immediately */ +	if (!(read_aux_reg(ARC_BCR_IC_BUILD) & CACHE_VER_NUM_MASK)) +		return 0; + +	return (read_aux_reg(ARC_AUX_IC_CTRL) & IC_CTRL_CACHE_DISABLE) != +	       IC_CTRL_CACHE_DISABLE; +} + +void icache_enable(void) +{ +	/* If no cache in CPU exit immediately */ +	if (!(read_aux_reg(ARC_BCR_IC_BUILD) & CACHE_VER_NUM_MASK)) +		return; + +	write_aux_reg(ARC_AUX_IC_CTRL, read_aux_reg(ARC_AUX_IC_CTRL) & +		      ~IC_CTRL_CACHE_DISABLE); +} + +void icache_disable(void) +{ +	/* If no cache in CPU exit immediately */ +	if (!(read_aux_reg(ARC_BCR_IC_BUILD) & CACHE_VER_NUM_MASK)) +		return; + +	write_aux_reg(ARC_AUX_IC_CTRL, read_aux_reg(ARC_AUX_IC_CTRL) | +		      IC_CTRL_CACHE_DISABLE); +} + +void invalidate_icache_all(void) +{ +#ifndef CONFIG_SYS_ICACHE_OFF +	/* Any write to IC_IVIC register triggers invalidation of entire I$ */ +	write_aux_reg(ARC_AUX_IC_IVIC, 1); +#endif /* CONFIG_SYS_ICACHE_OFF */ +} + +int dcache_status(void) +{ +	/* If no cache in CPU exit immediately */ +	if (!(read_aux_reg(ARC_BCR_DC_BUILD) & CACHE_VER_NUM_MASK)) +		return 0; + +	return (read_aux_reg(ARC_AUX_DC_CTRL) & DC_CTRL_CACHE_DISABLE) != +		DC_CTRL_CACHE_DISABLE; +} + +void dcache_enable(void) +{ +	/* If no cache in CPU exit immediately */ +	if (!(read_aux_reg(ARC_BCR_DC_BUILD) & CACHE_VER_NUM_MASK)) +		return; + +	write_aux_reg(ARC_AUX_DC_CTRL, read_aux_reg(ARC_AUX_DC_CTRL) & +		      ~(DC_CTRL_INV_MODE_FLUSH | DC_CTRL_CACHE_DISABLE)); +} + +void dcache_disable(void) +{ +	/* If no cache in CPU exit immediately */ +	if (!(read_aux_reg(ARC_BCR_DC_BUILD) & CACHE_VER_NUM_MASK)) +		return; + +	write_aux_reg(ARC_AUX_DC_CTRL, read_aux_reg(ARC_AUX_DC_CTRL) | +		      DC_CTRL_CACHE_DISABLE); +} + +void flush_dcache_all(void) +{ +	/* If no cache in CPU exit immediately */ +	if (!(read_aux_reg(ARC_BCR_DC_BUILD) & CACHE_VER_NUM_MASK)) +		return; + +	/* Do flush of entire cache */ +	write_aux_reg(ARC_AUX_DC_FLSH, 1); + +	/* Wait flush end */ +	while (read_aux_reg(ARC_AUX_DC_CTRL) & DC_CTRL_FLUSH_STATUS) +		; +} + +#ifndef CONFIG_SYS_DCACHE_OFF +static void dcache_flush_line(unsigned addr) +{ +#if (CONFIG_ARC_MMU_VER == 3) +	write_aux_reg(ARC_AUX_DC_PTAG, addr); +#endif +	write_aux_reg(ARC_AUX_DC_FLDL, addr); + +	/* Wait flush end */ +	while (read_aux_reg(ARC_AUX_DC_CTRL) & DC_CTRL_FLUSH_STATUS) +		; + +#ifndef CONFIG_SYS_ICACHE_OFF +	/* +	 * Invalidate I$ for addresses range just flushed from D$. +	 * If we try to execute data flushed above it will be valid/correct +	 */ +#if (CONFIG_ARC_MMU_VER == 3) +	write_aux_reg(ARC_AUX_IC_PTAG, addr); +#endif +	write_aux_reg(ARC_AUX_IC_IVIL, addr); +#endif /* CONFIG_SYS_ICACHE_OFF */ +} +#endif /* CONFIG_SYS_DCACHE_OFF */ + +void flush_dcache_range(unsigned long start, unsigned long end) +{ +#ifndef CONFIG_SYS_DCACHE_OFF +	unsigned int addr; + +	start = start & (~(CONFIG_SYS_CACHELINE_SIZE - 1)); +	end = end & (~(CONFIG_SYS_CACHELINE_SIZE - 1)); + +	for (addr = start; addr <= end; addr += CONFIG_SYS_CACHELINE_SIZE) +		dcache_flush_line(addr); +#endif /* CONFIG_SYS_DCACHE_OFF */ +} + +void invalidate_dcache_range(unsigned long start, unsigned long end) +{ +#ifndef CONFIG_SYS_DCACHE_OFF +	unsigned int addr; + +	start = start & (~(CONFIG_SYS_CACHELINE_SIZE - 1)); +	end = end & (~(CONFIG_SYS_CACHELINE_SIZE - 1)); + +	for (addr = start; addr <= end; addr += CONFIG_SYS_CACHELINE_SIZE) { +#if (CONFIG_ARC_MMU_VER == 3) +		write_aux_reg(ARC_AUX_DC_PTAG, addr); +#endif +		write_aux_reg(ARC_AUX_DC_IVDL, addr); +	} +#endif /* CONFIG_SYS_DCACHE_OFF */ +} + +void invalidate_dcache_all(void) +{ +#ifndef CONFIG_SYS_DCACHE_OFF +	/* Write 1 to DC_IVDC register triggers invalidation of entire D$ */ +	write_aux_reg(ARC_AUX_DC_IVDC, 1); +#endif /* CONFIG_SYS_DCACHE_OFF */ +} + +void flush_cache(unsigned long start, unsigned long size) +{ +	flush_dcache_range(start, start + size); +} diff --git a/arch/arc/lib/cpu.c b/arch/arc/lib/cpu.c new file mode 100644 index 00000000000..50634b860f7 --- /dev/null +++ b/arch/arc/lib/cpu.c @@ -0,0 +1,47 @@ +/* + * Copyright (C) 2013-2014 Synopsys, Inc. All rights reserved. + * + * SPDX-License-Identifier:	GPL-2.0+ + */ + +#include <common.h> +#include <asm/arcregs.h> +#include <asm/cache.h> + +DECLARE_GLOBAL_DATA_PTR; + +int arch_cpu_init(void) +{ +#ifdef CONFIG_SYS_ICACHE_OFF +	icache_disable(); +#else +	icache_enable(); +	invalidate_icache_all(); +#endif + +	flush_dcache_all(); +#ifdef CONFIG_SYS_DCACHE_OFF +	dcache_disable(); +#else +	dcache_enable(); +#endif +	timer_init(); + +/* In simulation (ISS) "CHIPID" and "ARCNUM" are all "ff" */ +	if ((read_aux_reg(ARC_AUX_IDENTITY) & 0xffffff00) == 0xffffff00) +		gd->arch.running_on_hw = 0; +	else +		gd->arch.running_on_hw = 1; + +	gd->cpu_clk = CONFIG_SYS_CLK_FREQ; +	gd->ram_size = CONFIG_SYS_SDRAM_SIZE; + +	return 0; +} + +int arch_early_init_r(void) +{ +	gd->bd->bi_memstart = CONFIG_SYS_SDRAM_BASE; +	gd->bd->bi_memsize = CONFIG_SYS_SDRAM_SIZE; +	return 0; +} diff --git a/arch/arc/lib/interrupts.c b/arch/arc/lib/interrupts.c new file mode 100644 index 00000000000..d7cab3bb409 --- /dev/null +++ b/arch/arc/lib/interrupts.c @@ -0,0 +1,143 @@ +/* + * Copyright (C) 2013-2014 Synopsys, Inc. All rights reserved. + * + * SPDX-License-Identifier:	GPL-2.0+ + */ + +#include <common.h> +#include <asm/arcregs.h> +#include <asm/ptrace.h> + +/* Bit values in STATUS32 */ +#define E1_MASK		(1 << 1)	/* Level 1 interrupts enable */ +#define E2_MASK		(1 << 2)	/* Level 2 interrupts enable */ + +int interrupt_init(void) +{ +	return 0; +} + +/* + * returns true if interrupts had been enabled before we disabled them + */ +int disable_interrupts(void) +{ +	int status = read_aux_reg(ARC_AUX_STATUS32); +	int state = (status & (E1_MASK | E2_MASK)) ? 1 : 0; + +	status &= ~(E1_MASK | E2_MASK); +	/* STATUS32 register is updated indirectly with "FLAG" instruction */ +	__asm__("flag %0" : : "r" (status)); +	return state; +} + +void enable_interrupts(void) +{ +	unsigned int status = read_aux_reg(ARC_AUX_STATUS32); + +	status |= E1_MASK | E2_MASK; +	/* STATUS32 register is updated indirectly with "FLAG" instruction */ +	__asm__("flag %0" : : "r" (status)); +} + +static void print_reg_file(long *reg_rev, int start_num) +{ +	unsigned int i; + +	/* Print 3 registers per line */ +	for (i = start_num; i < start_num + 25; i++) { +		printf("r%02u: 0x%08lx\t", i, (unsigned long)*reg_rev); +		if (((i + 1) % 3) == 0) +			printf("\n"); + +		/* Because pt_regs has registers reversed */ +		reg_rev--; +	} + +	/* Add new-line if none was inserted in the end of loop above */ +	if (((i + 1) % 3) != 0) +		printf("\n"); +} + +void show_regs(struct pt_regs *regs) +{ +	printf("ECR:\t0x%08lx\n", regs->ecr); +	printf("RET:\t0x%08lx\nBLINK:\t0x%08lx\nSTAT32:\t0x%08lx\n", +	       regs->ret, regs->blink, regs->status32); +	printf("GP: 0x%08lx\t r25: 0x%08lx\t\n", regs->r26, regs->r25); +	printf("BTA: 0x%08lx\t SP: 0x%08lx\t FP: 0x%08lx\n", regs->bta, +	       regs->sp, regs->fp); +	printf("LPS: 0x%08lx\tLPE: 0x%08lx\tLPC: 0x%08lx\n", regs->lp_start, +	       regs->lp_end, regs->lp_count); + +	print_reg_file(&(regs->r0), 0); +} + +void bad_mode(struct pt_regs *regs) +{ +	if (regs) +		show_regs(regs); + +	panic("Resetting CPU ...\n"); +} + +void do_memory_error(unsigned long address, struct pt_regs *regs) +{ +	printf("Memory error exception @ 0x%lx\n", address); +	bad_mode(regs); +} + +void do_instruction_error(unsigned long address, struct pt_regs *regs) +{ +	printf("Instruction error exception @ 0x%lx\n", address); +	bad_mode(regs); +} + +void do_machine_check_fault(unsigned long address, struct pt_regs *regs) +{ +	printf("Machine check exception @ 0x%lx\n", address); +	bad_mode(regs); +} + +void do_interrupt_handler(void) +{ +	printf("Interrupt fired\n"); +	bad_mode(0); +} + +void do_itlb_miss(struct pt_regs *regs) +{ +	printf("I TLB miss exception\n"); +	bad_mode(regs); +} + +void do_dtlb_miss(struct pt_regs *regs) +{ +	printf("D TLB miss exception\n"); +	bad_mode(regs); +} + +void do_tlb_prot_violation(unsigned long address, struct pt_regs *regs) +{ +	printf("TLB protection violation or misaligned access @ 0x%lx\n", +	       address); +	bad_mode(regs); +} + +void do_privilege_violation(struct pt_regs *regs) +{ +	printf("Privilege violation exception\n"); +	bad_mode(regs); +} + +void do_trap(struct pt_regs *regs) +{ +	printf("Trap exception\n"); +	bad_mode(regs); +} + +void do_extension(struct pt_regs *regs) +{ +	printf("Extension instruction exception\n"); +	bad_mode(regs); +} diff --git a/arch/arc/lib/libgcc2.c b/arch/arc/lib/libgcc2.c new file mode 100644 index 00000000000..d5ad327f655 --- /dev/null +++ b/arch/arc/lib/libgcc2.c @@ -0,0 +1,161 @@ +/* + * Copyright (C) 1989-2013 Free Software Foundation, Inc. + * + * SPDX-License-Identifier:	GPL-2.0+ + */ + +#include "libgcc2.h" + +DWtype +__ashldi3(DWtype u, shift_count_type b) +{ +	if (b == 0) +		return u; + +	const DWunion uu = {.ll = u}; +	const shift_count_type bm = W_TYPE_SIZE - b; +	DWunion w; + +	if (bm <= 0) { +		w.s.low = 0; +		w.s.high = (UWtype)uu.s.low << -bm; +	} else { +		const UWtype carries = (UWtype) uu.s.low >> bm; + +		w.s.low = (UWtype)uu.s.low << b; +		w.s.high = ((UWtype)uu.s.high << b) | carries; +	} + +	return w.ll; +} + +DWtype +__ashrdi3(DWtype u, shift_count_type b) +{ +	if (b == 0) +		return u; + +	const DWunion uu = {.ll = u}; +	const shift_count_type bm = W_TYPE_SIZE - b; +	DWunion w; + +	if (bm <= 0) { +		/* w.s.high = 1..1 or 0..0 */ +		w.s.high = uu.s.high >> (W_TYPE_SIZE - 1); +		w.s.low = uu.s.high >> -bm; +	} else { +		const UWtype carries = (UWtype) uu.s.high << bm; + +		w.s.high = uu.s.high >> b; +		w.s.low = ((UWtype)uu.s.low >> b) | carries; +	} + +	return w.ll; +} + +DWtype +__lshrdi3(DWtype u, shift_count_type b) +{ +	if (b == 0) +		return u; + +	const DWunion uu = {.ll = u}; +	const shift_count_type bm = W_TYPE_SIZE - b; +	DWunion w; + +	if (bm <= 0) { +		w.s.high = 0; +		w.s.low = (UWtype)uu.s.high >> -bm; +	} else { +		const UWtype carries = (UWtype)uu.s.high << bm; + +		w.s.high = (UWtype)uu.s.high >> b; +		w.s.low = ((UWtype)uu.s.low >> b) | carries; +	} + +	return w.ll; +} + +unsigned long +udivmodsi4(unsigned long num, unsigned long den, int modwanted) +{ +	unsigned long bit = 1; +	unsigned long res = 0; + +	while (den < num && bit && !(den & (1L<<31))) { +		den <<= 1; +		bit <<= 1; +	} + +	while (bit) { +		if (num >= den) { +			num -= den; +			res |= bit; +		} +		bit >>= 1; +		den >>= 1; +	} + +	if (modwanted) +		return num; + +	return res; +} + +long +__divsi3(long a, long b) +{ +	int neg = 0; +	long res; + +	if (a < 0) { +		a = -a; +		neg = !neg; +	} + +	if (b < 0) { +		b = -b; +		neg = !neg; +	} + +	res = udivmodsi4(a, b, 0); + +	if (neg) +		res = -res; + +	return res; +} + +long +__modsi3(long a, long b) +{ +	int neg = 0; +	long res; + +	if (a < 0) { +		a = -a; +		neg = 1; +	} + +	if (b < 0) +		b = -b; + +	res = udivmodsi4(a, b, 1); + +	if (neg) +		res = -res; + +	return res; +} + +long +__udivsi3(long a, long b) +{ +	return udivmodsi4(a, b, 0); +} + +long +__umodsi3(long a, long b) +{ +	return udivmodsi4(a, b, 1); +} diff --git a/arch/arc/lib/libgcc2.h b/arch/arc/lib/libgcc2.h new file mode 100644 index 00000000000..8813c3bf4ca --- /dev/null +++ b/arch/arc/lib/libgcc2.h @@ -0,0 +1,132 @@ +/* + * Copyright (C) 1989-2013 Free Software Foundation, Inc. + * + * SPDX-License-Identifier:	GPL-2.0+ + */ + +#ifndef __ASM_LIBGCC_H +#define __ASM_LIBGCC_H + +#define UNITS_PER_WORD 4	/* for ARC */ +#define BITS_PER_UNIT 8		/* for ARC */ + +#define W_TYPE_SIZE (4 * BITS_PER_UNIT) + +#define MIN_UNITS_PER_WORD UNITS_PER_WORD + +/* Work out the largest "word" size that we can deal with on this target.  */ +#if MIN_UNITS_PER_WORD > 4 +# define LIBGCC2_MAX_UNITS_PER_WORD 8 +#elif (MIN_UNITS_PER_WORD > 2 \ +       || (MIN_UNITS_PER_WORD > 1 && __SIZEOF_LONG_LONG__ > 4)) +# define LIBGCC2_MAX_UNITS_PER_WORD 4 +#else +# define LIBGCC2_MAX_UNITS_PER_WORD MIN_UNITS_PER_WORD +#endif + +/* Work out what word size we are using for this compilation. +   The value can be set on the command line.  */ +#ifndef LIBGCC2_UNITS_PER_WORD +#define LIBGCC2_UNITS_PER_WORD LIBGCC2_MAX_UNITS_PER_WORD +#endif + +typedef		 int QItype	__attribute__ ((mode (QI))); +typedef unsigned int UQItype	__attribute__ ((mode (QI))); +typedef		 int HItype	__attribute__ ((mode (HI))); +typedef unsigned int UHItype	__attribute__ ((mode (HI))); +#if MIN_UNITS_PER_WORD > 1 +/* These typedefs are usually forbidden on dsp's with UNITS_PER_WORD 1.  */ +typedef 	 int SItype	__attribute__ ((mode (SI))); +typedef unsigned int USItype	__attribute__ ((mode (SI))); +#if __SIZEOF_LONG_LONG__ > 4 +/* These typedefs are usually forbidden on archs with UNITS_PER_WORD 2.  */ +typedef		 int DItype	__attribute__ ((mode (DI))); +typedef unsigned int UDItype	__attribute__ ((mode (DI))); +#if MIN_UNITS_PER_WORD > 4 +/* These typedefs are usually forbidden on archs with UNITS_PER_WORD 4.  */ +typedef		 int TItype	__attribute__ ((mode (TI))); +typedef unsigned int UTItype	__attribute__ ((mode (TI))); +#endif +#endif +#endif + +#if LIBGCC2_UNITS_PER_WORD == 8 +#define W_TYPE_SIZE (8 * BITS_PER_UNIT) +#define Wtype	DItype +#define UWtype	UDItype +#define HWtype	DItype +#define UHWtype	UDItype +#define DWtype	TItype +#define UDWtype	UTItype +#ifdef LIBGCC2_GNU_PREFIX +#define __NW(a,b)	__gnu_ ## a ## di ## b +#define __NDW(a,b)	__gnu_ ## a ## ti ## b +#else +#define __NW(a,b)	__ ## a ## di ## b +#define __NDW(a,b)	__ ## a ## ti ## b +#endif +#elif LIBGCC2_UNITS_PER_WORD == 4 +#define W_TYPE_SIZE (4 * BITS_PER_UNIT) +#define Wtype	SItype +#define UWtype	USItype +#define HWtype	SItype +#define UHWtype	USItype +#define DWtype	DItype +#define UDWtype	UDItype +#ifdef LIBGCC2_GNU_PREFIX +#define __NW(a,b)	__gnu_ ## a ## si ## b +#define __NDW(a,b)	__gnu_ ## a ## di ## b +#else +#define __NW(a,b)	__ ## a ## si ## b +#define __NDW(a,b)	__ ## a ## di ## b +#endif +#elif LIBGCC2_UNITS_PER_WORD == 2 +#define W_TYPE_SIZE (2 * BITS_PER_UNIT) +#define Wtype	HItype +#define UWtype	UHItype +#define HWtype	HItype +#define UHWtype	UHItype +#define DWtype	SItype +#define UDWtype	USItype +#ifdef LIBGCC2_GNU_PREFIX +#define __NW(a,b)	__gnu_ ## a ## hi ## b +#define __NDW(a,b)	__gnu_ ## a ## si ## b +#else +#define __NW(a,b)	__ ## a ## hi ## b +#define __NDW(a,b)	__ ## a ## si ## b +#endif +#else +#define W_TYPE_SIZE BITS_PER_UNIT +#define Wtype	QItype +#define UWtype  UQItype +#define HWtype	QItype +#define UHWtype	UQItype +#define DWtype	HItype +#define UDWtype	UHItype +#ifdef LIBGCC2_GNU_PREFIX +#define __NW(a,b)	__gnu_ ## a ## qi ## b +#define __NDW(a,b)	__gnu_ ## a ## hi ## b +#else +#define __NW(a,b)	__ ## a ## qi ## b +#define __NDW(a,b)	__ ## a ## hi ## b +#endif +#endif + +typedef int shift_count_type __attribute__((mode (__libgcc_shift_count__))); + +#if __BYTE_ORDER__ != __ORDER_LITTLE_ENDIAN__ +	struct DWstruct {Wtype high, low;}; +#else +	struct DWstruct {Wtype low, high;}; +#endif + +/* We need this union to unpack/pack DImode values, since we don't have +   any arithmetic yet.  Incoming DImode parameters are stored into the +   `ll' field, and the unpacked result is read from the struct `s'.  */ + +typedef union { +	struct DWstruct s; +	DWtype ll; +} DWunion; + +#endif /* __ASM_LIBGCC_H */ diff --git a/arch/arc/lib/memcmp.S b/arch/arc/lib/memcmp.S index fa5aac5f674..87bccab51d7 100644 --- a/arch/arc/lib/memcmp.S +++ b/arch/arc/lib/memcmp.S @@ -29,6 +29,7 @@ memcmp:  	ld.a	%r4, [%r0, 8]  	ld.a	%r5, [%r1, 8]  	brne	WORD2, %r12, .Lodd +	nop  .Loop_end:  	asl_s	SHIFT, SHIFT, 3  	bhs_s	.Last_cmp @@ -105,6 +106,7 @@ memcmp:  	ldb.a	%r4, [%r0, 2]  	ldb.a	%r5, [%r1, 2]  	brne	%r3, %r12, .Lbyte_odd +	nop  .Lbyte_end:  	bcc	.Lbyte_even  	brne	%r4, %r5, .Lbyte_even diff --git a/arch/arc/lib/relocate.c b/arch/arc/lib/relocate.c index 2482bcdffcc..7797782563b 100644 --- a/arch/arc/lib/relocate.c +++ b/arch/arc/lib/relocate.c @@ -26,7 +26,7 @@ int do_elf_reloc_fixups(void)  		offset_ptr_rom = (Elf32_Addr *)re_src->r_offset;  		/* Check that the location of the relocation is in .text */ -		if (offset_ptr_rom >= (Elf32_Addr *)CONFIG_SYS_TEXT_BASE && +		if (offset_ptr_rom >= (Elf32_Addr *)&__image_copy_start &&  		    offset_ptr_rom > last_offset) {  			unsigned int val;  			/* Switch to the in-RAM version */ @@ -44,29 +44,22 @@ int do_elf_reloc_fixups(void)  #ifdef __LITTLE_ENDIAN__  			/* If location in ".text" section swap value */  			if ((unsigned int)offset_ptr_rom < -			    (unsigned int)&__text_end) +			    (unsigned int)&__ivt_end)  				val = (val << 16) | (val >> 16);  #endif -			/* Check that the target points into .text */ -			if (val >= CONFIG_SYS_TEXT_BASE && val <= -			    (unsigned int)&__bss_end) { +			/* Check that the target points into executable */ +			if (val >= (unsigned int)&__image_copy_start && val <= +			    (unsigned int)&__image_copy_end) {  				val += gd->reloc_off;  #ifdef __LITTLE_ENDIAN__  				/* If location in ".text" section swap value */  				if ((unsigned int)offset_ptr_rom < -				    (unsigned int)&__text_end) +				    (unsigned int)&__ivt_end)  					val = (val << 16) | (val >> 16);  #endif  				memcpy(offset_ptr_ram, &val, sizeof(int)); -			} else { -				debug("   %p: rom reloc %x, ram %p, value %x, limit %x\n", -				      re_src, re_src->r_offset, offset_ptr_ram, -				      val, (unsigned int)&__bss_end);  			} -		} else { -			debug("   %p: rom reloc %x, last %p\n", re_src, -			      re_src->r_offset, last_offset);  		}  		last_offset = offset_ptr_rom; diff --git a/arch/arc/lib/reset.c b/arch/arc/lib/reset.c new file mode 100644 index 00000000000..98ebf1d4456 --- /dev/null +++ b/arch/arc/lib/reset.c @@ -0,0 +1,19 @@ +/* + * Copyright (C) 2013-2014 Synopsys, Inc. All rights reserved. + * + * SPDX-License-Identifier:	GPL-2.0+ + */ + +#include <command.h> +#include <common.h> + +int do_reset(cmd_tbl_t *cmdtp, int flag, int argc, char *const argv[]) +{ +	printf("Put your restart handler here\n"); + +#ifdef DEBUG +	/* Stop debug session here */ +	__asm__("brk"); +#endif +	return 0; +} diff --git a/arch/arc/lib/sections.c b/arch/arc/lib/sections.c index b0b46a4e9ae..a72c6946d53 100644 --- a/arch/arc/lib/sections.c +++ b/arch/arc/lib/sections.c @@ -19,3 +19,5 @@ char __rel_dyn_end[0] __attribute__((section(".__rel_dyn_end")));  char __text_start[0] __attribute__((section(".__text_start")));  char __text_end[0] __attribute__((section(".__text_end")));  char __init_end[0] __attribute__((section(".__init_end"))); +char __ivt_start[0] __attribute__((section(".__ivt_start"))); +char __ivt_end[0] __attribute__((section(".__ivt_end"))); diff --git a/arch/arc/lib/timer.c b/arch/arc/lib/timer.c new file mode 100644 index 00000000000..a0acbbc01a4 --- /dev/null +++ b/arch/arc/lib/timer.c @@ -0,0 +1,24 @@ +/* + * Copyright (C) 2013-2014 Synopsys, Inc. All rights reserved. + * + * SPDX-License-Identifier:	GPL-2.0+ + */ + +#include <asm/arcregs.h> + +#define NH_MODE	(1 << 1)	/* Disable timer if CPU is halted */ + +int timer_init(void) +{ +	write_aux_reg(ARC_AUX_TIMER0_CTRL, NH_MODE); +	/* Set max value for counter/timer */ +	write_aux_reg(ARC_AUX_TIMER0_LIMIT, 0xffffffff); +	/* Set initial count value and restart counter/timer */ +	write_aux_reg(ARC_AUX_TIMER0_CNT, 0); +	return 0; +} + +unsigned long timer_read_counter(void) +{ +	return read_aux_reg(ARC_AUX_TIMER0_CNT); +} | 
