// Copyright 2008 Google Inc.
// Author: Lincoln Smith
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
//
// Unit tests for struct VCDiffCodeTableData, found in codetable.h.
#include <config.h>
#include "codetable.h"
#include "addrcache.h"
#include "testing.h"
namespace open_vcdiff {
namespace {
class CodeTableTest : public testing::Test {
protected:
CodeTableTest()
: code_table_data_(VCDiffCodeTableData::kDefaultCodeTableData) { }
virtual ~CodeTableTest() { }
virtual void SetUp() {
// Default code table must pass
EXPECT_TRUE(ValidateCodeTable());
}
static void AddExerciseOpcode(unsigned char inst1,
unsigned char mode1,
unsigned char size1,
unsigned char inst2,
unsigned char mode2,
unsigned char size2,
int opcode) {
g_exercise_code_table_->inst1[opcode] = inst1;
g_exercise_code_table_->mode1[opcode] = mode1;
g_exercise_code_table_->size1[opcode] = (inst1 == VCD_NOOP) ? 0 : size1;
g_exercise_code_table_->inst2[opcode] = inst2;
g_exercise_code_table_->mode2[opcode] = mode2;
g_exercise_code_table_->size2[opcode] = (inst2 == VCD_NOOP) ? 0 : size2;
}
static void SetUpTestCase() {
g_exercise_code_table_ = new VCDiffCodeTableData;
int opcode = 0;
for (unsigned char inst_mode1 = 0;
inst_mode1 <= VCD_LAST_INSTRUCTION_TYPE + kLastExerciseMode;
++inst_mode1) {
unsigned char inst1 = inst_mode1;
unsigned char mode1 = 0;
if (inst_mode1 > VCD_COPY) {
inst1 = VCD_COPY;
mode1 = inst_mode1 - VCD_COPY;
}
for (unsigned char inst_mode2 = 0;
inst_mode2 <= VCD_LAST_INSTRUCTION_TYPE + kLastExerciseMode;
++inst_mode2) {
unsigned char inst2 = inst_mode2;
unsigned char mode2 = 0;
if (inst_mode2 > VCD_COPY) {
inst2 = VCD_COPY;
mode2 = inst_mode2 - VCD_COPY;
}
AddExerciseOpcode(inst1, mode1, 0, inst2, mode2, 0, opcode++);
AddExerciseOpcode(inst1, mode1, 0, inst2, mode2, 255, opcode++);
AddExerciseOpcode(inst1, mode1, 255, inst2, mode2, 0, opcode++);
AddExerciseOpcode(inst1, mode1, 255, inst2, mode2, 255, opcode++);
}
}
// This is a CHECK rather than an EXPECT because it validates only
// the logic of the test, not of the code being tested.
CHECK_EQ(VCDiffCodeTableData::kCodeTableSize, opcode);
EXPECT_TRUE(VCDiffCodeTableData::kDefaultCodeTableData.Validate());
EXPECT_TRUE(g_exercise_code_table_->Validate(kLastExerciseMode));
}
static void TearDownTestCase() {
delete g_exercise_code_table_;
}
void VerifyInstruction(unsigned char opcode,
unsigned char inst,
unsigned char size,
unsigned char mode) {
EXPECT_EQ(inst, code_table_data_.inst1[opcode]);
EXPECT_EQ(size, code_table_data_.size1[opcode]);
EXPECT_EQ(mode, code_table_data_.mode1[opcode]);
EXPECT_EQ(VCD_NOOP, code_table_data_.inst2[opcode]);
EXPECT_EQ(0, code_table_data_.size2[opcode]);
EXPECT_EQ(0, code_table_data_.mode2[opcode]);
}
bool ValidateCodeTable() {
return code_table_data_.Validate();
}
// This value is designed so that the total number of inst values and modes
// will equal 8 (VCD_NOOP, VCD_ADD, VCD_RUN, VCD_COPY modes 0 - 4).
// Eight combinations of inst and mode, times two possible size values,
// squared (because there are two instructions per opcode), makes
// exactly 256 possible instruction combinations, which fits kCodeTableSize
// (the number of opcodes in the table.)
static const int kLastExerciseMode = 4;
// A code table that exercises as many combinations as possible:
// 2 instructions, each is a NOOP, ADD, RUN, or one of 5 copy modes
// (== 8 total combinations of inst and mode), and each has
// size == 0 or 255 (2 possibilities.)
static VCDiffCodeTableData* g_exercise_code_table_;
// The code table used by the current test.
VCDiffCodeTableData code_table_data_;
};
VCDiffCodeTableData* CodeTableTest::g_exercise_code_table_ = NULL;
// These tests make sure that ValidateCodeTable() catches particular
// error conditions in a custom code table.
// All possible combinations of inst and mode should have an opcode with size 0.
TEST_F(CodeTableTest, MissingCopyMode) {
VerifyInstruction(/* opcode */ 131, VCD_COPY, /* size */ 0, /* mode */ 7);
code_table_data_.size1[131] = 0xFF;
// Now there is no opcode expressing COPY with mode 7 and size 0.
EXPECT_FALSE(ValidateCodeTable());
}
TEST_F(CodeTableTest, MissingAdd) {
VerifyInstruction(/* opcode */ 1, VCD_ADD, /* size */ 0, /* mode */ 0);
code_table_data_.size1[1] = 0xFF; // Add size 0 => size 255
// Now there is no opcode expressing ADD with size 0.
EXPECT_FALSE(ValidateCodeTable());
}
TEST_F(CodeTableTest, MissingRun) {
VerifyInstruction(/* opcode */ 0, VCD_RUN, /* size */ 0, /* mode */ 0);
code_table_data_.size1[0] = 0xFF; // Run size 0 => size 255
// Now there is no opcode expressing RUN with size 0.
EXPECT_FALSE(ValidateCodeTable());
}
TEST_F(CodeTableTest, BadOpcode) {
VerifyInstruction(/* opcode */ 0, VCD_RUN, /* size */ 0, /* mode */ 0);
code_table_data_.inst1[0] = VCD_LAST_INSTRUCTION_TYPE + 1;
EXPECT_FALSE(ValidateCodeTable());
code_table_data_.inst1[0] = 0xFF;
EXPECT_FALSE(ValidateCodeTable());
}
TEST_F(CodeTableTest, BadMode) {
VerifyInstruction(/* opcode */ 131, VCD_COPY, /* size */ 0, /* mode */ 7);
code_table_data_.mode1[131] = VCDiffAddressCache::DefaultLastMode() + 1;
EXPECT_FALSE(ValidateCodeTable());
code_table_data_.mode1[131] = 0xFF;
EXPECT_FALSE(ValidateCodeTable());
}
TEST_F(CodeTableTest, AddWithNonzeroMode) {
VerifyInstruction(/* opcode */ 1, VCD_ADD, /* size */ 0, /* mode */ 0);
code_table_data_.mode1[1] = 1;
EXPECT_FALSE(ValidateCodeTable());
}
TEST_F(CodeTableTest, RunWithNonzeroMode) {
VerifyInstruction(/* opcode */ 0, VCD_RUN, /* size */ 0, /* mode */ 0);
code_table_data_.mode1[0] = 1;
EXPECT_FALSE(ValidateCodeTable());
}
TEST_F(CodeTableTest, NoOpWithNonzeroMode) {
VerifyInstruction(/* opcode */ 20, VCD_COPY, /* size */ 4, /* mode */ 0);
code_table_data_.inst1[20] = VCD_NOOP;
code_table_data_.mode1[20] = 0;
code_table_data_.size1[20] = 0;
EXPECT_TRUE(ValidateCodeTable());
code_table_data_.mode1[20] = 1;
EXPECT_FALSE(ValidateCodeTable());
}
TEST_F(CodeTableTest, NoOpWithNonzeroSize) {
VerifyInstruction(/* opcode */ 20, VCD_COPY, /* size */ 4, /* mode */ 0);
code_table_data_.inst1[20] = VCD_NOOP;
code_table_data_.mode1[20] = 0;
code_table_data_.size1[20] = 0;
EXPECT_TRUE(ValidateCodeTable());
code_table_data_.size1[20] = 1;
EXPECT_FALSE(ValidateCodeTable());
}
TEST_F(CodeTableTest, BadSecondOpcode) {
VerifyInstruction(/* opcode */ 20, VCD_COPY, /* size */ 4, /* mode */ 0);
code_table_data_.inst2[20] = VCD_LAST_INSTRUCTION_TYPE + 1;
EXPECT_FALSE(ValidateCodeTable());
code_table_data_.inst2[20] = 0xFF;
EXPECT_FALSE(ValidateCodeTable());
}
TEST_F(CodeTableTest, BadSecondMode) {
VerifyInstruction(/* opcode */ 20, VCD_COPY, /* size */ 4, /* mode */ 0);
code_table_data_.inst2[20] = VCD_COPY;
EXPECT_TRUE(ValidateCodeTable());
code_table_data_.mode2[20] = VCDiffAddressCache::DefaultLastMode() + 1;
EXPECT_FALSE(ValidateCodeTable());
code_table_data_.mode2[20] = 0xFF;
EXPECT_FALSE(ValidateCodeTable());
}
TEST_F(CodeTableTest, AddSecondWithNonzeroMode) {
VerifyInstruction(/* opcode */ 20, VCD_COPY, /* size */ 4, /* mode */ 0);
code_table_data_.inst2[20] = VCD_ADD;
EXPECT_TRUE(ValidateCodeTable());
code_table_data_.mode2[20] = 1;
EXPECT_FALSE(ValidateCodeTable());
}
TEST_F(CodeTableTest, RunSecondWithNonzeroMode) {
VerifyInstruction(/* opcode */ 20, VCD_COPY, /* size */ 4, /* mode */ 0);
code_table_data_.inst2[20] = VCD_RUN;
EXPECT_TRUE(ValidateCodeTable());
code_table_data_.mode2[20] = 1;
EXPECT_FALSE(ValidateCodeTable());
}
TEST_F(CodeTableTest, SecondNoOpWithNonzeroMode) {
VerifyInstruction(/* opcode */ 20, VCD_COPY, /* size */ 4, /* mode */ 0);
EXPECT_EQ(VCD_NOOP, code_table_data_.inst2[20]);
code_table_data_.mode2[20] = 1;
EXPECT_FALSE(ValidateCodeTable());
}
TEST_F(CodeTableTest, SecondNoOpWithNonzeroSize) {
VerifyInstruction(/* opcode */ 20, VCD_COPY, /* size */ 4, /* mode */ 0);
EXPECT_EQ(VCD_NOOP, code_table_data_.inst2[20]);
code_table_data_.size2[20] = 1;
EXPECT_FALSE(ValidateCodeTable());
}
TEST_F(CodeTableTest, ValidateExerciseCodeTable) {
EXPECT_TRUE(g_exercise_code_table_->Validate(kLastExerciseMode));
}
} // unnamed namespace
} // namespace open_vcdiff