diff options
Diffstat (limited to 'roms/ipxe/src/arch/i386/interface')
30 files changed, 9184 insertions, 0 deletions
| diff --git a/roms/ipxe/src/arch/i386/interface/pcbios/apm.c b/roms/ipxe/src/arch/i386/interface/pcbios/apm.c new file mode 100644 index 00000000..3b13e1cd --- /dev/null +++ b/roms/ipxe/src/arch/i386/interface/pcbios/apm.c @@ -0,0 +1,108 @@ +/* + * Copyright (C) 2013 Marin Hannache <ipxe@mareo.fr>. + * + * This program is free software; you can redistribute it and/or + * modify it under the terms of the GNU General Public License as + * published by the Free Software Foundation; either version 2 of the + * License, or any later version. + * + * 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. + * + * You should have received a copy of the GNU General Public License + * along with this program; if not, write to the Free Software + * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA + * 02110-1301, USA. + */ + +FILE_LICENCE ( GPL2_OR_LATER ); + +/** + * @file + * + * Advanced Power Management + * + */ + +#include <errno.h> +#include <realmode.h> +#include <ipxe/reboot.h> + +/** + * Power off the computer using APM + * + * @ret rc		Return status code + */ +static int apm_poweroff ( void ) { +	uint16_t apm_version; +	uint16_t apm_signature; +	uint16_t apm_flags; +	uint16_t carry; + +	/* APM check */ +	__asm__ __volatile__ ( REAL_CODE ( "int $0x15\n\t" +	                                   "adc %%edx,0\n\t" ) +	                       : "=a" ( apm_version ), "=b" ( apm_signature ), +                                 "=c" ( apm_flags ), "=d" ( carry ) +	                       : "a" ( 0x5300 ), "b" ( 0x0000 ), +				 "d" ( 0x0000 ) ); +	if ( carry ) { +		DBG ( "APM not present\n" ); +		return -ENOTSUP; +	} +	if ( apm_signature != 0x504d ) { /* signature 'PM' */ +		DBG ( "APM not present\n" ); +		return -ENOTSUP; +	} +	if ( apm_version < 0x0101 ) { /* Need version 1.1+ */ +		DBG ( "APM 1.1+ not supported\n" ); +		return -ENOTSUP; +	} +	if ( ( apm_flags & 0x8 ) == 0x8 ) { +		DBG ( "APM power management disabled\n" ); +		return -EPERM; +	} +	DBG2 ( "APM check completed\n" ); + +	/* APM initialisation */ +	__asm__ __volatile__ ( REAL_CODE ( "int $0x15\n\t" +	                                   "adc %%edx,0\n\t" ) +	                       : "=d" ( carry ) +	                       : "a" ( 0x5301 ), "b" ( 0x0000 ), +	                         "d" ( 0x0000 ) ); +	if ( carry ) { +		DBG ( "APM initialisation failed\n" ); +		return -EIO; +	} +	DBG2 ( "APM initialisation completed\n" ); + +	/* Set APM driver version */ +	__asm__ __volatile__ ( REAL_CODE ( "int $0x15\n\t" +	                                   "adc %%edx,0\n\t" ) +	                       : "=d" ( carry ) +	                       : "a" ( 0x530e ), "b" ( 0x0000 ), +	                         "c" ( 0x0101 ), "d" ( 0x0000 ) ); +	if ( carry ) { +		DBG ( "APM setting driver version failed\n" ); +		return -EIO; +	} +	DBG2 ( "APM driver version set\n" ); + +	/* Setting power state to off */ +	__asm__ __volatile__ ( REAL_CODE ( "int $0x15\n\t" +	                                   "adc %%edx,0\n\t" ) +	                       : "=d" ( carry ) +	                       : "a" ( 0x5307 ), "b" ( 0x0001 ), +	                         "c" ( 0x0003 ), "d" ( 0x0000) ); +	if ( carry ) { +		DBG ( "APM setting power state failed\n" ); +		return -ENOTTY; +	} + +	/* Should never happen */ +	return -ECANCELED; +} + +PROVIDE_REBOOT ( pcbios, poweroff, apm_poweroff ); diff --git a/roms/ipxe/src/arch/i386/interface/pcbios/bios_nap.c b/roms/ipxe/src/arch/i386/interface/pcbios/bios_nap.c new file mode 100644 index 00000000..1e7de756 --- /dev/null +++ b/roms/ipxe/src/arch/i386/interface/pcbios/bios_nap.c @@ -0,0 +1,16 @@ +#include <ipxe/nap.h> +#include <realmode.h> + +FILE_LICENCE ( GPL2_OR_LATER ); + +/** + * Save power by halting the CPU until the next interrupt + * + */ +static void bios_cpu_nap ( void ) { +	__asm__ __volatile__ ( "sti\n\t" +			       "hlt\n\t" +			       "cli\n\t" ); +} + +PROVIDE_NAP ( pcbios, cpu_nap, bios_cpu_nap ); diff --git a/roms/ipxe/src/arch/i386/interface/pcbios/bios_reboot.c b/roms/ipxe/src/arch/i386/interface/pcbios/bios_reboot.c new file mode 100644 index 00000000..68546b2e --- /dev/null +++ b/roms/ipxe/src/arch/i386/interface/pcbios/bios_reboot.c @@ -0,0 +1,48 @@ +/* + * Copyright (C) 2010 Michael Brown <mbrown@fensystems.co.uk>. + * + * This program is free software; you can redistribute it and/or + * modify it under the terms of the GNU General Public License as + * published by the Free Software Foundation; either version 2 of the + * License, or any later version. + * + * 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. + * + * You should have received a copy of the GNU General Public License + * along with this program; if not, write to the Free Software + * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA + * 02110-1301, USA. + */ + +FILE_LICENCE ( GPL2_OR_LATER ); + +/** @file + * + * Standard PC-BIOS reboot mechanism + * + */ + +#include <ipxe/reboot.h> +#include <realmode.h> +#include <bios.h> + +/** + * Reboot system + * + * @v warm		Perform a warm reboot + */ +static void bios_reboot ( int warm ) { +	uint16_t flag; + +	/* Configure BIOS for cold/warm reboot */ +	flag = ( warm ? BDA_REBOOT_WARM : 0 ); +	put_real ( flag, BDA_SEG, BDA_REBOOT ); + +	/* Jump to system reset vector */ +	__asm__ __volatile__ ( REAL_CODE ( "ljmp $0xf000, $0xfff0" ) : : ); +} + +PROVIDE_REBOOT ( pcbios, reboot, bios_reboot ); diff --git a/roms/ipxe/src/arch/i386/interface/pcbios/bios_smbios.c b/roms/ipxe/src/arch/i386/interface/pcbios/bios_smbios.c new file mode 100644 index 00000000..dd7897e2 --- /dev/null +++ b/roms/ipxe/src/arch/i386/interface/pcbios/bios_smbios.c @@ -0,0 +1,61 @@ +/* + * Copyright (C) 2007 Michael Brown <mbrown@fensystems.co.uk>. + * + * This program is free software; you can redistribute it and/or + * modify it under the terms of the GNU General Public License as + * published by the Free Software Foundation; either version 2 of the + * License, or any later version. + * + * 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. + * + * You should have received a copy of the GNU General Public License + * along with this program; if not, write to the Free Software + * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA + * 02110-1301, USA. + */ + +FILE_LICENCE ( GPL2_OR_LATER ); + +#include <stdint.h> +#include <string.h> +#include <errno.h> +#include <assert.h> +#include <ipxe/uaccess.h> +#include <ipxe/smbios.h> +#include <realmode.h> +#include <pnpbios.h> + +/** @file + * + * System Management BIOS + * + */ + +/** + * Find SMBIOS + * + * @v smbios		SMBIOS entry point descriptor structure to fill in + * @ret rc		Return status code + */ +static int bios_find_smbios ( struct smbios *smbios ) { +	struct smbios_entry entry; +	int rc; + +	/* Scan through BIOS segment to find SMBIOS entry point */ +	if ( ( rc = find_smbios_entry ( real_to_user ( BIOS_SEG, 0 ), 0x10000, +					&entry ) ) != 0 ) +		return rc; + +	/* Fill in entry point descriptor structure */ +	smbios->address = phys_to_user ( entry.smbios_address ); +	smbios->len = entry.smbios_len; +	smbios->count = entry.smbios_count; +	smbios->version = SMBIOS_VERSION ( entry.major, entry.minor ); + +	return 0; +} + +PROVIDE_SMBIOS ( pcbios, find_smbios, bios_find_smbios ); diff --git a/roms/ipxe/src/arch/i386/interface/pcbios/bios_timer.c b/roms/ipxe/src/arch/i386/interface/pcbios/bios_timer.c new file mode 100644 index 00000000..65bbf9e0 --- /dev/null +++ b/roms/ipxe/src/arch/i386/interface/pcbios/bios_timer.c @@ -0,0 +1,66 @@ +/* + * Copyright (C) 2008 Michael Brown <mbrown@fensystems.co.uk>. + * + * This program is free software; you can redistribute it and/or + * modify it under the terms of the GNU General Public License as + * published by the Free Software Foundation; either version 2 of the + * License, or any later version. + * + * 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. + * + * You should have received a copy of the GNU General Public License + * along with this program; if not, write to the Free Software + * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA + * 02110-1301, USA. + */ + +FILE_LICENCE ( GPL2_OR_LATER ); + +/** @file + * + * BIOS timer + * + */ + +#include <ipxe/timer.h> +#include <realmode.h> +#include <bios.h> + +/** + * Get current system time in ticks + * + * @ret ticks		Current time, in ticks + * + * Use direct memory access to BIOS variables, longword 0040:006C + * (ticks today) and byte 0040:0070 (midnight crossover flag) instead + * of calling timeofday BIOS interrupt. + */ +static unsigned long bios_currticks ( void ) { +	static int days = 0; +	uint32_t ticks; +	uint8_t midnight; + +	/* Re-enable interrupts so that the timer interrupt can occur */ +	__asm__ __volatile__ ( "sti\n\t" +			       "nop\n\t" +			       "nop\n\t" +			       "cli\n\t" ); + +	get_real ( ticks, BDA_SEG, 0x006c ); +	get_real ( midnight, BDA_SEG, 0x0070 ); + +	if ( midnight ) { +		midnight = 0; +		put_real ( midnight, BDA_SEG, 0x0070 ); +		days += 0x1800b0; +	} + +	return ( days + ticks ); +} + +PROVIDE_TIMER_INLINE ( pcbios, udelay ); +PROVIDE_TIMER ( pcbios, currticks, bios_currticks ); +PROVIDE_TIMER_INLINE ( pcbios, ticks_per_sec ); diff --git a/roms/ipxe/src/arch/i386/interface/pcbios/biosint.c b/roms/ipxe/src/arch/i386/interface/pcbios/biosint.c new file mode 100644 index 00000000..a193defa --- /dev/null +++ b/roms/ipxe/src/arch/i386/interface/pcbios/biosint.c @@ -0,0 +1,92 @@ +#include <errno.h> +#include <realmode.h> +#include <biosint.h> + +/** + * @file BIOS interrupts + * + */ + +FILE_LICENCE ( GPL2_OR_LATER ); + +/** + * Hook INT vector + * + * @v interrupt		INT number + * @v handler		Offset within .text16 to interrupt handler + * @v chain_vector	Vector for chaining to previous handler + * + * Hooks in an i386 INT handler.  The handler itself must reside + * within the .text16 segment.  @c chain_vector will be filled in with + * the address of the previously-installed handler for this interrupt; + * the handler should probably exit by ljmping via this vector. + */ +void hook_bios_interrupt ( unsigned int interrupt, unsigned int handler, +			   struct segoff *chain_vector ) { +	struct segoff vector = { +		.segment = rm_cs, +		.offset = handler, +	}; + +	DBG ( "Hooking INT %#02x to %04x:%04x\n", +	      interrupt, rm_cs, handler ); + +	if ( ( chain_vector->segment != 0 ) || +	     ( chain_vector->offset != 0 ) ) { +		/* Already hooked; do nothing */ +		DBG ( "...already hooked\n" ); +		return; +	} + +	copy_from_real ( chain_vector, 0, ( interrupt * 4 ), +			 sizeof ( *chain_vector ) ); +	DBG ( "...chaining to %04x:%04x\n", +	      chain_vector->segment, chain_vector->offset ); +	if ( DBG_LOG ) { +		char code[64]; +		copy_from_real ( code, chain_vector->segment, +				 chain_vector->offset, sizeof ( code ) ); +		DBG_HDA ( *chain_vector, code, sizeof ( code ) ); +	} + +	copy_to_real ( 0, ( interrupt * 4 ), &vector, sizeof ( vector ) ); +	hooked_bios_interrupts++; +} + +/** + * Unhook INT vector + * + * @v interrupt		INT number + * @v handler		Offset within .text16 to interrupt handler + * @v chain_vector	Vector containing address of previous handler + * + * Unhooks an i386 interrupt handler hooked by hook_i386_vector(). + * Note that this operation may fail, if some external code has hooked + * the vector since we hooked in our handler.  If it fails, it means + * that it is not possible to unhook our handler, and we must leave it + * (and its chaining vector) resident in memory. + */ +int unhook_bios_interrupt ( unsigned int interrupt, unsigned int handler, +			    struct segoff *chain_vector ) { +	struct segoff vector; + +	DBG ( "Unhooking INT %#02x from %04x:%04x\n", +	      interrupt, rm_cs, handler ); + +	copy_from_real ( &vector, 0, ( interrupt * 4 ), sizeof ( vector ) ); +	if ( ( vector.segment != rm_cs ) || ( vector.offset != handler ) ) { +		DBG ( "...cannot unhook; vector points to %04x:%04x\n", +		      vector.segment, vector.offset ); +		return -EBUSY; +	} + +	DBG ( "...restoring to %04x:%04x\n", +	      chain_vector->segment, chain_vector->offset ); +	copy_to_real ( 0, ( interrupt * 4 ), chain_vector, +		       sizeof ( *chain_vector ) ); + +	chain_vector->segment = 0; +	chain_vector->offset = 0; +	hooked_bios_interrupts--; +	return 0; +} diff --git a/roms/ipxe/src/arch/i386/interface/pcbios/int13.c b/roms/ipxe/src/arch/i386/interface/pcbios/int13.c new file mode 100644 index 00000000..1c7a8128 --- /dev/null +++ b/roms/ipxe/src/arch/i386/interface/pcbios/int13.c @@ -0,0 +1,1989 @@ +/* + * Copyright (C) 2006 Michael Brown <mbrown@fensystems.co.uk>. + * + * This program is free software; you can redistribute it and/or + * modify it under the terms of the GNU General Public License as + * published by the Free Software Foundation; either version 2 of the + * License, or any later version. + * + * 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. + * + * You should have received a copy of the GNU General Public License + * along with this program; if not, write to the Free Software + * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA + * 02110-1301, USA. + */ + +FILE_LICENCE ( GPL2_OR_LATER ); + +#include <stdint.h> +#include <stdlib.h> +#include <limits.h> +#include <byteswap.h> +#include <errno.h> +#include <assert.h> +#include <ipxe/list.h> +#include <ipxe/blockdev.h> +#include <ipxe/io.h> +#include <ipxe/open.h> +#include <ipxe/uri.h> +#include <ipxe/process.h> +#include <ipxe/xfer.h> +#include <ipxe/retry.h> +#include <ipxe/timer.h> +#include <ipxe/acpi.h> +#include <ipxe/sanboot.h> +#include <ipxe/device.h> +#include <ipxe/pci.h> +#include <ipxe/iso9660.h> +#include <ipxe/eltorito.h> +#include <realmode.h> +#include <bios.h> +#include <biosint.h> +#include <bootsector.h> +#include <int13.h> + +/** @file + * + * INT 13 emulation + * + * This module provides a mechanism for exporting block devices via + * the BIOS INT 13 disk interrupt interface.   + * + */ + +/** + * Overall timeout for INT 13 commands (independent of underlying device + * + * Underlying devices should ideally never become totally stuck. + * However, if they do, then the INT 13 mechanism provides no means + * for the caller to cancel the operation, and the machine appears to + * hang.  Use an overall timeout for all commands to avoid this + * problem and bounce timeout failures to the caller. + */ +#define INT13_COMMAND_TIMEOUT ( 15 * TICKS_PER_SEC ) + +/** An INT 13 emulated drive */ +struct int13_drive { +	/** Reference count */ +	struct refcnt refcnt; +	/** List of all registered drives */ +	struct list_head list; + +	/** Block device URI */ +	struct uri *uri; +	/** Underlying block device interface */ +	struct interface block; + +	/** BIOS in-use drive number (0x00-0xff) */ +	unsigned int drive; +	/** BIOS natural drive number (0x00-0xff) +	 * +	 * This is the drive number that would have been assigned by +	 * 'naturally' appending the drive to the end of the BIOS +	 * drive list. +	 * +	 * If the emulated drive replaces a preexisting drive, this is +	 * the drive number that the preexisting drive gets remapped +	 * to. +	 */ +	unsigned int natural_drive; + +	/** Block device capacity */ +	struct block_device_capacity capacity; +	/** INT 13 emulated blocksize shift +	 * +	 * To allow for emulation of CD-ROM access, this represents +	 * the left-shift required to translate from INT 13 blocks to +	 * underlying blocks. +	 */ +	unsigned int blksize_shift; + +	/** Number of cylinders +	 * +	 * The cylinder number field in an INT 13 call is ten bits +	 * wide, giving a maximum of 1024 cylinders.  Conventionally, +	 * when the 7.8GB limit of a CHS address is exceeded, it is +	 * the number of cylinders that is increased beyond the +	 * addressable limit. +	 */ +	unsigned int cylinders; +	/** Number of heads +	 * +	 * The head number field in an INT 13 call is eight bits wide, +	 * giving a maximum of 256 heads.  However, apparently all +	 * versions of MS-DOS up to and including Win95 fail with 256 +	 * heads, so the maximum encountered in practice is 255. +	 */ +	unsigned int heads; +	/** Number of sectors per track +	 * +	 * The sector number field in an INT 13 call is six bits wide, +	 * giving a maximum of 63 sectors, since sector numbering +	 * (unlike head and cylinder numbering) starts at 1, not 0. +	 */ +	unsigned int sectors_per_track; + +	/** Drive is a CD-ROM */ +	int is_cdrom; +	/** Address of El Torito boot catalog (if any) */ +	unsigned int boot_catalog; + +	/** Underlying device status, if in error */ +	int block_rc; +	/** Status of last operation */ +	int last_status; +}; + +/** Vector for chaining to other INT 13 handlers */ +static struct segoff __text16 ( int13_vector ); +#define int13_vector __use_text16 ( int13_vector ) + +/** Assembly wrapper */ +extern void int13_wrapper ( void ); + +/** Dummy floppy disk parameter table */ +static struct int13_fdd_parameters __data16 ( int13_fdd_params ) = { +	/* 512 bytes per sector */ +	.bytes_per_sector = 0x02, +	/* Highest sectors per track that we ever return */ +	.sectors_per_track = 48, +}; +#define int13_fdd_params __use_data16 ( int13_fdd_params ) + +/** List of registered emulated drives */ +static LIST_HEAD ( int13s ); + +/** + * Equipment word + * + * This is a cached copy of the BIOS Data Area equipment word at + * 40:10. + */ +static uint16_t equipment_word; + +/** + * Number of BIOS floppy disk drives + * + * This is derived from the equipment word.  It is held in .text16 to + * allow for easy access by the INT 13,08 wrapper. + */ +static uint8_t __text16 ( num_fdds ); +#define num_fdds __use_text16 ( num_fdds ) + +/** + * Number of BIOS hard disk drives + * + * This is a cached copy of the BIOS Data Area number of hard disk + * drives at 40:75.  It is held in .text16 to allow for easy access by + * the INT 13,08 wrapper. + */ +static uint8_t __text16 ( num_drives ); +#define num_drives __use_text16 ( num_drives ) + +/** + * Calculate INT 13 drive sector size + * + * @v int13		Emulated drive + * @ret blksize		Sector size + */ +static inline size_t int13_blksize ( struct int13_drive *int13 ) { +	return ( int13->capacity.blksize << int13->blksize_shift ); +} + +/** + * Calculate INT 13 drive capacity + * + * @v int13		Emulated drive + * @ret blocks		Number of blocks + */ +static inline uint64_t int13_capacity ( struct int13_drive *int13 ) { +	return ( int13->capacity.blocks >> int13->blksize_shift ); +} + +/** + * Calculate INT 13 drive capacity (limited to 32 bits) + * + * @v int13		Emulated drive + * @ret blocks		Number of blocks + */ +static inline uint32_t int13_capacity32 ( struct int13_drive *int13 ) { +	uint64_t capacity = int13_capacity ( int13 ); +	return ( ( capacity <= 0xffffffffUL ) ? capacity : 0xffffffff ); +} + +/** + * Test if INT 13 drive is a floppy disk drive + * + * @v int13		Emulated drive + * @ret is_fdd		Emulated drive is a floppy disk + */ +static inline int int13_is_fdd ( struct int13_drive *int13 ) { +	return ( ! ( int13->drive & 0x80 ) ); +} + +/** An INT 13 command */ +struct int13_command { +	/** Status */ +	int rc; +	/** INT 13 drive */ +	struct int13_drive *int13; +	/** Underlying block device interface */ +	struct interface block; +	/** Command timeout timer */ +	struct retry_timer timer; +}; + +/** + * Record INT 13 drive capacity + * + * @v command		INT 13 command + * @v capacity		Block device capacity + */ +static void int13_command_capacity ( struct int13_command *command, +				     struct block_device_capacity *capacity ) { +	memcpy ( &command->int13->capacity, capacity, +		 sizeof ( command->int13->capacity ) ); +} + +/** + * Close INT 13 command + * + * @v command		INT 13 command + * @v rc		Reason for close + */ +static void int13_command_close ( struct int13_command *command, int rc ) { +	intf_restart ( &command->block, rc ); +	stop_timer ( &command->timer ); +	command->rc = rc; +} + +/** + * Handle INT 13 command timer expiry + * + * @v timer		Timer + */ +static void int13_command_expired ( struct retry_timer *timer, +				    int over __unused ) { +	struct int13_command *command = +		container_of ( timer, struct int13_command, timer ); + +	int13_command_close ( command, -ETIMEDOUT ); +} + +/** INT 13 command interface operations */ +static struct interface_operation int13_command_op[] = { +	INTF_OP ( intf_close, struct int13_command *, int13_command_close ), +	INTF_OP ( block_capacity, struct int13_command *, +		  int13_command_capacity ), +}; + +/** INT 13 command interface descriptor */ +static struct interface_descriptor int13_command_desc = +	INTF_DESC ( struct int13_command, block, int13_command_op ); + +/** + * Open (or reopen) INT 13 emulated drive underlying block device + * + * @v int13		Emulated drive + * @ret rc		Return status code + */ +static int int13_reopen_block ( struct int13_drive *int13 ) { +	int rc; + +	/* Close any existing block device */ +	intf_restart ( &int13->block, -ECONNRESET ); + +	/* Open block device */ +	if ( ( rc = xfer_open_uri ( &int13->block, int13->uri ) ) != 0 ) { +		DBGC ( int13, "INT13 drive %02x could not reopen block " +		       "device: %s\n", int13->drive, strerror ( rc ) ); +		int13->block_rc = rc; +		return rc; +	} + +	/* Clear block device error status */ +	int13->block_rc = 0; + +	return 0; +} + +/** + * Prepare to issue INT 13 command + * + * @v command		INT 13 command + * @v int13		Emulated drive + * @ret rc		Return status code + */ +static int int13_command_start ( struct int13_command *command, +				 struct int13_drive *int13 ) { +	int rc; + +	/* Sanity check */ +	assert ( command->int13 == NULL ); +	assert ( ! timer_running ( &command->timer ) ); + +	/* Reopen block device if necessary */ +	if ( ( int13->block_rc != 0 ) && +	     ( ( rc = int13_reopen_block ( int13 ) ) != 0 ) ) +		return rc; + +	/* Initialise command */ +	command->rc = -EINPROGRESS; +	command->int13 = int13; +	start_timer_fixed ( &command->timer, INT13_COMMAND_TIMEOUT ); + +	/* Wait for block control interface to become ready */ +	while ( ( command->rc == -EINPROGRESS ) && +		( xfer_window ( &int13->block ) == 0 ) ) { +		step(); +	} + +	return ( ( command->rc == -EINPROGRESS ) ? +		 int13->block_rc : command->rc ); +} + +/** + * Wait for INT 13 command to complete + * + * @v command		INT 13 command + * @ret rc		Return status code + */ +static int int13_command_wait ( struct int13_command *command ) { + +	/* Sanity check */ +	assert ( timer_running ( &command->timer ) ); + +	/* Wait for command to complete */ +	while ( command->rc == -EINPROGRESS ) +		step(); + +	assert ( ! timer_running ( &command->timer ) ); +	return command->rc; +} + +/** + * Terminate INT 13 command + * + * @v command		INT 13 command + */ +static void int13_command_stop ( struct int13_command *command ) { +	stop_timer ( &command->timer ); +	command->int13 = NULL; +} + +/** The single active INT 13 command */ +static struct int13_command int13_command = { +	.block = INTF_INIT ( int13_command_desc ), +	.timer = TIMER_INIT ( int13_command_expired ), +}; + +/** + * Read from or write to INT 13 drive + * + * @v int13		Emulated drive + * @v lba		Starting logical block address + * @v count		Number of logical blocks + * @v buffer		Data buffer + * @v block_rw		Block read/write method + * @ret rc		Return status code + */ +static int int13_rw ( struct int13_drive *int13, uint64_t lba, +		      unsigned int count, userptr_t buffer, +		      int ( * block_rw ) ( struct interface *control, +					   struct interface *data, +					   uint64_t lba, unsigned int count, +					   userptr_t buffer, size_t len ) ) { +	struct int13_command *command = &int13_command; +	unsigned int frag_count; +	size_t frag_len; +	int rc; + +	/* Translate to underlying blocksize */ +	lba <<= int13->blksize_shift; +	count <<= int13->blksize_shift; + +	while ( count ) { + +		/* Determine fragment length */ +		frag_count = count; +		if ( frag_count > int13->capacity.max_count ) +			frag_count = int13->capacity.max_count; +		frag_len = ( int13->capacity.blksize * frag_count ); + +		/* Issue command */ +		if ( ( ( rc = int13_command_start ( command, int13 ) ) != 0 ) || +		     ( ( rc = block_rw ( &int13->block, &command->block, lba, +					 frag_count, buffer, +					 frag_len ) ) != 0 ) || +		     ( ( rc = int13_command_wait ( command ) ) != 0 ) ) { +			int13_command_stop ( command ); +			return rc; +		} +		int13_command_stop ( command ); + +		/* Move to next fragment */ +		lba += frag_count; +		count -= frag_count; +		buffer = userptr_add ( buffer, frag_len ); +	} + +	return 0; +} + +/** + * Read INT 13 drive capacity + * + * @v int13		Emulated drive + * @ret rc		Return status code + */ +static int int13_read_capacity ( struct int13_drive *int13 ) { +	struct int13_command *command = &int13_command; +	int rc; + +	/* Issue command */ +	if ( ( ( rc = int13_command_start ( command, int13 ) ) != 0 ) || +	     ( ( rc = block_read_capacity ( &int13->block, +					    &command->block ) ) != 0 ) || +	     ( ( rc = int13_command_wait ( command ) ) != 0 ) ) { +		int13_command_stop ( command ); +		return rc; +	} + +	int13_command_stop ( command ); +	return 0; +} + +/** + * Parse ISO9660 parameters + * + * @v int13		Emulated drive + * @v scratch		Scratch area for single-sector reads + * @ret rc		Return status code + * + * Reads and parses ISO9660 parameters, if present. + */ +static int int13_parse_iso9660 ( struct int13_drive *int13, void *scratch ) { +	static const struct iso9660_primary_descriptor_fixed primary_check = { +		.type = ISO9660_TYPE_PRIMARY, +		.id = ISO9660_ID, +	}; +	struct iso9660_primary_descriptor *primary = scratch; +	static const struct eltorito_descriptor_fixed boot_check = { +		.type = ISO9660_TYPE_BOOT, +		.id = ISO9660_ID, +		.version = 1, +		.system_id = "EL TORITO SPECIFICATION", +	}; +	struct eltorito_descriptor *boot = scratch; +	unsigned int blksize; +	unsigned int blksize_shift; +	int rc; + +	/* Calculate required blocksize shift */ +	blksize = int13_blksize ( int13 ); +	blksize_shift = 0; +	while ( blksize < ISO9660_BLKSIZE ) { +		blksize <<= 1; +		blksize_shift++; +	} +	if ( blksize > ISO9660_BLKSIZE ) { +		/* Do nothing if the blksize is invalid for CD-ROM access */ +		return 0; +	} + +	/* Read primary volume descriptor */ +	if ( ( rc = int13_rw ( int13, +			       ( ISO9660_PRIMARY_LBA << blksize_shift ), 1, +			       virt_to_user ( primary ), block_read ) ) != 0 ){ +		DBGC ( int13, "INT13 drive %02x could not read ISO9660 " +		       "primary volume descriptor: %s\n", +		       int13->drive, strerror ( rc ) ); +		return rc; +	} + +	/* Do nothing unless this is an ISO image */ +	if ( memcmp ( primary, &primary_check, sizeof ( primary_check ) ) != 0 ) +		return 0; +	DBGC ( int13, "INT13 drive %02x contains an ISO9660 filesystem; " +	       "treating as CD-ROM\n", int13->drive ); +	int13->is_cdrom = 1; + +	/* Read boot record volume descriptor */ +	if ( ( rc = int13_rw ( int13, +			       ( ELTORITO_LBA << blksize_shift ), 1, +			       virt_to_user ( boot ), block_read ) ) != 0 ) { +		DBGC ( int13, "INT13 drive %02x could not read El Torito boot " +		       "record volume descriptor: %s\n", +		       int13->drive, strerror ( rc ) ); +		return rc; +	} + +	/* Check for an El Torito boot catalog */ +	if ( memcmp ( boot, &boot_check, sizeof ( boot_check ) ) == 0 ) { +		int13->boot_catalog = boot->sector; +		DBGC ( int13, "INT13 drive %02x has an El Torito boot catalog " +		       "at LBA %08x\n", int13->drive, int13->boot_catalog ); +	} else { +		DBGC ( int13, "INT13 drive %02x has no El Torito boot " +		       "catalog\n", int13->drive ); +	} + +	/* Configure drive for no-emulation CD-ROM access */ +	int13->blksize_shift += blksize_shift; + +	return 0; +} + +/** + * Guess INT 13 hard disk drive geometry + * + * @v int13		Emulated drive + * @v scratch		Scratch area for single-sector reads + * @ret heads		Guessed number of heads + * @ret sectors		Guessed number of sectors per track + * @ret rc		Return status code + * + * Guesses the drive geometry by inspecting the partition table. + */ +static int int13_guess_geometry_hdd ( struct int13_drive *int13, void *scratch, +				      unsigned int *heads, +				      unsigned int *sectors ) { +	struct master_boot_record *mbr = scratch; +	struct partition_table_entry *partition; +	unsigned int i; +	int rc; + +	/* Default guess is xx/255/63 */ +	*heads = 255; +	*sectors = 63; + +	/* Read partition table */ +	if ( ( rc = int13_rw ( int13, 0, 1, virt_to_user ( mbr ), +			       block_read ) ) != 0 ) { +		DBGC ( int13, "INT13 drive %02x could not read " +		       "partition table to guess geometry: %s\n", +		       int13->drive, strerror ( rc ) ); +		return rc; +	} +	DBGC2 ( int13, "INT13 drive %02x has MBR:\n", int13->drive ); +	DBGC2_HDA ( int13, 0, mbr, sizeof ( *mbr ) ); +	DBGC ( int13, "INT13 drive %02x has signature %08x\n", +	       int13->drive, mbr->signature ); + +	/* Scan through partition table and modify guesses for +	 * heads and sectors_per_track if we find any used +	 * partitions. +	 */ +	for ( i = 0 ; i < 4 ; i++ ) { +		partition = &mbr->partitions[i]; +		if ( ! partition->type ) +			continue; +		*heads = ( PART_HEAD ( partition->chs_end ) + 1 ); +		*sectors = PART_SECTOR ( partition->chs_end ); +		DBGC ( int13, "INT13 drive %02x guessing C/H/S xx/%d/%d based " +		       "on partition %d\n", +		       int13->drive, *heads, *sectors, ( i + 1 ) ); +	} + +	return 0; +} + +/** Recognised floppy disk geometries */ +static const struct int13_fdd_geometry int13_fdd_geometries[] = { +	INT13_FDD_GEOMETRY ( 40, 1, 8 ), +	INT13_FDD_GEOMETRY ( 40, 1, 9 ), +	INT13_FDD_GEOMETRY ( 40, 2, 8 ), +	INT13_FDD_GEOMETRY ( 40, 1, 9 ), +	INT13_FDD_GEOMETRY ( 80, 2, 8 ), +	INT13_FDD_GEOMETRY ( 80, 2, 9 ), +	INT13_FDD_GEOMETRY ( 80, 2, 15 ), +	INT13_FDD_GEOMETRY ( 80, 2, 18 ), +	INT13_FDD_GEOMETRY ( 80, 2, 20 ), +	INT13_FDD_GEOMETRY ( 80, 2, 21 ), +	INT13_FDD_GEOMETRY ( 82, 2, 21 ), +	INT13_FDD_GEOMETRY ( 83, 2, 21 ), +	INT13_FDD_GEOMETRY ( 80, 2, 22 ), +	INT13_FDD_GEOMETRY ( 80, 2, 23 ), +	INT13_FDD_GEOMETRY ( 80, 2, 24 ), +	INT13_FDD_GEOMETRY ( 80, 2, 36 ), +	INT13_FDD_GEOMETRY ( 80, 2, 39 ), +	INT13_FDD_GEOMETRY ( 80, 2, 40 ), +	INT13_FDD_GEOMETRY ( 80, 2, 44 ), +	INT13_FDD_GEOMETRY ( 80, 2, 48 ), +}; + +/** + * Guess INT 13 floppy disk drive geometry + * + * @v int13		Emulated drive + * @ret heads		Guessed number of heads + * @ret sectors		Guessed number of sectors per track + * @ret rc		Return status code + * + * Guesses the drive geometry by inspecting the disk size. + */ +static int int13_guess_geometry_fdd ( struct int13_drive *int13, +				      unsigned int *heads, +				      unsigned int *sectors ) { +	unsigned int blocks = int13_capacity ( int13 ); +	const struct int13_fdd_geometry *geometry; +	unsigned int cylinders; +	unsigned int i; + +	/* Look for a match against a known geometry */ +	for ( i = 0 ; i < ( sizeof ( int13_fdd_geometries ) / +			    sizeof ( int13_fdd_geometries[0] ) ) ; i++ ) { +		geometry = &int13_fdd_geometries[i]; +		cylinders = INT13_FDD_CYLINDERS ( geometry ); +		*heads = INT13_FDD_HEADS ( geometry ); +		*sectors = INT13_FDD_SECTORS ( geometry ); +		if ( ( cylinders * (*heads) * (*sectors) ) == blocks ) { +			DBGC ( int13, "INT13 drive %02x guessing C/H/S " +			       "%d/%d/%d based on size %dK\n", int13->drive, +			       cylinders, *heads, *sectors, ( blocks / 2 ) ); +			return 0; +		} +	} + +	/* Otherwise, assume a partial disk image in the most common +	 * format (1440K, 80/2/18). +	 */ +	*heads = 2; +	*sectors = 18; +	DBGC ( int13, "INT13 drive %02x guessing C/H/S xx/%d/%d based on size " +	       "%dK\n", int13->drive, *heads, *sectors, ( blocks / 2 ) ); +	return 0; +} + +/** + * Guess INT 13 drive geometry + * + * @v int13		Emulated drive + * @v scratch		Scratch area for single-sector reads + * @ret rc		Return status code + */ +static int int13_guess_geometry ( struct int13_drive *int13, void *scratch ) { +	unsigned int guessed_heads; +	unsigned int guessed_sectors; +	unsigned int blocks; +	unsigned int blocks_per_cyl; +	int rc; + +	/* Don't even try when the blksize is invalid for C/H/S access */ +	if ( int13_blksize ( int13 ) != INT13_BLKSIZE ) +		return 0; + +	/* Guess geometry according to drive type */ +	if ( int13_is_fdd ( int13 ) ) { +		if ( ( rc = int13_guess_geometry_fdd ( int13, &guessed_heads, +						       &guessed_sectors )) != 0) +			return rc; +	} else { +		if ( ( rc = int13_guess_geometry_hdd ( int13, scratch, +						       &guessed_heads, +						       &guessed_sectors )) != 0) +			return rc; +	} + +	/* Apply guesses if no geometry already specified */ +	if ( ! int13->heads ) +		int13->heads = guessed_heads; +	if ( ! int13->sectors_per_track ) +		int13->sectors_per_track = guessed_sectors; +	if ( ! int13->cylinders ) { +		/* Avoid attempting a 64-bit divide on a 32-bit system */ +		blocks = int13_capacity32 ( int13 ); +		blocks_per_cyl = ( int13->heads * int13->sectors_per_track ); +		assert ( blocks_per_cyl != 0 ); +		int13->cylinders = ( blocks / blocks_per_cyl ); +		if ( int13->cylinders > 1024 ) +			int13->cylinders = 1024; +	} + +	return 0; +} + +/** + * Update BIOS drive count + */ +static void int13_sync_num_drives ( void ) { +	struct int13_drive *int13; +	uint8_t *counter; +	uint8_t max_drive; +	uint8_t required; + +	/* Get current drive counts */ +	get_real ( equipment_word, BDA_SEG, BDA_EQUIPMENT_WORD ); +	get_real ( num_drives, BDA_SEG, BDA_NUM_DRIVES ); +	num_fdds = ( ( equipment_word & 0x0001 ) ? +		     ( ( ( equipment_word >> 6 ) & 0x3 ) + 1 ) : 0 ); + +	/* Ensure count is large enough to cover all of our emulated drives */ +	list_for_each_entry ( int13, &int13s, list ) { +		counter = ( int13_is_fdd ( int13 ) ? &num_fdds : &num_drives ); +		max_drive = int13->drive; +		if ( max_drive < int13->natural_drive ) +			max_drive = int13->natural_drive; +		required = ( ( max_drive & 0x7f ) + 1 ); +		if ( *counter < required ) { +			*counter = required; +			DBGC ( int13, "INT13 drive %02x added to drive count: " +			       "%d HDDs, %d FDDs\n", +			       int13->drive, num_drives, num_fdds ); +		} +	} + +	/* Update current drive count */ +	equipment_word &= ~( ( 0x3 << 6 ) | 0x0001 ); +	if ( num_fdds ) { +		equipment_word |= ( 0x0001 | +				    ( ( ( num_fdds - 1 ) & 0x3 ) << 6 ) ); +	} +	put_real ( equipment_word, BDA_SEG, BDA_EQUIPMENT_WORD ); +	put_real ( num_drives, BDA_SEG, BDA_NUM_DRIVES ); +} + +/** + * Check number of drives + */ +static void int13_check_num_drives ( void ) { +	uint16_t check_equipment_word; +	uint8_t check_num_drives; + +	get_real ( check_equipment_word, BDA_SEG, BDA_EQUIPMENT_WORD ); +	get_real ( check_num_drives, BDA_SEG, BDA_NUM_DRIVES ); +	if ( ( check_equipment_word != equipment_word ) || +	     ( check_num_drives != num_drives ) ) { +		int13_sync_num_drives(); +	} +} + +/** + * INT 13, 00 - Reset disk system + * + * @v int13		Emulated drive + * @ret status		Status code + */ +static int int13_reset ( struct int13_drive *int13, +			 struct i386_all_regs *ix86 __unused ) { +	int rc; + +	DBGC2 ( int13, "Reset drive\n" ); + +	/* Reopen underlying block device */ +	if ( ( rc = int13_reopen_block ( int13 ) ) != 0 ) +		return -INT13_STATUS_RESET_FAILED; + +	/* Check that block device is functional */ +	if ( ( rc = int13_read_capacity ( int13 ) ) != 0 ) +		return -INT13_STATUS_RESET_FAILED; + +	return 0; +} + +/** + * INT 13, 01 - Get status of last operation + * + * @v int13		Emulated drive + * @ret status		Status code + */ +static int int13_get_last_status ( struct int13_drive *int13, +				   struct i386_all_regs *ix86 __unused ) { +	DBGC2 ( int13, "Get status of last operation\n" ); +	return int13->last_status; +} + +/** + * Read / write sectors + * + * @v int13		Emulated drive + * @v al		Number of sectors to read or write (must be nonzero) + * @v ch		Low bits of cylinder number + * @v cl (bits 7:6)	High bits of cylinder number + * @v cl (bits 5:0)	Sector number + * @v dh		Head number + * @v es:bx		Data buffer + * @v block_rw		Block read/write method + * @ret status		Status code + * @ret al		Number of sectors read or written + */ +static int int13_rw_sectors ( struct int13_drive *int13, +			      struct i386_all_regs *ix86, +			      int ( * block_rw ) ( struct interface *control, +						   struct interface *data, +						   uint64_t lba, +						   unsigned int count, +						   userptr_t buffer, +						   size_t len ) ) { +	unsigned int cylinder, head, sector; +	unsigned long lba; +	unsigned int count; +	userptr_t buffer; +	int rc; + +	/* Validate blocksize */ +	if ( int13_blksize ( int13 ) != INT13_BLKSIZE ) { +		DBGC ( int13, "\nINT 13 drive %02x invalid blocksize (%zd) " +		       "for non-extended read/write\n", +		       int13->drive, int13_blksize ( int13 ) ); +		return -INT13_STATUS_INVALID; +	} + +	/* Calculate parameters */ +	cylinder = ( ( ( ix86->regs.cl & 0xc0 ) << 2 ) | ix86->regs.ch ); +	head = ix86->regs.dh; +	sector = ( ix86->regs.cl & 0x3f ); +	if ( ( cylinder >= int13->cylinders ) || +	     ( head >= int13->heads ) || +	     ( sector < 1 ) || ( sector > int13->sectors_per_track ) ) { +		DBGC ( int13, "C/H/S %d/%d/%d out of range for geometry " +		       "%d/%d/%d\n", cylinder, head, sector, int13->cylinders, +		       int13->heads, int13->sectors_per_track ); +		return -INT13_STATUS_INVALID; +	} +	lba = ( ( ( ( cylinder * int13->heads ) + head ) +		  * int13->sectors_per_track ) + sector - 1 ); +	count = ix86->regs.al; +	buffer = real_to_user ( ix86->segs.es, ix86->regs.bx ); + +	DBGC2 ( int13, "C/H/S %d/%d/%d = LBA %08lx <-> %04x:%04x (count %d)\n", +		cylinder, head, sector, lba, ix86->segs.es, ix86->regs.bx, +		count ); + +	/* Read from / write to block device */ +	if ( ( rc = int13_rw ( int13, lba, count, buffer, block_rw ) ) != 0 ) { +		DBGC ( int13, "INT13 drive %02x I/O failed: %s\n", +		       int13->drive, strerror ( rc ) ); +		return -INT13_STATUS_READ_ERROR; +	} + +	return 0; +} + +/** + * INT 13, 02 - Read sectors + * + * @v int13		Emulated drive + * @v al		Number of sectors to read (must be nonzero) + * @v ch		Low bits of cylinder number + * @v cl (bits 7:6)	High bits of cylinder number + * @v cl (bits 5:0)	Sector number + * @v dh		Head number + * @v es:bx		Data buffer + * @ret status		Status code + * @ret al		Number of sectors read + */ +static int int13_read_sectors ( struct int13_drive *int13, +				struct i386_all_regs *ix86 ) { +	DBGC2 ( int13, "Read: " ); +	return int13_rw_sectors ( int13, ix86, block_read ); +} + +/** + * INT 13, 03 - Write sectors + * + * @v int13		Emulated drive + * @v al		Number of sectors to write (must be nonzero) + * @v ch		Low bits of cylinder number + * @v cl (bits 7:6)	High bits of cylinder number + * @v cl (bits 5:0)	Sector number + * @v dh		Head number + * @v es:bx		Data buffer + * @ret status		Status code + * @ret al		Number of sectors written + */ +static int int13_write_sectors ( struct int13_drive *int13, +				 struct i386_all_regs *ix86 ) { +	DBGC2 ( int13, "Write: " ); +	return int13_rw_sectors ( int13, ix86, block_write ); +} + +/** + * INT 13, 08 - Get drive parameters + * + * @v int13		Emulated drive + * @ret status		Status code + * @ret ch		Low bits of maximum cylinder number + * @ret cl (bits 7:6)	High bits of maximum cylinder number + * @ret cl (bits 5:0)	Maximum sector number + * @ret dh		Maximum head number + * @ret dl		Number of drives + */ +static int int13_get_parameters ( struct int13_drive *int13, +				  struct i386_all_regs *ix86 ) { +	unsigned int max_cylinder = int13->cylinders - 1; +	unsigned int max_head = int13->heads - 1; +	unsigned int max_sector = int13->sectors_per_track; /* sic */ + +	DBGC2 ( int13, "Get drive parameters\n" ); + +	/* Validate blocksize */ +	if ( int13_blksize ( int13 ) != INT13_BLKSIZE ) { +		DBGC ( int13, "\nINT 13 drive %02x invalid blocksize (%zd) " +		       "for non-extended parameters\n", +		       int13->drive, int13_blksize ( int13 ) ); +		return -INT13_STATUS_INVALID; +	} + +	/* Common parameters */ +	ix86->regs.ch = ( max_cylinder & 0xff ); +	ix86->regs.cl = ( ( ( max_cylinder >> 8 ) << 6 ) | max_sector ); +	ix86->regs.dh = max_head; +	ix86->regs.dl = ( int13_is_fdd ( int13 ) ? num_fdds : num_drives ); + +	/* Floppy-specific parameters */ +	if ( int13_is_fdd ( int13 ) ) { +		ix86->regs.bl = INT13_FDD_TYPE_1M44; +		ix86->segs.es = rm_ds; +		ix86->regs.di = __from_data16 ( &int13_fdd_params ); +	} + +	return 0; +} + +/** + * INT 13, 15 - Get disk type + * + * @v int13		Emulated drive + * @ret ah		Type code + * @ret cx:dx		Sector count + * @ret status		Status code / disk type + */ +static int int13_get_disk_type ( struct int13_drive *int13, +				 struct i386_all_regs *ix86 ) { +	uint32_t blocks; + +	DBGC2 ( int13, "Get disk type\n" ); + +	if ( int13_is_fdd ( int13 ) ) { +		return INT13_DISK_TYPE_FDD; +	} else { +		blocks = int13_capacity32 ( int13 ); +		ix86->regs.cx = ( blocks >> 16 ); +		ix86->regs.dx = ( blocks & 0xffff ); +		return INT13_DISK_TYPE_HDD; +	} +} + +/** + * INT 13, 41 - Extensions installation check + * + * @v int13		Emulated drive + * @v bx		0x55aa + * @ret bx		0xaa55 + * @ret cx		Extensions API support bitmap + * @ret status		Status code / API version + */ +static int int13_extension_check ( struct int13_drive *int13 __unused, +				   struct i386_all_regs *ix86 ) { +	if ( ix86->regs.bx == 0x55aa ) { +		DBGC2 ( int13, "INT13 extensions installation check\n" ); +		ix86->regs.bx = 0xaa55; +		ix86->regs.cx = ( INT13_EXTENSION_LINEAR | +				  INT13_EXTENSION_EDD | +				  INT13_EXTENSION_64BIT ); +		return INT13_EXTENSION_VER_3_0; +	} else { +		return -INT13_STATUS_INVALID; +	} +} + +/** + * Extended read / write + * + * @v int13		Emulated drive + * @v ds:si		Disk address packet + * @v block_rw		Block read/write method + * @ret status		Status code + */ +static int int13_extended_rw ( struct int13_drive *int13, +			       struct i386_all_regs *ix86, +			       int ( * block_rw ) ( struct interface *control, +						    struct interface *data, +						    uint64_t lba, +						    unsigned int count, +						    userptr_t buffer, +						    size_t len ) ) { +	struct int13_disk_address addr; +	uint8_t bufsize; +	uint64_t lba; +	unsigned long count; +	userptr_t buffer; +	int rc; + +	/* Extended reads are not allowed on floppy drives. +	 * ELTORITO.SYS seems to assume that we are really a CD-ROM if +	 * we support extended reads for a floppy drive. +	 */ +	if ( int13_is_fdd ( int13 ) ) +		return -INT13_STATUS_INVALID; + +	/* Get buffer size */ +	get_real ( bufsize, ix86->segs.ds, +		   ( ix86->regs.si + offsetof ( typeof ( addr ), bufsize ) ) ); +	if ( bufsize < offsetof ( typeof ( addr ), buffer_phys ) ) { +		DBGC2 ( int13, "<invalid buffer size %#02x\n>\n", bufsize ); +		return -INT13_STATUS_INVALID; +	} + +	/* Read parameters from disk address structure */ +	memset ( &addr, 0, sizeof ( addr ) ); +	copy_from_real ( &addr, ix86->segs.ds, ix86->regs.si, bufsize ); +	lba = addr.lba; +	DBGC2 ( int13, "LBA %08llx <-> ", ( ( unsigned long long ) lba ) ); +	if ( ( addr.count == 0xff ) || +	     ( ( addr.buffer.segment == 0xffff ) && +	       ( addr.buffer.offset == 0xffff ) ) ) { +		buffer = phys_to_user ( addr.buffer_phys ); +		DBGC2 ( int13, "%08llx", +			( ( unsigned long long ) addr.buffer_phys ) ); +	} else { +		buffer = real_to_user ( addr.buffer.segment, +					addr.buffer.offset ); +		DBGC2 ( int13, "%04x:%04x", addr.buffer.segment, +			addr.buffer.offset ); +	} +	if ( addr.count <= 0x7f ) { +		count = addr.count; +	} else if ( addr.count == 0xff ) { +		count = addr.long_count; +	} else { +		DBGC2 ( int13, " <invalid count %#02x>\n", addr.count ); +		return -INT13_STATUS_INVALID; +	} +	DBGC2 ( int13, " (count %ld)\n", count ); + +	/* Read from / write to block device */ +	if ( ( rc = int13_rw ( int13, lba, count, buffer, block_rw ) ) != 0 ) { +		DBGC ( int13, "INT13 drive %02x extended I/O failed: %s\n", +		       int13->drive, strerror ( rc ) ); +		/* Record that no blocks were transferred successfully */ +		addr.count = 0; +		put_real ( addr.count, ix86->segs.ds, +			   ( ix86->regs.si + +			     offsetof ( typeof ( addr ), count ) ) ); +		return -INT13_STATUS_READ_ERROR; +	} + +	return 0; +} + +/** + * INT 13, 42 - Extended read + * + * @v int13		Emulated drive + * @v ds:si		Disk address packet + * @ret status		Status code + */ +static int int13_extended_read ( struct int13_drive *int13, +				 struct i386_all_regs *ix86 ) { +	DBGC2 ( int13, "Extended read: " ); +	return int13_extended_rw ( int13, ix86, block_read ); +} + +/** + * INT 13, 43 - Extended write + * + * @v int13		Emulated drive + * @v ds:si		Disk address packet + * @ret status		Status code + */ +static int int13_extended_write ( struct int13_drive *int13, +				  struct i386_all_regs *ix86 ) { +	DBGC2 ( int13, "Extended write: " ); +	return int13_extended_rw ( int13, ix86, block_write ); +} + +/** + * INT 13, 44 - Verify sectors + * + * @v int13		Emulated drive + * @v ds:si		Disk address packet + * @ret status		Status code + */ +static int int13_extended_verify ( struct int13_drive *int13, +				   struct i386_all_regs *ix86 ) { +	struct int13_disk_address addr; +	uint64_t lba; +	unsigned long count; + +	/* Read parameters from disk address structure */ +	if ( DBG_EXTRA ) { +		copy_from_real ( &addr, ix86->segs.ds, ix86->regs.si, +				 sizeof ( addr )); +		lba = addr.lba; +		count = addr.count; +		DBGC2 ( int13, "Verify: LBA %08llx (count %ld)\n", +			( ( unsigned long long ) lba ), count ); +	} + +	/* We have no mechanism for verifying sectors */ +	return -INT13_STATUS_INVALID; +} + +/** + * INT 13, 44 - Extended seek + * + * @v int13		Emulated drive + * @v ds:si		Disk address packet + * @ret status		Status code + */ +static int int13_extended_seek ( struct int13_drive *int13, +				 struct i386_all_regs *ix86 ) { +	struct int13_disk_address addr; +	uint64_t lba; +	unsigned long count; + +	/* Read parameters from disk address structure */ +	if ( DBG_EXTRA ) { +		copy_from_real ( &addr, ix86->segs.ds, ix86->regs.si, +				 sizeof ( addr )); +		lba = addr.lba; +		count = addr.count; +		DBGC2 ( int13, "Seek: LBA %08llx (count %ld)\n", +			( ( unsigned long long ) lba ), count ); +	} + +	/* Ignore and return success */ +	return 0; +} + +/** + * Build device path information + * + * @v int13		Emulated drive + * @v dpi		Device path information + * @ret rc		Return status code + */ +static int int13_device_path_info ( struct int13_drive *int13, +				    struct edd_device_path_information *dpi ) { +	struct device *device; +	struct device_description *desc; +	unsigned int i; +	uint8_t sum = 0; +	int rc; + +	/* Reopen block device if necessary */ +	if ( ( int13->block_rc != 0 ) && +	     ( ( rc = int13_reopen_block ( int13 ) ) != 0 ) ) +		return rc; + +	/* Get underlying hardware device */ +	device = identify_device ( &int13->block ); +	if ( ! device ) { +		DBGC ( int13, "INT13 drive %02x cannot identify hardware " +		       "device\n", int13->drive ); +		return -ENODEV; +	} + +	/* Fill in bus type and interface path */ +	desc = &device->desc; +	switch ( desc->bus_type ) { +	case BUS_TYPE_PCI: +		dpi->host_bus_type.type = EDD_BUS_TYPE_PCI; +		dpi->interface_path.pci.bus = PCI_BUS ( desc->location ); +		dpi->interface_path.pci.slot = PCI_SLOT ( desc->location ); +		dpi->interface_path.pci.function = PCI_FUNC ( desc->location ); +		dpi->interface_path.pci.channel = 0xff; /* unused */ +		break; +	default: +		DBGC ( int13, "INT13 drive %02x unrecognised bus type %d\n", +		       int13->drive, desc->bus_type ); +		return -ENOTSUP; +	} + +	/* Get EDD block device description */ +	if ( ( rc = edd_describe ( &int13->block, &dpi->interface_type, +				   &dpi->device_path ) ) != 0 ) { +		DBGC ( int13, "INT13 drive %02x cannot identify block device: " +		       "%s\n", int13->drive, strerror ( rc ) ); +		return rc; +	} + +	/* Fill in common fields and fix checksum */ +	dpi->key = EDD_DEVICE_PATH_INFO_KEY; +	dpi->len = sizeof ( *dpi ); +	for ( i = 0 ; i < sizeof ( *dpi ) ; i++ ) +		sum += *( ( ( uint8_t * ) dpi ) + i ); +	dpi->checksum -= sum; + +	return 0; +} + +/** + * INT 13, 48 - Get extended parameters + * + * @v int13		Emulated drive + * @v ds:si		Drive parameter table + * @ret status		Status code + */ +static int int13_get_extended_parameters ( struct int13_drive *int13, +					   struct i386_all_regs *ix86 ) { +	struct int13_disk_parameters params; +	struct segoff address; +	size_t len = sizeof ( params ); +	uint16_t bufsize; +	int rc; + +	/* Get buffer size */ +	get_real ( bufsize, ix86->segs.ds, +		   ( ix86->regs.si + offsetof ( typeof ( params ), bufsize ))); + +	DBGC2 ( int13, "Get extended drive parameters to %04x:%04x+%02x\n", +		ix86->segs.ds, ix86->regs.si, bufsize ); + +	/* Build drive parameters */ +	memset ( ¶ms, 0, sizeof ( params ) ); +	params.flags = INT13_FL_DMA_TRANSPARENT; +	if ( ( int13->cylinders < 1024 ) && +	     ( int13_capacity ( int13 ) <= INT13_MAX_CHS_SECTORS ) ) { +		params.flags |= INT13_FL_CHS_VALID; +	} +	params.cylinders = int13->cylinders; +	params.heads = int13->heads; +	params.sectors_per_track = int13->sectors_per_track; +	params.sectors = int13_capacity ( int13 ); +	params.sector_size = int13_blksize ( int13 ); +	memset ( ¶ms.dpte, 0xff, sizeof ( params.dpte ) ); +	if ( ( rc = int13_device_path_info ( int13, ¶ms.dpi ) ) != 0 ) { +		DBGC ( int13, "INT13 drive %02x could not provide device " +		       "path information: %s\n", +		       int13->drive, strerror ( rc ) ); +		len = offsetof ( typeof ( params ), dpi ); +	} + +	/* Calculate returned "buffer size" (which will be less than +	 * the length actually copied if device path information is +	 * present). +	 */ +	if ( bufsize < offsetof ( typeof ( params ), dpte ) ) +		return -INT13_STATUS_INVALID; +	if ( bufsize < offsetof ( typeof ( params ), dpi ) ) { +		params.bufsize = offsetof ( typeof ( params ), dpte ); +	} else { +		params.bufsize = offsetof ( typeof ( params ), dpi ); +	} + +	DBGC ( int13, "INT 13 drive %02x described using extended " +	       "parameters:\n", int13->drive ); +	address.segment = ix86->segs.ds; +	address.offset = ix86->regs.si; +	DBGC_HDA ( int13, address, ¶ms, len ); + +	/* Return drive parameters */ +	if ( len > bufsize ) +		len = bufsize; +	copy_to_real ( ix86->segs.ds, ix86->regs.si, ¶ms, len ); + +	return 0; +} + +/** + * INT 13, 4b - Get status or terminate CD-ROM emulation + * + * @v int13		Emulated drive + * @v ds:si		Specification packet + * @ret status		Status code + */ +static int int13_cdrom_status_terminate ( struct int13_drive *int13, +					  struct i386_all_regs *ix86 ) { +	struct int13_cdrom_specification specification; + +	DBGC2 ( int13, "Get CD-ROM emulation status to %04x:%04x%s\n", +		ix86->segs.ds, ix86->regs.si, +		( ix86->regs.al ? "" : " and terminate" ) ); + +	/* Fail if we are not a CD-ROM */ +	if ( ! int13->is_cdrom ) { +		DBGC ( int13, "INT13 drive %02x is not a CD-ROM\n", +		       int13->drive ); +		return -INT13_STATUS_INVALID; +	} + +	/* Build specification packet */ +	memset ( &specification, 0, sizeof ( specification ) ); +	specification.size = sizeof ( specification ); +	specification.drive = int13->drive; + +	/* Return specification packet */ +	copy_to_real ( ix86->segs.ds, ix86->regs.si, &specification, +		       sizeof ( specification ) ); + +	return 0; +} + + +/** + * INT 13, 4d - Read CD-ROM boot catalog + * + * @v int13		Emulated drive + * @v ds:si		Command packet + * @ret status		Status code + */ +static int int13_cdrom_read_boot_catalog ( struct int13_drive *int13, +					   struct i386_all_regs *ix86 ) { +	struct int13_cdrom_boot_catalog_command command; +	int rc; + +	/* Read parameters from command packet */ +	copy_from_real ( &command, ix86->segs.ds, ix86->regs.si, +			 sizeof ( command ) ); +	DBGC2 ( int13, "Read CD-ROM boot catalog to %08x\n", command.buffer ); + +	/* Fail if we have no boot catalog */ +	if ( ! int13->boot_catalog ) { +		DBGC ( int13, "INT13 drive %02x has no boot catalog\n", +		       int13->drive ); +		return -INT13_STATUS_INVALID; +	} + +	/* Read from boot catalog */ +	if ( ( rc = int13_rw ( int13, ( int13->boot_catalog + command.start ), +			       command.count, phys_to_user ( command.buffer ), +			       block_read ) ) != 0 ) { +		DBGC ( int13, "INT13 drive %02x could not read boot catalog: " +		       "%s\n", int13->drive, strerror ( rc ) ); +		return -INT13_STATUS_READ_ERROR; +	} + +	return 0; +} + +/** + * INT 13 handler + * + */ +static __asmcall void int13 ( struct i386_all_regs *ix86 ) { +	int command = ix86->regs.ah; +	unsigned int bios_drive = ix86->regs.dl; +	struct int13_drive *int13; +	int status; + +	/* Check BIOS hasn't killed off our drive */ +	int13_check_num_drives(); + +	list_for_each_entry ( int13, &int13s, list ) { + +		if ( bios_drive != int13->drive ) { +			/* Remap any accesses to this drive's natural number */ +			if ( bios_drive == int13->natural_drive ) { +				DBGC2 ( int13, "INT13,%02x (%02x) remapped to " +					"(%02x)\n", ix86->regs.ah, +					bios_drive, int13->drive ); +				ix86->regs.dl = int13->drive; +				return; +			} else if ( ( ( bios_drive & 0x7f ) == 0x7f ) && +				    ( command == INT13_CDROM_STATUS_TERMINATE ) +				    && int13->is_cdrom ) { +				/* Catch non-drive-specific CD-ROM calls */ +			} else { +				continue; +			} +		} +		 +		DBGC2 ( int13, "INT13,%02x (%02x): ", +			ix86->regs.ah, bios_drive ); + +		switch ( command ) { +		case INT13_RESET: +			status = int13_reset ( int13, ix86 ); +			break; +		case INT13_GET_LAST_STATUS: +			status = int13_get_last_status ( int13, ix86 ); +			break; +		case INT13_READ_SECTORS: +			status = int13_read_sectors ( int13, ix86 ); +			break; +		case INT13_WRITE_SECTORS: +			status = int13_write_sectors ( int13, ix86 ); +			break; +		case INT13_GET_PARAMETERS: +			status = int13_get_parameters ( int13, ix86 ); +			break; +		case INT13_GET_DISK_TYPE: +			status = int13_get_disk_type ( int13, ix86 ); +			break; +		case INT13_EXTENSION_CHECK: +			status = int13_extension_check ( int13, ix86 ); +			break; +		case INT13_EXTENDED_READ: +			status = int13_extended_read ( int13, ix86 ); +			break; +		case INT13_EXTENDED_WRITE: +			status = int13_extended_write ( int13, ix86 ); +			break; +		case INT13_EXTENDED_VERIFY: +			status = int13_extended_verify ( int13, ix86 ); +			break; +		case INT13_EXTENDED_SEEK: +			status = int13_extended_seek ( int13, ix86 ); +			break; +		case INT13_GET_EXTENDED_PARAMETERS: +			status = int13_get_extended_parameters ( int13, ix86 ); +			break; +		case INT13_CDROM_STATUS_TERMINATE: +			status = int13_cdrom_status_terminate ( int13, ix86 ); +			break; +		case INT13_CDROM_READ_BOOT_CATALOG: +			status = int13_cdrom_read_boot_catalog ( int13, ix86 ); +			break; +		default: +			DBGC2 ( int13, "*** Unrecognised INT13 ***\n" ); +			status = -INT13_STATUS_INVALID; +			break; +		} + +		/* Store status for INT 13,01 */ +		int13->last_status = status; + +		/* Negative status indicates an error */ +		if ( status < 0 ) { +			status = -status; +			DBGC ( int13, "INT13,%02x (%02x) failed with status " +			       "%02x\n", ix86->regs.ah, int13->drive, status ); +		} else { +			ix86->flags &= ~CF; +		} +		ix86->regs.ah = status; + +		/* Set OF to indicate to wrapper not to chain this call */ +		ix86->flags |= OF; + +		return; +	} +} + +/** + * Hook INT 13 handler + * + */ +static void int13_hook_vector ( void ) { +	/* Assembly wrapper to call int13().  int13() sets OF if we +	 * should not chain to the previous handler.  (The wrapper +	 * clears CF and OF before calling int13()). +	 */ +	__asm__  __volatile__ ( +	       TEXT16_CODE ( "\nint13_wrapper:\n\t" +			     /* Preserve %ax and %dx for future reference */ +			     "pushw %%bp\n\t" +			     "movw %%sp, %%bp\n\t"			      +			     "pushw %%ax\n\t" +			     "pushw %%dx\n\t" +			     /* Clear OF, set CF, call int13() */ +			     "orb $0, %%al\n\t"  +			     "stc\n\t" +			     "pushl %0\n\t" +			     "pushw %%cs\n\t" +			     "call prot_call\n\t" +			     /* Chain if OF not set */ +			     "jo 1f\n\t" +			     "pushfw\n\t" +			     "lcall *%%cs:int13_vector\n\t" +			     "\n1:\n\t" +			     /* Overwrite flags for iret */ +			     "pushfw\n\t" +			     "popw 6(%%bp)\n\t" +			     /* Fix up %dl: +			      * +			      * INT 13,15 : do nothing if hard disk +			      * INT 13,08 : load with number of drives +			      * all others: restore original value +			      */ +			     "cmpb $0x15, -1(%%bp)\n\t" +			     "jne 2f\n\t" +			     "testb $0x80, -4(%%bp)\n\t" +			     "jnz 3f\n\t" +			     "\n2:\n\t" +			     "movb -4(%%bp), %%dl\n\t" +			     "cmpb $0x08, -1(%%bp)\n\t" +			     "jne 3f\n\t" +			     "testb $0x80, %%dl\n\t" +			     "movb %%cs:num_drives, %%dl\n\t" +			     "jnz 3f\n\t" +			     "movb %%cs:num_fdds, %%dl\n\t" +			     /* Return */ +			     "\n3:\n\t" +			     "movw %%bp, %%sp\n\t" +			     "popw %%bp\n\t" +			     "iret\n\t" ) +	       : : "i" ( int13 ) ); + +	hook_bios_interrupt ( 0x13, ( unsigned int ) int13_wrapper, +			      &int13_vector ); +} + +/** + * Unhook INT 13 handler + */ +static void int13_unhook_vector ( void ) { +	unhook_bios_interrupt ( 0x13, ( unsigned int ) int13_wrapper, +				&int13_vector ); +} + +/** + * Check INT13 emulated drive flow control window + * + * @v int13		Emulated drive + */ +static size_t int13_block_window ( struct int13_drive *int13 __unused ) { + +	/* We are never ready to receive data via this interface. +	 * This prevents objects that support both block and stream +	 * interfaces from attempting to send us stream data. +	 */ +	return 0; +} + +/** + * Handle INT 13 emulated drive underlying block device closing + * + * @v int13		Emulated drive + * @v rc		Reason for close + */ +static void int13_block_close ( struct int13_drive *int13, int rc ) { + +	/* Any closing is an error from our point of view */ +	if ( rc == 0 ) +		rc = -ENOTCONN; + +	DBGC ( int13, "INT13 drive %02x went away: %s\n", +	       int13->drive, strerror ( rc ) ); + +	/* Record block device error code */ +	int13->block_rc = rc; + +	/* Shut down interfaces */ +	intf_restart ( &int13->block, rc ); +} + +/** INT 13 drive interface operations */ +static struct interface_operation int13_block_op[] = { +	INTF_OP ( xfer_window, struct int13_drive *, int13_block_window ), +	INTF_OP ( intf_close, struct int13_drive *, int13_block_close ), +}; + +/** INT 13 drive interface descriptor */ +static struct interface_descriptor int13_block_desc = +	INTF_DESC ( struct int13_drive, block, int13_block_op ); + +/** + * Free INT 13 emulated drive + * + * @v refcnt		Reference count + */ +static void int13_free ( struct refcnt *refcnt ) { +	struct int13_drive *int13 = +		container_of ( refcnt, struct int13_drive, refcnt ); + +	uri_put ( int13->uri ); +	free ( int13 ); +} + +/** + * Hook INT 13 emulated drive + * + * @v uri		URI + * @v drive		Drive number + * @ret rc		Return status code + * + * Registers the drive with the INT 13 emulation subsystem, and hooks + * the INT 13 interrupt vector (if not already hooked). + */ +static int int13_hook ( struct uri *uri, unsigned int drive ) { +	struct int13_drive *int13; +	unsigned int natural_drive; +	void *scratch; +	int rc; + +	/* Calculate natural drive number */ +	int13_sync_num_drives(); +	natural_drive = ( ( drive & 0x80 ) ? ( num_drives | 0x80 ) : num_fdds ); + +	/* Check that drive number is not in use */ +	list_for_each_entry ( int13, &int13s, list ) { +		if ( int13->drive == drive ) { +			rc = -EADDRINUSE; +			goto err_in_use; +		} +	} + +	/* Allocate and initialise structure */ +	int13 = zalloc ( sizeof ( *int13 ) ); +	if ( ! int13 ) { +		rc = -ENOMEM; +		goto err_zalloc; +	} +	ref_init ( &int13->refcnt, int13_free ); +	intf_init ( &int13->block, &int13_block_desc, &int13->refcnt ); +	int13->uri = uri_get ( uri ); +	int13->drive = drive; +	int13->natural_drive = natural_drive; + +	/* Open block device interface */ +	if ( ( rc = int13_reopen_block ( int13 ) ) != 0 ) +		goto err_reopen_block; + +	/* Read device capacity */ +	if ( ( rc = int13_read_capacity ( int13 ) ) != 0 ) +		goto err_read_capacity; + +	/* Allocate scratch area */ +	scratch = malloc ( int13_blksize ( int13 ) ); +	if ( ! scratch ) +		goto err_alloc_scratch; + +	/* Parse parameters, if present */ +	if ( ( rc = int13_parse_iso9660 ( int13, scratch ) ) != 0 ) +		goto err_parse_iso9660; + +	/* Give drive a default geometry */ +	if ( ( rc = int13_guess_geometry ( int13, scratch ) ) != 0 ) +		goto err_guess_geometry; + +	DBGC ( int13, "INT13 drive %02x (naturally %02x) registered with C/H/S " +	       "geometry %d/%d/%d\n", int13->drive, int13->natural_drive, +	       int13->cylinders, int13->heads, int13->sectors_per_track ); + +	/* Hook INT 13 vector if not already hooked */ +	if ( list_empty ( &int13s ) ) { +		int13_hook_vector(); +		devices_get(); +	} + +	/* Add to list of emulated drives */ +	list_add ( &int13->list, &int13s ); + +	/* Update BIOS drive count */ +	int13_sync_num_drives(); + +	free ( scratch ); +	return 0; + + err_guess_geometry: + err_parse_iso9660: +	free ( scratch ); + err_alloc_scratch: + err_read_capacity: + err_reopen_block: +	intf_shutdown ( &int13->block, rc ); +	ref_put ( &int13->refcnt ); + err_zalloc: + err_in_use: +	return rc; +} + +/** + * Find INT 13 emulated drive by drive number + * + * @v drive		Drive number + * @ret int13		Emulated drive, or NULL + */ +static struct int13_drive * int13_find ( unsigned int drive ) { +	struct int13_drive *int13; + +	list_for_each_entry ( int13, &int13s, list ) { +		if ( int13->drive == drive ) +			return int13; +	} +	return NULL; +} + +/** + * Unhook INT 13 emulated drive + * + * @v drive		Drive number + * + * Unregisters the drive from the INT 13 emulation subsystem.  If this + * is the last emulated drive, the INT 13 vector is unhooked (if + * possible). + */ +static void int13_unhook ( unsigned int drive ) { +	struct int13_drive *int13; + +	/* Find drive */ +	int13 = int13_find ( drive ); +	if ( ! int13 ) { +		DBG ( "INT13 cannot find emulated drive %02x\n", drive ); +		return; +	} + +	/* Shut down interfaces */ +	intf_shutdown ( &int13->block, 0 ); + +	/* Remove from list of emulated drives */ +	list_del ( &int13->list ); + +	/* Should adjust BIOS drive count, but it's difficult +	 * to do so reliably. +	 */ + +	DBGC ( int13, "INT13 drive %02x unregistered\n", int13->drive ); + +	/* Unhook INT 13 vector if no more drives */ +	if ( list_empty ( &int13s ) ) { +		devices_put(); +		int13_unhook_vector(); +	} + +	/* Drop list's reference to drive */ +	ref_put ( &int13->refcnt ); +} + +/** + * Load and verify master boot record from INT 13 drive + * + * @v drive		Drive number + * @v address		Boot code address to fill in + * @ret rc		Return status code + */ +static int int13_load_mbr ( unsigned int drive, struct segoff *address ) { +	uint8_t status; +	int discard_b, discard_c, discard_d; +	uint16_t magic; + +	/* Use INT 13, 02 to read the MBR */ +	address->segment = 0; +	address->offset = 0x7c00; +	__asm__ __volatile__ ( REAL_CODE ( "pushw %%es\n\t" +					   "pushl %%ebx\n\t" +					   "popw %%bx\n\t" +					   "popw %%es\n\t" +					   "stc\n\t" +					   "sti\n\t" +					   "int $0x13\n\t" +					   "sti\n\t" /* BIOS bugs */ +					   "jc 1f\n\t" +					   "xorw %%ax, %%ax\n\t" +					   "\n1:\n\t" +					   "popw %%es\n\t" ) +			       : "=a" ( status ), "=b" ( discard_b ), +				 "=c" ( discard_c ), "=d" ( discard_d ) +			       : "a" ( 0x0201 ), "b" ( *address ), +				 "c" ( 1 ), "d" ( drive ) ); +	if ( status ) { +		DBG ( "INT13 drive %02x could not read MBR (status %02x)\n", +		      drive, status ); +		return -EIO; +	} + +	/* Check magic signature */ +	get_real ( magic, address->segment, +		   ( address->offset + +		     offsetof ( struct master_boot_record, magic ) ) ); +	if ( magic != INT13_MBR_MAGIC ) { +		DBG ( "INT13 drive %02x does not contain a valid MBR\n", +		      drive ); +		return -ENOEXEC; +	} + +	return 0; +} + +/** El Torito boot catalog command packet */ +static struct int13_cdrom_boot_catalog_command __data16 ( eltorito_cmd ) = { +	.size = sizeof ( struct int13_cdrom_boot_catalog_command ), +	.count = 1, +	.buffer = 0x7c00, +	.start = 0, +}; +#define eltorito_cmd __use_data16 ( eltorito_cmd ) + +/** El Torito disk address packet */ +static struct int13_disk_address __bss16 ( eltorito_address ); +#define eltorito_address __use_data16 ( eltorito_address ) + +/** + * Load and verify El Torito boot record from INT 13 drive + * + * @v drive		Drive number + * @v address		Boot code address to fill in + * @ret rc		Return status code + */ +static int int13_load_eltorito ( unsigned int drive, struct segoff *address ) { +	struct { +		struct eltorito_validation_entry valid; +		struct eltorito_boot_entry boot; +	} __attribute__ (( packed )) catalog; +	uint8_t status; + +	/* Use INT 13, 4d to read the boot catalog */ +	__asm__ __volatile__ ( REAL_CODE ( "stc\n\t" +					   "sti\n\t" +					   "int $0x13\n\t" +					   "sti\n\t" /* BIOS bugs */ +					   "jc 1f\n\t" +					   "xorw %%ax, %%ax\n\t" +					   "\n1:\n\t" ) +			       : "=a" ( status ) +			       : "a" ( 0x4d00 ), "d" ( drive ), +				 "S" ( __from_data16 ( &eltorito_cmd ) ) ); +	if ( status ) { +		DBG ( "INT13 drive %02x could not read El Torito boot catalog " +		      "(status %02x)\n", drive, status ); +		return -EIO; +	} +	copy_from_user ( &catalog, phys_to_user ( eltorito_cmd.buffer ), 0, +			 sizeof ( catalog ) ); + +	/* Sanity checks */ +	if ( catalog.valid.platform_id != ELTORITO_PLATFORM_X86 ) { +		DBG ( "INT13 drive %02x El Torito specifies unknown platform " +		      "%02x\n", drive, catalog.valid.platform_id ); +		return -ENOEXEC; +	} +	if ( catalog.boot.indicator != ELTORITO_BOOTABLE ) { +		DBG ( "INT13 drive %02x El Torito is not bootable\n", drive ); +		return -ENOEXEC; +	} +	if ( catalog.boot.media_type != ELTORITO_NO_EMULATION ) { +		DBG ( "INT13 drive %02x El Torito requires emulation " +		       "type %02x\n", drive, catalog.boot.media_type ); +		return -ENOTSUP; +	} +	DBG ( "INT13 drive %02x El Torito boot image at LBA %08x (count %d)\n", +	      drive, catalog.boot.start, catalog.boot.length ); +	address->segment = ( catalog.boot.load_segment ? +			     catalog.boot.load_segment : 0x7c0 ); +	address->offset = 0; +	DBG ( "INT13 drive %02x El Torito boot image loads at %04x:%04x\n", +	      drive, address->segment, address->offset ); + +	/* Use INT 13, 42 to read the boot image */ +	eltorito_address.bufsize = +		offsetof ( typeof ( eltorito_address ), buffer_phys ); +	eltorito_address.count = catalog.boot.length; +	eltorito_address.buffer = *address; +	eltorito_address.lba = catalog.boot.start; +	__asm__ __volatile__ ( REAL_CODE ( "stc\n\t" +					   "sti\n\t" +					   "int $0x13\n\t" +					   "sti\n\t" /* BIOS bugs */ +					   "jc 1f\n\t" +					   "xorw %%ax, %%ax\n\t" +					   "\n1:\n\t" ) +			       : "=a" ( status ) +			       : "a" ( 0x4200 ), "d" ( drive ), +				 "S" ( __from_data16 ( &eltorito_address ) ) ); +	if ( status ) { +		DBG ( "INT13 drive %02x could not read El Torito boot image " +		      "(status %02x)\n", drive, status ); +		return -EIO; +	} + +	return 0; +} + +/** + * Attempt to boot from an INT 13 drive + * + * @v drive		Drive number + * @ret rc		Return status code + * + * This boots from the specified INT 13 drive by loading the Master + * Boot Record to 0000:7c00 and jumping to it.  INT 18 is hooked to + * capture an attempt by the MBR to boot the next device.  (This is + * the closest thing to a return path from an MBR). + * + * Note that this function can never return success, by definition. + */ +static int int13_boot ( unsigned int drive ) { +	struct memory_map memmap; +	struct segoff address; +	int rc; + +	/* Look for a usable boot sector */ +	if ( ( ( rc = int13_load_mbr ( drive, &address ) ) != 0 ) && +	     ( ( rc = int13_load_eltorito ( drive, &address ) ) != 0 ) ) +		return rc; + +	/* Dump out memory map prior to boot, if memmap debugging is +	 * enabled.  Not required for program flow, but we have so +	 * many problems that turn out to be memory-map related that +	 * it's worth doing. +	 */ +	get_memmap ( &memmap ); + +	/* Jump to boot sector */ +	if ( ( rc = call_bootsector ( address.segment, address.offset, +				      drive ) ) != 0 ) { +		DBG ( "INT13 drive %02x boot returned: %s\n", +		      drive, strerror ( rc ) ); +		return rc; +	} + +	return -ECANCELED; /* -EIMPOSSIBLE */ +} + +/** A boot firmware table generated by iPXE */ +union xbft_table { +	/** ACPI header */ +	struct acpi_description_header acpi; +	/** Padding */ +	char pad[768]; +}; + +/** The boot firmware table generated by iPXE */ +static union xbft_table __bss16 ( xbftab ) __attribute__ (( aligned ( 16 ) )); +#define xbftab __use_data16 ( xbftab ) + +/** + * Describe INT 13 emulated drive for SAN-booted operating system + * + * @v drive		Drive number + * @ret rc		Return status code + */ +static int int13_describe ( unsigned int drive ) { +	struct int13_drive *int13; +	struct segoff xbft_address; +	int rc; + +	/* Find drive */ +	int13 = int13_find ( drive ); +	if ( ! int13 ) { +		DBG ( "INT13 cannot find emulated drive %02x\n", drive ); +		return -ENODEV; +	} + +	/* Reopen block device if necessary */ +	if ( ( int13->block_rc != 0 ) && +	     ( ( rc = int13_reopen_block ( int13 ) ) != 0 ) ) +		return rc; + +	/* Clear table */ +	memset ( &xbftab, 0, sizeof ( xbftab ) ); + +	/* Fill in common parameters */ +	strncpy ( xbftab.acpi.oem_id, "FENSYS", +		  sizeof ( xbftab.acpi.oem_id ) ); +	strncpy ( xbftab.acpi.oem_table_id, "iPXE", +		  sizeof ( xbftab.acpi.oem_table_id ) ); + +	/* Fill in remaining parameters */ +	if ( ( rc = acpi_describe ( &int13->block, &xbftab.acpi, +				    sizeof ( xbftab ) ) ) != 0 ) { +		DBGC ( int13, "INT13 drive %02x could not create ACPI " +		       "description: %s\n", int13->drive, strerror ( rc ) ); +		return rc; +	} + +	/* Fix up ACPI checksum */ +	acpi_fix_checksum ( &xbftab.acpi ); +	xbft_address.segment = rm_ds; +	xbft_address.offset = __from_data16 ( &xbftab ); +	DBGC ( int13, "INT13 drive %02x described using boot firmware " +	       "table:\n", int13->drive ); +	DBGC_HDA ( int13, xbft_address, &xbftab, +		   le32_to_cpu ( xbftab.acpi.length ) ); + +	return 0; +} + +PROVIDE_SANBOOT_INLINE ( pcbios, san_default_drive ); +PROVIDE_SANBOOT ( pcbios, san_hook, int13_hook ); +PROVIDE_SANBOOT ( pcbios, san_unhook, int13_unhook ); +PROVIDE_SANBOOT ( pcbios, san_boot, int13_boot ); +PROVIDE_SANBOOT ( pcbios, san_describe, int13_describe ); diff --git a/roms/ipxe/src/arch/i386/interface/pcbios/memtop_umalloc.c b/roms/ipxe/src/arch/i386/interface/pcbios/memtop_umalloc.c new file mode 100644 index 00000000..c382e3c3 --- /dev/null +++ b/roms/ipxe/src/arch/i386/interface/pcbios/memtop_umalloc.c @@ -0,0 +1,178 @@ +/* + * Copyright (C) 2007 Michael Brown <mbrown@fensystems.co.uk>. + * + * This program is free software; you can redistribute it and/or + * modify it under the terms of the GNU General Public License as + * published by the Free Software Foundation; either version 2 of the + * License, or any later version. + * + * 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. + * + * You should have received a copy of the GNU General Public License + * along with this program; if not, write to the Free Software + * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA + * 02110-1301, USA. + */ + +FILE_LICENCE ( GPL2_OR_LATER ); + +/** + * @file + * + * External memory allocation + * + */ + +#include <limits.h> +#include <errno.h> +#include <ipxe/uaccess.h> +#include <ipxe/hidemem.h> +#include <ipxe/io.h> +#include <ipxe/memblock.h> +#include <ipxe/umalloc.h> + +/** Alignment of external allocated memory */ +#define EM_ALIGN ( 4 * 1024 ) + +/** Equivalent of NOWHERE for user pointers */ +#define UNOWHERE ( ~UNULL ) + +/** An external memory block */ +struct external_memory { +	/** Size of this memory block (excluding this header) */ +	size_t size; +	/** Block is currently in use */ +	int used; +}; + +/** Top of heap */ +static userptr_t top = UNULL; + +/** Bottom of heap (current lowest allocated block) */ +static userptr_t bottom = UNULL; + +/** Remaining space on heap */ +static size_t heap_size; + +/** + * Initialise external heap + * + */ +static void init_eheap ( void ) { +	userptr_t base; + +	heap_size = largest_memblock ( &base ); +	bottom = top = userptr_add ( base, heap_size ); +	DBG ( "External heap grows downwards from %lx (size %zx)\n", +	      user_to_phys ( top, 0 ), heap_size ); +} + +/** + * Collect free blocks + * + */ +static void ecollect_free ( void ) { +	struct external_memory extmem; +	size_t len; + +	/* Walk the free list and collect empty blocks */ +	while ( bottom != top ) { +		copy_from_user ( &extmem, bottom, -sizeof ( extmem ), +				 sizeof ( extmem ) ); +		if ( extmem.used ) +			break; +		DBG ( "EXTMEM freeing [%lx,%lx)\n", user_to_phys ( bottom, 0 ), +		      user_to_phys ( bottom, extmem.size ) ); +		len = ( extmem.size + sizeof ( extmem ) ); +		bottom = userptr_add ( bottom, len ); +		heap_size += len; +	} +} + +/** + * Reallocate external memory + * + * @v old_ptr		Memory previously allocated by umalloc(), or UNULL + * @v new_size		Requested size + * @ret new_ptr		Allocated memory, or UNULL + * + * Calling realloc() with a new size of zero is a valid way to free a + * memory block. + */ +static userptr_t memtop_urealloc ( userptr_t ptr, size_t new_size ) { +	struct external_memory extmem; +	userptr_t new = ptr; +	size_t align; + +	/* (Re)initialise external memory allocator if necessary */ +	if ( bottom == top ) +		init_eheap(); + +	/* Get block properties into extmem */ +	if ( ptr && ( ptr != UNOWHERE ) ) { +		/* Determine old size */ +		copy_from_user ( &extmem, ptr, -sizeof ( extmem ), +				 sizeof ( extmem ) ); +	} else { +		/* Create a zero-length block */ +		if ( heap_size < sizeof ( extmem ) ) { +			DBG ( "EXTMEM out of space\n" ); +			return UNULL; +		} +		ptr = bottom = userptr_add ( bottom, -sizeof ( extmem ) ); +		heap_size -= sizeof ( extmem ); +		DBG ( "EXTMEM allocating [%lx,%lx)\n", +		      user_to_phys ( ptr, 0 ), user_to_phys ( ptr, 0 ) ); +		extmem.size = 0; +	} +	extmem.used = ( new_size > 0 ); + +	/* Expand/shrink block if possible */ +	if ( ptr == bottom ) { +		/* Update block */ +		if ( new_size > ( heap_size - extmem.size ) ) { +			DBG ( "EXTMEM out of space\n" ); +			return UNULL; +		} +		new = userptr_add ( ptr, - ( new_size - extmem.size ) ); +		align = ( user_to_phys ( new, 0 ) & ( EM_ALIGN - 1 ) ); +		new_size += align; +		new = userptr_add ( new, -align ); +		DBG ( "EXTMEM expanding [%lx,%lx) to [%lx,%lx)\n", +		      user_to_phys ( ptr, 0 ), +		      user_to_phys ( ptr, extmem.size ), +		      user_to_phys ( new, 0 ), +		      user_to_phys ( new, new_size )); +		memmove_user ( new, 0, ptr, 0, ( ( extmem.size < new_size ) ? +						 extmem.size : new_size ) ); +		bottom = new; +		heap_size -= ( new_size - extmem.size ); +		extmem.size = new_size; +	} else { +		/* Cannot expand; can only pretend to shrink */ +		if ( new_size > extmem.size ) { +			/* Refuse to expand */ +			DBG ( "EXTMEM cannot expand [%lx,%lx)\n", +			      user_to_phys ( ptr, 0 ), +			      user_to_phys ( ptr, extmem.size ) ); +			return UNULL; +		} +	} + +	/* Write back block properties */ +	copy_to_user ( new, -sizeof ( extmem ), &extmem, +		       sizeof ( extmem ) ); + +	/* Collect any free blocks and update hidden memory region */ +	ecollect_free(); +	hide_umalloc ( user_to_phys ( bottom, ( ( bottom == top ) ? +						0 : -sizeof ( extmem ) ) ), +		       user_to_phys ( top, 0 ) ); + +	return ( new_size ? new : UNOWHERE ); +} + +PROVIDE_UMALLOC ( memtop, urealloc, memtop_urealloc ); diff --git a/roms/ipxe/src/arch/i386/interface/pcbios/pcibios.c b/roms/ipxe/src/arch/i386/interface/pcbios/pcibios.c new file mode 100644 index 00000000..61873039 --- /dev/null +++ b/roms/ipxe/src/arch/i386/interface/pcbios/pcibios.c @@ -0,0 +1,115 @@ +/* + * Copyright (C) 2006 Michael Brown <mbrown@fensystems.co.uk>. + * + * This program is free software; you can redistribute it and/or + * modify it under the terms of the GNU General Public License as + * published by the Free Software Foundation; either version 2 of the + * License, or any later version. + * + * 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. + * + * You should have received a copy of the GNU General Public License + * along with this program; if not, write to the Free Software + * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA + * 02110-1301, USA. + */ + +FILE_LICENCE ( GPL2_OR_LATER ); + +#include <stdint.h> +#include <ipxe/pci.h> +#include <realmode.h> + +/** @file + * + * PCI configuration space access via PCI BIOS + * + */ + +/** + * Determine number of PCI buses within system + * + * @ret num_bus		Number of buses + */ +static int pcibios_num_bus ( void ) { +	int discard_a, discard_D; +	uint8_t max_bus; + +	__asm__ __volatile__ ( REAL_CODE ( "stc\n\t" +					   "int $0x1a\n\t" +					   "jnc 1f\n\t" +					   "xorw %%cx, %%cx\n\t" +					   "\n1:\n\t" ) +			       : "=c" ( max_bus ), "=a" ( discard_a ), +				 "=D" ( discard_D ) +			       : "a" ( PCIBIOS_INSTALLATION_CHECK >> 16 ), +				 "D" ( 0 ) +			       : "ebx", "edx" ); + +	return ( max_bus + 1 ); +} + +/** + * Read configuration space via PCI BIOS + * + * @v pci	PCI device + * @v command	PCI BIOS command + * @v value	Value read + * @ret rc	Return status code + */ +int pcibios_read ( struct pci_device *pci, uint32_t command, uint32_t *value ){ +	int discard_b, discard_D; +	int status; + +	__asm__ __volatile__ ( REAL_CODE ( "stc\n\t" +					   "int $0x1a\n\t" +					   "jnc 1f\n\t" +					   "xorl %%eax, %%eax\n\t" +					   "decl %%eax\n\t" +					   "movl %%eax, %%ecx\n\t" +					   "\n1:\n\t" ) +			       : "=a" ( status ), "=b" ( discard_b ), +				 "=c" ( *value ), "=D" ( discard_D ) +			       : "a" ( command >> 16 ), "D" ( command ), +				 "b" ( pci->busdevfn ) +			       : "edx" ); + +	return ( ( status >> 8 ) & 0xff ); +} + +/** + * Write configuration space via PCI BIOS + * + * @v pci	PCI device + * @v command	PCI BIOS command + * @v value	Value to be written + * @ret rc	Return status code + */ +int pcibios_write ( struct pci_device *pci, uint32_t command, uint32_t value ){ +	int discard_b, discard_c, discard_D; +	int status; + +	__asm__ __volatile__ ( REAL_CODE ( "stc\n\t" +					   "int $0x1a\n\t" +					   "jnc 1f\n\t" +					   "movb $0xff, %%ah\n\t" +					   "\n1:\n\t" ) +			       : "=a" ( status ), "=b" ( discard_b ), +				 "=c" ( discard_c ), "=D" ( discard_D ) +			       : "a" ( command >> 16 ),	"D" ( command ), +			         "b" ( pci->busdevfn ), "c" ( value ) +			       : "edx" ); +	 +	return ( ( status >> 8 ) & 0xff ); +} + +PROVIDE_PCIAPI ( pcbios, pci_num_bus, pcibios_num_bus ); +PROVIDE_PCIAPI_INLINE ( pcbios, pci_read_config_byte ); +PROVIDE_PCIAPI_INLINE ( pcbios, pci_read_config_word ); +PROVIDE_PCIAPI_INLINE ( pcbios, pci_read_config_dword ); +PROVIDE_PCIAPI_INLINE ( pcbios, pci_write_config_byte ); +PROVIDE_PCIAPI_INLINE ( pcbios, pci_write_config_word ); +PROVIDE_PCIAPI_INLINE ( pcbios, pci_write_config_dword ); diff --git a/roms/ipxe/src/arch/i386/interface/pcbios/rtc_entropy.c b/roms/ipxe/src/arch/i386/interface/pcbios/rtc_entropy.c new file mode 100644 index 00000000..fad421c2 --- /dev/null +++ b/roms/ipxe/src/arch/i386/interface/pcbios/rtc_entropy.c @@ -0,0 +1,199 @@ +/* + * Copyright (C) 2012 Michael Brown <mbrown@fensystems.co.uk>. + * + * This program is free software; you can redistribute it and/or + * modify it under the terms of the GNU General Public License as + * published by the Free Software Foundation; either version 2 of the + * License, or any later version. + * + * 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. + * + * You should have received a copy of the GNU General Public License + * along with this program; if not, write to the Free Software + * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA + * 02110-1301, USA. + */ + +FILE_LICENCE ( GPL2_OR_LATER ); + +/** @file + * + * RTC-based entropy source + * + */ + +#include <stdint.h> +#include <string.h> +#include <biosint.h> +#include <pic8259.h> +#include <rtc.h> +#include <ipxe/entropy.h> + +/** RTC "interrupt triggered" flag */ +static uint8_t __text16 ( rtc_flag ); +#define rtc_flag __use_text16 ( rtc_flag ) + +/** RTC interrupt handler */ +extern void rtc_isr ( void ); + +/** Previous RTC interrupt handler */ +static struct segoff rtc_old_handler; + +/** + * Hook RTC interrupt handler + * + */ +static void rtc_hook_isr ( void ) { + +	/* RTC interrupt handler */ +	__asm__ __volatile__ ( +		TEXT16_CODE ( "\nrtc_isr:\n\t" +			      /* Preserve registers */ +			      "pushw %%ax\n\t" +			      /* Set "interrupt triggered" flag */ +			      "cs movb $0x01, %c0\n\t" +			      /* Read RTC status register C to +			       * acknowledge interrupt +			       */ +			      "movb %3, %%al\n\t" +			      "outb %%al, %1\n\t" +			      "inb %2\n\t" +			      /* Send EOI */ +			      "movb $0x20, %%al\n\t" +			      "outb %%al, $0xa0\n\t" +			      "outb %%al, $0x20\n\t" +			      /* Restore registers and return */ +			      "popw %%ax\n\t" +			      "iret\n\t" ) +		: +		: "p" ( __from_text16 ( &rtc_flag ) ), +		  "i" ( CMOS_ADDRESS ), "i" ( CMOS_DATA ), +		  "i" ( RTC_STATUS_C ) ); + +	hook_bios_interrupt ( RTC_INT, ( unsigned int ) rtc_isr, +			      &rtc_old_handler ); +} + +/** + * Unhook RTC interrupt handler + * + */ +static void rtc_unhook_isr ( void ) { +	int rc; + +	rc = unhook_bios_interrupt ( RTC_INT, ( unsigned int ) rtc_isr, +				     &rtc_old_handler ); +	assert ( rc == 0 ); /* Should always be able to unhook */ +} + +/** + * Enable RTC interrupts + * + */ +static void rtc_enable_int ( void ) { +	uint8_t status_b; + +	/* Set Periodic Interrupt Enable bit in status register B */ +	outb ( ( RTC_STATUS_B | CMOS_DISABLE_NMI ), CMOS_ADDRESS ); +	status_b = inb ( CMOS_DATA ); +	outb ( ( RTC_STATUS_B | CMOS_DISABLE_NMI ), CMOS_ADDRESS ); +	outb ( ( status_b | RTC_STATUS_B_PIE ), CMOS_DATA ); + +	/* Re-enable NMI and reset to default address */ +	outb ( CMOS_DEFAULT_ADDRESS, CMOS_ADDRESS ); +	inb ( CMOS_DATA ); /* Discard; may be needed on some platforms */ +} + +/** + * Disable RTC interrupts + * + */ +static void rtc_disable_int ( void ) { +	uint8_t status_b; + +	/* Clear Periodic Interrupt Enable bit in status register B */ +	outb ( ( RTC_STATUS_B | CMOS_DISABLE_NMI ), CMOS_ADDRESS ); +	status_b = inb ( CMOS_DATA ); +	outb ( ( RTC_STATUS_B | CMOS_DISABLE_NMI ), CMOS_ADDRESS ); +	outb ( ( status_b & ~RTC_STATUS_B_PIE ), CMOS_DATA ); + +	/* Re-enable NMI and reset to default address */ +	outb ( CMOS_DEFAULT_ADDRESS, CMOS_ADDRESS ); +	inb ( CMOS_DATA ); /* Discard; may be needed on some platforms */ +} + +/** + * Enable entropy gathering + * + * @ret rc		Return status code + */ +static int rtc_entropy_enable ( void ) { + +	rtc_hook_isr(); +	enable_irq ( RTC_IRQ ); +	rtc_enable_int(); + +	return 0; +} + +/** + * Disable entropy gathering + * + */ +static void rtc_entropy_disable ( void ) { + +	rtc_disable_int(); +	disable_irq ( RTC_IRQ ); +	rtc_unhook_isr(); +} + +/** + * Measure a single RTC tick + * + * @ret delta		Length of RTC tick (in TSC units) + */ +uint8_t rtc_sample ( void ) { +	uint32_t before; +	uint32_t after; +	uint32_t temp; + +	__asm__ __volatile__ ( +		REAL_CODE ( /* Enable interrupts */ +			    "sti\n\t" +			    /* Wait for RTC interrupt */ +			    "cs movb %b2, %c4\n\t" +			    "\n1:\n\t" +			    "cs xchgb %b2, %c4\n\t" /* Serialize */ +			    "testb %b2, %b2\n\t" +			    "jz 1b\n\t" +			    /* Read "before" TSC */ +			    "rdtsc\n\t" +			    /* Store "before" TSC on stack */ +			    "pushl %0\n\t" +			    /* Wait for another RTC interrupt */ +			    "xorb %b2, %b2\n\t" +			    "cs movb %b2, %c4\n\t" +			    "\n1:\n\t" +			    "cs xchgb %b2, %c4\n\t" /* Serialize */ +			    "testb %b2, %b2\n\t" +			    "jz 1b\n\t" +			    /* Read "after" TSC */ +			    "rdtsc\n\t" +			    /* Retrieve "before" TSC on stack */ +			    "popl %1\n\t" +			    /* Disable interrupts */ +			    "cli\n\t" +			    ) +		: "=a" ( after ), "=d" ( before ), "=q" ( temp ) +		: "2" ( 0 ), "p" ( __from_text16 ( &rtc_flag ) ) ); + +	return ( after - before ); +} + +PROVIDE_ENTROPY_INLINE ( rtc, min_entropy_per_sample ); +PROVIDE_ENTROPY ( rtc, entropy_enable, rtc_entropy_enable ); +PROVIDE_ENTROPY ( rtc, entropy_disable, rtc_entropy_disable ); +PROVIDE_ENTROPY_INLINE ( rtc, get_noise ); diff --git a/roms/ipxe/src/arch/i386/interface/pcbios/rtc_time.c b/roms/ipxe/src/arch/i386/interface/pcbios/rtc_time.c new file mode 100644 index 00000000..67041d4c --- /dev/null +++ b/roms/ipxe/src/arch/i386/interface/pcbios/rtc_time.c @@ -0,0 +1,138 @@ +/* + * Copyright (C) 2012 Michael Brown <mbrown@fensystems.co.uk>. + * + * This program is free software; you can redistribute it and/or + * modify it under the terms of the GNU General Public License as + * published by the Free Software Foundation; either version 2 of the + * License, or any later version. + * + * 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. + * + * You should have received a copy of the GNU General Public License + * along with this program; if not, write to the Free Software + * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA + * 02110-1301, USA. + */ + +FILE_LICENCE ( GPL2_OR_LATER ); + +/** @file + * + * RTC-based time source + * + */ + +#include <stdint.h> +#include <time.h> +#include <rtc.h> +#include <ipxe/time.h> + +/** + * Read RTC register + * + * @v address		Register address + * @ret data		Data + */ +static unsigned int rtc_readb ( int address ) { +	outb ( address, CMOS_ADDRESS ); +	return inb ( CMOS_DATA ); +} + +/** + * Check if RTC update is in progress + * + * @ret is_busy		RTC update is in progress + */ +static int rtc_is_busy ( void ) { +	return ( rtc_readb ( RTC_STATUS_A ) & RTC_STATUS_A_UPDATE_IN_PROGRESS ); +} + +/** + * Read RTC BCD register + * + * @v address		Register address + * @ret value		Value + */ +static unsigned int rtc_readb_bcd ( int address ) { +	unsigned int bcd; + +	bcd = rtc_readb ( address ); +	return ( bcd - ( 6 * ( bcd >> 4 ) ) ); +} + +/** + * Read RTC time + * + * @ret time		Time, in seconds + */ +static time_t rtc_read_time ( void ) { +	unsigned int status_b; +	int is_binary; +	int is_24hour; +	unsigned int ( * read_component ) ( int address ); +	struct tm tm; +	int is_pm; +	unsigned int hour; +	time_t time; + +	/* Wait for any in-progress update to complete */ +	while ( rtc_is_busy() ) {} + +	/* Determine RTC mode */ +	status_b = rtc_readb ( RTC_STATUS_B ); +	is_binary = ( status_b & RTC_STATUS_B_BINARY ); +	is_24hour = ( status_b & RTC_STATUS_B_24_HOUR ); +	read_component = ( is_binary ? rtc_readb : rtc_readb_bcd ); + +	/* Read time values */ +	tm.tm_sec = read_component ( RTC_SEC ); +	tm.tm_min = read_component ( RTC_MIN ); +	hour = read_component ( RTC_HOUR ); +	if ( ! is_24hour ) { +		is_pm = ( hour >= 80 ); +		hour = ( ( ( ( hour & 0x7f ) % 80 ) % 12 ) + +			 ( is_pm ? 12 : 0 ) ); +	} +	tm.tm_hour = hour; +	tm.tm_mday = read_component ( RTC_MDAY ); +	tm.tm_mon = ( read_component ( RTC_MON ) - 1 ); +	tm.tm_year = ( read_component ( RTC_YEAR ) + +		       100 /* Assume we are in the 21st century, since +			    * this code was written in 2012 */ ); + +	DBGC ( RTC_STATUS_A, "RTCTIME is %04d-%02d-%02d %02d:%02d:%02d " +	       "(%s,%d-hour)\n", ( tm.tm_year + 1900 ), ( tm.tm_mon + 1 ), +	       tm.tm_mday, tm.tm_hour, tm.tm_min, tm.tm_sec, +	       ( is_binary ? "binary" : "BCD" ), ( is_24hour ? 24 : 12 ) ); + +	/* Convert to seconds since the Epoch */ +	time = mktime ( &tm ); + +	return time; +} + +/** + * Get current time in seconds + * + * @ret time		Time, in seconds + */ +static time_t rtc_now ( void ) { +	time_t time = 0; +	time_t last_time; + +	/* Read time until we get two matching values in a row, in +	 * case we end up reading a corrupted value in the middle of +	 * an update. +	 */ +	do { +		last_time = time; +		time = rtc_read_time(); +	} while ( time != last_time ); + +	return time; +} + +PROVIDE_TIME ( rtc, time_now, rtc_now ); diff --git a/roms/ipxe/src/arch/i386/interface/pcbios/vesafb.c b/roms/ipxe/src/arch/i386/interface/pcbios/vesafb.c new file mode 100644 index 00000000..2adc7b04 --- /dev/null +++ b/roms/ipxe/src/arch/i386/interface/pcbios/vesafb.c @@ -0,0 +1,538 @@ +/* + * Copyright (C) 2013 Michael Brown <mbrown@fensystems.co.uk>. + * + * This program is free software; you can redistribute it and/or + * modify it under the terms of the GNU General Public License as + * published by the Free Software Foundation; either version 2 of the + * License, or any later version. + * + * 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. + * + * You should have received a copy of the GNU General Public License + * along with this program; if not, write to the Free Software + * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA + * 02110-1301, USA. + */ + +FILE_LICENCE ( GPL2_OR_LATER ); + +/** @file + * + * VESA frame buffer console + * + */ + +#include <stdlib.h> +#include <errno.h> +#include <limits.h> +#include <realmode.h> +#include <ipxe/console.h> +#include <ipxe/io.h> +#include <ipxe/ansicol.h> +#include <ipxe/fbcon.h> +#include <ipxe/vesafb.h> +#include <config/console.h> + +/* Avoid dragging in BIOS console if not otherwise used */ +extern struct console_driver bios_console; +struct console_driver bios_console __attribute__ (( weak )); + +/* Disambiguate the various error causes */ +#define EIO_FAILED __einfo_error ( EINFO_EIO_FAILED ) +#define EINFO_EIO_FAILED						\ +	__einfo_uniqify ( EINFO_EIO, 0x01,				\ +			  "Function call failed" ) +#define EIO_HARDWARE __einfo_error ( EINFO_EIO_HARDWARE ) +#define EINFO_EIO_HARDWARE						\ +	__einfo_uniqify ( EINFO_EIO, 0x02,				\ +			  "Not supported in current configuration" ) +#define EIO_MODE __einfo_error ( EINFO_EIO_MODE ) +#define EINFO_EIO_MODE							\ +	__einfo_uniqify ( EINFO_EIO, 0x03,				\ +			  "Invalid in current video mode" ) +#define EIO_VBE( code )							\ +	EUNIQ ( EINFO_EIO, (code), EIO_FAILED, EIO_HARDWARE, EIO_MODE ) + +/* Set default console usage if applicable */ +#if ! ( defined ( CONSOLE_VESAFB ) && CONSOLE_EXPLICIT ( CONSOLE_VESAFB ) ) +#undef CONSOLE_VESAFB +#define CONSOLE_VESAFB ( CONSOLE_USAGE_ALL & ~CONSOLE_USAGE_LOG ) +#endif + +/** Font corresponding to selected character width and height */ +#define VESAFB_FONT VBE_FONT_8x16 + +/* Forward declaration */ +struct console_driver vesafb_console __console_driver; + +/** A VESA frame buffer */ +struct vesafb { +	/** Frame buffer console */ +	struct fbcon fbcon; +	/** Physical start address */ +	physaddr_t start; +	/** Pixel geometry */ +	struct fbcon_geometry pixel; +	/** Margin */ +	struct fbcon_margin margin; +	/** Colour mapping */ +	struct fbcon_colour_map map; +	/** Font definition */ +	struct fbcon_font font; +	/** Saved VGA mode */ +	uint8_t saved_mode; +}; + +/** The VESA frame buffer */ +static struct vesafb vesafb; + +/** Base memory buffer used for VBE calls */ +union vbe_buffer { +	/** VBE controller information block */ +	struct vbe_controller_info controller; +	/** VBE mode information block */ +	struct vbe_mode_info mode; +}; +static union vbe_buffer __bss16 ( vbe_buf ); +#define vbe_buf __use_data16 ( vbe_buf ) + +/** + * Convert VBE status code to iPXE status code + * + * @v status		VBE status code + * @ret rc		Return status code + */ +static int vesafb_rc ( unsigned int status ) { +	unsigned int code; + +	if ( ( status & 0xff ) != 0x4f ) +		return -ENOTSUP; +	code = ( ( status >> 8 ) & 0xff ); +	return ( code ? -EIO_VBE ( code ) : 0 ); +} + +/** + * Get font definition + * + */ +static void vesafb_font ( void ) { +	struct segoff font; + +	/* Get font information +	 * +	 * Working around gcc bugs is icky here.  The value we want is +	 * returned in %ebp, but there's no way to specify %ebp in an +	 * output constraint.  We can't put %ebp in the clobber list, +	 * because this tends to cause random build failures on some +	 * gcc versions.  We can't manually push/pop %ebp and return +	 * the value via a generic register output constraint, because +	 * gcc might choose to use %ebp to satisfy that constraint +	 * (and we have no way to prevent it from so doing). +	 * +	 * Work around this hideous mess by using %ecx and %edx as the +	 * output registers, since they get clobbered anyway. +	 */ +	__asm__ __volatile__ ( REAL_CODE ( "pushw %%bp\n\t" /* gcc bug */ +					   "int $0x10\n\t" +					   "movw %%es, %%cx\n\t" +					   "movw %%bp, %%dx\n\t" +					   "popw %%bp\n\t" /* gcc bug */ ) +			       : "=c" ( font.segment ), +				 "=d" ( font.offset ) +			       : "a" ( VBE_GET_FONT ), +				 "b" ( VESAFB_FONT ) ); +	DBGC ( &vbe_buf, "VESAFB has font %04x at %04x:%04x\n", +	       VESAFB_FONT, font.segment, font.offset ); +	vesafb.font.start = real_to_user ( font.segment, font.offset ); +} + +/** + * Get VBE mode list + * + * @ret mode_numbers	Mode number list (terminated with VBE_MODE_END) + * @ret rc		Return status code + * + * The caller is responsible for eventually freeing the mode list. + */ +static int vesafb_mode_list ( uint16_t **mode_numbers ) { +	struct vbe_controller_info *controller = &vbe_buf.controller; +	userptr_t video_mode_ptr; +	uint16_t mode_number; +	uint16_t status; +	size_t len; +	int rc; + +	/* Avoid returning uninitialised data on error */ +	*mode_numbers = NULL; + +	/* Get controller information block */ +	controller->vbe_signature = 0; +	__asm__ __volatile__ ( REAL_CODE ( "int $0x10" ) +			       : "=a" ( status ) +			       : "a" ( VBE_CONTROLLER_INFO ), +				 "D" ( __from_data16 ( controller ) ) +			       : "memory", "ebx", "edx" ); +	if ( ( rc = vesafb_rc ( status ) ) != 0 ) { +		DBGC ( &vbe_buf, "VESAFB could not get controller information: " +		       "[%04x] %s\n", status, strerror ( rc ) ); +		return rc; +	} +	if ( controller->vbe_signature != VBE_CONTROLLER_SIGNATURE ) { +		DBGC ( &vbe_buf, "VESAFB invalid controller signature " +		       "\"%c%c%c%c\"\n", ( controller->vbe_signature >> 0 ), +		       ( controller->vbe_signature >> 8 ), +		       ( controller->vbe_signature >> 16 ), +		       ( controller->vbe_signature >> 24 ) ); +		DBGC_HDA ( &vbe_buf, 0, controller, sizeof ( *controller ) ); +		return -EINVAL; +	} +	DBGC ( &vbe_buf, "VESAFB found VBE version %d.%d with mode list at " +	       "%04x:%04x\n", controller->vbe_major_version, +	       controller->vbe_minor_version, +	       controller->video_mode_ptr.segment, +	       controller->video_mode_ptr.offset ); + +	/* Calculate length of mode list */ +	video_mode_ptr = real_to_user ( controller->video_mode_ptr.segment, +					controller->video_mode_ptr.offset ); +	len = 0; +	do { +		copy_from_user ( &mode_number, video_mode_ptr, len, +				 sizeof ( mode_number ) ); +		len += sizeof ( mode_number ); +	} while ( mode_number != VBE_MODE_END ); + +	/* Allocate and fill mode list */ +	*mode_numbers = malloc ( len ); +	if ( ! *mode_numbers ) +		return -ENOMEM; +	copy_from_user ( *mode_numbers, video_mode_ptr, 0, len ); + +	return 0; +} + +/** + * Get video mode information + * + * @v mode_number	Mode number + * @ret rc		Return status code + */ +static int vesafb_mode_info ( unsigned int mode_number ) { +	struct vbe_mode_info *mode = &vbe_buf.mode; +	uint16_t status; +	int rc; + +	/* Get mode information */ +	__asm__ __volatile__ ( REAL_CODE ( "int $0x10" ) +			       : "=a" ( status ) +			       : "a" ( VBE_MODE_INFO ), +				 "c" ( mode_number ), +				 "D" ( __from_data16 ( mode ) ) +			       : "memory" ); +	if ( ( rc = vesafb_rc ( status ) ) != 0 ) { +		DBGC ( &vbe_buf, "VESAFB could not get mode %04x information: " +		       "[%04x] %s\n", mode_number, status, strerror ( rc ) ); +		return rc; +	} +	DBGC ( &vbe_buf, "VESAFB mode %04x %dx%d %dbpp(%d:%d:%d:%d) model " +	       "%02x [x%d]%s%s%s%s%s\n", mode_number, mode->x_resolution, +	       mode->y_resolution, mode->bits_per_pixel, mode->rsvd_mask_size, +	       mode->red_mask_size, mode->green_mask_size, mode->blue_mask_size, +	       mode->memory_model, ( mode->number_of_image_pages + 1 ), +	       ( ( mode->mode_attributes & VBE_MODE_ATTR_SUPPORTED ) ? +		 "" : " [unsupported]" ), +	       ( ( mode->mode_attributes & VBE_MODE_ATTR_TTY ) ? +		 " [tty]" : "" ), +	       ( ( mode->mode_attributes & VBE_MODE_ATTR_GRAPHICS ) ? +		 "" : " [text]" ), +	       ( ( mode->mode_attributes & VBE_MODE_ATTR_LINEAR ) ? +		 "" : " [nonlinear]" ), +	       ( ( mode->mode_attributes & VBE_MODE_ATTR_TRIPLE_BUF ) ? +		 " [buf]" : "" ) ); + +	return 0; +} + +/** + * Set video mode + * + * @v mode_number	Mode number + * @ret rc		Return status code + */ +static int vesafb_set_mode ( unsigned int mode_number ) { +	struct vbe_mode_info *mode = &vbe_buf.mode; +	uint16_t status; +	int rc; + +	/* Get mode information */ +	if ( ( rc = vesafb_mode_info ( mode_number ) ) != 0 ) +		return rc; + +	/* Record mode parameters */ +	vesafb.start = mode->phys_base_ptr; +	vesafb.pixel.width = mode->x_resolution; +	vesafb.pixel.height = mode->y_resolution; +	vesafb.pixel.len = ( ( mode->bits_per_pixel + 7 ) / 8 ); +	vesafb.pixel.stride = mode->bytes_per_scan_line; +	DBGC ( &vbe_buf, "VESAFB mode %04x has frame buffer at %08x\n", +	       mode_number, mode->phys_base_ptr ); + +	/* Initialise font colours */ +	vesafb.map.red_scale = ( 8 - mode->red_mask_size ); +	vesafb.map.green_scale = ( 8 - mode->green_mask_size ); +	vesafb.map.blue_scale = ( 8 - mode->blue_mask_size ); +	vesafb.map.red_lsb = mode->red_field_position; +	vesafb.map.green_lsb = mode->green_field_position; +	vesafb.map.blue_lsb = mode->blue_field_position; + +	/* Select this mode */ +	__asm__ __volatile__ ( REAL_CODE ( "int $0x10" ) +			       : "=a" ( status ) +			       : "a" ( VBE_SET_MODE ), +				 "b" ( mode_number ) ); +	if ( ( rc = vesafb_rc ( status ) ) != 0 ) { +		DBGC ( &vbe_buf, "VESAFB could not set mode %04x: [%04x] %s\n", +		       mode_number, status, strerror ( rc ) ); +		return rc; +	} + +	return 0; +} + +/** + * Select video mode + * + * @v mode_numbers	Mode number list (terminated with VBE_MODE_END) + * @v min_width		Minimum required width (in pixels) + * @v min_height	Minimum required height (in pixels) + * @v min_bpp		Minimum required colour depth (in bits per pixel) + * @ret mode_number	Mode number, or negative error + */ +static int vesafb_select_mode ( const uint16_t *mode_numbers, +				unsigned int min_width, unsigned int min_height, +				unsigned int min_bpp ) { +	struct vbe_mode_info *mode = &vbe_buf.mode; +	int best_mode_number = -ENOENT; +	unsigned int best_score = INT_MAX; +	unsigned int score; +	uint16_t mode_number; +	int rc; + +	/* Find the first suitable mode */ +	while ( ( mode_number = *(mode_numbers++) ) != VBE_MODE_END ) { + +		/* Force linear mode variant */ +		mode_number |= VBE_MODE_LINEAR; + +		/* Get mode information */ +		if ( ( rc = vesafb_mode_info ( mode_number ) ) != 0 ) +			continue; + +		/* Skip unusable modes */ +		if ( ( mode->mode_attributes & ( VBE_MODE_ATTR_SUPPORTED | +						 VBE_MODE_ATTR_GRAPHICS | +						 VBE_MODE_ATTR_LINEAR ) ) != +		     ( VBE_MODE_ATTR_SUPPORTED | VBE_MODE_ATTR_GRAPHICS | +		       VBE_MODE_ATTR_LINEAR ) ) { +			continue; +		} +		if ( mode->memory_model != VBE_MODE_MODEL_DIRECT_COLOUR ) +			continue; + +		/* Skip modes not meeting the requirements */ +		if ( ( mode->x_resolution < min_width ) || +		     ( mode->y_resolution < min_height ) || +		     ( mode->bits_per_pixel < min_bpp ) ) { +			continue; +		} + +		/* Select this mode if it has the best (i.e. lowest) +		 * score.  We choose the scoring system to favour +		 * modes close to the specified width and height; +		 * within modes of the same width and height we prefer +		 * a higher colour depth. +		 */ +		score = ( ( mode->x_resolution * mode->y_resolution ) - +			  mode->bits_per_pixel ); +		if ( score < best_score ) { +			best_mode_number = mode_number; +			best_score = score; +		} +	} + +	if ( best_mode_number >= 0 ) { +		DBGC ( &vbe_buf, "VESAFB selected mode %04x\n", +		       best_mode_number ); +	} else { +		DBGC ( &vbe_buf, "VESAFB found no suitable mode\n" ); +	} + +	return best_mode_number; +} + +/** + * Restore video mode + * + */ +static void vesafb_restore ( void ) { +	uint32_t discard_a; + +	/* Restore saved VGA mode */ +	__asm__ __volatile__ ( REAL_CODE ( "int $0x10" ) +			       : "=a" ( discard_a ) +			       : "a" ( VBE_SET_VGA_MODE | vesafb.saved_mode ) ); +	DBGC ( &vbe_buf, "VESAFB restored VGA mode %#02x\n", +	       vesafb.saved_mode ); +} + +/** + * Initialise VESA frame buffer + * + * @v config		Console configuration, or NULL to reset + * @ret rc		Return status code + */ +static int vesafb_init ( struct console_configuration *config ) { +	uint32_t discard_b; +	uint16_t *mode_numbers; +	unsigned int xgap; +	unsigned int ygap; +	unsigned int left; +	unsigned int right; +	unsigned int top; +	unsigned int bottom; +	int mode_number; +	int rc; + +	/* Record current VGA mode */ +	__asm__ __volatile__ ( REAL_CODE ( "int $0x10" ) +			       : "=a" ( vesafb.saved_mode ), "=b" ( discard_b ) +			       : "a" ( VBE_GET_VGA_MODE ) ); +	DBGC ( &vbe_buf, "VESAFB saved VGA mode %#02x\n", vesafb.saved_mode ); + +	/* Get VESA mode list */ +	if ( ( rc = vesafb_mode_list ( &mode_numbers ) ) != 0 ) +		goto err_mode_list; + +	/* Select mode */ +	if ( ( mode_number = vesafb_select_mode ( mode_numbers, config->width, +						  config->height, +						  config->depth ) ) < 0 ) { +		rc = mode_number; +		goto err_select_mode; +	} + +	/* Set mode */ +	if ( ( rc = vesafb_set_mode ( mode_number ) ) != 0 ) +		goto err_set_mode; + +	/* Calculate margin.  If the actual screen size is larger than +	 * the requested screen size, then update the margins so that +	 * the margin remains relative to the requested screen size. +	 * (As an exception, if a zero margin was specified then treat +	 * this as meaning "expand to edge of actual screen".) +	 */ +	xgap = ( vesafb.pixel.width - config->width ); +	ygap = ( vesafb.pixel.height - config->height ); +	left = ( xgap / 2 ); +	right = ( xgap - left ); +	top = ( ygap / 2 ); +	bottom = ( ygap - top ); +	vesafb.margin.left = ( config->left + ( config->left ? left : 0 ) ); +	vesafb.margin.right = ( config->right + ( config->right ? right : 0 ) ); +	vesafb.margin.top = ( config->top + ( config->top ? top : 0 ) ); +	vesafb.margin.bottom = +		( config->bottom + ( config->bottom ? bottom : 0 ) ); + +	/* Get font data */ +	vesafb_font(); + +	/* Initialise frame buffer console */ +	if ( ( rc = fbcon_init ( &vesafb.fbcon, phys_to_user ( vesafb.start ), +				 &vesafb.pixel, &vesafb.margin, &vesafb.map, +				 &vesafb.font, config->pixbuf ) ) != 0 ) +		goto err_fbcon_init; + +	free ( mode_numbers ); +	return 0; + +	fbcon_fini ( &vesafb.fbcon ); + err_fbcon_init: + err_set_mode: +	vesafb_restore(); + err_select_mode: +	free ( mode_numbers ); + err_mode_list: +	return rc; +} + +/** + * Finalise VESA frame buffer + * + */ +static void vesafb_fini ( void ) { + +	/* Finalise frame buffer console */ +	fbcon_fini ( &vesafb.fbcon ); + +	/* Restore saved VGA mode */ +	vesafb_restore(); +} + +/** + * Print a character to current cursor position + * + * @v character		Character + */ +static void vesafb_putchar ( int character ) { + +	fbcon_putchar ( &vesafb.fbcon, character ); +} + +/** + * Configure console + * + * @v config		Console configuration, or NULL to reset + * @ret rc		Return status code + */ +static int vesafb_configure ( struct console_configuration *config ) { +	int rc; + +	/* Reset console, if applicable */ +	if ( ! vesafb_console.disabled ) { +		vesafb_fini(); +		bios_console.disabled &= ~CONSOLE_DISABLED_OUTPUT; +		ansicol_reset_magic(); +	} +	vesafb_console.disabled = CONSOLE_DISABLED; + +	/* Do nothing more unless we have a usable configuration */ +	if ( ( config == NULL ) || +	     ( config->width == 0 ) || ( config->height == 0 ) ) { +		return 0; +	} + +	/* Initialise VESA frame buffer */ +	if ( ( rc = vesafb_init ( config ) ) != 0 ) +		return rc; + +	/* Mark console as enabled */ +	vesafb_console.disabled = 0; +	bios_console.disabled |= CONSOLE_DISABLED_OUTPUT; + +	/* Set magic colour to transparent if we have a background picture */ +	if ( config->pixbuf ) +		ansicol_set_magic_transparent(); + +	return 0; +} + +/** VESA frame buffer console driver */ +struct console_driver vesafb_console __console_driver = { +	.usage = CONSOLE_VESAFB, +	.putchar = vesafb_putchar, +	.configure = vesafb_configure, +	.disabled = CONSOLE_DISABLED, +}; diff --git a/roms/ipxe/src/arch/i386/interface/pxe/pxe_call.c b/roms/ipxe/src/arch/i386/interface/pxe/pxe_call.c new file mode 100644 index 00000000..657d47b6 --- /dev/null +++ b/roms/ipxe/src/arch/i386/interface/pxe/pxe_call.c @@ -0,0 +1,349 @@ +/* + * Copyright (C) 2006 Michael Brown <mbrown@fensystems.co.uk>. + * + * This program is free software; you can redistribute it and/or + * modify it under the terms of the GNU General Public License as + * published by the Free Software Foundation; either version 2 of the + * License, or any later version. + * + * 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. + * + * You should have received a copy of the GNU General Public License + * along with this program; if not, write to the Free Software + * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA + * 02110-1301, USA. + */ + +FILE_LICENCE ( GPL2_OR_LATER ); + +#include <ipxe/uaccess.h> +#include <ipxe/init.h> +#include <ipxe/profile.h> +#include <setjmp.h> +#include <registers.h> +#include <biosint.h> +#include <pxe.h> +#include <pxe_call.h> + +/** @file + * + * PXE API entry point + */ + +/* Disambiguate the various error causes */ +#define EINFO_EPXENBP							\ +	__einfo_uniqify ( EINFO_EPLATFORM, 0x01,			\ +			  "External PXE NBP error" ) +#define EPXENBP( status ) EPLATFORM ( EINFO_EPXENBP, status ) + +/** Vector for chaining INT 1A */ +extern struct segoff __text16 ( pxe_int_1a_vector ); +#define pxe_int_1a_vector __use_text16 ( pxe_int_1a_vector ) + +/** INT 1A handler */ +extern void pxe_int_1a ( void ); + +/** INT 1A hooked flag */ +static int int_1a_hooked = 0; + +/** PXENV_UNDI_TRANSMIT API call profiler */ +static struct profiler pxe_api_tx_profiler __profiler = +	{ .name = "pxeapi.tx" }; + +/** PXENV_UNDI_ISR API call profiler */ +static struct profiler pxe_api_isr_profiler __profiler = +	{ .name = "pxeapi.isr" }; + +/** PXE unknown API call profiler + * + * This profiler can be used to measure the overhead of a dummy PXE + * API call. + */ +static struct profiler pxe_api_unknown_profiler __profiler = +	{ .name = "pxeapi.unknown" }; + +/** Miscellaneous PXE API call profiler */ +static struct profiler pxe_api_misc_profiler __profiler = +	{ .name = "pxeapi.misc" }; + +/** + * Handle an unknown PXE API call + * + * @v pxenv_unknown 			Pointer to a struct s_PXENV_UNKNOWN + * @ret #PXENV_EXIT_FAILURE		Always + * @err #PXENV_STATUS_UNSUPPORTED	Always + */ +static PXENV_EXIT_t pxenv_unknown ( struct s_PXENV_UNKNOWN *pxenv_unknown ) { +	pxenv_unknown->Status = PXENV_STATUS_UNSUPPORTED; +	return PXENV_EXIT_FAILURE; +} + +/** Unknown PXE API call list */ +struct pxe_api_call pxenv_unknown_api __pxe_api_call = +	PXE_API_CALL ( PXENV_UNKNOWN, pxenv_unknown, struct s_PXENV_UNKNOWN ); + +/** + * Locate PXE API call + * + * @v opcode		Opcode + * @ret call		PXE API call, or NULL + */ +static struct pxe_api_call * find_pxe_api_call ( uint16_t opcode ) { +	struct pxe_api_call *call; + +	for_each_table_entry ( call, PXE_API_CALLS ) { +		if ( call->opcode == opcode ) +			return call; +	} +	return NULL; +} + +/** + * Determine applicable profiler (for debugging) + * + * @v opcode		PXE opcode + * @ret profiler	Profiler + */ +static struct profiler * pxe_api_profiler ( unsigned int opcode ) { + +	/* Determine applicable profiler */ +	switch ( opcode ) { +	case PXENV_UNDI_TRANSMIT: +		return &pxe_api_tx_profiler; +	case PXENV_UNDI_ISR: +		return &pxe_api_isr_profiler; +	case PXENV_UNKNOWN: +		return &pxe_api_unknown_profiler; +	default: +		return &pxe_api_misc_profiler; +	} +} + +/** + * Dispatch PXE API call + * + * @v bx		PXE opcode + * @v es:di		Address of PXE parameter block + * @ret ax		PXE exit code + */ +__asmcall void pxe_api_call ( struct i386_all_regs *ix86 ) { +	uint16_t opcode = ix86->regs.bx; +	userptr_t uparams = real_to_user ( ix86->segs.es, ix86->regs.di ); +	struct profiler *profiler = pxe_api_profiler ( opcode ); +	struct pxe_api_call *call; +	union u_PXENV_ANY params; +	PXENV_EXIT_t ret; + +	/* Start profiling */ +	profile_start ( profiler ); + +	/* Locate API call */ +	call = find_pxe_api_call ( opcode ); +	if ( ! call ) { +		DBGC ( &pxe_netdev, "PXENV_UNKNOWN_%04x\n", opcode ); +		call = &pxenv_unknown_api; +	} + +	/* Copy parameter block from caller */ +	copy_from_user ( ¶ms, uparams, 0, call->params_len ); + +	/* Set default status in case child routine fails to do so */ +	params.Status = PXENV_STATUS_FAILURE; + +	/* Hand off to relevant API routine */ +	ret = call->entry ( ¶ms ); + +	/* Copy modified parameter block back to caller and return */ +	copy_to_user ( uparams, 0, ¶ms, call->params_len ); +	ix86->regs.ax = ret; + +	/* Stop profiling, if applicable */ +	profile_stop ( profiler ); +} + +/** + * Dispatch weak PXE API call with PXE stack available + * + * @v ix86		Registers for PXE call + * @ret present		Zero (PXE stack present) + */ +int pxe_api_call_weak ( struct i386_all_regs *ix86 ) { +	pxe_api_call ( ix86 ); +	return 0; +} + +/** + * Dispatch PXE loader call + * + * @v es:di		Address of PXE parameter block + * @ret ax		PXE exit code + */ +__asmcall void pxe_loader_call ( struct i386_all_regs *ix86 ) { +	userptr_t uparams = real_to_user ( ix86->segs.es, ix86->regs.di ); +	struct s_UNDI_LOADER params; +	PXENV_EXIT_t ret; + +	/* Copy parameter block from caller */ +	copy_from_user ( ¶ms, uparams, 0, sizeof ( params ) ); + +	/* Fill in ROM segment address */ +	ppxe.UNDIROMID.segment = ix86->segs.ds; + +	/* Set default status in case child routine fails to do so */ +	params.Status = PXENV_STATUS_FAILURE; + +	/* Call UNDI loader */ +	ret = undi_loader ( ¶ms ); + +	/* Copy modified parameter block back to caller and return */ +	copy_to_user ( uparams, 0, ¶ms, sizeof ( params ) ); +	ix86->regs.ax = ret; +} + +/** + * Calculate byte checksum as used by PXE + * + * @v data		Data + * @v size		Length of data + * @ret sum		Checksum + */ +static uint8_t pxe_checksum ( void *data, size_t size ) { +	uint8_t *bytes = data; +	uint8_t sum = 0; + +	while ( size-- ) { +		sum += *bytes++; +	} +	return sum; +} + +/** + * Initialise !PXE and PXENV+ structures + * + */ +static void pxe_init_structures ( void ) { +	uint32_t rm_cs_phys = ( rm_cs << 4 ); +	uint32_t rm_ds_phys = ( rm_ds << 4 ); + +	/* Fill in missing segment fields */ +	ppxe.EntryPointSP.segment = rm_cs; +	ppxe.EntryPointESP.segment = rm_cs; +	ppxe.Stack.segment_address = rm_ds; +	ppxe.Stack.Physical_address = rm_ds_phys; +	ppxe.UNDIData.segment_address = rm_ds; +	ppxe.UNDIData.Physical_address = rm_ds_phys; +	ppxe.UNDICode.segment_address = rm_cs; +	ppxe.UNDICode.Physical_address = rm_cs_phys; +	ppxe.UNDICodeWrite.segment_address = rm_cs; +	ppxe.UNDICodeWrite.Physical_address = rm_cs_phys; +	pxenv.RMEntry.segment = rm_cs; +	pxenv.StackSeg = rm_ds; +	pxenv.UNDIDataSeg = rm_ds; +	pxenv.UNDICodeSeg = rm_cs; +	pxenv.PXEPtr.segment = rm_cs; + +	/* Update checksums */ +	ppxe.StructCksum -= pxe_checksum ( &ppxe, sizeof ( ppxe ) ); +	pxenv.Checksum -= pxe_checksum ( &pxenv, sizeof ( pxenv ) ); +} + +/** PXE structure initialiser */ +struct init_fn pxe_init_fn __init_fn ( INIT_NORMAL ) = { +	.initialise = pxe_init_structures, +}; + +/** + * Activate PXE stack + * + * @v netdev		Net device to use as PXE net device + */ +void pxe_activate ( struct net_device *netdev ) { + +	/* Ensure INT 1A is hooked */ +	if ( ! int_1a_hooked ) { +		hook_bios_interrupt ( 0x1a, ( unsigned int ) pxe_int_1a, +				      &pxe_int_1a_vector ); +		devices_get(); +		int_1a_hooked = 1; +	} + +	/* Set PXE network device */ +	pxe_set_netdev ( netdev ); +} + +/** + * Deactivate PXE stack + * + * @ret rc		Return status code + */ +int pxe_deactivate ( void ) { +	int rc; + +	/* Clear PXE network device */ +	pxe_set_netdev ( NULL ); + +	/* Ensure INT 1A is unhooked, if possible */ +	if ( int_1a_hooked ) { +		if ( ( rc = unhook_bios_interrupt ( 0x1a, +						    (unsigned int) pxe_int_1a, +						    &pxe_int_1a_vector ))!= 0){ +			DBG ( "Could not unhook INT 1A: %s\n", +			      strerror ( rc ) ); +			return rc; +		} +		devices_put(); +		int_1a_hooked = 0; +	} + +	return 0; +} + +/** Jump buffer for PXENV_RESTART_TFTP */ +rmjmp_buf pxe_restart_nbp; + +/** + * Start PXE NBP at 0000:7c00 + * + * @ret rc		Return status code + */ +int pxe_start_nbp ( void ) { +	int jmp; +	int discard_b, discard_c, discard_d, discard_D; +	uint16_t status; + +	/* Allow restarting NBP via PXENV_RESTART_TFTP */ +	jmp = rmsetjmp ( pxe_restart_nbp ); +	if ( jmp ) +		DBG ( "Restarting NBP (%x)\n", jmp ); + +	/* Far call to PXE NBP */ +	__asm__ __volatile__ ( REAL_CODE ( "pushl %%ebp\n\t" /* gcc bug */ +					   "movw %%cx, %%es\n\t" +					   "pushw %%es\n\t" +					   "pushw %%di\n\t" +					   "sti\n\t" +					   "lcall $0, $0x7c00\n\t" +					   "popl %%ebp\n\t" /* discard */ +					   "popl %%ebp\n\t" /* gcc bug */ ) +			       : "=a" ( status ), "=b" ( discard_b ), +				 "=c" ( discard_c ), "=d" ( discard_d ), +				 "=D" ( discard_D ) +			       : "a" ( 0 ), "b" ( __from_text16 ( &pxenv ) ), +			         "c" ( rm_cs ), +			         "d" ( virt_to_phys ( &pxenv ) ), +				 "D" ( __from_text16 ( &ppxe ) ) +			       : "esi", "memory" ); +	if ( status ) +		return -EPXENBP ( status ); + +	return 0; +} + +REQUIRE_OBJECT ( pxe_preboot ); +REQUIRE_OBJECT ( pxe_undi ); +REQUIRE_OBJECT ( pxe_udp ); +REQUIRE_OBJECT ( pxe_tftp ); +REQUIRE_OBJECT ( pxe_file ); diff --git a/roms/ipxe/src/arch/i386/interface/pxe/pxe_entry.S b/roms/ipxe/src/arch/i386/interface/pxe/pxe_entry.S new file mode 100644 index 00000000..6274264f --- /dev/null +++ b/roms/ipxe/src/arch/i386/interface/pxe/pxe_entry.S @@ -0,0 +1,217 @@ +/* + * Copyright (C) 2006 Michael Brown <mbrown@fensystems.co.uk>. + * + * This program is free software; you can redistribute it and/or + * modify it under the terms of the GNU General Public License as + * published by the Free Software Foundation; either version 2 of the + * License, or any later version. + * + * 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. + * + * You should have received a copy of the GNU General Public License + * along with this program; if not, write to the Free Software + * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA + * 02110-1301, USA. + * + */ + +FILE_LICENCE ( GPL2_OR_LATER ) + +	.arch i386 + +/**************************************************************************** + * !PXE structure + **************************************************************************** + */ +	.section ".text16.data", "aw", @progbits +	.globl ppxe +	.align 16 +ppxe: +	.ascii "!PXE"			/* Signature */ +	.byte pxe_length		/* StructLength */ +	.byte 0				/* StructCksum */ +	.byte 0				/* StructRev */ +	.byte 0				/* reserved_1 */ +	.word undiheader, 0		/* UNDIROMID */ +	.word 0, 0			/* BaseROMID */ +	.word pxe_entry_sp, 0		/* EntryPointSP */ +	.word pxe_entry_esp, 0		/* EntryPointESP */ +	.word -1, -1			/* StatusCallout */ +	.byte 0				/* reserved_2 */ +	.byte SegDescCnt		/* SegDescCnt */ +	.word 0				/* FirstSelector */ +pxe_segments: +	.word 0, 0, 0, _data16_memsz	/* Stack */ +	.word 0, 0, 0, _data16_memsz	/* UNDIData */ +	.word 0, 0, 0, _text16_memsz	/* UNDICode */ +	.word 0, 0, 0, _text16_memsz	/* UNDICodeWrite */ +	.word 0, 0, 0, 0		/* BC_Data */ +	.word 0, 0, 0, 0		/* BC_Code */ +	.word 0, 0, 0, 0		/* BC_CodeWrite */ +	.equ	SegDescCnt, ( ( . - pxe_segments ) / 8 ) +	.equ	pxe_length, . - ppxe +	.size	ppxe, . - ppxe + +	/* Define undiheader=0 as a weak symbol for non-ROM builds */ +	.section ".weak", "a", @nobits +	.weak	undiheader +undiheader: + +/**************************************************************************** + * PXENV+ structure + **************************************************************************** + */ +	.section ".text16.data", "aw", @progbits +	.globl pxenv +	.align 16 +pxenv: +	.ascii "PXENV+"			/* Signature */ +	.word 0x0201			/* Version */ +	.byte pxenv_length		/* Length */ +	.byte 0				/* Checksum */ +	.word pxenv_entry, 0		/* RMEntry */ +	.long 0				/* PMEntry */ +	.word 0				/* PMSelector */ +	.word 0				/* StackSeg */ +	.word _data16_memsz		/* StackSize */ +	.word 0				/* BC_CodeSeg */ +	.word 0				/* BC_CodeSize */ +	.word 0				/* BC_DataSeg */ +	.word 0				/* BC_DataSize */ +	.word 0				/* UNDIDataSeg */ +	.word _data16_memsz		/* UNDIDataSize */ +	.word 0				/* UNDICodeSeg */ +	.word _text16_memsz		/* UNDICodeSize */ +	.word ppxe, 0			/* PXEPtr */ +	.equ	pxenv_length, . - pxenv +	.size	pxenv, . - pxenv +  +/**************************************************************************** + * pxenv_entry (16-bit far call) + * + * PXE API call PXENV+ entry point + * + * Parameters: + *   %es:di : Far pointer to PXE parameter structure + *   %bx : PXE API call + * Returns: + *   %ax : PXE exit status + * Corrupts: + *   none + **************************************************************************** + */ +	/* Wyse Streaming Manager server (WLDRM13.BIN) assumes that +	 * the PXENV+ entry point is at UNDI_CS:0000; apparently, +	 * somebody at Wyse has difficulty distinguishing between the +	 * words "may" and "must"... +	 */ +	.section ".text16.null", "ax", @progbits +	.code16 +pxenv_null_entry: +	jmp	pxenv_entry + +	.section ".text16", "ax", @progbits +	.code16 +pxenv_entry: +	pushl	$pxe_api_call +	pushw	%cs +	call	prot_call +	addl	$4, %esp +	lret +	.size	pxenv_entry, . - pxenv_entry + +/**************************************************************************** + * pxe_entry + * + * PXE API call !PXE entry point + * + * Parameters: + *   stack : Far pointer to PXE parameter structure + *   stack : PXE API call + * Returns: + *   %ax : PXE exit status + * Corrupts: + *   none + **************************************************************************** + */ +	.section ".text16", "ax", @progbits +	.code16 +pxe_entry: +pxe_entry_sp: +	/* Preserve original %esp */ +	pushl	%esp +	/* Zero high word of %esp to allow use of common code */ +	movzwl	%sp, %esp +	jmp	pxe_entry_common +pxe_entry_esp: +	/* Preserve %esp to match behaviour of pxe_entry_sp */ +	pushl	%esp +pxe_entry_common: +	/* Save PXENV+ API call registers */ +	pushw	%es +	pushw	%di +	pushw	%bx +	/* Load !PXE parameters from stack into PXENV+ registers */ +	addr32 movw	18(%esp), %bx +	movw	%bx, %es +	addr32 movw	16(%esp), %di +	addr32 movw	14(%esp), %bx +	/* Make call as for PXENV+ */ +	pushw	%cs +	call	pxenv_entry +	/* Restore PXENV+ registers */ +	popw	%bx +	popw	%di +	popw	%es +	/* Restore original %esp and return */ +	popl	%esp +	lret +	.size	pxe_entry, . - pxe_entry + +/**************************************************************************** + * pxe_int_1a + * + * PXE INT 1A handler + * + * Parameters: + *   %ax : 0x5650 + * Returns: + *   %ax : 0x564e + *   %es:bx : Far pointer to the PXENV+ structure + *   %edx : Physical address of the PXENV+ structure + *   CF cleared + * Corrupts: + *   none + **************************************************************************** + */ +	.section ".text16", "ax", @progbits +	.code16 +	.globl	pxe_int_1a +pxe_int_1a: +	pushfw +	cmpw	$0x5650, %ax +	jne	1f +	/* INT 1A,5650 - PXE installation check */ +	xorl	%edx, %edx +	movw	%cs, %dx +	movw	%dx, %es +	movw	$pxenv, %bx +	shll	$4, %edx +	addl	$pxenv, %edx +	movw	$0x564e, %ax +	pushw	%bp +	movw	%sp, %bp +	andb	$~0x01, 8(%bp)	/* Clear CF on return */ +	popw	%bp +	popfw +	iret +1:	/* INT 1A,other - pass through */ +	popfw +	ljmp	*%cs:pxe_int_1a_vector + +	.section ".text16.data", "aw", @progbits +	.globl	pxe_int_1a_vector +pxe_int_1a_vector:	.long 0 diff --git a/roms/ipxe/src/arch/i386/interface/pxe/pxe_exit_hook.c b/roms/ipxe/src/arch/i386/interface/pxe/pxe_exit_hook.c new file mode 100644 index 00000000..9d189650 --- /dev/null +++ b/roms/ipxe/src/arch/i386/interface/pxe/pxe_exit_hook.c @@ -0,0 +1,61 @@ +/** @file + * + * PXE exit hook + * + */ + +/* + * Copyright (C) 2010 Shao Miller <shao.miller@yrdsb.edu.on.ca>. + * + * This program is free software; you can redistribute it and/or + * modify it under the terms of the GNU General Public License as + * published by the Free Software Foundation; either version 2 of the + * License, or any later version. + * + * 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. + * + * You should have received a copy of the GNU General Public License + * along with this program; if not, write to the Free Software + * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA + * 02110-1301, USA. + */ + +FILE_LICENCE ( GPL2_OR_LATER ); + +#include <stdint.h> +#include <realmode.h> +#include <pxe.h> + +/** PXE exit hook */ +extern segoff_t __data16 ( pxe_exit_hook ); +#define pxe_exit_hook __use_data16 ( pxe_exit_hook ) + +/** + * FILE EXIT HOOK + * + * @v file_exit_hook			Pointer to a struct + *					s_PXENV_FILE_EXIT_HOOK + * @v s_PXENV_FILE_EXIT_HOOK::Hook	SEG16:OFF16 to jump to + * @ret #PXENV_EXIT_SUCCESS		Successfully set hook + * @ret #PXENV_EXIT_FAILURE		We're not an NBP build + * @ret s_PXENV_FILE_EXIT_HOOK::Status	PXE status code + * + */ +static PXENV_EXIT_t +pxenv_file_exit_hook ( struct s_PXENV_FILE_EXIT_HOOK *file_exit_hook ) { +	DBG ( "PXENV_FILE_EXIT_HOOK" ); + +	/* We'll jump to the specified SEG16:OFF16 during exit */ +	pxe_exit_hook.segment = file_exit_hook->Hook.segment; +	pxe_exit_hook.offset = file_exit_hook->Hook.offset; +	file_exit_hook->Status = PXENV_STATUS_SUCCESS; +	return PXENV_EXIT_SUCCESS; +} + +/** PXE file API */ +struct pxe_api_call pxe_file_api_exit_hook __pxe_api_call = +	PXE_API_CALL ( PXENV_FILE_EXIT_HOOK, pxenv_file_exit_hook, +		       struct s_PXENV_FILE_EXIT_HOOK ); diff --git a/roms/ipxe/src/arch/i386/interface/pxe/pxe_file.c b/roms/ipxe/src/arch/i386/interface/pxe/pxe_file.c new file mode 100644 index 00000000..6e961029 --- /dev/null +++ b/roms/ipxe/src/arch/i386/interface/pxe/pxe_file.c @@ -0,0 +1,342 @@ +/** @file + * + * PXE FILE API + * + */ + +#include <stdlib.h> +#include <stdio.h> +#include <errno.h> +#include <byteswap.h> +#include <ipxe/uaccess.h> +#include <ipxe/posix_io.h> +#include <ipxe/features.h> +#include <pxe.h> +#include <realmode.h> + +/* + * Copyright (C) 2007 Michael Brown <mbrown@fensystems.co.uk>. + * + * This program is free software; you can redistribute it and/or + * modify it under the terms of the GNU General Public License as + * published by the Free Software Foundation; either version 2 of the + * License, or any later version. + * + * 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. + * + * You should have received a copy of the GNU General Public License + * along with this program; if not, write to the Free Software + * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA + * 02110-1301, USA. + */ + +FILE_LICENCE ( GPL2_OR_LATER ); + +FEATURE ( FEATURE_MISC, "PXEXT", DHCP_EB_FEATURE_PXE_EXT, 2 ); + +/** + * FILE OPEN + * + * @v file_open				Pointer to a struct s_PXENV_FILE_OPEN + * @v s_PXENV_FILE_OPEN::FileName	URL of file to open + * @ret #PXENV_EXIT_SUCCESS		File was opened + * @ret #PXENV_EXIT_FAILURE		File was not opened + * @ret s_PXENV_FILE_OPEN::Status	PXE status code + * @ret s_PXENV_FILE_OPEN::FileHandle	Handle of opened file + * + */ +static PXENV_EXIT_t pxenv_file_open ( struct s_PXENV_FILE_OPEN *file_open ) { +	userptr_t filename; +	size_t filename_len; +	int fd; + +	DBG ( "PXENV_FILE_OPEN" ); + +	/* Copy name from external program, and open it */ +	filename = real_to_user ( file_open->FileName.segment, +			      file_open->FileName.offset ); +	filename_len = strlen_user ( filename, 0 ); +	{ +		char uri_string[ filename_len + 1 ]; + +		copy_from_user ( uri_string, filename, 0, +				 sizeof ( uri_string ) ); +		DBG ( " %s", uri_string ); +		fd = open ( uri_string ); +	} + +	if ( fd < 0 ) { +		file_open->Status = PXENV_STATUS ( fd ); +		return PXENV_EXIT_FAILURE; +	} + +	DBG ( " as file %d", fd ); + +	file_open->FileHandle = fd; +	file_open->Status = PXENV_STATUS_SUCCESS; +	return PXENV_EXIT_SUCCESS; +} + +/** + * FILE CLOSE + * + * @v file_close			Pointer to a struct s_PXENV_FILE_CLOSE + * @v s_PXENV_FILE_CLOSE::FileHandle	File handle + * @ret #PXENV_EXIT_SUCCESS		File was closed + * @ret #PXENV_EXIT_FAILURE		File was not closed + * @ret s_PXENV_FILE_CLOSE::Status	PXE status code + * + */ +static PXENV_EXIT_t pxenv_file_close ( struct s_PXENV_FILE_CLOSE *file_close ) { + +	DBG ( "PXENV_FILE_CLOSE %d", file_close->FileHandle ); + +	close ( file_close->FileHandle ); +	file_close->Status = PXENV_STATUS_SUCCESS; +	return PXENV_EXIT_SUCCESS; +} + +/** + * FILE SELECT + * + * @v file_select			Pointer to a struct s_PXENV_FILE_SELECT + * @v s_PXENV_FILE_SELECT::FileHandle	File handle + * @ret #PXENV_EXIT_SUCCESS		File has been checked for readiness + * @ret #PXENV_EXIT_FAILURE		File has not been checked for readiness + * @ret s_PXENV_FILE_SELECT::Status	PXE status code + * @ret s_PXENV_FILE_SELECT::Ready	Indication of readiness + * + */ +static PXENV_EXIT_t +pxenv_file_select ( struct s_PXENV_FILE_SELECT *file_select ) { +	fd_set fdset; +	int ready; + +	DBG ( "PXENV_FILE_SELECT %d", file_select->FileHandle ); + +	FD_ZERO ( &fdset ); +	FD_SET ( file_select->FileHandle, &fdset ); +	if ( ( ready = select ( &fdset, 0 ) ) < 0 ) { +		file_select->Status = PXENV_STATUS ( ready ); +		return PXENV_EXIT_FAILURE; +	} + +	file_select->Ready = ( ready ? RDY_READ : 0 ); +	file_select->Status = PXENV_STATUS_SUCCESS; +	return PXENV_EXIT_SUCCESS; +} + +/** + * FILE READ + * + * @v file_read				Pointer to a struct s_PXENV_FILE_READ + * @v s_PXENV_FILE_READ::FileHandle	File handle + * @v s_PXENV_FILE_READ::BufferSize	Size of data buffer + * @v s_PXENV_FILE_READ::Buffer		Data buffer + * @ret #PXENV_EXIT_SUCCESS		Data has been read from file + * @ret #PXENV_EXIT_FAILURE		Data has not been read from file + * @ret s_PXENV_FILE_READ::Status	PXE status code + * @ret s_PXENV_FILE_READ::Ready	Indication of readiness + * @ret s_PXENV_FILE_READ::BufferSize	Length of data read + * + */ +static PXENV_EXIT_t pxenv_file_read ( struct s_PXENV_FILE_READ *file_read ) { +	userptr_t buffer; +	ssize_t len; + +	DBG ( "PXENV_FILE_READ %d to %04x:%04x+%04x", file_read->FileHandle, +	      file_read->Buffer.segment, file_read->Buffer.offset, +	      file_read->BufferSize ); + +	buffer = real_to_user ( file_read->Buffer.segment, +				file_read->Buffer.offset ); +	if ( ( len = read_user ( file_read->FileHandle, buffer, 0, +				file_read->BufferSize ) ) < 0 ) { +		file_read->Status = PXENV_STATUS ( len ); +		return PXENV_EXIT_FAILURE; +	} + +	DBG ( " read %04zx", ( ( size_t ) len ) ); + +	file_read->BufferSize = len; +	file_read->Status = PXENV_STATUS_SUCCESS; +	return PXENV_EXIT_SUCCESS; +} + +/** + * GET FILE SIZE + * + * @v get_file_size			Pointer to a struct s_PXENV_GET_FILE_SIZE + * @v s_PXENV_GET_FILE_SIZE::FileHandle	File handle + * @ret #PXENV_EXIT_SUCCESS		File size has been determined + * @ret #PXENV_EXIT_FAILURE		File size has not been determined + * @ret s_PXENV_GET_FILE_SIZE::Status	PXE status code + * @ret s_PXENV_GET_FILE_SIZE::FileSize	Size of file + */ +static PXENV_EXIT_t +pxenv_get_file_size ( struct s_PXENV_GET_FILE_SIZE *get_file_size ) { +	ssize_t filesize; + +	DBG ( "PXENV_GET_FILE_SIZE %d", get_file_size->FileHandle ); + +	filesize = fsize ( get_file_size->FileHandle ); +	if ( filesize < 0 ) { +		get_file_size->Status = PXENV_STATUS ( filesize ); +		return PXENV_EXIT_FAILURE; +	} + +	DBG ( " is %zd", ( ( size_t ) filesize ) ); + +	get_file_size->FileSize = filesize; +	get_file_size->Status = PXENV_STATUS_SUCCESS; +	return PXENV_EXIT_SUCCESS; +} + +/** + * FILE EXEC + * + * @v file_exec				Pointer to a struct s_PXENV_FILE_EXEC + * @v s_PXENV_FILE_EXEC::Command	Command to execute + * @ret #PXENV_EXIT_SUCCESS		Command was executed successfully + * @ret #PXENV_EXIT_FAILURE		Command was not executed successfully + * @ret s_PXENV_FILE_EXEC::Status	PXE status code + * + */ +static PXENV_EXIT_t pxenv_file_exec ( struct s_PXENV_FILE_EXEC *file_exec ) { +	userptr_t command; +	size_t command_len; +	int rc; + +	DBG ( "PXENV_FILE_EXEC" ); + +	/* Copy name from external program, and exec it */ +	command = real_to_user ( file_exec->Command.segment, +				 file_exec->Command.offset ); +	command_len = strlen_user ( command, 0 ); +	{ +		char command_string[ command_len + 1 ]; + +		copy_from_user ( command_string, command, 0, +				 sizeof ( command_string ) ); +		DBG ( " %s", command_string ); + +		if ( ( rc = system ( command_string ) ) != 0 ) { +			file_exec->Status = PXENV_STATUS ( rc ); +			return PXENV_EXIT_FAILURE; +		} +	} + +	file_exec->Status = PXENV_STATUS_SUCCESS; +	return PXENV_EXIT_SUCCESS; +} + +/** + * FILE CMDLINE + * + * @v file_cmdline			Pointer to a struct s_PXENV_FILE_CMDLINE + * @v s_PXENV_FILE_CMDLINE::Buffer	Buffer to contain command line + * @v s_PXENV_FILE_CMDLINE::BufferSize	Size of buffer + * @ret #PXENV_EXIT_SUCCESS		Command was executed successfully + * @ret #PXENV_EXIT_FAILURE		Command was not executed successfully + * @ret s_PXENV_FILE_EXEC::Status	PXE status code + * @ret s_PXENV_FILE_EXEC::BufferSize	Length of command line (including NUL) + * + */ +static PXENV_EXIT_t +pxenv_file_cmdline ( struct s_PXENV_FILE_CMDLINE *file_cmdline ) { +	userptr_t buffer; +	size_t max_len; +	size_t len; + +	DBG ( "PXENV_FILE_CMDLINE to %04x:%04x+%04x \"%s\"\n", +	      file_cmdline->Buffer.segment, file_cmdline->Buffer.offset, +	      file_cmdline->BufferSize, pxe_cmdline ); + +	buffer = real_to_user ( file_cmdline->Buffer.segment, +				file_cmdline->Buffer.offset ); +	len = file_cmdline->BufferSize; +	max_len = ( pxe_cmdline ? +		    ( strlen ( pxe_cmdline ) + 1 /* NUL */ ) : 0 ); +	if ( len > max_len ) +		len = max_len; +	copy_to_user ( buffer, 0, pxe_cmdline, len ); +	file_cmdline->BufferSize = max_len; + +	file_cmdline->Status = PXENV_STATUS_SUCCESS; +	return PXENV_EXIT_SUCCESS; +} + +/** + * FILE API CHECK + * + * @v file_exec				Pointer to a struct s_PXENV_FILE_API_CHECK + * @v s_PXENV_FILE_API_CHECK::Magic     Inbound magic number (0x91d447b2) + * @ret #PXENV_EXIT_SUCCESS		Command was executed successfully + * @ret #PXENV_EXIT_FAILURE		Command was not executed successfully + * @ret s_PXENV_FILE_API_CHECK::Status	PXE status code + * @ret s_PXENV_FILE_API_CHECK::Magic	Outbound magic number (0xe9c17b20) + * @ret s_PXENV_FILE_API_CHECK::Provider "iPXE" (0x45585067) + * @ret s_PXENV_FILE_API_CHECK::APIMask API function bitmask + * @ret s_PXENV_FILE_API_CHECK::Flags	Reserved + * + */ +static PXENV_EXIT_t +pxenv_file_api_check ( struct s_PXENV_FILE_API_CHECK *file_api_check ) { +	struct pxe_api_call *call; +	unsigned int mask = 0; +	unsigned int offset; + +	DBG ( "PXENV_FILE_API_CHECK" ); + +	/* Check for magic value */ +	if ( file_api_check->Magic != 0x91d447b2 ) { +		file_api_check->Status = PXENV_STATUS_BAD_FUNC; +		return PXENV_EXIT_FAILURE; +	} + +	/* Check for required parameter size */ +	if ( file_api_check->Size < sizeof ( *file_api_check ) ) { +		file_api_check->Status = PXENV_STATUS_OUT_OF_RESOURCES; +		return PXENV_EXIT_FAILURE; +	} + +	/* Determine supported calls */ +	for_each_table_entry ( call, PXE_API_CALLS ) { +		offset = ( call->opcode - PXENV_FILE_MIN ); +		if ( offset <= ( PXENV_FILE_MAX - PXENV_FILE_MIN ) ) +			mask |= ( 1 << offset ); +	} + +	/* Fill in parameters */ +	file_api_check->Size = sizeof ( *file_api_check ); +	file_api_check->Magic = 0xe9c17b20; +	file_api_check->Provider = 0x45585067; /* "iPXE" */ +	file_api_check->APIMask = mask; +	file_api_check->Flags = 0; /* None defined */ + +	file_api_check->Status = PXENV_STATUS_SUCCESS; +	return PXENV_EXIT_SUCCESS; +} + +/** PXE file API */ +struct pxe_api_call pxe_file_api[] __pxe_api_call = { +	PXE_API_CALL ( PXENV_FILE_OPEN, pxenv_file_open, +		       struct s_PXENV_FILE_OPEN ), +	PXE_API_CALL ( PXENV_FILE_CLOSE, pxenv_file_close, +		       struct s_PXENV_FILE_CLOSE ), +	PXE_API_CALL ( PXENV_FILE_SELECT, pxenv_file_select, +		       struct s_PXENV_FILE_SELECT ), +	PXE_API_CALL ( PXENV_FILE_READ, pxenv_file_read, +		       struct s_PXENV_FILE_READ ), +	PXE_API_CALL ( PXENV_GET_FILE_SIZE, pxenv_get_file_size, +		       struct s_PXENV_GET_FILE_SIZE ), +	PXE_API_CALL ( PXENV_FILE_EXEC, pxenv_file_exec, +		       struct s_PXENV_FILE_EXEC ), +	PXE_API_CALL ( PXENV_FILE_CMDLINE, pxenv_file_cmdline, +		       struct s_PXENV_FILE_CMDLINE ), +	PXE_API_CALL ( PXENV_FILE_API_CHECK, pxenv_file_api_check, +		       struct s_PXENV_FILE_API_CHECK ), +}; diff --git a/roms/ipxe/src/arch/i386/interface/pxe/pxe_loader.c b/roms/ipxe/src/arch/i386/interface/pxe/pxe_loader.c new file mode 100644 index 00000000..695af3b9 --- /dev/null +++ b/roms/ipxe/src/arch/i386/interface/pxe/pxe_loader.c @@ -0,0 +1,51 @@ +/* + * Copyright (C) 2007 Michael Brown <mbrown@fensystems.co.uk>. + * + * This program is free software; you can redistribute it and/or + * modify it under the terms of the GNU General Public License as + * published by the Free Software Foundation; either version 2 of the + * License, or any later version. + * + * 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. + * + * You should have received a copy of the GNU General Public License + * along with this program; if not, write to the Free Software + * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA + * 02110-1301, USA. + */ + +FILE_LICENCE ( GPL2_OR_LATER ); + +#include <ipxe/init.h> +#include "pxe.h" +#include "pxe_call.h" + +/** @file + * + * PXE UNDI loader + * + */ + +/* PXENV_UNDI_LOADER + * + */ +PXENV_EXIT_t undi_loader ( struct s_UNDI_LOADER *undi_loader ) { + +	/* Perform one-time initialisation (e.g. heap) */ +	initialise(); + +	DBG ( "[PXENV_UNDI_LOADER to CS %04x DS %04x]", +	      undi_loader->UNDI_CS, undi_loader->UNDI_DS ); + +	/* Fill in UNDI loader structure */ +	undi_loader->PXEptr.segment = rm_cs; +	undi_loader->PXEptr.offset = __from_text16 ( &ppxe ); +	undi_loader->PXENVptr.segment = rm_cs; +	undi_loader->PXENVptr.offset = __from_text16 ( &pxenv ); + +	undi_loader->Status = PXENV_STATUS_SUCCESS; +	return PXENV_EXIT_SUCCESS; +} diff --git a/roms/ipxe/src/arch/i386/interface/pxe/pxe_preboot.c b/roms/ipxe/src/arch/i386/interface/pxe/pxe_preboot.c new file mode 100644 index 00000000..534352b2 --- /dev/null +++ b/roms/ipxe/src/arch/i386/interface/pxe/pxe_preboot.c @@ -0,0 +1,385 @@ +/** @file + * + * PXE Preboot API + * + */ + +/* PXE API interface for Etherboot. + * + * Copyright (C) 2004 Michael Brown <mbrown@fensystems.co.uk>. + * + * This program is free software; you can redistribute it and/or + * modify it under the terms of the GNU General Public License as + * published by the Free Software Foundation; either version 2 of the + * License, or any later version. + * + * 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. + * + * You should have received a copy of the GNU General Public License + * along with this program; if not, write to the Free Software + * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA + * 02110-1301, USA. + */ + +FILE_LICENCE ( GPL2_OR_LATER ); + +#include <stdint.h> +#include <string.h> +#include <stdlib.h> +#include <setjmp.h> +#include <ipxe/uaccess.h> +#include <ipxe/dhcp.h> +#include <ipxe/fakedhcp.h> +#include <ipxe/device.h> +#include <ipxe/netdevice.h> +#include <ipxe/isapnp.h> +#include <ipxe/init.h> +#include <ipxe/if_ether.h> +#include <basemem_packet.h> +#include <biosint.h> +#include "pxe.h" +#include "pxe_call.h" + +/* Avoid dragging in isapnp.o unnecessarily */ +uint16_t isapnp_read_port; + +/** Zero-based versions of PXENV_GET_CACHED_INFO::PacketType */ +enum pxe_cached_info_indices { +	CACHED_INFO_DHCPDISCOVER = ( PXENV_PACKET_TYPE_DHCP_DISCOVER - 1 ), +	CACHED_INFO_DHCPACK = ( PXENV_PACKET_TYPE_DHCP_ACK - 1 ), +	CACHED_INFO_BINL = ( PXENV_PACKET_TYPE_CACHED_REPLY - 1 ), +	NUM_CACHED_INFOS +}; + +/** A cached DHCP packet */ +union pxe_cached_info { +	struct dhcphdr dhcphdr; +	/* This buffer must be *exactly* the size of a BOOTPLAYER_t +	 * structure, otherwise WinPE will die horribly.  It takes the +	 * size of *our* buffer and feeds it in to us as the size of +	 * one of *its* buffers.  If our buffer is larger than it +	 * expects, we therefore end up overwriting part of its data +	 * segment, since it tells us to do so.  (D'oh!) +	 * +	 * Note that a BOOTPLAYER_t is not necessarily large enough to +	 * hold a DHCP packet; this is a flaw in the PXE spec. +	 */ +	BOOTPLAYER_t packet; +} __attribute__ (( packed )); + +/** A PXE DHCP packet creator */ +struct pxe_dhcp_packet_creator { +	/** Create DHCP packet +	 * +	 * @v netdev		Network device +	 * @v data		Buffer for DHCP packet +	 * @v max_len		Size of DHCP packet buffer +	 * @ret rc		Return status code +	 */ +	int ( * create ) ( struct net_device *netdev, void *data, +			   size_t max_len ); +}; + +/** PXE DHCP packet creators */ +static struct pxe_dhcp_packet_creator pxe_dhcp_packet_creators[] = { +	[CACHED_INFO_DHCPDISCOVER] = { create_fakedhcpdiscover }, +	[CACHED_INFO_DHCPACK] = { create_fakedhcpack }, +	[CACHED_INFO_BINL] = { create_fakepxebsack }, +}; + +/** + * Name PXENV_GET_CACHED_INFO packet type + * + * @v packet_type	Packet type + * @ret name		Name of packet type + */ +static inline __attribute__ (( always_inline )) const char * +pxenv_get_cached_info_name ( int packet_type ) { +	switch ( packet_type ) { +	case PXENV_PACKET_TYPE_DHCP_DISCOVER: +		return "DHCPDISCOVER"; +	case PXENV_PACKET_TYPE_DHCP_ACK: +		return "DHCPACK"; +	case PXENV_PACKET_TYPE_CACHED_REPLY: +		return "BINL"; +	default: +		return "<INVALID>"; +	} +} + +/* The case in which the caller doesn't supply a buffer is really + * awkward to support given that we have multiple sources of options, + * and that we don't actually store the DHCP packets.  (We may not + * even have performed DHCP; we may have obtained all configuration + * from non-volatile stored options or from the command line.) + * + * Some NBPs rely on the buffers we provide being persistent, so we + * can't just use the temporary packet buffer.  4.5kB of base memory + * always wasted just because some clients are too lazy to provide + * their own buffers... + */ +static union pxe_cached_info __bss16_array ( cached_info, [NUM_CACHED_INFOS] ); +#define cached_info __use_data16 ( cached_info ) + +/** + * UNLOAD BASE CODE STACK + * + * @v None				- + * @ret ... + * + */ +static PXENV_EXIT_t +pxenv_unload_stack ( struct s_PXENV_UNLOAD_STACK *unload_stack ) { +	DBGC ( &pxe_netdev, "PXENV_UNLOAD_STACK\n" ); + +	unload_stack->Status = PXENV_STATUS_SUCCESS; +	return PXENV_EXIT_SUCCESS; +} + +/* PXENV_GET_CACHED_INFO + * + * Status: working + */ +static PXENV_EXIT_t +pxenv_get_cached_info ( struct s_PXENV_GET_CACHED_INFO *get_cached_info ) { +	struct pxe_dhcp_packet_creator *creator; +	union pxe_cached_info *info; +	unsigned int idx; +	size_t len; +	userptr_t buffer; +	int rc; + +	DBGC ( &pxe_netdev, "PXENV_GET_CACHED_INFO %s to %04x:%04x+%x", +	       pxenv_get_cached_info_name ( get_cached_info->PacketType ), +	       get_cached_info->Buffer.segment, +	       get_cached_info->Buffer.offset, get_cached_info->BufferSize ); + +	/* Sanity check */ +	if ( ! pxe_netdev ) { +		DBGC ( &pxe_netdev, "PXENV_GET_CACHED_INFO called with no " +		       "network device\n" ); +		get_cached_info->Status = PXENV_STATUS_UNDI_INVALID_STATE; +		return PXENV_EXIT_FAILURE; +	} + +	/* Sanity check */ +        idx = ( get_cached_info->PacketType - 1 ); +	if ( idx >= NUM_CACHED_INFOS ) { +		DBGC ( &pxe_netdev, " bad PacketType %d\n", +		       get_cached_info->PacketType ); +		goto err; +	} +	info = &cached_info[idx]; + +	/* Construct cached version of packet, if not already constructed. */ +	if ( ! info->dhcphdr.op ) { +		/* Construct DHCP packet */ +		creator = &pxe_dhcp_packet_creators[idx]; +		if ( ( rc = creator->create ( pxe_netdev, info, +					      sizeof ( *info ) ) ) != 0 ) { +			DBGC ( &pxe_netdev, " failed to build packet: %s\n", +			       strerror ( rc ) ); +			goto err; +		} +	} + +	len = get_cached_info->BufferSize; +	if ( len == 0 ) { +		/* Point client at our cached buffer. +		 * +		 * To add to the fun, Intel decided at some point in +		 * the evolution of the PXE specification to add the +		 * BufferLimit field, which we are meant to fill in +		 * with the length of our packet buffer, so that the +		 * caller can safely modify the boot server reply +		 * packet stored therein.  However, this field was not +		 * present in earlier versions of the PXE spec, and +		 * there is at least one PXE NBP (Altiris) which +		 * allocates only exactly enough space for this +		 * earlier, shorter version of the structure.  If we +		 * actually fill in the BufferLimit field, we +		 * therefore risk trashing random areas of the +		 * caller's memory.  If we *don't* fill it in, then +		 * the caller is at liberty to assume that whatever +		 * random value happened to be in that location +		 * represents the length of the buffer we've just +		 * passed back to it. +		 * +		 * Since older PXE stacks won't fill this field in +		 * anyway, it's probably safe to assume that no +		 * callers actually rely on it, so we choose to not +		 * fill it in. +		 */ +		get_cached_info->Buffer.segment = rm_ds; +		get_cached_info->Buffer.offset = __from_data16 ( info ); +		get_cached_info->BufferSize = sizeof ( *info ); +		DBGC ( &pxe_netdev, " using %04x:%04x+%04x['%x']", +		       get_cached_info->Buffer.segment, +		       get_cached_info->Buffer.offset, +		       get_cached_info->BufferSize, +		       get_cached_info->BufferLimit ); +	} else { +		/* Copy packet to client buffer */ +		if ( len > sizeof ( *info ) ) +			len = sizeof ( *info ); +		if ( len < sizeof ( *info ) ) +			DBGC ( &pxe_netdev, " buffer may be too short" ); +		buffer = real_to_user ( get_cached_info->Buffer.segment, +					get_cached_info->Buffer.offset ); +		copy_to_user ( buffer, 0, info, len ); +		get_cached_info->BufferSize = len; +	} + +	DBGC ( &pxe_netdev, "\n" ); +	get_cached_info->Status = PXENV_STATUS_SUCCESS; +	return PXENV_EXIT_SUCCESS; + + err: +	get_cached_info->Status = PXENV_STATUS_OUT_OF_RESOURCES; +	return PXENV_EXIT_FAILURE; +} + +/* PXENV_RESTART_TFTP + * + * Status: working + */ +static PXENV_EXIT_t +pxenv_restart_tftp ( struct s_PXENV_TFTP_READ_FILE *restart_tftp ) { +	PXENV_EXIT_t tftp_exit; + +	DBGC ( &pxe_netdev, "PXENV_RESTART_TFTP\n" ); + +	/* Words cannot describe the complete mismatch between the PXE +	 * specification and any possible version of reality... +	 */ +	restart_tftp->Buffer = PXE_LOAD_PHYS; /* Fixed by spec, apparently */ +	restart_tftp->BufferSize = ( 0xa0000 - PXE_LOAD_PHYS ); /* Near enough */ +	tftp_exit = pxenv_tftp_read_file ( restart_tftp ); +	if ( tftp_exit != PXENV_EXIT_SUCCESS ) +		return tftp_exit; + +	/* Restart NBP */ +	rmlongjmp ( pxe_restart_nbp, PXENV_RESTART_TFTP ); +} + +/* PXENV_START_UNDI + * + * Status: working + */ +static PXENV_EXIT_t pxenv_start_undi ( struct s_PXENV_START_UNDI *start_undi ) { +	unsigned int bus_type; +	unsigned int location; +	struct net_device *netdev; + +	DBGC ( &pxe_netdev, "PXENV_START_UNDI %04x:%04x:%04x\n", +	       start_undi->AX, start_undi->BX, start_undi->DX ); + +	/* Determine bus type and location.  Use a heuristic to decide +	 * whether we are PCI or ISAPnP +	 */ +	if ( ( start_undi->DX >= ISAPNP_READ_PORT_MIN ) && +	     ( start_undi->DX <= ISAPNP_READ_PORT_MAX ) && +	     ( start_undi->BX >= ISAPNP_CSN_MIN ) && +	     ( start_undi->BX <= ISAPNP_CSN_MAX ) ) { +		bus_type = BUS_TYPE_ISAPNP; +		location = start_undi->BX; +		/* Record ISAPnP read port for use by isapnp.c */ +		isapnp_read_port = start_undi->DX; +	} else { +		bus_type = BUS_TYPE_PCI; +		location = start_undi->AX; +	} + +	/* Probe for devices, etc. */ +	startup(); + +	/* Look for a matching net device */ +	netdev = find_netdev_by_location ( bus_type, location ); +	if ( ! netdev ) { +		DBGC ( &pxe_netdev, "PXENV_START_UNDI could not find matching " +		       "net device\n" ); +		start_undi->Status = PXENV_STATUS_UNDI_CANNOT_INITIALIZE_NIC; +		return PXENV_EXIT_FAILURE; +	} +	DBGC ( &pxe_netdev, "PXENV_START_UNDI found net device %s\n", +	       netdev->name ); + +	/* Activate PXE */ +	pxe_activate ( netdev ); + +	start_undi->Status = PXENV_STATUS_SUCCESS; +	return PXENV_EXIT_SUCCESS; +} + +/* PXENV_STOP_UNDI + * + * Status: working + */ +static PXENV_EXIT_t pxenv_stop_undi ( struct s_PXENV_STOP_UNDI *stop_undi ) { +	DBGC ( &pxe_netdev, "PXENV_STOP_UNDI\n" ); + +	/* Deactivate PXE */ +	pxe_deactivate(); + +	/* Prepare for unload */ +	shutdown_boot(); + +	/* Check to see if we still have any hooked interrupts */ +	if ( hooked_bios_interrupts != 0 ) { +		DBGC ( &pxe_netdev, "PXENV_STOP_UNDI failed: %d interrupts " +		       "still hooked\n", hooked_bios_interrupts ); +		stop_undi->Status = PXENV_STATUS_KEEP_UNDI; +		return PXENV_EXIT_FAILURE; +	} + +	stop_undi->Status = PXENV_STATUS_SUCCESS; +	return PXENV_EXIT_SUCCESS; +} + +/* PXENV_START_BASE + * + * Status: won't implement (requires major structural changes) + */ +static PXENV_EXIT_t pxenv_start_base ( struct s_PXENV_START_BASE *start_base ) { +	DBGC ( &pxe_netdev, "PXENV_START_BASE\n" ); + +	start_base->Status = PXENV_STATUS_UNSUPPORTED; +	return PXENV_EXIT_FAILURE; +} + +/* PXENV_STOP_BASE + * + * Status: working + */ +static PXENV_EXIT_t pxenv_stop_base ( struct s_PXENV_STOP_BASE *stop_base ) { +	DBGC ( &pxe_netdev, "PXENV_STOP_BASE\n" ); + +	/* The only time we will be called is when the NBP is trying +	 * to shut down the PXE stack.  There's nothing we need to do +	 * in this call. +	 */ + +	stop_base->Status = PXENV_STATUS_SUCCESS; +	return PXENV_EXIT_SUCCESS; +} + +/** PXE preboot API */ +struct pxe_api_call pxe_preboot_api[] __pxe_api_call = { +	PXE_API_CALL ( PXENV_UNLOAD_STACK, pxenv_unload_stack, +		       struct s_PXENV_UNLOAD_STACK ), +	PXE_API_CALL ( PXENV_GET_CACHED_INFO, pxenv_get_cached_info, +		       struct s_PXENV_GET_CACHED_INFO ), +	PXE_API_CALL ( PXENV_RESTART_TFTP, pxenv_restart_tftp, +		       struct s_PXENV_TFTP_READ_FILE ), +	PXE_API_CALL ( PXENV_START_UNDI, pxenv_start_undi, +		       struct s_PXENV_START_UNDI ), +	PXE_API_CALL ( PXENV_STOP_UNDI, pxenv_stop_undi, +		       struct s_PXENV_STOP_UNDI ), +	PXE_API_CALL ( PXENV_START_BASE, pxenv_start_base, +		       struct s_PXENV_START_BASE ), +	PXE_API_CALL ( PXENV_STOP_BASE, pxenv_stop_base, +		       struct s_PXENV_STOP_BASE ), +}; diff --git a/roms/ipxe/src/arch/i386/interface/pxe/pxe_tftp.c b/roms/ipxe/src/arch/i386/interface/pxe/pxe_tftp.c new file mode 100644 index 00000000..f4801bad --- /dev/null +++ b/roms/ipxe/src/arch/i386/interface/pxe/pxe_tftp.c @@ -0,0 +1,592 @@ +/** @file + * + * PXE TFTP API + * + */ + +/* + * Copyright (C) 2004 Michael Brown <mbrown@fensystems.co.uk>. + * + * This program is free software; you can redistribute it and/or + * modify it under the terms of the GNU General Public License as + * published by the Free Software Foundation; either version 2 of the + * License, or any later version. + * + * 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. + * + * You should have received a copy of the GNU General Public License + * along with this program; if not, write to the Free Software + * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA + * 02110-1301, USA. + */ + +FILE_LICENCE ( GPL2_OR_LATER ); + +#include <stdlib.h> +#include <stdio.h> +#include <errno.h> +#include <byteswap.h> +#include <ipxe/uaccess.h> +#include <ipxe/in.h> +#include <ipxe/tftp.h> +#include <ipxe/iobuf.h> +#include <ipxe/xfer.h> +#include <ipxe/open.h> +#include <ipxe/process.h> +#include <pxe.h> + +/** A PXE TFTP connection */ +struct pxe_tftp_connection { +	/** Data transfer interface */ +	struct interface xfer; +	/** Data buffer */ +	userptr_t buffer; +	/** Size of data buffer */ +	size_t size; +	/** Starting offset of data buffer */ +	size_t start; +	/** File position */ +	size_t offset; +	/** Maximum file position */ +	size_t max_offset; +	/** Block size */ +	size_t blksize; +	/** Block index */ +	unsigned int blkidx; +	/** Overall return status code */ +	int rc; +}; + +/** + * Close PXE TFTP connection + * + * @v pxe_tftp		PXE TFTP connection + * @v rc		Final status code + */ +static void pxe_tftp_close ( struct pxe_tftp_connection *pxe_tftp, int rc ) { +	intf_shutdown ( &pxe_tftp->xfer, rc ); +	pxe_tftp->rc = rc; +} + +/** + * Check flow control window + * + * @v pxe_tftp		PXE TFTP connection + * @ret len		Length of window + */ +static size_t pxe_tftp_xfer_window ( struct pxe_tftp_connection *pxe_tftp ) { + +	return pxe_tftp->blksize; +} + +/** + * Receive new data + * + * @v pxe_tftp		PXE TFTP connection + * @v iobuf		I/O buffer + * @v meta		Transfer metadata + * @ret rc		Return status code + */ +static int pxe_tftp_xfer_deliver ( struct pxe_tftp_connection *pxe_tftp, +				   struct io_buffer *iobuf, +				   struct xfer_metadata *meta ) { +	size_t len = iob_len ( iobuf ); +	int rc = 0; + +	/* Calculate new buffer position */ +	if ( meta->flags & XFER_FL_ABS_OFFSET ) +		pxe_tftp->offset = 0; +	pxe_tftp->offset += meta->offset; + +	/* Copy data block to buffer */ +	if ( len == 0 ) { +		/* No data (pure seek); treat as success */ +	} else if ( pxe_tftp->offset < pxe_tftp->start ) { +		DBG ( " buffer underrun at %zx (min %zx)", +		      pxe_tftp->offset, pxe_tftp->start ); +		rc = -ENOBUFS; +	} else if ( ( pxe_tftp->offset + len ) > +		    ( pxe_tftp->start + pxe_tftp->size ) ) { +		DBG ( " buffer overrun at %zx (max %zx)", +		      ( pxe_tftp->offset + len ), +		      ( pxe_tftp->start + pxe_tftp->size ) ); +		rc = -ENOBUFS; +	} else { +		copy_to_user ( pxe_tftp->buffer, +			       ( pxe_tftp->offset - pxe_tftp->start ), +			       iobuf->data, len ); +	} + +	/* Calculate new buffer position */ +	pxe_tftp->offset += len; + +	/* Record maximum offset as the file size */ +	if ( pxe_tftp->max_offset < pxe_tftp->offset ) +		pxe_tftp->max_offset = pxe_tftp->offset; + +	/* Terminate transfer on error */ +	if ( rc != 0 ) +		pxe_tftp_close ( pxe_tftp, rc ); + +	free_iob ( iobuf ); +	return rc; +} + +/** PXE TFTP connection interface operations */ +static struct interface_operation pxe_tftp_xfer_ops[] = { +	INTF_OP ( xfer_deliver, struct pxe_tftp_connection *, +		  pxe_tftp_xfer_deliver ), +	INTF_OP ( xfer_window, struct pxe_tftp_connection *, +		  pxe_tftp_xfer_window ), +	INTF_OP ( intf_close, struct pxe_tftp_connection *, pxe_tftp_close ), +}; + +/** PXE TFTP connection interface descriptor */ +static struct interface_descriptor pxe_tftp_xfer_desc = +	INTF_DESC ( struct pxe_tftp_connection, xfer, pxe_tftp_xfer_ops ); + +/** The PXE TFTP connection */ +static struct pxe_tftp_connection pxe_tftp = { +	.xfer = INTF_INIT ( pxe_tftp_xfer_desc ), +}; + +/** + * Maximum length of a PXE TFTP URI + * + * The PXE TFTP API provides 128 characters for the filename; the + * extra 128 bytes allow for the remainder of the URI. + */ +#define PXE_TFTP_URI_LEN 256 + +/** + * Open PXE TFTP connection + * + * @v ipaddress		IP address + * @v port		TFTP server port + * @v filename		File name + * @v blksize		Requested block size + * @ret rc		Return status code + */ +static int pxe_tftp_open ( uint32_t ipaddress, unsigned int port, +			   const unsigned char *filename, size_t blksize, +			   int sizeonly ) { +	char uri_string[PXE_TFTP_URI_LEN]; +	struct in_addr address; +	int rc; + +	/* Reset PXE TFTP connection structure */ +	memset ( &pxe_tftp, 0, sizeof ( pxe_tftp ) ); +	intf_init ( &pxe_tftp.xfer, &pxe_tftp_xfer_desc, NULL ); +	if ( blksize < TFTP_DEFAULT_BLKSIZE ) +		blksize = TFTP_DEFAULT_BLKSIZE; +	pxe_tftp.blksize = blksize; +	pxe_tftp.rc = -EINPROGRESS; + +	/* Construct URI string */ +	address.s_addr = ipaddress; +	if ( ! port ) +		port = htons ( TFTP_PORT ); +	snprintf ( uri_string, sizeof ( uri_string ), "tftp%s://%s:%d%s%s", +		   sizeonly ? "size" : "", inet_ntoa ( address ), +		   ntohs ( port ), ( ( filename[0] == '/' ) ? "" : "/" ), +		   filename ); +	DBG ( " %s", uri_string ); + +	/* Open PXE TFTP connection */ +	if ( ( rc = xfer_open_uri_string ( &pxe_tftp.xfer, +					   uri_string ) ) != 0 ) { +		DBG ( " could not open (%s)\n", strerror ( rc ) ); +		return rc; +	} + +	return 0; +} + +/** + * TFTP OPEN + * + * @v tftp_open				Pointer to a struct s_PXENV_TFTP_OPEN + * @v s_PXENV_TFTP_OPEN::ServerIPAddress TFTP server IP address + * @v s_PXENV_TFTP_OPEN::GatewayIPAddress Relay agent IP address, or 0.0.0.0 + * @v s_PXENV_TFTP_OPEN::FileName	Name of file to open + * @v s_PXENV_TFTP_OPEN::TFTPPort	TFTP server UDP port + * @v s_PXENV_TFTP_OPEN::PacketSize	TFTP blksize option to request + * @ret #PXENV_EXIT_SUCCESS		File was opened + * @ret #PXENV_EXIT_FAILURE		File was not opened + * @ret s_PXENV_TFTP_OPEN::Status	PXE status code + * @ret s_PXENV_TFTP_OPEN::PacketSize	Negotiated blksize + * @err #PXENV_STATUS_TFTP_INVALID_PACKET_SIZE Requested blksize too small + * + * Opens a TFTP connection for downloading a file a block at a time + * using pxenv_tftp_read(). + * + * If s_PXENV_TFTP_OPEN::GatewayIPAddress is 0.0.0.0, normal IP + * routing will take place.  See the relevant + * @ref pxe_routing "implementation note" for more details. + * + * On x86, you must set the s_PXE::StatusCallout field to a nonzero + * value before calling this function in protected mode.  You cannot + * call this function with a 32-bit stack segment.  (See the relevant + * @ref pxe_x86_pmode16 "implementation note" for more details.) + *  + * @note According to the PXE specification version 2.1, this call + * "opens a file for reading/writing", though how writing is to be + * achieved without the existence of an API call %pxenv_tftp_write() + * is not made clear. + * + * @note Despite the existence of the numerous statements within the + * PXE specification of the form "...if a TFTP/MTFTP or UDP connection + * is active...", you cannot use pxenv_tftp_open() and + * pxenv_tftp_read() to read a file via MTFTP; only via plain old + * TFTP.  If you want to use MTFTP, use pxenv_tftp_read_file() + * instead.  Astute readers will note that, since + * pxenv_tftp_read_file() is an atomic operation from the point of + * view of the PXE API, it is conceptually impossible to issue any + * other PXE API call "if an MTFTP connection is active". + */ +static PXENV_EXIT_t pxenv_tftp_open ( struct s_PXENV_TFTP_OPEN *tftp_open ) { +	int rc; + +	DBG ( "PXENV_TFTP_OPEN" ); + +	/* Guard against callers that fail to close before re-opening */ +	pxe_tftp_close ( &pxe_tftp, 0 ); + +	/* Open connection */ +	if ( ( rc = pxe_tftp_open ( tftp_open->ServerIPAddress, +				    tftp_open->TFTPPort, +				    tftp_open->FileName, +				    tftp_open->PacketSize, +				    0) ) != 0 ) { +		tftp_open->Status = PXENV_STATUS ( rc ); +		return PXENV_EXIT_FAILURE; +	} + +	/* Wait for OACK to arrive so that we have the block size */ +	while ( ( ( rc = pxe_tftp.rc ) == -EINPROGRESS ) && +		( pxe_tftp.max_offset == 0 ) ) { +		step(); +	} +	pxe_tftp.blksize = xfer_window ( &pxe_tftp.xfer ); +	tftp_open->PacketSize = pxe_tftp.blksize; +	DBG ( " blksize=%d", tftp_open->PacketSize ); + +	/* EINPROGRESS is normal; we don't wait for the whole transfer */ +	if ( rc == -EINPROGRESS ) +		rc = 0; + +	tftp_open->Status = PXENV_STATUS ( rc ); +	return ( rc ? PXENV_EXIT_FAILURE : PXENV_EXIT_SUCCESS ); +} + +/** + * TFTP CLOSE + * + * @v tftp_close			Pointer to a struct s_PXENV_TFTP_CLOSE + * @ret #PXENV_EXIT_SUCCESS		File was closed successfully + * @ret #PXENV_EXIT_FAILURE		File was not closed + * @ret s_PXENV_TFTP_CLOSE::Status	PXE status code + * @err None				- + * + * Close a connection previously opened with pxenv_tftp_open().  You + * must have previously opened a connection with pxenv_tftp_open(). + * + * On x86, you must set the s_PXE::StatusCallout field to a nonzero + * value before calling this function in protected mode.  You cannot + * call this function with a 32-bit stack segment.  (See the relevant + * @ref pxe_x86_pmode16 "implementation note" for more details.) + */ +static PXENV_EXIT_t pxenv_tftp_close ( struct s_PXENV_TFTP_CLOSE *tftp_close ) { +	DBG ( "PXENV_TFTP_CLOSE" ); + +	pxe_tftp_close ( &pxe_tftp, 0 ); +	tftp_close->Status = PXENV_STATUS_SUCCESS; +	return PXENV_EXIT_SUCCESS; +} + +/** + * TFTP READ + * + * @v tftp_read				Pointer to a struct s_PXENV_TFTP_READ + * @v s_PXENV_TFTP_READ::Buffer		Address of data buffer + * @ret #PXENV_EXIT_SUCCESS		Data was read successfully + * @ret #PXENV_EXIT_FAILURE		Data was not read + * @ret s_PXENV_TFTP_READ::Status	PXE status code + * @ret s_PXENV_TFTP_READ::PacketNumber	TFTP packet number + * @ret s_PXENV_TFTP_READ::BufferSize	Length of data written into buffer + * + * Reads a single packet from a connection previously opened with + * pxenv_tftp_open() into the data buffer pointed to by + * s_PXENV_TFTP_READ::Buffer.  You must have previously opened a + * connection with pxenv_tftp_open().  The data written into + * s_PXENV_TFTP_READ::Buffer is just the file data; the various + * network headers have already been removed. + * + * The buffer must be large enough to contain a packet of the size + * negotiated via the s_PXENV_TFTP_OPEN::PacketSize field in the + * pxenv_tftp_open() call.  It is worth noting that the PXE + * specification does @b not require the caller to fill in + * s_PXENV_TFTP_READ::BufferSize before calling pxenv_tftp_read(), so + * the PXE stack is free to ignore whatever value the caller might + * place there and just assume that the buffer is large enough.  That + * said, it may be worth the caller always filling in + * s_PXENV_TFTP_READ::BufferSize to guard against PXE stacks that + * mistake it for an input parameter. + * + * The length of the TFTP data packet will be returned via + * s_PXENV_TFTP_READ::BufferSize.  If this length is less than the + * blksize negotiated via s_PXENV_TFTP_OPEN::PacketSize in the call to + * pxenv_tftp_open(), this indicates that the block is the last block + * in the file.  Note that zero is a valid length for + * s_PXENV_TFTP_READ::BufferSize, and will occur when the length of + * the file is a multiple of the blksize. + * + * The PXE specification doesn't actually state that calls to + * pxenv_tftp_read() will return the data packets in strict sequential + * order, though most PXE stacks will probably do so.  The sequence + * number of the packet will be returned in + * s_PXENV_TFTP_READ::PacketNumber.  The first packet in the file has + * a sequence number of one, not zero. + * + * To guard against flawed PXE stacks, the caller should probably set + * s_PXENV_TFTP_READ::PacketNumber to one less than the expected + * returned value (i.e. set it to zero for the first call to + * pxenv_tftp_read() and then re-use the returned s_PXENV_TFTP_READ + * parameter block for subsequent calls without modifying + * s_PXENV_TFTP_READ::PacketNumber between calls).  The caller should + * also guard against potential problems caused by flawed + * implementations returning the occasional duplicate packet, by + * checking that the value returned in s_PXENV_TFTP_READ::PacketNumber + * is as expected (i.e. one greater than that returned from the + * previous call to pxenv_tftp_read()). + * + * On x86, you must set the s_PXE::StatusCallout field to a nonzero + * value before calling this function in protected mode.  You cannot + * call this function with a 32-bit stack segment.  (See the relevant + * @ref pxe_x86_pmode16 "implementation note" for more details.) + */ +static PXENV_EXIT_t pxenv_tftp_read ( struct s_PXENV_TFTP_READ *tftp_read ) { +	int rc; + +	DBG ( "PXENV_TFTP_READ to %04x:%04x", +	      tftp_read->Buffer.segment, tftp_read->Buffer.offset ); + +	/* Read single block into buffer */ +	pxe_tftp.buffer = real_to_user ( tftp_read->Buffer.segment, +					 tftp_read->Buffer.offset ); +	pxe_tftp.size = pxe_tftp.blksize; +	pxe_tftp.start = pxe_tftp.offset; +	while ( ( ( rc = pxe_tftp.rc ) == -EINPROGRESS ) && +		( pxe_tftp.offset == pxe_tftp.start ) ) +		step(); +	pxe_tftp.buffer = UNULL; +	tftp_read->BufferSize = ( pxe_tftp.offset - pxe_tftp.start ); +	tftp_read->PacketNumber = ++pxe_tftp.blkidx; + +	/* EINPROGRESS is normal if we haven't reached EOF yet */ +	if ( rc == -EINPROGRESS ) +		rc = 0; + +	tftp_read->Status = PXENV_STATUS ( rc ); +	return ( rc ? PXENV_EXIT_FAILURE : PXENV_EXIT_SUCCESS ); +} + +/** + * TFTP/MTFTP read file + * + * @v tftp_read_file		     Pointer to a struct s_PXENV_TFTP_READ_FILE + * @v s_PXENV_TFTP_READ_FILE::FileName		File name + * @v s_PXENV_TFTP_READ_FILE::BufferSize 	Size of the receive buffer + * @v s_PXENV_TFTP_READ_FILE::Buffer		Address of the receive buffer + * @v s_PXENV_TFTP_READ_FILE::ServerIPAddress	TFTP server IP address + * @v s_PXENV_TFTP_READ_FILE::GatewayIPAddress	Relay agent IP address + * @v s_PXENV_TFTP_READ_FILE::McastIPAddress	File's multicast IP address + * @v s_PXENV_TFTP_READ_FILE::TFTPClntPort	Client multicast UDP port + * @v s_PXENV_TFTP_READ_FILE::TFTPSrvPort	Server multicast UDP port + * @v s_PXENV_TFTP_READ_FILE::TFTPOpenTimeOut	Time to wait for first packet + * @v s_PXENV_TFTP_READ_FILE::TFTPReopenDelay	MTFTP inactivity timeout + * @ret #PXENV_EXIT_SUCCESS			File downloaded successfully + * @ret #PXENV_EXIT_FAILURE			File not downloaded + * @ret s_PXENV_TFTP_READ_FILE::Status		PXE status code + * @ret s_PXENV_TFTP_READ_FILE::BufferSize	Length of downloaded file + * + * Downloads an entire file via either TFTP or MTFTP into the buffer + * pointed to by s_PXENV_TFTP_READ_FILE::Buffer. + * + * The PXE specification does not make it clear how the caller + * requests that MTFTP be used rather than TFTP (or vice versa).  One + * reasonable guess is that setting + * s_PXENV_TFTP_READ_FILE::McastIPAddress to 0.0.0.0 would cause TFTP + * to be used instead of MTFTP, though it is conceivable that some PXE + * stacks would interpret that as "use the DHCP-provided multicast IP + * address" instead.  Some PXE stacks will not implement MTFTP at all, + * and will always use TFTP. + * + * It is not specified whether or not + * s_PXENV_TFTP_READ_FILE::TFTPSrvPort will be used as the TFTP server + * port for TFTP (rather than MTFTP) downloads.  Callers should assume + * that the only way to access a TFTP server on a non-standard port is + * to use pxenv_tftp_open() and pxenv_tftp_read(). + * + * If s_PXENV_TFTP_READ_FILE::GatewayIPAddress is 0.0.0.0, normal IP + * routing will take place.  See the relevant + * @ref pxe_routing "implementation note" for more details. + * + * It is interesting to note that s_PXENV_TFTP_READ_FILE::Buffer is an + * #ADDR32_t type, i.e. nominally a flat physical address.  Some PXE + * NBPs (e.g. NTLDR) are known to call pxenv_tftp_read_file() in real + * mode with s_PXENV_TFTP_READ_FILE::Buffer set to an address above + * 1MB.  This means that PXE stacks must be prepared to write to areas + * outside base memory.  Exactly how this is to be achieved is not + * specified, though using INT 15,87 is as close to a standard method + * as any, and should probably be used.  Switching to protected-mode + * in order to access high memory will fail if pxenv_tftp_read_file() + * is called in V86 mode; it is reasonably to expect that a V86 + * monitor would intercept the relatively well-defined INT 15,87 if it + * wants the PXE stack to be able to write to high memory. + * + * Things get even more interesting if pxenv_tftp_read_file() is + * called in protected mode, because there is then absolutely no way + * for the PXE stack to write to an absolute physical address.  You + * can't even get around the problem by creating a special "access + * everything" segment in the s_PXE data structure, because the + * #SEGDESC_t descriptors are limited to 64kB in size. + * + * Previous versions of the PXE specification (e.g. WfM 1.1a) provide + * a separate API call, %pxenv_tftp_read_file_pmode(), specifically to + * work around this problem.  The s_PXENV_TFTP_READ_FILE_PMODE + * parameter block splits s_PXENV_TFTP_READ_FILE::Buffer into + * s_PXENV_TFTP_READ_FILE_PMODE::BufferSelector and + * s_PXENV_TFTP_READ_FILE_PMODE::BufferOffset, i.e. it provides a + * protected-mode segment:offset address for the data buffer.  This + * API call is no longer present in version 2.1 of the PXE + * specification. + * + * Etherboot makes the assumption that s_PXENV_TFTP_READ_FILE::Buffer + * is an offset relative to the caller's data segment, when + * pxenv_tftp_read_file() is called in protected mode. + * + * On x86, you must set the s_PXE::StatusCallout field to a nonzero + * value before calling this function in protected mode.  You cannot + * call this function with a 32-bit stack segment.  (See the relevant + * @ref pxe_x86_pmode16 "implementation note" for more details.) + */ +PXENV_EXIT_t pxenv_tftp_read_file ( struct s_PXENV_TFTP_READ_FILE +				    *tftp_read_file ) { +	int rc; + +	DBG ( "PXENV_TFTP_READ_FILE to %08x+%x", tftp_read_file->Buffer, +	      tftp_read_file->BufferSize ); + +	/* Open TFTP file */ +	if ( ( rc = pxe_tftp_open ( tftp_read_file->ServerIPAddress, 0, +				    tftp_read_file->FileName, 0, 0 ) ) != 0 ) { +		tftp_read_file->Status = PXENV_STATUS ( rc ); +		return PXENV_EXIT_FAILURE; +	} + +	/* Read entire file */ +	pxe_tftp.buffer = phys_to_user ( tftp_read_file->Buffer ); +	pxe_tftp.size = tftp_read_file->BufferSize; +	while ( ( rc = pxe_tftp.rc ) == -EINPROGRESS ) +		step(); +	pxe_tftp.buffer = UNULL; +	tftp_read_file->BufferSize = pxe_tftp.max_offset; + +	/* Close TFTP file */ +	pxe_tftp_close ( &pxe_tftp, rc ); + +	tftp_read_file->Status = PXENV_STATUS ( rc ); +	return ( rc ? PXENV_EXIT_FAILURE : PXENV_EXIT_SUCCESS ); +} + +/** + * TFTP GET FILE SIZE + * + * @v tftp_get_fsize		     Pointer to a struct s_PXENV_TFTP_GET_FSIZE + * @v s_PXENV_TFTP_GET_FSIZE::ServerIPAddress	TFTP server IP address + * @v s_PXENV_TFTP_GET_FSIZE::GatewayIPAddress	Relay agent IP address + * @v s_PXENV_TFTP_GET_FSIZE::FileName	File name + * @ret #PXENV_EXIT_SUCCESS		File size was determined successfully + * @ret #PXENV_EXIT_FAILURE		File size was not determined + * @ret s_PXENV_TFTP_GET_FSIZE::Status	PXE status code + * @ret s_PXENV_TFTP_GET_FSIZE::FileSize	File size + * + * Determine the size of a file on a TFTP server.  This uses the + * "tsize" TFTP option, and so will not work with a TFTP server that + * does not support TFTP options, or that does not support the "tsize" + * option. + * + * The PXE specification states that this API call will @b not open a + * TFTP connection for subsequent use with pxenv_tftp_read().  (This + * is somewhat daft, since the only way to obtain the file size via + * the "tsize" option involves issuing a TFTP open request, but that's + * life.) + * + * You cannot call pxenv_tftp_get_fsize() while a TFTP or UDP + * connection is open. + * + * If s_PXENV_TFTP_GET_FSIZE::GatewayIPAddress is 0.0.0.0, normal IP + * routing will take place.  See the relevant + * @ref pxe_routing "implementation note" for more details. + * + * On x86, you must set the s_PXE::StatusCallout field to a nonzero + * value before calling this function in protected mode.  You cannot + * call this function with a 32-bit stack segment.  (See the relevant + * @ref pxe_x86_pmode16 "implementation note" for more details.) + *  + * @note There is no way to specify the TFTP server port with this API + * call.  Though you can open a file using a non-standard TFTP server + * port (via s_PXENV_TFTP_OPEN::TFTPPort or, potentially, + * s_PXENV_TFTP_READ_FILE::TFTPSrvPort), you can only get the size of + * a file from a TFTP server listening on the standard TFTP port. + * "Consistency" is not a word in Intel's vocabulary. + */ +static PXENV_EXIT_t pxenv_tftp_get_fsize ( struct s_PXENV_TFTP_GET_FSIZE +					   *tftp_get_fsize ) { +	int rc; + +	DBG ( "PXENV_TFTP_GET_FSIZE" ); + +	/* Open TFTP file */ +	if ( ( rc = pxe_tftp_open ( tftp_get_fsize->ServerIPAddress, 0, +				    tftp_get_fsize->FileName, 0, 1 ) ) != 0 ) { +		tftp_get_fsize->Status = PXENV_STATUS ( rc ); +		return PXENV_EXIT_FAILURE; +	} + +	/* Wait for initial seek to arrive, and record size */ +	while ( ( ( rc = pxe_tftp.rc ) == -EINPROGRESS ) && +		( pxe_tftp.max_offset == 0 ) ) { +		step(); +	} +	tftp_get_fsize->FileSize = pxe_tftp.max_offset; +	DBG ( " fsize=%d", tftp_get_fsize->FileSize ); + +	/* EINPROGRESS is normal; we don't wait for the whole transfer */ +	if ( rc == -EINPROGRESS ) +		rc = 0; + +	/* Close TFTP file */ +	pxe_tftp_close ( &pxe_tftp, rc ); + +	tftp_get_fsize->Status = PXENV_STATUS ( rc ); +	return ( rc ? PXENV_EXIT_FAILURE : PXENV_EXIT_SUCCESS ); +} + +/** PXE TFTP API */ +struct pxe_api_call pxe_tftp_api[] __pxe_api_call = { +	PXE_API_CALL ( PXENV_TFTP_OPEN, pxenv_tftp_open, +		       struct s_PXENV_TFTP_OPEN ), +	PXE_API_CALL ( PXENV_TFTP_CLOSE, pxenv_tftp_close, +		       struct s_PXENV_TFTP_CLOSE ), +	PXE_API_CALL ( PXENV_TFTP_READ, pxenv_tftp_read, +		       struct s_PXENV_TFTP_READ ), +	PXE_API_CALL ( PXENV_TFTP_READ_FILE, pxenv_tftp_read_file, +		       struct s_PXENV_TFTP_READ_FILE ), +	PXE_API_CALL ( PXENV_TFTP_GET_FSIZE, pxenv_tftp_get_fsize, +		       struct s_PXENV_TFTP_GET_FSIZE ), +}; diff --git a/roms/ipxe/src/arch/i386/interface/pxe/pxe_udp.c b/roms/ipxe/src/arch/i386/interface/pxe/pxe_udp.c new file mode 100644 index 00000000..32bc39c8 --- /dev/null +++ b/roms/ipxe/src/arch/i386/interface/pxe/pxe_udp.c @@ -0,0 +1,422 @@ +/** @file + * + * PXE UDP API + * + */ + +#include <string.h> +#include <byteswap.h> +#include <ipxe/iobuf.h> +#include <ipxe/xfer.h> +#include <ipxe/udp.h> +#include <ipxe/uaccess.h> +#include <ipxe/process.h> +#include <pxe.h> + +/* + * Copyright (C) 2004 Michael Brown <mbrown@fensystems.co.uk>. + * + * This program is free software; you can redistribute it and/or + * modify it under the terms of the GNU General Public License as + * published by the Free Software Foundation; either version 2 of the + * License, or any later version. + * + * 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. + * + * You should have received a copy of the GNU General Public License + * along with this program; if not, write to the Free Software + * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA + * 02110-1301, USA. + */ + +FILE_LICENCE ( GPL2_OR_LATER ); + +/** A PXE UDP connection */ +struct pxe_udp_connection { +	/** Data transfer interface to UDP stack */ +	struct interface xfer; +	/** Local address */ +	struct sockaddr_in local; +	/** Current PXENV_UDP_READ parameter block */ +	struct s_PXENV_UDP_READ *pxenv_udp_read; +}; + +/** + * Receive PXE UDP data + * + * @v pxe_udp			PXE UDP connection + * @v iobuf			I/O buffer + * @v meta			Data transfer metadata + * @ret rc			Return status code + * + * Receives a packet as part of the current pxenv_udp_read() + * operation. + */ +static int pxe_udp_deliver ( struct pxe_udp_connection *pxe_udp, +			     struct io_buffer *iobuf, +			     struct xfer_metadata *meta ) { +	struct s_PXENV_UDP_READ *pxenv_udp_read = pxe_udp->pxenv_udp_read; +	struct sockaddr_in *sin_src; +	struct sockaddr_in *sin_dest; +	userptr_t buffer; +	size_t len; +	int rc = 0; + +	if ( ! pxenv_udp_read ) { +		DBG ( "PXE discarded UDP packet\n" ); +		rc = -ENOBUFS; +		goto done; +	} + +	/* Copy packet to buffer and record length */ +	buffer = real_to_user ( pxenv_udp_read->buffer.segment, +				pxenv_udp_read->buffer.offset ); +	len = iob_len ( iobuf ); +	if ( len > pxenv_udp_read->buffer_size ) +		len = pxenv_udp_read->buffer_size; +	copy_to_user ( buffer, 0, iobuf->data, len ); +	pxenv_udp_read->buffer_size = len; + +	/* Fill in source/dest information */ +	assert ( meta ); +	sin_src = ( struct sockaddr_in * ) meta->src; +	assert ( sin_src ); +	assert ( sin_src->sin_family == AF_INET ); +	pxenv_udp_read->src_ip = sin_src->sin_addr.s_addr; +	pxenv_udp_read->s_port = sin_src->sin_port; +	sin_dest = ( struct sockaddr_in * ) meta->dest; +	assert ( sin_dest ); +	assert ( sin_dest->sin_family == AF_INET ); +	pxenv_udp_read->dest_ip = sin_dest->sin_addr.s_addr; +	pxenv_udp_read->d_port = sin_dest->sin_port; + +	/* Mark as received */ +	pxe_udp->pxenv_udp_read = NULL; + + done: +	free_iob ( iobuf ); +	return rc; +} + +/** PXE UDP data transfer interface operations */ +static struct interface_operation pxe_udp_xfer_operations[] = { +	INTF_OP ( xfer_deliver, struct pxe_udp_connection *, pxe_udp_deliver ), +}; + +/** PXE UDP data transfer interface descriptor */ +static struct interface_descriptor pxe_udp_xfer_desc = +	INTF_DESC ( struct pxe_udp_connection, xfer, pxe_udp_xfer_operations ); + +/** The PXE UDP connection */ +static struct pxe_udp_connection pxe_udp = { +	.xfer = INTF_INIT ( pxe_udp_xfer_desc ), +	.local = { +		.sin_family = AF_INET, +	}, +}; + +/** + * UDP OPEN + * + * @v pxenv_udp_open			Pointer to a struct s_PXENV_UDP_OPEN + * @v s_PXENV_UDP_OPEN::src_ip		IP address of this station, or 0.0.0.0 + * @ret #PXENV_EXIT_SUCCESS		Always + * @ret s_PXENV_UDP_OPEN::Status	PXE status code + * @err #PXENV_STATUS_UDP_OPEN		UDP connection already open + * @err #PXENV_STATUS_OUT_OF_RESOURCES	Could not open connection + * + * Prepares the PXE stack for communication using pxenv_udp_write() + * and pxenv_udp_read(). + * + * The IP address supplied in s_PXENV_UDP_OPEN::src_ip will be + * recorded and used as the local station's IP address for all further + * communication, including communication by means other than + * pxenv_udp_write() and pxenv_udp_read().  (If + * s_PXENV_UDP_OPEN::src_ip is 0.0.0.0, the local station's IP address + * will remain unchanged.) + * + * You can only have one open UDP connection at a time.  This is not a + * meaningful restriction, since pxenv_udp_write() and + * pxenv_udp_read() allow you to specify arbitrary local and remote + * ports and an arbitrary remote address for each packet.  According + * to the PXE specifiation, you cannot have a UDP connection open at + * the same time as a TFTP connection; this restriction does not apply + * to Etherboot. + * + * On x86, you must set the s_PXE::StatusCallout field to a nonzero + * value before calling this function in protected mode.  You cannot + * call this function with a 32-bit stack segment.  (See the relevant + * @ref pxe_x86_pmode16 "implementation note" for more details.) + * + * @note The PXE specification does not make it clear whether the IP + * address supplied in s_PXENV_UDP_OPEN::src_ip should be used only + * for this UDP connection, or retained for all future communication. + * The latter seems more consistent with typical PXE stack behaviour. + * + * @note Etherboot currently ignores the s_PXENV_UDP_OPEN::src_ip + * parameter. + * + */ +static PXENV_EXIT_t pxenv_udp_open ( struct s_PXENV_UDP_OPEN *pxenv_udp_open ) { +	int rc; + +	DBG ( "PXENV_UDP_OPEN" ); + +	/* Record source IP address */ +	pxe_udp.local.sin_addr.s_addr = pxenv_udp_open->src_ip; +	DBG ( " %s\n", inet_ntoa ( pxe_udp.local.sin_addr ) ); + +	/* Open promiscuous UDP connection */ +	intf_restart ( &pxe_udp.xfer, 0 ); +	if ( ( rc = udp_open_promisc ( &pxe_udp.xfer ) ) != 0 ) { +		DBG ( "PXENV_UDP_OPEN could not open promiscuous socket: %s\n", +		      strerror ( rc ) ); +		pxenv_udp_open->Status = PXENV_STATUS ( rc ); +		return PXENV_EXIT_FAILURE; +	} + +	pxenv_udp_open->Status = PXENV_STATUS_SUCCESS; +	return PXENV_EXIT_SUCCESS; +} + +/** + * UDP CLOSE + * + * @v pxenv_udp_close			Pointer to a struct s_PXENV_UDP_CLOSE + * @ret #PXENV_EXIT_SUCCESS		Always + * @ret s_PXENV_UDP_CLOSE::Status	PXE status code + * @err None				- + * + * Closes a UDP connection opened with pxenv_udp_open(). + * + * You can only have one open UDP connection at a time.  You cannot + * have a UDP connection open at the same time as a TFTP connection. + * You cannot use pxenv_udp_close() to close a TFTP connection; use + * pxenv_tftp_close() instead. + * + * On x86, you must set the s_PXE::StatusCallout field to a nonzero + * value before calling this function in protected mode.  You cannot + * call this function with a 32-bit stack segment.  (See the relevant + * @ref pxe_x86_pmode16 "implementation note" for more details.) + * + */ +static PXENV_EXIT_t +pxenv_udp_close ( struct s_PXENV_UDP_CLOSE *pxenv_udp_close ) { +	DBG ( "PXENV_UDP_CLOSE\n" ); + +	/* Close UDP connection */ +	intf_restart ( &pxe_udp.xfer, 0 ); + +	pxenv_udp_close->Status = PXENV_STATUS_SUCCESS; +	return PXENV_EXIT_SUCCESS; +} + +/** + * UDP WRITE + * + * @v pxenv_udp_write			Pointer to a struct s_PXENV_UDP_WRITE + * @v s_PXENV_UDP_WRITE::ip		Destination IP address + * @v s_PXENV_UDP_WRITE::gw		Relay agent IP address, or 0.0.0.0 + * @v s_PXENV_UDP_WRITE::src_port	Source UDP port, or 0 + * @v s_PXENV_UDP_WRITE::dst_port	Destination UDP port + * @v s_PXENV_UDP_WRITE::buffer_size	Length of the UDP payload + * @v s_PXENV_UDP_WRITE::buffer		Address of the UDP payload + * @ret #PXENV_EXIT_SUCCESS		Packet was transmitted successfully + * @ret #PXENV_EXIT_FAILURE		Packet could not be transmitted + * @ret s_PXENV_UDP_WRITE::Status	PXE status code + * @err #PXENV_STATUS_UDP_CLOSED	UDP connection is not open + * @err #PXENV_STATUS_UNDI_TRANSMIT_ERROR Could not transmit packet + * + * Transmits a single UDP packet.  A valid IP and UDP header will be + * prepended to the payload in s_PXENV_UDP_WRITE::buffer; the buffer + * should not contain precomputed IP and UDP headers, nor should it + * contain space allocated for these headers.  The first byte of the + * buffer will be transmitted as the first byte following the UDP + * header. + * + * If s_PXENV_UDP_WRITE::gw is 0.0.0.0, normal IP routing will take + * place.  See the relevant @ref pxe_routing "implementation note" for + * more details. + * + * If s_PXENV_UDP_WRITE::src_port is 0, port 2069 will be used. + * + * You must have opened a UDP connection with pxenv_udp_open() before + * calling pxenv_udp_write(). + * + * On x86, you must set the s_PXE::StatusCallout field to a nonzero + * value before calling this function in protected mode.  You cannot + * call this function with a 32-bit stack segment.  (See the relevant + * @ref pxe_x86_pmode16 "implementation note" for more details.) + * + * @note Etherboot currently ignores the s_PXENV_UDP_WRITE::gw + * parameter. + * + */ +static PXENV_EXIT_t +pxenv_udp_write ( struct s_PXENV_UDP_WRITE *pxenv_udp_write ) { +	struct sockaddr_in dest; +	struct xfer_metadata meta = { +		.src = ( struct sockaddr * ) &pxe_udp.local, +		.dest = ( struct sockaddr * ) &dest, +		.netdev = pxe_netdev, +	}; +	size_t len; +	struct io_buffer *iobuf; +	userptr_t buffer; +	int rc; + +	DBG ( "PXENV_UDP_WRITE" ); + +	/* Construct destination socket address */ +	memset ( &dest, 0, sizeof ( dest ) ); +	dest.sin_family = AF_INET; +	dest.sin_addr.s_addr = pxenv_udp_write->ip; +	dest.sin_port = pxenv_udp_write->dst_port; + +	/* Set local (source) port.  PXE spec says source port is 2069 +	 * if not specified.  Really, this ought to be set at UDP open +	 * time but hey, we didn't design this API. +	 */ +	pxe_udp.local.sin_port = pxenv_udp_write->src_port; +	if ( ! pxe_udp.local.sin_port ) +		pxe_udp.local.sin_port = htons ( 2069 ); + +	/* FIXME: we ignore the gateway specified, since we're +	 * confident of being able to do our own routing.  We should +	 * probably allow for multiple gateways. +	 */ + +	/* Allocate and fill data buffer */ +	len = pxenv_udp_write->buffer_size; +	iobuf = xfer_alloc_iob ( &pxe_udp.xfer, len ); +	if ( ! iobuf ) { +		DBG ( " out of memory\n" ); +		pxenv_udp_write->Status = PXENV_STATUS_OUT_OF_RESOURCES; +		return PXENV_EXIT_FAILURE; +	} +	buffer = real_to_user ( pxenv_udp_write->buffer.segment, +				pxenv_udp_write->buffer.offset ); +	copy_from_user ( iob_put ( iobuf, len ), buffer, 0, len ); + +	DBG ( " %04x:%04x+%x %d->%s:%d\n", pxenv_udp_write->buffer.segment, +	      pxenv_udp_write->buffer.offset, pxenv_udp_write->buffer_size, +	      ntohs ( pxenv_udp_write->src_port ), +	      inet_ntoa ( dest.sin_addr ), +	      ntohs ( pxenv_udp_write->dst_port ) ); +	 +	/* Transmit packet */ +	if ( ( rc = xfer_deliver ( &pxe_udp.xfer, iobuf, &meta ) ) != 0 ) { +		DBG ( "PXENV_UDP_WRITE could not transmit: %s\n", +		      strerror ( rc ) ); +		pxenv_udp_write->Status = PXENV_STATUS ( rc ); +		return PXENV_EXIT_FAILURE; +	} + +	pxenv_udp_write->Status = PXENV_STATUS_SUCCESS; +	return PXENV_EXIT_SUCCESS; +} + +/** + * UDP READ + * + * @v pxenv_udp_read			Pointer to a struct s_PXENV_UDP_READ + * @v s_PXENV_UDP_READ::dest_ip		Destination IP address, or 0.0.0.0 + * @v s_PXENV_UDP_READ::d_port		Destination UDP port, or 0 + * @v s_PXENV_UDP_READ::buffer_size	Size of the UDP payload buffer + * @v s_PXENV_UDP_READ::buffer		Address of the UDP payload buffer + * @ret #PXENV_EXIT_SUCCESS		A packet has been received + * @ret #PXENV_EXIT_FAILURE		No packet has been received + * @ret s_PXENV_UDP_READ::Status	PXE status code + * @ret s_PXENV_UDP_READ::src_ip	Source IP address + * @ret s_PXENV_UDP_READ::dest_ip	Destination IP address + * @ret s_PXENV_UDP_READ::s_port	Source UDP port + * @ret s_PXENV_UDP_READ::d_port	Destination UDP port + * @ret s_PXENV_UDP_READ::buffer_size	Length of UDP payload + * @err #PXENV_STATUS_UDP_CLOSED	UDP connection is not open + * @err #PXENV_STATUS_FAILURE		No packet was ready to read + * + * Receive a single UDP packet.  This is a non-blocking call; if no + * packet is ready to read, the call will return instantly with + * s_PXENV_UDP_READ::Status==PXENV_STATUS_FAILURE. + * + * If s_PXENV_UDP_READ::dest_ip is 0.0.0.0, UDP packets addressed to + * any IP address will be accepted and may be returned to the caller. + * + * If s_PXENV_UDP_READ::d_port is 0, UDP packets addressed to any UDP + * port will be accepted and may be returned to the caller. + * + * You must have opened a UDP connection with pxenv_udp_open() before + * calling pxenv_udp_read(). + * + * On x86, you must set the s_PXE::StatusCallout field to a nonzero + * value before calling this function in protected mode.  You cannot + * call this function with a 32-bit stack segment.  (See the relevant + * @ref pxe_x86_pmode16 "implementation note" for more details.) + * + * @note The PXE specification (version 2.1) does not state that we + * should fill in s_PXENV_UDP_READ::dest_ip and + * s_PXENV_UDP_READ::d_port, but Microsoft Windows' NTLDR program + * expects us to do so, and will fail if we don't. + * + */ +static PXENV_EXIT_t pxenv_udp_read ( struct s_PXENV_UDP_READ *pxenv_udp_read ) { +	struct in_addr dest_ip_wanted = { .s_addr = pxenv_udp_read->dest_ip }; +	struct in_addr dest_ip; +	uint16_t d_port_wanted = pxenv_udp_read->d_port; +	uint16_t d_port; + +	/* Try receiving a packet */ +	pxe_udp.pxenv_udp_read = pxenv_udp_read; +	step(); +	if ( pxe_udp.pxenv_udp_read ) { +		/* No packet received */ +		DBG2 ( "PXENV_UDP_READ\n" ); +		pxe_udp.pxenv_udp_read = NULL; +		goto no_packet; +	} +	dest_ip.s_addr = pxenv_udp_read->dest_ip; +	d_port = pxenv_udp_read->d_port; +	DBG ( "PXENV_UDP_READ" ); + +	/* Filter on destination address and/or port */ +	if ( dest_ip_wanted.s_addr && +	     ( dest_ip_wanted.s_addr != dest_ip.s_addr ) ) { +		DBG ( " wrong IP %s", inet_ntoa ( dest_ip ) ); +		DBG ( " (wanted %s)\n", inet_ntoa ( dest_ip_wanted ) ); +		goto no_packet; +	} +	if ( d_port_wanted && ( d_port_wanted != d_port ) ) { +		DBG ( " wrong port %d", htons ( d_port ) ); +		DBG ( " (wanted %d)\n", htons ( d_port_wanted ) ); +		goto no_packet; +	} + +	DBG ( " %04x:%04x+%x %s:", pxenv_udp_read->buffer.segment, +	      pxenv_udp_read->buffer.offset, pxenv_udp_read->buffer_size, +	      inet_ntoa ( *( ( struct in_addr * ) &pxenv_udp_read->src_ip ) )); +	DBG ( "%d<-%s:%d\n",  ntohs ( pxenv_udp_read->s_port ), +	      inet_ntoa ( *( ( struct in_addr * ) &pxenv_udp_read->dest_ip ) ), +	      ntohs ( pxenv_udp_read->d_port ) ); + +	pxenv_udp_read->Status = PXENV_STATUS_SUCCESS; +	return PXENV_EXIT_SUCCESS; + + no_packet: +	pxenv_udp_read->Status = PXENV_STATUS_FAILURE; +	return PXENV_EXIT_FAILURE; +} + +/** PXE UDP API */ +struct pxe_api_call pxe_udp_api[] __pxe_api_call = { +	PXE_API_CALL ( PXENV_UDP_OPEN, pxenv_udp_open, +		       struct s_PXENV_UDP_OPEN ), +	PXE_API_CALL ( PXENV_UDP_CLOSE, pxenv_udp_close, +		       struct s_PXENV_UDP_CLOSE ), +	PXE_API_CALL ( PXENV_UDP_WRITE, pxenv_udp_write, +		       struct s_PXENV_UDP_WRITE ), +	PXE_API_CALL ( PXENV_UDP_READ, pxenv_udp_read, +		       struct s_PXENV_UDP_READ ), +}; diff --git a/roms/ipxe/src/arch/i386/interface/pxe/pxe_undi.c b/roms/ipxe/src/arch/i386/interface/pxe/pxe_undi.c new file mode 100644 index 00000000..29e586ed --- /dev/null +++ b/roms/ipxe/src/arch/i386/interface/pxe/pxe_undi.c @@ -0,0 +1,1080 @@ +/** @file + * + * PXE UNDI API + * + */ + +/* + * Copyright (C) 2004 Michael Brown <mbrown@fensystems.co.uk>. + * + * This program is free software; you can redistribute it and/or + * modify it under the terms of the GNU General Public License as + * published by the Free Software Foundation; either version 2 of the + * License, or any later version. + * + * 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. + * + * You should have received a copy of the GNU General Public License + * along with this program; if not, write to the Free Software + * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA + * 02110-1301, USA. + */ + +FILE_LICENCE ( GPL2_OR_LATER ); + +#include <stdint.h> +#include <stdio.h> +#include <string.h> +#include <byteswap.h> +#include <basemem_packet.h> +#include <ipxe/netdevice.h> +#include <ipxe/iobuf.h> +#include <ipxe/device.h> +#include <ipxe/pci.h> +#include <ipxe/if_ether.h> +#include <ipxe/ip.h> +#include <ipxe/arp.h> +#include <ipxe/rarp.h> +#include <ipxe/profile.h> +#include "pxe.h" + +/** + * Count of outstanding transmitted packets + * + * This is incremented each time PXENV_UNDI_TRANSMIT is called, and + * decremented each time that PXENV_UNDI_ISR is called with the TX + * queue empty, stopping when the count reaches zero.  This allows us + * to provide a pessimistic approximation of TX completion events to + * the PXE NBP simply by monitoring the netdev's TX queue. + */ +static int undi_tx_count = 0; + +struct net_device *pxe_netdev = NULL; + +/** Transmit profiler */ +static struct profiler undi_tx_profiler __profiler = { .name = "undi.tx" }; + +/** + * Set network device as current PXE network device + * + * @v netdev		Network device, or NULL + */ +void pxe_set_netdev ( struct net_device *netdev ) { + +	if ( pxe_netdev ) { +		netdev_rx_unfreeze ( pxe_netdev ); +		netdev_put ( pxe_netdev ); +	} + +	pxe_netdev = NULL; + +	if ( netdev ) +		pxe_netdev = netdev_get ( netdev ); +} + +/** + * Open PXE network device + * + * @ret rc		Return status code + */ +static int pxe_netdev_open ( void ) { +	int rc; + +	assert ( pxe_netdev != NULL ); + +	if ( ( rc = netdev_open ( pxe_netdev ) ) != 0 ) +		return rc; + +	netdev_rx_freeze ( pxe_netdev ); +	netdev_irq ( pxe_netdev, 1 ); + +	return 0; +} + +/** + * Close PXE network device + * + */ +static void pxe_netdev_close ( void ) { + +	assert ( pxe_netdev != NULL ); +	netdev_rx_unfreeze ( pxe_netdev ); +	netdev_irq ( pxe_netdev, 0 ); +	netdev_close ( pxe_netdev ); +	undi_tx_count = 0; +} + +/** + * Dump multicast address list + * + * @v mcast		PXE multicast address list + */ +static void pxe_dump_mcast_list ( struct s_PXENV_UNDI_MCAST_ADDRESS *mcast ) { +	struct ll_protocol *ll_protocol = pxe_netdev->ll_protocol; +	unsigned int i; + +	for ( i = 0 ; i < mcast->MCastAddrCount ; i++ ) { +		DBGC ( &pxe_netdev, " %s", +		       ll_protocol->ntoa ( mcast->McastAddr[i] ) ); +	} +} + +/* PXENV_UNDI_STARTUP + * + * Status: working + */ +static PXENV_EXIT_t +pxenv_undi_startup ( struct s_PXENV_UNDI_STARTUP *undi_startup ) { +	DBGC ( &pxe_netdev, "PXENV_UNDI_STARTUP\n" ); + +	/* Sanity check */ +	if ( ! pxe_netdev ) { +		DBGC ( &pxe_netdev, "PXENV_UNDI_STARTUP called with no " +		       "network device\n" ); +		undi_startup->Status = PXENV_STATUS_UNDI_INVALID_STATE; +		return PXENV_EXIT_FAILURE; +	} + +	undi_startup->Status = PXENV_STATUS_SUCCESS; +	return PXENV_EXIT_SUCCESS; +} + +/* PXENV_UNDI_CLEANUP + * + * Status: working + */ +static PXENV_EXIT_t +pxenv_undi_cleanup ( struct s_PXENV_UNDI_CLEANUP *undi_cleanup ) { +	DBGC ( &pxe_netdev, "PXENV_UNDI_CLEANUP\n" ); + +	/* Sanity check */ +	if ( ! pxe_netdev ) { +		DBGC ( &pxe_netdev, "PXENV_UNDI_CLEANUP called with no " +		       "network device\n" ); +		undi_cleanup->Status = PXENV_STATUS_UNDI_INVALID_STATE; +		return PXENV_EXIT_FAILURE; +	} + +	/* Close network device */ +	pxe_netdev_close(); + +	undi_cleanup->Status = PXENV_STATUS_SUCCESS; +	return PXENV_EXIT_SUCCESS; +} + +/* PXENV_UNDI_INITIALIZE + * + * Status: working + */ +static PXENV_EXIT_t +pxenv_undi_initialize ( struct s_PXENV_UNDI_INITIALIZE *undi_initialize ) { +	DBGC ( &pxe_netdev, "PXENV_UNDI_INITIALIZE protocolini %08x\n", +	       undi_initialize->ProtocolIni ); + +	/* Sanity check */ +	if ( ! pxe_netdev ) { +		DBGC ( &pxe_netdev, "PXENV_UNDI_INITIALIZE called with no " +		       "network device\n" ); +		undi_initialize->Status = PXENV_STATUS_UNDI_INVALID_STATE; +		return PXENV_EXIT_FAILURE; +	} + +	undi_initialize->Status = PXENV_STATUS_SUCCESS; +	return PXENV_EXIT_SUCCESS; +} + +/* PXENV_UNDI_RESET_ADAPTER + * + * Status: working + */ +static PXENV_EXIT_t +pxenv_undi_reset_adapter ( struct s_PXENV_UNDI_RESET *undi_reset_adapter ) { +	int rc; + +	DBGC ( &pxe_netdev, "PXENV_UNDI_RESET_ADAPTER" ); +	pxe_dump_mcast_list ( &undi_reset_adapter->R_Mcast_Buf ); +	DBGC ( &pxe_netdev, "\n" ); + +	/* Sanity check */ +	if ( ! pxe_netdev ) { +		DBGC ( &pxe_netdev, "PXENV_UNDI_RESET_ADAPTER called with no " +		       "network device\n" ); +		undi_reset_adapter->Status = PXENV_STATUS_UNDI_INVALID_STATE; +		return PXENV_EXIT_FAILURE; +	} + +	/* Close and reopen network device */ +	pxe_netdev_close(); +	if ( ( rc = pxe_netdev_open() ) != 0 ) { +		DBGC ( &pxe_netdev, "PXENV_UNDI_RESET_ADAPTER could not " +		       "reopen %s: %s\n", pxe_netdev->name, strerror ( rc ) ); +		undi_reset_adapter->Status = PXENV_STATUS ( rc ); +		return PXENV_EXIT_FAILURE; +	} + +	undi_reset_adapter->Status = PXENV_STATUS_SUCCESS; +	return PXENV_EXIT_SUCCESS; +} + +/* PXENV_UNDI_SHUTDOWN + * + * Status: working + */ +static PXENV_EXIT_t +pxenv_undi_shutdown ( struct s_PXENV_UNDI_SHUTDOWN *undi_shutdown ) { +	DBGC ( &pxe_netdev, "PXENV_UNDI_SHUTDOWN\n" ); + +	/* Sanity check */ +	if ( ! pxe_netdev ) { +		DBGC ( &pxe_netdev, "PXENV_UNDI_SHUTDOWN called with no " +		       "network device\n" ); +		undi_shutdown->Status = PXENV_STATUS_UNDI_INVALID_STATE; +		return PXENV_EXIT_FAILURE; +	} + +	/* Close network device */ +	pxe_netdev_close(); + +	undi_shutdown->Status = PXENV_STATUS_SUCCESS; +	return PXENV_EXIT_SUCCESS; +} + +/* PXENV_UNDI_OPEN + * + * Status: working + */ +static PXENV_EXIT_t pxenv_undi_open ( struct s_PXENV_UNDI_OPEN *undi_open ) { +	int rc; + +	DBGC ( &pxe_netdev, "PXENV_UNDI_OPEN flag %04x filter %04x", +	       undi_open->OpenFlag, undi_open->PktFilter ); +	pxe_dump_mcast_list ( &undi_open->R_Mcast_Buf ); +	DBGC ( &pxe_netdev, "\n" ); + +	/* Sanity check */ +	if ( ! pxe_netdev ) { +		DBGC ( &pxe_netdev, "PXENV_UNDI_OPEN called with no " +		       "network device\n" ); +		undi_open->Status = PXENV_STATUS_UNDI_INVALID_STATE; +		return PXENV_EXIT_FAILURE; +	} + +	/* Open network device */ +	if ( ( rc = pxe_netdev_open() ) != 0 ) { +		DBGC ( &pxe_netdev, "PXENV_UNDI_OPEN could not open %s: %s\n", +		       pxe_netdev->name, strerror ( rc ) ); +		undi_open->Status = PXENV_STATUS ( rc ); +		return PXENV_EXIT_FAILURE; +	} + +	undi_open->Status = PXENV_STATUS_SUCCESS; +	return PXENV_EXIT_SUCCESS; +} + +/* PXENV_UNDI_CLOSE + * + * Status: working + */ +static PXENV_EXIT_t pxenv_undi_close ( struct s_PXENV_UNDI_CLOSE *undi_close ) { +	DBGC ( &pxe_netdev, "PXENV_UNDI_CLOSE\n" ); + +	/* Sanity check */ +	if ( ! pxe_netdev ) { +		DBGC ( &pxe_netdev, "PXENV_UNDI_CLOSE called with no " +		       "network device\n" ); +		undi_close->Status = PXENV_STATUS_UNDI_INVALID_STATE; +		return PXENV_EXIT_FAILURE; +	} + +	/* Close network device */ +	pxe_netdev_close(); + +	undi_close->Status = PXENV_STATUS_SUCCESS; +	return PXENV_EXIT_SUCCESS; +} + +/* PXENV_UNDI_TRANSMIT + * + * Status: working + */ +static PXENV_EXIT_t +pxenv_undi_transmit ( struct s_PXENV_UNDI_TRANSMIT *undi_transmit ) { +	struct s_PXENV_UNDI_TBD tbd; +	struct DataBlk *datablk; +	struct io_buffer *iobuf; +	struct net_protocol *net_protocol; +	struct ll_protocol *ll_protocol; +	char destaddr[MAX_LL_ADDR_LEN]; +	const void *ll_dest; +	size_t len; +	unsigned int i; +	int rc; + +	/* Start profiling */ +	profile_start ( &undi_tx_profiler ); + +	/* Sanity check */ +	if ( ! pxe_netdev ) { +		DBGC ( &pxe_netdev, "PXENV_UNDI_TRANSMIT called with no " +		       "network device\n" ); +		undi_transmit->Status = PXENV_STATUS_UNDI_INVALID_STATE; +		return PXENV_EXIT_FAILURE; +	} + +	DBGC2 ( &pxe_netdev, "PXENV_UNDI_TRANSMIT" ); + +	/* Forcibly enable interrupts and freeze receive queue +	 * processing at this point, to work around callers that never +	 * call PXENV_UNDI_OPEN before attempting to use the UNDI API. +	 */ +	if ( ! netdev_rx_frozen ( pxe_netdev ) ) { +		netdev_rx_freeze ( pxe_netdev ); +		netdev_irq ( pxe_netdev, 1 ); +	} + +	/* Identify network-layer protocol */ +	switch ( undi_transmit->Protocol ) { +	case P_IP:	net_protocol = &ipv4_protocol;	break; +	case P_ARP:	net_protocol = &arp_protocol;	break; +	case P_RARP:	net_protocol = &rarp_protocol;	break; +	case P_UNKNOWN: +		net_protocol = NULL; +		break; +	default: +		DBGC2 ( &pxe_netdev, " %02x invalid protocol\n", +			undi_transmit->Protocol ); +		undi_transmit->Status = PXENV_STATUS_UNDI_INVALID_PARAMETER; +		return PXENV_EXIT_FAILURE; +	} +	DBGC2 ( &pxe_netdev, " %s", +		( net_protocol ? net_protocol->name : "RAW" ) ); + +	/* Calculate total packet length */ +	copy_from_real ( &tbd, undi_transmit->TBD.segment, +			 undi_transmit->TBD.offset, sizeof ( tbd ) ); +	len = tbd.ImmedLength; +	DBGC2 ( &pxe_netdev, " %04x:%04x+%x", tbd.Xmit.segment, tbd.Xmit.offset, +		tbd.ImmedLength ); +	for ( i = 0 ; i < tbd.DataBlkCount ; i++ ) { +		datablk = &tbd.DataBlock[i]; +		len += datablk->TDDataLen; +		DBGC2 ( &pxe_netdev, " %04x:%04x+%x", +			datablk->TDDataPtr.segment, datablk->TDDataPtr.offset, +			datablk->TDDataLen ); +	} + +	/* Allocate and fill I/O buffer */ +	iobuf = alloc_iob ( MAX_LL_HEADER_LEN + +			    ( ( len > IOB_ZLEN ) ? len : IOB_ZLEN ) ); +	if ( ! iobuf ) { +		DBGC2 ( &pxe_netdev, " could not allocate iobuf\n" ); +		undi_transmit->Status = PXENV_STATUS_OUT_OF_RESOURCES; +		return PXENV_EXIT_FAILURE; +	} +	iob_reserve ( iobuf, MAX_LL_HEADER_LEN ); +	copy_from_real ( iob_put ( iobuf, tbd.ImmedLength ), tbd.Xmit.segment, +			 tbd.Xmit.offset, tbd.ImmedLength ); +	for ( i = 0 ; i < tbd.DataBlkCount ; i++ ) { +		datablk = &tbd.DataBlock[i]; +		copy_from_real ( iob_put ( iobuf, datablk->TDDataLen ), +				 datablk->TDDataPtr.segment, +				 datablk->TDDataPtr.offset, +				 datablk->TDDataLen ); +	} + +	/* Add link-layer header, if required to do so */ +	if ( net_protocol != NULL ) { + +		/* Calculate destination address */ +		ll_protocol = pxe_netdev->ll_protocol; +		if ( undi_transmit->XmitFlag == XMT_DESTADDR ) { +			copy_from_real ( destaddr, +					 undi_transmit->DestAddr.segment, +					 undi_transmit->DestAddr.offset, +					 ll_protocol->ll_addr_len ); +			ll_dest = destaddr; +			DBGC2 ( &pxe_netdev, " DEST %s", +				ll_protocol->ntoa ( ll_dest ) ); +		} else { +			ll_dest = pxe_netdev->ll_broadcast; +			DBGC2 ( &pxe_netdev, " BCAST" ); +		} + +		/* Add link-layer header */ +		if ( ( rc = ll_protocol->push ( pxe_netdev, iobuf, ll_dest, +						pxe_netdev->ll_addr, +						net_protocol->net_proto ))!=0){ +			DBGC2 ( &pxe_netdev, " could not add link-layer " +				"header: %s\n", strerror ( rc ) ); +			free_iob ( iobuf ); +			undi_transmit->Status = PXENV_STATUS ( rc ); +			return PXENV_EXIT_FAILURE; +		} +	} + +	/* Flag transmission as in-progress.  Do this before starting +	 * to transmit the packet, because the ISR may trigger before +	 * we return from netdev_tx(). +	 */ +	undi_tx_count++; + +	/* Transmit packet */ +	DBGC2 ( &pxe_netdev, "\n" ); +	if ( ( rc = netdev_tx ( pxe_netdev, iobuf ) ) != 0 ) { +		DBGC2 ( &pxe_netdev, "PXENV_UNDI_TRANSMIT could not transmit: " +			"%s\n", strerror ( rc ) ); +		undi_tx_count--; +		undi_transmit->Status = PXENV_STATUS ( rc ); +		return PXENV_EXIT_FAILURE; +	} + +	profile_stop ( &undi_tx_profiler ); +	undi_transmit->Status = PXENV_STATUS_SUCCESS; +	return PXENV_EXIT_SUCCESS; +} + +/* PXENV_UNDI_SET_MCAST_ADDRESS + * + * Status: working (for NICs that support receive-all-multicast) + */ +static PXENV_EXIT_t +pxenv_undi_set_mcast_address ( struct s_PXENV_UNDI_SET_MCAST_ADDRESS +			       *undi_set_mcast_address ) { +	DBGC ( &pxe_netdev, "PXENV_UNDI_SET_MCAST_ADDRESS" ); +	pxe_dump_mcast_list ( &undi_set_mcast_address->R_Mcast_Buf ); +	DBGC ( &pxe_netdev, "\n" ); + +	/* Sanity check */ +	if ( ! pxe_netdev ) { +		DBGC ( &pxe_netdev, "PXENV_UNDI_SET_MCAST_ADDRESS called with " +		       "no network device\n" ); +		undi_set_mcast_address->Status = +			PXENV_STATUS_UNDI_INVALID_STATE; +		return PXENV_EXIT_FAILURE; +	} + +	undi_set_mcast_address->Status = PXENV_STATUS_SUCCESS; +	return PXENV_EXIT_SUCCESS; +} + +/* PXENV_UNDI_SET_STATION_ADDRESS + * + * Status: working + */ +static PXENV_EXIT_t +pxenv_undi_set_station_address ( struct s_PXENV_UNDI_SET_STATION_ADDRESS +				 *undi_set_station_address ) { +	struct ll_protocol *ll_protocol; + +	/* Sanity check */ +	if ( ! pxe_netdev ) { +		DBGC ( &pxe_netdev, "PXENV_UNDI_SET_STATION_ADDRESS called " +		       "with no network device\n" ); +		undi_set_station_address->Status = +			PXENV_STATUS_UNDI_INVALID_STATE; +		return PXENV_EXIT_FAILURE; +	} + +	ll_protocol = pxe_netdev->ll_protocol; +	DBGC ( &pxe_netdev, "PXENV_UNDI_SET_STATION_ADDRESS %s", +	       ll_protocol->ntoa ( undi_set_station_address->StationAddress ) ); + +	/* If adapter is open, the change will have no effect; return +	 * an error +	 */ +	if ( netdev_is_open ( pxe_netdev ) ) { +		DBGC ( &pxe_netdev, " failed: netdev is open\n" ); +		undi_set_station_address->Status = +			PXENV_STATUS_UNDI_INVALID_STATE; +		return PXENV_EXIT_FAILURE; +	} + +	/* Update MAC address */ +	memcpy ( pxe_netdev->ll_addr, +		 &undi_set_station_address->StationAddress, +		 ll_protocol->ll_addr_len ); + +	DBGC ( &pxe_netdev, "\n" ); +	undi_set_station_address->Status = PXENV_STATUS_SUCCESS; +	return PXENV_EXIT_SUCCESS; +} + +/* PXENV_UNDI_SET_PACKET_FILTER + * + * Status: won't implement (would require driver API changes for no + * real benefit) + */ +static PXENV_EXIT_t +pxenv_undi_set_packet_filter ( struct s_PXENV_UNDI_SET_PACKET_FILTER +			       *undi_set_packet_filter ) { + +	DBGC ( &pxe_netdev, "PXENV_UNDI_SET_PACKET_FILTER %02x\n", +	       undi_set_packet_filter->filter ); + +	/* Sanity check */ +	if ( ! pxe_netdev ) { +		DBGC ( &pxe_netdev, "PXENV_UNDI_SET_PACKET_FILTER called with " +		       "no network device\n" ); +		undi_set_packet_filter->Status = +			PXENV_STATUS_UNDI_INVALID_STATE; +		return PXENV_EXIT_FAILURE; +	} + +	/* Pretend that we succeeded, otherwise the 3Com DOS UNDI +	 * driver refuses to load.  (We ignore the filter value in the +	 * PXENV_UNDI_OPEN call anyway.) +	 */ +	undi_set_packet_filter->Status = PXENV_STATUS_SUCCESS; + +	return PXENV_EXIT_SUCCESS; +} + +/* PXENV_UNDI_GET_INFORMATION + * + * Status: working + */ +static PXENV_EXIT_t +pxenv_undi_get_information ( struct s_PXENV_UNDI_GET_INFORMATION +			     *undi_get_information ) { +	struct device *dev; +	struct ll_protocol *ll_protocol; + +	/* Sanity check */ +	if ( ! pxe_netdev ) { +		DBGC ( &pxe_netdev, "PXENV_UNDI_GET_INFORMATION called with no " +		       "network device\n" ); +		undi_get_information->Status = PXENV_STATUS_UNDI_INVALID_STATE; +		return PXENV_EXIT_FAILURE; +	} + +	DBGC ( &pxe_netdev, "PXENV_UNDI_GET_INFORMATION" ); + +	/* Fill in information */ +	dev = pxe_netdev->dev; +	ll_protocol = pxe_netdev->ll_protocol; +	undi_get_information->BaseIo = dev->desc.ioaddr; +	undi_get_information->IntNumber = +		( netdev_irq_supported ( pxe_netdev ) ? dev->desc.irq : 0 ); +	/* Cheat: assume all cards can cope with this */ +	undi_get_information->MaxTranUnit = ETH_MAX_MTU; +	undi_get_information->HwType = ntohs ( ll_protocol->ll_proto ); +	undi_get_information->HwAddrLen = ll_protocol->ll_addr_len; +	assert ( ll_protocol->ll_addr_len <= +		 sizeof ( undi_get_information->CurrentNodeAddress ) ); +	memcpy ( &undi_get_information->CurrentNodeAddress, +		 pxe_netdev->ll_addr, +		 sizeof ( undi_get_information->CurrentNodeAddress ) ); +	ll_protocol->init_addr ( pxe_netdev->hw_addr, +				 &undi_get_information->PermNodeAddress ); +	undi_get_information->ROMAddress = 0; +		/* nic.rom_info->rom_segment; */ +	/* We only provide the ability to receive or transmit a single +	 * packet at a time.  This is a bootloader, not an OS. +	 */ +	undi_get_information->RxBufCt = 1; +	undi_get_information->TxBufCt = 1; + +	DBGC ( &pxe_netdev, " io %04x irq %d mtu %d %s %s\n", +	       undi_get_information->BaseIo, undi_get_information->IntNumber, +	       undi_get_information->MaxTranUnit, ll_protocol->name, +	       ll_protocol->ntoa ( &undi_get_information->CurrentNodeAddress )); +	undi_get_information->Status = PXENV_STATUS_SUCCESS; +	return PXENV_EXIT_SUCCESS; +} + +/* PXENV_UNDI_GET_STATISTICS + * + * Status: working + */ +static PXENV_EXIT_t +pxenv_undi_get_statistics ( struct s_PXENV_UNDI_GET_STATISTICS +			    *undi_get_statistics ) { + +	/* Sanity check */ +	if ( ! pxe_netdev ) { +		DBGC ( &pxe_netdev, "PXENV_UNDI_GET_STATISTICS called with no " +		       "network device\n" ); +		undi_get_statistics->Status = PXENV_STATUS_UNDI_INVALID_STATE; +		return PXENV_EXIT_FAILURE; +	} + +	DBGC ( &pxe_netdev, "PXENV_UNDI_GET_STATISTICS" ); + +	/* Report statistics */ +	undi_get_statistics->XmtGoodFrames = pxe_netdev->tx_stats.good; +	undi_get_statistics->RcvGoodFrames = pxe_netdev->rx_stats.good; +	undi_get_statistics->RcvCRCErrors = pxe_netdev->rx_stats.bad; +	undi_get_statistics->RcvResourceErrors = pxe_netdev->rx_stats.bad; +	DBGC ( &pxe_netdev, " txok %d rxok %d rxcrc %d rxrsrc %d\n", +	       undi_get_statistics->XmtGoodFrames, +	       undi_get_statistics->RcvGoodFrames, +	       undi_get_statistics->RcvCRCErrors, +	       undi_get_statistics->RcvResourceErrors ); + +	undi_get_statistics->Status = PXENV_STATUS_SUCCESS; +	return PXENV_EXIT_SUCCESS; +} + +/* PXENV_UNDI_CLEAR_STATISTICS + * + * Status: working + */ +static PXENV_EXIT_t +pxenv_undi_clear_statistics ( struct s_PXENV_UNDI_CLEAR_STATISTICS +			      *undi_clear_statistics ) { +	DBGC ( &pxe_netdev, "PXENV_UNDI_CLEAR_STATISTICS\n" ); + +	/* Sanity check */ +	if ( ! pxe_netdev ) { +		DBGC ( &pxe_netdev, "PXENV_UNDI_CLEAR_STATISTICS called with " +		       "no network device\n" ); +		undi_clear_statistics->Status = PXENV_STATUS_UNDI_INVALID_STATE; +		return PXENV_EXIT_FAILURE; +	} + +	/* Clear statistics */ +	memset ( &pxe_netdev->tx_stats, 0, sizeof ( pxe_netdev->tx_stats ) ); +	memset ( &pxe_netdev->rx_stats, 0, sizeof ( pxe_netdev->rx_stats ) ); + +	undi_clear_statistics->Status = PXENV_STATUS_SUCCESS; +	return PXENV_EXIT_SUCCESS; +} + +/* PXENV_UNDI_INITIATE_DIAGS + * + * Status: won't implement (would require driver API changes for no + * real benefit) + */ +static PXENV_EXIT_t +pxenv_undi_initiate_diags ( struct s_PXENV_UNDI_INITIATE_DIAGS +			    *undi_initiate_diags ) { +	DBGC ( &pxe_netdev, "PXENV_UNDI_INITIATE_DIAGS failed: unsupported\n" ); + +	/* Sanity check */ +	if ( ! pxe_netdev ) { +		DBGC ( &pxe_netdev, "PXENV_UNDI_INITIATE_DIAGS called with no " +		       "network device\n" ); +		undi_initiate_diags->Status = PXENV_STATUS_UNDI_INVALID_STATE; +		return PXENV_EXIT_FAILURE; +	} + +	undi_initiate_diags->Status = PXENV_STATUS_UNSUPPORTED; +	return PXENV_EXIT_FAILURE; +} + +/* PXENV_UNDI_FORCE_INTERRUPT + * + * Status: won't implement (would require driver API changes for no + * perceptible benefit) + */ +static PXENV_EXIT_t +pxenv_undi_force_interrupt ( struct s_PXENV_UNDI_FORCE_INTERRUPT +			     *undi_force_interrupt ) { +	DBGC ( &pxe_netdev, +	       "PXENV_UNDI_FORCE_INTERRUPT failed: unsupported\n" ); + +	/* Sanity check */ +	if ( ! pxe_netdev ) { +		DBGC ( &pxe_netdev, "PXENV_UNDI_FORCE_INTERRUPT called with no " +		       "network device\n" ); +		undi_force_interrupt->Status = PXENV_STATUS_UNDI_INVALID_STATE; +		return PXENV_EXIT_FAILURE; +	} + +	undi_force_interrupt->Status = PXENV_STATUS_UNSUPPORTED; +	return PXENV_EXIT_FAILURE; +} + +/* PXENV_UNDI_GET_MCAST_ADDRESS + * + * Status: working + */ +static PXENV_EXIT_t +pxenv_undi_get_mcast_address ( struct s_PXENV_UNDI_GET_MCAST_ADDRESS +			       *undi_get_mcast_address ) { +	struct ll_protocol *ll_protocol; +	struct in_addr ip = { .s_addr = undi_get_mcast_address->InetAddr }; +	int rc; + +	/* Sanity check */ +	if ( ! pxe_netdev ) { +		DBGC ( &pxe_netdev, "PXENV_UNDI_GET_MCAST_ADDRESS called with " +		       "no network device\n" ); +		undi_get_mcast_address->Status = +			PXENV_STATUS_UNDI_INVALID_STATE; +		return PXENV_EXIT_FAILURE; +	} + +	DBGC ( &pxe_netdev, "PXENV_UNDI_GET_MCAST_ADDRESS %s", +	       inet_ntoa ( ip ) ); + +	/* Hash address using the network device's link-layer protocol */ +	ll_protocol = pxe_netdev->ll_protocol; +	if ( ( rc = ll_protocol->mc_hash ( AF_INET, &ip, +				      undi_get_mcast_address->MediaAddr ))!=0){ +		DBGC ( &pxe_netdev, " failed: %s\n", strerror ( rc ) ); +		undi_get_mcast_address->Status = PXENV_STATUS ( rc ); +		return PXENV_EXIT_FAILURE; +	} +	DBGC ( &pxe_netdev, "=>%s\n", +	       ll_protocol->ntoa ( undi_get_mcast_address->MediaAddr ) ); + +	undi_get_mcast_address->Status = PXENV_STATUS_SUCCESS; +	return PXENV_EXIT_SUCCESS; +} + +/* PXENV_UNDI_GET_NIC_TYPE + * + * Status: working + */ +static PXENV_EXIT_t pxenv_undi_get_nic_type ( struct s_PXENV_UNDI_GET_NIC_TYPE +					      *undi_get_nic_type ) { +	struct device *dev; + +	/* Sanity check */ +	if ( ! pxe_netdev ) { +		DBGC ( &pxe_netdev, "PXENV_UNDI_GET_NIC_TYPE called with " +		       "no network device\n" ); +		undi_get_nic_type->Status = PXENV_STATUS_UNDI_INVALID_STATE; +		return PXENV_EXIT_FAILURE; +	} + +	DBGC ( &pxe_netdev, "PXENV_UNDI_GET_NIC_TYPE" ); + +	/* Fill in information */ +	memset ( &undi_get_nic_type->info, 0, +		 sizeof ( undi_get_nic_type->info ) ); +	dev = pxe_netdev->dev; +	switch ( dev->desc.bus_type ) { +	case BUS_TYPE_PCI: { +		struct pci_nic_info *info = &undi_get_nic_type->info.pci; + +		undi_get_nic_type->NicType = PCI_NIC; +		info->Vendor_ID = dev->desc.vendor; +		info->Dev_ID = dev->desc.device; +		info->Base_Class = PCI_BASE_CLASS ( dev->desc.class ); +		info->Sub_Class = PCI_SUB_CLASS ( dev->desc.class ); +		info->Prog_Intf = PCI_PROG_INTF ( dev->desc.class ); +		info->BusDevFunc = dev->desc.location; +		/* Earlier versions of the PXE specification do not +		 * have the SubVendor_ID and SubDevice_ID fields.  It +		 * is possible that some NBPs will not provide space +		 * for them, and so we must not fill them in. +		 */ +		DBGC ( &pxe_netdev, " PCI %02x:%02x.%x %04x:%04x " +		       "('%04x:%04x') %02x%02x%02x rev %02x\n", +		       PCI_BUS ( info->BusDevFunc ), +		       PCI_SLOT ( info->BusDevFunc ), +		       PCI_FUNC ( info->BusDevFunc ), info->Vendor_ID, +		       info->Dev_ID, info->SubVendor_ID, info->SubDevice_ID, +		       info->Base_Class, info->Sub_Class, info->Prog_Intf, +		       info->Rev ); +		break; } +	case BUS_TYPE_ISAPNP: { +		struct pnp_nic_info *info = &undi_get_nic_type->info.pnp; + +		undi_get_nic_type->NicType = PnP_NIC; +		info->EISA_Dev_ID = ( ( dev->desc.vendor << 16 ) | +				      dev->desc.device ); +		info->CardSelNum = dev->desc.location; +		/* Cheat: remaining fields are probably unnecessary, +		 * and would require adding extra code to isapnp.c. +		 */ +		DBGC ( &pxe_netdev, " ISAPnP CSN %04x %08x %02x%02x%02x\n", +		       info->CardSelNum, info->EISA_Dev_ID, +		       info->Base_Class, info->Sub_Class, info->Prog_Intf ); +		break; } +	default: +		DBGC ( &pxe_netdev, " failed: unknown bus type\n" ); +		undi_get_nic_type->Status = PXENV_STATUS_FAILURE; +		return PXENV_EXIT_FAILURE; +	} + +	undi_get_nic_type->Status = PXENV_STATUS_SUCCESS; +	return PXENV_EXIT_SUCCESS; +} + +/* PXENV_UNDI_GET_IFACE_INFO + * + * Status: working + */ +static PXENV_EXIT_t +pxenv_undi_get_iface_info ( struct s_PXENV_UNDI_GET_IFACE_INFO +			    *undi_get_iface_info ) { + +	/* Sanity check */ +	if ( ! pxe_netdev ) { +		DBGC ( &pxe_netdev, "PXENV_UNDI_GET_IFACE_INFO called with " +		       "no network device\n" ); +		undi_get_iface_info->Status = PXENV_STATUS_UNDI_INVALID_STATE; +		return PXENV_EXIT_FAILURE; +	} + +	DBGC ( &pxe_netdev, "PXENV_UNDI_GET_IFACE_INFO" ); + +	/* Just hand back some info, doesn't really matter what it is. +	 * Most PXE stacks seem to take this approach. +	 */ +	snprintf ( ( char * ) undi_get_iface_info->IfaceType, +		   sizeof ( undi_get_iface_info->IfaceType ), "DIX+802.3" ); +	undi_get_iface_info->LinkSpeed = 10000000; /* 10 Mbps */ +	undi_get_iface_info->ServiceFlags = +		( SUPPORTED_BROADCAST | SUPPORTED_MULTICAST | +		  SUPPORTED_SET_STATION_ADDRESS | SUPPORTED_RESET | +		  SUPPORTED_OPEN_CLOSE ); +	if ( netdev_irq_supported ( pxe_netdev ) ) +		undi_get_iface_info->ServiceFlags |= SUPPORTED_IRQ; +	memset ( undi_get_iface_info->Reserved, 0, +		 sizeof(undi_get_iface_info->Reserved) ); + +	DBGC ( &pxe_netdev, " %s %dbps flags %08x\n", +	       undi_get_iface_info->IfaceType, undi_get_iface_info->LinkSpeed, +	       undi_get_iface_info->ServiceFlags ); +	undi_get_iface_info->Status = PXENV_STATUS_SUCCESS; +	return PXENV_EXIT_SUCCESS; +} + +/* PXENV_UNDI_GET_STATE + * + * Status: impossible due to opcode collision + */ + +/* PXENV_UNDI_ISR + * + * Status: working + */ +static PXENV_EXIT_t pxenv_undi_isr ( struct s_PXENV_UNDI_ISR *undi_isr ) { +	struct io_buffer *iobuf; +	size_t len; +	struct ll_protocol *ll_protocol; +	const void *ll_dest; +	const void *ll_source; +	uint16_t net_proto; +	unsigned int flags; +	size_t ll_hlen; +	struct net_protocol *net_protocol; +	unsigned int prottype; +	int rc; + +	/* Use a different debug colour, since UNDI ISR messages are +	 * likely to be interspersed amongst other UNDI messages. +	 */ + +	/* Sanity check */ +	if ( ! pxe_netdev ) { +		DBGC ( &pxenv_undi_isr, "PXENV_UNDI_ISR called with " +		       "no network device\n" ); +		undi_isr->Status = PXENV_STATUS_UNDI_INVALID_STATE; +		return PXENV_EXIT_FAILURE; +	} + +	DBGC2 ( &pxenv_undi_isr, "PXENV_UNDI_ISR" ); + +	/* Just in case some idiot actually looks at these fields when +	 * we weren't meant to fill them in... +	 */ +	undi_isr->BufferLength = 0; +	undi_isr->FrameLength = 0; +	undi_isr->FrameHeaderLength = 0; +	undi_isr->ProtType = 0; +	undi_isr->PktType = 0; + +	switch ( undi_isr->FuncFlag ) { +	case PXENV_UNDI_ISR_IN_START : +		DBGC2 ( &pxenv_undi_isr, " START" ); + +		/* Call poll().  This should acknowledge the device +		 * interrupt and queue up any received packet. +		 */ +		net_poll(); + +		/* A 100% accurate determination of "OURS" vs "NOT +		 * OURS" is difficult to achieve without invasive and +		 * unpleasant changes to the driver model.  We settle +		 * for always returning "OURS" if interrupts are +		 * currently enabled. +		 * +		 * Returning "NOT OURS" when interrupts are disabled +		 * allows us to avoid a potential interrupt storm when +		 * we are on a shared interrupt line; if we were to +		 * always return "OURS" then the other device's ISR +		 * may never be called. +		 */ +		if ( netdev_irq_enabled ( pxe_netdev ) ) { +			DBGC2 ( &pxenv_undi_isr, " OURS" ); +			undi_isr->FuncFlag = PXENV_UNDI_ISR_OUT_OURS; +		} else { +			DBGC2 ( &pxenv_undi_isr, " NOT OURS" ); +			undi_isr->FuncFlag = PXENV_UNDI_ISR_OUT_NOT_OURS; +		} + +		/* Disable interrupts */ +		netdev_irq ( pxe_netdev, 0 ); + +		break; +	case PXENV_UNDI_ISR_IN_PROCESS : +	case PXENV_UNDI_ISR_IN_GET_NEXT : +		DBGC2 ( &pxenv_undi_isr, " %s", +			( ( undi_isr->FuncFlag == PXENV_UNDI_ISR_IN_PROCESS ) ? +			  "PROCESS" : "GET_NEXT" ) ); + +		/* Some dumb NBPs (e.g. emBoot's winBoot/i) never call +		 * PXENV_UNDI_ISR with FuncFlag=PXENV_UNDI_ISR_START; +		 * they just sit in a tight polling loop merrily +		 * violating the PXE spec with repeated calls to +		 * PXENV_UNDI_ISR_IN_PROCESS.  Force extra polls to +		 * cope with these out-of-spec clients. +		 */ +		net_poll(); + +		/* If we have not yet marked a TX as complete, and the +		 * netdev TX queue is empty, report the TX completion. +		 */ +		if ( undi_tx_count && list_empty ( &pxe_netdev->tx_queue ) ) { +			DBGC2 ( &pxenv_undi_isr, " TXC" ); +			undi_tx_count--; +			undi_isr->FuncFlag = PXENV_UNDI_ISR_OUT_TRANSMIT; +			break; +		} + +		/* Remove first packet from netdev RX queue */ +		iobuf = netdev_rx_dequeue ( pxe_netdev ); +		if ( ! iobuf ) { +			DBGC2 ( &pxenv_undi_isr, " DONE" ); +			/* No more packets remaining */ +			undi_isr->FuncFlag = PXENV_UNDI_ISR_OUT_DONE; +			/* Re-enable interrupts */ +			netdev_irq ( pxe_netdev, 1 ); +			break; +		} + +		/* Copy packet to base memory buffer */ +		len = iob_len ( iobuf ); +		DBGC2 ( &pxenv_undi_isr, " RX" ); +		if ( len > sizeof ( basemem_packet ) ) { +			/* Should never happen */ +			DBGC2 ( &pxenv_undi_isr, " overlength (%zx)", len ); +			len = sizeof ( basemem_packet ); +		} +		memcpy ( basemem_packet, iobuf->data, len ); + +		/* Strip link-layer header */ +		ll_protocol = pxe_netdev->ll_protocol; +		if ( ( rc = ll_protocol->pull ( pxe_netdev, iobuf, &ll_dest, +						&ll_source, &net_proto, +						&flags ) ) != 0 ) { +			/* Assume unknown net_proto and no ll_source */ +			net_proto = 0; +			ll_source = NULL; +		} +		ll_hlen = ( len - iob_len ( iobuf ) ); + +		/* Determine network-layer protocol */ +		switch ( net_proto ) { +		case htons ( ETH_P_IP ): +			net_protocol = &ipv4_protocol; +			prottype = P_IP; +			break; +		case htons ( ETH_P_ARP ): +			net_protocol = &arp_protocol; +			prottype = P_ARP; +			break; +		case htons ( ETH_P_RARP ): +			net_protocol = &rarp_protocol; +			prottype = P_RARP; +			break; +		default: +			net_protocol = NULL; +			prottype = P_UNKNOWN; +			break; +		} + +		/* Fill in UNDI_ISR structure */ +		undi_isr->FuncFlag = PXENV_UNDI_ISR_OUT_RECEIVE; +		undi_isr->BufferLength = len; +		undi_isr->FrameLength = len; +		undi_isr->FrameHeaderLength = ll_hlen; +		undi_isr->Frame.segment = rm_ds; +		undi_isr->Frame.offset = __from_data16 ( basemem_packet ); +		undi_isr->ProtType = prottype; +		if ( flags & LL_BROADCAST ) { +			undi_isr->PktType = P_BROADCAST; +		} else if ( flags & LL_MULTICAST ) { +			undi_isr->PktType = P_MULTICAST; +		} else { +			undi_isr->PktType = P_DIRECTED; +		} +		DBGC2 ( &pxenv_undi_isr, " %04x:%04x+%x(%x) %s hlen %d", +			undi_isr->Frame.segment, undi_isr->Frame.offset, +			undi_isr->BufferLength, undi_isr->FrameLength, +			( net_protocol ? net_protocol->name : "RAW" ), +			undi_isr->FrameHeaderLength ); + +		/* Free packet */ +		free_iob ( iobuf ); +		break; +	default : +		DBGC2 ( &pxenv_undi_isr, " INVALID(%04x)\n", +			undi_isr->FuncFlag ); + +		/* Should never happen */ +		undi_isr->FuncFlag = PXENV_UNDI_ISR_OUT_DONE; +		undi_isr->Status = PXENV_STATUS_UNDI_INVALID_PARAMETER; +		return PXENV_EXIT_FAILURE; +	} + +	DBGC2 ( &pxenv_undi_isr, "\n" ); +	undi_isr->Status = PXENV_STATUS_SUCCESS; +	return PXENV_EXIT_SUCCESS; +} + +/** PXE UNDI API */ +struct pxe_api_call pxe_undi_api[] __pxe_api_call = { +	PXE_API_CALL ( PXENV_UNDI_STARTUP, pxenv_undi_startup, +		       struct s_PXENV_UNDI_STARTUP ), +	PXE_API_CALL ( PXENV_UNDI_CLEANUP, pxenv_undi_cleanup, +		       struct s_PXENV_UNDI_CLEANUP ), +	PXE_API_CALL ( PXENV_UNDI_INITIALIZE, pxenv_undi_initialize, +		       struct s_PXENV_UNDI_INITIALIZE ), +	PXE_API_CALL ( PXENV_UNDI_RESET_ADAPTER, pxenv_undi_reset_adapter, +		       struct s_PXENV_UNDI_RESET ), +	PXE_API_CALL ( PXENV_UNDI_SHUTDOWN, pxenv_undi_shutdown, +		       struct s_PXENV_UNDI_SHUTDOWN ), +	PXE_API_CALL ( PXENV_UNDI_OPEN, pxenv_undi_open, +		       struct s_PXENV_UNDI_OPEN ), +	PXE_API_CALL ( PXENV_UNDI_CLOSE, pxenv_undi_close, +		       struct s_PXENV_UNDI_CLOSE ), +	PXE_API_CALL ( PXENV_UNDI_TRANSMIT, pxenv_undi_transmit, +		       struct s_PXENV_UNDI_TRANSMIT ), +	PXE_API_CALL ( PXENV_UNDI_SET_MCAST_ADDRESS, +		       pxenv_undi_set_mcast_address, +		       struct s_PXENV_UNDI_SET_MCAST_ADDRESS ), +	PXE_API_CALL ( PXENV_UNDI_SET_STATION_ADDRESS, +		       pxenv_undi_set_station_address, +		       struct s_PXENV_UNDI_SET_STATION_ADDRESS ), +	PXE_API_CALL ( PXENV_UNDI_SET_PACKET_FILTER, +		       pxenv_undi_set_packet_filter, +		       struct s_PXENV_UNDI_SET_PACKET_FILTER ), +	PXE_API_CALL ( PXENV_UNDI_GET_INFORMATION, pxenv_undi_get_information, +		       struct s_PXENV_UNDI_GET_INFORMATION ), +	PXE_API_CALL ( PXENV_UNDI_GET_STATISTICS, pxenv_undi_get_statistics, +		       struct s_PXENV_UNDI_GET_STATISTICS ), +	PXE_API_CALL ( PXENV_UNDI_CLEAR_STATISTICS, pxenv_undi_clear_statistics, +		       struct s_PXENV_UNDI_CLEAR_STATISTICS ), +	PXE_API_CALL ( PXENV_UNDI_INITIATE_DIAGS, pxenv_undi_initiate_diags, +		       struct s_PXENV_UNDI_INITIATE_DIAGS ), +	PXE_API_CALL ( PXENV_UNDI_FORCE_INTERRUPT, pxenv_undi_force_interrupt, +		       struct s_PXENV_UNDI_FORCE_INTERRUPT ), +	PXE_API_CALL ( PXENV_UNDI_GET_MCAST_ADDRESS, +		       pxenv_undi_get_mcast_address, +		       struct s_PXENV_UNDI_GET_MCAST_ADDRESS ), +	PXE_API_CALL ( PXENV_UNDI_GET_NIC_TYPE, pxenv_undi_get_nic_type, +		       struct s_PXENV_UNDI_GET_NIC_TYPE ), +	PXE_API_CALL ( PXENV_UNDI_GET_IFACE_INFO, pxenv_undi_get_iface_info, +		       struct s_PXENV_UNDI_GET_IFACE_INFO ), +	PXE_API_CALL ( PXENV_UNDI_ISR, pxenv_undi_isr, +		       struct s_PXENV_UNDI_ISR ), +}; diff --git a/roms/ipxe/src/arch/i386/interface/pxeparent/pxeparent.c b/roms/ipxe/src/arch/i386/interface/pxeparent/pxeparent.c new file mode 100644 index 00000000..0b6be9a0 --- /dev/null +++ b/roms/ipxe/src/arch/i386/interface/pxeparent/pxeparent.c @@ -0,0 +1,283 @@ +/* + * Copyright (C) 2007 Michael Brown <mbrown@fensystems.co.uk>. + * + * This program is free software; you can redistribute it and/or + * modify it under the terms of the GNU General Public License as + * published by the Free Software Foundation; either version 2 of the + * License, or any later version. + * + * 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. + * + * You should have received a copy of the GNU General Public License + * along with this program; if not, write to the Free Software + * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA + * 02110-1301, USA. + */ + +FILE_LICENCE ( GPL2_OR_LATER ); + +#include <ipxe/dhcp.h> +#include <ipxe/profile.h> +#include <pxeparent.h> +#include <pxe_api.h> +#include <pxe_types.h> +#include <pxe.h> + +/** @file + * + * Call interface to parent PXE stack + * + */ + +/* Disambiguate the various error causes */ +#define EINFO_EPXECALL							\ +	__einfo_uniqify ( EINFO_EPLATFORM, 0x01,			\ +			  "External PXE API error" ) +#define EPXECALL( status ) EPLATFORM ( EINFO_EPXECALL, status ) + +/** A parent PXE API call profiler */ +struct pxeparent_profiler { +	/** Total time spent performing REAL_CALL() */ +	struct profiler total; +	/** Time spent transitioning to real mode */ +	struct profiler p2r; +	/** Time spent in external code */ +	struct profiler ext; +	/** Time spent transitioning back to protected mode */ +	struct profiler r2p; +}; + +/** PXENV_UNDI_TRANSMIT profiler */ +static struct pxeparent_profiler pxeparent_tx_profiler __profiler = { +	{ .name = "pxeparent.tx" }, +	{ .name = "pxeparent.tx_p2r" }, +	{ .name = "pxeparent.tx_ext" }, +	{ .name = "pxeparent.tx_r2p" }, +}; + +/** PXENV_UNDI_ISR profiler + * + * Note that this profiler will not see calls to + * PXENV_UNDI_ISR_IN_START, which are handled by the UNDI ISR and do + * not go via pxeparent_call(). + */ +static struct pxeparent_profiler pxeparent_isr_profiler __profiler = { +	{ .name = "pxeparent.isr" }, +	{ .name = "pxeparent.isr_p2r" }, +	{ .name = "pxeparent.isr_ext" }, +	{ .name = "pxeparent.isr_r2p" }, +}; + +/** PXE unknown API call profiler + * + * This profiler can be used to measure the overhead of a dummy PXE + * API call. + */ +static struct pxeparent_profiler pxeparent_unknown_profiler __profiler = { +	{ .name = "pxeparent.unknown" }, +	{ .name = "pxeparent.unknown_p2r" }, +	{ .name = "pxeparent.unknown_ext" }, +	{ .name = "pxeparent.unknown_r2p" }, +}; + +/** Miscellaneous PXE API call profiler */ +static struct pxeparent_profiler pxeparent_misc_profiler __profiler = { +	{ .name = "pxeparent.misc" }, +	{ .name = "pxeparent.misc_p2r" }, +	{ .name = "pxeparent.misc_ext" }, +	{ .name = "pxeparent.misc_r2p" }, +}; + +/** + * Name PXE API call + * + * @v function		API call number + * @ret name		API call name + */ +static inline __attribute__ (( always_inline )) const char * +pxeparent_function_name ( unsigned int function ) { +	switch ( function ) { +	case PXENV_START_UNDI: +		return "PXENV_START_UNDI"; +	case PXENV_STOP_UNDI: +		return "PXENV_STOP_UNDI"; +	case PXENV_UNDI_STARTUP: +		return "PXENV_UNDI_STARTUP"; +	case PXENV_UNDI_CLEANUP: +		return "PXENV_UNDI_CLEANUP"; +	case PXENV_UNDI_INITIALIZE: +		return "PXENV_UNDI_INITIALIZE"; +	case PXENV_UNDI_RESET_ADAPTER: +		return "PXENV_UNDI_RESET_ADAPTER"; +	case PXENV_UNDI_SHUTDOWN: +		return "PXENV_UNDI_SHUTDOWN"; +	case PXENV_UNDI_OPEN: +		return "PXENV_UNDI_OPEN"; +	case PXENV_UNDI_CLOSE: +		return "PXENV_UNDI_CLOSE"; +	case PXENV_UNDI_TRANSMIT: +		return "PXENV_UNDI_TRANSMIT"; +	case PXENV_UNDI_SET_MCAST_ADDRESS: +		return "PXENV_UNDI_SET_MCAST_ADDRESS"; +	case PXENV_UNDI_SET_STATION_ADDRESS: +		return "PXENV_UNDI_SET_STATION_ADDRESS"; +	case PXENV_UNDI_SET_PACKET_FILTER: +		return "PXENV_UNDI_SET_PACKET_FILTER"; +	case PXENV_UNDI_GET_INFORMATION: +		return "PXENV_UNDI_GET_INFORMATION"; +	case PXENV_UNDI_GET_STATISTICS: +		return "PXENV_UNDI_GET_STATISTICS"; +	case PXENV_UNDI_CLEAR_STATISTICS: +		return "PXENV_UNDI_CLEAR_STATISTICS"; +	case PXENV_UNDI_INITIATE_DIAGS: +		return "PXENV_UNDI_INITIATE_DIAGS"; +	case PXENV_UNDI_FORCE_INTERRUPT: +		return "PXENV_UNDI_FORCE_INTERRUPT"; +	case PXENV_UNDI_GET_MCAST_ADDRESS: +		return "PXENV_UNDI_GET_MCAST_ADDRESS"; +	case PXENV_UNDI_GET_NIC_TYPE: +		return "PXENV_UNDI_GET_NIC_TYPE"; +	case PXENV_UNDI_GET_IFACE_INFO: +		return "PXENV_UNDI_GET_IFACE_INFO"; +	/* +	 * Duplicate case value; this is a bug in the PXE specification. +	 * +	 *	case PXENV_UNDI_GET_STATE: +	 *		return "PXENV_UNDI_GET_STATE"; +	 */ +	case PXENV_UNDI_ISR: +		return "PXENV_UNDI_ISR"; +	case PXENV_GET_CACHED_INFO: +		return "PXENV_GET_CACHED_INFO"; +	default: +		return "UNKNOWN API CALL"; +	} +} + +/** + * Determine applicable profiler pair (for debugging) + * + * @v function		API call number + * @ret profiler	Profiler + */ +static struct pxeparent_profiler * pxeparent_profiler ( unsigned int function ){ + +	/* Determine applicable profiler */ +	switch ( function ) { +	case PXENV_UNDI_TRANSMIT: +		return &pxeparent_tx_profiler; +	case PXENV_UNDI_ISR: +		return &pxeparent_isr_profiler; +	case PXENV_UNKNOWN: +		return &pxeparent_unknown_profiler; +	default: +		return &pxeparent_misc_profiler; +	} +} + +/** + * PXE parent parameter block + * + * Used as the parameter block for all parent PXE API calls.  Resides + * in base memory. + */ +static union u_PXENV_ANY __bss16 ( pxeparent_params ); +#define pxeparent_params __use_data16 ( pxeparent_params ) + +/** PXE parent entry point + * + * Used as the indirection vector for all parent PXE API calls.  Resides in + * base memory. + */ +SEGOFF16_t __bss16 ( pxeparent_entry_point ); +#define pxeparent_entry_point __use_data16 ( pxeparent_entry_point ) + +/** + * Issue parent PXE API call + * + * @v entry		Parent PXE stack entry point + * @v function		API call number + * @v params		PXE parameter block + * @v params_len	Length of PXE parameter block + * @ret rc		Return status code + */ +int pxeparent_call ( SEGOFF16_t entry, unsigned int function, +		     void *params, size_t params_len ) { +	struct pxeparent_profiler *profiler = pxeparent_profiler ( function ); +	PXENV_EXIT_t exit; +	unsigned long started; +	unsigned long stopped; +	int discard_D; +	int rc; + +	/* Copy parameter block and entry point */ +	assert ( params_len <= sizeof ( pxeparent_params ) ); +	memcpy ( &pxeparent_params, params, params_len ); +	memcpy ( &pxeparent_entry_point, &entry, sizeof ( entry ) ); + +	/* Call real-mode entry point.  This calling convention will +	 * work with both the !PXE and the PXENV+ entry points. +	 */ +	profile_start ( &profiler->total ); +	__asm__ __volatile__ ( REAL_CODE ( "pushl %%ebp\n\t" /* gcc bug */ +					   "rdtsc\n\t" +					   "pushl %%eax\n\t" +					   "pushw %%es\n\t" +					   "pushw %%di\n\t" +					   "pushw %%bx\n\t" +					   "lcall *pxeparent_entry_point\n\t" +					   "movw %%ax, %%bx\n\t" +					   "rdtsc\n\t" +					   "addw $6, %%sp\n\t" +					   "popl %%edx\n\t" +					   "popl %%ebp\n\t" /* gcc bug */ ) +			       : "=a" ( stopped ), "=d" ( started ), +				 "=b" ( exit ), "=D" ( discard_D ) +			       : "b" ( function ), +			         "D" ( __from_data16 ( &pxeparent_params ) ) +			       : "ecx", "esi" ); +	profile_stop ( &profiler->total ); +	profile_start_at ( &profiler->p2r, profile_started ( &profiler->total)); +	profile_stop_at ( &profiler->p2r, started ); +	profile_start_at ( &profiler->ext, started ); +	profile_stop_at ( &profiler->ext, stopped ); +	profile_start_at ( &profiler->r2p, stopped ); +	profile_stop_at ( &profiler->r2p, profile_stopped ( &profiler->total )); + +	/* Determine return status code based on PXENV_EXIT and +	 * PXENV_STATUS +	 */ +	rc = ( ( exit == PXENV_EXIT_SUCCESS ) ? +	       0 : -EPXECALL ( pxeparent_params.Status ) ); + +	/* If anything goes wrong, print as much debug information as +	 * it's possible to give. +	 */ +	if ( rc != 0 ) { +		SEGOFF16_t rm_params = { +			.segment = rm_ds, +			.offset = __from_data16 ( &pxeparent_params ), +		}; + +		DBG ( "PXEPARENT %s failed: %s\n", +		       pxeparent_function_name ( function ), strerror ( rc ) ); +		DBG ( "PXEPARENT parameters at %04x:%04x length " +		       "%#02zx, entry point at %04x:%04x\n", +		       rm_params.segment, rm_params.offset, params_len, +		       pxeparent_entry_point.segment, +		       pxeparent_entry_point.offset ); +		DBG ( "PXEPARENT parameters provided:\n" ); +		DBG_HDA ( rm_params, params, params_len ); +		DBG ( "PXEPARENT parameters returned:\n" ); +		DBG_HDA ( rm_params, &pxeparent_params, params_len ); +	} + +	/* Copy parameter block back */ +	memcpy ( params, &pxeparent_params, params_len ); + +	return rc; +} + diff --git a/roms/ipxe/src/arch/i386/interface/syslinux/com32_call.c b/roms/ipxe/src/arch/i386/interface/syslinux/com32_call.c new file mode 100644 index 00000000..75dcc238 --- /dev/null +++ b/roms/ipxe/src/arch/i386/interface/syslinux/com32_call.c @@ -0,0 +1,191 @@ +/* + * Copyright (C) 2008 Daniel Verkamp <daniel@drv.nu>. + * + * This program is free software; you can redistribute it and/or + * modify it under the terms of the GNU General Public License as + * published by the Free Software Foundation; either version 2 of the + * License, or any later version. + * + * 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. + * + * You should have received a copy of the GNU General Public License + * along with this program; if not, write to the Free Software + * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA + * 02110-1301, USA. + */ + +/** + * @file SYSLINUX COM32 helpers + * + */ + +FILE_LICENCE ( GPL2_OR_LATER ); + +#include <stdint.h> +#include <realmode.h> +#include <comboot.h> +#include <assert.h> +#include <ipxe/uaccess.h> + +static com32sys_t __bss16 ( com32_regs ); +#define com32_regs __use_data16 ( com32_regs ) + +static uint8_t __bss16 ( com32_int_vector ); +#define com32_int_vector __use_data16 ( com32_int_vector ) + +static uint32_t __bss16 ( com32_farcall_proc ); +#define com32_farcall_proc __use_data16 ( com32_farcall_proc ) + +uint16_t __bss16 ( com32_saved_sp ); + +/** + * Interrupt call helper + */ +void __asmcall com32_intcall ( uint8_t interrupt, physaddr_t inregs_phys, physaddr_t outregs_phys ) { + +	memcpy_user ( virt_to_user( &com32_regs ), 0, +	              phys_to_user ( inregs_phys ), 0, +	              sizeof(com32sys_t) ); + +	com32_int_vector = interrupt; + +	__asm__ __volatile__ ( +		REAL_CODE ( /* Save all registers */ +		            "pushal\n\t" +		            "pushw %%ds\n\t" +		            "pushw %%es\n\t" +		            "pushw %%fs\n\t" +		            "pushw %%gs\n\t" +		            /* Mask off unsafe flags */ +		            "movl (com32_regs + 40), %%eax\n\t" +		            "andl $0x200cd7, %%eax\n\t" +		            "movl %%eax, (com32_regs + 40)\n\t" +		            /* Load com32_regs into the actual registers */ +		            "movw %%sp, %%ss:(com32_saved_sp)\n\t" +		            "movw $com32_regs, %%sp\n\t" +		            "popw %%gs\n\t" +		            "popw %%fs\n\t" +		            "popw %%es\n\t" +		            "popw %%ds\n\t" +		            "popal\n\t" +		            "popfl\n\t" +		            "movw %%ss:(com32_saved_sp), %%sp\n\t" +		            /* patch INT instruction */ +		            "pushw %%ax\n\t" +		            "movb %%ss:(com32_int_vector), %%al\n\t" +		            "movb %%al, %%cs:(com32_intcall_instr + 1)\n\t"  +		            /* perform a jump to avoid problems with cache +		             * consistency in self-modifying code on some CPUs (486) +		             */ +		            "jmp 1f\n" +		            "1:\n\t" +		            "popw %%ax\n\t" +		            "com32_intcall_instr:\n\t" +		            /* INT instruction to be patched */ +		            "int $0xFF\n\t" +		            /* Copy regs back to com32_regs */ +		            "movw %%sp, %%ss:(com32_saved_sp)\n\t" +		            "movw $(com32_regs + 44), %%sp\n\t" +		            "pushfl\n\t" +		            "pushal\n\t" +		            "pushw %%ds\n\t" +		            "pushw %%es\n\t" +		            "pushw %%fs\n\t" +		            "pushw %%gs\n\t" +		            "movw %%ss:(com32_saved_sp), %%sp\n\t" +		            /* Restore registers */ +		            "popw %%gs\n\t" +		            "popw %%fs\n\t" +		            "popw %%es\n\t" +		            "popw %%ds\n\t" +		            "popal\n\t") +		            : : ); + +	if ( outregs_phys ) { +		memcpy_user ( phys_to_user ( outregs_phys ), 0, +		              virt_to_user( &com32_regs ), 0,  +		              sizeof(com32sys_t) ); +	} +} + +/** + * Farcall helper + */ +void __asmcall com32_farcall ( uint32_t proc, physaddr_t inregs_phys, physaddr_t outregs_phys ) { + +	memcpy_user ( virt_to_user( &com32_regs ), 0, +	              phys_to_user ( inregs_phys ), 0, +	              sizeof(com32sys_t) ); + +	com32_farcall_proc = proc; + +	__asm__ __volatile__ ( +		REAL_CODE ( /* Save all registers */ +		            "pushal\n\t" +		            "pushw %%ds\n\t" +		            "pushw %%es\n\t" +		            "pushw %%fs\n\t" +		            "pushw %%gs\n\t" +		            /* Mask off unsafe flags */ +		            "movl (com32_regs + 40), %%eax\n\t" +		            "andl $0x200cd7, %%eax\n\t" +		            "movl %%eax, (com32_regs + 40)\n\t" +		            /* Load com32_regs into the actual registers */ +		            "movw %%sp, %%ss:(com32_saved_sp)\n\t" +		            "movw $com32_regs, %%sp\n\t" +		            "popw %%gs\n\t" +		            "popw %%fs\n\t" +		            "popw %%es\n\t" +		            "popw %%ds\n\t" +		            "popal\n\t" +		            "popfl\n\t" +		            "movw %%ss:(com32_saved_sp), %%sp\n\t" +		            /* Call procedure */ +		            "lcall *%%ss:(com32_farcall_proc)\n\t" +		            /* Copy regs back to com32_regs */ +		            "movw %%sp, %%ss:(com32_saved_sp)\n\t" +		            "movw $(com32_regs + 44), %%sp\n\t" +		            "pushfl\n\t" +		            "pushal\n\t" +		            "pushw %%ds\n\t" +		            "pushw %%es\n\t" +		            "pushw %%fs\n\t" +		            "pushw %%gs\n\t" +		            "movw %%ss:(com32_saved_sp), %%sp\n\t" +		            /* Restore registers */ +		            "popw %%gs\n\t" +		            "popw %%fs\n\t" +		            "popw %%es\n\t" +		            "popw %%ds\n\t" +		            "popal\n\t") +		            : : ); + +	if ( outregs_phys ) { +		memcpy_user ( phys_to_user ( outregs_phys ), 0, +		              virt_to_user( &com32_regs ), 0,  +		              sizeof(com32sys_t) ); +	} +} + +/** + * CDECL farcall helper + */ +int __asmcall com32_cfarcall ( uint32_t proc, physaddr_t stack, size_t stacksz ) { +	int32_t eax; + +	copy_user_to_rm_stack ( phys_to_user ( stack ), stacksz ); +	com32_farcall_proc = proc; + +	__asm__ __volatile__ ( +		REAL_CODE ( "lcall *%%ss:(com32_farcall_proc)\n\t" ) +		: "=a" (eax) +		:  +		: "ecx", "edx" ); + +	remove_user_from_rm_stack ( 0, stacksz ); + +	return eax; +} diff --git a/roms/ipxe/src/arch/i386/interface/syslinux/com32_wrapper.S b/roms/ipxe/src/arch/i386/interface/syslinux/com32_wrapper.S new file mode 100644 index 00000000..c9d1452b --- /dev/null +++ b/roms/ipxe/src/arch/i386/interface/syslinux/com32_wrapper.S @@ -0,0 +1,97 @@ +/* + * Copyright (C) 2008 Daniel Verkamp <daniel@drv.nu>. + * + * This program is free software; you can redistribute it and/or + * modify it under the terms of the GNU General Public License as + * published by the Free Software Foundation; either version 2 of the + * License, or any later version. + * + * 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. + * + * You should have received a copy of the GNU General Public License + * along with this program; if not, write to the Free Software + * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA + * 02110-1301, USA. + */ + +FILE_LICENCE ( GPL2_OR_LATER ) + +	.text +	.arch i386 +	.code32 + +	.globl com32_farcall_wrapper +com32_farcall_wrapper: + +	movl $com32_farcall, %eax +	jmp com32_wrapper + + +	.globl com32_cfarcall_wrapper +com32_cfarcall_wrapper: + +	movl $com32_cfarcall, %eax +	jmp com32_wrapper + + +	.globl com32_intcall_wrapper +com32_intcall_wrapper: + +	movl $com32_intcall, %eax +	/*jmp com32_wrapper*/ /* fall through */ + +com32_wrapper: +	cli + +	/* Switch to internal virtual address space */ +	call _phys_to_virt + +	mov %eax, (com32_helper_function) + +	/* Save external COM32 stack pointer */ +	movl %esp, (com32_external_esp) + +	/* Copy arguments to caller-save registers */ +	movl 12(%esp), %eax +	movl 8(%esp), %ecx +	movl 4(%esp), %edx + +	/* Switch to internal stack */ +	movl (com32_internal_esp), %esp + +	/* Copy arguments to internal stack */ +	pushl %eax +	pushl %ecx +	pushl %edx + +	call *(com32_helper_function) + +	/* Clean up stack */ +	addl $12, %esp + +	/* Save internal stack pointer and restore external stack pointer */ +	movl %esp, (com32_internal_esp) +	movl (com32_external_esp), %esp + +	/* Switch to external flat physical address space */ +	call _virt_to_phys + +	sti +	ret + + +	.data + +/* Internal iPXE virtual address space %esp */ +.globl com32_internal_esp +.lcomm com32_internal_esp, 4 + +/* External flat physical address space %esp */ +.globl com32_external_esp +.lcomm com32_external_esp, 4 + +/* Function pointer of helper to call */ +.lcomm com32_helper_function, 4 diff --git a/roms/ipxe/src/arch/i386/interface/syslinux/comboot_call.c b/roms/ipxe/src/arch/i386/interface/syslinux/comboot_call.c new file mode 100644 index 00000000..1854501d --- /dev/null +++ b/roms/ipxe/src/arch/i386/interface/syslinux/comboot_call.c @@ -0,0 +1,714 @@ +/* + * Copyright (C) 2008 Daniel Verkamp <daniel@drv.nu>. + * + * This program is free software; you can redistribute it and/or + * modify it under the terms of the GNU General Public License as + * published by the Free Software Foundation; either version 2 of the + * License, or any later version. + * + * 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. + * + * You should have received a copy of the GNU General Public License + * along with this program; if not, write to the Free Software + * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA + * 02110-1301, USA. + */ + +/** + * @file SYSLINUX COMBOOT API + * + */ + +FILE_LICENCE ( GPL2_OR_LATER ); + +#include <errno.h> +#include <realmode.h> +#include <biosint.h> +#include <ipxe/console.h> +#include <stdlib.h> +#include <comboot.h> +#include <bzimage.h> +#include <pxe_call.h> +#include <setjmp.h> +#include <string.h> +#include <ipxe/posix_io.h> +#include <ipxe/process.h> +#include <ipxe/serial.h> +#include <ipxe/init.h> +#include <ipxe/image.h> +#include <ipxe/version.h> +#include <usr/imgmgmt.h> +#include "config/console.h" +#include "config/serial.h" + +/** The "SYSLINUX" version string */ +static char __bss16_array ( syslinux_version, [32] ); +#define syslinux_version __use_data16 ( syslinux_version ) + +/** The "SYSLINUX" copyright string */ +static char __data16_array ( syslinux_copyright, [] ) = " http://ipxe.org"; +#define syslinux_copyright __use_data16 ( syslinux_copyright ) + +static char __data16_array ( syslinux_configuration_file, [] ) = ""; +#define syslinux_configuration_file __use_data16 ( syslinux_configuration_file ) + +/** Feature flags */ +static uint8_t __data16 ( comboot_feature_flags ) = COMBOOT_FEATURE_IDLE_LOOP; +#define comboot_feature_flags __use_data16 ( comboot_feature_flags ) + +typedef union { +	syslinux_pm_regs pm; syslinux_rm_regs rm; +} syslinux_regs; + +/** Initial register values for INT 22h AX=1Ah and 1Bh */ +static syslinux_regs __text16 ( comboot_initial_regs ); +#define comboot_initial_regs __use_text16 ( comboot_initial_regs ) + +static struct segoff __text16 ( int20_vector ); +#define int20_vector __use_text16 ( int20_vector ) + +static struct segoff __text16 ( int21_vector ); +#define int21_vector __use_text16 ( int21_vector ) + +static struct segoff __text16 ( int22_vector ); +#define int22_vector __use_text16 ( int22_vector ) + +extern void int20_wrapper ( void ); +extern void int21_wrapper ( void ); +extern void int22_wrapper ( void ); + +/* setjmp/longjmp context buffer used to return after loading an image */ +rmjmp_buf comboot_return; + +/* Mode flags set by INT 22h AX=0017h */ +static uint16_t comboot_graphics_mode = 0; + + +/** + * Print a string with a particular terminator + */ +static void print_user_string ( unsigned int segment, unsigned int offset, char terminator ) { +	int i = 0; +	char c; +	userptr_t str = real_to_user ( segment, offset ); +	for ( ; ; ) { +		copy_from_user ( &c, str, i, 1 ); +		if ( c == terminator ) break; +		putchar ( c ); +		i++; +	} +} + + +/** + * Perform a series of memory copies from a list in low memory + */ +static void shuffle ( unsigned int list_segment, unsigned int list_offset, unsigned int count ) +{ +	comboot_shuffle_descriptor shuf[COMBOOT_MAX_SHUFFLE_DESCRIPTORS]; +	unsigned int i; + +	/* Copy shuffle descriptor list so it doesn't get overwritten */ +	copy_from_user ( shuf, real_to_user ( list_segment, list_offset ), 0, +	                 count * sizeof( comboot_shuffle_descriptor ) ); + +	/* Do the copies */ +	for ( i = 0; i < count; i++ ) { +		userptr_t src_u = phys_to_user ( shuf[ i ].src ); +		userptr_t dest_u = phys_to_user ( shuf[ i ].dest ); + +		if ( shuf[ i ].src == 0xFFFFFFFF ) { +			/* Fill with 0 instead of copying */ +			memset_user ( dest_u, 0, 0, shuf[ i ].len ); +		} else if ( shuf[ i ].dest == 0xFFFFFFFF ) { +			/* Copy new list of descriptors */ +			count = shuf[ i ].len / sizeof( comboot_shuffle_descriptor ); +			assert ( count <= COMBOOT_MAX_SHUFFLE_DESCRIPTORS ); +			copy_from_user ( shuf, src_u, 0, shuf[ i ].len ); +			i = -1; +		} else { +			/* Regular copy */ +			memmove_user ( dest_u, 0, src_u, 0, shuf[ i ].len ); +		} +	} +} + + +/** + * Set default text mode + */ +void comboot_force_text_mode ( void ) { +	if ( comboot_graphics_mode & COMBOOT_VIDEO_VESA ) { +		/* Set VGA mode 3 via VESA VBE mode set */ +		__asm__ __volatile__ ( +			REAL_CODE ( +				"mov $0x4F02, %%ax\n\t" +				"mov $0x03, %%bx\n\t" +				"int $0x10\n\t" +			) +		: : ); +	} else if ( comboot_graphics_mode & COMBOOT_VIDEO_GRAPHICS ) { +		/* Set VGA mode 3 via standard VGA mode set */ +		__asm__ __volatile__ ( +			REAL_CODE ( +				"mov $0x03, %%ax\n\t" +				"int $0x10\n\t" +			) +		: : ); +	} + +	comboot_graphics_mode = 0; +} + + +/** + * Fetch kernel and optional initrd + */ +static int comboot_fetch_kernel ( char *kernel_file, char *cmdline ) { +	struct image *kernel; +	struct image *initrd; +	char *initrd_file; +	int rc; + +	/* Find initrd= parameter, if any */ +	if ( ( initrd_file = strstr ( cmdline, "initrd=" ) ) != NULL ) { +		char *initrd_end; + +		/* skip "initrd=" */ +		initrd_file += 7; + +		/* Find terminating space, if any, and replace with NUL */ +		initrd_end = strchr ( initrd_file, ' ' ); +		if ( initrd_end ) +			*initrd_end = '\0'; + +		DBG ( "COMBOOT: fetching initrd '%s'\n", initrd_file ); + +		/* Fetch initrd */ +		if ( ( rc = imgdownload_string ( initrd_file, 0, +						 &initrd ) ) != 0 ) { +			DBG ( "COMBOOT: could not fetch initrd: %s\n", +			      strerror ( rc ) ); +			return rc; +		} + +		/* Restore space after initrd name, if applicable */ +		if ( initrd_end ) +			*initrd_end = ' '; +	} + +	DBG ( "COMBOOT: fetching kernel '%s'\n", kernel_file ); + +	/* Fetch kernel */ +	if ( ( rc = imgdownload_string ( kernel_file, 0, &kernel ) ) != 0 ) { +		DBG ( "COMBOOT: could not fetch kernel: %s\n", +		      strerror ( rc ) ); +		return rc; +	} + +	/* Replace comboot image with kernel */ +	if ( ( rc = image_replace ( kernel ) ) != 0 ) { +		DBG ( "COMBOOT: could not replace with kernel: %s\n", +		      strerror ( rc ) ); +		return rc; +	} + +	return 0; +} + + +/** + * Terminate program interrupt handler + */ +static __asmcall void int20 ( struct i386_all_regs *ix86 __unused ) { +	rmlongjmp ( comboot_return, COMBOOT_EXIT ); +} + + +/** + * DOS-compatible API + */ +static __asmcall void int21 ( struct i386_all_regs *ix86 ) { +	ix86->flags |= CF; + +	switch ( ix86->regs.ah ) { +	case 0x00: +	case 0x4C: /* Terminate program */ +		rmlongjmp ( comboot_return, COMBOOT_EXIT ); +		break; + +	case 0x01: /* Get Key with Echo */ +	case 0x08: /* Get Key without Echo */ +		/* TODO: handle extended characters? */ +		ix86->regs.al = getchar( ); + +		/* Enter */ +		if ( ix86->regs.al == 0x0A ) +			ix86->regs.al = 0x0D; + +		if ( ix86->regs.ah == 0x01 ) +			putchar ( ix86->regs.al ); + +		ix86->flags &= ~CF; +		break; + +	case 0x02: /* Write Character */ +		putchar ( ix86->regs.dl ); +		ix86->flags &= ~CF; +		break; + +	case 0x04: /* Write Character to Serial Port */ +		serial_putc ( ix86->regs.dl ); +		ix86->flags &= ~CF; +		break; + +	case 0x09: /* Write DOS String to Console */ +		print_user_string ( ix86->segs.ds, ix86->regs.dx, '$' ); +		ix86->flags &= ~CF; +		break; + +	case 0x0B: /* Check Keyboard */ +		if ( iskey() ) +			ix86->regs.al = 0xFF; +		else +			ix86->regs.al = 0x00; + +		ix86->flags &= ~CF; +		break; + +	case 0x30: /* Check DOS Version */ +		/* Bottom halves all 0; top halves spell "SYSLINUX" */ +		ix86->regs.eax = 0x59530000; +		ix86->regs.ebx = 0x4C530000; +		ix86->regs.ecx = 0x4E490000; +		ix86->regs.edx = 0x58550000; +		ix86->flags &= ~CF; +		break; + +	default: +		DBG ( "COMBOOT unknown int21 function %02x\n", ix86->regs.ah ); +		break; +	} +} + + +/** + * Dispatch PXE API call weakly + * + * @v ix86		Registers for PXE call + * @ret present		Zero if the PXE stack is present, nonzero if not + * + * A successful return only indicates that the PXE stack was available + * for dispatching the call; it says nothing about the success of + * whatever the call asked for. + */ +__weak int pxe_api_call_weak ( struct i386_all_regs *ix86 __unused ) { +	return -1; +} + +/** + * SYSLINUX API + */ +static __asmcall void int22 ( struct i386_all_regs *ix86 ) { +	ix86->flags |= CF; + +	switch ( ix86->regs.ax ) { +	case 0x0001: /* Get Version */ + +		/* Number of INT 22h API functions available */ +		ix86->regs.ax = 0x001D; + +		/* SYSLINUX version number */ +		ix86->regs.ch = 0; /* major */ +		ix86->regs.cl = 0; /* minor */ + +		/* SYSLINUX derivative ID */ +		ix86->regs.dl = BZI_LOADER_TYPE_IPXE; + +		/* SYSLINUX version */ +		snprintf ( syslinux_version, sizeof ( syslinux_version ), +			   "\r\niPXE %s", product_version ); + +		/* SYSLINUX version and copyright strings */ +		ix86->segs.es = rm_ds; +		ix86->regs.si = ( ( unsigned ) __from_data16 ( syslinux_version ) ); +		ix86->regs.di = ( ( unsigned ) __from_data16 ( syslinux_copyright ) ); + +		ix86->flags &= ~CF; +		break; + +	case 0x0002: /* Write String */ +		print_user_string ( ix86->segs.es, ix86->regs.bx, '\0' ); +		ix86->flags &= ~CF; +		break; + +	case 0x0003: /* Run command */ +		{ +			userptr_t cmd_u = real_to_user ( ix86->segs.es, ix86->regs.bx ); +			int len = strlen_user ( cmd_u, 0 ); +			char cmd[len + 1]; +			copy_from_user ( cmd, cmd_u, 0, len + 1 ); +			DBG ( "COMBOOT: executing command '%s'\n", cmd ); +			system ( cmd ); +			DBG ( "COMBOOT: exiting after executing command...\n" ); +			rmlongjmp ( comboot_return, COMBOOT_EXIT_COMMAND ); +		} +		break; + +	case 0x0004: /* Run default command */ +		/* FIXME: just exit for now */ +		rmlongjmp ( comboot_return, COMBOOT_EXIT_COMMAND ); +		break; + +	case 0x0005: /* Force text mode */ +		comboot_force_text_mode ( ); +		ix86->flags &= ~CF; +		break; + +	case 0x0006: /* Open file */ +		{ +			int fd; +			userptr_t file_u = real_to_user ( ix86->segs.es, ix86->regs.si ); +			int len = strlen_user ( file_u, 0 ); +			char file[len + 1]; + +			copy_from_user ( file, file_u, 0, len + 1 ); + +			if ( file[0] == '\0' ) { +				DBG ( "COMBOOT: attempted open with empty file name\n" ); +				break; +			} + +			DBG ( "COMBOOT: opening file '%s'\n", file ); + +			fd = open ( file ); + +			if ( fd < 0 ) { +				DBG ( "COMBOOT: error opening file %s\n", file ); +				break; +			} + +			/* This relies on the fact that a iPXE POSIX fd will +			 * always fit in 16 bits. +			 */ +#if (POSIX_FD_MAX > 65535) +#error POSIX_FD_MAX too large +#endif +			ix86->regs.si = (uint16_t) fd; + +			ix86->regs.cx = COMBOOT_FILE_BLOCKSZ; +			ix86->regs.eax = fsize ( fd ); +			ix86->flags &= ~CF; +		} +		break; + +	case 0x0007: /* Read file */ +		{ +			int fd = ix86->regs.si; +			int len = ix86->regs.cx * COMBOOT_FILE_BLOCKSZ; +			int rc; +			fd_set fds; +			userptr_t buf = real_to_user ( ix86->segs.es, ix86->regs.bx ); + +			/* Wait for data ready to read */ +			FD_ZERO ( &fds ); +			FD_SET ( fd, &fds ); + +			select ( &fds, 1 ); + +			rc = read_user ( fd, buf, 0, len ); +			if ( rc < 0 ) { +				DBG ( "COMBOOT: read failed\n" ); +				ix86->regs.si = 0; +				break; +			} + +			ix86->regs.ecx = rc; +			ix86->flags &= ~CF; +		} +		break; + +	case 0x0008: /* Close file */ +		{ +			int fd = ix86->regs.si; +			close ( fd ); +			ix86->flags &= ~CF; +		} +		break; + +	case 0x0009: /* Call PXE Stack */ +		if ( pxe_api_call_weak ( ix86 ) != 0 ) +			ix86->flags |= CF; +		else +			ix86->flags &= ~CF; +		break; + +	case 0x000A: /* Get Derivative-Specific Information */ + +		/* iPXE has its own derivative ID, so there is no defined +		 * output here; just return AL for now */ +		ix86->regs.al = BZI_LOADER_TYPE_IPXE; +		ix86->flags &= ~CF; +		break; + +	case 0x000B: /* Get Serial Console Configuration */ +#if defined(CONSOLE_SERIAL) && !defined(COMPRESERVE) +		ix86->regs.dx = COMCONSOLE; +		ix86->regs.cx = 115200 / COMSPEED; +		ix86->regs.bx = 0; +#else +		ix86->regs.dx = 0; +#endif + +		ix86->flags &= ~CF; +		break; + +	case 0x000E: /* Get configuration file name */ +		/* FIXME: stub */ +		ix86->segs.es = rm_ds; +		ix86->regs.bx = ( ( unsigned ) __from_data16 ( syslinux_configuration_file ) ); +		ix86->flags &= ~CF; +		break; + +	case 0x000F: /* Get IPAPPEND strings */ +		/* FIXME: stub */ +		ix86->regs.cx = 0; +		ix86->segs.es = 0; +		ix86->regs.bx = 0; +		ix86->flags &= ~CF; +		break; + +	case 0x0010: /* Resolve hostname */ +		{ +			userptr_t hostname_u = real_to_user ( ix86->segs.es, ix86->regs.bx ); +			int len = strlen_user ( hostname_u, 0 ); +			char hostname[len]; +			struct in_addr addr; + +			copy_from_user ( hostname, hostname_u, 0, len + 1 ); +			 +			/* TODO: +			 * "If the hostname does not contain a dot (.), the +			 * local domain name is automatically appended." +			 */ + +			comboot_resolv ( hostname, &addr ); + +			ix86->regs.eax = addr.s_addr; +			ix86->flags &= ~CF; +		} +		break; + +	case 0x0011: /* Maximum number of shuffle descriptors */ +		ix86->regs.cx = COMBOOT_MAX_SHUFFLE_DESCRIPTORS; +		ix86->flags &= ~CF; +		break; + +	case 0x0012: /* Cleanup, shuffle and boot */ +		if ( ix86->regs.cx > COMBOOT_MAX_SHUFFLE_DESCRIPTORS ) +			break; + +		/* Perform final cleanup */ +		shutdown_boot(); + +		/* Perform sequence of copies */ +		shuffle ( ix86->segs.es, ix86->regs.di, ix86->regs.cx ); + +		/* Jump to real-mode entry point */ +		__asm__ __volatile__ ( +			REAL_CODE (  +				"pushw %0\n\t" +				"popw %%ds\n\t" +				"pushl %1\n\t" +				"lret\n\t" +			) +			: +			: "r" ( ix86->segs.ds ), +			  "r" ( ix86->regs.ebp ), +			  "d" ( ix86->regs.ebx ), +			  "S" ( ix86->regs.esi ) ); + +		assert ( 0 ); /* Execution should never reach this point */ + +		break; + +	case 0x0013: /* Idle loop call */ +		step ( ); +		ix86->flags &= ~CF; +		break; + +	case 0x0015: /* Get feature flags */ +		ix86->segs.es = rm_ds; +		ix86->regs.bx = ( ( unsigned ) __from_data16 ( &comboot_feature_flags ) ); +		ix86->regs.cx = 1; /* Number of feature flag bytes */ +		ix86->flags &= ~CF; +		break; + +	case 0x0016: /* Run kernel image */ +		{ +			userptr_t file_u = real_to_user ( ix86->segs.ds, ix86->regs.si ); +			userptr_t cmd_u = real_to_user ( ix86->segs.es, ix86->regs.bx ); +			int file_len = strlen_user ( file_u, 0 ); +			int cmd_len = strlen_user ( cmd_u, 0 ); +			char file[file_len + 1]; +			char cmd[cmd_len + 1]; + +			copy_from_user ( file, file_u, 0, file_len + 1 ); +			copy_from_user ( cmd, cmd_u, 0, cmd_len + 1 ); + +			DBG ( "COMBOOT: run kernel %s %s\n", file, cmd ); +			comboot_fetch_kernel ( file, cmd ); +			/* Technically, we should return if we +			 * couldn't load the kernel, but it's not safe +			 * to do that since we have just overwritten +			 * part of the COMBOOT program's memory space. +			 */ +			DBG ( "COMBOOT: exiting to run kernel...\n" ); +			rmlongjmp ( comboot_return, COMBOOT_EXIT_RUN_KERNEL ); +		} +		break; + +	case 0x0017: /* Report video mode change */ +		comboot_graphics_mode = ix86->regs.bx; +		ix86->flags &= ~CF; +		break; + +	case 0x0018: /* Query custom font */ +		/* FIXME: stub */ +		ix86->regs.al = 0; +		ix86->segs.es = 0; +		ix86->regs.bx = 0; +		ix86->flags &= ~CF; +		break; + +	case 0x001B: /* Cleanup, shuffle and boot to real mode */ +		if ( ix86->regs.cx > COMBOOT_MAX_SHUFFLE_DESCRIPTORS ) +			break; + +		/* Perform final cleanup */ +		shutdown_boot(); + +		/* Perform sequence of copies */ +		shuffle ( ix86->segs.es, ix86->regs.di, ix86->regs.cx ); + +		/* Copy initial register values to .text16 */ +		memcpy_user ( real_to_user ( rm_cs, (unsigned) __from_text16 ( &comboot_initial_regs ) ), 0, +		              real_to_user ( ix86->segs.ds, ix86->regs.si ), 0, +		              sizeof(syslinux_rm_regs) ); + +		/* Load initial register values */ +		__asm__ __volatile__ ( +			REAL_CODE ( +				/* Point SS:SP at the register value structure */ +				"pushw %%cs\n\t" +				"popw %%ss\n\t" +				"movw $comboot_initial_regs, %%sp\n\t" + +				/* Segment registers */ +				"popw %%es\n\t" +				"popw %%ax\n\t" /* Skip CS */ +				"popw %%ds\n\t" +				"popw %%ax\n\t" /* Skip SS for now */ +				"popw %%fs\n\t" +				"popw %%gs\n\t" + +				/* GP registers */ +				"popl %%eax\n\t" +				"popl %%ecx\n\t" +				"popl %%edx\n\t" +				"popl %%ebx\n\t" +				"popl %%ebp\n\t" /* Skip ESP for now */ +				"popl %%ebp\n\t" +				"popl %%esi\n\t" +				"popl %%edi\n\t" + +				/* Load correct SS:ESP */ +				"movw $(comboot_initial_regs + 6), %%sp\n\t" +				"popw %%ss\n\t" +				"movl %%cs:(comboot_initial_regs + 28), %%esp\n\t" + +				"ljmp *%%cs:(comboot_initial_regs + 44)\n\t" +			) +			: : ); + +		break; + +	case 0x001C: /* Get pointer to auxilliary data vector */ +		/* FIXME: stub */ +		ix86->regs.cx = 0; /* Size of the ADV */ +		ix86->flags &= ~CF; +		break; + +	case 0x001D: /* Write auxilliary data vector */ +		/* FIXME: stub */ +		ix86->flags &= ~CF; +		break; + +	default: +		DBG ( "COMBOOT unknown int22 function %04x\n", ix86->regs.ax ); +		break; +	} +} + +/** + * Hook BIOS interrupts related to COMBOOT API (INT 20h, 21h, 22h) + */ +void hook_comboot_interrupts ( ) { + +	__asm__ __volatile__ ( +		TEXT16_CODE ( "\nint20_wrapper:\n\t" +		              "pushl %0\n\t" +		              "pushw %%cs\n\t" +		              "call prot_call\n\t" +		              "addw $4, %%sp\n\t" +			      "call patch_cf\n\t" +		              "iret\n\t" ) +		          : : "i" ( int20 ) ); + +	hook_bios_interrupt ( 0x20, ( unsigned int ) int20_wrapper, +		                      &int20_vector ); + +	__asm__ __volatile__ ( +		TEXT16_CODE ( "\nint21_wrapper:\n\t" +		              "pushl %0\n\t" +		              "pushw %%cs\n\t" +		              "call prot_call\n\t" +		              "addw $4, %%sp\n\t" +			      "call patch_cf\n\t" +		              "iret\n\t" ) +		          : : "i" ( int21 ) ); + +	hook_bios_interrupt ( 0x21, ( unsigned int ) int21_wrapper, +	                      &int21_vector ); + +	__asm__  __volatile__ ( +		TEXT16_CODE ( "\nint22_wrapper:\n\t" +		              "pushl %0\n\t" +		              "pushw %%cs\n\t" +		              "call prot_call\n\t" +		              "addw $4, %%sp\n\t" +			      "call patch_cf\n\t" +		              "iret\n\t" ) +		          : : "i" ( int22) ); + +	hook_bios_interrupt ( 0x22, ( unsigned int ) int22_wrapper, +	                      &int22_vector ); +} + +/** + * Unhook BIOS interrupts related to COMBOOT API (INT 20h, 21h, 22h) + */ +void unhook_comboot_interrupts ( ) { + +	unhook_bios_interrupt ( 0x20, ( unsigned int ) int20_wrapper, +				&int20_vector ); + +	unhook_bios_interrupt ( 0x21, ( unsigned int ) int21_wrapper, +				&int21_vector ); + +	unhook_bios_interrupt ( 0x22, ( unsigned int ) int22_wrapper, +				&int22_vector ); +} diff --git a/roms/ipxe/src/arch/i386/interface/syslinux/comboot_resolv.c b/roms/ipxe/src/arch/i386/interface/syslinux/comboot_resolv.c new file mode 100644 index 00000000..03bbfd04 --- /dev/null +++ b/roms/ipxe/src/arch/i386/interface/syslinux/comboot_resolv.c @@ -0,0 +1,61 @@ +#include <errno.h> +#include <comboot.h> +#include <ipxe/in.h> +#include <ipxe/list.h> +#include <ipxe/process.h> +#include <ipxe/resolv.h> + +FILE_LICENCE ( GPL2_OR_LATER ); + +struct comboot_resolver { +	struct interface intf; +	int rc; +	struct in_addr addr; +}; + +static void comboot_resolv_close ( struct comboot_resolver *comboot_resolver, +				   int rc ) { +	comboot_resolver->rc = rc; +	intf_shutdown ( &comboot_resolver->intf, rc ); +} + +static void comboot_resolv_done ( struct comboot_resolver *comboot_resolver, +				  struct sockaddr *sa ) { +	struct sockaddr_in *sin; + +	if ( sa->sa_family == AF_INET ) { +		sin = ( ( struct sockaddr_in * ) sa ); +		comboot_resolver->addr = sin->sin_addr; +	} +} + +static struct interface_operation comboot_resolv_op[] = { +	INTF_OP ( intf_close, struct comboot_resolver *, comboot_resolv_close ), +	INTF_OP ( resolv_done, struct comboot_resolver *, comboot_resolv_done ), +}; + +static struct interface_descriptor comboot_resolv_desc = +	INTF_DESC ( struct comboot_resolver, intf, comboot_resolv_op ); + +static struct comboot_resolver comboot_resolver = { +	.intf = INTF_INIT ( comboot_resolv_desc ), +}; + +int comboot_resolv ( const char *name, struct in_addr *address ) { +	int rc; + +	comboot_resolver.rc = -EINPROGRESS; +	comboot_resolver.addr.s_addr = 0; + +	if ( ( rc = resolv ( &comboot_resolver.intf, name, NULL ) ) != 0 ) +		return rc; + +	while ( comboot_resolver.rc == -EINPROGRESS ) +		step(); + +	if ( ! comboot_resolver.addr.s_addr ) +		return -EAFNOSUPPORT; + +	*address = comboot_resolver.addr; +	return comboot_resolver.rc; +} diff --git a/roms/ipxe/src/arch/i386/interface/vmware/guestinfo.c b/roms/ipxe/src/arch/i386/interface/vmware/guestinfo.c new file mode 100644 index 00000000..a0530c8d --- /dev/null +++ b/roms/ipxe/src/arch/i386/interface/vmware/guestinfo.c @@ -0,0 +1,271 @@ +/* + * Copyright (C) 2012 Michael Brown <mbrown@fensystems.co.uk>. + * + * This program is free software; you can redistribute it and/or + * modify it under the terms of the GNU General Public License as + * published by the Free Software Foundation; either version 2 of the + * License, or any later version. + * + * 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. + * + * You should have received a copy of the GNU General Public License + * along with this program; if not, write to the Free Software + * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA + * 02110-1301, USA. + */ + +FILE_LICENCE ( GPL2_OR_LATER ); + +/** @file + * + * VMware GuestInfo settings + * + */ + +#include <stdint.h> +#include <stdlib.h> +#include <stdio.h> +#include <string.h> +#include <errno.h> +#include <ipxe/init.h> +#include <ipxe/settings.h> +#include <ipxe/netdevice.h> +#include <ipxe/guestrpc.h> + +/** GuestInfo GuestRPC channel */ +static int guestinfo_channel; + +/** + * Fetch value of typed GuestInfo setting + * + * @v settings		Settings block + * @v setting		Setting to fetch + * @v type		Setting type to attempt (or NULL for default) + * @v data		Buffer to fill with setting data + * @v len		Length of buffer + * @ret found		Setting found in GuestInfo + * @ret len		Length of setting data, or negative error + */ +static int guestinfo_fetch_type ( struct settings *settings, +				  struct setting *setting, +				  const struct setting_type *type, +				  void *data, size_t len, int *found ) { +	const char *parent_name = settings->parent->name; +	char command[ 24 /* "info-get guestinfo.ipxe." */ + +		      strlen ( parent_name ) + 1 /* "." */ + +		      strlen ( setting->name ) + 1 /* "." */ + +		      ( type ? strlen ( type->name ) : 0 ) + 1 /* NUL */ ]; +	struct setting *predefined; +	char *info; +	int info_len; +	int check_len; +	int ret; + +	/* Construct info-get command */ +	snprintf ( command, sizeof ( command ), +		   "info-get guestinfo.ipxe.%s%s%s%s%s", +		   parent_name, ( parent_name[0] ? "." : "" ), setting->name, +		   ( type ? "." : "" ), ( type ? type->name : "" ) ); + +	/* Check for existence and obtain length of GuestInfo value */ +	info_len = guestrpc_command ( guestinfo_channel, command, NULL, 0 ); +	if ( info_len < 0 ) { +		ret = info_len; +		goto err_get_info_len; +	} + +	/* Mark as found */ +	*found = 1; + +	/* Determine default type if necessary */ +	if ( ! type ) { +		predefined = find_setting ( setting->name ); +		type = ( predefined ? predefined->type : &setting_type_string ); +	} +	assert ( type != NULL ); + +	/* Allocate temporary block to hold GuestInfo value */ +	info = zalloc ( info_len + 1 /* NUL */ ); +	if ( ! info ) { +		DBGC ( settings, "GuestInfo %p could not allocate %d bytes\n", +		       settings, info_len ); +		ret = -ENOMEM; +		goto err_alloc; +	} +	info[info_len] = '\0'; + +	/* Fetch GuestInfo value */ +	check_len = guestrpc_command ( guestinfo_channel, command, +				       info, info_len ); +	if ( check_len < 0 ) { +		ret = check_len; +		goto err_get_info; +	} +	if ( check_len != info_len ) { +		DBGC ( settings, "GuestInfo %p length mismatch (expected %d, " +		       "got %d)\n", settings, info_len, check_len ); +		ret = -EIO; +		goto err_get_info; +	} +	DBGC2 ( settings, "GuestInfo %p found %s = \"%s\"\n", +		settings, &command[9] /* Skip "info-get " */, info ); + +	/* Parse GuestInfo value according to type */ +	ret = setting_parse ( type, info, data, len ); +	if ( ret < 0 ) { +		DBGC ( settings, "GuestInfo %p could not parse \"%s\" as %s: " +		       "%s\n", settings, info, type->name, strerror ( ret ) ); +		goto err_parse; +	} + + err_parse: + err_get_info: +	free ( info ); + err_alloc: + err_get_info_len: +	return ret; +} + +/** + * Fetch value of GuestInfo setting + * + * @v settings		Settings block + * @v setting		Setting to fetch + * @v data		Buffer to fill with setting data + * @v len		Length of buffer + * @ret len		Length of setting data, or negative error + */ +static int guestinfo_fetch ( struct settings *settings, +			     struct setting *setting, +			     void *data, size_t len ) { +	struct setting_type *type; +	int found = 0; +	int ret; + +	/* Try default type first */ +	ret = guestinfo_fetch_type ( settings, setting, NULL, +				     data, len, &found ); +	if ( found ) +		return ret; + +	/* Otherwise, try all possible types */ +	for_each_table_entry ( type, SETTING_TYPES ) { +		ret = guestinfo_fetch_type ( settings, setting, type, +					     data, len, &found ); +		if ( found ) +			return ret; +	} + +	/* Not found */ +	return -ENOENT; +} + +/** GuestInfo settings operations */ +static struct settings_operations guestinfo_settings_operations = { +	.fetch = guestinfo_fetch, +}; + +/** GuestInfo settings */ +static struct settings guestinfo_settings = { +	.refcnt = NULL, +	.siblings = LIST_HEAD_INIT ( guestinfo_settings.siblings ), +	.children = LIST_HEAD_INIT ( guestinfo_settings.children ), +	.op = &guestinfo_settings_operations, +}; + +/** Initialise GuestInfo settings */ +static void guestinfo_init ( void ) { +	int rc; + +	/* Open GuestRPC channel */ +	guestinfo_channel = guestrpc_open(); +	if ( guestinfo_channel < 0 ) { +		rc = guestinfo_channel; +		DBG ( "GuestInfo could not open channel: %s\n", +		      strerror ( rc ) ); +		return; +	} + +	/* Register root GuestInfo settings */ +	if ( ( rc = register_settings ( &guestinfo_settings, NULL, +					"vmware" ) ) != 0 ) { +		DBG ( "GuestInfo could not register settings: %s\n", +		      strerror ( rc ) ); +		return; +	} +} + +/** GuestInfo settings initialiser */ +struct init_fn guestinfo_init_fn __init_fn ( INIT_NORMAL ) = { +	.initialise = guestinfo_init, +}; + +/** + * Create per-netdevice GuestInfo settings + * + * @v netdev		Network device + * @ret rc		Return status code + */ +static int guestinfo_net_probe ( struct net_device *netdev ) { +	struct settings *settings; +	int rc; + +	/* Do nothing unless we have a GuestInfo channel available */ +	if ( guestinfo_channel < 0 ) +		return 0; + +	/* Allocate and initialise settings block */ +	settings = zalloc ( sizeof ( *settings ) ); +	if ( ! settings ) { +		rc = -ENOMEM; +		goto err_alloc; +	} +	settings_init ( settings, &guestinfo_settings_operations, NULL, NULL ); + +	/* Register settings */ +	if ( ( rc = register_settings ( settings, netdev_settings ( netdev ), +					"vmware" ) ) != 0 ) { +		DBGC ( settings, "GuestInfo %p could not register for %s: %s\n", +		       settings, netdev->name, strerror ( rc ) ); +		goto err_register; +	} +	DBGC ( settings, "GuestInfo %p registered for %s\n", +	       settings, netdev->name ); + +	return 0; + + err_register: +	free ( settings ); + err_alloc: +	return rc; +} + +/** + * Remove per-netdevice GuestInfo settings + * + * @v netdev		Network device + */ +static void guestinfo_net_remove ( struct net_device *netdev ) { +	struct settings *parent = netdev_settings ( netdev ); +	struct settings *settings; + +	list_for_each_entry ( settings, &parent->children, siblings ) { +		if ( settings->op == &guestinfo_settings_operations ) { +			DBGC ( settings, "GuestInfo %p unregistered for %s\n", +			       settings, netdev->name ); +			unregister_settings ( settings ); +			free ( settings ); +			return; +		} +	} +} + +/** GuestInfo per-netdevice driver */ +struct net_driver guestinfo_net_driver __net_driver = { +	.name = "GuestInfo", +	.probe = guestinfo_net_probe, +	.remove = guestinfo_net_remove, +}; diff --git a/roms/ipxe/src/arch/i386/interface/vmware/guestrpc.c b/roms/ipxe/src/arch/i386/interface/vmware/guestrpc.c new file mode 100644 index 00000000..390fc554 --- /dev/null +++ b/roms/ipxe/src/arch/i386/interface/vmware/guestrpc.c @@ -0,0 +1,328 @@ +/* + * Copyright (C) 2012 Michael Brown <mbrown@fensystems.co.uk>. + * + * This program is free software; you can redistribute it and/or + * modify it under the terms of the GNU General Public License as + * published by the Free Software Foundation; either version 2 of the + * License, or any later version. + * + * 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. + * + * You should have received a copy of the GNU General Public License + * along with this program; if not, write to the Free Software + * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA + * 02110-1301, USA. + */ + +FILE_LICENCE ( GPL2_OR_LATER ); + +/** @file + * + * VMware GuestRPC mechanism + * + */ + +#include <stdint.h> +#include <string.h> +#include <errno.h> +#include <assert.h> +#include <ipxe/vmware.h> +#include <ipxe/guestrpc.h> + +/* Disambiguate the various error causes */ +#define EPROTO_OPEN __einfo_error ( EINFO_EPROTO_OPEN ) +#define EINFO_EPROTO_OPEN \ +	__einfo_uniqify ( EINFO_EPROTO, 0x00, "GuestRPC open failed" ) +#define EPROTO_COMMAND_LEN __einfo_error ( EINFO_EPROTO_COMMAND_LEN ) +#define EINFO_EPROTO_COMMAND_LEN \ +	__einfo_uniqify ( EINFO_EPROTO, 0x01, "GuestRPC command length failed" ) +#define EPROTO_COMMAND_DATA __einfo_error ( EINFO_EPROTO_COMMAND_DATA ) +#define EINFO_EPROTO_COMMAND_DATA \ +	__einfo_uniqify ( EINFO_EPROTO, 0x02, "GuestRPC command data failed" ) +#define EPROTO_REPLY_LEN __einfo_error ( EINFO_EPROTO_REPLY_LEN ) +#define EINFO_EPROTO_REPLY_LEN \ +	__einfo_uniqify ( EINFO_EPROTO, 0x03, "GuestRPC reply length failed" ) +#define EPROTO_REPLY_DATA __einfo_error ( EINFO_EPROTO_REPLY_DATA ) +#define EINFO_EPROTO_REPLY_DATA \ +	__einfo_uniqify ( EINFO_EPROTO, 0x04, "GuestRPC reply data failed" ) +#define EPROTO_REPLY_FINISH __einfo_error ( EINFO_EPROTO_REPLY_FINISH ) +#define EINFO_EPROTO_REPLY_FINISH \ +	__einfo_uniqify ( EINFO_EPROTO, 0x05, "GuestRPC reply finish failed" ) +#define EPROTO_CLOSE __einfo_error ( EINFO_EPROTO_CLOSE ) +#define EINFO_EPROTO_CLOSE \ +	__einfo_uniqify ( EINFO_EPROTO, 0x06, "GuestRPC close failed" ) + +/** + * Open GuestRPC channel + * + * @ret channel		Channel number, or negative error + */ +int guestrpc_open ( void ) { +	uint16_t channel; +	uint32_t discard_b; +	uint32_t status; + +	/* Issue GuestRPC command */ +	status = vmware_cmd_guestrpc ( 0, GUESTRPC_OPEN, GUESTRPC_MAGIC, +				       &channel, &discard_b ); +	if ( status != GUESTRPC_OPEN_SUCCESS ) { +		DBGC ( GUESTRPC_MAGIC, "GuestRPC open failed: status %08x\n", +		       status ); +		return -EPROTO_OPEN; +	} + +	DBGC ( GUESTRPC_MAGIC, "GuestRPC channel %d opened\n", channel ); +	return channel; +} + +/** + * Send GuestRPC command length + * + * @v channel		Channel number + * @v len		Command length + * @ret rc		Return status code + */ +static int guestrpc_command_len ( int channel, size_t len ) { +	uint16_t discard_d; +	uint32_t discard_b; +	uint32_t status; + +	/* Issue GuestRPC command */ +	status = vmware_cmd_guestrpc ( channel, GUESTRPC_COMMAND_LEN, len, +				       &discard_d, &discard_b ); +	if ( status != GUESTRPC_COMMAND_LEN_SUCCESS ) { +		DBGC ( GUESTRPC_MAGIC, "GuestRPC channel %d send command " +		       "length %zd failed: status %08x\n", +		       channel, len, status ); +		return -EPROTO_COMMAND_LEN; +	} + +	return 0; +} + +/** + * Send GuestRPC command data + * + * @v channel		Channel number + * @v data		Command data + * @ret rc		Return status code + */ +static int guestrpc_command_data ( int channel, uint32_t data ) { +	uint16_t discard_d; +	uint32_t discard_b; +	uint32_t status; + +	/* Issue GuestRPC command */ +	status = vmware_cmd_guestrpc ( channel, GUESTRPC_COMMAND_DATA, data, +				       &discard_d, &discard_b ); +	if ( status != GUESTRPC_COMMAND_DATA_SUCCESS ) { +		DBGC ( GUESTRPC_MAGIC, "GuestRPC channel %d send command " +		       "data %08x failed: status %08x\n", +		       channel, data, status ); +		return -EPROTO_COMMAND_DATA; +	} + +	return 0; +} + +/** + * Receive GuestRPC reply length + * + * @v channel		Channel number + * @ret reply_id	Reply ID + * @ret len		Reply length, or negative error + */ +static int guestrpc_reply_len ( int channel, uint16_t *reply_id ) { +	uint32_t len; +	uint32_t status; + +	/* Issue GuestRPC command */ +	status = vmware_cmd_guestrpc ( channel, GUESTRPC_REPLY_LEN, 0, +				       reply_id, &len ); +	if ( status != GUESTRPC_REPLY_LEN_SUCCESS ) { +		DBGC ( GUESTRPC_MAGIC, "GuestRPC channel %d receive reply " +		       "length failed: status %08x\n", channel, status ); +		return -EPROTO_REPLY_LEN; +	} + +	return len; +} + +/** + * Receive GuestRPC reply data + * + * @v channel		Channel number + * @v reply_id		Reply ID + * @ret data		Reply data + * @ret rc		Return status code + */ +static int guestrpc_reply_data ( int channel, uint16_t reply_id, +				 uint32_t *data ) { +	uint16_t discard_d; +	uint32_t status; + +	/* Issue GuestRPC command */ +	status = vmware_cmd_guestrpc ( channel, GUESTRPC_REPLY_DATA, reply_id, +				       &discard_d, data ); +	if ( status != GUESTRPC_REPLY_DATA_SUCCESS ) { +		DBGC ( GUESTRPC_MAGIC, "GuestRPC channel %d receive reply " +		       "%d data failed: status %08x\n", +		       channel, reply_id, status ); +		return -EPROTO_REPLY_DATA; +	} + +	return 0; +} + +/** + * Finish receiving GuestRPC reply + * + * @v channel		Channel number + * @v reply_id		Reply ID + * @ret rc		Return status code + */ +static int guestrpc_reply_finish ( int channel, uint16_t reply_id ) { +	uint16_t discard_d; +	uint32_t discard_b; +	uint32_t status; + +	/* Issue GuestRPC command */ +	status = vmware_cmd_guestrpc ( channel, GUESTRPC_REPLY_FINISH, reply_id, +				       &discard_d, &discard_b ); +	if ( status != GUESTRPC_REPLY_FINISH_SUCCESS ) { +		DBGC ( GUESTRPC_MAGIC, "GuestRPC channel %d finish reply %d " +		       "failed: status %08x\n", channel, reply_id, status ); +		return -EPROTO_REPLY_FINISH; +	} + +	return 0; +} + +/** + * Close GuestRPC channel + * + * @v channel		Channel number + */ +void guestrpc_close ( int channel ) { +	uint16_t discard_d; +	uint32_t discard_b; +	uint32_t status; + +	/* Issue GuestRPC command */ +	status = vmware_cmd_guestrpc ( channel, GUESTRPC_CLOSE, 0, +				       &discard_d, &discard_b ); +	if ( status != GUESTRPC_CLOSE_SUCCESS ) { +		DBGC ( GUESTRPC_MAGIC, "GuestRPC channel %d close failed: " +		       "status %08x\n", channel, status ); +		return; +	} + +	DBGC ( GUESTRPC_MAGIC, "GuestRPC channel %d closed\n", channel ); +} + +/** + * Issue GuestRPC command + * + * @v channel		Channel number + * @v command		Command + * @v reply		Reply buffer + * @v reply_len		Length of reply buffer + * @ret len		Length of reply, or negative error + * + * The actual length of the reply will be returned even if the buffer + * was too small. + */ +int guestrpc_command ( int channel, const char *command, char *reply, +		       size_t reply_len ) { +	const uint8_t *command_bytes = ( ( const void * ) command ); +	uint8_t *reply_bytes = ( ( void * ) reply ); +	size_t command_len = strlen ( command ); +	int orig_reply_len = reply_len; +	uint16_t status; +	uint8_t *status_bytes = ( ( void * ) &status ); +	size_t status_len = sizeof ( status ); +	uint32_t data; +	uint16_t reply_id; +	int len; +	int remaining; +	unsigned int i; +	int rc; + +	DBGC2 ( GUESTRPC_MAGIC, "GuestRPC channel %d issuing command:\n", +		channel ); +	DBGC2_HDA ( GUESTRPC_MAGIC, 0, command, command_len ); + +	/* Sanity check */ +	assert ( ( reply != NULL ) || ( reply_len == 0 ) ); + +	/* Send command length */ +	if ( ( rc = guestrpc_command_len ( channel, command_len ) ) < 0 ) +		return rc; + +	/* Send command data */ +	while ( command_len ) { +		data = 0; +		for ( i = sizeof ( data ) ; i ; i-- ) { +			if ( command_len ) { +				data = ( ( data & ~0xff ) | +					 *(command_bytes++) ); +				command_len--; +			} +			data = ( ( data << 24 ) | ( data >> 8 ) ); +		} +		if ( ( rc = guestrpc_command_data ( channel, data ) ) < 0 ) +			return rc; +	} + +	/* Receive reply length */ +	if ( ( len = guestrpc_reply_len ( channel, &reply_id ) ) < 0 ) { +		rc = len; +		return rc; +	} + +	/* Receive reply */ +	for ( remaining = len ; remaining > 0 ; remaining -= sizeof ( data ) ) { +		if ( ( rc = guestrpc_reply_data ( channel, reply_id, +						  &data ) ) < 0 ) { +			return rc; +		} +		for ( i = sizeof ( data ) ; i ; i-- ) { +			if ( status_len ) { +				*(status_bytes++) = ( data & 0xff ); +				status_len--; +				len--; +			} else if ( reply_len ) { +				*(reply_bytes++) = ( data & 0xff ); +				reply_len--; +			} +			data = ( ( data << 24 ) | ( data >> 8 ) ); +		} +	} + +	/* Finish receiving RPC reply */ +	if ( ( rc = guestrpc_reply_finish ( channel, reply_id ) ) < 0 ) +		return rc; + +	DBGC2 ( GUESTRPC_MAGIC, "GuestRPC channel %d received reply (id %d, " +		"length %d):\n", channel, reply_id, len ); +	DBGC2_HDA ( GUESTRPC_MAGIC, 0, &status, sizeof ( status ) ); +	DBGC2_HDA ( GUESTRPC_MAGIC, sizeof ( status ), reply, +		    ( ( len < orig_reply_len ) ? len : orig_reply_len ) ); + +	/* Check reply status */ +	if ( status != GUESTRPC_SUCCESS ) { +		DBGC ( GUESTRPC_MAGIC, "GuestRPC channel %d command failed " +		       "(status %04x, reply id %d, reply length %d):\n", +		       channel, status, reply_id, len ); +		DBGC_HDA ( GUESTRPC_MAGIC, 0, command, command_len ); +		DBGC_HDA ( GUESTRPC_MAGIC, 0, &status, sizeof ( status ) ); +		DBGC_HDA ( GUESTRPC_MAGIC, sizeof ( status ), reply, +			   ( ( len < orig_reply_len ) ? len : orig_reply_len )); +		return -EIO; +	} + +	return len; +} diff --git a/roms/ipxe/src/arch/i386/interface/vmware/vmconsole.c b/roms/ipxe/src/arch/i386/interface/vmware/vmconsole.c new file mode 100644 index 00000000..c6b9fff1 --- /dev/null +++ b/roms/ipxe/src/arch/i386/interface/vmware/vmconsole.c @@ -0,0 +1,134 @@ +/* + * Copyright (C) 2012 Michael Brown <mbrown@fensystems.co.uk>. + * + * This program is free software; you can redistribute it and/or + * modify it under the terms of the GNU General Public License as + * published by the Free Software Foundation; either version 2 of the + * License, or any later version. + * + * 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. + * + * You should have received a copy of the GNU General Public License + * along with this program; if not, write to the Free Software + * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA + * 02110-1301, USA. + */ + +FILE_LICENCE ( GPL2_OR_LATER ); + +/** @file + * + * VMware logfile console + * + */ + +#include <string.h> +#include <ipxe/console.h> +#include <ipxe/lineconsole.h> +#include <ipxe/init.h> +#include <ipxe/guestrpc.h> +#include <config/console.h> + +/** VMware logfile console buffer size */ +#define VMCONSOLE_BUFSIZE 128 + +/* Set default console usage if applicable */ +#if ! ( defined ( CONSOLE_VMWARE ) && CONSOLE_EXPLICIT ( CONSOLE_VMWARE ) ) +#undef CONSOLE_VMWARE +#define CONSOLE_VMWARE ( CONSOLE_USAGE_ALL & ~CONSOLE_USAGE_TUI ) +#endif + +/** VMware logfile console GuestRPC channel */ +static int vmconsole_channel; + +/** VMware logfile console line buffer */ +static struct { +	char prefix[4]; +	char message[VMCONSOLE_BUFSIZE]; +} vmconsole_buffer = { +	.prefix = "log ", +}; + +/** VMware logfile console ANSI escape sequence handlers */ +static struct ansiesc_handler vmconsole_handlers[] = { +	{ 0, NULL } +}; + +/** VMware logfile line console */ +static struct line_console vmconsole_line = { +	.buffer = vmconsole_buffer.message, +	.len = sizeof ( vmconsole_buffer.message ), +	.ctx = { +		.handlers = vmconsole_handlers, +	}, +}; + +/** VMware logfile console recursion marker */ +static int vmconsole_entered; + +/** + * Print a character to VMware logfile console + * + * @v character		Character to be printed + */ +static void vmconsole_putchar ( int character ) { +	int rc; + +	/* Ignore if we are already mid-logging */ +	if ( vmconsole_entered ) +		return; + +	/* Fill line buffer */ +	if ( line_putchar ( &vmconsole_line, character ) == 0 ) +		return; + +	/* Guard against re-entry */ +	vmconsole_entered = 1; + +	/* Send log message */ +	if ( ( rc = guestrpc_command ( vmconsole_channel, +				       vmconsole_buffer.prefix, NULL, 0 ) ) <0){ +		DBG ( "VMware console could not send log message: %s\n", +		      strerror ( rc ) ); +	} + +	/* Clear re-entry flag */ +	vmconsole_entered = 0; +} + +/** VMware logfile console driver */ +struct console_driver vmconsole __console_driver = { +	.putchar = vmconsole_putchar, +	.disabled = CONSOLE_DISABLED, +	.usage = CONSOLE_VMWARE, +}; + +/** + * Initialise VMware logfile console + * + */ +static void vmconsole_init ( void ) { +	int rc; + +	/* Attempt to open console */ +	vmconsole_channel = guestrpc_open(); +	if ( vmconsole_channel < 0 ) { +		rc = vmconsole_channel; +		DBG ( "VMware console could not be initialised: %s\n", +		      strerror ( rc ) ); +		return; +	} + +	/* Mark console as available */ +	vmconsole.disabled = 0; +} + +/** + * VMware logfile console initialisation function + */ +struct init_fn vmconsole_init_fn __init_fn ( INIT_CONSOLE ) = { +	.initialise = vmconsole_init, +}; diff --git a/roms/ipxe/src/arch/i386/interface/vmware/vmware.c b/roms/ipxe/src/arch/i386/interface/vmware/vmware.c new file mode 100644 index 00000000..8074e611 --- /dev/null +++ b/roms/ipxe/src/arch/i386/interface/vmware/vmware.c @@ -0,0 +1,58 @@ +/* + * Copyright (C) 2012 Michael Brown <mbrown@fensystems.co.uk>. + * + * This program is free software; you can redistribute it and/or + * modify it under the terms of the GNU General Public License as + * published by the Free Software Foundation; either version 2 of the + * License, or any later version. + * + * 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. + * + * You should have received a copy of the GNU General Public License + * along with this program; if not, write to the Free Software + * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA + * 02110-1301, USA. + */ + +FILE_LICENCE ( GPL2_OR_LATER ); + +/** @file + * + * VMware backdoor mechanism + * + * Based on the unofficial documentation at + * + *   http://sites.google.com/site/chitchatvmback/backdoor + * + */ + +#include <stdint.h> +#include <errno.h> +#include <ipxe/vmware.h> + +/** + * Detect VMware presence + * + * @ret rc		Return status code + */ +int vmware_present ( void ) { +	uint32_t version; +	uint32_t magic; +	uint32_t product_type; + +	/* Perform backdoor call */ +	vmware_cmd_get_version ( &version, &magic, &product_type ); + +	/* Check for VMware presence */ +	if ( magic != VMW_MAGIC ) { +		DBGC ( VMW_MAGIC, "VMware not present\n" ); +		return -ENOENT; +	} + +	DBGC ( VMW_MAGIC, "VMware product type %04x version %08x detected\n", +	       product_type, version ); +	return 0; +} | 
