mirror of
				https://github.com/dolphin-emu/dolphin.git
				synced 2025-10-24 17:09:06 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			894 lines
		
	
	
		
			No EOL
		
	
	
		
			37 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			894 lines
		
	
	
		
			No EOL
		
	
	
		
			37 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| 
 | |
| #include <time.h> // TODO: https://github.com/microsoft/wil/issues/44
 | |
| #include <wil/winrt.h>
 | |
| 
 | |
| #ifdef WIL_ENABLE_EXCEPTIONS
 | |
| #include <map>
 | |
| #include <string>
 | |
| #endif
 | |
| 
 | |
| // Required for pinterface template specializations that we depend on in this test
 | |
| #include <Windows.ApplicationModel.Chat.h>
 | |
| #pragma push_macro("GetCurrentTime")
 | |
| #undef GetCurrentTime
 | |
| #include <Windows.UI.Xaml.Data.h>
 | |
| #pragma pop_macro("GetCurrentTime")
 | |
| 
 | |
| #include "common.h"
 | |
| #include "FakeWinRTTypes.h"
 | |
| #include "test_objects.h"
 | |
| 
 | |
| using namespace ABI::Windows::Foundation;
 | |
| using namespace ABI::Windows::Foundation::Collections;
 | |
| using namespace ABI::Windows::Storage;
 | |
| using namespace ABI::Windows::System;
 | |
| using namespace Microsoft::WRL;
 | |
| using namespace Microsoft::WRL::Wrappers;
 | |
| 
 | |
| TEST_CASE("WinRTTests::VerifyTraitsTypes", "[winrt]")
 | |
| {
 | |
|     static_assert(wistd::is_same_v<bool, typename wil::details::LastType<int, bool>::type>, "");
 | |
|     static_assert(wistd::is_same_v<int, typename wil::details::LastType<int>::type>, "");
 | |
| 
 | |
|     static_assert(wistd::is_same_v<IAsyncAction*, decltype(wil::details::GetReturnParamPointerType(&IFileIOStatics::WriteTextAsync))>, "");
 | |
|     static_assert(wistd::is_same_v<IAsyncOperation<bool>*, decltype(wil::details::GetReturnParamPointerType(&ILauncherStatics::LaunchUriAsync))>, "");
 | |
| 
 | |
|     static_assert(wistd::is_same_v<void, decltype(wil::details::GetAsyncResultType(static_cast<IAsyncAction*>(nullptr)))>, "");
 | |
|     static_assert(wistd::is_same_v<boolean, decltype(wil::details::GetAsyncResultType(static_cast<IAsyncOperation<bool>*>(nullptr)))>, "");
 | |
|     static_assert(wistd::is_same_v<IStorageFile*, decltype(wil::details::GetAsyncResultType(static_cast<IAsyncOperation<StorageFile*>*>(nullptr)))>, "");
 | |
| }
 | |
| 
 | |
| template <bool InhibitArrayReferences, bool IgnoreCase, typename LhsT, typename RhsT>
 | |
| void DoHStringComparisonTest(LhsT&& lhs, RhsT&& rhs, int relation)
 | |
| {
 | |
|     using compare = wil::details::hstring_compare<InhibitArrayReferences, IgnoreCase>;
 | |
| 
 | |
|     // == and !=
 | |
|     REQUIRE(compare::equals(lhs, rhs) == (relation == 0));
 | |
|     REQUIRE(compare::not_equals(lhs, rhs) == (relation != 0));
 | |
| 
 | |
|     REQUIRE(compare::equals(rhs, lhs) == (relation == 0));
 | |
|     REQUIRE(compare::not_equals(rhs, lhs) == (relation != 0));
 | |
| 
 | |
|     // < and >=
 | |
|     REQUIRE(compare::less(lhs, rhs) == (relation < 0));
 | |
|     REQUIRE(compare::greater_equals(lhs, rhs) == (relation >= 0));
 | |
| 
 | |
|     REQUIRE(compare::less(rhs, lhs) == (relation > 0));
 | |
|     REQUIRE(compare::greater_equals(rhs, lhs) == (relation <= 0));
 | |
| 
 | |
|     // > and <=
 | |
|     REQUIRE(compare::greater(lhs, rhs) == (relation > 0));
 | |
|     REQUIRE(compare::less_equals(lhs, rhs) == (relation <= 0));
 | |
| 
 | |
|     REQUIRE(compare::greater(rhs, lhs) == (relation < 0));
 | |
|     REQUIRE(compare::less_equals(rhs, lhs) == (relation >= 0));
 | |
| 
 | |
|     // We wish to test with both const and non-const values. We can do this for free here so long as the type is
 | |
|     // not an array since changing the const-ness of an array may change the expected results
 | |
| #pragma warning(suppress: 4127)
 | |
|     if (!wistd::is_array<wistd::remove_reference_t<LhsT>>::value &&
 | |
|         !wistd::is_const<wistd::remove_reference_t<LhsT>>::value)
 | |
|     {
 | |
|         const wistd::remove_reference_t<LhsT>& constLhs = lhs;
 | |
|         DoHStringComparisonTest<InhibitArrayReferences, IgnoreCase>(constLhs, rhs, relation);
 | |
|     }
 | |
| 
 | |
| #pragma warning(suppress: 4127)
 | |
|     if (!wistd::is_array<wistd::remove_reference_t<RhsT>>::value &&
 | |
|         !wistd::is_const<wistd::remove_reference_t<RhsT>>::value)
 | |
|     {
 | |
|         const wistd::remove_reference_t<RhsT>& constRhs = rhs;
 | |
|         DoHStringComparisonTest<InhibitArrayReferences, IgnoreCase>(lhs, constRhs, relation);
 | |
|     }
 | |
| }
 | |
| 
 | |
| // The two string arguments are expected to compare equal to one another using the specified IgnoreCase argument and
 | |
| // contain at least one embedded null character
 | |
| template <bool InhibitArrayReferences, bool IgnoreCase, size_t Size>
 | |
| void DoHStringSameValueComparisonTest(const wchar_t (&lhs)[Size], const wchar_t (&rhs)[Size])
 | |
