Revert "[compiler] Support acq/rel accesses and atomic accesses on tagged"
This reverts commit faf2208a0b
.
Reason for revert: https://ci.chromium.org/ui/p/v8/builders/ci/V8%20Linux64%20-%20arm64%20-%20sim%20-%20pointer%20compression/10870/overview
Original change's description:
> [compiler] Support acq/rel accesses and atomic accesses on tagged
>
> This CL adds an AtomicMemoryOrder parameter to the various atomic load
> and store operators. Currently only acquire release (kAcqRel) and
> sequentially consistent (kSeqCst) orders are supported.
>
> Additionally, atomic loads and stores are extended to work with tagged
> values.
>
> This CL is a pre-requisite for supporting atomic accesses in Torque,
> which is in turn a pre-requisite for prototyping shared strings.
>
> Bug: v8:11995
> Change-Id: Ic77d2640e2dc7e5581b1211a054c93210c219355
> Reviewed-on: https://chromium-review.googlesource.com/c/v8/v8/+/3101765
> Reviewed-by: Nico Hartmann <nicohartmann@chromium.org>
> Reviewed-by: Zhi An Ng <zhin@chromium.org>
> Commit-Queue: Shu-yu Guo <syg@chromium.org>
> Cr-Commit-Position: refs/heads/main@{#76393}
Bug: v8:11995
Change-Id: Id9936672f9e96c509b1cdf866de1ac5303996945
No-Presubmit: true
No-Tree-Checks: true
No-Try: true
Reviewed-on: https://chromium-review.googlesource.com/c/v8/v8/+/3107229
Reviewed-by: Nico Hartmann <nicohartmann@chromium.org>
Commit-Queue: Nico Hartmann <nicohartmann@chromium.org>
Bot-Commit: Rubber Stamper <rubber-stamper@appspot.gserviceaccount.com>
Cr-Commit-Position: refs/heads/main@{#76394}
This commit is contained in:
parent
faf2208a0b
commit
746d62d4b9
1
BUILD.gn
1
BUILD.gn
@ -2449,7 +2449,6 @@ v8_header_set("v8_internal_headers") {
|
|||||||
"src/codegen/assembler-arch.h",
|
"src/codegen/assembler-arch.h",
|
||||||
"src/codegen/assembler-inl.h",
|
"src/codegen/assembler-inl.h",
|
||||||
"src/codegen/assembler.h",
|
"src/codegen/assembler.h",
|
||||||
"src/codegen/atomic-memory-order.h",
|
|
||||||
"src/codegen/bailout-reason.h",
|
"src/codegen/bailout-reason.h",
|
||||||
"src/codegen/callable.h",
|
"src/codegen/callable.h",
|
||||||
"src/codegen/code-comments.h",
|
"src/codegen/code-comments.h",
|
||||||
|
@ -204,28 +204,26 @@ TF_BUILTIN(AtomicsLoad, SharedArrayBufferBuiltinsAssembler) {
|
|||||||
arraysize(case_labels));
|
arraysize(case_labels));
|
||||||
|
|
||||||
BIND(&i8);
|
BIND(&i8);
|
||||||
Return(SmiFromInt32(AtomicLoad<Int8T>(AtomicMemoryOrder::kSeqCst,
|
Return(SmiFromInt32(AtomicLoad<Int8T>(backing_store, index_word)));
|
||||||
backing_store, index_word)));
|
|
||||||
|
|
||||||
BIND(&u8);
|
BIND(&u8);
|
||||||
Return(SmiFromInt32(AtomicLoad<Uint8T>(AtomicMemoryOrder::kSeqCst,
|
Return(SmiFromInt32(AtomicLoad<Uint8T>(backing_store, index_word)));
|
||||||
backing_store, index_word)));
|
|
||||||
|
|
||||||
BIND(&i16);
|
BIND(&i16);
|
||||||
Return(SmiFromInt32(AtomicLoad<Int16T>(
|
Return(
|
||||||
AtomicMemoryOrder::kSeqCst, backing_store, WordShl(index_word, 1))));
|
SmiFromInt32(AtomicLoad<Int16T>(backing_store, WordShl(index_word, 1))));
|
||||||
|
|
||||||
BIND(&u16);
|
BIND(&u16);
|
||||||
Return(SmiFromInt32(AtomicLoad<Uint16T>(
|
Return(
|
||||||
AtomicMemoryOrder::kSeqCst, backing_store, WordShl(index_word, 1))));
|
SmiFromInt32(AtomicLoad<Uint16T>(backing_store, WordShl(index_word, 1))));
|
||||||
|
|
||||||
BIND(&i32);
|
BIND(&i32);
|
||||||
Return(ChangeInt32ToTagged(AtomicLoad<Int32T>(
|
Return(ChangeInt32ToTagged(
|
||||||
AtomicMemoryOrder::kSeqCst, backing_store, WordShl(index_word, 2))));
|
AtomicLoad<Int32T>(backing_store, WordShl(index_word, 2))));
|
||||||
|
|
||||||
BIND(&u32);
|
BIND(&u32);
|
||||||
Return(ChangeUint32ToTagged(AtomicLoad<Uint32T>(
|
Return(ChangeUint32ToTagged(
|
||||||
AtomicMemoryOrder::kSeqCst, backing_store, WordShl(index_word, 2))));
|
AtomicLoad<Uint32T>(backing_store, WordShl(index_word, 2))));
|
||||||
#if V8_TARGET_ARCH_MIPS && !_MIPS_ARCH_MIPS32R6
|
#if V8_TARGET_ARCH_MIPS && !_MIPS_ARCH_MIPS32R6
|
||||||
BIND(&i64);
|
BIND(&i64);
|
||||||
Goto(&u64);
|
Goto(&u64);
|
||||||
@ -237,12 +235,12 @@ TF_BUILTIN(AtomicsLoad, SharedArrayBufferBuiltinsAssembler) {
|
|||||||
}
|
}
|
||||||
#else
|
#else
|
||||||
BIND(&i64);
|
BIND(&i64);
|
||||||
Return(BigIntFromSigned64(AtomicLoad64<AtomicInt64>(
|
Return(BigIntFromSigned64(
|
||||||
AtomicMemoryOrder::kSeqCst, backing_store, WordShl(index_word, 3))));
|
AtomicLoad64<AtomicInt64>(backing_store, WordShl(index_word, 3))));
|
||||||
|
|
||||||
BIND(&u64);
|
BIND(&u64);
|
||||||
Return(BigIntFromUnsigned64(AtomicLoad64<AtomicUint64>(
|
Return(BigIntFromUnsigned64(
|
||||||
AtomicMemoryOrder::kSeqCst, backing_store, WordShl(index_word, 3))));
|
AtomicLoad64<AtomicUint64>(backing_store, WordShl(index_word, 3))));
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
// This shouldn't happen, we've already validated the type.
|
// This shouldn't happen, we've already validated the type.
|
||||||
@ -309,18 +307,18 @@ TF_BUILTIN(AtomicsStore, SharedArrayBufferBuiltinsAssembler) {
|
|||||||
arraysize(case_labels));
|
arraysize(case_labels));
|
||||||
|
|
||||||
BIND(&u8);
|
BIND(&u8);
|
||||||
AtomicStore(MachineRepresentation::kWord8, AtomicMemoryOrder::kSeqCst,
|
AtomicStore(MachineRepresentation::kWord8, backing_store, index_word,
|
||||||
backing_store, index_word, value_word32);
|
value_word32);
|
||||||
Return(value_integer);
|
Return(value_integer);
|
||||||
|
|
||||||
BIND(&u16);
|
BIND(&u16);
|
||||||
AtomicStore(MachineRepresentation::kWord16, AtomicMemoryOrder::kSeqCst,
|
AtomicStore(MachineRepresentation::kWord16, backing_store,
|
||||||
backing_store, WordShl(index_word, 1), value_word32);
|
WordShl(index_word, 1), value_word32);
|
||||||
Return(value_integer);
|
Return(value_integer);
|
||||||
|
|
||||||
BIND(&u32);
|
BIND(&u32);
|
||||||
AtomicStore(MachineRepresentation::kWord32, AtomicMemoryOrder::kSeqCst,
|
AtomicStore(MachineRepresentation::kWord32, backing_store,
|
||||||
backing_store, WordShl(index_word, 2), value_word32);
|
WordShl(index_word, 2), value_word32);
|
||||||
Return(value_integer);
|
Return(value_integer);
|
||||||
|
|
||||||
BIND(&u64);
|
BIND(&u64);
|
||||||
@ -342,8 +340,7 @@ TF_BUILTIN(AtomicsStore, SharedArrayBufferBuiltinsAssembler) {
|
|||||||
TVARIABLE(UintPtrT, var_high);
|
TVARIABLE(UintPtrT, var_high);
|
||||||
BigIntToRawBytes(value_bigint, &var_low, &var_high);
|
BigIntToRawBytes(value_bigint, &var_low, &var_high);
|
||||||
TNode<UintPtrT> high = Is64() ? TNode<UintPtrT>() : var_high.value();
|
TNode<UintPtrT> high = Is64() ? TNode<UintPtrT>() : var_high.value();
|
||||||
AtomicStore64(AtomicMemoryOrder::kSeqCst, backing_store,
|
AtomicStore64(backing_store, WordShl(index_word, 3), var_low.value(), high);
|
||||||
WordShl(index_word, 3), var_low.value(), high);
|
|
||||||
Return(value_bigint);
|
Return(value_bigint);
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
@ -2919,18 +2919,6 @@ void TurboAssembler::StoreTaggedField(const Register& value,
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void TurboAssembler::AtomicStoreTaggedField(const Register& value,
|
|
||||||
const Register& dst_base,
|
|
||||||
const Register& dst_index,
|
|
||||||
const Register& temp) {
|
|
||||||
Add(temp, dst_base, dst_index);
|
|
||||||
if (COMPRESS_POINTERS_BOOL) {
|
|
||||||
Stlr(value.W(), temp);
|
|
||||||
} else {
|
|
||||||
Stlr(value, temp);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void TurboAssembler::DecompressTaggedSigned(const Register& destination,
|
void TurboAssembler::DecompressTaggedSigned(const Register& destination,
|
||||||
const MemOperand& field_operand) {
|
const MemOperand& field_operand) {
|
||||||
ASM_CODE_COMMENT(this);
|
ASM_CODE_COMMENT(this);
|
||||||
@ -2962,40 +2950,6 @@ void TurboAssembler::DecompressAnyTagged(const Register& destination,
|
|||||||
Add(destination, kPtrComprCageBaseRegister, destination);
|
Add(destination, kPtrComprCageBaseRegister, destination);
|
||||||
}
|
}
|
||||||
|
|
||||||
void TurboAssembler::AtomicDecompressTaggedSigned(const Register& destination,
|
|
||||||
const Register& base,
|
|
||||||
const Register& index,
|
|
||||||
const Register& temp) {
|
|
||||||
ASM_CODE_COMMENT(this);
|
|
||||||
Add(temp, base, index);
|
|
||||||
Ldar(destination.W(), temp);
|
|
||||||
if (FLAG_debug_code) {
|
|
||||||
// Corrupt the top 32 bits. Made up of 16 fixed bits and 16 pc offset bits.
|
|
||||||
Add(destination, destination,
|
|
||||||
((kDebugZapValue << 16) | (pc_offset() & 0xffff)) << 32);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void TurboAssembler::AtomicDecompressTaggedPointer(const Register& destination,
|
|
||||||
const Register& base,
|
|
||||||
const Register& index,
|
|
||||||
const Register& temp) {
|
|
||||||
ASM_CODE_COMMENT(this);
|
|
||||||
Add(temp, base, index);
|
|
||||||
Ldar(destination.W(), temp);
|
|
||||||
Add(destination, kPtrComprCageBaseRegister, destination);
|
|
||||||
}
|
|
||||||
|
|
||||||
void TurboAssembler::AtomicDecompressAnyTagged(const Register& destination,
|
|
||||||
const Register& base,
|
|
||||||
const Register& index,
|
|
||||||
const Register& temp) {
|
|
||||||
ASM_CODE_COMMENT(this);
|
|
||||||
Add(temp, base, index);
|
|
||||||
Ldar(destination.W(), temp);
|
|
||||||
Add(destination, kPtrComprCageBaseRegister, destination);
|
|
||||||
}
|
|
||||||
|
|
||||||
void TurboAssembler::CheckPageFlag(const Register& object, int mask,
|
void TurboAssembler::CheckPageFlag(const Register& object, int mask,
|
||||||
Condition cc, Label* condition_met) {
|
Condition cc, Label* condition_met) {
|
||||||
ASM_CODE_COMMENT(this);
|
ASM_CODE_COMMENT(this);
|
||||||
|
@ -1371,9 +1371,6 @@ class V8_EXPORT_PRIVATE TurboAssembler : public TurboAssemblerBase {
|
|||||||
void StoreTaggedField(const Register& value,
|
void StoreTaggedField(const Register& value,
|
||||||
const MemOperand& dst_field_operand);
|
const MemOperand& dst_field_operand);
|
||||||
|
|
||||||
void AtomicStoreTaggedField(const Register& value, const Register& dst_base,
|
|
||||||
const Register& dst_index, const Register& temp);
|
|
||||||
|
|
||||||
void DecompressTaggedSigned(const Register& destination,
|
void DecompressTaggedSigned(const Register& destination,
|
||||||
const MemOperand& field_operand);
|
const MemOperand& field_operand);
|
||||||
void DecompressTaggedPointer(const Register& destination,
|
void DecompressTaggedPointer(const Register& destination,
|
||||||
@ -1383,17 +1380,6 @@ class V8_EXPORT_PRIVATE TurboAssembler : public TurboAssemblerBase {
|
|||||||
void DecompressAnyTagged(const Register& destination,
|
void DecompressAnyTagged(const Register& destination,
|
||||||
const MemOperand& field_operand);
|
const MemOperand& field_operand);
|
||||||
|
|
||||||
void AtomicDecompressTaggedSigned(const Register& destination,
|
|
||||||
const Register& base, const Register& index,
|
|
||||||
const Register& temp);
|
|
||||||
void AtomicDecompressTaggedPointer(const Register& destination,
|
|
||||||
const Register& base,
|
|
||||||
const Register& index,
|
|
||||||
const Register& temp);
|
|
||||||
void AtomicDecompressAnyTagged(const Register& destination,
|
|
||||||
const Register& base, const Register& index,
|
|
||||||
const Register& temp);
|
|
||||||
|
|
||||||
// Restore FP and LR from the values stored in the current frame. This will
|
// Restore FP and LR from the values stored in the current frame. This will
|
||||||
// authenticate the LR when pointer authentication is enabled.
|
// authenticate the LR when pointer authentication is enabled.
|
||||||
void RestoreFPAndLR();
|
void RestoreFPAndLR();
|
||||||
|
@ -1,35 +0,0 @@
|
|||||||
// Copyright 2021 the V8 project authors. All rights reserved.
|
|
||||||
// Use of this source code is governed by a BSD-style license that can be
|
|
||||||
// found in the LICENSE file.
|
|
||||||
|
|
||||||
#ifndef V8_CODEGEN_ATOMIC_MEMORY_ORDER_H_
|
|
||||||
#define V8_CODEGEN_ATOMIC_MEMORY_ORDER_H_
|
|
||||||
|
|
||||||
#include <ostream>
|
|
||||||
|
|
||||||
#include "src/base/logging.h"
|
|
||||||
|
|
||||||
namespace v8 {
|
|
||||||
namespace internal {
|
|
||||||
|
|
||||||
// Atomic memory orders supported by the compiler.
|
|
||||||
enum class AtomicMemoryOrder : uint8_t { kAcqRel, kSeqCst };
|
|
||||||
|
|
||||||
inline size_t hash_value(AtomicMemoryOrder order) {
|
|
||||||
return static_cast<uint8_t>(order);
|
|
||||||
}
|
|
||||||
|
|
||||||
inline std::ostream& operator<<(std::ostream& os, AtomicMemoryOrder order) {
|
|
||||||
switch (order) {
|
|
||||||
case AtomicMemoryOrder::kAcqRel:
|
|
||||||
return os << "kAcqRel";
|
|
||||||
case AtomicMemoryOrder::kSeqCst:
|
|
||||||
return os << "kSeqCst";
|
|
||||||
}
|
|
||||||
UNREACHABLE();
|
|
||||||
}
|
|
||||||
|
|
||||||
} // namespace internal
|
|
||||||
} // namespace v8
|
|
||||||
|
|
||||||
#endif // V8_CODEGEN_ATOMIC_MEMORY_ORDER_H_
|
|
@ -688,14 +688,6 @@ void Assembler::movq(XMMRegister dst, Operand src) {
|
|||||||
emit_operand(dst, src);
|
emit_operand(dst, src);
|
||||||
}
|
}
|
||||||
|
|
||||||
void Assembler::movq(Operand dst, XMMRegister src) {
|
|
||||||
EnsureSpace ensure_space(this);
|
|
||||||
EMIT(0x66);
|
|
||||||
EMIT(0x0F);
|
|
||||||
EMIT(0xD6);
|
|
||||||
emit_operand(src, dst);
|
|
||||||
}
|
|
||||||
|
|
||||||
void Assembler::cmov(Condition cc, Register dst, Operand src) {
|
void Assembler::cmov(Condition cc, Register dst, Operand src) {
|
||||||
EnsureSpace ensure_space(this);
|
EnsureSpace ensure_space(this);
|
||||||
// Opcode: 0f 40 + cc /r.
|
// Opcode: 0f 40 + cc /r.
|
||||||
|
@ -535,7 +535,6 @@ class V8_EXPORT_PRIVATE Assembler : public AssemblerBase {
|
|||||||
void movzx_w(Register dst, Operand src);
|
void movzx_w(Register dst, Operand src);
|
||||||
|
|
||||||
void movq(XMMRegister dst, Operand src);
|
void movq(XMMRegister dst, Operand src);
|
||||||
void movq(Operand dst, XMMRegister src);
|
|
||||||
|
|
||||||
// Conditional moves
|
// Conditional moves
|
||||||
void cmov(Condition cc, Register dst, Register src) {
|
void cmov(Condition cc, Register dst, Register src) {
|
||||||
|
@ -294,17 +294,6 @@ void TurboAssembler::StoreTaggedSignedField(Operand dst_field_operand,
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void TurboAssembler::AtomicStoreTaggedField(Operand dst_field_operand,
|
|
||||||
Register value) {
|
|
||||||
if (COMPRESS_POINTERS_BOOL) {
|
|
||||||
movl(kScratchRegister, value);
|
|
||||||
xchgl(kScratchRegister, dst_field_operand);
|
|
||||||
} else {
|
|
||||||
movq(kScratchRegister, value);
|
|
||||||
xchgq(kScratchRegister, dst_field_operand);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void TurboAssembler::DecompressTaggedSigned(Register destination,
|
void TurboAssembler::DecompressTaggedSigned(Register destination,
|
||||||
Operand field_operand) {
|
Operand field_operand) {
|
||||||
ASM_CODE_COMMENT(this);
|
ASM_CODE_COMMENT(this);
|
||||||
|
@ -667,7 +667,6 @@ class V8_EXPORT_PRIVATE TurboAssembler : public SharedTurboAssembler {
|
|||||||
void StoreTaggedField(Operand dst_field_operand, Immediate immediate);
|
void StoreTaggedField(Operand dst_field_operand, Immediate immediate);
|
||||||
void StoreTaggedField(Operand dst_field_operand, Register value);
|
void StoreTaggedField(Operand dst_field_operand, Register value);
|
||||||
void StoreTaggedSignedField(Operand dst_field_operand, Smi value);
|
void StoreTaggedSignedField(Operand dst_field_operand, Smi value);
|
||||||
void AtomicStoreTaggedField(Operand dst_field_operand, Register value);
|
|
||||||
|
|
||||||
// The following macros work even when pointer compression is not enabled.
|
// The following macros work even when pointer compression is not enabled.
|
||||||
void DecompressTaggedSigned(Register destination, Operand field_operand);
|
void DecompressTaggedSigned(Register destination, Operand field_operand);
|
||||||
|
@ -329,11 +329,12 @@ Condition FlagsConditionToCondition(FlagsCondition condition) {
|
|||||||
__ dmb(ISH); \
|
__ dmb(ISH); \
|
||||||
} while (0)
|
} while (0)
|
||||||
|
|
||||||
#define ASSEMBLE_ATOMIC_STORE_INTEGER(asm_instr, order) \
|
#define ASSEMBLE_ATOMIC_STORE_INTEGER(asm_instr) \
|
||||||
do { \
|
do { \
|
||||||
__ dmb(ISH); \
|
__ dmb(ISH); \
|
||||||
__ asm_instr(i.InputRegister(0), i.InputOffset(1)); \
|
__ asm_instr(i.InputRegister(2), \
|
||||||
if (order == AtomicMemoryOrder::kSeqCst) __ dmb(ISH); \
|
MemOperand(i.InputRegister(0), i.InputRegister(1))); \
|
||||||
|
__ dmb(ISH); \
|
||||||
} while (0)
|
} while (0)
|
||||||
|
|
||||||
#define ASSEMBLE_ATOMIC_EXCHANGE_INTEGER(load_instr, store_instr) \
|
#define ASSEMBLE_ATOMIC_EXCHANGE_INTEGER(load_instr, store_instr) \
|
||||||
@ -926,24 +927,15 @@ CodeGenerator::CodeGenResult CodeGenerator::AssembleArchInstruction(
|
|||||||
i.InputDoubleRegister(0), DetermineStubCallMode());
|
i.InputDoubleRegister(0), DetermineStubCallMode());
|
||||||
DCHECK_EQ(LeaveCC, i.OutputSBit());
|
DCHECK_EQ(LeaveCC, i.OutputSBit());
|
||||||
break;
|
break;
|
||||||
case kArchStoreWithWriteBarrier: // Fall through.
|
case kArchStoreWithWriteBarrier: {
|
||||||
case kArchAtomicStoreWithWriteBarrier: {
|
RecordWriteMode mode =
|
||||||
RecordWriteMode mode;
|
static_cast<RecordWriteMode>(MiscField::decode(instr->opcode()));
|
||||||
if (arch_opcode == kArchStoreWithWriteBarrier) {
|
|
||||||
mode = static_cast<RecordWriteMode>(MiscField::decode(instr->opcode()));
|
|
||||||
} else {
|
|
||||||
mode = AtomicStoreRecordWriteModeField::decode(instr->opcode());
|
|
||||||
}
|
|
||||||
Register object = i.InputRegister(0);
|
Register object = i.InputRegister(0);
|
||||||
Register value = i.InputRegister(2);
|
Register value = i.InputRegister(2);
|
||||||
|
|
||||||
AddressingMode addressing_mode =
|
AddressingMode addressing_mode =
|
||||||
AddressingModeField::decode(instr->opcode());
|
AddressingModeField::decode(instr->opcode());
|
||||||
Operand offset(0);
|
Operand offset(0);
|
||||||
|
|
||||||
if (arch_opcode == kArchAtomicStoreWithWriteBarrier) {
|
|
||||||
__ dmb(ISH);
|
|
||||||
}
|
|
||||||
if (addressing_mode == kMode_Offset_RI) {
|
if (addressing_mode == kMode_Offset_RI) {
|
||||||
int32_t immediate = i.InputInt32(1);
|
int32_t immediate = i.InputInt32(1);
|
||||||
offset = Operand(immediate);
|
offset = Operand(immediate);
|
||||||
@ -954,12 +946,6 @@ CodeGenerator::CodeGenResult CodeGenerator::AssembleArchInstruction(
|
|||||||
offset = Operand(reg);
|
offset = Operand(reg);
|
||||||
__ str(value, MemOperand(object, reg));
|
__ str(value, MemOperand(object, reg));
|
||||||
}
|
}
|
||||||
if (arch_opcode == kArchAtomicStoreWithWriteBarrier &&
|
|
||||||
AtomicMemoryOrderField::decode(instr->opcode()) ==
|
|
||||||
AtomicMemoryOrder::kSeqCst) {
|
|
||||||
__ dmb(ISH);
|
|
||||||
}
|
|
||||||
|
|
||||||
auto ool = zone()->New<OutOfLineRecordWrite>(
|
auto ool = zone()->New<OutOfLineRecordWrite>(
|
||||||
this, object, offset, value, mode, DetermineStubCallMode(),
|
this, object, offset, value, mode, DetermineStubCallMode(),
|
||||||
&unwinding_info_writer_);
|
&unwinding_info_writer_);
|
||||||
@ -3328,16 +3314,13 @@ CodeGenerator::CodeGenResult CodeGenerator::AssembleArchInstruction(
|
|||||||
ASSEMBLE_ATOMIC_LOAD_INTEGER(ldr);
|
ASSEMBLE_ATOMIC_LOAD_INTEGER(ldr);
|
||||||
break;
|
break;
|
||||||
case kAtomicStoreWord8:
|
case kAtomicStoreWord8:
|
||||||
ASSEMBLE_ATOMIC_STORE_INTEGER(strb,
|
ASSEMBLE_ATOMIC_STORE_INTEGER(strb);
|
||||||
AtomicMemoryOrderField::decode(opcode));
|
|
||||||
break;
|
break;
|
||||||
case kAtomicStoreWord16:
|
case kAtomicStoreWord16:
|
||||||
ASSEMBLE_ATOMIC_STORE_INTEGER(strh,
|
ASSEMBLE_ATOMIC_STORE_INTEGER(strh);
|
||||||
AtomicMemoryOrderField::decode(opcode));
|
|
||||||
break;
|
break;
|
||||||
case kAtomicStoreWord32:
|
case kAtomicStoreWord32:
|
||||||
ASSEMBLE_ATOMIC_STORE_INTEGER(str,
|
ASSEMBLE_ATOMIC_STORE_INTEGER(str);
|
||||||
AtomicMemoryOrderField::decode(opcode));
|
|
||||||
break;
|
break;
|
||||||
case kAtomicExchangeInt8:
|
case kAtomicExchangeInt8:
|
||||||
ASSEMBLE_ATOMIC_EXCHANGE_INTEGER(ldrexb, strexb);
|
ASSEMBLE_ATOMIC_EXCHANGE_INTEGER(ldrexb, strexb);
|
||||||
|
@ -430,18 +430,17 @@ void EmitLoad(InstructionSelector* selector, InstructionCode opcode,
|
|||||||
void EmitStore(InstructionSelector* selector, InstructionCode opcode,
|
void EmitStore(InstructionSelector* selector, InstructionCode opcode,
|
||||||
size_t input_count, InstructionOperand* inputs, Node* index) {
|
size_t input_count, InstructionOperand* inputs, Node* index) {
|
||||||
ArmOperandGenerator g(selector);
|
ArmOperandGenerator g(selector);
|
||||||
ArchOpcode arch_opcode = ArchOpcodeField::decode(opcode);
|
|
||||||
|
|
||||||
if (g.CanBeImmediate(index, opcode)) {
|
if (g.CanBeImmediate(index, opcode)) {
|
||||||
inputs[input_count++] = g.UseImmediate(index);
|
inputs[input_count++] = g.UseImmediate(index);
|
||||||
opcode |= AddressingModeField::encode(kMode_Offset_RI);
|
opcode |= AddressingModeField::encode(kMode_Offset_RI);
|
||||||
} else if ((arch_opcode == kArmStr || arch_opcode == kAtomicStoreWord32) &&
|
} else if ((opcode == kArmStr) &&
|
||||||
TryMatchLSLImmediate(selector, &opcode, index, &inputs[2],
|
TryMatchLSLImmediate(selector, &opcode, index, &inputs[2],
|
||||||
&inputs[3])) {
|
&inputs[3])) {
|
||||||
input_count = 4;
|
input_count = 4;
|
||||||
} else {
|
} else {
|
||||||
inputs[input_count++] = g.UseRegister(index);
|
inputs[input_count++] = g.UseRegister(index);
|
||||||
if (arch_opcode == kArmVst1S128) {
|
if (opcode == kArmVst1S128) {
|
||||||
// Inputs are value, base, index, only care about base and index.
|
// Inputs are value, base, index, only care about base and index.
|
||||||
EmitAddBeforeS128LoadStore(selector, &opcode, &input_count, &inputs[1]);
|
EmitAddBeforeS128LoadStore(selector, &opcode, &input_count, &inputs[1]);
|
||||||
} else {
|
} else {
|
||||||
@ -641,60 +640,13 @@ void InstructionSelector::VisitProtectedLoad(Node* node) {
|
|||||||
UNIMPLEMENTED();
|
UNIMPLEMENTED();
|
||||||
}
|
}
|
||||||
|
|
||||||
namespace {
|
void InstructionSelector::VisitStore(Node* node) {
|
||||||
|
ArmOperandGenerator g(this);
|
||||||
ArchOpcode GetStoreOpcode(MachineRepresentation rep) {
|
|
||||||
switch (rep) {
|
|
||||||
case MachineRepresentation::kFloat32:
|
|
||||||
return kArmVstrF32;
|
|
||||||
case MachineRepresentation::kFloat64:
|
|
||||||
return kArmVstrF64;
|
|
||||||
case MachineRepresentation::kBit: // Fall through.
|
|
||||||
case MachineRepresentation::kWord8:
|
|
||||||
return kArmStrb;
|
|
||||||
case MachineRepresentation::kWord16:
|
|
||||||
return kArmStrh;
|
|
||||||
case MachineRepresentation::kTaggedSigned: // Fall through.
|
|
||||||
case MachineRepresentation::kTaggedPointer: // Fall through.
|
|
||||||
case MachineRepresentation::kTagged: // Fall through.
|
|
||||||
case MachineRepresentation::kWord32:
|
|
||||||
return kArmStr;
|
|
||||||
case MachineRepresentation::kSimd128:
|
|
||||||
return kArmVst1S128;
|
|
||||||
break;
|
|
||||||
case MachineRepresentation::kCompressedPointer: // Fall through.
|
|
||||||
case MachineRepresentation::kCompressed: // Fall through.
|
|
||||||
case MachineRepresentation::kWord64: // Fall through.
|
|
||||||
case MachineRepresentation::kMapWord: // Fall through.
|
|
||||||
case MachineRepresentation::kNone:
|
|
||||||
UNREACHABLE();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
ArchOpcode GetAtomicStoreOpcode(MachineRepresentation rep) {
|
|
||||||
switch (rep) {
|
|
||||||
case MachineRepresentation::kWord8:
|
|
||||||
return kAtomicStoreWord8;
|
|
||||||
case MachineRepresentation::kWord16:
|
|
||||||
return kAtomicStoreWord16;
|
|
||||||
case MachineRepresentation::kTaggedSigned: // Fall through.
|
|
||||||
case MachineRepresentation::kTaggedPointer: // Fall through.
|
|
||||||
case MachineRepresentation::kTagged: // Fall through.
|
|
||||||
case MachineRepresentation::kWord32:
|
|
||||||
return kAtomicStoreWord32;
|
|
||||||
default:
|
|
||||||
UNREACHABLE();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void VisitStoreCommon(InstructionSelector* selector, Node* node,
|
|
||||||
StoreRepresentation store_rep,
|
|
||||||
base::Optional<AtomicMemoryOrder> atomic_order) {
|
|
||||||
ArmOperandGenerator g(selector);
|
|
||||||
Node* base = node->InputAt(0);
|
Node* base = node->InputAt(0);
|
||||||
Node* index = node->InputAt(1);
|
Node* index = node->InputAt(1);
|
||||||
Node* value = node->InputAt(2);
|
Node* value = node->InputAt(2);
|
||||||
|
|
||||||
|
StoreRepresentation store_rep = StoreRepresentationOf(node->op());
|
||||||
WriteBarrierKind write_barrier_kind = store_rep.write_barrier_kind();
|
WriteBarrierKind write_barrier_kind = store_rep.write_barrier_kind();
|
||||||
MachineRepresentation rep = store_rep.representation();
|
MachineRepresentation rep = store_rep.representation();
|
||||||
|
|
||||||
@ -720,44 +672,58 @@ void VisitStoreCommon(InstructionSelector* selector, Node* node,
|
|||||||
inputs[input_count++] = g.UseUniqueRegister(value);
|
inputs[input_count++] = g.UseUniqueRegister(value);
|
||||||
RecordWriteMode record_write_mode =
|
RecordWriteMode record_write_mode =
|
||||||
WriteBarrierKindToRecordWriteMode(write_barrier_kind);
|
WriteBarrierKindToRecordWriteMode(write_barrier_kind);
|
||||||
InstructionCode code;
|
InstructionCode code = kArchStoreWithWriteBarrier;
|
||||||
if (!atomic_order) {
|
|
||||||
code = kArchStoreWithWriteBarrier;
|
|
||||||
code |= MiscField::encode(static_cast<int>(record_write_mode));
|
|
||||||
} else {
|
|
||||||
code = kArchAtomicStoreWithWriteBarrier;
|
|
||||||
code |= AtomicMemoryOrderField::encode(*atomic_order);
|
|
||||||
code |= AtomicStoreRecordWriteModeField::encode(record_write_mode);
|
|
||||||
}
|
|
||||||
code |= AddressingModeField::encode(addressing_mode);
|
code |= AddressingModeField::encode(addressing_mode);
|
||||||
selector->Emit(code, 0, nullptr, input_count, inputs);
|
code |= MiscField::encode(static_cast<int>(record_write_mode));
|
||||||
|
Emit(code, 0, nullptr, input_count, inputs);
|
||||||
} else {
|
} else {
|
||||||
InstructionCode opcode = kArchNop;
|
InstructionCode opcode = kArchNop;
|
||||||
if (!atomic_order) {
|
switch (rep) {
|
||||||
opcode = GetStoreOpcode(rep);
|
case MachineRepresentation::kFloat32:
|
||||||
} else {
|
opcode = kArmVstrF32;
|
||||||
// Release stores emit DMB ISH; STR while sequentially consistent stores
|
break;
|
||||||
// emit DMB ISH; STR; DMB ISH.
|
case MachineRepresentation::kFloat64:
|
||||||
// https://www.cl.cam.ac.uk/~pes20/cpp/cpp0xmappings.html
|
opcode = kArmVstrF64;
|
||||||
opcode = GetAtomicStoreOpcode(rep);
|
break;
|
||||||
opcode |= AtomicMemoryOrderField::encode(*atomic_order);
|
case MachineRepresentation::kBit: // Fall through.
|
||||||
|
case MachineRepresentation::kWord8:
|
||||||
|
opcode = kArmStrb;
|
||||||
|
break;
|
||||||
|
case MachineRepresentation::kWord16:
|
||||||
|
opcode = kArmStrh;
|
||||||
|
break;
|
||||||
|
case MachineRepresentation::kTaggedSigned: // Fall through.
|
||||||
|
case MachineRepresentation::kTaggedPointer: // Fall through.
|
||||||
|
case MachineRepresentation::kTagged: // Fall through.
|
||||||
|
case MachineRepresentation::kWord32:
|
||||||
|
opcode = kArmStr;
|
||||||
|
break;
|
||||||
|
case MachineRepresentation::kSimd128:
|
||||||
|
opcode = kArmVst1S128;
|
||||||
|
break;
|
||||||
|
case MachineRepresentation::kCompressedPointer: // Fall through.
|
||||||
|
case MachineRepresentation::kCompressed: // Fall through.
|
||||||
|
case MachineRepresentation::kWord64: // Fall through.
|
||||||
|
case MachineRepresentation::kMapWord: // Fall through.
|
||||||
|
case MachineRepresentation::kNone:
|
||||||
|
UNREACHABLE();
|
||||||
}
|
}
|
||||||
|
|
||||||
ExternalReferenceMatcher m(base);
|
ExternalReferenceMatcher m(base);
|
||||||
if (m.HasResolvedValue() &&
|
if (m.HasResolvedValue() &&
|
||||||
selector->CanAddressRelativeToRootsRegister(m.ResolvedValue())) {
|
CanAddressRelativeToRootsRegister(m.ResolvedValue())) {
|
||||||
Int32Matcher int_matcher(index);
|
Int32Matcher int_matcher(index);
|
||||||
if (int_matcher.HasResolvedValue()) {
|
if (int_matcher.HasResolvedValue()) {
|
||||||
ptrdiff_t const delta =
|
ptrdiff_t const delta =
|
||||||
int_matcher.ResolvedValue() +
|
int_matcher.ResolvedValue() +
|
||||||
TurboAssemblerBase::RootRegisterOffsetForExternalReference(
|
TurboAssemblerBase::RootRegisterOffsetForExternalReference(
|
||||||
selector->isolate(), m.ResolvedValue());
|
isolate(), m.ResolvedValue());
|
||||||
int input_count = 2;
|
int input_count = 2;
|
||||||
InstructionOperand inputs[2];
|
InstructionOperand inputs[2];
|
||||||
inputs[0] = g.UseRegister(value);
|
inputs[0] = g.UseRegister(value);
|
||||||
inputs[1] = g.UseImmediate(static_cast<int32_t>(delta));
|
inputs[1] = g.UseImmediate(static_cast<int32_t>(delta));
|
||||||
opcode |= AddressingModeField::encode(kMode_Root);
|
opcode |= AddressingModeField::encode(kMode_Root);
|
||||||
selector->Emit(opcode, 0, nullptr, input_count, inputs);
|
Emit(opcode, 0, nullptr, input_count, inputs);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -766,17 +732,10 @@ void VisitStoreCommon(InstructionSelector* selector, Node* node,
|
|||||||
size_t input_count = 0;
|
size_t input_count = 0;
|
||||||
inputs[input_count++] = g.UseRegister(value);
|
inputs[input_count++] = g.UseRegister(value);
|
||||||
inputs[input_count++] = g.UseRegister(base);
|
inputs[input_count++] = g.UseRegister(base);
|
||||||
EmitStore(selector, opcode, input_count, inputs, index);
|
EmitStore(this, opcode, input_count, inputs, index);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
} // namespace
|
|
||||||
|
|
||||||
void InstructionSelector::VisitStore(Node* node) {
|
|
||||||
VisitStoreCommon(this, node, StoreRepresentationOf(node->op()),
|
|
||||||
base::nullopt);
|
|
||||||
}
|
|
||||||
|
|
||||||
void InstructionSelector::VisitProtectedStore(Node* node) {
|
void InstructionSelector::VisitProtectedStore(Node* node) {
|
||||||
// TODO(eholk)
|
// TODO(eholk)
|
||||||
UNIMPLEMENTED();
|
UNIMPLEMENTED();
|
||||||
@ -2271,11 +2230,7 @@ void InstructionSelector::VisitMemoryBarrier(Node* node) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
void InstructionSelector::VisitWord32AtomicLoad(Node* node) {
|
void InstructionSelector::VisitWord32AtomicLoad(Node* node) {
|
||||||
// The memory order is ignored as both acquire and sequentially consistent
|
LoadRepresentation load_rep = LoadRepresentationOf(node->op());
|
||||||
// loads can emit LDR; DMB ISH.
|
|
||||||
// https://www.cl.cam.ac.uk/~pes20/cpp/cpp0xmappings.html
|
|
||||||
AtomicLoadParameters atomic_load_params = AtomicLoadParametersOf(node->op());
|
|
||||||
LoadRepresentation load_rep = atomic_load_params.representation();
|
|
||||||
ArmOperandGenerator g(this);
|
ArmOperandGenerator g(this);
|
||||||
Node* base = node->InputAt(0);
|
Node* base = node->InputAt(0);
|
||||||
Node* index = node->InputAt(1);
|
Node* index = node->InputAt(1);
|
||||||
@ -2287,9 +2242,6 @@ void InstructionSelector::VisitWord32AtomicLoad(Node* node) {
|
|||||||
case MachineRepresentation::kWord16:
|
case MachineRepresentation::kWord16:
|
||||||
opcode = load_rep.IsSigned() ? kAtomicLoadInt16 : kAtomicLoadUint16;
|
opcode = load_rep.IsSigned() ? kAtomicLoadInt16 : kAtomicLoadUint16;
|
||||||
break;
|
break;
|
||||||
case MachineRepresentation::kTaggedSigned: // Fall through.
|
|
||||||
case MachineRepresentation::kTaggedPointer: // Fall through.
|
|
||||||
case MachineRepresentation::kTagged: // Fall through.
|
|
||||||
case MachineRepresentation::kWord32:
|
case MachineRepresentation::kWord32:
|
||||||
opcode = kAtomicLoadWord32;
|
opcode = kAtomicLoadWord32;
|
||||||
break;
|
break;
|
||||||
@ -2301,9 +2253,34 @@ void InstructionSelector::VisitWord32AtomicLoad(Node* node) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
void InstructionSelector::VisitWord32AtomicStore(Node* node) {
|
void InstructionSelector::VisitWord32AtomicStore(Node* node) {
|
||||||
AtomicStoreParameters store_params = AtomicStoreParametersOf(node->op());
|
MachineRepresentation rep = AtomicStoreRepresentationOf(node->op());
|
||||||
VisitStoreCommon(this, node, store_params.store_representation(),
|
ArmOperandGenerator g(this);
|
||||||
store_params.order());
|
Node* base = node->InputAt(0);
|
||||||
|
Node* index = node->InputAt(1);
|
||||||
|
Node* value = node->InputAt(2);
|
||||||
|
ArchOpcode opcode;
|
||||||
|
switch (rep) {
|
||||||
|
case MachineRepresentation::kWord8:
|
||||||
|
opcode = kAtomicStoreWord8;
|
||||||
|
break;
|
||||||
|
case MachineRepresentation::kWord16:
|
||||||
|
opcode = kAtomicStoreWord16;
|
||||||
|
break;
|
||||||
|
case MachineRepresentation::kWord32:
|
||||||
|
opcode = kAtomicStoreWord32;
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
UNREACHABLE();
|
||||||
|
}
|
||||||
|
|
||||||
|
AddressingMode addressing_mode = kMode_Offset_RR;
|
||||||
|
InstructionOperand inputs[4];
|
||||||
|
size_t input_count = 0;
|
||||||
|
inputs[input_count++] = g.UseUniqueRegister(base);
|
||||||
|
inputs[input_count++] = g.UseUniqueRegister(index);
|
||||||
|
inputs[input_count++] = g.UseUniqueRegister(value);
|
||||||
|
InstructionCode code = opcode | AddressingModeField::encode(addressing_mode);
|
||||||
|
Emit(code, 0, nullptr, input_count, inputs);
|
||||||
}
|
}
|
||||||
|
|
||||||
void InstructionSelector::VisitWord32AtomicExchange(Node* node) {
|
void InstructionSelector::VisitWord32AtomicExchange(Node* node) {
|
||||||
|
@ -969,25 +969,6 @@ CodeGenerator::CodeGenResult CodeGenerator::AssembleArchInstruction(
|
|||||||
__ Bind(ool->exit());
|
__ Bind(ool->exit());
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
case kArchAtomicStoreWithWriteBarrier: {
|
|
||||||
DCHECK_EQ(AddressingModeField::decode(instr->opcode()), kMode_MRR);
|
|
||||||
RecordWriteMode mode =
|
|
||||||
static_cast<RecordWriteMode>(MiscField::decode(instr->opcode()));
|
|
||||||
Register object = i.InputRegister(0);
|
|
||||||
Register offset = i.InputRegister(1);
|
|
||||||
Register value = i.InputRegister(2);
|
|
||||||
auto ool = zone()->New<OutOfLineRecordWrite>(
|
|
||||||
this, object, offset, value, mode, DetermineStubCallMode(),
|
|
||||||
&unwinding_info_writer_);
|
|
||||||
__ AtomicStoreTaggedField(value, object, offset, i.TempRegister(0));
|
|
||||||
if (mode > RecordWriteMode::kValueIsPointer) {
|
|
||||||
__ JumpIfSmi(value, ool->exit());
|
|
||||||
}
|
|
||||||
__ CheckPageFlag(object, MemoryChunk::kPointersFromHereAreInterestingMask,
|
|
||||||
eq, ool->entry());
|
|
||||||
__ Bind(ool->exit());
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
case kArchStackSlot: {
|
case kArchStackSlot: {
|
||||||
FrameOffset offset =
|
FrameOffset offset =
|
||||||
frame_access_state()->GetFrameOffset(i.InputInt32(0));
|
frame_access_state()->GetFrameOffset(i.InputInt32(0));
|
||||||
@ -1830,18 +1811,6 @@ CodeGenerator::CodeGenResult CodeGenerator::AssembleArchInstruction(
|
|||||||
case kArm64LdrDecompressAnyTagged:
|
case kArm64LdrDecompressAnyTagged:
|
||||||
__ DecompressAnyTagged(i.OutputRegister(), i.MemoryOperand());
|
__ DecompressAnyTagged(i.OutputRegister(), i.MemoryOperand());
|
||||||
break;
|
break;
|
||||||
case kArm64LdarDecompressTaggedSigned:
|
|
||||||
__ AtomicDecompressTaggedSigned(i.OutputRegister(), i.InputRegister(0),
|
|
||||||
i.InputRegister(1), i.TempRegister(0));
|
|
||||||
break;
|
|
||||||
case kArm64LdarDecompressTaggedPointer:
|
|
||||||
__ AtomicDecompressTaggedPointer(i.OutputRegister(), i.InputRegister(0),
|
|
||||||
i.InputRegister(1), i.TempRegister(0));
|
|
||||||
break;
|
|
||||||
case kArm64LdarDecompressAnyTagged:
|
|
||||||
__ AtomicDecompressAnyTagged(i.OutputRegister(), i.InputRegister(0),
|
|
||||||
i.InputRegister(1), i.TempRegister(0));
|
|
||||||
break;
|
|
||||||
case kArm64Str:
|
case kArm64Str:
|
||||||
EmitOOLTrapIfNeeded(zone(), this, opcode, instr, __ pc_offset());
|
EmitOOLTrapIfNeeded(zone(), this, opcode, instr, __ pc_offset());
|
||||||
__ Str(i.InputOrZeroRegister64(0), i.MemoryOperand(1));
|
__ Str(i.InputOrZeroRegister64(0), i.MemoryOperand(1));
|
||||||
@ -1849,12 +1818,6 @@ CodeGenerator::CodeGenResult CodeGenerator::AssembleArchInstruction(
|
|||||||
case kArm64StrCompressTagged:
|
case kArm64StrCompressTagged:
|
||||||
__ StoreTaggedField(i.InputOrZeroRegister64(0), i.MemoryOperand(1));
|
__ StoreTaggedField(i.InputOrZeroRegister64(0), i.MemoryOperand(1));
|
||||||
break;
|
break;
|
||||||
case kArm64StlrCompressTagged:
|
|
||||||
// To be consistent with other STLR instructions, the value is stored at
|
|
||||||
// the 3rd input register instead of the 1st.
|
|
||||||
__ AtomicStoreTaggedField(i.InputRegister(2), i.InputRegister(0),
|
|
||||||
i.InputRegister(1), i.TempRegister(0));
|
|
||||||
break;
|
|
||||||
case kArm64LdrS:
|
case kArm64LdrS:
|
||||||
EmitOOLTrapIfNeeded(zone(), this, opcode, instr, __ pc_offset());
|
EmitOOLTrapIfNeeded(zone(), this, opcode, instr, __ pc_offset());
|
||||||
__ Ldr(i.OutputDoubleRegister().S(), i.MemoryOperand());
|
__ Ldr(i.OutputDoubleRegister().S(), i.MemoryOperand());
|
||||||
|
@ -177,12 +177,8 @@ namespace compiler {
|
|||||||
V(Arm64LdrDecompressTaggedSigned) \
|
V(Arm64LdrDecompressTaggedSigned) \
|
||||||
V(Arm64LdrDecompressTaggedPointer) \
|
V(Arm64LdrDecompressTaggedPointer) \
|
||||||
V(Arm64LdrDecompressAnyTagged) \
|
V(Arm64LdrDecompressAnyTagged) \
|
||||||
V(Arm64LdarDecompressTaggedSigned) \
|
|
||||||
V(Arm64LdarDecompressTaggedPointer) \
|
|
||||||
V(Arm64LdarDecompressAnyTagged) \
|
|
||||||
V(Arm64Str) \
|
V(Arm64Str) \
|
||||||
V(Arm64StrCompressTagged) \
|
V(Arm64StrCompressTagged) \
|
||||||
V(Arm64StlrCompressTagged) \
|
|
||||||
V(Arm64DmbIsh) \
|
V(Arm64DmbIsh) \
|
||||||
V(Arm64DsbIsb) \
|
V(Arm64DsbIsb) \
|
||||||
V(Arm64Sxtl) \
|
V(Arm64Sxtl) \
|
||||||
|
@ -377,9 +377,6 @@ int InstructionScheduler::GetTargetInstructionFlags(
|
|||||||
case kArm64LdrDecompressTaggedSigned:
|
case kArm64LdrDecompressTaggedSigned:
|
||||||
case kArm64LdrDecompressTaggedPointer:
|
case kArm64LdrDecompressTaggedPointer:
|
||||||
case kArm64LdrDecompressAnyTagged:
|
case kArm64LdrDecompressAnyTagged:
|
||||||
case kArm64LdarDecompressTaggedSigned:
|
|
||||||
case kArm64LdarDecompressTaggedPointer:
|
|
||||||
case kArm64LdarDecompressAnyTagged:
|
|
||||||
case kArm64Peek:
|
case kArm64Peek:
|
||||||
case kArm64LoadSplat:
|
case kArm64LoadSplat:
|
||||||
case kArm64LoadLane:
|
case kArm64LoadLane:
|
||||||
@ -402,7 +399,6 @@ int InstructionScheduler::GetTargetInstructionFlags(
|
|||||||
case kArm64StrW:
|
case kArm64StrW:
|
||||||
case kArm64Str:
|
case kArm64Str:
|
||||||
case kArm64StrCompressTagged:
|
case kArm64StrCompressTagged:
|
||||||
case kArm64StlrCompressTagged:
|
|
||||||
case kArm64DmbIsh:
|
case kArm64DmbIsh:
|
||||||
case kArm64DsbIsb:
|
case kArm64DsbIsb:
|
||||||
case kArm64StoreLane:
|
case kArm64StoreLane:
|
||||||
|
@ -144,6 +144,21 @@ class Arm64OperandGenerator final : public OperandGenerator {
|
|||||||
|
|
||||||
namespace {
|
namespace {
|
||||||
|
|
||||||
|
ArchOpcode GetAtomicStoreOpcode(MachineRepresentation rep) {
|
||||||
|
switch (rep) {
|
||||||
|
case MachineRepresentation::kWord8:
|
||||||
|
return kAtomicStoreWord8;
|
||||||
|
case MachineRepresentation::kWord16:
|
||||||
|
return kAtomicStoreWord16;
|
||||||
|
case MachineRepresentation::kWord32:
|
||||||
|
return kAtomicStoreWord32;
|
||||||
|
case MachineRepresentation::kWord64:
|
||||||
|
return kArm64Word64AtomicStoreWord64;
|
||||||
|
default:
|
||||||
|
UNREACHABLE();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
void VisitRR(InstructionSelector* selector, ArchOpcode opcode, Node* node) {
|
void VisitRR(InstructionSelector* selector, ArchOpcode opcode, Node* node) {
|
||||||
Arm64OperandGenerator g(selector);
|
Arm64OperandGenerator g(selector);
|
||||||
selector->Emit(opcode, g.DefineAsRegister(node),
|
selector->Emit(opcode, g.DefineAsRegister(node),
|
||||||
@ -2603,135 +2618,30 @@ void VisitAtomicCompareExchange(InstructionSelector* selector, Node* node,
|
|||||||
}
|
}
|
||||||
|
|
||||||
void VisitAtomicLoad(InstructionSelector* selector, Node* node,
|
void VisitAtomicLoad(InstructionSelector* selector, Node* node,
|
||||||
AtomicWidth width) {
|
ArchOpcode opcode, AtomicWidth width) {
|
||||||
Arm64OperandGenerator g(selector);
|
Arm64OperandGenerator g(selector);
|
||||||
Node* base = node->InputAt(0);
|
Node* base = node->InputAt(0);
|
||||||
Node* index = node->InputAt(1);
|
Node* index = node->InputAt(1);
|
||||||
InstructionOperand inputs[] = {g.UseRegister(base), g.UseRegister(index)};
|
InstructionOperand inputs[] = {g.UseRegister(base), g.UseRegister(index)};
|
||||||
InstructionOperand outputs[] = {g.DefineAsRegister(node)};
|
InstructionOperand outputs[] = {g.DefineAsRegister(node)};
|
||||||
InstructionOperand temps[] = {g.TempRegister()};
|
InstructionOperand temps[] = {g.TempRegister()};
|
||||||
|
InstructionCode code = opcode | AddressingModeField::encode(kMode_MRR) |
|
||||||
// The memory order is ignored as both acquire and sequentially consistent
|
AtomicWidthField::encode(width);
|
||||||
// loads can emit LDAR.
|
|
||||||
// https://www.cl.cam.ac.uk/~pes20/cpp/cpp0xmappings.html
|
|
||||||
AtomicLoadParameters atomic_load_params = AtomicLoadParametersOf(node->op());
|
|
||||||
LoadRepresentation load_rep = atomic_load_params.representation();
|
|
||||||
InstructionCode code;
|
|
||||||
switch (load_rep.representation()) {
|
|
||||||
case MachineRepresentation::kWord8:
|
|
||||||
DCHECK_IMPLIES(load_rep.IsSigned(), width == AtomicWidth::kWord32);
|
|
||||||
code = load_rep.IsSigned() ? kAtomicLoadInt8 : kAtomicLoadUint8;
|
|
||||||
break;
|
|
||||||
case MachineRepresentation::kWord16:
|
|
||||||
DCHECK_IMPLIES(load_rep.IsSigned(), width == AtomicWidth::kWord32);
|
|
||||||
code = load_rep.IsSigned() ? kAtomicLoadInt16 : kAtomicLoadUint16;
|
|
||||||
break;
|
|
||||||
case MachineRepresentation::kWord32:
|
|
||||||
code = kAtomicLoadWord32;
|
|
||||||
break;
|
|
||||||
case MachineRepresentation::kWord64:
|
|
||||||
code = kArm64Word64AtomicLoadUint64;
|
|
||||||
break;
|
|
||||||
#ifdef V8_COMPRESS_POINTERS
|
|
||||||
case MachineRepresentation::kTaggedSigned:
|
|
||||||
code = kArm64LdarDecompressTaggedSigned;
|
|
||||||
break;
|
|
||||||
case MachineRepresentation::kTaggedPointer:
|
|
||||||
code = kArm64LdarDecompressTaggedPointer;
|
|
||||||
break;
|
|
||||||
case MachineRepresentation::kTagged:
|
|
||||||
code = kArm64LdarDecompressAnyTagged;
|
|
||||||
break;
|
|
||||||
#else
|
|
||||||
case MachineRepresentation::kTaggedSigned: // Fall through.
|
|
||||||
case MachineRepresentation::kTaggedPointer: // Fall through.
|
|
||||||
case MachineRepresentation::kTagged:
|
|
||||||
if (kTaggedSize == 8) {
|
|
||||||
code = kArm64Word64AtomicLoadUint64;
|
|
||||||
} else {
|
|
||||||
code = kAtomicLoadWord32;
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
#endif
|
|
||||||
case MachineRepresentation::kCompressedPointer: // Fall through.
|
|
||||||
case MachineRepresentation::kCompressed:
|
|
||||||
DCHECK(COMPRESS_POINTERS_BOOL);
|
|
||||||
code = kAtomicLoadWord32;
|
|
||||||
break;
|
|
||||||
default:
|
|
||||||
UNREACHABLE();
|
|
||||||
}
|
|
||||||
code |=
|
|
||||||
AddressingModeField::encode(kMode_MRR) | AtomicWidthField::encode(width);
|
|
||||||
selector->Emit(code, arraysize(outputs), outputs, arraysize(inputs), inputs,
|
selector->Emit(code, arraysize(outputs), outputs, arraysize(inputs), inputs,
|
||||||
arraysize(temps), temps);
|
arraysize(temps), temps);
|
||||||
}
|
}
|
||||||
|
|
||||||
void VisitAtomicStore(InstructionSelector* selector, Node* node,
|
void VisitAtomicStore(InstructionSelector* selector, Node* node,
|
||||||
AtomicWidth width) {
|
ArchOpcode opcode, AtomicWidth width) {
|
||||||
Arm64OperandGenerator g(selector);
|
Arm64OperandGenerator g(selector);
|
||||||
Node* base = node->InputAt(0);
|
Node* base = node->InputAt(0);
|
||||||
Node* index = node->InputAt(1);
|
Node* index = node->InputAt(1);
|
||||||
Node* value = node->InputAt(2);
|
Node* value = node->InputAt(2);
|
||||||
|
|
||||||
// The memory order is ignored as both release and sequentially consistent
|
|
||||||
// stores can emit STLR.
|
|
||||||
// https://www.cl.cam.ac.uk/~pes20/cpp/cpp0xmappings.html
|
|
||||||
AtomicStoreParameters store_params = AtomicStoreParametersOf(node->op());
|
|
||||||
WriteBarrierKind write_barrier_kind = store_params.write_barrier_kind();
|
|
||||||
MachineRepresentation rep = store_params.representation();
|
|
||||||
|
|
||||||
if (FLAG_enable_unconditional_write_barriers &&
|
|
||||||
CanBeTaggedOrCompressedPointer(rep)) {
|
|
||||||
write_barrier_kind = kFullWriteBarrier;
|
|
||||||
}
|
|
||||||
|
|
||||||
InstructionOperand inputs[] = {g.UseRegister(base), g.UseRegister(index),
|
InstructionOperand inputs[] = {g.UseRegister(base), g.UseRegister(index),
|
||||||
g.UseUniqueRegister(value)};
|
g.UseUniqueRegister(value)};
|
||||||
InstructionOperand temps[] = {g.TempRegister()};
|
InstructionOperand temps[] = {g.TempRegister()};
|
||||||
InstructionCode code;
|
InstructionCode code = opcode | AddressingModeField::encode(kMode_MRR) |
|
||||||
|
AtomicWidthField::encode(width);
|
||||||
if (write_barrier_kind != kNoWriteBarrier && !FLAG_disable_write_barriers) {
|
|
||||||
DCHECK(CanBeTaggedOrCompressedPointer(rep));
|
|
||||||
DCHECK_EQ(AtomicWidthSize(width), kTaggedSize);
|
|
||||||
|
|
||||||
RecordWriteMode record_write_mode =
|
|
||||||
WriteBarrierKindToRecordWriteMode(write_barrier_kind);
|
|
||||||
code = kArchAtomicStoreWithWriteBarrier;
|
|
||||||
code |= MiscField::encode(static_cast<int>(record_write_mode));
|
|
||||||
} else {
|
|
||||||
switch (rep) {
|
|
||||||
case MachineRepresentation::kWord8:
|
|
||||||
code = kAtomicStoreWord8;
|
|
||||||
break;
|
|
||||||
case MachineRepresentation::kWord16:
|
|
||||||
code = kAtomicStoreWord16;
|
|
||||||
break;
|
|
||||||
case MachineRepresentation::kWord32:
|
|
||||||
code = kAtomicStoreWord32;
|
|
||||||
break;
|
|
||||||
case MachineRepresentation::kWord64:
|
|
||||||
DCHECK_EQ(width, AtomicWidth::kWord64);
|
|
||||||
code = kArm64Word64AtomicStoreWord64;
|
|
||||||
break;
|
|
||||||
case MachineRepresentation::kTaggedSigned: // Fall through.
|
|
||||||
case MachineRepresentation::kTaggedPointer: // Fall through.
|
|
||||||
case MachineRepresentation::kTagged:
|
|
||||||
DCHECK_EQ(AtomicWidthSize(width), kTaggedSize);
|
|
||||||
V8_FALLTHROUGH;
|
|
||||||
case MachineRepresentation::kCompressedPointer: // Fall through.
|
|
||||||
case MachineRepresentation::kCompressed:
|
|
||||||
CHECK(COMPRESS_POINTERS_BOOL);
|
|
||||||
DCHECK_EQ(width, AtomicWidth::kWord32);
|
|
||||||
code = kArm64StlrCompressTagged;
|
|
||||||
break;
|
|
||||||
default:
|
|
||||||
UNREACHABLE();
|
|
||||||
}
|
|
||||||
code |= AtomicWidthField::encode(width);
|
|
||||||
}
|
|
||||||
|
|
||||||
code |= AddressingModeField::encode(kMode_MRR);
|
|
||||||
selector->Emit(code, 0, nullptr, arraysize(inputs), inputs, arraysize(temps),
|
selector->Emit(code, 0, nullptr, arraysize(inputs), inputs, arraysize(temps),
|
||||||
temps);
|
temps);
|
||||||
}
|
}
|
||||||
@ -3292,19 +3202,55 @@ void InstructionSelector::VisitMemoryBarrier(Node* node) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
void InstructionSelector::VisitWord32AtomicLoad(Node* node) {
|
void InstructionSelector::VisitWord32AtomicLoad(Node* node) {
|
||||||
VisitAtomicLoad(this, node, AtomicWidth::kWord32);
|
LoadRepresentation load_rep = LoadRepresentationOf(node->op());
|
||||||
|
ArchOpcode opcode;
|
||||||
|
switch (load_rep.representation()) {
|
||||||
|
case MachineRepresentation::kWord8:
|
||||||
|
opcode = load_rep.IsSigned() ? kAtomicLoadInt8 : kAtomicLoadUint8;
|
||||||
|
break;
|
||||||
|
case MachineRepresentation::kWord16:
|
||||||
|
opcode = load_rep.IsSigned() ? kAtomicLoadInt16 : kAtomicLoadUint16;
|
||||||
|
break;
|
||||||
|
case MachineRepresentation::kWord32:
|
||||||
|
opcode = kAtomicLoadWord32;
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
UNREACHABLE();
|
||||||
|
}
|
||||||
|
VisitAtomicLoad(this, node, opcode, AtomicWidth::kWord32);
|
||||||
}
|
}
|
||||||
|
|
||||||
void InstructionSelector::VisitWord64AtomicLoad(Node* node) {
|
void InstructionSelector::VisitWord64AtomicLoad(Node* node) {
|
||||||
VisitAtomicLoad(this, node, AtomicWidth::kWord64);
|
LoadRepresentation load_rep = LoadRepresentationOf(node->op());
|
||||||
|
ArchOpcode opcode;
|
||||||
|
switch (load_rep.representation()) {
|
||||||
|
case MachineRepresentation::kWord8:
|
||||||
|
opcode = kAtomicLoadUint8;
|
||||||
|
break;
|
||||||
|
case MachineRepresentation::kWord16:
|
||||||
|
opcode = kAtomicLoadUint16;
|
||||||
|
break;
|
||||||
|
case MachineRepresentation::kWord32:
|
||||||
|
opcode = kAtomicLoadWord32;
|
||||||
|
break;
|
||||||
|
case MachineRepresentation::kWord64:
|
||||||
|
opcode = kArm64Word64AtomicLoadUint64;
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
UNREACHABLE();
|
||||||
|
}
|
||||||
|
VisitAtomicLoad(this, node, opcode, AtomicWidth::kWord64);
|
||||||
}
|
}
|
||||||
|
|
||||||
void InstructionSelector::VisitWord32AtomicStore(Node* node) {
|
void InstructionSelector::VisitWord32AtomicStore(Node* node) {
|
||||||
VisitAtomicStore(this, node, AtomicWidth::kWord32);
|
MachineRepresentation rep = AtomicStoreRepresentationOf(node->op());
|
||||||
|
DCHECK_NE(rep, MachineRepresentation::kWord64);
|
||||||
|
VisitAtomicStore(this, node, GetAtomicStoreOpcode(rep), AtomicWidth::kWord32);
|
||||||
}
|
}
|
||||||
|
|
||||||
void InstructionSelector::VisitWord64AtomicStore(Node* node) {
|
void InstructionSelector::VisitWord64AtomicStore(Node* node) {
|
||||||
VisitAtomicStore(this, node, AtomicWidth::kWord64);
|
MachineRepresentation rep = AtomicStoreRepresentationOf(node->op());
|
||||||
|
VisitAtomicStore(this, node, GetAtomicStoreOpcode(rep), AtomicWidth::kWord64);
|
||||||
}
|
}
|
||||||
|
|
||||||
void InstructionSelector::VisitWord32AtomicExchange(Node* node) {
|
void InstructionSelector::VisitWord32AtomicExchange(Node* node) {
|
||||||
|
@ -957,8 +957,7 @@ CodeGenerator::CodeGenResult CodeGenerator::AssembleArchInstruction(
|
|||||||
__ bind(ool->exit());
|
__ bind(ool->exit());
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
case kArchStoreWithWriteBarrier: // Fall thrugh.
|
case kArchStoreWithWriteBarrier: {
|
||||||
case kArchAtomicStoreWithWriteBarrier: {
|
|
||||||
RecordWriteMode mode =
|
RecordWriteMode mode =
|
||||||
static_cast<RecordWriteMode>(MiscField::decode(instr->opcode()));
|
static_cast<RecordWriteMode>(MiscField::decode(instr->opcode()));
|
||||||
Register object = i.InputRegister(0);
|
Register object = i.InputRegister(0);
|
||||||
@ -970,12 +969,7 @@ CodeGenerator::CodeGenResult CodeGenerator::AssembleArchInstruction(
|
|||||||
auto ool = zone()->New<OutOfLineRecordWrite>(this, object, operand, value,
|
auto ool = zone()->New<OutOfLineRecordWrite>(this, object, operand, value,
|
||||||
scratch0, scratch1, mode,
|
scratch0, scratch1, mode,
|
||||||
DetermineStubCallMode());
|
DetermineStubCallMode());
|
||||||
if (arch_opcode == kArchStoreWithWriteBarrier) {
|
|
||||||
__ mov(operand, value);
|
__ mov(operand, value);
|
||||||
} else {
|
|
||||||
__ mov(scratch0, value);
|
|
||||||
__ xchg(scratch0, operand);
|
|
||||||
}
|
|
||||||
if (mode > RecordWriteMode::kValueIsPointer) {
|
if (mode > RecordWriteMode::kValueIsPointer) {
|
||||||
__ JumpIfSmi(value, ool->exit());
|
__ JumpIfSmi(value, ool->exit());
|
||||||
}
|
}
|
||||||
@ -3841,31 +3835,17 @@ CodeGenerator::CodeGenResult CodeGenerator::AssembleArchInstruction(
|
|||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
case kIA32Word32AtomicPairLoad: {
|
case kIA32Word32AtomicPairLoad: {
|
||||||
__ movq(kScratchDoubleReg, i.MemoryOperand());
|
XMMRegister tmp = i.ToDoubleRegister(instr->TempAt(0));
|
||||||
__ Pextrd(i.OutputRegister(0), kScratchDoubleReg, 0);
|
__ movq(tmp, i.MemoryOperand());
|
||||||
__ Pextrd(i.OutputRegister(1), kScratchDoubleReg, 1);
|
__ Pextrd(i.OutputRegister(0), tmp, 0);
|
||||||
|
__ Pextrd(i.OutputRegister(1), tmp, 1);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
case kIA32Word32ReleasePairStore: {
|
case kIA32Word32AtomicPairStore: {
|
||||||
__ push(ebx);
|
|
||||||
i.MoveInstructionOperandToRegister(ebx, instr->InputAt(1));
|
|
||||||
__ push(ebx);
|
|
||||||
i.MoveInstructionOperandToRegister(ebx, instr->InputAt(0));
|
|
||||||
__ push(ebx);
|
|
||||||
frame_access_state()->IncreaseSPDelta(3);
|
|
||||||
__ movq(kScratchDoubleReg, MemOperand(esp, 0));
|
|
||||||
__ pop(ebx);
|
|
||||||
__ pop(ebx);
|
|
||||||
__ pop(ebx);
|
|
||||||
frame_access_state()->IncreaseSPDelta(-3);
|
|
||||||
__ movq(i.MemoryOperand(2), kScratchDoubleReg);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
case kIA32Word32SeqCstPairStore: {
|
|
||||||
Label store;
|
Label store;
|
||||||
__ bind(&store);
|
__ bind(&store);
|
||||||
__ mov(eax, i.MemoryOperand(2));
|
__ mov(i.TempRegister(0), i.MemoryOperand(2));
|
||||||
__ mov(edx, i.NextMemoryOperand(2));
|
__ mov(i.TempRegister(1), i.NextMemoryOperand(2));
|
||||||
__ push(ebx);
|
__ push(ebx);
|
||||||
frame_access_state()->IncreaseSPDelta(1);
|
frame_access_state()->IncreaseSPDelta(1);
|
||||||
i.MoveInstructionOperandToRegister(ebx, instr->InputAt(0));
|
i.MoveInstructionOperandToRegister(ebx, instr->InputAt(0));
|
||||||
|
@ -402,8 +402,7 @@ namespace compiler {
|
|||||||
V(IA32I16x8AllTrue) \
|
V(IA32I16x8AllTrue) \
|
||||||
V(IA32I8x16AllTrue) \
|
V(IA32I8x16AllTrue) \
|
||||||
V(IA32Word32AtomicPairLoad) \
|
V(IA32Word32AtomicPairLoad) \
|
||||||
V(IA32Word32ReleasePairStore) \
|
V(IA32Word32AtomicPairStore) \
|
||||||
V(IA32Word32SeqCstPairStore) \
|
|
||||||
V(IA32Word32AtomicPairAdd) \
|
V(IA32Word32AtomicPairAdd) \
|
||||||
V(IA32Word32AtomicPairSub) \
|
V(IA32Word32AtomicPairSub) \
|
||||||
V(IA32Word32AtomicPairAnd) \
|
V(IA32Word32AtomicPairAnd) \
|
||||||
|
@ -423,8 +423,7 @@ int InstructionScheduler::GetTargetInstructionFlags(
|
|||||||
case kIA32Word32AtomicPairLoad:
|
case kIA32Word32AtomicPairLoad:
|
||||||
return kIsLoadOperation;
|
return kIsLoadOperation;
|
||||||
|
|
||||||
case kIA32Word32ReleasePairStore:
|
case kIA32Word32AtomicPairStore:
|
||||||
case kIA32Word32SeqCstPairStore:
|
|
||||||
case kIA32Word32AtomicPairAdd:
|
case kIA32Word32AtomicPairAdd:
|
||||||
case kIA32Word32AtomicPairSub:
|
case kIA32Word32AtomicPairSub:
|
||||||
case kIA32Word32AtomicPairAnd:
|
case kIA32Word32AtomicPairAnd:
|
||||||
|
@ -246,41 +246,6 @@ class IA32OperandGenerator final : public OperandGenerator {
|
|||||||
|
|
||||||
namespace {
|
namespace {
|
||||||
|
|
||||||
ArchOpcode GetLoadOpcode(LoadRepresentation load_rep) {
|
|
||||||
ArchOpcode opcode;
|
|
||||||
switch (load_rep.representation()) {
|
|
||||||
case MachineRepresentation::kFloat32:
|
|
||||||
opcode = kIA32Movss;
|
|
||||||
break;
|
|
||||||
case MachineRepresentation::kFloat64:
|
|
||||||
opcode = kIA32Movsd;
|
|
||||||
break;
|
|
||||||
case MachineRepresentation::kBit: // Fall through.
|
|
||||||
case MachineRepresentation::kWord8:
|
|
||||||
opcode = load_rep.IsSigned() ? kIA32Movsxbl : kIA32Movzxbl;
|
|
||||||
break;
|
|
||||||
case MachineRepresentation::kWord16:
|
|
||||||
opcode = load_rep.IsSigned() ? kIA32Movsxwl : kIA32Movzxwl;
|
|
||||||
break;
|
|
||||||
case MachineRepresentation::kTaggedSigned: // Fall through.
|
|
||||||
case MachineRepresentation::kTaggedPointer: // Fall through.
|
|
||||||
case MachineRepresentation::kTagged: // Fall through.
|
|
||||||
case MachineRepresentation::kWord32:
|
|
||||||
opcode = kIA32Movl;
|
|
||||||
break;
|
|
||||||
case MachineRepresentation::kSimd128:
|
|
||||||
opcode = kIA32Movdqu;
|
|
||||||
break;
|
|
||||||
case MachineRepresentation::kCompressedPointer: // Fall through.
|
|
||||||
case MachineRepresentation::kCompressed: // Fall through.
|
|
||||||
case MachineRepresentation::kWord64: // Fall through.
|
|
||||||
case MachineRepresentation::kMapWord: // Fall through.
|
|
||||||
case MachineRepresentation::kNone:
|
|
||||||
UNREACHABLE();
|
|
||||||
}
|
|
||||||
return opcode;
|
|
||||||
}
|
|
||||||
|
|
||||||
void VisitRO(InstructionSelector* selector, Node* node, ArchOpcode opcode) {
|
void VisitRO(InstructionSelector* selector, Node* node, ArchOpcode opcode) {
|
||||||
IA32OperandGenerator g(selector);
|
IA32OperandGenerator g(selector);
|
||||||
Node* input = node->InputAt(0);
|
Node* input = node->InputAt(0);
|
||||||
@ -570,8 +535,41 @@ void InstructionSelector::VisitLoadTransform(Node* node) {
|
|||||||
Emit(code, 1, outputs, input_count, inputs);
|
Emit(code, 1, outputs, input_count, inputs);
|
||||||
}
|
}
|
||||||
|
|
||||||
void InstructionSelector::VisitLoad(Node* node, Node* value,
|
void InstructionSelector::VisitLoad(Node* node) {
|
||||||
InstructionCode opcode) {
|
LoadRepresentation load_rep = LoadRepresentationOf(node->op());
|
||||||
|
|
||||||
|
ArchOpcode opcode;
|
||||||
|
switch (load_rep.representation()) {
|
||||||
|
case MachineRepresentation::kFloat32:
|
||||||
|
opcode = kIA32Movss;
|
||||||
|
break;
|
||||||
|
case MachineRepresentation::kFloat64:
|
||||||
|
opcode = kIA32Movsd;
|
||||||
|
break;
|
||||||
|
case MachineRepresentation::kBit: // Fall through.
|
||||||
|
case MachineRepresentation::kWord8:
|
||||||
|
opcode = load_rep.IsSigned() ? kIA32Movsxbl : kIA32Movzxbl;
|
||||||
|
break;
|
||||||
|
case MachineRepresentation::kWord16:
|
||||||
|
opcode = load_rep.IsSigned() ? kIA32Movsxwl : kIA32Movzxwl;
|
||||||
|
break;
|
||||||
|
case MachineRepresentation::kTaggedSigned: // Fall through.
|
||||||
|
case MachineRepresentation::kTaggedPointer: // Fall through.
|
||||||
|
case MachineRepresentation::kTagged: // Fall through.
|
||||||
|
case MachineRepresentation::kWord32:
|
||||||
|
opcode = kIA32Movl;
|
||||||
|
break;
|
||||||
|
case MachineRepresentation::kSimd128:
|
||||||
|
opcode = kIA32Movdqu;
|
||||||
|
break;
|
||||||
|
case MachineRepresentation::kCompressedPointer: // Fall through.
|
||||||
|
case MachineRepresentation::kCompressed: // Fall through.
|
||||||
|
case MachineRepresentation::kWord64: // Fall through.
|
||||||
|
case MachineRepresentation::kNone:
|
||||||
|
case MachineRepresentation::kMapWord:
|
||||||
|
UNREACHABLE();
|
||||||
|
}
|
||||||
|
|
||||||
IA32OperandGenerator g(this);
|
IA32OperandGenerator g(this);
|
||||||
InstructionOperand outputs[1];
|
InstructionOperand outputs[1];
|
||||||
outputs[0] = g.DefineAsRegister(node);
|
outputs[0] = g.DefineAsRegister(node);
|
||||||
@ -583,97 +581,20 @@ void InstructionSelector::VisitLoad(Node* node, Node* value,
|
|||||||
Emit(code, 1, outputs, input_count, inputs);
|
Emit(code, 1, outputs, input_count, inputs);
|
||||||
}
|
}
|
||||||
|
|
||||||
void InstructionSelector::VisitLoad(Node* node) {
|
|
||||||
LoadRepresentation load_rep = LoadRepresentationOf(node->op());
|
|
||||||
DCHECK(!load_rep.IsMapWord());
|
|
||||||
VisitLoad(node, node, GetLoadOpcode(load_rep));
|
|
||||||
}
|
|
||||||
|
|
||||||
void InstructionSelector::VisitProtectedLoad(Node* node) {
|
void InstructionSelector::VisitProtectedLoad(Node* node) {
|
||||||
// TODO(eholk)
|
// TODO(eholk)
|
||||||
UNIMPLEMENTED();
|
UNIMPLEMENTED();
|
||||||
}
|
}
|
||||||
|
|
||||||
namespace {
|
void InstructionSelector::VisitStore(Node* node) {
|
||||||
|
IA32OperandGenerator g(this);
|
||||||
ArchOpcode GetStoreOpcode(MachineRepresentation rep) {
|
|
||||||
switch (rep) {
|
|
||||||
case MachineRepresentation::kFloat32:
|
|
||||||
return kIA32Movss;
|
|
||||||
case MachineRepresentation::kFloat64:
|
|
||||||
return kIA32Movsd;
|
|
||||||
case MachineRepresentation::kBit: // Fall through.
|
|
||||||
case MachineRepresentation::kWord8:
|
|
||||||
return kIA32Movb;
|
|
||||||
case MachineRepresentation::kWord16:
|
|
||||||
return kIA32Movw;
|
|
||||||
case MachineRepresentation::kTaggedSigned: // Fall through.
|
|
||||||
case MachineRepresentation::kTaggedPointer: // Fall through.
|
|
||||||
case MachineRepresentation::kTagged: // Fall through.
|
|
||||||
case MachineRepresentation::kWord32:
|
|
||||||
return kIA32Movl;
|
|
||||||
case MachineRepresentation::kSimd128:
|
|
||||||
return kIA32Movdqu;
|
|
||||||
case MachineRepresentation::kCompressedPointer: // Fall through.
|
|
||||||
case MachineRepresentation::kCompressed: // Fall through.
|
|
||||||
case MachineRepresentation::kWord64: // Fall through.
|
|
||||||
case MachineRepresentation::kMapWord: // Fall through.
|
|
||||||
case MachineRepresentation::kNone:
|
|
||||||
UNREACHABLE();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
ArchOpcode GetSeqCstStoreOpcode(MachineRepresentation rep) {
|
|
||||||
switch (rep) {
|
|
||||||
case MachineRepresentation::kWord8:
|
|
||||||
return kAtomicExchangeInt8;
|
|
||||||
case MachineRepresentation::kWord16:
|
|
||||||
return kAtomicExchangeInt16;
|
|
||||||
case MachineRepresentation::kTaggedSigned: // Fall through.
|
|
||||||
case MachineRepresentation::kTaggedPointer: // Fall through.
|
|
||||||
case MachineRepresentation::kTagged: // Fall through.
|
|
||||||
case MachineRepresentation::kWord32:
|
|
||||||
return kAtomicExchangeWord32;
|
|
||||||
default:
|
|
||||||
UNREACHABLE();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void VisitAtomicExchange(InstructionSelector* selector, Node* node,
|
|
||||||
ArchOpcode opcode, MachineRepresentation rep) {
|
|
||||||
IA32OperandGenerator g(selector);
|
|
||||||
Node* base = node->InputAt(0);
|
|
||||||
Node* index = node->InputAt(1);
|
|
||||||
Node* value = node->InputAt(2);
|
|
||||||
|
|
||||||
AddressingMode addressing_mode;
|
|
||||||
InstructionOperand value_operand = (rep == MachineRepresentation::kWord8)
|
|
||||||
? g.UseFixed(value, edx)
|
|
||||||
: g.UseUniqueRegister(value);
|
|
||||||
InstructionOperand inputs[] = {
|
|
||||||
value_operand, g.UseUniqueRegister(base),
|
|
||||||
g.GetEffectiveIndexOperand(index, &addressing_mode)};
|
|
||||||
InstructionOperand outputs[] = {
|
|
||||||
(rep == MachineRepresentation::kWord8)
|
|
||||||
// Using DefineSameAsFirst requires the register to be unallocated.
|
|
||||||
? g.DefineAsFixed(node, edx)
|
|
||||||
: g.DefineSameAsFirst(node)};
|
|
||||||
InstructionCode code = opcode | AddressingModeField::encode(addressing_mode);
|
|
||||||
selector->Emit(code, 1, outputs, arraysize(inputs), inputs);
|
|
||||||
}
|
|
||||||
|
|
||||||
void VisitStoreCommon(InstructionSelector* selector, Node* node,
|
|
||||||
StoreRepresentation store_rep,
|
|
||||||
base::Optional<AtomicMemoryOrder> atomic_order) {
|
|
||||||
IA32OperandGenerator g(selector);
|
|
||||||
Node* base = node->InputAt(0);
|
Node* base = node->InputAt(0);
|
||||||
Node* index = node->InputAt(1);
|
Node* index = node->InputAt(1);
|
||||||
Node* value = node->InputAt(2);
|
Node* value = node->InputAt(2);
|
||||||
|
|
||||||
|
StoreRepresentation store_rep = StoreRepresentationOf(node->op());
|
||||||
WriteBarrierKind write_barrier_kind = store_rep.write_barrier_kind();
|
WriteBarrierKind write_barrier_kind = store_rep.write_barrier_kind();
|
||||||
MachineRepresentation rep = store_rep.representation();
|
MachineRepresentation rep = store_rep.representation();
|
||||||
const bool is_seqcst =
|
|
||||||
atomic_order && *atomic_order == AtomicMemoryOrder::kSeqCst;
|
|
||||||
|
|
||||||
if (FLAG_enable_unconditional_write_barriers && CanBeTaggedPointer(rep)) {
|
if (FLAG_enable_unconditional_write_barriers && CanBeTaggedPointer(rep)) {
|
||||||
write_barrier_kind = kFullWriteBarrier;
|
write_barrier_kind = kFullWriteBarrier;
|
||||||
@ -690,23 +611,48 @@ void VisitStoreCommon(InstructionSelector* selector, Node* node,
|
|||||||
WriteBarrierKindToRecordWriteMode(write_barrier_kind);
|
WriteBarrierKindToRecordWriteMode(write_barrier_kind);
|
||||||
InstructionOperand temps[] = {g.TempRegister(), g.TempRegister()};
|
InstructionOperand temps[] = {g.TempRegister(), g.TempRegister()};
|
||||||
size_t const temp_count = arraysize(temps);
|
size_t const temp_count = arraysize(temps);
|
||||||
InstructionCode code = is_seqcst ? kArchAtomicStoreWithWriteBarrier
|
InstructionCode code = kArchStoreWithWriteBarrier;
|
||||||
: kArchStoreWithWriteBarrier;
|
|
||||||
code |= AddressingModeField::encode(addressing_mode);
|
code |= AddressingModeField::encode(addressing_mode);
|
||||||
code |= MiscField::encode(static_cast<int>(record_write_mode));
|
code |= MiscField::encode(static_cast<int>(record_write_mode));
|
||||||
selector->Emit(code, 0, nullptr, arraysize(inputs), inputs, temp_count,
|
Emit(code, 0, nullptr, arraysize(inputs), inputs, temp_count, temps);
|
||||||
temps);
|
|
||||||
} else if (is_seqcst) {
|
|
||||||
VisitAtomicExchange(selector, node, GetSeqCstStoreOpcode(rep), rep);
|
|
||||||
} else {
|
} else {
|
||||||
// Release and non-atomic stores emit MOV.
|
ArchOpcode opcode;
|
||||||
// https://www.cl.cam.ac.uk/~pes20/cpp/cpp0xmappings.html
|
switch (rep) {
|
||||||
|
case MachineRepresentation::kFloat32:
|
||||||
|
opcode = kIA32Movss;
|
||||||
|
break;
|
||||||
|
case MachineRepresentation::kFloat64:
|
||||||
|
opcode = kIA32Movsd;
|
||||||
|
break;
|
||||||
|
case MachineRepresentation::kBit: // Fall through.
|
||||||
|
case MachineRepresentation::kWord8:
|
||||||
|
opcode = kIA32Movb;
|
||||||
|
break;
|
||||||
|
case MachineRepresentation::kWord16:
|
||||||
|
opcode = kIA32Movw;
|
||||||
|
break;
|
||||||
|
case MachineRepresentation::kTaggedSigned: // Fall through.
|
||||||
|
case MachineRepresentation::kTaggedPointer: // Fall through.
|
||||||
|
case MachineRepresentation::kTagged: // Fall through.
|
||||||
|
case MachineRepresentation::kWord32:
|
||||||
|
opcode = kIA32Movl;
|
||||||
|
break;
|
||||||
|
case MachineRepresentation::kSimd128:
|
||||||
|
opcode = kIA32Movdqu;
|
||||||
|
break;
|
||||||
|
case MachineRepresentation::kCompressedPointer: // Fall through.
|
||||||
|
case MachineRepresentation::kCompressed: // Fall through.
|
||||||
|
case MachineRepresentation::kWord64: // Fall through.
|
||||||
|
case MachineRepresentation::kMapWord: // Fall through.
|
||||||
|
case MachineRepresentation::kNone:
|
||||||
|
UNREACHABLE();
|
||||||
|
}
|
||||||
|
|
||||||
InstructionOperand val;
|
InstructionOperand val;
|
||||||
if (g.CanBeImmediate(value)) {
|
if (g.CanBeImmediate(value)) {
|
||||||
val = g.UseImmediate(value);
|
val = g.UseImmediate(value);
|
||||||
} else if (!atomic_order && (rep == MachineRepresentation::kWord8 ||
|
} else if (rep == MachineRepresentation::kWord8 ||
|
||||||
rep == MachineRepresentation::kBit)) {
|
rep == MachineRepresentation::kBit) {
|
||||||
val = g.UseByteRegister(value);
|
val = g.UseByteRegister(value);
|
||||||
} else {
|
} else {
|
||||||
val = g.UseRegister(value);
|
val = g.UseRegister(value);
|
||||||
@ -717,20 +663,13 @@ void VisitStoreCommon(InstructionSelector* selector, Node* node,
|
|||||||
AddressingMode addressing_mode =
|
AddressingMode addressing_mode =
|
||||||
g.GetEffectiveAddressMemoryOperand(node, inputs, &input_count);
|
g.GetEffectiveAddressMemoryOperand(node, inputs, &input_count);
|
||||||
InstructionCode code =
|
InstructionCode code =
|
||||||
GetStoreOpcode(rep) | AddressingModeField::encode(addressing_mode);
|
opcode | AddressingModeField::encode(addressing_mode);
|
||||||
inputs[input_count++] = val;
|
inputs[input_count++] = val;
|
||||||
selector->Emit(code, 0, static_cast<InstructionOperand*>(nullptr),
|
Emit(code, 0, static_cast<InstructionOperand*>(nullptr), input_count,
|
||||||
input_count, inputs);
|
inputs);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
} // namespace
|
|
||||||
|
|
||||||
void InstructionSelector::VisitStore(Node* node) {
|
|
||||||
VisitStoreCommon(this, node, StoreRepresentationOf(node->op()),
|
|
||||||
base::nullopt);
|
|
||||||
}
|
|
||||||
|
|
||||||
void InstructionSelector::VisitProtectedStore(Node* node) {
|
void InstructionSelector::VisitProtectedStore(Node* node) {
|
||||||
// TODO(eholk)
|
// TODO(eholk)
|
||||||
UNIMPLEMENTED();
|
UNIMPLEMENTED();
|
||||||
@ -1686,6 +1625,29 @@ void VisitWordCompare(InstructionSelector* selector, Node* node,
|
|||||||
VisitWordCompare(selector, node, kIA32Cmp, cont);
|
VisitWordCompare(selector, node, kIA32Cmp, cont);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void VisitAtomicExchange(InstructionSelector* selector, Node* node,
|
||||||
|
ArchOpcode opcode, MachineRepresentation rep) {
|
||||||
|
IA32OperandGenerator g(selector);
|
||||||
|
Node* base = node->InputAt(0);
|
||||||
|
Node* index = node->InputAt(1);
|
||||||
|
Node* value = node->InputAt(2);
|
||||||
|
|
||||||
|
AddressingMode addressing_mode;
|
||||||
|
InstructionOperand value_operand = (rep == MachineRepresentation::kWord8)
|
||||||
|
? g.UseFixed(value, edx)
|
||||||
|
: g.UseUniqueRegister(value);
|
||||||
|
InstructionOperand inputs[] = {
|
||||||
|
value_operand, g.UseUniqueRegister(base),
|
||||||
|
g.GetEffectiveIndexOperand(index, &addressing_mode)};
|
||||||
|
InstructionOperand outputs[] = {
|
||||||
|
(rep == MachineRepresentation::kWord8)
|
||||||
|
// Using DefineSameAsFirst requires the register to be unallocated.
|
||||||
|
? g.DefineAsFixed(node, edx)
|
||||||
|
: g.DefineSameAsFirst(node)};
|
||||||
|
InstructionCode code = opcode | AddressingModeField::encode(addressing_mode);
|
||||||
|
selector->Emit(code, 1, outputs, arraysize(inputs), inputs);
|
||||||
|
}
|
||||||
|
|
||||||
void VisitAtomicBinOp(InstructionSelector* selector, Node* node,
|
void VisitAtomicBinOp(InstructionSelector* selector, Node* node,
|
||||||
ArchOpcode opcode, MachineRepresentation rep) {
|
ArchOpcode opcode, MachineRepresentation rep) {
|
||||||
AddressingMode addressing_mode;
|
AddressingMode addressing_mode;
|
||||||
@ -1995,25 +1957,32 @@ void InstructionSelector::VisitMemoryBarrier(Node* node) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
void InstructionSelector::VisitWord32AtomicLoad(Node* node) {
|
void InstructionSelector::VisitWord32AtomicLoad(Node* node) {
|
||||||
AtomicLoadParameters atomic_load_params = AtomicLoadParametersOf(node->op());
|
LoadRepresentation load_rep = LoadRepresentationOf(node->op());
|
||||||
LoadRepresentation load_rep = atomic_load_params.representation();
|
|
||||||
DCHECK(load_rep.representation() == MachineRepresentation::kWord8 ||
|
DCHECK(load_rep.representation() == MachineRepresentation::kWord8 ||
|
||||||
load_rep.representation() == MachineRepresentation::kWord16 ||
|
load_rep.representation() == MachineRepresentation::kWord16 ||
|
||||||
load_rep.representation() == MachineRepresentation::kWord32 ||
|
load_rep.representation() == MachineRepresentation::kWord32);
|
||||||
load_rep.representation() == MachineRepresentation::kTaggedSigned ||
|
|
||||||
load_rep.representation() == MachineRepresentation::kTaggedPointer ||
|
|
||||||
load_rep.representation() == MachineRepresentation::kTagged);
|
|
||||||
USE(load_rep);
|
USE(load_rep);
|
||||||
// The memory order is ignored as both acquire and sequentially consistent
|
VisitLoad(node);
|
||||||
// loads can emit MOV.
|
|
||||||
// https://www.cl.cam.ac.uk/~pes20/cpp/cpp0xmappings.html
|
|
||||||
VisitLoad(node, node, GetLoadOpcode(load_rep));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void InstructionSelector::VisitWord32AtomicStore(Node* node) {
|
void InstructionSelector::VisitWord32AtomicStore(Node* node) {
|
||||||
AtomicStoreParameters store_params = AtomicStoreParametersOf(node->op());
|
IA32OperandGenerator g(this);
|
||||||
VisitStoreCommon(this, node, store_params.store_representation(),
|
MachineRepresentation rep = AtomicStoreRepresentationOf(node->op());
|
||||||
store_params.order());
|
ArchOpcode opcode;
|
||||||
|
switch (rep) {
|
||||||
|
case MachineRepresentation::kWord8:
|
||||||
|
opcode = kAtomicExchangeInt8;
|
||||||
|
break;
|
||||||
|
case MachineRepresentation::kWord16:
|
||||||
|
opcode = kAtomicExchangeInt16;
|
||||||
|
break;
|
||||||
|
case MachineRepresentation::kWord32:
|
||||||
|
opcode = kAtomicExchangeWord32;
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
UNREACHABLE();
|
||||||
|
}
|
||||||
|
VisitAtomicExchange(this, node, opcode, rep);
|
||||||
}
|
}
|
||||||
|
|
||||||
void InstructionSelector::VisitWord32AtomicExchange(Node* node) {
|
void InstructionSelector::VisitWord32AtomicExchange(Node* node) {
|
||||||
@ -2106,8 +2075,6 @@ VISIT_ATOMIC_BINOP(Xor)
|
|||||||
#undef VISIT_ATOMIC_BINOP
|
#undef VISIT_ATOMIC_BINOP
|
||||||
|
|
||||||
void InstructionSelector::VisitWord32AtomicPairLoad(Node* node) {
|
void InstructionSelector::VisitWord32AtomicPairLoad(Node* node) {
|
||||||
// Both acquire and sequentially consistent loads can emit MOV.
|
|
||||||
// https://www.cl.cam.ac.uk/~pes20/cpp/cpp0xmappings.html
|
|
||||||
IA32OperandGenerator g(this);
|
IA32OperandGenerator g(this);
|
||||||
AddressingMode mode;
|
AddressingMode mode;
|
||||||
Node* base = node->InputAt(0);
|
Node* base = node->InputAt(0);
|
||||||
@ -2119,9 +2086,10 @@ void InstructionSelector::VisitWord32AtomicPairLoad(Node* node) {
|
|||||||
g.GetEffectiveIndexOperand(index, &mode)};
|
g.GetEffectiveIndexOperand(index, &mode)};
|
||||||
InstructionCode code =
|
InstructionCode code =
|
||||||
kIA32Word32AtomicPairLoad | AddressingModeField::encode(mode);
|
kIA32Word32AtomicPairLoad | AddressingModeField::encode(mode);
|
||||||
|
InstructionOperand temps[] = {g.TempDoubleRegister()};
|
||||||
InstructionOperand outputs[] = {g.DefineAsRegister(projection0),
|
InstructionOperand outputs[] = {g.DefineAsRegister(projection0),
|
||||||
g.DefineAsRegister(projection1)};
|
g.DefineAsRegister(projection1)};
|
||||||
Emit(code, 2, outputs, 2, inputs);
|
Emit(code, 2, outputs, 2, inputs, 1, temps);
|
||||||
} else if (projection0 || projection1) {
|
} else if (projection0 || projection1) {
|
||||||
// Only one word is needed, so it's enough to load just that.
|
// Only one word is needed, so it's enough to load just that.
|
||||||
ArchOpcode opcode = kIA32Movl;
|
ArchOpcode opcode = kIA32Movl;
|
||||||
@ -2142,31 +2110,12 @@ void InstructionSelector::VisitWord32AtomicPairLoad(Node* node) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
void InstructionSelector::VisitWord32AtomicPairStore(Node* node) {
|
void InstructionSelector::VisitWord32AtomicPairStore(Node* node) {
|
||||||
// Release pair stores emit a MOVQ via a double register, and sequentially
|
|
||||||
// consistent stores emit CMPXCHG8B.
|
|
||||||
// https://www.cl.cam.ac.uk/~pes20/cpp/cpp0xmappings.html
|
|
||||||
|
|
||||||
IA32OperandGenerator g(this);
|
IA32OperandGenerator g(this);
|
||||||
Node* base = node->InputAt(0);
|
Node* base = node->InputAt(0);
|
||||||
Node* index = node->InputAt(1);
|
Node* index = node->InputAt(1);
|
||||||
Node* value = node->InputAt(2);
|
Node* value = node->InputAt(2);
|
||||||
Node* value_high = node->InputAt(3);
|
Node* value_high = node->InputAt(3);
|
||||||
|
|
||||||
AtomicMemoryOrder order = OpParameter<AtomicMemoryOrder>(node->op());
|
|
||||||
if (order == AtomicMemoryOrder::kAcqRel) {
|
|
||||||
AddressingMode addressing_mode;
|
|
||||||
InstructionOperand inputs[] = {
|
|
||||||
g.UseUniqueRegisterOrSlotOrConstant(value),
|
|
||||||
g.UseUniqueRegisterOrSlotOrConstant(value_high),
|
|
||||||
g.UseUniqueRegister(base),
|
|
||||||
g.GetEffectiveIndexOperand(index, &addressing_mode),
|
|
||||||
};
|
|
||||||
InstructionCode code = kIA32Word32ReleasePairStore |
|
|
||||||
AddressingModeField::encode(addressing_mode);
|
|
||||||
Emit(code, 0, nullptr, arraysize(inputs), inputs);
|
|
||||||
} else {
|
|
||||||
DCHECK_EQ(order, AtomicMemoryOrder::kSeqCst);
|
|
||||||
|
|
||||||
AddressingMode addressing_mode;
|
AddressingMode addressing_mode;
|
||||||
InstructionOperand inputs[] = {
|
InstructionOperand inputs[] = {
|
||||||
g.UseUniqueRegisterOrSlotOrConstant(value), g.UseFixed(value_high, ecx),
|
g.UseUniqueRegisterOrSlotOrConstant(value), g.UseFixed(value_high, ecx),
|
||||||
@ -2177,11 +2126,10 @@ void InstructionSelector::VisitWord32AtomicPairStore(Node* node) {
|
|||||||
// and restored at the end of the instruction.
|
// and restored at the end of the instruction.
|
||||||
InstructionOperand temps[] = {g.TempRegister(eax), g.TempRegister(edx)};
|
InstructionOperand temps[] = {g.TempRegister(eax), g.TempRegister(edx)};
|
||||||
const int num_temps = arraysize(temps);
|
const int num_temps = arraysize(temps);
|
||||||
InstructionCode code = kIA32Word32SeqCstPairStore |
|
InstructionCode code =
|
||||||
AddressingModeField::encode(addressing_mode);
|
kIA32Word32AtomicPairStore | AddressingModeField::encode(addressing_mode);
|
||||||
Emit(code, 0, nullptr, arraysize(inputs), inputs, num_temps, temps);
|
Emit(code, 0, nullptr, arraysize(inputs), inputs, num_temps, temps);
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
void InstructionSelector::VisitWord32AtomicPairAdd(Node* node) {
|
void InstructionSelector::VisitWord32AtomicPairAdd(Node* node) {
|
||||||
VisitPairAtomicBinOp(this, node, kIA32Word32AtomicPairAdd);
|
VisitPairAtomicBinOp(this, node, kIA32Word32AtomicPairAdd);
|
||||||
|
@ -32,7 +32,6 @@
|
|||||||
#define TARGET_ADDRESSING_MODE_LIST(V)
|
#define TARGET_ADDRESSING_MODE_LIST(V)
|
||||||
#endif
|
#endif
|
||||||
#include "src/base/bit-field.h"
|
#include "src/base/bit-field.h"
|
||||||
#include "src/codegen/atomic-memory-order.h"
|
|
||||||
#include "src/compiler/write-barrier-kind.h"
|
#include "src/compiler/write-barrier-kind.h"
|
||||||
|
|
||||||
namespace v8 {
|
namespace v8 {
|
||||||
@ -102,7 +101,6 @@ inline RecordWriteMode WriteBarrierKindToRecordWriteMode(
|
|||||||
V(ArchParentFramePointer) \
|
V(ArchParentFramePointer) \
|
||||||
V(ArchTruncateDoubleToI) \
|
V(ArchTruncateDoubleToI) \
|
||||||
V(ArchStoreWithWriteBarrier) \
|
V(ArchStoreWithWriteBarrier) \
|
||||||
V(ArchAtomicStoreWithWriteBarrier) \
|
|
||||||
V(ArchStackSlot) \
|
V(ArchStackSlot) \
|
||||||
V(ArchStackPointerGreaterThan) \
|
V(ArchStackPointerGreaterThan) \
|
||||||
V(ArchStackCheckOffset) \
|
V(ArchStackCheckOffset) \
|
||||||
@ -267,16 +265,6 @@ enum MemoryAccessMode {
|
|||||||
|
|
||||||
enum class AtomicWidth { kWord32, kWord64 };
|
enum class AtomicWidth { kWord32, kWord64 };
|
||||||
|
|
||||||
inline size_t AtomicWidthSize(AtomicWidth width) {
|
|
||||||
switch (width) {
|
|
||||||
case AtomicWidth::kWord32:
|
|
||||||
return 4;
|
|
||||||
case AtomicWidth::kWord64:
|
|
||||||
return 8;
|
|
||||||
}
|
|
||||||
UNREACHABLE();
|
|
||||||
}
|
|
||||||
|
|
||||||
// The InstructionCode is an opaque, target-specific integer that encodes
|
// The InstructionCode is an opaque, target-specific integer that encodes
|
||||||
// what code to emit for an instruction in the code generator. It is not
|
// what code to emit for an instruction in the code generator. It is not
|
||||||
// interesting to the register allocator, as the inputs and flags on the
|
// interesting to the register allocator, as the inputs and flags on the
|
||||||
@ -302,16 +290,10 @@ using DeoptFrameStateOffsetField = base::BitField<int, 24, 8>;
|
|||||||
// size, an access mode, or both inside the overlapping MiscField.
|
// size, an access mode, or both inside the overlapping MiscField.
|
||||||
using LaneSizeField = base::BitField<int, 22, 8>;
|
using LaneSizeField = base::BitField<int, 22, 8>;
|
||||||
using AccessModeField = base::BitField<MemoryAccessMode, 30, 2>;
|
using AccessModeField = base::BitField<MemoryAccessMode, 30, 2>;
|
||||||
// AtomicWidthField overlaps with MiscField and is used for the various Atomic
|
// AtomicOperandWidth overlaps with MiscField and is used for the various Atomic
|
||||||
// opcodes. Only used on 64bit architectures. All atomic instructions on 32bit
|
// opcodes. Only used on 64bit architectures. All atomic instructions on 32bit
|
||||||
// architectures are assumed to be 32bit wide.
|
// architectures are assumed to be 32bit wide.
|
||||||
using AtomicWidthField = base::BitField<AtomicWidth, 22, 2>;
|
using AtomicWidthField = base::BitField<AtomicWidth, 22, 2>;
|
||||||
// AtomicMemoryOrderField overlaps with MiscField and is used for the various
|
|
||||||
// Atomic opcodes. This field is not used on all architectures. It is used on
|
|
||||||
// architectures where the codegen for kSeqCst and kAcqRel differ only by
|
|
||||||
// emitting fences.
|
|
||||||
using AtomicMemoryOrderField = base::BitField<AtomicMemoryOrder, 24, 2>;
|
|
||||||
using AtomicStoreRecordWriteModeField = base::BitField<RecordWriteMode, 26, 4>;
|
|
||||||
using MiscField = base::BitField<int, 22, 10>;
|
using MiscField = base::BitField<int, 22, 10>;
|
||||||
|
|
||||||
// This static assertion serves as an early warning if we are about to exhaust
|
// This static assertion serves as an early warning if we are about to exhaust
|
||||||
|
@ -328,7 +328,6 @@ int InstructionScheduler::GetInstructionFlags(const Instruction* instr) const {
|
|||||||
return kIsBarrier;
|
return kIsBarrier;
|
||||||
|
|
||||||
case kArchStoreWithWriteBarrier:
|
case kArchStoreWithWriteBarrier:
|
||||||
case kArchAtomicStoreWithWriteBarrier:
|
|
||||||
return kHasSideEffect;
|
return kHasSideEffect;
|
||||||
|
|
||||||
case kAtomicLoadInt8:
|
case kAtomicLoadInt8:
|
||||||
|
@ -1869,14 +1869,12 @@ void InstructionSelector::VisitNode(Node* node) {
|
|||||||
case IrOpcode::kMemoryBarrier:
|
case IrOpcode::kMemoryBarrier:
|
||||||
return VisitMemoryBarrier(node);
|
return VisitMemoryBarrier(node);
|
||||||
case IrOpcode::kWord32AtomicLoad: {
|
case IrOpcode::kWord32AtomicLoad: {
|
||||||
AtomicLoadParameters params = AtomicLoadParametersOf(node->op());
|
LoadRepresentation type = LoadRepresentationOf(node->op());
|
||||||
LoadRepresentation type = params.representation();
|
|
||||||
MarkAsRepresentation(type.representation(), node);
|
MarkAsRepresentation(type.representation(), node);
|
||||||
return VisitWord32AtomicLoad(node);
|
return VisitWord32AtomicLoad(node);
|
||||||
}
|
}
|
||||||
case IrOpcode::kWord64AtomicLoad: {
|
case IrOpcode::kWord64AtomicLoad: {
|
||||||
AtomicLoadParameters params = AtomicLoadParametersOf(node->op());
|
LoadRepresentation type = LoadRepresentationOf(node->op());
|
||||||
LoadRepresentation type = params.representation();
|
|
||||||
MarkAsRepresentation(type.representation(), node);
|
MarkAsRepresentation(type.representation(), node);
|
||||||
return VisitWord64AtomicLoad(node);
|
return VisitWord64AtomicLoad(node);
|
||||||
}
|
}
|
||||||
|
@ -1292,8 +1292,7 @@ CodeGenerator::CodeGenResult CodeGenerator::AssembleArchInstruction(
|
|||||||
__ movl(result, result);
|
__ movl(result, result);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
case kArchStoreWithWriteBarrier: // Fall through.
|
case kArchStoreWithWriteBarrier: {
|
||||||
case kArchAtomicStoreWithWriteBarrier: {
|
|
||||||
RecordWriteMode mode =
|
RecordWriteMode mode =
|
||||||
static_cast<RecordWriteMode>(MiscField::decode(instr->opcode()));
|
static_cast<RecordWriteMode>(MiscField::decode(instr->opcode()));
|
||||||
Register object = i.InputRegister(0);
|
Register object = i.InputRegister(0);
|
||||||
@ -1305,12 +1304,7 @@ CodeGenerator::CodeGenResult CodeGenerator::AssembleArchInstruction(
|
|||||||
auto ool = zone()->New<OutOfLineRecordWrite>(this, object, operand, value,
|
auto ool = zone()->New<OutOfLineRecordWrite>(this, object, operand, value,
|
||||||
scratch0, scratch1, mode,
|
scratch0, scratch1, mode,
|
||||||
DetermineStubCallMode());
|
DetermineStubCallMode());
|
||||||
if (arch_opcode == kArchStoreWithWriteBarrier) {
|
|
||||||
__ StoreTaggedField(operand, value);
|
__ StoreTaggedField(operand, value);
|
||||||
} else {
|
|
||||||
DCHECK_EQ(kArchAtomicStoreWithWriteBarrier, arch_opcode);
|
|
||||||
__ AtomicStoreTaggedField(operand, value);
|
|
||||||
}
|
|
||||||
if (mode > RecordWriteMode::kValueIsPointer) {
|
if (mode > RecordWriteMode::kValueIsPointer) {
|
||||||
__ JumpIfSmi(value, ool->exit());
|
__ JumpIfSmi(value, ool->exit());
|
||||||
}
|
}
|
||||||
@ -1318,7 +1312,6 @@ CodeGenerator::CodeGenResult CodeGenerator::AssembleArchInstruction(
|
|||||||
MemoryChunk::kPointersFromHereAreInterestingMask,
|
MemoryChunk::kPointersFromHereAreInterestingMask,
|
||||||
not_zero, ool->entry());
|
not_zero, ool->entry());
|
||||||
__ bind(ool->exit());
|
__ bind(ool->exit());
|
||||||
// TODO(syg): Support non-relaxed memory orders in TSAN.
|
|
||||||
EmitTSANStoreOOLIfNeeded(zone(), this, tasm(), operand, value, i,
|
EmitTSANStoreOOLIfNeeded(zone(), this, tasm(), operand, value, i,
|
||||||
DetermineStubCallMode(), kTaggedSize);
|
DetermineStubCallMode(), kTaggedSize);
|
||||||
break;
|
break;
|
||||||
|
@ -341,8 +341,8 @@ ArchOpcode GetStoreOpcode(StoreRepresentation store_rep) {
|
|||||||
UNREACHABLE();
|
UNREACHABLE();
|
||||||
}
|
}
|
||||||
|
|
||||||
ArchOpcode GetSeqCstStoreOpcode(StoreRepresentation store_rep) {
|
ArchOpcode GetAtomicStoreOp(MachineRepresentation rep) {
|
||||||
switch (store_rep.representation()) {
|
switch (rep) {
|
||||||
case MachineRepresentation::kWord8:
|
case MachineRepresentation::kWord8:
|
||||||
return kAtomicExchangeUint8;
|
return kAtomicExchangeUint8;
|
||||||
case MachineRepresentation::kWord16:
|
case MachineRepresentation::kWord16:
|
||||||
@ -351,15 +351,6 @@ ArchOpcode GetSeqCstStoreOpcode(StoreRepresentation store_rep) {
|
|||||||
return kAtomicExchangeWord32;
|
return kAtomicExchangeWord32;
|
||||||
case MachineRepresentation::kWord64:
|
case MachineRepresentation::kWord64:
|
||||||
return kX64Word64AtomicExchangeUint64;
|
return kX64Word64AtomicExchangeUint64;
|
||||||
case MachineRepresentation::kTaggedSigned: // Fall through.
|
|
||||||
case MachineRepresentation::kTaggedPointer: // Fall through.
|
|
||||||
case MachineRepresentation::kTagged:
|
|
||||||
if (COMPRESS_POINTERS_BOOL) return kAtomicExchangeWord32;
|
|
||||||
return kX64Word64AtomicExchangeUint64;
|
|
||||||
case MachineRepresentation::kCompressedPointer: // Fall through.
|
|
||||||
case MachineRepresentation::kCompressed:
|
|
||||||
CHECK(COMPRESS_POINTERS_BOOL);
|
|
||||||
return kAtomicExchangeWord32;
|
|
||||||
default:
|
default:
|
||||||
UNREACHABLE();
|
UNREACHABLE();
|
||||||
}
|
}
|
||||||
@ -508,38 +499,15 @@ void InstructionSelector::VisitLoad(Node* node) {
|
|||||||
|
|
||||||
void InstructionSelector::VisitProtectedLoad(Node* node) { VisitLoad(node); }
|
void InstructionSelector::VisitProtectedLoad(Node* node) { VisitLoad(node); }
|
||||||
|
|
||||||
namespace {
|
void InstructionSelector::VisitStore(Node* node) {
|
||||||
|
X64OperandGenerator g(this);
|
||||||
// Shared routine for Word32/Word64 Atomic Exchange
|
|
||||||
void VisitAtomicExchange(InstructionSelector* selector, Node* node,
|
|
||||||
ArchOpcode opcode, AtomicWidth width) {
|
|
||||||
X64OperandGenerator g(selector);
|
|
||||||
Node* base = node->InputAt(0);
|
|
||||||
Node* index = node->InputAt(1);
|
|
||||||
Node* value = node->InputAt(2);
|
|
||||||
AddressingMode addressing_mode;
|
|
||||||
InstructionOperand inputs[] = {
|
|
||||||
g.UseUniqueRegister(value), g.UseUniqueRegister(base),
|
|
||||||
g.GetEffectiveIndexOperand(index, &addressing_mode)};
|
|
||||||
InstructionOperand outputs[] = {g.DefineSameAsFirst(node)};
|
|
||||||
InstructionCode code = opcode | AddressingModeField::encode(addressing_mode) |
|
|
||||||
AtomicWidthField::encode(width);
|
|
||||||
selector->Emit(code, arraysize(outputs), outputs, arraysize(inputs), inputs);
|
|
||||||
}
|
|
||||||
|
|
||||||
void VisitStoreCommon(InstructionSelector* selector, Node* node,
|
|
||||||
StoreRepresentation store_rep,
|
|
||||||
base::Optional<AtomicMemoryOrder> atomic_order,
|
|
||||||
base::Optional<AtomicWidth> atomic_width) {
|
|
||||||
X64OperandGenerator g(selector);
|
|
||||||
Node* base = node->InputAt(0);
|
Node* base = node->InputAt(0);
|
||||||
Node* index = node->InputAt(1);
|
Node* index = node->InputAt(1);
|
||||||
Node* value = node->InputAt(2);
|
Node* value = node->InputAt(2);
|
||||||
|
|
||||||
|
StoreRepresentation store_rep = StoreRepresentationOf(node->op());
|
||||||
DCHECK_NE(store_rep.representation(), MachineRepresentation::kMapWord);
|
DCHECK_NE(store_rep.representation(), MachineRepresentation::kMapWord);
|
||||||
WriteBarrierKind write_barrier_kind = store_rep.write_barrier_kind();
|
WriteBarrierKind write_barrier_kind = store_rep.write_barrier_kind();
|
||||||
const bool is_seqcst =
|
|
||||||
atomic_order && *atomic_order == AtomicMemoryOrder::kSeqCst;
|
|
||||||
|
|
||||||
if (FLAG_enable_unconditional_write_barriers &&
|
if (FLAG_enable_unconditional_write_barriers &&
|
||||||
CanBeTaggedOrCompressedPointer(store_rep.representation())) {
|
CanBeTaggedOrCompressedPointer(store_rep.representation())) {
|
||||||
@ -556,19 +524,11 @@ void VisitStoreCommon(InstructionSelector* selector, Node* node,
|
|||||||
RecordWriteMode record_write_mode =
|
RecordWriteMode record_write_mode =
|
||||||
WriteBarrierKindToRecordWriteMode(write_barrier_kind);
|
WriteBarrierKindToRecordWriteMode(write_barrier_kind);
|
||||||
InstructionOperand temps[] = {g.TempRegister(), g.TempRegister()};
|
InstructionOperand temps[] = {g.TempRegister(), g.TempRegister()};
|
||||||
InstructionCode code = is_seqcst ? kArchAtomicStoreWithWriteBarrier
|
InstructionCode code = kArchStoreWithWriteBarrier;
|
||||||
: kArchStoreWithWriteBarrier;
|
|
||||||
code |= AddressingModeField::encode(addressing_mode);
|
code |= AddressingModeField::encode(addressing_mode);
|
||||||
code |= MiscField::encode(static_cast<int>(record_write_mode));
|
code |= MiscField::encode(static_cast<int>(record_write_mode));
|
||||||
selector->Emit(code, 0, nullptr, arraysize(inputs), inputs,
|
Emit(code, 0, nullptr, arraysize(inputs), inputs, arraysize(temps), temps);
|
||||||
arraysize(temps), temps);
|
|
||||||
} else if (is_seqcst) {
|
|
||||||
VisitAtomicExchange(selector, node, GetSeqCstStoreOpcode(store_rep),
|
|
||||||
*atomic_width);
|
|
||||||
} else {
|
} else {
|
||||||
// Release and non-atomic stores emit MOV.
|
|
||||||
// https://www.cl.cam.ac.uk/~pes20/cpp/cpp0xmappings.html
|
|
||||||
|
|
||||||
if ((ElementSizeLog2Of(store_rep.representation()) <
|
if ((ElementSizeLog2Of(store_rep.representation()) <
|
||||||
kSystemPointerSizeLog2) &&
|
kSystemPointerSizeLog2) &&
|
||||||
value->opcode() == IrOpcode::kTruncateInt64ToInt32) {
|
value->opcode() == IrOpcode::kTruncateInt64ToInt32) {
|
||||||
@ -598,18 +558,11 @@ void VisitStoreCommon(InstructionSelector* selector, Node* node,
|
|||||||
ArchOpcode opcode = GetStoreOpcode(store_rep);
|
ArchOpcode opcode = GetStoreOpcode(store_rep);
|
||||||
InstructionCode code =
|
InstructionCode code =
|
||||||
opcode | AddressingModeField::encode(addressing_mode);
|
opcode | AddressingModeField::encode(addressing_mode);
|
||||||
selector->Emit(code, 0, static_cast<InstructionOperand*>(nullptr),
|
Emit(code, 0, static_cast<InstructionOperand*>(nullptr), input_count,
|
||||||
input_count, inputs, temp_count, temps);
|
inputs, temp_count, temps);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
} // namespace
|
|
||||||
|
|
||||||
void InstructionSelector::VisitStore(Node* node) {
|
|
||||||
return VisitStoreCommon(this, node, StoreRepresentationOf(node->op()),
|
|
||||||
base::nullopt, base::nullopt);
|
|
||||||
}
|
|
||||||
|
|
||||||
void InstructionSelector::VisitProtectedStore(Node* node) {
|
void InstructionSelector::VisitProtectedStore(Node* node) {
|
||||||
X64OperandGenerator g(this);
|
X64OperandGenerator g(this);
|
||||||
Node* value = node->InputAt(2);
|
Node* value = node->InputAt(2);
|
||||||
@ -2387,6 +2340,23 @@ void VisitAtomicCompareExchange(InstructionSelector* selector, Node* node,
|
|||||||
selector->Emit(code, arraysize(outputs), outputs, arraysize(inputs), inputs);
|
selector->Emit(code, arraysize(outputs), outputs, arraysize(inputs), inputs);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Shared routine for Word32/Word64 Atomic Exchange
|
||||||
|
void VisitAtomicExchange(InstructionSelector* selector, Node* node,
|
||||||
|
ArchOpcode opcode, AtomicWidth width) {
|
||||||
|
X64OperandGenerator g(selector);
|
||||||
|
Node* base = node->InputAt(0);
|
||||||
|
Node* index = node->InputAt(1);
|
||||||
|
Node* value = node->InputAt(2);
|
||||||
|
AddressingMode addressing_mode;
|
||||||
|
InstructionOperand inputs[] = {
|
||||||
|
g.UseUniqueRegister(value), g.UseUniqueRegister(base),
|
||||||
|
g.GetEffectiveIndexOperand(index, &addressing_mode)};
|
||||||
|
InstructionOperand outputs[] = {g.DefineSameAsFirst(node)};
|
||||||
|
InstructionCode code = opcode | AddressingModeField::encode(addressing_mode) |
|
||||||
|
AtomicWidthField::encode(width);
|
||||||
|
selector->Emit(code, arraysize(outputs), outputs, arraysize(inputs), inputs);
|
||||||
|
}
|
||||||
|
|
||||||
} // namespace
|
} // namespace
|
||||||
|
|
||||||
// Shared routine for word comparison against zero.
|
// Shared routine for word comparison against zero.
|
||||||
@ -2754,44 +2724,29 @@ void InstructionSelector::VisitMemoryBarrier(Node* node) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
void InstructionSelector::VisitWord32AtomicLoad(Node* node) {
|
void InstructionSelector::VisitWord32AtomicLoad(Node* node) {
|
||||||
AtomicLoadParameters atomic_load_params = AtomicLoadParametersOf(node->op());
|
LoadRepresentation load_rep = LoadRepresentationOf(node->op());
|
||||||
LoadRepresentation load_rep = atomic_load_params.representation();
|
DCHECK(load_rep.representation() == MachineRepresentation::kWord8 ||
|
||||||
DCHECK(IsIntegral(load_rep.representation()) ||
|
load_rep.representation() == MachineRepresentation::kWord16 ||
|
||||||
IsAnyTagged(load_rep.representation()) ||
|
load_rep.representation() == MachineRepresentation::kWord32);
|
||||||
(COMPRESS_POINTERS_BOOL &&
|
USE(load_rep);
|
||||||
CanBeCompressedPointer(load_rep.representation())));
|
VisitLoad(node);
|
||||||
DCHECK_NE(load_rep.representation(), MachineRepresentation::kWord64);
|
|
||||||
DCHECK(!load_rep.IsMapWord());
|
|
||||||
// The memory order is ignored as both acquire and sequentially consistent
|
|
||||||
// loads can emit MOV.
|
|
||||||
// https://www.cl.cam.ac.uk/~pes20/cpp/cpp0xmappings.html
|
|
||||||
VisitLoad(node, node, GetLoadOpcode(load_rep));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void InstructionSelector::VisitWord64AtomicLoad(Node* node) {
|
void InstructionSelector::VisitWord64AtomicLoad(Node* node) {
|
||||||
AtomicLoadParameters atomic_load_params = AtomicLoadParametersOf(node->op());
|
LoadRepresentation load_rep = LoadRepresentationOf(node->op());
|
||||||
DCHECK(!atomic_load_params.representation().IsMapWord());
|
USE(load_rep);
|
||||||
// The memory order is ignored as both acquire and sequentially consistent
|
VisitLoad(node);
|
||||||
// loads can emit MOV.
|
|
||||||
// https://www.cl.cam.ac.uk/~pes20/cpp/cpp0xmappings.html
|
|
||||||
VisitLoad(node, node, GetLoadOpcode(atomic_load_params.representation()));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void InstructionSelector::VisitWord32AtomicStore(Node* node) {
|
void InstructionSelector::VisitWord32AtomicStore(Node* node) {
|
||||||
AtomicStoreParameters params = AtomicStoreParametersOf(node->op());
|
MachineRepresentation rep = AtomicStoreRepresentationOf(node->op());
|
||||||
DCHECK_NE(params.representation(), MachineRepresentation::kWord64);
|
DCHECK_NE(rep, MachineRepresentation::kWord64);
|
||||||
DCHECK_IMPLIES(CanBeTaggedOrCompressedPointer(params.representation()),
|
VisitAtomicExchange(this, node, GetAtomicStoreOp(rep), AtomicWidth::kWord32);
|
||||||
kTaggedSize == 4);
|
|
||||||
VisitStoreCommon(this, node, params.store_representation(), params.order(),
|
|
||||||
AtomicWidth::kWord32);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void InstructionSelector::VisitWord64AtomicStore(Node* node) {
|
void InstructionSelector::VisitWord64AtomicStore(Node* node) {
|
||||||
AtomicStoreParameters params = AtomicStoreParametersOf(node->op());
|
MachineRepresentation rep = AtomicStoreRepresentationOf(node->op());
|
||||||
DCHECK_IMPLIES(CanBeTaggedOrCompressedPointer(params.representation()),
|
VisitAtomicExchange(this, node, GetAtomicStoreOp(rep), AtomicWidth::kWord64);
|
||||||
kTaggedSize == 8);
|
|
||||||
VisitStoreCommon(this, node, params.store_representation(), params.order(),
|
|
||||||
AtomicWidth::kWord64);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void InstructionSelector::VisitWord32AtomicExchange(Node* node) {
|
void InstructionSelector::VisitWord32AtomicExchange(Node* node) {
|
||||||
|
@ -679,25 +679,22 @@ TNode<Object> CodeAssembler::LoadFullTagged(Node* base, TNode<IntPtrT> offset) {
|
|||||||
return BitcastWordToTagged(Load<RawPtrT>(base, offset));
|
return BitcastWordToTagged(Load<RawPtrT>(base, offset));
|
||||||
}
|
}
|
||||||
|
|
||||||
Node* CodeAssembler::AtomicLoad(MachineType type, AtomicMemoryOrder order,
|
Node* CodeAssembler::AtomicLoad(MachineType type, TNode<RawPtrT> base,
|
||||||
TNode<RawPtrT> base, TNode<WordT> offset) {
|
TNode<WordT> offset) {
|
||||||
DCHECK(!raw_assembler()->IsMapOffsetConstantMinusTag(offset));
|
DCHECK(!raw_assembler()->IsMapOffsetConstantMinusTag(offset));
|
||||||
return raw_assembler()->AtomicLoad(AtomicLoadParameters(type, order), base,
|
return raw_assembler()->AtomicLoad(type, base, offset);
|
||||||
offset);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
template <class Type>
|
template <class Type>
|
||||||
TNode<Type> CodeAssembler::AtomicLoad64(AtomicMemoryOrder order,
|
TNode<Type> CodeAssembler::AtomicLoad64(TNode<RawPtrT> base,
|
||||||
TNode<RawPtrT> base,
|
|
||||||
TNode<WordT> offset) {
|
TNode<WordT> offset) {
|
||||||
return UncheckedCast<Type>(raw_assembler()->AtomicLoad64(
|
return UncheckedCast<Type>(raw_assembler()->AtomicLoad64(base, offset));
|
||||||
AtomicLoadParameters(MachineType::Uint64(), order), base, offset));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
template TNode<AtomicInt64> CodeAssembler::AtomicLoad64<AtomicInt64>(
|
template TNode<AtomicInt64> CodeAssembler::AtomicLoad64<AtomicInt64>(
|
||||||
AtomicMemoryOrder order, TNode<RawPtrT> base, TNode<WordT> offset);
|
TNode<RawPtrT> base, TNode<WordT> offset);
|
||||||
template TNode<AtomicUint64> CodeAssembler::AtomicLoad64<AtomicUint64>(
|
template TNode<AtomicUint64> CodeAssembler::AtomicLoad64<AtomicUint64>(
|
||||||
AtomicMemoryOrder order, TNode<RawPtrT> base, TNode<WordT> offset);
|
TNode<RawPtrT> base, TNode<WordT> offset);
|
||||||
|
|
||||||
Node* CodeAssembler::LoadFromObject(MachineType type, TNode<Object> object,
|
Node* CodeAssembler::LoadFromObject(MachineType type, TNode<Object> object,
|
||||||
TNode<IntPtrT> offset) {
|
TNode<IntPtrT> offset) {
|
||||||
@ -862,22 +859,16 @@ void CodeAssembler::StoreFullTaggedNoWriteBarrier(TNode<RawPtrT> base,
|
|||||||
BitcastTaggedToWord(tagged_value));
|
BitcastTaggedToWord(tagged_value));
|
||||||
}
|
}
|
||||||
|
|
||||||
void CodeAssembler::AtomicStore(MachineRepresentation rep,
|
void CodeAssembler::AtomicStore(MachineRepresentation rep, TNode<RawPtrT> base,
|
||||||
AtomicMemoryOrder order, TNode<RawPtrT> base,
|
|
||||||
TNode<WordT> offset, TNode<Word32T> value) {
|
TNode<WordT> offset, TNode<Word32T> value) {
|
||||||
DCHECK(!raw_assembler()->IsMapOffsetConstantMinusTag(offset));
|
DCHECK(!raw_assembler()->IsMapOffsetConstantMinusTag(offset));
|
||||||
raw_assembler()->AtomicStore(
|
raw_assembler()->AtomicStore(rep, base, offset, value);
|
||||||
AtomicStoreParameters(rep, WriteBarrierKind::kNoWriteBarrier, order),
|
|
||||||
base, offset, value);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void CodeAssembler::AtomicStore64(AtomicMemoryOrder order, TNode<RawPtrT> base,
|
void CodeAssembler::AtomicStore64(TNode<RawPtrT> base, TNode<WordT> offset,
|
||||||
TNode<WordT> offset, TNode<UintPtrT> value,
|
TNode<UintPtrT> value,
|
||||||
TNode<UintPtrT> value_high) {
|
TNode<UintPtrT> value_high) {
|
||||||
raw_assembler()->AtomicStore64(
|
raw_assembler()->AtomicStore64(base, offset, value, value_high);
|
||||||
AtomicStoreParameters(MachineRepresentation::kWord64,
|
|
||||||
WriteBarrierKind::kNoWriteBarrier, order),
|
|
||||||
base, offset, value, value_high);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
#define ATOMIC_FUNCTION(name) \
|
#define ATOMIC_FUNCTION(name) \
|
||||||
|
@ -17,7 +17,6 @@
|
|||||||
#include "src/base/optional.h"
|
#include "src/base/optional.h"
|
||||||
#include "src/base/type-traits.h"
|
#include "src/base/type-traits.h"
|
||||||
#include "src/builtins/builtins.h"
|
#include "src/builtins/builtins.h"
|
||||||
#include "src/codegen/atomic-memory-order.h"
|
|
||||||
#include "src/codegen/code-factory.h"
|
#include "src/codegen/code-factory.h"
|
||||||
#include "src/codegen/machine-type.h"
|
#include "src/codegen/machine-type.h"
|
||||||
#include "src/codegen/source-position.h"
|
#include "src/codegen/source-position.h"
|
||||||
@ -744,14 +743,12 @@ class V8_EXPORT_PRIVATE CodeAssembler {
|
|||||||
return UncheckedCast<Type>(Load(MachineTypeOf<Type>::value, base, offset));
|
return UncheckedCast<Type>(Load(MachineTypeOf<Type>::value, base, offset));
|
||||||
}
|
}
|
||||||
template <class Type>
|
template <class Type>
|
||||||
TNode<Type> AtomicLoad(AtomicMemoryOrder order, TNode<RawPtrT> base,
|
TNode<Type> AtomicLoad(TNode<RawPtrT> base, TNode<WordT> offset) {
|
||||||
TNode<WordT> offset) {
|
|
||||||
return UncheckedCast<Type>(
|
return UncheckedCast<Type>(
|
||||||
AtomicLoad(MachineTypeOf<Type>::value, order, base, offset));
|
AtomicLoad(MachineTypeOf<Type>::value, base, offset));
|
||||||
}
|
}
|
||||||
template <class Type>
|
template <class Type>
|
||||||
TNode<Type> AtomicLoad64(AtomicMemoryOrder order, TNode<RawPtrT> base,
|
TNode<Type> AtomicLoad64(TNode<RawPtrT> base, TNode<WordT> offset);
|
||||||
TNode<WordT> offset);
|
|
||||||
// Load uncompressed tagged value from (most likely off JS heap) memory
|
// Load uncompressed tagged value from (most likely off JS heap) memory
|
||||||
// location.
|
// location.
|
||||||
TNode<Object> LoadFullTagged(Node* base);
|
TNode<Object> LoadFullTagged(Node* base);
|
||||||
@ -812,14 +809,12 @@ class V8_EXPORT_PRIVATE CodeAssembler {
|
|||||||
TNode<HeapObject> object,
|
TNode<HeapObject> object,
|
||||||
int offset, Node* value);
|
int offset, Node* value);
|
||||||
void OptimizedStoreMap(TNode<HeapObject> object, TNode<Map>);
|
void OptimizedStoreMap(TNode<HeapObject> object, TNode<Map>);
|
||||||
void AtomicStore(MachineRepresentation rep, AtomicMemoryOrder order,
|
void AtomicStore(MachineRepresentation rep, TNode<RawPtrT> base,
|
||||||
TNode<RawPtrT> base, TNode<WordT> offset,
|
TNode<WordT> offset, TNode<Word32T> value);
|
||||||
TNode<Word32T> value);
|
|
||||||
// {value_high} is used for 64-bit stores on 32-bit platforms, must be
|
// {value_high} is used for 64-bit stores on 32-bit platforms, must be
|
||||||
// nullptr in other cases.
|
// nullptr in other cases.
|
||||||
void AtomicStore64(AtomicMemoryOrder order, TNode<RawPtrT> base,
|
void AtomicStore64(TNode<RawPtrT> base, TNode<WordT> offset,
|
||||||
TNode<WordT> offset, TNode<UintPtrT> value,
|
TNode<UintPtrT> value, TNode<UintPtrT> value_high);
|
||||||
TNode<UintPtrT> value_high);
|
|
||||||
|
|
||||||
TNode<Word32T> AtomicAdd(MachineType type, TNode<RawPtrT> base,
|
TNode<Word32T> AtomicAdd(MachineType type, TNode<RawPtrT> base,
|
||||||
TNode<UintPtrT> offset, TNode<Word32T> value);
|
TNode<UintPtrT> offset, TNode<Word32T> value);
|
||||||
@ -1358,8 +1353,7 @@ class V8_EXPORT_PRIVATE CodeAssembler {
|
|||||||
const CallInterfaceDescriptor& descriptor, int input_count,
|
const CallInterfaceDescriptor& descriptor, int input_count,
|
||||||
Node* const* inputs);
|
Node* const* inputs);
|
||||||
|
|
||||||
Node* AtomicLoad(MachineType type, AtomicMemoryOrder order,
|
Node* AtomicLoad(MachineType type, TNode<RawPtrT> base, TNode<WordT> offset);
|
||||||
TNode<RawPtrT> base, TNode<WordT> offset);
|
|
||||||
|
|
||||||
Node* UnalignedLoad(MachineType type, TNode<RawPtrT> base,
|
Node* UnalignedLoad(MachineType type, TNode<RawPtrT> base,
|
||||||
TNode<WordT> offset);
|
TNode<WordT> offset);
|
||||||
|
@ -944,31 +944,29 @@ void Int64Lowering::LowerNode(Node* node) {
|
|||||||
}
|
}
|
||||||
case IrOpcode::kWord64AtomicLoad: {
|
case IrOpcode::kWord64AtomicLoad: {
|
||||||
DCHECK_EQ(4, node->InputCount());
|
DCHECK_EQ(4, node->InputCount());
|
||||||
AtomicLoadParameters params = AtomicLoadParametersOf(node->op());
|
MachineType type = AtomicOpType(node->op());
|
||||||
DefaultLowering(node, true);
|
DefaultLowering(node, true);
|
||||||
if (params.representation() == MachineType::Uint64()) {
|
if (type == MachineType::Uint64()) {
|
||||||
NodeProperties::ChangeOp(
|
NodeProperties::ChangeOp(node, machine()->Word32AtomicPairLoad());
|
||||||
node, machine()->Word32AtomicPairLoad(params.order()));
|
|
||||||
ReplaceNodeWithProjections(node);
|
ReplaceNodeWithProjections(node);
|
||||||
} else {
|
} else {
|
||||||
NodeProperties::ChangeOp(node, machine()->Word32AtomicLoad(params));
|
NodeProperties::ChangeOp(node, machine()->Word32AtomicLoad(type));
|
||||||
ReplaceNode(node, node, graph()->NewNode(common()->Int32Constant(0)));
|
ReplaceNode(node, node, graph()->NewNode(common()->Int32Constant(0)));
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
case IrOpcode::kWord64AtomicStore: {
|
case IrOpcode::kWord64AtomicStore: {
|
||||||
DCHECK_EQ(5, node->InputCount());
|
DCHECK_EQ(5, node->InputCount());
|
||||||
AtomicStoreParameters params = AtomicStoreParametersOf(node->op());
|
MachineRepresentation rep = AtomicStoreRepresentationOf(node->op());
|
||||||
if (params.representation() == MachineRepresentation::kWord64) {
|
if (rep == MachineRepresentation::kWord64) {
|
||||||
LowerMemoryBaseAndIndex(node);
|
LowerMemoryBaseAndIndex(node);
|
||||||
Node* value = node->InputAt(2);
|
Node* value = node->InputAt(2);
|
||||||
node->ReplaceInput(2, GetReplacementLow(value));
|
node->ReplaceInput(2, GetReplacementLow(value));
|
||||||
node->InsertInput(zone(), 3, GetReplacementHigh(value));
|
node->InsertInput(zone(), 3, GetReplacementHigh(value));
|
||||||
NodeProperties::ChangeOp(
|
NodeProperties::ChangeOp(node, machine()->Word32AtomicPairStore());
|
||||||
node, machine()->Word32AtomicPairStore(params.order()));
|
|
||||||
} else {
|
} else {
|
||||||
DefaultLowering(node, true);
|
DefaultLowering(node, true);
|
||||||
NodeProperties::ChangeOp(node, machine()->Word32AtomicStore(params));
|
NodeProperties::ChangeOp(node, machine()->Word32AtomicStore(rep));
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
@ -121,11 +121,6 @@ class MachineRepresentationInferrer {
|
|||||||
break;
|
break;
|
||||||
case IrOpcode::kWord32AtomicLoad:
|
case IrOpcode::kWord32AtomicLoad:
|
||||||
case IrOpcode::kWord64AtomicLoad:
|
case IrOpcode::kWord64AtomicLoad:
|
||||||
representation_vector_[node->id()] =
|
|
||||||
PromoteRepresentation(AtomicLoadParametersOf(node->op())
|
|
||||||
.representation()
|
|
||||||
.representation());
|
|
||||||
break;
|
|
||||||
case IrOpcode::kLoad:
|
case IrOpcode::kLoad:
|
||||||
case IrOpcode::kLoadImmutable:
|
case IrOpcode::kLoadImmutable:
|
||||||
case IrOpcode::kProtectedLoad:
|
case IrOpcode::kProtectedLoad:
|
||||||
@ -158,8 +153,8 @@ class MachineRepresentationInferrer {
|
|||||||
}
|
}
|
||||||
case IrOpcode::kWord32AtomicStore:
|
case IrOpcode::kWord32AtomicStore:
|
||||||
case IrOpcode::kWord64AtomicStore:
|
case IrOpcode::kWord64AtomicStore:
|
||||||
representation_vector_[node->id()] = PromoteRepresentation(
|
representation_vector_[node->id()] =
|
||||||
AtomicStoreParametersOf(node->op()).representation());
|
PromoteRepresentation(AtomicStoreRepresentationOf(node->op()));
|
||||||
break;
|
break;
|
||||||
case IrOpcode::kWord32AtomicPairLoad:
|
case IrOpcode::kWord32AtomicPairLoad:
|
||||||
case IrOpcode::kWord32AtomicPairStore:
|
case IrOpcode::kWord32AtomicPairStore:
|
||||||
@ -590,12 +585,9 @@ class MachineRepresentationChecker {
|
|||||||
case MachineRepresentation::kTaggedPointer:
|
case MachineRepresentation::kTaggedPointer:
|
||||||
case MachineRepresentation::kTaggedSigned:
|
case MachineRepresentation::kTaggedSigned:
|
||||||
if (COMPRESS_POINTERS_BOOL &&
|
if (COMPRESS_POINTERS_BOOL &&
|
||||||
((node->opcode() == IrOpcode::kStore &&
|
node->opcode() == IrOpcode::kStore &&
|
||||||
IsAnyTagged(StoreRepresentationOf(node->op())
|
IsAnyTagged(
|
||||||
.representation())) ||
|
StoreRepresentationOf(node->op()).representation())) {
|
||||||
(node->opcode() == IrOpcode::kWord32AtomicStore &&
|
|
||||||
IsAnyTagged(AtomicStoreParametersOf(node->op())
|
|
||||||
.representation())))) {
|
|
||||||
CheckValueInputIsCompressedOrTagged(node, 2);
|
CheckValueInputIsCompressedOrTagged(node, 2);
|
||||||
} else {
|
} else {
|
||||||
CheckValueInputIsTagged(node, 2);
|
CheckValueInputIsTagged(node, 2);
|
||||||
|
@ -32,41 +32,6 @@ std::ostream& operator<<(std::ostream& os, StoreRepresentation rep) {
|
|||||||
return os << rep.representation() << ", " << rep.write_barrier_kind();
|
return os << rep.representation() << ", " << rep.write_barrier_kind();
|
||||||
}
|
}
|
||||||
|
|
||||||
bool operator==(AtomicStoreParameters lhs, AtomicStoreParameters rhs) {
|
|
||||||
return lhs.store_representation() == rhs.store_representation() &&
|
|
||||||
lhs.order() == rhs.order();
|
|
||||||
}
|
|
||||||
|
|
||||||
bool operator!=(AtomicStoreParameters lhs, AtomicStoreParameters rhs) {
|
|
||||||
return !(lhs == rhs);
|
|
||||||
}
|
|
||||||
|
|
||||||
size_t hash_value(AtomicStoreParameters params) {
|
|
||||||
return base::hash_combine(hash_value(params.store_representation()),
|
|
||||||
params.order());
|
|
||||||
}
|
|
||||||
|
|
||||||
std::ostream& operator<<(std::ostream& os, AtomicStoreParameters params) {
|
|
||||||
return os << params.store_representation() << ", " << params.order();
|
|
||||||
}
|
|
||||||
|
|
||||||
bool operator==(AtomicLoadParameters lhs, AtomicLoadParameters rhs) {
|
|
||||||
return lhs.representation() == rhs.representation() &&
|
|
||||||
lhs.order() == rhs.order();
|
|
||||||
}
|
|
||||||
|
|
||||||
bool operator!=(AtomicLoadParameters lhs, AtomicLoadParameters rhs) {
|
|
||||||
return !(lhs == rhs);
|
|
||||||
}
|
|
||||||
|
|
||||||
size_t hash_value(AtomicLoadParameters params) {
|
|
||||||
return base::hash_combine(params.representation(), params.order());
|
|
||||||
}
|
|
||||||
|
|
||||||
std::ostream& operator<<(std::ostream& os, AtomicLoadParameters params) {
|
|
||||||
return os << params.representation() << ", " << params.order();
|
|
||||||
}
|
|
||||||
|
|
||||||
size_t hash_value(MemoryAccessKind kind) { return static_cast<size_t>(kind); }
|
size_t hash_value(MemoryAccessKind kind) { return static_cast<size_t>(kind); }
|
||||||
|
|
||||||
std::ostream& operator<<(std::ostream& os, MemoryAccessKind kind) {
|
std::ostream& operator<<(std::ostream& os, MemoryAccessKind kind) {
|
||||||
@ -156,29 +121,20 @@ bool operator==(LoadLaneParameters lhs, LoadLaneParameters rhs) {
|
|||||||
LoadRepresentation LoadRepresentationOf(Operator const* op) {
|
LoadRepresentation LoadRepresentationOf(Operator const* op) {
|
||||||
DCHECK(IrOpcode::kLoad == op->opcode() ||
|
DCHECK(IrOpcode::kLoad == op->opcode() ||
|
||||||
IrOpcode::kProtectedLoad == op->opcode() ||
|
IrOpcode::kProtectedLoad == op->opcode() ||
|
||||||
|
IrOpcode::kWord32AtomicLoad == op->opcode() ||
|
||||||
|
IrOpcode::kWord64AtomicLoad == op->opcode() ||
|
||||||
|
IrOpcode::kWord32AtomicPairLoad == op->opcode() ||
|
||||||
IrOpcode::kUnalignedLoad == op->opcode() ||
|
IrOpcode::kUnalignedLoad == op->opcode() ||
|
||||||
IrOpcode::kLoadImmutable == op->opcode());
|
IrOpcode::kLoadImmutable == op->opcode());
|
||||||
return OpParameter<LoadRepresentation>(op);
|
return OpParameter<LoadRepresentation>(op);
|
||||||
}
|
}
|
||||||
|
|
||||||
AtomicLoadParameters AtomicLoadParametersOf(Operator const* op) {
|
|
||||||
DCHECK(IrOpcode::kWord32AtomicLoad == op->opcode() ||
|
|
||||||
IrOpcode::kWord64AtomicLoad == op->opcode());
|
|
||||||
return OpParameter<AtomicLoadParameters>(op);
|
|
||||||
}
|
|
||||||
|
|
||||||
StoreRepresentation const& StoreRepresentationOf(Operator const* op) {
|
StoreRepresentation const& StoreRepresentationOf(Operator const* op) {
|
||||||
DCHECK(IrOpcode::kStore == op->opcode() ||
|
DCHECK(IrOpcode::kStore == op->opcode() ||
|
||||||
IrOpcode::kProtectedStore == op->opcode());
|
IrOpcode::kProtectedStore == op->opcode());
|
||||||
return OpParameter<StoreRepresentation>(op);
|
return OpParameter<StoreRepresentation>(op);
|
||||||
}
|
}
|
||||||
|
|
||||||
AtomicStoreParameters const& AtomicStoreParametersOf(Operator const* op) {
|
|
||||||
DCHECK(IrOpcode::kWord32AtomicStore == op->opcode() ||
|
|
||||||
IrOpcode::kWord64AtomicStore == op->opcode());
|
|
||||||
return OpParameter<AtomicStoreParameters>(op);
|
|
||||||
}
|
|
||||||
|
|
||||||
UnalignedStoreRepresentation const& UnalignedStoreRepresentationOf(
|
UnalignedStoreRepresentation const& UnalignedStoreRepresentationOf(
|
||||||
Operator const* op) {
|
Operator const* op) {
|
||||||
DCHECK_EQ(IrOpcode::kUnalignedStore, op->opcode());
|
DCHECK_EQ(IrOpcode::kUnalignedStore, op->opcode());
|
||||||
@ -225,6 +181,12 @@ StackSlotRepresentation const& StackSlotRepresentationOf(Operator const* op) {
|
|||||||
return OpParameter<StackSlotRepresentation>(op);
|
return OpParameter<StackSlotRepresentation>(op);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
MachineRepresentation AtomicStoreRepresentationOf(Operator const* op) {
|
||||||
|
DCHECK(IrOpcode::kWord32AtomicStore == op->opcode() ||
|
||||||
|
IrOpcode::kWord64AtomicStore == op->opcode());
|
||||||
|
return OpParameter<MachineRepresentation>(op);
|
||||||
|
}
|
||||||
|
|
||||||
MachineType AtomicOpType(Operator const* op) {
|
MachineType AtomicOpType(Operator const* op) {
|
||||||
return OpParameter<MachineType>(op);
|
return OpParameter<MachineType>(op);
|
||||||
}
|
}
|
||||||
@ -687,30 +649,6 @@ std::ostream& operator<<(std::ostream& os, TruncateKind kind) {
|
|||||||
V(S128Load32Zero) \
|
V(S128Load32Zero) \
|
||||||
V(S128Load64Zero)
|
V(S128Load64Zero)
|
||||||
|
|
||||||
#if TAGGED_SIZE_8_BYTES
|
|
||||||
|
|
||||||
#define ATOMIC_TAGGED_TYPE_LIST(V)
|
|
||||||
|
|
||||||
#define ATOMIC64_TAGGED_TYPE_LIST(V) \
|
|
||||||
V(TaggedSigned) \
|
|
||||||
V(TaggedPointer) \
|
|
||||||
V(AnyTagged) \
|
|
||||||
V(CompressedPointer) \
|
|
||||||
V(AnyCompressed)
|
|
||||||
|
|
||||||
#else
|
|
||||||
|
|
||||||
#define ATOMIC_TAGGED_TYPE_LIST(V) \
|
|
||||||
V(TaggedSigned) \
|
|
||||||
V(TaggedPointer) \
|
|
||||||
V(AnyTagged) \
|
|
||||||
V(CompressedPointer) \
|
|
||||||
V(AnyCompressed)
|
|
||||||
|
|
||||||
#define ATOMIC64_TAGGED_TYPE_LIST(V)
|
|
||||||
|
|
||||||
#endif // TAGGED_SIZE_8_BYTES
|
|
||||||
|
|
||||||
#define ATOMIC_U32_TYPE_LIST(V) \
|
#define ATOMIC_U32_TYPE_LIST(V) \
|
||||||
V(Uint8) \
|
V(Uint8) \
|
||||||
V(Uint16) \
|
V(Uint16) \
|
||||||
@ -726,28 +664,6 @@ std::ostream& operator<<(std::ostream& os, TruncateKind kind) {
|
|||||||
ATOMIC_U32_TYPE_LIST(V) \
|
ATOMIC_U32_TYPE_LIST(V) \
|
||||||
V(Uint64)
|
V(Uint64)
|
||||||
|
|
||||||
#if TAGGED_SIZE_8_BYTES
|
|
||||||
|
|
||||||
#define ATOMIC_TAGGED_REPRESENTATION_LIST(V)
|
|
||||||
|
|
||||||
#define ATOMIC64_TAGGED_REPRESENTATION_LIST(V) \
|
|
||||||
V(kTaggedSigned) \
|
|
||||||
V(kTaggedPointer) \
|
|
||||||
V(kTagged)
|
|
||||||
|
|
||||||
#else
|
|
||||||
|
|
||||||
#define ATOMIC_TAGGED_REPRESENTATION_LIST(V) \
|
|
||||||
V(kTaggedSigned) \
|
|
||||||
V(kTaggedPointer) \
|
|
||||||
V(kTagged) \
|
|
||||||
V(kCompressedPointer) \
|
|
||||||
V(kCompressed)
|
|
||||||
|
|
||||||
#define ATOMIC64_TAGGED_REPRESENTATION_LIST(V)
|
|
||||||
|
|
||||||
#endif // TAGGED_SIZE_8_BYTES
|
|
||||||
|
|
||||||
#define ATOMIC_REPRESENTATION_LIST(V) \
|
#define ATOMIC_REPRESENTATION_LIST(V) \
|
||||||
V(kWord8) \
|
V(kWord8) \
|
||||||
V(kWord16) \
|
V(kWord16) \
|
||||||
@ -1052,62 +968,54 @@ struct MachineOperatorGlobalCache {
|
|||||||
#undef STORE
|
#undef STORE
|
||||||
|
|
||||||
#define ATOMIC_LOAD(Type) \
|
#define ATOMIC_LOAD(Type) \
|
||||||
struct Word32SeqCstLoad##Type##Operator \
|
struct Word32AtomicLoad##Type##Operator final \
|
||||||
: public Operator1<AtomicLoadParameters> { \
|
: public Operator1<LoadRepresentation> { \
|
||||||
Word32SeqCstLoad##Type##Operator() \
|
Word32AtomicLoad##Type##Operator() \
|
||||||
: Operator1<AtomicLoadParameters>( \
|
: Operator1<LoadRepresentation>( \
|
||||||
IrOpcode::kWord32AtomicLoad, Operator::kEliminatable, \
|
IrOpcode::kWord32AtomicLoad, Operator::kEliminatable, \
|
||||||
"Word32AtomicLoad", 2, 1, 1, 1, 1, 0, \
|
"Word32AtomicLoad", 2, 1, 1, 1, 1, 0, MachineType::Type()) {} \
|
||||||
AtomicLoadParameters(MachineType::Type(), \
|
|
||||||
AtomicMemoryOrder::kSeqCst)) {} \
|
|
||||||
}; \
|
}; \
|
||||||
Word32SeqCstLoad##Type##Operator kWord32SeqCstLoad##Type;
|
Word32AtomicLoad##Type##Operator kWord32AtomicLoad##Type;
|
||||||
ATOMIC_TYPE_LIST(ATOMIC_LOAD)
|
ATOMIC_TYPE_LIST(ATOMIC_LOAD)
|
||||||
#undef ATOMIC_LOAD
|
#undef ATOMIC_LOAD
|
||||||
|
|
||||||
#define ATOMIC_LOAD(Type) \
|
#define ATOMIC_LOAD(Type) \
|
||||||
struct Word64SeqCstLoad##Type##Operator \
|
struct Word64AtomicLoad##Type##Operator final \
|
||||||
: public Operator1<AtomicLoadParameters> { \
|
: public Operator1<LoadRepresentation> { \
|
||||||
Word64SeqCstLoad##Type##Operator() \
|
Word64AtomicLoad##Type##Operator() \
|
||||||
: Operator1<AtomicLoadParameters>( \
|
: Operator1<LoadRepresentation>( \
|
||||||
IrOpcode::kWord64AtomicLoad, Operator::kEliminatable, \
|
IrOpcode::kWord64AtomicLoad, Operator::kEliminatable, \
|
||||||
"Word64AtomicLoad", 2, 1, 1, 1, 1, 0, \
|
"Word64AtomicLoad", 2, 1, 1, 1, 1, 0, MachineType::Type()) {} \
|
||||||
AtomicLoadParameters(MachineType::Type(), \
|
|
||||||
AtomicMemoryOrder::kSeqCst)) {} \
|
|
||||||
}; \
|
}; \
|
||||||
Word64SeqCstLoad##Type##Operator kWord64SeqCstLoad##Type;
|
Word64AtomicLoad##Type##Operator kWord64AtomicLoad##Type;
|
||||||
ATOMIC_U64_TYPE_LIST(ATOMIC_LOAD)
|
ATOMIC_U64_TYPE_LIST(ATOMIC_LOAD)
|
||||||
#undef ATOMIC_LOAD
|
#undef ATOMIC_LOAD
|
||||||
|
|
||||||
#define ATOMIC_STORE(Type) \
|
#define ATOMIC_STORE(Type) \
|
||||||
struct Word32SeqCstStore##Type##Operator \
|
struct Word32AtomicStore##Type##Operator \
|
||||||
: public Operator1<AtomicStoreParameters> { \
|
: public Operator1<MachineRepresentation> { \
|
||||||
Word32SeqCstStore##Type##Operator() \
|
Word32AtomicStore##Type##Operator() \
|
||||||
: Operator1<AtomicStoreParameters>( \
|
: Operator1<MachineRepresentation>( \
|
||||||
IrOpcode::kWord32AtomicStore, \
|
IrOpcode::kWord32AtomicStore, \
|
||||||
Operator::kNoDeopt | Operator::kNoRead | Operator::kNoThrow, \
|
Operator::kNoDeopt | Operator::kNoRead | Operator::kNoThrow, \
|
||||||
"Word32AtomicStore", 3, 1, 1, 0, 1, 0, \
|
"Word32AtomicStore", 3, 1, 1, 0, 1, 0, \
|
||||||
AtomicStoreParameters(MachineRepresentation::Type, \
|
MachineRepresentation::Type) {} \
|
||||||
kNoWriteBarrier, \
|
|
||||||
AtomicMemoryOrder::kSeqCst)) {} \
|
|
||||||
}; \
|
}; \
|
||||||
Word32SeqCstStore##Type##Operator kWord32SeqCstStore##Type;
|
Word32AtomicStore##Type##Operator kWord32AtomicStore##Type;
|
||||||
ATOMIC_REPRESENTATION_LIST(ATOMIC_STORE)
|
ATOMIC_REPRESENTATION_LIST(ATOMIC_STORE)
|
||||||
#undef ATOMIC_STORE
|
#undef ATOMIC_STORE
|
||||||
|
|
||||||
#define ATOMIC_STORE(Type) \
|
#define ATOMIC_STORE(Type) \
|
||||||
struct Word64SeqCstStore##Type##Operator \
|
struct Word64AtomicStore##Type##Operator \
|
||||||
: public Operator1<AtomicStoreParameters> { \
|
: public Operator1<MachineRepresentation> { \
|
||||||
Word64SeqCstStore##Type##Operator() \
|
Word64AtomicStore##Type##Operator() \
|
||||||
: Operator1<AtomicStoreParameters>( \
|
: Operator1<MachineRepresentation>( \
|
||||||
IrOpcode::kWord64AtomicStore, \
|
IrOpcode::kWord64AtomicStore, \
|
||||||
Operator::kNoDeopt | Operator::kNoRead | Operator::kNoThrow, \
|
Operator::kNoDeopt | Operator::kNoRead | Operator::kNoThrow, \
|
||||||
"Word64AtomicStore", 3, 1, 1, 0, 1, 0, \
|
"Word64AtomicStore", 3, 1, 1, 0, 1, 0, \
|
||||||
AtomicStoreParameters(MachineRepresentation::Type, \
|
MachineRepresentation::Type) {} \
|
||||||
kNoWriteBarrier, \
|
|
||||||
AtomicMemoryOrder::kSeqCst)) {} \
|
|
||||||
}; \
|
}; \
|
||||||
Word64SeqCstStore##Type##Operator kWord64SeqCstStore##Type;
|
Word64AtomicStore##Type##Operator kWord64AtomicStore##Type;
|
||||||
ATOMIC64_REPRESENTATION_LIST(ATOMIC_STORE)
|
ATOMIC64_REPRESENTATION_LIST(ATOMIC_STORE)
|
||||||
#undef ATOMIC_STORE
|
#undef ATOMIC_STORE
|
||||||
|
|
||||||
@ -1167,23 +1075,21 @@ struct MachineOperatorGlobalCache {
|
|||||||
ATOMIC_U64_TYPE_LIST(ATOMIC_COMPARE_EXCHANGE)
|
ATOMIC_U64_TYPE_LIST(ATOMIC_COMPARE_EXCHANGE)
|
||||||
#undef ATOMIC_COMPARE_EXCHANGE
|
#undef ATOMIC_COMPARE_EXCHANGE
|
||||||
|
|
||||||
struct Word32SeqCstPairLoadOperator : public Operator1<AtomicMemoryOrder> {
|
struct Word32AtomicPairLoadOperator : public Operator {
|
||||||
Word32SeqCstPairLoadOperator()
|
Word32AtomicPairLoadOperator()
|
||||||
: Operator1<AtomicMemoryOrder>(IrOpcode::kWord32AtomicPairLoad,
|
: Operator(IrOpcode::kWord32AtomicPairLoad,
|
||||||
Operator::kNoDeopt | Operator::kNoThrow,
|
Operator::kNoDeopt | Operator::kNoThrow,
|
||||||
"Word32AtomicPairLoad", 2, 1, 1, 2, 1, 0,
|
"Word32AtomicPairLoad", 2, 1, 1, 2, 1, 0) {}
|
||||||
AtomicMemoryOrder::kSeqCst) {}
|
|
||||||
};
|
};
|
||||||
Word32SeqCstPairLoadOperator kWord32SeqCstPairLoad;
|
Word32AtomicPairLoadOperator kWord32AtomicPairLoad;
|
||||||
|
|
||||||
struct Word32SeqCstPairStoreOperator : public Operator1<AtomicMemoryOrder> {
|
struct Word32AtomicPairStoreOperator : public Operator {
|
||||||
Word32SeqCstPairStoreOperator()
|
Word32AtomicPairStoreOperator()
|
||||||
: Operator1<AtomicMemoryOrder>(IrOpcode::kWord32AtomicPairStore,
|
: Operator(IrOpcode::kWord32AtomicPairStore,
|
||||||
Operator::kNoDeopt | Operator::kNoThrow,
|
Operator::kNoDeopt | Operator::kNoThrow,
|
||||||
"Word32AtomicPairStore", 4, 1, 1, 0, 1,
|
"Word32AtomicPairStore", 4, 1, 1, 0, 1, 0) {}
|
||||||
0, AtomicMemoryOrder::kSeqCst) {}
|
|
||||||
};
|
};
|
||||||
Word32SeqCstPairStoreOperator kWord32SeqCstPairStore;
|
Word32AtomicPairStoreOperator kWord32AtomicPairStore;
|
||||||
|
|
||||||
#define ATOMIC_PAIR_OP(op) \
|
#define ATOMIC_PAIR_OP(op) \
|
||||||
struct Word32AtomicPair##op##Operator : public Operator { \
|
struct Word32AtomicPair##op##Operator : public Operator { \
|
||||||
@ -1643,47 +1549,23 @@ const Operator* MachineOperatorBuilder::MemBarrier() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
const Operator* MachineOperatorBuilder::Word32AtomicLoad(
|
const Operator* MachineOperatorBuilder::Word32AtomicLoad(
|
||||||
AtomicLoadParameters params) {
|
LoadRepresentation rep) {
|
||||||
#define CACHED_LOAD(Type) \
|
|
||||||
if (params.representation() == MachineType::Type() && \
|
|
||||||
params.order() == AtomicMemoryOrder::kSeqCst) { \
|
|
||||||
return &cache_.kWord32SeqCstLoad##Type; \
|
|
||||||
}
|
|
||||||
ATOMIC_TYPE_LIST(CACHED_LOAD)
|
|
||||||
#undef CACHED_LOAD
|
|
||||||
|
|
||||||
#define LOAD(Type) \
|
#define LOAD(Type) \
|
||||||
if (params.representation() == MachineType::Type()) { \
|
if (rep == MachineType::Type()) { \
|
||||||
return zone_->New<Operator1<AtomicLoadParameters>>( \
|
return &cache_.kWord32AtomicLoad##Type; \
|
||||||
IrOpcode::kWord32AtomicLoad, Operator::kEliminatable, \
|
|
||||||
"Word32AtomicLoad", 2, 1, 1, 1, 1, 0, params); \
|
|
||||||
}
|
}
|
||||||
ATOMIC_TYPE_LIST(LOAD)
|
ATOMIC_TYPE_LIST(LOAD)
|
||||||
ATOMIC_TAGGED_TYPE_LIST(LOAD)
|
|
||||||
#undef LOAD
|
#undef LOAD
|
||||||
|
|
||||||
UNREACHABLE();
|
UNREACHABLE();
|
||||||
}
|
}
|
||||||
|
|
||||||
const Operator* MachineOperatorBuilder::Word32AtomicStore(
|
const Operator* MachineOperatorBuilder::Word32AtomicStore(
|
||||||
AtomicStoreParameters params) {
|
MachineRepresentation rep) {
|
||||||
#define CACHED_STORE(kRep) \
|
|
||||||
if (params.representation() == MachineRepresentation::kRep && \
|
|
||||||
params.order() == AtomicMemoryOrder::kSeqCst) { \
|
|
||||||
return &cache_.kWord32SeqCstStore##kRep; \
|
|
||||||
}
|
|
||||||
ATOMIC_REPRESENTATION_LIST(CACHED_STORE)
|
|
||||||
#undef CACHED_STORE
|
|
||||||
|
|
||||||
#define STORE(kRep) \
|
#define STORE(kRep) \
|
||||||
if (params.representation() == MachineRepresentation::kRep) { \
|
if (rep == MachineRepresentation::kRep) { \
|
||||||
return zone_->New<Operator1<AtomicStoreParameters>>( \
|
return &cache_.kWord32AtomicStore##kRep; \
|
||||||
IrOpcode::kWord32AtomicStore, \
|
|
||||||
Operator::kNoDeopt | Operator::kNoRead | Operator::kNoThrow, \
|
|
||||||
"Word32AtomicStore", 3, 1, 1, 0, 1, 0, params); \
|
|
||||||
}
|
}
|
||||||
ATOMIC_REPRESENTATION_LIST(STORE)
|
ATOMIC_REPRESENTATION_LIST(STORE)
|
||||||
ATOMIC_TAGGED_REPRESENTATION_LIST(STORE)
|
|
||||||
#undef STORE
|
#undef STORE
|
||||||
UNREACHABLE();
|
UNREACHABLE();
|
||||||
}
|
}
|
||||||
@ -1760,49 +1642,24 @@ const Operator* MachineOperatorBuilder::Word32AtomicXor(MachineType type) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
const Operator* MachineOperatorBuilder::Word64AtomicLoad(
|
const Operator* MachineOperatorBuilder::Word64AtomicLoad(
|
||||||
AtomicLoadParameters params) {
|
LoadRepresentation rep) {
|
||||||
#define CACHED_LOAD(Type) \
|
|
||||||
if (params.representation() == MachineType::Type() && \
|
|
||||||
params.order() == AtomicMemoryOrder::kSeqCst) { \
|
|
||||||
return &cache_.kWord64SeqCstLoad##Type; \
|
|
||||||
}
|
|
||||||
ATOMIC_U64_TYPE_LIST(CACHED_LOAD)
|
|
||||||
#undef CACHED_LOAD
|
|
||||||
|
|
||||||
#define LOAD(Type) \
|
#define LOAD(Type) \
|
||||||
if (params.representation() == MachineType::Type()) { \
|
if (rep == MachineType::Type()) { \
|
||||||
return zone_->New<Operator1<AtomicLoadParameters>>( \
|
return &cache_.kWord64AtomicLoad##Type; \
|
||||||
IrOpcode::kWord64AtomicLoad, Operator::kEliminatable, \
|
|
||||||
"Word64AtomicLoad", 2, 1, 1, 1, 1, 0, params); \
|
|
||||||
}
|
}
|
||||||
ATOMIC_U64_TYPE_LIST(LOAD)
|
ATOMIC_U64_TYPE_LIST(LOAD)
|
||||||
ATOMIC64_TAGGED_TYPE_LIST(LOAD)
|
|
||||||
#undef LOAD
|
#undef LOAD
|
||||||
|
|
||||||
UNREACHABLE();
|
UNREACHABLE();
|
||||||
}
|
}
|
||||||
|
|
||||||
const Operator* MachineOperatorBuilder::Word64AtomicStore(
|
const Operator* MachineOperatorBuilder::Word64AtomicStore(
|
||||||
AtomicStoreParameters params) {
|
MachineRepresentation rep) {
|
||||||
#define CACHED_STORE(kRep) \
|
|
||||||
if (params.representation() == MachineRepresentation::kRep && \
|
|
||||||
params.order() == AtomicMemoryOrder::kSeqCst) { \
|
|
||||||
return &cache_.kWord64SeqCstStore##kRep; \
|
|
||||||
}
|
|
||||||
ATOMIC64_REPRESENTATION_LIST(CACHED_STORE)
|
|
||||||
#undef CACHED_STORE
|
|
||||||
|
|
||||||
#define STORE(kRep) \
|
#define STORE(kRep) \
|
||||||
if (params.representation() == MachineRepresentation::kRep) { \
|
if (rep == MachineRepresentation::kRep) { \
|
||||||
return zone_->New<Operator1<AtomicStoreParameters>>( \
|
return &cache_.kWord64AtomicStore##kRep; \
|
||||||
IrOpcode::kWord64AtomicStore, \
|
|
||||||
Operator::kNoDeopt | Operator::kNoRead | Operator::kNoThrow, \
|
|
||||||
"Word64AtomicStore", 3, 1, 1, 0, 1, 0, params); \
|
|
||||||
}
|
}
|
||||||
ATOMIC64_REPRESENTATION_LIST(STORE)
|
ATOMIC64_REPRESENTATION_LIST(STORE)
|
||||||
ATOMIC64_TAGGED_REPRESENTATION_LIST(STORE)
|
|
||||||
#undef STORE
|
#undef STORE
|
||||||
|
|
||||||
UNREACHABLE();
|
UNREACHABLE();
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1877,24 +1734,12 @@ const Operator* MachineOperatorBuilder::Word64AtomicCompareExchange(
|
|||||||
UNREACHABLE();
|
UNREACHABLE();
|
||||||
}
|
}
|
||||||
|
|
||||||
const Operator* MachineOperatorBuilder::Word32AtomicPairLoad(
|
const Operator* MachineOperatorBuilder::Word32AtomicPairLoad() {
|
||||||
AtomicMemoryOrder order) {
|
return &cache_.kWord32AtomicPairLoad;
|
||||||
if (order == AtomicMemoryOrder::kSeqCst) {
|
|
||||||
return &cache_.kWord32SeqCstPairLoad;
|
|
||||||
}
|
|
||||||
return zone_->New<Operator1<AtomicMemoryOrder>>(
|
|
||||||
IrOpcode::kWord32AtomicPairLoad, Operator::kNoDeopt | Operator::kNoThrow,
|
|
||||||
"Word32AtomicPairLoad", 2, 1, 1, 2, 1, 0, order);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
const Operator* MachineOperatorBuilder::Word32AtomicPairStore(
|
const Operator* MachineOperatorBuilder::Word32AtomicPairStore() {
|
||||||
AtomicMemoryOrder order) {
|
return &cache_.kWord32AtomicPairStore;
|
||||||
if (order == AtomicMemoryOrder::kSeqCst) {
|
|
||||||
return &cache_.kWord32SeqCstPairStore;
|
|
||||||
}
|
|
||||||
return zone_->New<Operator1<AtomicMemoryOrder>>(
|
|
||||||
IrOpcode::kWord32AtomicPairStore, Operator::kNoDeopt | Operator::kNoThrow,
|
|
||||||
"Word32AtomicPairStore", 4, 1, 1, 0, 1, 0, order);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
const Operator* MachineOperatorBuilder::Word32AtomicPairAdd() {
|
const Operator* MachineOperatorBuilder::Word32AtomicPairAdd() {
|
||||||
@ -2018,12 +1863,8 @@ StackCheckKind StackCheckKindOf(Operator const* op) {
|
|||||||
#undef ATOMIC_TYPE_LIST
|
#undef ATOMIC_TYPE_LIST
|
||||||
#undef ATOMIC_U64_TYPE_LIST
|
#undef ATOMIC_U64_TYPE_LIST
|
||||||
#undef ATOMIC_U32_TYPE_LIST
|
#undef ATOMIC_U32_TYPE_LIST
|
||||||
#undef ATOMIC_TAGGED_TYPE_LIST
|
|
||||||
#undef ATOMIC64_TAGGED_TYPE_LIST
|
|
||||||
#undef ATOMIC_REPRESENTATION_LIST
|
#undef ATOMIC_REPRESENTATION_LIST
|
||||||
#undef ATOMIC_TAGGED_REPRESENTATION_LIST
|
|
||||||
#undef ATOMIC64_REPRESENTATION_LIST
|
#undef ATOMIC64_REPRESENTATION_LIST
|
||||||
#undef ATOMIC64_TAGGED_REPRESENTATION_LIST
|
|
||||||
#undef SIMD_LANE_OP_LIST
|
#undef SIMD_LANE_OP_LIST
|
||||||
#undef STACK_SLOT_CACHED_SIZES_ALIGNMENTS_LIST
|
#undef STACK_SLOT_CACHED_SIZES_ALIGNMENTS_LIST
|
||||||
#undef LOAD_TRANSFORM_LIST
|
#undef LOAD_TRANSFORM_LIST
|
||||||
|
@ -8,7 +8,6 @@
|
|||||||
#include "src/base/compiler-specific.h"
|
#include "src/base/compiler-specific.h"
|
||||||
#include "src/base/enum-set.h"
|
#include "src/base/enum-set.h"
|
||||||
#include "src/base/flags.h"
|
#include "src/base/flags.h"
|
||||||
#include "src/codegen/atomic-memory-order.h"
|
|
||||||
#include "src/codegen/machine-type.h"
|
#include "src/codegen/machine-type.h"
|
||||||
#include "src/compiler/globals.h"
|
#include "src/compiler/globals.h"
|
||||||
#include "src/compiler/write-barrier-kind.h"
|
#include "src/compiler/write-barrier-kind.h"
|
||||||
@ -51,32 +50,6 @@ using LoadRepresentation = MachineType;
|
|||||||
V8_EXPORT_PRIVATE LoadRepresentation LoadRepresentationOf(Operator const*)
|
V8_EXPORT_PRIVATE LoadRepresentation LoadRepresentationOf(Operator const*)
|
||||||
V8_WARN_UNUSED_RESULT;
|
V8_WARN_UNUSED_RESULT;
|
||||||
|
|
||||||
// A Word(32|64)AtomicLoad needs both a LoadRepresentation and a memory
|
|
||||||
// order.
|
|
||||||
class AtomicLoadParameters final {
|
|
||||||
public:
|
|
||||||
AtomicLoadParameters(LoadRepresentation representation,
|
|
||||||
AtomicMemoryOrder order)
|
|
||||||
: representation_(representation), order_(order) {}
|
|
||||||
|
|
||||||
LoadRepresentation representation() const { return representation_; }
|
|
||||||
AtomicMemoryOrder order() const { return order_; }
|
|
||||||
|
|
||||||
private:
|
|
||||||
LoadRepresentation representation_;
|
|
||||||
AtomicMemoryOrder order_;
|
|
||||||
};
|
|
||||||
|
|
||||||
V8_EXPORT_PRIVATE bool operator==(AtomicLoadParameters, AtomicLoadParameters);
|
|
||||||
bool operator!=(AtomicLoadParameters, AtomicLoadParameters);
|
|
||||||
|
|
||||||
size_t hash_value(AtomicLoadParameters);
|
|
||||||
|
|
||||||
V8_EXPORT_PRIVATE std::ostream& operator<<(std::ostream&, AtomicLoadParameters);
|
|
||||||
|
|
||||||
V8_EXPORT_PRIVATE AtomicLoadParameters AtomicLoadParametersOf(Operator const*)
|
|
||||||
V8_WARN_UNUSED_RESULT;
|
|
||||||
|
|
||||||
enum class MemoryAccessKind {
|
enum class MemoryAccessKind {
|
||||||
kNormal,
|
kNormal,
|
||||||
kUnaligned,
|
kUnaligned,
|
||||||
@ -158,43 +131,6 @@ V8_EXPORT_PRIVATE std::ostream& operator<<(std::ostream&, StoreRepresentation);
|
|||||||
V8_EXPORT_PRIVATE StoreRepresentation const& StoreRepresentationOf(
|
V8_EXPORT_PRIVATE StoreRepresentation const& StoreRepresentationOf(
|
||||||
Operator const*) V8_WARN_UNUSED_RESULT;
|
Operator const*) V8_WARN_UNUSED_RESULT;
|
||||||
|
|
||||||
// A Word(32|64)AtomicStore needs both a StoreRepresentation and a memory order.
|
|
||||||
class AtomicStoreParameters final {
|
|
||||||
public:
|
|
||||||
AtomicStoreParameters(MachineRepresentation representation,
|
|
||||||
WriteBarrierKind write_barrier_kind,
|
|
||||||
AtomicMemoryOrder order)
|
|
||||||
: store_representation_(representation, write_barrier_kind),
|
|
||||||
order_(order) {}
|
|
||||||
|
|
||||||
MachineRepresentation representation() const {
|
|
||||||
return store_representation_.representation();
|
|
||||||
}
|
|
||||||
WriteBarrierKind write_barrier_kind() const {
|
|
||||||
return store_representation_.write_barrier_kind();
|
|
||||||
}
|
|
||||||
AtomicMemoryOrder order() const { return order_; }
|
|
||||||
|
|
||||||
StoreRepresentation store_representation() const {
|
|
||||||
return store_representation_;
|
|
||||||
}
|
|
||||||
|
|
||||||
private:
|
|
||||||
StoreRepresentation store_representation_;
|
|
||||||
AtomicMemoryOrder order_;
|
|
||||||
};
|
|
||||||
|
|
||||||
V8_EXPORT_PRIVATE bool operator==(AtomicStoreParameters, AtomicStoreParameters);
|
|
||||||
bool operator!=(AtomicStoreParameters, AtomicStoreParameters);
|
|
||||||
|
|
||||||
size_t hash_value(AtomicStoreParameters);
|
|
||||||
|
|
||||||
V8_EXPORT_PRIVATE std::ostream& operator<<(std::ostream&,
|
|
||||||
AtomicStoreParameters);
|
|
||||||
|
|
||||||
V8_EXPORT_PRIVATE AtomicStoreParameters const& AtomicStoreParametersOf(
|
|
||||||
Operator const*) V8_WARN_UNUSED_RESULT;
|
|
||||||
|
|
||||||
// An UnalignedStore needs a MachineType.
|
// An UnalignedStore needs a MachineType.
|
||||||
using UnalignedStoreRepresentation = MachineRepresentation;
|
using UnalignedStoreRepresentation = MachineRepresentation;
|
||||||
|
|
||||||
@ -237,6 +173,9 @@ V8_EXPORT_PRIVATE std::ostream& operator<<(std::ostream&,
|
|||||||
V8_EXPORT_PRIVATE StackSlotRepresentation const& StackSlotRepresentationOf(
|
V8_EXPORT_PRIVATE StackSlotRepresentation const& StackSlotRepresentationOf(
|
||||||
Operator const* op) V8_WARN_UNUSED_RESULT;
|
Operator const* op) V8_WARN_UNUSED_RESULT;
|
||||||
|
|
||||||
|
MachineRepresentation AtomicStoreRepresentationOf(Operator const* op)
|
||||||
|
V8_WARN_UNUSED_RESULT;
|
||||||
|
|
||||||
MachineType AtomicOpType(Operator const* op) V8_WARN_UNUSED_RESULT;
|
MachineType AtomicOpType(Operator const* op) V8_WARN_UNUSED_RESULT;
|
||||||
|
|
||||||
class S128ImmediateParameter {
|
class S128ImmediateParameter {
|
||||||
@ -956,13 +895,13 @@ class V8_EXPORT_PRIVATE MachineOperatorBuilder final
|
|||||||
const Operator* MemBarrier();
|
const Operator* MemBarrier();
|
||||||
|
|
||||||
// atomic-load [base + index]
|
// atomic-load [base + index]
|
||||||
const Operator* Word32AtomicLoad(AtomicLoadParameters params);
|
const Operator* Word32AtomicLoad(LoadRepresentation rep);
|
||||||
// atomic-load [base + index]
|
// atomic-load [base + index]
|
||||||
const Operator* Word64AtomicLoad(AtomicLoadParameters params);
|
const Operator* Word64AtomicLoad(LoadRepresentation rep);
|
||||||
// atomic-store [base + index], value
|
// atomic-store [base + index], value
|
||||||
const Operator* Word32AtomicStore(AtomicStoreParameters params);
|
const Operator* Word32AtomicStore(MachineRepresentation rep);
|
||||||
// atomic-store [base + index], value
|
// atomic-store [base + index], value
|
||||||
const Operator* Word64AtomicStore(AtomicStoreParameters params);
|
const Operator* Word64AtomicStore(MachineRepresentation rep);
|
||||||
// atomic-exchange [base + index], value
|
// atomic-exchange [base + index], value
|
||||||
const Operator* Word32AtomicExchange(MachineType type);
|
const Operator* Word32AtomicExchange(MachineType type);
|
||||||
// atomic-exchange [base + index], value
|
// atomic-exchange [base + index], value
|
||||||
@ -992,9 +931,9 @@ class V8_EXPORT_PRIVATE MachineOperatorBuilder final
|
|||||||
// atomic-xor [base + index], value
|
// atomic-xor [base + index], value
|
||||||
const Operator* Word64AtomicXor(MachineType type);
|
const Operator* Word64AtomicXor(MachineType type);
|
||||||
// atomic-pair-load [base + index]
|
// atomic-pair-load [base + index]
|
||||||
const Operator* Word32AtomicPairLoad(AtomicMemoryOrder order);
|
const Operator* Word32AtomicPairLoad();
|
||||||
// atomic-pair-sub [base + index], value_high, value-low
|
// atomic-pair-sub [base + index], value_high, value-low
|
||||||
const Operator* Word32AtomicPairStore(AtomicMemoryOrder order);
|
const Operator* Word32AtomicPairStore();
|
||||||
// atomic-pair-add [base + index], value_high, value_low
|
// atomic-pair-add [base + index], value_high, value_low
|
||||||
const Operator* Word32AtomicPairAdd();
|
const Operator* Word32AtomicPairAdd();
|
||||||
// atomic-pair-sub [base + index], value_high, value-low
|
// atomic-pair-sub [base + index], value_high, value-low
|
||||||
|
@ -239,20 +239,20 @@ class V8_EXPORT_PRIVATE RawMachineAssembler {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Atomic memory operations.
|
// Atomic memory operations.
|
||||||
Node* AtomicLoad(AtomicLoadParameters rep, Node* base, Node* index) {
|
Node* AtomicLoad(MachineType type, Node* base, Node* index) {
|
||||||
DCHECK_NE(rep.representation().representation(),
|
DCHECK_NE(type.representation(), MachineRepresentation::kWord64);
|
||||||
MachineRepresentation::kWord64);
|
return AddNode(machine()->Word32AtomicLoad(type), base, index);
|
||||||
return AddNode(machine()->Word32AtomicLoad(rep), base, index);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
Node* AtomicLoad64(AtomicLoadParameters rep, Node* base, Node* index) {
|
Node* AtomicLoad64(Node* base, Node* index) {
|
||||||
if (machine()->Is64()) {
|
if (machine()->Is64()) {
|
||||||
// This uses Uint64() intentionally: AtomicLoad is not implemented for
|
// This uses Uint64() intentionally: AtomicLoad is not implemented for
|
||||||
// Int64(), which is fine because the machine instruction only cares
|
// Int64(), which is fine because the machine instruction only cares
|
||||||
// about words.
|
// about words.
|
||||||
return AddNode(machine()->Word64AtomicLoad(rep), base, index);
|
return AddNode(machine()->Word64AtomicLoad(MachineType::Uint64()), base,
|
||||||
|
index);
|
||||||
} else {
|
} else {
|
||||||
return AddNode(machine()->Word32AtomicPairLoad(rep.order()), base, index);
|
return AddNode(machine()->Word32AtomicPairLoad(), base, index);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -262,24 +262,22 @@ class V8_EXPORT_PRIVATE RawMachineAssembler {
|
|||||||
#define VALUE_HALVES value, value_high
|
#define VALUE_HALVES value, value_high
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
Node* AtomicStore(AtomicStoreParameters params, Node* base, Node* index,
|
Node* AtomicStore(MachineRepresentation rep, Node* base, Node* index,
|
||||||
Node* value) {
|
Node* value) {
|
||||||
DCHECK(!IsMapOffsetConstantMinusTag(index));
|
DCHECK(!IsMapOffsetConstantMinusTag(index));
|
||||||
DCHECK_NE(params.representation(), MachineRepresentation::kWord64);
|
DCHECK_NE(rep, MachineRepresentation::kWord64);
|
||||||
return AddNode(machine()->Word32AtomicStore(params), base, index, value);
|
return AddNode(machine()->Word32AtomicStore(rep), base, index, value);
|
||||||
}
|
}
|
||||||
|
|
||||||
Node* AtomicStore64(AtomicStoreParameters params, Node* base, Node* index,
|
Node* AtomicStore64(Node* base, Node* index, Node* value, Node* value_high) {
|
||||||
Node* value, Node* value_high) {
|
|
||||||
if (machine()->Is64()) {
|
if (machine()->Is64()) {
|
||||||
DCHECK_NULL(value_high);
|
DCHECK_NULL(value_high);
|
||||||
return AddNode(machine()->Word64AtomicStore(params), base, index, value);
|
return AddNode(
|
||||||
|
machine()->Word64AtomicStore(MachineRepresentation::kWord64), base,
|
||||||
|
index, value);
|
||||||
} else {
|
} else {
|
||||||
DCHECK(params.representation() != MachineRepresentation::kTaggedPointer &&
|
return AddNode(machine()->Word32AtomicPairStore(), base, index,
|
||||||
params.representation() != MachineRepresentation::kTaggedSigned &&
|
VALUE_HALVES);
|
||||||
params.representation() != MachineRepresentation::kTagged);
|
|
||||||
return AddNode(machine()->Word32AtomicPairStore(params.order()), base,
|
|
||||||
index, VALUE_HALVES);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -5175,26 +5175,16 @@ Node* WasmGraphBuilder::AtomicOp(wasm::WasmOpcode opcode, Node* const* inputs,
|
|||||||
const Operator* (MachineOperatorBuilder::*)(MachineType);
|
const Operator* (MachineOperatorBuilder::*)(MachineType);
|
||||||
using OperatorByRep =
|
using OperatorByRep =
|
||||||
const Operator* (MachineOperatorBuilder::*)(MachineRepresentation);
|
const Operator* (MachineOperatorBuilder::*)(MachineRepresentation);
|
||||||
using OperatorByAtomicLoadRep =
|
|
||||||
const Operator* (MachineOperatorBuilder::*)(AtomicLoadParameters);
|
|
||||||
using OperatorByAtomicStoreRep =
|
|
||||||
const Operator* (MachineOperatorBuilder::*)(AtomicStoreParameters);
|
|
||||||
|
|
||||||
const Type type;
|
const Type type;
|
||||||
const MachineType machine_type;
|
const MachineType machine_type;
|
||||||
const OperatorByType operator_by_type = nullptr;
|
const OperatorByType operator_by_type = nullptr;
|
||||||
const OperatorByRep operator_by_rep = nullptr;
|
const OperatorByRep operator_by_rep = nullptr;
|
||||||
const OperatorByAtomicLoadRep operator_by_atomic_load_params = nullptr;
|
|
||||||
const OperatorByAtomicStoreRep operator_by_atomic_store_rep = nullptr;
|
|
||||||
|
|
||||||
constexpr AtomicOpInfo(Type t, MachineType m, OperatorByType o)
|
constexpr AtomicOpInfo(Type t, MachineType m, OperatorByType o)
|
||||||
: type(t), machine_type(m), operator_by_type(o) {}
|
: type(t), machine_type(m), operator_by_type(o) {}
|
||||||
constexpr AtomicOpInfo(Type t, MachineType m, OperatorByRep o)
|
constexpr AtomicOpInfo(Type t, MachineType m, OperatorByRep o)
|
||||||
: type(t), machine_type(m), operator_by_rep(o) {}
|
: type(t), machine_type(m), operator_by_rep(o) {}
|
||||||
constexpr AtomicOpInfo(Type t, MachineType m, OperatorByAtomicLoadRep o)
|
|
||||||
: type(t), machine_type(m), operator_by_atomic_load_params(o) {}
|
|
||||||
constexpr AtomicOpInfo(Type t, MachineType m, OperatorByAtomicStoreRep o)
|
|
||||||
: type(t), machine_type(m), operator_by_atomic_store_rep(o) {}
|
|
||||||
|
|
||||||
// Constexpr, hence just a table lookup in most compilers.
|
// Constexpr, hence just a table lookup in most compilers.
|
||||||
static constexpr AtomicOpInfo Get(wasm::WasmOpcode opcode) {
|
static constexpr AtomicOpInfo Get(wasm::WasmOpcode opcode) {
|
||||||
@ -5303,21 +5293,11 @@ Node* WasmGraphBuilder::AtomicOp(wasm::WasmOpcode opcode, Node* const* inputs,
|
|||||||
// {offset} is validated to be within uintptr_t range in {BoundsCheckMem}.
|
// {offset} is validated to be within uintptr_t range in {BoundsCheckMem}.
|
||||||
uintptr_t capped_offset = static_cast<uintptr_t>(offset);
|
uintptr_t capped_offset = static_cast<uintptr_t>(offset);
|
||||||
if (info.type != AtomicOpInfo::kSpecial) {
|
if (info.type != AtomicOpInfo::kSpecial) {
|
||||||
const Operator* op;
|
const Operator* op =
|
||||||
if (info.operator_by_type) {
|
info.operator_by_type
|
||||||
op = (mcgraph()->machine()->*info.operator_by_type)(info.machine_type);
|
? (mcgraph()->machine()->*info.operator_by_type)(info.machine_type)
|
||||||
} else if (info.operator_by_rep) {
|
: (mcgraph()->machine()->*info.operator_by_rep)(
|
||||||
op = (mcgraph()->machine()->*info.operator_by_rep)(
|
|
||||||
info.machine_type.representation());
|
info.machine_type.representation());
|
||||||
} else if (info.operator_by_atomic_load_params) {
|
|
||||||
op = (mcgraph()->machine()->*info.operator_by_atomic_load_params)(
|
|
||||||
AtomicLoadParameters(info.machine_type, AtomicMemoryOrder::kSeqCst));
|
|
||||||
} else {
|
|
||||||
op = (mcgraph()->machine()->*info.operator_by_atomic_store_rep)(
|
|
||||||
AtomicStoreParameters(info.machine_type.representation(),
|
|
||||||
WriteBarrierKind::kNoWriteBarrier,
|
|
||||||
AtomicMemoryOrder::kSeqCst));
|
|
||||||
}
|
|
||||||
|
|
||||||
Node* input_nodes[6] = {MemBuffer(capped_offset), index};
|
Node* input_nodes[6] = {MemBuffer(capped_offset), index};
|
||||||
int num_actual_inputs = info.type;
|
int num_actual_inputs = info.type;
|
||||||
|
@ -89,7 +89,6 @@ v8_source_set("cctest_sources") {
|
|||||||
"compiler/function-tester.cc",
|
"compiler/function-tester.cc",
|
||||||
"compiler/function-tester.h",
|
"compiler/function-tester.h",
|
||||||
"compiler/node-observer-tester.h",
|
"compiler/node-observer-tester.h",
|
||||||
"compiler/test-atomic-load-store-codegen.cc",
|
|
||||||
"compiler/test-basic-block-profiler.cc",
|
"compiler/test-basic-block-profiler.cc",
|
||||||
"compiler/test-branch-combine.cc",
|
"compiler/test-branch-combine.cc",
|
||||||
"compiler/test-calls-with-arraylike-or-spread.cc",
|
"compiler/test-calls-with-arraylike-or-spread.cc",
|
||||||
|
@ -623,7 +623,6 @@
|
|||||||
'codegen-tester/*': [SKIP],
|
'codegen-tester/*': [SKIP],
|
||||||
'test-accessor-assembler/*': [SKIP],
|
'test-accessor-assembler/*': [SKIP],
|
||||||
'test-assembler-*': [SKIP],
|
'test-assembler-*': [SKIP],
|
||||||
'test-atomic-load-store-codegen/*': [SKIP],
|
|
||||||
'test-basic-block-profiler/*': [SKIP],
|
'test-basic-block-profiler/*': [SKIP],
|
||||||
'test-branch-combine/*': [SKIP],
|
'test-branch-combine/*': [SKIP],
|
||||||
'test-calls-with-arraylike-or-spread/*': [SKIP],
|
'test-calls-with-arraylike-or-spread/*': [SKIP],
|
||||||
|
@ -1,398 +0,0 @@
|
|||||||
// Copyright 2021 the V8 project authors. All rights reserved. Use of this
|
|
||||||
// source code is governed by a BSD-style license that can be found in the
|
|
||||||
// LICENSE file.
|
|
||||||
|
|
||||||
#include "src/base/bits.h"
|
|
||||||
#include "src/objects/objects-inl.h"
|
|
||||||
#include "test/cctest/cctest.h"
|
|
||||||
#include "test/cctest/compiler/codegen-tester.h"
|
|
||||||
#include "test/cctest/compiler/value-helper.h"
|
|
||||||
|
|
||||||
namespace v8 {
|
|
||||||
namespace internal {
|
|
||||||
namespace compiler {
|
|
||||||
|
|
||||||
#if V8_TARGET_LITTLE_ENDIAN
|
|
||||||
#define LSB(addr, bytes) addr
|
|
||||||
#elif V8_TARGET_BIG_ENDIAN
|
|
||||||
#define LSB(addr, bytes) reinterpret_cast<byte*>(addr + 1) - (bytes)
|
|
||||||
#else
|
|
||||||
#error "Unknown Architecture"
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#define TEST_ATOMIC_LOAD_INTEGER(ctype, itype, mach_type, order) \
|
|
||||||
do { \
|
|
||||||
ctype buffer[1]; \
|
|
||||||
\
|
|
||||||
RawMachineAssemblerTester<ctype> m; \
|
|
||||||
Node* base = m.PointerConstant(&buffer[0]); \
|
|
||||||
Node* index = m.Int32Constant(0); \
|
|
||||||
AtomicLoadParameters params(mach_type, order); \
|
|
||||||
if (mach_type.MemSize() == 8) { \
|
|
||||||
m.Return(m.AtomicLoad64(params, base, index)); \
|
|
||||||
} else { \
|
|
||||||
m.Return(m.AtomicLoad(params, base, index)); \
|
|
||||||
} \
|
|
||||||
\
|
|
||||||
FOR_INPUTS(ctype, itype, i) { \
|
|
||||||
buffer[0] = i; \
|
|
||||||
CHECK_EQ(i, m.Call()); \
|
|
||||||
} \
|
|
||||||
} while (false)
|
|
||||||
|
|
||||||
TEST(AcquireLoadInteger) {
|
|
||||||
TEST_ATOMIC_LOAD_INTEGER(int8_t, int8, MachineType::Int8(),
|
|
||||||
AtomicMemoryOrder::kAcqRel);
|
|
||||||
TEST_ATOMIC_LOAD_INTEGER(uint8_t, uint8, MachineType::Uint8(),
|
|
||||||
AtomicMemoryOrder::kAcqRel);
|
|
||||||
TEST_ATOMIC_LOAD_INTEGER(int16_t, int16, MachineType::Int16(),
|
|
||||||
AtomicMemoryOrder::kAcqRel);
|
|
||||||
TEST_ATOMIC_LOAD_INTEGER(uint16_t, uint16, MachineType::Uint16(),
|
|
||||||
AtomicMemoryOrder::kAcqRel);
|
|
||||||
TEST_ATOMIC_LOAD_INTEGER(int32_t, int32, MachineType::Int32(),
|
|
||||||
AtomicMemoryOrder::kAcqRel);
|
|
||||||
TEST_ATOMIC_LOAD_INTEGER(uint32_t, uint32, MachineType::Uint32(),
|
|
||||||
AtomicMemoryOrder::kAcqRel);
|
|
||||||
#if V8_TARGET_ARCH_64_BIT
|
|
||||||
TEST_ATOMIC_LOAD_INTEGER(uint64_t, uint64, MachineType::Uint64(),
|
|
||||||
AtomicMemoryOrder::kAcqRel);
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
|
|
||||||
TEST(SeqCstLoadInteger) {
|
|
||||||
TEST_ATOMIC_LOAD_INTEGER(int8_t, int8, MachineType::Int8(),
|
|
||||||
AtomicMemoryOrder::kSeqCst);
|
|
||||||
TEST_ATOMIC_LOAD_INTEGER(uint8_t, uint8, MachineType::Uint8(),
|
|
||||||
AtomicMemoryOrder::kSeqCst);
|
|
||||||
TEST_ATOMIC_LOAD_INTEGER(int16_t, int16, MachineType::Int16(),
|
|
||||||
AtomicMemoryOrder::kSeqCst);
|
|
||||||
TEST_ATOMIC_LOAD_INTEGER(uint16_t, uint16, MachineType::Uint16(),
|
|
||||||
AtomicMemoryOrder::kSeqCst);
|
|
||||||
TEST_ATOMIC_LOAD_INTEGER(int32_t, int32, MachineType::Int32(),
|
|
||||||
AtomicMemoryOrder::kSeqCst);
|
|
||||||
TEST_ATOMIC_LOAD_INTEGER(uint32_t, uint32, MachineType::Uint32(),
|
|
||||||
AtomicMemoryOrder::kSeqCst);
|
|
||||||
#if V8_TARGET_ARCH_64_BIT
|
|
||||||
TEST_ATOMIC_LOAD_INTEGER(uint64_t, uint64, MachineType::Uint64(),
|
|
||||||
AtomicMemoryOrder::kSeqCst);
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
|
|
||||||
namespace {
|
|
||||||
// Mostly same as CHECK_EQ() but customized for compressed tagged values.
|
|
||||||
template <typename CType>
|
|
||||||
void CheckEq(CType in_value, CType out_value) {
|
|
||||||
CHECK_EQ(in_value, out_value);
|
|
||||||
}
|
|
||||||
|
|
||||||
#ifdef V8_COMPRESS_POINTERS
|
|
||||||
// Specializations for checking the result of compressing store.
|
|
||||||
template <>
|
|
||||||
void CheckEq<Object>(Object in_value, Object out_value) {
|
|
||||||
// Compare only lower 32-bits of the value because tagged load/stores are
|
|
||||||
// 32-bit operations anyway.
|
|
||||||
CHECK_EQ(static_cast<Tagged_t>(in_value.ptr()),
|
|
||||||
static_cast<Tagged_t>(out_value.ptr()));
|
|
||||||
}
|
|
||||||
|
|
||||||
template <>
|
|
||||||
void CheckEq<HeapObject>(HeapObject in_value, HeapObject out_value) {
|
|
||||||
return CheckEq<Object>(in_value, out_value);
|
|
||||||
}
|
|
||||||
|
|
||||||
template <>
|
|
||||||
void CheckEq<Smi>(Smi in_value, Smi out_value) {
|
|
||||||
return CheckEq<Object>(in_value, out_value);
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
|
|
||||||
template <typename TaggedT>
|
|
||||||
void InitBuffer(TaggedT* buffer, size_t length, MachineType type) {
|
|
||||||
const size_t kBufferSize = sizeof(TaggedT) * length;
|
|
||||||
|
|
||||||
// Tagged field loads require values to be properly tagged because of
|
|
||||||
// pointer decompression that may be happenning during load.
|
|
||||||
Isolate* isolate = CcTest::InitIsolateOnce();
|
|
||||||
Smi* smi_view = reinterpret_cast<Smi*>(&buffer[0]);
|
|
||||||
if (type.IsTaggedSigned()) {
|
|
||||||
for (size_t i = 0; i < length; i++) {
|
|
||||||
smi_view[i] = Smi::FromInt(static_cast<int>(i + kBufferSize) ^ 0xABCDEF0);
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
memcpy(&buffer[0], &isolate->roots_table(), kBufferSize);
|
|
||||||
if (!type.IsTaggedPointer()) {
|
|
||||||
// Also add some Smis if we are checking AnyTagged case.
|
|
||||||
for (size_t i = 0; i < length / 2; i++) {
|
|
||||||
smi_view[i] =
|
|
||||||
Smi::FromInt(static_cast<int>(i + kBufferSize) ^ 0xABCDEF0);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
template <typename TaggedT>
|
|
||||||
void AtomicLoadTagged(MachineType type, AtomicMemoryOrder order) {
|
|
||||||
const int kNumElems = 16;
|
|
||||||
TaggedT buffer[kNumElems];
|
|
||||||
|
|
||||||
InitBuffer(buffer, kNumElems, type);
|
|
||||||
|
|
||||||
for (int i = 0; i < kNumElems; i++) {
|
|
||||||
BufferedRawMachineAssemblerTester<TaggedT> m;
|
|
||||||
TaggedT* base_pointer = &buffer[0];
|
|
||||||
if (COMPRESS_POINTERS_BOOL) {
|
|
||||||
base_pointer = reinterpret_cast<TaggedT*>(LSB(base_pointer, kTaggedSize));
|
|
||||||
}
|
|
||||||
Node* base = m.PointerConstant(base_pointer);
|
|
||||||
Node* index = m.Int32Constant(i * sizeof(buffer[0]));
|
|
||||||
AtomicLoadParameters params(type, order);
|
|
||||||
Node* load;
|
|
||||||
if (kTaggedSize == 8) {
|
|
||||||
load = m.AtomicLoad64(params, base, index);
|
|
||||||
} else {
|
|
||||||
load = m.AtomicLoad(params, base, index);
|
|
||||||
}
|
|
||||||
m.Return(load);
|
|
||||||
CheckEq<TaggedT>(buffer[i], m.Call());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
} // namespace
|
|
||||||
|
|
||||||
TEST(AcquireLoadTagged) {
|
|
||||||
AtomicLoadTagged<Smi>(MachineType::TaggedSigned(),
|
|
||||||
AtomicMemoryOrder::kAcqRel);
|
|
||||||
AtomicLoadTagged<HeapObject>(MachineType::TaggedPointer(),
|
|
||||||
AtomicMemoryOrder::kAcqRel);
|
|
||||||
AtomicLoadTagged<Object>(MachineType::AnyTagged(),
|
|
||||||
AtomicMemoryOrder::kAcqRel);
|
|
||||||
}
|
|
||||||
|
|
||||||
TEST(SeqCstLoadTagged) {
|
|
||||||
AtomicLoadTagged<Smi>(MachineType::TaggedSigned(),
|
|
||||||
AtomicMemoryOrder::kSeqCst);
|
|
||||||
AtomicLoadTagged<HeapObject>(MachineType::TaggedPointer(),
|
|
||||||
AtomicMemoryOrder::kSeqCst);
|
|
||||||
AtomicLoadTagged<Object>(MachineType::AnyTagged(),
|
|
||||||
AtomicMemoryOrder::kSeqCst);
|
|
||||||
}
|
|
||||||
|
|
||||||
#define TEST_ATOMIC_STORE_INTEGER(ctype, itype, mach_type, order) \
|
|
||||||
do { \
|
|
||||||
ctype buffer[1]; \
|
|
||||||
buffer[0] = static_cast<ctype>(-1); \
|
|
||||||
\
|
|
||||||
BufferedRawMachineAssemblerTester<int32_t> m(mach_type); \
|
|
||||||
Node* value = m.Parameter(0); \
|
|
||||||
Node* base = m.PointerConstant(&buffer[0]); \
|
|
||||||
Node* index = m.Int32Constant(0); \
|
|
||||||
AtomicStoreParameters params(mach_type.representation(), kNoWriteBarrier, \
|
|
||||||
order); \
|
|
||||||
if (mach_type.MemSize() == 8) { \
|
|
||||||
m.AtomicStore64(params, base, index, value, nullptr); \
|
|
||||||
} else { \
|
|
||||||
m.AtomicStore(params, base, index, value); \
|
|
||||||
} \
|
|
||||||
\
|
|
||||||
int32_t OK = 0x29000; \
|
|
||||||
m.Return(m.Int32Constant(OK)); \
|
|
||||||
\
|
|
||||||
FOR_INPUTS(ctype, itype, i) { \
|
|
||||||
CHECK_EQ(OK, m.Call(i)); \
|
|
||||||
CHECK_EQ(i, buffer[0]); \
|
|
||||||
} \
|
|
||||||
} while (false)
|
|
||||||
|
|
||||||
TEST(ReleaseStoreInteger) {
|
|
||||||
TEST_ATOMIC_STORE_INTEGER(int8_t, int8, MachineType::Int8(),
|
|
||||||
AtomicMemoryOrder::kAcqRel);
|
|
||||||
TEST_ATOMIC_STORE_INTEGER(uint8_t, uint8, MachineType::Uint8(),
|
|
||||||
AtomicMemoryOrder::kAcqRel);
|
|
||||||
TEST_ATOMIC_STORE_INTEGER(int16_t, int16, MachineType::Int16(),
|
|
||||||
AtomicMemoryOrder::kAcqRel);
|
|
||||||
TEST_ATOMIC_STORE_INTEGER(uint16_t, uint16, MachineType::Uint16(),
|
|
||||||
AtomicMemoryOrder::kAcqRel);
|
|
||||||
TEST_ATOMIC_STORE_INTEGER(int32_t, int32, MachineType::Int32(),
|
|
||||||
AtomicMemoryOrder::kAcqRel);
|
|
||||||
TEST_ATOMIC_STORE_INTEGER(uint32_t, uint32, MachineType::Uint32(),
|
|
||||||
AtomicMemoryOrder::kAcqRel);
|
|
||||||
#if V8_TARGET_ARCH_64_BIT
|
|
||||||
TEST_ATOMIC_STORE_INTEGER(uint64_t, uint64, MachineType::Uint64(),
|
|
||||||
AtomicMemoryOrder::kAcqRel);
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
|
|
||||||
TEST(SeqCstStoreInteger) {
|
|
||||||
TEST_ATOMIC_STORE_INTEGER(int8_t, int8, MachineType::Int8(),
|
|
||||||
AtomicMemoryOrder::kSeqCst);
|
|
||||||
TEST_ATOMIC_STORE_INTEGER(uint8_t, uint8, MachineType::Uint8(),
|
|
||||||
AtomicMemoryOrder::kSeqCst);
|
|
||||||
TEST_ATOMIC_STORE_INTEGER(int16_t, int16, MachineType::Int16(),
|
|
||||||
AtomicMemoryOrder::kSeqCst);
|
|
||||||
TEST_ATOMIC_STORE_INTEGER(uint16_t, uint16, MachineType::Uint16(),
|
|
||||||
AtomicMemoryOrder::kSeqCst);
|
|
||||||
TEST_ATOMIC_STORE_INTEGER(int32_t, int32, MachineType::Int32(),
|
|
||||||
AtomicMemoryOrder::kSeqCst);
|
|
||||||
TEST_ATOMIC_STORE_INTEGER(uint32_t, uint32, MachineType::Uint32(),
|
|
||||||
AtomicMemoryOrder::kSeqCst);
|
|
||||||
#if V8_TARGET_ARCH_64_BIT
|
|
||||||
TEST_ATOMIC_STORE_INTEGER(uint64_t, uint64, MachineType::Uint64(),
|
|
||||||
AtomicMemoryOrder::kSeqCst);
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
|
|
||||||
namespace {
|
|
||||||
template <typename TaggedT>
|
|
||||||
void AtomicStoreTagged(MachineType type, AtomicMemoryOrder order) {
|
|
||||||
// This tests that tagged values are correctly transferred by atomic loads and
|
|
||||||
// stores from in_buffer to out_buffer. For each particular element in
|
|
||||||
// in_buffer, it is copied to a different index in out_buffer, and all other
|
|
||||||
// indices are zapped, to test instructions of the correct width are emitted.
|
|
||||||
|
|
||||||
const int kNumElems = 16;
|
|
||||||
TaggedT in_buffer[kNumElems];
|
|
||||||
TaggedT out_buffer[kNumElems];
|
|
||||||
uintptr_t zap_data[] = {kZapValue, kZapValue};
|
|
||||||
TaggedT zap_value;
|
|
||||||
|
|
||||||
STATIC_ASSERT(sizeof(TaggedT) <= sizeof(zap_data));
|
|
||||||
MemCopy(&zap_value, &zap_data, sizeof(TaggedT));
|
|
||||||
InitBuffer(in_buffer, kNumElems, type);
|
|
||||||
|
|
||||||
#ifdef V8_TARGET_BIG_ENDIAN
|
|
||||||
int offset = sizeof(TaggedT) - ElementSizeInBytes(type.representation());
|
|
||||||
#else
|
|
||||||
int offset = 0;
|
|
||||||
#endif
|
|
||||||
|
|
||||||
for (int32_t x = 0; x < kNumElems; x++) {
|
|
||||||
int32_t y = kNumElems - x - 1;
|
|
||||||
|
|
||||||
RawMachineAssemblerTester<int32_t> m;
|
|
||||||
int32_t OK = 0x29000 + x;
|
|
||||||
Node* in_base = m.PointerConstant(in_buffer);
|
|
||||||
Node* in_index = m.IntPtrConstant(x * sizeof(TaggedT) + offset);
|
|
||||||
Node* out_base = m.PointerConstant(out_buffer);
|
|
||||||
Node* out_index = m.IntPtrConstant(y * sizeof(TaggedT) + offset);
|
|
||||||
|
|
||||||
Node* load;
|
|
||||||
AtomicLoadParameters load_params(type, order);
|
|
||||||
AtomicStoreParameters store_params(type.representation(), kNoWriteBarrier,
|
|
||||||
order);
|
|
||||||
if (kTaggedSize == 4) {
|
|
||||||
load = m.AtomicLoad(load_params, in_base, in_index);
|
|
||||||
m.AtomicStore(store_params, out_base, out_index, load);
|
|
||||||
} else {
|
|
||||||
DCHECK(m.machine()->Is64());
|
|
||||||
load = m.AtomicLoad64(load_params, in_base, in_index);
|
|
||||||
m.AtomicStore64(store_params, out_base, out_index, load, nullptr);
|
|
||||||
}
|
|
||||||
|
|
||||||
m.Return(m.Int32Constant(OK));
|
|
||||||
|
|
||||||
for (int32_t z = 0; z < kNumElems; z++) {
|
|
||||||
out_buffer[z] = zap_value;
|
|
||||||
}
|
|
||||||
CHECK_NE(in_buffer[x], out_buffer[y]);
|
|
||||||
CHECK_EQ(OK, m.Call());
|
|
||||||
// Mostly same as CHECK_EQ() but customized for compressed tagged values.
|
|
||||||
CheckEq<TaggedT>(in_buffer[x], out_buffer[y]);
|
|
||||||
for (int32_t z = 0; z < kNumElems; z++) {
|
|
||||||
if (z != y) CHECK_EQ(zap_value, out_buffer[z]);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
} // namespace
|
|
||||||
|
|
||||||
TEST(ReleaseStoreTagged) {
|
|
||||||
AtomicStoreTagged<Smi>(MachineType::TaggedSigned(),
|
|
||||||
AtomicMemoryOrder::kAcqRel);
|
|
||||||
AtomicStoreTagged<HeapObject>(MachineType::TaggedPointer(),
|
|
||||||
AtomicMemoryOrder::kAcqRel);
|
|
||||||
AtomicStoreTagged<Object>(MachineType::AnyTagged(),
|
|
||||||
AtomicMemoryOrder::kAcqRel);
|
|
||||||
}
|
|
||||||
|
|
||||||
TEST(SeqCstStoreTagged) {
|
|
||||||
AtomicStoreTagged<Smi>(MachineType::TaggedSigned(),
|
|
||||||
AtomicMemoryOrder::kSeqCst);
|
|
||||||
AtomicStoreTagged<HeapObject>(MachineType::TaggedPointer(),
|
|
||||||
AtomicMemoryOrder::kSeqCst);
|
|
||||||
AtomicStoreTagged<Object>(MachineType::AnyTagged(),
|
|
||||||
AtomicMemoryOrder::kSeqCst);
|
|
||||||
}
|
|
||||||
|
|
||||||
#if V8_TARGET_ARCH_32_BIT
|
|
||||||
|
|
||||||
namespace {
|
|
||||||
void TestAtomicPairLoadInteger(AtomicMemoryOrder order) {
|
|
||||||
uint64_t buffer[1];
|
|
||||||
uint32_t high;
|
|
||||||
uint32_t low;
|
|
||||||
|
|
||||||
BufferedRawMachineAssemblerTester<int32_t> m;
|
|
||||||
Node* base = m.PointerConstant(&buffer[0]);
|
|
||||||
Node* index = m.Int32Constant(0);
|
|
||||||
|
|
||||||
Node* pair_load = m.AtomicLoad64(
|
|
||||||
AtomicLoadParameters(MachineType::Uint64(), order), base, index);
|
|
||||||
m.StoreToPointer(&low, MachineRepresentation::kWord32,
|
|
||||||
m.Projection(0, pair_load));
|
|
||||||
m.StoreToPointer(&high, MachineRepresentation::kWord32,
|
|
||||||
m.Projection(1, pair_load));
|
|
||||||
|
|
||||||
int32_t OK = 0x29000;
|
|
||||||
m.Return(m.Int32Constant(OK));
|
|
||||||
|
|
||||||
FOR_UINT64_INPUTS(i) {
|
|
||||||
buffer[0] = i;
|
|
||||||
CHECK_EQ(OK, m.Call());
|
|
||||||
CHECK_EQ(i, make_uint64(high, low));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
} // namespace
|
|
||||||
|
|
||||||
TEST(AcquirePairLoadInteger) {
|
|
||||||
TestAtomicPairLoadInteger(AtomicMemoryOrder::kAcqRel);
|
|
||||||
}
|
|
||||||
|
|
||||||
TEST(SeqCstPairLoadInteger) {
|
|
||||||
TestAtomicPairLoadInteger(AtomicMemoryOrder::kSeqCst);
|
|
||||||
}
|
|
||||||
|
|
||||||
namespace {
|
|
||||||
void TestAtomicPairStoreInteger(AtomicMemoryOrder order) {
|
|
||||||
uint64_t buffer[1];
|
|
||||||
|
|
||||||
BufferedRawMachineAssemblerTester<int32_t> m(MachineType::Uint32(),
|
|
||||||
MachineType::Uint32());
|
|
||||||
Node* base = m.PointerConstant(&buffer[0]);
|
|
||||||
Node* index = m.Int32Constant(0);
|
|
||||||
|
|
||||||
m.AtomicStore64(AtomicStoreParameters(MachineRepresentation::kWord64,
|
|
||||||
kNoWriteBarrier, order),
|
|
||||||
base, index, m.Parameter(0), m.Parameter(1));
|
|
||||||
|
|
||||||
int32_t OK = 0x29000;
|
|
||||||
m.Return(m.Int32Constant(OK));
|
|
||||||
|
|
||||||
FOR_UINT64_INPUTS(i) {
|
|
||||||
CHECK_EQ(OK, m.Call(static_cast<uint32_t>(i & 0xFFFFFFFF),
|
|
||||||
static_cast<uint32_t>(i >> 32)));
|
|
||||||
CHECK_EQ(i, buffer[0]);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
} // namespace
|
|
||||||
|
|
||||||
TEST(ReleasePairStoreInteger) {
|
|
||||||
TestAtomicPairStoreInteger(AtomicMemoryOrder::kAcqRel);
|
|
||||||
}
|
|
||||||
|
|
||||||
TEST(SeqCstPairStoreInteger) {
|
|
||||||
TestAtomicPairStoreInteger(AtomicMemoryOrder::kSeqCst);
|
|
||||||
}
|
|
||||||
|
|
||||||
#endif // V8_TARGET_ARCH_32_BIT
|
|
||||||
|
|
||||||
} // namespace compiler
|
|
||||||
} // namespace internal
|
|
||||||
} // namespace v8
|
|
Loading…
Reference in New Issue
Block a user