From 72a9d65a2cda264efe77e0ad7b752d84777623a9 Mon Sep 17 00:00:00 2001 From: Paul Sandoz Date: Mon, 18 Jun 2018 10:13:58 -0700 Subject: [PATCH] 8170159: Improve the performance of BitSet traversal Reviewed-by: martin --- .../share/classes/java/util/BitSet.java | 31 ++++++++++++++++--- 1 file changed, 26 insertions(+), 5 deletions(-) diff --git a/src/java.base/share/classes/java/util/BitSet.java b/src/java.base/share/classes/java/util/BitSet.java index c9dafaed3e8..c72970def04 100644 --- a/src/java.base/share/classes/java/util/BitSet.java +++ b/src/java.base/share/classes/java/util/BitSet.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 1995, 2016, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 1995, 2018, Oracle and/or its affiliates. All rights reserved. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * * This code is free software; you can redistribute it and/or modify it @@ -1277,12 +1277,33 @@ public class BitSet implements Cloneable, java.io.Serializable { int hi = getFence(); int i = index; - int v = wordIndex(hi - 1); index = -1; - while (i >= 0 && i < hi) { - action.accept(i); - i = nextSetBit(i + 1, v); + + if (i >= 0 && i < hi) { + action.accept(i++); + + int u = wordIndex(i); // next lower word bound + int v = wordIndex(hi - 1); // upper word bound + + words_loop: + for (; u <= v && i <= hi; u++, i = u << ADDRESS_BITS_PER_WORD) { + long word = words[u] & (WORD_MASK << i); + while (word != 0) { + i = (u << ADDRESS_BITS_PER_WORD) + Long.numberOfTrailingZeros(word); + if (i >= hi) { + // Break out of outer loop to ensure check of + // Integer.MAX_VALUE bit set + break words_loop; + } + + // Flip the set bit + word &= ~(1L << i); + + action.accept(i); + } + } } + // Check if there is a final bit set for Integer.MAX_VALUE if (i == Integer.MAX_VALUE && hi == Integer.MAX_VALUE) { action.accept(Integer.MAX_VALUE);