1. Model advisor 检查规则简介
model advisor是simulink自带的静态模型检查器,能够根据设定的规则对模型进行检查,并且能够进行部分错误的自动修改。在汽车电子以及相关基于MBD(Model Based Design)开发的嵌入式项目中有重要的作用,能够帮助开发人员和团队搭建高质量的模型和使用高度统一的风格。自带的检查规则中,包含MAAB这种行业中应用广泛的规则,随着团队或者项目进程的积累,后续会慢慢发展出属于自己的一套建模规则和体系。此时则需要进行对model advisor的检查规则进行自定义。
- 下图为Model advisor的界面截图
2. 检查规则自定义
自定义规则分两种情况,一种是对现有规则进行勾选,组合成团队需要的检查体系,另外一种是编写脚本,创建新的检查规则。在确认现有检查规则中无法满足所需要的检查,则可以进行新规则的编写。例如:
- 找出所有具有integer rounding mode 配置的模块,并检查这些模块的这项配置是否配成
Simplest
。
这条模型检查规则在官方自带的检查中就没有,如果想要统一团队风格只能通过编写脚本或者组织人工审查,显然作为程序员的我们对于这种选择问题的思考是不存在的,答案只有一个,代码撸起来。
本文主要针对这条规则需求进行编写脚本的详细说明和示例。
2.1 创建新的规则及放到相应目录中
既然是在model advisor中添加新的检查规则,就需要将编写的检查规则都放到统一文件夹目录下,不能与其它检查规则混用。
%--- test_int_round_mode_check function description----------------------------
function test_int_round_mode_check
int_round_mode = ModelAdvisor.Check('int_round_mode');
int_round_mode.Title = 'int_round_mode:Check the integer rounding mode 检查整数舍入模式设定';
int_round_mode.TitleTips = ['Fail if the rounding mode config is not <Simplest>'...
'该模块设置如果不为<Simplest>则判断为违反本条规则'];
int_round_mode.setCallbackFcn(@test_int_round_mode_checkCallback,'None','StyleOne');
% publish the check
mdladvRoot = ModelAdvisor.Root;
mdladvRoot.publish(int_round_mode,'TEST');
这些代码定义了检查规则的名字、简介、调用具体函数的句柄test_int_round_mode_checkCallback
,并将这条规则发布在TEST
文件夹下。
2.2 确定是否需要增加自动修改功能
model advisor中,允许进行部分规则的自动修改,就是点一个按钮直接把模型改的符合本条检查规则,这个功能十分实用,尤其对于规模庞大的模型简直就是福音,所以回到本示例中,因为需要检查的内容单一,规则明确,所以可以增加自动修改的功能。当检查出整数舍入模式没有设置成Simplest
的模块,点击一个按钮,自动将这些模块舍入模式设成Simplest
。
% fix action description
int_round_modeAct = ModelAdvisor.Action;
int_round_modeAct.Name = 'Fix settings';
int_round_modeAct.Description = ['Click the button to set all blocks to Simplest'...
'按下按钮,将所有模块设置成 Simpleset'];
int_round_modeAct.setCallbackFcn(@test_int_round_mode_checkActCallback);
int_round_mode.setAction(int_round_modeAct);
这段代码定义了修复按钮,按钮名字为Fix settings
,点击按钮的句柄为test_int_round_mode_checkActCallback
。
2.3 关键功能分析
- 对于模型中,相关模块的查找,使用
find_system(system,'RegExp','On','RndMeth','.');
可以找出模型中所有具有 integer rouding mode 配置选项的模块。
- 其中system在脚本中表示当前检查的模型
- RegExp on 表示开启正则表达式
- RndMeth 为模块integer rounding mode选项的命令字
- '.' 指代正则表达式中,所有字符(不为空字符)
综上,这条命令是说,找到当前检查模型中,所有'RndMeth'设置有任意字符的模块。
2. 使用get_param(all_rnd_meth_block{i},'RndMeth');
得到模块的整形舍入模式设定选项
3. fix action 相关函数
- 使用如下两条命令创建修复列表模板和修复结果显示
int_round_mode_fc = ModelAdvisor.FormatTemplate('TableTemplate');
int_round_mode_fc.setColTitles({'Block','Old Setting','New Setting'});
- 使用如下命令在显示面板上增加所修复的目标
int_round_mode_fc.addRow({all_rnd_meth_block{i},current_rnd,correct_cfg_rnd});
- 使用如下命令进行模块整数舍入模式选项的设置
set_param(all_rnd_meth_block{i},'RndMeth',correct_cfg_rnd);
4. 脚本的文件名字一定要为sl_customization.m ,并且在将规则添加到model advisor中要使用sl_refresh_customizations这条命令,在不打开模型的条件下执行这条命令,这些是官方规定的操作,不能有改动。
2.4 脚本源码示例
%------------------------------------------------------------------------------
% Simulink scrip for customer check which used in model advisor.
% MATLAB version: R2017a
% Shibo Jiang 2017/8/20
% Version: 0.1
% Instructions: Type the 'sl_refresh_customizations' in matlab command
% window with all the model closed.
%------------------------------------------------------------------------------
function sl_customization(cm)
% --- register custom checks
cm.addModelAdvisorCheckFcn(@test_int_round_mode_check);
%-----Start of test_int_round_mode_check---------------------------------------
%--- test_int_round_mode_check function description------------------------
function test_int_round_mode_check
int_round_mode = ModelAdvisor.Check('int_round_mode');
int_round_mode.Title = 'int_round_mode:Check the integer rounding mode';
int_round_mode.TitleTips = ['Fail if the rounding mode config is not ' ...
'<Simplest>'];
int_round_mode.setCallbackFcn(@test_int_round_mode_checkCallback, ...
'None','StyleOne');
% fix action description
int_round_modeAct = ModelAdvisor.Action;
int_round_modeAct.Name = 'Fix settings';
int_round_modeAct.Description = ['Click the button to set all blocks' ...
' to Simplest'];
int_round_modeAct.setCallbackFcn(@test_int_round_mode_checkActCallback);
int_round_mode.setAction(int_round_modeAct);
% publish the check
mdladvRoot = ModelAdvisor.Root;
mdladvRoot.publish(int_round_mode,'TEST');
%--------------creates test_int_round_mode_checkCallback function----------
function int_round_mode_result = test_int_round_mode_checkCallback(system)
int_round_mode_obj = Simulink.ModelAdvisor.getModelAdvisor(system);
int_round_mode_result={};
int_round_mode_fc = ModelAdvisor.FormatTemplate('ListTemplate');
int_round_mode_fc.setInformation(['This check looks for the blocks' ...
' which integer rounding mode is not configed as [Simplest]']);
% find all blocks which have 'integer ronuding mode' setting
all_rnd_meth_block = find_system(system,'RegExp','On','RndMeth','.');
wrong_rnd_flag = 0;
if ~isempty(all_rnd_meth_block)
% find wrong config block
length_rnd = length(all_rnd_meth_block);
j = 1;
correct_cfg_rnd = 'Simplest';
for i = 1:length_rnd
current_rnd = get_param(all_rnd_meth_block{i},'RndMeth');
if 0 == strcmp(correct_cfg_rnd,current_rnd)
wrong_rnd_flag = 1;
wrong_rnd_block{j} = all_rnd_meth_block{i};
j = j + 1;
else
% Do nothing
end
end
else
wrong_rnd_flag = 0;
end
% report finding results
if 0 == wrong_rnd_flag
int_round_mode_fc.setSubResultStatusText(['Check has passed.']);
int_round_mode_fc.setSubResultStatus('pass');
int_round_mode_obj.setCheckResultStatus(true);
else
int_round_mode_fc.setSubResultStatusText(['Check has failed.' ...
' The integer rounding mode config is wrong']);
int_round_mode_fc.setListObj(wrong_rnd_block);
int_round_mode_fc.setSubResultStatus('warn');
int_round_mode_fc.setRecAction(['Config the integer rounding' ...
' mode as [Simplest] .']);
int_round_mode_obj.setCheckResultStatus(false);
int_round_mode_obj.setActionEnable(true);
end
int_round_mode_fc.setSubBar(0);
int_round_mode_result{end+1} = int_round_mode_fc;
%--------------creates test_int_round_mode_checkActCallback function-------
function int_round_mode_ActResult = ...
test_int_round_mode_checkActCallback(taskobj)
int_round_mode_obj = taskobj.MAObj;
int_round_mode_ActResult = {};
system = getfullname(int_round_mode_obj.System);
int_round_mode_fc = ModelAdvisor.FormatTemplate('TableTemplate');
int_round_mode_fc.setColTitles({'Block','Old Setting','New Setting'})
% find all blocks which have 'integer ronuding mode' setting
all_rnd_meth_block = find_system(system,'RegExp','On','RndMeth','.');
% fix wrong config block
length_rnd = length(all_rnd_meth_block);
correct_cfg_rnd = 'Simplest';
for i = 1:length_rnd
current_rnd = get_param(all_rnd_meth_block{i},'RndMeth');
if 0 == strcmp(correct_cfg_rnd,current_rnd)
int_round_mode_fc.addRow({all_rnd_meth_block{i}, ...
current_rnd,correct_cfg_rnd});
set_param(all_rnd_meth_block{i},'RndMeth',correct_cfg_rnd);
else
% Do nothing
end
end
int_round_mode_fc.setSubBar(0);
int_round_mode_ActResult = int_round_mode_fc;
int_round_mode_obj.setActionEnable(false);
%----End of test_int_round_mode_check -----------------------------------------
2.5 效果展示
- 将脚本放到matlab工作目录下,然后在命令窗口输入
sl_refresh_customizations
。
-
打开模型,并打开model advisor
可以看到,我们编写的规则已经在 By Product/TEST目录下了 -
点击检查本条规则
可以看到检查结果,并能够链接到模型上相应的模块。 -
点击修复按钮
能够看到修复的结果 - 再次点击检查按钮,验证本条规则是否通过
可以发现已经正确修改,本条规则已经检查通过了。
3. 备注
编写的规则存在一些使用上的bug,和Matlab版本有关,例如在Matlab2017a中,仅在第一次打开matlab时才能使用sl_refresh_customizations
命令进行model advisor的规则更新,如果修改了脚本代码,再次进行更新则会失败,此时需要重启Matlab才能更新,而这个问题在Matlab2016a中不存在。