Coverage Report

Created: 2023-03-04 07:00

/src/icu/icu4c/source/i18n/number_usageprefs.cpp
Line
Count
Source (jump to first uncovered line)
1
// © 2020 and later: Unicode, Inc. and others.
2
// License & terms of use: http://www.unicode.org/copyright.html
3
4
#include "unicode/utypes.h"
5
6
#if !UCONFIG_NO_FORMATTING
7
8
#include "number_usageprefs.h"
9
#include "cstring.h"
10
#include "number_decimalquantity.h"
11
#include "number_microprops.h"
12
#include "number_roundingutils.h"
13
#include "number_skeletons.h"
14
#include "unicode/char16ptr.h"
15
#include "unicode/currunit.h"
16
#include "unicode/fmtable.h"
17
#include "unicode/measure.h"
18
#include "unicode/numberformatter.h"
19
#include "unicode/platform.h"
20
#include "unicode/unum.h"
21
#include "unicode/urename.h"
22
#include "units_data.h"
23
24
using namespace icu;
25
using namespace icu::number;
26
using namespace icu::number::impl;
27
using icu::StringSegment;
28
using icu::units::ConversionRates;
29
30
// Copy constructor
31
0
StringProp::StringProp(const StringProp &other) : StringProp() {
32
0
    this->operator=(other);
33
0
}
34
35
// Copy assignment operator
36
0
StringProp &StringProp::operator=(const StringProp &other) {
37
0
    if (this == &other) { return *this; }  // self-assignment: no-op
38
0
    fLength = 0;
39
0
    fError = other.fError;
40
0
    if (fValue != nullptr) {
41
0
        uprv_free(fValue);
42
0
        fValue = nullptr;
43
0
    }
44
0
    if (other.fValue == nullptr) {
45
0
        return *this;
46
0
    }
47
0
    if (U_FAILURE(other.fError)) {
48
        // We don't bother trying to allocating memory if we're in any case busy
49
        // copying an errored StringProp.
50
0
        return *this;
51
0
    }
52
0
    fValue = (char *)uprv_malloc(other.fLength + 1);
53
0
    if (fValue == nullptr) {
54
0
        fError = U_MEMORY_ALLOCATION_ERROR;
55
0
        return *this;
56
0
    }
57
0
    fLength = other.fLength;
58
0
    uprv_strncpy(fValue, other.fValue, fLength + 1);
59
0
    return *this;
60
0
}
61
62
// Move constructor
63
StringProp::StringProp(StringProp &&src) noexcept : fValue(src.fValue),
64
                                                      fLength(src.fLength),
65
7.20k
                                                      fError(src.fError) {
66
    // Take ownership away from src if necessary
67
7.20k
    src.fValue = nullptr;
68
7.20k
}
69
70
// Move assignment operator
71
21.6k
StringProp &StringProp::operator=(StringProp &&src) noexcept {
72
21.6k
    if (this == &src) {
73
0
        return *this;
74
0
    }
75
21.6k
    if (fValue != nullptr) {
76
0
        uprv_free(fValue);
77
0
    }
78
21.6k
    fValue = src.fValue;
79
21.6k
    fLength = src.fLength;
80
21.6k
    fError = src.fError;
81
    // Take ownership away from src if necessary
82
21.6k
    src.fValue = nullptr;
83
21.6k
    return *this;
84
21.6k
}
85
86
34.7k
StringProp::~StringProp() {
87
34.7k
    if (fValue != nullptr) {
88
0
        uprv_free(fValue);
89
0
        fValue = nullptr;
90
0
    }
91
34.7k
}
92
93
0
void StringProp::set(StringPiece value) {
94
0
    if (fValue != nullptr) {
95
0
        uprv_free(fValue);
96
0
        fValue = nullptr;
97
0
    }
98
0
    fLength = value.length();
99
0
    fValue = (char *)uprv_malloc(fLength + 1);
100
0
    if (fValue == nullptr) {
101
0
        fLength = 0;
102
0
        fError = U_MEMORY_ALLOCATION_ERROR;
103
0
        return;
104
0
    }
105
0
    if (fLength > 0) {
106
0
        uprv_strncpy(fValue, value.data(), fLength);
107
0
    }
108
0
    fValue[fLength] = 0;
109
0
}
110
111
// Populates micros.mixedMeasures and modifies quantity, based on the values in
112
// measures.
113
void mixedMeasuresToMicros(const MaybeStackVector<Measure> &measures, DecimalQuantity *quantity,
114
0
                           MicroProps *micros, UErrorCode status) {
115
0
    micros->mixedMeasuresCount = measures.length();
116
117
0
    if (micros->mixedMeasures.getCapacity() < micros->mixedMeasuresCount) {
118
0
        if (micros->mixedMeasures.resize(micros->mixedMeasuresCount) == nullptr) {
119
0
            status = U_MEMORY_ALLOCATION_ERROR;
120
0
            return;
121
0
        }
122
0
    }
123
124
0
    for (int32_t i = 0; i < micros->mixedMeasuresCount; i++) {
125
0
        switch (measures[i]->getNumber().getType()) {
126
0
        case Formattable::kInt64:
127
0
            micros->mixedMeasures[i] = measures[i]->getNumber().getInt64();
128
0
            break;
129
130
0
        case Formattable::kDouble:
131
0
            U_ASSERT(micros->indexOfQuantity < 0);
132
0
            quantity->setToDouble(measures[i]->getNumber().getDouble());
133
0
            micros->indexOfQuantity = i;
134
0
            break;
135
136
0
        default:
137
0
            U_ASSERT(0 == "Found a Measure Number which is neither a double nor an int");
138
0
            UPRV_UNREACHABLE_EXIT;
139
0
            break;
140
0
        }
141
142
0
        if (U_FAILURE(status)) {
143
0
            return;
144
0
        }
145
0
    }
146
147
0
    if (micros->indexOfQuantity < 0) {
148
        // There is no quantity.
149
0
        status = U_INTERNAL_PROGRAM_ERROR;
150
0
    }
151
0
}
152
153
UsagePrefsHandler::UsagePrefsHandler(const Locale &locale,
154
                                     const MeasureUnit &inputUnit,
155
                                     const StringPiece usage,
156
                                     const MicroPropsGenerator *parent,
157
                                     UErrorCode &status)
