mirror of
https://github.com/saymrwulf/pytorch.git
synced 2026-05-14 20:57:59 +00:00
Summary: Change fx graph module's _replace_hook from a single hook, to a list of hooks. This is to prepare to registering more hooks for inductor provenance tracking, where we might need to register multiple hooks for node replacement. Test Plan: ``` buck run mode/dev-nosan caffe2/test:fx -- -r test_hooks_for_node_update buck run mode/dev-nosan caffe2/test:test_export -- -r test_replace_hook ``` Differential Revision: D66726724 Pull Request resolved: https://github.com/pytorch/pytorch/pull/142006 Approved by: https://github.com/zhxchen17
91 lines
3.1 KiB
Python
91 lines
3.1 KiB
Python
# Owner(s): ["module: fx"]
|
|
import torch
|
|
from torch.fx import symbolic_trace
|
|
from torch.testing._internal.common_utils import TestCase
|
|
|
|
|
|
class TestFXNodeHook(TestCase):
|
|
def test_hooks_for_node_update(self):
|
|
global create_node_hook1_called
|
|
global create_node_hook2_called
|
|
global erase_node_hook1_called
|
|
global erase_node_hook2_called
|
|
global replace_node_hook1_called
|
|
global replace_node_hook2_called
|
|
create_node_hook1_called = False
|
|
create_node_hook2_called = False
|
|
erase_node_hook1_called = False
|
|
erase_node_hook2_called = False
|
|
replace_node_hook1_called = False
|
|
replace_node_hook2_called = False
|
|
|
|
def fn(a, b, c):
|
|
x = torch.nn.functional.linear(a, b)
|
|
x = x + c
|
|
return x.cos()
|
|
|
|
def create_node_hook1(node):
|
|
global create_node_hook1_called
|
|
create_node_hook1_called = True
|
|
|
|
def create_node_hook2(node):
|
|
global create_node_hook2_called
|
|
create_node_hook2_called = True
|
|
|
|
def erase_node_hook1(node):
|
|
global erase_node_hook1_called
|
|
erase_node_hook1_called = True
|
|
|
|
def erase_node_hook2(node):
|
|
global erase_node_hook2_called
|
|
erase_node_hook2_called = True
|
|
|
|
def replace_node_hook1(old, new, user):
|
|
global replace_node_hook1_called
|
|
self.assertEqual(old.name, "a")
|
|
self.assertEqual(new, "a_1")
|
|
self.assertEqual(user.name, "linear")
|
|
replace_node_hook1_called = True
|
|
|
|
def replace_node_hook2(old, new, user):
|
|
global replace_node_hook2_called
|
|
replace_node_hook2_called = True
|
|
|
|
gm = symbolic_trace(fn)
|
|
gm._register_create_node_hook(create_node_hook1)
|
|
gm._register_create_node_hook(create_node_hook2)
|
|
gm._register_erase_node_hook(erase_node_hook1)
|
|
gm._register_erase_node_hook(erase_node_hook2)
|
|
gm._register_replace_node_hook(replace_node_hook1)
|
|
gm._register_replace_node_hook(replace_node_hook2)
|
|
|
|
graph = gm.graph
|
|
node_a = None
|
|
for node in graph.find_nodes(op="placeholder"):
|
|
node_a = node
|
|
break
|
|
assert node_a is not None
|
|
# This will create a new node
|
|
node_a_copy = graph.node_copy(node_a)
|
|
node_a.replace_all_uses_with(node_a_copy)
|
|
graph.erase_node(node_a)
|
|
|
|
assert (
|
|
create_node_hook1_called
|
|
and create_node_hook2_called
|
|
and erase_node_hook1_called
|
|
and erase_node_hook2_called
|
|
and replace_node_hook1_called
|
|
and replace_node_hook2_called
|
|
)
|
|
|
|
gm._unregister_create_node_hook(create_node_hook1)
|
|
gm._unregister_create_node_hook(create_node_hook2)
|
|
gm._unregister_erase_node_hook(erase_node_hook1)
|
|
gm._unregister_erase_node_hook(erase_node_hook2)
|
|
gm._unregister_replace_node_hook(replace_node_hook1)
|
|
gm._unregister_replace_node_hook(replace_node_hook2)
|
|
|
|
assert gm._create_node_hooks == []
|
|
assert gm._erase_node_hooks == []
|
|
assert gm._replace_hooks == []
|