#pragma once
#include "CacheUpdate.h"
#include "LeafExecutable.h"
#include "ParserModes.h"
#include <wtf/MallocPtr.h>
#include <wtf/Noncopyable.h>
#include <wtf/RefCounted.h>
#include <wtf/Vector.h>
namespace JSC {
class UnlinkedFunctionExecutable;
class CachedBytecode : public RefCounted<CachedBytecode> {
WTF_MAKE_NONCOPYABLE(CachedBytecode);
public:
static Ref<CachedBytecode> create()
{
return adoptRef(*new CachedBytecode(CachePayload::makeEmptyPayload()));
}
static Ref<CachedBytecode> create(FileSystem::MappedFileData&& data, LeafExecutableMap&& leafExecutables = { })
{
return adoptRef(*new CachedBytecode(CachePayload::makeMappedPayload(WTFMove(data)), WTFMove(leafExecutables)));
}
static Ref<CachedBytecode> create(MallocPtr<uint8_t, VMMalloc>&& data, size_t size, LeafExecutableMap&& leafExecutables)
{
return adoptRef(*new CachedBytecode(CachePayload::makeMallocPayload(WTFMove(data), size), WTFMove(leafExecutables)));
}
LeafExecutableMap& leafExecutables() { return m_leafExecutables; }
JS_EXPORT_PRIVATE void addGlobalUpdate(Ref<CachedBytecode>);
JS_EXPORT_PRIVATE void addFunctionUpdate(const UnlinkedFunctionExecutable*, CodeSpecializationKind, Ref<CachedBytecode>);
using ForEachUpdateCallback = Function<void(off_t, const void*, size_t)>;
JS_EXPORT_PRIVATE void commitUpdates(const ForEachUpdateCallback&) const;
const uint8_t* data() const { return m_payload.data(); }
size_t size() const { return m_payload.size(); }
bool hasUpdates() const { return !m_updates.isEmpty(); }
size_t sizeForUpdate() const { return m_size; }
private:
CachedBytecode(CachePayload&& payload, LeafExecutableMap&& leafExecutables = { })
: m_size(payload.size())
, m_payload(WTFMove(payload))
, m_leafExecutables(WTFMove(leafExecutables))
{
}
void copyLeafExecutables(const CachedBytecode&);
size_t m_size { 0 };
CachePayload m_payload;
LeafExecutableMap m_leafExecutables;
Vector<CacheUpdate> m_updates;
};
}