1.1 --- /dev/null Thu Jan 01 00:00:00 1970 +0000 1.2 +++ b/rtl/verilog/tpio.v Fri Aug 13 10:41:29 2010 +0100 1.3 @@ -0,0 +1,168 @@ 1.4 +// ============================================================================= 1.5 +// COPYRIGHT NOTICE 1.6 +// Copyright 2006 (c) Lattice Semiconductor Corporation 1.7 +// ALL RIGHTS RESERVED 1.8 +// This confidential and proprietary software may be used only as authorised by 1.9 +// a licensing agreement from Lattice Semiconductor Corporation. 1.10 +// The entire notice above must be reproduced on all authorized copies and 1.11 +// copies may only be made to the extent permitted by a licensing agreement from 1.12 +// Lattice Semiconductor Corporation. 1.13 +// 1.14 +// Lattice Semiconductor Corporation TEL : 1-800-Lattice (USA and Canada) 1.15 +// 5555 NE Moore Court 408-826-6000 (other locations) 1.16 +// Hillsboro, OR 97124 web : http://www.latticesemi.com/ 1.17 +// U.S.A email: techsupport@latticesemi.com 1.18 +// ============================================================================/ 1.19 +// FILE DETAILS 1.20 +// FILE DETAILS 1.21 +// Project : GPIO for LM32 1.22 +// File : tpio.v 1.23 +// Title : Tri State IO control 1.24 +// Dependencies : system_conf.v 1.25 +// Description : Implements the logic to interface tri-state IO with 1.26 +// Wishbone bus. 1.27 +// ============================================================================= 1.28 +// REVISION HISTORY 1.29 +// Version : 7.0 1.30 +// Mod. Date : Jun 27, 2005 1.31 +// Changes Made : Initial Creation 1.32 +// 1.33 +// Version : 7.0SP2, 3.0 1.34 +// Mod. Date : 20 Nov. 2007 1.35 +// Changes Made : Code clean up and add the BB for the inout port. 1.36 +// 1.37 +// Version : 3.1 1.38 +// Mod. Date : 11 Oct. 2008 1.39 +// Changes Made : Update the Edge Capture Register clean method 1.40 +// Make IRQ Mask register readable 1.41 +// ============================================================================= 1.42 +`ifndef TPIO_V 1.43 +`define TPIO_V 1.44 +`timescale 1ns/100 ps 1.45 +`include "system_conf.v" 1.46 +module TRI_PIO #(parameter DATA_WIDTH = 16, 1.47 + parameter IRQ_MODE = 1, 1.48 + parameter LEVEL = 0, 1.49 + parameter EDGE = 1, 1.50 + parameter POSE_EDGE_IRQ = 1, 1.51 + parameter NEGE_EDGE_IRQ = 0, 1.52 + parameter EITHER_EDGE_IRQ = 0) 1.53 + (RST_I, 1.54 + CLK_I, 1.55 + DAT_I, 1.56 + DAT_O, 1.57 + PIO_IO, 1.58 + IRQ_O, 1.59 + PIO_TRI_WR_EN, 1.60 + PIO_TRI_RE_EN, 1.61 + PIO_DATA_RE_EN, 1.62 + PIO_DATA_WR_EN, 1.63 + IRQ_MASK_RE_EN, 1.64 + IRQ_MASK_WR_EN, 1.65 + EDGE_CAP_WR_EN); 1.66 + 1.67 + parameter UDLY = 1;//user delay 1.68 + 1.69 + input RST_I; 1.70 + input CLK_I; 1.71 + input DAT_I; 1.72 + input PIO_TRI_RE_EN; 1.73 + input PIO_TRI_WR_EN; 1.74 + input PIO_DATA_RE_EN; 1.75 + input PIO_DATA_WR_EN; 1.76 + output DAT_O; 1.77 + input IRQ_MASK_RE_EN; 1.78 + input IRQ_MASK_WR_EN; 1.79 + input EDGE_CAP_WR_EN; 1.80 + output IRQ_O; 1.81 + inout PIO_IO; 1.82 + 1.83 + wire PIO_IO_I; 1.84 + wire DAT_O; 1.85 + wire IRQ_O; 1.86 + reg PIO_DATA_O; 1.87 + reg PIO_DATA_I; 1.88 + reg PIO_TRI; 1.89 + reg IRQ_MASK; 1.90 + reg IRQ_TEMP; 1.91 + reg EDGE_CAPTURE; 1.92 + reg PIO_DATA_DLY; 1.93 + 1.94 + always @(posedge CLK_I or posedge RST_I) 1.95 + if (RST_I) 1.96 + PIO_TRI <= #UDLY 0; 1.97 + else if (PIO_TRI_WR_EN) 1.98 + PIO_TRI <= #UDLY DAT_I; 1.99 + 1.100 + always @(posedge CLK_I or posedge RST_I) 1.101 + if (RST_I) 1.102 + PIO_DATA_O <= #UDLY 0; 1.103 + else if (PIO_DATA_WR_EN) 1.104 + PIO_DATA_O <= #UDLY DAT_I; 1.105 + 1.106 + always @(posedge CLK_I or posedge RST_I) 1.107 + if (RST_I) 1.108 + PIO_DATA_I <= #UDLY 0; 1.109 + else if (PIO_DATA_RE_EN) 1.110 + PIO_DATA_I <= #UDLY PIO_IO_I; 1.111 + 1.112 + BB tpio_inst(.I(PIO_DATA_O), .T(~PIO_TRI), .O(PIO_IO_I), .B(PIO_IO)); 1.113 + assign DAT_O = PIO_TRI_RE_EN ? PIO_TRI : 1.114 + IRQ_MASK_RE_EN ? IRQ_MASK : PIO_DATA_I; 1.115 + 1.116 + //IRQ_MODE 1.117 + 1.118 + generate 1.119 + if (IRQ_MODE == 1) begin 1.120 + //CONFIG THE IRQ_MASK REG. 1.121 + always @(posedge CLK_I or posedge RST_I) 1.122 + if (RST_I) 1.123 + IRQ_MASK <= #UDLY 0; 1.124 + else if (IRQ_MASK_WR_EN) 1.125 + IRQ_MASK <= #UDLY DAT_I; 1.126 + end 1.127 + endgenerate 1.128 + 1.129 + generate 1.130 + if (IRQ_MODE == 1 && LEVEL == 1) begin 1.131 + always @(posedge CLK_I or posedge RST_I) 1.132 + if (RST_I) 1.133 + IRQ_TEMP <= #UDLY 0; 1.134 + else 1.135 + IRQ_TEMP <= #UDLY PIO_IO_I & IRQ_MASK & ~PIO_TRI;//bit-and 1.136 + assign IRQ_O = IRQ_TEMP; 1.137 + end 1.138 + else if (IRQ_MODE == 1 && EDGE == 1) begin 1.139 + always @(posedge CLK_I or posedge RST_I) 1.140 + if (RST_I) 1.141 + PIO_DATA_DLY <= #UDLY 0; 1.142 + else 1.143 + PIO_DATA_DLY <= PIO_IO_I; 1.144 + 1.145 + always @(posedge CLK_I or posedge RST_I) 1.146 + if (RST_I) 1.147 + EDGE_CAPTURE <= #UDLY 0; 1.148 + else if ((PIO_IO_I & ~PIO_DATA_DLY & ~PIO_TRI) && POSE_EDGE_IRQ == 1) 1.149 + EDGE_CAPTURE <= #UDLY PIO_IO_I & ~PIO_DATA_DLY; 1.150 + else if ((~PIO_IO_I & PIO_DATA_DLY & ~PIO_TRI) && NEGE_EDGE_IRQ == 1) 1.151 + EDGE_CAPTURE <= #UDLY ~PIO_IO_I & PIO_DATA_DLY; 1.152 + else if ((PIO_IO_I & ~PIO_DATA_DLY & ~PIO_TRI) && EITHER_EDGE_IRQ == 1) 1.153 + EDGE_CAPTURE <= #UDLY PIO_IO_I & ~PIO_DATA_DLY; 1.154 + else if ((~PIO_IO_I & PIO_DATA_DLY & ~PIO_TRI) && EITHER_EDGE_IRQ == 1) 1.155 + EDGE_CAPTURE <= #UDLY ~PIO_IO_I & PIO_DATA_DLY; 1.156 + else if ( (~IRQ_MASK) & DAT_I & IRQ_MASK_WR_EN ) 1.157 + // interrupt mask's being set, so clear edge-capture 1.158 + EDGE_CAPTURE <= #UDLY 0; 1.159 + else if ( EDGE_CAP_WR_EN ) 1.160 + // user's writing to the edge-register, so update edge-capture 1.161 + // register 1.162 + EDGE_CAPTURE <= #UDLY EDGE_CAPTURE & DAT_I; 1.163 + 1.164 + assign IRQ_O = |(EDGE_CAPTURE & IRQ_MASK); 1.165 + end 1.166 + else // IRQ_MODE ==0 1.167 + assign IRQ_O = 0; 1.168 + endgenerate 1.169 +endmodule 1.170 +`endif // TPIO_V 1.171 +