| {
 | |
|     wchar_t lhsNonConstArray[Size + 5];
 | |
|     wchar_t rhsNonConstArray[Size + 5];
 | |
|     wcsncpy_s(lhsNonConstArray, lhs, Size);
 | |
|     wcsncpy_s(rhsNonConstArray, rhs, Size);
 | |
| 
 | |
|     // For non-const arrays, we should never deduce length, so even though we append different values to each string, we
 | |
|     // do so after the last null character, so they should never be read
 | |
|     wcsncpy_s(lhsNonConstArray + Size + 1, 4, L"foo", 3);
 | |
|     wcsncpy_s(rhsNonConstArray + Size + 1, 4, L"bar", 3);
 | |
| 
 | |
|     const wchar_t* lhsCstr = lhs;
 | |
|     const wchar_t* rhsCstr = rhs;
 | |
| 
 | |
|     HStringReference lhsRef(lhs);
 | |
|     HStringReference rhsRef(rhs);
 | |
|     HString lhsStr;
 | |
|     HString rhsStr;
 | |
|     REQUIRE_SUCCEEDED(lhsStr.Set(lhs));
 | |
|     REQUIRE_SUCCEEDED(rhsStr.Set(rhs));
 | |
|     auto lhsHstr = lhsStr.Get();
 | |
|     auto rhsHstr = rhsStr.Get();
 | |
| 
 | |
|     wil::unique_hstring lhsUniqueStr;
 | |
|     wil::unique_hstring rhsUniqueStr;
 | |
|     REQUIRE_SUCCEEDED(lhsStr.CopyTo(&lhsUniqueStr));
 | |
|     REQUIRE_SUCCEEDED(rhsStr.CopyTo(&rhsUniqueStr));
 | |
| 
 | |
|     // Const array - embedded nulls are included only if InhibitArrayReferences is false
 | |
|     DoHStringComparisonTest<InhibitArrayReferences, IgnoreCase>(lhs, rhs, 0);
 | |
|     DoHStringComparisonTest<InhibitArrayReferences, IgnoreCase>(lhs, rhsNonConstArray, InhibitArrayReferences ? 0 : 1);
 | |
|     DoHStringComparisonTest<InhibitArrayReferences, IgnoreCase>(lhs, rhsCstr, InhibitArrayReferences ? 0 : 1);
 | |
|     DoHStringComparisonTest<InhibitArrayReferences, IgnoreCase>(lhs, rhsRef, InhibitArrayReferences ? -1 : 0);
 | |
|     DoHStringComparisonTest<InhibitArrayReferences, IgnoreCase>(lhs, rhsStr, InhibitArrayReferences ? -1 : 0);
 | |
|     DoHStringComparisonTest<InhibitArrayReferences, IgnoreCase>(lhs, rhsHstr, InhibitArrayReferences ? -1 : 0);
 | |
|     DoHStringComparisonTest<InhibitArrayReferences, IgnoreCase>(lhs, rhsUniqueStr, InhibitArrayReferences ? -1 : 0);
 | |
| 
 | |
|     // Non-const array - *never* deduce length
 | |
|     DoHStringComparisonTest<InhibitArrayReferences, IgnoreCase>(lhsNonConstArray, rhsNonConstArray, 0);
 | |
|     DoHStringComparisonTest<InhibitArrayReferences, IgnoreCase>(lhsNonConstArray, rhsCstr, 0);
 | |
|     DoHStringComparisonTest<InhibitArrayReferences, IgnoreCase>(lhsNonConstArray, rhsRef, -1);
 | |
|     DoHStringComparisonTest<InhibitArrayReferences, IgnoreCase>(lhsNonConstArray, rhsStr, -1);
 | |
|     DoHStringComparisonTest<InhibitArrayReferences, IgnoreCase>(lhsNonConstArray, rhsHstr, -1);
 | |
|     DoHStringComparisonTest<InhibitArrayReferences, IgnoreCase>(lhsNonConstArray, rhsUniqueStr, -1);
 | |
| 
 | |
|     // C string - impossible to deduce length
 | |
|     DoHStringComparisonTest<InhibitArrayReferences, IgnoreCase>(lhsCstr, rhsCstr, 0);
 | |
|     DoHStringComparisonTest<InhibitArrayReferences, IgnoreCase>(lhsCstr, rhsRef, -1);
 | |
|     DoHStringComparisonTest<InhibitArrayReferences, IgnoreCase>(lhsCstr, rhsStr, -1);
 | |
|     DoHStringComparisonTest<InhibitArrayReferences, IgnoreCase>(lhsCstr, rhsHstr, -1);
 | |
|     DoHStringComparisonTest<InhibitArrayReferences, IgnoreCase>(lhsCstr, rhsUniqueStr, -1);
 | |
| 
 | |
|     // HStringReference
 | |
|     DoHStringComparisonTest<InhibitArrayReferences, IgnoreCase>(lhsRef, rhsRef, 0);
 | |
|     DoHStringComparisonTest<InhibitArrayReferences, IgnoreCase>(lhsRef, rhsStr, 0);
 | |
|     DoHStringComparisonTest<InhibitArrayReferences, IgnoreCase>(lhsRef, rhsHstr, 0);
 | |
|     DoHStringComparisonTest<InhibitArrayReferences, IgnoreCase>(lhsRef, rhsUniqueStr, 0);
 | |
| 
 | |
|     // HString
 | |
|     DoHStringComparisonTest<InhibitArrayReferences, IgnoreCase>(lhsStr, rhsStr, 0);
 | |
|     DoHStringComparisonTest<InhibitArrayReferences, IgnoreCase>(lhsStr, rhsHstr, 0);
 | |
|     DoHStringComparisonTest<InhibitArrayReferences, IgnoreCase>(lhsStr, rhsUniqueStr, 0);
 | |
| 
 | |
|     // Raw HSTRING
 | |
|     DoHStringComparisonTest<InhibitArrayReferences, IgnoreCase>(lhsHstr, rhsHstr, 0);
 | |
|     DoHStringComparisonTest<InhibitArrayReferences, IgnoreCase>(lhsHstr, rhsUniqueStr, 0);
 | |
| 
 | |
|     // wil::unique_hstring
 | |
|     DoHStringComparisonTest<InhibitArrayReferences, IgnoreCase>(lhsUniqueStr, rhsUniqueStr, 0);
 | |
| 
 | |
| #ifdef WIL_ENABLE_EXCEPTIONS
 | |
|     std::wstring lhsWstr(lhs, 7);
 | |
|     std::wstring rhsWstr(rhs, 7);
 | |
|     DoHStringComparisonTest<InhibitArrayReferences, IgnoreCase>(lhsWstr, rhsWstr, 0);
 | |
|     DoHStringComparisonTest<InhibitArrayReferences, IgnoreCase>(lhsWstr, rhs, InhibitArrayReferences ? 1 : 0);
 | |
|     DoHStringComparisonTest<InhibitArrayReferences, IgnoreCase>(lhsWstr, rhsNonConstArray, 1);
 | |
|     DoHStringComparisonTest<InhibitArrayReferences, IgnoreCase>(lhsWstr, rhsCstr, 1);
 | |
|     DoHStringComparisonTest<InhibitArrayReferences, IgnoreCase>(lhsWstr, rhsRef, 0);
 | |
|     DoHStringComparisonTest<InhibitArrayReferences, IgnoreCase>(lhsWstr, rhsStr, 0);
 | |
|     DoHStringComparisonTest<InhibitArrayReferences, IgnoreCase>(lhsWstr, rhsHstr, 0);
 | |
|     DoHStringComparisonTest<InhibitArrayReferences, IgnoreCase>(lhsWstr, rhsUniqueStr, 0);
 | |
| #endif
 | |
| }
 | |
| 
 | |
| // It's expected that the first argument (lhs) compares greater than the second argument (rhs)
 | |
| template <bool InhibitArrayReferences, bool IgnoreCase, size_t LhsSize, size_t RhsSize>
 | |
| void DoHStringDifferentValueComparisonTest(const wchar_t (&lhs)[LhsSize], const wchar_t (&rhs)[RhsSize])
 | |
