(原創) 如何將CMOS影像以RGB16的方式存入SDRAM? (SOC) (DE2-70) (TRDB-D5M) (TRDB-LTM)

Abstract
在數位視訊系統中,SDRAM扮演CMOS與LCD之間frame buffer的腳色,但若SDRAM另用他用,只好將RGB以16 bit方式存入SDRAM,將節省下來的空間做其他用途。

Introduction
使用環境:Quartus II 8.0 SP1 + DE2-70 (Cyclone II EP2C70F896C6N) + TRDB-D5M + TRDB-LTM

以RGB16(R5G6B5)的方式存入SDRAM的方式並不是我所創,在友晶提供的DE2_CCD_detect範例中,因為要做motion detection,要用前後兩個frame的資料做比對,所以整個SDRAM不能全部做成CMOS與VGA的frame buffer,必須有一塊區域存前一個frame的資料,而DE2又只有一顆8MB SDRAM,所以2個write port與2個read port不能全部用盡,CMOS與VGA只能占用1個write port與read port,而SDRAM的寬度為16 bit,所以只能以RGB16存進SDRAM。

或許你會有疑問,以RGB16來處理,不會失真嗎?理論上是會失真,不過由於是取高5 bit處理,所以還可接受,以下是用RGB16與RGB32結果的比較

左側為RGB16、右側為RGB32

de2-70_ltm_rgb16 de2-70_ltm_rgb32

兩張圖是有一點點差距,不過大體上來說還可以接受,若你用的是DE2,因為只有一顆8MB SDRAM,就只能靠RGB16的方式,才能讓SDRAM還能放其他的資料,而不讓2個write port與2個read port全被RGB32占滿,當然若你用的是DE2-70,因為有兩顆SDRAM,就沒這種問題。

網友就剛好遇到需要用RGB16的狀況,不過它的環境是DE2 + TRDB-D5M + TRDB-LTM,但因為我用的是DE2-70的三合一套件,也就是DE2-70 + TRDB-D5M + TRDB-LTM,所以我只能用DE2-70模擬DE2只有一顆SDRAM的狀況,然後使用RGB16的方式存入SDRAM。

Top Module部分
DE2_70.v / Verilog

  1 /* 
  2 (C) OOMusou 2008 http://oomusou.cnblogs.com
  3 
  4 Filename    : DE2_70.v
  5 Compiler    : Quartus II 8.0 SP1
  6 Description : Demo how to use TRDB-D5M with LTM on DE2-70 with RGB16
  7 Release     : 10/24/2008 1.0
  8 */
  9 
