aboutsummaryrefslogtreecommitdiff
path: root/arch/arm/include/asm/compiler.h
blob: 3cb1f163e634f5aba62f82e1610f49af78deb8b1 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
/* SPDX-License-Identifier: GPL-2.0 */
#ifndef __ASM_ARM_COMPILER_H
#define __ASM_ARM_COMPILER_H

/*
 * This is used to ensure the compiler did actually allocate the register we
 * asked it for some inline assembly sequences.  Apparently we can't trust
 * the compiler from one version to another so a bit of paranoia won't hurt.
 * This string is meant to be concatenated with the inline asm string and
 * will cause compilation to stop on mismatch.
 * (for details, see gcc PR 15089)
 * For compatibility with clang, we have to specifically take the equivalence
 * of 'r11' <-> 'fp' and 'r12' <-> 'ip' into account as well.
 */
#define __asmeq(x, y)				\
	".ifnc " x "," y "; "			\
	  ".ifnc " x y ",fpr11; " 		\
	    ".ifnc " x y ",r11fp; "		\
	      ".ifnc " x y ",ipr12; " 		\
	        ".ifnc " x y ",r12ip; "		\
	          ".err; "			\
	        ".endif; "			\
	      ".endif; "			\
	    ".endif; "				\
	  ".endif; "				\
	".endif\n\t"

/*
 * This is used for calling exported symbols from inline assembly code.
 */
#if defined(MODULE) && defined(CONFIG_MODULES_USE_LONG_CALLS)
#define __asmbl(cond, reg, target) \
	"movw	" reg ", #:lower16:" target "\n\t" \
	"movt	" reg ", #:upper16:" target "\n\t" \
	"blx" cond "	" reg "\n\t"
#define __asmbl_clobber(reg)	,reg
#else
#define __asmbl(cond, reg, target) "bl" cond "	" target"\n\t"
#define __asmbl_clobber(reg)
#endif

#endif /* __ASM_ARM_COMPILER_H */