| {
 | |
|     wchar_t lhsNonConstArray[LhsSize];
 | |
|     wchar_t rhsNonConstArray[RhsSize];
 | |
|     wcsncpy_s(lhsNonConstArray, lhs, LhsSize);
 | |
|     wcsncpy_s(rhsNonConstArray, rhs, RhsSize);
 | |
| 
 | |
|     const wchar_t* lhsCstr = lhs;
 | |
|     const wchar_t* rhsCstr = rhs;
 | |
| 
 | |
|     HStringReference lhsRef(lhs);
 | |
|     HStringReference rhsRef(rhs);
 | |
|     HString lhsStr;
 | |
|     HString rhsStr;
 | |
|     REQUIRE_SUCCEEDED(lhsStr.Set(lhs));
 | |
|     REQUIRE_SUCCEEDED(rhsStr.Set(rhs));
 | |
|     auto lhsHstr = lhsStr.Get();
 | |
|     auto rhsHstr = rhsStr.Get();
 | |
| 
 | |
|     wil::unique_hstring lhsUniqueStr;
 | |
|     wil::unique_hstring rhsUniqueStr;
 | |
|     REQUIRE_SUCCEEDED(lhsStr.CopyTo(&lhsUniqueStr));
 | |
|     REQUIRE_SUCCEEDED(rhsStr.CopyTo(&rhsUniqueStr));
 | |
| 
 | |
|     // Const array
 | |
|     DoHStringComparisonTest<InhibitArrayReferences, IgnoreCase>(lhs, rhs, 1);
 | |
|     DoHStringComparisonTest<InhibitArrayReferences, IgnoreCase>(lhs, rhsNonConstArray, 1);
 | |
|     DoHStringComparisonTest<InhibitArrayReferences, IgnoreCase>(lhs, rhsCstr, 1);
 | |
|     DoHStringComparisonTest<InhibitArrayReferences, IgnoreCase>(lhs, rhsRef, 1);
 | |
|     DoHStringComparisonTest<InhibitArrayReferences, IgnoreCase>(lhs, rhsStr, 1);
 | |
|     DoHStringComparisonTest<InhibitArrayReferences, IgnoreCase>(lhs, rhsHstr, 1);
 | |
|     DoHStringComparisonTest<InhibitArrayReferences, IgnoreCase>(lhs, rhsUniqueStr, 1);
 | |
| 
 | |
|     // Non-const array
 | |
|     DoHStringComparisonTest<InhibitArrayReferences, IgnoreCase>(lhsNonConstArray, rhsNonConstArray, 1);
 | |
|     DoHStringComparisonTest<InhibitArrayReferences, IgnoreCase>(lhsNonConstArray, rhsCstr, 1);
 | |
|     DoHStringComparisonTest<InhibitArrayReferences, IgnoreCase>(lhsNonConstArray, rhsRef, 1);
 | |
|     DoHStringComparisonTest<InhibitArrayReferences, IgnoreCase>(lhsNonConstArray, rhsStr, 1);
 | |
|     DoHStringComparisonTest<InhibitArrayReferences, IgnoreCase>(lhsNonConstArray, rhsHstr, 1);
 | |
|     DoHStringComparisonTest<InhibitArrayReferences, IgnoreCase>(lhsNonConstArray, rhsUniqueStr, 1);
 | |
| 
 | |
|     // C string
 | |
|     DoHStringComparisonTest<InhibitArrayReferences, IgnoreCase>(lhsCstr, rhsCstr, 1);
 | |
|     DoHStringComparisonTest<InhibitArrayReferences, IgnoreCase>(lhsCstr, rhsRef, 1);
 | |
|     DoHStringComparisonTest<InhibitArrayReferences, IgnoreCase>(lhsCstr, rhsStr, 1);
 | |
|     DoHStringComparisonTest<InhibitArrayReferences, IgnoreCase>(lhsCstr, rhsHstr, 1);
 | |
|     DoHStringComparisonTest<InhibitArrayReferences, IgnoreCase>(lhsCstr, rhsUniqueStr, 1);
 | |
| 
 | |
|     // HStringReference
 | |
|     DoHStringComparisonTest<InhibitArrayReferences, IgnoreCase>(lhsRef, rhsRef, 1);
 | |
|     DoHStringComparisonTest<InhibitArrayReferences, IgnoreCase>(lhsRef, rhsStr, 1);
 | |
|     DoHStringComparisonTest<InhibitArrayReferences, IgnoreCase>(lhsRef, rhsHstr, 1);
 | |
|     DoHStringComparisonTest<InhibitArrayReferences, IgnoreCase>(lhsRef, rhsUniqueStr, 1);
 | |
| 
 | |
|     // HString
 | |
|     DoHStringComparisonTest<InhibitArrayReferences, IgnoreCase>(lhsStr, rhsStr, 1);
 | |
|     DoHStringComparisonTest<InhibitArrayReferences, IgnoreCase>(lhsStr, rhsHstr, 1);
 | |
|     DoHStringComparisonTest<InhibitArrayReferences, IgnoreCase>(lhsStr, rhsUniqueStr, 1);
 | |
| 
 | |
|     // Raw HSTRING
 | |
|     DoHStringComparisonTest<InhibitArrayReferences, IgnoreCase>(lhsHstr, rhsHstr, 1);
 | |
|     DoHStringComparisonTest<InhibitArrayReferences, IgnoreCase>(lhsHstr, rhsUniqueStr, 1);
 | |
| 
 | |
|     // wil::unique_hstring
 | |
|     DoHStringComparisonTest<InhibitArrayReferences, IgnoreCase>(lhsUniqueStr, rhsUniqueStr, 1);
 | |
| 
 | |
| #ifdef WIL_ENABLE_EXCEPTIONS
 | |
|     std::wstring lhsWstr(lhs, 7);
 | |
|     std::wstring rhsWstr(rhs, 7);
 | |
|     DoHStringComparisonTest<InhibitArrayReferences, IgnoreCase>(lhsWstr, rhsWstr, 1);
 | |
|     DoHStringComparisonTest<InhibitArrayReferences, IgnoreCase>(lhsWstr, rhs, 1);
 | |
|     DoHStringComparisonTest<InhibitArrayReferences, IgnoreCase>(lhsWstr, rhsNonConstArray, 1);
 | |
|     DoHStringComparisonTest<InhibitArrayReferences, IgnoreCase>(lhsWstr, rhsCstr, 1);
 | |
|     DoHStringComparisonTest<InhibitArrayReferences, IgnoreCase>(lhsWstr, rhsRef, 1);
 | |
|     DoHStringComparisonTest<InhibitArrayReferences, IgnoreCase>(lhsWstr, rhsStr, 1);
 | |
|     DoHStringComparisonTest<InhibitArrayReferences, IgnoreCase>(lhsWstr, rhsHstr, 1);
 | |
|     DoHStringComparisonTest<InhibitArrayReferences, IgnoreCase>(lhsWstr, rhsUniqueStr, 1);
 | |
| #endif
 | |
| }
 | |
| 
 | |
| TEST_CASE("WinRTTests::HStringComparison", "[winrt][hstring_compare]")
 | |
