Fix findbugs warnings
[bgpcep.git] / bgp / path-selection-mode / src / main / java / org / opendaylight / protocol / bgp / mode / impl / base / OffsetMap.java
1 /*
2  * Copyright (c) 2015 Cisco Systems, Inc. and others.  All rights reserved.
3  *
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
7  */
8 package org.opendaylight.protocol.bgp.mode.impl.base;
9
10 import com.google.common.base.Preconditions;
11 import com.google.common.cache.CacheBuilder;
12 import com.google.common.cache.CacheLoader;
13 import com.google.common.cache.LoadingCache;
14 import com.google.common.collect.ImmutableSet;
15 import com.google.common.collect.ImmutableSet.Builder;
16 import com.google.common.primitives.UnsignedInteger;
17 import java.lang.reflect.Array;
18 import java.util.Arrays;
19 import java.util.Collections;
20 import java.util.Comparator;
21 import java.util.Set;
22 import javax.annotation.Nonnull;
23 import org.slf4j.Logger;
24 import org.slf4j.LoggerFactory;
25
26 /**
27  * A map of Router identifier to an offset. Used to maintain a simple
28  * offset-based lookup across multiple RouteEntry objects,
29  * which share either contributors or consumers.
30  * We also provide utility reformat methods, which provide access to
31  * array members and array management features.
32  */
33 final class OffsetMap {
34     private static final Logger LOG = LoggerFactory.getLogger(OffsetMap.class);
35     private static final String NEGATIVEOFFSET = "Invalid negative offset %s";
36     private static final String INVALIDOFFSET = "Invalid offset %s for %s router IDs";
37     private static final LoadingCache<Set<UnsignedInteger>, OffsetMap> OFFSETMAPS = CacheBuilder.newBuilder()
38             .weakValues().build(new CacheLoader<Set<UnsignedInteger>, OffsetMap>() {
39                 @Override
40                 public OffsetMap load(@Nonnull final Set<UnsignedInteger> key) {
41                     return new OffsetMap(key);
42                 }
43             });
44     private static final Comparator<UnsignedInteger> COMPARATOR = UnsignedInteger::compareTo;
45     private static final UnsignedInteger[] EMPTY_KEYS = new UnsignedInteger[0];
46     static final OffsetMap EMPTY = new OffsetMap(Collections.emptySet());
47     private final UnsignedInteger[] routeKeys;
48
49     private OffsetMap(final Set<UnsignedInteger> routerIds) {
50         final UnsignedInteger[] array = routerIds.toArray(EMPTY_KEYS);
51         Arrays.sort(array, COMPARATOR);
52         this.routeKeys = array;
53     }
54
55     UnsignedInteger getRouterKey(final int offset) {
56         Preconditions.checkArgument(offset >= 0);
57         return this.routeKeys[offset];
58     }
59
60     public int offsetOf(final UnsignedInteger key) {
61         return Arrays.binarySearch(this.routeKeys, key, COMPARATOR);
62     }
63
64     public int size() {
65         return this.routeKeys.length;
66     }
67
68     public OffsetMap with(final UnsignedInteger key) {
69         // TODO: we could make this faster if we had an array-backed Set and requiring
70         //       the caller to give us the result of offsetOf() -- as that indicates
71         //       where to insert the new routerId while maintaining the sorted nature
72         //       of the array
73         final Builder<UnsignedInteger> builder = ImmutableSet.builder();
74         builder.add(this.routeKeys);
75         builder.add(key);
76
77         return OFFSETMAPS.getUnchecked(builder.build());
78     }
79
80     public OffsetMap without(final UnsignedInteger key) {
81         final Builder<UnsignedInteger> builder = ImmutableSet.builder();
82         final int index = indexOfRouterId(key);
83         if (index < 0) {
84             LOG.trace("Router key {} not found", key);
85         } else {
86             builder.add(removeValue(this.routeKeys, index, EMPTY_KEYS));
87         }
88         return OFFSETMAPS.getUnchecked(builder.build());
89     }
90
91     private int indexOfRouterId(final UnsignedInteger key) {
92         for (int i = 0; i < this.routeKeys.length; i++) {
93             if (key.equals(this.routeKeys[i])) {
94                 return i;
95             }
96         }
97         return -1;
98     }
99
100     public <T> T getValue(final T[] array, final int offset) {
101         Preconditions.checkArgument(offset >= 0, NEGATIVEOFFSET, offset);
102         Preconditions.checkArgument(offset < this.routeKeys.length, INVALIDOFFSET, offset, this.routeKeys.length);
103         return array[offset];
104     }
105
106     public <T> void setValue(final T[] array, final int offset, final T value) {
107         Preconditions.checkArgument(offset >= 0, NEGATIVEOFFSET, offset);
108         Preconditions.checkArgument(offset < this.routeKeys.length, INVALIDOFFSET, offset, this.routeKeys.length);
109         array[offset] = value;
110     }
111
112     <T> T[] expand(final OffsetMap oldOffsets, final T[] oldArray, final int offset) {
113         @SuppressWarnings("unchecked")
114         final T[] ret = (T[]) Array.newInstance(oldArray.getClass().getComponentType(), this.routeKeys.length);
115         final int oldSize = oldOffsets.routeKeys.length;
116
117         System.arraycopy(oldArray, 0, ret, 0, offset);
118         System.arraycopy(oldArray, offset, ret, offset + 1, oldSize - offset);
119
120         return ret;
121     }
122
123     public <T> T[] removeValue(final T[] oldArray, final int offset, final T[] emptyArray) {
124         final int length = oldArray.length;
125         Preconditions.checkArgument(offset >= 0, NEGATIVEOFFSET, offset);
126         Preconditions.checkArgument(offset < this.routeKeys.length, INVALIDOFFSET, offset, length);
127
128         final int newLength = length - 1;
129         if (newLength == 0) {
130             Preconditions.checkArgument(emptyArray.length == 0);
131             return emptyArray;
132         }
133
134         final T[] ret = (T[]) Array.newInstance(oldArray.getClass().getComponentType(), newLength);
135         System.arraycopy(oldArray, 0, ret, 0, offset);
136         if (offset < newLength) {
137             System.arraycopy(oldArray, offset + 1, ret, offset, newLength - offset);
138         }
139
140         return ret;
141     }
142
143     boolean isEmpty() {
144         return this.size() == 0;
145     }
146 }