#include <iostream>
#include <vector>
#include <stack>
#include <string>
#include <bitset>
#include <chrono>

using namespace std;

struct Pieces
{
	uint16_t stick = 0;
	uint16_t hand = 0;
	uint16_t free = 0;
	uint16_t running = 0;
	uint16_t stop = 0;
	uint16_t stock = 0;
	uint16_t dead = 0;
	uint8_t stick_size = 0;
	uint8_t hand_size = 0;
	uint8_t free_size = 0;
	uint8_t running_size = 0;
	uint8_t dead_size = 0;
};

struct Position
{
	uint64_t board = 0;
	vector<Position *> children; // 24 Bytes
};

const uint8_t pos24[] = {
	0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 
	0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11
};

void coutBoard(uint64_t board, string c_name = "board", bool display_board = true)
{
	cout << c_name << "->";
	cout <<  "value: ";
	cout << int(board >> 60);
	cout <<  " ; player: ";
	cout << int((board >> 48) & 1);
	cout <<  " ; lastmove: ";
	cout << int((board >> 49) & 0xf);
	cout <<  " ; maxdeep: ";
	cout << int((board >> 53) & 0b1111111);
	if (display_board) {
		cout << endl <<  " ; board: ";
		for (int i = 0; i < 12; ++i) {
			cout << int((board >> (i << 2)) & 0xf);
			cout << ", ";
		}
	}
	cout << endl;
}

void coutPieces(uint16_t piece, string c_name = "piece")
{
	cout << c_name << ": ";
	cout << bitset<4>(piece >> 12);
	cout <<  " ";
	cout << bitset<4>(piece >> 8);
	cout <<  " ";
	cout << bitset<4>(piece >> 4);
	cout <<  " ";
	cout << bitset<4>(piece);
	cout << endl;
}

// uint64_t *board_map = new uint64_t[1ll << 24];
vector<Position*>board_map(1ll << 24);

uint32_t random_board[12][0xd] = {
	{0, 1804289383, 1505335290, 1205554746, 1968078301, 590011675, 290852541,
		1045618677, 757547896, 444454915, 1215069295, 1989311423, 1687063760},
	{0, 1227918265, 1995168598, 623271449, 319571911, 1086411056, 1857631170,
		1562469902, 188364873, 2017252061, 1708421557, 1416980517, 1110582131},
	{0, 661315303, 362860358, 1126333064, 818749156, 514134199, 1284681320,
		2055460636, 1759697306, 1439342445, 71876166, 842133572, 1610048249},
	{0, 89789692, 1920540202, 1633333913, 1322670498, 2099602078, 707676680,
		415587398, 111714879, 1956681178, 1646153577, 272081135, 2115950905},
	{0, 598268513, 286129175, 1049489039, 1834256778, 1526261789, 1224552512,
		912370068, 608905406, 296121872, 2144173174, 766615507, 469800673},
	{0, 1096799004, 1856146135, 1557536497, 183990277, 2023096691, 1714566222,
		351897519, 38645617, 1888747329, 1582528011, 1266305335, 2049663803},
	{0, 1599854414, 214780621, 988039572, 677741240, 1455091466, 81038786,
		839148289, 537411375, 1309058823, 2085997689, 707956661, 1473834340},
	{0, 1019288695, 1795081523, 421093279, 114810857, 1954323550, 1662292864,
		267189329, 1049253433, 1819267000, 1503847010, 128959393, 1976973646},
	{0, 1530438140, 158946141, 1992573474, 612787484, 1396673875, 1085856331,
		1854680614, 479406137, 171676246, 938891628, 640924275, 331817660},
	{0, 2029893588, 646125617, 353272483, 33267067, 820885701, 509160483,
		212803518, 979730751, 681628745, 1443730104, 1145702396, 1907921120},
	{0, 383803757, 1160721591, 1920542938, 543497852, 1318701851, 1012836014,
		708439997, 405830208, 101133139, 872194726, 574014618, 252959875},
	{0, 889299482, 1654321064, 283545993, 1058850617, 746974583, 1511305073,
		1220372842, 907127155, 594260387, 1372829053, 2140627706, 772686469}
};
uint32_t random_lastmove[] = {
	0, 462648444, 965671178, 1468590567, 901501096, 1396130115, 829939201,
	255697463, 755944791, 178959410, 691212257, 119126546, 631979362
};
uint32_t random_player[] = {842357681, 1505898414};

