2014-12-14 17:18:53 -05:00
|
|
|
module etx_io (/*AUTOARG*/
|
|
|
|
// Outputs
|
2015-04-13 23:35:21 -04:00
|
|
|
txo_lclk_p, txo_lclk_n, txo_frame_p, txo_frame_n, txo_data_p,
|
2015-11-08 23:28:39 -05:00
|
|
|
txo_data_n, tx_io_ack, tx_wr_wait, tx_rd_wait,
|
2014-12-14 17:18:53 -05:00
|
|
|
// Inputs
|
2015-11-08 23:28:39 -05:00
|
|
|
nreset, tx_lclk_io, tx_lclk90, txi_wr_wait_p, txi_wr_wait_n,
|
|
|
|
txi_rd_wait_p, txi_rd_wait_n, tx_packet, tx_access, tx_burst
|
2014-12-14 17:18:53 -05:00
|
|
|
);
|
2015-10-07 19:12:01 -04:00
|
|
|
|
2014-12-14 17:18:53 -05:00
|
|
|
parameter IOSTD_ELINK = "LVDS_25";
|
2015-09-14 22:03:22 -04:00
|
|
|
parameter PW = 104;
|
2015-11-06 22:41:43 -05:00
|
|
|
parameter ETYPE = 0; // 0 = parallella
|
|
|
|
// 1 = ephycard
|
2015-03-25 19:25:12 -04:00
|
|
|
//###########
|
2015-05-13 23:24:54 -04:00
|
|
|
//# reset, clocks
|
|
|
|
//##########
|
2015-11-06 16:51:57 -05:00
|
|
|
input nreset; //sync reset for io
|
2015-11-03 14:15:09 -05:00
|
|
|
input tx_lclk_io; //fast ODDR
|
2015-10-07 19:12:01 -04:00
|
|
|
input tx_lclk90; //fast 90deg shifted lclk
|
2014-12-14 17:18:53 -05:00
|
|
|
|
|
|
|
//###########
|
|
|
|
//# eLink pins
|
|
|
|
//###########
|
2015-05-13 23:24:54 -04:00
|
|
|
output txo_lclk_p, txo_lclk_n; // tx clock output
|
|
|
|
output txo_frame_p, txo_frame_n; // tx frame signal
|
|
|
|
output [7:0] txo_data_p, txo_data_n; // tx data (dual data rate)
|
|
|
|
input txi_wr_wait_p,txi_wr_wait_n; // tx write pushback
|
|
|
|
input txi_rd_wait_p, txi_rd_wait_n; // tx read pushback
|
2014-12-14 17:18:53 -05:00
|
|
|
|
|
|
|
//#############
|
2015-03-25 19:25:12 -04:00
|
|
|
//# Fabric interface
|
2014-12-14 17:18:53 -05:00
|
|
|
//#############
|
2015-05-13 23:24:54 -04:00
|
|
|
input [PW-1:0] tx_packet;
|
|
|
|
input tx_access;
|
|
|
|
input tx_burst;
|
2015-11-08 23:28:39 -05:00
|
|
|
output tx_io_ack;
|
2015-05-13 23:24:54 -04:00
|
|
|
output tx_wr_wait;
|
|
|
|
output tx_rd_wait;
|
2014-12-14 17:18:53 -05:00
|
|
|
|
2015-03-25 19:25:12 -04:00
|
|
|
//############
|
|
|
|
//# REGS
|
|
|
|
//############
|
2015-05-13 23:24:54 -04:00
|
|
|
reg [7:0] tx_pointer;
|
|
|
|
reg [15:0] tx_data16;
|
|
|
|
reg tx_access_reg;
|
|
|
|
reg tx_frame;
|
|
|
|
reg tx_io_wait_reg;
|
2015-05-18 15:38:30 -04:00
|
|
|
reg [PW-1:0] tx_packet_reg;
|
|
|
|
reg [63:0] tx_double;
|
|
|
|
reg [2:0] tx_state_reg;
|
|
|
|
reg [2:0] tx_state;
|
2014-12-14 17:18:53 -05:00
|
|
|
//############
|
|
|
|
//# WIRES
|
|
|
|
//############
|
2015-05-13 23:24:54 -04:00
|
|
|
wire new_tran;
|
|
|
|
wire access;
|
|
|
|
wire write;
|
|
|
|
wire [1:0] datamode;
|
|
|
|
wire [3:0] ctrlmode;
|
|
|
|
wire [31:0] dstaddr;
|
|
|
|
wire [31:0] data;
|
|
|
|
wire [31:0] srcaddr;
|
2015-05-15 15:31:01 -04:00
|
|
|
wire [7:0] txo_data;
|
|
|
|
wire txo_frame;
|
|
|
|
wire txo_lclk90;
|
2015-11-08 23:28:39 -05:00
|
|
|
reg tx_io_ack;
|
2015-11-09 21:57:25 -05:00
|
|
|
wire tx_new_frame;
|
|
|
|
wire tx_lclk90_ddr;
|
2015-10-07 19:12:01 -04:00
|
|
|
|
2014-12-14 17:18:53 -05:00
|
|
|
//#############################
|
2015-05-18 15:38:30 -04:00
|
|
|
//# Transmit state machine
|
2015-03-25 19:25:12 -04:00
|
|
|
//#############################
|
2015-05-18 15:38:30 -04:00
|
|
|
|
|
|
|
`define IDLE 3'b000
|
|
|
|
`define CYCLE1 3'b001
|
|
|
|
`define CYCLE2 3'b010
|
|
|
|
`define CYCLE3 3'b011
|
|
|
|
`define CYCLE4 3'b100
|
|
|
|
`define CYCLE5 3'b101
|
|
|
|
`define CYCLE6 3'b110
|
|
|
|
`define CYCLE7 3'b111
|
|
|
|
|
2015-11-08 23:28:39 -05:00
|
|
|
always @ (posedge tx_lclk_io)
|
2015-11-06 16:51:57 -05:00
|
|
|
if(!nreset)
|
2015-05-18 15:38:30 -04:00
|
|
|
tx_state[2:0] <= `IDLE;
|
|
|
|
else
|
|
|
|
case (tx_state[2:0])
|
|
|
|
`IDLE : tx_state[2:0] <= tx_access ? `CYCLE1 : `IDLE;
|
|
|
|
`CYCLE1 : tx_state[2:0] <= `CYCLE2;
|
|
|
|
`CYCLE2 : tx_state[2:0] <= `CYCLE3;
|
|
|
|
`CYCLE3 : tx_state[2:0] <= `CYCLE4;
|
|
|
|
`CYCLE4 : tx_state[2:0] <= `CYCLE5;
|
|
|
|
`CYCLE5 : tx_state[2:0] <= `CYCLE6;
|
|
|
|
`CYCLE6 : tx_state[2:0] <= `CYCLE7;
|
|
|
|
`CYCLE7 : tx_state[2:0] <= tx_burst ? `CYCLE4 : `IDLE;
|
|
|
|
endcase // case (tx_state)
|
2015-03-24 20:44:03 -04:00
|
|
|
|
2015-05-18 15:38:30 -04:00
|
|
|
assign tx_new_frame = (tx_state[2:0]==`CYCLE1);
|
|
|
|
|
|
|
|
|
2015-11-08 23:28:39 -05:00
|
|
|
//Creating wide acknowledge on cycle 4
|
|
|
|
always @ (posedge tx_lclk_io)
|
|
|
|
if(!nreset)
|
|
|
|
tx_io_ack <= 1'b0;
|
|
|
|
else if ((tx_state[2:0] ==`CYCLE4))
|
|
|
|
tx_io_ack <= 1'b1;
|
2015-11-04 20:02:45 -05:00
|
|
|
else if (tx_state[2:0]==`IDLE)
|
2015-11-08 23:28:39 -05:00
|
|
|
tx_io_ack <= 1'b0;
|
2015-05-18 15:38:30 -04:00
|
|
|
|
|
|
|
//Create frame signal for output
|
2015-11-08 23:28:39 -05:00
|
|
|
always @ (posedge tx_lclk_io)
|
2015-05-18 15:38:30 -04:00
|
|
|
begin
|
|
|
|
tx_state_reg[2:0] <= tx_state[2:0];
|
|
|
|
tx_frame <= |(tx_state_reg[2:0]);
|
|
|
|
end
|
2014-12-14 17:18:53 -05:00
|
|
|
|
2015-05-13 23:24:54 -04:00
|
|
|
//#############################
|
2015-05-18 15:38:30 -04:00
|
|
|
//# 2 CYCLE PACKET PIPELINE
|
2015-05-13 23:24:54 -04:00
|
|
|
//#############################
|
2015-11-08 23:28:39 -05:00
|
|
|
always @ (posedge tx_lclk_io)
|
2015-05-18 15:38:30 -04:00
|
|
|
if (tx_access)
|
|
|
|
tx_packet_reg[PW-1:0] <= tx_packet[PW-1:0];
|
|
|
|
|
2015-11-02 16:10:05 -05:00
|
|
|
packet2emesh p2e (
|
2015-05-18 15:38:30 -04:00
|
|
|
.write_out (write),
|
|
|
|
.datamode_out (datamode[1:0]),
|
|
|
|
.ctrlmode_out (ctrlmode[3:0]),
|
|
|
|
.dstaddr_out (dstaddr[31:0]),
|
|
|
|
.data_out (data[31:0]),
|
|
|
|
.srcaddr_out (srcaddr[31:0]),
|
|
|
|
.packet_in (tx_packet_reg[PW-1:0]));
|
|
|
|
|
2015-11-02 20:51:35 -05:00
|
|
|
|
|
|
|
/*
|
|
|
|
* The following format is used by the Epiphany multicore ASIC.
|
|
|
|
* Don't change it if you want to communicate with Epiphany.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
2015-11-08 23:28:39 -05:00
|
|
|
always @ (posedge tx_lclk_io)
|
2015-05-18 15:38:30 -04:00
|
|
|
if (tx_new_frame)
|
|
|
|
tx_double[63:0] <= {16'b0,//16
|
2015-06-30 13:32:05 +02:00
|
|
|
~write,7'b0,ctrlmode[3:0],//12
|
2015-11-02 16:10:05 -05:00
|
|
|
dstaddr[31:0],datamode[1:0],write,tx_access};//36
|
2015-05-18 15:38:30 -04:00
|
|
|
else if(tx_state[2:0]==`CYCLE4)
|
|
|
|
tx_double[63:0] <= {data[31:0],srcaddr[31:0]};
|
|
|
|
|
2015-05-13 23:24:54 -04:00
|
|
|
//#############################
|
2015-05-18 15:38:30 -04:00
|
|
|
//# SELECTING DATA FOR TRANSMIT
|
2015-05-13 23:24:54 -04:00
|
|
|
//#############################
|
2015-05-18 15:38:30 -04:00
|
|
|
|
2015-11-08 23:28:39 -05:00
|
|
|
always @ (posedge tx_lclk_io)
|
2015-05-18 15:38:30 -04:00
|
|
|
case(tx_state_reg[2:0])
|
2015-05-13 23:24:54 -04:00
|
|
|
//Cycle1
|
2015-05-18 15:38:30 -04:00
|
|
|
3'b001: tx_data16[15:0] <= tx_double[47:32];
|
2015-05-13 23:24:54 -04:00
|
|
|
//Cycle2
|
2015-05-18 15:38:30 -04:00
|
|
|
3'b010: tx_data16[15:0] <= tx_double[31:16];
|
2015-05-13 23:24:54 -04:00
|
|
|
//Cycle3
|
2015-05-18 15:38:30 -04:00
|
|
|
3'b011: tx_data16[15:0] <= tx_double[15:0];
|
2015-05-13 23:24:54 -04:00
|
|
|
//Cycle4
|
2015-05-18 15:38:30 -04:00
|
|
|
3'b100: tx_data16[15:0] <= tx_double[63:48];
|
2015-05-13 23:24:54 -04:00
|
|
|
//Cycle5
|
2015-05-18 15:38:30 -04:00
|
|
|
3'b101: tx_data16[15:0] <= tx_double[47:32];
|
2015-05-13 23:24:54 -04:00
|
|
|
//Cycle6
|
2015-05-18 15:38:30 -04:00
|
|
|
3'b110: tx_data16[15:0] <= tx_double[31:16];
|
|
|
|
//Cycle7
|
|
|
|
3'b111: tx_data16[15:0] <= tx_double[15:0];
|
|
|
|
default tx_data16[15:0] <= 16'b0;
|
|
|
|
endcase // case (tx_state[2:0])
|
2015-10-07 19:12:01 -04:00
|
|
|
|
2015-11-03 14:15:09 -05:00
|
|
|
//#############################
|
|
|
|
//# CLOCK DRIVERS
|
|
|
|
//#############################
|
2015-11-08 23:28:39 -05:00
|
|
|
//BUFIO i_lclk (.I(tx_lclk_io), .O(tx_lclk_ddr));
|
2015-11-03 14:15:09 -05:00
|
|
|
BUFIO i_lclk90 (.I(tx_lclk90), .O(tx_lclk90_ddr));
|
|
|
|
|
2015-05-13 23:24:54 -04:00
|
|
|
//#############################
|
2015-05-15 15:31:01 -04:00
|
|
|
//# ODDR DRIVERS
|
2015-05-13 23:24:54 -04:00
|
|
|
//#############################
|
|
|
|
|
2015-05-15 15:31:01 -04:00
|
|
|
//DATA
|
|
|
|
genvar i;
|
|
|
|
generate for(i=0; i<8; i=i+1)
|
|
|
|
begin : gen_oddr
|
|
|
|
ODDR #(.DDR_CLK_EDGE ("SAME_EDGE"))
|
|
|
|
oddr_data (
|
|
|
|
.Q (txo_data[i]),
|
2015-11-08 23:28:39 -05:00
|
|
|
.C (tx_lclk_io),
|
2015-05-15 15:31:01 -04:00
|
|
|
.CE (1'b1),
|
|
|
|
.D1 (tx_data16[i+8]),
|
|
|
|
.D2 (tx_data16[i]),
|
|
|
|
.R (1'b0),
|
|
|
|
.S (1'b0)
|
|
|
|
);
|
|
|
|
end
|
|
|
|
endgenerate
|
|
|
|
|
|
|
|
//FRAME
|
2015-11-03 14:15:09 -05:00
|
|
|
ODDR #(.DDR_CLK_EDGE ("SAME_EDGE"))
|
2015-05-15 15:31:01 -04:00
|
|
|
oddr_frame (
|
|
|
|
.Q (txo_frame),
|
2015-11-08 23:28:39 -05:00
|
|
|
.C (tx_lclk_io),
|
2015-05-15 15:31:01 -04:00
|
|
|
.CE (1'b1),
|
|
|
|
.D1 (tx_frame),
|
|
|
|
.D2 (tx_frame),
|
2015-10-08 10:34:59 -04:00
|
|
|
.R (1'b0), //reset
|
2015-05-15 15:31:01 -04:00
|
|
|
.S (1'b0)
|
|
|
|
);
|
|
|
|
|
|
|
|
//LCLK
|
|
|
|
ODDR #(.DDR_CLK_EDGE ("SAME_EDGE"))
|
|
|
|
oddr_lclk (
|
|
|
|
.Q (txo_lclk90),
|
2015-11-03 14:15:09 -05:00
|
|
|
.C (tx_lclk90_ddr),
|
2015-05-15 15:31:01 -04:00
|
|
|
.CE (1'b1),
|
|
|
|
.D1 (1'b1),
|
|
|
|
.D2 (1'b0),
|
2015-10-08 10:34:59 -04:00
|
|
|
.R (1'b0),//should be no reason to reset clock, static input
|
2015-05-15 15:31:01 -04:00
|
|
|
.S (1'b0)
|
|
|
|
);
|
2015-05-13 23:24:54 -04:00
|
|
|
|
|
|
|
//##############################
|
|
|
|
//# OUTPUT BUFFERS
|
|
|
|
//##############################
|
|
|
|
|
|
|
|
OBUFDS obufds_data[7:0] (
|
|
|
|
.O (txo_data_p[7:0]),
|
|
|
|
.OB (txo_data_n[7:0]),
|
2015-05-15 15:31:01 -04:00
|
|
|
.I (txo_data[7:0])
|
2015-05-13 23:24:54 -04:00
|
|
|
);
|
|
|
|
|
|
|
|
OBUFDS obufds_frame ( .O (txo_frame_p),
|
|
|
|
.OB (txo_frame_n),
|
2015-05-15 15:31:01 -04:00
|
|
|
.I (txo_frame)
|
2015-05-13 23:24:54 -04:00
|
|
|
);
|
|
|
|
|
|
|
|
OBUFDS obufds_lclk ( .O (txo_lclk_p),
|
|
|
|
.OB (txo_lclk_n),
|
2015-05-15 15:31:01 -04:00
|
|
|
.I (txo_lclk90)
|
2015-05-13 23:24:54 -04:00
|
|
|
);
|
|
|
|
|
2014-12-14 17:18:53 -05:00
|
|
|
//################################
|
|
|
|
//# Wait Input Buffers
|
|
|
|
//################################
|
2015-09-25 15:20:21 -04:00
|
|
|
|
2015-09-14 22:03:22 -04:00
|
|
|
generate
|
|
|
|
if(ETYPE==1)
|
|
|
|
begin
|
|
|
|
assign tx_wr_wait = txi_wr_wait_p;
|
|
|
|
end
|
|
|
|
else if (ETYPE==0)
|
|
|
|
begin
|
|
|
|
IBUFDS
|
|
|
|
#(.DIFF_TERM ("TRUE"), // Differential termination
|
|
|
|
.IOSTANDARD (IOSTD_ELINK))
|
|
|
|
ibufds_wrwait
|
|
|
|
(.I (txi_wr_wait_p),
|
|
|
|
.IB (txi_wr_wait_n),
|
|
|
|
.O (tx_wr_wait));
|
|
|
|
end
|
|
|
|
endgenerate
|
2015-11-08 23:28:39 -05:00
|
|
|
|
2015-04-13 23:35:21 -04:00
|
|
|
//TODO: Come up with cleaner defines for this
|
|
|
|
//Parallella and other platforms...
|
|
|
|
`ifdef TODO
|
|
|
|
IBUFDS
|
|
|
|
#(.DIFF_TERM ("TRUE"), // Differential termination
|
|
|
|
.IOSTANDARD (IOSTD_ELINK))
|
2015-09-14 22:03:22 -04:00
|
|
|
ibufds_rdwait
|
2015-04-13 23:35:21 -04:00
|
|
|
(.I (txi_rd_wait_p),
|
|
|
|
.IB (txi_rd_wait_n),
|
|
|
|
.O (tx_rd_wait));
|
|
|
|
`else
|
2015-03-25 19:25:12 -04:00
|
|
|
//On Parallella this signal comes in single-ended
|
2015-09-27 08:40:36 -04:00
|
|
|
assign tx_rd_wait = txi_rd_wait_p;
|
2015-04-13 23:35:21 -04:00
|
|
|
`endif
|
|
|
|
|
2015-03-25 19:25:12 -04:00
|
|
|
endmodule // etx_io
|
2015-05-13 23:24:54 -04:00
|
|
|
// Local Variables:
|
|
|
|
// verilog-library-directories:("." "../../emesh/hdl")
|
|
|
|
// End:
|
2014-12-14 17:18:53 -05:00
|
|
|
|