Renamed grc to grc65 to avoid name conflicts with another software named grc.
git-svn-id: svn://svn.cc65.org/cc65/trunk@4875 b7a2c559-68d2-44c3-8de9-860c34a00d81
This commit is contained in:
738
src/grc65/grc65.c
Normal file
738
src/grc65/grc65.c
Normal file
@@ -0,0 +1,738 @@
|
||||
|
||||
/*
|
||||
GEOS resource compiler
|
||||
|
||||
by Maciej 'YTM/Elysium' Witkowiak
|
||||
|
||||
see GEOSLib documentation for license info
|
||||
|
||||
*/
|
||||
|
||||
/*
|
||||
- make it work, then do it better
|
||||
- more or less comments? it was hard to code, should be even harder to
|
||||
understand =D
|
||||
- add loadable icons feature (binary - 63 bytes)
|
||||
*/
|
||||
|
||||
/* - err, maybe free allocated memory, huh? (who cares, it's just a little prog...)
|
||||
*/
|
||||
|
||||
#include <stdio.h>
|
||||
#include <stdlib.h>
|
||||
#include <stdarg.h>
|
||||
#include <string.h>
|
||||
#include <errno.h>
|
||||
#include <time.h>
|
||||
|
||||
#include "grc65.h"
|
||||
|
||||
/* common stuff */
|
||||
#include "fname.h"
|
||||
#include "abend.h"
|
||||
#include "chartype.h"
|
||||
#include "xmalloc.h"
|
||||
|
||||
void VLIRLinker(int argc, char *argv[]) {
|
||||
FILE *outCVT, *input;
|
||||
unsigned char *buffer;
|
||||
unsigned char vlirtabt[127];
|
||||
unsigned char vlirtabs[127];
|
||||
int i,j,lastarg;
|
||||
unsigned l;
|
||||
int bytes;
|
||||
int blocks,rest;
|
||||
|
||||
i=2;
|
||||
|
||||
/* check if we know enough */
|
||||
|
||||
if (argc<4)
|
||||
AbEnd("too few arguments, required [out] [cvthead] [vlir0] ...\n");
|
||||
|
||||
/* first open and copy CVT header */
|
||||
|
||||
outCVT = fopen(argv[i],"wb+");
|
||||
if (outCVT==NULL)
|
||||
AbEnd("can't open output:%s\n",strerror(errno));
|
||||
|
||||
++i;
|
||||
input = fopen(argv[i],"rb");
|
||||
if (input==NULL)
|
||||
AbEnd("can't open input:%s\n",strerror(errno));
|
||||
|
||||
buffer = xmalloc(THIS_BUFFER_IS_SOOO_HUGE);
|
||||
memset(buffer,0,THIS_BUFFER_IS_SOOO_HUGE);
|
||||
|
||||
bytes = fread(buffer,1,1024,input);
|
||||
fclose(input);
|
||||
if (bytes!=508)
|
||||
AbEnd("%s is not a cvt header\n",argv[i]);
|
||||
|
||||
fwrite(buffer,1,bytes,outCVT);
|
||||
|
||||
/* now put 254 bytes of VLIR table, to update later */
|
||||
|
||||
/* clear out things */
|
||||
memset(buffer,0,512);
|
||||
fwrite(buffer,1,254,outCVT);
|
||||
for (l=0;l<sizeof(vlirtabt)/sizeof(vlirtabt[0]);l++) {
|
||||
vlirtabt[l]=0;
|
||||
vlirtabs[l]=0;
|
||||
}
|
||||
|
||||
/* scan arguments for the last one that is not blank or nonexistant */
|
||||
j=argc;
|
||||
lastarg=argc-1;
|
||||
while (j!=3) {
|
||||
--j;
|
||||
if ((strcmp(argv[j],"blank")==0)||(strcmp(argv[j],"noexist")==0))
|
||||
--lastarg;
|
||||
}
|
||||
|
||||
/* now read all VLIR chains, align to 254 bytes */
|
||||
|
||||
++i;
|
||||
j=0;
|
||||
while (i!=argc) {
|
||||
if (strcmp(argv[i],"blank")==0) {
|
||||
vlirtabt[j]=0; vlirtabs[j]=0; }
|
||||
else if (strcmp(argv[i],"noexist")==0) {
|
||||
vlirtabt[j]=0; vlirtabs[j]=0xff; }
|
||||
else {
|
||||
memset(buffer,0,bytes+512);
|
||||
input = fopen(argv[i],"rb");
|
||||
if (input==NULL)
|
||||
AbEnd("couldn't open %s:%s\n",argv[i],strerror(errno));
|
||||
bytes = fread(buffer,1,THIS_BUFFER_IS_SOOO_HUGE,input);
|
||||
fclose(input);
|
||||
if (bytes==0)
|
||||
AbEnd("couldn't read %s:%s\n",argv[i],strerror(errno));
|
||||
blocks = bytes / 254;
|
||||
rest = bytes % 254;
|
||||
if (rest==0) {
|
||||
rest = 255;
|
||||
--blocks;
|
||||
} else {
|
||||
++rest;
|
||||
}
|
||||
vlirtabt[j]=blocks+1; vlirtabs[j]=rest;
|
||||
/* do not pad the last chain - it doesn't change or break anything
|
||||
but filesize in bytes of .cvt will be identical to native cvt */
|
||||
if (i==lastarg)
|
||||
fwrite(buffer,1,bytes,outCVT);
|
||||
else
|
||||
fwrite(buffer,1,(blocks+1)*254,outCVT);
|
||||
}
|
||||
++j;
|
||||
++i;
|
||||
}
|
||||
|
||||
free(buffer);
|
||||
|
||||
/* now rewind and update VLIR table */
|
||||
|
||||
fflush(outCVT);
|
||||
fseek(outCVT,508,SEEK_SET);
|
||||
for (l=0;l<sizeof(vlirtabt)/sizeof(vlirtabt[0]);l++) {
|
||||
fputc(vlirtabt[l],outCVT);
|
||||
fputc(vlirtabs[l],outCVT);
|
||||
}
|
||||
fclose(outCVT);
|
||||
exit(EXIT_SUCCESS);
|
||||
}
|
||||
|
||||
void printCHeader (void) {
|
||||
|
||||
fprintf(outputCFile, "\n/*\n\tThis file was generated by GEOS Resource Compiler\n"
|
||||
"\n\tDO NOT EDIT! Any changes will be lost!\n"
|
||||
"\n\tEdit proper resource file instead\n"
|
||||
"\n*/\n\n");
|
||||
}
|
||||
|
||||
void printSHeader (void) {
|
||||
|
||||
fprintf(outputSFile, "\n;\n;\tThis file was generated by GEOS Resource Compiler\n;"
|
||||
"\n;\tDO NOT EDIT! Any changes will be lost!\n;"
|
||||
"\n;\tEdit proper resource file instead\n;"
|
||||
"\n;\n\n");
|
||||
}
|
||||
|
||||
void printVHeader (void) {
|
||||
fprintf(outputVFile, "\n#\n#\tThis file was generated by GEOS Resource Compiler\n#"
|
||||
"\n#\tDO NOT EDIT! Any changes will be lost!\n#"
|
||||
"\n#\tEdit proper resource file instead\n#"
|
||||
"\n#\tLook at end of this file to find commandline that must be used\n"
|
||||
"#\tto invoke ld65 and grc65 (as VLIR linker)\n#"
|
||||
"\n#\n\n");
|
||||
}
|
||||
|
||||
void openCFile (void) {
|
||||
if ((CFnum==0) && (forceFlag==0)) {
|
||||
/* test if file exists already and no forcing*/
|
||||
if ((outputCFile = fopen (outputCName,"r"))!=0)
|
||||
AbEnd("file %s already exists, aborting\n", outputCName);
|
||||
}
|
||||
if ((outputCFile = fopen (outputCName,outputCMode))==0)
|
||||
AbEnd("can't open file %s for writing: %s\n",outputCName,strerror (errno));
|
||||
if (CFnum==0) {
|
||||
outputCMode[0]='a';
|
||||
printCHeader();
|
||||
CFnum++;
|
||||
}
|
||||
}
|
||||
|
||||
void openSFile (void) {
|
||||
if ((SFnum==0) && (forceFlag==0)) {
|
||||
/* test if file exists already and no forcing*/
|
||||
if ((outputSFile = fopen (outputSName,"r"))!=0)
|
||||
AbEnd("file %s already exists, aborting\n", outputSName);
|
||||
}
|
||||
if ((outputSFile = fopen (outputSName,outputSMode))==0)
|
||||
AbEnd("can't open file %s for writing: %s\n",outputSName,strerror (errno));
|
||||
if (SFnum==0) {
|
||||
outputSMode[0]='a';
|
||||
printSHeader();
|
||||
SFnum++;
|
||||
}
|
||||
}
|
||||
|
||||
void openVFile (void) {
|
||||
if ((VFnum==0) && (forceFlag==0)) {
|
||||
/* test if file exists already and no forcing*/
|
||||
if ((outputVFile = fopen (outputVName,"r"))!=0)
|
||||
AbEnd("file %s already exists, aborting\n", outputVName);
|
||||
}
|
||||
if ((outputVFile = fopen (outputVName,outputVMode))==0)
|
||||
AbEnd("can't open file %s for writting: %s\n",outputVName,strerror (errno));
|
||||
if (VFnum==0) {
|
||||
outputVMode[0]='a';
|
||||
printVHeader();
|
||||
VFnum++;
|
||||
}
|
||||
}
|
||||
|
||||
void printUsage (void) {
|
||||
printf("Usage: %s [options] file\n"
|
||||
"Options:\n"
|
||||
"\t-h, -?\t\tthis help\n"
|
||||
"\t-f\t\tforce writting files\n"
|
||||
"\t-o name\t\tname C output file\n"
|
||||
"\t-s name\t\tname asm output file\n"
|
||||
"\t-l name\t\tname ld65 config output file (for vlir)\n"
|
||||
"Or as VLIR linker: %s -vlir output.cvt header [vlir0] ... [blank] ... [vlir_n]\n",
|
||||
ProgName,ProgName);
|
||||
}
|
||||
|
||||
int findToken (const char **tokenTbl, const char *token) {
|
||||
/* takes as input table of tokens and token, returns position in table or -1 if not found */
|
||||
int a=0;
|
||||
|
||||
while (strlen(tokenTbl[a])!=0) {
|
||||
if (strcmp(tokenTbl[a],token)==0) break;
|
||||
a++;
|
||||
}
|
||||
if (strlen(tokenTbl[a])==0) a=-1;
|
||||
return a;
|
||||
}
|
||||
|
||||
char *nextPhrase() {
|
||||
return strtok(NULL, "\"");
|
||||
}
|
||||
|
||||
char *nextWord() {
|
||||
return strtok(NULL, " ");
|
||||
}
|
||||
|
||||
void setLen (char *name, unsigned len) {
|
||||
if (strlen(name)>len)
|
||||
name[len]='\0';
|
||||
}
|
||||
|
||||
void fillOut (char *name, int len, char *filler) {
|
||||
int a;
|
||||
setLen (name, len);
|
||||
fprintf(outputSFile, ".byte \"%s\"\n\t\t", name);
|
||||
a = len - strlen(name);
|
||||
if (a!=0) {
|
||||
fprintf(outputSFile, ".byte %s", filler);
|
||||
while (--a!=0) fprintf(outputSFile, ", %s", filler);
|
||||
fprintf(outputSFile, "\n\t\t");
|
||||
}
|
||||
}
|
||||
|
||||
char *bintos(unsigned char a, char out[7]) {
|
||||
int i=0;
|
||||
for (;i<8;i++) {
|
||||
out[7-i] = ((a & 1)==0) ? '0' : '1';
|
||||
a = a >> 1; };
|
||||
out[i]='\0';
|
||||
return out;
|
||||
}
|
||||
|
||||
int getNameSize (const char *word) {
|
||||
/* count length of a word using BSW 9 font table */
|
||||
int a=0, i=0;
|
||||
|
||||
while (word[i]!='\0') {
|
||||
a+=(BSWTab[word[i]-31] - BSWTab[word[i]-32]); i++; }
|
||||
|
||||
return a;
|
||||
}
|
||||
|
||||
void DoMenu (void) {
|
||||
|
||||
int a, size, tmpsize, item=0;
|
||||
char *token;
|
||||
char namebuff[255]="";
|
||||
struct menu myMenu;
|
||||
struct menuitem *curItem, *newItem;
|
||||
|
||||
openCFile();
|
||||
|
||||
myMenu.name=nextWord();
|
||||
myMenu.left=atoi(nextWord());
|
||||
myMenu.top=atoi(nextWord());
|
||||
myMenu.type=nextWord();
|
||||
|
||||
if (strcmp(nextWord(),"{")!=0) {
|
||||
AbEnd ("menu '%s' description has no opening bracket!\n", myMenu.name);
|
||||
}
|
||||
curItem = xmalloc(sizeof(struct menuitem));
|
||||
myMenu.item=curItem;
|
||||
do {
|
||||
token = nextWord();
|
||||
if (strcmp(token,"}")==0) break;
|
||||
if (token[strlen(token)-1]!='"') {
|
||||
strcpy (namebuff, token);
|
||||
do {
|
||||
token = nextWord();
|
||||
strcat (namebuff, " ");
|
||||
strcat (namebuff, token);
|
||||
} while (token[strlen(token)-1]!='"');
|
||||
token = xmalloc(strlen(namebuff));
|
||||
strcpy (token, namebuff);
|
||||
}
|
||||
curItem->name=token;
|
||||
curItem->type=nextWord();
|
||||
curItem->target=nextWord();
|
||||
newItem=xmalloc(sizeof(struct menuitem));
|
||||
curItem->next=newItem;
|
||||
curItem=newItem;
|
||||
item++;
|
||||
} while (strcmp(token,"}")!=0);
|
||||
if (item==0) AbEnd ("menu '%s' has 0 items!\n", myMenu.name);
|
||||
if (item>31) AbEnd ("menu '%s' has too many items!\n", myMenu.name);
|
||||
|
||||
curItem->next=NULL;
|
||||
|
||||
/* Count menu sizes */
|
||||
|
||||
size=0;
|
||||
curItem=myMenu.item;
|
||||
if (strstr(myMenu.type,"HORIZONTAL")!=NULL) {
|
||||
/* menu is HORIZONTAL, ysize=15, sum xsize of all items +~8?*/
|
||||
myMenu.bot=myMenu.top+15;
|
||||
for (a=0;a!=item;a++) {
|
||||
size+=getNameSize(curItem->name);
|
||||
curItem=curItem->next;
|
||||
};
|
||||
} else {
|
||||
/* menu is VERTICAL, ysize=item*15, count largest xsize of all items +~8? */
|
||||
myMenu.bot=myMenu.top+(14*item);
|
||||
for (a=0;a!=item;a++) {
|
||||
tmpsize=getNameSize(curItem->name);
|
||||
size = (size > tmpsize) ? size : tmpsize;
|
||||
curItem=curItem->next;
|
||||
};
|
||||
};
|
||||
myMenu.right=myMenu.left+size-1;
|
||||
|
||||
curItem=myMenu.item;
|
||||
for (a=0;a!=item;a++) {
|
||||
/* print prototype only if MENU_ACTION or DYN_SUB_MENU are present in type */
|
||||
if ((strstr(curItem->type, "MENU_ACTION")!=NULL) || (strstr(curItem->type, "DYN_SUB_MENU")!=NULL))
|
||||
fprintf(outputCFile, "void %s (void);\n", curItem->target);
|
||||
curItem=curItem->next;
|
||||
}
|
||||
|
||||
fprintf(outputCFile, "\nconst void %s = {\n\t(char)%i, (char)%i,\n\t(int)%i, (int)%i,\n\t"
|
||||
"(char)(%i | %s),\n", myMenu.name, myMenu.top, myMenu.bot, myMenu.left,
|
||||
myMenu.right, item, myMenu.type);
|
||||
|
||||
curItem=myMenu.item;
|
||||
for (a=0;a!=item;a++) {
|
||||
fprintf(outputCFile, "\t%s, (char)%s, (int)", curItem->name, curItem->type);
|
||||
if ((strstr(curItem->type, "SUB_MENU")!=NULL) && (strstr(curItem->type, "DYN_SUB_MENU")==NULL))
|
||||
fprintf(outputCFile, "&");
|
||||
fprintf(outputCFile, "%s,\n", curItem->target);
|
||||
curItem=curItem->next;
|
||||
}
|
||||
|
||||
fprintf(outputCFile, "\t};\n\n");
|
||||
|
||||
if (fclose (outputCFile)!=0)
|
||||
AbEnd("error closing %s: %s\n",outputCName,strerror (errno));
|
||||
}
|
||||
|
||||
void DoHeader (void) {
|
||||
|
||||
time_t t;
|
||||
struct tm *my_tm;
|
||||
|
||||
struct appheader myHead;
|
||||
char *token;
|
||||
char i1[9], i2[9], i3[9];
|
||||
int a, b;
|
||||
|
||||
openSFile();
|
||||
|
||||
token = nextWord();
|
||||
|
||||
a = findToken (hdrFTypes, token);
|
||||
|
||||
switch (a) {
|
||||
case 0: myHead.geostype = 6; break;
|
||||
case 1: myHead.geostype = 14; break;
|
||||
default: AbEnd("filetype '%s' is not supported yet\n", token);
|
||||
}
|
||||
|
||||
myHead.dosname = nextPhrase();
|
||||
nextPhrase();
|
||||
myHead.classname = nextPhrase();
|
||||
nextPhrase();
|
||||
myHead.version = nextPhrase();
|
||||
|
||||
/* put default values into myHead here */
|
||||
myHead.author = "cc65";
|
||||
myHead.info = "Program compiled with cc65 and GEOSLib.";
|
||||
myHead.dostype = 128+3;
|
||||
myHead.structure = 0;
|
||||
myHead.mode = 0;
|
||||
|
||||
t = time(NULL);
|
||||
my_tm = localtime (&t);
|
||||
|
||||
myHead.year = my_tm->tm_year;
|
||||
myHead.month = my_tm->tm_mon+1;
|
||||
myHead.day = my_tm->tm_mday;
|
||||
myHead.hour = my_tm->tm_hour;
|
||||
myHead.min = my_tm->tm_min;
|
||||
|
||||
if (strcmp(nextWord(),"{")!=0)
|
||||
AbEnd ("header '%s' has no opening bracket!\n", myHead.dosname);
|
||||
|
||||
do {
|
||||
token=nextWord();
|
||||
if (strcmp(token, "}")==0) break;
|
||||
switch (a = findToken (hdrFields, token)) {
|
||||
case -1:
|
||||
AbEnd ("unknown field '%s' in header '%s'\n", token, myHead.dosname);
|
||||
break;
|
||||
case 0: /* author */
|
||||
myHead.author = nextPhrase(); break;
|
||||
case 1: /* info */
|
||||
myHead.info = nextPhrase(); break;
|
||||
case 2: /* date */
|
||||
myHead.year = atoi(nextWord());
|
||||
myHead.month = atoi(nextWord());
|
||||
myHead.day = atoi(nextWord());
|
||||
myHead.hour = atoi(nextWord());
|
||||
myHead.min = atoi(nextWord());
|
||||
break;
|
||||
case 3: /* dostype */
|
||||
switch (b = findToken (hdrDOSTp, nextWord())) {
|
||||
case -1:
|
||||
AbEnd ("unknown dostype in header '%s'\n", myHead.dosname);
|
||||
break;
|
||||
default:
|
||||
myHead.dostype = b/2 + 128 + 1;
|
||||
break;
|
||||
}
|
||||
break;
|
||||
case 4: /* mode */
|
||||
switch (b = findToken (hdrModes, nextWord())) {
|
||||
case -1:
|
||||
AbEnd ("unknown mode in header '%s'\n", myHead.dosname);
|
||||
case 0:
|
||||
myHead.mode = 0x40; break;
|
||||
case 1:
|
||||
myHead.mode = 0x00; break;
|
||||
case 2:
|
||||
myHead.mode = 0xc0; break;
|
||||
case 3:
|
||||
myHead.mode = 0x80; break;
|
||||
}
|
||||
break;
|
||||
case 5: /* structure */
|
||||
switch (b = findToken(hdrStructTp, nextWord())) {
|
||||
case -1:
|
||||
AbEnd ("unknown structure type in header '%s'\n", myHead.dosname);
|
||||
case 0:
|
||||
case 1:
|
||||
myHead.structure = 0; break;
|
||||
case 2:
|
||||
case 3:
|
||||
myHead.structure = 1; break;
|
||||
}
|
||||
break;
|
||||
}
|
||||
|
||||
} while (strcmp(token, "}")!=0);
|
||||
|
||||
/* OK, all information is gathered, do flushout */
|
||||
|
||||
fprintf(outputSFile,
|
||||
"\t\t\t.segment \"HEADER\"\n\n\t\t.byte %i\n\t\t.word 0\n\t\t", myHead.dostype);
|
||||
|
||||
fillOut(myHead.dosname,16,"$a0");
|
||||
|
||||
fprintf(outputSFile,
|
||||
".word 0\n\t\t.byte %i\n\t\t.byte %i\n\t\t.byte %i, %i, %i, %i, %i\n\n\t\t"
|
||||
".word 0\n\t\t.byte \"PRG formatted GEOS file V1.0\"\n\n\t\t.res $c4\n\n\t\t"
|
||||
".byte 3, 21, 63 | $80\n\t\t",
|
||||
myHead.structure, myHead.geostype, myHead.year, myHead.month, myHead.day,
|
||||
myHead.hour, myHead.min);
|
||||
|
||||
for (a=0;a!=63;a=a+3) {
|
||||
fprintf(outputSFile,
|
||||
".byte %%%s, %%%s, %%%s\n\t\t",
|
||||
bintos(icon1[a], i1), bintos(icon1[a+1], i2), bintos(icon1[a+2], i3)); };
|
||||
|
||||
fprintf(outputSFile,
|
||||
"\n\t\t.byte %i, %i, %i\n\t\t.word $0400, $0400-1, $0400\n\n\t\t",
|
||||
myHead.dostype, myHead.geostype, myHead.structure);
|
||||
|
||||
fillOut(myHead.classname,12,"$20");
|
||||
|
||||
fillOut(myHead.version,4,"0");
|
||||
|
||||
fprintf(outputSFile,
|
||||
".byte 0, 0, 0\n\t\t.byte %i\n\n\t\t", myHead.mode);
|
||||
|
||||
setLen(myHead.author,62);
|
||||
fprintf(outputSFile,
|
||||
".byte \"%s\"\n\t\t.byte 0\n\t\t.res (63-%i)\n\n\t\t",
|
||||
myHead.author, (int) (strlen(myHead.author)+1));
|
||||
|
||||
setLen(myHead.info, 95);
|
||||
fprintf(outputSFile,
|
||||
".byte \"%s\"\n\t\t.byte 0\n\t\t.res (96-%i)\n\n",
|
||||
myHead.info, (int) (strlen(myHead.info)+1));
|
||||
|
||||
if (fclose (outputSFile)!=0)
|
||||
AbEnd("error closing %s: %s\n",outputSName,strerror (errno));
|
||||
|
||||
}
|
||||
|
||||
void DoVLIR (void) {
|
||||
|
||||
char *token;
|
||||
char *headname;
|
||||
int i,numchains,vlirbase;
|
||||
struct vlirentry {
|
||||
char *chainname;
|
||||
int exist;
|
||||
};
|
||||
|
||||
struct vlirentry vlirtable[127];
|
||||
|
||||
openVFile();
|
||||
|
||||
headname = nextWord();
|
||||
|
||||
vlirbase = strtol(nextWord(),NULL,0);
|
||||
|
||||
if (strcmp(nextWord(),"{")!=0)
|
||||
AbEnd ("VLIR description has no opening bracket!\n");
|
||||
|
||||
numchains=0;
|
||||
|
||||
do {
|
||||
token=nextWord();
|
||||
if (strcmp(token, "}")==0) break;
|
||||
numchains++;
|
||||
if (numchains>127) {
|
||||
AbEnd("Too many VLIR chains!\n");
|
||||
}
|
||||
vlirtable[numchains].chainname=token;
|
||||
|
||||
/* for first chain - name header */
|
||||
if (numchains==1) {
|
||||
fprintf(outputVFile,"MEMORY {\n\tHEADER: start = $204, size = 508, file = \"%s\";\n"
|
||||
"\tVLIR0: start = $0400, size = $5C00, file = \"%s\";\n",headname,token);
|
||||
} else {
|
||||
/* for all other - segment */
|
||||
/* ignore non-existing segments */
|
||||
vlirtable[numchains].exist=1;
|
||||
if ( (strcmp(token,"blank")==0) || (strcmp(token,"noexist")==0) ) {
|
||||
vlirtable[numchains].exist=0;
|
||||
fprintf(outputVFile,"#");
|
||||
}
|
||||
fprintf(outputVFile,"\tVLIR%i: start = $%x, size = $%x, file = \"%s\";\n",
|
||||
numchains-1,vlirbase,0x5c00-vlirbase,token);
|
||||
}
|
||||
|
||||
} while (strcmp(token, "}")!=0);
|
||||
fprintf(outputVFile,"}\n\n");
|
||||
|
||||
if (numchains==0) {
|
||||
AbEnd("There must be at least one VLIR chain.\n");
|
||||
};
|
||||
|
||||
/* now put segments info */
|
||||
fprintf(outputVFile,"SEGMENTS {\n\tHEADER: load = HEADER, type = ro;\n"
|
||||
"\tCODE: load = VLIR0, type = ro;\n"
|
||||
"\tRODATA: load = VLIR0, type = ro;\n"
|
||||
"\tDATA: load = VLIR0, type = rw;\n"
|
||||
"\tBSS: load = VLIR0, type = bss, define = yes;\n\n");
|
||||
|
||||
for (i=2;i<=numchains;i++) {
|
||||
if (vlirtable[i].exist==0) {
|
||||
fprintf(outputVFile,"#");
|
||||
}
|
||||
fprintf(outputVFile,"\tVLIR%i: load = VLIR%i, type = rw, define = yes;\n",i-1,i-1);
|
||||
}
|
||||
fprintf(outputVFile,"}\n");
|
||||
|
||||
/* now put usage info */
|
||||
fprintf(outputVFile,"\n# ld65 -o output.cvt -C %s file1.o file2.o ...",outputVName);
|
||||
fprintf(outputVFile,"\n# grc65 -vlir outputname %s",headname);
|
||||
for (i=1;i<=numchains;i++) {
|
||||
fprintf(outputVFile," %s",vlirtable[i].chainname);
|
||||
}
|
||||
fprintf(outputVFile,"\n");
|
||||
|
||||
if (fclose (outputVFile)!=0)
|
||||
AbEnd("error closing %s: %s\n",outputVName,strerror (errno));
|
||||
}
|
||||
|
||||
char *filterInput (FILE *F, char *tbl) {
|
||||
/* loads file into buffer filtering it out */
|
||||
int a, prevchar=-1, i=0, bracket=0, quote=1;
|
||||
|
||||
while (1) {
|
||||
a = getc(F);
|
||||
if ((a=='\n')||(a=='\015')) a = ' ';
|
||||
if (a==',' && quote) a = ' ';
|
||||
if (a=='\042') quote=!quote;
|
||||
if (quote) {
|
||||
if ((a=='{')||(a=='(')) bracket++;
|
||||
if ((a=='}')||(a==')')) bracket--;
|
||||
}
|
||||
if (a==EOF) { tbl[i]='\0'; xrealloc(tbl, i+1); break; };
|
||||
if (IsSpace(a)) {
|
||||
if ((prevchar!=' ') && (prevchar!=-1)) { tbl[i++]=' '; prevchar=' '; }
|
||||
} else {
|
||||
if (a==';' && quote) { do { a = getc (F); } while (a!='\n'); fseek(F,-1,SEEK_CUR); }
|
||||
else {
|
||||
tbl[i++]=a; prevchar=a; }
|
||||
}
|
||||
}
|
||||
|
||||
if (bracket!=0) AbEnd("there are unclosed brackets!\n");
|
||||
|
||||
return tbl;
|
||||
}
|
||||
|
||||
void processFile (const char *filename) {
|
||||
|
||||
FILE *F;
|
||||
|
||||
char *str;
|
||||
char *token;
|
||||
|
||||
int head=0; /* number of processed HEADER sections */
|
||||
int vlir=0; /* number of processed VLIR sections */
|
||||
|
||||
if ((F = fopen (filename,"r"))==0)
|
||||
AbEnd("can't open file %s for reading: %s\n",filename,strerror (errno));
|
||||
|
||||
str=filterInput(F, xmalloc(BLOODY_BIG_BUFFER));
|
||||
|
||||
token = strtok (str," ");
|
||||
|
||||
do {
|
||||
if (str!=NULL) {
|
||||
switch (findToken (mainToken, token)) {
|
||||
|
||||
case 0: DoMenu(); break;
|
||||
case 1:
|
||||
if (++head!=1) {
|
||||
AbEnd ("more than one HEADER section, aborting.\n");
|
||||
} else {
|
||||
DoHeader();
|
||||
}
|
||||
break;
|
||||
case 2: break; /* icon not implemented yet */
|
||||
case 3: break; /* dialog not implemented yet */
|
||||
case 4:
|
||||
if (++vlir!=1) {
|
||||
AbEnd ("more than one VLIR section, aborting.\n");
|
||||
} else {
|
||||
DoVLIR();
|
||||
}
|
||||
break;
|
||||
default: AbEnd ("unknown section %s.\n",token); break;
|
||||
}
|
||||
}
|
||||
token = nextWord();
|
||||
} while (token!=NULL);
|
||||
}
|
||||
|
||||
int main(int argc, char *argv[]) {
|
||||
|
||||
int ffile=0, i=1;
|
||||
|
||||
ProgName = argv[0];
|
||||
while (i < argc) {
|
||||
const char *arg = argv[i];
|
||||
if (arg[0] == '-') {
|
||||
switch (arg[1]) {
|
||||
case 'f':
|
||||
forceFlag=1;
|
||||
break;
|
||||
case 'o':
|
||||
outputCName=argv[++i];
|
||||
break;
|
||||
case 's':
|
||||
outputSName=argv[++i];
|
||||
break;
|
||||
case 'l':
|
||||
outputVName=argv[++i];
|
||||
break;
|
||||
case 'h':
|
||||
case '?':
|
||||
printUsage();
|
||||
exit (EXIT_SUCCESS);
|
||||
break;
|
||||
case 'v':
|
||||
if (strcmp(arg,"-vlir")==0) {
|
||||
VLIRLinker(argc,argv);
|
||||
exit (EXIT_SUCCESS);
|
||||
break;
|
||||
} else {
|
||||
AbEnd("unknown option %s\n",arg);
|
||||
break;
|
||||
}
|
||||
default: AbEnd("unknown option %s\n",arg);
|
||||
}
|
||||
} else {
|
||||
ffile++;
|
||||
|
||||
if (outputCName==NULL)
|
||||
outputCName = MakeFilename(arg,".h");
|
||||
if (outputSName==NULL)
|
||||
outputSName = MakeFilename(arg,".s");
|
||||
if (outputVName==NULL)
|
||||
outputVName = MakeFilename(arg,".cfg");
|
||||
|
||||
processFile(arg);
|
||||
|
||||
}
|
||||
i++;
|
||||
}
|
||||
if (ffile==0) AbEnd("no input file\n");
|
||||
|
||||
return EXIT_SUCCESS;
|
||||
}
|
||||
77
src/grc65/grc65.h
Normal file
77
src/grc65/grc65.h
Normal file
@@ -0,0 +1,77 @@
|
||||
|
||||
/* I hope that no one will be able to create a .grc bigger than this... */
|
||||
#define BLOODY_BIG_BUFFER 65000
|
||||
/* there are no 6MB GEOS binaries... I hope! */
|
||||
#define THIS_BUFFER_IS_SOOO_HUGE 6000000
|
||||
|
||||
struct menuitem {
|
||||
char *name;
|
||||
char *type;
|
||||
char *target;
|
||||
struct menuitem *next; };
|
||||
|
||||
struct menu {
|
||||
char *name;
|
||||
int top, left;
|
||||
int bot, right;
|
||||
char *type;
|
||||
struct menuitem *item; };
|
||||
|
||||
struct appheader {
|
||||
int year,month,day,hour,min;
|
||||
int mode;
|
||||
int dostype;
|
||||
int geostype;
|
||||
int structure;
|
||||
char *dosname;
|
||||
char *classname;
|
||||
char *version;
|
||||
char *author;
|
||||
char *info; };
|
||||
|
||||
const char *mainToken[] = {
|
||||
"MENU", "HEADER", "ICON", "DIALOG", "VLIR", "" };
|
||||
|
||||
const char *hdrFTypes[] = {
|
||||
"APPLICATION", "AUTO_EXEC", "DESK_ACC", "ASSEMBLY", "DISK_DEVICE", "PRINTER", "SYSTEM", "" };
|
||||
|
||||
const char *hdrFields[] = {
|
||||
"author", "info", "date", "dostype", "mode", "structure", "" };
|
||||
|
||||
const char *hdrDOSTp[] = {
|
||||
"seq", "SEQ", "prg", "PRG", "usr", "USR", "" };
|
||||
|
||||
const char *hdrStructTp[] = {
|
||||
"seq", "SEQ", "vlir", "VLIR", "" };
|
||||
|
||||
const char *hdrModes[] = {
|
||||
"any", "40only", "80only", "c64only", "" };
|
||||
|
||||
const int BSWTab[] = { 0,
|
||||
0x005, 0x007, 0x00b, 0x011, 0x017,
|
||||
0x01d, 0x023, 0x025, 0x029, 0x02d, 0x033, 0x039, 0x03c, 0x041, 0x043,
|
||||
0x04a, 0x04f, 0x052, 0x056, 0x05a, 0x05f, 0x063, 0x068, 0x06d, 0x072,
|
||||
0x077, 0x079, 0x07c, 0x080, 0x084, 0x088, 0x08e, 0x094, 0x09a, 0x09f,
|
||||
0x0a4, 0x0a9, 0x0ad, 0x0b1, 0x0b6, 0x0bc, 0x0be, 0x0c2, 0x0c8, 0x0cc,
|
||||
0x0d4, 0x0da, 0x0e0, 0x0e5, 0x0eb, 0x0f0, 0x0f5, 0x0f9, 0x0fe, 0x104,
|
||||
0x10c, 0x112, 0x118, 0x11e, 0x121, 0x129, 0x12c, 0x132, 0x13a, 0x13e,
|
||||
0x143, 0x148, 0x14d, 0x152, 0x157, 0x15a, 0x15f, 0x164, 0x166, 0x168,
|
||||
0x16d, 0x16f, 0x177, 0x17c, 0x182, 0x187, 0x18c, 0x18f, 0x193, 0x196,
|
||||
0x19b, 0x1a1, 0x1a9, 0x1af, 0x1b4, 0x1ba, 0x1be, 0x1c0, 0x1c4, 0x1ca,
|
||||
0x1d2, 0x1dd };
|
||||
|
||||
const unsigned char icon1[] = {
|
||||
255, 255, 255, 128, 0, 1, 128, 0, 1, 128, 0, 1, 128, 0, 1, 128, 0, 1,
|
||||
128, 0, 1, 128, 0, 1, 128, 0, 1, 128, 0, 1, 128, 0, 1,
|
||||
128, 0, 1, 128, 0, 1, 128, 0, 1, 128, 0, 1, 128, 0, 1,
|
||||
128, 0, 1, 128, 0, 1, 128, 0, 1, 128, 0, 1, 255, 255, 255 };
|
||||
|
||||
char *ProgName; /* for AbEnd, later remove and use common/cmdline.h */
|
||||
|
||||
char *outputCName=NULL, *outputSName=NULL, *outputVName=NULL;
|
||||
FILE *outputCFile, *outputSFile, *outputVFile;
|
||||
int CFnum=0, SFnum=0, VFnum=0;
|
||||
int forceFlag=0;
|
||||
char outputCMode[2]="w";
|
||||
char outputSMode[2]="w";
|
||||
char outputVMode[2]="w";
|
||||
58
src/grc65/make/gcc.mak
Normal file
58
src/grc65/make/gcc.mak
Normal file
@@ -0,0 +1,58 @@
|
||||
#
|
||||
# gcc Makefile for grc
|
||||
#
|
||||
|
||||
# ------------------------------------------------------------------------------
|
||||
|
||||
# The executable to build
|
||||
EXE = grc65
|
||||
|
||||
COMMON = ../common
|
||||
|
||||
#
|
||||
CC = gcc
|
||||
CFLAGS = -g -O2 -Wall -W -std=c89
|
||||
override CFLAGS += -I$(COMMON)
|
||||
LDFLAGS =
|
||||
EBIND = emxbind
|
||||
|
||||
|
||||
# -----------------------------------------------------------------------------
|
||||
# List of all object files
|
||||
|
||||
OBJS = grc65.o
|
||||
|
||||
LIBS = $(COMMON)/common.a
|
||||
|
||||
# ------------------------------------------------------------------------------
|
||||
# Makefile targets
|
||||
|
||||
# Main target - must be first
|
||||
.PHONY: all
|
||||
ifeq (.depend,$(wildcard .depend))
|
||||
all: $(EXE)
|
||||
include .depend
|
||||
else
|
||||
all: depend
|
||||
@$(MAKE) -f make/gcc.mak all
|
||||
endif
|
||||
|
||||
$(EXE): $(OBJS) $(LIBS)
|
||||
$(CC) $(LDFLAGS) $^ -o $@
|
||||
@if [ $(OS2_SHELL) ] ; then $(EBIND) $(EXE) ; fi
|
||||
|
||||
clean:
|
||||
$(RM) *~ core.* *.map
|
||||
|
||||
zap: clean
|
||||
$(RM) *.o $(EXE) .depend
|
||||
|
||||
# ------------------------------------------------------------------------------
|
||||
# Make the dependencies
|
||||
|
||||
.PHONY: depend dep
|
||||
depend dep: $(OBJS:.o=.c)
|
||||
@echo "Creating dependency information"
|
||||
$(CC) $(CFLAGS) -MM $^ > .depend
|
||||
|
||||
|
||||
97
src/grc65/make/watcom.mak
Normal file
97
src/grc65/make/watcom.mak
Normal file
@@ -0,0 +1,97 @@
|
||||
#
|
||||
# GRC Makefile for the Watcom compiler (using GNU make)
|
||||
#
|
||||
|
||||
# ------------------------------------------------------------------------------
|
||||
# Generic stuff
|
||||
|
||||
# Environment variables for the watcom compiler
|
||||
export WATCOM = c:\\watcom
|
||||
export INCLUDE = $(WATCOM)\\h
|
||||
|
||||
# We will use the windows compiler under linux (define as empty for windows)
|
||||
export WINEDEBUG=fixme-all
|
||||
WINE = wine
|
||||
|
||||
# Programs
|
||||
AR = $(WINE) wlib
|
||||
CC = $(WINE) wcc386
|
||||
LD = $(WINE) wlink
|
||||
WSTRIP = $(WINE) wstrip -q
|
||||
|
||||
LNKCFG = ld.tmp
|
||||
|
||||
# Program arguments
|
||||
CFLAGS = -d1 -obeilr -zp4 -5 -zq -w2 -i=..\\common
|
||||
|
||||
# Target files
|
||||
EXE = grc65.exe
|
||||
|
||||
# Create NT programs by default
|
||||
ifndef TARGET
|
||||
TARGET = NT
|
||||
endif
|
||||
|
||||
# --------------------- OS2 ---------------------
|
||||
ifeq ($(TARGET),OS2)
|
||||
SYSTEM = os2v2
|
||||
CFLAGS += -bt=$(TARGET)
|
||||
endif
|
||||
|
||||
# -------------------- DOS4G --------------------
|
||||
ifeq ($(TARGET),DOS32)
|
||||
SYSTEM = dos4g
|
||||
CFLAGS += -bt=$(TARGET)
|
||||
endif
|
||||
|
||||
# --------------------- NT ----------------------
|
||||
ifeq ($(TARGET),NT)
|
||||
SYSTEM = nt
|
||||
CFLAGS += -bt=$(TARGET)
|
||||
endif
|
||||
|
||||
# ------------------------------------------------------------------------------
|
||||
# Implicit rules
|
||||
|
||||
%.obj: %.c
|
||||
$(CC) $(CFLAGS) -fo=$@ $^
|
||||
|
||||
|
||||
# ------------------------------------------------------------------------------
|
||||
# All OBJ files
|
||||
|
||||
OBJS = grc65.obj
|
||||
|
||||
LIBS = ../common/common.lib
|
||||
|
||||
|
||||
# ------------------------------------------------------------------------------
|
||||
# Main targets
|
||||
|
||||
all: $(EXE)
|
||||
|
||||
|
||||
# ------------------------------------------------------------------------------
|
||||
# Other targets
|
||||
|
||||
|
||||
$(EXE): $(OBJS) $(LIBS)
|
||||
@echo "DEBUG ALL" > $(LNKCFG)
|
||||
@echo "OPTION QUIET" >> $(LNKCFG)
|
||||
@echo "OPTION MAP" >> $(LNKCFG)
|
||||
@echo "OPTION STACK=65536" >> $(LNKCFG)
|
||||
@echo "NAME $@" >> $(LNKCFG)
|
||||
@for i in $(OBJS); do echo "FILE $${i}"; done >> $(LNKCFG)
|
||||
@for i in $(LIBS); do echo "LIBRARY $${i}"; done >> $(LNKCFG)
|
||||
@$(LD) system $(SYSTEM) @$(LNKCFG)
|
||||
@rm $(LNKCFG)
|
||||
|
||||
clean:
|
||||
@rm -f *~ core
|
||||
|
||||
zap: clean
|
||||
@rm -f $(OBJS) $(EXE) $(EXE:.exe=.map)
|
||||
|
||||
strip:
|
||||
@-$(WSTRIP) $(EXE)
|
||||
|
||||
Reference in New Issue
Block a user