From a3dde3418b834f0018c693d4e80cb844184f80a0 Mon Sep 17 00:00:00 2001 From: xinyu Date: Sat, 3 Aug 2024 13:14:13 +0800 Subject: [PATCH 1/4] [CMIS] Return the CDB status value for the caller to check the status and perform the corresponding actions Signed-off-by: xinyu --- .../sonic_xcvr/api/public/cmis.py | 26 +++++++----- .../sonic_xcvr/api/public/cmisCDB.py | 25 +++++++----- tests/sonic_xcvr/test_cmis.py | 40 +++++++++++++------ tests/sonic_xcvr/test_cmisCDB.py | 8 ++-- 4 files changed, 65 insertions(+), 34 deletions(-) diff --git a/sonic_platform_base/sonic_xcvr/api/public/cmis.py b/sonic_platform_base/sonic_xcvr/api/public/cmis.py index cc6cd5920..8f6a63abb 100644 --- a/sonic_platform_base/sonic_xcvr/api/public/cmis.py +++ b/sonic_platform_base/sonic_xcvr/api/public/cmis.py @@ -1240,7 +1240,7 @@ def get_module_fw_mgmt_feature(self, verbose = False): """ txt = '' if self.cdb is None: - return {'status': False, 'info': "CDB Not supported", 'result': None} + return {'status': False, 'info': "CDB Not supported", 'feature': None} # get fw upgrade features (CMD 0041h) starttime = time.time() @@ -1252,8 +1252,11 @@ def get_module_fw_mgmt_feature(self, verbose = False): writelength = (writelength_raw + 1) * 8 txt += 'Auto page support: %s\n' %autopaging_flag txt += 'Max write length: %d\n' %writelength - rpllen, rpl_chkcode, rpl = self.cdb.get_fw_management_features() - if self.cdb.cdb_chkcode(rpl) == rpl_chkcode: + result = self.cdb.get_fw_management_features() + status = result['status'] + _, rpl_chkcode, rpl = result['rpl'] + + if status == 1 and self.cdb.cdb_chkcode(rpl) == rpl_chkcode: startLPLsize = rpl[2] txt += 'Start payload size %d\n' % startLPLsize maxblocksize = (rpl[4] + 1) * 8 @@ -1277,7 +1280,7 @@ def get_module_fw_mgmt_feature(self, verbose = False): else: txt += 'Reply payload check code error\n' - return {'status': False, 'info': txt, 'result': None} + return {'status': False, 'info': txt, 'feature': None} elapsedtime = time.time()-starttime logger.info('Get module FW upgrade features time: %.2f s\n' %elapsedtime) logger.info(txt) @@ -1297,20 +1300,25 @@ def get_module_fw_info(self): return {'status': False, 'info': "CDB Not supported", 'result': None} # get fw info (CMD 0100h) - rpllen, rpl_chkcode, rpl = self.cdb.get_fw_info() + result = self.cdb.get_fw_info() + status = result['status'] + rpllen, rpl_chkcode, rpl = result['rpl'] + # Interface NACK or timeout if (rpllen is None) or (rpl_chkcode is None): return {'status': False, 'info': "Interface fail", 'result': 0} # Return result 0 for distinguishing CDB is maybe in busy or failure. # password issue - if self.cdb.cdb_chkcode(rpl) != rpl_chkcode: + if status == 0x46: string = 'Get module FW info: Need to enter password\n' logger.info(string) # Reset password for module using CMIS 4.0 self.cdb.module_enter_password(0) - rpllen, rpl_chkcode, rpl = self.cdb.get_fw_info() + result = self.cdb.get_fw_info() + status = result['status'] + rpllen, rpl_chkcode, rpl = result['rpl'] - if self.cdb.cdb_chkcode(rpl) == rpl_chkcode: + if status == 1 and self.cdb.cdb_chkcode(rpl) == rpl_chkcode: # Regiter 9Fh:136 fwStatus = rpl[0] ImageARunning = (fwStatus & 0x01) # bit 0 - image A is running @@ -1603,7 +1611,7 @@ def module_fw_upgrade(self, imagepath): return result['status'], result['info'] result = self.get_module_fw_mgmt_feature() try: - startLPLsize, maxblocksize, lplonly_flag, autopaging_flag, writelength = result['result'] + startLPLsize, maxblocksize, lplonly_flag, autopaging_flag, writelength = result['feature'] except (ValueError, TypeError): return result['status'], result['info'] download_status, txt = self.module_fw_download(startLPLsize, maxblocksize, lplonly_flag, autopaging_flag, writelength, imagepath) diff --git a/sonic_platform_base/sonic_xcvr/api/public/cmisCDB.py b/sonic_platform_base/sonic_xcvr/api/public/cmisCDB.py index b21f1f651..97d33337f 100644 --- a/sonic_platform_base/sonic_xcvr/api/public/cmisCDB.py +++ b/sonic_platform_base/sonic_xcvr/api/public/cmisCDB.py @@ -155,13 +155,13 @@ def query_cdb_status(self): ''' This QUERY Status command may be used to retrieve the password acceptance status and to perform a test of the CDB interface. - It returns the reply message of this CDB command 0000h. + It returns the status and reply message of this CDB command 0000h. ''' cmd = bytearray(b'\x00\x00\x00\x00\x02\x00\x00\x00\x00\x10') cmd[133-INIT_OFFSET] = self.cdb_chkcode(cmd) self.write_cdb(cmd) status = self.cdb1_chkstatus() - if (status != 0x1): + if status != 0x1: if status > 127: txt = 'Query CDB status: Busy' else: @@ -170,7 +170,8 @@ def query_cdb_status(self): else: txt = 'Query CDB status: Success' logger.info(txt) - return self.read_cdb() + rpl = self.read_cdb() + return {'status': status, 'rpl': rpl} # Enter password def module_enter_password(self, psw = 0x00001011): @@ -199,7 +200,7 @@ def module_enter_password(self, psw = 0x00001011): def get_module_feature(self): ''' This command is used to query which CDB commands are supported. - It returns the reply message of this CDB command 0040h. + It returns the status and reply message of this CDB command 0040h. ''' cmd = bytearray(b'\x00\x40\x00\x00\x00\x00\x00\x00') cmd[133-INIT_OFFSET] = self.cdb_chkcode(cmd) @@ -214,13 +215,15 @@ def get_module_feature(self): else: txt = 'Get module feature status: Success' logger.info(txt) - return self.read_cdb() + + rpl = self.read_cdb() + return {'status': status, 'rpl': rpl} # Firmware Update Features Supported def get_fw_management_features(self): ''' This command is used to query supported firmware update features - It returns the reply message of this CDB command 0041h. + It returns the status and reply message of this CDB command 0041h. ''' cmd = bytearray(b'\x00\x41\x00\x00\x00\x00\x00\x00') cmd[133-INIT_OFFSET] = self.cdb_chkcode(cmd) @@ -235,14 +238,16 @@ def get_fw_management_features(self): else: txt = 'Get firmware management feature status: Success' logger.info(txt) - return self.read_cdb() + + rpl = self.read_cdb() + return {'status': status, 'rpl': rpl} # Get FW info def get_fw_info(self): ''' This command returns the firmware versions and firmware default running images that reside in the module - It returns the reply message of this CDB command 0100h. + It returns the status and reply message of this CDB command 0100h. ''' cmd = bytearray(b'\x01\x00\x00\x00\x00\x00\x00\x00') cmd[133-INIT_OFFSET] = self.cdb_chkcode(cmd) @@ -257,7 +262,9 @@ def get_fw_info(self): else: txt = 'Get firmware info status: Success' logger.info(txt) - return self.read_cdb() + + rpl = self.read_cdb() + return {'status': status, 'rpl': rpl} # Start FW download def start_fw_download(self, startLPLsize, header, imagesize): diff --git a/tests/sonic_xcvr/test_cmis.py b/tests/sonic_xcvr/test_cmis.py index 466621183..c94976d26 100644 --- a/tests/sonic_xcvr/test_cmis.py +++ b/tests/sonic_xcvr/test_cmis.py @@ -1171,14 +1171,15 @@ def test_get_module_level_flag(self, mock_response, expected): assert result == expected @pytest.mark.parametrize("mock_response, expected", [ - ((128, 1, [0] * 128), {'status': True, 'info': "", 'result': 0}), - ((None, 1, [0] * 128), {'status': False, 'info': "", 'result': 0}), - ((128, None, [0] * 128), {'status': False, 'info': "", 'result': 0}), - ((128, 0, [0] * 128), {'status': False, 'info': "", 'result': None}), - ((128, 1, [67, 3, 2, 2, 3, 183] + [0] * 104), {'status': True, 'info': "", 'result': None}), - ((128, 1, [52, 3, 2, 2, 3, 183] + [0] * 104), {'status': True, 'info': "", 'result': None}), - ((110, 1, [3, 3, 2, 2, 3, 183] + [0] * 104), {'status': True, 'info': "", 'result': None}), - ((110, 1, [48, 3, 2, 2, 3, 183] + [0] * 104), {'status': True, 'info': "", 'result': None}), + ({'status':1, 'rpl':(128, 1, [0] * 128)}, {'status': True, 'info': "", 'result': 0}), + ({'status':1, 'rpl':(None, 1, [0] * 128)}, {'status': False, 'info': "", 'result': 0}), + ({'status':1, 'rpl':(128, None, [0] * 128)}, {'status': False, 'info': "", 'result': 0}), + ({'status':1, 'rpl':(128, 0, [0] * 128)}, {'status': False, 'info': "", 'result': None}), + ({'status':1, 'rpl':(128, 1, [67, 3, 2, 2, 3, 183] + [0] * 104)}, {'status': True, 'info': "", 'result': None}), + ({'status':1, 'rpl':(128, 1, [52, 3, 2, 2, 3, 183] + [0] * 104)}, {'status': True, 'info': "", 'result': None}), + ({'status':1, 'rpl':(110, 1, [3, 3, 2, 2, 3, 183] + [0] * 104)}, {'status': True, 'info': "", 'result': None}), + ({'status':1, 'rpl':(110, 1, [48, 3, 2, 2, 3, 183] + [0] * 104)}, {'status': True, 'info': "", 'result': None}), + ({'status':0x46, 'rpl':(128, 0, [0] * 128)}, {'status': False, 'info': "", 'result': None}), ]) def test_get_module_fw_info(self, mock_response, expected): self.api.cdb = MagicMock() @@ -1195,6 +1196,21 @@ def test_get_module_fw_info(self, mock_response, expected): assert result['result'] == expected['result'] assert result['status'] == expected['status'] + @pytest.mark.parametrize("mock_response, expected", [ + ({'status':0, 'rpl':(18, 0, [0] * 18)}, {'status': False, 'info': "", 'feature': None}), + ({'status':1, 'rpl':(18, 1, [0] * 18)}, {'status': True, 'info': "", 'feature': (0, 8, False, True, 16)}) + ]) + def test_get_module_fw_mgmt_feature(self, mock_response, expected): + self.api.cdb = MagicMock() + self.api.cdb.cdb_chkcode = MagicMock() + self.api.cdb.cdb_chkcode.return_value = 1 + self.api.xcvr_eeprom.read = MagicMock() + self.api.xcvr_eeprom.read.side_effect = [1, 1] + self.api.cdb.get_fw_management_features = MagicMock() + self.api.cdb.get_fw_management_features.return_value = mock_response + result = self.api.get_module_fw_mgmt_feature() + assert result['feature'] == expected['feature'] + @pytest.mark.parametrize("input_param, mock_response, expected", [ (1, 1, (True, 'Module FW run: Success\n')), (1, 64, (False, 'Module FW run: Fail\nFW_run_status 64\n')), @@ -1220,22 +1236,22 @@ def test_module_fw_commit(self, mock_response, expected): @pytest.mark.parametrize("input_param, mock_response, expected", [ ( 'abc', - [{'status': True, 'info': '', 'result': ('a', 1, 1, 0, 'b', 0, 0, 0, 'a', 'b')}, {'status': True, 'info': '', 'result': (112, 2048, True, True, 2048)}, (True, ''), (True, '')], + [{'status': True, 'info': '', 'result': ('a', 1, 1, 0, 'b', 0, 0, 0, 'a', 'b')}, {'status': True, 'info': '', 'feature': (112, 2048, True, True, 2048)}, (True, ''), (True, '')], (True, '') ), ( 'abc', - [{'status': False, 'info': '', 'result': None}, {'status': True, 'info': '', 'result': (112, 2048, True, True, 2048)}, (True, ''), (True, '')], + [{'status': False, 'info': '', 'result': None}, {'status': True, 'info': '', 'feature': (112, 2048, True, True, 2048)}, (True, ''), (True, '')], (False, '') ), ( 'abc', - [{'status': True, 'info': '', 'result': ('a', 1, 1, 0, 'b', 0, 0, 0, 'a', 'b')}, {'status': False, 'info': '', 'result': None}, (True, ''), (True, '')], + [{'status': True, 'info': '', 'result': ('a', 1, 1, 0, 'b', 0, 0, 0, 'a', 'b')}, {'status': False, 'info': '', 'feature': None}, (True, ''), (True, '')], (False, '') ), ( 'abc', - [{'status': True, 'info': '', 'result': ('a', 1, 1, 0, 'b', 0, 0, 0, 'a', 'b')}, {'status': True, 'info': '', 'result': (112, 2048, True, True, 2048)}, (False, ''), (True, '')], + [{'status': True, 'info': '', 'result': ('a', 1, 1, 0, 'b', 0, 0, 0, 'a', 'b')}, {'status': True, 'info': '', 'feature': (112, 2048, True, True, 2048)}, (False, ''), (True, '')], (False, '') ), ]) diff --git a/tests/sonic_xcvr/test_cmisCDB.py b/tests/sonic_xcvr/test_cmisCDB.py index 6973e0245..c0ae07b77 100644 --- a/tests/sonic_xcvr/test_cmisCDB.py +++ b/tests/sonic_xcvr/test_cmisCDB.py @@ -78,7 +78,7 @@ def test_query_cdb_status(self, mock_response, expected): self.api.cdb1_chkstatus.return_value = mock_response[0] self.api.read_cdb = MagicMock() self.api.read_cdb.return_value = mock_response[1] - result = self.api.query_cdb_status() + result = self.api.query_cdb_status()['rpl'] assert result == expected @pytest.mark.parametrize("mock_response, expected", [ @@ -102,7 +102,7 @@ def test_get_module_feature(self, mock_response, expected): self.api.cdb1_chkstatus.return_value = mock_response[0] self.api.read_cdb = MagicMock() self.api.read_cdb.return_value = mock_response[1] - result = self.api.get_module_feature() + result = self.api.get_module_feature()['rpl'] assert result == expected @pytest.mark.parametrize("mock_response, expected", [ @@ -115,7 +115,7 @@ def test_get_fw_management_features(self, mock_response, expected): self.api.cdb1_chkstatus.return_value = mock_response[0] self.api.read_cdb = MagicMock() self.api.read_cdb.return_value = mock_response[1] - result = self.api.get_fw_management_features() + result = self.api.get_fw_management_features()['rpl'] assert result == expected @pytest.mark.parametrize("mock_response, expected", [ @@ -128,7 +128,7 @@ def test_get_fw_info(self, mock_response, expected): self.api.cdb1_chkstatus.return_value = mock_response[0] self.api.read_cdb = MagicMock() self.api.read_cdb.return_value = mock_response[1] - result = self.api.get_fw_info() + result = self.api.get_fw_info()['rpl'] assert result == expected @pytest.mark.parametrize("input_param, mock_response, expected", [ From 7b098c762f6c4896acc1359db6e1f4e2d1cadac4 Mon Sep 17 00:00:00 2001 From: xinyu Date: Sat, 3 Aug 2024 13:28:31 +0800 Subject: [PATCH 2/4] [CMIS] Log an error when failing to retrieve the CDB firmware management feature command Signed-off-by: xinyu --- sonic_platform_base/sonic_xcvr/api/public/cmis.py | 4 +++- 1 file changed, 3 insertions(+), 1 deletion(-) diff --git a/sonic_platform_base/sonic_xcvr/api/public/cmis.py b/sonic_platform_base/sonic_xcvr/api/public/cmis.py index 8f6a63abb..97f6fe098 100644 --- a/sonic_platform_base/sonic_xcvr/api/public/cmis.py +++ b/sonic_platform_base/sonic_xcvr/api/public/cmis.py @@ -1279,7 +1279,9 @@ def get_module_fw_mgmt_feature(self, verbose = False): txt += 'Read to LPL/EPL {:#x}\n'.format(rpl[6]) else: - txt += 'Reply payload check code error\n' + txt += 'Status or reply payload check code error\n' + logger.info(txt) + logger.info('Fail to get fw mgnt feature, cdb status: 0x{}\n'.format(status)) return {'status': False, 'info': txt, 'feature': None} elapsedtime = time.time()-starttime logger.info('Get module FW upgrade features time: %.2f s\n' %elapsedtime) From 4d919c8546aa5832bb977eb029fab9984ffaffd6 Mon Sep 17 00:00:00 2001 From: xinyu Date: Tue, 6 Aug 2024 16:31:57 +0800 Subject: [PATCH 3/4] [CMIS] Log get fw mgmt feature RPL check code result to provide clearer data when an error occurs Signed-off-by: xinyu --- sonic_platform_base/sonic_xcvr/api/public/cmis.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/sonic_platform_base/sonic_xcvr/api/public/cmis.py b/sonic_platform_base/sonic_xcvr/api/public/cmis.py index 97f6fe098..d290d143c 100644 --- a/sonic_platform_base/sonic_xcvr/api/public/cmis.py +++ b/sonic_platform_base/sonic_xcvr/api/public/cmis.py @@ -1281,7 +1281,7 @@ def get_module_fw_mgmt_feature(self, verbose = False): else: txt += 'Status or reply payload check code error\n' logger.info(txt) - logger.info('Fail to get fw mgnt feature, cdb status: 0x{}\n'.format(status)) + logger.info('Fail to get fw mgmt feature, cdb status: {:#x}, cdb_chkcode: {:#x}, rpl_chkcode: {:#x}\n'.format(status, self.cdb.cdb_chkcode(rpl), rpl_chkcode)) return {'status': False, 'info': txt, 'feature': None} elapsedtime = time.time()-starttime logger.info('Get module FW upgrade features time: %.2f s\n' %elapsedtime) From ea7ee0df0bd8b92469c984fd9b06fc6cdb6794d8 Mon Sep 17 00:00:00 2001 From: xinyu Date: Tue, 13 Aug 2024 16:15:29 +0800 Subject: [PATCH 4/4] [CMIS] Chagne log level from info to error for fail to get fw mgmt feature Signed-off-by: xinyu --- sonic_platform_base/sonic_xcvr/api/public/cmis.py | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/sonic_platform_base/sonic_xcvr/api/public/cmis.py b/sonic_platform_base/sonic_xcvr/api/public/cmis.py index d290d143c..3f1cc82a1 100644 --- a/sonic_platform_base/sonic_xcvr/api/public/cmis.py +++ b/sonic_platform_base/sonic_xcvr/api/public/cmis.py @@ -1280,8 +1280,8 @@ def get_module_fw_mgmt_feature(self, verbose = False): else: txt += 'Status or reply payload check code error\n' - logger.info(txt) - logger.info('Fail to get fw mgmt feature, cdb status: {:#x}, cdb_chkcode: {:#x}, rpl_chkcode: {:#x}\n'.format(status, self.cdb.cdb_chkcode(rpl), rpl_chkcode)) + logger.error(txt) + logger.error('Fail to get fw mgmt feature, cdb status: {:#x}, cdb_chkcode: {:#x}, rpl_chkcode: {:#x}\n'.format(status, self.cdb.cdb_chkcode(rpl), rpl_chkcode)) return {'status': False, 'info': txt, 'feature': None} elapsedtime = time.time()-starttime logger.info('Get module FW upgrade features time: %.2f s\n' %elapsedtime)