| {
 | |
|     SECTION("Don't inhibit arrays")
 | |
|     {
 | |
|         DoHStringSameValueComparisonTest<false, false>(L"foo\0bar", L"foo\0bar");
 | |
|         DoHStringDifferentValueComparisonTest<false, false>(L"foo", L"bar");
 | |
|     }
 | |
| 
 | |
|     SECTION("Inhibit arrays")
 | |
|     {
 | |
|         DoHStringSameValueComparisonTest<true, false>(L"foo\0bar", L"foo\0bar");
 | |
|         DoHStringDifferentValueComparisonTest<true, false>(L"foo", L"bar");
 | |
|     }
 | |
| 
 | |
|     SECTION("Ignore case")
 | |
|     {
 | |
|         DoHStringSameValueComparisonTest<true, true>(L"foo\0bar", L"FoO\0bAR");
 | |
|         DoHStringDifferentValueComparisonTest<true, true>(L"Foo", L"baR");
 | |
|     }
 | |
| 
 | |
|     SECTION("Empty string")
 | |
|     {
 | |
|         const wchar_t constArray[] = L"";
 | |
|         wchar_t nonConstArray[] = L"";
 | |
|         const wchar_t* cstr = constArray;
 | |
|         const wchar_t* nullCstr = nullptr;
 | |
| 
 | |
|         // str may end up referencing a null HSTRING. That's fine; we'll just test null HSTRING twice
 | |
|         HString str;
 | |
|         REQUIRE_SUCCEEDED(str.Set(constArray));
 | |
|         HSTRING nullHstr = nullptr;
 | |
| 
 | |
|         // Const array - impossible to use null value
 | |
|         DoHStringComparisonTest<false, false>(constArray, constArray, 0);
 | |
|         DoHStringComparisonTest<false, false>(constArray, nonConstArray, 0);
 | |
|         DoHStringComparisonTest<false, false>(constArray, cstr, 0);
 | |
|         DoHStringComparisonTest<false, false>(constArray, nullCstr, 0);
 | |
|         DoHStringComparisonTest<false, false>(constArray, str.Get(), 0);
 | |
|         DoHStringComparisonTest<false, false>(constArray, nullHstr, 0);
 | |
| 
 | |
|         // Non-const array - impossible to use null value
 | |
|         DoHStringComparisonTest<false, false>(nonConstArray, nonConstArray, 0);
 | |
|         DoHStringComparisonTest<false, false>(nonConstArray, cstr, 0);
 | |
|         DoHStringComparisonTest<false, false>(nonConstArray, nullCstr, 0);
 | |
|         DoHStringComparisonTest<false, false>(nonConstArray, str.Get(), 0);
 | |
|         DoHStringComparisonTest<false, false>(nonConstArray, nullHstr, 0);
 | |
| 
 | |
|         // Non-null c-string
 | |
|         DoHStringComparisonTest<false, false>(cstr, cstr, 0);
 | |
|         DoHStringComparisonTest<false, false>(cstr, nullCstr, 0);
 | |
|         DoHStringComparisonTest<false, false>(cstr, str.Get(), 0);
 | |
|         DoHStringComparisonTest<false, false>(cstr, nullHstr, 0);
 | |
| 
 | |
|         // Null c-string
 | |
|         DoHStringComparisonTest<false, false>(nullCstr, nullCstr, 0);
 | |
|         DoHStringComparisonTest<false, false>(nullCstr, str.Get(), 0);
 | |
|         DoHStringComparisonTest<false, false>(nullCstr, nullHstr, 0);
 | |
| 
 | |
|         // (Possibly) non-null HSTRING
 | |
|         DoHStringComparisonTest<false, false>(str.Get(), str.Get(), 0);
 | |
|         DoHStringComparisonTest<false, false>(str.Get(), nullHstr, 0);
 | |
| 
 | |
|         // Null HSTRING
 | |
|         DoHStringComparisonTest<false, false>(nullHstr, nullHstr, 0);
 | |
| 
 | |
| #ifdef WIL_ENABLE_EXCEPTIONS
 | |
|         std::wstring wstr;
 | |
|         DoHStringComparisonTest<false, false>(wstr, wstr, 0);
 | |
|         DoHStringComparisonTest<false, false>(wstr, constArray, 0);
 | |
|         DoHStringComparisonTest<false, false>(wstr, nonConstArray, 0);
 | |
|         DoHStringComparisonTest<false, false>(wstr, cstr, 0);
 | |
|         DoHStringComparisonTest<false, false>(wstr, nullCstr, 0);
 | |
|         DoHStringComparisonTest<false, false>(wstr, str.Get(), 0);
 | |
|         DoHStringComparisonTest<false, false>(wstr, nullHstr, 0);
 | |
| #endif
 | |
|     }
 | |
| }
 | |
| 
 | |
| #ifdef WIL_ENABLE_EXCEPTIONS
 | |
| TEST_CASE("WinRTTests::HStringMapTest", "[winrt][hstring_compare]")
 | |
| {
 | |
|     int nextValue = 0;
 | |
|     std::map<std::wstring, int> wstringMap;
 | |
|     wstringMap.emplace(L"foo", nextValue++);
 | |
|     wstringMap.emplace(L"bar", nextValue++);
 | |
|     wstringMap.emplace(std::wstring(L"foo\0bar", 7), nextValue++);
 | |
|     wstringMap.emplace(L"adding", nextValue++);
 | |
|     wstringMap.emplace(L"quite", nextValue++);
 | |
|     wstringMap.emplace(L"a", nextValue++);
 | |
|     wstringMap.emplace(L"few", nextValue++);
 | |
|     wstringMap.emplace(L"more", nextValue++);
 | |
|     wstringMap.emplace(L"values", nextValue++);
 | |
|     wstringMap.emplace(L"for", nextValue++);
 | |
|     wstringMap.emplace(L"testing", nextValue++);
 | |
|     wstringMap.emplace(L"", nextValue++);
 | |
| 
 | |
|     std::map<HString, int> hstringMap;
 | |
|     for (auto& pair : wstringMap)
 | |
|     {
 | |
|         HString str;
 | |
|         THROW_IF_FAILED(str.Set(pair.first.c_str(), static_cast<UINT>(pair.first.length())));
 | |
|         hstringMap.emplace(std::move(str), pair.second);
 | |
|     }
 | |
| 
 | |
|     // Order should be the same as the map of wstring
 | |
|     auto itr = hstringMap.begin();
 | |
|     for (auto& pair : wstringMap)
 | |
|     {
 | |
|         REQUIRE(itr != hstringMap.end());
 | |
|         REQUIRE(itr->first == HStringReference(pair.first.c_str(), static_cast<UINT>(pair.first.length())));
 | |
| 
 | |
|         // Should also be able to find the value
 | |
|         REQUIRE(hstringMap.find(pair.first) != hstringMap.end());
 | |
| 
 | |
|         ++itr;
 | |
|     }
 | |
|     REQUIRE(itr == hstringMap.end());
 | |
| 
 | |
|     const wchar_t constArray[] = L"foo\0bar";
 | |
|     wchar_t nonConstArray[] = L"foo\0bar";
 | |
|     const wchar_t* cstr = constArray;
 | |
| 
 | |
|     HString key;
 | |
|     wil::unique_hstring uniqueHstr;
 | |
|     THROW_IF_FAILED(key.Set(constArray));
 | |
|     THROW_IF_FAILED(key.CopyTo(&uniqueHstr));
 | |
| 
 | |
|     HStringReference ref(constArray);
 | |
|     std::wstring wstr(constArray, 7);
 | |
| 
 | |
|     auto verifyFunc = [&](int expectedValue, auto&& keyValue)
 | |
|     {
 | |
|         auto itr = hstringMap.find(std::forward<decltype(keyValue)>(keyValue));
 | |
|         REQUIRE(itr != hstringMap.end());
 | |
|         REQUIRE(expectedValue == itr->second);
 | |
|     };
 | |
| 
 | |
|     // The following should find "foo\0bar"
 | |
|     auto expectedValue = wstringMap[wstr];
 | |
|     verifyFunc(expectedValue, uniqueHstr);
 | |
|     verifyFunc(expectedValue, key);
 | |
|     verifyFunc(expectedValue, key.Get());
 | |
|     verifyFunc(expectedValue, ref);
 | |
|     verifyFunc(expectedValue, wstr);
 | |
| 
 | |
|     // Arrays/strings should not deduce length and should therefore find "foo"
 | |
|     expectedValue = wstringMap[L"foo"];
 | |
|     verifyFunc(expectedValue, constArray);
 | |
|     verifyFunc(expectedValue, nonConstArray);
 | |
|     verifyFunc(expectedValue, cstr);
 | |
| 
 | |
|     // Should not ignore case
 | |
|     REQUIRE(hstringMap.find(L"FOO") == hstringMap.end());
 | |
| 
 | |
|     // Should also be able to find empty values
 | |
|     const wchar_t constEmptyArray[] = L"";
 | |
|     wchar_t nonConstEmptyArray[] = L"";
 | |
|     const wchar_t* emptyCstr = constEmptyArray;
 | |
|     const wchar_t* nullCstr = nullptr;
 | |
| 
 | |
|     HString emptyStr;
 | |
|     HSTRING nullHstr = nullptr;
 | |
| 
 | |
|     std::wstring emptyWstr;
 | |
| 
 | |
|     expectedValue = wstringMap[L""];
 | |
|     verifyFunc(expectedValue, constEmptyArray);
 | |
|     verifyFunc(expectedValue, nonConstEmptyArray);
 | |
|     verifyFunc(expectedValue, emptyCstr);
 | |
|     verifyFunc(expectedValue, nullCstr);
 | |
|     verifyFunc(expectedValue, emptyStr);
 | |
|     verifyFunc(expectedValue, nullHstr);
 | |
|     verifyFunc(expectedValue, emptyWstr);
 | |
| }
 | |
