#include "config.h"
#include "SerializedNFA.h"
#include "NFA.h"
#if ENABLE(CONTENT_EXTENSIONS)
namespace WebCore {
namespace ContentExtensions {
template<typename T>
bool writeAllToFile(FileSystem::PlatformFileHandle file, const T& container)
{
const char* bytes = reinterpret_cast<const char*>(container.data());
size_t bytesLength = container.size() * sizeof(container[0]);
const char* end = bytes + bytesLength;
while (bytes < end) {
auto written = FileSystem::writeToFile(file, bytes, bytesLength);
if (written == -1)
return false;
bytes += written;
bytesLength -= written;
}
return true;
}
Optional<SerializedNFA> SerializedNFA::serialize(NFA&& nfa)
{
auto file = FileSystem::invalidPlatformFileHandle;
auto filename = FileSystem::openTemporaryFile("SerializedNFA", file);
if (!FileSystem::isHandleValid(file))
return WTF::nullopt;
bool wroteSuccessfully = writeAllToFile(file, nfa.nodes)
&& writeAllToFile(file, nfa.transitions)
&& writeAllToFile(file, nfa.targets)
&& writeAllToFile(file, nfa.epsilonTransitionsTargets)
&& writeAllToFile(file, nfa.actions);
if (!wroteSuccessfully) {
FileSystem::closeFile(file);
FileSystem::deleteFile(filename);
return WTF::nullopt;
}
bool mappedSuccessfully = false;
FileSystem::MappedFileData mappedFile(file, FileSystem::MappedFileMode::Private, mappedSuccessfully);
FileSystem::closeFile(file);
FileSystem::deleteFile(filename);
if (!mappedSuccessfully)
return WTF::nullopt;
Metadata metadata {
nfa.nodes.size(),
nfa.transitions.size(),
nfa.targets.size(),
nfa.epsilonTransitionsTargets.size(),
nfa.actions.size(),
0,
nfa.nodes.size() * sizeof(nfa.nodes[0]),
nfa.nodes.size() * sizeof(nfa.nodes[0])
+ nfa.transitions.size() * sizeof(nfa.transitions[0]),
nfa.nodes.size() * sizeof(nfa.nodes[0])
+ nfa.transitions.size() * sizeof(nfa.transitions[0])
+ nfa.targets.size() * sizeof(nfa.targets[0]),
nfa.nodes.size() * sizeof(nfa.nodes[0])
+ nfa.transitions.size() * sizeof(nfa.transitions[0])
+ nfa.targets.size() * sizeof(nfa.targets[0])
+ nfa.epsilonTransitionsTargets.size() * sizeof(nfa.epsilonTransitionsTargets[0])
};
nfa.clear();
return {{ WTFMove(mappedFile), WTFMove(metadata) }};
}
SerializedNFA::SerializedNFA(FileSystem::MappedFileData&& file, Metadata&& metadata)
: m_file(WTFMove(file))
, m_metadata(WTFMove(metadata))
{
}
template<typename T>
const T* SerializedNFA::pointerAtOffsetInFile(size_t offset) const
{
return reinterpret_cast<const T*>(reinterpret_cast<const uint8_t*>(m_file.data()) + offset);
}
auto SerializedNFA::nodes() const -> const Range<ImmutableNFANode>
{
return { pointerAtOffsetInFile<ImmutableNFANode>(m_metadata.nodesOffset), m_metadata.nodesSize };
}
auto SerializedNFA::transitions() const -> const Range<ImmutableRange<char>>
{
return { pointerAtOffsetInFile<ImmutableRange<char>>(m_metadata.transitionsOffset), m_metadata.transitionsSize };
}
auto SerializedNFA::targets() const -> const Range<uint32_t>
{
return { pointerAtOffsetInFile<uint32_t>(m_metadata.targetsOffset), m_metadata.targetsSize };
}
auto SerializedNFA::epsilonTransitionsTargets() const -> const Range<uint32_t>
{
return { pointerAtOffsetInFile<uint32_t>(m_metadata.epsilonTransitionsTargetsOffset), m_metadata.epsilonTransitionsTargetsSize };
}
auto SerializedNFA::actions() const -> const Range<uint64_t>
{
return { pointerAtOffsetInFile<uint64_t>(m_metadata.actionsOffset), m_metadata.actionsSize };
}
} }
#endif // ENABLE(CONTENT_EXTENSIONS)