Skip to content

Commit

Permalink
fixes
Browse files Browse the repository at this point in the history
  • Loading branch information
irenaby committed Nov 24, 2024
1 parent e4adf42 commit 4f40c03
Show file tree
Hide file tree
Showing 4 changed files with 38 additions and 34 deletions.
Original file line number Diff line number Diff line change
Expand Up @@ -31,7 +31,7 @@
from model_compression_toolkit.constants import REUSE, REUSE_GROUP
from model_compression_toolkit.core.keras.constants import KERNEL, BIAS, USE_BIAS, FILTERS, PADDING, \
KERNEL_SIZE, DEPTH_MULTIPLIER, STRIDES, DILATIONS, DILATION_RATE, DEPTHWISE_KERNEL, RATE, \
ACTIVATION, LINEAR, DATA_FORMAT, GROUPS
ACTIVATION, LINEAR, DATA_FORMAT, GROUPS, CHANNELS_FORMAT_FIRST, CHANNELS_FORMAT_LAST


def extract_bias_node_data(_node: FunctionalNode, _graph: Graph) -> np.ndarray:
Expand Down Expand Up @@ -138,20 +138,19 @@ def substitute(self,
if len(conv_func_node.op_call_args) > 0:
Logger.critical(f"node {conv_func_node.name} expected to have only kwargs but got args={conv_func_node.op_call_args}.") # pragma: no cover

strides = self._parse_tf_2d_kwarg(conv_func_node, STRIDES)
strides = self._parse_tf_stride_dilation(conv_func_node, STRIDES)
if strides is None:
# Non-standard strides -> skip substitution.
return graph
conv_fw_attr[STRIDES] = strides

if PADDING in conv_func_node.op_call_kwargs:
padding = conv_func_node.op_call_kwargs[PADDING]
if not isinstance(padding, str):
# Non-standard padding, Layer only support either 'valid' or 'same' -> skip substitution.
return graph # pragma: no cover
conv_fw_attr[PADDING] = padding
padding = conv_func_node.op_call_kwargs.get(PADDING) or 'VALID'
if not isinstance(padding, str):
# Non-standard padding, Layer only support either 'valid' or 'same' -> skip substitution.
return graph # pragma: no cover
conv_fw_attr[PADDING] = padding

dilations = self._parse_tf_2d_kwarg(conv_func_node, DILATIONS)
dilations = self._parse_tf_stride_dilation(conv_func_node, DILATIONS)
if dilations is None:
# Non-standard dilations -> skip substitution.
return graph
Expand All @@ -163,7 +162,7 @@ def substitute(self,
weights[BIAS] = b

data_format = conv_func_node.op_call_kwargs.get(DATA_FORMAT, 'NHWC')
conv_fw_attr['data_format'] = {'NHWC': 'channels_last', 'NCHW': 'channels_first'}[data_format]
conv_fw_attr[DATA_FORMAT] = {'NHWC': CHANNELS_FORMAT_LAST, 'NCHW': CHANNELS_FORMAT_FIRST}[data_format]

conv_fw_attr[GROUPS] = 1

Expand All @@ -174,7 +173,7 @@ def substitute(self,
replace_conv_node(graph, conv_node, conv_func_node, remove_add_node=b is not None)
return graph

def _parse_tf_2d_kwarg(self, node, key) -> Optional[Tuple[int, int]]:
def _parse_tf_stride_dilation(self, node, key) -> Optional[Tuple[int, int]]:
"""
Extract stride/dilation param from tf node and convert it to keras format (suitable for Conv2D).
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -39,6 +39,9 @@

class ConvFuncSubstitutionsTest(BaseKerasFeatureNetworkTest):

def __init__(self, unit_test):
super().__init__(unit_test, input_shape=(32, 32, 3))

def get_tpc(self):
tp = generate_test_tp_model({'enable_weights_quantization': False,
'enable_activation_quantization': False})
Expand Down Expand Up @@ -67,6 +70,18 @@ def create_networks(self):
x = tf.nn.convolution(x, np.random.random((3, 3, 2, 4)).astype(np.float32),
[2, 1], padding='SAME')
x = tf.nn.bias_add(x, np.random.random((4,)).astype(np.float32))

# default values and various formats
x = tf.nn.conv2d(x, np.random.random((3, 3, 4, 8)), 1, 'VALID')
x = tf.nn.conv2d(x, np.random.random((3, 3, 8, 16)), strides=[1], padding='SAME', dilations=1)
x = tf.nn.conv2d(x, np.random.random((3, 3, 16, 8)), strides=[1, 1], padding='VALID', dilations=[1])
x = tf.nn.conv2d(x, filters=np.random.random((3, 3, 8, 4)), strides=[1, 1], padding='SAME', dilations=[1, 1])

x = tf.nn.convolution(x, np.random.random((3, 3, 4, 16)).astype(np.float32))
x = tf.nn.convolution(x, np.random.random((3, 3, 16, 32)).astype(np.float32), strides=[1], padding='SAME', dilations=1)
x = tf.nn.convolution(x, np.random.random((3, 3, 32, 8)).astype(np.float32), strides=[1, 1], padding='VALID', dilations=[1])
x = tf.nn.convolution(x, filters=np.random.random((3, 3, 8, 4)).astype(np.float32), strides=[1, 1], padding='VALID', dilations=[1, 1])

return tf.keras.Model(inputs=_in, outputs=x)

def compare(self, quantized_model, float_model, input_x=None, quantization_info=None):
Expand All @@ -75,7 +90,7 @@ def compare(self, quantized_model, float_model, input_x=None, quantization_info=
cs = cosine_similarity(out_float.numpy(), out_quant.numpy())
self.unit_test.assertTrue(np.isclose(cs, 1), msg=f'fail cosine similarity check: {cs}')

self.unit_test.assertTrue(len(get_layers_from_model_by_type(quantized_model, Conv2D)) == 4,
self.unit_test.assertTrue(len(get_layers_from_model_by_type(quantized_model, Conv2D)) == 12,
"Not all conv functions were substituted.")
self.unit_test.assertTrue(len(get_layers_from_model_by_type(quantized_model, DepthwiseConv2D)) == 2,
"Not all dw-conv functions were substituted.")
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -116,20 +116,23 @@ def compare(self, quantized_model, float_model, input_x=None, quantization_info=
self.unit_test.assertTrue(len(layer.weights) == 2,msg=f'fail Bias should appear in weights!!')


class FuncConv2DCollapsingTest(FourConv2DCollapsingTest):
class FuncConvCollapsingTest(FourConv2DCollapsingTest):
def create_networks(self):
# tests the combination of functional conv to Conv2D substitution with linear collapsing
# Tests the combination of functional conv to Conv2D substitution with linear collapsing
# (in case of default values, tf layer doesn't contain these attributes, and they must be added explicitly
# to node's attributes dict, which is not covered by substitution test)
h, w, c = self.get_input_shapes()[0][1:]
inputs = layers.Input(shape=(h, w, c))
x = tf.nn.conv2d(inputs, tf.random.uniform((3, 3, c, 128)), 1, 'VALID')
x = tf.nn.conv2d(x, filters=tf.random.uniform((1, 1, 128, 64)), strides=[1], padding='SAME', dilations=1)
x = tf.nn.conv2d(x, tf.random.uniform((1, 1, 64, 64)), strides=[1, 1], padding='VALID', dilations=[1])
y = tf.nn.conv2d(x, tf.random.uniform((1, 1, 64, 4)), strides=[1, 1], padding='SAME', dilations=[1, 1])
x = tf.nn.conv2d(inputs, tf.random.uniform((3, 3, c, 16)), 1, 'SAME')
x = tf.nn.convolution(x, tf.random.uniform((1, 1, 16, 8)))
x = tf.nn.relu(x)
x = tf.nn.convolution(x, tf.random.uniform((3, 3, 8, 32)))
y = tf.nn.conv2d(x, tf.random.uniform((1, 1, 32, 4)), 1, 'VALID')
return tf.keras.models.Model(inputs=inputs, outputs=y)

def compare(self, quantized_model, float_model, input_x=None, quantization_info=None):
convs = [l for l in quantized_model.layers if isinstance(l, layers.Conv2D)]
self.unit_test.assertTrue(len(convs) == 1)
self.unit_test.assertTrue(len(convs) == 2)

y = float_model.predict(input_x)
y_hat = quantized_model.predict(input_x)
Expand All @@ -138,18 +141,6 @@ def compare(self, quantized_model, float_model, input_x=None, quantization_info=
self.unit_test.assertTrue(np.isclose(cs, 1), msg=f'fail cosine similarity check:{cs}')


class FuncConvolutionCollapsingTest(FuncConv2DCollapsingTest):
def create_networks(self):
# tests the combination of functional conv to Conv2D substitution with linear collapsing
h, w, c = self.get_input_shapes()[0][1:]
inputs = layers.Input(shape=(h, w, c))
x = tf.nn.convolution(inputs, tf.random.uniform((3, 3, c, 128)))
x = tf.nn.convolution(x, filters=tf.random.uniform((1, 1, 128, 64)), strides=[1], padding='SAME', dilations=1)
x = tf.nn.convolution(x, tf.random.uniform((1, 1, 64, 64)), strides=[1, 1], padding='VALID', dilations=[1])
y = tf.nn.convolution(x, tf.random.uniform((1, 1, 64, 4)), strides=[1, 1], padding='VALID', dilations=[1, 1])
return tf.keras.models.Model(inputs=inputs, outputs=y)


class SixConv2DCollapsingTest(BaseConv2DCollapsingTest):
def __init__(self, unit_test):
super().__init__(unit_test)
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -57,7 +57,7 @@
InputScalingConvTest, InputScalingDWTest, InputScalingZeroPadTest
from tests.keras_tests.feature_networks_tests.feature_networks.linear_collapsing_test import TwoConv2DCollapsingTest, \
ThreeConv2DCollapsingTest, FourConv2DCollapsingTest, SixConv2DCollapsingTest, Op2DAddConstCollapsingTest, \
FuncConv2DCollapsingTest, FuncConvolutionCollapsingTest
FuncConvCollapsingTest
from tests.keras_tests.feature_networks_tests.feature_networks.lut_quantizer import LUTWeightsQuantizerTest, \
LUTActivationQuantizerTest
from tests.keras_tests.feature_networks_tests.feature_networks.manual_bit_selection import ManualBitWidthSelectionTest, \
Expand Down Expand Up @@ -606,8 +606,7 @@ def test_linear_collapsing(self):
FourConv2DCollapsingTest(self).run_test()
SixConv2DCollapsingTest(self).run_test()
Op2DAddConstCollapsingTest(self).run_test()
FuncConv2DCollapsingTest(self).run_test()
FuncConvolutionCollapsingTest(self).run_test()
FuncConvCollapsingTest(self).run_test()

def test_const_quantization(self):
c = (np.ones((32, 32, 16)) + np.random.random((32, 32, 16))).astype(np.float32)
Expand Down

0 comments on commit 4f40c03

Please sign in to comment.