diff --git a/.ci-builds/.azure-pipelines-icu4c.yml b/.ci-builds/.azure-pipelines-icu4c.yml index a749764931e..10a3af1e3f8 100644 --- a/.ci-builds/.azure-pipelines-icu4c.yml +++ b/.ci-builds/.azure-pipelines-icu4c.yml @@ -381,7 +381,7 @@ jobs: #------------------------------------------------------------------------- - job: ICU4C_MSVC_x64_Debug displayName: 'C: MSVC x64 Debug (VS 2022)' - timeoutInMinutes: 35 + timeoutInMinutes: 60 pool: vmImage: 'windows-2022' demands: @@ -487,7 +487,7 @@ jobs: #------------------------------------------------------------------------- - job: ICU4C_MSVC_x86_Debug_VS2019 displayName: 'C: MSVC 32-bit Debug (VS 2019)' - timeoutInMinutes: 60 + timeoutInMinutes: 90 pool: vmImage: 'windows-2019' demands: diff --git a/icu4c/source/i18n/messageformat2_allocation.h b/icu4c/source/i18n/messageformat2_allocation.h index 431862cecef..7be27e22252 100644 --- a/icu4c/source/i18n/messageformat2_allocation.h +++ b/icu4c/source/i18n/messageformat2_allocation.h @@ -24,16 +24,15 @@ namespace message2 { // Helpers template - static T* copyArray(const T* source, int32_t& len) { // `len` is an in/out param - if (source == nullptr) { - len = 0; + static T* copyArray(const T* source, int32_t len, UErrorCode& status) { + if (U_FAILURE(status)) { return nullptr; } + U_ASSERT(source != nullptr); + U_ASSERT(len >= 0); T* dest = new T[len]; if (dest == nullptr) { - // Set length to 0 to prevent the - // array from being accessed - len = 0; + status = U_MEMORY_ALLOCATION_ERROR; } else { for (int32_t i = 0; i < len; i++) { dest[i] = source[i]; @@ -43,13 +42,14 @@ namespace message2 { } template - static T* copyVectorToArray(const UVector& source, int32_t& len) { - len = source.size(); + static T* copyVectorToArray(const UVector& source, UErrorCode& status) { + if (U_FAILURE(status)) { + return nullptr; + } + int32_t len = source.size(); T* dest = new T[len]; if (dest == nullptr) { - // Set length to 0 to prevent the - // array from being accessed - len = 0; + status = U_MEMORY_ALLOCATION_ERROR; } else { for (int32_t i = 0; i < len; i++) { dest[i] = *(static_cast(source.elementAt(i))); @@ -59,19 +59,21 @@ namespace message2 { } template - static T* moveVectorToArray(UVector& source, int32_t& len) { - len = source.size(); + static T* moveVectorToArray(UVector& source, UErrorCode& status) { + if (U_FAILURE(status)) { + return nullptr; + } + + int32_t len = source.size(); T* dest = new T[len]; if (dest == nullptr) { - // Set length to 0 to prevent the - // array from being accessed - len = 0; + status = U_MEMORY_ALLOCATION_ERROR; } else { for (int32_t i = 0; i < len; i++) { dest[i] = std::move(*static_cast(source.elementAt(i))); } + source.removeAllElements(); } - source.removeAllElements(); return dest; } diff --git a/icu4c/source/i18n/messageformat2_data_model.cpp b/icu4c/source/i18n/messageformat2_data_model.cpp index 822d20c3b84..8255de01af0 100644 --- a/icu4c/source/i18n/messageformat2_data_model.cpp +++ b/icu4c/source/i18n/messageformat2_data_model.cpp @@ -77,15 +77,10 @@ SelectorKeys::Builder::~Builder() { } SelectorKeys::SelectorKeys(const UVector& ks, UErrorCode& status) : len(ks.size()) { + Key* result = copyVectorToArray(ks, status); if (U_FAILURE(status)) { return; } - Key* result = copyVectorToArray(ks, len); - if (result == nullptr) { - status = U_MEMORY_ALLOCATION_ERROR; - len = 0; - return; - } keys.adoptInstead(result); } @@ -95,7 +90,13 @@ SelectorKeys& SelectorKeys::operator=(SelectorKeys other) noexcept { } SelectorKeys::SelectorKeys(const SelectorKeys& other) : len(other.len) { - keys.adoptInstead(copyArray(other.keys.getAlias(), len)); + UErrorCode localErrorCode = U_ZERO_ERROR; + if (len != 0) { + keys.adoptInstead(copyArray(other.keys.getAlias(), len, localErrorCode)); + } + if (U_FAILURE(localErrorCode)) { + len = 0; + } } SelectorKeys::~SelectorKeys() { @@ -198,9 +199,18 @@ Key::~Key() {} // ------------ Reserved // Copy constructor -Reserved::Reserved(const Reserved& other) { - len = other.len; - parts.adoptInstead(copyArray(other.parts.getAlias(), len)); +Reserved::Reserved(const Reserved& other) : len(other.len) { + U_ASSERT(!other.bogus); + + UErrorCode localErrorCode = U_ZERO_ERROR; + if (len == 0) { + parts.adoptInstead(nullptr); + } else { + parts.adoptInstead(copyArray(other.parts.getAlias(), len, localErrorCode)); + } + if (U_FAILURE(localErrorCode)) { + bogus = true; + } } Reserved& Reserved::operator=(Reserved other) noexcept { @@ -212,14 +222,16 @@ Reserved::Reserved(const UVector& ps, UErrorCode& status) noexcept : len(ps.size if (U_FAILURE(status)) { return; } - parts = LocalArray(copyVectorToArray(ps, len)); + parts = LocalArray(copyVectorToArray(ps, status)); } int32_t Reserved::numParts() const { + U_ASSERT(!bogus); return len; } const Literal& Reserved::getPart(int32_t i) const { + U_ASSERT(!bogus); U_ASSERT(i < numParts()); return parts[i]; } @@ -257,14 +269,10 @@ Reserved::~Reserved() { //------------------------ Operator -OptionMap::OptionMap(const UVector& opts, UErrorCode& status) { - CHECK_ERROR(status); - - len = opts.size(); - Option* result = copyVectorToArray