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
[];
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 struct idset
*idset_sch_new(void)
64 return idset_new(max_ssid
+ 1, __MAX_SUBCHANNEL
+ 1);
67 void idset_sch_add(struct idset
*set
, struct subchannel_id schid
)
69 idset_add(set
, schid
.ssid
, schid
.sch_no
);
72 void idset_sch_del(struct idset
*set
, struct subchannel_id schid
)
74 idset_del(set
, schid
.ssid
, schid
.sch_no
);
77 /* Clear ids starting from @schid up to end of subchannel set. */
78 void idset_sch_del_subseq(struct idset
*set
, struct subchannel_id schid
)
80 int pos
= schid
.ssid
* set
->num_id
+ schid
.sch_no
;
82 bitmap_clear(set
->bitmap
, pos
, set
->num_id
- schid
.sch_no
);
85 int idset_sch_contains(struct idset
*set
, struct subchannel_id schid
)
87 return idset_contains(set
, schid
.ssid
, schid
.sch_no
);
90 int idset_is_empty(struct idset
*set
)
92 return bitmap_empty(set
->bitmap
, set
->num_ssid
* set
->num_id
);
95 void idset_add_set(struct idset
*to
, struct idset
*from
)
97 int len
= min(to
->num_ssid
* to
->num_id
, from
->num_ssid
* from
->num_id
);
99 bitmap_or(to
->bitmap
, to
->bitmap
, from
->bitmap
, len
);