blob: 9bb7dbccc03e03abf97351550edf7e773cd42b43 [file]
// Licensed to the Apache Software Foundation (ASF) under one
// or more contributor license agreements. See the NOTICE file
// distributed with this work for additional information
// regarding copyright ownership. The ASF 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
//
// 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.
#pragma once
#include <list>
#include <memory>
#include <string>
#include <unordered_map>
#include <unordered_set>
#include <utility>
#include <vector>
#include "plasma/common.h"
#include "plasma/eviction_policy.h"
#include "plasma/plasma.h"
namespace plasma {
/// Reserve this fraction of memory for shared usage. Attempts to set client
/// quotas that would cause the global LRU memory fraction to fall below this
/// value will be rejected.
constexpr double kGlobalLruReserveFraction = 0.3;
/// Extends the basic eviction policy to implement per-client memory quotas.
/// This effectively gives each client its own LRU queue, which caps its
/// memory usage and protects this memory from being evicted by other clients.
///
/// The quotas are enforced when objects are first created, by evicting the
/// necessary number of objects from the client's own LRU queue to cap its
/// memory usage. Once that is done, allocation is handled by the normal
/// eviction policy. This may result in the eviction of objects from the
/// global LRU queue, if not enough memory can be allocated even after the
/// evictions from the client's own LRU queue.
///
/// Some special cases:
/// - When a pinned object is "evicted" from a per-client queue, it is
/// instead transferred into the global LRU queue.
/// - When a client disconnects, its LRU queue is merged into the head of the
/// global LRU queue.
class QuotaAwarePolicy : public EvictionPolicy {
public:
/// Construct a quota-aware eviction policy.
///
/// \param store_info Information about the Plasma store that is exposed
/// to the eviction policy.
/// \param max_size Max size in bytes total of objects to store.
explicit QuotaAwarePolicy(PlasmaStoreInfo* store_info, int64_t max_size);
void ObjectCreated(const ObjectID& object_id, Client* client, bool is_create) override;
bool SetClientQuota(Client* client, int64_t output_memory_quota) override;
bool EnforcePerClientQuota(Client* client, int64_t size, bool is_create,
std::vector<ObjectID>* objects_to_evict) override;
void ClientDisconnected(Client* client) override;
void BeginObjectAccess(const ObjectID& object_id) override;
void EndObjectAccess(const ObjectID& object_id) override;
void RemoveObject(const ObjectID& object_id) override;
void RefreshObjects(const std::vector<ObjectID>& object_ids) override;
std::string DebugString() const override;
private:
/// Returns whether we are enforcing memory quotas for an operation.
bool HasQuota(Client* client, bool is_create);
/// Per-client LRU caches, if quota is enabled.
std::unordered_map<Client*, std::unique_ptr<LRUCache>> per_client_cache_;
/// Tracks which client created which object. This only applies to clients
/// that have a memory quota set.
std::unordered_map<ObjectID, Client*> owned_by_client_;
/// Tracks which objects are mapped for read and hence can't be evicted.
/// However these objects are still tracked within the client caches.
std::unordered_set<ObjectID> shared_for_read_;
};
} // namespace plasma