]> git.sur5r.net Git - cc65/blobdiff - src/cc65/compile.c
Fixed two compiler warnings.
[cc65] / src / cc65 / compile.c
index 4301fb7375f81f634a2522671d2efb84c7d3da05..d7167706b21de9ca4ec2e9ba05944fe201ee3c29 100644 (file)
@@ -2,14 +2,14 @@
 /*                                                                           */
 /*                                compile.c                                 */
 /*                                                                           */
-/*                      Top level compiler subroutine                       */
+/*                      Top level compiler subroutine                       */
 /*                                                                           */
 /*                                                                           */
 /*                                                                           */
-/* (C) 2000-2004 Ullrich von Bassewitz                                       */
-/*               Römerstrasse 52                                             */
-/*               D-70794 Filderstadt                                         */
-/* EMail:        uz@cc65.org                                                 */
+/* (C) 2000-2009, Ullrich von Bassewitz                                      */
+/*                Roemerstrasse 52                                           */
+/*                D-70794 Filderstadt                                        */
+/* EMail:         uz@cc65.org                                                */
 /*                                                                           */
 /*                                                                           */
 /* This software is provided 'as-is', without any expressed or implied       */
@@ -46,6 +46,7 @@
 #include "asmlabel.h"
 #include "asmstmt.h"
 #include "codegen.h"
+#include "codeopt.h"
 #include "compile.h"
 #include "declare.h"
 #include "error.h"
@@ -55,7 +56,9 @@
 #include "input.h"
 #include "litpool.h"
 #include "macrotab.h"
+#include "output.h"
 #include "pragma.h"
+#include "preproc.h"
 #include "standard.h"
 #include "symtab.h"
 
@@ -81,8 +84,6 @@ static void Parse (void)
     while (CurTok.Tok != TOK_CEOF) {
 
        DeclSpec        Spec;
-       Declaration     Decl;
-       int             NeedStorage;
 
        /* Check for empty statements */
        if (CurTok.Tok == TOK_SEMI) {
@@ -90,10 +91,13 @@ static void Parse (void)
            continue;
        }
 
-       /* Check for an ASM statement (which is allowed also on global level) */
+               /* Disallow ASM statements on global level */
        if (CurTok.Tok == TOK_ASM) {
+            Error ("__asm__ is not allowed here");                 
+            /* Parse and remove the statement for error recovery */
            AsmStatement ();
            ConsumeSemi ();
+            RemoveGlobalCode ();
            continue;
        }
 
@@ -122,22 +126,12 @@ static void Parse (void)
            continue;
        }
 
-               /* Check if we must reserve storage for the variable. We do
-        * this if we don't had a storage class given ("int i") or
-        * if the storage class is explicitly specified as static.
-        * This means that "extern int i" will not get storage
-        * allocated.
-        */
-       NeedStorage = (Spec.StorageClass & SC_TYPEDEF) == 0 &&
-                     ((Spec.Flags & DS_DEF_STORAGE) != 0  ||
-                     (Spec.StorageClass & (SC_STATIC | SC_EXTERN)) == SC_STATIC);
-
        /* Read declarations for this type */
        Entry = 0;
        comma = 0;
                while (1) {
 
-           unsigned SymFlags;
+            Declaration         Decl;
 
            /* Read the next declaration */
            ParseDecl (&Spec, &Decl, DM_NEED_IDENT);
@@ -146,22 +140,49 @@ static void Parse (void)
                break;
            }
 