/*
void setRandomBoard()
{
	for (uint32_t i = 0; i < 12; ++i) {
		random_board[i][0] = 0;
		for (uint32_t j = 1; j <= 12; ++j) {
			srand((i << 4) | j);
			random_board[i][j] = rand();
		}
	}
	random_lastmove[0] = 0;
	for (uint32_t i = 1; i <= 12; ++i) {
		srand(i << 4);
		random_lastmove[i] = rand();
	}
	srand(0x0000ffff);
	random_player[0] = rand();
	srand(0xffff0000);
	random_player[1] = rand();
}
*/

uint32_t getBoardMapKey(uint64_t board)
{
	// make the key
	uint32_t key = random_board[0][board & 0xf]
		^ random_board[1][(board >> 4) & 0xf]
		^ random_board[2][(board >> 8) & 0xf]
		^ random_board[3][(board >> 12) & 0xf]
		^ random_board[4][(board >> 16) & 0xf]
		^ random_board[5][(board >> 20) & 0xf]
		^ random_board[6][(board >> 24) & 0xf]
		^ random_board[7][(board >> 28) & 0xf]
		^ random_board[8][(board >> 32) & 0xf]
		^ random_board[9][(board >> 36) & 0xf]
		^ random_board[10][(board >> 40) & 0xf]
		^ random_board[11][(board >> 44) & 0xf]
		^ random_player[(board >> 48) & 1]
		^ random_lastmove[(board >> 49) & 0xf];
	return key;
}

Position *getBoardMap(Position *pos)
{
	uint32_t key = getBoardMapKey(pos->board);
	uint64_t sub_board = pos->board << 11;
	uint32_t masked_key = key & 0xffffff;
	while (board_map[masked_key] != nullptr) {
		if (board_map[masked_key]->board << 11 == sub_board) {
			return board_map[masked_key];
		}
		key++;
		masked_key = key & 0xffffff;
	}
	return pos;
}

void setBoardMap(Position *pos)
{
	uint32_t key = getBoardMapKey(pos->board);
	uint64_t sub_board = pos->board << 11;
	uint32_t masked_key = key & 0xffffff;
	while (board_map[masked_key] != nullptr) {
		if (board_map[masked_key]->board << 11 == sub_board) {
			break;
		}
		key++;
		masked_key = key & 0xffffff;
	}
	board_map[masked_key] = pos;
}

void coutMovelist(vector<uint8_t> &movelist)
{
	cout << "movelist: ";
	for (int i = 0; i < movelist.size(); ++i) {
		cout << (int)movelist[i] << " ";
	}
	cout << endl;
}

// indexOfBoard
int8_t iob(uint64_t board, uint8_t c)
{
	for (uint8_t i = 0; i < 12; ++i) {
		if (c == (board >> (i << 2) & 0xf)) {
			return i;
		}
	}
	return -1;
}

// pieceOfBoard
inline uint8_t pob(uint64_t board, int8_t c_pos)
{
	return board >> (c_pos << 2) & 0xf;
}

uint16_t getRunPos(uint64_t board, uint8_t c, int8_t c_pos)
{
	uint16_t run_pos = 0;
	// while loop next pos
	uint16_t next_pos = pos24[c_pos + c];
	uint8_t piece_of_next_pos = pob(board, next_pos);
	while (piece_of_next_pos != next_pos + 1) {
		run_pos |= (1 << next_pos);
		// run back to start position
		if (next_pos == c_pos) {
			run_pos |= 1 << 15;
			break;
		}
		// run into its right position
		if (c == next_pos + 1) {
			break;
		}
		next_pos = pos24[next_pos + c];
		piece_of_next_pos = pob(board, next_pos);
	}
	return run_pos;
}

