mirror of
https://github.com/dart-lang/sdk
synced 2024-10-14 15:42:20 +00:00
2bfecc160b
TEST=build Change-Id: I2dd8ae69764af27f480a19995b491e98f52476ae Reviewed-on: https://dart-review.googlesource.com/c/sdk/+/293902 Reviewed-by: Liam Appelbe <liama@google.com> Commit-Queue: Ryan Macnak <rmacnak@google.com>
521 lines
18 KiB
C++
521 lines
18 KiB
C++
// Copyright (c) 2013, the Dart project authors. Please see the AUTHORS file
|
|
// for details. All rights reserved. Use of this source code is governed by a
|
|
// BSD-style license that can be found in the LICENSE file.
|
|
|
|
#ifndef RUNTIME_VM_CONSTANTS_IA32_H_
|
|
#define RUNTIME_VM_CONSTANTS_IA32_H_
|
|
|
|
#ifndef RUNTIME_VM_CONSTANTS_H_
|
|
#error Do not include constants_ia32.h directly; use constants.h instead.
|
|
#endif
|
|
|
|
#include "platform/assert.h"
|
|
#include "platform/globals.h"
|
|
#include "platform/utils.h"
|
|
|
|
#include "vm/constants_base.h"
|
|
|
|
namespace dart {
|
|
|
|
#define R(reg) (1 << (reg))
|
|
|
|
enum Register {
|
|
EAX = 0,
|
|
ECX = 1,
|
|
EDX = 2,
|
|
EBX = 3,
|
|
ESP = 4, // SP
|
|
EBP = 5, // FP
|
|
ESI = 6, // THR
|
|
EDI = 7,
|
|
kNumberOfCpuRegisters = 8,
|
|
kNoRegister = -1, // Signals an illegal register.
|
|
};
|
|
|
|
// Low and high bytes registers of the first four general purpose registers.
|
|
// The other four general purpose registers do not have byte registers.
|
|
enum ByteRegister {
|
|
AL = 0,
|
|
CL = 1,
|
|
DL = 2,
|
|
BL = 3,
|
|
AH = 4,
|
|
CH = 5,
|
|
DH = 6,
|
|
BH = 7,
|
|
kNumberOfByteRegisters = 8,
|
|
kNoByteRegister = -1 // Signals an illegal register.
|
|
};
|
|
|
|
inline ByteRegister ByteRegisterOf(Register reg) {
|
|
// This only works for EAX, ECX, EDX, EBX.
|
|
// Remaining Register values map to high byte of the above registers.
|
|
RELEASE_ASSERT(reg == Register::EAX || reg == Register::ECX ||
|
|
reg == Register::EDX || reg == Register::EBX);
|
|
return static_cast<ByteRegister>(reg);
|
|
}
|
|
|
|
enum XmmRegister {
|
|
XMM0 = 0,
|
|
XMM1 = 1,
|
|
XMM2 = 2,
|
|
XMM3 = 3,
|
|
XMM4 = 4,
|
|
XMM5 = 5,
|
|
XMM6 = 6,
|
|
XMM7 = 7,
|
|
kNumberOfXmmRegisters = 8,
|
|
kNoXmmRegister = -1 // Signals an illegal register.
|
|
};
|
|
|
|
// Architecture independent aliases.
|
|
typedef XmmRegister FpuRegister;
|
|
const FpuRegister FpuTMP = XMM7;
|
|
const int kFpuRegisterSize = 16;
|
|
typedef simd128_value_t fpu_register_t;
|
|
const int kNumberOfFpuRegisters = kNumberOfXmmRegisters;
|
|
const FpuRegister kNoFpuRegister = kNoXmmRegister;
|
|
|
|
extern const char* const cpu_reg_names[kNumberOfCpuRegisters];
|
|
extern const char* const cpu_reg_abi_names[kNumberOfCpuRegisters];
|
|
extern const char* const cpu_reg_byte_names[kNumberOfByteRegisters];
|
|
extern const char* const fpu_reg_names[kNumberOfXmmRegisters];
|
|
|
|
// Register aliases.
|
|
const Register TMP = kNoRegister; // No scratch register used by assembler.
|
|
const Register TMP2 = kNoRegister; // No second assembler scratch register.
|
|
const Register CODE_REG = EDI;
|
|
// Set when calling Dart functions in JIT mode, used by LazyCompileStub.
|
|
const Register FUNCTION_REG = EAX;
|
|
const Register PP = kNoRegister; // No object pool pointer.
|
|
const Register SPREG = ESP; // Stack pointer register.
|
|
const Register FPREG = EBP; // Frame pointer register.
|
|
const Register IC_DATA_REG = ECX; // ICData/MegamorphicCache register.
|
|
const Register ARGS_DESC_REG = EDX; // Arguments descriptor register.
|
|
const Register THR = ESI; // Caches current thread in generated code.
|
|
const Register CALLEE_SAVED_TEMP = EBX;
|
|
const Register CALLEE_SAVED_TEMP2 = EDI;
|
|
|
|
// ABI for catch-clause entry point.
|
|
const Register kExceptionObjectReg = EAX;
|
|
const Register kStackTraceObjectReg = EDX;
|
|
|
|
// ABI for write barrier stub.
|
|
const Register kWriteBarrierObjectReg = EDX;
|
|
const Register kWriteBarrierValueReg = EBX;
|
|
const Register kWriteBarrierSlotReg = EDI;
|
|
|
|
// Common ABI for shared slow path stubs.
|
|
struct SharedSlowPathStubABI {
|
|
static constexpr Register kResultReg = EAX;
|
|
};
|
|
|
|
// ABI for instantiation stubs.
|
|
struct InstantiationABI {
|
|
static constexpr Register kUninstantiatedTypeArgumentsReg = EBX;
|
|
static constexpr Register kInstantiatorTypeArgumentsReg = EDX;
|
|
static constexpr Register kFunctionTypeArgumentsReg = ECX;
|
|
static constexpr Register kResultTypeArgumentsReg = EAX;
|
|
static constexpr Register kResultTypeReg = EAX;
|
|
static constexpr Register kScratchReg =
|
|
EDI; // On ia32 we don't use CODE_REG.
|
|
};
|
|
|
|
// Registers in addition to those listed in InstantiationABI used inside the
|
|
// implementation of the InstantiateTypeArguments stubs.
|
|
struct InstantiateTAVInternalRegs {
|
|
// On IA32, we don't do hash cache checks in the stub. We only define
|
|
// kSavedRegisters to avoid needing to #ifdef uses of it.
|
|
static constexpr intptr_t kSavedRegisters = 0;
|
|
};
|
|
|
|
// Calling convention when calling SubtypeTestCacheStub.
|
|
// Although ia32 uses a stack-based calling convention, we keep the same
|
|
// 'TypeTestABI' name for symmetry with other architectures with a proper ABI.
|
|
// Note that ia32 has no support for type testing stubs.
|
|
struct TypeTestABI {
|
|
static constexpr Register kInstanceReg = EAX;
|
|
static constexpr Register kDstTypeReg = EBX;
|
|
static constexpr Register kInstantiatorTypeArgumentsReg = EDX;
|
|
static constexpr Register kFunctionTypeArgumentsReg = ECX;
|
|
static constexpr Register kSubtypeTestCacheReg =
|
|
EDI; // On ia32 we don't use CODE_REG.
|
|
|
|
// For call to InstanceOfStub.
|
|
static constexpr Register kInstanceOfResultReg = kInstanceReg;
|
|
// For call to SubtypeNTestCacheStub.
|
|
static constexpr Register kSubtypeTestCacheResultReg =
|
|
TypeTestABI::kSubtypeTestCacheReg;
|
|
};
|
|
|
|
// Calling convention when calling kSubtypeCheckRuntimeEntry, to match other
|
|
// architectures. We don't generate a call to the AssertSubtypeStub because we
|
|
// need CODE_REG to store a fifth argument.
|
|
struct AssertSubtypeABI {
|
|
static constexpr Register kSubTypeReg = EAX;
|
|
static constexpr Register kSuperTypeReg = EBX;
|
|
static constexpr Register kInstantiatorTypeArgumentsReg = EDX;
|
|
static constexpr Register kFunctionTypeArgumentsReg = ECX;
|
|
static constexpr Register kDstNameReg =
|
|
EDI; /// On ia32 we don't use CODE_REG.
|
|
|
|
// No result register, as AssertSubtype is only run for side effect
|
|
// (throws if the subtype check fails).
|
|
};
|
|
|
|
// For calling the ia32-specific AssertAssignableStub
|
|
struct AssertAssignableStubABI {
|
|
static constexpr Register kDstNameReg = EBX;
|
|
static constexpr Register kSubtypeTestReg = ECX;
|
|
|
|
static constexpr intptr_t kInstanceSlotFromFp = 2 + 3;
|
|
static constexpr intptr_t kDstTypeSlotFromFp = 2 + 2;
|
|
static constexpr intptr_t kInstantiatorTAVSlotFromFp = 2 + 1;
|
|
static constexpr intptr_t kFunctionTAVSlotFromFp = 2 + 0;
|
|
};
|
|
|
|
// ABI for InitStaticFieldStub.
|
|
struct InitStaticFieldABI {
|
|
static constexpr Register kFieldReg = EDX;
|
|
static constexpr Register kResultReg = EAX;
|
|
};
|
|
|
|
// Registers used inside the implementation of InitLateStaticFieldStub.
|
|
struct InitLateStaticFieldInternalRegs {
|
|
static constexpr Register kAddressReg = ECX;
|
|
static constexpr Register kScratchReg = EDI;
|
|
};
|
|
|
|
// ABI for InitInstanceFieldStub.
|
|
struct InitInstanceFieldABI {
|
|
static constexpr Register kInstanceReg = EBX;
|
|
static constexpr Register kFieldReg = EDX;
|
|
static constexpr Register kResultReg = EAX;
|
|
};
|
|
|
|
// Registers used inside the implementation of InitLateInstanceFieldStub.
|
|
struct InitLateInstanceFieldInternalRegs {
|
|
static constexpr Register kAddressReg = ECX;
|
|
static constexpr Register kScratchReg = EDI;
|
|
};
|
|
|
|
// ABI for LateInitializationError stubs.
|
|
struct LateInitializationErrorABI {
|
|
static constexpr Register kFieldReg = EDI;
|
|
};
|
|
|
|
// ABI for ThrowStub.
|
|
struct ThrowABI {
|
|
static constexpr Register kExceptionReg = EAX;
|
|
};
|
|
|
|
// ABI for ReThrowStub.
|
|
struct ReThrowABI {
|
|
static constexpr Register kExceptionReg = EAX;
|
|
static constexpr Register kStackTraceReg = EBX;
|
|
};
|
|
|
|
// ABI for AssertBooleanStub.
|
|
struct AssertBooleanABI {
|
|
static constexpr Register kObjectReg = EAX;
|
|
};
|
|
|
|
// ABI for RangeErrorStub.
|
|
struct RangeErrorABI {
|
|
static constexpr Register kLengthReg = EAX;
|
|
static constexpr Register kIndexReg = EBX;
|
|
};
|
|
|
|
// ABI for AllocateObjectStub.
|
|
struct AllocateObjectABI {
|
|
static constexpr Register kResultReg = EAX;
|
|
static constexpr Register kTypeArgumentsReg = EDX;
|
|
static constexpr Register kTagsReg = kNoRegister; // Not used.
|
|
};
|
|
|
|
// ABI for Allocate{Mint,Double,Float32x4,Float64x2}Stub.
|
|
struct AllocateBoxABI {
|
|
static constexpr Register kResultReg = AllocateObjectABI::kResultReg;
|
|
static constexpr Register kTempReg = EBX;
|
|
};
|
|
|
|
// ABI for AllocateClosureStub.
|
|
struct AllocateClosureABI {
|
|
static constexpr Register kResultReg = AllocateObjectABI::kResultReg;
|
|
static constexpr Register kFunctionReg = EBX;
|
|
static constexpr Register kContextReg = ECX;
|
|
static constexpr Register kScratchReg = EDX;
|
|
};
|
|
|
|
// ABI for AllocateArrayStub.
|
|
struct AllocateArrayABI {
|
|
static constexpr Register kResultReg = AllocateObjectABI::kResultReg;
|
|
static constexpr Register kLengthReg = EDX;
|
|
static constexpr Register kTypeArgumentsReg = ECX;
|
|
};
|
|
|
|
// ABI for AllocateRecordStub.
|
|
struct AllocateRecordABI {
|
|
static constexpr Register kResultReg = AllocateObjectABI::kResultReg;
|
|
static constexpr Register kShapeReg = EDX;
|
|
static constexpr Register kTemp1Reg = EBX;
|
|
static constexpr Register kTemp2Reg = EDI;
|
|
};
|
|
|
|
// ABI for AllocateSmallRecordStub (AllocateRecord2, AllocateRecord2Named,
|
|
// AllocateRecord3, AllocateRecord3Named).
|
|
struct AllocateSmallRecordABI {
|
|
static constexpr Register kResultReg = AllocateObjectABI::kResultReg;
|
|
static constexpr Register kShapeReg = EBX;
|
|
static constexpr Register kValue0Reg = ECX;
|
|
static constexpr Register kValue1Reg = EDX;
|
|
static constexpr Register kValue2Reg = kNoRegister;
|
|
static constexpr Register kTempReg = EDI;
|
|
};
|
|
|
|
// ABI for AllocateTypedDataArrayStub.
|
|
struct AllocateTypedDataArrayABI {
|
|
static constexpr Register kResultReg = AllocateObjectABI::kResultReg;
|
|
static constexpr Register kLengthReg = kResultReg;
|
|
};
|
|
|
|
// ABI for BoxDoubleStub.
|
|
struct BoxDoubleStubABI {
|
|
static constexpr FpuRegister kValueReg = XMM0;
|
|
static constexpr Register kTempReg = EBX;
|
|
static constexpr Register kResultReg = EAX;
|
|
};
|
|
|
|
// ABI for DoubleToIntegerStub.
|
|
struct DoubleToIntegerStubABI {
|
|
static constexpr FpuRegister kInputReg = XMM0;
|
|
static constexpr Register kRecognizedKindReg = EAX;
|
|
static constexpr Register kResultReg = EAX;
|
|
};
|
|
|
|
// ABI for SuspendStub (AwaitStub, AwaitWithTypeCheckStub, YieldAsyncStarStub,
|
|
// SuspendSyncStarAtStartStub, SuspendSyncStarAtYieldStub).
|
|
struct SuspendStubABI {
|
|
static constexpr Register kArgumentReg = EAX;
|
|
static constexpr Register kTypeArgsReg = EDX; // Can be the same as kTempReg
|
|
static constexpr Register kTempReg = EDX;
|
|
static constexpr Register kFrameSizeReg = ECX;
|
|
static constexpr Register kSuspendStateReg = EBX;
|
|
static constexpr Register kFunctionDataReg = EDI;
|
|
// Can reuse THR.
|
|
static constexpr Register kSrcFrameReg = ESI;
|
|
// Can reuse kFunctionDataReg.
|
|
static constexpr Register kDstFrameReg = EDI;
|
|
|
|
// Number of bytes to skip after
|
|
// suspend stub return address in order to resume.
|
|
// IA32: mov esp, ebp; pop ebp; ret
|
|
static constexpr intptr_t kResumePcDistance = 4;
|
|
};
|
|
|
|
// ABI for InitSuspendableFunctionStub (InitAsyncStub, InitAsyncStarStub,
|
|
// InitSyncStarStub).
|
|
struct InitSuspendableFunctionStubABI {
|
|
static constexpr Register kTypeArgsReg = EAX;
|
|
};
|
|
|
|
// ABI for ResumeStub
|
|
struct ResumeStubABI {
|
|
static constexpr Register kSuspendStateReg = EBX;
|
|
static constexpr Register kTempReg = EDX;
|
|
// Registers for the frame copying (the 1st part).
|
|
static constexpr Register kFrameSizeReg = ECX;
|
|
// Can reuse THR.
|
|
static constexpr Register kSrcFrameReg = ESI;
|
|
// Can reuse CODE_REG.
|
|
static constexpr Register kDstFrameReg = EDI;
|
|
// Registers for control transfer.
|
|
// (the 2nd part, can reuse registers from the 1st part)
|
|
static constexpr Register kResumePcReg = ECX;
|
|
// Can also reuse kSuspendStateReg but should not conflict with CODE_REG.
|
|
static constexpr Register kExceptionReg = EAX;
|
|
static constexpr Register kStackTraceReg = EBX;
|
|
};
|
|
|
|
// ABI for ReturnStub (ReturnAsyncStub, ReturnAsyncNotFutureStub,
|
|
// ReturnAsyncStarStub).
|
|
struct ReturnStubABI {
|
|
static constexpr Register kSuspendStateReg = EBX;
|
|
};
|
|
|
|
// ABI for AsyncExceptionHandlerStub.
|
|
struct AsyncExceptionHandlerStubABI {
|
|
static constexpr Register kSuspendStateReg = EBX;
|
|
};
|
|
|
|
// ABI for CloneSuspendStateStub.
|
|
struct CloneSuspendStateStubABI {
|
|
static constexpr Register kSourceReg = EAX;
|
|
static constexpr Register kDestinationReg = EBX;
|
|
static constexpr Register kTempReg = EDX;
|
|
static constexpr Register kFrameSizeReg = ECX;
|
|
// Can reuse THR.
|
|
static constexpr Register kSrcFrameReg = ESI;
|
|
static constexpr Register kDstFrameReg = EDI;
|
|
};
|
|
|
|
// ABI for DispatchTableNullErrorStub and consequently for all dispatch
|
|
// table calls (though normal functions will not expect or use this
|
|
// register). This ABI is added to distinguish memory corruption errors from
|
|
// null errors.
|
|
// Note: dispatch table calls are never actually generated on IA32, this
|
|
// declaration is only added for completeness.
|
|
struct DispatchTableNullErrorABI {
|
|
static constexpr Register kClassIdReg = EAX;
|
|
};
|
|
|
|
typedef uint32_t RegList;
|
|
const RegList kAllCpuRegistersList = 0xFF;
|
|
const RegList kAllFpuRegistersList = (1 << kNumberOfFpuRegisters) - 1;
|
|
|
|
const intptr_t kReservedCpuRegisters = (1 << SPREG) | (1 << FPREG) | (1 << THR);
|
|
constexpr intptr_t kNumberOfReservedCpuRegisters =
|
|
Utils::CountOneBits32(kReservedCpuRegisters);
|
|
// CPU registers available to Dart allocator.
|
|
const RegList kDartAvailableCpuRegs =
|
|
kAllCpuRegistersList & ~kReservedCpuRegisters;
|
|
constexpr int kNumberOfDartAvailableCpuRegs =
|
|
kNumberOfCpuRegisters - kNumberOfReservedCpuRegisters;
|
|
// No reason to prefer certain registers on IA32.
|
|
constexpr int kRegisterAllocationBias = 0;
|
|
constexpr int kStoreBufferWrapperSize = 11;
|
|
|
|
const RegList kAbiPreservedCpuRegs = (1 << EDI) | (1 << ESI) | (1 << EBX);
|
|
|
|
// Registers available to Dart that are not preserved by runtime calls.
|
|
const RegList kDartVolatileCpuRegs =
|
|
kDartAvailableCpuRegs & ~kAbiPreservedCpuRegs;
|
|
|
|
const RegList kAbiVolatileFpuRegs = kAllFpuRegistersList;
|
|
|
|
#undef R
|
|
|
|
enum ScaleFactor {
|
|
TIMES_1 = 0,
|
|
TIMES_2 = 1,
|
|
TIMES_4 = 2,
|
|
TIMES_8 = 3,
|
|
TIMES_16 = 4,
|
|
// We can't include vm/compiler/runtime_api.h, so just be explicit instead
|
|
// of using (dart::)kWordSizeLog2.
|
|
#if defined(TARGET_ARCH_IS_32_BIT)
|
|
// Used for Smi-boxed indices.
|
|
TIMES_HALF_WORD_SIZE = kInt32SizeLog2 - 1,
|
|
// Used for unboxed indices.
|
|
TIMES_WORD_SIZE = kInt32SizeLog2,
|
|
#else
|
|
#error "Unexpected word size"
|
|
#endif
|
|
#if !defined(DART_COMPRESSED_POINTERS)
|
|
TIMES_COMPRESSED_WORD_SIZE = TIMES_WORD_SIZE,
|
|
#else
|
|
#error Cannot compress IA32
|
|
#endif
|
|
// Used for Smi-boxed indices.
|
|
TIMES_COMPRESSED_HALF_WORD_SIZE = TIMES_COMPRESSED_WORD_SIZE - 1,
|
|
};
|
|
|
|
class Instr {
|
|
public:
|
|
static constexpr uint8_t kHltInstruction = 0xF4;
|
|
// We prefer not to use the int3 instruction since it conflicts with gdb.
|
|
static constexpr uint8_t kBreakPointInstruction = kHltInstruction;
|
|
static constexpr int kBreakPointInstructionSize = 1;
|
|
|
|
bool IsBreakPoint() {
|
|
ASSERT(kBreakPointInstructionSize == 1);
|
|
return (*reinterpret_cast<const uint8_t*>(this)) == kBreakPointInstruction;
|
|
}
|
|
|
|
// Instructions are read out of a code stream. The only way to get a
|
|
// reference to an instruction is to convert a pointer. There is no way
|
|
// to allocate or create instances of class Instr.
|
|
// Use the At(pc) function to create references to Instr.
|
|
static Instr* At(uword pc) { return reinterpret_cast<Instr*>(pc); }
|
|
|
|
private:
|
|
DISALLOW_ALLOCATION();
|
|
// We need to prevent the creation of instances of class Instr.
|
|
DISALLOW_IMPLICIT_CONSTRUCTORS(Instr);
|
|
};
|
|
|
|
// The largest multibyte nop we will emit. This could go up to 15 if it
|
|
// becomes important to us.
|
|
const int MAX_NOP_SIZE = 8;
|
|
|
|
class CallingConventions {
|
|
public:
|
|
static const Register ArgumentRegisters[];
|
|
static constexpr intptr_t kArgumentRegisters = 0;
|
|
static constexpr intptr_t kFpuArgumentRegisters = 0;
|
|
static constexpr intptr_t kNumArgRegs = 0;
|
|
static constexpr Register kPointerToReturnStructRegisterCall = kNoRegister;
|
|
|
|
static const XmmRegister FpuArgumentRegisters[];
|
|
static constexpr intptr_t kXmmArgumentRegisters = 0;
|
|
static constexpr intptr_t kNumFpuArgRegs = 0;
|
|
|
|
static constexpr intptr_t kCalleeSaveCpuRegisters = kAbiPreservedCpuRegs;
|
|
|
|
static constexpr bool kArgumentIntRegXorFpuReg = false;
|
|
|
|
static constexpr Register kReturnReg = EAX;
|
|
static constexpr Register kSecondReturnReg = EDX;
|
|
static constexpr Register kPointerToReturnStructRegisterReturn = kReturnReg;
|
|
|
|
// Whether the callee uses `ret 4` instead of `ret` to return with struct
|
|
// return values.
|
|
// See: https://c9x.me/x86/html/file_module_x86_id_280.html
|
|
#if defined(_WIN32)
|
|
static constexpr bool kUsesRet4 = false;
|
|
#else
|
|
static constexpr bool kUsesRet4 = true;
|
|
#endif
|
|
|
|
// Floating point values are returned on the "FPU stack" (in "ST" registers).
|
|
// However, we use XMM0 in our compiler pipeline as the location.
|
|
// The move from and to ST is done in FfiCallInstr::EmitNativeCode and
|
|
// NativeReturnInstr::EmitNativeCode.
|
|
static constexpr XmmRegister kReturnFpuReg = XMM0;
|
|
|
|
static constexpr Register kFfiAnyNonAbiRegister = EBX;
|
|
static constexpr Register kFirstNonArgumentRegister = EAX;
|
|
static constexpr Register kSecondNonArgumentRegister = ECX;
|
|
static constexpr Register kStackPointerRegister = SPREG;
|
|
|
|
// Whether larger than wordsize arguments are aligned to even registers.
|
|
static constexpr AlignmentStrategy kArgumentRegisterAlignment =
|
|
kAlignedToWordSize;
|
|
static constexpr AlignmentStrategy kArgumentRegisterAlignmentVarArgs =
|
|
kArgumentRegisterAlignment;
|
|
|
|
// How stack arguments are aligned.
|
|
static constexpr AlignmentStrategy kArgumentStackAlignment =
|
|
kAlignedToWordSize;
|
|
|
|
// How fields in compounds are aligned.
|
|
#if defined(DART_TARGET_OS_WINDOWS)
|
|
static constexpr AlignmentStrategy kFieldAlignment = kAlignedToValueSize;
|
|
#else
|
|
static constexpr AlignmentStrategy kFieldAlignment =
|
|
kAlignedToValueSizeBut8AlignedTo4;
|
|
#endif
|
|
|
|
// Whether 1 or 2 byte-sized arguments or return values are passed extended
|
|
// to 4 bytes.
|
|
static constexpr ExtensionStrategy kReturnRegisterExtension = kNotExtended;
|
|
static constexpr ExtensionStrategy kArgumentRegisterExtension = kNotExtended;
|
|
static constexpr ExtensionStrategy kArgumentStackExtension = kExtendedTo4;
|
|
};
|
|
|
|
const uword kBreakInstructionFiller = 0xCCCCCCCC;
|
|
|
|
} // namespace dart
|
|
|
|
#endif // RUNTIME_VM_CONSTANTS_IA32_H_
|