#pragma once
#include "JSCConfig.h"
#include "JSExportMacros.h"
#include <stdint.h>
#include <stdio.h>
#include <wtf/ForbidHeapAllocation.h>
#include <wtf/Noncopyable.h>
#include <wtf/PrintStream.h>
#include <wtf/StdLibExtras.h>
namespace WTF {
class StringBuilder;
}
using WTF::StringBuilder;
namespace JSC {
class Options {
public:
enum class DumpLevel : uint8_t {
None = 0,
Overridden,
All,
Verbose
};
enum class Availability : uint8_t {
Normal = 0,
Restricted,
Configurable
};
#define DECLARE_OPTION_ID(type_, name_, defaultValue_, availability_, description_) \
name_##ID,
enum ID : uint16_t {
FOR_EACH_JSC_OPTION(DECLARE_OPTION_ID)
};
#undef DECLARE_OPTION_ID
enum class Type : uint8_t {
Bool,
Unsigned,
Double,
Int32,
Size,
OptionRange,
OptionString,
GCLogLevel,
};
class AllowUnfinalizedAccessScope {
WTF_MAKE_NONCOPYABLE(AllowUnfinalizedAccessScope);
WTF_FORBID_HEAP_ALLOCATION;
public:
#if ASSERT_ENABLED
AllowUnfinalizedAccessScope()
{
if (!g_jscConfig.options.isFinalized) {
m_savedAllowUnfinalizedUse = g_jscConfig.options.allowUnfinalizedAccess;
g_jscConfig.options.allowUnfinalizedAccess = true;
}
}
~AllowUnfinalizedAccessScope()
{
if (!g_jscConfig.options.isFinalized)
g_jscConfig.options.allowUnfinalizedAccess = m_savedAllowUnfinalizedUse;
}
private:
bool m_savedAllowUnfinalizedUse;
#else
ALWAYS_INLINE AllowUnfinalizedAccessScope() = default;
ALWAYS_INLINE ~AllowUnfinalizedAccessScope() { }
#endif
};
JS_EXPORT_PRIVATE static void initialize();
static void finalize();
JS_EXPORT_PRIVATE static bool setOptions(const char* optionsList);
JS_EXPORT_PRIVATE static bool setOption(const char* arg);
JS_EXPORT_PRIVATE static void dumpAllOptions(FILE*, DumpLevel, const char* title = nullptr);
JS_EXPORT_PRIVATE static void dumpAllOptionsInALine(StringBuilder&);
JS_EXPORT_PRIVATE static void ensureOptionsAreCoherent();
static void recomputeDependentOptions();
#define DECLARE_OPTION_ACCESSORS(type_, name_, defaultValue_, availability_, description_) \
private: \
ALWAYS_INLINE static OptionsStorage::type_& name_##Default() { return g_jscConfig.options.name_##Default; } \
public: \
ALWAYS_INLINE static OptionsStorage::type_& name_() \
{ \
ASSERT(g_jscConfig.options.allowUnfinalizedAccess || g_jscConfig.options.isFinalized); \
return g_jscConfig.options.name_; \
}
FOR_EACH_JSC_OPTION(DECLARE_OPTION_ACCESSORS)
#undef DECLARE_OPTION_ACCESSORS
static bool isAvailable(ID, Availability);
private:
struct ConstMetaData {
const char* name;
const char* description;
Type type;
Availability availability;
uint16_t offsetOfOption;
uint16_t offsetOfOptionDefault;
};
Options();
enum DumpDefaultsOption {
DontDumpDefaults,
DumpDefaults
};
static void dumpOptionsIfNeeded();
static void dumpAllOptions(StringBuilder&, DumpLevel, const char* title,
const char* separator, const char* optionHeader, const char* optionFooter, DumpDefaultsOption);
static void dumpOption(StringBuilder&, DumpLevel, ID,
const char* optionHeader, const char* optionFooter, DumpDefaultsOption);
static bool setOptionWithoutAlias(const char* arg);
static bool setAliasedOption(const char* arg);
static bool overrideAliasedOptionWithHeuristic(const char* name);
inline static void* addressOfOption(Options::ID);
inline static void* addressOfOptionDefault(Options::ID);
static const ConstMetaData s_constMetaData[NumberOfOptions];
friend struct OptionReader;
};
}