Changeset 11927
- Timestamp:
- 08/31/11 16:54:53 (13 years ago)
- File:
-
- 1 edited
Legend:
- Unmodified
- Added
- Removed
-
trunk/FACT++/src/biasctrl.cc
r11860 r11927 1 1 #include <functional> 2 3 #include <boost/bind.hpp> 2 4 3 5 #include "Dim.h" … … 25 27 class ConnectionBias : public ConnectionUSB 26 28 { 27 vector<uint8_t> fBuffer;28 29 bool fIsVerbose;30 31 29 enum 32 30 { … … 38 36 enum Command_t 39 37 { 40 kCmdReset = 0, 41 kCmdRead = 1, 42 kCmdGlobalSet = 2, 43 kCmdChannelSet = 3, 44 kCmdPrint = 4 38 // Communication commands 39 kCmdReset = 0, 40 kCmdRead = 1, 41 kCmdGlobalSet = 2, 42 kCmdChannelSet = 3, 43 44 // Internal command names 45 kResetChannels = 0x10|kCmdChannelSet, 46 kUpdate = 0x10|kCmdRead, 47 kExpertChannelSet = 0x14|kCmdChannelSet, 48 kSynchronize = 0x1e, 49 //kOverCurReset = 20, 45 50 }; 46 51 47 // Resistance in Ohm for voltage correction 48 #define RESISTOR float(1000) 52 enum 53 { 54 kMaxDac = 0xfff 55 }; 56 57 boost::asio::deadline_timer fSyncTimer; 58 boost::asio::deadline_timer fRampTimer; 59 boost::asio::deadline_timer fUpdateTimer; 60 61 vector<uint8_t> fBuffer; 62 63 bool fIsVerbose; 64 65 vector<uint16_t> fVoltCmd; // Current command voltage in DAC units (12bit = 90V) 66 vector<uint16_t> fVoltGapd; 67 68 vector<bool> fPresent; 69 70 int64_t fWrapCounter; 71 int64_t fSendCounter; 72 73 int16_t fGlobalVoltCmd; // Command value to be reached 74 // uint16_t fExpertVoltRef; // Command value to be reached 75 76 int16_t fRampStep; 77 int16_t fRampTime; 78 79 uint16_t fUpdateTime; 80 81 bool fIsInitializing; 82 bool fIsRamping; 83 // bool fWaitingForAnswer; 49 84 50 85 protected: 51 52 vector<uint16_t> fVolt; // Voltage in DAC units (12bit = 90V) 53 vector<uint16_t> fRefVolt; 86 vector<uint16_t> fVolt; // Current voltage in DAC units (12bit = 90V) 87 vector<uint16_t> fVoltRef; // Current reference voltage in DAC units (12bit = 90V) 54 88 55 89 vector<int16_t> fCurrent; // Current in ADC units (12bit = 5mA) 56 vector<int16_t> fRefCurrent;57 58 vector<bool> fPresent;59 60 int fWrapCounter;61 62 90 63 91 virtual void UpdateA() … … 70 98 71 99 private: 72 void HandleReceivedData(const bs::error_code& err, size_t bytes_received, int type) 73 { 74 if (type==kCmdPrint && bytes_received==0 && !err) 75 { 76 Print(); 77 return; 78 } 79 100 bool CheckChDac(const string &cmd, uint16_t dac, uint16_t ch=0) 101 { 102 if (dac>kMaxDac) 103 { 104 Error(cmd+" - DAC value out of range."); 105 return false; 106 } 107 108 if (ch>=kNumChannels) 109 { 110 Error(cmd+" - Channel out of range."); 111 return false; 112 } 113 114 return true; 115 } 116 117 vector<char> GetCmd(uint16_t board, uint16_t channel, Command_t cmd, uint16_t dac=0) 118 { 119 vector<char> data(3); 120 121 /* 122 if (board>kNumBoards) 123 return; 124 if (channel>kNumChannelsPerBoard) 125 return; 126 if (dac>0xfff) 127 return; 128 */ 129 130 data[0] = (cmd<<5) | (board<<1) | (((channel&16)>>4) & 1); 131 data[1] = (channel<<4) | (dac>>8); 132 data[2] = dac&0xff; 133 134 return data; 135 } 136 137 vector<char> GetCmd(Command_t cmd, uint16_t id=0, uint16_t dac=0) 138 { 139 const unsigned int board = id/kNumChannelsPerBoard; 140 const unsigned int channel = id%kNumChannelsPerBoard; 141 142 return GetCmd(board, channel, cmd, dac); 143 } 144 145 bool CheckMessageLength(int received, int expected, const string &msg) 146 { 147 if (received!=expected) 148 return false; 149 150 ostringstream str; 151 str << msg << ": Expected " << expected << " bytes in answer, but got " << received << endl; 152 Error(str); 153 154 PostClose(false); 155 return false; 156 } 157 158 bool EvalAnswer(uint8_t *answer, uint16_t id, int command) 159 { 160 answer += id*3; 161 162 const uint16_t status = (answer[0]>>7)&1; 163 const uint16_t wrap = (answer[0]>>4)&7; 164 const uint16_t ddd = ((uint16_t(answer[0])&0xf)<<8) | answer[1]; 165 const uint16_t error = (answer[2]>>4)&0xf; 166 const uint16_t board = answer[2]&0xf; 167 168 if (fWrapCounter>=0) 169 { 170 if ((fWrapCounter+1)%8 != wrap) 171 { 172 ostringstream msg; 173 msg << "Corrupted answer: received wrap counter " << wrap << " is not last received counter (" << fWrapCounter << "+1)%8."; 174 Error(msg); 175 return false; 176 } 177 } 178 179 fWrapCounter = wrap; 180 181 if (command==kSynchronize) 182 { 183 ostringstream msg; 184 msg << hex << setfill('0'); 185 msg << "Initial answer received:"; 186 msg << " 0x" << setw(2) << answer[0]; 187 msg << " 0x" << setw(2) << answer[1]; 188 msg << " 0x" << setw(2) << answer[2]; 189 Message(msg); 190 191 if (status!=0 || ddd!=0 || error!=0 || board==0) 192 Warn("Initial answer doesn't seem to be a reset as naively expected."); 193 194 fSendCounter = wrap; 195 196 return true; 197 } 198 199 if (error==0x8) // No device 200 { 201 Message("Reset button on crate pressed!"); 202 SetZero(); 203 return true; 204 } 205 206 if (command==kCmdReset) 207 { 208 if (status==0 && ddd==0 && error==0 && board==0) 209 { 210 Message("Reset successfully executed."); 211 return true; 212 } 213 214 Warn("Answer to 'reset' command contains unexpected data."); 215 return false; 216 } 217 218 if (command==kCmdGlobalSet) 219 { 220 if (status==0 && ddd==0 && error==0 && board==0) 221 { 222 for (int i=0; i<kNumChannels; i++) 223 fVolt[i] = fGlobalVoltCmd; 224 225 fGlobalVoltCmd = -1; 226 227 return true; 228 } 229 230 Warn("Answer to 'global set' command contains unexpected data."); 231 return false; 232 } 233 234 if ((command&0xff)==kExpertChannelSet) 235 id = command>>8; 236 237 const int cmd = command&3; 238 239 if (cmd==kCmdRead || cmd==kCmdChannelSet) 240 { 241 if (board!=id/kNumChannelsPerBoard) 242 { 243 ostringstream out; 244 out << "Talked to board " << id/kNumChannelsPerBoard << ", but got answer from board " << board << "."; 245 Error(out); 246 return false; 247 } 248 249 // Not present 250 if (error==0x7 || error==0xf) 251 { 252 fPresent[board] = false; 253 fCurrent[id] = 0x8000; 254 return true; 255 } 256 257 fCurrent[id] = status ? -ddd : ddd; 258 fPresent[board] = true; 259 } 260 261 if (cmd==kCmdChannelSet) 262 fVolt[id] = fVoltCmd[id]; 263 264 return true; 265 266 } 267 268 private: 269 void HandleReceivedData(const bs::error_code& err, size_t bytes_received, int command, int send_counter) 270 { 80 271 // Do not schedule a new read if the connection failed. 81 272 if (bytes_received==0 || err) … … 94 285 Error(str); 95 286 } 96 PostClose(err!=ba::error::basic_errors::operation_aborted); 97 return; 98 } 99 287 PostClose(false);//err!=ba::error::basic_errors::operation_aborted); 288 return; 289 } 290 291 // Check if the number of received bytes is correctly dividable by 3 292 // This check should never fail - just for sanity 100 293 if (bytes_received%3) 101 294 { 102 295 Error("Number of received bytes not a multiple of 3, can't read data."); 103 PostClose(true); 104 return; 105 } 106 296 PostClose(false); 297 return; 298 } 299 300 // Now print the received message if requested by the user 107 301 if (fIsVerbose) 108 302 { 109 //Out() << endl << kBold << "Data received (size=" << bytes_received << "):" << endl; 110 //Out() << Converter::GetHex<uint8_t>(fBuffer, 32) << endl; 111 // FIXME: Check why more is printed than expected 112 } 113 114 const uint16_t command = type&0xf; 115 const uint16_t id = type>>4; 116 const uint16_t status = (fBuffer[0]>>7)&1; 117 const uint16_t wrap = (fBuffer[0]>>4)&7; 118 const uint16_t ddd = ((uint16_t(fBuffer[0])&0xf)<<8) | fBuffer[1]; 119 const uint16_t error = (fBuffer[2]>>4)&0xf; 120 const uint16_t board = fBuffer[2]&0xf; 121 122 if (fWrapCounter>=0 && (fWrapCounter+1)%8 != wrap) 123 { 124 Error("Corrupted answer (wrap counter not as it ought to be."); 125 return; 126 } 127 fWrapCounter = wrap; 128 129 if (error==0x8) // No device 130 { 131 ostringstream out; 132 out << "HV down requested!"; 133 Fatal(out); 134 135 GlobalSetDac(0); 136 137 // Resynchronize input and output 138 // SystemReset and status request 139 PostClose(true); 140 141 return; 142 } 143 144 if (command==kCmdReset) 145 { 146 if (status==0 && ddd==0 && error==0 && board==0) 303 vector<uint32_t> vout((bytes_received/3)*4); 304 305 for (size_t i=0; i<bytes_received/3; i++) 147 306 { 148 Message("Reset successfully executed."); 307 vout[i] = 308 (uint32_t(fBuffer[i*3+0])<<16) | 309 (uint32_t(fBuffer[i*3+1])<< 8) | 310 (uint32_t(fBuffer[i*3+2])<< 0); 311 } 312 313 Out() << endl << kBold << "Data received (size=" << bytes_received << "):" << endl; 314 Out() << " fWrapCounter=" << fWrapCounter << " fSendCounter=" << fSendCounter << " fIsInitializing=" << fIsInitializing << " fIsRamping=" << fIsRamping << endl; 315 Out() << Converter::GetHex<uint32_t>(vout, 16) << endl; 316 } 317 318 const int cmd = command&0xf; 319 320 // Check the number of received_byted according to the answer expected 321 if ((cmd==kSynchronize && !CheckMessageLength(bytes_received, 3, "Synchronization")) || 322 (cmd==kCmdReset && !CheckMessageLength(bytes_received, 3, "CmdReset")) || 323 (cmd==kCmdRead && !CheckMessageLength(bytes_received, 3*kNumChannels, "CmdRead")) || 324 (cmd==kCmdChannelSet && !CheckMessageLength(bytes_received, 3*kNumChannels, "CmdChannelSet")) || 325 (cmd==kExpertChannelSet && !CheckMessageLength(bytes_received, 3, "CmdExpertChannelSet"))) 326 return; 327 328 // Now evaluate the whole bunch of messages 329 for (size_t i=0; i<bytes_received/3; i++) 330 { 331 if (!EvalAnswer(fBuffer.data(), i, command)) 332 { 333 PostClose(false); 149 334 return; 150 335 } 151 152 Warn("Answer to 'reset' command contains unexpected data."); 153 return; 154 } 155 156 if (command==kCmdGlobalSet) 157 { 158 if (status==0 && ddd==0 && error==0 && board==0) 336 } 337 338 if (send_counter%8 != fWrapCounter) 339 { 340 ostringstream msg; 341 msg << "Corrupted answer: received wrap counter " << fWrapCounter << " is not send counter " << fSendCounter << "%8."; 342 Error(msg); 343 PostClose(false); 344 } 345 346 // Now we are ready to send a new message 347 // fWaitingForAnswer = false; 348 349 if (command==kSynchronize) 350 { 351 Message("Stream successfully synchronized."); 352 fIsInitializing = false; 353 354 // Cancel sending of the next 0 355 fSyncTimer.cancel(); 356 357 // Start continous reading of all channels 358 ScheduleUpdate(100); 359 } 360 361 // Take action depending on what is going on 362 if (command==kCmdReset) 363 Message("Reset command successfully answered."); 364 365 if (cmd==kCmdRead || cmd==kCmdChannelSet || cmd==kExpertChannelSet) 366 { 367 UpdateV(); 368 UpdateA(); 369 return; 370 } 371 372 if (cmd==kCmdReset || command==kResetChannels) 373 { 374 // Re-start cyclic reading of values after a short time 375 // to allow the currents to become stable 376 fUpdateTimer.cancel(); 377 ScheduleUpdate(100); 378 } 379 380 if (command==kUpdate) 381 ScheduleUpdate(fUpdateTime); 382 383 // If we are ramping, schedule a new ramp step 384 if (command==kCmdChannelSet && fIsRamping) 385 { 386 ScheduleRampStep(); 387 return; 388 } 389 } 390 391 // -------------------------------------------------------------------- 392 393 void HandleSyncTimer(int counter, const bs::error_code &error) 394 { 395 if (error==ba::error::basic_errors::operation_aborted) 396 { 397 Warn("Synchronization aborted..."); 398 fIsRamping = false; 399 return; 400 } 401 402 if (error) 403 { 404 ostringstream str; 405 str << "Synchronization timer: " << error.message() << " (" << error << ")";// << endl; 406 Error(str); 407 408 PostClose(false); 409 return; 410 } 411 412 if (!is_open()) 413 { 414 Warn("Synchronization in progress, but disconnected."); 415 return; 416 } 417 418 ostringstream msg; 419 msg << "Synchronization time expired (" << counter << ")" << endl; 420 Info(msg); 421 422 if (fIsInitializing) 423 { 424 PostMessage("\0", 1); 425 426 if (counter==2) 159 427 { 160 Message("GlobalSet successfully executed."); 428 Error("Synchronization attempt timed out."); 429 PostClose(false); 161 430 return; 162 431 } 163 432 164 Warn("Answer to 'global set' command contains unexpected data."); 165 return; 166 } 167 168 if (command==kCmdRead || command==kCmdChannelSet) 169 { 170 if (error==0x7 || error==0xf) 171 { 172 fPresent[board] = false; 173 fCurrent[id] = 0x8000; 174 return; 175 } 176 177 if (board!=id/kNumChannelsPerBoard) 178 { 179 ostringstream out; 180 out << "Talked to board " << id/kNumChannelsPerBoard << " but board " << board << " answered."; 181 Error(out); 182 return; 183 } 184 185 fCurrent[id] = status ? -ddd : ddd; 186 fPresent[board] = true; 187 188 UpdateA(); 189 190 return; 191 } 433 ScheduleSync(counter+1); 434 return; 435 } 436 437 Info("Synchronisation successfull."); 438 } 439 440 void ScheduleSync(int counter=0) 441 { 442 fSyncTimer.expires_from_now(boost::posix_time::milliseconds(333)); 443 fSyncTimer.async_wait(boost::bind(&ConnectionBias::HandleSyncTimer, this, counter, dummy::error)); 192 444 } 193 445 … … 195 447 void ConnectionEstablished() 196 448 { 197 fWrapCounter = -1; 198 fBuffer.resize(3); 199 200 SystemReset(); 201 ReadAllChannels(); 202 } 203 204 void HandleReadTimeout(const bs::error_code &error) 449 // Reset everything.... 450 fSendCounter = -1; 451 fWrapCounter = -1; 452 fGlobalVoltCmd = -1; 453 fIsInitializing = true; 454 455 fVolt.assign( 0, kNumChannels); 456 fVoltRef.assign(0, kNumChannels); 457 fVoltCmd.assign(0, kNumChannels); 458 459 // Send a single 0 (and possible two consecutive 0's 460 // to make sure we are in sync with the device) 461 PostMessage("\0", 1); 462 AsyncRead(ba::buffer(fBuffer, 3), kSynchronize, ++fSendCounter); 463 // fWaitingForAnswer = true; 464 465 // Wait for some time before sending the next 0 466 ScheduleSync(); 467 } 468 469 // -------------------------------------------------------------------- 470 471 void HandleUpdateTimer(const bs::error_code &error) 205 472 { 206 473 if (error==ba::error::basic_errors::operation_aborted) 207 return; 474 { 475 Warn("Update timer aborted..."); 476 fIsRamping = false; 477 return; 478 } 208 479 209 480 if (error) 210 481 { 211 482 ostringstream str; 212 str << " Read timeout of " << URL() << ": " << error.message() << " (" << error << ")";// << endl;483 str << "Update timer: " << error.message() << " (" << error << ")";// << endl; 213 484 Error(str); 214 485 215 PostClose(); 486 PostClose(false); 487 return; 488 } 489 490 if (is_open()) 491 ReadAllChannels(true); 492 } 493 494 void ScheduleUpdate(int millisec) 495 { 496 fUpdateTimer.expires_from_now(boost::posix_time::milliseconds(millisec)); 497 fUpdateTimer.async_wait(boost::bind(&ConnectionBias::HandleUpdateTimer, this, dummy::error)); 498 } 499 500 // -------------------------------------------------------------------- 501 502 void SetAllChannels(const vector<uint16_t> &dac, bool special=false) 503 { 504 vector<char> data; 505 data.reserve(kNumChannels*3); 506 507 for (int ch=0; ch<kNumChannels; ch++) 508 { 509 const vector<char> cmd = GetCmd(kCmdChannelSet, ch, dac[ch]); 510 data.insert(data.end(), cmd.begin(), cmd.end()); 511 512 fVoltCmd[ch] = dac[ch]; 513 } 514 515 fSendCounter += kNumChannels; 516 517 PostMessage(data); 518 AsyncRead(ba::buffer(fBuffer, kNumChannels*3), 519 special ? kResetChannels : kCmdChannelSet, fSendCounter); 520 // fWaitingForAnswer = true; 521 } 522 523 uint16_t RampOneStep(uint16_t ch) 524 { 525 if (fVoltRef[ch]>fVolt[ch]) 526 return fVolt[ch]+fRampStep>fVoltRef[ch] ? fVoltRef[ch] : fVolt[ch]+fRampStep; 527 528 if (fVoltRef[ch]<fVolt[ch]) 529 return fVolt[ch]-fRampStep<fVoltRef[ch] ? fVoltRef[ch] : fVolt[ch]-fRampStep; 530 531 return fVolt[ch]; 532 } 533 534 bool RampOneStep() 535 { 536 vector<uint16_t> dac(kNumChannels); 537 538 bool identical = true; 539 for (int ch=0; ch<kNumChannels; ch++) 540 { 541 dac[ch] = RampOneStep(ch); 542 if (dac[ch]!=fVolt[ch]) 543 identical = false; 544 } 545 546 SetAllChannels(dac); 547 548 if (identical) 549 Info("Ramping: target values reached."); 550 551 return !identical; 552 } 553 554 void HandleRampTimer(const bs::error_code &error) 555 { 556 if (error==ba::error::basic_errors::operation_aborted) 557 { 558 Warn("Ramping aborted..."); 559 fIsRamping = false; 560 return; 561 } 562 563 if (error) 564 { 565 ostringstream str; 566 str << "Ramping timer: " << error.message() << " (" << error << ")";// << endl; 567 Error(str); 568 569 PostClose(false); 216 570 return; 217 571 } … … 219 573 if (!is_open()) 220 574 { 221 // For example: Here we could schedule a new accept if we 222 // would not want to allow two connections at the same time. 223 return; 224 } 225 226 // Check whether the deadline has passed. We compare the deadline 227 // against the current time since a new asynchronous operation 228 // may have moved the deadline before this actor had a chance 229 // to run. 230 if (fInTimeout.expires_at() > ba::deadline_timer::traits_type::now()) 231 return; 232 233 Error("Timeout reading data from "+URL()); 234 235 PostClose(); 236 } 237 238 vector<char> GetCmd(uint16_t id, Command_t cmd, uint16_t dac=0) 239 { 240 const unsigned int board = id/kNumChannelsPerBoard; 241 const unsigned int channel = id%kNumChannelsPerBoard; 242 243 return GetCmd(board, channel, cmd, dac); 244 } 245 246 vector<char> GetCmd(uint16_t board, uint16_t channel, Command_t cmd, uint16_t dac=0) 247 { 248 vector<char> data(3); 249 250 /* 251 if (board>kNumBoards) 252 return; 253 if (channel>kNumChannelsPerBoard) 254 return; 255 if (dac>0xfff) 256 return; 257 */ 258 259 data[0] = (cmd<<5) | (board<<1) | (((channel&16)>>4) & 1); 260 data[1] = (channel<<4) | (dac>>8); 261 data[2] = dac&0xff; 262 263 return data; 264 } 265 266 /* 267 void SetChannels(const map<uint16_t, uint16_t> &vals) 268 { 269 if (vals.empty()) 270 return; 271 272 // Build and execute commands 273 for (map<uint16_t, uint16_t>::const_iterator it=vals.begin(); 274 it!=vals.end(); it++) 275 { 276 // If DAC value unchanged, do not send command 277 if (fVolt[it->first] == it->second) 278 continue; 279 280 const vector<char> cmd = GetCmd(it->first, kCmdChannelSet, it->second); 281 PostMessage(cmd); 282 AsyncRead(ba::buffer(fBuffer), kCmdChannelSet|(it->first<<4)); 283 } 284 }*/ 285 286 /* 287 // ***** Synchronize board ***** 288 bool Crate::Synch() 289 { 290 //############################################################ 291 int Trial = 0; 292 vector<unsigned char> Data; 293 294 while(++Trial <= 3) { 295 Data = Communicate(string(1, 0)); 296 if (Data.size() == 3) return true; 297 } 298 return false; 299 //############################################################ 300 } 301 */ 575 Warn("Ramping in progress, but disconnected."); 576 return; 577 } 578 579 if (!fIsRamping) 580 { 581 Error("Ramp handler called although no ramping in progress."); 582 return; 583 } 584 585 fIsRamping = RampOneStep(); 586 } 587 588 void ScheduleRampStep() 589 { 590 fRampTimer.expires_from_now(boost::posix_time::milliseconds(fRampTime)); 591 fRampTimer.async_wait(boost::bind(&ConnectionBias::HandleRampTimer, this, dummy::error)); 592 } 302 593 303 594 public: 304 595 ConnectionBias(ba::io_service& ioservice, MessageImp &imp) : ConnectionUSB(ioservice, imp()), 596 fSyncTimer(ioservice), 597 fRampTimer(ioservice), 598 fUpdateTimer(ioservice), 599 fBuffer(3*kNumChannels), 305 600 fIsVerbose(true), 601 fVoltCmd(kNumChannels), 602 fVoltGapd(kNumChannels), 603 //fRefCurrent(kNumChannels), 604 fPresent(kNumBoards), 605 fRampStep(-1), 606 fRampTime(-1), 607 fUpdateTime(3000), 608 fIsRamping(false), 306 609 fVolt(kNumChannels), 307 fRefVolt(kNumChannels), 308 fCurrent(kNumChannels), 309 fRefCurrent(kNumChannels), 310 fPresent(kNumBoards) 610 fVoltRef(kNumChannels), 611 fCurrent(kNumChannels) 311 612 { 312 613 SetLogStream(&imp); 313 614 } 314 615 315 void SystemReset() 316 { 317 Message("Sending system reset."); 318 PostMessage(GetCmd(0, kCmdReset)); 319 AsyncRead(ba::buffer(fBuffer), kCmdReset); 320 } 321 322 void ReadChannel(int ch) 323 { 324 PostMessage(GetCmd(ch, kCmdRead)); 325 AsyncRead(ba::buffer(fBuffer), kCmdRead|(ch<<4)); 326 } 327 328 329 void ReadAllChannels() 330 { 331 Message("Requesting full system status."); 332 333 // Prepare command to read all channels 334 for (int i=0; i<kNumChannels; i++) 335 ReadChannel(i); 336 337 //vector<char> buf; 338 //AsyncRead(ba::buffer(buf), kCmdPrint); 616 void OverCurrentReset() 617 { 618 if (fIsRamping) 619 { 620 Warn("OverCurrentReset - Ramping in progres."); 621 RampStop(); 622 } 623 624 vector<uint16_t> dac(kNumChannels); 625 626 for (int ch=0; ch<kNumChannels; ch++) 627 dac[ch] = fCurrent[ch]<0 ? 0 : fVolt[ch]; 628 629 SetAllChannels(dac, true); 630 } 631 632 void ReadAllChannels(bool special = false) 633 { 634 vector<char> data; 635 data.reserve(kNumChannels*3); 636 637 for (int ch=0; ch<kNumChannels; ch++) 638 { 639 const vector<char> cmd = GetCmd(kCmdRead, ch); 640 data.insert(data.end(), cmd.begin(), cmd.end()); 641 } 642 643 fSendCounter += kNumChannels; 644 645 PostMessage(data); 646 AsyncRead(ba::buffer(fBuffer, kNumChannels*3), 647 special ? kUpdate : kCmdRead, fSendCounter); 648 // fWaitingForAnswer = true; 649 } 650 651 // -------------------------------------------------------------------- 652 653 bool ChannelSetDac(uint16_t ch, uint16_t dac) 654 { 655 if (!CheckChDac("ChannelSetDac", dac, ch)) 656 return false; 657 658 fVoltRef[ch] = dac; 659 660 if (!fIsRamping) 661 fIsRamping = RampOneStep(); 662 663 return true; 664 } 665 666 bool ChannelSetVolt(uint16_t ch, double volt) 667 { 668 return ChannelSetDac(ch, volt*4096/90.); 339 669 } 340 670 341 671 bool GlobalSetDac(uint16_t dac) 342 672 { 343 if (dac>0xfff) 344 return false; 345 346 PostMessage(GetCmd(0, kCmdGlobalSet, dac)); 347 AsyncRead(ba::buffer(fBuffer), kCmdGlobalSet); 348 349 ReadAllChannels(); 350 351 fVolt.assign(kNumChannels, dac); 352 UpdateV(); 673 if (!CheckChDac("GlobalSetDac", dac)) 674 return false; 675 676 for (size_t ch=0; ch<kNumChannels; ch++) 677 fVoltRef[ch] = dac; 678 679 if (!fIsRamping) 680 fIsRamping = RampOneStep(); 353 681 354 682 return true; 355 683 } 356 684 357 bool GlobalSet(double voltage) 358 { 359 return GlobalSetDac(voltage*4096/90); 360 } 361 362 bool ChannelSetDac(uint16_t ch, uint16_t dac) 363 { 364 if (dac>0xfff) 365 return false; 366 367 if (ch>=kNumChannels) 368 return false; 369 370 if (fVolt[ch]==dac) 371 return true; 372 373 PostMessage(GetCmd(ch, kCmdChannelSet, dac)); 374 AsyncRead(ba::buffer(fBuffer), kCmdChannelSet|(ch<<4)); 375 376 ReadChannel(ch); 377 378 fVolt[ch] = dac; 379 UpdateV(); 685 bool GlobalSetVolt(float volt) 686 { 687 return GlobalSetDac(volt*4096/90); 688 } 689 690 // -------------------------------------------------------------------- 691 692 bool SetGapdVoltage() 693 { 694 for (size_t ch=0; ch<kNumChannels; ch++) 695 if (fVoltGapd[ch]>kMaxDac) 696 { 697 Error("SetGapdVoltage - Voltage reference for G-APD channel out of range."); 698 return false; 699 } 700 701 for (size_t ch=0; ch<kNumChannels; ch++) 702 fVoltRef[ch] = fVoltGapd[ch]; 703 704 if (!fIsRamping) 705 fIsRamping = RampOneStep(); 380 706 381 707 return true; 382 708 } 383 709 384 bool ChannelSet(uint16_t ch, double voltage) 385 { 386 return ChannelSetDac(ch, voltage*4096/90); 387 } 388 389 void SetVoltage(int ch, int32_t dac) 390 { 391 if (dac<0) 392 dac = 0; 393 394 if (dac>0xfff) 395 dac = 0xfff; 396 397 ChannelSetDac(ch, dac); 398 } 399 400 // ***** Correct voltages according to current ***** 401 void AdaptVoltages() 402 { 403 for (int i=0; i<kNumChannels; i++) 404 { 405 if (fRefVolt[i]==0 || fCurrent[i]<0 || fRefCurrent[i]<0) 406 continue; 407 408 // Calculate difference and convert ADC units to Amp 409 // const double diffcur = (fRefCurrent[i]-fCurrent[i])*5000/4096 410 //const int32_t diffcur = int32_t(fRefCurrent[i]-fCurrent[i])*5000; 411 412 // Calculate voltage difference 413 // #define RESISTOR 1000 // Ohm 414 //const double diffvolt = diffcur*RESISTOR/1e6; 415 416 // Calculate new vlaue by onverting voltage difference to DAC units 417 //const int32_t dac = fRefVolt[i] + diffvolt*4096/90.0; 418 SetVoltage(i, fRefVolt[i] + (fRefCurrent[i]-fCurrent[i])/18); 419 } 420 } 421 422 void SetReferenceCurrent() 423 { 424 fRefCurrent = fCurrent; 425 } 426 427 bool SetReferenceVoltage(const vector<float> &volt) 710 void SetZero() 711 { 712 for (size_t ch=0; ch<kNumChannels; ch++) 713 fVoltRef[ch] = 0; 714 715 if (!fIsRamping) 716 fIsRamping = RampOneStep(); 717 } 718 719 bool SetNewGapdVoltage(const vector<float> &volt) 428 720 { 429 721 if (volt.size()!=kNumChannels) 430 722 { 431 723 ostringstream out; 432 out << "Set ReferenceVoltage - Given vector has " << volt.size() << " elements - expected " << kNumChannels << endl;724 out << "SetNewGapdVoltage - Given vector has " << volt.size() << " elements - expected " << kNumChannels << endl; 433 725 Error(out); 434 726 return false; 435 727 } 436 728 437 for (size_t i=0; i< volt.size(); i++)438 f RefVolt[i] = volt[i]*4096/90;729 for (size_t i=0; i<kNumChannels; i++) 730 fVoltGapd[i] = volt[i]*4096/90; 439 731 440 732 return true; 441 733 } 442 734 443 void ApplyReferenceVoltage() 444 { 445 for (size_t i=0; i<fRefVolt.size(); i++) 446 SetVoltage(i, fRefVolt[i]); 447 } 735 // -------------------------------------------------------------------- 736 737 void RampStop() 738 { 739 fRampTimer.cancel(); 740 fIsRamping = false; 741 742 Message("Ramping stopped."); 743 } 744 745 void RampStart() 746 { 747 if (fIsRamping) 748 { 749 Warn("RampStart - Ramping in progress... ignored."); 750 return; 751 } 752 753 fIsRamping = RampOneStep(); 754 } 755 756 void SetRampTime(uint16_t val) 757 { 758 fRampTime = val; 759 } 760 761 void SetRampStep(uint16_t val) 762 { 763 fRampStep = val; 764 } 765 766 bool IsRamping() const { return fIsRamping; } 767 768 // ------------------------------------------------------------------- 769 770 void ExpertReset() 771 { 772 Warn("EXPERT MODE: Sending reset."); 773 PostMessage(GetCmd(kCmdReset)); 774 AsyncRead(ba::buffer(fBuffer, 3), kCmdReset); 775 // fWaitingForAnswer = true; 776 } 777 778 779 bool ExpertChannelSetDac(uint16_t ch, uint16_t dac) 780 { 781 if (!CheckChDac("ExpertChannelSetDac", dac, ch)) 782 return false; 783 784 fVoltCmd[ch] = dac; 785 786 // FIXME: How to ensure the correct evaluation of the answer? 787 788 ostringstream msg; 789 msg << "EXPERT MODE: Sending 'ChannelSet' (set ch " << ch << " to DAC=" << dac << ")"; 790 Warn(msg); 791 792 PostMessage(GetCmd(kCmdChannelSet, ch, dac)); 793 AsyncRead(ba::buffer(fBuffer, 3), kExpertChannelSet|(ch<<8), ++fSendCounter); 794 // fWaitingForAnswer = true; 795 796 return true; 797 } 798 799 bool ExpertChannelSetVolt(uint16_t ch, double volt) 800 { 801 return ExpertChannelSetDac(ch, volt*4096/90.); 802 } 803 804 bool ExpertGlobalSetDac(uint16_t dac) 805 { 806 if (!CheckChDac("ExpertGlobalSetDac", dac)) 807 return false; 808 809 if (fGlobalVoltCmd>=0) 810 { 811 Error("ExpertGlobalSetDac - Still waiting for previous global-set's answer."); 812 return false; 813 } 814 815 fGlobalVoltCmd = dac; 816 817 ostringstream msg; 818 msg << "EXPERT MODE: Sending 'GlobalSet' (DAC=" << dac << ")"; 819 Warn(msg); 820 821 PostMessage(GetCmd(kCmdGlobalSet, 0, dac)); 822 AsyncRead(ba::buffer(fBuffer, 3), kCmdGlobalSet, ++fSendCounter); 823 // fWaitingForAnswer = true; 824 825 return true; 826 } 827 828 bool ExpertGlobalSetVolt(float volt) 829 { 830 return GlobalSetDac(volt*4096/90); 831 } 832 833 // -------------------------------------------------------------------- 448 834 449 835 void SetVerbose(bool b) … … 468 854 469 855 } 856 470 857 void Print() 471 858 { … … 485 872 } 486 873 } 874 875 // ------------------------------------------------------------------- 876 /* 877 void AdaptVoltages() 878 { 879 // Correct voltages according to current 880 for (int i=0; i<kNumChannels; i++) 881 { 882 if (fVoltRef[i]==0 || fCurrent[i]<0 || fRefCurrent[i]<0) 883 continue; 884 885 // Calculate difference and convert ADC units to Amp 886 // const double diffcur = (fRefCurrent[i]-fCurrent[i])*5000/4096 887 //const int32_t diffcur = int32_t(fRefCurrent[i]-fCurrent[i])*5000; 888 889 // Calculate voltage difference 890 // #define RESISTOR 1000 // Ohm 891 //const double diffvolt = diffcur*RESISTOR/1e6; 892 893 // Calculate new vlaue by onverting voltage difference to DAC units 894 //const int32_t dac = fRefVolt[i] + diffvolt*4096/90.0; 895 SetVoltage(i, fRefVolt[i] + (fRefCurrent[i]-fCurrent[i])/18); 896 } 897 } 898 899 void SetReferenceCurrent() 900 { 901 fRefCurrent = fCurrent; 902 } 903 */ 904 905 enum States_t 906 { 907 kDisconnected = StateMachineImp::kSM_UserMode, 908 kConnecting, 909 kInitializing, 910 kConnected, 911 kRamping, 912 kExpertMode // 'forward' declaration to be used in StateMachineBias 913 }; 914 915 int GetStatus() 916 { 917 if (!IsConnected()) 918 return kDisconnected; 919 920 if (IsConnecting()) 921 return kConnecting; 922 923 if (fIsInitializing) 924 return kInitializing; 925 926 if (fIsRamping) 927 return kRamping; 928 929 return kConnected; 930 } 487 931 }; 488 932 … … 505 949 void UpdateV() 506 950 { 507 fDimVoltage.Update(fVolt); 508 } 951 vector<uint16_t> vec; 952 vec.insert(vec.end(), fVolt.begin(), fVolt.end()); 953 vec.insert(vec.end(), fVoltRef.begin(), fVoltRef.end()); 954 fDimVoltage.Update(vec); 955 } 956 509 957 510 958 public: … … 512 960 ConnectionBias(ioservice, imp), 513 961 fDimCurrent("BIAS_CONTROL/CURRENT", "S:416", ""), 514 fDimVoltage("BIAS_CONTROL/VOLTAGE", "S:416 ", "")962 fDimVoltage("BIAS_CONTROL/VOLTAGE", "S:416;S:416", "") 515 963 { 516 964 } … … 549 997 S fBias; 550 998 551 enum states_t 552 { 553 kStateDisconnected = 1, 554 kStateConnected = 2, 555 }; 556 557 int SetGlobal(const EventImp &evt) 558 { 559 if (!CheckEventSize(evt.GetSize(), "SetGlobal", 4)) 560 return false; 561 562 if (!fBias.GlobalSet(evt.GetFloat())) 999 bool fExpertMode; 1000 1001 // -------------------------------------------------------------------- 1002 1003 int SetGlobalVolt(const EventImp &evt) 1004 { 1005 if (!CheckEventSize(evt.GetSize(), "SetGlobalVolt", 4)) 1006 return false; 1007 1008 if (!fBias.GlobalSetVolt(evt.GetFloat())) 563 1009 T::Error("Supplied voltage out of range (0-90)"); 564 1010 … … 577 1023 } 578 1024 579 int SetChannel (const EventImp &evt)580 { 581 if (!CheckEventSize(evt.GetSize(), "SetChannel ", 6))582 return false; 583 584 if (!fBias.ChannelSet (evt.GetUShort(), evt.Get<float>(2)))1025 int SetChannelVolt(const EventImp &evt) 1026 { 1027 if (!CheckEventSize(evt.GetSize(), "SetChannelVolt", 6)) 1028 return false; 1029 1030 if (!fBias.ChannelSetVolt(evt.GetUShort(), evt.Get<float>(2))) 585 1031 T::Error("Value out of range"); 586 1032 … … 599 1045 } 600 1046 1047 // -------------------------------------------------------------------- 1048 1049 int ExpertSetGlobalVolt(const EventImp &evt) 1050 { 1051 if (!CheckEventSize(evt.GetSize(), "ExpertSetGlobalVolt", 4)) 1052 return false; 1053 1054 if (!fBias.ExpertGlobalSetVolt(evt.GetFloat())) 1055 T::Error("Supplied voltage out of range (0-90)"); 1056 1057 return T::GetCurrentState(); 1058 } 1059 1060 int ExpertSetGlobalDac(const EventImp &evt) 1061 { 1062 if (!CheckEventSize(evt.GetSize(), "ExpertSetGlobalDac", 2)) 1063 return false; 1064 1065 if (!fBias.ExpertGlobalSetDac(evt.GetUShort())) 1066 T::Error("Supplied voltage out of range (0-90)"); 1067 1068 return T::GetCurrentState(); 1069 } 1070 1071 int ExpertSetChannelVolt(const EventImp &evt) 1072 { 1073 if (!CheckEventSize(evt.GetSize(), "ExpertSetChannelVolt", 6)) 1074 return false; 1075 1076 if (!fBias.ExpertChannelSetVolt(evt.GetUShort(), evt.Get<float>(2))) 1077 T::Error("Value out of range"); 1078 1079 return T::GetCurrentState(); 1080 } 1081 1082 int ExpertSetChannelDac(const EventImp &evt) 1083 { 1084 if (!CheckEventSize(evt.GetSize(), "ExpertSetChannelDac", 4)) 1085 return false; 1086 1087 if (!fBias.ExpertChannelSetDac(evt.Get<uint16_t>(), evt.Get<uint16_t>(2))) 1088 T::Error("Value out of range"); 1089 1090 return T::GetCurrentState(); 1091 } 1092 1093 // -------------------------------------------------------------------- 601 1094 602 1095 int Disconnect() … … 628 1121 // Now we can reopen the connection 629 1122 fBias.PostClose(true); 1123 1124 return T::GetCurrentState(); 1125 } 1126 1127 int SetVerbosity(const EventImp &evt) 1128 { 1129 if (!CheckEventSize(evt.GetSize(), "SetVerbosity", 1)) 1130 return T::kSM_FatalError; 1131 1132 fBias.SetVerbose(evt.GetBool()); 1133 1134 return T::GetCurrentState(); 1135 } 1136 1137 int SetExpertMode(const EventImp &evt) 1138 { 1139 if (!CheckEventSize(evt.GetSize(), "SetExpertMode", 1)) 1140 return T::kSM_FatalError; 1141 1142 fExpertMode = evt.GetBool(); 1143 1144 if (fExpertMode) 1145 T::Warn("Expert commands enabled -- please ensure that you EXACTLY know what you do. These commands can destroy the system."); 630 1146 631 1147 return T::GetCurrentState(); … … 641 1157 poll_one(); 642 1158 643 return fBias.IsConnected() ? kStateConnected : kStateDisconnected; 644 } 645 646 int SetVerbosity(const EventImp &evt) 647 { 648 if (!CheckEventSize(evt.GetSize(), "SetVerbosity", 1)) 649 return T::kSM_FatalError; 650 651 fBias.SetVerbose(evt.GetBool()); 652 653 return T::GetCurrentState(); 1159 return fExpertMode && fBias.GetStatus()==ConnectionBias::kConnected ? 1160 ConnectionBias::kExpertMode : fBias.GetStatus(); 654 1161 } 655 1162 … … 657 1164 StateMachineBias(ostream &out=cout) : 658 1165 T(out, "BIAS_CONTROL"), ba::io_service::work(static_cast<ba::io_service&>(*this)), 659 fBias(*this, *this) 1166 fBias(*this, *this), fExpertMode(false) 660 1167 { 661 1168 // ba::io_service::work is a kind of keep_alive for the loop. … … 666 1173 // deletion and creation of threads and more. 667 1174 1175 T::Warn("FIXME -- implement a state for 'at reference'"); 1176 668 1177 // State names 669 AddStateName(kStateDisconnected, "Disconnected", 670 "Bias-power supply not connected via USB."); 671 672 AddStateName(kStateConnected, "Connected", 673 "USB connection to bias-power supply established."); 1178 T::AddStateName(ConnectionBias::kDisconnected, "Disconnected", 1179 "Bias-power supply not connected via USB."); 1180 1181 T::AddStateName(ConnectionBias::kConnecting, "Connecting", 1182 "Trying to establish USB connection to bias-power supply."); 1183 1184 T::AddStateName(ConnectionBias::kInitializing, "Initializing", 1185 "USB connection to bias-power supply established, synchronizing USB stream."); 1186 1187 T::AddStateName(ConnectionBias::kConnected, "Connected", 1188 "USB connection to bias-power supply established."); 1189 1190 T::AddStateName(ConnectionBias::kExpertMode, "ExpertMode", 1191 "Special (risky!) mode to directly send command to the bias-power supply."); 1192 1193 T::AddStateName(ConnectionBias::kRamping, "Ramping", 1194 "Voltage ramping in progress."); 674 1195 675 1196 // Verbosity commands 676 1197 T::AddEvent("SET_VERBOSE", "B") 677 (bind(&StateMachineBias::SetVerbosity, this, _1))1198 (bind(&StateMachineBias::SetVerbosity, this, placeholders::_1)) 678 1199 ("set verbosity state" 679 1200 "|verbosity[bool]:disable or enable verbosity for received data (yes/no), except dynamic data"); 680 1201 681 1202 // Conenction commands 682 AddEvent("DISCONNECT", kStateConnected)1203 T::AddEvent("DISCONNECT", ConnectionBias::kConnected) 683 1204 (bind(&StateMachineBias::Disconnect, this)) 684 1205 ("disconnect from ethernet"); 685 1206 686 AddEvent("RECONNECT", "O", kStateDisconnected, kStateConnected)687 (bind(&StateMachineBias::Reconnect, this, _1))1207 T::AddEvent("RECONNECT", "O", ConnectionBias::kDisconnected, ConnectionBias::kConnected) 1208 (bind(&StateMachineBias::Reconnect, this, placeholders::_1)) 688 1209 ("(Re)connect ethernet connection to FTM, a new address can be given" 689 1210 "|[host][string]:new ethernet address in the form <host:port>"); 690 1211 691 1212 692 AddEvent("REQUEST_STATUS", kStateConnected) 693 (Wrapper(bind(&ConnectionBias::ReadAllChannels, &fBias))) 694 (""); 695 696 AddEvent("SET_GLOBAL_VOLTAGE", "F:1", kStateConnected) 697 (bind(&StateMachineBias::SetGlobal, this, _1)) 698 (""); 699 700 AddEvent("SET_GLOBAL_DAC", "S:1", kStateConnected) 701 (bind(&StateMachineBias::SetGlobalDac, this, _1)) 702 (""); 703 704 AddEvent("SET_CHANNEL_VOLTAGE", "S:1;F:1", kStateConnected) 705 (bind(&StateMachineBias::SetChannel, this, _1)) 706 (""); 707 708 AddEvent("SET_CHANNEL_DAC", "S:1;S:1", kStateConnected) 709 (bind(&StateMachineBias::SetChannelDac, this, _1)) 710 (""); 711 712 AddEvent("RESET", kStateConnected) 713 (Wrapper(bind(&ConnectionBias::SystemReset, &fBias))) 714 (""); 715 716 AddEvent("SET_REFERENCE_CURRENT", kStateConnected) 717 (Wrapper(bind(&ConnectionBias::SetReferenceCurrent, &fBias))) 718 (""); 719 720 AddEvent("APPLY_REFERENCE_VOLTAGE", kStateConnected) 721 (Wrapper(bind(&ConnectionBias::ApplyReferenceVoltage, &fBias))) 722 (""); 723 724 AddEvent("ADAPT_VOLTAGES", kStateConnected) 725 (Wrapper(bind(&ConnectionBias::AdaptVoltages, &fBias))) 726 (""); 727 728 AddEvent("PRINT", kStateConnected) 1213 1214 T::AddEvent("REQUEST_STATUS", ConnectionBias::kConnected, ConnectionBias::kRamping) 1215 (Wrapper(bind(&ConnectionBias::ReadAllChannels, &fBias, false))) 1216 (""); 1217 1218 T::AddEvent("RESET_OVER_CURRENT_STATUS", ConnectionBias::kConnected) 1219 (Wrapper(bind(&ConnectionBias::OverCurrentReset, &fBias))) 1220 (""); 1221 1222 1223 1224 T::AddEvent("SET_GLOBAL_VOLTAGE", "F:1", ConnectionBias::kConnected, ConnectionBias::kRamping) 1225 (bind(&StateMachineBias::SetGlobalVolt, this, placeholders::_1)) 1226 (""); 1227 1228 T::AddEvent("SET_GLOBAL_DAC", "S:1", ConnectionBias::kConnected, ConnectionBias::kRamping) 1229 (bind(&StateMachineBias::SetGlobalDac, this, placeholders::_1)) 1230 (""); 1231 1232 T::AddEvent("SET_CHANNEL_VOLTAGE", "S:1;F:1", ConnectionBias::kConnected, ConnectionBias::kRamping) 1233 (bind(&StateMachineBias::SetChannelVolt, this, placeholders::_1)) 1234 (""); 1235 1236 T::AddEvent("SET_CHANNEL_DAC", "S:1;S:1", ConnectionBias::kConnected, ConnectionBias::kRamping) 1237 (bind(&StateMachineBias::SetChannelDac, this, placeholders::_1)) 1238 (""); 1239 1240 T::AddEvent("SET_GAPD_REFERENCE_VOLTAGE", ConnectionBias::kConnected, ConnectionBias::kRamping) 1241 (Wrapper(bind(&ConnectionBias::SetGapdVoltage, &fBias))) 1242 (""); 1243 1244 T::AddEvent("SET_ZERO_VOLTAGE", ConnectionBias::kConnected, ConnectionBias::kRamping) 1245 (Wrapper(bind(&ConnectionBias::SetZero, &fBias))) 1246 (""); 1247 1248 1249 1250 T::AddEvent("STOP", ConnectionBias::kConnected, ConnectionBias::kRamping) 1251 (Wrapper(bind(&ConnectionBias::RampStop, &fBias))) 1252 (""); 1253 1254 T::AddEvent("START", ConnectionBias::kConnected) 1255 (Wrapper(bind(&ConnectionBias::RampStart, &fBias))) 1256 (""); 1257 1258 1259 1260 T::AddEvent("PRINT", ConnectionBias::kConnected, ConnectionBias::kRamping) 729 1261 (Wrapper(bind(&ConnectionBias::Print, &fBias))) 1262 (""); 1263 1264 1265 T::AddEvent("EXPERT_MODE", "B:1") 1266 (bind(&StateMachineBias::SetExpertMode, this, placeholders::_1)) 1267 (""); 1268 1269 T::AddEvent("EXPERT_RESET", ConnectionBias::kExpertMode) 1270 (Wrapper(bind(&ConnectionBias::ExpertReset, &fBias))) 1271 (""); 1272 1273 T::AddEvent("EXPERT_SET_GLOBAL_VOLTAGE", "F:1", ConnectionBias::kExpertMode) 1274 (bind(&StateMachineBias::ExpertSetGlobalVolt, this, placeholders::_1)) 1275 (""); 1276 1277 T::AddEvent("EXPERT_SET_GLOBAL_DAC", "S:1", ConnectionBias::kExpertMode) 1278 (bind(&StateMachineBias::ExpertSetGlobalDac, this, placeholders::_1)) 1279 (""); 1280 1281 T::AddEvent("EXPERT_SET_CHANNEL_VOLTAGE", "S:1;F:1", ConnectionBias::kExpertMode) 1282 (bind(&StateMachineBias::ExpertSetChannelVolt, this, placeholders::_1)) 1283 (""); 1284 1285 T::AddEvent("EXPERT_SET_CHANNEL_DAC", "S:1;S:1", ConnectionBias::kExpertMode) 1286 (bind(&StateMachineBias::ExpertSetChannelDac, this, placeholders::_1)) 730 1287 (""); 731 1288 } … … 737 1294 fBias.SetEndpoint(conf.Get<string>("dev")); 738 1295 T::Message("Setting device to "+fBias.URL()); 1296 1297 const uint16_t step = conf.Get<uint16_t>("ramp-step"); 1298 const uint16_t time = conf.Get<uint16_t>("ramp-time"); 1299 1300 if (step>230) // 5V 1301 { 1302 T::Error("ramp-step exceeds allowed range."); 1303 return 1; 1304 } 1305 if (time>10000) // 5V 1306 { 1307 T::Error("ramp-time exceeds allowed range."); 1308 return 2; 1309 } 1310 1311 fBias.SetRampStep(step); 1312 fBias.SetRampTime(time); 739 1313 740 1314 // -------------------------------------------------------------------------- … … 770 1344 { 771 1345 T::Error("Invalid board/channel read from FACTmapV5.txt."); 772 return 1;1346 return 3; 773 1347 } 774 1348 … … 780 1354 { 781 1355 T::Error("Reading reference voltages from FACTmapV5.txt failed."); 782 return 2;783 } 784 785 if (!fBias.Set ReferenceVoltage(vec))1356 return 4; 1357 } 1358 1359 if (!fBias.SetNewGapdVoltage(vec)) 786 1360 { 787 1361 T::Error("Setting reference voltages failed."); 788 return 3;1362 return 5; 789 1363 } 790 1364 … … 813 1387 ("no-dim,d", po_bool(), "Disable dim services") 814 1388 ("dev", var<string>("FTE00FOH"), "Device address of USB port to bias-power supply") 815 ("quiet,q", po_bool(), "Disable printing contents of all received messages (except dynamic data) in clear text.") 1389 ("quiet,q", po_bool(), "Disable printing contents of all received messages (except dynamic data) in clear text.") 1390 ("ramp-time", var<uint16_t>(), "") 1391 ("ramp-step", var<uint16_t>(), "") 1392 ("ramp-volt", var<float>(), "") 816 1393 ; 817 1394
Note:
See TracChangeset
for help on using the changeset viewer.