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-05-13 23:24:54 -04:00
|
|
|
txo_data_n, tx_io_wait, tx_wr_wait, tx_rd_wait,
|
2014-12-14 17:18:53 -05:00
|
|
|
// Inputs
|
2015-05-13 23:24:54 -04:00
|
|
|
ioreset, tx_lclk, tx_lclk90, tx_lclk_div4, 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
|
|
|
);
|
|
|
|
|
|
|
|
parameter IOSTD_ELINK = "LVDS_25";
|
2015-05-13 23:24:54 -04:00
|
|
|
parameter PW = 104;
|
|
|
|
|
2015-03-25 19:25:12 -04:00
|
|
|
//###########
|
2015-05-13 23:24:54 -04:00
|
|
|
//# reset, clocks
|
|
|
|
//##########
|
|
|
|
input ioreset; //reset for io
|
|
|
|
input tx_lclk; // fast clock for io
|
|
|
|
input tx_lclk90; // fast 90deg shifted lclk
|
|
|
|
input tx_lclk_div4; // slow clock for rest of logic
|
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;
|
|
|
|
output tx_io_wait;
|
|
|
|
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 [7:0] tx_data;
|
|
|
|
reg tx_access_reg;
|
|
|
|
reg tx_frame;
|
|
|
|
reg tx_io_wait_reg;
|
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;
|
|
|
|
|
2014-12-14 17:18:53 -05:00
|
|
|
//#############################
|
2015-05-13 23:24:54 -04:00
|
|
|
//# Disassemble packet (for clarity)
|
2015-03-25 19:25:12 -04:00
|
|
|
//#############################
|
2015-05-13 23:24:54 -04:00
|
|
|
packet2emesh p2e (
|
|
|
|
// Outputs
|
|
|
|
.access_out (access),
|
|
|
|
.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]),
|
|
|
|
// Inputs
|
|
|
|
.packet_in (tx_packet[PW-1:0]));
|
2015-03-24 20:44:03 -04:00
|
|
|
|
2015-05-13 23:24:54 -04:00
|
|
|
|
|
|
|
//#############################
|
|
|
|
//# Transaction state machine
|
|
|
|
//#############################
|
|
|
|
always @ (posedge tx_lclk)
|
|
|
|
if (~tx_access)
|
|
|
|
tx_pointer[7:0] <= 8'b00000001; //new transaction
|
|
|
|
else if (tx_pointer[6] & tx_burst)
|
|
|
|
tx_pointer[7:0] <= 8'b00001000; //burst
|
|
|
|
else
|
|
|
|
tx_pointer[7:0] <= {tx_pointer[6:0],1'b0};
|
2014-12-14 17:18:53 -05:00
|
|
|
|
2015-05-13 23:24:54 -04:00
|
|
|
//#############################
|
|
|
|
//# Frame Signal
|
|
|
|
//#############################
|
|
|
|
//TODO: cleanup
|
|
|
|
assign tx_io_wait = tx_access & ~tx_burst & ~tx_io_wait_reg;
|
|
|
|
|
|
|
|
always @ (posedge tx_lclk_div4 or posedge ioreset)
|
|
|
|
if(ioreset)
|
|
|
|
tx_io_wait_reg <= 1'b0;
|
|
|
|
else
|
|
|
|
tx_io_wait_reg <= tx_io_wait;
|
|
|
|
|
|
|
|
//#############################
|
|
|
|
//# Frame Signal
|
|
|
|
//#############################
|
|
|
|
always @ (posedge tx_lclk or posedge ioreset)
|
|
|
|
if(ioreset)
|
|
|
|
tx_frame <= 1'b0;
|
|
|
|
else if(tx_pointer[0] & tx_access)
|
|
|
|
tx_frame <= 1'b1;
|
|
|
|
else if(tx_pointer[7] & ~tx_burst)
|
|
|
|
tx_frame <= 1'b0;
|
|
|
|
|
|
|
|
//#############################
|
|
|
|
//# SELECTING DATA PER CYCLE
|
|
|
|
//#############################
|
|
|
|
//optimize later...
|
|
|
|
always @ (negedge tx_lclk)
|
|
|
|
case(tx_pointer[6:0])
|
|
|
|
//Cycle0
|
|
|
|
7'b0000001: tx_data16[15:0] <= {ctrlmode[3:0],dstaddr[31:28],~write,7'b0};
|
|
|
|
//Cycle1
|
|
|
|
7'b0000010: tx_data16[15:0] <= {dstaddr[19:12],dstaddr[27:20]};
|
|
|
|
//Cycle2
|
|
|
|
7'b0000100: tx_data16[15:0] <= {dstaddr[3:0],datamode[1:0],write,access,
|
|
|
|
dstaddr[11:4]};
|
|
|
|
//Cycle3
|
|
|
|
7'b0001000: tx_data16[15:0] <= {data[23:16],data[31:24]};
|
|
|
|
//Cycle4
|
|
|
|
7'b0010000: tx_data16[15:0] <= {data[7:0],data[15:8]};
|
|
|
|
//Cycle5
|
|
|
|
7'b0100000: tx_data16[15:0] <= {srcaddr[23:16],srcaddr[31:24]};
|
|
|
|
//Cycle6
|
|
|
|
7'b1000000: tx_data16[15:0] <= {srcaddr[7:0],srcaddr[15:8]};
|
|
|
|
default tx_data16[15:0] <= 16'b0;
|
|
|
|
endcase // case (tx_pointer[7:0])
|
|
|
|
|
|
|
|
//#############################
|
|
|
|
//# DATA (DDR)
|
|
|
|
//#############################
|
|
|
|
always @ (negedge tx_lclk)
|
|
|
|
tx_data[7:0] <= tx_data16[15:8];
|
|
|
|
|
|
|
|
always @ (posedge tx_lclk)
|
|
|
|
tx_data[7:0] <= tx_data16[7:0];
|
|
|
|
|
|
|
|
//##############################
|
|
|
|
//# OUTPUT BUFFERS
|
|
|
|
//##############################
|
|
|
|
|
|
|
|
OBUFDS obufds_data[7:0] (
|
|
|
|
.O (txo_data_p[7:0]),
|
|
|
|
.OB (txo_data_n[7:0]),
|
|
|
|
.I (tx_data)
|
|
|
|
);
|
|
|
|
|
|
|
|
OBUFDS obufds_frame ( .O (txo_frame_p),
|
|
|
|
.OB (txo_frame_n),
|
|
|
|
.I (tx_frame)
|
|
|
|
);
|
|
|
|
|
|
|
|
OBUFDS obufds_lclk ( .O (txo_lclk_p),
|
|
|
|
.OB (txo_lclk_n),
|
|
|
|
.I (tx_lclk90)
|
|
|
|
);
|
|
|
|
|
2014-12-14 17:18:53 -05:00
|
|
|
//################################
|
|
|
|
//# Wait Input Buffers
|
|
|
|
//################################
|
2015-03-25 19:25:12 -04:00
|
|
|
|
2014-12-14 17:18:53 -05:00
|
|
|
IBUFDS
|
2015-03-25 19:25:12 -04:00
|
|
|
#(.DIFF_TERM ("TRUE"), // Differential termination
|
2014-12-14 17:18:53 -05:00
|
|
|
.IOSTANDARD (IOSTD_ELINK))
|
2015-05-13 23:24:54 -04:00
|
|
|
ibufds_wrwait
|
2015-04-13 23:35:21 -04:00
|
|
|
(.I (txi_wr_wait_p),
|
|
|
|
.IB (txi_wr_wait_n),
|
2014-12-14 17:18:53 -05:00
|
|
|
.O (tx_wr_wait));
|
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-05-13 23:24:54 -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-04-13 23:35:21 -04:00
|
|
|
assign tx_rd_wait = txi_rd_wait_p;
|
|
|
|
`endif
|
|
|
|
|
2015-04-08 13:20:25 -04:00
|
|
|
|
2015-04-13 23:35:21 -04:00
|
|
|
|
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
|
|
|
|
2015-04-13 23:35:21 -04:00
|
|
|
|
2015-05-13 23:24:54 -04:00
|
|
|
/*
|
2015-04-13 23:35:21 -04:00
|
|
|
Copyright (C) 2014 Adapteva, Inc.
|
2015-04-17 22:21:08 -04:00
|
|
|
Contributed by Andreas Olofsson <andreas@adapteva.com>
|
|
|
|
|
2015-04-13 23:35:21 -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/>.
|
|
|
|
*/
|