/** @file | |
x64-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> | |
} | |
TEST(ConversionTestSuite, TestSafeInt32ToUintn) { | |
RETURN_STATUS Status; | |
INT32 Operand; | |
UINTN Result; | |
// | |
// 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); | |
} | |
TEST(ConversionTestSuite, TestSafeUint32ToIntn) { | |
RETURN_STATUS Status; | |
UINT32 Operand; | |
INTN Result; | |
// | |
// For x64, INTN is same as INT64 which is a superset of INT32 | |
// This is just a cast then, and it'll never fail | |
// | |
// | |
// If Operand is non-negative, then it's a cast | |
// | |
Operand = 0xabababab; | |
Result = 0; | |
Status = SafeUint32ToIntn (Operand, &Result); | |
ASSERT_EQ (Status, RETURN_SUCCESS); | |
ASSERT_EQ (0xabababab, Result); | |
} | |
TEST(ConversionTestSuite, TestSafeIntnToInt32) { | |
RETURN_STATUS Status; | |
INTN Operand; | |
INT32 Result; | |
// | |
// If Operand is between MIN_INT32 and MAX_INT32 inclusive, then it's a cast | |
// | |
Operand = 0x5bababab; | |
Result = 0; | |
Status = SafeIntnToInt32 (Operand, &Result); | |
ASSERT_EQ (Status, RETURN_SUCCESS); | |
ASSERT_EQ (0x5bababab, Result); | |
Operand = (-1537977259); | |
Status = SafeIntnToInt32 (Operand, &Result); | |
ASSERT_EQ (Status, RETURN_SUCCESS); | |
ASSERT_EQ ((-1537977259), Result); | |
// | |
// Otherwise should result in an error status | |
// | |
Operand = (0x5babababefefefef); | |
Status = SafeIntnToInt32 (Operand, &Result); | |
ASSERT_EQ (RETURN_BUFFER_TOO_SMALL, Status); | |
Operand = (-6605562033422200815); | |
Status = SafeIntnToInt32 (Operand, &Result); | |
ASSERT_EQ (RETURN_BUFFER_TOO_SMALL, Status); | |
} | |
TEST(ConversionTestSuite, TestSafeIntnToUint32) { | |
RETURN_STATUS Status; | |
INTN Operand; | |
UINT32 Result; | |
// | |
// If Operand is between 0 and MAX_UINT32 inclusive, then it's a cast | |
// | |
Operand = 0xabababab; | |
Result = 0; | |
Status = SafeIntnToUint32 (Operand, &Result); | |
ASSERT_EQ (Status, RETURN_SUCCESS); | |
ASSERT_EQ (0xabababab, Result); | |
// | |
// Otherwise should result in an error status | |
// | |
Operand = (0x5babababefefefef); | |
Status = SafeIntnToUint32 (Operand, &Result); | |
ASSERT_EQ (RETURN_BUFFER_TOO_SMALL, Status); | |
Operand = (-6605562033422200815); | |
Status = SafeIntnToUint32 (Operand, &Result); | |
ASSERT_EQ (RETURN_BUFFER_TOO_SMALL, Status); | |
} | |
TEST(ConversionTestSuite, TestSafeUintnToUint32) { | |
RETURN_STATUS Status; | |
UINTN Operand; | |
UINT32 Result; | |
// | |
// If Operand is <= MAX_UINT32, then it's a cast | |
// | |
Operand = 0xabababab; | |
Result = 0; | |
Status = SafeUintnToUint32 (Operand, &Result); | |
ASSERT_EQ (Status, RETURN_SUCCESS); | |
ASSERT_EQ (0xabababab, Result); | |
// | |
// Otherwise should result in an error status | |
// | |
Operand = (0xababababefefefef); | |
Status = SafeUintnToUint32 (Operand, &Result); | |
ASSERT_EQ (RETURN_BUFFER_TOO_SMALL, Status); | |
} | |
TEST(ConversionTestSuite, TestSafeUintnToIntn) { | |
RETURN_STATUS Status; | |
UINTN Operand; | |
INTN Result; | |
// | |
// If Operand is <= MAX_INTN (0x7fff_ffff_ffff_ffff), then it's a cast | |
// | |
Operand = 0x5babababefefefef; | |
Result = 0; | |
Status = SafeUintnToIntn (Operand, &Result); | |
ASSERT_EQ (Status, RETURN_SUCCESS); | |
ASSERT_EQ (0x5babababefefefef, Result); | |
// | |
// Otherwise should result in an error status | |
// | |
Operand = (0xababababefefefef); | |
Status = SafeUintnToIntn (Operand, &Result); | |
ASSERT_EQ (RETURN_BUFFER_TOO_SMALL, Status); | |
} | |
TEST(ConversionTestSuite, TestSafeUintnToInt64) { | |
RETURN_STATUS Status; | |
UINTN Operand; | |
INT64 Result; | |
// | |
// If Operand is <= MAX_INT64, then it's a cast | |
// | |
Operand = 0x5babababefefefef; | |
Result = 0; | |
Status = SafeUintnToInt64 (Operand, &Result); | |
ASSERT_EQ (Status, RETURN_SUCCESS); | |
ASSERT_EQ (0x5babababefefefef, Result); | |
// | |
// Otherwise should result in an error status | |
// | |
Operand = (0xababababefefefef); | |
Status = SafeUintnToInt64 (Operand, &Result); | |
ASSERT_EQ (RETURN_BUFFER_TOO_SMALL, Status); | |
} | |
TEST(ConversionTestSuite, TestSafeInt64ToIntn) { | |
RETURN_STATUS Status; | |
INT64 Operand; | |
INTN Result; | |
// | |
// INTN is same as INT64 in x64, so this is just a cast | |
// | |
Operand = 0x5babababefefefef; | |
Result = 0; | |
Status = SafeInt64ToIntn (Operand, &Result); | |
ASSERT_EQ (Status, RETURN_SUCCESS); | |
ASSERT_EQ (0x5babababefefefef, Result); | |
} | |
TEST(ConversionTestSuite, TestSafeInt64ToUintn) { | |
RETURN_STATUS Status; | |
INT64 Operand; | |
UINTN Result; | |
// | |
// If Operand is non-negative, then it's a cast | |
// | |
Operand = 0x5babababefefefef; | |
Result = 0; | |
Status = SafeInt64ToUintn (Operand, &Result); | |
ASSERT_EQ (Status, RETURN_SUCCESS); | |
ASSERT_EQ ((UINTN)0x5babababefefefef, Result); | |
// | |
// Otherwise should result in an error status | |
// | |
Operand = (-6605562033422200815); | |
Status = SafeInt64ToUintn (Operand, &Result); | |
ASSERT_EQ (RETURN_BUFFER_TOO_SMALL, Status); | |
} | |
TEST(ConversionTestSuite, TestSafeUint64ToIntn) { | |
RETURN_STATUS Status; | |
UINT64 Operand; | |
INTN Result; | |
// | |
// If Operand is <= MAX_INTN (0x7fff_ffff_ffff_ffff), then it's a cast | |
// | |
Operand = 0x5babababefefefef; | |
Result = 0; | |
Status = SafeUint64ToIntn (Operand, &Result); | |
ASSERT_EQ (Status, RETURN_SUCCESS); | |
ASSERT_EQ (0x5babababefefefef, Result); | |
// | |
// Otherwise should result in an error status | |
// | |
Operand = (0xababababefefefef); | |
Status = SafeUint64ToIntn (Operand, &Result); | |
ASSERT_EQ (RETURN_BUFFER_TOO_SMALL, Status); | |
} | |
TEST(ConversionTestSuite, TestSafeUint64ToUintn) { | |
RETURN_STATUS Status; | |
UINT64 Operand; | |
UINTN Result; | |
// | |
// UINTN is same as UINT64 in x64, so this is just a cast | |
// | |
Operand = 0xababababefefefef; | |
Result = 0; | |
Status = SafeUint64ToUintn (Operand, &Result); | |
ASSERT_EQ (Status, RETURN_SUCCESS); | |
ASSERT_EQ (0xababababefefefef, Result); | |
} | |
TEST(AdditionSubtractionTestSuite, TestSafeUintnAdd) { | |
RETURN_STATUS Status; | |
UINTN Augend; | |
UINTN Addend; | |
UINTN Result; | |
// | |
// If the result of addition doesn't overflow MAX_UINTN, then it's addition | |
// | |
Augend = 0x3a3a3a3a12121212; | |
Addend = 0x3a3a3a3a12121212; | |
Result = 0; | |
Status = SafeUintnAdd (Augend, Addend, &Result); | |
ASSERT_EQ (Status, RETURN_SUCCESS); | |
ASSERT_EQ ((UINTN)0x7474747424242424, Result); | |
// | |
// Otherwise should result in an error status | |
// | |
Augend = 0xababababefefefef; | |
Addend = 0xbcbcbcbcdededede; | |
Status = SafeUintnAdd (Augend, Addend, &Result); | |
ASSERT_EQ (RETURN_BUFFER_TOO_SMALL, Status); | |
} | |
TEST(AdditionSubtractionTestSuite, TestSafeIntnAdd) { | |
RETURN_STATUS Status; | |
INTN Augend; | |
INTN Addend; | |
INTN Result; | |
// | |
// If the result of addition doesn't overflow MAX_INTN | |
// and doesn't underflow MIN_INTN, then it's addition | |
// | |
Augend = 0x3a3a3a3a3a3a3a3a; | |
Addend = 0x3a3a3a3a3a3a3a3a; | |
Result = 0; | |
Status = SafeIntnAdd (Augend, Addend, &Result); | |
ASSERT_EQ (Status, RETURN_SUCCESS); | |
ASSERT_EQ (0x7474747474747474, Result); | |
Augend = (-4195730024608447034); | |
Addend = (-4195730024608447034); | |
Status = SafeIntnAdd (Augend, Addend, &Result); | |
ASSERT_EQ (Status, RETURN_SUCCESS); | |
ASSERT_EQ ((-8391460049216894068), Result); | |
// | |
// Otherwise should result in an error status | |
// | |
Augend = 0x5a5a5a5a5a5a5a5a; | |
Addend = 0x5a5a5a5a5a5a5a5a; | |
Status = SafeIntnAdd (Augend, Addend, &Result); | |
ASSERT_EQ (RETURN_BUFFER_TOO_SMALL, Status); | |
Augend = (-6510615555426900570); | |
Addend = (-6510615555426900570); | |
Status = SafeIntnAdd (Augend, Addend, &Result); | |
ASSERT_EQ (RETURN_BUFFER_TOO_SMALL, Status); | |
} | |
TEST(AdditionSubtractionTestSuite, TestSafeUintnSub) { | |
RETURN_STATUS Status; | |
UINTN Minuend; | |
UINTN Subtrahend; | |
UINTN Result; | |
// | |
// If Minuend >= Subtrahend, then it's subtraction | |
// | |
Minuend = 0x5a5a5a5a5a5a5a5a; | |
Subtrahend = 0x3b3b3b3b3b3b3b3b; | |
Result = 0; | |
Status = SafeUintnSub (Minuend, Subtrahend, &Result); | |
ASSERT_EQ (Status, RETURN_SUCCESS); | |
ASSERT_EQ ((UINTN)0x1f1f1f1f1f1f1f1f, Result); | |
// | |
// Otherwise should result in an error status | |
// | |
Minuend = 0x5a5a5a5a5a5a5a5a; | |
Subtrahend = 0x6d6d6d6d6d6d6d6d; | |
Status = SafeUintnSub (Minuend, Subtrahend, &Result); | |
ASSERT_EQ (RETURN_BUFFER_TOO_SMALL, Status); | |
} | |
TEST(AdditionSubtractionTestSuite, TestSafeIntnSub) { | |
RETURN_STATUS Status; | |
INTN Minuend; | |
INTN Subtrahend; | |
INTN Result; | |
// | |
// If the result of subtractions doesn't overflow MAX_INTN or | |
// underflow MIN_INTN, then it's subtraction | |
// | |
Minuend = 0x5a5a5a5a5a5a5a5a; | |
Subtrahend = 0x3a3a3a3a3a3a3a3a; | |
Result = 0; | |
Status = SafeIntnSub (Minuend, Subtrahend, &Result); | |
ASSERT_EQ (Status, RETURN_SUCCESS); | |
ASSERT_EQ (0x2020202020202020, Result); | |
Minuend = 0x3a3a3a3a3a3a3a3a; | |
Subtrahend = 0x5a5a5a5a5a5a5a5a; | |
Status = SafeIntnSub (Minuend, Subtrahend, &Result); | |
ASSERT_EQ (Status, RETURN_SUCCESS); | |
ASSERT_EQ ((-2314885530818453536), Result); | |
// | |
// Otherwise should result in an error status | |
// | |
Minuend = (-8825501086245354106); | |
Subtrahend = 8825501086245354106; | |
Status = SafeIntnSub (Minuend, Subtrahend, &Result); | |
ASSERT_EQ (RETURN_BUFFER_TOO_SMALL, Status); | |
Minuend = (8825501086245354106); | |
Subtrahend = (-8825501086245354106); | |
Status = SafeIntnSub (Minuend, Subtrahend, &Result); | |
ASSERT_EQ (RETURN_BUFFER_TOO_SMALL, Status); | |
} | |
TEST(MultiplicationTestSuite, TestSafeUintnMult) { | |
RETURN_STATUS Status; | |
UINTN Multiplicand; | |
UINTN Multiplier; | |
UINTN Result; | |
// | |
// If the result of multiplication doesn't overflow MAX_UINTN, it will succeed | |
// | |
Multiplicand = 0x123456789a; | |
Multiplier = 0x1234567; | |
Result = 0; | |
Status = SafeUintnMult (Multiplicand, Multiplier, &Result); | |
ASSERT_EQ (Status, RETURN_SUCCESS); | |
ASSERT_EQ ((UINTN)0x14b66db9745a07f6, Result); | |
// | |
// Otherwise should result in an error status | |
// | |
Multiplicand = 0x123456789a; | |
Multiplier = 0x12345678; | |
Status = SafeUintnMult (Multiplicand, Multiplier, &Result); | |
ASSERT_EQ (RETURN_BUFFER_TOO_SMALL, Status); | |
} | |
TEST(MultiplicationTestSuite, TestSafeIntnMult) { | |
RETURN_STATUS Status; | |
INTN Multiplicand; | |
INTN Multiplier; | |
INTN Result; | |
// | |
// If the result of multiplication doesn't overflow MAX_INTN and doesn't | |
// underflow MIN_UINTN, it will succeed | |
// | |
Multiplicand = 0x123456789; | |
Multiplier = 0x6789abcd; | |
Result = 0; | |
Status = SafeIntnMult (Multiplicand, Multiplier, &Result); | |
ASSERT_EQ (Status, RETURN_SUCCESS); | |
ASSERT_EQ (0x75cd9045220d6bb5, Result); | |
// | |
// Otherwise should result in an error status | |
// | |
Multiplicand = 0x123456789; | |
Multiplier = 0xa789abcd; | |
Status = SafeIntnMult (Multiplicand, Multiplier, &Result); | |
ASSERT_EQ (RETURN_BUFFER_TOO_SMALL, Status); | |
} |