2005-04-16 15:20:36 -07:00
/*
* Extensible Firmware Interface
*
* Based on Extensible Firmware Interface Specification version 1.0
*
* Copyright ( C ) 1999 VA Linux Systems
* Copyright ( C ) 1999 Walt Drummond < drummond @ valinux . com >
* Copyright ( C ) 1999 - 2002 Hewlett - Packard Co .
* David Mosberger - Tang < davidm @ hpl . hp . com >
* Stephane Eranian < eranian @ hpl . hp . com >
*
* All EFI Runtime Services are not implemented yet as EFI only
* supports physical mode addressing on SoftSDV . This is to be fixed
* in a future version . - - drummond 1999 - 07 - 20
*
* Implemented EFI runtime services and virtual mode calls . - - davidm
*
* Goutham Rao : < goutham . rao @ intel . com >
* Skip non - WB memory and ignore empty memory ranges .
*/
# include <linux/kernel.h>
# include <linux/types.h>
# include <linux/ioport.h>
# include <linux/efi.h>
# include <asm/io.h>
# include <asm/page.h>
# include <asm/pgtable.h>
# include <asm/tlbflush.h>
2008-02-13 13:26:39 -08:00
# include <asm/efi.h>
2005-04-16 15:20:36 -07:00
/*
* To make EFI call EFI runtime service in physical addressing mode we need
* prelog / epilog before / after the invocation to disable interrupt , to
* claim EFI runtime service handler exclusively and to duplicate a memory in
* low memory space say 0 - 3 G .
*/
static unsigned long efi_rt_eflags ;
static pgd_t efi_bak_pg_dir_pointer [ 2 ] ;
2008-01-30 13:32:11 +01:00
void efi_call_phys_prelog ( void )
2005-04-16 15:20:36 -07:00
{
unsigned long cr4 ;
unsigned long temp ;
2008-01-30 13:31:12 +01:00
struct desc_ptr gdt_descr ;
2005-04-16 15:20:36 -07:00
local_irq_save ( efi_rt_eflags ) ;
/*
2008-07-02 22:48:03 +01:00
* If I don ' t have PAE , I should just duplicate two entries in page
* directory . If I have PAE , I just need to duplicate one entry in
2005-04-16 15:20:36 -07:00
* page directory .
*/
2005-09-03 15:56:36 -07:00
cr4 = read_cr4 ( ) ;
2005-04-16 15:20:36 -07:00
2008-07-02 22:48:03 +01:00
if ( cr4 & X86_CR4_PAE ) {
2005-04-16 15:20:36 -07:00
efi_bak_pg_dir_pointer [ 0 ] . pgd =
swapper_pg_dir [ pgd_index ( 0 ) ] . pgd ;
swapper_pg_dir [ 0 ] . pgd =
swapper_pg_dir [ pgd_index ( PAGE_OFFSET ) ] . pgd ;
} else {
efi_bak_pg_dir_pointer [ 0 ] . pgd =
swapper_pg_dir [ pgd_index ( 0 ) ] . pgd ;
efi_bak_pg_dir_pointer [ 1 ] . pgd =
swapper_pg_dir [ pgd_index ( 0x400000 ) ] . pgd ;
swapper_pg_dir [ pgd_index ( 0 ) ] . pgd =
swapper_pg_dir [ pgd_index ( PAGE_OFFSET ) ] . pgd ;
temp = PAGE_OFFSET + 0x400000 ;
swapper_pg_dir [ pgd_index ( 0x400000 ) ] . pgd =
swapper_pg_dir [ pgd_index ( temp ) ] . pgd ;
}
/*
* After the lock is released , the original page table is restored .
*/
2008-01-30 13:32:11 +01:00
__flush_tlb_all ( ) ;
2005-04-16 15:20:36 -07:00
2007-05-02 19:27:11 +02:00
gdt_descr . address = __pa ( get_cpu_gdt_table ( 0 ) ) ;
gdt_descr . size = GDT_SIZE - 1 ;
load_gdt ( & gdt_descr ) ;
2005-04-16 15:20:36 -07:00
}
2008-01-30 13:32:11 +01:00
void efi_call_phys_epilog ( void )
2005-04-16 15:20:36 -07:00
{
unsigned long cr4 ;
2008-01-30 13:31:12 +01:00
struct desc_ptr gdt_descr ;
2005-04-16 15:20:36 -07:00
2007-05-02 19:27:11 +02:00
gdt_descr . address = ( unsigned long ) get_cpu_gdt_table ( 0 ) ;
gdt_descr . size = GDT_SIZE - 1 ;
load_gdt ( & gdt_descr ) ;
2006-02-24 13:04:14 -08:00
2005-09-03 15:56:36 -07:00
cr4 = read_cr4 ( ) ;
2005-04-16 15:20:36 -07:00
2008-07-02 22:48:03 +01:00
if ( cr4 & X86_CR4_PAE ) {
2005-04-16 15:20:36 -07:00
swapper_pg_dir [ pgd_index ( 0 ) ] . pgd =
efi_bak_pg_dir_pointer [ 0 ] . pgd ;
} else {
swapper_pg_dir [ pgd_index ( 0 ) ] . pgd =
efi_bak_pg_dir_pointer [ 0 ] . pgd ;
swapper_pg_dir [ pgd_index ( 0x400000 ) ] . pgd =
efi_bak_pg_dir_pointer [ 1 ] . pgd ;
}
/*
* After the lock is released , the original page table is restored .
*/
2008-01-30 13:32:11 +01:00
__flush_tlb_all ( ) ;
2005-04-16 15:20:36 -07:00
local_irq_restore ( efi_rt_eflags ) ;
}