org.iq80.leveldb.util.Slices Maven / Gradle / Ivy
/*
* Copyright (C) 2011 the original author or authors.
* See the notice.md file distributed with this work for additional
* information regarding copyright ownership.
*
* 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 org.iq80.leveldb.util;
import com.google.common.base.Preconditions;
import java.nio.ByteBuffer;
import java.nio.ByteOrder;
import java.nio.CharBuffer;
import java.nio.charset.CharacterCodingException;
import java.nio.charset.Charset;
import java.nio.charset.CharsetDecoder;
import java.nio.charset.CharsetEncoder;
import java.nio.charset.CoderResult;
import java.nio.charset.CodingErrorAction;
import java.util.IdentityHashMap;
import java.util.Map;
public final class Slices
{
public static Slice readLengthPrefixedBytes(SliceInput sliceInput)
{
int length = VariableLengthQuantity.readVariableLengthInt(sliceInput);
return sliceInput.readBytes(length);
}
public static void writeLengthPrefixedBytes(SliceOutput sliceOutput, Slice value)
{
VariableLengthQuantity.writeVariableLengthInt(value.length(), sliceOutput);
sliceOutput.writeBytes(value);
}
/**
* A buffer whose capacity is {@code 0}.
*/
public static final Slice EMPTY_SLICE = new Slice(0);
private Slices()
{
}
public static Slice ensureSize(Slice existingSlice, int minWritableBytes)
{
if (existingSlice == null) {
existingSlice = EMPTY_SLICE;
}
if (minWritableBytes <= existingSlice.length()) {
return existingSlice;
}
int newCapacity;
if (existingSlice.length() == 0) {
newCapacity = 1;
}
else {
newCapacity = existingSlice.length();
}
int minNewCapacity = existingSlice.length() + minWritableBytes;
while (newCapacity < minNewCapacity) {
newCapacity <<= 1;
}
Slice newSlice = allocate(newCapacity);
newSlice.setBytes(0, existingSlice, 0, existingSlice.length());
return newSlice;
}
public static Slice allocate(int capacity)
{
if (capacity == 0) {
return EMPTY_SLICE;
}
return new Slice(capacity);
}
public static Slice wrappedBuffer(byte[] array)
{
if (array.length == 0) {
return EMPTY_SLICE;
}
return new Slice(array);
}
public static Slice copiedBuffer(ByteBuffer source, int sourceOffset, int length)
{
Preconditions.checkNotNull(source, "source is null");
int newPosition = source.position() + sourceOffset;
return copiedBuffer((ByteBuffer) source.duplicate().order(ByteOrder.LITTLE_ENDIAN).clear().limit(newPosition + length).position(newPosition));
}
public static Slice copiedBuffer(ByteBuffer source)
{
Preconditions.checkNotNull(source, "source is null");
Slice copy = allocate(source.limit() - source.position());
copy.setBytes(0, source.duplicate().order(ByteOrder.LITTLE_ENDIAN));
return copy;
}
public static Slice copiedBuffer(String string, Charset charset)
{
Preconditions.checkNotNull(string, "string is null");
Preconditions.checkNotNull(charset, "charset is null");
return wrappedBuffer(string.getBytes(charset));
}
public static ByteBuffer encodeString(CharBuffer src, Charset charset)
{
CharsetEncoder encoder = getEncoder(charset);
ByteBuffer dst = ByteBuffer.allocate(
(int) ((double) src.remaining() * encoder.maxBytesPerChar()));
try {
CoderResult cr = encoder.encode(src, dst, true);
if (!cr.isUnderflow()) {
cr.throwException();
}
cr = encoder.flush(dst);
if (!cr.isUnderflow()) {
cr.throwException();
}
}
catch (CharacterCodingException x) {
throw new IllegalStateException(x);
}
dst.flip();
return dst;
}
public static String decodeString(ByteBuffer src, Charset charset)
{
CharsetDecoder decoder = getDecoder(charset);
CharBuffer dst = CharBuffer.allocate(
(int) ((double) src.remaining() * decoder.maxCharsPerByte()));
try {
CoderResult cr = decoder.decode(src, dst, true);
if (!cr.isUnderflow()) {
cr.throwException();
}
cr = decoder.flush(dst);
if (!cr.isUnderflow()) {
cr.throwException();
}
}
catch (CharacterCodingException x) {
throw new IllegalStateException(x);
}
return dst.flip().toString();
}
private static final ThreadLocal