Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Added support for QONNX Resize node ingestion and tested with tiny UNet model #1122

Merged
merged 21 commits into from
Nov 21, 2024
Merged
Show file tree
Hide file tree
Changes from 14 commits
Commits
Show all changes
21 commits
Select commit Hold shift + click to select a range
a761da9
Added support for `Resize` node from QONNX model
nghielme Nov 12, 2024
b62468a
Added a test on tiny UNet model in order to test `Resize` node
nghielme Nov 12, 2024
40a431f
pre-commit restyling
nghielme Nov 12, 2024
be55945
Aesthetic fix
nghielme Nov 12, 2024
743831f
Second aesthetic fix
nghielme Nov 12, 2024
aa46dbe
Merge branch 'main' into resize_pr
nghielme Nov 12, 2024
4f82810
Added one test on a simpler model extracted from UNet model `branched…
nghielme Nov 12, 2024
7e6b9af
Example models commit updated
nghielme Nov 12, 2024
5757ac6
An empty list is now appended to the shape of all the inputs of the c…
nghielme Nov 12, 2024
5e13800
Merge branch 'main' into resize_pr
nghielme Nov 14, 2024
cf80f64
Cleaned some code and added the removal of RoI input from `Resize` node
nghielme Nov 15, 2024
c7f6983
Merge branch 'resize_pr' of https://github.com/fastmachinelearning/hl…
nghielme Nov 15, 2024
a5e32c5
Merge branch 'main' into resize_pr
nghielme Nov 15, 2024
b07e998
revert some unneeded changes
jmitrevs Nov 16, 2024
354b535
Added some minor checks related to sizes parameter
nghielme Nov 18, 2024
3b5f8db
Merge branch 'resize_pr' of https://github.com/fastmachinelearning/hl…
nghielme Nov 18, 2024
3254942
Merge branch 'main' into resize_pr
nghielme Nov 18, 2024
9943350
Minor fix
nghielme Nov 18, 2024
5ff517b
Minor modification of the error msg
nghielme Nov 19, 2024
6a10129
Minor fixes
nghielme Nov 20, 2024
20ab44f
Merge branch 'main' into resize_pr
nghielme Nov 21, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
2 changes: 1 addition & 1 deletion example-models
24 changes: 23 additions & 1 deletion hls4ml/converters/onnx/reshape.py
Original file line number Diff line number Diff line change
@@ -1,4 +1,4 @@
from hls4ml.converters.onnx_to_hls import onnx_handler
from hls4ml.converters.onnx_to_hls import get_onnx_attribute, onnx_handler


@onnx_handler('Transpose')
Expand Down Expand Up @@ -36,3 +36,25 @@ def parse_flatten_layer(node, input_names, input_shapes, graph):
layer['target_shape'] = [-1] # does not contain batch dimension

return layer


@onnx_handler('Resize')
def parse_resize_layer(node, input_names, input_shapes, graph):
layer = {}
layer['name'] = node.name
layer['class_name'] = 'Resize'
layer['inputs'] = input_names
layer['outputs'] = list(node.output)
layer['in_height'] = input_shapes[0][2]
layer['in_width'] = input_shapes[0][1]
layer['out_width'] = input_shapes[0][1]
layer['out_height'] = input_shapes[0][2]
layer['n_chan'] = input_shapes[0][3]
layer['algorithm'] = get_onnx_attribute(node, 'mode')
# The following is used in initialize() method.
# Probably a better solution would be to have a channels last parameter at QONNX level
layer['data_format'] = (
'channels_last' if any(node.domain == 'qonnx.custom_op.channels_last' for node in graph.node) else 'channels_first'
)

return layer
63 changes: 50 additions & 13 deletions hls4ml/model/layers.py
Original file line number Diff line number Diff line change
Expand Up @@ -1147,20 +1147,57 @@ class Resize(Layer):
def initialize(self):
inp = self.get_input_variable()

if self.get_attr('data_format') == 'channels_last':
if len(inp.shape) == 2: # 1D -> width + chan
shape = [self.get_attr('out_width'), self.get_attr('n_chan')]
dims = [f'OUT_WIDTH_{self.index}', f'N_CHAN_{self.index}']
elif len(inp.shape) == 3: # 2D -> height + width + chan
shape = [self.get_attr('out_height'), self.get_attr('out_width'), self.get_attr('n_chan')]
dims = [f'OUT_HEIGHT_{self.index}', f'OUT_WIDTH_{self.index}', f'N_CHAN_{self.index}']
if len(self.inputs) > 1:
# get the scales of Resize node from QONNX frontend
# see doc here: https://onnx.ai/onnx/operators/onnx__Resize.html
scales_idx = 2 if len(self.inputs) == 3 or len(self.inputs) == 4 else 1
jmitrevs marked this conversation as resolved.
Show resolved Hide resolved
scales = self.get_input_node(self.inputs[scales_idx]).get_attr('value')
if len(scales) == 4: # Resize 2D
self.set_attr('out_width', int(self.get_attr('in_width') * scales[1]))
self.set_attr('out_height', int(self.get_attr('in_height') * scales[2]))
self.set_attr('n_chan', int(self.get_attr('n_chan') * scales[3]))
elif len(scales) == 3: # Resize 1D
self.set_attr('out_width', int(self.get_attr('in_width') * scales[1]))
self.set_attr('n_chan', int(self.get_attr('n_chan') * scales[2]))
else:
raise Exception('Resize 1D and Resize 2D are the ones supported in hls4ml')
if self.get_attr('data_format') == 'channels_last':
if len(inp.shape) == 2: # 1D -> width + chan
shape = [int(self.get_attr('out_width')), int(self.get_attr('n_chan'))]
dims = [f'OUT_WIDTH_{self.index}', f'N_CHAN_{self.index}']
elif len(inp.shape) == 3: # 2D -> height + width + chan
shape = [
int(self.get_attr('out_height')),
int(self.get_attr('out_width')),
int(self.get_attr('n_chan')),
]
dims = [f'OUT_HEIGHT_{self.index}', f'OUT_WIDTH_{self.index}', f'N_CHAN_{self.index}']
else:
if len(inp.shape) == 2: # 1D -> width + chan
shape = [int(self.get_attr('n_chan')), int(self.get_attr('out_width'))]
dims = [f'N_CHAN_{self.index}', f'OUT_WIDTH_{self.index}']
elif len(inp.shape) == 3: # 2D -> height + width + chan
shape = [
int(self.get_attr('n_chan')),
int(self.get_attr('out_height')),
int(self.get_attr('out_width')),
]
dims = [f'N_CHAN_{self.index}', f'OUT_HEIGHT_{self.index}', f'OUT_WIDTH_{self.index}']
else:
if len(inp.shape) == 2: # 1D -> width + chan
shape = [self.get_attr('n_chan'), self.get_attr('out_width')]
dims = [f'N_CHAN_{self.index}', f'OUT_WIDTH_{self.index}']
elif len(inp.shape) == 3: # 2D -> height + width + chan
shape = [self.get_attr('n_chan'), self.get_attr('out_height'), self.get_attr('out_width')]
dims = [f'N_CHAN_{self.index}', f'OUT_HEIGHT_{self.index}', f'OUT_WIDTH_{self.index}']
if self.get_attr('data_format') == 'channels_last':
if len(inp.shape) == 2: # 1D -> width + chan
shape = [self.get_attr('out_width'), self.get_attr('n_chan')]
dims = [f'OUT_WIDTH_{self.index}', f'N_CHAN_{self.index}']
elif len(inp.shape) == 3: # 2D -> height + width + chan
shape = [self.get_attr('out_height'), self.get_attr('out_width'), self.get_attr('n_chan')]
dims = [f'OUT_HEIGHT_{self.index}', f'OUT_WIDTH_{self.index}', f'N_CHAN_{self.index}']
else:
if len(inp.shape) == 2: # 1D -> width + chan
shape = [self.get_attr('n_chan'), self.get_attr('out_width')]
dims = [f'N_CHAN_{self.index}', f'OUT_WIDTH_{self.index}']
elif len(inp.shape) == 3: # 2D -> height + width + chan
shape = [self.get_attr('n_chan'), self.get_attr('out_height'), self.get_attr('out_width')]
dims = [f'N_CHAN_{self.index}', f'OUT_HEIGHT_{self.index}', f'OUT_WIDTH_{self.index}']

self.add_output_variable(shape, dims, precision=inp.type.precision)

Expand Down
1 change: 1 addition & 0 deletions hls4ml/model/optimizer/__init__.py
Original file line number Diff line number Diff line change
Expand Up @@ -34,6 +34,7 @@
'parse_qonnx',
[
'reshape_constant',
'resize_remove_constants',
'quant_constant_parameters',
'quant_to_activation',
'fuse_quant_with_constant',
Expand Down
1 change: 0 additions & 1 deletion hls4ml/model/optimizer/passes/linear.py
Original file line number Diff line number Diff line change
Expand Up @@ -40,7 +40,6 @@ def transform(self, model, node):
# if the activation has a quantizer (usually from a QONNX Quant node), set the previous node's output precision
if quantizer is not None:
prev_node.set_attr("quantizer", quantizer)
prev_node.types['result_t'] = quantizer.hls_type
prev_node.get_output_variable().type.precision = quantizer.hls_type
model.remove_node(node)
return True
42 changes: 42 additions & 0 deletions hls4ml/model/optimizer/passes/resize_remove_constants.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,42 @@
from hls4ml.model.layers import Constant, Resize
from hls4ml.model.optimizer import OptimizerPass


class ResizeRemoveConstants(OptimizerPass):
"""
This optimizer is intended to clean the Resize node from RoI and Scales parameters that if left cause issues in hls4ml.
"""

def match(self, node):
is_match = isinstance(node, Resize) and len(node.inputs) > 1
return is_match

def transform(self, model, node):
"""
Remove RoI and Scale Constant from new shape input.
"""
# see doc here: https://onnx.ai/onnx/operators/onnx__Resize.html
scales_idx = 2 if len(node.inputs) == 3 or len(node.inputs) == 4 else 1
scales_node = node.get_input_node(node.inputs[scales_idx])
node.inputs[scales_idx] = ''
if not isinstance(scales_node, Constant):
raise RuntimeError("Non-constant shape inputs are not supported")
model.remove_node(scales_node, rewire=False)
if len(node.inputs) >= 3 and node.inputs[1] != '':
# RoI is present only if more than 3 inputs are specified
# RoI position is always 1 when present
roi_node = node.get_input_node(node.inputs[1])
node.inputs[1] = ''
if not isinstance(roi_node, Constant):
jmitrevs marked this conversation as resolved.
Show resolved Hide resolved
raise RuntimeError("Non-constant RoI inputs are not supported")
model.remove_node(roi_node, rewire=False)
if len(node.inputs) == 4:
jmitrevs marked this conversation as resolved.
Show resolved Hide resolved
# Remove sizes node
sizes_node = node.get_input_node(node.inputs[-1])
node.inputs[-1] = ''
if not isinstance(sizes_node, Constant):
raise RuntimeError("Non-constant RoI inputs are not supported")
model.remove_node(sizes_node, rewire=False)
# Clean all the '' inputs
node.inputs = list(filter(None, node.inputs))
return True
78 changes: 78 additions & 0 deletions test/pytest/test_qonnx.py
Original file line number Diff line number Diff line change
Expand Up @@ -101,6 +101,32 @@ def sep_conv_model():
return model


@pytest.fixture(scope='module')
def branched_model():
"""
Load branched model using separable convs, already channels-last and cleaned
"""
dl_file = str(example_model_path / "onnx/branched_model_ch_last.onnx")
assert os.path.isfile(dl_file)

model = ModelWrapper(dl_file)

return model


@pytest.fixture(scope='module')
def tiny_unet_model():
"""
Load tiny unet model, already channels-last and cleaned
"""
dl_file = str(example_model_path / "onnx/tiny_unet_ch_last.onnx")
assert os.path.isfile(dl_file)

model = ModelWrapper(dl_file)

return model


@pytest.fixture(scope='module')
def two_layer_keras_model():
"""
Expand Down Expand Up @@ -309,6 +335,58 @@ def test_sep_conv(sep_conv_model, backend):
np.testing.assert_allclose(y_qonnx.ravel(), y_hls4ml.ravel(), atol=1e-2, rtol=1)


@pytest.mark.parametrize('backend', ['Vitis'])
def test_branched_model(branched_model, backend):
model = branched_model
ishape = tuple(model.get_tensor_shape(model.graph.input[0].name))
X = np.random.uniform(low=0, high=1, size=np.prod(ishape)).reshape(ishape)
X = (np.round(X * 2**16) * 2**-16).astype(np.float32)
idict = {model.graph.input[0].name: X}
y_qonnx = oxe.execute_onnx(model, idict)[model.graph.output[0].name]

config = hls4ml.utils.config.config_from_onnx_model(
model, granularity='name', backend=backend, default_precision='fixed<32,16>'
)
hls_model = hls4ml.converters.convert_from_onnx_model(
model,
output_dir=str(test_root_path / f'hls4mlprj_qonnx_branched_model_{backend}'),
io_type='io_stream',
backend=backend,
hls_config=config,
)
hls_model.compile()
y_hls4ml = hls_model.predict(np.ascontiguousarray(X))

np.testing.assert_array_equal(y_qonnx.ravel(), y_hls4ml.ravel())


@pytest.mark.parametrize('backend', ['Vitis'])
def test_tiny_unet_model(tiny_unet_model, backend):

model = tiny_unet_model
ishape = tuple(model.get_tensor_shape(model.graph.input[0].name))
X = np.random.uniform(low=0, high=1, size=np.prod(ishape)).reshape(ishape)
X = (np.round(X * 2**16) * 2**-16).astype(np.float32)
idict = {model.graph.input[0].name: X}
y_qonnx = oxe.execute_onnx(model, idict)[model.graph.output[0].name]

config = hls4ml.utils.config.config_from_onnx_model(
model, granularity='name', backend=backend, default_precision='fixed<32,16>'
)

hls_model = hls4ml.converters.convert_from_onnx_model(
model,
output_dir=str(test_root_path / f'hls4mlprj_qonnx_tiny_unet_model_{backend}'),
io_type='io_stream',
backend=backend,
hls_config=config,
)
hls_model.compile()
y_hls4ml = hls_model.predict(np.ascontiguousarray(X))

np.testing.assert_array_equal(y_qonnx.ravel(), y_hls4ml.ravel())


@pytest.mark.parametrize(
'model_name',
[
Expand Down
Loading