mirror of https://github.com/dnomd343/klotski.git
Dnomd343
2 years ago
5 changed files with 56 additions and 33 deletions
@ -1,47 +1,51 @@ |
|||||
#include "common.h" |
#include "common.h" |
||||
#include "all_cases.h" |
#include "all_cases.h" |
||||
|
|
||||
std::mutex AllCases::all_cases_building; |
std::mutex AllCases::building; |
||||
bool AllCases::all_cases_available = false; |
bool AllCases::available = false; |
||||
std::vector<uint32_t> AllCases::all_cases[]; |
std::vector<uint32_t> AllCases::data[]; |
||||
|
|
||||
AllCases::Status AllCases::status() { // get all cases status
|
AllCases::Status AllCases::status() { // get all cases status
|
||||
if (all_cases_available) { |
if (AllCases::available) { |
||||
return AVAILABLE; // all cases already built
|
return AVAILABLE; // all cases already built
|
||||
} |
} |
||||
if (!all_cases_building.try_lock()) { // fail to lock mutex -> another thread working
|
if (!AllCases::building.try_lock()) { // fail to lock mutex
|
||||
return BUILDING; |
return BUILDING; // another thread working
|
||||
} |
} |
||||
all_cases_building.unlock(); // release mutex
|
AllCases::building.unlock(); // release mutex
|
||||
return NO_INIT; |
return NO_INIT; |
||||
} |
} |
||||
|
|
||||
const std::vector<uint32_t> (*AllCases::fetch())[16] { // get const ptr of all cases
|
const std::vector<uint32_t> (*AllCases::fetch())[16] { // get all cases content
|
||||
if (status() != AllCases::AVAILABLE) { |
if (status() != AVAILABLE) { |
||||
AllCases::build(); // all cases initialize
|
AllCases::build(); // all cases initialize
|
||||
} |
} |
||||
return &all_cases; // return ptr
|
return &AllCases::data; // return const ptr
|
||||
} |
} |
||||
|
|
||||
void AllCases::build() { // build all cases
|
void AllCases::build() { // ensure that all cases available
|
||||
if (AllCases::all_cases_available) { |
if (!AllCases::available) { |
||||
return; // all cases already built
|
if (AllCases::building.try_lock()) { // mutex lock success
|
||||
|
build_data(); // start build process
|
||||
|
AllCases::available = true; // set available flag
|
||||
|
} else { |
||||
|
AllCases::building.lock(); // blocking waiting
|
||||
|
} |
||||
|
AllCases::building.unlock(); |
||||
} |
} |
||||
if (AllCases::all_cases_building.try_lock()) { // lock success -> start building
|
} |
||||
|
|
||||
|
void AllCases::build_data() { // find all cases
|
||||
|
for (uint32_t head = 0; head < 16; ++head) { // address of 2x2 block
|
||||
|
if ((head & 0b11) == 0b11) { |
||||
|
continue; // invalid 2x2 address
|
||||
|
} |
||||
/// head -> 0/1/2 / 4/5/6 / 8/9/10 / 12/13/14
|
/// head -> 0/1/2 / 4/5/6 / 8/9/10 / 12/13/14
|
||||
for (uint32_t head = 0; head < 16; ++head) { // address of 2x2 block
|
data[head].reserve(ALL_CASES_SIZE[head]); // memory pre-allocated
|
||||
if ((head & 0b11) == 0b11) { |
for (auto const &range : *BasicRanges::fetch()) { // check base on 2x2 address and range
|
||||
continue; // invalid 2x2 address
|
if (Common::check_case(head, range)) { |
||||
} |
data[head].emplace_back(Common::range_reverse(range)); // found valid case
|
||||
for (uint32_t const &range : *BasicRanges::fetch()) { // check base on 2x2 address and range
|
|
||||
if (Common::check_case(head, range)) { |
|
||||
all_cases[head].emplace_back(Common::range_reverse(range)); // found valid case
|
|
||||
} |
|
||||
} |
} |
||||
} |
} |
||||
AllCases::all_cases_available = true; // set available flag
|
|
||||
} else { // another thread building
|
|
||||
AllCases::all_cases_building.lock(); // blocking waiting
|
|
||||
} |
} |
||||
AllCases::all_cases_building.unlock(); |
|
||||
} |
} |
||||
|
Loading…
Reference in new issue