From d810ed97d255b7e51ea91d62e713cf5ca7c0d416 Mon Sep 17 00:00:00 2001
From: "ol.sc"
Date: Mon, 26 Dec 2011 22:54:04 +0000
Subject: [PATCH] Redesigned GEOS VLIR linking: - No more post-linking with
resource compiler, rather ld65 directly creates the VLIR CVT file. - No more
dynamic linker config creation, rather the built-in 'geos' config is usable
both for SEQ CVT and VLIR CVT files.
ToDos:
- Have ld65 accept alignment to $FD.
- Adjust docs / samples.
git-svn-id: svn://svn.cc65.org/cc65/trunk@5314 b7a2c559-68d2-44c3-8de9-860c34a00d81
---
libsrc/geos/devel/crt0.s | 6 +-
src/grc65/grc65.c | 1178 ++++++++++++++++++++------------------
src/grc65/grc65.h | 77 ---
src/ld65/cfg/geos.cfg | 70 ++-
4 files changed, 665 insertions(+), 666 deletions(-)
delete mode 100644 src/grc65/grc65.h
diff --git a/libsrc/geos/devel/crt0.s b/libsrc/geos/devel/crt0.s
index fb0289a4c..1e138e8e4 100644
--- a/libsrc/geos/devel/crt0.s
+++ b/libsrc/geos/devel/crt0.s
@@ -7,7 +7,7 @@
.export _exit
.export __STARTUP__ : absolute = 1 ; Mark as startup
- .import __RAM_START__, __RAM_SIZE__ ; Linker generated
+ .import __VLIR0_START__, __VLIR0_SIZE__ ; Linker generated
.import __STACKSIZE__ ; Linker generated
.import initlib, donelib
.import callmain
@@ -26,9 +26,9 @@
; Setup stack
- lda #<(__RAM_START__ + __RAM_SIZE__ + __STACKSIZE__)
+ lda #<(__VLIR0_START__ + __VLIR0_SIZE__ + __STACKSIZE__)
sta sp
- lda #>(__RAM_START__ + __RAM_SIZE__ + __STACKSIZE__)
+ lda #>(__VLIR0_START__ + __VLIR0_SIZE__ + __STACKSIZE__)
sta sp+1 ; Set argument stack ptr
; Call module constructors
diff --git a/src/grc65/grc65.c b/src/grc65/grc65.c
index ac5564632..75d33b2cd 100644
--- a/src/grc65/grc65.c
+++ b/src/grc65/grc65.c
@@ -1,18 +1,14 @@
+/* GEOS resource compiler
-/*
- GEOS resource compiler
-
- by Maciej 'YTM/Elysium' Witkowiak
-
- see GEOSLib documentation for license info
+ 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)
+/* - 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...)
@@ -25,377 +21,374 @@
#include
#include
-#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;llen)
- name[len]='\0';
+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");
- }
+
+void fillOut(char *name, int len, char *filler) {
+
+ int a;
+
+ setLen(name, len);
+ fprintf(outputSFile, "\t.byte \"%s\"\n", name);
+
+ a = strlen(name);
+ if (a < len) {
+ fprintf(outputSFile, "\t.res (%i - %i), %s\n", len, a, filler);
+ }
}
+
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 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++; }
+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;
+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();
+ 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);
- }
+ if (strcmp(nextWord(), "{") != 0) {
+ AbEnd("menu '%s' description has no opening bracket!\n", myMenu.name);
+ }
curItem = xmalloc(sizeof(struct menuitem));
- myMenu.item=curItem;
+ 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));
+ 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,
+ "\n"
+ "const 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,
+ "};\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;
+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;
+ struct appheader myHead;
+ char *token;
+ char i1[9], i2[9], i3[9];
+ int a, b;
openSFile();
token = nextWord();
- a = findToken (hdrFTypes, token);
+ 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);
- }
+ 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();
@@ -406,12 +399,12 @@ int a, b;
/* put default values into myHead here */
myHead.author = "cc65";
myHead.info = "Program compiled with cc65 and GEOSLib.";
- myHead.dostype = 128+3;
+ myHead.dostype = 128 + 3;
myHead.structure = 0;
myHead.mode = 0;
t = time(NULL);
- my_tm = localtime (&t);
+ my_tm = localtime(&t);
myHead.year = my_tm->tm_year;
myHead.month = my_tm->tm_mon+1;
@@ -419,320 +412,361 @@ int a, b;
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);
+ 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;
+ 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);
+ } 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);
+ "\n"
+ "\t\t.segment \"DIRENTRY\"\n\n"
+ "\t.byte %i\n"
+ "\t.word 0\n", myHead.dostype);
- fillOut(myHead.dosname,16,"$a0");
+ 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);
+ "\t.word 0\n"
+ "\t.byte %i\n"
+ "\t.byte %i\n"
+ "\t.byte %i, %i, %i, %i, %i\n\n"
+ "\t.word 0\n"
+ "\t.byte \"PRG formatted GEOS file V1.0\"\n\n",
+ 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.segment \"FILEINFO\"\n\n"
+ "\t.import __VLIR0_START__, __STARTUP_RUN__\n\n"
+ "\t.byte 3, 21, 63 | $80\n");
+
+ for (a = 0; a != 63; a = a + 3) {
+ fprintf(outputSFile,
+ "\t.byte %%%s, %%%s, %%%s\n",
+ 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);
+ "\t.byte %i, %i, %i\n"
+ "\t.word __VLIR0_START__, __VLIR0_START__ - 1, __STARTUP_RUN__\n\n",
+ myHead.dostype, myHead.geostype, myHead.structure);
- fillOut(myHead.classname,12,"$20");
+ fillOut(myHead.classname, 12, "$20");
- fillOut(myHead.version,4,"0");
+ fillOut(myHead.version, 4, "0");
fprintf(outputSFile,
- ".byte 0, 0, 0\n\t\t.byte %i\n\n\t\t", myHead.mode);
+ "\t.byte 0, 0, 0\n"
+ "\t.byte %i\n\n", myHead.mode);
- setLen(myHead.author,62);
+ 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));
+ "\t.byte \"%s\"\n"
+ "\t.byte 0\n"
+ "\t.res (63 - %i)\n\n",
+ 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));
+ "\t.byte \"%s\"\n"
+ "\t.byte 0\n\n", myHead.info);
+ 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;
-};
+void DoVLIR(void) {
-struct vlirentry vlirtable[127];
+ char *token;
+ int record, lastrecord;
+ int vlirsize, vlirtable[127];
- openVFile();
-
- headname = nextWord();
+ openSFile();
- vlirbase = strtol(nextWord(),NULL,0);
+ vlirsize = strtol(nextWord(), NULL, 0);
- if (strcmp(nextWord(),"{")!=0)
- AbEnd ("VLIR description has no opening bracket!\n");
+ if (strcmp(nextWord(), "{") != 0) {
+ AbEnd ("VLIR description has no opening bracket!\n");
+ }
- numchains=0;
+ lastrecord = -1;
+ memset(vlirtable, 0, sizeof(vlirtable));
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);
+ token = nextWord();
+ if (strcmp(token, "}") == 0) break;
+
+ record = atoi(token);
+ if (record < 0 || record > 126) {
+ AbEnd("VLIR record %i is out of range 0-126.\n", record);
+ }
+ if (vlirtable[record] == 1) {
+ AbEnd("VLIR record %i is defined twice.\n", record);
+ }
+
+ vlirtable[record] = 1;
+ if (record > lastrecord) lastrecord = record;
+ } while (strcmp(token, "}") != 0);
+
+ if (lastrecord == -1) {
+ AbEnd("There must be at least one VLIR record.\n");
}
- fprintf(outputVFile,"}\n");
+
+ /* always include record 0 */
+ vlirtable[0] = 1;
+
+ /* OK, all information is gathered, do flushout */
- /* 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(outputSFile,
+ "\n"
+ "\t\t.segment \"RECORDS\"\n\n"
+ "\t.export __OVERLAYSIZE__ : absolute = $%04x\n\n",
+ vlirsize);
+
+ for (record = 0; record <= lastrecord; record++) {
+ if (vlirtable[record] == 1) {
+ fprintf(outputSFile,
+ "\t.import __VLIR%i_START__, __VLIR%i_LAST__\n",
+ record, record);
+ }
+ }
+ fprintf(outputSFile,
+ "\n");
+
+ for (record = 0; record <= lastrecord; record++) {
+ if (vlirtable[record] == 1) {
+ fprintf(outputSFile,
+ "\t.byte .lobyte ((__VLIR%i_LAST__ - __VLIR%i_START__ - 1) / 254) + 1\n"
+ "\t.byte .lobyte ((__VLIR%i_LAST__ - __VLIR%i_START__ - 1) .MOD 254) + 2\n",
+ record, record, record, record);
+ } else {
+ fprintf(outputSFile,
+ "\t.byte $00\n"
+ "\t.byte $FF\n");
+ }
}
- fprintf(outputVFile,"\n");
+ fprintf(outputSFile,
+ "\n");
+
+ if (fclose(outputSFile) != 0)
+ AbEnd("error closing %s: %s\n", outputSName, strerror(errno));
+
+ openCFile();
- if (fclose (outputVFile)!=0)
- AbEnd("error closing %s: %s\n",outputVName,strerror (errno));
+ fprintf(outputCFile,
+ "extern void _OVERLAYADDR__;\n"
+ "extern void _OVERLAYSIZE__;\n\n"
+ "#define OVERLAY_ADDR (char*) &_OVERLAYADDR__\n"
+ "#define OVERLAY_SIZE (unsigned)&_OVERLAYSIZE__\n\n");
+
+ if (fclose(outputCFile) != 0)
+ AbEnd("error closing %s: %s\n", outputCName, 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;
+
+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; }
- }
+ 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");
+ if (bracket != 0) AbEnd("there are unclosed brackets!\n");
return tbl;
}
-void processFile (const char *filename) {
-FILE *F;
+void processFile(const char *filename) {
-char *str;
-char *token;
+ FILE *F;
-int head=0; /* number of processed HEADER sections */
-int vlir=0; /* number of processed VLIR sections */
+ char *str;
+ char *token;
- if ((F = fopen (filename,"r"))==0)
- AbEnd("can't open file %s for reading: %s\n",filename,strerror (errno));
+ 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));
+ str = filterInput(F, xmalloc(BLOODY_BIG_BUFFER));
- token = strtok (str," ");
+ 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);
+ 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;
+ 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");
+ 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 'h':
+ case '?':
+ printUsage();
+ exit(EXIT_SUCCESS);
+ break;
+ default:
+ AbEnd("unknown option %s\n", arg);
+ }
+ } else {
+ ffile++;
+
+ if (outputCName == NULL) outputCName = MakeFilename(arg, ".h");
+ if (outputSName == NULL) outputSName = MakeFilename(arg, ".s");
+
+ processFile(arg);
+ }
+
+ i++;
+ }
+
+ if (ffile == 0) AbEnd("no input file\n");
return EXIT_SUCCESS;
}
diff --git a/src/grc65/grc65.h b/src/grc65/grc65.h
deleted file mode 100644
index 9825fe931..000000000
--- a/src/grc65/grc65.h
+++ /dev/null
@@ -1,77 +0,0 @@
-
-/* 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";
diff --git a/src/ld65/cfg/geos.cfg b/src/ld65/cfg/geos.cfg
index 7c71059ab..4dcc5fca4 100644
--- a/src/ld65/cfg/geos.cfg
+++ b/src/ld65/cfg/geos.cfg
@@ -1,22 +1,64 @@
SYMBOLS {
- __STACKSIZE__: type = weak, value = $0400; # 1k stack
+ __STACKSIZE__: type = weak, value = $0400; # 1k stack
+ __OVERLAYSIZE__: type = weak, value = $0000; # no overlays by default
+ __OVERLAYADDR__: type = weak, value = $6000 - __OVERLAYSIZE__;
}
MEMORY {
- ZP: file = "", define = yes, start = $0058, size = $0028;
- HEADER: file = %O, start = $0204, size = $01FC;
- RAM: file = %O, define = yes, start = $0400, size = $5C00 - __STACKSIZE__;
+ CVT: file = %O, start = $0, size = $80000;
+ ZP: define = yes, start = $58, size = $1A + $06;
+ VLIR0: define = yes, start = $0400, size = __OVERLAYADDR__ - __STACKSIZE__ - $0400;
+ VLIR1: define = yes, start = __OVERLAYADDR__, size = __OVERLAYSIZE__;
+ VLIR2: define = yes, start = __OVERLAYADDR__, size = __OVERLAYSIZE__;
+ VLIR3: define = yes, start = __OVERLAYADDR__, size = __OVERLAYSIZE__;
+ VLIR4: define = yes, start = __OVERLAYADDR__, size = __OVERLAYSIZE__;
+ VLIR5: define = yes, start = __OVERLAYADDR__, size = __OVERLAYSIZE__;
+ VLIR6: define = yes, start = __OVERLAYADDR__, size = __OVERLAYSIZE__;
+ VLIR7: define = yes, start = __OVERLAYADDR__, size = __OVERLAYSIZE__;
+ VLIR8: define = yes, start = __OVERLAYADDR__, size = __OVERLAYSIZE__;
+ VLIR9: define = yes, start = __OVERLAYADDR__, size = __OVERLAYSIZE__;
+ VLIR10: define = yes, start = __OVERLAYADDR__, size = __OVERLAYSIZE__;
+ VLIR11: define = yes, start = __OVERLAYADDR__, size = __OVERLAYSIZE__;
+ VLIR12: define = yes, start = __OVERLAYADDR__, size = __OVERLAYSIZE__;
+ VLIR13: define = yes, start = __OVERLAYADDR__, size = __OVERLAYSIZE__;
+ VLIR14: define = yes, start = __OVERLAYADDR__, size = __OVERLAYSIZE__;
+ VLIR15: define = yes, start = __OVERLAYADDR__, size = __OVERLAYSIZE__;
+ VLIR16: define = yes, start = __OVERLAYADDR__, size = __OVERLAYSIZE__;
+ VLIR17: define = yes, start = __OVERLAYADDR__, size = __OVERLAYSIZE__;
+ VLIR18: define = yes, start = __OVERLAYADDR__, size = __OVERLAYSIZE__;
+ VLIR19: define = yes, start = __OVERLAYADDR__, size = __OVERLAYSIZE__;
}
SEGMENTS {
- HEADER: load = HEADER, type = ro;
- STARTUP: load = RAM, type = ro;
- LOWCODE: load = RAM, type = ro, optional = yes;
- INIT: load = RAM, type = ro, define = yes, optional = yes;
- CODE: load = RAM, type = ro;
- RODATA: load = RAM, type = ro;
- DATA: load = RAM, type = rw;
- BSS: load = RAM, type = bss, define = yes;
- ZEROPAGE: load = ZP, type = zp;
- EXTZP: load = ZP, type = zp;
+ ZEROPAGE: type = zp, load = ZP;
+ EXTZP: type = zp, load = ZP;
+ DIRENTRY: type = ro, load = CVT, align = $FE;
+ FILEINFO: type = ro, load = CVT, align = $FE;
+ RECORDS: type = ro, load = CVT, align = $FE, optional = yes;
+ STARTUP: type = ro, run = VLIR0, load = CVT, align_load = $FE, define = yes;
+ LOWCODE: type = ro, run = VLIR0, load = CVT, optional = yes;
+ INIT: type = ro, run = VLIR0, load = CVT, define = yes, optional = yes;
+ CODE: type = ro, run = VLIR0, load = CVT;
+ RODATA: type = ro, run = VLIR0, load = CVT;
+ DATA: type = rw, run = VLIR0, load = CVT;
+ BSS: type = bss, load = VLIR0, define = yes;
+ OVERLAY1: type = ro, run = VLIR1, load = CVT, align_load = $FE, optional = yes;
+ OVERLAY2: type = ro, run = VLIR2, load = CVT, align_load = $FE, optional = yes;
+ OVERLAY3: type = ro, run = VLIR3, load = CVT, align_load = $FE, optional = yes;
+ OVERLAY4: type = ro, run = VLIR4, load = CVT, align_load = $FE, optional = yes;
+ OVERLAY5: type = ro, run = VLIR5, load = CVT, align_load = $FE, optional = yes;
+ OVERLAY6: type = ro, run = VLIR6, load = CVT, align_load = $FE, optional = yes;
+ OVERLAY7: type = ro, run = VLIR7, load = CVT, align_load = $FE, optional = yes;
+ OVERLAY8: type = ro, run = VLIR8, load = CVT, align_load = $FE, optional = yes;
+ OVERLAY9: type = ro, run = VLIR9, load = CVT, align_load = $FE, optional = yes;
+ OVERLAY10: type = ro, run = VLIR10, load = CVT, align_load = $FE, optional = yes;
+ OVERLAY11: type = ro, run = VLIR11, load = CVT, align_load = $FE, optional = yes;
+ OVERLAY12: type = ro, run = VLIR12, load = CVT, align_load = $FE, optional = yes;
+ OVERLAY13: type = ro, run = VLIR13, load = CVT, align_load = $FE, optional = yes;
+ OVERLAY14: type = ro, run = VLIR14, load = CVT, align_load = $FE, optional = yes;
+ OVERLAY15: type = ro, run = VLIR15, load = CVT, align_load = $FE, optional = yes;
+ OVERLAY16: type = ro, run = VLIR16, load = CVT, align_load = $FE, optional = yes;
+ OVERLAY17: type = ro, run = VLIR17, load = CVT, align_load = $FE, optional = yes;
+ OVERLAY18: type = ro, run = VLIR18, load = CVT, align_load = $FE, optional = yes;
+ OVERLAY19: type = ro, run = VLIR19, load = CVT, align_load = $FE, optional = yes;
}
FEATURES {
CONDES: segment = INIT,
--
2.39.5