// © 2016 and later: Unicode, Inc. and others. // License & terms of use: http://www.unicode.org/copyright.html /****************************************************************************** * Copyright (C) 2008-2015, International Business Machines * Corporation and others. All Rights Reserved. ******************************************************************************* */ #include "unicode/utypes.h" #include "unicode/localpointer.h" #include "unicode/putil.h" #include "cstring.h" #include "toolutil.h" #include "uoptions.h" #include "uparse.h" #include "package.h" #include "pkg_icu.h" #include <stdio.h> #include <stdlib.h> #include <string.h> // read a file list -------------------------------------------------------- *** U_NAMESPACE_USE static const struct { const char *suffix; int32_t length; } listFileSuffixes[]={ { ".txt", 4 }, { ".lst", 4 }, { ".tmp", 4 } }; /* check for multiple text file suffixes to see if this list name is a text file name */ static UBool isListTextFile(const char *listname) { const char *listNameEnd=strchr(listname, 0); const char *suffix; int32_t i, length; for(i=0; i<UPRV_LENGTHOF(listFileSuffixes); ++i) { suffix=listFileSuffixes[i].suffix; length=listFileSuffixes[i].length; if((listNameEnd-listname)>length && 0==memcmp(listNameEnd-length, suffix, length)) { return TRUE; } } return FALSE; } /* * Read a file list. * If the listname ends with ".txt", then read the list file * (in the system/ invariant charset). * If the listname ends with ".dat", then read the ICU .dat package file. * Otherwise, read the file itself as a single-item list. */ U_CAPI Package * U_EXPORT2 readList(const char *filesPath, const char *listname, UBool readContents, Package *listPkgIn) { Package *listPkg = listPkgIn; FILE *file; const char *listNameEnd; if(listname==NULL || listname[0]==0) { fprintf(stderr, "missing list file\n"); return NULL; } if (listPkg == NULL) { listPkg=new Package(); if(listPkg==NULL) { fprintf(stderr, "icupkg: not enough memory\n"); exit(U_MEMORY_ALLOCATION_ERROR); } } listNameEnd=strchr(listname, 0); if(isListTextFile(listname)) { // read the list file char line[1024]; char *end; const char *start; file=fopen(listname, "r"); if(file==NULL) { fprintf(stderr, "icupkg: unable to open list file \"%s\"\n", listname); delete listPkg; exit(U_FILE_ACCESS_ERROR); } while(fgets(line, sizeof(line), file)) { // remove comments end=strchr(line, '#'); if(end!=NULL) { *end=0; } else { // remove trailing CR LF end=strchr(line, 0); while(line<end && (*(end-1)=='\r' || *(end-1)=='\n')) { *--end=0; } } // check first non-whitespace character and // skip empty lines and // skip lines starting with reserved characters start=u_skipWhitespace(line); if(*start==0 || NULL!=strchr(U_PKG_RESERVED_CHARS, *start)) { continue; } // take whitespace-separated items from the line for(;;) { // find whitespace after the item or the end of the line for(end=(char *)start; *end!=0 && *end!=' ' && *end!='\t'; ++end) {} if(*end==0) { // this item is the last one on the line end=NULL; } else { // the item is terminated by whitespace, terminate it with NUL *end=0; } if(readContents) { listPkg->addFile(filesPath, start); } else { listPkg->addItem(start); } // find the start of the next item or exit the loop if(end==NULL || *(start=u_skipWhitespace(end+1))==0) { break; } } } fclose(file); } else if((listNameEnd-listname)>4 && 0==memcmp(listNameEnd-4, ".dat", 4)) { // read the ICU .dat package // Accept a .dat file whose name differs from the ToC prefixes. listPkg->setAutoPrefix(); listPkg->readPackage(listname); } else { // list the single file itself if(readContents) { listPkg->addFile(filesPath, listname); } else { listPkg->addItem(listname); } } return listPkg; } U_CAPI int U_EXPORT2 writePackageDatFile(const char *outFilename, const char *outComment, const char *sourcePath, const char *addList, Package *pkg, char outType) { LocalPointer<Package> ownedPkg; LocalPointer<Package> addListPkg; if (pkg == NULL) { ownedPkg.adoptInstead(new Package); if(ownedPkg.isNull()) { fprintf(stderr, "icupkg: not enough memory\n"); return U_MEMORY_ALLOCATION_ERROR; } pkg = ownedPkg.getAlias(); addListPkg.adoptInstead(readList(sourcePath, addList, TRUE, NULL)); if(addListPkg.isValid()) { pkg->addItems(*addListPkg); } else { return U_ILLEGAL_ARGUMENT_ERROR; } } pkg->writePackage(outFilename, outType, outComment); return 0; }