home *** CD-ROM | disk | FTP | other *** search
/ World of A1200 / World_Of_A1200.iso / programs / develop / galer / galer_deutsch / source / galersrcd.lha / GALasm.c < prev    next >
C/C++ Source or Header  |  1993-03-28  |  31KB  |  1,199 lines

  1. /****************************************************************/
  2. /*                                */
  3. /* GALAsm.c - enthält GAL-Assembler                */
  4. /*                                */
  5. /* compilieren: cc GALAsm.c                    */
  6. /*                                */
  7. /****************************************************************/
  8.  
  9.  
  10.  
  11. #include <exec/types.h>
  12. #include <exec/memory.h>
  13. #include <intuition/intuition.h>
  14. #include <ctype.h>
  15. #include <stdio.h>
  16. #include <functions.h>
  17.  
  18. #include "GALer.h"
  19.  
  20.  
  21.  
  22. extern    int    linenum, MaxFuseAdr, SigAdr, GALType;
  23. extern    UBYTE    *actptr, *buffend;
  24. extern  char    path;
  25.  
  26. extern    struct    JedecStruct    Jedec;
  27. extern    struct    Configuration    Config;
  28. extern    struct    RastPort    *rp;
  29. extern    struct    MenuItem    MenuItem13, MenuItem13b;  /*GAL16V8, GAL20V8*/
  30.  
  31.  
  32. struct    Pin        actPin;
  33. struct    GAL_OLMC    OLMC[8];
  34.  
  35.  
  36.             /*Diese Arrays geben an in welche Spalte der ent- */
  37.             /*sprechende Pin eingekoppelt (bzw. rückgekoppelt)*/
  38.             /*wird. Für die invertierende Einkopplung ist 1 zu*/
  39.             /*addieren, um die entsprechende Spalte zu erhalten*/
  40.             /* -1: keine Einkopplung auf Fuse-Matrix vorhanden */
  41.             /*GAL16V8:*/
  42. int    PinToFuse16Mode1[20]    = {  2, 0, 4, 8,12,16,20,24,28,-1,
  43.                     30,26,22,18,-1,-1,14,10, 6,-1 };
  44. int    PinToFuse16Mode2[20]    = {  2, 0, 4, 8,12,16,20,24,28,-1,
  45.                     30,-1,26,22,18,14,10, 6,-1,-1 };
  46. int    PinToFuse16Mode3[20]    = { -1, 0, 4, 8,12,16,20,24,28,-1,
  47.                     -1,30,26,22,18,14,10, 6, 2,-1 };
  48.  
  49.             /*GAL20V8:*/
  50. int    PinToFuse20Mode1[24]    = {  2, 0, 4, 8,12,16,20,24,28,32,36,-1,
  51.                     38,34,30,26,22,-1,-1,18,14,10, 6,-1 };
  52. int    PinToFuse20Mode2[24]    = {  2, 0, 4, 8,12,16,20,24,28,32,36,-1,
  53.                     38,34,-1,30,26,22,18,14,10,-1, 6,-1 };
  54. int    PinToFuse20Mode3[24]    = { -1, 0, 4, 8,12,16,20,24,28,32,36,-1,
  55.                     -1,38,34,30,26,22,18,14,10, 6, 2,-1 };
  56.  
  57.             /*dieses Array gibt an in welcher Zeile der*/
  58.             /*erste Eingang der OLMC-Zelle liegt*/
  59. int    ToOLMC[8]        = { 56,48,40,32,24,16, 8, 0 };
  60.  
  61.  
  62.  
  63. UBYTE    PinNames[24][10], PinNamesOpt[24][10];
  64. UBYTE    PinDecNeg[24];
  65. UBYTE    ModeErrorStr[] = {"Modus x:  Pin xx"};
  66. UBYTE    *pinnames;
  67.  
  68.  
  69. LONG    fsize;
  70. UBYTE    *fbuff;
  71.  
  72. int    num_of_pins, num_of_col, modus, gal_type;
  73. int    asmreadyflag;
  74.  
  75.  
  76. /* AssembleInputFile:
  77.    Eingabe-Datei assemblieren
  78.    Aufruf: AssembleInputFile(OpMode);    (OpMode = OperationMode)
  79.    Parameter: OpMode = ASSEMBLER : assembliere das Source-File und erstelle
  80.                    die Jedec-Struktur
  81.              = OPTIMIZER : assembliere das Source-File bis zu den
  82.                    Booleschen Gleichung und brich dann ab
  83.    Ergebnis: 0: alles o.k.   sonst: Fehler oder Abbruch
  84. */
  85. int AssembleInputFile(OpMode)
  86. int    OpMode;
  87. {
  88. UBYTE    chr;
  89. UBYTE    *bool_start;
  90. char    prevOp;
  91. char    *filereqtxt;
  92. int    i, j, k, l, n, m;
  93. int    oldMaxFuseAdr, max_chr;
  94. int    pass, pin_num, bool_linenum;
  95. int    actOLMC, row_offset, a_type, newline, oldline;
  96.  
  97.  
  98. m = YES;
  99. if (OpMode == ASSEMBLER)
  100.   m = AsmRequester();
  101.  
  102. if (m) {
  103.  if (OpMode == ASSEMBLER)
  104.    filereqtxt = (char *)"Source-File laden";
  105.  else
  106.    filereqtxt = (char *)"zu optimierendes Source-File laden";
  107.  
  108.  if (MyFileReq(filereqtxt,(char *)".pld",YES)) {
  109.    fsize=FileSize(&path);
  110.    switch (fsize) {
  111.      case -1L: {
  112.        ErrorReq(1);
  113.        return(-1);
  114.        break;
  115.       }
  116.      case -2L: {
  117.        ErrorReq(2);
  118.        return(-2);
  119.        break;
  120.       }
  121.      case 0L: {
  122.        ErrorReq(4);
  123.        return(-4);
  124.        break;
  125.       }
  126.     }
  127.    if ((fbuff=(UBYTE *)AllocMem(fsize,MEMF_PUBLIC))) {
  128.      if ((ReadFile(&path,fsize,fbuff))) {
  129.        PrintText((UBYTE *)"Datei ist geladen",1);
  130.  
  131.  
  132.        actptr  = fbuff;
  133.        buffend = fbuff+fsize;
  134.        linenum = 1;
  135.  
  136.        for (n=0; n<sizeof(Jedec); n++) {    /*Jedec-Sruktur zurücksetzen*/
  137.      if (n < LOGIC20_SIZE)
  138.        Jedec.GALLogic[n] = 1;        /*Fuse-Matrix auf 1*/
  139.      else
  140.        Jedec.GALLogic[n] = 0;        /*ACW, PT usw. auf 0*/
  141.     }
  142.  
  143.        for (n=0; n<8; n++) {            /*OLMC-Sruktur auf 0 setzen*/
  144.      OLMC[n].Active   = 0;
  145.      OLMC[n].PinType  = 0;
  146.      OLMC[n].TriCon   = 0;
  147.      OLMC[n].FeedBack = 0;
  148.     }
  149.  
  150.                     /*GAL-Typ feststellen*/
  151.        a_type = 0;
  152.        if ((!strncmp(actptr,(UBYTE *)"GAL16V8A",8)) || (!strncmp(actptr,(UBYTE *)"GAL20V8A",8)) ||
  153.            (!strncmp(actptr,(UBYTE *)"GAL16V8B",8)) || (!strncmp(actptr,(UBYTE *)"GAL20V8B",8))) {
  154.      a_type = 1;
  155.      if ( (*(actptr+8L) != ' ') && (*(actptr+8L) != 0x0A) && (*(actptr+8L) != 0x09) ) {
  156.        AsmError(1);
  157.        return(-1);
  158.       }
  159.     }
  160.  
  161.        if (!strncmp(actptr,(UBYTE *)"GAL16V8",7)) {
  162.      num_of_pins = 20;            /*Anzahl der Pins   */
  163.      num_of_col  = MAX_FUSE_ADR16 + 1;    /*Anzahl der Spalten*/
  164.      gal_type    = GAL16V8;
  165.         }
  166.        else
  167.          if (!strncmp(actptr,(UBYTE *)"GAL20V8",7)) {
  168.        num_of_pins = 24;            /*Anzahl der Pins   */
  169.        num_of_col  = MAX_FUSE_ADR20 + 1;    /*Anzahl der Spalten*/
  170.        gal_type    = GAL20V8;
  171.       }
  172.      else {
  173.        AsmError(1);
  174.        return(-1);
  175.       }
  176.  
  177.        if ((!a_type) && ((*(actptr+7L) != ' ') && (*(actptr+7L) != 0x0A) && (*(actptr+7L) != 0x09))) {
  178.      AsmError(1);
  179.      return(-1);
  180.     }
  181.  
  182.  
  183.  
  184.                     /*Signatur (8 Bytes der 2.Zeile) in Puffer schreiben*/
  185.        if (GetNextLine()) {            /*Datei-Ende erreicht?*/
  186.      AsmError(2);                /*ja, dann Fehler*/
  187.      return(-1);
  188.     }
  189.        n = m = 0;                /*Signatur in Jedec-Stuktur schreiben*/
  190.        while((*actptr != 0x0A) && (*actptr != 0x09) && (n < 8)) { /*CR,TAB*/
  191.      chr = *actptr;
  192.      for (m=0; m<8; m++) {
  193.        Jedec.GALSig[n*8+m] = (chr>>(7-m)) & 0x1; 
  194.       }
  195.      actptr++;
  196.      n++;
  197.      if (actptr>buffend) {            /*Fileende erreicht ?*/
  198.        AsmError(2);                /*ja, dann Fehler*/
  199.        return(-1);
  200.       }
  201.     } 
  202.  
  203.                     /*Pin-Namen eintragen*/
  204.        for (n=0; n<24; PinDecNeg[n++]=0); /*Merker für Neg. bei Pindeklara-*/
  205.                       /*tion*/
  206.  
  207.        pinnames = &PinNames[0][0];    /*für Assembler in "PinNames"*/
  208.        if (OpMode == OPTIMIZER)
  209.      pinnames = &PinNamesOpt[0][0];    /*für Optimizer in "PinNamesOpt"*/
  210.  
  211.        asmreadyflag = 0;            /*noch nicht assembliert*/
  212.        GetNextLine();
  213.        for (n=0; n<num_of_pins; n++) {
  214.       if (GetNextChar()) {            /*Dateiende?*/
  215.         AsmError(2);            /*ja, dann Fehler*/
  216.         return(-1);
  217.        }
  218.       m = 0;
  219.       chr = *actptr;
  220.       if (chr == '/') {
  221.         max_chr      = 10;
  222.         PinDecNeg[n] = 1;            /*Neg. bei Pindeklaration*/
  223.        }
  224.       else
  225.         max_chr = 9;
  226.       if (!(isalpha(chr) || isdigit(chr) || chr=='/')) {
  227.         AsmError(5);
  228.         return(-1);
  229.        }
  230.       k = 0;
  231.       while (isalpha(chr) || isdigit(chr) || chr=='/') {
  232.          if ((chr == '/') && (k != 0)) {    /* '/' nicht am Anfang des Pinnamens?*/
  233.            AsmError(10);
  234.            return(-1);
  235.           }
  236.          k = 1;
  237.          actptr++;
  238.          if ((chr=='/') && (!(isalpha(*actptr) || isdigit(*actptr)))) {
  239.            AsmError(3);
  240.            return(-1);
  241.           }
  242.          *(pinnames+n*10+m) = chr;
  243.          m++;
  244.          chr = *actptr;
  245.          if (m == max_chr) {        /*zuviele Buchstaben?*/
  246.            AsmError(4);
  247.            return(-1);
  248.           }
  249.         }
  250.       *(pinnames+n*10+m) = 0;        /*Stringende kennzeichnen*/
  251.  
  252.       for (l=0; l<n; l++) {            /*Pinname doppelt?*/
  253.         if (strcmp(pinnames+l*10,(UBYTE *)"NC")) {
  254.           i = j = 0;
  255.           if (*(pinnames+l*10) == '/') i = 1;
  256.           if (*(pinnames+n*10) == '/') j = 1;
  257.           if (!strcmp(pinnames+l*10+i,pinnames+n*10+j)) {
  258.         AsmError(9);
  259.         return(-1);
  260.            }
  261.          }
  262.        }
  263.                         /*GND an entsprechenden Pin?*/
  264.       if (!strcmp(pinnames+n*10,(UBYTE *)"GND")) {
  265.         if (n+1 != num_of_pins/2) {
  266.           AsmError(6);
  267.           return(-1);
  268.          }
  269.        }
  270.       if (n+1 == num_of_pins/2) {
  271.         if (strcmp(pinnames+n*10,(UBYTE *)"GND")) {
  272.           AsmError(8);
  273.           return(-1);
  274.          }
  275.        }
  276.                         /*VCC an entsprechenden Pin?*/
  277.       if (!strcmp(pinnames+n*10,(UBYTE *)"VCC")) {
  278.         if (n+1 != num_of_pins) {
  279.           AsmError(6);
  280.           return(-1);
  281.          }
  282.        }
  283.       if (n+1 == num_of_pins) {
  284.         if (strcmp(pinnames+n*10,(UBYTE *)"VCC")) {
  285.           AsmError(7);
  286.           return(-1);
  287.          }
  288.        }
  289.     }
  290.  
  291.  
  292.       if (OpMode == OPTIMIZER)        /*falls vom Optimizer aufgerufen,*/
  293.     return(0);            /*dann fertig*/
  294.  
  295.  
  296.  
  297. /* Boolean-Equations auswerten:
  298.    Dabei werden die Boolean-Equations zweimal untersucht. Beim ersten
  299.    Durchlauf werden die OLMC-Pins ausgewertet und die OLMC-Struktur ge-
  300.    füllt. Mit Hilfe dieser Struktur läßt sich auf dem notwendigen Modus
  301.    (1, 2 oder 3) schließen. Beim zweiten Durchlauf wird dann die
  302.    Fuse-Matrix erstellt.
  303.  */
  304.  
  305.        if (GetNextChar()) {            /*Dateiende?*/
  306.      AsmError(2);                /*ja, dann Fehler*/
  307.      return(-1);
  308.     }
  309.  
  310.                         /*Gleichungen vorhanden?*/
  311.        if (!strncmp(actptr,(UBYTE *)"DESCRIPTION",11)) {
  312.      AsmError(33);                /*nein, dann Fehlermeldung*/
  313.      return(-1);
  314.     }
  315.  
  316.  
  317.        bool_start   = actptr;            /*Zeiger auf Anfang der Equations*/
  318.        bool_linenum = linenum;            /*Zeilennummer merken*/
  319.  
  320.        for (pass=0; pass<2; pass++) {        /*2 Durchläufe*/
  321.  
  322.      if (pass) {            /*2. Durchlauf?->ACW erstellen*/
  323.        modus = 0;            /*und Modus bestimmen*/
  324.        for (n=0; n<8; n++) {        /*alle OLMCs untersuchen*/
  325.          if (OLMC[n].PinType == REGOUT) {    /*OLMC als Reg. vorhanden?*/
  326.            modus = MODE3;            /*ja, dann MUß Mode3 vorliegen*/
  327.            Jedec.GALSYN = 0;        /*SYN- und AC0-Bit setzen*/
  328.            Jedec.GALAC0 = 1;
  329.            break;
  330.           }
  331.         }
  332.  
  333.        if (!modus) {
  334.          for (n=0; n<8; n++) {
  335.            if (OLMC[n].PinType == TRIOUT) {    /*OLMC als Tri. vorhanden?*/
  336.              modus = MODE2;            /*ja, dann Mode2*/
  337.              Jedec.GALSYN = 1;        /*SYN- und AC0-Bit setzen*/
  338.              Jedec.GALAC0 = 1;
  339.          break;
  340.             }
  341.           }
  342.         }
  343.  
  344.  
  345.        if (!modus) {            /*wenn Verstöße gegen Mode1,*/
  346.          for (n=0; n<8; n++) {        /*dann automatisch in Mode2*/
  347.            if (OLMC[n].PinType == INPUT) {    /*schalten*/
  348.              if (gal_type == GAL16V8) {
  349.            pin_num = n + 12;
  350.                if ((pin_num == 15) || (pin_num == 16)) {
  351.              modus = MODE2;        /*Mode2*/
  352.              Jedec.GALSYN = 1;        /*SYN- und AC0-Bit setzen*/
  353.              Jedec.GALAC0 = 1;
  354.              break;
  355.                 }
  356.               }
  357.              if (gal_type == GAL20V8) {
  358.            pin_num = n + 15;
  359.                if ((pin_num == 18) || (pin_num == 19)) {
  360.              modus = MODE2;        /*Mode2*/
  361.              Jedec.GALSYN = 1;        /*SYN- und AC0-Bit setzen*/
  362.              Jedec.GALAC0 = 1;
  363.              break;
  364.                 }
  365.               }
  366.             }        /*wenn Ausgang mit Rückkopplung, dann Mode2*/
  367.            if ((OLMC[n].PinType == COM_TRI_OUT) && (OLMC[n].FeedBack)) {
  368.          modus = MODE2;            /*Mode2*/
  369.          Jedec.GALSYN = 1;        /*SYN- und AC0-Bit setzen*/
  370.          Jedec.GALAC0 = 1;
  371.          break;
  372.         }
  373.           }
  374.         }
  375.  
  376.  
  377.        if (!modus) {
  378.          modus = MODE1;            /*noch kein Mode? dann Mode1*/
  379.          Jedec.GALSYN = 1;            /*SYN- und AC0-Bit setzen*/
  380.          Jedec.GALAC0 = 0;
  381.         }
  382.  
  383.  
  384.  
  385.                     /*falls Modus 1 voliegt, alle*/
  386.                     /*Ausgänge deren Typ nicht an-*/
  387.                     /*gegeben wurde in komb. Ausgänge*/
  388.                     /*umwandeln*/
  389.                     /*falls Modus 2 oder 3 vorliegt,*/
  390.                     /*alle Ausgänge deren Typ nicht an-*/
  391.                     /*gegeben wurde in Tristate mit per-*/
  392.        for (n=0; n<8; n++) {    /*manenter Freischaltung umwandeln*/
  393.          if (OLMC[n].PinType == COM_TRI_OUT) {
  394.            if (modus == MODE1) {
  395.          OLMC[n].PinType = COMOUT;
  396.         }
  397.            else {
  398.              OLMC[n].PinType = TRIOUT;
  399.              OLMC[n].TriCon  = TRI_VCC;
  400.         }
  401.           }
  402.         }
  403.     
  404.                     /*ACW erstellen: SYN und AC0 sind*/
  405.                     /*bereits festgelegt*/
  406.        for (n=0; n<PT_SIZE; n++)        /*Produkftermfreigabe auf 1*/
  407.          Jedec.GALPT[n] = 1;
  408.  
  409.                     /*AC1-Bits bestimmen*/
  410.        for (n=0; n<AC1_SIZE; n++) {
  411.          if ((OLMC[n].PinType == INPUT) || (OLMC[n].PinType == TRIOUT))
  412.            Jedec.GALAC1[AC1_SIZE-1-n] = 1;
  413.         }
  414.  
  415.        for (n=0; n<XOR_SIZE; n++) {        /*XOR-Bits bestimmen*/
  416.          if (((OLMC[n].PinType == COMOUT) ||
  417.           (OLMC[n].PinType == TRIOUT) ||
  418.           (OLMC[n].PinType == REGOUT)) &&
  419.           (OLMC[n].Active  == ACTIVE_HIGH))
  420.            Jedec.GALXOR[XOR_SIZE-1-n] = 1;
  421.         }
  422.  
  423.       }
  424.  
  425.  
  426.  
  427.      actptr  = bool_start;
  428.      linenum = bool_linenum;
  429.      newline = linenum;
  430.          goto label1;
  431.  
  432. loop1:     if (GetNextChar()) {                /*Dateiende?*/
  433.        AsmError(2);                    /*ja, dann Fehler*/
  434.        return(-1);
  435.       }
  436.      chr = 0x00;
  437.      if (*actptr == '.') {
  438.        actptr++;
  439.        chr = *actptr;
  440.        if (!((chr == 'T') || (chr == 'E') || (chr == 'R'))) {
  441.          AsmError(13);
  442.          return(-1);
  443.         }         
  444.        actptr++;
  445.        if (GetNextChar()) {                /*Dateiende?*/
  446.          AsmError(2);                /*ja, dann Fehler*/
  447.          return(-1);
  448.         }
  449.       }
  450.  
  451.      actOLMC = (int)actPin.p_Pin;        /*OLMC-Offset merken*/
  452.      if (gal_type == GAL16V8)
  453.        actOLMC -= 12;
  454.      else
  455.        actOLMC -= 15;
  456.      row_offset  = 0;            /*Offset für OR am OLMC*/
  457.      prevOp         = 0;            /*vorherige Verknüpfung*/
  458.                         /*chr enthält T,R,E oder 0x00*/
  459.      if (!pass) {                /*Pass 1?*/
  460.        if ( ((gal_type == GAL16V8)        /*OLMC-Pin?*/
  461.          && (actPin.p_Pin >= 12) && (actPin.p_Pin <= 19)) ||
  462.         ((gal_type == GAL20V8)
  463.          && (actPin.p_Pin >= 15) && (actPin.p_Pin <= 22)) ) {
  464.          if (gal_type == GAL16V8)
  465.            n=actPin.p_Pin-12;
  466.          if (gal_type == GAL20V8)
  467.            n=actPin.p_Pin-15;
  468.          if (chr != 'E') {            /*keine Tristate-Freigabe?*/
  469.                         /*mehrfache Zuweisung?*/
  470.            if ((!OLMC[n].PinType) || (OLMC[n].PinType == INPUT)) {
  471.          if (actPin.p_Neg)
  472.            OLMC[n].Active = ACTIVE_LOW;
  473.          else
  474.            OLMC[n].Active = ACTIVE_HIGH;
  475.  
  476.          switch (chr) {
  477.            case 'T': { OLMC[n].PinType = TRIOUT;
  478.                    break;
  479.                  }
  480.            case 'R': { OLMC[n].PinType = REGOUT;
  481.                    break;
  482.                  }
  483.            case 0x00:{ OLMC[n].PinType = COM_TRI_OUT;
  484.                    break;
  485.                  }
  486.           }
  487.         }
  488.            else {
  489.          AsmError(16);
  490.          return(-1);
  491.         }
  492.           }
  493.          else {
  494.            if (OLMC[n].PinType == TRIOUT) {
  495.          if (actPin.p_Neg)        /*bei Tri.-Kontrolle '/'*/
  496.            actPin.p_Neg = 0;        /*nicht beachten*/
  497.          if (OLMC[n].TriCon) {        /*Tri.-Kontrolle doppelt?*/
  498.            AsmError(22);
  499.            return(-1);
  500.           }
  501.          OLMC[n].TriCon = TRICON;    /*Tri.-Kontrolle angegeben*/
  502.         }
  503.            if ((!OLMC[n].PinType) || (OLMC[n].PinType == INPUT)) {
  504.          AsmError(17);            /*zuerst Tri.Kon.?*/
  505.          return(-1);            /*dann Fehler*/
  506.         }
  507.            if (OLMC[n].PinType == REGOUT) { /*Register? dann Fehler*/
  508.          AsmError(23);
  509.          return(-1);
  510.         }
  511.            if (OLMC[n].PinType == COM_TRI_OUT) { /*kein klares .T?*/
  512.          AsmError(24);
  513.          return(-1);
  514.         }
  515.           }
  516.         }
  517.        else {
  518.          AsmError(15);
  519.          return(-1);
  520.         }
  521.       }
  522.  
  523.  
  524.  
  525.      if (*actptr != '=') {                /* '=' ?*/
  526.        AsmError(14);                /*nein, dann Fehler*/
  527.        return(-1);
  528.       }         
  529. loop2:
  530.      actptr++;
  531.      if (GetNextChar()) {                /*Dateiende?*/
  532.        AsmError(2);                    /*ja, dann Fehler*/
  533.        return(-1);
  534.           }
  535.      IsPinName(pinnames,num_of_pins);
  536.      if (!actPin.p_Pin) {                /*Pinname?*/
  537.        AsmError(11);                /*nein, dann Fehler*/
  538.        return(-1);
  539.       }
  540.      if (actPin.p_Pin == NC_PIN) {            /*NC als Pinname?*/
  541.        AsmError(12);                /*ja, dann Fehler*/
  542.        return(-1);
  543.       }
  544.      if (*(pinnames+(long)((actPin.p_Pin-1)*10)) == '/') {
  545.        actPin.p_Neg = !actPin.p_Neg;    /*Negation bei Pindek.*/
  546.       }                    /*berücksichtigen*/
  547.      oldline = linenum;
  548.      if (GetNextChar()) {                /*Dateiende?*/
  549.        AsmError(2);                    /*ja, dann Fehler*/
  550.        return(-1);
  551.       }
  552.      newline = linenum;
  553.      linenum = oldline;
  554.  
  555.      if (!pass) {                    /*Pass 1?*/
  556.        if ( ((gal_type == GAL16V8)            /*OLMC-Pin?*/
  557.          && (actPin.p_Pin >= 12) && (actPin.p_Pin <= 19)) ||
  558.         ((gal_type == GAL20V8)
  559.          && (actPin.p_Pin >= 15) && (actPin.p_Pin <= 22)) ) {
  560.          if (gal_type == GAL16V8)
  561.            n=actPin.p_Pin-12;
  562.          if (gal_type == GAL20V8)
  563.            n=actPin.p_Pin-15;
  564.          if (!OLMC[n].PinType) {            /*OLMC bereits def.?*/
  565.            OLMC[n].PinType = INPUT;            /*nein, dann Eingang*/
  566.           }
  567.          OLMC[n].FeedBack = YES;    /*wenn hinter dem "=" ein OLMC-Pin*/
  568.         }                /*vorkommt, dann FeedBack benötigt*/
  569.       }
  570.                     /*im 2.Durchlauf Fuse-Matrix*/
  571.                     /*erstellen; chr = T,R,E oder 0x00*/
  572.      if (pass) {
  573.        if (chr != 'E') {
  574.          if (!row_offset) {            /*Zeilenoffset noch 0?*/
  575.            if ( (modus != MODE1) && (OLMC[actOLMC].PinType != REGOUT)) {
  576.              row_offset = 1;
  577.             }
  578.           }
  579.         }
  580.                     /*auf Verstoß gegen Modus prüfen*/
  581.        pin_num = actPin.p_Pin;
  582.        if (modus == MODE2) {        /*Verstoß gegen Mode 2?*/
  583.          if ((gal_type==GAL16V8) && ((pin_num==12) || (pin_num==19))) {
  584.            AsmError(20);
  585.            return(-1);
  586.           }
  587.          if ((gal_type==GAL20V8) && ((pin_num==15) || (pin_num==22))) {
  588.            AsmError(21);
  589.            return(-1);
  590.           }
  591.         }
  592.        if (modus == MODE3) {        /*Verstoß gegen Mode 3?*/
  593.          if ((gal_type==GAL16V8) && ((pin_num==1) || (pin_num==11))) {
  594.            AsmError(26);
  595.            return(-1);
  596.           }
  597.          if ((gal_type==GAL20V8) && ((pin_num==1) || (pin_num==13))) {
  598.            AsmError(27);
  599.            return(-1);
  600.           }
  601.         }
  602.                     /*wenn GND dann Zeile auf 0 setzen*/
  603.        if ((pin_num == num_of_pins) || (pin_num == num_of_pins/2)) {
  604.          if (actPin.p_Neg) {        /*kein /VCC, /GND zulassen*/
  605.            AsmError(25);
  606.            return(-1);
  607.           }
  608.          if (!prevOp && (*actptr != '*' ) && (*actptr != '+')) {
  609.            if (pin_num == num_of_pins/2) {
  610.          if (chr == 'E')         /*bei "x.E=GND|VCC" 1.Zeile*/
  611.            m = ToOLMC[actOLMC] * num_of_col;    /*bei x, x.R, x.T*/
  612.          else                /*row_offset berücksichtigen*/
  613.            m = (ToOLMC[actOLMC] + row_offset) * num_of_col;
  614.          for (n=m; n<m+num_of_col; Jedec.GALLogic[n++]=0);
  615.         }
  616.           }
  617.          else {
  618.            AsmError(28);
  619.            return(-1);
  620.           }
  621.         }
  622.        else {
  623.          if (chr == 'E') {
  624.            if (prevOp == '+') {        /*max. 1 Produktterm*/
  625.              AsmError(29);            /*mehr, dann Fehler*/
  626.              return(-1);
  627.             }                     /*UND-Verknüpfung erstellen*/
  628.            SetAND(ToOLMC[actOLMC], pin_num, actPin.p_Neg);
  629.           }
  630.          else {
  631.            if (prevOp == '+') {        /*ODER-Verknüpfung?*/
  632.              row_offset++;            /*ja, dann nächste Zeile*/
  633.              if (row_offset == MAX_OR) {    /*zuviele Produktterme?*/
  634.            if ((modus != MODE1) && (OLMC[actOLMC].PinType != REGOUT)) {
  635.              AsmError(31);
  636.              return(-1);
  637.             }
  638.            else {
  639.              AsmError(30);
  640.              return(-1);
  641.             }
  642.           }
  643.             }                /*UND-Verknüpfung setzen*/
  644.            SetAND(ToOLMC[actOLMC]+row_offset, pin_num, actPin.p_Neg);
  645.           }
  646.         }
  647.                     /*kommt noch eine Verknüpfung?*/
  648.        if ((*actptr != '+') && (*actptr != '*') && (chr != 'E')) { /*nein*/
  649.          row_offset++;                /*Rest der OLMC- */
  650.          if (row_offset != MAX_OR) {        /*Zeilen auf 0    */
  651.            m = (ToOLMC[actOLMC] + row_offset) * num_of_col;
  652.            for (n=m; n<m+(MAX_OR-row_offset) * num_of_col; n++)
  653.          Jedec.GALLogic[n] = 0;
  654.           }
  655.         }
  656.       }
  657.  
  658.  
  659.      linenum = newline;
  660.  
  661.      if ((*actptr == '+') || (*actptr == '*')) {
  662.        prevOp = *actptr;
  663.        goto loop2;
  664.       }
  665.  
  666.  
  667.      if (strncmp(actptr,(UBYTE *)"DESCRIPTION",11)) {
  668. label1:       linenum = newline;
  669.        IsPinName(pinnames,num_of_pins);
  670.        if (!actPin.p_Pin) {                /*Pinname?*/
  671.          AsmError(11);                /*nein, dann Fehler*/
  672.          return(-1);
  673.         }
  674.        if (actPin.p_Pin==NC_PIN) {            /*NC als Pinname?*/
  675.          AsmError(12);                /*ja, dann Fehler*/
  676.          return(-1);
  677.         }
  678.        if (*(pinnames+(long)((actPin.p_Pin-1)*10)) == '/') {
  679.          actPin.p_Neg = !actPin.p_Neg;    /*Negation bei Pindek.*/
  680.         }                    /*berücksichtigen*/
  681.  
  682.        goto loop1;
  683.       }
  684.  
  685.     }
  686.                     /*Fuse-Matrix der nichtverwendeten*/
  687.                     /*OLMCs und der OLMCs die als Eingang*/
  688.                     /*programmiert sind auf 0 setzen*/
  689.        for (n=0; n<8; n++) {
  690.      if ((OLMC[n].PinType == NOTUSED) || (OLMC[n].PinType == INPUT)) {
  691.        l = ToOLMC[n];
  692.        l = l * num_of_col;
  693.        m = l + 8 * num_of_col;
  694.        for (k=l; k<m; k++)
  695.         Jedec.GALLogic[k] = 0;
  696.       }
  697.     }
  698.                     /*Jedec-Struktur ist fertig (jubel)*/
  699.  
  700.        asmreadyflag = 1;        /*File fertig assembliert*/
  701.  
  702.        FreeMem(fbuff, fsize);        /*Puffer für File .pld*/
  703.  
  704.  
  705.                     /* symbolisches GAL einstellen */
  706.                     /*gal_type: aus dem Source-File*/
  707.                     /*GALType : eingestelltes GAL*/
  708.        if (Config.AutoGAL == YES) {
  709.      if (gal_type == GAL16V8)
  710.        SetGALType (GAL16V8,a_type);
  711.      if (gal_type == GAL20V8)
  712.        SetGALType (GAL20V8,a_type);
  713.      PrintPinNames();
  714.     }
  715.  
  716.                     /* angewählte Files erstellen*/
  717.        if (Config.GenJedec==YES) {
  718.          if (MyFileReq((char *)"JEDEC-Datei schreiben",(char *)".jed",!Config.AutoSave)) {
  719.        oldMaxFuseAdr = MaxFuseAdr;
  720.        if (gal_type == GAL16V8) 
  721.          MaxFuseAdr = MAX_FUSE_ADR16;
  722.        else
  723.          MaxFuseAdr = MAX_FUSE_ADR20;
  724.        WriteJedecFile();
  725.        MaxFuseAdr = oldMaxFuseAdr;
  726.       }
  727.     }
  728.        if (Config.GenFuse==YES)
  729.      WriteFuseFile();
  730.        if (Config.GenChip==YES)
  731.      WriteChipFile();
  732.        if (Config.GenPin==YES)
  733.      WritePinFile();
  734.        PrintText((UBYTE *)"Datei wurde assembliert.",1);
  735.        return(0);                /*kein Fehler aufgetreten*/
  736.       }
  737.      else {
  738.        ErrorReq(3);                /*Lesefehler*/
  739.        FreeMem(fbuff,fsize);
  740.        return(-2);
  741.       }
  742.     }
  743.    else {
  744.      ErrorReq(2);                /*kein Speicher*/
  745.      return(-2);
  746.     }
  747.   }
  748.  else return(-1);            /*Abbruch beim FileRequester*/
  749.  }
  750. else return(-1);            /*Abbruch beim AsmRequester*/
  751. }
  752.  
  753.  
  754.  
  755.  
  756.  
  757. /* setze eine UND-Verknüpfung (=0) in der Fuse-Matrix
  758.    row     :  Zeile in der die UND-Verknüpfung entstehen soll
  759.    pinnum  :  Pin der UND-verknüpft werden soll
  760.    negation:  0: Pin nicht negiert; 1: Pin negiert ('/')
  761. */
  762. void SetAND(row, pinnum, negation)
  763. int    row, pinnum, negation;
  764. {
  765. int column;
  766.  
  767.  if (gal_type == GAL16V8) {
  768.    if (modus == MODE1)
  769.      column = PinToFuse16Mode1[pinnum - 1];
  770.    if (modus == MODE2)
  771.      column = PinToFuse16Mode2[pinnum - 1];
  772.    if (modus == MODE3)
  773.      column = PinToFuse16Mode3[pinnum - 1];
  774.   }
  775.  else {
  776.    if (modus == MODE1)
  777.      column  =  PinToFuse20Mode1[pinnum - 1];
  778.    if (modus == MODE2)
  779.      column  =  PinToFuse20Mode2[pinnum - 1];
  780.    if (modus == MODE3)
  781.      column  =  PinToFuse20Mode3[pinnum - 1];
  782.   }
  783.  Jedec.GALLogic[row * num_of_col + column + negation] = 0;
  784. }
  785.  
  786.  
  787.  
  788.  
  789. /* Überprüfe ob bei "actptr" ein Pinname steht der in "pinnames" eingetragen
  790.    ist und der kein NC (not connected) ist. Die Anzahl der zu untersuchenden
  791.    Pins steht in "numofpins".
  792.    "actptr" wird auf erstes Zeichen hinter den Pinnamen gestellt
  793.    actPin.p_Pin: Pinnummer oder NC_PIN; 0: kein Pinname
  794.    actPin.p_Neg: Pinname mit '/' = 1;  ohne '/' = 0
  795. */
  796. void IsPinName(pinnames, numofpins)
  797. UBYTE    *pinnames;
  798. int    numofpins;
  799. {
  800. int    i, k, n;
  801. UBYTE    *oldactptr;
  802.  
  803.  actPin.p_Neg = 0;        /*Pin-Struktur zurücksetzen*/
  804.  actPin.p_Pin = 0;
  805.  
  806.  if (*actptr == '/' ) {        /*Negation?*/
  807.    actptr++;
  808.    actPin.p_Neg = 1;
  809.   }
  810.  
  811.  n = 0;                /*Länge des möglichen Pinnamens ermitteln*/
  812.  oldactptr = actptr;
  813.  while (isalpha(*actptr) || isdigit(*actptr)) {
  814.    actptr++;
  815.    n++;
  816.   }
  817.  
  818.  if (n)
  819.    if ((n == 2 ) && !strncmp(oldactptr,(UBYTE *)"NC",2))  /*NC ?*/
  820.      actPin.p_Pin = NC_PIN;                /*ja, dann NC-Pin*/
  821.    else
  822.      for (k=0; k<numofpins; k++) {     /*List der Pinnamen durchsuchen*/
  823.        i = 0;
  824.        if (*(pinnames+k*10) == '/')
  825.          i = 1;
  826.        if (n == strlen(pinnames+k*10+i))    /*Stringlängen gleich?*/       
  827.          if (!(strncmp(oldactptr,pinnames+k*10+i,n))) {     /*ja, dann Strings*/
  828.        actPin.p_Pin = k+1;                 /*vergeichen      */
  829.        break;
  830.           }
  831.       }
  832. }
  833.  
  834.  
  835.  
  836.  
  837. /*suche ab "actptr" das nächste Zeichen das kein Space, TAB, LF ist
  838.   Ergebins: 0:Zeichen gefunden, actptr=Zeiger auf dieses Zeichen
  839.         1:kein Zeichen mehr da
  840. */
  841. int GetNextChar()
  842. {
  843.  for(;;) {
  844.    switch (*actptr) {
  845.      case 0x0A: {                /*LineFeed*/
  846.        actptr++;
  847.        linenum++;
  848.        break;
  849.       }
  850.      case  ' ':                    /*Space*/
  851.      case 0x09: {                /*TAB*/
  852.        actptr++;
  853.        break;
  854.       }
  855.      default  : {
  856.        if ((*actptr>' ') && (*actptr<='~')) return(0);    /*Zeichen gefunden?*/
  857.        else actptr++;
  858.       }
  859.     }
  860.    if (actptr>buffend) return(1);            /*Fileende erreicht?*/
  861.   }
  862. }
  863.  
  864.  
  865. /*Zeiger auf zächste Zeile holen
  866.   Ergebins: 0:Zeile gefunden, actptr=Zeiger auf diese Zeile
  867.         1:File-Ende erreicht
  868. */
  869. int GetNextLine()
  870. {
  871.  for(;;) {
  872.    if (*actptr == 0x0A) {
  873.      actptr++;
  874.      linenum++;
  875.      return(0);
  876.     }
  877.    if (actptr>buffend) return(1);            /*Fileende erreicht*/
  878.    actptr++;
  879.   }
  880. }
  881.  
  882.  
  883.  
  884.  
  885. /* gibt Fehlermeldungen des GAL-Assemblers aus
  886.    und gibt Speicher für File-Puffer wieder frei
  887. */
  888. void AsmError(errornum)
  889. int    errornum;
  890. {
  891.  FreeMem(fbuff,fsize);
  892.  MyRequest(ERR_REQ,(UBYTE *)"Fehler in der Eingabe-Datei");
  893.  PrintErrorLine(linenum);
  894.  switch (errornum) {
  895.    case  1:
  896.      PrintText((UBYTE *)"Zeile 1: GAL-Typ erwartet",1);
  897.      break;
  898.    case  2:
  899.      PrintText((UBYTE *)"unerwartetes Ende der Eingabe-Datei",1);
  900.      break;
  901.    case  3:
  902.      PrintText((UBYTE *)"habe nach '/' Pinnamen erwartet",1);
  903.      break;
  904.    case  4:
  905.      PrintText((UBYTE *)"Pinname hat mehr als 8 Buchstaben und Ziffern",1);
  906.      break;
  907.    case  5:
  908.      PrintText((UBYTE *)"unerlaubtes Zeichen bei Pin-Definition",1);
  909.      break;
  910.    case  6:
  911.      PrintText((UBYTE *)"VCC/GND-Zuweisung nur an den entsprechenden Pins erlaubt",1);
  912.      break;
  913.    case  7:
  914.      PrintText((UBYTE *)"für VCC-Pin Zuweisung VCC erwartet",1);
  915.      break;
  916.    case  8:
  917.      PrintText((UBYTE *)"für GND-Pin Zuweisung GND erwartet",1);
  918.      break;
  919.    case  9:
  920.      PrintText((UBYTE *)"gleicher Pinname kommt mehrfach vor",1);
  921.      break;
  922.    case 10:
  923.      PrintText((UBYTE *)"unerlaubte Verwendung von '/'",1);
  924.      break;
  925.    case 11:
  926.      PrintText((UBYTE *)"unbekannter Pin-Name in den Booleschen Gleichungen",1);
  927.      break;
  928.    case 12:
  929.      PrintText((UBYTE *)"NC (Not Connected) in Booleschen Gleichungen nicht erlaubt",1);
  930.      break;
  931.    case 13:
  932.      PrintText((UBYTE *)"nach '.' wird ein 'T', 'E' oder 'R' erwartet",1);
  933.      break;
  934.    case 14:
  935.      PrintText((UBYTE *)"'=' wurde erwartet",1);
  936.      break;
  937.    case 15:
  938.      PrintText((UBYTE *)"Pin kann nicht als Ausgang programmiert werden",1);
  939.      break;
  940.    case 16:
  941.      PrintText((UBYTE *)"gleicher Pin ist mehrfach als Ausgang definiert worden",1);
  942.      break;
  943.    case 17:
  944.      PrintText((UBYTE *)"Tristate-Kontrolle: Tristate-Ausgang noch nicht definiert",1);
  945.      break;
  946.  
  947.    case 20:
  948.      PrintText((UBYTE *)"Betriebsmodus 2: Pins 12,19 können nicht als Eingang verwendet werden",1);
  949.      break;
  950.    case 21:
  951.      PrintText((UBYTE *)"Betriebsmodus 2: Pins 15,22 können nicht als Eingang verwendet werden",1);
  952.      break;
  953.    case 22:
  954.      PrintText((UBYTE *)"Tristate-Kontrolle wurde mehrfach angegeben",1);
  955.      break;
  956.    case 23:
  957.      PrintText((UBYTE *)"Tristate-Kontrolle wurde auf Registerausgang angewendet",1);
  958.      break;
  959.    case 24:
  960.      PrintText((UBYTE *)"Tristate-Kontrolle ohne vorheriges '.T'",1);
  961.      break;
  962.    case 25:
  963.      PrintText((UBYTE *)"statt /VCC, /GND bitte GND, VCC verwenden",1);
  964.      break;
  965.    case 26:
  966.      PrintText((UBYTE *)"Betriebsmodus 3: Pins 1,11 für 'Clock' und '/OE' reserviert",1);
  967.      break;
  968.    case 27:
  969.      PrintText((UBYTE *)"Betriebsmodus 3: Pins 1,13 für 'Clock' und '/OE' reserviert",1);
  970.      break;
  971.    case 28:
  972.      PrintText((UBYTE *)"VCC und GND nicht in Booleschen Ausdrücken erlaubt",1);
  973.      break;
  974.    case 29:
  975.      PrintText((UBYTE *)"bei der Tristate-Kontrolle ist nur EIN Produktterm erlaubt",1);
  976.      break;
  977.    case 30:
  978.      PrintText((UBYTE *)"maximal 8 Produktterme erlaubt",1);
  979.      break;
  980.    case 31:
  981.      PrintText((UBYTE *)"maximal 7 Produktterme erlaubt",1);
  982.      break;
  983.    case 33:
  984.      PrintText((UBYTE *)"keine Booleschen Gleichungen vorhanden",1);
  985.      break;
  986.   }
  987. }
  988.  
  989.  
  990.  
  991.  
  992.  
  993.  
  994. /****************************************************************/
  995. /* ab hier stehen die Routinen, die für die Erstellung der      */
  996. /* Dokumentations-Files zuständig sind                */
  997. /****************************************************************/
  998.  
  999.  
  1000. /* erstelle das Chip-File*/
  1001. void WriteChipFile()
  1002. {
  1003. extern    char    path;
  1004. FILE    *fp;
  1005. int    n;
  1006.  
  1007.    if (MyFileReq((char *)"Chip-Datei schreiben",(char *)".chp",!Config.AutoSave)) {
  1008.      PrintText((UBYTE *)"schreibe Chip-File...",1);
  1009.      if (fp=fopen(&path,(UBYTE *)"w")) {
  1010.        fprintf(fp,"\n\n");
  1011.        WriteSpaces(fp,32);
  1012.        if (gal_type == GAL16V8)    fprintf(fp,"GAL16V8\n\n");
  1013.        else fprintf(fp,"GAL20V8\n\n");
  1014.        WriteSpaces(fp,26);
  1015.        fprintf(fp,"-------\\___/-------\n");
  1016.  
  1017.        for (n=0; n<num_of_pins/2; n++) {
  1018.      WriteSpaces(fp,25-strlen(pinnames+n*10));
  1019.      fprintf(fp,"%s | %2d           %2d | %s\n",pinnames+n*10,n+1,num_of_pins-n,pinnames+(num_of_pins-n-1)*10);
  1020.      if (n<num_of_pins/2-1) {
  1021.        WriteSpaces(fp,26);
  1022.        fprintf(fp,"|                 |\n");
  1023.       }
  1024.     }
  1025.  
  1026.        WriteSpaces(fp,26);
  1027.        fprintf(fp,"-------------------\n");
  1028.  
  1029.        if (fclose(fp)==EOF) {
  1030.      PrintText((UBYTE *)" Fehler!",0);
  1031.          ErrorReq(8);            /*Datei läßt sich nicht schließen*/
  1032.      return;
  1033.     }
  1034.       }
  1035.      else {
  1036.        PrintText((UBYTE *)" Fehler!",0);
  1037.        ErrorReq(13);
  1038.        return;
  1039.       }
  1040.      PrintText((UBYTE *)" o.k.",0);
  1041.     }
  1042. }
  1043.  
  1044.  
  1045.  
  1046.  
  1047.  
  1048. /* erstelle das Pin-File*/
  1049. void WritePinFile()
  1050. {
  1051. extern    char    path;
  1052. FILE    *fp;
  1053. int    k, n, flag;
  1054.  
  1055.    if (MyFileReq((char *)"Pin-Datei schreiben",(char *)".pin",!Config.AutoSave)) {
  1056.      PrintText((UBYTE *)"schreibe Pin-File...",1);
  1057.      if (fp=fopen(&path,(UBYTE *)"w")) {
  1058.        fprintf(fp,"\n\n");
  1059.        fprintf(fp," Pin # | Name     | Pin Type\n");
  1060.        fprintf(fp,"-----------------------------\n");
  1061.  
  1062.        for (n=1; n<=num_of_pins; n++) {
  1063.      fprintf(fp,"  %2d   | ",n);
  1064.      fprintf(fp,"%s",pinnames+(n-1)*10);
  1065.      WriteSpaces(fp,9-strlen(pinnames+(n-1)*10));
  1066.  
  1067.      flag = 0;
  1068.      if (n == num_of_pins/2) {
  1069.        fprintf(fp,"| GND\n");
  1070.        flag = 1;
  1071.       }
  1072.      if (n == num_of_pins) {
  1073.        fprintf(fp,"| VCC\n\n");
  1074.        flag = 1;
  1075.       }
  1076.      if ((modus == MODE3) && (n == 1)) {
  1077.        fprintf(fp,"| Clock\n");
  1078.        flag = 1;
  1079.       }
  1080.      if (modus == MODE3) {
  1081.        if ((gal_type == GAL16V8) && (n == 11)) {
  1082.          fprintf(fp,"| /OE\n");
  1083.          flag = 1;
  1084.         }
  1085.        if ((gal_type == GAL20V8) && (n == 13)) {
  1086.          fprintf(fp,"| /OE\n");
  1087.          flag = 1;
  1088.         }
  1089.       }
  1090.                             /*OLMC-Pin?*/
  1091.      if ( ((gal_type == GAL16V8) && (n >= 12) && (n <= 19)) ||
  1092.           ((gal_type == GAL20V8) && (n >= 15) && (n <= 22)) ) {
  1093.        if (gal_type == GAL16V8)
  1094.          k = n-12;
  1095.        else
  1096.          k = n-15;
  1097.        if (OLMC[k].PinType != INPUT)
  1098.          if (OLMC[k].PinType)
  1099.            fprintf(fp,"| Output\n");
  1100.          else
  1101.            fprintf(fp,"| NC\n");
  1102.        else
  1103.          fprintf(fp,"| Input\n");
  1104.       }
  1105.      else {
  1106.        if (!flag) 
  1107.          fprintf(fp,"| Input\n");
  1108.       }
  1109.     }
  1110.        if (fclose(fp)==EOF) {
  1111.      PrintText((UBYTE *)" Fehler!",0);
  1112.      ErrorReq(8);            /*Datei läßt sich nicht schließen*/
  1113.      return;
  1114.     }
  1115.       }
  1116.      else {
  1117.        PrintText((UBYTE *)" Fehler!",0);
  1118.        ErrorReq(13);
  1119.        return;
  1120.       }
  1121.      PrintText((UBYTE *)" o.k.",0);
  1122.     }
  1123. }
  1124.  
  1125.  
  1126.  
  1127. /* erstelle das Fuse-File*/
  1128. void WriteFuseFile()
  1129. {
  1130. extern    char    path;
  1131. FILE    *fp;
  1132. int    row, col, pin;
  1133. int    xor, ac1;
  1134.  
  1135.    if (MyFileReq((char *)"Fuse-Datei schreiben",(char *)".fus",!Config.AutoSave)) {
  1136.     PrintText((UBYTE *)"schreibe Fuse-File...",1);
  1137.     if (fp=fopen(&path,(UBYTE *)"w")) {
  1138.        if (gal_type == GAL16V8)
  1139.      pin = 19;
  1140.        else
  1141.      pin = 22;
  1142.        for (row=0; row<ROW_SIZE; row++) {
  1143.      if (!((row) % 8)) {
  1144.        fprintf(fp,"\n\nPin %2d = ",pin);
  1145.        fprintf(fp,"%s",pinnames+(pin-1)*10);
  1146.        WriteSpaces(fp,13-strlen(pinnames+(pin-1)*10));
  1147.        if (gal_type == GAL16V8) {
  1148.          xor = Jedec.GALXOR[19-pin];
  1149.          ac1 = Jedec.GALAC1[19-pin];
  1150.         }
  1151.        else {
  1152.          xor = Jedec.GALXOR[22-pin];
  1153.          ac1 = Jedec.GALAC1[22-pin];
  1154.         }
  1155.        fprintf(fp,"XOR = %1d   AC1 = %1d",xor,ac1);
  1156.        pin--;    
  1157.       }
  1158.      fprintf(fp,"\n%2d ",row);
  1159.      for (col=0; col<num_of_col; col++) {
  1160.        if (!( (col) % 4) )
  1161.          fprintf(fp," ");
  1162.        if (Jedec.GALLogic[row * num_of_col + col])
  1163.          fprintf(fp,"-");
  1164.        else
  1165.          fprintf(fp,"x");
  1166.       }
  1167.     }
  1168.        fprintf(fp,"\n\n");
  1169.        if (fclose(fp)==EOF) {
  1170.      PrintText((UBYTE *)" Fehler!",0);
  1171.      ErrorReq(8);            /*Datei läßt sich nicht schließen*/
  1172.      return;
  1173.     }
  1174.       }
  1175.      else {
  1176.        PrintText((UBYTE *)" Fehler!",0);
  1177.        ErrorReq(13);
  1178.        return;
  1179.       }
  1180.      PrintText((UBYTE *)" o.k.",0);
  1181.     }
  1182. }
  1183.  
  1184.  
  1185.  
  1186.  
  1187.  
  1188. /* schreibt Anzahl "numof" Spaces in File "fp"*/
  1189. void WriteSpaces(fp, numof)
  1190. FILE    *fp;
  1191. int    numof;
  1192. {
  1193. int n;
  1194.  
  1195.  for (n=0; n<numof; n++)
  1196.    fprintf(fp," ");
  1197. }
  1198.  
  1199.