2 Copyright (C) 2002 Free Software Foundation, Inc.
3 Contributed by Michael Hayes (m.hayes@elec.canterbury.ac.nz).
5 This program is free software; you can redistribute it and/or modify
6 it under the terms of the GNU General Public License as published by
7 the Free Software Foundation; either version 2 of the License, or
8 (at your option) any later version.
10 This program is distributed in the hope that it will be useful,
11 but WITHOUT ANY WARRANTY; without even the implied warranty of
12 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
13 GNU General Public License for more details.
15 You should have received a copy of the GNU General Public License
16 along with this program; if not, write to the Free Software
17 Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. */
30 static void bitset_print
PARAMS ((FILE *, bitset
, int));
33 #define BITSET_STATS_FILE "bitset.dat"
35 #define BITSET_LOG_COUNT_BINS 10
36 #define BITSET_LOG_SIZE_BINS 16
37 #define BITSET_DENSITY_BINS 20
39 struct bitset_type_stats_struct
41 unsigned int xmallocs
;
43 unsigned int oballocs
;
46 unsigned int list_counts
[BITSET_LOG_COUNT_BINS
];
47 unsigned int list_sizes
[BITSET_LOG_SIZE_BINS
];
48 unsigned int list_density
[BITSET_DENSITY_BINS
];
51 struct bitset_stats_struct
54 struct bitset_type_stats_struct types
[BITSET_TYPE_NUM
];
57 struct bitset_stats_struct bitset_stats_data
;
58 struct bitset_stats_struct
*bitset_stats
;
60 static void bitset_percent_histogram_print
PARAMS ((FILE *, const char *,
64 static void bitset_log_histogram_print
PARAMS ((FILE *, const char *,
68 static void bitset_stats_print_1
69 PARAMS ((FILE *, const char *, struct bitset_type_stats_struct
*));
70 static void bitset_stats_print
PARAMS ((FILE *, int));
71 static void bitset_stats_read
PARAMS ((void));
72 static void bitset_stats_write
PARAMS ((void));
74 #define BITSET_STATS_XMALLOCS_INC(TYPE) \
76 bitset_stats->types[(TYPE)].xmallocs++
78 #define BITSET_STATS_XFREES_INC(BSET) \
80 bitset_stats->types[BITSET_TYPE_ (BSET)].xfrees++
82 #define BITSET_STATS_OBALLOCS_INC(TYPE) \
84 bitset_stats->types[(TYPE)].oballocs++
86 #define BITSET_STATS_OBFREES_INC(BSET) \
88 bitset_stats->types[BITSET_TYPE_ (BSET)].obfrees++
90 #define BITSET_STATS_LISTS_INC(BSET) \
92 bitset_stats->types[BITSET_TYPE_ (BSET)].lists++
94 #define BITSET_STATS_LIST_COUNTS_INC(BSET, I) \
96 bitset_stats->types[BITSET_TYPE_ (BSET)].list_counts[(I)]++
98 #define BITSET_STATS_LIST_SIZES_INC(BSET, I) \
100 bitset_stats->types[BITSET_TYPE_ (BSET)].list_sizes[(I)]++
102 #define BITSET_STATS_LIST_DENSITY_INC(BSET, I) \
104 bitset_stats->types[BITSET_TYPE_ (BSET)].list_density[(I)]++
107 #define BITSET_STATS_XMALLOCS_INC(TYPE)
109 #define BITSET_STATS_XFREES_INC(BSET)
111 #define BITSET_STATS_OBALLOCS_INC(TYPE)
113 #define BITSET_STATS_OBFREES_INC(BSET)
115 #define BITSET_STATS_LISTS_INC(BSET)
117 #define BITSET_STATS_LIST_COUNTS_INC(BSET, I)
119 #define BITSET_STATS_LIST_SIZES_INC(BSET, I)
121 #define BITSET_STATS_LIST_DENSITY_INC(BSET, I)
122 #endif /* BITSET_STATS */
125 /* Return number of bytes required to create a N_BIT bitset
126 of TYPE. The bitset may grow to require more bytes than this. */
128 bitset_bytes (type
, n_bits
)
129 enum bitset_type type
;
130 bitset_bindex n_bits
;
137 bytes
= sbitset_bytes (n_bits
);
141 bytes
= lbitset_bytes (n_bits
);
145 bytes
= ebitset_bytes (n_bits
);
156 /* Initialise bitset BSET of TYPE for N_BITS. */
158 bitset_init (bset
, n_bits
, type
)
160 bitset_bindex n_bits
;
161 enum bitset_type type
;
166 return sbitset_init (bset
, n_bits
);
169 return lbitset_init (bset
, n_bits
);
172 return ebitset_init (bset
, n_bits
);
180 /* Select a bitset type for a set of N_BITS and with attribute hints
181 specified by ATTR. For variable size bitsets, N_BITS is only a
182 hint and may be zero. */
184 bitset_type_choose (n_bits
, attr
)
185 bitset_bindex n_bits ATTRIBUTE_UNUSED
;
188 enum bitset_type type
;
191 /* Check attributes. */
192 if (attr
& BITSET_FIXED
&& attr
& BITSET_VARIABLE
)
194 if (attr
& BITSET_SPARSE
&& attr
& BITSET_DENSE
)
197 /* Note that sometimes we will be asked for a zero length
198 fixed size bitset. */
201 /* Choose the type of bitset. */
204 /* Currently, the simple bitsets do not support a variable size. */
205 if (attr
& BITSET_VARIABLE
|| attr
& BITSET_SPARSE
)
208 if (attr
& BITSET_DENSE
|| attr
& BITSET_GREEDY
)
216 /* Create a bitset of N_BITS of type TYPE. */
218 bitset_alloc (n_bits
, type
)
219 bitset_bindex n_bits
;
220 enum bitset_type type
;
225 BITSET_STATS_XMALLOCS_INC (type
);
227 bytes
= bitset_bytes (type
, n_bits
);
229 bset
= (bitset
) xcalloc (1, bytes
);
231 /* The cache is disabled until some elements are allocated. If we
232 have variable length arrays, then we may need to allocate dummy
235 return bitset_init (bset
, n_bits
, type
);
239 /* Create a bitset of N_BITS of type TYPE. */
241 bitset_obstack_alloc (bobstack
, n_bits
, type
)
242 struct obstack
*bobstack
;
243 bitset_bindex n_bits
;
244 enum bitset_type type
;
249 BITSET_STATS_OBALLOCS_INC (type
);
251 bytes
= bitset_bytes (type
, n_bits
);
253 bset
= obstack_alloc (bobstack
, bytes
);
254 memset (bset
, 0, bytes
);
256 return bitset_init (bset
, n_bits
, type
);
260 /* Create a bitset of N_BITS and with attribute hints specified by
263 bitset_create (n_bits
, attr
)
264 bitset_bindex n_bits
;
267 enum bitset_type type
;
269 type
= bitset_type_choose (n_bits
, attr
);
271 return bitset_alloc (n_bits
, type
);
275 /* Free bitset BSET. */
280 BITSET_STATS_XFREES_INC (bset
);
287 /* Free bitset BSET allocated on obstack. */
289 bitset_obstack_free (bset
)
292 BITSET_STATS_OBFREES_INC (bset
);
298 /* Find next bit set in SRC starting from and including BITNO.
299 Return -1 if SRC empty. */
301 bitset_next (src
, bitno
)
306 bitset_bindex next
= bitno
;
308 if (!bitset_list (src
, &val
, 1, &next
))
314 /* Find previous bit set in SRC starting from and including BITNO.
315 Return -1 if SRC empty. */
317 bitset_prev (src
, bitno
)
322 bitset_bindex next
= bitno
;
324 if (!bitset_reverse_list (src
, &val
, 1, &next
))
330 /* Find first set bit. */
335 return bitset_next (src
, 0);
339 /* Find last set bit. */
344 return bitset_prev (src
, 0);
348 /* Return non-zero if BITNO in SRC is the only set bit. */
350 bitset_only_set_p (src
, bitno
)
354 bitset_bindex val
[2];
355 bitset_bindex next
= 0;
357 if (bitset_list (src
, val
, 2, &next
) != 1)
359 return val
[0] == bitno
;
363 /* Toggle bit BITNO in bitset BSET and return non-zero if now set. */
365 bitset_toggle (bset
, bitno
)
369 /* This routine is for completeness. It could be optimized if
371 if (bitset_test (bset
, bitno
))
373 bitset_reset (bset
, bitno
);
378 bitset_set (bset
, bitno
);
384 /* Print contents of bitset BSET to FILE. */
386 bitset_print (file
, bset
, verbose
)
394 fprintf (file
, "n_bits = %d, set = {", bitset_size (bset
));
397 BITSET_EXECUTE (bset
, 0, i
,
401 fprintf (file
, "\n");
405 fprintf (file
, "%d ", i
);
406 pos
+= 1 + (i
>= 10) + (i
>= 100);
410 fprintf (file
, "}\n");
414 /* DST = SRC. Return non-zero if DST != SRC. */
416 bitset_copy (dst
, src
)
422 if (BITSET_COMPATIBLE_ (dst
, src
))
423 return BITSET_COPY_ (dst
, src
);
425 /* Convert bitset types. We assume that the DST bitset
426 is large enough to hold the SRC bitset. */
428 BITSET_EXECUTE (src
, 0, i
,
437 /* Return size in bits of bitset SRC. */
442 return BITSET_SIZE_ (src
);
446 /* Return number of bits set in bitset SRC. */
451 bitset_bindex list
[BITSET_LIST_SIZE
];
457 for (count
= 0; (num
= bitset_list (src
, list
, BITSET_LIST_SIZE
, &next
));
470 return BITSET_ZERO_ (dst
);
479 return BITSET_ONES_ (dst
);
483 /* Return SRC == 0. */
488 return BITSET_EMPTY_P_ (src
);
492 /* Return DST == DST | SRC. */
494 bitset_subset_p (dst
, src
)
498 BITSET_CHECK2_ (dst
, src
);
499 return BITSET_SUBSET_P_ (dst
, src
);
503 /* Return DST == SRC. */
505 bitset_equal_p (dst
, src
)
509 BITSET_CHECK2_ (dst
, src
);
510 return BITSET_EQUAL_P_ (dst
, src
);
514 /* Return DST & SRC == 0. */
516 bitset_disjoint_p (dst
, src
)
520 BITSET_CHECK2_ (dst
, src
);
521 return BITSET_DISJOINT_P_ (dst
, src
);
527 bitset_not (dst
, src
)
531 BITSET_CHECK2_ (dst
, src
);
532 return BITSET_NOT_ (dst
, src
);
536 /* DST = SRC1 | SRC2. Return non-zero if DST != SRC1 | SRC2. */
538 bitset_or (dst
, src1
, src2
)
543 BITSET_CHECK3_ (dst
, src1
, src2
);
544 return BITSET_OR_ (dst
, src1
, src2
);
548 /* DST = SRC1 & SRC2. Return non-zero if DST != SRC1 & SRC2. */
550 bitset_and (dst
, src1
, src2
)
555 BITSET_CHECK3_ (dst
, src1
, src2
);
556 return BITSET_AND_ (dst
, src1
, src2
);
560 /* DST = SRC1 ^ SRC2. Return non-zero if DST != SRC1 ^ SRC2. */
562 bitset_xor (dst
, src1
, src2
)
567 BITSET_CHECK3_ (dst
, src1
, src2
);
568 return BITSET_XOR_ (dst
, src1
, src2
);
572 /* DST = SRC1 & ~SRC2. Return non-zero if DST != SRC1 & ~SRC2. */
574 bitset_andn (dst
, src1
, src2
)
579 BITSET_CHECK3_ (dst
, src1
, src2
);
580 return BITSET_ANDN_ (dst
, src1
, src2
);
585 bitset_op4 (dst
, src1
, src2
, src3
, op
)
595 /* Create temporary bitset. */
596 tmp
= bitset_alloc (0, BITSET_TYPE_ (dst
));
600 case BITSET_OP_OR_AND
:
601 BITSET_OR_ (tmp
, src1
, src2
);
602 changed
= BITSET_AND_ (dst
, src3
, tmp
);
605 case BITSET_OP_AND_OR
:
606 BITSET_AND_ (tmp
, src1
, src2
);
607 changed
= BITSET_OR_ (dst
, src3
, tmp
);
610 case BITSET_OP_ANDN_OR
:
611 BITSET_ANDN_ (tmp
, src1
, src2
);
612 changed
= BITSET_OR_ (dst
, src3
, tmp
);
624 /* DST = (SRC1 | SRC2) & SRC3. Return non-zero if
625 DST != (SRC1 | SRC2) & SRC3. */
627 bitset_or_and (dst
, src1
, src2
, src3
)
633 BITSET_CHECK4_ (dst
, src1
, src2
, src3
);
634 return BITSET_OR_AND_ (dst
, src1
, src2
, src3
);
638 /* DST = (SRC1 & SRC2) | SRC3. Return non-zero if
639 DST != (SRC1 & SRC2) | SRC3. */
641 bitset_and_or (dst
, src1
, src2
, src3
)
647 BITSET_CHECK4_ (dst
, src1
, src2
, src3
);
648 return BITSET_AND_OR_ (dst
, src1
, src2
, src3
);
652 /* DST = (SRC1 & ~SRC2) | SRC3. Return non-zero if
653 DST != (SRC1 & ~SRC2) | SRC3. */
655 bitset_andn_or (dst
, src1
, src2
, src3
)
661 BITSET_CHECK4_ (dst
, src1
, src2
, src3
);
662 return BITSET_ANDN_OR_ (dst
, src1
, src2
, src3
);
666 /* Dump bitset BSET to FILE. */
668 bitset_dump (file
, bset
)
672 bitset_print (file
, bset
, 0);
676 /* Function to be called from debugger to print bitset. */
682 bitset_print (stderr
, bset
, 1);
686 /* Release memory associated with bitsets. */
688 bitset_release_memory ()
690 lbitset_release_memory ();
691 ebitset_release_memory ();
697 bitset_list (bset
, list
, num
, next
)
705 count
= BITSET_LIST_ (bset
, list
, num
, next
);
712 enum bitset_type type
;
714 type
= BITSET_TYPE_ (bset
);
715 BITSET_STATS_LISTS_INC (bset
);
717 /* Log histogram of number of set bits. */
718 for (i
= 0, tmp
= count
; tmp
; tmp
>>= 1, i
++)
720 if (i
>= BITSET_LOG_COUNT_BINS
)
721 i
= BITSET_LOG_COUNT_BINS
- 1;
722 BITSET_STATS_LIST_COUNTS_INC (bset
, i
);
724 /* Log histogram of number of bits in set. */
725 size
= bitset_size (bset
);
726 for (i
= 0, tmp
= size
; tmp
; tmp
>>= 1, i
++)
728 if (i
>= BITSET_LOG_SIZE_BINS
)
729 i
= BITSET_LOG_SIZE_BINS
- 1;
730 BITSET_STATS_LIST_SIZES_INC (bset
, i
);
732 /* Histogram of fraction of bits set. */
733 i
= size
? (count
* BITSET_DENSITY_BINS
) / size
: 0;
734 if (i
>= BITSET_DENSITY_BINS
)
735 i
= BITSET_DENSITY_BINS
- 1;
736 BITSET_STATS_LIST_DENSITY_INC (bset
, i
);
742 /* Print a percentage histogram with message MSG to FILE. */
744 bitset_percent_histogram_print (file
, name
, msg
, n_bins
, bins
)
755 for (i
= 0; i
< n_bins
; i
++)
761 fprintf (file
, "%s %s", name
, msg
);
762 for (i
= 0; i
< n_bins
; i
++)
763 fprintf (file
, "%.0f-%.0f%%\t%8d (%5.1f%%)\n",
765 (i
+ 1) * 100.0 / n_bins
, bins
[i
],
766 (100.0 * bins
[i
]) / total
);
770 /* Print a log histogram with message MSG to FILE. */
772 bitset_log_histogram_print (file
, name
, msg
, n_bins
, bins
)
781 unsigned int max_width
;
784 for (i
= 0; i
< n_bins
; i
++)
790 /* 2 * ceil (log10(2) * (N - 1)) + 1 */
791 max_width
= 2 * (unsigned int) (0.30103 * (n_bins
- 1) + 0.9999) + 1;
793 fprintf (file
, "%s %s", name
, msg
);
794 for (i
= 0; i
< 2; i
++)
795 fprintf (file
, "%*d\t%8d (%5.1f%%)\n",
796 max_width
, i
, bins
[i
], 100.0 * bins
[i
] / total
);
798 /* Perhaps we should bail out once the histogram goes to zero. */
799 for (; i
< n_bins
; i
++)
800 fprintf (file
, "%*d-%d\t%8d (%5.1f%%)\n",
801 max_width
- ((unsigned int) (0.30103 * (i
) + 0.9999) + 1),
802 1 << (i
- 1), (1 << i
) - 1, bins
[i
],
803 (100.0 * bins
[i
]) / total
);
807 /* Print bitset statistics to FILE. */
809 bitset_stats_print_1 (file
, name
, stats
)
812 struct bitset_type_stats_struct
*stats
;
817 fprintf (file
, "%d %ss xmalloced, %d freed.\n",
818 stats
->xmallocs
, name
, stats
->xfrees
);
819 fprintf (file
, "%d %ss oballoced, %d freed.\n",
820 stats
->oballocs
, name
, stats
->obfrees
);
822 fprintf (file
, "%d bitset_lists\n", stats
->lists
);
824 bitset_log_histogram_print (file
, name
, "count log histogram\n",
825 BITSET_LOG_COUNT_BINS
, stats
->list_counts
);
827 bitset_log_histogram_print (file
, name
, "size log histogram\n",
828 BITSET_LOG_SIZE_BINS
, stats
->list_sizes
);
830 bitset_percent_histogram_print (file
, name
, "density histogram\n",
831 BITSET_DENSITY_BINS
, stats
->list_density
);
835 /* Print all bitset statistics to FILE. */
837 bitset_stats_print (file
, verbose
)
839 int verbose ATTRIBUTE_UNUSED
;
842 static const char *names
[] = BITSET_TYPE_NAMES
;
847 fprintf (file
, "Bitset statistics:\n\n");
849 if (bitset_stats
->runs
> 1)
850 fprintf (file
, "Accumulated runs = %d\n", bitset_stats
->runs
);
852 for (i
= 0; i
< BITSET_TYPE_NUM
; i
++)
853 bitset_stats_print_1 (file
, names
[i
], &bitset_stats
->types
[i
]);
855 #endif /* BITSET_STATS */
858 /* Initialise bitset statistics logging. */
863 bitset_stats
= &bitset_stats_data
;
864 bitset_stats_read ();
865 #endif /* BITSET_STATS */
869 /* Read bitset statistics file. */
878 file
= fopen (BITSET_STATS_FILE
, "r");
881 if (fread (&bitset_stats_data
, sizeof (bitset_stats_data
),
885 perror ("Could not read stats file.");
887 fprintf (stderr
, "Bad stats file size.\n");
891 bitset_stats_data
.runs
++;
895 /* Write bitset statistics file. */
897 bitset_stats_write ()
904 file
= fopen (BITSET_STATS_FILE
, "w");
907 if (fwrite (&bitset_stats_data
, sizeof (bitset_stats_data
),
909 perror ("Could not write stats file.");
913 perror ("Could not open stats file for writing.");
917 /* Dump bitset statistics to FILE. */
919 bitset_stats_dump (file
)
923 bitset_stats_print (file
, 0);
924 bitset_stats_write ();
925 #endif /* BITSET_STATS */
929 /* Function to be called from debugger to print bitset stats. */
931 debug_bitset_stats (void)
934 bitset_stats_print (stderr
, 1);
935 #endif /* BITSET_STATS */