memcpy ( apple, &efi_null_apple, sizeof ( *apple ) );
}
+
+/******************************************************************************
+ *
+ * USB I/O Protocol
+ *
+ ******************************************************************************
+ */
+
+static EFI_STATUS EFIAPI
+efi_null_usb_control_transfer ( EFI_USB_IO_PROTOCOL *usbio __unused,
+ EFI_USB_DEVICE_REQUEST *packet __unused,
+ EFI_USB_DATA_DIRECTION direction __unused,
+ UINT32 timeout __unused, VOID *data __unused,
+ UINTN len __unused, UINT32 *status __unused ) {
+ return EFI_UNSUPPORTED;
+}
+
+static EFI_STATUS EFIAPI
+efi_null_usb_bulk_transfer ( EFI_USB_IO_PROTOCOL *usbio __unused,
+ UINT8 endpoint __unused, VOID *data __unused,
+ UINTN *len __unused, UINTN timeout __unused,
+ UINT32 *status __unused ) {
+ return EFI_UNSUPPORTED;
+}
+
+static EFI_STATUS EFIAPI
+efi_null_usb_sync_interrupt_transfer ( EFI_USB_IO_PROTOCOL *usbio __unused,
+ UINT8 endpoint __unused,
+ VOID *data __unused,
+ UINTN *len __unused,
+ UINTN timeout __unused,
+ UINT32 *status __unused ) {
+ return EFI_UNSUPPORTED;
+}
+
+static EFI_STATUS EFIAPI
+efi_null_usb_async_interrupt_transfer ( EFI_USB_IO_PROTOCOL *usbio __unused,
+ UINT8 endpoint __unused,
+ BOOLEAN start __unused,
+ UINTN interval __unused,
+ UINTN len __unused,
+ EFI_ASYNC_USB_TRANSFER_CALLBACK
+ callback __unused,
+ VOID *context __unused ) {
+ return EFI_UNSUPPORTED;
+}
+
+static EFI_STATUS EFIAPI
+efi_null_usb_isochronous_transfer ( EFI_USB_IO_PROTOCOL *usbio __unused,
+ UINT8 endpoint __unused,
+ VOID *data __unused, UINTN len __unused,
+ UINT32 *status __unused ) {
+ return EFI_UNSUPPORTED;
+}
+
+static EFI_STATUS EFIAPI
+efi_null_usb_async_isochronous_transfer ( EFI_USB_IO_PROTOCOL *usbio __unused,
+ UINT8 endpoint __unused,
+ VOID *data __unused,
+ UINTN len __unused,
+ EFI_ASYNC_USB_TRANSFER_CALLBACK
+ callback __unused,
+ VOID *context __unused ) {
+ return EFI_UNSUPPORTED;
+}
+
+static EFI_STATUS EFIAPI
+efi_null_usb_get_device_descriptor ( EFI_USB_IO_PROTOCOL *usbio __unused,
+ EFI_USB_DEVICE_DESCRIPTOR
+ *efidesc __unused ) {
+ return EFI_UNSUPPORTED;
+}
+
+static EFI_STATUS EFIAPI
+efi_null_usb_get_config_descriptor ( EFI_USB_IO_PROTOCOL *usbio __unused,
+ EFI_USB_CONFIG_DESCRIPTOR
+ *efidesc __unused ) {
+ return EFI_UNSUPPORTED;
+}
+
+static EFI_STATUS EFIAPI
+efi_null_usb_get_interface_descriptor ( EFI_USB_IO_PROTOCOL *usbio __unused,
+ EFI_USB_INTERFACE_DESCRIPTOR
+ *efidesc __unused ) {
+ return EFI_UNSUPPORTED;
+}
+
+static EFI_STATUS EFIAPI
+efi_null_usb_get_endpoint_descriptor ( EFI_USB_IO_PROTOCOL *usbio __unused,
+ UINT8 index __unused,
+ EFI_USB_ENDPOINT_DESCRIPTOR
+ *efidesc __unused ) {
+ return EFI_UNSUPPORTED;
+}
+
+static EFI_STATUS EFIAPI
+efi_null_usb_get_string_descriptor ( EFI_USB_IO_PROTOCOL *usbio __unused,
+ UINT16 language __unused,
+ UINT8 index __unused,
+ CHAR16 **string __unused ) {
+ return EFI_UNSUPPORTED;
+}
+
+static EFI_STATUS EFIAPI
+efi_null_usb_get_supported_languages ( EFI_USB_IO_PROTOCOL *usbio __unused,
+ UINT16 **languages __unused,
+ UINT16 *len __unused ) {
+ return EFI_UNSUPPORTED;
+}
+
+static EFI_STATUS EFIAPI
+efi_null_usb_port_reset ( EFI_USB_IO_PROTOCOL *usbio __unused ) {
+ return EFI_UNSUPPORTED;
+}
+
+static EFI_USB_IO_PROTOCOL efi_null_usbio = {
+ .UsbControlTransfer = efi_null_usb_control_transfer,
+ .UsbBulkTransfer = efi_null_usb_bulk_transfer,
+ .UsbAsyncInterruptTransfer = efi_null_usb_async_interrupt_transfer,
+ .UsbSyncInterruptTransfer = efi_null_usb_sync_interrupt_transfer,
+ .UsbIsochronousTransfer = efi_null_usb_isochronous_transfer,
+ .UsbAsyncIsochronousTransfer = efi_null_usb_async_isochronous_transfer,
+ .UsbGetDeviceDescriptor = efi_null_usb_get_device_descriptor,
+ .UsbGetConfigDescriptor = efi_null_usb_get_config_descriptor,
+ .UsbGetInterfaceDescriptor = efi_null_usb_get_interface_descriptor,
+ .UsbGetEndpointDescriptor = efi_null_usb_get_endpoint_descriptor,
+ .UsbGetStringDescriptor = efi_null_usb_get_string_descriptor,
+ .UsbGetSupportedLanguages = efi_null_usb_get_supported_languages,
+ .UsbPortReset = efi_null_usb_port_reset,
+};
+
+/**
+ * Nullify USB I/O protocol
+ *
+ * @v usbio USB I/O protocol
+ */
+void efi_nullify_usbio ( EFI_USB_IO_PROTOCOL *usbio ) {
+
+ memcpy ( usbio, &efi_null_usbio, sizeof ( *usbio ) );
+}
#include <ipxe/efi/efi.h>
#include <ipxe/efi/efi_path.h>
#include <ipxe/efi/efi_driver.h>
+#include <ipxe/efi/efi_null.h>
#include <ipxe/efi/efi_usb.h>
#include <ipxe/usb.h>
EFI_BOOT_SERVICES *bs = efi_systab->BootServices;
struct usb_function *func = usbdev->func;
struct efi_usb_interface *usbintf;
+ int leak = 0;
EFI_STATUS efirc;
int rc;
usbintf->name, efi_handle_name ( usbintf->handle ) );
return 0;
- bs->UninstallMultipleProtocolInterfaces (
+ if ( ( efirc = bs->UninstallMultipleProtocolInterfaces (
usbintf->handle,
&efi_usb_io_protocol_guid, &usbintf->usbio,
&efi_device_path_protocol_guid, usbintf->path,
- NULL );
+ NULL ) ) != 0 ) {
+ DBGC ( usbdev, "USBDEV %s could not uninstall: %s\n",
+ usbintf->name, strerror ( -EEFI ( efirc ) ) );
+ leak = 1;
+ }
+ efi_nullify_usbio ( &usbintf->usbio );
err_install_protocol:
efi_usb_close_all ( usbintf );
efi_usb_free_all ( usbintf );
list_del ( &usbintf->list );
- free ( usbintf->path );
+ if ( ! leak )
+ free ( usbintf->path );
err_path:
- free ( usbintf );
+ if ( ! leak )
+ free ( usbintf );
err_alloc:
+ if ( leak ) {
+ DBGC ( usbdev, "USBDEV %s nullified and leaked\n",
+ usbintf->name );
+ }
return rc;
}
static void efi_usb_uninstall ( struct efi_usb_interface *usbintf ) {
EFI_BOOT_SERVICES *bs = efi_systab->BootServices;
struct efi_usb_device *usbdev = usbintf->usbdev;
+ int leak = efi_shutdown_in_progress;
+ EFI_STATUS efirc;
DBGC ( usbdev, "USBDEV %s uninstalling %s\n",
usbintf->name, efi_handle_name ( usbintf->handle ) );
* seems to be required on some platforms to avoid failures
* when uninstalling protocols.
*/
- bs->DisconnectController ( usbintf->handle, NULL, NULL );
+ if ( ! efi_shutdown_in_progress )
+ bs->DisconnectController ( usbintf->handle, NULL, NULL );
/* Uninstall protocols */
- bs->UninstallMultipleProtocolInterfaces (
+ if ( ( ! efi_shutdown_in_progress ) &&
+ ( ( efirc = bs->UninstallMultipleProtocolInterfaces (
usbintf->handle,
&efi_usb_io_protocol_guid, &usbintf->usbio,
&efi_device_path_protocol_guid, usbintf->path,
- NULL );
+ NULL ) ) != 0 ) ) {
+ DBGC ( usbdev, "USBDEV %s could not uninstall: %s\n",
+ usbintf->name, strerror ( -EEFI ( efirc ) ) );
+ leak = 1;
+ }
+ efi_nullify_usbio ( &usbintf->usbio );
/* Close and free all endpoints */
efi_usb_close_all ( usbintf );
list_del ( &usbintf->list );
/* Free device path */
- free ( usbintf->path );
+ if ( ! leak )
+ free ( usbintf->path );
/* Free interface */
- free ( usbintf );
+ if ( ! leak )
+ free ( usbintf );
+
+ /* Report leakage, if applicable */
+ if ( leak && ( ! efi_shutdown_in_progress ) ) {
+ DBGC ( usbdev, "USBDEV %s nullified and leaked\n",
+ usbintf->name );
+ }
}
/**