]> git.saurik.com Git - apple/icu.git/blame - icuSources/tools/genrb/derb.cpp
ICU-66108.tar.gz
[apple/icu.git] / icuSources / tools / genrb / derb.cpp
CommitLineData
f3c0d7a5
A
1// © 2016 and later: Unicode, Inc. and others.
2// License & terms of use: http://www.unicode.org/copyright.html
b75a7d8f
A
3/*
4*******************************************************************************
5*
2ca993e8 6* Copyright (C) 1999-2016, International Business Machines
b75a7d8f
A
7* Corporation and others. All Rights Reserved.
8*
9*******************************************************************************
2ca993e8 10* file name: derb.cpp
f3c0d7a5 11* encoding: UTF-8
b75a7d8f
A
12* tab size: 8 (not used)
13* indentation:4
14*
15* created on: 2000sep6
16* created by: Vladimir Weinstein as an ICU workshop example
17* maintained by: Yves Arrouye <yves@realnames.com>
18*/
19
2ca993e8 20#include "unicode/stringpiece.h"
b75a7d8f 21#include "unicode/ucnv.h"
2ca993e8 22#include "unicode/unistr.h"
b75a7d8f 23#include "unicode/ustring.h"
374ca955 24#include "unicode/putil.h"
57a6839d 25#include "unicode/ustdio.h"
b75a7d8f 26
2ca993e8 27#include "charstr.h"
b75a7d8f
A
28#include "uresimp.h"
29#include "cmemory.h"
30#include "cstring.h"
31#include "uoptions.h"
32#include "toolutil.h"
33#include "ustrfmt.h"
34
57a6839d 35#if !UCONFIG_NO_FORMATTING
b75a7d8f 36
57a6839d 37#define DERB_VERSION "1.1"
b75a7d8f
A
38
39#define DERB_DEFAULT_TRUNC 80
40
b75a7d8f
A
41static const int32_t indentsize = 4;
42static int32_t truncsize = DERB_DEFAULT_TRUNC;
51004dcb 43static UBool opt_truncate = FALSE;
b75a7d8f 44
b75a7d8f
A
45static const char *getEncodingName(const char *encoding);
46static void reportError(const char *pname, UErrorCode *status, const char *when);
47static UChar *quotedString(const UChar *string);
2ca993e8
A
48static void printOutBundle(UFILE *out, UResourceBundle *resource, int32_t indent, const char *pname, UErrorCode *status);
49static void printString(UFILE *out, const UChar *str, int32_t len);
50static void printCString(UFILE *out, const char *str, int32_t len);
51static void printIndent(UFILE *out, int32_t indent);
52static void printHex(UFILE *out, uint8_t what);
b75a7d8f
A
53
54static UOption options[]={
55 UOPTION_HELP_H,
56 UOPTION_HELP_QUESTION_MARK,
57/* 2 */ UOPTION_ENCODING,
58/* 3 */ { "to-stdout", NULL, NULL, NULL, 'c', UOPT_NO_ARG, 0 } ,
59/* 4 */ { "truncate", NULL, NULL, NULL, 't', UOPT_OPTIONAL_ARG, 0 },
60/* 5 */ UOPTION_VERBOSE,
61/* 6 */ UOPTION_DESTDIR,
62/* 7 */ UOPTION_SOURCEDIR,
63/* 8 */ { "bom", NULL, NULL, NULL, 0, UOPT_NO_ARG, 0 },
64/* 9 */ UOPTION_ICUDATADIR,
65/* 10 */ UOPTION_VERSION,
57a6839d 66/* 11 */ { "suppressAliases", NULL, NULL, NULL, 'A', UOPT_NO_ARG, 0 },
b75a7d8f
A
67};
68
69static UBool verbose = FALSE;
70static UBool suppressAliases = FALSE;
57a6839d 71static UFILE *ustderr = NULL;
b75a7d8f
A
72
73extern int
74main(int argc, char* argv[]) {
75 const char *encoding = NULL;
76 const char *outputDir = NULL; /* NULL = no output directory, use current */
77 const char *inputDir = ".";
78 int tostdout = 0;
79 int prbom = 0;
80
81 const char *pname;
82
83 UResourceBundle *bundle = NULL;
b75a7d8f
A
84 int32_t i = 0;
85
b75a7d8f
A
86 const char* arg;
87
88 /* Get the name of tool. */
89 pname = uprv_strrchr(*argv, U_FILE_SEP_CHAR);
73c04bcf 90#if U_FILE_SEP_CHAR != U_FILE_ALT_SEP_CHAR
b75a7d8f 91 if (!pname) {
73c04bcf 92 pname = uprv_strrchr(*argv, U_FILE_ALT_SEP_CHAR);
b75a7d8f
A
93 }
94#endif
95 if (!pname) {
96 pname = *argv;
97 } else {
98 ++pname;
99 }
100
101 /* error handling, printing usage message */
2ca993e8 102 argc=u_parseArgs(argc, argv, UPRV_LENGTHOF(options), options);
b75a7d8f
A
103
104 /* error handling, printing usage message */
105 if(argc<0) {
106 fprintf(stderr,
107 "%s: error in command line argument \"%s\"\n", pname,
108 argv[-argc]);
109 }
110 if(argc<0 || options[0].doesOccur || options[1].doesOccur) {
111 fprintf(argc < 0 ? stderr : stdout,
112 "%csage: %s [ -h, -?, --help ] [ -V, --version ]\n"
113 " [ -v, --verbose ] [ -e, --encoding encoding ] [ --bom ]\n"
114 " [ -t, --truncate [ size ] ]\n"
115 " [ -s, --sourcedir source ] [ -d, --destdir destination ]\n"
116 " [ -i, --icudatadir directory ] [ -c, --to-stdout ]\n"
117 " [ -A, --suppressAliases]\n"
118 " bundle ...\n", argc < 0 ? 'u' : 'U',
119 pname);
120 return argc<0 ? U_ILLEGAL_ARGUMENT_ERROR : U_ZERO_ERROR;
121 }
122
123 if(options[10].doesOccur) {
124 fprintf(stderr,
125 "%s version %s (ICU version %s).\n"
126 "%s\n",
127 pname, DERB_VERSION, U_ICU_VERSION, U_COPYRIGHT_STRING);
128 return U_ZERO_ERROR;
129 }
130 if(options[2].doesOccur) {
131 encoding = options[2].value;
132 }
133
134 if (options[3].doesOccur) {
57a6839d
A
135 if(options[2].doesOccur) {
136 fprintf(stderr, "%s: Error: don't specify an encoding (-e) when writing to stdout (-c).\n", pname);
137 return 3;
138 }
139 tostdout = 1;
b75a7d8f
A
140 }
141
142 if(options[4].doesOccur) {
51004dcb 143 opt_truncate = TRUE;
b75a7d8f
A
144 if(options[4].value != NULL) {
145 truncsize = atoi(options[4].value); /* user defined printable size */
146 } else {
147 truncsize = DERB_DEFAULT_TRUNC; /* we'll use default omitting size */
148 }
149 } else {
51004dcb 150 opt_truncate = FALSE;
b75a7d8f
A
151 }
152
153 if(options[5].doesOccur) {
154 verbose = TRUE;
155 }
156
157 if (options[6].doesOccur) {
158 outputDir = options[6].value;
159 }
160
161 if(options[7].doesOccur) {
162 inputDir = options[7].value; /* we'll use users resources */
163 }
164
165 if (options[8].doesOccur) {
166 prbom = 1;
167 }
168
169 if (options[9].doesOccur) {
170 u_setDataDirectory(options[9].value);
171 }
172
173 if (options[11].doesOccur) {
174 suppressAliases = TRUE;
175 }
176
57a6839d
A
177 fflush(stderr); // use ustderr now.
178 ustderr = u_finit(stderr, NULL, NULL);
b75a7d8f
A
179
180 for (i = 1; i < argc; ++i) {
181 static const UChar sp[] = { 0x0020 }; /* " " */
b75a7d8f
A
182
183 arg = getLongPathname(argv[i]);
184
185 if (verbose) {
57a6839d 186 u_fprintf(ustderr, "processing bundle \"%s\"\n", argv[i]);
b75a7d8f
A
187 }
188
2ca993e8
A
189 icu::CharString locale;
190 UErrorCode status = U_ZERO_ERROR;
191 {
192 const char *p = findBasename(arg);
193 const char *q = uprv_strrchr(p, '.');
194 if (q == NULL) {
195 locale.append(p, status);
196 } else {
197 locale.append(p, (int32_t)(q - p), status);
198 }
b75a7d8f 199 }
2ca993e8
A
200 if (U_FAILURE(status)) {
201 return status;
b75a7d8f 202 }
b75a7d8f 203
2ca993e8
A
204 icu::CharString infile;
205 const char *thename = NULL;
206 UBool fromICUData = !uprv_strcmp(inputDir, "-");
207 if (!fromICUData) {
b75a7d8f 208 UBool absfilename = *arg == U_FILE_SEP_CHAR;
4388f060 209#if U_PLATFORM_HAS_WIN32_API
b75a7d8f
A
210 if (!absfilename) {
211 absfilename = (uprv_strlen(arg) > 2 && isalpha(arg[0])
73c04bcf 212 && arg[1] == ':' && arg[2] == U_FILE_SEP_CHAR);
b75a7d8f
A
213 }
214#endif
215 if (absfilename) {
216 thename = arg;
217 } else {
2ca993e8 218 const char *q = uprv_strrchr(arg, U_FILE_SEP_CHAR);
73c04bcf
A
219#if U_FILE_SEP_CHAR != U_FILE_ALT_SEP_CHAR
220 if (q == NULL) {
221 q = uprv_strrchr(arg, U_FILE_ALT_SEP_CHAR);
222 }
223#endif
2ca993e8 224 infile.append(inputDir, status);
73c04bcf 225 if(q != NULL) {
2ca993e8
A
226 infile.appendPathPart(icu::StringPiece(arg, (int32_t)(q - arg)), status);
227 }
228 if (U_FAILURE(status)) {
229 return status;
73c04bcf 230 }
2ca993e8 231 thename = infile.data();
b75a7d8f
A
232 }
233 }
b75a7d8f 234 if (thename) {
2ca993e8 235 bundle = ures_openDirect(thename, locale.data(), &status);
b75a7d8f 236 } else {
2ca993e8 237 bundle = ures_open(fromICUData ? 0 : inputDir, locale.data(), &status);
b75a7d8f 238 }
2ca993e8 239 if (U_SUCCESS(status)) {
57a6839d 240 UFILE *out = NULL;
b75a7d8f
A
241
242 const char *filename = 0;
243 const char *ext = 0;
244
2ca993e8
A
245 if (locale.isEmpty() || !tostdout) {
246 filename = findBasename(arg);
247 ext = uprv_strrchr(filename, '.');
b75a7d8f 248 if (!ext) {
2ca993e8 249 ext = uprv_strchr(filename, 0);
b75a7d8f
A
250 }
251 }
252
253 if (tostdout) {
57a6839d 254 out = u_get_stdout();
b75a7d8f 255 } else {
2ca993e8 256 icu::CharString thefile;
b75a7d8f 257 if (outputDir) {
2ca993e8 258 thefile.append(outputDir, status);
b75a7d8f 259 }
2ca993e8
A
260 thefile.appendPathPart(filename, status);
261 if (*ext) {
262 thefile.truncate(thefile.length() - (int32_t)uprv_strlen(ext));
263 }
264 thefile.append(".txt", status);
265 if (U_FAILURE(status)) {
266 return status;
b75a7d8f 267 }
b75a7d8f 268
2ca993e8 269 out = u_fopen(thefile.data(), "w", NULL, encoding);
b75a7d8f 270 if (!out) {
2ca993e8 271 u_fprintf(ustderr, "%s: couldn't create %s\n", pname, thefile.data());
57a6839d
A
272 u_fclose(ustderr);
273 return 4;
b75a7d8f
A
274 }
275 }
276
57a6839d
A
277 // now, set the callback.
278 ucnv_setFromUCallBack(u_fgetConverter(out), UCNV_FROM_U_CALLBACK_ESCAPE, UCNV_ESCAPE_C, 0, 0, &status);
279 if (U_FAILURE(status)) {
280 u_fprintf(ustderr, "%s: couldn't configure converter for encoding\n", pname);
281 u_fclose(ustderr);
282 if(!tostdout) {
283 u_fclose(out);
284 }
285 return 3;
b75a7d8f
A
286 }
287
57a6839d
A
288 if (prbom) { /* XXX: Should be done only for UTFs */
289 u_fputc(0xFEFF, out);
290 }
291 u_fprintf(out, "// -*- Coding: %s; -*-\n//\n", encoding ? encoding : getEncodingName(ucnv_getDefaultName()));
292 u_fprintf(out, "// This file was dumped by derb(8) from ");
b75a7d8f 293 if (thename) {
57a6839d 294 u_fprintf(out, "%s", thename);
b75a7d8f 295 } else if (fromICUData) {
2ca993e8 296 u_fprintf(out, "the ICU internal %s locale", locale.data());
b75a7d8f
A
297 }
298
57a6839d 299 u_fprintf(out, "\n// derb(8) by Vladimir Weinstein and Yves Arrouye\n\n");
b75a7d8f 300
2ca993e8
A
301 if (!locale.isEmpty()) {
302 u_fprintf(out, "%s", locale.data());
b75a7d8f 303 } else {
2ca993e8 304 u_fprintf(out, "%.*s%.*S", (int32_t)(ext - filename), filename, UPRV_LENGTHOF(sp), sp);
b75a7d8f 305 }
2ca993e8 306 printOutBundle(out, bundle, 0, pname, &status);
b75a7d8f 307
57a6839d
A
308 if (!tostdout) {
309 u_fclose(out);
b75a7d8f 310 }
73c04bcf
A
311 }
312 else {
b75a7d8f
A
313 reportError(pname, &status, "opening resource file");
314 }
315
316 ures_close(bundle);
317 }
318
b75a7d8f
A
319 return 0;
320}
321
322static UChar *quotedString(const UChar *string) {
323 int len = u_strlen(string);
324 int alen = len;
325 const UChar *sp;
326 UChar *newstr, *np;
327
328 for (sp = string; *sp; ++sp) {
329 switch (*sp) {
330 case '\n':
331 case 0x0022:
332 ++alen;
333 break;
334 }
335 }
336
2ca993e8 337 newstr = (UChar *) uprv_malloc((1 + alen) * U_SIZEOF_UCHAR);
b75a7d8f
A
338 for (sp = string, np = newstr; *sp; ++sp) {
339 switch (*sp) {
340 case '\n':
341 *np++ = 0x005C;
342 *np++ = 0x006E;
343 break;
344
345 case 0x0022:
346 *np++ = 0x005C;
2ca993e8 347 U_FALLTHROUGH;
b75a7d8f
A
348 default:
349 *np++ = *sp;
350 break;
351 }
352 }
353 *np = 0;
354
355 return newstr;
356}
357
358
2ca993e8 359static void printString(UFILE *out, const UChar *str, int32_t len) {
57a6839d 360 u_file_write(str, len, out);
b75a7d8f
A
361}
362
2ca993e8 363static void printCString(UFILE *out, const char *str, int32_t len) {
57a6839d
A
364 if(len==-1) {
365 u_fprintf(out, "%s", str);
366 } else {
367 u_fprintf(out, "%.*s", len, str);
368 }
b75a7d8f
A
369}
370
2ca993e8
A
371static void printIndent(UFILE *out, int32_t indent) {
372 icu::UnicodeString inchar(indent, 0x20, indent);
f3c0d7a5 373 printString(out, (const UChar *)inchar.getBuffer(), indent);
b75a7d8f
A
374}
375
2ca993e8 376static void printHex(UFILE *out, uint8_t what) {
b75a7d8f
A
377 static const char map[] = "0123456789ABCDEF";
378 UChar hex[2];
379
380 hex[0] = map[what >> 4];
381 hex[1] = map[what & 0xf];
382
2ca993e8 383 printString(out, hex, 2);
b75a7d8f 384}
b75a7d8f 385
2ca993e8
A
386static void printOutAlias(UFILE *out, UResourceBundle *parent, Resource r, const char *key, int32_t indent, const char *pname, UErrorCode *status) {
387 static const UChar cr[] = { 0xA }; // LF
b75a7d8f 388 int32_t len = 0;
729e4ab9 389 const UChar* thestr = res_getAlias(&(parent->fResData), r, &len);
b75a7d8f 390 UChar *string = quotedString(thestr);
51004dcb 391 if(opt_truncate && len > truncsize) {
b75a7d8f 392 char msg[128];
2ca993e8 393 printIndent(out, indent);
b75a7d8f
A
394 sprintf(msg, "// WARNING: this resource, size %li is truncated to %li\n",
395 (long)len, (long)truncsize/2);
2ca993e8 396 printCString(out, msg, -1);
b75a7d8f
A
397 len = truncsize;
398 }
399 if(U_SUCCESS(*status)) {
374ca955
A
400 static const UChar openStr[] = { 0x003A, 0x0061, 0x006C, 0x0069, 0x0061, 0x0073, 0x0020, 0x007B, 0x0020, 0x0022 }; /* ":alias { \"" */
401 static const UChar closeStr[] = { 0x0022, 0x0020, 0x007D, 0x0020 }; /* "\" } " */
2ca993e8 402 printIndent(out, indent);
b75a7d8f 403 if(key != NULL) {
2ca993e8 404 printCString(out, key, -1);
b75a7d8f 405 }
2ca993e8
A
406 printString(out, openStr, UPRV_LENGTHOF(openStr));
407 printString(out, string, len);
408 printString(out, closeStr, UPRV_LENGTHOF(closeStr));
b75a7d8f 409 if(verbose) {
2ca993e8 410 printCString(out, " // ALIAS", -1);
b75a7d8f 411 }
2ca993e8 412 printString(out, cr, UPRV_LENGTHOF(cr));
b75a7d8f
A
413 } else {
414 reportError(pname, status, "getting binary value");
415 }
46f4442e 416 uprv_free(string);
b75a7d8f
A
417}
418
2ca993e8 419static void printOutBundle(UFILE *out, UResourceBundle *resource, int32_t indent, const char *pname, UErrorCode *status)
b75a7d8f 420{
2ca993e8 421 static const UChar cr[] = { 0xA }; // LF
b75a7d8f
A
422
423/* int32_t noOfElements = ures_getSize(resource);*/
424 int32_t i = 0;
425 const char *key = ures_getKey(resource);
426
427 switch(ures_getType(resource)) {
729e4ab9 428 case URES_STRING :
b75a7d8f
A
429 {
430 int32_t len=0;
431 const UChar* thestr = ures_getString(resource, &len, status);
46f4442e 432 UChar *string = quotedString(thestr);
b75a7d8f
A
433
434 /* TODO: String truncation */
51004dcb 435 if(opt_truncate && len > truncsize) {
b75a7d8f 436 char msg[128];
2ca993e8 437 printIndent(out, indent);
b75a7d8f
A
438 sprintf(msg, "// WARNING: this resource, size %li is truncated to %li\n",
439 (long)len, (long)(truncsize/2));
2ca993e8 440 printCString(out, msg, -1);
b75a7d8f
A
441 len = truncsize/2;
442 }
2ca993e8 443 printIndent(out, indent);
b75a7d8f 444 if(key != NULL) {
374ca955
A
445 static const UChar openStr[] = { 0x0020, 0x007B, 0x0020, 0x0022 }; /* " { \"" */
446 static const UChar closeStr[] = { 0x0022, 0x0020, 0x007D }; /* "\" }" */
2ca993e8
A
447 printCString(out, key, (int32_t)uprv_strlen(key));
448 printString(out, openStr, UPRV_LENGTHOF(openStr));
449 printString(out, string, len);
450 printString(out, closeStr, UPRV_LENGTHOF(closeStr));
b75a7d8f 451 } else {
374ca955
A
452 static const UChar openStr[] = { 0x0022 }; /* "\"" */
453 static const UChar closeStr[] = { 0x0022, 0x002C }; /* "\"," */
b75a7d8f 454
2ca993e8
A
455 printString(out, openStr, UPRV_LENGTHOF(openStr));
456 printString(out, string, (int32_t)(u_strlen(string)));
457 printString(out, closeStr, UPRV_LENGTHOF(closeStr));
b75a7d8f
A
458 }
459
460 if(verbose) {
2ca993e8 461 printCString(out, "// STRING", -1);
b75a7d8f 462 }
2ca993e8 463 printString(out, cr, UPRV_LENGTHOF(cr));
b75a7d8f
A
464
465 uprv_free(string);
466 }
467 break;
468
729e4ab9 469 case URES_INT :
b75a7d8f 470 {
374ca955
A
471 static const UChar openStr[] = { 0x003A, 0x0069, 0x006E, 0x0074, 0x0020, 0x007B, 0x0020 }; /* ":int { " */
472 static const UChar closeStr[] = { 0x0020, 0x007D }; /* " }" */
b75a7d8f
A
473 UChar num[20];
474
2ca993e8 475 printIndent(out, indent);
b75a7d8f 476 if(key != NULL) {
2ca993e8 477 printCString(out, key, -1);
b75a7d8f 478 }
2ca993e8 479 printString(out, openStr, UPRV_LENGTHOF(openStr));
b75a7d8f 480 uprv_itou(num, 20, ures_getInt(resource, status), 10, 0);
2ca993e8
A
481 printString(out, num, u_strlen(num));
482 printString(out, closeStr, UPRV_LENGTHOF(closeStr));
b75a7d8f
A
483
484 if(verbose) {
2ca993e8 485 printCString(out, "// INT", -1);
b75a7d8f 486 }
2ca993e8 487 printString(out, cr, UPRV_LENGTHOF(cr));
b75a7d8f
A
488 break;
489 }
729e4ab9 490 case URES_BINARY :
b75a7d8f
A
491 {
492 int32_t len = 0;
493 const int8_t *data = (const int8_t *)ures_getBinary(resource, &len, status);
51004dcb 494 if(opt_truncate && len > truncsize) {
b75a7d8f 495 char msg[128];
2ca993e8 496 printIndent(out, indent);
b75a7d8f
A
497 sprintf(msg, "// WARNING: this resource, size %li is truncated to %li\n",
498 (long)len, (long)(truncsize/2));
2ca993e8 499 printCString(out, msg, -1);
b75a7d8f
A
500 len = truncsize;
501 }
502 if(U_SUCCESS(*status)) {
374ca955
A
503 static const UChar openStr[] = { 0x003A, 0x0062, 0x0069, 0x006E, 0x0061, 0x0072, 0x0079, 0x0020, 0x007B, 0x0020 }; /* ":binary { " */
504 static const UChar closeStr[] = { 0x0020, 0x007D, 0x0020 }; /* " } " */
2ca993e8 505 printIndent(out, indent);
b75a7d8f 506 if(key != NULL) {
2ca993e8 507 printCString(out, key, -1);
b75a7d8f 508 }
2ca993e8 509 printString(out, openStr, UPRV_LENGTHOF(openStr));
b75a7d8f 510 for(i = 0; i<len; i++) {
2ca993e8 511 printHex(out, *data++);
b75a7d8f 512 }
2ca993e8 513 printString(out, closeStr, UPRV_LENGTHOF(closeStr));
b75a7d8f 514 if(verbose) {
2ca993e8 515 printCString(out, " // BINARY", -1);
b75a7d8f 516 }
2ca993e8 517 printString(out, cr, UPRV_LENGTHOF(cr));
b75a7d8f
A
518 } else {
519 reportError(pname, status, "getting binary value");
520 }
521 }
522 break;
729e4ab9 523 case URES_INT_VECTOR :
b75a7d8f
A
524 {
525 int32_t len = 0;
526 const int32_t *data = ures_getIntVector(resource, &len, status);
527 if(U_SUCCESS(*status)) {
374ca955
A
528 static const UChar openStr[] = { 0x003A, 0x0069, 0x006E, 0x0074, 0x0076, 0x0065, 0x0063, 0x0074, 0x006F, 0x0072, 0x0020, 0x007B, 0x0020 }; /* ":intvector { " */
529 static const UChar closeStr[] = { 0x0020, 0x007D, 0x0020 }; /* " } " */
b75a7d8f
A
530 UChar num[20];
531
2ca993e8 532 printIndent(out, indent);
b75a7d8f 533 if(key != NULL) {
2ca993e8 534 printCString(out, key, -1);
b75a7d8f 535 }
2ca993e8 536 printString(out, openStr, UPRV_LENGTHOF(openStr));
b75a7d8f
A
537 for(i = 0; i < len - 1; i++) {
538 int32_t numLen = uprv_itou(num, 20, data[i], 10, 0);
539 num[numLen++] = 0x002C; /* ',' */
540 num[numLen++] = 0x0020; /* ' ' */
541 num[numLen] = 0;
2ca993e8 542 printString(out, num, u_strlen(num));
b75a7d8f
A
543 }
544 if(len > 0) {
545 uprv_itou(num, 20, data[len - 1], 10, 0);
2ca993e8 546 printString(out, num, u_strlen(num));
b75a7d8f 547 }
2ca993e8 548 printString(out, closeStr, UPRV_LENGTHOF(closeStr));
b75a7d8f 549 if(verbose) {
2ca993e8 550 printCString(out, "// INTVECTOR", -1);
b75a7d8f 551 }
2ca993e8 552 printString(out, cr, UPRV_LENGTHOF(cr));
b75a7d8f
A
553 } else {
554 reportError(pname, status, "getting int vector");
555 }
556 }
557 break;
729e4ab9
A
558 case URES_TABLE :
559 case URES_ARRAY :
b75a7d8f 560 {
374ca955
A
561 static const UChar openStr[] = { 0x007B }; /* "{" */
562 static const UChar closeStr[] = { 0x007D, '\n' }; /* "}\n" */
b75a7d8f
A
563
564 UResourceBundle *t = NULL;
565 ures_resetIterator(resource);
2ca993e8 566 printIndent(out, indent);
b75a7d8f 567 if(key != NULL) {
2ca993e8 568 printCString(out, key, -1);
b75a7d8f 569 }
2ca993e8 570 printString(out, openStr, UPRV_LENGTHOF(openStr));
b75a7d8f 571 if(verbose) {
729e4ab9 572 if(ures_getType(resource) == URES_TABLE) {
2ca993e8 573 printCString(out, "// TABLE", -1);
b75a7d8f 574 } else {
2ca993e8 575 printCString(out, "// ARRAY", -1);
b75a7d8f
A
576 }
577 }
2ca993e8 578 printString(out, cr, UPRV_LENGTHOF(cr));
b75a7d8f
A
579
580 if(suppressAliases == FALSE) {
581 while(U_SUCCESS(*status) && ures_hasNext(resource)) {
582 t = ures_getNextResource(resource, t, status);
583 if(U_SUCCESS(*status)) {
2ca993e8 584 printOutBundle(out, t, indent+indentsize, pname, status);
b75a7d8f
A
585 } else {
586 reportError(pname, status, "While processing table");
587 *status = U_ZERO_ERROR;
588 }
589 }
590 } else { /* we have to use low level access to do this */
729e4ab9
A
591 Resource r;
592 int32_t resSize = ures_getSize(resource);
593 UBool isTable = (UBool)(ures_getType(resource) == URES_TABLE);
594 for(i = 0; i < resSize; i++) {
b75a7d8f 595 /* need to know if it's an alias */
729e4ab9
A
596 if(isTable) {
597 r = res_getTableItemByIndex(&resource->fResData, resource->fRes, i, &key);
b75a7d8f 598 } else {
729e4ab9 599 r = res_getArrayItem(&resource->fResData, resource->fRes, i);
b75a7d8f
A
600 }
601 if(U_SUCCESS(*status)) {
729e4ab9 602 if(res_getPublicType(r) == URES_ALIAS) {
2ca993e8 603 printOutAlias(out, resource, r, key, indent+indentsize, pname, status);
b75a7d8f
A
604 } else {
605 t = ures_getByIndex(resource, i, t, status);
2ca993e8 606 printOutBundle(out, t, indent+indentsize, pname, status);
b75a7d8f
A
607 }
608 } else {
609 reportError(pname, status, "While processing table");
610 *status = U_ZERO_ERROR;
611 }
612 }
613 }
614
2ca993e8
A
615 printIndent(out, indent);
616 printString(out, closeStr, UPRV_LENGTHOF(closeStr));
b75a7d8f
A
617 ures_close(t);
618 }
619 break;
620 default:
621 break;
622 }
623
624}
625
626static const char *getEncodingName(const char *encoding) {
627 UErrorCode err;
628 const char *enc;
629
630 err = U_ZERO_ERROR;
631 if (!(enc = ucnv_getStandardName(encoding, "MIME", &err))) {
632 err = U_ZERO_ERROR;
633 if (!(enc = ucnv_getStandardName(encoding, "IANA", &err))) {
340931cb 634 // do nothing
b75a7d8f
A
635 }
636 }
637
638 return enc;
639}
640
641static void reportError(const char *pname, UErrorCode *status, const char *when) {
57a6839d 642 u_fprintf(ustderr, "%s: error %d while %s: %s\n", pname, *status, when, u_errorName(*status));
b75a7d8f
A
643}
644
57a6839d
A
645#else
646extern int
647main(int argc, char* argv[]) {
648 /* Changing stdio.h ustdio.h requires that formatting not be disabled. */
649 return 3;
650}
651#endif /* !UCONFIG_NO_FORMATTING */
652
b75a7d8f
A
653/*
654 * Local Variables:
655 * indent-tabs-mode: nil
656 * End:
657 */