154 lines
3.4 KiB
C
154 lines
3.4 KiB
C
/*
|
|
* CDE - Common Desktop Environment
|
|
*
|
|
* Copyright (c) 1993-2012, The Open Group. All rights reserved.
|
|
*
|
|
* These libraries and programs are free software; you can
|
|
* redistribute them and/or modify them under the terms of the GNU
|
|
* Lesser General Public License as published by the Free Software
|
|
* Foundation; either version 2 of the License, or (at your option)
|
|
* any later version.
|
|
*
|
|
* These libraries and programs are distributed in the hope that
|
|
* they will be useful, but WITHOUT ANY WARRANTY; without even the
|
|
* implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR
|
|
* PURPOSE. See the GNU Lesser General Public License for more
|
|
* details.
|
|
*
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
* License along with these librararies and programs; if not, write
|
|
* to the Free Software Foundation, Inc., 51 Franklin Street, Fifth
|
|
* Floor, Boston, MA 02110-1301 USA
|
|
*/
|
|
/* $XConsortium: ISet.C /main/1 1996/07/29 16:54:10 cde-hp $ */
|
|
// Copyright (c) 1994 James Clark
|
|
// See the file COPYING for copying permission.
|
|
|
|
#ifndef ISet_DEF_INCLUDED
|
|
#define ISet_DEF_INCLUDED 1
|
|
|
|
#include <stdlib.h>
|
|
|
|
#ifdef SP_NAMESPACE
|
|
namespace SP_NAMESPACE {
|
|
#endif
|
|
|
|
template<class T>
|
|
ISet<T>::ISet()
|
|
{
|
|
}
|
|
|
|
template<class T>
|
|
ISet<T>::~ISet()
|
|
{
|
|
}
|
|
|
|
template<class T>
|
|
ISet<T>::ISet(const T *v, size_t n)
|
|
{
|
|
for (size_t i = 0; i < n; i++)
|
|
add(v[i]);
|
|
}
|
|
|
|
template<class T>
|
|
Boolean ISet<T>::contains(T x) const
|
|
{
|
|
for (size_t i = 0; i < r_.size(); i++)
|
|
if (r_[i].max >= x)
|
|
return r_[i].min <= x ? 1 : 0;
|
|
return 0;
|
|
}
|
|
|
|
template<class T>
|
|
void ISet<T>::addRange(T min, T max)
|
|
{
|
|
size_t i;
|
|
if (min == 0)
|
|
i = 0;
|
|
else {
|
|
for (i = r_.size(); i > 0 && min - 1 <= r_[i - 1].max; i--)
|
|
;
|
|
}
|
|
// r_[i - 1].max < min - 1 <= r_[i].max
|
|
if (i < r_.size() && (r_[i].min == 0 || max >= r_[i].min - 1)) {
|
|
// we can coelesce
|
|
if (min < r_[i].min)
|
|
r_[i].min = min;
|
|
if (max > r_[i].max) {
|
|
r_[i].max = max;
|
|
size_t j;
|
|
for (j = i + 1; j < r_.size() && r_[i].max >= r_[j].min - 1; j++)
|
|
r_[i].max = r_[j].max;
|
|
// get rid of i + 1 ... j - 1
|
|
if (j > i + 1) {
|
|
for (size_t k = j; k < r_.size(); k++)
|
|
r_[k - (j - i - 1)] = r_[k];
|
|
r_.resize(r_.size() - (j - i - 1));
|
|
}
|
|
}
|
|
}
|
|
else {
|
|
// r_[i - 1].max < min - 1
|
|
// max + 1 < r_[i].min
|
|
r_.resize(r_.size() + 1);
|
|
for (size_t j = r_.size() - 1; j > i; j--)
|
|
r_[j] = r_[j - 1];
|
|
r_[i].max = max;
|
|
r_[i].min = min;
|
|
}
|
|
}
|
|
|
|
template<class T>
|
|
void ISet<T>::remove(T c)
|
|
{
|
|
for (size_t i = 0; i < r_.size(); i++)
|
|
if (r_[i].max >= c) {
|
|
if (r_[i].min <= c) {
|
|
if (r_[i].min == r_[i].max) {
|
|
while (++i < r_.size())
|
|
r_[i - 1] = r_[i];
|
|
r_.resize(r_.size() - 1);
|
|
}
|
|
else if (c == r_[i].min)
|
|
r_[i].min += 1;
|
|
else if (c == r_[i].max)
|
|
r_[i].max -= 1;
|
|
else {
|
|
r_.resize(r_.size() + 1);
|
|
// split the range
|
|
// subtracting 2 is safe since we know that the length is >= 2
|
|
for (size_t j = r_.size() - 2; j > i; j--)
|
|
r_[j + 1] = r_[j];
|
|
r_[i + 1].max = r_[i].max;
|
|
r_[i + 1].min = c + 1;
|
|
r_[i].max = c - 1;
|
|
}
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
|
|
template<class T>
|
|
void ISet<T>::check()
|
|
{
|
|
for (size_t i = 0; i < r_.size(); i++) {
|
|
if (r_[i].min > r_[i].max)
|
|
abort();
|
|
// adjacent ranges must be coalesced
|
|
if (i > 0 && r_[i].min - 1 <= r_[i - 1].max)
|
|
abort();
|
|
}
|
|
}
|
|
|
|
template<class T>
|
|
void ISet<T>::clear()
|
|
{
|
|
r_.resize(0);
|
|
}
|
|
|
|
#ifdef SP_NAMESPACE
|
|
}
|
|
#endif
|
|
|
|
#endif /* not ISet_DEF_INCLUDED */
|