让我们假设我有一个c源文件(测试源文件),内容:
uint8 reg1 = I2CRead(20)
uint8 reg2 = I2CRead(24)
uint8 reg3 = I2CRead(28)
if (reg1 != 0x10) || (reg2 != 0x11) || (reg3 != 0x12)
{
register content wrong ...
}
else
{
register content is OK...
}
i2Cread()函数的构建如下:
uint8 I2CRead (uint8 address)
{
I2C_Hw_Istance()->DTR = address //DTR = Data Transmit Register
return I2C_Hw_Instance()-> DRR //DRR = Data Receive Register
}
现在,我正在尝试编写一个单元测试(使用cpputest框架),我想在其中"伪造"函数读取的值 i2cread(),以便满足if()中的所有平等表达式。
单位测试文件是从"正在测试的源文件"中隔离的,但是我可以以特殊的方式访问以下功能在正在测试的源文件中: i2c_hw_instance()。
使用此函数我可以在单元测试文件中"伪造"我的寄存器值:
I2C_Hw_Istance()->DTR = 20;
I2C_Hw_Istance()->DRR = 0x10;
因为我有3 i2cread()函数连续称呼,所以我需要通过每个 i2cread()单独拨打DTR和DRR。因此,这意味着单位测试文件需要知道何时要求何时呼吁何时dtr和drr。
通常如何做到这一点?
我的想法是要有一种与单位测试一起构建的假文件(只是"概念"):
fakefile.c
void fakeRegisters()
{
if first call of I2CRead() than:
I2C_Hw_Istance()->DTR = 20;
I2C_Hw_Istance()->DRR = 0x10;
if second call of I2CRead()than:
I2C_Hw_Istance()->DTR = 24;
I2C_Hw_Istance()->DRR = 0x11;
if tird call of I2CRead() than:
I2C_Hw_Istance()->DTR = 28;
I2C_Hw_Istance()->DRR = 0x12;
}
您需要选择I2CRead()
的实现:
uint8_t I2CRead_i2c(uint8_t address)
{
I2C_Hw_Istance()->DTR = address //DTR = Data Transmit Register
return I2C_Hw_Instance()-> DRR //DRR = Data Receive Register
}
uint8_t I2CRead_fake(uint8_t address)
{
switch (address) {
case 20: return 0x10;
case 24: return 0x11;
case 28: return 0x12;
}
fail_test(); /* however you do this in your xUnit */
}
现在,在您的代码中,您通常使用"真实"实现:
uint8 (*I2CRead)(uint8) = I2CRead_i2c;
运行测试时,您需要注入模拟实现:
I2CRead = I2CRead_fake;
这是最简单的伪造实现之一;如果您需要,您可以使用全部伪像(您可能需要一个包含寄存器内容的数组,以便您可以测试起动器的成功和失败路径)。