summaryrefslogblamecommitdiffstats
path: root/src/interface/efi/efi_blacklist.c
blob: 292b28e8c0318cebc43e0268097e45b257a0620c (plain) (tree)












































































































































































































































                                                                               
/*
 * Copyright (C) 2019 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_OR_UBDL );

#include <stddef.h>
#include <string.h>
#include <errno.h>
#include <ipxe/settings.h>
#include <ipxe/efi/efi.h>
#include <ipxe/efi/Protocol/DriverBinding.h>
#include <ipxe/efi/Protocol/LoadedImage.h>
#include <ipxe/efi/Protocol/ComponentName.h>
#include <ipxe/efi/efi_blacklist.h>

/** @file
 *
 * EFI driver blacklist
 *
 */

/** A blacklisted driver */
struct efi_blacklist {
	/** Name */
	const char *name;
	/**
	 * Check if driver is blacklisted
	 *
	 * @v binding		Driver binding protocol
	 * @v loaded		Loaded image protocol
	 * @v wtf		Component name protocol, if present
	 * @ret blacklisted	Driver is the blacklisted driver
	 */
	int ( * blacklist ) ( EFI_DRIVER_BINDING_PROTOCOL *binding,
			      EFI_LOADED_IMAGE_PROTOCOL *loaded,
			      EFI_COMPONENT_NAME_PROTOCOL *wtf );
};

/**
 * Blacklist Dell Ip4ConfigDxe driver
 *
 * @v binding		Driver binding protocol
 * @v loaded		Loaded image protocol
 * @v wtf		Component name protocol, if present
 * @ret blacklisted	Driver is the blacklisted driver
 */
static int
efi_blacklist_dell_ip4config ( EFI_DRIVER_BINDING_PROTOCOL *binding __unused,
			       EFI_LOADED_IMAGE_PROTOCOL *loaded __unused,
			       EFI_COMPONENT_NAME_PROTOCOL *wtf ) {
	static const CHAR16 ip4cfg[] = L"IP4 CONFIG Network Service Driver";
	static const char dell[] = "Dell Inc.";
	char manufacturer[ sizeof ( dell ) ];
	CHAR16 *name;

	/* Check driver name */
	if ( ! wtf )
		return 0;
	if ( wtf->GetDriverName ( wtf, "eng", &name ) != 0 )
		return 0;
	if ( memcmp ( name, ip4cfg, sizeof ( ip4cfg ) ) != 0 )
		return 0;

	/* Check manufacturer */
	fetch_string_setting ( NULL, &manufacturer_setting, manufacturer,
			       sizeof ( manufacturer ) );
	if ( strcmp ( manufacturer, dell ) != 0 )
		return 0;

	return 1;
}

/** Blacklisted drivers */
static struct efi_blacklist efi_blacklists[] = {
	{
		.name = "Dell Ip4Config",
		.blacklist = efi_blacklist_dell_ip4config,
	},
};

/**
 * Find driver blacklisting, if any
 *
 * @v driver		Driver binding handle
 * @ret blacklist	Driver blacklisting, or NULL
 * @ret rc		Return status code
 */
