/*********************************************************************** * Copyright (C) 2016 and later: Unicode, Inc. and others. * License & terms of use: http://www.unicode.org/copyright.html#License *********************************************************************** *********************************************************************** * COPYRIGHT: * Copyright (C) 2001-2012 IBM, Inc. All Rights Reserved. * ***********************************************************************/ /******************************************************************************** * * File CALLCOLL.C * * Modification History: * Name Description * Andy Heninger First Version * ********************************************************************************* */ // // This program tests string collation and sort key generation performance. // Three APIs can be teste: ICU C , Unix strcoll, strxfrm and Windows LCMapString // A file of names is required as input, one per line. It must be in utf-8 or utf-16 format, // and include a byte order mark. Either LE or BE format is OK. // const char gUsageString[] = "usage: collperf options...\n" "-help Display this message.\n" "-file file_name utf-16 format file of names.\n" "-locale name ICU locale to use. Default is en_US\n" "-rules file_name Collation rules file (overrides locale)\n" "-langid 0x1234 Windows Language ID number. Default to value for -locale option\n" " see http://msdn.microsoft.com/library/psdk/winbase/nls_8xo3.htm\n" "-win Run test using Windows native services. (ICU is default)\n" "-unix Run test using Unix strxfrm, strcoll services.\n" "-uselen Use API with string lengths. Default is null-terminated strings\n" "-usekeys Run tests using sortkeys rather than strcoll\n" "-strcmp Run tests using u_strcmp rather than strcoll\n" "-strcmpCPO Run tests using u_strcmpCodePointOrder rather than strcoll\n" "-loop nnnn Loopcount for test. Adjust for reasonable total running time.\n" "-iloop n Inner Loop Count. Default = 1. Number of calls to function\n" " under test at each call point. For measuring test overhead.\n" "-terse Terse numbers-only output. Intended for use by scripts.\n" "-french French accent ordering\n" "-frenchoff No French accent ordering (for use with French locales.)\n" "-norm Normalizing mode on\n" "-shifted Shifted mode\n" "-lower Lower case first\n" "-upper Upper case first\n" "-case Enable separate case level\n" "-level n Sort level, 1 to 5, for Primary, Secndary, Tertiary, Quaternary, Identical\n" "-keyhist Produce a table sort key size vs. string length\n" "-binsearch Binary Search timing test\n" "-keygen Sort Key Generation timing test\n" "-qsort Quicksort timing test\n" "-iter Iteration Performance Test\n" "-dump Display strings, sort keys and CEs.\n" ; #include <stdio.h> #include <string.h> #include <stdlib.h> #include <math.h> #include <locale.h> #include <errno.h> #include <unicode/utypes.h> #include <unicode/ucol.h> #include <unicode/ucoleitr.h> #include <unicode/uloc.h> #include <unicode/ustring.h> #include <unicode/ures.h> #include <unicode/uchar.h> #include <unicode/ucnv.h> #include <unicode/utf8.h> #ifdef WIN32 #include <windows.h> #else // // Stubs for Windows API functions when building on UNIXes. // typedef int DWORD; inline int CompareStringW(DWORD, DWORD, UChar *, int, UChar *, int) {return 0;} #include <sys/time.h> unsigned long timeGetTime() { struct timeval t; gettimeofday(&t, 0); unsigned long val = t.tv_sec * 1000; // Let it overflow. Who cares. val += t.tv_usec / 1000; return val; } inline int LCMapStringW(DWORD, DWORD, UChar *, int, UChar *, int) {return 0;} const int LCMAP_SORTKEY = 0; #define MAKELCID(a,b) 0 const int SORT_DEFAULT = 0; #endif // // Command line option variables // These global variables are set according to the options specified // on the command line by the user. char * opt_fName = 0; const char * opt_locale = "en_US"; int opt_langid = 0; // Defaults to value corresponding to opt_locale. char * opt_rules = 0; UBool opt_help = FALSE; int opt_loopCount = 1; int opt_iLoopCount = 1; UBool opt_terse = FALSE; UBool opt_qsort = FALSE; UBool opt_binsearch = FALSE; UBool opt_icu = TRUE; UBool opt_win = FALSE; // Run with Windows native functions. UBool opt_unix = FALSE; // Run with UNIX strcoll, strxfrm functions. UBool opt_uselen = FALSE; UBool opt_usekeys = FALSE; UBool opt_strcmp = FALSE; UBool opt_strcmpCPO = FALSE; UBool opt_norm = FALSE; UBool opt_keygen = FALSE; UBool opt_french = FALSE; UBool opt_frenchoff = FALSE; UBool opt_shifted = FALSE; UBool opt_lower = FALSE; UBool opt_upper = FALSE; UBool opt_case = FALSE; int opt_level = 0; UBool opt_keyhist = FALSE; UBool opt_itertest = FALSE; UBool opt_dump = FALSE; // // Definitions for the command line options // struct OptSpec { const char *name; enum {FLAG, NUM, STRING} type; void *pVar; }; OptSpec opts[] = { {"-file", OptSpec::STRING, &opt_fName}, {"-locale", OptSpec::STRING, &opt_locale}, {"-langid", OptSpec::NUM, &opt_langid}, {"-rules", OptSpec::STRING, &opt_rules}, {"-qsort", OptSpec::FLAG, &opt_qsort}, {"-binsearch", OptSpec::FLAG, &opt_binsearch}, {"-iter", OptSpec::FLAG, &opt_itertest}, {"-win", OptSpec::FLAG, &opt_win}, {"-unix", OptSpec::FLAG, &opt_unix}, {"-uselen", OptSpec::FLAG, &opt_uselen}, {"-usekeys", OptSpec::FLAG, &opt_usekeys}, {"-strcmp", OptSpec::FLAG, &opt_strcmp}, {"-strcmpCPO", OptSpec::FLAG, &opt_strcmpCPO}, {"-norm", OptSpec::FLAG, &opt_norm}, {"-french", OptSpec::FLAG, &opt_french}, {"-frenchoff", OptSpec::FLAG, &opt_frenchoff}, {"-shifted", OptSpec::FLAG, &opt_shifted}, {"-lower", OptSpec::FLAG, &opt_lower}, {"-upper", OptSpec::FLAG, &opt_upper}, {"-case", OptSpec::FLAG, &opt_case}, {"-level", OptSpec::NUM, &opt_level}, {"-keyhist", OptSpec::FLAG, &opt_keyhist}, {"-keygen", OptSpec::FLAG, &opt_keygen}, {"-loop", OptSpec::NUM, &opt_loopCount}, {"-iloop", OptSpec::NUM, &opt_iLoopCount}, {"-terse", OptSpec::FLAG, &opt_terse}, {"-dump", OptSpec::FLAG, &opt_dump}, {"-help", OptSpec::FLAG, &opt_help}, {"-?", OptSpec::FLAG, &opt_help}, {0, OptSpec::FLAG, 0} }; //--------------------------------------------------------------------------- // // Global variables pointing to and describing the test file // //--------------------------------------------------------------------------- // // struct Line // // Each line from the source file (containing a name, presumably) gets // one of these structs. // struct Line { UChar *name; int len; char *winSortKey; char *icuSortKey; char *unixSortKey; char *unixName; }; Line *gFileLines; // Ptr to array of Line structs, one per line in the file. int gNumFileLines; UCollator *gCol; DWORD gWinLCID; Line **gSortedLines; Line **gRandomLines; int gCount; //--------------------------------------------------------------------------- // // ProcessOptions() Function to read the command line options. // //--------------------------------------------------------------------------- UBool ProcessOptions(int argc, const char **argv, OptSpec opts[]) { int i; int argNum; const char *pArgName; OptSpec *pOpt; for (argNum=1; argNum<argc; argNum++) { pArgName = argv[argNum]; for (pOpt = opts; pOpt->name != 0; pOpt++) { if (strcmp(pOpt->name, pArgName) == 0) { switch (pOpt->type) { case OptSpec::FLAG: *(UBool *)(pOpt->pVar) = TRUE; break; case OptSpec::STRING: argNum ++; if (argNum >= argc) { fprintf(stderr, "value expected for \"%s\" option.\n", pOpt->name); return FALSE; } *(const char **)(pOpt->pVar) = argv[argNum]; break; case OptSpec::NUM: argNum ++; if (argNum >= argc) { fprintf(stderr, "value expected for \"%s\" option.\n", pOpt->name); return FALSE; } char *endp; i = strtol(argv[argNum], &endp, 0); if (endp == argv[argNum]) { fprintf(stderr, "integer value expected for \"%s\" option.\n", pOpt->name); return FALSE; } *(int *)(pOpt->pVar) = i; } break; } } if (pOpt->name == 0) { fprintf(stderr, "Unrecognized option \"%s\"\n", pArgName); return FALSE; } } return TRUE; } //--------------------------------------------------------------------------------------- // // Comparison functions for use by qsort. // // Six flavors, ICU or Windows, SortKey or String Compare, Strings with length // or null terminated. // //--------------------------------------------------------------------------------------- int ICUstrcmpK(const void *a, const void *b) { gCount++; int t = strcmp((*(Line **)a)->icuSortKey, (*(Line **)b)->icuSortKey); return t; } int ICUstrcmpL(const void *a, const void *b) { gCount++; UCollationResult t; t = ucol_strcoll(gCol, (*(Line **)a)->name, (*(Line **)a)->len, (*(Line **)b)->name, (*(Line **)b)->len); if (t == UCOL_LESS) return -1; if (t == UCOL_GREATER) return +1; return 0; } int ICUstrcmp(const void *a, const void *b) { gCount++; UCollationResult t; t = ucol_strcoll(gCol, (*(Line **)a)->name, -1, (*(Line **)b)->name, -1); if (t == UCOL_LESS) return -1; if (t == UCOL_GREATER) return +1; return 0; } int Winstrcmp(const void *a, const void *b) { gCount++; int t; t = CompareStringW(gWinLCID, 0, (*(Line **)a)->name, -1, (*(Line **)b)->name, -1); return t-2; } int UNIXstrcmp(const void *a, const void *b) { gCount++; int t; t = strcoll((*(Line **)a)->unixName, (*(Line **)b)->unixName); return t; } int WinstrcmpL(const void *a, const void *b) { gCount++; int t; t = CompareStringW(gWinLCID, 0, (*(Line **)a)->name, (*(Line **)a)->len, (*(Line **)b)->name, (*(Line **)b)->len); return t-2; } int WinstrcmpK(const void *a, const void *b) { gCount++; int t = strcmp((*(Line **)a)->winSortKey, (*(Line **)b)->winSortKey); return t; } //--------------------------------------------------------------------------------------- // // Function for sorting the names (lines) into a random order. // Order is based on a hash of the ICU Sort key for the lines // The randomized order is used as input for the sorting timing tests. // //--------------------------------------------------------------------------------------- int ICURandomCmp(const void *a, const void *b) { char *ask = (*(Line **)a)->icuSortKey; char *bsk = (*(Line **)b)->icuSortKey; int aVal = 0; int bVal = 0; int retVal; while (*ask != 0) { aVal += aVal*37 + *ask++; } while (*bsk != 0) { bVal += bVal*37 + *bsk++; } retVal = -1; if (aVal == bVal) { retVal = 0; } else if (aVal > bVal) { retVal = 1; } return retVal; } //--------------------------------------------------------------------------------------- // // doKeyGen() Key Generation Timing Test // //--------------------------------------------------------------------------------------- void doKeyGen() { int line; int loops = 0; int iLoop; int t; int len=-1; // Adjust loop count to compensate for file size. Should be order n double dLoopCount = double(opt_loopCount) * (1000. / double(gNumFileLines)); int adj_loopCount = int(dLoopCount); if (adj_loopCount < 1) adj_loopCount = 1; unsigned long startTime = timeGetTime(); if (opt_win) { for (loops=0; loops<adj_loopCount; loops++) { for (line=0; line < gNumFileLines; line++) { if (opt_uselen) { len = gFileLines[line].len; } for (iLoop=0; iLoop < opt_iLoopCount; iLoop++) { t=LCMapStringW(gWinLCID, LCMAP_SORTKEY, gFileLines[line].name, len, (unsigned short *)gFileLines[line].winSortKey, 5000); // TODO something with length. } } } } else if (opt_icu) { for (loops=0; loops<adj_loopCount; loops++) { for (line=0; line < gNumFileLines; line++) { if (opt_uselen) { len = gFileLines[line].len; } for (iLoop=0; iLoop < opt_iLoopCount; iLoop++) { t = ucol_getSortKey(gCol, gFileLines[line].name, len, (unsigned char *)gFileLines[line].icuSortKey, 5000); } } } } else if (opt_unix) { for (loops=0; loops<adj_loopCount; loops++) { for (line=0; line < gNumFileLines; line++) { for (iLoop=0; iLoop < opt_iLoopCount; iLoop++) { t = strxfrm(gFileLines[line].unixSortKey, gFileLines[line].unixName, 5000); } } } } unsigned long elapsedTime = timeGetTime() - startTime; int ns = (int)(float(1000000) * (float)elapsedTime / (float)(adj_loopCount*gNumFileLines)); if (opt_terse == FALSE) { printf("Sort Key Generation: total # of keys = %d\n", loops*gNumFileLines); printf("Sort Key Generation: time per key = %d ns\n", ns); } else { printf("%d, ", ns); } int totalKeyLen = 0; int totalChars = 0; for (line=0; line<gNumFileLines; line++) { totalChars += u_strlen(gFileLines[line].name); if (opt_win) { totalKeyLen += strlen(gFileLines[line].winSortKey); } else if (opt_icu) { totalKeyLen += strlen(gFileLines[line].icuSortKey); } else if (opt_unix) { totalKeyLen += strlen(gFileLines[line].unixSortKey); } } if (opt_terse == FALSE) { printf("Key Length / character = %f\n", (float)totalKeyLen / (float)totalChars); } else { printf("%f, ", (float)totalKeyLen / (float)totalChars); } } //--------------------------------------------------------------------------------------- // // doBinarySearch() Binary Search timing test. Each name from the list // is looked up in the full sorted list of names. // //--------------------------------------------------------------------------------------- void doBinarySearch() { gCount = 0; int line; int loops = 0; int iLoop = 0; unsigned long elapsedTime = 0; // Adjust loop count to compensate for file size. Should be order n (lookups) * log n (compares/lookup) // Accurate timings do not depend on this being perfect. The correction is just to try to // get total running times of about the right order, so the that user doesn't need to // manually adjust the loop count for every different file size. double dLoopCount = double(opt_loopCount) * 3000. / (log10((double)gNumFileLines) * double(gNumFileLines)); if (opt_usekeys) dLoopCount *= 5; int adj_loopCount = int(dLoopCount); if (adj_loopCount < 1) adj_loopCount = 1; for (;;) { // not really a loop, just allows "break" to work, to simplify // inadvertantly running more than one test through here. if (opt_strcmp || opt_strcmpCPO) { unsigned long startTime = timeGetTime(); typedef int32_t (U_EXPORT2 *PF)(const UChar *, const UChar *); PF pf = u_strcmp; if (opt_strcmpCPO) {pf = u_strcmpCodePointOrder;} //if (opt_strcmp && opt_win) {pf = (PF)wcscmp;} // Damn the difference between int32_t and int // which forces the use of a cast here. int r = 0; for (loops=0; loops<adj_loopCount; loops++) { for (line=0; line < gNumFileLines; line++) { int hi = gNumFileLines-1; int lo = 0; int guess = -1; for (;;) { int newGuess = (hi + lo) / 2; if (newGuess == guess) break; guess = newGuess; for (iLoop=0; iLoop < opt_iLoopCount; iLoop++) { r = (*pf)((gSortedLines[line])->name, (gSortedLines[guess])->name); } gCount++; if (r== 0) break; if (r < 0) hi = guess; else lo = guess; } } } elapsedTime = timeGetTime() - startTime; break; } if (opt_icu) { unsigned long startTime = timeGetTime(); UCollationResult r = UCOL_EQUAL; for (loops=0; loops<adj_loopCount; loops++) { for (line=0; line < gNumFileLines; line++) { int lineLen = -1; int guessLen = -1; if (opt_uselen) { lineLen = (gSortedLines[line])->len; } int hi = gNumFileLines-1; int lo = 0; int guess = -1; for (;;) { int newGuess = (hi + lo) / 2; if (newGuess == guess) break; guess = newGuess; int ri = 0; if (opt_usekeys) { for (iLoop=0; iLoop < opt_iLoopCount; iLoop++) { ri = strcmp((gSortedLines[line])->icuSortKey, (gSortedLines[guess])->icuSortKey); } gCount++; r=UCOL_GREATER; if(ri<0) {r=UCOL_LESS;} else if (ri==0) {r=UCOL_EQUAL;} } else { if (opt_uselen) { guessLen = (gSortedLines[guess])->len; } for (iLoop=0; iLoop < opt_iLoopCount; iLoop++) { r = ucol_strcoll(gCol, (gSortedLines[line])->name, lineLen, (gSortedLines[guess])->name, guessLen); } gCount++; } if (r== UCOL_EQUAL) break; if (r == UCOL_LESS) hi = guess; else lo = guess; } } } elapsedTime = timeGetTime() - startTime; break; } if (opt_win) { unsigned long startTime = timeGetTime(); int r = 0; for (loops=0; loops<adj_loopCount; loops++) { for (line=0; line < gNumFileLines; line++) { int lineLen = -1; int guessLen = -1; if (opt_uselen) { lineLen = (gSortedLines[line])->len; } int hi = gNumFileLines-1; int lo = 0; int guess = -1; for (;;) { int newGuess = (hi + lo) / 2; if (newGuess == guess) break; guess = newGuess; if (opt_usekeys) { for (iLoop=0; iLoop < opt_iLoopCount; iLoop++) { r = strcmp((gSortedLines[line])->winSortKey, (gSortedLines[guess])->winSortKey); } gCount++; r+=2; } else { if (opt_uselen) { guessLen = (gSortedLines[guess])->len; } for (iLoop=0; iLoop < opt_iLoopCount; iLoop++) { r = CompareStringW(gWinLCID, 0, (gSortedLines[line])->name, lineLen, (gSortedLines[guess])->name, guessLen); } if (r == 0) { if (opt_terse == FALSE) { fprintf(stderr, "Error returned from Windows CompareStringW.\n"); } exit(-1); } gCount++; } if (r== 2) // strings == break; if (r == 1) // line < guess hi = guess; else // line > guess lo = guess; } } } elapsedTime = timeGetTime() - startTime; break; } if (opt_unix) { unsigned long startTime = timeGetTime(); int r = 0; for (loops=0; loops<adj_loopCount; loops++) { for (line=0; line < gNumFileLines; line++) { int hi = gNumFileLines-1; int lo = 0; int guess = -1; for (;;) { int newGuess = (hi + lo) / 2; if (newGuess == guess) break; guess = newGuess; if (opt_usekeys) { for (iLoop=0; iLoop < opt_iLoopCount; iLoop++) { r = strcmp((gSortedLines[line])->unixSortKey, (gSortedLines[guess])->unixSortKey); } gCount++; } else { for (iLoop=0; iLoop < opt_iLoopCount; iLoop++) { r = strcoll((gSortedLines[line])->unixName, (gSortedLines[guess])->unixName); } errno = 0; if (errno != 0) { fprintf(stderr, "Error %d returned from strcoll.\n", errno); exit(-1); } gCount++; } if (r == 0) // strings == break; if (r < 0) // line < guess hi = guess; else // line > guess lo = guess; } } } elapsedTime = timeGetTime() - startTime; break; } break; } int ns = (int)(float(1000000) * (float)elapsedTime / (float)gCount); if (opt_terse == FALSE) { printf("binary search: total # of string compares = %d\n", gCount); printf("binary search: compares per loop = %d\n", gCount / loops); printf("binary search: time per compare = %d ns\n", ns); } else { printf("%d, ", ns); } } //--------------------------------------------------------------------------------------- // // doQSort() The quick sort timing test. Uses the C library qsort function. // //--------------------------------------------------------------------------------------- void doQSort() { int i; Line **sortBuf = new Line *[gNumFileLines]; // Adjust loop count to compensate for file size. QSort should be n log(n) double dLoopCount = double(opt_loopCount) * 3000. / (log10((double)gNumFileLines) * double(gNumFileLines)); if (opt_usekeys) dLoopCount *= 5; int adj_loopCount = int(dLoopCount); if (adj_loopCount < 1) adj_loopCount = 1; gCount = 0; unsigned long startTime = timeGetTime(); if (opt_win && opt_usekeys) { for (i=0; i<opt_loopCount; i++) { memcpy(sortBuf, gRandomLines, gNumFileLines * sizeof(Line *)); qsort(sortBuf, gNumFileLines, sizeof(Line *), WinstrcmpK); } } else if (opt_win && opt_uselen) { for (i=0; i<adj_loopCount; i++) { memcpy(sortBuf, gRandomLines, gNumFileLines * sizeof(Line *)); qsort(sortBuf, gNumFileLines, sizeof(Line *), WinstrcmpL); } } else if (opt_win && !opt_uselen) { for (i=0; i<adj_loopCount; i++) { memcpy(sortBuf, gRandomLines, gNumFileLines * sizeof(Line *)); qsort(sortBuf, gNumFileLines, sizeof(Line *), Winstrcmp); } } else if (opt_icu && opt_usekeys) { for (i=0; i<adj_loopCount; i++) { memcpy(sortBuf, gRandomLines, gNumFileLines * sizeof(Line *)); qsort(sortBuf, gNumFileLines, sizeof(Line *), ICUstrcmpK); } } else if (opt_icu && opt_uselen) { for (i=0; i<adj_loopCount; i++) { memcpy(sortBuf, gRandomLines, gNumFileLines * sizeof(Line *)); qsort(sortBuf, gNumFileLines, sizeof(Line *), ICUstrcmpL); } } else if (opt_icu && !opt_uselen) { for (i=0; i<adj_loopCount; i++) { memcpy(sortBuf, gRandomLines, gNumFileLines * sizeof(Line *)); qsort(sortBuf, gNumFileLines, sizeof(Line *), ICUstrcmp); } } else if (opt_unix && !opt_usekeys) { for (i=0; i<adj_loopCount; i++) { memcpy(sortBuf, gRandomLines, gNumFileLines * sizeof(Line *)); qsort(sortBuf, gNumFileLines, sizeof(Line *), UNIXstrcmp); } } unsigned long elapsedTime = timeGetTime() - startTime; int ns = (int)(float(1000000) * (float)elapsedTime / (float)gCount); if (opt_terse == FALSE) { printf("qsort: total # of string compares = %d\n", gCount); printf("qsort: time per compare = %d ns\n", ns); } else { printf("%d, ", ns); } } //--------------------------------------------------------------------------------------- // // doKeyHist() Output a table of data for // average sort key size vs. string length. // //--------------------------------------------------------------------------------------- void doKeyHist() { int i; int maxLen = 0; // Find the maximum string length for (i=0; i<gNumFileLines; i++) { if (gFileLines[i].len > maxLen) maxLen = gFileLines[i].len; } // Allocate arrays to hold the histogram data int *accumulatedLen = new int[maxLen+1]; int *numKeysOfSize = new int[maxLen+1]; for (i=0; i<=maxLen; i++) { accumulatedLen[i] = 0; numKeysOfSize[i] = 0; } // Fill the arrays... for (i=0; i<gNumFileLines; i++) { int len = gFileLines[i].len; accumulatedLen[len] += strlen(gFileLines[i].icuSortKey); numKeysOfSize[len] += 1; } // And write out averages printf("String Length, Avg Key Length, Avg Key Len per char\n"); for (i=1; i<=maxLen; i++) { if (numKeysOfSize[i] > 0) { printf("%d, %f, %f\n", i, (float)accumulatedLen[i] / (float)numKeysOfSize[i], (float)accumulatedLen[i] / (float)(numKeysOfSize[i] * i)); } } delete []accumulatedLen; delete []numKeysOfSize ; } //--------------------------------------------------------------------------------------- // // doForwardIterTest(UBool) Forward iteration test // argument null-terminated string used // //--------------------------------------------------------------------------------------- void doForwardIterTest(UBool haslen) { int count = 0; UErrorCode error = U_ZERO_ERROR; printf("\n\nPerforming forward iteration performance test with "); if (haslen) { printf("non-null terminated data -----------\n"); } else { printf("null terminated data -----------\n"); } printf("performance test on strings from file -----------\n"); UChar dummytext[] = {0, 0}; UCollationElements *iter = ucol_openElements(gCol, NULL, 0, &error); ucol_setText(iter, dummytext, 1, &error); gCount = 0; unsigned long startTime = timeGetTime(); while (count < opt_loopCount) { int linecount = 0; while (linecount < gNumFileLines) { UChar *str = gFileLines[linecount].name; int strlen = haslen?gFileLines[linecount].len:-1; ucol_setText(iter, str, strlen, &error); while (ucol_next(iter, &error) != UCOL_NULLORDER) { gCount++; } linecount ++; } count ++; } unsigned long elapsedTime = timeGetTime() - startTime; printf("elapsedTime %ld\n", elapsedTime); // empty loop recalculation count = 0; startTime = timeGetTime(); while (count < opt_loopCount) { int linecount = 0; while (linecount < gNumFileLines) { UChar *str = gFileLines[linecount].name; int strlen = haslen?gFileLines[linecount].len:-1; ucol_setText(iter, str, strlen, &error); linecount ++; } count ++; } elapsedTime -= (timeGetTime() - startTime); printf("elapsedTime %ld\n", elapsedTime); ucol_closeElements(iter); int ns = (int)(float(1000000) * (float)elapsedTime / (float)gCount); printf("Total number of strings compared %d in %d loops\n", gNumFileLines, opt_loopCount); printf("Average time per ucol_next() nano seconds %d\n", ns); printf("performance test on skipped-5 concatenated strings from file -----------\n"); UChar *str; int strlen = 0; // appending all the strings int linecount = 0; while (linecount < gNumFileLines) { strlen += haslen?gFileLines[linecount].len: u_strlen(gFileLines[linecount].name); linecount ++; } str = (UChar *)malloc(sizeof(UChar) * strlen); int strindex = 0; linecount = 0; while (strindex < strlen) { int len = 0; len += haslen?gFileLines[linecount].len: u_strlen(gFileLines[linecount].name); memcpy(str + strindex, gFileLines[linecount].name, sizeof(UChar) * len); strindex += len; linecount ++; } printf("Total size of strings %d\n", strlen); gCount = 0; count = 0; if (!haslen) { strlen = -1; } iter = ucol_openElements(gCol, str, strlen, &error); if (!haslen) { strlen = u_strlen(str); } strlen -= 5; // any left over characters are not iterated, // this is to ensure the backwards and forwards iterators // gets the same position startTime = timeGetTime(); while (count < opt_loopCount) { int count5 = 5; strindex = 0; ucol_setOffset(iter, strindex, &error); while (TRUE) { if (ucol_next(iter, &error) == UCOL_NULLORDER) { break; } gCount++; count5 --; if (count5 == 0) { strindex += 10; if (strindex > strlen) { break; } ucol_setOffset(iter, strindex, &error); count5 = 5; } } count ++; } elapsedTime = timeGetTime() - startTime; printf("elapsedTime %ld\n", elapsedTime); // empty loop recalculation int tempgCount = 0; count = 0; startTime = timeGetTime(); while (count < opt_loopCount) { int count5 = 5; strindex = 0; ucol_setOffset(iter, strindex, &error); while (TRUE) { tempgCount ++; count5 --; if (count5 == 0) { strindex += 10; if (strindex > strlen) { break; } ucol_setOffset(iter, strindex, &error); count5 = 5; } } count ++; } elapsedTime -= (timeGetTime() - startTime); printf("elapsedTime %ld\n", elapsedTime); ucol_closeElements(iter); printf("gCount %d\n", gCount); ns = (int)(float(1000000) * (float)elapsedTime / (float)gCount); printf("Average time per ucol_next() nano seconds %d\n", ns); } //--------------------------------------------------------------------------------------- // // doBackwardIterTest(UBool) Backwards iteration test // argument null-terminated string used // //--------------------------------------------------------------------------------------- void doBackwardIterTest(UBool haslen) { int count = 0; UErrorCode error = U_ZERO_ERROR; printf("\n\nPerforming backward iteration performance test with "); if (haslen) { printf("non-null terminated data -----------\n"); } else { printf("null terminated data -----------\n"); } printf("performance test on strings from file -----------\n"); UCollationElements *iter = ucol_openElements(gCol, NULL, 0, &error); UChar dummytext[] = {0, 0}; ucol_setText(iter, dummytext, 1, &error); gCount = 0; unsigned long startTime = timeGetTime(); while (count < opt_loopCount) { int linecount = 0; while (linecount < gNumFileLines) { UChar *str = gFileLines[linecount].name; int strlen = haslen?gFileLines[linecount].len:-1; ucol_setText(iter, str, strlen, &error); while (ucol_previous(iter, &error) != UCOL_NULLORDER) { gCount ++; } linecount ++; } count ++; } unsigned long elapsedTime = timeGetTime() - startTime; printf("elapsedTime %ld\n", elapsedTime); // empty loop recalculation count = 0; startTime = timeGetTime(); while (count < opt_loopCount) { int linecount = 0; while (linecount < gNumFileLines) { UChar *str = gFileLines[linecount].name; int strlen = haslen?gFileLines[linecount].len:-1; ucol_setText(iter, str, strlen, &error); linecount ++; } count ++; } elapsedTime -= (timeGetTime() - startTime); printf("elapsedTime %ld\n", elapsedTime); ucol_closeElements(iter); int ns = (int)(float(1000000) * (float)elapsedTime / (float)gCount); printf("Total number of strings compared %d in %d loops\n", gNumFileLines, opt_loopCount); printf("Average time per ucol_previous() nano seconds %d\n", ns); printf("performance test on skipped-5 concatenated strings from file -----------\n"); UChar *str; int strlen = 0; // appending all the strings int linecount = 0; while (linecount < gNumFileLines) { strlen += haslen?gFileLines[linecount].len: u_strlen(gFileLines[linecount].name); linecount ++; } str = (UChar *)malloc(sizeof(UChar) * strlen); int strindex = 0; linecount = 0; while (strindex < strlen) { int len = 0; len += haslen?gFileLines[linecount].len: u_strlen(gFileLines[linecount].name); memcpy(str + strindex, gFileLines[linecount].name, sizeof(UChar) * len); strindex += len; linecount ++; } printf("Total size of strings %d\n", strlen); gCount = 0; count = 0; if (!haslen) { strlen = -1; } iter = ucol_openElements(gCol, str, strlen, &error); if (!haslen) { strlen = u_strlen(str); } startTime = timeGetTime(); while (count < opt_loopCount) { int count5 = 5; strindex = 5; ucol_setOffset(iter, strindex, &error); while (TRUE) { if (ucol_previous(iter, &error) == UCOL_NULLORDER) { break; } gCount ++; count5 --; if (count5 == 0) { strindex += 10; if (strindex > strlen) { break; } ucol_setOffset(iter, strindex, &error); count5 = 5; } } count ++; } elapsedTime = timeGetTime() - startTime; printf("elapsedTime %ld\n", elapsedTime); // empty loop recalculation count = 0; int tempgCount = 0; startTime = timeGetTime(); while (count < opt_loopCount) { int count5 = 5; strindex = 5; ucol_setOffset(iter, strindex, &error); while (TRUE) { tempgCount ++; count5 --; if (count5 == 0) { strindex += 10; if (strindex > strlen) { break; } ucol_setOffset(iter, strindex, &error); count5 = 5; } } count ++; } elapsedTime -= (timeGetTime() - startTime); printf("elapsedTime %ld\n", elapsedTime); ucol_closeElements(iter); printf("gCount %d\n", gCount); ns = (int)(float(1000000) * (float)elapsedTime / (float)gCount); printf("Average time per ucol_previous() nano seconds %d\n", ns); } //--------------------------------------------------------------------------------------- // // doIterTest() Iteration test // //--------------------------------------------------------------------------------------- void doIterTest() { doForwardIterTest(opt_uselen); doBackwardIterTest(opt_uselen); } //---------------------------------------------------------------------------------------- // // UnixConvert -- Convert the lines of the file to the encoding for UNIX // Since it appears that Unicode support is going in the general // direction of the use of UTF-8 locales, that is the approach // that is used here. // //---------------------------------------------------------------------------------------- void UnixConvert() { int line; UConverter *cvrtr; // An ICU code page converter. UErrorCode status = U_ZERO_ERROR; cvrtr = ucnv_open("utf-8", &status); // we are just doing UTF-8 locales for now. if (U_FAILURE(status)) { fprintf(stderr, "ICU Converter open failed.: %s\n", u_errorName(status)); exit(-1); } for (line=0; line < gNumFileLines; line++) { int sizeNeeded = ucnv_fromUChars(cvrtr, 0, // ptr to target buffer. 0, // length of target buffer. gFileLines[line].name, -1, // source is null terminated &status); if (status != U_BUFFER_OVERFLOW_ERROR && status != U_ZERO_ERROR) { //fprintf(stderr, "Conversion from Unicode, something is wrong.\n"); //exit(-1); } status = U_ZERO_ERROR; gFileLines[line].unixName = new char[sizeNeeded+1]; sizeNeeded = ucnv_fromUChars(cvrtr, gFileLines[line].unixName, // ptr to target buffer. sizeNeeded+1, // length of target buffer. gFileLines[line].name, -1, // source is null terminated &status); if (U_FAILURE(status)) { fprintf(stderr, "ICU Conversion Failed.: %d\n", status); exit(-1); } gFileLines[line].unixName[sizeNeeded] = 0; }; ucnv_close(cvrtr); } //---------------------------------------------------------------------------------------- // // class UCharFile Class to hide all the gorp to read a file in // and produce a stream of UChars. // //---------------------------------------------------------------------------------------- class UCharFile { public: UCharFile(const char *fileName); ~UCharFile(); UChar get(); UBool eof() {return fEof;}; UBool error() {return fError;}; private: UCharFile (const UCharFile & /*other*/) {}; // No copy constructor. UCharFile & operator = (const UCharFile &/*other*/) {return *this;}; // No assignment op FILE *fFile; const char *fName; UBool fEof; UBool fError; UChar fPending2ndSurrogate; enum {UTF16LE, UTF16BE, UTF8} fEncoding; }; UCharFile::UCharFile(const char * fileName) { fEof = FALSE; fError = FALSE; fName = fileName; fFile = fopen(fName, "rb"); fPending2ndSurrogate = 0; if (fFile == NULL) { fprintf(stderr, "Can not open file \"%s\"\n", opt_fName); fError = TRUE; return; } // // Look for the byte order mark at the start of the file. // int BOMC1, BOMC2, BOMC3; BOMC1 = fgetc(fFile); BOMC2 = fgetc(fFile); if (BOMC1 == 0xff && BOMC2 == 0xfe) { fEncoding = UTF16LE; } else if (BOMC1 == 0xfe && BOMC2 == 0xff) { fEncoding = UTF16BE; } else if (BOMC1 == 0xEF && BOMC2 == 0xBB && (BOMC3 = fgetc(fFile)) == 0xBF ) { fEncoding = UTF8; } else { fprintf(stderr, "collperf: file \"%s\" encoding must be UTF-8 or UTF-16, and " "must include a BOM.\n", fileName); fError = true; return; } } UCharFile::~UCharFile() { fclose(fFile); } UChar UCharFile::get() { UChar c; switch (fEncoding) { case UTF16LE: { int cL, cH; cL = fgetc(fFile); cH = fgetc(fFile); c = cL | (cH << 8); if (cH == EOF) { c = 0; fEof = TRUE; } break; } case UTF16BE: { int cL, cH; cH = fgetc(fFile); cL = fgetc(fFile); c = cL | (cH << 8); if (cL == EOF) { c = 0; fEof = TRUE; } break; } case UTF8: { if (fPending2ndSurrogate != 0) { c = fPending2ndSurrogate; fPending2ndSurrogate = 0; break; } int ch = fgetc(fFile); // Note: c and ch are separate cause eof test doesn't work on UChar type. if (ch == EOF) { c = 0; fEof = TRUE; break; } if (ch <= 0x7f) { // It's ascii. No further utf-8 conversion. c = ch; break; } // Figure out the lenght of the char and read the rest of the bytes // into a temp array. int nBytes; if (ch >= 0xF0) {nBytes=4;} else if (ch >= 0xE0) {nBytes=3;} else if (ch >= 0xC0) {nBytes=2;} else { fprintf(stderr, "utf-8 encoded file contains corrupt data.\n"); fError = TRUE; return 0; } unsigned char bytes[10]; bytes[0] = (unsigned char)ch; int i; for (i=1; i<nBytes; i++) { bytes[i] = fgetc(fFile); if (bytes[i] < 0x80 || bytes[i] >= 0xc0) { fprintf(stderr, "utf-8 encoded file contains corrupt data.\n"); fError = TRUE; return 0; } } // Convert the bytes from the temp array to a Unicode char. i = 0; uint32_t cp; U8_NEXT_UNSAFE(bytes, i, cp); c = (UChar)cp; if (cp >= 0x10000) { // The code point needs to be broken up into a utf-16 surrogate pair. // Process first half this time through the main loop, and // remember the other half for the next time through. UChar utf16Buf[3]; i = 0; UTF16_APPEND_CHAR_UNSAFE(utf16Buf, i, cp); fPending2ndSurrogate = utf16Buf[1]; c = utf16Buf[0]; } break; }; default: c = 0xFFFD; /* Error, unspecified codepage*/ fprintf(stderr, "UCharFile: Error: unknown fEncoding\n"); exit(1); } return c; } //---------------------------------------------------------------------------------------- // // openRulesCollator - Command line specified a rules file. Read it in // and open a collator with it. // //---------------------------------------------------------------------------------------- UCollator *openRulesCollator() { UCharFile f(opt_rules); if (f.error()) { return 0; } int bufLen = 10000; UChar *buf = (UChar *)malloc(bufLen * sizeof(UChar)); UChar *tmp; int i = 0; for(;;) { buf[i] = f.get(); if (f.eof()) { break; } if (f.error()) { return 0; } i++; if (i >= bufLen) { tmp = buf; bufLen += 10000; buf = (UChar *)realloc(buf, bufLen); if (buf == NULL) { free(tmp); return 0; } } } buf[i] = 0; UErrorCode status = U_ZERO_ERROR; UCollator *coll = ucol_openRules(buf, u_strlen(buf), UCOL_OFF, UCOL_DEFAULT_STRENGTH, NULL, &status); if (U_FAILURE(status)) { fprintf(stderr, "ICU ucol_openRules() open failed.: %d\n", status); return 0; } free(buf); return coll; } //---------------------------------------------------------------------------------------- // // Main -- process command line, read in and pre-process the test file, // call other functions to do the actual tests. // //---------------------------------------------------------------------------------------- int main(int argc, const char** argv) { if (ProcessOptions(argc, argv, opts) != TRUE || opt_help || opt_fName == 0) { printf(gUsageString); exit (1); } // Make sure that we've only got one API selected. if (opt_unix || opt_win) opt_icu = FALSE; if (opt_unix) opt_win = FALSE; // // Set up an ICU collator // UErrorCode status = U_ZERO_ERROR; if (opt_rules != 0) { gCol = openRulesCollator(); if (gCol == 0) {return -1;} } else { gCol = ucol_open(opt_locale, &status); if (U_FAILURE(status)) { fprintf(stderr, "Collator creation failed.: %d\n", status); return -1; } } if (status==U_USING_DEFAULT_WARNING && opt_terse==FALSE) { fprintf(stderr, "Warning, U_USING_DEFAULT_WARNING for %s\n", opt_locale); } if (status==U_USING_FALLBACK_WARNING && opt_terse==FALSE) { fprintf(stderr, "Warning, U_USING_FALLBACK_ERROR for %s\n", opt_locale); } if (opt_norm) { ucol_setAttribute(gCol, UCOL_NORMALIZATION_MODE, UCOL_ON, &status); } if (opt_french && opt_frenchoff) { fprintf(stderr, "collperf: Error, specified both -french and -frenchoff options."); exit(-1); } if (opt_french) { ucol_setAttribute(gCol, UCOL_FRENCH_COLLATION, UCOL_ON, &status); } if (opt_frenchoff) { ucol_setAttribute(gCol, UCOL_FRENCH_COLLATION, UCOL_OFF, &status); } if (opt_lower) { ucol_setAttribute(gCol, UCOL_CASE_FIRST, UCOL_LOWER_FIRST, &status); } if (opt_upper) { ucol_setAttribute(gCol, UCOL_CASE_FIRST, UCOL_UPPER_FIRST, &status); } if (opt_case) { ucol_setAttribute(gCol, UCOL_CASE_LEVEL, UCOL_ON, &status); } if (opt_shifted) { ucol_setAttribute(gCol, UCOL_ALTERNATE_HANDLING, UCOL_SHIFTED, &status); } if (opt_level != 0) { switch (opt_level) { case 1: ucol_setAttribute(gCol, UCOL_STRENGTH, UCOL_PRIMARY, &status); break; case 2: ucol_setAttribute(gCol, UCOL_STRENGTH, UCOL_SECONDARY, &status); break; case 3: ucol_setAttribute(gCol, UCOL_STRENGTH, UCOL_TERTIARY, &status); break; case 4: ucol_setAttribute(gCol, UCOL_STRENGTH, UCOL_QUATERNARY, &status); break; case 5: ucol_setAttribute(gCol, UCOL_STRENGTH, UCOL_IDENTICAL, &status); break; default: fprintf(stderr, "-level param must be between 1 and 5\n"); exit(-1); } } if (U_FAILURE(status)) { fprintf(stderr, "Collator attribute setting failed.: %d\n", status); return -1; } // // Set up a Windows LCID // if (opt_langid != 0) { gWinLCID = MAKELCID(opt_langid, SORT_DEFAULT); } else { gWinLCID = uloc_getLCID(opt_locale); } // // Set the UNIX locale // if (opt_unix) { if (setlocale(LC_ALL, opt_locale) == 0) { fprintf(stderr, "setlocale(LC_ALL, %s) failed.\n", opt_locale); exit(-1); } } // Read in the input file. // File assumed to be utf-16. // Lines go onto heap buffers. Global index array to line starts is created. // Lines themselves are null terminated. // UCharFile f(opt_fName); if (f.error()) { exit(-1); } const int MAXLINES = 100000; gFileLines = new Line[MAXLINES]; UChar buf[1024]; int column = 0; // Read the file, split into lines, and save in memory. // Loop runs once per utf-16 value from the input file, // (The number of bytes read from file per loop iteration depends on external encoding.) for (;;) { UChar c = f.get(); if (f.error()){ exit(-1); } // We now have a good UTF-16 value in c. // Watch for CR, LF, EOF; these finish off a line. if (c == 0xd) { continue; } if (f.eof() || c == 0x0a || c==0x2028) { // Unipad inserts 2028 line separators! buf[column++] = 0; if (column > 1) { gFileLines[gNumFileLines].name = new UChar[column]; gFileLines[gNumFileLines].len = column-1; memcpy(gFileLines[gNumFileLines].name, buf, column * sizeof(UChar)); gNumFileLines++; column = 0; if (gNumFileLines >= MAXLINES) { fprintf(stderr, "File too big. Max number of lines is %d\n", MAXLINES); exit(-1); } } if (c == 0xa || c == 0x2028) continue; else break; // EOF } buf[column++] = c; if (column >= 1023) { static UBool warnFlag = TRUE; if (warnFlag) { fprintf(stderr, "Warning - file line longer than 1023 chars truncated.\n"); warnFlag = FALSE; } column--; } } if (opt_terse == FALSE) { printf("file \"%s\", %d lines.\n", opt_fName, gNumFileLines); } // Convert the lines to the UNIX encoding. if (opt_unix) { UnixConvert(); } // // Pre-compute ICU sort keys for the lines of the file. // int line; int32_t t; for (line=0; line<gNumFileLines; line++) { t = ucol_getSortKey(gCol, gFileLines[line].name, -1, (unsigned char *)buf, sizeof(buf)); gFileLines[line].icuSortKey = new char[t]; if (t > (int32_t)sizeof(buf)) { t = ucol_getSortKey(gCol, gFileLines[line].name, -1, (unsigned char *)gFileLines[line].icuSortKey , t); } else { memcpy(gFileLines[line].icuSortKey, buf, t); } } // // Pre-compute Windows sort keys for the lines of the file. // for (line=0; line<gNumFileLines; line++) { t=LCMapStringW(gWinLCID, LCMAP_SORTKEY, gFileLines[line].name, -1, buf, sizeof(buf)); gFileLines[line].winSortKey = new char[t]; if (t > (int32_t)sizeof(buf)) { t = LCMapStringW(gWinLCID, LCMAP_SORTKEY, gFileLines[line].name, -1, (unsigned short *)(gFileLines[line].winSortKey), t); } else { memcpy(gFileLines[line].winSortKey, buf, t); } } // // Pre-compute UNIX sort keys for the lines of the file. // if (opt_unix) { for (line=0; line<gNumFileLines; line++) { t=strxfrm((char *)buf, gFileLines[line].unixName, sizeof(buf)); gFileLines[line].unixSortKey = new char[t]; if (t > (int32_t)sizeof(buf)) { t = strxfrm(gFileLines[line].unixSortKey, gFileLines[line].unixName, sizeof(buf)); } else { memcpy(gFileLines[line].unixSortKey, buf, t); } } } // // Dump file lines, CEs, Sort Keys if requested. // if (opt_dump) { int i; for (line=0; line<gNumFileLines; line++) { for (i=0;;i++) { UChar c = gFileLines[line].name[i]; if (c == 0) break; if (c < 0x20 || c > 0x7e) { printf("\\u%.4x", c); } else { printf("%c", c); } } printf("\n"); printf(" CEs: "); UCollationElements *CEiter = ucol_openElements(gCol, gFileLines[line].name, -1, &status); int32_t ce; i = 0; for (;;) { ce = ucol_next(CEiter, &status); if (ce == UCOL_NULLORDER) { break; } printf(" %.8x", ce); if (++i > 8) { printf("\n "); i = 0; } } printf("\n"); ucol_closeElements(CEiter); printf(" ICU Sort Key: "); for (i=0; ; i++) { unsigned char c = gFileLines[line].icuSortKey[i]; printf("%02x ", c); if (c == 0) { break; } if (i > 0 && i % 20 == 0) { printf("\n "); } } printf("\n"); } } // // Pre-sort the lines. // int i; gSortedLines = new Line *[gNumFileLines]; for (i=0; i<gNumFileLines; i++) { gSortedLines[i] = &gFileLines[i]; } if (opt_win) { qsort(gSortedLines, gNumFileLines, sizeof(Line *), Winstrcmp); } else if (opt_unix) { qsort(gSortedLines, gNumFileLines, sizeof(Line *), UNIXstrcmp); } else /* ICU */ { qsort(gSortedLines, gNumFileLines, sizeof(Line *), ICUstrcmp); } // // Make up a randomized order, will be used for sorting tests. // gRandomLines = new Line *[gNumFileLines]; for (i=0; i<gNumFileLines; i++) { gRandomLines[i] = &gFileLines[i]; } qsort(gRandomLines, gNumFileLines, sizeof(Line *), ICURandomCmp); // // We've got the file read into memory. Go do something with it. // if (opt_qsort) doQSort(); if (opt_binsearch) doBinarySearch(); if (opt_keygen) doKeyGen(); if (opt_keyhist) doKeyHist(); if (opt_itertest) doIterTest(); return 0; }