123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244 |
- // -*- Mode: C++; tab-width: 4; c-basic-offset: 4; indent-tabs-mode: nil -*-
- // vim:tabstop=4:shiftwidth=4:expandtab:
- /*
- * Copyright (C) 2004-2008 Wu Yongwei <adah at users dot sourceforge dot net>
- *
- * This software is provided 'as-is', without any express or implied
- * warranty. In no event will the authors be held liable for any
- * damages arising from the use of this software.
- *
- * Permission is granted to anyone to use this software for any purpose,
- * including commercial applications, and to alter it and redistribute
- * it freely, subject to the following restrictions:
- *
- * 1. The origin of this software must not be misrepresented; you must
- * not claim that you wrote the original software. If you use this
- * software in a product, an acknowledgement in the product
- * documentation would be appreciated but is not required.
- * 2. Altered source versions must be plainly marked as such, and must
- * not be misrepresented as being the original software.
- * 3. This notice may not be removed or altered from any source
- * distribution.
- *
- * This file is part of Stones of Nvwa:
- * http://sourceforge.net/projects/nvwa
- *
- */
- /*!
- * \file bool_array.h
- * \ingroup QxMemLeak
- *
- * Header file for class bool_array (packed boolean array).
- *
- * \version 3.1, 2005/08/25
- * \author Wu Yongwei
- *
- */
- #ifndef QT_NO_DEBUG
- #ifndef _QX_MODE_RELEASE
- #if _QX_USE_MEM_LEAK_DETECTION
- #ifndef _BOOL_ARRAY_H
- #define _BOOL_ARRAY_H
- #ifdef _MSC_VER
- #pragma once
- #endif
- #include <assert.h> // assert
- #include <stdlib.h> // exit, free, and NULL
- #include <new> // std::bad_alloc
- #include <stdexcept> // std::out_of_range
- #include <string> // for exception constructors
- namespace qx {
- namespace memory {
- #ifndef _BYTE_DEFINED
- #define _BYTE_DEFINED
- typedef unsigned char BYTE;
- #endif // !_BYTE_DEFINED
- /**
- * Class to represent a packed boolean array.
- *
- * This was first written in April 1995, before I knew of any existing
- * implementation of this kind of classes. Of course, the C++ Standard
- * Template Library now demands an implementation of packed boolean
- * array as `vector<bool>', but the code here should still be useful
- * for the following three reasons: (1) STL support of MSVC 6 did not
- * implement this specialization (nor did it have a `bit_vector'); (2) I
- * incorporated some useful member functions from the STL bitset into
- * this `bool_array', including `reset', `set', `flip', and `count';
- * (3) In my tests under MSVC 6 and GCC 2.95.3/3.2.3 my code is really
- * FASTER than vector<bool> or the normal boolean array.
- */
- class QX_DLL_EXPORT bool_array
- {
- /** Class to represent a reference to an array element. */
- class QX_DLL_EXPORT _Element
- {
- public:
- _Element(BYTE* __ptr, unsigned long __idx);
- bool operator=(bool ___value);
- operator bool() const;
- private:
- BYTE* _M_byte_ptr;
- size_t _M_byte_idx;
- size_t _M_bit_idx;
- };
- public:
- bool_array() : _M_byte_ptr(NULL), _M_length(0) {}
- explicit bool_array(unsigned long __size);
- ~bool_array() { if (_M_byte_ptr != NULL) free(_M_byte_ptr); }
- bool create(unsigned long __size);
- void initialize(bool ___value);
- // Using unsigned type here can increase performance!
- _Element operator[](unsigned long __idx);
- bool at(unsigned long __idx) const;
- void reset(unsigned long __idx);
- void set(unsigned long __idx);
- unsigned long size() const { return _M_length; }
- unsigned long count() const;
- unsigned long count(unsigned long __beg, unsigned long __end) const;
- void flip();
- private:
- BYTE* _M_byte_ptr;
- unsigned long _M_length;
- static BYTE _S_bit_count[256];
- };
- /* Inline functions */
- /**
- * Constructs a reference to an array element.
- *
- * @param __ptr pointer to the interal boolean data
- * @param __idx index of the array element to access
- */
- inline bool_array::_Element::_Element(BYTE* __ptr, unsigned long __idx)
- {
- _M_byte_ptr = __ptr;
- _M_byte_idx = (size_t)(__idx / 8);
- _M_bit_idx = (size_t)(__idx % 8);
- }
- /**
- * Assigns a new boolean value to an array element.
- *
- * @param ___value the new boolean value
- * @return the assigned boolean value
- */
- inline bool bool_array::_Element::operator=(bool ___value)
- {
- if (___value)
- *(_M_byte_ptr + _M_byte_idx) |= 1 << _M_bit_idx;
- else
- *(_M_byte_ptr + _M_byte_idx) &= ~(1 << _M_bit_idx);
- return ___value;
- }
- /**
- * Reads the boolean value from an array element.
- *
- * @return the boolean value of the accessed array element
- */
- inline bool_array::_Element::operator bool() const
- {
- return *(_M_byte_ptr + _M_byte_idx) & (1 << _M_bit_idx) ? true : false;
- }
- /**
- * Constructs the packed boolean array with a specific size.
- *
- * @param __size size of the array
- * @throw std::out_of_range if \a __size equals \c 0
- * @throw std::bad_alloc if memory is insufficient
- */
- inline bool_array::bool_array(unsigned long __size)
- : _M_byte_ptr(NULL), _M_length(0)
- {
- if (__size == 0)
- throw std::out_of_range("invalid bool_array size");
- if (!create(__size))
- throw std::bad_alloc();
- }
- /**
- * Creates a reference to an array element.
- *
- * @param __idx index of the array element to access
- */
- inline bool_array::_Element bool_array::operator[](unsigned long __idx)
- {
- assert(_M_byte_ptr);
- assert(__idx < _M_length);
- return _Element(_M_byte_ptr, __idx);
- }
- /**
- * Reads the boolean value of an array element via an index.
- *
- * @param __idx index of the array element to access
- * @return the boolean value of the accessed array element
- * @throw std::out_of_range when the index is too big
- */
- inline bool bool_array::at(unsigned long __idx) const
- {
- size_t __byte_idx, __bit_idx;
- if (__idx >= _M_length)
- throw std::out_of_range("invalid bool_array subscript");
- __byte_idx = (size_t)(__idx / 8);
- __bit_idx = (size_t)(__idx % 8);
- return *(_M_byte_ptr + __byte_idx) & (1 << __bit_idx) ? true : false;
- }
- /**
- * Resets an array element to \c false via an index.
- *
- * @param __idx index of the array element to access
- * @throw std::out_of_range when the index is too big
- */
- inline void bool_array::reset(unsigned long __idx)
- {
- size_t __byte_idx, __bit_idx;
- if (__idx >= _M_length)
- throw std::out_of_range("invalid bool_array subscript");
- __byte_idx = (size_t)(__idx / 8);
- __bit_idx = (size_t)(__idx % 8);
- *(_M_byte_ptr + __byte_idx) &= ~(1 << __bit_idx);
- }
- /**
- * Sets an array element to \c true via an index.
- *
- * @param __idx index of the array element to access
- * @throw std::out_of_range when the index is too big
- */
- inline void bool_array::set(unsigned long __idx)
- {
- size_t __byte_idx, __bit_idx;
- if (__idx >= _M_length)
- throw std::out_of_range("invalid bool_array subscript");
- __byte_idx = (size_t)(__idx / 8);
- __bit_idx = (size_t)(__idx % 8);
- *(_M_byte_ptr + __byte_idx) |= 1 << __bit_idx;
- }
- } // namespace memory
- } // namespace qx
- #endif // _BOOL_ARRAY_H
- #endif // _QX_USE_MEM_LEAK_DETECTION
- #endif // _QX_MODE_RELEASE
- #endif // QT_NO_DEBUG
|