• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /**
2  ************************************************************************************
3  * Copyright (C) 2006-2009,2011, International Business Machines Corporation        *
4  * and others. All Rights Reserved.                                                 *
5  ************************************************************************************
6  */
7 
8 #include "unicode/utypes.h"
9 
10 #if !UCONFIG_NO_BREAK_ITERATION
11 
12 #include "brkeng.h"
13 #include "dictbe.h"
14 #include "triedict.h"
15 #include "unicode/uchar.h"
16 #include "unicode/uniset.h"
17 #include "unicode/chariter.h"
18 #include "unicode/ures.h"
19 #include "unicode/udata.h"
20 #include "unicode/putil.h"
21 #include "unicode/ustring.h"
22 #include "unicode/uscript.h"
23 #include "uvector.h"
24 #include "umutex.h"
25 #include "uresimp.h"
26 #include "ubrkimpl.h"
27 
28 U_NAMESPACE_BEGIN
29 
30 /*
31  ******************************************************************
32  */
33 
LanguageBreakEngine()34 LanguageBreakEngine::LanguageBreakEngine() {
35 }
36 
~LanguageBreakEngine()37 LanguageBreakEngine::~LanguageBreakEngine() {
38 }
39 
40 /*
41  ******************************************************************
42  */
43 
LanguageBreakFactory()44 LanguageBreakFactory::LanguageBreakFactory() {
45 }
46 
~LanguageBreakFactory()47 LanguageBreakFactory::~LanguageBreakFactory() {
48 }
49 
50 /*
51  ******************************************************************
52  */
53 
UnhandledEngine(UErrorCode &)54 UnhandledEngine::UnhandledEngine(UErrorCode &/*status*/) {
55     for (int32_t i = 0; i < (int32_t)(sizeof(fHandled)/sizeof(fHandled[0])); ++i) {
56         fHandled[i] = 0;
57     }
58 }
59 
~UnhandledEngine()60 UnhandledEngine::~UnhandledEngine() {
61     for (int32_t i = 0; i < (int32_t)(sizeof(fHandled)/sizeof(fHandled[0])); ++i) {
62         if (fHandled[i] != 0) {
63             delete fHandled[i];
64         }
65     }
66 }
67 
68 UBool
handles(UChar32 c,int32_t breakType) const69 UnhandledEngine::handles(UChar32 c, int32_t breakType) const {
70     return (breakType >= 0 && breakType < (int32_t)(sizeof(fHandled)/sizeof(fHandled[0]))
71         && fHandled[breakType] != 0 && fHandled[breakType]->contains(c));
72 }
73 
74 int32_t
findBreaks(UText * text,int32_t startPos,int32_t endPos,UBool reverse,int32_t breakType,UStack &) const75 UnhandledEngine::findBreaks( UText *text,
76                                  int32_t startPos,
77                                  int32_t endPos,
78                                  UBool reverse,
79                                  int32_t breakType,
80                                  UStack &/*foundBreaks*/ ) const {
81     if (breakType >= 0 && breakType < (int32_t)(sizeof(fHandled)/sizeof(fHandled[0]))) {
82         UChar32 c = utext_current32(text);
83         if (reverse) {
84             while((int32_t)utext_getNativeIndex(text) > startPos && fHandled[breakType]->contains(c)) {
85                 c = utext_previous32(text);
86             }
87         }
88         else {
89             while((int32_t)utext_getNativeIndex(text) < endPos && fHandled[breakType]->contains(c)) {
90                 utext_next32(text);            // TODO:  recast loop to work with post-increment operations.
91                 c = utext_current32(text);
92             }
93         }
94     }
95     return 0;
96 }
97 
98 void
handleCharacter(UChar32 c,int32_t breakType)99 UnhandledEngine::handleCharacter(UChar32 c, int32_t breakType) {
100     if (breakType >= 0 && breakType < (int32_t)(sizeof(fHandled)/sizeof(fHandled[0]))) {
101         if (fHandled[breakType] == 0) {
102             fHandled[breakType] = new UnicodeSet();
103             if (fHandled[breakType] == 0) {
104                 return;
105             }
106         }
107         if (!fHandled[breakType]->contains(c)) {
108             UErrorCode status = U_ZERO_ERROR;
109             // Apply the entire script of the character.
110             int32_t script = u_getIntPropertyValue(c, UCHAR_SCRIPT);
111             fHandled[breakType]->applyIntPropertyValue(UCHAR_SCRIPT, script, status);
112         }
113     }
114 }
115 
116 /*
117  ******************************************************************
118  */
119 
ICULanguageBreakFactory(UErrorCode &)120 ICULanguageBreakFactory::ICULanguageBreakFactory(UErrorCode &/*status*/) {
121     fEngines = 0;
122 }
123 
~ICULanguageBreakFactory()124 ICULanguageBreakFactory::~ICULanguageBreakFactory() {
125     if (fEngines != 0) {
126         delete fEngines;
127     }
128 }
129 
130 U_NAMESPACE_END
131 U_CDECL_BEGIN
_deleteEngine(void * obj)132 static void U_CALLCONV _deleteEngine(void *obj) {
133     delete (const U_NAMESPACE_QUALIFIER LanguageBreakEngine *) obj;
134 }
135 U_CDECL_END
136 U_NAMESPACE_BEGIN
137 
138 const LanguageBreakEngine *
getEngineFor(UChar32 c,int32_t breakType)139 ICULanguageBreakFactory::getEngineFor(UChar32 c, int32_t breakType) {
140     UBool       needsInit;
141     int32_t     i;
142     const LanguageBreakEngine *lbe = NULL;
143     UErrorCode  status = U_ZERO_ERROR;
144 
145     // TODO: The global mutex should not be used.
146     // The global mutex should only be used for short periods.
147     // A ICULanguageBreakFactory specific mutex should be used.
148     umtx_lock(NULL);
149     needsInit = (UBool)(fEngines == NULL);
150     if (!needsInit) {
151         i = fEngines->size();
152         while (--i >= 0) {
153             lbe = (const LanguageBreakEngine *)(fEngines->elementAt(i));
154             if (lbe != NULL && lbe->handles(c, breakType)) {
155                 break;
156             }
157             lbe = NULL;
158         }
159     }
160     umtx_unlock(NULL);
161 
162     if (lbe != NULL) {
163         return lbe;
164     }
165 
166     if (needsInit) {
167         UStack  *engines = new UStack(_deleteEngine, NULL, status);
168         if (U_SUCCESS(status) && engines == NULL) {
169             status = U_MEMORY_ALLOCATION_ERROR;
170         }
171         else if (U_FAILURE(status)) {
172             delete engines;
173             engines = NULL;
174         }
175         else {
176             umtx_lock(NULL);
177             if (fEngines == NULL) {
178                 fEngines = engines;
179                 engines = NULL;
180             }
181             umtx_unlock(NULL);
182             delete engines;
183         }
184     }
185 
186     if (fEngines == NULL) {
187         return NULL;
188     }
189 
190     // We didn't find an engine the first time through, or there was no
191     // stack. Create an engine.
192     const LanguageBreakEngine *newlbe = loadEngineFor(c, breakType);
193 
194     // Now get the lock, and see if someone else has created it in the
195     // meantime
196     umtx_lock(NULL);
197     i = fEngines->size();
198     while (--i >= 0) {
199         lbe = (const LanguageBreakEngine *)(fEngines->elementAt(i));
200         if (lbe != NULL && lbe->handles(c, breakType)) {
201             break;
202         }
203         lbe = NULL;
204     }
205     if (lbe == NULL && newlbe != NULL) {
206         fEngines->push((void *)newlbe, status);
207         lbe = newlbe;
208         newlbe = NULL;
209     }
210     umtx_unlock(NULL);
211 
212     delete newlbe;
213 
214     return lbe;
215 }
216 
217 const LanguageBreakEngine *
loadEngineFor(UChar32 c,int32_t breakType)218 ICULanguageBreakFactory::loadEngineFor(UChar32 c, int32_t breakType) {
219     UErrorCode status = U_ZERO_ERROR;
220     UScriptCode code = uscript_getScript(c, &status);
221     if (U_SUCCESS(status)) {
222         const CompactTrieDictionary *dict = loadDictionaryFor(code, breakType);
223         if (dict != NULL) {
224             const LanguageBreakEngine *engine = NULL;
225             switch(code) {
226             case USCRIPT_THAI:
227                 engine = new ThaiBreakEngine(dict, status);
228                 break;
229             case USCRIPT_KHMER:
230                 engine = new KhmerBreakEngine(dict, status);
231                 break;
232             default:
233                 break;
234             }
235             if (engine == NULL) {
236                 delete dict;
237             }
238             else if (U_FAILURE(status)) {
239                 delete engine;
240                 engine = NULL;
241             }
242             return engine;
243         }
244     }
245     return NULL;
246 }
247 
248 const CompactTrieDictionary *
loadDictionaryFor(UScriptCode script,int32_t)249 ICULanguageBreakFactory::loadDictionaryFor(UScriptCode script, int32_t /*breakType*/) {
250     UErrorCode status = U_ZERO_ERROR;
251     // Open root from brkitr tree.
252     char dictnbuff[256];
253     char ext[4]={'\0'};
254 
255     UResourceBundle *b = ures_open(U_ICUDATA_BRKITR, "", &status);
256     b = ures_getByKeyWithFallback(b, "dictionaries", b, &status);
257     b = ures_getByKeyWithFallback(b, uscript_getShortName(script), b, &status);
258     int32_t dictnlength = 0;
259     const UChar *dictfname = ures_getString(b, &dictnlength, &status);
260     if (U_SUCCESS(status) && (size_t)dictnlength >= sizeof(dictnbuff)) {
261         dictnlength = 0;
262         status = U_BUFFER_OVERFLOW_ERROR;
263     }
264     if (U_SUCCESS(status) && dictfname) {
265         UChar* extStart=u_strchr(dictfname, 0x002e);
266         int len = 0;
267         if(extStart!=NULL){
268             len = (int)(extStart-dictfname);
269             u_UCharsToChars(extStart+1, ext, sizeof(ext)); // nul terminates the buff
270             u_UCharsToChars(dictfname, dictnbuff, len);
271         }
272         dictnbuff[len]=0; // nul terminate
273     }
274     ures_close(b);
275     UDataMemory *file = udata_open(U_ICUDATA_BRKITR, ext, dictnbuff, &status);
276     if (U_SUCCESS(status)) {
277         const CompactTrieDictionary *dict = new CompactTrieDictionary(
278             file, status);
279         if (U_SUCCESS(status) && dict == NULL) {
280             status = U_MEMORY_ALLOCATION_ERROR;
281         }
282         if (U_FAILURE(status)) {
283             delete dict;
284             dict = NULL;
285         }
286         return dict;
287     }
288     return NULL;
289 }
290 
291 U_NAMESPACE_END
292 
293 #endif /* #if !UCONFIG_NO_BREAK_ITERATION */
294