/** @file
  This library provides help functions for REST EX Protocol.
  (C) Copyright 2020 Hewlett Packard Enterprise Development LP
  SPDX-License-Identifier: BSD-2-Clause-Patent
**/
#include 
#include 
#include 
#include 
#include 
#include 
#include 
#define REST_EX_CONFIG_DATA_LEN_UNKNOWN  0xff
/**
  This function allows the caller to create child handle for specific
  REST server.
  @param[in]  Image                The image handle used to open service.
  @param[in]  AccessMode           Access mode of REST server.
  @param[in]  ConfigType           Underlying configuration to communicate with REST server.
  @param[in]  ServiceType          REST service type.
  @param[out] ChildInstanceHandle  The handle to receive the create child.
  @retval  EFI_SUCCESS            Can't create the corresponding REST EX child instance.
  @retval  EFI_INVALID_PARAMETERS Any of input parameters is improper.
**/
EFI_STATUS
RestExLibCreateChild (
  IN EFI_HANDLE                       Image,
  IN EFI_REST_EX_SERVICE_ACCESS_MODE  AccessMode,
  IN EFI_REST_EX_CONFIG_TYPE          ConfigType,
  IN EFI_REST_EX_SERVICE_TYPE         ServiceType,
  OUT EFI_HANDLE                      *ChildInstanceHandle
  )
{
  EFI_STATUS                Status;
  UINTN                     NoBuffer;
  EFI_HANDLE                *Handle;
  EFI_HANDLE                ChildHandle;
  EFI_REST_EX_PROTOCOL      *RestEx;
  EFI_REST_EX_SERVICE_INFO  *RestExServiceInfo;
  UINT8                     LenOfConfig;
  if ((Image == NULL) ||
      (AccessMode >= EfiRestExServiceModeMax) ||
      (ConfigType >= EfiRestExConfigTypeMax) ||
      (ServiceType >= EfiRestExServiceTypeMax) ||
      (ChildInstanceHandle == NULL)
      )
  {
    return EFI_INVALID_PARAMETER;
  }
  *ChildInstanceHandle = NULL;
  //
  // Locate all REST EX binding service.
  //
  Handle   = NULL;
  NoBuffer = 0;
  Status   = gBS->LocateHandleBuffer (
                    ByProtocol,
                    &gEfiRestExServiceBindingProtocolGuid,
                    NULL,
                    &NoBuffer,
                    &Handle
                    );
  if (EFI_ERROR (Status) && (Status != EFI_BUFFER_TOO_SMALL)) {
    return Status;
  }
  Handle = (EFI_HANDLE *)AllocateZeroPool (sizeof (EFI_HANDLE) * NoBuffer);
  if (Handle == NULL) {
    return EFI_OUT_OF_RESOURCES;
  }
  Status = gBS->LocateHandleBuffer (
                  ByProtocol,
                  &gEfiRestExServiceBindingProtocolGuid,
                  NULL,
                  &NoBuffer,
                  &Handle
                  );
  if (EFI_ERROR (Status)) {
    FreePool (Handle);
    return Status;
  }
  //
  // Search for the proper REST EX instance.
  //
  while (NoBuffer != 0) {
    ChildHandle = NULL;
    Status      = NetLibCreateServiceChild (
                    *(Handle + (NoBuffer - 1)),
                    Image,
                    &gEfiRestExServiceBindingProtocolGuid,
                    &ChildHandle
                    );
    if (!EFI_ERROR (Status)) {
      Status = gBS->OpenProtocol (
                      ChildHandle,
                      &gEfiRestExProtocolGuid,
                      (VOID **)&RestEx,
                      Image,
                      NULL,
                      EFI_OPEN_PROTOCOL_GET_PROTOCOL
                      );
      if (EFI_ERROR (Status)) {
        goto ON_ERROR;
      }
      //
      // Get the information of REST service provided by this EFI REST EX driver
      //
      Status = RestEx->GetService (
                         RestEx,
                         &RestExServiceInfo
                         );
      if (EFI_ERROR (Status)) {
        goto ON_ERROR;
      }
      //
      // Check REST EX property.
      //
      switch (ConfigType) {
        case EfiRestExConfigHttp:
          LenOfConfig = sizeof (EFI_REST_EX_HTTP_CONFIG_DATA);
          break;
        case EfiRestExConfigUnspecific:
          LenOfConfig = REST_EX_CONFIG_DATA_LEN_UNKNOWN;
          break;
        default:
          goto ON_ERROR;
      }
      if ((RestExServiceInfo->EfiRestExServiceInfoV10.RestServiceAccessMode != AccessMode) ||
          (RestExServiceInfo->EfiRestExServiceInfoV10.RestServiceType != ServiceType) ||
          (RestExServiceInfo->EfiRestExServiceInfoV10.RestExConfigType != ConfigType) ||
          ((LenOfConfig != REST_EX_CONFIG_DATA_LEN_UNKNOWN) && (RestExServiceInfo->EfiRestExServiceInfoV10.RestExConfigDataLength != LenOfConfig)))
      {
        goto ON_ERROR;
      }
    }
    //
    // This is proper REST EX instance.
    //
    *ChildInstanceHandle = ChildHandle;
    FreePool (Handle);
    return EFI_SUCCESS;
ON_ERROR:;
    NetLibDestroyServiceChild (
      *(Handle + (NoBuffer - 1)),
      Image,
      &gEfiRestExServiceBindingProtocolGuid,
      ChildHandle
      );
    NoBuffer--;
  }
  FreePool (Handle);
  return EFI_NOT_FOUND;
}