static int efi_blacklist ( EFI_HANDLE driver,
			   struct efi_blacklist **blacklist ) {
	EFI_BOOT_SERVICES *bs = efi_systab->BootServices;
	union {
		EFI_DRIVER_BINDING_PROTOCOL *binding;
		void *interface;
	} binding;
	union {
		EFI_LOADED_IMAGE_PROTOCOL *loaded;
		void *interface;
	} loaded;
	union {
		EFI_COMPONENT_NAME_PROTOCOL *wtf;
		void *interface;
	} wtf;
	unsigned int i;
	EFI_HANDLE image;
	EFI_STATUS efirc;
	int rc;

	DBGC2 ( &efi_blacklists, "EFIBL checking %s\n",
		efi_handle_name ( driver ) );

	/* Mark as not blacklisted */
	*blacklist = NULL;

	/* Open driver binding protocol */
	if ( ( efirc = bs->OpenProtocol (
			driver, &efi_driver_binding_protocol_guid,
			&binding.interface, efi_image_handle, driver,
			EFI_OPEN_PROTOCOL_GET_PROTOCOL ) ) != 0 ) {
		rc = -EEFI ( efirc );
		DBGC ( driver, "EFIBL %s could not open driver binding "
		       "protocol: %s\n", efi_handle_name ( driver ),
		       strerror ( rc ) );
		goto err_binding;
	}
	image = binding.binding->ImageHandle;

	/* Open loaded image protocol */
	if ( ( efirc = bs->OpenProtocol (
			image, &efi_loaded_image_protocol_guid,
			&loaded.interface, efi_image_handle, image,
			EFI_OPEN_PROTOCOL_GET_PROTOCOL ) ) != 0 ) {
		rc = -EEFI ( efirc );
		DBGC ( driver, "EFIBL %s could not open",
		       efi_handle_name ( driver ) );
		DBGC ( driver, " %s loaded image protocol: %s\n",
		       efi_handle_name ( image ), strerror ( rc ) );
		goto err_loaded;
	}

	/* Open component name protocol, if present*/
	if ( ( efirc = bs->OpenProtocol (
			driver, &efi_component_name_protocol_guid,
			&wtf.interface, efi_image_handle, driver,
			EFI_OPEN_PROTOCOL_GET_PROTOCOL ) ) != 0 ) {
		/* Ignore failure; is not required to be present */
		wtf.interface = NULL;
	}

	/* Check blacklistings */
	for ( i = 0 ; i < ( sizeof ( efi_blacklists ) /
			    sizeof ( efi_blacklists[0] ) ) ; i++ ) {
		if ( efi_blacklists[i].blacklist ( binding.binding,
						   loaded.loaded, wtf.wtf ) ) {
			*blacklist = &efi_blacklists[i];
			break;
		}
	}

	/* Success */
	rc = 0;

	/* Close protocols */
	if ( wtf.wtf ) {
		bs->CloseProtocol ( driver, &efi_component_name_protocol_guid,
				    efi_image_handle, driver );
	}
	bs->CloseProtocol ( image, &efi_loaded_image_protocol_guid,
			    efi_image_handle, image );
 err_loaded:
	bs->CloseProtocol ( driver, &efi_driver_binding_protocol_guid,
			    efi_image_handle, driver );
 err_binding:
	return rc;
}

/**
 * Unload any blacklisted drivers
 *
 */
void efi_unload_blacklist ( void ) {
	EFI_BOOT_SERVICES *bs = efi_systab->BootServices;
	struct efi_blacklist *blacklist;
	EFI_HANDLE *drivers;
	EFI_HANDLE driver;
	UINTN num_drivers;
	unsigned int i;
	EFI_STATUS efirc;
	int rc;

	/* Locate all driver binding protocol handles */
	if ( ( efirc = bs->LocateHandleBuffer (
			ByProtocol, &efi_driver_binding_protocol_guid,
			NULL, &num_drivers, &drivers ) ) != 0 ) {
		rc = -EEFI ( efirc );
		DBGC ( &efi_blacklists, "EFIBL could not list all drivers: "
		       "%s\n", strerror ( rc ) );
		return;
	}

	/* Unload any blacklisted drivers */
	for ( i = 0 ; i < num_drivers ; i++ ) {
		driver = drivers[i];
		if ( ( rc = efi_blacklist ( driver, &blacklist ) ) != 0 ) {
			DBGC ( driver, "EFIBL could not determine "
			       "blacklisting for %s: %s\n",
			       efi_handle_name ( driver ), strerror ( rc ) );
			continue;
		}
		if ( ! blacklist )
			continue;
		DBGC ( driver, "EFIBL unloading %s (%s)\n",
		       efi_handle_name ( driver ), blacklist->name );
		if ( ( efirc = bs->UnloadImage ( driver ) ) != 0 ) {
			DBGC ( driver, "EFIBL could not unload %s: %s\n",
			       efi_handle_name ( driver ), strerror ( rc ) );
		}
	}

	/* Free handle list */
	bs->FreePool ( drivers );
}