NetworkPkg:Enable Http Boot over Ipv6 stack
Add new features to support Http boot over ipv6 stack. Contributed-under: TianoCore Contribution Agreement 1.0 Signed-off-by: Zhang Lubo <lubo.zhang@intel.com> Reviewed-by: Fu Siyuan <siyuan.fu@intel.com> Reviewed-by: Ye Ting <ting.ye@intel.com> Reviewed-by: Wu Jiaxin <jiaxin.wu@intel.com> git-svn-id: https://svn.code.sf.net/p/edk2/code/trunk/edk2@18743 6f19259b-4bc3-4df7-8a09-765794883524
This commit is contained in:
@ -54,14 +54,27 @@ HttpBootUpdateDevicePath (
|
|||||||
Node->Ipv4.StaticIpAddress = FALSE;
|
Node->Ipv4.StaticIpAddress = FALSE;
|
||||||
CopyMem (&Node->Ipv4.GatewayIpAddress, &Private->GatewayIp, sizeof (EFI_IPv4_ADDRESS));
|
CopyMem (&Node->Ipv4.GatewayIpAddress, &Private->GatewayIp, sizeof (EFI_IPv4_ADDRESS));
|
||||||
CopyMem (&Node->Ipv4.SubnetMask, &Private->SubnetMask, sizeof (EFI_IPv4_ADDRESS));
|
CopyMem (&Node->Ipv4.SubnetMask, &Private->SubnetMask, sizeof (EFI_IPv4_ADDRESS));
|
||||||
|
} else {
|
||||||
TmpDevicePath = AppendDevicePathNode (Private->ParentDevicePath, (EFI_DEVICE_PATH_PROTOCOL*) Node);
|
Node = AllocateZeroPool (sizeof (IPv6_DEVICE_PATH));
|
||||||
FreePool (Node);
|
if (Node == NULL) {
|
||||||
if (TmpDevicePath == NULL) {
|
|
||||||
return EFI_OUT_OF_RESOURCES;
|
return EFI_OUT_OF_RESOURCES;
|
||||||
}
|
}
|
||||||
} else {
|
Node->Ipv6.Header.Type = MESSAGING_DEVICE_PATH;
|
||||||
ASSERT (FALSE);
|
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;
|
||||||
}
|
}
|
||||||
|
|
||||||
//
|
//
|
||||||
@ -85,21 +98,39 @@ HttpBootUpdateDevicePath (
|
|||||||
return EFI_OUT_OF_RESOURCES;
|
return EFI_OUT_OF_RESOURCES;
|
||||||
}
|
}
|
||||||
|
|
||||||
//
|
if (!Private->UsingIpv6) {
|
||||||
// Reinstall the device path protocol of the child handle.
|
//
|
||||||
//
|
// Reinstall the device path protocol of the child handle.
|
||||||
Status = gBS->ReinstallProtocolInterface (
|
//
|
||||||
Private->ChildHandle,
|
Status = gBS->ReinstallProtocolInterface (
|
||||||
&gEfiDevicePathProtocolGuid,
|
Private->Ip4Nic->Controller,
|
||||||
Private->DevicePath,
|
&gEfiDevicePathProtocolGuid,
|
||||||
NewDevicePath
|
Private->Ip4Nic->DevicePath,
|
||||||
);
|
NewDevicePath
|
||||||
if (EFI_ERROR (Status)) {
|
);
|
||||||
return Status;
|
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;
|
||||||
}
|
}
|
||||||
|
|
||||||
FreePool (Private->DevicePath);
|
|
||||||
Private->DevicePath = NewDevicePath;
|
|
||||||
return EFI_SUCCESS;
|
return EFI_SUCCESS;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -113,7 +144,7 @@ HttpBootUpdateDevicePath (
|
|||||||
|
|
||||||
**/
|
**/
|
||||||
EFI_STATUS
|
EFI_STATUS
|
||||||
HttpBootExtractUriInfo (
|
HttpBootDhcp4ExtractUriInfo (
|
||||||
IN HTTP_BOOT_PRIVATE_DATA *Private
|
IN HTTP_BOOT_PRIVATE_DATA *Private
|
||||||
)
|
)
|
||||||
{
|
{
|
||||||
@ -192,6 +223,159 @@ HttpBootExtractUriInfo (
|
|||||||
return Status;
|
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.
|
Discover all the boot information for boot file.
|
||||||
|
|
||||||
@ -218,9 +402,9 @@ HttpBootDiscoverBootInfo (
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (!Private->UsingIpv6) {
|
if (!Private->UsingIpv6) {
|
||||||
Status = HttpBootExtractUriInfo (Private);
|
Status = HttpBootDhcp4ExtractUriInfo (Private);
|
||||||
} else {
|
} else {
|
||||||
ASSERT (FALSE);
|
Status = HttpBootDhcp6ExtractUriInfo (Private);
|
||||||
}
|
}
|
||||||
|
|
||||||
return Status;
|
return Status;
|
||||||
@ -247,12 +431,14 @@ HttpBootCreateHttpIo (
|
|||||||
|
|
||||||
ZeroMem (&ConfigData, sizeof (HTTP_IO_CONFIG_DATA));
|
ZeroMem (&ConfigData, sizeof (HTTP_IO_CONFIG_DATA));
|
||||||
if (!Private->UsingIpv6) {
|
if (!Private->UsingIpv6) {
|
||||||
ConfigData.Config4.HttpVersion = HttpVersion11;
|
ConfigData.Config4.HttpVersion = HttpVersion11;
|
||||||
ConfigData.Config4.RequestTimeOut = HTTP_BOOT_REQUEST_TIMEOUT;
|
ConfigData.Config4.RequestTimeOut = HTTP_BOOT_REQUEST_TIMEOUT;
|
||||||
IP4_COPY_ADDRESS (&ConfigData.Config4.LocalIp, &Private->StationIp.v4);
|
IP4_COPY_ADDRESS (&ConfigData.Config4.LocalIp, &Private->StationIp.v4);
|
||||||
IP4_COPY_ADDRESS (&ConfigData.Config4.SubnetMask, &Private->SubnetMask.v4);
|
IP4_COPY_ADDRESS (&ConfigData.Config4.SubnetMask, &Private->SubnetMask.v4);
|
||||||
} else {
|
} else {
|
||||||
ASSERT (FALSE);
|
ConfigData.Config6.HttpVersion = HttpVersion11;
|
||||||
|
ConfigData.Config6.RequestTimeOut = HTTP_BOOT_REQUEST_TIMEOUT;
|
||||||
|
IP6_COPY_ADDRESS (&ConfigData.Config6.LocalIp, &Private->StationIp.v6);
|
||||||
}
|
}
|
||||||
|
|
||||||
Status = HttpIoCreateIo (
|
Status = HttpIoCreateIo (
|
||||||
|
@ -151,7 +151,10 @@ HttpBootDxeComponentNameGetControllerName (
|
|||||||
|
|
||||||
NicHandle = HttpBootGetNicByIp4Children (ControllerHandle);
|
NicHandle = HttpBootGetNicByIp4Children (ControllerHandle);
|
||||||
if (NicHandle == NULL) {
|
if (NicHandle == NULL) {
|
||||||
return EFI_UNSUPPORTED;
|
NicHandle = HttpBootGetNicByIp6Children(ControllerHandle);
|
||||||
|
if (NicHandle == NULL) {
|
||||||
|
return EFI_UNSUPPORTED;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
//
|
//
|
||||||
|
@ -319,7 +319,7 @@ HttpBootParseDhcp4Packet (
|
|||||||
}
|
}
|
||||||
|
|
||||||
//
|
//
|
||||||
// The offer with "HttpClient" is a Http offer.
|
// The offer with "HTTPClient" is a Http offer.
|
||||||
//
|
//
|
||||||
Option = Options[HTTP_BOOT_DHCP4_TAG_INDEX_CLASS_ID];
|
Option = Options[HTTP_BOOT_DHCP4_TAG_INDEX_CLASS_ID];
|
||||||
if ((Option != NULL) && (Option->Length >= 9) &&
|
if ((Option != NULL) && (Option->Length >= 9) &&
|
||||||
@ -461,13 +461,13 @@ HttpBootCacheDhcp4Offer (
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
Select an DHCPv4 offer, and record SelectIndex and SelectProxyType.
|
Select an DHCPv4 or DHCP6 offer, and record SelectIndex and SelectProxyType.
|
||||||
|
|
||||||
@param[in] Private Pointer to HTTP boot driver private data.
|
@param[in] Private Pointer to HTTP boot driver private data.
|
||||||
|
|
||||||
**/
|
**/
|
||||||
VOID
|
VOID
|
||||||
HttpBootSelectDhcp4Offer (
|
HttpBootSelectDhcpOffer (
|
||||||
IN HTTP_BOOT_PRIVATE_DATA *Private
|
IN HTTP_BOOT_PRIVATE_DATA *Private
|
||||||
)
|
)
|
||||||
{
|
{
|
||||||
@ -590,7 +590,7 @@ HttpBootDhcp4CallBack (
|
|||||||
// Select offer according to the priority in UEFI spec, and record the SelectIndex
|
// Select offer according to the priority in UEFI spec, and record the SelectIndex
|
||||||
// and SelectProxyType.
|
// and SelectProxyType.
|
||||||
//
|
//
|
||||||
HttpBootSelectDhcp4Offer (Private);
|
HttpBootSelectDhcpOffer (Private);
|
||||||
|
|
||||||
if (Private->SelectIndex == 0) {
|
if (Private->SelectIndex == 0) {
|
||||||
Status = EFI_ABORTED;
|
Status = EFI_ABORTED;
|
||||||
@ -689,7 +689,7 @@ HttpBootRegisterIp4Dns (
|
|||||||
|
|
||||||
**/
|
**/
|
||||||
EFI_STATUS
|
EFI_STATUS
|
||||||
HttpBootSetIpPolicy (
|
HttpBootSetIp4Policy (
|
||||||
IN HTTP_BOOT_PRIVATE_DATA *Private
|
IN HTTP_BOOT_PRIVATE_DATA *Private
|
||||||
)
|
)
|
||||||
{
|
{
|
||||||
@ -752,7 +752,7 @@ HttpBootDhcp4Dora (
|
|||||||
Dhcp4 = Private->Dhcp4;
|
Dhcp4 = Private->Dhcp4;
|
||||||
ASSERT (Dhcp4 != NULL);
|
ASSERT (Dhcp4 != NULL);
|
||||||
|
|
||||||
Status = HttpBootSetIpPolicy (Private);
|
Status = HttpBootSetIp4Policy (Private);
|
||||||
if (EFI_ERROR (Status)) {
|
if (EFI_ERROR (Status)) {
|
||||||
return Status;
|
return Status;
|
||||||
}
|
}
|
||||||
|
@ -245,6 +245,17 @@ typedef struct {
|
|||||||
EFI_DHCP4_PACKET_OPTION *OptList[HTTP_BOOT_DHCP4_TAG_INDEX_MAX];
|
EFI_DHCP4_PACKET_OPTION *OptList[HTTP_BOOT_DHCP4_TAG_INDEX_MAX];
|
||||||
} HTTP_BOOT_DHCP4_PACKET_CACHE;
|
} HTTP_BOOT_DHCP4_PACKET_CACHE;
|
||||||
|
|
||||||
|
/**
|
||||||
|
Select an DHCPv4 or DHCP6 offer, and record SelectIndex and SelectProxyType.
|
||||||
|
|
||||||
|
@param[in] Private Pointer to HTTP boot driver private data.
|
||||||
|
|
||||||
|
**/
|
||||||
|
VOID
|
||||||
|
HttpBootSelectDhcpOffer (
|
||||||
|
IN HTTP_BOOT_PRIVATE_DATA *Private
|
||||||
|
);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
Start the D.O.R.A DHCPv4 process to acquire the IPv4 address and other Http boot information.
|
Start the D.O.R.A DHCPv4 process to acquire the IPv4 address and other Http boot information.
|
||||||
|
|
||||||
|
984
NetworkPkg/HttpBootDxe/HttpBootDhcp6.c
Normal file
984
NetworkPkg/HttpBootDxe/HttpBootDhcp6.c
Normal file
@ -0,0 +1,984 @@
|
|||||||
|
/** @file
|
||||||
|
Functions implementation related with DHCPv6 for HTTP boot driver.
|
||||||
|
|
||||||
|
Copyright (c) 2015, Intel Corporation. All rights reserved.<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"
|
||||||
|
|
||||||
|
/**
|
||||||
|
Build the options buffer for the DHCPv6 request packet.
|
||||||
|
|
||||||
|
@param[in] Private The pointer to HTTP BOOT driver private data.
|
||||||
|
@param[out] OptList The pointer to the option pointer array.
|
||||||
|
@param[in] Buffer The pointer to the buffer to contain the option list.
|
||||||
|
|
||||||
|
@return Index The count of the built-in options.
|
||||||
|
|
||||||
|
**/
|
||||||
|
UINT32
|
||||||
|
HttpBootBuildDhcp6Options (
|
||||||
|
IN HTTP_BOOT_PRIVATE_DATA *Private,
|
||||||
|
OUT EFI_DHCP6_PACKET_OPTION **OptList,
|
||||||
|
IN UINT8 *Buffer
|
||||||
|
)
|
||||||
|
{
|
||||||
|
HTTP_BOOT_DHCP6_OPTION_ENTRY OptEnt;
|
||||||
|
UINT16 Value;
|
||||||
|
UINT32 Index;
|
||||||
|
|
||||||
|
Index = 0;
|
||||||
|
OptList[0] = (EFI_DHCP6_PACKET_OPTION *) Buffer;
|
||||||
|
|
||||||
|
//
|
||||||
|
// Append client option request option
|
||||||
|
//
|
||||||
|
OptList[Index]->OpCode = HTONS (HTTP_BOOT_DHCP6_OPT_ORO);
|
||||||
|
OptList[Index]->OpLen = HTONS (8);
|
||||||
|
OptEnt.Oro = (HTTP_BOOT_DHCP6_OPTION_ORO *) OptList[Index]->Data;
|
||||||
|
OptEnt.Oro->OpCode[0] = HTONS(HTTP_BOOT_DHCP6_OPT_BOOT_FILE_URL);
|
||||||
|
OptEnt.Oro->OpCode[1] = HTONS(HTTP_BOOT_DHCP6_OPT_BOOT_FILE_PARAM);
|
||||||
|
OptEnt.Oro->OpCode[2] = HTONS(HTTP_BOOT_DHCP6_OPT_DNS_SERVERS);
|
||||||
|
OptEnt.Oro->OpCode[3] = HTONS(HTTP_BOOT_DHCP6_OPT_VENDOR_CLASS);
|
||||||
|
Index++;
|
||||||
|
OptList[Index] = GET_NEXT_DHCP6_OPTION (OptList[Index - 1]);
|
||||||
|
|
||||||
|
//
|
||||||
|
// Append client network device interface option
|
||||||
|
//
|
||||||
|
OptList[Index]->OpCode = HTONS (HTTP_BOOT_DHCP6_OPT_UNDI);
|
||||||
|
OptList[Index]->OpLen = HTONS ((UINT16)3);
|
||||||
|
OptEnt.Undi = (HTTP_BOOT_DHCP6_OPTION_UNDI *) OptList[Index]->Data;
|
||||||
|
|
||||||
|
if (Private->Nii != NULL) {
|
||||||
|
OptEnt.Undi->Type = Private->Nii->Type;
|
||||||
|
OptEnt.Undi->MajorVer = Private->Nii->MajorVer;
|
||||||
|
OptEnt.Undi->MinorVer = Private->Nii->MinorVer;
|
||||||
|
} else {
|
||||||
|
OptEnt.Undi->Type = DEFAULT_UNDI_TYPE;
|
||||||
|
OptEnt.Undi->MajorVer = DEFAULT_UNDI_MAJOR;
|
||||||
|
OptEnt.Undi->MinorVer = DEFAULT_UNDI_MINOR;
|
||||||
|
}
|
||||||
|
|
||||||
|
Index++;
|
||||||
|
OptList[Index] = GET_NEXT_DHCP6_OPTION (OptList[Index - 1]);
|
||||||
|
|
||||||
|
//
|
||||||
|
// Append client system architecture option
|
||||||
|
//
|
||||||
|
OptList[Index]->OpCode = HTONS (HTTP_BOOT_DHCP6_OPT_ARCH);
|
||||||
|
OptList[Index]->OpLen = HTONS ((UINT16) sizeof (HTTP_BOOT_DHCP6_OPTION_ARCH));
|
||||||
|
OptEnt.Arch = (HTTP_BOOT_DHCP6_OPTION_ARCH *) OptList[Index]->Data;
|
||||||
|
Value = HTONS (EFI_HTTP_BOOT_CLIENT_SYSTEM_ARCHITECTURE);
|
||||||
|
CopyMem (&OptEnt.Arch->Type, &Value, sizeof (UINT16));
|
||||||
|
Index++;
|
||||||
|
OptList[Index] = GET_NEXT_DHCP6_OPTION (OptList[Index - 1]);
|
||||||
|
|
||||||
|
//
|
||||||
|
// Append vendor class identify option.
|
||||||
|
//
|
||||||
|
OptList[Index]->OpCode = HTONS (HTTP_BOOT_DHCP6_OPT_VENDOR_CLASS);
|
||||||
|
OptList[Index]->OpLen = HTONS ((UINT16) sizeof (HTTP_BOOT_DHCP6_OPTION_VENDOR_CLASS));
|
||||||
|
OptEnt.VendorClass = (HTTP_BOOT_DHCP6_OPTION_VENDOR_CLASS *) OptList[Index]->Data;
|
||||||
|
OptEnt.VendorClass->Vendor = HTONL (HTTP_BOOT_DHCP6_ENTERPRISE_NUM);
|
||||||
|
OptEnt.VendorClass->ClassLen = HTONS ((UINT16) sizeof (HTTP_BOOT_CLASS_ID));
|
||||||
|
CopyMem (
|
||||||
|
&OptEnt.VendorClass->ClassId,
|
||||||
|
DEFAULT_CLASS_ID_DATA,
|
||||||
|
sizeof (HTTP_BOOT_CLASS_ID)
|
||||||
|
);
|
||||||
|
HttpBootUintnToAscDecWithFormat (
|
||||||
|
EFI_HTTP_BOOT_CLIENT_SYSTEM_ARCHITECTURE,
|
||||||
|
OptEnt.VendorClass->ClassId.ArchitectureType,
|
||||||
|
sizeof (OptEnt.VendorClass->ClassId.ArchitectureType)
|
||||||
|
);
|
||||||
|
|
||||||
|
if (Private->Nii != NULL) {
|
||||||
|
CopyMem (
|
||||||
|
OptEnt.VendorClass->ClassId.InterfaceName,
|
||||||
|
Private->Nii->StringId,
|
||||||
|
sizeof (OptEnt.VendorClass->ClassId.InterfaceName)
|
||||||
|
);
|
||||||
|
HttpBootUintnToAscDecWithFormat (
|
||||||
|
Private->Nii->MajorVer,
|
||||||
|
OptEnt.VendorClass->ClassId.UndiMajor,
|
||||||
|
sizeof (OptEnt.VendorClass->ClassId.UndiMajor)
|
||||||
|
);
|
||||||
|
HttpBootUintnToAscDecWithFormat (
|
||||||
|
Private->Nii->MinorVer,
|
||||||
|
OptEnt.VendorClass->ClassId.UndiMinor,
|
||||||
|
sizeof (OptEnt.VendorClass->ClassId.UndiMinor)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
Index++;
|
||||||
|
|
||||||
|
return Index;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
Parse out a DHCPv6 option by OptTag, and find the position in buffer.
|
||||||
|
|
||||||
|
@param[in] Buffer The pointer to the option buffer.
|
||||||
|
@param[in] Length Length of the option buffer.
|
||||||
|
@param[in] OptTag The required option tag.
|
||||||
|
|
||||||
|
@retval NULL Failed to parse the required option.
|
||||||
|
@retval Others The postion of the required option in buffer.
|
||||||
|
|
||||||
|
**/
|
||||||
|
EFI_DHCP6_PACKET_OPTION *
|
||||||
|
HttpBootParseDhcp6Options (
|
||||||
|
IN UINT8 *Buffer,
|
||||||
|
IN UINT32 Length,
|
||||||
|
IN UINT16 OptTag
|
||||||
|
)
|
||||||
|
{
|
||||||
|
EFI_DHCP6_PACKET_OPTION *Option;
|
||||||
|
UINT32 Offset;
|
||||||
|
|
||||||
|
Option = (EFI_DHCP6_PACKET_OPTION *) Buffer;
|
||||||
|
Offset = 0;
|
||||||
|
|
||||||
|
//
|
||||||
|
// OpLen and OpCode here are both stored in network order.
|
||||||
|
//
|
||||||
|
while (Offset < Length) {
|
||||||
|
|
||||||
|
if (NTOHS (Option->OpCode) == OptTag) {
|
||||||
|
|
||||||
|
return Option;
|
||||||
|
}
|
||||||
|
|
||||||
|
Offset += (NTOHS(Option->OpLen) + 4);
|
||||||
|
Option = (EFI_DHCP6_PACKET_OPTION *) (Buffer + Offset);
|
||||||
|
}
|
||||||
|
|
||||||
|
return NULL;
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
Parse the cached DHCPv6 packet, including all the options.
|
||||||
|
|
||||||
|
@param[in] Cache6 The pointer to a cached DHCPv6 packet.
|
||||||
|
|
||||||
|
@retval EFI_SUCCESS Parsed the DHCPv6 packet successfully.
|
||||||
|
@retval EFI_DEVICE_ERROR Failed to parse and invalid the packet.
|
||||||
|
|
||||||
|
**/
|
||||||
|
EFI_STATUS
|
||||||
|
HttpBootParseDhcp6Packet (
|
||||||
|
IN HTTP_BOOT_DHCP6_PACKET_CACHE *Cache6
|
||||||
|
)
|
||||||
|
{
|
||||||
|
EFI_DHCP6_PACKET *Offer;
|
||||||
|
EFI_DHCP6_PACKET_OPTION **Options;
|
||||||
|
EFI_DHCP6_PACKET_OPTION *Option;
|
||||||
|
HTTP_BOOT_OFFER_TYPE OfferType;
|
||||||
|
EFI_IPv6_ADDRESS IpAddr;
|
||||||
|
BOOLEAN IsProxyOffer;
|
||||||
|
BOOLEAN IsHttpOffer;
|
||||||
|
BOOLEAN IsDnsOffer;
|
||||||
|
BOOLEAN IpExpressedUri;
|
||||||
|
EFI_STATUS Status;
|
||||||
|
UINT32 Offset;
|
||||||
|
UINT32 Length;
|
||||||
|
|
||||||
|
IsDnsOffer = FALSE;
|
||||||
|
IpExpressedUri = FALSE;
|
||||||
|
IsProxyOffer = TRUE;
|
||||||
|
IsHttpOffer = FALSE;
|
||||||
|
Offer = &Cache6->Packet.Offer;
|
||||||
|
Options = Cache6->OptList;
|
||||||
|
|
||||||
|
ZeroMem (Cache6->OptList, sizeof (Cache6->OptList));
|
||||||
|
|
||||||
|
Option = (EFI_DHCP6_PACKET_OPTION *) (Offer->Dhcp6.Option);
|
||||||
|
Offset = 0;
|
||||||
|
Length = GET_DHCP6_OPTION_SIZE (Offer);
|
||||||
|
|
||||||
|
//
|
||||||
|
// OpLen and OpCode here are both stored in network order, since they are from original packet.
|
||||||
|
//
|
||||||
|
while (Offset < Length) {
|
||||||
|
|
||||||
|
if (NTOHS (Option->OpCode) == HTTP_BOOT_DHCP6_OPT_IA_NA) {
|
||||||
|
Options[HTTP_BOOT_DHCP6_IDX_IA_NA] = Option;
|
||||||
|
} else if (NTOHS (Option->OpCode) == HTTP_BOOT_DHCP6_OPT_BOOT_FILE_URL) {
|
||||||
|
//
|
||||||
|
// The server sends this option to inform the client about an URL to a boot file.
|
||||||
|
//
|
||||||
|
Options[HTTP_BOOT_DHCP6_IDX_BOOT_FILE_URL] = Option;
|
||||||
|
} else if (NTOHS (Option->OpCode) == HTTP_BOOT_DHCP6_OPT_BOOT_FILE_PARAM) {
|
||||||
|
Options[HTTP_BOOT_DHCP6_IDX_BOOT_FILE_PARAM] = Option;
|
||||||
|
} else if (NTOHS (Option->OpCode) == HTTP_BOOT_DHCP6_OPT_VENDOR_CLASS) {
|
||||||
|
Options[HTTP_BOOT_DHCP6_IDX_VENDOR_CLASS] = Option;
|
||||||
|
} else if (NTOHS (Option->OpCode) == HTTP_BOOT_DHCP6_OPT_DNS_SERVERS) {
|
||||||
|
Options[HTTP_BOOT_DHCP6_IDX_DNS_SERVER] = Option;
|
||||||
|
}
|
||||||
|
|
||||||
|
Offset += (NTOHS (Option->OpLen) + 4);
|
||||||
|
Option = (EFI_DHCP6_PACKET_OPTION *) (Offer->Dhcp6.Option + Offset);
|
||||||
|
}
|
||||||
|
//
|
||||||
|
// The offer with assigned client address is NOT a proxy offer.
|
||||||
|
// An ia_na option, embeded with valid ia_addr option and a status_code of success.
|
||||||
|
//
|
||||||
|
Option = Options[HTTP_BOOT_DHCP6_IDX_IA_NA];
|
||||||
|
if (Option != NULL) {
|
||||||
|
Option = HttpBootParseDhcp6Options (
|
||||||
|
Option->Data + 12,
|
||||||
|
NTOHS (Option->OpLen),
|
||||||
|
HTTP_BOOT_DHCP6_OPT_STATUS_CODE
|
||||||
|
);
|
||||||
|
if ((Option != NULL && Option->Data[0] == 0) || (Option == NULL)) {
|
||||||
|
IsProxyOffer = FALSE;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
//
|
||||||
|
// The offer with "HTTPClient" is a Http offer.
|
||||||
|
//
|
||||||
|
Option = Options[HTTP_BOOT_DHCP6_IDX_VENDOR_CLASS];
|
||||||
|
|
||||||
|
if (Option != NULL &&
|
||||||
|
NTOHS(Option->OpLen) >= 10 &&
|
||||||
|
CompareMem (Option->Data, DEFAULT_CLASS_ID_DATA, 10) == 0) {
|
||||||
|
IsHttpOffer = TRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
//
|
||||||
|
// The offer with Domain Server is a DNS offer.
|
||||||
|
//
|
||||||
|
Option = Options[HTTP_BOOT_DHCP6_IDX_DNS_SERVER];
|
||||||
|
if (Option != NULL) {
|
||||||
|
IsDnsOffer = TRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
//
|
||||||
|
// Http offer must have a boot URI.
|
||||||
|
//
|
||||||
|
if (IsHttpOffer && Options[HTTP_BOOT_DHCP6_IDX_BOOT_FILE_URL] == NULL) {
|
||||||
|
return EFI_DEVICE_ERROR;
|
||||||
|
}
|
||||||
|
|
||||||
|
//
|
||||||
|
// Try to retrieve the IP of HTTP server from URI.
|
||||||
|
//
|
||||||
|
if (IsHttpOffer) {
|
||||||
|
Status = HttpParseUrl (
|
||||||
|
(CHAR8*) Options[HTTP_BOOT_DHCP6_IDX_BOOT_FILE_URL]->Data,
|
||||||
|
(UINT32) AsciiStrLen ((CHAR8*) Options[HTTP_BOOT_DHCP6_IDX_BOOT_FILE_URL]->Data),
|
||||||
|
FALSE,
|
||||||
|
&Cache6->UriParser
|
||||||
|
);
|
||||||
|
if (EFI_ERROR (Status)) {
|
||||||
|
return EFI_DEVICE_ERROR;
|
||||||
|
}
|
||||||
|
|
||||||
|
Status = HttpUrlGetIp6 (
|
||||||
|
(CHAR8*) Options[HTTP_BOOT_DHCP6_IDX_BOOT_FILE_URL]->Data,
|
||||||
|
Cache6->UriParser,
|
||||||
|
&IpAddr
|
||||||
|
);
|
||||||
|
IpExpressedUri = !EFI_ERROR (Status);
|
||||||
|
}
|
||||||
|
|
||||||
|
//
|
||||||
|
// Determine offer type of the DHCPv6 packet.
|
||||||
|
//
|
||||||
|
if (IsHttpOffer) {
|
||||||
|
if (IpExpressedUri) {
|
||||||
|
OfferType = IsProxyOffer ? HttpOfferTypeProxyIpUri : HttpOfferTypeDhcpIpUri;
|
||||||
|
} else {
|
||||||
|
if (!IsProxyOffer) {
|
||||||
|
OfferType = IsDnsOffer ? HttpOfferTypeDhcpNameUriDns : HttpOfferTypeDhcpNameUri;
|
||||||
|
} else {
|
||||||
|
OfferType = HttpOfferTypeProxyNameUri;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
} else {
|
||||||
|
if (!IsProxyOffer) {
|
||||||
|
OfferType = IsDnsOffer ? HttpOfferTypeDhcpDns : HttpOfferTypeDhcpOnly;
|
||||||
|
} else {
|
||||||
|
return EFI_DEVICE_ERROR;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Cache6->OfferType = OfferType;
|
||||||
|
return EFI_SUCCESS;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
Cache the DHCPv6 packet.
|
||||||
|
|
||||||
|
@param[in] Dst The pointer to the cache buffer for DHCPv6 packet.
|
||||||
|
@param[in] Src The pointer to the DHCPv6 packet to be cached.
|
||||||
|
|
||||||
|
**/
|
||||||
|
VOID
|
||||||
|
HttpBootCacheDhcp6Packet (
|
||||||
|
IN EFI_DHCP6_PACKET *Dst,
|
||||||
|
IN EFI_DHCP6_PACKET *Src
|
||||||
|
)
|
||||||
|
{
|
||||||
|
ASSERT (Dst->Size >= Src->Length);
|
||||||
|
|
||||||
|
CopyMem (&Dst->Dhcp6, &Src->Dhcp6, Src->Length);
|
||||||
|
Dst->Length = Src->Length;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
Cache all the received DHCPv6 offers, and set OfferIndex and OfferCount.
|
||||||
|
|
||||||
|
@param[in] Private The pointer to HTTP_BOOT_PRIVATE_DATA.
|
||||||
|
@param[in] RcvdOffer The pointer to the received offer packet.
|
||||||
|
|
||||||
|
**/
|
||||||
|
VOID
|
||||||
|
HttpBootCacheDhcp6Offer (
|
||||||
|
IN HTTP_BOOT_PRIVATE_DATA *Private,
|
||||||
|
IN EFI_DHCP6_PACKET *RcvdOffer
|
||||||
|
)
|
||||||
|
{
|
||||||
|
HTTP_BOOT_DHCP6_PACKET_CACHE *Cache6;
|
||||||
|
EFI_DHCP6_PACKET *Offer;
|
||||||
|
HTTP_BOOT_OFFER_TYPE OfferType;
|
||||||
|
|
||||||
|
Cache6 = &Private->OfferBuffer[Private->OfferNum].Dhcp6;
|
||||||
|
Offer = &Cache6->Packet.Offer;
|
||||||
|
|
||||||
|
//
|
||||||
|
// Cache the content of DHCPv6 packet firstly.
|
||||||
|
//
|
||||||
|
HttpBootCacheDhcp6Packet(Offer, RcvdOffer);
|
||||||
|
|
||||||
|
//
|
||||||
|
// Validate the DHCPv6 packet, and parse the options and offer type.
|
||||||
|
//
|
||||||
|
if (EFI_ERROR (HttpBootParseDhcp6Packet (Cache6))) {
|
||||||
|
return ;
|
||||||
|
}
|
||||||
|
|
||||||
|
//
|
||||||
|
// Determine whether cache the current offer by type, and record OfferIndex and OfferCount.
|
||||||
|
//
|
||||||
|
OfferType = Cache6->OfferType;
|
||||||
|
ASSERT (OfferType < HttpOfferTypeMax);
|
||||||
|
ASSERT (Private->OfferCount[OfferType] < HTTP_BOOT_OFFER_MAX_NUM);
|
||||||
|
Private->OfferIndex[OfferType][Private->OfferCount[OfferType]] = Private->OfferNum;
|
||||||
|
Private->OfferCount[OfferType]++;
|
||||||
|
Private->OfferNum++;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
EFI_DHCP6_CALLBACK is provided by the consumer of the EFI DHCPv6 Protocol driver
|
||||||
|
to intercept events that occurred in the configuration process.
|
||||||
|
|
||||||
|
@param[in] This The pointer to the EFI DHCPv6 Protocol.
|
||||||
|
@param[in] Context The pointer to the context set by EFI_DHCP6_PROTOCOL.Configure().
|
||||||
|
@param[in] CurrentState The current operational state of the EFI DHCPv Protocol driver.
|
||||||
|
@param[in] Dhcp6Event The event that occurs in the current state, which usually means a
|
||||||
|
state transition.
|
||||||
|
@param[in] Packet The DHCPv6 packet that is going to be sent or was already received.
|
||||||
|
@param[out] NewPacket The packet that is used to replace the Packet above.
|
||||||
|
|
||||||
|
@retval EFI_SUCCESS Told the EFI DHCPv6 Protocol driver to continue the DHCP process.
|
||||||
|
@retval EFI_NOT_READY Only used in the Dhcp6Selecting state. The EFI DHCPv6 Protocol
|
||||||
|
driver will continue to wait for more packets.
|
||||||
|
@retval EFI_ABORTED Told the EFI DHCPv6 Protocol driver to abort the current process.
|
||||||
|
|
||||||
|
**/
|
||||||
|
EFI_STATUS
|
||||||
|
EFIAPI
|
||||||
|
HttpBootDhcp6CallBack (
|
||||||
|
IN EFI_DHCP6_PROTOCOL *This,
|
||||||
|
IN VOID *Context,
|
||||||
|
IN EFI_DHCP6_STATE CurrentState,
|
||||||
|
IN EFI_DHCP6_EVENT Dhcp6Event,
|
||||||
|
IN EFI_DHCP6_PACKET *Packet,
|
||||||
|
OUT EFI_DHCP6_PACKET **NewPacket OPTIONAL
|
||||||
|
)
|
||||||
|
{
|
||||||
|
HTTP_BOOT_PRIVATE_DATA *Private;
|
||||||
|
EFI_DHCP6_PACKET *SelectAd;
|
||||||
|
EFI_STATUS Status;
|
||||||
|
if ((Dhcp6Event != Dhcp6RcvdAdvertise) && (Dhcp6Event != Dhcp6SelectAdvertise)) {
|
||||||
|
return EFI_SUCCESS;
|
||||||
|
}
|
||||||
|
|
||||||
|
ASSERT (Packet != NULL);
|
||||||
|
|
||||||
|
Private = (HTTP_BOOT_PRIVATE_DATA *) Context;
|
||||||
|
Status = EFI_SUCCESS;
|
||||||
|
switch (Dhcp6Event) {
|
||||||
|
|
||||||
|
case Dhcp6RcvdAdvertise:
|
||||||
|
Status = EFI_NOT_READY;
|
||||||
|
if (Private->OfferNum < HTTP_BOOT_OFFER_MAX_NUM) {
|
||||||
|
//
|
||||||
|
// Cache the dhcp offers to OfferBuffer[] for select later, and record
|
||||||
|
// the OfferIndex and OfferCount.
|
||||||
|
//
|
||||||
|
HttpBootCacheDhcp6Offer (Private, Packet);
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
|
||||||
|
case Dhcp6SelectAdvertise:
|
||||||
|
//
|
||||||
|
// Select offer by the default policy or by order, and record the SelectIndex
|
||||||
|
// and SelectProxyType.
|
||||||
|
//
|
||||||
|
HttpBootSelectDhcpOffer (Private);
|
||||||
|
|
||||||
|
if (Private->SelectIndex == 0) {
|
||||||
|
Status = EFI_ABORTED;
|
||||||
|
} else {
|
||||||
|
ASSERT (NewPacket != NULL);
|
||||||
|
SelectAd = &Private->OfferBuffer[Private->SelectIndex - 1].Dhcp6.Packet.Offer;
|
||||||
|
*NewPacket = AllocateZeroPool (SelectAd->Size);
|
||||||
|
ASSERT (*NewPacket != NULL);
|
||||||
|
CopyMem (*NewPacket, SelectAd, SelectAd->Size);
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
|
||||||
|
default:
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
return Status;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
Check whether IP driver could route the message which will be sent to ServerIp address.
|
||||||
|
|
||||||
|
This function will check the IP6 route table every 1 seconds until specified timeout is expired, if a valid
|
||||||
|
route is found in IP6 route table, the address will be filed in GatewayAddr and return.
|
||||||
|
|
||||||
|
@param[in] Private The pointer to HTTP_BOOT_PRIVATE_DATA.
|
||||||
|
@param[in] TimeOutInSecond Timeout value in seconds.
|
||||||
|
@param[out] GatewayAddr Pointer to store the gateway IP address.
|
||||||
|
|
||||||
|
@retval EFI_SUCCESS Found a valid gateway address successfully.
|
||||||
|
@retval EFI_TIMEOUT The operation is time out.
|
||||||
|
@retval Other Unexpect error happened.
|
||||||
|
|
||||||
|
**/
|
||||||
|
EFI_STATUS
|
||||||
|
HttpBootCheckRouteTable (
|
||||||
|
IN HTTP_BOOT_PRIVATE_DATA *Private,
|
||||||
|
IN UINTN TimeOutInSecond,
|
||||||
|
OUT EFI_IPv6_ADDRESS *GatewayAddr
|
||||||
|
)
|
||||||
|
{
|
||||||
|
EFI_STATUS Status;
|
||||||
|
EFI_IP6_PROTOCOL *Ip6;
|
||||||
|
EFI_IP6_MODE_DATA Ip6ModeData;
|
||||||
|
UINTN Index;
|
||||||
|
EFI_EVENT TimeOutEvt;
|
||||||
|
UINTN RetryCount;
|
||||||
|
BOOLEAN GatewayIsFound;
|
||||||
|
|
||||||
|
ASSERT (GatewayAddr != NULL);
|
||||||
|
ASSERT (Private != NULL);
|
||||||
|
|
||||||
|
Ip6 = Private->Ip6;
|
||||||
|
GatewayIsFound = FALSE;
|
||||||
|
RetryCount = 0;
|
||||||
|
TimeOutEvt = NULL;
|
||||||
|
Status = EFI_SUCCESS;
|
||||||
|
ZeroMem (GatewayAddr, sizeof (EFI_IPv6_ADDRESS));
|
||||||
|
|
||||||
|
while (TRUE) {
|
||||||
|
Status = Ip6->GetModeData (Ip6, &Ip6ModeData, NULL, NULL);
|
||||||
|
if (EFI_ERROR (Status)) {
|
||||||
|
goto ON_EXIT;
|
||||||
|
}
|
||||||
|
|
||||||
|
//
|
||||||
|
// Find out the gateway address which can route the message which send to ServerIp.
|
||||||
|
//
|
||||||
|
for (Index = 0; Index < Ip6ModeData.RouteCount; Index++) {
|
||||||
|
if (NetIp6IsNetEqual (&Private->ServerIp.v6, &Ip6ModeData.RouteTable[Index].Destination, Ip6ModeData.RouteTable[Index].PrefixLength)) {
|
||||||
|
IP6_COPY_ADDRESS (GatewayAddr, &Ip6ModeData.RouteTable[Index].Gateway);
|
||||||
|
GatewayIsFound = TRUE;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (Ip6ModeData.AddressList != NULL) {
|
||||||
|
FreePool (Ip6ModeData.AddressList);
|
||||||
|
}
|
||||||
|
if (Ip6ModeData.GroupTable != NULL) {
|
||||||
|
FreePool (Ip6ModeData.GroupTable);
|
||||||
|
}
|
||||||
|
if (Ip6ModeData.RouteTable != NULL) {
|
||||||
|
FreePool (Ip6ModeData.RouteTable);
|
||||||
|
}
|
||||||
|
if (Ip6ModeData.NeighborCache != NULL) {
|
||||||
|
FreePool (Ip6ModeData.NeighborCache);
|
||||||
|
}
|
||||||
|
if (Ip6ModeData.PrefixTable != NULL) {
|
||||||
|
FreePool (Ip6ModeData.PrefixTable);
|
||||||
|
}
|
||||||
|
if (Ip6ModeData.IcmpTypeList != NULL) {
|
||||||
|
FreePool (Ip6ModeData.IcmpTypeList);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (GatewayIsFound || RetryCount == TimeOutInSecond) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
RetryCount++;
|
||||||
|
|
||||||
|
//
|
||||||
|
// Delay 1 second then recheck it again.
|
||||||
|
//
|
||||||
|
if (TimeOutEvt == NULL) {
|
||||||
|
Status = gBS->CreateEvent (
|
||||||
|
EVT_TIMER,
|
||||||
|
TPL_CALLBACK,
|
||||||
|
NULL,
|
||||||
|
NULL,
|
||||||
|
&TimeOutEvt
|
||||||
|
);
|
||||||
|
if (EFI_ERROR (Status)) {
|
||||||
|
goto ON_EXIT;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Status = gBS->SetTimer (TimeOutEvt, TimerRelative, TICKS_PER_SECOND);
|
||||||
|
if (EFI_ERROR (Status)) {
|
||||||
|
goto ON_EXIT;
|
||||||
|
}
|
||||||
|
while (EFI_ERROR (gBS->CheckEvent (TimeOutEvt))) {
|
||||||
|
Ip6->Poll (Ip6);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
ON_EXIT:
|
||||||
|
if (TimeOutEvt != NULL) {
|
||||||
|
gBS->CloseEvent (TimeOutEvt);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (GatewayIsFound) {
|
||||||
|
Status = EFI_SUCCESS;
|
||||||
|
} else if (RetryCount == TimeOutInSecond) {
|
||||||
|
Status = EFI_TIMEOUT;
|
||||||
|
}
|
||||||
|
|
||||||
|
return Status;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
Set the IP6 policy to Automatic.
|
||||||
|
|
||||||
|
@param[in] Private The pointer to HTTP_BOOT_PRIVATE_DATA.
|
||||||
|
|
||||||
|
@retval EFI_SUCCESS Switch the IP policy succesfully.
|
||||||
|
@retval Others Unexpect error happened.
|
||||||
|
|
||||||
|
**/
|
||||||
|
EFI_STATUS
|
||||||
|
HttpBootSetIp6Policy (
|
||||||
|
IN HTTP_BOOT_PRIVATE_DATA *Private
|
||||||
|
)
|
||||||
|
{
|
||||||
|
EFI_IP6_CONFIG_POLICY Policy;
|
||||||
|
EFI_IP6_CONFIG_PROTOCOL *Ip6Config;
|
||||||
|
EFI_STATUS Status;
|
||||||
|
UINTN DataSize;
|
||||||
|
|
||||||
|
Ip6Config = Private->Ip6Config;
|
||||||
|
DataSize = sizeof (EFI_IP6_CONFIG_POLICY);
|
||||||
|
|
||||||
|
//
|
||||||
|
// Get and store the current policy of IP6 driver.
|
||||||
|
//
|
||||||
|
Status = Ip6Config->GetData (
|
||||||
|
Ip6Config,
|
||||||
|
Ip6ConfigDataTypePolicy,
|
||||||
|
&DataSize,
|
||||||
|
&Policy
|
||||||
|
);
|
||||||
|
if (EFI_ERROR (Status)) {
|
||||||
|
return Status;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (Policy == Ip6ConfigPolicyManual) {
|
||||||
|
Policy = Ip6ConfigPolicyAutomatic;
|
||||||
|
Status = Ip6Config->SetData (
|
||||||
|
Ip6Config,
|
||||||
|
Ip6ConfigDataTypePolicy,
|
||||||
|
sizeof(EFI_IP6_CONFIG_POLICY),
|
||||||
|
&Policy
|
||||||
|
);
|
||||||
|
if (EFI_ERROR (Status)) {
|
||||||
|
return Status;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return EFI_SUCCESS;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
This function will register the default DNS addresses to the network device.
|
||||||
|
|
||||||
|
@param[in] Private The pointer to HTTP_BOOT_PRIVATE_DATA.
|
||||||
|
@param[in] DataLength Size of the buffer pointed to by DnsServerData in bytes.
|
||||||
|
@param[in] DnsServerData Point a list of DNS server address in an array
|
||||||
|
of EFI_IPv6_ADDRESS instances.
|
||||||
|
|
||||||
|
@retval EFI_SUCCESS The DNS configuration has been configured successfully.
|
||||||
|
@retval Others Failed to configure the address.
|
||||||
|
|
||||||
|
**/
|
||||||
|
EFI_STATUS
|
||||||
|
HttpBootSetIp6Dns (
|
||||||
|
IN HTTP_BOOT_PRIVATE_DATA *Private,
|
||||||
|
IN UINTN DataLength,
|
||||||
|
IN VOID *DnsServerData
|
||||||
|
)
|
||||||
|
{
|
||||||
|
EFI_IP6_CONFIG_PROTOCOL *Ip6Config;
|
||||||
|
|
||||||
|
ASSERT (Private->UsingIpv6);
|
||||||
|
|
||||||
|
Ip6Config = Private->Ip6Config;
|
||||||
|
|
||||||
|
return Ip6Config->SetData (
|
||||||
|
Ip6Config,
|
||||||
|
Ip6ConfigDataTypeDnsServer,
|
||||||
|
DataLength,
|
||||||
|
DnsServerData
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
This function will register the IPv6 gateway address to the network device.
|
||||||
|
|
||||||
|
@param[in] Private The pointer to HTTP_BOOT_PRIVATE_DATA.
|
||||||
|
|
||||||
|
@retval EFI_SUCCESS The new IP configuration has been configured successfully.
|
||||||
|
@retval Others Failed to configure the address.
|
||||||
|
|
||||||
|
**/
|
||||||
|
EFI_STATUS
|
||||||
|
HttpBootSetIp6Gateway (
|
||||||
|
IN HTTP_BOOT_PRIVATE_DATA *Private
|
||||||
|
)
|
||||||
|
{
|
||||||
|
EFI_IP6_CONFIG_PROTOCOL *Ip6Config;
|
||||||
|
EFI_STATUS Status;
|
||||||
|
|
||||||
|
ASSERT (Private->UsingIpv6);
|
||||||
|
Ip6Config = Private->Ip6Config;
|
||||||
|
|
||||||
|
//
|
||||||
|
// Set the default gateway address.
|
||||||
|
//
|
||||||
|
if (!Private->NoGateway && !NetIp6IsUnspecifiedAddr (&Private->GatewayIp.v6)) {
|
||||||
|
Status = Ip6Config->SetData (
|
||||||
|
Ip6Config,
|
||||||
|
Ip6ConfigDataTypeGateway,
|
||||||
|
sizeof (EFI_IPv6_ADDRESS),
|
||||||
|
&Private->GatewayIp.v6
|
||||||
|
);
|
||||||
|
if (EFI_ERROR(Status)) {
|
||||||
|
return Status;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return EFI_SUCCESS;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
This function will register the station IP address.
|
||||||
|
|
||||||
|
@param[in] Private The pointer to HTTP_BOOT_PRIVATE_DATA.
|
||||||
|
|
||||||
|
@retval EFI_SUCCESS The new IP address has been configured successfully.
|
||||||
|
@retval Others Failed to configure the address.
|
||||||
|
|
||||||
|
**/
|
||||||
|
EFI_STATUS
|
||||||
|
HttpBootSetIp6Address (
|
||||||
|
IN HTTP_BOOT_PRIVATE_DATA *Private
|
||||||
|
)
|
||||||
|
{
|
||||||
|
EFI_STATUS Status;
|
||||||
|
EFI_IP6_PROTOCOL *Ip6;
|
||||||
|
EFI_IP6_CONFIG_PROTOCOL *Ip6Cfg;
|
||||||
|
EFI_IP6_CONFIG_POLICY Policy;
|
||||||
|
EFI_IP6_CONFIG_MANUAL_ADDRESS CfgAddr;
|
||||||
|
EFI_IPv6_ADDRESS *Ip6Addr;
|
||||||
|
EFI_IPv6_ADDRESS GatewayAddr;
|
||||||
|
EFI_IP6_CONFIG_DATA Ip6CfgData;
|
||||||
|
EFI_EVENT MappedEvt;
|
||||||
|
UINTN DataSize;
|
||||||
|
BOOLEAN IsAddressOk;
|
||||||
|
UINTN Index;
|
||||||
|
|
||||||
|
ASSERT (Private->UsingIpv6);
|
||||||
|
|
||||||
|
MappedEvt = NULL;
|
||||||
|
IsAddressOk = FALSE;
|
||||||
|
Ip6Addr = NULL;
|
||||||
|
Ip6Cfg = Private->Ip6Config;
|
||||||
|
Ip6 = Private->Ip6;
|
||||||
|
|
||||||
|
ZeroMem (&CfgAddr, sizeof (EFI_IP6_CONFIG_MANUAL_ADDRESS));
|
||||||
|
CopyMem (&CfgAddr, &Private->StationIp.v6, sizeof (EFI_IPv6_ADDRESS));
|
||||||
|
ZeroMem (&Ip6CfgData, sizeof (EFI_IP6_CONFIG_DATA));
|
||||||
|
|
||||||
|
Ip6CfgData.AcceptIcmpErrors = TRUE;
|
||||||
|
Ip6CfgData.DefaultProtocol = IP6_ICMP;
|
||||||
|
Ip6CfgData.HopLimit = HTTP_BOOT_DEFAULT_HOPLIMIT;
|
||||||
|
Ip6CfgData.ReceiveTimeout = HTTP_BOOT_DEFAULT_LIFETIME;
|
||||||
|
Ip6CfgData.TransmitTimeout = HTTP_BOOT_DEFAULT_LIFETIME;
|
||||||
|
|
||||||
|
Status = Ip6->Configure (Ip6, &Ip6CfgData);
|
||||||
|
if (EFI_ERROR (Status)) {
|
||||||
|
goto ON_EXIT;
|
||||||
|
}
|
||||||
|
|
||||||
|
//
|
||||||
|
// Retrieve the gateway address from IP6 route table.
|
||||||
|
//
|
||||||
|
Status = HttpBootCheckRouteTable (Private, HTTP_BOOT_IP6_ROUTE_TABLE_TIMEOUT, &GatewayAddr);
|
||||||
|
if (EFI_ERROR (Status)) {
|
||||||
|
Private->NoGateway = TRUE;
|
||||||
|
} else {
|
||||||
|
IP6_COPY_ADDRESS (&Private->GatewayIp.v6, &GatewayAddr);
|
||||||
|
}
|
||||||
|
|
||||||
|
//
|
||||||
|
// Set the new address by Ip6ConfigProtocol manually.
|
||||||
|
//
|
||||||
|
Policy = Ip6ConfigPolicyManual;
|
||||||
|
Status = Ip6Cfg->SetData (
|
||||||
|
Ip6Cfg,
|
||||||
|
Ip6ConfigDataTypePolicy,
|
||||||
|
sizeof(EFI_IP6_CONFIG_POLICY),
|
||||||
|
&Policy
|
||||||
|
);
|
||||||
|
if (EFI_ERROR (Status)) {
|
||||||
|
goto ON_EXIT;
|
||||||
|
}
|
||||||
|
|
||||||
|
//
|
||||||
|
// Create a notify event to set address flag when DAD if IP6 driver succeeded.
|
||||||
|
//
|
||||||
|
Status = gBS->CreateEvent (
|
||||||
|
EVT_NOTIFY_SIGNAL,
|
||||||
|
TPL_NOTIFY,
|
||||||
|
HttpBootCommonNotify,
|
||||||
|
&IsAddressOk,
|
||||||
|
&MappedEvt
|
||||||
|
);
|
||||||
|
if (EFI_ERROR (Status)) {
|
||||||
|
goto ON_EXIT;
|
||||||
|
}
|
||||||
|
|
||||||
|
//
|
||||||
|
// Set static host ip6 address. This is a asynchronous process.
|
||||||
|
//
|
||||||
|
Status = Ip6Cfg->RegisterDataNotify (
|
||||||
|
Ip6Cfg,
|
||||||
|
Ip6ConfigDataTypeManualAddress,
|
||||||
|
MappedEvt
|
||||||
|
);
|
||||||
|
if (EFI_ERROR(Status)) {
|
||||||
|
goto ON_EXIT;
|
||||||
|
}
|
||||||
|
|
||||||
|
Status = Ip6Cfg->SetData (
|
||||||
|
Ip6Cfg,
|
||||||
|
Ip6ConfigDataTypeManualAddress,
|
||||||
|
sizeof (EFI_IP6_CONFIG_MANUAL_ADDRESS),
|
||||||
|
&CfgAddr
|
||||||
|
);
|
||||||
|
if (EFI_ERROR (Status) && Status != EFI_NOT_READY) {
|
||||||
|
goto ON_EXIT;
|
||||||
|
} else if (Status == EFI_NOT_READY) {
|
||||||
|
//
|
||||||
|
// Poll the network until the asynchronous process is finished.
|
||||||
|
//
|
||||||
|
while (!IsAddressOk) {
|
||||||
|
Ip6->Poll (Ip6);
|
||||||
|
}
|
||||||
|
//
|
||||||
|
// Check whether the Ip6 Address setting is successed.
|
||||||
|
//
|
||||||
|
DataSize = 0;
|
||||||
|
Status = Ip6Cfg->GetData (
|
||||||
|
Ip6Cfg,
|
||||||
|
Ip6ConfigDataTypeManualAddress,
|
||||||
|
&DataSize,
|
||||||
|
NULL
|
||||||
|
);
|
||||||
|
if (Status != EFI_BUFFER_TOO_SMALL || DataSize == 0) {
|
||||||
|
Status = EFI_DEVICE_ERROR;
|
||||||
|
goto ON_EXIT;
|
||||||
|
}
|
||||||
|
|
||||||
|
Ip6Addr = AllocatePool (DataSize);
|
||||||
|
if (Ip6Addr == NULL) {
|
||||||
|
return EFI_OUT_OF_RESOURCES;
|
||||||
|
}
|
||||||
|
Status = Ip6Cfg->GetData (
|
||||||
|
Ip6Cfg,
|
||||||
|
Ip6ConfigDataTypeManualAddress,
|
||||||
|
&DataSize,
|
||||||
|
(VOID *) Ip6Addr
|
||||||
|
);
|
||||||
|
if (EFI_ERROR (Status)) {
|
||||||
|
Status = EFI_DEVICE_ERROR;
|
||||||
|
goto ON_EXIT;
|
||||||
|
}
|
||||||
|
|
||||||
|
for (Index = 0; Index < DataSize / sizeof (EFI_IPv6_ADDRESS); Index ++) {
|
||||||
|
if (CompareMem (Ip6Addr + Index, &CfgAddr, sizeof (EFI_IPv6_ADDRESS)) == 0) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (Index == DataSize / sizeof (EFI_IPv6_ADDRESS)) {
|
||||||
|
Status = EFI_ABORTED;
|
||||||
|
goto ON_EXIT;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
ON_EXIT:
|
||||||
|
if (MappedEvt != NULL) {
|
||||||
|
Ip6Cfg->UnregisterDataNotify (
|
||||||
|
Ip6Cfg,
|
||||||
|
Ip6ConfigDataTypeManualAddress,
|
||||||
|
MappedEvt
|
||||||
|
);
|
||||||
|
gBS->CloseEvent (MappedEvt);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (Ip6Addr != NULL) {
|
||||||
|
FreePool (Ip6Addr);
|
||||||
|
}
|
||||||
|
|
||||||
|
return Status;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
Start the S.A.R.R DHCPv6 process to acquire the IPv6 address and other Http boot information.
|
||||||
|
|
||||||
|
@param[in] Private Pointer to HTTP_BOOT private data.
|
||||||
|
|
||||||
|
@retval EFI_SUCCESS The S.A.R.R process successfully finished.
|
||||||
|
@retval Others Failed to finish the S.A.R.R process.
|
||||||
|
|
||||||
|
**/
|
||||||
|
EFI_STATUS
|
||||||
|
HttpBootDhcp6Sarr (
|
||||||
|
IN HTTP_BOOT_PRIVATE_DATA *Private
|
||||||
|
)
|
||||||
|
{
|
||||||
|
EFI_DHCP6_PROTOCOL *Dhcp6;
|
||||||
|
EFI_DHCP6_CONFIG_DATA Config;
|
||||||
|
EFI_DHCP6_MODE_DATA Mode;
|
||||||
|
EFI_DHCP6_RETRANSMISSION *Retransmit;
|
||||||
|
EFI_DHCP6_PACKET_OPTION *OptList[HTTP_BOOT_DHCP6_OPTION_MAX_NUM];
|
||||||
|
UINT32 OptCount;
|
||||||
|
UINT8 Buffer[HTTP_BOOT_DHCP6_OPTION_MAX_SIZE];
|
||||||
|
EFI_STATUS Status;
|
||||||
|
|
||||||
|
Dhcp6 = Private->Dhcp6;
|
||||||
|
ASSERT (Dhcp6 != NULL);
|
||||||
|
|
||||||
|
//
|
||||||
|
// Build options list for the request packet.
|
||||||
|
//
|
||||||
|
OptCount = HttpBootBuildDhcp6Options (Private, OptList, Buffer);
|
||||||
|
ASSERT (OptCount >0);
|
||||||
|
|
||||||
|
Retransmit = AllocateZeroPool (sizeof (EFI_DHCP6_RETRANSMISSION));
|
||||||
|
if (Retransmit == NULL) {
|
||||||
|
return EFI_OUT_OF_RESOURCES;
|
||||||
|
}
|
||||||
|
|
||||||
|
ZeroMem (&Mode, sizeof (EFI_DHCP6_MODE_DATA));
|
||||||
|
ZeroMem (&Config, sizeof (EFI_DHCP6_CONFIG_DATA));
|
||||||
|
|
||||||
|
Config.OptionCount = OptCount;
|
||||||
|
Config.OptionList = OptList;
|
||||||
|
Config.Dhcp6Callback = HttpBootDhcp6CallBack;
|
||||||
|
Config.CallbackContext = Private;
|
||||||
|
Config.IaInfoEvent = NULL;
|
||||||
|
Config.RapidCommit = FALSE;
|
||||||
|
Config.ReconfigureAccept = FALSE;
|
||||||
|
Config.IaDescriptor.IaId = NET_RANDOM (NetRandomInitSeed ());
|
||||||
|
Config.IaDescriptor.Type = EFI_DHCP6_IA_TYPE_NA;
|
||||||
|
Config.SolicitRetransmission = Retransmit;
|
||||||
|
Retransmit->Irt = 4;
|
||||||
|
Retransmit->Mrc = 4;
|
||||||
|
Retransmit->Mrt = 32;
|
||||||
|
Retransmit->Mrd = 60;
|
||||||
|
|
||||||
|
//
|
||||||
|
// Configure the DHCPv6 instance for HTTP boot.
|
||||||
|
//
|
||||||
|
Status = Dhcp6->Configure (Dhcp6, &Config);
|
||||||
|
FreePool (Retransmit);
|
||||||
|
if (EFI_ERROR (Status)) {
|
||||||
|
goto ON_EXIT;
|
||||||
|
}
|
||||||
|
//
|
||||||
|
// Initialize the record fields for DHCPv6 offer in private data.
|
||||||
|
//
|
||||||
|
Private->OfferNum = 0;
|
||||||
|
Private->SelectIndex = 0;
|
||||||
|
ZeroMem (Private->OfferCount, sizeof (Private->OfferCount));
|
||||||
|
ZeroMem (Private->OfferIndex, sizeof (Private->OfferIndex));
|
||||||
|
|
||||||
|
//
|
||||||
|
// Start DHCPv6 S.A.R.R. process to acquire IPv6 address.
|
||||||
|
//
|
||||||
|
Status = Dhcp6->Start (Dhcp6);
|
||||||
|
if (EFI_ERROR (Status)) {
|
||||||
|
goto ON_EXIT;
|
||||||
|
}
|
||||||
|
|
||||||
|
//
|
||||||
|
// Get the acquired IPv6 address and store them.
|
||||||
|
//
|
||||||
|
Status = Dhcp6->GetModeData (Dhcp6, &Mode, NULL);
|
||||||
|
if (EFI_ERROR (Status)) {
|
||||||
|
goto ON_EXIT;
|
||||||
|
}
|
||||||
|
|
||||||
|
ASSERT (Mode.Ia->State == Dhcp6Bound);
|
||||||
|
CopyMem (&Private->StationIp.v6, &Mode.Ia->IaAddress[0].IpAddress, sizeof (EFI_IPv6_ADDRESS));
|
||||||
|
|
||||||
|
AsciiPrint ("\n Station IPv6 address is ");
|
||||||
|
HttpBootShowIp6Addr (&Private->StationIp.v6);
|
||||||
|
AsciiPrint ("\n");
|
||||||
|
|
||||||
|
ON_EXIT:
|
||||||
|
if (EFI_ERROR (Status)) {
|
||||||
|
Dhcp6->Stop (Dhcp6);
|
||||||
|
Dhcp6->Configure (Dhcp6, NULL);
|
||||||
|
} else {
|
||||||
|
ZeroMem (&Config, sizeof (EFI_DHCP6_CONFIG_DATA));
|
||||||
|
ZeroMem (&Mode, sizeof (EFI_DHCP6_MODE_DATA));
|
||||||
|
Dhcp6->Configure (Dhcp6, &Config);
|
||||||
|
}
|
||||||
|
|
||||||
|
return Status;
|
||||||
|
|
||||||
|
}
|
||||||
|
|
198
NetworkPkg/HttpBootDxe/HttpBootDhcp6.h
Normal file
198
NetworkPkg/HttpBootDxe/HttpBootDhcp6.h
Normal file
@ -0,0 +1,198 @@
|
|||||||
|
/** @file
|
||||||
|
Functions declaration related with DHCPv6 for HTTP boot driver.
|
||||||
|
|
||||||
|
Copyright (c) 2015, Intel Corporation. All rights reserved.<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.
|
||||||
|
|
||||||
|
**/
|
||||||
|
|
||||||
|
|
||||||
|
#ifndef __EFI_HTTP_BOOT_DHCP6_H__
|
||||||
|
#define __EFI_HTTP_BOOT_DHCP6_H__
|
||||||
|
|
||||||
|
#define HTTP_BOOT_OFFER_MAX_NUM 16
|
||||||
|
#define HTTP_BOOT_DHCP6_OPTION_MAX_NUM 16
|
||||||
|
#define HTTP_BOOT_DHCP6_OPTION_MAX_SIZE 312
|
||||||
|
#define HTTP_BOOT_DHCP6_PACKET_MAX_SIZE 1472
|
||||||
|
#define HTTP_BOOT_IP6_ROUTE_TABLE_TIMEOUT 10
|
||||||
|
#define HTTP_BOOT_DEFAULT_HOPLIMIT 64
|
||||||
|
#define HTTP_BOOT_DEFAULT_LIFETIME 50000
|
||||||
|
|
||||||
|
|
||||||
|
#define HTTP_BOOT_DHCP6_OPT_CLIENT_ID 1
|
||||||
|
#define HTTP_BOOT_DHCP6_OPT_SERVER_ID 2
|
||||||
|
#define HTTP_BOOT_DHCP6_OPT_IA_NA 3
|
||||||
|
#define HTTP_BOOT_DHCP6_OPT_IA_TA 4
|
||||||
|
#define HTTP_BOOT_DHCP6_OPT_IAADDR 5
|
||||||
|
#define HTTP_BOOT_DHCP6_OPT_ORO 6
|
||||||
|
#define HTTP_BOOT_DHCP6_OPT_PREFERENCE 7
|
||||||
|
#define HTTP_BOOT_DHCP6_OPT_ELAPSED_TIME 8
|
||||||
|
#define HTTP_BOOT_DHCP6_OPT_REPLAY_MSG 9
|
||||||
|
#define HTTP_BOOT_DHCP6_OPT_AUTH 11
|
||||||
|
#define HTTP_BOOT_DHCP6_OPT_UNICAST 12
|
||||||
|
#define HTTP_BOOT_DHCP6_OPT_STATUS_CODE 13
|
||||||
|
#define HTTP_BOOT_DHCP6_OPT_RAPID_COMMIT 14
|
||||||
|
#define HTTP_BOOT_DHCP6_OPT_USER_CLASS 15
|
||||||
|
#define HTTP_BOOT_DHCP6_OPT_VENDOR_CLASS 16
|
||||||
|
#define HTTP_BOOT_DHCP6_OPT_VENDOR_OPTS 17
|
||||||
|
#define HTTP_BOOT_DHCP6_OPT_INTERFACE_ID 18
|
||||||
|
#define HTTP_BOOT_DHCP6_OPT_RECONFIG_MSG 19
|
||||||
|
#define HTTP_BOOT_DHCP6_OPT_RECONFIG_ACCEPT 20
|
||||||
|
#define HTTP_BOOT_DHCP6_OPT_DNS_SERVERS 23
|
||||||
|
#define HTTP_BOOT_DHCP6_OPT_BOOT_FILE_URL 59 // Assigned by IANA, RFC 5970
|
||||||
|
#define HTTP_BOOT_DHCP6_OPT_BOOT_FILE_PARAM 60 // Assigned by IANA, RFC 5970
|
||||||
|
#define HTTP_BOOT_DHCP6_OPT_ARCH 61 // Assigned by IANA, RFC 5970
|
||||||
|
#define HTTP_BOOT_DHCP6_OPT_UNDI 62 // Assigned by IANA, RFC 5970
|
||||||
|
#define HTTP_BOOT_DHCP6_ENTERPRISE_NUM 343 // TODO: IANA TBD: temporarily using Intel's
|
||||||
|
#define HTTP_BOOT_DHCP6_MAX_BOOT_FILE_SIZE 65535 // It's a limitation of bit length, 65535*512 bytes.
|
||||||
|
|
||||||
|
#define HTTP_BOOT_DHCP6_IDX_IA_NA 0
|
||||||
|
#define HTTP_BOOT_DHCP6_IDX_BOOT_FILE_URL 1
|
||||||
|
#define HTTP_BOOT_DHCP6_IDX_BOOT_FILE_PARAM 2
|
||||||
|
#define HTTP_BOOT_DHCP6_IDX_VENDOR_CLASS 3
|
||||||
|
#define HTTP_BOOT_DHCP6_IDX_DNS_SERVER 4
|
||||||
|
#define HTTP_BOOT_DHCP6_IDX_MAX 5
|
||||||
|
|
||||||
|
#pragma pack(1)
|
||||||
|
typedef struct {
|
||||||
|
UINT16 OpCode[256];
|
||||||
|
} HTTP_BOOT_DHCP6_OPTION_ORO;
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
UINT8 Type;
|
||||||
|
UINT8 MajorVer;
|
||||||
|
UINT8 MinorVer;
|
||||||
|
} HTTP_BOOT_DHCP6_OPTION_UNDI;
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
UINT16 Type;
|
||||||
|
} HTTP_BOOT_DHCP6_OPTION_ARCH;
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
UINT8 ClassIdentifier[10];
|
||||||
|
UINT8 ArchitecturePrefix[5];
|
||||||
|
UINT8 ArchitectureType[5];
|
||||||
|
UINT8 Lit3[1];
|
||||||
|
UINT8 InterfaceName[4];
|
||||||
|
UINT8 Lit4[1];
|
||||||
|
UINT8 UndiMajor[3];
|
||||||
|
UINT8 UndiMinor[3];
|
||||||
|
} HTTP_BOOT_CLASS_ID;
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
UINT32 Vendor;
|
||||||
|
UINT16 ClassLen;
|
||||||
|
HTTP_BOOT_CLASS_ID ClassId;
|
||||||
|
} HTTP_BOOT_DHCP6_OPTION_VENDOR_CLASS;
|
||||||
|
|
||||||
|
#pragma pack()
|
||||||
|
|
||||||
|
typedef union {
|
||||||
|
HTTP_BOOT_DHCP6_OPTION_ORO *Oro;
|
||||||
|
HTTP_BOOT_DHCP6_OPTION_UNDI *Undi;
|
||||||
|
HTTP_BOOT_DHCP6_OPTION_ARCH *Arch;
|
||||||
|
HTTP_BOOT_DHCP6_OPTION_VENDOR_CLASS *VendorClass;
|
||||||
|
} HTTP_BOOT_DHCP6_OPTION_ENTRY;
|
||||||
|
|
||||||
|
typedef union {
|
||||||
|
EFI_DHCP6_PACKET Offer;
|
||||||
|
EFI_DHCP6_PACKET Ack;
|
||||||
|
UINT8 Buffer[HTTP_BOOT_DHCP6_PACKET_MAX_SIZE];
|
||||||
|
} HTTP_BOOT_DHCP6_PACKET;
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
HTTP_BOOT_DHCP6_PACKET Packet;
|
||||||
|
HTTP_BOOT_OFFER_TYPE OfferType;
|
||||||
|
EFI_DHCP6_PACKET_OPTION *OptList[HTTP_BOOT_DHCP6_IDX_MAX];
|
||||||
|
VOID *UriParser;
|
||||||
|
} HTTP_BOOT_DHCP6_PACKET_CACHE;
|
||||||
|
|
||||||
|
#define GET_NEXT_DHCP6_OPTION(Opt) \
|
||||||
|
(EFI_DHCP6_PACKET_OPTION *) ((UINT8 *) (Opt) + \
|
||||||
|
sizeof (EFI_DHCP6_PACKET_OPTION) + (NTOHS ((Opt)->OpLen)) - 1)
|
||||||
|
|
||||||
|
#define GET_DHCP6_OPTION_SIZE(Pkt) \
|
||||||
|
((Pkt)->Length - sizeof (EFI_DHCP6_HEADER))
|
||||||
|
|
||||||
|
/**
|
||||||
|
Start the S.A.R.R DHCPv6 process to acquire the IPv6 address and other Http boot information.
|
||||||
|
|
||||||
|
@param[in] Private Pointer to HTTP_BOOT private data.
|
||||||
|
|
||||||
|
@retval EFI_SUCCESS The S.A.R.R process successfully finished.
|
||||||
|
@retval Others Failed to finish the S.A.R.R process.
|
||||||
|
|
||||||
|
**/
|
||||||
|
EFI_STATUS
|
||||||
|
HttpBootDhcp6Sarr (
|
||||||
|
IN HTTP_BOOT_PRIVATE_DATA *Private
|
||||||
|
);
|
||||||
|
|
||||||
|
/**
|
||||||
|
Set the IP6 policy to Automatic.
|
||||||
|
|
||||||
|
@param[in] Private The pointer to HTTP_BOOT_PRIVATE_DATA.
|
||||||
|
|
||||||
|
@retval EFI_SUCCESS Switch the IP policy succesfully.
|
||||||
|
@retval Others Unexpect error happened.
|
||||||
|
|
||||||
|
**/
|
||||||
|
EFI_STATUS
|
||||||
|
HttpBootSetIp6Policy (
|
||||||
|
IN HTTP_BOOT_PRIVATE_DATA *Private
|
||||||
|
);
|
||||||
|
|
||||||
|
/**
|
||||||
|
This function will register the default DNS addresses to the network device.
|
||||||
|
|
||||||
|
@param[in] Private The pointer to HTTP_BOOT_PRIVATE_DATA.
|
||||||
|
@param[in] DataLength Size of the buffer pointed to by DnsServerData in bytes.
|
||||||
|
@param[in] DnsServerData Point a list of DNS server address in an array
|
||||||
|
of EFI_IPv6_ADDRESS instances.
|
||||||
|
|
||||||
|
@retval EFI_SUCCESS The DNS configuration has been configured successfully.
|
||||||
|
@retval Others Failed to configure the address.
|
||||||
|
|
||||||
|
**/
|
||||||
|
EFI_STATUS
|
||||||
|
HttpBootSetIp6Dns (
|
||||||
|
IN HTTP_BOOT_PRIVATE_DATA *Private,
|
||||||
|
IN UINTN DataLength,
|
||||||
|
IN VOID *DnsServerData
|
||||||
|
);
|
||||||
|
|
||||||
|
/**
|
||||||
|
This function will register the IPv6 gateway address to the network device.
|
||||||
|
|
||||||
|
@param[in] Private The pointer to HTTP_BOOT_PRIVATE_DATA.
|
||||||
|
|
||||||
|
@retval EFI_SUCCESS The new IP configuration has been configured successfully.
|
||||||
|
@retval Others Failed to configure the address.
|
||||||
|
|
||||||
|
**/
|
||||||
|
EFI_STATUS
|
||||||
|
HttpBootSetIp6Gateway (
|
||||||
|
IN HTTP_BOOT_PRIVATE_DATA *Private
|
||||||
|
);
|
||||||
|
|
||||||
|
/**
|
||||||
|
This function will register the station IP address.
|
||||||
|
|
||||||
|
@param[in] Private The pointer to HTTP_BOOT_PRIVATE_DATA.
|
||||||
|
|
||||||
|
@retval EFI_SUCCESS The new IP address has been configured successfully.
|
||||||
|
@retval Others Failed to configure the address.
|
||||||
|
|
||||||
|
**/
|
||||||
|
EFI_STATUS
|
||||||
|
HttpBootSetIp6Address (
|
||||||
|
IN HTTP_BOOT_PRIVATE_DATA *Private
|
||||||
|
);
|
||||||
|
|
||||||
|
#endif
|
File diff suppressed because it is too large
Load Diff
@ -41,9 +41,11 @@ WITHOUT WARRANTIES OR REPRESENTATIONS OF ANY KIND, EITHER EXPRESS OR IMPLIED.
|
|||||||
//
|
//
|
||||||
#include <Protocol/NetworkInterfaceIdentifier.h>
|
#include <Protocol/NetworkInterfaceIdentifier.h>
|
||||||
#include <Protocol/Dhcp4.h>
|
#include <Protocol/Dhcp4.h>
|
||||||
|
#include <Protocol/Dhcp6.h>
|
||||||
|
#include <Protocol/Dns6.h>
|
||||||
#include <Protocol/Http.h>
|
#include <Protocol/Http.h>
|
||||||
#include <Protocol/Ip4Config2.h>
|
#include <Protocol/Ip4Config2.h>
|
||||||
|
#include <Protocol/Ip6Config.h>
|
||||||
//
|
//
|
||||||
// Produced Protocols
|
// Produced Protocols
|
||||||
//
|
//
|
||||||
@ -65,29 +67,45 @@ extern EFI_COMPONENT_NAME_PROTOCOL gHttpBootDxeComponentName;
|
|||||||
// Private data structure
|
// Private data structure
|
||||||
//
|
//
|
||||||
typedef struct _HTTP_BOOT_PRIVATE_DATA HTTP_BOOT_PRIVATE_DATA;
|
typedef struct _HTTP_BOOT_PRIVATE_DATA HTTP_BOOT_PRIVATE_DATA;
|
||||||
|
typedef struct _HTTP_BOOT_VIRTUAL_NIC HTTP_BOOT_VIRTUAL_NIC;
|
||||||
|
|
||||||
//
|
//
|
||||||
// Include files with internal function prototypes
|
// Include files with internal function prototypes
|
||||||
//
|
//
|
||||||
#include "HttpBootComponentName.h"
|
#include "HttpBootComponentName.h"
|
||||||
#include "HttpBootDhcp4.h"
|
#include "HttpBootDhcp4.h"
|
||||||
|
#include "HttpBootDhcp6.h"
|
||||||
#include "HttpBootImpl.h"
|
#include "HttpBootImpl.h"
|
||||||
#include "HttpBootSupport.h"
|
#include "HttpBootSupport.h"
|
||||||
#include "HttpBootClient.h"
|
#include "HttpBootClient.h"
|
||||||
|
|
||||||
typedef union {
|
typedef union {
|
||||||
HTTP_BOOT_DHCP4_PACKET_CACHE Dhcp4;
|
HTTP_BOOT_DHCP4_PACKET_CACHE Dhcp4;
|
||||||
|
HTTP_BOOT_DHCP6_PACKET_CACHE Dhcp6;
|
||||||
} HTTP_BOOT_DHCP_PACKET_CACHE;
|
} HTTP_BOOT_DHCP_PACKET_CACHE;
|
||||||
|
|
||||||
|
struct _HTTP_BOOT_VIRTUAL_NIC {
|
||||||
|
UINT32 Signature;
|
||||||
|
EFI_HANDLE Controller;
|
||||||
|
EFI_LOAD_FILE_PROTOCOL LoadFile;
|
||||||
|
EFI_DEVICE_PATH_PROTOCOL *DevicePath;
|
||||||
|
HTTP_BOOT_PRIVATE_DATA *Private;
|
||||||
|
};
|
||||||
|
|
||||||
struct _HTTP_BOOT_PRIVATE_DATA {
|
struct _HTTP_BOOT_PRIVATE_DATA {
|
||||||
UINT32 Signature;
|
UINT32 Signature;
|
||||||
EFI_HANDLE Controller;
|
EFI_HANDLE Controller;
|
||||||
EFI_HANDLE Image;
|
EFI_HANDLE Image;
|
||||||
|
|
||||||
|
HTTP_BOOT_VIRTUAL_NIC *Ip4Nic;
|
||||||
|
HTTP_BOOT_VIRTUAL_NIC *Ip6Nic;
|
||||||
|
|
||||||
//
|
//
|
||||||
// Cousumed children
|
// Cousumed children
|
||||||
//
|
//
|
||||||
|
EFI_HANDLE Ip6Child;
|
||||||
EFI_HANDLE Dhcp4Child;
|
EFI_HANDLE Dhcp4Child;
|
||||||
|
EFI_HANDLE Dhcp6Child;
|
||||||
HTTP_IO HttpIo;
|
HTTP_IO HttpIo;
|
||||||
BOOLEAN HttpCreated;
|
BOOLEAN HttpCreated;
|
||||||
|
|
||||||
@ -95,14 +113,13 @@ struct _HTTP_BOOT_PRIVATE_DATA {
|
|||||||
// Consumed protocol
|
// Consumed protocol
|
||||||
//
|
//
|
||||||
EFI_NETWORK_INTERFACE_IDENTIFIER_PROTOCOL *Nii;
|
EFI_NETWORK_INTERFACE_IDENTIFIER_PROTOCOL *Nii;
|
||||||
|
EFI_IP6_PROTOCOL *Ip6;
|
||||||
EFI_IP4_CONFIG2_PROTOCOL *Ip4Config2;
|
EFI_IP4_CONFIG2_PROTOCOL *Ip4Config2;
|
||||||
|
EFI_IP6_CONFIG_PROTOCOL *Ip6Config;
|
||||||
EFI_DHCP4_PROTOCOL *Dhcp4;
|
EFI_DHCP4_PROTOCOL *Dhcp4;
|
||||||
|
EFI_DHCP6_PROTOCOL *Dhcp6;
|
||||||
EFI_DEVICE_PATH_PROTOCOL *ParentDevicePath;
|
EFI_DEVICE_PATH_PROTOCOL *ParentDevicePath;
|
||||||
|
|
||||||
//
|
|
||||||
// Produced children
|
|
||||||
//
|
|
||||||
EFI_HANDLE ChildHandle;
|
|
||||||
|
|
||||||
//
|
//
|
||||||
// Produced protocol
|
// Produced protocol
|
||||||
@ -119,10 +136,12 @@ struct _HTTP_BOOT_PRIVATE_DATA {
|
|||||||
EFI_IP_ADDRESS StationIp;
|
EFI_IP_ADDRESS StationIp;
|
||||||
EFI_IP_ADDRESS SubnetMask;
|
EFI_IP_ADDRESS SubnetMask;
|
||||||
EFI_IP_ADDRESS GatewayIp;
|
EFI_IP_ADDRESS GatewayIp;
|
||||||
|
EFI_IP_ADDRESS ServerIp;
|
||||||
UINT16 Port;
|
UINT16 Port;
|
||||||
CHAR8 *BootFileUri;
|
CHAR8 *BootFileUri;
|
||||||
VOID *BootFileUriParser;
|
VOID *BootFileUriParser;
|
||||||
UINTN BootFileSize;
|
UINTN BootFileSize;
|
||||||
|
BOOLEAN NoGateway;
|
||||||
|
|
||||||
//
|
//
|
||||||
// Cached HTTP data
|
// Cached HTTP data
|
||||||
@ -167,9 +186,10 @@ struct _HTTP_BOOT_PRIVATE_DATA {
|
|||||||
};
|
};
|
||||||
|
|
||||||
#define HTTP_BOOT_PRIVATE_DATA_SIGNATURE SIGNATURE_32 ('H', 'B', 'P', 'D')
|
#define HTTP_BOOT_PRIVATE_DATA_SIGNATURE SIGNATURE_32 ('H', 'B', 'P', 'D')
|
||||||
|
#define HTTP_BOOT_VIRTUAL_NIC_SIGNATURE SIGNATURE_32 ('H', 'B', 'V', 'N')
|
||||||
#define HTTP_BOOT_PRIVATE_DATA_FROM_LOADFILE(a) CR (a, HTTP_BOOT_PRIVATE_DATA, LoadFile, HTTP_BOOT_PRIVATE_DATA_SIGNATURE)
|
#define HTTP_BOOT_PRIVATE_DATA_FROM_LOADFILE(a) CR (a, HTTP_BOOT_PRIVATE_DATA, LoadFile, HTTP_BOOT_PRIVATE_DATA_SIGNATURE)
|
||||||
#define HTTP_BOOT_PRIVATE_DATA_FROM_ID(a) CR (a, HTTP_BOOT_PRIVATE_DATA, Id, HTTP_BOOT_PRIVATE_DATA_SIGNATURE)
|
#define HTTP_BOOT_PRIVATE_DATA_FROM_ID(a) CR (a, HTTP_BOOT_PRIVATE_DATA, Id, HTTP_BOOT_PRIVATE_DATA_SIGNATURE)
|
||||||
|
#define HTTP_BOOT_VIRTUAL_NIC_FROM_LOADFILE(a) CR (a, HTTP_BOOT_VIRTUAL_NIC, LoadFile, HTTP_BOOT_VIRTUAL_NIC_SIGNATURE)
|
||||||
extern EFI_LOAD_FILE_PROTOCOL gHttpBootDxeLoadFile;
|
extern EFI_LOAD_FILE_PROTOCOL gHttpBootDxeLoadFile;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -300,4 +320,131 @@ HttpBootIp4DxeDriverBindingStop (
|
|||||||
IN EFI_HANDLE *ChildHandleBuffer OPTIONAL
|
IN EFI_HANDLE *ChildHandleBuffer OPTIONAL
|
||||||
);
|
);
|
||||||
|
|
||||||
|
/**
|
||||||
|
Tests to see if this driver supports a given controller. If a child device is provided,
|
||||||
|
it further tests to see if this driver supports creating a handle for the specified child device.
|
||||||
|
|
||||||
|
This function checks to see if the driver specified by This supports the device specified by
|
||||||
|
ControllerHandle. Drivers will typically use the device path attached to
|
||||||
|
ControllerHandle and/or the services from the bus I/O abstraction attached to
|
||||||
|
ControllerHandle to determine if the driver supports ControllerHandle. This function
|
||||||
|
may be called many times during platform initialization. In order to reduce boot times, the tests
|
||||||
|
performed by this function must be very small, and take as little time as possible to execute. This
|
||||||
|
function must not change the state of any hardware devices, and this function must be aware that the
|
||||||
|
device specified by ControllerHandle may already be managed by the same driver or a
|
||||||
|
different driver. This function must match its calls to AllocatePages() with FreePages(),
|
||||||
|
AllocatePool() with FreePool(), and OpenProtocol() with CloseProtocol().
|
||||||
|
Because ControllerHandle may have been previously started by the same driver, if a protocol is
|
||||||
|
already in the opened state, then it must not be closed with CloseProtocol(). This is required
|
||||||
|
to guarantee the state of ControllerHandle is not modified by this function.
|
||||||
|
|
||||||
|
@param[in] This A pointer to the EFI_DRIVER_BINDING_PROTOCOL instance.
|
||||||
|
@param[in] ControllerHandle The handle of the controller to test. This handle
|
||||||
|
must support a protocol interface that supplies
|
||||||
|
an I/O abstraction to the driver.
|
||||||
|
@param[in] RemainingDevicePath A pointer to the remaining portion of a device path. This
|
||||||
|
parameter is ignored by device drivers, and is optional for bus
|
||||||
|
drivers. For bus drivers, if this parameter is not NULL, then
|
||||||
|
the bus driver must determine if the bus controller specified
|
||||||
|
by ControllerHandle and the child controller specified
|
||||||
|
by RemainingDevicePath are both supported by this
|
||||||
|
bus driver.
|
||||||
|
|
||||||
|
@retval EFI_SUCCESS The device specified by ControllerHandle and
|
||||||
|
RemainingDevicePath is supported by the driver specified by This.
|
||||||
|
@retval EFI_ALREADY_STARTED The device specified by ControllerHandle and
|
||||||
|
RemainingDevicePath is already being managed by the driver
|
||||||
|
specified by This.
|
||||||
|
@retval EFI_ACCESS_DENIED The device specified by ControllerHandle and
|
||||||
|
RemainingDevicePath is already being managed by a different
|
||||||
|
driver or an application that requires exclusive access.
|
||||||
|
Currently not implemented.
|
||||||
|
@retval EFI_UNSUPPORTED The device specified by ControllerHandle and
|
||||||
|
RemainingDevicePath is not supported by the driver specified by This.
|
||||||
|
**/
|
||||||
|
EFI_STATUS
|
||||||
|
EFIAPI
|
||||||
|
HttpBootIp6DxeDriverBindingSupported (
|
||||||
|
IN EFI_DRIVER_BINDING_PROTOCOL *This,
|
||||||
|
IN EFI_HANDLE ControllerHandle,
|
||||||
|
IN EFI_DEVICE_PATH_PROTOCOL *RemainingDevicePath OPTIONAL
|
||||||
|
);
|
||||||
|
|
||||||
|
/**
|
||||||
|
Starts a device controller or a bus controller.
|
||||||
|
|
||||||
|
The Start() function is designed to be invoked from the EFI boot service ConnectController().
|
||||||
|
As a result, much of the error checking on the parameters to Start() has been moved into this
|
||||||
|
common boot service. It is legal to call Start() from other locations,
|
||||||
|
but the following calling restrictions must be followed, or the system behavior will not be deterministic.
|
||||||
|
1. ControllerHandle must be a valid EFI_HANDLE.
|
||||||
|
2. If RemainingDevicePath is not NULL, then it must be a pointer to a naturally aligned
|
||||||
|
EFI_DEVICE_PATH_PROTOCOL.
|
||||||
|
3. Prior to calling Start(), the Supported() function for the driver specified by This must
|
||||||
|
have been called with the same calling parameters, and Supported() must have returned EFI_SUCCESS.
|
||||||
|
|
||||||
|
@param[in] This A pointer to the EFI_DRIVER_BINDING_PROTOCOL instance.
|
||||||
|
@param[in] ControllerHandle The handle of the controller to start. This handle
|
||||||
|
must support a protocol interface that supplies
|
||||||
|
an I/O abstraction to the driver.
|
||||||
|
@param[in] RemainingDevicePath A pointer to the remaining portion of a device path. This
|
||||||
|
parameter is ignored by device drivers, and is optional for bus
|
||||||
|
drivers. For a bus driver, if this parameter is NULL, then handles
|
||||||
|
for all the children of Controller are created by this driver.
|
||||||
|
If this parameter is not NULL and the first Device Path Node is
|
||||||
|
not the End of Device Path Node, then only the handle for the
|
||||||
|
child device specified by the first Device Path Node of
|
||||||
|
RemainingDevicePath is created by this driver.
|
||||||
|
If the first Device Path Node of RemainingDevicePath is
|
||||||
|
the End of Device Path Node, no child handle is created by this
|
||||||
|
driver.
|
||||||
|
|
||||||
|
@retval EFI_SUCCESS The device was started.
|
||||||
|
@retval EFI_DEVICE_ERROR The device could not be started due to a device error.Currently not implemented.
|
||||||
|
@retval EFI_OUT_OF_RESOURCES The request could not be completed due to a lack of resources.
|
||||||
|
@retval Others The driver failded to start the device.
|
||||||
|
|
||||||
|
**/
|
||||||
|
EFI_STATUS
|
||||||
|
EFIAPI
|
||||||
|
HttpBootIp6DxeDriverBindingStart (
|
||||||
|
IN EFI_DRIVER_BINDING_PROTOCOL *This,
|
||||||
|
IN EFI_HANDLE ControllerHandle,
|
||||||
|
IN EFI_DEVICE_PATH_PROTOCOL *RemainingDevicePath OPTIONAL
|
||||||
|
);
|
||||||
|
|
||||||
|
/**
|
||||||
|
Stops a device controller or a bus controller.
|
||||||
|
|
||||||
|
The Stop() function is designed to be invoked from the EFI boot service DisconnectController().
|
||||||
|
As a result, much of the error checking on the parameters to Stop() has been moved
|
||||||
|
into this common boot service. It is legal to call Stop() from other locations,
|
||||||
|
but the following calling restrictions must be followed, or the system behavior will not be deterministic.
|
||||||
|
1. ControllerHandle must be a valid EFI_HANDLE that was used on a previous call to this
|
||||||
|
same driver's Start() function.
|
||||||
|
2. The first NumberOfChildren handles of ChildHandleBuffer must all be a valid
|
||||||
|
EFI_HANDLE. In addition, all of these handles must have been created in this driver's
|
||||||
|
Start() function, and the Start() function must have called OpenProtocol() on
|
||||||
|
ControllerHandle with an Attribute of EFI_OPEN_PROTOCOL_BY_CHILD_CONTROLLER.
|
||||||
|
|
||||||
|
@param[in] This A pointer to the EFI_DRIVER_BINDING_PROTOCOL instance.
|
||||||
|
@param[in] ControllerHandle A handle to the device being stopped. The handle must
|
||||||
|
support a bus specific I/O protocol for the driver
|
||||||
|
to use to stop the device.
|
||||||
|
@param[in] NumberOfChildren The number of child device handles in ChildHandleBuffer.
|
||||||
|
@param[in] ChildHandleBuffer An array of child handles to be freed. May be NULL
|
||||||
|
if NumberOfChildren is 0.
|
||||||
|
|
||||||
|
@retval EFI_SUCCESS The device was stopped.
|
||||||
|
@retval EFI_DEVICE_ERROR The device could not be stopped due to a device error.
|
||||||
|
|
||||||
|
**/
|
||||||
|
EFI_STATUS
|
||||||
|
EFIAPI
|
||||||
|
HttpBootIp6DxeDriverBindingStop (
|
||||||
|
IN EFI_DRIVER_BINDING_PROTOCOL *This,
|
||||||
|
IN EFI_HANDLE ControllerHandle,
|
||||||
|
IN UINTN NumberOfChildren,
|
||||||
|
IN EFI_HANDLE *ChildHandleBuffer OPTIONAL
|
||||||
|
);
|
||||||
#endif
|
#endif
|
||||||
|
@ -35,6 +35,8 @@
|
|||||||
HttpBootImpl.c
|
HttpBootImpl.c
|
||||||
HttpBootDhcp4.h
|
HttpBootDhcp4.h
|
||||||
HttpBootDhcp4.c
|
HttpBootDhcp4.c
|
||||||
|
HttpBootDhcp6.h
|
||||||
|
HttpBootDhcp6.c
|
||||||
HttpBootSupport.h
|
HttpBootSupport.h
|
||||||
HttpBootSupport.c
|
HttpBootSupport.c
|
||||||
HttpBootClient.h
|
HttpBootClient.h
|
||||||
@ -62,6 +64,13 @@
|
|||||||
gEfiDhcp4ServiceBindingProtocolGuid ## TO_START
|
gEfiDhcp4ServiceBindingProtocolGuid ## TO_START
|
||||||
gEfiDhcp4ProtocolGuid ## TO_START
|
gEfiDhcp4ProtocolGuid ## TO_START
|
||||||
gEfiIp4Config2ProtocolGuid ## TO_START
|
gEfiIp4Config2ProtocolGuid ## TO_START
|
||||||
|
gEfiDhcp6ServiceBindingProtocolGuid ## TO_START
|
||||||
|
gEfiDhcp6ProtocolGuid ## TO_START
|
||||||
|
gEfiDns6ServiceBindingProtocolGuid ## SOMETIMES_CONSUMES
|
||||||
|
gEfiDns6ProtocolGuid ## SOMETIMES_CONSUMES
|
||||||
|
gEfiIp6ServiceBindingProtocolGuid ## TO_START
|
||||||
|
gEfiIp6ProtocolGuid ## TO_START
|
||||||
|
gEfiIp6ConfigProtocolGuid ## TO_START
|
||||||
gEfiNetworkInterfaceIdentifierProtocolGuid_31 ## SOMETIMES_CONSUMES
|
gEfiNetworkInterfaceIdentifierProtocolGuid_31 ## SOMETIMES_CONSUMES
|
||||||
|
|
||||||
[UserExtensions.TianoCore."ExtraFiles"]
|
[UserExtensions.TianoCore."ExtraFiles"]
|
||||||
|
Binary file not shown.
@ -18,6 +18,9 @@ WITHOUT WARRANTIES OR REPRESENTATIONS OF ANY KIND, EITHER EXPRESS OR IMPLIED.
|
|||||||
Enable the use of UEFI HTTP boot function.
|
Enable the use of UEFI HTTP boot function.
|
||||||
|
|
||||||
@param[in] Private The pointer to the driver's private data.
|
@param[in] Private The pointer to the driver's private data.
|
||||||
|
@param[in] UsingIpv6 Specifies the type of IP addresses that are to be
|
||||||
|
used during the session that is being started.
|
||||||
|
Set to TRUE for IPv6, and FALSE for IPv4.
|
||||||
|
|
||||||
@retval EFI_SUCCESS HTTP boot was successfully enabled.
|
@retval EFI_SUCCESS HTTP boot was successfully enabled.
|
||||||
@retval EFI_INVALID_PARAMETER Private is NULL.
|
@retval EFI_INVALID_PARAMETER Private is NULL.
|
||||||
@ -26,10 +29,12 @@ WITHOUT WARRANTIES OR REPRESENTATIONS OF ANY KIND, EITHER EXPRESS OR IMPLIED.
|
|||||||
**/
|
**/
|
||||||
EFI_STATUS
|
EFI_STATUS
|
||||||
HttpBootStart (
|
HttpBootStart (
|
||||||
IN HTTP_BOOT_PRIVATE_DATA *Private
|
IN HTTP_BOOT_PRIVATE_DATA *Private,
|
||||||
|
IN BOOLEAN UsingIpv6
|
||||||
)
|
)
|
||||||
{
|
{
|
||||||
UINTN Index;
|
UINTN Index;
|
||||||
|
EFI_STATUS Status;
|
||||||
|
|
||||||
if (Private == NULL) {
|
if (Private == NULL) {
|
||||||
return EFI_INVALID_PARAMETER;
|
return EFI_INVALID_PARAMETER;
|
||||||
@ -39,25 +44,47 @@ HttpBootStart (
|
|||||||
return EFI_ALREADY_STARTED;
|
return EFI_ALREADY_STARTED;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
//
|
||||||
|
// Detect whether using ipv6 or not, and set it to the private data.
|
||||||
|
//
|
||||||
|
if (UsingIpv6 && Private->Ip6Nic != NULL) {
|
||||||
|
Private->UsingIpv6 = TRUE;
|
||||||
|
} else if (!UsingIpv6 && Private->Ip4Nic != NULL) {
|
||||||
|
Private->UsingIpv6 = FALSE;
|
||||||
|
} else {
|
||||||
|
return EFI_UNSUPPORTED;
|
||||||
|
}
|
||||||
|
|
||||||
|
//
|
||||||
|
// Init the content of cached DHCP offer list.
|
||||||
|
//
|
||||||
|
ZeroMem (Private->OfferBuffer, sizeof (Private->OfferBuffer));
|
||||||
if (!Private->UsingIpv6) {
|
if (!Private->UsingIpv6) {
|
||||||
//
|
|
||||||
// Init the content of cached DHCP offer list.
|
|
||||||
//
|
|
||||||
ZeroMem (Private->OfferBuffer, sizeof (Private->OfferBuffer));
|
|
||||||
for (Index = 0; Index < HTTP_BOOT_OFFER_MAX_NUM; Index++) {
|
for (Index = 0; Index < HTTP_BOOT_OFFER_MAX_NUM; Index++) {
|
||||||
Private->OfferBuffer[Index].Dhcp4.Packet.Offer.Size = HTTP_BOOT_DHCP4_PACKET_MAX_SIZE;
|
Private->OfferBuffer[Index].Dhcp4.Packet.Offer.Size = HTTP_BOOT_DHCP4_PACKET_MAX_SIZE;
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
ASSERT (FALSE);
|
for (Index = 0; Index < HTTP_BOOT_OFFER_MAX_NUM; Index++) {
|
||||||
|
Private->OfferBuffer[Index].Dhcp6.Packet.Offer.Size = HTTP_BOOT_DHCP6_PACKET_MAX_SIZE;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (Private->UsingIpv6) {
|
||||||
|
//
|
||||||
|
// Set Ip6 policy to Automatic to start the Ip6 router discovery.
|
||||||
|
//
|
||||||
|
Status = HttpBootSetIp6Policy (Private);
|
||||||
|
if (EFI_ERROR (Status)) {
|
||||||
|
return Status;
|
||||||
|
}
|
||||||
|
}
|
||||||
Private->Started = TRUE;
|
Private->Started = TRUE;
|
||||||
|
|
||||||
return EFI_SUCCESS;
|
return EFI_SUCCESS;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
Attempt to complete a DHCPv4 D.O.R.A sequence to retrieve the boot resource information.
|
Attempt to complete a DHCPv4 D.O.R.A or DHCPv6 S.R.A.A sequence to retrieve the boot resource information.
|
||||||
|
|
||||||
@param[in] Private The pointer to the driver's private data.
|
@param[in] Private The pointer to the driver's private data.
|
||||||
|
|
||||||
@ -86,9 +113,15 @@ HttpBootDhcp (
|
|||||||
Status = EFI_DEVICE_ERROR;
|
Status = EFI_DEVICE_ERROR;
|
||||||
|
|
||||||
if (!Private->UsingIpv6) {
|
if (!Private->UsingIpv6) {
|
||||||
|
//
|
||||||
|
// Start D.O.R.A process to get a IPv4 address and other boot information.
|
||||||
|
//
|
||||||
Status = HttpBootDhcp4Dora (Private);
|
Status = HttpBootDhcp4Dora (Private);
|
||||||
} else {
|
} else {
|
||||||
ASSERT (FALSE);
|
//
|
||||||
|
// Start S.A.R.R process to get a IPv6 address and other boot information.
|
||||||
|
//
|
||||||
|
Status = HttpBootDhcp6Sarr (Private);
|
||||||
}
|
}
|
||||||
|
|
||||||
return Status;
|
return Status;
|
||||||
@ -241,7 +274,7 @@ HttpBootStop (
|
|||||||
Private->BootFileUriParser = NULL;
|
Private->BootFileUriParser = NULL;
|
||||||
Private->BootFileSize = 0;
|
Private->BootFileSize = 0;
|
||||||
Private->SelectIndex = 0;
|
Private->SelectIndex = 0;
|
||||||
Private->SelectProxyType = HttpOfferTypeMax;
|
Private->SelectProxyType = HttpOfferTypeMax;
|
||||||
|
|
||||||
if (!Private->UsingIpv6) {
|
if (!Private->UsingIpv6) {
|
||||||
//
|
//
|
||||||
@ -256,7 +289,17 @@ HttpBootStop (
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
ASSERT (FALSE);
|
//
|
||||||
|
// Stop and release the DHCP6 child.
|
||||||
|
//
|
||||||
|
Private->Dhcp6->Stop (Private->Dhcp6);
|
||||||
|
Private->Dhcp6->Configure (Private->Dhcp6, NULL);
|
||||||
|
|
||||||
|
for (Index = 0; Index < HTTP_BOOT_OFFER_MAX_NUM; Index++) {
|
||||||
|
if (Private->OfferBuffer[Index].Dhcp6.UriParser) {
|
||||||
|
HttpUrlFreeParser (Private->OfferBuffer[Index].Dhcp6.UriParser);
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
ZeroMem (Private->OfferBuffer, sizeof (Private->OfferBuffer));
|
ZeroMem (Private->OfferBuffer, sizeof (Private->OfferBuffer));
|
||||||
@ -309,7 +352,9 @@ HttpBootDxeLoadFile (
|
|||||||
)
|
)
|
||||||
{
|
{
|
||||||
HTTP_BOOT_PRIVATE_DATA *Private;
|
HTTP_BOOT_PRIVATE_DATA *Private;
|
||||||
|
HTTP_BOOT_VIRTUAL_NIC *VirtualNic;
|
||||||
BOOLEAN MediaPresent;
|
BOOLEAN MediaPresent;
|
||||||
|
BOOLEAN UsingIpv6;
|
||||||
EFI_STATUS Status;
|
EFI_STATUS Status;
|
||||||
|
|
||||||
if (This == NULL || BufferSize == NULL) {
|
if (This == NULL || BufferSize == NULL) {
|
||||||
@ -323,8 +368,10 @@ HttpBootDxeLoadFile (
|
|||||||
return EFI_UNSUPPORTED;
|
return EFI_UNSUPPORTED;
|
||||||
}
|
}
|
||||||
|
|
||||||
Private = HTTP_BOOT_PRIVATE_DATA_FROM_LOADFILE (This);
|
VirtualNic = HTTP_BOOT_VIRTUAL_NIC_FROM_LOADFILE (This);
|
||||||
|
Private = VirtualNic->Private;
|
||||||
|
UsingIpv6 = FALSE;
|
||||||
|
|
||||||
//
|
//
|
||||||
// Check media status before HTTP boot start
|
// Check media status before HTTP boot start
|
||||||
//
|
//
|
||||||
@ -334,10 +381,26 @@ HttpBootDxeLoadFile (
|
|||||||
return EFI_NO_MEDIA;
|
return EFI_NO_MEDIA;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
//
|
||||||
|
// Check whether the virtual nic is using IPv6 or not.
|
||||||
|
//
|
||||||
|
if (VirtualNic == Private->Ip6Nic) {
|
||||||
|
UsingIpv6 = TRUE;
|
||||||
|
}
|
||||||
|
|
||||||
//
|
//
|
||||||
// Initialize HTTP boot and load the boot file.
|
// Initialize HTTP boot and load the boot file.
|
||||||
//
|
//
|
||||||
Status = HttpBootStart (Private);
|
Status = HttpBootStart (Private, UsingIpv6);
|
||||||
|
if (Status == EFI_ALREADY_STARTED && UsingIpv6 != Private->UsingIpv6) {
|
||||||
|
//
|
||||||
|
// Http boot Driver has already been started but not on the required IP version, restart it.
|
||||||
|
//
|
||||||
|
Status = HttpBootStop (Private);
|
||||||
|
if (!EFI_ERROR (Status)) {
|
||||||
|
Status = HttpBootStart (Private, UsingIpv6);
|
||||||
|
}
|
||||||
|
}
|
||||||
if (Status == EFI_SUCCESS || Status == EFI_ALREADY_STARTED) {
|
if (Status == EFI_SUCCESS || Status == EFI_ALREADY_STARTED) {
|
||||||
Status = HttpBootLoadFile (Private, BufferSize, Buffer);
|
Status = HttpBootLoadFile (Private, BufferSize, Buffer);
|
||||||
}
|
}
|
||||||
@ -345,11 +408,19 @@ HttpBootDxeLoadFile (
|
|||||||
if (Status != EFI_SUCCESS && Status != EFI_BUFFER_TOO_SMALL) {
|
if (Status != EFI_SUCCESS && Status != EFI_BUFFER_TOO_SMALL) {
|
||||||
HttpBootStop (Private);
|
HttpBootStop (Private);
|
||||||
} else {
|
} else {
|
||||||
//
|
if (!Private->UsingIpv6) {
|
||||||
// Stop and release the DHCP4 child.
|
//
|
||||||
//
|
// Stop and release the DHCP4 child.
|
||||||
Private->Dhcp4->Stop (Private->Dhcp4);
|
//
|
||||||
Private->Dhcp4->Configure (Private->Dhcp4, NULL);
|
Private->Dhcp4->Stop (Private->Dhcp4);
|
||||||
|
Private->Dhcp4->Configure (Private->Dhcp4, NULL);
|
||||||
|
} else {
|
||||||
|
//
|
||||||
|
// Stop and release the DHCP6 child.
|
||||||
|
//
|
||||||
|
Private->Dhcp6->Stop (Private->Dhcp6);
|
||||||
|
Private->Dhcp6->Configure (Private->Dhcp6, NULL);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return Status;
|
return Status;
|
||||||
|
@ -15,7 +15,7 @@ WITHOUT WARRANTIES OR REPRESENTATIONS OF ANY KIND, EITHER EXPRESS OR IMPLIED.
|
|||||||
#define __EFI_HTTP_BOOT_IMPL_H__
|
#define __EFI_HTTP_BOOT_IMPL_H__
|
||||||
|
|
||||||
/**
|
/**
|
||||||
Attempt to complete a DHCPv4 D.O.R.A sequence to retrieve the boot resource information.
|
Attempt to complete a DHCPv4 D.O.R.A or DHCPv6 S.R.A.A sequence to retrieve the boot resource information.
|
||||||
|
|
||||||
@param[in] Private The pointer to the driver's private data.
|
@param[in] Private The pointer to the driver's private data.
|
||||||
|
|
||||||
|
@ -42,6 +42,31 @@ HttpBootGetNicByIp4Children (
|
|||||||
return NicHandle;
|
return NicHandle;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
Get the Nic handle using any child handle in the IPv6 stack.
|
||||||
|
|
||||||
|
@param[in] ControllerHandle Pointer to child handle over IPv6.
|
||||||
|
|
||||||
|
@return NicHandle The pointer to the Nic handle.
|
||||||
|
@return NULL Can't find the Nic handle.
|
||||||
|
|
||||||
|
**/
|
||||||
|
EFI_HANDLE
|
||||||
|
HttpBootGetNicByIp6Children (
|
||||||
|
IN EFI_HANDLE ControllerHandle
|
||||||
|
)
|
||||||
|
{
|
||||||
|
EFI_HANDLE NicHandle;
|
||||||
|
NicHandle = NetLibGetNicHandle (ControllerHandle, &gEfiHttpProtocolGuid);
|
||||||
|
if (NicHandle == NULL) {
|
||||||
|
NicHandle = NetLibGetNicHandle (ControllerHandle, &gEfiDhcp6ProtocolGuid);
|
||||||
|
if (NicHandle == NULL) {
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return NicHandle;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
This function is to convert UINTN to ASCII string with the required formatting.
|
This function is to convert UINTN to ASCII string with the required formatting.
|
||||||
@ -89,6 +114,242 @@ HttpBootShowIp4Addr (
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
This function is to display the IPv6 address.
|
||||||
|
|
||||||
|
@param[in] Ip The pointer to the IPv6 address.
|
||||||
|
|
||||||
|
**/
|
||||||
|
VOID
|
||||||
|
HttpBootShowIp6Addr (
|
||||||
|
IN EFI_IPv6_ADDRESS *Ip
|
||||||
|
)
|
||||||
|
{
|
||||||
|
UINTN Index;
|
||||||
|
|
||||||
|
for (Index = 0; Index < 16; Index++) {
|
||||||
|
|
||||||
|
if (Ip->Addr[Index] != 0) {
|
||||||
|
AsciiPrint ("%x", Ip->Addr[Index]);
|
||||||
|
}
|
||||||
|
Index++;
|
||||||
|
if (Index > 15) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (((Ip->Addr[Index] & 0xf0) == 0) && (Ip->Addr[Index - 1] != 0)) {
|
||||||
|
AsciiPrint ("0");
|
||||||
|
}
|
||||||
|
AsciiPrint ("%x", Ip->Addr[Index]);
|
||||||
|
if (Index < 15) {
|
||||||
|
AsciiPrint (":");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
Notify the callback function when an event is triggered.
|
||||||
|
|
||||||
|
@param[in] Event The triggered event.
|
||||||
|
@param[in] Context The opaque parameter to the function.
|
||||||
|
|
||||||
|
**/
|
||||||
|
VOID
|
||||||
|
EFIAPI
|
||||||
|
HttpBootCommonNotify (
|
||||||
|
IN EFI_EVENT Event,
|
||||||
|
IN VOID *Context
|
||||||
|
)
|
||||||
|
{
|
||||||
|
*((BOOLEAN *) Context) = TRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
Retrieve the host address using the EFI_DNS6_PROTOCOL.
|
||||||
|
|
||||||
|
@param[in] Private The pointer to the driver's private data.
|
||||||
|
@param[in] HostName Pointer to buffer containing hostname.
|
||||||
|
@param[out] IpAddress On output, pointer to buffer containing IPv6 address.
|
||||||
|
|
||||||
|
@retval EFI_SUCCESS Operation succeeded.
|
||||||
|
@retval EFI_DEVICE_ERROR An unexpected network error occurred.
|
||||||
|
@retval Others Other errors as indicated.
|
||||||
|
**/
|
||||||
|
EFI_STATUS
|
||||||
|
HttpBootDns (
|
||||||
|
IN HTTP_BOOT_PRIVATE_DATA *Private,
|
||||||
|
IN CHAR16 *HostName,
|
||||||
|
OUT EFI_IPv6_ADDRESS *IpAddress
|
||||||
|
)
|
||||||
|
{
|
||||||
|
EFI_STATUS Status;
|
||||||
|
EFI_DNS6_PROTOCOL *Dns6;
|
||||||
|
EFI_DNS6_CONFIG_DATA Dns6ConfigData;
|
||||||
|
EFI_DNS6_COMPLETION_TOKEN Token;
|
||||||
|
EFI_HANDLE Dns6Handle;
|
||||||
|
EFI_IP6_CONFIG_PROTOCOL *Ip6Config;
|
||||||
|
EFI_IPv6_ADDRESS *DnsServerList;
|
||||||
|
UINTN DnsServerListCount;
|
||||||
|
UINTN DataSize;
|
||||||
|
BOOLEAN IsDone;
|
||||||
|
|
||||||
|
DnsServerList = NULL;
|
||||||
|
DnsServerListCount = 0;
|
||||||
|
Dns6 = NULL;
|
||||||
|
Dns6Handle = NULL;
|
||||||
|
ZeroMem (&Token, sizeof (EFI_DNS6_COMPLETION_TOKEN));
|
||||||
|
|
||||||
|
//
|
||||||
|
// Get DNS server list from EFI IPv6 Configuration protocol.
|
||||||
|
//
|
||||||
|
Status = gBS->HandleProtocol (Private->Controller, &gEfiIp6ConfigProtocolGuid, (VOID **) &Ip6Config);
|
||||||
|
if (!EFI_ERROR (Status)) {
|
||||||
|
//
|
||||||
|
// Get the required size.
|
||||||
|
//
|
||||||
|
DataSize = 0;
|
||||||
|
Status = Ip6Config->GetData (Ip6Config, Ip6ConfigDataTypeDnsServer, &DataSize, NULL);
|
||||||
|
if (Status == EFI_BUFFER_TOO_SMALL) {
|
||||||
|
DnsServerList = AllocatePool (DataSize);
|
||||||
|
if (DnsServerList == NULL) {
|
||||||
|
return EFI_OUT_OF_RESOURCES;
|
||||||
|
}
|
||||||
|
|
||||||
|
Status = Ip6Config->GetData (Ip6Config, Ip6ConfigDataTypeDnsServer, &DataSize, DnsServerList);
|
||||||
|
if (EFI_ERROR (Status)) {
|
||||||
|
FreePool (DnsServerList);
|
||||||
|
DnsServerList = NULL;
|
||||||
|
} else {
|
||||||
|
DnsServerListCount = DataSize / sizeof (EFI_IPv6_ADDRESS);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
//
|
||||||
|
// Create a DNSv6 child instance and get the protocol.
|
||||||
|
//
|
||||||
|
Status = NetLibCreateServiceChild (
|
||||||
|
Private->Controller,
|
||||||
|
Private->Image,
|
||||||
|
&gEfiDns6ServiceBindingProtocolGuid,
|
||||||
|
&Dns6Handle
|
||||||
|
);
|
||||||
|
if (EFI_ERROR (Status)) {
|
||||||
|
goto Exit;
|
||||||
|
}
|
||||||
|
|
||||||
|
Status = gBS->OpenProtocol (
|
||||||
|
Dns6Handle,
|
||||||
|
&gEfiDns6ProtocolGuid,
|
||||||
|
(VOID **) &Dns6,
|
||||||
|
Private->Image,
|
||||||
|
Private->Controller,
|
||||||
|
EFI_OPEN_PROTOCOL_BY_DRIVER
|
||||||
|
);
|
||||||
|
if (EFI_ERROR (Status)) {
|
||||||
|
goto Exit;
|
||||||
|
}
|
||||||
|
|
||||||
|
//
|
||||||
|
// Configure DNS6 instance for the DNS server address and protocol.
|
||||||
|
//
|
||||||
|
ZeroMem (&Dns6ConfigData, sizeof (EFI_DNS6_CONFIG_DATA));
|
||||||
|
Dns6ConfigData.DnsServerCount = (UINT32)DnsServerListCount;
|
||||||
|
Dns6ConfigData.DnsServerList = DnsServerList;
|
||||||
|
Dns6ConfigData.EnableDnsCache = TRUE;
|
||||||
|
Dns6ConfigData.Protocol = EFI_IP_PROTO_UDP;
|
||||||
|
IP6_COPY_ADDRESS (&Dns6ConfigData.StationIp,&Private->StationIp.v6);
|
||||||
|
Status = Dns6->Configure (
|
||||||
|
Dns6,
|
||||||
|
&Dns6ConfigData
|
||||||
|
);
|
||||||
|
if (EFI_ERROR (Status)) {
|
||||||
|
goto Exit;
|
||||||
|
}
|
||||||
|
|
||||||
|
Token.Status = EFI_NOT_READY;
|
||||||
|
IsDone = FALSE;
|
||||||
|
//
|
||||||
|
// Create event to set the IsDone flag when name resolution is finished.
|
||||||
|
//
|
||||||
|
Status = gBS->CreateEvent (
|
||||||
|
EVT_NOTIFY_SIGNAL,
|
||||||
|
TPL_NOTIFY,
|
||||||
|
HttpBootCommonNotify,
|
||||||
|
&IsDone,
|
||||||
|
&Token.Event
|
||||||
|
);
|
||||||
|
if (EFI_ERROR (Status)) {
|
||||||
|
goto Exit;
|
||||||
|
}
|
||||||
|
|
||||||
|
//
|
||||||
|
// Start asynchronous name resolution.
|
||||||
|
//
|
||||||
|
Status = Dns6->HostNameToIp (Dns6, HostName, &Token);
|
||||||
|
if (EFI_ERROR (Status)) {
|
||||||
|
goto Exit;
|
||||||
|
}
|
||||||
|
|
||||||
|
while (!IsDone) {
|
||||||
|
Dns6->Poll (Dns6);
|
||||||
|
}
|
||||||
|
|
||||||
|
//
|
||||||
|
// Name resolution is done, check result.
|
||||||
|
//
|
||||||
|
Status = Token.Status;
|
||||||
|
if (!EFI_ERROR (Status)) {
|
||||||
|
if (Token.RspData.H2AData == NULL) {
|
||||||
|
Status = EFI_DEVICE_ERROR;
|
||||||
|
goto Exit;
|
||||||
|
}
|
||||||
|
if (Token.RspData.H2AData->IpCount == 0 || Token.RspData.H2AData->IpList == NULL) {
|
||||||
|
Status = EFI_DEVICE_ERROR;
|
||||||
|
goto Exit;
|
||||||
|
}
|
||||||
|
//
|
||||||
|
// We just return the first IPv6 address from DNS protocol.
|
||||||
|
//
|
||||||
|
IP6_COPY_ADDRESS (IpAddress, Token.RspData.H2AData->IpList);
|
||||||
|
Status = EFI_SUCCESS;
|
||||||
|
}
|
||||||
|
Exit:
|
||||||
|
|
||||||
|
if (Token.Event != NULL) {
|
||||||
|
gBS->CloseEvent (Token.Event);
|
||||||
|
}
|
||||||
|
if (Token.RspData.H2AData != NULL) {
|
||||||
|
if (Token.RspData.H2AData->IpList != NULL) {
|
||||||
|
FreePool (Token.RspData.H2AData->IpList);
|
||||||
|
}
|
||||||
|
FreePool (Token.RspData.H2AData);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (Dns6 != NULL) {
|
||||||
|
Dns6->Configure (Dns6, NULL);
|
||||||
|
|
||||||
|
gBS->CloseProtocol (
|
||||||
|
Dns6Handle,
|
||||||
|
&gEfiDns6ProtocolGuid,
|
||||||
|
Private->Image,
|
||||||
|
Private->Controller
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (Dns6Handle != NULL) {
|
||||||
|
NetLibDestroyServiceChild (
|
||||||
|
Private->Controller,
|
||||||
|
Private->Image,
|
||||||
|
&gEfiDns6ServiceBindingProtocolGuid,
|
||||||
|
Dns6Handle
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (DnsServerList != NULL) {
|
||||||
|
FreePool (DnsServerList);
|
||||||
|
}
|
||||||
|
|
||||||
|
return Status;
|
||||||
|
}
|
||||||
/**
|
/**
|
||||||
Create a HTTP_IO_HEADER to hold the HTTP header items.
|
Create a HTTP_IO_HEADER to hold the HTTP header items.
|
||||||
|
|
||||||
@ -100,7 +361,7 @@ HttpBootShowIp4Addr (
|
|||||||
HTTP_IO_HEADER *
|
HTTP_IO_HEADER *
|
||||||
HttpBootCreateHeader (
|
HttpBootCreateHeader (
|
||||||
UINTN MaxHeaderCount
|
UINTN MaxHeaderCount
|
||||||
)
|
)
|
||||||
{
|
{
|
||||||
HTTP_IO_HEADER *HttpIoHeader;
|
HTTP_IO_HEADER *HttpIoHeader;
|
||||||
|
|
||||||
@ -254,23 +515,6 @@ HttpBootSetHeader (
|
|||||||
return EFI_SUCCESS;
|
return EFI_SUCCESS;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
Notify the callback function when an event is triggered.
|
|
||||||
|
|
||||||
@param[in] Event The triggered event.
|
|
||||||
@param[in] Context The opaque parameter to the function.
|
|
||||||
|
|
||||||
**/
|
|
||||||
VOID
|
|
||||||
EFIAPI
|
|
||||||
HttpIoCommonNotify (
|
|
||||||
IN EFI_EVENT Event,
|
|
||||||
IN VOID *Context
|
|
||||||
)
|
|
||||||
{
|
|
||||||
*((BOOLEAN *) Context) = TRUE;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
Create a HTTP_IO to access the HTTP service. It will create and configure
|
Create a HTTP_IO to access the HTTP service. It will create and configure
|
||||||
a HTTP child handle.
|
a HTTP child handle.
|
||||||
@ -301,6 +545,7 @@ HttpIoCreateIo (
|
|||||||
EFI_STATUS Status;
|
EFI_STATUS Status;
|
||||||
EFI_HTTP_CONFIG_DATA HttpConfigData;
|
EFI_HTTP_CONFIG_DATA HttpConfigData;
|
||||||
EFI_HTTPv4_ACCESS_POINT Http4AccessPoint;
|
EFI_HTTPv4_ACCESS_POINT Http4AccessPoint;
|
||||||
|
EFI_HTTPv6_ACCESS_POINT Http6AccessPoint;
|
||||||
EFI_HTTP_PROTOCOL *Http;
|
EFI_HTTP_PROTOCOL *Http;
|
||||||
EFI_EVENT Event;
|
EFI_EVENT Event;
|
||||||
|
|
||||||
@ -359,7 +604,10 @@ HttpIoCreateIo (
|
|||||||
IP4_COPY_ADDRESS (&Http4AccessPoint.LocalSubnet, &ConfigData->Config4.SubnetMask);
|
IP4_COPY_ADDRESS (&Http4AccessPoint.LocalSubnet, &ConfigData->Config4.SubnetMask);
|
||||||
HttpConfigData.AccessPoint.IPv4Node = &Http4AccessPoint;
|
HttpConfigData.AccessPoint.IPv4Node = &Http4AccessPoint;
|
||||||
} else {
|
} else {
|
||||||
ASSERT (FALSE);
|
HttpConfigData.LocalAddressIsIPv6 = TRUE;
|
||||||
|
Http6AccessPoint.LocalPort = ConfigData->Config6.LocalPort;
|
||||||
|
IP6_COPY_ADDRESS (&Http6AccessPoint.LocalAddress, &ConfigData->Config6.LocalIp);
|
||||||
|
HttpConfigData.AccessPoint.IPv6Node = &Http6AccessPoint;
|
||||||
}
|
}
|
||||||
|
|
||||||
Status = Http->Configure (Http, &HttpConfigData);
|
Status = Http->Configure (Http, &HttpConfigData);
|
||||||
@ -373,7 +621,7 @@ HttpIoCreateIo (
|
|||||||
Status = gBS->CreateEvent (
|
Status = gBS->CreateEvent (
|
||||||
EVT_NOTIFY_SIGNAL,
|
EVT_NOTIFY_SIGNAL,
|
||||||
TPL_NOTIFY,
|
TPL_NOTIFY,
|
||||||
HttpIoCommonNotify,
|
HttpBootCommonNotify,
|
||||||
&HttpIo->IsTxDone,
|
&HttpIo->IsTxDone,
|
||||||
&Event
|
&Event
|
||||||
);
|
);
|
||||||
@ -386,7 +634,7 @@ HttpIoCreateIo (
|
|||||||
Status = gBS->CreateEvent (
|
Status = gBS->CreateEvent (
|
||||||
EVT_NOTIFY_SIGNAL,
|
EVT_NOTIFY_SIGNAL,
|
||||||
TPL_NOTIFY,
|
TPL_NOTIFY,
|
||||||
HttpIoCommonNotify,
|
HttpBootCommonNotify,
|
||||||
&HttpIo->IsRxDone,
|
&HttpIo->IsRxDone,
|
||||||
&Event
|
&Event
|
||||||
);
|
);
|
||||||
@ -579,7 +827,7 @@ HttpIoRecvResponse (
|
|||||||
//
|
//
|
||||||
// Store the received data into the wrapper.
|
// Store the received data into the wrapper.
|
||||||
//
|
//
|
||||||
Status = HttpIo->ReqToken.Status;
|
Status = HttpIo->RspToken.Status;
|
||||||
if (!EFI_ERROR (Status)) {
|
if (!EFI_ERROR (Status)) {
|
||||||
ResponseData->HeaderCount = HttpIo->RspToken.Message->HeaderCount;
|
ResponseData->HeaderCount = HttpIo->RspToken.Message->HeaderCount;
|
||||||
ResponseData->Headers = HttpIo->RspToken.Message->Headers;
|
ResponseData->Headers = HttpIo->RspToken.Message->Headers;
|
||||||
|
@ -29,6 +29,20 @@ HttpBootGetNicByIp4Children (
|
|||||||
IN EFI_HANDLE ControllerHandle
|
IN EFI_HANDLE ControllerHandle
|
||||||
);
|
);
|
||||||
|
|
||||||
|
/**
|
||||||
|
Get the Nic handle using any child handle in the IPv6 stack.
|
||||||
|
|
||||||
|
@param[in] ControllerHandle Pointer to child handle over IPv6.
|
||||||
|
|
||||||
|
@return NicHandle The pointer to the Nic handle.
|
||||||
|
@return NULL Can't find the Nic handle.
|
||||||
|
|
||||||
|
**/
|
||||||
|
EFI_HANDLE
|
||||||
|
HttpBootGetNicByIp6Children (
|
||||||
|
IN EFI_HANDLE ControllerHandle
|
||||||
|
);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
This function is to convert UINTN to ASCII string with the required formatting.
|
This function is to convert UINTN to ASCII string with the required formatting.
|
||||||
|
|
||||||
@ -56,6 +70,17 @@ HttpBootShowIp4Addr (
|
|||||||
IN EFI_IPv4_ADDRESS *Ip
|
IN EFI_IPv4_ADDRESS *Ip
|
||||||
);
|
);
|
||||||
|
|
||||||
|
/**
|
||||||
|
This function is to display the IPv6 address.
|
||||||
|
|
||||||
|
@param[in] Ip The pointer to the IPv6 address.
|
||||||
|
|
||||||
|
**/
|
||||||
|
VOID
|
||||||
|
HttpBootShowIp6Addr (
|
||||||
|
IN EFI_IPv6_ADDRESS *Ip
|
||||||
|
);
|
||||||
|
|
||||||
//
|
//
|
||||||
// A wrapper structure to hold the HTTP headers.
|
// A wrapper structure to hold the HTTP headers.
|
||||||
//
|
//
|
||||||
@ -122,11 +147,24 @@ typedef struct {
|
|||||||
UINT16 LocalPort;
|
UINT16 LocalPort;
|
||||||
} HTTP4_IO_CONFIG_DATA;
|
} HTTP4_IO_CONFIG_DATA;
|
||||||
|
|
||||||
|
//
|
||||||
|
// HTTP_IO configuration data for IPv6
|
||||||
|
//
|
||||||
|
typedef struct {
|
||||||
|
EFI_HTTP_VERSION HttpVersion;
|
||||||
|
UINT32 RequestTimeOut; // In milliseconds.
|
||||||
|
BOOLEAN UseDefaultAddress;
|
||||||
|
EFI_IPv6_ADDRESS LocalIp;
|
||||||
|
UINT16 LocalPort;
|
||||||
|
} HTTP6_IO_CONFIG_DATA;
|
||||||
|
|
||||||
|
|
||||||
//
|
//
|
||||||
// HTTP_IO configuration
|
// HTTP_IO configuration
|
||||||
//
|
//
|
||||||
typedef union {
|
typedef union {
|
||||||
HTTP4_IO_CONFIG_DATA Config4;
|
HTTP4_IO_CONFIG_DATA Config4;
|
||||||
|
HTTP6_IO_CONFIG_DATA Config6;
|
||||||
} HTTP_IO_CONFIG_DATA;
|
} HTTP_IO_CONFIG_DATA;
|
||||||
|
|
||||||
//
|
//
|
||||||
@ -160,6 +198,38 @@ typedef struct {
|
|||||||
CHAR8 *Body;
|
CHAR8 *Body;
|
||||||
} HTTP_IO_RESOPNSE_DATA;
|
} HTTP_IO_RESOPNSE_DATA;
|
||||||
|
|
||||||
|
/**
|
||||||
|
Retrieve the host address using the EFI_DNS6_PROTOCOL.
|
||||||
|
|
||||||
|
@param[in] Private The pointer to the driver's private data.
|
||||||
|
@param[in] HostName Pointer to buffer containing hostname.
|
||||||
|
@param[out] IpAddress On output, pointer to buffer containing IPv6 address.
|
||||||
|
|
||||||
|
@retval EFI_SUCCESS Operation succeeded.
|
||||||
|
@retval EFI_DEVICE_ERROR An unexpected network error occurred.
|
||||||
|
@retval Others Other errors as indicated.
|
||||||
|
**/
|
||||||
|
EFI_STATUS
|
||||||
|
HttpBootDns (
|
||||||
|
IN HTTP_BOOT_PRIVATE_DATA *Private,
|
||||||
|
IN CHAR16 *HostName,
|
||||||
|
OUT EFI_IPv6_ADDRESS *IpAddress
|
||||||
|
);
|
||||||
|
|
||||||
|
/**
|
||||||
|
Notify the callback function when an event is triggered.
|
||||||
|
|
||||||
|
@param[in] Event The triggered event.
|
||||||
|
@param[in] Context The opaque parameter to the function.
|
||||||
|
|
||||||
|
**/
|
||||||
|
VOID
|
||||||
|
EFIAPI
|
||||||
|
HttpBootCommonNotify (
|
||||||
|
IN EFI_EVENT Event,
|
||||||
|
IN VOID *Context
|
||||||
|
);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
Create a HTTP_IO to access the HTTP service. It will create and configure
|
Create a HTTP_IO to access the HTTP service. It will create and configure
|
||||||
a HTTP child handle.
|
a HTTP child handle.
|
||||||
|
@ -194,11 +194,11 @@ Exit:
|
|||||||
Dns4->Configure (Dns4, NULL);
|
Dns4->Configure (Dns4, NULL);
|
||||||
|
|
||||||
gBS->CloseProtocol (
|
gBS->CloseProtocol (
|
||||||
Dns4Handle,
|
Dns4Handle,
|
||||||
&gEfiDns4ProtocolGuid,
|
&gEfiDns4ProtocolGuid,
|
||||||
Service->ImageHandle,
|
Service->ImageHandle,
|
||||||
Service->ControllerHandle
|
Service->ControllerHandle
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (Dns4Handle != NULL) {
|
if (Dns4Handle != NULL) {
|
||||||
@ -216,3 +216,200 @@ Exit:
|
|||||||
|
|
||||||
return Status;
|
return Status;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
Retrieve the host address using the EFI_DNS6_PROTOCOL.
|
||||||
|
|
||||||
|
@param[in] HttpInstance Pointer to HTTP_PROTOCOL instance.
|
||||||
|
@param[in] HostName Pointer to buffer containing hostname.
|
||||||
|
@param[out] IpAddress On output, pointer to buffer containing IPv6 address.
|
||||||
|
|
||||||
|
@retval EFI_SUCCESS Operation succeeded.
|
||||||
|
@retval EFI_OUT_OF_RESOURCES Failed to allocate needed resources.
|
||||||
|
@retval EFI_DEVICE_ERROR An unexpected network error occurred.
|
||||||
|
@retval Others Other errors as indicated.
|
||||||
|
|
||||||
|
**/
|
||||||
|
EFI_STATUS
|
||||||
|
HttpDns6 (
|
||||||
|
IN HTTP_PROTOCOL *HttpInstance,
|
||||||
|
IN CHAR16 *HostName,
|
||||||
|
OUT EFI_IPv6_ADDRESS *IpAddress
|
||||||
|
)
|
||||||
|
{
|
||||||
|
EFI_STATUS Status;
|
||||||
|
HTTP_SERVICE *Service;
|
||||||
|
EFI_DNS6_PROTOCOL *Dns6;
|
||||||
|
EFI_DNS6_CONFIG_DATA Dns6ConfigData;
|
||||||
|
EFI_DNS6_COMPLETION_TOKEN Token;
|
||||||
|
EFI_HANDLE Dns6Handle;
|
||||||
|
EFI_IP6_CONFIG_PROTOCOL *Ip6Config;
|
||||||
|
EFI_IPv6_ADDRESS *DnsServerList;
|
||||||
|
UINTN DnsServerListCount;
|
||||||
|
UINTN DataSize;
|
||||||
|
BOOLEAN IsDone;
|
||||||
|
|
||||||
|
|
||||||
|
Service = HttpInstance->Service;
|
||||||
|
ASSERT (Service != NULL);
|
||||||
|
|
||||||
|
DnsServerList = NULL;
|
||||||
|
DnsServerListCount = 0;
|
||||||
|
Dns6 = NULL;
|
||||||
|
Dns6Handle = NULL;
|
||||||
|
ZeroMem (&Token, sizeof (EFI_DNS6_COMPLETION_TOKEN));
|
||||||
|
|
||||||
|
//
|
||||||
|
// Get DNS server list from EFI IPv6 Configuration protocol.
|
||||||
|
//
|
||||||
|
Status = gBS->HandleProtocol (Service->ControllerHandle, &gEfiIp6ConfigProtocolGuid, (VOID **) &Ip6Config);
|
||||||
|
if (!EFI_ERROR (Status)) {
|
||||||
|
//
|
||||||
|
// Get the required size.
|
||||||
|
//
|
||||||
|
DataSize = 0;
|
||||||
|
Status = Ip6Config->GetData (Ip6Config, Ip6ConfigDataTypeDnsServer, &DataSize, NULL);
|
||||||
|
if (Status == EFI_BUFFER_TOO_SMALL) {
|
||||||
|
DnsServerList = AllocatePool (DataSize);
|
||||||
|
if (DnsServerList == NULL) {
|
||||||
|
return EFI_OUT_OF_RESOURCES;
|
||||||
|
}
|
||||||
|
|
||||||
|
Status = Ip6Config->GetData (Ip6Config, Ip6ConfigDataTypeDnsServer, &DataSize, DnsServerList);
|
||||||
|
if (EFI_ERROR (Status)) {
|
||||||
|
FreePool (DnsServerList);
|
||||||
|
DnsServerList = NULL;
|
||||||
|
} else {
|
||||||
|
DnsServerListCount = DataSize / sizeof (EFI_IPv6_ADDRESS);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
//
|
||||||
|
// Create a DNSv6 child instance and get the protocol.
|
||||||
|
//
|
||||||
|
Status = NetLibCreateServiceChild (
|
||||||
|
Service->ControllerHandle,
|
||||||
|
Service->ImageHandle,
|
||||||
|
&gEfiDns6ServiceBindingProtocolGuid,
|
||||||
|
&Dns6Handle
|
||||||
|
);
|
||||||
|
if (EFI_ERROR (Status)) {
|
||||||
|
goto Exit;
|
||||||
|
}
|
||||||
|
|
||||||
|
Status = gBS->OpenProtocol (
|
||||||
|
Dns6Handle,
|
||||||
|
&gEfiDns6ProtocolGuid,
|
||||||
|
(VOID **) &Dns6,
|
||||||
|
Service->ImageHandle,
|
||||||
|
Service->ControllerHandle,
|
||||||
|
EFI_OPEN_PROTOCOL_BY_DRIVER
|
||||||
|
);
|
||||||
|
if (EFI_ERROR (Status)) {
|
||||||
|
goto Exit;
|
||||||
|
}
|
||||||
|
|
||||||
|
//
|
||||||
|
// Configure DNS6 instance for the DNS server address and protocol.
|
||||||
|
//
|
||||||
|
ZeroMem (&Dns6ConfigData, sizeof (EFI_DNS6_CONFIG_DATA));
|
||||||
|
Dns6ConfigData.DnsServerCount = (UINT32)DnsServerListCount;
|
||||||
|
Dns6ConfigData.DnsServerList = DnsServerList;
|
||||||
|
Dns6ConfigData.EnableDnsCache = TRUE;
|
||||||
|
Dns6ConfigData.Protocol = EFI_IP_PROTO_UDP;
|
||||||
|
IP6_COPY_ADDRESS (&Dns6ConfigData.StationIp, &HttpInstance->Ipv6Node.LocalAddress);
|
||||||
|
Status = Dns6->Configure (
|
||||||
|
Dns6,
|
||||||
|
&Dns6ConfigData
|
||||||
|
);
|
||||||
|
if (EFI_ERROR (Status)) {
|
||||||
|
goto Exit;
|
||||||
|
}
|
||||||
|
|
||||||
|
Token.Status = EFI_NOT_READY;
|
||||||
|
IsDone = FALSE;
|
||||||
|
//
|
||||||
|
// Create event to set the IsDone flag when name resolution is finished.
|
||||||
|
//
|
||||||
|
Status = gBS->CreateEvent (
|
||||||
|
EVT_NOTIFY_SIGNAL,
|
||||||
|
TPL_NOTIFY,
|
||||||
|
HttpCommonNotify,
|
||||||
|
&IsDone,
|
||||||
|
&Token.Event
|
||||||
|
);
|
||||||
|
if (EFI_ERROR (Status)) {
|
||||||
|
goto Exit;
|
||||||
|
}
|
||||||
|
|
||||||
|
//
|
||||||
|
// Start asynchronous name resolution.
|
||||||
|
//
|
||||||
|
Status = Dns6->HostNameToIp (Dns6, HostName, &Token);
|
||||||
|
if (EFI_ERROR (Status)) {
|
||||||
|
goto Exit;
|
||||||
|
}
|
||||||
|
|
||||||
|
while (!IsDone) {
|
||||||
|
Dns6->Poll (Dns6);
|
||||||
|
}
|
||||||
|
|
||||||
|
//
|
||||||
|
// Name resolution is done, check result.
|
||||||
|
//
|
||||||
|
Status = Token.Status;
|
||||||
|
if (!EFI_ERROR (Status)) {
|
||||||
|
if (Token.RspData.H2AData == NULL) {
|
||||||
|
Status = EFI_DEVICE_ERROR;
|
||||||
|
goto Exit;
|
||||||
|
}
|
||||||
|
if (Token.RspData.H2AData->IpCount == 0 || Token.RspData.H2AData->IpList == NULL) {
|
||||||
|
Status = EFI_DEVICE_ERROR;
|
||||||
|
goto Exit;
|
||||||
|
}
|
||||||
|
//
|
||||||
|
// We just return the first IPv6 address from DNS protocol.
|
||||||
|
//
|
||||||
|
IP6_COPY_ADDRESS (IpAddress, Token.RspData.H2AData->IpList);
|
||||||
|
Status = EFI_SUCCESS;
|
||||||
|
}
|
||||||
|
|
||||||
|
Exit:
|
||||||
|
|
||||||
|
if (Token.Event != NULL) {
|
||||||
|
gBS->CloseEvent (Token.Event);
|
||||||
|
}
|
||||||
|
if (Token.RspData.H2AData != NULL) {
|
||||||
|
if (Token.RspData.H2AData->IpList != NULL) {
|
||||||
|
FreePool (Token.RspData.H2AData->IpList);
|
||||||
|
}
|
||||||
|
FreePool (Token.RspData.H2AData);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (Dns6 != NULL) {
|
||||||
|
Dns6->Configure (Dns6, NULL);
|
||||||
|
|
||||||
|
gBS->CloseProtocol (
|
||||||
|
Dns6Handle,
|
||||||
|
&gEfiDns6ProtocolGuid,
|
||||||
|
Service->ImageHandle,
|
||||||
|
Service->ControllerHandle
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (Dns6Handle != NULL) {
|
||||||
|
NetLibDestroyServiceChild (
|
||||||
|
Service->ControllerHandle,
|
||||||
|
Service->ImageHandle,
|
||||||
|
&gEfiDns6ServiceBindingProtocolGuid,
|
||||||
|
Dns6Handle
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (DnsServerList != NULL) {
|
||||||
|
FreePool (DnsServerList);
|
||||||
|
}
|
||||||
|
|
||||||
|
return Status;
|
||||||
|
}
|
||||||
|
@ -35,4 +35,24 @@ HttpDns4 (
|
|||||||
OUT EFI_IPv4_ADDRESS *IpAddress
|
OUT EFI_IPv4_ADDRESS *IpAddress
|
||||||
);
|
);
|
||||||
|
|
||||||
|
/**
|
||||||
|
Retrieve the host address using the EFI_DNS6_PROTOCOL.
|
||||||
|
|
||||||
|
@param[in] HttpInstance Pointer to HTTP_PROTOCOL instance.
|
||||||
|
@param[in] HostName Pointer to buffer containing hostname.
|
||||||
|
@param[out] IpAddress On output, pointer to buffer containing IPv6 address.
|
||||||
|
|
||||||
|
@retval EFI_SUCCESS Operation succeeded.
|
||||||
|
@retval EFI_OUT_OF_RESOURCES Failed to allocate needed resources.
|
||||||
|
@retval EFI_DEVICE_ERROR An unexpected network error occurred.
|
||||||
|
@retval Others Other errors as indicated.
|
||||||
|
|
||||||
|
**/
|
||||||
|
EFI_STATUS
|
||||||
|
HttpDns6 (
|
||||||
|
IN HTTP_PROTOCOL *HttpInstance,
|
||||||
|
IN CHAR16 *HostName,
|
||||||
|
OUT EFI_IPv6_ADDRESS *IpAddress
|
||||||
|
);
|
||||||
|
|
||||||
#endif
|
#endif
|
@ -20,15 +20,25 @@ EFI_HTTP_UTILITIES_PROTOCOL *mHttpUtilities = NULL;
|
|||||||
///
|
///
|
||||||
/// Driver Binding Protocol instance
|
/// Driver Binding Protocol instance
|
||||||
///
|
///
|
||||||
EFI_DRIVER_BINDING_PROTOCOL gHttpDxeDriverBinding = {
|
EFI_DRIVER_BINDING_PROTOCOL gHttpDxeIp4DriverBinding = {
|
||||||
HttpDxeDriverBindingSupported,
|
HttpDxeIp4DriverBindingSupported,
|
||||||
HttpDxeDriverBindingStart,
|
HttpDxeIp4DriverBindingStart,
|
||||||
HttpDxeDriverBindingStop,
|
HttpDxeIp4DriverBindingStop,
|
||||||
HTTP_DRIVER_VERSION,
|
HTTP_DRIVER_VERSION,
|
||||||
NULL,
|
NULL,
|
||||||
NULL
|
NULL
|
||||||
};
|
};
|
||||||
|
|
||||||
|
EFI_DRIVER_BINDING_PROTOCOL gHttpDxeIp6DriverBinding = {
|
||||||
|
HttpDxeIp6DriverBindingSupported,
|
||||||
|
HttpDxeIp6DriverBindingStart,
|
||||||
|
HttpDxeIp6DriverBindingStop,
|
||||||
|
HTTP_DRIVER_VERSION,
|
||||||
|
NULL,
|
||||||
|
NULL
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
Create a HTTP driver service binding private instance.
|
Create a HTTP driver service binding private instance.
|
||||||
|
|
||||||
@ -73,33 +83,59 @@ HttpCreateService (
|
|||||||
/**
|
/**
|
||||||
Release all the resource used the HTTP service binding instance.
|
Release all the resource used the HTTP service binding instance.
|
||||||
|
|
||||||
@param HttpService The HTTP private instance.
|
@param[in] HttpService The HTTP private instance.
|
||||||
|
@param[in] UsingIpv6 Indicate use TCP4 protocol or TCP6 protocol.
|
||||||
|
if TRUE, use Tcp6 protocol.
|
||||||
|
if FALSE, use Tcp4 protocl.
|
||||||
**/
|
**/
|
||||||
VOID
|
VOID
|
||||||
HttpCleanService (
|
HttpCleanService (
|
||||||
IN HTTP_SERVICE *HttpService
|
IN HTTP_SERVICE *HttpService,
|
||||||
|
IN BOOLEAN UsingIpv6
|
||||||
)
|
)
|
||||||
{
|
{
|
||||||
|
|
||||||
if (HttpService == NULL) {
|
if (HttpService == NULL) {
|
||||||
return ;
|
return ;
|
||||||
}
|
}
|
||||||
|
if (!UsingIpv6) {
|
||||||
if (HttpService->TcpChildHandle != NULL) {
|
if (HttpService->Tcp4ChildHandle != NULL) {
|
||||||
gBS->CloseProtocol (
|
gBS->CloseProtocol (
|
||||||
HttpService->TcpChildHandle,
|
HttpService->Tcp4ChildHandle,
|
||||||
&gEfiTcp4ProtocolGuid,
|
&gEfiTcp4ProtocolGuid,
|
||||||
HttpService->ImageHandle,
|
HttpService->ImageHandle,
|
||||||
HttpService->ControllerHandle
|
HttpService->ControllerHandle
|
||||||
);
|
);
|
||||||
|
|
||||||
NetLibDestroyServiceChild (
|
NetLibDestroyServiceChild (
|
||||||
HttpService->ControllerHandle,
|
HttpService->ControllerHandle,
|
||||||
HttpService->ImageHandle,
|
HttpService->ImageHandle,
|
||||||
&gEfiTcp4ServiceBindingProtocolGuid,
|
&gEfiTcp4ServiceBindingProtocolGuid,
|
||||||
HttpService->TcpChildHandle
|
HttpService->Tcp4ChildHandle
|
||||||
);
|
);
|
||||||
|
|
||||||
|
HttpService->Tcp4ChildHandle = NULL;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if (HttpService->Tcp6ChildHandle != NULL) {
|
||||||
|
gBS->CloseProtocol (
|
||||||
|
HttpService->Tcp6ChildHandle,
|
||||||
|
&gEfiTcp6ProtocolGuid,
|
||||||
|
HttpService->ImageHandle,
|
||||||
|
HttpService->ControllerHandle
|
||||||
|
);
|
||||||
|
|
||||||
|
NetLibDestroyServiceChild (
|
||||||
|
HttpService->ControllerHandle,
|
||||||
|
HttpService->ImageHandle,
|
||||||
|
&gEfiTcp6ServiceBindingProtocolGuid,
|
||||||
|
HttpService->Tcp6ChildHandle
|
||||||
|
);
|
||||||
|
|
||||||
|
HttpService->Tcp6ChildHandle = NULL;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -107,7 +143,7 @@ HttpCleanService (
|
|||||||
in the system.
|
in the system.
|
||||||
|
|
||||||
@param[in] Event Not used.
|
@param[in] Event Not used.
|
||||||
@param[in] Context The pointer to the IP4 config2 instance data.
|
@param[in] Context The pointer to the IP4 config2 instance data or IP6 Config instance data.
|
||||||
|
|
||||||
**/
|
**/
|
||||||
VOID
|
VOID
|
||||||
@ -122,7 +158,7 @@ HttpUtilitiesInstalledCallback (
|
|||||||
NULL,
|
NULL,
|
||||||
(VOID **) &mHttpUtilities
|
(VOID **) &mHttpUtilities
|
||||||
);
|
);
|
||||||
|
|
||||||
//
|
//
|
||||||
// Close the event if Http utilities protocol is loacted.
|
// Close the event if Http utilities protocol is loacted.
|
||||||
//
|
//
|
||||||
@ -150,6 +186,7 @@ HttpDxeDriverEntryPoint (
|
|||||||
IN EFI_SYSTEM_TABLE *SystemTable
|
IN EFI_SYSTEM_TABLE *SystemTable
|
||||||
)
|
)
|
||||||
{
|
{
|
||||||
|
EFI_STATUS Status;
|
||||||
VOID *Registration;
|
VOID *Registration;
|
||||||
|
|
||||||
gBS->LocateProtocol (
|
gBS->LocateProtocol (
|
||||||
@ -174,14 +211,39 @@ HttpDxeDriverEntryPoint (
|
|||||||
//
|
//
|
||||||
// Install UEFI Driver Model protocol(s).
|
// Install UEFI Driver Model protocol(s).
|
||||||
//
|
//
|
||||||
return EfiLibInstallDriverBindingComponentName2 (
|
Status = EfiLibInstallDriverBindingComponentName2 (
|
||||||
ImageHandle,
|
ImageHandle,
|
||||||
SystemTable,
|
SystemTable,
|
||||||
&gHttpDxeDriverBinding,
|
&gHttpDxeIp4DriverBinding,
|
||||||
|
ImageHandle,
|
||||||
|
&gHttpDxeComponentName,
|
||||||
|
&gHttpDxeComponentName2
|
||||||
|
);
|
||||||
|
if (EFI_ERROR (Status)) {
|
||||||
|
return Status;
|
||||||
|
}
|
||||||
|
|
||||||
|
Status = EfiLibInstallDriverBindingComponentName2 (
|
||||||
|
ImageHandle,
|
||||||
|
SystemTable,
|
||||||
|
&gHttpDxeIp6DriverBinding,
|
||||||
|
NULL,
|
||||||
|
&gHttpDxeComponentName,
|
||||||
|
&gHttpDxeComponentName2
|
||||||
|
);
|
||||||
|
if (EFI_ERROR (Status)) {
|
||||||
|
gBS->UninstallMultipleProtocolInterfaces (
|
||||||
ImageHandle,
|
ImageHandle,
|
||||||
|
&gEfiDriverBindingProtocolGuid,
|
||||||
|
&gHttpDxeIp4DriverBinding,
|
||||||
|
&gEfiComponentName2ProtocolGuid,
|
||||||
|
&gHttpDxeComponentName2,
|
||||||
|
&gEfiComponentNameProtocolGuid,
|
||||||
&gHttpDxeComponentName,
|
&gHttpDxeComponentName,
|
||||||
&gHttpDxeComponentName2
|
NULL
|
||||||
);
|
);
|
||||||
|
}
|
||||||
|
return Status;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -223,6 +285,309 @@ HttpDestroyChildEntryInHandleBuffer (
|
|||||||
return ServiceBinding->DestroyChild (ServiceBinding, HttpInstance->Handle);
|
return ServiceBinding->DestroyChild (ServiceBinding, HttpInstance->Handle);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
Test to see if this driver supports ControllerHandle. This is the worker function for
|
||||||
|
HttpDxeIp4(6)DriverBindingSupported.
|
||||||
|
|
||||||
|
@param[in] This The pointer to the driver binding protocol.
|
||||||
|
@param[in] ControllerHandle The handle of device to be tested.
|
||||||
|
@param[in] RemainingDevicePath Optional parameter used to pick a specific child
|
||||||
|
device to be started.
|
||||||
|
@param[in] IpVersion IP_VERSION_4 or IP_VERSION_6.
|
||||||
|
|
||||||
|
@retval EFI_SUCCESS This driver supports this device.
|
||||||
|
@retval EFI_UNSUPPORTED This driver does not support this device.
|
||||||
|
|
||||||
|
**/
|
||||||
|
EFI_STATUS
|
||||||
|
EFIAPI
|
||||||
|
HttpDxeSupported (
|
||||||
|
IN EFI_DRIVER_BINDING_PROTOCOL *This,
|
||||||
|
IN EFI_HANDLE ControllerHandle,
|
||||||
|
IN EFI_DEVICE_PATH_PROTOCOL *RemainingDevicePath OPTIONAL,
|
||||||
|
IN UINT8 IpVersion
|
||||||
|
)
|
||||||
|
{
|
||||||
|
EFI_STATUS Status;
|
||||||
|
EFI_GUID *TcpServiceBindingProtocolGuid;
|
||||||
|
|
||||||
|
if (IpVersion == IP_VERSION_4) {
|
||||||
|
TcpServiceBindingProtocolGuid = &gEfiTcp4ServiceBindingProtocolGuid;
|
||||||
|
} else {
|
||||||
|
TcpServiceBindingProtocolGuid = &gEfiTcp6ServiceBindingProtocolGuid;
|
||||||
|
}
|
||||||
|
|
||||||
|
Status = gBS->OpenProtocol (
|
||||||
|
ControllerHandle,
|
||||||
|
TcpServiceBindingProtocolGuid,
|
||||||
|
NULL,
|
||||||
|
This->DriverBindingHandle,
|
||||||
|
ControllerHandle,
|
||||||
|
EFI_OPEN_PROTOCOL_TEST_PROTOCOL
|
||||||
|
);
|
||||||
|
|
||||||
|
if (EFI_ERROR (Status)) {
|
||||||
|
return EFI_UNSUPPORTED;
|
||||||
|
}
|
||||||
|
|
||||||
|
return EFI_SUCCESS;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
Start this driver on ControllerHandle. This is the worker function for
|
||||||
|
HttpDxeIp4(6)DriverBindingStart.
|
||||||
|
|
||||||
|
@param[in] This The pointer to the driver binding protocol.
|
||||||
|
@param[in] ControllerHandle The handle of device to be started.
|
||||||
|
@param[in] RemainingDevicePath Optional parameter used to pick a specific child
|
||||||
|
device to be started.
|
||||||
|
@param[in] IpVersion IP_VERSION_4 or IP_VERSION_6.
|
||||||
|
|
||||||
|
|
||||||
|
@retval EFI_SUCCESS This driver is installed to ControllerHandle.
|
||||||
|
@retval EFI_ALREADY_STARTED This driver is already running on ControllerHandle.
|
||||||
|
@retval other This driver does not support this device.
|
||||||
|
|
||||||
|
**/
|
||||||
|
EFI_STATUS
|
||||||
|
EFIAPI
|
||||||
|
HttpDxeStart (
|
||||||
|
IN EFI_DRIVER_BINDING_PROTOCOL *This,
|
||||||
|
IN EFI_HANDLE ControllerHandle,
|
||||||
|
IN EFI_DEVICE_PATH_PROTOCOL *RemainingDevicePath OPTIONAL,
|
||||||
|
IN UINT8 IpVersion
|
||||||
|
)
|
||||||
|
{
|
||||||
|
EFI_STATUS Status;
|
||||||
|
EFI_SERVICE_BINDING_PROTOCOL *ServiceBinding;
|
||||||
|
HTTP_SERVICE *HttpService;
|
||||||
|
VOID *Interface;
|
||||||
|
BOOLEAN UsingIpv6;
|
||||||
|
|
||||||
|
UsingIpv6 = FALSE;
|
||||||
|
|
||||||
|
//
|
||||||
|
// Test for the Http service binding protocol
|
||||||
|
//
|
||||||
|
Status = gBS->OpenProtocol (
|
||||||
|
ControllerHandle,
|
||||||
|
&gEfiHttpServiceBindingProtocolGuid,
|
||||||
|
(VOID **) &ServiceBinding,
|
||||||
|
This->DriverBindingHandle,
|
||||||
|
ControllerHandle,
|
||||||
|
EFI_OPEN_PROTOCOL_GET_PROTOCOL
|
||||||
|
);
|
||||||
|
|
||||||
|
if (!EFI_ERROR (Status)) {
|
||||||
|
HttpService = HTTP_SERVICE_FROM_PROTOCOL (ServiceBinding);
|
||||||
|
} else {
|
||||||
|
Status = HttpCreateService (ControllerHandle, This->DriverBindingHandle, &HttpService);
|
||||||
|
if (EFI_ERROR (Status)) {
|
||||||
|
return Status;
|
||||||
|
}
|
||||||
|
|
||||||
|
ASSERT (HttpService != NULL);
|
||||||
|
|
||||||
|
//
|
||||||
|
// Install the HttpServiceBinding Protocol onto Controller
|
||||||
|
//
|
||||||
|
Status = gBS->InstallMultipleProtocolInterfaces (
|
||||||
|
&ControllerHandle,
|
||||||
|
&gEfiHttpServiceBindingProtocolGuid,
|
||||||
|
&HttpService->ServiceBinding,
|
||||||
|
NULL
|
||||||
|
);
|
||||||
|
|
||||||
|
if (EFI_ERROR (Status)) {
|
||||||
|
goto ON_ERROR;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (IpVersion == IP_VERSION_4) {
|
||||||
|
|
||||||
|
if (HttpService->Tcp4ChildHandle == NULL) {
|
||||||
|
//
|
||||||
|
// Create a TCP4 child instance, but do not configure it. This will establish the parent-child relationship.
|
||||||
|
//
|
||||||
|
Status = NetLibCreateServiceChild (
|
||||||
|
ControllerHandle,
|
||||||
|
This->DriverBindingHandle,
|
||||||
|
&gEfiTcp4ServiceBindingProtocolGuid,
|
||||||
|
&HttpService->Tcp4ChildHandle
|
||||||
|
);
|
||||||
|
|
||||||
|
if (EFI_ERROR (Status)) {
|
||||||
|
goto ON_ERROR;
|
||||||
|
}
|
||||||
|
|
||||||
|
Status = gBS->OpenProtocol (
|
||||||
|
HttpService->Tcp4ChildHandle,
|
||||||
|
&gEfiTcp4ProtocolGuid,
|
||||||
|
&Interface,
|
||||||
|
This->DriverBindingHandle,
|
||||||
|
ControllerHandle,
|
||||||
|
EFI_OPEN_PROTOCOL_BY_DRIVER
|
||||||
|
);
|
||||||
|
|
||||||
|
if (EFI_ERROR (Status)) {
|
||||||
|
goto ON_ERROR;
|
||||||
|
}
|
||||||
|
|
||||||
|
} else {
|
||||||
|
return EFI_ALREADY_STARTED;
|
||||||
|
}
|
||||||
|
|
||||||
|
} else {
|
||||||
|
UsingIpv6 = TRUE;
|
||||||
|
|
||||||
|
if (HttpService->Tcp6ChildHandle == NULL) {
|
||||||
|
//
|
||||||
|
// Create a TCP6 child instance, but do not configure it. This will establish the parent-child relationship.
|
||||||
|
//
|
||||||
|
Status = NetLibCreateServiceChild (
|
||||||
|
ControllerHandle,
|
||||||
|
This->DriverBindingHandle,
|
||||||
|
&gEfiTcp6ServiceBindingProtocolGuid,
|
||||||
|
&HttpService->Tcp6ChildHandle
|
||||||
|
);
|
||||||
|
|
||||||
|
if (EFI_ERROR (Status)) {
|
||||||
|
goto ON_ERROR;
|
||||||
|
}
|
||||||
|
|
||||||
|
Status = gBS->OpenProtocol (
|
||||||
|
HttpService->Tcp6ChildHandle,
|
||||||
|
&gEfiTcp6ProtocolGuid,
|
||||||
|
&Interface,
|
||||||
|
This->DriverBindingHandle,
|
||||||
|
ControllerHandle,
|
||||||
|
EFI_OPEN_PROTOCOL_BY_DRIVER
|
||||||
|
);
|
||||||
|
|
||||||
|
if (EFI_ERROR (Status)) {
|
||||||
|
goto ON_ERROR;
|
||||||
|
}
|
||||||
|
|
||||||
|
} else {
|
||||||
|
return EFI_ALREADY_STARTED;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
return EFI_SUCCESS;
|
||||||
|
|
||||||
|
ON_ERROR:
|
||||||
|
|
||||||
|
if (HttpService != NULL) {
|
||||||
|
HttpCleanService (HttpService, UsingIpv6);
|
||||||
|
if (HttpService->Tcp4ChildHandle == NULL && HttpService->Tcp6ChildHandle == NULL) {
|
||||||
|
FreePool (HttpService);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return Status;
|
||||||
|
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
Stop this driver on ControllerHandle. This is the worker function for
|
||||||
|
HttpDxeIp4(6)DriverBindingStop.
|
||||||
|
|
||||||
|
@param[in] This Protocol instance pointer.
|
||||||
|
@param[in] ControllerHandle Handle of device to stop driver on.
|
||||||
|
@param[in] NumberOfChildren Number of Handles in ChildHandleBuffer. If number of
|
||||||
|
children is zero stop the entire bus driver.
|
||||||
|
@param[in] ChildHandleBuffer List of Child Handles to Stop.
|
||||||
|
@param[in] IpVersion IP_VERSION_4 or IP_VERSION_6.
|
||||||
|
|
||||||
|
@retval EFI_SUCCESS This driver was removed ControllerHandle.
|
||||||
|
@retval EFI_DEVICE_ERROR An unexpected system or network error occurred.
|
||||||
|
@retval Others This driver was not removed from this device
|
||||||
|
|
||||||
|
**/
|
||||||
|
EFI_STATUS
|
||||||
|
EFIAPI
|
||||||
|
HttpDxeStop (
|
||||||
|
IN EFI_DRIVER_BINDING_PROTOCOL *This,
|
||||||
|
IN EFI_HANDLE ControllerHandle,
|
||||||
|
IN UINTN NumberOfChildren,
|
||||||
|
IN EFI_HANDLE *ChildHandleBuffer,
|
||||||
|
IN UINT8 IpVersion
|
||||||
|
)
|
||||||
|
{
|
||||||
|
EFI_HANDLE NicHandle;
|
||||||
|
EFI_STATUS Status;
|
||||||
|
EFI_SERVICE_BINDING_PROTOCOL *ServiceBinding;
|
||||||
|
HTTP_SERVICE *HttpService;
|
||||||
|
LIST_ENTRY *List;
|
||||||
|
HTTP_DESTROY_CHILD_IN_HANDLE_BUF_CONTEXT Context;
|
||||||
|
BOOLEAN UsingIpv6;
|
||||||
|
|
||||||
|
//
|
||||||
|
// HTTP driver opens TCP4(6) child, So, Controller is a TCP4(6)
|
||||||
|
// child handle. Locate the Nic handle first. Then get the
|
||||||
|
// HTTP private data back.
|
||||||
|
//
|
||||||
|
if (IpVersion == IP_VERSION_4) {
|
||||||
|
UsingIpv6 = FALSE;
|
||||||
|
NicHandle = NetLibGetNicHandle (ControllerHandle, &gEfiTcp4ProtocolGuid);
|
||||||
|
} else {
|
||||||
|
UsingIpv6 = TRUE;
|
||||||
|
NicHandle = NetLibGetNicHandle (ControllerHandle, &gEfiTcp6ProtocolGuid);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (NicHandle == NULL) {
|
||||||
|
return EFI_SUCCESS;
|
||||||
|
}
|
||||||
|
|
||||||
|
Status = gBS->OpenProtocol (
|
||||||
|
NicHandle,
|
||||||
|
&gEfiHttpServiceBindingProtocolGuid,
|
||||||
|
(VOID **) &ServiceBinding,
|
||||||
|
This->DriverBindingHandle,
|
||||||
|
NicHandle,
|
||||||
|
EFI_OPEN_PROTOCOL_GET_PROTOCOL
|
||||||
|
);
|
||||||
|
|
||||||
|
if (!EFI_ERROR (Status)) {
|
||||||
|
|
||||||
|
HttpService = HTTP_SERVICE_FROM_PROTOCOL (ServiceBinding);
|
||||||
|
|
||||||
|
if (NumberOfChildren != 0) {
|
||||||
|
//
|
||||||
|
// Destroy the HTTP child instance in ChildHandleBuffer.
|
||||||
|
//
|
||||||
|
List = &HttpService->ChildrenList;
|
||||||
|
Context.ServiceBinding = ServiceBinding;
|
||||||
|
Context.NumberOfChildren = NumberOfChildren;
|
||||||
|
Context.ChildHandleBuffer = ChildHandleBuffer;
|
||||||
|
Status = NetDestroyLinkList (
|
||||||
|
List,
|
||||||
|
HttpDestroyChildEntryInHandleBuffer,
|
||||||
|
&Context,
|
||||||
|
NULL
|
||||||
|
);
|
||||||
|
} else {
|
||||||
|
|
||||||
|
HttpCleanService (HttpService, UsingIpv6);
|
||||||
|
|
||||||
|
if (HttpService->Tcp4ChildHandle == NULL && HttpService->Tcp6ChildHandle == NULL) {
|
||||||
|
gBS->UninstallProtocolInterface (
|
||||||
|
NicHandle,
|
||||||
|
&gEfiHttpServiceBindingProtocolGuid,
|
||||||
|
ServiceBinding
|
||||||
|
);
|
||||||
|
FreePool (HttpService);
|
||||||
|
}
|
||||||
|
Status = EFI_SUCCESS;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return Status;
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
Tests to see if this driver supports a given controller. If a child device is provided,
|
Tests to see if this driver supports a given controller. If a child device is provided,
|
||||||
it further tests to see if this driver supports creating a handle for the specified child device.
|
it further tests to see if this driver supports creating a handle for the specified child device.
|
||||||
@ -267,41 +632,18 @@ HttpDestroyChildEntryInHandleBuffer (
|
|||||||
**/
|
**/
|
||||||
EFI_STATUS
|
EFI_STATUS
|
||||||
EFIAPI
|
EFIAPI
|
||||||
HttpDxeDriverBindingSupported (
|
HttpDxeIp4DriverBindingSupported (
|
||||||
IN EFI_DRIVER_BINDING_PROTOCOL *This,
|
IN EFI_DRIVER_BINDING_PROTOCOL *This,
|
||||||
IN EFI_HANDLE ControllerHandle,
|
IN EFI_HANDLE ControllerHandle,
|
||||||
IN EFI_DEVICE_PATH_PROTOCOL *RemainingDevicePath OPTIONAL
|
IN EFI_DEVICE_PATH_PROTOCOL *RemainingDevicePath OPTIONAL
|
||||||
)
|
)
|
||||||
{
|
{
|
||||||
EFI_STATUS Status;
|
return HttpDxeSupported (
|
||||||
|
This,
|
||||||
//
|
ControllerHandle,
|
||||||
// Test for the HttpServiceBinding protocol.
|
RemainingDevicePath,
|
||||||
//
|
IP_VERSION_4
|
||||||
Status = gBS->OpenProtocol (
|
);
|
||||||
ControllerHandle,
|
|
||||||
&gEfiHttpServiceBindingProtocolGuid,
|
|
||||||
NULL,
|
|
||||||
This->DriverBindingHandle,
|
|
||||||
ControllerHandle,
|
|
||||||
EFI_OPEN_PROTOCOL_TEST_PROTOCOL
|
|
||||||
);
|
|
||||||
if (!EFI_ERROR (Status)) {
|
|
||||||
return EFI_ALREADY_STARTED;
|
|
||||||
}
|
|
||||||
|
|
||||||
//
|
|
||||||
// Test for the Tcp4 Protocol
|
|
||||||
//
|
|
||||||
return gBS->OpenProtocol (
|
|
||||||
ControllerHandle,
|
|
||||||
&gEfiTcp4ServiceBindingProtocolGuid,
|
|
||||||
NULL,
|
|
||||||
This->DriverBindingHandle,
|
|
||||||
ControllerHandle,
|
|
||||||
EFI_OPEN_PROTOCOL_TEST_PROTOCOL
|
|
||||||
);
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -342,90 +684,18 @@ HttpDxeDriverBindingSupported (
|
|||||||
**/
|
**/
|
||||||
EFI_STATUS
|
EFI_STATUS
|
||||||
EFIAPI
|
EFIAPI
|
||||||
HttpDxeDriverBindingStart (
|
HttpDxeIp4DriverBindingStart (
|
||||||
IN EFI_DRIVER_BINDING_PROTOCOL *This,
|
IN EFI_DRIVER_BINDING_PROTOCOL *This,
|
||||||
IN EFI_HANDLE ControllerHandle,
|
IN EFI_HANDLE ControllerHandle,
|
||||||
IN EFI_DEVICE_PATH_PROTOCOL *RemainingDevicePath OPTIONAL
|
IN EFI_DEVICE_PATH_PROTOCOL *RemainingDevicePath OPTIONAL
|
||||||
)
|
)
|
||||||
{
|
{
|
||||||
EFI_STATUS Status;
|
return HttpDxeStart (
|
||||||
HTTP_SERVICE *HttpService;
|
This,
|
||||||
VOID *Interface;
|
ControllerHandle,
|
||||||
|
RemainingDevicePath,
|
||||||
//
|
IP_VERSION_4
|
||||||
// Test for the Http service binding protocol
|
);
|
||||||
//
|
|
||||||
Status = gBS->OpenProtocol (
|
|
||||||
ControllerHandle,
|
|
||||||
&gEfiHttpServiceBindingProtocolGuid,
|
|
||||||
NULL,
|
|
||||||
This->DriverBindingHandle,
|
|
||||||
ControllerHandle,
|
|
||||||
EFI_OPEN_PROTOCOL_TEST_PROTOCOL
|
|
||||||
);
|
|
||||||
|
|
||||||
if (Status == EFI_SUCCESS) {
|
|
||||||
return EFI_ALREADY_STARTED;
|
|
||||||
}
|
|
||||||
|
|
||||||
Status = HttpCreateService (ControllerHandle, This->DriverBindingHandle, &HttpService);
|
|
||||||
if (EFI_ERROR (Status)) {
|
|
||||||
return Status;
|
|
||||||
}
|
|
||||||
|
|
||||||
ASSERT (HttpService != NULL);
|
|
||||||
|
|
||||||
//
|
|
||||||
// Create a TCP child instance, but do not configure it. This will establish the parent-child relationship.
|
|
||||||
//
|
|
||||||
Status = NetLibCreateServiceChild (
|
|
||||||
ControllerHandle,
|
|
||||||
This->DriverBindingHandle,
|
|
||||||
&gEfiTcp4ServiceBindingProtocolGuid,
|
|
||||||
&HttpService->TcpChildHandle
|
|
||||||
);
|
|
||||||
|
|
||||||
if (EFI_ERROR (Status)) {
|
|
||||||
goto ON_ERROR;
|
|
||||||
}
|
|
||||||
|
|
||||||
Status = gBS->OpenProtocol (
|
|
||||||
HttpService->TcpChildHandle,
|
|
||||||
&gEfiTcp4ProtocolGuid,
|
|
||||||
&Interface,
|
|
||||||
This->DriverBindingHandle,
|
|
||||||
ControllerHandle,
|
|
||||||
EFI_OPEN_PROTOCOL_BY_DRIVER
|
|
||||||
);
|
|
||||||
|
|
||||||
if (EFI_ERROR (Status)) {
|
|
||||||
goto ON_ERROR;
|
|
||||||
}
|
|
||||||
|
|
||||||
//
|
|
||||||
// Install the HttpServiceBinding Protocol onto Controller
|
|
||||||
//
|
|
||||||
Status = gBS->InstallMultipleProtocolInterfaces (
|
|
||||||
&ControllerHandle,
|
|
||||||
&gEfiHttpServiceBindingProtocolGuid,
|
|
||||||
&HttpService->ServiceBinding,
|
|
||||||
NULL
|
|
||||||
);
|
|
||||||
|
|
||||||
if (EFI_ERROR (Status)) {
|
|
||||||
goto ON_ERROR;
|
|
||||||
}
|
|
||||||
|
|
||||||
return EFI_SUCCESS;
|
|
||||||
|
|
||||||
ON_ERROR:
|
|
||||||
|
|
||||||
if (HttpService != NULL) {
|
|
||||||
HttpCleanService (HttpService);
|
|
||||||
FreePool (HttpService);
|
|
||||||
}
|
|
||||||
|
|
||||||
return Status;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -456,78 +726,176 @@ ON_ERROR:
|
|||||||
**/
|
**/
|
||||||
EFI_STATUS
|
EFI_STATUS
|
||||||
EFIAPI
|
EFIAPI
|
||||||
HttpDxeDriverBindingStop (
|
HttpDxeIp4DriverBindingStop (
|
||||||
IN EFI_DRIVER_BINDING_PROTOCOL *This,
|
IN EFI_DRIVER_BINDING_PROTOCOL *This,
|
||||||
IN EFI_HANDLE ControllerHandle,
|
IN EFI_HANDLE ControllerHandle,
|
||||||
IN UINTN NumberOfChildren,
|
IN UINTN NumberOfChildren,
|
||||||
IN EFI_HANDLE *ChildHandleBuffer OPTIONAL
|
IN EFI_HANDLE *ChildHandleBuffer OPTIONAL
|
||||||
)
|
)
|
||||||
{
|
{
|
||||||
EFI_HANDLE NicHandle;
|
return HttpDxeStop (
|
||||||
EFI_STATUS Status;
|
This,
|
||||||
EFI_SERVICE_BINDING_PROTOCOL *ServiceBinding;
|
ControllerHandle,
|
||||||
HTTP_SERVICE *HttpService;
|
NumberOfChildren,
|
||||||
LIST_ENTRY *List;
|
ChildHandleBuffer,
|
||||||
HTTP_DESTROY_CHILD_IN_HANDLE_BUF_CONTEXT Context;
|
IP_VERSION_4
|
||||||
|
|
||||||
//
|
|
||||||
// HTTP driver opens TCP child, So, Controller is a TCP
|
|
||||||
// child handle. Locate the Nic handle first. Then get the
|
|
||||||
// HTTP private data back.
|
|
||||||
//
|
|
||||||
NicHandle = NetLibGetNicHandle (ControllerHandle, &gEfiTcp4ProtocolGuid);
|
|
||||||
if (NicHandle == NULL) {
|
|
||||||
return EFI_SUCCESS;
|
|
||||||
}
|
|
||||||
|
|
||||||
Status = gBS->OpenProtocol (
|
|
||||||
NicHandle,
|
|
||||||
&gEfiHttpServiceBindingProtocolGuid,
|
|
||||||
(VOID **) &ServiceBinding,
|
|
||||||
This->DriverBindingHandle,
|
|
||||||
NicHandle,
|
|
||||||
EFI_OPEN_PROTOCOL_GET_PROTOCOL
|
|
||||||
);
|
|
||||||
|
|
||||||
if (EFI_ERROR (Status)) {
|
|
||||||
return EFI_DEVICE_ERROR;
|
|
||||||
}
|
|
||||||
|
|
||||||
HttpService = HTTP_SERVICE_FROM_PROTOCOL (ServiceBinding);
|
|
||||||
|
|
||||||
if (!IsListEmpty (&HttpService->ChildrenList)) {
|
|
||||||
//
|
|
||||||
// Destroy the HTTP child instance in ChildHandleBuffer.
|
|
||||||
//
|
|
||||||
List = &HttpService->ChildrenList;
|
|
||||||
Context.ServiceBinding = ServiceBinding;
|
|
||||||
Context.NumberOfChildren = NumberOfChildren;
|
|
||||||
Context.ChildHandleBuffer = ChildHandleBuffer;
|
|
||||||
Status = NetDestroyLinkList (
|
|
||||||
List,
|
|
||||||
HttpDestroyChildEntryInHandleBuffer,
|
|
||||||
&Context,
|
|
||||||
NULL
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (NumberOfChildren == 0 && IsListEmpty (&HttpService->ChildrenList)) {
|
|
||||||
gBS->UninstallProtocolInterface (
|
|
||||||
NicHandle,
|
|
||||||
&gEfiHttpServiceBindingProtocolGuid,
|
|
||||||
ServiceBinding
|
|
||||||
);
|
);
|
||||||
|
|
||||||
HttpCleanService (HttpService);
|
|
||||||
|
|
||||||
FreePool (HttpService);
|
|
||||||
|
|
||||||
Status = EFI_SUCCESS;
|
|
||||||
}
|
|
||||||
|
|
||||||
return Status;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
Tests to see if this driver supports a given controller. If a child device is provided,
|
||||||
|
it further tests to see if this driver supports creating a handle for the specified child device.
|
||||||
|
|
||||||
|
This function checks to see if the driver specified by This supports the device specified by
|
||||||
|
ControllerHandle. Drivers will typically use the device path attached to
|
||||||
|
ControllerHandle and/or the services from the bus I/O abstraction attached to
|
||||||
|
ControllerHandle to determine if the driver supports ControllerHandle. This function
|
||||||
|
may be called many times during platform initialization. In order to reduce boot times, the tests
|
||||||
|
performed by this function must be very small, and take as little time as possible to execute. This
|
||||||
|
function must not change the state of any hardware devices, and this function must be aware that the
|
||||||
|
device specified by ControllerHandle may already be managed by the same driver or a
|
||||||
|
different driver. This function must match its calls to AllocatePages() with FreePages(),
|
||||||
|
AllocatePool() with FreePool(), and OpenProtocol() with CloseProtocol().
|
||||||
|
Because ControllerHandle may have been previously started by the same driver, if a protocol is
|
||||||
|
already in the opened state, then it must not be closed with CloseProtocol(). This is required
|
||||||
|
to guarantee the state of ControllerHandle is not modified by this function.
|
||||||
|
|
||||||
|
@param[in] This A pointer to the EFI_DRIVER_BINDING_PROTOCOL instance.
|
||||||
|
@param[in] ControllerHandle The handle of the controller to test. This handle
|
||||||
|
must support a protocol interface that supplies
|
||||||
|
an I/O abstraction to the driver.
|
||||||
|
@param[in] RemainingDevicePath A pointer to the remaining portion of a device path. This
|
||||||
|
parameter is ignored by device drivers, and is optional for bus
|
||||||
|
drivers. For bus drivers, if this parameter is not NULL, then
|
||||||
|
the bus driver must determine if the bus controller specified
|
||||||
|
by ControllerHandle and the child controller specified
|
||||||
|
by RemainingDevicePath are both supported by this
|
||||||
|
bus driver.
|
||||||
|
|
||||||
|
@retval EFI_SUCCESS The device specified by ControllerHandle and
|
||||||
|
RemainingDevicePath is supported by the driver specified by This.
|
||||||
|
@retval EFI_ALREADY_STARTED The device specified by ControllerHandle and
|
||||||
|
RemainingDevicePath is already being managed by the driver
|
||||||
|
specified by This.
|
||||||
|
@retval EFI_ACCESS_DENIED The device specified by ControllerHandle and
|
||||||
|
RemainingDevicePath is already being managed by a different
|
||||||
|
driver or an application that requires exclusive access.
|
||||||
|
Currently not implemented.
|
||||||
|
@retval EFI_UNSUPPORTED The device specified by ControllerHandle and
|
||||||
|
RemainingDevicePath is not supported by the driver specified by This.
|
||||||
|
**/
|
||||||
|
EFI_STATUS
|
||||||
|
EFIAPI
|
||||||
|
HttpDxeIp6DriverBindingSupported (
|
||||||
|
IN EFI_DRIVER_BINDING_PROTOCOL *This,
|
||||||
|
IN EFI_HANDLE ControllerHandle,
|
||||||
|
IN EFI_DEVICE_PATH_PROTOCOL *RemainingDevicePath OPTIONAL
|
||||||
|
)
|
||||||
|
{
|
||||||
|
return HttpDxeSupported (
|
||||||
|
This,
|
||||||
|
ControllerHandle,
|
||||||
|
RemainingDevicePath,
|
||||||
|
IP_VERSION_6
|
||||||
|
);
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
Starts a device controller or a bus controller.
|
||||||
|
|
||||||
|
The Start() function is designed to be invoked from the EFI boot service ConnectController().
|
||||||
|
As a result, much of the error checking on the parameters to Start() has been moved into this
|
||||||
|
common boot service. It is legal to call Start() from other locations,
|
||||||
|
but the following calling restrictions must be followed, or the system behavior will not be deterministic.
|
||||||
|
1. ControllerHandle must be a valid EFI_HANDLE.
|
||||||
|
2. If RemainingDevicePath is not NULL, then it must be a pointer to a naturally aligned
|
||||||
|
EFI_DEVICE_PATH_PROTOCOL.
|
||||||
|
3. Prior to calling Start(), the Supported() function for the driver specified by This must
|
||||||
|
have been called with the same calling parameters, and Supported() must have returned EFI_SUCCESS.
|
||||||
|
|
||||||
|
@param[in] This A pointer to the EFI_DRIVER_BINDING_PROTOCOL instance.
|
||||||
|
@param[in] ControllerHandle The handle of the controller to start. This handle
|
||||||
|
must support a protocol interface that supplies
|
||||||
|
an I/O abstraction to the driver.
|
||||||
|
@param[in] RemainingDevicePath A pointer to the remaining portion of a device path. This
|
||||||
|
parameter is ignored by device drivers, and is optional for bus
|
||||||
|
drivers. For a bus driver, if this parameter is NULL, then handles
|
||||||
|
for all the children of Controller are created by this driver.
|
||||||
|
If this parameter is not NULL and the first Device Path Node is
|
||||||
|
not the End of Device Path Node, then only the handle for the
|
||||||
|
child device specified by the first Device Path Node of
|
||||||
|
RemainingDevicePath is created by this driver.
|
||||||
|
If the first Device Path Node of RemainingDevicePath is
|
||||||
|
the End of Device Path Node, no child handle is created by this
|
||||||
|
driver.
|
||||||
|
|
||||||
|
@retval EFI_SUCCESS The device was started.
|
||||||
|
@retval EFI_ALREADY_STARTED This device is already running on ControllerHandle.
|
||||||
|
@retval EFI_DEVICE_ERROR The device could not be started due to a device error.Currently not implemented.
|
||||||
|
@retval EFI_OUT_OF_RESOURCES The request could not be completed due to a lack of resources.
|
||||||
|
@retval Others The driver failded to start the device.
|
||||||
|
|
||||||
|
**/
|
||||||
|
EFI_STATUS
|
||||||
|
EFIAPI
|
||||||
|
HttpDxeIp6DriverBindingStart (
|
||||||
|
IN EFI_DRIVER_BINDING_PROTOCOL *This,
|
||||||
|
IN EFI_HANDLE ControllerHandle,
|
||||||
|
IN EFI_DEVICE_PATH_PROTOCOL *RemainingDevicePath OPTIONAL
|
||||||
|
)
|
||||||
|
{
|
||||||
|
return HttpDxeStart (
|
||||||
|
This,
|
||||||
|
ControllerHandle,
|
||||||
|
RemainingDevicePath,
|
||||||
|
IP_VERSION_6
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
Stops a device controller or a bus controller.
|
||||||
|
|
||||||
|
The Stop() function is designed to be invoked from the EFI boot service DisconnectController().
|
||||||
|
As a result, much of the error checking on the parameters to Stop() has been moved
|
||||||
|
into this common boot service. It is legal to call Stop() from other locations,
|
||||||
|
but the following calling restrictions must be followed, or the system behavior will not be deterministic.
|
||||||
|
1. ControllerHandle must be a valid EFI_HANDLE that was used on a previous call to this
|
||||||
|
same driver's Start() function.
|
||||||
|
2. The first NumberOfChildren handles of ChildHandleBuffer must all be a valid
|
||||||
|
EFI_HANDLE. In addition, all of these handles must have been created in this driver's
|
||||||
|
Start() function, and the Start() function must have called OpenProtocol() on
|
||||||
|
ControllerHandle with an Attribute of EFI_OPEN_PROTOCOL_BY_CHILD_CONTROLLER.
|
||||||
|
|
||||||
|
@param[in] This A pointer to the EFI_DRIVER_BINDING_PROTOCOL instance.
|
||||||
|
@param[in] ControllerHandle A handle to the device being stopped. The handle must
|
||||||
|
support a bus specific I/O protocol for the driver
|
||||||
|
to use to stop the device.
|
||||||
|
@param[in] NumberOfChildren The number of child device handles in ChildHandleBuffer.
|
||||||
|
@param[in] ChildHandleBuffer An array of child handles to be freed. May be NULL
|
||||||
|
if NumberOfChildren is 0.
|
||||||
|
|
||||||
|
@retval EFI_SUCCESS The device was stopped.
|
||||||
|
@retval EFI_DEVICE_ERROR The device could not be stopped due to a device error.
|
||||||
|
|
||||||
|
**/
|
||||||
|
EFI_STATUS
|
||||||
|
EFIAPI
|
||||||
|
HttpDxeIp6DriverBindingStop (
|
||||||
|
IN EFI_DRIVER_BINDING_PROTOCOL *This,
|
||||||
|
IN EFI_HANDLE ControllerHandle,
|
||||||
|
IN UINTN NumberOfChildren,
|
||||||
|
IN EFI_HANDLE *ChildHandleBuffer OPTIONAL
|
||||||
|
)
|
||||||
|
{
|
||||||
|
return HttpDxeStop (
|
||||||
|
This,
|
||||||
|
ControllerHandle,
|
||||||
|
NumberOfChildren,
|
||||||
|
ChildHandleBuffer,
|
||||||
|
IP_VERSION_6
|
||||||
|
);
|
||||||
|
}
|
||||||
/**
|
/**
|
||||||
Creates a child handle and installs a protocol.
|
Creates a child handle and installs a protocol.
|
||||||
|
|
||||||
@ -557,7 +925,6 @@ HttpServiceBindingCreateChild (
|
|||||||
HTTP_SERVICE *HttpService;
|
HTTP_SERVICE *HttpService;
|
||||||
HTTP_PROTOCOL *HttpInstance;
|
HTTP_PROTOCOL *HttpInstance;
|
||||||
EFI_STATUS Status;
|
EFI_STATUS Status;
|
||||||
VOID *Interface;
|
|
||||||
EFI_TPL OldTpl;
|
EFI_TPL OldTpl;
|
||||||
|
|
||||||
if ((This == NULL) || (ChildHandle == NULL)) {
|
if ((This == NULL) || (ChildHandle == NULL)) {
|
||||||
@ -569,6 +936,12 @@ HttpServiceBindingCreateChild (
|
|||||||
if (HttpInstance == NULL) {
|
if (HttpInstance == NULL) {
|
||||||
return EFI_OUT_OF_RESOURCES;
|
return EFI_OUT_OF_RESOURCES;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
HttpInstance->Signature = HTTP_PROTOCOL_SIGNATURE;
|
||||||
|
HttpInstance->Service = HttpService;
|
||||||
|
CopyMem (&HttpInstance->Http, &mEfiHttpTemplate, sizeof (HttpInstance->Http));
|
||||||
|
NetMapInit (&HttpInstance->TxTokens);
|
||||||
|
NetMapInit (&HttpInstance->RxTokens);
|
||||||
|
|
||||||
//
|
//
|
||||||
// Install HTTP protocol onto ChildHandle
|
// Install HTTP protocol onto ChildHandle
|
||||||
@ -584,27 +957,7 @@ HttpServiceBindingCreateChild (
|
|||||||
goto ON_ERROR;
|
goto ON_ERROR;
|
||||||
}
|
}
|
||||||
|
|
||||||
HttpInstance->Handle = *ChildHandle;
|
HttpInstance->Handle = *ChildHandle;
|
||||||
|
|
||||||
Status = HttpInitProtocol (HttpService, HttpInstance);
|
|
||||||
if (EFI_ERROR (Status)) {
|
|
||||||
goto ON_ERROR;
|
|
||||||
}
|
|
||||||
|
|
||||||
//
|
|
||||||
// Open the default Tcp4 protocol by child.
|
|
||||||
//
|
|
||||||
Status = gBS->OpenProtocol (
|
|
||||||
HttpService->TcpChildHandle,
|
|
||||||
&gEfiTcp4ProtocolGuid,
|
|
||||||
(VOID **) &Interface,
|
|
||||||
gHttpDxeDriverBinding.DriverBindingHandle,
|
|
||||||
HttpInstance->Handle,
|
|
||||||
EFI_OPEN_PROTOCOL_BY_CHILD_CONTROLLER
|
|
||||||
);
|
|
||||||
if (EFI_ERROR (Status)) {
|
|
||||||
goto ON_ERROR;
|
|
||||||
}
|
|
||||||
|
|
||||||
//
|
//
|
||||||
// Add it to the HTTP service's child list.
|
// Add it to the HTTP service's child list.
|
||||||
@ -619,8 +972,9 @@ HttpServiceBindingCreateChild (
|
|||||||
return EFI_SUCCESS;
|
return EFI_SUCCESS;
|
||||||
|
|
||||||
ON_ERROR:
|
ON_ERROR:
|
||||||
|
|
||||||
HttpCleanProtocol (HttpInstance);
|
NetMapClean (&HttpInstance->TxTokens);
|
||||||
|
NetMapClean (&HttpInstance->RxTokens);
|
||||||
FreePool (HttpInstance);
|
FreePool (HttpInstance);
|
||||||
|
|
||||||
return Status;
|
return Status;
|
||||||
@ -664,8 +1018,8 @@ HttpServiceBindingDestroyChild (
|
|||||||
ChildHandle,
|
ChildHandle,
|
||||||
&gEfiHttpProtocolGuid,
|
&gEfiHttpProtocolGuid,
|
||||||
(VOID **) &Http,
|
(VOID **) &Http,
|
||||||
gHttpDxeDriverBinding.DriverBindingHandle,
|
NULL,
|
||||||
ChildHandle,
|
NULL,
|
||||||
EFI_OPEN_PROTOCOL_GET_PROTOCOL
|
EFI_OPEN_PROTOCOL_GET_PROTOCOL
|
||||||
);
|
);
|
||||||
if (EFI_ERROR (Status)) {
|
if (EFI_ERROR (Status)) {
|
||||||
@ -681,16 +1035,6 @@ HttpServiceBindingDestroyChild (
|
|||||||
return EFI_SUCCESS;
|
return EFI_SUCCESS;
|
||||||
}
|
}
|
||||||
|
|
||||||
//
|
|
||||||
// Close the Tcp4 protocol.
|
|
||||||
//
|
|
||||||
gBS->CloseProtocol (
|
|
||||||
HttpService->TcpChildHandle,
|
|
||||||
&gEfiTcp4ProtocolGuid,
|
|
||||||
gHttpDxeDriverBinding.DriverBindingHandle,
|
|
||||||
ChildHandle
|
|
||||||
);
|
|
||||||
|
|
||||||
HttpInstance->InDestroy = TRUE;
|
HttpInstance->InDestroy = TRUE;
|
||||||
|
|
||||||
//
|
//
|
||||||
@ -706,11 +1050,11 @@ HttpServiceBindingDestroyChild (
|
|||||||
HttpInstance->InDestroy = FALSE;
|
HttpInstance->InDestroy = FALSE;
|
||||||
return Status;
|
return Status;
|
||||||
}
|
}
|
||||||
|
|
||||||
OldTpl = gBS->RaiseTPL (TPL_CALLBACK);
|
|
||||||
|
|
||||||
HttpCleanProtocol (HttpInstance);
|
HttpCleanProtocol (HttpInstance);
|
||||||
|
|
||||||
|
OldTpl = gBS->RaiseTPL (TPL_CALLBACK);
|
||||||
|
|
||||||
RemoveEntryList (&HttpInstance->Link);
|
RemoveEntryList (&HttpInstance->Link);
|
||||||
HttpService->ChildrenNumber--;
|
HttpService->ChildrenNumber--;
|
||||||
|
|
||||||
|
@ -43,8 +43,12 @@
|
|||||||
//
|
//
|
||||||
#include <Protocol/HttpUtilities.h>
|
#include <Protocol/HttpUtilities.h>
|
||||||
#include <Protocol/Tcp4.h>
|
#include <Protocol/Tcp4.h>
|
||||||
|
#include <Protocol/Tcp6.h>
|
||||||
#include <Protocol/Dns4.h>
|
#include <Protocol/Dns4.h>
|
||||||
|
#include <Protocol/Dns6.h>
|
||||||
#include <Protocol/Ip4Config2.h>
|
#include <Protocol/Ip4Config2.h>
|
||||||
|
#include <Protocol/Ip6Config.h>
|
||||||
|
|
||||||
|
|
||||||
//
|
//
|
||||||
// Produced Protocols
|
// Produced Protocols
|
||||||
@ -59,7 +63,9 @@
|
|||||||
//
|
//
|
||||||
// Protocol instances
|
// Protocol instances
|
||||||
//
|
//
|
||||||
extern EFI_DRIVER_BINDING_PROTOCOL gHttpDxeDriverBinding;
|
extern EFI_DRIVER_BINDING_PROTOCOL gHttpDxeIp4DriverBinding;
|
||||||
|
extern EFI_DRIVER_BINDING_PROTOCOL gHttpDxeIp6DriverBinding;
|
||||||
|
|
||||||
extern EFI_COMPONENT_NAME2_PROTOCOL gHttpDxeComponentName2;
|
extern EFI_COMPONENT_NAME2_PROTOCOL gHttpDxeComponentName2;
|
||||||
extern EFI_COMPONENT_NAME_PROTOCOL gHttpDxeComponentName;
|
extern EFI_COMPONENT_NAME_PROTOCOL gHttpDxeComponentName;
|
||||||
|
|
||||||
@ -123,7 +129,7 @@ typedef struct {
|
|||||||
**/
|
**/
|
||||||
EFI_STATUS
|
EFI_STATUS
|
||||||
EFIAPI
|
EFIAPI
|
||||||
HttpDxeDriverBindingSupported (
|
HttpDxeIp4DriverBindingSupported (
|
||||||
IN EFI_DRIVER_BINDING_PROTOCOL *This,
|
IN EFI_DRIVER_BINDING_PROTOCOL *This,
|
||||||
IN EFI_HANDLE ControllerHandle,
|
IN EFI_HANDLE ControllerHandle,
|
||||||
IN EFI_DEVICE_PATH_PROTOCOL *RemainingDevicePath OPTIONAL
|
IN EFI_DEVICE_PATH_PROTOCOL *RemainingDevicePath OPTIONAL
|
||||||
@ -166,7 +172,7 @@ HttpDxeDriverBindingSupported (
|
|||||||
**/
|
**/
|
||||||
EFI_STATUS
|
EFI_STATUS
|
||||||
EFIAPI
|
EFIAPI
|
||||||
HttpDxeDriverBindingStart (
|
HttpDxeIp4DriverBindingStart (
|
||||||
IN EFI_DRIVER_BINDING_PROTOCOL *This,
|
IN EFI_DRIVER_BINDING_PROTOCOL *This,
|
||||||
IN EFI_HANDLE ControllerHandle,
|
IN EFI_HANDLE ControllerHandle,
|
||||||
IN EFI_DEVICE_PATH_PROTOCOL *RemainingDevicePath OPTIONAL
|
IN EFI_DEVICE_PATH_PROTOCOL *RemainingDevicePath OPTIONAL
|
||||||
@ -200,13 +206,142 @@ HttpDxeDriverBindingStart (
|
|||||||
**/
|
**/
|
||||||
EFI_STATUS
|
EFI_STATUS
|
||||||
EFIAPI
|
EFIAPI
|
||||||
HttpDxeDriverBindingStop (
|
HttpDxeIp4DriverBindingStop (
|
||||||
IN EFI_DRIVER_BINDING_PROTOCOL *This,
|
IN EFI_DRIVER_BINDING_PROTOCOL *This,
|
||||||
IN EFI_HANDLE ControllerHandle,
|
IN EFI_HANDLE ControllerHandle,
|
||||||
IN UINTN NumberOfChildren,
|
IN UINTN NumberOfChildren,
|
||||||
IN EFI_HANDLE *ChildHandleBuffer OPTIONAL
|
IN EFI_HANDLE *ChildHandleBuffer OPTIONAL
|
||||||
);
|
);
|
||||||
|
|
||||||
|
/**
|
||||||
|
Tests to see if this driver supports a given controller. If a child device is provided,
|
||||||
|
it further tests to see if this driver supports creating a handle for the specified child device.
|
||||||
|
|
||||||
|
This function checks to see if the driver specified by This supports the device specified by
|
||||||
|
ControllerHandle. Drivers will typically use the device path attached to
|
||||||
|
ControllerHandle and/or the services from the bus I/O abstraction attached to
|
||||||
|
ControllerHandle to determine if the driver supports ControllerHandle. This function
|
||||||
|
may be called many times during platform initialization. In order to reduce boot times, the tests
|
||||||
|
performed by this function must be very small, and take as little time as possible to execute. This
|
||||||
|
function must not change the state of any hardware devices, and this function must be aware that the
|
||||||
|
device specified by ControllerHandle may already be managed by the same driver or a
|
||||||
|
different driver. This function must match its calls to AllocatePages() with FreePages(),
|
||||||
|
AllocatePool() with FreePool(), and OpenProtocol() with CloseProtocol().
|
||||||
|
Because ControllerHandle may have been previously started by the same driver, if a protocol is
|
||||||
|
already in the opened state, then it must not be closed with CloseProtocol(). This is required
|
||||||
|
to guarantee the state of ControllerHandle is not modified by this function.
|
||||||
|
|
||||||
|
@param[in] This A pointer to the EFI_DRIVER_BINDING_PROTOCOL instance.
|
||||||
|
@param[in] ControllerHandle The handle of the controller to test. This handle
|
||||||
|
must support a protocol interface that supplies
|
||||||
|
an I/O abstraction to the driver.
|
||||||
|
@param[in] RemainingDevicePath A pointer to the remaining portion of a device path. This
|
||||||
|
parameter is ignored by device drivers, and is optional for bus
|
||||||
|
drivers. For bus drivers, if this parameter is not NULL, then
|
||||||
|
the bus driver must determine if the bus controller specified
|
||||||
|
by ControllerHandle and the child controller specified
|
||||||
|
by RemainingDevicePath are both supported by this
|
||||||
|
bus driver.
|
||||||
|
|
||||||
|
@retval EFI_SUCCESS The device specified by ControllerHandle and
|
||||||
|
RemainingDevicePath is supported by the driver specified by This.
|
||||||
|
@retval EFI_ALREADY_STARTED The device specified by ControllerHandle and
|
||||||
|
RemainingDevicePath is already being managed by the driver
|
||||||
|
specified by This.
|
||||||
|
@retval EFI_ACCESS_DENIED The device specified by ControllerHandle and
|
||||||
|
RemainingDevicePath is already being managed by a different
|
||||||
|
driver or an application that requires exclusive access.
|
||||||
|
Currently not implemented.
|
||||||
|
@retval EFI_UNSUPPORTED The device specified by ControllerHandle and
|
||||||
|
RemainingDevicePath is not supported by the driver specified by This.
|
||||||
|
**/
|
||||||
|
EFI_STATUS
|
||||||
|
EFIAPI
|
||||||
|
HttpDxeIp6DriverBindingSupported (
|
||||||
|
IN EFI_DRIVER_BINDING_PROTOCOL *This,
|
||||||
|
IN EFI_HANDLE ControllerHandle,
|
||||||
|
IN EFI_DEVICE_PATH_PROTOCOL *RemainingDevicePath OPTIONAL
|
||||||
|
);
|
||||||
|
|
||||||
|
/**
|
||||||
|
Starts a device controller or a bus controller.
|
||||||
|
|
||||||
|
The Start() function is designed to be invoked from the EFI boot service ConnectController().
|
||||||
|
As a result, much of the error checking on the parameters to Start() has been moved into this
|
||||||
|
common boot service. It is legal to call Start() from other locations,
|
||||||
|
but the following calling restrictions must be followed, or the system behavior will not be deterministic.
|
||||||
|
1. ControllerHandle must be a valid EFI_HANDLE.
|
||||||
|
2. If RemainingDevicePath is not NULL, then it must be a pointer to a naturally aligned
|
||||||
|
EFI_DEVICE_PATH_PROTOCOL.
|
||||||
|
3. Prior to calling Start(), the Supported() function for the driver specified by This must
|
||||||
|
have been called with the same calling parameters, and Supported() must have returned EFI_SUCCESS.
|
||||||
|
|
||||||
|
@param[in] This A pointer to the EFI_DRIVER_BINDING_PROTOCOL instance.
|
||||||
|
@param[in] ControllerHandle The handle of the controller to start. This handle
|
||||||
|
must support a protocol interface that supplies
|
||||||
|
an I/O abstraction to the driver.
|
||||||
|
@param[in] RemainingDevicePath A pointer to the remaining portion of a device path. This
|
||||||
|
parameter is ignored by device drivers, and is optional for bus
|
||||||
|
drivers. For a bus driver, if this parameter is NULL, then handles
|
||||||
|
for all the children of Controller are created by this driver.
|
||||||
|
If this parameter is not NULL and the first Device Path Node is
|
||||||
|
not the End of Device Path Node, then only the handle for the
|
||||||
|
child device specified by the first Device Path Node of
|
||||||
|
RemainingDevicePath is created by this driver.
|
||||||
|
If the first Device Path Node of RemainingDevicePath is
|
||||||
|
the End of Device Path Node, no child handle is created by this
|
||||||
|
driver.
|
||||||
|
|
||||||
|
@retval EFI_SUCCESS The device was started.
|
||||||
|
@retval EFI_ALREADY_STARTED This device is already running on ControllerHandle.
|
||||||
|
@retval EFI_DEVICE_ERROR The device could not be started due to a device error.Currently not implemented.
|
||||||
|
@retval EFI_OUT_OF_RESOURCES The request could not be completed due to a lack of resources.
|
||||||
|
@retval Others The driver failded to start the device.
|
||||||
|
|
||||||
|
**/
|
||||||
|
EFI_STATUS
|
||||||
|
EFIAPI
|
||||||
|
HttpDxeIp6DriverBindingStart (
|
||||||
|
IN EFI_DRIVER_BINDING_PROTOCOL *This,
|
||||||
|
IN EFI_HANDLE ControllerHandle,
|
||||||
|
IN EFI_DEVICE_PATH_PROTOCOL *RemainingDevicePath OPTIONAL
|
||||||
|
);
|
||||||
|
|
||||||
|
/**
|
||||||
|
Stops a device controller or a bus controller.
|
||||||
|
|
||||||
|
The Stop() function is designed to be invoked from the EFI boot service DisconnectController().
|
||||||
|
As a result, much of the error checking on the parameters to Stop() has been moved
|
||||||
|
into this common boot service. It is legal to call Stop() from other locations,
|
||||||
|
but the following calling restrictions must be followed, or the system behavior will not be deterministic.
|
||||||
|
1. ControllerHandle must be a valid EFI_HANDLE that was used on a previous call to this
|
||||||
|
same driver's Start() function.
|
||||||
|
2. The first NumberOfChildren handles of ChildHandleBuffer must all be a valid
|
||||||
|
EFI_HANDLE. In addition, all of these handles must have been created in this driver's
|
||||||
|
Start() function, and the Start() function must have called OpenProtocol() on
|
||||||
|
ControllerHandle with an Attribute of EFI_OPEN_PROTOCOL_BY_CHILD_CONTROLLER.
|
||||||
|
|
||||||
|
@param[in] This A pointer to the EFI_DRIVER_BINDING_PROTOCOL instance.
|
||||||
|
@param[in] ControllerHandle A handle to the device being stopped. The handle must
|
||||||
|
support a bus specific I/O protocol for the driver
|
||||||
|
to use to stop the device.
|
||||||
|
@param[in] NumberOfChildren The number of child device handles in ChildHandleBuffer.
|
||||||
|
@param[in] ChildHandleBuffer An array of child handles to be freed. May be NULL
|
||||||
|
if NumberOfChildren is 0.
|
||||||
|
|
||||||
|
@retval EFI_SUCCESS The device was stopped.
|
||||||
|
@retval EFI_DEVICE_ERROR The device could not be stopped due to a device error.
|
||||||
|
|
||||||
|
**/
|
||||||
|
EFI_STATUS
|
||||||
|
EFIAPI
|
||||||
|
HttpDxeIp6DriverBindingStop (
|
||||||
|
IN EFI_DRIVER_BINDING_PROTOCOL *This,
|
||||||
|
IN EFI_HANDLE ControllerHandle,
|
||||||
|
IN UINTN NumberOfChildren,
|
||||||
|
IN EFI_HANDLE *ChildHandleBuffer OPTIONAL
|
||||||
|
);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
Creates a child handle and installs a protocol.
|
Creates a child handle and installs a protocol.
|
||||||
|
|
||||||
|
@ -56,9 +56,14 @@
|
|||||||
gEfiHttpUtilitiesProtocolGuid ## CONSUMES
|
gEfiHttpUtilitiesProtocolGuid ## CONSUMES
|
||||||
gEfiTcp4ServiceBindingProtocolGuid ## TO_START
|
gEfiTcp4ServiceBindingProtocolGuid ## TO_START
|
||||||
gEfiTcp4ProtocolGuid ## TO_START
|
gEfiTcp4ProtocolGuid ## TO_START
|
||||||
|
gEfiTcp6ServiceBindingProtocolGuid ## TO_START
|
||||||
|
gEfiTcp6ProtocolGuid ## TO_START
|
||||||
gEfiDns4ServiceBindingProtocolGuid ## SOMETIMES_CONSUMES
|
gEfiDns4ServiceBindingProtocolGuid ## SOMETIMES_CONSUMES
|
||||||
gEfiDns4ProtocolGuid ## SOMETIMES_CONSUMES
|
gEfiDns4ProtocolGuid ## SOMETIMES_CONSUMES
|
||||||
|
gEfiDns6ServiceBindingProtocolGuid ## SOMETIMES_CONSUMES
|
||||||
|
gEfiDns6ProtocolGuid ## SOMETIMES_CONSUMES
|
||||||
gEfiIp4Config2ProtocolGuid ## SOMETIMES_CONSUMES
|
gEfiIp4Config2ProtocolGuid ## SOMETIMES_CONSUMES
|
||||||
|
gEfiIp6ConfigProtocolGuid ## SOMETIMES_CONSUMES
|
||||||
|
|
||||||
[UserExtensions.TianoCore."ExtraFiles"]
|
[UserExtensions.TianoCore."ExtraFiles"]
|
||||||
HttpDxeExtra.uni
|
HttpDxeExtra.uni
|
Binary file not shown.
@ -51,6 +51,8 @@ EfiHttpGetModeData (
|
|||||||
)
|
)
|
||||||
{
|
{
|
||||||
HTTP_PROTOCOL *HttpInstance;
|
HTTP_PROTOCOL *HttpInstance;
|
||||||
|
EFI_HTTPv4_ACCESS_POINT *Http4AccessPoint;
|
||||||
|
EFI_HTTPv6_ACCESS_POINT *Http6AccessPoint;
|
||||||
|
|
||||||
if ((This == NULL) || (HttpConfigData == NULL)) {
|
if ((This == NULL) || (HttpConfigData == NULL)) {
|
||||||
return EFI_INVALID_PARAMETER;
|
return EFI_INVALID_PARAMETER;
|
||||||
@ -58,24 +60,32 @@ EfiHttpGetModeData (
|
|||||||
|
|
||||||
HttpInstance = HTTP_INSTANCE_FROM_PROTOCOL (This);
|
HttpInstance = HTTP_INSTANCE_FROM_PROTOCOL (This);
|
||||||
ASSERT (HttpInstance != NULL);
|
ASSERT (HttpInstance != NULL);
|
||||||
|
|
||||||
if (HttpInstance->State < HTTP_STATE_HTTP_CONFIGED) {
|
if (HttpInstance->State < HTTP_STATE_HTTP_CONFIGED) {
|
||||||
return EFI_NOT_STARTED;
|
return EFI_NOT_STARTED;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (HttpConfigData->AccessPoint.IPv4Node == NULL) {
|
|
||||||
return EFI_INVALID_PARAMETER;
|
|
||||||
}
|
|
||||||
|
|
||||||
HttpConfigData->HttpVersion = HttpInstance->HttpVersion;
|
HttpConfigData->HttpVersion = HttpInstance->HttpVersion;
|
||||||
HttpConfigData->TimeOutMillisec = HttpInstance->TimeOutMillisec;
|
HttpConfigData->TimeOutMillisec = HttpInstance->TimeOutMillisec;
|
||||||
HttpConfigData->LocalAddressIsIPv6 = HttpInstance->LocalAddressIsIPv6;
|
HttpConfigData->LocalAddressIsIPv6 = HttpInstance->LocalAddressIsIPv6;
|
||||||
|
|
||||||
CopyMem (
|
if (HttpInstance->LocalAddressIsIPv6) {
|
||||||
HttpConfigData->AccessPoint.IPv4Node,
|
Http6AccessPoint = AllocateZeroPool (sizeof (EFI_HTTPv6_ACCESS_POINT));
|
||||||
&HttpInstance->IPv4Node,
|
CopyMem (
|
||||||
sizeof (HttpInstance->IPv4Node)
|
Http6AccessPoint,
|
||||||
|
&HttpInstance->Ipv6Node,
|
||||||
|
sizeof (HttpInstance->Ipv6Node)
|
||||||
);
|
);
|
||||||
|
HttpConfigData->AccessPoint.IPv6Node = Http6AccessPoint;
|
||||||
|
} else {
|
||||||
|
Http4AccessPoint = AllocateZeroPool (sizeof (EFI_HTTPv4_ACCESS_POINT));
|
||||||
|
CopyMem (
|
||||||
|
Http4AccessPoint,
|
||||||
|
&HttpInstance->IPv4Node,
|
||||||
|
sizeof (HttpInstance->IPv4Node)
|
||||||
|
);
|
||||||
|
HttpConfigData->AccessPoint.IPv4Node = Http4AccessPoint;
|
||||||
|
}
|
||||||
|
|
||||||
return EFI_SUCCESS;
|
return EFI_SUCCESS;
|
||||||
}
|
}
|
||||||
@ -120,8 +130,13 @@ EfiHttpConfigure (
|
|||||||
{
|
{
|
||||||
HTTP_PROTOCOL *HttpInstance;
|
HTTP_PROTOCOL *HttpInstance;
|
||||||
EFI_STATUS Status;
|
EFI_STATUS Status;
|
||||||
|
|
||||||
if (This == NULL) {
|
//
|
||||||
|
// Check input parameters.
|
||||||
|
//
|
||||||
|
if (This == NULL ||
|
||||||
|
(HttpConfigData != NULL && ((HttpConfigData->LocalAddressIsIPv6 && HttpConfigData->AccessPoint.IPv6Node == NULL) ||
|
||||||
|
(!HttpConfigData->LocalAddressIsIPv6 && HttpConfigData->AccessPoint.IPv4Node == NULL)))) {
|
||||||
return EFI_INVALID_PARAMETER;
|
return EFI_INVALID_PARAMETER;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -129,18 +144,7 @@ EfiHttpConfigure (
|
|||||||
ASSERT (HttpInstance != NULL && HttpInstance->Service != NULL);
|
ASSERT (HttpInstance != NULL && HttpInstance->Service != NULL);
|
||||||
|
|
||||||
if (HttpConfigData != NULL) {
|
if (HttpConfigData != NULL) {
|
||||||
//
|
|
||||||
// Check input parameters.
|
|
||||||
//
|
|
||||||
if (HttpConfigData->LocalAddressIsIPv6) {
|
|
||||||
if (HttpConfigData->AccessPoint.IPv6Node == NULL) {
|
|
||||||
return EFI_INVALID_PARAMETER;
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
if (HttpConfigData->AccessPoint.IPv4Node == NULL) {
|
|
||||||
return EFI_INVALID_PARAMETER;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
//
|
//
|
||||||
// Now configure this HTTP instance.
|
// Now configure this HTTP instance.
|
||||||
//
|
//
|
||||||
@ -151,33 +155,38 @@ EfiHttpConfigure (
|
|||||||
HttpInstance->HttpVersion = HttpConfigData->HttpVersion;
|
HttpInstance->HttpVersion = HttpConfigData->HttpVersion;
|
||||||
HttpInstance->TimeOutMillisec = HttpConfigData->TimeOutMillisec;
|
HttpInstance->TimeOutMillisec = HttpConfigData->TimeOutMillisec;
|
||||||
HttpInstance->LocalAddressIsIPv6 = HttpConfigData->LocalAddressIsIPv6;
|
HttpInstance->LocalAddressIsIPv6 = HttpConfigData->LocalAddressIsIPv6;
|
||||||
|
|
||||||
if (HttpConfigData->LocalAddressIsIPv6) {
|
if (HttpConfigData->LocalAddressIsIPv6) {
|
||||||
return EFI_UNSUPPORTED;
|
CopyMem (
|
||||||
|
&HttpInstance->Ipv6Node,
|
||||||
|
HttpConfigData->AccessPoint.IPv6Node,
|
||||||
|
sizeof (HttpInstance->Ipv6Node)
|
||||||
|
);
|
||||||
} else {
|
} else {
|
||||||
CopyMem (
|
CopyMem (
|
||||||
&HttpInstance->IPv4Node,
|
&HttpInstance->IPv4Node,
|
||||||
HttpConfigData->AccessPoint.IPv4Node,
|
HttpConfigData->AccessPoint.IPv4Node,
|
||||||
sizeof (HttpInstance->IPv4Node)
|
sizeof (HttpInstance->IPv4Node)
|
||||||
);
|
);
|
||||||
|
|
||||||
HttpInstance->State = HTTP_STATE_HTTP_CONFIGED;
|
|
||||||
return EFI_SUCCESS;
|
|
||||||
}
|
}
|
||||||
|
//
|
||||||
|
// Creat Tcp child
|
||||||
|
//
|
||||||
|
Status = HttpInitProtocol (HttpInstance, HttpInstance->LocalAddressIsIPv6);
|
||||||
|
if (EFI_ERROR (Status)) {
|
||||||
|
return Status;
|
||||||
|
}
|
||||||
|
|
||||||
|
HttpInstance->State = HTTP_STATE_HTTP_CONFIGED;
|
||||||
|
return EFI_SUCCESS;
|
||||||
|
|
||||||
} else {
|
} else {
|
||||||
if (HttpInstance->LocalAddressIsIPv6) {
|
//
|
||||||
return EFI_UNSUPPORTED;
|
// Reset all the resources related to HttpInsance.
|
||||||
} else {
|
//
|
||||||
HttpCleanProtocol (HttpInstance);
|
HttpCleanProtocol (HttpInstance);
|
||||||
Status = HttpInitProtocol (HttpInstance->Service, HttpInstance);
|
HttpInstance->State = HTTP_STATE_UNCONFIGED;
|
||||||
if (EFI_ERROR (Status)) {
|
return EFI_SUCCESS;
|
||||||
return Status;
|
|
||||||
}
|
|
||||||
|
|
||||||
HttpInstance->State = HTTP_STATE_UNCONFIGED;
|
|
||||||
return EFI_SUCCESS;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -264,10 +273,6 @@ EfiHttpRequest (
|
|||||||
return EFI_NOT_STARTED;
|
return EFI_NOT_STARTED;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (HttpInstance->LocalAddressIsIPv6) {
|
|
||||||
return EFI_UNSUPPORTED;
|
|
||||||
}
|
|
||||||
|
|
||||||
//
|
//
|
||||||
// Check whether the token already existed.
|
// Check whether the token already existed.
|
||||||
//
|
//
|
||||||
@ -291,7 +296,8 @@ EfiHttpRequest (
|
|||||||
}
|
}
|
||||||
FreePool (HttpInstance->Url);
|
FreePool (HttpInstance->Url);
|
||||||
HttpInstance->Url = Url;
|
HttpInstance->Url = Url;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
UnicodeStrToAsciiStr (Request->Url, Url);
|
UnicodeStrToAsciiStr (Request->Url, Url);
|
||||||
UrlParser = NULL;
|
UrlParser = NULL;
|
||||||
@ -340,7 +346,7 @@ EfiHttpRequest (
|
|||||||
Wrap->HttpToken = Token;
|
Wrap->HttpToken = Token;
|
||||||
Wrap->HttpInstance = HttpInstance;
|
Wrap->HttpInstance = HttpInstance;
|
||||||
|
|
||||||
Status = HttpCreateTcp4TxEvent (Wrap);
|
Status = HttpCreateTcpTxEvent (Wrap);
|
||||||
if (EFI_ERROR (Status)) {
|
if (EFI_ERROR (Status)) {
|
||||||
goto Error1;
|
goto Error1;
|
||||||
}
|
}
|
||||||
@ -379,24 +385,35 @@ EfiHttpRequest (
|
|||||||
|
|
||||||
if (Configure) {
|
if (Configure) {
|
||||||
//
|
//
|
||||||
// Parse Url for IPv4 address, if failed, perform DNS resolution.
|
// Parse Url for IPv4 or IPv6 address, if failed, perform DNS resolution.
|
||||||
//
|
//
|
||||||
Status = NetLibAsciiStrToIp4 (HostName, &HttpInstance->RemoteAddr);
|
if (!HttpInstance->LocalAddressIsIPv6) {
|
||||||
|
Status = NetLibAsciiStrToIp4 (HostName, &HttpInstance->RemoteAddr);
|
||||||
|
} else {
|
||||||
|
Status = NetLibAsciiStrToIp6 (HostName, &HttpInstance->RemoteIpv6Addr);
|
||||||
|
}
|
||||||
|
|
||||||
if (EFI_ERROR (Status)) {
|
if (EFI_ERROR (Status)) {
|
||||||
HostNameStr = AllocateZeroPool ((AsciiStrLen (HostName) + 1) * sizeof (UINT16));
|
HostNameStr = AllocateZeroPool ((AsciiStrLen (HostName) + 1) * sizeof (CHAR16));
|
||||||
if (HostNameStr == NULL) {
|
if (HostNameStr == NULL) {
|
||||||
Status = EFI_OUT_OF_RESOURCES;
|
Status = EFI_OUT_OF_RESOURCES;
|
||||||
goto Error1;
|
goto Error1;
|
||||||
}
|
}
|
||||||
|
|
||||||
AsciiStrToUnicodeStr (HostName, HostNameStr);
|
AsciiStrToUnicodeStr (HostName, HostNameStr);
|
||||||
Status = HttpDns4 (HttpInstance, HostNameStr, &HttpInstance->RemoteAddr);
|
if (!HttpInstance->LocalAddressIsIPv6) {
|
||||||
|
Status = HttpDns4 (HttpInstance, HostNameStr, &HttpInstance->RemoteAddr);
|
||||||
|
} else {
|
||||||
|
Status = HttpDns6 (HttpInstance, HostNameStr, &HttpInstance->RemoteIpv6Addr);
|
||||||
|
}
|
||||||
|
|
||||||
FreePool (HostNameStr);
|
FreePool (HostNameStr);
|
||||||
if (EFI_ERROR (Status)) {
|
if (EFI_ERROR (Status)) {
|
||||||
goto Error1;
|
goto Error1;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
//
|
//
|
||||||
// Save the RemotePort and RemoteHost.
|
// Save the RemotePort and RemoteHost.
|
||||||
//
|
//
|
||||||
@ -410,7 +427,7 @@ EfiHttpRequest (
|
|||||||
//
|
//
|
||||||
// The request URL is different from previous calls to Request(), close existing TCP instance.
|
// The request URL is different from previous calls to Request(), close existing TCP instance.
|
||||||
//
|
//
|
||||||
ASSERT (HttpInstance->Tcp4 != NULL);
|
ASSERT (HttpInstance->Tcp4 != NULL &&HttpInstance->Tcp6 != NULL);
|
||||||
HttpCloseConnection (HttpInstance);
|
HttpCloseConnection (HttpInstance);
|
||||||
EfiHttpCancel (This, NULL);
|
EfiHttpCancel (This, NULL);
|
||||||
}
|
}
|
||||||
@ -429,25 +446,16 @@ EfiHttpRequest (
|
|||||||
Wrap->TcpWrap.Method = Request->Method;
|
Wrap->TcpWrap.Method = Request->Method;
|
||||||
|
|
||||||
if (Configure) {
|
if (Configure) {
|
||||||
//
|
Status = HttpInitTcp (HttpInstance, Wrap);
|
||||||
// Configure TCP instance.
|
|
||||||
//
|
|
||||||
Status = HttpConfigureTcp4 (HttpInstance, Wrap);
|
|
||||||
if (EFI_ERROR (Status)) {
|
|
||||||
goto Error1;
|
|
||||||
}
|
|
||||||
//
|
|
||||||
// Connect TCP.
|
|
||||||
//
|
|
||||||
Status = HttpConnectTcp4 (HttpInstance);
|
|
||||||
if (EFI_ERROR (Status)) {
|
if (EFI_ERROR (Status)) {
|
||||||
goto Error2;
|
goto Error2;
|
||||||
}
|
}
|
||||||
|
|
||||||
} else {
|
} else {
|
||||||
//
|
//
|
||||||
// For the new HTTP token, create TX TCP token events.
|
// For the new HTTP token, create TX TCP token events.
|
||||||
//
|
//
|
||||||
Status = HttpCreateTcp4TxEvent (Wrap);
|
Status = HttpCreateTcpTxEvent (Wrap);
|
||||||
if (EFI_ERROR (Status)) {
|
if (EFI_ERROR (Status)) {
|
||||||
goto Error1;
|
goto Error1;
|
||||||
}
|
}
|
||||||
@ -488,7 +496,7 @@ EfiHttpRequest (
|
|||||||
//
|
//
|
||||||
// Transmit the request message.
|
// Transmit the request message.
|
||||||
//
|
//
|
||||||
Status = HttpTransmitTcp4 (
|
Status = HttpTransmitTcp (
|
||||||
HttpInstance,
|
HttpInstance,
|
||||||
Wrap,
|
Wrap,
|
||||||
(UINT8*) RequestStr,
|
(UINT8*) RequestStr,
|
||||||
@ -499,11 +507,11 @@ EfiHttpRequest (
|
|||||||
}
|
}
|
||||||
|
|
||||||
DispatchDpc ();
|
DispatchDpc ();
|
||||||
|
|
||||||
if (HostName != NULL) {
|
if (HostName != NULL) {
|
||||||
FreePool (HostName);
|
FreePool (HostName);
|
||||||
}
|
}
|
||||||
|
|
||||||
return EFI_SUCCESS;
|
return EFI_SUCCESS;
|
||||||
|
|
||||||
Error5:
|
Error5:
|
||||||
@ -517,15 +525,19 @@ Error4:
|
|||||||
Error3:
|
Error3:
|
||||||
HttpCloseConnection (HttpInstance);
|
HttpCloseConnection (HttpInstance);
|
||||||
|
|
||||||
|
|
||||||
Error2:
|
Error2:
|
||||||
HttpCloseTcp4ConnCloseEvent (HttpInstance);
|
HttpCloseTcpConnCloseEvent (HttpInstance);
|
||||||
if (NULL != Wrap->TcpWrap.TxToken.CompletionToken.Event) {
|
if (NULL != Wrap->TcpWrap.Tx4Token.CompletionToken.Event) {
|
||||||
gBS->CloseEvent (Wrap->TcpWrap.TxToken.CompletionToken.Event);
|
gBS->CloseEvent (Wrap->TcpWrap.Tx4Token.CompletionToken.Event);
|
||||||
Wrap->TcpWrap.TxToken.CompletionToken.Event = NULL;
|
Wrap->TcpWrap.Tx4Token.CompletionToken.Event = NULL;
|
||||||
|
}
|
||||||
|
if (NULL != Wrap->TcpWrap.Tx6Token.CompletionToken.Event) {
|
||||||
|
gBS->CloseEvent (Wrap->TcpWrap.Tx6Token.CompletionToken.Event);
|
||||||
|
Wrap->TcpWrap.Tx6Token.CompletionToken.Event = NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
Error1:
|
Error1:
|
||||||
|
|
||||||
if (HostName != NULL) {
|
if (HostName != NULL) {
|
||||||
FreePool (HostName);
|
FreePool (HostName);
|
||||||
}
|
}
|
||||||
@ -541,7 +553,7 @@ Error1:
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
Cancel a TxToken or RxToken.
|
Cancel a user's Token.
|
||||||
|
|
||||||
@param[in] Map The HTTP instance's token queue.
|
@param[in] Map The HTTP instance's token queue.
|
||||||
@param[in] Item Object container for one HTTP token and token's wrap.
|
@param[in] Item Object container for one HTTP token and token's wrap.
|
||||||
@ -562,6 +574,7 @@ HttpCancelTokens (
|
|||||||
|
|
||||||
EFI_HTTP_TOKEN *Token;
|
EFI_HTTP_TOKEN *Token;
|
||||||
HTTP_TOKEN_WRAP *Wrap;
|
HTTP_TOKEN_WRAP *Wrap;
|
||||||
|
HTTP_PROTOCOL *HttpInstance;
|
||||||
|
|
||||||
Token = (EFI_HTTP_TOKEN *) Context;
|
Token = (EFI_HTTP_TOKEN *) Context;
|
||||||
|
|
||||||
@ -575,23 +588,40 @@ HttpCancelTokens (
|
|||||||
|
|
||||||
Wrap = (HTTP_TOKEN_WRAP *) Item->Value;
|
Wrap = (HTTP_TOKEN_WRAP *) Item->Value;
|
||||||
ASSERT (Wrap != NULL);
|
ASSERT (Wrap != NULL);
|
||||||
|
HttpInstance = Wrap->HttpInstance;
|
||||||
|
|
||||||
//
|
//
|
||||||
// Free resources.
|
// Free resources.
|
||||||
//
|
//
|
||||||
NetMapRemoveItem (Map, Item, NULL);
|
NetMapRemoveItem (Map, Item, NULL);
|
||||||
|
|
||||||
if (Wrap->TcpWrap.TxToken.CompletionToken.Event != NULL) {
|
if (!HttpInstance->LocalAddressIsIPv6) {
|
||||||
gBS->CloseEvent (Wrap->TcpWrap.TxToken.CompletionToken.Event);
|
if (Wrap->TcpWrap.Tx4Token.CompletionToken.Event != NULL) {
|
||||||
|
gBS->CloseEvent (Wrap->TcpWrap.Tx4Token.CompletionToken.Event);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (Wrap->TcpWrap.Rx4Token.CompletionToken.Event != NULL) {
|
||||||
|
gBS->CloseEvent (Wrap->TcpWrap.Rx4Token.CompletionToken.Event);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (Wrap->TcpWrap.Rx4Token.Packet.RxData->FragmentTable[0].FragmentBuffer != NULL) {
|
||||||
|
FreePool (Wrap->TcpWrap.Rx4Token.Packet.RxData->FragmentTable[0].FragmentBuffer);
|
||||||
|
}
|
||||||
|
|
||||||
|
} else {
|
||||||
|
if (Wrap->TcpWrap.Tx6Token.CompletionToken.Event != NULL) {
|
||||||
|
gBS->CloseEvent (Wrap->TcpWrap.Tx6Token.CompletionToken.Event);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (Wrap->TcpWrap.Rx6Token.CompletionToken.Event != NULL) {
|
||||||
|
gBS->CloseEvent (Wrap->TcpWrap.Rx6Token.CompletionToken.Event);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (Wrap->TcpWrap.Rx6Token.Packet.RxData->FragmentTable[0].FragmentBuffer != NULL) {
|
||||||
|
FreePool (Wrap->TcpWrap.Rx6Token.Packet.RxData->FragmentTable[0].FragmentBuffer);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (Wrap->TcpWrap.RxToken.CompletionToken.Event != NULL) {
|
|
||||||
gBS->CloseEvent (Wrap->TcpWrap.RxToken.CompletionToken.Event);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (Wrap->TcpWrap.RxToken.Packet.RxData->FragmentTable[0].FragmentBuffer != NULL) {
|
|
||||||
FreePool (Wrap->TcpWrap.RxToken.Packet.RxData->FragmentTable[0].FragmentBuffer);
|
|
||||||
}
|
|
||||||
|
|
||||||
FreePool (Wrap);
|
FreePool (Wrap);
|
||||||
|
|
||||||
@ -747,7 +777,7 @@ HttpBodyParserCallback (
|
|||||||
Wrap->HttpInstance->NextMsg = Data;
|
Wrap->HttpInstance->NextMsg = Data;
|
||||||
|
|
||||||
//
|
//
|
||||||
// Free TxToken since already received corrsponding HTTP response.
|
// Free Tx4Token or Tx6Token since already received corrsponding HTTP response.
|
||||||
//
|
//
|
||||||
FreePool (Wrap);
|
FreePool (Wrap);
|
||||||
|
|
||||||
@ -761,7 +791,7 @@ HttpBodyParserCallback (
|
|||||||
|
|
||||||
@retval EFI_SUCCESS Allocation succeeded.
|
@retval EFI_SUCCESS Allocation succeeded.
|
||||||
@retval EFI_OUT_OF_RESOURCES Failed to complete the opration due to lack of resources.
|
@retval EFI_OUT_OF_RESOURCES Failed to complete the opration due to lack of resources.
|
||||||
@retval EFI_NOT_READY Can't find a corresponding TxToken or
|
@retval EFI_NOT_READY Can't find a corresponding Tx4Token/Tx6Token or
|
||||||
the EFI_HTTP_UTILITIES_PROTOCOL is not available.
|
the EFI_HTTP_UTILITIES_PROTOCOL is not available.
|
||||||
|
|
||||||
**/
|
**/
|
||||||
@ -772,12 +802,9 @@ HttpResponseWorker (
|
|||||||
{
|
{
|
||||||
EFI_STATUS Status;
|
EFI_STATUS Status;
|
||||||
EFI_HTTP_MESSAGE *HttpMsg;
|
EFI_HTTP_MESSAGE *HttpMsg;
|
||||||
EFI_TCP4_IO_TOKEN *RxToken;
|
|
||||||
EFI_TCP4_PROTOCOL *Tcp4;
|
|
||||||
CHAR8 *EndofHeader;
|
CHAR8 *EndofHeader;
|
||||||
CHAR8 *HttpHeaders;
|
CHAR8 *HttpHeaders;
|
||||||
UINTN SizeofHeaders;
|
UINTN SizeofHeaders;
|
||||||
CHAR8 *Buffer;
|
|
||||||
UINTN BufferSize;
|
UINTN BufferSize;
|
||||||
UINTN StatusCode;
|
UINTN StatusCode;
|
||||||
CHAR8 *Tmp;
|
CHAR8 *Tmp;
|
||||||
@ -796,23 +823,21 @@ HttpResponseWorker (
|
|||||||
|
|
||||||
HttpInstance = Wrap->HttpInstance;
|
HttpInstance = Wrap->HttpInstance;
|
||||||
Token = Wrap->HttpToken;
|
Token = Wrap->HttpToken;
|
||||||
|
|
||||||
HttpMsg = Token->Message;
|
HttpMsg = Token->Message;
|
||||||
|
|
||||||
Tcp4 = HttpInstance->Tcp4;
|
HttpInstance->EndofHeader = NULL;
|
||||||
ASSERT (Tcp4 != NULL);
|
HttpInstance->HttpHeaders = NULL;
|
||||||
HttpMsg->Headers = NULL;
|
HttpMsg->Headers = NULL;
|
||||||
HttpHeaders = NULL;
|
HttpHeaders = NULL;
|
||||||
SizeofHeaders = 0;
|
SizeofHeaders = 0;
|
||||||
Buffer = NULL;
|
BufferSize = 0;
|
||||||
BufferSize = 0;
|
EndofHeader = NULL;
|
||||||
EndofHeader = NULL;
|
|
||||||
|
|
||||||
if (HttpMsg->Data.Response != NULL) {
|
if (HttpMsg->Data.Response != NULL) {
|
||||||
//
|
//
|
||||||
// Need receive the HTTP headers, prepare buffer.
|
// Need receive the HTTP headers, prepare buffer.
|
||||||
//
|
//
|
||||||
Status = HttpCreateTcp4RxEventForHeader (HttpInstance);
|
Status = HttpCreateTcpRxEventForHeader (HttpInstance);
|
||||||
if (EFI_ERROR (Status)) {
|
if (EFI_ERROR (Status)) {
|
||||||
goto Error;
|
goto Error;
|
||||||
}
|
}
|
||||||
@ -843,71 +868,16 @@ HttpResponseWorker (
|
|||||||
// Check whether we cached the whole HTTP headers.
|
// Check whether we cached the whole HTTP headers.
|
||||||
//
|
//
|
||||||
EndofHeader = AsciiStrStr (HttpHeaders, HTTP_END_OF_HDR_STR);
|
EndofHeader = AsciiStrStr (HttpHeaders, HTTP_END_OF_HDR_STR);
|
||||||
}
|
}
|
||||||
|
|
||||||
RxToken = &HttpInstance->RxToken;
|
HttpInstance->EndofHeader = &EndofHeader;
|
||||||
RxToken->Packet.RxData->FragmentTable[0].FragmentBuffer = AllocateZeroPool (DEF_BUF_LEN);
|
HttpInstance->HttpHeaders = &HttpHeaders;
|
||||||
if (RxToken->Packet.RxData->FragmentTable[0].FragmentBuffer == NULL) {
|
|
||||||
Status = EFI_OUT_OF_RESOURCES;
|
Status = HttpTcpReceiveHeader (HttpInstance, &SizeofHeaders, &BufferSize);
|
||||||
|
if (EFI_ERROR (Status)) {
|
||||||
goto Error;
|
goto Error;
|
||||||
}
|
}
|
||||||
|
|
||||||
//
|
|
||||||
// Receive the HTTP headers only when EFI_HTTP_RESPONSE_DATA is not NULL.
|
|
||||||
//
|
|
||||||
while (EndofHeader == NULL) {
|
|
||||||
HttpInstance->IsRxDone = FALSE;
|
|
||||||
RxToken->Packet.RxData->DataLength = DEF_BUF_LEN;
|
|
||||||
RxToken->Packet.RxData->FragmentTable[0].FragmentLength = DEF_BUF_LEN;
|
|
||||||
Status = Tcp4->Receive (Tcp4, RxToken);
|
|
||||||
if (EFI_ERROR (Status)) {
|
|
||||||
DEBUG ((EFI_D_ERROR, "Tcp4 receive failed: %r\n", Status));
|
|
||||||
goto Error;
|
|
||||||
}
|
|
||||||
|
|
||||||
while (!HttpInstance->IsRxDone) {
|
|
||||||
Tcp4->Poll (Tcp4);
|
|
||||||
}
|
|
||||||
|
|
||||||
Status = RxToken->CompletionToken.Status;
|
|
||||||
if (EFI_ERROR (Status)) {
|
|
||||||
goto Error;
|
|
||||||
}
|
|
||||||
|
|
||||||
//
|
|
||||||
// Append the response string.
|
|
||||||
//
|
|
||||||
BufferSize = SizeofHeaders + RxToken->Packet.RxData->FragmentTable[0].FragmentLength;
|
|
||||||
Buffer = AllocateZeroPool (BufferSize);
|
|
||||||
if (Buffer == NULL) {
|
|
||||||
Status = EFI_OUT_OF_RESOURCES;
|
|
||||||
goto Error;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (HttpHeaders != NULL) {
|
|
||||||
CopyMem (Buffer, HttpHeaders, SizeofHeaders);
|
|
||||||
FreePool (HttpHeaders);
|
|
||||||
}
|
|
||||||
|
|
||||||
CopyMem (
|
|
||||||
Buffer + SizeofHeaders,
|
|
||||||
RxToken->Packet.RxData->FragmentTable[0].FragmentBuffer,
|
|
||||||
RxToken->Packet.RxData->FragmentTable[0].FragmentLength
|
|
||||||
);
|
|
||||||
HttpHeaders = Buffer;
|
|
||||||
SizeofHeaders = BufferSize;
|
|
||||||
|
|
||||||
//
|
|
||||||
// Check whether we received end of HTTP headers.
|
|
||||||
//
|
|
||||||
EndofHeader = AsciiStrStr (HttpHeaders, HTTP_END_OF_HDR_STR);
|
|
||||||
};
|
|
||||||
|
|
||||||
//
|
|
||||||
// Skip the CRLF after the HTTP headers.
|
|
||||||
//
|
|
||||||
EndofHeader = EndofHeader + AsciiStrLen (HTTP_END_OF_HDR_STR);
|
|
||||||
|
|
||||||
//
|
//
|
||||||
// Cache the part of body.
|
// Cache the part of body.
|
||||||
//
|
//
|
||||||
@ -927,9 +897,6 @@ HttpResponseWorker (
|
|||||||
HttpInstance->CacheLen = BodyLen;
|
HttpInstance->CacheLen = BodyLen;
|
||||||
}
|
}
|
||||||
|
|
||||||
FreePool (RxToken->Packet.RxData->FragmentTable[0].FragmentBuffer);
|
|
||||||
RxToken->Packet.RxData->FragmentTable[0].FragmentBuffer = NULL;
|
|
||||||
|
|
||||||
//
|
//
|
||||||
// Search for Status Code.
|
// Search for Status Code.
|
||||||
//
|
//
|
||||||
@ -997,7 +964,7 @@ HttpResponseWorker (
|
|||||||
}
|
}
|
||||||
|
|
||||||
//
|
//
|
||||||
// The first TxToken not transmitted yet, insert back and return error.
|
// The first Tx Token not transmitted yet, insert back and return error.
|
||||||
//
|
//
|
||||||
if (!ValueInItem->TcpWrap.IsTxDone) {
|
if (!ValueInItem->TcpWrap.IsTxDone) {
|
||||||
goto Error2;
|
goto Error2;
|
||||||
@ -1108,16 +1075,8 @@ HttpResponseWorker (
|
|||||||
//
|
//
|
||||||
// We still need receive more data when there is no cache data and MsgParser is not NULL;
|
// We still need receive more data when there is no cache data and MsgParser is not NULL;
|
||||||
//
|
//
|
||||||
RxToken = &Wrap->TcpWrap.RxToken;
|
Status = HttpTcpReceiveBody (Wrap, HttpMsg);
|
||||||
|
|
||||||
RxToken->Packet.RxData->DataLength = (UINT32) HttpMsg->BodyLength;
|
|
||||||
RxToken->Packet.RxData->FragmentTable[0].FragmentLength = (UINT32) HttpMsg->BodyLength;
|
|
||||||
RxToken->Packet.RxData->FragmentTable[0].FragmentBuffer = (VOID *) HttpMsg->Body;
|
|
||||||
|
|
||||||
RxToken->CompletionToken.Status = EFI_NOT_READY;
|
|
||||||
Status = Tcp4->Receive (Tcp4, RxToken);
|
|
||||||
if (EFI_ERROR (Status)) {
|
if (EFI_ERROR (Status)) {
|
||||||
DEBUG ((EFI_D_ERROR, "Tcp4 receive failed: %r\n", Status));
|
|
||||||
goto Error;
|
goto Error;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1130,18 +1089,7 @@ Exit:
|
|||||||
}
|
}
|
||||||
Token->Status = Status;
|
Token->Status = Status;
|
||||||
gBS->SignalEvent (Token->Event);
|
gBS->SignalEvent (Token->Event);
|
||||||
|
HttpCloseTcpRxEvent (Wrap);
|
||||||
if (Wrap != NULL) {
|
|
||||||
if (Wrap->TcpWrap.RxToken.CompletionToken.Event != NULL) {
|
|
||||||
gBS->CloseEvent (Wrap->TcpWrap.RxToken.CompletionToken.Event);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (HttpInstance->RxToken.CompletionToken.Event != NULL) {
|
|
||||||
gBS->CloseEvent (HttpInstance->RxToken.CompletionToken.Event);
|
|
||||||
HttpInstance->RxToken.CompletionToken.Event = NULL;
|
|
||||||
}
|
|
||||||
|
|
||||||
FreePool (Wrap);
|
FreePool (Wrap);
|
||||||
return Status;
|
return Status;
|
||||||
|
|
||||||
@ -1149,28 +1097,7 @@ Error2:
|
|||||||
NetMapInsertHead (&HttpInstance->TxTokens, ValueInItem->HttpToken, ValueInItem);
|
NetMapInsertHead (&HttpInstance->TxTokens, ValueInItem->HttpToken, ValueInItem);
|
||||||
|
|
||||||
Error:
|
Error:
|
||||||
if (Wrap != NULL) {
|
HttpTcpTokenCleanup (Wrap);
|
||||||
if (Wrap->TcpWrap.RxToken.CompletionToken.Event != NULL) {
|
|
||||||
gBS->CloseEvent (Wrap->TcpWrap.RxToken.CompletionToken.Event);
|
|
||||||
}
|
|
||||||
RxToken = &Wrap->TcpWrap.RxToken;
|
|
||||||
if (RxToken->Packet.RxData->FragmentTable[0].FragmentBuffer != NULL) {
|
|
||||||
FreePool (RxToken->Packet.RxData->FragmentTable[0].FragmentBuffer);
|
|
||||||
RxToken->Packet.RxData->FragmentTable[0].FragmentBuffer = NULL;
|
|
||||||
}
|
|
||||||
FreePool (Wrap);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (HttpInstance->RxToken.CompletionToken.Event != NULL) {
|
|
||||||
gBS->CloseEvent (HttpInstance->RxToken.CompletionToken.Event);
|
|
||||||
HttpInstance->RxToken.CompletionToken.Event = NULL;
|
|
||||||
}
|
|
||||||
|
|
||||||
RxToken = &HttpInstance->RxToken;
|
|
||||||
if (RxToken->Packet.RxData->FragmentTable[0].FragmentBuffer != NULL) {
|
|
||||||
FreePool (RxToken->Packet.RxData->FragmentTable[0].FragmentBuffer);
|
|
||||||
RxToken->Packet.RxData->FragmentTable[0].FragmentBuffer = NULL;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (HttpHeaders != NULL) {
|
if (HttpHeaders != NULL) {
|
||||||
FreePool (HttpHeaders);
|
FreePool (HttpHeaders);
|
||||||
@ -1268,10 +1195,6 @@ EfiHttpResponse (
|
|||||||
return EFI_NOT_STARTED;
|
return EFI_NOT_STARTED;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (HttpInstance->LocalAddressIsIPv6) {
|
|
||||||
return EFI_UNSUPPORTED;
|
|
||||||
}
|
|
||||||
|
|
||||||
//
|
//
|
||||||
// Check whether the token already existed.
|
// Check whether the token already existed.
|
||||||
//
|
//
|
||||||
@ -1287,7 +1210,7 @@ EfiHttpResponse (
|
|||||||
Wrap->HttpInstance = HttpInstance;
|
Wrap->HttpInstance = HttpInstance;
|
||||||
Wrap->HttpToken = Token;
|
Wrap->HttpToken = Token;
|
||||||
|
|
||||||
Status = HttpCreateTcp4RxEvent (Wrap);
|
Status = HttpCreateTcpRxEvent (Wrap);
|
||||||
if (EFI_ERROR (Status)) {
|
if (EFI_ERROR (Status)) {
|
||||||
goto Error;
|
goto Error;
|
||||||
}
|
}
|
||||||
@ -1308,8 +1231,12 @@ EfiHttpResponse (
|
|||||||
|
|
||||||
Error:
|
Error:
|
||||||
if (Wrap != NULL) {
|
if (Wrap != NULL) {
|
||||||
if (Wrap->TcpWrap.RxToken.CompletionToken.Event != NULL) {
|
if (Wrap->TcpWrap.Rx4Token.CompletionToken.Event != NULL) {
|
||||||
gBS->CloseEvent (Wrap->TcpWrap.RxToken.CompletionToken.Event);
|
gBS->CloseEvent (Wrap->TcpWrap.Rx4Token.CompletionToken.Event);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (Wrap->TcpWrap.Rx6Token.CompletionToken.Event != NULL) {
|
||||||
|
gBS->CloseEvent (Wrap->TcpWrap.Rx6Token.CompletionToken.Event);
|
||||||
}
|
}
|
||||||
FreePool (Wrap);
|
FreePool (Wrap);
|
||||||
}
|
}
|
||||||
@ -1343,8 +1270,8 @@ EfiHttpPoll (
|
|||||||
IN EFI_HTTP_PROTOCOL *This
|
IN EFI_HTTP_PROTOCOL *This
|
||||||
)
|
)
|
||||||
{
|
{
|
||||||
HTTP_PROTOCOL *HttpInstance;
|
|
||||||
EFI_STATUS Status;
|
EFI_STATUS Status;
|
||||||
|
HTTP_PROTOCOL *HttpInstance;
|
||||||
|
|
||||||
if (This == NULL) {
|
if (This == NULL) {
|
||||||
return EFI_INVALID_PARAMETER;
|
return EFI_INVALID_PARAMETER;
|
||||||
@ -1353,17 +1280,18 @@ EfiHttpPoll (
|
|||||||
HttpInstance = HTTP_INSTANCE_FROM_PROTOCOL (This);
|
HttpInstance = HTTP_INSTANCE_FROM_PROTOCOL (This);
|
||||||
ASSERT (HttpInstance != NULL);
|
ASSERT (HttpInstance != NULL);
|
||||||
|
|
||||||
if (HttpInstance->LocalAddressIsIPv6) {
|
if (HttpInstance->State != HTTP_STATE_TCP_CONNECTED || (HttpInstance->Tcp4 == NULL &&
|
||||||
return EFI_UNSUPPORTED;
|
HttpInstance->Tcp6 == NULL)) {
|
||||||
}
|
|
||||||
|
|
||||||
if (HttpInstance->Tcp4 == NULL || HttpInstance->State != HTTP_STATE_TCP_CONNECTED) {
|
|
||||||
return EFI_NOT_STARTED;
|
return EFI_NOT_STARTED;
|
||||||
}
|
}
|
||||||
|
|
||||||
Status = HttpInstance->Tcp4->Poll (HttpInstance->Tcp4);
|
if (HttpInstance->LocalAddressIsIPv6) {
|
||||||
|
Status = HttpInstance->Tcp6->Poll (HttpInstance->Tcp6);
|
||||||
|
} else {
|
||||||
|
Status = HttpInstance->Tcp4->Poll (HttpInstance->Tcp4);
|
||||||
|
}
|
||||||
|
|
||||||
DispatchDpc ();
|
DispatchDpc ();
|
||||||
|
|
||||||
return Status;
|
return Status;
|
||||||
}
|
}
|
||||||
|
File diff suppressed because it is too large
Load Diff
@ -27,6 +27,7 @@ WITHOUT WARRANTIES OR REPRESENTATIONS OF ANY KIND, EITHER EXPRESS OR IMPLIED.
|
|||||||
HTTP_SERVICE_SIGNATURE \
|
HTTP_SERVICE_SIGNATURE \
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
//
|
//
|
||||||
// The state of HTTP protocol. It starts from UNCONFIGED.
|
// The state of HTTP protocol. It starts from UNCONFIGED.
|
||||||
//
|
//
|
||||||
@ -58,18 +59,23 @@ typedef struct _HTTP_SERVICE {
|
|||||||
EFI_SERVICE_BINDING_PROTOCOL ServiceBinding;
|
EFI_SERVICE_BINDING_PROTOCOL ServiceBinding;
|
||||||
EFI_HANDLE ImageHandle;
|
EFI_HANDLE ImageHandle;
|
||||||
EFI_HANDLE ControllerHandle;
|
EFI_HANDLE ControllerHandle;
|
||||||
|
EFI_HANDLE Tcp4ChildHandle;
|
||||||
|
EFI_HANDLE Tcp6ChildHandle;
|
||||||
LIST_ENTRY ChildrenList;
|
LIST_ENTRY ChildrenList;
|
||||||
UINTN ChildrenNumber;
|
UINTN ChildrenNumber;
|
||||||
EFI_HANDLE TcpChildHandle;
|
|
||||||
INTN State;
|
INTN State;
|
||||||
} HTTP_SERVICE;
|
} HTTP_SERVICE;
|
||||||
|
|
||||||
typedef struct {
|
typedef struct {
|
||||||
EFI_TCP4_IO_TOKEN TxToken;
|
EFI_TCP4_IO_TOKEN Tx4Token;
|
||||||
EFI_TCP4_TRANSMIT_DATA TxData;
|
EFI_TCP4_TRANSMIT_DATA Tx4Data;
|
||||||
|
EFI_TCP6_IO_TOKEN Tx6Token;
|
||||||
|
EFI_TCP6_TRANSMIT_DATA Tx6Data;
|
||||||
|
EFI_TCP4_IO_TOKEN Rx4Token;
|
||||||
|
EFI_TCP4_RECEIVE_DATA Rx4Data;
|
||||||
|
EFI_TCP6_IO_TOKEN Rx6Token;
|
||||||
|
EFI_TCP6_RECEIVE_DATA Rx6Data;
|
||||||
BOOLEAN IsTxDone;
|
BOOLEAN IsTxDone;
|
||||||
EFI_TCP4_IO_TOKEN RxToken;
|
|
||||||
EFI_TCP4_RECEIVE_DATA RxData;
|
|
||||||
BOOLEAN IsRxDone;
|
BOOLEAN IsRxDone;
|
||||||
UINTN BodyLen;
|
UINTN BodyLen;
|
||||||
EFI_HTTP_METHOD Method;
|
EFI_HTTP_METHOD Method;
|
||||||
@ -84,26 +90,43 @@ typedef struct _HTTP_PROTOCOL {
|
|||||||
BOOLEAN InDestroy;
|
BOOLEAN InDestroy;
|
||||||
INTN State;
|
INTN State;
|
||||||
|
|
||||||
EFI_HANDLE TcpChildHandle;
|
EFI_HANDLE Tcp4ChildHandle;
|
||||||
EFI_TCP4_PROTOCOL *Tcp4;
|
EFI_TCP4_PROTOCOL *Tcp4;
|
||||||
EFI_TCP4_CONFIG_DATA Tcp4CfgData;
|
EFI_TCP4_CONFIG_DATA Tcp4CfgData;
|
||||||
EFI_TCP4_OPTION Tcp4Option;
|
EFI_TCP4_OPTION Tcp4Option;
|
||||||
|
|
||||||
EFI_TCP4_CONNECTION_TOKEN ConnToken;
|
EFI_TCP4_CONNECTION_TOKEN Tcp4ConnToken;
|
||||||
BOOLEAN IsConnDone;
|
BOOLEAN IsTcp4ConnDone;
|
||||||
EFI_TCP4_CLOSE_TOKEN CloseToken;
|
EFI_TCP4_CLOSE_TOKEN Tcp4CloseToken;
|
||||||
BOOLEAN IsCloseDone;
|
BOOLEAN IsTcp4CloseDone;
|
||||||
|
|
||||||
CHAR8 *RemoteHost;
|
CHAR8 *RemoteHost;
|
||||||
UINT16 RemotePort;
|
UINT16 RemotePort;
|
||||||
EFI_IPv4_ADDRESS RemoteAddr;
|
EFI_IPv4_ADDRESS RemoteAddr;
|
||||||
|
|
||||||
|
EFI_HANDLE Tcp6ChildHandle;
|
||||||
|
EFI_TCP6_PROTOCOL *Tcp6;
|
||||||
|
EFI_TCP6_CONFIG_DATA Tcp6CfgData;
|
||||||
|
EFI_TCP6_OPTION Tcp6Option;
|
||||||
|
|
||||||
|
EFI_TCP6_CONNECTION_TOKEN Tcp6ConnToken;
|
||||||
|
BOOLEAN IsTcp6ConnDone;
|
||||||
|
EFI_TCP6_CLOSE_TOKEN Tcp6CloseToken;
|
||||||
|
BOOLEAN IsTcp6CloseDone;
|
||||||
|
EFI_IPv6_ADDRESS RemoteIpv6Addr;
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
//
|
//
|
||||||
// RxToken used for receiving HTTP header.
|
// Rx4Token or Rx6Token used for receiving HTTP header.
|
||||||
//
|
//
|
||||||
EFI_TCP4_IO_TOKEN RxToken;
|
EFI_TCP4_IO_TOKEN Rx4Token;
|
||||||
EFI_TCP4_RECEIVE_DATA RxData;
|
EFI_TCP4_RECEIVE_DATA Rx4Data;
|
||||||
|
EFI_TCP6_IO_TOKEN Rx6Token;
|
||||||
|
EFI_TCP6_RECEIVE_DATA Rx6Data;
|
||||||
BOOLEAN IsRxDone;
|
BOOLEAN IsRxDone;
|
||||||
|
|
||||||
|
CHAR8 **EndofHeader;
|
||||||
|
CHAR8 **HttpHeaders;
|
||||||
CHAR8 *CacheBody;
|
CHAR8 *CacheBody;
|
||||||
CHAR8 *NextMsg;
|
CHAR8 *NextMsg;
|
||||||
UINTN CacheLen;
|
UINTN CacheLen;
|
||||||
@ -119,6 +142,7 @@ typedef struct _HTTP_PROTOCOL {
|
|||||||
BOOLEAN LocalAddressIsIPv6;
|
BOOLEAN LocalAddressIsIPv6;
|
||||||
|
|
||||||
EFI_HTTPv4_ACCESS_POINT IPv4Node;
|
EFI_HTTPv4_ACCESS_POINT IPv4Node;
|
||||||
|
EFI_HTTPv6_ACCESS_POINT Ipv6Node;
|
||||||
|
|
||||||
NET_MAP TxTokens;
|
NET_MAP TxTokens;
|
||||||
NET_MAP RxTokens;
|
NET_MAP RxTokens;
|
||||||
@ -158,7 +182,7 @@ HttpCommonNotify (
|
|||||||
);
|
);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
Create events for the TCP4 connection token and TCP4 close token.
|
Create events for the TCP connection token and TCP close token.
|
||||||
|
|
||||||
@param[in] HttpInstance Pointer to HTTP_PROTOCOL structure.
|
@param[in] HttpInstance Pointer to HTTP_PROTOCOL structure.
|
||||||
|
|
||||||
@ -167,23 +191,23 @@ HttpCommonNotify (
|
|||||||
|
|
||||||
**/
|
**/
|
||||||
EFI_STATUS
|
EFI_STATUS
|
||||||
HttpCreateTcp4ConnCloseEvent (
|
HttpCreateTcpConnCloseEvent (
|
||||||
IN HTTP_PROTOCOL *HttpInstance
|
IN HTTP_PROTOCOL *HttpInstance
|
||||||
);
|
);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
Close events in the TCP4 connection token and TCP4 close token.
|
Close events in the TCP connection token and TCP close token.
|
||||||
|
|
||||||
@param[in] HttpInstance Pointer to HTTP_PROTOCOL structure.
|
@param[in] HttpInstance Pointer to HTTP_PROTOCOL structure.
|
||||||
|
|
||||||
**/
|
**/
|
||||||
VOID
|
VOID
|
||||||
HttpCloseTcp4ConnCloseEvent (
|
HttpCloseTcpConnCloseEvent (
|
||||||
IN HTTP_PROTOCOL *HttpInstance
|
IN HTTP_PROTOCOL *HttpInstance
|
||||||
);
|
);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
Create event for the TCP4 transmit token.
|
Create event for the TCP transmit token.
|
||||||
|
|
||||||
@param[in] Wrap Point to HTTP token's wrap data.
|
@param[in] Wrap Point to HTTP token's wrap data.
|
||||||
|
|
||||||
@ -192,12 +216,12 @@ HttpCloseTcp4ConnCloseEvent (
|
|||||||
|
|
||||||
**/
|
**/
|
||||||
EFI_STATUS
|
EFI_STATUS
|
||||||
HttpCreateTcp4TxEvent (
|
HttpCreateTcpTxEvent (
|
||||||
IN HTTP_TOKEN_WRAP *Wrap
|
IN HTTP_TOKEN_WRAP *Wrap
|
||||||
);
|
);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
Create event for the TCP4 receive token which is used to receive HTTP header.
|
Create event for the TCP receive token which is used to receive HTTP header.
|
||||||
|
|
||||||
@param[in] HttpInstance Pointer to HTTP_PROTOCOL structure.
|
@param[in] HttpInstance Pointer to HTTP_PROTOCOL structure.
|
||||||
|
|
||||||
@ -206,12 +230,12 @@ HttpCreateTcp4TxEvent (
|
|||||||
|
|
||||||
**/
|
**/
|
||||||
EFI_STATUS
|
EFI_STATUS
|
||||||
HttpCreateTcp4RxEventForHeader (
|
HttpCreateTcpRxEventForHeader (
|
||||||
IN HTTP_PROTOCOL *HttpInstance
|
IN HTTP_PROTOCOL *HttpInstance
|
||||||
);
|
);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
Create event for the TCP4 receive token which is used to receive HTTP body.
|
Create event for the TCP receive token which is used to receive HTTP body.
|
||||||
|
|
||||||
@param[in] Wrap Point to HTTP token's wrap data.
|
@param[in] Wrap Point to HTTP token's wrap data.
|
||||||
|
|
||||||
@ -220,15 +244,26 @@ HttpCreateTcp4RxEventForHeader (
|
|||||||
|
|
||||||
**/
|
**/
|
||||||
EFI_STATUS
|
EFI_STATUS
|
||||||
HttpCreateTcp4RxEvent (
|
HttpCreateTcpRxEvent (
|
||||||
IN HTTP_TOKEN_WRAP *Wrap
|
IN HTTP_TOKEN_WRAP *Wrap
|
||||||
);
|
);
|
||||||
|
|
||||||
|
/**
|
||||||
|
Close Events for Tcp Receive Tokens for HTTP body and HTTP header.
|
||||||
|
|
||||||
|
@param[in] Wrap Pointer to HTTP token's wrap data.
|
||||||
|
|
||||||
|
**/
|
||||||
|
VOID
|
||||||
|
HttpCloseTcpRxEvent (
|
||||||
|
IN HTTP_TOKEN_WRAP *Wrap
|
||||||
|
);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
Intiialize the HTTP_PROTOCOL structure to the unconfigured state.
|
Intiialize the HTTP_PROTOCOL structure to the unconfigured state.
|
||||||
|
|
||||||
@param[in] HttpSb The HTTP service private instance.
|
|
||||||
@param[in, out] HttpInstance Pointer to HTTP_PROTOCOL structure.
|
@param[in, out] HttpInstance Pointer to HTTP_PROTOCOL structure.
|
||||||
|
@param[in] IpVersion Indicate us TCP4 protocol or TCP6 protocol.
|
||||||
|
|
||||||
@retval EFI_SUCCESS HTTP_PROTOCOL structure is initialized successfully.
|
@retval EFI_SUCCESS HTTP_PROTOCOL structure is initialized successfully.
|
||||||
@retval Others Other error as indicated.
|
@retval Others Other error as indicated.
|
||||||
@ -236,8 +271,8 @@ HttpCreateTcp4RxEvent (
|
|||||||
**/
|
**/
|
||||||
EFI_STATUS
|
EFI_STATUS
|
||||||
HttpInitProtocol (
|
HttpInitProtocol (
|
||||||
IN HTTP_SERVICE *HttpSb,
|
IN OUT HTTP_PROTOCOL *HttpInstance,
|
||||||
IN OUT HTTP_PROTOCOL *HttpInstance
|
IN BOOLEAN IpVersion
|
||||||
);
|
);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -295,6 +330,22 @@ HttpConfigureTcp4 (
|
|||||||
IN HTTP_TOKEN_WRAP *Wrap
|
IN HTTP_TOKEN_WRAP *Wrap
|
||||||
);
|
);
|
||||||
|
|
||||||
|
/**
|
||||||
|
Configure TCP6 protocol child.
|
||||||
|
|
||||||
|
@param[in] HttpInstance The HTTP instance private data.
|
||||||
|
@param[in] Wrap The HTTP token's wrap data.
|
||||||
|
|
||||||
|
@retval EFI_SUCCESS The TCP6 protocol child is configured.
|
||||||
|
@retval Others Other error as indicated.
|
||||||
|
|
||||||
|
**/
|
||||||
|
EFI_STATUS
|
||||||
|
HttpConfigureTcp6 (
|
||||||
|
IN HTTP_PROTOCOL *HttpInstance,
|
||||||
|
IN HTTP_TOKEN_WRAP *Wrap
|
||||||
|
);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
Check existing TCP connection, if in error state, receover TCP4 connection.
|
Check existing TCP connection, if in error state, receover TCP4 connection.
|
||||||
|
|
||||||
@ -311,7 +362,22 @@ HttpConnectTcp4 (
|
|||||||
);
|
);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
Send the HTTP message through TCP4.
|
Check existing TCP connection, if in error state, recover TCP6 connection.
|
||||||
|
|
||||||
|
@param[in] HttpInstance The HTTP instance private data.
|
||||||
|
|
||||||
|
@retval EFI_SUCCESS The TCP connection is established.
|
||||||
|
@retval EFI_NOT_READY TCP6 protocol child is not created or configured.
|
||||||
|
@retval Others Other error as indicated.
|
||||||
|
|
||||||
|
**/
|
||||||
|
EFI_STATUS
|
||||||
|
HttpConnectTcp6 (
|
||||||
|
IN HTTP_PROTOCOL *HttpInstance
|
||||||
|
);
|
||||||
|
|
||||||
|
/**
|
||||||
|
Send the HTTP message through TCP4 or TCP6.
|
||||||
|
|
||||||
@param[in] HttpInstance The HTTP instance private data.
|
@param[in] HttpInstance The HTTP instance private data.
|
||||||
@param[in] Wrap The HTTP token's wrap data.
|
@param[in] Wrap The HTTP token's wrap data.
|
||||||
@ -323,7 +389,7 @@ HttpConnectTcp4 (
|
|||||||
|
|
||||||
**/
|
**/
|
||||||
EFI_STATUS
|
EFI_STATUS
|
||||||
HttpTransmitTcp4 (
|
HttpTransmitTcp (
|
||||||
IN HTTP_PROTOCOL *HttpInstance,
|
IN HTTP_PROTOCOL *HttpInstance,
|
||||||
IN HTTP_TOKEN_WRAP *Wrap,
|
IN HTTP_TOKEN_WRAP *Wrap,
|
||||||
IN UINT8 *TxString,
|
IN UINT8 *TxString,
|
||||||
@ -346,7 +412,7 @@ HttpMappingToStatusCode (
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
Check whether the user's token or event has already
|
Check whether the user's token or event has already
|
||||||
been enqueue on HTTP TxToken or RxToken list.
|
been enqueue on HTTP Tx or Rx Token list.
|
||||||
|
|
||||||
@param[in] Map The container of either user's transmit or receive
|
@param[in] Map The container of either user's transmit or receive
|
||||||
token.
|
token.
|
||||||
@ -367,7 +433,7 @@ HttpTokenExist (
|
|||||||
);
|
);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
Check whether the HTTP message associated with TxToken is already sent out.
|
Check whether the HTTP message associated with TxToken or Tx6Token is already sent out.
|
||||||
|
|
||||||
@param[in] Map The container of TxToken.
|
@param[in] Map The container of TxToken.
|
||||||
@param[in] Item Current item to check against.
|
@param[in] Item Current item to check against.
|
||||||
@ -385,10 +451,26 @@ HttpTcpNotReady (
|
|||||||
IN VOID *Context
|
IN VOID *Context
|
||||||
);
|
);
|
||||||
|
|
||||||
|
/**
|
||||||
|
Initialize TCP related data.
|
||||||
|
|
||||||
|
@param[in] HttpInstance The HTTP instance private data.
|
||||||
|
@param[in] Wrap The HTTP token's wrap data.
|
||||||
|
|
||||||
|
@retval EFI_SUCCESS The initialization of TCP instance is done.
|
||||||
|
@retval Others Other error as indicated.
|
||||||
|
|
||||||
|
**/
|
||||||
|
EFI_STATUS
|
||||||
|
HttpInitTcp (
|
||||||
|
IN HTTP_PROTOCOL *HttpInstance,
|
||||||
|
IN HTTP_TOKEN_WRAP *Wrap
|
||||||
|
);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
Transmit the HTTP mssage by processing the associated HTTP token.
|
Transmit the HTTP mssage by processing the associated HTTP token.
|
||||||
|
|
||||||
@param[in] Map The container of TxToken.
|
@param[in] Map The container of TxToken or Tx6Token.
|
||||||
@param[in] Item Current item to check against.
|
@param[in] Item Current item to check against.
|
||||||
@param[in] Context The Token to check againist.
|
@param[in] Context The Token to check againist.
|
||||||
|
|
||||||
@ -408,7 +490,7 @@ HttpTcpTransmit (
|
|||||||
/**
|
/**
|
||||||
Receive the HTTP response by processing the associated HTTP token.
|
Receive the HTTP response by processing the associated HTTP token.
|
||||||
|
|
||||||
@param[in] Map The container of RxToken.
|
@param[in] Map The container of Rx4Token or Rx6Token.
|
||||||
@param[in] Item Current item to check against.
|
@param[in] Item Current item to check against.
|
||||||
@param[in] Context The Token to check againist.
|
@param[in] Context The Token to check againist.
|
||||||
|
|
||||||
@ -425,6 +507,51 @@ HttpTcpReceive (
|
|||||||
IN VOID *Context
|
IN VOID *Context
|
||||||
);
|
);
|
||||||
|
|
||||||
|
/**
|
||||||
|
Receive the HTTP header by processing the associated HTTP token.
|
||||||
|
|
||||||
|
@param[in] HttpInstance The HTTP instance private data.
|
||||||
|
@param[in, out] SizeofHeaders The HTTP header length.
|
||||||
|
@param[in, out] BufferSize The size of buffer to cacahe the header message.
|
||||||
|
|
||||||
|
@retval EFI_SUCCESS The HTTP header is received.
|
||||||
|
@retval Others Other errors as indicated.
|
||||||
|
|
||||||
|
**/
|
||||||
|
EFI_STATUS
|
||||||
|
HttpTcpReceiveHeader (
|
||||||
|
IN HTTP_PROTOCOL *HttpInstance,
|
||||||
|
IN OUT UINTN *SizeofHeaders,
|
||||||
|
IN OUT UINTN *BufferSize
|
||||||
|
);
|
||||||
|
|
||||||
|
/**
|
||||||
|
Receive the HTTP body by processing the associated HTTP token.
|
||||||
|
|
||||||
|
@param[in] Wrap The HTTP token's wrap data.
|
||||||
|
@param[in] HttpMsg The HTTP message data.
|
||||||
|
|
||||||
|
@retval EFI_SUCCESS The HTTP body is received.
|
||||||
|
@retval Others Other error as indicated.
|
||||||
|
|
||||||
|
**/
|
||||||
|
EFI_STATUS
|
||||||
|
HttpTcpReceiveBody (
|
||||||
|
IN HTTP_TOKEN_WRAP *Wrap,
|
||||||
|
IN EFI_HTTP_MESSAGE *HttpMsg
|
||||||
|
);
|
||||||
|
|
||||||
|
/**
|
||||||
|
Clean up Tcp Tokens while the Tcp transmission error occurs.
|
||||||
|
|
||||||
|
@param[in] Wrap Pointer to HTTP token's wrap data.
|
||||||
|
|
||||||
|
**/
|
||||||
|
VOID
|
||||||
|
HttpTcpTokenCleanup (
|
||||||
|
IN HTTP_TOKEN_WRAP *Wrap
|
||||||
|
);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
Generate HTTP request string.
|
Generate HTTP request string.
|
||||||
|
|
||||||
|
Reference in New Issue
Block a user