Commit 8d8026f3 authored by Linus Torvalds's avatar Linus Torvalds

Merge tag 'xtensa-20230627' of https://github.com/jcmvbkbc/linux-xtensa

Pull xtensa updates from Max Filippov:

 - clean up platform_* interface of the xtensa architecture

 - enable HAVE_ASM_MODVERSIONS

 - drop ARCH_WANT_FRAME_POINTERS

 - clean up unaligned access exception handler

 - provide handler for load/store exceptions

 - various small fixes and cleanups

* tag 'xtensa-20230627' of https://github.com/jcmvbkbc/linux-xtensa:
  xtensa: dump userspace code around the exception PC
  xtensa: rearrange show_stack output
  xtensa: add load/store exception handler
  xtensa: rearrange unaligned exception handler
  xtensa: always install slow handler for unaligned access exception
  xtensa: move early_trap_init from kasan_early_init to init_arch
  xtensa: drop ARCH_WANT_FRAME_POINTERS
  xtensa: report trax and perf counters in cpuinfo
  xtensa: add asm-prototypes.h
  xtensa: only build __strncpy_user with CONFIG_ARCH_HAS_STRNCPY_FROM_USER
  xtensa: drop bcopy implementation
  xtensa: drop EXPORT_SYMBOL for common_exception_return
  xtensa: boot-redboot: clean up Makefile
  xtensa: clean up default platform functions
  xtensa: drop platform_halt and platform_power_off
  xtensa: drop platform_restart
  xtensa: drop platform_heartbeat
  xtensa: xt2000: drop empty platform_init
