summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--inc/id.h18
-rw-r--r--inc/instr.h47
-rw-r--r--inc/storage.h2
-rw-r--r--inc/utils.h18
-rw-r--r--src/sim/id.cc58
-rw-r--r--src/sim/instr.cc27
-rw-r--r--src/utils/utils.cc49
-rw-r--r--tests/utils.cc108
8 files changed, 142 insertions, 185 deletions
diff --git a/inc/id.h b/inc/id.h
index eafe274..1503457 100644
--- a/inc/id.h
+++ b/inc/id.h
@@ -15,6 +15,24 @@ class ID : public Stage
ID(Stage *next);
Response advance(InstrDTO &i) override;
+
+ /**
+ * Parse an instruction into a type, opcode, and fields. If the type is
+ * invalid, only the type field will be set.
+ * @param the resulting first field, which varies per type. To call this
+ * function properly, this field must contain the full instruction bytes on
+ * function entry.
+ * @param the resulting second field, which varies per type.
+ * @param the resulting third field, which varies per type.
+ * @param the resulting type.
+ * @param the resulting opcode.
+ */
+ void get_instr_fields(
+ signed int &s1,
+ signed int &s2,
+ signed int &s3,
+ unsigned int &type,
+ unsigned int &opcode);
};
#endif /* ID_D_INCLUDED */
diff --git a/inc/instr.h b/inc/instr.h
index ce08685..98ecf1d 100644
--- a/inc/instr.h
+++ b/inc/instr.h
@@ -1,13 +1,56 @@
#ifndef INSTR_H
#define INSTR_H
-
#include <functional>
+#include <iostream>
#include <map>
+enum Mnemonic {
+ ADD,
+ SUB,
+ MUL,
+ QUOT,
+ REM,
+ SFTR,
+ SFTL,
+ AND,
+ OR,
+ NOT,
+ XOR,
+ ADDV,
+ SUBV,
+ MULV,
+ DIVV,
+ CMP,
+ CEV,
+ LOAD,
+ LOADV,
+ ADDI,
+ SUBI,
+ SFTRI,
+ SFTLI,
+ ANDI,
+ ORI,
+ XORI,
+ STORE,
+ STOREV,
+ JMP,
+ JRL,
+ JAL,
+ BEQ,
+ BGT,
+ BUF,
+ BOF,
+ PUSH,
+ POP,
+};
+
+std::ostream &operator<<(std::ostream &os, Mnemonic a);
+
namespace instr
{
// clang-format off
-extern const std::map<int,std::map<int,std::function<void(signed int &s1, signed int &s2, signed int &s3)>>> instr_map;
+ extern const std::map<unsigned int, Mnemonic> mnemonic_map;
+ extern const std::map<Mnemonic, std::function<void(signed int &s1, signed int &s2, signed int &s3)>> instr_map;
// clang-format on
} // namespace instr
diff --git a/inc/storage.h b/inc/storage.h
index ff1fbcb..d6fa094 100644
--- a/inc/storage.h
+++ b/inc/storage.h
@@ -80,7 +80,7 @@ class Storage
/**
* The accessor currently being serviced.
*/
- enum Accessor requester;
+ Accessor requester;
/**
* The number of cycles until the current request is completed.
*/
diff --git a/inc/utils.h b/inc/utils.h
index 7fed157..a375b68 100644
--- a/inc/utils.h
+++ b/inc/utils.h
@@ -13,24 +13,6 @@
void get_cache_fields(int address, int *tag, int *index, int *offset);
/**
- * Parse an instruction into a type, opcode, and fields. If the type is invalid,
- * only the type field will be set.
- * @param the resulting first field, which varies per type. To call this
- * function properly, this field must contain the full instruction bytes on
- * function entry.
- * @param the resulting second field, which varies per type.
- * @param the resulting third field, which varies per type.
- * @param the resulting type.
- * @param the resulting opcode.
- */
-void get_instr_fields(
- signed int &s1,
- signed int &s2,
- signed int &s3,
- unsigned int &type,
- unsigned int &opcode);
-
-/**
* Formats a string using snprintf.
* @param an object that represents the format string
* @param arguments to be formatted
diff --git a/src/sim/id.cc b/src/sim/id.cc
index 87dce0c..65acef5 100644
--- a/src/sim/id.cc
+++ b/src/sim/id.cc
@@ -1,9 +1,65 @@
#include "id.h"
#include "accessor.h"
#include "instrDTO.h"
+#include "logger.h"
#include "response.h"
#include "stage.h"
+#include "utils.h"
+
+static Logger *global_log = Logger::getInstance();
ID::ID(Stage *stage) : Stage(stage) { this->id = DCDE; }
-Response ID::advance(InstrDTO &i) { return OK; }
+Response ID::advance(InstrDTO &i) { Response r; }
+
+void ID::get_instr_fields(
+ signed int &s1,
+ signed int &s2,
+ signed int &s3,
+ unsigned int &type,
+ unsigned int &opcode)
+{
+ // unsigned int &opcode;
+ int opcode_bits;
+
+ type = GET_LS_BITS(s1, TYPE_SIZE);
+ opcode_bits = (type == 0b0) ? R_OPCODE_SIZE : OPCODE_SIZE;
+
+ switch (type) {
+ case 0:
+ // R-TYPE
+ opcode += GET_MID_BITS(s1, TYPE_SIZE, TYPE_SIZE + opcode_bits);
+ s3 = GET_MID_BITS(
+ s1, TYPE_SIZE + opcode_bits + (REG_SIZE * 2),
+ TYPE_SIZE + opcode_bits + (REG_SIZE * 3));
+ s2 = GET_MID_BITS(
+ s1, TYPE_SIZE + opcode_bits + REG_SIZE,
+ TYPE_SIZE + opcode_bits + (REG_SIZE * 2));
+ s1 = GET_MID_BITS(
+ s1, TYPE_SIZE + opcode_bits,
+ TYPE_SIZE + opcode_bits + REG_SIZE);
+ break;
+ case 1:
+ // I-TYPE
+ opcode = GET_MID_BITS(s1, TYPE_SIZE, TYPE_SIZE + opcode_bits);
+ s3 = GET_MID_BITS(
+ s1, TYPE_SIZE + opcode_bits + (REG_SIZE * 2), WORD_SPEC);
+ s2 = GET_MID_BITS(
+ s1, TYPE_SIZE + opcode_bits + REG_SIZE,
+ TYPE_SIZE + opcode_bits + (REG_SIZE * 2));
+ s1 = GET_MID_BITS(
+ s1, TYPE_SIZE + opcode_bits, TYPE_SIZE + opcode_bits + REG_SIZE);
+ break;
+ case 2:
+ // J-TYPE
+ opcode = GET_MID_BITS(s1, TYPE_SIZE, TYPE_SIZE + opcode_bits);
+ s2 = GET_MID_BITS(s1, TYPE_SIZE + OPCODE_SIZE + REG_SIZE, WORD_SPEC);
+ s1 = GET_MID_BITS(
+ s1, TYPE_SIZE + OPCODE_SIZE, TYPE_SIZE + OPCODE_SIZE + REG_SIZE);
+ break;
+ default:
+ global_log->log(
+ DEBUG,
+ string_format("%s returning invalid type: %d", __FUNCTION__, type));
+ }
+}
diff --git a/src/sim/instr.cc b/src/sim/instr.cc
index 608b871..50aa71d 100644
--- a/src/sim/instr.cc
+++ b/src/sim/instr.cc
@@ -3,16 +3,16 @@
#include <map>
// clang-format off
-#define INIT_INSTRUCTION(type, opcode, body) \
- {type, {{opcode, [](signed int &s1, signed int &s2, signed int &s3) { \
+#define INIT_INSTRUCTION(mnemonic, body) \
+ {mnemonic, [](signed int &s1, signed int &s2, signed int &s3) { \
body; \
- }}}}
+ }}
// clang-format on
namespace instr
{
// clang-format off
-const std::map<int, std::map<int, std::function<void(signed int &s1, signed int &s2, signed int &s3)>>>
+const std::map<Mnemonic, std::function<void(signed int &s1, signed int &s2, signed int &s3)>>
// clang-format on
instr_map = {
@@ -20,11 +20,26 @@ const std::map<int, std::map<int, std::function<void(signed int &s1, signed int
// TODO these need to be WRAPPED with a function that sets overflow.
// future note to self, if these are more than 2 lines each, you're
// doing it wrong
- INIT_INSTRUCTION(0b00, 0b00001, s3 = s1 + s2),
- INIT_INSTRUCTION(0b00, 0b00010, s3 = s1 - s2),
+ INIT_INSTRUCTION(ADD, s3 = s1 + s2),
+ INIT_INSTRUCTION(SUB, s3 = s1 - s2),
/* I type instructions */
/* J type instructions */
};
+
+const std::map<unsigned int, Mnemonic> mnemonic_map = {
+ {0b0000100, ADD}, {0b0001000, SUB}, {0b0001100, MUL}, {0b0010000, QUOT},
+ {0b0010100, REM}, {0b0011000, SFTR}, {0b0011100, SFTL}, {0b0100000, AND},
+ {0b0100100, OR}, {0b0101000, NOT}, {0b0101100, XOR}, {0b0110000, ADDV},
+ {0b0110100, SUBV}, {0b0111000, MULV}, {0b0111100, DIVV}, {0b1000000, CMP},
+ {0b1000100, CEV}, {0b000101, LOAD}, {0b001001, LOADV}, {0b001101, ADDI},
+ {0b010001, SUBI}, {0b010101, SFTRI}, {0b011101, SFTLI}, {0b100001, ANDI},
+ {0b100101, ORI}, {0b101001, XORI}, {0b101101, STORE}, {0b110001, STOREV},
+ {0b000101, CEV}, {0b000101, LOAD}, {0b001001, LOADV}, {0b001001, LOADV},
+ {0b000110, JMP}, {0b001010, JRL}, {0b001110, JAL}, {0b010010, BEQ},
+ {0b010110, BGT}, {0b011010, BUF}, {0b011110, BOF}, {0b100010, PUSH},
+ {0b100110, POP},
+
+};
} // namespace instr
diff --git a/src/utils/utils.cc b/src/utils/utils.cc
index 3b140e0..e12a0e0 100644
--- a/src/utils/utils.cc
+++ b/src/utils/utils.cc
@@ -4,8 +4,6 @@
#include <string>
#include <vector>
-static Logger *global_log = Logger::getInstance();
-
void get_cache_fields(int address, int *tag, int *index, int *offset)
{
*tag = GET_MID_BITS(address, L1_CACHE_LINE_SPEC + LINE_SPEC, MEM_WORD_SPEC);
@@ -13,53 +11,6 @@ void get_cache_fields(int address, int *tag, int *index, int *offset)
*offset = GET_LS_BITS(address, LINE_SPEC);
}
-void get_instr_fields(
- signed int &s1,
- signed int &s2,
- signed int &s3,
- unsigned int &type,
- unsigned int &opcode)
-{
- type = GET_LS_BITS(s1, TYPE_SIZE);
- switch (type) {
- case 0:
- // R-TYPE
- opcode = GET_MID_BITS(s1, TYPE_SIZE, TYPE_SIZE + R_OPCODE_SIZE);
- s3 = GET_MID_BITS(
- s1, TYPE_SIZE + R_OPCODE_SIZE + (REG_SIZE * 2),
- TYPE_SIZE + R_OPCODE_SIZE + (REG_SIZE * 3));
- s2 = GET_MID_BITS(
- s1, TYPE_SIZE + R_OPCODE_SIZE + REG_SIZE,
- TYPE_SIZE + R_OPCODE_SIZE + (REG_SIZE * 2));
- s1 = GET_MID_BITS(
- s1, TYPE_SIZE + R_OPCODE_SIZE,
- TYPE_SIZE + R_OPCODE_SIZE + REG_SIZE);
- break;
- case 1:
- // I-TYPE
- opcode = GET_MID_BITS(s1, TYPE_SIZE, TYPE_SIZE + OPCODE_SIZE);
- s3 = GET_MID_BITS(
- s1, TYPE_SIZE + OPCODE_SIZE + (REG_SIZE * 2), WORD_SPEC);
- s2 = GET_MID_BITS(
- s1, TYPE_SIZE + OPCODE_SIZE + REG_SIZE,
- TYPE_SIZE + OPCODE_SIZE + (REG_SIZE * 2));
- s1 = GET_MID_BITS(
- s1, TYPE_SIZE + OPCODE_SIZE, TYPE_SIZE + OPCODE_SIZE + REG_SIZE);
- break;
- case 2:
- // J-TYPE
- opcode = GET_MID_BITS(s1, TYPE_SIZE, TYPE_SIZE + OPCODE_SIZE);
- s2 = GET_MID_BITS(s1, TYPE_SIZE + OPCODE_SIZE + REG_SIZE, WORD_SPEC);
- s1 = GET_MID_BITS(
- s1, TYPE_SIZE + OPCODE_SIZE, TYPE_SIZE + OPCODE_SIZE + REG_SIZE);
- break;
- default:
- global_log->log(
- DEBUG,
- string_format("%s returning invalid type: %d", __FUNCTION__, type));
- }
-}
-
const std::string string_format(const char *const zcFormat, ...)
{
va_list vaArgs;
diff --git a/tests/utils.cc b/tests/utils.cc
index 6c72694..2e0e934 100644
--- a/tests/utils.cc
+++ b/tests/utils.cc
@@ -22,114 +22,6 @@ TEST_CASE("Parse arbitrary fields # two", "[utils]")
CHECK(offset == 0b11);
}
-TEST_CASE("Parse invalid type", "[utils]")
-{
- signed int s1 = 0, s2 = 0, s3 = 0;
- unsigned int type = 0, opcode = 0;
-
- s1 = 0x00FF00FF;
- get_instr_fields(s1, s2, s3, type, opcode);
-
- REQUIRE(type == 0b11);
- // behavior does nothing
- CHECK(s1 == 0x00FF00FF);
- CHECK(s2 == 0b0);
- CHECK(s3 == 0b0);
- CHECK(opcode == 0b0);
-}
-
-TEST_CASE("Parse arbitrary r-type # one", "[utils]")
-{
- signed int s1 = 0, s2 = 0, s3 = 0;
- unsigned int type = 0, opcode = 0;
-
- s1 = 0xCCCCCCCC;
- get_instr_fields(s1, s2, s3, type, opcode);
-
- REQUIRE(type == 0b00);
- CHECK(s1 == 0b11001);
- CHECK(s2 == 0b01100);
- CHECK(s3 == 0b00110);
- CHECK(opcode == 0b10011);
-}
-
-TEST_CASE("Parse arbitrary r-type # two", "[utils]")
-{
- signed int s1 = 0, s2 = 0, s3 = 0;
- unsigned int type = 0, opcode = 0;
-
- s1 = 0x99AABBCC;
- get_instr_fields(s1, s2, s3, type, opcode);
-
- REQUIRE(type == 0b00);
- CHECK(s1 == 0b10111);
- CHECK(s2 == 0b01011);
- CHECK(s3 == 0b10101);
- CHECK(opcode == 0b10011);
-}
-
-TEST_CASE("Parse arbitrary i-type # one", "[utils]")
-{
- signed int s1 = 0, s2 = 0, s3 = 0;
- unsigned int type = 0, opcode = 0;
-
- s1 = 0xDDDDDDDD;
- get_instr_fields(s1, s2, s3, type, opcode);
-
- REQUIRE(type == 0b01);
- CHECK(s1 == 0b10111);
- CHECK(s2 == 0b11011);
- CHECK(s3 == 0xDDDD);
- CHECK(opcode == 0b0111);
-}
-
-TEST_CASE("Parse arbitrary i-type # two", "[utils]")
-{
- signed int s1 = 0, s2 = 0, s3 = 0;
- unsigned int type = 0, opcode = 0;
-
- s1 = 0xAABBCCDD;
- get_instr_fields(s1, s2, s3, type, opcode);
-
- REQUIRE(type == 0b01);
- CHECK(s1 == 0b10011);
- CHECK(s2 == 0b11001);
- CHECK(s3 == 0xAABB);
- CHECK(opcode == 0b0111);
-}
-
-TEST_CASE("Parse arbitrary j-type # one", "[utils]")
-{
- signed int s1 = 0, s2 = 0, s3 = 0;
- unsigned int type = 0, opcode = 0;
-
- s1 = 0xEEEEEEEE;
- get_instr_fields(s1, s2, s3, type, opcode);
-
- REQUIRE(type == 0b10);
- CHECK(s1 == 0b11011);
- CHECK(s2 == 0b111011101110111011101);
- CHECK(opcode == 0b1011);
- // behavior does nothing
- CHECK(s3 == 0b0);
-}
-
-TEST_CASE("Parse arbitrary j-type # two", "[utils]")
-{
- signed int s1 = 0, s2 = 0, s3 = 0;
- unsigned int type = 0, opcode = 0;
-
- s1 = 0xBBCCDDEE;
- get_instr_fields(s1, s2, s3, type, opcode);
-
- REQUIRE(type == 0b10);
- CHECK(s1 == 0b10111);
- CHECK(s2 == 0b101110111100110011011);
- CHECK(opcode == 0b1011);
- // behavior does nothing
- CHECK(s3 == 0b0);
-}
-
TEST_CASE("wrap address outside upper bound", "[utils]")
{
int address = MEM_WORDS + 25;