builtins-extras-utils.cc 3.15 KB
Newer Older
1 2 3 4 5 6
// Copyright 2016 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.

#include "src/builtins/builtins-utils-inl.h"
#include "src/builtins/builtins.h"
7
#include "src/objects/elements.h"
8

9
#include "src/logging/counters.h"
10
#include "src/objects/objects-inl.h"
11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41

namespace v8 {
namespace internal {

namespace {
enum UncurryThisFunctionContextSlot {
  kFunctionSlot = Context::MIN_CONTEXT_SLOTS,
  kFunctionContextLength,
};
}  // namespace

// These functions are key for safe meta-programming:
// http://wiki.ecmascript.org/doku.php?id=conventions:safe_meta_programming
//
// Technically they could all be derived from combinations of
// Function.prototype.{bind,call,apply} but that introduces lots of layers of
// indirection.
//
// Equivalent to:
//
//   function uncurryThis(func) {
//     return function(thisArg, ...args) {
//       return %reflect_apply(func, thisArg, args);
//     };
//   };
//
BUILTIN(ExtrasUtilsUncurryThis) {
  HandleScope scope(isolate);

  DCHECK_EQ(2, args.length());
  Handle<JSFunction> function = args.at<JSFunction>(1);
42
  Handle<NativeContext> native_context(isolate->context().native_context(),
43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66
                                       isolate);
  Handle<Context> context = isolate->factory()->NewBuiltinContext(
      native_context,
      static_cast<int>(UncurryThisFunctionContextSlot::kFunctionContextLength));

  context->set(static_cast<int>(UncurryThisFunctionContextSlot::kFunctionSlot),
               *function);

  Handle<SharedFunctionInfo> info =
      isolate->factory()->NewSharedFunctionInfoForBuiltin(
          isolate->factory()->empty_string(),
          Builtins::kExtrasUtilsCallReflectApply, kNormalFunction);
  info->DontAdaptArguments();

  Handle<Map> map = isolate->strict_function_without_prototype_map();
  Handle<JSFunction> new_bound_function =
      isolate->factory()->NewFunctionFromSharedFunctionInfo(map, info, context);

  return *new_bound_function;
}

BUILTIN(ExtrasUtilsCallReflectApply) {
  HandleScope scope(isolate);
  Handle<Context> context(isolate->context(), isolate);
67
  Handle<NativeContext> native_context(isolate->context().native_context(),
68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93
                                       isolate);
  Handle<JSFunction> function(
      JSFunction::cast(context->get(
          static_cast<int>(UncurryThisFunctionContextSlot::kFunctionSlot))),
      isolate);

  Handle<Object> this_arg = args.at(1);

  int const rest_args_atart = 2;
  Arguments argv(args.length() - rest_args_atart,
                 args.address_of_arg_at(rest_args_atart));
  Handle<JSArray> rest_args_array = isolate->factory()->NewJSArray(0);
  RETURN_FAILURE_ON_EXCEPTION(
      isolate, ArrayConstructInitializeElements(rest_args_array, &argv));

  Handle<Object> reflect_apply_args[] = {function, this_arg, rest_args_array};
  Handle<JSFunction> reflect_apply(native_context->reflect_apply(), isolate);
  RETURN_RESULT_OR_FAILURE(
      isolate,
      Execution::Call(isolate, reflect_apply,
                      isolate->factory()->undefined_value(),
                      arraysize(reflect_apply_args), reflect_apply_args));
}

}  // namespace internal
}  // namespace v8