summaryrefslogtreecommitdiff
path: root/src
diff options
context:
space:
mode:
Diffstat (limited to 'src')
-rw-r--r--src/common.svh115
-rw-r--r--src/rv2insn.sv107
-rw-r--r--src/sched.sv12
3 files changed, 226 insertions, 8 deletions
diff --git a/src/common.svh b/src/common.svh
index 42fdb71..67a87d2 100644
--- a/src/common.svh
+++ b/src/common.svh
@@ -1,6 +1,119 @@
`ifndef __COMMON_SVH__
`define __COMMON_SVH__
-`define NOP 0
+`define RV_OP_FIELD 6:0
+typedef logic [31:0] rv_t;
+typedef logic [31:0] xlen_t;
+
+typedef struct packed {
+ logic [6:0] funct7;
+ logic [4:0] rs2;
+ logic [4:0] rs1;
+ logic [2:0] funct3;
+ logic [4:0] rd;
+ logic [6:0] op;
+} rtype_t;
+
+typedef struct packed {
+ logic [11:0] imm;
+ logic [4:0] rs1;
+ logic [2:0] funct3;
+ logic [4:0] rd;
+ logic [6:0] op;
+} itype_t;
+
+typedef struct packed {
+ logic [6:0] imm1;
+ logic [4:0] rs2;
+ logic [4:0] rs1;
+ logic [2:0] funct3;
+ logic [4:0] imm0;
+ logic [6:0] op;
+} stype_t;
+
+typedef struct packed {
+ logic [0:0] imm3;
+ logic [5:0] imm2;
+ logic [4:0] rs2;
+ logic [4:0] rs1;
+ logic [2:0] funct3;
+ logic [3:0] imm1;
+ logic [0:0] imm0;
+ logic [6:0] op;
+} btype_t;
+
+typedef struct packed {
+ logic [19:0] imm;
+ logic [4:0] rd;
+ logic [6:0] op;
+} utype_t;
+
+typedef struct packed {
+ logic [0:0] imm3;
+ logic [9:0] imm2;
+ logic [0:0] imm1;
+ logic [7:0] imm0;
+ logic [4:0] rd;
+ logic [6:0] op;
+} jtype_t;
+
+typedef enum logic [6:0] {
+ LOAD = 7'b0000011,
+ LOAD_FP = 7'b0000111,
+ MISC_MEM = 7'b0001111,
+ OP_IMM = 7'b0010011,
+ AUIPC = 7'b0010111,
+ OP_IMM_32 = 7'b0011011,
+
+ STORE = 7'b0100011,
+ STORE_FP = 7'b0100111,
+ AMO = 7'b0101111,
+ OP = 7'b0110011,
+ LUI = 7'b0110111,
+ OP_32 = 7'b0111011,
+
+ MADD = 7'b1000011,
+ MSUB = 7'b1000111,
+ NMSUB = 7'b1001011,
+ NMADD = 7'b1001111,
+ OP_FP = 7'b1010011,
+
+ BRANCH = 7'b1100011,
+ JALR = 7'b1100111,
+ JAL = 7'b1101111,
+ SYSTEM = 7'b1110011
+} opcode_t;
+
+typedef logic[7:0] port_t;
+
+typedef enum port_t {
+ NOP = 8'b00000000,
+ REG = 8'b00100000,
+ ALU = 8'b01000000
+} port_base_t;
+
+typedef struct packed {
+ port_t src;
+ port_t dst;
+} mov_t;
+
+typedef logic[7:0] op_t;
+typedef logic[19:0] imm_t;
+
+typedef struct packed {
+ mov_t[1:0] in;
+ mov_t out;
+ op_t op;
+ imm_t imm;
+} insn_t;
+
+typedef enum op_t {
+ ALU_ADD,
+ ALU_SUB,
+ ALU_ADDI
+} alu_op_t;
+
+`define rn(r) (REG + port_t'(r))
+`define alu(fu) (ALU + port_t'(fu))
`endif /* __COMMON_SVH__ */
diff --git a/src/rv2insn.sv b/src/rv2insn.sv
new file mode 100644
index 0000000..1d9996c
--- /dev/null
+++ b/src/rv2insn.sv
@@ -0,0 +1,107 @@
+`include "common.svh"
+
+module rv2insn #(
+ parameter QUE_DEPTH,
+ parameter ALU_COUNT
+)(
+ input clk_i,
+ input rst_ni,
+ input xlen_t pc_i,
+ input rv_t[QUE_DEPTH-1:0] rv_i,
+ output insn_t[QUE_DEPTH-1:0] que_o
+);
+
+typedef logic[$clog2(ALU_COUNT):0] alu_counter_t;
+
+alu_counter_t alu_counter_r;
+alu_counter_t alu_counter;
+
+typedef struct packed {
+ insn_t insn;
+ alu_counter_t alu_counter;
+} decode_out_t;
+
+function automatic insn_t halt();
+ /* TODO come up with some proper instruction here */
+ return -1;
+endfunction
+
+function automatic alu_counter_t next_alu(alu_counter_t alu_counter);
+ return (alu_counter + 1) % ALU_COUNT;
+endfunction
+
+function automatic decode_out_t op_imm(xlen_t pc, itype_t rv, alu_counter_t alu_counter);
+ alu_counter_t fu = next_alu(alu_counter);
+ unique case (rv.funct3)
+ 3'b000: return {{NOP, NOP},
+ {`rn(rv.rs1), `alu(fu)},
+ {`alu(fu), `rn(rv.rd)},
+ ALU_ADDI, imm_t'(signed'(rv.imm)), fu};
+
+ default: return {halt(), fu};
+ endcase
+
+ return 0;
+endfunction
+
+function automatic decode_out_t op(xlen_t pc, rtype_t rv, alu_counter_t alu_counter);
+ alu_counter_t fu = next_alu(alu_counter);
+ unique case (rv.funct3)
+ 3'b000: begin
+ unique case (rv.funct7)
+ 7'b0000000:
+ return {{`rn(rv.rs1), `alu(fu)},
+ {`rn(rv.rs2), `alu(fu)},
+ {`alu(fu), `rn(rv.rd)},
+ ALU_ADD, imm_t'(0), fu};
+ 7'b0100000:
+ return {{`rn(rv.rs1), `alu(fu)},
+ {`rn(rv.rs2), `alu(fu)},
+ {`alu(fu), `rn(rv.rd)},
+ ALU_SUB, imm_t'(0), fu};
+
+ default: return {halt(), fu};
+ endcase
+ end
+ default: return {halt(), fu};
+ endcase
+endfunction
+
+function automatic decode_out_t decode(xlen_t pc, rv_t rv, alu_counter_t alu_counter);
+ unique case (rv[`RV_OP_FIELD])
+ OP_IMM: return op_imm (pc, rv, alu_counter);
+ OP: return op (pc, rv, alu_counter);
+ default: return {halt(), alu_counter};
+ endcase
+endfunction
+
+typedef struct packed {
+ insn_t[QUE_DEPTH-1:0] que;
+ alu_counter_t alu_counter;
+} stages_out_t;
+
+function automatic stages_out_t stages(int i, xlen_t pc,
+ rv_t[QUE_DEPTH-1:0] rv, insn_t[QUE_DEPTH-1:0] que, alu_counter_t alu_counter);
+ if (i < QUE_DEPTH) begin
+ decode_out_t out = decode(pc + i, rv[i], alu_counter);
+ que[i] = out.insn;
+ return stages(i + 1, pc, rv, que, out.alu_counter);
+ end else begin
+ return {que, alu_counter};
+ end
+endfunction
+
+always_comb begin
+ stages_out_t out = stages(0, pc_i, rv_i, '0, alu_counter_r);
+ alu_counter = out.alu_counter;
+ que_o = out.que;
+end
+
+always_ff @(posedge clk_i or negedge rst_ni) begin
+ if (!rst_ni) begin
+ alu_counter_r <= 0;
+ end else begin
+ alu_counter_r <= alu_counter;
+ end
+end
+endmodule
diff --git a/src/sched.sv b/src/sched.sv
index c538dc2..16a1e54 100644
--- a/src/sched.sv
+++ b/src/sched.sv
@@ -5,9 +5,6 @@ module sched
parameter QUE_DEPTH,
parameter SLOT_COUNT,
parameter SLOT_DEPTH,
- type port_t,
- type mov_t,
- type insn_t,
type slot_t
)(
input insn_t[QUE_DEPTH-1:0] que_i,
@@ -28,7 +25,7 @@ endfunction
function automatic logic check_dst(mov_t d1, mov_t[SLOT_DEPTH-1:0] d2);
logic[SLOT_DEPTH-1:0] dep = '0;
for (int i = 0; i < SLOT_DEPTH; ++i) begin
- dep[i] = d1.dst != `NOP & d1.dst == d2[i].dst;
+ dep[i] = d1.dst != NOP & d1.dst == d2[i].dst;
end
return dep != 0;
endfunction
@@ -37,7 +34,7 @@ endfunction
function automatic logic check_overlap(mov_t d1, mov_t[SLOT_DEPTH-1:0] d2);
logic[SLOT_DEPTH-1:0] dep = '0;
for (int i = 0; i < SLOT_DEPTH; ++i) begin
- dep[i] = d1.src != `NOP & d1.src == d2[i].dst;
+ dep[i] = d1.src != NOP & d1.src == d2[i].dst;
end
return dep != 0;
endfunction
@@ -50,7 +47,7 @@ function automatic logic depends(insn_t i, slot_t[SLOT_COUNT-1:0] slots);
logic overlap1 = check_overlap(i.in[0], slots[ii].out);
logic overlap2 = check_overlap(i.in[1], slots[ii].out);
- logic same_src1 = i.in[0].dst != `NOP
+ logic same_src1 = i.in[0].dst != NOP
& i.in[0].dst == slots[ii].in[0].dst;
/* op must have trigger so don't need to check against NOP */
@@ -62,7 +59,7 @@ function automatic logic depends(insn_t i, slot_t[SLOT_COUNT-1:0] slots);
endfunction
function automatic logic is_noop(mov_t m);
- return m.src == `NOP && m.dst == `NOP;
+ return m.src == NOP && m.dst == NOP;
endfunction
function automatic mov_t[SLOT_DEPTH-1:0] place_out(int i, logic placed,
@@ -80,6 +77,7 @@ endfunction
function automatic slot_t place(insn_t i, slot_t src);
slot_t slot;
slot.op = i.op;
+ slot.imm = i.imm;
slot.in = i.in;
slot.out = place_out(0, 0, i.out, src.out);
return slot;