1 // © 2016 and later: Unicode, Inc. and others.
2 // License & terms of use: http://www.unicode.org/copyright.html
3 /*
4 *******************************************************************************
5 *
6 * Copyright (C) 1999-2014, International Business Machines
7 * Corporation and others. All Rights Reserved.
8 *
9 *******************************************************************************
10 * file name: letest.cpp
11 *
12 * created on: 11/06/2000
13 * created by: Eric R. Mader
14 */
15
16 #include "unicode/utypes.h"
17 #include "unicode/uclean.h"
18 #include "unicode/uchar.h"
19 #include "unicode/unistr.h"
20 #include "unicode/uscript.h"
21 #include "unicode/putil.h"
22 #include "unicode/ctest.h"
23
24 #include "layout/LETypes.h"
25 #include "layout/LEScripts.h"
26 #include "layout/LayoutEngine.h"
27
28 #include "layout/ParagraphLayout.h"
29 #include "layout/RunArrays.h"
30
31 #include "PortableFontInstance.h"
32 #include "SimpleFontInstance.h"
33
34 #include "letsutil.h"
35 #include "letest.h"
36
37 #include "xmlparser.h"
38 #include "putilimp.h" // for uprv_getUTCtime()
39
40 #include <stdlib.h>
41 #include <string.h>
42
43 U_NAMESPACE_USE
44
45 #define CH_COMMA 0x002C
46
47 U_CDECL_BEGIN
48
ScriptTest(void)49 static void U_CALLCONV ScriptTest(void)
50 {
51 if ((int)scriptCodeCount != (int)USCRIPT_CODE_LIMIT) {
52 log_err("ScriptCodes::scriptCodeCount = %d, but UScriptCode::USCRIPT_CODE_LIMIT = %d\n", scriptCodeCount, USCRIPT_CODE_LIMIT);
53 }
54 }
55
ParamTest(void)56 static void U_CALLCONV ParamTest(void)
57 {
58 LEErrorCode status = LE_NO_ERROR;
59 SimpleFontInstance *font = new SimpleFontInstance(12, status);
60 LayoutEngine *engine = LayoutEngine::layoutEngineFactory(font, arabScriptCode, -1, status);
61 LEGlyphID *glyphs = NULL;
62 le_int32 *indices = NULL;
63 float *positions = NULL;
64 le_int32 glyphCount = 0;
65
66 glyphCount = engine->getGlyphCount();
67 if (glyphCount != 0) {
68 log_err("Calling getGlyphCount() on an empty layout returned %d.\n", glyphCount);
69 }
70
71 glyphs = NEW_ARRAY(LEGlyphID, glyphCount + 10);
72 indices = NEW_ARRAY(le_int32, glyphCount + 10);
73 positions = NEW_ARRAY(float, glyphCount + 10);
74
75 engine->getGlyphs(NULL, status);
76
77 if (status != LE_ILLEGAL_ARGUMENT_ERROR) {
78 log_err("Calling getGlyphs(NULL, status) did not return LE_ILLEGAL_ARGUMENT_ERROR.\n");
79 }
80
81 status = LE_NO_ERROR;
82 engine->getGlyphs(glyphs, status);
83
84 if (status != LE_NO_LAYOUT_ERROR) {
85 log_err("Calling getGlyphs(glyphs, status) on an empty layout did not return LE_NO_LAYOUT_ERROR.\n");
86 }
87
88 status = LE_NO_ERROR;
89 engine->getGlyphs(NULL, 0xFF000000L, status);
90
91 if (status != LE_ILLEGAL_ARGUMENT_ERROR) {
92 log_err("Calling getGlyphs(NULL, 0xFF000000L, status) did not return LE_ILLEGAL_ARGUMENT_ERROR.\n");
93 }
94
95 status = LE_NO_ERROR;
96 engine->getGlyphs(glyphs, 0xFF000000L, status);
97
98 if (status != LE_NO_LAYOUT_ERROR) {
99 log_err("Calling getGlyphs(glyphs, 0xFF000000L, status) on an empty layout did not return LE_NO_LAYOUT_ERROR.\n");
100 }
101
102 status = LE_NO_ERROR;
103 engine->getCharIndices(NULL, status);
104
105 if (status != LE_ILLEGAL_ARGUMENT_ERROR) {
106 log_err("Calling getCharIndices(NULL, status) did not return LE_ILLEGAL_ARGUMENT_ERROR.\n");
107 }
108
109 status = LE_NO_ERROR;
110 engine->getCharIndices(indices, status);
111
112 if (status != LE_NO_LAYOUT_ERROR) {
113 log_err("Calling getCharIndices(indices, status) on an empty layout did not return LE_NO_LAYOUT_ERROR.\n");
114 }
115
116 status = LE_NO_ERROR;
117 engine->getCharIndices(NULL, 1024, status);
118
119 if (status != LE_ILLEGAL_ARGUMENT_ERROR) {
120 log_err("Calling getCharIndices(NULL, 1024, status) did not return LE_ILLEGAL_ARGUMENT_ERROR.\n");
121 }
122
123 status = LE_NO_ERROR;
124 engine->getCharIndices(indices, 1024, status);
125
126 if (status != LE_NO_LAYOUT_ERROR) {
127 log_err("Calling getCharIndices(indices, 1024, status) on an empty layout did not return LE_NO_LAYOUT_ERROR.\n");
128 }
129
130 status = LE_NO_ERROR;
131 engine->getGlyphPositions(NULL, status);
132
133 if (status != LE_ILLEGAL_ARGUMENT_ERROR) {
134 log_err("Calling getGlyphPositions(NULL, status) did not return LE_ILLEGAL_ARGUMENT_ERROR.\n");
135 }
136
137 status = LE_NO_ERROR;
138 engine->getGlyphPositions(positions, status);
139
140 if (status != LE_NO_LAYOUT_ERROR) {
141 log_err("Calling getGlyphPositions(positions, status) on an empty layout did not return LE_NO_LAYOUT_ERROR.\n");
142 }
143
144 DELETE_ARRAY(positions);
145 DELETE_ARRAY(indices);
146 DELETE_ARRAY(glyphs);
147
148 status = LE_NO_ERROR;
149 glyphCount = engine->layoutChars(NULL, 0, 0, 0, false, 0.0, 0.0, status);
150
151 if (status != LE_ILLEGAL_ARGUMENT_ERROR) {
152 log_err("Calling layoutChars(NULL, 0, 0, 0, false, 0.0, 0.0, status) did not fail w/ LE_ILLEGAL_ARGUMENT_ERROR.\n");
153 }
154
155 LEUnicode chars[] = {
156 0x0045, 0x006E, 0x0067, 0x006C, 0x0069, 0x0073, 0x0068, 0x0020, // "English "
157 0x0645, 0x0627, 0x0646, 0x062A, 0x0648, 0x0634, // MEM ALIF KAF NOON TEH WAW SHEEN
158 0x0020, 0x0074, 0x0065, 0x0078, 0x0074, 0x02E // " text."
159 };
160
161 status = LE_NO_ERROR;
162 glyphCount = engine->layoutChars(chars, -1, 6, 20, true, 0.0, 0.0, status);
163
164 if (status != LE_ILLEGAL_ARGUMENT_ERROR) {
165 log_err("Calling layoutChars(chars, -1, 6, 20, true, 0.0, 0.0, status) did not fail w/ LE_ILLEGAL_ARGUMENT_ERROR.\n");
166 }
167
168 status = LE_NO_ERROR;
169 glyphCount = engine->layoutChars(chars, 8, -1, 20, true, 0.0, 0.0, status);
170
171 if (status != LE_ILLEGAL_ARGUMENT_ERROR) {
172 log_err("Calling layoutChars(chars, 8, -1, 20, true, 0.0, 0.0, status) did not fail w/ LE_ILLEGAL_ARGUMENT_ERROR.\n");
173 }
174
175 status = LE_NO_ERROR;
176 glyphCount = engine->layoutChars(chars, 8, 6, -1, true, 0.0, 0.0, status);
177
178 if (status != LE_ILLEGAL_ARGUMENT_ERROR) {
179 log_err("Calling layoutChars((chars, 8, 6, -1, true, 0.0, 0.0, status) did not fail w/ LE_ILLEGAL_ARGUMENT_ERROR.\n");
180 }
181
182 status = LE_NO_ERROR;
183 glyphCount = engine->layoutChars(chars, 8, 6, 10, true, 0.0, 0.0, status);
184
185 if (status != LE_ILLEGAL_ARGUMENT_ERROR) {
186 log_err("Calling layoutChars(chars, 8, 6, 10, true, 0.0, 0.0, status) did not fail w/ LE_ILLEGAL_ARGUMENT_ERROR.\n");
187 }
188
189 float x = 0.0, y = 0.0;
190
191 status = LE_NO_ERROR;
192 glyphCount = engine->layoutChars(chars, 8, 6, 20, true, 0.0, 0.0, status);
193
194 if (LE_FAILURE(status)) {
195 log_err("Calling layoutChars(chars, 8, 6, 20, true, 0.0, 0.0, status) failed.\n");
196 goto bail;
197 }
198
199 engine->getGlyphPosition(-1, x, y, status);
200
201 if (status != LE_INDEX_OUT_OF_BOUNDS_ERROR) {
202 log_err("Calling getGlyphPosition(-1, x, y, status) did not fail w/ LE_INDEX_OUT_OF_BOUNDS_ERROR.\n");
203 }
204
205 status = LE_NO_ERROR;
206 engine->getGlyphPosition(glyphCount + 1, x, y, status);
207
208 if (status != LE_INDEX_OUT_OF_BOUNDS_ERROR) {
209 log_err("Calling getGlyphPosition(glyphCount + 1, x, y, status) did not fail w/ LE_INDEX_OUT_OF_BOUNDS_ERROR.\n");
210 }
211
212 bail:
213 delete engine;
214 delete font;
215 }
216 U_CDECL_END
217
218 U_CDECL_BEGIN
FactoryTest(void)219 static void U_CALLCONV FactoryTest(void)
220 {
221 LEErrorCode status = LE_NO_ERROR;
222 SimpleFontInstance *font = new SimpleFontInstance(12, status);
223 LayoutEngine *engine = NULL;
224
225 for(le_int32 scriptCode = 0; scriptCode < scriptCodeCount; scriptCode += 1) {
226 status = LE_NO_ERROR;
227 engine = LayoutEngine::layoutEngineFactory(font, scriptCode, -1, status);
228
229 if (LE_FAILURE(status)) {
230 log_err("Could not create a LayoutEngine for script \'%s\'.\n", uscript_getShortName((UScriptCode)scriptCode));
231 }
232
233 delete engine;
234 }
235
236 delete font;
237 }
238 U_CDECL_END
239
240 U_CDECL_BEGIN
AccessTest(void)241 static void U_CALLCONV AccessTest(void)
242 {
243 LEErrorCode status = LE_NO_ERROR;
244 SimpleFontInstance *font = new SimpleFontInstance(12, status);
245 LayoutEngine *engine = LayoutEngine::layoutEngineFactory(font, arabScriptCode, -1, status);
246 le_int32 glyphCount;
247 LEGlyphID glyphs[6], extraBitGlyphs[6];
248 le_int32 biasedIndices[6], indices[6], glyph;
249 float positions[6 * 2 + 2];
250 LEUnicode chars[] = {
251 0x0045, 0x006E, 0x0067, 0x006C, 0x0069, 0x0073, 0x0068, 0x0020, // "English "
252 0x0645, 0x0627, 0x0646, 0x062A, 0x0648, 0x0634, // MEM ALIF KAF NOON TEH WAW SHEEN
253 0x0020, 0x0074, 0x0065, 0x0078, 0x0074, 0x02E // " text."
254 };
255
256 if (LE_FAILURE(status)) {
257 log_err("Could not create LayoutEngine.\n");
258 goto bail;
259 }
260
261 glyphCount = engine->layoutChars(chars, 8, 6, 20, true, 0.0, 0.0, status);
262
263 if (LE_FAILURE(status) || glyphCount != 6) {
264 log_err("layoutChars(chars, 8, 6, 20, true, 0.0, 0.0, status) failed.\n");
265 goto bail;
266 }
267
268 engine->getGlyphs(glyphs, status);
269 engine->getCharIndices(indices, status);
270 engine->getGlyphPositions(positions, status);
271
272 if (LE_FAILURE(status)) {
273 log_err("Could not get glyph, indices and position arrays.\n");
274 goto bail;
275 }
276
277 engine->getGlyphs(extraBitGlyphs, 0xFF000000L, status);
278
279 if (LE_FAILURE(status)) {
280 log_err("getGlyphs(extraBitGlyphs, 0xFF000000L, status); failed.\n");
281 } else {
282 for(glyph = 0; glyph < glyphCount; glyph += 1) {
283 if (extraBitGlyphs[glyph] != (glyphs[glyph] | 0xFF000000L)) {
284 log_err("extraBigGlyphs[%d] != glyphs[%d] | 0xFF000000L: %8X, %8X\n",
285 glyph, glyph, extraBitGlyphs[glyph], glyphs[glyph]);
286 break;
287 }
288 }
289 }
290
291 status = LE_NO_ERROR;
292 engine->getCharIndices(biasedIndices, 1024, status);
293
294 if (LE_FAILURE(status)) {
295 log_err("getCharIndices(biasedIndices, 1024, status) failed.\n");
296 } else {
297 for (glyph = 0; glyph < glyphCount; glyph += 1) {
298 if (biasedIndices[glyph] != (indices[glyph] + 1024)) {
299 log_err("biasedIndices[%d] != indices[%d] + 1024: %8X, %8X\n",
300 glyph, glyph, biasedIndices[glyph], indices[glyph]);
301 break;
302 }
303 }
304 }
305
306 status = LE_NO_ERROR;
307 for (glyph = 0; glyph <= glyphCount; glyph += 1) {
308 float x = 0.0, y = 0.0;
309
310 engine->getGlyphPosition(glyph, x, y, status);
311
312 if (LE_FAILURE(status)) {
313 log_err("getGlyphPosition(%d, x, y, status) failed.\n", glyph);
314 break;
315 }
316
317 if (x != positions[glyph*2] || y != positions[glyph*2 + 1]) {
318 log_err("getGlyphPosition(%d, x, y, status) returned bad position: (%f, %f) != (%f, %f)\n",
319 glyph, x, y, positions[glyph*2], positions[glyph*2 + 1]);
320 break;
321 }
322 }
323
324 bail:
325 delete engine;
326 delete font;
327 }
328 U_CDECL_END
329
compareResults(const char * testID,TestResult * expected,TestResult * actual)330 le_bool compareResults(const char *testID, TestResult *expected, TestResult *actual)
331 {
332 /* NOTE: we'll stop on the first failure 'cause once there's one error, it may cascade... */
333 if (actual->glyphCount != expected->glyphCount) {
334 log_err("Test %s: incorrect glyph count: exptected %d, got %d\n",
335 testID, expected->glyphCount, actual->glyphCount);
336 return false;
337 }
338
339 le_int32 i;
340
341 for (i = 0; i < actual->glyphCount; i += 1) {
342 if (actual->glyphs[i] != expected->glyphs[i]) {
343 log_err("Test %s: incorrect id for glyph %d: expected %4X, got %4X\n",
344 testID, i, expected->glyphs[i], actual->glyphs[i]);
345 return false;
346 }
347 }
348
349 for (i = 0; i < actual->glyphCount; i += 1) {
350 if (actual->indices[i] != expected->indices[i]) {
351 log_err("Test %s: incorrect index for glyph %d: expected %8X, got %8X\n",
352 testID, i, expected->indices[i], actual->indices[i]);
353 return false;
354 }
355 }
356
357 for (i = 0; i <= actual->glyphCount; i += 1) {
358 double xError = uprv_fabs(actual->positions[i * 2] - expected->positions[i * 2]);
359
360 if (xError > 0.0001) {
361 log_err("Test %s: incorrect x position for glyph %d: expected %f, got %f\n",
362 testID, i, expected->positions[i * 2], actual->positions[i * 2]);
363 return false;
364 }
365
366 double yError = uprv_fabs(actual->positions[i * 2 + 1] - expected->positions[i * 2 + 1]);
367
368 if (yError < 0) {
369 yError = -yError;
370 }
371
372 if (yError > 0.0001) {
373 log_err("Test %s: incorrect y position for glyph %d: expected %f, got %f\n",
374 testID, i, expected->positions[i * 2 + 1], actual->positions[i * 2 + 1]);
375 return false;
376 }
377 }
378
379 return true;
380 }
381
checkFontVersion(PortableFontInstance * fontInstance,const char * testVersionString,le_uint32 testChecksum,const char * testID)382 static void checkFontVersion(PortableFontInstance *fontInstance, const char *testVersionString,
383 le_uint32 testChecksum, const char *testID)
384 {
385 le_uint32 fontChecksum = fontInstance->getFontChecksum();
386
387 if (fontChecksum != testChecksum) {
388 const char *fontVersionString = fontInstance->getNameString(NAME_VERSION_STRING,
389 PLATFORM_MACINTOSH, MACINTOSH_ROMAN, MACINTOSH_ENGLISH);
390 const LEUnicode *uFontVersionString = NULL;
391
392 // The standard recommends that the Macintosh Roman/English name string be present, but
393 // if it's not, try the Microsoft Unicode/English string.
394 if (fontVersionString == NULL) {
395 uFontVersionString = fontInstance->getUnicodeNameString(NAME_VERSION_STRING,
396 PLATFORM_MICROSOFT, MICROSOFT_UNICODE_BMP, MICROSOFT_ENGLISH);
397 }
398
399 log_info("Test %s: this may not be the same font used to generate the test data.\n", testID);
400
401 if (uFontVersionString != NULL) {
402 log_info("Your font's version string is \"%S\"\n", uFontVersionString);
403 fontInstance->deleteNameString(uFontVersionString);
404 } else {
405 log_info("Your font's version string is \"%s\"\n", fontVersionString);
406 fontInstance->deleteNameString(fontVersionString);
407 }
408
409 log_info("The expected version string is \"%s\"\n", testVersionString);
410 log_info("If you see errors, they may be due to the version of the font you're using.\n");
411 }
412 }
413
414 /* Returns the path to icu/source/test/testdata/ */
getSourceTestData()415 const char *getSourceTestData() {
416 const char *srcDataDir = NULL;
417 #ifdef U_TOPSRCDIR
418 srcDataDir = U_TOPSRCDIR U_FILE_SEP_STRING "test" U_FILE_SEP_STRING "testdata" U_FILE_SEP_STRING;
419 #else
420 srcDataDir = ".." U_FILE_SEP_STRING ".." U_FILE_SEP_STRING "test" U_FILE_SEP_STRING "testdata" U_FILE_SEP_STRING;
421 FILE *f = fopen(".." U_FILE_SEP_STRING ".." U_FILE_SEP_STRING "test" U_FILE_SEP_STRING "testdata" U_FILE_SEP_STRING "rbbitst.txt", "r");
422
423 if (f != NULL) {
424 /* We're in icu/source/test/letest/ */
425 fclose(f);
426 } else {
427 /* We're in icu/source/test/letest/(Debug|Release) */
428 srcDataDir = ".." U_FILE_SEP_STRING ".." U_FILE_SEP_STRING ".." U_FILE_SEP_STRING "test"
429 U_FILE_SEP_STRING "testdata" U_FILE_SEP_STRING;
430 }
431 #endif
432
433 return srcDataDir;
434 }
435
getPath(char buffer[2048],const char * filename)436 const char *getPath(char buffer[2048], const char *filename) {
437 const char *testDataDirectory = getSourceTestData();
438
439 strcpy(buffer, testDataDirectory);
440 strcat(buffer, filename);
441
442 return buffer;
443 }
444
getHexArray(const UnicodeString & numbers,int32_t & arraySize)445 le_uint32 *getHexArray(const UnicodeString &numbers, int32_t &arraySize)
446 {
447 int32_t offset = -1;
448
449 arraySize = 1;
450 while((offset = numbers.indexOf(CH_COMMA, offset + 1)) >= 0) {
451 arraySize += 1;
452 }
453
454 le_uint32 *array = NEW_ARRAY(le_uint32, arraySize);
455 char number[16];
456 le_int32 count = 0;
457 le_int32 start = 0, end = 0;
458 le_int32 len = 0;
459
460 // trim leading whitespace
461 while(u_isUWhiteSpace(numbers[start])) {
462 start += 1;
463 }
464
465 while((end = numbers.indexOf(CH_COMMA, start)) >= 0) {
466 len = numbers.extract(start, end - start, number, ARRAY_SIZE(number), US_INV);
467 number[len] = '\0';
468 start = end + 1;
469
470 sscanf(number, "%x", &array[count++]);
471
472 // trim whitespace following the comma
473 while(u_isUWhiteSpace(numbers[start])) {
474 start += 1;
475 }
476 }
477
478 // trim trailing whitespace
479 end = numbers.length();
480 while(u_isUWhiteSpace(numbers[end - 1])) {
481 end -= 1;
482 }
483
484 len = numbers.extract(start, end - start, number, ARRAY_SIZE(number), US_INV);
485 number[len] = '\0';
486 sscanf(number, "%x", &array[count]);
487
488 return array;
489 }
490
getFloatArray(const UnicodeString & numbers,int32_t & arraySize)491 float *getFloatArray(const UnicodeString &numbers, int32_t &arraySize)
492 {
493 int32_t offset = -1;
494
495 arraySize = 1;
496 while((offset = numbers.indexOf(CH_COMMA, offset + 1)) >= 0) {
497 arraySize += 1;
498 }
499
500 float *array = NEW_ARRAY(float, arraySize);
501 char number[32];
502 le_int32 count = 0;
503 le_int32 start = 0, end = 0;
504 le_int32 len = 0;
505
506 // trim leading whitespace
507 while(u_isUWhiteSpace(numbers[start])) {
508 start += 1;
509 }
510
511 while((end = numbers.indexOf(CH_COMMA, start)) >= 0) {
512 len = numbers.extract(start, end - start, number, ARRAY_SIZE(number), US_INV);
513 number[len] = '\0';
514 start = end + 1;
515
516 sscanf(number, "%f", &array[count++]);
517
518 // trim whiteapce following the comma
519 while(u_isUWhiteSpace(numbers[start])) {
520 start += 1;
521 }
522 }
523
524 while(u_isUWhiteSpace(numbers[start])) {
525 start += 1;
526 }
527
528 // trim trailing whitespace
529 end = numbers.length();
530 while(u_isUWhiteSpace(numbers[end - 1])) {
531 end -= 1;
532 }
533
534 len = numbers.extract(start, end - start, number, ARRAY_SIZE(number), US_INV);
535 number[len] = '\0';
536 sscanf(number, "%f", &array[count]);
537
538 return array;
539 }
540
openFont(const char * fontName,const char * checksum,const char * version,const char * testID)541 LEFontInstance *openFont(const char *fontName, const char *checksum, const char *version, const char *testID)
542 {
543 char path[2048];
544 PortableFontInstance *font;
545 LEErrorCode fontStatus = LE_NO_ERROR;
546
547
548 font = new PortableFontInstance(getPath(path, fontName), 12, fontStatus);
549
550 if (LE_FAILURE(fontStatus)) {
551 log_info("Test %s: can't open font %s - test skipped.\n", testID, fontName);
552 delete font;
553 return NULL;
554 } else {
555 le_uint32 cksum = 0;
556
557 sscanf(checksum, "%x", &cksum);
558
559 checkFontVersion(font, version, cksum, testID);
560 }
561
562 return font;
563 }
564
565 U_CDECL_BEGIN
DataDrivenTest(void)566 static void U_CALLCONV DataDrivenTest(void)
567 {
568 #if !UCONFIG_NO_REGULAR_EXPRESSIONS
569 UErrorCode status = U_ZERO_ERROR;
570 char path[2048];
571 const char *testFilePath = getPath(path, "letest.xml");
572
573 UXMLParser *parser = UXMLParser::createParser(status);
574 UXMLElement *root = parser->parseFile(testFilePath, status);
575
576 if (root == NULL) {
577 log_err("Could not open the test data file: %s\n", testFilePath);
578 delete parser;
579 return;
580 }
581
582 UnicodeString test_case = UNICODE_STRING_SIMPLE("test-case");
583 UnicodeString test_text = UNICODE_STRING_SIMPLE("test-text");
584 UnicodeString test_font = UNICODE_STRING_SIMPLE("test-font");
585 UnicodeString result_glyphs = UNICODE_STRING_SIMPLE("result-glyphs");
586 UnicodeString result_indices = UNICODE_STRING_SIMPLE("result-indices");
587 UnicodeString result_positions = UNICODE_STRING_SIMPLE("result-positions");
588
589 // test-case attributes
590 UnicodeString id_attr = UNICODE_STRING_SIMPLE("id");
591 UnicodeString script_attr = UNICODE_STRING_SIMPLE("script");
592 UnicodeString lang_attr = UNICODE_STRING_SIMPLE("lang");
593
594 // test-font attributes
595 UnicodeString name_attr = UNICODE_STRING_SIMPLE("name");
596 UnicodeString ver_attr = UNICODE_STRING_SIMPLE("version");
597 UnicodeString cksum_attr = UNICODE_STRING_SIMPLE("checksum");
598
599 const UXMLElement *testCase;
600 int32_t tc = 0;
601
602 while((testCase = root->nextChildElement(tc)) != NULL) {
603 if (testCase->getTagName().compare(test_case) == 0) {
604 char *id = getCString(testCase->getAttribute(id_attr));
605 char *script = getCString(testCase->getAttribute(script_attr));
606 char *lang = getCString(testCase->getAttribute(lang_attr));
607 LEFontInstance *font = NULL;
608 const UXMLElement *element;
609 int32_t ec = 0;
610 int32_t charCount = 0;
611 int32_t typoFlags = 3; // kerning + ligatures...
612 UScriptCode scriptCode;
613 le_int32 languageCode = -1;
614 UnicodeString text, glyphs, indices, positions;
615 int32_t glyphCount = 0, indexCount = 0, positionCount = 0;
616 TestResult expected = {0, NULL, NULL, NULL};
617 TestResult actual = {0, NULL, NULL, NULL};
618 LEErrorCode success = LE_NO_ERROR;
619 LayoutEngine *engine = NULL;
620
621 uscript_getCode(script, &scriptCode, 1, &status);
622 if (LE_FAILURE(status)) {
623 log_err("invalid script name: %s.\n", script);
624 goto free_c_strings;
625 }
626
627 if (lang != NULL) {
628 languageCode = getLanguageCode(lang);
629
630 if (languageCode < 0) {
631 log_err("invalid language name: %s.\n", lang);
632 goto free_c_strings;
633 }
634 }
635
636 while((element = testCase->nextChildElement(ec)) != NULL) {
637 UnicodeString tag = element->getTagName();
638
639 // TODO: make sure that each element is only used once.
640 if (tag.compare(test_font) == 0) {
641 char *fontName = getCString(element->getAttribute(name_attr));
642 char *fontVer = getCString(element->getAttribute(ver_attr));
643 char *fontCksum = getCString(element->getAttribute(cksum_attr));
644
645 font = openFont(fontName, fontCksum, fontVer, id);
646 freeCString(fontCksum);
647 freeCString(fontVer);
648 freeCString(fontName);
649
650 if (font == NULL) {
651 // warning message already displayed...
652 goto free_c_strings;
653 }
654 } else if (tag.compare(test_text) == 0) {
655 text = element->getText(true);
656 charCount = text.length();
657 } else if (tag.compare(result_glyphs) == 0) {
658 glyphs = element->getText(true);
659 } else if (tag.compare(result_indices) == 0) {
660 indices = element->getText(true);
661 } else if (tag.compare(result_positions) == 0) {
662 positions = element->getText(true);
663 } else {
664 // an unknown tag...
665 char *cTag = getCString(&tag);
666
667 log_info("Test %s: unknown element with tag \"%s\"\n", id, cTag);
668 freeCString(cTag);
669 }
670 }
671
672 // TODO: make sure that the font, test-text, result-glyphs, result-indices and result-positions
673 // have all been provided
674 if (font == NULL) {
675 LEErrorCode fontStatus = LE_NO_ERROR;
676
677 font = new SimpleFontInstance(12, fontStatus);
678 typoFlags |= 0x80000000L; // use CharSubstitutionFilter...
679 }
680
681 expected.glyphs = (LEGlyphID *) getHexArray(glyphs, glyphCount);
682 expected.indices = (le_int32 *) getHexArray(indices, indexCount);
683 expected.positions = getFloatArray(positions, positionCount);
684
685 expected.glyphCount = glyphCount;
686
687 if (glyphCount < charCount || indexCount != glyphCount || positionCount < glyphCount * 2 + 2) {
688 log_err("Test %s: inconsistent input data: charCount = %d, glyphCount = %d, indexCount = %d, positionCount = %d\n",
689 id, charCount, glyphCount, indexCount, positionCount);
690 goto free_expected;
691 };
692
693 engine = LayoutEngine::layoutEngineFactory(font, scriptCode, languageCode, typoFlags, success);
694
695 if (LE_FAILURE(success)) {
696 log_err("Test %s: could not create a LayoutEngine.\n", id);
697 goto free_expected;
698 }
699
700 actual.glyphCount = engine->layoutChars(text.getBuffer(), 0, charCount, charCount, getRTL(text), 0, 0, success);
701
702 actual.glyphs = NEW_ARRAY(LEGlyphID, actual.glyphCount);
703 actual.indices = NEW_ARRAY(le_int32, actual.glyphCount);
704 actual.positions = NEW_ARRAY(float, actual.glyphCount * 2 + 2);
705
706 engine->getGlyphs(actual.glyphs, success);
707 engine->getCharIndices(actual.indices, success);
708 engine->getGlyphPositions(actual.positions, success);
709
710 compareResults(id, &expected, &actual);
711
712 DELETE_ARRAY(actual.positions);
713 DELETE_ARRAY(actual.indices);
714 DELETE_ARRAY(actual.glyphs);
715
716 delete engine;
717
718 log_verbose("OK - %4d glyphs: %s\n", actual.glyphCount, id);
719 free_expected:
720 DELETE_ARRAY(expected.positions);
721 DELETE_ARRAY(expected.indices);
722 DELETE_ARRAY(expected.glyphs);
723
724 delete font;
725
726 free_c_strings:
727 freeCString(lang);
728 freeCString(script);
729 freeCString(id);
730 }
731 }
732
733 delete root;
734 delete parser;
735 #endif
736 }
737 U_CDECL_END
738
739 U_CDECL_BEGIN
740 /*
741 * From ticket:5923:
742 *
743 * Build a paragraph that contains a mixture of left to right and right to left text.
744 * Break it into multiple lines and make sure that the glyphToCharMap for run in each
745 * line is correct.
746 *
747 * Note: it might be a good idea to also check the glyphs and positions for each run,
748 * that we get the expected number of runs per line and that the line breaks are where
749 * we expect them to be. Really, it would be a good idea to make a whole test suite
750 * for ParagraphLayout.
751 */
GlyphToCharTest(void)752 static void U_CALLCONV GlyphToCharTest(void)
753 {
754 #if !UCONFIG_NO_BREAK_ITERATION
755 LEErrorCode status = LE_NO_ERROR;
756 LEFontInstance *font;
757 FontRuns fontRuns(0);
758 ParagraphLayout *paragraphLayout;
759 const ParagraphLayout::Line *line;
760 /*
761 * This is the same text that's in <icu>/source/samples/layout/Sample.txt
762 */
763 LEUnicode chars[] = {
764 /*BOM*/ 0x0054, 0x0068, 0x0065, 0x0020, 0x004c, 0x0061, 0x0079,
765 0x006f, 0x0075, 0x0074, 0x0045, 0x006e, 0x0067, 0x0069, 0x006e,
766 0x0065, 0x0020, 0x0064, 0x006f, 0x0065, 0x0073, 0x0020, 0x0061,
767 0x006c, 0x006c, 0x0020, 0x0074, 0x0068, 0x0065, 0x0020, 0x0077,
768 0x006f, 0x0072, 0x006b, 0x0020, 0x006e, 0x0065, 0x0063, 0x0065,
769 0x0073, 0x0073, 0x0061, 0x0072, 0x0079, 0x0020, 0x0074, 0x006f,
770 0x0020, 0x0064, 0x0069, 0x0073, 0x0070, 0x006c, 0x0061, 0x0079,
771 0x0020, 0x0055, 0x006e, 0x0069, 0x0063, 0x006f, 0x0064, 0x0065,
772 0x0020, 0x0074, 0x0065, 0x0078, 0x0074, 0x0020, 0x0077, 0x0072,
773 0x0069, 0x0074, 0x0074, 0x0065, 0x006e, 0x0020, 0x0069, 0x006e,
774 0x0020, 0x006c, 0x0061, 0x006e, 0x0067, 0x0075, 0x0061, 0x0067,
775 0x0065, 0x0073, 0x0020, 0x0077, 0x0069, 0x0074, 0x0068, 0x0020,
776 0x0063, 0x006f, 0x006d, 0x0070, 0x006c, 0x0065, 0x0078, 0x0020,
777 0x0077, 0x0072, 0x0069, 0x0074, 0x0069, 0x006e, 0x0067, 0x0020,
778 0x0073, 0x0079, 0x0073, 0x0074, 0x0065, 0x006d, 0x0073, 0x0020,
779 0x0073, 0x0075, 0x0063, 0x0068, 0x0020, 0x0061, 0x0073, 0x0020,
780 0x0048, 0x0069, 0x006e, 0x0064, 0x0069, 0x0020, 0x0028, 0x0939,
781 0x093f, 0x0928, 0x094d, 0x0926, 0x0940, 0x0029, 0x0020, 0x0054,
782 0x0068, 0x0061, 0x0069, 0x0020, 0x0028, 0x0e44, 0x0e17, 0x0e22,
783 0x0029, 0x0020, 0x0061, 0x006e, 0x0064, 0x0020, 0x0041, 0x0072,
784 0x0061, 0x0062, 0x0069, 0x0063, 0x0020, 0x0028, 0x0627, 0x0644,
785 0x0639, 0x0631, 0x0628, 0x064a, 0x0629, 0x0029, 0x002e, 0x0020,
786 0x0048, 0x0065, 0x0072, 0x0065, 0x0027, 0x0073, 0x0020, 0x0061,
787 0x0020, 0x0073, 0x0061, 0x006d, 0x0070, 0x006c, 0x0065, 0x0020,
788 0x006f, 0x0066, 0x0020, 0x0073, 0x006f, 0x006d, 0x0065, 0x0020,
789 0x0074, 0x0065, 0x0078, 0x0074, 0x0020, 0x0077, 0x0072, 0x0069,
790 0x0074, 0x0074, 0x0065, 0x006e, 0x0020, 0x0069, 0x006e, 0x0020,
791 0x0053, 0x0061, 0x006e, 0x0073, 0x006b, 0x0072, 0x0069, 0x0074,
792 0x003a, 0x0020, 0x0936, 0x094d, 0x0930, 0x0940, 0x092e, 0x0926,
793 0x094d, 0x0020, 0x092d, 0x0917, 0x0935, 0x0926, 0x094d, 0x0917,
794 0x0940, 0x0924, 0x093e, 0x0020, 0x0905, 0x0927, 0x094d, 0x092f,
795 0x093e, 0x092f, 0x0020, 0x0905, 0x0930, 0x094d, 0x091c, 0x0941,
796 0x0928, 0x0020, 0x0935, 0x093f, 0x0937, 0x093e, 0x0926, 0x0020,
797 0x092f, 0x094b, 0x0917, 0x0020, 0x0927, 0x0943, 0x0924, 0x0930,
798 0x093e, 0x0937, 0x094d, 0x091f, 0x094d, 0x0930, 0x0020, 0x0909,
799 0x0935, 0x093e, 0x091a, 0x0964, 0x0020, 0x0927, 0x0930, 0x094d,
800 0x092e, 0x0915, 0x094d, 0x0937, 0x0947, 0x0924, 0x094d, 0x0930,
801 0x0947, 0x0020, 0x0915, 0x0941, 0x0930, 0x0941, 0x0915, 0x094d,
802 0x0937, 0x0947, 0x0924, 0x094d, 0x0930, 0x0947, 0x0020, 0x0938,
803 0x092e, 0x0935, 0x0947, 0x0924, 0x093e, 0x0020, 0x092f, 0x0941,
804 0x092f, 0x0941, 0x0924, 0x094d, 0x0938, 0x0935, 0x0903, 0x0020,
805 0x092e, 0x093e, 0x092e, 0x0915, 0x093e, 0x0903, 0x0020, 0x092a,
806 0x093e, 0x0923, 0x094d, 0x0921, 0x0935, 0x093e, 0x0936, 0x094d,
807 0x091a, 0x0948, 0x0935, 0x0020, 0x0915, 0x093f, 0x092e, 0x0915,
808 0x0941, 0x0930, 0x094d, 0x0935, 0x0924, 0x0020, 0x0938, 0x0902,
809 0x091c, 0x092f, 0x0020, 0x0048, 0x0065, 0x0072, 0x0065, 0x0027,
810 0x0073, 0x0020, 0x0061, 0x0020, 0x0073, 0x0061, 0x006d, 0x0070,
811 0x006c, 0x0065, 0x0020, 0x006f, 0x0066, 0x0020, 0x0073, 0x006f,
812 0x006d, 0x0065, 0x0020, 0x0074, 0x0065, 0x0078, 0x0074, 0x0020,
813 0x0077, 0x0072, 0x0069, 0x0074, 0x0074, 0x0065, 0x006e, 0x0020,
814 0x0069, 0x006e, 0x0020, 0x0041, 0x0072, 0x0061, 0x0062, 0x0069,
815 0x0063, 0x003a, 0x0020, 0x0623, 0x0633, 0x0627, 0x0633, 0x064b,
816 0x0627, 0x060c, 0x0020, 0x062a, 0x062a, 0x0639, 0x0627, 0x0645,
817 0x0644, 0x0020, 0x0627, 0x0644, 0x062d, 0x0648, 0x0627, 0x0633,
818 0x064a, 0x0628, 0x0020, 0x0641, 0x0642, 0x0637, 0x0020, 0x0645,
819 0x0639, 0x0020, 0x0627, 0x0644, 0x0623, 0x0631, 0x0642, 0x0627,
820 0x0645, 0x060c, 0x0020, 0x0648, 0x062a, 0x0642, 0x0648, 0x0645,
821 0x0020, 0x0628, 0x062a, 0x062e, 0x0632, 0x064a, 0x0646, 0x0020,
822 0x0627, 0x0644, 0x0623, 0x062d, 0x0631, 0x0641, 0x0020, 0x0648,
823 0x0627, 0x0644, 0x0645, 0x062d, 0x0627, 0x0631, 0x0641, 0x0020,
824 0x0627, 0x0644, 0x0623, 0x062e, 0x0631, 0x0649, 0x0020, 0x0628,
825 0x0639, 0x062f, 0x0020, 0x0623, 0x0646, 0x0020, 0x062a, 0x064f,
826 0x0639, 0x0637, 0x064a, 0x0020, 0x0631, 0x0642, 0x0645, 0x0627,
827 0x0020, 0x0645, 0x0639, 0x064a, 0x0646, 0x0627, 0x0020, 0x0644,
828 0x0643, 0x0644, 0x0020, 0x0648, 0x0627, 0x062d, 0x062f, 0x0020,
829 0x0645, 0x0646, 0x0647, 0x0627, 0x002e, 0x0020, 0x0648, 0x0642,
830 0x0628, 0x0644, 0x0020, 0x0627, 0x062e, 0x062a, 0x0631, 0x0627,
831 0x0639, 0x0020, 0x0022, 0x064a, 0x0648, 0x0646, 0x0650, 0x0643,
832 0x0648, 0x062f, 0x0022, 0x060c, 0x0020, 0x0643, 0x0627, 0x0646,
833 0x0020, 0x0647, 0x0646, 0x0627, 0x0643, 0x0020, 0x0645, 0x0626,
834 0x0627, 0x062a, 0x0020, 0x0627, 0x0644, 0x0623, 0x0646, 0x0638,
835 0x0645, 0x0629, 0x0020, 0x0644, 0x0644, 0x062a, 0x0634, 0x0641,
836 0x064a, 0x0631, 0x0020, 0x0648, 0x062a, 0x062e, 0x0635, 0x064a,
837 0x0635, 0x0020, 0x0647, 0x0630, 0x0647, 0x0020, 0x0627, 0x0644,
838 0x0623, 0x0631, 0x0642, 0x0627, 0x0645, 0x0020, 0x0644, 0x0644,
839 0x0645, 0x062d, 0x0627, 0x0631, 0x0641, 0x060c, 0x0020, 0x0648,
840 0x0644, 0x0645, 0x0020, 0x064a, 0x0648, 0x062c, 0x062f, 0x0020,
841 0x0646, 0x0638, 0x0627, 0x0645, 0x0020, 0x062a, 0x0634, 0x0641,
842 0x064a, 0x0631, 0x0020, 0x0648, 0x0627, 0x062d, 0x062f, 0x0020,
843 0x064a, 0x062d, 0x062a, 0x0648, 0x064a, 0x0020, 0x0639, 0x0644,
844 0x0649, 0x0020, 0x062c, 0x0645, 0x064a, 0x0639, 0x0020, 0x0627,
845 0x0644, 0x0645, 0x062d, 0x0627, 0x0631, 0x0641, 0x0020, 0x0627,
846 0x0644, 0x0636, 0x0631, 0x0648, 0x0631, 0x064a, 0x0629, 0x0020,
847 0x0061, 0x006e, 0x0064, 0x0020, 0x0068, 0x0065, 0x0072, 0x0065,
848 0x0027, 0x0073, 0x0020, 0x0061, 0x0020, 0x0073, 0x0061, 0x006d,
849 0x0070, 0x006c, 0x0065, 0x0020, 0x006f, 0x0066, 0x0020, 0x0073,
850 0x006f, 0x006d, 0x0065, 0x0020, 0x0074, 0x0065, 0x0078, 0x0074,
851 0x0020, 0x0077, 0x0072, 0x0069, 0x0074, 0x0074, 0x0065, 0x006e,
852 0x0020, 0x0069, 0x006e, 0x0020, 0x0054, 0x0068, 0x0061, 0x0069,
853 0x003a, 0x0020, 0x0e1a, 0x0e17, 0x0e17, 0x0e35, 0x0e48, 0x0e51,
854 0x0e1e, 0x0e32, 0x0e22, 0x0e38, 0x0e44, 0x0e0b, 0x0e42, 0x0e04,
855 0x0e25, 0x0e19, 0x0e42, 0x0e14, 0x0e42, 0x0e23, 0x0e18, 0x0e35,
856 0x0e2d, 0x0e32, 0x0e28, 0x0e31, 0x0e22, 0x0e2d, 0x0e22, 0x0e39,
857 0x0e48, 0x0e17, 0x0e48, 0x0e32, 0x0e21, 0x0e01, 0x0e25, 0x0e32,
858 0x0e07, 0x0e17, 0x0e38, 0x0e48, 0x0e07, 0x0e43, 0x0e2b, 0x0e0d,
859 0x0e48, 0x0e43, 0x0e19, 0x0e41, 0x0e04, 0x0e19, 0x0e0b, 0x0e31,
860 0x0e2a, 0x0e01, 0x0e31, 0x0e1a, 0x0e25, 0x0e38, 0x0e07, 0x0e40,
861 0x0e2e, 0x0e19, 0x0e23, 0x0e35, 0x0e0a, 0x0e32, 0x0e27, 0x0e44,
862 0x0e23, 0x0e48, 0x0e41, 0x0e25, 0x0e30, 0x0e1b, 0x0e49, 0x0e32,
863 0x0e40, 0x0e2d, 0x0e47, 0x0e21, 0x0e20, 0x0e23, 0x0e23, 0x0e22,
864 0x0e32, 0x0e0a, 0x0e32, 0x0e27, 0x0e44, 0x0e23, 0x0e48, 0x0e1a,
865 0x0e49, 0x0e32, 0x0e19, 0x0e02, 0x0e2d, 0x0e07, 0x0e1e, 0x0e27,
866 0x0e01, 0x0e40, 0x0e02, 0x0e32, 0x0e2b, 0x0e25, 0x0e31, 0x0e07,
867 0x0e40, 0x0e25, 0x0e47, 0x0e01, 0x0e40, 0x0e1e, 0x0e23, 0x0e32,
868 0x0e30, 0x0e44, 0x0e21, 0x0e49, 0x0e2a, 0x0e23, 0x0e49, 0x0e32,
869 0x0e07, 0x0e1a, 0x0e49, 0x0e32, 0x0e19, 0x0e15, 0x0e49, 0x0e2d,
870 0x0e07, 0x0e02, 0x0e19, 0x0e21, 0x0e32, 0x0e14, 0x0e49, 0x0e27,
871 0x0e22, 0x0e40, 0x0e01, 0x0e27, 0x0e35, 0x0e22, 0x0e19, 0x0e40,
872 0x0e1b, 0x0e47, 0x0e19, 0x0e23, 0x0e30, 0x0e22, 0x0e30, 0x0e17,
873 0x0e32, 0x0e07, 0x0e2b, 0x0e25, 0x0e32, 0x0e22, 0x0e44, 0x0e21,
874 0x0e25, 0x0e4c
875 };
876 le_int32 charCount = LE_ARRAY_SIZE(chars);
877 le_int32 charIndex = 0, lineNumber = 1;
878 const float lineWidth = 600;
879
880 font = new SimpleFontInstance(12, status);
881
882 if (LE_FAILURE(status)) {
883 goto finish;
884 }
885
886 fontRuns.add(font, charCount);
887
888 paragraphLayout = new ParagraphLayout(chars, charCount, &fontRuns, NULL, NULL, NULL, 0, false, status);
889
890 if (LE_FAILURE(status)) {
891 goto close_font;
892 }
893
894 paragraphLayout->reflow();
895 while ((line = paragraphLayout->nextLine(lineWidth)) != NULL) {
896 le_int32 runCount = line->countRuns();
897
898 for(le_int32 run = 0; run < runCount; run += 1) {
899 const ParagraphLayout::VisualRun *visualRun = line->getVisualRun(run);
900 le_int32 glyphCount = visualRun->getGlyphCount();
901 const le_int32 *glyphToCharMap = visualRun->getGlyphToCharMap();
902
903 if (visualRun->getDirection() == UBIDI_RTL) {
904 /*
905 * For a right to left run, make sure that the character indices
906 * increase from the right most glyph to the left most glyph. If
907 * there are any one to many glyph substitutions, we might get several
908 * glyphs in a row with the same character index.
909 */
910 for(le_int32 i = glyphCount - 1; i >= 0; i -= 1) {
911 le_int32 ix = glyphToCharMap[i];
912
913 if (ix != charIndex) {
914 if (ix != charIndex - 1) {
915 log_err("Bad glyph to char index for glyph %d on line %d: expected %d, got %d\n",
916 i, lineNumber, charIndex, ix);
917 goto close_paragraph; // once there's one error, we can't count on anything else...
918 }
919 } else {
920 charIndex += 1;
921 }
922 }
923 } else {
924 /*
925 * We can't just check the order of the character indices
926 * for left to right runs because Indic text might have been
927 * reordered. What we can do is find the minimum and maximum
928 * character indices in the run and make sure that the minimum
929 * is equal to charIndex and then advance charIndex to the maximum.
930 */
931 le_int32 minIndex = 0x7FFFFFFF, maxIndex = -1;
932
933 for(le_int32 i = 0; i < glyphCount; i += 1) {
934 le_int32 ix = glyphToCharMap[i];
935
936 if (ix > maxIndex) {
937 maxIndex = ix;
938 }
939
940 if (ix < minIndex) {
941 minIndex = ix;
942 }
943 }
944
945 if (minIndex != charIndex) {
946 log_err("Bad minIndex for run %d on line %d: expected %d, got %d\n",
947 run, lineNumber, charIndex, minIndex);
948 goto close_paragraph; // once there's one error, we can't count on anything else...
949 }
950
951 charIndex = maxIndex + 1;
952 }
953 }
954
955 lineNumber += 1;
956 }
957 close_paragraph:
958 delete paragraphLayout;
959
960 close_font:
961 delete font;
962
963 finish:
964 return;
965 #endif
966 }
967 U_CDECL_END
968
addAllTests(TestNode ** root)969 static void addAllTests(TestNode **root)
970 {
971 addTest(root, &ScriptTest, "api/ScriptTest");
972 addTest(root, &ParamTest, "api/ParameterTest");
973 addTest(root, &FactoryTest, "api/FactoryTest");
974 addTest(root, &AccessTest, "layout/AccessTest");
975 addTest(root, &DataDrivenTest, "layout/DataDrivenTest");
976 addTest(root, &GlyphToCharTest, "paragraph/GlyphToCharTest");
977
978 #ifndef USING_ICULEHB
979 addCTests(root);
980 #endif
981 }
982
983 /* returns the path to icu/source/data/out */
ctest_dataOutDir()984 static const char *ctest_dataOutDir()
985 {
986 static const char *dataOutDir = NULL;
987
988 if(dataOutDir) {
989 return dataOutDir;
990 }
991
992 /* U_TOPBUILDDIR is set by the makefiles on UNIXes when building cintltst and intltst
993 // to point to the top of the build hierarchy, which may or
994 // may not be the same as the source directory, depending on
995 // the configure options used. At any rate,
996 // set the data path to the built data from this directory.
997 // The value is complete with quotes, so it can be used
998 // as-is as a string constant.
999 */
1000 #if defined (U_TOPBUILDDIR)
1001 {
1002 dataOutDir = U_TOPBUILDDIR "data" U_FILE_SEP_STRING "out" U_FILE_SEP_STRING;
1003 }
1004 #else
1005
1006 /* On Windows, the file name obtained from __FILE__ includes a full path.
1007 * This file is "wherever\icu\source\test\cintltst\cintltst.c"
1008 * Change to "wherever\icu\source\data"
1009 */
1010 {
1011 static char p[sizeof(__FILE__) + 20];
1012 char *pBackSlash;
1013 int i;
1014
1015 strcpy(p, __FILE__);
1016 /* We want to back over three '\' chars. */
1017 /* Only Windows should end up here, so looking for '\' is safe. */
1018 for (i=1; i<=3; i++) {
1019 pBackSlash = strrchr(p, U_FILE_SEP_CHAR);
1020 if (pBackSlash != NULL) {
1021 *pBackSlash = 0; /* Truncate the string at the '\' */
1022 }
1023 }
1024
1025 if (pBackSlash != NULL) {
1026 /* We found and truncated three names from the path.
1027 * Now append "source\data" and set the environment
1028 */
1029 strcpy(pBackSlash, U_FILE_SEP_STRING "data" U_FILE_SEP_STRING "out" U_FILE_SEP_STRING);
1030 dataOutDir = p;
1031 }
1032 else {
1033 /* __FILE__ on MSVC7 does not contain the directory */
1034 FILE *file = fopen(".." U_FILE_SEP_STRING ".." U_FILE_SEP_STRING "data" U_FILE_SEP_STRING "Makefile.in", "r");
1035 if (file) {
1036 fclose(file);
1037 dataOutDir = ".." U_FILE_SEP_STRING ".." U_FILE_SEP_STRING "data" U_FILE_SEP_STRING "out" U_FILE_SEP_STRING;
1038 }
1039 else {
1040 dataOutDir = ".." U_FILE_SEP_STRING".." U_FILE_SEP_STRING".." U_FILE_SEP_STRING "data" U_FILE_SEP_STRING "out" U_FILE_SEP_STRING;
1041 }
1042 }
1043 }
1044 #endif
1045
1046 return dataOutDir;
1047 }
1048
1049 /* ctest_setICU_DATA - if the ICU_DATA environment variable is not already
1050 * set, try to deduce the directory in which ICU was built,
1051 * and set ICU_DATA to "icu/source/data" in that location.
1052 * The intent is to allow the tests to have a good chance
1053 * of running without requiring that the user manually set
1054 * ICU_DATA. Common data isn't a problem, since it is
1055 * picked up via a static (build time) reference, but the
1056 * tests dynamically load some data.
1057 */
ctest_setICU_DATA()1058 static void ctest_setICU_DATA() {
1059
1060 /* No location for the data dir was identifiable.
1061 * Add other fallbacks for the test data location here if the need arises
1062 */
1063 if (getenv("ICU_DATA") == NULL) {
1064 /* If ICU_DATA isn't set, set it to the usual location */
1065 u_setDataDirectory(ctest_dataOutDir());
1066 }
1067 }
1068
main(int argc,char * argv[])1069 int main(int argc, char* argv[])
1070 {
1071 int32_t nerrors = 0;
1072 TestNode *root = NULL;
1073 UErrorCode errorCode = U_ZERO_ERROR;
1074 UDate startTime, endTime;
1075 int32_t diffTime;
1076
1077 startTime = uprv_getUTCtime();
1078
1079 if (!initArgs(argc, argv, NULL, NULL)) {
1080 /* Error already displayed. */
1081 return -1;
1082 }
1083
1084 /* Check whether ICU will initialize without forcing the build data directory into
1085 * the ICU_DATA path. Success here means either the data dll contains data, or that
1086 * this test program was run with ICU_DATA set externally. Failure of this check
1087 * is normal when ICU data is not packaged into a shared library.
1088 *
1089 * Whether or not this test succeeds, we want to cleanup and reinitialize
1090 * with a data path so that data loading from individual files can be tested.
1091 */
1092 u_init(&errorCode);
1093
1094 if (U_FAILURE(errorCode)) {
1095 fprintf(stderr,
1096 "#### Note: ICU Init without build-specific setDataDirectory() failed.\n");
1097 }
1098
1099 u_cleanup();
1100 errorCode = U_ZERO_ERROR;
1101
1102 if (!initArgs(argc, argv, NULL, NULL)) {
1103 /* Error already displayed. */
1104 return -1;
1105 }
1106 /* Initialize ICU */
1107 ctest_setICU_DATA(); /* u_setDataDirectory() must happen Before u_init() */
1108 u_init(&errorCode);
1109
1110 if (U_FAILURE(errorCode)) {
1111 fprintf(stderr,
1112 "#### ERROR! %s: u_init() failed with status = \"%s\".\n"
1113 "*** Check the ICU_DATA environment variable and \n"
1114 "*** check that the data files are present.\n", argv[0], u_errorName(errorCode));
1115 return 1;
1116 }
1117
1118 addAllTests(&root);
1119 nerrors = runTestRequest(root, argc, argv);
1120
1121 cleanUpTestTree(root);
1122 u_cleanup();
1123
1124 endTime = uprv_getUTCtime();
1125 diffTime = (int32_t)(endTime - startTime);
1126 printf("Elapsed Time: %02d:%02d:%02d.%03d\n",
1127 (int)((diffTime%U_MILLIS_PER_DAY)/U_MILLIS_PER_HOUR),
1128 (int)((diffTime%U_MILLIS_PER_HOUR)/U_MILLIS_PER_MINUTE),
1129 (int)((diffTime%U_MILLIS_PER_MINUTE)/U_MILLIS_PER_SECOND),
1130 (int)(diffTime%U_MILLIS_PER_SECOND));
1131
1132 return nerrors;
1133 }
1134
1135