aboutsummaryrefslogtreecommitdiff
path: root/qtmips_machine/registers.cpp
blob: 514987ae3cd953ff34815e8ded1c406e8c798c13 (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
#include "registers.h"
#include "qtmipsexception.h"
#include "utils.h"

// TODO should this be configurable?
//////////////////////////////////////////////////////////////////////////////
/// Program counter initial value
#define PC_INIT 0x80020000
//////////////////////////////////////////////////////////////////////////////

Registers::Registers() {
    this->pc = PC_INIT; // Initialize to beginning program section
    for (int i = 0; i < 31; i++)
        this->gp[i] = 0;
    this->hi = this->lo = 0;
}

std::uint32_t Registers::read_pc() {
    return this->pc;
}

std::uint32_t Registers::pc_inc() {
    this->pc += 4;
    return this->pc;
}

std::uint32_t Registers::pc_jmp(std::int32_t offset) {
    if (offset % 4)
        throw QTMIPS_EXCEPTION(UnalignedJump, "Trying to jump by unaligned offset", to_string_hex(offset));
    this->pc += offset;
    return this->pc;
}

void Registers::pc_abs_jmp(std::uint32_t address) {
    if (address % 4)
        throw QTMIPS_EXCEPTION(UnalignedJump, "Trying to jump to unaligned address", to_string_hex(address));
    this->pc = address;
}

std::uint32_t Registers::read_gp(std::uint8_t i) {
    SANITY_ASSERT(i < 32, std::string("Trying to read from register ") + std::to_string(i));
    if (!i) // $0 always reads as 0
        return 0;
    return this->gp[i - 1];
}

void Registers::write_gp(std::uint8_t i, std::uint32_t value) {
    SANITY_ASSERT(i < 32, std::string("Trying to write to register ") + std::to_string(i));
    if (i == 0) // Skip write to $0
        return;
    this->gp[i - 1] = value;
}

std::uint32_t Registers::read_hi_lo(bool hi) {
    if (hi)
        return this->hi;
    else
        return this->lo;
}

void Registers::write_hi_lo(bool hi, std::uint32_t value) {
    if (hi)
        this->hi = value;
    else
        this->lo = value;
}