calling.h 6.6 KB
Newer Older
1
#include <linux/jump_label.h>
2
#include <asm/unwind_hints.h>
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 43 44 45 46 47 48 49 50 51

 x86 function call convention, 64-bit:
 -------------------------------------
  arguments           |  callee-saved      | extra caller-saved | return
 [callee-clobbered]   |                    | [callee-clobbered] |
 ---------------------------------------------------------------------------
 rdi rsi rdx rcx r8-9 | rbx rbp [*] r12-15 | r10-11             | rax, rdx [**]

 ( rsp is obviously invariant across normal function calls. (gcc can 'merge'
   functions when it sees tail-call optimization possibilities) rflags is
   clobbered. Leftover arguments are passed over the stack frame.)

 [*]  In the frame-pointers case rbp is fixed to the stack frame.

 [**] for struct return values wider than 64 bits the return convention is a
      bit more complex: up to 128 bits width we return small structures
      straight in rax, rdx. For structures larger than that (3 words or
      larger) the caller puts a pointer to an on-stack return struct
      [allocated in the caller's stack frame] into the first argument - i.e.
      into rdi. All other arguments shift up by one in this case.
      Fortunately this case is rare in the kernel.

For 32-bit we have the following conventions - kernel is built with
-mregparm=3 and -freg-struct-return:

 x86 function calling convention, 32-bit:
 ----------------------------------------
  arguments         | callee-saved        | extra caller-saved | return
 [callee-clobbered] |                     | [callee-clobbered] |
 -------------------------------------------------------------------------
 eax edx ecx        | ebx edi esi ebp [*] | <none>             | eax, edx [**]

 ( here too esp is obviously invariant across normal function calls. eflags
   is clobbered. Leftover arguments are passed over the stack frame. )

 [*]  In the frame-pointers case ebp is fixed to the stack frame.

 [**] We build with -freg-struct-return, which on 32-bit means similar
      semantics as on 64-bit: edx can be used for a second return value
      (i.e. covering integer and structure sizes up to 64 bits) - after that
      it gets more complex and more expensive: 3-word or larger struct returns
      get done in the caller's frame and the pointer to the return struct goes
      into regparm0, i.e. eax - the other arguments shift up and the
      function's register parameters degenerate to regparm=2 in essence.

*/

52 53
#ifdef CONFIG_X86_64

54
/*
T
Tao Guo 已提交
55 56
 * 64-bit system call stack frame layout defines and helpers,
 * for assembly code:
57
 */
L
Linus Torvalds 已提交
58

59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91
/* The layout forms the "struct pt_regs" on the stack: */
/*
 * C ABI says these regs are callee-preserved. They aren't saved on kernel entry
 * unless syscall needs a complete, fully filled "struct pt_regs".
 */
#define R15		0*8
#define R14		1*8
#define R13		2*8
#define R12		3*8
#define RBP		4*8
#define RBX		5*8
/* These regs are callee-clobbered. Always saved on kernel entry. */
#define R11		6*8
#define R10		7*8
#define R9		8*8
#define R8		9*8
#define RAX		10*8
#define RCX		11*8
#define RDX		12*8
#define RSI		13*8
#define RDI		14*8
/*
 * On syscall entry, this is syscall#. On CPU exception, this is error code.
 * On hw interrupt, it's IRQ number:
 */
#define ORIG_RAX	15*8
/* Return frame for iretq */
#define RIP		16*8
#define CS		17*8
#define EFLAGS		18*8
#define RSP		19*8
#define SS		20*8

92 93
#define SIZEOF_PTREGS	21*8

94 95
	.macro ALLOC_PT_GPREGS_ON_STACK
	addq	$-(15*8), %rsp
96
	.endm
97

98 99
	.macro SAVE_C_REGS_HELPER offset=0 rax=1 rcx=1 r8910=1 r11=1
	.if \r11
100
	movq %r11, 6*8+\offset(%rsp)
101 102
	.endif
	.if \r8910
103 104 105
	movq %r10, 7*8+\offset(%rsp)
	movq %r9,  8*8+\offset(%rsp)
	movq %r8,  9*8+\offset(%rsp)
106
	.endif
107
	.if \rax
108
	movq %rax, 10*8+\offset(%rsp)
109 110
	.endif
	.if \rcx
111
	movq %rcx, 11*8+\offset(%rsp)
L
Linus Torvalds 已提交
112
	.endif
113 114 115
	movq %rdx, 12*8+\offset(%rsp)
	movq %rsi, 13*8+\offset(%rsp)
	movq %rdi, 14*8+\offset(%rsp)
116
	UNWIND_HINT_REGS offset=\offset extra=0
117 118
	.endm
	.macro SAVE_C_REGS offset=0
119
	SAVE_C_REGS_HELPER \offset, 1, 1, 1, 1
120 121
	.endm
	.macro SAVE_C_REGS_EXCEPT_RAX_RCX offset=0
122
	SAVE_C_REGS_HELPER \offset, 0, 0, 1, 1
123 124
	.endm
	.macro SAVE_C_REGS_EXCEPT_R891011
125
	SAVE_C_REGS_HELPER 0, 1, 1, 0, 0
126 127
	.endm
	.macro SAVE_C_REGS_EXCEPT_RCX_R891011
128 129 130 131
	SAVE_C_REGS_HELPER 0, 1, 0, 0, 0
	.endm
	.macro SAVE_C_REGS_EXCEPT_RAX_RCX_R11
	SAVE_C_REGS_HELPER 0, 0, 0, 1, 0
132
	.endm
133

134
	.macro SAVE_EXTRA_REGS offset=0
135 136 137 138 139 140
	movq %r15, 0*8+\offset(%rsp)
	movq %r14, 1*8+\offset(%rsp)
	movq %r13, 2*8+\offset(%rsp)
	movq %r12, 3*8+\offset(%rsp)
	movq %rbp, 4*8+\offset(%rsp)
	movq %rbx, 5*8+\offset(%rsp)
141
	UNWIND_HINT_REGS offset=\offset
142
	.endm
L
Linus Torvalds 已提交
143

144
	.macro RESTORE_EXTRA_REGS offset=0
145 146 147 148 149 150
	movq 0*8+\offset(%rsp), %r15
	movq 1*8+\offset(%rsp), %r14
	movq 2*8+\offset(%rsp), %r13
	movq 3*8+\offset(%rsp), %r12
	movq 4*8+\offset(%rsp), %rbp
	movq 5*8+\offset(%rsp), %rbx
151
	UNWIND_HINT_REGS offset=\offset extra=0
152
	.endm
153

154
	.macro RESTORE_C_REGS_HELPER rstor_rax=1, rstor_rcx=1, rstor_r11=1, rstor_r8910=1, rstor_rdx=1
155
	.if \rstor_r11
156
	movq 6*8(%rsp), %r11
L
Linus Torvalds 已提交
157
	.endif
158
	.if \rstor_r8910
159 160 161
	movq 7*8(%rsp), %r10
	movq 8*8(%rsp), %r9
	movq 9*8(%rsp), %r8
L
Linus Torvalds 已提交
162
	.endif
163
	.if \rstor_rax
164
	movq 10*8(%rsp), %rax
L
Linus Torvalds 已提交
165
	.endif
166
	.if \rstor_rcx
167
	movq 11*8(%rsp), %rcx
L
Linus Torvalds 已提交
168
	.endif
169
	.if \rstor_rdx
170
	movq 12*8(%rsp), %rdx
L
Linus Torvalds 已提交
171
	.endif
172 173
	movq 13*8(%rsp), %rsi
	movq 14*8(%rsp), %rdi
174
	UNWIND_HINT_IRET_REGS offset=16*8
175
	.endm
176 177
	.macro RESTORE_C_REGS
	RESTORE_C_REGS_HELPER 1,1,1,1,1
L
Linus Torvalds 已提交
178
	.endm
179 180
	.macro RESTORE_C_REGS_EXCEPT_RAX
	RESTORE_C_REGS_HELPER 0,1,1,1,1
181
	.endm
182 183
	.macro RESTORE_C_REGS_EXCEPT_RCX
	RESTORE_C_REGS_HELPER 1,0,1,1,1
L
Linus Torvalds 已提交
184
	.endm
185 186 187
	.macro RESTORE_C_REGS_EXCEPT_R11
	RESTORE_C_REGS_HELPER 1,1,0,1,1
	.endm
188 189 190
	.macro RESTORE_C_REGS_EXCEPT_RCX_R11
	RESTORE_C_REGS_HELPER 1,0,0,1,1
	.endm
191

192
	.macro REMOVE_PT_GPREGS_FROM_STACK addskip=0
193
	subq $-(15*8+\addskip), %rsp
L
Linus Torvalds 已提交
194 195 196 197 198
	.endm

	.macro icebp
	.byte 0xf1
	.endm
199

200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219
/*
 * This is a sneaky trick to help the unwinder find pt_regs on the stack.  The
 * frame pointer is replaced with an encoded pointer to pt_regs.  The encoding
 * is just setting the LSB, which makes it an invalid stack address and is also
 * a signal to the unwinder that it's a pt_regs pointer in disguise.
 *
 * NOTE: This macro must be used *after* SAVE_EXTRA_REGS because it corrupts
 * the original rbp.
 */
.macro ENCODE_FRAME_POINTER ptregs_offset=0
#ifdef CONFIG_FRAME_POINTER
	.if \ptregs_offset
		leaq \ptregs_offset(%rsp), %rbp
	.else
		mov %rsp, %rbp
	.endif
	orq	$0x1, %rbp
#endif
.endm

220 221
#endif /* CONFIG_X86_64 */

222 223 224 225 226 227 228 229 230 231 232 233 234
/*
 * This does 'call enter_from_user_mode' unless we can avoid it based on
 * kernel config or using the static jump infrastructure.
 */
.macro CALL_enter_from_user_mode
#ifdef CONFIG_CONTEXT_TRACKING
#ifdef HAVE_JUMP_LABEL
	STATIC_JUMP_IF_FALSE .Lafter_call_\@, context_tracking_enabled, def=0
#endif
	call enter_from_user_mode
.Lafter_call_\@:
#endif
.endm