10 module DE2_70 (
11   //////////////////////// Clock Input ////////////////////////
12   input          iCLK_28,           // 28.63636 MHz
13   input          iCLK_50,           // 50 MHz
14   input          iCLK_50_2,         // 50 MHz
15   input          iCLK_50_3,         // 50 MHz
16   input          iCLK_50_4,         // 50 MHz
17   input          iEXT_CLOCK,        // External Clock
18   //////////////////////// Push Button ////////////////////////
19   input  [3:0]   iKEY,              // Pushbutton[3:0]
20   //////////////////////// DPDT Switch ////////////////////////
21   input  [17:0]  iSW,               // Toggle Switch[17:0]
22   //////////////////////// 7-SEG Dispaly ////////////////////////
23   output [6:0]   oHEX0_D,           // Seven Segment Digit 0
24   output         oHEX0_DP,          // Seven Segment Digit 0 decimal point
25   output [6:0]   oHEX1_D,           // Seven Segment Digit 1
26   output         oHEX1_DP,          // Seven Segment Digit 1 decimal point
27   output [6:0]   oHEX2_D,           // Seven Segment Digit 2
28   output         oHEX2_DP,          // Seven Segment Digit 2 decimal point
29   output [6:0]   oHEX3_D,           // Seven Segment Digit 3
30   output         oHEX3_DP,          // Seven Segment Digit 3 decimal point
31   output [6:0]   oHEX4_D,           // Seven Segment Digit 4
32   output         oHEX4_DP,          // Seven Segment Digit 4 decimal point
33   output [6:0]   oHEX5_D,           // Seven Segment Digit 5
34   output         oHEX5_DP,          // Seven Segment Digit 5 decimal point
35   output [6:0]   oHEX6_D,           // Seven Segment Digit 6
36   output         oHEX6_DP,          // Seven Segment Digit 6 decimal point
37   output [6:0]   oHEX7_D,           // Seven Segment Digit 7
38   output         oHEX7_DP,          // Seven Segment Digit 7 decimal point
39   //////////////////////////// LED ////////////////////////////
40   output [8:0]   oLEDG,             // LED Green[8:0]
41   output [17:0]  oLEDR,             // LED Red[17:0]
42   //////////////////////////// UART ////////////////////////////
43   output         oUART_TXD,         // UART Transmitter
44   input          iUART_RXD,         // UART Receiver
45   output         oUART_CTS,         // UART Clear To Send
46   input          iUART_RTS,         // UART Requst To Send
47   //////////////////////////// IRDA ////////////////////////////
48   output         oIRDA_TXD,         // IRDA Transmitter
49   input          iIRDA_RXD,         // IRDA Receiver
50   /////////////////////// SDRAM Interface ////////////////////////
51   inout   [31:0] DRAM_DQ,           // SDRAM Data bus 32 Bits
52   output  [12:0] oDRAM0_A,          // SDRAM0 Address bus 13 Bits
53   output  [12:0] oDRAM1_A,          // SDRAM1 Address bus 13 Bits
54   output         oDRAM0_LDQM0,      // SDRAM0 Low-byte Data Mask
55   output         oDRAM1_LDQM0,      // SDRAM1 Low-byte Data Mask
56   output         oDRAM0_UDQM1,      // SDRAM0 High-byte Data Mask
57   output         oDRAM1_UDQM1,      // SDRAM1 High-byte Data Mask
58   output         oDRAM0_WE_N,       // SDRAM0 Write Enable
59   output         oDRAM1_WE_N,       // SDRAM1 Write Enable
60   output         oDRAM0_CAS_N,      // SDRAM0 Column Address Strobe
61   output         oDRAM1_CAS_N,      // SDRAM1 Column Address Strobe
62   output         oDRAM0_RAS_N,      // SDRAM0 Row Address Strobe
63   output         oDRAM1_RAS_N,      // SDRAM1 Row Address Strobe
64   output         oDRAM0_CS_N,       // SDRAM0 Chip Select
65   output         oDRAM1_CS_N,       // SDRAM1 Chip Select
66   output  [1:0]  oDRAM0_BA,         // SDRAM0 Bank Address
67   output  [1:0]  oDRAM1_BA,         // SDRAM1 Bank Address
68   output         oDRAM0_CLK,        // SDRAM0 Clock
69   output         oDRAM1_CLK,        // SDRAM1 Clock
70   output         oDRAM0_CKE,        // SDRAM0 Clock Enable
71   output         oDRAM1_CKE,        // SDRAM1 Clock Enable
72   //////////////////////// Flash Interface ////////////////////////
73   inout   [14:0] FLASH_DQ,          // FLASH Data bus 15 Bits (0 to 14)
74   inout          FLASH_DQ15_AM1,    // FLASH Data bus Bit 15 or Address A-1
75   output  [21:0] oFLASH_A,          // FLASH Address bus 26 Bits
76   output         oFLASH_WE_N,       // FLASH Write Enable
77   output         oFLASH_RST_N,      // FLASH Reset
78   output         oFLASH_WP_N,       // FLASH Write Protect /Programming Acceleration
79   input          iFLASH_RY_N,       // FLASH Ready/Busy output
80   output         oFLASH_BYTE_N,     // FLASH Byte/Word Mode Configuration
81   output         oFLASH_OE_N,       // FLASH Output Enable
82   output         oFLASH_CE_N,       // FLASH Chip Enable
83   //////////////////////// SRAM Interface ////////////////////////
84   inout   [31:0] SRAM_DQ,           // SRAM Data Bus 32 Bits
85   inout   [3:0]  SRAM_DPA,          // SRAM Parity Data Bus
86   output  [18:0] oSRAM_A,           // SRAM Address bus 21 Bits
87   output         oSRAM_ADSC_N,      // SRAM Controller Address Status    
88   output         oSRAM_ADSP_N,      // SRAM Processor Address Status
89   output         oSRAM_ADV_N,       // SRAM Burst Address Advance
90   output  [3:0]  oSRAM_BE_N,        // SRAM Byte Write Enable
91   output         oSRAM_CE1_N,       // SRAM Chip Enable
92   output         oSRAM_CE2,         // SRAM Chip Enable
93   output         oSRAM_CE3_N,       // SRAM Chip Enable
94   output         oSRAM_CLK,         // SRAM Clock
95   output         oSRAM_GW_N,        // SRAM Global Write Enable
96   output         oSRAM_OE_N,        // SRAM Output Enable
97   output         oSRAM_WE_N,        // SRAM Write Enable
98   //////////////////// ISP1362 Interface ////////////////////////
99   inout   [15:0] OTG_D,             // ISP1362 Data bus 16 Bits
100   output  [1:0]  oOTG_A,            // ISP1362 Address 2 Bits
101   output         oOTG_CS_N,         // ISP1362 Chip Select
102   output         oOTG_OE_N,         // ISP1362 Read
103   output         oOTG_WE_N,         // ISP1362 Write
104   output         oOTG_RESET_N,      // ISP1362 Reset
105   inout          OTG_FSPEED,        // USB Full Speed,    0 = Enable, Z = Disable
106   inout          OTG_LSPEED,        // USB Low Speed,     0 = Enable, Z = Disable
107   input          iOTG_INT0,         // ISP1362 Interrupt 0
108   input          iOTG_INT1,         // ISP1362 Interrupt 1
109   input          iOTG_DREQ0,        // ISP1362 DMA Request 0
110   input          iOTG_DREQ1,        // ISP1362 DMA Request 1
111   output         oOTG_DACK0_N,      // ISP1362 DMA Acknowledge 0
112   output         oOTG_DACK1_N,      // ISP1362 DMA Acknowledge 1
113   //////////////////// LCD Module 16X2 ////////////////////////////
114   inout   [7:0]  LCD_D,             // LCD Data bus 8 bits
115   output         oLCD_ON,           // LCD Power ON/OFF
116   output         oLCD_BLON,         // LCD Back Light ON/OFF
117   output         oLCD_RW,           // LCD Read/Write Select, 0 = Write, 1 = Read
118   output         oLCD_EN,           // LCD Enable
119   output         oLCD_RS,           // LCD Command/Data Select, 0 = Command, 1 = Data
120   //////////////////// SD Card Interface ////////////////////////
121   inout          SD_DAT,            // SD Card Data
122   inout          SD_DAT3,           // SD Card Data 3
123   inout          SD_CMD,            // SD Card Command Signal
124   output         oSD_CLK,           // SD Card Clock
125   //////////////////////// I2C ////////////////////////////////
126   inout          I2C_SDAT,          // I2C Data
127   output         oI2C_SCLK,         // I2C Clock
128   //////////////////////// PS2 ////////////////////////////////
129   inout          PS2_KBDAT,         // PS2 Keyboard Data
130   inout          PS2_KBCLK,         // PS2 Keyboard Clock
131   inout          PS2_MSDAT,         // PS2 Mouse Data
132   inout          PS2_MSCLK,         // PS2 Mouse Clock
133   //////////////////////// VGA ////////////////////////////
134   output         oVGA_CLOCK,        // VGA Clock
135   output         oVGA_HS,           // VGA H_SYNC
136   output         oVGA_VS,           // VGA V_SYNC
137   output         oVGA_BLANK_N,      // VGA BLANK
138   output         oVGA_SYNC_N,       // VGA SYNC
139   output  [9:0]  oVGA_R,            // VGA Red[9:0]
140   output  [9:0]  oVGA_G,            // VGA Green[9:0]
141   output  [9:0]  oVGA_B,            // VGA Blue[9:0]
142   //////////////// Ethernet Interface ////////////////////////////
143   inout   [15:0] ENET_D,            // DM9000A DATA bus 16Bits
144   output         oENET_CMD,         // DM9000A Command/Data Select, 0 = Command, 1 = Data
145   output         oENET_CS_N,        // DM9000A Chip Select
146   output         oENET_IOW_N,       // DM9000A Write
147   output         oENET_IOR_N,       // DM9000A Read
148   output         oENET_RESET_N,     // DM9000A Reset
149   input          iENET_INT,         // DM9000A Interrupt
150   output         oENET_CLK,         // DM9000A Clock 25 MHz
151   //////////////////// Audio CODEC  ////////////////////////////
152   inout          AUD_ADCLRCK,       // Audio CODEC ADC LR Clock
153   input          iAUD_ADCDAT,       // Audio CODEC ADC Data
154   inout          AUD_DACLRCK,       // Audio CODEC DAC LR Clock
155   output         oAUD_DACDAT,       // Audio CODEC DAC Data
156   inout          AUD_BCLK,          // Audio CODEC Bit-Stream Clock
157   output         oAUD_XCK,          // Audio CODEC Chip Clock
158   //////////////////// TV Devoder   ////////////////////////////
159   input          iTD1_CLK27,        // TV Decoder1 Line_Lock Output Clock
160   input   [7:0]  iTD1_D,            // TV Decoder1 Data bus 8 bits
161   input          iTD1_HS,           // TV Decoder1 H_SYNC
162   input          iTD1_VS,           // TV Decoder1 V_SYNC
163   output         oTD1_RESET_N,      // TV Decoder1 Reset
164   input          iTD2_CLK27,        // TV Decoder2 Line_Lock Output Clock        
165   input   [7:0]  iTD2_D,            // TV Decoder2 Data bus 8 bits
166   input          iTD2_HS,           // TV Decoder2 H_SYNC
167   input          iTD2_VS,           // TV Decoder2 V_SYNC
168   output         oTD2_RESET_N,      // TV Decoder2 Reset
169   //////////////////////// GPIO ////////////////////////////////
170   inout   [31:0] GPIO_0,            // GPIO Connection 0 I/O
171   input          GPIO_CLKIN_N0,     // GPIO Connection 0 Clock Input 0
172   input          GPIO_CLKIN_P0,     // GPIO Connection 0 Clock Input 1
173   inout          GPIO_CLKOUT_N0,    // GPIO Connection 0 Clock Output 0
174   inout          GPIO_CLKOUT_P0,    // GPIO Connection 0 Clock Output 1
175   inout   [31:0] GPIO_1,            // GPIO Connection 1 I/O
176   input          GPIO_CLKIN_N1,     // GPIO Connection 1 Clock Input 0
177   input          GPIO_CLKIN_P1,     // GPIO Connection 1 Clock Input 1
178   inout          GPIO_CLKOUT_N1,    // GPIO Connection 1 Clock Output 0
179   inout          GPIO_CLKOUT_P1     // GPIO Connection 1 Clock Output 1
180 );
181 
182 //  CCD
183 wire  [11:0]  CCD_DATA;
184 wire          CCD_SDAT;
185 wire          CCD_SCLK;
186 wire          CCD_FLASH;
187 wire          CCD_FVAL;
188 wire          CCD_LVAL;
189 wire          CCD_PIXCLK;
190 wire          CCD_MCLK; //  CCD Master Clock
191 
192 wire  [15:0]  Read_DATA1;
193 wire  [15:0]  Read_DATA2;
194 wire  [15:0]  Read_DATA3; // add by oomusou for RGB16
195 wire  [15:0]  Read_DATA4; // add by oomusou for RGB16
196 wire          VGA_CTRL_CLK;
197 wire  [11:0]  mCCD_DATA;
198 wire          mCCD_DVAL;
199 wire          mCCD_DVAL_d;
200 wire  [15:0]  X_Cont;
201 wire  [15:0]  Y_Cont;
202 wire  [9:0]   X_ADDR;
203 wire  [31:0]  Frame_Cont;
204 wire          DLY_RST_0;
205 wire          DLY_RST_1;
206 wire          DLY_RST_2;
207 wire          Read;
208 reg   [11:0]  rCCD_DATA;
209 reg           rCCD_LVAL;
210 reg           rCCD_FVAL;
211 wire  [11:0]  sCCD_R;
212 wire  [11:0]  sCCD_G;
213 wire  [11:0]  sCCD_B;
214 wire          sCCD_DVAL;
215 reg   [1:0]   rClk;
216 wire          sdram_ctrl_clk;
217 
218 // Touch panel signal
219 wire  [7:0] ltm_r;    //  LTM Red Data 8 Bits
220 wire  [7:0] ltm_g;    //  LTM Green Data 8 Bits
221 wire  [7:0] ltm_b;    //  LTM Blue Data 8 Bits
222 wire        ltm_nclk; //  LTM Clcok
223 wire        ltm_hd;
224 wire        ltm_vd;
225 wire        ltm_den;
226 wire        adc_dclk;
227 wire        adc_cs;
228 wire        adc_penirq_n;
229 wire        adc_busy;
230 wire        adc_din;
231 wire        adc_dout;
232 wire        adc_ltm_sclk;
233 wire        ltm_grst;
234 
235 // LTM Config
236 wire      ltm_sclk;
237 wire      ltm_sda;
238 wire      ltm_scen;
239 wire      ltm_3wirebusy_n;
240 
241 assign  CCD_DATA[0]     = GPIO_1[11];
242 assign  CCD_DATA[1]     = GPIO_1[10];
243 assign  CCD_DATA[2]     = GPIO_1[9];
244 assign  CCD_DATA[3]     = GPIO_1[8];
245 assign  CCD_DATA[4]     = GPIO_1[7];
246 assign  CCD_DATA[5]     = GPIO_1[6];
247 assign  CCD_DATA[6]     = GPIO_1[5];
248 assign  CCD_DATA[7]     = GPIO_1[4];
249 assign  CCD_DATA[8]     = GPIO_1[3];
250 assign  CCD_DATA[9]     = GPIO_1[2];
251 assign  CCD_DATA[10]    = GPIO_1[1];
252 assign  CCD_DATA[11]    = GPIO_1[0];
253 assign  GPIO_CLKOUT_N1  = CCD_MCLK;
254 assign  CCD_FVAL        = GPIO_1[18];
255 assign  CCD_LVAL        = GPIO_1[17];
256 assign  CCD_PIXCLK      = GPIO_CLKIN_N1;
257 assign  GPIO_1[15]      = 1'b1;  // tRIGGER
258 assign  GPIO_1[14]      = DLY_RST_1;
259 
260 assign  oLEDR           = iSW;
261 assign  oLEDG           = Y_Cont;
262 
263 assign  oTD1_RESET_N    = 1'b1;
264 assign  oVGA_CLOCK      = ~VGA_CTRL_CLK;
265 
266 assign  CCD_MCLK        = rClk[0];
267 assign  oUART_TXD       = iUART_RXD;
268 
269 assign  adc_penirq_n    = GPIO_CLKIN_N0;
270 assign  adc_dout        = GPIO_0[0];
271 assign  adc_busy        = GPIO_CLKIN_P0;
272 assign  GPIO_0[1]       = adc_din;
273 assign  GPIO_0[2]       = adc_ltm_sclk;
274 assign  GPIO_0[3]       = ltm_b[3];
275 assign  GPIO_0[4]       = ltm_b[2];
276 assign  GPIO_0[5]       = ltm_b[1];
277 assign  GPIO_0[6]       = ltm_b[0];
278 assign  GPIO_0[7]       =~ltm_nclk;
279 assign  GPIO_0[8]       = ltm_den;
280 assign  GPIO_0[9]       = ltm_hd;
281 assign  GPIO_0[10]      = ltm_vd;
282 assign  GPIO_0[11]      = ltm_b[4];
283 assign  GPIO_0[12]      = ltm_b[5];
284 assign  GPIO_0[13]      = ltm_b[6];
285 assign  GPIO_CLKOUT_N0  = ltm_b[7];
286 assign  GPIO_0[14]      = ltm_g[0];
287 assign  GPIO_CLKOUT_P0  = ltm_g[1];
288 assign  GPIO_0[15]      = ltm_g[2];
289 assign  GPIO_0[16]      = ltm_g[3];
290 assign  GPIO_0[17]      = ltm_g[4];
291 assign  GPIO_0[18]      = ltm_g[5];
292 assign  GPIO_0[19]      = ltm_g[6];
293 assign  GPIO_0[20]      = ltm_g[7];
294 assign  GPIO_0[21]      = ltm_r[0];
295 assign  GPIO_0[22]      = ltm_r[1];
296 assign  GPIO_0[23]      = ltm_r[2];
297 assign  GPIO_0[24]      = ltm_r[3];
298 assign  GPIO_0[25]      = ltm_r[4];
299 assign  GPIO_0[26]      = ltm_r[5];
300 assign  GPIO_0[27]      = ltm_r[6];
301 assign  GPIO_0[28]      = ltm_r[7];
302 assign  GPIO_0[29]      = ltm_grst;
303 assign  GPIO_0[30]      = ltm_scen;
304 assign  GPIO_0[31]      = ltm_sda;
305 
306 assign ltm_grst         = iKEY[0];
307 assign adc_ltm_sclk     = ltm_sclk;
308 
309 always@(posedge iCLK_50)
310   rClk    <=  rClk  + 1;
311 
312 always@(posedge CCD_PIXCLK) begin
313   rCCD_DATA <=  CCD_DATA;
314   rCCD_LVAL <=  CCD_LVAL;
315   rCCD_FVAL <=  CCD_FVAL;
316 end
317 
318 Reset_Delay reset0  (
319   .iCLK(iCLK_50),
320   .iRST(iKEY[0]),
321   .oRST_0(DLY_RST_0),
322   .oRST_1(DLY_RST_1),
323   .oRST_2(DLY_RST_2)
324 );
325 
326 CCD_Capture capture0 (
327   .oDATA(mCCD_DATA),
328   .oDVAL(mCCD_DVAL),
329   .oX_Cont(X_Cont),
330   .oY_Cont(Y_Cont),
331   .oFrame_Cont(Frame_Cont),
332   .iDATA(rCCD_DATA),
333   .iFVAL(rCCD_FVAL),
334   .iLVAL(rCCD_LVAL),
335   .iSTART(!iKEY[3]),
336   .iEND(!iKEY[2]),
337   .iCLK(CCD_PIXCLK),
338   .iRST(DLY_RST_2)
339 );
340 
341 RAW2RGB rgb0 (
342   .iCLK(CCD_PIXCLK),
343   .iRST_n(DLY_RST_1),
344   .iData(mCCD_DATA),
345   .iDval(mCCD_DVAL),
346   .oRed(sCCD_R),
347   .oGreen(sCCD_G),
348   .oBlue(sCCD_B),
349   .oDval(sCCD_DVAL),
350   .iMIRROR(iSW[17]),
351   .iX_Cont(X_Cont),
352   .iY_Cont(Y_Cont)
353 );
354 
355 SEG7_LUT_8  seg0 (
356   .oSEG0(oHEX0_D),
357   .oSEG1(oHEX1_D),
358   .oSEG2(oHEX2_D),
359   .oSEG3(oHEX3_D),
360   .oSEG4(oHEX4_D),
361   .oSEG5(oHEX5_D),
362   .oSEG6(oHEX6_D),
363   .oSEG7(oHEX7_D),
364   .iDIG(Frame_Cont[31:0])
365 );
366 
367 vga_pll vga_pll0 (
368   .inclk0(iCLK_50_2),
369   .c0(ltm_nclk)
370 );
371 
372 sdram_pll sdram_pll0 (
373   .inclk0(iCLK_50_3),
374   .c0(sdram_ctrl_clk),
375   .c1(oDRAM0_CLK),
376   .c2(oDRAM1_CLK)
377 );
378 
379 Sdram_Control_4Port sdram0 (
380   //  HOST Side
381   .REF_CLK(iCLK_50),
382   .RESET_N(1'b1),
383   .CLK(sdram_ctrl_clk),
384   //  FIFO Write Side 1
385   .WR1_DATA({sCCD_R[11:7], sCCD_G[11:6],  sCCD_B[11:7]}),
386   .WR1(sCCD_DVAL),
387   .WR1_ADDR(0),
388   .WR1_MAX_ADDR(800*480),
389   .WR1_LENGTH(9'h100),
390   .WR1_LOAD(!DLY_RST_0),
391   .WR1_CLK(CCD_PIXCLK),
392   //  FIFO Write Side 1
393   .WR2_DATA({sCCD_R[11:7], sCCD_G[11:6],  sCCD_B[11:7]}),
394   .WR2(sCCD_DVAL),
395   .WR2_ADDR(22'h100000),
396   .WR2_MAX_ADDR(22'h100000 + 800*480),
397   .WR2_LENGTH(9'h100),
398   .WR2_LOAD(!DLY_RST_0),
399   .WR2_CLK(CCD_PIXCLK),
400   //  FIFO Read Side 1 to LTM
401   .RD1_DATA(Read_DATA3),
402   .RD1(Read),
403   .RD1_ADDR(0),
404   .RD1_MAX_ADDR(800*480),
405   .RD1_LENGTH(9'h100),
406   .RD1_LOAD(!DLY_RST_0),
407   .RD1_CLK(~ltm_nclk),
408   //  FIFO Read Side 1
409   .RD2_DATA(Read_DATA4),
410   .RD2(Read),
411   .RD2_ADDR(22'h100000 + 800*480 + 200),
412   .RD2_MAX_ADDR(22'h100000 + 800*480 + 300),
413   .RD2_LENGTH(9'h100),
414   .RD2_LOAD(!DLY_RST_0),
415   .RD2_CLK(~ltm_nclk),
416   //  SDRAM Side
417   .SA(oDRAM0_A[11:0]),
418   .BA(oDRAM0_BA),
419   .CS_N(oDRAM0_CS_N),
420   .CKE(oDRAM0_CKE),
421   .RAS_N(oDRAM0_RAS_N),
422   .CAS_N(oDRAM0_CAS_N),
423   .WE_N(oDRAM0_WE_N),
424   .DQ(DRAM_DQ[15:0]),
425   .DQM({oDRAM0_UDQM1,oDRAM0_LDQM0})
426 );
427 
428 // add by oomusou for RGB16
429 assign Read_DATA1 = {Read_DATA3[10:6], Read_DATA3[4:0], 5'h00};
430 assign Read_DATA2 = {Read_DATA3[5:5], 4'h0, Read_DATA3[15:11], 5'h00};
431 
432 I2C_CCD_Config ccd_config0 (
433   //  Host Side
434   .iCLK(iCLK_50),
435   .iRST_N(DLY_RST_1),
436   .iEXPOSURE_ADJ(iKEY[1]),
437   .iEXPOSURE_DEC_p(iSW[0]),
438   .iMIRROR_SW(iSW[17]),
439   //  I2C Side
440   .I2C_SCLK(GPIO_1[20]),
441   .I2C_SDAT(GPIO_1[19])
442 );
443 
444 touch_tcon tcon0 (
445   .iCLK(ltm_nclk),
446   .iRST_n(DLY_RST_2),
447   // sdram side
448   .iREAD_DATA1(Read_DATA1),
449   .iREAD_DATA2(Read_DATA2),
450   .oREAD_SDRAM_EN(Read),
451   // lcd side
452   .oLCD_R(ltm_r),
453   .oLCD_G(ltm_g),
454   .oLCD_B(ltm_b),
455   .oHD(ltm_hd),
456   .oVD(ltm_vd),
457   .oDEN(ltm_den)
458 );
459 
460 lcd_3wire_config wire0 (
461   // Host Side
462   .iCLK(iCLK_50),
463   .iRST_n(DLY_RST_0),
464   // 3 wire Side
465   .o3WIRE_SCLK(ltm_sclk),
466   .io3WIRE_SDAT(ltm_sda),
467   .o3WIRE_SCEN(ltm_scen),
468   .o3WIRE_BUSY_n(ltm_3wirebusy_n)
469 );
470 
471 endmodule


379行

Sdram_Control_4Port sdram0 (
 
//  HOST Side
  .REF_CLK(iCLK_50),
  .RESET_N(
1'b1),
  .CLK(sdram_ctrl_clk),
 
//  FIFO Write Side 1
  .WR1_DATA({sCCD_R[11:7], sCCD_G[11:6],  sCCD_B[11:7]}),
  .WR1(sCCD_DVAL),
  .WR1_ADDR(
0),
  .WR1_MAX_ADDR(
800*480),
  .WR1_LENGTH(
9'h100),
  .WR1_LOAD(!DLY_RST_0),
  .WR1_CLK(CCD_PIXCLK),
 
//  FIFO Write Side 1
  .WR2_DATA({sCCD_R[11:7], sCCD_G[11:6],  sCCD_B[11:7]}),
  .WR2(sCCD_DVAL),
  .WR2_ADDR(
22'h100000),
  .WR2_MAX_ADDR(22'h100000 + 800*480),
  .WR2_LENGTH(9'h100),
  .WR2_LOAD(!DLY_RST_0),
  .WR2_CLK(CCD_PIXCLK),
 
//  FIFO Read Side 1 to LTM
  .RD1_DATA(Read_DATA3),
  .RD1(Read),
  .RD1_ADDR(
0),
  .RD1_MAX_ADDR(
800*480),
  .RD1_LENGTH(
9'h100),
  .RD1_LOAD(!DLY_RST_0),
  .RD1_CLK(
~ltm_nclk),
 
//  FIFO Read Side 1
  .RD2_DATA(Read_DATA4),
  .RD2(Read),
  .RD2_ADDR(
22'h100000 + 800*480 + 200),
  .RD2_MAX_ADDR(22'h100000 + 800*480 + 300),
  .RD2_LENGTH(9'h100),
  .RD2_LOAD(!DLY_RST_0),
  .RD2_CLK(
~ltm_nclk),
 
//  SDRAM Side
  .SA(oDRAM0_A[11:0]),
  .BA(oDRAM0_BA),
  .CS_N(oDRAM0_CS_N),
  .CKE(oDRAM0_CKE),
  .RAS_N(oDRAM0_RAS_N),
  .CAS_N(oDRAM0_CAS_N),
  .WE_N(oDRAM0_WE_N),
  .DQ(DRAM_DQ[
15:0]),
  .DQM({oDRAM0_UDQM1,oDRAM0_LDQM0})
);


原來在DE2_70_D5M_LTM中,是使用2顆SDRAM來做frame buffer,為了模擬DE2的1顆SDRAM,所以我只用了一個Sdram_Control_4Port。

384行

//  FIFO Write Side 1
.WR1_DATA({sCCD_R[11:7], sCCD_G[11:6],  sCCD_B[11:7]}),
.WR1(sCCD_DVAL),
.WR1_ADDR(
0),
.WR1_MAX_ADDR(
800*480),
.WR1_LENGTH(
9'h100),
.WR1_LOAD(!DLY_RST_0),
.WR1_CLK(CCD_PIXCLK),
 


.WR1_DATA為16 bit,使用RGB16(R5G6B5)的方式傳入,不一定得跟我一樣使用R5G6B5的方式,若你要R5G5G5也可以,反正最多只能16 bit就好。

sCCD_R原來為12 bit,我只取高5 bit而以。

392行

//  FIFO Write Side 1
.WR2_DATA({sCCD_R[11:7], sCCD_G[11:6],  sCCD_B[11:7]}),
.WR2(sCCD_DVAL),
.WR2_ADDR(
22'h100000),
.WR2_MAX_ADDR(22'h100000 + 800*480),
.WR2_LENGTH(9'h100),
.WR2_LOAD(!DLY_RST_0),
.WR2_CLK(CCD_PIXCLK),
 


FIFO Write Side 2就可以自由應用了,這裡之所以也傳入RGB16,是因為網友想將capture的影像放在SDRAM的其他位置做處理,所以在此範例又傳入RGB16,只是address有所不同,實際可依你的需要善用FIFO Write Side 2。

400行

//  FIFO Read Side 1 to LTM
.RD1_DATA(Read_DATA3),
.RD1(Read),
.RD1_ADDR(
0),
.RD1_MAX_ADDR(
800*480),
.RD1_LENGTH(
9'h100),
.RD1_LOAD(!DLY_RST_0),
.RD1_CLK(
~ltm_nclk),
 


FIFO Read Side1負責將資料傳給LTM controller,注意我另外用了wire Read_DATA3,而不是原來的Read_DATA1,理由我稍後會做解釋。

408行

//  FIFO Read Side 1
.RD2_DATA(Read_DATA4),
.RD2(Read),
.RD2_ADDR(
22'h100000 + 800*480 + 200),
.RD2_MAX_ADDR(22'h100000 + 800*480 + 300),
.RD2_LENGTH(9'h100),
.RD2_LOAD(!DLY_RST_0),
.RD2_CLK(
~ltm_nclk),


FIFO Read Side 1也可以自由應用,因為網友想要解取部分影像,所以才這樣寫,實際可依你的需要善用FIFO Read Side 2。

444行

touch_tcon tcon0 (
  .iCLK(ltm_nclk),
  .iRST_n(DLY_RST_2),
 
// sdram side
  .iREAD_DATA1(Read_DATA1),
  .iREAD_DATA2(Read_DATA2),
  .oREAD_SDRAM_EN(Read),
 
// lcd side
  .oLCD_R(ltm_r),
  .oLCD_G(ltm_g),
  .oLCD_B(ltm_b),
  .oHD(ltm_hd),
  .oVD(ltm_vd),
  .oDEN(ltm_den)
);


這是LTM controller,需要傳入Read_DATA1與Read_DATA2,不過因為我們已經改用RGB16,在不更改touch_tcon module的前提下,我們需要對Read_DATA1與Read_DATA2做手腳。

428行

// add by oomusou for RGB16
assign Read_DATA1 = {Read_DATA3[10:6], Read_DATA3[4:0], 5'h00};
assign Read_DATA2 = {Read_DATA3[5:5], 4'h0, Read_DATA3[15:11], 5'h00};


本文最關鍵的地方在這裡,原來的Read_DATA1是由{sCCD_G[11:7],  sCCD_B[11:2]}所構成,而Read_DATA2是由{sCCD_G[6:2], sCCD_R[11:2]}所構成,但現在由Sdram_Control_4Port的RD1_DATA吐出來的是Read_DATA3,是由{sCCD_R[11:7], sCCD_G[11:6],  sCCD_B[11:7]}所構成,所以必須將Read_DATA3轉換成Read_DATA1與Read_DATA2原本的格式,LTM才會顯示正常,否則會有色偏的情形

由於RGB16取的是高5 bit,所以R/G/B的低5 bit必須補0,如此RGB皆以10 bit傳進LTM controller。

完整程式下載
DE2_70_D5M_LTM_RGB32.7z
DE2_70_D5M_LTM_RGB16.7z

Conclusion
藉由RGB16的方式,可以省下一半的SDRAM另做他用,而且畫質仍可接受,這種技巧在DE2這種只有1顆SDRAM的環境下非常有用。

See Also
(原創) 如何計算SDRAM使用頻寬? (SOC) (DE2) (DE2-70) (TRDB-DC2) (TRDB-D5M) (TRDB-LCM) (TRDB-LTM)

posted on 2008-10-24 17:40  真 OO无双  阅读(10459)  评论(44编辑  收藏  举报

导航