2010-07-09 16:14:46 +00:00
|
|
|
/*
|
|
|
|
**********************************************************************
|
|
|
|
* Copyright (c) 2002-2010,International Business Machines
|
|
|
|
* Corporation and others. All Rights Reserved.
|
|
|
|
**********************************************************************
|
|
|
|
**********************************************************************
|
|
|
|
*/
|
|
|
|
|
|
|
|
#ifndef _DATEFMTPERF_H
|
|
|
|
#define _DATEFMTPERF_H
|
|
|
|
|
2010-08-05 00:26:41 +00:00
|
|
|
|
|
|
|
#include "unicode/stringpiece.h"
|
2010-07-09 16:14:46 +00:00
|
|
|
#include "unicode/unistr.h"
|
2010-08-05 00:26:41 +00:00
|
|
|
#include "unicode/uperf.h"
|
|
|
|
|
|
|
|
#include "unicode/utypes.h"
|
2010-07-09 16:14:46 +00:00
|
|
|
#include "unicode/datefmt.h"
|
|
|
|
#include "unicode/calendar.h"
|
|
|
|
#include "unicode/uclean.h"
|
|
|
|
#include "unicode/brkiter.h"
|
2010-07-29 19:09:01 +00:00
|
|
|
#include "unicode/numfmt.h"
|
|
|
|
#include "unicode/coll.h"
|
2010-07-09 16:14:46 +00:00
|
|
|
#include "util.h"
|
2010-07-29 19:09:01 +00:00
|
|
|
|
2010-07-09 16:14:46 +00:00
|
|
|
#include "datedata.h"
|
|
|
|
#include "breakdata.h"
|
2010-07-29 19:09:01 +00:00
|
|
|
#include "collationdata.h"
|
|
|
|
|
2010-07-09 16:14:46 +00:00
|
|
|
#include <stdlib.h>
|
|
|
|
#include <fstream>
|
|
|
|
#include <string>
|
|
|
|
|
|
|
|
#include <iostream>
|
|
|
|
using namespace std;
|
|
|
|
|
|
|
|
// Stubs for Windows API functions when building on UNIXes.
|
|
|
|
//
|
|
|
|
#if defined(U_WINDOWS)
|
|
|
|
// 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)
|
2010-07-29 19:09:01 +00:00
|
|
|
{
|
2010-07-09 16:14:46 +00:00
|
|
|
for(int i = 0; i < num; i++)
|
|
|
|
{
|
2010-07-29 19:09:01 +00:00
|
|
|
boundary = BreakIterator::createWordInstance("en", *status);
|
2010-07-09 16:14:46 +00:00
|
|
|
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 );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2010-07-29 19:09:01 +00:00
|
|
|
|
2010-07-09 16:14:46 +00:00
|
|
|
else // character iteration
|
|
|
|
{
|
|
|
|
for(int i = 0; i < num; i++)
|
2010-07-29 19:09:01 +00:00
|
|
|
{
|
2010-07-09 16:14:46 +00:00
|
|
|
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;
|
2010-07-29 19:09:01 +00:00
|
|
|
char locale[25];
|
2010-07-09 16:14:46 +00:00
|
|
|
public:
|
|
|
|
|
|
|
|
DateFmtFunction()
|
|
|
|
{
|
|
|
|
num = -1;
|
|
|
|
}
|
|
|
|
|
2010-07-29 19:09:01 +00:00
|
|
|
DateFmtFunction(int a, const char* loc)
|
2010-07-09 16:14:46 +00:00
|
|
|
{
|
|
|
|
num = a;
|
2010-07-29 19:09:01 +00:00
|
|
|
strcpy(locale, loc);
|
2010-07-09 16:14:46 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
virtual void call(UErrorCode* status)
|
|
|
|
{
|
2010-07-29 19:09:01 +00:00
|
|
|
|
2010-07-09 16:14:46 +00:00
|
|
|
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);
|
|
|
|
|
2010-07-29 19:09:01 +00:00
|
|
|
Locale loc(locale);
|
2010-07-09 16:14:46 +00:00
|
|
|
DateFormat *fmt;
|
|
|
|
fmt = DateFormat::createDateTimeInstance(
|
2010-07-29 19:09:01 +00:00
|
|
|
DateFormat::kShort, DateFormat::kFull, loc);
|
2010-07-09 16:14:46 +00:00
|
|
|
|
|
|
|
|
|
|
|
// (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);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
};
|
|
|
|
|
2010-07-29 19:09:01 +00:00
|
|
|
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;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
class CollationFunction : public UPerfFunction
|
|
|
|
{
|
|
|
|
|
|
|
|
private:
|
|
|
|
int num;
|
|
|
|
char locale[25];
|
|
|
|
public:
|
|
|
|
|
|
|
|
CollationFunction()
|
|
|
|
{
|
|
|
|
num = -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
CollationFunction(int a, const char* loc)
|
|
|
|
{
|
|
|
|
num = a;
|
|
|
|
strcpy(locale, loc);
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual void call(UErrorCode* status2)
|
|
|
|
{
|
|
|
|
uint32_t listSize = sizeof(collation_strings)/sizeof(collation_strings[0]);
|
|
|
|
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;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2010-07-09 16:14:46 +00:00
|
|
|
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* BreakItWord250();
|
|
|
|
UPerfFunction* BreakItWord10000();
|
|
|
|
UPerfFunction* BreakItChar250();
|
|
|
|
UPerfFunction* BreakItChar10000();
|
2010-07-29 19:09:01 +00:00
|
|
|
UPerfFunction* NumFmt10000();
|
|
|
|
UPerfFunction* NumFmt100000();
|
|
|
|
UPerfFunction* Collation10000();
|
|
|
|
UPerfFunction* Collation100000();
|
2010-07-09 16:14:46 +00:00
|
|
|
};
|
|
|
|
|
2010-08-05 00:26:41 +00:00
|
|
|
#endif // DateFmtPerf
|