ia32entry.S 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514
  1. /*
  2. * Compatibility mode system call entry point for x86-64.
  3. *
  4. * Copyright 2000-2002 Andi Kleen, SuSE Labs.
  5. */
  6. #include <asm/dwarf2.h>
  7. #include <asm/calling.h>
  8. #include <asm/asm-offsets.h>
  9. #include <asm/current.h>
  10. #include <asm/errno.h>
  11. #include <asm/ia32_unistd.h>
  12. #include <asm/thread_info.h>
  13. #include <asm/segment.h>
  14. #include <asm/irqflags.h>
  15. #include <asm/asm.h>
  16. #include <asm/smap.h>
  17. #include <linux/linkage.h>
  18. #include <linux/err.h>
  19. /* Avoid __ASSEMBLER__'ifying <linux/audit.h> just for this. */
  20. #include <linux/elf-em.h>
  21. #define AUDIT_ARCH_I386 (EM_386|__AUDIT_ARCH_LE)
  22. #define __AUDIT_ARCH_LE 0x40000000
  23. #ifndef CONFIG_AUDITSYSCALL
  24. #define sysexit_audit ia32_ret_from_sys_call
  25. #define sysretl_audit ia32_ret_from_sys_call
  26. #endif
  27. .section .entry.text, "ax"
  28. .macro IA32_ARG_FIXUP noebp=0
  29. movl %edi,%r8d
  30. .if \noebp
  31. .else
  32. movl %ebp,%r9d
  33. .endif
  34. xchg %ecx,%esi
  35. movl %ebx,%edi
  36. movl %edx,%edx /* zero extension */
  37. .endm
  38. /* clobbers %eax */
  39. .macro CLEAR_RREGS offset=0, _r9=rax
  40. xorl %eax,%eax
  41. movq %rax,\offset+R11(%rsp)
  42. movq %rax,\offset+R10(%rsp)
  43. movq %\_r9,\offset+R9(%rsp)
  44. movq %rax,\offset+R8(%rsp)
  45. .endm
  46. /*
  47. * Reload arg registers from stack in case ptrace changed them.
  48. * We don't reload %eax because syscall_trace_enter() returned
  49. * the %rax value we should see. Instead, we just truncate that
  50. * value to 32 bits again as we did on entry from user mode.
  51. * If it's a new value set by user_regset during entry tracing,
  52. * this matches the normal truncation of the user-mode value.
  53. * If it's -1 to make us punt the syscall, then (u32)-1 is still
  54. * an appropriately invalid value.
  55. */
  56. .macro LOAD_ARGS32 offset, _r9=0
  57. .if \_r9
  58. movl \offset+16(%rsp),%r9d
  59. .endif
  60. movl \offset+40(%rsp),%ecx
  61. movl \offset+48(%rsp),%edx
  62. movl \offset+56(%rsp),%esi
  63. movl \offset+64(%rsp),%edi
  64. movl %eax,%eax /* zero extension */
  65. .endm
  66. .macro CFI_STARTPROC32 simple
  67. CFI_STARTPROC \simple
  68. CFI_UNDEFINED r8
  69. CFI_UNDEFINED r9
  70. CFI_UNDEFINED r10
  71. CFI_UNDEFINED r11
  72. CFI_UNDEFINED r12
  73. CFI_UNDEFINED r13
  74. CFI_UNDEFINED r14
  75. CFI_UNDEFINED r15
  76. .endm
  77. #ifdef CONFIG_PARAVIRT
  78. ENTRY(native_usergs_sysret32)
  79. swapgs
  80. sysretl
  81. ENDPROC(native_usergs_sysret32)
  82. ENTRY(native_irq_enable_sysexit)
  83. swapgs
  84. sti
  85. sysexit
  86. ENDPROC(native_irq_enable_sysexit)
  87. #endif
  88. /*
  89. * 32bit SYSENTER instruction entry.
  90. *
  91. * Arguments:
  92. * %eax System call number.
  93. * %ebx Arg1
  94. * %ecx Arg2
  95. * %edx Arg3
  96. * %esi Arg4
  97. * %edi Arg5
  98. * %ebp user stack
  99. * 0(%ebp) Arg6
  100. *
  101. * Interrupts off.
  102. *
  103. * This is purely a fast path. For anything complicated we use the int 0x80
  104. * path below. Set up a complete hardware stack frame to share code
  105. * with the int 0x80 path.
  106. */
  107. ENTRY(ia32_sysenter_target)
  108. CFI_STARTPROC32 simple
  109. CFI_SIGNAL_FRAME
  110. CFI_DEF_CFA rsp,0
  111. CFI_REGISTER rsp,rbp
  112. SWAPGS_UNSAFE_STACK
  113. movq PER_CPU_VAR(kernel_stack), %rsp
  114. addq $(KERNEL_STACK_OFFSET),%rsp
  115. /*
  116. * No need to follow this irqs on/off section: the syscall
  117. * disabled irqs, here we enable it straight after entry:
  118. */
  119. ENABLE_INTERRUPTS(CLBR_NONE)
  120. movl %ebp,%ebp /* zero extension */
  121. pushq_cfi $__USER32_DS
  122. /*CFI_REL_OFFSET ss,0*/
  123. pushq_cfi %rbp
  124. CFI_REL_OFFSET rsp,0
  125. pushfq_cfi
  126. /*CFI_REL_OFFSET rflags,0*/
  127. movl TI_sysenter_return+THREAD_INFO(%rsp,3*8-KERNEL_STACK_OFFSET),%r10d
  128. CFI_REGISTER rip,r10
  129. pushq_cfi $__USER32_CS
  130. /*CFI_REL_OFFSET cs,0*/
  131. movl %eax, %eax
  132. pushq_cfi %r10
  133. CFI_REL_OFFSET rip,0
  134. pushq_cfi %rax
  135. cld
  136. SAVE_ARGS 0,1,0
  137. /* no need to do an access_ok check here because rbp has been
  138. 32bit zero extended */
  139. ASM_STAC
  140. 1: movl (%rbp),%ebp
  141. _ASM_EXTABLE(1b,ia32_badarg)
  142. ASM_CLAC
  143. /*
  144. * Sysenter doesn't filter flags, so we need to clear NT
  145. * ourselves. To save a few cycles, we can check whether
  146. * NT was set instead of doing an unconditional popfq.
  147. */
  148. testl $X86_EFLAGS_NT,EFLAGS(%rsp) /* saved EFLAGS match cpu */
  149. jnz sysenter_fix_flags
  150. sysenter_flags_fixed:
  151. orl $TS_COMPAT,TI_status+THREAD_INFO(%rsp,RIP-ARGOFFSET)
  152. testl $_TIF_WORK_SYSCALL_ENTRY,TI_flags+THREAD_INFO(%rsp,RIP-ARGOFFSET)
  153. CFI_REMEMBER_STATE
  154. jnz sysenter_tracesys
  155. cmpq $(IA32_NR_syscalls-1),%rax
  156. ja ia32_badsys
  157. sysenter_do_call:
  158. IA32_ARG_FIXUP
  159. sysenter_dispatch:
  160. call *ia32_sys_call_table(,%rax,8)
  161. movq %rax,RAX-ARGOFFSET(%rsp)
  162. DISABLE_INTERRUPTS(CLBR_NONE)
  163. TRACE_IRQS_OFF
  164. testl $_TIF_ALLWORK_MASK,TI_flags+THREAD_INFO(%rsp,RIP-ARGOFFSET)
  165. jnz sysexit_audit
  166. sysexit_from_sys_call:
  167. andl $~TS_COMPAT,TI_status+THREAD_INFO(%rsp,RIP-ARGOFFSET)
  168. /* clear IF, that popfq doesn't enable interrupts early */
  169. andl $~0x200,EFLAGS-R11(%rsp)
  170. movl RIP-R11(%rsp),%edx /* User %eip */
  171. CFI_REGISTER rip,rdx
  172. RESTORE_ARGS 0,24,0,0,0,0
  173. xorq %r8,%r8
  174. xorq %r9,%r9
  175. xorq %r10,%r10
  176. xorq %r11,%r11
  177. popfq_cfi
  178. /*CFI_RESTORE rflags*/
  179. popq_cfi %rcx /* User %esp */
  180. CFI_REGISTER rsp,rcx
  181. TRACE_IRQS_ON
  182. ENABLE_INTERRUPTS_SYSEXIT32
  183. CFI_RESTORE_STATE
  184. #ifdef CONFIG_AUDITSYSCALL
  185. .macro auditsys_entry_common
  186. movl %esi,%r9d /* 6th arg: 4th syscall arg */
  187. movl %edx,%r8d /* 5th arg: 3rd syscall arg */
  188. /* (already in %ecx) 4th arg: 2nd syscall arg */
  189. movl %ebx,%edx /* 3rd arg: 1st syscall arg */
  190. movl %eax,%esi /* 2nd arg: syscall number */
  191. movl $AUDIT_ARCH_I386,%edi /* 1st arg: audit arch */
  192. call __audit_syscall_entry
  193. movl RAX-ARGOFFSET(%rsp),%eax /* reload syscall number */
  194. cmpq $(IA32_NR_syscalls-1),%rax
  195. ja ia32_badsys
  196. movl %ebx,%edi /* reload 1st syscall arg */
  197. movl RCX-ARGOFFSET(%rsp),%esi /* reload 2nd syscall arg */
  198. movl RDX-ARGOFFSET(%rsp),%edx /* reload 3rd syscall arg */
  199. movl RSI-ARGOFFSET(%rsp),%ecx /* reload 4th syscall arg */
  200. movl RDI-ARGOFFSET(%rsp),%r8d /* reload 5th syscall arg */
  201. .endm
  202. .macro auditsys_exit exit
  203. testl $(_TIF_ALLWORK_MASK & ~_TIF_SYSCALL_AUDIT),TI_flags+THREAD_INFO(%rsp,RIP-ARGOFFSET)
  204. jnz ia32_ret_from_sys_call
  205. TRACE_IRQS_ON
  206. ENABLE_INTERRUPTS(CLBR_NONE)
  207. movl %eax,%esi /* second arg, syscall return value */
  208. cmpl $-MAX_ERRNO,%eax /* is it an error ? */
  209. jbe 1f
  210. movslq %eax, %rsi /* if error sign extend to 64 bits */
  211. 1: setbe %al /* 1 if error, 0 if not */
  212. movzbl %al,%edi /* zero-extend that into %edi */
  213. call __audit_syscall_exit
  214. movq RAX-ARGOFFSET(%rsp),%rax /* reload syscall return value */
  215. movl $(_TIF_ALLWORK_MASK & ~_TIF_SYSCALL_AUDIT),%edi
  216. DISABLE_INTERRUPTS(CLBR_NONE)
  217. TRACE_IRQS_OFF
  218. testl %edi,TI_flags+THREAD_INFO(%rsp,RIP-ARGOFFSET)
  219. jz \exit
  220. CLEAR_RREGS -ARGOFFSET
  221. jmp int_with_check
  222. .endm
  223. sysenter_auditsys:
  224. auditsys_entry_common
  225. movl %ebp,%r9d /* reload 6th syscall arg */
  226. jmp sysenter_dispatch
  227. sysexit_audit:
  228. auditsys_exit sysexit_from_sys_call
  229. #endif
  230. sysenter_fix_flags:
  231. pushq_cfi $(X86_EFLAGS_IF|X86_EFLAGS_FIXED)
  232. popfq_cfi
  233. jmp sysenter_flags_fixed
  234. sysenter_tracesys:
  235. #ifdef CONFIG_AUDITSYSCALL
  236. testl $(_TIF_WORK_SYSCALL_ENTRY & ~_TIF_SYSCALL_AUDIT),TI_flags+THREAD_INFO(%rsp,RIP-ARGOFFSET)
  237. jz sysenter_auditsys
  238. #endif
  239. SAVE_REST
  240. CLEAR_RREGS
  241. movq $-ENOSYS,RAX(%rsp)/* ptrace can change this for a bad syscall */
  242. movq %rsp,%rdi /* &pt_regs -> arg1 */
  243. call syscall_trace_enter
  244. LOAD_ARGS32 ARGOFFSET /* reload args from stack in case ptrace changed it */
  245. RESTORE_REST
  246. cmpq $(IA32_NR_syscalls-1),%rax
  247. ja int_ret_from_sys_call /* sysenter_tracesys has set RAX(%rsp) */
  248. jmp sysenter_do_call
  249. CFI_ENDPROC
  250. ENDPROC(ia32_sysenter_target)
  251. /*
  252. * 32bit SYSCALL instruction entry.
  253. *
  254. * Arguments:
  255. * %eax System call number.
  256. * %ebx Arg1
  257. * %ecx return EIP
  258. * %edx Arg3
  259. * %esi Arg4
  260. * %edi Arg5
  261. * %ebp Arg2 [note: not saved in the stack frame, should not be touched]
  262. * %esp user stack
  263. * 0(%esp) Arg6
  264. *
  265. * Interrupts off.
  266. *
  267. * This is purely a fast path. For anything complicated we use the int 0x80
  268. * path below. Set up a complete hardware stack frame to share code
  269. * with the int 0x80 path.
  270. */
  271. ENTRY(ia32_cstar_target)
  272. CFI_STARTPROC32 simple
  273. CFI_SIGNAL_FRAME
  274. CFI_DEF_CFA rsp,KERNEL_STACK_OFFSET
  275. CFI_REGISTER rip,rcx
  276. /*CFI_REGISTER rflags,r11*/
  277. SWAPGS_UNSAFE_STACK
  278. movl %esp,%r8d
  279. CFI_REGISTER rsp,r8
  280. movq PER_CPU_VAR(kernel_stack),%rsp
  281. /*
  282. * No need to follow this irqs on/off section: the syscall
  283. * disabled irqs and here we enable it straight after entry:
  284. */
  285. ENABLE_INTERRUPTS(CLBR_NONE)
  286. SAVE_ARGS 8,0,0
  287. movl %eax,%eax /* zero extension */
  288. movq %rax,ORIG_RAX-ARGOFFSET(%rsp)
  289. movq %rcx,RIP-ARGOFFSET(%rsp)
  290. CFI_REL_OFFSET rip,RIP-ARGOFFSET
  291. movq %rbp,RCX-ARGOFFSET(%rsp) /* this lies slightly to ptrace */
  292. movl %ebp,%ecx
  293. movq $__USER32_CS,CS-ARGOFFSET(%rsp)
  294. movq $__USER32_DS,SS-ARGOFFSET(%rsp)
  295. movq %r11,EFLAGS-ARGOFFSET(%rsp)
  296. /*CFI_REL_OFFSET rflags,EFLAGS-ARGOFFSET*/
  297. movq %r8,RSP-ARGOFFSET(%rsp)
  298. CFI_REL_OFFSET rsp,RSP-ARGOFFSET
  299. /* no need to do an access_ok check here because r8 has been
  300. 32bit zero extended */
  301. /* hardware stack frame is complete now */
  302. ASM_STAC
  303. 1: movl (%r8),%r9d
  304. _ASM_EXTABLE(1b,ia32_badarg)
  305. ASM_CLAC
  306. orl $TS_COMPAT,TI_status+THREAD_INFO(%rsp,RIP-ARGOFFSET)
  307. testl $_TIF_WORK_SYSCALL_ENTRY,TI_flags+THREAD_INFO(%rsp,RIP-ARGOFFSET)
  308. CFI_REMEMBER_STATE
  309. jnz cstar_tracesys
  310. cmpq $IA32_NR_syscalls-1,%rax
  311. ja ia32_badsys
  312. cstar_do_call:
  313. IA32_ARG_FIXUP 1
  314. cstar_dispatch:
  315. call *ia32_sys_call_table(,%rax,8)
  316. movq %rax,RAX-ARGOFFSET(%rsp)
  317. DISABLE_INTERRUPTS(CLBR_NONE)
  318. TRACE_IRQS_OFF
  319. testl $_TIF_ALLWORK_MASK,TI_flags+THREAD_INFO(%rsp,RIP-ARGOFFSET)
  320. jnz sysretl_audit
  321. sysretl_from_sys_call:
  322. andl $~TS_COMPAT,TI_status+THREAD_INFO(%rsp,RIP-ARGOFFSET)
  323. RESTORE_ARGS 0,-ARG_SKIP,0,0,0
  324. movl RIP-ARGOFFSET(%rsp),%ecx
  325. CFI_REGISTER rip,rcx
  326. movl EFLAGS-ARGOFFSET(%rsp),%r11d
  327. /*CFI_REGISTER rflags,r11*/
  328. xorq %r10,%r10
  329. xorq %r9,%r9
  330. xorq %r8,%r8
  331. TRACE_IRQS_ON
  332. movl RSP-ARGOFFSET(%rsp),%esp
  333. CFI_RESTORE rsp
  334. USERGS_SYSRET32
  335. #ifdef CONFIG_AUDITSYSCALL
  336. cstar_auditsys:
  337. CFI_RESTORE_STATE
  338. movl %r9d,R9-ARGOFFSET(%rsp) /* register to be clobbered by call */
  339. auditsys_entry_common
  340. movl R9-ARGOFFSET(%rsp),%r9d /* reload 6th syscall arg */
  341. jmp cstar_dispatch
  342. sysretl_audit:
  343. auditsys_exit sysretl_from_sys_call
  344. #endif
  345. cstar_tracesys:
  346. #ifdef CONFIG_AUDITSYSCALL
  347. testl $(_TIF_WORK_SYSCALL_ENTRY & ~_TIF_SYSCALL_AUDIT),TI_flags+THREAD_INFO(%rsp,RIP-ARGOFFSET)
  348. jz cstar_auditsys
  349. #endif
  350. xchgl %r9d,%ebp
  351. SAVE_REST
  352. CLEAR_RREGS 0, r9
  353. movq $-ENOSYS,RAX(%rsp) /* ptrace can change this for a bad syscall */
  354. movq %rsp,%rdi /* &pt_regs -> arg1 */
  355. call syscall_trace_enter
  356. LOAD_ARGS32 ARGOFFSET, 1 /* reload args from stack in case ptrace changed it */
  357. RESTORE_REST
  358. xchgl %ebp,%r9d
  359. cmpq $(IA32_NR_syscalls-1),%rax
  360. ja int_ret_from_sys_call /* cstar_tracesys has set RAX(%rsp) */
  361. jmp cstar_do_call
  362. END(ia32_cstar_target)
  363. ia32_badarg:
  364. ASM_CLAC
  365. movq $-EFAULT,%rax
  366. jmp ia32_sysret
  367. CFI_ENDPROC
  368. /*
  369. * Emulated IA32 system calls via int 0x80.
  370. *
  371. * Arguments:
  372. * %eax System call number.
  373. * %ebx Arg1
  374. * %ecx Arg2
  375. * %edx Arg3
  376. * %esi Arg4
  377. * %edi Arg5
  378. * %ebp Arg6 [note: not saved in the stack frame, should not be touched]
  379. *
  380. * Notes:
  381. * Uses the same stack frame as the x86-64 version.
  382. * All registers except %eax must be saved (but ptrace may violate that)
  383. * Arguments are zero extended. For system calls that want sign extension and
  384. * take long arguments a wrapper is needed. Most calls can just be called
  385. * directly.
  386. * Assumes it is only called from user space and entered with interrupts off.
  387. */
  388. ENTRY(ia32_syscall)
  389. CFI_STARTPROC32 simple
  390. CFI_SIGNAL_FRAME
  391. CFI_DEF_CFA rsp,SS+8-RIP
  392. /*CFI_REL_OFFSET ss,SS-RIP*/
  393. CFI_REL_OFFSET rsp,RSP-RIP
  394. /*CFI_REL_OFFSET rflags,EFLAGS-RIP*/
  395. /*CFI_REL_OFFSET cs,CS-RIP*/
  396. CFI_REL_OFFSET rip,RIP-RIP
  397. PARAVIRT_ADJUST_EXCEPTION_FRAME
  398. SWAPGS
  399. /*
  400. * No need to follow this irqs on/off section: the syscall
  401. * disabled irqs and here we enable it straight after entry:
  402. */
  403. ENABLE_INTERRUPTS(CLBR_NONE)
  404. movl %eax,%eax
  405. pushq_cfi %rax
  406. cld
  407. /* note the registers are not zero extended to the sf.
  408. this could be a problem. */
  409. SAVE_ARGS 0,1,0
  410. orl $TS_COMPAT,TI_status+THREAD_INFO(%rsp,RIP-ARGOFFSET)
  411. testl $_TIF_WORK_SYSCALL_ENTRY,TI_flags+THREAD_INFO(%rsp,RIP-ARGOFFSET)
  412. jnz ia32_tracesys
  413. cmpq $(IA32_NR_syscalls-1),%rax
  414. ja ia32_badsys
  415. ia32_do_call:
  416. IA32_ARG_FIXUP
  417. call *ia32_sys_call_table(,%rax,8) # xxx: rip relative
  418. ia32_sysret:
  419. movq %rax,RAX-ARGOFFSET(%rsp)
  420. ia32_ret_from_sys_call:
  421. CLEAR_RREGS -ARGOFFSET
  422. jmp int_ret_from_sys_call
  423. ia32_tracesys:
  424. SAVE_REST
  425. CLEAR_RREGS
  426. movq $-ENOSYS,RAX(%rsp) /* ptrace can change this for a bad syscall */
  427. movq %rsp,%rdi /* &pt_regs -> arg1 */
  428. call syscall_trace_enter
  429. LOAD_ARGS32 ARGOFFSET /* reload args from stack in case ptrace changed it */
  430. RESTORE_REST
  431. cmpq $(IA32_NR_syscalls-1),%rax
  432. ja int_ret_from_sys_call /* ia32_tracesys has set RAX(%rsp) */
  433. jmp ia32_do_call
  434. END(ia32_syscall)
  435. ia32_badsys:
  436. movq $0,ORIG_RAX-ARGOFFSET(%rsp)
  437. movq $-ENOSYS,%rax
  438. jmp ia32_sysret
  439. CFI_ENDPROC
  440. .macro PTREGSCALL label, func
  441. ALIGN
  442. GLOBAL(\label)
  443. leaq \func(%rip),%rax
  444. jmp ia32_ptregs_common
  445. .endm
  446. CFI_STARTPROC32
  447. PTREGSCALL stub32_rt_sigreturn, sys32_rt_sigreturn
  448. PTREGSCALL stub32_sigreturn, sys32_sigreturn
  449. PTREGSCALL stub32_execve, compat_sys_execve
  450. PTREGSCALL stub32_fork, sys_fork
  451. PTREGSCALL stub32_vfork, sys_vfork
  452. ALIGN
  453. GLOBAL(stub32_clone)
  454. leaq sys_clone(%rip),%rax
  455. mov %r8, %rcx
  456. jmp ia32_ptregs_common
  457. ALIGN
  458. ia32_ptregs_common:
  459. popq %r11
  460. CFI_ENDPROC
  461. CFI_STARTPROC32 simple
  462. CFI_SIGNAL_FRAME
  463. CFI_DEF_CFA rsp,SS+8-ARGOFFSET
  464. CFI_REL_OFFSET rax,RAX-ARGOFFSET
  465. CFI_REL_OFFSET rcx,RCX-ARGOFFSET
  466. CFI_REL_OFFSET rdx,RDX-ARGOFFSET
  467. CFI_REL_OFFSET rsi,RSI-ARGOFFSET
  468. CFI_REL_OFFSET rdi,RDI-ARGOFFSET
  469. CFI_REL_OFFSET rip,RIP-ARGOFFSET
  470. /* CFI_REL_OFFSET cs,CS-ARGOFFSET*/
  471. /* CFI_REL_OFFSET rflags,EFLAGS-ARGOFFSET*/
  472. CFI_REL_OFFSET rsp,RSP-ARGOFFSET
  473. /* CFI_REL_OFFSET ss,SS-ARGOFFSET*/
  474. SAVE_REST
  475. call *%rax
  476. RESTORE_REST
  477. jmp ia32_sysret /* misbalances the return cache */
  478. CFI_ENDPROC
  479. END(ia32_ptregs_common)