[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:
parent
ee16640ef6
commit
6c6deee7ea
@ -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());
|
||||||
}
|
}
|
||||||
|
@ -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());
|
||||||
}
|
}
|
||||||
|
@ -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) {
|
||||||
|
@ -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) {
|
||||||
|
@ -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 =
|
||||||
|
@ -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);
|
||||||
|
|
||||||
|
@ -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(
|
||||||
|
@ -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);
|
||||||
|
@ -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);
|
||||||
|
@ -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,
|
||||||
|
Loading…
Reference in New Issue
Block a user