mirror of https://gitee.com/openkylin/linux.git
68 lines
1.3 KiB
ArmAsm
68 lines
1.3 KiB
ArmAsm
|
/*
|
||
|
* thunks.S - assembly helpers for mixed-bitness code
|
||
|
* Copyright (c) 2015 Andrew Lutomirski
|
||
|
*
|
||
|
* This program is free software; you can redistribute it and/or modify
|
||
|
* it under the terms and conditions of the GNU General Public License,
|
||
|
* version 2, as published by the Free Software Foundation.
|
||
|
*
|
||
|
* This program is distributed in the hope it will be useful, but
|
||
|
* WITHOUT ANY WARRANTY; without even the implied warranty of
|
||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||
|
* General Public License for more details.
|
||
|
*
|
||
|
* These are little helpers that make it easier to switch bitness on
|
||
|
* the fly.
|
||
|
*/
|
||
|
|
||
|
.text
|
||
|
|
||
|
.global call32_from_64
|
||
|
.type call32_from_64, @function
|
||
|
call32_from_64:
|
||
|
// rdi: stack to use
|
||
|
// esi: function to call
|
||
|
|
||
|
// Save registers
|
||
|
pushq %rbx
|
||
|
pushq %rbp
|
||
|
pushq %r12
|
||
|
pushq %r13
|
||
|
pushq %r14
|
||
|
pushq %r15
|
||
|
pushfq
|
||
|
|
||
|
// Switch stacks
|
||
|
mov %rsp,(%rdi)
|
||
|
mov %rdi,%rsp
|
||
|
|
||
|
// Switch to compatibility mode
|
||
|
pushq $0x23 /* USER32_CS */
|
||
|
pushq $1f
|
||
|
lretq
|
||
|
|
||
|
1:
|
||
|
.code32
|
||
|
// Call the function
|
||
|
call *%esi
|
||
|
// Switch back to long mode
|
||
|
jmp $0x33,$1f
|
||
|
.code64
|
||
|
|
||
|
1:
|
||
|
// Restore the stack
|
||
|
mov (%rsp),%rsp
|
||
|
|
||
|
// Restore registers
|
||
|
popfq
|
||
|
popq %r15
|
||
|
popq %r14
|
||
|
popq %r13
|
||
|
popq %r12
|
||
|
popq %rbp
|
||
|
popq %rbx
|
||
|
|
||
|
ret
|
||
|
|
||
|
.size call32_from_64, .-call32_from_64
|