#define CB ((i ^ player) & 1)
#define CBI ((~(i ^ player)) & 1)
#define PCB (((i | p) ^ player) & 1)
#define PCBI ((~((i | p) ^ player)) & 1)
unsigned int max_loop_depth = 24;
Pieces piecesValue(uint64_t board, int8_t depth)
{
	Pieces new_pieces;
	uint8_t player = (board >> 48) & 1; // 0 is odd, 1 is even
	uint16_t run_pos_sum = 0, stop_pos_sum = 0;
	uint16_t odd_run_pos_sum = 0, odd_next_pos_sum = 0;
	uint16_t even_in_odd = 0, even_in_odd_sum = 0, even_in_odd_run_pos = 0;
	uint16_t if_can_loop = 0;
	for (uint8_t i = 0; i < 12; ++i) {
		uint16_t c = i + 1;
		int8_t c_pos = iob(board, c);
		// stick
		if (c == pob(board, i)) {
			new_pieces.stick |= (1 << i);
			new_pieces.stick_size = (new_pieces.stick_size & (0xf << (CBI << 2)))
			| (((new_pieces.stick_size >> (CB << 2)) & 0xf) + 1) << (CB << 2);
		}
		// hand
		else if (c_pos == -1) {
			new_pieces.hand |= (1 << i);
			new_pieces.hand_size = (new_pieces.hand_size & (0xf << (CBI << 2)))
			| (((new_pieces.hand_size >> (CB << 2)) & 0xf) + 1) << (CB << 2);
		}
		// run, stop
		else {
			uint16_t c_run_pos = getRunPos(board, c, c_pos);
			if (c_run_pos == 0) {
				new_pieces.stop |= (1 << i);
				stop_pos_sum |= (1 << c_pos);
			} else {
				run_pos_sum |= c_run_pos;
				// if odd
				if (c & 1) {
					odd_run_pos_sum |= c_run_pos;
					odd_next_pos_sum |= 1 << pos24[c_pos + c];
				}
				// if piece is 12, or depth less than max,
				// or next step eating, or no loop
				if (c == 12 || depth <= max_loop_depth ||
					pob(board, pos24[c_pos + c]) > 0 || ~(c_run_pos >> 15) & 1) {
					new_pieces.running |= (1 << i);
					new_pieces.running_size++;
					// if even in odd
					if (i & 1 && ~c_pos & 1 && c != 12) {
						even_in_odd_sum++;
						even_in_odd_run_pos = c_run_pos;
						even_in_odd_run_pos |= 1 << c_pos;
						even_in_odd = 1 << i;
					}
				} else if ((c_run_pos >> 15) & 1) {
					if_can_loop |= 1 << i;
				}
			}
		}
	}
	// even in odd is dead
	if (1 == even_in_odd_sum &&
			!((odd_run_pos_sum & even_in_odd_run_pos) << 1)) {
		if (even_in_odd_run_pos >> 15 &&
				!((stop_pos_sum & even_in_odd_run_pos) << 1)) {
			new_pieces.running ^= even_in_odd;
			new_pieces.running_size--;
		}
		new_pieces.dead |= even_in_odd;
		new_pieces.dead_size |= 1 << ((~player & 1) << 2);
	}
	// if loop prevent can run
	for (int i = 1; i < 10; ++i) {
		if ((if_can_loop >> i) & 1) {
			int c = i + 1;
			int c_pos = iob(board, c);
			int c_next_pos = pos24[c_pos + c];
			// if player is even
			if (player) {
				// if piece is even, and position is odd,
				// and current pos can [not] be eated by odd,
				// and next pos can be eated by odd
				if (i & 1 && ~c_pos & 1 && ~(odd_next_pos_sum >> c_pos) & 1
					&& (odd_next_pos_sum >> c_next_pos) & 1) {
					new_pieces.running |= (1 << i);
					new_pieces.running_size++;
				}
			}
			// if player is odd
			else {
				// if piece is even, and position is odd,
				// and current pos can be eated by odd,
				// and next pos can [not] be eated by odd
				if (i & 1 && ~c_pos & 1 && (odd_next_pos_sum >> c_pos) & 1
					&& ~(odd_next_pos_sum >> c_next_pos) & 1) {
					new_pieces.running |= (1 << i);
					new_pieces.running_size++;
				}
				// if piece is odd, and position is even,
				else if (c & 1 && c_pos & 1) {
					new_pieces.running |= (1 << i);
					new_pieces.running_size++;
				}
			}
		}
	}
	// free
	uint16_t mergehandfree = new_pieces.hand | new_pieces.free;
	for (uint8_t p = 0; p < 2; ++p) {
		for (int8_t i = 0; i < 12 ; i += 2) {
			if ((new_pieces.stop >> (i | p )) & 1) {
				uint8_t c = (i | p) + 1;
				int8_t c_pos = iob(board, c);
				if (!((p ^ c_pos) & 1) && (mergehandfree >> c_pos) & 1) {
					new_pieces.free |= (1 << c) >> 1;
					new_pieces.free_size =
						(new_pieces.free_size & (0xf << (PCBI << 2))) |
						(((new_pieces.free_size >> (PCB << 2)) & 0xf) + 1)
						<< (PCB << 2);
					new_pieces.stop &= ~((1 << c) >> 1);
					mergehandfree = new_pieces.hand | new_pieces.free;
					i = -2;
				}
			}
		}
	}
	// stock
	for (uint8_t i = 0; i < 12 ; ++i) {
		if ((new_pieces.stop >> i) & 1) {
			uint8_t c = i + 1;
			int8_t c_pos = iob(board, c);
			if (!((run_pos_sum >> c_pos) & 1)) {
				new_pieces.stock |= (1 << c) >> 1;
				new_pieces.stop &= ~((1 << c) >> 1);
			}
		}
	}
	// dead
	for (int8_t i = 0; i < 12 ; ++i) {
		if ((new_pieces.stock >> i) & 1) {
			uint8_t c = i + 1;
			int8_t c_pos = iob(board, c);
			// if in other player, or in itself but dead
			if ((c_pos ^ i) & 1 || (new_pieces.dead >> c_pos) & 1) {
				new_pieces.dead |= (1 << c) >> 1;
				new_pieces.dead_size = (new_pieces.dead_size & (0xf << (CBI << 2)))
				| (((new_pieces.dead_size >> (CB << 2)) & 0xf) + 1) << (CB << 2);
				new_pieces.stock &= ~((1 << c) >> 1);
				i = -1;
			}
		}
	}
	// if multiple stock
	for (uint8_t i = 0; i < 12 ; ++i) {
		if ((new_pieces.stock >> i) & 1) {
			uint8_t c = i + 1;
			int8_t c_pos = iob(board, c);
			uint8_t ms = c_pos + 1;
			int8_t ms_pos = iob(board, ms);
			uint8_t ts = ms_pos + 1;
			int8_t ts_pos = iob(board, ts);
			if (
				(new_pieces.stock << 1 >> ms) & 1
				&& (
					ms_pos + 1 == c
					|| (
						(new_pieces.stock << 1 >> ts) & 1
						&& ts_pos + 1 == c
						)
					)
				) {
				new_pieces.dead |= (1 << c) >> 1;
				new_pieces.dead_size = (new_pieces.dead_size & (0xf << (CBI << 2)))
				| (((new_pieces.dead_size >> (CB << 2)) & 0xf) + 1) << (CB << 2);
			}
		}
	}
	new_pieces.stock &= ~new_pieces.dead;
	return new_pieces;
}

