Skip to content

Commit

Permalink
Merge pull request cc65#2499 from kugelfuhr/kugelfuhr/disable-recursi…
Browse files Browse the repository at this point in the history
…ve-calls-to-main

Disallow recursive calls to main() in cc65 mode
  • Loading branch information
mrdudz authored Sep 7, 2024
2 parents 7bd0a1d + 79606c4 commit 4e2a3bd
Show file tree
Hide file tree
Showing 11 changed files with 63 additions and 22 deletions.
5 changes: 5 additions & 0 deletions doc/cc65.sgml
Original file line number Diff line number Diff line change
Expand Up @@ -824,6 +824,11 @@ and the one defined by the ISO standard:
as it sounds, since the 6502 has so few registers that it isn't
possible to keep values in registers anyway.
<p>
<item> In <tt/cc65/ mode, <tt/main()/ cannot be called recursively. If this
is necessary, the program must be compiled in <tt/c89/ or <tt/c99/ mode
using the <tt><ref id="option--standard" name="--standard"></tt>
command line option.
<p>
</itemize>

There may be some more minor differences I'm currently not aware of. The
Expand Down
8 changes: 4 additions & 4 deletions src/cc65/codegen.c
Original file line number Diff line number Diff line change
Expand Up @@ -507,13 +507,13 @@ void g_enter (unsigned flags, unsigned argsize)



