/* * virbitmap.c: Simple bitmap operations * * Copyright (C) 2010-2013 Red Hat, Inc. * Copyright (C) 2010 Novell, Inc. * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License as published by the Free Software Foundation; either * version 2.1 of the License, or (at your option) any later version. * * This library is distributed in the hope that it 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 this library. If not, see * . * * Author: Jim Fehlig */ #include #include #include #include #include #include #include #include "virbitmap.h" #include "viralloc.h" #include "virbuffer.h" #include "c-ctype.h" #include "count-one-bits.h" #include "virstring.h" #include "virerror.h" #define VIR_FROM_THIS VIR_FROM_NONE struct _virBitmap { size_t max_bit; size_t map_len; unsigned long *map; }; #define VIR_BITMAP_BITS_PER_UNIT ((int) sizeof(unsigned long) * CHAR_BIT) #define VIR_BITMAP_UNIT_OFFSET(b) ((b) / VIR_BITMAP_BITS_PER_UNIT) #define VIR_BITMAP_BIT_OFFSET(b) ((b) % VIR_BITMAP_BITS_PER_UNIT) #define VIR_BITMAP_BIT(b) (1UL << VIR_BITMAP_BIT_OFFSET(b)) /** * virBitmapNew: * @size: number of bits * * Allocate a bitmap capable of containing @size bits. * * Returns a pointer to the allocated bitmap or NULL if * memory cannot be allocated. */ virBitmapPtr virBitmapNew(size_t size) { virBitmapPtr bitmap; size_t sz; if (SIZE_MAX - VIR_BITMAP_BITS_PER_UNIT < size || size == 0) { virReportOOMError(); return NULL; } sz = (size + VIR_BITMAP_BITS_PER_UNIT - 1) / VIR_BITMAP_BITS_PER_UNIT; if (VIR_ALLOC(bitmap) < 0) return NULL; if (VIR_ALLOC_N(bitmap->map, sz) < 0) { VIR_FREE(bitmap); return NULL; } bitmap->max_bit = size; bitmap->map_len = sz; return bitmap; } /** * virBitmapFree: * @bitmap: previously allocated bitmap * * Free @bitmap previously allocated by virBitmapNew. */ void virBitmapFree(virBitmapPtr bitmap) { if (bitmap) { VIR_FREE(bitmap->map); VIR_FREE(bitmap); } } int virBitmapCopy(virBitmapPtr dst, virBitmapPtr src) { if (dst->max_bit != src->max_bit) { errno = EINVAL; return -1; } memcpy(dst->map, src->map, src->map_len * sizeof(src->map[0])); return 0; } /** * virBitmapSetBit: * @bitmap: Pointer to bitmap * @b: bit position to set * * Set bit position @b in @bitmap * * Returns 0 on if bit is successfully set, -1 on error. */ int virBitmapSetBit(virBitmapPtr bitmap, size_t b) { if (bitmap->max_bit <= b) return -1; bitmap->map[VIR_BITMAP_UNIT_OFFSET(b)] |= VIR_BITMAP_BIT(b); return 0; } /** * virBitmapClearBit: * @bitmap: Pointer to bitmap * @b: bit position to clear * * Clear bit position @b in @bitmap * * Returns 0 on if bit is successfully clear, -1 on error. */ int virBitmapClearBit(virBitmapPtr bitmap, size_t b) { if (bitmap->max_bit <= b) return -1; bitmap->map[VIR_BITMAP_UNIT_OFFSET(b)] &= ~VIR_BITMAP_BIT(b); return 0; } /* Helper function. caller must ensure b < bitmap->max_bit */ static bool virBitmapIsSet(virBitmapPtr bitmap, size_t b) { return !!(bitmap->map[VIR_BITMAP_UNIT_OFFSET(b)] & VIR_BITMAP_BIT(b)); } /** * virBitmapGetBit: * @bitmap: Pointer to bitmap * @b: bit position to get * @result: bool pointer to receive bit setting * * Get setting of bit position @b in @bitmap and store in @result * * On success, @result will contain the setting of @b and 0 is * returned. On failure, -1 is returned and @result is unchanged. */ int virBitmapGetBit(virBitmapPtr bitmap, size_t b, bool *result) { if (bitmap->max_bit <= b) return -1; *result = virBitmapIsSet(bitmap, b); return 0; } /** * virBitmapString: * @bitmap: Pointer to bitmap * * Convert @bitmap to printable string. * * Returns pointer to the string or NULL on error. */ char *virBitmapString(virBitmapPtr bitmap) { virBuffer buf = VIR_BUFFER_INITIALIZER; size_t sz; virBufferAddLit(&buf, "0x"); sz = bitmap->map_len; while (sz--) { virBufferAsprintf(&buf, "%0*lx", VIR_BITMAP_BITS_PER_UNIT / 4, bitmap->map[sz]); } if (virBufferError(&buf)) { virBufferFreeAndReset(&buf); return NULL; } return virBufferContentAndReset(&buf); } /** * virBitmapFormat: * @bitmap: the bitmap * * This function is the counterpart of virBitmapParse. This function creates * a human-readable string representing the bits in bitmap. * * See virBitmapParse for the format of @str. * * If bitmap is NULL or it has no bits set, an empty string is returned. * * Returns the string on success or NULL otherwise. Caller should call * VIR_FREE to free the string. */ char *virBitmapFormat(virBitmapPtr bitmap) { virBuffer buf = VIR_BUFFER_INITIALIZER; bool first = true; int start, cur, prev; if (!bitmap || (cur = virBitmapNextSetBit(bitmap, -1)) < 0) { char *ret; ignore_value(VIR_STRDUP(ret, "")); return ret; } start = prev = cur; while (prev >= 0) { cur = virBitmapNextSetBit(bitmap, prev); if (cur == prev + 1) { prev = cur; continue; } /* cur < 0 or cur > prev + 1 */ if (!first) virBufferAddLit(&buf, ","); else first = false; if (prev == start) virBufferAsprintf(&buf, "%d", start); else virBufferAsprintf(&buf, "%d-%d", start, prev); start = prev = cur; } if (virBufferError(&buf)) { virBufferFreeAndReset(&buf); virReportOOMError(); return NULL; } return virBufferContentAndReset(&buf); } /** * virBitmapParse: * @str: points to a string representing a human-readable bitmap * @terminator: character separating the bitmap to parse * @bitmap: a bitmap created from @str * @bitmapSize: the upper limit of num of bits in created bitmap * * This function is the counterpart of virBitmapFormat. This function creates * a bitmap, in which bits are set according to the content of @str. * * @str is a comma separated string of fields N, which means a number of bit * to set, and ^N, which means to unset the bit, and N-M for ranges of bits * to set. * * To allow parsing of bitmaps within larger strings it is possible to set * a termination character in the argument @terminator. When the character * in @terminator is encountered in @str, the parsing of the bitmap stops. * Pass 0 as @terminator if it is not needed. Whitespace characters may not * be used as terminators. * * Returns the number of bits set in @bitmap, or -1 in case of error. */ int virBitmapParse(const char *str, char terminator, virBitmapPtr *bitmap, size_t bitmapSize) { bool neg = false; const char *cur = str; char *tmp; size_t i; int start, last; if (!(*bitmap = virBitmapNew(bitmapSize))) return -1; if (!str) goto error; virSkipSpaces(&cur); if (*cur == '\0') goto error; while (*cur != 0 && *cur != terminator) { /* * 3 constructs are allowed: * - N : a single CPU number * - N-M : a range of CPU numbers with N < M * - ^N : remove a single CPU number from the current set */ if (*cur == '^') { cur++; neg = true; } if (!c_isdigit(*cur)) goto error; if (virStrToLong_i(cur, &tmp, 10, &start) < 0) goto error; if (start < 0) goto error; cur = tmp; virSkipSpaces(&cur); if (*cur == ',' || *cur == 0 || *cur == terminator) { if (neg) { if (virBitmapClearBit(*bitmap, start) < 0) goto error; } else { if (virBitmapSetBit(*bitmap, start) < 0) goto error; } } else if (*cur == '-') { if (neg) goto error; cur++; virSkipSpaces(&cur); if (virStrToLong_i(cur, &tmp, 10, &last) < 0) goto error; if (last < start) goto error; cur = tmp; for (i = start; i <= last; i++) { if (virBitmapSetBit(*bitmap, i) < 0) goto error; } virSkipSpaces(&cur); } if (*cur == ',') { cur++; virSkipSpaces(&cur); neg = false; } else if (*cur == 0 || *cur == terminator) { break; } else { goto error; } } return virBitmapCountBits(*bitmap); error: virReportError(VIR_ERR_INVALID_ARG, _("Failed to parse bitmap '%s'"), str); virBitmapFree(*bitmap); *bitmap = NULL; return -1; } /** * virBitmapNewCopy: * @src: the source bitmap. * * Makes a copy of bitmap @src. * * returns the copied bitmap on success, or NULL otherwise. Caller * should call virBitmapFree to free the returned bitmap. */ virBitmapPtr virBitmapNewCopy(virBitmapPtr src) { virBitmapPtr dst; if ((dst = virBitmapNew(src->max_bit)) == NULL) return NULL; if (virBitmapCopy(dst, src) != 0) { virBitmapFree(dst); return NULL; } return dst; } /** * virBitmapNewData: * @data: the data * @len: length of @data in bytes * * Allocate a bitmap from a chunk of data containing bits * information * * Returns a pointer to the allocated bitmap or NULL if * memory cannot be allocated. */ virBitmapPtr virBitmapNewData(void *data, int len) { virBitmapPtr bitmap; size_t i, j; unsigned long *p; unsigned char *bytes = data; bitmap = virBitmapNew(len * CHAR_BIT); if (!bitmap) return NULL; /* le64toh is not provided by gnulib, so we do the conversion by hand */ p = bitmap->map; for (i = j = 0; i < len; i++, j++) { if (j == sizeof(*p)) { j = 0; p++; } *p |= (unsigned long) bytes[i] << (j * CHAR_BIT); } return bitmap; } /** * virBitmapToData: * @data: the data * @len: len of @data in byte * * Convert a bitmap to a chunk of data containing bits information. * Data consists of sequential bytes, with lower bytes containing * lower bits. * * Returns 0 on success, -1 otherwise. */ int virBitmapToData(virBitmapPtr bitmap, unsigned char **data, int *dataLen) { int len; unsigned long *l; size_t i, j; unsigned char *bytes; len = (bitmap->max_bit + CHAR_BIT - 1) / CHAR_BIT; if (VIR_ALLOC_N(*data, len) < 0) return -1; bytes = *data; *dataLen = len; /* htole64 is not provided by gnulib, so we do the conversion by hand */ l = bitmap->map; for (i = j = 0; i < len; i++, j++) { if (j == sizeof(*l)) { j = 0; l++; } bytes[i] = *l >> (j * CHAR_BIT); } return 0; } /** * virBitmapEqual: * @b1: bitmap 1 * @b2: bitmap 2 * * Compares two bitmaps, whose lengths can be different from each other. * * Returns true if two bitmaps have exactly the same set of bits set, * otherwise false. */ bool virBitmapEqual(virBitmapPtr b1, virBitmapPtr b2) { virBitmapPtr tmp; size_t i; if (b1->max_bit > b2->max_bit) { tmp = b1; b1 = b2; b2 = tmp; } /* Now b1 is the smaller one, if not equal */ for (i = 0; i < b1->map_len; i++) { if (b1->map[i] != b2->map[i]) return false; } for (; i < b2->map_len; i++) { if (b2->map[i]) return false; } return true; } size_t virBitmapSize(virBitmapPtr bitmap) { return bitmap->max_bit; } /** * virBitmapSetAll: * @bitmap: the bitmap * * set all bits in @bitmap. */ void virBitmapSetAll(virBitmapPtr bitmap) { int tail = bitmap->max_bit % VIR_BITMAP_BITS_PER_UNIT; memset(bitmap->map, 0xff, bitmap->map_len * (VIR_BITMAP_BITS_PER_UNIT / CHAR_BIT)); /* Ensure tail bits are clear. */ if (tail) bitmap->map[bitmap->map_len - 1] &= -1UL >> (VIR_BITMAP_BITS_PER_UNIT - tail); } /** * virBitmapClearAll: * @bitmap: the bitmap * * clear all bits in @bitmap. */ void virBitmapClearAll(virBitmapPtr bitmap) { memset(bitmap->map, 0, bitmap->map_len * (VIR_BITMAP_BITS_PER_UNIT / CHAR_BIT)); } /** * virBitmapIsAllSet: * @bitmap: the bitmap to check * * check if all bits in @bitmap are set. */ bool virBitmapIsAllSet(virBitmapPtr bitmap) { size_t i; int unusedBits; size_t sz; unusedBits = bitmap->map_len * VIR_BITMAP_BITS_PER_UNIT - bitmap->max_bit; sz = bitmap->map_len; if (unusedBits > 0) sz--; for (i = 0; i < sz; i++) if (bitmap->map[i] != -1) return false; if (unusedBits > 0) { if ((bitmap->map[sz] & ((1UL << (VIR_BITMAP_BITS_PER_UNIT - unusedBits)) - 1)) != ((1UL << (VIR_BITMAP_BITS_PER_UNIT - unusedBits)) - 1)) return false; } return true; } /** * virBitmapIsAllClear: * @bitmap: the bitmap to check * * check if all bits in @bitmap are clear */ bool virBitmapIsAllClear(virBitmapPtr bitmap) { size_t i; for (i = 0; i < bitmap->map_len; i++) if (bitmap->map[i] != 0) return false; return true; } /** * virBitmapNextSetBit: * @bitmap: the bitmap * @pos: the position after which to search for a set bit * * Search for the first set bit after position @pos in bitmap @bitmap. * @pos can be -1 to search for the first set bit. Position starts * at 0. * * Returns the position of the found bit, or -1 if no bit found. */ ssize_t virBitmapNextSetBit(virBitmapPtr bitmap, ssize_t pos) { size_t nl; size_t nb; unsigned long bits; if (pos < 0) pos = -1; pos++; if (pos >= bitmap->max_bit) return -1; nl = pos / VIR_BITMAP_BITS_PER_UNIT; nb = pos % VIR_BITMAP_BITS_PER_UNIT; bits = bitmap->map[nl] & ~((1UL << nb) - 1); while (bits == 0 && ++nl < bitmap->map_len) { bits = bitmap->map[nl]; } if (bits == 0) return -1; return ffsl(bits) - 1 + nl * VIR_BITMAP_BITS_PER_UNIT; } /** * virBitmapNextClearBit: * @bitmap: the bitmap * @pos: the position after which to search for a clear bit * * Search for the first clear bit after position @pos in bitmap @bitmap. * @pos can be -1 to search for the first set bit. Position starts * at 0. * * Returns the position of the found bit, or -1 if no bit found. */ ssize_t virBitmapNextClearBit(virBitmapPtr bitmap, ssize_t pos) { size_t nl; size_t nb; unsigned long bits; if (pos < 0) pos = -1; pos++; if (pos >= bitmap->max_bit) return -1; nl = pos / VIR_BITMAP_BITS_PER_UNIT; nb = pos % VIR_BITMAP_BITS_PER_UNIT; bits = ~bitmap->map[nl] & ~((1UL << nb) - 1); while (bits == 0 && ++nl < bitmap->map_len) { bits = ~bitmap->map[nl]; } if (nl == bitmap->map_len - 1) { /* Ensure tail bits are ignored. */ int tail = bitmap->max_bit % VIR_BITMAP_BITS_PER_UNIT; if (tail) bits &= -1UL >> (VIR_BITMAP_BITS_PER_UNIT - tail); } if (bits == 0) return -1; return ffsl(bits) - 1 + nl * VIR_BITMAP_BITS_PER_UNIT; } /* Return the number of bits currently set in the map. */ size_t virBitmapCountBits(virBitmapPtr bitmap) { size_t i; size_t ret = 0; for (i = 0; i < bitmap->map_len; i++) ret += count_one_bits_l(bitmap->map[i]); return ret; } /** * virBitmapDataToString: * @data: the data * @len: length of @data in bytes * * Convert a chunk of data containing bits information to a human * readable string, e.g.: 0-1,4 * * Returns: a string representation of the data, or NULL on error */ char * virBitmapDataToString(void *data, int len) { virBitmapPtr map = NULL; char *ret = NULL; if (!(map = virBitmapNewData(data, len))) return NULL; ret = virBitmapFormat(map); virBitmapFree(map); return ret; }