uint8_t posValue(uint64_t board, const Pieces &pieces_value)
{
	uint8_t player = (board >> 48) & 1;
	uint8_t my_stick = pieces_value.stick_size & 0xf;
	uint8_t your_stick = pieces_value.stick_size >> 4;
	uint8_t my_handle = (pieces_value.hand_size & 0xf)
		+ (pieces_value.free_size & 0xf);
	uint8_t your_handle = (pieces_value.hand_size >> 4)
		+ (pieces_value.free_size >> 4);
	uint8_t my_dead = pieces_value.dead_size & 0xf;
	uint8_t your_dead = pieces_value.dead_size >> 4;
	// two win
	const bool two_win = my_stick + my_handle == 6 &&
		 your_stick + your_handle == 6 &&
		 my_stick  == your_stick - player;
	// I win
	const bool i_win = (my_stick - player == 6 && your_stick < 6) ||
		(my_stick + my_handle == 6 &&
		(your_dead > 0 || my_stick - your_stick >= (~player & 1)));
	// I lose
	const bool i_lose = (my_stick + player < 6 && your_stick == 6) ||
		(your_stick + your_handle == 6 &&
		(my_dead > 0 || your_stick - my_stick >= (player + 1)));
	// two lose
	const bool two_lose = my_dead > 0 && your_dead > 0;
	if (two_win) {
		return 3;
	} else if (i_win) {
		return 4;
	} else if (i_lose) {
		return 1;
	} else if (two_lose) {
		return 2;
	}
	return 0;
}

