macro-assembler.h 7.48 KB
Newer Older
1
// Copyright 2012 the V8 project authors. All rights reserved.
2 3
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
4 5 6 7

#ifndef V8_MACRO_ASSEMBLER_H_
#define V8_MACRO_ASSEMBLER_H_

8
#include "src/assembler-inl.h"
9 10 11 12 13 14 15 16

// Helper types to make boolean flag easier to read at call-site.
enum InvokeFlag {
  CALL_FUNCTION,
  JUMP_FUNCTION
};


17 18 19 20 21 22
// Flags used for the AllocateInNewSpace functions.
enum AllocationFlags {
  // No special flags.
  NO_ALLOCATION_FLAGS = 0,
  // The content of the result register already contains the allocation top in
  // new space.
23
  RESULT_CONTAINS_TOP = 1 << 0,
24 25
  // Specify that the requested size of the space to allocate is specified in
  // words instead of bytes.
26
  SIZE_IN_WORDS = 1 << 1,
27
  // Align the allocation to a multiple of kDoubleSize
28
  DOUBLE_ALIGNMENT = 1 << 2,
29
  // Directly allocate in old space
30
  PRETENURE = 1 << 3,
31 32 33 34
  // Allocation folding dominator
  ALLOCATION_FOLDING_DOMINATOR = 1 << 4,
  // Folded allocation
  ALLOCATION_FOLDED = 1 << 5
35 36
};

37
#if V8_TARGET_ARCH_IA32
38
#include "src/ia32/macro-assembler-ia32.h"
39
#elif V8_TARGET_ARCH_X64
40
#include "src/x64/macro-assembler-x64.h"
41
#elif V8_TARGET_ARCH_ARM64
42 43
#include "src/arm64/constants-arm64.h"
#include "src/arm64/macro-assembler-arm64.h"
44
#include "src/arm64/macro-assembler-arm64-inl.h"
45
#elif V8_TARGET_ARCH_ARM
46 47
#include "src/arm/constants-arm.h"
#include "src/arm/macro-assembler-arm.h"
48
#elif V8_TARGET_ARCH_PPC
49
#include "src/ppc/constants-ppc.h"
50
#include "src/ppc/macro-assembler-ppc.h"
51
#elif V8_TARGET_ARCH_MIPS
52
#include "src/mips/constants-mips.h"
53
#include "src/mips/macro-assembler-mips.h"
54
#elif V8_TARGET_ARCH_MIPS64
55
#include "src/mips64/constants-mips64.h"
56
#include "src/mips64/macro-assembler-mips64.h"
57 58 59
#elif V8_TARGET_ARCH_S390
#include "src/s390/constants-s390.h"
#include "src/s390/macro-assembler-s390.h"
danno@chromium.org's avatar
danno@chromium.org committed
60
#elif V8_TARGET_ARCH_X87
61
#include "src/x87/macro-assembler-x87.h"
62 63
#else
#error Unsupported target architecture.
64 65
#endif

66 67 68
namespace v8 {
namespace internal {

69 70 71
class FrameScope {
 public:
  explicit FrameScope(MacroAssembler* masm, StackFrame::Type type)
72
      : masm_(masm), type_(type), old_has_frame_(masm->has_frame()) {
73 74 75 76 77 78 79 80 81 82
    masm->set_has_frame(true);
    if (type != StackFrame::MANUAL && type_ != StackFrame::NONE) {
      masm->EnterFrame(type);
    }
  }

  ~FrameScope() {
    if (type_ != StackFrame::MANUAL && type_ != StackFrame::NONE) {
      masm_->LeaveFrame(type_);
    }
83
    masm_->set_has_frame(old_has_frame_);
84 85 86 87 88 89 90 91
  }

  // Normally we generate the leave-frame code when this object goes
  // out of scope.  Sometimes we may need to generate the code somewhere else
  // in addition.  Calling this will achieve that, but the object stays in
  // scope, the MacroAssembler is still marked as being in a frame scope, and
  // the code will be generated again when it goes out of scope.
  void GenerateLeaveFrame() {
92
    DCHECK(type_ != StackFrame::MANUAL && type_ != StackFrame::NONE);
93 94 95 96 97 98
    masm_->LeaveFrame(type_);
  }

 private:
  MacroAssembler* masm_;
  StackFrame::Type type_;
99
  bool old_has_frame_;
100 101
};

102 103 104 105 106 107
class FrameAndConstantPoolScope {
 public:
  FrameAndConstantPoolScope(MacroAssembler* masm, StackFrame::Type type)
      : masm_(masm),
        type_(type),
        old_has_frame_(masm->has_frame()),
108 109
        old_constant_pool_available_(FLAG_enable_embedded_constant_pool &&
                                     masm->is_constant_pool_available()) {
110
    masm->set_has_frame(true);
111 112
    if (FLAG_enable_embedded_constant_pool) {
      masm->set_constant_pool_available(true);
113 114 115 116 117 118 119 120 121
    }
    if (type_ != StackFrame::MANUAL && type_ != StackFrame::NONE) {
      masm->EnterFrame(type, !old_constant_pool_available_);
    }
  }

