2015-08-14 15:37:37 -04:00
|
|
|
/*
|
|
|
|
This block receives the IO transaction and converts to a 104 bit packet.
|
|
|
|
*/
|
2015-07-01 00:14:46 +02:00
|
|
|
`include "elink_constants.v"
|
2014-12-14 17:18:53 -05:00
|
|
|
module erx_io (/*AUTOARG*/
|
|
|
|
// Outputs
|
2015-05-14 22:31:42 -04:00
|
|
|
rx_lclk_pll, rxo_wr_wait_p, rxo_wr_wait_n, rxo_rd_wait_p,
|
2015-05-13 23:26:41 -04:00
|
|
|
rxo_rd_wait_n, rx_access, rx_burst, rx_packet,
|
2014-12-14 17:18:53 -05:00
|
|
|
// Inputs
|
2015-08-24 21:08:49 +02:00
|
|
|
reset, rx_lclk, rx_lclk_div4, rxi_lclk_p, rxi_lclk_n,
|
2015-05-14 22:31:42 -04:00
|
|
|
rxi_frame_p, rxi_frame_n, rxi_data_p, rxi_data_n, rx_wr_wait,
|
|
|
|
rx_rd_wait
|
2014-12-14 17:18:53 -05:00
|
|
|
);
|
|
|
|
|
2015-06-30 13:32:05 +02:00
|
|
|
parameter IOSTD_ELINK = "LVDS_25";
|
2015-05-13 23:26:41 -04:00
|
|
|
parameter PW = 104;
|
2015-04-08 13:20:25 -04:00
|
|
|
|
2015-08-24 22:39:51 +02:00
|
|
|
// Can we do this in a better way?
|
|
|
|
parameter [8:0] RX_TAP_DELAY [8:0] = {10,11,10,10,10,11,11,11,11};
|
|
|
|
|
|
|
|
|
2015-05-13 23:26:41 -04:00
|
|
|
//#########################
|
|
|
|
//# reset, clocks
|
|
|
|
//#########################
|
|
|
|
input reset; // reset
|
|
|
|
input rx_lclk; // fast I/O clock
|
|
|
|
input rx_lclk_div4; // slow clock
|
2015-05-14 22:31:42 -04:00
|
|
|
output rx_lclk_pll; // clock output for pll
|
2015-05-13 23:26:41 -04:00
|
|
|
|
|
|
|
//##########################
|
|
|
|
//# elink pins
|
|
|
|
//##########################
|
|
|
|
input rxi_lclk_p, rxi_lclk_n; // rx clock input
|
|
|
|
input rxi_frame_p, rxi_frame_n; // rx frame signal
|
|
|
|
input [7:0] rxi_data_p, rxi_data_n; // rx data
|
|
|
|
output rxo_wr_wait_p,rxo_wr_wait_n; // rx write pushback output
|
|
|
|
output rxo_rd_wait_p,rxo_rd_wait_n; // rx read pushback output
|
2015-04-13 23:35:21 -04:00
|
|
|
|
2015-05-13 23:26:41 -04:00
|
|
|
//##########################
|
|
|
|
//# erx logic interface
|
|
|
|
//##########################
|
|
|
|
output rx_access;
|
|
|
|
output rx_burst;
|
|
|
|
output [PW-1:0] rx_packet;
|
|
|
|
input rx_wr_wait;
|
|
|
|
input rx_rd_wait;
|
2014-12-14 17:18:53 -05:00
|
|
|
|
|
|
|
//############
|
|
|
|
//# WIRES
|
|
|
|
//############
|
2015-05-15 15:31:01 -04:00
|
|
|
wire [7:0] rxi_data;
|
|
|
|
wire rxi_frame;
|
|
|
|
wire rxi_lclk;
|
2015-05-13 23:26:41 -04:00
|
|
|
wire access_wide;
|
|
|
|
reg valid_packet;
|
2015-05-15 15:31:01 -04:00
|
|
|
wire [15:0] rx_word;
|
2015-08-24 21:39:31 +02:00
|
|
|
reg [15:0] rx_word_sync;
|
|
|
|
|
2015-05-13 23:26:41 -04:00
|
|
|
//############
|
|
|
|
//# REGS
|
|
|
|
//############
|
|
|
|
reg [7:0] data_even_reg;
|
|
|
|
reg [7:0] data_odd_reg;
|
2015-08-24 21:39:31 +02:00
|
|
|
wire [1:0] rx_frame;
|
|
|
|
reg [1:] rx_frame_sync;
|
2015-05-15 15:31:01 -04:00
|
|
|
wire rx_frame_old;
|
2015-05-13 23:26:41 -04:00
|
|
|
reg [111:0] rx_sample;
|
2015-05-15 15:31:01 -04:00
|
|
|
reg [6:0] rx_pointer;
|
2015-05-13 23:26:41 -04:00
|
|
|
reg access;
|
|
|
|
reg burst;
|
|
|
|
reg [PW-1:0] rx_packet_lclk;
|
|
|
|
reg rx_access;
|
|
|
|
reg [PW-1:0] rx_packet;
|
|
|
|
reg rx_burst;
|
2015-05-15 15:31:01 -04:00
|
|
|
wire rx_lclk_iddr;
|
2015-05-16 22:06:40 -04:00
|
|
|
wire [8:0] rxi_delay_in;
|
|
|
|
wire [8:0] rxi_delay_out;
|
2015-08-24 21:39:31 +02:00
|
|
|
wire reset_sync;
|
|
|
|
|
|
|
|
//Reset sync
|
|
|
|
always @ (posedge rx_lclk)
|
|
|
|
reset_sync <= reset;
|
2014-12-14 17:18:53 -05:00
|
|
|
|
2015-05-13 23:26:41 -04:00
|
|
|
//#####################
|
|
|
|
//#CREATE 112 BIT PACKET
|
|
|
|
//#####################
|
2015-08-24 21:39:31 +02:00
|
|
|
//TODO: clean up!
|
2015-05-13 23:26:41 -04:00
|
|
|
//write Pointer
|
|
|
|
always @ (posedge rx_lclk)
|
2015-08-24 21:39:31 +02:00
|
|
|
if (~rx_frame_sync[1]) begin
|
|
|
|
rx_pointer <= 3'b0; //new frame
|
|
|
|
access <= 1'b0;
|
|
|
|
end
|
|
|
|
else if (rx_pointer != 3'd6 && (rx_frame_sync != 2'b00))
|
2015-05-13 23:26:41 -04:00
|
|
|
begin
|
2015-08-24 21:39:31 +02:00
|
|
|
rx_pointer <= rx_pointer + 1; //anticipate burst
|
|
|
|
access <= 1'b0;
|
|
|
|
end
|
|
|
|
else begin
|
|
|
|
rx_pointer <= 3'd3;//middle of frame
|
|
|
|
access <= 1'b1;
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
|
|
// shift register for rx_word
|
|
|
|
always @ (posedge rx_lclk)
|
|
|
|
case (rx_frame_sync[1:0])
|
|
|
|
2'b01 : begin
|
|
|
|
rx_sample[111:8] <= rx_sample[103:0];
|
|
|
|
rx_sample[7:0] <= rx_word_sync[7:0];
|
|
|
|
end
|
|
|
|
2'b10 : begin
|
|
|
|
rx_sample[111:8] <= rx_sample[103:0];
|
|
|
|
rx_sample[7:0] <= rx_word_sync[15:8];
|
|
|
|
end
|
|
|
|
2'b11 : begin
|
|
|
|
rx_sample[111:16] <= rx_sample[95:0];
|
|
|
|
rx_sample[15:0] <= rx_word_sync[15:0];
|
2015-05-13 23:26:41 -04:00
|
|
|
end
|
2015-08-24 21:39:31 +02:00
|
|
|
default :
|
|
|
|
rx_sample <= rx_sample;
|
|
|
|
endcase // case (rx_frame_sync[1:0])
|
|
|
|
|
2014-12-14 17:18:53 -05:00
|
|
|
|
2015-05-13 23:26:41 -04:00
|
|
|
//#####################
|
|
|
|
//#DATA VALID SIGNAL
|
2015-05-15 15:31:01 -04:00
|
|
|
//####################
|
2015-08-24 21:39:31 +02:00
|
|
|
|
2015-05-13 23:26:41 -04:00
|
|
|
always @ (posedge rx_lclk)
|
|
|
|
begin
|
|
|
|
valid_packet <= access;//data pipeline
|
|
|
|
end
|
2015-05-18 15:38:30 -04:00
|
|
|
|
|
|
|
reg burst_detect;
|
|
|
|
always @ (posedge rx_lclk)
|
2015-08-24 21:39:31 +02:00
|
|
|
if(access & rx_frame_sync[1])
|
2015-05-18 15:38:30 -04:00
|
|
|
burst_detect <= 1'b1;
|
2015-08-24 21:39:31 +02:00
|
|
|
else if(~rx_frame_sync)
|
2015-05-18 15:38:30 -04:00
|
|
|
burst_detect <= 1'b0;
|
|
|
|
|
2015-05-13 23:26:41 -04:00
|
|
|
//###################################
|
|
|
|
//#SAMPLE AND HOLD DATA
|
|
|
|
//###################################
|
|
|
|
|
|
|
|
//(..and shuffle data for 104 bit packet)
|
|
|
|
always @ (posedge rx_lclk)
|
|
|
|
if(access)
|
|
|
|
begin
|
2015-05-18 15:38:30 -04:00
|
|
|
//pipelin burst (delay by one frame)
|
|
|
|
burst <= burst_detect;
|
2015-08-24 21:39:31 +02:00
|
|
|
|
2015-05-13 23:26:41 -04:00
|
|
|
//access
|
2015-08-24 21:39:31 +02:00
|
|
|
rx_packet_lclk[0] <= rx_sample[64];
|
|
|
|
|
2015-05-13 23:26:41 -04:00
|
|
|
//write
|
2015-08-24 21:39:31 +02:00
|
|
|
rx_packet_lclk[1] <= rx_sample[65];
|
|
|
|
|
2015-05-13 23:26:41 -04:00
|
|
|
//datamode
|
2015-08-24 21:39:31 +02:00
|
|
|
rx_packet_lclk[3:2] <= rx_sample[67:66];
|
|
|
|
|
2015-05-13 23:26:41 -04:00
|
|
|
//ctrlmode
|
2015-08-24 21:39:31 +02:00
|
|
|
rx_packet_lclk[7:4] <= rx_sample[103:100];
|
|
|
|
|
2015-05-13 23:26:41 -04:00
|
|
|
//dstaddr
|
2015-08-24 21:39:31 +02:00
|
|
|
rx_packet_lclk[39:8] <= rx_sample[99:68];
|
|
|
|
|
2015-05-13 23:26:41 -04:00
|
|
|
//data
|
2015-08-24 21:39:31 +02:00
|
|
|
rx_packet_lclk[71:40] <= rx_sample[63:32];
|
|
|
|
|
2015-05-13 23:26:41 -04:00
|
|
|
//srcaddr
|
2015-08-24 21:39:31 +02:00
|
|
|
rx_packet_lclk[103:72]<= rx_sample[31:0];
|
|
|
|
end
|
2015-05-13 23:26:41 -04:00
|
|
|
|
|
|
|
//###################################
|
|
|
|
//#SYNCHRONIZE TO SLOW CLK
|
|
|
|
//###################################
|
|
|
|
|
|
|
|
//stretch access pulse to 4 cycles
|
2015-08-24 22:39:51 +02:00
|
|
|
pulse_stretcher #(.DW(3))
|
|
|
|
ps0 (
|
|
|
|
.out(access_wide),
|
|
|
|
.in(valid_packet),
|
|
|
|
.clk(rx_lclk),
|
|
|
|
.reset(reset_sync));
|
2015-05-13 23:26:41 -04:00
|
|
|
|
2015-08-24 22:39:51 +02:00
|
|
|
|
2015-05-13 23:26:41 -04:00
|
|
|
always @ (posedge rx_lclk_div4)
|
|
|
|
rx_access <= access_wide;
|
|
|
|
|
|
|
|
always @ (posedge rx_lclk_div4)
|
|
|
|
if(access_wide)
|
|
|
|
begin
|
|
|
|
rx_packet[PW-1:0] <= rx_packet_lclk[PW-1:0];
|
2015-05-18 15:38:30 -04:00
|
|
|
rx_burst <= burst;
|
2015-05-13 23:26:41 -04:00
|
|
|
end
|
2014-12-14 17:18:53 -05:00
|
|
|
|
2015-05-16 22:06:40 -04:00
|
|
|
|
|
|
|
//################################
|
|
|
|
//# I/O Buffers Instantiation
|
|
|
|
//################################
|
|
|
|
|
2015-06-30 13:32:05 +02:00
|
|
|
IBUFDS #(.DIFF_TERM ("TRUE"),.IOSTANDARD (IOSTD_ELINK))
|
2015-05-16 22:06:40 -04:00
|
|
|
ibuf_data[7:0]
|
|
|
|
(.I (rxi_data_p[7:0]),
|
|
|
|
.IB (rxi_data_n[7:0]),
|
|
|
|
.O (rxi_data[7:0]));
|
|
|
|
|
2015-06-30 13:32:05 +02:00
|
|
|
IBUFDS #(.DIFF_TERM ("TRUE"), .IOSTANDARD (IOSTD_ELINK))
|
2015-05-16 22:06:40 -04:00
|
|
|
ibuf_frame
|
|
|
|
(.I (rxi_frame_p),
|
|
|
|
.IB (rxi_frame_n),
|
|
|
|
.O (rxi_frame));
|
|
|
|
|
2015-06-30 13:32:05 +02:00
|
|
|
IBUFDS #(.DIFF_TERM ("TRUE"),.IOSTANDARD (IOSTD_ELINK))
|
2015-05-16 22:06:40 -04:00
|
|
|
ibuf_lclk (.I (rxi_lclk_p),
|
|
|
|
.IB (rxi_lclk_n),
|
|
|
|
.O (rxi_lclk)
|
|
|
|
);
|
2015-07-01 00:14:46 +02:00
|
|
|
|
|
|
|
`ifdef EPHYCARD
|
|
|
|
OBUFT #(.IOSTANDARD("LVCMOS18"), .SLEW("SLOW"))
|
|
|
|
obuft_wrwait (
|
|
|
|
.O(rxo_wr_wait_p),
|
|
|
|
.T(rx_wr_wait),
|
|
|
|
.I(1'b0)
|
|
|
|
);
|
|
|
|
|
|
|
|
OBUFT #(.IOSTANDARD("LVCMOS18"), .SLEW("SLOW"))
|
|
|
|
obuft_rdwait (
|
|
|
|
.O(rxo_rd_wait_p),
|
|
|
|
.T(rx_rd_wait),
|
|
|
|
.I(1'b0)
|
|
|
|
);
|
|
|
|
|
|
|
|
`else
|
2015-05-16 22:06:40 -04:00
|
|
|
|
2015-06-30 13:32:05 +02:00
|
|
|
OBUFDS #(.IOSTANDARD(IOSTD_ELINK),.SLEW("SLOW"))
|
2015-05-16 22:06:40 -04:00
|
|
|
obufds_wrwait (
|
|
|
|
.O(rxo_wr_wait_p),
|
|
|
|
.OB(rxo_wr_wait_n),
|
|
|
|
.I(rx_wr_wait)
|
|
|
|
);
|
2014-12-14 17:18:53 -05:00
|
|
|
|
2015-06-30 13:32:05 +02:00
|
|
|
OBUFDS #(.IOSTANDARD(IOSTD_ELINK),.SLEW("SLOW"))
|
2015-05-16 22:06:40 -04:00
|
|
|
obufds_rdwait (.O(rxo_rd_wait_p),
|
|
|
|
.OB(rxo_rd_wait_n),
|
|
|
|
.I(rx_rd_wait)
|
|
|
|
);
|
2015-07-01 00:14:46 +02:00
|
|
|
`endif
|
2015-05-16 22:06:40 -04:00
|
|
|
//###################################
|
|
|
|
//#RX CLOCK
|
|
|
|
//###################################
|
2015-05-18 15:38:30 -04:00
|
|
|
|
2015-08-24 22:39:51 +02:00
|
|
|
assign rx_lclk_pll = rxi_lclk;
|
2015-05-16 22:06:40 -04:00
|
|
|
|
|
|
|
//###################################
|
|
|
|
//#IDELAY CIRCUIT
|
|
|
|
//###################################
|
|
|
|
|
|
|
|
assign rxi_delay_in[8:0] ={rxi_frame,rxi_data[7:0]};
|
|
|
|
|
|
|
|
genvar j;
|
|
|
|
generate for(j=0; j<9; j=j+1)
|
2015-08-24 21:08:49 +02:00
|
|
|
begin : gen_idelay
|
|
|
|
(* IODELAY_GROUP = "IDELAY_GROUP" *) // Group name for IDELAYCTRL
|
2015-05-16 22:06:40 -04:00
|
|
|
IDELAYE2 #(.CINVCTRL_SEL("FALSE"),
|
|
|
|
.DELAY_SRC("IDATAIN"),
|
|
|
|
.HIGH_PERFORMANCE_MODE("FALSE"),
|
|
|
|
.IDELAY_TYPE("FIXED"),
|
2015-08-24 22:39:51 +02:00
|
|
|
.IDELAY_VALUE(RX_TAP_DELAY[j]),
|
2015-05-16 22:06:40 -04:00
|
|
|
.PIPE_SEL("FALSE"),
|
|
|
|
.REFCLK_FREQUENCY(200.0),
|
|
|
|
.SIGNAL_PATTERN("DATA"))
|
|
|
|
idelay_inst (.CNTVALUEOUT(),
|
|
|
|
.DATAOUT(rxi_delay_out[j]),
|
|
|
|
.C(1'b0),
|
|
|
|
.CE(1'b0),
|
|
|
|
.CINVCTRL(1'b0),
|
|
|
|
.CNTVALUEIN(5'b0),
|
|
|
|
.DATAIN(1'b0),
|
|
|
|
.IDATAIN(rxi_delay_in[j]),
|
|
|
|
.INC(1'b0),
|
|
|
|
.LD(1'b0),
|
|
|
|
.LDPIPEEN(1'b0),
|
|
|
|
.REGRST(1'b0)
|
|
|
|
);
|
|
|
|
end // block: gen_idelay
|
|
|
|
endgenerate
|
|
|
|
|
|
|
|
|
|
|
|
//#############################
|
|
|
|
//# IDDR SAMPLERS
|
|
|
|
//#############################
|
2015-08-24 22:39:51 +02:00
|
|
|
|
2015-05-16 22:06:40 -04:00
|
|
|
//DATA
|
|
|
|
genvar i;
|
|
|
|
generate for(i=0; i<8; i=i+1)
|
|
|
|
begin : gen_iddr
|
2015-08-24 21:39:31 +02:00
|
|
|
IDDR #(.DDR_CLK_EDGE ("SAME_EDGE"), .SRTYPE("ASYNC"))
|
2015-05-16 22:06:40 -04:00
|
|
|
iddr_data (
|
|
|
|
.Q1 (rx_word[i]),
|
|
|
|
.Q2 (rx_word[i+8]),
|
2015-08-24 22:39:51 +02:00
|
|
|
.C (rx_lclk),
|
2015-05-16 22:06:40 -04:00
|
|
|
.CE (1'b1),
|
|
|
|
.D (rxi_delay_out[i]),
|
2015-08-24 21:39:31 +02:00
|
|
|
.R (reset_sync),
|
2015-05-16 22:06:40 -04:00
|
|
|
.S (1'b0)
|
|
|
|
);
|
|
|
|
end
|
|
|
|
endgenerate
|
2015-05-13 23:26:41 -04:00
|
|
|
|
2015-05-16 22:06:40 -04:00
|
|
|
//FRAME
|
2015-08-24 21:39:31 +02:00
|
|
|
IDDR #(.DDR_CLK_EDGE ("SAME_EDGE"), .SRTYPE("ASYNC"))
|
2015-05-16 22:06:40 -04:00
|
|
|
iddr_frame (
|
2015-08-24 21:39:31 +02:00
|
|
|
.Q1 (rx_frame[0]),
|
|
|
|
.Q2 (rx_frame[1]),
|
2015-08-24 22:39:51 +02:00
|
|
|
.C (rx_lclk),
|
2015-05-16 22:06:40 -04:00
|
|
|
.CE (1'b1),
|
|
|
|
.D (rxi_delay_out[8]),
|
2015-08-24 21:39:31 +02:00
|
|
|
.R (reset_sync),
|
2015-05-16 22:06:40 -04:00
|
|
|
.S (1'b0)
|
2015-08-24 21:39:31 +02:00
|
|
|
);
|
|
|
|
|
|
|
|
//Pipe stage
|
|
|
|
always @ (posedge rx_lclk)
|
|
|
|
rx_frame_sync <= rx_frame;
|
|
|
|
|
|
|
|
//Pipe stage
|
|
|
|
always @ (posedge rx_lclk)
|
|
|
|
rx_word_sync <= rx_word;
|
|
|
|
|
2015-05-16 22:06:40 -04:00
|
|
|
|
2015-04-14 14:00:23 -04:00
|
|
|
endmodule // erx_io
|
2015-05-13 23:26:41 -04:00
|
|
|
// Local Variables:
|
|
|
|
// verilog-library-directories:("." "../../emesh/hdl" "../../common/hdl")
|
|
|
|
// End:
|
2015-04-14 14:00:23 -04:00
|
|
|
|
|
|
|
/*
|
2015-05-13 23:26:41 -04:00
|
|
|
Copyright (C) 2014 Adapteva, Inc.
|
2015-05-14 22:31:42 -04:00
|
|
|
Contributed by Andreas Olofsson <andreas@adapteva.com>
|
2015-05-16 22:06:40 -04:00
|
|
|
Contributed by Gunnar Hillerstrom
|
2015-04-14 14:00:23 -04:00
|
|
|
|
2015-05-13 23:26:41 -04:00
|
|
|
This program is free software: you can redistribute it and/or modify
|
|
|
|
it under the terms of the GNU General Public License as published by
|
|
|
|
the Free Software Foundation, either version 3 of the License, or
|
|
|
|
(at your option) any later version.
|
|
|
|
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
GNU General Public License for more details.
|
|
|
|
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
|
|
along with this program (see the file COPYING). If not, see
|
|
|
|
<http://www.gnu.org/licenses/>.
|
2015-04-14 14:00:23 -04:00
|
|
|
*/
|