2 * Copyright (c) 2021 PANTHEON.tech, s.r.o. and others. All rights reserved.
4 * This program and the accompanying materials are made available under the
5 * terms of the Eclipse Public License v1.0 which accompanies this distribution,
6 * and is available at http://www.eclipse.org/legal/epl-v10.html
8 package org.opendaylight.controller.cluster.datastore.utils;
10 import com.google.common.annotations.Beta;
11 import com.google.common.collect.Collections2;
12 import com.google.common.collect.ImmutableRangeSet;
13 import com.google.common.primitives.UnsignedLong;
14 import java.util.TreeSet;
15 import org.eclipse.jdt.annotation.NonNull;
16 import org.opendaylight.yangtools.concepts.Mutable;
19 public final class MutableUnsignedLongSet extends UnsignedLongSet implements Mutable {
20 MutableUnsignedLongSet(final TreeSet<Entry> ranges) {
24 public static @NonNull MutableUnsignedLongSet of() {
25 return new MutableUnsignedLongSet(new TreeSet<>());
29 public ImmutableUnsignedLongSet immutableCopy() {
30 return ImmutableUnsignedLongSet.copyOf(this);
33 public void add(final long longBits) {
34 final var ranges = trustedRanges();
35 final var range = Entry.of(longBits);
37 // We need Iterator.remove() to perform efficient merge below
38 final var headIt = ranges.headSet(range, true).descendingIterator();
39 if (headIt.hasNext()) {
40 final var head = headIt.next();
41 if (Long.compareUnsigned(head.upperBits, longBits) >= 0) {
42 // Already contained, this is a no-op
46 // Merge into head entry if possible
47 if (head.upperBits + 1 == longBits) {
48 head.upperBits = longBits;
50 // Potentially merge head entry and tail entry
51 final var tail = ranges.higher(range);
53 if (tail.lowerBits - 1 == longBits) {
54 // Expand tail, remove head
55 tail.lowerBits = head.lowerBits;
63 final var tail = ranges.higher(range);
65 // Merge into tail entry if possible
66 if (tail.lowerBits - 1 == longBits) {
67 tail.lowerBits = longBits;
72 // No luck, store a new entry
76 public ImmutableRangeSet<UnsignedLong> toRangeSet() {
77 return ImmutableRangeSet.copyOf(Collections2.transform(trustedRanges(), Entry::toUnsigned));