| // © 2016 and later: Unicode, Inc. and others. | 
 | // License & terms of use: http://www.unicode.org/copyright.html | 
 | /******************************************************************** | 
 |  * COPYRIGHT:  | 
 |  * Copyright (c) 1997-2016, International Business Machines Corporation and | 
 |  * others. All Rights Reserved. | 
 |  ********************************************************************/ | 
 |  | 
 | #include <utility> | 
 |  | 
 | /** | 
 |  * IntlTestUtilities is the medium level test class for everything in the directory "utility". | 
 |  */ | 
 |  | 
 | #include "unicode/utypes.h" | 
 | #include "unicode/errorcode.h" | 
 | #include "unicode/localpointer.h" | 
 | #include "charstr.h" | 
 | #include "itutil.h" | 
 | #include "strtest.h" | 
 | #include "loctest.h" | 
 | #include "localebuildertest.h" | 
 | #include "citrtest.h" | 
 | #include "ustrtest.h" | 
 | #include "ucdtest.h" | 
 | #include "restest.h" | 
 | #include "restsnew.h" | 
 | #include "tsmthred.h" | 
 | #include "tsputil.h" | 
 | #include "uobjtest.h" | 
 | #include "utxttest.h" | 
 | #include "v32test.h" | 
 | #include "uvectest.h"  | 
 | #include "aliastst.h" | 
 | #include "usettest.h" | 
 |  | 
 | extern IntlTest *createBytesTrieTest(); | 
 | extern IntlTest *createLocaleMatcherTest(); | 
 | static IntlTest *createLocalPointerTest(); | 
 | extern IntlTest *createUCharsTrieTest(); | 
 | static IntlTest *createEnumSetTest(); | 
 | extern IntlTest *createSimpleFormatterTest(); | 
 | extern IntlTest *createUnifiedCacheTest(); | 
 | extern IntlTest *createQuantityFormatterTest(); | 
 | extern IntlTest *createPluralMapTest(); | 
 | #if !UCONFIG_NO_FORMATTING | 
 | extern IntlTest *createStaticUnicodeSetsTest(); | 
 | #endif | 
 |  | 
 | void IntlTestUtilities::runIndexedTest( int32_t index, UBool exec, const char* &name, char* par ) | 
 | { | 
 |     if (exec) logln("TestSuite Utilities: "); | 
 |     TESTCASE_AUTO_BEGIN; | 
 |     TESTCASE_AUTO_CLASS(MultithreadTest); | 
 |     TESTCASE_AUTO_CLASS(StringTest); | 
 |     TESTCASE_AUTO_CLASS(UnicodeStringTest); | 
 |     TESTCASE_AUTO_CLASS(LocaleTest); | 
 |     TESTCASE_AUTO_CLASS(CharIterTest); | 
 |     TESTCASE_AUTO_CLASS(UObjectTest); | 
 |     TESTCASE_AUTO_CLASS(UnicodeTest); | 
 |     TESTCASE_AUTO_CLASS(ResourceBundleTest); | 
 |     TESTCASE_AUTO_CLASS(NewResourceBundleTest); | 
 |     TESTCASE_AUTO_CLASS(PUtilTest); | 
 |     TESTCASE_AUTO_CLASS(UVector32Test); | 
 |     TESTCASE_AUTO_CLASS(UVectorTest); | 
 |     TESTCASE_AUTO_CLASS(UTextTest); | 
 |     TESTCASE_AUTO_CLASS(LocaleAliasTest); | 
 |     TESTCASE_AUTO_CLASS(UnicodeSetTest); | 
 |     TESTCASE_AUTO_CLASS(ErrorCodeTest); | 
 |     TESTCASE_AUTO_CREATE_CLASS(LocalPointerTest); | 
 |     TESTCASE_AUTO_CREATE_CLASS(BytesTrieTest); | 
 |     TESTCASE_AUTO_CREATE_CLASS(UCharsTrieTest); | 
 |     TESTCASE_AUTO_CREATE_CLASS(EnumSetTest); | 
 |     TESTCASE_AUTO_CREATE_CLASS(SimpleFormatterTest); | 
 |     TESTCASE_AUTO_CREATE_CLASS(UnifiedCacheTest); | 
 |     TESTCASE_AUTO_CREATE_CLASS(QuantityFormatterTest); | 
 |     TESTCASE_AUTO_CREATE_CLASS(PluralMapTest); | 
 | #if !UCONFIG_NO_FORMATTING | 
 |     TESTCASE_AUTO_CREATE_CLASS(StaticUnicodeSetsTest); | 
 | #endif | 
 |     TESTCASE_AUTO_CLASS(LocaleBuilderTest); | 
 |     TESTCASE_AUTO_CREATE_CLASS(LocaleMatcherTest); | 
 |     TESTCASE_AUTO_END; | 
 | } | 
 |  | 
 | void ErrorCodeTest::runIndexedTest(int32_t index, UBool exec, const char* &name, char* /*par*/) { | 
 |     if (exec) logln("TestSuite Utilities: "); | 
 |     switch (index) { | 
 |         case 0: name = "TestErrorCode"; if (exec) TestErrorCode(); break; | 
 |         case 1: name = "TestSubclass"; if (exec) TestSubclass(); break; | 
 |         case 2: name = "TestIcuTestErrorCode"; if (exec) TestIcuTestErrorCode(); break; | 
 |         default: name = ""; break; //needed to end loop | 
 |     } | 
 | } | 
 |  | 
 | static void RefPlusOne(UErrorCode &code) { code=(UErrorCode)(code+1); } | 
 | static void PtrPlusTwo(UErrorCode *code) { *code=(UErrorCode)(*code+2); } | 
 |  | 
 | void ErrorCodeTest::TestErrorCode() { | 
 |     ErrorCode errorCode; | 
 |     if(errorCode.get()!=U_ZERO_ERROR || !errorCode.isSuccess() || errorCode.isFailure()) { | 
 |         errln("ErrorCode did not initialize properly"); | 
 |         return; | 
 |     } | 
 |     errorCode.assertSuccess(); | 
 |     if(errorCode.errorName()!=u_errorName(U_ZERO_ERROR)) { | 
 |         errln("ErrorCode did not format error message string properly"); | 
 |     } | 
 |     RefPlusOne(errorCode); | 
 |     if(errorCode.get()!=U_ILLEGAL_ARGUMENT_ERROR || errorCode.isSuccess() || !errorCode.isFailure()) { | 
 |         errln("ErrorCode did not yield a writable reference"); | 
 |     } | 
 |     PtrPlusTwo(errorCode); | 
 |     if(errorCode.get()!=U_INVALID_FORMAT_ERROR || errorCode.isSuccess() || !errorCode.isFailure()) { | 
 |         errln("ErrorCode did not yield a writable pointer"); | 
 |     } | 
 |     errorCode.set(U_PARSE_ERROR); | 
 |     if(errorCode.get()!=U_PARSE_ERROR || errorCode.isSuccess() || !errorCode.isFailure()) { | 
 |         errln("ErrorCode.set() failed"); | 
 |     } | 
 |     if( errorCode.reset()!=U_PARSE_ERROR || errorCode.get()!=U_ZERO_ERROR || | 
 |         !errorCode.isSuccess() || errorCode.isFailure() | 
 |     ) { | 
 |         errln("ErrorCode did not reset properly"); | 
 |     } | 
 | } | 
 |  | 
 | class MyErrorCode: public ErrorCode { | 
 | public: | 
 |     MyErrorCode(int32_t &countChecks, int32_t &countDests) | 
 |         : checks(countChecks), dests(countDests) {} | 
 |     ~MyErrorCode() { | 
 |         if(isFailure()) { | 
 |             ++dests; | 
 |         } | 
 |     } | 
 | private: | 
 |     virtual void handleFailure() const { | 
 |         ++checks; | 
 |     } | 
 |     int32_t &checks; | 
 |     int32_t &dests; | 
 | }; | 
 |  | 
 | void ErrorCodeTest::TestSubclass() { | 
 |     int32_t countChecks=0; | 
 |     int32_t countDests=0; | 
 |     { | 
 |         MyErrorCode errorCode(countChecks, countDests); | 
 |         if( errorCode.get()!=U_ZERO_ERROR || !errorCode.isSuccess() || errorCode.isFailure() || | 
 |             countChecks!=0 || countDests!=0 | 
 |         ) { | 
 |             errln("ErrorCode did not initialize properly"); | 
 |             return; | 
 |         } | 
 |         errorCode.assertSuccess(); | 
 |         if(countChecks!=0) { | 
 |             errln("ErrorCode.assertSuccess() called handleFailure() despite success"); | 
 |         } | 
 |         RefPlusOne(errorCode); | 
 |         if(errorCode.get()!=U_ILLEGAL_ARGUMENT_ERROR || errorCode.isSuccess() || !errorCode.isFailure()) { | 
 |             errln("ErrorCode did not yield a writable reference"); | 
 |         } | 
 |         errorCode.assertSuccess(); | 
 |         if(countChecks!=1) { | 
 |             errln("ErrorCode.assertSuccess() did not handleFailure()"); | 
 |         } | 
 |         PtrPlusTwo(errorCode); | 
 |         if(errorCode.get()!=U_INVALID_FORMAT_ERROR || errorCode.isSuccess() || !errorCode.isFailure()) { | 
 |             errln("ErrorCode did not yield a writable pointer"); | 
 |         } | 
 |         errorCode.assertSuccess(); | 
 |         if(countChecks!=2) { | 
 |             errln("ErrorCode.assertSuccess() did not handleFailure()"); | 
 |         } | 
 |         errorCode.set(U_PARSE_ERROR); | 
 |         if(errorCode.get()!=U_PARSE_ERROR || errorCode.isSuccess() || !errorCode.isFailure()) { | 
 |             errln("ErrorCode.set() failed"); | 
 |         } | 
 |         if( errorCode.reset()!=U_PARSE_ERROR || errorCode.get()!=U_ZERO_ERROR || | 
 |             !errorCode.isSuccess() || errorCode.isFailure() | 
 |         ) { | 
 |             errln("ErrorCode did not reset properly"); | 
 |         } | 
 |         errorCode.assertSuccess(); | 
 |         if(countChecks!=2) { | 
 |             errln("ErrorCode.assertSuccess() called handleFailure() despite success"); | 
 |         } | 
 |     } | 
 |     if(countDests!=0) { | 
 |         errln("MyErrorCode destructor detected failure despite success"); | 
 |     } | 
 |     countChecks=countDests=0; | 
 |     { | 
 |         MyErrorCode errorCode(countChecks, countDests); | 
 |         errorCode.set(U_PARSE_ERROR); | 
 |     } | 
 |     if(countDests!=1) { | 
 |         errln("MyErrorCode destructor failed to detect failure"); | 
 |     } | 
 | } | 
 |  | 
 | class IcuTestErrorCodeTestHelper : public IntlTest { | 
 |   public: | 
 |     void errln( const UnicodeString &message ) U_OVERRIDE { | 
 |         test->assertFalse("Already saw an error", seenError); | 
 |         seenError = TRUE; | 
 |         test->assertEquals("Message for Error", expectedErrln, message); | 
 |         if (expectedDataErr) { | 
 |             test->errln("Got non-data error, but expected data error"); | 
 |         } | 
 |     } | 
 |  | 
 |     void dataerrln( const UnicodeString &message ) U_OVERRIDE { | 
 |         test->assertFalse("Already saw an error", seenError); | 
 |         seenError = TRUE; | 
 |         test->assertEquals("Message for Error", expectedErrln, message); | 
 |         if (!expectedDataErr) { | 
 |             test->errln("Got data error, but expected non-data error"); | 
 |         } | 
 |     } | 
 |  | 
 |     IntlTest* test; | 
 |     UBool expectedDataErr; | 
 |     UnicodeString expectedErrln; | 
 |     UBool seenError; | 
 | }; | 
 |  | 
 | void ErrorCodeTest::TestIcuTestErrorCode() { | 
 |     IcuTestErrorCodeTestHelper helper; | 
 |     helper.test = this; | 
 |  | 
 |     // Test destructor message | 
 |     helper.expectedErrln = u"AAA destructor: expected success but got error: U_ILLEGAL_PAD_POSITION"; | 
 |     helper.expectedDataErr = FALSE; | 
 |     helper.seenError = FALSE; | 
 |     { | 
 |         IcuTestErrorCode testStatus(helper, "AAA"); | 
 |         testStatus.set(U_ILLEGAL_PAD_POSITION); | 
 |     } | 
 |     assertTrue("Should have seen an error", helper.seenError); | 
 |  | 
 |     // Test destructor message with scope | 
 |     helper.expectedErrln = u"BBB destructor: expected success but got error: U_ILLEGAL_PAD_POSITION scope: foo"; | 
 |     helper.expectedDataErr = FALSE; | 
 |     helper.seenError = FALSE; | 
 |     { | 
 |         IcuTestErrorCode testStatus(helper, "BBB"); | 
 |         testStatus.setScope("foo"); | 
 |         testStatus.set(U_ILLEGAL_PAD_POSITION); | 
 |     } | 
 |     assertTrue("Should have seen an error", helper.seenError); | 
 |  | 
 |     // Check errIfFailure message with scope | 
 |     helper.expectedErrln = u"CCC expected success but got error: U_ILLEGAL_PAD_POSITION scope: foo"; | 
 |     helper.expectedDataErr = FALSE; | 
 |     helper.seenError = FALSE; | 
 |     { | 
 |         IcuTestErrorCode testStatus(helper, "CCC"); | 
 |         testStatus.setScope("foo"); | 
 |         testStatus.set(U_ILLEGAL_PAD_POSITION); | 
 |         testStatus.errIfFailureAndReset(); | 
 |         assertTrue("Should have seen an error", helper.seenError); | 
 |         helper.seenError = FALSE; | 
 |         helper.expectedErrln = u"CCC expected success but got error: U_ILLEGAL_CHAR_FOUND scope: foo - 5.4300"; | 
 |         testStatus.set(U_ILLEGAL_CHAR_FOUND); | 
 |         testStatus.errIfFailureAndReset("%6.4f", 5.43); | 
 |         assertTrue("Should have seen an error", helper.seenError); | 
 |     } | 
 |  | 
 |     // Check errDataIfFailure message without scope | 
 |     helper.expectedErrln = u"DDD data: expected success but got error: U_ILLEGAL_PAD_POSITION"; | 
 |     helper.expectedDataErr = TRUE; | 
 |     helper.seenError = FALSE; | 
 |     { | 
 |         IcuTestErrorCode testStatus(helper, "DDD"); | 
 |         testStatus.set(U_ILLEGAL_PAD_POSITION); | 
 |         testStatus.errDataIfFailureAndReset(); | 
 |         assertTrue("Should have seen an error", helper.seenError); | 
 |         helper.seenError = FALSE; | 
 |         helper.expectedErrln = u"DDD data: expected success but got error: U_ILLEGAL_CHAR_FOUND - 5.4300"; | 
 |         testStatus.set(U_ILLEGAL_CHAR_FOUND); | 
 |         testStatus.errDataIfFailureAndReset("%6.4f", 5.43); | 
 |         assertTrue("Should have seen an error", helper.seenError); | 
 |     } | 
 |  | 
 |     // Check expectFailure | 
 |     helper.expectedErrln = u"EEE expected: U_ILLEGAL_CHAR_FOUND but got error: U_ILLEGAL_PAD_POSITION"; | 
 |     helper.expectedDataErr = FALSE; | 
 |     helper.seenError = FALSE; | 
 |     { | 
 |         IcuTestErrorCode testStatus(helper, "EEE"); | 
 |         testStatus.set(U_ILLEGAL_PAD_POSITION); | 
 |         testStatus.expectErrorAndReset(U_ILLEGAL_PAD_POSITION); | 
 |         assertFalse("Should NOT have seen an error", helper.seenError); | 
 |         testStatus.set(U_ILLEGAL_PAD_POSITION); | 
 |         testStatus.expectErrorAndReset(U_ILLEGAL_CHAR_FOUND); | 
 |         assertTrue("Should have seen an error", helper.seenError); | 
 |         helper.seenError = FALSE; | 
 |         helper.expectedErrln = u"EEE expected: U_ILLEGAL_CHAR_FOUND but got error: U_ZERO_ERROR scope: scopety scope - 5.4300"; | 
 |         testStatus.setScope("scopety scope"); | 
 |         testStatus.set(U_ILLEGAL_PAD_POSITION); | 
 |         testStatus.expectErrorAndReset(U_ILLEGAL_PAD_POSITION, "%6.4f", 5.43); | 
 |         assertFalse("Should NOT have seen an error", helper.seenError); | 
 |         testStatus.expectErrorAndReset(U_ILLEGAL_CHAR_FOUND, "%6.4f", 5.43); | 
 |         assertTrue("Should have seen an error", helper.seenError); | 
 |     } | 
 | } | 
 |  | 
 |  | 
 | class LocalPointerTest : public IntlTest { | 
 | public: | 
 |     LocalPointerTest() {} | 
 |  | 
 |     void runIndexedTest(int32_t index, UBool exec, const char *&name, char *par=NULL); | 
 |  | 
 |     void TestLocalPointer(); | 
 |     void TestLocalPointerMoveSwap(); | 
 |     void TestLocalPointerStdUniquePtr(); | 
 |     void TestLocalArray(); | 
 |     void TestLocalArrayMoveSwap(); | 
 |     void TestLocalArrayStdUniquePtr(); | 
 |     void TestLocalXyzPointer(); | 
 |     void TestLocalXyzPointerMoveSwap(); | 
 |     void TestLocalXyzPointerNull(); | 
 |     void TestLocalXyzStdUniquePtr(); | 
 | }; | 
 |  | 
 | static IntlTest *createLocalPointerTest() { | 
 |     return new LocalPointerTest(); | 
 | } | 
 |  | 
 | void LocalPointerTest::runIndexedTest(int32_t index, UBool exec, const char *&name, char * /*par*/) { | 
 |     if(exec) { | 
 |         logln("TestSuite LocalPointerTest: "); | 
 |     } | 
 |     TESTCASE_AUTO_BEGIN; | 
 |     TESTCASE_AUTO(TestLocalPointer); | 
 |     TESTCASE_AUTO(TestLocalPointerMoveSwap); | 
 |     TESTCASE_AUTO(TestLocalPointerStdUniquePtr); | 
 |     TESTCASE_AUTO(TestLocalArray); | 
 |     TESTCASE_AUTO(TestLocalArrayMoveSwap); | 
 |     TESTCASE_AUTO(TestLocalArrayStdUniquePtr); | 
 |     TESTCASE_AUTO(TestLocalXyzPointer); | 
 |     TESTCASE_AUTO(TestLocalXyzPointerMoveSwap); | 
 |     TESTCASE_AUTO(TestLocalXyzPointerNull); | 
 |     TESTCASE_AUTO(TestLocalXyzStdUniquePtr); | 
 |     TESTCASE_AUTO_END; | 
 | } | 
 |  | 
 | // Exercise almost every LocalPointer and LocalPointerBase method. | 
 | void LocalPointerTest::TestLocalPointer() { | 
 |     // constructor | 
 |     LocalPointer<UnicodeString> s(new UnicodeString((UChar32)0x50005)); | 
 |     // isNULL(), isValid(), operator==(), operator!=() | 
 |     if(s.isNull() || !s.isValid() || s==NULL || !(s!=NULL)) { | 
 |         errln("LocalPointer constructor or NULL test failure"); | 
 |         return; | 
 |     } | 
 |     // getAlias(), operator->, operator* | 
 |     if(s.getAlias()->length()!=2 || s->length()!=2 || (*s).length()!=2) { | 
 |         errln("LocalPointer access failure"); | 
 |     } | 
 |     // adoptInstead(), orphan() | 
 |     s.adoptInstead(new UnicodeString((UChar)0xfffc)); | 
 |     if(s->length()!=1) { | 
 |         errln("LocalPointer adoptInstead(U+FFFC) failure"); | 
 |     } | 
 |     UnicodeString *orphan=s.orphan(); | 
 |     if(orphan==NULL || orphan->length()!=1 || s.isValid() || s!=NULL) { | 
 |         errln("LocalPointer orphan() failure"); | 
 |     } | 
 |     delete orphan; | 
 |     s.adoptInstead(new UnicodeString()); | 
 |     if(s->length()!=0) { | 
 |         errln("LocalPointer adoptInstead(empty) failure"); | 
 |     } | 
 |  | 
 |     // LocalPointer(p, errorCode) sets U_MEMORY_ALLOCATION_ERROR if p==NULL. | 
 |     UErrorCode errorCode = U_ZERO_ERROR; | 
 |     LocalPointer<CharString> csx(new CharString("some chars", errorCode), errorCode); | 
 |     if(csx.isNull() && U_SUCCESS(errorCode)) { | 
 |         errln("LocalPointer(p, errorCode) failure"); | 
 |         return; | 
 |     } | 
 |     errorCode = U_ZERO_ERROR; | 
 |     csx.adoptInsteadAndCheckErrorCode(new CharString("different chars", errorCode), errorCode); | 
 |     if(csx.isNull() && U_SUCCESS(errorCode)) { | 
 |         errln("adoptInsteadAndCheckErrorCode(p, errorCode) failure"); | 
 |         return; | 
 |     } | 
 |     // Incoming failure: Keep the current object and delete the input object. | 
 |     errorCode = U_ILLEGAL_ARGUMENT_ERROR; | 
 |     csx.adoptInsteadAndCheckErrorCode(new CharString("unused", errorCode), errorCode); | 
 |     if(csx.isValid() && strcmp(csx->data(), "different chars") != 0) { | 
 |         errln("adoptInsteadAndCheckErrorCode(p, U_FAILURE) did not retain the old object"); | 
 |         return; | 
 |     } | 
 |     errorCode = U_ZERO_ERROR; | 
 |     csx.adoptInsteadAndCheckErrorCode(NULL, errorCode); | 
 |     if(errorCode != U_MEMORY_ALLOCATION_ERROR) { | 
 |         errln("adoptInsteadAndCheckErrorCode(NULL, errorCode) did not set U_MEMORY_ALLOCATION_ERROR"); | 
 |         return; | 
 |     } | 
 |     if(csx.isValid()) { | 
 |         errln("adoptInsteadAndCheckErrorCode(NULL, errorCode) kept the object"); | 
 |         return; | 
 |     } | 
 |     errorCode = U_ZERO_ERROR; | 
 |     LocalPointer<CharString> null(NULL, errorCode); | 
 |     if(errorCode != U_MEMORY_ALLOCATION_ERROR) { | 
 |         errln("LocalPointer(NULL, errorCode) did not set U_MEMORY_ALLOCATION_ERROR"); | 
 |         return; | 
 |     } | 
 |  | 
 |     // destructor | 
 | } | 
 |  | 
 | // Try to avoid clang -Wself-move warnings from s1 = std::move(s1); | 
 | template<typename T> | 
 | void moveFrom(T &dest, T &src) { | 
 |     dest = std::move(src); | 
 | } | 
 |  | 
 | void LocalPointerTest::TestLocalPointerMoveSwap() { | 
 |     UnicodeString *p1 = new UnicodeString((UChar)0x61); | 
 |     UnicodeString *p2 = new UnicodeString((UChar)0x62); | 
 |     LocalPointer<UnicodeString> s1(p1); | 
 |     LocalPointer<UnicodeString> s2(p2); | 
 |     s1.swap(s2); | 
 |     if(s1.getAlias() != p2 || s2.getAlias() != p1) { | 
 |         errln("LocalPointer.swap() did not swap"); | 
 |     } | 
 |     swap(s1, s2); | 
 |     if(s1.getAlias() != p1 || s2.getAlias() != p2) { | 
 |         errln("swap(LocalPointer) did not swap back"); | 
 |     } | 
 |     LocalPointer<UnicodeString> s3; | 
 |     s3 = std::move(s1); | 
 |     if(s3.getAlias() != p1 || s1.isValid()) { | 
 |         errln("LocalPointer = std::move() did not move"); | 
 |     } | 
 |     infoln("TestLocalPointerMoveSwap() with rvalue references"); | 
 |     s1 = static_cast<LocalPointer<UnicodeString> &&>(s3); | 
 |     if(s1.getAlias() != p1 || s3.isValid()) { | 
 |         errln("LocalPointer move assignment operator did not move"); | 
 |     } | 
 |     LocalPointer<UnicodeString> s4(static_cast<LocalPointer<UnicodeString> &&>(s2)); | 
 |     if(s4.getAlias() != p2 || s2.isValid()) { | 
 |         errln("LocalPointer move constructor did not move"); | 
 |     } | 
 |  | 
 |     // Move self assignment leaves the object valid but in an undefined state. | 
 |     // Do it to make sure there is no crash, | 
 |     // but do not check for any particular resulting value. | 
 |     moveFrom(s1, s1); | 
 |     moveFrom(s3, s3); | 
 | } | 
 |  | 
 | void LocalPointerTest::TestLocalPointerStdUniquePtr() { | 
 |     auto* ptr = new UnicodeString((UChar32)0x50005); | 
 |     // Implicit conversion operator | 
 |     std::unique_ptr<UnicodeString> s = LocalPointer<UnicodeString>(ptr); | 
 |     // Explicit move constructor | 
 |     LocalPointer<UnicodeString> s2(std::move(s)); | 
 |     // Conversion operator should also work with std::move | 
 |     s = std::move(s2); | 
 |     // Back again with move assignment | 
 |     s2 = std::move(s); | 
 |     assertTrue("Pointer should remain the same", ptr == s2.getAlias()); | 
 | } | 
 |  | 
 | // Exercise almost every LocalArray method (but not LocalPointerBase). | 
 | void LocalPointerTest::TestLocalArray() { | 
 |     // constructor | 
 |     LocalArray<UnicodeString> a(new UnicodeString[2]); | 
 |     // operator[]() | 
 |     a[0].append((UChar)0x61); | 
 |     a[1].append((UChar32)0x60006); | 
 |     if(a[0].length()!=1 || a[1].length()!=2) { | 
 |         errln("LocalArray access failure"); | 
 |     } | 
 |     // adoptInstead() | 
 |     a.adoptInstead(new UnicodeString[4]); | 
 |     a[3].append((UChar)0x62).append((UChar)0x63).reverse(); | 
 |     if(a[3].length()!=2 || a[3][1]!=0x62) { | 
 |         errln("LocalArray adoptInstead() failure"); | 
 |     } | 
 |  | 
 |     // LocalArray(p, errorCode) sets U_MEMORY_ALLOCATION_ERROR if p==NULL. | 
 |     UErrorCode errorCode = U_ZERO_ERROR; | 
 |     LocalArray<UnicodeString> ua(new UnicodeString[3], errorCode); | 
 |     if(ua.isNull() && U_SUCCESS(errorCode)) { | 
 |         errln("LocalArray(p, errorCode) failure"); | 
 |         return; | 
 |     } | 
 |     errorCode = U_ZERO_ERROR; | 
 |     UnicodeString *u4 = new UnicodeString[4]; | 
 |     ua.adoptInsteadAndCheckErrorCode(u4, errorCode); | 
 |     if(ua.isNull() && U_SUCCESS(errorCode)) { | 
 |         errln("adoptInsteadAndCheckErrorCode(p, errorCode) failure"); | 
 |         return; | 
 |     } | 
 |     // Incoming failure: Keep the current object and delete the input object. | 
 |     errorCode = U_ILLEGAL_ARGUMENT_ERROR; | 
 |     ua.adoptInsteadAndCheckErrorCode(new UnicodeString[5], errorCode); | 
 |     if(ua.isValid() && ua.getAlias() != u4) { | 
 |         errln("adoptInsteadAndCheckErrorCode(p, U_FAILURE) did not retain the old array"); | 
 |         return; | 
 |     } | 
 |     errorCode = U_ZERO_ERROR; | 
 |     ua.adoptInsteadAndCheckErrorCode(NULL, errorCode); | 
 |     if(errorCode != U_MEMORY_ALLOCATION_ERROR) { | 
 |         errln("adoptInsteadAndCheckErrorCode(NULL, errorCode) did not set U_MEMORY_ALLOCATION_ERROR"); | 
 |         return; | 
 |     } | 
 |     if(ua.isValid()) { | 
 |         errln("adoptInsteadAndCheckErrorCode(NULL, errorCode) kept the array"); | 
 |         return; | 
 |     } | 
 |     errorCode = U_ZERO_ERROR; | 
 |     LocalArray<UnicodeString> null(NULL, errorCode); | 
 |     if(errorCode != U_MEMORY_ALLOCATION_ERROR) { | 
 |         errln("LocalArray(NULL, errorCode) did not set U_MEMORY_ALLOCATION_ERROR"); | 
 |         return; | 
 |     } | 
 |  | 
 |     // destructor | 
 | } | 
 |  | 
 | void LocalPointerTest::TestLocalArrayMoveSwap() { | 
 |     UnicodeString *p1 = new UnicodeString[2]; | 
 |     UnicodeString *p2 = new UnicodeString[3]; | 
 |     LocalArray<UnicodeString> a1(p1); | 
 |     LocalArray<UnicodeString> a2(p2); | 
 |     a1.swap(a2); | 
 |     if(a1.getAlias() != p2 || a2.getAlias() != p1) { | 
 |         errln("LocalArray.swap() did not swap"); | 
 |     } | 
 |     swap(a1, a2); | 
 |     if(a1.getAlias() != p1 || a2.getAlias() != p2) { | 
 |         errln("swap(LocalArray) did not swap back"); | 
 |     } | 
 |     LocalArray<UnicodeString> a3; | 
 |     a3 = std::move(a1); | 
 |     if(a3.getAlias() != p1 || a1.isValid()) { | 
 |         errln("LocalArray = std::move() did not move"); | 
 |     } | 
 |     infoln("TestLocalArrayMoveSwap() with rvalue references"); | 
 |     a1 = static_cast<LocalArray<UnicodeString> &&>(a3); | 
 |     if(a1.getAlias() != p1 || a3.isValid()) { | 
 |         errln("LocalArray move assignment operator did not move"); | 
 |     } | 
 |     LocalArray<UnicodeString> a4(static_cast<LocalArray<UnicodeString> &&>(a2)); | 
 |     if(a4.getAlias() != p2 || a2.isValid()) { | 
 |         errln("LocalArray move constructor did not move"); | 
 |     } | 
 |  | 
 |     // Move self assignment leaves the object valid but in an undefined state. | 
 |     // Do it to make sure there is no crash, | 
 |     // but do not check for any particular resulting value. | 
 |     moveFrom(a1, a1); | 
 |     moveFrom(a3, a3); | 
 | } | 
 |  | 
 | void LocalPointerTest::TestLocalArrayStdUniquePtr() { | 
 |     auto* ptr = new UnicodeString[2]; | 
 |     // Implicit conversion operator | 
 |     std::unique_ptr<UnicodeString[]> a = LocalArray<UnicodeString>(ptr); | 
 |     // Explicit move constructor | 
 |     LocalArray<UnicodeString> a2(std::move(a)); | 
 |     // Conversion operator should also work with std::move | 
 |     a = std::move(a2); | 
 |     // Back again with move assignment | 
 |     a2 = std::move(a); | 
 |     assertTrue("Pointer should remain the same", ptr == a2.getAlias()); | 
 | } | 
 |  | 
 | #include "unicode/ucnvsel.h" | 
 | #include "unicode/ucal.h" | 
 | #include "unicode/udatpg.h" | 
 | #include "unicode/uidna.h" | 
 | #include "unicode/uldnames.h" | 
 | #include "unicode/umsg.h" | 
 | #include "unicode/unorm2.h" | 
 | #include "unicode/uregex.h" | 
 | #include "unicode/utrans.h" | 
 | #include "unicode/uformattedvalue.h" | 
 |  | 
 | // Use LocalXyzPointer types that are not covered elsewhere in the intltest suite. | 
 | void LocalPointerTest::TestLocalXyzPointer() { | 
 |     IcuTestErrorCode errorCode(*this, "TestLocalXyzPointer"); | 
 |  | 
 |     static const char *const encoding="ISO-8859-1"; | 
 |     LocalUConverterSelectorPointer sel( | 
 |         ucnvsel_open(&encoding, 1, NULL, UCNV_ROUNDTRIP_SET, errorCode)); | 
 |     if(errorCode.errIfFailureAndReset("ucnvsel_open()")) { | 
 |         return; | 
 |     } | 
 |     if(sel.isNull()) { | 
 |         errln("LocalUConverterSelectorPointer failure"); | 
 |         return; | 
 |     } | 
 |  | 
 | #if !UCONFIG_NO_FORMATTING | 
 |     LocalUCalendarPointer cal(ucal_open(NULL, 0, "root", UCAL_GREGORIAN, errorCode)); | 
 |     if(errorCode.errDataIfFailureAndReset("ucal_open()")) { | 
 |         return; | 
 |     } | 
 |     if(cal.isNull()) { | 
 |         errln("LocalUCalendarPointer failure"); | 
 |         return; | 
 |     } | 
 |  | 
 |     LocalUDateTimePatternGeneratorPointer patgen(udatpg_open("root", errorCode)); | 
 |     if(errorCode.errDataIfFailureAndReset("udatpg_open()")) { | 
 |         return; | 
 |     } | 
 |     if(patgen.isNull()) { | 
 |         errln("LocalUDateTimePatternGeneratorPointer failure"); | 
 |         return; | 
 |     } | 
 |  | 
 |     LocalULocaleDisplayNamesPointer ldn(uldn_open("de-CH", ULDN_STANDARD_NAMES, errorCode)); | 
 |     if(errorCode.errIfFailureAndReset("uldn_open()")) { | 
 |         return; | 
 |     } | 
 |     if(ldn.isNull()) { | 
 |         errln("LocalULocaleDisplayNamesPointer failure"); | 
 |         return; | 
 |     } | 
 |  | 
 |     UnicodeString hello=UNICODE_STRING_SIMPLE("Hello {0}!"); | 
 |     LocalUMessageFormatPointer msg( | 
 |         umsg_open(hello.getBuffer(), hello.length(), "root", NULL, errorCode)); | 
 |     if(errorCode.errIfFailureAndReset("umsg_open()")) { | 
 |         return; | 
 |     } | 
 |     if(msg.isNull()) { | 
 |         errln("LocalUMessageFormatPointer failure"); | 
 |         return; | 
 |     } | 
 | #endif  /* UCONFIG_NO_FORMATTING  */ | 
 |  | 
 | #if !UCONFIG_NO_NORMALIZATION | 
 |     const UNormalizer2 *nfc=unorm2_getNFCInstance(errorCode); | 
 |     UnicodeSet emptySet; | 
 |     LocalUNormalizer2Pointer fn2(unorm2_openFiltered(nfc, emptySet.toUSet(), errorCode)); | 
 |     if(errorCode.errIfFailureAndReset("unorm2_openFiltered()")) { | 
 |         return; | 
 |     } | 
 |     if(fn2.isNull()) { | 
 |         errln("LocalUNormalizer2Pointer failure"); | 
 |         return; | 
 |     } | 
 | #endif /* !UCONFIG_NO_NORMALIZATION */ | 
 |  | 
 | #if !UCONFIG_NO_IDNA | 
 |     LocalUIDNAPointer idna(uidna_openUTS46(0, errorCode)); | 
 |     if(errorCode.errIfFailureAndReset("uidna_openUTS46()")) { | 
 |         return; | 
 |     } | 
 |     if(idna.isNull()) { | 
 |         errln("LocalUIDNAPointer failure"); | 
 |         return; | 
 |     } | 
 | #endif  /* !UCONFIG_NO_IDNA */ | 
 |  | 
 | #if !UCONFIG_NO_REGULAR_EXPRESSIONS | 
 |     UnicodeString pattern=UNICODE_STRING_SIMPLE("abc|xy+z"); | 
 |     LocalURegularExpressionPointer regex( | 
 |         uregex_open(pattern.getBuffer(), pattern.length(), 0, NULL, errorCode)); | 
 |     if(errorCode.errIfFailureAndReset("uregex_open()")) { | 
 |         return; | 
 |     } | 
 |     if(regex.isNull()) { | 
 |         errln("LocalURegularExpressionPointer failure"); | 
 |         return; | 
 |     } | 
 | #endif /* UCONFIG_NO_REGULAR_EXPRESSIONS */ | 
 |  | 
 | #if !UCONFIG_NO_TRANSLITERATION | 
 |     UnicodeString id=UNICODE_STRING_SIMPLE("Grek-Latn"); | 
 |     LocalUTransliteratorPointer trans( | 
 |         utrans_openU(id.getBuffer(), id.length(), UTRANS_FORWARD, NULL, 0, NULL, errorCode)); | 
 |     if(errorCode.errIfFailureAndReset("utrans_open()")) { | 
 |         return; | 
 |     } | 
 |     if(trans.isNull()) { | 
 |         errln("LocalUTransliteratorPointer failure"); | 
 |         return; | 
 |     } | 
 | #endif /* !UCONFIG_NO_TRANSLITERATION */ | 
 |  | 
 |     // destructors | 
 | } | 
 |  | 
 | void LocalPointerTest::TestLocalXyzPointerMoveSwap() { | 
 | #if !UCONFIG_NO_NORMALIZATION | 
 |     IcuTestErrorCode errorCode(*this, "TestLocalXyzPointerMoveSwap"); | 
 |     const UNormalizer2 *nfc=unorm2_getNFCInstance(errorCode); | 
 |     const UNormalizer2 *nfd=unorm2_getNFDInstance(errorCode); | 
 |     if(errorCode.errIfFailureAndReset("unorm2_getNF[CD]Instance()")) { | 
 |         return; | 
 |     } | 
 |     UnicodeSet emptySet; | 
 |     UNormalizer2 *p1 = unorm2_openFiltered(nfc, emptySet.toUSet(), errorCode); | 
 |     UNormalizer2 *p2 = unorm2_openFiltered(nfd, emptySet.toUSet(), errorCode); | 
 |     LocalUNormalizer2Pointer f1(p1); | 
 |     LocalUNormalizer2Pointer f2(p2); | 
 |     if(errorCode.errIfFailureAndReset("unorm2_openFiltered()")) { | 
 |         return; | 
 |     } | 
 |     if(f1.isNull() || f2.isNull()) { | 
 |         errln("LocalUNormalizer2Pointer failure"); | 
 |         return; | 
 |     } | 
 |     f1.swap(f2); | 
 |     if(f1.getAlias() != p2 || f2.getAlias() != p1) { | 
 |         errln("LocalUNormalizer2Pointer.swap() did not swap"); | 
 |     } | 
 |     swap(f1, f2); | 
 |     if(f1.getAlias() != p1 || f2.getAlias() != p2) { | 
 |         errln("swap(LocalUNormalizer2Pointer) did not swap back"); | 
 |     } | 
 |     LocalUNormalizer2Pointer f3; | 
 |     f3 = std::move(f1); | 
 |     if(f3.getAlias() != p1 || f1.isValid()) { | 
 |         errln("LocalUNormalizer2Pointer = std::move() did not move"); | 
 |     } | 
 |     infoln("TestLocalXyzPointerMoveSwap() with rvalue references"); | 
 |     f1 = static_cast<LocalUNormalizer2Pointer &&>(f3); | 
 |     if(f1.getAlias() != p1 || f3.isValid()) { | 
 |         errln("LocalUNormalizer2Pointer move assignment operator did not move"); | 
 |     } | 
 |     LocalUNormalizer2Pointer f4(static_cast<LocalUNormalizer2Pointer &&>(f2)); | 
 |     if(f4.getAlias() != p2 || f2.isValid()) { | 
 |         errln("LocalUNormalizer2Pointer move constructor did not move"); | 
 |     } | 
 |     // Move self assignment leaves the object valid but in an undefined state. | 
 |     // Do it to make sure there is no crash, | 
 |     // but do not check for any particular resulting value. | 
 |     moveFrom(f1, f1); | 
 |     moveFrom(f3, f3); | 
 | #endif /* !UCONFIG_NO_NORMALIZATION */ | 
 | } | 
 |  | 
 | // Try LocalXyzPointer types with NULL pointers. | 
 | void LocalPointerTest::TestLocalXyzPointerNull() { | 
 |     { | 
 |         IcuTestErrorCode errorCode(*this, "TestLocalXyzPointerNull/LocalUConverterSelectorPointer"); | 
 |         static const char *const encoding="ISO-8859-1"; | 
 |         LocalUConverterSelectorPointer null; | 
 |         LocalUConverterSelectorPointer sel( | 
 |             ucnvsel_open(&encoding, 1, NULL, UCNV_ROUNDTRIP_SET, errorCode)); | 
 |         sel.adoptInstead(NULL); | 
 |     } | 
 | #if !UCONFIG_NO_FORMATTING | 
 |     { | 
 |         IcuTestErrorCode errorCode(*this, "TestLocalXyzPointerNull/LocalUCalendarPointer"); | 
 |         LocalUCalendarPointer null; | 
 |         LocalUCalendarPointer cal(ucal_open(NULL, 0, "root", UCAL_GREGORIAN, errorCode)); | 
 |         if(!errorCode.errDataIfFailureAndReset("ucal_open()")) { | 
 |             cal.adoptInstead(NULL); | 
 |         } | 
 |     } | 
 |     { | 
 |         IcuTestErrorCode errorCode(*this, "TestLocalXyzPointerNull/LocalUDateTimePatternGeneratorPointer"); | 
 |         LocalUDateTimePatternGeneratorPointer null; | 
 |         LocalUDateTimePatternGeneratorPointer patgen(udatpg_open("root", errorCode)); | 
 |         patgen.adoptInstead(NULL); | 
 |     } | 
 |     { | 
 |         IcuTestErrorCode errorCode(*this, "TestLocalXyzPointerNull/LocalUMessageFormatPointer"); | 
 |         UnicodeString hello=UNICODE_STRING_SIMPLE("Hello {0}!"); | 
 |         LocalUMessageFormatPointer null; | 
 |         LocalUMessageFormatPointer msg( | 
 |             umsg_open(hello.getBuffer(), hello.length(), "root", NULL, errorCode)); | 
 |         msg.adoptInstead(NULL); | 
 |     } | 
 | #endif /* !UCONFIG_NO_FORMATTING */ | 
 |  | 
 | #if !UCONFIG_NO_REGULAR_EXPRESSIONS | 
 |     { | 
 |         IcuTestErrorCode errorCode(*this, "TestLocalXyzPointerNull/LocalURegularExpressionPointer"); | 
 |         UnicodeString pattern=UNICODE_STRING_SIMPLE("abc|xy+z"); | 
 |         LocalURegularExpressionPointer null; | 
 |         LocalURegularExpressionPointer regex( | 
 |             uregex_open(pattern.getBuffer(), pattern.length(), 0, NULL, errorCode)); | 
 |         if(!errorCode.errDataIfFailureAndReset("urege_open()")) { | 
 |             regex.adoptInstead(NULL); | 
 |         } | 
 |     } | 
 | #endif /* !UCONFIG_NO_REGULAR_EXPRESSIONS */ | 
 |  | 
 | #if !UCONFIG_NO_TRANSLITERATION | 
 |     { | 
 |         IcuTestErrorCode errorCode(*this, "TestLocalXyzPointerNull/LocalUTransliteratorPointer"); | 
 |         UnicodeString id=UNICODE_STRING_SIMPLE("Grek-Latn"); | 
 |         LocalUTransliteratorPointer null; | 
 |         LocalUTransliteratorPointer trans( | 
 |             utrans_openU(id.getBuffer(), id.length(), UTRANS_FORWARD, NULL, 0, NULL, errorCode)); | 
 |         if(!errorCode.errDataIfFailureAndReset("utrans_openU()")) { | 
 |             trans.adoptInstead(NULL); | 
 |         } | 
 |     } | 
 | #endif /* !UCONFIG_NO_TRANSLITERATION */ | 
 |  | 
 | } | 
 |  | 
 | void LocalPointerTest::TestLocalXyzStdUniquePtr() { | 
 |     IcuTestErrorCode status(*this, "TestLocalXyzStdUniquePtr"); | 
 | #if !UCONFIG_NO_FORMATTING | 
 |     auto* ptr = ucfpos_open(status); | 
 |     // Implicit conversion operator | 
 |     std::unique_ptr<UConstrainedFieldPosition, void(*)(UConstrainedFieldPosition*)> a = | 
 |         LocalUConstrainedFieldPositionPointer(ptr); | 
 |     // Explicit move constructor | 
 |     LocalUConstrainedFieldPositionPointer a2(std::move(a)); | 
 |     // Conversion operator should also work with std::move | 
 |     a = std::move(a2); | 
 |     // Back again with move assignment | 
 |     a2 = std::move(a); | 
 |     assertTrue("Pointer should remain the same", ptr == a2.getAlias()); | 
 | #endif // UCONFIG_NO_FORMATTING | 
 | } | 
 |  | 
 | /** EnumSet test **/ | 
 | #include "unicode/enumset.h" | 
 |  | 
 | class EnumSetTest : public IntlTest { | 
 | public: | 
 |   EnumSetTest() {} | 
 |   virtual void runIndexedTest(int32_t index, UBool exec, const char *&name, char *par=NULL); | 
 |   void TestEnumSet(); | 
 | }; | 
 |  | 
 | static IntlTest *createEnumSetTest() { | 
 |     return new EnumSetTest(); | 
 | } | 
 |  | 
 | void EnumSetTest::runIndexedTest(int32_t index, UBool exec, const char *&name, char * /*par*/) { | 
 |   TESTCASE_AUTO_BEGIN; | 
 |   TESTCASE_AUTO(TestEnumSet); | 
 |   TESTCASE_AUTO_END; | 
 | } | 
 | enum myEnum { | 
 |     MAX_NONBOOLEAN=-1, | 
 |     THING1, | 
 |     THING2, | 
 |     THING3, | 
 |     LIMIT_BOOLEAN | 
 | }; | 
 |  | 
 | void EnumSetTest::TestEnumSet() { | 
 |     EnumSet<myEnum, | 
 |             MAX_NONBOOLEAN+1, | 
 |             LIMIT_BOOLEAN> | 
 |                             flags; | 
 |  | 
 |     logln("Enum is from [%d..%d]\n", MAX_NONBOOLEAN+1, | 
 |           LIMIT_BOOLEAN); | 
 |  | 
 |     assertFalse(WHERE, flags.get(THING1)); | 
 |     assertFalse(WHERE, flags.get(THING2)); | 
 |     assertFalse(WHERE, flags.get(THING3)); | 
 |  | 
 |     logln("get(thing1)=%d, get(thing2)=%d, get(thing3)=%d\n",          flags.get(THING1),          flags.get(THING2),          flags.get(THING3)); | 
 |     logln("Value now: %d\n", flags.getAll()); | 
 |     flags.clear(); | 
 |     logln("clear -Value now: %d\n", flags.getAll()); | 
 |     logln("get(thing1)=%d, get(thing2)=%d, get(thing3)=%d\n",          flags.get(THING1),          flags.get(THING2),          flags.get(THING3)); | 
 |     assertFalse(WHERE, flags.get(THING1)); | 
 |     assertFalse(WHERE, flags.get(THING2)); | 
 |     assertFalse(WHERE, flags.get(THING3)); | 
 |     flags.add(THING1); | 
 |     logln("set THING1 -Value now: %d\n", flags.getAll()); | 
 |     assertTrue(WHERE, flags.get(THING1)); | 
 |     assertFalse(WHERE, flags.get(THING2)); | 
 |     assertFalse(WHERE, flags.get(THING3)); | 
 |     logln("get(thing1)=%d, get(thing2)=%d, get(thing3)=%d\n",          flags.get(THING1),          flags.get(THING2),          flags.get(THING3)); | 
 |     flags.add(THING3); | 
 |     logln("set THING3 -Value now: %d\n", flags.getAll()); | 
 |     assertTrue(WHERE, flags.get(THING1)); | 
 |     assertFalse(WHERE, flags.get(THING2)); | 
 |     assertTrue(WHERE, flags.get(THING3)); | 
 |     logln("get(thing1)=%d, get(thing2)=%d, get(thing3)=%d\n",          flags.get(THING1),          flags.get(THING2),          flags.get(THING3)); | 
 |     flags.remove(THING2); | 
 |     assertTrue(WHERE, flags.get(THING1)); | 
 |     assertFalse(WHERE, flags.get(THING2)); | 
 |     assertTrue(WHERE, flags.get(THING3)); | 
 |     logln("remove THING2 -Value now: %d\n", flags.getAll()); | 
 |     logln("get(thing1)=%d, get(thing2)=%d, get(thing3)=%d\n",          flags.get(THING1),          flags.get(THING2),          flags.get(THING3)); | 
 |     flags.remove(THING1); | 
 |     assertFalse(WHERE, flags.get(THING1)); | 
 |     assertFalse(WHERE, flags.get(THING2)); | 
 |     assertTrue(WHERE, flags.get(THING3)); | 
 |     logln("remove THING1 -Value now: %d\n", flags.getAll()); | 
 |     logln("get(thing1)=%d, get(thing2)=%d, get(thing3)=%d\n",          flags.get(THING1),          flags.get(THING2),          flags.get(THING3)); | 
 |  | 
 |     flags.clear(); | 
 |     logln("clear -Value now: %d\n", flags.getAll()); | 
 |     logln("get(thing1)=%d, get(thing2)=%d, get(thing3)=%d\n",          flags.get(THING1),          flags.get(THING2),          flags.get(THING3)); | 
 |     assertFalse(WHERE, flags.get(THING1)); | 
 |     assertFalse(WHERE, flags.get(THING2)); | 
 |     assertFalse(WHERE, flags.get(THING3)); | 
 | } |