2022-06-21 04:52:31 +02:00
|
|
|
/** @file
|
|
|
|
IA32-specific functions for unit-testing INTN and UINTN functions in
|
|
|
|
SafeIntLib.
|
|
|
|
|
|
|
|
Copyright (c) Microsoft Corporation.<BR>
|
|
|
|
Copyright (c) 2019 - 2020, Intel Corporation. All rights reserved.<BR>
|
|
|
|
SPDX-License-Identifier: BSD-2-Clause-Patent
|
|
|
|
|
|
|
|
**/
|
|
|
|
|
|
|
|
#include <gtest/gtest.h>
|
|
|
|
extern "C" {
|
|
|
|
#include <Base.h>
|
|
|
|
#include <Library/SafeIntLib.h>
|
|
|
|
}
|
|
|
|
|
2023-08-16 23:15:19 +02:00
|
|
|
TEST (ConversionTestSuite, TestSafeInt32ToUintn) {
|
2022-06-21 04:52:31 +02:00
|
|
|
RETURN_STATUS Status;
|
2023-08-16 23:15:19 +02:00
|
|
|
INT32 Operand;
|
|
|
|
UINTN Result;
|
2022-06-21 04:52:31 +02:00
|
|
|
|
|
|
|
//
|
|
|
|
// If Operand is non-negative, then it's a cast
|
|
|
|
//
|
|
|
|
Operand = 0x5bababab;
|
|
|
|
Result = 0;
|
|
|
|
Status = SafeInt32ToUintn (Operand, &Result);
|
|
|
|
ASSERT_EQ (Status, RETURN_SUCCESS);
|
|
|
|
ASSERT_EQ ((UINTN)0x5bababab, Result);
|
|
|
|
|
|
|
|
//
|
|
|
|
// Otherwise should result in an error status
|
|
|
|
//
|
|
|
|
Operand = (-1537977259);
|
|
|
|
Status = SafeInt32ToUintn (Operand, &Result);
|
|
|
|
ASSERT_EQ (RETURN_BUFFER_TOO_SMALL, Status);
|
|
|
|
}
|
|
|
|
|
2023-08-16 23:15:19 +02:00
|
|
|
TEST (ConversionTestSuite, TestSafeUint32ToIntn) {
|
2022-06-21 04:52:31 +02:00
|
|
|
RETURN_STATUS Status;
|
2023-08-16 23:15:19 +02:00
|
|
|
UINT32 Operand;
|
|
|
|
INTN Result;
|
2022-06-21 04:52:31 +02:00
|
|
|
|
|
|
|
//
|
|
|
|
// If Operand is <= MAX_INTN, then it's a cast
|
|
|
|
//
|
|
|
|
Operand = 0x5bababab;
|
|
|
|
Result = 0;
|
|
|
|
Status = SafeUint32ToIntn (Operand, &Result);
|
|
|
|
ASSERT_EQ (Status, RETURN_SUCCESS);
|
|
|
|
ASSERT_EQ (0x5bababab, Result);
|
|
|
|
|
|
|
|
//
|
|
|
|
// Otherwise should result in an error status
|
|
|
|
//
|
|
|
|
Operand = (0xabababab);
|
|
|
|
Status = SafeUint32ToIntn (Operand, &Result);
|
|
|
|
ASSERT_EQ (RETURN_BUFFER_TOO_SMALL, Status);
|
|
|
|
}
|
|
|
|
|
2023-08-16 23:15:19 +02:00
|
|
|
TEST (ConversionTestSuite, TestSafeIntnToInt32) {
|
2022-06-21 04:52:31 +02:00
|
|
|
RETURN_STATUS Status;
|
2023-08-16 23:15:19 +02:00
|
|
|
INTN Operand;
|
|
|
|
INT32 Result;
|
2022-06-21 04:52:31 +02:00
|
|
|
|
|
|
|
//
|
|
|
|
// INTN is same as INT32 in IA32, so this is just a cast
|
|
|
|
//
|
|
|
|
Operand = 0x5bababab;
|
|
|
|
Result = 0;
|
|
|
|
Status = SafeIntnToInt32 (Operand, &Result);
|
|
|
|
ASSERT_EQ (Status, RETURN_SUCCESS);
|
|
|
|
ASSERT_EQ (0x5bababab, Result);
|
|
|
|
}
|
|
|
|
|
2023-08-16 23:15:19 +02:00
|
|
|
TEST (ConversionTestSuite, TestSafeIntnToUint32) {
|
2022-06-21 04:52:31 +02:00
|
|
|
RETURN_STATUS Status;
|
2023-08-16 23:15:19 +02:00
|
|
|
INTN Operand;
|
|
|
|
UINT32 Result;
|
2022-06-21 04:52:31 +02:00
|
|
|
|
|
|
|
//
|
|
|
|
// If Operand is non-negative, then it's a cast
|
|
|
|
//
|
|
|
|
Operand = 0x5bababab;
|
|
|
|
Result = 0;
|
|
|
|
Status = SafeIntnToUint32 (Operand, &Result);
|
|
|
|
ASSERT_EQ (Status, RETURN_SUCCESS);
|
|
|
|
ASSERT_EQ ((UINT32)0x5bababab, Result);
|
|
|
|
|
|
|
|
//
|
|
|
|
// Otherwise should result in an error status
|
|
|
|
//
|
|
|
|
Operand = (-1537977259);
|
|
|
|
Status = SafeIntnToUint32 (Operand, &Result);
|
|
|
|
ASSERT_EQ (RETURN_BUFFER_TOO_SMALL, Status);
|
|
|
|
}
|
|
|
|
|
2023-08-16 23:15:19 +02:00
|
|
|
TEST (ConversionTestSuite, TestSafeUintnToUint32) {
|
2022-06-21 04:52:31 +02:00
|
|
|
RETURN_STATUS Status;
|
2023-08-16 23:15:19 +02:00
|
|
|
UINTN Operand;
|
|
|
|
UINT32 Result;
|
2022-06-21 04:52:31 +02:00
|
|
|
|
|
|
|
//
|
|
|
|
// UINTN is same as UINT32 in IA32, so this is just a cast
|
|
|
|
//
|
|
|
|
Operand = 0xabababab;
|
|
|
|
Result = 0;
|
|
|
|
Status = SafeUintnToUint32 (Operand, &Result);
|
|
|
|
ASSERT_EQ (Status, RETURN_SUCCESS);
|
|
|
|
ASSERT_EQ (0xabababab, Result);
|
|
|
|
}
|
|
|
|
|
2023-08-16 23:15:19 +02:00
|
|
|
TEST (ConversionTestSuite, TestSafeUintnToIntn) {
|
2022-06-21 04:52:31 +02:00
|
|
|
RETURN_STATUS Status;
|
2023-08-16 23:15:19 +02:00
|
|
|
UINTN Operand;
|
|
|
|
INTN Result;
|
2022-06-21 04:52:31 +02:00
|
|
|
|
|
|
|
//
|
|
|
|
// If Operand is <= MAX_INTN, then it's a cast
|
|
|
|
//
|
|
|
|
Operand = 0x5bababab;
|
|
|
|
Result = 0;
|
|
|
|
Status = SafeUintnToIntn (Operand, &Result);
|
|
|
|
ASSERT_EQ (Status, RETURN_SUCCESS);
|
|
|
|
ASSERT_EQ (0x5bababab, Result);
|
|
|
|
|
|
|
|
//
|
|
|
|
// Otherwise should result in an error status
|
|
|
|
//
|
|
|
|
Operand = (0xabababab);
|
|
|
|
Status = SafeUintnToIntn (Operand, &Result);
|
|
|
|
ASSERT_EQ (RETURN_BUFFER_TOO_SMALL, Status);
|
|
|
|
}
|
|
|
|
|
2023-08-16 23:15:19 +02:00
|
|
|
TEST (ConversionTestSuite, TestSafeUintnToInt64) {
|
2022-06-21 04:52:31 +02:00
|
|
|
RETURN_STATUS Status;
|
2023-08-16 23:15:19 +02:00
|
|
|
UINTN Operand;
|
|
|
|
INT64 Result;
|
2022-06-21 04:52:31 +02:00
|
|
|
|
|
|
|
//
|
|
|
|
// UINTN is same as UINT32 in IA32, and UINT32 is a subset of
|
|
|
|
// INT64, so this is just a cast
|
|
|
|
//
|
|
|
|
Operand = 0xabababab;
|
|
|
|
Result = 0;
|
|
|
|
Status = SafeUintnToInt64 (Operand, &Result);
|
|
|
|
ASSERT_EQ (Status, RETURN_SUCCESS);
|
|
|
|
ASSERT_EQ (0xabababab, Result);
|
|
|
|
}
|
|
|
|
|
2023-08-16 23:15:19 +02:00
|
|
|
TEST (ConversionTestSuite, TestSafeInt64ToIntn) {
|
2022-06-21 04:52:31 +02:00
|
|
|
RETURN_STATUS Status;
|
2023-08-16 23:15:19 +02:00
|
|
|
INT64 Operand;
|
|
|
|
INTN Result;
|
2022-06-21 04:52:31 +02:00
|
|
|
|
|
|
|
//
|
|
|
|
// If Operand is between MIN_INTN and MAX_INTN2 inclusive, then it's a cast
|
|
|
|
//
|
|
|
|
Operand = 0x5bababab;
|
|
|
|
Result = 0;
|
|
|
|
Status = SafeInt64ToIntn (Operand, &Result);
|
|
|
|
ASSERT_EQ (Status, RETURN_SUCCESS);
|
|
|
|
ASSERT_EQ (0x5bababab, Result);
|
|
|
|
|
|
|
|
Operand = (-1537977259);
|
|
|
|
Status = SafeInt64ToIntn (Operand, &Result);
|
|
|
|
ASSERT_EQ (Status, RETURN_SUCCESS);
|
|
|
|
ASSERT_EQ ((-1537977259), Result);
|
|
|
|
|
|
|
|
//
|
|
|
|
// Otherwise should result in an error status
|
|
|
|
//
|
|
|
|
Operand = (0x5babababefefefef);
|
|
|
|
Status = SafeInt64ToIntn (Operand, &Result);
|
|
|
|
ASSERT_EQ (RETURN_BUFFER_TOO_SMALL, Status);
|
|
|
|
|
|
|
|
Operand = (-6605562033422200815);
|
|
|
|
Status = SafeInt64ToIntn (Operand, &Result);
|
|
|
|
ASSERT_EQ (RETURN_BUFFER_TOO_SMALL, Status);
|
|
|
|
}
|
|
|
|
|
2023-08-16 23:15:19 +02:00
|
|
|
TEST (ConversionTestSuite, TestSafeInt64ToUintn) {
|
2022-06-21 04:52:31 +02:00
|
|
|
RETURN_STATUS Status;
|
2023-08-16 23:15:19 +02:00
|
|
|
INT64 Operand;
|
|
|
|
UINTN Result;
|
2022-06-21 04:52:31 +02:00
|
|
|
|
|
|
|
//
|
|
|
|
// If Operand is between 0 and MAX_UINTN inclusive, then it's a cast
|
|
|
|
//
|
|
|
|
Operand = 0xabababab;
|
|
|
|
Result = 0;
|
|
|
|
Status = SafeInt64ToUintn (Operand, &Result);
|
|
|
|
ASSERT_EQ (Status, RETURN_SUCCESS);
|
|
|
|
ASSERT_EQ (0xabababab, Result);
|
|
|
|
|
|
|
|
//
|
|
|
|
// Otherwise should result in an error status
|
|
|
|
//
|
|
|
|
Operand = (0x5babababefefefef);
|
|
|
|
Status = SafeInt64ToUintn (Operand, &Result);
|
|
|
|
ASSERT_EQ (RETURN_BUFFER_TOO_SMALL, Status);
|
|
|
|
|
|
|
|
Operand = (-6605562033422200815);
|
|
|
|
Status = SafeInt64ToUintn (Operand, &Result);
|
|
|
|
ASSERT_EQ (RETURN_BUFFER_TOO_SMALL, Status);
|
|
|
|
}
|
|
|
|
|
2023-08-16 23:15:19 +02:00
|
|
|
TEST (ConversionTestSuite, TestSafeUint64ToIntn) {
|
2022-06-21 04:52:31 +02:00
|
|
|
RETURN_STATUS Status;
|
2023-08-16 23:15:19 +02:00
|
|
|
UINT64 Operand;
|
|
|
|
INTN Result;
|
2022-06-21 04:52:31 +02:00
|
|
|
|
|
|
|
//
|
|
|
|
// If Operand is <= MAX_INTN, then it's a cast
|
|
|
|
//
|
|
|
|
Operand = 0x5bababab;
|
|
|
|
Result = 0;
|
|
|
|
Status = SafeUint64ToIntn (Operand, &Result);
|
|
|
|
ASSERT_EQ (Status, RETURN_SUCCESS);
|
|
|
|
ASSERT_EQ (0x5bababab, Result);
|
|
|
|
|
|
|
|
//
|
|
|
|
// Otherwise should result in an error status
|
|
|
|
//
|
|
|
|
Operand = (0xababababefefefef);
|
|
|
|
Status = SafeUint64ToIntn (Operand, &Result);
|
|
|
|
ASSERT_EQ (RETURN_BUFFER_TOO_SMALL, Status);
|
|
|
|
}
|
|
|
|
|
2023-08-16 23:15:19 +02:00
|
|
|
TEST (ConversionTestSuite, TestSafeUint64ToUintn) {
|
2022-06-21 04:52:31 +02:00
|
|
|
RETURN_STATUS Status;
|
2023-08-16 23:15:19 +02:00
|
|
|
UINT64 Operand;
|
|
|
|
UINTN Result;
|
2022-06-21 04:52:31 +02:00
|
|
|
|
|
|
|
//
|
|
|
|
// If Operand is <= MAX_UINTN, then it's a cast
|
|
|
|
//
|
|
|
|
Operand = 0xabababab;
|
|
|
|
Result = 0;
|
|
|
|
Status = SafeUint64ToUintn (Operand, &Result);
|
|
|
|
ASSERT_EQ (Status, RETURN_SUCCESS);
|
|
|
|
ASSERT_EQ (0xabababab, Result);
|
|
|
|
|
|
|
|
//
|
|
|
|
// Otherwise should result in an error status
|
|
|
|
//
|
|
|
|
Operand = (0xababababefefefef);
|
|
|
|
Status = SafeUint64ToUintn (Operand, &Result);
|
|
|
|
ASSERT_EQ (RETURN_BUFFER_TOO_SMALL, Status);
|
|
|
|
}
|
|
|
|
|
2023-08-16 23:15:19 +02:00
|
|
|
TEST (AdditionSubtractionTestSuite, TestSafeUintnAdd) {
|
2022-06-21 04:52:31 +02:00
|
|
|
RETURN_STATUS Status;
|
2023-08-16 23:15:19 +02:00
|
|
|
UINTN Augend;
|
|
|
|
UINTN Addend;
|
|
|
|
UINTN Result;
|
2022-06-21 04:52:31 +02:00
|
|
|
|
|
|
|
//
|
|
|
|
// If the result of addition doesn't overflow MAX_UINTN, then it's addition
|
|
|
|
//
|
|
|
|
Augend = 0x3a3a3a3a;
|
|
|
|
Addend = 0x3a3a3a3a;
|
|
|
|
Result = 0;
|
|
|
|
Status = SafeUintnAdd (Augend, Addend, &Result);
|
|
|
|
ASSERT_EQ (Status, RETURN_SUCCESS);
|
|
|
|
ASSERT_EQ ((UINTN)0x74747474, Result);
|
|
|
|
|
|
|
|
//
|
|
|
|
// Otherwise should result in an error status
|
|
|
|
//
|
|
|
|
Augend = 0xabababab;
|
|
|
|
Addend = 0xbcbcbcbc;
|
|
|
|
Status = SafeUintnAdd (Augend, Addend, &Result);
|
|
|
|
ASSERT_EQ (RETURN_BUFFER_TOO_SMALL, Status);
|
|
|
|
}
|
|
|
|
|
2023-08-16 23:15:19 +02:00
|
|
|
TEST (AdditionSubtractionTestSuite, TestSafeIntnAdd) {
|
2022-06-21 04:52:31 +02:00
|
|
|
RETURN_STATUS Status;
|
2023-08-16 23:15:19 +02:00
|
|
|
INTN Augend;
|
|
|
|
INTN Addend;
|
|
|
|
INTN Result;
|
2022-06-21 04:52:31 +02:00
|
|
|
|
|
|
|
//
|
|
|
|
// If the result of addition doesn't overflow MAX_INTN
|
|
|
|
// and doesn't underflow MIN_INTN, then it's addition
|
|
|
|
//
|
|
|
|
Augend = 0x3a3a3a3a;
|
|
|
|
Addend = 0x3a3a3a3a;
|
|
|
|
Result = 0;
|
|
|
|
Status = SafeIntnAdd (Augend, Addend, &Result);
|
|
|
|
ASSERT_EQ (Status, RETURN_SUCCESS);
|
|
|
|
ASSERT_EQ (0x74747474, Result);
|
|
|
|
|
|
|
|
Augend = (-976894522);
|
|
|
|
Addend = (-976894522);
|
|
|
|
Status = SafeIntnAdd (Augend, Addend, &Result);
|
|
|
|
ASSERT_EQ (Status, RETURN_SUCCESS);
|
|
|
|
ASSERT_EQ ((-1953789044), Result);
|
|
|
|
|
|
|
|
//
|
|
|
|
// Otherwise should result in an error status
|
|
|
|
//
|
|
|
|
Augend = 0x5a5a5a5a;
|
|
|
|
Addend = 0x5a5a5a5a;
|
|
|
|
Status = SafeIntnAdd (Augend, Addend, &Result);
|
|
|
|
ASSERT_EQ (RETURN_BUFFER_TOO_SMALL, Status);
|
|
|
|
|
|
|
|
Augend = (-1515870810);
|
|
|
|
Addend = (-1515870810);
|
|
|
|
Status = SafeIntnAdd (Augend, Addend, &Result);
|
|
|
|
ASSERT_EQ (RETURN_BUFFER_TOO_SMALL, Status);
|
|
|
|
}
|
|
|
|
|
2023-08-16 23:15:19 +02:00
|
|
|
TEST (AdditionSubtractionTestSuite, TestSafeUintnSub) {
|
2022-06-21 04:52:31 +02:00
|
|
|
RETURN_STATUS Status;
|
2023-08-16 23:15:19 +02:00
|
|
|
UINTN Minuend;
|
|
|
|
UINTN Subtrahend;
|
|
|
|
UINTN Result;
|
2022-06-21 04:52:31 +02:00
|
|
|
|
|
|
|
//
|
|
|
|
// If Minuend >= Subtrahend, then it's subtraction
|
|
|
|
//
|
|
|
|
Minuend = 0x5a5a5a5a;
|
|
|
|
Subtrahend = 0x3b3b3b3b;
|
|
|
|
Result = 0;
|
|
|
|
Status = SafeUintnSub (Minuend, Subtrahend, &Result);
|
|
|
|
ASSERT_EQ (Status, RETURN_SUCCESS);
|
|
|
|
ASSERT_EQ ((UINTN)0x1f1f1f1f, Result);
|
|
|
|
|
|
|
|
//
|
|
|
|
// Otherwise should result in an error status
|
|
|
|
//
|
|
|
|
Minuend = 0x5a5a5a5a;
|
|
|
|
Subtrahend = 0x6d6d6d6d;
|
|
|
|
Status = SafeUintnSub (Minuend, Subtrahend, &Result);
|
|
|
|
ASSERT_EQ (RETURN_BUFFER_TOO_SMALL, Status);
|
|
|
|
}
|
|
|
|
|
2023-08-16 23:15:19 +02:00
|
|
|
TEST (AdditionSubtractionTestSuite, TestSafeIntnSub) {
|
2022-06-21 04:52:31 +02:00
|
|
|
RETURN_STATUS Status;
|
2023-08-16 23:15:19 +02:00
|
|
|
INTN Minuend;
|
|
|
|
INTN Subtrahend;
|
|
|
|
INTN Result;
|
2022-06-21 04:52:31 +02:00
|
|
|
|
|
|
|
//
|
|
|
|
// If the result of subtractions doesn't overflow MAX_INTN or
|
|
|
|
// underflow MIN_INTN, then it's subtraction
|
|
|
|
//
|
|
|
|
Minuend = 0x5a5a5a5a;
|
|
|
|
Subtrahend = 0x3a3a3a3a;
|
|
|
|
Result = 0;
|
|
|
|
Status = SafeIntnSub (Minuend, Subtrahend, &Result);
|
|
|
|
ASSERT_EQ (Status, RETURN_SUCCESS);
|
|
|
|
ASSERT_EQ (0x20202020, Result);
|
|
|
|
|
|
|
|
Minuend = 0x3a3a3a3a;
|
|
|
|
Subtrahend = 0x5a5a5a5a;
|
|
|
|
Status = SafeIntnSub (Minuend, Subtrahend, &Result);
|
|
|
|
ASSERT_EQ (Status, RETURN_SUCCESS);
|
|
|
|
ASSERT_EQ ((-538976288), Result);
|
|
|
|
|
|
|
|
//
|
|
|
|
// Otherwise should result in an error status
|
|
|
|
//
|
|
|
|
Minuend = (-2054847098);
|
|
|
|
Subtrahend = 2054847098;
|
|
|
|
Status = SafeIntnSub (Minuend, Subtrahend, &Result);
|
|
|
|
ASSERT_EQ (RETURN_BUFFER_TOO_SMALL, Status);
|
|
|
|
|
|
|
|
Minuend = (2054847098);
|
|
|
|
Subtrahend = (-2054847098);
|
|
|
|
Status = SafeIntnSub (Minuend, Subtrahend, &Result);
|
|
|
|
ASSERT_EQ (RETURN_BUFFER_TOO_SMALL, Status);
|
|
|
|
}
|
|
|
|
|
2023-08-16 23:15:19 +02:00
|
|
|
TEST (MultiplicationTestSuite, TestSafeUintnMult) {
|
2022-06-21 04:52:31 +02:00
|
|
|
RETURN_STATUS Status;
|
2023-08-16 23:15:19 +02:00
|
|
|
UINTN Multiplicand;
|
|
|
|
UINTN Multiplier;
|
|
|
|
UINTN Result;
|
2022-06-21 04:52:31 +02:00
|
|
|
|
|
|
|
//
|
|
|
|
// If the result of multiplication doesn't overflow MAX_UINTN, it will succeed
|
|
|
|
//
|
|
|
|
Multiplicand = 0xa122a;
|
|
|
|
Multiplier = 0xd23;
|
|
|
|
Result = 0;
|
|
|
|
Status = SafeUintnMult (Multiplicand, Multiplier, &Result);
|
|
|
|
ASSERT_EQ (Status, RETURN_SUCCESS);
|
|
|
|
ASSERT_EQ (0x844c9dbe, Result);
|
|
|
|
|
|
|
|
//
|
|
|
|
// Otherwise should result in an error status
|
|
|
|
//
|
|
|
|
Multiplicand = 0xa122a;
|
|
|
|
Multiplier = 0xed23;
|
|
|
|
Status = SafeUintnMult (Multiplicand, Multiplier, &Result);
|
|
|
|
ASSERT_EQ (RETURN_BUFFER_TOO_SMALL, Status);
|
|
|
|
}
|
|
|
|
|
2023-08-16 23:15:19 +02:00
|
|
|
TEST (MultiplicationTestSuite, TestSafeIntnMult) {
|
2022-06-21 04:52:31 +02:00
|
|
|
RETURN_STATUS Status;
|
2023-08-16 23:15:19 +02:00
|
|
|
INTN Multiplicand;
|
|
|
|
INTN Multiplier;
|
|
|
|
INTN Result;
|
2022-06-21 04:52:31 +02:00
|
|
|
|
|
|
|
//
|
|
|
|
// If the result of multiplication doesn't overflow MAX_INTN and doesn't
|
|
|
|
// underflow MIN_UINTN, it will succeed
|
|
|
|
//
|
|
|
|
Multiplicand = 0x123456;
|
|
|
|
Multiplier = 0x678;
|
|
|
|
Result = 0;
|
|
|
|
Status = SafeIntnMult (Multiplicand, Multiplier, &Result);
|
|
|
|
ASSERT_EQ (Status, RETURN_SUCCESS);
|
|
|
|
ASSERT_EQ (0x75c28c50, Result);
|
|
|
|
|
|
|
|
//
|
|
|
|
// Otherwise should result in an error status
|
|
|
|
//
|
|
|
|
Multiplicand = 0x123456;
|
|
|
|
Multiplier = 0xabc;
|
|
|
|
Status = SafeIntnMult (Multiplicand, Multiplier, &Result);
|
|
|
|
ASSERT_EQ (RETURN_BUFFER_TOO_SMALL, Status);
|
|
|
|
}
|