summaryrefslogtreecommitdiffstats
path: root/src/drivers/net/efi/snp.c
diff options
context:
space:
mode:
authorMichael Brown2014-07-01 18:58:09 +0200
committerMichael Brown2014-07-03 16:28:17 +0200
commitc7051d826b43954b1e191667a75b21b44ec02c35 (patch)
treec29a4e22878c42f4c31900e5582a055287bbe939 /src/drivers/net/efi/snp.c
parent[build] Add yet another potential location for isolinux.bin (diff)
downloadipxe-c7051d826b43954b1e191667a75b21b44ec02c35.tar.gz
ipxe-c7051d826b43954b1e191667a75b21b44ec02c35.tar.xz
ipxe-c7051d826b43954b1e191667a75b21b44ec02c35.zip
[efi] Allow network devices to be created on top of arbitrary SNP devices
Signed-off-by: Michael Brown <mcb30@ipxe.org>
Diffstat (limited to 'src/drivers/net/efi/snp.c')
-rw-r--r--src/drivers/net/efi/snp.c219
1 files changed, 219 insertions, 0 deletions
diff --git a/src/drivers/net/efi/snp.c b/src/drivers/net/efi/snp.c
new file mode 100644
index 00000000..98619d66
--- /dev/null
+++ b/src/drivers/net/efi/snp.c
@@ -0,0 +1,219 @@
+/*
+ * Copyright (C) 2014 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 <string.h>
+#include <errno.h>
+#include <ipxe/efi/efi.h>
+#include <ipxe/efi/Protocol/SimpleNetwork.h>
+#include <ipxe/efi/efi_driver.h>
+#include <ipxe/efi/efi_snp.h>
+#include <ipxe/efi/efi_pci.h>
+#include "snpnet.h"
+#include "snp.h"
+
+/** @file
+ *
+ * SNP driver
+ *
+ */
+
+/** EFI simple network protocol GUID */
+static EFI_GUID efi_simple_network_protocol_guid
+ = EFI_SIMPLE_NETWORK_PROTOCOL_GUID;
+
+/** EFI PCI I/O protocol GUID */
+static EFI_GUID efi_pci_io_protocol_guid
+ = EFI_PCI_IO_PROTOCOL_GUID;
+
+/**
+ * Check to see if driver supports a device
+ *
+ * @v device EFI device handle
+ * @ret rc Return status code
+ */
+static int snp_supported ( EFI_HANDLE device ) {
+ EFI_BOOT_SERVICES *bs = efi_systab->BootServices;
+ EFI_STATUS efirc;
+
+ /* Check that this is not a device we are providing ourselves */
+ if ( find_snpdev ( device ) != NULL ) {
+ DBGCP ( device, "SNP %p %s is provided by this binary\n",
+ device, efi_handle_devpath_text ( device ) );
+ return -ENOTTY;
+ }
+
+ /* Test for presence of simple network protocol */
+ if ( ( efirc = bs->OpenProtocol ( device,
+ &efi_simple_network_protocol_guid,
+ NULL, efi_image_handle, device,
+ EFI_OPEN_PROTOCOL_TEST_PROTOCOL))!=0){
+ DBGCP ( device, "SNP %p %s is not an SNP device\n",
+ device, efi_handle_devpath_text ( device ) );
+ return -EEFI ( efirc );
+ }
+ DBGC ( device, "SNP %p %s is an SNP device\n",
+ device, efi_handle_devpath_text ( device ) );
+
+ return 0;
+}
+
+/**
+ * Get underlying PCI device information
+ *
+ * @v snpdev SNP device
+ * @ret rc Return status code
+ */
+static int snp_pci_info ( struct snp_device *snpdev ) {
+ EFI_BOOT_SERVICES *bs = efi_systab->BootServices;
+ struct efi_device *efidev = snpdev->efidev;
+ EFI_DEVICE_PATH_PROTOCOL *devpath = efidev->path;
+ struct pci_device pci;
+ EFI_HANDLE device;
+ EFI_STATUS efirc;
+ int rc;
+
+ /* Check for presence of PCI I/O protocol */
+ if ( ( efirc = bs->LocateDevicePath ( &efi_pci_io_protocol_guid,
+ &devpath, &device ) ) != 0 ) {
+ DBGC ( efidev->device, "SNP %p %s is not a PCI device\n",
+ efidev->device, efi_devpath_text ( efidev->path ) );
+ return -EEFI ( efirc );
+ }
+
+ /* Get PCI device information */
+ if ( ( rc = efipci_info ( device, &pci ) ) != 0 ) {
+ DBGC ( efidev->device, "SNP %p %s could not get PCI "
+ "information: %s\n", efidev->device,
+ efi_devpath_text ( efidev->path ), strerror ( rc ) );
+ return rc;
+ }
+
+ /* Populate SNP device information */
+ memcpy ( &snpdev->dev.desc, &pci.dev.desc, sizeof ( snpdev->dev.desc ));
+ snprintf ( snpdev->dev.name, sizeof ( snpdev->dev.name ), "SNP-%s",
+ pci.dev.name );
+
+ return 0;
+}
+
+/**
+ * Attach driver to device
+ *
+ * @v efidev EFI device
+ * @ret rc Return status code
+ */
+static int snp_start ( struct efi_device *efidev ) {
+ EFI_BOOT_SERVICES *bs = efi_systab->BootServices;
+ EFI_HANDLE device = efidev->device;
+ struct snp_device *snpdev;
+ union {
+ EFI_SIMPLE_NETWORK_PROTOCOL *snp;
+ void *interface;
+ } snp;
+ EFI_STATUS efirc;
+ int rc;
+
+ /* Check that this is not a device we are providing ourselves */
+ if ( find_snpdev ( efidev->device ) != NULL ) {
+ DBGCP ( device, "SNP %p %s is provided by this binary\n",
+ device, efi_devpath_text ( efidev->path ) );
+ rc = -ENOTTY;
+ goto err_own;
+ }
+
+ /* Allocate and initialise structure */
+ snpdev = zalloc ( sizeof ( *snpdev ) );
+ if ( ! snpdev ) {
+ rc = -ENOMEM;
+ goto err_alloc;
+ }
+ snpdev->efidev = efidev;
+ snpdev->dev.driver_name = "SNP";
+ INIT_LIST_HEAD ( &snpdev->dev.children );
+
+ /* See if device is an SNP device */
+ if ( ( efirc = bs->OpenProtocol ( device,
+ &efi_simple_network_protocol_guid,
+ &snp.interface, efi_image_handle,
+ device,
+ ( EFI_OPEN_PROTOCOL_BY_DRIVER |
+ EFI_OPEN_PROTOCOL_EXCLUSIVE )))!=0){
+ rc = -EEFI ( efirc );
+ DBGCP ( device, "SNP %p %s cannot open SNP protocol: %s\n",
+ device, efi_devpath_text ( efidev->path ),
+ strerror ( rc ) );
+ goto err_open_protocol;
+ }
+ snpdev->snp = snp.snp;
+
+ /* Get underlying device information */
+ if ( ( rc = snp_pci_info ( snpdev ) ) != 0 )
+ goto err_info;
+
+ /* Mark SNP device as a child of the EFI device */
+ snpdev->dev.parent = &efidev->dev;
+ list_add ( &snpdev->dev.siblings, &efidev->dev.children );
+
+ /* Create SNP network device */
+ if ( ( rc = snpnet_probe ( snpdev ) ) != 0 )
+ goto err_probe;
+
+ efidev_set_drvdata ( efidev, snpdev );
+ return 0;
+
+ snpnet_remove ( snpdev );
+ err_probe:
+ list_del ( &snpdev->dev.siblings );
+ err_info:
+ bs->CloseProtocol ( device, &efi_simple_network_protocol_guid,
+ efi_image_handle, device );
+ err_open_protocol:
+ free ( snpdev );
+ err_alloc:
+ err_own:
+ return rc;
+}
+
+/**
+ * Detach driver from device
+ *
+ * @v efidev EFI device
+ */
+static void snp_stop ( struct efi_device *efidev ) {
+ EFI_BOOT_SERVICES *bs = efi_systab->BootServices;
+ struct snp_device *snpdev = efidev_get_drvdata ( efidev );
+
+ snpnet_remove ( snpdev );
+ list_del ( &snpdev->dev.siblings );
+ bs->CloseProtocol ( efidev->device, &efi_simple_network_protocol_guid,
+ efi_image_handle, efidev->device );
+ free ( snpdev );
+}
+
+/** EFI SNP driver */
+struct efi_driver snp_driver __efi_driver ( EFI_DRIVER_NORMAL ) = {
+ .name = "SNP",
+ .supported = snp_supported,
+ .start = snp_start,
+ .stop = snp_stop,
+};