#define RAND() ((unsigned int)((rand() & 0xff) | ((rand() & 0xff) << 8) | ((rand() & 0xff) << 16) | ((rand() & 0xff) << 24)))
#define GIGA (uint64_t)((uint64_t)0x400*(uint64_t)0x400*(uint64_t)0x400)
#define MAX_LENGTH_SECTION 0x1f
#define MAX_LENGTH_TEXT 0x7f
#define MAX_LENGTH_BUF 0xff
static char const * asz_mon[ 12 ] = {
"Jan" ,"Feb" ,"Mar" ,"Apr" ,"May" ,"Jun" ,
"Jul" ,"Aug" ,"Sep" ,"Oct" ,"Nov" ,"Dec" } ;
static char * gen_string1( char * sz_string, size_t u_length) {
unsigned char * psz_string = ( unsigned char * ) sz_string;
unsigned char * psz_string_end = ( unsigned char * ) sz_string + u_length;
for ( ; psz_string < psz_string_end; ++ psz_string) {
* psz_string = 0x21 + ( RAND( ) % 94 ) ;
}
* psz_string = '\0 ' ;
return sz_string;
}
static char * gen_string2( char * sz_string, size_t u_length) {
unsigned char c;
char * psz_string = sz_string;
char * psz_string_end = sz_string + u_length;
for ( ; psz_string < psz_string_end; ++ psz_string) {
c = RAND( ) % 52 ;
if ( c < 26 ) {
* psz_string = 'A' + c;
} else {
* psz_string = 'a' + ( c - 26 ) ;
}
}
* psz_string = '\0 ' ;
return sz_string;
}
int gen_data( FILE * fp, size_t u_GiB) {
char sz_buf[ MAX_LENGTH_BUF+ 1 ] ;
unsigned int u_string_recs;
unsigned int u_hatena;
unsigned int u_xy_recs;
unsigned int u_num_recs;
char x_or_y;
uint64_t u_pos;
time_t u_time = time ( NULL ) ;
tm * pt_tm = :: localtime ( & u_time) ;
char sz_timestamp[ 0x20 ] ;
:: sprintf ( sz_timestamp, "%s %.2d %02d:%02d:%02d %d" , asz_mon[ pt_tm- > tm_mon] , pt_tm- > tm_mday, pt_tm- > tm_hour, pt_tm- > tm_min, pt_tm- > tm_sec, pt_tm- > tm_year+ 1900 ) ;
char sz_qq_buf[ MAX_LENGTH_BUF+ 0x40 + 1 ] ;
// タイトル、倍率(mag)。(この係数を使って④の数値から割る、正の整数。unsigned int)
:: srand ( ( unsigned int ) time ( NULL ) ) ;
:: fprintf ( fp, "%s %u\x0a " , gen_string2( sz_buf, 1 + ( RAND( ) % MAX_LENGTH_SECTION) ) , RAND( ) ) ;
for ( ;; ) {
// ①セクション名
// ex. SECTION_NAME ①
:: fprintf ( fp, "%s\x0a " , gen_string2( sz_buf, 1 + ( RAND( ) % MAX_LENGTH_SECTION) ) ) ;
// ②セクション集計値
// ex. 11200 11200 2 Jun 9 23:23:00 2018 ②
// u_xy_recs = RAND() % 0x8000000 + 1;
u_xy_recs = RAND( ) % 0x8000 + 1 ;
u_hatena = u_xy_recs;
u_string_recs = RAND( ) % 0x20 + 1 ;
:: fprintf ( fp, "%u %u %u %s\x0a " , u_xy_recs, u_hatena, u_string_recs, sz_timestamp) ;
// ③テキスト
// ex. This is pen. ③
// hello world. ③
for ( unsigned int j = 0 ; j < u_string_recs; ++ j) {
:: fprintf ( fp, "%s\x0a " , gen_string1( sz_buf, 1 + ( RAND( ) % MAX_LENGTH_TEXT) ) ) ;
}
// ④数値
// ex. x 1 2 ④
for ( unsigned int j = 0 ; j < u_xy_recs; ++ j) {
x_or_y = ( RAND( ) % 2 ) ? 'x' : 'y' ;
// u_num_recs = RAND() % 0x400 + 1;
u_num_recs = RAND( ) % 0x100 + 1 ;
switch ( RAND( ) % 3 ) {
case 0 :
// ⑤QQ 謎の行
// QQ subname -1 0 0 1 -21000000 600000 2
// →この行が無いファイルもある。
// →7,8カラムは1行目の倍率数字で割る(数値行と同じ計算)
// →3~6カラムは -1, 0, 1, のいずれか。この4つの組み合わせであとで計算に使うことがある。
// 1 0 0 1 →TYPE_A
// 0 1 -1 0 →TYPE_B
// -1 0 0 -1 →TYPE_C
// 0 -1 1 0 →TYPE_D
// -1 0 0 1 →TYPE_E
// 0 -1 -1 0 →TYPE_F
// 1 0 0 -1 →TYPE_G
// 0 1 1 0 →TYPE_H
// →9カラム目は正の整数。(unsigned int)
// →下記6つのバリエーションがあります。
// QQ subname
// QQ subname c
// QQ subname -1 0 0 1 -21000000 600000
// QQ subname -1 0 0 1 -21000000 600000 2
// QQ subname c -1 0 0 1 -21000000 600000
// QQ subname c -1 0 0 1 -21000000 600000 2
// ※cはcという文字列固定
// ※1つのファイルに存在するのはいずれか1つのフォーマット
switch ( RAND( ) % 3 ) {
case 0 :
:: sprintf ( sz_qq_buf, "%s" ,
RAND( ) % 2 ? " c" : "" ) ;
break ;
case 1 :
:: sprintf ( sz_qq_buf, "%s %d %d %d %d %d %d" ,
RAND( ) % 2 ? " c" : "" ,
( int ) ( RAND( ) % 3 ) - 1 , ( int ) ( RAND( ) % 3 ) - 1 , ( int ) ( RAND( ) % 3 ) - 1 , ( int ) ( RAND( ) % 3 ) - 1 ,
( int32_t ) RAND( ) , ( int32_t ) RAND( ) ) ;
break ;
default :
:: sprintf ( sz_qq_buf, "%s %d %d %d %d %d %d %u" ,
RAND( ) % 2 ? " c" : "" ,
( int ) ( RAND( ) % 3 ) - 1 , ( int ) ( RAND( ) % 3 ) - 1 , ( int ) ( RAND( ) % 3 ) - 1 , ( int ) ( RAND( ) % 3 ) - 1 ,
( int32_t ) RAND( ) , ( int32_t ) RAND( ) , RAND( ) ) ;
break ;
}
// 上記QQ行は先頭のQQがなくなってx, y行に存在することがあります。
// 1つのファイル内でQQ行とx, yの後ろにくるパターンの両方は存在せず、片方が出てきたらもう片方は出てきません。
// x, yと同じ行にくる場合のフォーマットもQQ行の規則と同じです。
// x 1 234 subname -1 0 0 1 -21000000 600000 2
// y 2 234 subname -1 0 0 1 -21000000 600000 2
if ( RAND( ) % 2 ) {
:: fprintf ( fp, "%c %u %u%s\x0a " , x_or_y, j + 1 , u_num_recs, sz_qq_buf) ;
} else {
:: fprintf ( fp, "%c %u %u\x0a " , x_or_y, j + 1 , u_num_recs) ;
:: fprintf ( fp, "QQ %s%s\x0a " , gen_string2( sz_buf, 1 + ( RAND( ) % MAX_LENGTH_SECTION) ) , sz_qq_buf) ;
}
break ;
case 1 :
// RR行があります。
// これもQQ同様x, yの次の行にあったりなかったりします。
// QQとRRが一緒に出てくることは無いと思います。
// フォーマットは「RR 正の整数」でQQのようにx,yの後ろに来ることはありません。
:: fprintf ( fp, "%c %u %u\x0a " , x_or_y, j + 1 , u_num_recs) ;
:: fprintf ( fp, "RR %u\x0a " , RAND( ) ) ;
break ;
default :
:: fprintf ( fp, "%c %u %u\x0a " , x_or_y, j + 1 , u_num_recs) ;
break ;
}
if ( x_or_y == 'x' ) {
for ( unsigned int k = 0 ; k < u_num_recs; ++ k) {
// 100 1 -2000 10
:: fprintf ( fp, "%d %d %d %d\x0a " ,
( int32_t ) RAND( ) , ( int32_t ) RAND( ) , ( int32_t ) RAND( ) , ( int32_t ) RAND( ) ) ;
}
} else {
for ( unsigned int k = 0 ; k < u_num_recs; ++ k) {
// -100 10000
:: fprintf ( fp, "%d %d\x0a " ,
( int32_t ) RAND( ) , ( int32_t ) RAND( ) ) ;
}
}
}
u_pos = ftello( fp) ;
if ( u_pos > GIGA * ( uint64_t ) u_GiB) {
break ;
}
}
return 0 ;
}
int main( int argc, char ** argv) {
if ( argc ! = 3 ) {
:: fprintf ( stderr , "Usage: %s <filepath> <GiB size>\n " , argv[ 0 ] ) ;
return 1 ;
}
char * filename = argv[ 1 ] ;
size_t u_GiB = :: strtoul ( argv[ 2 ] , NULL , 10 ) ;;
if ( u_GiB == 0 ) {
:: fprintf ( stderr , "Usage: %s <filepath> <GiB size>\n " , argv[ 0 ] ) ;
:: fprintf ( stderr , "<GiB size> must be greater than zero\n " ) ;
return 1 ;
}
FILE * fp = :: fopen ( filename, "wb" ) ;
if ( fp == NULL ) {
:: fprintf ( stderr , "fatal error: open file error \" %s\" \n " , filename) ;
return 1 ;
}
if ( :: gen_data ( fp, u_GiB) < 0 ) { fclose ( fp) ; return - 1 ; }
fclose ( fp) ;
return 0 ;
}

