1996-12-09 14:05:35 +00:00
|
|
|
/*-
|
2017-11-27 15:37:16 +00:00
|
|
|
* SPDX-License-Identifier: BSD-2-Clause-FreeBSD
|
|
|
|
*
|
1996-12-10 23:59:04 +00:00
|
|
|
* Copyright (C) 1996
|
|
|
|
* David L. Nugent. All rights reserved.
|
1996-12-09 14:05:35 +00:00
|
|
|
*
|
|
|
|
* Redistribution and use in source and binary forms, with or without
|
|
|
|
* modification, are permitted provided that the following conditions
|
|
|
|
* are met:
|
|
|
|
* 1. Redistributions of source code must retain the above copyright
|
1996-12-10 23:59:04 +00:00
|
|
|
* notice, this list of conditions and the following disclaimer.
|
1996-12-09 14:05:35 +00:00
|
|
|
* 2. Redistributions in binary form must reproduce the above copyright
|
|
|
|
* notice, this list of conditions and the following disclaimer in the
|
|
|
|
* documentation and/or other materials provided with the distribution.
|
|
|
|
*
|
1996-12-10 23:59:04 +00:00
|
|
|
* THIS SOFTWARE IS PROVIDED BY DAVID L. NUGENT AND CONTRIBUTORS ``AS IS'' AND
|
1996-12-09 14:05:35 +00:00
|
|
|
* ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
|
|
|
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
|
1996-12-10 23:59:04 +00:00
|
|
|
* ARE DISCLAIMED. IN NO EVENT SHALL DAVID L. NUGENT OR CONTRIBUTORS BE LIABLE
|
1996-12-09 14:05:35 +00:00
|
|
|
* FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
|
|
|
|
* DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
|
|
|
|
* OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
|
|
|
|
* HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
|
|
|
|
* LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
|
|
|
|
* OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
|
|
|
|
* SUCH DAMAGE.
|
|
|
|
*/
|
|
|
|
|
1997-10-10 06:23:42 +00:00
|
|
|
#ifndef lint
|
|
|
|
static const char rcsid[] =
|
1999-08-28 01:35:59 +00:00
|
|
|
"$FreeBSD$";
|
1997-10-10 06:23:42 +00:00
|
|
|
#endif /* not lint */
|
|
|
|
|
1996-12-09 14:05:35 +00:00
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
|
|
|
|
|
|
|
#include "bitmap.h"
|
|
|
|
|
|
|
|
struct bitmap
|
|
|
|
bm_alloc(int size)
|
|
|
|
{
|
|
|
|
struct bitmap bm;
|
|
|
|
int szmap = (size / 8) + !!(size % 8);
|
|
|
|
|
|
|
|
bm.size = size;
|
|
|
|
bm.map = malloc(szmap);
|
|
|
|
if (bm.map)
|
|
|
|
memset(bm.map, 0, szmap);
|
|
|
|
return bm;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
bm_dealloc(struct bitmap * bm)
|
|
|
|
{
|
2012-12-05 13:56:43 +00:00
|
|
|
free(bm->map);
|
1996-12-09 14:05:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
bm_getmask(int *pos, unsigned char *bmask)
|
|
|
|
{
|
|
|
|
*bmask = (unsigned char) (1 << (*pos % 8));
|
|
|
|
*pos /= 8;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
bm_setbit(struct bitmap * bm, int pos)
|
|
|
|
{
|
|
|
|
unsigned char bmask;
|
|
|
|
|
|
|
|
bm_getmask(&pos, &bmask);
|
|
|
|
bm->map[pos] |= bmask;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
bm_clrbit(struct bitmap * bm, int pos)
|
|
|
|
{
|
|
|
|
unsigned char bmask;
|
|
|
|
|
|
|
|
bm_getmask(&pos, &bmask);
|
|
|
|
bm->map[pos] &= ~bmask;
|
|
|
|
}
|
|
|
|
|
|
|
|
int
|
|
|
|
bm_isset(struct bitmap * bm, int pos)
|
|
|
|
{
|
|
|
|
unsigned char bmask;
|
|
|
|
|
|
|
|
bm_getmask(&pos, &bmask);
|
|
|
|
return !!(bm->map[pos] & bmask);
|
|
|
|
}
|
|
|
|
|
|
|
|
int
|
|
|
|
bm_firstunset(struct bitmap * bm)
|
|
|
|
{
|
|
|
|
int szmap = (bm->size / 8) + !!(bm->size % 8);
|
|
|
|
int at = 0;
|
|
|
|
int pos = 0;
|
|
|
|
|
|
|
|
while (pos < szmap) {
|
|
|
|
unsigned char bmv = bm->map[pos++];
|
|
|
|
unsigned char bmask = 1;
|
|
|
|
|
|
|
|
while (bmask & 0xff) {
|
|
|
|
if ((bmv & bmask) == 0)
|
|
|
|
return at;
|
|
|
|
bmask <<= 1;
|
|
|
|
++at;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return at;
|
|
|
|
}
|
|
|
|
|
|
|
|
int
|
|
|
|
bm_lastset(struct bitmap * bm)
|
|
|
|
{
|
|
|
|
int szmap = (bm->size / 8) + !!(bm->size % 8);
|
|
|
|
int at = 0;
|
|
|
|
int pos = 0;
|
|
|
|
int ofs = 0;
|
|
|
|
|
|
|
|
while (pos < szmap) {
|
|
|
|
unsigned char bmv = bm->map[pos++];
|
|
|
|
unsigned char bmask = 1;
|
|
|
|
|
|
|
|
while (bmask & 0xff) {
|
|
|
|
if ((bmv & bmask) != 0)
|
|
|
|
ofs = at;
|
|
|
|
bmask <<= 1;
|
|
|
|
++at;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return ofs;
|
|
|
|
}
|