我尝试使用cmocka对一些源代码进行单元测试。基本上(相关)源代码看起来类似于Source.c。
单元测试分别调用每个功能。在测试 Add()函数时,此函数最终将调用 util_malloc()(此函数通常在malloc之前检查0大小),该函数由单元测试包装。在包装函数 __ wrap_util_malloc()中,在Wrappers.c中,首先检查size的大小,然后使用malloc分配内存。
接下来测试 Remove()函数,其中释放先前分配的内存。
运行测试cmocka时会返回以下故障:
<failure><![CDATA[Blocks allocated... project_path/wrappers.c:46: note: block 00341e58 allocated here ERROR: Add_Test leaked 1 block(s) ]]></failure>
和
<failure><![CDATA[EXCEPTION_ACCESS_VIOLATION occurred at 004060af.
To debug in Visual Studio... [...]
]]></failure>
现在我已经在 Add_Test()函数的末尾添加了 Remove()调用(以及 Add()在 Remove_Test()的开头。这似乎解决了这个问题。从这一点来看,应该在每个单独的单元测试中释放所有分配的内存。
现在我的问题:
static ST_SOME_STRUCT GlobStruct;
void Add()
{
GlobStruct = util_malloc(sizeof(ST_SOME_STRUCT));
}
void Remove()
{
util_free(&GlobStruct);
}
void DoStuff()
{
//Do stuff using the global structure GlobStruct
}
int main( int argc, char **argv )
{
const struct CMUnitTest Test[] =
{
cmocka_unit_test(Add_Test),
cmocka_unit_test(Remove_Test),
};
cmocka_set_message_output( CM_OUTPUT_XML );
return cmocka_run_group_tests( Test, NULL, NULL );
}
static void Add_Test (void** state)
{
expect_value(__wrap_util_malloc, size, sizeof(ST_SOME_STRUCT ));
Add();
}
static void Remove_Test (void** state)
{
expect_not_value(__wrap_util_free, memory, cast_ptr_to_largest_integral_type(NULL));
Remove();
}
void *__wrap_util_malloc(int size)
{
check_expected(size);
return malloc(size);
}
void __wrap_util_free(void *memory)
{
check_expected_ptr(memory);
free(memory);
}
答案 0 :(得分:0)
使用cmocka_run_group_tests
运行测试时,cmocka仍然会为每个测试运行单独的设置或拆卸功能,以及检查是否忘记释放块的内部函数(因此消息Add_Test leaked 1 block(s)
)。在每次测试之后,cmocka还将释放在该测试中分配的任何块,然后再运行下一个。
常见方法是每个测试都应该执行在单个测试中分配的任何内容的清理。如果要在组测试开始时初始化共享状态,请指定组设置/拆卸功能以初始化共享状态(将作为参数传递给每个单元测试),或者使用测试设置/拆卸功能之前调用每次测试。
与往常一样,拥有全局共享变量是一个坏主意(并使测试更难)。您的Add
和Remove
函数(以及处理此对象的所有其他函数)应该接受指向包含数据的结构的指针。