| 
 | |
| TEST_CASE("WinRTTests::HStringCaseInsensitiveMapTest", "[winrt][hstring_compare]")
 | |
| {
 | |
|     std::map<HString, int, wil::hstring_insensitive_less> hstringMap;
 | |
| 
 | |
|     auto emplaceFunc = [&](auto&& key, int value)
 | |
|     {
 | |
|         HString str;
 | |
|         THROW_IF_FAILED(str.Set(std::forward<decltype(key)>(key)));
 | |
|         hstringMap.emplace(std::move(str), value);
 | |
|     };
 | |
| 
 | |
|     int nextValue = 0;
 | |
|     int fooValue = nextValue++;
 | |
|     emplaceFunc(L"foo", fooValue);
 | |
|     emplaceFunc(L"bar", nextValue++);
 | |
|     int foobarValue = nextValue++;
 | |
|     emplaceFunc(L"foo\0bar", foobarValue);
 | |
|     emplaceFunc(L"foobar", nextValue++);
 | |
|     emplaceFunc(L"adding", nextValue++);
 | |
|     emplaceFunc(L"some", nextValue++);
 | |
|     emplaceFunc(L"more", nextValue++);
 | |
|     emplaceFunc(L"values", nextValue++);
 | |
|     emplaceFunc(L"for", nextValue++);
 | |
|     emplaceFunc(L"testing", nextValue++);
 | |
|     WI_ASSERT(static_cast<size_t>(nextValue) == hstringMap.size());
 | |
| 
 | |
|     const wchar_t constArray[] = L"FoO\0BAr";
 | |
|     wchar_t nonConstArray[] = L"fOo\0baR";
 | |
|     const wchar_t* cstr = constArray;
 | |
| 
 | |
|     HString key;
 | |
|     wil::unique_hstring uniqueHstr;
 | |
|     THROW_IF_FAILED(key.Set(constArray));
 | |
|     THROW_IF_FAILED(key.CopyTo(&uniqueHstr));
 | |
| 
 | |
|     HStringReference ref(constArray);
 | |
|     std::wstring wstr(constArray, 7);
 | |
| 
 | |
|     auto verifyFunc = [&](int expectedValue, auto&& key)
 | |
|     {
 | |
|         auto itr = hstringMap.find(std::forward<decltype(key)>(key));
 | |
|         REQUIRE(itr != std::end(hstringMap));
 | |
|         REQUIRE(expectedValue == itr->second);
 | |
|     };
 | |
| 
 | |
|     // The following should find "foo\0bar"
 | |
|     verifyFunc(foobarValue, uniqueHstr);
 | |
|     verifyFunc(foobarValue, key);
 | |
|     verifyFunc(foobarValue, key.Get());
 | |
|     verifyFunc(foobarValue, ref);
 | |
|     verifyFunc(foobarValue, wstr);
 | |
| 
 | |
|     // Arrays/strings should not deduce length and should therefore find "foo"
 | |
|     verifyFunc(fooValue, constArray);
 | |
|     verifyFunc(fooValue, nonConstArray);
 | |
|     verifyFunc(fooValue, cstr);
 | |
| }
 | |
| #endif
 | |
| 
 | |
| // This is not a test method, nor should it be called. This is a compilation-only test.
 | |
| #ifdef WIL_ENABLE_EXCEPTIONS
 | |
| void RunWhenCompleteCompilationTest()
 | |
| {
 | |
|     {
 | |
|         ComPtr<IAsyncOperation<HSTRING>> stringOp;
 | |
|         wil::run_when_complete(stringOp.Get(), [](HRESULT /* result */, HSTRING /* value */) {});
 | |
| #if WINAPI_FAMILY_PARTITION(WINAPI_PARTITION_DESKTOP | WINAPI_PARTITION_SYSTEM)
 | |
|         auto result = wil::wait_for_completion(stringOp.Get());
 | |
| #endif
 | |
|     }
 | |
| 
 | |
|     {
 | |
|         ComPtr<IAsyncOperationWithProgress<HSTRING, UINT64>> stringOpWithProgress;
 | |
|         wil::run_when_complete(stringOpWithProgress.Get(), [](HRESULT /* result */, HSTRING /* value */) {});
 | |
| #if WINAPI_FAMILY_PARTITION(WINAPI_PARTITION_DESKTOP | WINAPI_PARTITION_SYSTEM)
 | |
|         auto result = wil::wait_for_completion(stringOpWithProgress.Get());
 | |
| #endif
 | |
|     }
 | |
| }
 | |
| #endif
 | |
| 
 | |
| TEST_CASE("WinRTTests::RunWhenCompleteMoveOnlyTest", "[winrt][run_when_complete]")
 | |
| {
 | |
|     auto op = Make<FakeAsyncOperation<int>>();
 | |
|     REQUIRE(op);
 | |
| 
 | |
|     bool gotEvent = false;
 | |
|     auto hr = wil::run_when_complete_nothrow(op.Get(), [&gotEvent, enforce = cannot_copy{}](HRESULT hr, int result)
 | |
|     {
 | |
|         (void)enforce;
 | |
|         REQUIRE_SUCCEEDED(hr);
 | |
|         REQUIRE(result == 42);
 | |
|         gotEvent = true;
 | |
|         return S_OK;
 | |
|     });
 | |
|     REQUIRE_SUCCEEDED(hr);
 | |
| 
 | |
|     op->Complete(S_OK, 42);
 | |
|     REQUIRE(gotEvent);
 | |
| }
 | |
