QuakeGod
2023-02-01 2364f06313efb8eb97d4ae741a31cdd2efc175d2
提交 | 用户 | age
bfc108 1 /**
Q 2   ******************************************************************************
3   * @file           : PLCfunctions.c
4   * @brief          : PLC funcstions program body
5   ******************************************************************************
6     */
7 //#include "globaldef.h"
8 #include "PLCfunctions.h"
9 #include "string.h"
10 #include "stm32f0xx_hal.h"
11 #include <core_cmInstr.h>
12
13 extern __IO uint32_t uwTick;
14
15 //unsigned short WDFs[TOTAL_WDFS];
16
17 inline unsigned int GetTick(void)
18 {
19 //    unsigned short Clk1=SysTick->VAL;
20         return uwTick;
21 }
22
23 //unsigned char CurVAL;
24 //unsigned char CurVALs[TOTAL_CurVAL];
25 //stTimer Timers[TOTALTIMERS];
26
27 //unsigned short WX[13];
28 //unsigned short WY[13];
29 //unsigned short WR[64];
30 //unsigned short DT[256];
31
32 //unsigned short SDT[256];
33
34 stPLCMem PLCMem;
35
36 int InitTimer(int nIndex, int nType)
37 {
38     if (nIndex >= TOTALTIMERS) return -1;
a7db3c 39     KMem.Timers[nIndex].StatByte = 0x0010 | nType;
bfc108 40 //    Timers[nIndex].nType = 0;
a7db3c 41     KMem.SV[nIndex] = 0;
Q 42     KMem.EV[nIndex] = 0;
43     KMem.Timers[nIndex].LastActTime = GetTick();
bfc108 44     return 0;
Q 45 }
46
a7db3c 47 int RunTimer(int nIndex , int SV)
bfc108 48 {
Q 49     if (nIndex >= TOTALTIMERS) return -1;    
a7db3c 50     if (!KMem.Timers[nIndex].bSet)
bfc108 51     {
a7db3c 52         KMem.SV[nIndex] = SV;
Q 53         KMem.EV[nIndex]= 0;
54         KMem.Timers[nIndex].LastActTime = GetTick();        
55         KMem.Timers[nIndex].bSet = 1;
bfc108 56     }
Q 57     return 0;
58 }
59
60 int StopTimer(int nIndex)
61 {
62     if (nIndex >= TOTALTIMERS) return -1;    
a7db3c 63     if (KMem.Timers[nIndex].bSet)
bfc108 64     {
a7db3c 65         KMem.EV[nIndex] = 0;
Q 66         KMem.Timers[nIndex].LastActTime = GetTick();        
67         KMem.Timers[nIndex].bSet = 0;        
bfc108 68     }
Q 69     return 0;
70 }
71 int ResetTimer(int nIndex)
72 {
73     if (nIndex >= TOTALTIMERS) return -1;
a7db3c 74     KMem.EV[nIndex] = 0;
Q 75     KMem.Timers[nIndex].bTon = 0;
76     KMem.Timers[nIndex].LastActTime=GetTick();
bfc108 77     return 0;
Q 78 }
79
80 int SetTimerValue(int nIndex, int bSet, int SV)
81 {
82     if (nIndex >= TOTALTIMERS) return -1;    
a7db3c 83     if (bSet) {RunTimer(nIndex, SV);}
bfc108 84     else {StopTimer(nIndex);}
a7db3c 85     return KMem.Timers[nIndex].bTon;
bfc108 86 }
Q 87
88 int ProcessTimer(int nIndex)
89 {
90     if (nIndex >= TOTALTIMERS) return -1;
a7db3c 91     if (!KMem.Timers[nIndex].nInited) return 0;
Q 92     if (KMem.Timers[nIndex].bSet)        // bSet =1;
bfc108 93     {
a7db3c 94         if (!KMem.Timers[nIndex].bTon)
bfc108 95         {
a7db3c 96             int TimeDiff = GetTick() - KMem.Timers[nIndex].LastActTime;
bfc108 97             int nScale = TICK_OF_MS;
a7db3c 98             if (KMem.Timers[nIndex].nScale == 0)
bfc108 99             {nScale = TICK_OF_MS;
a7db3c 100             }else if (KMem.Timers[nIndex].nScale == 1)
bfc108 101             {nScale = TICK_OF_RS;
a7db3c 102             }else if (KMem.Timers[nIndex].nScale == 2)
bfc108 103             {nScale = TICK_OF_XS;
a7db3c 104             }else if (KMem.Timers[nIndex].nScale == 3)
bfc108 105             {nScale = TICK_OF_YS;
Q 106             }else {}
107             
108             
109             if (TimeDiff >= nScale)
110             {
111                 int TimeDiffmS = TimeDiff / nScale;
a7db3c 112                 unsigned short NextEV = KMem.EV[nIndex] + TimeDiffmS;
Q 113                 KMem.Timers[nIndex].LastActTime += TimeDiffmS*nScale;
bfc108 114                 
a7db3c 115                 if (NextEV >= KMem.SV[nIndex]) 
bfc108 116                 {
a7db3c 117                     NextEV = KMem.SV[nIndex];
Q 118                     KMem.Timers[nIndex].bTon =1;
bfc108 119                 }
a7db3c 120                 KMem.EV[nIndex] = NextEV;
bfc108 121             }
Q 122         }
123     }else         //bSet=0;
124     {
a7db3c 125         if(KMem.Timers[nIndex].bTon) 
bfc108 126         {
a7db3c 127             KMem.Timers[nIndex].bTon =    0;
bfc108 128         }
Q 129     }
a7db3c 130     SetCoilValue(KLCoilTypeT, nIndex, KMem.Timers[nIndex].bTon);    
Q 131     return KMem.Timers[nIndex].bTon;
bfc108 132 }
Q 133
134 int IsTimerOn(int nIndex)
135 {
136     if (nIndex >= TOTALTIMERS) return 0;
137     ProcessTimer(nIndex);
a7db3c 138     return KMem.Timers[nIndex].bTon;
bfc108 139
Q 140 }
141
142 int GetTimerSV(int nIndex)
143 {
144     if (nIndex >= TOTALTIMERS) return 0;
145 //    ProcessTimer(nIndex);    
a7db3c 146     return KMem.SV[nIndex];
bfc108 147 //    return 0;    
Q 148 }
149 int GetTimerEV(int nIndex)
150 {
151     if (nIndex >= TOTALTIMERS) return 0;
152 //    ProcessTimer(nIndex);    
a7db3c 153     return KMem.EV[nIndex];
bfc108 154 //    return 0;
Q 155 }
156
157 int PushInVal(void)
158 {
a7db3c 159     for (int i=TOTAL_CurVAL -1 ;i>0;i--)
bfc108 160     {
a7db3c 161         KMem.CurVALs[i]=KMem.CurVALs[i-1];
bfc108 162     }
a7db3c 163     KMem.CurVALs[0]=KMem.CurVAL;
Q 164     return KMem.CurVAL;
bfc108 165 }
Q 166
167 int PopOutVal(void)
168 {
a7db3c 169     unsigned char theVAL=KMem.CurVALs[0];
bfc108 170     for (int i=0;i<TOTAL_CurVAL-1;i++)
Q 171     {
a7db3c 172         KMem.CurVALs[i]=KMem.CurVALs[i+1];
bfc108 173     }
Q 174     return theVAL;
175 }
176
a7db3c 177 stBinProg1 const  prog1[]= //__attribute__((at(0X8008000)))
bfc108 178 {
a7db3c 179     {OP_ST,KLCoilTypeSR,13},
Q 180     {OP_MV,0,50},    {KLDataTypeDEC,KLDataTypeDT,1},
181     {OP_MV,0,20},    {KLDataTypeDEC,KLDataTypeDT,2},
182     {OP_MV,0,30},    {KLDataTypeDEC,KLDataTypeDT,3},
183     {OP_MV,0,40},    {KLDataTypeDEC,KLDataTypeDT,4},
184     {OP_SET,KLCoilTypeR,0},
185 //    {OP_SET,KLCoilTypeY,0},
bfc108 186     
a7db3c 187     {OP_ST,KLCoilTypeR,0},
Q 188         {OP_TMX,1,1},    {KLDataTypeDT,0,0},
189         {OP_DF},
190         {OP_SET,KLCoilTypeR,10},
191         
192     {OP_ST,KLCoilTypeX,0},
193     {OP_DF},
194     {OP_SET,KLCoilTypeR,10},
bfc108 195
a7db3c 196     {OP_ST,KLCoilTypeX,1},
Q 197     {OP_DF},
198     {OP_RESET,KLCoilTypeR,10},
199 /*    
200     {OP_ST,KLCoilTypeR,10},
201     {OP_AN,KLCoilTypeR,51},    
202     {OP_AN,KLCoilTypeR,52},    
203     {OP_AN,KLCoilTypeR,53},    
204     {OP_ADD3,0,21},        {KLDataTypeDT,KLDataTypeDT,31},        {0,KLDataTypeDT,32},    
205
206     {OP_ST,KLCoilTypeR,10},
207     {OP_AN,KLCoilTypeR,54},    
208     {OP_AN,KLCoilTypeR,55},    
209     {OP_AN,KLCoilTypeR,56},    
210     {OP_ADD3,0,23},        {KLDataTypeDT,KLDataTypeDT,33},        {0,KLDataTypeDT,34},    
211 */
212     {OP_ST,KLCoilTypeSR,1},
213     {OP_PSHS},    
214     {OP_AN,KLCoilTypeR,51},    
215     {OP_OUT,KLCoilTypeY,1},    
216     {OP_RDS},    
217     {OP_AN,KLCoilTypeR,52},    
218     {OP_OUT,KLCoilTypeY,2},    
219     {OP_RDS},    
220     {OP_AN,KLCoilTypeR,53},    
221     {OP_OUT,KLCoilTypeY,3},    
222     {OP_RDS},    
223     {OP_AN,KLCoilTypeR,54},    
224     {OP_OUT,KLCoilTypeY,4},    
225     {OP_RDS},    
226     {OP_AN,KLCoilTypeR,55},    
227     {OP_OUT,KLCoilTypeY,5},    
228     {OP_POPS},    
229     {OP_AN,KLCoilTypeR,56},    
230     {OP_OUT,KLCoilTypeY,6},    
231
232     {OP_ST,KLCoilTypeR,10},
233     {OP_DF},
234     {OP_PSHS},    
235     {OP_MV,0,150},    {KLDataTypeDEC,KLDataTypeDT,11},
236     {OP_MV,0,30},    {KLDataTypeDEC,KLDataTypeDT,12},
237     {OP_RDS},    
238     {OP_MV,0,150},    {KLDataTypeDEC,KLDataTypeDT,13},
239     {OP_MV,0,30},    {KLDataTypeDEC,KLDataTypeDT,14},
240     {OP_POPS},    
241     {OP_AN_,KLCoilTypeR,11},    
242     {OP_AN_,KLCoilTypeR,12},    
243     {OP_AN_,KLCoilTypeR,13},    
244     {OP_AN_,KLCoilTypeR,14},    
245     {OP_SET,KLCoilTypeR,14},
246
247     {OP_ST,KLCoilTypeR,10},
248     {OP_PSHS},    
249     {OP_AN,KLCoilTypeR,11},    
250     {OP_DF},
251     {OP_SET,KLCoilTypeR,51},
252     {OP_RESET,KLCoilTypeR,52},
253     {OP_RESET,KLCoilTypeR,53},
254     {OP_RESET,KLCoilTypeR,54},
255     {OP_RESET,KLCoilTypeR,55},
256     {OP_SET,KLCoilTypeR,56},
257
258     {OP_RDS},    
259     {OP_AN,KLCoilTypeR,11},    
260
261         {OP_PSHS},    
262         {OP_TMX,11,11},    {KLDataTypeDT,0,0},
263             {OP_RESET,KLCoilTypeR,11},
264             {OP_SET,KLCoilTypeR,12},
265         {OP_POPS},
266             {OP_SUB3,0,11},        {KLDataTypeSV,KLDataTypeEV,11},        {0,KLDataTypeDT,21},        
267             {OP_AN_LE,0,21},{KLDataTypeDT,KLDataTypeDEC,30},
268             {OP_PSHS},    
269                 {OP_DIV,0,21},        {KLDataTypeDT,KLDataTypeDEC,10},        {0,KLDataTypeDT,31},        
270             {OP_RDS},    
271                 {OP_AN_GE,0,32},{KLDataTypeDT,KLDataTypeDEC,5},
272                 {OP_SET,KLCoilTypeR,51},
273             {OP_POPS},
274                 {OP_AN_LT,0,32},{KLDataTypeDT,KLDataTypeDEC,5},
275                 {OP_RESET,KLCoilTypeR,51},
276     {OP_RDS},            
277     {OP_AN,KLCoilTypeR,12},    
278     {OP_DF},
279     {OP_RESET,KLCoilTypeR,51},
280     {OP_SET,KLCoilTypeR,52},
281     {OP_RDS},            
282     {OP_AN,KLCoilTypeR,12},    
283         {OP_TMX,12,12},    {KLDataTypeDT,0,0},
284             {OP_RESET,KLCoilTypeR,12},
285             {OP_SET,KLCoilTypeR,13},
286     {OP_POPS},
287     {OP_AN,KLCoilTypeR,12},    
288     {OP_OUT,KLCoilTypeR,52},    
bfc108 289     
a7db3c 290     {OP_ST,KLCoilTypeR,10},
Q 291     {OP_PSHS},    
292     {OP_AN,KLCoilTypeR,13},    
293     {OP_DF},
294     {OP_RESET,KLCoilTypeR,52},
295     {OP_SET,KLCoilTypeR,53},
296     {OP_SET,KLCoilTypeR,54},
297     {OP_RESET,KLCoilTypeR,56},
298     {OP_RDS},    
299     {OP_AN,KLCoilTypeR,13},    
300         {OP_TMX,13,13},    {KLDataTypeDT,0,0},
301             {OP_RESET,KLCoilTypeR,13},
302             {OP_SET,KLCoilTypeR,14},
303     {OP_RDS},    
304     {OP_AN,KLCoilTypeR,13},    
305             {OP_SUB3,0,13},        {KLDataTypeSV,KLDataTypeEV,13},        {0,KLDataTypeDT,23},        
306             {OP_AN_LE,0,23},{KLDataTypeDT,KLDataTypeDEC,30},
307             {OP_PSHS},    
308                 {OP_DIV,0,23},        {KLDataTypeDT,KLDataTypeDEC,10},        {0,KLDataTypeDT,33},        
309             {OP_RDS},    
310                 {OP_AN_GE,0,34},{KLDataTypeDT,KLDataTypeDEC,5},
311                 {OP_SET,KLCoilTypeR,54},
312             {OP_POPS},
313                 {OP_AN_LT,0,34},{KLDataTypeDT,KLDataTypeDEC,5},
314                 {OP_RESET,KLCoilTypeR,54},
315     {OP_RDS},    
316     {OP_AN,KLCoilTypeR,14},    
317     {OP_DF},
318     {OP_RESET,KLCoilTypeR,54},
319     {OP_SET,KLCoilTypeR,55},
320     {OP_POPS},
321     {OP_AN,KLCoilTypeR,14},    
322         {OP_TMX,14,14},    {KLDataTypeDT,0,0},
323             {OP_RESET,KLCoilTypeR,14},
324             {OP_SET,KLCoilTypeR,11},
bfc108 325 };
Q 326
327 /*
328
329     {OP_ST,Addr_R,1},
330     {OP_PSHS,0,0},
331     {OP_AN_,Addr_Y,1},
332     {OP_TMR,5,200},
333     {OP_SET,Addr_Y,1},
334     {OP_POPS,0,0},
335     {OP_AN,Addr_Y,1},
336     {OP_TMR,6,200},
337     {OP_RESET,Addr_Y,1},
338
339 */
a7db3c 340 int nSizeProg1=sizeof(prog1)/sizeof(stBinProg1);
bfc108 341
a7db3c 342 int InitPLC()
bfc108 343 {
a7db3c 344     PLCMem.nScanCount=0;
Q 345     for (int i=0;i<1024;i++){PLCMem.ProgTrace[i]=0;}
346     for (int i=0;i<16;i++)    {
347         KMem.WR[i]=0;
348     }
349     for (int i=0;i<256;i++)    {
350         KMem.DT[i]=0;
351     }
bfc108 352     return 0;
Q 353 }
354
a7db3c 355 int StartPLC()
bfc108 356 {
a7db3c 357     PLCMem.nScanCount = 0;
Q 358     for (int i=0;i<1024;i++){PLCMem.ProgTrace[i]=0;}
359     for (int i=0;i<16;i++)    {
360         KMem.WR[i]=0;
361     }
362     for (int i=0;i<256;i++)    {
363         KMem.DT[i]=0;
364     }
365     PLCMem.bPLCRunning=1;
366     return 0;
bfc108 367 }
a7db3c 368
Q 369 int StopPLC()
bfc108 370 {
a7db3c 371     PLCMem.bPLCRunning=0;
Q 372     return 0;
bfc108 373 }
a7db3c 374
0fe6b0 375 inline void SetAddrBit(unsigned short * pW, unsigned char bitAddr)
Q 376 {
377     (*pW)|=1<<(bitAddr&0xf);
378 }
379
380 inline void ResetBit(unsigned short * pW, unsigned char bitAddr)
381 {
382     (*pW)&=~(1<<(bitAddr&0xf));
383 }
384 static inline void SetBitValue(unsigned short * pW, unsigned char bitAddr, unsigned char Value)
385 {
386     if (Value)    {    SetAddrBit(pW, bitAddr);}
387     else {ResetBit(pW, bitAddr);}
388 }
389
390 static inline unsigned char GetBitValue(unsigned short W, unsigned char bitAddr)
391 {
392     if (W&(1<<(bitAddr&0xf))) return 1;
393     else return 0;
394 }
395
396
a7db3c 397 int ProcessPLCBinProg(const stBinProg1 * pBinprog, int nStepSize)
bfc108 398 {
a7db3c 399     if (!PLCMem.bPLCRunning) return 0;
Q 400     
401     if (PLCMem.nScanCount == 0) {
402         SetCoilValue(KLCoilTypeSR, 13, 1);
403         SetCoilValue(KLCoilTypeSR, 0, 0);
404         SetCoilValue(KLCoilTypeSR, 1, 1);
405     }
406     else
bfc108 407     {
a7db3c 408         SetCoilValue(KLCoilTypeSR, 13, 0);
Q 409         SetCoilValue(KLCoilTypeSR, 0, 0);
410         SetCoilValue(KLCoilTypeSR, 1, 1);
411     }
412     for (int i = 0; i < TOTAL_CurVAL; i++) {
413         KMem.CurVALs[i] = 0;
414     }
415     int CurPos = 0;
416 //    stBinProg1 * pBinProg1;
417     stBinProg15 * pBinProg15;
418     stBinProg2 * pBinProg2;
419     stBinProg3 * pBinProg3;
420
421     int lastScanInputVal = 1;//??????,????????,? ?? ???
422
423     while (CurPos < nStepSize)
424     {
425         unsigned int nNextPos = 1;
426         unsigned int thisOP = pBinprog[CurPos].nOp;
427 //        unsigned int nParamCount = 0
428         unsigned char thisAddrType = pBinprog[CurPos].nParamType;
429         unsigned short thisAddr = pBinprog[CurPos].nParamAddr;
430
431
bfc108 432         switch (thisOP)
Q 433         {
a7db3c 434 //        case OP_NONE:
Q 435 //            break;
436         case OP_NOP:
437             break;
438             //??? ??
439         case OP_NOT:
440         case OP_ANS:
441         case OP_ORS:
442         case OP_PSHS:
443         case OP_RDS:
444         case OP_POPS:
445         case OP_DF:
446         case OP_DF_:
447             switch (thisOP)
448             {
bfc108 449             case OP_NOT:
a7db3c 450                 KMem.CurVAL = !KMem.CurVAL;
Q 451                 break;
452             case OP_ANS:
453                 KMem.CurVAL = PopOutVal() && KMem.CurVAL;
454                 break;
455             case OP_ORS:
456                 KMem.CurVAL = PopOutVal() || KMem.CurVAL;
bfc108 457                 break;
Q 458             case OP_PSHS:
459                 PushInVal();
460                 break;
a7db3c 461             case OP_RDS:
Q 462                 KMem.CurVAL = KMem.CurVALs[0] != 0;
463                 break;
bfc108 464             case OP_POPS:
a7db3c 465                 KMem.CurVAL = PopOutVal();
bfc108 466                 break;
a7db3c 467             case OP_DF:
Q 468                 KMem.CurVAL = KMem.CurVAL && !lastScanInputVal;
bfc108 469                 break;
a7db3c 470             case OP_DF_:
Q 471                 KMem.CurVAL = !KMem.CurVAL && lastScanInputVal;
bfc108 472                 break;
Q 473
474             default:
475                 break;
a7db3c 476             }
Q 477             break;
478             // 1????
479         case OP_ST:
480         case OP_ST_:
481         case OP_AN:
482         case OP_AN_:
483         case OP_OR:
484         case OP_OR_:
485             switch (thisOP)
486             {
487             case OP_ST:
488                 PushInVal();
489                 KMem.CurVAL = GetCoilValue(thisAddrType, thisAddr);
490                 break;
491             case OP_ST_:
492                 PushInVal();
493                 KMem.CurVAL = !GetCoilValue(thisAddrType, thisAddr);
494                 break;
495             case OP_AN:
496                 KMem.CurVAL = KMem.CurVAL&&GetCoilValue(thisAddrType, thisAddr);
497                 break;
498             case OP_AN_:
499                 KMem.CurVAL = KMem.CurVAL && (!GetCoilValue(thisAddrType, thisAddr));
500                 break;
501             case OP_OR:
502                 KMem.CurVAL = KMem.CurVAL || GetCoilValue(thisAddrType, thisAddr);
503                 break;
504             case OP_OR_:
505                 KMem.CurVAL = KMem.CurVAL || (!GetCoilValue(thisAddrType, thisAddr));
506                 break;
507             default:
508                 break;
509             }
510             break;
511             // 1 ?? ??
512         case OP_OUT:
513         case OP_SET:
514         case OP_RESET:
515             switch (thisOP)
516             {
517             case OP_OUT:
518                 SetCoilValue(thisAddrType, thisAddr, KMem.CurVAL);
519                 break;
520             case OP_SET:
521                 if (KMem.CurVAL) SetCoilValue(thisAddrType, thisAddr, 1);
522                 break;
523             case OP_RESET:
524                 if (KMem.CurVAL) SetCoilValue(thisAddrType, thisAddr, 0);
525                 break;
526             default:
527                 break;
528             }
529             break;
530             // ????
531         case OP_ST_EQ:
532         case OP_ST_NE:
533         case OP_ST_LT:
534         case OP_ST_GT:
535         case OP_ST_LE:
536         case OP_ST_GE:
537         case OP_AN_EQ:
538         case OP_AN_NE:
539         case OP_AN_LT:
540         case OP_AN_GT:
541         case OP_AN_LE:
542         case OP_AN_GE:
543         case OP_OR_EQ:
544         case OP_OR_NE:
545         case OP_OR_LT:
546         case OP_OR_GT:
547         case OP_OR_LE:
548         case OP_OR_GE:
549             pBinProg2 = (stBinProg2 *)&pBinprog[CurPos];
550             thisAddrType = pBinProg2->nParamType1;
551
552             switch (thisOP)
553             {
554             case OP_ST_EQ:
555                 PushInVal();
556                 KMem.CurVAL = (GetVarData(thisAddrType, thisAddr) == GetVarData(pBinProg2->nParamType2, pBinProg2->nParamAddr2));
557                 break;
558             case OP_ST_NE:
559                 PushInVal();
560                 KMem.CurVAL = (GetVarData(thisAddrType, thisAddr) != GetVarData(pBinProg2->nParamType2, pBinProg2->nParamAddr2));
561                 break;
562             case OP_ST_LT:
563                 PushInVal();
564                 KMem.CurVAL = (GetVarData(thisAddrType, thisAddr) < GetVarData(pBinProg2->nParamType2, pBinProg2->nParamAddr2));
565                 break;
566             case OP_ST_GT:
567                 PushInVal();
568                 KMem.CurVAL = (GetVarData(thisAddrType, thisAddr) > GetVarData(pBinProg2->nParamType2, pBinProg2->nParamAddr2));
569                 break;
570             case OP_ST_LE:
571                 PushInVal();
572                 KMem.CurVAL = (GetVarData(thisAddrType, thisAddr) <= GetVarData(pBinProg2->nParamType2, pBinProg2->nParamAddr2));
573                 break;
574             case OP_ST_GE:
575                 PushInVal();
576                 KMem.CurVAL = (GetVarData(thisAddrType, thisAddr) >= GetVarData(pBinProg2->nParamType2, pBinProg2->nParamAddr2));
577                 break;
578             case OP_AN_EQ:
579                 KMem.CurVAL = KMem.CurVAL && (GetVarData(thisAddrType, thisAddr) == GetVarData(pBinProg2->nParamType2, pBinProg2->nParamAddr2));
580                 break;
581             case OP_AN_NE:
582                 KMem.CurVAL = KMem.CurVAL && (GetVarData(thisAddrType, thisAddr) != GetVarData(pBinProg2->nParamType2, pBinProg2->nParamAddr2));
583                 break;
584             case OP_AN_LT:
585                 KMem.CurVAL = KMem.CurVAL && (GetVarData(thisAddrType, thisAddr) < GetVarData(pBinProg2->nParamType2, pBinProg2->nParamAddr2));
586                 break;
587             case OP_AN_GT:
588                 KMem.CurVAL = KMem.CurVAL && (GetVarData(thisAddrType, thisAddr) > GetVarData(pBinProg2->nParamType2, pBinProg2->nParamAddr2));
589                 break;
590             case OP_AN_LE:
591                 KMem.CurVAL = KMem.CurVAL && (GetVarData(thisAddrType, thisAddr) <= GetVarData(pBinProg2->nParamType2, pBinProg2->nParamAddr2));
592                 break;
593             case OP_AN_GE:
594                 KMem.CurVAL = KMem.CurVAL && (GetVarData(thisAddrType, thisAddr) >= GetVarData(pBinProg2->nParamType2, pBinProg2->nParamAddr2));
595                 break;
596
597             case OP_OR_EQ:
598                 KMem.CurVAL = KMem.CurVAL || (GetVarData(thisAddrType, thisAddr) == GetVarData(pBinProg2->nParamType2, pBinProg2->nParamAddr2));
599                 break;
600             case OP_OR_NE:
601                 KMem.CurVAL = KMem.CurVAL || (GetVarData(thisAddrType, thisAddr) != GetVarData(pBinProg2->nParamType2, pBinProg2->nParamAddr2));
602                 break;
603             case OP_OR_LT:
604                 KMem.CurVAL = KMem.CurVAL || (GetVarData(thisAddrType, thisAddr) < GetVarData(pBinProg2->nParamType2, pBinProg2->nParamAddr2));
605                 break;
606             case OP_OR_GT:
607                 KMem.CurVAL = KMem.CurVAL || (GetVarData(thisAddrType, thisAddr) > GetVarData(pBinProg2->nParamType2, pBinProg2->nParamAddr2));
608                 break;
609             case OP_OR_LE:
610                 KMem.CurVAL = KMem.CurVAL || (GetVarData(thisAddrType, thisAddr) <= GetVarData(pBinProg2->nParamType2, pBinProg2->nParamAddr2));
611                 break;
612             case OP_OR_GE:
613                 KMem.CurVAL = KMem.CurVAL || (GetVarData(thisAddrType, thisAddr) >= GetVarData(pBinProg2->nParamType2, pBinProg2->nParamAddr2));
614                 break;
615
616             default:
617                 break;
618             }
619             nNextPos = 2;
620             break;
621             // ???
622         case OP_TML:
623         case OP_TMR:
624         case OP_TMX:
625         case OP_TMY:
626             pBinProg15 = (stBinProg15 *)(&pBinprog[CurPos]);
627             {
628                 unsigned char thisNum= pBinProg15->nOpNum;
629                 thisAddrType = pBinProg15->nParamType1;
630                 thisAddr = pBinProg15->nParamAddr1;
631                 switch (thisOP)
632                 {
633                 case OP_TML:
634                     if (!KMem.Timers[thisNum].nInited) InitTimer(thisNum, 0);
635                     if (KMem.CurVAL) RunTimer(thisNum, GetVarData(thisAddrType, thisAddr));
636                     else StopTimer(thisNum);
637                     KMem.CurVAL = ProcessTimer(thisNum);
638
639                     break;
640                 case OP_TMR:
641                     if (!KMem.Timers[thisNum].nInited) InitTimer(thisNum, 1);
642                     if (KMem.CurVAL) RunTimer(thisNum, GetVarData(thisAddrType, thisAddr));
643                     else StopTimer(thisNum);
644                     KMem.CurVAL = ProcessTimer(thisNum);
645                     break;
646                 case OP_TMX:
647                     if (!KMem.Timers[thisNum].nInited) InitTimer(thisNum, 2);
648                     if (KMem.CurVAL) RunTimer(thisNum, GetVarData(thisAddrType, thisAddr));
649                     else StopTimer(thisNum);
650                     KMem.CurVAL = ProcessTimer(thisNum);
651
652                     break;
653                 case OP_TMY:
654                     if (!KMem.Timers[thisNum].nInited) InitTimer(thisNum, 3);
655                     if (KMem.CurVAL) RunTimer(thisNum, GetVarData(thisAddrType, thisAddr));
656                     else StopTimer(thisNum);
657                     KMem.CurVAL = ProcessTimer(thisNum);
658                     break;
659                 default:
660                     break;
661                 }
662
663             }
664             nNextPos = 2;
665             break;
666             // 1 ??????
667         case OP_INC:
668         case OP_DEC:
669             pBinProg15 = (stBinProg15 *)(&pBinprog[CurPos]);
670             thisAddrType = pBinProg15->nParamType1;
671             thisAddr = pBinProg15->nParamAddr1;
672             nNextPos = 2;
673             switch (thisOP)
674             {
675             case OP_INC:
676                 if (KMem.CurVAL) SetVarData(thisAddrType, thisAddr, GetVarData(thisAddrType, thisAddr) + 1);
677                 break;
678             case OP_DEC:
679                 if (KMem.CurVAL) SetVarData(thisAddrType, thisAddr, GetVarData(thisAddrType, thisAddr) - 1);
680                 break;
681
682             default:
683                 break;
684             }
685             break;
686             // 2??????
687         case OP_MV:
688         case OP_ADD2:
689         case OP_SUB2:
690             pBinProg2 = (stBinProg2 *)(&pBinprog[CurPos]);
691             {
692                 int nParamType2, nParamAddr2;
693                 thisAddrType = pBinProg2->nParamType1;
694                 thisAddr = pBinProg2->nParamAddr1;
695                 nParamType2 = pBinProg2->nParamType2;
696                 nParamAddr2 = pBinProg2->nParamAddr2;
697
698                 switch (thisOP)
699                 {
700                 case OP_MV:
701                     if (KMem.CurVAL) SetVarData(nParamType2, nParamAddr2, GetVarData(thisAddrType, thisAddr));
702                     break;
703                 case OP_ADD2:
704                     if (KMem.CurVAL) SetVarData(nParamType2, nParamAddr2, GetVarData(thisAddrType, thisAddr) + GetVarData(nParamType2, nParamAddr2));
705                     break;
706                 case OP_SUB2:
707                     if (KMem.CurVAL) SetVarData(nParamType2, nParamAddr2, GetVarData(nParamType2, nParamAddr2) - GetVarData(thisAddrType, thisAddr));
708                     break;
709
710                 default:
711                     break;
712                 }
713
714             }
715             nNextPos = 2;
716             break;
717             // 3 ??????
718         case OP_ADD3:
719         case OP_SUB3:
720         case OP_MUL:
721         case OP_DIV:
722             pBinProg3 = (stBinProg3 *)(&pBinprog[CurPos]);
723             int nParamType2, nParamAddr2;
724             int nParamType3, nParamAddr3;
725             thisAddrType = pBinProg3->nParamType1;
726             thisAddr = pBinProg3->nParamAddr1;
727             nParamType2 = pBinProg3->nParamType2;
728             nParamAddr2 = pBinProg3->nParamAddr2;
729             nParamType3 = pBinProg3->nParamType3;
730             nParamAddr3 = pBinProg3->nParamAddr3;
731             switch (thisOP)
732             {
733             case OP_ADD3:
734                 if (KMem.CurVAL) SetVarData(nParamType3, nParamAddr3, GetVarData(thisAddrType, thisAddr) + GetVarData(nParamType2, nParamAddr2));
735                 break;
736             case OP_SUB3:
737                 if (KMem.CurVAL) SetVarData(nParamType3, nParamAddr3, GetVarData(thisAddrType, thisAddr) - GetVarData(nParamType2, nParamAddr2));
738                 break;
739             case OP_MUL:
740                 if (KMem.CurVAL) {
741                     short multiplicand = GetVarData(thisAddrType, thisAddr);
742                     short multiplier = GetVarData(nParamType2, nParamAddr2);
743                     int product = multiplicand * multiplier;
744                     SetVarData(nParamType3, nParamAddr3, product);
745                     SetVarData(nParamType3, nParamAddr3 + 1, product >> 16);
746                 }
747                 break;
748             case OP_DIV:
749                 if (KMem.CurVAL) {
750                     short dividend = GetVarData(thisAddrType, thisAddr);
751                     short divisor = GetVarData(nParamType2, nParamAddr2);
752                     short quotient = dividend / divisor;
753                     short remainder = dividend % divisor;
754                     SetVarData(nParamType3, nParamAddr3, quotient);
755                     SetVarData(nParamType3, nParamAddr3 + 1, remainder);
756                 }
757                 break;
758
759             default:
760                 break;
761             }
762             nNextPos = 3;
763             break;
764
765         default:
766             break;
bfc108 767         }
a7db3c 768
0fe6b0 769         lastScanInputVal =  PLCMem.ProgTrace[CurPos]; //GetBitValue(KMem.WDFs);
a7db3c 770         PLCMem.ProgTrace[CurPos] = KMem.CurVAL;
Q 771         CurPos += nNextPos;
bfc108 772     }
a7db3c 773     PLCMem.nScanCount++;
bfc108 774     return 0;
Q 775 }