1d2912cb1SThomas Gleixner /* SPDX-License-Identifier: GPL-2.0-only */
24baa9922SRussell King /*
34baa9922SRussell King * arch/arm/include/asm/processor.h
44baa9922SRussell King *
54baa9922SRussell King * Copyright (C) 1995-1999 Russell King
64baa9922SRussell King */
74baa9922SRussell King
84baa9922SRussell King #ifndef __ASM_ARM_PROCESSOR_H
94baa9922SRussell King #define __ASM_ARM_PROCESSOR_H
104baa9922SRussell King
114baa9922SRussell King #ifdef __KERNEL__
124baa9922SRussell King
13864232faSWill Deacon #include <asm/hw_breakpoint.h>
144baa9922SRussell King #include <asm/ptrace.h>
154baa9922SRussell King #include <asm/types.h>
1627a84793SWill Deacon #include <asm/unified.h>
1778c85161SVincenzo Frascino #include <asm/vdso/processor.h>
184baa9922SRussell King
194baa9922SRussell King #ifdef __KERNEL__
20794baba6SLennert Buytenhek #define STACK_TOP ((current->personality & ADDR_LIMIT_32BIT) ? \
214baa9922SRussell King TASK_SIZE : TASK_SIZE_26)
224baa9922SRussell King #define STACK_TOP_MAX TASK_SIZE
234baa9922SRussell King #endif
244baa9922SRussell King
254baa9922SRussell King struct debug_info {
26864232faSWill Deacon #ifdef CONFIG_HAVE_HW_BREAKPOINT
27864232faSWill Deacon struct perf_event *hbp[ARM_MAX_HBP_SLOTS];
28864232faSWill Deacon #endif
294baa9922SRussell King };
304baa9922SRussell King
314baa9922SRussell King struct thread_struct {
324baa9922SRussell King /* fault info */
334baa9922SRussell King unsigned long address;
344baa9922SRussell King unsigned long trap_no;
354baa9922SRussell King unsigned long error_code;
364baa9922SRussell King /* debugging */
374baa9922SRussell King struct debug_info debug;
384baa9922SRussell King };
394baa9922SRussell King
4008626a60SKees Cook /*
4108626a60SKees Cook * Everything usercopied to/from thread_struct is statically-sized, so
4208626a60SKees Cook * no hardened usercopy whitelist is needed.
4308626a60SKees Cook */
arch_thread_struct_whitelist(unsigned long * offset,unsigned long * size)4408626a60SKees Cook static inline void arch_thread_struct_whitelist(unsigned long *offset,
4508626a60SKees Cook unsigned long *size)
4608626a60SKees Cook {
4708626a60SKees Cook *offset = *size = 0;
4808626a60SKees Cook }
4908626a60SKees Cook
504baa9922SRussell King #define INIT_THREAD { }
514baa9922SRussell King
524baa9922SRussell King #define start_thread(regs,pc,sp) \
534baa9922SRussell King ({ \
545e588114SNicolas Pitre unsigned long r7, r8, r9; \
555e588114SNicolas Pitre \
565e588114SNicolas Pitre if (IS_ENABLED(CONFIG_BINFMT_ELF_FDPIC)) { \
575e588114SNicolas Pitre r7 = regs->ARM_r7; \
585e588114SNicolas Pitre r8 = regs->ARM_r8; \
595e588114SNicolas Pitre r9 = regs->ARM_r9; \
605e588114SNicolas Pitre } \
6159f0cb0fSRussell King memset(regs->uregs, 0, sizeof(regs->uregs)); \
625e588114SNicolas Pitre if (IS_ENABLED(CONFIG_BINFMT_ELF_FDPIC) && \
635e588114SNicolas Pitre current->personality & FDPIC_FUNCPTRS) { \
645e588114SNicolas Pitre regs->ARM_r7 = r7; \
655e588114SNicolas Pitre regs->ARM_r8 = r8; \
665e588114SNicolas Pitre regs->ARM_r9 = r9; \
675e588114SNicolas Pitre regs->ARM_r10 = current->mm->start_data; \
685e588114SNicolas Pitre } else if (!IS_ENABLED(CONFIG_MMU)) \
695e588114SNicolas Pitre regs->ARM_r10 = current->mm->start_data; \
704baa9922SRussell King if (current->personality & ADDR_LIMIT_32BIT) \
714baa9922SRussell King regs->ARM_cpsr = USR_MODE; \
724baa9922SRussell King else \
734baa9922SRussell King regs->ARM_cpsr = USR26_MODE; \
744baa9922SRussell King if (elf_hwcap & HWCAP_THUMB && pc & 1) \
754baa9922SRussell King regs->ARM_cpsr |= PSR_T_BIT; \
7626584853SCatalin Marinas regs->ARM_cpsr |= PSR_ENDSTATE; \
774baa9922SRussell King regs->ARM_pc = pc & ~1; /* pc */ \
784baa9922SRussell King regs->ARM_sp = sp; /* sp */ \
794baa9922SRussell King })
804baa9922SRussell King
814baa9922SRussell King /* Forward declaration, a strange C thing */
824baa9922SRussell King struct task_struct;
834baa9922SRussell King
8442a20f86SKees Cook unsigned long __get_wchan(struct task_struct *p);
854baa9922SRussell King
864baa9922SRussell King #define task_pt_regs(p) \
874baa9922SRussell King ((struct pt_regs *)(THREAD_START_SP + task_stack_page(p)) - 1)
884baa9922SRussell King
894baa9922SRussell King #define KSTK_EIP(tsk) task_pt_regs(tsk)->ARM_pc
904baa9922SRussell King #define KSTK_ESP(tsk) task_pt_regs(tsk)->ARM_sp
914baa9922SRussell King
9227a84793SWill Deacon #ifdef CONFIG_SMP
9327a84793SWill Deacon #define __ALT_SMP_ASM(smp, up) \
9427a84793SWill Deacon "9998: " smp "\n" \
9527a84793SWill Deacon " .pushsection \".alt.smp.init\", \"a\"\n" \
96*9f80ccdaSArd Biesheuvel " .align 2\n" \
97450abd38SArd Biesheuvel " .long 9998b - .\n" \
9827a84793SWill Deacon " " up "\n" \
9927a84793SWill Deacon " .popsection\n"
10027a84793SWill Deacon #else
10127a84793SWill Deacon #define __ALT_SMP_ASM(smp, up) up
10227a84793SWill Deacon #endif
10327a84793SWill Deacon
1044baa9922SRussell King /*
1054baa9922SRussell King * Prefetching support - only ARMv5.
1064baa9922SRussell King */
1074baa9922SRussell King #if __LINUX_ARM_ARCH__ >= 5
1084baa9922SRussell King
1094baa9922SRussell King #define ARCH_HAS_PREFETCH
prefetch(const void * ptr)1104baa9922SRussell King static inline void prefetch(const void *ptr)
1114baa9922SRussell King {
1124baa9922SRussell King __asm__ __volatile__(
11316f719deSNicolas Pitre "pld\t%a0"
114e744dff7SWill Deacon :: "p" (ptr));
1154baa9922SRussell King }
1164baa9922SRussell King
117d8f57aa4SWill Deacon #if __LINUX_ARM_ARCH__ >= 7 && defined(CONFIG_SMP)
1184baa9922SRussell King #define ARCH_HAS_PREFETCHW
prefetchw(const void * ptr)119d8f57aa4SWill Deacon static inline void prefetchw(const void *ptr)
120d8f57aa4SWill Deacon {
121d8f57aa4SWill Deacon __asm__ __volatile__(
122d8f57aa4SWill Deacon ".arch_extension mp\n"
123d8f57aa4SWill Deacon __ALT_SMP_ASM(
12443947b88SStefan Agner "pldw\t%a0",
12543947b88SStefan Agner "pld\t%a0"
126d8f57aa4SWill Deacon )
127d8f57aa4SWill Deacon :: "p" (ptr));
128d8f57aa4SWill Deacon }
129d8f57aa4SWill Deacon #endif
1304baa9922SRussell King #endif
1314baa9922SRussell King
1324baa9922SRussell King #endif
1334baa9922SRussell King
1344baa9922SRussell King #endif /* __ASM_ARM_PROCESSOR_H */
135