Commit a95eb1d0 authored by Linus Torvalds's avatar Linus Torvalds

Merge tag 'loongarch-fixes-5.19-5' of...

Merge tag 'loongarch-fixes-5.19-5' of git://git.kernel.org/pub/scm/linux/kernel/git/chenhuacai/linux-loongson

Pull LoongArch fixes from Huacai Chen:

 - Fix cache size calculation, stack protection attributes, ptrace's
   fpr_set and "ROM Size" in boardinfo

 - Some cleanups and improvements of assembly

 - Some cleanups of unused code and useless code

* tag 'loongarch-fixes-5.19-5' of git://git.kernel.org/pub/scm/linux/kernel/git/chenhuacai/linux-loongson:
  LoongArch: Fix wrong "ROM Size" of boardinfo
  LoongArch: Fix missing fcsr in ptrace's fpr_set
  LoongArch: Fix shared cache size calculation
  LoongArch: Disable executable stack by default
  LoongArch: Remove unused variables
  LoongArch: Remove clock setting during cpu hotplug stage
  LoongArch: Remove useless header compiler.h
  LoongArch: Remove several syntactic sugar macros for branches
  LoongArch: Re-tab the assembly files
  LoongArch: Simplify "BGT foo, zero" with BGTZ
  LoongArch: Simplify "BLT foo, zero" with BLTZ
  LoongArch: Simplify "BEQ/BNE foo, zero" with BEQZ/BNEZ
  LoongArch: Use the "move" pseudo-instruction where applicable
  LoongArch: Use the "jr" pseudo-instruction where applicable
  LoongArch: Use ABI names of registers where appropriate
