/*
* Implementation of a gateway into 32bit space. Stub functions
* can be called from Bochs BIOS which call functions with a compatible
* signature in 32bit space. All interrupts are disabled while in
* 32 bit mode.
*
* This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Lesser General Public
* License as published by the Free Software Foundation; either
* version 2 of the License, or (at your option) any later version.
*
* This library is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public
* License along with this library; If not, see .
*
* Copyright (C) IBM Corporation, 2006
* Copyright (c) 2008, Citrix Systems, Inc.
*
* Author: Stefan Berger
* Author: Keir Fraser
*/
/*
* Note:
* BCC's ABI does not require to preserve any 16bit registers ax, bx, cs, dx
* by a called function. So these registers need not be preserved while
* calling a function in 32bit space, either.
*
* When bcc calls a function with 16bit parameters it pushes 2 bytes onto
* the stack for such a parameter. GCC, however, expects 32bit parameters
* (4 bytes) even for uint16_t, so casting to 32bit from bcc is a good idea.
*/
/* At most 32 bytes in argument list to a 32-bit function. */
#define MAX_ARG_BYTES 32
#define REAL_MODE_CODE_OFFSET 0xf0000
/* Definitions of code/data segment descriptors. */
#define PM_32BIT_CS (gdt_entry_pm_32bit_cs - gdt_base)
#define PM_16BIT_CS (gdt_entry_pm_16bit_cs - gdt_base)
#define PM_32BIT_DS (gdt_entry_pm_32bit_ds - gdt_base)
#define PM_16BIT_DS (gdt_entry_pm_16bit_ds - gdt_base)
.align 16
gdt_base:
.word 0,0
.byte 0,0,0,0
gdt_entry_pm_32bit_cs:
.word 0xffff, 0x0000
.byte 0x00, 0x9b, 0xcf, 0x00
gdt_entry_pm_16bit_cs:
.word 0xffff, 0x0000
.byte REAL_MODE_CODE_OFFSET >> 16, 0x9b, 0x8f, 0x0
gdt_entry_pm_32bit_ds:
.word 0xffff, 0x0000
.byte 0x0, 0x93, 0xcf, 0x0
gdt_entry_pm_16bit_ds:
.word 0xffff, 0x0000
.byte 0x0, 0x93, 0x8f, 0x0
gdt_entry_end:
protmode_gdtdesc:
.word (gdt_entry_end - gdt_base) - 1
.long gdt_base | REAL_MODE_CODE_OFFSET
realmode_gdtdesc:
.word 0xffff
.long 0x0
Upcall:
; Do an upcall into 32 bit space
;
; Input:
; bx: index of function to call
; Ouput:
; dx, ax: 32 bit result of call (even if 'void' is expected)
; Save caller state, stack frame offsets listed below
#define esp_off 0
#define ss_off 4
#define es_off 6
#define ds_off 8
#define flags_off 10
#define retaddr_off 12
#define args_off 14
pushf
cli
push ds
push es
push ss
push esp
; Calculate protected-mode esp from ss:sp
and esp, #0xffff
xor eax, eax
mov ax, ss
shl eax, #4
add esp, eax
; Switch to protected mode
seg cs
lgdt protmode_gdtdesc
mov eax, cr0
or al, #0x1 ; protected mode on
mov cr0, eax
jmpf DWORD (REAL_MODE_CODE_OFFSET|upcall1), #PM_32BIT_CS
upcall1:
USE32
mov ax, #PM_32BIT_DS
mov ds, ax
mov es, ax
mov ss, ax
; Marshal arguments and call 32-bit function
mov ecx, #MAX_ARG_BYTES/4
upcall2:
push MAX_ARG_BYTES-4+args_off[esp]
loop upcall2
mov eax, [BIOS_INFO_PHYSICAL_ADDRESS + BIOSINFO_OFF_bios32_entry]
call eax
add esp, #MAX_ARG_BYTES
mov ecx, eax ; Result in ecx
; Restore real-mode stack pointer
xor eax, eax
mov ax, ss_off[esp]
mov bx, ax ; Real-mode ss in bx
shl eax, 4
sub esp, eax
; Return to real mode
jmpf upcall3, #PM_16BIT_CS
upcall3:
USE16
mov ax, #PM_16BIT_DS
mov ds, ax
mov es, ax
mov ss, ax
mov eax, cr0
and al, #0xfe ; protected mode off
mov cr0, eax
jmpf upcall4, #REAL_MODE_CODE_OFFSET>>4
upcall4:
seg cs
lgdt realmode_gdtdesc
; Restore real-mode ss
mov ss, bx
; Convert result into dx:ax format
mov eax, ecx
ror eax, #16
mov dx, ax
ror eax, #16
; Restore caller state and return
pop esp
pop bx ; skip ss
pop es
pop ds
popf
ret
MACRO DoUpcall
mov bx, #?1
jmp Upcall
MEND
#define X(idx, ret, fn, args...) _ ## fn: DoUpcall(idx)
#include "32bitprotos.h"
#undef X