#define RAND() ((unsigned int)((rand() & 0xff) | ((rand() & 0xff) << 8) | ((rand() & 0xff) << 16) | ((rand() & 0xff) << 24)))
#define GIGA               (uint64_t)((uint64_t)0x400*(uint64_t)0x400*(uint64_t)0x400)
#define MAX_LENGTH_SECTION 0x1f
#define MAX_LENGTH_TEXT    0x7f
#define MAX_LENGTH_BUF     0xff

static char const* asz_mon[12] = {
"Jan","Feb","Mar","Apr","May","Jun",
"Jul","Aug","Sep","Oct","Nov","Dec"};

static char* gen_string1(char* sz_string, size_t u_length) {
	
	unsigned char* psz_string = (unsigned char*)sz_string;
	unsigned char* psz_string_end = (unsigned char*)sz_string + u_length;

	for (; psz_string < psz_string_end; ++psz_string) {
		*psz_string = 0x21 + (RAND() % 94);
	}

	*psz_string = '\0';
	return sz_string;
}

static char* gen_string2(char* sz_string, size_t u_length) {
	
	unsigned char c;
	char* psz_string = sz_string;
	char* psz_string_end = sz_string + u_length;

	for (; psz_string < psz_string_end; ++psz_string) {
		c = RAND() % 52;
		if (c < 26) {
			*psz_string = 'A' + c;
		} else {
			*psz_string = 'a' + (c -26);
		}
	}

	*psz_string = '\0';
	return sz_string;
}

