2011-10-26 08:02:47 +02:00
|
|
|
package Top {
|
|
|
|
|
|
|
|
import Chisel._
|
|
|
|
import Node._;
|
|
|
|
import Constants._;
|
|
|
|
|
|
|
|
class ioDivider(width: Int) extends Bundle {
|
|
|
|
// requests
|
2012-01-18 19:28:48 +01:00
|
|
|
val div_val = Bool(INPUT);
|
|
|
|
val div_kill = Bool(INPUT);
|
|
|
|
val div_rdy = Bool(OUTPUT);
|
|
|
|
val dw = UFix(1, INPUT);
|
|
|
|
val div_fn = UFix(2, INPUT);
|
2012-02-09 12:47:59 +01:00
|
|
|
val div_tag = UFix(5, INPUT);
|
|
|
|
val in0 = Bits(width, INPUT);
|
|
|
|
val in1 = Bits(width, INPUT);
|
2011-10-26 08:02:47 +02:00
|
|
|
// responses
|
2012-02-09 12:47:59 +01:00
|
|
|
val result = Bits(width, OUTPUT);
|
|
|
|
val result_tag = UFix(5, OUTPUT);
|
|
|
|
val result_val = Bool(OUTPUT);
|
|
|
|
val result_rdy = Bool(INPUT);
|
2011-10-26 08:02:47 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
class rocketDivider(width : Int) extends Component {
|
|
|
|
val io = new ioDivider(width);
|
|
|
|
|
|
|
|
val s_ready :: s_neg_inputs :: s_busy :: s_neg_outputs :: s_done :: Nil = Enum(5) { UFix() };
|
|
|
|
val state = Reg(resetVal = s_ready);
|
|
|
|
|
2012-01-02 01:09:40 +01:00
|
|
|
val count = Reg() { UFix() };
|
|
|
|
val divby0 = Reg() { Bool() };
|
|
|
|
val neg_quo = Reg() { Bool() };
|
|
|
|
val neg_rem = Reg() { Bool() };
|
2012-02-09 12:47:59 +01:00
|
|
|
val reg_tag = Reg() { UFix() };
|
2012-01-02 01:09:40 +01:00
|
|
|
val rem = Reg() { Bool() };
|
|
|
|
val half = Reg() { Bool() };
|
2011-10-26 08:02:47 +02:00
|
|
|
|
2012-01-02 01:09:40 +01:00
|
|
|
val divisor = Reg() { UFix() };
|
|
|
|
val remainder = Reg() { UFix() };
|
2011-10-26 08:02:47 +02:00
|
|
|
val subtractor = remainder(2*width, width).toUFix - divisor;
|
|
|
|
|
2011-12-20 01:57:53 +01:00
|
|
|
val tc = (io.div_fn === DIV_D) || (io.div_fn === DIV_R);
|
2012-01-03 00:42:39 +01:00
|
|
|
|
2012-02-12 02:20:33 +01:00
|
|
|
val do_kill = io.div_kill && Reg(io.div_rdy) // kill on 1st cycle only
|
|
|
|
|
2011-10-26 08:02:47 +02:00
|
|
|
switch (state) {
|
|
|
|
is (s_ready) {
|
2012-02-12 02:20:33 +01:00
|
|
|
when (io.div_val) {
|
|
|
|
state := Mux(tc, s_neg_inputs, s_busy)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
is (s_neg_inputs) {
|
|
|
|
state := Mux(do_kill, s_ready, s_busy)
|
2011-10-26 08:02:47 +02:00
|
|
|
}
|
|
|
|
is (s_busy) {
|
2012-02-12 02:20:33 +01:00
|
|
|
when (do_kill) {
|
|
|
|
state := s_ready
|
|
|
|
}
|
|
|
|
.elsewhen (count === UFix(width)) {
|
|
|
|
state := Mux(neg_quo || neg_rem, s_neg_outputs, s_done)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
is (s_neg_outputs) {
|
|
|
|
state := s_done
|
2011-10-26 08:02:47 +02:00
|
|
|
}
|
|
|
|
is (s_done) {
|
2012-02-12 02:20:33 +01:00
|
|
|
when (io.result_rdy) {
|
|
|
|
state := s_ready
|
|
|
|
}
|
2011-10-26 08:02:47 +02:00
|
|
|
}
|
|
|
|
}
|
2012-02-12 02:20:33 +01:00
|
|
|
|
|
|
|
// state machine
|
2011-10-26 08:02:47 +02:00
|
|
|
|
2012-02-09 12:47:59 +01:00
|
|
|
val lhs_sign = tc && Mux(io.dw === DW_64, io.in0(width-1), io.in0(width/2-1)).toBool
|
|
|
|
val lhs_hi = Mux(io.dw === DW_64, io.in0(width-1,width/2), Fill(width/2, lhs_sign))
|
|
|
|
val lhs_in = Cat(lhs_hi, io.in0(width/2-1,0))
|
2011-10-26 08:02:47 +02:00
|
|
|
|
2012-02-09 12:47:59 +01:00
|
|
|
val rhs_sign = tc && Mux(io.dw === DW_64, io.in1(width-1), io.in1(width/2-1)).toBool
|
|
|
|
val rhs_hi = Mux(io.dw === DW_64, io.in1(width-1,width/2), Fill(width/2, rhs_sign))
|
|
|
|
val rhs_in = Cat(rhs_hi, io.in1(width/2-1,0))
|
2011-10-26 08:02:47 +02:00
|
|
|
|
|
|
|
when ((state === s_ready) && io.div_val) {
|
2012-02-12 02:20:33 +01:00
|
|
|
count := UFix(0, log2up(width+1));
|
|
|
|
half := (io.dw === DW_32);
|
|
|
|
neg_quo := Bool(false);
|
|
|
|
neg_rem := Bool(false);
|
|
|
|
rem := (io.div_fn === DIV_R) || (io.div_fn === DIV_RU);
|
|
|
|
reg_tag := io.div_tag;
|
|
|
|
divby0 := Bool(true);
|
|
|
|
divisor := rhs_in.toUFix;
|
|
|
|
remainder := Cat(UFix(0,width+1), lhs_in).toUFix;
|
2011-10-26 08:02:47 +02:00
|
|
|
}
|
|
|
|
when (state === s_neg_inputs) {
|
2012-02-12 02:20:33 +01:00
|
|
|
neg_rem := remainder(width-1).toBool;
|
|
|
|
neg_quo := (remainder(width-1) != divisor(width-1));
|
2011-10-26 08:02:47 +02:00
|
|
|
when (remainder(width-1).toBool) {
|
2012-02-12 02:20:33 +01:00
|
|
|
remainder := Cat(remainder(2*width, width), -remainder(width-1,0)).toUFix;
|
2011-10-26 08:02:47 +02:00
|
|
|
}
|
|
|
|
when (divisor(width-1).toBool) {
|
2012-02-12 02:20:33 +01:00
|
|
|
divisor := subtractor(width-1,0);
|
2011-10-26 08:02:47 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
when (state === s_neg_outputs) {
|
|
|
|
when (neg_rem && neg_quo && !divby0) {
|
2012-02-12 02:20:33 +01:00
|
|
|
remainder := Cat(-remainder(2*width, width+1), remainder(width), -remainder(width-1,0)).toUFix;
|
2011-10-26 08:02:47 +02:00
|
|
|
}
|
2012-02-12 02:20:33 +01:00
|
|
|
.elsewhen (neg_quo && !divby0) {
|
|
|
|
remainder := Cat(remainder(2*width, width), -remainder(width-1,0)).toUFix;
|
2011-10-26 08:02:47 +02:00
|
|
|
}
|
2012-02-12 02:20:33 +01:00
|
|
|
.elsewhen (neg_rem) {
|
|
|
|
remainder := Cat(-remainder(2*width, width+1), remainder(width,0)).toUFix;
|
2011-10-26 08:02:47 +02:00
|
|
|
}
|
2012-02-12 02:20:33 +01:00
|
|
|
|
2011-10-26 08:02:47 +02:00
|
|
|
when (divisor(width-1).toBool) {
|
2012-02-12 02:20:33 +01:00
|
|
|
divisor := subtractor(width-1,0);
|
2011-10-26 08:02:47 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
when (state === s_busy) {
|
2012-02-12 02:20:33 +01:00
|
|
|
count := count + UFix(1);
|
|
|
|
divby0 := divby0 && !subtractor(width).toBool;
|
|
|
|
remainder := Mux(subtractor(width).toBool,
|
2011-10-26 08:02:47 +02:00
|
|
|
Cat(remainder(2*width-1, width), remainder(width-1,0), ~subtractor(width)),
|
|
|
|
Cat(subtractor(width-1, 0), remainder(width-1,0), ~subtractor(width))).toUFix;
|
|
|
|
}
|
|
|
|
|
|
|
|
val result = Mux(rem, remainder(2*width, width+1), remainder(width-1,0));
|
|
|
|
|
2012-02-09 12:47:59 +01:00
|
|
|
io.result := Mux(half, Cat(Fill(width/2, result(width/2-1)), result(width/2-1,0)), result);
|
|
|
|
io.result_tag := reg_tag;
|
|
|
|
io.result_val := (state === s_done);
|
|
|
|
|
2011-10-26 08:02:47 +02:00
|
|
|
io.div_rdy := (state === s_ready);
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|