/** @file | |
Utility functions for expression evaluation. | |
Copyright (c) 2007 - 2008, Intel Corporation | |
All rights reserved. This program and the accompanying materials | |
are licensed and made available under the terms and conditions of the BSD License | |
which 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 "Setup.h" | |
// | |
// Global stack used to evaluate boolean expresions | |
// | |
EFI_HII_VALUE *mOpCodeScopeStack = NULL; | |
EFI_HII_VALUE *mOpCodeScopeStackEnd = NULL; | |
EFI_HII_VALUE *mOpCodeScopeStackPointer = NULL; | |
EFI_HII_VALUE *mExpressionEvaluationStack = NULL; | |
EFI_HII_VALUE *mExpressionEvaluationStackEnd = NULL; | |
EFI_HII_VALUE *mExpressionEvaluationStackPointer = NULL; | |
// | |
// Unicode collation protocol interface | |
// | |
EFI_UNICODE_COLLATION_PROTOCOL *mUnicodeCollation = NULL; | |
/** | |
Grow size of the stack. | |
This is an internal function. | |
@param Stack On input: old stack; On output: new stack | |
@param StackPtr On input: old stack pointer; On output: new stack | |
pointer | |
@param StackEnd On input: old stack end; On output: new stack end | |
@retval EFI_SUCCESS Grow stack success. | |
@retval EFI_OUT_OF_RESOURCES No enough memory for stack space. | |
**/ | |
EFI_STATUS | |
GrowStack ( | |
IN OUT EFI_HII_VALUE **Stack, | |
IN OUT EFI_HII_VALUE **StackPtr, | |
IN OUT EFI_HII_VALUE **StackEnd | |
) | |
{ | |
UINTN Size; | |
EFI_HII_VALUE *NewStack; | |
Size = EXPRESSION_STACK_SIZE_INCREMENT; | |
if (*StackPtr != NULL) { | |
Size = Size + (*StackEnd - *Stack); | |
} | |
NewStack = AllocatePool (Size * sizeof (EFI_HII_VALUE)); | |
if (NewStack == NULL) { | |
return EFI_OUT_OF_RESOURCES; | |
} | |
if (*StackPtr != NULL) { | |
// | |
// Copy from Old Stack to the New Stack | |
// | |
CopyMem ( | |
NewStack, | |
*Stack, | |
(*StackEnd - *Stack) * sizeof (EFI_HII_VALUE) | |
); | |
// | |
// Free The Old Stack | |
// | |
FreePool (*Stack); | |
} | |
// | |
// Make the Stack pointer point to the old data in the new stack | |
// | |
*StackPtr = NewStack + (*StackPtr - *Stack); | |
*Stack = NewStack; | |
*StackEnd = NewStack + Size; | |
return EFI_SUCCESS; | |
} | |
/** | |
Push an element onto the Boolean Stack. | |
@param Stack On input: old stack; On output: new stack | |
@param StackPtr On input: old stack pointer; On output: new stack | |
pointer | |
@param StackEnd On input: old stack end; On output: new stack end | |
@param Data Data to push. | |
@retval EFI_SUCCESS Push stack success. | |
**/ | |
EFI_STATUS | |
PushStack ( | |
IN OUT EFI_HII_VALUE **Stack, | |
IN OUT EFI_HII_VALUE **StackPtr, | |
IN OUT EFI_HII_VALUE **StackEnd, | |
IN EFI_HII_VALUE *Data | |
) | |
{ | |
EFI_STATUS Status; | |
// | |
// Check for a stack overflow condition | |
// | |
if (*StackPtr >= *StackEnd) { | |
// | |
// Grow the stack | |
// | |
Status = GrowStack (Stack, StackPtr, StackEnd); | |
if (EFI_ERROR (Status)) { | |
return Status; | |
} | |
} | |
// | |
// Push the item onto the stack | |
// | |
CopyMem (*StackPtr, Data, sizeof (EFI_HII_VALUE)); | |
*StackPtr = *StackPtr + 1; | |
return EFI_SUCCESS; | |
} | |
/** | |
Pop an element from the stack. | |
@param Stack On input: old stack; On output: new stack | |
@param StackPtr On input: old stack pointer; On output: new stack | |
pointer | |
@param StackEnd On input: old stack end; On output: new stack end | |
@param Data Data to pop. | |
@retval EFI_SUCCESS The value was popped onto the stack. | |
@retval EFI_ACCESS_DENIED The pop operation underflowed the stack | |
**/ | |
EFI_STATUS | |
PopStack ( | |
IN OUT EFI_HII_VALUE **Stack, | |
IN OUT EFI_HII_VALUE **StackPtr, | |
IN OUT EFI_HII_VALUE **StackEnd, | |
OUT EFI_HII_VALUE *Data | |
) | |
{ | |
// | |
// Check for a stack underflow condition | |
// | |
if (*StackPtr == *Stack) { | |
return EFI_ACCESS_DENIED; | |
} | |
// | |
// Pop the item off the stack | |
// | |
*StackPtr = *StackPtr - 1; | |
CopyMem (Data, *StackPtr, sizeof (EFI_HII_VALUE)); | |
return EFI_SUCCESS; | |
} | |
/** | |
Reset stack pointer to begin of the stack. | |
**/ | |
VOID | |
ResetScopeStack ( | |
VOID | |
) | |
{ | |
mOpCodeScopeStackPointer = mOpCodeScopeStack; | |
} | |
/** | |
Push an Operand onto the Stack | |
@param Operand Operand to push. | |
@retval EFI_SUCCESS The value was pushed onto the stack. | |
@retval EFI_OUT_OF_RESOURCES There is not enough system memory to grow the | |
stack. | |
**/ | |
EFI_STATUS | |
PushScope ( | |
IN UINT8 Operand | |
) | |
{ | |
EFI_HII_VALUE Data; | |
Data.Type = EFI_IFR_TYPE_NUM_SIZE_8; | |
Data.Value.u8 = Operand; | |
return PushStack ( | |
&mOpCodeScopeStack, | |
&mOpCodeScopeStackPointer, | |
&mOpCodeScopeStackEnd, | |
&Data | |
); | |
} | |
/** | |
Pop an Operand from the Stack | |
@param Operand Operand to pop. | |
@retval EFI_SUCCESS The value was pushed onto the stack. | |
@retval EFI_OUT_OF_RESOURCES There is not enough system memory to grow the | |
stack. | |
**/ | |
EFI_STATUS | |
PopScope ( | |
OUT UINT8 *Operand | |
) | |
{ | |
EFI_STATUS Status; | |
EFI_HII_VALUE Data; | |
Status = PopStack ( | |
&mOpCodeScopeStack, | |
&mOpCodeScopeStackPointer, | |
&mOpCodeScopeStackEnd, | |
&Data | |
); | |
*Operand = Data.Value.u8; | |
return Status; | |
} | |
/** | |
Reset stack pointer to begin of the stack. | |
**/ | |
VOID | |
ResetExpressionStack ( | |
VOID | |
) | |
{ | |
mExpressionEvaluationStackPointer = mExpressionEvaluationStack; | |
} | |
/** | |
Push an Expression value onto the Stack | |
@param Value Expression value to push. | |
@retval EFI_SUCCESS The value was pushed onto the stack. | |
@retval EFI_OUT_OF_RESOURCES There is not enough system memory to grow the | |
stack. | |
**/ | |
EFI_STATUS | |
PushExpression ( | |
IN EFI_HII_VALUE *Value | |
) | |
{ | |
return PushStack ( | |
&mExpressionEvaluationStack, | |
&mExpressionEvaluationStackPointer, | |
&mExpressionEvaluationStackEnd, | |
Value | |
); | |
} | |
/** | |
Pop an Expression value from the stack. | |
@param Value Expression value to pop. | |
@retval EFI_SUCCESS The value was popped onto the stack. | |
@retval EFI_ACCESS_DENIED The pop operation underflowed the stack | |
**/ | |
EFI_STATUS | |
PopExpression ( | |
OUT EFI_HII_VALUE *Value | |
) | |
{ | |
return PopStack ( | |
&mExpressionEvaluationStack, | |
&mExpressionEvaluationStackPointer, | |
&mExpressionEvaluationStackEnd, | |
Value | |
); | |
} | |
/** | |
Get Form given its FormId. | |
@param FormSet The formset which contains this form. | |
@param FormId Id of this form. | |
@retval Pointer The form. | |
@retval NULL Specified Form is not found in the formset. | |
**/ | |
FORM_BROWSER_FORM * | |
IdToForm ( | |
IN FORM_BROWSER_FORMSET *FormSet, | |
IN UINT16 FormId | |
) | |
{ | |
LIST_ENTRY *Link; | |
FORM_BROWSER_FORM *Form; | |
Link = GetFirstNode (&FormSet->FormListHead); | |
while (!IsNull (&FormSet->FormListHead, Link)) { | |
Form = FORM_BROWSER_FORM_FROM_LINK (Link); | |
if (Form->FormId == FormId) { | |
return Form; | |
} | |
Link = GetNextNode (&FormSet->FormListHead, Link); | |
} | |
return NULL; | |
} | |
/** | |
Search a Question in Form scope using its QuestionId. | |
@param Form The form which contains this Question. | |
@param QuestionId Id of this Question. | |
@retval Pointer The Question. | |
@retval NULL Specified Question not found in the form. | |
**/ | |
FORM_BROWSER_STATEMENT * | |
IdToQuestion2 ( | |
IN FORM_BROWSER_FORM *Form, | |
IN UINT16 QuestionId | |
) | |
{ | |
LIST_ENTRY *Link; | |
FORM_BROWSER_STATEMENT *Question; | |
if (QuestionId == 0) { | |
// | |
// The value of zero is reserved | |
// | |
return NULL; | |
} | |
Link = GetFirstNode (&Form->StatementListHead); | |
while (!IsNull (&Form->StatementListHead, Link)) { | |
Question = FORM_BROWSER_STATEMENT_FROM_LINK (Link); | |
if (Question->QuestionId == QuestionId) { | |
return Question; | |
} | |
Link = GetNextNode (&Form->StatementListHead, Link); | |
} | |
return NULL; | |
} | |
/** | |
Search a Question in Formset scope using its QuestionId. | |
@param FormSet The formset which contains this form. | |
@param Form The form which contains this Question. | |
@param QuestionId Id of this Question. | |
@retval Pointer The Question. | |
@retval NULL Specified Question not found in the form. | |
**/ | |
FORM_BROWSER_STATEMENT * | |
IdToQuestion ( | |
IN FORM_BROWSER_FORMSET *FormSet, | |
IN FORM_BROWSER_FORM *Form, | |
IN UINT16 QuestionId | |
) | |
{ | |
LIST_ENTRY *Link; | |
FORM_BROWSER_STATEMENT *Question; | |
// | |
// Search in the form scope first | |
// | |
Question = IdToQuestion2 (Form, QuestionId); | |
if (Question != NULL) { | |
return Question; | |
} | |
// | |
// Search in the formset scope | |
// | |
Link = GetFirstNode (&FormSet->FormListHead); | |
while (!IsNull (&FormSet->FormListHead, Link)) { | |
Form = FORM_BROWSER_FORM_FROM_LINK (Link); | |
Question = IdToQuestion2 (Form, QuestionId); | |
if (Question != NULL) { | |
// | |
// EFI variable storage may be updated by Callback() asynchronous, | |
// to keep synchronous, always reload the Question Value. | |
// | |
if (Question->Storage->Type == EFI_HII_VARSTORE_EFI_VARIABLE) { | |
GetQuestionValue (FormSet, Form, Question, FALSE); | |
} | |
return Question; | |
} | |
Link = GetNextNode (&FormSet->FormListHead, Link); | |
} | |
return NULL; | |
} | |
/** | |
Get Expression given its RuleId. | |
@param Form The form which contains this Expression. | |
@param RuleId Id of this Expression. | |
@retval Pointer The Expression. | |
@retval NULL Specified Expression not found in the form. | |
**/ | |
FORM_EXPRESSION * | |
RuleIdToExpression ( | |
IN FORM_BROWSER_FORM *Form, | |
IN UINT8 RuleId | |
) | |
{ | |
LIST_ENTRY *Link; | |
FORM_EXPRESSION *Expression; | |
Link = GetFirstNode (&Form->ExpressionListHead); | |
while (!IsNull (&Form->ExpressionListHead, Link)) { | |
Expression = FORM_EXPRESSION_FROM_LINK (Link); | |
if (Expression->Type == EFI_HII_EXPRESSION_RULE && Expression->RuleId == RuleId) { | |
return Expression; | |
} | |
Link = GetNextNode (&Form->ExpressionListHead, Link); | |
} | |
return NULL; | |
} | |
/** | |
Locate the Unicode Collation Protocol interface for later use. | |
@retval EFI_SUCCESS Protocol interface initialize success. | |
@retval Other Protocol interface initialize failed. | |
**/ | |
EFI_STATUS | |
InitializeUnicodeCollationProtocol ( | |
VOID | |
) | |
{ | |
EFI_STATUS Status; | |
if (mUnicodeCollation != NULL) { | |
return EFI_SUCCESS; | |
} | |
// | |
// BUGBUG: Proper impelmentation is to locate all Unicode Collation Protocol | |
// instances first and then select one which support English language. | |
// Current implementation just pick the first instance. | |
// | |
Status = gBS->LocateProtocol ( | |
&gEfiUnicodeCollation2ProtocolGuid, | |
NULL, | |
(VOID **) &mUnicodeCollation | |
); | |
return Status; | |
} | |
/** | |
Convert the input Unicode character to upper. | |
@param String Th Unicode character to be converted. | |
**/ | |
VOID | |
IfrStrToUpper ( | |
IN CHAR16 *String | |
) | |
{ | |
while (*String != 0) { | |
if ((*String >= 'a') && (*String <= 'z')) { | |
*String = (UINT16) ((*String) & ((UINT16) ~0x20)); | |
} | |
String++; | |
} | |
} | |
/** | |
Evaluate opcode EFI_IFR_TO_STRING. | |
@param FormSet Formset which contains this opcode. | |
@param Format String format in EFI_IFR_TO_STRING. | |
@param Result Evaluation result for this opcode. | |
@retval EFI_SUCCESS Opcode evaluation success. | |
@retval Other Opcode evaluation failed. | |
**/ | |
EFI_STATUS | |
IfrToString ( | |
IN FORM_BROWSER_FORMSET *FormSet, | |
IN UINT8 Format, | |
OUT EFI_HII_VALUE *Result | |
) | |
{ | |
EFI_STATUS Status; | |
EFI_HII_VALUE Value; | |
CHAR16 *String; | |
CHAR16 *PrintFormat; | |
CHAR16 Buffer[MAXIMUM_VALUE_CHARACTERS]; | |
UINTN BufferSize; | |
Status = PopExpression (&Value); | |
if (EFI_ERROR (Status)) { | |
return Status; | |
} | |
switch (Value.Type) { | |
case EFI_IFR_TYPE_NUM_SIZE_8: | |
case EFI_IFR_TYPE_NUM_SIZE_16: | |
case EFI_IFR_TYPE_NUM_SIZE_32: | |
case EFI_IFR_TYPE_NUM_SIZE_64: | |
BufferSize = MAXIMUM_VALUE_CHARACTERS * sizeof (CHAR16); | |
switch (Format) { | |
case EFI_IFR_STRING_UNSIGNED_DEC: | |
case EFI_IFR_STRING_SIGNED_DEC: | |
PrintFormat = L"%ld"; | |
break; | |
case EFI_IFR_STRING_LOWERCASE_HEX: | |
PrintFormat = L"%lx"; | |
break; | |
case EFI_IFR_STRING_UPPERCASE_HEX: | |
PrintFormat = L"%lX"; | |
break; | |
default: | |
return EFI_UNSUPPORTED; | |
} | |
UnicodeSPrint (Buffer, BufferSize, PrintFormat, Value.Value.u64); | |
String = Buffer; | |
break; | |
case EFI_IFR_TYPE_STRING: | |
CopyMem (Result, &Value, sizeof (EFI_HII_VALUE)); | |
return EFI_SUCCESS; | |
case EFI_IFR_TYPE_BOOLEAN: | |
String = (Value.Value.b) ? L"True" : L"False"; | |
break; | |
default: | |
return EFI_UNSUPPORTED; | |
} | |
Result->Type = EFI_IFR_TYPE_STRING; | |
Result->Value.string = NewString (String, FormSet->HiiHandle); | |
return EFI_SUCCESS; | |
} | |
/** | |
Evaluate opcode EFI_IFR_TO_UINT. | |
@param FormSet Formset which contains this opcode. | |
@param Result Evaluation result for this opcode. | |
@retval EFI_SUCCESS Opcode evaluation success. | |
@retval Other Opcode evaluation failed. | |
**/ | |
EFI_STATUS | |
IfrToUint ( | |
IN FORM_BROWSER_FORMSET *FormSet, | |
OUT EFI_HII_VALUE *Result | |
) | |
{ | |
EFI_STATUS Status; | |
EFI_HII_VALUE Value; | |
CHAR16 *String; | |
CHAR16 *StringPtr; | |
Status = PopExpression (&Value); | |
if (EFI_ERROR (Status)) { | |
return Status; | |
} | |
if (Value.Type >= EFI_IFR_TYPE_OTHER) { | |
return EFI_UNSUPPORTED; | |
} | |
Status = EFI_SUCCESS; | |
if (Value.Type == EFI_IFR_TYPE_STRING) { | |
String = GetToken (Value.Value.string, FormSet->HiiHandle); | |
if (String == NULL) { | |
return EFI_NOT_FOUND; | |
} | |
IfrStrToUpper (String); | |
StringPtr = StrStr (String, L"0X"); | |
if (StringPtr != NULL) { | |
// | |
// Hex string | |
// | |
Result->Value.u64 = StrHexToUint64 (String); | |
} else { | |
// | |
// decimal string | |
// | |
Result->Value.u64 = StrDecimalToUint64 (String); | |
} | |
FreePool (String); | |
} else { | |
CopyMem (Result, &Value, sizeof (EFI_HII_VALUE)); | |
} | |
Result->Type = EFI_IFR_TYPE_NUM_SIZE_64; | |
return Status; | |
} | |
/** | |
Evaluate opcode EFI_IFR_CATENATE. | |
@param FormSet Formset which contains this opcode. | |
@param Result Evaluation result for this opcode. | |
@retval EFI_SUCCESS Opcode evaluation success. | |
@retval Other Opcode evaluation failed. | |
**/ | |
EFI_STATUS | |
IfrCatenate ( | |
IN FORM_BROWSER_FORMSET *FormSet, | |
OUT EFI_HII_VALUE *Result | |
) | |
{ | |
EFI_STATUS Status; | |
EFI_HII_VALUE Value; | |
CHAR16 *String[2]; | |
UINTN Index; | |
CHAR16 *StringPtr; | |
UINTN Size; | |
// | |
// String[0] - The second string | |
// String[1] - The first string | |
// | |
String[0] = NULL; | |
String[1] = NULL; | |
StringPtr = NULL; | |
Status = EFI_SUCCESS; | |
for (Index = 0; Index < 2; Index++) { | |
Status = PopExpression (&Value); | |
if (EFI_ERROR (Status)) { | |
goto Done; | |
} | |
if (Value.Type != EFI_IFR_TYPE_STRING) { | |
Status = EFI_UNSUPPORTED; | |
goto Done; | |
} | |
String[Index] = GetToken (Value.Value.string, FormSet->HiiHandle); | |
if (String[Index] == NULL) { | |
Status = EFI_NOT_FOUND; | |
goto Done; | |
} | |
} | |
Size = StrSize (String[0]); | |
StringPtr= AllocatePool (StrSize (String[1]) + Size); | |
ASSERT (StringPtr != NULL); | |
StrCpy (StringPtr, String[1]); | |
StrCat (StringPtr, String[0]); | |
Result->Type = EFI_IFR_TYPE_STRING; | |
Result->Value.string = NewString (StringPtr, FormSet->HiiHandle); | |
Done: | |
if (String[0] != NULL) { | |
FreePool (String[0]); | |
} | |
if (String[1] != NULL) { | |
FreePool (String[1]); | |
} | |
if (StringPtr != NULL) { | |
FreePool (StringPtr); | |
} | |
return Status; | |
} | |
/** | |
Evaluate opcode EFI_IFR_MATCH. | |
@param FormSet Formset which contains this opcode. | |
@param Result Evaluation result for this opcode. | |
@retval EFI_SUCCESS Opcode evaluation success. | |
@retval Other Opcode evaluation failed. | |
**/ | |
EFI_STATUS | |
IfrMatch ( | |
IN FORM_BROWSER_FORMSET *FormSet, | |
OUT EFI_HII_VALUE *Result | |
) | |
{ | |
EFI_STATUS Status; | |
EFI_HII_VALUE Value; | |
CHAR16 *String[2]; | |
UINTN Index; | |
// | |
// String[0] - The string to search | |
// String[1] - pattern | |
// | |
String[0] = NULL; | |
String[1] = NULL; | |
Status = EFI_SUCCESS; | |
for (Index = 0; Index < 2; Index++) { | |
Status = PopExpression (&Value); | |
if (EFI_ERROR (Status)) { | |
goto Done; | |
} | |
if (Value.Type != EFI_IFR_TYPE_STRING) { | |
Status = EFI_UNSUPPORTED; | |
goto Done; | |
} | |
String[Index] = GetToken (Value.Value.string, FormSet->HiiHandle); | |
if (String [Index] == NULL) { | |
Status = EFI_NOT_FOUND; | |
goto Done; | |
} | |
} | |
Result->Type = EFI_IFR_TYPE_BOOLEAN; | |
Result->Value.b = mUnicodeCollation->MetaiMatch (mUnicodeCollation, String[0], String[1]); | |
Done: | |
if (String[0] != NULL) { | |
FreePool (String[0]); | |
} | |
if (String[1] != NULL) { | |
FreePool (String[1]); | |
} | |
return Status; | |
} | |
/** | |
Evaluate opcode EFI_IFR_FIND. | |
@param FormSet Formset which contains this opcode. | |
@param Format Case sensitive or insensitive. | |
@param Result Evaluation result for this opcode. | |
@retval EFI_SUCCESS Opcode evaluation success. | |
@retval Other Opcode evaluation failed. | |
**/ | |
EFI_STATUS | |
IfrFind ( | |
IN FORM_BROWSER_FORMSET *FormSet, | |
IN UINT8 Format, | |
OUT EFI_HII_VALUE *Result | |
) | |
{ | |
EFI_STATUS Status; | |
EFI_HII_VALUE Value; | |
CHAR16 *String[2]; | |
UINTN Base; | |
CHAR16 *StringPtr; | |
UINTN Index; | |
if (Format > EFI_IFR_FF_CASE_INSENSITIVE) { | |
return EFI_UNSUPPORTED; | |
} | |
Status = PopExpression (&Value); | |
if (EFI_ERROR (Status)) { | |
return Status; | |
} | |
if (Value.Type > EFI_IFR_TYPE_NUM_SIZE_64) { | |
return EFI_UNSUPPORTED; | |
} | |
Base = (UINTN) Value.Value.u64; | |
// | |
// String[0] - sub-string | |
// String[1] - The string to search | |
// | |
String[0] = NULL; | |
String[1] = NULL; | |
for (Index = 0; Index < 2; Index++) { | |
Status = PopExpression (&Value); | |
if (EFI_ERROR (Status)) { | |
goto Done; | |
} | |
if (Value.Type != EFI_IFR_TYPE_STRING) { | |
Status = EFI_UNSUPPORTED; | |
goto Done; | |
} | |
String[Index] = GetToken (Value.Value.string, FormSet->HiiHandle); | |
if (String[Index] == NULL) { | |
Status = EFI_NOT_FOUND; | |
goto Done; | |
} | |
if (Format == EFI_IFR_FF_CASE_INSENSITIVE) { | |
// | |
// Case insensitive, convert both string to upper case | |
// | |
IfrStrToUpper (String[Index]); | |
} | |
} | |
Result->Type = EFI_IFR_TYPE_NUM_SIZE_64; | |
if (Base >= StrLen (String[1])) { | |
Result->Value.u64 = 0xFFFFFFFFFFFFFFFFULL; | |
} else { | |
StringPtr = StrStr (String[1] + Base, String[0]); | |
Result->Value.u64 = (StringPtr == NULL) ? 0xFFFFFFFFFFFFFFFFULL : (StringPtr - String[1]); | |
} | |
Done: | |
if (String[0] != NULL) { | |
FreePool (String[0]); | |
} | |
if (String[1] != NULL) { | |
FreePool (String[1]); | |
} | |
return Status; | |
} | |
/** | |
Evaluate opcode EFI_IFR_MID. | |
@param FormSet Formset which contains this opcode. | |
@param Result Evaluation result for this opcode. | |
@retval EFI_SUCCESS Opcode evaluation success. | |
@retval Other Opcode evaluation failed. | |
**/ | |
EFI_STATUS | |
IfrMid ( | |
IN FORM_BROWSER_FORMSET *FormSet, | |
OUT EFI_HII_VALUE *Result | |
) | |
{ | |
EFI_STATUS Status; | |
EFI_HII_VALUE Value; | |
CHAR16 *String; | |
UINTN Base; | |
UINTN Length; | |
CHAR16 *SubString; | |
Status = PopExpression (&Value); | |
if (EFI_ERROR (Status)) { | |
return Status; | |
} | |
if (Value.Type > EFI_IFR_TYPE_NUM_SIZE_64) { | |
return EFI_UNSUPPORTED; | |
} | |
Length = (UINTN) Value.Value.u64; | |
Status = PopExpression (&Value); | |
if (EFI_ERROR (Status)) { | |
return Status; | |
} | |
if (Value.Type > EFI_IFR_TYPE_NUM_SIZE_64) { | |
return EFI_UNSUPPORTED; | |
} | |
Base = (UINTN) Value.Value.u64; | |
Status = PopExpression (&Value); | |
if (EFI_ERROR (Status)) { | |
return Status; | |
} | |
if (Value.Type != EFI_IFR_TYPE_STRING) { | |
return EFI_UNSUPPORTED; | |
} | |
String = GetToken (Value.Value.string, FormSet->HiiHandle); | |
if (String == NULL) { | |
return EFI_NOT_FOUND; | |
} | |
if (Length == 0 || Base >= StrLen (String)) { | |
SubString = gEmptyString; | |
} else { | |
SubString = String + Base; | |
if ((Base + Length) < StrLen (String)) { | |
SubString[Length] = L'\0'; | |
} | |
} | |
Result->Type = EFI_IFR_TYPE_STRING; | |
Result->Value.string = NewString (SubString, FormSet->HiiHandle); | |
FreePool (String); | |
return Status; | |
} | |
/** | |
Evaluate opcode EFI_IFR_TOKEN. | |
@param FormSet Formset which contains this opcode. | |
@param Result Evaluation result for this opcode. | |
@retval EFI_SUCCESS Opcode evaluation success. | |
@retval Other Opcode evaluation failed. | |
**/ | |
EFI_STATUS | |
IfrToken ( | |
IN FORM_BROWSER_FORMSET *FormSet, | |
OUT EFI_HII_VALUE *Result | |
) | |
{ | |
EFI_STATUS Status; | |
EFI_HII_VALUE Value; | |
CHAR16 *String[2]; | |
UINTN Count; | |
CHAR16 *Delimiter; | |
CHAR16 *SubString; | |
CHAR16 *StringPtr; | |
UINTN Index; | |
Status = PopExpression (&Value); | |
if (EFI_ERROR (Status)) { | |
return Status; | |
} | |
if (Value.Type > EFI_IFR_TYPE_NUM_SIZE_64) { | |
return EFI_UNSUPPORTED; | |
} | |
Count = (UINTN) Value.Value.u64; | |
// | |
// String[0] - Delimiter | |
// String[1] - The string to search | |
// | |
String[0] = NULL; | |
String[1] = NULL; | |
for (Index = 0; Index < 2; Index++) { | |
Status = PopExpression (&Value); | |
if (EFI_ERROR (Status)) { | |
goto Done; | |
} | |
if (Value.Type != EFI_IFR_TYPE_STRING) { | |
Status = EFI_UNSUPPORTED; | |
goto Done; | |
} | |
String[Index] = GetToken (Value.Value.string, FormSet->HiiHandle); | |
if (String[Index] == NULL) { | |
Status = EFI_NOT_FOUND; | |
goto Done; | |
} | |
} | |
Delimiter = String[0]; | |
SubString = String[1]; | |
while (Count > 0) { | |
SubString = StrStr (SubString, Delimiter); | |
if (SubString != NULL) { | |
// | |
// Skip over the delimiter | |
// | |
SubString = SubString + StrLen (Delimiter); | |
} else { | |
break; | |
} | |
Count--; | |
} | |
if (SubString == NULL) { | |
// | |
// nth delimited sub-string not found, push an empty string | |
// | |
SubString = gEmptyString; | |
} else { | |
// | |
// Put a NULL terminator for nth delimited sub-string | |
// | |
StringPtr = StrStr (SubString, Delimiter); | |
if (StringPtr != NULL) { | |
*StringPtr = L'\0'; | |
} | |
} | |
Result->Type = EFI_IFR_TYPE_STRING; | |
Result->Value.string = NewString (SubString, FormSet->HiiHandle); | |
Done: | |
if (String[0] != NULL) { | |
FreePool (String[0]); | |
} | |
if (String[1] != NULL) { | |
FreePool (String[1]); | |
} | |
return Status; | |
} | |
/** | |
Evaluate opcode EFI_IFR_SPAN. | |
@param FormSet Formset which contains this opcode. | |
@param Flags FIRST_MATCHING or FIRST_NON_MATCHING. | |
@param Result Evaluation result for this opcode. | |
@retval EFI_SUCCESS Opcode evaluation success. | |
@retval Other Opcode evaluation failed. | |
**/ | |
EFI_STATUS | |
IfrSpan ( | |
IN FORM_BROWSER_FORMSET *FormSet, | |
IN UINT8 Flags, | |
OUT EFI_HII_VALUE *Result | |
) | |
{ | |
EFI_STATUS Status; | |
EFI_HII_VALUE Value; | |
CHAR16 *String[2]; | |
CHAR16 *Charset; | |
UINTN Base; | |
UINTN Index; | |
CHAR16 *StringPtr; | |
BOOLEAN Found; | |
Status = PopExpression (&Value); | |
if (EFI_ERROR (Status)) { | |
return Status; | |
} | |
if (Value.Type > EFI_IFR_TYPE_NUM_SIZE_64) { | |
return EFI_UNSUPPORTED; | |
} | |
Base = (UINTN) Value.Value.u64; | |
// | |
// String[0] - Charset | |
// String[1] - The string to search | |
// | |
String[0] = NULL; | |
String[1] = NULL; | |
for (Index = 0; Index < 2; Index++) { | |
Status = PopExpression (&Value); | |
if (EFI_ERROR (Status)) { | |
goto Done; | |
} | |
if (Value.Type != EFI_IFR_TYPE_STRING) { | |
Status = EFI_UNSUPPORTED; | |
goto Done; | |
} | |
String[Index] = GetToken (Value.Value.string, FormSet->HiiHandle); | |
if (String [Index] == NULL) { | |
Status = EFI_NOT_FOUND; | |
goto Done; | |
} | |
} | |
if (Base >= StrLen (String[1])) { | |
Status = EFI_UNSUPPORTED; | |
goto Done; | |
} | |
Found = FALSE; | |
StringPtr = String[1] + Base; | |
Charset = String[0]; | |
while (*StringPtr != 0 && !Found) { | |
Index = 0; | |
while (Charset[Index] != 0) { | |
if (*StringPtr >= Charset[Index] && *StringPtr <= Charset[Index + 1]) { | |
if (Flags == EFI_IFR_FLAGS_FIRST_MATCHING) { | |
Found = TRUE; | |
break; | |
} | |
} else { | |
if (Flags == EFI_IFR_FLAGS_FIRST_NON_MATCHING) { | |
Found = TRUE; | |
break; | |
} | |
} | |
// | |
// Skip characters pair representing low-end of a range and high-end of a range | |
// | |
Index += 2; | |
} | |
if (!Found) { | |
StringPtr++; | |
} | |
} | |
Result->Type = EFI_IFR_TYPE_NUM_SIZE_64; | |
Result->Value.u64 = StringPtr - String[1]; | |
Done: | |
if (String[0] != NULL) { | |
FreePool (String[0]); | |
} | |
if (String[1] != NULL) { | |
FreePool (String[1]); | |
} | |
return Status; | |
} | |
/** | |
Zero extend integer/boolean/date/time to UINT64 for comparing. | |
@param Value HII Value to be converted. | |
**/ | |
VOID | |
ExtendValueToU64 ( | |
IN EFI_HII_VALUE *Value | |
) | |
{ | |
UINT64 Temp; | |
Temp = 0; | |
switch (Value->Type) { | |
case EFI_IFR_TYPE_NUM_SIZE_8: | |
Temp = Value->Value.u8; | |
break; | |
case EFI_IFR_TYPE_NUM_SIZE_16: | |
Temp = Value->Value.u16; | |
break; | |
case EFI_IFR_TYPE_NUM_SIZE_32: | |
Temp = Value->Value.u32; | |
break; | |
case EFI_IFR_TYPE_BOOLEAN: | |
Temp = Value->Value.b; | |
break; | |
case EFI_IFR_TYPE_TIME: | |
Temp = Value->Value.u32 & 0xffffff; | |
break; | |
case EFI_IFR_TYPE_DATE: | |
Temp = Value->Value.u32; | |
break; | |
default: | |
return; | |
} | |
Value->Value.u64 = Temp; | |
} | |
/** | |
Compare two Hii value. | |
@param Value1 Expression value to compare on left-hand. | |
@param Value2 Expression value to compare on right-hand. | |
@param HiiHandle Only required for string compare. | |
@retval EFI_INVALID_PARAMETER Could not perform comparation on two values. | |
@retval 0 Two operators equeal. | |
@return Positive value if Value1 is greater than Value2. | |
@retval Negative value if Value1 is less than Value2. | |
**/ | |
INTN | |
CompareHiiValue ( | |
IN EFI_HII_VALUE *Value1, | |
IN EFI_HII_VALUE *Value2, | |
IN EFI_HII_HANDLE HiiHandle OPTIONAL | |
) | |
{ | |
INTN Result; | |
INT64 Temp64; | |
CHAR16 *Str1; | |
CHAR16 *Str2; | |
if (Value1->Type >= EFI_IFR_TYPE_OTHER || Value2->Type >= EFI_IFR_TYPE_OTHER ) { | |
return EFI_INVALID_PARAMETER; | |
} | |
if (Value1->Type == EFI_IFR_TYPE_STRING || Value2->Type == EFI_IFR_TYPE_STRING ) { | |
if (Value1->Type != Value2->Type) { | |
// | |
// Both Operator should be type of String | |
// | |
return EFI_INVALID_PARAMETER; | |
} | |
if (Value1->Value.string == 0 || Value2->Value.string == 0) { | |
// | |
// StringId 0 is reserved | |
// | |
return EFI_INVALID_PARAMETER; | |
} | |
if (Value1->Value.string == Value2->Value.string) { | |
return 0; | |
} | |
Str1 = GetToken (Value1->Value.string, HiiHandle); | |
if (Str1 == NULL) { | |
// | |
// String not found | |
// | |
return EFI_INVALID_PARAMETER; | |
} | |
Str2 = GetToken (Value2->Value.string, HiiHandle); | |
if (Str2 == NULL) { | |
FreePool (Str1); | |
return EFI_INVALID_PARAMETER; | |
} | |
Result = StrCmp (Str1, Str2); | |
FreePool (Str1); | |
FreePool (Str2); | |
return Result; | |
} | |
// | |
// Take remain types(integer, boolean, date/time) as integer | |
// | |
Temp64 = (INT64) (Value1->Value.u64 - Value2->Value.u64); | |
if (Temp64 > 0) { | |
Result = 1; | |
} else if (Temp64 < 0) { | |
Result = -1; | |
} else { | |
Result = 0; | |
} | |
return Result; | |
} | |
/** | |
Evaluate the result of a HII expression. | |
If Expression is NULL, then ASSERT. | |
@param FormSet FormSet associated with this expression. | |
@param Form Form associated with this expression. | |
@param Expression Expression to be evaluated. | |
@retval EFI_SUCCESS The expression evaluated successfuly | |
@retval EFI_NOT_FOUND The Question which referenced by a QuestionId | |
could not be found. | |
@retval EFI_OUT_OF_RESOURCES There is not enough system memory to grow the | |
stack. | |
@retval EFI_ACCESS_DENIED The pop operation underflowed the stack | |
@retval EFI_INVALID_PARAMETER Syntax error with the Expression | |
**/ | |
EFI_STATUS | |
EvaluateExpression ( | |
IN FORM_BROWSER_FORMSET *FormSet, | |
IN FORM_BROWSER_FORM *Form, | |
IN OUT FORM_EXPRESSION *Expression | |
) | |
{ | |
EFI_STATUS Status; | |
LIST_ENTRY *Link; | |
EXPRESSION_OPCODE *OpCode; | |
FORM_BROWSER_STATEMENT *Question; | |
FORM_BROWSER_STATEMENT *Question2; | |
UINT16 Index; | |
EFI_HII_VALUE Data1; | |
EFI_HII_VALUE Data2; | |
EFI_HII_VALUE Data3; | |
FORM_EXPRESSION *RuleExpression; | |
EFI_HII_VALUE *Value; | |
INTN Result; | |
CHAR16 *StrPtr; | |
UINT32 TempValue; | |
// | |
// Always reset the stack before evaluating an Expression | |
// | |
ResetExpressionStack (); | |
ASSERT (Expression != NULL); | |
Expression->Result.Type = EFI_IFR_TYPE_OTHER; | |
Link = GetFirstNode (&Expression->OpCodeListHead); | |
while (!IsNull (&Expression->OpCodeListHead, Link)) { | |
OpCode = EXPRESSION_OPCODE_FROM_LINK (Link); | |
Link = GetNextNode (&Expression->OpCodeListHead, Link); | |
ZeroMem (&Data1, sizeof (EFI_HII_VALUE)); | |
ZeroMem (&Data2, sizeof (EFI_HII_VALUE)); | |
ZeroMem (&Data3, sizeof (EFI_HII_VALUE)); | |
Value = &Data3; | |
Value->Type = EFI_IFR_TYPE_BOOLEAN; | |
Status = EFI_SUCCESS; | |
switch (OpCode->Operand) { | |
// | |
// Built-in functions | |
// | |
case EFI_IFR_EQ_ID_VAL_OP: | |
Question = IdToQuestion (FormSet, Form, OpCode->QuestionId); | |
if (Question == NULL) { | |
return EFI_NOT_FOUND; | |
} | |
Result = CompareHiiValue (&Question->HiiValue, &OpCode->Value, NULL); | |
if (Result == EFI_INVALID_PARAMETER) { | |
return EFI_INVALID_PARAMETER; | |
} | |
Value->Value.b = (BOOLEAN) ((Result == 0) ? TRUE : FALSE); | |
break; | |
case EFI_IFR_EQ_ID_ID_OP: | |
Question = IdToQuestion (FormSet, Form, OpCode->QuestionId); | |
if (Question == NULL) { | |
return EFI_NOT_FOUND; | |
} | |
Question2 = IdToQuestion (FormSet, Form, OpCode->QuestionId2); | |
if (Question2 == NULL) { | |
return EFI_NOT_FOUND; | |
} | |
Result = CompareHiiValue (&Question->HiiValue, &Question2->HiiValue, FormSet->HiiHandle); | |
if (Result == EFI_INVALID_PARAMETER) { | |
return EFI_INVALID_PARAMETER; | |
} | |
Value->Value.b = (BOOLEAN) ((Result == 0) ? TRUE : FALSE); | |
break; | |
case EFI_IFR_EQ_ID_LIST_OP: | |
Question = IdToQuestion (FormSet, Form, OpCode->QuestionId); | |
if (Question == NULL) { | |
return EFI_NOT_FOUND; | |
} | |
Value->Value.b = FALSE; | |
for (Index =0; Index < OpCode->ListLength; Index++) { | |
if (Question->HiiValue.Value.u16 == OpCode->ValueList[Index]) { | |
Value->Value.b = TRUE; | |
break; | |
} | |
} | |
break; | |
case EFI_IFR_DUP_OP: | |
Status = PopExpression (Value); | |
if (EFI_ERROR (Status)) { | |
return Status; | |
} | |
Status = PushExpression (Value); | |
break; | |
case EFI_IFR_QUESTION_REF1_OP: | |
case EFI_IFR_THIS_OP: | |
Question = IdToQuestion (FormSet, Form, OpCode->QuestionId); | |
if (Question == NULL) { | |
return EFI_NOT_FOUND; | |
} | |
Value = &Question->HiiValue; | |
break; | |
case EFI_IFR_QUESTION_REF3_OP: | |
if (OpCode->DevicePath == 0) { | |
// | |
// EFI_IFR_QUESTION_REF3 | |
// Pop an expression from the expression stack | |
// | |
Status = PopExpression (Value); | |
if (EFI_ERROR (Status)) { | |
return Status; | |
} | |
// | |
// Validate the expression value | |
// | |
if ((Value->Type > EFI_IFR_TYPE_NUM_SIZE_64) || (Value->Value.u64 > 0xffff)) { | |
return EFI_NOT_FOUND; | |
} | |
Question = IdToQuestion (FormSet, Form, Value->Value.u16); | |
if (Question == NULL) { | |
return EFI_NOT_FOUND; | |
} | |
// | |
// push the questions' value on to the expression stack | |
// | |
Value = &Question->HiiValue; | |
} else { | |
// | |
// BUGBUG: push 0 for EFI_IFR_QUESTION_REF3_2 and EFI_IFR_QUESTION_REF3_3, | |
// since it is impractical to evaluate the value of a Question in another | |
// Hii Package list. | |
// | |
ZeroMem (Value, sizeof (EFI_HII_VALUE)); | |
} | |
break; | |
case EFI_IFR_RULE_REF_OP: | |
// | |
// Find expression for this rule | |
// | |
RuleExpression = RuleIdToExpression (Form, OpCode->RuleId); | |
if (RuleExpression == NULL) { | |
return EFI_NOT_FOUND; | |
} | |
// | |
// Evaluate this rule expression | |
// | |
Status = EvaluateExpression (FormSet, Form, RuleExpression); | |
if (EFI_ERROR (Status)) { | |
return Status; | |
} | |
Value = &RuleExpression->Result; | |
break; | |
case EFI_IFR_STRING_REF1_OP: | |
Value->Type = EFI_IFR_TYPE_STRING; | |
Value->Value.string = OpCode->Value.Value.string; | |
break; | |
// | |
// Constant | |
// | |
case EFI_IFR_TRUE_OP: | |
case EFI_IFR_FALSE_OP: | |
case EFI_IFR_ONE_OP: | |
case EFI_IFR_ONES_OP: | |
case EFI_IFR_UINT8_OP: | |
case EFI_IFR_UINT16_OP: | |
case EFI_IFR_UINT32_OP: | |
case EFI_IFR_UINT64_OP: | |
case EFI_IFR_UNDEFINED_OP: | |
case EFI_IFR_VERSION_OP: | |
case EFI_IFR_ZERO_OP: | |
Value = &OpCode->Value; | |
break; | |
// | |
// unary-op | |
// | |
case EFI_IFR_LENGTH_OP: | |
Status = PopExpression (Value); | |
if (EFI_ERROR (Status)) { | |
return Status; | |
} | |
if (Value->Type != EFI_IFR_TYPE_STRING) { | |
return EFI_INVALID_PARAMETER; | |
} | |
StrPtr = GetToken (Value->Value.string, FormSet->HiiHandle); | |
if (StrPtr == NULL) { | |
return EFI_INVALID_PARAMETER; | |
} | |
Value->Type = EFI_IFR_TYPE_NUM_SIZE_64; | |
Value->Value.u64 = StrLen (StrPtr); | |
FreePool (StrPtr); | |
break; | |
case EFI_IFR_NOT_OP: | |
Status = PopExpression (Value); | |
if (EFI_ERROR (Status)) { | |
return Status; | |
} | |
if (Value->Type != EFI_IFR_TYPE_BOOLEAN) { | |
return EFI_INVALID_PARAMETER; | |
} | |
Value->Value.b = (BOOLEAN) (!Value->Value.b); | |
break; | |
case EFI_IFR_QUESTION_REF2_OP: | |
// | |
// Pop an expression from the expression stack | |
// | |
Status = PopExpression (Value); | |
if (EFI_ERROR (Status)) { | |
return Status; | |
} | |
// | |
// Validate the expression value | |
// | |
if ((Value->Type > EFI_IFR_TYPE_NUM_SIZE_64) || (Value->Value.u64 > 0xffff)) { | |
return EFI_NOT_FOUND; | |
} | |
Question = IdToQuestion (FormSet, Form, Value->Value.u16); | |
if (Question == NULL) { | |
return EFI_NOT_FOUND; | |
} | |
Value = &Question->HiiValue; | |
break; | |
case EFI_IFR_STRING_REF2_OP: | |
// | |
// Pop an expression from the expression stack | |
// | |
Status = PopExpression (Value); | |
if (EFI_ERROR (Status)) { | |
return Status; | |
} | |
// | |
// Validate the expression value | |
// | |
if ((Value->Type > EFI_IFR_TYPE_NUM_SIZE_64) || (Value->Value.u64 > 0xffff)) { | |
return EFI_NOT_FOUND; | |
} | |
Value->Type = EFI_IFR_TYPE_STRING; | |
StrPtr = GetToken (Value->Value.u16, FormSet->HiiHandle); | |
if (StrPtr == NULL) { | |
// | |
// If String not exit, push an empty string | |
// | |
Value->Value.string = NewString (gEmptyString, FormSet->HiiHandle); | |
} else { | |
Index = (UINT16) Value->Value.u64; | |
Value->Value.string = Index; | |
FreePool (StrPtr); | |
} | |
break; | |
case EFI_IFR_TO_BOOLEAN_OP: | |
// | |
// Pop an expression from the expression stack | |
// | |
Status = PopExpression (Value); | |
if (EFI_ERROR (Status)) { | |
return Status; | |
} | |
// | |
// Convert an expression to a Boolean | |
// | |
if (Value->Type <= EFI_IFR_TYPE_DATE) { | |
// | |
// When converting from an unsigned integer, zero will be converted to | |
// FALSE and any other value will be converted to TRUE. | |
// | |
Value->Value.b = (BOOLEAN) ((Value->Value.u64) ? TRUE : FALSE); | |
Value->Type = EFI_IFR_TYPE_BOOLEAN; | |
} else if (Value->Type == EFI_IFR_TYPE_STRING) { | |
// | |
// When converting from a string, if case-insensitive compare | |
// with "true" is True, then push True. If a case-insensitive compare | |
// with "false" is True, then push False. | |
// | |
StrPtr = GetToken (Value->Value.string, FormSet->HiiHandle); | |
if (StrPtr == NULL) { | |
return EFI_INVALID_PARAMETER; | |
} | |
if ((StrCmp (StrPtr, L"true") == 0) || (StrCmp (StrPtr, L"false") == 0)){ | |
Value->Value.b = TRUE; | |
} else { | |
Value->Value.b = FALSE; | |
} | |
FreePool (StrPtr); | |
Value->Type = EFI_IFR_TYPE_BOOLEAN; | |
} | |
break; | |
case EFI_IFR_TO_STRING_OP: | |
Status = IfrToString (FormSet, OpCode->Format, Value); | |
break; | |
case EFI_IFR_TO_UINT_OP: | |
Status = IfrToUint (FormSet, Value); | |
break; | |
case EFI_IFR_TO_LOWER_OP: | |
case EFI_IFR_TO_UPPER_OP: | |
Status = InitializeUnicodeCollationProtocol (); | |
if (EFI_ERROR (Status)) { | |
return Status; | |
} | |
Status = PopExpression (Value); | |
if (EFI_ERROR (Status)) { | |
return Status; | |
} | |
if (Value->Type != EFI_IFR_TYPE_STRING) { | |
return EFI_UNSUPPORTED; | |
} | |
StrPtr = GetToken (Value->Value.string, FormSet->HiiHandle); | |
if (StrPtr == NULL) { | |
return EFI_NOT_FOUND; | |
} | |
if (OpCode->Operand == EFI_IFR_TO_LOWER_OP) { | |
mUnicodeCollation->StrLwr (mUnicodeCollation, StrPtr); | |
} else { | |
mUnicodeCollation->StrUpr (mUnicodeCollation, StrPtr); | |
} | |
Value->Value.string = NewString (StrPtr, FormSet->HiiHandle); | |
FreePool (StrPtr); | |
break; | |
case EFI_IFR_BITWISE_NOT_OP: | |
// | |
// Pop an expression from the expression stack | |
// | |
Status = PopExpression (Value); | |
if (EFI_ERROR (Status)) { | |
return Status; | |
} | |
if (Value->Type > EFI_IFR_TYPE_DATE) { | |
return EFI_INVALID_PARAMETER; | |
} | |
Value->Type = EFI_IFR_TYPE_NUM_SIZE_64; | |
Value->Value.u64 = ~Value->Value.u64; | |
break; | |
// | |
// binary-op | |
// | |
case EFI_IFR_ADD_OP: | |
case EFI_IFR_SUBTRACT_OP: | |
case EFI_IFR_MULTIPLY_OP: | |
case EFI_IFR_DIVIDE_OP: | |
case EFI_IFR_MODULO_OP: | |
case EFI_IFR_BITWISE_AND_OP: | |
case EFI_IFR_BITWISE_OR_OP: | |
case EFI_IFR_SHIFT_LEFT_OP: | |
case EFI_IFR_SHIFT_RIGHT_OP: | |
// | |
// Pop an expression from the expression stack | |
// | |
Status = PopExpression (&Data2); | |
if (EFI_ERROR (Status)) { | |
return Status; | |
} | |
if (Data2.Type > EFI_IFR_TYPE_DATE) { | |
return EFI_INVALID_PARAMETER; | |
} | |
// | |
// Pop another expression from the expression stack | |
// | |
Status = PopExpression (&Data1); | |
if (EFI_ERROR (Status)) { | |
return Status; | |
} | |
if (Data1.Type > EFI_IFR_TYPE_DATE) { | |
return EFI_INVALID_PARAMETER; | |
} | |
Value->Type = EFI_IFR_TYPE_NUM_SIZE_64; | |
switch (OpCode->Operand) { | |
case EFI_IFR_ADD_OP: | |
Value->Value.u64 = Data1.Value.u64 + Data2.Value.u64; | |
break; | |
case EFI_IFR_SUBTRACT_OP: | |
Value->Value.u64 = Data1.Value.u64 - Data2.Value.u64; | |
break; | |
case EFI_IFR_MULTIPLY_OP: | |
Value->Value.u64 = MultU64x32 (Data1.Value.u64, (UINT32) Data2.Value.u64); | |
break; | |
case EFI_IFR_DIVIDE_OP: | |
Value->Value.u64 = DivU64x32 (Data1.Value.u64, (UINT32) Data2.Value.u64); | |
break; | |
case EFI_IFR_MODULO_OP: | |
DivU64x32Remainder (Data1.Value.u64, (UINT32) Data2.Value.u64, &TempValue); | |
Value->Value.u64 = TempValue; | |
break; | |
case EFI_IFR_BITWISE_AND_OP: | |
Value->Value.u64 = Data1.Value.u64 & Data2.Value.u64; | |
break; | |
case EFI_IFR_BITWISE_OR_OP: | |
Value->Value.u64 = Data1.Value.u64 | Data2.Value.u64; | |
break; | |
case EFI_IFR_SHIFT_LEFT_OP: | |
Value->Value.u64 = LShiftU64 (Data1.Value.u64, (UINTN) Data2.Value.u64); | |
break; | |
case EFI_IFR_SHIFT_RIGHT_OP: | |
Value->Value.u64 = RShiftU64 (Data1.Value.u64, (UINTN) Data2.Value.u64); | |
break; | |
default: | |
break; | |
} | |
break; | |
case EFI_IFR_AND_OP: | |
case EFI_IFR_OR_OP: | |
// | |
// Two Boolean operator | |
// | |
Status = PopExpression (&Data2); | |
if (EFI_ERROR (Status)) { | |
return Status; | |
} | |
if (Data2.Type != EFI_IFR_TYPE_BOOLEAN) { | |
return EFI_INVALID_PARAMETER; | |
} | |
// | |
// Pop another expression from the expression stack | |
// | |
Status = PopExpression (&Data1); | |
if (EFI_ERROR (Status)) { | |
return Status; | |
} | |
if (Data1.Type != EFI_IFR_TYPE_BOOLEAN) { | |
return EFI_INVALID_PARAMETER; | |
} | |
if (OpCode->Operand == EFI_IFR_AND_OP) { | |
Value->Value.b = (BOOLEAN) (Data1.Value.b && Data2.Value.b); | |
} else { | |
Value->Value.b = (BOOLEAN) (Data1.Value.b || Data2.Value.b); | |
} | |
break; | |
case EFI_IFR_EQUAL_OP: | |
case EFI_IFR_NOT_EQUAL_OP: | |
case EFI_IFR_GREATER_EQUAL_OP: | |
case EFI_IFR_GREATER_THAN_OP: | |
case EFI_IFR_LESS_EQUAL_OP: | |
case EFI_IFR_LESS_THAN_OP: | |
// | |
// Compare two integer, string, boolean or date/time | |
// | |
Status = PopExpression (&Data2); | |
if (EFI_ERROR (Status)) { | |
return Status; | |
} | |
if (Data2.Type > EFI_IFR_TYPE_BOOLEAN && Data2.Type != EFI_IFR_TYPE_STRING) { | |
return EFI_INVALID_PARAMETER; | |
} | |
// | |
// Pop another expression from the expression stack | |
// | |
Status = PopExpression (&Data1); | |
if (EFI_ERROR (Status)) { | |
return Status; | |
} | |
Result = CompareHiiValue (&Data1, &Data2, FormSet->HiiHandle); | |
if (Result == EFI_INVALID_PARAMETER) { | |
return EFI_INVALID_PARAMETER; | |
} | |
switch (OpCode->Operand) { | |
case EFI_IFR_EQUAL_OP: | |
Value->Value.b = (BOOLEAN) ((Result == 0) ? TRUE : FALSE); | |
break; | |
case EFI_IFR_NOT_EQUAL_OP: | |
Value->Value.b = (BOOLEAN) ((Result == 0) ? TRUE : FALSE); | |
break; | |
case EFI_IFR_GREATER_EQUAL_OP: | |
Value->Value.b = (BOOLEAN) ((Result >= 0) ? TRUE : FALSE); | |
break; | |
case EFI_IFR_GREATER_THAN_OP: | |
Value->Value.b = (BOOLEAN) ((Result > 0) ? TRUE : FALSE); | |
break; | |
case EFI_IFR_LESS_EQUAL_OP: | |
Value->Value.b = (BOOLEAN) ((Result <= 0) ? TRUE : FALSE); | |
break; | |
case EFI_IFR_LESS_THAN_OP: | |
Value->Value.b = (BOOLEAN) ((Result < 0) ? TRUE : FALSE); | |
break; | |
default: | |
break; | |
} | |
break; | |
case EFI_IFR_MATCH_OP: | |
Status = IfrMatch (FormSet, Value); | |
break; | |
case EFI_IFR_CATENATE_OP: | |
Status = IfrCatenate (FormSet, Value); | |
break; | |
// | |
// ternary-op | |
// | |
case EFI_IFR_CONDITIONAL_OP: | |
// | |
// Pop third expression from the expression stack | |
// | |
Status = PopExpression (&Data3); | |
if (EFI_ERROR (Status)) { | |
return Status; | |
} | |
// | |
// Pop second expression from the expression stack | |
// | |
Status = PopExpression (&Data2); | |
if (EFI_ERROR (Status)) { | |
return Status; | |
} | |
// | |
// Pop first expression from the expression stack | |
// | |
Status = PopExpression (&Data1); | |
if (EFI_ERROR (Status)) { | |
return Status; | |
} | |
if (Data1.Type != EFI_IFR_TYPE_BOOLEAN) { | |
return EFI_INVALID_PARAMETER; | |
} | |
if (Data1.Value.b) { | |
Value = &Data3; | |
} else { | |
Value = &Data2; | |
} | |
break; | |
case EFI_IFR_FIND_OP: | |
Status = IfrFind (FormSet, OpCode->Format, Value); | |
break; | |
case EFI_IFR_MID_OP: | |
Status = IfrMid (FormSet, Value); | |
break; | |
case EFI_IFR_TOKEN_OP: | |
Status = IfrToken (FormSet, Value); | |
break; | |
case EFI_IFR_SPAN_OP: | |
Status = IfrSpan (FormSet, OpCode->Flags, Value); | |
break; | |
default: | |
break; | |
} | |
if (EFI_ERROR (Status)) { | |
return Status; | |
} | |
Status = PushExpression (Value); | |
if (EFI_ERROR (Status)) { | |
return Status; | |
} | |
} | |
// | |
// Pop the final result from expression stack | |
// | |
Value = &Data1; | |
Status = PopExpression (Value); | |
if (EFI_ERROR (Status)) { | |
return Status; | |
} | |
// | |
// After evaluating an expression, there should be only one value left on the expression stack | |
// | |
if (PopExpression (Value) != EFI_ACCESS_DENIED) { | |
return EFI_INVALID_PARAMETER; | |
} | |
CopyMem (&Expression->Result, Value, sizeof (EFI_HII_VALUE)); | |
return EFI_SUCCESS; | |
} |