All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.neo4j.kernel.impl.api.state.AppendOnlyValuesContainer Maven / Gradle / Ivy

/*
 * Copyright (c) "Neo4j"
 * Neo4j Sweden AB [https://neo4j.com]
 *
 * This file is part of Neo4j.
 *
 * Neo4j is free software: you can redistribute it and/or modify
 * it under the terms of the GNU General Public License as published by
 * the Free Software Foundation, either version 3 of the License, or
 * (at your option) any later version.
 *
 * This program is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 * GNU General Public License for more details.
 *
 * You should have received a copy of the GNU General Public License
 * along with this program.  If not, see .
 */
package org.neo4j.kernel.impl.api.state;

import static java.lang.Math.max;
import static java.util.Objects.requireNonNull;
import static org.neo4j.util.Preconditions.checkArgument;
import static org.neo4j.util.Preconditions.checkState;
import static org.neo4j.values.storable.ValueByteBufferCodec.VALUE_TYPES;

import java.nio.ByteBuffer;
import java.util.ArrayList;
import java.util.List;
import org.neo4j.io.ByteUnit;
import org.neo4j.kernel.impl.util.collection.Memory;
import org.neo4j.kernel.impl.util.collection.MemoryAllocator;
import org.neo4j.memory.MemoryTracker;
import org.neo4j.util.VisibleForTesting;
import org.neo4j.values.storable.Value;
import org.neo4j.values.storable.ValueByteBufferCodec;
import org.neo4j.values.storable.ValueByteBufferCodec.Writer;

public class AppendOnlyValuesContainer implements ValuesContainer {
    private static final int CHUNK_SIZE = (int) ByteUnit.kibiBytes(512);
    private static final int REMOVED = 0xFF;

    private final int chunkSize;
    private final List chunks = new ArrayList<>();
    private final List allocated = new ArrayList<>();
    private final Writer writer;
    private final MemoryAllocator allocator;
    private final MemoryTracker memoryTracker;
    private ByteBuffer currentChunk;
    private boolean closed;

    public AppendOnlyValuesContainer(MemoryAllocator allocator, MemoryTracker memoryTracker) {
        this(CHUNK_SIZE, allocator, memoryTracker);
    }

    @VisibleForTesting
    AppendOnlyValuesContainer(int chunkSize, MemoryAllocator allocator, MemoryTracker memoryTracker) {
        this.chunkSize = chunkSize;
        this.allocator = allocator;
        this.memoryTracker = memoryTracker;
        this.writer = new Writer(chunkSize, new OffHeapByteBufferAllocator(allocator, memoryTracker));
    }

    @Override
    public long add(Value value) {
        assertNotClosed();
        requireNonNull(value, "value cannot be null");
        final ByteBuffer buf = writer.write(value);
        if (currentChunk == null || buf.remaining() > currentChunk.remaining()) {
            currentChunk = addNewChunk(max(chunkSize, buf.remaining()));
        }

        final long ref = ((chunks.size() - 1L) << 32) | currentChunk.position();
        currentChunk.put(buf);
        return ref;
    }

    @Override
    public Value get(long ref) {
        assertNotClosed();
        final int chunkIdx = (int) (ref >>> 32);
        int offset = (int) ref;

        checkArgument(
                chunkIdx >= 0 && chunkIdx < chunks.size(),
                "invalid chunk idx %d (total #%d chunks), ref: 0x%X",
                chunkIdx,
                chunks.size(),
                ref);
        final ByteBuffer chunk = chunks.get(chunkIdx);
        checkArgument(offset >= 0 && offset < chunk.position(), "invalid chunk offset (%d), ref: 0x%X", offset, ref);
        final int typeId = chunk.get(offset) & 0xFF;
        checkArgument(typeId != REMOVED, "element is already removed, ref: 0x%X", ref);
        checkArgument(typeId < VALUE_TYPES.length, "invaling typeId (%d) for ref 0x%X", typeId, ref);
        offset++;

        final ValueByteBufferCodec.ValueType type = VALUE_TYPES[typeId];
        return type.getReader().read(chunk, offset);
    }

    @Override
    public Value remove(long ref) {
        assertNotClosed();
        final Value removed = get(ref);
        final int chunkIdx = (int) (ref >>> 32);
        final int chunkOffset = (int) ref;
        final ByteBuffer chunk = chunks.get(chunkIdx);
        chunk.put(chunkOffset, (byte) REMOVED);
        return removed;
    }

    @Override
    public void close() {
        assertNotClosed();
        closed = true;
        allocated.forEach(m -> m.free(memoryTracker));
        allocated.clear();
        chunks.clear();
        writer.close();
        currentChunk = null;
    }

    private void assertNotClosed() {
        checkState(!closed, "Container is closed");
    }

    private ByteBuffer addNewChunk(int size) {
        final Memory memory = allocator.allocate(size, false, memoryTracker);
        final ByteBuffer chunk = memory.asByteBuffer();
        allocated.add(memory);
        chunks.add(chunk);
        return chunk;
    }

    private static class OffHeapByteBufferAllocator implements ValueByteBufferCodec.ByteBufferAllocator {
        private final MemoryAllocator allocator;
        private final MemoryTracker memoryTracker;
        private Memory bufMemory;

        public OffHeapByteBufferAllocator(MemoryAllocator allocator, MemoryTracker memoryTracker) {
            this.allocator = allocator;
            this.memoryTracker = memoryTracker;
        }

        @Override
        public ByteBuffer allocate(long capacity) {
            this.bufMemory = allocator.allocate(capacity, false, memoryTracker);
            return bufMemory.asByteBuffer();
        }

        @Override
        public void free() {
            bufMemory.free(memoryTracker);
            bufMemory = null;
        }
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy