| /** @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> | |
| } | |
| 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; | |
| // | |
| // 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); | |
| } | |
| TEST (ConversionTestSuite, TestSafeIntnToInt32) { | |
| RETURN_STATUS Status; | |
| INTN Operand; | |
| INT32 Result; | |
| // | |
| // 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); | |
| } | |
| TEST (ConversionTestSuite, TestSafeIntnToUint32) { | |
| RETURN_STATUS Status; | |
| INTN Operand; | |
| UINT32 Result; | |
| // | |
| // 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); | |
| } | |
| TEST (ConversionTestSuite, TestSafeUintnToUint32) { | |
| RETURN_STATUS Status; | |
| UINTN Operand; | |
| UINT32 Result; | |
| // | |
| // 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); | |
| } | |
| TEST (ConversionTestSuite, TestSafeUintnToIntn) { | |
| RETURN_STATUS Status; | |
| UINTN Operand; | |
| INTN Result; | |
| // | |
| // 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); | |
| } | |
| TEST (ConversionTestSuite, TestSafeUintnToInt64) { | |
| RETURN_STATUS Status; | |
| UINTN Operand; | |
| INT64 Result; | |
| // | |
| // 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); | |
| } | |
| TEST (ConversionTestSuite, TestSafeInt64ToIntn) { | |
| RETURN_STATUS Status; | |
| INT64 Operand; | |
| INTN Result; | |
| // | |
| // 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); | |
| } | |
| TEST (ConversionTestSuite, TestSafeInt64ToUintn) { | |
| RETURN_STATUS Status; | |
| INT64 Operand; | |
| UINTN Result; | |
| // | |
| // 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); | |
| } | |
| TEST (ConversionTestSuite, TestSafeUint64ToIntn) { | |
| RETURN_STATUS Status; | |
| UINT64 Operand; | |
| INTN Result; | |
| // | |
| // 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); | |
| } | |
| TEST (ConversionTestSuite, TestSafeUint64ToUintn) { | |
| RETURN_STATUS Status; | |
| UINT64 Operand; | |
| UINTN Result; | |
| // | |
| // 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); | |
| } | |
| 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 = 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); | |
| } | |
| 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 = 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); | |
| } | |
| TEST (AdditionSubtractionTestSuite, TestSafeUintnSub) { | |
| RETURN_STATUS Status; | |
| UINTN Minuend; | |
| UINTN Subtrahend; | |
| UINTN Result; | |
| // | |
| // 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); | |
| } | |
| 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 = 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); | |
| } | |
| 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 = 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); | |
| } | |
| 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 = 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); | |
| } |