1 В избранное 0 Ответвления 0

OSCHINA-MIRROR/zsl588-OpenCorePkg

Присоединиться к Gitlife
Откройте для себя и примите участие в публичных проектах с открытым исходным кодом с участием более 10 миллионов разработчиков. Приватные репозитории также полностью бесплатны :)
Присоединиться бесплатно
Клонировать/Скачать
OcHeciLib.c 14 КБ
Копировать Редактировать Исходные данные Просмотреть построчно История
vit9696 Отправлено 5 лет назад 8a4a443
123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601
/** @file
This file implements interaction with HECI.
Copyright (c) 2019, vit9696. All rights reserved.<BR>
Portions copyright (c) 2019, savvas. All rights reserved.<BR>
SPDX-License-Identifier: BSD-2-Clause
**/
#include <PiDxe.h>
#include <Library/BaseMemoryLib.h>
#include <Library/OcHeciLib.h>
#include <Library/OcDebugLogLib.h>
#include <Library/OcMiscLib.h>
#include <Library/UefiBootServicesTableLib.h>
#include <Protocol/Heci.h>
#include <Protocol/Heci2.h>
#include <IndustryStandard/AppleProvisioning.h>
#include <IndustryStandard/HeciMsg.h>
#include <IndustryStandard/HeciClientMsg.h>
STATIC UINT8 mCurrentMeClientRequestedReceiveMsg;
STATIC UINT8 mCurrentMeClientCanReceiveMsg;
STATIC UINT8 mCurrentMeClientAddress;
STATIC EFI_HECI_PROTOCOL *mHeci;
STATIC EFI_HECI2_PROTOCOL *mHeci2;
STATIC BOOLEAN mSendingHeciCommand;
STATIC BOOLEAN mSendingHeciCommandPerClient;
EFI_STATUS
HeciReadMessage (
IN UINT32 Blocking,
IN UINT32 *MessageBody,
IN OUT UINT32 *Length
)
{
if (mHeci != NULL) {
return mHeci->ReadMsg (
Blocking,
MessageBody,
Length
);
}
if (mHeci2 != NULL) {
return mHeci2->ReadMsg (
HECI_DEFAULT_DEVICE,
Blocking,
MessageBody,
Length
);
}
DEBUG ((DEBUG_INFO, "OCME: No ME protocol loaded, cannot read message\n"));
return EFI_NOT_FOUND;
}
EFI_STATUS
HeciSendMessage (
IN UINT32 *Message,
IN UINT32 Length,
IN UINT8 HostAddress,
IN UINT8 MEAddress
)
{
if (mHeci != NULL) {
return mHeci->SendMsg (
Message,
Length,
HostAddress,
MEAddress
);
}
if (mHeci2 != NULL) {
return mHeci2->SendMsg (
HECI_DEFAULT_DEVICE,
Message,
Length,
HostAddress,
MEAddress
);
}
DEBUG ((DEBUG_INFO, "OCME: No ME protocol loaded, cannot send message\n"));
return EFI_NOT_FOUND;
}
EFI_STATUS
HeciLocateProtocol (
VOID
)
{
EFI_STATUS Status;
if (mHeci != NULL || mHeci2 != NULL) {
return EFI_SUCCESS;
}
Status = gBS->LocateProtocol (
&gEfiHeciProtocolGuid,
NULL,
(VOID **) &mHeci
);
if (EFI_ERROR (Status)) {
DEBUG ((DEBUG_INFO, "OCME: Falling back to HECI 2 protocol - %r\n", Status));
//
// Ensure we don't have both set
//
mHeci = NULL;
Status = gBS->LocateProtocol (
&gEfiHeci2ProtocolGuid,
NULL,
(VOID **) &mHeci2
);
}
if (EFI_ERROR (Status)) {
DEBUG ((DEBUG_INFO, "OCME: Failed to find any HECI protocol - %r\n", Status));
}
return Status;
}
VOID
HeciUpdateReceiveMsgStatus (
VOID
)
{
EFI_STATUS Status;
UINT32 Size;
HBM_FLOW_CONTROL Command;
if (mSendingHeciCommandPerClient) {
ZeroMem (&Command, sizeof (Command));
Size = sizeof (Command);
Status = HeciReadMessage (
BLOCKING,
(UINT32 *) &Command,
&Size
);
if (!EFI_ERROR (Status) && Command.Command.Fields.Command == FLOW_CONTROL) {
++mCurrentMeClientCanReceiveMsg;
}
}
}
EFI_STATUS
HeciGetResponse (
OUT VOID *MessageData,
IN UINT32 ResponseSize
)
{
EFI_STATUS Status;
HBM_FLOW_CONTROL Command;
Status = EFI_NOT_READY;
STATIC_ASSERT (sizeof (HBM_FLOW_CONTROL) == 8, "Invalid ME command size");
if (mSendingHeciCommandPerClient || mSendingHeciCommand) {
ZeroMem (MessageData, ResponseSize);
//
// Note, this was reworked to make more sense.
// https://github.com/osy86/OpenCorePkg/commit/8f7188d41876109aec2fe3a721f69daf979dd268.diff
//
if (!mCurrentMeClientRequestedReceiveMsg) {
ZeroMem (&Command, sizeof (Command));
Command.Command.Fields.Command = FLOW_CONTROL;
Command.MeAddress = mCurrentMeClientAddress;
Command.HostAddress = HBM_CLIENT_ADDRESS;
Status = HeciSendMessage (
(UINT32 *) &Command,
sizeof (Command),
HBM_HOST_ADDRESS,
HBM_ME_ADDRESS
);
if (!EFI_ERROR (Status)) {
++mCurrentMeClientRequestedReceiveMsg;
}
}
Status = HeciReadMessage (
BLOCKING,
MessageData,
&ResponseSize
);
if (!EFI_ERROR (Status)) {
--mCurrentMeClientRequestedReceiveMsg;
}
}
return Status;
}
EFI_STATUS
HeciSendMessageWithResponse (
IN OUT VOID *MessageData,
IN UINT32 RequestSize,
IN UINT32 ResponseSize
)
{
HECI_BUS_MESSAGE *Message;
HBM_COMMAND Command;
EFI_STATUS Status;
mSendingHeciCommand = TRUE;
Message = (HECI_BUS_MESSAGE *) MessageData;
Command = Message->Command;
Status = HeciSendMessage (
MessageData,
RequestSize,
HBM_HOST_ADDRESS,
HBM_ME_ADDRESS
);
if (!EFI_ERROR (Status)) {
Status = HeciGetResponse (MessageData, ResponseSize);
if (!EFI_ERROR (Status)
&& Command.Fields.Command != Message->Command.Fields.Command) {
Status = EFI_PROTOCOL_ERROR;
}
}
mSendingHeciCommand = FALSE;
return Status;
}
EFI_STATUS
HeciGetClientMap (
OUT UINT8 *ClientMap,
OUT UINT8 *ClientActiveCount
)
{
EFI_STATUS Status;
HBM_HOST_ENUMERATION_BUFFER Command;
UINTN Index;
UINTN Index2;
UINT8 *ValidAddressesPtr;
UINT32 ValidAddresses;
*ClientActiveCount = 0;
Status = HeciLocateProtocol ();
if (EFI_ERROR (Status)) {
return Status;
}
STATIC_ASSERT (sizeof (Command.Request) == 4, "Invalid ME command size");
STATIC_ASSERT (sizeof (Command.Response) == 36, "Invalid ME command size");
ZeroMem (&Command, sizeof (Command));
Command.Request.Command.Fields.Command = HOST_ENUMERATION_REQUEST;
Status = HeciSendMessageWithResponse (
&Command,
sizeof (Command.Request),
sizeof (Command.Response)
);
if (EFI_ERROR (Status)) {
return Status;
}
ValidAddressesPtr = &Command.Response.ValidAddresses[0];
for (Index = 0; Index < HBM_ME_CLIENT_MAX; Index += OC_CHAR_BIT) {
ValidAddresses = *ValidAddressesPtr;
for (Index2 = 0; Index2 < OC_CHAR_BIT; Index2++) {
if ((ValidAddresses & (1U << Index2)) != 0) {
ClientMap[*ClientActiveCount] = (UINT8) (Index + Index2);
++(*ClientActiveCount);
}
}
++ValidAddressesPtr;
}
return Status;
}
EFI_STATUS
HeciGetClientProperties (
IN UINT8 Address,
OUT HECI_CLIENT_PROPERTIES *Properties
)
{
EFI_STATUS Status;
HBM_HOST_CLIENT_PROPERTIES_BUFFER Command;
Status = HeciLocateProtocol ();
if (EFI_ERROR (Status)) {
return Status;
}
STATIC_ASSERT (sizeof (Command.Request) == 4, "Invalid ME command size");
STATIC_ASSERT (sizeof (Command.Response) == 28, "Invalid ME command size");
ZeroMem (&Command, sizeof (Command));
Command.Request.Command.Fields.Command = HOST_CLIENT_PROPERTIES_REQUEST;
Command.Request.Address = Address;
Status = HeciSendMessageWithResponse (
&Command,
sizeof (Command.Request),
sizeof (Command.Response)
);
CopyMem (
Properties,
&Command.Response.ClientProperties,
sizeof (*Properties)
);
return Status;
}
EFI_STATUS
HeciConnectToClient (
IN UINT8 Address
)
{
EFI_STATUS Status;
HBM_CLIENT_CONNECT_BUFFER Command;
Status = HeciLocateProtocol ();
if (EFI_ERROR (Status)) {
return Status;
}
ZeroMem (&Command, sizeof (Command));
STATIC_ASSERT (sizeof (Command.Request) == 4, "Invalid ME command size");
Command.Request.Command.Fields.Command = CLIENT_CONNECT_REQUEST;
Command.Request.MeAddress = Address;
Command.Request.HostAddress = HBM_CLIENT_ADDRESS;
Status = HeciSendMessageWithResponse (
&Command,
sizeof (Command.Request),
sizeof (Command.Response)
);
DEBUG ((DEBUG_INFO, "OCME: Connect to client %X code %d - %r\n", Address, Command.Response.Status, Status));
if (EFI_ERROR (Status)) {
return Status;
}
switch (Command.Response.Status) {
case HBM_CLIENT_CONNECT_NOT_FOUND:
return EFI_NOT_FOUND;
case HBM_CLIENT_CONNECT_ALREADY_CONNECTED:
return EFI_ALREADY_STARTED;
case HBM_CLIENT_CONNECT_OUT_OF_RESOURCES:
return EFI_OUT_OF_RESOURCES;
case HBM_CLIENT_CONNECT_INVALID_PARAMETER:
return EFI_INVALID_PARAMETER;
default:
mSendingHeciCommandPerClient = TRUE;
mCurrentMeClientRequestedReceiveMsg = 0;
mCurrentMeClientCanReceiveMsg = 0;
mCurrentMeClientAddress = Address;
return EFI_SUCCESS;
}
}
EFI_STATUS
HeciSendMessagePerClient (
IN VOID *Message,
IN UINT32 Size
)
{
EFI_STATUS Status;
Status = EFI_SUCCESS;
if (mSendingHeciCommandPerClient) {
if (!mCurrentMeClientCanReceiveMsg) {
HeciUpdateReceiveMsgStatus();
}
Status = HeciSendMessage (
Message,
Size,
HBM_CLIENT_ADDRESS,
mCurrentMeClientAddress
);
if (!EFI_ERROR (Status)) {
--mCurrentMeClientCanReceiveMsg;
}
}
return Status;
}
EFI_STATUS
HeciDisconnectFromClients (
VOID
)
{
EFI_STATUS Status;
HBM_CLIENT_DISCONNECT_BUFFER Command;
Status = EFI_SUCCESS;
if (mSendingHeciCommandPerClient) {
//
// Note, this is different between HECI 1 and HECI 2.
// HECI 1 has 4 byte response, and it does not require HeciUpdateReceiveMsgStatus.
//
STATIC_ASSERT (sizeof (Command.Request) == 4, "Invalid ME command req size");
STATIC_ASSERT (sizeof (Command.Response) == 8, "Invalid ME command rsp size");
if (!mCurrentMeClientCanReceiveMsg) {
HeciUpdateReceiveMsgStatus();
}
ZeroMem (&Command, sizeof (Command));
Command.Request.Command.Fields.Command = CLIENT_DISCONNECT_REQUEST;
Command.Request.MeAddress = mCurrentMeClientAddress;
Command.Request.HostAddress = HBM_CLIENT_ADDRESS;
++mCurrentMeClientRequestedReceiveMsg;
Status = HeciSendMessageWithResponse (
&Command,
sizeof (Command.Request),
sizeof (Command.Response)
);
DEBUG ((
DEBUG_INFO,
"OCME: Disconnect from client %X code %d - %r\n",
mCurrentMeClientAddress,
Command.Response.Status,
Status
));
if (!EFI_ERROR (Status)) {
mSendingHeciCommandPerClient = FALSE;
}
}
return Status;
}
EFI_STATUS
HeciPavpRequestProvisioning (
OUT UINT32 *EpidStatus,
OUT UINT32 *EpidGroupId
)
{
EFI_STATUS Status;
ME_PAVP_PROVISION_REQUEST_BUFFER Command;
STATIC_ASSERT (sizeof (Command.Request) == 16, "Invalid ME command size");
STATIC_ASSERT (sizeof (Command.Response) == 24, "Invalid ME command size");
ZeroMem (&Command, sizeof (Command));
Command.Request.Header.Version = ME_PAVP_PROTOCOL_VERSION;
Command.Request.Header.Command = ME_PAVP_PROVISION_REQUEST_COMMAND;
HeciSendMessagePerClient (&Command, sizeof (Command.Request));
ZeroMem (&Command, sizeof (Command));
Status = HeciGetResponse (&Command, sizeof (Command.Response));
if (!EFI_ERROR (Status)) {
*EpidStatus = Command.Response.Status;
*EpidGroupId = Command.Response.GroupId;
}
return Status;
}
EFI_STATUS
HeciPavpPerformProvisioning (
IN EPID_CERTIFICATE *EpidCertificate,
IN EPID_GROUP_PUBLIC_KEY *EpidGroupPublicKey,
OUT BOOLEAN *SetVar OPTIONAL
)
{
EFI_STATUS Status;
ME_PAVP_PROVISION_PERFORM_BUFFER Command;
UINTN Index;
STATIC_ASSERT (sizeof (Command.Request) == 1284, "Invalid ME command size");
STATIC_ASSERT (sizeof (Command.Response) == 16, "Invalid ME command size");
if (SetVar != NULL) {
*SetVar = FALSE;
}
ZeroMem (&Command, sizeof (Command));
Command.Request.Header.Version = ME_PAVP_PROTOCOL_VERSION;
Command.Request.Header.Command = ME_PAVP_PROVISION_PERFORM_COMMAND;
Command.Request.Header.PayloadSize = ME_PAVP_PROVISION_PERFORM_PAYLOAD_SIZE;
CopyMem (&Command.Request.Certificate, EpidCertificate, sizeof (Command.Request.Certificate));
CopyMem (&Command.Request.PublicKey, EpidGroupPublicKey, sizeof (Command.Request.PublicKey));
Status = HeciSendMessagePerClient (&Command, sizeof (Command.Request));
ZeroMem (&Command, sizeof (Command));
if (EFI_ERROR (Status)) {
DEBUG ((DEBUG_INFO, "OCME: Failed to send provisioning command - %r\n", Status));
return EFI_DEVICE_ERROR;
}
for (Index = 0; Index < 3; ++Index) {
Status = HeciGetResponse (&Command, sizeof (Command.Response));
if (Status != EFI_TIMEOUT) {
break;
}
}
DEBUG ((
DEBUG_INFO,
"OCME: Finished provisioning command with status %x - %r\n",
Command.Response.Header.Status,
Status
));
if (EFI_ERROR (Status)) {
return EFI_DEVICE_ERROR;
}
if (Command.Response.Header.Status != EPID_STATUS_PROVISIONED) {
Status = EFI_DEVICE_ERROR;
}
if (Command.Response.Header.Status == EPID_STATUS_FAIL_PROVISION) {
if (SetVar != NULL) {
*SetVar = TRUE;
}
}
return Status;
}
EFI_STATUS
HeciFpfGetStatus (
OUT UINT32 *FpfStatus
)
{
EFI_STATUS Status;
UINT32 Response[11];
UINT32 Request[4];
ZeroMem (Request, sizeof (Request));
Request[0] = 3;
HeciSendMessagePerClient (Request, sizeof (Request));
ZeroMem (Response, sizeof (Response));
Status = HeciGetResponse (Response, sizeof (Response));
if (!EFI_ERROR (Status)) {
*FpfStatus = Response[1];
}
return Status;
}
EFI_STATUS
HeciFpfProvision (
OUT UINT32 *FpfStatus
)
{
EFI_STATUS Status;
UINT32 Response[2];
UINT32 Request[3];
ZeroMem (Request, sizeof (Request));
Request[0] = 5;
Request[1] = 1;
Request[2] = 255;
HeciSendMessagePerClient (Request, sizeof (Request));
ZeroMem (Response, sizeof (Response));
Status = HeciGetResponse (Response, sizeof (Response));
if (!EFI_ERROR (Status)) {
*FpfStatus = Response[1];
}
return Status;
}

Опубликовать ( 0 )

Вы можете оставить комментарий после Вход в систему

1
https://gitlife.ru/oschina-mirror/zsl588-OpenCorePkg.git
git@gitlife.ru:oschina-mirror/zsl588-OpenCorePkg.git
oschina-mirror
zsl588-OpenCorePkg
zsl588-OpenCorePkg
0.7.7