1 // SPDX-License-Identifier: GPL-2.0+ 2 /* 3 * Porting to u-boot: 4 * 5 * (C) Copyright 2010 6 * Stefano Babic, DENX Software Engineering, sbabic@denx.de. 7 * 8 * Lattice ispVME Embedded code to load Lattice's FPGA: 9 * 10 * Copyright 2009 Lattice Semiconductor Corp. 11 * 12 * ispVME Embedded allows programming of Lattice's suite of FPGA 13 * devices on embedded systems through the JTAG port. The software 14 * is distributed in source code form and is open to re - distribution 15 * and modification where applicable. 16 * 17 * Revision History of ivm_core.c module: 18 * 4/25/06 ht Change some variables from unsigned short or int 19 * to long int to make the code compiler independent. 20 * 5/24/06 ht Support using RESET (TRST) pin as a special purpose 21 * control pin such as triggering the loading of known 22 * state exit. 23 * 3/6/07 ht added functions to support output to terminals 24 * 25 * 09/11/07 NN Type cast mismatch variables 26 * Moved the sclock() function to hardware.c 27 * 08/28/08 NN Added Calculate checksum support. 28 * 4/1/09 Nguyen replaced the recursive function call codes on 29 * the ispVMLCOUNT function 30 */ 31 32 #include <common.h> 33 #include <linux/string.h> 34 #include <malloc.h> 35 #include <lattice.h> 36 37 #define vme_out_char(c) printf("%c", c) 38 #define vme_out_hex(c) printf("%x", c) 39 #define vme_out_string(s) printf("%s", s) 40 41 /* 42 * 43 * Global variables used to specify the flow control and data type. 44 * 45 * g_usFlowControl: flow control register. Each bit in the 46 * register can potentially change the 47 * personality of the embedded engine. 48 * g_usDataType: holds the data type of the current row. 49 * 50 */ 51 52 static unsigned short g_usFlowControl; 53 unsigned short g_usDataType; 54 55 /* 56 * 57 * Global variables used to specify the ENDDR and ENDIR. 58 * 59 * g_ucEndDR: the state that the device goes to after SDR. 60 * g_ucEndIR: the state that the device goes to after SIR. 61 * 62 */ 63 64 unsigned char g_ucEndDR = DRPAUSE; 65 unsigned char g_ucEndIR = IRPAUSE; 66 67 /* 68 * 69 * Global variables used to support header/trailer. 70 * 71 * g_usHeadDR: the number of lead devices in bypass. 72 * g_usHeadIR: the sum of IR length of lead devices. 73 * g_usTailDR: the number of tail devices in bypass. 74 * g_usTailIR: the sum of IR length of tail devices. 75 * 76 */ 77 78 static unsigned short g_usHeadDR; 79 static unsigned short g_usHeadIR; 80 static unsigned short g_usTailDR; 81 static unsigned short g_usTailIR; 82 83 /* 84 * 85 * Global variable to store the number of bits of data or instruction 86 * to be shifted into or out from the device. 87 * 88 */ 89 90 static unsigned short g_usiDataSize; 91 92 /* 93 * 94 * Stores the frequency. Default to 1 MHz. 95 * 96 */ 97 98 static int g_iFrequency = 1000; 99 100 /* 101 * 102 * Stores the maximum amount of ram needed to hold a row of data. 103 * 104 */ 105 106 static unsigned short g_usMaxSize; 107 108 /* 109 * 110 * Stores the LSH or RSH value. 111 * 112 */ 113 114 static unsigned short g_usShiftValue; 115 116 /* 117 * 118 * Stores the current repeat loop value. 119 * 120 */ 121 122 static unsigned short g_usRepeatLoops; 123 124 /* 125 * 126 * Stores the current vendor. 127 * 128 */ 129 130 static signed char g_cVendor = LATTICE; 131 132 /* 133 * 134 * Stores the VME file CRC. 135 * 136 */ 137 138 unsigned short g_usCalculatedCRC; 139 140 /* 141 * 142 * Stores the Device Checksum. 143 * 144 */ 145 /* 08/28/08 NN Added Calculate checksum support. */ 146 unsigned long g_usChecksum; 147 static unsigned int g_uiChecksumIndex; 148 149 /* 150 * 151 * Stores the current state of the JTAG state machine. 152 * 153 */ 154 155 static signed char g_cCurrentJTAGState; 156 157 /* 158 * 159 * Global variables used to support looping. 160 * 161 * g_pucHeapMemory: holds the entire repeat loop. 162 * g_iHeapCounter: points to the current byte in the repeat loop. 163 * g_iHEAPSize: the current size of the repeat in bytes. 164 * 165 */ 166 167 unsigned char *g_pucHeapMemory; 168 unsigned short g_iHeapCounter; 169 unsigned short g_iHEAPSize; 170 static unsigned short previous_size; 171 172 /* 173 * 174 * Global variables used to support intelligent programming. 175 * 176 * g_usIntelDataIndex: points to the current byte of the 177 * intelligent buffer. 178 * g_usIntelBufferSize: holds the size of the intelligent 179 * buffer. 180 * 181 */ 182 183 unsigned short g_usIntelDataIndex; 184 unsigned short g_usIntelBufferSize; 185 186 /* 187 * 188 * Supported VME versions. 189 * 190 */ 191 192 const char *const g_szSupportedVersions[] = { 193 "__VME2.0", "__VME3.0", "____12.0", "____12.1", 0}; 194 195 /* 196 * 197 * Holds the maximum size of each respective buffer. These variables are used 198 * to write the HEX files when converting VME to HEX. 199 * 200 */ 201 202 static unsigned short g_usTDOSize; 203 static unsigned short g_usMASKSize; 204 static unsigned short g_usTDISize; 205 static unsigned short g_usDMASKSize; 206 static unsigned short g_usLCOUNTSize; 207 static unsigned short g_usHDRSize; 208 static unsigned short g_usTDRSize; 209 static unsigned short g_usHIRSize; 210 static unsigned short g_usTIRSize; 211 static unsigned short g_usHeapSize; 212 213 /* 214 * 215 * Global variables used to store data. 216 * 217 * g_pucOutMaskData: local RAM to hold one row of MASK data. 218 * g_pucInData: local RAM to hold one row of TDI data. 219 * g_pucOutData: local RAM to hold one row of TDO data. 220 * g_pucHIRData: local RAM to hold the current SIR header. 221 * g_pucTIRData: local RAM to hold the current SIR trailer. 222 * g_pucHDRData: local RAM to hold the current SDR header. 223 * g_pucTDRData: local RAM to hold the current SDR trailer. 224 * g_pucIntelBuffer: local RAM to hold the current intelligent buffer 225 * g_pucOutDMaskData: local RAM to hold one row of DMASK data. 226 * 227 */ 228 229 unsigned char *g_pucOutMaskData = NULL, 230 *g_pucInData = NULL, 231 *g_pucOutData = NULL, 232 *g_pucHIRData = NULL, 233 *g_pucTIRData = NULL, 234 *g_pucHDRData = NULL, 235 *g_pucTDRData = NULL, 236 *g_pucIntelBuffer = NULL, 237 *g_pucOutDMaskData = NULL; 238 239 /* 240 * 241 * JTAG state machine transition table. 242 * 243 */ 244 245 struct { 246 unsigned char CurState; /* From this state */ 247 unsigned char NextState; /* Step to this state */ 248 unsigned char Pattern; /* The tragetory of TMS */ 249 unsigned char Pulses; /* The number of steps */ 250 } g_JTAGTransistions[25] = { 251 { RESET, RESET, 0xFC, 6 }, /* Transitions from RESET */ 252 { RESET, IDLE, 0x00, 1 }, 253 { RESET, DRPAUSE, 0x50, 5 }, 254 { RESET, IRPAUSE, 0x68, 6 }, 255 { IDLE, RESET, 0xE0, 3 }, /* Transitions from IDLE */ 256 { IDLE, DRPAUSE, 0xA0, 4 }, 257 { IDLE, IRPAUSE, 0xD0, 5 }, 258 { DRPAUSE, RESET, 0xF8, 5 }, /* Transitions from DRPAUSE */ 259 { DRPAUSE, IDLE, 0xC0, 3 }, 260 { DRPAUSE, IRPAUSE, 0xF4, 7 }, 261 { DRPAUSE, DRPAUSE, 0xE8, 6 },/* 06/14/06 Support POLL STATUS LOOP*/ 262 { IRPAUSE, RESET, 0xF8, 5 }, /* Transitions from IRPAUSE */ 263 { IRPAUSE, IDLE, 0xC0, 3 }, 264 { IRPAUSE, DRPAUSE, 0xE8, 6 }, 265 { DRPAUSE, SHIFTDR, 0x80, 2 }, /* Extra transitions using SHIFTDR */ 266 { IRPAUSE, SHIFTDR, 0xE0, 5 }, 267 { SHIFTDR, DRPAUSE, 0x80, 2 }, 268 { SHIFTDR, IDLE, 0xC0, 3 }, 269 { IRPAUSE, SHIFTIR, 0x80, 2 },/* Extra transitions using SHIFTIR */ 270 { SHIFTIR, IRPAUSE, 0x80, 2 }, 271 { SHIFTIR, IDLE, 0xC0, 3 }, 272 { DRPAUSE, DRCAPTURE, 0xE0, 4 }, /* 11/15/05 Support DRCAPTURE*/ 273 { DRCAPTURE, DRPAUSE, 0x80, 2 }, 274 { IDLE, DRCAPTURE, 0x80, 2 }, 275 { IRPAUSE, DRCAPTURE, 0xE0, 4 } 276 }; 277 278 /* 279 * 280 * List to hold all LVDS pairs. 281 * 282 */ 283 284 LVDSPair *g_pLVDSList; 285 unsigned short g_usLVDSPairCount; 286 287 /* 288 * 289 * Function prototypes. 290 * 291 */ 292 293 static signed char ispVMDataCode(void); 294 static long int ispVMDataSize(void); 295 static void ispVMData(unsigned char *Data); 296 static signed char ispVMShift(signed char Code); 297 static signed char ispVMAmble(signed char Code); 298 static signed char ispVMLoop(unsigned short a_usLoopCount); 299 static signed char ispVMBitShift(signed char mode, unsigned short bits); 300 static void ispVMComment(unsigned short a_usCommentSize); 301 static void ispVMHeader(unsigned short a_usHeaderSize); 302 static signed char ispVMLCOUNT(unsigned short a_usCountSize); 303 static void ispVMClocks(unsigned short Clocks); 304 static void ispVMBypass(signed char ScanType, unsigned short Bits); 305 static void ispVMStateMachine(signed char NextState); 306 static signed char ispVMSend(unsigned short int); 307 static signed char ispVMRead(unsigned short int); 308 static signed char ispVMReadandSave(unsigned short int); 309 static signed char ispVMProcessLVDS(unsigned short a_usLVDSCount); 310 static void ispVMMemManager(signed char types, unsigned short size); 311 312 /* 313 * 314 * External variables and functions in hardware.c module 315 * 316 */ 317 static signed char g_cCurrentJTAGState; 318 319 #ifdef DEBUG 320 321 /* 322 * 323 * GetState 324 * 325 * Returns the state as a string based on the opcode. Only used 326 * for debugging purposes. 327 * 328 */ 329 330 const char *GetState(unsigned char a_ucState) 331 { 332 switch (a_ucState) { 333 case RESET: 334 return "RESET"; 335 case IDLE: 336 return "IDLE"; 337 case IRPAUSE: 338 return "IRPAUSE"; 339 case DRPAUSE: 340 return "DRPAUSE"; 341 case SHIFTIR: 342 return "SHIFTIR"; 343 case SHIFTDR: 344 return "SHIFTDR"; 345 case DRCAPTURE:/* 11/15/05 support DRCAPTURE*/ 346 return "DRCAPTURE"; 347 default: 348 break; 349 } 350 351 return 0; 352 } 353 354 /* 355 * 356 * PrintData 357 * 358 * Prints the data. Only used for debugging purposes. 359 * 360 */ 361 362 void PrintData(unsigned short a_iDataSize, unsigned char *a_pucData) 363 { 364 /* 09/11/07 NN added local variables initialization */ 365 unsigned short usByteSize = 0; 366 unsigned short usBitIndex = 0; 367 signed short usByteIndex = 0; 368 unsigned char ucByte = 0; 369 unsigned char ucFlipByte = 0; 370 371 if (a_iDataSize % 8) { 372 /* 09/11/07 NN Type cast mismatch variables */ 373 usByteSize = (unsigned short)(a_iDataSize / 8 + 1); 374 } else { 375 /* 09/11/07 NN Type cast mismatch variables */ 376 usByteSize = (unsigned short)(a_iDataSize / 8); 377 } 378 puts("("); 379 /* 09/11/07 NN Type cast mismatch variables */ 380 for (usByteIndex = (signed short)(usByteSize - 1); 381 usByteIndex >= 0; usByteIndex--) { 382 ucByte = a_pucData[usByteIndex]; 383 ucFlipByte = 0x00; 384 385 /* 386 * 387 * Flip each byte. 388 * 389 */ 390 391 for (usBitIndex = 0; usBitIndex < 8; usBitIndex++) { 392 ucFlipByte <<= 1; 393 if (ucByte & 0x1) { 394 ucFlipByte |= 0x1; 395 } 396 397 ucByte >>= 1; 398 } 399 400 /* 401 * 402 * Print the flipped byte. 403 * 404 */ 405 406 printf("%.02X", ucFlipByte); 407 if ((usByteSize - usByteIndex) % 40 == 39) { 408 puts("\n\t\t"); 409 } 410 if (usByteIndex < 0) 411 break; 412 } 413 puts(")"); 414 } 415 #endif /* DEBUG */ 416 417 void ispVMMemManager(signed char cTarget, unsigned short usSize) 418 { 419 switch (cTarget) { 420 case XTDI: 421 case TDI: 422 if (g_pucInData != NULL) { 423 if (previous_size == usSize) {/*memory exist*/ 424 break; 425 } else { 426 free(g_pucInData); 427 g_pucInData = NULL; 428 } 429 } 430 g_pucInData = (unsigned char *) malloc(usSize / 8 + 2); 431 previous_size = usSize; 432 case XTDO: 433 case TDO: 434 if (g_pucOutData != NULL) { 435 if (previous_size == usSize) { /*already exist*/ 436 break; 437 } else { 438 free(g_pucOutData); 439 g_pucOutData = NULL; 440 } 441 } 442 g_pucOutData = (unsigned char *) malloc(usSize / 8 + 2); 443 previous_size = usSize; 444 break; 445 case MASK: 446 if (g_pucOutMaskData != NULL) { 447 if (previous_size == usSize) {/*already allocated*/ 448 break; 449 } else { 450 free(g_pucOutMaskData); 451 g_pucOutMaskData = NULL; 452 } 453 } 454 g_pucOutMaskData = (unsigned char *) malloc(usSize / 8 + 2); 455 previous_size = usSize; 456 break; 457 case HIR: 458 if (g_pucHIRData != NULL) { 459 free(g_pucHIRData); 460 g_pucHIRData = NULL; 461 } 462 g_pucHIRData = (unsigned char *) malloc(usSize / 8 + 2); 463 break; 464 case TIR: 465 if (g_pucTIRData != NULL) { 466 free(g_pucTIRData); 467 g_pucTIRData = NULL; 468 } 469 g_pucTIRData = (unsigned char *) malloc(usSize / 8 + 2); 470 break; 471 case HDR: 472 if (g_pucHDRData != NULL) { 473 free(g_pucHDRData); 474 g_pucHDRData = NULL; 475 } 476 g_pucHDRData = (unsigned char *) malloc(usSize / 8 + 2); 477 break; 478 case TDR: 479 if (g_pucTDRData != NULL) { 480 free(g_pucTDRData); 481 g_pucTDRData = NULL; 482 } 483 g_pucTDRData = (unsigned char *) malloc(usSize / 8 + 2); 484 break; 485 case HEAP: 486 if (g_pucHeapMemory != NULL) { 487 free(g_pucHeapMemory); 488 g_pucHeapMemory = NULL; 489 } 490 g_pucHeapMemory = (unsigned char *) malloc(usSize + 2); 491 break; 492 case DMASK: 493 if (g_pucOutDMaskData != NULL) { 494 if (previous_size == usSize) { /*already allocated*/ 495 break; 496 } else { 497 free(g_pucOutDMaskData); 498 g_pucOutDMaskData = NULL; 499 } 500 } 501 g_pucOutDMaskData = (unsigned char *) malloc(usSize / 8 + 2); 502 previous_size = usSize; 503 break; 504 case LHEAP: 505 if (g_pucIntelBuffer != NULL) { 506 free(g_pucIntelBuffer); 507 g_pucIntelBuffer = NULL; 508 } 509 g_pucIntelBuffer = (unsigned char *) malloc(usSize + 2); 510 break; 511 case LVDS: 512 if (g_pLVDSList != NULL) { 513 free(g_pLVDSList); 514 g_pLVDSList = NULL; 515 } 516 g_pLVDSList = (LVDSPair *) malloc(usSize * sizeof(LVDSPair)); 517 if (g_pLVDSList) 518 memset(g_pLVDSList, 0, usSize * sizeof(LVDSPair)); 519 break; 520 default: 521 return; 522 } 523 } 524 525 void ispVMFreeMem(void) 526 { 527 if (g_pucHeapMemory != NULL) { 528 free(g_pucHeapMemory); 529 g_pucHeapMemory = NULL; 530 } 531 532 if (g_pucOutMaskData != NULL) { 533 free(g_pucOutMaskData); 534 g_pucOutMaskData = NULL; 535 } 536 537 if (g_pucInData != NULL) { 538 free(g_pucInData); 539 g_pucInData = NULL; 540 } 541 542 if (g_pucOutData != NULL) { 543 free(g_pucOutData); 544 g_pucOutData = NULL; 545 } 546 547 if (g_pucHIRData != NULL) { 548 free(g_pucHIRData); 549 g_pucHIRData = NULL; 550 } 551 552 if (g_pucTIRData != NULL) { 553 free(g_pucTIRData); 554 g_pucTIRData = NULL; 555 } 556 557 if (g_pucHDRData != NULL) { 558 free(g_pucHDRData); 559 g_pucHDRData = NULL; 560 } 561 562 if (g_pucTDRData != NULL) { 563 free(g_pucTDRData); 564 g_pucTDRData = NULL; 565 } 566 567 if (g_pucOutDMaskData != NULL) { 568 free(g_pucOutDMaskData); 569 g_pucOutDMaskData = NULL; 570 } 571 572 if (g_pucIntelBuffer != NULL) { 573 free(g_pucIntelBuffer); 574 g_pucIntelBuffer = NULL; 575 } 576 577 if (g_pLVDSList != NULL) { 578 free(g_pLVDSList); 579 g_pLVDSList = NULL; 580 } 581 } 582 583 584 /* 585 * 586 * ispVMDataSize 587 * 588 * Returns a VME-encoded number, usually used to indicate the 589 * bit length of an SIR/SDR command. 590 * 591 */ 592 593 long int ispVMDataSize() 594 { 595 /* 09/11/07 NN added local variables initialization */ 596 long int iSize = 0; 597 signed char cCurrentByte = 0; 598 signed char cIndex = 0; 599 cIndex = 0; 600 while ((cCurrentByte = GetByte()) & 0x80) { 601 iSize |= ((long int) (cCurrentByte & 0x7F)) << cIndex; 602 cIndex += 7; 603 } 604 iSize |= ((long int) (cCurrentByte & 0x7F)) << cIndex; 605 return iSize; 606 } 607 608 /* 609 * 610 * ispVMCode 611 * 612 * This is the heart of the embedded engine. All the high-level opcodes 613 * are extracted here. Once they have been identified, then it 614 * will call other functions to handle the processing. 615 * 616 */ 617 618 signed char ispVMCode() 619 { 620 /* 09/11/07 NN added local variables initialization */ 621 unsigned short iRepeatSize = 0; 622 signed char cOpcode = 0; 623 signed char cRetCode = 0; 624 unsigned char ucState = 0; 625 unsigned short usDelay = 0; 626 unsigned short usToggle = 0; 627 unsigned char usByte = 0; 628 629 /* 630 * 631 * Check the compression flag only if this is the first time 632 * this function is entered. Do not check the compression flag if 633 * it is being called recursively from other functions within 634 * the embedded engine. 635 * 636 */ 637 638 if (!(g_usDataType & LHEAP_IN) && !(g_usDataType & HEAP_IN)) { 639 usByte = GetByte(); 640 if (usByte == 0xf1) { 641 g_usDataType |= COMPRESS; 642 } else if (usByte == 0xf2) { 643 g_usDataType &= ~COMPRESS; 644 } else { 645 return VME_INVALID_FILE; 646 } 647 } 648 649 /* 650 * 651 * Begin looping through all the VME opcodes. 652 * 653 */ 654 655 while ((cOpcode = GetByte()) >= 0) { 656 657 switch (cOpcode) { 658 case STATE: 659 660 /* 661 * Step the JTAG state machine. 662 */ 663 664 ucState = GetByte(); 665 666 /* 667 * Step the JTAG state machine to DRCAPTURE 668 * to support Looping. 669 */ 670 671 if ((g_usDataType & LHEAP_IN) && 672 (ucState == DRPAUSE) && 673 (g_cCurrentJTAGState == ucState)) { 674 ispVMStateMachine(DRCAPTURE); 675 } 676 677 ispVMStateMachine(ucState); 678 679 #ifdef DEBUG 680 if (g_usDataType & LHEAP_IN) { 681 debug("LDELAY %s ", GetState(ucState)); 682 } else { 683 debug("STATE %s;\n", GetState(ucState)); 684 } 685 #endif /* DEBUG */ 686 break; 687 case SIR: 688 case SDR: 689 case XSDR: 690 691 #ifdef DEBUG 692 switch (cOpcode) { 693 case SIR: 694 puts("SIR "); 695 break; 696 case SDR: 697 case XSDR: 698 if (g_usDataType & LHEAP_IN) { 699 puts("LSDR "); 700 } else { 701 puts("SDR "); 702 } 703 break; 704 } 705 #endif /* DEBUG */ 706 /* 707 * 708 * Shift in data into the device. 709 * 710 */ 711 712 cRetCode = ispVMShift(cOpcode); 713 if (cRetCode != 0) { 714 return cRetCode; 715 } 716 break; 717 case WAIT: 718 719 /* 720 * 721 * Observe delay. 722 * 723 */ 724 725 /* 09/11/07 NN Type cast mismatch variables */ 726 usDelay = (unsigned short) ispVMDataSize(); 727 ispVMDelay(usDelay); 728 729 #ifdef DEBUG 730 if (usDelay & 0x8000) { 731 732 /* 733 * Since MSB is set, the delay time must be 734 * decoded to millisecond. The SVF2VME encodes 735 * the MSB to represent millisecond. 736 */ 737 738 usDelay &= ~0x8000; 739 if (g_usDataType & LHEAP_IN) { 740 printf("%.2E SEC;\n", 741 (float) usDelay / 1000); 742 } else { 743 printf("RUNTEST %.2E SEC;\n", 744 (float) usDelay / 1000); 745 } 746 } else { 747 /* 748 * Since MSB is not set, the delay time 749 * is given as microseconds. 750 */ 751 752 if (g_usDataType & LHEAP_IN) { 753 printf("%.2E SEC;\n", 754 (float) usDelay / 1000000); 755 } else { 756 printf("RUNTEST %.2E SEC;\n", 757 (float) usDelay / 1000000); 758 } 759 } 760 #endif /* DEBUG */ 761 break; 762 case TCK: 763 764 /* 765 * Issue clock toggles. 766 */ 767 768 /* 09/11/07 NN Type cast mismatch variables */ 769 usToggle = (unsigned short) ispVMDataSize(); 770 ispVMClocks(usToggle); 771 772 #ifdef DEBUG 773 printf("RUNTEST %d TCK;\n", usToggle); 774 #endif /* DEBUG */ 775 break; 776 case ENDDR: 777 778 /* 779 * 780 * Set the ENDDR. 781 * 782 */ 783 784 g_ucEndDR = GetByte(); 785 786 #ifdef DEBUG 787 printf("ENDDR %s;\n", GetState(g_ucEndDR)); 788 #endif /* DEBUG */ 789 break; 790 case ENDIR: 791 792 /* 793 * 794 * Set the ENDIR. 795 * 796 */ 797 798 g_ucEndIR = GetByte(); 799 800 #ifdef DEBUG 801 printf("ENDIR %s;\n", GetState(g_ucEndIR)); 802 #endif /* DEBUG */ 803 break; 804 case HIR: 805 case TIR: 806 case HDR: 807 case TDR: 808 809 #ifdef DEBUG 810 switch (cOpcode) { 811 case HIR: 812 puts("HIR "); 813 break; 814 case TIR: 815 puts("TIR "); 816 break; 817 case HDR: 818 puts("HDR "); 819 break; 820 case TDR: 821 puts("TDR "); 822 break; 823 } 824 #endif /* DEBUG */ 825 /* 826 * Set the header/trailer of the device in order 827 * to bypass 828 * successfully. 829 */ 830 831 cRetCode = ispVMAmble(cOpcode); 832 if (cRetCode != 0) { 833 return cRetCode; 834 } 835 836 #ifdef DEBUG 837 puts(";\n"); 838 #endif /* DEBUG */ 839 break; 840 case MEM: 841 842 /* 843 * The maximum RAM required to support 844 * processing one row of the VME file. 845 */ 846 847 /* 09/11/07 NN Type cast mismatch variables */ 848 g_usMaxSize = (unsigned short) ispVMDataSize(); 849 850 #ifdef DEBUG 851 printf("// MEMSIZE %d\n", g_usMaxSize); 852 #endif /* DEBUG */ 853 break; 854 case VENDOR: 855 856 /* 857 * 858 * Set the VENDOR type. 859 * 860 */ 861 862 cOpcode = GetByte(); 863 switch (cOpcode) { 864 case LATTICE: 865 #ifdef DEBUG 866 puts("// VENDOR LATTICE\n"); 867 #endif /* DEBUG */ 868 g_cVendor = LATTICE; 869 break; 870 case ALTERA: 871 #ifdef DEBUG 872 puts("// VENDOR ALTERA\n"); 873 #endif /* DEBUG */ 874 g_cVendor = ALTERA; 875 break; 876 case XILINX: 877 #ifdef DEBUG 878 puts("// VENDOR XILINX\n"); 879 #endif /* DEBUG */ 880 g_cVendor = XILINX; 881 break; 882 default: 883 break; 884 } 885 break; 886 case SETFLOW: 887 888 /* 889 * Set the flow control. Flow control determines 890 * the personality of the embedded engine. 891 */ 892 893 /* 09/11/07 NN Type cast mismatch variables */ 894 g_usFlowControl |= (unsigned short) ispVMDataSize(); 895 break; 896 case RESETFLOW: 897 898 /* 899 * 900 * Unset the flow control. 901 * 902 */ 903 904 /* 09/11/07 NN Type cast mismatch variables */ 905 g_usFlowControl &= (unsigned short) ~(ispVMDataSize()); 906 break; 907 case HEAP: 908 909 /* 910 * 911 * Allocate heap size to store loops. 912 * 913 */ 914 915 cRetCode = GetByte(); 916 if (cRetCode != SECUREHEAP) { 917 return VME_INVALID_FILE; 918 } 919 /* 09/11/07 NN Type cast mismatch variables */ 920 g_iHEAPSize = (unsigned short) ispVMDataSize(); 921 922 /* 923 * Store the maximum size of the HEAP buffer. 924 * Used to convert VME to HEX. 925 */ 926 927 if (g_iHEAPSize > g_usHeapSize) { 928 g_usHeapSize = g_iHEAPSize; 929 } 930 931 ispVMMemManager(HEAP, (unsigned short) g_iHEAPSize); 932 break; 933 case REPEAT: 934 935 /* 936 * 937 * Execute loops. 938 * 939 */ 940 941 g_usRepeatLoops = 0; 942 943 /* 09/11/07 NN Type cast mismatch variables */ 944 iRepeatSize = (unsigned short) ispVMDataSize(); 945 946 cRetCode = ispVMLoop((unsigned short) iRepeatSize); 947 if (cRetCode != 0) { 948 return cRetCode; 949 } 950 break; 951 case ENDLOOP: 952 953 /* 954 * 955 * Exit point from processing loops. 956 * 957 */ 958 959 return cRetCode; 960 case ENDVME: 961 962 /* 963 * The only valid exit point that indicates 964 * end of programming. 965 */ 966 967 return cRetCode; 968 case SHR: 969 970 /* 971 * 972 * Right-shift address. 973 * 974 */ 975 976 g_usFlowControl |= SHIFTRIGHT; 977 978 /* 09/11/07 NN Type cast mismatch variables */ 979 g_usShiftValue = (unsigned short) (g_usRepeatLoops * 980 (unsigned short)GetByte()); 981 break; 982 case SHL: 983 984 /* 985 * Left-shift address. 986 */ 987 988 g_usFlowControl |= SHIFTLEFT; 989 990 /* 09/11/07 NN Type cast mismatch variables */ 991 g_usShiftValue = (unsigned short) (g_usRepeatLoops * 992 (unsigned short)GetByte()); 993 break; 994 case FREQUENCY: 995 996 /* 997 * 998 * Set the frequency. 999 * 1000 */ 1001 1002 /* 09/11/07 NN Type cast mismatch variables */ 1003 g_iFrequency = (int) (ispVMDataSize() / 1000); 1004 if (g_iFrequency == 1) 1005 g_iFrequency = 1000; 1006 1007 #ifdef DEBUG 1008 printf("FREQUENCY %.2E HZ;\n", 1009 (float) g_iFrequency * 1000); 1010 #endif /* DEBUG */ 1011 break; 1012 case LCOUNT: 1013 1014 /* 1015 * 1016 * Process LCOUNT command. 1017 * 1018 */ 1019 1020 cRetCode = ispVMLCOUNT((unsigned short)ispVMDataSize()); 1021 if (cRetCode != 0) { 1022 return cRetCode; 1023 } 1024 break; 1025 case VUES: 1026 1027 /* 1028 * 1029 * Set the flow control to verify USERCODE. 1030 * 1031 */ 1032 1033 g_usFlowControl |= VERIFYUES; 1034 break; 1035 case COMMENT: 1036 1037 /* 1038 * 1039 * Display comment. 1040 * 1041 */ 1042 1043 ispVMComment((unsigned short) ispVMDataSize()); 1044 break; 1045 case LVDS: 1046 1047 /* 1048 * 1049 * Process LVDS command. 1050 * 1051 */ 1052 1053 ispVMProcessLVDS((unsigned short) ispVMDataSize()); 1054 break; 1055 case HEADER: 1056 1057 /* 1058 * 1059 * Discard header. 1060 * 1061 */ 1062 1063 ispVMHeader((unsigned short) ispVMDataSize()); 1064 break; 1065 /* 03/14/06 Support Toggle ispENABLE signal*/ 1066 case ispEN: 1067 ucState = GetByte(); 1068 if ((ucState == ON) || (ucState == 0x01)) 1069 writePort(g_ucPinENABLE, 0x01); 1070 else 1071 writePort(g_ucPinENABLE, 0x00); 1072 ispVMDelay(1); 1073 break; 1074 /* 05/24/06 support Toggle TRST pin*/ 1075 case TRST: 1076 ucState = GetByte(); 1077 if (ucState == 0x01) 1078 writePort(g_ucPinTRST, 0x01); 1079 else 1080 writePort(g_ucPinTRST, 0x00); 1081 ispVMDelay(1); 1082 break; 1083 default: 1084 1085 /* 1086 * 1087 * Invalid opcode encountered. 1088 * 1089 */ 1090 1091 #ifdef DEBUG 1092 printf("\nINVALID OPCODE: 0x%.2X\n", cOpcode); 1093 #endif /* DEBUG */ 1094 1095 return VME_INVALID_FILE; 1096 } 1097 } 1098 1099 /* 1100 * 1101 * Invalid exit point. Processing the token 'ENDVME' is the only 1102 * valid way to exit the embedded engine. 1103 * 1104 */ 1105 1106 return VME_INVALID_FILE; 1107 } 1108 1109 /* 1110 * 1111 * ispVMDataCode 1112 * 1113 * Processes the TDI/TDO/MASK/DMASK etc of an SIR/SDR command. 1114 * 1115 */ 1116 1117 signed char ispVMDataCode() 1118 { 1119 /* 09/11/07 NN added local variables initialization */ 1120 signed char cDataByte = 0; 1121 signed char siDataSource = 0; /*source of data from file by default*/ 1122 1123 if (g_usDataType & HEAP_IN) { 1124 siDataSource = 1; /*the source of data from memory*/ 1125 } 1126 1127 /* 1128 * 1129 * Clear the data type register. 1130 * 1131 **/ 1132 1133 g_usDataType &= ~(MASK_DATA + TDI_DATA + 1134 TDO_DATA + DMASK_DATA + CMASK_DATA); 1135 1136 /* 1137 * Iterate through SIR/SDR command and look for TDI, 1138 * TDO, MASK, etc. 1139 */ 1140 1141 while ((cDataByte = GetByte()) >= 0) { 1142 ispVMMemManager(cDataByte, g_usMaxSize); 1143 switch (cDataByte) { 1144 case TDI: 1145 1146 /* 1147 * Store the maximum size of the TDI buffer. 1148 * Used to convert VME to HEX. 1149 */ 1150 1151 if (g_usiDataSize > g_usTDISize) { 1152 g_usTDISize = g_usiDataSize; 1153 } 1154 /* 1155 * Updated data type register to indicate that 1156 * TDI data is currently being used. Process the 1157 * data in the VME file into the TDI buffer. 1158 */ 1159 1160 g_usDataType |= TDI_DATA; 1161 ispVMData(g_pucInData); 1162 break; 1163 case XTDO: 1164 1165 /* 1166 * Store the maximum size of the TDO buffer. 1167 * Used to convert VME to HEX. 1168 */ 1169 1170 if (g_usiDataSize > g_usTDOSize) { 1171 g_usTDOSize = g_usiDataSize; 1172 } 1173 1174 /* 1175 * Updated data type register to indicate that 1176 * TDO data is currently being used. 1177 */ 1178 1179 g_usDataType |= TDO_DATA; 1180 break; 1181 case TDO: 1182 1183 /* 1184 * Store the maximum size of the TDO buffer. 1185 * Used to convert VME to HEX. 1186 */ 1187 1188 if (g_usiDataSize > g_usTDOSize) { 1189 g_usTDOSize = g_usiDataSize; 1190 } 1191 1192 /* 1193 * Updated data type register to indicate 1194 * that TDO data is currently being used. 1195 * Process the data in the VME file into the 1196 * TDO buffer. 1197 */ 1198 1199 g_usDataType |= TDO_DATA; 1200 ispVMData(g_pucOutData); 1201 break; 1202 case MASK: 1203 1204 /* 1205 * Store the maximum size of the MASK buffer. 1206 * Used to convert VME to HEX. 1207 */ 1208 1209 if (g_usiDataSize > g_usMASKSize) { 1210 g_usMASKSize = g_usiDataSize; 1211 } 1212 1213 /* 1214 * Updated data type register to indicate that 1215 * MASK data is currently being used. Process 1216 * the data in the VME file into the MASK buffer 1217 */ 1218 1219 g_usDataType |= MASK_DATA; 1220 ispVMData(g_pucOutMaskData); 1221 break; 1222 case DMASK: 1223 1224 /* 1225 * Store the maximum size of the DMASK buffer. 1226 * Used to convert VME to HEX. 1227 */ 1228 1229 if (g_usiDataSize > g_usDMASKSize) { 1230 g_usDMASKSize = g_usiDataSize; 1231 } 1232 1233 /* 1234 * Updated data type register to indicate that 1235 * DMASK data is currently being used. Process 1236 * the data in the VME file into the DMASK 1237 * buffer. 1238 */ 1239 1240 g_usDataType |= DMASK_DATA; 1241 ispVMData(g_pucOutDMaskData); 1242 break; 1243 case CMASK: 1244 1245 /* 1246 * Updated data type register to indicate that 1247 * MASK data is currently being used. Process 1248 * the data in the VME file into the MASK buffer 1249 */ 1250 1251 g_usDataType |= CMASK_DATA; 1252 ispVMData(g_pucOutMaskData); 1253 break; 1254 case CONTINUE: 1255 return 0; 1256 default: 1257 /* 1258 * Encountered invalid opcode. 1259 */ 1260 return VME_INVALID_FILE; 1261 } 1262 1263 switch (cDataByte) { 1264 case TDI: 1265 1266 /* 1267 * Left bit shift. Used when performing 1268 * algorithm looping. 1269 */ 1270 1271 if (g_usFlowControl & SHIFTLEFT) { 1272 ispVMBitShift(SHL, g_usShiftValue); 1273 g_usFlowControl &= ~SHIFTLEFT; 1274 } 1275 1276 /* 1277 * Right bit shift. Used when performing 1278 * algorithm looping. 1279 */ 1280 1281 if (g_usFlowControl & SHIFTRIGHT) { 1282 ispVMBitShift(SHR, g_usShiftValue); 1283 g_usFlowControl &= ~SHIFTRIGHT; 1284 } 1285 default: 1286 break; 1287 } 1288 1289 if (siDataSource) { 1290 g_usDataType |= HEAP_IN; /*restore from memory*/ 1291 } 1292 } 1293 1294 if (siDataSource) { /*fetch data from heap memory upon return*/ 1295 g_usDataType |= HEAP_IN; 1296 } 1297 1298 if (cDataByte < 0) { 1299 1300 /* 1301 * Encountered invalid opcode. 1302 */ 1303 1304 return VME_INVALID_FILE; 1305 } else { 1306 return 0; 1307 } 1308 } 1309 1310 /* 1311 * 1312 * ispVMData 1313 * Extract one row of data operand from the current data type opcode. Perform 1314 * the decompression if necessary. Extra RAM is not required for the 1315 * decompression process. The decompression scheme employed in this module 1316 * is on row by row basis. The format of the data stream: 1317 * [compression code][compressed data stream] 1318 * 0x00 --No compression 1319 * 0x01 --Compress by 0x00. 1320 * Example: 1321 * Original stream: 0x000000000000000000000001 1322 * Compressed stream: 0x01000901 1323 * Detail: 0x01 is the code, 0x00 is the key, 1324 * 0x09 is the count of 0x00 bytes, 1325 * 0x01 is the uncompressed byte. 1326 * 0x02 --Compress by 0xFF. 1327 * Example: 1328 * Original stream: 0xFFFFFFFFFFFFFFFFFFFFFF01 1329 * Compressed stream: 0x02FF0901 1330 * Detail: 0x02 is the code, 0xFF is the key, 1331 * 0x09 is the count of 0xFF bytes, 1332 * 0x01 is the uncompressed byte. 1333 * 0x03 1334 * : : 1335 * 0xFE -- Compress by nibble blocks. 1336 * Example: 1337 * Original stream: 0x84210842108421084210 1338 * Compressed stream: 0x0584210 1339 * Detail: 0x05 is the code, means 5 nibbles block. 1340 * 0x84210 is the 5 nibble blocks. 1341 * The whole row is 80 bits given by g_usiDataSize. 1342 * The number of times the block repeat itself 1343 * is found by g_usiDataSize/(4*0x05) which is 4. 1344 * 0xFF -- Compress by the most frequently happen byte. 1345 * Example: 1346 * Original stream: 0x04020401030904040404 1347 * Compressed stream: 0xFF04(0,1,0x02,0,1,0x01,1,0x03,1,0x09,0,0,0) 1348 * or: 0xFF044090181C240 1349 * Detail: 0xFF is the code, 0x04 is the key. 1350 * a bit of 0 represent the key shall be put into 1351 * the current bit position and a bit of 1 1352 * represent copying the next of 8 bits of data 1353 * in. 1354 * 1355 */ 1356 1357 void ispVMData(unsigned char *ByteData) 1358 { 1359 /* 09/11/07 NN added local variables initialization */ 1360 unsigned short size = 0; 1361 unsigned short i, j, m, getData = 0; 1362 unsigned char cDataByte = 0; 1363 unsigned char compress = 0; 1364 unsigned short FFcount = 0; 1365 unsigned char compr_char = 0xFF; 1366 unsigned short index = 0; 1367 signed char compression = 0; 1368 1369 /*convert number in bits to bytes*/ 1370 if (g_usiDataSize % 8 > 0) { 1371 /* 09/11/07 NN Type cast mismatch variables */ 1372 size = (unsigned short)(g_usiDataSize / 8 + 1); 1373 } else { 1374 /* 09/11/07 NN Type cast mismatch variables */ 1375 size = (unsigned short)(g_usiDataSize / 8); 1376 } 1377 1378 /* 1379 * If there is compression, then check if compress by key 1380 * of 0x00 or 0xFF or by other keys or by nibble blocks 1381 */ 1382 1383 if (g_usDataType & COMPRESS) { 1384 compression = 1; 1385 compress = GetByte(); 1386 if ((compress == VAR) && (g_usDataType & HEAP_IN)) { 1387 getData = 1; 1388 g_usDataType &= ~(HEAP_IN); 1389 compress = GetByte(); 1390 } 1391 1392 switch (compress) { 1393 case 0x00: 1394 /* No compression */ 1395 compression = 0; 1396 break; 1397 case 0x01: 1398 /* Compress by byte 0x00 */ 1399 compr_char = 0x00; 1400 break; 1401 case 0x02: 1402 /* Compress by byte 0xFF */ 1403 compr_char = 0xFF; 1404 break; 1405 case 0xFF: 1406 /* Huffman encoding */ 1407 compr_char = GetByte(); 1408 i = 8; 1409 for (index = 0; index < size; index++) { 1410 ByteData[index] = 0x00; 1411 if (i > 7) { 1412 cDataByte = GetByte(); 1413 i = 0; 1414 } 1415 if ((cDataByte << i++) & 0x80) 1416 m = 8; 1417 else { 1418 ByteData[index] = compr_char; 1419 m = 0; 1420 } 1421 1422 for (j = 0; j < m; j++) { 1423 if (i > 7) { 1424 cDataByte = GetByte(); 1425 i = 0; 1426 } 1427 ByteData[index] |= 1428 ((cDataByte << i++) & 0x80) >> j; 1429 } 1430 } 1431 size = 0; 1432 break; 1433 default: 1434 for (index = 0; index < size; index++) 1435 ByteData[index] = 0x00; 1436 for (index = 0; index < compress; index++) { 1437 if (index % 2 == 0) 1438 cDataByte = GetByte(); 1439 for (i = 0; i < size * 2 / compress; i++) { 1440 j = (unsigned short)(index + 1441 (i * (unsigned short)compress)); 1442 /*clear the nibble to zero first*/ 1443 if (j%2) { 1444 if (index % 2) 1445 ByteData[j/2] |= 1446 cDataByte & 0xF; 1447 else 1448 ByteData[j/2] |= 1449 cDataByte >> 4; 1450 } else { 1451 if (index % 2) 1452 ByteData[j/2] |= 1453 cDataByte << 4; 1454 else 1455 ByteData[j/2] |= 1456 cDataByte & 0xF0; 1457 } 1458 } 1459 } 1460 size = 0; 1461 break; 1462 } 1463 } 1464 1465 FFcount = 0; 1466 1467 /* Decompress by byte 0x00 or 0xFF */ 1468 for (index = 0; index < size; index++) { 1469 if (FFcount <= 0) { 1470 cDataByte = GetByte(); 1471 if ((cDataByte == VAR) && (g_usDataType&HEAP_IN) && 1472 !getData && !(g_usDataType&COMPRESS)) { 1473 getData = 1; 1474 g_usDataType &= ~(HEAP_IN); 1475 cDataByte = GetByte(); 1476 } 1477 ByteData[index] = cDataByte; 1478 if ((compression) && (cDataByte == compr_char)) 1479 /* 09/11/07 NN Type cast mismatch variables */ 1480 FFcount = (unsigned short) ispVMDataSize(); 1481 /*The number of 0xFF or 0x00 bytes*/ 1482 } else { 1483 FFcount--; /*Use up the 0xFF chain first*/ 1484 ByteData[index] = compr_char; 1485 } 1486 } 1487 1488 if (getData) { 1489 g_usDataType |= HEAP_IN; 1490 getData = 0; 1491 } 1492 } 1493 1494 /* 1495 * 1496 * ispVMShift 1497 * 1498 * Processes the SDR/XSDR/SIR commands. 1499 * 1500 */ 1501 1502 signed char ispVMShift(signed char a_cCode) 1503 { 1504 /* 09/11/07 NN added local variables initialization */ 1505 unsigned short iDataIndex = 0; 1506 unsigned short iReadLoop = 0; 1507 signed char cRetCode = 0; 1508 1509 cRetCode = 0; 1510 /* 09/11/07 NN Type cast mismatch variables */ 1511 g_usiDataSize = (unsigned short) ispVMDataSize(); 1512 1513 /*clear the flags first*/ 1514 g_usDataType &= ~(SIR_DATA + EXPRESS + SDR_DATA); 1515 switch (a_cCode) { 1516 case SIR: 1517 g_usDataType |= SIR_DATA; 1518 /* 1519 * 1/15/04 If performing cascading, then go directly to SHIFTIR. 1520 * Else, go to IRPAUSE before going to SHIFTIR 1521 */ 1522 if (g_usFlowControl & CASCADE) { 1523 ispVMStateMachine(SHIFTIR); 1524 } else { 1525 ispVMStateMachine(IRPAUSE); 1526 ispVMStateMachine(SHIFTIR); 1527 if (g_usHeadIR > 0) { 1528 ispVMBypass(HIR, g_usHeadIR); 1529 sclock(); 1530 } 1531 } 1532 break; 1533 case XSDR: 1534 g_usDataType |= EXPRESS; /*mark simultaneous in and out*/ 1535 case SDR: 1536 g_usDataType |= SDR_DATA; 1537 /* 1538 * 1/15/04 If already in SHIFTDR, then do not move state or 1539 * shift in header. This would imply that the previously 1540 * shifted frame was a cascaded frame. 1541 */ 1542 if (g_cCurrentJTAGState != SHIFTDR) { 1543 /* 1544 * 1/15/04 If performing cascading, then go directly 1545 * to SHIFTDR. Else, go to DRPAUSE before going 1546 * to SHIFTDR 1547 */ 1548 if (g_usFlowControl & CASCADE) { 1549 if (g_cCurrentJTAGState == DRPAUSE) { 1550 ispVMStateMachine(SHIFTDR); 1551 /* 1552 * 1/15/04 If cascade flag has been seat 1553 * and the current state is DRPAUSE, 1554 * this implies that the first cascaded 1555 * frame is about to be shifted in. The 1556 * header must be shifted prior to 1557 * shifting the first cascaded frame. 1558 */ 1559 if (g_usHeadDR > 0) { 1560 ispVMBypass(HDR, g_usHeadDR); 1561 sclock(); 1562 } 1563 } else { 1564 ispVMStateMachine(SHIFTDR); 1565 } 1566 } else { 1567 ispVMStateMachine(DRPAUSE); 1568 ispVMStateMachine(SHIFTDR); 1569 if (g_usHeadDR > 0) { 1570 ispVMBypass(HDR, g_usHeadDR); 1571 sclock(); 1572 } 1573 } 1574 } 1575 break; 1576 default: 1577 return VME_INVALID_FILE; 1578 } 1579 1580 cRetCode = ispVMDataCode(); 1581 1582 if (cRetCode != 0) { 1583 return VME_INVALID_FILE; 1584 } 1585 1586 #ifdef DEBUG 1587 printf("%d ", g_usiDataSize); 1588 1589 if (g_usDataType & TDI_DATA) { 1590 puts("TDI "); 1591 PrintData(g_usiDataSize, g_pucInData); 1592 } 1593 1594 if (g_usDataType & TDO_DATA) { 1595 puts("\n\t\tTDO "); 1596 PrintData(g_usiDataSize, g_pucOutData); 1597 } 1598 1599 if (g_usDataType & MASK_DATA) { 1600 puts("\n\t\tMASK "); 1601 PrintData(g_usiDataSize, g_pucOutMaskData); 1602 } 1603 1604 if (g_usDataType & DMASK_DATA) { 1605 puts("\n\t\tDMASK "); 1606 PrintData(g_usiDataSize, g_pucOutDMaskData); 1607 } 1608 1609 puts(";\n"); 1610 #endif /* DEBUG */ 1611 1612 if (g_usDataType & TDO_DATA || g_usDataType & DMASK_DATA) { 1613 if (g_usDataType & DMASK_DATA) { 1614 cRetCode = ispVMReadandSave(g_usiDataSize); 1615 if (!cRetCode) { 1616 if (g_usTailDR > 0) { 1617 sclock(); 1618 ispVMBypass(TDR, g_usTailDR); 1619 } 1620 ispVMStateMachine(DRPAUSE); 1621 ispVMStateMachine(SHIFTDR); 1622 if (g_usHeadDR > 0) { 1623 ispVMBypass(HDR, g_usHeadDR); 1624 sclock(); 1625 } 1626 for (iDataIndex = 0; 1627 iDataIndex < g_usiDataSize / 8 + 1; 1628 iDataIndex++) 1629 g_pucInData[iDataIndex] = 1630 g_pucOutData[iDataIndex]; 1631 g_usDataType &= ~(TDO_DATA + DMASK_DATA); 1632 cRetCode = ispVMSend(g_usiDataSize); 1633 } 1634 } else { 1635 cRetCode = ispVMRead(g_usiDataSize); 1636 if (cRetCode == -1 && g_cVendor == XILINX) { 1637 for (iReadLoop = 0; iReadLoop < 30; 1638 iReadLoop++) { 1639 cRetCode = ispVMRead(g_usiDataSize); 1640 if (!cRetCode) { 1641 break; 1642 } else { 1643 /* Always DRPAUSE */ 1644 ispVMStateMachine(DRPAUSE); 1645 /* 1646 * Bypass other devices 1647 * when appropriate 1648 */ 1649 ispVMBypass(TDR, g_usTailDR); 1650 ispVMStateMachine(g_ucEndDR); 1651 ispVMStateMachine(IDLE); 1652 ispVMDelay(1000); 1653 } 1654 } 1655 } 1656 } 1657 } else { /*TDI only*/ 1658 cRetCode = ispVMSend(g_usiDataSize); 1659 } 1660 1661 /*transfer the input data to the output buffer for the next verify*/ 1662 if ((g_usDataType & EXPRESS) || (a_cCode == SDR)) { 1663 if (g_pucOutData) { 1664 for (iDataIndex = 0; iDataIndex < g_usiDataSize / 8 + 1; 1665 iDataIndex++) 1666 g_pucOutData[iDataIndex] = 1667 g_pucInData[iDataIndex]; 1668 } 1669 } 1670 1671 switch (a_cCode) { 1672 case SIR: 1673 /* 1/15/04 If not performing cascading, then shift ENDIR */ 1674 if (!(g_usFlowControl & CASCADE)) { 1675 if (g_usTailIR > 0) { 1676 sclock(); 1677 ispVMBypass(TIR, g_usTailIR); 1678 } 1679 ispVMStateMachine(g_ucEndIR); 1680 } 1681 break; 1682 case XSDR: 1683 case SDR: 1684 /* 1/15/04 If not performing cascading, then shift ENDDR */ 1685 if (!(g_usFlowControl & CASCADE)) { 1686 if (g_usTailDR > 0) { 1687 sclock(); 1688 ispVMBypass(TDR, g_usTailDR); 1689 } 1690 ispVMStateMachine(g_ucEndDR); 1691 } 1692 break; 1693 default: 1694 break; 1695 } 1696 1697 return cRetCode; 1698 } 1699 1700 /* 1701 * 1702 * ispVMAmble 1703 * 1704 * This routine is to extract Header and Trailer parameter for SIR and 1705 * SDR operations. 1706 * 1707 * The Header and Trailer parameter are the pre-amble and post-amble bit 1708 * stream need to be shifted into TDI or out of TDO of the devices. Mostly 1709 * is for the purpose of bypassing the leading or trailing devices. ispVM 1710 * supports only shifting data into TDI to bypass the devices. 1711 * 1712 * For a single device, the header and trailer parameters are all set to 0 1713 * as default by ispVM. If it is for multiple devices, the header and trailer 1714 * value will change as specified by the VME file. 1715 * 1716 */ 1717 1718 signed char ispVMAmble(signed char Code) 1719 { 1720 signed char compress = 0; 1721 /* 09/11/07 NN Type cast mismatch variables */ 1722 g_usiDataSize = (unsigned short)ispVMDataSize(); 1723 1724 #ifdef DEBUG 1725 printf("%d", g_usiDataSize); 1726 #endif /* DEBUG */ 1727 1728 if (g_usiDataSize) { 1729 1730 /* 1731 * Discard the TDI byte and set the compression bit in the data 1732 * type register to false if compression is set because TDI data 1733 * after HIR/HDR/TIR/TDR is not compressed. 1734 */ 1735 1736 GetByte(); 1737 if (g_usDataType & COMPRESS) { 1738 g_usDataType &= ~(COMPRESS); 1739 compress = 1; 1740 } 1741 } 1742 1743 switch (Code) { 1744 case HIR: 1745 1746 /* 1747 * Store the maximum size of the HIR buffer. 1748 * Used to convert VME to HEX. 1749 */ 1750 1751 if (g_usiDataSize > g_usHIRSize) { 1752 g_usHIRSize = g_usiDataSize; 1753 } 1754 1755 /* 1756 * Assign the HIR value and allocate memory. 1757 */ 1758 1759 g_usHeadIR = g_usiDataSize; 1760 if (g_usHeadIR) { 1761 ispVMMemManager(HIR, g_usHeadIR); 1762 ispVMData(g_pucHIRData); 1763 1764 #ifdef DEBUG 1765 puts(" TDI "); 1766 PrintData(g_usHeadIR, g_pucHIRData); 1767 #endif /* DEBUG */ 1768 } 1769 break; 1770 case TIR: 1771 1772 /* 1773 * Store the maximum size of the TIR buffer. 1774 * Used to convert VME to HEX. 1775 */ 1776 1777 if (g_usiDataSize > g_usTIRSize) { 1778 g_usTIRSize = g_usiDataSize; 1779 } 1780 1781 /* 1782 * Assign the TIR value and allocate memory. 1783 */ 1784 1785 g_usTailIR = g_usiDataSize; 1786 if (g_usTailIR) { 1787 ispVMMemManager(TIR, g_usTailIR); 1788 ispVMData(g_pucTIRData); 1789 1790 #ifdef DEBUG 1791 puts(" TDI "); 1792 PrintData(g_usTailIR, g_pucTIRData); 1793 #endif /* DEBUG */ 1794 } 1795 break; 1796 case HDR: 1797 1798 /* 1799 * Store the maximum size of the HDR buffer. 1800 * Used to convert VME to HEX. 1801 */ 1802 1803 if (g_usiDataSize > g_usHDRSize) { 1804 g_usHDRSize = g_usiDataSize; 1805 } 1806 1807 /* 1808 * Assign the HDR value and allocate memory. 1809 * 1810 */ 1811 1812 g_usHeadDR = g_usiDataSize; 1813 if (g_usHeadDR) { 1814 ispVMMemManager(HDR, g_usHeadDR); 1815 ispVMData(g_pucHDRData); 1816 1817 #ifdef DEBUG 1818 puts(" TDI "); 1819 PrintData(g_usHeadDR, g_pucHDRData); 1820 #endif /* DEBUG */ 1821 } 1822 break; 1823 case TDR: 1824 1825 /* 1826 * Store the maximum size of the TDR buffer. 1827 * Used to convert VME to HEX. 1828 */ 1829 1830 if (g_usiDataSize > g_usTDRSize) { 1831 g_usTDRSize = g_usiDataSize; 1832 } 1833 1834 /* 1835 * Assign the TDR value and allocate memory. 1836 * 1837 */ 1838 1839 g_usTailDR = g_usiDataSize; 1840 if (g_usTailDR) { 1841 ispVMMemManager(TDR, g_usTailDR); 1842 ispVMData(g_pucTDRData); 1843 1844 #ifdef DEBUG 1845 puts(" TDI "); 1846 PrintData(g_usTailDR, g_pucTDRData); 1847 #endif /* DEBUG */ 1848 } 1849 break; 1850 default: 1851 break; 1852 } 1853 1854 /* 1855 * 1856 * Re-enable compression if it was previously set. 1857 * 1858 **/ 1859 1860 if (compress) { 1861 g_usDataType |= COMPRESS; 1862 } 1863 1864 if (g_usiDataSize) { 1865 Code = GetByte(); 1866 if (Code == CONTINUE) { 1867 return 0; 1868 } else { 1869 1870 /* 1871 * Encountered invalid opcode. 1872 */ 1873 1874 return VME_INVALID_FILE; 1875 } 1876 } 1877 1878 return 0; 1879 } 1880 1881 /* 1882 * 1883 * ispVMLoop 1884 * 1885 * Perform the function call upon by the REPEAT opcode. 1886 * Memory is to be allocated to store the entire loop from REPEAT to ENDLOOP. 1887 * After the loop is stored then execution begin. The REPEATLOOP flag is set 1888 * on the g_usFlowControl register to indicate the repeat loop is in session 1889 * and therefore fetch opcode from the memory instead of from the file. 1890 * 1891 */ 1892 1893 signed char ispVMLoop(unsigned short a_usLoopCount) 1894 { 1895 /* 09/11/07 NN added local variables initialization */ 1896 signed char cRetCode = 0; 1897 unsigned short iHeapIndex = 0; 1898 unsigned short iLoopIndex = 0; 1899 1900 g_usShiftValue = 0; 1901 for (iHeapIndex = 0; iHeapIndex < g_iHEAPSize; iHeapIndex++) { 1902 g_pucHeapMemory[iHeapIndex] = GetByte(); 1903 } 1904 1905 if (g_pucHeapMemory[iHeapIndex - 1] != ENDLOOP) { 1906 return VME_INVALID_FILE; 1907 } 1908 1909 g_usFlowControl |= REPEATLOOP; 1910 g_usDataType |= HEAP_IN; 1911 1912 for (iLoopIndex = 0; iLoopIndex < a_usLoopCount; iLoopIndex++) { 1913 g_iHeapCounter = 0; 1914 cRetCode = ispVMCode(); 1915 g_usRepeatLoops++; 1916 if (cRetCode < 0) { 1917 break; 1918 } 1919 } 1920 1921 g_usDataType &= ~(HEAP_IN); 1922 g_usFlowControl &= ~(REPEATLOOP); 1923 return cRetCode; 1924 } 1925 1926 /* 1927 * 1928 * ispVMBitShift 1929 * 1930 * Shift the TDI stream left or right by the number of bits. The data in 1931 * *g_pucInData is of the VME format, so the actual shifting is the reverse of 1932 * IEEE 1532 or SVF format. 1933 * 1934 */ 1935 1936 signed char ispVMBitShift(signed char mode, unsigned short bits) 1937 { 1938 /* 09/11/07 NN added local variables initialization */ 1939 unsigned short i = 0; 1940 unsigned short size = 0; 1941 unsigned short tmpbits = 0; 1942 1943 if (g_usiDataSize % 8 > 0) { 1944 /* 09/11/07 NN Type cast mismatch variables */ 1945 size = (unsigned short)(g_usiDataSize / 8 + 1); 1946 } else { 1947 /* 09/11/07 NN Type cast mismatch variables */ 1948 size = (unsigned short)(g_usiDataSize / 8); 1949 } 1950 1951 switch (mode) { 1952 case SHR: 1953 for (i = 0; i < size; i++) { 1954 if (g_pucInData[i] != 0) { 1955 tmpbits = bits; 1956 while (tmpbits > 0) { 1957 g_pucInData[i] <<= 1; 1958 if (g_pucInData[i] == 0) { 1959 i--; 1960 g_pucInData[i] = 1; 1961 } 1962 tmpbits--; 1963 } 1964 } 1965 } 1966 break; 1967 case SHL: 1968 for (i = 0; i < size; i++) { 1969 if (g_pucInData[i] != 0) { 1970 tmpbits = bits; 1971 while (tmpbits > 0) { 1972 g_pucInData[i] >>= 1; 1973 if (g_pucInData[i] == 0) { 1974 i--; 1975 g_pucInData[i] = 8; 1976 } 1977 tmpbits--; 1978 } 1979 } 1980 } 1981 break; 1982 default: 1983 return VME_INVALID_FILE; 1984 } 1985 1986 return 0; 1987 } 1988 1989 /* 1990 * 1991 * ispVMComment 1992 * 1993 * Displays the SVF comments. 1994 * 1995 */ 1996 1997 void ispVMComment(unsigned short a_usCommentSize) 1998 { 1999 char cCurByte = 0; 2000 for (; a_usCommentSize > 0; a_usCommentSize--) { 2001 /* 2002 * 2003 * Print character to the terminal. 2004 * 2005 **/ 2006 cCurByte = GetByte(); 2007 vme_out_char(cCurByte); 2008 } 2009 cCurByte = '\n'; 2010 vme_out_char(cCurByte); 2011 } 2012 2013 /* 2014 * 2015 * ispVMHeader 2016 * 2017 * Iterate the length of the header and discard it. 2018 * 2019 */ 2020 2021 void ispVMHeader(unsigned short a_usHeaderSize) 2022 { 2023 for (; a_usHeaderSize > 0; a_usHeaderSize--) { 2024 GetByte(); 2025 } 2026 } 2027 2028 /* 2029 * 2030 * ispVMCalculateCRC32 2031 * 2032 * Calculate the 32-bit CRC. 2033 * 2034 */ 2035 2036 void ispVMCalculateCRC32(unsigned char a_ucData) 2037 { 2038 /* 09/11/07 NN added local variables initialization */ 2039 unsigned char ucIndex = 0; 2040 unsigned char ucFlipData = 0; 2041 unsigned short usCRCTableEntry = 0; 2042 unsigned int crc_table[16] = { 2043 0x0000, 0xCC01, 0xD801, 2044 0x1400, 0xF001, 0x3C00, 2045 0x2800, 0xE401, 0xA001, 2046 0x6C00, 0x7800, 0xB401, 2047 0x5000, 0x9C01, 0x8801, 2048 0x4400 2049 }; 2050 2051 for (ucIndex = 0; ucIndex < 8; ucIndex++) { 2052 ucFlipData <<= 1; 2053 if (a_ucData & 0x01) { 2054 ucFlipData |= 0x01; 2055 } 2056 a_ucData >>= 1; 2057 } 2058 2059 /* 09/11/07 NN Type cast mismatch variables */ 2060 usCRCTableEntry = (unsigned short)(crc_table[g_usCalculatedCRC & 0xF]); 2061 g_usCalculatedCRC = (unsigned short)((g_usCalculatedCRC >> 4) & 0x0FFF); 2062 g_usCalculatedCRC = (unsigned short)(g_usCalculatedCRC ^ 2063 usCRCTableEntry ^ crc_table[ucFlipData & 0xF]); 2064 usCRCTableEntry = (unsigned short)(crc_table[g_usCalculatedCRC & 0xF]); 2065 g_usCalculatedCRC = (unsigned short)((g_usCalculatedCRC >> 4) & 0x0FFF); 2066 g_usCalculatedCRC = (unsigned short)(g_usCalculatedCRC ^ 2067 usCRCTableEntry ^ crc_table[(ucFlipData >> 4) & 0xF]); 2068 } 2069 2070 /* 2071 * 2072 * ispVMLCOUNT 2073 * 2074 * Process the intelligent programming loops. 2075 * 2076 */ 2077 2078 signed char ispVMLCOUNT(unsigned short a_usCountSize) 2079 { 2080 unsigned short usContinue = 1; 2081 unsigned short usIntelBufferIndex = 0; 2082 unsigned short usCountIndex = 0; 2083 signed char cRetCode = 0; 2084 signed char cRepeatHeap = 0; 2085 signed char cOpcode = 0; 2086 unsigned char ucState = 0; 2087 unsigned short usDelay = 0; 2088 unsigned short usToggle = 0; 2089 2090 g_usIntelBufferSize = (unsigned short)ispVMDataSize(); 2091 2092 /* 2093 * Allocate memory for intel buffer. 2094 * 2095 */ 2096 2097 ispVMMemManager(LHEAP, g_usIntelBufferSize); 2098 2099 /* 2100 * Store the maximum size of the intelligent buffer. 2101 * Used to convert VME to HEX. 2102 */ 2103 2104 if (g_usIntelBufferSize > g_usLCOUNTSize) { 2105 g_usLCOUNTSize = g_usIntelBufferSize; 2106 } 2107 2108 /* 2109 * Copy intel data to the buffer. 2110 */ 2111 2112 for (usIntelBufferIndex = 0; usIntelBufferIndex < g_usIntelBufferSize; 2113 usIntelBufferIndex++) { 2114 g_pucIntelBuffer[usIntelBufferIndex] = GetByte(); 2115 } 2116 2117 /* 2118 * Set the data type register to get data from the intelligent 2119 * data buffer. 2120 */ 2121 2122 g_usDataType |= LHEAP_IN; 2123 2124 /* 2125 * 2126 * If the HEAP_IN flag is set, temporarily unset the flag so data will be 2127 * retrieved from the status buffer. 2128 * 2129 **/ 2130 2131 if (g_usDataType & HEAP_IN) { 2132 g_usDataType &= ~HEAP_IN; 2133 cRepeatHeap = 1; 2134 } 2135 2136 #ifdef DEBUG 2137 printf("LCOUNT %d;\n", a_usCountSize); 2138 #endif /* DEBUG */ 2139 2140 /* 2141 * Iterate through the intelligent programming command. 2142 */ 2143 2144 for (usCountIndex = 0; usCountIndex < a_usCountSize; usCountIndex++) { 2145 2146 /* 2147 * 2148 * Initialize the intel data index to 0 before each iteration. 2149 * 2150 **/ 2151 2152 g_usIntelDataIndex = 0; 2153 cOpcode = 0; 2154 ucState = 0; 2155 usDelay = 0; 2156 usToggle = 0; 2157 usContinue = 1; 2158 2159 /* 2160 * 2161 * Begin looping through all the VME opcodes. 2162 * 2163 */ 2164 /* 2165 * 4/1/09 Nguyen replaced the recursive function call codes on 2166 * the ispVMLCOUNT function 2167 * 2168 */ 2169 while (usContinue) { 2170 cOpcode = GetByte(); 2171 switch (cOpcode) { 2172 case HIR: 2173 case TIR: 2174 case HDR: 2175 case TDR: 2176 /* 2177 * Set the header/trailer of the device in order 2178 * to bypass successfully. 2179 */ 2180 2181 ispVMAmble(cOpcode); 2182 break; 2183 case STATE: 2184 2185 /* 2186 * Step the JTAG state machine. 2187 */ 2188 2189 ucState = GetByte(); 2190 /* 2191 * Step the JTAG state machine to DRCAPTURE 2192 * to support Looping. 2193 */ 2194 2195 if ((g_usDataType & LHEAP_IN) && 2196 (ucState == DRPAUSE) && 2197 (g_cCurrentJTAGState == ucState)) { 2198 ispVMStateMachine(DRCAPTURE); 2199 } 2200 ispVMStateMachine(ucState); 2201 #ifdef DEBUG 2202 printf("LDELAY %s ", GetState(ucState)); 2203 #endif /* DEBUG */ 2204 break; 2205 case SIR: 2206 #ifdef DEBUG 2207 printf("SIR "); 2208 #endif /* DEBUG */ 2209 /* 2210 * Shift in data into the device. 2211 */ 2212 2213 cRetCode = ispVMShift(cOpcode); 2214 break; 2215 case SDR: 2216 2217 #ifdef DEBUG 2218 printf("LSDR "); 2219 #endif /* DEBUG */ 2220 /* 2221 * Shift in data into the device. 2222 */ 2223 2224 cRetCode = ispVMShift(cOpcode); 2225 break; 2226 case WAIT: 2227 2228 /* 2229 * 2230 * Observe delay. 2231 * 2232 */ 2233 2234 usDelay = (unsigned short)ispVMDataSize(); 2235 ispVMDelay(usDelay); 2236 2237 #ifdef DEBUG 2238 if (usDelay & 0x8000) { 2239 2240 /* 2241 * Since MSB is set, the delay time must 2242 * be decoded to millisecond. The 2243 * SVF2VME encodes the MSB to represent 2244 * millisecond. 2245 */ 2246 2247 usDelay &= ~0x8000; 2248 printf("%.2E SEC;\n", 2249 (float) usDelay / 1000); 2250 } else { 2251 /* 2252 * Since MSB is not set, the delay time 2253 * is given as microseconds. 2254 */ 2255 2256 printf("%.2E SEC;\n", 2257 (float) usDelay / 1000000); 2258 } 2259 #endif /* DEBUG */ 2260 break; 2261 case TCK: 2262 2263 /* 2264 * Issue clock toggles. 2265 */ 2266 2267 usToggle = (unsigned short)ispVMDataSize(); 2268 ispVMClocks(usToggle); 2269 2270 #ifdef DEBUG 2271 printf("RUNTEST %d TCK;\n", usToggle); 2272 #endif /* DEBUG */ 2273 break; 2274 case ENDLOOP: 2275 2276 /* 2277 * Exit point from processing loops. 2278 */ 2279 usContinue = 0; 2280 break; 2281 2282 case COMMENT: 2283 2284 /* 2285 * Display comment. 2286 */ 2287 2288 ispVMComment((unsigned short) ispVMDataSize()); 2289 break; 2290 case ispEN: 2291 ucState = GetByte(); 2292 if ((ucState == ON) || (ucState == 0x01)) 2293 writePort(g_ucPinENABLE, 0x01); 2294 else 2295 writePort(g_ucPinENABLE, 0x00); 2296 ispVMDelay(1); 2297 break; 2298 case TRST: 2299 if (GetByte() == 0x01) 2300 writePort(g_ucPinTRST, 0x01); 2301 else 2302 writePort(g_ucPinTRST, 0x00); 2303 ispVMDelay(1); 2304 break; 2305 default: 2306 2307 /* 2308 * Invalid opcode encountered. 2309 */ 2310 2311 debug("\nINVALID OPCODE: 0x%.2X\n", cOpcode); 2312 2313 return VME_INVALID_FILE; 2314 } 2315 } 2316 if (cRetCode >= 0) { 2317 /* 2318 * Break if intelligent programming is successful. 2319 */ 2320 2321 break; 2322 } 2323 2324 } 2325 /* 2326 * If HEAP_IN flag was temporarily disabled, 2327 * re-enable it before exiting 2328 */ 2329 2330 if (cRepeatHeap) { 2331 g_usDataType |= HEAP_IN; 2332 } 2333 2334 /* 2335 * Set the data type register to not get data from the 2336 * intelligent data buffer. 2337 */ 2338 2339 g_usDataType &= ~LHEAP_IN; 2340 return cRetCode; 2341 } 2342 /* 2343 * 2344 * ispVMClocks 2345 * 2346 * Applies the specified number of pulses to TCK. 2347 * 2348 */ 2349 2350 void ispVMClocks(unsigned short Clocks) 2351 { 2352 unsigned short iClockIndex = 0; 2353 for (iClockIndex = 0; iClockIndex < Clocks; iClockIndex++) { 2354 sclock(); 2355 } 2356 } 2357 2358 /* 2359 * 2360 * ispVMBypass 2361 * 2362 * This procedure takes care of the HIR, HDR, TIR, TDR for the 2363 * purpose of putting the other devices into Bypass mode. The 2364 * current state is checked to find out if it is at DRPAUSE or 2365 * IRPAUSE. If it is at DRPAUSE, perform bypass register scan. 2366 * If it is at IRPAUSE, scan into instruction registers the bypass 2367 * instruction. 2368 * 2369 */ 2370 2371 void ispVMBypass(signed char ScanType, unsigned short Bits) 2372 { 2373 /* 09/11/07 NN added local variables initialization */ 2374 unsigned short iIndex = 0; 2375 unsigned short iSourceIndex = 0; 2376 unsigned char cBitState = 0; 2377 unsigned char cCurByte = 0; 2378 unsigned char *pcSource = NULL; 2379 2380 if (Bits <= 0) { 2381 return; 2382 } 2383 2384 switch (ScanType) { 2385 case HIR: 2386 pcSource = g_pucHIRData; 2387 break; 2388 case TIR: 2389 pcSource = g_pucTIRData; 2390 break; 2391 case HDR: 2392 pcSource = g_pucHDRData; 2393 break; 2394 case TDR: 2395 pcSource = g_pucTDRData; 2396 break; 2397 default: 2398 break; 2399 } 2400 2401 iSourceIndex = 0; 2402 cBitState = 0; 2403 for (iIndex = 0; iIndex < Bits - 1; iIndex++) { 2404 /* Scan instruction or bypass register */ 2405 if (iIndex % 8 == 0) { 2406 cCurByte = pcSource[iSourceIndex++]; 2407 } 2408 cBitState = (unsigned char) (((cCurByte << iIndex % 8) & 0x80) 2409 ? 0x01 : 0x00); 2410 writePort(g_ucPinTDI, cBitState); 2411 sclock(); 2412 } 2413 2414 if (iIndex % 8 == 0) { 2415 cCurByte = pcSource[iSourceIndex++]; 2416 } 2417 2418 cBitState = (unsigned char) (((cCurByte << iIndex % 8) & 0x80) 2419 ? 0x01 : 0x00); 2420 writePort(g_ucPinTDI, cBitState); 2421 } 2422 2423 /* 2424 * 2425 * ispVMStateMachine 2426 * 2427 * This procedure steps all devices in the daisy chain from a given 2428 * JTAG state to the next desirable state. If the next state is TLR, 2429 * the JTAG state machine is brute forced into TLR by driving TMS 2430 * high and pulse TCK 6 times. 2431 * 2432 */ 2433 2434 void ispVMStateMachine(signed char cNextJTAGState) 2435 { 2436 /* 09/11/07 NN added local variables initialization */ 2437 signed char cPathIndex = 0; 2438 signed char cStateIndex = 0; 2439 2440 if ((g_cCurrentJTAGState == cNextJTAGState) && 2441 (cNextJTAGState != RESET)) { 2442 return; 2443 } 2444 2445 for (cStateIndex = 0; cStateIndex < 25; cStateIndex++) { 2446 if ((g_cCurrentJTAGState == 2447 g_JTAGTransistions[cStateIndex].CurState) && 2448 (cNextJTAGState == 2449 g_JTAGTransistions[cStateIndex].NextState)) { 2450 break; 2451 } 2452 } 2453 2454 g_cCurrentJTAGState = cNextJTAGState; 2455 for (cPathIndex = 0; 2456 cPathIndex < g_JTAGTransistions[cStateIndex].Pulses; 2457 cPathIndex++) { 2458 if ((g_JTAGTransistions[cStateIndex].Pattern << cPathIndex) 2459 & 0x80) { 2460 writePort(g_ucPinTMS, (unsigned char) 0x01); 2461 } else { 2462 writePort(g_ucPinTMS, (unsigned char) 0x00); 2463 } 2464 sclock(); 2465 } 2466 2467 writePort(g_ucPinTDI, 0x00); 2468 writePort(g_ucPinTMS, 0x00); 2469 } 2470 2471 /* 2472 * 2473 * ispVMStart 2474 * 2475 * Enable the port to the device and set the state to RESET (TLR). 2476 * 2477 */ 2478 2479 void ispVMStart() 2480 { 2481 #ifdef DEBUG 2482 printf("// ISPVM EMBEDDED ADDED\n"); 2483 printf("STATE RESET;\n"); 2484 #endif 2485 g_usFlowControl = 0; 2486 g_usDataType = g_uiChecksumIndex = g_cCurrentJTAGState = 0; 2487 g_usHeadDR = g_usHeadIR = g_usTailDR = g_usTailIR = 0; 2488 g_usMaxSize = g_usShiftValue = g_usRepeatLoops = 0; 2489 g_usTDOSize = g_usMASKSize = g_usTDISize = 0; 2490 g_usDMASKSize = g_usLCOUNTSize = g_usHDRSize = 0; 2491 g_usTDRSize = g_usHIRSize = g_usTIRSize = g_usHeapSize = 0; 2492 g_pLVDSList = NULL; 2493 g_usLVDSPairCount = 0; 2494 previous_size = 0; 2495 2496 ispVMStateMachine(RESET); /*step devices to RESET state*/ 2497 } 2498 2499 /* 2500 * 2501 * ispVMEnd 2502 * 2503 * Set the state of devices to RESET to enable the devices and disable 2504 * the port. 2505 * 2506 */ 2507 2508 void ispVMEnd() 2509 { 2510 #ifdef DEBUG 2511 printf("// ISPVM EMBEDDED ADDED\n"); 2512 printf("STATE RESET;\n"); 2513 printf("RUNTEST 1.00E-001 SEC;\n"); 2514 #endif 2515 2516 ispVMStateMachine(RESET); /*step devices to RESET state */ 2517 ispVMDelay(1000); /*wake up devices*/ 2518 } 2519 2520 /* 2521 * 2522 * ispVMSend 2523 * 2524 * Send the TDI data stream to devices. The data stream can be 2525 * instructions or data. 2526 * 2527 */ 2528 2529 signed char ispVMSend(unsigned short a_usiDataSize) 2530 { 2531 /* 09/11/07 NN added local variables initialization */ 2532 unsigned short iIndex = 0; 2533 unsigned short iInDataIndex = 0; 2534 unsigned char cCurByte = 0; 2535 unsigned char cBitState = 0; 2536 2537 for (iIndex = 0; iIndex < a_usiDataSize - 1; iIndex++) { 2538 if (iIndex % 8 == 0) { 2539 cCurByte = g_pucInData[iInDataIndex++]; 2540 } 2541 cBitState = (unsigned char)(((cCurByte << iIndex % 8) & 0x80) 2542 ? 0x01 : 0x00); 2543 writePort(g_ucPinTDI, cBitState); 2544 sclock(); 2545 } 2546 2547 if (iIndex % 8 == 0) { 2548 /* Take care of the last bit */ 2549 cCurByte = g_pucInData[iInDataIndex]; 2550 } 2551 2552 cBitState = (unsigned char) (((cCurByte << iIndex % 8) & 0x80) 2553 ? 0x01 : 0x00); 2554 2555 writePort(g_ucPinTDI, cBitState); 2556 if (g_usFlowControl & CASCADE) { 2557 /*1/15/04 Clock in last bit for the first n-1 cascaded frames */ 2558 sclock(); 2559 } 2560 2561 return 0; 2562 } 2563 2564 /* 2565 * 2566 * ispVMRead 2567 * 2568 * Read the data stream from devices and verify. 2569 * 2570 */ 2571 2572 signed char ispVMRead(unsigned short a_usiDataSize) 2573 { 2574 /* 09/11/07 NN added local variables initialization */ 2575 unsigned short usDataSizeIndex = 0; 2576 unsigned short usErrorCount = 0; 2577 unsigned short usLastBitIndex = 0; 2578 unsigned char cDataByte = 0; 2579 unsigned char cMaskByte = 0; 2580 unsigned char cInDataByte = 0; 2581 unsigned char cCurBit = 0; 2582 unsigned char cByteIndex = 0; 2583 unsigned short usBufferIndex = 0; 2584 unsigned char ucDisplayByte = 0x00; 2585 unsigned char ucDisplayFlag = 0x01; 2586 char StrChecksum[256] = {0}; 2587 unsigned char g_usCalculateChecksum = 0x00; 2588 2589 /* 09/11/07 NN Type cast mismatch variables */ 2590 usLastBitIndex = (unsigned short)(a_usiDataSize - 1); 2591 2592 #ifndef DEBUG 2593 /* 2594 * If mask is not all zeros, then set the display flag to 0x00, 2595 * otherwise it shall be set to 0x01 to indicate that data read 2596 * from the device shall be displayed. If DEBUG is defined, 2597 * always display data. 2598 */ 2599 2600 for (usDataSizeIndex = 0; usDataSizeIndex < (a_usiDataSize + 7) / 8; 2601 usDataSizeIndex++) { 2602 if (g_usDataType & MASK_DATA) { 2603 if (g_pucOutMaskData[usDataSizeIndex] != 0x00) { 2604 ucDisplayFlag = 0x00; 2605 break; 2606 } 2607 } else if (g_usDataType & CMASK_DATA) { 2608 g_usCalculateChecksum = 0x01; 2609 ucDisplayFlag = 0x00; 2610 break; 2611 } else { 2612 ucDisplayFlag = 0x00; 2613 break; 2614 } 2615 } 2616 #endif /* DEBUG */ 2617 2618 /* 2619 * 2620 * Begin shifting data in and out of the device. 2621 * 2622 **/ 2623 2624 for (usDataSizeIndex = 0; usDataSizeIndex < a_usiDataSize; 2625 usDataSizeIndex++) { 2626 if (cByteIndex == 0) { 2627 2628 /* 2629 * Grab byte from TDO buffer. 2630 */ 2631 2632 if (g_usDataType & TDO_DATA) { 2633 cDataByte = g_pucOutData[usBufferIndex]; 2634 } 2635 2636 /* 2637 * Grab byte from MASK buffer. 2638 */ 2639 2640 if (g_usDataType & MASK_DATA) { 2641 cMaskByte = g_pucOutMaskData[usBufferIndex]; 2642 } else { 2643 cMaskByte = 0xFF; 2644 } 2645 2646 /* 2647 * Grab byte from CMASK buffer. 2648 */ 2649 2650 if (g_usDataType & CMASK_DATA) { 2651 cMaskByte = 0x00; 2652 g_usCalculateChecksum = 0x01; 2653 } 2654 2655 /* 2656 * Grab byte from TDI buffer. 2657 */ 2658 2659 if (g_usDataType & TDI_DATA) { 2660 cInDataByte = g_pucInData[usBufferIndex]; 2661 } 2662 2663 usBufferIndex++; 2664 } 2665 2666 cCurBit = readPort(); 2667 2668 if (ucDisplayFlag) { 2669 ucDisplayByte <<= 1; 2670 ucDisplayByte |= cCurBit; 2671 } 2672 2673 /* 2674 * Check if data read from port matches with expected TDO. 2675 */ 2676 2677 if (g_usDataType & TDO_DATA) { 2678 /* 08/28/08 NN Added Calculate checksum support. */ 2679 if (g_usCalculateChecksum) { 2680 if (cCurBit == 0x01) 2681 g_usChecksum += 2682 (1 << (g_uiChecksumIndex % 8)); 2683 g_uiChecksumIndex++; 2684 } else { 2685 if ((((cMaskByte << cByteIndex) & 0x80) 2686 ? 0x01 : 0x00)) { 2687 if (cCurBit != (unsigned char) 2688 (((cDataByte << cByteIndex) & 0x80) 2689 ? 0x01 : 0x00)) { 2690 usErrorCount++; 2691 } 2692 } 2693 } 2694 } 2695 2696 /* 2697 * Write TDI data to the port. 2698 */ 2699 2700 writePort(g_ucPinTDI, 2701 (unsigned char)(((cInDataByte << cByteIndex) & 0x80) 2702 ? 0x01 : 0x00)); 2703 2704 if (usDataSizeIndex < usLastBitIndex) { 2705 2706 /* 2707 * Clock data out from the data shift register. 2708 */ 2709 2710 sclock(); 2711 } else if (g_usFlowControl & CASCADE) { 2712 2713 /* 2714 * Clock in last bit for the first N - 1 cascaded frames 2715 */ 2716 2717 sclock(); 2718 } 2719 2720 /* 2721 * Increment the byte index. If it exceeds 7, then reset it back 2722 * to zero. 2723 */ 2724 2725 cByteIndex++; 2726 if (cByteIndex >= 8) { 2727 if (ucDisplayFlag) { 2728 2729 /* 2730 * Store displayed data in the TDO buffer. By reusing 2731 * the TDO buffer to store displayed data, there is no 2732 * need to allocate a buffer simply to hold display 2733 * data. This will not cause any false verification 2734 * errors because the true TDO byte has already 2735 * been consumed. 2736 */ 2737 2738 g_pucOutData[usBufferIndex - 1] = ucDisplayByte; 2739 ucDisplayByte = 0; 2740 } 2741 2742 cByteIndex = 0; 2743 } 2744 /* 09/12/07 Nguyen changed to display the 1 bit expected data */ 2745 else if (a_usiDataSize == 1) { 2746 if (ucDisplayFlag) { 2747 2748 /* 2749 * Store displayed data in the TDO buffer. 2750 * By reusing the TDO buffer to store displayed 2751 * data, there is no need to allocate 2752 * a buffer simply to hold display data. This 2753 * will not cause any false verification errors 2754 * because the true TDO byte has already 2755 * been consumed. 2756 */ 2757 2758 /* 2759 * Flip ucDisplayByte and store it in cDataByte. 2760 */ 2761 cDataByte = 0x00; 2762 for (usBufferIndex = 0; usBufferIndex < 8; 2763 usBufferIndex++) { 2764 cDataByte <<= 1; 2765 if (ucDisplayByte & 0x01) { 2766 cDataByte |= 0x01; 2767 } 2768 ucDisplayByte >>= 1; 2769 } 2770 g_pucOutData[0] = cDataByte; 2771 ucDisplayByte = 0; 2772 } 2773 2774 cByteIndex = 0; 2775 } 2776 } 2777 2778 if (ucDisplayFlag) { 2779 2780 #ifdef DEBUG 2781 debug("RECEIVED TDO ("); 2782 #else 2783 vme_out_string("Display Data: 0x"); 2784 #endif /* DEBUG */ 2785 2786 /* 09/11/07 NN Type cast mismatch variables */ 2787 for (usDataSizeIndex = (unsigned short) 2788 ((a_usiDataSize + 7) / 8); 2789 usDataSizeIndex > 0 ; usDataSizeIndex--) { 2790 cMaskByte = g_pucOutData[usDataSizeIndex - 1]; 2791 cDataByte = 0x00; 2792 2793 /* 2794 * Flip cMaskByte and store it in cDataByte. 2795 */ 2796 2797 for (usBufferIndex = 0; usBufferIndex < 8; 2798 usBufferIndex++) { 2799 cDataByte <<= 1; 2800 if (cMaskByte & 0x01) { 2801 cDataByte |= 0x01; 2802 } 2803 cMaskByte >>= 1; 2804 } 2805 #ifdef DEBUG 2806 printf("%.2X", cDataByte); 2807 if ((((a_usiDataSize + 7) / 8) - usDataSizeIndex) 2808 % 40 == 39) { 2809 printf("\n\t\t"); 2810 } 2811 #else 2812 vme_out_hex(cDataByte); 2813 #endif /* DEBUG */ 2814 } 2815 2816 #ifdef DEBUG 2817 printf(")\n\n"); 2818 #else 2819 vme_out_string("\n\n"); 2820 #endif /* DEBUG */ 2821 /* 09/02/08 Nguyen changed to display the data Checksum */ 2822 if (g_usChecksum != 0) { 2823 g_usChecksum &= 0xFFFF; 2824 sprintf(StrChecksum, "Data Checksum: %.4lX\n\n", 2825 g_usChecksum); 2826 vme_out_string(StrChecksum); 2827 g_usChecksum = 0; 2828 } 2829 } 2830 2831 if (usErrorCount > 0) { 2832 if (g_usFlowControl & VERIFYUES) { 2833 vme_out_string( 2834 "USERCODE verification failed. " 2835 "Continue programming......\n\n"); 2836 g_usFlowControl &= ~(VERIFYUES); 2837 return 0; 2838 } else { 2839 2840 #ifdef DEBUG 2841 printf("TOTAL ERRORS: %d\n", usErrorCount); 2842 #endif /* DEBUG */ 2843 2844 return VME_VERIFICATION_FAILURE; 2845 } 2846 } else { 2847 if (g_usFlowControl & VERIFYUES) { 2848 vme_out_string("USERCODE verification passed. " 2849 "Programming aborted.\n\n"); 2850 g_usFlowControl &= ~(VERIFYUES); 2851 return 1; 2852 } else { 2853 return 0; 2854 } 2855 } 2856 } 2857 2858 /* 2859 * 2860 * ispVMReadandSave 2861 * 2862 * Support dynamic I/O. 2863 * 2864 */ 2865 2866 signed char ispVMReadandSave(unsigned short int a_usiDataSize) 2867 { 2868 /* 09/11/07 NN added local variables initialization */ 2869 unsigned short int usDataSizeIndex = 0; 2870 unsigned short int usLastBitIndex = 0; 2871 unsigned short int usBufferIndex = 0; 2872 unsigned short int usOutBitIndex = 0; 2873 unsigned short int usLVDSIndex = 0; 2874 unsigned char cDataByte = 0; 2875 unsigned char cDMASKByte = 0; 2876 unsigned char cInDataByte = 0; 2877 unsigned char cCurBit = 0; 2878 unsigned char cByteIndex = 0; 2879 signed char cLVDSByteIndex = 0; 2880 2881 /* 09/11/07 NN Type cast mismatch variables */ 2882 usLastBitIndex = (unsigned short) (a_usiDataSize - 1); 2883 2884 /* 2885 * 2886 * Iterate through the data bits. 2887 * 2888 */ 2889 2890 for (usDataSizeIndex = 0; usDataSizeIndex < a_usiDataSize; 2891 usDataSizeIndex++) { 2892 if (cByteIndex == 0) { 2893 2894 /* 2895 * Grab byte from DMASK buffer. 2896 */ 2897 2898 if (g_usDataType & DMASK_DATA) { 2899 cDMASKByte = g_pucOutDMaskData[usBufferIndex]; 2900 } else { 2901 cDMASKByte = 0x00; 2902 } 2903 2904 /* 2905 * Grab byte from TDI buffer. 2906 */ 2907 2908 if (g_usDataType & TDI_DATA) { 2909 cInDataByte = g_pucInData[usBufferIndex]; 2910 } 2911 2912 usBufferIndex++; 2913 } 2914 2915 cCurBit = readPort(); 2916 cDataByte = (unsigned char)(((cInDataByte << cByteIndex) & 0x80) 2917 ? 0x01 : 0x00); 2918 2919 /* 2920 * Initialize the byte to be zero. 2921 */ 2922 2923 if (usOutBitIndex % 8 == 0) { 2924 g_pucOutData[usOutBitIndex / 8] = 0x00; 2925 } 2926 2927 /* 2928 * Use TDI, DMASK, and device TDO to create new TDI (actually 2929 * stored in g_pucOutData). 2930 */ 2931 2932 if ((((cDMASKByte << cByteIndex) & 0x80) ? 0x01 : 0x00)) { 2933 2934 if (g_pLVDSList) { 2935 for (usLVDSIndex = 0; 2936 usLVDSIndex < g_usLVDSPairCount; 2937 usLVDSIndex++) { 2938 if (g_pLVDSList[usLVDSIndex]. 2939 usNegativeIndex == 2940 usDataSizeIndex) { 2941 g_pLVDSList[usLVDSIndex]. 2942 ucUpdate = 0x01; 2943 break; 2944 } 2945 } 2946 } 2947 2948 /* 2949 * DMASK bit is 1, use TDI. 2950 */ 2951 2952 g_pucOutData[usOutBitIndex / 8] |= (unsigned char) 2953 (((cDataByte & 0x1) ? 0x01 : 0x00) << 2954 (7 - usOutBitIndex % 8)); 2955 } else { 2956 2957 /* 2958 * DMASK bit is 0, use device TDO. 2959 */ 2960 2961 g_pucOutData[usOutBitIndex / 8] |= (unsigned char) 2962 (((cCurBit & 0x1) ? 0x01 : 0x00) << 2963 (7 - usOutBitIndex % 8)); 2964 } 2965 2966 /* 2967 * Shift in TDI in order to get TDO out. 2968 */ 2969 2970 usOutBitIndex++; 2971 writePort(g_ucPinTDI, cDataByte); 2972 if (usDataSizeIndex < usLastBitIndex) { 2973 sclock(); 2974 } 2975 2976 /* 2977 * Increment the byte index. If it exceeds 7, then reset it back 2978 * to zero. 2979 */ 2980 2981 cByteIndex++; 2982 if (cByteIndex >= 8) { 2983 cByteIndex = 0; 2984 } 2985 } 2986 2987 /* 2988 * If g_pLVDSList exists and pairs need updating, then update 2989 * the negative-pair to receive the flipped positive-pair value. 2990 */ 2991 2992 if (g_pLVDSList) { 2993 for (usLVDSIndex = 0; usLVDSIndex < g_usLVDSPairCount; 2994 usLVDSIndex++) { 2995 if (g_pLVDSList[usLVDSIndex].ucUpdate) { 2996 2997 /* 2998 * Read the positive value and flip it. 2999 */ 3000 3001 cDataByte = (unsigned char) 3002 (((g_pucOutData[g_pLVDSList[usLVDSIndex]. 3003 usPositiveIndex / 8] 3004 << (g_pLVDSList[usLVDSIndex]. 3005 usPositiveIndex % 8)) & 0x80) ? 3006 0x01 : 0x00); 3007 /* 09/11/07 NN Type cast mismatch variables */ 3008 cDataByte = (unsigned char) (!cDataByte); 3009 3010 /* 3011 * Get the byte that needs modification. 3012 */ 3013 3014 cInDataByte = 3015 g_pucOutData[g_pLVDSList[usLVDSIndex]. 3016 usNegativeIndex / 8]; 3017 3018 if (cDataByte) { 3019 3020 /* 3021 * Copy over the current byte and 3022 * set the negative bit to 1. 3023 */ 3024 3025 cDataByte = 0x00; 3026 for (cLVDSByteIndex = 7; 3027 cLVDSByteIndex >= 0; 3028 cLVDSByteIndex--) { 3029 cDataByte <<= 1; 3030 if (7 - 3031 (g_pLVDSList[usLVDSIndex]. 3032 usNegativeIndex % 8) == 3033 cLVDSByteIndex) { 3034 3035 /* 3036 * Set negative bit to 1 3037 */ 3038 3039 cDataByte |= 0x01; 3040 } else if (cInDataByte & 0x80) { 3041 cDataByte |= 0x01; 3042 } 3043 3044 cInDataByte <<= 1; 3045 } 3046 3047 /* 3048 * Store the modified byte. 3049 */ 3050 3051 g_pucOutData[g_pLVDSList[usLVDSIndex]. 3052 usNegativeIndex / 8] = cDataByte; 3053 } else { 3054 3055 /* 3056 * Copy over the current byte and set 3057 * the negative bit to 0. 3058 */ 3059 3060 cDataByte = 0x00; 3061 for (cLVDSByteIndex = 7; 3062 cLVDSByteIndex >= 0; 3063 cLVDSByteIndex--) { 3064 cDataByte <<= 1; 3065 if (7 - 3066 (g_pLVDSList[usLVDSIndex]. 3067 usNegativeIndex % 8) == 3068 cLVDSByteIndex) { 3069 3070 /* 3071 * Set negative bit to 0 3072 */ 3073 3074 cDataByte |= 0x00; 3075 } else if (cInDataByte & 0x80) { 3076 cDataByte |= 0x01; 3077 } 3078 3079 cInDataByte <<= 1; 3080 } 3081 3082 /* 3083 * Store the modified byte. 3084 */ 3085 3086 g_pucOutData[g_pLVDSList[usLVDSIndex]. 3087 usNegativeIndex / 8] = cDataByte; 3088 } 3089 3090 break; 3091 } 3092 } 3093 } 3094 3095 return 0; 3096 } 3097 3098 signed char ispVMProcessLVDS(unsigned short a_usLVDSCount) 3099 { 3100 unsigned short usLVDSIndex = 0; 3101 3102 /* 3103 * Allocate memory to hold LVDS pairs. 3104 */ 3105 3106 ispVMMemManager(LVDS, a_usLVDSCount); 3107 g_usLVDSPairCount = a_usLVDSCount; 3108 3109 #ifdef DEBUG 3110 printf("LVDS %d (", a_usLVDSCount); 3111 #endif /* DEBUG */ 3112 3113 /* 3114 * Iterate through each given LVDS pair. 3115 */ 3116 3117 for (usLVDSIndex = 0; usLVDSIndex < g_usLVDSPairCount; usLVDSIndex++) { 3118 3119 /* 3120 * Assign the positive and negative indices of the LVDS pair. 3121 */ 3122 3123 /* 09/11/07 NN Type cast mismatch variables */ 3124 g_pLVDSList[usLVDSIndex].usPositiveIndex = 3125 (unsigned short) ispVMDataSize(); 3126 /* 09/11/07 NN Type cast mismatch variables */ 3127 g_pLVDSList[usLVDSIndex].usNegativeIndex = 3128 (unsigned short)ispVMDataSize(); 3129 3130 #ifdef DEBUG 3131 if (usLVDSIndex < g_usLVDSPairCount - 1) { 3132 printf("%d:%d, ", 3133 g_pLVDSList[usLVDSIndex].usPositiveIndex, 3134 g_pLVDSList[usLVDSIndex].usNegativeIndex); 3135 } else { 3136 printf("%d:%d", 3137 g_pLVDSList[usLVDSIndex].usPositiveIndex, 3138 g_pLVDSList[usLVDSIndex].usNegativeIndex); 3139 } 3140 #endif /* DEBUG */ 3141 3142 } 3143 3144 #ifdef DEBUG 3145 printf(");\n"); 3146 #endif /* DEBUG */ 3147 3148 return 0; 3149 } 3150