158
    : fUnitsRouter(inputUnit, locale, usage, status),
159
0
      fParent(parent) {
160
0
}
161
162
void UsagePrefsHandler::processQuantity(DecimalQuantity &quantity, MicroProps &micros,
163
0
                                        UErrorCode &status) const {
164
0
    fParent->processQuantity(quantity, micros, status);
165
0
    if (U_FAILURE(status)) {
166
0
        return;
167
0
    }
168
169
0
    quantity.roundToInfinity(); // Enables toDouble
170
0
    const units::RouteResult routed = fUnitsRouter.route(quantity.toDouble(), &micros.rounder, status);
171
0
    if (U_FAILURE(status)) {
172
0
        return;
173
0
    }
174
0
    const MaybeStackVector<Measure>& routedMeasures = routed.measures;
175
0
    micros.outputUnit = routed.outputUnit.copy(status).build(status);
176
0
    if (U_FAILURE(status)) {
177
0
        return;
178
0
    }
179
180
0
    mixedMeasuresToMicros(routedMeasures, &quantity, &micros, status);
181
0
}
182
183
UnitConversionHandler::UnitConversionHandler(const MeasureUnit &targetUnit,
184
                                             const MicroPropsGenerator *parent, UErrorCode &status)
185
0
    : fOutputUnit(targetUnit), fParent(parent) {
186
0
    MeasureUnitImpl tempInput, tempOutput;
187
188
0
    ConversionRates conversionRates(status);
189
0
    if (U_FAILURE(status)) {
190
0
        return;
191
0
    }
192
193
0
    const MeasureUnitImpl &targetUnitImpl =
194
0
        MeasureUnitImpl::forMeasureUnit(targetUnit, tempOutput, status);
195
0
    fUnitConverter.adoptInsteadAndCheckErrorCode(
196
0
        new ComplexUnitsConverter(targetUnitImpl, conversionRates, status), status);
197
0
}
198
199
void UnitConversionHandler::processQuantity(DecimalQuantity &quantity, MicroProps &micros,
200
0
                                            UErrorCode &status) const {
201
0
    fParent->processQuantity(quantity, micros, status);
202
0
    if (U_FAILURE(status)) {
203
0
        return;
204
0
    }
205
0
    quantity.roundToInfinity(); // Enables toDouble
206
0
    MaybeStackVector<Measure> measures =
207
0
        fUnitConverter->convert(quantity.toDouble(), &micros.rounder, status);
208
0
    micros.outputUnit = fOutputUnit;
209
0
    if (U_FAILURE(status)) {
210
0
        return;
211
0
    }
212
213
0
    mixedMeasuresToMicros(measures, &quantity, &micros, status);
214
0
}
215
216
#endif /* #if !UCONFIG_NO_FORMATTING */