blob: 7198d1834dec566da69b35ccafe5d53aeac5394d [file]
/** @file
Library that implements the Arm CCA Realm Service Interface calls.
Copyright (c) 2022 - 2024, Arm Limited. All rights reserved.<BR>
SPDX-License-Identifier: BSD-2-Clause-Patent
@par Glossary:
- Rsi or RSI - Realm Service Interface
- IPA - Intermediate Physical Address
- RIPAS - Realm IPA state
- REM - Realm Extensible Measurement
@par Reference(s):
- Realm Management Monitor (RMM) Specification, version 1.0-rel0
(https://developer.arm.com/documentation/den0137/)
**/
#include <Base.h>
#include <IndustryStandard/ArmStdSmc.h>
#include <Library/ArmCcaRsiLib.h>
#include <Library/ArmLib.h>
#include <Library/ArmSmcLib.h>
#include <Library/BaseMemoryLib.h>
#include <Library/DebugLib.h>
#include <Library/MemoryAllocationLib.h>
#include "ArmCcaRsi.h"
/** The version of RSI specification implemented by this module.
*/
#define RSI_IMPL_VERSION RMM_VERSION (1, 0);
/** A macro to test if a pointer or address is Realm Granule size aligned.
*/
#define IS_REALM_GRANULE_ALIGNED(Ptr) \
ADDRESS_IS_ALIGNED (Ptr, ARM_CCA_REALM_GRANULE_SIZE)
/**
Convert the RSI status code to EFI Status code.
@param [in] RsiCommandReturnCode RSI status code.
@retval RETURN_SUCCESS Success.
@retval RETURN_INVALID_PARAMETER A parameter is invalid.
@retval RETURN_ABORTED The operation was aborted as the state
of the Realm or REC does not match the
state expected by the command.
@retval RETURN_NOT_READY The operation requested by the command
is not complete.
**/
STATIC
RETURN_STATUS
ArmCcaRsiCmdStatusToReturnStatus (
IN UINT64 RsiCommandReturnCode
)
{
switch (RsiCommandReturnCode) {
case RSI_SUCCESS:
return RETURN_SUCCESS;
case RSI_ERROR_INPUT:
return RETURN_INVALID_PARAMETER;
case RSI_ERROR_STATE:
case RSI_ERROR_UNKNOWN:
return RETURN_ABORTED;
case RSI_INCOMPLETE:
return RETURN_NOT_READY;
default:
// Unknown error code.
ASSERT (0);
break;
} // switch
return RETURN_ABORTED;
}
/**
Continue the operation to retrieve an attestation token.
@param [out] TokenBuffer Pointer to a buffer to store the
retrieved attestation token.
@param [in] Offset Offset within Token buffer granule
to start of buffer in bytes.
@param [in,out] TokenSize On input size of the token buffer,
and on output size of the token
returned if operation is successful,
otherwise 0.
@retval RETURN_SUCCESS Success.
@retval RETURN_INVALID_PARAMETER A parameter is invalid.
@retval RETURN_ABORTED The operation was aborted as the state
of the Realm or REC does not match the
state expected by the command.
Or the Token generation failed for an
unknown or IMPDEF reason.
@retval RETURN_NOT_READY The operation requested by the command
is not complete.
**/
STATIC
RETURN_STATUS
EFIAPI
ArmCcaRsiAttestationTokenContinue (
OUT UINT8 *CONST TokenBuffer,
IN UINT64 CONST Offset,
IN OUT UINT64 *CONST TokenSize
)
{
RETURN_STATUS Status;
ARM_SMC_ARGS SmcCmd;
ZeroMem (&SmcCmd, sizeof (SmcCmd));
SmcCmd.Arg0 = ARM_CCA_FID_RSI_ATTESTATION_TOKEN_CONTINUE;
// Set the IPA of the Granule to which the token will be written.
SmcCmd.Arg1 = (UINTN)TokenBuffer;
// Set the Offset within Granule to start of buffer in bytes
SmcCmd.Arg2 = (UINTN)Offset;
// Set the size of the buffer in bytes
SmcCmd.Arg3 = (UINTN)*TokenSize;
ArmCallSmc (&SmcCmd);
Status = ArmCcaRsiCmdStatusToReturnStatus (SmcCmd.Arg0);
if (!RETURN_ERROR (Status)) {
// Update the token size
*TokenSize = SmcCmd.Arg1;
} else {
// Clear the TokenBuffer on error.
ZeroMem (TokenBuffer, *TokenSize);
*TokenSize = 0;
}
return Status;
}
/**
Initialize the operation to retrieve an attestation token.
@param [in] ChallengeData Pointer to the challenge data to be
included in the attestation token.
@param [in] ChallengeDataSizeBits Size of the challenge data in bits.
@param [out] MaxTokenSize Pointer to an integer to retrieve
the maximum attestation token size.
@retval RETURN_SUCCESS Success.
@retval RETURN_INVALID_PARAMETER A parameter is invalid.
**/
STATIC
RETURN_STATUS
EFIAPI
ArmCcaRsiAttestationTokenInit (
IN CONST UINT8 *CONST ChallengeData,
IN UINT64 ChallengeDataSizeBits,
OUT UINT64 *CONST MaxTokenSize
)
{
RETURN_STATUS Status;
ARM_SMC_ARGS SmcCmd;
UINT8 *Buffer8;
CONST UINT8 *Data8;
UINT64 Count;
UINT8 TailBits;
/* See A7.2.2 Attestation token generation, RMM Specification, version 1.0-rel0
IWTKDD - If the size of the challenge provided by the relying party is less
than 64 bytes, it should be zero-padded prior to calling
RSI_ATTESTATION_TOKEN_INIT.
Therefore, zero out the SmcCmd memory before coping the ChallengeData
bits.
*/
ZeroMem (&SmcCmd, sizeof (SmcCmd));
SmcCmd.Arg0 = ARM_CCA_FID_RSI_ATTESTATION_TOKEN_INIT;
// Copy challenge data.
Buffer8 = (UINT8 *)&SmcCmd.Arg1;
Data8 = ChallengeData;
TailBits = ChallengeDataSizeBits & 0x7;
Count = (ChallengeDataSizeBits - TailBits) / 8;
// First copy whole bytes
CopyMem (Buffer8, Data8, Count);
// Now copy any remaining tail bits.
if (TailBits > 0) {
// Advance buffer pointers.
Buffer8 += Count;
Data8 += Count;
// Copy tail byte.
*Buffer8 = *Data8;
// Clear unused tail bits.
*Buffer8 &= ~(0xFF << TailBits);
}
ArmCallSmc (&SmcCmd);
Status = ArmCcaRsiCmdStatusToReturnStatus (SmcCmd.Arg0);
if (RETURN_ERROR (Status)) {
// Set the max token size to zero
*MaxTokenSize = 0;
} else {
*MaxTokenSize = SmcCmd.Arg1;
}
return Status;
}
/**
Free the attestation token buffer.
@param [in] TokenBuffer Pointer to the retrieved
attestation token.
@param [in] TokenBufferSize Size of the token buffer.
**/
VOID
ArmCcaRsiFreeAttestationToken (
IN UINT8 *CONST TokenBuffer,
IN UINT64 CONST TokenBufferSize
)
{
if (TokenBuffer != NULL) {
if (TokenBufferSize > 0) {
// Scrub the token buffer
ZeroMem (TokenBuffer, TokenBufferSize);
}
FreePool (TokenBuffer);
}
}
/**
A helper function to Retrieve an attestation token from the RMM.
@param [in] Token Pointer to a buffer to store the
retrieved attestation token.
@param [in,out] TokenLength Length of token buffer in input and
length of token data returned in output.
@retval RETURN_SUCCESS Success.
@retval RETURN_INVALID_PARAMETER A parameter is invalid.
@retval RETURN_OUT_OF_RESOURCES Out of resources.
@retval RETURN_ABORTED The operation was aborted as the state
of the Realm or REC does not match the
state expected by the command.
Or the Token generation failed for an
unknown or IMPDEF reason.
@retval RETURN_NOT_READY The operation requested by the command
is not complete.
@retval RETURN_BAD_BUFFER_SIZE The token buffer size returned in an
earlier call to RSI_ATTESTATION_TOKEN_INIT
was insufficient to complete the operation.
**/
STATIC
RETURN_STATUS
EFIAPI
ArmCcaRsiGetAttestationTokenHelper (
IN UINT8 *Token,
IN OUT UINT64 *TokenLength
)
{
RETURN_STATUS Status;
UINT8 *Granule;
UINT64 GranuleSize;
UINT64 Offset;
UINT64 TokenSize;
UINT64 MaxTokenSize;
if ((Token == NULL) || (TokenLength == NULL) || (*TokenLength == 0)) {
return RETURN_INVALID_PARAMETER;
}
// Allocate a granule to retrieve the attestation token chunk.
Granule = (UINT8 *)AllocateAlignedPages (
EFI_SIZE_TO_PAGES (ARM_CCA_REALM_GRANULE_SIZE),
ARM_CCA_REALM_GRANULE_SIZE
);
if (Granule == NULL) {
ASSERT (0);
return RETURN_OUT_OF_RESOURCES;
}
MaxTokenSize = *TokenLength;
TokenSize = 0;
do {
// Retrieve one Granule of data per loop iteration
ZeroMem (Granule, ARM_CCA_REALM_GRANULE_SIZE);
Offset = 0;
do {
// Retrieve sub-Granule chunk of data per loop iteration
GranuleSize = ARM_CCA_REALM_GRANULE_SIZE - Offset;
Status = ArmCcaRsiAttestationTokenContinue (
Granule,
Offset,
&GranuleSize
);
Offset += GranuleSize;
} while ((Status == RETURN_NOT_READY) && (Offset < ARM_CCA_REALM_GRANULE_SIZE));
if (RETURN_ERROR (Status) && (Status != RETURN_NOT_READY)) {
ASSERT (0);
TokenSize = 0;
break;
}
if (Offset > (MaxTokenSize - TokenSize)) {
Status = RETURN_BAD_BUFFER_SIZE;
TokenSize = 0;
break;
}
// "Offset" bytes of data are now ready for consumption from "Granule"
// Copy the new token data from the Granule.
CopyMem (&Token[TokenSize], Granule, Offset);
TokenSize += Offset;
} while ((Status == RETURN_NOT_READY) && (TokenSize < MaxTokenSize));
// Update the TokenLength to reflect the token data retrieved.
*TokenLength = TokenSize;
// Scrub the Granule buffer
ZeroMem (Granule, ARM_CCA_REALM_GRANULE_SIZE);
FreeAlignedPages (Granule, EFI_SIZE_TO_PAGES (ARM_CCA_REALM_GRANULE_SIZE));
return Status;
}
/**
Retrieve an attestation token from the RMM.
@param [in] ChallengeData Pointer to the challenge data to be
included in the attestation token.
@param [in] ChallengeDataSizeBits Size of the challenge data in bits.
@param [out] TokenBuffer Pointer to a buffer to store the
retrieved attestation token.
@param [out] TokenBufferSize Length of token data returned.
Note: The TokenBuffer allocated must be freed by the caller
using RsiFreeAttestationToken().
@retval RETURN_SUCCESS Success.
@retval RETURN_INVALID_PARAMETER A parameter is invalid.
@retval RETURN_OUT_OF_RESOURCES Out of resources.
@retval RETURN_ABORTED The operation was aborted as the state
of the Realm or REC does not match the
state expected by the command.
Or the Token generation failed for an
unknown or IMPDEF reason.
@retval RETURN_NOT_READY The operation requested by the command
is not complete.
@retval RETURN_BAD_BUFFER_SIZE The token buffer size returned in an
earlier call to RSI_ATTESTATION_TOKEN_INIT
was insufficient to complete the operation.
**/
RETURN_STATUS
EFIAPI
ArmCcaRsiGetAttestationToken (
IN CONST UINT8 *CONST ChallengeData,
IN UINT64 ChallengeDataSizeBits,
OUT UINT8 **CONST TokenBuffer,
OUT UINT64 *CONST TokenBufferSize
)
{
RETURN_STATUS Status;
UINT8 *Token;
UINT64 TokenLength;
UINT64 MaxTokenSize;
if ((TokenBuffer == NULL) ||
(TokenBufferSize == NULL) ||
(ChallengeData == NULL))
{
return RETURN_INVALID_PARAMETER;
}
if (ChallengeDataSizeBits > ARM_CCA_MAX_CHALLENGE_DATA_SIZE_BITS) {
return RETURN_INVALID_PARAMETER;
}
/* See A7.2.2 Attestation token generation, RMM Specification, version 1.0-rel0
IWTKDD - Arm recommends that the challenge should contain at least 32 bytes
of unique data.
*/
if (ChallengeDataSizeBits < ARM_CCA_MIN_CHALLENGE_DATA_SIZE_BITS) {
DEBUG ((DEBUG_WARN, "Minimum Challenge data size should be 32 bytes\n"));
}
Status = ArmCcaRsiAttestationTokenInit (
ChallengeData,
ChallengeDataSizeBits,
&MaxTokenSize
);
if (RETURN_ERROR (Status)) {
ASSERT_RETURN_ERROR (Status);
return Status;
}
// Allocate a buffer to store the retrieved attestation token.
Token = AllocateZeroPool (MaxTokenSize);
if (Token == NULL) {
ASSERT (0);
return RETURN_OUT_OF_RESOURCES;
}
// Retrieve the attestation token.
TokenLength = MaxTokenSize;
Status = ArmCcaRsiGetAttestationTokenHelper (Token, &TokenLength);
if (RETURN_ERROR (Status)) {
// Scrub the Token on failure.
ZeroMem (Token, MaxTokenSize);
FreePool (Token);
*TokenBuffer = NULL;
*TokenBufferSize = 0;
} else {
*TokenBuffer = Token;
*TokenBufferSize = TokenLength;
}
return Status;
}
/**
Returns the IPA state for the page pointed by the address.
@param [in] Base Base of target IPA region.
@param [in, out] Top End of target IPA region on input.
Top of IPA region which has the
reported RIPAS value on return.
@param [out] State The RIPAS state for the address specified.
@retval RETURN_SUCCESS Success.
@retval RETURN_INVALID_PARAMETER A parameter is invalid.
**/
RETURN_STATUS
EFIAPI
ArmCcaRsiGetIpaState (
IN UINT64 *Base,
IN OUT UINT64 **Top,
OUT ARM_CCA_RIPAS *State
)
{
RETURN_STATUS Status;
ARM_SMC_ARGS SmcCmd;
if ((State == NULL) ||
(!IS_REALM_GRANULE_ALIGNED (Base)) ||
(!IS_REALM_GRANULE_ALIGNED (*Top)) ||
(*Top < Base))
{
return RETURN_INVALID_PARAMETER;
}
ZeroMem (&SmcCmd, sizeof (SmcCmd));
SmcCmd.Arg0 = ARM_CCA_FID_RSI_IPA_STATE_GET;
SmcCmd.Arg1 = (UINTN)Base;
SmcCmd.Arg2 = (UINTN)*Top;
ArmCallSmc (&SmcCmd);
Status = ArmCcaRsiCmdStatusToReturnStatus (SmcCmd.Arg0);
if (!RETURN_ERROR (Status)) {
*Top = (UINT64 *)SmcCmd.Arg1;
*State = (ARM_CCA_RIPAS)(SmcCmd.Arg2 & ARM_CCA_RIPAS_TYPE_MASK);
}
return Status;
}
/**
Sets the IPA state for the pages pointed by the memory range.
@param [in] Address Address to the start of the memory range.
@param [in] Size Length of the memory range.
@param [in] State The RIPAS state to be configured.
@param [in] Flags The RIPAS change flags.
@retval RETURN_SUCCESS Success.
@retval RETURN_INVALID_PARAMETER A parameter is invalid.
@retval RETURN_ACCESS_DENIED RIPAS change request was rejected.
**/
RETURN_STATUS
EFIAPI
ArmCcaRsiSetIpaState (
IN UINT64 *Address,
IN UINT64 Size,
IN ARM_CCA_RIPAS State,
IN UINT64 Flags
)
{
RETURN_STATUS Status;
UINT64 *BaseAddress;
UINT64 *EndAddress;
ARM_SMC_ARGS SmcCmd;
if ((Size == 0) ||
(!IS_REALM_GRANULE_ALIGNED (Address)) ||
(!IS_REALM_GRANULE_ALIGNED ((UINT8 *)Address + Size)) ||
((State != RipasEmpty) && (State != RipasRam)))
{
return RETURN_INVALID_PARAMETER;
}
BaseAddress = Address;
// Divide Size by 8 for the pointer arithmetic
// to work, as we are adding to UINT64*.
EndAddress = Address + (Size / 8);
while (Size > 0) {
ZeroMem (&SmcCmd, sizeof (SmcCmd));
SmcCmd.Arg0 = ARM_CCA_FID_RSI_IPA_STATE_SET;
SmcCmd.Arg1 = (UINTN)BaseAddress;
SmcCmd.Arg2 = (UINTN)EndAddress;
SmcCmd.Arg3 = (UINTN)State;
SmcCmd.Arg4 = Flags;
ArmCallSmc (&SmcCmd);
Status = ArmCcaRsiCmdStatusToReturnStatus (SmcCmd.Arg0);
if (RETURN_ERROR (Status)) {
break;
}
BaseAddress = (UINT64 *)SmcCmd.Arg1;
Size = EndAddress - BaseAddress;
if ((SmcCmd.Arg2 & ARM_CCA_RSI_RESPONSE_MASK) == ARM_CCA_RIPAS_CHANGE_RESPONSE_REJECT) {
Status = RETURN_ACCESS_DENIED;
break;
}
} // while
return Status;
}
/**
Extends a measurement to a REM.
@param [in] MeasurementIndex Index of the REM.
@param [in] Measurement Pointer to the measurement buffer.
@param [in] MeasurementSize Size of the measurement data.
@retval RETURN_SUCCESS Success.
@retval RETURN_INVALID_PARAMETER A parameter is invalid.
**/
RETURN_STATUS
EFIAPI
ArmCcaRsiExtendMeasurement (
IN UINTN MeasurementIndex,
IN CONST UINT8 *CONST Measurement,
IN UINTN MeasurementSize
)
{
ARM_SMC_ARGS SmcCmd;
UINT64 *Data64;
if ((MeasurementIndex < ARM_CCA_MIN_REM_INDEX) ||
(MeasurementIndex > ARM_CCA_MAX_REM_INDEX) ||
(Measurement == NULL) ||
(MeasurementSize == 0) ||
(MeasurementSize > ARM_CCA_MAX_MEASUREMENT_DATA_SIZE_BYTES))
{
return RETURN_INVALID_PARAMETER;
}
ZeroMem (&SmcCmd, sizeof (SmcCmd));
SmcCmd.Arg0 = ARM_CCA_FID_RSI_MEASUREMENT_EXTEND;
SmcCmd.Arg1 = MeasurementIndex;
SmcCmd.Arg2 = MeasurementSize;
Data64 = &SmcCmd.Arg3;
CopyMem (Data64, Measurement, MeasurementSize);
ArmCallSmc (&SmcCmd);
return ArmCcaRsiCmdStatusToReturnStatus (SmcCmd.Arg0);
}
/**
Read the measurement value from a REM.
@param [in] MeasurementIndex Index of the REM.
@param [out] MeasurementBuffer Pointer to store the measurement data.
@param [in] MeasurementBufferSize Size of the measurement buffer.
@retval RETURN_SUCCESS Success.
@retval RETURN_INVALID_PARAMETER A parameter is invalid.
**/
RETURN_STATUS
EFIAPI
ArmCcaRsiReadMeasurement (
IN UINTN MeasurementIndex,
OUT UINT8 *CONST MeasurementBuffer,
IN UINTN MeasurementBufferSize
)
{
RETURN_STATUS Status;
ARM_SMC_ARGS SmcCmd;
UINT64 *Data64;
if ((MeasurementIndex < ARM_CCA_MIN_REM_INDEX) ||
(MeasurementIndex > ARM_CCA_MAX_REM_INDEX) ||
(MeasurementBuffer == NULL))
{
return RETURN_INVALID_PARAMETER;
}
if (MeasurementBufferSize < ARM_CCA_MAX_MEASUREMENT_DATA_SIZE_BYTES) {
return RETURN_BUFFER_TOO_SMALL;
}
ZeroMem (&SmcCmd, sizeof (SmcCmd));
SmcCmd.Arg0 = ARM_CCA_FID_RSI_MEASUREMENT_READ;
SmcCmd.Arg1 = MeasurementIndex;
ArmCallSmc (&SmcCmd);
Status = ArmCcaRsiCmdStatusToReturnStatus (SmcCmd.Arg0);
if (!RETURN_ERROR (Status)) {
Data64 = &SmcCmd.Arg1;
CopyMem (MeasurementBuffer, Data64, ARM_CCA_MAX_MEASUREMENT_DATA_SIZE_BYTES);
}
return Status;
}
/**
Read the Realm Configuration.
@param [out] Config Pointer to the address of the buffer to retrieve
the Realm configuration.
Note: The buffer to retrieve the Realm configuration must be aligned to the
Realm granule size.
@retval RETURN_SUCCESS Success.
@retval RETURN_INVALID_PARAMETER A parameter is invalid.
**/
RETURN_STATUS
EFIAPI
ArmCcaRsiGetRealmConfig (
OUT ARM_CCA_REALM_CONFIG *Config
)
{
ARM_SMC_ARGS SmcCmd;
if ((Config == NULL) || (!IS_REALM_GRANULE_ALIGNED (Config))) {
return RETURN_INVALID_PARAMETER;
}
ZeroMem (&SmcCmd, sizeof (SmcCmd));
SmcCmd.Arg0 = ARM_CCA_FID_RSI_REALM_CONFIG;
SmcCmd.Arg1 = (UINTN)Config;
ArmCallSmc (&SmcCmd);
return ArmCcaRsiCmdStatusToReturnStatus (SmcCmd.Arg0);
}
/**
Make a Host Call.
A Host call can be used by a Realm to make a hypercall.
On Realm execution of HVC, an Unknown exception is taken to the Realm.
@param [in] Args Pointer to the IPA of the Host call data
structure.
Note: The IPA of the Host call arguments data structure must be aligned
to the Realm granule size.
@retval RETURN_SUCCESS Success.
@retval RETURN_INVALID_PARAMETER A parameter is invalid.
**/
RETURN_STATUS
EFIAPI
ArmCcaRsiHostCall (
IN ARM_CCA_RSI_HOST_CALL_ARGS *Args
)
{
ARM_SMC_ARGS SmcCmd;
// The RMM specification, version 1.0-rel0, section B5.3.3.2 Failure
// conditions specifies the alignment requirement for the RsiHostCall
// structure as 256 bytes.
if ((Args == NULL) || !ADDRESS_IS_ALIGNED (Args, ARM_CCA_RSI_HOST_CALL_ARGS_SIZE)) {
return RETURN_INVALID_PARAMETER;
}
// Clear the reserved fields
ZeroMem (&Args->Reserved1, sizeof (Args->Reserved1));
ZeroMem (&SmcCmd, sizeof (SmcCmd));
SmcCmd.Arg0 = ARM_CCA_FID_RSI_HOST_CALL;
SmcCmd.Arg1 = (UINTN)Args;
ArmCallSmc (&SmcCmd);
return ArmCcaRsiCmdStatusToReturnStatus (SmcCmd.Arg0);
}
/**
Get the version of the RSI implementation.
@param [out] UefiImpl The version of the RSI specification
implemented by the UEFI firmware.
@param [out] RmmImplLow The low version of the RSI specification
implemented by the RMM.
@param [out] RmmImplHigh The high version of the RSI specification
implemented by the RMM.
@retval RETURN_SUCCESS Success.
@retval RETURN_UNSUPPORTED The execution context is not a Realm.
@retval RETURN_INCOMPATIBLE_VERSION The Firmware and RMM specification
revisions are not compatible.
@retval RETURN_INVALID_PARAMETER A parameter is invalid.
**/
RETURN_STATUS
EFIAPI
ArmCcaRsiGetVersion (
OUT UINT32 *CONST UefiImpl,
OUT UINT32 *CONST RmmImplLow,
OUT UINT32 *CONST RmmImplHigh
)
{
RETURN_STATUS Status;
ARM_SMC_ARGS SmcCmd;
if ((UefiImpl == NULL) || (RmmImplLow == NULL) || (RmmImplHigh == NULL)) {
return RETURN_INVALID_PARAMETER;
}
ZeroMem (&SmcCmd, sizeof (SmcCmd));
SmcCmd.Arg0 = ARM_CCA_FID_RSI_VERSION;
SmcCmd.Arg1 = RSI_IMPL_VERSION;
ArmCallSmc (&SmcCmd);
if (SmcCmd.Arg0 == ARM_SMC_MM_RET_NOT_SUPPORTED) {
// This FID is not implemented, which means
// we are not running in a Realm, therefore
// return the error code as unsupported.
return RETURN_UNSUPPORTED;
}
*RmmImplLow = (SmcCmd.Arg1 & RSI_VERSION_MASK);
*RmmImplHigh = (SmcCmd.Arg2 & RSI_VERSION_MASK);
*UefiImpl = RSI_IMPL_VERSION;
// The RSI_VERSION command does not have any failure
// conditions see section B5.3.10.2 Failure conditions
// Therefore the only defined return values are
// RSI_SUCCESS and RSI_ERROR_INPUT.
Status = ArmCcaRsiCmdStatusToReturnStatus (SmcCmd.Arg0);
if (Status == RETURN_INVALID_PARAMETER) {
// RSI_VERSION returns RSI_ERROR_INPUT when
// the RMM does not support an interface revision
// which is compatible with the requested revision.
// Since RSI_ERROR_INPUT is mapped to RETURN_INVALID_PARAMETER
// by ArmCcaRsiCmdStatusToReturnStatus(), return the status code as
// RETURN_INCOMPATIBLE_VERSION.
return RETURN_INCOMPATIBLE_VERSION;
}
// Add an assert in case RMM returns a different error code than expected.
ASSERT (Status == RETURN_SUCCESS);
return Status;
}
/**
Get the features supported by the RSI implementation.
RMM implementations across different CCA platforms may support
disparate features and may offer disparate configuration options
for Realms. The features supported by an RSI implementation are
discovered by reading feature pseudo-register values using the
RSI_FEATURES command.
@param [in] FeatureRegIndex The Feature Register Index.
@param [out] FeatureRegValue The Feature Register Value.
@retval RETURN_SUCCESS Success.
@retval RETURN_INVALID_PARAMETER A parameter is invalid.
**/
RETURN_STATUS
EFIAPI
ArmCcaRsiGetFeatures (
IN UINT64 FeatureRegIndex,
OUT UINT64 *FeatureRegValue
)
{
ARM_SMC_ARGS SmcCmd;
if (FeatureRegValue == NULL) {
return RETURN_INVALID_PARAMETER;
}
ZeroMem (&SmcCmd, sizeof (SmcCmd));
SmcCmd.Arg0 = ARM_CCA_FID_RSI_FEATURES;
SmcCmd.Arg1 = FeatureRegIndex;
ArmCallSmc (&SmcCmd);
*FeatureRegValue = SmcCmd.Arg1;
return ArmCcaRsiCmdStatusToReturnStatus (SmcCmd.Arg0);
}