| 
 | |
| #if WINAPI_FAMILY_PARTITION(WINAPI_PARTITION_DESKTOP | WINAPI_PARTITION_SYSTEM)
 | |
| TEST_CASE("WinRTTests::WaitForCompletionTimeout", "[winrt][wait_for_completion]")
 | |
| {
 | |
|     auto op = Make<FakeAsyncOperation<bool, boolean>>();
 | |
|     REQUIRE(op);
 | |
| 
 | |
|     // The wait_for_completion* functions don't properly deduce the "decayed" async type, so force it here
 | |
|     auto asyncOp = static_cast<IAsyncOperation<bool>*>(op.Get());
 | |
| 
 | |
|     bool timedOut = false;
 | |
|     REQUIRE_SUCCEEDED(wil::wait_for_completion_or_timeout_nothrow(asyncOp, 1, &timedOut));
 | |
|     REQUIRE(timedOut);
 | |
| }
 | |
| 
 | |
| // This is not a test method, nor should it be called. This is a compilation-only test.
 | |
| #pragma warning(push)
 | |
| #pragma warning(disable: 4702) // Unreachable code
 | |
| void WaitForCompletionCompilationTest()
 | |
| {
 | |
|     // Ensure the wait_for_completion variants compile
 | |
|     FAIL_FAST_HR_MSG(E_UNEXPECTED, "This is a compilation test, and should not be called");
 | |
| 
 | |
|     // template <typename TAsync = ABI::Windows::Foundation::IAsyncAction>
 | |
|     // inline HRESULT wait_for_completion_nothrow(_In_ TAsync* operation, COWAIT_FLAGS flags = COWAIT_DISPATCH_CALLS, DWORD timeout = INFINITE);
 | |
|     IAsyncAction* action = nullptr;
 | |
|     wil::wait_for_completion_nothrow(action);
 | |
|     wil::wait_for_completion_nothrow(action, COWAIT_DEFAULT);
 | |
| 
 | |
|     // template <typename TResult>
 | |
|     // HRESULT wait_for_completion_nothrow(_In_ ABI::Windows::Foundation::IAsyncOperation<TResult>* operation,
 | |
|     //     _Out_ typename wil::details::MapAsyncOpResultType<TResult>::type* result,
 | |
|     //     COWAIT_FLAGS flags = COWAIT_DISPATCH_CALLS, DWORD timeout = INFINITE);
 | |
|     IAsyncOperation<bool>* operation = nullptr;
 | |
|     wil::wait_for_completion_nothrow(operation);
 | |
|     wil::wait_for_completion_nothrow(operation, COWAIT_DEFAULT);
 | |
| 
 | |
|     // template <typename TResult, typename TProgress>
 | |
|     // HRESULT wait_for_completion_nothrow(_In_ ABI::Windows::Foundation::IAsyncOperationWithProgress<TResult, TProgress>* operation,
 | |
|     //     _Out_ typename wil::details::MapAsyncOpProgressResultType<TResult, TProgress>::type* result,
 | |
|     //     COWAIT_FLAGS flags = COWAIT_DISPATCH_CALLS, DWORD timeout = INFINITE);
 | |
| 
 | |
|     ComPtr<IAsyncOperation<bool>> operationWithResult;
 | |
|     boolean result = false;
 | |
|     wil::wait_for_completion_nothrow(operationWithResult.Get(), &result);
 | |
|     wil::wait_for_completion_nothrow(operationWithResult.Get(), &result, COWAIT_DEFAULT);
 | |
| 
 | |
|     DWORD timeoutValue = 1000; // arbitrary
 | |
|     bool timedOut = false;
 | |
| 
 | |
|     // template <typename TAsync = ABI::Windows::Foundation::IAsyncAction>
 | |
|     // inline HRESULT wait_for_completion_or_timeout_nothrow(_In_ TAsync* operation,
 | |
|     //     DWORD timeoutValue, _Out_ bool* timedOut, COWAIT_FLAGS flags = COWAIT_DISPATCH_CALLS);
 | |
|     wil::wait_for_completion_or_timeout_nothrow(action, timeoutValue, &timedOut);
 | |
|     wil::wait_for_completion_or_timeout_nothrow(action, timeoutValue, &timedOut, COWAIT_DEFAULT);
 | |
| 
 | |
|     // template <typename TResult>
 | |
|     // HRESULT wait_for_completion_or_timeout_nothrow(_In_ ABI::Windows::Foundation::IAsyncOperation<TResult>* operation,
 | |
|     //     _Out_ typename wil::details::MapAsyncOpResultType<TResult>::type* result,
 | |
|     //     DWORD timeoutValue, _Out_ bool* timedOut, COWAIT_FLAGS flags = COWAIT_DISPATCH_CALLS);
 | |
|     wil::wait_for_completion_or_timeout_nothrow(operation, timeoutValue, &timedOut);
 | |
|     wil::wait_for_completion_or_timeout_nothrow(operation, timeoutValue, &timedOut, COWAIT_DEFAULT);
 | |
| 
 | |
|     // template <typename TResult, typename TProgress>
 | |
|     // HRESULT wait_for_completion_or_timeout_nothrow(_In_ ABI::Windows::Foundation::IAsyncOperationWithProgress<TResult, TProgress>* operation,
 | |
|     //     _Out_ typename wil::details::MapAsyncOpProgressResultType<TResult, TProgress>::type* result,
 | |
|     //     DWORD timeoutValue, _Out_ bool* timedOut, COWAIT_FLAGS flags = COWAIT_DISPATCH_CALLS);
 | |
|     wil::wait_for_completion_or_timeout_nothrow(operationWithResult.Get(), &result, timeoutValue, &timedOut);
 | |
|     wil::wait_for_completion_or_timeout_nothrow(operationWithResult.Get(), &result, timeoutValue, &timedOut, COWAIT_DEFAULT);
 | |
| 
 | |
| #ifdef WIL_ENABLE_EXCEPTIONS
 | |
|     // template <typename TAsync = ABI::Windows::Foundation::IAsyncAction>
 | |
|     // inline void wait_for_completion(_In_ TAsync* operation, COWAIT_FLAGS flags = COWAIT_DISPATCH_CALLS, DWORD timeout = INFINITE);
 | |
|     wil::wait_for_completion(action);
 | |
|     wil::wait_for_completion(action, COWAIT_DEFAULT);
 | |
| 
 | |
|     // template <typename TResult, typename TReturn = typename wil::details::MapToSmartType<typename wil::details::MapAsyncOpResultType<TResult>::type>::type>
 | |
|     // TReturn
 | |
|     //     wait_for_completion(_In_ ABI::Windows::Foundation::IAsyncOperation<TResult>* operation, COWAIT_FLAGS flags = COWAIT_DISPATCH_CALLS, DWORD timeout = INFINITE);
 | |
|     wil::wait_for_completion(operation);
 | |
|     wil::wait_for_completion(operation, COWAIT_DEFAULT);
 | |
| 
 | |
|     // template <typename TResult, typename TProgress, typename TReturn = typename wil::details::MapToSmartType<typename wil::details::MapAsyncOpResultType<TResult>::type>::type>
 | |
|     // TReturn
 | |
|     //     wait_for_completion(_In_ ABI::Windows::Foundation::IAsyncOperationWithProgress<TResult, TProgress>* operation, COWAIT_FLAGS flags = COWAIT_DISPATCH_CALLS, DWORD timeout = INFINITE);
 | |
|     result = wil::wait_for_completion(operationWithResult.Get());
 | |
|     result = wil::wait_for_completion(operationWithResult.Get(), COWAIT_DEFAULT);
 | |
| #endif
 | |
| }
 | |
