1 // © 2016 and later: Unicode, Inc. and others.
2 // License & terms of use: http://www.unicode.org/copyright.html
3 /*
4 **********************************************************************
5 * Copyright (C) 1999-2014, International Business Machines
6 * Corporation and others. All Rights Reserved.
7 **********************************************************************
8 * Date Name Description
9 * 11/17/99 aliu Creation.
10 **********************************************************************
11 */
12
13 #include "unicode/utypes.h"
14 #include "umutex.h"
15
16 #if !UCONFIG_NO_TRANSLITERATION
17
18 #include "unicode/unistr.h"
19 #include "unicode/uniset.h"
20 #include "rbt_data.h"
21 #include "hash.h"
22 #include "cmemory.h"
23
24 U_NAMESPACE_BEGIN
25
TransliterationRuleData(UErrorCode & status)26 TransliterationRuleData::TransliterationRuleData(UErrorCode& status)
27 : UMemory(), ruleSet(status), variableNames(status),
28 variables(0), variablesAreOwned(true)
29 {
30 if (U_FAILURE(status)) {
31 return;
32 }
33 variableNames.setValueDeleter(uprv_deleteUObject);
34 variables = 0;
35 variablesLength = 0;
36 }
37
TransliterationRuleData(const TransliterationRuleData & other)38 TransliterationRuleData::TransliterationRuleData(const TransliterationRuleData& other) :
39 UMemory(other), ruleSet(other.ruleSet),
40 variablesAreOwned(true),
41 variablesBase(other.variablesBase),
42 variablesLength(other.variablesLength)
43 {
44 UErrorCode status = U_ZERO_ERROR;
45 int32_t i = 0;
46 variableNames.setValueDeleter(uprv_deleteUObject);
47 int32_t pos = UHASH_FIRST;
48 const UHashElement *e;
49 while ((e = other.variableNames.nextElement(pos)) != 0) {
50 UnicodeString* value =
51 new UnicodeString(*(const UnicodeString*)e->value.pointer);
52 // Exit out if value could not be created.
53 if (value == nullptr) {
54 return;
55 }
56 variableNames.put(*(UnicodeString*)e->key.pointer, value, status);
57 }
58
59 variables = 0;
60 if (other.variables != 0) {
61 variables = (UnicodeFunctor **)uprv_malloc(variablesLength * sizeof(UnicodeFunctor *));
62 /* test for nullptr */
63 if (variables == 0) {
64 status = U_MEMORY_ALLOCATION_ERROR;
65 return;
66 }
67 for (i=0; i<variablesLength; ++i) {
68 variables[i] = other.variables[i]->clone();
69 if (variables[i] == nullptr) {
70 status = U_MEMORY_ALLOCATION_ERROR;
71 break;
72 }
73 }
74 }
75 // Remove the array and exit if memory allocation error occurred.
76 if (U_FAILURE(status)) {
77 for (int32_t n = i-1; n >= 0; n--) {
78 delete variables[n];
79 }
80 uprv_free(variables);
81 variables = nullptr;
82 return;
83 }
84
85 // Do this last, _after_ setting up variables[].
86 ruleSet.setData(this); // ruleSet must already be frozen
87 }
88
~TransliterationRuleData()89 TransliterationRuleData::~TransliterationRuleData() {
90 if (variablesAreOwned && variables != 0) {
91 for (int32_t i=0; i<variablesLength; ++i) {
92 delete variables[i];
93 }
94 }
95 uprv_free(variables);
96 }
97
98 UnicodeFunctor*
lookup(UChar32 standIn) const99 TransliterationRuleData::lookup(UChar32 standIn) const {
100 int32_t i = standIn - variablesBase;
101 return (i >= 0 && i < variablesLength) ? variables[i] : 0;
102 }
103
104 UnicodeMatcher*
lookupMatcher(UChar32 standIn) const105 TransliterationRuleData::lookupMatcher(UChar32 standIn) const {
106 UnicodeFunctor *f = lookup(standIn);
107 return (f != 0) ? f->toMatcher() : 0;
108 }
109
110 UnicodeReplacer*
lookupReplacer(UChar32 standIn) const111 TransliterationRuleData::lookupReplacer(UChar32 standIn) const {
112 UnicodeFunctor *f = lookup(standIn);
113 return (f != 0) ? f->toReplacer() : 0;
114 }
115
116
117 U_NAMESPACE_END
118
119 #endif /* #if !UCONFIG_NO_TRANSLITERATION */
120