x393  1.0
FPGAcodeforElphelNC393camera
imu_timestamps393.v
Go to the documentation of this file.
1 
39 `timescale 1ns/1ps
40 
42  input xclk, // 80 MHz, posedge
43  input rst, // sync reset (@posedge xclk)
44  output reg ts_snap, // request to take a local time snapshot
45  input ts_stb, // one clock pulse before receiving a local TS data
46  input [7:0] ts_data, // local timestamp data (s0,s1,s2,s3,u0,u1,u2,u3==0)
47 
48  input [3:0] ts_rq,// requests to create timestamps (4 channels), @posedge xclk
49  output [3:0] ts_ackn, // timestamp for this channel is stored
50  input [3:0] ra, // read address (2 MSBs - channel number, 2 LSBs - usec_low, (usec_high ORed with channel <<24), sec_low, sec_high
51  output [15:0] dout);// output data
52  reg ts_rcv;
53  reg ts_busy;
54  reg [1:0] chn; // channel for which timestamp is bein requested/received
55  wire [3:0] rq_pri; // 1-hot prioritized timestamp request
56  wire [1:0] rq_enc; // encoded request channel
57  reg [2:0] cntr; // ts rcv counter
58  wire pre_snap;
59  reg [7:0] ts_data_r; // previous value of ts_data
60  reg [15:0] ts_ram[0:15];
61  reg rcv_last; // receiving last byte (usec MSB)
62  reg [3:0] ts_ackn_r;
63  wire [3:0] chn1hot;
64  wire pre_ackn;
65 
66  assign rq_pri = {ts_rq[3] & ~(|ts_rq[2:0]),
67  ts_rq[2] & ~(|ts_rq[1:0]),
68  ts_rq[1] & ~ ts_rq[0],
69  ts_rq[0]};
70  assign rq_enc = {rq_pri[3] | rq_pri[2],
71  rq_pri[3] | rq_pri[1]};
72 
73  assign pre_snap = (|ts_rq) && !ts_busy;
74  assign chn1hot = {chn[1] & chn[0], chn[1] & ~chn[0], ~chn[1] & chn[0], ~chn[1] & ~chn[0]};
75  assign pre_ackn = ts_rcv && (cntr == 3'h6);
76 
77 
78  assign ts_ackn = ts_ackn_r;
79 
80 
81  always @ (posedge xclk) begin
82  ts_snap <= pre_snap && !rst;
83 
84  if (ts_rcv) ts_data_r <= ts_data;
85 
86  if (rst) ts_busy <= 0;
87  else if (pre_snap) ts_busy <= 1;
88  else if (ts_rcv && (cntr == 3'h6)) ts_busy <= 0; // adjust 6?
89 
90  rcv_last <= ts_rcv && (cntr == 3'h6);
91 
92  if (rst) ts_rcv <= 0;
93  else if (ts_stb) ts_rcv <= 1;
94  else if (rcv_last) ts_rcv <= 0;
95 
96  if (!ts_rcv) cntr <= 0;
97  else cntr <= cntr + 1;
98  if (pre_snap) chn <= rq_enc;
99  // insert channel instead of the usec MSB, swap usec <-> sec
100  if (ts_rcv && cntr[0]) ts_ram[{chn, ~cntr[2], cntr[1]}] <= {rcv_last ? {6'b0,chn} : ts_data, ts_data_r};
101 
102  if (rst) ts_ackn_r <= 4'hf;
103  else ts_ackn_r <= ts_rq & (ts_ackn_r | (chn1hot & {4{pre_ackn}}));
104 
105 
106 
107 
108  end
109 
110  assign dout[15:0] = ts_ram[ra];
111 
112 endmodule
113 
[0:15] 3716ts_ramreg[15:0]