Commit d4d0cf32 authored by Gus Caplan's avatar Gus Caplan Committed by Commit Bot

[api] add reflection apis for v8::Data

Allows reflection of v8::Data types, such as being able to check if a
value is a v8::Module. This is useful for libraries which wrap the V8
API, such as rusty_v8.

Change-Id: I4841c5f7f60885b20e1504c8562e278844ff7ec3
Reviewed-on: https://chromium-review.googlesource.com/c/v8/v8/+/2382719Reviewed-by: 's avatarCamillo Bruni <cbruni@chromium.org>
Commit-Queue: Gus Caplan <snek@chromium.org>
Cr-Commit-Position: refs/heads/master@{#69649}
parent 10e2311f
......@@ -1319,6 +1319,32 @@ class V8_EXPORT SealHandleScope {
* The superclass of objects that can reside on V8's heap.
*/
class V8_EXPORT Data {
public:
/**
* Returns true if this data is a |v8::Value|.
*/
bool IsValue() const;
/**
* Returns true if this data is a |v8::Module|.
*/
bool IsModule() const;
/**
* Returns true if this data is a |v8::Private|.
*/
bool IsPrivate() const;
/**
* Returns true if this data is a |v8::ObjectTemplate|.
*/
bool IsObjectTemplate() const;
/**
* Returns true if this data is a |v8::FunctionTemplate|.
*/
bool IsFunctionTemplate() const;
private:
Data();
};
......@@ -1630,6 +1656,11 @@ class V8_EXPORT Module : public Data {
"the latter will crash with a failed CHECK().")
void SetSyntheticModuleExport(Local<String> export_name,
Local<Value> export_value);
V8_INLINE static Module* Cast(Data* data);
private:
static void CheckCast(Data* obj);
};
/**
......@@ -2925,6 +2956,8 @@ class V8_EXPORT Value : public Data {
bool FullIsUndefined() const;
bool FullIsNull() const;
bool FullIsString() const;
static void CheckCast(Data* that);
};
......@@ -11545,6 +11578,13 @@ template <class T> Value* Value::Cast(T* value) {
return static_cast<Value*>(value);
}
template <>
V8_INLINE Value* Value::Cast(Data* value) {
#ifdef V8_ENABLE_CHECKS
CheckCast(value);
#endif
return static_cast<Value*>(value);
}
Boolean* Boolean::Cast(v8::Value* value) {
#ifdef V8_ENABLE_CHECKS
......@@ -11577,6 +11617,12 @@ Private* Private::Cast(Data* data) {
return reinterpret_cast<Private*>(data);
}
Module* Module::Cast(Data* data) {
#ifdef V8_ENABLE_CHECKS
CheckCast(data);
#endif
return reinterpret_cast<Module*>(data);
}
Number* Number::Cast(v8::Value* value) {
#ifdef V8_ENABLE_CHECKS
......
......@@ -1192,6 +1192,34 @@ void* SealHandleScope::operator new[](size_t) { base::OS::Abort(); }
void SealHandleScope::operator delete(void*, size_t) { base::OS::Abort(); }
void SealHandleScope::operator delete[](void*, size_t) { base::OS::Abort(); }
bool Data::IsModule() const { return Utils::OpenHandle(this)->IsModule(); }
bool Data::IsValue() const {
i::DisallowHeapAllocation no_gc;
i::Handle<i::Object> self = Utils::OpenHandle(this);
if (self->IsSmi()) {
return true;
}
i::HeapObject heap_object = i::HeapObject::cast(*self);
DCHECK(!heap_object.IsTheHole());
if (heap_object.IsSymbol()) {
return !i::Symbol::cast(heap_object).is_private();
}
return heap_object.IsPrimitiveHeapObject() || heap_object.IsJSReceiver();
}
bool Data::IsPrivate() const {
return Utils::OpenHandle(this)->IsPrivateSymbol();
}
bool Data::IsObjectTemplate() const {
return Utils::OpenHandle(this)->IsObjectTemplateInfo();
}
bool Data::IsFunctionTemplate() const {
return Utils::OpenHandle(this)->IsFunctionTemplateInfo();
}
void Context::Enter() {
i::Handle<i::Context> env = Utils::OpenHandle(this);
i::Isolate* isolate = env->GetIsolate();
......@@ -3394,7 +3422,9 @@ bool Value::FullIsString() const {
return result;
}
bool Value::IsSymbol() const { return Utils::OpenHandle(this)->IsSymbol(); }
bool Value::IsSymbol() const {
return Utils::OpenHandle(this)->IsPublicSymbol();
}
bool Value::IsArray() const { return Utils::OpenHandle(this)->IsJSArray(); }
......@@ -3649,6 +3679,10 @@ void i::Internals::CheckInitializedImpl(v8::Isolate* external_isolate) {
"Isolate is not initialized or V8 has died");
}
void v8::Value::CheckCast(Data* that) {
Utils::ApiCheck(that->IsValue(), "v8::Value::Cast", "Data is not a Value");
}
void External::CheckCast(v8::Value* that) {
Utils::ApiCheck(that->IsExternal(), "v8::External::Cast",
"Value is not an External");
......@@ -3694,6 +3728,11 @@ void v8::Private::CheckCast(v8::Data* that) {
"v8::Private::Cast", "Value is not a Private");
}
void v8::Module::CheckCast(v8::Data* that) {
i::Handle<i::Object> obj = Utils::OpenHandle(that);
Utils::ApiCheck(obj->IsModule(), "v8::Module::Cast", "Value is not a Module");
}
void v8::Number::CheckCast(v8::Value* that) {
i::Handle<i::Object> obj = Utils::OpenHandle(that);
Utils::ApiCheck(obj->IsNumber(), "v8::Number::Cast()",
......
......@@ -2250,6 +2250,69 @@ static void TestObjectTemplateInheritedWithoutInstanceTemplate(
}
}
THREADED_TEST(TestDataTypeChecks) {
LocalContext env;
v8::Isolate* isolate = CcTest::isolate();
v8::HandleScope scope(isolate);
v8::Local<v8::Data> values[] = {
v8::Undefined(isolate),
v8::Null(isolate),
v8::True(isolate),
v8::Integer::New(isolate, 10),
v8::Number::New(isolate, 3.14),
v8::BigInt::NewFromUnsigned(isolate, 10),
v8::Symbol::New(isolate),
v8::String::NewFromUtf8Literal(isolate, "hello"),
};
for (auto x : values) {
CHECK(!x->IsModule());
CHECK(x->IsValue());
CHECK(!x->IsPrivate());
CHECK(!x->IsObjectTemplate());
CHECK(!x->IsFunctionTemplate());
v8::Local<v8::Value>::Cast(x);
}
v8::ScriptOrigin origin(
v8_str(""), Local<v8::Integer>(), Local<v8::Integer>(),
Local<v8::Boolean>(), Local<v8::Integer>(), Local<v8::Value>(),
Local<v8::Boolean>(), Local<v8::Boolean>(), True(isolate));
v8::ScriptCompiler::Source source(v8::String::NewFromUtf8Literal(isolate, ""),
origin);
v8::Local<v8::Data> module =
v8::ScriptCompiler::CompileModule(isolate, &source).ToLocalChecked();
CHECK(module->IsModule());
CHECK(!module->IsValue());
CHECK(!module->IsPrivate());
CHECK(!module->IsObjectTemplate());
CHECK(!module->IsFunctionTemplate());
v8::Local<v8::Module>::Cast(module);
v8::Local<v8::Data> p = v8::Private::New(isolate);
CHECK(!p->IsModule());
CHECK(!p->IsValue());
CHECK(p->IsPrivate());
CHECK(!p->IsObjectTemplate());
CHECK(!p->IsFunctionTemplate());
CHECK(!(*reinterpret_cast<Local<Value>*>(&p))->IsSymbol());
v8::Local<v8::Private>::Cast(p);
v8::Local<v8::Data> otmpl = v8::ObjectTemplate::New(isolate);
CHECK(!otmpl->IsModule());
CHECK(!otmpl->IsValue());
CHECK(!otmpl->IsPrivate());
CHECK(otmpl->IsObjectTemplate());
CHECK(!otmpl->IsFunctionTemplate());
v8::Local<v8::Data> ftmpl = v8::FunctionTemplate::New(isolate);
CHECK(!ftmpl->IsModule());
CHECK(!ftmpl->IsValue());
CHECK(!ftmpl->IsPrivate());
CHECK(!ftmpl->IsObjectTemplate());
CHECK(ftmpl->IsFunctionTemplate());
}
THREADED_TEST(TestObjectTemplateInheritedWithPrototype1) {
TestObjectTemplateInheritedWithoutInstanceTemplate(
Constructor_GetFunction_NewInstance);
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment