X-Git-Url: https://git.sesse.net/?p=stockfish;a=blobdiff_plain;f=src%2Fsearch.cpp;h=d385e3eb7c4a07a8094465245e7286328d46c7af;hp=b7c86e42e3f5ade0ba44a6db1f1bf0ce4100a2d3;hb=95212222c7444538b84326208e433ac12f15e9fb;hpb=2786aed19554dee742013d438742adae03a3ef8c diff --git a/src/search.cpp b/src/search.cpp index b7c86e42..d385e3eb 100644 --- a/src/search.cpp +++ b/src/search.cpp @@ -129,7 +129,7 @@ namespace { void extract_pv_from_tt(Position& pos); void insert_pv_in_tt(Position& pos); - std::string pv_info_to_uci(Position& pos, int depth, Value alpha, Value beta, int pvLine = 0); + std::string pv_info_to_uci(Position& pos, int depth, Value alpha, Value beta, int pvLine); int64_t nodes; Value pv_score; @@ -306,7 +306,7 @@ namespace { int current_search_time(); std::string value_to_uci(Value v); - int nps(const Position& pos); + std::string speed_to_uci(int64_t nodes); void poll(const Position& pos); void wait_for_stop_or_ponderhit(); @@ -544,12 +544,13 @@ bool think(Position& pos, bool infinite, bool ponder, int time[], int increment[ std::string name = Options["Search Log Filename"].value(); LogFile.open(name.c_str(), std::ios::out | std::ios::app); - LogFile << "Searching: " << pos.to_fen() - << "\ninfinite: " << infinite - << " ponder: " << ponder - << " time: " << myTime - << " increment: " << myIncrement - << " moves to go: " << movesToGo << endl; + LogFile << "\nSearching: " << pos.to_fen() + << "\ninfinite: " << infinite + << " ponder: " << ponder + << " time: " << myTime + << " increment: " << myIncrement + << " moves to go: " << movesToGo + << endl; } // We're ready to start thinking. Call the iterative deepening loop function @@ -557,25 +558,20 @@ bool think(Position& pos, bool infinite, bool ponder, int time[], int increment[ Move bestMove = id_loop(pos, searchMoves, &ponderMove); // Print final search statistics - cout << "info nodes " << pos.nodes_searched() - << " nps " << nps(pos) - << " time " << current_search_time() << endl; + cout << "info" << speed_to_uci(pos.nodes_searched()) << endl; if (UseLogFile) { - LogFile << "\nNodes: " << pos.nodes_searched() - << "\nNodes/second: " << nps(pos) - << "\nBest move: " << move_to_san(pos, bestMove); + int t = current_search_time(); + + LogFile << "Nodes: " << pos.nodes_searched() + << "\nNodes/second: " << (t > 0 ? int(pos.nodes_searched() * 1000 / t) : 0) + << "\nBest move: " << move_to_san(pos, bestMove); StateInfo st; pos.do_move(bestMove, st); - LogFile << "\nPonder move: " - << move_to_san(pos, ponderMove) // Works also with MOVE_NONE - << endl; - - // Return from think() with unchanged position - pos.undo_move(bestMove); - + LogFile << "\nPonder move: " << move_to_san(pos, ponderMove) << endl; + pos.undo_move(bestMove); // Return from think() with unchanged position LogFile.close(); } @@ -615,7 +611,7 @@ namespace { // Initialize FIXME move before Rml.init() TT.new_search(); H.clear(); - memset(ss, 0, PLY_MAX_PLUS_2 * sizeof(SearchStack)); + memset(ss, 0, 4 * sizeof(SearchStack)); *ponderMove = bestMove = easyMove = MOVE_NONE; depth = aspirationDelta = 0; ss->currentMove = MOVE_NULL; // Hack to skip update_gains() @@ -701,6 +697,9 @@ namespace { bestValues[depth] = value; bestMoveChanges[depth] = Rml.bestMoveChanges; + if (UseLogFile) + LogFile << pretty_pv(pos, depth, value, current_search_time(), Rml[0].pv) << endl; + // Drop the easy move if differs from the new best move if (bestMove != easyMove) easyMove = MOVE_NONE; @@ -802,7 +801,8 @@ namespace { bestValue = alpha; // Step 1. Initialize node and poll. Polling can abort search - ss->currentMove = ss->bestMove = threatMove = MOVE_NONE; + ss->currentMove = ss->bestMove = threatMove = (ss+1)->excludedMove = MOVE_NONE; + (ss+1)->skipNullMove = false; (ss+1)->reduction = DEPTH_ZERO; (ss+2)->killers[0] = (ss+2)->killers[1] = (ss+2)->mateKiller = MOVE_NONE; if (threadID == 0 && ++NodesSincePoll > NodesBetweenPolls) @@ -1028,9 +1028,7 @@ split_point_start: // At split points actual search starts from here if (SendSearchedNodes) { SendSearchedNodes = false; - cout << "info nodes " << nodes - << " nps " << nps(pos) - << " time " << current_search_time() << endl; + cout << "info" << speed_to_uci(pos.nodes_searched()) << endl; } if (current_search_time() >= 1000) @@ -1071,7 +1069,7 @@ split_point_start: // At split points actual search starts from here // Update current move (this must be done after singular extension search) ss->currentMove = move; - newDepth = depth - (!Root ? ONE_PLY : DEPTH_ZERO) + ext; + newDepth = depth - ONE_PLY + ext; // Step 12. Futility pruning (is omitted in PV nodes) if ( !PvNode @@ -1154,8 +1152,7 @@ split_point_start: // At split points actual search starts from here && ss->killers[0] != move && ss->killers[1] != move) { - ss->reduction = Root ? reduction(depth, moveCount - MultiPV + 1) - : reduction(depth, moveCount); + ss->reduction = reduction(depth, moveCount); if (ss->reduction) { alpha = SpNode ? sp->alpha : alpha; @@ -1194,14 +1191,14 @@ split_point_start: // At split points actual search starts from here alpha = sp->alpha; } - if (!Root && value > bestValue && !(SpNode && ThreadsMgr.cutoff_at_splitpoint(threadID))) + if (value > bestValue && !(SpNode && ThreadsMgr.cutoff_at_splitpoint(threadID))) { bestValue = value; if (SpNode) sp->bestValue = value; - if (value > alpha) + if (!Root && value > alpha) { if (PvNode && value < beta) // We want always alpha < beta { @@ -1219,16 +1216,12 @@ split_point_start: // At split points actual search starts from here ss->bestMove = move; if (SpNode) - sp->parentSstack->bestMove = move; + sp->ss->bestMove = move; } } if (Root) { - // To avoid to exit with bestValue == -VALUE_INFINITE - if (value > bestValue) - bestValue = value; - // Finished searching the move. If StopRequest is true, the search // was aborted because the user interrupted the search or because we // ran out of time. In this case, the return value of the search cannot @@ -1240,13 +1233,9 @@ split_point_start: // At split points actual search starts from here // Remember searched nodes counts for this move mp.rm->nodes += pos.nodes_searched() - nodes; - // Step 17. Check for new best move - if (!isPvMove && value <= alpha) - mp.rm->pv_score = -VALUE_INFINITE; - else + // PV move or new best move ? + if (isPvMove || value > alpha) { - // PV move or new best move! - // Update PV ss->bestMove = move; mp.rm->pv_score = value; @@ -1266,9 +1255,11 @@ split_point_start: // At split points actual search starts from here alpha = Rml[Min(moveCount, MultiPV) - 1].pv_score; // FIXME why moveCount? else if (value > alpha) alpha = value; + } + else + mp.rm->pv_score = -VALUE_INFINITE; - } // PV move or new best move - } + } // Root // Step 18. Check for split if ( !Root @@ -1865,6 +1856,14 @@ split_point_start: // At split points actual search starts from here H.update_gain(pos.piece_on(move_to(m)), move_to(m), -(before + after)); } + // current_search_time() returns the number of milliseconds which have passed + // since the beginning of the current search. + + int current_search_time() { + + return get_system_time() - SearchStartTime; + } + // value_to_uci() converts a value to a string suitable for use with the UCI // protocol specifications: @@ -1880,27 +1879,25 @@ split_point_start: // At split points actual search starts from here if (abs(v) < VALUE_MATE - PLY_MAX * ONE_PLY) s << "cp " << int(v) * 100 / int(PawnValueMidgame); // Scale to centipawns else - s << "mate " << (v > 0 ? (VALUE_MATE - v + 1) / 2 : -(VALUE_MATE + v) / 2 ); + s << "mate " << (v > 0 ? (VALUE_MATE - v + 1) / 2 : -(VALUE_MATE + v) / 2); return s.str(); } - // current_search_time() returns the number of milliseconds which have passed - // since the beginning of the current search. - - int current_search_time() { - - return get_system_time() - SearchStartTime; - } + // speed_to_uci() returns a string with time stats of current search suitable + // to be sent to UCI gui. + std::string speed_to_uci(int64_t nodes) { - // nps() computes the current nodes/second count + std::stringstream s; + int t = current_search_time(); - int nps(const Position& pos) { + s << " nodes " << nodes + << " nps " << (t > 0 ? int(nodes * 1000 / t) : 0) + << " time " << t; - int t = current_search_time(); - return (t > 0 ? int((pos.nodes_searched() * 1000) / t) : 0); + return s.str(); } @@ -2117,16 +2114,19 @@ split_point_start: // At split points actual search starts from here threads[threadID].state = THREAD_SEARCHING; - // Here we call search() with SplitPoint template parameter set to true + // Copy SplitPoint position and search stack and call search() + // with SplitPoint template parameter set to true. + SearchStack ss[PLY_MAX_PLUS_2]; SplitPoint* tsp = threads[threadID].splitPoint; Position pos(*tsp->pos, threadID); - SearchStack* ss = tsp->sstack[threadID] + 1; - ss->sp = tsp; + + memcpy(ss, tsp->ss - 1, 4 * sizeof(SearchStack)); + (ss+1)->sp = tsp; if (tsp->pvNode) - search(pos, ss, tsp->alpha, tsp->beta, tsp->depth, tsp->ply); + search(pos, ss+1, tsp->alpha, tsp->beta, tsp->depth, tsp->ply); else - search(pos, ss, tsp->alpha, tsp->beta, tsp->depth, tsp->ply); + search(pos, ss+1, tsp->alpha, tsp->beta, tsp->depth, tsp->ply); assert(threads[threadID].state == THREAD_SEARCHING); @@ -2371,7 +2371,7 @@ split_point_start: // At split points actual search starts from here splitPoint.moveCount = moveCount; splitPoint.pos = &pos; splitPoint.nodes = 0; - splitPoint.parentSstack = ss; + splitPoint.ss = ss; for (i = 0; i < activeThreads; i++) splitPoint.slaves[i] = 0; @@ -2398,12 +2398,10 @@ split_point_start: // At split points actual search starts from here lock_release(&mpLock); // Tell the threads that they have work to do. This will make them leave - // their idle loop. But before copy search stack tail for each thread. + // their idle loop. for (i = 0; i < activeThreads; i++) if (i == master || splitPoint.slaves[i]) { - memcpy(splitPoint.sstack[i], ss - 1, 4 * sizeof(SearchStack)); - assert(i == master || threads[i].state == THREAD_BOOKED); threads[i].state = THREAD_WORKISWAITING; // This makes the slave to exit from idle_loop() @@ -2528,8 +2526,8 @@ split_point_start: // At split points actual search starts from here } // pv_info_to_uci() returns a string with information on the current PV line - // formatted according to UCI specification and eventually writes the info - // to a log file. It is called at each iteration or after a new pv is found. + // formatted according to UCI specification. It is called at each iteration + // or after a new pv is found. std::string RootMove::pv_info_to_uci(Position& pos, int depth, Value alpha, Value beta, int pvLine) { @@ -2544,18 +2542,9 @@ split_point_start: // At split points actual search starts from here << " multipv " << pvLine + 1 << " score " << value_to_uci(pv_score) << (pv_score >= beta ? " lowerbound" : pv_score <= alpha ? " upperbound" : "") - << " time " << current_search_time() - << " nodes " << pos.nodes_searched() - << " nps " << nps(pos) + << speed_to_uci(pos.nodes_searched()) << " pv " << l.str(); - if (UseLogFile && pvLine == 0) - { - ValueType t = pv_score >= beta ? VALUE_TYPE_LOWER : - pv_score <= alpha ? VALUE_TYPE_UPPER : VALUE_TYPE_EXACT; - - LogFile << pretty_pv(pos, current_search_time(), depth, pv_score, t, pv) << endl; - } return s.str(); }