123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278 |
- /*
- * Copyright (c) Contributors to the Open 3D Engine Project.
- * For complete copyright and license terms please see the LICENSE at the root of this distribution.
- *
- * SPDX-License-Identifier: Apache-2.0 OR MIT
- *
- */
- #include <AzCore/RTTI/RTTI.h>
- #include <AzCore/RTTI/ReflectionManager.h>
- #include <AzCore/Serialization/SerializeContext.h>
- #include <AzCore/UnitTest/TestTypes.h>
- #include <AzCore/std/containers/array.h>
- #include <AzCore/std/containers/bitset.h>
- #include <AzCore/std/containers/fixed_forward_list.h>
- #include <AzCore/std/containers/fixed_list.h>
- #include <AzCore/std/containers/fixed_unordered_map.h>
- #include <AzCore/std/containers/fixed_unordered_set.h>
- #include <AzCore/std/containers/fixed_vector.h>
- #include <AzCore/std/containers/forward_list.h>
- #include <AzCore/std/containers/map.h>
- #include <AzCore/std/containers/set.h>
- #include <AzCore/std/containers/unordered_map.h>
- #include <AzCore/std/containers/unordered_set.h>
- #include <AzCore/std/containers/variant.h>
- #include <AzCore/std/containers/vector.h>
- #include <AzCore/std/functional.h>
- #include <AzCore/std/parallel/thread.h>
- #include <AzCore/std/smart_ptr/unique_ptr.h>
- #include <AzCore/std/string/string.h>
- #include <AzCore/std/tuple.h>
- #include <AzCore/std/utils.h>
- // Compile checks that a primitive type(int) and qualified types have AzTypeInfo
- static_assert(AZ::Internal::HasAzTypeInfo_v<int>, "int value type should is missing GetO3deTypeName or GetO3deTypeId overload");
- static_assert(AZ::Internal::HasAzTypeInfo_v<int*>, "pointer to int type is missing GetO3deTypeName or GetO3deTypeId overload");
- static_assert(AZ::Internal::HasAzTypeInfo_v<int const>, "const int value type is missing GetO3deTypeName or GetO3deTypeId overload");
- static_assert(AZ::Internal::HasAzTypeInfo_v<int&>, "lvalue reference to non-const int type is missing GetO3deTypeName or GetO3deTypeId overload");
- static_assert(AZ::Internal::HasAzTypeInfo_v<int const&>, "lvalue reference to const int type is missing GetO3deTypeName or GetO3deTypeId overload");
- static_assert(AZ::Internal::HasAzTypeInfo_v<int&&>, "rvalue reference to non-const int type is missing GetO3deTypeName or GetO3deTypeId overload");
- static_assert(AZ::Internal::HasAzTypeInfo_v<int const&&>, "rvalue reference to const int type is missing GetO3deTypeName or GetO3deTypeId overload");
- // Check unique_ptr<T>
- static_assert(AZ::Internal::HasAzTypeInfo_v<AZStd::unique_ptr<int>>, "unique_ptr<T> is missing GetO3deTypeName or GetO3deTypeId overload");
- static_assert(AZStd::is_same_v<decltype(GetO3deTemplateId(AZ::Adl{},
- AZ::AzGenericTypeInfo::Internal::TemplateIdentityTypes<std::unique_ptr>{})), AZ::TemplateId>,
- "unique_ptr is missing GetO3deTemplateId overload");
- namespace UnitTest
- {
- template<class T>
- struct AliasTest;
- }
- namespace UnitTestUsing
- {
- using UnitTest::AliasTest;
- }
- namespace UnitTestAlias
- {
- template<class T>
- using AliasTest = UnitTest::AliasTest<T>;
- }
- // The `using` declaration brings the UnitTest::AliasTest type into the UnitTestUsing scope
- // So canonically they are treated as the same temp[late
- static_assert(AZStd::is_same_v<AZ::AzGenericTypeInfo::Internal::TemplateIdentityTypes<UnitTest::AliasTest>,
- AZ::AzGenericTypeInfo::Internal::TemplateIdentityTypes<UnitTestUsing::AliasTest>>);
- // An alias template are a separate template from any types it references
- // An alias template doesn't actually alias another template, but is its own construct.
- // For example `template<class T> void_t = void`, doesn't "reference" another template, it just maps
- // any instantiation of the template to void
- // Similarly `template<class T> unique_ptr = std::unique_ptr<T>`, is not an alias for the `std::unique_ptr`
- // template, the result of it's instantiation results in a class type of `std::unique_ptr<T>`
- #if !defined(AZ_COMPILER_GCC)
- static_assert(!AZStd::is_same_v<AZ::AzGenericTypeInfo::Internal::TemplateIdentityTypes<UnitTest::AliasTest>,
- AZ::AzGenericTypeInfo::Internal::TemplateIdentityTypes<UnitTestAlias::AliasTest>>);
- #else
- // On GCC the Alias template matches actual template identifier
- static_assert(AZStd::is_same_v<AZ::AzGenericTypeInfo::Internal::TemplateIdentityTypes<UnitTest::AliasTest>,
- AZ::AzGenericTypeInfo::Internal::TemplateIdentityTypes<UnitTestAlias::AliasTest>>);
- #endif
- namespace UnitTest
- {
- // Non intrusive typeinfo for external and integral types
- struct ExternalClass
- {
- };
- AZ_TYPE_INFO_SPECIALIZE(ExternalClass, "{38380915-084B-4886-8D3D-B8439E9E987C}");
- }
- // These 2 types must only EVER be used by the MultiThreadedTypeInfo test, or else
- // that test is invalidated because the statics will have been initialized already
- struct MTTI {};
- struct MTTI2
- {
- AZ_TYPE_INFO(MTTI2, "{CBC94693-5ECD-4CBF-A8DB-9B122E697E8D}");
- };
- namespace AZ
- {
- enum class PlatformID;
- AZ_TYPE_INFO_SPECIALIZE(MTTI, "{4876C017-0C26-4D0D-9A1F-2A738BAE6449}");
- }
- namespace UnitTest
- {
- class Rtti
- : public LeakDetectionFixture
- {
- };
- // Intrusive TypeInfo
- struct MyClass
- {
- AZ_TYPE_INFO(MyClass, "{CADA6BA7-D479-4C20-B7F0-121A1DF4E9CC}");
- };
- template<class T1, class T2>
- struct MyClassTemplate
- {
- };
- template<class... Args>
- struct MyClassVariadicTemplate
- {
- };
- }
- namespace AZ
- {
- AZ_TYPE_INFO_TEMPLATE(UnitTest::MyClassTemplate, "{EBFE7ADF-1FCE-47F0-B417-14FE06BAF02D}", AZ_TYPE_INFO_CLASS, AZ_TYPE_INFO_CLASS);
- AZ_TYPE_INFO_TEMPLATE(UnitTest::MyClassVariadicTemplate, "{60C1D809-09FA-48EB-A9B7-0BD8DBFF21C8}", AZ_TYPE_INFO_CLASS_VARARGS);
- }
- namespace UnitTest
- {
- // Tests if known types maintain their assigned/constructed uuids properly. Changes to this can have significant impact
- // various systems such as serialization.
- TEST_F(Rtti, KnownTypes)
- {
- EXPECT_EQ(AZ::Uuid("{3AB0037F-AF8D-48ce-BCA0-A170D18B2C03}"), azrtti_typeid<char>());
- EXPECT_EQ(AZ::Uuid("{58422C0E-1E47-4854-98E6-34098F6FE12D}"), azrtti_typeid<AZ::s8>());
- EXPECT_EQ(AZ::Uuid("{B8A56D56-A10D-4dce-9F63-405EE243DD3C}"), azrtti_typeid<short>());
- EXPECT_EQ(AZ::Uuid("{72039442-EB38-4d42-A1AD-CB68F7E0EEF6}"), azrtti_typeid<int>());
- EXPECT_EQ(AZ::Uuid("{8F24B9AD-7C51-46cf-B2F8-277356957325}"), azrtti_typeid<long>());
- EXPECT_EQ(AZ::Uuid("{70D8A282-A1EA-462d-9D04-51EDE81FAC2F}"), azrtti_typeid<AZ::s64>());
- EXPECT_EQ(AZ::Uuid("{72B9409A-7D1A-4831-9CFE-FCB3FADD3426}"), azrtti_typeid<unsigned char>());
- EXPECT_EQ(AZ::Uuid("{ECA0B403-C4F8-4b86-95FC-81688D046E40}"), azrtti_typeid<unsigned short>());
- EXPECT_EQ(AZ::Uuid("{43DA906B-7DEF-4ca8-9790-854106D3F983}"), azrtti_typeid<unsigned int>());
- EXPECT_EQ(AZ::Uuid("{5EC2D6F7-6859-400f-9215-C106F5B10E53}"), azrtti_typeid<unsigned long>());
- EXPECT_EQ(AZ::Uuid("{D6597933-47CD-4fc8-B911-63F3E2B0993A}"), azrtti_typeid<AZ::u64>());
- EXPECT_EQ(AZ::Uuid("{EA2C3E90-AFBE-44d4-A90D-FAAF79BAF93D}"), azrtti_typeid<float>());
- EXPECT_EQ(AZ::Uuid("{110C4B14-11A8-4e9d-8638-5051013A56AC}"), azrtti_typeid<double>());
- EXPECT_EQ(AZ::Uuid("{A0CA880C-AFE4-43cb-926C-59AC48496112}"), azrtti_typeid<bool>());
- EXPECT_EQ(AZ::Uuid("{E152C105-A133-4d03-BBF8-3D4B2FBA3E2A}"), azrtti_typeid<AZ::Uuid>());
- EXPECT_EQ(AZ::Uuid("{C0F1AFAD-5CB3-450E-B0F5-ADB5D46B0E22}"), azrtti_typeid<void>());
- EXPECT_EQ(AZ::Uuid("{9F4E062E-06A0-46D4-85DF-E0DA96467D3A}"), azrtti_typeid<AZ::Crc32>());
- EXPECT_EQ(AZ::Uuid("{0635D08E-DDD2-48DE-A7AE-73CC563C57C3}"), azrtti_typeid<AZ::PlatformID>());
- EXPECT_EQ(AZ::Uuid("{72039442-EB38-4D42-A1AD-CB68F7E0EEF6}"), azrtti_typeid<int*>());
- EXPECT_EQ(AZ::Uuid("{72039442-EB38-4D42-A1AD-CB68F7E0EEF6}"), azrtti_typeid<int&>());
- EXPECT_EQ(AZ::Uuid("{72039442-EB38-4D42-A1AD-CB68F7E0EEF6}"), azrtti_typeid<int&&>());
- EXPECT_EQ(AZ::Uuid("{72039442-EB38-4D42-A1AD-CB68F7E0EEF6}"), azrtti_typeid<const int*>());
- EXPECT_EQ(AZ::Uuid("{72039442-EB38-4D42-A1AD-CB68F7E0EEF6}"), azrtti_typeid<const int&>());
- EXPECT_EQ(AZ::Uuid("{72039442-EB38-4D42-A1AD-CB68F7E0EEF6}"), azrtti_typeid<const int&&>());
- EXPECT_EQ(AZ::Uuid("{72039442-EB38-4D42-A1AD-CB68F7E0EEF6}"), azrtti_typeid<const int>());
- // TypeId aggregation with template ID as prefix
- // Aggregation uses right fold for addition (Id1 + (Id2 + (Id3 + (... + Idn))))
- EXPECT_EQ(azrtti_typeid<int>() + azrtti_typeid<char>(), (AZ::Internal::AggregateTypes<int, char>::GetCanonicalTypeId()));
- EXPECT_EQ(azrtti_typeid<AZStd::tuple>() + azrtti_typeid<int>(), azrtti_typeid<AZStd::tuple<int>>());
- EXPECT_EQ(azrtti_typeid<AZStd::tuple>() + (azrtti_typeid<int>() + azrtti_typeid<char>()), (azrtti_typeid<AZStd::tuple<int, char>>()));
- EXPECT_NE(azrtti_typeid<AZStd::tuple>() + azrtti_typeid<int>() + azrtti_typeid<char>(), (azrtti_typeid<AZStd::tuple<int, char>>()));
- EXPECT_EQ(AZ::Uuid("{B2F5707A-08FA-566A-BE44-226C634405BE}"), (azrtti_typeid<AZStd::less<int>>()));
- EXPECT_EQ(AZ::Uuid("{6D2500BA-EE64-5288-9766-4C7CD8A10476}"), (azrtti_typeid<AZStd::less_equal<int>>()));
- EXPECT_EQ(AZ::Uuid("{5959973B-2113-5789-BC8C-2F1E4A917953}"), (azrtti_typeid<AZStd::greater<int>>()));
- EXPECT_EQ(AZ::Uuid("{7769141C-BF97-5E9B-B77F-F075FA915905}"), (azrtti_typeid<AZStd::greater_equal<int>>()));
- EXPECT_EQ(AZ::Uuid("{39487937-0E1C-5F78-8A7E-B24EFE32F48F}"), (azrtti_typeid<AZStd::equal_to<int>>()));
- EXPECT_EQ(AZ::Uuid("{AE785799-21A1-5D89-A083-E4441E1F81A8}"), (azrtti_typeid<AZStd::hash<int>>()));
- EXPECT_EQ(AZ::Uuid("{64503325-ECF4-5F02-95F9-E37D00810E59}"), (azrtti_typeid<AZStd::pair<int, int>>()));
- EXPECT_EQ(AZ::Uuid("{853CDD8D-12FF-5619-9A42-10178785620A}"), (azrtti_typeid<AZStd::tuple<int, char, float, double>>()));
- EXPECT_EQ(AZ::Uuid("{85AFA5E8-AA5C-50A3-9CAB-B8C483DA88C5}"), (azrtti_typeid<AZStd::vector<int>>()));
- EXPECT_EQ(AZ::Uuid("{09C2272F-2353-5337-BDCB-B1D0D6A2A778}"), (azrtti_typeid<AZStd::list<int>>()));
- EXPECT_EQ(AZ::Uuid("{2D875DAD-A157-5792-AE25-96D909E1BE4C}"), (azrtti_typeid<AZStd::forward_list<int>>()));
- EXPECT_EQ(AZ::Uuid("{9DF03CD1-931A-544D-A93B-0546907B70CA}"), (azrtti_typeid<AZStd::set<int>>()));
- EXPECT_EQ(AZ::Uuid("{243A34FA-C6F6-51D1-8166-06DED5141370}"), (azrtti_typeid<AZStd::unordered_set<int>>()));
- EXPECT_EQ(AZ::Uuid("{79F4B21A-02CD-58C1-9669-FA2E5E7A142A}"), (azrtti_typeid<AZStd::unordered_multiset<int>>()));
- EXPECT_EQ(AZ::Uuid("{BB54671F-18E6-5F96-B659-FA236D1B7D31}"), (azrtti_typeid<AZStd::map<int, int>>()));
- EXPECT_EQ(AZ::Uuid("{C543E26A-7772-5511-8CE1-A8FA6441CAD3}"), (azrtti_typeid<AZStd::unordered_map<int, int>>()));
- EXPECT_EQ(AZ::Uuid("{FD30FBC0-B826-51CF-A75B-E00466FEB0F0}"), (azrtti_typeid<AZStd::unordered_map<AZStd::string, MyClass>>()));
- EXPECT_EQ(AZ::Uuid("{64E53B04-DD49-55DB-8299-5B4ED53A5F1C}"), (azrtti_typeid<AZStd::unordered_multimap<int, int>>()));
- EXPECT_EQ(AZ::Uuid("{1C213FE1-ED58-5889-8FC9-48D0E11D2E7E}"), (azrtti_typeid<AZStd::unordered_multimap<AZStd::string, MyClass>>()));
- EXPECT_EQ(AZ::Uuid("{0BF83553-00B0-5B7C-9BF3-A87C811F0752}"), (azrtti_typeid<AZStd::shared_ptr<int>>()));
- EXPECT_EQ(AZ::Uuid("{E91D2018-767D-57D4-AF21-5CBEA51A15EC}"), (azrtti_typeid<AZStd::optional<int>>()));
- EXPECT_EQ(AZ::Uuid("{03AAAB3F-5C47-5A66-9EBC-D5FA4DB353C9}"), (azrtti_typeid<AZStd::basic_string<char>>()));
- EXPECT_EQ(AZ::Uuid("{406E9B16-A89C-5289-B10E-17F338588559}"), (azrtti_typeid<AZStd::char_traits<char>>()));
- EXPECT_EQ(AZ::Uuid("{7114E998-A8B4-519B-9342-A86D1587B4F7}"), (azrtti_typeid<AZStd::basic_string_view<char>>()));
- EXPECT_EQ(AZ::Uuid("{A3C35B6E-E2DE-58F7-A897-06C64C5BC1E3}"), (azrtti_typeid<AZStd::fixed_vector<int, 4>>()));
- EXPECT_EQ(AZ::Uuid("{A3C35B6E-E2DE-58F7-A897-06C64C5BC1E3}"), (azrtti_typeid<AZStd::fixed_vector<int, 4>>()));
- EXPECT_EQ(AZ::Uuid("{F670463F-FB3F-5CF3-A1FE-A7CC6DB312E8}"), (azrtti_typeid<AZStd::fixed_list<int, 4>>()));
- EXPECT_EQ(AZ::Uuid("{71C90433-74CE-5018-BEFD-FC98F4451AEF}"), (azrtti_typeid<AZStd::fixed_forward_list<int, 4>>()));
- EXPECT_EQ(AZ::Uuid("{DD9565F2-A80F-5DD3-B33F-0B0BF1C24A4F}"), (azrtti_typeid<AZStd::array<int, 4>>()));
- EXPECT_EQ(AZ::Uuid("{E5848517-FBDC-5D0F-9012-B16951027D9E}"), (azrtti_typeid<AZStd::bitset<8>>()));
- EXPECT_EQ(AZ::Uuid("{2962AD0B-AD39-5D19-9548-7AB0E68A1787}"), (azrtti_typeid<AZStd::function<bool(int)>>()));
- EXPECT_EQ(AZ::Uuid("{B1E9136B-D77A-4643-BE8E-2ABDA246AE0E}"), (azrtti_typeid<AZStd::monostate>()));
- EXPECT_EQ(AZ::Uuid("{7570E0E7-0BA8-5382-BB14-CEB7B1C0DBEB}"), (azrtti_typeid<AZStd::variant<int, char>>()));
- }
- TEST_F(Rtti, TypeInfoQualifierTest)
- {
- // The Uuid of a value type should result of a pointer type
- EXPECT_EQ(azrtti_typeid<int>(), azrtti_typeid<int*>());
- EXPECT_EQ(AZ::AzTypeInfo<int>::Uuid(), AZ::AzTypeInfo<int*>::Uuid());
- EXPECT_EQ(AZ::AzTypeInfo<int>::GetCanonicalTypeId(), AZ::AzTypeInfo<int*>::GetPointeeTypeId());
- EXPECT_EQ(AZStd::string_view{ AZ::AzTypeInfo<int>::Name() }, AZStd::string_view("int"));
- EXPECT_EQ(AZStd::string_view{ AZ::AzTypeInfo<int const>::Name() }, AZStd::string_view("int const"));
- EXPECT_EQ(AZStd::string_view{ AZ::AzTypeInfo<int const&>::Name() }, AZStd::string_view("int const&"));
- EXPECT_EQ(AZStd::string_view{ AZ::AzTypeInfo<int const&&>::Name() }, AZStd::string_view("int const&&"));
- EXPECT_EQ(AZStd::string_view{ AZ::AzTypeInfo<int&>::Name() }, AZStd::string_view("int&"));
- EXPECT_EQ(AZStd::string_view{ AZ::AzTypeInfo<int&&>::Name() }, AZStd::string_view("int&&"));
- EXPECT_EQ(AZStd::string_view{ AZ::AzTypeInfo<const int*>::Name() }, AZStd::string_view("int const*"));
- EXPECT_EQ(AZStd::string_view{ AZ::AzTypeInfo<const int* const>::Name() }, AZStd::string_view("int const* const"));
- EXPECT_EQ(AZStd::string_view{ AZ::AzTypeInfo<const int* const&>::Name() }, AZStd::string_view("int const* const&"));
- EXPECT_EQ(AZStd::string_view{ AZ::AzTypeInfo<const int* const&&>::Name() }, AZStd::string_view("int const* const&&"));
- EXPECT_EQ(AZStd::string_view{ AZ::AzTypeInfo<const int* &>::Name() }, AZStd::string_view("int const*&"));
- EXPECT_EQ(AZStd::string_view{ AZ::AzTypeInfo<const int* &&>::Name() }, AZStd::string_view("int const*&&"));
- EXPECT_EQ(AZ::AzTypeInfo<const int*>::Uuid(), AZ::AzTypeInfo<int*>::Uuid());
- // Validate that a container of value tpyes have a different type id than a container of pointers
- EXPECT_NE(azrtti_typeid<AZStd::vector<int>>(), azrtti_typeid<AZStd::vector<int*>>());
- }
- TEST_F(Rtti, TypeInfoTest)
- {
- EXPECT_EQ(AZ::Uuid("{CADA6BA7-D479-4C20-B7F0-121A1DF4E9CC}"), AZ::AzTypeInfo<MyClass>::Uuid());
- EXPECT_STREQ("MyClass", AZ::AzTypeInfo<MyClass>::Name());
- EXPECT_EQ(AZ::Uuid("{38380915-084B-4886-8D3D-B8439E9E987C}"), AZ::AzTypeInfo<ExternalClass>::Uuid());
- EXPECT_STREQ("ExternalClass", AZ::AzTypeInfo<ExternalClass>::Name());
- // template templates
- {
- // Check if the correct type id is returned.
- AZ::Uuid templateUuid = AZ::Uuid("{EBFE7ADF-1FCE-47F0-B417-14FE06BAF02D}");
- EXPECT_EQ(templateUuid, AZ::AzGenericTypeInfo::Uuid<MyClassTemplate>());
- // Check that the uuid of the template is returned if AzGenericTypeInfo is used to return the uuid.
- EXPECT_EQ(templateUuid, (AZ::AzGenericTypeInfo::Uuid<MyClassTemplate<MyClass, int>>()));
- using MyClassTemplateType = MyClassTemplate<MyClass, int>;
- EXPECT_EQ(templateUuid, (AZ::AzGenericTypeInfo::Uuid<MyClassTemplateType>()));
- // Check all combinations return a valid id.
- EXPECT_EQ(AZ::Uuid("{911B2EA8-CCB1-4F0C-A535-540AD00173AE}"), AZ::AzGenericTypeInfo::Uuid<AZStd::array>());
- EXPECT_EQ(AZ::Uuid("{6BAE9836-EC49-466A-85F2-F4B1B70839FB}"), AZ::AzGenericTypeInfo::Uuid<AZStd::bitset>());
- EXPECT_EQ(AZ::Uuid("{C9F9C644-CCC3-4F77-A792-F5B5DBCA746E}"), AZ::AzGenericTypeInfo::Uuid<AZStd::function>());
- EXPECT_EQ(AZ::Uuid("{A60E3E61-1FF6-4982-B6B8-9E4350C4C679}"), AZ::AzGenericTypeInfo::Uuid<AZStd::vector>());
- }
- // templates
- {
- AZ::Uuid templateUuid = AZ::Uuid("{EBFE7ADF-1FCE-47F0-B417-14FE06BAF02D}") + AZ::Internal::AggregateTypes<MyClass, int>::GetCanonicalTypeId();
- typedef MyClassTemplate<MyClass, int> MyClassTemplateType;
- EXPECT_EQ(templateUuid, AZ::AzTypeInfo<MyClassTemplateType>::Uuid());
- const char* myClassTemplatename = AZ::AzTypeInfo<MyClassTemplateType>::Name();
- EXPECT_NE(nullptr, strstr(myClassTemplatename, "MyClassTemplate"));
- EXPECT_NE(nullptr, strstr(myClassTemplatename, "MyClass"));
- EXPECT_NE(nullptr, strstr(myClassTemplatename, "int"));
- }
- // variadic templates
- {
- AZ::Uuid templateUuid = AZ::Uuid("{60C1D809-09FA-48EB-A9B7-0BD8DBFF21C8}") + AZ::Internal::AggregateTypes<MyClass, int>::GetCanonicalTypeId();
- using MyClassVariadicTemplateType = MyClassVariadicTemplate<MyClass, int>;
- EXPECT_EQ(templateUuid, AZ::AzTypeInfo<MyClassVariadicTemplateType>::Uuid());
- const char* myClassTemplatename = AZ::AzTypeInfo<MyClassVariadicTemplateType>::Name();
- EXPECT_NE(nullptr, strstr(myClassTemplatename, "MyClassVariadicTemplate"));
- EXPECT_NE(nullptr, strstr(myClassTemplatename, "MyClass"));
- EXPECT_NE(nullptr, strstr(myClassTemplatename, "int"));
- }
- }
- class MyBase
- {
- public:
- AZ_TYPE_INFO(MyBase, "{6A0855E5-6899-482B-B470-C3E5C13D13F5}");
- virtual ~MyBase() {}
- int dataMyBase;
- };
- class MyBase1
- : public MyBase
- {
- public:
- ~MyBase1() override {}
- // Event though MyBase doesn't have RTTI we do allow to be noted as a base class, of course it will NOT be
- // part of the RTTI chain. The goal is to allow AZ_RTTI to declare any base classes without worry if they have RTTI or not
- AZ_RTTI(MyBase1, "{F3F97A32-15D2-48FF-B741-B89EA2DD2280}", MyBase);
- int data1MyBase1;
- int data2MyBase1;
- };
- class MyDerived
- : public MyBase1
- {
- public:
- ~MyDerived() override {}
- AZ_RTTI(MyDerived, "{3BE0590A-F20F-4056-96AF-C2F0565C2EA5}", MyBase1);
- int dataMyDerived;
- };
- class MyDerived1
- {
- public:
- virtual ~MyDerived1() {}
- AZ_RTTI(MyDerived1, "{527B6166-1A4F-4782-8D06-F228860B1102}");
- int datatypename;
- };
- class MyDerived2
- : public MyDerived
- {
- public:
- ~MyDerived2() override {}
- AZ_RTTI(MyDerived2, "{8902C46B-61C5-4294-82A2-06CB61ACA314}", MyDerived);
- int dataMyDerived2;
- };
- class MyClassMix
- : public MyDerived2
- , public MyDerived1
- {
- public:
- ~MyClassMix() override {}
- AZ_RTTI(MyClassMix, "{F6CDCF25-3161-46AE-A46C-0F9B8A1027AF}", MyDerived2, MyDerived1);
- int dataMix;
- };
- class MyClassA
- {
- public:
- virtual ~MyClassA() {}
- AZ_RTTI(MyClassA, "{F2D44607-1BB6-4A6D-8D8B-4FDE27B488CF}");
- int dataClassA;
- };
- class MyClassB
- {
- public:
- virtual ~MyClassB() {}
- AZ_RTTI(MyClassB, "{E46477C8-4833-4F8C-A57A-02EAFA0C33D8}");
- int dataClassB;
- };
- class MyClassC
- {
- public:
- virtual ~MyClassC() {}
- AZ_RTTI(MyClassC, "{614F230F-1AD0-419D-8376-18891112F55D}");
- int dataClassC;
- };
- class MyClassD
- : public MyClassA
- {
- public:
- ~MyClassD() override {}
- AZ_RTTI(MyClassD, "{8E047831-1445-4D13-8F6F-DD36C871FD05}", MyClassA);
- int dataClassD;
- };
- class MyClassMaxMix
- : public MyDerived2
- , public MyDerived1
- , public MyClassB
- , public MyClassC
- , public MyClassD
- {
- public:
- ~MyClassMaxMix() override {}
- AZ_RTTI(MyClassMaxMix, "{49A7F45B-D039-44ED-A6BF-E500CB84E867}", MyDerived2, MyDerived1, MyClassB, MyClassC, MyClassD);
- int dataMaxMix;
- };
- TEST_F(Rtti, IsTypeOfTest)
- {
- using TypeIdArray = AZStd::vector<AZ::Uuid>;
- auto EnumTypes = [](const AZ::Uuid& id, void* userData)
- {
- TypeIdArray* idArray = reinterpret_cast<TypeIdArray*>(userData);
- idArray->push_back(id);
- };
- MyBase1 mb1;
- MyDerived md;
- MyDerived2 md2;
- MyClassMix mcm;
- MyClassMaxMix mcmm;
- EXPECT_FALSE(azrtti_istypeof<MyBase>(mb1));// MyBase has not RTTI enabled, even though it's a base class
- EXPECT_FALSE(azrtti_istypeof<MyDerived>(mb1));
- EXPECT_TRUE(azrtti_istypeof<MyBase1>(md));
- EXPECT_FALSE(azrtti_istypeof<MyBase>(md));
- EXPECT_FALSE(azrtti_istypeof<MyDerived2>(md));
- EXPECT_TRUE(azrtti_istypeof<MyDerived>(md2));
- EXPECT_TRUE(azrtti_istypeof<MyBase1>(md2));
- EXPECT_FALSE(azrtti_istypeof<MyBase>(md2));
- EXPECT_TRUE(azrtti_istypeof<MyDerived1>(mcm));
- EXPECT_TRUE(azrtti_istypeof<MyDerived2>(mcm));
- EXPECT_TRUE(azrtti_istypeof<MyDerived>(mcm));
- EXPECT_TRUE(azrtti_istypeof<MyBase1>(mcm));
- EXPECT_FALSE(azrtti_istypeof<MyBase>(mcm));
- EXPECT_TRUE(azrtti_istypeof<MyDerived1>(&mcmm));
- EXPECT_TRUE(azrtti_istypeof<MyDerived2>(mcmm));
- EXPECT_TRUE(azrtti_istypeof<MyDerived>(mcmm));
- EXPECT_TRUE(azrtti_istypeof<MyBase1>(mcmm));
- EXPECT_TRUE(azrtti_istypeof<MyClassA>(mcmm));
- EXPECT_TRUE(azrtti_istypeof<MyClassB>(mcmm));
- EXPECT_TRUE(azrtti_istypeof<MyClassC>(mcmm));
- EXPECT_TRUE(azrtti_istypeof<MyClassD>(mcmm));
- EXPECT_FALSE(azrtti_istypeof<MyBase>(mcmm));
- // type checks
- EXPECT_TRUE(azrtti_istypeof<MyBase1&>(md));
- EXPECT_TRUE(azrtti_istypeof<const MyBase1&>(md));
- EXPECT_TRUE(azrtti_istypeof<const MyBase1>(md));
- EXPECT_TRUE(azrtti_istypeof<MyBase1>(&md));
- EXPECT_TRUE(azrtti_istypeof<MyBase1&>(&md));
- EXPECT_TRUE(azrtti_istypeof<const MyBase1&>(&md));
- EXPECT_TRUE(azrtti_istypeof<const MyBase1>(&md));
- EXPECT_TRUE(azrtti_istypeof<MyBase1*>(&md));
- EXPECT_TRUE(azrtti_istypeof<MyBase1*>(md));
- EXPECT_TRUE(azrtti_istypeof<const MyBase1*>(md));
- EXPECT_TRUE(azrtti_istypeof<const MyBase1*>(&md));
- EXPECT_TRUE(azrtti_istypeof(AZ::AzTypeInfo<const MyBase1>::Uuid(), &md));
- EXPECT_TRUE(azrtti_istypeof(AZ::AzTypeInfo<MyBase1>::Uuid(), md));
- // template templates
- AZStd::vector<int> vector;
- AZStd::array<int, 1> array;
- AZStd::bitset<8> bitset;
- AZStd::function<void()> function;
- EXPECT_TRUE(azrtti_istypeof<AZStd::vector>(vector));
- EXPECT_TRUE(azrtti_istypeof<AZStd::array>(array));
- EXPECT_TRUE(azrtti_istypeof<AZStd::bitset>(bitset));
- EXPECT_FALSE(azrtti_istypeof<AZStd::vector>(mb1)); // MyBase has not RTTI enabled, even though it's a base class
- EXPECT_FALSE(azrtti_istypeof<AZStd::vector>(md));
- // check type enumeration
- TypeIdArray typeIds;
- // check a single type (no base types)
- MyDerived1::RTTI_EnumHierarchy(EnumTypes, &typeIds);
- EXPECT_EQ(1, typeIds.size());
- EXPECT_EQ(AZ::AzTypeInfo<MyDerived1>::Uuid(), typeIds[0]);
- // check a simple inheritance
- typeIds.clear();
- MyDerived::RTTI_EnumHierarchy(EnumTypes, &typeIds);
- EXPECT_EQ(2, typeIds.size());
- EXPECT_NE(typeIds.end(), AZStd::find(typeIds.begin(), typeIds.end(), AZ::AzTypeInfo<MyBase1>::Uuid()));
- EXPECT_NE(typeIds.end(), AZStd::find(typeIds.begin(), typeIds.end(), AZ::AzTypeInfo<MyDerived>::Uuid()));
- // check a little more complicated one
- typeIds.clear();
- MyClassMix::RTTI_EnumHierarchy(EnumTypes, &typeIds);
- EXPECT_EQ(5, typeIds.size());
- EXPECT_NE(typeIds.end(), AZStd::find(typeIds.begin(), typeIds.end(), AZ::AzTypeInfo<MyBase1>::Uuid()));
- EXPECT_NE(typeIds.end(), AZStd::find(typeIds.begin(), typeIds.end(), AZ::AzTypeInfo<MyDerived>::Uuid()));
- EXPECT_NE(typeIds.end(), AZStd::find(typeIds.begin(), typeIds.end(), AZ::AzTypeInfo<MyDerived1>::Uuid()));
- EXPECT_NE(typeIds.end(), AZStd::find(typeIds.begin(), typeIds.end(), AZ::AzTypeInfo<MyDerived2>::Uuid()));
- EXPECT_NE(typeIds.end(), AZStd::find(typeIds.begin(), typeIds.end(), AZ::AzTypeInfo<MyClassMix>::Uuid()));
- // now check the virtual full time selection
- MyBase1* mb1Ptr = &mcm;
- typeIds.clear();
- mb1Ptr->RTTI_EnumTypes(EnumTypes, &typeIds);
- EXPECT_EQ(5, typeIds.size());
- EXPECT_NE(typeIds.end(), AZStd::find(typeIds.begin(), typeIds.end(), AZ::AzTypeInfo<MyBase1>::Uuid()));
- EXPECT_NE(typeIds.end(), AZStd::find(typeIds.begin(), typeIds.end(), AZ::AzTypeInfo<MyDerived>::Uuid()));
- EXPECT_NE(typeIds.end(), AZStd::find(typeIds.begin(), typeIds.end(), AZ::AzTypeInfo<MyDerived1>::Uuid()));
- EXPECT_NE(typeIds.end(), AZStd::find(typeIds.begin(), typeIds.end(), AZ::AzTypeInfo<MyDerived2>::Uuid()));
- EXPECT_NE(typeIds.end(), AZStd::find(typeIds.begin(), typeIds.end(), AZ::AzTypeInfo<MyClassMix>::Uuid()));
- }
- TEST_F(Rtti, GetGenericTypeIdTest)
- {
- using IntVector = AZStd::vector<int>;
- AZ::IRttiHelper* helper = AZ::GetRttiHelper<IntVector>();
- EXPECT_EQ(azrtti_typeid<IntVector>(), helper->GetTypeId());
- EXPECT_EQ(azrtti_typeid<AZStd::vector>(), helper->GetGenericTypeId());
- EXPECT_EQ(AZ::AzTypeInfo<IntVector>::GetTemplateId(), helper->GetGenericTypeId());
- helper = AZ::GetRttiHelper<MyClassMix>();
- EXPECT_EQ(azrtti_typeid<MyClassMix>(), helper->GetTypeId());
- // MyClassMix isn't a template
- EXPECT_TRUE(helper->GetGenericTypeId().IsNull());
- // Validate that ADL lookup of Class Template Ids via the AzTypeInfo<T>::GetTemplateId() function
- // will work
- static_assert(AZ::HasGetO3deTypeName_v<IntVector>);
- static_assert(AZ::HasGetO3deTypeId_v<IntVector>);
- auto vectorTemplateId = GetO3deClassTemplateId(AZ::Adl{}, AZStd::type_identity<IntVector>{});
- EXPECT_FALSE(vectorTemplateId.IsNull());
- static_assert(AZ::HasGetO3deClassTemplateId_v<IntVector>);
- }
- class ExampleAbstractClass
- {
- public:
- AZ_RTTI(ExampleAbstractClass, "{F99EC269-3077-4984-A1B6-FA5656A65AC9}")
- virtual void AbstractFunction1() = 0;
- virtual void AbstractFunction2() = 0;
- };
- class ExampleFullImplementationClass : public ExampleAbstractClass
- {
- public:
- AZ_RTTI(ExampleFullImplementationClass, "{81B043ED-3770-414E-8B54-0F623C035926}", ExampleAbstractClass)
- void AbstractFunction1() override {}
- void AbstractFunction2() override {}
- };
- class ExamplePartialImplementationClass1
- : public ExampleAbstractClass
- {
- public:
- AZ_RTTI(ExamplePartialImplementationClass1, "{049B29D7-0414-4C5F-8FB2-589D0833121B}", ExampleAbstractClass)
- void AbstractFunction1() override {}
- };
- class ExampleCombined
- : public ExamplePartialImplementationClass1
- {
- public:
- AZ_RTTI(ExampleCombined, "{0D03E811-F8F1-4AA5-8DA2-4CD6B7FB7080}", ExamplePartialImplementationClass1)
- void AbstractFunction2() override {}
- };
- TEST_F(Rtti, IsAbstract)
- {
- // compile time proof that the two non-abstract classes are not abstract at compile time:
- [[maybe_unused]] ExampleFullImplementationClass one;
- [[maybe_unused]] ExampleCombined two;
- ASSERT_NE(AZ::GetRttiHelper<ExampleAbstractClass>(), nullptr);
- ASSERT_NE(AZ::GetRttiHelper<ExampleFullImplementationClass>(), nullptr);
- ASSERT_NE(AZ::GetRttiHelper<ExamplePartialImplementationClass1>(), nullptr);
- ASSERT_NE(AZ::GetRttiHelper<ExampleCombined>(), nullptr);
- EXPECT_TRUE(AZ::GetRttiHelper<ExampleAbstractClass>()->IsAbstract());
- EXPECT_FALSE(AZ::GetRttiHelper<ExampleFullImplementationClass>()->IsAbstract());
- EXPECT_TRUE(AZ::GetRttiHelper<ExamplePartialImplementationClass1>()->IsAbstract());
- EXPECT_FALSE(AZ::GetRttiHelper<ExampleCombined>()->IsAbstract());
- }
- TEST_F(Rtti, DynamicCastTest)
- {
- MyBase1 i_mb1;
- MyDerived i_md;
- MyDerived2 i_md2;
- MyClassMix i_mcm;
- MyClassMaxMix i_mcmm;
- MyBase1* mb1 = &i_mb1;
- MyDerived* md = &i_md;
- MyDerived2* md2 = &i_md2;
- MyClassMix* mcm = &i_mcm;
- MyClassMaxMix* mcmm = &i_mcmm;
- // downcast
- EXPECT_EQ(nullptr, azdynamic_cast<MyBase*>(mb1));// MyBase has not RTTI enabled, even though it's a base class
- EXPECT_EQ(nullptr, azdynamic_cast<MyDerived*>(mb1));
- EXPECT_NE(nullptr, azdynamic_cast<MyBase1*>(md));
- EXPECT_EQ(nullptr, azdynamic_cast<MyBase*>(md));
- EXPECT_EQ(nullptr, azdynamic_cast<MyDerived2*>(md));
- EXPECT_NE(nullptr, azdynamic_cast<MyDerived*>(md2));
- EXPECT_NE(nullptr, azdynamic_cast<MyBase1*>(md2));
- EXPECT_EQ(nullptr, azdynamic_cast<MyBase*>(md2));
- EXPECT_NE(nullptr, azdynamic_cast<MyDerived1*>(mcm));
- EXPECT_NE(nullptr, azdynamic_cast<MyDerived2*>(mcm));
- EXPECT_NE(nullptr, azdynamic_cast<MyDerived*>(mcm));
- EXPECT_NE(nullptr, azdynamic_cast<MyBase1*>(mcm));
- EXPECT_EQ(nullptr, azdynamic_cast<MyBase*>(mcm));
- EXPECT_NE(nullptr, azdynamic_cast<MyDerived1*>(mcmm));
- EXPECT_NE(nullptr, azdynamic_cast<MyDerived2*>(mcmm));
- EXPECT_NE(nullptr, azdynamic_cast<MyDerived*>(mcmm));
- EXPECT_NE(nullptr, azdynamic_cast<MyBase1*>(mcmm));
- EXPECT_NE(nullptr, azdynamic_cast<MyClassA*>(mcmm));
- EXPECT_NE(nullptr, azdynamic_cast<MyClassB*>(mcmm));
- EXPECT_NE(nullptr, azdynamic_cast<MyClassC*>(mcmm));
- EXPECT_NE(nullptr, azdynamic_cast<MyClassD*>(mcmm));
- EXPECT_EQ(nullptr, azdynamic_cast<MyBase*>(mcmm));
- // up cast
- mb1 = mcmm;
- MyClassA* mca = mcmm;
- int i_i;
- int* pi = &i_i;
- EXPECT_EQ(nullptr, azdynamic_cast<MyBase*>(nullptr));
- EXPECT_EQ(nullptr, azdynamic_cast<MyBase*>(pi));
- EXPECT_EQ(pi, azdynamic_cast<int*>(pi));
- EXPECT_NE(nullptr, azdynamic_cast<MyDerived*>(mb1));
- EXPECT_NE(nullptr, azdynamic_cast<MyDerived2*>(mb1));
- EXPECT_NE(nullptr, azdynamic_cast<MyClassMaxMix*>(mb1));
- EXPECT_NE(nullptr, azdynamic_cast<MyClassD*>(mca));
- EXPECT_NE(nullptr, azdynamic_cast<MyClassMaxMix*>(mca));
- // type checks
- const MyDerived* cmd = md;
- EXPECT_NE(nullptr, azdynamic_cast<const MyBase1*>(md));
- EXPECT_NE(nullptr, azdynamic_cast<const volatile MyBase1*>(md));
- EXPECT_NE(nullptr, azdynamic_cast<const MyBase1*>(cmd));
- EXPECT_NE(nullptr, azdynamic_cast<const volatile MyBase1*>(cmd));
- // unrelated cast not supported (we can, but why)
- //AZ_TEST_ASSERT(azdynamic_cast<MyBase1*>(mca));
- md = mcmm;
- // serialization helpers
- EXPECT_EQ(mcmm, mca->RTTI_AddressOf(AZ::AzTypeInfo<MyClassMaxMix>::Uuid()));
- EXPECT_EQ(mcmm, mb1->RTTI_AddressOf(AZ::AzTypeInfo<MyClassMaxMix>::Uuid()));
- EXPECT_EQ(mca, mb1->RTTI_AddressOf(AZ::AzTypeInfo<MyClassA>::Uuid()));
- EXPECT_EQ(md, mb1->RTTI_AddressOf(AZ::AzTypeInfo<MyDerived>::Uuid()));
- EXPECT_EQ(nullptr, md2->RTTI_AddressOf(AZ::AzTypeInfo<MyClassA>::Uuid()));
- EXPECT_EQ(mca, mcmm->RTTI_AddressOf(AZ::AzTypeInfo<MyClassA>::Uuid()));
- EXPECT_EQ(mb1, mcmm->RTTI_AddressOf(AZ::AzTypeInfo<MyBase1>::Uuid()));
- // typeid
- EXPECT_EQ(AZ::AzTypeInfo<MyBase>::Uuid(), azrtti_typeid<MyBase>());
- EXPECT_EQ(AZ::AzTypeInfo<MyBase1>::Uuid(), azrtti_typeid(i_mb1));
- EXPECT_EQ(AZ::AzTypeInfo<MyDerived2>::Uuid(), azrtti_typeid(md2));
- EXPECT_EQ(AZ::AzTypeInfo<MyClassMaxMix>::Uuid(), azrtti_typeid(mca));
- MyClassA& mcar = i_mcmm;
- EXPECT_EQ(AZ::AzTypeInfo<MyClassMaxMix>::Uuid(), azrtti_typeid(mcar));
- EXPECT_EQ(AZ::AzTypeInfo<int>::Uuid(), azrtti_typeid<int>());
- }
- // Struct which uses the Curiously Recuring Template Pattern to provide static
- // polymorphism
- template <class Derived>
- struct RttiVisitorCRTP
- {
- virtual ~RttiVisitorCRTP() = default;
- AZ_RTTI((RttiVisitorCRTP, "{07750558-F992-4F19-A31F-8BF7A2B41AF0}", Derived));
- };
- template <class T>
- struct RttiDerivedVisitor;
- AZ_TYPE_INFO_TEMPLATE(RttiDerivedVisitor, "{484C63B4-A9AE-4E82-90A0-C863E9411604}", AZ_TYPE_INFO_CLASS);
- template <class T>
- struct RttiDerivedVisitor
- : RttiVisitorCRTP<RttiDerivedVisitor<T>>
- {
- using BaseType = RttiVisitorCRTP<RttiDerivedVisitor<T>>;
- // Provide declaration for RTTI functions
- AZ_RTTI_NO_TYPE_INFO_DECL();
- };
- // Provide the implemetnation for RTTI functions
- AZ_RTTI_NO_TYPE_INFO_IMPL((RttiDerivedVisitor, AZ_TYPE_INFO_CLASS), (RttiVisitorCRTP<RttiDerivedVisitor<T1>>));
- TEST_F(Rtti, Rtti_DeclAndImpl_CanWorkaroundCRTP_WithAzTypeInfo_Instantiation)
- {
- AZStd::unique_ptr<RttiVisitorCRTP<RttiDerivedVisitor<int>>> basePtr = AZStd::make_unique<RttiDerivedVisitor<int>>();
- // Check if the instance is of the derived visitor type
- EXPECT_TRUE(basePtr->RTTI_IsTypeOf(azrtti_typeid<RttiDerivedVisitor<int>>()));
- // Check if the instance is also of the derived base class type
- EXPECT_TRUE(basePtr->RTTI_IsTypeOf(azrtti_typeid<RttiVisitorCRTP<RttiDerivedVisitor<int>>>()));
- }
- TEST_F(Rtti, MultiThreadedTypeInfo)
- {
- // These must be AZ::Uuids so that they don't engage the AZ::UuidHolder code
- const AZ::Uuid expectedMtti("{4876C017-0C26-4D0D-9A1F-2A738BAE6449}");
- const AZ::Uuid expectedMtti2("{CBC94693-5ECD-4CBF-A8DB-9B122E697E8D}");
- // Create 2x of each of these threads which are doing RTTI ops and
- // let the scheduler run them at random. This is attempting to crash
- // them into each other as best we can
- auto threadFunc1 = [&expectedMtti, &expectedMtti2]()
- {
- AZStd::this_thread::sleep_for(AZStd::chrono::milliseconds(1));
- const AZ::TypeId& mtti = azrtti_typeid<MTTI>();
- const AZ::TypeId& mtti2 = azrtti_typeid<MTTI2>();
- EXPECT_FALSE(mtti.IsNull());
- EXPECT_EQ(expectedMtti, mtti);
- EXPECT_FALSE(mtti2.IsNull());
- EXPECT_EQ(expectedMtti2, mtti2);
- };
- auto threadFunc2 = []()
- {
- AZStd::this_thread::sleep_for(AZStd::chrono::milliseconds(1));
- MTTI* mtti = new MTTI();
- bool castSucceeded = (azrtti_cast<MTTI2*>(mtti) != nullptr);
- EXPECT_FALSE(castSucceeded);
- delete mtti;
- };
- auto threadFunc3 = []()
- {
- AZStd::this_thread::sleep_for(AZStd::chrono::milliseconds(1));
- MTTI2* mtti2 = new MTTI2();
- bool castSucceeded = (azrtti_cast<MTTI*>(mtti2) != nullptr);
- EXPECT_FALSE(castSucceeded);
- delete mtti2;
- };
- auto threadFunc4 = []()
- {
- AZStd::this_thread::sleep_for(AZStd::chrono::milliseconds(1));
- MTTI* mtti = new MTTI();
- bool castSucceeded = (azrtti_cast<MTTI*>(mtti) != nullptr);
- EXPECT_TRUE(castSucceeded);
- delete mtti;
- };
- auto threadFunc5 = []()
- {
- AZStd::this_thread::sleep_for(AZStd::chrono::milliseconds(1));
- MTTI2* mtti2 = new MTTI2();
- bool castSucceeded = (azrtti_cast<MTTI2*>(mtti2) != nullptr);
- EXPECT_TRUE(castSucceeded);
- delete mtti2;
- };
- AZStd::fixed_vector<AZStd::function<void()>, 5> threadFuncs({ threadFunc1, threadFunc2, threadFunc3, threadFunc4, threadFunc5 });
- AZStd::thread threads[10];
- for (size_t threadIdx = 0; threadIdx < AZ_ARRAY_SIZE(threads); ++threadIdx)
- {
- auto threadFunc = threadFuncs[threadIdx % threadFuncs.size()];
- threads[threadIdx] = AZStd::thread(threadFunc);
- }
- for (auto& thread : threads)
- {
- thread.join();
- }
- }
- static void ExternalRttiEnumHeirarchyHelper(const AZ::TypeId&, void* userData)
- {
- auto totalClassesEnumerated = reinterpret_cast<size_t*>(userData);
- ++*totalClassesEnumerated;
- }
- class MyBaseExternal
- {
- public:
- AZ_TYPE_INFO(MyBaseExternal, "{F0F36BB2-14E6-4C44-B3D5-E0CBFD783C99}");
- int32_t m_intValue;
- };
- class MyDerivedExternal
- : public MyBaseExternal
- {
- public:
- AZ_TYPE_INFO(MyDerivedExternal, "{FFD1C3B7-7957-4270-BF10-700CE8BE2B53}");
- float m_floatValue;
- };
- class MyConvertibleExternal
- {
- public:
- AZ_TYPE_INFO(MyConvertibleExternal, "{3962F510-309B-4E32-8CE5-6DEE85F351A9}");
- MyConvertibleExternal() = default;
- MyConvertibleExternal(const MyBaseExternal& baseExternal)
- : m_baseExternal(baseExternal)
- {
- }
- operator MyBaseExternal() const
- {
- return m_baseExternal;
- }
- MyBaseExternal m_baseExternal;
- };
- class MyBaseIntrusive
- {
- public:
- AZ_RTTI(MyBaseIntrusive, "{06D41B30-CEDB-46C9-BD98-B8672A04F71F}");
- virtual ~MyBaseIntrusive() = default;
- uint64_t m_uintValue;
- };
- class MyDerivedIntrusive
- : public MyBaseIntrusive
- {
- public:
- AZ_RTTI(MyDerivedIntrusive, "{6F3FA2A5-CD05-424F-8E37-1DEDA7CE8816}", MyBaseIntrusive);
- ~MyDerivedIntrusive() override = default;
- double m_doubleValue;
- };
- class MyExternalDerivedFromExternalAndIntrusive
- : public MyDerivedExternal
- , public MyDerivedIntrusive
- {
- public:
- AZ_TYPE_INFO(MyExternalDerivedFromExternalAndIntrusive, "{79DC295D-98C5-4FEB-9DC0-0AC3D5A91855}");
- };
- }
- namespace AZ
- {
- AZ_EXTERNAL_RTTI_SPECIALIZE(UnitTest::MyBaseExternal);
- AZ_EXTERNAL_RTTI_SPECIALIZE(UnitTest::MyDerivedExternal, UnitTest::MyBaseExternal);
- AZ_EXTERNAL_RTTI_SPECIALIZE(UnitTest::MyExternalDerivedFromExternalAndIntrusive, UnitTest::MyDerivedExternal, UnitTest::MyDerivedIntrusive);
- AZ_EXTERNAL_RTTI_SPECIALIZE(UnitTest::MyConvertibleExternal, UnitTest::MyBaseExternal);
- }
- namespace UnitTest
- {
- class MyIntrusiveDerivedFromExternalAndIntrusive
- : public MyDerivedExternal
- , public MyDerivedIntrusive
- {
- public:
- AZ_RTTI(MyIntrusiveDerivedFromExternalAndIntrusive, "{3822CF8D-6AC7-4B71-B755-5C69B9DF5A3C}", MyDerivedExternal, MyDerivedIntrusive);
- ~MyIntrusiveDerivedFromExternalAndIntrusive() override = default;
- };
- TEST_F(Rtti, ExternalRtti)
- {
- MyBaseExternal baseInstance{ 7 };
- MyDerivedExternal derivedInstance;
- derivedInstance.m_intValue = 15;
- derivedInstance.m_floatValue = 0.0f;
- MyConvertibleExternal convertibleInstance(MyBaseExternal{ 24 });
- MyExternalDerivedFromExternalAndIntrusive externalDerivedFromExternalAndIntrusiveInstance;
- externalDerivedFromExternalAndIntrusiveInstance.m_intValue = -1;
- externalDerivedFromExternalAndIntrusiveInstance.m_uintValue = 2;
- externalDerivedFromExternalAndIntrusiveInstance.m_floatValue = 2.0f;
- externalDerivedFromExternalAndIntrusiveInstance.m_doubleValue = -32.0;
- MyIntrusiveDerivedFromExternalAndIntrusive intrusiveDerivedFromExternalAndIntrusiveInstance;
- intrusiveDerivedFromExternalAndIntrusiveInstance.m_intValue = -55;
- intrusiveDerivedFromExternalAndIntrusiveInstance.m_uintValue = 256;
- intrusiveDerivedFromExternalAndIntrusiveInstance.m_floatValue = -1023.0f;
- intrusiveDerivedFromExternalAndIntrusiveInstance.m_doubleValue = .0223;
- AZ::IRttiHelper* baseExternal = AZ::GetRttiHelper<MyBaseExternal>();
- AZ::IRttiHelper* derivedExternal = AZ::GetRttiHelper<MyDerivedExternal>();
- AZ::IRttiHelper* convertibleExternal = AZ::GetRttiHelper<MyConvertibleExternal>();
- AZ::IRttiHelper* externalDerivedFromExternalAndIntrusive = AZ::GetRttiHelper<MyExternalDerivedFromExternalAndIntrusive>();
- AZ::IRttiHelper* intrusiveDerivedFromExternalAndIntrusive = AZ::GetRttiHelper<MyIntrusiveDerivedFromExternalAndIntrusive>();
- ASSERT_NE(nullptr, baseExternal);
- ASSERT_NE(nullptr, derivedExternal);
- ASSERT_NE(nullptr, convertibleExternal);
- ASSERT_NE(nullptr, externalDerivedFromExternalAndIntrusive);
- ASSERT_NE(nullptr, intrusiveDerivedFromExternalAndIntrusive);
- // Base Class External RTTI
- {
- EXPECT_EQ(AZ::AzTypeInfo<MyBaseExternal>::Uuid(), baseExternal->GetTypeId());
- EXPECT_TRUE(baseExternal->IsTypeOf(AZ::AzTypeInfo<MyBaseExternal>::Uuid()));
- size_t enumHierarchyTotalClasses{};
- baseExternal->EnumHierarchy(&ExternalRttiEnumHeirarchyHelper, &enumHierarchyTotalClasses);
- // MyBaseExternal has no other base classes so this count should be 1
- EXPECT_EQ(1, enumHierarchyTotalClasses);
- }
- // Derived Class External RTTI
- {
- EXPECT_EQ(AZ::AzTypeInfo<MyDerivedExternal>::Uuid(), derivedExternal->GetTypeId());
- EXPECT_TRUE(derivedExternal->IsTypeOf(AZ::AzTypeInfo<MyDerivedExternal>::Uuid()));
- EXPECT_TRUE(derivedExternal->IsTypeOf(AZ::AzTypeInfo<MyBaseExternal>::Uuid()));
- size_t enumHierarchyTotalClasses{};
- derivedExternal->EnumHierarchy(&ExternalRttiEnumHeirarchyHelper, &enumHierarchyTotalClasses);
- // MyDerivedExternal has MyBaseExternal as a base classes so this count should be 2
- EXPECT_EQ(2, enumHierarchyTotalClasses);
- // MyDerivedExternal -> MyDerivedExternal - succeeds
- EXPECT_NE(nullptr, derivedExternal->Cast(&derivedInstance, AZ::AzTypeInfo<MyDerivedExternal>::Uuid()));
- // MyDerivedExternal -> MyBaseExternal - succeeds
- EXPECT_NE(nullptr, derivedExternal->Cast(&derivedInstance, AZ::AzTypeInfo<MyBaseExternal>::Uuid()));
- // MyBaseExternal -> MyDerivedExternal - fails
- EXPECT_EQ(nullptr, baseExternal->Cast<MyDerivedExternal>(&baseInstance));
- // MyBaseExternal -> MyBaseExternal(using derived class RttiHelper)- succeeds
- EXPECT_NE(nullptr, derivedExternal->Cast(&baseInstance, AZ::AzTypeInfo<MyBaseExternal>::Uuid()));
- // MyBaseExternal -> MyBaseExternal(using RttiCast function which must lookup RTTI information from the derived instance)- fails
- // The reason why this fails is because the instance data does not have RTTI on it so it must lookup using
- // using the supplied template type id
- EXPECT_NE(nullptr, AZ::RttiCast<MyBaseExternal*>(&derivedInstance));
- }
- // Convertible Class External RTTI
- {
- EXPECT_EQ(AZ::AzTypeInfo<MyConvertibleExternal>::Uuid(), convertibleExternal->GetTypeId());
- EXPECT_TRUE(convertibleExternal->IsTypeOf(AZ::AzTypeInfo<MyConvertibleExternal>::Uuid()));
- EXPECT_TRUE(convertibleExternal->IsTypeOf(AZ::AzTypeInfo<MyBaseExternal>::Uuid()));
- size_t enumHierarchyTotalClasses{};
- convertibleExternal->EnumHierarchy(&ExternalRttiEnumHeirarchyHelper, &enumHierarchyTotalClasses);
- // MyConvertibleExternal specifies "MyBaseExternal" as a base classes even though it really is not,
- // but EnumHierarchy should still enumerate for the "MyBaseExternal" typeid. Therefore the count should be 2
- EXPECT_EQ(2, enumHierarchyTotalClasses);
- // MyConvertibleExternal -> MyConvertibleExternal - succeeds
- EXPECT_NE(nullptr, convertibleExternal->Cast(&convertibleInstance, AZ::AzTypeInfo<MyConvertibleExternal>::Uuid()));
- // MyConvertibleExternal -> MyBaseExternal - succeeds
- EXPECT_NE(nullptr, convertibleExternal->Cast(&convertibleInstance, AZ::AzTypeInfo<MyBaseExternal>::Uuid()));
- // MyBaseExternal -> MyConvertibleExternal - fails
- EXPECT_EQ(nullptr, baseExternal->Cast<MyConvertibleExternal>(&baseInstance));
- // MyBaseExternal -> MyBaseExternal(using convertible class RttiHelper)- succeeds
- EXPECT_NE(nullptr, convertibleExternal->Cast(&baseInstance, AZ::AzTypeInfo<MyBaseExternal>::Uuid()));
- // MyBaseExternal -> MyBaseExternal(using RttiCast function which must lookup RTTI information from the derived instance)- succeeds
- EXPECT_NE(nullptr, AZ::RttiCast<MyBaseExternal*>(&derivedInstance));
- }
- // Derived class with External RTTI which inherits from a class with external RTTI and intrusive RTTI
- {
- EXPECT_EQ(AZ::AzTypeInfo<MyExternalDerivedFromExternalAndIntrusive>::Uuid(), externalDerivedFromExternalAndIntrusive->GetTypeId());
- EXPECT_TRUE(externalDerivedFromExternalAndIntrusive->IsTypeOf(AZ::AzTypeInfo<MyExternalDerivedFromExternalAndIntrusive>::Uuid()));
- EXPECT_TRUE(externalDerivedFromExternalAndIntrusive->IsTypeOf(AZ::AzTypeInfo<MyDerivedExternal>::Uuid()));
- EXPECT_TRUE(externalDerivedFromExternalAndIntrusive->IsTypeOf(AZ::AzTypeInfo<MyDerivedIntrusive>::Uuid()));
- EXPECT_TRUE(externalDerivedFromExternalAndIntrusive->IsTypeOf(AZ::AzTypeInfo<MyBaseExternal>::Uuid()));
- EXPECT_TRUE(externalDerivedFromExternalAndIntrusive->IsTypeOf(AZ::AzTypeInfo<MyBaseIntrusive>::Uuid()));
- size_t enumHierarchyTotalClasses{};
- externalDerivedFromExternalAndIntrusive->EnumHierarchy(&ExternalRttiEnumHeirarchyHelper, &enumHierarchyTotalClasses);
- // MyDerivedFromExternalAndIntrusive inherits from MyDerivedExternal which has one base class with external RTTI.
- // This adds 2 to the enumeration count.
- // MyDerivedFromExternalAndIntrusive also inherits from MyDerivedIntrusive which has one base with intrusive RTTI
- // This adds 2 more the enumeration count. Combining these counts with the one for this class the count value should be 5
- EXPECT_EQ(5, enumHierarchyTotalClasses);
- // MyDerivedFromExternalAndIntrusive -> MyDerivedFromExternalAndIntrusive - succeeds
- EXPECT_NE(nullptr, externalDerivedFromExternalAndIntrusive->Cast(&externalDerivedFromExternalAndIntrusiveInstance, AZ::AzTypeInfo<MyExternalDerivedFromExternalAndIntrusive>::Uuid()));
- // MyDerivedFromExternalAndIntrusive -> MyDerivedExternal - succeeds
- EXPECT_NE(nullptr, externalDerivedFromExternalAndIntrusive->Cast(&externalDerivedFromExternalAndIntrusiveInstance, AZ::AzTypeInfo<MyDerivedExternal>::Uuid()));
- // MyDerivedFromExternalAndIntrusive -> MyBaseExternal - succeeds
- EXPECT_NE(nullptr, externalDerivedFromExternalAndIntrusive->Cast(&externalDerivedFromExternalAndIntrusiveInstance, AZ::AzTypeInfo<MyBaseExternal>::Uuid()));
- // MyDerivedFromExternalAndIntrusive -> MyDerivedIntrusive- succeeds
- MyDerivedIntrusive* castedDerivedIntrusiveInstance = externalDerivedFromExternalAndIntrusive->Cast<MyDerivedIntrusive>(&externalDerivedFromExternalAndIntrusiveInstance);
- ASSERT_NE(nullptr, castedDerivedIntrusiveInstance);
- EXPECT_DOUBLE_EQ(-32.0, castedDerivedIntrusiveInstance->m_doubleValue);
- castedDerivedIntrusiveInstance->m_doubleValue = -64.0; // Verify that access doesn't crash due to invalid memory address
- // MyDerivedFromExternalAndIntrusive -> MyBaseIntrusive- succeeds
- MyBaseIntrusive* castedBaseIntrusiveInstance = externalDerivedFromExternalAndIntrusive->Cast<MyBaseIntrusive>(&externalDerivedFromExternalAndIntrusiveInstance);
- ASSERT_NE(nullptr, castedBaseIntrusiveInstance);
- EXPECT_EQ(2U, castedBaseIntrusiveInstance->m_uintValue);
- castedDerivedIntrusiveInstance->m_uintValue = 4U;
- // MyDerivedExternal -> MyDerivedFromExternalAndIntrusive - fails
- EXPECT_EQ(nullptr, derivedExternal->Cast<MyExternalDerivedFromExternalAndIntrusive>(&derivedInstance));
- // MyBaseExternal -> MyDerivedFromExternalAndIntrusive - fails
- EXPECT_EQ(nullptr, baseExternal->Cast<MyExternalDerivedFromExternalAndIntrusive>(&baseInstance));
- // MyBaseExternal -> MyBaseExternal(using externalDerivedFromExternalAndIntrusive class RttiHelper)- succeeds
- EXPECT_NE(nullptr, externalDerivedFromExternalAndIntrusive->Cast(&baseInstance, AZ::AzTypeInfo<MyBaseExternal>::Uuid()));
- // MyDerivedExternal -> MyBaseExternal(using externalDerivedFromExternalAndIntrusive class RttiHelper)- succeeds
- EXPECT_NE(nullptr, externalDerivedFromExternalAndIntrusive->Cast(&derivedInstance, AZ::AzTypeInfo<MyBaseExternal>::Uuid()));
- // MyBaseIntrusive -> MyBaseIntrusive(using externalDerivedFromExternalAndIntrusive class RttiHelper)- succeeds
- MyBaseIntrusive baseIntrusiveInstance;
- baseIntrusiveInstance.m_uintValue = 3456893U;
- EXPECT_NE(nullptr, externalDerivedFromExternalAndIntrusive->Cast(&baseIntrusiveInstance, AZ::AzTypeInfo<MyBaseIntrusive>::Uuid()));
- // MyDerivedIntrusive-> MyBaseIntrusive(using externalDerivedFromExternalAndIntrusive class RttiHelper)- succeeds
- MyDerivedIntrusive derivedIntrusiveInstance;
- derivedIntrusiveInstance.m_uintValue = 1700U;
- derivedIntrusiveInstance.m_doubleValue = 24.0f;
- EXPECT_NE(nullptr, externalDerivedFromExternalAndIntrusive->Cast(&derivedIntrusiveInstance, AZ::AzTypeInfo<MyBaseIntrusive>::Uuid()));
- // Test Rtti Free functions for External class with external Rtti
- enumHierarchyTotalClasses = 0;
- AZ::RttiEnumHierarchy<MyExternalDerivedFromExternalAndIntrusive>(&ExternalRttiEnumHeirarchyHelper, &enumHierarchyTotalClasses);
- EXPECT_EQ(5, enumHierarchyTotalClasses);
- // This should fail
- EXPECT_EQ(nullptr, AZ::RttiCast<MyExternalDerivedFromExternalAndIntrusive*>(&derivedIntrusiveInstance));
- EXPECT_NE(nullptr, AZ::RttiCast<MyExternalDerivedFromExternalAndIntrusive*>(&externalDerivedFromExternalAndIntrusiveInstance));
- void* baseIntrusiveAddress = AZ::RttiAddressOf(&externalDerivedFromExternalAndIntrusiveInstance, AZ::AzTypeInfo<MyBaseIntrusive>::Uuid());
- ASSERT_NE(nullptr, baseIntrusiveAddress);
- EXPECT_EQ(4U, static_cast<MyBaseIntrusive*>(baseIntrusiveAddress)->m_uintValue);
- EXPECT_FALSE(AZ::RttiIsTypeOf(AZ::AzTypeInfo<MyConvertibleExternal>::Uuid(), externalDerivedFromExternalAndIntrusiveInstance));
- EXPECT_FALSE(AZ::RttiIsTypeOf<MyConvertibleExternal>(externalDerivedFromExternalAndIntrusiveInstance));
- EXPECT_TRUE(AZ::RttiIsTypeOf(AZ::AzTypeInfo<MyDerivedIntrusive>::Uuid(), externalDerivedFromExternalAndIntrusiveInstance));
- EXPECT_TRUE(AZ::RttiIsTypeOf<MyDerivedExternal>(externalDerivedFromExternalAndIntrusiveInstance));
- // Check pointer case template specializations for RttiIsTypeOf
- EXPECT_TRUE(AZ::RttiIsTypeOf(AZ::AzTypeInfo<MyBaseExternal>::Uuid(), &externalDerivedFromExternalAndIntrusiveInstance));
- EXPECT_TRUE(AZ::RttiIsTypeOf<MyBaseExternal>(&externalDerivedFromExternalAndIntrusiveInstance));
- EXPECT_EQ(AZ::AzTypeInfo<MyExternalDerivedFromExternalAndIntrusive>::Uuid(), AZ::RttiTypeId(externalDerivedFromExternalAndIntrusiveInstance));
- // Check pointer case template specializations for RttiTypeId
- EXPECT_EQ(AZ::AzTypeInfo<MyExternalDerivedFromExternalAndIntrusive>::Uuid(), AZ::RttiTypeId(&externalDerivedFromExternalAndIntrusiveInstance));
- }
- // Derived class with Intrusive RTTI which inherits from a class with external RTTI and intrusive RTTI
- {
- EXPECT_EQ(AZ::AzTypeInfo<MyIntrusiveDerivedFromExternalAndIntrusive>::Uuid(), intrusiveDerivedFromExternalAndIntrusive->GetTypeId());
- EXPECT_TRUE(intrusiveDerivedFromExternalAndIntrusive->IsTypeOf(AZ::AzTypeInfo<MyIntrusiveDerivedFromExternalAndIntrusive>::Uuid()));
- EXPECT_TRUE(intrusiveDerivedFromExternalAndIntrusive->IsTypeOf(AZ::AzTypeInfo<MyDerivedExternal>::Uuid()));
- EXPECT_TRUE(intrusiveDerivedFromExternalAndIntrusive->IsTypeOf(AZ::AzTypeInfo<MyDerivedIntrusive>::Uuid()));
- EXPECT_TRUE(intrusiveDerivedFromExternalAndIntrusive->IsTypeOf(AZ::AzTypeInfo<MyBaseExternal>::Uuid()));
- EXPECT_TRUE(intrusiveDerivedFromExternalAndIntrusive->IsTypeOf(AZ::AzTypeInfo<MyBaseIntrusive>::Uuid()));
- size_t enumHierarchyTotalClasses{};
- intrusiveDerivedFromExternalAndIntrusive->EnumHierarchy(&ExternalRttiEnumHeirarchyHelper, &enumHierarchyTotalClasses);
- // MyIntrusiveDerivedFromExternalAndIntrusive inherits from MyDerivedExternal which has one base class with intrusive RTTI.
- // This adds 2 to the enumeration count.
- // MyIntrusiveDerivedFromExternalAndIntrusive also inherits from MyDerivedIntrusive which has one base with intrusive RTTI
- // This adds 2 more the enumeration count. Combining these counts with the one for this class the count value should be 5
- EXPECT_EQ(5, enumHierarchyTotalClasses);
- // MyIntrusiveDerivedFromExternalAndIntrusive -> MyIntrusiveDerivedFromExternalAndIntrusive - succeeds
- EXPECT_NE(nullptr, intrusiveDerivedFromExternalAndIntrusive->Cast(&intrusiveDerivedFromExternalAndIntrusiveInstance, AZ::AzTypeInfo<MyIntrusiveDerivedFromExternalAndIntrusive>::Uuid()));
- // MyIntrusiveDerivedFromExternalAndIntrusive -> MyDerivedExternal - succeeds
- EXPECT_NE(nullptr, intrusiveDerivedFromExternalAndIntrusive->Cast(&intrusiveDerivedFromExternalAndIntrusiveInstance, AZ::AzTypeInfo<MyDerivedExternal>::Uuid()));
- // MyIntrusiveDerivedFromExternalAndIntrusive -> MyBaseExternal - succeeds
- EXPECT_NE(nullptr, intrusiveDerivedFromExternalAndIntrusive->Cast(&intrusiveDerivedFromExternalAndIntrusiveInstance, AZ::AzTypeInfo<MyBaseExternal>::Uuid()));
- // MyIntrusiveDerivedFromExternalAndIntrusive -> MyDerivedIntrusive- succeeds
- MyDerivedIntrusive* castedDerivedIntrusiveInstance = intrusiveDerivedFromExternalAndIntrusive->Cast<MyDerivedIntrusive>(&intrusiveDerivedFromExternalAndIntrusiveInstance);
- ASSERT_NE(nullptr, castedDerivedIntrusiveInstance);
- EXPECT_DOUBLE_EQ(.0223, castedDerivedIntrusiveInstance->m_doubleValue);
- castedDerivedIntrusiveInstance->m_doubleValue = -64.0; // Verify that access doesn't crash due to invalid memory address
- // MyIntrusiveDerivedFromExternalAndIntrusive -> MyBaseIntrusive- succeeds
- MyBaseIntrusive* castedBaseIntrusiveInstance = intrusiveDerivedFromExternalAndIntrusive->Cast<MyBaseIntrusive>(&intrusiveDerivedFromExternalAndIntrusiveInstance);
- ASSERT_NE(nullptr, castedBaseIntrusiveInstance);
- EXPECT_EQ(256U, castedBaseIntrusiveInstance->m_uintValue);
- castedDerivedIntrusiveInstance->m_uintValue = 71U;
- // MyDerivedExternal -> MyIntrusiveDerivedFromExternalAndIntrusive - fails
- EXPECT_EQ(nullptr, derivedExternal->Cast<MyIntrusiveDerivedFromExternalAndIntrusive>(&derivedInstance));
- // MyBaseExternal -> MyIntrusiveDerivedFromExternalAndIntrusive - fails
- EXPECT_EQ(nullptr, baseExternal->Cast<MyIntrusiveDerivedFromExternalAndIntrusive>(&baseInstance));
- // MyBaseIntrusive -> MyBaseIntrusive(using intrusiveDerivedFromExternalAndIntrusive class RttiHelper)- succeeds
- MyBaseIntrusive baseIntrusiveInstance;
- baseIntrusiveInstance.m_uintValue = 3456893U;
- EXPECT_NE(nullptr, intrusiveDerivedFromExternalAndIntrusive->Cast(&baseIntrusiveInstance, AZ::AzTypeInfo<MyBaseIntrusive>::Uuid()));
- // MyDerivedIntrusive-> MyBaseIntrusive(using intrusiveDerivedFromExternalAndIntrusive class RttiHelper)- succeeds
- MyDerivedIntrusive derivedIntrusiveInstance;
- derivedIntrusiveInstance.m_uintValue = 1700U;
- derivedIntrusiveInstance.m_doubleValue = 24.0f;
- EXPECT_NE(nullptr, intrusiveDerivedFromExternalAndIntrusive->Cast(&derivedIntrusiveInstance, AZ::AzTypeInfo<MyBaseIntrusive>::Uuid()));
- // Test Rtti Free functions for class with intrusive Rtti
- enumHierarchyTotalClasses = 0;
- AZ::RttiEnumHierarchy<MyIntrusiveDerivedFromExternalAndIntrusive>(&ExternalRttiEnumHeirarchyHelper, &enumHierarchyTotalClasses);
- EXPECT_EQ(5, enumHierarchyTotalClasses);
- // This should fail
- EXPECT_EQ(nullptr, AZ::RttiCast<MyIntrusiveDerivedFromExternalAndIntrusive*>(&derivedIntrusiveInstance));
- EXPECT_NE(nullptr, AZ::RttiCast<MyIntrusiveDerivedFromExternalAndIntrusive*>(&intrusiveDerivedFromExternalAndIntrusiveInstance));
- void* baseIntrusiveAddress = AZ::RttiAddressOf(&intrusiveDerivedFromExternalAndIntrusiveInstance, AZ::AzTypeInfo<MyBaseIntrusive>::Uuid());
- ASSERT_NE(nullptr, baseIntrusiveAddress);
- EXPECT_EQ(71U, static_cast<MyBaseIntrusive*>(baseIntrusiveAddress)->m_uintValue);
- EXPECT_FALSE(AZ::RttiIsTypeOf(AZ::AzTypeInfo<MyConvertibleExternal>::Uuid(), intrusiveDerivedFromExternalAndIntrusiveInstance));
- EXPECT_FALSE(AZ::RttiIsTypeOf<MyConvertibleExternal>(intrusiveDerivedFromExternalAndIntrusiveInstance));
- EXPECT_TRUE(AZ::RttiIsTypeOf(AZ::AzTypeInfo<MyDerivedIntrusive>::Uuid(), intrusiveDerivedFromExternalAndIntrusiveInstance));
- EXPECT_TRUE(AZ::RttiIsTypeOf<MyDerivedExternal>(intrusiveDerivedFromExternalAndIntrusiveInstance));
- // Check pointer case template specializations for RttiIsTypeOf
- EXPECT_TRUE(AZ::RttiIsTypeOf(AZ::AzTypeInfo<MyBaseExternal>::Uuid(), &intrusiveDerivedFromExternalAndIntrusiveInstance));
- EXPECT_TRUE(AZ::RttiIsTypeOf<MyBaseExternal>(&intrusiveDerivedFromExternalAndIntrusiveInstance));
- EXPECT_EQ(AZ::AzTypeInfo<MyIntrusiveDerivedFromExternalAndIntrusive>::Uuid(), AZ::RttiTypeId(intrusiveDerivedFromExternalAndIntrusiveInstance));
- // Check pointer case template specializations for RttiTypeId
- EXPECT_EQ(AZ::AzTypeInfo<MyIntrusiveDerivedFromExternalAndIntrusive>::Uuid(), AZ::RttiTypeId(&intrusiveDerivedFromExternalAndIntrusiveInstance));
- }
- }
- TEST_F(Rtti, ExternalRttiStoresTypeTraits)
- {
- AZ::IRttiHelper* externalRtti = AZ::GetRttiHelper<UnitTest::MyExternalDerivedFromExternalAndIntrusive>();
- ASSERT_NE(nullptr, externalRtti);
- EXPECT_NE(AZ::TypeTraits::is_signed, externalRtti->GetTypeTraits() & AZ::TypeTraits::is_signed);
- EXPECT_NE(AZ::TypeTraits::is_unsigned, externalRtti->GetTypeTraits() & AZ::TypeTraits::is_unsigned);
- }
- TEST_F(Rtti, InternalRttiStoresTypeTraits)
- {
- AZ::IRttiHelper* internalRtti = AZ::GetRttiHelper<UnitTest::ExampleCombined>();
- ASSERT_NE(nullptr, internalRtti);
- EXPECT_NE(AZ::TypeTraits::is_signed, internalRtti->GetTypeTraits() & AZ::TypeTraits::is_signed);
- EXPECT_NE(AZ::TypeTraits::is_unsigned, internalRtti->GetTypeTraits() & AZ::TypeTraits::is_unsigned);
- }
- enum TestEnumWithTypeInfo : uint16_t
- {};
- }
- namespace AZ
- {
- AZ_TYPE_INFO_SPECIALIZE(UnitTest::TestEnumWithTypeInfo, "{6C2F6697-4E32-4E54-8A9E-AF2FB3F77C69}");
- }
- namespace UnitTest
- {
- TEST_F(Rtti, TypeInfoStoresTypeTraits)
- {
- AZ::IRttiHelper* internalRtti = AZ::GetRttiHelper<int>();
- ASSERT_NE(nullptr, internalRtti);
- EXPECT_EQ(AZ::TypeTraits::is_signed, internalRtti->GetTypeTraits() & AZ::TypeTraits::is_signed);
- EXPECT_NE(AZ::TypeTraits::is_unsigned, internalRtti->GetTypeTraits() & AZ::TypeTraits::is_unsigned);
- }
- class ReflectionManagerTest
- : public LeakDetectionFixture
- {
- public:
- void SetUp() override
- {
- LeakDetectionFixture::SetUp();
- m_reflection = AZStd::make_unique<AZ::ReflectionManager>();
- }
- void TearDown() override
- {
- m_reflection.reset();
- LeakDetectionFixture::TearDown();
- }
- protected:
- AZStd::unique_ptr<AZ::ReflectionManager> m_reflection;
- };
- class TestReflectedClass
- {
- public:
- static bool s_isReflected;
- static void Reflect(AZ::ReflectContext* context)
- {
- s_isReflected = !context->IsRemovingReflection();
- }
- };
- bool TestReflectedClass::s_isReflected = false;
- TEST_F(ReflectionManagerTest, AddContext_AddClass)
- {
- m_reflection->AddReflectContext<AZ::SerializeContext>();
- m_reflection->Reflect(&TestReflectedClass::Reflect);
- EXPECT_TRUE(TestReflectedClass::s_isReflected);
- m_reflection->RemoveReflectContext<AZ::SerializeContext>();
- EXPECT_FALSE(TestReflectedClass::s_isReflected);
- }
- TEST_F(ReflectionManagerTest, AddClass_AddContext)
- {
- m_reflection->Reflect(&TestReflectedClass::Reflect);
- m_reflection->AddReflectContext<AZ::SerializeContext>();
- EXPECT_TRUE(TestReflectedClass::s_isReflected);
- m_reflection->Unreflect(&TestReflectedClass::Reflect);
- EXPECT_FALSE(TestReflectedClass::s_isReflected);
- }
- TEST_F(ReflectionManagerTest, UnreflectOnDestruct)
- {
- m_reflection->Reflect(&TestReflectedClass::Reflect);
- m_reflection->AddReflectContext<AZ::SerializeContext>();
- EXPECT_TRUE(TestReflectedClass::s_isReflected);
- m_reflection.reset();
- EXPECT_FALSE(TestReflectedClass::s_isReflected);
- }
- TEST_F(ReflectionManagerTest, UnreflectReReflect)
- {
- m_reflection->AddReflectContext<AZ::SerializeContext>();
- m_reflection->Reflect(&TestReflectedClass::Reflect);
- EXPECT_TRUE(TestReflectedClass::s_isReflected);
- m_reflection->Unreflect(&TestReflectedClass::Reflect);
- EXPECT_FALSE(TestReflectedClass::s_isReflected);
- m_reflection->Reflect(&TestReflectedClass::Reflect);
- EXPECT_TRUE(TestReflectedClass::s_isReflected);
- m_reflection->RemoveReflectContext<AZ::SerializeContext>();
- EXPECT_FALSE(TestReflectedClass::s_isReflected);
- m_reflection->AddReflectContext<AZ::SerializeContext>();
- EXPECT_TRUE(TestReflectedClass::s_isReflected);
- m_reflection.reset();
- EXPECT_FALSE(TestReflectedClass::s_isReflected);
- }
- }
|