summaryrefslogtreecommitdiff
path: root/tests/cache_1_1.cc
blob: 3cc83a0fd9a23f0c65eacff425fa37d6110c0cef (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
#include "c11.h"
#include "cache.h"
#include "dram.h"
#include <catch2/catch_test_macros.hpp>

TEST_CASE_METHOD(C11, "store 0th element in DELAY cycles", "[dram]")
{
	int r;
	signed int w;
	CHECK(expected == actual);

	w = 0x11223344;
	// delay + 1 due to internal logic, when mem
	// finishes is_address_missing still returns '1'
	this->wait_then_do(this->m_delay + this->c_delay + 1, [this, w]() {
		return this->c->write_word(this->mem, w, 0b0);
	});

	r = c->write_word(this->mem, w, 0b0);
	CHECK(r);

	expected.at(0) = w;
	actual = c->get_data()[0];
	REQUIRE(expected == actual);
}

TEST_CASE_METHOD(C11, "store 0th, 1st element in DELAY cycles, with conflict", "[cache]")
{
	signed int w;
	int r, i;
	CHECK(expected == actual);

	w = 0x11223344;
	// delay + 1 due to internal logic, when mem
	// finishes is_address_missing still returns '1'
	for (i = 0; i < this->m_delay + this->c_delay + 1; ++i) {
		r = c->write_word(this->mem, w, 0b0);
		CHECK(!r);
		r = c->write_word(this->fetch, w, 0b1);
		CHECK(!r);

		// check for early modifications
		actual = c->get_data()[0];
		REQUIRE(this->expected == this->actual);
	}

	r = c->write_word(this->mem, w, 0b0);
	CHECK(r);

	expected.at(0) = w;
	actual = c->get_data()[0];
	REQUIRE(expected == actual);

	// this should have been loaded already!
	this->wait_then_do(
		this->c_delay, [this, w]() { return this->c->write_word(this->fetch, w, 0b1); });

	r = c->write_word(this->fetch, w, 0b1);
	CHECK(r);

	expected.at(1) = w;
	actual = c->get_data()[0];
	REQUIRE(expected == actual);
}

TEST_CASE_METHOD(
	C11, "store 0th, 1st element different tags, in DELAY cycles, no conflict", "[cache]")
{
	int r;
	signed int w;
	CHECK(expected == actual);

	w = 0x11223344;
	// delay + 1 due to internal logic, when mem
	// finishes is_address_missing still returns '1'
	this->wait_then_do(this->m_delay + this->c_delay + 1, [this, w]() {
		return this->c->write_word(this->mem, w, 0b0);
	});

	r = c->write_word(this->mem, w, 0b0);
	CHECK(r);

	expected.at(0) = w;
	actual = c->get_data()[0];
	REQUIRE(expected == actual);

	// write back to memory
	// fetch new address (don't run the completion cycle yet)
	this->wait_then_do(this->m_delay + this->m_delay + 1, [this, w]() {
		return this->c->write_word(this->fetch, w, 0b10000001);
	});

	// after the fetch, this cache line should be empty
	this->c->write_word(this->fetch, w, 0b10000001);
	CHECK(r);

	expected.at(0) = 0;
	actual = c->get_data()[0];
	CHECK(expected == actual);

	this->wait_then_do(
		this->c_delay, [this, w]() { return this->c->write_word(this->fetch, w, 0b10000001); });

	r = c->write_word(this->fetch, w, 0b10000001);
	CHECK(r);

	expected.at(0) = 0;
	expected.at(1) = w;
	actual = c->get_data()[0];
	REQUIRE(expected == actual);
}