-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathDecoder.sv
151 lines (126 loc) · 5.47 KB
/
Decoder.sv
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
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
module decoder (
input logic [1:0] Op,
input logic [5:0] Funct,
input logic [3:0] Rd,
output logic [1:0] FlagW,
output logic PCS, RegW, MemW, MemB,
output logic MemtoReg, ALUSrcA, ALUSrcB,
output logic [1:0] ImmSrc, RegSrc,
output logic [2:0] ALUControl
);
logic [11:0] controls;
logic Branch, ALUOp;
// Main Decoder
always_comb
case(Op)
///////////////////////////////////////////////////////////
// DATA PROCESSING
///////////////////////////////////////////////////////////
2'b00:
case(Funct[4:1])
4'b1101 :
//MOV REGISTER, IMMEDIATE
if (Funct[5]) controls = 12'b000011010000;
//MOV REGISTER, REGISTER AND LSL|LSR|ASR|ROR
else controls = 12'b000010010000;
4'b1010 :
//CMP REGISTER, IMMEDIATE
if (Funct[5]) controls = 12'b000001000001;
//CMP REGISTER, REGISTER
else controls = 12'b000000000001;
4'b1011 :
//CMN REGISTER, IMMEDIATE
if (Funct[5]) controls = 12'b000001000001;
//CMN REGISTER, REGISTER
else controls = 12'b000000000001;
4'b1000 :
//TST REGISTER, IMMEDIATE
if (Funct[5]) controls = 12'b000001000001;
//TST REGISTER, REGISTER
else controls = 12'b000000000001;
4'b1001 :
//TEQ REGISTER, IMMEDIATE
if (Funct[5]) controls = 12'b000001000001;
//TEQ REGISTER, REGISTER
else controls = 12'b000000000001;
default:
//ALU FUNCTIONS ADD|SUB|AND|ORR
if (Funct[5]) controls = 12'b000001010001;
else controls = 12'b000000010001;
endcase
///////////////////////////////////////////////////////////
// MEMORY
///////////////////////////////////////////////////////////
2'b01:
// LDR
if (Funct[0])
//Imediato
if (~Funct[5])
//Byte
if(Funct[2]) controls = 12'b000101110100;
//Normal
else controls = 12'b000101110000;
//Registrador
else
//Byte
if(Funct[2]) controls = 12'b000100110100;
//Normal
else controls = 12'b000100110000;
// STR
else
//Imediato
if (~Funct[5])
//Byte
if(Funct[2]) controls = 12'b100101101100;
//Normal
else controls = 12'b100101101000;
//Registrador
else
//Byte
if(Funct[2]) controls = 12'b000100101100;
//Normal
else controls = 12'b000100101000;
///////////////////////////////////////////////////////////
// BRENCH
///////////////////////////////////////////////////////////
2'b10: controls = 12'b011001000010;
// Unimplemented
default: controls = 12'bx;
endcase
assign {RegSrc, ImmSrc, ALUSrcA, ALUSrcB, MemtoReg,
RegW, MemW, MemB, Branch, ALUOp} = controls;
// ALU Decoder
always_comb
if (ALUOp) begin // which DP Instr?
case(Funct[4:1])
4'b0100: ALUControl = 3'b000; // ADD
4'b0010: ALUControl = 3'b001; // SUB
4'b0000: ALUControl = 3'b010; // AND
4'b1100: ALUControl = 3'b011; // ORR
4'b0001: ALUControl = 3'b100; // EOR (XOR)
4'b1000: ALUControl = 3'b010; // TST (AND)
4'b1001: ALUControl = 3'b100; // TEQ (XOR)
4'b1010: ALUControl = 3'b001; // CMP (ADD)
4'b1011: ALUControl = 3'b001; // CMN (SUB)
default: ALUControl = 3'bx; // unimplemented
endcase
// update flags if S bit is set
// (C & V only updated for arith instructions)
FlagW[1] = Funct[0]; // FlagW[1] = S-bit
// FlagW[0] = S-bit & (ADD | SUB)
FlagW[0] = Funct[0] & (ALUControl == 3'b000 | ALUControl == 3'b001);
end else begin
ALUControl = 3'b000; // add for non-DP instructions
FlagW = 2'b00; // don't update Flags
end
// PC Logic
assign PCS = ((Rd == 4'b1111) & RegW) | Branch;
endmodule
//CMP ? compare
//-Flags set to result of (Rn - Operand2).
//CMN ? compare negative
//-Flags set to result of (Rn + Operand2).
//TST ? bitwise test
//-Flags set to result of (Rn AND Operand2).
//TEQ ? test equivalence
//-Flags set to result of (Rn EOR Operand2).