// Copyright 2012 the V8 project authors. All rights reserved. // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. #ifndef V8_ACCESSORS_H_ #define V8_ACCESSORS_H_ #include "include/v8.h" #include "src/allocation.h" #include "src/globals.h" #include "src/property-details.h" namespace v8 { namespace internal { // Forward declarations. class AccessorInfo; template class Handle; class FieldIndex; class JavaScriptFrame; // The list of accessor descriptors. This is a second-order macro // taking a macro to be applied to all accessor descriptor names. // V(accessor_name, AccessorName, GetterSideEffectType, SetterSideEffectType) #define ACCESSOR_INFO_LIST_GENERATOR(V, _) \ V(_, arguments_iterator, ArgumentsIterator, kHasNoSideEffect, \ kHasSideEffectToReceiver) \ V(_, array_length, ArrayLength, kHasNoSideEffect, kHasSideEffectToReceiver) \ V(_, bound_function_length, BoundFunctionLength, kHasNoSideEffect, \ kHasSideEffectToReceiver) \ V(_, bound_function_name, BoundFunctionName, kHasNoSideEffect, \ kHasSideEffectToReceiver) \ V(_, error_stack, ErrorStack, kHasSideEffectToReceiver, \ kHasSideEffectToReceiver) \ V(_, function_arguments, FunctionArguments, kHasNoSideEffect, \ kHasSideEffectToReceiver) \ V(_, function_caller, FunctionCaller, kHasNoSideEffect, \ kHasSideEffectToReceiver) \ V(_, function_name, FunctionName, kHasNoSideEffect, \ kHasSideEffectToReceiver) \ V(_, function_length, FunctionLength, kHasNoSideEffect, \ kHasSideEffectToReceiver) \ V(_, function_prototype, FunctionPrototype, kHasNoSideEffect, \ kHasSideEffectToReceiver) \ V(_, string_length, StringLength, kHasNoSideEffect, kHasSideEffectToReceiver) #define ACCESSOR_SETTER_LIST(V) \ V(ArrayLengthSetter) \ V(ErrorStackSetter) \ V(FunctionPrototypeSetter) \ V(ModuleNamespaceEntrySetter) \ V(ReconfigureToDataProperty) // Accessors contains all predefined proxy accessors. class Accessors : public AllStatic { public: #define ACCESSOR_GETTER_DECLARATION(_, accessor_name, AccessorName, ...) \ static void AccessorName##Getter( \ v8::Local name, \ const v8::PropertyCallbackInfo& info); ACCESSOR_INFO_LIST_GENERATOR(ACCESSOR_GETTER_DECLARATION, /* not used */) #undef ACCESSOR_GETTER_DECLARATION #define ACCESSOR_SETTER_DECLARATION(accessor_name) \ static void accessor_name( \ v8::Local name, v8::Local value, \ const v8::PropertyCallbackInfo& info); ACCESSOR_SETTER_LIST(ACCESSOR_SETTER_DECLARATION) #undef ACCESSOR_SETTER_DECLARATION static constexpr int kAccessorInfoCount = #define COUNT_ACCESSOR(...) +1 ACCESSOR_INFO_LIST_GENERATOR(COUNT_ACCESSOR, /* not used */); #undef COUNT_ACCESSOR static constexpr int kAccessorSetterCount = #define COUNT_ACCESSOR(...) +1 ACCESSOR_SETTER_LIST(COUNT_ACCESSOR); #undef COUNT_ACCESSOR static void ModuleNamespaceEntryGetter( v8::Local name, const v8::PropertyCallbackInfo& info); static Handle MakeModuleNamespaceEntryInfo(Isolate* isolate, Handle name); // Accessor function called directly from the runtime system. Returns the // newly materialized arguments object for the given {frame}. Note that for // optimized frames it is possible to specify an {inlined_jsframe_index}. static Handle FunctionGetArguments(JavaScriptFrame* frame, int inlined_jsframe_index); // Returns true for properties that are accessors to object fields. // If true, the matching FieldIndex is returned through |field_index|. static bool IsJSObjectFieldAccessor(Isolate* isolate, Handle map, Handle name, FieldIndex* field_index); static MaybeHandle ReplaceAccessorWithDataProperty( Handle receiver, Handle holder, Handle name, Handle value); // Create an AccessorInfo. The setter is optional (can be nullptr). // // Note that the type of setter is AccessorNameBooleanSetterCallback instead // of v8::AccessorNameSetterCallback. The difference is that the former can // set a (boolean) return value. The setter should roughly follow the same // conventions as many of the internal methods in objects.cc: // - The return value is unset iff there was an exception. // - If the ShouldThrow argument is true, the return value must not be false. typedef void (*AccessorNameBooleanSetterCallback)( Local property, Local value, const PropertyCallbackInfo& info); static Handle MakeAccessor( Isolate* isolate, Handle name, AccessorNameGetterCallback getter, AccessorNameBooleanSetterCallback setter); private: #define ACCESSOR_INFO_DECLARATION(_, accessor_name, AccessorName, ...) \ static Handle Make##AccessorName##Info(Isolate* isolate); ACCESSOR_INFO_LIST_GENERATOR(ACCESSOR_INFO_DECLARATION, /* not used */) #undef ACCESSOR_INFO_DECLARATION friend class Heap; }; } // namespace internal } // namespace v8 #endif // V8_ACCESSORS_H_