mirror of
https://github.com/Suwayomi/Tachidesk.git
synced 2026-01-30 07:24:25 +01:00
android support! thanks to TachiWeb devs.
This commit is contained in:
854
AndroidCompat/src/main/java/android/util/ArrayMap.java
Normal file
854
AndroidCompat/src/main/java/android/util/ArrayMap.java
Normal file
@@ -0,0 +1,854 @@
|
||||
/*
|
||||
* Copyright (C) 2013 The Android Open Source Project
|
||||
*
|
||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||
* you may not use this file except in compliance with the License.
|
||||
* You may obtain a copy of the License at
|
||||
*
|
||||
* http://www.apache.org/licenses/LICENSE-2.0
|
||||
*
|
||||
* Unless required by applicable law or agreed to in writing, software
|
||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
* See the License for the specific language governing permissions and
|
||||
* limitations under the License.
|
||||
*/
|
||||
package android.util;
|
||||
import libcore.util.EmptyArray;
|
||||
import java.util.Collection;
|
||||
import java.util.ConcurrentModificationException;
|
||||
import java.util.Map;
|
||||
import java.util.Set;
|
||||
/**
|
||||
* ArrayMap is a generic key->value mapping data structure that is
|
||||
* designed to be more memory efficient than a traditional {@link java.util.HashMap}.
|
||||
* It keeps its mappings in an array data structure -- an integer array of hash
|
||||
* codes for each item, and an Object array of the key/value pairs. This allows it to
|
||||
* avoid having to create an extra object for every entry put in to the map, and it
|
||||
* also tries to control the growth of the size of these arrays more aggressively
|
||||
* (since growing them only requires copying the entries in the array, not rebuilding
|
||||
* a hash map).
|
||||
*
|
||||
* <p>Note that this implementation is not intended to be appropriate for data structures
|
||||
* that may contain large numbers of items. It is generally slower than a traditional
|
||||
* HashMap, since lookups require a binary search and adds and removes require inserting
|
||||
* and deleting entries in the array. For containers holding up to hundreds of items,
|
||||
* the performance difference is not significant, less than 50%.</p>
|
||||
*
|
||||
* <p>Because this container is intended to better balance memory use, unlike most other
|
||||
* standard Java containers it will shrink its array as items are removed from it. Currently
|
||||
* you have no control over this shrinking -- if you set a capacity and then remove an
|
||||
* item, it may reduce the capacity to better match the current size. In the future an
|
||||
* explicit call to set the capacity should turn off this aggressive shrinking behavior.</p>
|
||||
*/
|
||||
public final class ArrayMap<K, V> implements Map<K, V> {
|
||||
private static final boolean DEBUG = false;
|
||||
private static final String TAG = "ArrayMap";
|
||||
/**
|
||||
* Attempt to spot concurrent modifications to this data structure.
|
||||
*
|
||||
* It's best-effort, but any time we can throw something more diagnostic than an
|
||||
* ArrayIndexOutOfBoundsException deep in the ArrayMap internals it's going to
|
||||
* save a lot of development time.
|
||||
*
|
||||
* Good times to look for CME include after any allocArrays() call and at the end of
|
||||
* functions that change mSize (put/remove/clear).
|
||||
*/
|
||||
private static final boolean CONCURRENT_MODIFICATION_EXCEPTIONS = true;
|
||||
/**
|
||||
* The minimum amount by which the capacity of a ArrayMap will increase.
|
||||
* This is tuned to be relatively space-efficient.
|
||||
*/
|
||||
private static final int BASE_SIZE = 4;
|
||||
/**
|
||||
* Maximum number of entries to have in array caches.
|
||||
*/
|
||||
private static final int CACHE_SIZE = 10;
|
||||
/**
|
||||
* Special hash array value that indicates the container is immutable.
|
||||
*/
|
||||
static final int[] EMPTY_IMMUTABLE_INTS = new int[0];
|
||||
/**
|
||||
* @hide Special immutable empty ArrayMap.
|
||||
*/
|
||||
public static final ArrayMap EMPTY = new ArrayMap<>(-1);
|
||||
/**
|
||||
* Caches of small array objects to avoid spamming garbage. The cache
|
||||
* Object[] variable is a pointer to a linked list of array objects.
|
||||
* The first entry in the array is a pointer to the next array in the
|
||||
* list; the second entry is a pointer to the int[] hash code array for it.
|
||||
*/
|
||||
static Object[] mBaseCache;
|
||||
static int mBaseCacheSize;
|
||||
static Object[] mTwiceBaseCache;
|
||||
static int mTwiceBaseCacheSize;
|
||||
final boolean mIdentityHashCode;
|
||||
int[] mHashes;
|
||||
Object[] mArray;
|
||||
int mSize;
|
||||
MapCollections<K, V> mCollections;
|
||||
private static int binarySearchHashes(int[] hashes, int N, int hash) {
|
||||
try {
|
||||
return ContainerHelpers.binarySearch(hashes, N, hash);
|
||||
} catch (ArrayIndexOutOfBoundsException e) {
|
||||
if (CONCURRENT_MODIFICATION_EXCEPTIONS) {
|
||||
throw new ConcurrentModificationException();
|
||||
} else {
|
||||
throw e; // the cache is poisoned at this point, there's not much we can do
|
||||
}
|
||||
}
|
||||
}
|
||||
int indexOf(Object key, int hash) {
|
||||
final int N = mSize;
|
||||
// Important fast case: if nothing is in here, nothing to look for.
|
||||
if (N == 0) {
|
||||
return ~0;
|
||||
}
|
||||
int index = binarySearchHashes(mHashes, N, hash);
|
||||
// If the hash code wasn't found, then we have no entry for this key.
|
||||
if (index < 0) {
|
||||
return index;
|
||||
}
|
||||
// If the key at the returned index matches, that's what we want.
|
||||
if (key.equals(mArray[index<<1])) {
|
||||
return index;
|
||||
}
|
||||
// Search for a matching key after the index.
|
||||
int end;
|
||||
for (end = index + 1; end < N && mHashes[end] == hash; end++) {
|
||||
if (key.equals(mArray[end << 1])) return end;
|
||||
}
|
||||
// Search for a matching key before the index.
|
||||
for (int i = index - 1; i >= 0 && mHashes[i] == hash; i--) {
|
||||
if (key.equals(mArray[i << 1])) return i;
|
||||
}
|
||||
// Key not found -- return negative value indicating where a
|
||||
// new entry for this key should go. We use the end of the
|
||||
// hash chain to reduce the number of array entries that will
|
||||
// need to be copied when inserting.
|
||||
return ~end;
|
||||
}
|
||||
int indexOfNull() {
|
||||
final int N = mSize;
|
||||
// Important fast case: if nothing is in here, nothing to look for.
|
||||
if (N == 0) {
|
||||
return ~0;
|
||||
}
|
||||
int index = binarySearchHashes(mHashes, N, 0);
|
||||
// If the hash code wasn't found, then we have no entry for this key.
|
||||
if (index < 0) {
|
||||
return index;
|
||||
}
|
||||
// If the key at the returned index matches, that's what we want.
|
||||
if (null == mArray[index<<1]) {
|
||||
return index;
|
||||
}
|
||||
// Search for a matching key after the index.
|
||||
int end;
|
||||
for (end = index + 1; end < N && mHashes[end] == 0; end++) {
|
||||
if (null == mArray[end << 1]) return end;
|
||||
}
|
||||
// Search for a matching key before the index.
|
||||
for (int i = index - 1; i >= 0 && mHashes[i] == 0; i--) {
|
||||
if (null == mArray[i << 1]) return i;
|
||||
}
|
||||
// Key not found -- return negative value indicating where a
|
||||
// new entry for this key should go. We use the end of the
|
||||
// hash chain to reduce the number of array entries that will
|
||||
// need to be copied when inserting.
|
||||
return ~end;
|
||||
}
|
||||
private void allocArrays(final int size) {
|
||||
if (mHashes == EMPTY_IMMUTABLE_INTS) {
|
||||
throw new UnsupportedOperationException("ArrayMap is immutable");
|
||||
}
|
||||
if (size == (BASE_SIZE*2)) {
|
||||
synchronized (ArrayMap.class) {
|
||||
if (mTwiceBaseCache != null) {
|
||||
final Object[] array = mTwiceBaseCache;
|
||||
mArray = array;
|
||||
mTwiceBaseCache = (Object[])array[0];
|
||||
mHashes = (int[])array[1];
|
||||
array[0] = array[1] = null;
|
||||
mTwiceBaseCacheSize--;
|
||||
if (DEBUG) Log.d(TAG, "Retrieving 2x cache " + mHashes
|
||||
+ " now have " + mTwiceBaseCacheSize + " entries");
|
||||
return;
|
||||
}
|
||||
}
|
||||
} else if (size == BASE_SIZE) {
|
||||
synchronized (ArrayMap.class) {
|
||||
if (mBaseCache != null) {
|
||||
final Object[] array = mBaseCache;
|
||||
mArray = array;
|
||||
mBaseCache = (Object[])array[0];
|
||||
mHashes = (int[])array[1];
|
||||
array[0] = array[1] = null;
|
||||
mBaseCacheSize--;
|
||||
if (DEBUG) Log.d(TAG, "Retrieving 1x cache " + mHashes
|
||||
+ " now have " + mBaseCacheSize + " entries");
|
||||
return;
|
||||
}
|
||||
}
|
||||
}
|
||||
mHashes = new int[size];
|
||||
mArray = new Object[size<<1];
|
||||
}
|
||||
private static void freeArrays(final int[] hashes, final Object[] array, final int size) {
|
||||
if (hashes.length == (BASE_SIZE*2)) {
|
||||
synchronized (ArrayMap.class) {
|
||||
if (mTwiceBaseCacheSize < CACHE_SIZE) {
|
||||
array[0] = mTwiceBaseCache;
|
||||
array[1] = hashes;
|
||||
for (int i=(size<<1)-1; i>=2; i--) {
|
||||
array[i] = null;
|
||||
}
|
||||
mTwiceBaseCache = array;
|
||||
mTwiceBaseCacheSize++;
|
||||
if (DEBUG) Log.d(TAG, "Storing 2x cache " + array
|
||||
+ " now have " + mTwiceBaseCacheSize + " entries");
|
||||
}
|
||||
}
|
||||
} else if (hashes.length == BASE_SIZE) {
|
||||
synchronized (ArrayMap.class) {
|
||||
if (mBaseCacheSize < CACHE_SIZE) {
|
||||
array[0] = mBaseCache;
|
||||
array[1] = hashes;
|
||||
for (int i=(size<<1)-1; i>=2; i--) {
|
||||
array[i] = null;
|
||||
}
|
||||
mBaseCache = array;
|
||||
mBaseCacheSize++;
|
||||
if (DEBUG) Log.d(TAG, "Storing 1x cache " + array
|
||||
+ " now have " + mBaseCacheSize + " entries");
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
/**
|
||||
* Create a new empty ArrayMap. The default capacity of an array map is 0, and
|
||||
* will grow once items are added to it.
|
||||
*/
|
||||
public ArrayMap() {
|
||||
this(0, false);
|
||||
}
|
||||
/**
|
||||
* Create a new ArrayMap with a given initial capacity.
|
||||
*/
|
||||
public ArrayMap(int capacity) {
|
||||
this(capacity, false);
|
||||
}
|
||||
/** {@hide} */
|
||||
public ArrayMap(int capacity, boolean identityHashCode) {
|
||||
mIdentityHashCode = identityHashCode;
|
||||
// If this is immutable, use the sentinal EMPTY_IMMUTABLE_INTS
|
||||
// instance instead of the usual EmptyArray.INT. The reference
|
||||
// is checked later to see if the array is allowed to grow.
|
||||
if (capacity < 0) {
|
||||
mHashes = EMPTY_IMMUTABLE_INTS;
|
||||
mArray = EmptyArray.OBJECT;
|
||||
} else if (capacity == 0) {
|
||||
mHashes = EmptyArray.INT;
|
||||
mArray = EmptyArray.OBJECT;
|
||||
} else {
|
||||
allocArrays(capacity);
|
||||
}
|
||||
mSize = 0;
|
||||
}
|
||||
/**
|
||||
* Create a new ArrayMap with the mappings from the given ArrayMap.
|
||||
*/
|
||||
public ArrayMap(ArrayMap<K, V> map) {
|
||||
this();
|
||||
if (map != null) {
|
||||
putAll(map);
|
||||
}
|
||||
}
|
||||
/**
|
||||
* Make the array map empty. All storage is released.
|
||||
*/
|
||||
@Override
|
||||
public void clear() {
|
||||
if (mSize > 0) {
|
||||
final int[] ohashes = mHashes;
|
||||
final Object[] oarray = mArray;
|
||||
final int osize = mSize;
|
||||
mHashes = EmptyArray.INT;
|
||||
mArray = EmptyArray.OBJECT;
|
||||
mSize = 0;
|
||||
freeArrays(ohashes, oarray, osize);
|
||||
}
|
||||
if (CONCURRENT_MODIFICATION_EXCEPTIONS && mSize > 0) {
|
||||
throw new ConcurrentModificationException();
|
||||
}
|
||||
}
|
||||
/**
|
||||
* @hide
|
||||
* Like {@link #clear}, but doesn't reduce the capacity of the ArrayMap.
|
||||
*/
|
||||
public void erase() {
|
||||
if (mSize > 0) {
|
||||
final int N = mSize<<1;
|
||||
final Object[] array = mArray;
|
||||
for (int i=0; i<N; i++) {
|
||||
array[i] = null;
|
||||
}
|
||||
mSize = 0;
|
||||
}
|
||||
}
|
||||
/**
|
||||
* Ensure the array map can hold at least <var>minimumCapacity</var>
|
||||
* items.
|
||||
*/
|
||||
public void ensureCapacity(int minimumCapacity) {
|
||||
final int osize = mSize;
|
||||
if (mHashes.length < minimumCapacity) {
|
||||
final int[] ohashes = mHashes;
|
||||
final Object[] oarray = mArray;
|
||||
allocArrays(minimumCapacity);
|
||||
if (mSize > 0) {
|
||||
System.arraycopy(ohashes, 0, mHashes, 0, osize);
|
||||
System.arraycopy(oarray, 0, mArray, 0, osize<<1);
|
||||
}
|
||||
freeArrays(ohashes, oarray, osize);
|
||||
}
|
||||
if (CONCURRENT_MODIFICATION_EXCEPTIONS && mSize != osize) {
|
||||
throw new ConcurrentModificationException();
|
||||
}
|
||||
}
|
||||
/**
|
||||
* Check whether a key exists in the array.
|
||||
*
|
||||
* @param key The key to search for.
|
||||
* @return Returns true if the key exists, else false.
|
||||
*/
|
||||
@Override
|
||||
public boolean containsKey(Object key) {
|
||||
return indexOfKey(key) >= 0;
|
||||
}
|
||||
/**
|
||||
* Returns the index of a key in the set.
|
||||
*
|
||||
* @param key The key to search for.
|
||||
* @return Returns the index of the key if it exists, else a negative integer.
|
||||
*/
|
||||
public int indexOfKey(Object key) {
|
||||
return key == null ? indexOfNull()
|
||||
: indexOf(key, mIdentityHashCode ? System.identityHashCode(key) : key.hashCode());
|
||||
}
|
||||
int indexOfValue(Object value) {
|
||||
final int N = mSize*2;
|
||||
final Object[] array = mArray;
|
||||
if (value == null) {
|
||||
for (int i=1; i<N; i+=2) {
|
||||
if (array[i] == null) {
|
||||
return i>>1;
|
||||
}
|
||||
}
|
||||
} else {
|
||||
for (int i=1; i<N; i+=2) {
|
||||
if (value.equals(array[i])) {
|
||||
return i>>1;
|
||||
}
|
||||
}
|
||||
}
|
||||
return -1;
|
||||
}
|
||||
/**
|
||||
* Check whether a value exists in the array. This requires a linear search
|
||||
* through the entire array.
|
||||
*
|
||||
* @param value The value to search for.
|
||||
* @return Returns true if the value exists, else false.
|
||||
*/
|
||||
@Override
|
||||
public boolean containsValue(Object value) {
|
||||
return indexOfValue(value) >= 0;
|
||||
}
|
||||
/**
|
||||
* Retrieve a value from the array.
|
||||
* @param key The key of the value to retrieve.
|
||||
* @return Returns the value associated with the given key,
|
||||
* or null if there is no such key.
|
||||
*/
|
||||
@Override
|
||||
public V get(Object key) {
|
||||
final int index = indexOfKey(key);
|
||||
return index >= 0 ? (V)mArray[(index<<1)+1] : null;
|
||||
}
|
||||
/**
|
||||
* Return the key at the given index in the array.
|
||||
* @param index The desired index, must be between 0 and {@link #size()}-1.
|
||||
* @return Returns the key stored at the given index.
|
||||
*/
|
||||
public K keyAt(int index) {
|
||||
return (K)mArray[index << 1];
|
||||
}
|
||||
/**
|
||||
* Return the value at the given index in the array.
|
||||
* @param index The desired index, must be between 0 and {@link #size()}-1.
|
||||
* @return Returns the value stored at the given index.
|
||||
*/
|
||||
public V valueAt(int index) {
|
||||
return (V)mArray[(index << 1) + 1];
|
||||
}
|
||||
/**
|
||||
* Set the value at a given index in the array.
|
||||
* @param index The desired index, must be between 0 and {@link #size()}-1.
|
||||
* @param value The new value to store at this index.
|
||||
* @return Returns the previous value at the given index.
|
||||
*/
|
||||
public V setValueAt(int index, V value) {
|
||||
index = (index << 1) + 1;
|
||||
V old = (V)mArray[index];
|
||||
mArray[index] = value;
|
||||
return old;
|
||||
}
|
||||
/**
|
||||
* Return true if the array map contains no items.
|
||||
*/
|
||||
@Override
|
||||
public boolean isEmpty() {
|
||||
return mSize <= 0;
|
||||
}
|
||||
/**
|
||||
* Add a new value to the array map.
|
||||
* @param key The key under which to store the value. If
|
||||
* this key already exists in the array, its value will be replaced.
|
||||
* @param value The value to store for the given key.
|
||||
* @return Returns the old value that was stored for the given key, or null if there
|
||||
* was no such key.
|
||||
*/
|
||||
@Override
|
||||
public V put(K key, V value) {
|
||||
final int osize = mSize;
|
||||
final int hash;
|
||||
int index;
|
||||
if (key == null) {
|
||||
hash = 0;
|
||||
index = indexOfNull();
|
||||
} else {
|
||||
hash = mIdentityHashCode ? System.identityHashCode(key) : key.hashCode();
|
||||
index = indexOf(key, hash);
|
||||
}
|
||||
if (index >= 0) {
|
||||
index = (index<<1) + 1;
|
||||
final V old = (V)mArray[index];
|
||||
mArray[index] = value;
|
||||
return old;
|
||||
}
|
||||
index = ~index;
|
||||
if (osize >= mHashes.length) {
|
||||
final int n = osize >= (BASE_SIZE*2) ? (osize+(osize>>1))
|
||||
: (osize >= BASE_SIZE ? (BASE_SIZE*2) : BASE_SIZE);
|
||||
if (DEBUG) Log.d(TAG, "put: grow from " + mHashes.length + " to " + n);
|
||||
final int[] ohashes = mHashes;
|
||||
final Object[] oarray = mArray;
|
||||
allocArrays(n);
|
||||
if (CONCURRENT_MODIFICATION_EXCEPTIONS && osize != mSize) {
|
||||
throw new ConcurrentModificationException();
|
||||
}
|
||||
if (mHashes.length > 0) {
|
||||
if (DEBUG) Log.d(TAG, "put: copy 0-" + osize + " to 0");
|
||||
System.arraycopy(ohashes, 0, mHashes, 0, ohashes.length);
|
||||
System.arraycopy(oarray, 0, mArray, 0, oarray.length);
|
||||
}
|
||||
freeArrays(ohashes, oarray, osize);
|
||||
}
|
||||
if (index < osize) {
|
||||
if (DEBUG) Log.d(TAG, "put: move " + index + "-" + (osize-index)
|
||||
+ " to " + (index+1));
|
||||
System.arraycopy(mHashes, index, mHashes, index + 1, osize - index);
|
||||
System.arraycopy(mArray, index << 1, mArray, (index + 1) << 1, (mSize - index) << 1);
|
||||
}
|
||||
if (CONCURRENT_MODIFICATION_EXCEPTIONS) {
|
||||
if (osize != mSize || index >= mHashes.length) {
|
||||
throw new ConcurrentModificationException();
|
||||
}
|
||||
}
|
||||
mHashes[index] = hash;
|
||||
mArray[index<<1] = key;
|
||||
mArray[(index<<1)+1] = value;
|
||||
mSize++;
|
||||
return null;
|
||||
}
|
||||
/**
|
||||
* Special fast path for appending items to the end of the array without validation.
|
||||
* The array must already be large enough to contain the item.
|
||||
* @hide
|
||||
*/
|
||||
public void append(K key, V value) {
|
||||
int index = mSize;
|
||||
final int hash = key == null ? 0
|
||||
: (mIdentityHashCode ? System.identityHashCode(key) : key.hashCode());
|
||||
if (index >= mHashes.length) {
|
||||
throw new IllegalStateException("Array is full");
|
||||
}
|
||||
if (index > 0 && mHashes[index-1] > hash) {
|
||||
RuntimeException e = new RuntimeException("here");
|
||||
e.fillInStackTrace();
|
||||
Log.w(TAG, "New hash " + hash
|
||||
+ " is before end of array hash " + mHashes[index-1]
|
||||
+ " at index " + index + " key " + key, e);
|
||||
put(key, value);
|
||||
return;
|
||||
}
|
||||
mSize = index+1;
|
||||
mHashes[index] = hash;
|
||||
index <<= 1;
|
||||
mArray[index] = key;
|
||||
mArray[index+1] = value;
|
||||
}
|
||||
/**
|
||||
* The use of the {@link #append} function can result in invalid array maps, in particular
|
||||
* an array map where the same key appears multiple times. This function verifies that
|
||||
* the array map is valid, throwing IllegalArgumentException if a problem is found. The
|
||||
* main use for this method is validating an array map after unpacking from an IPC, to
|
||||
* protect against malicious callers.
|
||||
* @hide
|
||||
*/
|
||||
public void validate() {
|
||||
final int N = mSize;
|
||||
if (N <= 1) {
|
||||
// There can't be dups.
|
||||
return;
|
||||
}
|
||||
int basehash = mHashes[0];
|
||||
int basei = 0;
|
||||
for (int i=1; i<N; i++) {
|
||||
int hash = mHashes[i];
|
||||
if (hash != basehash) {
|
||||
basehash = hash;
|
||||
basei = i;
|
||||
continue;
|
||||
}
|
||||
// We are in a run of entries with the same hash code. Go backwards through
|
||||
// the array to see if any keys are the same.
|
||||
final Object cur = mArray[i<<1];
|
||||
for (int j=i-1; j>=basei; j--) {
|
||||
final Object prev = mArray[j<<1];
|
||||
if (cur == prev) {
|
||||
throw new IllegalArgumentException("Duplicate key in ArrayMap: " + cur);
|
||||
}
|
||||
if (cur != null && prev != null && cur.equals(prev)) {
|
||||
throw new IllegalArgumentException("Duplicate key in ArrayMap: " + cur);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
/**
|
||||
* Perform a {@link #put(Object, Object)} of all key/value pairs in <var>array</var>
|
||||
* @param array The array whose contents are to be retrieved.
|
||||
*/
|
||||
public void putAll(ArrayMap<? extends K, ? extends V> array) {
|
||||
final int N = array.mSize;
|
||||
ensureCapacity(mSize + N);
|
||||
if (mSize == 0) {
|
||||
if (N > 0) {
|
||||
System.arraycopy(array.mHashes, 0, mHashes, 0, N);
|
||||
System.arraycopy(array.mArray, 0, mArray, 0, N<<1);
|
||||
mSize = N;
|
||||
}
|
||||
} else {
|
||||
for (int i=0; i<N; i++) {
|
||||
put(array.keyAt(i), array.valueAt(i));
|
||||
}
|
||||
}
|
||||
}
|
||||
/**
|
||||
* Remove an existing key from the array map.
|
||||
* @param key The key of the mapping to remove.
|
||||
* @return Returns the value that was stored under the key, or null if there
|
||||
* was no such key.
|
||||
*/
|
||||
@Override
|
||||
public V remove(Object key) {
|
||||
final int index = indexOfKey(key);
|
||||
if (index >= 0) {
|
||||
return removeAt(index);
|
||||
}
|
||||
return null;
|
||||
}
|
||||
/**
|
||||
* Remove the key/value mapping at the given index.
|
||||
* @param index The desired index, must be between 0 and {@link #size()}-1.
|
||||
* @return Returns the value that was stored at this index.
|
||||
*/
|
||||
public V removeAt(int index) {
|
||||
final Object old = mArray[(index << 1) + 1];
|
||||
final int osize = mSize;
|
||||
final int nsize;
|
||||
if (osize <= 1) {
|
||||
// Now empty.
|
||||
if (DEBUG) Log.d(TAG, "remove: shrink from " + mHashes.length + " to 0");
|
||||
freeArrays(mHashes, mArray, osize);
|
||||
mHashes = EmptyArray.INT;
|
||||
mArray = EmptyArray.OBJECT;
|
||||
nsize = 0;
|
||||
} else {
|
||||
nsize = osize - 1;
|
||||
if (mHashes.length > (BASE_SIZE*2) && mSize < mHashes.length/3) {
|
||||
// Shrunk enough to reduce size of arrays. We don't allow it to
|
||||
// shrink smaller than (BASE_SIZE*2) to avoid flapping between
|
||||
// that and BASE_SIZE.
|
||||
final int n = osize > (BASE_SIZE*2) ? (osize + (osize>>1)) : (BASE_SIZE*2);
|
||||
if (DEBUG) Log.d(TAG, "remove: shrink from " + mHashes.length + " to " + n);
|
||||
final int[] ohashes = mHashes;
|
||||
final Object[] oarray = mArray;
|
||||
allocArrays(n);
|
||||
if (CONCURRENT_MODIFICATION_EXCEPTIONS && osize != mSize) {
|
||||
throw new ConcurrentModificationException();
|
||||
}
|
||||
if (index > 0) {
|
||||
if (DEBUG) Log.d(TAG, "remove: copy from 0-" + index + " to 0");
|
||||
System.arraycopy(ohashes, 0, mHashes, 0, index);
|
||||
System.arraycopy(oarray, 0, mArray, 0, index << 1);
|
||||
}
|
||||
if (index < nsize) {
|
||||
if (DEBUG) Log.d(TAG, "remove: copy from " + (index+1) + "-" + nsize
|
||||
+ " to " + index);
|
||||
System.arraycopy(ohashes, index + 1, mHashes, index, nsize - index);
|
||||
System.arraycopy(oarray, (index + 1) << 1, mArray, index << 1,
|
||||
(nsize - index) << 1);
|
||||
}
|
||||
} else {
|
||||
if (index < nsize) {
|
||||
if (DEBUG) Log.d(TAG, "remove: move " + (index+1) + "-" + nsize
|
||||
+ " to " + index);
|
||||
System.arraycopy(mHashes, index + 1, mHashes, index, nsize - index);
|
||||
System.arraycopy(mArray, (index + 1) << 1, mArray, index << 1,
|
||||
(nsize - index) << 1);
|
||||
}
|
||||
mArray[nsize << 1] = null;
|
||||
mArray[(nsize << 1) + 1] = null;
|
||||
}
|
||||
}
|
||||
if (CONCURRENT_MODIFICATION_EXCEPTIONS && osize != mSize) {
|
||||
throw new ConcurrentModificationException();
|
||||
}
|
||||
mSize = nsize;
|
||||
return (V)old;
|
||||
}
|
||||
/**
|
||||
* Return the number of items in this array map.
|
||||
*/
|
||||
@Override
|
||||
public int size() {
|
||||
return mSize;
|
||||
}
|
||||
/**
|
||||
* {@inheritDoc}
|
||||
*
|
||||
* <p>This implementation returns false if the object is not a map, or
|
||||
* if the maps have different sizes. Otherwise, for each key in this map,
|
||||
* values of both maps are compared. If the values for any key are not
|
||||
* equal, the method returns false, otherwise it returns true.
|
||||
*/
|
||||
@Override
|
||||
public boolean equals(Object object) {
|
||||
if (this == object) {
|
||||
return true;
|
||||
}
|
||||
if (object instanceof Map) {
|
||||
Map<?, ?> map = (Map<?, ?>) object;
|
||||
if (size() != map.size()) {
|
||||
return false;
|
||||
}
|
||||
try {
|
||||
for (int i=0; i<mSize; i++) {
|
||||
K key = keyAt(i);
|
||||
V mine = valueAt(i);
|
||||
Object theirs = map.get(key);
|
||||
if (mine == null) {
|
||||
if (theirs != null || !map.containsKey(key)) {
|
||||
return false;
|
||||
}
|
||||
} else if (!mine.equals(theirs)) {
|
||||
return false;
|
||||
}
|
||||
}
|
||||
} catch (NullPointerException ignored) {
|
||||
return false;
|
||||
} catch (ClassCastException ignored) {
|
||||
return false;
|
||||
}
|
||||
return true;
|
||||
}
|
||||
return false;
|
||||
}
|
||||
/**
|
||||
* {@inheritDoc}
|
||||
*/
|
||||
@Override
|
||||
public int hashCode() {
|
||||
final int[] hashes = mHashes;
|
||||
final Object[] array = mArray;
|
||||
int result = 0;
|
||||
for (int i = 0, v = 1, s = mSize; i < s; i++, v+=2) {
|
||||
Object value = array[v];
|
||||
result += hashes[i] ^ (value == null ? 0 : value.hashCode());
|
||||
}
|
||||
return result;
|
||||
}
|
||||
/**
|
||||
* {@inheritDoc}
|
||||
*
|
||||
* <p>This implementation composes a string by iterating over its mappings. If
|
||||
* this map contains itself as a key or a value, the string "(this Map)"
|
||||
* will appear in its place.
|
||||
*/
|
||||
@Override
|
||||
public String toString() {
|
||||
if (isEmpty()) {
|
||||
return "{}";
|
||||
}
|
||||
StringBuilder buffer = new StringBuilder(mSize * 28);
|
||||
buffer.append('{');
|
||||
for (int i=0; i<mSize; i++) {
|
||||
if (i > 0) {
|
||||
buffer.append(", ");
|
||||
}
|
||||
Object key = keyAt(i);
|
||||
if (key != this) {
|
||||
buffer.append(key);
|
||||
} else {
|
||||
buffer.append("(this Map)");
|
||||
}
|
||||
buffer.append('=');
|
||||
Object value = valueAt(i);
|
||||
if (value != this) {
|
||||
buffer.append(value);
|
||||
} else {
|
||||
buffer.append("(this Map)");
|
||||
}
|
||||
}
|
||||
buffer.append('}');
|
||||
return buffer.toString();
|
||||
}
|
||||
// ------------------------------------------------------------------------
|
||||
// Interop with traditional Java containers. Not as efficient as using
|
||||
// specialized collection APIs.
|
||||
// ------------------------------------------------------------------------
|
||||
private MapCollections<K, V> getCollection() {
|
||||
if (mCollections == null) {
|
||||
mCollections = new MapCollections<K, V>() {
|
||||
@Override
|
||||
protected int colGetSize() {
|
||||
return mSize;
|
||||
}
|
||||
@Override
|
||||
protected Object colGetEntry(int index, int offset) {
|
||||
return mArray[(index<<1) + offset];
|
||||
}
|
||||
@Override
|
||||
protected int colIndexOfKey(Object key) {
|
||||
return indexOfKey(key);
|
||||
}
|
||||
@Override
|
||||
protected int colIndexOfValue(Object value) {
|
||||
return indexOfValue(value);
|
||||
}
|
||||
@Override
|
||||
protected Map<K, V> colGetMap() {
|
||||
return ArrayMap.this;
|
||||
}
|
||||
@Override
|
||||
protected void colPut(K key, V value) {
|
||||
put(key, value);
|
||||
}
|
||||
@Override
|
||||
protected V colSetValue(int index, V value) {
|
||||
return setValueAt(index, value);
|
||||
}
|
||||
@Override
|
||||
protected void colRemoveAt(int index) {
|
||||
removeAt(index);
|
||||
}
|
||||
@Override
|
||||
protected void colClear() {
|
||||
clear();
|
||||
}
|
||||
};
|
||||
}
|
||||
return mCollections;
|
||||
}
|
||||
/**
|
||||
* Determine if the array map contains all of the keys in the given collection.
|
||||
* @param collection The collection whose contents are to be checked against.
|
||||
* @return Returns true if this array map contains a key for every entry
|
||||
* in <var>collection</var>, else returns false.
|
||||
*/
|
||||
public boolean containsAll(Collection<?> collection) {
|
||||
return MapCollections.containsAllHelper(this, collection);
|
||||
}
|
||||
/**
|
||||
* Perform a {@link #put(Object, Object)} of all key/value pairs in <var>map</var>
|
||||
* @param map The map whose contents are to be retrieved.
|
||||
*/
|
||||
@Override
|
||||
public void putAll(Map<? extends K, ? extends V> map) {
|
||||
ensureCapacity(mSize + map.size());
|
||||
for (Map.Entry<? extends K, ? extends V> entry : map.entrySet()) {
|
||||
put(entry.getKey(), entry.getValue());
|
||||
}
|
||||
}
|
||||
/**
|
||||
* Remove all keys in the array map that exist in the given collection.
|
||||
* @param collection The collection whose contents are to be used to remove keys.
|
||||
* @return Returns true if any keys were removed from the array map, else false.
|
||||
*/
|
||||
public boolean removeAll(Collection<?> collection) {
|
||||
return MapCollections.removeAllHelper(this, collection);
|
||||
}
|
||||
/**
|
||||
* Remove all keys in the array map that do <b>not</b> exist in the given collection.
|
||||
* @param collection The collection whose contents are to be used to determine which
|
||||
* keys to keep.
|
||||
* @return Returns true if any keys were removed from the array map, else false.
|
||||
*/
|
||||
public boolean retainAll(Collection<?> collection) {
|
||||
return MapCollections.retainAllHelper(this, collection);
|
||||
}
|
||||
/**
|
||||
* Return a {@link java.util.Set} for iterating over and interacting with all mappings
|
||||
* in the array map.
|
||||
*
|
||||
* <p><b>Note:</b> this is a very inefficient way to access the array contents, it
|
||||
* requires generating a number of temporary objects and allocates additional state
|
||||
* information associated with the container that will remain for the life of the container.</p>
|
||||
*
|
||||
* <p><b>Note:</b></p> the semantics of this
|
||||
* Set are subtly different than that of a {@link java.util.HashMap}: most important,
|
||||
* the {@link java.util.Map.Entry Map.Entry} object returned by its iterator is a single
|
||||
* object that exists for the entire iterator, so you can <b>not</b> hold on to it
|
||||
* after calling {@link java.util.Iterator#next() Iterator.next}.</p>
|
||||
*/
|
||||
@Override
|
||||
public Set<Map.Entry<K, V>> entrySet() {
|
||||
return getCollection().getEntrySet();
|
||||
}
|
||||
/**
|
||||
* Return a {@link java.util.Set} for iterating over and interacting with all keys
|
||||
* in the array map.
|
||||
*
|
||||
* <p><b>Note:</b> this is a fairly inefficient way to access the array contents, it
|
||||
* requires generating a number of temporary objects and allocates additional state
|
||||
* information associated with the container that will remain for the life of the container.</p>
|
||||
*/
|
||||
@Override
|
||||
public Set<K> keySet() {
|
||||
return getCollection().getKeySet();
|
||||
}
|
||||
/**
|
||||
* Return a {@link java.util.Collection} for iterating over and interacting with all values
|
||||
* in the array map.
|
||||
*
|
||||
* <p><b>Note:</b> this is a fairly inefficient way to access the array contents, it
|
||||
* requires generating a number of temporary objects and allocates additional state
|
||||
* information associated with the container that will remain for the life of the container.</p>
|
||||
*/
|
||||
@Override
|
||||
public Collection<V> values() {
|
||||
return getCollection().getValues();
|
||||
}
|
||||
}
|
||||
676
AndroidCompat/src/main/java/android/util/Base64.java
Normal file
676
AndroidCompat/src/main/java/android/util/Base64.java
Normal file
@@ -0,0 +1,676 @@
|
||||
/*
|
||||
* Copyright (C) 2010 The Android Open Source Project
|
||||
*
|
||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||
* you may not use this file except in compliance with the License.
|
||||
* You may obtain a copy of the License at
|
||||
*
|
||||
* http://www.apache.org/licenses/LICENSE-2.0
|
||||
*
|
||||
* Unless required by applicable law or agreed to in writing, software
|
||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
* See the License for the specific language governing permissions and
|
||||
* limitations under the License.
|
||||
*/
|
||||
package android.util;
|
||||
|
||||
import java.nio.charset.StandardCharsets;
|
||||
|
||||
/**
|
||||
* Utilities for encoding and decoding the Base64 representation of
|
||||
* binary data. See RFCs <a
|
||||
* href="http://www.ietf.org/rfc/rfc2045.txt">2045</a> and <a
|
||||
* href="http://www.ietf.org/rfc/rfc3548.txt">3548</a>.
|
||||
*/
|
||||
public class Base64 {
|
||||
/**
|
||||
* Default values for encoder/decoder flags.
|
||||
*/
|
||||
public static final int DEFAULT = 0;
|
||||
/**
|
||||
* Encoder flag bit to omit the padding '=' characters at the end
|
||||
* of the output (if any).
|
||||
*/
|
||||
public static final int NO_PADDING = 1;
|
||||
/**
|
||||
* Encoder flag bit to omit all line terminators (i.e., the output
|
||||
* will be on one long line).
|
||||
*/
|
||||
public static final int NO_WRAP = 2;
|
||||
/**
|
||||
* Encoder flag bit to indicate lines should be terminated with a
|
||||
* CRLF pair instead of just an LF. Has no effect if {@code
|
||||
* NO_WRAP} is specified as well.
|
||||
*/
|
||||
public static final int CRLF = 4;
|
||||
/**
|
||||
* Encoder/decoder flag bit to indicate using the "URL and
|
||||
* filename safe" variant of Base64 (see RFC 3548 section 4) where
|
||||
* {@code -} and {@code _} are used in place of {@code +} and
|
||||
* {@code /}.
|
||||
*/
|
||||
public static final int URL_SAFE = 8;
|
||||
/**
|
||||
* Flag to pass to {@link Base64OutputStream} to indicate that it
|
||||
* should not close the output stream it is wrapping when it
|
||||
* itself is closed.
|
||||
*/
|
||||
public static final int NO_CLOSE = 16;
|
||||
|
||||
private Base64() {
|
||||
} // don't instantiate
|
||||
// --------------------------------------------------------
|
||||
// decoding
|
||||
// --------------------------------------------------------
|
||||
|
||||
/**
|
||||
* Decode the Base64-encoded data in input and return the data in
|
||||
* a new byte array.
|
||||
*
|
||||
* <p>The padding '=' characters at the end are considered optional, but
|
||||
* if any are present, there must be the correct number of them.
|
||||
*
|
||||
* @param str the input String to decode, which is converted to
|
||||
* bytes using the default charset
|
||||
* @param flags controls certain features of the decoded output.
|
||||
* Pass {@code DEFAULT} to decode standard Base64.
|
||||
* @throws IllegalArgumentException if the input contains
|
||||
* incorrect padding
|
||||
*/
|
||||
public static byte[] decode(String str, int flags) {
|
||||
return decode(str.getBytes(), flags);
|
||||
}
|
||||
|
||||
/**
|
||||
* Decode the Base64-encoded data in input and return the data in
|
||||
* a new byte array.
|
||||
*
|
||||
* <p>The padding '=' characters at the end are considered optional, but
|
||||
* if any are present, there must be the correct number of them.
|
||||
*
|
||||
* @param input the input array to decode
|
||||
* @param flags controls certain features of the decoded output.
|
||||
* Pass {@code DEFAULT} to decode standard Base64.
|
||||
* @throws IllegalArgumentException if the input contains
|
||||
* incorrect padding
|
||||
*/
|
||||
public static byte[] decode(byte[] input, int flags) {
|
||||
return decode(input, 0, input.length, flags);
|
||||
}
|
||||
|
||||
/**
|
||||
* Decode the Base64-encoded data in input and return the data in
|
||||
* a new byte array.
|
||||
*
|
||||
* <p>The padding '=' characters at the end are considered optional, but
|
||||
* if any are present, there must be the correct number of them.
|
||||
*
|
||||
* @param input the data to decode
|
||||
* @param offset the position within the input array at which to start
|
||||
* @param len the number of bytes of input to decode
|
||||
* @param flags controls certain features of the decoded output.
|
||||
* Pass {@code DEFAULT} to decode standard Base64.
|
||||
* @throws IllegalArgumentException if the input contains
|
||||
* incorrect padding
|
||||
*/
|
||||
public static byte[] decode(byte[] input, int offset, int len, int flags) {
|
||||
// Allocate space for the most data the input could represent.
|
||||
// (It could contain less if it contains whitespace, etc.)
|
||||
Decoder decoder = new Decoder(flags, new byte[len * 3 / 4]);
|
||||
if (!decoder.process(input, offset, len, true)) {
|
||||
throw new IllegalArgumentException("bad base-64");
|
||||
}
|
||||
// Maybe we got lucky and allocated exactly enough output space.
|
||||
if (decoder.op == decoder.output.length) {
|
||||
return decoder.output;
|
||||
}
|
||||
// Need to shorten the array, so allocate a new one of the
|
||||
// right size and copy.
|
||||
byte[] temp = new byte[decoder.op];
|
||||
System.arraycopy(decoder.output, 0, temp, 0, decoder.op);
|
||||
return temp;
|
||||
}
|
||||
|
||||
/**
|
||||
* Base64-encode the given data and return a newly allocated
|
||||
* String with the result.
|
||||
*
|
||||
* @param input the data to encode
|
||||
* @param flags controls certain features of the encoded output.
|
||||
* Passing {@code DEFAULT} results in output that
|
||||
* adheres to RFC 2045.
|
||||
*/
|
||||
public static String encodeToString(byte[] input, int flags) {
|
||||
return new String(encode(input, flags), StandardCharsets.US_ASCII);
|
||||
}
|
||||
// --------------------------------------------------------
|
||||
// encoding
|
||||
// --------------------------------------------------------
|
||||
|
||||
/**
|
||||
* Base64-encode the given data and return a newly allocated
|
||||
* String with the result.
|
||||
*
|
||||
* @param input the data to encode
|
||||
* @param offset the position within the input array at which to
|
||||
* start
|
||||
* @param len the number of bytes of input to encode
|
||||
* @param flags controls certain features of the encoded output.
|
||||
* Passing {@code DEFAULT} results in output that
|
||||
* adheres to RFC 2045.
|
||||
*/
|
||||
public static String encodeToString(byte[] input, int offset, int len, int flags) {
|
||||
return new String(encode(input, offset, len, flags), StandardCharsets.US_ASCII);
|
||||
}
|
||||
|
||||
/**
|
||||
* Base64-encode the given data and return a newly allocated
|
||||
* byte[] with the result.
|
||||
*
|
||||
* @param input the data to encode
|
||||
* @param flags controls certain features of the encoded output.
|
||||
* Passing {@code DEFAULT} results in output that
|
||||
* adheres to RFC 2045.
|
||||
*/
|
||||
public static byte[] encode(byte[] input, int flags) {
|
||||
return encode(input, 0, input.length, flags);
|
||||
}
|
||||
|
||||
/**
|
||||
* Base64-encode the given data and return a newly allocated
|
||||
* byte[] with the result.
|
||||
*
|
||||
* @param input the data to encode
|
||||
* @param offset the position within the input array at which to
|
||||
* start
|
||||
* @param len the number of bytes of input to encode
|
||||
* @param flags controls certain features of the encoded output.
|
||||
* Passing {@code DEFAULT} results in output that
|
||||
* adheres to RFC 2045.
|
||||
*/
|
||||
public static byte[] encode(byte[] input, int offset, int len, int flags) {
|
||||
Encoder encoder = new Encoder(flags, null);
|
||||
// Compute the exact length of the array we will produce.
|
||||
int output_len = len / 3 * 4;
|
||||
// Account for the tail of the data and the padding bytes, if any.
|
||||
if (encoder.do_padding) {
|
||||
if (len % 3 > 0) {
|
||||
output_len += 4;
|
||||
}
|
||||
} else {
|
||||
switch (len % 3) {
|
||||
case 0:
|
||||
break;
|
||||
case 1:
|
||||
output_len += 2;
|
||||
break;
|
||||
case 2:
|
||||
output_len += 3;
|
||||
break;
|
||||
}
|
||||
}
|
||||
// Account for the newlines, if any.
|
||||
if (encoder.do_newline && len > 0) {
|
||||
output_len += (((len - 1) / (3 * Encoder.LINE_GROUPS)) + 1) *
|
||||
(encoder.do_cr ? 2 : 1);
|
||||
}
|
||||
encoder.output = new byte[output_len];
|
||||
encoder.process(input, offset, len, true);
|
||||
assert encoder.op == output_len;
|
||||
return encoder.output;
|
||||
}
|
||||
|
||||
// --------------------------------------------------------
|
||||
// shared code
|
||||
// --------------------------------------------------------
|
||||
/* package */ static abstract class Coder {
|
||||
public byte[] output;
|
||||
public int op;
|
||||
|
||||
/**
|
||||
* Encode/decode another block of input data. this.output is
|
||||
* provided by the caller, and must be big enough to hold all
|
||||
* the coded data. On exit, this.opwill be set to the length
|
||||
* of the coded data.
|
||||
*
|
||||
* @param finish true if this is the final call to process for
|
||||
* this object. Will finalize the coder state and
|
||||
* include any final bytes in the output.
|
||||
* @return true if the input so far is good; false if some
|
||||
* error has been detected in the input stream..
|
||||
*/
|
||||
public abstract boolean process(byte[] input, int offset, int len, boolean finish);
|
||||
|
||||
/**
|
||||
* @return the maximum number of bytes a call to process()
|
||||
* could produce for the given number of input bytes. This may
|
||||
* be an overestimate.
|
||||
*/
|
||||
public abstract int maxOutputSize(int len);
|
||||
}
|
||||
|
||||
/* package */ static class Decoder extends Coder {
|
||||
/**
|
||||
* Lookup table for turning bytes into their position in the
|
||||
* Base64 alphabet.
|
||||
*/
|
||||
private static final int[] DECODE = {
|
||||
-1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1,
|
||||
-1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1,
|
||||
-1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, 62, -1, -1, -1, 63,
|
||||
52, 53, 54, 55, 56, 57, 58, 59, 60, 61, -1, -1, -1, -2, -1, -1,
|
||||
-1, 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14,
|
||||
15, 16, 17, 18, 19, 20, 21, 22, 23, 24, 25, -1, -1, -1, -1, -1,
|
||||
-1, 26, 27, 28, 29, 30, 31, 32, 33, 34, 35, 36, 37, 38, 39, 40,
|
||||
41, 42, 43, 44, 45, 46, 47, 48, 49, 50, 51, -1, -1, -1, -1, -1,
|
||||
-1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1,
|
||||
-1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1,
|
||||
-1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1,
|
||||
-1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1,
|
||||
-1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1,
|
||||
-1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1,
|
||||
-1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1,
|
||||
-1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1,
|
||||
};
|
||||
/**
|
||||
* Decode lookup table for the "web safe" variant (RFC 3548
|
||||
* sec. 4) where - and _ replace + and /.
|
||||
*/
|
||||
private static final int[] DECODE_WEBSAFE = {
|
||||
-1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1,
|
||||
-1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1,
|
||||
-1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, 62, -1, -1,
|
||||
52, 53, 54, 55, 56, 57, 58, 59, 60, 61, -1, -1, -1, -2, -1, -1,
|
||||
-1, 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14,
|
||||
15, 16, 17, 18, 19, 20, 21, 22, 23, 24, 25, -1, -1, -1, -1, 63,
|
||||
-1, 26, 27, 28, 29, 30, 31, 32, 33, 34, 35, 36, 37, 38, 39, 40,
|
||||
41, 42, 43, 44, 45, 46, 47, 48, 49, 50, 51, -1, -1, -1, -1, -1,
|
||||
-1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1,
|
||||
-1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1,
|
||||
-1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1,
|
||||
-1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1,
|
||||
-1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1,
|
||||
-1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1,
|
||||
-1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1,
|
||||
-1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1,
|
||||
};
|
||||
/**
|
||||
* Non-data values in the DECODE arrays.
|
||||
*/
|
||||
private static final int SKIP = -1;
|
||||
private static final int EQUALS = -2;
|
||||
final private int[] alphabet;
|
||||
/**
|
||||
* States 0-3 are reading through the next input tuple.
|
||||
* State 4 is having read one '=' and expecting exactly
|
||||
* one more.
|
||||
* State 5 is expecting no more data or padding characters
|
||||
* in the input.
|
||||
* State 6 is the error state; an error has been detected
|
||||
* in the input and no future input can "fix" it.
|
||||
*/
|
||||
private int state; // state number (0 to 6)
|
||||
private int value;
|
||||
|
||||
public Decoder(int flags, byte[] output) {
|
||||
this.output = output;
|
||||
alphabet = ((flags & URL_SAFE) == 0) ? DECODE : DECODE_WEBSAFE;
|
||||
state = 0;
|
||||
value = 0;
|
||||
}
|
||||
|
||||
/**
|
||||
* @return an overestimate for the number of bytes {@code
|
||||
* len} bytes could decode to.
|
||||
*/
|
||||
public int maxOutputSize(int len) {
|
||||
return len * 3 / 4 + 10;
|
||||
}
|
||||
|
||||
/**
|
||||
* Decode another block of input data.
|
||||
*
|
||||
* @return true if the state machine is still healthy. false if
|
||||
* bad base-64 data has been detected in the input stream.
|
||||
*/
|
||||
public boolean process(byte[] input, int offset, int len, boolean finish) {
|
||||
if (this.state == 6) return false;
|
||||
int p = offset;
|
||||
len += offset;
|
||||
// Using local variables makes the decoder about 12%
|
||||
// faster than if we manipulate the member variables in
|
||||
// the loop. (Even alphabet makes a measurable
|
||||
// difference, which is somewhat surprising to me since
|
||||
// the member variable is final.)
|
||||
int state = this.state;
|
||||
int value = this.value;
|
||||
int op = 0;
|
||||
final byte[] output = this.output;
|
||||
final int[] alphabet = this.alphabet;
|
||||
while (p < len) {
|
||||
// Try the fast path: we're starting a new tuple and the
|
||||
// next four bytes of the input stream are all data
|
||||
// bytes. This corresponds to going through states
|
||||
// 0-1-2-3-0. We expect to use this method for most of
|
||||
// the data.
|
||||
//
|
||||
// If any of the next four bytes of input are non-data
|
||||
// (whitespace, etc.), value will end up negative. (All
|
||||
// the non-data values in decode are small negative
|
||||
// numbers, so shifting any of them up and or'ing them
|
||||
// together will result in a value with its top bit set.)
|
||||
//
|
||||
// You can remove this whole block and the output should
|
||||
// be the same, just slower.
|
||||
if (state == 0) {
|
||||
while (p + 4 <= len &&
|
||||
(value = ((alphabet[input[p] & 0xff] << 18) |
|
||||
(alphabet[input[p + 1] & 0xff] << 12) |
|
||||
(alphabet[input[p + 2] & 0xff] << 6) |
|
||||
(alphabet[input[p + 3] & 0xff]))) >= 0) {
|
||||
output[op + 2] = (byte) value;
|
||||
output[op + 1] = (byte) (value >> 8);
|
||||
output[op] = (byte) (value >> 16);
|
||||
op += 3;
|
||||
p += 4;
|
||||
}
|
||||
if (p >= len) break;
|
||||
}
|
||||
// The fast path isn't available -- either we've read a
|
||||
// partial tuple, or the next four input bytes aren't all
|
||||
// data, or whatever. Fall back to the slower state
|
||||
// machine implementation.
|
||||
int d = alphabet[input[p++] & 0xff];
|
||||
switch (state) {
|
||||
case 0:
|
||||
if (d >= 0) {
|
||||
value = d;
|
||||
++state;
|
||||
} else if (d != SKIP) {
|
||||
this.state = 6;
|
||||
return false;
|
||||
}
|
||||
break;
|
||||
case 1:
|
||||
if (d >= 0) {
|
||||
value = (value << 6) | d;
|
||||
++state;
|
||||
} else if (d != SKIP) {
|
||||
this.state = 6;
|
||||
return false;
|
||||
}
|
||||
break;
|
||||
case 2:
|
||||
if (d >= 0) {
|
||||
value = (value << 6) | d;
|
||||
++state;
|
||||
} else if (d == EQUALS) {
|
||||
// Emit the last (partial) output tuple;
|
||||
// expect exactly one more padding character.
|
||||
output[op++] = (byte) (value >> 4);
|
||||
state = 4;
|
||||
} else if (d != SKIP) {
|
||||
this.state = 6;
|
||||
return false;
|
||||
}
|
||||
break;
|
||||
case 3:
|
||||
if (d >= 0) {
|
||||
// Emit the output triple and return to state 0.
|
||||
value = (value << 6) | d;
|
||||
output[op + 2] = (byte) value;
|
||||
output[op + 1] = (byte) (value >> 8);
|
||||
output[op] = (byte) (value >> 16);
|
||||
op += 3;
|
||||
state = 0;
|
||||
} else if (d == EQUALS) {
|
||||
// Emit the last (partial) output tuple;
|
||||
// expect no further data or padding characters.
|
||||
output[op + 1] = (byte) (value >> 2);
|
||||
output[op] = (byte) (value >> 10);
|
||||
op += 2;
|
||||
state = 5;
|
||||
} else if (d != SKIP) {
|
||||
this.state = 6;
|
||||
return false;
|
||||
}
|
||||
break;
|
||||
case 4:
|
||||
if (d == EQUALS) {
|
||||
++state;
|
||||
} else if (d != SKIP) {
|
||||
this.state = 6;
|
||||
return false;
|
||||
}
|
||||
break;
|
||||
case 5:
|
||||
if (d != SKIP) {
|
||||
this.state = 6;
|
||||
return false;
|
||||
}
|
||||
break;
|
||||
}
|
||||
}
|
||||
if (!finish) {
|
||||
// We're out of input, but a future call could provide
|
||||
// more.
|
||||
this.state = state;
|
||||
this.value = value;
|
||||
this.op = op;
|
||||
return true;
|
||||
}
|
||||
// Done reading input. Now figure out where we are left in
|
||||
// the state machine and finish up.
|
||||
switch (state) {
|
||||
case 0:
|
||||
// Output length is a multiple of three. Fine.
|
||||
break;
|
||||
case 1:
|
||||
// Read one extra input byte, which isn't enough to
|
||||
// make another output byte. Illegal.
|
||||
this.state = 6;
|
||||
return false;
|
||||
case 2:
|
||||
// Read two extra input bytes, enough to emit 1 more
|
||||
// output byte. Fine.
|
||||
output[op++] = (byte) (value >> 4);
|
||||
break;
|
||||
case 3:
|
||||
// Read three extra input bytes, enough to emit 2 more
|
||||
// output bytes. Fine.
|
||||
output[op++] = (byte) (value >> 10);
|
||||
output[op++] = (byte) (value >> 2);
|
||||
break;
|
||||
case 4:
|
||||
// Read one padding '=' when we expected 2. Illegal.
|
||||
this.state = 6;
|
||||
return false;
|
||||
case 5:
|
||||
// Read all the padding '='s we expected and no more.
|
||||
// Fine.
|
||||
break;
|
||||
}
|
||||
this.state = state;
|
||||
this.op = op;
|
||||
return true;
|
||||
}
|
||||
}
|
||||
|
||||
/* package */ static class Encoder extends Coder {
|
||||
/**
|
||||
* Emit a new line every this many output tuples. Corresponds to
|
||||
* a 76-character line length (the maximum allowable according to
|
||||
* <a href="http://www.ietf.org/rfc/rfc2045.txt">RFC 2045</a>).
|
||||
*/
|
||||
public static final int LINE_GROUPS = 19;
|
||||
/**
|
||||
* Lookup table for turning Base64 alphabet positions (6 bits)
|
||||
* into output bytes.
|
||||
*/
|
||||
private static final byte[] ENCODE = {
|
||||
'A', 'B', 'C', 'D', 'E', 'F', 'G', 'H', 'I', 'J', 'K', 'L', 'M', 'N', 'O', 'P',
|
||||
'Q', 'R', 'S', 'T', 'U', 'V', 'W', 'X', 'Y', 'Z', 'a', 'b', 'c', 'd', 'e', 'f',
|
||||
'g', 'h', 'i', 'j', 'k', 'l', 'm', 'n', 'o', 'p', 'q', 'r', 's', 't', 'u', 'v',
|
||||
'w', 'x', 'y', 'z', '0', '1', '2', '3', '4', '5', '6', '7', '8', '9', '+', '/',
|
||||
};
|
||||
/**
|
||||
* Lookup table for turning Base64 alphabet positions (6 bits)
|
||||
* into output bytes.
|
||||
*/
|
||||
private static final byte[] ENCODE_WEBSAFE = {
|
||||
'A', 'B', 'C', 'D', 'E', 'F', 'G', 'H', 'I', 'J', 'K', 'L', 'M', 'N', 'O', 'P',
|
||||
'Q', 'R', 'S', 'T', 'U', 'V', 'W', 'X', 'Y', 'Z', 'a', 'b', 'c', 'd', 'e', 'f',
|
||||
'g', 'h', 'i', 'j', 'k', 'l', 'm', 'n', 'o', 'p', 'q', 'r', 's', 't', 'u', 'v',
|
||||
'w', 'x', 'y', 'z', '0', '1', '2', '3', '4', '5', '6', '7', '8', '9', '-', '_',
|
||||
};
|
||||
final public boolean do_padding;
|
||||
final public boolean do_newline;
|
||||
final public boolean do_cr;
|
||||
final private byte[] tail;
|
||||
final private byte[] alphabet;
|
||||
/* package */ int tailLen;
|
||||
private int count;
|
||||
|
||||
public Encoder(int flags, byte[] output) {
|
||||
this.output = output;
|
||||
do_padding = (flags & NO_PADDING) == 0;
|
||||
do_newline = (flags & NO_WRAP) == 0;
|
||||
do_cr = (flags & CRLF) != 0;
|
||||
alphabet = ((flags & URL_SAFE) == 0) ? ENCODE : ENCODE_WEBSAFE;
|
||||
tail = new byte[2];
|
||||
tailLen = 0;
|
||||
count = do_newline ? LINE_GROUPS : -1;
|
||||
}
|
||||
|
||||
/**
|
||||
* @return an overestimate for the number of bytes {@code
|
||||
* len} bytes could encode to.
|
||||
*/
|
||||
public int maxOutputSize(int len) {
|
||||
return len * 8 / 5 + 10;
|
||||
}
|
||||
|
||||
public boolean process(byte[] input, int offset, int len, boolean finish) {
|
||||
// Using local variables makes the encoder about 9% faster.
|
||||
final byte[] alphabet = this.alphabet;
|
||||
final byte[] output = this.output;
|
||||
int op = 0;
|
||||
int count = this.count;
|
||||
int p = offset;
|
||||
len += offset;
|
||||
int v = -1;
|
||||
// First we need to concatenate the tail of the previous call
|
||||
// with any input bytes available now and see if we can empty
|
||||
// the tail.
|
||||
switch (tailLen) {
|
||||
case 0:
|
||||
// There was no tail.
|
||||
break;
|
||||
case 1:
|
||||
if (p + 2 <= len) {
|
||||
// A 1-byte tail with at least 2 bytes of
|
||||
// input available now.
|
||||
v = ((tail[0] & 0xff) << 16) |
|
||||
((input[p++] & 0xff) << 8) |
|
||||
(input[p++] & 0xff);
|
||||
tailLen = 0;
|
||||
}
|
||||
break;
|
||||
case 2:
|
||||
if (p + 1 <= len) {
|
||||
// A 2-byte tail with at least 1 byte of input.
|
||||
v = ((tail[0] & 0xff) << 16) |
|
||||
((tail[1] & 0xff) << 8) |
|
||||
(input[p++] & 0xff);
|
||||
tailLen = 0;
|
||||
}
|
||||
break;
|
||||
}
|
||||
if (v != -1) {
|
||||
output[op++] = alphabet[(v >> 18) & 0x3f];
|
||||
output[op++] = alphabet[(v >> 12) & 0x3f];
|
||||
output[op++] = alphabet[(v >> 6) & 0x3f];
|
||||
output[op++] = alphabet[v & 0x3f];
|
||||
if (--count == 0) {
|
||||
if (do_cr) output[op++] = '\r';
|
||||
output[op++] = '\n';
|
||||
count = LINE_GROUPS;
|
||||
}
|
||||
}
|
||||
// At this point either there is no tail, or there are fewer
|
||||
// than 3 bytes of input available.
|
||||
// The main loop, turning 3 input bytes into 4 output bytes on
|
||||
// each iteration.
|
||||
while (p + 3 <= len) {
|
||||
v = ((input[p] & 0xff) << 16) |
|
||||
((input[p + 1] & 0xff) << 8) |
|
||||
(input[p + 2] & 0xff);
|
||||
output[op] = alphabet[(v >> 18) & 0x3f];
|
||||
output[op + 1] = alphabet[(v >> 12) & 0x3f];
|
||||
output[op + 2] = alphabet[(v >> 6) & 0x3f];
|
||||
output[op + 3] = alphabet[v & 0x3f];
|
||||
p += 3;
|
||||
op += 4;
|
||||
if (--count == 0) {
|
||||
if (do_cr) output[op++] = '\r';
|
||||
output[op++] = '\n';
|
||||
count = LINE_GROUPS;
|
||||
}
|
||||
}
|
||||
if (finish) {
|
||||
// Finish up the tail of the input. Note that we need to
|
||||
// consume any bytes in tail before any bytes
|
||||
// remaining in input; there should be at most two bytes
|
||||
// total.
|
||||
if (p - tailLen == len - 1) {
|
||||
int t = 0;
|
||||
v = ((tailLen > 0 ? tail[t++] : input[p++]) & 0xff) << 4;
|
||||
tailLen -= t;
|
||||
output[op++] = alphabet[(v >> 6) & 0x3f];
|
||||
output[op++] = alphabet[v & 0x3f];
|
||||
if (do_padding) {
|
||||
output[op++] = '=';
|
||||
output[op++] = '=';
|
||||
}
|
||||
if (do_newline) {
|
||||
if (do_cr) output[op++] = '\r';
|
||||
output[op++] = '\n';
|
||||
}
|
||||
} else if (p - tailLen == len - 2) {
|
||||
int t = 0;
|
||||
v = (((tailLen > 1 ? tail[t++] : input[p++]) & 0xff) << 10) |
|
||||
(((tailLen > 0 ? tail[t++] : input[p++]) & 0xff) << 2);
|
||||
tailLen -= t;
|
||||
output[op++] = alphabet[(v >> 12) & 0x3f];
|
||||
output[op++] = alphabet[(v >> 6) & 0x3f];
|
||||
output[op++] = alphabet[v & 0x3f];
|
||||
if (do_padding) {
|
||||
output[op++] = '=';
|
||||
}
|
||||
if (do_newline) {
|
||||
if (do_cr) output[op++] = '\r';
|
||||
output[op++] = '\n';
|
||||
}
|
||||
} else if (do_newline && op > 0 && count != LINE_GROUPS) {
|
||||
if (do_cr) output[op++] = '\r';
|
||||
output[op++] = '\n';
|
||||
}
|
||||
assert tailLen == 0;
|
||||
assert p == len;
|
||||
} else {
|
||||
// Save the leftovers in tail to be consumed on the next
|
||||
// call to encodeInternal.
|
||||
if (p == len - 1) {
|
||||
tail[tailLen++] = input[p];
|
||||
} else if (p == len - 2) {
|
||||
tail[tailLen++] = input[p];
|
||||
tail[tailLen++] = input[p + 1];
|
||||
}
|
||||
}
|
||||
this.op = op;
|
||||
this.count = count;
|
||||
return true;
|
||||
}
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,51 @@
|
||||
/*
|
||||
* Copyright (C) 2013 The Android Open Source Project
|
||||
*
|
||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||
* you may not use this file except in compliance with the License.
|
||||
* You may obtain a copy of the License at
|
||||
*
|
||||
* http://www.apache.org/licenses/LICENSE-2.0
|
||||
*
|
||||
* Unless required by applicable law or agreed to in writing, software
|
||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
* See the License for the specific language governing permissions and
|
||||
* limitations under the License.
|
||||
*/
|
||||
package android.util;
|
||||
class ContainerHelpers {
|
||||
// This is Arrays.binarySearch(), but doesn't do any argument validation.
|
||||
static int binarySearch(int[] array, int size, int value) {
|
||||
int lo = 0;
|
||||
int hi = size - 1;
|
||||
while (lo <= hi) {
|
||||
final int mid = (lo + hi) >>> 1;
|
||||
final int midVal = array[mid];
|
||||
if (midVal < value) {
|
||||
lo = mid + 1;
|
||||
} else if (midVal > value) {
|
||||
hi = mid - 1;
|
||||
} else {
|
||||
return mid; // value found
|
||||
}
|
||||
}
|
||||
return ~lo; // value not present
|
||||
}
|
||||
static int binarySearch(long[] array, int size, long value) {
|
||||
int lo = 0;
|
||||
int hi = size - 1;
|
||||
while (lo <= hi) {
|
||||
final int mid = (lo + hi) >>> 1;
|
||||
final long midVal = array[mid];
|
||||
if (midVal < value) {
|
||||
lo = mid + 1;
|
||||
} else if (midVal > value) {
|
||||
hi = mid - 1;
|
||||
} else {
|
||||
return mid; // value found
|
||||
}
|
||||
}
|
||||
return ~lo; // value not present
|
||||
}
|
||||
}
|
||||
139
AndroidCompat/src/main/java/android/util/Log.java
Normal file
139
AndroidCompat/src/main/java/android/util/Log.java
Normal file
@@ -0,0 +1,139 @@
|
||||
//
|
||||
// Source code recreated from a .class file by IntelliJ IDEA
|
||||
// (powered by Fernflower decompiler)
|
||||
//
|
||||
|
||||
package android.util;
|
||||
|
||||
import org.slf4j.Logger;
|
||||
import org.slf4j.LoggerFactory;
|
||||
|
||||
import java.io.PrintWriter;
|
||||
import java.io.StringWriter;
|
||||
|
||||
public final class Log {
|
||||
public static final int ASSERT = 7;
|
||||
public static final int DEBUG = 3;
|
||||
public static final int ERROR = 6;
|
||||
public static final int INFO = 4;
|
||||
public static final int VERBOSE = 2;
|
||||
public static final int WARN = 5;
|
||||
|
||||
private static Logger logger = LoggerFactory.getLogger(Log.class);
|
||||
|
||||
public static int v(String tag, String msg) {
|
||||
return log(VERBOSE, tag, msg);
|
||||
}
|
||||
|
||||
public static int v(String tag, String msg, Throwable tr) {
|
||||
return log(VERBOSE, tag, msg, tr);
|
||||
}
|
||||
|
||||
public static int d(String tag, String msg) {
|
||||
return log(DEBUG, tag, msg);
|
||||
}
|
||||
|
||||
public static int d(String tag, String msg, Throwable tr) {
|
||||
return log(DEBUG, tag, msg, tr);
|
||||
}
|
||||
|
||||
public static int i(String tag, String msg) {
|
||||
return log(INFO, tag, msg);
|
||||
}
|
||||
|
||||
public static int i(String tag, String msg, Throwable tr) {
|
||||
return log(INFO, tag, msg, tr);
|
||||
}
|
||||
|
||||
public static int w(String tag, String msg) {
|
||||
return log(WARN, tag, msg);
|
||||
}
|
||||
|
||||
public static int w(String tag, String msg, Throwable tr) {
|
||||
return log(WARN, tag, msg, tr);
|
||||
}
|
||||
|
||||
public static boolean isLoggable(String var0, int var1) {
|
||||
return true;
|
||||
}
|
||||
|
||||
public static int w(String tag, Throwable tr) {
|
||||
return log(WARN, tag, tr);
|
||||
}
|
||||
|
||||
public static int e(String tag, String msg) {
|
||||
return log(ERROR, tag, msg);
|
||||
}
|
||||
|
||||
public static int e(String tag, String msg, Throwable tr) {
|
||||
return log(ERROR, tag, msg, tr);
|
||||
}
|
||||
|
||||
//Level?
|
||||
public static int wtf(String tag, String msg) {
|
||||
return log(ERROR, tag, msg);
|
||||
}
|
||||
public static int wtf(String tag, Throwable tr) {
|
||||
return log(ERROR, tag, tr);
|
||||
}
|
||||
public static int wtf(String tag, String msg, Throwable tr) {
|
||||
return log(ERROR, tag, msg, tr);
|
||||
}
|
||||
|
||||
public static String getStackTraceString(Throwable tr) {
|
||||
final StringWriter sw = new StringWriter();
|
||||
final PrintWriter pw = new PrintWriter(sw, true);
|
||||
tr.printStackTrace(pw);
|
||||
return sw.getBuffer().toString();
|
||||
}
|
||||
|
||||
public static int println(int priority, String tag, String msg) {
|
||||
return log(priority, tag, msg);
|
||||
}
|
||||
|
||||
private static int log(int level, String tag, String msg) {
|
||||
logger.info(formatLog(level, tag, msg));
|
||||
return tag.length() + msg.length(); //Not accurate, but never used anyways
|
||||
}
|
||||
|
||||
private static int log(int level, String tag, Throwable t) {
|
||||
return log(level, tag, "An exception occured!", t);
|
||||
}
|
||||
|
||||
private static int log(int level, String tag, String msg, Throwable t) {
|
||||
logger.info(formatLog(level, tag, msg), t);
|
||||
return tag.length() + msg.length(); //Not accurate, but never used anyways
|
||||
}
|
||||
|
||||
private static String formatLog(int level, String tag, String msg) {
|
||||
StringBuilder first = new StringBuilder("[");
|
||||
switch(level) {
|
||||
case ASSERT:
|
||||
first.append("ASSERT");
|
||||
break;
|
||||
case DEBUG:
|
||||
first.append("DEBUG");
|
||||
break;
|
||||
case ERROR:
|
||||
first.append("ERROR");
|
||||
break;
|
||||
case INFO:
|
||||
first.append("INFO");
|
||||
break;
|
||||
case VERBOSE:
|
||||
first.append("VERBOSE");
|
||||
break;
|
||||
case WARN:
|
||||
first.append("WARN");
|
||||
break;
|
||||
default:
|
||||
first.append("UNKNOWN");
|
||||
break;
|
||||
}
|
||||
first.append("] ");
|
||||
first.append(tag);
|
||||
first.append(": ");
|
||||
first.append(msg);
|
||||
return first.toString();
|
||||
}
|
||||
}
|
||||
482
AndroidCompat/src/main/java/android/util/MapCollections.java
Normal file
482
AndroidCompat/src/main/java/android/util/MapCollections.java
Normal file
@@ -0,0 +1,482 @@
|
||||
/*
|
||||
* Copyright (C) 2013 The Android Open Source Project
|
||||
*
|
||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||
* you may not use this file except in compliance with the License.
|
||||
* You may obtain a copy of the License at
|
||||
*
|
||||
* http://www.apache.org/licenses/LICENSE-2.0
|
||||
*
|
||||
* Unless required by applicable law or agreed to in writing, software
|
||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
* See the License for the specific language governing permissions and
|
||||
* limitations under the License.
|
||||
*/
|
||||
package android.util;
|
||||
import java.lang.reflect.Array;
|
||||
import java.util.*;
|
||||
|
||||
/**
|
||||
* Helper for writing standard Java collection interfaces to a data
|
||||
* structure like {@link ArrayMap}.
|
||||
* @hide
|
||||
*/
|
||||
abstract class MapCollections<K, V> {
|
||||
EntrySet mEntrySet;
|
||||
KeySet mKeySet;
|
||||
ValuesCollection mValues;
|
||||
final class ArrayIterator<T> implements Iterator<T> {
|
||||
final int mOffset;
|
||||
int mSize;
|
||||
int mIndex;
|
||||
boolean mCanRemove = false;
|
||||
ArrayIterator(int offset) {
|
||||
mOffset = offset;
|
||||
mSize = colGetSize();
|
||||
}
|
||||
@Override
|
||||
public boolean hasNext() {
|
||||
return mIndex < mSize;
|
||||
}
|
||||
@Override
|
||||
public T next() {
|
||||
if (!hasNext()) throw new NoSuchElementException();
|
||||
Object res = colGetEntry(mIndex, mOffset);
|
||||
mIndex++;
|
||||
mCanRemove = true;
|
||||
return (T)res;
|
||||
}
|
||||
@Override
|
||||
public void remove() {
|
||||
if (!mCanRemove) {
|
||||
throw new IllegalStateException();
|
||||
}
|
||||
mIndex--;
|
||||
mSize--;
|
||||
mCanRemove = false;
|
||||
colRemoveAt(mIndex);
|
||||
}
|
||||
}
|
||||
final class MapIterator implements Iterator<Map.Entry<K, V>>, Map.Entry<K, V> {
|
||||
int mEnd;
|
||||
int mIndex;
|
||||
boolean mEntryValid = false;
|
||||
MapIterator() {
|
||||
mEnd = colGetSize() - 1;
|
||||
mIndex = -1;
|
||||
}
|
||||
@Override
|
||||
public boolean hasNext() {
|
||||
return mIndex < mEnd;
|
||||
}
|
||||
@Override
|
||||
public Map.Entry<K, V> next() {
|
||||
if (!hasNext()) throw new NoSuchElementException();
|
||||
mIndex++;
|
||||
mEntryValid = true;
|
||||
return this;
|
||||
}
|
||||
@Override
|
||||
public void remove() {
|
||||
if (!mEntryValid) {
|
||||
throw new IllegalStateException();
|
||||
}
|
||||
colRemoveAt(mIndex);
|
||||
mIndex--;
|
||||
mEnd--;
|
||||
mEntryValid = false;
|
||||
}
|
||||
@Override
|
||||
public K getKey() {
|
||||
if (!mEntryValid) {
|
||||
throw new IllegalStateException(
|
||||
"This container does not support retaining Map.Entry objects");
|
||||
}
|
||||
return (K)colGetEntry(mIndex, 0);
|
||||
}
|
||||
@Override
|
||||
public V getValue() {
|
||||
if (!mEntryValid) {
|
||||
throw new IllegalStateException(
|
||||
"This container does not support retaining Map.Entry objects");
|
||||
}
|
||||
return (V)colGetEntry(mIndex, 1);
|
||||
}
|
||||
@Override
|
||||
public V setValue(V object) {
|
||||
if (!mEntryValid) {
|
||||
throw new IllegalStateException(
|
||||
"This container does not support retaining Map.Entry objects");
|
||||
}
|
||||
return colSetValue(mIndex, object);
|
||||
}
|
||||
@Override
|
||||
public final boolean equals(Object o) {
|
||||
if (!mEntryValid) {
|
||||
throw new IllegalStateException(
|
||||
"This container does not support retaining Map.Entry objects");
|
||||
}
|
||||
if (!(o instanceof Map.Entry)) {
|
||||
return false;
|
||||
}
|
||||
Map.Entry<?, ?> e = (Map.Entry<?, ?>) o;
|
||||
return Objects.equals(e.getKey(), colGetEntry(mIndex, 0))
|
||||
&& Objects.equals(e.getValue(), colGetEntry(mIndex, 1));
|
||||
}
|
||||
@Override
|
||||
public final int hashCode() {
|
||||
if (!mEntryValid) {
|
||||
throw new IllegalStateException(
|
||||
"This container does not support retaining Map.Entry objects");
|
||||
}
|
||||
final Object key = colGetEntry(mIndex, 0);
|
||||
final Object value = colGetEntry(mIndex, 1);
|
||||
return (key == null ? 0 : key.hashCode()) ^
|
||||
(value == null ? 0 : value.hashCode());
|
||||
}
|
||||
@Override
|
||||
public final String toString() {
|
||||
return getKey() + "=" + getValue();
|
||||
}
|
||||
}
|
||||
final class EntrySet implements Set<Map.Entry<K, V>> {
|
||||
@Override
|
||||
public boolean add(Map.Entry<K, V> object) {
|
||||
throw new UnsupportedOperationException();
|
||||
}
|
||||
@Override
|
||||
public boolean addAll(Collection<? extends Map.Entry<K, V>> collection) {
|
||||
int oldSize = colGetSize();
|
||||
for (Map.Entry<K, V> entry : collection) {
|
||||
colPut(entry.getKey(), entry.getValue());
|
||||
}
|
||||
return oldSize != colGetSize();
|
||||
}
|
||||
@Override
|
||||
public void clear() {
|
||||
colClear();
|
||||
}
|
||||
@Override
|
||||
public boolean contains(Object o) {
|
||||
if (!(o instanceof Map.Entry))
|
||||
return false;
|
||||
Map.Entry<?, ?> e = (Map.Entry<?, ?>) o;
|
||||
int index = colIndexOfKey(e.getKey());
|
||||
if (index < 0) {
|
||||
return false;
|
||||
}
|
||||
Object foundVal = colGetEntry(index, 1);
|
||||
return Objects.equals(foundVal, e.getValue());
|
||||
}
|
||||
@Override
|
||||
public boolean containsAll(Collection<?> collection) {
|
||||
Iterator<?> it = collection.iterator();
|
||||
while (it.hasNext()) {
|
||||
if (!contains(it.next())) {
|
||||
return false;
|
||||
}
|
||||
}
|
||||
return true;
|
||||
}
|
||||
@Override
|
||||
public boolean isEmpty() {
|
||||
return colGetSize() == 0;
|
||||
}
|
||||
@Override
|
||||
public Iterator<Map.Entry<K, V>> iterator() {
|
||||
return new MapIterator();
|
||||
}
|
||||
@Override
|
||||
public boolean remove(Object object) {
|
||||
throw new UnsupportedOperationException();
|
||||
}
|
||||
@Override
|
||||
public boolean removeAll(Collection<?> collection) {
|
||||
throw new UnsupportedOperationException();
|
||||
}
|
||||
@Override
|
||||
public boolean retainAll(Collection<?> collection) {
|
||||
throw new UnsupportedOperationException();
|
||||
}
|
||||
@Override
|
||||
public int size() {
|
||||
return colGetSize();
|
||||
}
|
||||
@Override
|
||||
public Object[] toArray() {
|
||||
throw new UnsupportedOperationException();
|
||||
}
|
||||
@Override
|
||||
public <T> T[] toArray(T[] array) {
|
||||
throw new UnsupportedOperationException();
|
||||
}
|
||||
@Override
|
||||
public boolean equals(Object object) {
|
||||
return equalsSetHelper(this, object);
|
||||
}
|
||||
@Override
|
||||
public int hashCode() {
|
||||
int result = 0;
|
||||
for (int i=colGetSize()-1; i>=0; i--) {
|
||||
final Object key = colGetEntry(i, 0);
|
||||
final Object value = colGetEntry(i, 1);
|
||||
result += ( (key == null ? 0 : key.hashCode()) ^
|
||||
(value == null ? 0 : value.hashCode()) );
|
||||
}
|
||||
return result;
|
||||
}
|
||||
};
|
||||
final class KeySet implements Set<K> {
|
||||
@Override
|
||||
public boolean add(K object) {
|
||||
throw new UnsupportedOperationException();
|
||||
}
|
||||
@Override
|
||||
public boolean addAll(Collection<? extends K> collection) {
|
||||
throw new UnsupportedOperationException();
|
||||
}
|
||||
@Override
|
||||
public void clear() {
|
||||
colClear();
|
||||
}
|
||||
@Override
|
||||
public boolean contains(Object object) {
|
||||
return colIndexOfKey(object) >= 0;
|
||||
}
|
||||
@Override
|
||||
public boolean containsAll(Collection<?> collection) {
|
||||
return containsAllHelper(colGetMap(), collection);
|
||||
}
|
||||
@Override
|
||||
public boolean isEmpty() {
|
||||
return colGetSize() == 0;
|
||||
}
|
||||
@Override
|
||||
public Iterator<K> iterator() {
|
||||
return new ArrayIterator<K>(0);
|
||||
}
|
||||
@Override
|
||||
public boolean remove(Object object) {
|
||||
int index = colIndexOfKey(object);
|
||||
if (index >= 0) {
|
||||
colRemoveAt(index);
|
||||
return true;
|
||||
}
|
||||
return false;
|
||||
}
|
||||
@Override
|
||||
public boolean removeAll(Collection<?> collection) {
|
||||
return removeAllHelper(colGetMap(), collection);
|
||||
}
|
||||
@Override
|
||||
public boolean retainAll(Collection<?> collection) {
|
||||
return retainAllHelper(colGetMap(), collection);
|
||||
}
|
||||
@Override
|
||||
public int size() {
|
||||
return colGetSize();
|
||||
}
|
||||
@Override
|
||||
public Object[] toArray() {
|
||||
return toArrayHelper(0);
|
||||
}
|
||||
@Override
|
||||
public <T> T[] toArray(T[] array) {
|
||||
return toArrayHelper(array, 0);
|
||||
}
|
||||
@Override
|
||||
public boolean equals(Object object) {
|
||||
return equalsSetHelper(this, object);
|
||||
}
|
||||
@Override
|
||||
public int hashCode() {
|
||||
int result = 0;
|
||||
for (int i=colGetSize()-1; i>=0; i--) {
|
||||
Object obj = colGetEntry(i, 0);
|
||||
result += obj == null ? 0 : obj.hashCode();
|
||||
}
|
||||
return result;
|
||||
}
|
||||
};
|
||||
final class ValuesCollection implements Collection<V> {
|
||||
@Override
|
||||
public boolean add(V object) {
|
||||
throw new UnsupportedOperationException();
|
||||
}
|
||||
@Override
|
||||
public boolean addAll(Collection<? extends V> collection) {
|
||||
throw new UnsupportedOperationException();
|
||||
}
|
||||
@Override
|
||||
public void clear() {
|
||||
colClear();
|
||||
}
|
||||
@Override
|
||||
public boolean contains(Object object) {
|
||||
return colIndexOfValue(object) >= 0;
|
||||
}
|
||||
@Override
|
||||
public boolean containsAll(Collection<?> collection) {
|
||||
Iterator<?> it = collection.iterator();
|
||||
while (it.hasNext()) {
|
||||
if (!contains(it.next())) {
|
||||
return false;
|
||||
}
|
||||
}
|
||||
return true;
|
||||
}
|
||||
@Override
|
||||
public boolean isEmpty() {
|
||||
return colGetSize() == 0;
|
||||
}
|
||||
@Override
|
||||
public Iterator<V> iterator() {
|
||||
return new ArrayIterator<V>(1);
|
||||
}
|
||||
@Override
|
||||
public boolean remove(Object object) {
|
||||
int index = colIndexOfValue(object);
|
||||
if (index >= 0) {
|
||||
colRemoveAt(index);
|
||||
return true;
|
||||
}
|
||||
return false;
|
||||
}
|
||||
@Override
|
||||
public boolean removeAll(Collection<?> collection) {
|
||||
int N = colGetSize();
|
||||
boolean changed = false;
|
||||
for (int i=0; i<N; i++) {
|
||||
Object cur = colGetEntry(i, 1);
|
||||
if (collection.contains(cur)) {
|
||||
colRemoveAt(i);
|
||||
i--;
|
||||
N--;
|
||||
changed = true;
|
||||
}
|
||||
}
|
||||
return changed;
|
||||
}
|
||||
@Override
|
||||
public boolean retainAll(Collection<?> collection) {
|
||||
int N = colGetSize();
|
||||
boolean changed = false;
|
||||
for (int i=0; i<N; i++) {
|
||||
Object cur = colGetEntry(i, 1);
|
||||
if (!collection.contains(cur)) {
|
||||
colRemoveAt(i);
|
||||
i--;
|
||||
N--;
|
||||
changed = true;
|
||||
}
|
||||
}
|
||||
return changed;
|
||||
}
|
||||
@Override
|
||||
public int size() {
|
||||
return colGetSize();
|
||||
}
|
||||
@Override
|
||||
public Object[] toArray() {
|
||||
return toArrayHelper(1);
|
||||
}
|
||||
@Override
|
||||
public <T> T[] toArray(T[] array) {
|
||||
return toArrayHelper(array, 1);
|
||||
}
|
||||
};
|
||||
public static <K, V> boolean containsAllHelper(Map<K, V> map, Collection<?> collection) {
|
||||
Iterator<?> it = collection.iterator();
|
||||
while (it.hasNext()) {
|
||||
if (!map.containsKey(it.next())) {
|
||||
return false;
|
||||
}
|
||||
}
|
||||
return true;
|
||||
}
|
||||
public static <K, V> boolean removeAllHelper(Map<K, V> map, Collection<?> collection) {
|
||||
int oldSize = map.size();
|
||||
Iterator<?> it = collection.iterator();
|
||||
while (it.hasNext()) {
|
||||
map.remove(it.next());
|
||||
}
|
||||
return oldSize != map.size();
|
||||
}
|
||||
public static <K, V> boolean retainAllHelper(Map<K, V> map, Collection<?> collection) {
|
||||
int oldSize = map.size();
|
||||
Iterator<K> it = map.keySet().iterator();
|
||||
while (it.hasNext()) {
|
||||
if (!collection.contains(it.next())) {
|
||||
it.remove();
|
||||
}
|
||||
}
|
||||
return oldSize != map.size();
|
||||
}
|
||||
public Object[] toArrayHelper(int offset) {
|
||||
final int N = colGetSize();
|
||||
Object[] result = new Object[N];
|
||||
for (int i=0; i<N; i++) {
|
||||
result[i] = colGetEntry(i, offset);
|
||||
}
|
||||
return result;
|
||||
}
|
||||
public <T> T[] toArrayHelper(T[] array, int offset) {
|
||||
final int N = colGetSize();
|
||||
if (array.length < N) {
|
||||
@SuppressWarnings("unchecked") T[] newArray
|
||||
= (T[]) Array.newInstance(array.getClass().getComponentType(), N);
|
||||
array = newArray;
|
||||
}
|
||||
for (int i=0; i<N; i++) {
|
||||
array[i] = (T)colGetEntry(i, offset);
|
||||
}
|
||||
if (array.length > N) {
|
||||
array[N] = null;
|
||||
}
|
||||
return array;
|
||||
}
|
||||
public static <T> boolean equalsSetHelper(Set<T> set, Object object) {
|
||||
if (set == object) {
|
||||
return true;
|
||||
}
|
||||
if (object instanceof Set) {
|
||||
Set<?> s = (Set<?>) object;
|
||||
try {
|
||||
return set.size() == s.size() && set.containsAll(s);
|
||||
} catch (NullPointerException ignored) {
|
||||
return false;
|
||||
} catch (ClassCastException ignored) {
|
||||
return false;
|
||||
}
|
||||
}
|
||||
return false;
|
||||
}
|
||||
public Set<Map.Entry<K, V>> getEntrySet() {
|
||||
if (mEntrySet == null) {
|
||||
mEntrySet = new EntrySet();
|
||||
}
|
||||
return mEntrySet;
|
||||
}
|
||||
public Set<K> getKeySet() {
|
||||
if (mKeySet == null) {
|
||||
mKeySet = new KeySet();
|
||||
}
|
||||
return mKeySet;
|
||||
}
|
||||
public Collection<V> getValues() {
|
||||
if (mValues == null) {
|
||||
mValues = new ValuesCollection();
|
||||
}
|
||||
return mValues;
|
||||
}
|
||||
protected abstract int colGetSize();
|
||||
protected abstract Object colGetEntry(int index, int offset);
|
||||
protected abstract int colIndexOfKey(Object key);
|
||||
protected abstract int colIndexOfValue(Object key);
|
||||
protected abstract Map<K, V> colGetMap();
|
||||
protected abstract void colPut(K key, V value);
|
||||
protected abstract V colSetValue(int index, V value);
|
||||
protected abstract void colRemoveAt(int index);
|
||||
protected abstract void colClear();
|
||||
}
|
||||
Reference in New Issue
Block a user