123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152 |
- // Copyright (c) 2019 GitHub, Inc. All rights reserved.
- // Use of this source code is governed by the MIT license that can be
- // found in the LICENSE file.
- #ifndef SHELL_COMMON_GIN_HELPER_CALLBACK_H_
- #define SHELL_COMMON_GIN_HELPER_CALLBACK_H_
- #include <utility>
- #include <vector>
- #include "base/bind.h"
- #include "gin/dictionary.h"
- #include "shell/common/gin_converters/std_converter.h"
- #include "shell/common/gin_helper/function_template.h"
- #include "shell/common/gin_helper/locker.h"
- #include "shell/common/gin_helper/microtasks_scope.h"
- // Implements safe conversions between JS functions and base::RepeatingCallback.
- namespace gin_helper {
- template <typename T>
- class RefCountedGlobal;
- // Manages the V8 function with RAII.
- class SafeV8Function {
- public:
- SafeV8Function(v8::Isolate* isolate, v8::Local<v8::Value> value);
- SafeV8Function(const SafeV8Function& other);
- ~SafeV8Function();
- bool IsAlive() const;
- v8::Local<v8::Function> NewHandle(v8::Isolate* isolate) const;
- private:
- scoped_refptr<RefCountedGlobal<v8::Function>> v8_function_;
- };
- // Helper to invoke a V8 function with C++ parameters.
- template <typename Sig>
- struct V8FunctionInvoker {};
- template <typename... ArgTypes>
- struct V8FunctionInvoker<v8::Local<v8::Value>(ArgTypes...)> {
- static v8::Local<v8::Value> Go(v8::Isolate* isolate,
- const SafeV8Function& function,
- ArgTypes... raw) {
- gin_helper::Locker locker(isolate);
- v8::EscapableHandleScope handle_scope(isolate);
- if (!function.IsAlive())
- return v8::Null(isolate);
- gin_helper::MicrotasksScope microtasks_scope(isolate, true);
- v8::Local<v8::Function> holder = function.NewHandle(isolate);
- v8::Local<v8::Context> context = holder->CreationContext();
- v8::Context::Scope context_scope(context);
- std::vector<v8::Local<v8::Value>> args{
- gin::ConvertToV8(isolate, std::forward<ArgTypes>(raw))...};
- v8::MaybeLocal<v8::Value> ret = holder->Call(
- context, holder, args.size(), args.empty() ? nullptr : &args.front());
- if (ret.IsEmpty())
- return v8::Undefined(isolate);
- else
- return handle_scope.Escape(ret.ToLocalChecked());
- }
- };
- template <typename... ArgTypes>
- struct V8FunctionInvoker<void(ArgTypes...)> {
- static void Go(v8::Isolate* isolate,
- const SafeV8Function& function,
- ArgTypes... raw) {
- gin_helper::Locker locker(isolate);
- v8::HandleScope handle_scope(isolate);
- if (!function.IsAlive())
- return;
- gin_helper::MicrotasksScope microtasks_scope(isolate, true);
- v8::Local<v8::Function> holder = function.NewHandle(isolate);
- v8::Local<v8::Context> context = holder->CreationContext();
- v8::Context::Scope context_scope(context);
- std::vector<v8::Local<v8::Value>> args{
- gin::ConvertToV8(isolate, std::forward<ArgTypes>(raw))...};
- holder
- ->Call(context, holder, args.size(),
- args.empty() ? nullptr : &args.front())
- .IsEmpty();
- }
- };
- template <typename ReturnType, typename... ArgTypes>
- struct V8FunctionInvoker<ReturnType(ArgTypes...)> {
- static ReturnType Go(v8::Isolate* isolate,
- const SafeV8Function& function,
- ArgTypes... raw) {
- gin_helper::Locker locker(isolate);
- v8::HandleScope handle_scope(isolate);
- ReturnType ret = ReturnType();
- if (!function.IsAlive())
- return ret;
- gin_helper::MicrotasksScope microtasks_scope(isolate, true);
- v8::Local<v8::Function> holder = function.NewHandle(isolate);
- v8::Local<v8::Context> context = holder->CreationContext();
- v8::Context::Scope context_scope(context);
- std::vector<v8::Local<v8::Value>> args{
- gin::ConvertToV8(isolate, std::forward<ArgTypes>(raw))...};
- v8::Local<v8::Value> result;
- auto maybe_result = holder->Call(context, holder, args.size(),
- args.empty() ? nullptr : &args.front());
- if (maybe_result.ToLocal(&result))
- gin::Converter<ReturnType>::FromV8(isolate, result, &ret);
- return ret;
- }
- };
- // Helper to pass a C++ function to JavaScript.
- using Translater = base::RepeatingCallback<void(gin::Arguments* args)>;
- v8::Local<v8::Value> CreateFunctionFromTranslater(v8::Isolate* isolate,
- const Translater& translater,
- bool one_time);
- v8::Local<v8::Value> BindFunctionWith(v8::Isolate* isolate,
- v8::Local<v8::Context> context,
- v8::Local<v8::Function> func,
- v8::Local<v8::Value> arg1,
- v8::Local<v8::Value> arg2);
- // Calls callback with Arguments.
- template <typename Sig>
- struct NativeFunctionInvoker {};
- template <typename ReturnType, typename... ArgTypes>
- struct NativeFunctionInvoker<ReturnType(ArgTypes...)> {
- static void Go(base::RepeatingCallback<ReturnType(ArgTypes...)> val,
- gin::Arguments* args) {
- using Indices = typename IndicesGenerator<sizeof...(ArgTypes)>::type;
- Invoker<Indices, ArgTypes...> invoker(args, 0);
- if (invoker.IsOK())
- invoker.DispatchToCallback(val);
- }
- };
- // Convert a callback to V8 without the call number limitation, this can easily
- // cause memory leaks so use it with caution.
- template <typename Sig>
- v8::Local<v8::Value> CallbackToV8Leaked(
- v8::Isolate* isolate,
- const base::RepeatingCallback<Sig>& val) {
- Translater translater =
- base::BindRepeating(&NativeFunctionInvoker<Sig>::Go, val);
- return CreateFunctionFromTranslater(isolate, translater, false);
- }
- } // namespace gin_helper
- #endif // SHELL_COMMON_GIN_HELPER_CALLBACK_H_
|