Skip to content

Commit

Permalink
Look deeper in a node's source position to detect the Truffle languag…
Browse files Browse the repository at this point in the history
…e being used.
  • Loading branch information
nirvdrum committed Mar 25, 2024
1 parent 1b726bd commit e96af0f
Show file tree
Hide file tree
Showing 3 changed files with 74 additions and 61 deletions.
1 change: 1 addition & 0 deletions CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -8,6 +8,7 @@ Bug Fixes:

Changes:

* Improve Truffle language detection by looking deeper into a node's source position (@nirvdrum).

# 0.16

Expand Down
10 changes: 10 additions & 0 deletions lib/seafoam/passes/truffle_translators/translators.rb
Original file line number Diff line number Diff line change
Expand Up @@ -21,6 +21,16 @@ def get_translator(node)
subpackage = declaring_class[:declaring_class].match(/^(\w+\.\w+)\./)[1]
translator = TRUFFLE_LANGUAGES[subpackage]

unless translator
Seafoam::Graal::Source.walk(entry.props.dig("nodeSourcePosition")) do |method|
declaring_class = method[:declaring_class]
subpackage = declaring_class.match(/^(\w+\.\w+)\./)[1]
translator = TRUFFLE_LANGUAGES[subpackage]

break if translator
end
end

break const_get(translator).new if translator
end
end
Expand Down
124 changes: 63 additions & 61 deletions spec/seafoam/passes/truffle_spec.rb
Original file line number Diff line number Diff line change
Expand Up @@ -50,82 +50,84 @@
end

describe "with :simplify_truffle_args" do
before :each do
@filename = File.expand_path(
"../../../examples/#{graalvm_version}/ruby/example_polymorphic_receiver.bgv.gz",
__dir__,
)
@phase_index = "After TruffleTier"
@graph = Seafoam::SpecHelpers.example_graph(@filename, @phase_index)
end

it "substitutes TruffleArgument nodes for LoadIndexed nodes" do
before_load_indexed_nodes = @graph.nodes.values.select do |n|
n.node_class.end_with?(".compiler.nodes.java.LoadIndexedNode")
end.map(&:dup)
before_truffle_argument_nodes = @graph.nodes.values.select do |n|
n.props[:synthetic_class] == "TruffleArgument"
["example_polymorphic_receiver", "example_arith_operator"].each do |example|
before :each do
@filename = File.expand_path(
"../../../examples/#{graalvm_version}/ruby/#{example}.bgv.gz",
__dir__,
)
@phase_index = "After TruffleTier"
@graph = Seafoam::SpecHelpers.example_graph(@filename, @phase_index)
end

# Verify initial graph state.
expect(before_load_indexed_nodes).not_to(be_empty)
expect(before_truffle_argument_nodes).to(be_empty)
it "substitutes TruffleArgument nodes for LoadIndexed nodes" do
before_load_indexed_nodes = @graph.nodes.values.select do |n|
n.node_class.end_with?(".compiler.nodes.java.LoadIndexedNode")
end.map(&:dup)
before_truffle_argument_nodes = @graph.nodes.values.select do |n|
n.props[:synthetic_class] == "TruffleArgument"
end

# Run the Truffle argument simplification pass.
graph = Seafoam::SpecHelpers.example_graph(@filename, @phase_index)
pass = Seafoam::Passes::TrufflePass.new(simplify_truffle_args: true, hide_null_fields: true)
pass.apply(graph)
# Verify initial graph state.
expect(before_load_indexed_nodes).not_to(be_empty)
expect(before_truffle_argument_nodes).to(be_empty)

# Verify graph state after the pass has run.
after_load_indexed_nodes = graph.nodes.values.select do |n|
n.node_class.end_with?(".compiler.nodes.java.LoadIndexedNode")
end
after_truffle_argument_nodes = graph.nodes.values.select do |n|
n.props[:synthetic_class] == "TruffleArgument"
end
# Run the Truffle argument simplification pass.
graph = Seafoam::SpecHelpers.example_graph(@filename, @phase_index)
pass = Seafoam::Passes::TrufflePass.new(simplify_truffle_args: true, hide_null_fields: true)
pass.apply(graph)

