/** @file
  Http IO Helper Library.
  (C) Copyright 2020 Hewlett-Packard Development Company, L.P.
  SPDX-License-Identifier: BSD-2-Clause-Patent
**/
#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 
/**
  Notify the callback function when an event is triggered.
  @param[in]  Context         The opaque parameter to the function.
**/
VOID
EFIAPI
HttpIoNotifyDpc (
  IN VOID                *Context
  )
{
  *((BOOLEAN *) Context) = TRUE;
}
/**
  Request HttpIoNotifyDpc as a DPC at TPL_CALLBACK.
  @param[in]  Event                 The event signaled.
  @param[in]  Context               The opaque parameter to the function.
**/
VOID
EFIAPI
HttpIoNotify (
  IN EFI_EVENT              Event,
  IN VOID                   *Context
  )
{
  //
  // Request HttpIoNotifyDpc as a DPC at TPL_CALLBACK
  //
  QueueDpc (TPL_CALLBACK, HttpIoNotifyDpc, Context);
}
/**
  Destroy the HTTP_IO and release the resources.
  @param[in]  HttpIo          The HTTP_IO which wraps the HTTP service to be destroyed.
**/
VOID
HttpIoDestroyIo (
  IN HTTP_IO                *HttpIo
  )
{
  EFI_HTTP_PROTOCOL         *Http;
  EFI_EVENT                 Event;
  if (HttpIo == NULL) {
    return;
  }
  Event = HttpIo->ReqToken.Event;
  if (Event != NULL) {
    gBS->CloseEvent (Event);
  }
  Event = HttpIo->RspToken.Event;
  if (Event != NULL) {
    gBS->CloseEvent (Event);
  }
  Event = HttpIo->TimeoutEvent;
  if (Event != NULL) {
    gBS->CloseEvent (Event);
  }
  Http = HttpIo->Http;
  if (Http != NULL) {
    Http->Configure (Http, NULL);
    gBS->CloseProtocol (
           HttpIo->Handle,
           &gEfiHttpProtocolGuid,
           HttpIo->Image,
           HttpIo->Controller
           );
  }
  NetLibDestroyServiceChild (
    HttpIo->Controller,
    HttpIo->Image,
    &gEfiHttpServiceBindingProtocolGuid,
    HttpIo->Handle
    );
}
/**
  Create a HTTP_IO to access the HTTP service. It will create and configure
  a HTTP child handle.
  @param[in]  Image          The handle of the driver image.
  @param[in]  Controller     The handle of the controller.
  @param[in]  IpVersion      IP_VERSION_4 or IP_VERSION_6.
  @param[in]  ConfigData     The HTTP_IO configuration data ,
                             NULL means not to configure the HTTP child.
  @param[in]  Callback       Callback function which will be invoked when specified
                             HTTP_IO_CALLBACK_EVENT happened.
  @param[in]  Context        The Context data which will be passed to the Callback function.
  @param[out] HttpIo         The HTTP_IO.
  @retval EFI_SUCCESS            The HTTP_IO is created and configured.
  @retval EFI_INVALID_PARAMETER  One or more parameters are invalid.
  @retval EFI_UNSUPPORTED        One or more of the control options are not
                                 supported in the implementation.
  @retval EFI_OUT_OF_RESOURCES   Failed to allocate memory.
  @retval Others                 Failed to create the HTTP_IO or configure it.
**/
EFI_STATUS
HttpIoCreateIo (
  IN EFI_HANDLE             Image,
  IN EFI_HANDLE             Controller,
  IN UINT8                  IpVersion,
  IN HTTP_IO_CONFIG_DATA    *ConfigData, OPTIONAL
  IN HTTP_IO_CALLBACK       Callback,
  IN VOID                   *Context,
  OUT HTTP_IO               *HttpIo
  )
{
  EFI_STATUS                Status;
  EFI_HTTP_CONFIG_DATA      HttpConfigData;
  EFI_HTTPv4_ACCESS_POINT   Http4AccessPoint;
  EFI_HTTPv6_ACCESS_POINT   Http6AccessPoint;
  EFI_HTTP_PROTOCOL         *Http;
  EFI_EVENT                 Event;
  if ((Image == NULL) || (Controller == NULL) || (HttpIo == NULL)) {
    return EFI_INVALID_PARAMETER;
  }
  if (IpVersion != IP_VERSION_4 && IpVersion != IP_VERSION_6) {
    return EFI_UNSUPPORTED;
  }
  ZeroMem (HttpIo, sizeof (HTTP_IO));
  ZeroMem (&HttpConfigData, sizeof (EFI_HTTP_CONFIG_DATA));
  //
  // Create the HTTP child instance and get the HTTP protocol.
  //
  Status = NetLibCreateServiceChild (
             Controller,
             Image,
             &gEfiHttpServiceBindingProtocolGuid,
             &HttpIo->Handle
             );
  if (EFI_ERROR (Status)) {
    return Status;
  }
  Status = gBS->OpenProtocol (
                  HttpIo->Handle,
                  &gEfiHttpProtocolGuid,
                  (VOID **) &Http,
                  Image,
                  Controller,
                  EFI_OPEN_PROTOCOL_BY_DRIVER
                  );
  if (EFI_ERROR (Status) || (Http == NULL)) {
    goto ON_ERROR;
  }
  //
  // Init the configuration data and configure the HTTP child.
  //
  HttpIo->Image       = Image;
  HttpIo->Controller  = Controller;
  HttpIo->IpVersion   = IpVersion;
  HttpIo->Http        = Http;
  HttpIo->Callback    = Callback;
  HttpIo->Context     = Context;
  if (ConfigData != NULL) {
    if (HttpIo->IpVersion == IP_VERSION_4) {
      HttpConfigData.LocalAddressIsIPv6  = FALSE;
      HttpConfigData.HttpVersion         = ConfigData->Config4.HttpVersion;
      HttpConfigData.TimeOutMillisec     = ConfigData->Config4.RequestTimeOut;
      Http4AccessPoint.UseDefaultAddress = ConfigData->Config4.UseDefaultAddress;
      Http4AccessPoint.LocalPort         = ConfigData->Config4.LocalPort;
      IP4_COPY_ADDRESS (&Http4AccessPoint.LocalAddress, &ConfigData->Config4.LocalIp);
      IP4_COPY_ADDRESS (&Http4AccessPoint.LocalSubnet, &ConfigData->Config4.SubnetMask);
      HttpConfigData.AccessPoint.IPv4Node = &Http4AccessPoint;
    } else {
      HttpConfigData.LocalAddressIsIPv6 = TRUE;
      HttpConfigData.HttpVersion         = ConfigData->Config6.HttpVersion;
      HttpConfigData.TimeOutMillisec     = ConfigData->Config6.RequestTimeOut;
      Http6AccessPoint.LocalPort         = ConfigData->Config6.LocalPort;
      IP6_COPY_ADDRESS (&Http6AccessPoint.LocalAddress, &ConfigData->Config6.LocalIp);
      HttpConfigData.AccessPoint.IPv6Node = &Http6AccessPoint;
    }
    Status = Http->Configure (Http, &HttpConfigData);
    if (EFI_ERROR (Status)) {
      goto ON_ERROR;
    }
  }
  //
  // Create events for variuos asynchronous operations.
  //
  Status = gBS->CreateEvent (
                  EVT_NOTIFY_SIGNAL,
                  TPL_NOTIFY,
                  HttpIoNotify,
                  &HttpIo->IsTxDone,
                  &Event
                  );
  if (EFI_ERROR (Status)) {
    goto ON_ERROR;
  }
  HttpIo->ReqToken.Event = Event;
  HttpIo->ReqToken.Message = &HttpIo->ReqMessage;
  Status = gBS->CreateEvent (
                  EVT_NOTIFY_SIGNAL,
                  TPL_NOTIFY,
                  HttpIoNotify,
                  &HttpIo->IsRxDone,
                  &Event
                  );
  if (EFI_ERROR (Status)) {
    goto ON_ERROR;
  }
  HttpIo->RspToken.Event = Event;
  HttpIo->RspToken.Message = &HttpIo->RspMessage;
  //
  // Create TimeoutEvent for response
  //
  Status = gBS->CreateEvent (
                  EVT_TIMER,
                  TPL_CALLBACK,
                  NULL,
                  NULL,
                  &Event
                  );
  if (EFI_ERROR (Status)) {
    goto ON_ERROR;
  }
  HttpIo->TimeoutEvent = Event;
  return EFI_SUCCESS;
ON_ERROR:
  HttpIoDestroyIo (HttpIo);
  return Status;
}
/**
  Synchronously send a HTTP REQUEST message to the server.
  @param[in]   HttpIo           The HttpIo wrapping the HTTP service.
  @param[in]   Request          A pointer to storage such data as URL and HTTP method.
  @param[in]   HeaderCount      Number of HTTP header structures in Headers list.
  @param[in]   Headers          Array containing list of HTTP headers.
  @param[in]   BodyLength       Length in bytes of the HTTP body.
  @param[in]   Body             Body associated with the HTTP request.
  @retval EFI_SUCCESS            The HTTP request is trasmitted.
  @retval EFI_INVALID_PARAMETER  One or more parameters are invalid.
  @retval EFI_OUT_OF_RESOURCES   Failed to allocate memory.
  @retval EFI_DEVICE_ERROR       An unexpected network or system error occurred.
  @retval Others                 Other errors as indicated.
**/
EFI_STATUS
HttpIoSendRequest (
  IN  HTTP_IO                *HttpIo,
  IN  EFI_HTTP_REQUEST_DATA  *Request,
  IN  UINTN                  HeaderCount,
  IN  EFI_HTTP_HEADER        *Headers,
  IN  UINTN                  BodyLength,
  IN  VOID                   *Body
  )
{
  EFI_STATUS                 Status;
  EFI_HTTP_PROTOCOL          *Http;
  if (HttpIo == NULL || HttpIo->Http == NULL) {
    return EFI_INVALID_PARAMETER;
  }
  HttpIo->ReqToken.Status  = EFI_NOT_READY;
  HttpIo->ReqToken.Message->Data.Request = Request;
  HttpIo->ReqToken.Message->HeaderCount  = HeaderCount;
  HttpIo->ReqToken.Message->Headers      = Headers;
  HttpIo->ReqToken.Message->BodyLength   = BodyLength;
  HttpIo->ReqToken.Message->Body         = Body;
  if (HttpIo->Callback != NULL) {
    Status = HttpIo->Callback (
               HttpIoRequest,
               HttpIo->ReqToken.Message,
               HttpIo->Context
               );
    if (EFI_ERROR (Status)) {
      return Status;
    }
  }
  //
  // Queue the request token to HTTP instances.
  //
  Http = HttpIo->Http;
  HttpIo->IsTxDone = FALSE;
  Status = Http->Request (
                   Http,
                   &HttpIo->ReqToken
                   );
  if (EFI_ERROR (Status)) {
    return Status;
  }
  //
  // Poll the network until transmit finish.
  //
  while (!HttpIo->IsTxDone) {
    Http->Poll (Http);
  }
  return HttpIo->ReqToken.Status;
}
/**
  Synchronously receive a HTTP RESPONSE message from the server.
  @param[in]   HttpIo           The HttpIo wrapping the HTTP service.
  @param[in]   RecvMsgHeader    TRUE to receive a new HTTP response (from message header).
                                FALSE to continue receive the previous response message.
  @param[out]  ResponseData     Point to a wrapper of the received response data.
  @retval EFI_SUCCESS            The HTTP response is received.
  @retval EFI_INVALID_PARAMETER  One or more parameters are invalid.
  @retval EFI_OUT_OF_RESOURCES   Failed to allocate memory.
  @retval EFI_DEVICE_ERROR       An unexpected network or system error occurred.
  @retval Others                 Other errors as indicated.
**/
EFI_STATUS
HttpIoRecvResponse (
  IN      HTTP_IO                  *HttpIo,
  IN      BOOLEAN                  RecvMsgHeader,
  OUT     HTTP_IO_RESPONSE_DATA    *ResponseData
  )
{
  EFI_STATUS                 Status;
  EFI_HTTP_PROTOCOL          *Http;
  if (HttpIo == NULL || HttpIo->Http == NULL || ResponseData == NULL) {
    return EFI_INVALID_PARAMETER;
  }
  //
  // Queue the response token to HTTP instances.
  //
  HttpIo->RspToken.Status  = EFI_NOT_READY;
  if (RecvMsgHeader) {
    HttpIo->RspToken.Message->Data.Response = &ResponseData->Response;
  } else {
    HttpIo->RspToken.Message->Data.Response = NULL;
  }
  HttpIo->RspToken.Message->HeaderCount   = 0;
  HttpIo->RspToken.Message->Headers       = NULL;
  HttpIo->RspToken.Message->BodyLength    = ResponseData->BodyLength;
  HttpIo->RspToken.Message->Body          = ResponseData->Body;
  Http = HttpIo->Http;
  HttpIo->IsRxDone = FALSE;
  //
  // Start the timer, and wait Timeout seconds to receive the header packet.
  //
  Status = gBS->SetTimer (HttpIo->TimeoutEvent, TimerRelative, HttpIo->Timeout * TICKS_PER_MS);
  if (EFI_ERROR (Status)) {
    return Status;
  }
  Status = Http->Response (
                   Http,
                   &HttpIo->RspToken
                   );
  if (EFI_ERROR (Status)) {
    //
    // Remove timeout timer from the event list.
    //
    gBS->SetTimer (HttpIo->TimeoutEvent, TimerCancel, 0);
    return Status;
  }
  //
  // Poll the network until receive finish.
  //
  while (!HttpIo->IsRxDone && EFI_ERROR (gBS->CheckEvent (HttpIo->TimeoutEvent))) {
    Http->Poll (Http);
  }
  //
  // Remove timeout timer from the event list.
  //
  gBS->SetTimer (HttpIo->TimeoutEvent, TimerCancel, 0);
  if (!HttpIo->IsRxDone) {
    //
    // Timeout occurs, cancel the response token.
    //
    Http->Cancel (Http, &HttpIo->RspToken);
    Status = EFI_TIMEOUT;
    return Status;
  } else {
    HttpIo->IsRxDone = FALSE;
  }
  if ((HttpIo->Callback != NULL) &&
      (HttpIo->RspToken.Status == EFI_SUCCESS || HttpIo->RspToken.Status == EFI_HTTP_ERROR)) {
    Status = HttpIo->Callback (
               HttpIoResponse,
               HttpIo->RspToken.Message,
               HttpIo->Context
               );
    if (EFI_ERROR (Status)) {
      return Status;
    }
  }
  //
  // Store the received data into the wrapper.
  //
  ResponseData->Status = HttpIo->RspToken.Status;
  ResponseData->HeaderCount = HttpIo->RspToken.Message->HeaderCount;
  ResponseData->Headers     = HttpIo->RspToken.Message->Headers;
  ResponseData->BodyLength  = HttpIo->RspToken.Message->BodyLength;
  return Status;
}
/**
  Get the value of the content length if there is a "Content-Length" header.
  @param[in]    HeaderCount        Number of HTTP header structures in Headers.
  @param[in]    Headers            Array containing list of HTTP headers.
  @param[out]   ContentLength      Pointer to save the value of the content length.
  @retval EFI_SUCCESS              Successfully get the content length.
  @retval EFI_NOT_FOUND            No "Content-Length" header in the Headers.
**/
EFI_STATUS
HttpIoGetContentLength (
  IN     UINTN                HeaderCount,
  IN     EFI_HTTP_HEADER      *Headers,
  OUT    UINTN                *ContentLength
  )
{
  EFI_HTTP_HEADER       *Header;
  Header = HttpFindHeader (HeaderCount, Headers, HTTP_HEADER_CONTENT_LENGTH);
  if (Header == NULL) {
    return EFI_NOT_FOUND;
  }
  return AsciiStrDecimalToUintnS (Header->FieldValue, (CHAR8 **) NULL, ContentLength);
}
/**
  Send HTTP request in chunks.
  @param[in]   HttpIo             The HttpIo wrapping the HTTP service.
  @param[in]   SendChunkProcess   Pointer to current chunk process status.
  @param[in]   RequestMessage     Request to send.
  @retval EFI_SUCCESS             Successfully to send chunk data according to SendChunkProcess.
  @retval Other                   Other errors.
**/
EFI_STATUS
HttpIoSendChunkedTransfer (
  IN  HTTP_IO                    *HttpIo,
  IN  HTTP_IO_SEND_CHUNK_PROCESS *SendChunkProcess,
  IN  EFI_HTTP_MESSAGE           *RequestMessage
)
{
  EFI_STATUS            Status;
  EFI_HTTP_HEADER       *NewHeaders;
  EFI_HTTP_HEADER       *ContentLengthHeader;
  UINTN                 AddNewHeader;
  UINTN                 HeaderCount;
  CHAR8                 *MessageBody;
  UINTN                 MessageBodyLength;
  UINTN                 ChunkLength;
  CHAR8                 ChunkLengthStr [HTTP_IO_CHUNK_SIZE_STRING_LEN];
  EFI_HTTP_REQUEST_DATA *SentRequestData;
  AddNewHeader        = 0;
  NewHeaders          = NULL;
  MessageBody         = NULL;
  ContentLengthHeader = NULL;
  MessageBodyLength   = 0;
  switch (*SendChunkProcess) {
  case HttpIoSendChunkHeaderZeroContent:
      ContentLengthHeader = HttpFindHeader (RequestMessage->HeaderCount, RequestMessage->Headers, HTTP_HEADER_CONTENT_LENGTH);
      if (ContentLengthHeader == NULL) {
        AddNewHeader = 1;
      }
      NewHeaders = AllocateZeroPool ((RequestMessage->HeaderCount + AddNewHeader) * sizeof(EFI_HTTP_HEADER));
      CopyMem ((VOID*)NewHeaders, (VOID *)RequestMessage->Headers, RequestMessage->HeaderCount * sizeof (EFI_HTTP_HEADER));
      if (AddNewHeader == 0) {
        //
        // Override content-length to Transfer-Encoding.
        //
        ContentLengthHeader = HttpFindHeader (RequestMessage->HeaderCount, NewHeaders, HTTP_HEADER_CONTENT_LENGTH);
        ContentLengthHeader->FieldName = NULL;
        ContentLengthHeader->FieldValue = NULL;
      } else {
        ContentLengthHeader = NewHeaders + RequestMessage->HeaderCount;
      }
      HttpSetFieldNameAndValue (ContentLengthHeader, HTTP_HEADER_TRANSFER_ENCODING, HTTP_HEADER_TRANSFER_ENCODING_CHUNKED);
      HeaderCount = RequestMessage->HeaderCount + AddNewHeader;
      MessageBodyLength = 0;
      MessageBody = NULL;
      SentRequestData = RequestMessage->Data.Request;
      break;
  case HttpIoSendChunkContent:
      HeaderCount = 0;
      NewHeaders = NULL;
      SentRequestData = NULL;
      if (RequestMessage->BodyLength > HTTP_IO_MAX_SEND_PAYLOAD) {
        MessageBodyLength = HTTP_IO_MAX_SEND_PAYLOAD;
      } else {
        MessageBodyLength = RequestMessage->BodyLength;
      }
      AsciiSPrint (
          ChunkLengthStr,
          HTTP_IO_CHUNK_SIZE_STRING_LEN,
          "%x%c%c",
          MessageBodyLength,
          CHUNKED_TRANSFER_CODING_CR,
          CHUNKED_TRANSFER_CODING_LF
          );
      ChunkLength = AsciiStrLen (ChunkLengthStr);
      MessageBody = AllocatePool (ChunkLength + MessageBodyLength + 2);
      if (MessageBody == NULL) {
        DEBUG((DEBUG_ERROR, "Not enough memory for chunk transfer\n"));
        return EFI_OUT_OF_RESOURCES;
      }
      //
      // Build up the chunk transfer paylaod.
      //
      CopyMem (MessageBody, ChunkLengthStr, ChunkLength);
      CopyMem (MessageBody + ChunkLength, RequestMessage->Body, MessageBodyLength);
      *(MessageBody + ChunkLength + MessageBodyLength) = CHUNKED_TRANSFER_CODING_CR;
      *(MessageBody + ChunkLength + MessageBodyLength + 1) = CHUNKED_TRANSFER_CODING_LF;
      //
      // Change variables for the next chunk trasnfer.
      //
      RequestMessage->BodyLength -= MessageBodyLength;
      RequestMessage->Body = (VOID *)((CHAR8 *)RequestMessage->Body + MessageBodyLength);
      MessageBodyLength += (ChunkLength + 2);
      if (RequestMessage->BodyLength == 0) {
        *SendChunkProcess = HttpIoSendChunkEndChunk;
      }
      break;
  case HttpIoSendChunkEndChunk:
      HeaderCount = 0;
      NewHeaders = NULL;
      SentRequestData = NULL;
      AsciiSPrint (
        ChunkLengthStr,
        HTTP_IO_CHUNK_SIZE_STRING_LEN,
        "0%c%c%c%c",
        CHUNKED_TRANSFER_CODING_CR,
        CHUNKED_TRANSFER_CODING_LF,
        CHUNKED_TRANSFER_CODING_CR,
        CHUNKED_TRANSFER_CODING_LF
        );
      MessageBody = AllocatePool (AsciiStrLen(ChunkLengthStr));
      if (MessageBody == NULL) {
        DEBUG((DEBUG_ERROR, "Not enough memory for the end chunk transfer\n"));
        return EFI_OUT_OF_RESOURCES;
      }
      CopyMem (MessageBody, ChunkLengthStr, AsciiStrLen (ChunkLengthStr));
      MessageBodyLength = AsciiStrLen (ChunkLengthStr);
      *SendChunkProcess = HttpIoSendChunkFinish;
      break;
  default:
      return EFI_INVALID_PARAMETER;
  }
  Status = HttpIoSendRequest (
             HttpIo,
             SentRequestData,
             HeaderCount,
             NewHeaders,
             MessageBodyLength,
             MessageBody
             );
  if (ContentLengthHeader != NULL) {
    if (ContentLengthHeader->FieldName != NULL) {
      FreePool (ContentLengthHeader->FieldName);
    }
    if (ContentLengthHeader->FieldValue != NULL) {
      FreePool (ContentLengthHeader->FieldValue);
    }
  }
  if (NewHeaders != NULL) {
    FreePool (NewHeaders);
  }
  if (MessageBody != NULL) {
    FreePool (MessageBody);
  }
  return Status;
}
/**
  Synchronously receive a HTTP RESPONSE message from the server.
  @param[in]   HttpIo           The HttpIo wrapping the HTTP service.
  @param[in]   HeaderCount      Number of headers in Headers.
  @param[in]   Headers          Array containing list of HTTP headers.
  @param[out]  ChunkListHead    A pointer to receive list head
                                of chunked data. Caller has to
                                release memory of ChunkListHead
                                and all list entries.
  @param[out]  ContentLength    Total content length
  @retval EFI_SUCCESS            The HTTP chunked transfer is received.
  @retval EFI_NOT_FOUND          No chunked transfer coding header found.
  @retval EFI_OUT_OF_RESOURCES   Failed to allocate memory.
  @retval EFI_INVALID_PARAMETER  Improper parameters.
  @retval Others                 Other errors as indicated.
**/
EFI_STATUS
HttpIoGetChunkedTransferContent (
  IN     HTTP_IO              *HttpIo,
  IN     UINTN                HeaderCount,
  IN     EFI_HTTP_HEADER      *Headers,
  OUT    LIST_ENTRY           **ChunkListHead,
  OUT    UINTN                *ContentLength
  )
{
  EFI_HTTP_HEADER       *Header;
  CHAR8                 ChunkSizeAscii [256];
  EFI_STATUS            Status;
  UINTN                 Index;
  HTTP_IO_RESPONSE_DATA ResponseData;
  UINTN                 TotalLength;
  UINTN                 MaxTotalLength;
  LIST_ENTRY            *HttpChunks;
  HTTP_IO_CHUNKS        *ThisChunk;
  LIST_ENTRY            *ThisListEntry;
  if (ChunkListHead == NULL || ContentLength == NULL) {
    return EFI_INVALID_PARAMETER;
  }
  *ContentLength = 0;
  Header = HttpFindHeader (HeaderCount, Headers, HTTP_HEADER_TRANSFER_ENCODING);
  if (Header == NULL) {
    return EFI_NOT_FOUND;
  }
  if (AsciiStrCmp (Header->FieldValue, HTTP_HEADER_TRANSFER_ENCODING_CHUNKED) != 0) {
    return EFI_NOT_FOUND;
  }
  //
  // Loop to get all chunks.
  //
  TotalLength = 0;
  MaxTotalLength = PcdGet32 (PcdMaxHttpChunkTransfer);
  HttpChunks = (LIST_ENTRY *)AllocateZeroPool (sizeof (LIST_ENTRY));
  if (HttpChunks == NULL) {
    Status = EFI_OUT_OF_RESOURCES;
    goto ExitDeleteChunks;
  }
  InitializeListHead (HttpChunks);
  DEBUG ((DEBUG_INFO, "     Chunked transfer\n"));
  while (TRUE) {
    ZeroMem((VOID *)&ResponseData, sizeof(HTTP_IO_RESPONSE_DATA));
    ResponseData.BodyLength = HTTP_IO_CHUNKED_TRANSFER_CODING_DATA_LENGTH;
    ResponseData.Body = ChunkSizeAscii;
    Status = HttpIoRecvResponse (
               HttpIo,
               FALSE,
               &ResponseData
               );
    if (EFI_ERROR (Status)) {
      goto ExitDeleteChunks;
    }
    //
    // Decoding Chunked Transfer Coding.
    // Only decode chunk-size and last chunk.
    //
    DEBUG ((DEBUG_INFO, "     Chunk HTTP Response StatusCode - %d\n", ResponseData.Response.StatusCode));
    //
    // Break if this is last chunk.
    //
    if (ChunkSizeAscii [0] == CHUNKED_TRANSFER_CODING_LAST_CHUNK) {
      //
      // Check if this is a valid Last-Chunk.
      //
      if ((ChunkSizeAscii [1] != CHUNKED_TRANSFER_CODING_CR) ||
          (ChunkSizeAscii [2] != CHUNKED_TRANSFER_CODING_LF)
          ) {
        DEBUG ((DEBUG_ERROR, "     This is an invalid Last-chunk\n"));
        Status = EFI_INVALID_PARAMETER;
        goto ExitDeleteChunks;
      }
      Status = EFI_SUCCESS;
      DEBUG ((DEBUG_INFO, "     Last-chunk\n"));
      ThisChunk = (HTTP_IO_CHUNKS *)AllocateZeroPool (sizeof (HTTP_IO_CHUNKS));
      if (ThisChunk == NULL) {
        Status = EFI_OUT_OF_RESOURCES;
        goto ExitDeleteChunks;
      }
      InitializeListHead (&ThisChunk->NextChunk);
      ThisChunk->Length = ResponseData.BodyLength - 1 - 2; // Minus sizeof '0' and CRLF.
      ThisChunk->Data = (CHAR8 *)AllocatePool (ThisChunk->Length);
      if (ThisChunk->Data == NULL) {
        FreePool ((UINT8 *)ThisChunk);
        Status = EFI_OUT_OF_RESOURCES;
        goto ExitDeleteChunks;
      }
      CopyMem ((UINT8 *)ThisChunk->Data, (UINT8 *)ResponseData.Body + 1, ThisChunk->Length);
      TotalLength += ThisChunk->Length;
      InsertTailList (HttpChunks, &ThisChunk->NextChunk);
      break;
    }
    //
    // Get the chunk length
    //
    Index = 0;
    while ((ChunkSizeAscii [Index] != CHUNKED_TRANSFER_CODING_EXTENSION_SEPARATOR) &&
           (ChunkSizeAscii [Index] != (CHAR8)CHUNKED_TRANSFER_CODING_CR) &&
           (Index != HTTP_IO_CHUNKED_TRANSFER_CODING_DATA_LENGTH)) {
      Index ++;
    }
    if (Index == HTTP_IO_CHUNKED_TRANSFER_CODING_DATA_LENGTH) {
      Status = EFI_NOT_FOUND;
      goto ExitDeleteChunks;
    }
    ChunkSizeAscii[Index] = 0;
    AsciiStrHexToUintnS (ChunkSizeAscii, NULL, ContentLength);
    DEBUG ((DEBUG_INFO, "     Length of this chunk %d\n", *ContentLength));
    //
    // Receive the data;
    //
    ThisChunk = (HTTP_IO_CHUNKS *)AllocateZeroPool (sizeof (HTTP_IO_CHUNKS));
    if (ThisChunk == NULL) {
      Status = EFI_OUT_OF_RESOURCES;
      goto ExitDeleteChunks;
    }
    ResponseData.BodyLength = *ContentLength;
    ResponseData.Body = (CHAR8 *)AllocatePool (*ContentLength);
    if (ResponseData.Body == NULL) {
      FreePool (ThisChunk);
      Status = EFI_OUT_OF_RESOURCES;
      goto ExitDeleteChunks;
    }
    InitializeListHead (&ThisChunk->NextChunk);
    ThisChunk->Length = *ContentLength;
    ThisChunk->Data = ResponseData.Body;
    InsertTailList (HttpChunks, &ThisChunk->NextChunk);
    Status = HttpIoRecvResponse (
               HttpIo,
               FALSE,
               &ResponseData
               );
    if (EFI_ERROR (Status)) {
      goto ExitDeleteChunks;
    }
    //
    // Read CRLF
    //
    ZeroMem((VOID *)&ResponseData, sizeof(HTTP_IO_RESPONSE_DATA));
    ResponseData.BodyLength = 2;
    ResponseData.Body = ChunkSizeAscii;
    Status = HttpIoRecvResponse (
               HttpIo,
               FALSE,
               &ResponseData
               );
    if (EFI_ERROR (Status)) {
      goto ExitDeleteChunks;
    }
    //
    // Verify the end of chunk payload.
    //
    if ((ChunkSizeAscii [0] != CHUNKED_TRANSFER_CODING_CR) ||
        (ChunkSizeAscii [1] != CHUNKED_TRANSFER_CODING_LF)
       ) {
       DEBUG ((DEBUG_ERROR, "     This is an invalid End-of-chunk notation.\n"));
       goto ExitDeleteChunks;
    }
    TotalLength += *ContentLength;
    if (TotalLength > MaxTotalLength) {
       DEBUG ((DEBUG_ERROR, "     Total chunk transfer payload exceeds the size defined by PcdMaxHttpChunkTransfer.\n"));
       goto ExitDeleteChunks;
    }
  }
  *ContentLength = TotalLength;
  *ChunkListHead = HttpChunks;
  DEBUG ((DEBUG_INFO, "     Total of lengh of chunks :%d\n", TotalLength));
  return EFI_SUCCESS;
ExitDeleteChunks:
  if (HttpChunks != NULL) {
    while (!IsListEmpty(HttpChunks)) {
      ThisListEntry = GetFirstNode (HttpChunks);
      RemoveEntryList (ThisListEntry);
      ThisChunk = (HTTP_IO_CHUNKS *)ThisListEntry;
      if (ThisChunk->Data != NULL) {
        FreePool (ThisChunk->Data);
      }
      FreePool(ThisListEntry);
    }
    FreePool (HttpChunks);
  }
  return Status;
}