xref: /third_party/node/deps/v8/src/api/api-macros.h (revision 1cb0ef41)
1// Copyright 2021 the V8 project authors. All rights reserved.
2// Use of this source code is governed by a BSD-style license that can be
3// found in the LICENSE file.
4
5// Note 1: Any file that includes this one should include api-macros-undef.h
6// at the bottom.
7
8// Note 2: This file is deliberately missing the include guards (the undeffing
9// approach wouldn't work otherwise).
10//
11// PRESUBMIT_INTENTIONALLY_MISSING_INCLUDE_GUARD
12
13/*
14 * Most API methods should use one of the three macros:
15 *
16 * ENTER_V8, ENTER_V8_NO_SCRIPT, ENTER_V8_NO_SCRIPT_NO_EXCEPTION.
17 *
18 * The latter two assume that no script is executed, and no exceptions are
19 * scheduled in addition (respectively). Creating a pending exception and
20 * removing it before returning is ok.
21 *
22 * Exceptions should be handled either by invoking one of the
23 * RETURN_ON_FAILED_EXECUTION* macros.
24 *
25 * API methods that are part of the debug interface should use
26 *
27 * PREPARE_FOR_DEBUG_INTERFACE_EXECUTION_WITH_ISOLATE
28 *
29 * in a similar fashion to ENTER_V8.
30 *
31 * Don't use macros with DO_NOT_USE in their name.
32 *
33 * TODO(cbruni): Document LOG_API and other RuntimeCallStats macros.
34 * TODO(verwaest): All API methods should invoke one of the ENTER_V8* macros.
35 * TODO(verwaest): Remove calls form API methods to DO_NOT_USE macros.
36 */
37
38#define API_RCS_SCOPE(isolate, class_name, function_name) \
39  RCS_SCOPE(isolate,                                      \
40            i::RuntimeCallCounterId::kAPI_##class_name##_##function_name);
41
42#define ENTER_V8_DO_NOT_USE(isolate) i::VMState<v8::OTHER> __state__((isolate))
43
44#define ENTER_V8_HELPER_DO_NOT_USE(isolate, context, class_name,  \
45                                   function_name, bailout_value,  \
46                                   HandleScopeClass, do_callback) \
47  if (IsExecutionTerminatingCheck(isolate)) {                     \
48    return bailout_value;                                         \
49  }                                                               \
50  HandleScopeClass handle_scope(isolate);                         \
51  CallDepthScope<do_callback> call_depth_scope(isolate, context); \
52  API_RCS_SCOPE(isolate, class_name, function_name);              \
53  i::VMState<v8::OTHER> __state__((isolate));                     \
54  bool has_pending_exception = false
55
56#define PREPARE_FOR_DEBUG_INTERFACE_EXECUTION_WITH_ISOLATE(isolate, T)       \
57  if (IsExecutionTerminatingCheck(isolate)) {                                \
58    return MaybeLocal<T>();                                                  \
59  }                                                                          \
60  InternalEscapableScope handle_scope(isolate);                              \
61  CallDepthScope<false> call_depth_scope(isolate, v8::Local<v8::Context>()); \
62  i::VMState<v8::OTHER> __state__((isolate));                                \
63  bool has_pending_exception = false
64
65#define PREPARE_FOR_EXECUTION_WITH_CONTEXT(context, class_name, function_name, \
66                                           bailout_value, HandleScopeClass,    \
67                                           do_callback)                        \
68  auto isolate = context.IsEmpty()                                             \
69                     ? i::Isolate::Current()                                   \
70                     : reinterpret_cast<i::Isolate*>(context->GetIsolate());   \
71  ENTER_V8_HELPER_DO_NOT_USE(isolate, context, class_name, function_name,      \
72                             bailout_value, HandleScopeClass, do_callback);
73
74#define PREPARE_FOR_EXECUTION(context, class_name, function_name, T)          \
75  PREPARE_FOR_EXECUTION_WITH_CONTEXT(context, class_name, function_name,      \
76                                     MaybeLocal<T>(), InternalEscapableScope, \
77                                     false)
78
79#define ENTER_V8(isolate, context, class_name, function_name, bailout_value, \
80                 HandleScopeClass)                                           \
81  ENTER_V8_HELPER_DO_NOT_USE(isolate, context, class_name, function_name,    \
82                             bailout_value, HandleScopeClass, true)
83
84#ifdef DEBUG
85#define ENTER_V8_NO_SCRIPT(isolate, context, class_name, function_name,   \
86                           bailout_value, HandleScopeClass)               \
87  ENTER_V8_HELPER_DO_NOT_USE(isolate, context, class_name, function_name, \
88                             bailout_value, HandleScopeClass, false);     \
89  i::DisallowJavascriptExecutionDebugOnly __no_script__((isolate))
90
91// Lightweight version for APIs that don't require an active context.
92#define ASSERT_NO_SCRIPT_NO_EXCEPTION(isolate)                      \
93  i::DisallowJavascriptExecutionDebugOnly __no_script__((isolate)); \
94  i::DisallowExceptions __no_exceptions__((isolate))
95
96#define ENTER_V8_NO_SCRIPT_NO_EXCEPTION(isolate) \
97  i::VMState<v8::OTHER> __state__((isolate));    \
98  ASSERT_NO_SCRIPT_NO_EXCEPTION(isolate)
99
100#define ENTER_V8_FOR_NEW_CONTEXT(isolate)     \
101  i::VMState<v8::OTHER> __state__((isolate)); \
102  i::DisallowExceptions __no_exceptions__((isolate))
103#else
104#define ENTER_V8_NO_SCRIPT(isolate, context, class_name, function_name,   \
105                           bailout_value, HandleScopeClass)               \
106  ENTER_V8_HELPER_DO_NOT_USE(isolate, context, class_name, function_name, \
107                             bailout_value, HandleScopeClass, false)
108
109#define ASSERT_NO_SCRIPT_NO_EXCEPTION(isolate)
110
111#define ENTER_V8_NO_SCRIPT_NO_EXCEPTION(isolate) \
112  i::VMState<v8::OTHER> __state__((isolate));
113
114#define ENTER_V8_FOR_NEW_CONTEXT(isolate) \
115  i::VMState<v8::OTHER> __state__((isolate));
116#endif  // DEBUG
117
118#define EXCEPTION_BAILOUT_CHECK_SCOPED_DO_NOT_USE(isolate, value) \
119  do {                                                            \
120    if (has_pending_exception) {                                  \
121      call_depth_scope.Escape();                                  \
122      return value;                                               \
123    }                                                             \
124  } while (false)
125
126#define RETURN_ON_FAILED_EXECUTION(T) \
127  EXCEPTION_BAILOUT_CHECK_SCOPED_DO_NOT_USE(isolate, MaybeLocal<T>())
128
129#define RETURN_ON_FAILED_EXECUTION_PRIMITIVE(T) \
130  EXCEPTION_BAILOUT_CHECK_SCOPED_DO_NOT_USE(isolate, Nothing<T>())
131
132#define RETURN_ESCAPED(value) return handle_scope.Escape(value);
133