/* * Copyright 2021 The Netty Project * * The Netty Project licenses this file to you 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: * * https://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 io.netty.buffer.api.memseg; import io.netty.buffer.api.AllocationType; import io.netty.buffer.api.AllocatorControl; import io.netty.buffer.api.Buffer; import io.netty.buffer.api.Drop; import io.netty.buffer.api.MemoryManager; import io.netty.buffer.api.StandardAllocationTypes; import io.netty.buffer.api.internal.ArcDrop; import io.netty.buffer.api.internal.Statics; import jdk.incubator.foreign.MemorySegment; import jdk.incubator.foreign.ResourceScope; import java.lang.ref.Cleaner; import java.util.concurrent.ConcurrentHashMap; import java.util.function.Function; import static jdk.incubator.foreign.ResourceScope.newSharedScope; public class SegmentMemoryManagers implements MemoryManager { private static final ConcurrentHashMap CLEANUP_ACTIONS = new ConcurrentHashMap<>(); private static final Function CLEANUP_ACTION_MAKER = s -> new ReduceNativeMemoryUsage(s); static Runnable getCleanupAction(long size) { return CLEANUP_ACTIONS.computeIfAbsent(size, CLEANUP_ACTION_MAKER); } private static MemorySegment createHeapSegment(long size) { return MemorySegment.ofArray(new byte[Math.toIntExact(size)]); } private static MemorySegment createNativeSegment(long size, Cleaner cleaner) { final ResourceScope scope = cleaner == null ? newSharedScope() : newSharedScope(cleaner); scope.addCloseAction(getCleanupAction(size)); var segment = MemorySegment.allocateNative(size, scope); Statics.MEM_USAGE_NATIVE.add(size); return segment; } @Override public Buffer allocateShared(AllocatorControl allocatorControl, long size, Drop drop, AllocationType type) { if (type instanceof StandardAllocationTypes stype) { var segment = switch (stype) { case ON_HEAP -> createHeapSegment(size); case OFF_HEAP -> createNativeSegment(size, null); }; return new MemSegBuffer(segment, segment, Statics.convert(drop), allocatorControl); } throw new IllegalArgumentException("Unknown allocation type: " + type); } @Override public Buffer allocateConstChild(Buffer readOnlyConstParent) { assert readOnlyConstParent.readOnly(); MemSegBuffer buf = (MemSegBuffer) readOnlyConstParent; return new MemSegBuffer(buf); } @Override public Drop drop() { return Statics.convert(MemSegBuffer.SEGMENT_CLOSE); } @Override public Object unwrapRecoverableMemory(Buffer buf) { var b = (MemSegBuffer) buf; return b.recoverableMemory(); } @Override public Buffer recoverMemory(AllocatorControl allocatorControl, Object recoverableMemory, Drop drop) { var segment = (MemorySegment) recoverableMemory; return new MemSegBuffer(segment, segment, Statics.convert(ArcDrop.acquire(drop)), allocatorControl); } @Override public Object sliceMemory(Object memory, int offset, int length) { var segment = (MemorySegment) memory; return segment.asSlice(offset, length); } @Override public String implementationName() { return "MemorySegment"; } }