x393  1.0
FPGAcodeforElphelNC393camera
cmprs_frame_sync.v
Go to the documentation of this file.
1 
39 `timescale 1ns/1ps
40 
42  parameter FRAME_HEIGHT_BITS= 16, // Maximal frame height
43  parameter LAST_FRAME_BITS= 16, // number of bits in frame counter (before rolls over)
44  parameter CMPRS_TIMEOUT_BITS= 12,
45  parameter CMPRS_TIMEOUT= 1000 // mclk cycles
46 
47 )(
48 // input rst,
49  input xclk, // global clock input, compressor single clock rate
50  input mclk, // global system/memory clock
51  input mrst, // @posedge mclk, sync reset
52  input xrst, // @posedge xclk, sync reset
53 
54  input cmprs_en, // @mclk 0 resets immediately
55  output cmprs_en_extend, // @mclk keep compressor enabled for graceful shutdown
56 
57 // TODO: delay vsync_late if still compressing previous frame in multi-frame buffer mode
58 
59  input cmprs_run, // @mclk enable propagation of vsync_late to frame_start_dst in bonded(sync to src) mode
60  input cmprs_standalone, // @mclk single-cycle: generate a single frame_start_dst in unbonded (not synchronized) mode.
61  // cmprs_run should be off
62  input sigle_frame_buf, // memory controller uses a single frame buffer (frame_number_* == 0), use other sync
63  input vsync_late, // @mclk delayed start of frame, @xclk. In 353 it was 16 lines after VACT active
64  // source channel should already start, some delay give time for sequencer commands
65  // that should arrive before it
66  input frame_started, // @xclk started first macroblock (checking for broken frames)
67 
68  output frame_start_dst, // @mclk - trigger receive (tiled) memory channel (it will take care of single/repetitive
69  // this output either follows vsync_late (reclocks it) or generated in non-bonded mode
70  // (compress from memory)
71  input [FRAME_HEIGHT_BITS-1:0] line_unfinished_src,// number of the current (unfinished ) line, in the source (sensor) channel (RELATIVE TO FRAME, NOT WINDOW?)
72  input [LAST_FRAME_BITS-1:0] frame_number_src, // current frame number (for multi-frame ranges) in the source (sensor) channel
73  input frame_done_src, // single-cycle pulse when the full frame (window) was transferred to/from DDR3 memory
74  // frame_done_src is later than line_unfinished_src/ frame_number_src changes
75  // Used withe a single-frame buffers
76 
77  input [FRAME_HEIGHT_BITS-1:0] line_unfinished, // number of the current (unfinished ) line in this (compressor) channel
78  input [LAST_FRAME_BITS-1:0] frame_number, // current frame number (for multi-frame ranges) in this (compressor channel
79  input frame_done, // input - single-cycle pulse when the full frame (window) was transferred to/from DDR3 memory
80  output reg suspend, // suspend reading data for this channel - waiting for the source data
81 
82  input stuffer_running, // @xclk2x stuffer is running/flushing
83  output reg force_flush_long, // force flush (abort frame), can be any clock and may last until stuffer_done_mclk
84  // stuffer will re-clock and extract 0->1 transition
86  output reading_frame,
87  output frame_started_mclk // use to store frame number
88 );
89 /*
90  Abort frame (force flush) if:
91  a) "broken frame" - attempted to start a new frame before previous one was completely read from the memory
92  b) turned off enable while frame was being compressed
93  Abort frame lasts until flush end or timeout expire
94 */
95 // wire vsync_late_mclk; // single mclk cycle, reclocked from vsync_late
96 // wire frame_started_mclk;
99  reg frames_differ; // src and dest point to different frames (single-frame buffer mode), disregard line_unfinished_*
100  reg frames_numbers_differ; // src and dest point to different frames (multi-frame buffer mode), disregard line_unfinished_*
101  reg line_numbers_sync; // src unfinished line number is > this unfinished line number
102 
103  reg reading_frame_r; // compressor is reading frame data (make sure input is done before starting next frame, otherwise make it a broken frame
110 // reg cmprs_en_xclk;
113 
116 
117 // always @ (posedge xclk) begin
118 // cmprs_en_xclk <=cmprs_en;
119 // end
120  always @ (posedge mclk) begin
121  if (mrst) cmprs_en_extend_r <= 0;
122  else if (cmprs_en) cmprs_en_extend_r <= 1;
123  else if ((timeout == 0) || !stuffer_running_mclk_r) cmprs_en_extend_r <= 0;
124  end
125 
126  always @ (posedge mclk) begin
127  stuffer_running_mclk_r <= stuffer_running; // re-clock from negedge xclk2x
128 
130  else if (!cmprs_en_extend_r) timeout <= 0;
131  else timeout <= timeout - 1;
132 
133  cmprs_en_d <= cmprs_en;
134 
135  broken_frame <= cmprs_en && cmprs_run && vsync_late && reading_frame_r; // single xclk pulse
137 
139  else if (broken_frame || aborted_frame) force_flush_long <= 1;
140 
141 
143  else if (frame_started_mclk) reading_frame_r <= 1;
144 
146  if (!cmprs_en) bonded_mode <= 0;
147  else if (cmprs_run) bonded_mode <= 1;
148  else if (cmprs_standalone) bonded_mode <= 0;
149 
150  if (!cmprs_en || !cmprs_run || vsync_late) frames_differ <= 0;
151  else if (frame_done_src) frames_differ <= 1'b1;
152 
154 
156 
157 // suspend <= !bonded_mode && ((sigle_frame_buf ? frames_differ : frames_numbers_differ) || line_numbers_sync);
158 // suspend <= bonded_mode && ((sigle_frame_buf ? frames_differ : frames_numbers_differ) || !line_numbers_sync);
160 
161 
162  end
163 
164 // pulse_cross_clock vsync_late_mclk_i (.rst(xrst), .src_clk(xclk), .dst_clk(mclk), .in_pulse(cmprs_en_xclk && vsync_late), .out_pulse(vsync_late_mclk),.busy());
166 
167 endmodule
168 
[LAST_FRAME_BITS-1:0] 1600frame_number_src
frame_started_i pulse_cross_clock
1620timeoutreg[CMPRS_TIMEOUT_BITS-1:0]
[LAST_FRAME_BITS-1:0] 1603frame_number
[FRAME_HEIGHT_BITS-1:0] 1599line_unfinished_src
[FRAME_HEIGHT_BITS-1:0] 1602line_unfinished