luau/CodeGen/include/Luau/SharedCodeAllocator.h

181 lines
6.1 KiB
C
Raw Normal View History

2024-03-22 17:21:27 +00:00
// This file is part of the Luau programming language and is licensed under MIT License; see LICENSE.txt for details
#pragma once
2024-04-12 11:44:40 +01:00
#include "Luau/CodeGen.h"
2024-03-22 17:21:27 +00:00
#include "Luau/Common.h"
#include "Luau/NativeProtoExecData.h"
#include <array>
#include <atomic>
#include <memory>
#include <mutex>
2024-04-12 11:44:40 +01:00
#include <optional>
2024-03-22 17:21:27 +00:00
#include <stdint.h>
#include <unordered_map>
#include <vector>
namespace Luau
{
namespace CodeGen
{
// SharedCodeAllocator is a native executable code allocator that provides
// shared ownership of the native code. Code is allocated on a per-module
// basis. Each module is uniquely identifiable via an id, which may be a hash
// or other unique value. Each module may contain multiple natively compiled
// functions (protos).
//
// The module is the unit of shared ownership (i.e., it is where the reference
// count is maintained).
2024-04-05 18:41:05 +01:00
struct CodeAllocator;
2024-03-22 17:21:27 +00:00
class NativeModule;
class NativeModuleRef;
class SharedCodeAllocator;
// A NativeModule represents a single natively-compiled module (script). It is
// the unit of shared ownership and is thus where the reference count is
// maintained. It owns a set of NativeProtos, with associated native exec data,
// and the allocated native data and code.
class NativeModule
{
public:
2024-08-02 00:25:12 +01:00
NativeModule(
SharedCodeAllocator* allocator,
const std::optional<ModuleId>& moduleId,
const uint8_t* moduleBaseAddress,
std::vector<NativeProtoExecDataPtr> nativeProtos
) noexcept;
2024-03-22 17:21:27 +00:00
NativeModule(const NativeModule&) = delete;
NativeModule(NativeModule&&) = delete;
NativeModule& operator=(const NativeModule&) = delete;
NativeModule& operator=(NativeModule&&) = delete;
// The NativeModule must not be destroyed if there are any outstanding
// references. It should thus only be destroyed by a call to release()
// that releases the last reference.
~NativeModule() noexcept;
size_t addRef() const noexcept;
2024-04-05 18:41:05 +01:00
size_t addRefs(size_t count) const noexcept;
2024-03-22 17:21:27 +00:00
size_t release() const noexcept;
[[nodiscard]] size_t getRefcount() const noexcept;
2024-04-12 11:44:40 +01:00
[[nodiscard]] const std::optional<ModuleId>& getModuleId() const noexcept;
2024-03-22 17:21:27 +00:00
// Gets the base address of the executable native code for the module.
[[nodiscard]] const uint8_t* getModuleBaseAddress() const noexcept;
// Attempts to find the NativeProto with the given bytecode id. If no
// NativeProto for that bytecode id exists, a null pointer is returned.
2024-04-05 18:41:05 +01:00
[[nodiscard]] const uint32_t* tryGetNativeProto(uint32_t bytecodeId) const noexcept;
[[nodiscard]] const std::vector<NativeProtoExecDataPtr>& getNativeProtos() const noexcept;
2024-03-22 17:21:27 +00:00
private:
mutable std::atomic<size_t> refcount = 0;
SharedCodeAllocator* allocator = nullptr;
2024-04-12 11:44:40 +01:00
std::optional<ModuleId> moduleId = {};
2024-03-22 17:21:27 +00:00
const uint8_t* moduleBaseAddress = nullptr;
2024-04-05 18:41:05 +01:00
std::vector<NativeProtoExecDataPtr> nativeProtos = {};
2024-03-22 17:21:27 +00:00
};
// A NativeModuleRef is an owning reference to a NativeModule. (Note: We do
// not use shared_ptr, to avoid complex state management in the Luau GC Proto
// object.)
class NativeModuleRef
{
public:
NativeModuleRef() noexcept = default;
2024-04-12 11:44:40 +01:00
NativeModuleRef(const NativeModule* nativeModule) noexcept;
2024-03-22 17:21:27 +00:00
NativeModuleRef(const NativeModuleRef& other) noexcept;
NativeModuleRef(NativeModuleRef&& other) noexcept;
NativeModuleRef& operator=(NativeModuleRef other) noexcept;
~NativeModuleRef() noexcept;
void reset() noexcept;
void swap(NativeModuleRef& other) noexcept;
[[nodiscard]] bool empty() const noexcept;
explicit operator bool() const noexcept;
[[nodiscard]] const NativeModule* get() const noexcept;
[[nodiscard]] const NativeModule* operator->() const noexcept;
[[nodiscard]] const NativeModule& operator*() const noexcept;
private:
const NativeModule* nativeModule = nullptr;
};
class SharedCodeAllocator
{
public:
2024-04-05 18:41:05 +01:00
SharedCodeAllocator(CodeAllocator* codeAllocator) noexcept;
2024-03-22 17:21:27 +00:00
SharedCodeAllocator(const SharedCodeAllocator&) = delete;
SharedCodeAllocator(SharedCodeAllocator&&) = delete;
SharedCodeAllocator& operator=(const SharedCodeAllocator&) = delete;
SharedCodeAllocator& operator=(SharedCodeAllocator&&) = delete;
~SharedCodeAllocator() noexcept;
// If we have a NativeModule for the given ModuleId, an owning reference to
// it is returned. Otherwise, an empty NativeModuleRef is returned.
[[nodiscard]] NativeModuleRef tryGetNativeModule(const ModuleId& moduleId) const noexcept;
// If we have a NativeModule for the given ModuleId, an owning reference to
// it is returned. Otherwise, a new NativeModule is created for that ModuleId
// using the provided NativeProtos, data, and code (space is allocated for the
2024-04-05 18:41:05 +01:00
// data and code such that it can be executed). Like std::map::insert, the
// bool result is true if a new module was created; false if an existing
// module is being returned.
2024-08-02 00:25:12 +01:00
std::pair<NativeModuleRef, bool> getOrInsertNativeModule(
const ModuleId& moduleId,
std::vector<NativeProtoExecDataPtr> nativeProtos,
const uint8_t* data,
size_t dataSize,
const uint8_t* code,
size_t codeSize
);
2024-03-22 17:21:27 +00:00
2024-04-12 11:44:40 +01:00
NativeModuleRef insertAnonymousNativeModule(
2024-08-02 00:25:12 +01:00
std::vector<NativeProtoExecDataPtr> nativeProtos,
const uint8_t* data,
size_t dataSize,
const uint8_t* code,
size_t codeSize
);
2024-04-12 11:44:40 +01:00
2024-03-22 17:21:27 +00:00
// If a NativeModule exists for the given ModuleId and that NativeModule
// is no longer referenced, the NativeModule is destroyed. This should
// usually only be called by NativeModule::release() when the reference
// count becomes zero
2024-04-12 11:44:40 +01:00
void eraseNativeModuleIfUnreferenced(const NativeModule& nativeModule);
2024-03-22 17:21:27 +00:00
private:
struct ModuleIdHash
{
[[nodiscard]] size_t operator()(const ModuleId& moduleId) const noexcept;
};
[[nodiscard]] NativeModuleRef tryGetNativeModuleWithLockHeld(const ModuleId& moduleId) const noexcept;
mutable std::mutex mutex;
2024-04-12 11:44:40 +01:00
std::unordered_map<ModuleId, std::unique_ptr<NativeModule>, ModuleIdHash, std::equal_to<>> identifiedModules;
std::atomic<size_t> anonymousModuleCount = 0;
2024-04-05 18:41:05 +01:00
CodeAllocator* codeAllocator = nullptr;
2024-03-22 17:21:27 +00:00
};
} // namespace CodeGen
} // namespace Luau