mirror of
https://github.com/ipxe/ipxe
synced 2025-12-18 10:30:23 +03:00
[i386] Move iSCSI and AoE boot code to arch/i386/interface/pcbios
This commit is contained in:
108
src/arch/i386/interface/pcbios/iscsiboot.c
Normal file
108
src/arch/i386/interface/pcbios/iscsiboot.c
Normal file
@@ -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,
|
||||
};
|
||||
Reference in New Issue
Block a user