2008-04-09 19:50:41 -07:00
/*
2010-02-25 15:37:17 -05:00
* Copyright 2007 - 2010 Red Hat , Inc .
2008-04-09 19:50:41 -07:00
* by Peter Jones < pjones @ redhat . com >
* Copyright 2007 IBM , Inc .
* by Konrad Rzeszutek < konradr @ linux . vnet . ibm . com >
* Copyright 2008
* by Konrad Rzeszutek < ketuzsezr @ darnok . org >
*
* This code finds the iSCSI Boot Format Table .
*
* This program is free software ; you can redistribute it and / or modify
* it under the terms of the GNU General Public License v2 .0 as published by
* the Free Software Foundation
*
* This program 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 General Public License for more details .
*/
# include <linux/bootmem.h>
# include <linux/blkdev.h>
# include <linux/ctype.h>
# include <linux/device.h>
2010-02-25 15:37:17 -05:00
# include <linux/efi.h>
2008-04-09 19:50:41 -07:00
# include <linux/err.h>
# include <linux/init.h>
# include <linux/limits.h>
# include <linux/module.h>
# include <linux/pci.h>
# include <linux/stat.h>
# include <linux/string.h>
# include <linux/types.h>
2010-02-25 15:37:17 -05:00
# include <linux/acpi.h>
# include <linux/iscsi_ibft.h>
2008-04-09 19:50:41 -07:00
# include <asm/mmzone.h>
/*
* Physical location of iSCSI Boot Format Table .
*/
2010-02-25 15:37:17 -05:00
struct acpi_table_ibft * ibft_addr ;
2008-04-09 19:50:41 -07:00
EXPORT_SYMBOL_GPL ( ibft_addr ) ;
2011-04-14 23:50:57 -04:00
static const struct {
char * sign ;
} ibft_signs [ ] = {
# ifdef CONFIG_ACPI
/*
* One spec says " IBFT " , the other says " iBFT " . We have to check
* for both .
*/
{ ACPI_SIG_IBFT } ,
# endif
{ " iBFT " } ,
{ " BIFT " } , /* Broadcom iSCSI Offload */
} ;
2008-04-09 19:50:41 -07:00
# define IBFT_SIGN_LEN 4
# define IBFT_START 0x80000 /* 512kB */
# define IBFT_END 0x100000 /* 1MB */
# define VGA_MEM 0xA0000 /* VGA buffer */
# define VGA_SIZE 0x20000 /* 128kB */
2010-02-25 15:37:17 -05:00
# ifdef CONFIG_ACPI
static int __init acpi_find_ibft ( struct acpi_table_header * header )
{
ibft_addr = ( struct acpi_table_ibft * ) header ;
return 0 ;
}
# endif /* CONFIG_ACPI */
2008-04-09 19:50:41 -07:00
2010-04-08 12:35:55 -04:00
static int __init find_ibft_in_mem ( void )
2008-04-09 19:50:41 -07:00
{
unsigned long pos ;
unsigned int len = 0 ;
void * virt ;
2011-04-14 23:50:57 -04:00
int i ;
2008-04-09 19:50:41 -07:00
for ( pos = IBFT_START ; pos < IBFT_END ; pos + = 16 ) {
/* The table can't be inside the VGA BIOS reserved space,
* so skip that area */
if ( pos = = VGA_MEM )
pos + = VGA_SIZE ;
2009-10-02 16:12:39 +01:00
virt = isa_bus_to_virt ( pos ) ;
2011-04-14 23:50:57 -04:00
for ( i = 0 ; i < ARRAY_SIZE ( ibft_signs ) ; i + + ) {
if ( memcmp ( virt , ibft_signs [ i ] . sign , IBFT_SIGN_LEN ) = =
0 ) {
unsigned long * addr =
( unsigned long * ) isa_bus_to_virt ( pos + 4 ) ;
len = * addr ;
/* if the length of the table extends past 1M,
* the table cannot be valid . */
if ( pos + len < = ( IBFT_END - 1 ) ) {
ibft_addr = ( struct acpi_table_ibft * ) virt ;
goto done ;
}
2008-04-09 19:50:41 -07:00
}
}
}
2011-04-14 23:50:57 -04:00
done :
2010-04-08 12:35:55 -04:00
return len ;
}
/*
* Routine used to find the iSCSI Boot Format Table . The logical
* kernel address is set in the ibft_addr global variable .
*/
unsigned long __init find_ibft_region ( unsigned long * sizep )
{
2011-06-05 03:35:09 -07:00
# ifdef CONFIG_ACPI
2011-04-14 23:50:57 -04:00
int i ;
2011-06-05 03:35:09 -07:00
# endif
2010-04-08 12:35:55 -04:00
ibft_addr = NULL ;
2010-02-25 15:37:17 -05:00
# ifdef CONFIG_ACPI
2011-04-14 23:50:57 -04:00
for ( i = 0 ; i < ARRAY_SIZE ( ibft_signs ) & & ! ibft_addr ; i + + )
acpi_table_parse ( ibft_signs [ i ] . sign , acpi_find_ibft ) ;
2010-02-25 15:37:17 -05:00
# endif /* CONFIG_ACPI */
2010-04-08 12:35:55 -04:00
/* iBFT 1.03 section 1.4.3.1 mandates that UEFI machines will
* only use ACPI for this */
if ( ! ibft_addr & & ! efi_enabled )
find_ibft_in_mem ( ) ;
2010-04-01 14:32:43 -07:00
if ( ibft_addr ) {
2010-02-25 15:37:17 -05:00
* sizep = PAGE_ALIGN ( ibft_addr - > header . length ) ;
return ( u64 ) isa_virt_to_bus ( ibft_addr ) ;
2010-04-01 14:32:43 -07:00
}
* sizep = 0 ;
return 0 ;
2008-04-09 19:50:41 -07:00
}