#include "dram.h" #include "definitions.h" #include #include TEST_CASE("Construct singleton dram", "[dram]") { Dram *d = new Dram(1, 1); std::array expected = {0, 0, 0, 0}; std::array actual = d->view(0, 1)[0]; REQUIRE(expected == actual); delete d; } TEST_CASE( "Construct singleton dram, store 0th element in zero cycles", "[dram]") { Dram *d = new Dram(1, 0); std::array expected = {0, 0, 0, 0}; std::array actual = d->view(0, 1)[0]; CHECK(expected == actual); signed int w = 0x11223344; Response r = d->write(MEM, w, 0x00000000); CHECK(r == OK); expected.at(0) = w; actual = d->view(0, 1)[0]; REQUIRE(expected == actual); delete d; } TEST_CASE( "Construct singleton dram, store 0th element in three cycles", "[dram]") { int delay = 3; Dram *d = new Dram(1, delay); std::array expected = {0, 0, 0, 0}; std::array actual = d->view(0, 1)[0]; CHECK(expected == actual); signed int w = 0x11223344; int i; Response r; for (i = 0; i < delay; ++i) { r = d->write(MEM, w, 0x00000000); CHECK(r == WAIT); actual = d->view(0, 1)[0]; REQUIRE(expected == actual); d->resolve(); } r = d->write(MEM, w, 0x00000000); CHECK(r == OK); d->resolve(); expected.at(0) = w; actual = d->view(0, 1)[0]; REQUIRE(expected == actual); delete d; } TEST_CASE( "Construct singleton dram, store 0, 1th element in three cycles no " "conflict", "[dram]") { int delay = 3; Dram *d = new Dram(1, delay); std::array expected = {0, 0, 0, 0}; std::array actual = d->view(0, 1)[0]; CHECK(expected == actual); signed int w = 0x11223344; int i; Response r; for (i = 0; i < delay; ++i) { r = d->write(MEM, w, 0x00000000); CHECK(r == WAIT); actual = d->view(0, 1)[0]; REQUIRE(expected == actual); d->resolve(); } r = d->write(MEM, w, 0x00000000); REQUIRE(r == OK); // clock cycle did NOT resolve yet! // this fetch should not make progress r = d->write(FETCH, w, 0x00000001); CHECK(r == WAIT); actual = d->view(0, 1)[0]; CHECK(r == WAIT); d->resolve(); expected.at(0) = w; actual = d->view(0, 1)[0]; REQUIRE(expected == actual); for (i = 0; i < delay; ++i) { r = d->write(FETCH, w, 0x00000001); CHECK(r == WAIT); actual = d->view(0, 1)[0]; REQUIRE(expected == actual); d->resolve(); } r = d->write(FETCH, w, 0x00000001); actual = d->view(0, 1)[0]; CHECK(r == OK); expected.at(1) = w; actual = d->view(0, 1)[0]; REQUIRE(expected == actual); delete d; } TEST_CASE( "Construct singleton dram, store 0, 1th element in three cycles much " "conflict", "[dram]") { int delay = 2; Dram *d = new Dram(1, 2); std::array expected = {0, 0, 0, 0}; std::array actual = d->view(0, 1)[0]; CHECK(expected == actual); signed int w = 0x11223344; int i; Response r; for (i = 0; i < delay; ++i) { r = d->write(MEM, w, 0x00000000); CHECK(r == WAIT); r = d->write(FETCH, w, 0x00000001); CHECK(r == WAIT); actual = d->view(0, 1)[0]; REQUIRE(expected == actual); d->resolve(); } r = d->write(MEM, w, 0x00000000); CHECK(r == OK); r = d->write(FETCH, w, 0x00000001); CHECK(r == WAIT); d->resolve(); actual = d->view(0, 1)[0]; expected.at(0) = w; REQUIRE(expected == actual); for (i = 0; i < delay; ++i) { r = d->write(FETCH, w, 0x00000001); CHECK(r == WAIT); r = d->write(MEM, w, 0x00000003); CHECK(r == WAIT); actual = d->view(0, 1)[0]; REQUIRE(expected == actual); d->resolve(); } r = d->write(FETCH, w, 0x00000001); actual = d->view(0, 1)[0]; CHECK(r == OK); r = d->write(MEM, w, 0x00000003); CHECK(r == WAIT); expected.at(1) = w; actual = d->view(0, 1)[0]; REQUIRE(expected == actual); delete d; } TEST_CASE("Construct singleton dram, write a line to an address", "[dram]") { Dram *d = new Dram(1, 0); std::array expected = {0, 0, 0, 0}; std::array actual = d->view(0, 1)[0]; CHECK(expected == actual); signed int w = 0x11223311; expected = {w, w+1, w+2, w+3}; int addr = 0x00000000; d->write_line(MEM, expected, addr); Response r = d->read(MEM, 0x00000000, actual); CHECK(r == OK); REQUIRE(expected == actual); r = d->read(MEM, 0x00000001, actual); CHECK(r == OK); REQUIRE(expected == actual); r = d->read(MEM, 0x00000002, actual); CHECK(r == OK); REQUIRE(expected == actual); r = d->read(MEM, 0x00000003, actual); CHECK(r == OK); REQUIRE(expected == actual); delete d; } TEST_CASE("Construct singleton dram, write a line to an address one element at a time, read it in zero cycles", "[dram]") { Dram *d = new Dram(1, 0); std::array expected = {0, 0, 0, 0}; std::array actual = d->view(0, 1)[0]; CHECK(expected == actual); signed int w = 0x11223311; int addr = 0x00000000; for(int i=0; iwrite(MEM, w, addr++); CHECK(r == OK); expected.at(i) = w++; } Response r = d->read(MEM, 0x00000000, actual); CHECK(r == OK); REQUIRE(expected == actual); r = d->read(MEM, 0x00000001, actual); CHECK(r == OK); REQUIRE(expected == actual); r = d->read(MEM, 0x00000002, actual); CHECK(r == OK); REQUIRE(expected == actual); r = d->read(MEM, 0x00000003, actual); CHECK(r == OK); REQUIRE(expected == actual); delete d; } TEST_CASE("Construct singleton dram, write a line to an address one element at a time in 12 cycles, read it in three cycles", "[dram]") { int delay = 3; Dram *d = new Dram(1, delay); std::array expected = {0, 0, 0, 0}; std::array actual = d->view(0, 1)[0]; CHECK(expected == actual); signed int w = 0x11223311; int addr = 0x00000000; int i; Response r; for(i=0; iwrite(MEM, w, addr); d->resolve(); } r = d->write(MEM, w, addr++); d->resolve(); expected.at(i) = w++; } for (i = 0; i < delay; ++i) { r = d->read(MEM, 0x00000000, actual); CHECK(r == WAIT); REQUIRE(expected != actual); d->resolve(); } r = d->read(MEM, 0x00000000, actual); CHECK(r == OK); d->resolve(); REQUIRE(expected == actual); delete d; } TEST_CASE( "Construct singleton dram, store line one element at a time in 12 cycles, read line in 3 cycles with no conflict","[dram]") { int delay = 3; Dram *d = new Dram(1, delay); std::array expected = {0, 0, 0, 0}; std::array actual = d->view(0, 1)[0]; CHECK(expected == actual); signed int w = 0x11223311; int addr = 0x00000000; int i; Response r; for(i=0; iwrite(MEM, w, addr); d->resolve(); } r = d->write(MEM, w, addr++); d->resolve(); expected.at(i) = w++; } for (i = 0; i < delay; ++i) { r = d->read(MEM, 0x00000000, actual); CHECK(r == WAIT); REQUIRE(expected != actual); d->resolve(); } r = d->read(MEM, 0x00000000, actual); REQUIRE(r == OK); r = d->read(FETCH, 0x00000003, actual); CHECK(r == WAIT); d->resolve(); REQUIRE(expected == actual); actual = {0,0,0,0}; for (i = 0; i < delay; ++i) { r = d->read(FETCH, 0x00000000, actual); CHECK(r == WAIT); REQUIRE(expected != actual); d->resolve(); } r = d->read(FETCH, 0x00000000, actual); REQUIRE(r == OK); r = d->read(MEM, 0x00000002, actual); CHECK(r == WAIT); d->resolve(); REQUIRE(expected == actual); delete d; } TEST_CASE( "Construct singleton dram, store line one element at a time in 12 cycles, read line in 3 cycles with much conflict","[dram]") { int delay = 3; Dram *d = new Dram(1, delay); std::array expected = {0, 0, 0, 0}; std::array actual = d->view(0, 1)[0]; CHECK(expected == actual); signed int w = 0x11223311; int addr = 0x00000000; int i; Response r; for(i=0; iwrite(MEM, w, addr); d->resolve(); } r = d->write(MEM, w, addr++); d->resolve(); expected.at(i) = w++; } for (i = 0; i < delay; ++i) { r = d->read(MEM, 0x00000000, actual); CHECK(r == WAIT); REQUIRE(expected != actual); r = d->read(FETCH, 0x00000002, actual); CHECK(r == WAIT); REQUIRE(expected != actual); d->resolve(); } r = d->read(MEM, 0x00000000, actual); REQUIRE(r == OK); r = d->read(FETCH, 0x00000003, actual); CHECK(r == WAIT); d->resolve(); REQUIRE(expected == actual); actual = {0,0,0,0}; for (i = 0; i < delay; ++i) { r = d->read(FETCH, 0x00000000, actual); CHECK(r == WAIT); REQUIRE(expected != actual); r = d->read(MEM, 0x00000002, actual); CHECK(r == WAIT); REQUIRE(expected != actual); d->resolve(); } r = d->read(FETCH, 0x00000000, actual); REQUIRE(r == OK); r = d->read(MEM, 0x00000002, actual); CHECK(r == WAIT); d->resolve(); REQUIRE(expected == actual); delete d; } TEST_CASE("Sidedoor bypasses delay", "[dram]") { int delay = 3; Dram *d = new Dram(1, delay); std::array expected = {0, 0, 0, 0}; std::array actual = d->view(0, 1)[0]; CHECK(expected == actual); signed int w = 0x11223344; int i; Response r; for (i = 0; i < delay - 1; ++i) { r = d->write(MEM, w, 0x00000000); CHECK(r == WAIT); actual = d->view(0, 1)[0]; REQUIRE(expected == actual); d->resolve(); } r = d->write(MEM, w, 0x00000000); CHECK(r == WAIT); actual = d->view(0, 1)[0]; REQUIRE(expected == actual); r = d->write(SIDE, w, 0x00000001); actual = d->view(0, 1)[0]; CHECK(r == OK); expected.at(1) = w; actual = d->view(0, 1)[0]; REQUIRE(expected == actual); delete d; }