Fix various typos in EmbeddedPkg. Signed-off-by: Coeur <coeur@gmx.fr> Reviewed-by: Leif Lindholm <leif.lindholm@linaro.org>
		
			
				
	
	
		
			548 lines
		
	
	
		
			15 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			548 lines
		
	
	
		
			15 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
/** @file
 | 
						|
  Serial IO Abstraction for GDB stub. This allows an EFI consoles that shows up on the system
 | 
						|
  running GDB. One console for error information and another console for user input/output.
 | 
						|
 | 
						|
  Basic packet format is $packet-data#checksum. So every command has 4 bytes of overhead: $,
 | 
						|
  #, 0, 0. The 0 and 0 are the ascii characters for the checksum.
 | 
						|
 | 
						|
 | 
						|
  Copyright (c) 2008 - 2009, Apple Inc. All rights reserved.<BR>
 | 
						|
 | 
						|
  SPDX-License-Identifier: BSD-2-Clause-Patent
 | 
						|
 | 
						|
**/
 | 
						|
 | 
						|
#include <GdbStubInternal.h>
 | 
						|
 | 
						|
//
 | 
						|
// Set TRUE if F Reply package signals a ctrl-c. We can not process the Ctrl-c
 | 
						|
// here we need to wait for the periodic callback to do this.
 | 
						|
//
 | 
						|
BOOLEAN gCtrlCBreakFlag = FALSE;
 | 
						|
 | 
						|
//
 | 
						|
// If the periodic callback is called while we are processing an F packet we need
 | 
						|
// to let the callback know to not read from the serial stream as it could steal
 | 
						|
// characters from the F response packet
 | 
						|
//
 | 
						|
BOOLEAN gProcessingFPacket = FALSE;
 | 
						|
 | 
						|
/**
 | 
						|
  Process a control-C break message.
 | 
						|
 | 
						|
  Currently a place holder, remove the ASSERT when it gets implemented.
 | 
						|
 | 
						|
  @param  ErrNo   Error information from the F reply packet or other source
 | 
						|
 | 
						|
**/
 | 
						|
 | 
						|
VOID
 | 
						|