  ~FrameAndConstantPoolScope() {
    masm_->LeaveFrame(type_);
    masm_->set_has_frame(old_has_frame_);
122 123
    if (FLAG_enable_embedded_constant_pool) {
      masm_->set_constant_pool_available(old_constant_pool_available_);
124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150
    }
  }

  // Normally we generate the leave-frame code when this object goes
  // out of scope.  Sometimes we may need to generate the code somewhere else
  // in addition.  Calling this will achieve that, but the object stays in
  // scope, the MacroAssembler is still marked as being in a frame scope, and
  // the code will be generated again when it goes out of scope.
  void GenerateLeaveFrame() {
    DCHECK(type_ != StackFrame::MANUAL && type_ != StackFrame::NONE);
    masm_->LeaveFrame(type_);
  }

 private:
  MacroAssembler* masm_;
  StackFrame::Type type_;
  bool old_has_frame_;
  bool old_constant_pool_available_;

  DISALLOW_IMPLICIT_CONSTRUCTORS(FrameAndConstantPoolScope);
};

// Class for scoping the the unavailability of constant pool access.
class ConstantPoolUnavailableScope {
 public:
  explicit ConstantPoolUnavailableScope(MacroAssembler* masm)
      : masm_(masm),
151 152 153 154
        old_constant_pool_available_(FLAG_enable_embedded_constant_pool &&
                                     masm->is_constant_pool_available()) {
    if (FLAG_enable_embedded_constant_pool) {
      masm_->set_constant_pool_available(false);
155 156 157
    }
  }
  ~ConstantPoolUnavailableScope() {
158 159
    if (FLAG_enable_embedded_constant_pool) {
      masm_->set_constant_pool_available(old_constant_pool_available_);
160 161 162 163 164 165 166 167 168 169
    }
  }

 private:
  MacroAssembler* masm_;
  int old_constant_pool_available_;

  DISALLOW_IMPLICIT_CONSTRUCTORS(ConstantPoolUnavailableScope);
};

170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194

class AllowExternalCallThatCantCauseGC: public FrameScope {
 public:
  explicit AllowExternalCallThatCantCauseGC(MacroAssembler* masm)
      : FrameScope(masm, StackFrame::NONE) { }
};


class NoCurrentFrameScope {
 public:
  explicit NoCurrentFrameScope(MacroAssembler* masm)
      : masm_(masm), saved_(masm->has_frame()) {
    masm->set_has_frame(false);
  }

  ~NoCurrentFrameScope() {
    masm_->set_has_frame(saved_);
  }

 private:
  MacroAssembler* masm_;
  bool saved_;
};


195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216
// Support for "structured" code comments.
#ifdef DEBUG

class Comment {
 public:
  Comment(MacroAssembler* masm, const char* msg);
  ~Comment();

 private:
  MacroAssembler* masm_;
  const char* msg_;
};

#else

class Comment {
 public:
  Comment(MacroAssembler*, const char*)  {}
};

#endif  // DEBUG

217

218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246
// Wrapper class for passing expected and actual parameter counts as
// either registers or immediate values. Used to make sure that the
// caller provides exactly the expected number of parameters to the
// callee.
class ParameterCount BASE_EMBEDDED {
 public:
  explicit ParameterCount(Register reg) : reg_(reg), immediate_(0) {}
  explicit ParameterCount(int imm) : reg_(no_reg), immediate_(imm) {}

  bool is_reg() const { return !reg_.is(no_reg); }
  bool is_immediate() const { return !is_reg(); }

  Register reg() const {
    DCHECK(is_reg());
    return reg_;
  }
  int immediate() const {
    DCHECK(is_immediate());
    return immediate_;
  }

 private:
  const Register reg_;
  const int immediate_;

  DISALLOW_IMPLICIT_CONSTRUCTORS(ParameterCount);
};


247 248 249 250
class AllocationUtils {
 public:
  static ExternalReference GetAllocationTopReference(
      Isolate* isolate, AllocationFlags flags) {
251 252
    if ((flags & PRETENURE) != 0) {
      return ExternalReference::old_space_allocation_top_address(isolate);
253 254
    }
    return ExternalReference::new_space_allocation_top_address(isolate);
255 256 257 258 259
  }


  static ExternalReference GetAllocationLimitReference(
      Isolate* isolate, AllocationFlags flags) {
260 261
    if ((flags & PRETENURE) != 0) {
      return ExternalReference::old_space_allocation_limit_address(isolate);
262 263
    }
    return ExternalReference::new_space_allocation_limit_address(isolate);
264 265 266 267
  }
};


268 269
}  // namespace internal
}  // namespace v8
270

271
#endif  // V8_MACRO_ASSEMBLER_H_