|  | @@ -0,0 +1,105 @@
 | 
	
		
			
				|  |  | +/*
 | 
	
		
			
				|  |  | + * Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
 | 
	
		
			
				|  |  | + * or more contributor license agreements. Licensed under the Elastic License
 | 
	
		
			
				|  |  | + * 2.0 and the Server Side Public License, v 1; you may not use this file except
 | 
	
		
			
				|  |  | + * in compliance with, at your election, the Elastic License 2.0 or the Server
 | 
	
		
			
				|  |  | + * Side Public License, v 1.
 | 
	
		
			
				|  |  | + */
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +package org.elasticsearch.common.util;
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +import org.apache.lucene.util.BytesRef;
 | 
	
		
			
				|  |  | +import org.apache.lucene.util.RamUsageEstimator;
 | 
	
		
			
				|  |  | +import org.elasticsearch.common.bytes.BytesReference;
 | 
	
		
			
				|  |  | +import org.elasticsearch.common.bytes.ReleasableBytesReference;
 | 
	
		
			
				|  |  | +import org.elasticsearch.common.io.stream.StreamInput;
 | 
	
		
			
				|  |  | +import org.elasticsearch.common.io.stream.StreamOutput;
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +import java.io.IOException;
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +import static org.elasticsearch.common.util.BigArrays.indexIsInt;
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +public class ReleasableByteArray implements ByteArray {
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +    private static final long SHALLOW_SIZE = RamUsageEstimator.shallowSizeOfInstance(ReleasableByteArray.class);
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +    private final ReleasableBytesReference ref;
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +    ReleasableByteArray(StreamInput in) throws IOException {
 | 
	
		
			
				|  |  | +        this.ref = in.readReleasableBytesReference();
 | 
	
		
			
				|  |  | +    }
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +    @Override
 | 
	
		
			
				|  |  | +    public void writeTo(StreamOutput out) throws IOException {
 | 
	
		
			
				|  |  | +        out.writeBytesReference(ref);
 | 
	
		
			
				|  |  | +    }
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +    @Override
 | 
	
		
			
				|  |  | +    public long size() {
 | 
	
		
			
				|  |  | +        return ref.length() / Byte.BYTES;
 | 
	
		
			
				|  |  | +    }
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +    @Override
 | 
	
		
			
				|  |  | +    public byte get(long index) {
 | 
	
		
			
				|  |  | +        assert indexIsInt(index);
 | 
	
		
			
				|  |  | +        return ref.get((int) index);
 | 
	
		
			
				|  |  | +    }
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +    @Override
 | 
	
		
			
				|  |  | +    public boolean get(long index, int len, BytesRef ref) {
 | 
	
		
			
				|  |  | +        assert indexIsInt(index);
 | 
	
		
			
				|  |  | +        BytesReference sliced = this.ref.slice((int) index, len);
 | 
	
		
			
				|  |  | +        if (sliced.length() != 0) {
 | 
	
		
			
				|  |  | +            ref.offset = sliced.arrayOffset();
 | 
	
		
			
				|  |  | +            ref.length = sliced.length();
 | 
	
		
			
				|  |  | +            ref.bytes = sliced.array();
 | 
	
		
			
				|  |  | +            return true;
 | 
	
		
			
				|  |  | +        } else {
 | 
	
		
			
				|  |  | +            return false;
 | 
	
		
			
				|  |  | +        }
 | 
	
		
			
				|  |  | +    }
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +    @Override
 | 
	
		
			
				|  |  | +    public byte set(long index, byte value) {
 | 
	
		
			
				|  |  | +        throw new UnsupportedOperationException();
 | 
	
		
			
				|  |  | +    }
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +    @Override
 | 
	
		
			
				|  |  | +    public void set(long index, byte[] buf, int offset, int len) {
 | 
	
		
			
				|  |  | +        throw new UnsupportedOperationException();
 | 
	
		
			
				|  |  | +    }
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +    @Override
 | 
	
		
			
				|  |  | +    public void fill(long fromIndex, long toIndex, byte value) {
 | 
	
		
			
				|  |  | +        throw new UnsupportedOperationException();
 | 
	
		
			
				|  |  | +    }
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +    @Override
 | 
	
		
			
				|  |  | +    public boolean hasArray() {
 | 
	
		
			
				|  |  | +        return ref.hasArray();
 | 
	
		
			
				|  |  | +    }
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +    @Override
 | 
	
		
			
				|  |  | +    public byte[] array() {
 | 
	
		
			
				|  |  | +        // The assumption of this method is that the returned array has valid entries starting from slot 0 and
 | 
	
		
			
				|  |  | +        // this isn't case when just returning the array from ReleasableBytesReference#array().
 | 
	
		
			
				|  |  | +        // The interface that this class implements should have something like an arrayOffset() method,
 | 
	
		
			
				|  |  | +        // so that callers know from what array offset the first actual byte starts.
 | 
	
		
			
				|  |  | +        throw new UnsupportedOperationException();
 | 
	
		
			
				|  |  | +    }
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +    @Override
 | 
	
		
			
				|  |  | +    public long ramBytesUsed() {
 | 
	
		
			
				|  |  | +        /*
 | 
	
		
			
				|  |  | +         * If we return the size of the buffer that we've sliced
 | 
	
		
			
				|  |  | +         * we're likely to double count things.
 | 
	
		
			
				|  |  | +         */
 | 
	
		
			
				|  |  | +        return SHALLOW_SIZE;
 | 
	
		
			
				|  |  | +    }
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +    @Override
 | 
	
		
			
				|  |  | +    public void close() {
 | 
	
		
			
				|  |  | +        ref.decRef();
 | 
	
		
			
				|  |  | +    }
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +}
 |