parents 0ae61229 f7667ca1
...@@ -16,7 +16,6 @@ config XTENSA ...@@ -16,7 +16,6 @@ config XTENSA
select ARCH_USE_MEMTEST select ARCH_USE_MEMTEST
select ARCH_USE_QUEUED_RWLOCKS select ARCH_USE_QUEUED_RWLOCKS
select ARCH_USE_QUEUED_SPINLOCKS select ARCH_USE_QUEUED_SPINLOCKS
select ARCH_WANT_FRAME_POINTERS
select ARCH_WANT_IPC_PARSE_VERSION select ARCH_WANT_IPC_PARSE_VERSION
select BUILDTIME_TABLE_SORT select BUILDTIME_TABLE_SORT
select CLONE_BACKWARDS select CLONE_BACKWARDS
...@@ -35,6 +34,7 @@ config XTENSA ...@@ -35,6 +34,7 @@ config XTENSA
select HAVE_ARCH_KCSAN select HAVE_ARCH_KCSAN
select HAVE_ARCH_SECCOMP_FILTER select HAVE_ARCH_SECCOMP_FILTER
select HAVE_ARCH_TRACEHOOK select HAVE_ARCH_TRACEHOOK
select HAVE_ASM_MODVERSIONS
select HAVE_CONTEXT_TRACKING_USER select HAVE_CONTEXT_TRACKING_USER
select HAVE_DEBUG_KMEMLEAK select HAVE_DEBUG_KMEMLEAK
select HAVE_DMA_CONTIGUOUS select HAVE_DMA_CONTIGUOUS
...@@ -203,6 +203,18 @@ config XTENSA_UNALIGNED_USER ...@@ -203,6 +203,18 @@ config XTENSA_UNALIGNED_USER
Say Y here to enable unaligned memory access in user space. Say Y here to enable unaligned memory access in user space.
config XTENSA_LOAD_STORE
bool "Load/store exception handler for memory only readable with l32"
help
The Xtensa architecture only allows reading memory attached to its
instruction bus with l32r and l32i instructions, all other
instructions raise an exception with the LoadStoreErrorCause code.
This makes it hard to use some configurations, e.g. store string
literals in FLASH memory attached to the instruction bus.
Say Y here to enable exception handler that allows transparent
byte and 2-byte access to memory attached to instruction bus.
config HAVE_SMP config HAVE_SMP
bool "System Supports SMP (MX)" bool "System Supports SMP (MX)"
depends on XTENSA_VARIANT_CUSTOM depends on XTENSA_VARIANT_CUSTOM
......
...@@ -38,3 +38,11 @@ config PRINT_STACK_DEPTH ...@@ -38,3 +38,11 @@ config PRINT_STACK_DEPTH
help help
This option allows you to set the stack depth that the kernel This option allows you to set the stack depth that the kernel
prints in stack traces. prints in stack traces.
config PRINT_USER_CODE_ON_UNHANDLED_EXCEPTION
bool "Dump user code around unhandled exception address"
help
Enable this option to display user code around PC of the unhandled
exception (starting at address aligned on 16 byte boundary).
This may simplify finding faulting code in the absence of other
debug facilities.
...@@ -6,16 +6,12 @@ ...@@ -6,16 +6,12 @@
OBJCOPY_ARGS := -O $(if $(CONFIG_CPU_BIG_ENDIAN),elf32-xtensa-be,elf32-xtensa-le) OBJCOPY_ARGS := -O $(if $(CONFIG_CPU_BIG_ENDIAN),elf32-xtensa-be,elf32-xtensa-le)
LD_ARGS = -T $(srctree)/$(obj)/boot.ld
boot-y := bootstrap.o boot-y := bootstrap.o
targets += $(boot-y) targets += $(boot-y)
OBJS := $(addprefix $(obj)/,$(boot-y)) OBJS := $(addprefix $(obj)/,$(boot-y))
LIBS := arch/xtensa/boot/lib/lib.a arch/xtensa/lib/lib.a LIBS := arch/xtensa/boot/lib/lib.a arch/xtensa/lib/lib.a
LIBGCC := $(shell $(CC) $(KBUILD_CFLAGS) -print-libgcc-file-name)
$(obj)/zImage.o: $(obj)/../vmlinux.bin.gz $(OBJS) $(obj)/zImage.o: $(obj)/../vmlinux.bin.gz $(OBJS)
$(Q)$(OBJCOPY) $(OBJCOPY_ARGS) -R .comment \ $(Q)$(OBJCOPY) $(OBJCOPY_ARGS) -R .comment \
--add-section image=$< \ --add-section image=$< \
...@@ -23,7 +19,10 @@ $(obj)/zImage.o: $(obj)/../vmlinux.bin.gz $(OBJS) ...@@ -23,7 +19,10 @@ $(obj)/zImage.o: $(obj)/../vmlinux.bin.gz $(OBJS)
$(OBJS) $@ $(OBJS) $@
$(obj)/zImage.elf: $(obj)/zImage.o $(LIBS) $(obj)/zImage.elf: $(obj)/zImage.o $(LIBS)
$(Q)$(LD) $(LD_ARGS) -o $@ $^ -L/xtensa-elf/lib $(LIBGCC) $(Q)$(LD) $(KBUILD_LDFLAGS) \
-T $(srctree)/$(obj)/boot.ld \
--build-id=none \
-o $@ $^
$(obj)/../zImage.redboot: $(obj)/zImage.elf $(obj)/../zImage.redboot: $(obj)/zImage.elf
$(Q)$(OBJCOPY) -S -O binary $< $@ $(Q)$(OBJCOPY) -S -O binary $< $@
......
/* SPDX-License-Identifier: GPL-2.0 */
#ifndef __ASM_PROTOTYPES_H
#define __ASM_PROTOTYPES_H
#include <asm/cacheflush.h>
#include <asm/checksum.h>
#include <asm/ftrace.h>
#include <asm/page.h>
#include <asm/string.h>
#include <asm/uaccess.h>
#include <asm-generic/asm-prototypes.h>
/*
* gcc internal math functions
*/
long long __ashrdi3(long long, int);
long long __ashldi3(long long, int);
long long __bswapdi2(long long);
int __bswapsi2(int);
long long __lshrdi3(long long, int);
int __divsi3(int, int);
int __modsi3(int, int);
int __mulsi3(int, int);
unsigned int __udivsi3(unsigned int, unsigned int);
unsigned int __umodsi3(unsigned int, unsigned int);
unsigned long long __umulsidi3(unsigned int, unsigned int);
#endif /* __ASM_PROTOTYPES_H */
...@@ -11,6 +11,7 @@ ...@@ -11,6 +11,7 @@
#ifndef _XTENSA_ASMMACRO_H #ifndef _XTENSA_ASMMACRO_H
#define _XTENSA_ASMMACRO_H #define _XTENSA_ASMMACRO_H
#include <asm-generic/export.h>
#include <asm/core.h> #include <asm/core.h>
/* /*
......
...@@ -26,6 +26,14 @@ ...@@ -26,6 +26,14 @@
#define XCHAL_SPANNING_WAY 0 #define XCHAL_SPANNING_WAY 0
#endif #endif
#ifndef XCHAL_HAVE_TRAX
#define XCHAL_HAVE_TRAX 0
#endif
#ifndef XCHAL_NUM_PERF_COUNTERS
#define XCHAL_NUM_PERF_COUNTERS 0
#endif
#if XCHAL_HAVE_WINDOWED #if XCHAL_HAVE_WINDOWED
#if defined(CONFIG_USER_ABI_DEFAULT) || defined(CONFIG_USER_ABI_CALL0_PROBE) #if defined(CONFIG_USER_ABI_DEFAULT) || defined(CONFIG_USER_ABI_CALL0_PROBE)
/* Whether windowed ABI is supported in userspace. */ /* Whether windowed ABI is supported in userspace. */
......
...@@ -13,17 +13,8 @@ ...@@ -13,17 +13,8 @@
#include <asm/processor.h> #include <asm/processor.h>
#ifndef __ASSEMBLY__ #ifndef __ASSEMBLY__
#define ftrace_return_address0 ({ unsigned long a0, a1; \
__asm__ __volatile__ ( \
"mov %0, a0\n" \
"mov %1, a1\n" \
: "=r"(a0), "=r"(a1)); \
MAKE_PC_FROM_RA(a0, a1); })
#ifdef CONFIG_FRAME_POINTER
extern unsigned long return_address(unsigned level); extern unsigned long return_address(unsigned level);
#define ftrace_return_address(n) return_address(n) #define ftrace_return_address(n) return_address(n)
#endif
#endif /* __ASSEMBLY__ */ #endif /* __ASSEMBLY__ */
#ifdef CONFIG_FUNCTION_TRACER #ifdef CONFIG_FUNCTION_TRACER
......
...@@ -27,31 +27,11 @@ extern void platform_init(bp_tag_t*); ...@@ -27,31 +27,11 @@ extern void platform_init(bp_tag_t*);
*/ */
extern void platform_setup (char **); extern void platform_setup (char **);
/*
* platform_restart is called to restart the system.
*/
extern void platform_restart (void);
/*
* platform_halt is called to stop the system and halt.
*/
extern void platform_halt (void);
/*
* platform_power_off is called to stop the system and power it off.
*/
extern void platform_power_off (void);
/* /*
* platform_idle is called from the idle function. * platform_idle is called from the idle function.
*/ */
extern void platform_idle (void); extern void platform_idle (void);
/*
* platform_heartbeat is called every HZ
*/
extern void platform_heartbeat (void);
/* /*
* platform_calibrate_ccount calibrates cpu clock freq (CONFIG_XTENSA_CALIBRATE) * platform_calibrate_ccount calibrates cpu clock freq (CONFIG_XTENSA_CALIBRATE)
*/ */
......
...@@ -118,9 +118,6 @@ extern void *__memcpy(void *__to, __const__ void *__from, size_t __n); ...@@ -118,9 +118,6 @@ extern void *__memcpy(void *__to, __const__ void *__from, size_t __n);
extern void *memmove(void *__dest, __const__ void *__src, size_t __n); extern void *memmove(void *__dest, __const__ void *__src, size_t __n);
extern void *__memmove(void *__dest, __const__ void *__src, size_t __n); extern void *__memmove(void *__dest, __const__ void *__src, size_t __n);
/* Don't build bcopy at all ... */
#define __HAVE_ARCH_BCOPY
#if defined(CONFIG_KASAN) && !defined(__SANITIZE_ADDRESS__) #if defined(CONFIG_KASAN) && !defined(__SANITIZE_ADDRESS__)
/* /*
......
...@@ -47,6 +47,7 @@ __init trap_set_handler(int cause, xtensa_exception_handler *handler); ...@@ -47,6 +47,7 @@ __init trap_set_handler(int cause, xtensa_exception_handler *handler);
asmlinkage void fast_illegal_instruction_user(void); asmlinkage void fast_illegal_instruction_user(void);
asmlinkage void fast_syscall_user(void); asmlinkage void fast_syscall_user(void);
asmlinkage void fast_alloca(void); asmlinkage void fast_alloca(void);
asmlinkage void fast_load_store(void);
asmlinkage void fast_unaligned(void); asmlinkage void fast_unaligned(void);
asmlinkage void fast_second_level_miss(void); asmlinkage void fast_second_level_miss(void);
asmlinkage void fast_store_prohibited(void); asmlinkage void fast_store_prohibited(void);
...@@ -64,8 +65,14 @@ void do_unhandled(struct pt_regs *regs); ...@@ -64,8 +65,14 @@ void do_unhandled(struct pt_regs *regs);
static inline void __init early_trap_init(void) static inline void __init early_trap_init(void)
{ {
static struct exc_table init_exc_table __initdata = { static struct exc_table init_exc_table __initdata = {
#ifdef CONFIG_XTENSA_LOAD_STORE
.fast_kernel_handler[EXCCAUSE_LOAD_STORE_ERROR] =
fast_load_store,
#endif
#ifdef CONFIG_MMU
.fast_kernel_handler[EXCCAUSE_DTLB_MISS] = .fast_kernel_handler[EXCCAUSE_DTLB_MISS] =
fast_second_level_miss, fast_second_level_miss,
#endif
}; };
xtensa_set_sr(&init_exc_table, excsave1); xtensa_set_sr(&init_exc_table, excsave1);
} }
......
This diff is collapsed.
...@@ -78,6 +78,7 @@ ENTRY(_mcount) ...@@ -78,6 +78,7 @@ ENTRY(_mcount)
#error Unsupported Xtensa ABI #error Unsupported Xtensa ABI
#endif #endif
ENDPROC(_mcount) ENDPROC(_mcount)
EXPORT_SYMBOL(_mcount)
ENTRY(ftrace_stub) ENTRY(ftrace_stub)
abi_entry_default abi_entry_default
......
...@@ -17,27 +17,28 @@ ...@@ -17,27 +17,28 @@
#include <asm/platform.h> #include <asm/platform.h>
#include <asm/timex.h> #include <asm/timex.h>
#define _F(r,f,a,b) \
r __platform_##f a b; \
r platform_##f a __attribute__((weak, alias("__platform_"#f)))
/* /*
* Default functions that are used if no platform specific function is defined. * Default functions that are used if no platform specific function is defined.
* (Please, refer to include/asm-xtensa/platform.h for more information) * (Please, refer to arch/xtensa/include/asm/platform.h for more information)
*/ */
_F(void, init, (bp_tag_t *first), { }); void __weak __init platform_init(bp_tag_t *first)
_F(void, setup, (char** cmd), { }); {
_F(void, restart, (void), { while(1); }); }
_F(void, halt, (void), { while(1); });
_F(void, power_off, (void), { while(1); }); void __weak __init platform_setup(char **cmd)
_F(void, idle, (void), { __asm__ __volatile__ ("waiti 0" ::: "memory"); }); {
_F(void, heartbeat, (void), { }); }
void __weak platform_idle(void)
{
__asm__ __volatile__ ("waiti 0" ::: "memory");
}
#ifdef CONFIG_XTENSA_CALIBRATE_CCOUNT #ifdef CONFIG_XTENSA_CALIBRATE_CCOUNT
_F(void, calibrate_ccount, (void), void __weak platform_calibrate_ccount(void)
{ {
pr_err("ERROR: Cannot calibrate cpu frequency! Assuming 10MHz.\n"); pr_err("ERROR: Cannot calibrate cpu frequency! Assuming 10MHz.\n");
ccount_freq = 10 * 1000000UL; ccount_freq = 10 * 1000000UL;
}); }
#endif #endif
...@@ -22,6 +22,7 @@ ...@@ -22,6 +22,7 @@
#include <linux/screen_info.h> #include <linux/screen_info.h>
#include <linux/kernel.h> #include <linux/kernel.h>
#include <linux/percpu.h> #include <linux/percpu.h>
#include <linux/reboot.h>
#include <linux/cpu.h> #include <linux/cpu.h>
#include <linux/of.h> #include <linux/of.h>
#include <linux/of_fdt.h> #include <linux/of_fdt.h>
...@@ -46,6 +47,7 @@ ...@@ -46,6 +47,7 @@
#include <asm/smp.h> #include <asm/smp.h>
#include <asm/sysmem.h> #include <asm/sysmem.h>
#include <asm/timex.h> #include <asm/timex.h>
#include <asm/traps.h>
#if defined(CONFIG_VGA_CONSOLE) || defined(CONFIG_DUMMY_CONSOLE) #if defined(CONFIG_VGA_CONSOLE) || defined(CONFIG_DUMMY_CONSOLE)
struct screen_info screen_info = { struct screen_info screen_info = {
...@@ -241,6 +243,12 @@ void __init early_init_devtree(void *params) ...@@ -241,6 +243,12 @@ void __init early_init_devtree(void *params)
void __init init_arch(bp_tag_t *bp_start) void __init init_arch(bp_tag_t *bp_start)
{ {
/* Initialize basic exception handling if configuration may need it */
if (IS_ENABLED(CONFIG_KASAN) ||
IS_ENABLED(CONFIG_XTENSA_LOAD_STORE))
early_trap_init();
/* Initialize MMU. */ /* Initialize MMU. */
init_mmu(); init_mmu();
...@@ -522,19 +530,30 @@ void cpu_reset(void) ...@@ -522,19 +530,30 @@ void cpu_reset(void)
void machine_restart(char * cmd) void machine_restart(char * cmd)
{ {
platform_restart(); local_irq_disable();
smp_send_stop();
do_kernel_restart(cmd);
pr_err("Reboot failed -- System halted\n");
while (1)
cpu_relax();
} }
void machine_halt(void) void machine_halt(void)
{ {
platform_halt(); local_irq_disable();
while (1); smp_send_stop();
do_kernel_power_off();
while (1)
cpu_relax();
} }
void machine_power_off(void) void machine_power_off(void)
{ {
platform_power_off(); local_irq_disable();
while (1); smp_send_stop();
do_kernel_power_off();
while (1)
cpu_relax();
} }
#ifdef CONFIG_PROC_FS #ifdef CONFIG_PROC_FS
...@@ -574,6 +593,12 @@ c_show(struct seq_file *f, void *slot) ...@@ -574,6 +593,12 @@ c_show(struct seq_file *f, void *slot)
# if XCHAL_HAVE_OCD # if XCHAL_HAVE_OCD
"ocd " "ocd "
# endif # endif
#if XCHAL_HAVE_TRAX
"trax "
#endif
#if XCHAL_NUM_PERF_COUNTERS
"perf "
#endif
#endif #endif
#if XCHAL_HAVE_DENSITY #if XCHAL_HAVE_DENSITY
"density " "density "
...@@ -623,11 +648,13 @@ c_show(struct seq_file *f, void *slot) ...@@ -623,11 +648,13 @@ c_show(struct seq_file *f, void *slot)
seq_printf(f,"physical aregs\t: %d\n" seq_printf(f,"physical aregs\t: %d\n"
"misc regs\t: %d\n" "misc regs\t: %d\n"
"ibreak\t\t: %d\n" "ibreak\t\t: %d\n"
"dbreak\t\t: %d\n", "dbreak\t\t: %d\n"
"perf counters\t: %d\n",
XCHAL_NUM_AREGS, XCHAL_NUM_AREGS,
XCHAL_NUM_MISC_REGS, XCHAL_NUM_MISC_REGS,
XCHAL_NUM_IBREAK, XCHAL_NUM_IBREAK,
XCHAL_NUM_DBREAK); XCHAL_NUM_DBREAK,
XCHAL_NUM_PERF_COUNTERS);
/* Interrupt. */ /* Interrupt. */
......
...@@ -237,8 +237,6 @@ EXPORT_SYMBOL_GPL(save_stack_trace); ...@@ -237,8 +237,6 @@ EXPORT_SYMBOL_GPL(save_stack_trace);
#endif #endif
#ifdef CONFIG_FRAME_POINTER
struct return_addr_data { struct return_addr_data {
unsigned long addr; unsigned long addr;
unsigned skip; unsigned skip;
...@@ -271,5 +269,3 @@ unsigned long return_address(unsigned level) ...@@ -271,5 +269,3 @@ unsigned long return_address(unsigned level)
return r.addr; return r.addr;
} }
EXPORT_SYMBOL(return_address); EXPORT_SYMBOL(return_address);
#endif
...@@ -121,10 +121,6 @@ static irqreturn_t timer_interrupt(int irq, void *dev_id) ...@@ -121,10 +121,6 @@ static irqreturn_t timer_interrupt(int irq, void *dev_id)
set_linux_timer(get_linux_timer()); set_linux_timer(get_linux_timer());
evt->event_handler(evt); evt->event_handler(evt);
/* Allow platform to do something useful (Wdog). */
platform_heartbeat();
return IRQ_HANDLED; return IRQ_HANDLED;
} }
......
...@@ -54,9 +54,10 @@ static void do_interrupt(struct pt_regs *regs); ...@@ -54,9 +54,10 @@ static void do_interrupt(struct pt_regs *regs);
#if XTENSA_FAKE_NMI #if XTENSA_FAKE_NMI
static void do_nmi(struct pt_regs *regs); static void do_nmi(struct pt_regs *regs);
#endif #endif
#if XCHAL_UNALIGNED_LOAD_EXCEPTION || XCHAL_UNALIGNED_STORE_EXCEPTION #ifdef CONFIG_XTENSA_LOAD_STORE
static void do_unaligned_user(struct pt_regs *regs); static void do_load_store(struct pt_regs *regs);
#endif #endif
static void do_unaligned_user(struct pt_regs *regs);
static void do_multihit(struct pt_regs *regs); static void do_multihit(struct pt_regs *regs);
#if XTENSA_HAVE_COPROCESSORS #if XTENSA_HAVE_COPROCESSORS
static void do_coprocessor(struct pt_regs *regs); static void do_coprocessor(struct pt_regs *regs);
...@@ -91,7 +92,10 @@ static dispatch_init_table_t __initdata dispatch_init_table[] = { ...@@ -91,7 +92,10 @@ static dispatch_init_table_t __initdata dispatch_init_table[] = {
{ EXCCAUSE_SYSTEM_CALL, USER, fast_syscall_user }, { EXCCAUSE_SYSTEM_CALL, USER, fast_syscall_user },
{ EXCCAUSE_SYSTEM_CALL, 0, system_call }, { EXCCAUSE_SYSTEM_CALL, 0, system_call },
/* EXCCAUSE_INSTRUCTION_FETCH unhandled */ /* EXCCAUSE_INSTRUCTION_FETCH unhandled */
/* EXCCAUSE_LOAD_STORE_ERROR unhandled*/ #ifdef CONFIG_XTENSA_LOAD_STORE
{ EXCCAUSE_LOAD_STORE_ERROR, USER|KRNL, fast_load_store },
{ EXCCAUSE_LOAD_STORE_ERROR, 0, do_load_store },
#endif
{ EXCCAUSE_LEVEL1_INTERRUPT, 0, do_interrupt }, { EXCCAUSE_LEVEL1_INTERRUPT, 0, do_interrupt },
#ifdef SUPPORT_WINDOWED #ifdef SUPPORT_WINDOWED
{ EXCCAUSE_ALLOCA, USER|KRNL, fast_alloca }, { EXCCAUSE_ALLOCA, USER|KRNL, fast_alloca },
...@@ -102,9 +106,9 @@ static dispatch_init_table_t __initdata dispatch_init_table[] = { ...@@ -102,9 +106,9 @@ static dispatch_init_table_t __initdata dispatch_init_table[] = {
#ifdef CONFIG_XTENSA_UNALIGNED_USER #ifdef CONFIG_XTENSA_UNALIGNED_USER
{ EXCCAUSE_UNALIGNED, USER, fast_unaligned }, { EXCCAUSE_UNALIGNED, USER, fast_unaligned },
#endif #endif
{ EXCCAUSE_UNALIGNED, 0, do_unaligned_user },
{ EXCCAUSE_UNALIGNED, KRNL, fast_unaligned }, { EXCCAUSE_UNALIGNED, KRNL, fast_unaligned },
#endif #endif
{ EXCCAUSE_UNALIGNED, 0, do_unaligned_user },
#ifdef CONFIG_MMU #ifdef CONFIG_MMU
{ EXCCAUSE_ITLB_MISS, 0, do_page_fault }, { EXCCAUSE_ITLB_MISS, 0, do_page_fault },
{ EXCCAUSE_ITLB_MISS, USER|KRNL, fast_second_level_miss}, { EXCCAUSE_ITLB_MISS, USER|KRNL, fast_second_level_miss},
...@@ -171,6 +175,23 @@ __die_if_kernel(const char *str, struct pt_regs *regs, long err) ...@@ -171,6 +175,23 @@ __die_if_kernel(const char *str, struct pt_regs *regs, long err)
die(str, regs, err); die(str, regs, err);
} }
#ifdef CONFIG_PRINT_USER_CODE_ON_UNHANDLED_EXCEPTION
static inline void dump_user_code(struct pt_regs *regs)
{
char buf[32];
if (copy_from_user(buf, (void __user *)(regs->pc & -16), sizeof(buf)) == 0) {
print_hex_dump(KERN_INFO, " ", DUMP_PREFIX_NONE,
32, 1, buf, sizeof(buf), false);
}
}
#else
static inline void dump_user_code(struct pt_regs *regs)
{
}
#endif
/* /*
* Unhandled Exceptions. Kill user task or panic if in kernel space. * Unhandled Exceptions. Kill user task or panic if in kernel space.
*/ */
...@@ -186,6 +207,7 @@ void do_unhandled(struct pt_regs *regs) ...@@ -186,6 +207,7 @@ void do_unhandled(struct pt_regs *regs)
"\tEXCCAUSE is %ld\n", "\tEXCCAUSE is %ld\n",
current->comm, task_pid_nr(current), regs->pc, current->comm, task_pid_nr(current), regs->pc,
regs->exccause); regs->exccause);
dump_user_code(regs);
force_sig(SIGILL); force_sig(SIGILL);
} }
...@@ -349,6 +371,19 @@ static void do_div0(struct pt_regs *regs) ...@@ -349,6 +371,19 @@ static void do_div0(struct pt_regs *regs)
force_sig_fault(SIGFPE, FPE_INTDIV, (void __user *)regs->pc); force_sig_fault(SIGFPE, FPE_INTDIV, (void __user *)regs->pc);
} }
#ifdef CONFIG_XTENSA_LOAD_STORE
static void do_load_store(struct pt_regs *regs)
{
__die_if_kernel("Unhandled load/store exception in kernel",
regs, SIGKILL);
pr_info_ratelimited("Load/store error to %08lx in '%s' (pid = %d, pc = %#010lx)\n",
regs->excvaddr, current->comm,
task_pid_nr(current), regs->pc);
force_sig_fault(SIGBUS, BUS_ADRERR, (void *)regs->excvaddr);
}
#endif
/* /*
* Handle unaligned memory accesses from user space. Kill task. * Handle unaligned memory accesses from user space. Kill task.
* *
...@@ -356,7 +391,6 @@ static void do_div0(struct pt_regs *regs) ...@@ -356,7 +391,6 @@ static void do_div0(struct pt_regs *regs)
* accesses causes from user space. * accesses causes from user space.
*/ */
#if XCHAL_UNALIGNED_LOAD_EXCEPTION || XCHAL_UNALIGNED_STORE_EXCEPTION
static void do_unaligned_user(struct pt_regs *regs) static void do_unaligned_user(struct pt_regs *regs)
{ {
__die_if_kernel("Unhandled unaligned exception in kernel", __die_if_kernel("Unhandled unaligned exception in kernel",
...@@ -368,7 +402,6 @@ static void do_unaligned_user(struct pt_regs *regs) ...@@ -368,7 +402,6 @@ static void do_unaligned_user(struct pt_regs *regs)
task_pid_nr(current), regs->pc); task_pid_nr(current), regs->pc);
force_sig_fault(SIGBUS, BUS_ADRALN, (void *) regs->excvaddr); force_sig_fault(SIGBUS, BUS_ADRALN, (void *) regs->excvaddr);
} }
#endif
#if XTENSA_HAVE_COPROCESSORS #if XTENSA_HAVE_COPROCESSORS
static void do_coprocessor(struct pt_regs *regs) static void do_coprocessor(struct pt_regs *regs)
...@@ -534,31 +567,58 @@ static void show_trace(struct task_struct *task, unsigned long *sp, ...@@ -534,31 +567,58 @@ static void show_trace(struct task_struct *task, unsigned long *sp,
} }
#define STACK_DUMP_ENTRY_SIZE 4 #define STACK_DUMP_ENTRY_SIZE 4
#define STACK_DUMP_LINE_SIZE 32 #define STACK_DUMP_LINE_SIZE 16
static size_t kstack_depth_to_print = CONFIG_PRINT_STACK_DEPTH; static size_t kstack_depth_to_print = CONFIG_PRINT_STACK_DEPTH;
struct stack_fragment
{
size_t len;
size_t off;
u8 *sp;
const char *loglvl;
};
static int show_stack_fragment_cb(struct stackframe *frame, void *data)
{
struct stack_fragment *sf = data;
while (sf->off < sf->len) {
u8 line[STACK_DUMP_LINE_SIZE];
size_t line_len = sf->len - sf->off > STACK_DUMP_LINE_SIZE ?
STACK_DUMP_LINE_SIZE : sf->len - sf->off;
bool arrow = sf->off == 0;
if (frame && frame->sp == (unsigned long)(sf->sp + sf->off))
arrow = true;
__memcpy(line, sf->sp + sf->off, line_len);
print_hex_dump(sf->loglvl, arrow ? "> " : " ", DUMP_PREFIX_NONE,
STACK_DUMP_LINE_SIZE, STACK_DUMP_ENTRY_SIZE,
line, line_len, false);
sf->off += STACK_DUMP_LINE_SIZE;
if (arrow)
return 0;
}
return 1;
}
void show_stack(struct task_struct *task, unsigned long *sp, const char *loglvl) void show_stack(struct task_struct *task, unsigned long *sp, const char *loglvl)
{ {
size_t len, off = 0; struct stack_fragment sf;
if (!sp) if (!sp)
sp = stack_pointer(task); sp = stack_pointer(task);
len = min((-(size_t)sp) & (THREAD_SIZE - STACK_DUMP_ENTRY_SIZE), sf.len = min((-(size_t)sp) & (THREAD_SIZE - STACK_DUMP_ENTRY_SIZE),
kstack_depth_to_print * STACK_DUMP_ENTRY_SIZE); kstack_depth_to_print * STACK_DUMP_ENTRY_SIZE);
sf.off = 0;
sf.sp = (u8 *)sp;
sf.loglvl = loglvl;
printk("%sStack:\n", loglvl); printk("%sStack:\n", loglvl);
while (off < len) { walk_stackframe(sp, show_stack_fragment_cb, &sf);
u8 line[STACK_DUMP_LINE_SIZE]; while (sf.off < sf.len)
size_t line_len = len - off > STACK_DUMP_LINE_SIZE ? show_stack_fragment_cb(NULL, &sf);
STACK_DUMP_LINE_SIZE : len - off;
__memcpy(line, (u8 *)sp + off, line_len);
print_hex_dump(loglvl, " ", DUMP_PREFIX_NONE,
STACK_DUMP_LINE_SIZE, STACK_DUMP_ENTRY_SIZE,
line, line_len, false);
off += STACK_DUMP_LINE_SIZE;
}
show_trace(task, sp, loglvl); show_trace(task, sp, loglvl);
} }
......
...@@ -13,71 +13,10 @@ ...@@ -13,71 +13,10 @@
*/ */
#include <linux/module.h> #include <linux/module.h>
#include <linux/string.h> #include <asm/pgtable.h>
#include <linux/mm.h>
#include <linux/interrupt.h>
#include <asm/irq.h>
#include <linux/in6.h>
#include <linux/uaccess.h>
#include <asm/cacheflush.h>
#include <asm/checksum.h>
#include <asm/dma.h>
#include <asm/io.h>
#include <asm/page.h>
#include <asm/ftrace.h>
#ifdef CONFIG_BLK_DEV_FD
#include <asm/floppy.h>
#endif
#ifdef CONFIG_NET
#include <net/checksum.h>
#endif /* CONFIG_NET */
/*
* String functions
*/
EXPORT_SYMBOL(memset);
EXPORT_SYMBOL(memcpy);
EXPORT_SYMBOL(memmove);
EXPORT_SYMBOL(__memset);
EXPORT_SYMBOL(__memcpy);
EXPORT_SYMBOL(__memmove);
#ifdef CONFIG_ARCH_HAS_STRNCPY_FROM_USER
EXPORT_SYMBOL(__strncpy_user);
#endif
EXPORT_SYMBOL(clear_page);
EXPORT_SYMBOL(copy_page);
EXPORT_SYMBOL(empty_zero_page); EXPORT_SYMBOL(empty_zero_page);
/*
* gcc internal math functions
*/
extern long long __ashrdi3(long long, int);
extern long long __ashldi3(long long, int);
extern long long __bswapdi2(long long);
extern int __bswapsi2(int);
extern long long __lshrdi3(long long, int);
extern int __divsi3(int, int);
extern int __modsi3(int, int);
extern int __mulsi3(int, int);
extern unsigned int __udivsi3(unsigned int, unsigned int);
extern unsigned int __umodsi3(unsigned int, unsigned int);
extern unsigned long long __umulsidi3(unsigned int, unsigned int);
EXPORT_SYMBOL(__ashldi3);
EXPORT_SYMBOL(__ashrdi3);
EXPORT_SYMBOL(__bswapdi2);
EXPORT_SYMBOL(__bswapsi2);
EXPORT_SYMBOL(__lshrdi3);
EXPORT_SYMBOL(__divsi3);
EXPORT_SYMBOL(__modsi3);
EXPORT_SYMBOL(__mulsi3);
EXPORT_SYMBOL(__udivsi3);
EXPORT_SYMBOL(__umodsi3);
EXPORT_SYMBOL(__umulsidi3);
unsigned int __sync_fetch_and_and_4(volatile void *p, unsigned int v) unsigned int __sync_fetch_and_and_4(volatile void *p, unsigned int v)
{ {
BUG(); BUG();
...@@ -89,35 +28,3 @@ unsigned int __sync_fetch_and_or_4(volatile void *p, unsigned int v) ...@@ -89,35 +28,3 @@ unsigned int __sync_fetch_and_or_4(volatile void *p, unsigned int v)
BUG(); BUG();
} }
EXPORT_SYMBOL(__sync_fetch_and_or_4); EXPORT_SYMBOL(__sync_fetch_and_or_4);
/*
* Networking support
*/
EXPORT_SYMBOL(csum_partial);
EXPORT_SYMBOL(csum_partial_copy_generic);
/*
* Architecture-specific symbols
*/
EXPORT_SYMBOL(__xtensa_copy_user);
EXPORT_SYMBOL(__invalidate_icache_range);
/*
* Kernel hacking ...
*/
#if defined(CONFIG_VGA_CONSOLE) || defined(CONFIG_DUMMY_CONSOLE)
// FIXME EXPORT_SYMBOL(screen_info);
#endif
extern long common_exception_return;
EXPORT_SYMBOL(common_exception_return);
#ifdef CONFIG_FUNCTION_TRACER
EXPORT_SYMBOL(_mcount);
#endif
EXPORT_SYMBOL(__invalidate_dcache_range);
#if XCHAL_DCACHE_IS_WRITEBACK
EXPORT_SYMBOL(__flush_dcache_range);
#endif
...@@ -6,7 +6,8 @@ ...@@ -6,7 +6,8 @@
lib-y += memcopy.o memset.o checksum.o \ lib-y += memcopy.o memset.o checksum.o \
ashldi3.o ashrdi3.o bswapdi2.o bswapsi2.o lshrdi3.o \ ashldi3.o ashrdi3.o bswapdi2.o bswapsi2.o lshrdi3.o \
divsi3.o udivsi3.o modsi3.o umodsi3.o mulsi3.o umulsidi3.o \ divsi3.o udivsi3.o modsi3.o umodsi3.o mulsi3.o umulsidi3.o \
usercopy.o strncpy_user.o strnlen_user.o usercopy.o strnlen_user.o
lib-$(CONFIG_ARCH_HAS_STRNCPY_FROM_USER) += strncpy_user.o
lib-$(CONFIG_PCI) += pci-auto.o lib-$(CONFIG_PCI) += pci-auto.o
lib-$(CONFIG_KCSAN) += kcsan-stubs.o lib-$(CONFIG_KCSAN) += kcsan-stubs.o
KCSAN_SANITIZE_kcsan-stubs.o := n KCSAN_SANITIZE_kcsan-stubs.o := n
...@@ -26,3 +26,4 @@ ENTRY(__ashldi3) ...@@ -26,3 +26,4 @@ ENTRY(__ashldi3)
abi_ret_default abi_ret_default
ENDPROC(__ashldi3) ENDPROC(__ashldi3)
EXPORT_SYMBOL(__ashldi3)
...@@ -26,3 +26,4 @@ ENTRY(__ashrdi3) ...@@ -26,3 +26,4 @@ ENTRY(__ashrdi3)
abi_ret_default abi_ret_default
ENDPROC(__ashrdi3) ENDPROC(__ashrdi3)
EXPORT_SYMBOL(__ashrdi3)
...@@ -19,3 +19,4 @@ ENTRY(__bswapdi2) ...@@ -19,3 +19,4 @@ ENTRY(__bswapdi2)
abi_ret_default abi_ret_default
ENDPROC(__bswapdi2) ENDPROC(__bswapdi2)
EXPORT_SYMBOL(__bswapdi2)
...@@ -14,3 +14,4 @@ ENTRY(__bswapsi2) ...@@ -14,3 +14,4 @@ ENTRY(__bswapsi2)
abi_ret_default abi_ret_default
ENDPROC(__bswapsi2) ENDPROC(__bswapsi2)
EXPORT_SYMBOL(__bswapsi2)
...@@ -169,6 +169,7 @@ ENTRY(csum_partial) ...@@ -169,6 +169,7 @@ ENTRY(csum_partial)
j 5b /* branch to handle the remaining byte */ j 5b /* branch to handle the remaining byte */
ENDPROC(csum_partial) ENDPROC(csum_partial)
EXPORT_SYMBOL(csum_partial)
/* /*
* Copy from ds while checksumming, otherwise like csum_partial * Copy from ds while checksumming, otherwise like csum_partial
...@@ -346,6 +347,7 @@ EX(10f) s8i a8, a3, 1 ...@@ -346,6 +347,7 @@ EX(10f) s8i a8, a3, 1
j 4b /* process the possible trailing odd byte */ j 4b /* process the possible trailing odd byte */
ENDPROC(csum_partial_copy_generic) ENDPROC(csum_partial_copy_generic)
EXPORT_SYMBOL(csum_partial_copy_generic)
# Exception handler: # Exception handler:
......
...@@ -72,3 +72,4 @@ ENTRY(__divsi3) ...@@ -72,3 +72,4 @@ ENTRY(__divsi3)
abi_ret_default abi_ret_default
ENDPROC(__divsi3) ENDPROC(__divsi3)
EXPORT_SYMBOL(__divsi3)
...@@ -26,3 +26,4 @@ ENTRY(__lshrdi3) ...@@ -26,3 +26,4 @@ ENTRY(__lshrdi3)
abi_ret_default abi_ret_default
ENDPROC(__lshrdi3) ENDPROC(__lshrdi3)
EXPORT_SYMBOL(__lshrdi3)
...@@ -273,21 +273,8 @@ WEAK(memcpy) ...@@ -273,21 +273,8 @@ WEAK(memcpy)
abi_ret_default abi_ret_default
ENDPROC(__memcpy) ENDPROC(__memcpy)
EXPORT_SYMBOL(__memcpy)
/* EXPORT_SYMBOL(memcpy)
* void bcopy(const void *src, void *dest, size_t n);
*/
ENTRY(bcopy)
abi_entry_default
# a2=src, a3=dst, a4=len
mov a5, a3
mov a3, a2
mov a2, a5
j .Lmovecommon # go to common code for memmove+bcopy
ENDPROC(bcopy)
/* /*
* void *memmove(void *dst, const void *src, size_t len); * void *memmove(void *dst, const void *src, size_t len);
...@@ -551,3 +538,5 @@ WEAK(memmove) ...@@ -551,3 +538,5 @@ WEAK(memmove)
abi_ret_default abi_ret_default
ENDPROC(__memmove) ENDPROC(__memmove)
EXPORT_SYMBOL(__memmove)
EXPORT_SYMBOL(memmove)
...@@ -142,6 +142,8 @@ EX(10f) s8i a3, a5, 0 ...@@ -142,6 +142,8 @@ EX(10f) s8i a3, a5, 0
abi_ret_default abi_ret_default
ENDPROC(__memset) ENDPROC(__memset)
EXPORT_SYMBOL(__memset)
EXPORT_SYMBOL(memset)
.section .fixup, "ax" .section .fixup, "ax"
.align 4 .align 4
......
...@@ -60,6 +60,7 @@ ENTRY(__modsi3) ...@@ -60,6 +60,7 @@ ENTRY(__modsi3)
abi_ret_default abi_ret_default
ENDPROC(__modsi3) ENDPROC(__modsi3)
EXPORT_SYMBOL(__modsi3)
#if !XCHAL_HAVE_NSA #if !XCHAL_HAVE_NSA
.section .rodata .section .rodata
......
...@@ -131,3 +131,4 @@ ENTRY(__mulsi3) ...@@ -131,3 +131,4 @@ ENTRY(__mulsi3)
abi_ret_default abi_ret_default
ENDPROC(__mulsi3) ENDPROC(__mulsi3)
EXPORT_SYMBOL(__mulsi3)
...@@ -201,6 +201,7 @@ EX(10f) s8i a9, a11, 0 ...@@ -201,6 +201,7 @@ EX(10f) s8i a9, a11, 0
abi_ret_default abi_ret_default
ENDPROC(__strncpy_user) ENDPROC(__strncpy_user)
EXPORT_SYMBOL(__strncpy_user)
.section .fixup, "ax" .section .fixup, "ax"
.align 4 .align 4
......
...@@ -133,6 +133,7 @@ EX(10f) l32i a9, a4, 0 # get word with first two bytes of string ...@@ -133,6 +133,7 @@ EX(10f) l32i a9, a4, 0 # get word with first two bytes of string
abi_ret_default abi_ret_default
ENDPROC(__strnlen_user) ENDPROC(__strnlen_user)
EXPORT_SYMBOL(__strnlen_user)
.section .fixup, "ax" .section .fixup, "ax"
.align 4 .align 4
......
...@@ -66,3 +66,4 @@ ENTRY(__udivsi3) ...@@ -66,3 +66,4 @@ ENTRY(__udivsi3)
abi_ret_default abi_ret_default
ENDPROC(__udivsi3) ENDPROC(__udivsi3)
EXPORT_SYMBOL(__udivsi3)
...@@ -55,3 +55,4 @@ ENTRY(__umodsi3) ...@@ -55,3 +55,4 @@ ENTRY(__umodsi3)
abi_ret_default abi_ret_default
ENDPROC(__umodsi3) ENDPROC(__umodsi3)
EXPORT_SYMBOL(__umodsi3)
...@@ -228,3 +228,4 @@ ENTRY(__umulsidi3) ...@@ -228,3 +228,4 @@ ENTRY(__umulsidi3)
#endif /* XCHAL_NO_MUL */ #endif /* XCHAL_NO_MUL */
ENDPROC(__umulsidi3) ENDPROC(__umulsidi3)
EXPORT_SYMBOL(__umulsidi3)
...@@ -283,6 +283,7 @@ EX(10f) s8i a6, a5, 0 ...@@ -283,6 +283,7 @@ EX(10f) s8i a6, a5, 0
abi_ret(STACK_SIZE) abi_ret(STACK_SIZE)
ENDPROC(__xtensa_copy_user) ENDPROC(__xtensa_copy_user)
EXPORT_SYMBOL(__xtensa_copy_user)
.section .fixup, "ax" .section .fixup, "ax"
.align 4 .align 4
......
...@@ -14,7 +14,6 @@ ...@@ -14,7 +14,6 @@
#include <linux/kernel.h> #include <linux/kernel.h>
#include <asm/initialize_mmu.h> #include <asm/initialize_mmu.h>
#include <asm/tlbflush.h> #include <asm/tlbflush.h>
#include <asm/traps.h>
void __init kasan_early_init(void) void __init kasan_early_init(void)
{ {
...@@ -31,7 +30,6 @@ void __init kasan_early_init(void) ...@@ -31,7 +30,6 @@ void __init kasan_early_init(void)
BUG_ON(!pmd_none(*pmd)); BUG_ON(!pmd_none(*pmd));
set_pmd(pmd, __pmd((unsigned long)kasan_early_shadow_pte)); set_pmd(pmd, __pmd((unsigned long)kasan_early_shadow_pte));
} }
early_trap_init();
} }
static void __init populate(void *start, void *end) static void __init populate(void *start, void *end)
......
...@@ -47,6 +47,7 @@ ENTRY(clear_page) ...@@ -47,6 +47,7 @@ ENTRY(clear_page)
abi_ret_default abi_ret_default
ENDPROC(clear_page) ENDPROC(clear_page)
EXPORT_SYMBOL(clear_page)
/* /*
* copy_page and copy_user_page are the same for non-cache-aliased configs. * copy_page and copy_user_page are the same for non-cache-aliased configs.
...@@ -89,6 +90,7 @@ ENTRY(copy_page) ...@@ -89,6 +90,7 @@ ENTRY(copy_page)
abi_ret_default abi_ret_default
ENDPROC(copy_page) ENDPROC(copy_page)
EXPORT_SYMBOL(copy_page)
#ifdef CONFIG_MMU #ifdef CONFIG_MMU
/* /*
...@@ -367,6 +369,7 @@ ENTRY(__invalidate_icache_range) ...@@ -367,6 +369,7 @@ ENTRY(__invalidate_icache_range)
abi_ret_default abi_ret_default
ENDPROC(__invalidate_icache_range) ENDPROC(__invalidate_icache_range)
EXPORT_SYMBOL(__invalidate_icache_range)
/* /*
* void __flush_invalidate_dcache_range(ulong start, ulong size) * void __flush_invalidate_dcache_range(ulong start, ulong size)
...@@ -397,6 +400,7 @@ ENTRY(__flush_dcache_range) ...@@ -397,6 +400,7 @@ ENTRY(__flush_dcache_range)
abi_ret_default abi_ret_default
ENDPROC(__flush_dcache_range) ENDPROC(__flush_dcache_range)
EXPORT_SYMBOL(__flush_dcache_range)
/* /*
* void _invalidate_dcache_range(ulong start, ulong size) * void _invalidate_dcache_range(ulong start, ulong size)
...@@ -411,6 +415,7 @@ ENTRY(__invalidate_dcache_range) ...@@ -411,6 +415,7 @@ ENTRY(__invalidate_dcache_range)
abi_ret_default abi_ret_default
ENDPROC(__invalidate_dcache_range) ENDPROC(__invalidate_dcache_range)
EXPORT_SYMBOL(__invalidate_dcache_range)
/* /*
* void _invalidate_icache_all(void) * void _invalidate_icache_all(void)
......
...@@ -16,6 +16,7 @@ ...@@ -16,6 +16,7 @@
#include <linux/notifier.h> #include <linux/notifier.h>
#include <linux/panic_notifier.h> #include <linux/panic_notifier.h>
#include <linux/printk.h> #include <linux/printk.h>
#include <linux/reboot.h>
#include <linux/string.h> #include <linux/string.h>
#include <asm/platform.h> #include <asm/platform.h>
...@@ -24,26 +25,27 @@ ...@@ -24,26 +25,27 @@
#include <platform/simcall.h> #include <platform/simcall.h>
void platform_halt(void) static int iss_power_off(struct sys_off_data *unused)
{
pr_info(" ** Called platform_halt() **\n");
simc_exit(0);
}
void platform_power_off(void)
{ {
pr_info(" ** Called platform_power_off() **\n"); pr_info(" ** Called platform_power_off() **\n");
simc_exit(0); simc_exit(0);
return NOTIFY_DONE;
} }
void platform_restart(void) static int iss_restart(struct notifier_block *this,
unsigned long event, void *ptr)
{ {
/* Flush and reset the mmu, simulate a processor reset, and /* Flush and reset the mmu, simulate a processor reset, and
* jump to the reset vector. */ * jump to the reset vector. */
cpu_reset(); cpu_reset();
/* control never gets here */
return NOTIFY_DONE;
} }
static struct notifier_block iss_restart_block = {
.notifier_call = iss_restart,
};
static int static int
iss_panic_event(struct notifier_block *this, unsigned long event, void *ptr) iss_panic_event(struct notifier_block *this, unsigned long event, void *ptr)
{ {
...@@ -82,4 +84,8 @@ void __init platform_setup(char **p_cmdline) ...@@ -82,4 +84,8 @@ void __init platform_setup(char **p_cmdline)
} }
atomic_notifier_chain_register(&panic_notifier_list, &iss_panic_block); atomic_notifier_chain_register(&panic_notifier_list, &iss_panic_block);
register_restart_handler(&iss_restart_block);
register_sys_off_handler(SYS_OFF_MODE_POWER_OFF,
SYS_OFF_PRIO_PLATFORM,
iss_power_off, NULL);
} }
...@@ -23,6 +23,7 @@ ...@@ -23,6 +23,7 @@
#include <linux/platform_device.h> #include <linux/platform_device.h>
#include <linux/serial.h> #include <linux/serial.h>
#include <linux/serial_8250.h> #include <linux/serial_8250.h>
#include <linux/timer.h>
#include <asm/processor.h> #include <asm/processor.h>
#include <asm/platform.h> #include <asm/platform.h>
...@@ -41,51 +42,46 @@ static void led_print (int f, char *s) ...@@ -41,51 +42,46 @@ static void led_print (int f, char *s)
break; break;
} }
void platform_halt(void) static int xt2000_power_off(struct sys_off_data *unused)
{
led_print (0, " HALT ");
local_irq_disable();
while (1);
}
void platform_power_off(void)
{ {
led_print (0, "POWEROFF"); led_print (0, "POWEROFF");
local_irq_disable(); local_irq_disable();
while (1); while (1);
return NOTIFY_DONE;
} }
void platform_restart(void) static int xt2000_restart(struct notifier_block *this,
unsigned long event, void *ptr)
{ {
/* Flush and reset the mmu, simulate a processor reset, and /* Flush and reset the mmu, simulate a processor reset, and
* jump to the reset vector. */ * jump to the reset vector. */
cpu_reset(); cpu_reset();
/* control never gets here */
return NOTIFY_DONE;
} }
static struct notifier_block xt2000_restart_block = {
.notifier_call = xt2000_restart,
};
void __init platform_setup(char** cmdline) void __init platform_setup(char** cmdline)
{ {
led_print (0, "LINUX "); led_print (0, "LINUX ");
} }
/* early initialization */ /* Heartbeat. Let the LED blink. */
void __init platform_init(bp_tag_t *first) static void xt2000_heartbeat(struct timer_list *unused);
{
}
/* Heartbeat. Let the LED blink. */ static DEFINE_TIMER(heartbeat_timer, xt2000_heartbeat);
void platform_heartbeat(void) static void xt2000_heartbeat(struct timer_list *unused)
{ {
static int i, t; static int i;
if (--t < 0) led_print(7, i ? "." : " ");
{
t = 59;
led_print(7, i ? ".": " ");
i ^= 1; i ^= 1;
} mod_timer(&heartbeat_timer, jiffies + HZ / 2);
} }
//#define RS_TABLE_SIZE 2 //#define RS_TABLE_SIZE 2
...@@ -143,7 +139,11 @@ static int __init xt2000_setup_devinit(void) ...@@ -143,7 +139,11 @@ static int __init xt2000_setup_devinit(void)
{ {
platform_device_register(&xt2000_serial8250_device); platform_device_register(&xt2000_serial8250_device);
platform_device_register(&xt2000_sonic_device); platform_device_register(&xt2000_sonic_device);
mod_timer(&heartbeat_timer, jiffies + HZ / 2);
register_restart_handler(&xt2000_restart_block);
register_sys_off_handler(SYS_OFF_MODE_POWER_OFF,
SYS_OFF_PRIO_DEFAULT,
xt2000_power_off, NULL);
return 0; return 0;
} }
......
...@@ -33,23 +33,17 @@ ...@@ -33,23 +33,17 @@
#include <platform/lcd.h> #include <platform/lcd.h>
#include <platform/hardware.h> #include <platform/hardware.h>
void platform_halt(void) static int xtfpga_power_off(struct sys_off_data *unused)
{
lcd_disp_at_pos(" HALT ", 0);
local_irq_disable();
while (1)
cpu_relax();
}
void platform_power_off(void)
{ {
lcd_disp_at_pos("POWEROFF", 0); lcd_disp_at_pos("POWEROFF", 0);
local_irq_disable(); local_irq_disable();
while (1) while (1)
cpu_relax(); cpu_relax();
return NOTIFY_DONE;
} }
void platform_restart(void) static int xtfpga_restart(struct notifier_block *this,
unsigned long event, void *ptr)
{ {
/* Try software reset first. */ /* Try software reset first. */
WRITE_ONCE(*(u32 *)XTFPGA_SWRST_VADDR, 0xdead); WRITE_ONCE(*(u32 *)XTFPGA_SWRST_VADDR, 0xdead);
...@@ -58,9 +52,14 @@ void platform_restart(void) ...@@ -58,9 +52,14 @@ void platform_restart(void)
* simulate a processor reset, and jump to the reset vector. * simulate a processor reset, and jump to the reset vector.
*/ */
cpu_reset(); cpu_reset();
/* control never gets here */
return NOTIFY_DONE;
} }
static struct notifier_block xtfpga_restart_block = {
.notifier_call = xtfpga_restart,
};
#ifdef CONFIG_XTENSA_CALIBRATE_CCOUNT #ifdef CONFIG_XTENSA_CALIBRATE_CCOUNT
void __init platform_calibrate_ccount(void) void __init platform_calibrate_ccount(void)
...@@ -70,6 +69,14 @@ void __init platform_calibrate_ccount(void) ...@@ -70,6 +69,14 @@ void __init platform_calibrate_ccount(void)
#endif #endif
static void __init xtfpga_register_handlers(void)
{
register_restart_handler(&xtfpga_restart_block);
register_sys_off_handler(SYS_OFF_MODE_POWER_OFF,
SYS_OFF_PRIO_DEFAULT,
xtfpga_power_off, NULL);
}
#ifdef CONFIG_USE_OF #ifdef CONFIG_USE_OF
static void __init xtfpga_clk_setup(struct device_node *np) static void __init xtfpga_clk_setup(struct device_node *np)
...@@ -134,6 +141,9 @@ static int __init machine_setup(void) ...@@ -134,6 +141,9 @@ static int __init machine_setup(void)
if ((eth = of_find_compatible_node(eth, NULL, "opencores,ethoc"))) if ((eth = of_find_compatible_node(eth, NULL, "opencores,ethoc")))
update_local_mac(eth); update_local_mac(eth);
of_node_put(eth); of_node_put(eth);
xtfpga_register_handlers();
return 0; return 0;
} }
arch_initcall(machine_setup); arch_initcall(machine_setup);
...@@ -281,6 +291,8 @@ static int __init xtavnet_init(void) ...@@ -281,6 +291,8 @@ static int __init xtavnet_init(void)
pr_info("XTFPGA: Ethernet MAC %pM\n", ethoc_pdata.hwaddr); pr_info("XTFPGA: Ethernet MAC %pM\n", ethoc_pdata.hwaddr);
ethoc_pdata.eth_clkfreq = *(long *)XTFPGA_CLKFRQ_VADDR; ethoc_pdata.eth_clkfreq = *(long *)XTFPGA_CLKFRQ_VADDR;
xtfpga_register_handlers();
return 0; return 0;
} }
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment