mirror of https://github.com/acidanthera/audk.git
1086 lines
33 KiB
C
1086 lines
33 KiB
C
/** @file
|
|
Implementation of the boot file download function.
|
|
|
|
Copyright (c) 2015 - 2016, Intel Corporation. All rights reserved.<BR>
|
|
(C) Copyright 2016 Hewlett Packard Enterprise Development LP<BR>
|
|
This program and the accompanying materials are licensed and made available under
|
|
the terms and conditions of the BSD License that accompanies this distribution.
|
|
The full text of the license may be found at
|
|
http://opensource.org/licenses/bsd-license.php.
|
|
|
|
THE PROGRAM IS DISTRIBUTED UNDER THE BSD LICENSE ON AN "AS IS" BASIS,
|
|
WITHOUT WARRANTIES OR REPRESENTATIONS OF ANY KIND, EITHER EXPRESS OR IMPLIED.
|
|
|
|
**/
|
|
|
|
#include "HttpBootDxe.h"
|
|
|
|
/**
|
|
Update the IP and URL device path node to include the boot resource information.
|
|
|
|
@param[in] Private The pointer to the driver's private data.
|
|
|
|
@retval EFI_SUCCESS Device patch successfully updated.
|
|
@retval EFI_OUT_OF_RESOURCES Could not allocate needed resources.
|
|
@retval Others Unexpected error happened.
|
|
|
|
**/
|
|
EFI_STATUS
|
|
HttpBootUpdateDevicePath (
|
|
IN HTTP_BOOT_PRIVATE_DATA *Private
|
|
)
|
|
{
|
|
EFI_DEV_PATH *Node;
|
|
EFI_DEVICE_PATH_PROTOCOL *TmpDevicePath;
|
|
EFI_DEVICE_PATH_PROTOCOL *NewDevicePath;
|
|
UINTN Length;
|
|
EFI_STATUS Status;
|
|
|
|
TmpDevicePath = NULL;
|
|
|
|
//
|
|
// Update the IP node with DHCP assigned information.
|
|
//
|
|
if (!Private->UsingIpv6) {
|
|
Node = AllocateZeroPool (sizeof (IPv4_DEVICE_PATH));
|
|
if (Node == NULL) {
|
|
return EFI_OUT_OF_RESOURCES;
|
|
}
|
|
Node->Ipv4.Header.Type = MESSAGING_DEVICE_PATH;
|
|
Node->Ipv4.Header.SubType = MSG_IPv4_DP;
|
|
SetDevicePathNodeLength (Node, sizeof (IPv4_DEVICE_PATH));
|
|
CopyMem (&Node->Ipv4.LocalIpAddress, &Private->StationIp, sizeof (EFI_IPv4_ADDRESS));
|
|
Node->Ipv4.RemotePort = Private->Port;
|
|
Node->Ipv4.Protocol = EFI_IP_PROTO_TCP;
|
|
Node->Ipv4.StaticIpAddress = FALSE;
|
|
CopyMem (&Node->Ipv4.GatewayIpAddress, &Private->GatewayIp, sizeof (EFI_IPv4_ADDRESS));
|
|
CopyMem (&Node->Ipv4.SubnetMask, &Private->SubnetMask, sizeof (EFI_IPv4_ADDRESS));
|
|
} else {
|
|
Node = AllocateZeroPool (sizeof (IPv6_DEVICE_PATH));
|
|
if (Node == NULL) {
|
|
return EFI_OUT_OF_RESOURCES;
|
|
}
|
|
Node->Ipv6.Header.Type = MESSAGING_DEVICE_PATH;
|
|
Node->Ipv6.Header.SubType = MSG_IPv6_DP;
|
|
SetDevicePathNodeLength (Node, sizeof (IPv6_DEVICE_PATH));
|
|
Node->Ipv6.PrefixLength = IP6_PREFIX_LENGTH;
|
|
Node->Ipv6.RemotePort = Private->Port;
|
|
Node->Ipv6.Protocol = EFI_IP_PROTO_TCP;
|
|
Node->Ipv6.IpAddressOrigin = 0;
|
|
CopyMem (&Node->Ipv6.LocalIpAddress, &Private->StationIp.v6, sizeof (EFI_IPv6_ADDRESS));
|
|
CopyMem (&Node->Ipv6.RemoteIpAddress, &Private->ServerIp.v6, sizeof (EFI_IPv6_ADDRESS));
|
|
CopyMem (&Node->Ipv6.GatewayIpAddress, &Private->GatewayIp.v6, sizeof (EFI_IPv6_ADDRESS));
|
|
}
|
|
|
|
TmpDevicePath = AppendDevicePathNode (Private->ParentDevicePath, (EFI_DEVICE_PATH_PROTOCOL*) Node);
|
|
FreePool (Node);
|
|
if (TmpDevicePath == NULL) {
|
|
return EFI_OUT_OF_RESOURCES;
|
|
}
|
|
|
|
//
|
|
// Update the URI node with the boot file URI.
|
|
//
|
|
Length = sizeof (EFI_DEVICE_PATH_PROTOCOL) + AsciiStrSize (Private->BootFileUri);
|
|
Node = AllocatePool (Length);
|
|
if (Node == NULL) {
|
|
FreePool (TmpDevicePath);
|
|
return EFI_OUT_OF_RESOURCES;
|
|
}
|
|
Node->DevPath.Type = MESSAGING_DEVICE_PATH;
|
|
Node->DevPath.SubType = MSG_URI_DP;
|
|
SetDevicePathNodeLength (Node, Length);
|
|
CopyMem ((UINT8*) Node + sizeof (EFI_DEVICE_PATH_PROTOCOL), Private->BootFileUri, AsciiStrSize (Private->BootFileUri));
|
|
|
|
NewDevicePath = AppendDevicePathNode (TmpDevicePath, (EFI_DEVICE_PATH_PROTOCOL*) Node);
|
|
FreePool (Node);
|
|
FreePool (TmpDevicePath);
|
|
if (NewDevicePath == NULL) {
|
|
return EFI_OUT_OF_RESOURCES;
|
|
}
|
|
|
|
if (!Private->UsingIpv6) {
|
|
//
|
|
// Reinstall the device path protocol of the child handle.
|
|
//
|
|
Status = gBS->ReinstallProtocolInterface (
|
|
Private->Ip4Nic->Controller,
|
|
&gEfiDevicePathProtocolGuid,
|
|
Private->Ip4Nic->DevicePath,
|
|
NewDevicePath
|
|
);
|
|
if (EFI_ERROR (Status)) {
|
|
return Status;
|
|
}
|
|
|
|
FreePool (Private->Ip4Nic->DevicePath);
|
|
Private->Ip4Nic->DevicePath = NewDevicePath;
|
|
} else {
|
|
//
|
|
// Reinstall the device path protocol of the child handle.
|
|
//
|
|
Status = gBS->ReinstallProtocolInterface (
|
|
Private->Ip6Nic->Controller,
|
|
&gEfiDevicePathProtocolGuid,
|
|
Private->Ip6Nic->DevicePath,
|
|
NewDevicePath
|
|
);
|
|
if (EFI_ERROR (Status)) {
|
|
return Status;
|
|
}
|
|
FreePool (Private->Ip6Nic->DevicePath);
|
|
Private->Ip6Nic->DevicePath = NewDevicePath;
|
|
}
|
|
|
|
return EFI_SUCCESS;
|
|
}
|
|
|
|
/**
|
|
Parse the boot file URI information from the selected Dhcp4 offer packet.
|
|
|
|
@param[in] Private The pointer to the driver's private data.
|
|
|
|
@retval EFI_SUCCESS Successfully parsed out all the boot information.
|
|
@retval Others Failed to parse out the boot information.
|
|
|
|
**/
|
|
EFI_STATUS
|
|
HttpBootDhcp4ExtractUriInfo (
|
|
IN HTTP_BOOT_PRIVATE_DATA *Private
|
|
)
|
|
{
|
|
HTTP_BOOT_DHCP4_PACKET_CACHE *SelectOffer;
|
|
HTTP_BOOT_DHCP4_PACKET_CACHE *HttpOffer;
|
|
UINT32 SelectIndex;
|
|
UINT32 ProxyIndex;
|
|
EFI_DHCP4_PACKET_OPTION *Option;
|
|
EFI_STATUS Status;
|
|
|
|
ASSERT (Private != NULL);
|
|
ASSERT (Private->SelectIndex != 0);
|
|
SelectIndex = Private->SelectIndex - 1;
|
|
ASSERT (SelectIndex < HTTP_BOOT_OFFER_MAX_NUM);
|
|
|
|
Status = EFI_SUCCESS;
|
|
|
|
//
|
|
// SelectOffer contains the IP address configuration and name server configuration.
|
|
// HttpOffer contains the boot file URL.
|
|
//
|
|
SelectOffer = &Private->OfferBuffer[SelectIndex].Dhcp4;
|
|
if ((SelectOffer->OfferType == HttpOfferTypeDhcpIpUri) || (SelectOffer->OfferType == HttpOfferTypeDhcpNameUriDns)) {
|
|
HttpOffer = SelectOffer;
|
|
} else {
|
|
ASSERT (Private->SelectProxyType != HttpOfferTypeMax);
|
|
ProxyIndex = Private->OfferIndex[Private->SelectProxyType][0];
|
|
HttpOffer = &Private->OfferBuffer[ProxyIndex].Dhcp4;
|
|
}
|
|
|
|
//
|
|
// Configure the default DNS server if server assigned.
|
|
//
|
|
if ((SelectOffer->OfferType == HttpOfferTypeDhcpNameUriDns) || (SelectOffer->OfferType == HttpOfferTypeDhcpDns)) {
|
|
Option = SelectOffer->OptList[HTTP_BOOT_DHCP4_TAG_INDEX_DNS_SERVER];
|
|
ASSERT (Option != NULL);
|
|
Status = HttpBootRegisterIp4Dns (
|
|
Private,
|
|
Option->Length,
|
|
Option->Data
|
|
);
|
|
if (EFI_ERROR (Status)) {
|
|
return Status;
|
|
}
|
|
}
|
|
|
|
//
|
|
// Extract the port from URL, and use default HTTP port 80 if not provided.
|
|
//
|
|
Status = HttpUrlGetPort (
|
|
(CHAR8*) HttpOffer->OptList[HTTP_BOOT_DHCP4_TAG_INDEX_BOOTFILE]->Data,
|
|
HttpOffer->UriParser,
|
|
&Private->Port
|
|
);
|
|
if (EFI_ERROR (Status) || Private->Port == 0) {
|
|
Private->Port = 80;
|
|
}
|
|
|
|
//
|
|
// Record the URI of boot file from the selected HTTP offer.
|
|
//
|
|
Private->BootFileUriParser = HttpOffer->UriParser;
|
|
Private->BootFileUri = (CHAR8*) HttpOffer->OptList[HTTP_BOOT_DHCP4_TAG_INDEX_BOOTFILE]->Data;
|
|
|
|
|
|
//
|
|
// All boot informations are valid here.
|
|
//
|
|
AsciiPrint ("\n URI: %a", Private->BootFileUri);
|
|
|
|
//
|
|
// Update the device path to include the IP and boot URI information.
|
|
//
|
|
Status = HttpBootUpdateDevicePath (Private);
|
|
|
|
return Status;
|
|
}
|
|
|
|
/**
|
|
Parse the boot file URI information from the selected Dhcp6 offer packet.
|
|
|
|
@param[in] Private The pointer to the driver's private data.
|
|
|
|
@retval EFI_SUCCESS Successfully parsed out all the boot information.
|
|
@retval Others Failed to parse out the boot information.
|
|
|
|
**/
|
|
EFI_STATUS
|
|
HttpBootDhcp6ExtractUriInfo (
|
|
IN HTTP_BOOT_PRIVATE_DATA *Private
|
|
)
|
|
{
|
|
HTTP_BOOT_DHCP6_PACKET_CACHE *SelectOffer;
|
|
HTTP_BOOT_DHCP6_PACKET_CACHE *HttpOffer;
|
|
UINT32 SelectIndex;
|
|
UINT32 ProxyIndex;
|
|
EFI_DHCP6_PACKET_OPTION *Option;
|
|
EFI_IPv6_ADDRESS IpAddr;
|
|
CHAR8 *HostName;
|
|
CHAR16 *HostNameStr;
|
|
EFI_STATUS Status;
|
|
|
|
ASSERT (Private != NULL);
|
|
ASSERT (Private->SelectIndex != 0);
|
|
SelectIndex = Private->SelectIndex - 1;
|
|
ASSERT (SelectIndex < HTTP_BOOT_OFFER_MAX_NUM);
|
|
|
|
Status = EFI_SUCCESS;
|
|
HostName = NULL;
|
|
//
|
|
// SelectOffer contains the IP address configuration and name server configuration.
|
|
// HttpOffer contains the boot file URL.
|
|
//
|
|
SelectOffer = &Private->OfferBuffer[SelectIndex].Dhcp6;
|
|
if ((SelectOffer->OfferType == HttpOfferTypeDhcpIpUri) || (SelectOffer->OfferType == HttpOfferTypeDhcpNameUriDns)) {
|
|
HttpOffer = SelectOffer;
|
|
} else {
|
|
ASSERT (Private->SelectProxyType != HttpOfferTypeMax);
|
|
ProxyIndex = Private->OfferIndex[Private->SelectProxyType][0];
|
|
HttpOffer = &Private->OfferBuffer[ProxyIndex].Dhcp6;
|
|
}
|
|
|
|
//
|
|
// Set the Local station address to IP layer.
|
|
//
|
|
Status = HttpBootSetIp6Address (Private);
|
|
if (EFI_ERROR (Status)) {
|
|
return Status;
|
|
}
|
|
|
|
//
|
|
// Configure the default DNS server if server assigned.
|
|
//
|
|
if ((SelectOffer->OfferType == HttpOfferTypeDhcpNameUriDns) || (SelectOffer->OfferType == HttpOfferTypeDhcpDns)) {
|
|
Option = SelectOffer->OptList[HTTP_BOOT_DHCP6_IDX_DNS_SERVER];
|
|
ASSERT (Option != NULL);
|
|
Status = HttpBootSetIp6Dns (
|
|
Private,
|
|
HTONS (Option->OpLen),
|
|
Option->Data
|
|
);
|
|
if (EFI_ERROR (Status)) {
|
|
return Status;
|
|
}
|
|
}
|
|
|
|
//
|
|
// Extract the HTTP server Ip frome URL. This is used to Check route table
|
|
// whether can send message to HTTP Server Ip through the GateWay.
|
|
//
|
|
Status = HttpUrlGetIp6 (
|
|
(CHAR8*) HttpOffer->OptList[HTTP_BOOT_DHCP6_IDX_BOOT_FILE_URL]->Data,
|
|
HttpOffer->UriParser,
|
|
&IpAddr
|
|
);
|
|
|
|
if (EFI_ERROR (Status)) {
|
|
//
|
|
// The Http server address is expressed by Name Ip, so perform DNS resolution
|
|
//
|
|
Status = HttpUrlGetHostName (
|
|
(CHAR8*) HttpOffer->OptList[HTTP_BOOT_DHCP6_IDX_BOOT_FILE_URL]->Data,
|
|
HttpOffer->UriParser,
|
|
&HostName
|
|
);
|
|
if (EFI_ERROR (Status)) {
|
|
return Status;
|
|
}
|
|
|
|
HostNameStr = AllocateZeroPool ((AsciiStrLen (HostName) + 1) * sizeof (CHAR16));
|
|
if (HostNameStr == NULL) {
|
|
Status = EFI_OUT_OF_RESOURCES;
|
|
goto Error;
|
|
}
|
|
|
|
AsciiStrToUnicodeStr (HostName, HostNameStr);
|
|
Status = HttpBootDns (Private, HostNameStr, &IpAddr);
|
|
FreePool (HostNameStr);
|
|
if (EFI_ERROR (Status)) {
|
|
goto Error;
|
|
}
|
|
}
|
|
|
|
CopyMem (&Private->ServerIp.v6, &IpAddr, sizeof (EFI_IPv6_ADDRESS));
|
|
|
|
//
|
|
// register the IPv6 gateway address to the network device.
|
|
//
|
|
Status = HttpBootSetIp6Gateway (Private);
|
|
if (EFI_ERROR (Status)) {
|
|
return Status;
|
|
}
|
|
|
|
//
|
|
// Extract the port from URL, and use default HTTP port 80 if not provided.
|
|
//
|
|
Status = HttpUrlGetPort (
|
|
(CHAR8*) HttpOffer->OptList[HTTP_BOOT_DHCP6_IDX_BOOT_FILE_URL]->Data,
|
|
HttpOffer->UriParser,
|
|
&Private->Port
|
|
);
|
|
if (EFI_ERROR (Status) || Private->Port == 0) {
|
|
Private->Port = 80;
|
|
}
|
|
|
|
//
|
|
// Record the URI of boot file from the selected HTTP offer.
|
|
//
|
|
Private->BootFileUriParser = HttpOffer->UriParser;
|
|
Private->BootFileUri = (CHAR8*) HttpOffer->OptList[HTTP_BOOT_DHCP6_IDX_BOOT_FILE_URL]->Data;
|
|
|
|
|
|
//
|
|
// All boot informations are valid here.
|
|
//
|
|
AsciiPrint ("\n URI: %a", Private->BootFileUri);
|
|
//
|
|
// Update the device path to include the IP and boot URI information.
|
|
//
|
|
Status = HttpBootUpdateDevicePath (Private);
|
|
|
|
Error:
|
|
|
|
if (HostName != NULL) {
|
|
FreePool (HostName);
|
|
}
|
|
|
|
return Status;
|
|
}
|
|
|
|
|
|
/**
|
|
Discover all the boot information for boot file.
|
|
|
|
@param[in, out] Private The pointer to the driver's private data.
|
|
|
|
@retval EFI_SUCCESS Successfully obtained all the boot information .
|
|
@retval Others Failed to retrieve the boot information.
|
|
|
|
**/
|
|
EFI_STATUS
|
|
HttpBootDiscoverBootInfo (
|
|
IN OUT HTTP_BOOT_PRIVATE_DATA *Private
|
|
)
|
|
{
|
|
EFI_STATUS Status;
|
|
|
|
//
|
|
// Start D.O.R.A/S.A.R.R exchange to acquire station ip address and
|
|
// other Http boot information.
|
|
//
|
|
Status = HttpBootDhcp (Private);
|
|
if (EFI_ERROR (Status)) {
|
|
return Status;
|
|
}
|
|
|
|
if (!Private->UsingIpv6) {
|
|
Status = HttpBootDhcp4ExtractUriInfo (Private);
|
|
} else {
|
|
Status = HttpBootDhcp6ExtractUriInfo (Private);
|
|
}
|
|
|
|
return Status;
|
|
}
|
|
|
|
/**
|
|
Create a HttpIo instance for the file download.
|
|
|
|
@param[in] Private The pointer to the driver's private data.
|
|
|
|
@retval EFI_SUCCESS Successfully created.
|
|
@retval Others Failed to create HttpIo.
|
|
|
|
**/
|
|
EFI_STATUS
|
|
HttpBootCreateHttpIo (
|
|
IN HTTP_BOOT_PRIVATE_DATA *Private
|
|
)
|
|
{
|
|
HTTP_IO_CONFIG_DATA ConfigData;
|
|
EFI_STATUS Status;
|
|
|
|
ASSERT (Private != NULL);
|
|
|
|
ZeroMem (&ConfigData, sizeof (HTTP_IO_CONFIG_DATA));
|
|
if (!Private->UsingIpv6) {
|
|
ConfigData.Config4.HttpVersion = HttpVersion11;
|
|
ConfigData.Config4.RequestTimeOut = HTTP_BOOT_REQUEST_TIMEOUT;
|
|
IP4_COPY_ADDRESS (&ConfigData.Config4.LocalIp, &Private->StationIp.v4);
|
|
IP4_COPY_ADDRESS (&ConfigData.Config4.SubnetMask, &Private->SubnetMask.v4);
|
|
} else {
|
|
ConfigData.Config6.HttpVersion = HttpVersion11;
|
|
ConfigData.Config6.RequestTimeOut = HTTP_BOOT_REQUEST_TIMEOUT;
|
|
IP6_COPY_ADDRESS (&ConfigData.Config6.LocalIp, &Private->StationIp.v6);
|
|
}
|
|
|
|
Status = HttpIoCreateIo (
|
|
Private->Image,
|
|
Private->Controller,
|
|
Private->UsingIpv6 ? IP_VERSION_6 : IP_VERSION_4,
|
|
&ConfigData,
|
|
&Private->HttpIo
|
|
);
|
|
if (EFI_ERROR (Status)) {
|
|
return Status;
|
|
}
|
|
|
|
Private->HttpCreated = TRUE;
|
|
return EFI_SUCCESS;
|
|
}
|
|
|
|
/**
|
|
Release all the resource of a cache item.
|
|
|
|
@param[in] Cache The pointer to the cache item.
|
|
|
|
**/
|
|
VOID
|
|
HttpBootFreeCache (
|
|
IN HTTP_BOOT_CACHE_CONTENT *Cache
|
|
)
|
|
{
|
|
UINTN Index;
|
|
LIST_ENTRY *Entry;
|
|
LIST_ENTRY *NextEntry;
|
|
HTTP_BOOT_ENTITY_DATA *EntityData;
|
|
|
|
if (Cache != NULL) {
|
|
//
|
|
// Free the request data
|
|
//
|
|
if (Cache->RequestData != NULL) {
|
|
if (Cache->RequestData->Url != NULL) {
|
|
FreePool (Cache->RequestData->Url);
|
|
}
|
|
FreePool (Cache->RequestData);
|
|
}
|
|
|
|
//
|
|
// Free the response header
|
|
//
|
|
if (Cache->ResponseData != NULL) {
|
|
if (Cache->ResponseData->Headers != NULL) {
|
|
for (Index = 0; Index < Cache->ResponseData->HeaderCount; Index++) {
|
|
FreePool (Cache->ResponseData->Headers[Index].FieldName);
|
|
FreePool (Cache->ResponseData->Headers[Index].FieldValue);
|
|
}
|
|
FreePool (Cache->ResponseData->Headers);
|
|
}
|
|
}
|
|
|
|
//
|
|
// Free the response body
|
|
//
|
|
NET_LIST_FOR_EACH_SAFE (Entry, NextEntry, &Cache->EntityDataList) {
|
|
EntityData = NET_LIST_USER_STRUCT (Entry, HTTP_BOOT_ENTITY_DATA, Link);
|
|
if (EntityData->Block != NULL) {
|
|
FreePool (EntityData->Block);
|
|
}
|
|
RemoveEntryList (&EntityData->Link);
|
|
FreePool (EntityData);
|
|
}
|
|
|
|
FreePool (Cache);
|
|
}
|
|
}
|
|
|
|
/**
|
|
Clean up all cached data.
|
|
|
|
@param[in] Private The pointer to the driver's private data.
|
|
|
|
**/
|
|
VOID
|
|
HttpBootFreeCacheList (
|
|
IN HTTP_BOOT_PRIVATE_DATA *Private
|
|
)
|
|
{
|
|
LIST_ENTRY *Entry;
|
|
LIST_ENTRY *NextEntry;
|
|
HTTP_BOOT_CACHE_CONTENT *Cache;
|
|
|
|
NET_LIST_FOR_EACH_SAFE (Entry, NextEntry, &Private->CacheList) {
|
|
Cache = NET_LIST_USER_STRUCT (Entry, HTTP_BOOT_CACHE_CONTENT, Link);
|
|
RemoveEntryList (&Cache->Link);
|
|
HttpBootFreeCache (Cache);
|
|
}
|
|
}
|
|
|
|
/**
|
|
Get the file content from cached data.
|
|
|
|
@param[in] Private The pointer to the driver's private data.
|
|
@param[in] Uri Uri of the file to be retrieved from cache.
|
|
@param[in, out] BufferSize On input the size of Buffer in bytes. On output with a return
|
|
code of EFI_SUCCESS, the amount of data transferred to
|
|
Buffer. On output with a return code of EFI_BUFFER_TOO_SMALL,
|
|
the size of Buffer required to retrieve the requested file.
|
|
@param[out] Buffer The memory buffer to transfer the file to. IF Buffer is NULL,
|
|
then the size of the requested file is returned in
|
|
BufferSize.
|
|
|
|
@retval EFI_SUCCESS Successfully created.
|
|
@retval Others Failed to create HttpIo.
|
|
|
|
**/
|
|
EFI_STATUS
|
|
HttpBootGetFileFromCache (
|
|
IN HTTP_BOOT_PRIVATE_DATA *Private,
|
|
IN CHAR16 *Uri,
|
|
IN OUT UINTN *BufferSize,
|
|
OUT UINT8 *Buffer
|
|
)
|
|
{
|
|
LIST_ENTRY *Entry;
|
|
LIST_ENTRY *Entry2;
|
|
HTTP_BOOT_CACHE_CONTENT *Cache;
|
|
HTTP_BOOT_ENTITY_DATA *EntityData;
|
|
UINTN CopyedSize;
|
|
|
|
if (Uri == NULL || BufferSize == 0 || Buffer == NULL) {
|
|
return EFI_INVALID_PARAMETER;
|
|
}
|
|
|
|
//
|
|
// Search file in the cache list, the cache entry will be released upon a successful
|
|
// match.
|
|
//
|
|
NET_LIST_FOR_EACH (Entry, &Private->CacheList) {
|
|
Cache = NET_LIST_USER_STRUCT (Entry, HTTP_BOOT_CACHE_CONTENT, Link);
|
|
//
|
|
// Compare the URI to see whether we already have a cache for this file.
|
|
//
|
|
if ((Cache->RequestData != NULL) &&
|
|
(Cache->RequestData->Url != NULL) &&
|
|
(StrCmp (Uri, Cache->RequestData->Url) == 0))
|
|
{
|
|
//
|
|
// Hit cache, check buffer size.
|
|
//
|
|
if (*BufferSize < Cache->EntityLength) {
|
|
*BufferSize = Cache->EntityLength;
|
|
return EFI_BUFFER_TOO_SMALL;
|
|
}
|
|
|
|
//
|
|
// Fill data to buffer.
|
|
//
|
|
CopyedSize = 0;
|
|
NET_LIST_FOR_EACH (Entry2, &Cache->EntityDataList) {
|
|
EntityData = NET_LIST_USER_STRUCT (Entry2, HTTP_BOOT_ENTITY_DATA, Link);
|
|
if (*BufferSize > CopyedSize) {
|
|
CopyMem (
|
|
Buffer + CopyedSize,
|
|
EntityData->DataStart,
|
|
MIN (EntityData->DataLength, *BufferSize - CopyedSize)
|
|
);
|
|
CopyedSize += MIN (EntityData->DataLength, *BufferSize - CopyedSize);
|
|
}
|
|
}
|
|
*BufferSize = CopyedSize;
|
|
|
|
//
|
|
// On success, free the cached data to release the memory resource.
|
|
//
|
|
RemoveEntryList (&Cache->Link);
|
|
HttpBootFreeCache (Cache);
|
|
return EFI_SUCCESS;
|
|
}
|
|
}
|
|
|
|
return EFI_NOT_FOUND;
|
|
}
|
|
|
|
/**
|
|
A callback function to intercept events during message parser.
|
|
|
|
This function will be invoked during HttpParseMessageBody() with various events type. An error
|
|
return status of the callback function will cause the HttpParseMessageBody() aborted.
|
|
|
|
@param[in] EventType Event type of this callback call.
|
|
@param[in] Data A pointer to data buffer.
|
|
@param[in] Length Length in bytes of the Data.
|
|
@param[in] Context Callback context set by HttpInitMsgParser().
|
|
|
|
@retval EFI_SUCCESS Continue to parser the message body.
|
|
@retval Others Abort the parse.
|
|
|
|
**/
|
|
EFI_STATUS
|
|
EFIAPI
|
|
HttpBootGetBootFileCallback (
|
|
IN HTTP_BODY_PARSE_EVENT EventType,
|
|
IN CHAR8 *Data,
|
|
IN UINTN Length,
|
|
IN VOID *Context
|
|
)
|
|
{
|
|
HTTP_BOOT_CALLBACK_DATA *CallbackData;
|
|
HTTP_BOOT_ENTITY_DATA *NewEntityData;
|
|
|
|
//
|
|
// We only care about the entity data.
|
|
//
|
|
if (EventType != BodyParseEventOnData) {
|
|
return EFI_SUCCESS;
|
|
}
|
|
|
|
CallbackData = (HTTP_BOOT_CALLBACK_DATA *) Context;
|
|
//
|
|
// Copy data if caller has provided a buffer.
|
|
//
|
|
if (CallbackData->BufferSize > CallbackData->CopyedSize) {
|
|
CopyMem (
|
|
CallbackData->Buffer + CallbackData->CopyedSize,
|
|
Data,
|
|
MIN (Length, CallbackData->BufferSize - CallbackData->CopyedSize)
|
|
);
|
|
CallbackData->CopyedSize += MIN (Length, CallbackData->BufferSize - CallbackData->CopyedSize);
|
|
}
|
|
|
|
//
|
|
// The caller doesn't provide a buffer, save the block into cache list.
|
|
//
|
|
if (CallbackData->Cache != NULL) {
|
|
NewEntityData = AllocatePool (sizeof (HTTP_BOOT_ENTITY_DATA));
|
|
if (NewEntityData == NULL) {
|
|
return EFI_OUT_OF_RESOURCES;
|
|
}
|
|
if (CallbackData->NewBlock) {
|
|
NewEntityData->Block = CallbackData->Block;
|
|
CallbackData->Block = NULL;
|
|
}
|
|
NewEntityData->DataLength = Length;
|
|
NewEntityData->DataStart = (UINT8*) Data;
|
|
InsertTailList (&CallbackData->Cache->EntityDataList, &NewEntityData->Link);
|
|
}
|
|
return EFI_SUCCESS;
|
|
}
|
|
|
|
/**
|
|
This function download the boot file by using UEFI HTTP protocol.
|
|
|
|
@param[in] Private The pointer to the driver's private data.
|
|
@param[in] HeaderOnly Only request the response header, it could save a lot of time if
|
|
the caller only want to know the size of the requested file.
|
|
@param[in, out] BufferSize On input the size of Buffer in bytes. On output with a return
|
|
code of EFI_SUCCESS, the amount of data transferred to
|
|
Buffer. On output with a return code of EFI_BUFFER_TOO_SMALL,
|
|
the size of Buffer required to retrieve the requested file.
|
|
@param[out] Buffer The memory buffer to transfer the file to. IF Buffer is NULL,
|
|
then the size of the requested file is returned in
|
|
BufferSize.
|
|
|
|
@retval EFI_SUCCESS The file was loaded.
|
|
@retval EFI_INVALID_PARAMETER BufferSize is NULL or Buffer Size is not NULL but Buffer is NULL.
|
|
@retval EFI_OUT_OF_RESOURCES Could not allocate needed resources
|
|
@retval EFI_BUFFER_TOO_SMALL The BufferSize is too small to read the current directory entry.
|
|
BufferSize has been updated with the size needed to complete
|
|
the request.
|
|
@retval Others Unexpected error happened.
|
|
|
|
**/
|
|
EFI_STATUS
|
|
HttpBootGetBootFile (
|
|
IN HTTP_BOOT_PRIVATE_DATA *Private,
|
|
IN BOOLEAN HeaderOnly,
|
|
IN OUT UINTN *BufferSize,
|
|
OUT UINT8 *Buffer
|
|
)
|
|
{
|
|
EFI_STATUS Status;
|
|
EFI_HTTP_STATUS_CODE StatusCode;
|
|
CHAR8 *HostName;
|
|
EFI_HTTP_REQUEST_DATA *RequestData;
|
|
HTTP_IO_RESPONSE_DATA *ResponseData;
|
|
HTTP_IO_RESPONSE_DATA ResponseBody;
|
|
HTTP_IO *HttpIo;
|
|
HTTP_IO_HEADER *HttpIoHeader;
|
|
VOID *Parser;
|
|
HTTP_BOOT_CALLBACK_DATA Context;
|
|
UINTN ContentLength;
|
|
HTTP_BOOT_CACHE_CONTENT *Cache;
|
|
UINT8 *Block;
|
|
CHAR16 *Url;
|
|
BOOLEAN IdentityMode;
|
|
UINTN ReceivedSize;
|
|
|
|
ASSERT (Private != NULL);
|
|
ASSERT (Private->HttpCreated);
|
|
|
|
if (BufferSize == NULL) {
|
|
return EFI_INVALID_PARAMETER;
|
|
}
|
|
|
|
if (*BufferSize != 0 && Buffer == NULL) {
|
|
return EFI_INVALID_PARAMETER;
|
|
}
|
|
|
|
//
|
|
// First, check whether we already cached the requested Uri.
|
|
//
|
|
Url = AllocatePool ((AsciiStrLen (Private->BootFileUri) + 1) * sizeof (CHAR16));
|
|
if (Url == NULL) {
|
|
return EFI_OUT_OF_RESOURCES;
|
|
}
|
|
AsciiStrToUnicodeStr (Private->BootFileUri, Url);
|
|
if (!HeaderOnly) {
|
|
Status = HttpBootGetFileFromCache (Private, Url, BufferSize, Buffer);
|
|
if (Status != EFI_NOT_FOUND) {
|
|
FreePool (Url);
|
|
return Status;
|
|
}
|
|
}
|
|
|
|
//
|
|
// Not found in cache, try to download it through HTTP.
|
|
//
|
|
|
|
//
|
|
// 1. Create a temp cache item for the requested URI if caller doesn't provide buffer.
|
|
//
|
|
Cache = NULL;
|
|
if ((!HeaderOnly) && (*BufferSize == 0)) {
|
|
Cache = AllocateZeroPool (sizeof (HTTP_BOOT_CACHE_CONTENT));
|
|
if (Cache == NULL) {
|
|
Status = EFI_OUT_OF_RESOURCES;
|
|
goto ERROR_1;
|
|
}
|
|
InitializeListHead (&Cache->EntityDataList);
|
|
}
|
|
|
|
//
|
|
// 2. Send HTTP request message.
|
|
//
|
|
|
|
//
|
|
// 2.1 Build HTTP header for the request, 3 header is needed to download a boot file:
|
|
// Host
|
|
// Accept
|
|
// User-Agent
|
|
//
|
|
HttpIoHeader = HttpBootCreateHeader (3);
|
|
if (HttpIoHeader == NULL) {
|
|
Status = EFI_OUT_OF_RESOURCES;
|
|
goto ERROR_2;
|
|
}
|
|
|
|
//
|
|
// Add HTTP header field 1: Host
|
|
//
|
|
HostName = NULL;
|
|
Status = HttpUrlGetHostName (
|
|
Private->BootFileUri,
|
|
Private->BootFileUriParser,
|
|
&HostName
|
|
);
|
|
if (EFI_ERROR (Status)) {
|
|
goto ERROR_3;
|
|
}
|
|
Status = HttpBootSetHeader (
|
|
HttpIoHeader,
|
|
HTTP_HEADER_HOST,
|
|
HostName
|
|
);
|
|
FreePool (HostName);
|
|
if (EFI_ERROR (Status)) {
|
|
goto ERROR_3;
|
|
}
|
|
|
|
//
|
|
// Add HTTP header field 2: Accept
|
|
//
|
|
Status = HttpBootSetHeader (
|
|
HttpIoHeader,
|
|
HTTP_HEADER_ACCEPT,
|
|
"*/*"
|
|
);
|
|
if (EFI_ERROR (Status)) {
|
|
goto ERROR_3;
|
|
}
|
|
|
|
//
|
|
// Add HTTP header field 3: User-Agent
|
|
//
|
|
Status = HttpBootSetHeader (
|
|
HttpIoHeader,
|
|
HTTP_HEADER_USER_AGENT,
|
|
HTTP_USER_AGENT_EFI_HTTP_BOOT
|
|
);
|
|
if (EFI_ERROR (Status)) {
|
|
goto ERROR_3;
|
|
}
|
|
|
|
//
|
|
// 2.2 Build the rest of HTTP request info.
|
|
//
|
|
RequestData = AllocatePool (sizeof (EFI_HTTP_REQUEST_DATA));
|
|
if (RequestData == NULL) {
|
|
Status = EFI_OUT_OF_RESOURCES;
|
|
goto ERROR_3;
|
|
}
|
|
RequestData->Method = HeaderOnly ? HttpMethodHead : HttpMethodGet;
|
|
RequestData->Url = Url;
|
|
if (RequestData->Url == NULL) {
|
|
Status = EFI_OUT_OF_RESOURCES;
|
|
goto ERROR_4;
|
|
}
|
|
AsciiStrToUnicodeStr (Private->BootFileUri, RequestData->Url);
|
|
|
|
//
|
|
// 2.3 Record the request info in a temp cache item.
|
|
//
|
|
if (Cache != NULL) {
|
|
Cache->RequestData = RequestData;
|
|
}
|
|
|
|
//
|
|
// 2.4 Send out the request to HTTP server.
|
|
//
|
|
HttpIo = &Private->HttpIo;
|
|
Status = HttpIoSendRequest (
|
|
HttpIo,
|
|
RequestData,
|
|
HttpIoHeader->HeaderCount,
|
|
HttpIoHeader->Headers,
|
|
0,
|
|
NULL
|
|
);
|
|
if (EFI_ERROR (Status)) {
|
|
goto ERROR_4;
|
|
}
|
|
|
|
//
|
|
// 3. Receive HTTP response message.
|
|
//
|
|
|
|
//
|
|
// 3.1 First step, use zero BodyLength to only receive the response headers.
|
|
//
|
|
ResponseData = AllocateZeroPool (sizeof(HTTP_IO_RESPONSE_DATA));
|
|
if (ResponseData == NULL) {
|
|
Status = EFI_OUT_OF_RESOURCES;
|
|
goto ERROR_4;
|
|
}
|
|
Status = HttpIoRecvResponse (
|
|
&Private->HttpIo,
|
|
TRUE,
|
|
ResponseData
|
|
);
|
|
if (EFI_ERROR (Status) || EFI_ERROR (ResponseData->Status)) {
|
|
if (EFI_ERROR (ResponseData->Status)) {
|
|
StatusCode = HttpIo->RspToken.Message->Data.Response->StatusCode;
|
|
HttpBootPrintErrorMessage (StatusCode);
|
|
Status = ResponseData->Status;
|
|
}
|
|
goto ERROR_5;
|
|
}
|
|
|
|
//
|
|
// 3.2 Cache the response header.
|
|
//
|
|
if (Cache != NULL) {
|
|
Cache->ResponseData = ResponseData;
|
|
}
|
|
|
|
//
|
|
// 3.3 Init a message-body parser from the header information.
|
|
//
|
|
Parser = NULL;
|
|
Context.NewBlock = FALSE;
|
|
Context.Block = NULL;
|
|
Context.CopyedSize = 0;
|
|
Context.Buffer = Buffer;
|
|
Context.BufferSize = *BufferSize;
|
|
Context.Cache = Cache;
|
|
Status = HttpInitMsgParser (
|
|
HeaderOnly? HttpMethodHead : HttpMethodGet,
|
|
ResponseData->Response.StatusCode,
|
|
ResponseData->HeaderCount,
|
|
ResponseData->Headers,
|
|
HttpBootGetBootFileCallback,
|
|
(VOID*) &Context,
|
|
&Parser
|
|
);
|
|
if (EFI_ERROR (Status)) {
|
|
goto ERROR_6;
|
|
}
|
|
|
|
//
|
|
// 3.4 Continue to receive and parse message-body if needed.
|
|
//
|
|
Block = NULL;
|
|
if (!HeaderOnly) {
|
|
//
|
|
// 3.4.1, check whether we are in identity transfer-coding.
|
|
//
|
|
ContentLength = 0;
|
|
Status = HttpGetEntityLength (Parser, &ContentLength);
|
|
if (!EFI_ERROR (Status)) {
|
|
IdentityMode = TRUE;
|
|
} else {
|
|
IdentityMode = FALSE;
|
|
}
|
|
|
|
//
|
|
// 3.4.2, start the message-body download, the identity and chunked transfer-coding
|
|
// is handled in different path here.
|
|
//
|
|
ZeroMem (&ResponseBody, sizeof (HTTP_IO_RESPONSE_DATA));
|
|
if (IdentityMode) {
|
|
//
|
|
// In identity transfer-coding there is no need to parse the message body,
|
|
// just download the message body to the user provided buffer directly.
|
|
//
|
|
ReceivedSize = 0;
|
|
while (ReceivedSize < ContentLength) {
|
|
ResponseBody.Body = (CHAR8*) Buffer + ReceivedSize;
|
|
ResponseBody.BodyLength = *BufferSize - ReceivedSize;
|
|
Status = HttpIoRecvResponse (
|
|
&Private->HttpIo,
|
|
FALSE,
|
|
&ResponseBody
|
|
);
|
|
if (EFI_ERROR (Status)) {
|
|
goto ERROR_6;
|
|
}
|
|
ReceivedSize += ResponseBody.BodyLength;
|
|
}
|
|
} else {
|
|
//
|
|
// In "chunked" transfer-coding mode, so we need to parse the received
|
|
// data to get the real entity content.
|
|
//
|
|
Block = NULL;
|
|
while (!HttpIsMessageComplete (Parser)) {
|
|
//
|
|
// Allocate a buffer in Block to hold the message-body.
|
|
// If caller provides a buffer, this Block will be reused in every HttpIoRecvResponse().
|
|
// Otherwise a buffer, the buffer in Block will be cached and we should allocate a new before
|
|
// every HttpIoRecvResponse().
|
|
//
|
|
if (Block == NULL || Context.BufferSize == 0) {
|
|
Block = AllocatePool (HTTP_BOOT_BLOCK_SIZE);
|
|
if (Block == NULL) {
|
|
Status = EFI_OUT_OF_RESOURCES;
|
|
goto ERROR_6;
|
|
}
|
|
Context.NewBlock = TRUE;
|
|
Context.Block = Block;
|
|
} else {
|
|
Context.NewBlock = FALSE;
|
|
}
|
|
|
|
ResponseBody.Body = (CHAR8*) Block;
|
|
ResponseBody.BodyLength = HTTP_BOOT_BLOCK_SIZE;
|
|
Status = HttpIoRecvResponse (
|
|
&Private->HttpIo,
|
|
FALSE,
|
|
&ResponseBody
|
|
);
|
|
if (EFI_ERROR (Status)) {
|
|
goto ERROR_6;
|
|
}
|
|
|
|
//
|
|
// Parse the new received block of the message-body, the block will be saved in cache.
|
|
//
|
|
Status = HttpParseMessageBody (
|
|
Parser,
|
|
ResponseBody.BodyLength,
|
|
ResponseBody.Body
|
|
);
|
|
if (EFI_ERROR (Status)) {
|
|
goto ERROR_6;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
//
|
|
// 3.5 Message-body receive & parse is completed, we should be able to get the file size now.
|
|
//
|
|
Status = HttpGetEntityLength (Parser, &ContentLength);
|
|
if (EFI_ERROR (Status)) {
|
|
goto ERROR_6;
|
|
}
|
|
|
|
if (*BufferSize < ContentLength) {
|
|
Status = EFI_BUFFER_TOO_SMALL;
|
|
}
|
|
*BufferSize = ContentLength;
|
|
|
|
//
|
|
// 4. Save the cache item to driver's cache list and return.
|
|
//
|
|
if (Cache != NULL) {
|
|
Cache->EntityLength = ContentLength;
|
|
InsertTailList (&Private->CacheList, &Cache->Link);
|
|
}
|
|
|
|
if (Parser != NULL) {
|
|
HttpFreeMsgParser (Parser);
|
|
}
|
|
|
|
return EFI_SUCCESS;
|
|
|
|
ERROR_6:
|
|
if (Parser != NULL) {
|
|
HttpFreeMsgParser (Parser);
|
|
}
|
|
if (Context.Block != NULL) {
|
|
FreePool (Context.Block);
|
|
}
|
|
HttpBootFreeCache (Cache);
|
|
|
|
ERROR_5:
|
|
if (ResponseData != NULL) {
|
|
FreePool (ResponseData);
|
|
}
|
|
ERROR_4:
|
|
if (RequestData != NULL) {
|
|
FreePool (RequestData);
|
|
}
|
|
ERROR_3:
|
|
HttpBootFreeHeader (HttpIoHeader);
|
|
ERROR_2:
|
|
if (Cache != NULL) {
|
|
FreePool (Cache);
|
|
}
|
|
ERROR_1:
|
|
if (Url != NULL) {
|
|
FreePool (Url);
|
|
}
|
|
|
|
return Status;
|
|
}
|