Home | History | Annotate | Download | only in icuinfo
      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-2016, International Business Machines
      7 *   Corporation and others.  All Rights Reserved.
      8 *
      9 *******************************************************************************
     10 *   file name:  icuinfo.cpp
     11 *   encoding:   UTF-8
     12 *   tab size:   8 (not used)
     13 *   indentation:4
     14 *
     15 *   created on: 2009-2010
     16 *   created by: Steven R. Loomis
     17 *
     18 *   This program shows some basic info about the current ICU.
     19 */
     20 
     21 #include <stdio.h>
     22 #include <stdlib.h>
     23 #include "unicode/utypes.h"
     24 #include "unicode/putil.h"
     25 #include "unicode/uclean.h"
     26 #include "udbgutil.h"
     27 #include "unewdata.h"
     28 #include "cmemory.h"
     29 #include "cstring.h"
     30 #include "uoptions.h"
     31 #include "toolutil.h"
     32 #include "icuplugimp.h"
     33 #include <unicode/uloc.h>
     34 #include <unicode/ucnv.h>
     35 #include "unicode/ucal.h"
     36 #include <unicode/ulocdata.h>
     37 #include "putilimp.h"
     38 #include "unicode/uchar.h"
     39 
     40 static UOption options[]={
     41   /*0*/ UOPTION_HELP_H,
     42   /*1*/ UOPTION_HELP_QUESTION_MARK,
     43   /*2*/ UOPTION_ICUDATADIR,
     44   /*3*/ UOPTION_VERBOSE,
     45   /*4*/ UOPTION_DEF("list-plugins", 'L', UOPT_NO_ARG), // may be a no-op if disabled
     46   /*5*/ UOPTION_DEF("milisecond-time", 'm', UOPT_NO_ARG),
     47   /*6*/ UOPTION_DEF("cleanup", 'K', UOPT_NO_ARG),
     48   /*7*/ UOPTION_DEF("xml", 'x', UOPT_REQUIRES_ARG),
     49 };
     50 
     51 static UErrorCode initStatus = U_ZERO_ERROR;
     52 static UBool icuInitted = FALSE;
     53 
     54 static void do_init() {
     55     if(!icuInitted) {
     56       u_init(&initStatus);
     57       icuInitted = TRUE;
     58     }
     59 }
     60 
     61 
     62 void cmd_millis()
     63 {
     64   printf("Milliseconds since Epoch: %.0f\n", uprv_getUTCtime());
     65 }
     66 
     67 void cmd_version(UBool /* noLoad */, UErrorCode &errorCode)
     68 {
     69 
     70     do_init();
     71 
     72     udbg_writeIcuInfo(stdout); /* print the XML format */
     73 
     74     union {
     75         uint8_t byte;
     76         uint16_t word;
     77     } u;
     78     u.word=0x0100;
     79     if(U_IS_BIG_ENDIAN==u.byte) {
     80       //printf("U_IS_BIG_ENDIAN: %d\n", U_IS_BIG_ENDIAN);
     81     } else {
     82         fprintf(stderr, "  error: U_IS_BIG_ENDIAN=%d != %d=actual 'is big endian'\n",
     83                 U_IS_BIG_ENDIAN, u.byte);
     84         errorCode=U_INTERNAL_PROGRAM_ERROR;
     85     }
     86 
     87     if(U_SIZEOF_WCHAR_T==sizeof(wchar_t)) {
     88       //printf("U_SIZEOF_WCHAR_T: %d\n", U_SIZEOF_WCHAR_T);
     89     } else {
     90         fprintf(stderr, "  error: U_SIZEOF_WCHAR_T=%d != %d=sizeof(wchar_t)\n",
     91                 U_SIZEOF_WCHAR_T, (int)sizeof(wchar_t));
     92         errorCode=U_INTERNAL_PROGRAM_ERROR;
     93     }
     94 
     95     int charsetFamily;
     96     if('A'==0x41) {
     97         charsetFamily=U_ASCII_FAMILY;
     98     } else if('A'==0xc1) {
     99         charsetFamily=U_EBCDIC_FAMILY;
    100     } else {
    101         charsetFamily=-1;  // unknown
    102     }
    103     if(U_CHARSET_FAMILY==charsetFamily) {
    104       //printf("U_CHARSET_FAMILY: %d\n", U_CHARSET_FAMILY);
    105     } else {
    106         fprintf(stderr, "  error: U_CHARSET_FAMILY=%d != %d=actual charset family\n",
    107                 U_CHARSET_FAMILY, charsetFamily);
    108         errorCode=U_INTERNAL_PROGRAM_ERROR;
    109     }
    110 
    111     printf("\n\nICU Initialization returned: %s\n", u_errorName(initStatus));
    112 
    113 
    114 #if UCONFIG_ENABLE_PLUGINS
    115 #if U_ENABLE_DYLOAD
    116     const char *pluginFile = uplug_getPluginFile();
    117     printf("Plugin file is: %s\n", (pluginFile&&*pluginFile)?pluginFile:"(not set. try setting ICU_PLUGINS to a directory.)");
    118 #else
    119     fprintf(stderr, "Dynamic Loading: is disabled. No plugins will be loaded at start-up.\n");
    120 #endif
    121 #else
    122     fprintf(stderr, "Plugins are disabled.\n");
    123 #endif
    124 }
    125 
    126 void cmd_cleanup()
    127 {
    128     u_cleanup();
    129     fprintf(stdout, "ICU u_cleanup() called.\n");
    130 }
    131 
    132 
    133 void cmd_listplugins() {
    134 #if UCONFIG_ENABLE_PLUGINS
    135     int32_t i;
    136     UPlugData *plug;
    137 
    138     do_init();
    139     printf("ICU Initialized: u_init() returned %s\n", u_errorName(initStatus));
    140 
    141     printf("Plugins: \n");
    142     printf(    "# %6s   %s \n",
    143                        "Level",
    144                        "Name" );
    145     printf(    "    %10s:%-10s\n",
    146                        "Library",
    147                        "Symbol"
    148             );
    149 
    150 
    151     printf(    "       config| (configuration string)\n");
    152     printf(    " >>>   Error          | Explanation \n");
    153     printf(    "-----------------------------------\n");
    154 
    155     for(i=0;(plug=uplug_getPlugInternal(i))!=NULL;i++) {
    156         UErrorCode libStatus = U_ZERO_ERROR;
    157         const char *name = uplug_getPlugName(plug);
    158         const char *sym = uplug_getSymbolName(plug);
    159         const char *lib = uplug_getLibraryName(plug, &libStatus);
    160         const char *config = uplug_getConfiguration(plug);
    161         UErrorCode loadStatus = uplug_getPlugLoadStatus(plug);
    162         const char *message = NULL;
    163 
    164         printf("\n#%d  %-6s %s \n",
    165             i+1,
    166             udbg_enumName(UDBG_UPlugLevel,(int32_t)uplug_getPlugLevel(plug)),
    167             name!=NULL?(*name?name:"this plugin did not call uplug_setPlugName()"):"(null)"
    168         );
    169         printf("    plugin| %10s:%-10s\n",
    170             (U_SUCCESS(libStatus)?(lib!=NULL?lib:"(null)"):u_errorName(libStatus)),
    171             sym!=NULL?sym:"(null)"
    172         );
    173 
    174         if(config!=NULL&&*config) {
    175             printf("    config| %s\n", config);
    176         }
    177 
    178         switch(loadStatus) {
    179             case U_PLUGIN_CHANGED_LEVEL_WARNING:
    180                 message = "Note: This plugin changed the system level (by allocating memory or calling something which does). Later plugins may not load.";
    181                 break;
    182 
    183             case U_PLUGIN_DIDNT_SET_LEVEL:
    184                 message = "Error: This plugin did not call uplug_setPlugLevel during QUERY.";
    185                 break;
    186 
    187             case U_PLUGIN_TOO_HIGH:
    188                 message = "Error: This plugin couldn't load because the system level was too high. Try loading this plugin earlier.";
    189                 break;
    190 
    191             case U_ZERO_ERROR:
    192                 message = NULL; /* no message */
    193                 break;
    194             default:
    195                 if(U_FAILURE(loadStatus)) {
    196                     message = "error loading:";
    197                 } else {
    198                     message = "warning during load:";
    199                 }
    200         }
    201 
    202         if(message!=NULL) {
    203             printf("\\\\\\ status| %s\n"
    204                    "/// %s\n", u_errorName(loadStatus), message);
    205         }
    206 
    207     }
    208 	if(i==0) {
    209 		printf("No plugins loaded.\n");
    210 	}
    211 #endif
    212 }
    213 
    214 
    215 
    216 extern int
    217 main(int argc, char* argv[]) {
    218     UErrorCode errorCode = U_ZERO_ERROR;
    219     UBool didSomething = FALSE;
    220 
    221     /* preset then read command line options */
    222     argc=u_parseArgs(argc, argv, UPRV_LENGTHOF(options), options);
    223 
    224     /* error handling, printing usage message */
    225     if(argc<0) {
    226         fprintf(stderr,
    227             "error in command line argument \"%s\"\n",
    228             argv[-argc]);
    229     }
    230     if( options[0].doesOccur || options[1].doesOccur) {
    231       fprintf(stderr, "%s: Output information about the current ICU\n", argv[0]);
    232       fprintf(stderr, "Options:\n"
    233               " -h     or  --help                 - Print this help message.\n"
    234               " -m     or  --millisecond-time     - Print the current UTC time in milliseconds.\n"
    235               " -d <dir>   or  --icudatadir <dir> - Set the ICU Data Directory\n"
    236               " -v                                - Print version and configuration information about ICU\n"
    237 #if UCONFIG_ENABLE_PLUGINS
    238               " -L         or  --list-plugins     - List and diagnose issues with ICU Plugins\n"
    239 #endif
    240               " -K         or  --cleanup          - Call u_cleanup() before exitting (will attempt to unload plugins)\n"
    241               "\n"
    242               "If no arguments are given, the tool will print ICU version and configuration information.\n"
    243               );
    244       fprintf(stderr, "International Components for Unicode %s\n%s\n", U_ICU_VERSION, U_COPYRIGHT_STRING );
    245       return argc<0 ? U_ILLEGAL_ARGUMENT_ERROR : U_ZERO_ERROR;
    246     }
    247 
    248     if(options[2].doesOccur) {
    249       u_setDataDirectory(options[2].value);
    250     }
    251 
    252     if(options[5].doesOccur) {
    253       cmd_millis();
    254       didSomething=TRUE;
    255     }
    256     if(options[4].doesOccur) {
    257       cmd_listplugins();
    258       didSomething = TRUE;
    259     }
    260 
    261     if(options[3].doesOccur) {
    262       cmd_version(FALSE, errorCode);
    263       didSomething = TRUE;
    264     }
    265 
    266     if(options[7].doesOccur) {  /* 2nd part of version: cleanup */
    267       FILE *out = fopen(options[7].value, "w");
    268       if(out==NULL) {
    269         fprintf(stderr,"ERR: can't write to XML file %s\n", options[7].value);
    270         return 1;
    271       }
    272       /* todo: API for writing DTD? */
    273       fprintf(out, "<?xml version=\"1.0\" encoding=\"UTF-8\" ?>\n");
    274       udbg_writeIcuInfo(out);
    275       fclose(out);
    276       didSomething = TRUE;
    277     }
    278 
    279     if(options[6].doesOccur) {  /* 2nd part of version: cleanup */
    280       cmd_cleanup();
    281       didSomething = TRUE;
    282     }
    283 
    284     if(!didSomething) {
    285       cmd_version(FALSE, errorCode);  /* at least print the version # */
    286     }
    287 
    288     return U_FAILURE(errorCode);
    289 }
    290