forked from coderdj/redax
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathV1724.cc
534 lines (462 loc) · 16.9 KB
/
V1724.cc
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
#include "V1724.hh"
#include "Options.hh"
V1724::V1724(MongoLog *log){
fOptions = NULL;
fBoardHandle=fLink=fCrate=fBID=-1;
fBaseAddress=0;
fLog = log;
}
V1724::~V1724(){
End();
}
int V1724::Init(Options *options, int link, int crate, int bid, unsigned int address=0){
/*
// I am hoping that minesweeper will no longer be needed
// MINESWEEPER
stringstream command;
command<<"(cd /home/xedaq/minesweeper && echo `./minesweeper -l "<<
link<<" -c "<<crate<<"`)";
cout<<"Sending command: "<<command.str()<<endl;
int retsys = system(command.str().c_str());
cout<<"Returned: "<<retsys<<endl;
usleep(1000);
*/
int a = CAENVME_Init(cvV2718, link, crate, &fBoardHandle);
if(a != cvSuccess){
cout<<"Failed to init board, error code: "<<a<<", handle: "<<fBoardHandle<<
" at link "<<link<<" and bdnum "<<crate<<endl;
fBoardHandle = -1;
return -1;
}
fOptions = options;
// To start we do not know which FW version we're dealing with (for data parsing)
fFirmwareVersion = fOptions->GetInt("firmware_version", -1);
if(fFirmwareVersion == -1){
cout<<"Firmware version unspecified in options"<<endl;
return -1;
}
if((fFirmwareVersion != 0) && (fFirmwareVersion != 1)){
cout<<"Firmware version unidentified, accepted versions are {0, 1}"<<endl;
return -1;
}
fLink = link;
fCrate = crate;
fBID = bid;
fBaseAddress=address;
cout<<"Successfully initialized board at "<<fBoardHandle<<endl;
clock_counter = 0;
last_time = 0;
seen_over_15 = false;
seen_under_5 = true; // starts run as true
return 0;
}
int V1724::GetClockCounter(u_int32_t timestamp){
// The V1724 has a 31-bit on board clock counter that counts 10ns samples.
// So it will reset every 21 seconds. We need to count the resets or we
// can't run longer than that. But it's not as simple as incementing a
// counter every time a timestamp is less than the previous one because
// we're multi-threaded and channels are quasi-independent. So we need
// this fancy logic here.
//Seen under 5, true first time you see something under 5. False first time you
// see something under 15 but >5
// Seen over 15, true first time you se something >15 if under 5=false. False first
// time you see something under 5
// First, is this number greater than the previous?
if(timestamp > last_time){
// Case 1. This is over 15s but seen_under_5 is true. Give 1 back
if(timestamp >= 15e8 && seen_under_5 && clock_counter != 0)
return clock_counter-1;
// Case 2. This is over 5s and seen_under_5 is true.
else if(timestamp >= 5e8 && timestamp < 15e8 && seen_under_5){
seen_under_5 = false;
last_time = timestamp;
return clock_counter;
}
// Case 3. This is over 15s and seen_under_5 is false
else if(timestamp >= 15e8 && !seen_under_5){
seen_over_15 = true;
last_time = timestamp;
return clock_counter;
}
// Case 5. Anything else where the clock is progressing correctly
else{
last_time = timestamp;
return clock_counter;
}
}
// Second, is this number less than the previous?
else if(timestamp < last_time){
// Case 1. Genuine clock reset. under 5s is false and over 15s is true
if(timestamp < 5e8 && !seen_under_5 && seen_over_15){
seen_under_5 = true;
seen_over_15 = false;
last_time = timestamp;
clock_counter++;
return clock_counter;
}
// Case 2: Any other jitter within the 21 seconds, just return
else{
return clock_counter;
}
}
else{
std::stringstream err;
err<<"Something odd in your clock counters. t_new: "<<timestamp<<
" last time: "<<last_time<<" over 15: "<<seen_over_15<<
" under 5: "<<seen_under_5;
fLog->Entry(err.str(), MongoLog::Warning);
// Counter equal to last time, so we're happy and keep the same counter
return clock_counter;
}
}
int V1724::WriteRegister(unsigned int reg, unsigned int value){
//std::cout<<"Writing reg:val: "<<hex<<reg<<":"<<value<<dec<<std::endl;
u_int32_t write=0;
write+=value;
if(CAENVME_WriteCycle(fBoardHandle, fBaseAddress+reg,
&write,cvA32_U_DATA,cvD32) != cvSuccess){
std::stringstream err;
err<<"Failed to write register 0x"<<hex<<reg<<dec<<" to board "<<fBID<<
" with value "<<hex<<value<<dec<<" board handle "<<fBoardHandle<<endl;
fLog->Entry(err.str(), MongoLog::Warning);
return -1;
}
// std::cout<<hex<<"Wrote register "<<reg<<" with value "<<value<<" for board "<<dec<<fBID<<std::endl;
return 0;
}
unsigned int V1724::ReadRegister(unsigned int reg){
unsigned int temp;
int ret = -100;
if((ret = CAENVME_ReadCycle(fBoardHandle, fBaseAddress+reg, &temp,
cvA32_U_DATA, cvD32)) != cvSuccess){
std::stringstream err;
std::cout<<"Read returned: "<<ret<<" "<<hex<<temp<<std::endl;
err<<"Failed to read register 0x"<<hex<<reg<<dec<<" on board "<<fBID<<
": "<<ret<<endl;
fLog->Entry(err.str(), MongoLog::Warning);
return 0xFFFFFFFF;
}
return temp;
}
u_int32_t V1724::ReadMBLT(unsigned int *&buffer){
// Initialize
unsigned int blt_bytes=0;
int nb=0,ret=-5;
// The best-equipped V1724E has 4MS/channel memory = 8 MB/channel
// the other, V1724G, has 512 MS/channel = 1MB/channel
//unsigned int BLT_SIZE=8388608; //8*8388608; // 8MB buffer size
unsigned int BLT_SIZE=524288;
unsigned int BUFFER_SIZE = 8388608; // 8 MB memory of digi (even allocating more than needed)
u_int32_t *tempBuffer = new u_int32_t[BUFFER_SIZE];
int count = 0;
do{
try{
ret = CAENVME_FIFOBLTReadCycle(fBoardHandle, fBaseAddress,
((unsigned char*)tempBuffer)+blt_bytes,
BLT_SIZE, cvA32_U_MBLT, cvD64, &nb);
}catch(std::exception E){
std::cout<<fBoardHandle<<" sucks"<<std::endl;
std::cout<<"BLT_BYTES: "<<blt_bytes<<std::endl;
std::cout<<"nb: "<<nb<<std::endl;
std::cout<<E.what()<<std::endl;
throw;
};
if( (ret != cvSuccess) && (ret != cvBusError) ){
stringstream err;
err<<"Read error in board "<<fBID<<" after "<<count<<" reads: "<<dec<<ret;
fLog->Entry(err.str(), MongoLog::Error);
u_int32_t data=0;
WriteRegister(0xEF24, 0xFFFFFFFF);
data = ReadRegister(0x8104);
std::cout<<"Board status: "<<hex<<data<<dec<<std::endl;
delete[] tempBuffer;
return 0;
}
count++;
blt_bytes+=nb;
if(blt_bytes>BUFFER_SIZE){
stringstream err;
err<<"You managed to transfer more data than fits on board."<<
"Transferred: "<<blt_bytes<<" bytes, Buffer: "<<BUFFER_SIZE<<" bytes.";
fLog->Entry(err.str(), MongoLog::Error);
delete[] tempBuffer;
return 0;
}
}while(ret != cvBusError);
// Now, unfortunately we need to make one copy of the data here or else our memory
// usage explodes. We declare above a buffer of several MB, which is the maximum capacity
// of the board in case every channel is 100% saturated (the practical largest
// capacity is certainly smaller depending on settings). But if we just keep reserving
// O(MB) blocks and filling 50kB with actual data, we're gonna run out of memory.
// So here we declare the return buffer as *just* large enough to hold the actual
// data and free up the rest of the memory reserved as buffer.
// In tests this does not seem to impact our ability to read out the V1724 at the
// maximum bandwidth of the link.
if(blt_bytes>0){
buffer = new u_int32_t[blt_bytes/(sizeof(u_int32_t))];
std::memcpy(buffer, tempBuffer, blt_bytes);
}
delete[] tempBuffer;
return blt_bytes;
}
int V1724::ConfigureBaselines(vector <u_int16_t> &end_values,
int nominal_value, int ntries){
// The point of this function is to set the voltage offset per channel such
// that the baseline is at exactly 16000 (or whatever value is set in the
// config file). The DAC seems to be a very sensitive thing and there
// are some strategically placed sleep statements (placed via trial, error,
// and tears) throughout the code. Take care if changing things here.
// Initial parameters:
int adjustment_threshold = 5;
int current_iteration=0;
int nChannels = 8;
int repeat_this_many=5;
int triggers_per_iteration = 10;
// Determine starting values. If the flag 'start_provided' is set then the
// initial argument vector already has our start values. If not then we can
// make a decent guess here.
u_int32_t starting_value = u_int32_t( (0x3fff-nominal_value)*
((0.9*0xffff)/0x3fff) + 3277);
vector<u_int16_t> dac_values(nChannels, starting_value);
if(end_values[0]!=0 && end_values.size() ==
(unsigned int)(nChannels)){ // use start values if sent
std::cout<<"Found good start values for digi "<<fBID<<": ";
for(unsigned int x=0; x<end_values.size(); x++){
dac_values[x] = end_values[x];
std::cout<<dac_values[x]<<" ";
}
std::cout<<std::endl;
}
vector<int> channel_finished(nChannels, 0);
vector<bool> update_dac(nChannels, true);
// Load up the DAC values
if(LoadDAC(dac_values, update_dac)!=0){
std::stringstream error;
error<<"Digitizer "<<fBID<<" failed to load DAC in baseline routine.";
fLog->Entry(error.str(), MongoLog::Error);
return -2;
}
// ****************************
// Main loop
// ****************************
while(current_iteration < ntries){
bool breakout=true;
for(unsigned int x=0; x<channel_finished.size(); x++){
if(channel_finished[x]<repeat_this_many)
breakout=false;
}
if(breakout)
break;
// enable adc
WriteRegister(0x8100,0x4);//x24? // Acq control reg
if(MonitorRegister(0x8104, 0x4, 1000, 1000) != true){
fLog->Entry("Timed out waiting for acquisition to start in baselines", MongoLog::Warning);
return -1;
}
//write trigger
for(int ntrig=0; ntrig<triggers_per_iteration; ntrig++){
WriteRegister(0x8108,0x1); // Software trig reg
usleep(1000); // Give time for event?
}
// disable adc
WriteRegister(0x8100,0x0);//x24? // Acq control reg
// Read data
u_int32_t *buff = NULL;
u_int32_t size = 0;
size = ReadMBLT(buff);
// Check for mal formed data
if(size>0 && size<=16){
std::cout<<"Delete undersized buffer ("<<size<<")"<<std::endl;
delete[] buff;
continue;
}
if(size == 0){
std::cout<<"No event though board said there would be one"<<std::endl;
if(buff != NULL) delete[] buff;
continue;
}
// Now we're going to acquire 'n' triggers
std::vector<double>baseline_per_channel(nChannels, 0);
std::vector<double>good_triggers_per_channel(nChannels, 0);
// Parse
unsigned int idx = 0;
while(idx < size/sizeof(u_int32_t)){
if(buff[idx]>>20==0xA00){ // header
u_int32_t esize = buff[idx]&0xFFFFFFF;
u_int32_t cmask = buff[idx+1]&0xFF;
u_int32_t csize = (esize - 4) / cmask;
idx += 4;
// Loop through channels
for(unsigned int channel=0; channel<8; channel++){
if(fFirmwareVersion == 0){
csize = buff[idx] - 2; // In words (4 bytes). The -2 is cause of header
idx += 2;
}
else if(fFirmwareVersion == 1){
csize = (esize - 4) / __builtin_popcount(cmask);
}
float baseline = -1.;
long int tbase = 0;
int bcount = 0;
unsigned int minval = 0x3fff, maxval=0;
if(!((cmask>>channel)&1))
continue;
if(channel_finished[channel]>=repeat_this_many){
idx+=csize;
continue;
}
for(unsigned int i=0; i<csize; i++){
if(((buff[idx+i]&0xFFFF)==0) || (((buff[idx+i]>>16)&0xFFFF)==0))
continue;
tbase += buff[idx+i]&0xFFFF;
tbase += (buff[idx+i]>>16)&0xFFFF;
bcount+=2;
if((buff[idx+i]&0xFFFF)<minval)
minval = buff[idx+i]&0xFFFF;
if((buff[idx+i]&0xFFFF)>maxval)
maxval = buff[idx+i]&0xFFFF;
if(((buff[idx+i]>>16)&0xFFFF)<minval)
minval=(buff[idx+i]>>16)&0xFFFF;
if(((buff[idx+i]>>16)&0xFFFF)>maxval)
maxval=(buff[idx+i]>>16)&0xFFFF;
}
idx += csize;
// Toss if signal inside
if(abs((int)(maxval)-(int)(minval))>30){
std::cout<<"Signal in baseline, channel "<<channel
<<" min: "<<minval<<" max: "<<maxval<<std::endl;
}
else{
baseline = (float(tbase) / ((float(bcount))));
// Add to total
baseline_per_channel[channel]+= baseline;
good_triggers_per_channel[channel]+=1.;
}
} // end for loop through channels
//delete[] buff;
//break;
}
else
idx++;
}// end parse data
delete[] buff;
// Get average from total
for(int channel=0; channel<nChannels; channel++)
baseline_per_channel[channel]/=good_triggers_per_channel[channel];
// Compute update to baseline if any
// Time for the **magic**. We want to see how far we are off from nominal and
// adjust up and down accordingly. We will always adjust just a tiny bit
// less than we think we need to to avoid getting into some overshoot
// see-saw type loop where we never hit the target.
for(int channel=0; channel<nChannels; channel++){
if(channel_finished[channel]>=repeat_this_many)
continue;
if(good_triggers_per_channel[channel]==0)
continue;
float absolute_unit = float(0xffff)/float(0x3fff);
int adjustment = .5*int(absolute_unit*((float(baseline_per_channel[channel])-
float(nominal_value))));
if(abs(float(baseline_per_channel[channel])-float(nominal_value)) < adjustment_threshold){
channel_finished[channel]++;
std::cout<<"Channel "<<channel<<" converging at step "<<
channel_finished[channel]<<"/5"<<std::endl;
}
else{
channel_finished[channel]=0;
update_dac[channel] = true;
if(adjustment<0 && (u_int32_t(abs(adjustment)))>dac_values[channel]){
dac_values[channel]=0x0;
std::cout<<"Channel "<<channel<<" DAC to zero"<<std::endl;
}
else if(adjustment>0 &&dac_values[channel]+adjustment>0xffff){
dac_values[channel]=0xffff;
std::cout<<"Channel "<<channel<<" DAC to 0xffff"<<std::endl;
}
else {
std::cout<<"Had channel "<<channel<<" at "<<dac_values[channel];
dac_values[channel]+=(adjustment);
std::cout<<" but now it's at "<<dac_values[channel]<<" (adjustment) BL: "<<
baseline_per_channel[channel]<<
" ("<<good_triggers_per_channel[channel]<<" iterations)"<<std::endl;
}
}
} // End final channel adjustment
current_iteration++;
// Load DAC
if(LoadDAC(dac_values, update_dac)!=0){
std::stringstream error;
error<<"Digitizer "<<fBID<<" failed to load DAC in baseline routine.";
fLog->Entry(error.str(), MongoLog::Error);
return -2;
}
} // end while(current_iteration < ntries)
for(unsigned int x=0; x<channel_finished.size(); x++){
if(channel_finished[x]<2){ // Be a little more lenient in case it's just starting to converge
std::stringstream error;
error<<"Baseline routine did not finish for channel "<<x<<" (and maybe others)."<<std::endl;
fLog->Entry(error.str(), MongoLog::Message);
return -1;
}
}
end_values = dac_values;
return 0;
}
int V1724::LoadDAC(vector<u_int16_t>dac_values, vector<bool> &update_dac){
// Loads DAC values into registers
for(unsigned int x=0; x<dac_values.size(); x++){
if(x>7 || update_dac[x]==false) // oops
continue;
// We updated, or at least tried to update
//update_dac[x]=false;
// Give the DAC time to be set if needed
if(MonitorRegister((0x1088)+(0x100*x), 0x4, 100, 1000, 0) != true){
stringstream errorstr;
errorstr<<"Timed out waiting for channel "<<x<<" in DAC setting";
fLog->Entry(errorstr.str(), MongoLog::Error);
return -1;
}
// Now write channel DAC values
if(WriteRegister((0x1098)+(0x100*x), dac_values[x])!=0){
stringstream errorstr;
errorstr<<"Failed writing DAC "<<hex<<dac_values[x]<<dec<<" in channel "<<x;
fLog->Entry(errorstr.str(), MongoLog::Error);
return -1;
}
// Give the DAC time to be set if needed
if(MonitorRegister((0x1088)+(0x100*x), 0x4, 100, 1000, 0) != true){
stringstream errorstr;
errorstr<<"Timed out waiting for channel "<<x<<" after DAC setting";
fLog->Entry(errorstr.str(), MongoLog::Error);
return -1;
}
}
// Sleep a bit because apparently checking the register means nothing and you
// gotta wait a little for the actual voltage to be updated
usleep(5000);
return 0;
}
int V1724::End(){
if(fBoardHandle>=0)
CAENVME_End(fBoardHandle);
fBoardHandle=fLink=fCrate=fBID=-1;
fBaseAddress=0;
return 0;
}
bool V1724::MonitorRegister(u_int32_t reg, u_int32_t mask, int ntries, int sleep, u_int32_t val){
int counter = 0;
u_int32_t rval = 0;
if(val == 0) rval = 0xffffffff;
while(counter < ntries){
rval = ReadRegister(reg);
if(rval == 0xffffffff)
return false;
if((val == 1 && (rval&mask)) || (val == 0 && !(rval&mask)))
return true;
counter++;
usleep(sleep);
}
std::cout<<"MonitorRegister failed for "<<hex<<reg<<" with mask "<<
mask<<" and register value "<<rval<<"... couldn't get "<<val<<dec<<
std::endl;
return false;
}