unsigned int max_roll = 1.2e7;
unsigned int max_depth = 36;
unsigned int roll_sum = 0;
unsigned int result_sum = 0;
Position *roll(Position *pos, int8_t depth)
{
	roll_sum++;
	// pos value
	Pieces pieces_value = piecesValue(pos->board, depth);
	uint64_t pos_value = posValue(pos->board, pieces_value);
	pos->board |= pos_value << 60;
	setBoardMap(pos);
	pos->children.clear();
	// depth
	pos->board &= ~(0b1111111ll << 53);
	pos->board |= (uint64_t)depth << 53;
	// end if too much
	if (depth >= max_depth || roll_sum >= max_roll) {
		return pos;
	}
	// end if has a value
	if (pos_value > 0) {
		result_sum++;
	}
	// roll if value is 0
	else {
		uint8_t player = (pos->board >> 48) & 1;
		uint8_t lastmove = (pos->board >> 49) & 0xf;
		uint8_t hand_size = pieces_value.hand_size & 0xf;
		uint8_t running_size = pieces_value.running_size;
		// remove lastmove
		if ((pieces_value.running << 1 >> lastmove) & 1) {
			running_size--;
			pieces_value.running &= ~(1 << lastmove >> 1);
		}
		// remove 12 when player == 0
		if (!player && (pieces_value.running << 1 >> 12) & 1) {
			running_size--;
			pieces_value.running &= ~(1 << 12 >> 1);
		}
		// children size
		uint8_t children_size = hand_size + running_size;
		pos->children.resize(children_size);
		uint8_t x = 0;
		uint16_t if_roll = 0;
		bool if_win = false;
		// hand's children
		for (uint8_t i = 0; i < 12 && !if_win; i += 2) {
			uint64_t c = (i | player) + 1;
			if ((pieces_value.hand << 1 >> c) & 1) {
				Position *new_pos = new Position();
				new_pos->board = pos->board;
				uint8_t outc = (new_pos->board << 4 >> (c << 2)) & 0xf;
				// player
				uint64_t next_player = outc > 0 ?
					outc & 1 : ~player & 1;
				new_pos->board &= ~(1ll << 48);
				new_pos->board |= next_player << 48;
				// hand on stick
				new_pos->board &= ~(0xfll << (c << 2) >> 4);
				new_pos->board |= c << (c << 2) >> 4;
				// lastmove
				new_pos->board &= ~(0xfll << 49);
				new_pos->board |= c << 49;
				// push map
				Position *map_item = getBoardMap(new_pos);
				if (map_item == new_pos) {
					if_roll |= 1 << x;
					pos->children[x] = roll(new_pos, depth + 1);
				} else {
					delete new_pos;
					pos->children[x] = map_item;
				}
				// if win
				if ((pos->children[x]->board >> 60 == 4
					&& next_player == player)
					|| (pos->children[x]->board >> 60 == 1
					&& next_player != player)) {
					if_win = true;
				}
				x++;
			}
		}
		// running's children
		for (uint8_t i = 0; i < 12 && !if_win; ++i) {
			uint64_t c = i + 1;
			if ((pieces_value.running << 1 >> c) & 1) {
				Position *new_pos = new Position();
				new_pos->board = pos->board;
				// player
				uint64_t next_player = ~player & 1;
				new_pos->board &= ~(1ll << 48);
				new_pos->board |= next_player << 48;
				// run on position
				int8_t c_pos = iob(pos->board, c);
				int8_t c_newpos = pos24[c_pos + c];
				new_pos->board &= ~(0xfll << (c_pos << 2));
				if (12 != c) {
					new_pos->board &= ~(0xfll << (c_newpos << 2));
					new_pos->board |= c << (c_newpos << 2);
				}
				// lastmove
				new_pos->board &= ~(0xfll << 49);
				new_pos->board |= c << 49;
				// push map
				Position *map_item = getBoardMap(new_pos);
				if (map_item == new_pos) {
					if_roll |= 1 << x;
					pos->children[x] = roll(new_pos, depth + 1);
				} else {
					delete new_pos;
					pos->children[x] = map_item;
				}
				// if win
				if (pos->children[x]->board >> 60 == 1) {
					if_win = true;
				}
				x++;
			}
		}
		if (if_win) {
			pos->children.resize(x);
			pos->board &= ~(0xfll << 60);
			pos->board |= 4ll << 60;
			setBoardMap(pos);
		} else {
			if (children_size < 1) {
				if (0 == lastmove) {
					// two lose
					pos->board &= ~(0xfll << 60);
					pos->board |= 2ll << 60;
					return pos;
				} else {
					Position *new_pos = new Position();
					new_pos->board = pos->board;
					// player
					uint64_t next_player = ~player & 1;
					new_pos->board &= ~(1ll << 48);
					new_pos->board |= next_player << 48;
					// lastmove is 0
					new_pos->board &= ~(0xfll << 49);
					// roll
					if_roll = 1;
					pos->children.emplace_back(roll(new_pos, depth + 1));
				}
			}
			// value
			uint64_t max_value = pos->board >> 60;
			for (Position *child : pos->children) {
				int child_value = child->board >> 60;
				if ((child_value == 4 || child_value == 1) &&
					((child->board >> 48) & 1) != player) {
					child_value = child_value == 4 ? 1 : 4;
				}
				if (max_value < child_value) {
					max_value = child_value;
				}
			}
			pos->board &= ~(0xfll << 60);
			pos->board |= max_value << 60;
			setBoardMap(pos);
		}
		// max depth of this pos
		uint64_t pos_depth = depth;
		for (int i = 0; i < pos->children.size(); i++) {
			int child_depth = (pos->children[i]->board >> 53) & 0b1111111;
			if ((if_roll >> i) & 1 && pos_depth < child_depth) {
				pos_depth = child_depth;
			}
		}
		pos->board &= ~(0b1111111ll << 53);
		pos->board |= pos_depth << 53;
	}
	if (depth == 1) {
		cout << "current step: " << (int)((pos->board >> 49) & 0xf);
		cout << "  depth: " << (int)((pos->board >> 53) & 0x7f);
		cout << "  roll_sum: " << roll_sum;
		cout << endl;
	}
	return pos;
}

