The C based gRPC (C++, Python, Ruby, Objective-C, PHP, C#) https://grpc.io/
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 
 

424 lines
13 KiB

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <inttypes.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <sys/time.h>
#include <fcntl.h>
#include <unistd.h>
int utf8_to16_iconv(const unsigned char *buf8, size_t len8,
unsigned short *buf16, size_t *len16);
int utf8_to16_naive(const unsigned char *buf8, size_t len8,
unsigned short *buf16, size_t *len16);
static struct ftab {
const char *name;
int (*func)(const unsigned char *buf8, size_t len8,
unsigned short *buf16, size_t *len16);
} ftab[] = {
{
.name = "iconv",
.func = utf8_to16_iconv,
}, {
.name = "naive",
.func = utf8_to16_naive,
},
};
static unsigned char *load_test_buf(int len)
{
const char utf8[] = "\xF0\x90\xBF\x80";
const int utf8_len = sizeof(utf8)/sizeof(utf8[0]) - 1;
unsigned char *data = malloc(len);
unsigned char *p = data;
while (len >= utf8_len) {
memcpy(p, utf8, utf8_len);
p += utf8_len;
len -= utf8_len;
}
while (len--)
*p++ = 0x7F;
return data;
}
static unsigned char *load_test_file(int *len)
{
unsigned char *data;
int fd;
struct stat stat;
fd = open("../UTF-8-demo.txt", O_RDONLY);
if (fd == -1) {
printf("Failed to open ../UTF-8-demo.txt!\n");
exit(1);
}
if (fstat(fd, &stat) == -1) {
printf("Failed to get file size!\n");
exit(1);
}
*len = stat.st_size;
data = malloc(*len);
if (read(fd, data, *len) != *len) {
printf("Failed to read file!\n");
exit(1);
}
close(fd);
return data;
}
static void print_test(const unsigned char *data, int len)
{
printf(" [len=%d] \"", len);
while (len--)
printf("\\x%02X", *data++);
printf("\"\n");
}
struct test {
const unsigned char *data;
int len;
};
static void prepare_test_buf(unsigned char *buf, const struct test *pos,
int pos_len, int pos_idx)
{
/* Round concatenate correct tokens to 1024 bytes */
int buf_idx = 0;
while (buf_idx < 1024) {
int buf_len = 1024 - buf_idx;
if (buf_len >= pos[pos_idx].len) {
memcpy(buf+buf_idx, pos[pos_idx].data, pos[pos_idx].len);
buf_idx += pos[pos_idx].len;
} else {
memset(buf+buf_idx, 0, buf_len);
buf_idx += buf_len;
}
if (++pos_idx == pos_len)
pos_idx = 0;
}
}
/* Return 0 on success, -1 on error */
static int test_manual(const struct ftab *ftab, unsigned short *buf16,
unsigned short *_buf16)
{
#define LEN16 4096
#pragma GCC diagnostic push
#pragma GCC diagnostic ignored "-Wpointer-sign"
/* positive tests */
static const struct test pos[] = {
{"", 0},
{"\x00", 1},
{"\x66", 1},
{"\x7F", 1},
{"\x00\x7F", 2},
{"\x7F\x00", 2},
{"\xC2\x80", 2},
{"\xDF\xBF", 2},
{"\xE0\xA0\x80", 3},
{"\xE0\xA0\xBF", 3},
{"\xED\x9F\x80", 3},
{"\xEF\x80\xBF", 3},
{"\xF0\x90\xBF\x80", 4},
{"\xF2\x81\xBE\x99", 4},
{"\xF4\x8F\x88\xAA", 4},
};
/* negative tests */
static const struct test neg[] = {
{"\x80", 1},
{"\xBF", 1},
{"\xC0\x80", 2},
{"\xC1\x00", 2},
{"\xC2\x7F", 2},
{"\xDF\xC0", 2},
{"\xE0\x9F\x80", 3},
{"\xE0\xC2\x80", 3},
{"\xED\xA0\x80", 3},
{"\xED\x7F\x80", 3},
{"\xEF\x80\x00", 3},
{"\xF0\x8F\x80\x80", 4},
{"\xF0\xEE\x80\x80", 4},
{"\xF2\x90\x91\x7F", 4},
{"\xF4\x90\x88\xAA", 4},
{"\xF4\x00\xBF\xBF", 4},
{"\x00\x00\x00\x00\x00\xC2\x80\x00\x00\x00\xE1\x80\x80\x00\x00\xC2" \
"\xC2\x80\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00",
32},
{"\x00\x00\x00\x00\x00\xC2\xC2\x80\x00\x00\xE1\x80\x80\x00\x00\x00",
16},
{"\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00" \
"\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\xF1\x80",
32},
{"\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00" \
"\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\xF1",
32},
{"\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00" \
"\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\xF1\x80" \
"\x80", 33},
{"\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00" \
"\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\xF1\x80" \
"\xC2\x80", 34},
{"\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00" \
"\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\xF0" \
"\x80\x80\x80", 35},
};
#pragma GCC diagnostic push
size_t len16 = LEN16, _len16 = LEN16;
int ret, _ret;
/* Test single token */
for (int i = 0; i < sizeof(pos)/sizeof(pos[0]); ++i) {
ret = ftab->func(pos[i].data, pos[i].len, buf16, &len16);
_ret = utf8_to16_iconv(pos[i].data, pos[i].len, _buf16, &_len16);
if (ret != _ret || len16 != _len16 || memcmp(buf16, _buf16, len16)) {
printf("FAILED positive test(%d:%d, %lu:%lu): ",
ret, _ret, len16, _len16);
print_test(pos[i].data, pos[i].len);
return -1;
}
len16 = _len16 = LEN16;
}
for (int i = 0; i < sizeof(neg)/sizeof(neg[0]); ++i) {
ret = ftab->func(neg[i].data, neg[i].len, buf16, &len16);
_ret = utf8_to16_iconv(neg[i].data, neg[i].len, _buf16, &_len16);
if (ret != _ret || len16 != _len16 || memcmp(buf16, _buf16, len16)) {
printf("FAILED negitive test(%d:%d, %lu:%lu): ",
ret, _ret, len16, _len16);
print_test(neg[i].data, neg[i].len);
return -1;
}
len16 = _len16 = LEN16;
}
/* Test shifted buffer to cover 1k length */
/* buffer size must be greater than 1024 + 16 + max(test string length) */
const int max_size = 1024*2;
uint64_t buf64[max_size/8 + 2];
/* Offset 8 bytes by 1 byte */
unsigned char *buf = ((unsigned char *)buf64) + 1;
int buf_len;
for (int i = 0; i < sizeof(pos)/sizeof(pos[0]); ++i) {
/* Positive test: shift 16 bytes, validate each shift */
prepare_test_buf(buf, pos, sizeof(pos)/sizeof(pos[0]), i);
buf_len = 1024;
for (int j = 0; j < 16; ++j) {
ret = ftab->func(buf, buf_len, buf16, &len16);
_ret = utf8_to16_iconv(buf, buf_len, _buf16, &_len16);
if (ret != _ret || len16 != _len16 || \
memcmp(buf16, _buf16, len16)) {
printf("FAILED positive test(%d:%d, %lu:%lu): ",
ret, _ret, len16, _len16);
print_test(buf, buf_len);
return -1;
}
len16 = _len16 = LEN16;
for (int k = buf_len; k >= 1; --k)
buf[k] = buf[k-1];
buf[0] = '\x55';
++buf_len;
}
/* Negative test: trunk last non ascii */
while (buf_len >= 1 && buf[buf_len-1] <= 0x7F)
--buf_len;
if (buf_len) {
ret = ftab->func(buf, buf_len-1, buf16, &len16);
_ret = utf8_to16_iconv(buf, buf_len-1, _buf16, &_len16);
if (ret != _ret || len16 != _len16 || \
memcmp(buf16, _buf16, len16)) {
printf("FAILED negative test(%d:%d, %lu:%lu): ",
ret, _ret, len16, _len16);
print_test(buf, buf_len-1);
return -1;
}
len16 = _len16 = LEN16;
}
}
/* Negative test */
for (int i = 0; i < sizeof(neg)/sizeof(neg[0]); ++i) {
/* Append one error token, shift 16 bytes, validate each shift */
int pos_idx = i % (sizeof(pos)/sizeof(pos[0]));
prepare_test_buf(buf, pos, sizeof(pos)/sizeof(pos[0]), pos_idx);
memcpy(buf+1024, neg[i].data, neg[i].len);
buf_len = 1024 + neg[i].len;
for (int j = 0; j < 16; ++j) {
ret = ftab->func(buf, buf_len, buf16, &len16);
_ret = utf8_to16_iconv(buf, buf_len, _buf16, &_len16);
if (ret != _ret || len16 != _len16 || \
memcmp(buf16, _buf16, len16)) {
printf("FAILED negative test(%d:%d, %lu:%lu): ",
ret, _ret, len16, _len16);
print_test(buf, buf_len);
return -1;
}
len16 = _len16 = LEN16;
for (int k = buf_len; k >= 1; --k)
buf[k] = buf[k-1];
buf[0] = '\x66';
++buf_len;
}
}
return 0;
}
static void test(const unsigned char *buf8, size_t len8,
unsigned short *buf16, size_t len16, const struct ftab *ftab)
{
/* Use iconv as the reference answer */
if (strcmp(ftab->name, "iconv") == 0)
return;
printf("%s\n", ftab->name);
/* Test file or buffer */
size_t _len16 = len16;
unsigned short *_buf16 = (unsigned short *)malloc(_len16);
if (utf8_to16_iconv(buf8, len8, _buf16, &_len16)) {
printf("Invalid test file or buffer!\n");
exit(1);
}
printf("standard test: ");
if (ftab->func(buf8, len8, buf16, &len16) || len16 != _len16 || \
memcmp(buf16, _buf16, len16) != 0)
printf("FAIL\n");
else
printf("pass\n");
free(_buf16);
/* Manual cases */
unsigned short *mbuf8 = (unsigned short *)malloc(LEN16);
unsigned short *mbuf16 = (unsigned short *)malloc(LEN16);
printf("manual test: %s\n",
test_manual(ftab, mbuf8, mbuf16) ? "FAIL" : "pass");
free(mbuf8);
free(mbuf16);
printf("\n");
}
static void bench(const unsigned char *buf8, size_t len8,
unsigned short *buf16, size_t len16, const struct ftab *ftab)
{
const int loops = 1024*1024*1024/len8;
int ret = 0;
double time, size;
struct timeval tv1, tv2;
fprintf(stderr, "bench %s... ", ftab->name);
gettimeofday(&tv1, 0);
for (int i = 0; i < loops; ++i)
ret |= ftab->func(buf8, len8, buf16, &len16);
gettimeofday(&tv2, 0);
printf("%s\n", ret?"FAIL":"pass");
time = tv2.tv_usec - tv1.tv_usec;
time = time / 1000000 + tv2.tv_sec - tv1.tv_sec;
size = ((double)len8 * loops) / (1024*1024);
printf("time: %.4f s\n", time);
printf("data: %.0f MB\n", size);
printf("BW: %.2f MB/s\n", size / time);
printf("\n");
}
static void usage(const char *bin)
{
printf("Usage:\n");
printf("%s test [alg] ==> test all or one algorithm\n", bin);
printf("%s bench [alg] ==> benchmark all or one algorithm\n", bin);
printf("%s bench size NUM ==> benchmark with specific buffer size\n", bin);
printf("alg = ");
for (int i = 0; i < sizeof(ftab)/sizeof(ftab[0]); ++i)
printf("%s ", ftab[i].name);
printf("\nNUM = buffer size in bytes, 1 ~ 67108864(64M)\n");
}
int main(int argc, char *argv[])
{
int len8 = 0, len16;
unsigned char *buf8;
unsigned short *buf16;
const char *alg = NULL;
void (*tb)(const unsigned char *buf8, size_t len8,
unsigned short *buf16, size_t len16, const struct ftab *ftab);
tb = NULL;
if (argc >= 2) {
if (strcmp(argv[1], "test") == 0)
tb = test;
else if (strcmp(argv[1], "bench") == 0)
tb = bench;
if (argc >= 3) {
alg = argv[2];
if (strcmp(alg, "size") == 0) {
if (argc < 4) {
tb = NULL;
} else {
alg = NULL;
len8 = atoi(argv[3]);
if (len8 <= 0 || len8 > 67108864) {
printf("Buffer size error!\n\n");
tb = NULL;
}
}
}
}
}
if (tb == NULL) {
usage(argv[0]);
return 1;
}
/* Load UTF8 test buffer */
if (len8)
buf8 = load_test_buf(len8);
else
buf8 = load_test_file(&len8);
/* Prepare UTF16 buffer large enough */
len16 = len8 * 2;
buf16 = (unsigned short *)malloc(len16);
if (tb == bench)
printf("============== Bench UTF8 (%d bytes) ==============\n", len8);
for (int i = 0; i < sizeof(ftab)/sizeof(ftab[0]); ++i) {
if (alg && strcmp(alg, ftab[i].name) != 0)
continue;
tb((const unsigned char *)buf8, len8, buf16, len16, &ftab[i]);
}
#if 0
if (tb == bench) {
printf("==================== Bench ASCII ====================\n");
/* Change test buffer to ascii */
for (int i = 0; i < len; i++)
data[i] &= 0x7F;
for (int i = 0; i < sizeof(ftab)/sizeof(ftab[0]); ++i) {
if (alg && strcmp(alg, ftab[i].name) != 0)
continue;
tb((const unsigned char *)data, len, &ftab[i]);
printf("\n");
}
}
#endif
return 0;
}