| 
									
										
										
										
											2011-03-30 16:30:11 +02:00
										 |  |  | #include <stdio.h>
 | 
					
						
							|  |  |  | #include <stdlib.h>
 | 
					
						
							|  |  |  | #include <string.h>
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <helpers/bitmask.h>
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* How many bits in an unsigned long */ | 
					
						
							|  |  |  | #define bitsperlong (8 * sizeof(unsigned long))
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* howmany(a,b) : how many elements of size b needed to hold all of a */ | 
					
						
							| 
									
										
										
										
											2011-04-19 20:16:05 +02:00
										 |  |  | #define howmany(x, y) (((x)+((y)-1))/(y))
 | 
					
						
							| 
									
										
										
										
											2011-03-30 16:30:11 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | /* How many longs in mask of n bits */ | 
					
						
							|  |  |  | #define longsperbits(n) howmany(n, bitsperlong)
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-04-19 20:16:05 +02:00
										 |  |  | #define max(a, b) ((a) > (b) ? (a) : (b))
 | 
					
						
							| 
									
										
										
										
											2011-03-30 16:30:11 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | /*
 | 
					
						
							|  |  |  |  * Allocate and free `struct bitmask *` | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Allocate a new `struct bitmask` with a size of n bits */ | 
					
						
							|  |  |  | struct bitmask *bitmask_alloc(unsigned int n) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct bitmask *bmp; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	bmp = malloc(sizeof(*bmp)); | 
					
						
							|  |  |  | 	if (bmp == 0) | 
					
						
							|  |  |  | 		return 0; | 
					
						
							|  |  |  | 	bmp->size = n; | 
					
						
							|  |  |  | 	bmp->maskp = calloc(longsperbits(n), sizeof(unsigned long)); | 
					
						
							|  |  |  | 	if (bmp->maskp == 0) { | 
					
						
							|  |  |  | 		free(bmp); | 
					
						
							|  |  |  | 		return 0; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return bmp; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Free `struct bitmask` */ | 
					
						
							|  |  |  | void bitmask_free(struct bitmask *bmp) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	if (bmp == 0) | 
					
						
							|  |  |  | 		return; | 
					
						
							|  |  |  | 	free(bmp->maskp); | 
					
						
							|  |  |  | 	bmp->maskp = (unsigned long *)0xdeadcdef;  /* double free tripwire */ | 
					
						
							|  |  |  | 	free(bmp); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*
 | 
					
						
							|  |  |  |  * The routines _getbit() and _setbit() are the only | 
					
						
							|  |  |  |  * routines that actually understand the layout of bmp->maskp[]. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * On little endian architectures, this could simply be an array of | 
					
						
							|  |  |  |  * bytes.  But the kernel layout of bitmasks _is_ visible to userspace | 
					
						
							|  |  |  |  * via the sched_(set/get)affinity calls in Linux 2.6, and on big | 
					
						
							|  |  |  |  * endian architectures, it is painfully obvious that this is an | 
					
						
							|  |  |  |  * array of unsigned longs. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Return the value (0 or 1) of bit n in bitmask bmp */ | 
					
						
							|  |  |  | static unsigned int _getbit(const struct bitmask *bmp, unsigned int n) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	if (n < bmp->size) | 
					
						
							|  |  |  | 		return (bmp->maskp[n/bitsperlong] >> (n % bitsperlong)) & 1; | 
					
						
							|  |  |  | 	else | 
					
						
							|  |  |  | 		return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Set bit n in bitmask bmp to value v (0 or 1) */ | 
					
						
							|  |  |  | static void _setbit(struct bitmask *bmp, unsigned int n, unsigned int v) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	if (n < bmp->size) { | 
					
						
							|  |  |  | 		if (v) | 
					
						
							|  |  |  | 			bmp->maskp[n/bitsperlong] |= 1UL << (n % bitsperlong); | 
					
						
							|  |  |  | 		else | 
					
						
							| 
									
										
										
										
											2011-04-19 20:16:05 +02:00
										 |  |  | 			bmp->maskp[n/bitsperlong] &= | 
					
						
							|  |  |  | 				~(1UL << (n % bitsperlong)); | 
					
						
							| 
									
										
										
										
											2011-03-30 16:30:11 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*
 | 
					
						
							|  |  |  |  * When parsing bitmask lists, only allow numbers, separated by one | 
					
						
							|  |  |  |  * of the allowed next characters. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * The parameter 'sret' is the return from a sscanf "%u%c".  It is | 
					
						
							|  |  |  |  * -1 if the sscanf input string was empty.  It is 0 if the first | 
					
						
							|  |  |  |  * character in the sscanf input string was not a decimal number. | 
					
						
							|  |  |  |  * It is 1 if the unsigned number matching the "%u" was the end of the | 
					
						
							|  |  |  |  * input string.  It is 2 if one or more additional characters followed | 
					
						
							|  |  |  |  * the matched unsigned number.  If it is 2, then 'nextc' is the first | 
					
						
							|  |  |  |  * character following the number.  The parameter 'ok_next_chars' | 
					
						
							|  |  |  |  * is the nul-terminated list of allowed next characters. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * The mask term just scanned was ok if and only if either the numbers | 
					
						
							|  |  |  |  * matching the %u were all of the input or if the next character in | 
					
						
							|  |  |  |  * the input past the numbers was one of the allowed next characters. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | static int scan_was_ok(int sret, char nextc, const char *ok_next_chars) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	return sret == 1 || | 
					
						
							|  |  |  | 		(sret == 2 && strchr(ok_next_chars, nextc) != NULL); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static const char *nexttoken(const char *q,  int sep) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	if (q) | 
					
						
							|  |  |  | 		q = strchr(q, sep); | 
					
						
							|  |  |  | 	if (q) | 
					
						
							|  |  |  | 		q++; | 
					
						
							|  |  |  | 	return q; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Set a single bit i in bitmask */ | 
					
						
							|  |  |  | struct bitmask *bitmask_setbit(struct bitmask *bmp, unsigned int i) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	_setbit(bmp, i, 1); | 
					
						
							|  |  |  | 	return bmp; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Set all bits in bitmask: bmp = ~0 */ | 
					
						
							|  |  |  | struct bitmask *bitmask_setall(struct bitmask *bmp) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	unsigned int i; | 
					
						
							|  |  |  | 	for (i = 0; i < bmp->size; i++) | 
					
						
							|  |  |  | 		_setbit(bmp, i, 1); | 
					
						
							|  |  |  | 	return bmp; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Clear all bits in bitmask: bmp = 0 */ | 
					
						
							|  |  |  | struct bitmask *bitmask_clearall(struct bitmask *bmp) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	unsigned int i; | 
					
						
							|  |  |  | 	for (i = 0; i < bmp->size; i++) | 
					
						
							|  |  |  | 		_setbit(bmp, i, 0); | 
					
						
							|  |  |  | 	return bmp; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* True if all bits are clear */ | 
					
						
							|  |  |  | int bitmask_isallclear(const struct bitmask *bmp) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	unsigned int i; | 
					
						
							|  |  |  | 	for (i = 0; i < bmp->size; i++) | 
					
						
							|  |  |  | 		if (_getbit(bmp, i)) | 
					
						
							|  |  |  | 			return 0; | 
					
						
							|  |  |  | 	return 1; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* True if specified bit i is set */ | 
					
						
							|  |  |  | int bitmask_isbitset(const struct bitmask *bmp, unsigned int i) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	return _getbit(bmp, i); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Number of lowest set bit (min) */ | 
					
						
							|  |  |  | unsigned int bitmask_first(const struct bitmask *bmp) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	return bitmask_next(bmp, 0); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Number of highest set bit (max) */ | 
					
						
							|  |  |  | unsigned int bitmask_last(const struct bitmask *bmp) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	unsigned int i; | 
					
						
							|  |  |  | 	unsigned int m = bmp->size; | 
					
						
							|  |  |  | 	for (i = 0; i < bmp->size; i++) | 
					
						
							|  |  |  | 		if (_getbit(bmp, i)) | 
					
						
							|  |  |  | 			m = i; | 
					
						
							|  |  |  | 	return m; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Number of next set bit at or above given bit i */ | 
					
						
							|  |  |  | unsigned int bitmask_next(const struct bitmask *bmp, unsigned int i) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	unsigned int n; | 
					
						
							|  |  |  | 	for (n = i; n < bmp->size; n++) | 
					
						
							|  |  |  | 		if (_getbit(bmp, n)) | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 	return n; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*
 | 
					
						
							|  |  |  |  * Parses a comma-separated list of numbers and ranges of numbers, | 
					
						
							|  |  |  |  * with optional ':%u' strides modifying ranges, into provided bitmask. | 
					
						
							|  |  |  |  * Some examples of input lists and their equivalent simple list: | 
					
						
							|  |  |  |  *	Input		Equivalent to | 
					
						
							|  |  |  |  *	0-3		0,1,2,3 | 
					
						
							|  |  |  |  *	0-7:2		0,2,4,6 | 
					
						
							|  |  |  |  *	1,3,5-7		1,3,5,6,7 | 
					
						
							| 
									
										
										
										
											2011-04-19 20:16:05 +02:00
										 |  |  |  *	0-3:2,8-15:4	0,2,8,12 | 
					
						
							| 
									
										
										
										
											2011-03-30 16:30:11 +02:00
										 |  |  |  */ | 
					
						
							|  |  |  | int bitmask_parselist(const char *buf, struct bitmask *bmp) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	const char *p, *q; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	bitmask_clearall(bmp); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	q = buf; | 
					
						
							|  |  |  | 	while (p = q, q = nexttoken(q, ','), p) { | 
					
						
							|  |  |  | 		unsigned int a;		/* begin of range */ | 
					
						
							|  |  |  | 		unsigned int b;		/* end of range */ | 
					
						
							|  |  |  | 		unsigned int s;		/* stride */ | 
					
						
							|  |  |  | 		const char *c1, *c2;	/* next tokens after '-' or ',' */ | 
					
						
							|  |  |  | 		char nextc;		/* char after sscanf %u match */ | 
					
						
							|  |  |  | 		int sret;		/* sscanf return (number of matches) */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		sret = sscanf(p, "%u%c", &a, &nextc); | 
					
						
							|  |  |  | 		if (!scan_was_ok(sret, nextc, ",-")) | 
					
						
							|  |  |  | 			goto err; | 
					
						
							|  |  |  | 		b = a; | 
					
						
							|  |  |  | 		s = 1; | 
					
						
							|  |  |  | 		c1 = nexttoken(p, '-'); | 
					
						
							|  |  |  | 		c2 = nexttoken(p, ','); | 
					
						
							|  |  |  | 		if (c1 != NULL && (c2 == NULL || c1 < c2)) { | 
					
						
							|  |  |  | 			sret = sscanf(c1, "%u%c", &b, &nextc); | 
					
						
							|  |  |  | 			if (!scan_was_ok(sret, nextc, ",:")) | 
					
						
							|  |  |  | 				goto err; | 
					
						
							|  |  |  | 			c1 = nexttoken(c1, ':'); | 
					
						
							|  |  |  | 			if (c1 != NULL && (c2 == NULL || c1 < c2)) { | 
					
						
							|  |  |  | 				sret = sscanf(c1, "%u%c", &s, &nextc); | 
					
						
							|  |  |  | 				if (!scan_was_ok(sret, nextc, ",")) | 
					
						
							|  |  |  | 					goto err; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		if (!(a <= b)) | 
					
						
							|  |  |  | 			goto err; | 
					
						
							|  |  |  | 		if (b >= bmp->size) | 
					
						
							|  |  |  | 			goto err; | 
					
						
							|  |  |  | 		while (a <= b) { | 
					
						
							|  |  |  | 			_setbit(bmp, a, 1); | 
					
						
							|  |  |  | 			a += s; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | err: | 
					
						
							|  |  |  | 	bitmask_clearall(bmp); | 
					
						
							|  |  |  | 	return -1; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*
 | 
					
						
							|  |  |  |  * emit(buf, buflen, rbot, rtop, len) | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Helper routine for bitmask_displaylist().  Write decimal number | 
					
						
							|  |  |  |  * or range to buf+len, suppressing output past buf+buflen, with optional | 
					
						
							|  |  |  |  * comma-prefix.  Return len of what would be written to buf, if it | 
					
						
							|  |  |  |  * all fit. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static inline int emit(char *buf, int buflen, int rbot, int rtop, int len) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	if (len > 0) | 
					
						
							|  |  |  | 		len += snprintf(buf + len, max(buflen - len, 0), ","); | 
					
						
							|  |  |  | 	if (rbot == rtop) | 
					
						
							|  |  |  | 		len += snprintf(buf + len, max(buflen - len, 0), "%d", rbot); | 
					
						
							|  |  |  | 	else | 
					
						
							| 
									
										
										
										
											2011-04-19 20:16:05 +02:00
										 |  |  | 		len += snprintf(buf + len, max(buflen - len, 0), "%d-%d", | 
					
						
							|  |  |  | 				rbot, rtop); | 
					
						
							| 
									
										
										
										
											2011-03-30 16:30:11 +02:00
										 |  |  | 	return len; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*
 | 
					
						
							|  |  |  |  * Write decimal list representation of bmp to buf. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Output format is a comma-separated list of decimal numbers and | 
					
						
							|  |  |  |  * ranges.  Consecutively set bits are shown as two hyphen-separated | 
					
						
							|  |  |  |  * decimal numbers, the smallest and largest bit numbers set in | 
					
						
							|  |  |  |  * the range.  Output format is compatible with the format | 
					
						
							|  |  |  |  * accepted as input by bitmap_parselist(). | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * The return value is the number of characters which would be | 
					
						
							|  |  |  |  * generated for the given input, excluding the trailing '\0', as | 
					
						
							|  |  |  |  * per ISO C99. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int bitmask_displaylist(char *buf, int buflen, const struct bitmask *bmp) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int len = 0; | 
					
						
							|  |  |  | 	/* current bit is 'cur', most recently seen range is [rbot, rtop] */ | 
					
						
							|  |  |  | 	unsigned int cur, rbot, rtop; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (buflen > 0) | 
					
						
							|  |  |  | 		*buf = 0; | 
					
						
							|  |  |  | 	rbot = cur = bitmask_first(bmp); | 
					
						
							|  |  |  | 	while (cur < bmp->size) { | 
					
						
							|  |  |  | 		rtop = cur; | 
					
						
							|  |  |  | 		cur = bitmask_next(bmp, cur+1); | 
					
						
							|  |  |  | 		if (cur >= bmp->size || cur > rtop + 1) { | 
					
						
							|  |  |  | 			len = emit(buf, buflen, rbot, rtop, len); | 
					
						
							|  |  |  | 			rbot = cur; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return len; | 
					
						
							|  |  |  | } |