3

我正在阅读几篇关于模拟 C 函数(如CMockCMocka)的文章,但我不确定在此过程中如何用模拟函数替换实际函数。例如,CMocka 依赖于使用 GNU 编译器的自动包装,该编译器支持--wrap__wrap前缀附加到函数调用或允许您覆盖任何您喜欢的符号的弱符号等参数。

但是对于几乎所有其他框架,您如何在 Visual Studio 中执行此操作?

例如,CMock 有一个类似的例子(这里简化了很多):

// myfunc.c
#include <parsestuff.h>

// this is the function we would like to test
int MyFunc(char* Command)
{
    // this is the call to the function we will mock
    return ParseStuff(Command);
}

还有实际的实现,它包含链接器应该在实际应用程序中找到的实际功能:

// parsestuff.c

int ParseStuff(char* cmd)
{
    // do some actual work
    return 42;
}

现在,在测试期间,Ruby 脚本会创建模拟函数,例如:

// MockParseStuff.c (auto created by cmock)

int ParseStuff(char* Cmd);
void ParseStuff_ExpectAndReturn(char* Cmd, int toReturn);
  1. 但是如果 VS 项目已经包含parsestuff.c了,那么调用 from 怎么可能myfunc.c结束MockParseStuff.c呢?

  2. 这是否意味着我不能parsestuff.c包含在单元测试项目中?但如果是这种情况,那么也不可能模拟,例如,MyFuncmyfunc.c任何测试中,因为我已经必须包含文件才能测试它?

(更新)我也知道我可以包含.c文件而不是.h文件,然后做一些预处理器的东西来替换原来的调用,比如:

// replace ParseStuff with ParseStuff_wrap
#define ParseStuff ParseStuff_wrap
// include the source instead of the header
#include <myfunc.c>
#undef ParseStuff

int ParseStuff_wrap(char* cmd) 
{
    // this will get called from MyFunc,
    // which is now statically included
}

但这似乎是很多管道,我什至没有看到任何地方提到它。

4

2 回答 2

2

这是一个简单而简短的 hippomocks 解决方案:

我创建了一个空的 Win32 控制台应用程序

  • 主文件
  • myfunc.c + myfunc.h
  • parsestuff.c、parsestuff.h

并添加了您示例中的代码。

在 hippomock 的帮助下,您可以模拟每个 C-Function。这是我的 main.cpp 的样子:

#include "stdafx.h"
#include "myfunc.h"
#include "hippomocks.h"


extern "C" int ParseStuff(char* cmd);

int _tmain(int argc, _TCHAR* argv[])
{
    MockRepository mocks;

    mocks.ExpectCallFunc(ParseStuff).Return(4711);

    char buf[10] = "";

    int result = MyFunc(buf);

    return result; //assert result is 4711
}

HippoMocks 是一个免费、简单且非常强大的单头框架,可以在 GitHub 上下载。

希望我赢得了赏金:)

更新,它是如何工作的:

  1. HippoMocks 获取指向 ParseStuff 的 func 指针
  2. HippoMocks 构建了一个替换 func 指针,指向具有相同签名和自己实现的模板函数。
  3. Hippomocks 修补内存中函数调用序言中的 jmp 操作码,使其指向被替换的函数。
  4. 调用后或在析构函数中释放替换和内存补丁。

这是它在我的机器上的样子:

@ILT+3080(_ParseStuff):
00D21C0D  jmp HippoMocks::mockFuncs<char,int>::static_expectation1<0,char *> (0D21DB1h)  

如果您在内存窗口中观察内存地址 00D21C0D(可能与运行不同),您会看到它在调用 ExpectCallFunc 后被修补。

于 2016-08-02T11:21:18.643 回答
0

我没有处理过 C 模拟库或 Visual Studio,但我在自己的项目中考虑过这一点。Feathers 书建议使用预处理器接缝或链接接缝作为处理此问题的工具。您已经提到了预处理器接缝,所以我将专注于链接接缝。

链接接缝要求模拟函数在库中,模拟函数在库中。测试可以链接到模拟函数库,而目标应用程序可以链接到原始库。

当然,正如您所提到的,要模拟 MyFunc() 您将必须创建另一个库和一个单独的测试应用程序来链接它(或在测试应用程序中动态加载和卸载库)。

这听起来很费力,这就是为什么我拖延在自己的应用程序中添加测试的原因!

希望这可以帮助!

于 2016-08-01T13:11:45.880 回答