x393  1.0
FPGAcodeforElphelNC393camera
encoderDCAC393.v
Go to the documentation of this file.
1 
38 
39 
40 // Accepts 13-bits signed data (only 12-bit can be ecoded), so DC difference (to be encoded) is limited (saturated) to 12 bits, not the value itself
41 // AC - always limited to 800 .. 7ff
43  input clk, // pixel clock, posedge
44  input en, // enable (0 resets)
45  input lasti, // was "last MCU in a frame" (@ stb)
46  input first_blocki, // first block in frame - save fifo write address (@ stb)
47  input [ 2:0] comp_numberi, // [2:0] component number 0..2 in color, 0..3 - in jp4diff, >= 4 - don't use (@ stb)
48  input comp_firsti, // first this component in a frame (reset DC) (@ stb)
49  input comp_colori, // use color - huffman? (@ stb)
50  input comp_lastinmbi, // last component in a macroblock (@ stb) is it needed?
51  input stb, // strobe that writes firsti, lasti, tni,average
52  input [12:0] zdi, // [11:0] zigzag-reordered data input
53  input first_blockz, // first block input (@zds)
54  input zds, // strobe - one ahead of the DC component output
55  output reg last, //
56  output reg [15:0] do,
57  output reg dv,
58  // just for debug
60  output [2:0] dbg_block_mem_ra,
61  output [2:0] dbg_block_mem_wa,
62  output [2:0] dbg_block_mem_wa_save
63  );
64 
65 
66 // 8x13 DC storage memory
67  reg [12:0] dc_mem[7:0];
68  reg [12:0] dc_diff0, dc_diff;
69  wire [11:0] dc_diff_limited= (dc_diff[12]==dc_diff[11])?
70  dc_diff[11:0] :
71  {~dc_diff[11],{11{dc_diff[11]}}}; // difference (to be encoded) limited to fit 12 bits
72  reg [12:0] dc_restored; // corrected DC value of the current block, compensated to fit difference to 12 bits
73  reg [ 5:0] rll_cntr;
74  reg [5:0] cntr;
75  reg [11:0] ac_in;
76 
77  wire izero=(ac_in[11:0]==12'b0);
78 
79  reg [14:0] val_r; // DC diff/AC values to be sent out, registered
80 
81  reg DCACen; // enable DC/AC (2 cycles ahead of do
82  wire rll_out;
83  wire pre_dv;
85  reg [12:0] zdi_d;
86  reg [3:0] zds_d;
87  wire DC_tosend= zds_d[2];
88  wire pre_DCACen= zds_d[1];
89 
90  wire [2:0] comp_numbero; // [2:0] component number 0..2 in color, 0..3 - in jp4diff, >= 4 - don't use
91  wire comp_firsto; // first this component in a frame (reset DC)
92  wire comp_coloro; // use color - huffman?
93 // wire comp_lastinmbo; // last component in a macroblock
94  wire lasto; // last macroblock in a frame
95  reg [2:0] block_mem_ra;
96  reg [2:0] block_mem_wa;
97  reg [2:0] block_mem_wa_save;
98  reg [6:0] block_mem_ram[0:7];
100 
101  assign comp_numbero[2:0]= block_mem_o[2:0];
102  assign comp_firsto= block_mem_o[3];
103  assign comp_coloro= block_mem_o[4];
104  assign comp_lastinmbo= block_mem_o[5];
105  assign lasto= block_mem_o[6];
106 
110 
111  always @ (posedge clk) begin
113  if (!en) block_mem_wa[2:0] <= 3'h0;
114  else if (stb) block_mem_wa[2:0] <= block_mem_wa[2:0] +1;
115 
116  if (stb && first_blocki) block_mem_wa_save[2:0] <= block_mem_wa[2:0];
117 
118  if (!en) block_mem_ra[2:0] <= 3'h0;
119  else if (zds) block_mem_ra[2:0] <= first_blockz?block_mem_wa_save[2:0]:(block_mem_ra[2:0] +1);
120  end
121 
122  assign rll_out= ((val_r[12] && !val_r[14]) || (ac_in[11:0]!=12'b0)) && (rll_cntr[5:0]!=6'b0);
123  assign pre_dv=rll_out || val_r[14] || was_nonzero_AC;
124 
125  always @ (posedge clk) begin
126  val_r[14:0] <={ DC_tosend?
127  {en,
128  comp_coloro,
129  comp_lastinmbo && lasto, // last component's in a frame DC coefficient
130  dc_diff_limited[11:0]}:
131  {2'b0,
132  (cntr[5:0]==6'h3f),
133  ac_in[11:0]}};
134  was_nonzero_AC <= en && (ac_in[11:0]!=12'b0) && DCACen;
135  if (pre_dv) do <= rll_out? {3'b0,val_r[12],6'b0,rll_cntr[5:0]}:{1'b1,val_r[14:0]};
136  dv <= pre_dv;
137  DCACen <= en && (pre_DCACen || (DCACen && (cntr[5:0]!=6'h3f))); // adjust
138  if (!DCACen) cntr[5:0] <=6'b0;
139  else cntr[5:0] <=cntr[5:0]+1;
140  end
141 
142  always @ (posedge clk) begin
143  zdi_d[12:0] <= zdi[12:0];
144  ac_in[11:0] <= (zdi_d[12]==zdi_d[11])? zdi_d[11:0]:{~zdi_d[11],{11{zdi_d[11]}}}; // always // delay + saturation
145 
146  if (DC_tosend || !izero || !DCACen) rll_cntr[5:0] <= 6'h0;
147  else if (DCACen) rll_cntr[5:0] <= rll_cntr[5:0] +1 ;
148  if (DC_tosend) last <= lasto;
149  end
150 
151 // DC components
152  always @ (posedge clk) begin
153  zds_d[3:0] <= {zds_d[2:0], zds};
154  if (zds_d[0]) dc_diff0[12:0] <= comp_firsto?13'b0:dc_mem[comp_numbero[2:0]];
155  if (zds_d[1]) dc_diff [12:0] <= zdi_d[12:0]-dc_diff0[12:0];
156  if (zds_d[2]) dc_restored[12:0] <= dc_diff0[12:0] + {dc_diff_limited[11],dc_diff_limited[11:0]};
157  if (zds_d[3]) dc_mem[comp_numbero[2:0]] <= dc_restored[12:0];
158  end
159 
160 // Generate output stream to facilitate huffman encoding. The data will go to FIFO (16x) to compensate for possible long Huffman codes
161 // and/or zero-byte insertions
162 // format:
163 // {2'b11, color,last block, dc[11:0]} - DC data
164 // {2'b10, 1'b0, last coeff, ac[11:0]} - AC data (last coeff is set if it is last- 63-rd AC coefficient)
165 // {2'h00, 2'b00, 6'b0,rll[ 5:0]} - RLL zeroes.
166 // {2'h00, 2'b01, 6'b0,rll[ 5:0]} - end of block. lower 6 bits will have length that should be ignored
167 
168 endmodule
2322dc_diffreg[12:0]
[7:0] 2320dc_memreg[12:0]
2334zdi_dreg[12:0]
2321dc_diff0reg[12:0]
2343block_mem_wareg[2:0]
2346block_mem_owire[6:0]
[2:0] 2317dbg_block_mem_ra
[ 2:0] 2305comp_numberi
[2:0] 2318dbg_block_mem_wa
2342block_mem_rareg[2:0]
[0:7] 2345block_mem_ramreg[6:0]
2344block_mem_wa_savereg[2:0]
2335zds_dreg[3:0]
2329val_rreg[14:0]
reg [15:0] 2314do
2325rll_cntrreg[5:0]
2338comp_numberowire[2:0]
2324dc_restoredreg[12:0]
2323dc_diff_limitedwire[11:0]
2327ac_inreg[11:0]
2326cntrreg[5:0]
[2:0] 2319dbg_block_mem_wa_save