e52b771671687084c0e798db5000996d2d5c93a2
[calu.git] / 3c_disasm / instr / subi.cpp
1 /*   `Deep Thought', a softcore CPU implemented on a FPGA
2
3     Copyright (C) 2010 Markus Hofstaetter <markus.manrow@gmx.at>
4     Copyright (C) 2010 Martin Perner <e0725782@student.tuwien.ac.at>
5     Copyright (C) 2010 Stefan Rebernig <stefan.rebernig@gmail.com>
6     Copyright (C) 2010 Manfred Schwarz <e0725898@student.tuwien.ac.at>
7     Copyright (C) 2010 Bernhard Urban <lewurm@gmail.com>
8
9     This program is free software: you can redistribute it and/or modify
10     it under the terms of the GNU General Public License as published by
11     the Free Software Foundation, either version 3 of the License, or
12     (at your option) any later version.
13
14     This program is distributed in the hope that it will be useful,
15     but WITHOUT ANY WARRANTY; without even the implied warranty of
16     MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
17     GNU General Public License for more details.
18
19     You should have received a copy of the GNU General Public License
20     along with this program.  If not, see <http://www.gnu.org/licenses/>. */
21
22 #include "../Iinstr.hpp"
23
24 class Csubi : public Iinstr {
25         public:
26                 Csubi();
27                 void evalInstr();
28                 void execInstr();
29                 std::string toString();
30                 Iinstr* getNew();
31 };
32
33 /**
34  * Name:      create_instruction
35  * Purpose:   if compiled as shared library, this functions creates the 
36               instruction object
37
38  * Returns:   pointer to instruction object
39  */
40 extern "C" Iinstr* create_instruction() {
41     return new Csubi();
42 }
43
44 Iinstr* Csubi::getNew() {
45         return new Csubi();
46 }
47
48 /**
49  * Name:      destroy_instruction
50  * Purpose:   if compiled as shared library, this functions destoys the 
51               instruction object
52
53  * Parameter: IInstruction - the instruction object to delete
54  */
55 extern "C" void destroy_instruction(Iinstr* p) {
56     delete p;
57 }
58
59 Csubi::Csubi()
60 {
61         opcode = B5(00011);
62         name = "subi";
63 }
64
65 void Csubi::evalInstr()
66 {
67         this->m_s = argbits[2];
68         this->m_c = argbits[1];
69         this->m_d = argbits[0];
70
71         argbits >>= 3;
72         dynamic_bitset<> immb = argbits;
73         immb.resize(12);
74         this->m_imm = this->generate12ImmSign(immb.to_ulong());
75
76         argbits >>= 12;
77         m_ra = this->getRegister(argbits);
78
79         argbits >>= 4;
80         m_rd = this->getRegister(argbits);
81 }
82
83 void Csubi::execInstr()
84 {
85         //cout << "should exec " << this->toString() << endl;
86         CDat ra = this->m_cpu->getRegister(m_ra);
87         CDatd reg = ra - this->m_imm;
88         this->m_cpu->setRegister(m_rd, reg);
89         if(!this->m_d) {
90                 this->m_cpu->updateFlags(reg, ra, (~this->m_imm)+1);
91         }
92 }
93
94 std::string Csubi::toString()
95 {
96         stringstream op;
97         op << this->getName();
98
99         if(m_d) op << 'D';
100         if(m_s) op << 'S';
101         if(m_c) op << 'C';
102
103         op << this->getConditionFlag() << " r" << m_rd << ", r" << m_ra << ", " << m_imm;
104
105         return op.str();
106 }