/* * nextpnr -- Next Generation Place and Route * * Copyright (C) 2021 Symbiflow Authors * * * Permission to use, copy, modify, and/or distribute this software for any * purpose with or without fee is hereby granted, provided that the above * copyright notice and this permission notice appear in all copies. * * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. * */ #ifndef SAMPLER_H_ #define SAMPLER_H_ #include #include #include #include #include "nextpnr_namespaces.h" NEXTPNR_NAMESPACE_BEGIN // Given a set of coordinates, generates random samples that are geometric // distributed. struct Sampler { void divide_samples(size_t target_sample_count, const std::vector> &samples); size_t number_of_regions() const { return splits.size() - 1; } size_t get_sample_from_region(size_t region, std::function rng) const { if (region >= (splits.size() - 1)) { throw std::runtime_error("region out of range"); } size_t split_begin = splits[region]; size_t split_end = splits[region + 1]; if (split_begin == split_end) { throw std::runtime_error("Splits should never be empty!"); } // Pick a random element from that region. return indicies[split_begin + (rng() % (split_end - split_begin))]; } size_t get_sample(std::function rng) const { size_t region = rng() % number_of_regions(); return get_sample_from_region(region, rng); } std::vector indicies; std::vector splits; }; NEXTPNR_NAMESPACE_END #endif /* SAMPLER_H_ */