// example:
// 1,2,0,4,0,6,7,3,9,10,12,11;1;6|14000000,36,12
// 1,2,0,4,0,6,7,3,9,10,12,11;1|14000000,36,12
// 1,2,0,4,0,6,7,3,9,10,12,11|14000000,36,12
// 1,2,0,4,0,6,7,3,9,10,12,11;1;6
// 1,2,0,4,0,6,7,3,9,10,12,11;1
// 1,2,0,4,0,6,7,3,9,10,12,11
Position initBoard(string pos_start)
{
	Position *new_position = new Position();
	size_t pos_find, last_pos_find, substr_len;
	vector<size_t> pos_split;
	string pos_string;
	uint64_t player, last_move;
	pos_find = 0;
	while (pos_start.find(';', pos_find) != string::npos) {
		pos_find = pos_start.find(';', pos_find);
		pos_split.emplace_back(pos_find);
		pos_find++;
	}
	if (pos_start.find('|', pos_find) != string::npos) {
		size_t setting_pos = pos_start.find('|', pos_find);
		size_t setting_max_roll = pos_start.find(',', setting_pos + 1);
		size_t setting_max_depth = pos_start.find(',', setting_max_roll + 1);
		max_roll = stoi(pos_start.substr(setting_pos + 1, 
			setting_max_roll - setting_pos - 1));
		max_depth = stoi(pos_start.substr(setting_max_roll + 1, 
			setting_max_depth - setting_max_roll - 1));
		max_loop_depth = stoi(pos_start.substr(setting_max_depth + 1));
	}
	cout << "max_roll: " << max_roll << endl;
	cout << "max_depth: " << max_depth << endl;
	cout << "max_loop_depth: " << max_loop_depth << endl;
	cout << endl;
	if (pos_split.size() == 2) {
		pos_string = pos_start.substr(0, pos_split[0]);
		player = stoi(pos_start.substr(pos_split[0] + 1, 1));
		last_move = stoi(pos_start.substr(pos_split[1] + 1));
	} else if (pos_split.size() == 1) {
		pos_string = pos_start.substr(0, pos_split[0]);
		player = stoi(pos_start.substr(pos_split[0] + 1));
		last_move = 0;
	} else {
		pos_string = pos_start;
		player = 1;
		last_move = 0;
	}
	cout << "board: ";
	pos_find = 0;
	uint64_t c = 0;
	for (uint8_t i = 0; i < 12; i++) {
		if (pos_start.find(',', pos_find) == string::npos) {
			c = stoi(pos_start.substr(pos_find));
		} else {
			last_pos_find = pos_find;
			pos_find = pos_start.find(',', pos_find);
			substr_len = pos_find - last_pos_find;
			c = stoi(pos_start.substr(last_pos_find, substr_len));
		}
		new_position->board |= (c << (i << 2));
		cout << (int)c << ", ";
		pos_find++;
	}
	cout << endl;
	cout << "player: " << player << endl;
	new_position->board |= (player << 48);
	cout << "last_move: " << last_move << endl;
	new_position->board |= (last_move << 49);
	cout << endl;
	return *new_position;
}

