1 // SPDX-License-Identifier: GPL-2.0
3 * Copyright IBM Corp. 2007, 2012
4 * Author(s): Peter Oberparleiter <peter.oberparleiter@de.ibm.com>
7 #include <linux/vmalloc.h>
8 #include <linux/bitmap.h>
9 #include <linux/bitops.h>
16 unsigned long bitmap
[0];
19 static inline unsigned long bitmap_size(int num_ssid
, int num_id
)
21 return BITS_TO_LONGS(num_ssid
* num_id
) * sizeof(unsigned long);
24 static struct idset
*idset_new(int num_ssid
, int num_id
)
28 set
= vmalloc(sizeof(struct idset
) + bitmap_size(num_ssid
, num_id
));
30 set
->num_ssid
= num_ssid
;
32 memset(set
->bitmap
, 0, bitmap_size(num_ssid
, num_id
));
37 void idset_free(struct idset
*set
)
42 void idset_fill(struct idset
*set
)
44 memset(set
->bitmap
, 0xff, bitmap_size(set
->num_ssid
, set
->num_id
));
47 static inline void idset_add(struct idset
*set
, int ssid
, int id
)
49 set_bit(ssid
* set
->num_id
+ id
, set
->bitmap
);
52 static inline void idset_del(struct idset
*set
, int ssid
, int id
)
54 clear_bit(ssid
* set
->num_id
+ id
, set
->bitmap
);
57 static inline int idset_contains(struct idset
*set
, int ssid
, int id
)
59 return test_bit(ssid
* set
->num_id
+ id
, set
->bitmap
);
62 static inline int idset_get_first(struct idset
*set
, int *ssid
, int *id
)
66 bitnum
= find_first_bit(set
->bitmap
, set
->num_ssid
* set
->num_id
);
67 if (bitnum
>= set
->num_ssid
* set
->num_id
)
69 *ssid
= bitnum
/ set
->num_id
;
70 *id
= bitnum
% set
->num_id
;
74 struct idset
*idset_sch_new(void)
76 return idset_new(max_ssid
+ 1, __MAX_SUBCHANNEL
+ 1);
79 void idset_sch_add(struct idset
*set
, struct subchannel_id schid
)
81 idset_add(set
, schid
.ssid
, schid
.sch_no
);
84 void idset_sch_del(struct idset
*set
, struct subchannel_id schid
)
86 idset_del(set
, schid
.ssid
, schid
.sch_no
);
89 /* Clear ids starting from @schid up to end of subchannel set. */
90 void idset_sch_del_subseq(struct idset
*set
, struct subchannel_id schid
)
92 int pos
= schid
.ssid
* set
->num_id
+ schid
.sch_no
;
94 bitmap_clear(set
->bitmap
, pos
, set
->num_id
- schid
.sch_no
);
97 int idset_sch_contains(struct idset
*set
, struct subchannel_id schid
)
99 return idset_contains(set
, schid
.ssid
, schid
.sch_no
);
102 int idset_is_empty(struct idset
*set
)
104 return bitmap_empty(set
->bitmap
, set
->num_ssid
* set
->num_id
);
107 void idset_add_set(struct idset
*to
, struct idset
*from
)
109 int len
= min(to
->num_ssid
* to
->num_id
, from
->num_ssid
* from
->num_id
);
111 bitmap_or(to
->bitmap
, to
->bitmap
, from
->bitmap
, len
);