/* -*- mode: C++; c-basic-offset: 2; indent-tabs-mode: nil -*- */ /* * Main authors: * Mikael Lagerkvist * Christian Schulte * * Copyright: * Mikael Lagerkvist, 2007 * Christian Schulte, 2017 * * This file is part of Gecode, the generic constraint * development environment: * http://www.gecode.org * * Permission is hereby granted, free of charge, to any person obtaining * a copy of this software and associated documentation files (the * "Software"), to deal in the Software without restriction, including * without limitation the rights to use, copy, modify, merge, publish, * distribute, sublicense, and/or sell copies of the Software, and to * permit persons to whom the Software is furnished to do so, subject to * the following conditions: * * The above copyright notice and this permission notice shall be * included in all copies or substantial portions of the Software. * * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND * NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE * LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION * OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION * WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. * */ #include namespace Gecode { /* * Ranges * */ forceinline unsigned int TupleSet::Range::width(void) const { return static_cast(max - min + 1); } forceinline const TupleSet::BitSetData* TupleSet::Range::supports(unsigned int n_words, int n) const { assert((min <= n) && (n <= max)); return s + n_words * static_cast(n - min); } /* * Tuple set data * */ forceinline TupleSet::Data::Data(int a) : arity(a), n_words(0U), // To be initialized in finalize n_tuples(0), n_free(n_initial_free), min(Int::Limits::max), max(Int::Limits::min), key(0), td(heap.alloc(n_initial_free * a)), vd(heap.alloc(a)), range(nullptr), support(nullptr) { } forceinline bool TupleSet::Data::finalized(void) const { return n_free < 0; } forceinline TupleSet::Tuple TupleSet::Data::add(void) { if (n_free == 0) resize(); assert(n_free > 0); n_free--; Tuple t = td + n_tuples*arity; n_tuples++; return t; } forceinline TupleSet::Tuple TupleSet::Data::get(int i) const { assert((i >= 0) && (i < n_tuples)); return td + i*arity; } forceinline unsigned int TupleSet::ValueData::start(int k) const { if (n > 1U) { unsigned int l=0U, h=n-1U; while (true) { assert(l<=h); unsigned int m = l + ((h-l) >> 1); if (k < r[m].min) h=m-1U; else if (k > r[m].max) l=m+1U; else return m; } GECODE_NEVER; } else { return 0U; } } forceinline void TupleSet::Data::set(BitSetData* d, unsigned int i) { d[i / BitSetData::bpb].set(i % BitSetData::bpb); } forceinline bool TupleSet::Data::get(const BitSetData* d, unsigned int i) { return d[i / BitSetData::bpb].get(i % BitSetData::bpb); } forceinline unsigned int TupleSet::Data::tuple2idx(Tuple t) const { return static_cast((t - td) / static_cast(arity)); } forceinline const TupleSet::Range* TupleSet::Data::fst(int i) const { return &vd[i].r[0]; } forceinline const TupleSet::Range* TupleSet::Data::lst(int i) const { return &vd[i].r[vd[i].n-1U]; } /* * Tuple set * */ forceinline TupleSet& TupleSet::add(const IntArgs& t) { _add(t); return *this; } forceinline TupleSet::TupleSet(void) {} forceinline TupleSet::operator bool(void) const { return object() != nullptr; } forceinline void TupleSet::finalize(void) { Data* d = static_cast(object()); if (!d->finalized()) d->finalize(); } forceinline bool TupleSet::finalized(void) const { return static_cast(object())->finalized(); } forceinline TupleSet::Data& TupleSet::data(void) const { assert(finalized()); return *static_cast(object()); } forceinline TupleSet::Data& TupleSet::raw(void) const { return *static_cast(object()); } forceinline bool TupleSet::operator !=(const TupleSet& t) const { return !(*this == t); } forceinline int TupleSet::arity(void) const { return raw().arity; } forceinline int TupleSet::tuples(void) const { return raw().n_tuples; } forceinline unsigned int TupleSet::words(void) const { return data().n_words; } forceinline int TupleSet::min(void) const { return data().min; } forceinline int TupleSet::max(void) const { return data().max; } forceinline TupleSet::Tuple TupleSet::operator [](int i) const { return data().get(i); } forceinline const TupleSet::Range* TupleSet::fst(int i) const { return data().fst(i); } forceinline const TupleSet::Range* TupleSet::lst(int i) const { return data().lst(i); } forceinline bool TupleSet::operator ==(const TupleSet& t) const { if (tuples() != t.tuples()) return false; if (arity() != t.arity()) return false; if (min() != t.min()) return false; if (max() != t.max()) return false; return equal(t); } forceinline std::size_t TupleSet::hash(void) const { return data().key; } template std::basic_ostream& operator <<(std::basic_ostream& os, const TupleSet& ts) { std::basic_ostringstream s; s.copyfmt(os); s.width(0); s << "Number of tuples: " << ts.tuples() << " (number of words: " << ts.words() << " with " << Support::BitSetData::bpb << " bits)" << std::endl; for (int a=0; a < ts.arity(); a++) { unsigned int size = 0U; for (const TupleSet::Range* c=ts.fst(a); c<=ts.lst(a); c++) size += c->width(); s << "\t[" << a << "] size: " << size << ", width: " << static_cast(ts.lst(a)->max - ts.fst(a)->min + 1) << ", ranges: " << (ts.lst(a) - ts.fst(a) + 1U) << std::endl; } return os << s.str(); } /* * Range iterator * */ forceinline TupleSet::Ranges::Ranges(const TupleSet& ts, int i) { c = &(ts.data().vd[i].r[0]); l = c + ts.data().vd[i].n; } forceinline bool TupleSet::Ranges::operator ()(void) const { return cmin; } forceinline int TupleSet::Ranges::max(void) const { return c->max; } forceinline unsigned int TupleSet::Ranges::width(void) const { return c->width(); } } // STATISTICS: int-prop