1 | /*-------------------------------------------------------------------------*/
|
---|
2 | /* DIM section for Array definition (arrays are globals) */
|
---|
3 | /*-------------------------------------------------------------------------*/
|
---|
4 |
|
---|
5 | DIM errlist[9] /* idx 1=number of valid entries */
|
---|
6 |
|
---|
7 | /* ----------------------------------------------------------------------- */
|
---|
8 | /* */
|
---|
9 | /* Version: */
|
---|
10 | /* */
|
---|
11 | kVERSION = 0 /* */
|
---|
12 | kSUBVERSION = 69 /* */
|
---|
13 | /* */
|
---|
14 | /* HISTORY: */
|
---|
15 | /* */
|
---|
16 | /* * V0.69: */
|
---|
17 | /* - Implemented Pre-Operational/Operational state */
|
---|
18 | /* means: + cannot start RPM mode */
|
---|
19 | /* + SDO 0x6004 (POSA/POSR) ignored */
|
---|
20 | /* + new SDO 0x1000 */
|
---|
21 | /* */
|
---|
22 | /* * V0.68: */
|
---|
23 | /* - Made NoWait=On the default */
|
---|
24 | /* - Implemented error 89 (CAN I/O) */
|
---|
25 | /* - changed "REOPEN 0 0" to "REOPEN 1 0" */
|
---|
26 | /* */
|
---|
27 | /* * V0.67: */
|
---|
28 | /* - trying to implement the control cabinet stuff */
|
---|
29 | /* */
|
---|
30 | /* * V0.66: */
|
---|
31 | /* - implemented canopen node guarding (0x4000) */
|
---|
32 | /* - implemented some kind of host guarding (in conjuction with */
|
---|
33 | /* the implemented nodeguarding) */
|
---|
34 | /* - removed old guarding using SDO 0x4000 */
|
---|
35 | /* - implemented SDO 0x100c (guardtime) */
|
---|
36 | /* - implemented SDO 0x100d (lifetimefactor) */
|
---|
37 | /* */
|
---|
38 | /* * V0.65: */
|
---|
39 | /* - fixed a bug in the handling of the endswitches */
|
---|
40 | /* */
|
---|
41 | /* * V0.64: */
|
---|
42 | /* - removed FORCEHOME */
|
---|
43 | /* */
|
---|
44 | /* * V0.63: */
|
---|
45 | /* - added movement handshake timeout (removed 0x400 WAITAX) */
|
---|
46 | /* - added brackets around string SDOs */
|
---|
47 | /* - changed SDO 0x3007 to support both sync modes */
|
---|
48 | /* */
|
---|
49 | /* * V0.62: */
|
---|
50 | /* - changed handling of 0x2000/1/2 added /3 */
|
---|
51 | /* */
|
---|
52 | /* * V0.61: */
|
---|
53 | /* - corrected problems with the error handling */
|
---|
54 | /* */
|
---|
55 | /* * V0.60: */
|
---|
56 | /* - introduced syncronisation */
|
---|
57 | /* */
|
---|
58 | /* * V0.52: */
|
---|
59 | /* - changed the handling of the endswitch error (unknown switch) */
|
---|
60 | /* */
|
---|
61 | /* * V0.51: */
|
---|
62 | /* - made errlist working */
|
---|
63 | /* */
|
---|
64 | /* * V0.50: */
|
---|
65 | /* - Rearanged many object numbers */
|
---|
66 | /* - Added object dictonary to comments */
|
---|
67 | /* */
|
---|
68 | /* * V0.43: */
|
---|
69 | /* - Added Object 0x1003, 0x1004, 0x1005 */
|
---|
70 | /* */
|
---|
71 | /* * V0.42: */
|
---|
72 | /* - Added APOS to PDO1 */
|
---|
73 | /* */
|
---|
74 | /* * V0.41: */
|
---|
75 | /* - Period Interrupt diabled while HOME */
|
---|
76 | /* */
|
---|
77 | /* * V0.40: */
|
---|
78 | /* - Introduced object 0x1010, 0x1011 */
|
---|
79 | /* */
|
---|
80 | /* * V0.38: */
|
---|
81 | /* - Introduced object 0x100a */
|
---|
82 | /* */
|
---|
83 | /* * V0.37: */
|
---|
84 | /* - ON ERROR GOSUB moved after new init section */
|
---|
85 | /* */
|
---|
86 | /* * V0.36: */
|
---|
87 | /* - Enahnced Initialization (NOWAIT OFF, etc.) */
|
---|
88 | /* */
|
---|
89 | /* * V0.35: */
|
---|
90 | /* - SDO 0x4003/EXIT introduced */
|
---|
91 | /* */
|
---|
92 | /* * V0.34: */
|
---|
93 | /* - PDO1 as answer to a SDO added (maybe SDO changes state) */
|
---|
94 | /* */
|
---|
95 | /* ----------------------------------------------------------------------- */
|
---|
96 |
|
---|
97 | PRINT "Magic Mics V", kVERSION, ".", kSUBVERSION, " (CANbus controlled)" /**/
|
---|
98 |
|
---|
99 | /* ----------------------------------------------------------------------- */
|
---|
100 | /* */
|
---|
101 | /* Object Dictionary: */
|
---|
102 | /* */
|
---|
103 | /* 0x1000 1 rw Start node (Pre-oprational --> operational) */
|
---|
104 | /* 0x1003 x rw Read delete error list (subidx 0-9) */
|
---|
105 | /* 0x1004 0 ro Nr of PDOs (transmit) */
|
---|
106 | /* 1 ro Nr of PDOs (synchron) */
|
---|
107 | /* 2 ro Nr of PDOs (asynchron) */
|
---|
108 | /* 0x1005 x ro COB ID for Syncs */
|
---|
109 | /* 0x100a x ro Software Version */
|
---|
110 | /* 0x100b x ro Node number */
|
---|
111 | /* 0x100c x rw Guardtime (ms) */
|
---|
112 | /* 0x100d x rw Lifetime factor */
|
---|
113 | /* 0x100e x ro COB ID for Guarding */
|
---|
114 | /* 0x1010 x wo Write data to EEPROM */
|
---|
115 | /* 0x1014 x ro COB ID for Emergency */
|
---|
116 | /* 0x1800 x rw Enable PDO1 (Axe Status, Position) */
|
---|
117 | /* 0x2000 0 rw Maximum positioning error */
|
---|
118 | /* 1 rw Negative Software Endswitch Value (Set=Enable) */
|
---|
119 | /* 2 rw Positive Software Endswitch Value (Set=Enable) */
|
---|
120 | /* 3 rw Enable/Disable Software Endswitch */
|
---|
121 | /* 0x2002 x rw Velocity */
|
---|
122 | /* 0x2003 0 wo Acceleration */
|
---|
123 | /* 1 wo Deceleration */
|
---|
124 | /* 0x3000 x wo Motor 'on', 'off', 'stop' (Warning: Motor off can have */
|
---|
125 | /* strange side effects if RF=1!) */
|
---|
126 | /* 0x3001 x wo Home 'home' */
|
---|
127 | /* 0x3002 x wo Reopen Communication 'open' */
|
---|
128 | /* 0x3003 x wo Exit Program 'exit' */
|
---|
129 | /* 0x3006 0 wo Velocity Mode 'strt', 'stop' */
|
---|
130 | /* 1 wo VelMode Velocity */
|
---|
131 | /* 0x3007 0 wo Velocity Syncronisation 'sync' */
|
---|
132 | /* 1 wo Position Syncronisation 'sync' */
|
---|
133 | /* 0x3008 x wo Nowait 'on', 'off' */
|
---|
134 | /* 0x4000 0 wo Reset timeout timer (Nodeguard) */
|
---|
135 | /* 0x6000 x rw Rotation Direction */
|
---|
136 | /* 0x6002 x rw Velocity Resolution */
|
---|
137 | /* 0x6003 0 wo Define present position as origin ('set') */
|
---|
138 | /* 1 wo Define new origin (0=delete) */
|
---|
139 | /* 2 rw Home Offset */
|
---|
140 | /* 0x6004 0 rw Absolute Position */
|
---|
141 | /* 1 wo Relative Position */
|
---|
142 | /* 1 ro Control Position */
|
---|
143 | /* 0x6501 x rw Encoder Resolution */
|
---|
144 | /* 0x6502 x rw Maximum Velocity */
|
---|
145 | /* 0x6508 x ro Time since switch on */
|
---|
146 | /* */
|
---|
147 | /* ----------------------------------------------------------------------- */
|
---|
148 |
|
---|
149 |
|
---|
150 | /*-------------------------------------------------------------------------*/
|
---|
151 | /* section for global constants */
|
---|
152 | /*-------------------------------------------------------------------------*/
|
---|
153 | SET PRGPAR -1 /* Don't restart any Program on Exit */
|
---|
154 |
|
---|
155 | SET ENCODERTYPE 0 /* Incremental Encoder */
|
---|
156 | SET MENCODERTYPE 0 /* Incremental Encoder (Master) */
|
---|
157 |
|
---|
158 | SET ENDSWMOD 0 /* No End Switch */
|
---|
159 | SET ERRCOND 2 /* Motor Stop, position control, no break */
|
---|
160 | SET POSDRCT -1 /* rotation direction */
|
---|
161 | /* OLD: SET POSDRCT -1 /* rotation direction */
|
---|
162 | SET POSFACT_Z 1 /* 1 user unit (be) = POSFACT_Z/POSFACT_N qc */
|
---|
163 | SET POSFACT_N 1 /* */
|
---|
164 |
|
---|
165 | SET HOME_FORCE 0 /* Don't force Home positioning on mainloopup */
|
---|
166 | SET HOME_OFFSET 0 /* Offset between index and home position */
|
---|
167 | SET HOMETYPE 0 /* drive to home, reverse, go to next index */
|
---|
168 |
|
---|
169 | /*----------------*/
|
---|
170 | /* syncronisation */
|
---|
171 | /*----------------*/
|
---|
172 | SET SYNCFACTM 1 /* Master Sync Velocity factor */
|
---|
173 | SET SYNCFACTS 1 /* Slave Sync Velocity factor */
|
---|
174 | SET SYNCPOSOFFS 0 /* Sync Position offset between M/S */
|
---|
175 | SET SYNCACCURACY 50 /* When to set Accuracy Flag */
|
---|
176 | SET REVERS 0 /* How to handle reversation of vel */
|
---|
177 |
|
---|
178 | /*----------------*/
|
---|
179 | /* Inputs */
|
---|
180 | /*----------------*/
|
---|
181 | SET I_REFSWITCH 0 /* Reference Switch */
|
---|
182 | SET I_POSLIMITSW 0 /* Pos Limit Switch */
|
---|
183 | SET I_NEGLIMITSW 0 /* Neg Limit Switch */
|
---|
184 | SET I_BREAK 0 /* Input which brakes a running program */
|
---|
185 | SET I_CONTINUE 0 /* Input to continue a broken program */
|
---|
186 | SET I_ERRCLR 0 /* Input to clear error */
|
---|
187 |
|
---|
188 | /*----------------*/
|
---|
189 | /* Outputs */
|
---|
190 | /*----------------*/
|
---|
191 | SET O_AXMOVE 0 /* Motor control is working */
|
---|
192 | SET O_BRAKE 0 /* Brake */
|
---|
193 | SET O_ERROR 0 /* error occured */
|
---|
194 |
|
---|
195 | /*----------------*/
|
---|
196 | /* Unit param. */
|
---|
197 | /*----------------*/
|
---|
198 | SET RAMPTYPE 1 /* Ramp Type: 0=Trapez, 1=Sinus */
|
---|
199 | /* OLD: SET RAMPTYPE 0 /* Ramp Type: 0=Trapez, 1=Sinus */
|
---|
200 | SET ENCODER 1500 /* Encoder has 1500 Ticks */
|
---|
201 | SET MENCODER 1500 /* Encoder has 500 Ticks (Master) */
|
---|
202 | SET VELMAX 3000 /* Motor: Maximum revolutions per minute */
|
---|
203 | SET POSERR 1500 /* Maximum tolarable Position error (qc) 0.1° */
|
---|
204 | SET RAMPMIN 10000 /* Shortest Ramp 10s */
|
---|
205 |
|
---|
206 | /*----------------*/
|
---|
207 | /* Dflt vel & acc */
|
---|
208 | /*----------------*/
|
---|
209 |
|
---|
210 | /* Prop=100, Div=300, Int=800 */
|
---|
211 | if (get cannr==1) then
|
---|
212 | SET KPROP 100
|
---|
213 | SET KDER 300
|
---|
214 | SET KINT 1000
|
---|
215 | elseif (get cannr==2) then
|
---|
216 | SET KPROP 100
|
---|
217 | SET KDER 200
|
---|
218 | SET KINT 150
|
---|
219 | else
|
---|
220 | SET KPROP 350
|
---|
221 | SET KDER 50
|
---|
222 | SET KINT 350
|
---|
223 | endif
|
---|
224 |
|
---|
225 | vres = (GET ENCODER)*(GET VELMAX) /* ticks/R * R/M = ticks/min */
|
---|
226 | SET VELRES vres /* Set velocity units */
|
---|
227 |
|
---|
228 |
|
---|
229 | /* OLD: SET HOME_VEL -(25*vres%100) /* Home position velocity: 25% */
|
---|
230 | /* OLD: SET HOME_RAMP (25*vres%100) /* Home position accel: 25% */
|
---|
231 |
|
---|
232 | /*----------------*/
|
---|
233 | /* Manual control */
|
---|
234 | /*----------------*/
|
---|
235 | SET RAMPTYPE 1 /* Ramp: 0=linear, 1=sinus */
|
---|
236 | defacc = 60*vres%100
|
---|
237 |
|
---|
238 | SET DFLTACC defacc /* Default acceleratio: [%] */
|
---|
239 | ACC defacc
|
---|
240 | DEC defacc*2
|
---|
241 | /* Velocity which is reached in
|
---|
242 | a time given by RAMPMIN */
|
---|
243 | SET DFLTVEL (1*vres%100) /* Default velocity [%] */
|
---|
244 |
|
---|
245 | /*manvel = (4*vres%100) /* 150 U/min */ /* Max speed in man mode: [%] */
|
---|
246 |
|
---|
247 | print "Vel Res (vel max): ", GET VELRES, " Encoder Ticks/min"
|
---|
248 | /*print "V_man: ", manvel, " Encoder Ticks/min"*/
|
---|
249 |
|
---|
250 | /*-------------------------------------------------------------------------*/
|
---|
251 | /* const section for constant velues */
|
---|
252 | /*-------------------------------------------------------------------------*/
|
---|
253 | kTRUE = 1
|
---|
254 | kFALSE = 0
|
---|
255 |
|
---|
256 | pdotime = 100
|
---|
257 | pdo1on = kFALSE
|
---|
258 |
|
---|
259 | guardtime = 0
|
---|
260 | lifetimefactor = 0
|
---|
261 | timeouttime = TIME
|
---|
262 | firsttimeout = 0
|
---|
263 |
|
---|
264 | preop = kTRUE
|
---|
265 |
|
---|
266 | /*-------------------------------------------------------------------------*/
|
---|
267 | /* Can Open Definitions */
|
---|
268 | /*-------------------------------------------------------------------------*/
|
---|
269 | /* The CAN Object are static object. This is why they must be deleted. */
|
---|
270 | /* The program should run in any of our nodes. */
|
---|
271 | /* Therefor the standard CAN objects (SDo, PDO1, PDO2) for communication */
|
---|
272 | /* are defined. The nodenumber is part of the object ID (this is somehow */
|
---|
273 | /* similar to the TCP/IP ports) */
|
---|
274 | /*-------------------------------------------------------------------------*/
|
---|
275 | CANDEL -1
|
---|
276 |
|
---|
277 | nodenr = GET CANNR
|
---|
278 | PRINT "Initializing Node Nr.", nodenr
|
---|
279 | pdo1 = DEFCANOUT (0x180+nodenr) 8
|
---|
280 | pdo2 = DEFCANOUT (0x280+nodenr) 8
|
---|
281 | pdo3 = DEFCANOUT (0x380+nodenr) 8
|
---|
282 | sdotx = DEFCANOUT (0x580+nodenr) 8
|
---|
283 | sdorx = DEFCANIN (0x600+nodenr) 8
|
---|
284 | /*
|
---|
285 | guardrx = DEFCANIN (0x700+nodenr) 8
|
---|
286 | guardtx = DEFCANOUT (0x700+nodenr) 8
|
---|
287 | */
|
---|
288 |
|
---|
289 | /* Close and reopen communication, enable buffering */
|
---|
290 | err = REOPEN 1 0
|
---|
291 |
|
---|
292 | /*-------------------------------------------------------------------------*/
|
---|
293 | /* Before the motor control hardware is enabled (hi on output 1) */
|
---|
294 | /* the commands make sure, that the motor will not start moving. */
|
---|
295 | /* As default positioning commands doesn't stop the further execution */
|
---|
296 | /* of the program. */
|
---|
297 | /*-------------------------------------------------------------------------*/
|
---|
298 | /*MOTOR STOP
|
---|
299 | MOTOR OFF
|
---|
300 | CVEL 0
|
---|
301 | OUT 1 1
|
---|
302 | */
|
---|
303 | NOWAIT ON
|
---|
304 |
|
---|
305 | /*-------------------------------------------------------------------------*/
|
---|
306 | /* ON ... GOSUB ... definitions */
|
---|
307 | /*-------------------------------------------------------------------------*/
|
---|
308 | /* The errorlist one can retreive using the corresponding CAN object */
|
---|
309 | /* should be emty when the node is initialized (arrays are static objects) */
|
---|
310 | /* therefor it must be deleted. */
|
---|
311 | /* Errors are handled in an interrupt procedure called PROC_ERROR */
|
---|
312 | /*-------------------------------------------------------------------------*/
|
---|
313 | /* ON CANMSG GOSUB PROC_CANMSG */
|
---|
314 | i = 9
|
---|
315 | while (i) do
|
---|
316 | errlist[i] = 0
|
---|
317 | i = i - 1
|
---|
318 | endwhile
|
---|
319 |
|
---|
320 | ON ERROR GOSUB PROC_ERROR
|
---|
321 |
|
---|
322 | /***************************************************************************/
|
---|
323 | kIoModule = 4*256
|
---|
324 |
|
---|
325 | /*-------------------------------------------------------------------------*/
|
---|
326 | /* Init */
|
---|
327 | /*-------------------------------------------------------------------------*/
|
---|
328 | /* Program Main Loop */
|
---|
329 | /*-------------------------------------------------------------------------*/
|
---|
330 | /* The main loop is the core of the program which handles incoming */
|
---|
331 | /* objects. In principal CANIN should wait until an object is received, */
|
---|
332 | /* but it stops waiting when an interrupt occurs. This is the reason why */
|
---|
333 | /* the validity of the message must be checked. */
|
---|
334 | /*-------------------------------------------------------------------------*/
|
---|
335 | MAIN:
|
---|
336 | canhi = 0
|
---|
337 | canlo = 1
|
---|
338 |
|
---|
339 | brake = 0
|
---|
340 | RF = 0
|
---|
341 |
|
---|
342 | init = 0
|
---|
343 |
|
---|
344 | gosub reset
|
---|
345 |
|
---|
346 | PRINT "Starting Mainloop..."
|
---|
347 |
|
---|
348 | rc = 1
|
---|
349 | MAINLOOP:
|
---|
350 | if (rc==0) then /* It must be tested because ON PERIOD breaks 'wait for obj' */
|
---|
351 | gosub PROC_SDORX
|
---|
352 | else
|
---|
353 | fuse = in (kIoModule+1)
|
---|
354 | emcy = in (kIoModule+2)
|
---|
355 | vltg = in (kIoModule+3)
|
---|
356 | mode = in (kIoModule+4)
|
---|
357 | ready = in 1
|
---|
358 |
|
---|
359 | if (ready==0) and (RF==1) then
|
---|
360 | print "DKC not ready, but RF set... setting RF=AH=0!"
|
---|
361 | gosub reset
|
---|
362 | goto mainloop
|
---|
363 | elseif (mode==1) then
|
---|
364 | print "Control not in PC mode!"
|
---|
365 | gosub reset
|
---|
366 | SET PRGPAR 0
|
---|
367 | exit
|
---|
368 | elseif (fuse==0) then
|
---|
369 | print "Motor-Power Fuse not OK!"
|
---|
370 | gosub reset
|
---|
371 | goto mainloop
|
---|
372 | elseif (vltg==0) then
|
---|
373 | print "Overvoltage control broken!"
|
---|
374 | gosub reset
|
---|
375 | goto mainloop
|
---|
376 | elseif (emcy==0) then
|
---|
377 | print "Please release Emergency Stop!"
|
---|
378 | gosub reset
|
---|
379 | goto mainloop
|
---|
380 | elseif (ready==1) and (RF==0) then
|
---|
381 | print "DKC powered, RF=0... setting RF=AH=1!"
|
---|
382 | /*
|
---|
383 | * After switching on power wait at least 300ms until
|
---|
384 | * control changed state 'bb' to 'ab'
|
---|
385 | */
|
---|
386 | cvel 0
|
---|
387 | waitt 300
|
---|
388 | out 1 0
|
---|
389 | out 2 0
|
---|
390 | motor off
|
---|
391 | waitt 100
|
---|
392 | out 1 1
|
---|
393 | out 2 1
|
---|
394 | RF = 1
|
---|
395 | waitt 100
|
---|
396 |
|
---|
397 | if (brake==0 and get cannr==3) then
|
---|
398 | out (kIoModule+1) 1
|
---|
399 | brake = 1
|
---|
400 | waitt 1000
|
---|
401 | endif
|
---|
402 |
|
---|
403 | /* This "motor on" is necessary to make sure the floating analog output
|
---|
404 | doesn't move the telescope (slowly) */
|
---|
405 | motor on
|
---|
406 |
|
---|
407 | canout pdo3 (ready | (fuse<<1) | (emcy<<2) | (vltg<<3) | (mode<<4) | (rf<<5) | (brake<<6)) 0
|
---|
408 | /*
|
---|
409 | if (get cannr==2) then
|
---|
410 | syncv
|
---|
411 | print "Synchronizing speed..."
|
---|
412 | else
|
---|
413 | print "Starting CAN mode..." ^
|
---|
414 | goto CANSTART
|
---|
415 | endif */
|
---|
416 | elseif (ready==0) or (RF==0) then
|
---|
417 | print "No Power, no RF..."
|
---|
418 | canout pdo3 (ready | (fuse<<1) | (emcy<<2) | (vltg<<3) | (mode<<4) | (rf<5) | (brake<<6)) 0
|
---|
419 | waitt 500
|
---|
420 | goto mainloop
|
---|
421 | endif
|
---|
422 | endif
|
---|
423 |
|
---|
424 | rc = CANIN sdorx -1 0 canhi canlo
|
---|
425 | goto mainloop
|
---|
426 |
|
---|
427 | ENDMAIN:
|
---|
428 | MOTOR STOP
|
---|
429 | MOTOR OFF
|
---|
430 | OUT 1 0
|
---|
431 | EXIT
|
---|
432 |
|
---|
433 | /*-------------------------------------------------------------------------*/
|
---|
434 | /* Part for Programs called with GOSUB */
|
---|
435 | /*-------------------------------------------------------------------------*/
|
---|
436 |
|
---|
437 | SUBMAINPROG
|
---|
438 | SUBPROG reset
|
---|
439 | init = 0
|
---|
440 |
|
---|
441 | out 1 0
|
---|
442 | out 2 0
|
---|
443 | RF = 0
|
---|
444 | motor off
|
---|
445 | waitt 1000
|
---|
446 |
|
---|
447 | if (brake==1 and get cannr==3) then
|
---|
448 | waitt 3000 /* wait 3s for DKC to stop the motor */
|
---|
449 | out (kIoModule+1) 0 /* brake the brake */
|
---|
450 | waitt 1000
|
---|
451 | endif
|
---|
452 |
|
---|
453 | canout pdo3 (ready | (fuse<<1) | (emcy<<2) | (vltg<<3) | (mode<<4)) (rf | (brake<<1))
|
---|
454 | return
|
---|
455 |
|
---|
456 | /*----------------------------------*/
|
---|
457 | /* PROC_CANOPENMSG */
|
---|
458 | /*----------------------------------*/
|
---|
459 | SUBPROG PROC_SDOSET
|
---|
460 | idx = canhi&0xff00 | (canhi>>16)&0xff
|
---|
461 | subidx = canhi&0xff
|
---|
462 | sdoval = (canlo&0xff)<<24 | (canlo&0xff00)<<8 | (canlo>>8)&0xff00 | (canlo>>24)&0xff
|
---|
463 | canlo = 0
|
---|
464 | /*
|
---|
465 | PRINT "Setting Idx:", idx, "/", subidx, " to ", sdoval
|
---|
466 | */
|
---|
467 | if (idx==0x1000 and subidx==1 and sdoval==1) then
|
---|
468 | preop = kFALSE
|
---|
469 | elseif (idx==0x1003 and subidx==0 and sdoval==0) then
|
---|
470 | i = 9
|
---|
471 | while (i) do
|
---|
472 | errlist[i] = 0
|
---|
473 | i = i - 1
|
---|
474 | endwhile
|
---|
475 | elseif (idx==0x100c) then
|
---|
476 | guardtime = sdoval
|
---|
477 | ON PERIOD 0 GOSUB PROC_Timeout
|
---|
478 | if (lifetimefactor>0 and guardtime>0) then
|
---|
479 | timeouttime = TIME + guardtime*lifetimefactor
|
---|
480 | ON PERIOD guardtime GOSUB PROC_Timeout
|
---|
481 | endif
|
---|
482 | elseif (idx == 0x100d) then
|
---|
483 | lifetimefactor = sdoval
|
---|
484 | if (lifetimefactor==0) then
|
---|
485 | ON PERIOD 0 GOSUB PROC_Timeout
|
---|
486 | endif
|
---|
487 | elseif (idx == 0x1010 and sdoval == ('s'<<24|'a'<<16|'v'<<8|'e')) then
|
---|
488 | SAVEPROM
|
---|
489 | elseif (idx == 0x1800 and subidx == 1) then
|
---|
490 | ON PERIOD 0 GOSUB PROC_PDO1
|
---|
491 | if (sdoval>>31) then
|
---|
492 | pdo1on = kFALSE
|
---|
493 | else
|
---|
494 | ON PERIOD pdotime GOSUB PROC_PDO1
|
---|
495 | pdo1on = kTRUE
|
---|
496 | endif
|
---|
497 | elseif (idx == 0x2000) then
|
---|
498 | if (subidx == 0) then
|
---|
499 | SET POSERR sdoval
|
---|
500 | elseif (subidx == 1) then
|
---|
501 | SET NEGLIMIT sdoval
|
---|
502 | SET SWNEGLIMACT 1
|
---|
503 | elseif (subidx == 2) then
|
---|
504 | SET POSLIMIT sdoval
|
---|
505 | SET SWPOSLIMACT 1
|
---|
506 | elseif (subidx == 3) then
|
---|
507 | SET SWNEGLIMACT sdoval&1
|
---|
508 | SET SWPOSLIMACT (sdoval>>1)&1
|
---|
509 | endif
|
---|
510 | elseif (idx == 0x2002) then
|
---|
511 | VEL sdoval
|
---|
512 | elseif (idx == 0x2003) then
|
---|
513 | if (subidx) then
|
---|
514 | DEC sdoval
|
---|
515 | else
|
---|
516 | ACC sdoval
|
---|
517 | endif
|
---|
518 | elseif (idx == 0x3000) then
|
---|
519 | if (sdoval == ('o'<<24|'n'<<16)) then
|
---|
520 | MOTOR ON
|
---|
521 | elseif (sdoval == ('o'<<24|'f'<<16|'f'<<8)) then
|
---|
522 | MOTOR OFF
|
---|
523 | elseif (sdoval == ('s'<<24|'t'<<16|'o'<<8|'p')) then
|
---|
524 | MOTOR STOP
|
---|
525 | endif
|
---|
526 | elseif (idx == 0x3001) then
|
---|
527 | if (sdoval == ('h'<<24|'o'<<16|'m'<<8|'e')) then
|
---|
528 | limitsw = GET I_POSLIMITSW
|
---|
529 | set I_POSLIMITSW 0
|
---|
530 | /* Disable Interrupt - same problem than with CANIN */
|
---|
531 | ON PERIOD 0 GOSUB PROC_PDO1
|
---|
532 | HOME
|
---|
533 | /* Reenable interrupt */
|
---|
534 | if (pdo1on) then
|
---|
535 | ON PERIOD pdotime GOSUB PROC_PDO1
|
---|
536 | endif
|
---|
537 | SET I_POSLIMITSW limitsw
|
---|
538 | endif
|
---|
539 | elseif (idx == 0x3002 and sdoval == ('o'<<24|'p'<<16|'e'<<8|'n')) then
|
---|
540 | sdoval=REOPEN 2 0
|
---|
541 | if (sdoval) then
|
---|
542 | PRINT "Error Reopen"
|
---|
543 | endif
|
---|
544 | elseif (idx == 0x3003 and sdoval == ('e'<<24|'x'<<16|'i'<<8|'t')) then
|
---|
545 | CANOUT sdotx (canhi&0xffffff | 0x60000000) 0
|
---|
546 | EXIT
|
---|
547 | elseif (idx == 0x3006) then
|
---|
548 | if (subidx == 0) then
|
---|
549 | if (sdoval == ('s'<<24|'t'<<16|'r'<<8|'t')) then
|
---|
550 | if (preop==kFALSE) then
|
---|
551 | CVEL 0
|
---|
552 | CSTART
|
---|
553 | else
|
---|
554 | canlo = 1
|
---|
555 | endif
|
---|
556 | elseif (sdoval == ('s'<<24|'t'<<16|'o'<<8|'p')) then
|
---|
557 | CSTOP
|
---|
558 | endif
|
---|
559 | elseif (subidx == 1) then
|
---|
560 | CVEL sdoval
|
---|
561 | endif
|
---|
562 | elseif (idx == 0x3007) then
|
---|
563 | if (preop==kFALSE) then
|
---|
564 | if (subidx==0 and sdoval == ('s'<<24|'y'<<16|'n'<<8|'c')) then
|
---|
565 | SYNCV
|
---|
566 | elseif (subidx==1 and sdoval == ('s'<<24|'y'<<16|'n'<<8|'c')) then
|
---|
567 | SYNCP
|
---|
568 | endif
|
---|
569 | else
|
---|
570 | canlo = 1
|
---|
571 | endif
|
---|
572 | elseif (idx == 0x3008) then
|
---|
573 | if (sdoval == ('o'<<24|'n'<<16)) then
|
---|
574 | NOWAIT ON
|
---|
575 | elseif (sdoval == ('o'<<24|'f'<<16|'f'<<8)) then
|
---|
576 | NOWAIT OFF
|
---|
577 | endif
|
---|
578 | elseif (idx == 0x4000) then
|
---|
579 | timeouttime = TIME + guardtime*lifetimefactor
|
---|
580 | elseif (idx == 0x6000) then
|
---|
581 | if (sdoval&1) then
|
---|
582 | SET POSDRCT -1
|
---|
583 | else
|
---|
584 | SET POSDRCT 1
|
---|
585 | endif
|
---|
586 | elseif (idx == 0x6002) then
|
---|
587 | SET VELRES sdoval
|
---|
588 | elseif (idx == 0x6003) then
|
---|
589 | if (subidx == 0 and sdoval == ('s'<<24|'e'<<16|'t'<<8)) then
|
---|
590 | DEF ORIGIN
|
---|
591 | elseif (subidx == 1) then
|
---|
592 | if (sdoval==0) then
|
---|
593 | RST ORIGIN
|
---|
594 | else
|
---|
595 | SET ORIGIN sdoval
|
---|
596 | endif
|
---|
597 | elseif (subidx == 2) then
|
---|
598 | SET HOME_OFFSET sdoval
|
---|
599 | endif
|
---|
600 | elseif (idx == 0x6004) then
|
---|
601 | if (preop==kFALSE) then
|
---|
602 | if (subidx==0) then
|
---|
603 | POSA sdoval
|
---|
604 | elseif (subidx==1) then
|
---|
605 | POSR sdoval
|
---|
606 | endif
|
---|
607 | else
|
---|
608 | canlo = 1
|
---|
609 | endif
|
---|
610 | elseif (idx == 0x6200) then
|
---|
611 | pdotime = sdoval
|
---|
612 | if (pdo1on) then
|
---|
613 | ON PERIOD 0 GOSUB PROC_PDO1
|
---|
614 | ON PERIOD pdotime GOSUB PROC_PDO1
|
---|
615 | endif
|
---|
616 | elseif (idx == 0x6501) then
|
---|
617 | SET ENCODER sdoval
|
---|
618 | elseif (idx == 0x6502) then
|
---|
619 | SET VELMAX sdoval
|
---|
620 | else
|
---|
621 | CANOUT sdotx (canhi&0xffffff | (1<<31)/*&0x80000000*/) 0
|
---|
622 | PRINT "Unknown SDO: idx=", idx, ", subidx=", subidx
|
---|
623 | goto ENDSDOSET
|
---|
624 | endif
|
---|
625 |
|
---|
626 | if (pdo1on) then
|
---|
627 | GOSUB PROC_PDO1
|
---|
628 | endif
|
---|
629 |
|
---|
630 | CANOUT sdotx (canhi&0xffffff | 0x60000000) canlo
|
---|
631 | /*
|
---|
632 | PRINT "Sdo Set ", idx, "/", subidx
|
---|
633 | */
|
---|
634 | ENDSDOSET:
|
---|
635 | RETURN
|
---|
636 |
|
---|
637 | /*----------------------------------*/
|
---|
638 |
|
---|
639 | SUBPROG PROC_SDOREQ
|
---|
640 | idx = canhi&0xff00 | (canhi>>16)&0xff
|
---|
641 | subidx = canhi&0xff
|
---|
642 | /*
|
---|
643 | PRINT "Requesting Idx:", idx, "/", subidx
|
---|
644 | */
|
---|
645 | if (idx == 0x1000 and subidx==1) then
|
---|
646 | sdoval = preop
|
---|
647 | elseif (idx == 0x1003) then
|
---|
648 | if (subidx >=0 and subidx<=9) then
|
---|
649 | sdoval = errlist[subidx-1]
|
---|
650 | endif
|
---|
651 | elseif (idx == 0x1004) then
|
---|
652 | if (subidx == 0) then
|
---|
653 | sdoval = 1
|
---|
654 | elseif (subidx == 1) then
|
---|
655 | sdoval = 0
|
---|
656 | elseif (subidx == 2) then
|
---|
657 | sdoval = 1
|
---|
658 | endif
|
---|
659 | elseif (idx == 0x1005) then
|
---|
660 | sdoval = 1<<31 | 0x80
|
---|
661 | elseif (idx == 0x100b) then
|
---|
662 | sdoval = nodenr
|
---|
663 | elseif (idx == 0x100a) then
|
---|
664 | sdoval = (kVERSION<<16) | kSUBVERSION
|
---|
665 | elseif (idx == 0x100c) then
|
---|
666 | sdoval = guardtime
|
---|
667 | elseif (idx == 0x100d) then
|
---|
668 | sdoval = lifetimefactor
|
---|
669 | elseif (idx == 0x100e) then
|
---|
670 | sdoval = 0x600|nodenr /*0x700 | nodenr*/
|
---|
671 | elseif (idx == 0x1010) then
|
---|
672 | sdoval = 1
|
---|
673 | elseif (idx == 0x1011) then
|
---|
674 | sdoval = 0
|
---|
675 | elseif (idx == 0x1014) then
|
---|
676 | sdoval = 0x80 | nodenr
|
---|
677 | elseif (idx == 0x1800) then
|
---|
678 | if (subidx == 1) then
|
---|
679 | sdoval = (~pdo1on)<<31 | (0x0180 + nodenr)
|
---|
680 | elseif (subidx == 2) then
|
---|
681 | sdoval = 0xfe
|
---|
682 | elseif (subidx == 3) then
|
---|
683 | sdoval = 0
|
---|
684 | endif
|
---|
685 | elseif (idx == 0x2000) then
|
---|
686 | if (subidx == 0) then
|
---|
687 | sdoval = GET POSERR
|
---|
688 | elseif (subidx == 1) then
|
---|
689 | sdoval = GET NEGLIMIT | (GET SWNEGLIMACT) << 30
|
---|
690 | elseif (subidx == 2) then
|
---|
691 | sdoval = GET POSLIMIT | (GET SWPOSLIMACT) << 31
|
---|
692 | elseif (subidx == 3) then
|
---|
693 | sdoval = (GET SWNEGLIMACT) | ((GET SWPOSLIMACT)<<1)
|
---|
694 | endif
|
---|
695 | elseif (idx == 0x2001) then
|
---|
696 | sdoval = AXEND
|
---|
697 | elseif (idx == 0x2002) then
|
---|
698 | sdoval = AVEL
|
---|
699 | elseif (idx == 0x2003) then
|
---|
700 | if (subidx==0) then
|
---|
701 | sdoval = INB 0
|
---|
702 | elseif (subidx>0 and subidx<2) then
|
---|
703 | sdoval = IN subidx
|
---|
704 | endif
|
---|
705 | elseif (idx == 0x2004) then
|
---|
706 | sdoval = STAT
|
---|
707 | elseif (idx == 0x6000) then
|
---|
708 | if (GET POSDRCT == 1) then
|
---|
709 | sdoval = 0
|
---|
710 | elseif (GET POSDRCT == -1) then
|
---|
711 | sdoval = 1
|
---|
712 | endif
|
---|
713 | elseif (idx == 0x6002) then
|
---|
714 | sdoval = GET VELRES
|
---|
715 | elseif (idx == 0x6003) then
|
---|
716 | sdoval = GET HOME_OFFSET
|
---|
717 | elseif (idx == 0x6004) then
|
---|
718 | if (subidx == 0) then
|
---|
719 | sdoval = APOS
|
---|
720 | elseif (subidx==1) then
|
---|
721 | sdoval = CPOS
|
---|
722 | endif
|
---|
723 | elseif (idx == 0x6501) then
|
---|
724 | sdoval = GET ENCODER
|
---|
725 | elseif (idx == 0x6502) then
|
---|
726 | sdoval = GET VELMAX
|
---|
727 | elseif (idx == 0x6508) thenä
|
---|
728 | sdoval = TIME
|
---|
729 | else
|
---|
730 | CANOUT sdotx (canhi&0xffffff | 0x80000000) 0
|
---|
731 | goto ENDSDOREQ
|
---|
732 | endif
|
---|
733 |
|
---|
734 | canlo = (sdoval&0xff)<<24 | (sdoval&0xff00)<<8 | (sdoval>>8)&0xff00 | (sdoval>>24)&0xff
|
---|
735 | CANOUT sdotx (canhi&0xffffff | 0x43000000) canlo
|
---|
736 | /*
|
---|
737 | PRINT "Returning: ", sdoval
|
---|
738 | */
|
---|
739 | ENDSDOREQ:
|
---|
740 | RETURN
|
---|
741 |
|
---|
742 | /*----------------------------------------------------------------------*/
|
---|
743 | /* PROC_SDORX */
|
---|
744 | /*----------------------------------------------------------------------*/
|
---|
745 | /* This procedure handles incoming objects, it is called from the main */
|
---|
746 | /* loop. If the object ID (COB ID) identifies a SDO object, it is */
|
---|
747 | /* whether it is a object to set data (write into the object */
|
---|
748 | /* dictionary) or data is requested (read from object dictionary) */
|
---|
749 | /* If it isn't a valid SDO a error message is send. */
|
---|
750 | /* Remark: Only objects with the right node number are received by the */
|
---|
751 | /* main loop. */
|
---|
752 | /*----------------------------------------------------------------------*/
|
---|
753 |
|
---|
754 | SUBPROG PROC_SDORX
|
---|
755 | /* --Echo--
|
---|
756 | CANOUT sdotx canhi canlo
|
---|
757 | */
|
---|
758 | cmd = canhi>>24
|
---|
759 | if (cmd==0x23 OR cmd==0x2B OR cmd==0x2F) then
|
---|
760 | gosub PROC_SDOSET
|
---|
761 | elseif (cmd == 0x40) then
|
---|
762 | gosub PROC_SDOREQ
|
---|
763 | else
|
---|
764 | PRINT "Unknown SDO cmd", cmd
|
---|
765 | CANOUT sdotx (canhi&0xffffff | 0x80000000) 0
|
---|
766 | endif
|
---|
767 | ENDSDORX:
|
---|
768 | RETURN
|
---|
769 |
|
---|
770 | /*----------------------------------*/
|
---|
771 | /* PROC_CANMSG */
|
---|
772 | /* called if a canmsg with */
|
---|
773 | /* cobid=2*CANNR+1 is received */
|
---|
774 | /* Warning: This doesn't fit to */
|
---|
775 | /* CanOpen specification */
|
---|
776 | /*----------------------------------*/
|
---|
777 | SUBPROG PROC_CANMSG
|
---|
778 | varnr = InMsg(-1)
|
---|
779 | PRINT "varnr=", varnr, "msgval=", msgval
|
---|
780 | RETURN
|
---|
781 |
|
---|
782 | /*-------------------------------------------------------------------------*/
|
---|
783 | /* PDO 1 Interrupt */
|
---|
784 | /*-------------------------------------------------------------------------*/
|
---|
785 | SUBPROG PROC_PDO1
|
---|
786 | CANOUT pdo1 AXEND APOS
|
---|
787 | RETURN
|
---|
788 |
|
---|
789 | /*-------------------------------------------------------------------------*/
|
---|
790 | /* Timeout Interrupt */
|
---|
791 | /*-------------------------------------------------------------------------*/
|
---|
792 | SUBPROG PROC_Timeout
|
---|
793 | if (TIME > timeouttime) then
|
---|
794 | MOTOR STOP
|
---|
795 |
|
---|
796 | if (firsttimeout==0) then
|
---|
797 |
|
---|
798 | /* Tell the bus that an error occured */
|
---|
799 | CANOUT pdo2 0 0
|
---|
800 |
|
---|
801 | i = errlist[1] + 1 /* Fill status of array */
|
---|
802 | while (i>2) do /* shift errors by one */
|
---|
803 | errlist[i] = errlist[i-1]
|
---|
804 | i = i - 1
|
---|
805 | endwhile /* set new errornumber */
|
---|
806 | errlist[2] = 100 /* User Error #100 */
|
---|
807 | if (errlist[1]<8) then /* write new size if enhanced */
|
---|
808 | errlist[1] = errlist[1] + 1
|
---|
809 | endif
|
---|
810 |
|
---|
811 | errinf = 0
|
---|
812 |
|
---|
813 | PRINT "User Timeout!"
|
---|
814 |
|
---|
815 | /* tell the bus what exactly happened */
|
---|
816 | CANOUT pdo2 errlist[2] errinf
|
---|
817 | endif
|
---|
818 |
|
---|
819 | firsttimeout = 1
|
---|
820 | else
|
---|
821 | firsttimeout = 0
|
---|
822 | endif
|
---|
823 | RETURN
|
---|
824 |
|
---|
825 | /*-------------------------------------------------------------------------*/
|
---|
826 | /* Error sub proc */
|
---|
827 | /*-------------------------------------------------------------------------*/
|
---|
828 | SUBPROG PROC_ERROR
|
---|
829 | /*MOTOR STOP*/
|
---|
830 | out 1 0
|
---|
831 | out 2 0
|
---|
832 | RF = 0
|
---|
833 |
|
---|
834 | /* Tell the bus that an error occured */
|
---|
835 | CANOUT pdo2 0 0
|
---|
836 |
|
---|
837 | waitt 100
|
---|
838 |
|
---|
839 | if (brake==1 and get cannr==3) then
|
---|
840 | waitt 5000
|
---|
841 | out (kIoModule+1) 0
|
---|
842 | waitt 500
|
---|
843 | endif
|
---|
844 |
|
---|
845 | print "Error #", errno
|
---|
846 |
|
---|
847 | i = errlist[1] + 1 /* Fill status of array */
|
---|
848 | while (i>2) do /* shift errors by one */
|
---|
849 | errlist[i] = errlist[i-1]
|
---|
850 | i = i - 1
|
---|
851 | endwhile /* set new errornumber */
|
---|
852 | errlist[2] = ERRNO
|
---|
853 | if (errlist[1]<8) then /* write new size if enhanced */
|
---|
854 | errlist[1] = errlist[1] + 1
|
---|
855 | endif
|
---|
856 |
|
---|
857 | errinf = 0
|
---|
858 |
|
---|
859 | /* check if the error is repairable and repair */
|
---|
860 | if (errlist[2]==6) then
|
---|
861 | PRINT "No home forced!"
|
---|
862 | ERRCLR
|
---|
863 | elseif (errlist[2]==8) then
|
---|
864 | PRINT "Control deviation overflow."
|
---|
865 | ERRCLR
|
---|
866 | errinf = 0xaffe
|
---|
867 | elseif (errlist[2]==9) then
|
---|
868 | PRINT "Did'n find zero index."
|
---|
869 | ERRCLR
|
---|
870 | elseif (errlist[2]==11) then
|
---|
871 | poslsw = GET POSLIMIT
|
---|
872 | neglsw = GET NEGLIMIT
|
---|
873 | if ((GET SWNEGLIMACT) and APOS<=neglsw) then
|
---|
874 | PRINT "Negative software endswitch (", neglsw, ") activated at position ", APOS
|
---|
875 | SET SWNEGLIMACT 0
|
---|
876 | ERRCLR
|
---|
877 | CVEL (vres%100)
|
---|
878 | ACC (10*vres%100)
|
---|
879 | DEC (10*vres%100)
|
---|
880 | POSA neglsw + 100
|
---|
881 | SET SWNEGLIMACT 1
|
---|
882 | errinf = -1
|
---|
883 | elseif ((GET SWPOSLIMACT) and APOS>=poslsw) then
|
---|
884 | PRINT "Positive software endswitch (", poslsw, ") activated at position ", APOS
|
---|
885 | SET SWPOSLIMACT 0
|
---|
886 | ERRCLR
|
---|
887 | CVEL (vres%100) /* 1% */
|
---|
888 | ACC (10*vres%100) /* 10% */
|
---|
889 | DEC (10*vres%100) /* 10% */
|
---|
890 | POSA poslsw - 100
|
---|
891 | SET SWPOSLIMACT 1
|
---|
892 | errinf = 1
|
---|
893 | else
|
---|
894 | PRINT "Software endswitch activated - command skipped. Pos: ", APOS
|
---|
895 | ERRCLR
|
---|
896 | endif
|
---|
897 | elseif (errlist[2]==25) then
|
---|
898 | /* FIXME: To handle this correct you must make sure,
|
---|
899 | that the endswitch numbers are negative */
|
---|
900 | poslsw = -(GET I_POSLIMITSW)
|
---|
901 | neglsw = -(GET I_NEGLIMITSW)
|
---|
902 | if (IN poslsw == 0) then
|
---|
903 | PRINT "Positive endswitch activated at position ", APOS
|
---|
904 | SET I_POSLIMITSW 0
|
---|
905 | ERRCLR
|
---|
906 | CVEL (vres%100) /* 1% */
|
---|
907 | ACC (10*vres%100) /* 10% */
|
---|
908 | DEC (10*vres%100) /* 10% */
|
---|
909 | CSTART
|
---|
910 | WHILE (IN poslsw == 0) DO ENDWHILE
|
---|
911 | CSTOP
|
---|
912 | SET I_POSLIMITSW -poslsw
|
---|
913 | errinf = 1
|
---|
914 | elseif (IN neglsw == 0) then
|
---|
915 | PRINT "Negative endswitch activated at position ", APOS
|
---|
916 | SET I_NEGLIMITSW 0
|
---|
917 | ERRCLR
|
---|
918 | vres = GET VELRES
|
---|
919 | CVEL -(vres%100) /* 1% */
|
---|
920 | ACC (10*vres%100) /* 10% */
|
---|
921 | DEC (10*vres%100) /* 10% */
|
---|
922 | CSTART
|
---|
923 | WHILE (IN neglsw == 0) DO ENDWHILE
|
---|
924 | CSTOP
|
---|
925 | SET I_NEGLIMITSW -neglsw
|
---|
926 | errinf = -1
|
---|
927 | endif
|
---|
928 | elseif (errlist[2]==84) then
|
---|
929 | PRINT "Too many (>12) ON TIME interrupts."
|
---|
930 | ERRCLR
|
---|
931 | elseif (errlist[2]==89) then
|
---|
932 | PRINT "CAN I/O error."
|
---|
933 | errinf = REOPEN 1 0
|
---|
934 | ERRCLR
|
---|
935 | ELSE
|
---|
936 | PRINT "Error Function Called: ERRNO=", errlist[2]
|
---|
937 | endif
|
---|
938 |
|
---|
939 | /* tell the bus what exactly happened */
|
---|
940 | CANOUT pdo2 errlist[2] errinf
|
---|
941 | RETURN
|
---|
942 |
|
---|
943 | /*-------------------------------------------------------------------------*/
|
---|
944 | /* End of part for Programs called with GOSUB */
|
---|
945 | /*-------------------------------------------------------------------------*/
|
---|
946 |
|
---|
947 | ENDPROG
|
---|