summaryrefslogblamecommitdiffstats
path: root/src/drivers/net/efi/snponly.c
blob: 6fcc54a0193a1491051f80be17ae33f19b5bfae6 (plain) (tree)



















































































































                                                                              
                                                           
   
                                              


                                                                            
                      





                                                                    
/*
 * Copyright (C) 2010 VMware, Inc.  All Rights Reserved.
 *
 * 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 St, Fifth Floor, Boston, MA 02110-1301 USA.
 */

FILE_LICENCE ( GPL2_OR_LATER );

#include <string.h>
#include <errno.h>
#include <ipxe/device.h>
#include <ipxe/init.h>
#include <ipxe/efi/efi.h>
#include <ipxe/efi/Protocol/SimpleNetwork.h>
#include "snp.h"
#include "snpnet.h"

/** @file
 *
 * Chain-loading Simple Network Protocol Bus Driver
 *
 * This bus driver allows iPXE to use the EFI Simple Network Protocol provided
 * by the platform to transmit and receive packets. It attaches to only the
 * device handle that iPXE was loaded from, that is, it will only use the
 * Simple Network Protocol on the current loaded image's device handle.
 *
 * Eseentially, this driver provides the EFI equivalent of the "undionly"
 * driver.
 */

/** The one and only SNP network device */
static struct snp_device snponly_dev;

/** EFI simple network protocol GUID */
static EFI_GUID efi_simple_network_protocol_guid
	= EFI_SIMPLE_NETWORK_PROTOCOL_GUID;

/**
 * Probe SNP root bus
 *
 * @v rootdev		SNP bus root device
 *
 * Look at the loaded image's device handle and see if the simple network
 * protocol exists. If so, register a driver for it.
 */
static int snpbus_probe ( struct root_device *rootdev ) {
	EFI_BOOT_SERVICES *bs = efi_systab->BootServices;
	EFI_STATUS efirc;
	int rc;
	void *snp;

	efirc = bs->OpenProtocol ( efi_loaded_image->DeviceHandle,
				   &efi_simple_network_protocol_guid,
				   &snp, efi_image_handle, NULL,
				   EFI_OPEN_PROTOCOL_GET_PROTOCOL );
	if ( efirc ) {
		DBG ( "Could not find Simple Network Protocol!\n" );
		return -ENODEV;
	}
	snponly_dev.snp = snp;

	/* Add to device hierarchy */
	strncpy ( snponly_dev.dev.name, "EFI SNP",
		  ( sizeof ( snponly_dev.dev.name ) - 1 ) );
	snponly_dev.dev.parent = &rootdev->dev;
	list_add ( &snponly_dev.dev.siblings, &rootdev->dev.children);
	INIT_LIST_HEAD ( &snponly_dev.dev.children );

	/* Create network device */
	if ( ( rc = snpnet_probe ( &snponly_dev ) ) != 0 )
		goto err;

	return 0;

err:
	list_del ( &snponly_dev.dev.siblings );
	return rc;
}

/**
 * Remove SNP root bus
 *
 * @v rootdev		SNP bus root device
 */
static void snpbus_remove ( struct root_device *rootdev __unused ) {
	snpnet_remove ( &snponly_dev );
	list_del ( &snponly_dev.dev.siblings );
}

/** SNP bus root device driver */
static struct root_driver snp_root_driver = {
	.probe = snpbus_probe,
	.remove = snpbus_remove,
};

/** SNP bus root device */
struct root_device snp_root_device __root_device = {
	.dev = { .name = "EFI SNP" },
	.driver = &snp_root_driver,
};

/**
 * Prepare for exit
 *
 * @v booting		System is shutting down for OS boot
 */
static void snponly_shutdown ( int booting ) {
	/* If we are shutting down to boot an OS, make sure the SNP does not
	 * stay active.
	 */
	if ( booting )
		snponly_dev.removal_state = EfiSimpleNetworkStopped;
}

struct startup_fn startup_snponly __startup_fn ( STARTUP_LATE ) = {
	.shutdown = snponly_shutdown,
};