int main()
{
	// read position
	cout << "input a position: ";
	string pos_start;
	cin >> pos_start;
	// start roll
	auto start_time = chrono::high_resolution_clock::now();
	Position pos = initBoard(pos_start);
	Position *result_pos = roll(&pos, 0);
	board_map.clear();
	// end roll
	auto end_time = chrono::high_resolution_clock::now();
	auto duration = chrono::duration_cast<chrono::milliseconds>(end_time - start_time);
	cout << endl << "roll it took " << duration.count() << " ms" << endl;
	// print data
	string pick_child;
	cout << "roll_sum:" << roll_sum << endl;
	cout << "result_sum:" << result_sum << endl;
	cout << endl;
	// start game
	stack<Position*> poslist;
	vector<uint8_t> movelist;
	int this_depth = 0;
	poslist.push(result_pos);
	do {
		coutMovelist(movelist);
		coutBoard(poslist.top()->board);
		cout << "this_depth: " << this_depth;
		cout << " ; available move:" << poslist.top()->children.size() << endl;
		for (size_t lm = 0; lm < poslist.top()->children.size(); lm++) {
			cout << "  " << lm << ": ";
			coutBoard(poslist.top()->children[lm]->board, "", false);
		}
		cout << endl << "select one option: ";
		cin >> pick_child;
		if (stoi(pick_child) >= 0
			&& stoi(pick_child) < poslist.top()->children.size()) {
			poslist.push(poslist.top()->children[stoi(pick_child)]);
			movelist.push_back((poslist.top()->board >> 49) & 0xf);
			this_depth++;
		} else if (pick_child == "-2") {
			poslist.pop();
			movelist.pop_back();
			this_depth--;
		} else if (pick_child == "-1") {
			while (this_depth > 0) {
				poslist.pop();
				movelist.pop_back();
				this_depth--;
			}
		} else if (pick_child == "-3") {
			cout << "Goodbye!" << endl;
		} else {
			cout << "Wrong choice, enter again." << endl;
		}
	} while (pick_child != "-3");
	return 0;
}