void g_leave (int IsMainFunc)
void g_leave (int DoCleanup)
/* Function epilogue */
{
/* In the main function nothing has to be dropped because the program
** is terminated anyway.
/* In the main function in cc65 mode nothing has to be dropped because
** the program is terminated anyway.
*/
if (!IsMainFunc) {
if (DoCleanup) {
/* How many bytes of locals do we have to drop? */
unsigned ToDrop = (unsigned) -StackPtr;

Expand Down
2 changes: 1 addition & 1 deletion src/cc65/codegen.h
Original file line number Diff line number Diff line change
Expand Up @@ -247,7 +247,7 @@ void g_scale (unsigned flags, long val);
void g_enter (unsigned flags, unsigned argsize);
/* Function prologue */

void g_leave (int IsMainFunc);
void g_leave (int DoCleanup);
/* Function epilogue */


Expand Down
26 changes: 22 additions & 4 deletions src/cc65/expr.c
Original file line number Diff line number Diff line change
Expand Up @@ -1219,9 +1219,6 @@ static void Primary (ExprDesc* E)
/* Is the symbol known? */
if (Sym) {

/* We found the symbol - skip the name token */
NextToken ();

/* Check for illegal symbol types */
CHECK ((Sym->Flags & SC_TYPEMASK) != SC_LABEL);
if ((Sym->Flags & SC_TYPEMASK) == SC_TYPEDEF) {
Expand All @@ -1230,9 +1227,14 @@ static void Primary (ExprDesc* E)
/* Assume an int type to make E valid */
E->Flags = E_LOC_STACK | E_RTYPE_LVAL;
E->Type = type_int;
/* Skip the erroneous token */
NextToken ();
break;
}

/* Skip the name token */
NextToken ();

/* Mark the symbol as referenced */
Sym->Flags |= SC_REF;

Expand Down Expand Up @@ -1286,7 +1288,23 @@ static void Primary (ExprDesc* E)
** rvalue, too, because we cannot store anything in a function.
** So fix the flags depending on the type.
*/
if (IsTypeArray (E->Type) || IsTypeFunc (E->Type)) {
if (IsTypeArray (E->Type)) {
ED_AddrExpr (E);
} else if (IsTypeFunc (E->Type)) {
/* In cc65 mode we cannot call or take the address of
** main().
*/
if (IS_Get (&Standard) == STD_CC65 &&
strcmp (Sym->Name, "main") == 0) {
/* Adjust the error message depending on a call or an
** address operation.
*/
if (CurTok.Tok == TOK_LPAREN) {
Error ("'main' must not be called recursively");
} else {
Error ("The address of 'main' cannot be taken");
}
}
ED_AddrExpr (E);
}

Expand Down
10 changes: 7 additions & 3 deletions src/cc65/function.c
Original file line number Diff line number Diff line change
Expand Up @@ -646,13 +646,17 @@ void NewFunc (SymEntry* Func, FuncDesc* D)
/* Output the function exit code label */
g_defcodelabel (F_GetRetLab (CurrentFunc));

/* Restore the register variables (not necessary for main function) */
if (!F_IsMainFunc (CurrentFunc)) {
/* Restore the register variables (not necessary for the main function in
** cc65 mode)
*/
int CleanupOnExit = (IS_Get (&Standard) != STD_CC65) ||
!F_IsMainFunc (CurrentFunc);
if (CleanupOnExit) {
F_RestoreRegVars (CurrentFunc);
}

/* Generate the exit code */
g_leave (F_IsMainFunc (CurrentFunc));
g_leave (CleanupOnExit);

/* Emit references to imports/exports */
EmitExternals ();
Expand Down
15 changes: 11 additions & 4 deletions src/cc65/locals.c
Original file line number Diff line number Diff line change
Expand Up @@ -111,6 +111,13 @@ static void ParseRegisterDecl (Declarator* Decl, int Reg)
/* Get the size of the variable */
unsigned Size = SizeOf (Decl->Type);

/* Check if this is the main function and we are in cc65 mode. If so, we
** won't save the old contents of the register variables since in cc65
** mode main() may not be called recursively.
*/
int SaveRegVars = (IS_Get (&Standard) != STD_CC65) ||
!F_IsMainFunc (CurrentFunc);

/* Check for an optional initialization */
if (CurTok.Tok == TOK_ASSIGN) {

Expand All @@ -126,13 +133,13 @@ static void ParseRegisterDecl (Declarator* Decl, int Reg)
/* Save the current contents of the register variable on stack. This is
** not necessary for the main function.
*/
if (!F_IsMainFunc (CurrentFunc)) {
if (SaveRegVars) {
g_save_regvars (Reg, Size);
}

/* Add the symbol to the symbol table. We do that now, because for
** register variables the current stack pointer is implicitly used
** as location for the save area (unused in case of main()).
** as location for the save area (maybe unused in case of main()).
*/
Sym = AddLocalSym (Decl->Ident, Decl->Type, Decl->StorageClass, Reg);

Expand Down Expand Up @@ -187,14 +194,14 @@ static void ParseRegisterDecl (Declarator* Decl, int Reg)
/* Save the current contents of the register variable on stack. This is
** not necessary for the main function.
*/
if (!F_IsMainFunc (CurrentFunc)) {
if (SaveRegVars) {
F_AllocLocalSpace (CurrentFunc);
g_save_regvars (Reg, Size);
}

/* Add the symbol to the symbol table. We do that now, because for
** register variables the current stack pointer is implicitly used
** as location for the save area (unused in case of main()).
** as location for the save area (maybe unused in case of main()).
*/
Sym = AddLocalSym (Decl->Ident, Decl->Type, Decl->StorageClass, Reg);
}
Expand Down
4 changes: 3 additions & 1 deletion targettest/atari/mem.c
Original file line number Diff line number Diff line change
Expand Up @@ -21,6 +21,8 @@ unsigned int *MEMTOP = (unsigned int *)741;
unsigned int *MEMLO = (unsigned int *)743;
void *allocmem;

void code(void) { }

int main(void)
{
allocmem = malloc(257);
Expand All @@ -35,7 +37,7 @@ int main(void)
printf(" MEMLO = $%04X (%u)\n", *MEMLO, *MEMLO);

printf(" ----------------------\n");
printf(" main: $%04X (code)\n", &main);
printf(" code: $%04X (code)\n", &code);
printf(" data: $%04X (data)\n", &data);
printf(" _dos_type: $%04X (bss)\n", &_dos_type);
printf(" allocmem: $%04X (dyn. data)\n", allocmem);
Expand Down
4 changes: 3 additions & 1 deletion targettest/pce/conio.c
Original file line number Diff line number Diff line change
Expand Up @@ -11,6 +11,8 @@ static char hex[16] = { "0123456789abcdef" };
static char charbuf[0x20];
static char colbuf[0x20];

void func(void) { }

void main(void)
{
int stackvar = 42;
Expand Down Expand Up @@ -65,7 +67,7 @@ void main(void)
p[8],p[9],p[10],p[11],p[12],p[13],p[14],p[15]
);
}
memcpy(p, main, i = 0); /* test that a zero length doesn't copy 64K */
memcpy(p, func, i = 0); /* test that a zero length doesn't copy 64K */

gotoxy(0,ysize - 1);
for (i = 0; i < xsize; ++i) {
Expand Down
4 changes: 3 additions & 1 deletion targettest/scanf-test.c
Original file line number Diff line number Diff line change
Expand Up @@ -159,12 +159,14 @@ static void Pause(void) {
#endif
}

static void Nil() { }

int main(void) {
long n0;
unsigned t;
int c, n1 = 12345, n2, n3;
char s1[80], s2[80];
void *p1 = main, *p2 = main, *p3 = main, *p4 = main;
void *p1 = Nil, *p2 = Nil, *p3 = Nil, *p4 = Nil;

#ifndef USE_STDIO
clrscr();
Expand Down
3 changes: 1 addition & 2 deletions test/ref/custom-reference-error.c
Original file line number Diff line number Diff line change
Expand Up @@ -22,6 +22,5 @@ return_t main(int argc, char* argv[])
n = 0; /* produce an error */
/* produce a warning */
}

int arr[main(0, 0)]; /* produce an error */
int b = 0;
int arr[b]; /* produce an error */
4 changes: 3 additions & 1 deletion test/val/nullptr.c
Original file line number Diff line number Diff line change
Expand Up @@ -28,6 +28,8 @@ struct S {
} \
} while(0);

void func() { }

int main()
{
int a;
Expand Down Expand Up @@ -60,7 +62,7 @@ int main()
TEST_NON_NULL(((struct S*)&a)->a)

/* Non-null pointer obtained with cast and -> */
TEST_NON_NULL(((struct S*)&main)->a)
TEST_NON_NULL(((struct S*)&func)->a)

if (failures != 0)
{
Expand Down

0 comments on commit 4e2a3bd

Please sign in to comment.