| #pragma warning(pop)
 | |
| #endif
 | |
| 
 | |
| TEST_CASE("WinRTTests::TimeTTests", "[winrt][time_t]")
 | |
| {
 | |
|     // Verifying that converting DateTime variable set as the date that means 0 as time_t works
 | |
|     DateTime time1 = { wil::SecondsToStartOf1970 * wil::HundredNanoSecondsInSecond };
 | |
|     __time64_t time_t1 = wil::DateTime_to_time_t(time1);
 | |
|     REQUIRE(time_t1 == 0);
 | |
| 
 | |
|     // Verifying that converting back to DateTime would return the same value
 | |
|     DateTime time2 = wil::time_t_to_DateTime(time_t1);
 | |
|     REQUIRE(time1.UniversalTime == time2.UniversalTime);
 | |
| 
 | |
|     // Verifying that converting to time_t for non-zero value also works
 | |
|     time2.UniversalTime += wil::HundredNanoSecondsInSecond * 123;
 | |
|     __time64_t time_t2 = wil::DateTime_to_time_t(time2);
 | |
|     REQUIRE(time_t2 - time_t1 == 123);
 | |
| 
 | |
|     // Verifying that converting back to DateTime for non-zero value also works
 | |
|     time1 = wil::time_t_to_DateTime(time_t2);
 | |
|     REQUIRE(time1.UniversalTime == time2.UniversalTime);
 | |
| }
 | |
| 
 | |
| ComPtr<IVector<IInspectable*>> MakeSampleInspectableVector()
 | |
| {
 | |
|     auto result = Make<FakeVector<IInspectable*>>();
 | |
|     REQUIRE(result);
 | |
| 
 | |
|     ComPtr<IPropertyValueStatics> propStatics;
 | |
|     REQUIRE_SUCCEEDED(GetActivationFactory(HStringReference(RuntimeClass_Windows_Foundation_PropertyValue).Get(), &propStatics));
 | |
| 
 | |
|     for (UINT32 i = 0; i < 5; ++i)
 | |
|     {
 | |
|         ComPtr<IInspectable> myProp;
 | |
|         REQUIRE_SUCCEEDED(propStatics->CreateUInt32(i, &myProp));
 | |
|         REQUIRE_SUCCEEDED(result->Append(myProp.Get()));
 | |
|     }
 | |
| 
 | |
|     return result;
 | |
| }
 | |
| 
 | |
| ComPtr<IVector<HSTRING>> MakeSampleStringVector()
 | |
| {
 | |
|     auto result = Make<FakeVector<HSTRING>>();
 | |
|     REQUIRE(result);
 | |
| 
 | |
|     const HStringReference items[] = { HStringReference(L"one"), HStringReference(L"two"), HStringReference(L"three") };
 | |
|     for (const auto& i : items)
 | |
|     {
 | |
|         REQUIRE_SUCCEEDED(result->Append(i.Get()));
 | |
|     }
 | |
| 
 | |
|     return result;
 | |
| }
 | |
| 
 | |
| ComPtr<IVector<Point>> MakeSamplePointVector()
 | |
| {
 | |
|     auto result = Make<FakeVector<Point>>();
 | |
|     REQUIRE(result);
 | |
| 
 | |
|     for (int i = 0; i < 5; ++i)
 | |
|     {
 | |
|         auto value = static_cast<float>(i);
 | |
|         REQUIRE_SUCCEEDED(result->Append(Point{ value, value }));
 | |
|     }
 | |
| 
 | |
|     return result;
 | |
| }
 | |
| 
 | |
| TEST_CASE("WinRTTests::VectorRangeTest", "[winrt][vector_range]")
 | |
