Home | History | Annotate | Download | only in pcm
      1 /*
      2  *  Mask inlines
      3  *  Copyright (c) 2000 by Abramo Bagnara <abramo (at) alsa-project.org>
      4  *
      5  *
      6  *   This library is free software; you can redistribute it and/or modify
      7  *   it under the terms of the GNU Lesser General Public License as
      8  *   published by the Free Software Foundation; either version 2.1 of
      9  *   the License, or (at your option) any later version.
     10  *
     11  *   This program is distributed in the hope that it will be useful,
     12  *   but WITHOUT ANY WARRANTY; without even the implied warranty of
     13  *   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
     14  *   GNU Lesser General Public License for more details.
     15  *
     16  *   You should have received a copy of the GNU Lesser General Public
     17  *   License along with this library; if not, write to the Free Software
     18  *   Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307 USA
     19  *
     20  */
     21 
     22 #include <sys/types.h>
     23 
     24 #define MASK_INLINE static inline
     25 
     26 #define MASK_MAX SND_MASK_MAX
     27 #define MASK_SIZE (MASK_MAX / 32)
     28 
     29 #define MASK_OFS(i)	((i) >> 5)
     30 #define MASK_BIT(i)	(1U << ((i) & 31))
     31 
     32 MASK_INLINE unsigned int ld2(u_int32_t v)
     33 {
     34         unsigned r = 0;
     35 
     36         if (v >= 0x10000) {
     37                 v >>= 16;
     38                 r += 16;
     39         }
     40         if (v >= 0x100) {
     41                 v >>= 8;
     42                 r += 8;
     43         }
     44         if (v >= 0x10) {
     45                 v >>= 4;
     46                 r += 4;
     47         }
     48         if (v >= 4) {
     49                 v >>= 2;
     50                 r += 2;
     51         }
     52         if (v >= 2)
     53                 r++;
     54         return r;
     55 }
     56 
     57 MASK_INLINE unsigned int hweight32(u_int32_t v)
     58 {
     59         v = (v & 0x55555555) + ((v >> 1) & 0x55555555);
     60         v = (v & 0x33333333) + ((v >> 2) & 0x33333333);
     61         v = (v & 0x0F0F0F0F) + ((v >> 4) & 0x0F0F0F0F);
     62         v = (v & 0x00FF00FF) + ((v >> 8) & 0x00FF00FF);
     63         return (v & 0x0000FFFF) + ((v >> 16) & 0x0000FFFF);
     64 }
     65 
     66 MASK_INLINE size_t snd_mask_sizeof(void)
     67 {
     68 	return sizeof(snd_mask_t);
     69 }
     70 
     71 MASK_INLINE void snd_mask_none(snd_mask_t *mask)
     72 {
     73 	memset(mask, 0, sizeof(*mask));
     74 }
     75 
     76 MASK_INLINE void snd_mask_any(snd_mask_t *mask)
     77 {
     78 	memset(mask, 0xff, MASK_SIZE * sizeof(u_int32_t));
     79 }
     80 
     81 MASK_INLINE int snd_mask_empty(const snd_mask_t *mask)
     82 {
     83 	int i;
     84 	for (i = 0; i < MASK_SIZE; i++)
     85 		if (mask->bits[i])
     86 			return 0;
     87 	return 1;
     88 }
     89 
     90 MASK_INLINE int snd_mask_full(const snd_mask_t *mask)
     91 {
     92 	int i;
     93 	for (i = 0; i < MASK_SIZE; i++)
     94 		if (mask->bits[i] != 0xffffffff)
     95 			return 0;
     96 	return 1;
     97 }
     98 
     99 MASK_INLINE unsigned int snd_mask_count(const snd_mask_t *mask)
    100 {
    101 	int i, w = 0;
    102 	for (i = 0; i < MASK_SIZE; i++)
    103 		w += hweight32(mask->bits[i]);
    104 	return w;
    105 }
    106 
    107 MASK_INLINE unsigned int snd_mask_min(const snd_mask_t *mask)
    108 {
    109 	int i;
    110 	assert(!snd_mask_empty(mask));
    111 	for (i = 0; i < MASK_SIZE; i++) {
    112 		if (mask->bits[i])
    113 			return ffs(mask->bits[i]) - 1 + (i << 5);
    114 	}
    115 	return 0;
    116 }
    117 
    118 MASK_INLINE unsigned int snd_mask_max(const snd_mask_t *mask)
    119 {
    120 	int i;
    121 	assert(!snd_mask_empty(mask));
    122 	for (i = MASK_SIZE - 1; i >= 0; i--) {
    123 		if (mask->bits[i])
    124 			return ld2(mask->bits[i]) + (i << 5);
    125 	}
    126 	return 0;
    127 }
    128 
    129 MASK_INLINE void snd_mask_set(snd_mask_t *mask, unsigned int val)
    130 {
    131 	assert(val <= SND_MASK_MAX);
    132 	mask->bits[MASK_OFS(val)] |= MASK_BIT(val);
    133 }
    134 
    135 MASK_INLINE void snd_mask_reset(snd_mask_t *mask, unsigned int val)
    136 {
    137 	assert(val <= SND_MASK_MAX);
    138 	mask->bits[MASK_OFS(val)] &= ~MASK_BIT(val);
    139 }
    140 
    141 MASK_INLINE void snd_mask_set_range(snd_mask_t *mask, unsigned int from, unsigned int to)
    142 {
    143 	unsigned int i;
    144 	assert(to <= SND_MASK_MAX && from <= to);
    145 	for (i = from; i <= to; i++)
    146 		mask->bits[MASK_OFS(i)] |= MASK_BIT(i);
    147 }
    148 
    149 MASK_INLINE void snd_mask_reset_range(snd_mask_t *mask, unsigned int from, unsigned int to)
    150 {
    151 	unsigned int i;
    152 	assert(to <= SND_MASK_MAX && from <= to);
    153 	for (i = from; i <= to; i++)
    154 		mask->bits[MASK_OFS(i)] &= ~MASK_BIT(i);
    155 }
    156 
    157 MASK_INLINE void snd_mask_leave(snd_mask_t *mask, unsigned int val)
    158 {
    159 	unsigned int v;
    160 	assert(val <= SND_MASK_MAX);
    161 	v = mask->bits[MASK_OFS(val)] & MASK_BIT(val);
    162 	snd_mask_none(mask);
    163 	mask->bits[MASK_OFS(val)] = v;
    164 }
    165 
    166 MASK_INLINE void snd_mask_intersect(snd_mask_t *mask, const snd_mask_t *v)
    167 {
    168 	int i;
    169 	for (i = 0; i < MASK_SIZE; i++)
    170 		mask->bits[i] &= v->bits[i];
    171 }
    172 
    173 MASK_INLINE void snd_mask_union(snd_mask_t *mask, const snd_mask_t *v)
    174 {
    175 	int i;
    176 	for (i = 0; i < MASK_SIZE; i++)
    177 		mask->bits[i] |= v->bits[i];
    178 }
    179 
    180 MASK_INLINE int snd_mask_eq(const snd_mask_t *mask, const snd_mask_t *v)
    181 {
    182 	return ! memcmp(mask, v, MASK_SIZE * 4);
    183 }
    184 
    185 MASK_INLINE void snd_mask_copy(snd_mask_t *mask, const snd_mask_t *v)
    186 {
    187 	*mask = *v;
    188 }
    189 
    190 MASK_INLINE int snd_mask_test(const snd_mask_t *mask, unsigned int val)
    191 {
    192 	assert(val <= SND_MASK_MAX);
    193 	return mask->bits[MASK_OFS(val)] & MASK_BIT(val);
    194 }
    195 
    196 MASK_INLINE int snd_mask_single(const snd_mask_t *mask)
    197 {
    198 	int i, c = 0;
    199 	assert(!snd_mask_empty(mask));
    200 	for (i = 0; i < MASK_SIZE; i++) {
    201 		if (! mask->bits[i])
    202 			continue;
    203 		if (mask->bits[i] & (mask->bits[i] - 1))
    204 			return 0;
    205 		if (c)
    206 			return 0;
    207 		c++;
    208 	}
    209 	return 1;
    210 }
    211 
    212 MASK_INLINE int snd_mask_refine(snd_mask_t *mask, const snd_mask_t *v)
    213 {
    214 	snd_mask_t old;
    215 	if (snd_mask_empty(mask))
    216 		return -ENOENT;
    217 	snd_mask_copy(&old, mask);
    218 	snd_mask_intersect(mask, v);
    219 	if (snd_mask_empty(mask))
    220 		return -EINVAL;
    221 	return !snd_mask_eq(mask, &old);
    222 }
    223 
    224 MASK_INLINE int snd_mask_refine_first(snd_mask_t *mask)
    225 {
    226 	if (snd_mask_empty(mask))
    227 		return -ENOENT;
    228 	if (snd_mask_single(mask))
    229 		return 0;
    230 	snd_mask_leave(mask, snd_mask_min(mask));
    231 	return 1;
    232 }
    233 
    234 MASK_INLINE int snd_mask_refine_last(snd_mask_t *mask)
    235 {
    236 	if (snd_mask_empty(mask))
    237 		return -ENOENT;
    238 	if (snd_mask_single(mask))
    239 		return 0;
    240 	snd_mask_leave(mask, snd_mask_max(mask));
    241 	return 1;
    242 }
    243 
    244 MASK_INLINE int snd_mask_refine_min(snd_mask_t *mask, unsigned int val)
    245 {
    246 	if (snd_mask_empty(mask))
    247 		return -ENOENT;
    248 	if (snd_mask_min(mask) >= val)
    249 		return 0;
    250 	snd_mask_reset_range(mask, 0, val - 1);
    251 	if (snd_mask_empty(mask))
    252 		return -EINVAL;
    253 	return 1;
    254 }
    255 
    256 MASK_INLINE int snd_mask_refine_max(snd_mask_t *mask, unsigned int val)
    257 {
    258 	if (snd_mask_empty(mask))
    259 		return -ENOENT;
    260 	if (snd_mask_max(mask) <= val)
    261 		return 0;
    262 	snd_mask_reset_range(mask, val + 1, SND_MASK_MAX);
    263 	if (snd_mask_empty(mask))
    264 		return -EINVAL;
    265 	return 1;
    266 }
    267 
    268 MASK_INLINE int snd_mask_refine_set(snd_mask_t *mask, unsigned int val)
    269 {
    270 	int changed;
    271 	if (snd_mask_empty(mask))
    272 		return -ENOENT;
    273 	changed = !snd_mask_single(mask);
    274 	snd_mask_leave(mask, val);
    275 	if (snd_mask_empty(mask))
    276 		return -EINVAL;
    277 	return changed;
    278 }
    279 
    280 MASK_INLINE int snd_mask_value(const snd_mask_t *mask)
    281 {
    282 	assert(!snd_mask_empty(mask));
    283 	return snd_mask_min(mask);
    284 }
    285 
    286 MASK_INLINE int snd_mask_always_eq(const snd_mask_t *m1, const snd_mask_t *m2)
    287 {
    288 	return snd_mask_single(m1) && snd_mask_single(m2) &&
    289 		snd_mask_value(m1) == snd_mask_value(m2);
    290 }
    291 
    292 MASK_INLINE int snd_mask_never_eq(const snd_mask_t *m1, const snd_mask_t *m2)
    293 {
    294 	int i;
    295 	for (i = 0; i < MASK_SIZE; i++)
    296 		if (m1->bits[i] & m2->bits[i])
    297 			return 0;
    298 	return 1;
    299 }
    300