URL
https://opencores.org/ocsvn/ulpi_wrapper/ulpi_wrapper/trunk
Subversion Repositories ulpi_wrapper
[/] [ulpi_wrapper/] [trunk/] [rtl/] [ulpi_wrapper.v] - Rev 3
Go to most recent revision | Compare with Previous | Blame | View Log
//----------------------------------------------------------------- // ULPI (Link) Wrapper // V0.1 // Ultra-Embedded.com // Copyright 2015 // // Email: admin@ultra-embedded.com // // License: GPL // If you would like a version with a more permissive license for // use in closed source commercial applications please contact me // for details. //----------------------------------------------------------------- // // This file is open source HDL; 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 2 of // the License, or (at your option) any later version. // // This file 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 file; if not, write to the Free Software // Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 // USA //----------------------------------------------------------------- //----------------------------------------------------------------- // Module: UTMI+ to ULPI Wrapper // // Description: // - Converts from UTMI interface to reduced pin count ULPI. // - No support for low power mode. // - I/O synchronous to 60MHz ULPI clock input (from PHY) // - Tested against SMSC/Microchip USB3300 in device mode. //----------------------------------------------------------------- module ulpi_wrapper ( // ULPI Interface (PHY) input ulpi_clk60_i, input ulpi_rst_i, input [7:0] ulpi_data_i, output [7:0] ulpi_data_o, input ulpi_dir_i, input ulpi_nxt_i, output ulpi_stp_o, // Register access (Wishbone pipelined access type) // NOTE: Tie inputs to 0 if unused input [7:0] reg_addr_i, input reg_stb_i, input reg_we_i, input [7:0] reg_data_i, output [7:0] reg_data_o, output reg_ack_o, // UTMI Interface (SIE) input utmi_txvalid_i, output utmi_txready_o, output utmi_rxvalid_o, output utmi_rxactive_o, output utmi_rxerror_o, output [7:0] utmi_data_o, input [7:0] utmi_data_i, input [1:0] utmi_xcvrselect_i, input utmi_termselect_i, input [1:0] utmi_opmode_i, input utmi_dppulldown_i, input utmi_dmpulldown_i, output [1:0] utmi_linestate_o ); //----------------------------------------------------------------- // States //----------------------------------------------------------------- localparam STATE_W = 2; localparam STATE_IDLE = 2'd0; localparam STATE_CMD = 2'd1; localparam STATE_DATA = 2'd2; localparam STATE_REG = 2'd3; reg [STATE_W-1:0] state_q; //----------------------------------------------------------------- // Local Params //----------------------------------------------------------------- localparam REG_FUNC_CTRL = 8'h84; localparam REG_OTG_CTRL = 8'h8a; localparam REG_TRANSMIT = 8'h40; localparam REG_WRITE = 8'h80; localparam REG_READ = 8'hC0; //----------------------------------------------------------------- // UTMI Mode Select //----------------------------------------------------------------- reg mode_update_q; reg [1:0] xcvrselect_q; reg termselect_q; reg [1:0] opmode_q; reg phy_reset_q; reg auto_wr_q; reg reg_wr_q; always @ (posedge ulpi_clk60_i or posedge ulpi_rst_i) if (ulpi_rst_i) begin mode_update_q <= 1'b0; xcvrselect_q <= 2'b0; termselect_q <= 1'b0; opmode_q <= 2'b11; phy_reset_q <= 1'b1; end else begin xcvrselect_q <= utmi_xcvrselect_i; termselect_q <= utmi_termselect_i; opmode_q <= utmi_opmode_i; if (mode_update_q && (state_q == STATE_CMD) && (ulpi_data_o == REG_FUNC_CTRL)) begin mode_update_q <= 1'b0; phy_reset_q <= 1'b0; end else if (opmode_q != utmi_opmode_i || termselect_q != utmi_termselect_i || xcvrselect_q != utmi_xcvrselect_i) mode_update_q <= 1'b1; end //----------------------------------------------------------------- // UTMI OTG Control //----------------------------------------------------------------- reg otg_update_q; reg dppulldown_q; reg dmpulldown_q; always @ (posedge ulpi_clk60_i or posedge ulpi_rst_i) if (ulpi_rst_i) begin otg_update_q <= 1'b0; dppulldown_q <= 1'b1; dmpulldown_q <= 1'b1; end else begin dppulldown_q <= utmi_dppulldown_i; dmpulldown_q <= utmi_dmpulldown_i; if (otg_update_q && (state_q == STATE_CMD) && (ulpi_data_o == REG_OTG_CTRL)) otg_update_q <= 1'b0; else if (dppulldown_q != utmi_dppulldown_i || dmpulldown_q != utmi_dmpulldown_i) otg_update_q <= 1'b1; end //----------------------------------------------------------------- // Bus turnaround detect //----------------------------------------------------------------- reg ulpi_dir_q; always @ (posedge ulpi_clk60_i or posedge ulpi_rst_i) if (ulpi_rst_i) ulpi_dir_q <= 1'b0; else ulpi_dir_q <= ulpi_dir_i; wire turnaround_w = ulpi_dir_q ^ ulpi_dir_i; reg ulpi_rxcmd_q; always @ (posedge ulpi_clk60_i or posedge ulpi_rst_i) if (ulpi_rst_i) ulpi_rxcmd_q <= 1'b0; // Switch to input with NXT asserted in turnaround cycle else if (!ulpi_dir_q && ulpi_dir_i && ulpi_nxt_i) ulpi_rxcmd_q <= 1'b1; // Switch to output (turnaround cycle) else if (ulpi_dir_q && !ulpi_dir_i) ulpi_rxcmd_q <= 1'b0; //----------------------------------------------------------------- // Register Access //----------------------------------------------------------------- reg reg_wr_pending_q; reg reg_rd_pending_q; reg [7:0] reg_addr_q; reg [7:0] reg_data_q; reg reg_ack_q; wire reg_ready_w = (reg_wr_pending_q && state_q == STATE_REG && ulpi_nxt_i && reg_wr_q) || (reg_rd_pending_q && !turnaround_w && ulpi_dir_i && !ulpi_rxcmd_q); always @ (posedge ulpi_clk60_i or posedge ulpi_rst_i) if (ulpi_rst_i) begin reg_wr_pending_q <= 1'b0; reg_rd_pending_q <= 1'b0; reg_addr_q <= 8'b0; end else if (reg_stb_i) begin reg_addr_q <= reg_addr_i; reg_wr_pending_q <= reg_we_i; reg_rd_pending_q <= ~reg_we_i; end else if (reg_ready_w) begin reg_wr_pending_q <= 1'b0; reg_rd_pending_q <= 1'b0; end always @ (posedge ulpi_clk60_i or posedge ulpi_rst_i) if (ulpi_rst_i) reg_data_q <= 8'b0; else if (reg_stb_i && reg_we_i) reg_data_q <= reg_data_i; assign reg_data_o = utmi_data_o; always @ (posedge ulpi_clk60_i or posedge ulpi_rst_i) if (ulpi_rst_i) reg_ack_q <= 1'b0; else reg_ack_q <= reg_ready_w; assign reg_ack_o = reg_ack_q; //----------------------------------------------------------------- // Rx - Tx delay //----------------------------------------------------------------- localparam TX_DELAY_W = 3; localparam TX_START_DELAY = 3'd7; reg [TX_DELAY_W-1:0] tx_delay_q; always @ (posedge ulpi_clk60_i or posedge ulpi_rst_i) if (ulpi_rst_i) tx_delay_q <= {TX_DELAY_W{1'b0}}; else if (utmi_rxactive_o) tx_delay_q <= TX_START_DELAY; else if (tx_delay_q != {TX_DELAY_W{1'b0}}) tx_delay_q <= tx_delay_q - 1; wire tx_delay_complete_w = (tx_delay_q == {TX_DELAY_W{1'b0}}); //----------------------------------------------------------------- // Tx Buffer - decouple UTMI Tx from PHY I/O //----------------------------------------------------------------- reg [7:0] tx_buffer_q[0:1]; reg tx_valid_q[0:1]; reg tx_wr_idx_q; reg tx_rd_idx_q; wire utmi_tx_ready_w; wire utmi_tx_accept_w; always @ (posedge ulpi_clk60_i or posedge ulpi_rst_i) if (ulpi_rst_i) begin tx_buffer_q[0] <= 8'b0; tx_buffer_q[1] <= 8'b0; tx_valid_q[0] <= 1'b0; tx_valid_q[1] <= 1'b0; tx_wr_idx_q <= 1'b0; tx_rd_idx_q <= 1'b0; end else begin // Push if (utmi_txvalid_i && utmi_txready_o) begin tx_buffer_q[tx_wr_idx_q] <= utmi_data_i; tx_valid_q[tx_wr_idx_q] <= 1'b1; tx_wr_idx_q <= tx_wr_idx_q + 1'b1; end // Pop if (utmi_tx_ready_w && utmi_tx_accept_w) begin tx_valid_q[tx_rd_idx_q] <= 1'b0; tx_rd_idx_q <= tx_rd_idx_q + 1'b1; end end // Tx buffer space (only accept after Rx->Tx turnaround delay) assign utmi_txready_o = ~tx_valid_q[tx_wr_idx_q] & tx_delay_complete_w; assign utmi_tx_ready_w = tx_valid_q[tx_rd_idx_q]; wire [7:0] utmi_tx_data_w = tx_buffer_q[tx_rd_idx_q]; //----------------------------------------------------------------- // Implementation //----------------------------------------------------------------- // Xilinx placement pragmas: //synthesis attribute IOB of ulpi_data_q is "TRUE" //synthesis attribute IOB of ulpi_stp_q is "TRUE" reg [7:0] ulpi_data_q; reg ulpi_stp_q; reg [7:0] data_q; reg utmi_rxvalid_q; reg utmi_rxerror_q; reg utmi_rxactive_q; reg [1:0] utmi_linestate_q; reg [7:0] utmi_data_q; always @ (posedge ulpi_clk60_i or posedge ulpi_rst_i) begin if (ulpi_rst_i) begin state_q <= STATE_IDLE; ulpi_data_q <= 8'b0; data_q <= 8'b0; ulpi_stp_q <= 1'b0; utmi_rxvalid_q <= 1'b0; utmi_rxerror_q <= 1'b0; utmi_rxactive_q <= 1'b0; utmi_linestate_q <= 2'b0; utmi_data_q <= 8'b0; auto_wr_q <= 1'b0; reg_wr_q <= 1'b0; end else begin ulpi_stp_q <= 1'b0; utmi_rxvalid_q <= 1'b0; if (!turnaround_w) begin //----------------------------------------------------------------- // Input: RX_DATA //----------------------------------------------------------------- if (ulpi_dir_i && ulpi_rxcmd_q) begin utmi_rxvalid_q <= ulpi_nxt_i; utmi_data_q <= ulpi_data_i; // No valid data, extract phy status if (!ulpi_nxt_i) begin utmi_linestate_q <= ulpi_data_i[1:0]; case (ulpi_data_i[5:4]) 2'b00: begin utmi_rxactive_q <= 1'b0; utmi_rxerror_q <= 1'b0; end 2'b01: begin utmi_rxactive_q <= 1'b1; utmi_rxerror_q <= 1'b0; end 2'b11: begin utmi_rxactive_q <= 1'b1; utmi_rxerror_q <= 1'b1; end default: ; // HOST_DISCONNECTED endcase end // RxValid (so force RxActive) else utmi_rxactive_q <= 1'b1; end //----------------------------------------------------------------- // Input: REG_DATA //----------------------------------------------------------------- else if (ulpi_dir_i) begin utmi_rxvalid_q <= 1'b0; utmi_data_q <= ulpi_data_i; end //----------------------------------------------------------------- // Output //----------------------------------------------------------------- else begin // IDLE: Pending mode update if ((state_q == STATE_IDLE) && mode_update_q) begin data_q <= {1'b0, 1'b1, phy_reset_q, opmode_q, termselect_q, xcvrselect_q}; ulpi_data_q <= REG_FUNC_CTRL; state_q <= STATE_CMD; auto_wr_q <= 1'b1; reg_wr_q <= 1'b0; end // IDLE: Pending OTG control update else if ((state_q == STATE_IDLE) && otg_update_q) begin data_q <= {5'b0, dmpulldown_q, dppulldown_q, 1'b0}; ulpi_data_q <= REG_OTG_CTRL; state_q <= STATE_CMD; auto_wr_q <= 1'b1; reg_wr_q <= 1'b0; end // IDLE: Pending register access else if ((state_q == STATE_IDLE) && (reg_wr_pending_q || reg_rd_pending_q)) begin data_q <= reg_data_q; if (reg_wr_pending_q) ulpi_data_q <= REG_WRITE | {2'b0, reg_addr_q[5:0]}; else ulpi_data_q <= REG_READ | {2'b0, reg_addr_q[5:0]}; state_q <= STATE_CMD; auto_wr_q <= 1'b0; reg_wr_q <= reg_wr_pending_q; end // IDLE: Pending transmit else if ((state_q == STATE_IDLE) && utmi_tx_ready_w) begin ulpi_data_q <= REG_TRANSMIT | {4'b0, utmi_tx_data_w[3:0]}; state_q <= STATE_DATA; auto_wr_q <= 1'b0; reg_wr_q <= 1'b0; end // Command else if ((state_q == STATE_CMD) && ulpi_nxt_i) begin // Read Register if (!reg_wr_q && !auto_wr_q) begin state_q <= STATE_IDLE; ulpi_data_q <= 8'b0; end // Write Register else begin state_q <= STATE_REG; ulpi_data_q <= data_q; end end // Data (register write) else if (state_q == STATE_REG && ulpi_nxt_i) begin state_q <= STATE_IDLE; ulpi_data_q <= 8'b0; // IDLE ulpi_stp_q <= 1'b1; auto_wr_q <= 1'b0; reg_wr_q <= 1'b0; end // Data else if (state_q == STATE_DATA && ulpi_nxt_i) begin // End of packet if (!utmi_tx_ready_w) begin state_q <= STATE_IDLE; ulpi_data_q <= 8'b0; // IDLE ulpi_stp_q <= 1'b1; end else begin state_q <= STATE_DATA; ulpi_data_q <= utmi_tx_data_w; end end end end end end // Accept from buffer assign utmi_tx_accept_w = ((state_q == STATE_IDLE) && !(mode_update_q || otg_update_q || turnaround_w || reg_wr_pending_q || reg_rd_pending_q) && !ulpi_dir_i) || (state_q == STATE_DATA && ulpi_nxt_i && !ulpi_dir_i); //----------------------------------------------------------------- // Assignments //----------------------------------------------------------------- // ULPI Interface assign ulpi_data_o = ulpi_data_q; assign ulpi_stp_o = ulpi_stp_q; // UTMI Interface assign utmi_linestate_o = utmi_linestate_q; assign utmi_data_o = utmi_data_q; assign utmi_rxerror_o = utmi_rxerror_q; assign utmi_rxactive_o = utmi_rxactive_q; assign utmi_rxvalid_o = utmi_rxvalid_q; endmodule
Go to most recent revision | Compare with Previous | Blame | View Log