1*03f9172cSAndroid Build Coastguard Worker /* 2*03f9172cSAndroid Build Coastguard Worker * Bitfield 3*03f9172cSAndroid Build Coastguard Worker * Copyright (c) 2013, Jouni Malinen <[email protected]> 4*03f9172cSAndroid Build Coastguard Worker * 5*03f9172cSAndroid Build Coastguard Worker * This software may be distributed under the terms of the BSD license. 6*03f9172cSAndroid Build Coastguard Worker * See README for more details. 7*03f9172cSAndroid Build Coastguard Worker */ 8*03f9172cSAndroid Build Coastguard Worker 9*03f9172cSAndroid Build Coastguard Worker #include "includes.h" 10*03f9172cSAndroid Build Coastguard Worker 11*03f9172cSAndroid Build Coastguard Worker #include "common.h" 12*03f9172cSAndroid Build Coastguard Worker #include "bitfield.h" 13*03f9172cSAndroid Build Coastguard Worker 14*03f9172cSAndroid Build Coastguard Worker 15*03f9172cSAndroid Build Coastguard Worker struct bitfield { 16*03f9172cSAndroid Build Coastguard Worker u8 *bits; 17*03f9172cSAndroid Build Coastguard Worker size_t max_bits; 18*03f9172cSAndroid Build Coastguard Worker }; 19*03f9172cSAndroid Build Coastguard Worker 20*03f9172cSAndroid Build Coastguard Worker bitfield_alloc(size_t max_bits)21*03f9172cSAndroid Build Coastguard Workerstruct bitfield * bitfield_alloc(size_t max_bits) 22*03f9172cSAndroid Build Coastguard Worker { 23*03f9172cSAndroid Build Coastguard Worker struct bitfield *bf; 24*03f9172cSAndroid Build Coastguard Worker 25*03f9172cSAndroid Build Coastguard Worker bf = os_zalloc(sizeof(*bf) + (max_bits + 7) / 8); 26*03f9172cSAndroid Build Coastguard Worker if (bf == NULL) 27*03f9172cSAndroid Build Coastguard Worker return NULL; 28*03f9172cSAndroid Build Coastguard Worker bf->bits = (u8 *) (bf + 1); 29*03f9172cSAndroid Build Coastguard Worker bf->max_bits = max_bits; 30*03f9172cSAndroid Build Coastguard Worker return bf; 31*03f9172cSAndroid Build Coastguard Worker } 32*03f9172cSAndroid Build Coastguard Worker 33*03f9172cSAndroid Build Coastguard Worker bitfield_free(struct bitfield * bf)34*03f9172cSAndroid Build Coastguard Workervoid bitfield_free(struct bitfield *bf) 35*03f9172cSAndroid Build Coastguard Worker { 36*03f9172cSAndroid Build Coastguard Worker os_free(bf); 37*03f9172cSAndroid Build Coastguard Worker } 38*03f9172cSAndroid Build Coastguard Worker 39*03f9172cSAndroid Build Coastguard Worker bitfield_set(struct bitfield * bf,size_t bit)40*03f9172cSAndroid Build Coastguard Workervoid bitfield_set(struct bitfield *bf, size_t bit) 41*03f9172cSAndroid Build Coastguard Worker { 42*03f9172cSAndroid Build Coastguard Worker if (bit >= bf->max_bits) 43*03f9172cSAndroid Build Coastguard Worker return; 44*03f9172cSAndroid Build Coastguard Worker bf->bits[bit / 8] |= BIT(bit % 8); 45*03f9172cSAndroid Build Coastguard Worker } 46*03f9172cSAndroid Build Coastguard Worker 47*03f9172cSAndroid Build Coastguard Worker bitfield_clear(struct bitfield * bf,size_t bit)48*03f9172cSAndroid Build Coastguard Workervoid bitfield_clear(struct bitfield *bf, size_t bit) 49*03f9172cSAndroid Build Coastguard Worker { 50*03f9172cSAndroid Build Coastguard Worker if (bit >= bf->max_bits) 51*03f9172cSAndroid Build Coastguard Worker return; 52*03f9172cSAndroid Build Coastguard Worker bf->bits[bit / 8] &= ~BIT(bit % 8); 53*03f9172cSAndroid Build Coastguard Worker } 54*03f9172cSAndroid Build Coastguard Worker 55*03f9172cSAndroid Build Coastguard Worker bitfield_is_set(struct bitfield * bf,size_t bit)56*03f9172cSAndroid Build Coastguard Workerint bitfield_is_set(struct bitfield *bf, size_t bit) 57*03f9172cSAndroid Build Coastguard Worker { 58*03f9172cSAndroid Build Coastguard Worker if (bit >= bf->max_bits) 59*03f9172cSAndroid Build Coastguard Worker return 0; 60*03f9172cSAndroid Build Coastguard Worker return !!(bf->bits[bit / 8] & BIT(bit % 8)); 61*03f9172cSAndroid Build Coastguard Worker } 62*03f9172cSAndroid Build Coastguard Worker 63*03f9172cSAndroid Build Coastguard Worker first_zero(u8 val)64*03f9172cSAndroid Build Coastguard Workerstatic int first_zero(u8 val) 65*03f9172cSAndroid Build Coastguard Worker { 66*03f9172cSAndroid Build Coastguard Worker int i; 67*03f9172cSAndroid Build Coastguard Worker for (i = 0; i < 8; i++) { 68*03f9172cSAndroid Build Coastguard Worker if (!(val & 0x01)) 69*03f9172cSAndroid Build Coastguard Worker return i; 70*03f9172cSAndroid Build Coastguard Worker val >>= 1; 71*03f9172cSAndroid Build Coastguard Worker } 72*03f9172cSAndroid Build Coastguard Worker return -1; 73*03f9172cSAndroid Build Coastguard Worker } 74*03f9172cSAndroid Build Coastguard Worker 75*03f9172cSAndroid Build Coastguard Worker bitfield_get_first_zero(struct bitfield * bf)76*03f9172cSAndroid Build Coastguard Workerint bitfield_get_first_zero(struct bitfield *bf) 77*03f9172cSAndroid Build Coastguard Worker { 78*03f9172cSAndroid Build Coastguard Worker size_t i; 79*03f9172cSAndroid Build Coastguard Worker for (i = 0; i < (bf->max_bits + 7) / 8; i++) { 80*03f9172cSAndroid Build Coastguard Worker if (bf->bits[i] != 0xff) 81*03f9172cSAndroid Build Coastguard Worker break; 82*03f9172cSAndroid Build Coastguard Worker } 83*03f9172cSAndroid Build Coastguard Worker if (i == (bf->max_bits + 7) / 8) 84*03f9172cSAndroid Build Coastguard Worker return -1; 85*03f9172cSAndroid Build Coastguard Worker i = i * 8 + first_zero(bf->bits[i]); 86*03f9172cSAndroid Build Coastguard Worker if (i >= bf->max_bits) 87*03f9172cSAndroid Build Coastguard Worker return -1; 88*03f9172cSAndroid Build Coastguard Worker return i; 89*03f9172cSAndroid Build Coastguard Worker } 90