/* *********************************************************************** * Copyright (C) 2016 and later: Unicode, Inc. and others. * License & terms of use: http://www.unicode.org/copyright.html#License *********************************************************************** *********************************************************************** * Copyright (c) 2002-2016,International Business Machines * Corporation and others. All Rights Reserved. *********************************************************************** *********************************************************************** */ #ifndef _DATEFMTPERF_H #define _DATEFMTPERF_H #include "cmemory.h" #include "unicode/stringpiece.h" #include "unicode/unistr.h" #include "unicode/uperf.h" #include "unicode/dtitvfmt.h" #include "unicode/utypes.h" #include "unicode/datefmt.h" #include "unicode/calendar.h" #include "unicode/uclean.h" #include "unicode/brkiter.h" #include "unicode/numfmt.h" #include "unicode/coll.h" #include "util.h" #include "datedata.h" #include "breakdata.h" #include "collationdata.h" #include <stdlib.h> #include <string.h> #include <fstream> #include <iostream> using namespace std; // Stubs for Windows API functions when building on UNIXes. // #if U_PLATFORM_USES_ONLY_WIN32_API // do nothing #else #define _UNICODE typedef int DWORD; inline int FoldStringW(DWORD dwMapFlags, const UChar* lpSrcStr,int cchSrc, UChar* lpDestStr,int cchDest); #endif class BreakItFunction : public UPerfFunction { private: int num; bool wordIteration; public: BreakItFunction(){num = -1;} BreakItFunction(int a, bool b){num = a; wordIteration = b;} virtual void call(UErrorCode * status) { BreakIterator* boundary; if(wordIteration) { for(int i = 0; i < num; i++) { boundary = BreakIterator::createWordInstance("en", *status); boundary->setText(str); int32_t start = boundary->first(); for (int32_t end = boundary->next(); end != BreakIterator::DONE; start = end, end = boundary->next()) { printTextRange( *boundary, start, end ); } } } else // character iteration { for(int i = 0; i < num; i++) { boundary = BreakIterator::createCharacterInstance(Locale::getUS(), *status); boundary->setText(str); int32_t start = boundary->first(); for (int32_t end = boundary->next(); end != BreakIterator::DONE; start = end, end = boundary->next()) { printTextRange( *boundary, start, end ); } } } } virtual long getOperationsPerIteration() { if(wordIteration) return 125*num; else return 355*num; } void printUnicodeString(const UnicodeString &s) { char charBuf[1000]; s.extract(0, s.length(), charBuf, sizeof(charBuf)-1, 0); charBuf[sizeof(charBuf)-1] = 0; printf("%s", charBuf); } void printTextRange( BreakIterator& iterator, int32_t start, int32_t end ) { CharacterIterator *strIter = iterator.getText().clone(); UnicodeString s; strIter->getText(s); //printUnicodeString(UnicodeString(s, start, end-start)); //puts(""); delete strIter; } // Print the given string to stdout (for debugging purposes) void uprintf(const UnicodeString &str) { char *buf = 0; int32_t len = str.length(); int32_t bufLen = len + 16; int32_t actualLen; buf = new char[bufLen + 1]; actualLen = str.extract(0, len, buf/*, bufLen*/); // Default codepage conversion buf[actualLen] = 0; printf("%s", buf); delete[] buf; } }; class DateFmtFunction : public UPerfFunction { private: int num; char locale[25]; public: DateFmtFunction() { num = -1; } DateFmtFunction(int a, const char* loc) { num = a; strcpy(locale, loc); } virtual void call(UErrorCode* status) { UErrorCode status2 = U_ZERO_ERROR; Calendar *cal; TimeZone *zone; UnicodeString str; UDate date; cal = Calendar::createInstance(status2); check(status2, "Calendar::createInstance"); zone = TimeZone::createTimeZone("GMT"); // Create a GMT zone cal->adoptTimeZone(zone); Locale loc(locale); DateFormat *fmt; fmt = DateFormat::createDateTimeInstance( DateFormat::kShort, DateFormat::kFull, loc); // (dates are imported from datedata.h) for(int j = 0; j < num; j++) for(int i = 0; i < NUM_DATES; i++) { cal->clear(); cal->set(years[i], months[i], days[i]); date = cal->getTime(status2); check(status2, "Calendar::getTime"); fmt->setCalendar(*cal); // Format the date str.remove(); fmt->format(date, str, status2); // Display the formatted date string //uprintf(str); //printf("\n"); } delete fmt; delete cal; //u_cleanup(); } virtual long getOperationsPerIteration() { return NUM_DATES * num; } // Print the given string to stdout (for debugging purposes) void uprintf(const UnicodeString &str) { char *buf = 0; int32_t len = str.length(); int32_t bufLen = len + 16; int32_t actualLen; buf = new char[bufLen + 1]; actualLen = str.extract(0, len, buf/*, bufLen*/); // Default codepage conversion buf[actualLen] = 0; printf("%s", buf); delete[] buf; } // Verify that a UErrorCode is successful; exit(1) if not void check(UErrorCode& status, const char* msg) { if (U_FAILURE(status)) { printf("ERROR: %s (%s)\n", u_errorName(status), msg); exit(1); } } }; class DateFmtCreateFunction : public UPerfFunction { private: int num; char locale[25]; public: DateFmtCreateFunction(int a, const char* loc) { num = a; strcpy(locale, loc); } virtual void call(UErrorCode* /* status */) { Locale loc(locale); DateFormat *fmt; // (dates are imported from datedata.h) for(int j = 0; j < num; j++) { fmt = DateFormat::createDateTimeInstance( DateFormat::kShort, DateFormat::kFull, loc); delete fmt; } } virtual long getOperationsPerIteration() { return num; } }; class DateFmtCopyFunction : public UPerfFunction { private: int num; char locale[25]; public: DateFmtCopyFunction() { num = -1; } DateFmtCopyFunction(int a, const char* loc) { num = a; strcpy(locale, loc); } virtual void call(UErrorCode* /* status */) { Locale loc(locale); UErrorCode status2 = U_ZERO_ERROR; DateFormat *fmt = DateFormat::createDateTimeInstance( DateFormat::kShort, DateFormat::kFull, loc); for(int j = 0; j < num; j++) { Format *cp = fmt->clone(); delete cp; } delete fmt; } virtual long getOperationsPerIteration() { return num; } // Verify that a UErrorCode is successful; exit(1) if not void check(UErrorCode& status, const char* msg) { if (U_FAILURE(status)) { printf("ERROR: %s (%s)\n", u_errorName(status), msg); exit(1); } } }; class DIFCreateFunction : public UPerfFunction { private: int num; char locale[25]; public: DIFCreateFunction() { num = -1; } DIFCreateFunction(int a, const char* loc) { num = a; strcpy(locale, loc); } virtual void call(UErrorCode* /* status */) { UErrorCode status2 = U_ZERO_ERROR; Calendar *cal; TimeZone *zone; cal = Calendar::createInstance(status2); check(status2, "Calendar::createInstance"); zone = TimeZone::createTimeZone("GMT"); // Create a GMT zone cal->adoptTimeZone(zone); Locale loc(locale); UnicodeString skeleton("yMMMMdHms"); for(int j = 0; j < num; j++) { DateIntervalFormat* fmt(DateIntervalFormat::createInstance(skeleton, loc, status2)); delete fmt; } delete cal; } virtual long getOperationsPerIteration() { return num; } // Verify that a UErrorCode is successful; exit(1) if not void check(UErrorCode& status, const char* msg) { if (U_FAILURE(status)) { printf("ERROR: %s (%s)\n", u_errorName(status), msg); exit(1); } } }; class TimeZoneCreateFunction : public UPerfFunction { private: int num; char locale[25]; public: TimeZoneCreateFunction() { num = -1; } TimeZoneCreateFunction(int a, const char* loc) { num = a; strcpy(locale, loc); } virtual void call(UErrorCode* /* status */) { Locale loc(locale); UnicodeString tzname("UTC"); for(int j = 0; j < num; j++) { TimeZone* tz(TimeZone::createTimeZone(tzname)); delete tz; } } virtual long getOperationsPerIteration() { return num; } // Verify that a UErrorCode is successful; exit(1) if not void check(UErrorCode& status, const char* msg) { if (U_FAILURE(status)) { printf("ERROR: %s (%s)\n", u_errorName(status), msg); exit(1); } } }; class DTPatternGeneratorCreateFunction : public UPerfFunction { private: int num; char locale[25]; public: DTPatternGeneratorCreateFunction() { num = -1; } DTPatternGeneratorCreateFunction(int a, const char* loc) { num = a; strcpy(locale, loc); } virtual void call(UErrorCode* /* status */) { UErrorCode status2 = U_ZERO_ERROR; Locale loc(locale); for(int j = 0; j < num; j++) { DateTimePatternGenerator* gen(DateTimePatternGenerator::createInstance(loc, status2)); delete gen; } } virtual long getOperationsPerIteration() { return num; } // Verify that a UErrorCode is successful; exit(1) if not void check(UErrorCode& status, const char* msg) { if (U_FAILURE(status)) { printf("ERROR: %s (%s)\n", u_errorName(status), msg); exit(1); } } }; class DTPatternGeneratorCopyFunction : public UPerfFunction { private: int num; char locale[25]; public: DTPatternGeneratorCopyFunction() { num = -1; } DTPatternGeneratorCopyFunction(int a, const char* loc) { num = a; strcpy(locale, loc); } virtual void call(UErrorCode* /* status */) { UErrorCode status2 = U_ZERO_ERROR; Locale loc(locale); DateTimePatternGenerator* gen(DateTimePatternGenerator::createInstance(loc, status2)); for(int j = 0; j < num; j++) { DateTimePatternGenerator *cl = gen->clone(); delete cl; } delete gen; } virtual long getOperationsPerIteration() { return num; } // Verify that a UErrorCode is successful; exit(1) if not void check(UErrorCode& status, const char* msg) { if (U_FAILURE(status)) { printf("ERROR: %s (%s)\n", u_errorName(status), msg); exit(1); } } }; class DTPatternGeneratorBestValueFunction : public UPerfFunction { private: int num; char locale[25]; public: DTPatternGeneratorBestValueFunction() { num = -1; } DTPatternGeneratorBestValueFunction(int a, const char* loc) { num = a; strcpy(locale, loc); } virtual void call(UErrorCode* /* status */) { UErrorCode status2 = U_ZERO_ERROR; Locale loc(locale); DateTimePatternGenerator* gen(DateTimePatternGenerator::createInstance(loc, status2)); UnicodeString skeleton("yMMMMdHms"); for(int j = 0; j < num; j++) { gen->getBestPattern(skeleton, status2); } check(status2, "getBestPattern"); delete gen; } virtual long getOperationsPerIteration() { return num; } // Verify that a UErrorCode is successful; exit(1) if not void check(UErrorCode& status, const char* msg) { if (U_FAILURE(status)) { printf("ERROR: %s (%s)\n", u_errorName(status), msg); exit(1); } } }; class NumFmtFunction : public UPerfFunction { private: int num; char locale[25]; public: NumFmtFunction() { num = -1; } NumFmtFunction(int a, const char* loc) { num = a; strcpy(locale, loc); } virtual void call(UErrorCode* status2) { Locale loc(locale); UErrorCode status = U_ZERO_ERROR; // Create a number formatter for the locale NumberFormat *fmt = NumberFormat::createInstance(loc, status); // Parse a string. The string uses the digits '0' through '9' // and the decimal separator '.', standard in the US locale for(int i = 0; i < num; i++) { UnicodeString str("9876543210.123"); Formattable result; fmt->parse(str, result, status); //uprintf(formattableToString(result)); //printf("\n"); // Take the number parsed above, and use the formatter to // format it. str.remove(); // format() will APPEND to this string fmt->format(result, str, status); //uprintf(str); //printf("\n"); } delete fmt; // Release the storage used by the formatter } enum { U_SPACE=0x20, U_DQUOTE=0x22, U_COMMA=0x2c, U_LEFT_SQUARE_BRACKET=0x5b, U_BACKSLASH=0x5c, U_RIGHT_SQUARE_BRACKET=0x5d, U_SMALL_U=0x75 }; // Create a display string for a formattable UnicodeString formattableToString(const Formattable& f) { switch (f.getType()) { case Formattable::kDate: // TODO: Finish implementing this return UNICODE_STRING_SIMPLE("Formattable_DATE_TBD"); case Formattable::kDouble: { char buf[256]; sprintf(buf, "%gD", f.getDouble()); return UnicodeString(buf, ""); } case Formattable::kLong: case Formattable::kInt64: { char buf[256]; sprintf(buf, "%ldL", f.getLong()); return UnicodeString(buf, ""); } case Formattable::kString: return UnicodeString((UChar)U_DQUOTE).append(f.getString()).append((UChar)U_DQUOTE); case Formattable::kArray: { int32_t i, count; const Formattable* array = f.getArray(count); UnicodeString result((UChar)U_LEFT_SQUARE_BRACKET); for (i=0; i<count; ++i) { if (i > 0) { (result += (UChar)U_COMMA) += (UChar)U_SPACE; } result += formattableToString(array[i]); } result += (UChar)U_RIGHT_SQUARE_BRACKET; return result; } default: return UNICODE_STRING_SIMPLE("INVALID_Formattable"); } } virtual long getOperationsPerIteration() { return num; } // Print the given string to stdout using the UTF-8 converter (for debugging purposes only) void uprintf(const UnicodeString &str) { char stackBuffer[100]; char *buf = 0; int32_t bufLen = str.extract(0, 0x7fffffff, stackBuffer, sizeof(stackBuffer), "UTF-8"); if(bufLen < sizeof(stackBuffer)) { buf = stackBuffer; } else { buf = new char[bufLen + 1]; bufLen = str.extract(0, 0x7fffffff, buf, bufLen + 1, "UTF-8"); } printf("%s", buf); if(buf != stackBuffer) { delete[] buf; } } }; #define NUM_STRING "9876543210.123" #define NUM_NUM 9876543210.123 class StdioNumFmtFunction : public UPerfFunction { private: int num; char locale[25]; public: StdioNumFmtFunction() { num = -1; } StdioNumFmtFunction(int a, const char* loc) { num = a; strcpy(locale, loc); } virtual void call(UErrorCode* status2) { Locale loc(locale); UErrorCode status = U_ZERO_ERROR; // Parse a string. The string uses the digits '0' through '9' // and the decimal separator '.', standard in the US locale double result; char outbuf[500]; const char *str = NUM_STRING; for(int i = 0; i < num; i++) { if(sscanf(str, "%lg", &result)!=1) { cout << "Failed Stdio: failed to sscanf" << endl; *status2 = U_PARSE_ERROR; return; } sprintf(outbuf, "%lg", result); } if(result!=NUM_NUM) { cout << "Failed Stdio: sscanf got wrong result, expected " << NUM_NUM << " got " << result << endl; *status2 = U_PARSE_ERROR; } if(strcmp(str,NUM_STRING)) { cout << "Failed Stdio: sprintf got wrong result, expected " << NUM_STRING << " got " << str << endl; *status2 = U_PARSE_ERROR; } } virtual long getOperationsPerIteration() { return num; } }; class CollationFunction : public UPerfFunction { private: int num; char locale[25]; UnicodeString *collation_strings; /** * Unescape the strings */ void init() { uint32_t listSize = UPRV_LENGTHOF(collation_strings_escaped); collation_strings = new UnicodeString[listSize]; for(uint32_t k=0;k<listSize;k++) { collation_strings[k] = collation_strings_escaped[k].unescape(); } UnicodeString shorty((UChar32)0x12345); } public: CollationFunction() { num = -1; init(); } ~CollationFunction() { delete [] collation_strings; } CollationFunction(int a, const char* loc) { num = a; strcpy(locale, loc); init(); } virtual void call(UErrorCode* status2) { uint32_t listSize = UPRV_LENGTHOF(collation_strings_escaped); UErrorCode status = U_ZERO_ERROR; Collator *coll = Collator::createInstance(Locale(locale), status); for(int k = 0; k < num; k++) { uint32_t i, j; for(i=listSize-1; i>=1; i--) { for(j=0; j<i; j++) { if(coll->compare(collation_strings[j], collation_strings[j+1]) == UCOL_LESS) { //cout << "Success!" << endl; } } } } delete coll; } virtual long getOperationsPerIteration() { return num; } }; class DateFormatPerfTest : public UPerfTest { private: public: DateFormatPerfTest(int32_t argc, const char* argv[], UErrorCode& status); ~DateFormatPerfTest(); virtual UPerfFunction* runIndexedTest(int32_t index, UBool exec,const char* &name, char* par); UPerfFunction* DateFmt250(); UPerfFunction* DateFmt10000(); UPerfFunction* DateFmt100000(); UPerfFunction* DateFmtCreate250(); UPerfFunction* DateFmtCreate10000(); UPerfFunction* DateFmtCopy250(); UPerfFunction* DateFmtCopy10000(); UPerfFunction* BreakItWord250(); UPerfFunction* BreakItWord10000(); UPerfFunction* BreakItChar250(); UPerfFunction* BreakItChar10000(); UPerfFunction* NumFmt10000(); UPerfFunction* NumFmt100000(); UPerfFunction* Collation10000(); UPerfFunction* Collation100000(); UPerfFunction* DIFCreate250(); UPerfFunction* DIFCreate10000(); UPerfFunction* TimeZoneCreate250(); UPerfFunction* TimeZoneCreate10000(); UPerfFunction* DTPatternGeneratorCreate250(); UPerfFunction* DTPatternGeneratorCreate10000(); UPerfFunction* DTPatternGeneratorCopy250(); UPerfFunction* DTPatternGeneratorCopy10000(); UPerfFunction* DTPatternGeneratorBestValue250(); UPerfFunction* DTPatternGeneratorBestValue10000(); }; #endif // DateFmtPerf