2008-08-27 10:11:39 +00:00
|
|
|
// Copyright (c) 1994-2006 Sun Microsystems Inc.
|
|
|
|
// All Rights Reserved.
|
|
|
|
//
|
2008-07-03 15:10:15 +00:00
|
|
|
// Redistribution and use in source and binary forms, with or without
|
2008-08-27 10:11:39 +00:00
|
|
|
// modification, are permitted provided that the following conditions
|
|
|
|
// are met:
|
|
|
|
//
|
|
|
|
// - Redistributions of source code must retain the above copyright notice,
|
|
|
|
// this list of conditions and the following disclaimer.
|
2008-07-03 15:10:15 +00:00
|
|
|
//
|
2008-08-27 10:11:39 +00:00
|
|
|
// - Redistribution in binary form must reproduce the above copyright
|
|
|
|
// notice, this list of conditions and the following disclaimer in the
|
|
|
|
// documentation and/or other materials provided with the
|
|
|
|
// distribution.
|
|
|
|
//
|
|
|
|
// - Neither the name of Sun Microsystems or the names of contributors may
|
|
|
|
// be used to endorse or promote products derived from this software without
|
|
|
|
// specific prior written permission.
|
2008-07-03 15:10:15 +00:00
|
|
|
//
|
|
|
|
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
|
|
|
// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
2008-08-27 10:11:39 +00:00
|
|
|
// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
|
|
|
|
// FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE
|
|
|
|
// COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
|
|
|
|
// INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
|
|
|
|
// (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
|
|
|
|
// SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
|
|
|
|
// HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
|
|
|
|
// STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
|
|
|
|
// ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED
|
|
|
|
// OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
|
|
|
2010-01-25 11:54:10 +00:00
|
|
|
// The original source code covered by the above license above has been
|
|
|
|
// modified significantly by Google Inc.
|
2012-01-10 16:59:55 +00:00
|
|
|
// Copyright 2012 the V8 project authors. All rights reserved.
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
// A light-weight ARM Assembler
|
|
|
|
// Generates user mode instructions for the ARM architecture up to version 5
|
|
|
|
|
2009-05-04 13:36:43 +00:00
|
|
|
#ifndef V8_ARM_ASSEMBLER_ARM_H_
|
|
|
|
#define V8_ARM_ASSEMBLER_ARM_H_
|
2009-08-31 12:40:37 +00:00
|
|
|
#include <stdio.h>
|
2008-07-03 15:10:15 +00:00
|
|
|
#include "assembler.h"
|
2011-01-26 08:32:54 +00:00
|
|
|
#include "constants-arm.h"
|
2009-11-13 12:32:57 +00:00
|
|
|
#include "serialize.h"
|
2008-07-03 15:10:15 +00:00
|
|
|
|
2009-05-25 10:05:56 +00:00
|
|
|
namespace v8 {
|
|
|
|
namespace internal {
|
2008-07-03 15:10:15 +00:00
|
|
|
|
2013-01-23 16:29:48 +00:00
|
|
|
// CpuFeatures keeps track of which features are supported by the target CPU.
|
|
|
|
// Supported features must be enabled by a Scope before use.
|
|
|
|
class CpuFeatures : public AllStatic {
|
|
|
|
public:
|
|
|
|
// Detect features of the target CPU. Set safe defaults if the serializer
|
|
|
|
// is enabled (snapshots must be portable).
|
|
|
|
static void Probe();
|
|
|
|
|
|
|
|
// Check whether a feature is supported by the target CPU.
|
|
|
|
static bool IsSupported(CpuFeature f) {
|
|
|
|
ASSERT(initialized_);
|
|
|
|
if (f == VFP3 && !FLAG_enable_vfp3) return false;
|
|
|
|
if (f == VFP2 && !FLAG_enable_vfp2) return false;
|
|
|
|
if (f == SUDIV && !FLAG_enable_sudiv) return false;
|
|
|
|
if (f == UNALIGNED_ACCESSES && !FLAG_enable_unaligned_accesses) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (f == VFP32DREGS && !FLAG_enable_32dregs) return false;
|
|
|
|
return (supported_ & (1u << f)) != 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef DEBUG
|
|
|
|
// Check whether a feature is currently enabled.
|
|
|
|
static bool IsEnabled(CpuFeature f) {
|
|
|
|
ASSERT(initialized_);
|
|
|
|
Isolate* isolate = Isolate::UncheckedCurrent();
|
|
|
|
if (isolate == NULL) {
|
|
|
|
// When no isolate is available, work as if we're running in
|
|
|
|
// release mode.
|
|
|
|
return IsSupported(f);
|
|
|
|
}
|
|
|
|
unsigned enabled = static_cast<unsigned>(isolate->enabled_cpu_features());
|
|
|
|
return (enabled & (1u << f)) != 0;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
// Enable a specified feature within a scope.
|
|
|
|
class Scope BASE_EMBEDDED {
|
|
|
|
#ifdef DEBUG
|
|
|
|
|
|
|
|
public:
|
|
|
|
explicit Scope(CpuFeature f) {
|
|
|
|
unsigned mask = 1u << f;
|
|
|
|
// VFP2 and ARMv7 are implied by VFP3.
|
|
|
|
if (f == VFP3) mask |= 1u << VFP2 | 1u << ARMv7;
|
|
|
|
ASSERT(CpuFeatures::IsSupported(f));
|
|
|
|
ASSERT(!Serializer::enabled() ||
|
|
|
|
(CpuFeatures::found_by_runtime_probing_ & mask) == 0);
|
|
|
|
isolate_ = Isolate::UncheckedCurrent();
|
|
|
|
old_enabled_ = 0;
|
|
|
|
if (isolate_ != NULL) {
|
|
|
|
old_enabled_ = static_cast<unsigned>(isolate_->enabled_cpu_features());
|
|
|
|
isolate_->set_enabled_cpu_features(old_enabled_ | mask);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
~Scope() {
|
|
|
|
ASSERT_EQ(Isolate::UncheckedCurrent(), isolate_);
|
|
|
|
if (isolate_ != NULL) {
|
|
|
|
isolate_->set_enabled_cpu_features(old_enabled_);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
Isolate* isolate_;
|
|
|
|
unsigned old_enabled_;
|
|
|
|
#else
|
|
|
|
|
|
|
|
public:
|
|
|
|
explicit Scope(CpuFeature f) {}
|
|
|
|
#endif
|
|
|
|
};
|
|
|
|
|
|
|
|
class TryForceFeatureScope BASE_EMBEDDED {
|
|
|
|
public:
|
|
|
|
explicit TryForceFeatureScope(CpuFeature f)
|
|
|
|
: old_supported_(CpuFeatures::supported_) {
|
|
|
|
if (CanForce()) {
|
|
|
|
CpuFeatures::supported_ |= (1u << f);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
~TryForceFeatureScope() {
|
|
|
|
if (CanForce()) {
|
|
|
|
CpuFeatures::supported_ = old_supported_;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
static bool CanForce() {
|
|
|
|
// It's only safe to temporarily force support of CPU features
|
|
|
|
// when there's only a single isolate, which is guaranteed when
|
|
|
|
// the serializer is enabled.
|
|
|
|
return Serializer::enabled();
|
|
|
|
}
|
|
|
|
|
|
|
|
const unsigned old_supported_;
|
|
|
|
};
|
|
|
|
|
|
|
|
private:
|
|
|
|
#ifdef DEBUG
|
|
|
|
static bool initialized_;
|
|
|
|
#endif
|
|
|
|
static unsigned supported_;
|
|
|
|
static unsigned found_by_runtime_probing_;
|
|
|
|
|
|
|
|
friend class ExternalReference;
|
|
|
|
DISALLOW_COPY_AND_ASSIGN(CpuFeatures);
|
|
|
|
};
|
|
|
|
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
// CPU Registers.
|
|
|
|
//
|
|
|
|
// 1) We would prefer to use an enum, but enum values are assignment-
|
|
|
|
// compatible with int, which has caused code-generation bugs.
|
|
|
|
//
|
|
|
|
// 2) We would prefer to use a class instead of a struct but we don't like
|
|
|
|
// the register initialization to depend on the particular initialization
|
|
|
|
// order (which appears to be different on OS X, Linux, and Windows for the
|
|
|
|
// installed versions of C++ we tried). Using a struct permits C-style
|
|
|
|
// "initialization". Also, the Register objects cannot be const as this
|
|
|
|
// forces initialization stubs in MSVC, making us dependent on initialization
|
|
|
|
// order.
|
|
|
|
//
|
|
|
|
// 3) By not using an enum, we are possibly preventing the compiler from
|
|
|
|
// doing certain constant folds, which may significantly reduce the
|
|
|
|
// code generated for some assembly instructions (because they boil down
|
|
|
|
// to a few constants). If this is a problem, we could change the code
|
|
|
|
// such that we use an enum in optimized mode, and the struct in debug
|
|
|
|
// mode. This way we get the compile-time error checking in debug mode
|
|
|
|
// and best performance in optimized code.
|
2011-01-04 14:32:54 +00:00
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
// Core register
|
|
|
|
struct Register {
|
2010-12-07 11:31:57 +00:00
|
|
|
static const int kNumRegisters = 16;
|
2012-12-18 16:25:45 +00:00
|
|
|
static const int kMaxNumAllocatableRegisters = 8;
|
2011-04-21 07:15:43 +00:00
|
|
|
static const int kSizeInBytes = 4;
|
2012-12-18 16:25:45 +00:00
|
|
|
static const int kGPRsPerNonVFP2Double = 2;
|
|
|
|
|
|
|
|
inline static int NumAllocatableRegisters();
|
2010-12-07 11:31:57 +00:00
|
|
|
|
|
|
|
static int ToAllocationIndex(Register reg) {
|
2012-12-18 16:25:45 +00:00
|
|
|
ASSERT(reg.code() < kMaxNumAllocatableRegisters);
|
2010-12-07 11:31:57 +00:00
|
|
|
return reg.code();
|
|
|
|
}
|
|
|
|
|
|
|
|
static Register FromAllocationIndex(int index) {
|
2012-12-18 16:25:45 +00:00
|
|
|
ASSERT(index >= 0 && index < kMaxNumAllocatableRegisters);
|
2010-12-07 11:31:57 +00:00
|
|
|
return from_code(index);
|
|
|
|
}
|
|
|
|
|
|
|
|
static const char* AllocationIndexToString(int index) {
|
2012-12-18 16:25:45 +00:00
|
|
|
ASSERT(index >= 0 && index < kMaxNumAllocatableRegisters);
|
2010-12-07 11:31:57 +00:00
|
|
|
const char* const names[] = {
|
|
|
|
"r0",
|
|
|
|
"r1",
|
|
|
|
"r2",
|
|
|
|
"r3",
|
|
|
|
"r4",
|
|
|
|
"r5",
|
|
|
|
"r6",
|
|
|
|
"r7",
|
|
|
|
};
|
|
|
|
return names[index];
|
|
|
|
}
|
|
|
|
|
|
|
|
static Register from_code(int code) {
|
|
|
|
Register r = { code };
|
|
|
|
return r;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool is_valid() const { return 0 <= code_ && code_ < kNumRegisters; }
|
2010-09-24 08:25:31 +00:00
|
|
|
bool is(Register reg) const { return code_ == reg.code_; }
|
|
|
|
int code() const {
|
2008-07-03 15:10:15 +00:00
|
|
|
ASSERT(is_valid());
|
|
|
|
return code_;
|
|
|
|
}
|
2010-09-24 08:25:31 +00:00
|
|
|
int bit() const {
|
2008-07-03 15:10:15 +00:00
|
|
|
ASSERT(is_valid());
|
|
|
|
return 1 << code_;
|
|
|
|
}
|
|
|
|
|
2010-05-18 06:38:42 +00:00
|
|
|
void set_code(int code) {
|
|
|
|
code_ = code;
|
|
|
|
ASSERT(is_valid());
|
|
|
|
}
|
|
|
|
|
2010-02-04 21:32:02 +00:00
|
|
|
// Unfortunately we can't make this private in a struct.
|
2008-07-03 15:10:15 +00:00
|
|
|
int code_;
|
|
|
|
};
|
|
|
|
|
2012-03-12 13:56:56 +00:00
|
|
|
// These constants are used in several locations, including static initializers
|
|
|
|
const int kRegister_no_reg_Code = -1;
|
|
|
|
const int kRegister_r0_Code = 0;
|
|
|
|
const int kRegister_r1_Code = 1;
|
|
|
|
const int kRegister_r2_Code = 2;
|
|
|
|
const int kRegister_r3_Code = 3;
|
|
|
|
const int kRegister_r4_Code = 4;
|
|
|
|
const int kRegister_r5_Code = 5;
|
|
|
|
const int kRegister_r6_Code = 6;
|
|
|
|
const int kRegister_r7_Code = 7;
|
|
|
|
const int kRegister_r8_Code = 8;
|
|
|
|
const int kRegister_r9_Code = 9;
|
|
|
|
const int kRegister_r10_Code = 10;
|
|
|
|
const int kRegister_fp_Code = 11;
|
|
|
|
const int kRegister_ip_Code = 12;
|
|
|
|
const int kRegister_sp_Code = 13;
|
|
|
|
const int kRegister_lr_Code = 14;
|
|
|
|
const int kRegister_pc_Code = 15;
|
|
|
|
|
|
|
|
const Register no_reg = { kRegister_no_reg_Code };
|
|
|
|
|
|
|
|
const Register r0 = { kRegister_r0_Code };
|
|
|
|
const Register r1 = { kRegister_r1_Code };
|
|
|
|
const Register r2 = { kRegister_r2_Code };
|
|
|
|
const Register r3 = { kRegister_r3_Code };
|
|
|
|
const Register r4 = { kRegister_r4_Code };
|
|
|
|
const Register r5 = { kRegister_r5_Code };
|
|
|
|
const Register r6 = { kRegister_r6_Code };
|
|
|
|
const Register r7 = { kRegister_r7_Code };
|
|
|
|
// Used as context register.
|
|
|
|
const Register r8 = { kRegister_r8_Code };
|
|
|
|
// Used as lithium codegen scratch register.
|
|
|
|
const Register r9 = { kRegister_r9_Code };
|
|
|
|
// Used as roots register.
|
|
|
|
const Register r10 = { kRegister_r10_Code };
|
|
|
|
const Register fp = { kRegister_fp_Code };
|
|
|
|
const Register ip = { kRegister_ip_Code };
|
|
|
|
const Register sp = { kRegister_sp_Code };
|
|
|
|
const Register lr = { kRegister_lr_Code };
|
|
|
|
const Register pc = { kRegister_pc_Code };
|
|
|
|
|
2009-12-09 11:14:45 +00:00
|
|
|
// Single word VFP register.
|
|
|
|
struct SwVfpRegister {
|
2010-09-24 08:25:31 +00:00
|
|
|
bool is_valid() const { return 0 <= code_ && code_ < 32; }
|
|
|
|
bool is(SwVfpRegister reg) const { return code_ == reg.code_; }
|
|
|
|
int code() const {
|
2009-12-09 11:14:45 +00:00
|
|
|
ASSERT(is_valid());
|
|
|
|
return code_;
|
|
|
|
}
|
2010-09-24 08:25:31 +00:00
|
|
|
int bit() const {
|
2009-12-09 11:14:45 +00:00
|
|
|
ASSERT(is_valid());
|
|
|
|
return 1 << code_;
|
|
|
|
}
|
2010-09-24 08:25:31 +00:00
|
|
|
void split_code(int* vm, int* m) const {
|
2010-08-26 08:53:00 +00:00
|
|
|
ASSERT(is_valid());
|
|
|
|
*m = code_ & 0x1;
|
|
|
|
*vm = code_ >> 1;
|
|
|
|
}
|
2009-12-09 11:14:45 +00:00
|
|
|
|
|
|
|
int code_;
|
|
|
|
};
|
|
|
|
|
|
|
|
|
2012-12-03 17:16:51 +00:00
|
|
|
// Double word VFP register.
|
|
|
|
struct DwVfpRegister {
|
2013-02-04 12:01:59 +00:00
|
|
|
static const int kMaxNumRegisters = 32;
|
2011-06-29 10:51:06 +00:00
|
|
|
// A few double registers are reserved: one as a scratch register and one to
|
|
|
|
// hold 0.0, that does not fit in the immediate field of vmov instructions.
|
|
|
|
// d14: 0.0
|
|
|
|
// d15: scratch register.
|
|
|
|
static const int kNumReservedRegisters = 2;
|
2013-02-04 12:01:59 +00:00
|
|
|
static const int kMaxNumAllocatableRegisters = kMaxNumRegisters -
|
2011-06-29 10:51:06 +00:00
|
|
|
kNumReservedRegisters;
|
2010-12-07 11:31:57 +00:00
|
|
|
|
2013-01-23 16:29:48 +00:00
|
|
|
// Note: the number of registers can be different at snapshot and run-time.
|
|
|
|
// Any code included in the snapshot must be able to run both with 16 or 32
|
|
|
|
// registers.
|
2012-12-18 16:25:45 +00:00
|
|
|
inline static int NumRegisters();
|
|
|
|
inline static int NumAllocatableRegisters();
|
2013-01-23 16:29:48 +00:00
|
|
|
|
2012-12-03 17:16:51 +00:00
|
|
|
inline static int ToAllocationIndex(DwVfpRegister reg);
|
2012-12-18 16:25:45 +00:00
|
|
|
static const char* AllocationIndexToString(int index);
|
2013-01-23 16:29:48 +00:00
|
|
|
inline static DwVfpRegister FromAllocationIndex(int index);
|
2010-12-07 11:31:57 +00:00
|
|
|
|
|
|
|
static DwVfpRegister from_code(int code) {
|
2012-12-03 17:16:51 +00:00
|
|
|
DwVfpRegister r = { code };
|
|
|
|
return r;
|
2010-12-07 11:31:57 +00:00
|
|
|
}
|
|
|
|
|
2013-01-23 16:29:48 +00:00
|
|
|
bool is_valid() const {
|
2013-02-04 12:01:59 +00:00
|
|
|
return 0 <= code_ && code_ < kMaxNumRegisters;
|
2013-01-23 16:29:48 +00:00
|
|
|
}
|
2010-09-24 08:25:31 +00:00
|
|
|
bool is(DwVfpRegister reg) const { return code_ == reg.code_; }
|
|
|
|
SwVfpRegister low() const {
|
2013-01-23 16:29:48 +00:00
|
|
|
ASSERT(code_ < 16);
|
2010-07-08 12:38:02 +00:00
|
|
|
SwVfpRegister reg;
|
|
|
|
reg.code_ = code_ * 2;
|
|
|
|
|
|
|
|
ASSERT(reg.is_valid());
|
|
|
|
return reg;
|
|
|
|
}
|
2010-09-24 08:25:31 +00:00
|
|
|
SwVfpRegister high() const {
|
2013-01-23 16:29:48 +00:00
|
|
|
ASSERT(code_ < 16);
|
2010-07-08 12:38:02 +00:00
|
|
|
SwVfpRegister reg;
|
|
|
|
reg.code_ = (code_ * 2) + 1;
|
|
|
|
|
|
|
|
ASSERT(reg.is_valid());
|
|
|
|
return reg;
|
|
|
|
}
|
2010-09-24 08:25:31 +00:00
|
|
|
int code() const {
|
2009-12-09 11:14:45 +00:00
|
|
|
ASSERT(is_valid());
|
|
|
|
return code_;
|
|
|
|
}
|
2010-09-24 08:25:31 +00:00
|
|
|
int bit() const {
|
2009-12-09 11:14:45 +00:00
|
|
|
ASSERT(is_valid());
|
|
|
|
return 1 << code_;
|
|
|
|
}
|
2010-09-24 08:25:31 +00:00
|
|
|
void split_code(int* vm, int* m) const {
|
2010-08-26 08:53:00 +00:00
|
|
|
ASSERT(is_valid());
|
|
|
|
*m = (code_ & 0x10) >> 4;
|
|
|
|
*vm = code_ & 0x0F;
|
|
|
|
}
|
2012-12-03 15:51:05 +00:00
|
|
|
|
2012-12-03 17:16:51 +00:00
|
|
|
int code_;
|
2009-12-09 11:14:45 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
|
2012-12-03 17:16:51 +00:00
|
|
|
typedef DwVfpRegister DoubleRegister;
|
2010-12-07 11:31:57 +00:00
|
|
|
|
|
|
|
|
2010-03-01 08:34:09 +00:00
|
|
|
// Support for the VFP registers s0 to s31 (d0 to d15).
|
2009-12-09 11:14:45 +00:00
|
|
|
// Note that "s(N):s(N+1)" is the same as "d(N/2)".
|
2010-03-01 08:34:09 +00:00
|
|
|
const SwVfpRegister s0 = { 0 };
|
|
|
|
const SwVfpRegister s1 = { 1 };
|
|
|
|
const SwVfpRegister s2 = { 2 };
|
|
|
|
const SwVfpRegister s3 = { 3 };
|
|
|
|
const SwVfpRegister s4 = { 4 };
|
|
|
|
const SwVfpRegister s5 = { 5 };
|
|
|
|
const SwVfpRegister s6 = { 6 };
|
|
|
|
const SwVfpRegister s7 = { 7 };
|
|
|
|
const SwVfpRegister s8 = { 8 };
|
|
|
|
const SwVfpRegister s9 = { 9 };
|
|
|
|
const SwVfpRegister s10 = { 10 };
|
|
|
|
const SwVfpRegister s11 = { 11 };
|
|
|
|
const SwVfpRegister s12 = { 12 };
|
|
|
|
const SwVfpRegister s13 = { 13 };
|
|
|
|
const SwVfpRegister s14 = { 14 };
|
|
|
|
const SwVfpRegister s15 = { 15 };
|
|
|
|
const SwVfpRegister s16 = { 16 };
|
|
|
|
const SwVfpRegister s17 = { 17 };
|
|
|
|
const SwVfpRegister s18 = { 18 };
|
|
|
|
const SwVfpRegister s19 = { 19 };
|
|
|
|
const SwVfpRegister s20 = { 20 };
|
|
|
|
const SwVfpRegister s21 = { 21 };
|
|
|
|
const SwVfpRegister s22 = { 22 };
|
|
|
|
const SwVfpRegister s23 = { 23 };
|
|
|
|
const SwVfpRegister s24 = { 24 };
|
|
|
|
const SwVfpRegister s25 = { 25 };
|
|
|
|
const SwVfpRegister s26 = { 26 };
|
|
|
|
const SwVfpRegister s27 = { 27 };
|
|
|
|
const SwVfpRegister s28 = { 28 };
|
|
|
|
const SwVfpRegister s29 = { 29 };
|
|
|
|
const SwVfpRegister s30 = { 30 };
|
|
|
|
const SwVfpRegister s31 = { 31 };
|
|
|
|
|
2012-12-03 17:16:51 +00:00
|
|
|
const DwVfpRegister no_dreg = { -1 };
|
|
|
|
const DwVfpRegister d0 = { 0 };
|
|
|
|
const DwVfpRegister d1 = { 1 };
|
|
|
|
const DwVfpRegister d2 = { 2 };
|
|
|
|
const DwVfpRegister d3 = { 3 };
|
|
|
|
const DwVfpRegister d4 = { 4 };
|
|
|
|
const DwVfpRegister d5 = { 5 };
|
|
|
|
const DwVfpRegister d6 = { 6 };
|
|
|
|
const DwVfpRegister d7 = { 7 };
|
|
|
|
const DwVfpRegister d8 = { 8 };
|
|
|
|
const DwVfpRegister d9 = { 9 };
|
|
|
|
const DwVfpRegister d10 = { 10 };
|
|
|
|
const DwVfpRegister d11 = { 11 };
|
|
|
|
const DwVfpRegister d12 = { 12 };
|
|
|
|
const DwVfpRegister d13 = { 13 };
|
|
|
|
const DwVfpRegister d14 = { 14 };
|
|
|
|
const DwVfpRegister d15 = { 15 };
|
2013-01-23 16:29:48 +00:00
|
|
|
const DwVfpRegister d16 = { 16 };
|
|
|
|
const DwVfpRegister d17 = { 17 };
|
|
|
|
const DwVfpRegister d18 = { 18 };
|
|
|
|
const DwVfpRegister d19 = { 19 };
|
|
|
|
const DwVfpRegister d20 = { 20 };
|
|
|
|
const DwVfpRegister d21 = { 21 };
|
|
|
|
const DwVfpRegister d22 = { 22 };
|
|
|
|
const DwVfpRegister d23 = { 23 };
|
|
|
|
const DwVfpRegister d24 = { 24 };
|
|
|
|
const DwVfpRegister d25 = { 25 };
|
|
|
|
const DwVfpRegister d26 = { 26 };
|
|
|
|
const DwVfpRegister d27 = { 27 };
|
|
|
|
const DwVfpRegister d28 = { 28 };
|
|
|
|
const DwVfpRegister d29 = { 29 };
|
|
|
|
const DwVfpRegister d30 = { 30 };
|
|
|
|
const DwVfpRegister d31 = { 31 };
|
2009-12-09 11:14:45 +00:00
|
|
|
|
2012-12-18 16:25:45 +00:00
|
|
|
const Register sfpd_lo = { kRegister_r6_Code };
|
|
|
|
const Register sfpd_hi = { kRegister_r7_Code };
|
|
|
|
|
2012-01-24 16:36:55 +00:00
|
|
|
// Aliases for double registers. Defined using #define instead of
|
|
|
|
// "static const DwVfpRegister&" because Clang complains otherwise when a
|
|
|
|
// compilation unit that includes this header doesn't use the variables.
|
|
|
|
#define kFirstCalleeSavedDoubleReg d8
|
|
|
|
#define kLastCalleeSavedDoubleReg d15
|
|
|
|
#define kDoubleRegZero d14
|
|
|
|
#define kScratchDoubleReg d15
|
2011-06-22 06:24:34 +00:00
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
// Coprocessor register
|
|
|
|
struct CRegister {
|
2010-09-24 08:25:31 +00:00
|
|
|
bool is_valid() const { return 0 <= code_ && code_ < 16; }
|
|
|
|
bool is(CRegister creg) const { return code_ == creg.code_; }
|
|
|
|
int code() const {
|
2008-07-03 15:10:15 +00:00
|
|
|
ASSERT(is_valid());
|
|
|
|
return code_;
|
|
|
|
}
|
2010-09-24 08:25:31 +00:00
|
|
|
int bit() const {
|
2008-07-03 15:10:15 +00:00
|
|
|
ASSERT(is_valid());
|
|
|
|
return 1 << code_;
|
|
|
|
}
|
|
|
|
|
2010-02-04 21:32:02 +00:00
|
|
|
// Unfortunately we can't make this private in a struct.
|
2008-07-03 15:10:15 +00:00
|
|
|
int code_;
|
|
|
|
};
|
|
|
|
|
|
|
|
|
2010-03-01 08:34:09 +00:00
|
|
|
const CRegister no_creg = { -1 };
|
|
|
|
|
|
|
|
const CRegister cr0 = { 0 };
|
|
|
|
const CRegister cr1 = { 1 };
|
|
|
|
const CRegister cr2 = { 2 };
|
|
|
|
const CRegister cr3 = { 3 };
|
|
|
|
const CRegister cr4 = { 4 };
|
|
|
|
const CRegister cr5 = { 5 };
|
|
|
|
const CRegister cr6 = { 6 };
|
|
|
|
const CRegister cr7 = { 7 };
|
|
|
|
const CRegister cr8 = { 8 };
|
|
|
|
const CRegister cr9 = { 9 };
|
|
|
|
const CRegister cr10 = { 10 };
|
|
|
|
const CRegister cr11 = { 11 };
|
|
|
|
const CRegister cr12 = { 12 };
|
|
|
|
const CRegister cr13 = { 13 };
|
|
|
|
const CRegister cr14 = { 14 };
|
|
|
|
const CRegister cr15 = { 15 };
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
|
|
|
|
// Coprocessor number
|
|
|
|
enum Coprocessor {
|
|
|
|
p0 = 0,
|
|
|
|
p1 = 1,
|
|
|
|
p2 = 2,
|
|
|
|
p3 = 3,
|
|
|
|
p4 = 4,
|
|
|
|
p5 = 5,
|
|
|
|
p6 = 6,
|
|
|
|
p7 = 7,
|
|
|
|
p8 = 8,
|
|
|
|
p9 = 9,
|
|
|
|
p10 = 10,
|
|
|
|
p11 = 11,
|
|
|
|
p12 = 12,
|
|
|
|
p13 = 13,
|
|
|
|
p14 = 14,
|
|
|
|
p15 = 15
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
// -----------------------------------------------------------------------------
|
|
|
|
// Machine instruction Operands
|
|
|
|
|
|
|
|
// Class Operand represents a shifter operand in data processing instructions
|
|
|
|
class Operand BASE_EMBEDDED {
|
|
|
|
public:
|
|
|
|
// immediate
|
2008-09-22 13:57:03 +00:00
|
|
|
INLINE(explicit Operand(int32_t immediate,
|
2013-01-04 10:56:24 +00:00
|
|
|
RelocInfo::Mode rmode = RelocInfo::NONE32));
|
2011-08-23 12:00:09 +00:00
|
|
|
INLINE(static Operand Zero()) {
|
|
|
|
return Operand(static_cast<int32_t>(0));
|
|
|
|
}
|
2008-07-03 15:10:15 +00:00
|
|
|
INLINE(explicit Operand(const ExternalReference& f));
|
|
|
|
explicit Operand(Handle<Object> handle);
|
|
|
|
INLINE(explicit Operand(Smi* value));
|
|
|
|
|
|
|
|
// rm
|
|
|
|
INLINE(explicit Operand(Register rm));
|
|
|
|
|
|
|
|
// rm <shift_op> shift_imm
|
|
|
|
explicit Operand(Register rm, ShiftOp shift_op, int shift_imm);
|
|
|
|
|
|
|
|
// rm <shift_op> rs
|
|
|
|
explicit Operand(Register rm, ShiftOp shift_op, Register rs);
|
|
|
|
|
2008-08-13 09:32:07 +00:00
|
|
|
// Return true if this is a register operand.
|
|
|
|
INLINE(bool is_reg() const);
|
|
|
|
|
2011-02-17 10:07:13 +00:00
|
|
|
// Return true if this operand fits in one instruction so that no
|
2011-03-10 13:58:20 +00:00
|
|
|
// 2-instruction solution with a load into the ip register is necessary. If
|
|
|
|
// the instruction this operand is used for is a MOV or MVN instruction the
|
|
|
|
// actual instruction to use is required for this calculation. For other
|
|
|
|
// instructions instr is ignored.
|
2012-08-10 12:24:06 +00:00
|
|
|
bool is_single_instruction(const Assembler* assembler, Instr instr = 0) const;
|
2012-10-18 12:21:42 +00:00
|
|
|
bool must_output_reloc_info(const Assembler* assembler) const;
|
2010-06-14 11:20:36 +00:00
|
|
|
|
|
|
|
inline int32_t immediate() const {
|
|
|
|
ASSERT(!rm_.is_valid());
|
|
|
|
return imm32_;
|
|
|
|
}
|
|
|
|
|
2008-08-13 09:32:07 +00:00
|
|
|
Register rm() const { return rm_; }
|
2010-07-21 07:42:51 +00:00
|
|
|
Register rs() const { return rs_; }
|
|
|
|
ShiftOp shift_op() const { return shift_op_; }
|
2008-08-13 09:32:07 +00:00
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
private:
|
|
|
|
Register rm_;
|
|
|
|
Register rs_;
|
|
|
|
ShiftOp shift_op_;
|
|
|
|
int shift_imm_; // valid if rm_ != no_reg && rs_ == no_reg
|
|
|
|
int32_t imm32_; // valid if rm_ == no_reg
|
2008-09-22 13:57:03 +00:00
|
|
|
RelocInfo::Mode rmode_;
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
friend class Assembler;
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
// Class MemOperand represents a memory operand in load and store instructions
|
|
|
|
class MemOperand BASE_EMBEDDED {
|
|
|
|
public:
|
|
|
|
// [rn +/- offset] Offset/NegOffset
|
|
|
|
// [rn +/- offset]! PreIndex/NegPreIndex
|
|
|
|
// [rn], +/- offset PostIndex/NegPostIndex
|
|
|
|
// offset is any signed 32-bit value; offset is first loaded to register ip if
|
|
|
|
// it does not fit the addressing mode (12-bit unsigned and sign bit)
|
|
|
|
explicit MemOperand(Register rn, int32_t offset = 0, AddrMode am = Offset);
|
|
|
|
|
|
|
|
// [rn +/- rm] Offset/NegOffset
|
|
|
|
// [rn +/- rm]! PreIndex/NegPreIndex
|
|
|
|
// [rn], +/- rm PostIndex/NegPostIndex
|
|
|
|
explicit MemOperand(Register rn, Register rm, AddrMode am = Offset);
|
|
|
|
|
|
|
|
// [rn +/- rm <shift_op> shift_imm] Offset/NegOffset
|
|
|
|
// [rn +/- rm <shift_op> shift_imm]! PreIndex/NegPreIndex
|
|
|
|
// [rn], +/- rm <shift_op> shift_imm PostIndex/NegPostIndex
|
|
|
|
explicit MemOperand(Register rn, Register rm,
|
|
|
|
ShiftOp shift_op, int shift_imm, AddrMode am = Offset);
|
|
|
|
|
2010-05-07 20:02:57 +00:00
|
|
|
void set_offset(int32_t offset) {
|
|
|
|
ASSERT(rm_.is(no_reg));
|
|
|
|
offset_ = offset;
|
|
|
|
}
|
|
|
|
|
2011-02-17 10:07:13 +00:00
|
|
|
uint32_t offset() const {
|
2010-05-07 20:02:57 +00:00
|
|
|
ASSERT(rm_.is(no_reg));
|
|
|
|
return offset_;
|
|
|
|
}
|
|
|
|
|
2010-05-18 06:38:42 +00:00
|
|
|
Register rn() const { return rn_; }
|
|
|
|
Register rm() const { return rm_; }
|
2011-06-22 19:18:04 +00:00
|
|
|
AddrMode am() const { return am_; }
|
2010-05-07 20:02:57 +00:00
|
|
|
|
2011-02-17 10:07:13 +00:00
|
|
|
bool OffsetIsUint12Encodable() const {
|
|
|
|
return offset_ >= 0 ? is_uint12(offset_) : is_uint12(-offset_);
|
|
|
|
}
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
private:
|
|
|
|
Register rn_; // base
|
|
|
|
Register rm_; // register offset
|
|
|
|
int32_t offset_; // valid if rm_ == no_reg
|
|
|
|
ShiftOp shift_op_;
|
|
|
|
int shift_imm_; // valid if rm_ != no_reg && rs_ == no_reg
|
|
|
|
AddrMode am_; // bits P, U, and W
|
|
|
|
|
|
|
|
friend class Assembler;
|
|
|
|
};
|
|
|
|
|
2009-09-14 06:57:24 +00:00
|
|
|
extern const Instr kMovLrPc;
|
2010-03-19 14:05:11 +00:00
|
|
|
extern const Instr kLdrPCMask;
|
2009-09-14 06:57:24 +00:00
|
|
|
extern const Instr kLdrPCPattern;
|
2010-03-19 14:05:11 +00:00
|
|
|
extern const Instr kBlxRegMask;
|
|
|
|
extern const Instr kBlxRegPattern;
|
2012-03-23 12:16:40 +00:00
|
|
|
extern const Instr kBlxIp;
|
2009-09-14 06:57:24 +00:00
|
|
|
|
2010-06-14 11:20:36 +00:00
|
|
|
extern const Instr kMovMvnMask;
|
|
|
|
extern const Instr kMovMvnPattern;
|
|
|
|
extern const Instr kMovMvnFlip;
|
|
|
|
|
2010-06-22 08:38:32 +00:00
|
|
|
extern const Instr kMovLeaveCCMask;
|
|
|
|
extern const Instr kMovLeaveCCPattern;
|
|
|
|
extern const Instr kMovwMask;
|
|
|
|
extern const Instr kMovwPattern;
|
|
|
|
extern const Instr kMovwLeaveCCFlip;
|
|
|
|
|
2010-06-14 11:20:36 +00:00
|
|
|
extern const Instr kCmpCmnMask;
|
|
|
|
extern const Instr kCmpCmnPattern;
|
|
|
|
extern const Instr kCmpCmnFlip;
|
|
|
|
extern const Instr kAddSubFlip;
|
|
|
|
extern const Instr kAndBicFlip;
|
2009-09-14 06:57:24 +00:00
|
|
|
|
2011-01-26 08:32:54 +00:00
|
|
|
|
|
|
|
|
2011-03-22 13:20:04 +00:00
|
|
|
class Assembler : public AssemblerBase {
|
2008-07-03 15:10:15 +00:00
|
|
|
public:
|
|
|
|
// Create an assembler. Instructions and relocation information are emitted
|
|
|
|
// into a buffer, with the instructions starting from the beginning and the
|
|
|
|
// relocation information starting from the end of the buffer. See CodeDesc
|
|
|
|
// for a detailed comment on the layout (globals.h).
|
|
|
|
//
|
|
|
|
// If the provided buffer is NULL, the assembler allocates and grows its own
|
|
|
|
// buffer, and buffer_size determines the initial buffer size. The buffer is
|
|
|
|
// owned by the assembler and deallocated upon destruction of the assembler.
|
|
|
|
//
|
|
|
|
// If the provided buffer is not NULL, the assembler uses the provided buffer
|
|
|
|
// for code generation and assumes its size to be buffer_size. If the buffer
|
|
|
|
// is too small, a fatal error occurs. No deallocation of the buffer is done
|
|
|
|
// upon destruction of the assembler.
|
2011-03-31 16:17:37 +00:00
|
|
|
Assembler(Isolate* isolate, void* buffer, int buffer_size);
|
2012-11-22 10:28:29 +00:00
|
|
|
virtual ~Assembler();
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
// GetCode emits any pending (non-emitted) code and fills the descriptor
|
|
|
|
// desc. GetCode() is idempotent; it returns the same result if no other
|
2009-01-15 19:08:34 +00:00
|
|
|
// Assembler functions are invoked in between GetCode() calls.
|
2008-07-03 15:10:15 +00:00
|
|
|
void GetCode(CodeDesc* desc);
|
|
|
|
|
|
|
|
// Label operations & relative jumps (PPUM Appendix D)
|
|
|
|
//
|
|
|
|
// Takes a branch opcode (cc) and a label (L) and generates
|
|
|
|
// either a backward branch or a forward branch and links it
|
|
|
|
// to the label fixup chain. Usage:
|
|
|
|
//
|
|
|
|
// Label L; // unbound label
|
|
|
|
// j(cc, &L); // forward branch to unbound label
|
|
|
|
// bind(&L); // bind label to the current pc
|
|
|
|
// j(cc, &L); // backward branch to bound label
|
|
|
|
// bind(&L); // illegal: a label may be bound only once
|
|
|
|
//
|
|
|
|
// Note: The same Label can be used for forward and backward branches
|
|
|
|
// but it may be bound only once.
|
|
|
|
|
|
|
|
void bind(Label* L); // binds an unbound label L to the current code position
|
|
|
|
|
|
|
|
// Returns the branch offset to the given label from the current code position
|
|
|
|
// Links the label to the current position if it is still unbound
|
2008-08-06 10:02:49 +00:00
|
|
|
// Manages the jump elimination optimization if the second parameter is true.
|
|
|
|
int branch_offset(Label* L, bool jump_elimination_allowed);
|
2008-07-03 15:10:15 +00:00
|
|
|
|
2009-08-31 12:40:37 +00:00
|
|
|
// Puts a labels target address at the given position.
|
|
|
|
// The high 8 bits are set to zero.
|
|
|
|
void label_at_put(Label* L, int at_offset);
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
// Return the address in the constant pool of the code target address used by
|
2012-10-18 12:21:42 +00:00
|
|
|
// the branch/call instruction at pc, or the object in a mov.
|
|
|
|
INLINE(static Address target_pointer_address_at(Address pc));
|
|
|
|
|
|
|
|
// Read/Modify the pointer in the branch/call/move instruction at pc.
|
|
|
|
INLINE(static Address target_pointer_at(Address pc));
|
|
|
|
INLINE(static void set_target_pointer_at(Address pc, Address target));
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
// Read/Modify the code target address in the branch/call instruction at pc.
|
|
|
|
INLINE(static Address target_address_at(Address pc));
|
|
|
|
INLINE(static void set_target_address_at(Address pc, Address target));
|
|
|
|
|
2012-10-18 12:21:42 +00:00
|
|
|
// Return the code target address at a call site from the return address
|
|
|
|
// of that call in the instruction stream.
|
|
|
|
INLINE(static Address target_address_from_return_address(Address pc));
|
|
|
|
|
|
|
|
// Given the address of the beginning of a call, return the address
|
|
|
|
// in the instruction stream that the call will return from.
|
|
|
|
INLINE(static Address return_address_from_call_start(Address pc));
|
|
|
|
|
2009-10-30 10:23:12 +00:00
|
|
|
// This sets the branch destination (which is in the constant pool on ARM).
|
|
|
|
// This is for calls and branches within generated code.
|
2012-03-21 14:29:14 +00:00
|
|
|
inline static void deserialization_set_special_target_at(
|
|
|
|
Address constant_pool_entry, Address target);
|
2009-10-30 10:23:12 +00:00
|
|
|
|
|
|
|
// This sets the branch destination (which is in the constant pool on ARM).
|
|
|
|
// This is for calls and branches to runtime code.
|
|
|
|
inline static void set_external_target_at(Address constant_pool_entry,
|
2012-03-21 14:29:14 +00:00
|
|
|
Address target);
|
2009-10-28 12:37:54 +00:00
|
|
|
|
2009-10-27 11:54:01 +00:00
|
|
|
// Here we are patching the address in the constant pool, not the actual call
|
|
|
|
// instruction. The address in the constant pool is the same size as a
|
|
|
|
// pointer.
|
2012-03-21 14:29:14 +00:00
|
|
|
static const int kSpecialTargetSize = kPointerSize;
|
2009-10-27 11:54:01 +00:00
|
|
|
|
2009-09-14 06:57:24 +00:00
|
|
|
// Size of an instruction.
|
|
|
|
static const int kInstrSize = sizeof(Instr);
|
|
|
|
|
2009-08-19 10:18:30 +00:00
|
|
|
// Distance between start of patched return sequence and the emitted address
|
|
|
|
// to jump to.
|
2010-03-19 14:05:11 +00:00
|
|
|
#ifdef USE_BLX
|
2010-06-08 12:04:49 +00:00
|
|
|
// Patched return sequence is:
|
2010-03-19 14:05:11 +00:00
|
|
|
// ldr ip, [pc, #0] @ emited address and start
|
|
|
|
// blx ip
|
|
|
|
static const int kPatchReturnSequenceAddressOffset = 0 * kInstrSize;
|
|
|
|
#else
|
2010-06-08 12:04:49 +00:00
|
|
|
// Patched return sequence is:
|
2010-03-19 14:05:11 +00:00
|
|
|
// mov lr, pc @ start of sequence
|
|
|
|
// ldr pc, [pc, #-4] @ emited address
|
|
|
|
static const int kPatchReturnSequenceAddressOffset = kInstrSize;
|
|
|
|
#endif
|
2008-07-03 15:10:15 +00:00
|
|
|
|
2010-06-08 12:04:49 +00:00
|
|
|
// Distance between start of patched debug break slot and the emitted address
|
|
|
|
// to jump to.
|
|
|
|
#ifdef USE_BLX
|
|
|
|
// Patched debug break slot code is:
|
|
|
|
// ldr ip, [pc, #0] @ emited address and start
|
|
|
|
// blx ip
|
|
|
|
static const int kPatchDebugBreakSlotAddressOffset = 0 * kInstrSize;
|
|
|
|
#else
|
|
|
|
// Patched debug break slot code is:
|
|
|
|
// mov lr, pc @ start of sequence
|
|
|
|
// ldr pc, [pc, #-4] @ emited address
|
|
|
|
static const int kPatchDebugBreakSlotAddressOffset = kInstrSize;
|
|
|
|
#endif
|
|
|
|
|
2012-10-18 12:21:42 +00:00
|
|
|
#ifdef USE_BLX
|
|
|
|
static const int kPatchDebugBreakSlotReturnOffset = 2 * kInstrSize;
|
|
|
|
#else
|
|
|
|
static const int kPatchDebugBreakSlotReturnOffset = kInstrSize;
|
|
|
|
#endif
|
|
|
|
|
2009-08-31 12:40:37 +00:00
|
|
|
// Difference between address of current opcode and value read from pc
|
|
|
|
// register.
|
|
|
|
static const int kPcLoadDelta = 8;
|
|
|
|
|
2010-06-08 12:04:49 +00:00
|
|
|
static const int kJSReturnSequenceInstructions = 4;
|
|
|
|
static const int kDebugBreakSlotInstructions = 3;
|
|
|
|
static const int kDebugBreakSlotLength =
|
|
|
|
kDebugBreakSlotInstructions * kInstrSize;
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
// ---------------------------------------------------------------------------
|
|
|
|
// Code generation
|
|
|
|
|
|
|
|
// Insert the smallest number of nop instructions
|
|
|
|
// possible to align the pc offset to a multiple
|
|
|
|
// of m. m must be a power of 2 (>= 4).
|
|
|
|
void Align(int m);
|
2010-06-22 10:07:57 +00:00
|
|
|
// Aligns code to something that's optimal for a jump target for the platform.
|
|
|
|
void CodeTargetAlign();
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
// Branch instructions
|
|
|
|
void b(int branch_offset, Condition cond = al);
|
|
|
|
void bl(int branch_offset, Condition cond = al);
|
|
|
|
void blx(int branch_offset); // v5 and above
|
|
|
|
void blx(Register target, Condition cond = al); // v5 and above
|
|
|
|
void bx(Register target, Condition cond = al); // v5 and above, plus v4t
|
|
|
|
|
|
|
|
// Convenience branch instructions using labels
|
2008-08-06 10:02:49 +00:00
|
|
|
void b(Label* L, Condition cond = al) {
|
|
|
|
b(branch_offset(L, cond == al), cond);
|
|
|
|
}
|
|
|
|
void b(Condition cond, Label* L) { b(branch_offset(L, cond == al), cond); }
|
|
|
|
void bl(Label* L, Condition cond = al) { bl(branch_offset(L, false), cond); }
|
|
|
|
void bl(Condition cond, Label* L) { bl(branch_offset(L, false), cond); }
|
|
|
|
void blx(Label* L) { blx(branch_offset(L, false)); } // v5 and above
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
// Data-processing instructions
|
2010-02-05 08:46:41 +00:00
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
void and_(Register dst, Register src1, const Operand& src2,
|
|
|
|
SBit s = LeaveCC, Condition cond = al);
|
|
|
|
|
|
|
|
void eor(Register dst, Register src1, const Operand& src2,
|
|
|
|
SBit s = LeaveCC, Condition cond = al);
|
|
|
|
|
|
|
|
void sub(Register dst, Register src1, const Operand& src2,
|
|
|
|
SBit s = LeaveCC, Condition cond = al);
|
2008-09-12 03:29:06 +00:00
|
|
|
void sub(Register dst, Register src1, Register src2,
|
|
|
|
SBit s = LeaveCC, Condition cond = al) {
|
|
|
|
sub(dst, src1, Operand(src2), s, cond);
|
|
|
|
}
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
void rsb(Register dst, Register src1, const Operand& src2,
|
|
|
|
SBit s = LeaveCC, Condition cond = al);
|
|
|
|
|
|
|
|
void add(Register dst, Register src1, const Operand& src2,
|
|
|
|
SBit s = LeaveCC, Condition cond = al);
|
2010-06-01 21:11:38 +00:00
|
|
|
void add(Register dst, Register src1, Register src2,
|
|
|
|
SBit s = LeaveCC, Condition cond = al) {
|
|
|
|
add(dst, src1, Operand(src2), s, cond);
|
|
|
|
}
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
void adc(Register dst, Register src1, const Operand& src2,
|
|
|
|
SBit s = LeaveCC, Condition cond = al);
|
|
|
|
|
|
|
|
void sbc(Register dst, Register src1, const Operand& src2,
|
|
|
|
SBit s = LeaveCC, Condition cond = al);
|
|
|
|
|
|
|
|
void rsc(Register dst, Register src1, const Operand& src2,
|
|
|
|
SBit s = LeaveCC, Condition cond = al);
|
|
|
|
|
|
|
|
void tst(Register src1, const Operand& src2, Condition cond = al);
|
2008-09-12 03:29:06 +00:00
|
|
|
void tst(Register src1, Register src2, Condition cond = al) {
|
|
|
|
tst(src1, Operand(src2), cond);
|
|
|
|
}
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
void teq(Register src1, const Operand& src2, Condition cond = al);
|
|
|
|
|
|
|
|
void cmp(Register src1, const Operand& src2, Condition cond = al);
|
2008-09-12 03:29:06 +00:00
|
|
|
void cmp(Register src1, Register src2, Condition cond = al) {
|
|
|
|
cmp(src1, Operand(src2), cond);
|
|
|
|
}
|
2011-02-09 14:57:24 +00:00
|
|
|
void cmp_raw_immediate(Register src1, int raw_immediate, Condition cond = al);
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
void cmn(Register src1, const Operand& src2, Condition cond = al);
|
|
|
|
|
|
|
|
void orr(Register dst, Register src1, const Operand& src2,
|
|
|
|
SBit s = LeaveCC, Condition cond = al);
|
2008-09-12 03:29:06 +00:00
|
|
|
void orr(Register dst, Register src1, Register src2,
|
|
|
|
SBit s = LeaveCC, Condition cond = al) {
|
|
|
|
orr(dst, src1, Operand(src2), s, cond);
|
|
|
|
}
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
void mov(Register dst, const Operand& src,
|
|
|
|
SBit s = LeaveCC, Condition cond = al);
|
2008-09-12 03:29:06 +00:00
|
|
|
void mov(Register dst, Register src, SBit s = LeaveCC, Condition cond = al) {
|
|
|
|
mov(dst, Operand(src), s, cond);
|
|
|
|
}
|
2008-07-03 15:10:15 +00:00
|
|
|
|
2010-06-22 08:38:32 +00:00
|
|
|
// ARMv7 instructions for loading a 32 bit immediate in two instructions.
|
|
|
|
// This may actually emit a different mov instruction, but on an ARMv7 it
|
|
|
|
// is guaranteed to only emit one instruction.
|
|
|
|
void movw(Register reg, uint32_t immediate, Condition cond = al);
|
|
|
|
// The constant for movt should be in the range 0-0xffff.
|
|
|
|
void movt(Register reg, uint32_t immediate, Condition cond = al);
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
void bic(Register dst, Register src1, const Operand& src2,
|
|
|
|
SBit s = LeaveCC, Condition cond = al);
|
|
|
|
|
|
|
|
void mvn(Register dst, const Operand& src,
|
|
|
|
SBit s = LeaveCC, Condition cond = al);
|
|
|
|
|
|
|
|
// Multiply instructions
|
|
|
|
|
|
|
|
void mla(Register dst, Register src1, Register src2, Register srcA,
|
|
|
|
SBit s = LeaveCC, Condition cond = al);
|
|
|
|
|
2012-10-01 21:27:33 +00:00
|
|
|
void mls(Register dst, Register src1, Register src2, Register srcA,
|
|
|
|
Condition cond = al);
|
|
|
|
|
|
|
|
void sdiv(Register dst, Register src1, Register src2,
|
|
|
|
Condition cond = al);
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
void mul(Register dst, Register src1, Register src2,
|
|
|
|
SBit s = LeaveCC, Condition cond = al);
|
|
|
|
|
|
|
|
void smlal(Register dstL, Register dstH, Register src1, Register src2,
|
|
|
|
SBit s = LeaveCC, Condition cond = al);
|
|
|
|
|
|
|
|
void smull(Register dstL, Register dstH, Register src1, Register src2,
|
|
|
|
SBit s = LeaveCC, Condition cond = al);
|
|
|
|
|
|
|
|
void umlal(Register dstL, Register dstH, Register src1, Register src2,
|
|
|
|
SBit s = LeaveCC, Condition cond = al);
|
|
|
|
|
|
|
|
void umull(Register dstL, Register dstH, Register src1, Register src2,
|
|
|
|
SBit s = LeaveCC, Condition cond = al);
|
|
|
|
|
|
|
|
// Miscellaneous arithmetic instructions
|
|
|
|
|
|
|
|
void clz(Register dst, Register src, Condition cond = al); // v5 and above
|
|
|
|
|
2010-07-21 07:42:51 +00:00
|
|
|
// Saturating instructions. v6 and above.
|
|
|
|
|
|
|
|
// Unsigned saturate.
|
|
|
|
//
|
|
|
|
// Saturate an optionally shifted signed value to an unsigned range.
|
|
|
|
//
|
|
|
|
// usat dst, #satpos, src
|
|
|
|
// usat dst, #satpos, src, lsl #sh
|
|
|
|
// usat dst, #satpos, src, asr #sh
|
|
|
|
//
|
|
|
|
// Register dst will contain:
|
|
|
|
//
|
|
|
|
// 0, if s < 0
|
|
|
|
// (1 << satpos) - 1, if s > ((1 << satpos) - 1)
|
|
|
|
// s, otherwise
|
|
|
|
//
|
|
|
|
// where s is the contents of src after shifting (if used.)
|
|
|
|
void usat(Register dst, int satpos, const Operand& src, Condition cond = al);
|
|
|
|
|
2010-05-27 13:46:18 +00:00
|
|
|
// Bitfield manipulation instructions. v7 and above.
|
|
|
|
|
|
|
|
void ubfx(Register dst, Register src, int lsb, int width,
|
|
|
|
Condition cond = al);
|
|
|
|
|
|
|
|
void sbfx(Register dst, Register src, int lsb, int width,
|
|
|
|
Condition cond = al);
|
|
|
|
|
|
|
|
void bfc(Register dst, int lsb, int width, Condition cond = al);
|
|
|
|
|
|
|
|
void bfi(Register dst, Register src, int lsb, int width,
|
|
|
|
Condition cond = al);
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
// Status register access instructions
|
|
|
|
|
|
|
|
void mrs(Register dst, SRegister s, Condition cond = al);
|
|
|
|
void msr(SRegisterFieldMask fields, const Operand& src, Condition cond = al);
|
|
|
|
|
|
|
|
// Load/Store instructions
|
|
|
|
void ldr(Register dst, const MemOperand& src, Condition cond = al);
|
|
|
|
void str(Register src, const MemOperand& dst, Condition cond = al);
|
|
|
|
void ldrb(Register dst, const MemOperand& src, Condition cond = al);
|
|
|
|
void strb(Register src, const MemOperand& dst, Condition cond = al);
|
|
|
|
void ldrh(Register dst, const MemOperand& src, Condition cond = al);
|
|
|
|
void strh(Register src, const MemOperand& dst, Condition cond = al);
|
|
|
|
void ldrsb(Register dst, const MemOperand& src, Condition cond = al);
|
|
|
|
void ldrsh(Register dst, const MemOperand& src, Condition cond = al);
|
2010-05-26 09:43:54 +00:00
|
|
|
void ldrd(Register dst1,
|
|
|
|
Register dst2,
|
|
|
|
const MemOperand& src, Condition cond = al);
|
|
|
|
void strd(Register src1,
|
|
|
|
Register src2,
|
|
|
|
const MemOperand& dst, Condition cond = al);
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
// Load/Store multiple instructions
|
|
|
|
void ldm(BlockAddrMode am, Register base, RegList dst, Condition cond = al);
|
|
|
|
void stm(BlockAddrMode am, Register base, RegList src, Condition cond = al);
|
|
|
|
|
|
|
|
// Exception-generating instructions and debugging support
|
2010-10-28 07:35:07 +00:00
|
|
|
void stop(const char* msg,
|
|
|
|
Condition cond = al,
|
|
|
|
int32_t code = kDefaultStopCode);
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
void bkpt(uint32_t imm16); // v5 and above
|
2010-10-28 07:35:07 +00:00
|
|
|
void svc(uint32_t imm24, Condition cond = al);
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
// Coprocessor instructions
|
|
|
|
|
|
|
|
void cdp(Coprocessor coproc, int opcode_1,
|
|
|
|
CRegister crd, CRegister crn, CRegister crm,
|
|
|
|
int opcode_2, Condition cond = al);
|
|
|
|
|
|
|
|
void cdp2(Coprocessor coproc, int opcode_1,
|
|
|
|
CRegister crd, CRegister crn, CRegister crm,
|
|
|
|
int opcode_2); // v5 and above
|
|
|
|
|
|
|
|
void mcr(Coprocessor coproc, int opcode_1,
|
|
|
|
Register rd, CRegister crn, CRegister crm,
|
|
|
|
int opcode_2 = 0, Condition cond = al);
|
|
|
|
|
|
|
|
void mcr2(Coprocessor coproc, int opcode_1,
|
|
|
|
Register rd, CRegister crn, CRegister crm,
|
|
|
|
int opcode_2 = 0); // v5 and above
|
|
|
|
|
|
|
|
void mrc(Coprocessor coproc, int opcode_1,
|
|
|
|
Register rd, CRegister crn, CRegister crm,
|
|
|
|
int opcode_2 = 0, Condition cond = al);
|
|
|
|
|
|
|
|
void mrc2(Coprocessor coproc, int opcode_1,
|
|
|
|
Register rd, CRegister crn, CRegister crm,
|
|
|
|
int opcode_2 = 0); // v5 and above
|
|
|
|
|
|
|
|
void ldc(Coprocessor coproc, CRegister crd, const MemOperand& src,
|
|
|
|
LFlag l = Short, Condition cond = al);
|
|
|
|
void ldc(Coprocessor coproc, CRegister crd, Register base, int option,
|
|
|
|
LFlag l = Short, Condition cond = al);
|
|
|
|
|
|
|
|
void ldc2(Coprocessor coproc, CRegister crd, const MemOperand& src,
|
|
|
|
LFlag l = Short); // v5 and above
|
|
|
|
void ldc2(Coprocessor coproc, CRegister crd, Register base, int option,
|
|
|
|
LFlag l = Short); // v5 and above
|
|
|
|
|
2009-11-12 13:04:02 +00:00
|
|
|
// Support for VFP.
|
2013-01-23 16:29:48 +00:00
|
|
|
// All these APIs support S0 to S31 and D0 to D31.
|
2009-11-12 13:04:02 +00:00
|
|
|
|
2010-01-25 11:54:10 +00:00
|
|
|
void vldr(const DwVfpRegister dst,
|
|
|
|
const Register base,
|
2011-02-17 10:07:13 +00:00
|
|
|
int offset,
|
|
|
|
const Condition cond = al);
|
|
|
|
void vldr(const DwVfpRegister dst,
|
|
|
|
const MemOperand& src,
|
2010-01-25 11:54:10 +00:00
|
|
|
const Condition cond = al);
|
2010-03-23 13:38:04 +00:00
|
|
|
|
|
|
|
void vldr(const SwVfpRegister dst,
|
|
|
|
const Register base,
|
2011-02-17 10:07:13 +00:00
|
|
|
int offset,
|
|
|
|
const Condition cond = al);
|
|
|
|
void vldr(const SwVfpRegister dst,
|
|
|
|
const MemOperand& src,
|
2010-03-23 13:38:04 +00:00
|
|
|
const Condition cond = al);
|
|
|
|
|
2010-01-25 11:54:10 +00:00
|
|
|
void vstr(const DwVfpRegister src,
|
|
|
|
const Register base,
|
2011-02-17 10:07:13 +00:00
|
|
|
int offset,
|
|
|
|
const Condition cond = al);
|
|
|
|
void vstr(const DwVfpRegister src,
|
|
|
|
const MemOperand& dst,
|
2010-01-25 11:54:10 +00:00
|
|
|
const Condition cond = al);
|
2010-06-30 12:22:15 +00:00
|
|
|
|
2010-08-17 08:43:45 +00:00
|
|
|
void vstr(const SwVfpRegister src,
|
|
|
|
const Register base,
|
2011-02-17 10:07:13 +00:00
|
|
|
int offset,
|
|
|
|
const Condition cond = al);
|
|
|
|
void vstr(const SwVfpRegister src,
|
|
|
|
const MemOperand& dst,
|
2010-08-17 08:43:45 +00:00
|
|
|
const Condition cond = al);
|
|
|
|
|
2011-04-06 09:06:23 +00:00
|
|
|
void vldm(BlockAddrMode am,
|
|
|
|
Register base,
|
|
|
|
DwVfpRegister first,
|
|
|
|
DwVfpRegister last,
|
|
|
|
Condition cond = al);
|
|
|
|
|
|
|
|
void vstm(BlockAddrMode am,
|
|
|
|
Register base,
|
|
|
|
DwVfpRegister first,
|
|
|
|
DwVfpRegister last,
|
|
|
|
Condition cond = al);
|
|
|
|
|
|
|
|
void vldm(BlockAddrMode am,
|
|
|
|
Register base,
|
|
|
|
SwVfpRegister first,
|
|
|
|
SwVfpRegister last,
|
|
|
|
Condition cond = al);
|
|
|
|
|
|
|
|
void vstm(BlockAddrMode am,
|
|
|
|
Register base,
|
|
|
|
SwVfpRegister first,
|
|
|
|
SwVfpRegister last,
|
|
|
|
Condition cond = al);
|
|
|
|
|
2010-07-08 12:38:02 +00:00
|
|
|
void vmov(const DwVfpRegister dst,
|
|
|
|
double imm,
|
2012-09-25 14:32:07 +00:00
|
|
|
const Register scratch = no_reg,
|
2010-07-08 12:38:02 +00:00
|
|
|
const Condition cond = al);
|
|
|
|
void vmov(const SwVfpRegister dst,
|
|
|
|
const SwVfpRegister src,
|
|
|
|
const Condition cond = al);
|
2010-06-30 12:22:15 +00:00
|
|
|
void vmov(const DwVfpRegister dst,
|
|
|
|
const DwVfpRegister src,
|
|
|
|
const Condition cond = al);
|
2013-01-23 16:29:48 +00:00
|
|
|
void vmov(const DwVfpRegister dst,
|
|
|
|
int index,
|
|
|
|
const Register src,
|
|
|
|
const Condition cond = al);
|
2009-12-09 11:14:45 +00:00
|
|
|
void vmov(const DwVfpRegister dst,
|
|
|
|
const Register src1,
|
|
|
|
const Register src2,
|
|
|
|
const Condition cond = al);
|
|
|
|
void vmov(const Register dst1,
|
|
|
|
const Register dst2,
|
|
|
|
const DwVfpRegister src,
|
2009-11-12 13:04:02 +00:00
|
|
|
const Condition cond = al);
|
2009-12-09 11:14:45 +00:00
|
|
|
void vmov(const SwVfpRegister dst,
|
2009-11-12 13:04:02 +00:00
|
|
|
const Register src,
|
|
|
|
const Condition cond = al);
|
2009-12-09 11:14:45 +00:00
|
|
|
void vmov(const Register dst,
|
|
|
|
const SwVfpRegister src,
|
|
|
|
const Condition cond = al);
|
2010-03-23 13:38:04 +00:00
|
|
|
void vcvt_f64_s32(const DwVfpRegister dst,
|
|
|
|
const SwVfpRegister src,
|
2011-02-04 07:08:50 +00:00
|
|
|
VFPConversionMode mode = kDefaultRoundToZero,
|
2010-03-23 13:38:04 +00:00
|
|
|
const Condition cond = al);
|
|
|
|
void vcvt_f32_s32(const SwVfpRegister dst,
|
|
|
|
const SwVfpRegister src,
|
2011-02-04 07:08:50 +00:00
|
|
|
VFPConversionMode mode = kDefaultRoundToZero,
|
2010-03-23 13:38:04 +00:00
|
|
|
const Condition cond = al);
|
|
|
|
void vcvt_f64_u32(const DwVfpRegister dst,
|
|
|
|
const SwVfpRegister src,
|
2011-02-04 07:08:50 +00:00
|
|
|
VFPConversionMode mode = kDefaultRoundToZero,
|
2010-03-23 13:38:04 +00:00
|
|
|
const Condition cond = al);
|
|
|
|
void vcvt_s32_f64(const SwVfpRegister dst,
|
|
|
|
const DwVfpRegister src,
|
2011-02-04 07:08:50 +00:00
|
|
|
VFPConversionMode mode = kDefaultRoundToZero,
|
2010-03-23 13:38:04 +00:00
|
|
|
const Condition cond = al);
|
|
|
|
void vcvt_u32_f64(const SwVfpRegister dst,
|
|
|
|
const DwVfpRegister src,
|
2011-02-04 07:08:50 +00:00
|
|
|
VFPConversionMode mode = kDefaultRoundToZero,
|
2010-03-23 13:38:04 +00:00
|
|
|
const Condition cond = al);
|
|
|
|
void vcvt_f64_f32(const DwVfpRegister dst,
|
|
|
|
const SwVfpRegister src,
|
2011-02-04 07:08:50 +00:00
|
|
|
VFPConversionMode mode = kDefaultRoundToZero,
|
2010-03-23 13:38:04 +00:00
|
|
|
const Condition cond = al);
|
|
|
|
void vcvt_f32_f64(const SwVfpRegister dst,
|
|
|
|
const DwVfpRegister src,
|
2011-02-04 07:08:50 +00:00
|
|
|
VFPConversionMode mode = kDefaultRoundToZero,
|
2010-03-23 13:38:04 +00:00
|
|
|
const Condition cond = al);
|
2009-12-09 11:14:45 +00:00
|
|
|
|
2011-03-15 11:19:13 +00:00
|
|
|
void vneg(const DwVfpRegister dst,
|
|
|
|
const DwVfpRegister src,
|
|
|
|
const Condition cond = al);
|
2011-01-31 10:16:28 +00:00
|
|
|
void vabs(const DwVfpRegister dst,
|
|
|
|
const DwVfpRegister src,
|
|
|
|
const Condition cond = al);
|
2009-12-09 11:14:45 +00:00
|
|
|
void vadd(const DwVfpRegister dst,
|
|
|
|
const DwVfpRegister src1,
|
|
|
|
const DwVfpRegister src2,
|
|
|
|
const Condition cond = al);
|
|
|
|
void vsub(const DwVfpRegister dst,
|
|
|
|
const DwVfpRegister src1,
|
|
|
|
const DwVfpRegister src2,
|
|
|
|
const Condition cond = al);
|
|
|
|
void vmul(const DwVfpRegister dst,
|
|
|
|
const DwVfpRegister src1,
|
|
|
|
const DwVfpRegister src2,
|
|
|
|
const Condition cond = al);
|
2012-11-14 11:01:18 +00:00
|
|
|
void vmla(const DwVfpRegister dst,
|
|
|
|
const DwVfpRegister src1,
|
|
|
|
const DwVfpRegister src2,
|
|
|
|
const Condition cond = al);
|
2009-12-09 11:14:45 +00:00
|
|
|
void vdiv(const DwVfpRegister dst,
|
|
|
|
const DwVfpRegister src1,
|
|
|
|
const DwVfpRegister src2,
|
|
|
|
const Condition cond = al);
|
|
|
|
void vcmp(const DwVfpRegister src1,
|
|
|
|
const DwVfpRegister src2,
|
2009-11-12 13:04:02 +00:00
|
|
|
const Condition cond = al);
|
2010-08-16 07:52:49 +00:00
|
|
|
void vcmp(const DwVfpRegister src1,
|
|
|
|
const double src2,
|
|
|
|
const Condition cond = al);
|
2009-11-12 13:04:02 +00:00
|
|
|
void vmrs(const Register dst,
|
|
|
|
const Condition cond = al);
|
2010-11-09 08:26:02 +00:00
|
|
|
void vmsr(const Register dst,
|
|
|
|
const Condition cond = al);
|
2010-06-29 09:40:36 +00:00
|
|
|
void vsqrt(const DwVfpRegister dst,
|
|
|
|
const DwVfpRegister src,
|
|
|
|
const Condition cond = al);
|
2009-11-12 13:04:02 +00:00
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
// Pseudo instructions
|
2010-11-24 09:40:58 +00:00
|
|
|
|
|
|
|
// Different nop operations are used by the code generator to detect certain
|
|
|
|
// states of the generated code.
|
|
|
|
enum NopMarkerTypes {
|
|
|
|
NON_MARKING_NOP = 0,
|
|
|
|
DEBUG_BREAK_NOP,
|
|
|
|
// IC markers.
|
|
|
|
PROPERTY_ACCESS_INLINED,
|
|
|
|
PROPERTY_ACCESS_INLINED_CONTEXT,
|
|
|
|
PROPERTY_ACCESS_INLINED_CONTEXT_DONT_DELETE,
|
|
|
|
// Helper values.
|
|
|
|
LAST_CODE_MARKER,
|
|
|
|
FIRST_IC_MARKER = PROPERTY_ACCESS_INLINED
|
|
|
|
};
|
|
|
|
|
|
|
|
void nop(int type = 0); // 0 is the default non-marking type.
|
2008-07-03 15:10:15 +00:00
|
|
|
|
2009-05-12 20:11:01 +00:00
|
|
|
void push(Register src, Condition cond = al) {
|
|
|
|
str(src, MemOperand(sp, 4, NegPreIndex), cond);
|
2008-08-13 09:32:07 +00:00
|
|
|
}
|
|
|
|
|
2009-10-02 13:36:20 +00:00
|
|
|
void pop(Register dst, Condition cond = al) {
|
|
|
|
ldr(dst, MemOperand(sp, 4, PostIndex), cond);
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
2008-08-13 09:32:07 +00:00
|
|
|
void pop() {
|
|
|
|
add(sp, sp, Operand(kPointerSize));
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Jump unconditionally to given label.
|
|
|
|
void jmp(Label* L) { b(L, al); }
|
|
|
|
|
2012-10-18 14:37:04 +00:00
|
|
|
static bool use_immediate_embedded_pointer_loads(
|
2012-10-18 12:21:42 +00:00
|
|
|
const Assembler* assembler) {
|
2012-10-18 14:37:04 +00:00
|
|
|
#ifdef USE_BLX
|
2012-10-18 12:21:42 +00:00
|
|
|
return CpuFeatures::IsSupported(MOVW_MOVT_IMMEDIATE_LOADS) &&
|
|
|
|
(assembler == NULL || !assembler->predictable_code_size());
|
2012-10-18 14:37:04 +00:00
|
|
|
#else
|
|
|
|
// If not using BLX, all loads from the constant pool cannot be immediate,
|
|
|
|
// because the ldr pc, [pc + #xxxx] used for calls must be a single
|
|
|
|
// instruction and cannot be easily distinguished out of context from
|
|
|
|
// other loads that could use movw/movt.
|
|
|
|
return false;
|
|
|
|
#endif
|
2012-10-18 12:21:42 +00:00
|
|
|
}
|
|
|
|
|
2009-09-14 06:57:24 +00:00
|
|
|
// Check the code size generated from label to here.
|
2011-06-30 11:26:15 +00:00
|
|
|
int SizeOfCodeGeneratedSince(Label* label) {
|
|
|
|
return pc_offset() - label->pos();
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check the number of instructions generated from label to here.
|
|
|
|
int InstructionsGeneratedSince(Label* label) {
|
|
|
|
return SizeOfCodeGeneratedSince(label) / kInstrSize;
|
2009-09-14 06:57:24 +00:00
|
|
|
}
|
2008-07-03 15:10:15 +00:00
|
|
|
|
2009-11-04 10:04:22 +00:00
|
|
|
// Check whether an immediate fits an addressing mode 1 instruction.
|
|
|
|
bool ImmediateFitsAddrMode1Instruction(int32_t imm32);
|
|
|
|
|
2010-04-21 09:43:45 +00:00
|
|
|
// Class for scoping postponing the constant pool generation.
|
|
|
|
class BlockConstPoolScope {
|
|
|
|
public:
|
|
|
|
explicit BlockConstPoolScope(Assembler* assem) : assem_(assem) {
|
2010-04-23 07:42:45 +00:00
|
|
|
assem_->StartBlockConstPool();
|
2010-04-21 09:43:45 +00:00
|
|
|
}
|
|
|
|
~BlockConstPoolScope() {
|
2010-04-23 07:42:45 +00:00
|
|
|
assem_->EndBlockConstPool();
|
2010-04-21 09:43:45 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
Assembler* assem_;
|
2010-04-21 10:20:55 +00:00
|
|
|
|
|
|
|
DISALLOW_IMPLICIT_CONSTRUCTORS(BlockConstPoolScope);
|
2010-04-21 09:43:45 +00:00
|
|
|
};
|
2009-11-04 14:45:50 +00:00
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
// Debugging
|
|
|
|
|
2009-09-14 06:57:24 +00:00
|
|
|
// Mark address of the ExitJSFrame code.
|
|
|
|
void RecordJSReturn();
|
|
|
|
|
2010-06-08 12:04:49 +00:00
|
|
|
// Mark address of a debug break slot.
|
|
|
|
void RecordDebugBreakSlot();
|
|
|
|
|
2011-04-27 15:02:59 +00:00
|
|
|
// Record the AST id of the CallIC being compiled, so that it can be placed
|
|
|
|
// in the relocation information.
|
2012-08-06 14:13:09 +00:00
|
|
|
void SetRecordedAstId(TypeFeedbackId ast_id) {
|
|
|
|
ASSERT(recorded_ast_id_.IsNone());
|
2011-07-18 10:44:13 +00:00
|
|
|
recorded_ast_id_ = ast_id;
|
|
|
|
}
|
|
|
|
|
2012-08-06 14:13:09 +00:00
|
|
|
TypeFeedbackId RecordedAstId() {
|
|
|
|
ASSERT(!recorded_ast_id_.IsNone());
|
2011-07-18 10:44:13 +00:00
|
|
|
return recorded_ast_id_;
|
|
|
|
}
|
|
|
|
|
2012-08-06 14:13:09 +00:00
|
|
|
void ClearRecordedAstId() { recorded_ast_id_ = TypeFeedbackId::None(); }
|
2011-04-27 15:02:59 +00:00
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
// Record a comment relocation entry that can be used by a disassembler.
|
2010-12-07 11:31:57 +00:00
|
|
|
// Use --code-comments to enable.
|
2008-07-03 15:10:15 +00:00
|
|
|
void RecordComment(const char* msg);
|
|
|
|
|
2012-06-14 11:16:47 +00:00
|
|
|
// Record the emission of a constant pool.
|
|
|
|
//
|
|
|
|
// The emission of constant pool depends on the size of the code generated and
|
|
|
|
// the number of RelocInfo recorded.
|
|
|
|
// The Debug mechanism needs to map code offsets between two versions of a
|
|
|
|
// function, compiled with and without debugger support (see for example
|
|
|
|
// Debug::PrepareForBreakPoints()).
|
|
|
|
// Compiling functions with debugger support generates additional code
|
|
|
|
// (Debug::GenerateSlot()). This may affect the emission of the constant
|
|
|
|
// pools and cause the version of the code with debugger support to have
|
|
|
|
// constant pools generated in different places.
|
|
|
|
// Recording the position and size of emitted constant pools allows to
|
|
|
|
// correctly compute the offset mappings between the different versions of a
|
|
|
|
// function in all situations.
|
|
|
|
//
|
|
|
|
// The parameter indicates the size of the constant pool (in bytes), including
|
|
|
|
// the marker and branch over the data.
|
|
|
|
void RecordConstPool(int size);
|
|
|
|
|
2011-01-17 12:45:39 +00:00
|
|
|
// Writes a single byte or word of data in the code stream. Used
|
|
|
|
// for inline tables, e.g., jump-tables. The constant pool should be
|
|
|
|
// emitted before any use of db and dd to ensure that constant pools
|
|
|
|
// are not emitted as part of the tables generated.
|
2010-12-07 11:31:57 +00:00
|
|
|
void db(uint8_t data);
|
|
|
|
void dd(uint32_t data);
|
|
|
|
|
2010-11-04 15:12:03 +00:00
|
|
|
PositionsRecorder* positions_recorder() { return &positions_recorder_; }
|
2008-09-18 10:40:25 +00:00
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
// Read/patch instructions
|
2011-09-19 18:36:47 +00:00
|
|
|
Instr instr_at(int pos) { return *reinterpret_cast<Instr*>(buffer_ + pos); }
|
|
|
|
void instr_at_put(int pos, Instr instr) {
|
|
|
|
*reinterpret_cast<Instr*>(buffer_ + pos) = instr;
|
|
|
|
}
|
2009-09-14 06:57:24 +00:00
|
|
|
static Instr instr_at(byte* pc) { return *reinterpret_cast<Instr*>(pc); }
|
2010-04-22 07:18:30 +00:00
|
|
|
static void instr_at_put(byte* pc, Instr instr) {
|
2008-07-03 15:10:15 +00:00
|
|
|
*reinterpret_cast<Instr*>(pc) = instr;
|
|
|
|
}
|
2011-02-09 14:57:24 +00:00
|
|
|
static Condition GetCondition(Instr instr);
|
2010-04-23 07:42:45 +00:00
|
|
|
static bool IsBranch(Instr instr);
|
|
|
|
static int GetBranchOffset(Instr instr);
|
2010-04-22 07:18:30 +00:00
|
|
|
static bool IsLdrRegisterImmediate(Instr instr);
|
2012-12-28 13:34:15 +00:00
|
|
|
static bool IsVldrDRegisterImmediate(Instr instr);
|
2010-04-22 07:18:30 +00:00
|
|
|
static int GetLdrRegisterImmediateOffset(Instr instr);
|
2012-12-28 13:34:15 +00:00
|
|
|
static int GetVldrDRegisterImmediateOffset(Instr instr);
|
2010-04-22 07:18:30 +00:00
|
|
|
static Instr SetLdrRegisterImmediateOffset(Instr instr, int offset);
|
2012-12-28 13:34:15 +00:00
|
|
|
static Instr SetVldrDRegisterImmediateOffset(Instr instr, int offset);
|
2010-07-22 08:17:40 +00:00
|
|
|
static bool IsStrRegisterImmediate(Instr instr);
|
|
|
|
static Instr SetStrRegisterImmediateOffset(Instr instr, int offset);
|
|
|
|
static bool IsAddRegisterImmediate(Instr instr);
|
|
|
|
static Instr SetAddRegisterImmediateOffset(Instr instr, int offset);
|
2010-05-17 10:51:41 +00:00
|
|
|
static Register GetRd(Instr instr);
|
2011-02-09 14:57:24 +00:00
|
|
|
static Register GetRn(Instr instr);
|
|
|
|
static Register GetRm(Instr instr);
|
2010-05-17 10:51:41 +00:00
|
|
|
static bool IsPush(Instr instr);
|
|
|
|
static bool IsPop(Instr instr);
|
|
|
|
static bool IsStrRegFpOffset(Instr instr);
|
|
|
|
static bool IsLdrRegFpOffset(Instr instr);
|
|
|
|
static bool IsStrRegFpNegOffset(Instr instr);
|
|
|
|
static bool IsLdrRegFpNegOffset(Instr instr);
|
2010-11-24 09:40:58 +00:00
|
|
|
static bool IsLdrPcImmediateOffset(Instr instr);
|
2012-12-28 13:34:15 +00:00
|
|
|
static bool IsVldrDPcImmediateOffset(Instr instr);
|
2011-02-09 14:57:24 +00:00
|
|
|
static bool IsTstImmediate(Instr instr);
|
|
|
|
static bool IsCmpRegister(Instr instr);
|
|
|
|
static bool IsCmpImmediate(Instr instr);
|
|
|
|
static Register GetCmpImmediateRegister(Instr instr);
|
|
|
|
static int GetCmpImmediateRawImmediate(Instr instr);
|
2010-11-24 09:40:58 +00:00
|
|
|
static bool IsNop(Instr instr, int type = NON_MARKING_NOP);
|
2012-10-18 12:21:42 +00:00
|
|
|
static bool IsMovT(Instr instr);
|
|
|
|
static bool IsMovW(Instr instr);
|
2010-04-22 07:18:30 +00:00
|
|
|
|
2011-05-23 12:48:17 +00:00
|
|
|
// Constants in pools are accessed via pc relative addressing, which can
|
2012-12-28 13:34:15 +00:00
|
|
|
// reach +/-4KB for integer PC-relative loads and +/-1KB for floating-point
|
|
|
|
// PC-relative loads, thereby defining a maximum distance between the
|
|
|
|
// instruction and the accessed constant.
|
|
|
|
static const int kMaxDistToIntPool = 4*KB;
|
|
|
|
static const int kMaxDistToFPPool = 1*KB;
|
|
|
|
// All relocations could be integer, it therefore acts as the limit.
|
|
|
|
static const int kMaxNumPendingRelocInfo = kMaxDistToIntPool/kInstrSize;
|
2011-05-23 12:48:17 +00:00
|
|
|
|
2011-06-16 07:00:46 +00:00
|
|
|
// Postpone the generation of the constant pool for the specified number of
|
|
|
|
// instructions.
|
|
|
|
void BlockConstPoolFor(int instructions);
|
|
|
|
|
|
|
|
// Check if is time to emit a constant pool.
|
2010-12-07 11:31:57 +00:00
|
|
|
void CheckConstPool(bool force_emit, bool require_jump);
|
2010-04-23 07:42:45 +00:00
|
|
|
|
2010-04-22 07:18:30 +00:00
|
|
|
protected:
|
2011-04-27 15:02:59 +00:00
|
|
|
// Relocation for a type-recording IC has the AST id added to it. This
|
|
|
|
// member variable is a way to pass the information from the call site to
|
|
|
|
// the relocation info.
|
2012-08-06 14:13:09 +00:00
|
|
|
TypeFeedbackId recorded_ast_id_;
|
2011-04-27 15:02:59 +00:00
|
|
|
|
2010-04-22 07:18:30 +00:00
|
|
|
int buffer_space() const { return reloc_info_writer.pos() - pc_; }
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
// Decode branch instruction at pos and return branch target pos
|
|
|
|
int target_at(int pos);
|
|
|
|
|
|
|
|
// Patch branch instruction at pos to branch to given branch target pos
|
|
|
|
void target_at_put(int pos, int target_pos);
|
|
|
|
|
2011-06-16 07:00:46 +00:00
|
|
|
// Prevent contant pool emission until EndBlockConstPool is called.
|
|
|
|
// Call to this function can be nested but must be followed by an equal
|
|
|
|
// number of call to EndBlockConstpool.
|
2010-04-27 09:09:51 +00:00
|
|
|
void StartBlockConstPool() {
|
2011-06-16 07:00:46 +00:00
|
|
|
if (const_pool_blocked_nesting_++ == 0) {
|
|
|
|
// Prevent constant pool checks happening by setting the next check to
|
|
|
|
// the biggest possible offset.
|
|
|
|
next_buffer_check_ = kMaxInt;
|
|
|
|
}
|
2010-04-27 09:09:51 +00:00
|
|
|
}
|
2011-06-16 07:00:46 +00:00
|
|
|
|
|
|
|
// Resume constant pool emission. Need to be called as many time as
|
|
|
|
// StartBlockConstPool to have an effect.
|
2010-04-27 09:09:51 +00:00
|
|
|
void EndBlockConstPool() {
|
2011-06-16 07:00:46 +00:00
|
|
|
if (--const_pool_blocked_nesting_ == 0) {
|
|
|
|
// Check the constant pool hasn't been blocked for too long.
|
|
|
|
ASSERT((num_pending_reloc_info_ == 0) ||
|
2012-12-28 13:34:15 +00:00
|
|
|
(pc_offset() < (first_const_pool_use_ + kMaxDistToIntPool)));
|
|
|
|
ASSERT((num_pending_64_bit_reloc_info_ == 0) ||
|
|
|
|
(pc_offset() < (first_const_pool_use_ + kMaxDistToFPPool)));
|
2011-06-16 07:00:46 +00:00
|
|
|
// Two cases:
|
|
|
|
// * no_const_pool_before_ >= next_buffer_check_ and the emission is
|
|
|
|
// still blocked
|
|
|
|
// * no_const_pool_before_ < next_buffer_check_ and the next emit will
|
|
|
|
// trigger a check.
|
|
|
|
next_buffer_check_ = no_const_pool_before_;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
bool is_const_pool_blocked() const {
|
|
|
|
return (const_pool_blocked_nesting_ > 0) ||
|
|
|
|
(pc_offset() < no_const_pool_before_);
|
2010-04-27 09:09:51 +00:00
|
|
|
}
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
private:
|
|
|
|
int next_buffer_check_; // pc offset of next buffer check
|
|
|
|
|
|
|
|
// Code generation
|
|
|
|
// The relocation writer's position is at least kGap bytes below the end of
|
|
|
|
// the generated instructions. This is so that multi-instruction sequences do
|
|
|
|
// not have to check for overflow. The same is true for writes of large
|
|
|
|
// relocation info entries.
|
|
|
|
static const int kGap = 32;
|
|
|
|
|
|
|
|
// Constant pool generation
|
|
|
|
// Pools are emitted in the instruction stream, preferably after unconditional
|
|
|
|
// jumps or after returns from functions (in dead code locations).
|
|
|
|
// If a long code sequence does not contain unconditional jumps, it is
|
|
|
|
// necessary to emit the constant pool before the pool gets too far from the
|
|
|
|
// location it is accessed from. In this case, we emit a jump over the emitted
|
|
|
|
// constant pool.
|
|
|
|
// Constants in the pool may be addresses of functions that gets relocated;
|
|
|
|
// if so, a relocation info entry is associated to the constant pool entry.
|
|
|
|
|
|
|
|
// Repeated checking whether the constant pool should be emitted is rather
|
|
|
|
// expensive. By default we only check again once a number of instructions
|
|
|
|
// has been generated. That also means that the sizing of the buffers is not
|
|
|
|
// an exact science, and that we rely on some slop to not overrun buffers.
|
2011-06-16 07:00:46 +00:00
|
|
|
static const int kCheckPoolIntervalInst = 32;
|
|
|
|
static const int kCheckPoolInterval = kCheckPoolIntervalInst * kInstrSize;
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
|
2010-04-21 09:43:45 +00:00
|
|
|
// Emission of the constant pool may be blocked in some code sequences.
|
|
|
|
int const_pool_blocked_nesting_; // Block emission if this is not zero.
|
|
|
|
int no_const_pool_before_; // Block emission before this pc offset.
|
2008-07-03 15:10:15 +00:00
|
|
|
|
2011-06-16 07:00:46 +00:00
|
|
|
// Keep track of the first instruction requiring a constant pool entry
|
|
|
|
// since the previous constant pool was emitted.
|
|
|
|
int first_const_pool_use_;
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
// Relocation info generation
|
|
|
|
// Each relocation is encoded as a variable size value
|
|
|
|
static const int kMaxRelocSize = RelocInfoWriter::kMaxSize;
|
|
|
|
RelocInfoWriter reloc_info_writer;
|
2011-06-16 07:00:46 +00:00
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
// Relocation info records are also used during code generation as temporary
|
|
|
|
// containers for constants and code target addresses until they are emitted
|
|
|
|
// to the constant pool. These pending relocation info records are temporarily
|
|
|
|
// stored in a separate buffer until a constant pool is emitted.
|
|
|
|
// If every instruction in a long sequence is accessing the pool, we need one
|
|
|
|
// pending relocation entry per instruction.
|
2011-06-16 07:00:46 +00:00
|
|
|
|
|
|
|
// the buffer of pending relocation info
|
|
|
|
RelocInfo pending_reloc_info_[kMaxNumPendingRelocInfo];
|
|
|
|
// number of pending reloc info entries in the buffer
|
|
|
|
int num_pending_reloc_info_;
|
2012-12-28 13:34:15 +00:00
|
|
|
// Number of pending reloc info entries included above which also happen to
|
|
|
|
// be 64-bit.
|
|
|
|
int num_pending_64_bit_reloc_info_;
|
2008-07-03 15:10:15 +00:00
|
|
|
|
2008-09-30 09:42:08 +00:00
|
|
|
// The bound position, before this we cannot do instruction elimination.
|
2008-07-03 15:10:15 +00:00
|
|
|
int last_bound_pos_;
|
|
|
|
|
|
|
|
// Code emission
|
|
|
|
inline void CheckBuffer();
|
|
|
|
void GrowBuffer();
|
|
|
|
inline void emit(Instr x);
|
|
|
|
|
2012-10-18 12:21:42 +00:00
|
|
|
// 32-bit immediate values
|
|
|
|
void move_32_bit_immediate(Condition cond,
|
|
|
|
Register rd,
|
|
|
|
SBit s,
|
|
|
|
const Operand& x);
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
// Instruction generation
|
|
|
|
void addrmod1(Instr instr, Register rn, Register rd, const Operand& x);
|
|
|
|
void addrmod2(Instr instr, Register rd, const MemOperand& x);
|
|
|
|
void addrmod3(Instr instr, Register rd, const MemOperand& x);
|
|
|
|
void addrmod4(Instr instr, Register rn, RegList rl);
|
|
|
|
void addrmod5(Instr instr, CRegister crd, const MemOperand& x);
|
|
|
|
|
|
|
|
// Labels
|
|
|
|
void print(Label* L);
|
|
|
|
void bind_to(Label* L, int pos);
|
|
|
|
void link_to(Label* L, Label* appendix);
|
|
|
|
void next(Label* L);
|
|
|
|
|
2012-10-18 12:21:42 +00:00
|
|
|
enum UseConstantPoolMode {
|
|
|
|
USE_CONSTANT_POOL,
|
|
|
|
DONT_USE_CONSTANT_POOL
|
|
|
|
};
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
// Record reloc info for current pc_
|
2012-10-18 12:21:42 +00:00
|
|
|
void RecordRelocInfo(RelocInfo::Mode rmode, intptr_t data = 0,
|
|
|
|
UseConstantPoolMode mode = USE_CONSTANT_POOL);
|
2012-12-28 13:34:15 +00:00
|
|
|
void RecordRelocInfo(double data);
|
|
|
|
void RecordRelocInfoConstantPoolEntryHelper(const RelocInfo& rinfo);
|
2009-08-31 12:40:37 +00:00
|
|
|
|
|
|
|
friend class RegExpMacroAssemblerARM;
|
2009-09-14 06:57:24 +00:00
|
|
|
friend class RelocInfo;
|
|
|
|
friend class CodePatcher;
|
2010-04-21 09:43:45 +00:00
|
|
|
friend class BlockConstPoolScope;
|
2010-11-04 15:12:03 +00:00
|
|
|
|
|
|
|
PositionsRecorder positions_recorder_;
|
|
|
|
friend class PositionsRecorder;
|
|
|
|
friend class EnsureSpace;
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
class EnsureSpace BASE_EMBEDDED {
|
|
|
|
public:
|
2010-11-04 15:39:06 +00:00
|
|
|
explicit EnsureSpace(Assembler* assembler) {
|
2010-11-04 15:12:03 +00:00
|
|
|
assembler->CheckBuffer();
|
|
|
|
}
|
2008-07-03 15:10:15 +00:00
|
|
|
};
|
|
|
|
|
2010-11-04 15:12:03 +00:00
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
} } // namespace v8::internal
|
|
|
|
|
2009-05-04 13:36:43 +00:00
|
|
|
#endif // V8_ARM_ASSEMBLER_ARM_H_
|