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

io.trino.memory.MemoryPool Maven / Gradle / Ivy

/*
 * 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 io.trino.memory;

import com.google.common.annotations.VisibleForTesting;
import com.google.common.collect.ImmutableMap;
import com.google.common.util.concurrent.AbstractFuture;
import com.google.common.util.concurrent.ListenableFuture;
import com.google.errorprone.annotations.concurrent.GuardedBy;
import io.airlift.units.DataSize;
import io.trino.execution.TaskId;
import io.trino.spi.QueryId;
import io.trino.spi.memory.MemoryAllocation;
import io.trino.spi.memory.MemoryPoolInfo;
import jakarta.annotation.Nullable;
import org.weakref.jmx.Managed;

import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.Map.Entry;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.CopyOnWriteArrayList;

import static com.google.common.base.MoreObjects.toStringHelper;
import static com.google.common.base.Preconditions.checkArgument;
import static com.google.common.base.Preconditions.checkState;
import static com.google.common.collect.ImmutableMap.toImmutableMap;
import static io.trino.operator.Operator.NOT_BLOCKED;
import static java.util.Objects.requireNonNull;

public class MemoryPool
{
    private final long maxBytes;

    @GuardedBy("this")
    private long reservedBytes;
    @GuardedBy("this")
    private long reservedRevocableBytes;

    @Nullable
    @GuardedBy("this")
    private NonCancellableMemoryFuture future;

    // TODO: It would be better if we just tracked QueryContexts, but their lifecycle is managed by a weak reference, so we can't do that
    // It is guarded for updates by this, but can be read without holding a lock
    private final Map queryMemoryReservations = new ConcurrentHashMap<>();

    // This map keeps track of all the tagged allocations, e.g., query-1 -> ['TableScanOperator': 10MB, 'LazyOutputBuffer': 5MB, ...]
    @GuardedBy("this")
    private final Map> taggedMemoryAllocations = new HashMap<>();

    @GuardedBy("this")
    private final Map queryRevocableMemoryReservations = new HashMap<>();

    @GuardedBy("this")
    private final Map taskMemoryReservations = new HashMap<>();

    @GuardedBy("this")
    private final Map taskRevocableMemoryReservations = new HashMap<>();

    @GuardedBy("this")
    private long connectorsReservedBytes;

    private final List listeners = new CopyOnWriteArrayList<>();

    public MemoryPool(DataSize size)
    {
        requireNonNull(size, "size is null");
        maxBytes = size.toBytes();
    }

    public synchronized MemoryPoolInfo getInfo()
    {
        Map> memoryAllocations = new HashMap<>();
        for (Entry> entry : taggedMemoryAllocations.entrySet()) {
            List allocations = new ArrayList<>();
            if (entry.getValue() != null) {
                entry.getValue().forEach((tag, allocation) -> allocations.add(new MemoryAllocation(tag, allocation)));
            }
            memoryAllocations.put(entry.getKey(), allocations);
        }

        Map stringKeyedTaskMemoryReservations = taskMemoryReservations.entrySet().stream()
                .collect(toImmutableMap(
                        entry -> entry.getKey().toString(),
                        Entry::getValue));

        Map stringKeyedTaskRevocableMemoryReservations = taskRevocableMemoryReservations.entrySet().stream()
                .collect(toImmutableMap(
                        entry -> entry.getKey().toString(),
                        Entry::getValue));

        return new MemoryPoolInfo(
                maxBytes,
                reservedBytes,
                reservedRevocableBytes,
                queryMemoryReservations,
                memoryAllocations,
                queryRevocableMemoryReservations,
                stringKeyedTaskMemoryReservations,
                stringKeyedTaskRevocableMemoryReservations);
    }

    public void addListener(MemoryPoolListener listener)
    {
        listeners.add(requireNonNull(listener, "listener cannot be null"));
    }

    public void removeListener(MemoryPoolListener listener)
    {
        listeners.remove(requireNonNull(listener, "listener cannot be null"));
    }

    /**
     * Reserves the given number of bytes. Caller should wait on the returned future, before allocating more memory.
     */
    public ListenableFuture reserve(TaskId taskId, String allocationTag, long bytes)
    {
        checkArgument(bytes >= 0, "'%s' is negative", bytes);
        ListenableFuture result;
        synchronized (this) {
            if (bytes != 0) {
                QueryId queryId = taskId.getQueryId();
                queryMemoryReservations.merge(queryId, bytes, Long::sum);
                updateTaggedMemoryAllocations(queryId, allocationTag, bytes);
                taskMemoryReservations.merge(taskId, bytes, Long::sum);
            }
            reservedBytes += bytes;
            if (getFreeBytes() <= 0) {
                if (future == null) {
                    future = NonCancellableMemoryFuture.create();
                }
                checkState(!future.isDone(), "future is already completed");
                result = future;
            }
            else {
                result = NOT_BLOCKED;
            }
        }

        onMemoryReserved();
        return result;
    }

    private void onMemoryReserved()
    {
        listeners.forEach(listener -> listener.onMemoryReserved(this));
    }

    public ListenableFuture reserveRevocable(TaskId taskId, long bytes)
    {
        checkArgument(bytes >= 0, "'%s' is negative", bytes);

        ListenableFuture result;
        synchronized (this) {
            if (bytes != 0) {
                queryRevocableMemoryReservations.merge(taskId.getQueryId(), bytes, Long::sum);
                taskRevocableMemoryReservations.merge(taskId, bytes, Long::sum);
            }
            reservedRevocableBytes += bytes;
            if (getFreeBytes() <= 0) {
                if (future == null) {
                    future = NonCancellableMemoryFuture.create();
                }
                checkState(!future.isDone(), "future is already completed");
                result = future;
            }
            else {
                result = NOT_BLOCKED;
            }
        }

        onMemoryReserved();
        return result;
    }

    /**
     * Try to reserve the given number of bytes. Return value indicates whether the caller may use the requested memory.
     */
    public boolean tryReserve(TaskId taskId, String allocationTag, long bytes)
    {
        checkArgument(bytes >= 0, "'%s' is negative", bytes);
        synchronized (this) {
            if (getFreeBytes() - bytes < 0) {
                return false;
            }
            reservedBytes += bytes;
            if (bytes != 0) {
                QueryId queryId = taskId.getQueryId();
                queryMemoryReservations.merge(queryId, bytes, Long::sum);
                updateTaggedMemoryAllocations(queryId, allocationTag, bytes);
                taskMemoryReservations.merge(taskId, bytes, Long::sum);
            }
        }

        onMemoryReserved();
        return true;
    }

    public boolean tryReserveConnectorMemory(long bytes)
    {
        checkArgument(bytes >= 0, "'%s' is negative", bytes);
        synchronized (this) {
            if (getFreeBytes() - bytes < 0) {
                return false;
            }
            connectorsReservedBytes += bytes;
            reservedBytes += bytes;
        }
        onMemoryReserved();
        return true;
    }

    public boolean tryReserveRevocable(long bytes)
    {
        checkArgument(bytes >= 0, "'%s' is negative", bytes);
        synchronized (this) {
            if (getFreeBytes() - bytes < 0) {
                return false;
            }
            reservedRevocableBytes += bytes;
        }

        onMemoryReserved();
        return true;
    }

    public synchronized void free(TaskId taskId, String allocationTag, long bytes)
    {
        checkArgument(bytes >= 0, "'%s' is negative", bytes);
        checkArgument(reservedBytes >= bytes, "tried to free more memory than is reserved");
        if (bytes == 0) {
            // Freeing zero bytes is a no-op
            return;
        }

        QueryId queryId = taskId.getQueryId();
        Long queryReservation = queryMemoryReservations.get(queryId);
        requireNonNull(queryReservation, "queryReservation is null");
        checkArgument(queryReservation >= bytes, "tried to free more memory than is reserved by query");

        Long taskReservation = taskMemoryReservations.get(taskId);
        requireNonNull(taskReservation, "taskReservation is null");
        checkArgument(taskReservation >= bytes, "tried to free more memory than is reserved by task");

        queryReservation -= bytes;
        if (queryReservation == 0) {
            queryMemoryReservations.remove(queryId);
            taggedMemoryAllocations.remove(queryId);
        }
        else {
            queryMemoryReservations.put(queryId, queryReservation);
            updateTaggedMemoryAllocations(queryId, allocationTag, -bytes);
        }

        taskReservation -= bytes;
        if (taskReservation == 0) {
            taskMemoryReservations.remove(taskId);
        }
        else {
            taskMemoryReservations.put(taskId, taskReservation);
        }

        reservedBytes -= bytes;
        if (getFreeBytes() > 0 && future != null) {
            future.set(null);
            future = null;
        }
    }

    public synchronized void freeRevocable(TaskId taskId, long bytes)
    {
        checkArgument(bytes >= 0, "'%s' is negative", bytes);
        checkArgument(reservedRevocableBytes >= bytes, "tried to free more revocable memory than is reserved");
        if (bytes == 0) {
            // Freeing zero bytes is a no-op
            return;
        }

        QueryId queryId = taskId.getQueryId();
        Long queryReservation = queryRevocableMemoryReservations.get(queryId);
        requireNonNull(queryReservation, "queryReservation is null");
        checkArgument(queryReservation >= bytes, "tried to free more revocable memory than is reserved by query");

        Long taskReservation = taskRevocableMemoryReservations.get(taskId);
        requireNonNull(taskReservation, "taskReservation is null");
        checkArgument(taskReservation >= bytes, "tried to free more revocable memory than is reserved by task");

        queryReservation -= bytes;
        if (queryReservation == 0) {
            queryRevocableMemoryReservations.remove(queryId);
        }
        else {
            queryRevocableMemoryReservations.put(queryId, queryReservation);
        }

        taskReservation -= bytes;
        if (taskReservation == 0) {
            taskRevocableMemoryReservations.remove(taskId);
        }
        else {
            taskRevocableMemoryReservations.put(taskId, taskReservation);
        }

        reservedRevocableBytes -= bytes;
        if (getFreeBytes() > 0 && future != null) {
            future.set(null);
            future = null;
        }
    }

    public synchronized void freeRevocable(long bytes)
    {
        checkArgument(bytes >= 0, "'%s' is negative", bytes);
        checkArgument(reservedRevocableBytes >= bytes, "tried to free more revocable memory than is reserved");
        if (bytes == 0) {
            // Freeing zero bytes is a no-op
            return;
        }

        reservedRevocableBytes -= bytes;
        if (getFreeBytes() > 0 && future != null) {
            future.set(null);
            future = null;
        }
    }

    public synchronized void freeConnectorMemory(long bytes)
    {
        checkArgument(bytes >= 0, "'%s' is negative", bytes);
        checkArgument(reservedBytes >= bytes, "tried to free more memory than is reserved");
        checkArgument(connectorsReservedBytes >= bytes, "tried to free more memory for the connectors than is reserved");
        if (bytes == 0) {
            // Freeing zero bytes is a no-op
            return;
        }

        connectorsReservedBytes -= bytes;
        reservedBytes -= bytes;
        if (getFreeBytes() > 0 && future != null) {
            future.set(null);
            future = null;
        }
    }

    /**
     * Returns the number of free bytes. This value may be negative, which indicates that the pool is over-committed.
     */
    @Managed
    public synchronized long getFreeBytes()
    {
        return maxBytes - reservedBytes - reservedRevocableBytes;
    }

    @Managed
    public long getMaxBytes()
    {
        return maxBytes;
    }

    @Managed
    public synchronized long getReservedBytes()
    {
        return reservedBytes;
    }

    @Managed
    public synchronized long getReservedRevocableBytes()
    {
        return reservedRevocableBytes;
    }

    @Managed
    public synchronized long getConnectorsReservedBytes()
    {
        return connectorsReservedBytes;
    }

    long getQueryMemoryReservation(QueryId queryId)
    {
        return queryMemoryReservations.getOrDefault(queryId, 0L);
    }

    @VisibleForTesting
    synchronized long getQueryRevocableMemoryReservation(QueryId queryId)
    {
        return queryRevocableMemoryReservations.getOrDefault(queryId, 0L);
    }

    @VisibleForTesting
    synchronized long getTaskMemoryReservation(TaskId taskId)
    {
        return taskMemoryReservations.getOrDefault(taskId, 0L);
    }

    @VisibleForTesting
    synchronized long getTaskRevocableMemoryReservation(TaskId taskId)
    {
        return taskRevocableMemoryReservations.getOrDefault(taskId, 0L);
    }

    @Override
    public synchronized String toString()
    {
        return toStringHelper(this)
                .add("maxBytes", maxBytes)
                .add("freeBytes", getFreeBytes())
                .add("reservedBytes", reservedBytes)
                .add("reservedRevocableBytes", reservedRevocableBytes)
                .add("future", future)
                .toString();
    }

    private static class NonCancellableMemoryFuture
            extends AbstractFuture
    {
        public static  NonCancellableMemoryFuture create()
        {
            return new NonCancellableMemoryFuture<>();
        }

        @Override
        public boolean set(@Nullable V value)
        {
            return super.set(value);
        }

        @Override
        public boolean cancel(boolean mayInterruptIfRunning)
        {
            throw new UnsupportedOperationException("cancellation is not supported");
        }
    }

    private synchronized void updateTaggedMemoryAllocations(QueryId queryId, String allocationTag, long delta)
    {
        if (delta == 0) {
            return;
        }

        Map allocations = taggedMemoryAllocations.computeIfAbsent(queryId, _ -> new HashMap<>());
        allocations.compute(allocationTag, (ignored, oldValue) -> {
            if (oldValue == null) {
                return delta;
            }
            long newValue = oldValue.longValue() + delta;
            if (newValue == 0) {
                return null;
            }
            return newValue;
        });
    }

    @VisibleForTesting
    public synchronized Map getQueryMemoryReservations()
    {
        return ImmutableMap.copyOf(queryMemoryReservations);
    }

    @VisibleForTesting
    public synchronized Map> getTaggedMemoryAllocations()
    {
        return ImmutableMap.copyOf(taggedMemoryAllocations);
    }

    @VisibleForTesting
    public synchronized Map getQueryRevocableMemoryReservations()
    {
        return ImmutableMap.copyOf(queryRevocableMemoryReservations);
    }

    @VisibleForTesting
    public synchronized Map getTaskMemoryReservations()
    {
        return ImmutableMap.copyOf(taskMemoryReservations);
    }

    @VisibleForTesting
    public synchronized Map getTaskRevocableMemoryReservations()
    {
        return ImmutableMap.copyOf(taskRevocableMemoryReservations);
    }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy