[compiler] Support AcqRel in MemoryBarrier and expose to CodeAssembler

AcqRel barriers are currently unused and will be used by the shared
value barrier in a future CL.

Bug: v8:12547
Change-Id: I8ae40b9e17f007441125dfa5d0a04f46565785fd
Reviewed-on: https://chromium-review.googlesource.com/c/v8/v8/+/3827319
Reviewed-by: Tobias Tebbi <tebbi@chromium.org>
Commit-Queue: Shu-yu Guo <syg@chromium.org>
Cr-Commit-Position: refs/heads/main@{#82568}
This commit is contained in:
Shu-yu Guo 2022-08-17 14:14:03 -07:00 committed by V8 LUCI CQ
parent ee16640ef6
commit 6c6deee7ea
10 changed files with 50 additions and 15 deletions

View File

@ -2274,6 +2274,7 @@ void InstructionSelector::VisitFloat64InsertHighWord32(Node* node) {
} }
void InstructionSelector::VisitMemoryBarrier(Node* node) { void InstructionSelector::VisitMemoryBarrier(Node* node) {
// Use DMB ISH for both acquire-release and sequentially consistent barriers.
ArmOperandGenerator g(this); ArmOperandGenerator g(this);
Emit(kArmDmbIsh, g.NoOutput()); Emit(kArmDmbIsh, g.NoOutput());
} }

View File

@ -3387,6 +3387,7 @@ void InstructionSelector::VisitFloat64Mul(Node* node) {
} }
void InstructionSelector::VisitMemoryBarrier(Node* node) { void InstructionSelector::VisitMemoryBarrier(Node* node) {
// Use DMB ISH for both acquire-release and sequentially consistent barriers.
Arm64OperandGenerator g(this); Arm64OperandGenerator g(this);
Emit(kArm64DmbIsh, g.NoOutput()); Emit(kArm64DmbIsh, g.NoOutput());
} }

View File

@ -2018,8 +2018,15 @@ void InstructionSelector::VisitFloat64SilenceNaN(Node* node) {
} }
void InstructionSelector::VisitMemoryBarrier(Node* node) { void InstructionSelector::VisitMemoryBarrier(Node* node) {
IA32OperandGenerator g(this); // ia32 is no weaker than release-acquire and only needs to emit an
Emit(kIA32MFence, g.NoOutput()); // instruction for SeqCst memory barriers.
AtomicMemoryOrder order = OpParameter<AtomicMemoryOrder>(node->op());
if (order == AtomicMemoryOrder::kSeqCst) {
IA32OperandGenerator g(this);
Emit(kIA32MFence, g.NoOutput());
return;
}
DCHECK_EQ(AtomicMemoryOrder::kAcqRel, order);
} }
void InstructionSelector::VisitWord32AtomicLoad(Node* node) { void InstructionSelector::VisitWord32AtomicLoad(Node* node) {

View File

@ -3055,8 +3055,15 @@ void InstructionSelector::VisitFloat64SilenceNaN(Node* node) {
} }
void InstructionSelector::VisitMemoryBarrier(Node* node) { void InstructionSelector::VisitMemoryBarrier(Node* node) {
X64OperandGenerator g(this); // x64 is no weaker than release-acquire and only needs to emit an instruction
Emit(kX64MFence, g.NoOutput()); // for SeqCst memory barriers.
AtomicMemoryOrder order = OpParameter<AtomicMemoryOrder>(node->op());
if (order == AtomicMemoryOrder::kSeqCst) {
X64OperandGenerator g(this);
Emit(kX64MFence, g.NoOutput());
return;
}
DCHECK_EQ(AtomicMemoryOrder::kAcqRel, order);
} }
void InstructionSelector::VisitWord32AtomicLoad(Node* node) { void InstructionSelector::VisitWord32AtomicLoad(Node* node) {

View File

@ -944,6 +944,10 @@ CodeAssembler::AtomicCompareExchange64<AtomicUint64>(
TNode<UintPtrT> new_value, TNode<UintPtrT> old_value_high, TNode<UintPtrT> new_value, TNode<UintPtrT> old_value_high,
TNode<UintPtrT> new_value_high); TNode<UintPtrT> new_value_high);
void CodeAssembler::MemoryBarrier(AtomicMemoryOrder order) {
raw_assembler()->MemoryBarrier(order);
}
void CodeAssembler::StoreRoot(RootIndex root_index, TNode<Object> value) { void CodeAssembler::StoreRoot(RootIndex root_index, TNode<Object> value) {
DCHECK(!RootsTable::IsImmortalImmovable(root_index)); DCHECK(!RootsTable::IsImmortalImmovable(root_index));
TNode<ExternalReference> isolate_root = TNode<ExternalReference> isolate_root =

View File

@ -862,6 +862,8 @@ class V8_EXPORT_PRIVATE CodeAssembler {
TNode<UintPtrT> old_value_high, TNode<UintPtrT> old_value_high,
TNode<UintPtrT> new_value_high); TNode<UintPtrT> new_value_high);
void MemoryBarrier(AtomicMemoryOrder order);
// Store a value to the root array. // Store a value to the root array.
void StoreRoot(RootIndex root_index, TNode<Object> value); void StoreRoot(RootIndex root_index, TNode<Object> value);

View File

@ -1293,13 +1293,15 @@ struct MachineOperatorGlobalCache {
}; };
Word32AtomicPairCompareExchangeOperator kWord32AtomicPairCompareExchange; Word32AtomicPairCompareExchangeOperator kWord32AtomicPairCompareExchange;
struct MemoryBarrierOperator : public Operator { template <AtomicMemoryOrder order>
struct MemoryBarrierOperator : public Operator1<AtomicMemoryOrder> {
MemoryBarrierOperator() MemoryBarrierOperator()
: Operator(IrOpcode::kMemoryBarrier, : Operator1<AtomicMemoryOrder>(
Operator::kNoDeopt | Operator::kNoThrow, "MemoryBarrier", 0, IrOpcode::kMemoryBarrier, Operator::kNoDeopt | Operator::kNoThrow,
1, 1, 0, 1, 0) {} "SeqCstMemoryBarrier", 0, 1, 1, 0, 1, 0, order) {}
}; };
MemoryBarrierOperator kMemoryBarrier; MemoryBarrierOperator<AtomicMemoryOrder::kSeqCst> kSeqCstMemoryBarrier;
MemoryBarrierOperator<AtomicMemoryOrder::kAcqRel> kAcqRelMemoryBarrier;
// The {BitcastWordToTagged} operator must not be marked as pure (especially // The {BitcastWordToTagged} operator must not be marked as pure (especially
// not idempotent), because otherwise the splitting logic in the Scheduler // not idempotent), because otherwise the splitting logic in the Scheduler
@ -1734,8 +1736,15 @@ const Operator* MachineOperatorBuilder::Comment(const char* msg) {
return zone_->New<CommentOperator>(msg); return zone_->New<CommentOperator>(msg);
} }
const Operator* MachineOperatorBuilder::MemBarrier() { const Operator* MachineOperatorBuilder::MemoryBarrier(AtomicMemoryOrder order) {
return &cache_.kMemoryBarrier; switch (order) {
case AtomicMemoryOrder::kSeqCst:
return &cache_.kSeqCstMemoryBarrier;
case AtomicMemoryOrder::kAcqRel:
return &cache_.kAcqRelMemoryBarrier;
default:
UNREACHABLE();
}
} }
const Operator* MachineOperatorBuilder::Word32AtomicLoad( const Operator* MachineOperatorBuilder::Word32AtomicLoad(

View File

@ -1003,8 +1003,7 @@ class V8_EXPORT_PRIVATE MachineOperatorBuilder final
// Runtime::kStackGuardWithGap call. // Runtime::kStackGuardWithGap call.
const Operator* LoadStackCheckOffset(); const Operator* LoadStackCheckOffset();
// Memory barrier. const Operator* MemoryBarrier(AtomicMemoryOrder order);
const Operator* MemBarrier();
// atomic-load [base + index] // atomic-load [base + index]
const Operator* Word32AtomicLoad(AtomicLoadParameters params); const Operator* Word32AtomicLoad(AtomicLoadParameters params);

View File

@ -335,6 +335,10 @@ class V8_EXPORT_PRIVATE RawMachineAssembler {
} }
} }
Node* MemoryBarrier(AtomicMemoryOrder order) {
return AddNode(machine()->MemoryBarrier(order));
}
// Arithmetic Operations. // Arithmetic Operations.
Node* WordAnd(Node* a, Node* b) { Node* WordAnd(Node* a, Node* b) {
return AddNode(machine()->WordAnd(), a, b); return AddNode(machine()->WordAnd(), a, b);

View File

@ -5009,8 +5009,9 @@ Node* WasmGraphBuilder::AtomicOp(wasm::WasmOpcode opcode, Node* const* inputs,
} }
void WasmGraphBuilder::AtomicFence() { void WasmGraphBuilder::AtomicFence() {
SetEffect(graph()->NewNode(mcgraph()->machine()->MemBarrier(), effect(), SetEffect(graph()->NewNode(
control())); mcgraph()->machine()->MemoryBarrier(AtomicMemoryOrder::kSeqCst), effect(),
control()));
} }
void WasmGraphBuilder::MemoryInit(uint32_t data_segment_index, Node* dst, void WasmGraphBuilder::MemoryInit(uint32_t data_segment_index, Node* dst,