xref: /openbmc/linux/tools/power/cpupower/utils/helpers/bitmask.c (revision 4b4193256c8d3bc3a5397b5cd9494c2ad386317d)
1  // SPDX-License-Identifier: GPL-2.0
2  #include <stdio.h>
3  #include <stdlib.h>
4  #include <string.h>
5  
6  #include <helpers/bitmask.h>
7  
8  /* How many bits in an unsigned long */
9  #define bitsperlong (8 * sizeof(unsigned long))
10  
11  /* howmany(a,b) : how many elements of size b needed to hold all of a */
12  #define howmany(x, y) (((x)+((y)-1))/(y))
13  
14  /* How many longs in mask of n bits */
15  #define longsperbits(n) howmany(n, bitsperlong)
16  
17  #define max(a, b) ((a) > (b) ? (a) : (b))
18  
19  /*
20   * Allocate and free `struct bitmask *`
21   */
22  
23  /* Allocate a new `struct bitmask` with a size of n bits */
bitmask_alloc(unsigned int n)24  struct bitmask *bitmask_alloc(unsigned int n)
25  {
26  	struct bitmask *bmp;
27  
28  	bmp = malloc(sizeof(*bmp));
29  	if (!bmp)
30  		return 0;
31  	bmp->size = n;
32  	bmp->maskp = calloc(longsperbits(n), sizeof(unsigned long));
33  	if (!bmp->maskp) {
34  		free(bmp);
35  		return 0;
36  	}
37  	return bmp;
38  }
39  
40  /* Free `struct bitmask` */
bitmask_free(struct bitmask * bmp)41  void bitmask_free(struct bitmask *bmp)
42  {
43  	if (!bmp)
44  		return;
45  	free(bmp->maskp);
46  	bmp->maskp = (unsigned long *)0xdeadcdef;  /* double free tripwire */
47  	free(bmp);
48  }
49  
50  /*
51   * The routines _getbit() and _setbit() are the only
52   * routines that actually understand the layout of bmp->maskp[].
53   *
54   * On little endian architectures, this could simply be an array of
55   * bytes.  But the kernel layout of bitmasks _is_ visible to userspace
56   * via the sched_(set/get)affinity calls in Linux 2.6, and on big
57   * endian architectures, it is painfully obvious that this is an
58   * array of unsigned longs.
59   */
60  
61  /* Return the value (0 or 1) of bit n in bitmask bmp */
_getbit(const struct bitmask * bmp,unsigned int n)62  static unsigned int _getbit(const struct bitmask *bmp, unsigned int n)
63  {
64  	if (n < bmp->size)
65  		return (bmp->maskp[n/bitsperlong] >> (n % bitsperlong)) & 1;
66  	else
67  		return 0;
68  }
69  
70  /* Set bit n in bitmask bmp to value v (0 or 1) */
_setbit(struct bitmask * bmp,unsigned int n,unsigned int v)71  static void _setbit(struct bitmask *bmp, unsigned int n, unsigned int v)
72  {
73  	if (n < bmp->size) {
74  		if (v)
75  			bmp->maskp[n/bitsperlong] |= 1UL << (n % bitsperlong);
76  		else
77  			bmp->maskp[n/bitsperlong] &=
78  				~(1UL << (n % bitsperlong));
79  	}
80  }
81  
82  /*
83   * When parsing bitmask lists, only allow numbers, separated by one
84   * of the allowed next characters.
85   *
86   * The parameter 'sret' is the return from a sscanf "%u%c".  It is
87   * -1 if the sscanf input string was empty.  It is 0 if the first
88   * character in the sscanf input string was not a decimal number.
89   * It is 1 if the unsigned number matching the "%u" was the end of the
90   * input string.  It is 2 if one or more additional characters followed
91   * the matched unsigned number.  If it is 2, then 'nextc' is the first
92   * character following the number.  The parameter 'ok_next_chars'
93   * is the nul-terminated list of allowed next characters.
94   *
95   * The mask term just scanned was ok if and only if either the numbers
96   * matching the %u were all of the input or if the next character in
97   * the input past the numbers was one of the allowed next characters.
98   */
scan_was_ok(int sret,char nextc,const char * ok_next_chars)99  static int scan_was_ok(int sret, char nextc, const char *ok_next_chars)
100  {
101  	return sret == 1 ||
102  		(sret == 2 && strchr(ok_next_chars, nextc) != NULL);
103  }
104  
nexttoken(const char * q,int sep)105  static const char *nexttoken(const char *q,  int sep)
106  {
107  	if (q)
108  		q = strchr(q, sep);
109  	if (q)
110  		q++;
111  	return q;
112  }
113  
114  /* Set a single bit i in bitmask */
bitmask_setbit(struct bitmask * bmp,unsigned int i)115  struct bitmask *bitmask_setbit(struct bitmask *bmp, unsigned int i)
116  {
117  	_setbit(bmp, i, 1);
118  	return bmp;
119  }
120  
121  /* Set all bits in bitmask: bmp = ~0 */
bitmask_setall(struct bitmask * bmp)122  struct bitmask *bitmask_setall(struct bitmask *bmp)
123  {
124  	unsigned int i;
125  	for (i = 0; i < bmp->size; i++)
126  		_setbit(bmp, i, 1);
127  	return bmp;
128  }
129  
130  /* Clear all bits in bitmask: bmp = 0 */
bitmask_clearall(struct bitmask * bmp)131  struct bitmask *bitmask_clearall(struct bitmask *bmp)
132  {
133  	unsigned int i;
134  	for (i = 0; i < bmp->size; i++)
135  		_setbit(bmp, i, 0);
136  	return bmp;
137  }
138  
139  /* True if all bits are clear */
bitmask_isallclear(const struct bitmask * bmp)140  int bitmask_isallclear(const struct bitmask *bmp)
141  {
142  	unsigned int i;
143  	for (i = 0; i < bmp->size; i++)
144  		if (_getbit(bmp, i))
145  			return 0;
146  	return 1;
147  }
148  
149  /* True if specified bit i is set */
bitmask_isbitset(const struct bitmask * bmp,unsigned int i)150  int bitmask_isbitset(const struct bitmask *bmp, unsigned int i)
151  {
152  	return _getbit(bmp, i);
153  }
154  
155  /* Number of lowest set bit (min) */
bitmask_first(const struct bitmask * bmp)156  unsigned int bitmask_first(const struct bitmask *bmp)
157  {
158  	return bitmask_next(bmp, 0);
159  }
160  
161  /* Number of highest set bit (max) */
bitmask_last(const struct bitmask * bmp)162  unsigned int bitmask_last(const struct bitmask *bmp)
163  {
164  	unsigned int i;
165  	unsigned int m = bmp->size;
166  	for (i = 0; i < bmp->size; i++)
167  		if (_getbit(bmp, i))
168  			m = i;
169  	return m;
170  }
171  
172  /* Number of next set bit at or above given bit i */
bitmask_next(const struct bitmask * bmp,unsigned int i)173  unsigned int bitmask_next(const struct bitmask *bmp, unsigned int i)
174  {
175  	unsigned int n;
176  	for (n = i; n < bmp->size; n++)
177  		if (_getbit(bmp, n))
178  			break;
179  	return n;
180  }
181  
182  /*
183   * Parses a comma-separated list of numbers and ranges of numbers,
184   * with optional ':%u' strides modifying ranges, into provided bitmask.
185   * Some examples of input lists and their equivalent simple list:
186   *	Input		Equivalent to
187   *	0-3		0,1,2,3
188   *	0-7:2		0,2,4,6
189   *	1,3,5-7		1,3,5,6,7
190   *	0-3:2,8-15:4	0,2,8,12
191   */
bitmask_parselist(const char * buf,struct bitmask * bmp)192  int bitmask_parselist(const char *buf, struct bitmask *bmp)
193  {
194  	const char *p, *q;
195  
196  	bitmask_clearall(bmp);
197  
198  	q = buf;
199  	while (p = q, q = nexttoken(q, ','), p) {
200  		unsigned int a;		/* begin of range */
201  		unsigned int b;		/* end of range */
202  		unsigned int s;		/* stride */
203  		const char *c1, *c2;	/* next tokens after '-' or ',' */
204  		char nextc;		/* char after sscanf %u match */
205  		int sret;		/* sscanf return (number of matches) */
206  
207  		sret = sscanf(p, "%u%c", &a, &nextc);
208  		if (!scan_was_ok(sret, nextc, ",-"))
209  			goto err;
210  		b = a;
211  		s = 1;
212  		c1 = nexttoken(p, '-');
213  		c2 = nexttoken(p, ',');
214  		if (c1 != NULL && (c2 == NULL || c1 < c2)) {
215  			sret = sscanf(c1, "%u%c", &b, &nextc);
216  			if (!scan_was_ok(sret, nextc, ",:"))
217  				goto err;
218  			c1 = nexttoken(c1, ':');
219  			if (c1 != NULL && (c2 == NULL || c1 < c2)) {
220  				sret = sscanf(c1, "%u%c", &s, &nextc);
221  				if (!scan_was_ok(sret, nextc, ","))
222  					goto err;
223  			}
224  		}
225  		if (!(a <= b))
226  			goto err;
227  		if (b >= bmp->size)
228  			goto err;
229  		while (a <= b) {
230  			_setbit(bmp, a, 1);
231  			a += s;
232  		}
233  	}
234  	return 0;
235  err:
236  	bitmask_clearall(bmp);
237  	return -1;
238  }
239  
240  /*
241   * emit(buf, buflen, rbot, rtop, len)
242   *
243   * Helper routine for bitmask_displaylist().  Write decimal number
244   * or range to buf+len, suppressing output past buf+buflen, with optional
245   * comma-prefix.  Return len of what would be written to buf, if it
246   * all fit.
247   */
248  
emit(char * buf,int buflen,int rbot,int rtop,int len)249  static inline int emit(char *buf, int buflen, int rbot, int rtop, int len)
250  {
251  	if (len > 0)
252  		len += snprintf(buf + len, max(buflen - len, 0), ",");
253  	if (rbot == rtop)
254  		len += snprintf(buf + len, max(buflen - len, 0), "%d", rbot);
255  	else
256  		len += snprintf(buf + len, max(buflen - len, 0), "%d-%d",
257  				rbot, rtop);
258  	return len;
259  }
260  
261  /*
262   * Write decimal list representation of bmp to buf.
263   *
264   * Output format is a comma-separated list of decimal numbers and
265   * ranges.  Consecutively set bits are shown as two hyphen-separated
266   * decimal numbers, the smallest and largest bit numbers set in
267   * the range.  Output format is compatible with the format
268   * accepted as input by bitmap_parselist().
269   *
270   * The return value is the number of characters which would be
271   * generated for the given input, excluding the trailing '\0', as
272   * per ISO C99.
273   */
274  
bitmask_displaylist(char * buf,int buflen,const struct bitmask * bmp)275  int bitmask_displaylist(char *buf, int buflen, const struct bitmask *bmp)
276  {
277  	int len = 0;
278  	/* current bit is 'cur', most recently seen range is [rbot, rtop] */
279  	unsigned int cur, rbot, rtop;
280  
281  	if (buflen > 0)
282  		*buf = 0;
283  	rbot = cur = bitmask_first(bmp);
284  	while (cur < bmp->size) {
285  		rtop = cur;
286  		cur = bitmask_next(bmp, cur+1);
287  		if (cur >= bmp->size || cur > rtop + 1) {
288  			len = emit(buf, buflen, rbot, rtop, len);
289  			rbot = cur;
290  		}
291  	}
292  	return len;
293  }
294