-           /* Get the symbol flags */
-           SymFlags = Spec.StorageClass;
-           if (IsTypeFunc (Decl.Type)) {
-               SymFlags |= SC_FUNC;
-           } else {
-               if (NeedStorage) {
-                   /* We will allocate storage, variable is defined */
-                   SymFlags |= SC_STORAGE | SC_DEF;
-               }
+            /* Check if we must reserve storage for the variable. We do this,
+             *
+             *   - if it is not a typedef or function,
+             *   - if we don't had a storage class given ("int i")
+             *   - if the storage class is explicitly specified as static,
+             *   - or if there is an initialization.
+             *
+             * This means that "extern int i;" will not get storage allocated.
+             */
+           if ((Decl.StorageClass & SC_FUNC) != SC_FUNC          &&
+                (Decl.StorageClass & SC_TYPEDEF) != SC_TYPEDEF    &&
+                ((Spec.Flags & DS_DEF_STORAGE) != 0         ||
+                 (Decl.StorageClass & (SC_EXTERN|SC_STATIC)) == SC_STATIC ||
+                 ((Decl.StorageClass & SC_EXTERN) != 0 &&
+                  CurTok.Tok == TOK_ASSIGN))) {
+
+                /* We will allocate storage */
+               Decl.StorageClass |= SC_STORAGE | SC_DEF;
            }
 
+            /* If this is a function declarator that is not followed by a comma
+             * or semicolon, it must be followed by a function body. If this is
+             * the case, convert an empty parameter list into one accepting no
+             * parameters (same as void) as required by the standard.
+             */
+            if ((Decl.StorageClass & SC_FUNC) != 0 &&
+                (CurTok.Tok != TOK_COMMA)          &&
+                (CurTok.Tok != TOK_SEMI)) {
+
+                FuncDesc* D = GetFuncDesc (Decl.Type);
+                if (D->Flags & FD_EMPTY) {
+                    D->Flags = (D->Flags & ~(FD_EMPTY | FD_VARIADIC)) | FD_VOID_PARAM;
+                }
+            }
+
            /* Add an entry to the symbol table */
-           Entry = AddGlobalSym (Decl.Ident, Decl.Type, SymFlags);
+           Entry = AddGlobalSym (Decl.Ident, Decl.Type, Decl.StorageClass);
+
+            /* Add declaration attributes */
+            SymUseAttr (Entry, &Decl);
 
            /* Reserve storage for the variable if we need to */
-                   if (SymFlags & SC_STORAGE) {
+                   if (Decl.StorageClass & SC_STORAGE) {
 
                /* Get the size of the variable */
                unsigned Size = SizeOf (Decl.Type);
@@ -178,15 +199,20 @@ static void Parse (void)
                                /* Size is unknown and not an array */
                                Error ("Variable `%s' has unknown size", Decl.Ident);
                            }
-                       } else if (ANSI) {
+                       } else if (IS_Get (&Standard) != STD_CC65) {
                            /* We cannot declare variables of type void */
                            Error ("Illegal type for variable `%s'", Decl.Ident);
                        }
                    }
 
-                   /* Switch to the data or rodata segment */
-                   if (IsQualConst (Decl.Type)) {
-                       g_userodata ();
+                   /* Switch to the data or rodata segment. For arrays, check
+                      * the element qualifiers, since not the array but its
+                      * elements are const.
+                      */
+                   if (IsQualConst (Decl.Type) ||
+                        (IsTypeArray (Decl.Type) &&
+                         IsQualConst (GetElementType (Decl.Type)))) {
+                       g_userodata ();
                    } else {
                        g_usedata ();
                    }
@@ -246,12 +272,15 @@ static void Parse (void)
                if (CurTok.Tok == TOK_SEMI) {
                    /* Prototype only */
                    NextToken ();
-               } else if (Entry) {
-                    /* Function body definition */
+               } else {
+
+                    /* Function body. Check for duplicate function definitions */
                     if (SymIsDef (Entry)) {
                         Error ("Body for function `%s' has already been defined",
                                Entry->Name);
                     }
+
+                    /* Parse the function body */
                     NewFunc (Entry);
                }
            }
@@ -270,9 +299,8 @@ static void Parse (void)
 void Compile (const char* FileName)
 /* Top level compile routine. Will setup things and call the parser. */
 {
-    char        Buf[20];
-    char        DateStr[20];
-    char        TimeStr[20];
+    char        DateStr[32];
+    char        TimeStr[32];
     time_t      Time;
     struct tm*  TM;
 
@@ -285,7 +313,7 @@ void Compile (const char* FileName)
     };
 
     /* Add macros that are always defined */
-    DefineNumericMacro ("__CC65__", VERSION);
+    DefineNumericMacro ("__CC65__", GetVersionAsNumber ());
 
     /* Language standard that is supported */
     DefineNumericMacro ("__CC65_STD_C89__", STD_C89);