int gen_data(FILE* fp, size_t u_GiB) {

	char sz_buf[MAX_LENGTH_BUF+1];

	unsigned int u_string_recs;
	unsigned int u_hatena;
	unsigned int u_xy_recs;
	unsigned int u_num_recs;
	char x_or_y;
	uint64_t u_pos;

	time_t u_time = time(NULL);
	tm*    pt_tm = ::localtime(&u_time);
	char sz_timestamp[0x20];
	::sprintf(sz_timestamp, "%s %.2d %02d:%02d:%02d %d", asz_mon[pt_tm->tm_mon], pt_tm->tm_mday, pt_tm->tm_hour, pt_tm->tm_min, pt_tm->tm_sec, pt_tm->tm_year+1900);

	char sz_qq_buf[MAX_LENGTH_BUF+0x40+1];

	// タイトル、倍率(mag)。(この係数を使って④の数値から割る、正の整数。unsigned int)
	::srand((unsigned int)time(NULL));
	::fprintf(fp, "%s %u\x0a", gen_string2(sz_buf, 1 + (RAND() % MAX_LENGTH_SECTION)), RAND());

	for (;;) {
		// ①ｾｸｼｮﾝ名
		// ex. SECTION_NAME　①
		::fprintf(fp, "%s\x0a", gen_string2(sz_buf, 1 + (RAND() % MAX_LENGTH_SECTION)));

		// ②ｾｸｼｮﾝ集計値
		// ex. 11200 11200 2 Jun  9 23:23:00 2018　②
//		u_xy_recs = RAND() % 0x8000000 + 1;
		u_xy_recs = RAND() % 0x8000 + 1;
		u_hatena  = u_xy_recs;
		u_string_recs = RAND() % 0x20 + 1;

		::fprintf(fp, "%u %u %u %s\x0a", u_xy_recs, u_hatena, u_string_recs, sz_timestamp);
	
		// ③ﾃｷｽﾄ
		// ex. This is pen.　③
		//     hello world.　③
		for (unsigned int j = 0; j < u_string_recs; ++j) {
			::fprintf(fp, "%s\x0a", gen_string1(sz_buf, 1 + (RAND() % MAX_LENGTH_TEXT)));
		}

		// ④数値
		// ex. x 1 2　④
		for (unsigned int j = 0; j < u_xy_recs; ++j) {
			x_or_y = (RAND() % 2) ? 'x' : 'y';
//			u_num_recs = RAND() % 0x400 + 1;
			u_num_recs = RAND() % 0x100 + 1;

			switch (RAND() % 3) {

				case 0:
					// ⑤QQ 謎の行
					// QQ subname -1 0 0 1 -21000000 600000 2
					// →この行が無いファイルもある。
					// →7,8カラムは1行目の倍率数字で割る（数値行と同じ計算）
					// →3～6カラムは -1, 0, 1, のいずれか。この4つの組み合わせであとで計算に使うことがある。
					// 1 0 0 1 →TYPE_A
					// 0 1 -1 0 →TYPE_B
					// -1 0 0 -1 →TYPE_C
					// 0 -1 1 0 →TYPE_D
					// -1 0 0 1 →TYPE_E
					// 0 -1 -1 0 →TYPE_F
					// 1 0 0 -1 →TYPE_G
					// 0 1 1 0 →TYPE_H
					// →9カラム目は正の整数。(unsigned int)
					// →下記6つのバリエーションがあります。
					// QQ subname
					// QQ subname c
					// QQ subname -1 0 0 1 -21000000 600000
					// QQ subname -1 0 0 1 -21000000 600000 2
					// QQ subname c -1 0 0 1 -21000000 600000
					// QQ subname c -1 0 0 1 -21000000 600000 2
					// ※cはcという文字列固定
					// ※1つのファイルに存在するのはいずれか1つのフォーマット
					switch (RAND() % 3) {
						case 0:
							::sprintf(sz_qq_buf, "%s",
							RAND() % 2 ? " c" : "");
							break;
						case 1:
							::sprintf(sz_qq_buf, "%s %d %d %d %d %d %d",
							RAND() % 2 ? " c" : "",
							(int)(RAND() % 3) - 1, (int)(RAND() % 3) - 1, (int)(RAND() % 3) - 1, (int)(RAND() % 3) - 1,
							(int32_t)RAND(), (int32_t)RAND());
							break;
						default:
							::sprintf(sz_qq_buf, "%s %d %d %d %d %d %d %u",
							RAND() % 2 ? " c" : "",
							(int)(RAND() % 3) - 1, (int)(RAND() % 3) - 1, (int)(RAND() % 3) - 1, (int)(RAND() % 3) - 1,
							(int32_t)RAND(), (int32_t)RAND(), RAND());
							break;
					}
					// 上記QQ行は先頭のQQがなくなってx, y行に存在することがあります。
					// 1つのファイル内でQQ行とx, yの後ろにくるパターンの両方は存在せず、片方が出てきたらもう片方は出てきません。
					// x, yと同じ行にくる場合のフォーマットもQQ行の規則と同じです。
					// x 1 234 subname -1 0 0 1 -21000000 600000 2
					// y 2 234 subname -1 0 0 1 -21000000 600000 2
					if (RAND() % 2) {
						::fprintf(fp, "%c %u %u%s\x0a", x_or_y, j + 1, u_num_recs, sz_qq_buf);
					} else {
						::fprintf(fp, "%c %u %u\x0a", x_or_y, j + 1, u_num_recs);
						::fprintf(fp, "QQ %s%s\x0a", gen_string2(sz_buf, 1 + (RAND() % MAX_LENGTH_SECTION)), sz_qq_buf);
					}
					break;

				case 1:
					// RR行があります。
					// これもQQ同様x, yの次の行にあったりなかったりします。
					// QQとRRが一緒に出てくることは無いと思います。
					// フォーマットは「RR 正の整数」でQQのようにx,yの後ろに来ることはありません。
					::fprintf(fp, "%c %u %u\x0a", x_or_y, j + 1, u_num_recs);
					::fprintf(fp, "RR %u\x0a", RAND());
					break;

				default:
					::fprintf(fp, "%c %u %u\x0a", x_or_y, j + 1, u_num_recs);
					break;
			}

			if (x_or_y == 'x') {
				for (unsigned int k = 0; k < u_num_recs; ++k) {
					// 100 1 -2000 10
					::fprintf(fp, "%d %d %d %d\x0a",
						(int32_t)RAND(), (int32_t)RAND(), (int32_t)RAND(), (int32_t)RAND());
				}
			} else {
				for (unsigned int k = 0; k < u_num_recs; ++k) {
					// -100 10000
					::fprintf(fp, "%d %d\x0a",
						(int32_t)RAND(), (int32_t)RAND());
				}
			}
		}

		u_pos = ftello(fp);

		if (u_pos > GIGA * (uint64_t)u_GiB) {
			break;
		}
	}

	return 0;
}

int main(int argc, char** argv) {

	if (argc != 3) {
		::fprintf(stderr, "Usage: %s <filepath> <GiB size>\n", argv[0]);
		return 1;
	}

	char* filename = argv[1];
	size_t u_GiB = ::strtoul(argv[2], NULL, 10);;

	if (u_GiB == 0) {
		::fprintf(stderr, "Usage: %s <filepath> <GiB size>\n", argv[0]);
		::fprintf(stderr, "<GiB size> must be greater than zero\n");
		return 1;
	}

	FILE* fp = ::fopen(filename, "wb");
	if (fp == NULL) {
		::fprintf(stderr, "fatal error: open file error \"%s\"\n", filename);
		return 1;
	}

	if (::gen_data(fp, u_GiB) < 0) {fclose(fp); return -1;}
	fclose(fp);
	return 0;
}
