diff options
Diffstat (limited to 'src/arch/i386/interface/pcbios/iscsiboot.c')
-rw-r--r-- | src/arch/i386/interface/pcbios/iscsiboot.c | 108 |
1 files changed, 108 insertions, 0 deletions
diff --git a/src/arch/i386/interface/pcbios/iscsiboot.c b/src/arch/i386/interface/pcbios/iscsiboot.c new file mode 100644 index 00000000..cdf7790b --- /dev/null +++ b/src/arch/i386/interface/pcbios/iscsiboot.c @@ -0,0 +1,108 @@ +#include <stdint.h> +#include <string.h> +#include <stdlib.h> +#include <stdio.h> +#include <errno.h> +#include <gpxe/iscsi.h> +#include <gpxe/settings.h> +#include <gpxe/dhcp.h> +#include <gpxe/netdevice.h> +#include <gpxe/ibft.h> +#include <gpxe/init.h> +#include <gpxe/sanboot.h> +#include <int13.h> +#include <usr/autoboot.h> + +struct setting keep_san_setting __setting = { + .name = "keep-san", + .description = "Preserve SAN connection", + .tag = DHCP_EB_KEEP_SAN, + .type = &setting_type_int8, +}; + +/** + * Guess boot network device + * + * @ret netdev Boot network device + */ +static struct net_device * guess_boot_netdev ( void ) { + struct net_device *netdev; + + /* Just use the first network device */ + for_each_netdev ( netdev ) { + if ( netdev->state & NETDEV_OPEN ) + return netdev; + } + + return NULL; +} + +static int iscsiboot ( const char *root_path ) { + struct scsi_device *scsi; + struct int13_drive *drive; + int keep_san; + int rc; + + scsi = zalloc ( sizeof ( *scsi ) ); + if ( ! scsi ) { + rc = -ENOMEM; + goto err_alloc_scsi; + } + drive = zalloc ( sizeof ( *drive ) ); + if ( ! drive ) { + rc = -ENOMEM; + goto err_alloc_drive; + } + + printf ( "iSCSI booting from %s\n", root_path ); + + if ( ( rc = iscsi_attach ( scsi, root_path ) ) != 0 ) { + printf ( "Could not attach iSCSI device: %s\n", + strerror ( rc ) ); + goto err_attach; + } + if ( ( rc = init_scsidev ( scsi ) ) != 0 ) { + printf ( "Could not initialise iSCSI device: %s\n", + strerror ( rc ) ); + goto err_init; + } + + drive->blockdev = &scsi->blockdev; + + /* FIXME: ugly, ugly hack */ + struct net_device *netdev = guess_boot_netdev(); + struct iscsi_session *iscsi = + container_of ( scsi->backend, struct iscsi_session, refcnt ); + ibft_fill_data ( netdev, iscsi ); + + register_int13_drive ( drive ); + printf ( "Registered as BIOS drive %#02x\n", drive->drive ); + printf ( "Booting from BIOS drive %#02x\n", drive->drive ); + rc = int13_boot ( drive->drive ); + printf ( "Boot failed\n" ); + + /* Leave drive registered, if instructed to do so */ + keep_san = fetch_intz_setting ( NULL, &keep_san_setting ); + if ( keep_san ) { + printf ( "Preserving connection to SAN disk\n" ); + shutdown_exit_flags |= SHUTDOWN_KEEP_DEVICES; + return rc; + } + + printf ( "Unregistering BIOS drive %#02x\n", drive->drive ); + unregister_int13_drive ( drive ); + + err_init: + iscsi_detach ( scsi ); + err_attach: + free ( drive ); + err_alloc_drive: + free ( scsi ); + err_alloc_scsi: + return rc; +} + +struct sanboot_protocol iscsi_sanboot_protocol __sanboot_protocol = { + .prefix = "iscsi:", + .boot = iscsiboot, +}; |