# Verify state after the pass has run. This test only makes sense if the TruffleArgument node list is
# non-empty.
expect(after_load_indexed_nodes.size).to(eq(before_load_indexed_nodes.size))
expect(after_truffle_argument_nodes).not_to(be_empty)
# Verify graph state after the pass has run.
after_load_indexed_nodes = graph.nodes.values.select do |n|
n.node_class.end_with?(".compiler.nodes.java.LoadIndexedNode")
end
after_truffle_argument_nodes = graph.nodes.values.select do |n|
n.props[:synthetic_class] == "TruffleArgument"
end

after_truffle_argument_nodes.each_with_index do |arg_node, index|
# The node_class should be the synthetic class value.
expect(arg_node.node_class).to(eq("TruffleArgument"))
# Verify state after the pass has run. This test only makes sense if the TruffleArgument node list is
# non-empty.
expect(after_load_indexed_nodes.size).to(eq(before_load_indexed_nodes.size))
expect(after_truffle_argument_nodes).not_to(be_empty)

# The "next" edges are never connected to the TruffleArgument node.
rewritten_edges = before_load_indexed_nodes[index].outputs.select { |edge| edge.props[:name] != "next" }
after_truffle_argument_nodes.each_with_index do |arg_node, index|
# The node_class should be the synthetic class value.
expect(arg_node.node_class).to(eq("TruffleArgument"))

# The original LoadIndexed nodes should have no outgoing edges.
expect(after_load_indexed_nodes[index].outputs).to(be_empty)
# The "next" edges are never connected to the TruffleArgument node.
rewritten_edges = before_load_indexed_nodes[index].outputs.select { |edge| edge.props[:name] != "next" }

# The new TruffleArgument nodes should have no incoming edges.
expect(arg_node.inputs).to(be_empty)
# The original LoadIndexed nodes should have no outgoing edges.
expect(after_load_indexed_nodes[index].outputs).to(be_empty)

# The new TruffleArgument nodes should have edges pointing to the same edges the original LoadIndexed nodes
# did, minus the "next" nodes.
expect(arg_node.outputs.map(&:to).map(&:id)).to(eq(rewritten_edges.map(&:to).map(&:id)))
# The new TruffleArgument nodes should have no incoming edges.
expect(arg_node.inputs).to(be_empty)

# The new TruffleArgument nodes should have edges pointing to the same edges the original LoadIndexed
# nodes did, minus the "next" nodes.
expect(arg_node.outputs.map(&:to).map(&:id)).to(eq(rewritten_edges.map(&:to).map(&:id)))
end
end
end

it "uses symbolic Truffle argument names for TruffleRuby" do
# Run the Truffle argument simplification pass.
pass = Seafoam::Passes::TrufflePass.new(simplify_truffle_args: true, hide_null_fields: true)
pass.apply(@graph)
it "uses symbolic Truffle argument names for TruffleRuby" do
# Run the Truffle argument simplification pass.
pass = Seafoam::Passes::TrufflePass.new(simplify_truffle_args: true, hide_null_fields: true)
pass.apply(@graph)

# Verify graph state after the pass has run.
after_truffle_argument_nodes = @graph.nodes.values.select do |n|
n.props[:synthetic_class] == "TruffleArgument"
end
expect(after_truffle_argument_nodes).not_to(be_empty)
# Verify graph state after the pass has run.
after_truffle_argument_nodes = @graph.nodes.values.select do |n|
n.props[:synthetic_class] == "TruffleArgument"
end
expect(after_truffle_argument_nodes).not_to(be_empty)

truffleruby_arg_pattern = Regexp.union(
*Seafoam::Passes::TruffleTranslators::TruffleRuby::TRUFFLERUBY_ARGS,
/args\[\d+\]/,
)
truffleruby_arg_pattern = Regexp.union(
*Seafoam::Passes::TruffleTranslators::TruffleRuby::TRUFFLERUBY_ARGS,
/args\[\d+\]/,
)

after_truffle_argument_nodes.each do |node|
expect(node.props[:label]).to(match(truffleruby_arg_pattern))
after_truffle_argument_nodes.each do |node|
expect(node.props[:label]).to(match(truffleruby_arg_pattern))
end
end
end
end
Expand Down

0 comments on commit e96af0f

Please sign in to comment.