parents 9d928d9b 45b53c90
...@@ -69,7 +69,6 @@ config LOONGARCH ...@@ -69,7 +69,6 @@ config LOONGARCH
select GENERIC_TIME_VSYSCALL select GENERIC_TIME_VSYSCALL
select GPIOLIB select GPIOLIB
select HAVE_ARCH_AUDITSYSCALL select HAVE_ARCH_AUDITSYSCALL
select HAVE_ARCH_COMPILER_H
select HAVE_ARCH_MMAP_RND_BITS if MMU select HAVE_ARCH_MMAP_RND_BITS if MMU
select HAVE_ARCH_SECCOMP_FILTER select HAVE_ARCH_SECCOMP_FILTER
select HAVE_ARCH_TRACEHOOK select HAVE_ARCH_TRACEHOOK
......
...@@ -274,16 +274,4 @@ ...@@ -274,16 +274,4 @@
nor \dst, \src, zero nor \dst, \src, zero
.endm .endm
.macro bgt r0 r1 label
blt \r1, \r0, \label
.endm
.macro bltz r0 label
blt \r0, zero, \label
.endm
.macro bgez r0 label
bge \r0, zero, \label
.endm
#endif /* _ASM_ASMMACRO_H */ #endif /* _ASM_ASMMACRO_H */
...@@ -10,7 +10,6 @@ ...@@ -10,7 +10,6 @@
#include <linux/types.h> #include <linux/types.h>
#include <asm/barrier.h> #include <asm/barrier.h>
#include <asm/cmpxchg.h> #include <asm/cmpxchg.h>
#include <asm/compiler.h>
#if __SIZEOF_LONG__ == 4 #if __SIZEOF_LONG__ == 4
#define __LL "ll.w " #define __LL "ll.w "
...@@ -157,27 +156,25 @@ static inline int arch_atomic_sub_if_positive(int i, atomic_t *v) ...@@ -157,27 +156,25 @@ static inline int arch_atomic_sub_if_positive(int i, atomic_t *v)
__asm__ __volatile__( __asm__ __volatile__(
"1: ll.w %1, %2 # atomic_sub_if_positive\n" "1: ll.w %1, %2 # atomic_sub_if_positive\n"
" addi.w %0, %1, %3 \n" " addi.w %0, %1, %3 \n"
" or %1, %0, $zero \n" " move %1, %0 \n"
" blt %0, $zero, 2f \n" " bltz %0, 2f \n"
" sc.w %1, %2 \n" " sc.w %1, %2 \n"
" beq $zero, %1, 1b \n" " beqz %1, 1b \n"
"2: \n" "2: \n"
__WEAK_LLSC_MB __WEAK_LLSC_MB
: "=&r" (result), "=&r" (temp), : "=&r" (result), "=&r" (temp), "+ZC" (v->counter)
"+" GCC_OFF_SMALL_ASM() (v->counter)
: "I" (-i)); : "I" (-i));
} else { } else {
__asm__ __volatile__( __asm__ __volatile__(
"1: ll.w %1, %2 # atomic_sub_if_positive\n" "1: ll.w %1, %2 # atomic_sub_if_positive\n"
" sub.w %0, %1, %3 \n" " sub.w %0, %1, %3 \n"
" or %1, %0, $zero \n" " move %1, %0 \n"
" blt %0, $zero, 2f \n" " bltz %0, 2f \n"
" sc.w %1, %2 \n" " sc.w %1, %2 \n"
" beq $zero, %1, 1b \n" " beqz %1, 1b \n"
"2: \n" "2: \n"
__WEAK_LLSC_MB __WEAK_LLSC_MB
: "=&r" (result), "=&r" (temp), : "=&r" (result), "=&r" (temp), "+ZC" (v->counter)
"+" GCC_OFF_SMALL_ASM() (v->counter)
: "r" (i)); : "r" (i));
} }
...@@ -320,27 +317,25 @@ static inline long arch_atomic64_sub_if_positive(long i, atomic64_t *v) ...@@ -320,27 +317,25 @@ static inline long arch_atomic64_sub_if_positive(long i, atomic64_t *v)
__asm__ __volatile__( __asm__ __volatile__(
"1: ll.d %1, %2 # atomic64_sub_if_positive \n" "1: ll.d %1, %2 # atomic64_sub_if_positive \n"
" addi.d %0, %1, %3 \n" " addi.d %0, %1, %3 \n"
" or %1, %0, $zero \n" " move %1, %0 \n"
" blt %0, $zero, 2f \n" " bltz %0, 2f \n"
" sc.d %1, %2 \n" " sc.d %1, %2 \n"
" beq %1, $zero, 1b \n" " beqz %1, 1b \n"
"2: \n" "2: \n"
__WEAK_LLSC_MB __WEAK_LLSC_MB
: "=&r" (result), "=&r" (temp), : "=&r" (result), "=&r" (temp), "+ZC" (v->counter)
"+" GCC_OFF_SMALL_ASM() (v->counter)
: "I" (-i)); : "I" (-i));
} else { } else {
__asm__ __volatile__( __asm__ __volatile__(
"1: ll.d %1, %2 # atomic64_sub_if_positive \n" "1: ll.d %1, %2 # atomic64_sub_if_positive \n"
" sub.d %0, %1, %3 \n" " sub.d %0, %1, %3 \n"
" or %1, %0, $zero \n" " move %1, %0 \n"
" blt %0, $zero, 2f \n" " bltz %0, 2f \n"
" sc.d %1, %2 \n" " sc.d %1, %2 \n"
" beq %1, $zero, 1b \n" " beqz %1, 1b \n"
"2: \n" "2: \n"
__WEAK_LLSC_MB __WEAK_LLSC_MB
: "=&r" (result), "=&r" (temp), : "=&r" (result), "=&r" (temp), "+ZC" (v->counter)
"+" GCC_OFF_SMALL_ASM() (v->counter)
: "r" (i)); : "r" (i));
} }
......
...@@ -48,9 +48,9 @@ static inline unsigned long array_index_mask_nospec(unsigned long index, ...@@ -48,9 +48,9 @@ static inline unsigned long array_index_mask_nospec(unsigned long index,
__asm__ __volatile__( __asm__ __volatile__(
"sltu %0, %1, %2\n\t" "sltu %0, %1, %2\n\t"
#if (__SIZEOF_LONG__ == 4) #if (__SIZEOF_LONG__ == 4)
"sub.w %0, $r0, %0\n\t" "sub.w %0, $zero, %0\n\t"
#elif (__SIZEOF_LONG__ == 8) #elif (__SIZEOF_LONG__ == 8)
"sub.d %0, $r0, %0\n\t" "sub.d %0, $zero, %0\n\t"
#endif #endif
: "=r" (mask) : "=r" (mask)
: "r" (index), "r" (size) : "r" (index), "r" (size)
......
...@@ -55,9 +55,9 @@ static inline unsigned long __xchg(volatile void *ptr, unsigned long x, ...@@ -55,9 +55,9 @@ static inline unsigned long __xchg(volatile void *ptr, unsigned long x,
__asm__ __volatile__( \ __asm__ __volatile__( \
"1: " ld " %0, %2 # __cmpxchg_asm \n" \ "1: " ld " %0, %2 # __cmpxchg_asm \n" \
" bne %0, %z3, 2f \n" \ " bne %0, %z3, 2f \n" \
" or $t0, %z4, $zero \n" \ " move $t0, %z4 \n" \
" " st " $t0, %1 \n" \ " " st " $t0, %1 \n" \
" beq $zero, $t0, 1b \n" \ " beqz $t0, 1b \n" \
"2: \n" \ "2: \n" \
__WEAK_LLSC_MB \ __WEAK_LLSC_MB \
: "=&r" (__ret), "=ZB"(*m) \ : "=&r" (__ret), "=ZB"(*m) \
......
/* SPDX-License-Identifier: GPL-2.0 */
/*
* Copyright (C) 2020-2022 Loongson Technology Corporation Limited
*/
#ifndef _ASM_COMPILER_H
#define _ASM_COMPILER_H
#define GCC_OFF_SMALL_ASM() "ZC"
#define LOONGARCH_ISA_LEVEL "loongarch"
#define LOONGARCH_ISA_ARCH_LEVEL "arch=loongarch"
#define LOONGARCH_ISA_LEVEL_RAW loongarch
#define LOONGARCH_ISA_ARCH_LEVEL_RAW LOONGARCH_ISA_LEVEL_RAW
#endif /* _ASM_COMPILER_H */
...@@ -288,8 +288,6 @@ struct arch_elf_state { ...@@ -288,8 +288,6 @@ struct arch_elf_state {
.interp_fp_abi = LOONGARCH_ABI_FP_ANY, \ .interp_fp_abi = LOONGARCH_ABI_FP_ANY, \
} }
#define elf_read_implies_exec(ex, exec_stk) (exec_stk == EXSTACK_DEFAULT)
extern int arch_elf_pt_proc(void *ehdr, void *phdr, struct file *elf, extern int arch_elf_pt_proc(void *ehdr, void *phdr, struct file *elf,
bool is_interp, struct arch_elf_state *state); bool is_interp, struct arch_elf_state *state);
......
...@@ -8,7 +8,6 @@ ...@@ -8,7 +8,6 @@
#include <linux/futex.h> #include <linux/futex.h>
#include <linux/uaccess.h> #include <linux/uaccess.h>
#include <asm/barrier.h> #include <asm/barrier.h>
#include <asm/compiler.h>
#include <asm/errno.h> #include <asm/errno.h>
#define __futex_atomic_op(insn, ret, oldval, uaddr, oparg) \ #define __futex_atomic_op(insn, ret, oldval, uaddr, oparg) \
...@@ -17,7 +16,7 @@ ...@@ -17,7 +16,7 @@
"1: ll.w %1, %4 # __futex_atomic_op\n" \ "1: ll.w %1, %4 # __futex_atomic_op\n" \
" " insn " \n" \ " " insn " \n" \
"2: sc.w $t0, %2 \n" \ "2: sc.w $t0, %2 \n" \
" beq $t0, $zero, 1b \n" \ " beqz $t0, 1b \n" \
"3: \n" \ "3: \n" \
" .section .fixup,\"ax\" \n" \ " .section .fixup,\"ax\" \n" \
"4: li.w %0, %6 \n" \ "4: li.w %0, %6 \n" \
...@@ -82,9 +81,9 @@ futex_atomic_cmpxchg_inatomic(u32 *uval, u32 __user *uaddr, u32 oldval, u32 newv ...@@ -82,9 +81,9 @@ futex_atomic_cmpxchg_inatomic(u32 *uval, u32 __user *uaddr, u32 oldval, u32 newv
"# futex_atomic_cmpxchg_inatomic \n" "# futex_atomic_cmpxchg_inatomic \n"
"1: ll.w %1, %3 \n" "1: ll.w %1, %3 \n"
" bne %1, %z4, 3f \n" " bne %1, %z4, 3f \n"
" or $t0, %z5, $zero \n" " move $t0, %z5 \n"
"2: sc.w $t0, %2 \n" "2: sc.w $t0, %2 \n"
" beq $zero, $t0, 1b \n" " beqz $t0, 1b \n"
"3: \n" "3: \n"
__WEAK_LLSC_MB __WEAK_LLSC_MB
" .section .fixup,\"ax\" \n" " .section .fixup,\"ax\" \n"
...@@ -95,8 +94,8 @@ futex_atomic_cmpxchg_inatomic(u32 *uval, u32 __user *uaddr, u32 oldval, u32 newv ...@@ -95,8 +94,8 @@ futex_atomic_cmpxchg_inatomic(u32 *uval, u32 __user *uaddr, u32 oldval, u32 newv
" "__UA_ADDR "\t1b, 4b \n" " "__UA_ADDR "\t1b, 4b \n"
" "__UA_ADDR "\t2b, 4b \n" " "__UA_ADDR "\t2b, 4b \n"
" .previous \n" " .previous \n"
: "+r" (ret), "=&r" (val), "=" GCC_OFF_SMALL_ASM() (*uaddr) : "+r" (ret), "=&r" (val), "=ZC" (*uaddr)
: GCC_OFF_SMALL_ASM() (*uaddr), "Jr" (oldval), "Jr" (newval), : "ZC" (*uaddr), "Jr" (oldval), "Jr" (newval),
"i" (-EFAULT) "i" (-EFAULT)
: "memory", "t0"); : "memory", "t0");
......
...@@ -9,7 +9,6 @@ ...@@ -9,7 +9,6 @@
#include <linux/compiler.h> #include <linux/compiler.h>
#include <linux/stringify.h> #include <linux/stringify.h>
#include <asm/compiler.h>
#include <asm/loongarch.h> #include <asm/loongarch.h>
static inline void arch_local_irq_enable(void) static inline void arch_local_irq_enable(void)
......
...@@ -9,7 +9,6 @@ ...@@ -9,7 +9,6 @@
#include <linux/bitops.h> #include <linux/bitops.h>
#include <linux/atomic.h> #include <linux/atomic.h>
#include <asm/cmpxchg.h> #include <asm/cmpxchg.h>
#include <asm/compiler.h>
typedef struct { typedef struct {
atomic_long_t a; atomic_long_t a;
......
...@@ -39,18 +39,6 @@ extern const struct plat_smp_ops loongson3_smp_ops; ...@@ -39,18 +39,6 @@ extern const struct plat_smp_ops loongson3_smp_ops;
#define MAX_PACKAGES 16 #define MAX_PACKAGES 16
/* Chip Config register of each physical cpu package */
extern u64 loongson_chipcfg[MAX_PACKAGES];
#define LOONGSON_CHIPCFG(id) (*(volatile u32 *)(loongson_chipcfg[id]))
/* Chip Temperature register of each physical cpu package */
extern u64 loongson_chiptemp[MAX_PACKAGES];
#define LOONGSON_CHIPTEMP(id) (*(volatile u32 *)(loongson_chiptemp[id]))
/* Freq Control register of each physical cpu package */
extern u64 loongson_freqctrl[MAX_PACKAGES];
#define LOONGSON_FREQCTRL(id) (*(volatile u32 *)(loongson_freqctrl[id]))
#define xconf_readl(addr) readl(addr) #define xconf_readl(addr) readl(addr)
#define xconf_readq(addr) readq(addr) #define xconf_readq(addr) readq(addr)
...@@ -58,7 +46,7 @@ static inline void xconf_writel(u32 val, volatile void __iomem *addr) ...@@ -58,7 +46,7 @@ static inline void xconf_writel(u32 val, volatile void __iomem *addr)
{ {
asm volatile ( asm volatile (
" st.w %[v], %[hw], 0 \n" " st.w %[v], %[hw], 0 \n"
" ld.b $r0, %[hw], 0 \n" " ld.b $zero, %[hw], 0 \n"
: :
: [hw] "r" (addr), [v] "r" (val) : [hw] "r" (addr), [v] "r" (val)
); );
...@@ -68,7 +56,7 @@ static inline void xconf_writeq(u64 val64, volatile void __iomem *addr) ...@@ -68,7 +56,7 @@ static inline void xconf_writeq(u64 val64, volatile void __iomem *addr)
{ {
asm volatile ( asm volatile (
" st.d %[v], %[hw], 0 \n" " st.d %[v], %[hw], 0 \n"
" ld.b $r0, %[hw], 0 \n" " ld.b $zero, %[hw], 0 \n"
: :
: [hw] "r" (addr), [v] "r" (val64) : [hw] "r" (addr), [v] "r" (val64)
); );
......
...@@ -23,13 +23,13 @@ ...@@ -23,13 +23,13 @@
static __always_inline void prepare_frametrace(struct pt_regs *regs) static __always_inline void prepare_frametrace(struct pt_regs *regs)
{ {
__asm__ __volatile__( __asm__ __volatile__(
/* Save $r1 */ /* Save $ra */
STORE_ONE_REG(1) STORE_ONE_REG(1)
/* Use $r1 to save PC */ /* Use $ra to save PC */
"pcaddi $r1, 0\n\t" "pcaddi $ra, 0\n\t"
STR_LONG_S " $r1, %0\n\t" STR_LONG_S " $ra, %0\n\t"
/* Restore $r1 */ /* Restore $ra */
STR_LONG_L " $r1, %1, "STR_LONGSIZE"\n\t" STR_LONG_L " $ra, %1, "STR_LONGSIZE"\n\t"
STORE_ONE_REG(2) STORE_ONE_REG(2)
STORE_ONE_REG(3) STORE_ONE_REG(3)
STORE_ONE_REG(4) STORE_ONE_REG(4)
......
...@@ -44,14 +44,14 @@ struct thread_info { ...@@ -44,14 +44,14 @@ struct thread_info {
} }
/* How to get the thread information struct from C. */ /* How to get the thread information struct from C. */
register struct thread_info *__current_thread_info __asm__("$r2"); register struct thread_info *__current_thread_info __asm__("$tp");
static inline struct thread_info *current_thread_info(void) static inline struct thread_info *current_thread_info(void)
{ {
return __current_thread_info; return __current_thread_info;
} }
register unsigned long current_stack_pointer __asm__("$r3"); register unsigned long current_stack_pointer __asm__("$sp");
#endif /* !__ASSEMBLY__ */ #endif /* !__ASSEMBLY__ */
......
...@@ -162,7 +162,7 @@ do { \ ...@@ -162,7 +162,7 @@ do { \
"2: \n" \ "2: \n" \
" .section .fixup,\"ax\" \n" \ " .section .fixup,\"ax\" \n" \
"3: li.w %0, %3 \n" \ "3: li.w %0, %3 \n" \
" or %1, $r0, $r0 \n" \ " move %1, $zero \n" \
" b 2b \n" \ " b 2b \n" \
" .previous \n" \ " .previous \n" \
" .section __ex_table,\"a\" \n" \ " .section __ex_table,\"a\" \n" \
......
...@@ -4,8 +4,9 @@ ...@@ -4,8 +4,9 @@
* *
* Copyright (C) 2020-2022 Loongson Technology Corporation Limited * Copyright (C) 2020-2022 Loongson Technology Corporation Limited
*/ */
#include <asm/cpu-info.h>
#include <linux/cacheinfo.h> #include <linux/cacheinfo.h>
#include <asm/bootinfo.h>
#include <asm/cpu-info.h>
/* Populates leaf and increments to next leaf */ /* Populates leaf and increments to next leaf */
#define populate_cache(cache, leaf, c_level, c_type) \ #define populate_cache(cache, leaf, c_level, c_type) \
...@@ -17,6 +18,8 @@ do { \ ...@@ -17,6 +18,8 @@ do { \
leaf->ways_of_associativity = c->cache.ways; \ leaf->ways_of_associativity = c->cache.ways; \
leaf->size = c->cache.linesz * c->cache.sets * \ leaf->size = c->cache.linesz * c->cache.sets * \
c->cache.ways; \ c->cache.ways; \
if (leaf->level > 2) \
leaf->size *= nodes_per_package; \
leaf++; \ leaf++; \
} while (0) } while (0)
...@@ -95,11 +98,15 @@ static void cache_cpumap_setup(unsigned int cpu) ...@@ -95,11 +98,15 @@ static void cache_cpumap_setup(unsigned int cpu)
int populate_cache_leaves(unsigned int cpu) int populate_cache_leaves(unsigned int cpu)
{ {
int level = 1; int level = 1, nodes_per_package = 1;
struct cpuinfo_loongarch *c = &current_cpu_data; struct cpuinfo_loongarch *c = &current_cpu_data;
struct cpu_cacheinfo *this_cpu_ci = get_cpu_cacheinfo(cpu); struct cpu_cacheinfo *this_cpu_ci = get_cpu_cacheinfo(cpu);
struct cacheinfo *this_leaf = this_cpu_ci->info_list; struct cacheinfo *this_leaf = this_cpu_ci->info_list;
if (loongson_sysconf.nr_nodes > 1)
nodes_per_package = loongson_sysconf.cores_per_package
/ loongson_sysconf.cores_per_node;
if (c->icache.waysize) { if (c->icache.waysize) {
populate_cache(dcache, this_leaf, level, CACHE_TYPE_DATA); populate_cache(dcache, this_leaf, level, CACHE_TYPE_DATA);
populate_cache(icache, this_leaf, level++, CACHE_TYPE_INST); populate_cache(icache, this_leaf, level++, CACHE_TYPE_INST);
......
...@@ -17,21 +17,6 @@ u64 efi_system_table; ...@@ -17,21 +17,6 @@ u64 efi_system_table;
struct loongson_system_configuration loongson_sysconf; struct loongson_system_configuration loongson_sysconf;
EXPORT_SYMBOL(loongson_sysconf); EXPORT_SYMBOL(loongson_sysconf);
u64 loongson_chipcfg[MAX_PACKAGES];
u64 loongson_chiptemp[MAX_PACKAGES];
u64 loongson_freqctrl[MAX_PACKAGES];
unsigned long long smp_group[MAX_PACKAGES];
static void __init register_addrs_set(u64 *registers, const u64 addr, int num)
{
u64 i;
for (i = 0; i < num; i++) {
*registers = (i << 44) | addr;
registers++;
}
}
void __init init_environ(void) void __init init_environ(void)
{ {
int efi_boot = fw_arg0; int efi_boot = fw_arg0;
...@@ -50,11 +35,6 @@ void __init init_environ(void) ...@@ -50,11 +35,6 @@ void __init init_environ(void)
efi_memmap_init_early(&data); efi_memmap_init_early(&data);
memblock_reserve(data.phys_map & PAGE_MASK, memblock_reserve(data.phys_map & PAGE_MASK,
PAGE_ALIGN(data.size + (data.phys_map & ~PAGE_MASK))); PAGE_ALIGN(data.size + (data.phys_map & ~PAGE_MASK)));
register_addrs_set(smp_group, TO_UNCACHE(0x1fe01000), 16);
register_addrs_set(loongson_chipcfg, TO_UNCACHE(0x1fe00180), 16);
register_addrs_set(loongson_chiptemp, TO_UNCACHE(0x1fe0019c), 16);
register_addrs_set(loongson_freqctrl, TO_UNCACHE(0x1fe001d0), 16);
} }
static int __init init_cpu_fullname(void) static int __init init_cpu_fullname(void)
......
...@@ -153,7 +153,7 @@ SYM_FUNC_START(_save_fp) ...@@ -153,7 +153,7 @@ SYM_FUNC_START(_save_fp)
fpu_save_csr a0 t1 fpu_save_csr a0 t1
fpu_save_double a0 t1 # clobbers t1 fpu_save_double a0 t1 # clobbers t1
fpu_save_cc a0 t1 t2 # clobbers t1, t2 fpu_save_cc a0 t1 t2 # clobbers t1, t2
jirl zero, ra, 0 jr ra
SYM_FUNC_END(_save_fp) SYM_FUNC_END(_save_fp)
EXPORT_SYMBOL(_save_fp) EXPORT_SYMBOL(_save_fp)
...@@ -164,7 +164,7 @@ SYM_FUNC_START(_restore_fp) ...@@ -164,7 +164,7 @@ SYM_FUNC_START(_restore_fp)
fpu_restore_double a0 t1 # clobbers t1 fpu_restore_double a0 t1 # clobbers t1
fpu_restore_csr a0 t1 fpu_restore_csr a0 t1
fpu_restore_cc a0 t1 t2 # clobbers t1, t2 fpu_restore_cc a0 t1 t2 # clobbers t1, t2
jirl zero, ra, 0 jr ra
SYM_FUNC_END(_restore_fp) SYM_FUNC_END(_restore_fp)
/* /*
...@@ -216,7 +216,7 @@ SYM_FUNC_START(_init_fpu) ...@@ -216,7 +216,7 @@ SYM_FUNC_START(_init_fpu)
movgr2fr.d $f30, t1 movgr2fr.d $f30, t1
movgr2fr.d $f31, t1 movgr2fr.d $f31, t1
jirl zero, ra, 0 jr ra
SYM_FUNC_END(_init_fpu) SYM_FUNC_END(_init_fpu)
/* /*
...@@ -229,7 +229,7 @@ SYM_FUNC_START(_save_fp_context) ...@@ -229,7 +229,7 @@ SYM_FUNC_START(_save_fp_context)
sc_save_fcsr a2 t1 sc_save_fcsr a2 t1
sc_save_fp a0 sc_save_fp a0
li.w a0, 0 # success li.w a0, 0 # success
jirl zero, ra, 0 jr ra
SYM_FUNC_END(_save_fp_context) SYM_FUNC_END(_save_fp_context)
/* /*
...@@ -242,10 +242,10 @@ SYM_FUNC_START(_restore_fp_context) ...@@ -242,10 +242,10 @@ SYM_FUNC_START(_restore_fp_context)
sc_restore_fcc a1 t1 t2 sc_restore_fcc a1 t1 t2
sc_restore_fcsr a2 t1 sc_restore_fcsr a2 t1
li.w a0, 0 # success li.w a0, 0 # success
jirl zero, ra, 0 jr ra
SYM_FUNC_END(_restore_fp_context) SYM_FUNC_END(_restore_fp_context)
SYM_FUNC_START(fault) SYM_FUNC_START(fault)
li.w a0, -EFAULT # failure li.w a0, -EFAULT # failure
jirl zero, ra, 0 jr ra
SYM_FUNC_END(fault) SYM_FUNC_END(fault)
...@@ -28,7 +28,7 @@ SYM_FUNC_START(__arch_cpu_idle) ...@@ -28,7 +28,7 @@ SYM_FUNC_START(__arch_cpu_idle)
nop nop
idle 0 idle 0
/* end of rollback region */ /* end of rollback region */
1: jirl zero, ra, 0 1: jr ra
SYM_FUNC_END(__arch_cpu_idle) SYM_FUNC_END(__arch_cpu_idle)
SYM_FUNC_START(handle_vint) SYM_FUNC_START(handle_vint)
...@@ -91,5 +91,5 @@ SYM_FUNC_END(except_vec_cex) ...@@ -91,5 +91,5 @@ SYM_FUNC_END(except_vec_cex)
SYM_FUNC_START(handle_sys) SYM_FUNC_START(handle_sys)
la.abs t0, handle_syscall la.abs t0, handle_syscall
jirl zero, t0, 0 jr t0
SYM_FUNC_END(handle_sys) SYM_FUNC_END(handle_sys)
...@@ -32,7 +32,7 @@ SYM_CODE_START(kernel_entry) # kernel entry point ...@@ -32,7 +32,7 @@ SYM_CODE_START(kernel_entry) # kernel entry point
/* We might not get launched at the address the kernel is linked to, /* We might not get launched at the address the kernel is linked to,
so we jump there. */ so we jump there. */
la.abs t0, 0f la.abs t0, 0f
jirl zero, t0, 0 jr t0
0: 0:
la t0, __bss_start # clear .bss la t0, __bss_start # clear .bss
st.d zero, t0, 0 st.d zero, t0, 0
...@@ -50,7 +50,7 @@ SYM_CODE_START(kernel_entry) # kernel entry point ...@@ -50,7 +50,7 @@ SYM_CODE_START(kernel_entry) # kernel entry point
/* KSave3 used for percpu base, initialized as 0 */ /* KSave3 used for percpu base, initialized as 0 */
csrwr zero, PERCPU_BASE_KS csrwr zero, PERCPU_BASE_KS
/* GPR21 used for percpu base (runtime), initialized as 0 */ /* GPR21 used for percpu base (runtime), initialized as 0 */
or u0, zero, zero move u0, zero
la tp, init_thread_union la tp, init_thread_union
/* Set the SP after an empty pt_regs. */ /* Set the SP after an empty pt_regs. */
...@@ -86,7 +86,7 @@ SYM_CODE_START(smpboot_entry) ...@@ -86,7 +86,7 @@ SYM_CODE_START(smpboot_entry)
ld.d tp, t0, CPU_BOOT_TINFO ld.d tp, t0, CPU_BOOT_TINFO
la.abs t0, 0f la.abs t0, 0f
jirl zero, t0, 0 jr t0
0: 0:
bl start_secondary bl start_secondary
SYM_CODE_END(smpboot_entry) SYM_CODE_END(smpboot_entry)
......
...@@ -193,7 +193,7 @@ static int fpr_set(struct task_struct *target, ...@@ -193,7 +193,7 @@ static int fpr_set(struct task_struct *target,
const void *kbuf, const void __user *ubuf) const void *kbuf, const void __user *ubuf)
{ {
const int fcc_start = NUM_FPU_REGS * sizeof(elf_fpreg_t); const int fcc_start = NUM_FPU_REGS * sizeof(elf_fpreg_t);
const int fcc_end = fcc_start + sizeof(u64); const int fcsr_start = fcc_start + sizeof(u64);
int err; int err;
BUG_ON(count % sizeof(elf_fpreg_t)); BUG_ON(count % sizeof(elf_fpreg_t));
...@@ -209,10 +209,12 @@ static int fpr_set(struct task_struct *target, ...@@ -209,10 +209,12 @@ static int fpr_set(struct task_struct *target,
if (err) if (err)
return err; return err;
if (count > 0)
err |= user_regset_copyin(&pos, &count, &kbuf, &ubuf, err |= user_regset_copyin(&pos, &count, &kbuf, &ubuf,
&target->thread.fpu.fcc, &target->thread.fpu.fcc, fcc_start,
fcc_start, fcc_end); fcc_start + sizeof(u64));
err |= user_regset_copyin(&pos, &count, &kbuf, &ubuf,
&target->thread.fpu.fcsr, fcsr_start,
fcsr_start + sizeof(u32));
return err; return err;
} }
......
...@@ -13,7 +13,6 @@ ...@@ -13,7 +13,6 @@
#include <linux/console.h> #include <linux/console.h>
#include <acpi/reboot.h> #include <acpi/reboot.h>
#include <asm/compiler.h>
#include <asm/idle.h> #include <asm/idle.h>
#include <asm/loongarch.h> #include <asm/loongarch.h>
#include <asm/reboot.h> #include <asm/reboot.h>
......
...@@ -126,7 +126,7 @@ static void __init parse_bios_table(const struct dmi_header *dm) ...@@ -126,7 +126,7 @@ static void __init parse_bios_table(const struct dmi_header *dm)
char *dmi_data = (char *)dm; char *dmi_data = (char *)dm;
bios_extern = *(dmi_data + SMBIOS_BIOSEXTERN_OFFSET); bios_extern = *(dmi_data + SMBIOS_BIOSEXTERN_OFFSET);
b_info.bios_size = *(dmi_data + SMBIOS_BIOSSIZE_OFFSET); b_info.bios_size = (*(dmi_data + SMBIOS_BIOSSIZE_OFFSET) + 1) << 6;
if (bios_extern & LOONGSON_EFI_ENABLE) if (bios_extern & LOONGSON_EFI_ENABLE)
set_bit(EFI_BOOT, &efi.flags); set_bit(EFI_BOOT, &efi.flags);
......
...@@ -278,116 +278,29 @@ void loongson3_cpu_die(unsigned int cpu) ...@@ -278,116 +278,29 @@ void loongson3_cpu_die(unsigned int cpu)
mb(); mb();
} }
/* void play_dead(void)
* The target CPU should go to XKPRANGE (uncached area) and flush
* ICache/DCache/VCache before the control CPU can safely disable its clock.
*/
static void loongson3_play_dead(int *state_addr)
{ {
register int val; register uint64_t addr;
register void *addr;
register void (*init_fn)(void); register void (*init_fn)(void);
__asm__ __volatile__( idle_task_exit();
" li.d %[addr], 0x8000000000000000\n"
"1: cacop 0x8, %[addr], 0 \n" /* flush ICache */
" cacop 0x8, %[addr], 1 \n"
" cacop 0x8, %[addr], 2 \n"
" cacop 0x8, %[addr], 3 \n"
" cacop 0x9, %[addr], 0 \n" /* flush DCache */
" cacop 0x9, %[addr], 1 \n"
" cacop 0x9, %[addr], 2 \n"
" cacop 0x9, %[addr], 3 \n"
" addi.w %[sets], %[sets], -1 \n"
" addi.d %[addr], %[addr], 0x40 \n"
" bnez %[sets], 1b \n"
" li.d %[addr], 0x8000000000000000\n"
"2: cacop 0xa, %[addr], 0 \n" /* flush VCache */
" cacop 0xa, %[addr], 1 \n"
" cacop 0xa, %[addr], 2 \n"
" cacop 0xa, %[addr], 3 \n"
" cacop 0xa, %[addr], 4 \n"
" cacop 0xa, %[addr], 5 \n"
" cacop 0xa, %[addr], 6 \n"
" cacop 0xa, %[addr], 7 \n"
" cacop 0xa, %[addr], 8 \n"
" cacop 0xa, %[addr], 9 \n"
" cacop 0xa, %[addr], 10 \n"
" cacop 0xa, %[addr], 11 \n"
" cacop 0xa, %[addr], 12 \n"
" cacop 0xa, %[addr], 13 \n"
" cacop 0xa, %[addr], 14 \n"
" cacop 0xa, %[addr], 15 \n"
" addi.w %[vsets], %[vsets], -1 \n"
" addi.d %[addr], %[addr], 0x40 \n"
" bnez %[vsets], 2b \n"
" li.w %[val], 0x7 \n" /* *state_addr = CPU_DEAD; */
" st.w %[val], %[state_addr], 0 \n"
" dbar 0 \n"
" cacop 0x11, %[state_addr], 0 \n" /* flush entry of *state_addr */
: [addr] "=&r" (addr), [val] "=&r" (val)
: [state_addr] "r" (state_addr),
[sets] "r" (cpu_data[smp_processor_id()].dcache.sets),
[vsets] "r" (cpu_data[smp_processor_id()].vcache.sets));
local_irq_enable(); local_irq_enable();
change_csr_ecfg(ECFG0_IM, ECFGF_IPI); set_csr_ecfg(ECFGF_IPI);
__this_cpu_write(cpu_state, CPU_DEAD);
__smp_mb();
do {
__asm__ __volatile__("idle 0\n\t");
addr = iocsr_read64(LOONGARCH_IOCSR_MBUF0);
} while (addr == 0);
__asm__ __volatile__( init_fn = (void *)TO_CACHE(addr);
" idle 0 \n" iocsr_write32(0xffffffff, LOONGARCH_IOCSR_IPI_CLEAR);
" li.w $t0, 0x1020 \n"
" iocsrrd.d %[init_fn], $t0 \n" /* Get init PC */
: [init_fn] "=&r" (addr)
: /* No Input */
: "a0");
init_fn = __va(addr);
init_fn(); init_fn();
unreachable(); unreachable();
} }
void play_dead(void)
{
int *state_addr;
unsigned int cpu = smp_processor_id();
void (*play_dead_uncached)(int *s);
idle_task_exit();
play_dead_uncached = (void *)TO_UNCACHE(__pa((unsigned long)loongson3_play_dead));
state_addr = &per_cpu(cpu_state, cpu);
mb();
play_dead_uncached(state_addr);
}
static int loongson3_enable_clock(unsigned int cpu)
{
uint64_t core_id = cpu_data[cpu].core;
uint64_t package_id = cpu_data[cpu].package;
LOONGSON_FREQCTRL(package_id) |= 1 << (core_id * 4 + 3);
return 0;
}
static int loongson3_disable_clock(unsigned int cpu)
{
uint64_t core_id = cpu_data[cpu].core;
uint64_t package_id = cpu_data[cpu].package;
LOONGSON_FREQCTRL(package_id) &= ~(1 << (core_id * 4 + 3));
return 0;
}
static int register_loongson3_notifier(void)
{
return cpuhp_setup_state_nocalls(CPUHP_LOONGARCH_SOC_PREPARE,
"loongarch/loongson:prepare",
loongson3_enable_clock,
loongson3_disable_clock);
}
early_initcall(register_loongson3_notifier);
#endif #endif
/* /*
......
...@@ -32,7 +32,7 @@ SYM_FUNC_START(__clear_user) ...@@ -32,7 +32,7 @@ SYM_FUNC_START(__clear_user)
1: st.b zero, a0, 0 1: st.b zero, a0, 0
addi.d a0, a0, 1 addi.d a0, a0, 1
addi.d a1, a1, -1 addi.d a1, a1, -1
bgt a1, zero, 1b bgtz a1, 1b
2: move a0, a1 2: move a0, a1
jr ra jr ra
......
...@@ -35,7 +35,7 @@ SYM_FUNC_START(__copy_user) ...@@ -35,7 +35,7 @@ SYM_FUNC_START(__copy_user)
addi.d a0, a0, 1 addi.d a0, a0, 1
addi.d a1, a1, 1 addi.d a1, a1, 1
addi.d a2, a2, -1 addi.d a2, a2, -1
bgt a2, zero, 1b bgtz a2, 1b
3: move a0, a2 3: move a0, a2
jr ra jr ra
......
...@@ -7,7 +7,6 @@ ...@@ -7,7 +7,6 @@
#include <linux/smp.h> #include <linux/smp.h>
#include <linux/timex.h> #include <linux/timex.h>
#include <asm/compiler.h>
#include <asm/processor.h> #include <asm/processor.h>
void __delay(unsigned long cycles) void __delay(unsigned long cycles)
......
...@@ -32,7 +32,7 @@ SYM_FUNC_START(clear_page) ...@@ -32,7 +32,7 @@ SYM_FUNC_START(clear_page)
st.d zero, a0, -8 st.d zero, a0, -8
bne t0, a0, 1b bne t0, a0, 1b
jirl $r0, ra, 0 jr ra
SYM_FUNC_END(clear_page) SYM_FUNC_END(clear_page)
EXPORT_SYMBOL(clear_page) EXPORT_SYMBOL(clear_page)
...@@ -79,6 +79,6 @@ SYM_FUNC_START(copy_page) ...@@ -79,6 +79,6 @@ SYM_FUNC_START(copy_page)
st.d t7, a0, -8 st.d t7, a0, -8
bne t8, a0, 1b bne t8, a0, 1b
jirl $r0, ra, 0 jr ra
SYM_FUNC_END(copy_page) SYM_FUNC_END(copy_page)
EXPORT_SYMBOL(copy_page) EXPORT_SYMBOL(copy_page)
...@@ -47,7 +47,7 @@ SYM_FUNC_START(handle_tlb_load) ...@@ -47,7 +47,7 @@ SYM_FUNC_START(handle_tlb_load)
* The vmalloc handling is not in the hotpath. * The vmalloc handling is not in the hotpath.
*/ */
csrrd t0, LOONGARCH_CSR_BADV csrrd t0, LOONGARCH_CSR_BADV
blt t0, $r0, vmalloc_load bltz t0, vmalloc_load
csrrd t1, LOONGARCH_CSR_PGDL csrrd t1, LOONGARCH_CSR_PGDL
vmalloc_done_load: vmalloc_done_load:
...@@ -80,7 +80,7 @@ vmalloc_done_load: ...@@ -80,7 +80,7 @@ vmalloc_done_load:
* see if we need to jump to huge tlb processing. * see if we need to jump to huge tlb processing.
*/ */
andi t0, ra, _PAGE_HUGE andi t0, ra, _PAGE_HUGE
bne t0, $r0, tlb_huge_update_load bnez t0, tlb_huge_update_load
csrrd t0, LOONGARCH_CSR_BADV csrrd t0, LOONGARCH_CSR_BADV
srli.d t0, t0, (PAGE_SHIFT + PTE_ORDER) srli.d t0, t0, (PAGE_SHIFT + PTE_ORDER)
...@@ -100,12 +100,12 @@ smp_pgtable_change_load: ...@@ -100,12 +100,12 @@ smp_pgtable_change_load:
srli.d ra, t0, _PAGE_PRESENT_SHIFT srli.d ra, t0, _PAGE_PRESENT_SHIFT
andi ra, ra, 1 andi ra, ra, 1
beq ra, $r0, nopage_tlb_load beqz ra, nopage_tlb_load
ori t0, t0, _PAGE_VALID ori t0, t0, _PAGE_VALID
#ifdef CONFIG_SMP #ifdef CONFIG_SMP
sc.d t0, t1, 0 sc.d t0, t1, 0
beq t0, $r0, smp_pgtable_change_load beqz t0, smp_pgtable_change_load
#else #else
st.d t0, t1, 0 st.d t0, t1, 0
#endif #endif
...@@ -139,23 +139,23 @@ tlb_huge_update_load: ...@@ -139,23 +139,23 @@ tlb_huge_update_load:
#endif #endif
srli.d ra, t0, _PAGE_PRESENT_SHIFT srli.d ra, t0, _PAGE_PRESENT_SHIFT
andi ra, ra, 1 andi ra, ra, 1
beq ra, $r0, nopage_tlb_load beqz ra, nopage_tlb_load
tlbsrch tlbsrch
ori t0, t0, _PAGE_VALID ori t0, t0, _PAGE_VALID
#ifdef CONFIG_SMP #ifdef CONFIG_SMP
sc.d t0, t1, 0 sc.d t0, t1, 0
beq t0, $r0, tlb_huge_update_load beqz t0, tlb_huge_update_load
ld.d t0, t1, 0 ld.d t0, t1, 0
#else #else
st.d t0, t1, 0 st.d t0, t1, 0
#endif #endif
addu16i.d t1, $r0, -(CSR_TLBIDX_EHINV >> 16) addu16i.d t1, zero, -(CSR_TLBIDX_EHINV >> 16)
addi.d ra, t1, 0 addi.d ra, t1, 0
csrxchg ra, t1, LOONGARCH_CSR_TLBIDX csrxchg ra, t1, LOONGARCH_CSR_TLBIDX
tlbwr tlbwr
csrxchg $r0, t1, LOONGARCH_CSR_TLBIDX csrxchg zero, t1, LOONGARCH_CSR_TLBIDX
/* /*
* A huge PTE describes an area the size of the * A huge PTE describes an area the size of the
...@@ -178,27 +178,27 @@ tlb_huge_update_load: ...@@ -178,27 +178,27 @@ tlb_huge_update_load:
addi.d t0, ra, 0 addi.d t0, ra, 0
/* Convert to entrylo1 */ /* Convert to entrylo1 */
addi.d t1, $r0, 1 addi.d t1, zero, 1
slli.d t1, t1, (HPAGE_SHIFT - 1) slli.d t1, t1, (HPAGE_SHIFT - 1)
add.d t0, t0, t1 add.d t0, t0, t1
csrwr t0, LOONGARCH_CSR_TLBELO1 csrwr t0, LOONGARCH_CSR_TLBELO1
/* Set huge page tlb entry size */ /* Set huge page tlb entry size */
addu16i.d t0, $r0, (CSR_TLBIDX_PS >> 16) addu16i.d t0, zero, (CSR_TLBIDX_PS >> 16)
addu16i.d t1, $r0, (PS_HUGE_SIZE << (CSR_TLBIDX_PS_SHIFT - 16)) addu16i.d t1, zero, (PS_HUGE_SIZE << (CSR_TLBIDX_PS_SHIFT - 16))
csrxchg t1, t0, LOONGARCH_CSR_TLBIDX csrxchg t1, t0, LOONGARCH_CSR_TLBIDX
tlbfill tlbfill
addu16i.d t0, $r0, (CSR_TLBIDX_PS >> 16) addu16i.d t0, zero, (CSR_TLBIDX_PS >> 16)
addu16i.d t1, $r0, (PS_DEFAULT_SIZE << (CSR_TLBIDX_PS_SHIFT - 16)) addu16i.d t1, zero, (PS_DEFAULT_SIZE << (CSR_TLBIDX_PS_SHIFT - 16))
csrxchg t1, t0, LOONGARCH_CSR_TLBIDX csrxchg t1, t0, LOONGARCH_CSR_TLBIDX
nopage_tlb_load: nopage_tlb_load:
dbar 0 dbar 0
csrrd ra, EXCEPTION_KS2 csrrd ra, EXCEPTION_KS2
la.abs t0, tlb_do_page_fault_0 la.abs t0, tlb_do_page_fault_0
jirl $r0, t0, 0 jr t0
SYM_FUNC_END(handle_tlb_load) SYM_FUNC_END(handle_tlb_load)
SYM_FUNC_START(handle_tlb_store) SYM_FUNC_START(handle_tlb_store)
...@@ -210,7 +210,7 @@ SYM_FUNC_START(handle_tlb_store) ...@@ -210,7 +210,7 @@ SYM_FUNC_START(handle_tlb_store)
* The vmalloc handling is not in the hotpath. * The vmalloc handling is not in the hotpath.
*/ */
csrrd t0, LOONGARCH_CSR_BADV csrrd t0, LOONGARCH_CSR_BADV
blt t0, $r0, vmalloc_store bltz t0, vmalloc_store
csrrd t1, LOONGARCH_CSR_PGDL csrrd t1, LOONGARCH_CSR_PGDL
vmalloc_done_store: vmalloc_done_store:
...@@ -244,7 +244,7 @@ vmalloc_done_store: ...@@ -244,7 +244,7 @@ vmalloc_done_store:
* see if we need to jump to huge tlb processing. * see if we need to jump to huge tlb processing.
*/ */
andi t0, ra, _PAGE_HUGE andi t0, ra, _PAGE_HUGE
bne t0, $r0, tlb_huge_update_store bnez t0, tlb_huge_update_store
csrrd t0, LOONGARCH_CSR_BADV csrrd t0, LOONGARCH_CSR_BADV
srli.d t0, t0, (PAGE_SHIFT + PTE_ORDER) srli.d t0, t0, (PAGE_SHIFT + PTE_ORDER)
...@@ -265,12 +265,12 @@ smp_pgtable_change_store: ...@@ -265,12 +265,12 @@ smp_pgtable_change_store:
srli.d ra, t0, _PAGE_PRESENT_SHIFT srli.d ra, t0, _PAGE_PRESENT_SHIFT
andi ra, ra, ((_PAGE_PRESENT | _PAGE_WRITE) >> _PAGE_PRESENT_SHIFT) andi ra, ra, ((_PAGE_PRESENT | _PAGE_WRITE) >> _PAGE_PRESENT_SHIFT)
xori ra, ra, ((_PAGE_PRESENT | _PAGE_WRITE) >> _PAGE_PRESENT_SHIFT) xori ra, ra, ((_PAGE_PRESENT | _PAGE_WRITE) >> _PAGE_PRESENT_SHIFT)
bne ra, $r0, nopage_tlb_store bnez ra, nopage_tlb_store
ori t0, t0, (_PAGE_VALID | _PAGE_DIRTY | _PAGE_MODIFIED) ori t0, t0, (_PAGE_VALID | _PAGE_DIRTY | _PAGE_MODIFIED)
#ifdef CONFIG_SMP #ifdef CONFIG_SMP
sc.d t0, t1, 0 sc.d t0, t1, 0
beq t0, $r0, smp_pgtable_change_store beqz t0, smp_pgtable_change_store
#else #else
st.d t0, t1, 0 st.d t0, t1, 0
#endif #endif
...@@ -306,24 +306,24 @@ tlb_huge_update_store: ...@@ -306,24 +306,24 @@ tlb_huge_update_store:
srli.d ra, t0, _PAGE_PRESENT_SHIFT srli.d ra, t0, _PAGE_PRESENT_SHIFT
andi ra, ra, ((_PAGE_PRESENT | _PAGE_WRITE) >> _PAGE_PRESENT_SHIFT) andi ra, ra, ((_PAGE_PRESENT | _PAGE_WRITE) >> _PAGE_PRESENT_SHIFT)
xori ra, ra, ((_PAGE_PRESENT | _PAGE_WRITE) >> _PAGE_PRESENT_SHIFT) xori ra, ra, ((_PAGE_PRESENT | _PAGE_WRITE) >> _PAGE_PRESENT_SHIFT)
bne ra, $r0, nopage_tlb_store bnez ra, nopage_tlb_store
tlbsrch tlbsrch
ori t0, t0, (_PAGE_VALID | _PAGE_DIRTY | _PAGE_MODIFIED) ori t0, t0, (_PAGE_VALID | _PAGE_DIRTY | _PAGE_MODIFIED)
#ifdef CONFIG_SMP #ifdef CONFIG_SMP
sc.d t0, t1, 0 sc.d t0, t1, 0
beq t0, $r0, tlb_huge_update_store beqz t0, tlb_huge_update_store
ld.d t0, t1, 0 ld.d t0, t1, 0
#else #else
st.d t0, t1, 0 st.d t0, t1, 0
#endif #endif
addu16i.d t1, $r0, -(CSR_TLBIDX_EHINV >> 16) addu16i.d t1, zero, -(CSR_TLBIDX_EHINV >> 16)
addi.d ra, t1, 0 addi.d ra, t1, 0
csrxchg ra, t1, LOONGARCH_CSR_TLBIDX csrxchg ra, t1, LOONGARCH_CSR_TLBIDX
tlbwr tlbwr
csrxchg $r0, t1, LOONGARCH_CSR_TLBIDX csrxchg zero, t1, LOONGARCH_CSR_TLBIDX
/* /*
* A huge PTE describes an area the size of the * A huge PTE describes an area the size of the
* configured huge page size. This is twice the * configured huge page size. This is twice the
...@@ -345,28 +345,28 @@ tlb_huge_update_store: ...@@ -345,28 +345,28 @@ tlb_huge_update_store:
addi.d t0, ra, 0 addi.d t0, ra, 0
/* Convert to entrylo1 */ /* Convert to entrylo1 */
addi.d t1, $r0, 1 addi.d t1, zero, 1
slli.d t1, t1, (HPAGE_SHIFT - 1) slli.d t1, t1, (HPAGE_SHIFT - 1)
add.d t0, t0, t1 add.d t0, t0, t1
csrwr t0, LOONGARCH_CSR_TLBELO1 csrwr t0, LOONGARCH_CSR_TLBELO1
/* Set huge page tlb entry size */ /* Set huge page tlb entry size */
addu16i.d t0, $r0, (CSR_TLBIDX_PS >> 16) addu16i.d t0, zero, (CSR_TLBIDX_PS >> 16)
addu16i.d t1, $r0, (PS_HUGE_SIZE << (CSR_TLBIDX_PS_SHIFT - 16)) addu16i.d t1, zero, (PS_HUGE_SIZE << (CSR_TLBIDX_PS_SHIFT - 16))
csrxchg t1, t0, LOONGARCH_CSR_TLBIDX csrxchg t1, t0, LOONGARCH_CSR_TLBIDX
tlbfill tlbfill
/* Reset default page size */ /* Reset default page size */
addu16i.d t0, $r0, (CSR_TLBIDX_PS >> 16) addu16i.d t0, zero, (CSR_TLBIDX_PS >> 16)
addu16i.d t1, $r0, (PS_DEFAULT_SIZE << (CSR_TLBIDX_PS_SHIFT - 16)) addu16i.d t1, zero, (PS_DEFAULT_SIZE << (CSR_TLBIDX_PS_SHIFT - 16))
csrxchg t1, t0, LOONGARCH_CSR_TLBIDX csrxchg t1, t0, LOONGARCH_CSR_TLBIDX
nopage_tlb_store: nopage_tlb_store:
dbar 0 dbar 0
csrrd ra, EXCEPTION_KS2 csrrd ra, EXCEPTION_KS2
la.abs t0, tlb_do_page_fault_1 la.abs t0, tlb_do_page_fault_1
jirl $r0, t0, 0 jr t0
SYM_FUNC_END(handle_tlb_store) SYM_FUNC_END(handle_tlb_store)
SYM_FUNC_START(handle_tlb_modify) SYM_FUNC_START(handle_tlb_modify)
...@@ -378,7 +378,7 @@ SYM_FUNC_START(handle_tlb_modify) ...@@ -378,7 +378,7 @@ SYM_FUNC_START(handle_tlb_modify)
* The vmalloc handling is not in the hotpath. * The vmalloc handling is not in the hotpath.
*/ */
csrrd t0, LOONGARCH_CSR_BADV csrrd t0, LOONGARCH_CSR_BADV
blt t0, $r0, vmalloc_modify bltz t0, vmalloc_modify
csrrd t1, LOONGARCH_CSR_PGDL csrrd t1, LOONGARCH_CSR_PGDL
vmalloc_done_modify: vmalloc_done_modify:
...@@ -411,7 +411,7 @@ vmalloc_done_modify: ...@@ -411,7 +411,7 @@ vmalloc_done_modify:
* see if we need to jump to huge tlb processing. * see if we need to jump to huge tlb processing.
*/ */
andi t0, ra, _PAGE_HUGE andi t0, ra, _PAGE_HUGE
bne t0, $r0, tlb_huge_update_modify bnez t0, tlb_huge_update_modify
csrrd t0, LOONGARCH_CSR_BADV csrrd t0, LOONGARCH_CSR_BADV
srli.d t0, t0, (PAGE_SHIFT + PTE_ORDER) srli.d t0, t0, (PAGE_SHIFT + PTE_ORDER)
...@@ -431,12 +431,12 @@ smp_pgtable_change_modify: ...@@ -431,12 +431,12 @@ smp_pgtable_change_modify:
srli.d ra, t0, _PAGE_WRITE_SHIFT srli.d ra, t0, _PAGE_WRITE_SHIFT
andi ra, ra, 1 andi ra, ra, 1
beq ra, $r0, nopage_tlb_modify beqz ra, nopage_tlb_modify
ori t0, t0, (_PAGE_VALID | _PAGE_DIRTY | _PAGE_MODIFIED) ori t0, t0, (_PAGE_VALID | _PAGE_DIRTY | _PAGE_MODIFIED)
#ifdef CONFIG_SMP #ifdef CONFIG_SMP
sc.d t0, t1, 0 sc.d t0, t1, 0
beq t0, $r0, smp_pgtable_change_modify beqz t0, smp_pgtable_change_modify
#else #else
st.d t0, t1, 0 st.d t0, t1, 0
#endif #endif
...@@ -471,14 +471,14 @@ tlb_huge_update_modify: ...@@ -471,14 +471,14 @@ tlb_huge_update_modify:
srli.d ra, t0, _PAGE_WRITE_SHIFT srli.d ra, t0, _PAGE_WRITE_SHIFT
andi ra, ra, 1 andi ra, ra, 1
beq ra, $r0, nopage_tlb_modify beqz ra, nopage_tlb_modify
tlbsrch tlbsrch
ori t0, t0, (_PAGE_VALID | _PAGE_DIRTY | _PAGE_MODIFIED) ori t0, t0, (_PAGE_VALID | _PAGE_DIRTY | _PAGE_MODIFIED)
#ifdef CONFIG_SMP #ifdef CONFIG_SMP
sc.d t0, t1, 0 sc.d t0, t1, 0
beq t0, $r0, tlb_huge_update_modify beqz t0, tlb_huge_update_modify
ld.d t0, t1, 0 ld.d t0, t1, 0
#else #else
st.d t0, t1, 0 st.d t0, t1, 0
...@@ -504,28 +504,28 @@ tlb_huge_update_modify: ...@@ -504,28 +504,28 @@ tlb_huge_update_modify:
addi.d t0, ra, 0 addi.d t0, ra, 0
/* Convert to entrylo1 */ /* Convert to entrylo1 */
addi.d t1, $r0, 1 addi.d t1, zero, 1
slli.d t1, t1, (HPAGE_SHIFT - 1) slli.d t1, t1, (HPAGE_SHIFT - 1)
add.d t0, t0, t1 add.d t0, t0, t1
csrwr t0, LOONGARCH_CSR_TLBELO1 csrwr t0, LOONGARCH_CSR_TLBELO1
/* Set huge page tlb entry size */ /* Set huge page tlb entry size */
addu16i.d t0, $r0, (CSR_TLBIDX_PS >> 16) addu16i.d t0, zero, (CSR_TLBIDX_PS >> 16)
addu16i.d t1, $r0, (PS_HUGE_SIZE << (CSR_TLBIDX_PS_SHIFT - 16)) addu16i.d t1, zero, (PS_HUGE_SIZE << (CSR_TLBIDX_PS_SHIFT - 16))
csrxchg t1, t0, LOONGARCH_CSR_TLBIDX csrxchg t1, t0, LOONGARCH_CSR_TLBIDX
tlbwr tlbwr
/* Reset default page size */ /* Reset default page size */
addu16i.d t0, $r0, (CSR_TLBIDX_PS >> 16) addu16i.d t0, zero, (CSR_TLBIDX_PS >> 16)
addu16i.d t1, $r0, (PS_DEFAULT_SIZE << (CSR_TLBIDX_PS_SHIFT - 16)) addu16i.d t1, zero, (PS_DEFAULT_SIZE << (CSR_TLBIDX_PS_SHIFT - 16))
csrxchg t1, t0, LOONGARCH_CSR_TLBIDX csrxchg t1, t0, LOONGARCH_CSR_TLBIDX
nopage_tlb_modify: nopage_tlb_modify:
dbar 0 dbar 0
csrrd ra, EXCEPTION_KS2 csrrd ra, EXCEPTION_KS2
la.abs t0, tlb_do_page_fault_1 la.abs t0, tlb_do_page_fault_1
jirl $r0, t0, 0 jr t0
SYM_FUNC_END(handle_tlb_modify) SYM_FUNC_END(handle_tlb_modify)
SYM_FUNC_START(handle_tlb_refill) SYM_FUNC_START(handle_tlb_refill)
......
...@@ -130,7 +130,6 @@ enum cpuhp_state { ...@@ -130,7 +130,6 @@ enum cpuhp_state {
CPUHP_ZCOMP_PREPARE, CPUHP_ZCOMP_PREPARE,
CPUHP_TIMERS_PREPARE, CPUHP_TIMERS_PREPARE,
CPUHP_MIPS_SOC_PREPARE, CPUHP_MIPS_SOC_PREPARE,
CPUHP_LOONGARCH_SOC_PREPARE,
CPUHP_BP_PREPARE_DYN, CPUHP_BP_PREPARE_DYN,
CPUHP_BP_PREPARE_DYN_END = CPUHP_BP_PREPARE_DYN + 20, CPUHP_BP_PREPARE_DYN_END = CPUHP_BP_PREPARE_DYN + 20,
CPUHP_BRINGUP_CPU, CPUHP_BRINGUP_CPU,
......
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