Rev 169 | Show entire file | Ignore whitespace | Details | Blame | Last modification | View Log | RSS feed
Rev 169 | Rev 185 | ||
---|---|---|---|
Line 4... | Line 4... | ||
4 | Copyright (C) 2008-2015 Marco Costalba, Joona Kiiski, Tord Romstad |
4 | Copyright (C) 2008-2015 Marco Costalba, Joona Kiiski, Tord Romstad |
5 | Copyright (C) 2015- |
5 | Copyright (C) 2015-2019 Marco Costalba, Joona Kiiski, Gary Linscott, Tord Romstad |
6 | 6 | ||
7 | Stockfish is free software: you can redistribute it and/or modify |
7 | Stockfish is free software: you can redistribute it and/or modify |
8 | it under the terms of the GNU General Public License as published by |
8 | it under the terms of the GNU General Public License as published by |
9 | the Free Software Foundation, either version 3 of the License, or |
9 | the Free Software Foundation, either version 3 of the License, or |
10 | (at your option) any later version. |
10 | (at your option) any later version. |
Line 18... | Line 18... | ||
18 | along with this program. If not, see <http://www.gnu.org/licenses/>. |
18 | along with this program. If not, see <http://www.gnu.org/licenses/>. |
19 | */ |
19 | */ |
20 | 20 | ||
21 | #include <cstring> // For std::memset |
21 | #include <cstring> // For std::memset |
22 | #include <iostream> |
22 | #include <iostream> |
- | 23 | #include <thread> |
|
23 | 24 | ||
24 | #include "bitboard.h" |
25 | #include "bitboard.h" |
- | 26 | #include "misc.h" |
|
25 | #include "tt.h" |
27 | #include "tt.h" |
- | 28 | #include "uci.h" |
|
26 | 29 | ||
27 | TranspositionTable TT; // Our global transposition table |
30 | TranspositionTable TT; // Our global transposition table |
- | 31 | ||
- | 32 | /// TTEntry::save saves a TTEntry |
|
- | 33 | void TTEntry::save(Key k, Value v, Bound b, Depth d, Move m, Value ev) { |
|
- | 34 | ||
- | 35 | assert(d / ONE_PLY * ONE_PLY == d); |
|
- | 36 | ||
- | 37 | // Preserve any existing move for the same position |
|
- | 38 | if (m || (k >> 48) != key16) |
|
- | 39 | move16 = (uint16_t)m; |
|
- | 40 | ||
- | 41 | // Overwrite less valuable entries |
|
- | 42 | if ( (k >> 48) != key16 |
|
- | 43 | || d / ONE_PLY > depth8 - 4 |
|
- | 44 | || b == BOUND_EXACT) |
|
- | 45 | { |
|
- | 46 | key16 = (uint16_t)(k >> 48); |
|
- | 47 | value16 = (int16_t)v; |
|
- | 48 | eval16 = (int16_t)ev; |
|
- | 49 | genBound8 = (uint8_t)(TT.generation8 | b); |
|
- | 50 | depth8 = (int8_t)(d / ONE_PLY); |
|
- | 51 | } |
|
- | 52 | } |
|
28 | 53 | ||
29 | 54 | ||
30 | /// TranspositionTable::resize() sets the size of the transposition table, |
55 | /// TranspositionTable::resize() sets the size of the transposition table, |
31 | /// measured in megabytes. Transposition table consists of a power of 2 number |
56 | /// measured in megabytes. Transposition table consists of a power of 2 number |
32 | /// of clusters and each cluster consists of ClusterSize number of TTEntry. |
57 | /// of clusters and each cluster consists of ClusterSize number of TTEntry. |
33 | 58 | ||
34 | void TranspositionTable::resize(size_t mbSize) { |
59 | void TranspositionTable::resize(size_t mbSize) { |
35 | 60 | ||
36 |
|
61 | clusterCount = mbSize * 1024 * 1024 / sizeof(Cluster); |
37 | - | ||
38 | if (newClusterCount == clusterCount) |
- | |
39 | return; |
- | |
40 | - | ||
41 | clusterCount = newClusterCount; |
- | |
42 | 62 | ||
43 | free(mem); |
63 | free(mem); |
44 | mem = malloc(clusterCount * sizeof(Cluster) + CacheLineSize - 1); |
64 | mem = malloc(clusterCount * sizeof(Cluster) + CacheLineSize - 1); |
45 | 65 | ||
46 | if (!mem) |
66 | if (!mem) |
Line 53... | Line 73... | ||
53 | table = (Cluster*)((uintptr_t(mem) + CacheLineSize - 1) & ~(CacheLineSize - 1)); |
73 | table = (Cluster*)((uintptr_t(mem) + CacheLineSize - 1) & ~(CacheLineSize - 1)); |
54 | clear(); |
74 | clear(); |
55 | } |
75 | } |
56 | 76 | ||
57 | 77 | ||
58 | /// TranspositionTable::clear() |
78 | /// TranspositionTable::clear() initializes the entire transposition table to zero, |
59 |
|
79 | // in a multi-threaded way. |
60 | /// user asks the program to clear the table (from the UCI interface). |
- | |
61 | 80 | ||
62 | void TranspositionTable::clear() { |
81 | void TranspositionTable::clear() { |
63 | 82 | ||
64 | std:: |
83 | std::vector<std::thread> threads; |
65 | } |
- | |
66 | 84 | ||
- | 85 | for (size_t idx = 0; idx < Options["Threads"]; idx++) |
|
- | 86 | { |
|
- | 87 | threads.emplace_back([this, idx]() { |
|
- | 88 | ||
- | 89 | // Thread binding gives faster search on systems with a first-touch policy |
|
- | 90 | if (Options["Threads"] > 8) |
|
- | 91 | WinProcGroup::bindThisThread(idx); |
|
- | 92 | ||
- | 93 | // Each thread will zero its part of the hash table |
|
- | 94 | const size_t stride = clusterCount / Options["Threads"], |
|
- | 95 | start = stride * idx, |
|
- | 96 | len = idx != Options["Threads"] - 1 ? |
|
- | 97 | stride : clusterCount - start; |
|
- | 98 | ||
- | 99 | std::memset(&table[start], 0, len * sizeof(Cluster)); |
|
- | 100 | }); |
|
- | 101 | } |
|
- | 102 | ||
- | 103 | for (std::thread& th: threads) |
|
- | 104 | th.join(); |
|
- | 105 | } |
|
67 | 106 | ||
68 | /// TranspositionTable::probe() looks up the current position in the transposition |
107 | /// TranspositionTable::probe() looks up the current position in the transposition |
69 | /// table. It returns true and a pointer to the TTEntry if the position is found. |
108 | /// table. It returns true and a pointer to the TTEntry if the position is found. |
70 | /// Otherwise, it returns false and a pointer to an empty or least valuable TTEntry |
109 | /// Otherwise, it returns false and a pointer to an empty or least valuable TTEntry |
71 | /// to be replaced later. The replace value of an entry is calculated as its depth |
110 | /// to be replaced later. The replace value of an entry is calculated as its depth |
Line 78... | Line 117... | ||
78 | const uint16_t key16 = key >> 48; // Use the high 16 bits as key inside the cluster |
117 | const uint16_t key16 = key >> 48; // Use the high 16 bits as key inside the cluster |
79 | 118 | ||
80 | for (int i = 0; i < ClusterSize; ++i) |
119 | for (int i = 0; i < ClusterSize; ++i) |
81 | if (!tte[i].key16 || tte[i].key16 == key16) |
120 | if (!tte[i].key16 || tte[i].key16 == key16) |
82 | { |
121 | { |
83 | if ((tte[i].genBound8 & 0xFC) != generation8 && tte[i].key16) |
- | |
84 |
|
122 | tte[i].genBound8 = uint8_t(generation8 | tte[i].bound()); // Refresh |
85 | 123 | ||
86 | return found = (bool)tte[i].key16, &tte[i]; |
124 | return found = (bool)tte[i].key16, &tte[i]; |
87 | } |
125 | } |
88 | 126 | ||
89 | // Find an entry to be replaced according to the replacement strategy |
127 | // Find an entry to be replaced according to the replacement strategy |