source: trunk/MagicSoft/Cosy/devdrv/macs.cc@ 1705

Last change on this file since 1705 was 1703, checked in by tbretz, 22 years ago
*** empty log message ***
File size: 19.8 KB
Line 
1#include "macs.h"
2
3#include <iostream.h>
4#include <sys/time.h> // timeval->tv_sec
5
6#include "timer.h"
7#include "network.h"
8#include "MLogManip.h"
9
10ClassImp(Macs);
11
12/*
13 ---------------------------
14 For test purposes
15 ---------------------------
16
17class MyTimer : public TTimer
18{
19public:
20 MyTimer(TObject *obj, Long_t ms, Bool_t mode) : TTimer(obj, ms, mode) {}
21 Bool_t Notify()
22 {
23 cout << "Notify" << endl;
24 TTimer::Notify();
25 return kTRUE;
26 }
27};
28*/
29
30Macs::Macs(const BYTE_t nodeid, const char *name, MLog &out)
31 : NodeDrv(nodeid, name, out), fMacId(2*nodeid+1),
32 fPos(0), fPosTime(0.0), fPdoPos(0), fPdoTime(0.0),
33 fPosActive(0), fRpmActive(0)
34{
35 fTimeout = new TTimer(this, 100, kFALSE); // 100ms, asynchronous
36}
37
38Macs::~Macs()
39{
40 fTimerOn = kFALSE;
41 delete fTimeout;
42}
43
44void Macs::HandleSDO(WORD_t idx, BYTE_t subidx, LWORD_t val, timeval_t *tv)
45{
46 switch (idx)
47 {
48 case 0x100a:
49 lout << "- " << GetNodeName() << ": Using Software Version V" << dec << (int)(val>>16) << "." << (int)(val&0xff) << endl;
50 fSoftVersion = val;
51 return;
52
53 case 0x100b:
54 // Do not display, this is used for CheckConnection
55 // lout << "Node ID: " << dec << val << endl;
56 return;
57
58 case 0x2002:
59 cout << GetNodeName() << ": Actual velocity: " << dec << val << endl;
60 fVel = val;
61 return;
62
63 case 0x4000:
64 switch (subidx)
65 {
66 case 1:
67 cout << GetNodeName() << ": Timeout timer is " << (val?"en":"dis") << "abled." << endl;
68 return;
69 case 2:
70 cout << GetNodeName() << ": Actual timeout time: " << dec << val << "ms" << endl;
71 return;
72 }
73 break;
74
75 case 0x6004:
76 if (subidx)
77 return;
78
79// lout << "Actual Position: " << dec << (signed long)val << endl;
80 fPos = (LWORDS_t)val;
81 fPosTime.SetTimer(tv);
82 return;
83/*
84 case 0x2001:
85 cout << "Axe Status: 0x" << hex << val << endl;
86 cout << " - Motor " << (val&0x01?"standing":"moving") << endl;
87 cout << " - Positioning " << (val&0x02?"active":"inactive") << endl;
88 cout << " - Rotary mode " << (val&0x04?"active":"inactive") << endl;
89 cout << " - Attitude control: " << (val&0x40?"off":"on") << endl;
90 cout << " - Axe resetted: " << (val&0x80?"yes":"no") << endl;
91 fPosActive = val&0x02;
92 fRpmActive = val&0x04;
93 return;
94 case 0x2003:
95 if (!subidx)
96 {
97 cout << "Input State: ";
98 for (int i=0; i<8; i++)
99 cout << (int)(val&(1<<i)?1:0);
100 cout <<endl;
101 return;
102 }
103 cout << "Input No." << subidx << (val?"hi":"lo") << endl;
104 return;
105 case 0x2004:
106 cout << "Status Value of Axis: 0x" << hex << val << endl;
107 cout << " - Attitude control: " << (val&0x800000?"off":"on") << endl;
108 cout << " - Movement done: " << (val&0x040000?"yes":"no") << endl;
109 cout << " - Number of Control Units: " << (int)((val>>8)&0xff) << endl;
110 cout << " - Attitude control: " << (val&0x04?"off":"on") << endl;
111 cout << " - Startswitch active: " << (val&0x20?"yes":"no") << endl;
112 cout << " - Referenceswitch active: " << (val&0x40?"yes":"no") << endl;
113 cout << " - Endswitch active: " << (val&0x80?"yes":"no") << endl;
114 return;
115*/
116
117 case 0x6002:
118 lout << "- " << GetNodeName() << ": Velocity resolution = " << dec << val << " ticks/min" << endl;
119 fVelRes = val;
120 return;
121
122 case 0x6501:
123 lout << "- " << GetNodeName() << ": Encoder resolution = " << dec << val << " ticks/min" << endl;
124 fRes = val;
125 return;
126 }
127 cout << "Macs: SDO, idx=0x"<< hex << idx << "/" << (int)subidx;
128 cout << ", val=0x"<<val<<endl;
129}
130
131void Macs::HandleSDOOK(WORD_t idx, BYTE_t subidx)
132{
133 switch (idx)
134 {
135 case 0x2002:
136 switch (subidx)
137 {
138 case 0:
139 lout << ddev(MLog::eGui);
140 lout << "- " << GetNodeName() << ": Velocity set." << endl;
141 lout << edev(MLog::eGui);
142 return;
143 }
144 break;
145 case 0x2003:
146 switch (subidx)
147 {
148 case 0:
149 lout << ddev(MLog::eGui);
150 lout << "- " << GetNodeName() << ": Acceleration set." << endl;
151 lout << edev(MLog::eGui);
152 return;
153 case 1:
154 lout << ddev(MLog::eGui);
155 lout << "- " << GetNodeName() << ": Decceleration set." << endl;
156 lout << edev(MLog::eGui);
157 return;
158 }
159 break;
160 case 0x3006:
161 switch (subidx)
162 {
163 case 0:
164 lout << ddev(MLog::eGui);
165 lout << "- " << GetNodeName() << ": RPM mode switched." << endl;
166 lout << edev(MLog::eGui);
167 return;
168
169 case 1:
170 /*
171 lout << ddev(MLog::eGui);
172 lout << "- Velocity set (" << GetNodeName() << ")" << endl;
173 lout << edev(MLog::eGui);
174 */
175 return;
176 }
177 break;
178 case 0x4000:
179 if (subidx==0 && fTimerOn)
180 {
181 ResetTimeout();
182 return;
183 }
184 break;
185 case 0x6004:
186 switch (subidx)
187 {
188 case 0:
189 lout << ddev(MLog::eGui);
190 lout << "- " << GetNodeName() << ": Absolute positioning started." << endl;
191 lout << edev(MLog::eGui);
192 return;
193
194 case 1:
195 lout << ddev(MLog::eGui);
196 lout << "- " << GetNodeName() << ": Relative positioning started." << endl;
197 lout << edev(MLog::eGui);
198 return;
199 }
200 break;
201
202
203 }
204 NodeDrv::HandleSDOOK(idx, subidx);
205}
206
207
208void Macs::ReqVelRes()
209{
210 lout << "- " << GetNodeName() << ": Requesting velocity resolution (velres, 0x6002)." << endl;
211 RequestSDO(0x6002);
212 WaitForSdo(0x6002);
213}
214
215void Macs::ReqRes()
216{
217 lout << "- " << GetNodeName() << ": Requesting encoder resolution (res, 0x6501)." << endl;
218 RequestSDO(0x6501);
219 WaitForSdo(0x6501);
220}
221
222void Macs::SetPDO1On(BYTE_t flag)
223{
224 lout << "- " << GetNodeName() << ": " << (flag?"Enable":"Disable") << " PDO1." << endl;
225 SendSDO(0x1800, 1, (LWORD_t)(flag?0:1)<<31);
226 WaitForSdo(0x1800, 1);
227}
228
229void Macs::CheckConnection()
230{
231 RequestSDO(0x100b);
232 WaitForSdo(0x100b);
233
234 // FIXME! Not statically linked!
235 // if (fSoftVersion<0x00000035)
236 // fIsZombie = true;
237}
238
239
240void Macs::Init()
241{
242 lout << "- " << GetNodeName() << ": Requesting Mac Software Version." << endl;
243 RequestSDO(0x100a);
244 WaitForSdo(0x100a);
245
246 if (fIsZombie)
247 {
248 lout << GetNodeName() << " - InitDevice failed!" << endl;
249 return;
250 }
251
252 EnableTimeout(kFALSE);
253 SetRpmMode(FALSE);
254
255 ReqRes(); // Init fRes
256 ReqVelRes(); // Init fVelRes
257
258 lout << "- " << GetNodeName() << ": Motor on." << endl;
259 SendSDO(0x3000, string('o', 'n'));
260 WaitForSdo(0x3000);
261
262// SetHome(250000);
263
264// lout << "- Requesting SDO 0x2001 of " << (int)GetId() << endl;
265// RequestSDO(0x2001);
266// WaitForSdo(0x2001);
267
268 SetPDO1On(FALSE); // this is a workaround for the Macs
269 SetPDO1On(TRUE);
270
271 SetNoWait(TRUE);
272}
273
274void Macs::StopMotor()
275{
276 //
277 // Stop the motor and switch off the position control unit
278 //
279 SendSDO(0x3000, string('s','t','o','p'));
280 WaitForSdo(0x3000);
281}
282
283void Macs::StopDevice()
284{
285 EnableTimeout(kFALSE);
286
287 SetNoWait(FALSE);
288
289 //
290 // FIXME: This isn't called if the initialization isn't done completely!
291 //
292
293 SetRpmMode(FALSE);
294
295 SetPDO1On(FALSE);
296
297 lout << "- " << GetNodeName() << ": Motor off." << endl;
298 SendSDO(0x3000, string('o', 'f', 'f'));
299 WaitForSdo(0x3000);
300
301 /*
302 lout << "- Stopping Program of " << (int)GetId() << endl;
303 SendSDO(0x4000, (LWORD_t)0xaffe);
304 WaitForSdo();
305 */
306}
307
308void Macs::ReqPos()
309{
310 lout << "- " << GetNodeName() << ": Requesting Position." << endl;
311 RequestSDO(0x6004);
312 WaitForSdo(0x6004);
313}
314
315void Macs::ReqVel()
316{
317 lout << "- " << GetNodeName() << ": Requesting Velocity." << endl;
318 RequestSDO(0x2002);
319 WaitForSdo(0x2002);
320}
321
322void Macs::SetHome(LWORDS_t pos, WORD_t maxtime)
323{
324 /*
325 Bool_t to = fTimerOn;
326
327 if (to)
328 EnableTimeout(kFALSE);
329 */
330 lout << "- " << GetNodeName() << ": Driving to home position, Offset=" << dec << pos << endl;
331 SendSDO(0x6003, 2, (LWORD_t)pos); // home
332 WaitForSdo(0x6003, 2);
333
334 // home also defines the zero point of the system
335 // maximum time allowd for home drive: 25.000ms
336 SendSDO(0x3001, string('h','o','m','e')); // home
337 WaitForSdo(0x3001, 0, maxtime*1000);
338 lout << "- " << GetNodeName() << ": Home position reached. " << endl;
339
340 SendSDO(0x6003, 0, string('s','e','t')); // home
341 WaitForSdo(0x6003, 0);
342
343 //if (to)
344 // EnableTimeout(kTRUE);
345}
346
347void Macs::SetVelocity(LWORD_t vel)
348{
349 SendSDO(0x2002, vel); // velocity
350 WaitForSdo(0x2002);
351}
352
353void Macs::SetAcceleration(LWORD_t acc)
354{
355 SendSDO(0x2003, 0, acc); // acceleration
356 WaitForSdo(0x2003, 0);
357}
358
359void Macs::SetDeceleration(LWORD_t dec)
360{
361 SendSDO(0x2003, 1, dec); // acceleration
362 WaitForSdo(0x2003, 1);
363}
364
365void Macs::SetRpmMode(BYTE_t mode)
366{
367 //
368 // SetRpmMode(FALSE) stop the motor, but lets the position control unit on
369 //
370 SendSDO(0x3006, 0, mode ? string('s','t','r','t') : string('s','t','o','p'));
371 WaitForSdo(0x3006, 0);
372}
373
374void Macs::SetRpmVelocity(LWORDS_t cvel)
375{
376 SendSDO(0x3006, 1, (LWORD_t)cvel);
377 WaitForSdo(0x3006, 1);
378}
379
380void Macs::StartRelPos(LWORDS_t pos)
381{
382 SendSDO(0x6004, 1, (LWORD_t)pos);
383}
384
385void Macs::StartAbsPos(LWORDS_t pos)
386{
387 SendSDO(0x6004, 0, (LWORD_t)pos);
388}
389
390void Macs::SetNoWait(BYTE_t flag)
391{
392 lout << "- " << GetNodeName() << ": Setting NOWAIT " << (flag?"ON":"OFF") << "." << endl;
393 SendSDO(0x3008, flag ? string('o', 'n') : string('o', 'f', 'f'));
394 WaitForSdo(0x3008);
395}
396
397void Macs::StartVelSync()
398{
399 //
400 // The syncronization mode is disabled by a 'MOTOR STOP'
401 // or by a positioning command (POSA, ...)
402 //
403 lout << "- " << GetNodeName() << ": Starting RPM Sync Mode." << endl;
404 SendSDO(0x3007, 0, string('s', 'y', 'n', 'c'));
405 WaitForSdo(0x3007, 0);
406}
407
408void Macs::StartPosSync()
409{
410 //
411 // The syncronization mode is disabled by a 'MOTOR STOP'
412 // or by a positioning command (POSA, ...)
413 //
414 lout << "- " << GetNodeName() << ": Starting Position Sync Mode." << endl;
415 SendSDO(0x3007, 1, string('s', 'y', 'n', 'c'));
416 WaitForSdo(0x3007, 1);
417}
418/*
419void Macs::ReqAxEnd()
420{
421 RequestSDO(0x2001);
422 WaitForSdo(0x2001);
423}
424*/
425void Macs::SendMsg(BYTE_t data[6])
426{
427 GetNetwork()->SendCanFrame(fMacId, 0, 0, data[0], data[1], data[2], data[3], data[4], data[5]);
428}
429
430void Macs::SendMsg(BYTE_t d0=0, BYTE_t d1=0, BYTE_t d2=0,
431 BYTE_t d3=0, BYTE_t d4=0, BYTE_t d5=0)
432{
433 GetNetwork()->SendCanFrame(fMacId, 0, 0, d0, d1, d2, d3, d4, d5);
434}
435
436void Macs::HandlePDO1(BYTE_t *data, timeval_t *tv)
437{
438 fPdoPos = (data[4]<<24) | (data[5]<<16) | (data[6]<<8) | data[7];
439
440 // data[3]&0x01; // motor not moving
441 fPosActive = data[3]&kPosActive; // positioning active
442 fRpmActive = data[3]&kRpmActive; // RPM mode switched on
443 // data[3]&0x08; // - unused -
444 // data[3]&0x10; // - unused -
445 // data[3]&0x20; // - unused -
446 fInControl = data[3]&0x40; // motor uncontrolled
447 // data[3]&0x80; // axis resetted (after errclr, motor stop, motor on)
448
449 fStatus = data[3];
450
451 fPdoTime.SetTimer(tv);
452}
453
454void Macs::HandlePDO2(BYTE_t *data, timeval_t *tv)
455{
456 LWORDS_t errnum = (data[0]<<24) | (data[1]<<16) | (data[2]<<8) | data[3];
457 LWORDS_t errinf = (data[4]<<24) | (data[5]<<16) | (data[6]<<8) | data[7];
458
459 //
460 // errnum==0 gives a sudden information that something happened. Now the
461 // microcontroller is running inside its interrupt procedure which
462 // stopped the normal program. The interrupt procedure should try to clear
463 // the error state of the hardware. This should never create a new error!
464 //
465 if (!errnum)
466 {
467 lout << "- " << GetNodeName() << ": reports Error occursion." << endl;
468 SetError(-1);
469 return;
470 }
471
472 //
473 // Now the error is handled by the hardware now it is the software part
474 // to react on it. The Error flag now is set to the correct value.
475 //
476 if (GetError()>0)
477 {
478 lout << GetNodeName() << ": WARNING! Error #" << GetError() << " unhandled (not cleared) by software." << endl;
479
480 //
481 // If the error is unhadled and/or not cleared, don't try it again.
482 //
483 if (GetError()==errnum)
484 return;
485 }
486
487 SetError(errnum);
488
489 lout << GetNodeName() << " reports: ";
490 switch (errnum)
491 {
492 case 6:
493 //
494 // Report the error to the user. All possible movements should have
495 // been stopped anyhow. Now delete the error to prevent the system
496 // from reporting this error a thousands of times.
497 //
498 lout << "Home position not the first positioning command." << endl;
499 SetError(0);
500 return;
501
502 case 8:
503 lout << "Control deviation overflow." << endl;
504 return;
505
506 case 9:
507 lout << "Zero index not found." << endl;
508 return;
509
510 case 11:
511 case 25:
512 switch (errinf)
513 {
514 case -1:
515 lout << "Negative";
516 break;
517 case 1:
518 lout << "Positive";
519 break;
520 default:
521 lout << "-unknown-";
522 }
523 switch (errnum)
524 {
525 case 11:
526 lout << " software endswitch activated." << endl;
527 break;
528 case 25:
529 lout << " hardware endswitch activated." << endl;
530 break;
531 }
532 return;
533
534 case 84:
535 lout << "Too many (>12) ON TIME calls." << endl;
536 return;
537
538 case 100:
539 lout << "Connection timed out." << endl;
540 EnableTimeout(false);
541 return;
542
543 default:
544 lout << "Error Nr. " << errnum << ", " << errinf << endl;
545 }
546}
547
548// FIXME? Handling of fIsZombie?
549void Macs::HandleError()
550{
551 //
552 // If there is no error we must not handle anything
553 //
554 if (!HasError())
555 return;
556
557 //
558 // If the program got into the: HandleError state before the hardware
559 // has finished handeling the error we have to wait for the hardware
560 // handeling the error
561 //
562 // FIXME: Timeout???
563 //
564 while (GetError()<0)
565 usleep(1);
566
567 //
568 // After this software and hardware should be in a state so that
569 // we can go on working 'as usual' Eg. Initialize a Display Update
570 //
571 cout << GetNodeName() << " Handling Error #" << GetError() << endl;
572 switch (GetError())
573 {
574 case 6: // home
575 case 8: // control dev
576 case 9: // zero idx
577 case 84: // ON TIME
578 // Stop program?
579 return;
580
581 case 11: // software endswitch
582 case 25: // hardware endswitch
583 case 100: // timeout (movement has been stopped, so we can go on)
584 DelError();
585 return;
586
587 case 101:
588 lout << "Warning: " << GetNodeName() << " didn't respond in timeout window - try again." << endl;
589 DelError();
590 return;
591 }
592}
593
594double Macs::GetTime()
595{
596 return fPosTime.Now();
597}
598
599double Macs::GetMjd()
600{
601 return fPosTime.CalcMjd();
602}
603
604double Macs::GetPdoTime()
605{
606 return fPdoTime.Now();
607}
608
609double Macs::GetPdoMjd()
610{
611 return fPdoTime.CalcMjd();
612}
613
614/* 0x2000 0 rw Maximum positioning error */
615/* 1 rw Negative Software Endswitch */
616/* 2 rw Positive Software Endswitch */
617void Macs::SetNegEndswitch(LWORDS_t val)
618{
619 SendSDO(0x2000, 1, (LWORD_t)val);
620 WaitForSdo(0x2000, 1);
621}
622
623void Macs::SetPosEndswitch(LWORDS_t val)
624{
625 SendSDO(0x2000, 2, (LWORD_t)val);
626 WaitForSdo(0x2000, 2);
627}
628
629void Macs::EnableEndswitches(bool neg, bool pos)
630{
631 SendSDO(0x2000, 3, (LWORD_t)(neg|(pos<<1)));
632 WaitForSdo(0x2000, 3);
633}
634
635void Macs::ReqTimeoutTime()
636{
637 RequestSDO(0x4000, 2);
638 WaitForSdo(0x4000, 2);
639}
640
641void Macs::EnableTimeout(bool enable, LWORDS_t ms)
642{
643 lout << "- " << GetNodeName() << ": " << (enable?"En":"Dis") << "able timeout, " << dec << ms << "ms." << endl;
644 if (!enable)
645 {
646 SendSDO(0x4000, 1, string('o', 'f', 'f'));
647 WaitForSdo(0x4000, 1);
648
649 lout << "- " << GetNodeName() << ": Stopping handshake (PC)." << endl;
650
651 fTimeout->Stop(); //kTRUE);
652
653 fTimerOn = kFALSE;
654 }
655 else
656 {
657 if (ms>0)
658 SetTimeoutTime(ms);
659
660 ResetTimeout();
661
662 fTimerOn = kTRUE;
663 fTimeout->Start(fGuardTime/*/3*2*/, kTRUE); //kFALSE); //TRUE);
664
665 //
666 // Start with kFALSE would be a continous timer, but this
667 // timer seems to stop it's activity at some stage without
668 // any reason
669 //
670 lout << "- " << GetNodeName() << ": starting handshake." << endl;
671 SendSDO(0x4000, 1, string('o', 'n'));
672 WaitForSdo(0x4000, 1);
673 }
674 lout << "- " << GetNodeName() << ": Timeout timer turned "
675 << (enable?"on.":"off.") << endl;
676}
677
678Bool_t Macs::HandleTimer(TTimer *t)
679{
680 /*
681 Fons:
682 -----
683
684 timers never trigger at the same time or when in a TTimer::Notify.
685 Little explanation:
686
687 - there are two types of timers synchronous and a-synchronous.
688 - synchronous timers are only handled via the ROOT eventloop
689 (see TUnixSystem::DispatchOneEvent()). If there are no mouse/keyboard
690 events then the synchronous timer queue is checked. So if the processing
691 of a mouse/keyboard event takes a long time synchronous timers are not
692 called for a while. To prevent this from happening one can call in long
693 procedures gSystem->ProcessEvents(). The system schedules only the
694 next timer in the queue when the current one's Notify() has finished.
695 - a-synchronous timers are triggered via SIGALARM, i.e. the program is
696 interupted and execution jumps to the Notify() function. When the
697 notify is finished the next a-sync timer is scheduled and the system
698 resumes from the place where it was initially interrupted. One of the
699 things to remember when using a-sync timers is don't make any graphics
700 calls in them. X11 is not re-entrant and it might be that the SIGALARM
701 signal interrupted the system while being in X11. A-sync timers are best
702 used to set flags that you can test at a convenient and controlled
703 time.
704 */
705
706 //
707 // FIXME! Use NMT!
708 //
709 // --- lout << ddev(MLog::eGui);
710 // --- lout << "Send 0x4000: " << GetNodeName() << endl;
711
712 SendSDO(0x4000, 0, (LWORD_t)0, false);
713
714 // --- lout << "Done 0x4000: " << GetNodeName() << endl;
715
716 Timer time;
717
718 // --- lout << "dT " << GetNodeName() << ": " << dec <<(int)((fTimeoutTime-time.Now())*1000) << endl;
719 // --- lout << edev(MLog::eGui);
720
721 if (time.Now() > fTimeoutTime)
722 {
723 lout << ddev(MLog::eGui);
724 lout << "Warning: " << GetNodeName() << " didn't respond in timeout window." << endl;
725 lout << edev(MLog::eGui);
726 SetError(101);
727 }
728
729 //WaitForSdo(0x4000, 0, kDontWait);
730 //
731 // Would not be necessary if I would Start the timer with
732 // kFALSE. This would be a continous timer, but this
733 // timer seems to stop it's activity at some stage without
734 // any reason
735 //
736 if (fTimerOn)
737 fTimeout->Start(fGuardTime/*/3*2*/, kTRUE);
738
739 return kTRUE;
740}
741
742void Macs::ResetTimeout()
743{
744 Timer time;
745
746 // --- lout << ddev(MLog::eGui);
747 // --- lout << "Reset " << GetNodeName() << ": " << dec << (int)((fTimeoutTime-time.Now())*1000) << endl;
748 // --- lout << edev(MLog::eGui);
749
750 fTimeoutTime = time.Now() + 3.*fGuardTime/1000.;
751}
752
753void Macs::SetTimeoutTime(LWORD_t ms)
754{
755 // FIXME: Is '/2' the best choose?
756 fGuardTime = ms/3; // how often do we send/request the handshake
757
758 SendSDO(0x4000, 2, ms*2); // How often do we check for the handshake
759 WaitForSdo(0x4000, 2);
760}
761
Note: See TracBrowser for help on using the repository browser.