@@ -299,55 +327,71 @@ void Compile (const char* FileName)
      */
     if (IS_Get (&Optimize)) {
         long CodeSize = IS_Get (&CodeSizeFactor);
-       DefineNumericMacro ("__OPT__", 1);
+       DefineNumericMacro ("__OPT__", 1);
                if (CodeSize > 100) {
-           DefineNumericMacro ("__OPT_i__", CodeSize);
-       }
-       if (IS_Get (&EnableRegVars)) {
-           DefineNumericMacro ("__OPT_r__", 1);
-       }
+           DefineNumericMacro ("__OPT_i__", CodeSize);
+       }
+       if (IS_Get (&EnableRegVars)) {
+           DefineNumericMacro ("__OPT_r__", 1);
+       }
                if (IS_Get (&InlineStdFuncs)) {
-           DefineNumericMacro ("__OPT_s__", 1);
-       }
+           DefineNumericMacro ("__OPT_s__", 1);
+       }
     }
 
     /* __TIME__ and __DATE__ macros */
     Time = time (0);
     TM   = localtime (&Time);
-    strftime (Buf, sizeof (Buf), "%e %Y", TM);
-    xsprintf (DateStr, sizeof (DateStr), "\"%s %s\"", MonthNames[TM->tm_mon], Buf);
+    xsprintf (DateStr, sizeof (DateStr), "\"%s %2d %d\"",
+              MonthNames[TM->tm_mon], TM->tm_mday, TM->tm_year + 1900);
     strftime (TimeStr, sizeof (TimeStr), "\"%H:%M:%S\"", TM);
     DefineTextMacro ("__DATE__", DateStr);
     DefineTextMacro ("__TIME__", TimeStr);
 
-    /* Initialize the literal pool */
-    InitLiteralPool ();
+    /* Other standard macros */
+    /* DefineNumericMacro ("__STDC__", 1);      <- not now */
+    DefineNumericMacro ("__STDC_HOSTED__", 1);
 
     /* Create the base lexical level */
     EnterGlobalLevel ();
 
+    /* Create the global code and data segments */
+    CreateGlobalSegments ();
+
+    /* Initialize the literal pool */
+    InitLiteralPool ();
+
     /* Generate the code generator preamble */
     g_preamble ();
 
     /* Open the input file */
     OpenMainFile (FileName);
 
-    /* Ok, start the ball rolling... */
-    Parse ();
+    /* Are we supposed to compile or just preprocess the input? */
+    if (PreprocessOnly) {
 
-    /* Dump the literal pool. */
-    DumpLiteralPool ();
+       /* Open the file */
+        OpenOutputFile ();
 
-    /* Write imported/exported symbols */
-    EmitExternals ();
+        /* Preprocess each line and write it to the output file */
+        while (NextLine ()) {
+            Preprocess ();
+            WriteOutput ("%.*s\n", (int) SB_GetLen (Line), SB_GetConstBuf (Line));
+        }
+
+        /* Close the output file */
+        CloseOutputFile ();
+
+    } else {
+
+        /* Ok, start the ball rolling... */
+        Parse ();
 
-    if (Debug) {
-       PrintLiteralPoolStats (stdout);
-       PrintMacroStats (stdout);
     }
 
-    /* Leave the main lexical level */
-    LeaveGlobalLevel ();
+    if (Debug) {
+        PrintMacroStats (stdout);
+    }
 
     /* Print an error report */
     ErrorReport ();
@@ -355,4 +399,37 @@ void Compile (const char* FileName)
 
 
 
+void FinishCompile (void)
+/* Emit literals, externals, debug info, do cleanup and optimizations */
+{
+    SymTable* SymTab;
+    SymEntry* Func;
+
+    /* Walk over all functions, doing cleanup, optimizations ... */
+    SymTab = GetGlobalSymTab ();
+    Func   = SymTab->SymHead;
+    while (Func) {
+        if (SymIsOutputFunc (Func)) {
+            /* Function which is defined and referenced or extern */
+            MoveLiteralPool (Func->V.F.LitPool);
+            CS_MergeLabels (Func->V.F.Seg->Code);
+            RunOpt (Func->V.F.Seg->Code);
+        }
+        Func = Func->NextSym;
+    }
+
+    /* Output the literal pool */
+    OutputLiteralPool ();
+
+    /* Emit debug infos if enabled */
+    EmitDebugInfo ();
+
+    /* Write imported/exported symbols */
+    EmitExternals ();
+
+    /* Leave the main lexical level */
+    LeaveGlobalLevel ();
+}
+
+