/* ********************************************************************** * Copyright (C) 2010-2011, International Business Machines * Corporation and others. All Rights Reserved. ********************************************************************** * file name: dicttrieperf.cpp * encoding: US-ASCII * tab size: 8 (not used) * indentation:4 * * created on: 2010dec09 * created by: Markus W. Scherer * * Performance test program for dictionary-type tries. * * Usage from within /test/perf/dicttrieperf/ : * (Linux) * make * export LD_LIBRARY_PATH=../../../lib:../../../stubdata:../../../tools/ctestfw * ./dicttrieperf --sourcedir /data/out/tmp --passes 3 --iterations 1000 * or * ./dicttrieperf -f /source/data/brkitr/thaidict.txt --passes 3 --iterations 250 */ #include #include #include "unicode/uperf.h" #include "unicode/utext.h" #include "bytestrie.h" #include "bytestriebuilder.h" #include "charstr.h" #include "package.h" #include "toolutil.h" #include "triedict.h" #include "ucbuf.h" // struct ULine #include "ucharstrie.h" #include "ucharstriebuilder.h" #include "uoptions.h" #include "uvectr32.h" #define LENGTHOF(array) (int32_t)(sizeof(array)/sizeof((array)[0])) // Test object. class DictionaryTriePerfTest : public UPerfTest { public: DictionaryTriePerfTest(int32_t argc, const char *argv[], UErrorCode &status) : UPerfTest(argc, argv, NULL, 0, "", status), numTextLines(0) { if(hasFile()) { getLines(status); for(int32_t i=0; i=0x41) { ++numTextLines; // Remove trailing CR LF. int32_t len=lines[i].len; UChar c; while(len>0 && ((c=lines[i].name[len-1])==0xa || c==0xd)) { --len; } lines[i].len=len; } } } } virtual UPerfFunction *runIndexedTest(int32_t index, UBool exec, const char *&name, char *par=NULL); const char *getSourceDir() const { return sourceDir; } UBool hasFile() const { return ucharBuf!=NULL; } const ULine *getCachedLines() const { return lines; } int32_t getNumLines() const { return numLines; } int32_t numTextLines; // excluding comment lines }; // Performance test function object. // Loads icudt46l.dat (or whatever its current versioned filename) // from the -s or --sourcedir path. class PackageLookup : public UPerfFunction { protected: PackageLookup(const DictionaryTriePerfTest &perf) { IcuToolErrorCode errorCode("PackageLookup()"); CharString filename(perf.getSourceDir(), errorCode); int32_t filenameLength=filename.length(); if(filenameLength>0 && filename[filenameLength-1]!=U_FILE_SEP_CHAR && filename[filenameLength-1]!=U_FILE_ALT_SEP_CHAR) { filename.append(U_FILE_SEP_CHAR, errorCode); } filename.append(U_ICUDATA_NAME, errorCode); filename.append(".dat", errorCode); pkg.readPackage(filename.data()); } public: virtual ~PackageLookup() {} // virtual void call(UErrorCode* pErrorCode) { ... } virtual long getOperationsPerIteration() { return pkg.getItemCount(); } // virtual long getEventsPerIteration(); protected: Package pkg; }; struct TOCEntry { int32_t nameOffset, dataOffset; }; // Similar to ICU 4.6 offsetTOCLookupFn() (in ucmndata.c). static int32_t simpleBinarySearch(const char *s, const char *names, const TOCEntry *toc, int32_t count) { int32_t start=0; int32_t limit=count; int32_t lastNumber=limit; for(;;) { int32_t number=(start+limit)/2; if(lastNumber==number) { // have we moved? return -1; // not found } lastNumber=number; int32_t cmp=strcmp(s, names+toc[number].nameOffset); if(cmp<0) { limit=number; } else if(cmp>0) { start=number; } else { // found s return number; } } } class BinarySearchPackageLookup : public PackageLookup { public: BinarySearchPackageLookup(const DictionaryTriePerfTest &perf) : PackageLookup(perf) { IcuToolErrorCode errorCode("BinarySearchPackageLookup()"); int32_t count=pkg.getItemCount(); toc=new TOCEntry[count]; for(int32_t i=0; iname; itemNames.append("icudt46l/", errorCode); itemNames.append(name, strlen(name)+1, errorCode); } printf("size of item names: %6ld\n", (long)itemNames.length()); printf("size of TOC: %6ld\n", (long)(count*8)); printf("total index size: %6ld\n", (long)(itemNames.length()+count*8)); } virtual ~BinarySearchPackageLookup() { delete[] toc; } virtual void call(UErrorCode * /*pErrorCode*/) { int32_t count=pkg.getItemCount(); const char *itemNameChars=itemNames.data(); const char *name=itemNameChars; for(int32_t i=0; iname; int32_t offset=itemNames.length(); itemNames.append("icudt46l/", errorCode); itemNames.append(name, -1, errorCode); // As value, set the data item index. // In a real implementation, we would use that to get the // start and limit offset of the data item. StringPiece fullName(itemNames.toStringPiece()); fullName.remove_prefix(offset); builder.add(fullName, i, errorCode); // NUL-terminate the name for call() to find the next one. itemNames.append(0, errorCode); } int32_t length=builder.build(USTRINGTRIE_BUILD_SMALL, errorCode).length(); printf("size of BytesTrie: %6ld\n", (long)length); // count+1: +1 for the last-item limit offset which we should have always had printf("size of dataOffsets:%6ld\n", (long)((count+1)*4)); printf("total index size: %6ld\n", (long)(length+(count+1)*4)); } virtual ~BytesTriePackageLookup() {} virtual void call(UErrorCode *pErrorCode) { int32_t count=pkg.getItemCount(); const char *nameTrieBytes=builder.build(USTRINGTRIE_BUILD_SMALL, *pErrorCode).data(); const char *name=itemNames.data(); for(int32_t i=0; i/source/data/brkitr/thaidict.txt. class DictLookup : public UPerfFunction { public: DictLookup(const DictionaryTriePerfTest &perfTest) : perf(perfTest) {} virtual long getOperationsPerIteration() { return perf.numTextLines; } protected: const DictionaryTriePerfTest &perf; }; class CompactTrieDictLookup : public DictLookup { public: CompactTrieDictLookup(const DictionaryTriePerfTest &perfTest) : DictLookup(perfTest), ctd(NULL) { IcuToolErrorCode errorCode("UCharsTrieDictLookup()"); // U+0E1C is the median code unit, from // the UCharsTrie root node (split-branch node) for thaidict.txt. MutableTrieDictionary builder(0xe1c, errorCode); const ULine *lines=perf.getCachedLines(); int32_t numLines=perf.getNumLines(); for(int32_t i=0; idataSize(); printf("size of CompactTrieDict: %6ld bytes\n", (long)length); } virtual ~CompactTrieDictLookup() { delete ctd; } virtual void call(UErrorCode *pErrorCode) { UText text=UTEXT_INITIALIZER; int32_t lengths[20]; const ULine *lines=perf.getCachedLines(); int32_t numLines=perf.getNumLines(); for(int32_t i=0; imatches(&text, lines[i].len, lengths, count, LENGTHOF(lengths)); if(count==0 || lengths[count-1]!=lines[i].len) { fprintf(stderr, "word %ld (0-based) not found\n", (long)i); } } } protected: CompactTrieDictionary *ctd; }; // Closely imitate CompactTrieDictionary::matches(). // Note: CompactTrieDictionary::matches() is part of its trie implementation, // and while it loops over the text, it knows the current state. // By contrast, this implementation uses UCharsTrie API functions that have to // check the trie state each time and load/store state in the object. // (Whether it hasNext() and whether it is in the middle of a linear-match node.) static int32_t ucharsTrieMatches(UCharsTrie &trie, UText *text, int32_t textLimit, int32_t *lengths, int &count, int limit ) { UChar32 c=utext_next32(text); // Notes: // a) CompactTrieDictionary::matches() does not check for U_SENTINEL. // b) It also ignores non-BMP code points by casting to UChar! if(c<0) { return 0; } // Should be firstForCodePoint() but CompactTrieDictionary // handles only code units. UStringTrieResult result=trie.first(c); int32_t numChars=1; count=0; for(;;) { if(USTRINGTRIE_HAS_VALUE(result)) { if(count=textLimit) { // Note: Why do we have both a text limit and a UText that knows its length? break; } UChar32 c=utext_next32(text); // Notes: // a) CompactTrieDictionary::matches() does not check for U_SENTINEL. // b) It also ignores non-BMP code points by casting to UChar! if(c<0) { break; } ++numChars; // Should be nextForCodePoint() but CompactTrieDictionary // handles only code units. result=trie.next(c); } #if 0 // Note: CompactTrieDictionary::matches() comments say that it leaves the UText // after the longest prefix match and returns the number of characters // that were matched. if(index!=lastMatch) { utext_setNativeIndex(text, lastMatch); } return lastMatch-start; // However, it does not do either of these, so I am not trying to // imitate it (or its docs) 100%. #endif return numChars; } class UCharsTrieDictLookup : public DictLookup { public: UCharsTrieDictLookup(const DictionaryTriePerfTest &perfTest) : DictLookup(perfTest) { IcuToolErrorCode errorCode("UCharsTrieDictLookup()"); const ULine *lines=perf.getCachedLines(); int32_t numLines=perf.getNumLines(); for(int32_t i=0; i=0) { str.append((char)b, errorCode); } else { fprintf(stderr, "thaiWordToBytes(): unable to encode U+%04X as a byte\n", c); return FALSE; } } return TRUE; } class BytesTrieDictLookup : public DictLookup { public: BytesTrieDictLookup(const DictionaryTriePerfTest &perfTest) : DictLookup(perfTest), noDict(FALSE) { IcuToolErrorCode errorCode("BytesTrieDictLookup()"); CharString str; const ULine *lines=perf.getCachedLines(); int32_t numLines=perf.getNumLines(); for(int32_t i=0; i=textLimit) { break; } UChar32 c=utext_next32(text); if(c<0) { break; } ++numChars; result=trie.next(thaiCharToByte(c)); } return numChars; } class BytesTrieDictMatches : public BytesTrieDictLookup { public: BytesTrieDictMatches(const DictionaryTriePerfTest &perfTest) : BytesTrieDictLookup(perfTest) {} virtual void call(UErrorCode *pErrorCode) { if(noDict) { return; } BytesTrie trie(builder.build(USTRINGTRIE_BUILD_SMALL, *pErrorCode).data()); UText text=UTEXT_INITIALIZER; int32_t lengths[20]; const ULine *lines=perf.getCachedLines(); int32_t numLines=perf.getNumLines(); for(int32_t i=0; i