| {
 | |
|     auto uninit = wil::RoInitialize_failfast();
 | |
| 
 | |
|     auto inspectables = MakeSampleInspectableVector();
 | |
|     unsigned count = 0;
 | |
|     REQUIRE_SUCCEEDED(inspectables->get_Size(&count));
 | |
| 
 | |
|     unsigned idx = 0;
 | |
|     HRESULT success = S_OK;
 | |
|     for (const auto& i : wil::get_range_nothrow(inspectables.Get(), &success))
 | |
|     {
 | |
|         // Duplications are not a typo - they verify the thing is callable twice
 | |
| 
 | |
|         UINT32 value;
 | |
|         ComPtr<IReference<UINT32>> intRef;
 | |
|         REQUIRE_SUCCEEDED(i.CopyTo(IID_PPV_ARGS(&intRef)));
 | |
|         REQUIRE_SUCCEEDED(intRef->get_Value(&value));
 | |
|         REQUIRE(idx == value);
 | |
|         REQUIRE_SUCCEEDED(i.CopyTo(IID_PPV_ARGS(&intRef)));
 | |
|         REQUIRE_SUCCEEDED(intRef->get_Value(&value));
 | |
|         REQUIRE(idx == value);
 | |
| 
 | |
|         ++idx;
 | |
| 
 | |
|         HString rtc;
 | |
|         REQUIRE_SUCCEEDED(i->GetRuntimeClassName(rtc.GetAddressOf()));
 | |
|         REQUIRE_SUCCEEDED(i->GetRuntimeClassName(rtc.GetAddressOf()));
 | |
|     }
 | |
|     REQUIRE_SUCCEEDED(success);
 | |
|     REQUIRE(count == idx);
 | |
| 
 | |
|     auto strings = MakeSampleStringVector();
 | |
|     for (const auto& i : wil::get_range_nothrow(strings.Get(), &success))
 | |
|     {
 | |
|         REQUIRE(i.Get());
 | |
|         REQUIRE(i.Get());
 | |
|     }
 | |
|     REQUIRE_SUCCEEDED(success);
 | |
| 
 | |
|     int index = 0;
 | |
|     auto points = MakeSamplePointVector();
 | |
|     for (auto value : wil::get_range_nothrow(points.Get(), &success))
 | |
|     {
 | |
|         REQUIRE(index++ == value.Get().X);
 | |
|     }
 | |
|     REQUIRE_SUCCEEDED(success);
 | |
| 
 | |
|     // operator-> should not clear out the pointer
 | |
|     auto inspRange = wil::get_range_nothrow(inspectables.Get());
 | |
|     for (auto itr = inspRange.begin(); itr != inspRange.end(); ++itr)
 | |
|     {
 | |
|         REQUIRE(itr->Get());
 | |
|     }
 | |
| 
 | |
|     auto strRange = wil::get_range_nothrow(strings.Get());
 | |
|     for (auto itr = strRange.begin(); itr != strRange.end(); ++itr)
 | |
|     {
 | |
|         REQUIRE(itr->Get());
 | |
|     }
 | |
| 
 | |
|     index = 0;
 | |
|     auto pointRange = wil::get_range_nothrow(points.Get());
 | |
|     for (auto itr = pointRange.begin(); itr != pointRange.end(); ++itr)
 | |
|     {
 | |
|         REQUIRE(index++ == itr->Get().X);
 | |
|     }
 | |
| 
 | |
| #if (defined WIL_ENABLE_EXCEPTIONS)
 | |
|     idx = 0;
 | |
|     for (const auto& i : wil::get_range(inspectables.Get()))
 | |
|     {
 | |
|         // Duplications are not a typo - they verify the thing is callable twice
 | |
| 
 | |
|         UINT32 value;
 | |
|         ComPtr<IReference<UINT32>> intRef;
 | |
|         REQUIRE_SUCCEEDED(i.CopyTo(IID_PPV_ARGS(&intRef)));
 | |
|         REQUIRE_SUCCEEDED(intRef->get_Value(&value));
 | |
|         REQUIRE(idx == value);
 | |
|         REQUIRE_SUCCEEDED(i.CopyTo(IID_PPV_ARGS(&intRef)));
 | |
|         REQUIRE_SUCCEEDED(intRef->get_Value(&value));
 | |
|         REQUIRE(idx == value);
 | |
| 
 | |
|         ++idx;
 | |
| 
 | |
|         HString rtc;
 | |
|         REQUIRE_SUCCEEDED(i->GetRuntimeClassName(rtc.GetAddressOf()));
 | |
|         REQUIRE_SUCCEEDED(i->GetRuntimeClassName(rtc.GetAddressOf()));
 | |
|     }
 | |
|     REQUIRE(count == idx);
 | |
| 
 | |
|     for (const auto& i : wil::get_range(strings.Get()))
 | |
|     {
 | |
|         REQUIRE(i.Get());
 | |
|         REQUIRE(i.Get());
 | |
|     }
 | |
| 
 | |
|     index = 0;
 | |
|     for (auto value : wil::get_range(points.Get()))
 | |
|     {
 | |
|         REQUIRE(index++ == value.Get().X);
 | |
|     }
 | |
| 
 | |
|     // operator-> should not clear out the pointer
 | |
|     for (auto itr = inspRange.begin(); itr != inspRange.end(); ++itr)
 | |
|     {
 | |
|         REQUIRE(itr->Get());
 | |
|     }
 | |
| 
 | |
|     for (auto itr = strRange.begin(); itr != strRange.end(); ++itr)
 | |
|     {
 | |
|         REQUIRE(itr->Get());
 | |
|     }
 | |
| 
 | |
|     index = 0;
 | |
|     for (auto itr = pointRange.begin(); itr != pointRange.end(); ++itr)
 | |
|     {
 | |
|         REQUIRE(index++ == itr->Get().X);
 | |
|     }
 | |
| 
 | |
|     // Iterator self-assignment is a nop.
 | |
|     {
 | |
|         auto inspRange2 = wil::get_range(inspectables.Get());
 | |
|         auto itr = inspRange2.begin();
 | |
|         REQUIRE(itr != inspRange2.end()); // should have something in it
 | |
|         auto& ref = *itr;
 | |
|         auto val = ref;
 | |
|         itr = itr;
 | |
|         REQUIRE(val == ref);
 | |
|         itr = std::move(itr);
 | |
|         REQUIRE(val == ref);
 | |
|     }
 | |
| 
 | |
|     {
 | |
|         auto strRange2 = wil::get_range(strings.Get());
 | |
|         auto itr = strRange2.begin();
 | |
|         REQUIRE(itr != strRange2.end()); // should have something in it
 | |
|         auto& ref = *itr;
 | |
|         auto val = ref.Get();
 | |
|         itr = itr;
 | |
|         REQUIRE(val == ref);
 | |
|         itr = std::move(itr);
 | |
|         REQUIRE(val == ref.Get());
 | |
|     }
 | |
| #endif
 | |
| }
 | |
| 
 | |
| unsigned long GetComObjectRefCount(IUnknown* unk) { unk->AddRef(); return unk->Release(); }
 | |
| 
 | |
| TEST_CASE("WinRTTests::VectorRangeLeakTest", "[winrt][vector_range]")
 | |
| {
 | |
|     auto uninit = wil::RoInitialize_failfast();
 | |
| 
 | |
|     auto inspectables = MakeSampleInspectableVector();
 | |
|     ComPtr<IInspectable> verifyNotLeaked;
 | |
|     HRESULT hr = S_OK;
 | |
|     for (const auto& ptr : wil::get_range_nothrow(inspectables.Get(), &hr))
 | |
|     {
 | |
|         if (!verifyNotLeaked)
 | |
|         {
 | |
|             verifyNotLeaked = ptr;
 | |
|         }
 | |
|     }
 | |
|     inspectables = nullptr; // clear all refs to verifyNotLeaked
 | |
|     REQUIRE_SUCCEEDED(hr);
 | |
|     REQUIRE(GetComObjectRefCount(verifyNotLeaked.Get()) == 1);
 | |
| 
 | |
|     inspectables = MakeSampleInspectableVector();
 | |
|     for (const auto& ptr : wil::get_range_failfast(inspectables.Get()))
 | |
|     {
 | |
|         if (!verifyNotLeaked)
 | |
|         {
 | |
|             verifyNotLeaked = ptr;
 | |
|         }
 | |
|     }
 | |
|     inspectables = nullptr; // clear all refs to verifyNotLeaked
 | |
|     REQUIRE(GetComObjectRefCount(verifyNotLeaked.Get()) == 1);
 | |
| 
 | |
| #if (defined WIL_ENABLE_EXCEPTIONS)
 | |
|     inspectables = MakeSampleInspectableVector();
 | |
|     for (const auto& ptr : wil::get_range(inspectables.Get()))
 | |
|     {
 | |
|         if (!verifyNotLeaked)
 | |
|         {
 | |
|             verifyNotLeaked = ptr;
 | |
|         }
 | |
|     }
 | |
|     inspectables = nullptr; // clear all refs to verifyNotLeaked
 | |
|     REQUIRE(GetComObjectRefCount(verifyNotLeaked.Get()) == 1);
 | |
| #endif
 | |
| }
 | |
| 
 | |
| TEST_CASE("WinRTTests::TwoPhaseConstructor", "[winrt][hstring]")
 | |
| {
 | |
|     const wchar_t left[] = L"left";
 | |
|     const wchar_t right[] = L"right";
 | |
|     ULONG needed = ARRAYSIZE(left) + ARRAYSIZE(right) - 1;
 | |
|     auto maker = wil::TwoPhaseHStringConstructor::Preallocate(needed);
 | |
|     REQUIRE_SUCCEEDED(StringCbCopyW(maker.Get(), maker.ByteSize(), left));
 | |
|     REQUIRE_SUCCEEDED(StringCbCatW(maker.Get(), maker.ByteSize(), right));
 | |
|     REQUIRE_SUCCEEDED(maker.Validate(needed * sizeof(wchar_t)));
 | |
| 
 | |
|     wil::unique_hstring promoted{ maker.Promote() };
 | |
|     REQUIRE(wcscmp(L"leftright", str_raw_ptr(promoted)) == 0);
 | |
| } |