GdbCtrlCBreakMessage (
 | 
						|
  IN  UINTN ErrNo
 | 
						|
  )
 | 
						|
{
 | 
						|
  // See D.10.5 of gdb.pdf
 | 
						|
  // This should look like a break message. Should look like SIGINT
 | 
						|
 | 
						|
  /* TODO: Make sure if we should do anything with ErrNo */
 | 
						|
  //Turn on the global Ctrl-C flag.
 | 
						|
  gCtrlCBreakFlag = TRUE;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
/**
 | 
						|
  Parse the F reply packet and extract the return value and an ErrNo if it exists.
 | 
						|
 | 
						|
  @param  Packet  Packet to parse like an F reply packet
 | 
						|
  @param  ErrNo   Buffer to hold Count bytes that were read
 | 
						|
 | 
						|
  @retval -1      Error, not a valid F reply packet
 | 
						|
  @retval other   Return the return code from the F reply packet
 | 
						|
 | 
						|
**/
 | 
						|
INTN
 | 
						|
GdbParseFReplyPacket (
 | 
						|
  IN  CHAR8   *Packet,
 | 
						|
  OUT UINTN   *ErrNo
 | 
						|
  )
 | 
						|
{
 | 
						|
  INTN   RetCode;
 | 
						|
 | 
						|
  if (Packet[0] != 'F') {
 | 
						|
    // A valid response would be an F packet
 | 
						|
    return -1;
 | 
						|
  }
 | 
						|
 | 
						|
  RetCode = AsciiStrHexToUintn (&Packet[1]);
 | 
						|
 | 
						|
  // Find 1st comma
 | 
						|
  for (;*Packet != '\0' && *Packet != ',';  Packet++);
 | 
						|
  if (*Packet == '\0') {
 | 
						|
    *ErrNo = 0;
 | 
						|
    return RetCode;
 | 
						|
  }
 | 
						|
 | 
						|
  *ErrNo = AsciiStrHexToUintn (++Packet);
 | 
						|
 | 
						|
  // Find 2nd comma
 | 
						|
  for (;*Packet != '\0' && *Packet != ',';  Packet++);
 | 
						|
  if (*Packet == '\0') {
 | 
						|
    return RetCode;
 | 
						|
  }
 | 
						|
 | 
						|
  if (*(++Packet) == 'C') {
 | 
						|
    GdbCtrlCBreakMessage (*ErrNo);
 | 
						|
  }
 | 
						|
 | 
						|
  return RetCode;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
/**
 | 
						|
  Read data from a FileDescriptor. On success number of bytes read is returned. Zero indicates
 | 
						|
  the end of a file. On error -1 is returned. If count is zero, GdbRead returns zero.
 | 
						|
 | 
						|
  @param  FileDescriptor   Device to talk to.
 | 
						|
  @param  Buffer           Buffer to hold Count bytes that were read
 | 
						|
  @param  Count            Number of bytes to transfer.
 | 
						|
 | 
						|
  @retval -1               Error
 | 
						|
  @retval {other}          Number of bytes read.
 | 
						|
 | 
						|
**/
 | 
						|
INTN
 | 
						|
GdbRead (
 | 
						|
  IN  INTN    FileDescriptor,
 | 
						|
  OUT VOID    *Buffer,
 | 
						|
  IN  UINTN   Count
 | 
						|
  )
 | 
						|
{
 | 
						|
  CHAR8   Packet[128];
 | 
						|
  UINTN   Size;
 | 
						|
  INTN    RetCode;
 | 
						|
  UINTN   ErrNo;
 | 
						|
  BOOLEAN ReceiveDone = FALSE;
 | 
						|
 | 
						|
  // Send:
 | 
						|
  // "Fread,XX,YYYYYYYY,XX
 | 
						|
  //
 | 
						|
  // XX - FileDescriptor in ASCII
 | 
						|
  // YYYYYYYY - Buffer address in ASCII
 | 
						|
  // XX - Count in ASCII
 | 
						|
  // SS - check sum
 | 
						|
  //
 | 
						|
  Size = AsciiSPrint (Packet, sizeof (Packet), "Fread,%x,%x,%x", FileDescriptor, Buffer, Count);
 | 
						|
  // Packet array is too small if you got this ASSERT
 | 
						|
  ASSERT (Size < sizeof (Packet));
 | 
						|
 | 
						|
  gProcessingFPacket = TRUE;
 | 
						|
  SendPacket (Packet);
 | 
						|
  Print ((CHAR16 *)L"Packet sent..\n");
 | 
						|
 | 
						|
  do {
 | 
						|
    // Reply:
 | 
						|
    ReceivePacket (Packet, sizeof (Packet));
 | 
						|
    Print ((CHAR16 *)L"Command received..%c\n", Packet[0]);
 | 
						|
 | 
						|
    // Process GDB commands
 | 
						|
    switch (Packet[0]) {
 | 
						|
      //Write memory command.
 | 
						|
      //M addr,length:XX...
 | 
						|
      case    'M':
 | 
						|
        WriteToMemory (Packet);
 | 
						|
        break;
 | 
						|
 | 
						|
      //Fretcode, errno, Ctrl-C flag
 | 
						|
      //retcode - Count read
 | 
						|
      case    'F':
 | 
						|
        //Once target receives F reply packet that means the previous
 | 
						|
        //transactions are finished.
 | 
						|
        ReceiveDone = TRUE;
 | 
						|
        break;
 | 
						|
 | 
						|
      //Send empty buffer
 | 
						|
      default    :
 | 
						|
        SendNotSupported();
 | 
						|
        break;
 | 
						|
    }
 | 
						|
  } while (ReceiveDone == FALSE);
 | 
						|
 | 
						|
  RetCode = GdbParseFReplyPacket (Packet, &ErrNo);
 | 
						|
  Print ((CHAR16 *)L"RetCode: %x..ErrNo: %x..\n", RetCode, ErrNo);
 | 
						|
 | 
						|
  if (ErrNo > 0) {
 | 
						|
    //Send error to the host if there is any.
 | 
						|
    SendError ((UINT8)ErrNo);
 | 
						|
  }
 | 
						|
 | 
						|
  gProcessingFPacket = FALSE;
 | 
						|
 | 
						|
  return RetCode;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
/**
 | 
						|
  Write data to a FileDescriptor. On success number of bytes written is returned. Zero indicates
 | 
						|
  nothing was written. On error -1 is returned.
 | 
						|
 | 
						|
  @param  FileDescriptor   Device to talk to.
 | 
						|
  @param  Buffer           Buffer to hold Count bytes that are to be written
 | 
						|
  @param  Count            Number of bytes to transfer.
 | 
						|
 | 
						|
  @retval -1               Error
 | 
						|
  @retval {other}          Number of bytes written.
 | 
						|
 | 
						|
**/
 | 
						|
INTN
 | 
						|
GdbWrite (
 | 
						|
  IN  INTN          FileDescriptor,
 | 
						|
  OUT CONST VOID    *Buffer,
 | 
						|
  IN  UINTN         Count
 | 
						|
  )
 | 
						|
{
 | 
						|
  CHAR8   Packet[128];
 | 
						|
  UINTN   Size;
 | 
						|
  INTN    RetCode;
 | 
						|
  UINTN   ErrNo;
 | 
						|
  BOOLEAN ReceiveDone = FALSE;
 | 
						|
 | 
						|
  // Send:
 | 
						|
  // #Fwrite,XX,YYYYYYYY,XX$SS
 | 
						|
  //
 | 
						|
  // XX - FileDescriptor in ASCII
 | 
						|
  // YYYYYYYY - Buffer address in ASCII
 | 
						|
  // XX - Count in ASCII
 | 
						|
  // SS - check sum
 | 
						|
  //
 | 
						|
  Size = AsciiSPrint (Packet, sizeof (Packet), "Fwrite,%x,%x,%x", FileDescriptor, Buffer, Count);
 | 
						|
  // Packet array is too small if you got this ASSERT
 | 
						|
  ASSERT (Size < sizeof (Packet));
 | 
						|
 | 
						|
  SendPacket (Packet);
 | 
						|
  Print ((CHAR16 *)L"Packet sent..\n");
 | 
						|
 | 
						|
  do {
 | 
						|
    // Reply:
 | 
						|
    ReceivePacket (Packet, sizeof (Packet));
 | 
						|
    Print ((CHAR16 *)L"Command received..%c\n", Packet[0]);
 | 
						|
 | 
						|
    // Process GDB commands
 | 
						|
    switch (Packet[0]) {
 | 
						|
      //Read memory command.
 | 
						|
      //m addr,length.
 | 
						|
      case    'm':
 | 
						|
        ReadFromMemory (Packet);
 | 
						|
        break;
 | 
						|
 | 
						|
      //Fretcode, errno, Ctrl-C flag
 | 
						|
      //retcode - Count read
 | 
						|
      case    'F':
 | 
						|
        //Once target receives F reply packet that means the previous
 | 
						|
        //transactions are finished.
 | 
						|
        ReceiveDone = TRUE;
 | 
						|
        break;
 | 
						|
 | 
						|
      //Send empty buffer
 | 
						|
      default    :
 | 
						|
        SendNotSupported();
 | 
						|
        break;
 | 
						|
    }
 | 
						|
  } while (ReceiveDone == FALSE);
 | 
						|
 | 
						|
  RetCode = GdbParseFReplyPacket (Packet, &ErrNo);
 | 
						|
  Print ((CHAR16 *)L"RetCode: %x..ErrNo: %x..\n", RetCode, ErrNo);
 | 
						|
 | 
						|
  //Send error to the host if there is any.
 | 
						|
  if (ErrNo > 0) {
 | 
						|
    SendError((UINT8)ErrNo);
 | 
						|
  }
 | 
						|
 | 
						|
  return RetCode;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
/**
 | 
						|
  Reset the serial device.
 | 
						|
 | 
						|
  @param  This              Protocol instance pointer.
 | 
						|
 | 
						|
  @retval EFI_SUCCESS       The device was reset.
 | 
						|
  @retval EFI_DEVICE_ERROR  The serial device could not be reset.
 | 
						|
 | 
						|
**/
 | 
						|
EFI_STATUS
 | 
						|
EFIAPI
 | 
						|
GdbSerialReset (
 | 
						|
  IN EFI_SERIAL_IO_PROTOCOL  *This
 | 
						|
  )
 | 
						|
{
 | 
						|
  return EFI_SUCCESS;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
/**
 | 
						|
  Sets the baud rate, receive FIFO depth, transmit/receive time out, parity,
 | 
						|
  data buts, and stop bits on a serial device.
 | 
						|
 | 
						|
  @param  This             Protocol instance pointer.
 | 
						|
  @param  BaudRate         The requested baud rate. A BaudRate value of 0 will use the the
 | 
						|
                           device's default interface speed.
 | 
						|
  @param  ReceiveFifoDepth The requested depth of the FIFO on the receive side of the
 | 
						|
                           serial interface. A ReceiveFifoDepth value of 0 will use
 | 
						|
                           the device's default FIFO depth.
 | 
						|
  @param  Timeout          The requested time out for a single character in microseconds.
 | 
						|
                           This timeout applies to both the transmit and receive side of the
 | 
						|
                           interface. A Timeout value of 0 will use the device's default time
 | 
						|
                           out value.
 | 
						|
  @param  Parity           The type of parity to use on this serial device. A Parity value of
 | 
						|
                           DefaultParity will use the device's default parity value.
 | 
						|
  @param  DataBits         The number of data bits to use on the serial device. A DataBits
 | 
						|
                           value of 0 will use the device's default data bit setting.
 | 
						|
  @param  StopBits         The number of stop bits to use on this serial device. A StopBits
 | 
						|
                           value of DefaultStopBits will use the device's default number of
 | 
						|
                           stop bits.
 | 
						|
 | 
						|
  @retval EFI_SUCCESS      The device was reset.
 | 
						|
  @retval EFI_DEVICE_ERROR The serial device could not be reset.
 | 
						|
 | 
						|
**/
 | 
						|
EFI_STATUS
 | 
						|
EFIAPI
 | 
						|
GdbSerialSetAttributes (
 | 
						|
  IN EFI_SERIAL_IO_PROTOCOL  *This,
 | 
						|
  IN UINT64                  BaudRate,
 | 
						|
  IN UINT32                  ReceiveFifoDepth,
 | 
						|
  IN UINT32                  Timeout,
 | 
						|
  IN EFI_PARITY_TYPE         Parity,
 | 
						|
  IN UINT8                   DataBits,
 | 
						|
  IN EFI_STOP_BITS_TYPE      StopBits
 | 
						|
  )
 | 
						|
{
 | 
						|
  return EFI_UNSUPPORTED;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
/**
 | 
						|
  Set the control bits on a serial device
 | 
						|
 | 
						|
  @param  This             Protocol instance pointer.
 | 
						|
  @param  Control          Set the bits of Control that are settable.
 | 
						|
 | 
						|
  @retval EFI_SUCCESS      The new control bits were set on the serial device.
 | 
						|
  @retval EFI_UNSUPPORTED  The serial device does not support this operation.
 | 
						|
  @retval EFI_DEVICE_ERROR The serial device is not functioning correctly.
 | 
						|
 | 
						|
**/
 | 
						|
EFI_STATUS
 | 
						|
EFIAPI
 | 
						|
GdbSerialSetControl (
 | 
						|
  IN EFI_SERIAL_IO_PROTOCOL  *This,
 | 
						|
  IN UINT32                  Control
 | 
						|
  )
 | 
						|
{
 | 
						|
  return EFI_UNSUPPORTED;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
/**
 | 
						|
  Retrieves the status of the control bits on a serial device
 | 
						|
 | 
						|
  @param  This              Protocol instance pointer.
 | 
						|
  @param  Control           A pointer to return the current Control signals from the serial device.
 | 
						|
 | 
						|
  @retval EFI_SUCCESS       The control bits were read from the serial device.
 | 
						|
  @retval EFI_DEVICE_ERROR  The serial device is not functioning correctly.
 | 
						|
 | 
						|
**/
 | 
						|
EFI_STATUS
 | 
						|
EFIAPI
 | 
						|
GdbSerialGetControl (
 | 
						|
  IN EFI_SERIAL_IO_PROTOCOL  *This,
 | 
						|
  OUT UINT32                 *Control
 | 
						|
  )
 | 
						|
{
 | 
						|
  return EFI_UNSUPPORTED;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
/**
 | 
						|
  Writes data to a serial device.
 | 
						|
 | 
						|
  @param  This              Protocol instance pointer.
 | 
						|
  @param  BufferSize        On input, the size of the Buffer. On output, the amount of
 | 
						|
                            data actually written.
 | 
						|
  @param  Buffer            The buffer of data to write
 | 
						|
 | 
						|
  @retval EFI_SUCCESS       The data was written.
 | 
						|
  @retval EFI_DEVICE_ERROR  The device reported an error.
 | 
						|
  @retval EFI_TIMEOUT       The data write was stopped due to a timeout.
 | 
						|
 | 
						|
**/
 | 
						|
EFI_STATUS
 | 
						|
EFIAPI
 | 
						|
GdbSerialWrite (
 | 
						|
  IN EFI_SERIAL_IO_PROTOCOL  *This,
 | 
						|
  IN OUT UINTN               *BufferSize,
 | 
						|
  IN VOID                    *Buffer
 | 
						|
  )
 | 
						|
{
 | 
						|
  GDB_SERIAL_DEV  *SerialDev;
 | 
						|
  UINTN            Return;
 | 
						|
 | 
						|
  SerialDev = GDB_SERIAL_DEV_FROM_THIS (This);
 | 
						|
 | 
						|
  Return = GdbWrite (SerialDev->OutFileDescriptor, Buffer, *BufferSize);
 | 
						|
  if (Return == (UINTN)-1) {
 | 
						|
    return EFI_DEVICE_ERROR;
 | 
						|
  }
 | 
						|
 | 
						|
  if (Return != *BufferSize) {
 | 
						|
    *BufferSize = Return;
 | 
						|
  }
 | 
						|
 | 
						|
  return EFI_SUCCESS;
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
  Writes data to a serial device.
 | 
						|
 | 
						|
  @param  This              Protocol instance pointer.
 | 
						|
  @param  BufferSize        On input, the size of the Buffer. On output, the amount of
 | 
						|
                            data returned in Buffer.
 | 
						|
  @param  Buffer            The buffer to return the data into.
 | 
						|
 | 
						|
  @retval EFI_SUCCESS       The data was read.
 | 
						|
  @retval EFI_DEVICE_ERROR  The device reported an error.
 | 
						|
  @retval EFI_TIMEOUT       The data write was stopped due to a timeout.
 | 
						|
 | 
						|
**/
 | 
						|
 | 
						|
EFI_STATUS
 | 
						|
EFIAPI
 | 
						|
GdbSerialRead (
 | 
						|
  IN EFI_SERIAL_IO_PROTOCOL  *This,
 | 
						|
  IN OUT UINTN               *BufferSize,
 | 
						|
  OUT VOID                   *Buffer
 | 
						|
  )
 | 
						|
{
 | 
						|
  GDB_SERIAL_DEV  *SerialDev;
 | 
						|
  UINTN            Return;
 | 
						|
 | 
						|
  SerialDev = GDB_SERIAL_DEV_FROM_THIS (This);
 | 
						|
 | 
						|
  Return = GdbRead (SerialDev->InFileDescriptor, Buffer, *BufferSize);
 | 
						|
  if (Return == (UINTN)-1) {
 | 
						|
    return EFI_DEVICE_ERROR;
 | 
						|
  }
 | 
						|
 | 
						|
  if (Return != *BufferSize) {
 | 
						|
    *BufferSize = Return;
 | 
						|
  }
 | 
						|
 | 
						|
  return EFI_SUCCESS;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
//
 | 
						|
// Template used to initialize the GDB Serial IO protocols
 | 
						|
//
 | 
						|
GDB_SERIAL_DEV gdbSerialDevTemplate = {
 | 
						|
  GDB_SERIAL_DEV_SIGNATURE,
 | 
						|
  NULL,
 | 
						|
 | 
						|
  { // SerialIo
 | 
						|
    SERIAL_IO_INTERFACE_REVISION,
 | 
						|
    GdbSerialReset,
 | 
						|
    GdbSerialSetAttributes,
 | 
						|
    GdbSerialSetControl,
 | 
						|
    GdbSerialGetControl,
 | 
						|
    GdbSerialWrite,
 | 
						|
    GdbSerialRead,
 | 
						|
    NULL
 | 
						|
  },
 | 
						|
  { // SerialMode
 | 
						|
    0,      // ControlMask
 | 
						|
    0,      // Timeout
 | 
						|
    0,      // BaudRate
 | 
						|
    1,      // RceiveFifoDepth
 | 
						|
    0,      // DataBits
 | 
						|
    0,      // Parity
 | 
						|
    0       // StopBits
 | 
						|
  },
 | 
						|
  {
 | 
						|
    {
 | 
						|
      {
 | 
						|
        HARDWARE_DEVICE_PATH,
 | 
						|
        HW_VENDOR_DP,
 | 
						|
        {
 | 
						|
          (UINT8) (sizeof (VENDOR_DEVICE_PATH) + sizeof (UINT32)),
 | 
						|
          (UINT8) ((sizeof (VENDOR_DEVICE_PATH) + sizeof (UINT32)) >> 8)
 | 
						|
        },
 | 
						|
      },
 | 
						|
      EFI_SERIAL_IO_PROTOCOL_GUID
 | 
						|
    },
 | 
						|
    0,
 | 
						|
    {
 | 
						|
      END_DEVICE_PATH_TYPE,
 | 
						|
      END_ENTIRE_DEVICE_PATH_SUBTYPE,
 | 
						|
      {
 | 
						|
        (UINT8) (sizeof (EFI_DEVICE_PATH_PROTOCOL)),
 | 
						|
        (UINT8) (sizeof (EFI_DEVICE_PATH_PROTOCOL) >> 8)
 | 
						|
      }
 | 
						|
    },
 | 
						|
  },
 | 
						|
  GDB_STDIN,
 | 
						|
  GDB_STDOUT
 | 
						|
};
 | 
						|
 | 
						|
 | 
						|
/**
 | 
						|
  Make two serial consoles: 1) StdIn and StdOut via GDB. 2) StdErr via GDB.
 | 
						|
 | 
						|
  These console show up on the remote system running GDB
 | 
						|
 | 
						|
**/
 | 
						|
VOID
 | 
						|
GdbInitializeSerialConsole (
 | 
						|
  VOID
 | 
						|
  )
 | 
						|
{
 | 
						|
  EFI_STATUS      Status;
 | 
						|
  GDB_SERIAL_DEV  *StdOutSerialDev;
 | 
						|
  GDB_SERIAL_DEV  *StdErrSerialDev;
 | 
						|
 | 
						|
  // Use the template to make a copy of the Serial Console private data structure.
 | 
						|
  StdOutSerialDev = AllocateCopyPool (sizeof (GDB_SERIAL_DEV),  &gdbSerialDevTemplate);
 | 
						|
  ASSERT (StdOutSerialDev != NULL);
 | 
						|
 | 
						|
  // Fixup pointer after the copy
 | 
						|
  StdOutSerialDev->SerialIo.Mode = &StdOutSerialDev->SerialMode;
 | 
						|
 | 
						|
  StdErrSerialDev = AllocateCopyPool (sizeof (GDB_SERIAL_DEV),  &gdbSerialDevTemplate);
 | 
						|
  ASSERT (StdErrSerialDev != NULL);
 | 
						|
 | 
						|
  // Fixup pointer and modify stuff that is different for StdError
 | 
						|
  StdErrSerialDev->SerialIo.Mode = &StdErrSerialDev->SerialMode;
 | 
						|
  StdErrSerialDev->DevicePath.Index = 1;
 | 
						|
  StdErrSerialDev->OutFileDescriptor = GDB_STDERR;
 | 
						|
 | 
						|
  // Make a new handle with Serial IO protocol and its device path on it.
 | 
						|
  Status = gBS->InstallMultipleProtocolInterfaces (
 | 
						|
                  &StdOutSerialDev->Handle,
 | 
						|
                  &gEfiSerialIoProtocolGuid,   &StdOutSerialDev->SerialIo,
 | 
						|
                  &gEfiDevicePathProtocolGuid, &StdOutSerialDev->DevicePath,
 | 
						|
                  NULL
 | 
						|
                  );
 | 
						|
  ASSERT_EFI_ERROR (Status);
 | 
						|
 | 
						|
  // Make a new handle with Serial IO protocol and its device path on it.
 | 
						|
  Status = gBS->InstallMultipleProtocolInterfaces (
 | 
						|
                  &StdErrSerialDev->Handle,
 | 
						|
                  &gEfiSerialIoProtocolGuid,   &StdErrSerialDev->SerialIo,
 | 
						|
                  &gEfiDevicePathProtocolGuid, &StdErrSerialDev->DevicePath,
 | 
						|
                  NULL
 | 
						|
                  );
 | 
						|
  ASSERT_EFI_ERROR (Status);
 | 
						|
}
 | 
						|
 |