]> git.sur5r.net Git - cc65/blobdiff - src/cc65/codegen.c
remove TABs
[cc65] / src / cc65 / codegen.c
index dc0fd520d0b282167b8fd674cc4fe8c8c17ac482..a611f4f6a809ec0789c5c975bfc7876104e10b8e 100644 (file)
@@ -1,15 +1,15 @@
 /*****************************************************************************/
 /*                                                                           */
-/*                                 codegen.c                                */
+/*                                 codegen.c                                 */
 /*                                                                           */
-/*                            6502 code generator                           */
+/*                            6502 code generator                            */
 /*                                                                           */
 /*                                                                           */
 /*                                                                           */
-/* (C) 1998-2004 Ullrich von Bassewitz                                       */
-/*               Römerstraße 52                                              */
-/*               D-70794 Filderstadt                                         */
-/* EMail:        uz@cc65.org                                                 */
+/* (C) 1998-2013, Ullrich von Bassewitz                                      */
+/*                Roemerstrasse 52                                           */
+/*                D-70794 Filderstadt                                        */
+/* EMail:         uz@cc65.org                                                */
 /*                                                                           */
 /*                                                                           */
 /* This software is provided 'as-is', without any expressed or implied       */
 /* common */
 #include "check.h"
 #include "cpu.h"
+#include "inttypes.h"
 #include "strbuf.h"
-#include "version.h"
 #include "xmalloc.h"
 #include "xsprintf.h"
+#include "version.h"
 
 /* cc65 */
 #include "asmcode.h"
@@ -55,6 +56,7 @@
 #include "global.h"
 #include "segments.h"
 #include "stackptr.h"
+#include "stdfunc.h"
 #include "textseg.h"
 #include "util.h"
 #include "codegen.h"
@@ -62,7 +64,7 @@
 
 
 /*****************************************************************************/
-/*                                  Helpers                                 */
+/*                                  Helpers                                  */
 /*****************************************************************************/
 
 
 static void typeerror (unsigned type)
 /* Print an error message about an invalid operand type */
 {
-    Internal ("Invalid type in CF flags: %04X, type = %u", type, type & CF_TYPE);
+    /* Special handling for floats here: */
+    if ((type & CF_TYPEMASK) == CF_FLOAT) {
+        Fatal ("Floating point type is currently unsupported");
+    } else {
+        Internal ("Invalid type in CF flags: %04X, type = %u", type, type & CF_TYPEMASK);
+    }
 }
 
 
@@ -86,7 +93,7 @@ static void CheckLocalOffs (unsigned Offs)
 
 
 
-static const char* GetLabelName (unsigned Flags, unsigned long Label, long Offs)
+static const char* GetLabelName (unsigned Flags, uintptr_t Label, long Offs)
 {
     static char Buf [256];              /* Label name */
 
@@ -98,22 +105,22 @@ static const char* GetLabelName (unsigned Flags, unsigned long Label, long Offs)
             if (Offs) {
                 xsprintf (Buf, sizeof (Buf), "%s%+ld", LocalLabelName (Label), Offs);
             } else {
-                       xsprintf (Buf, sizeof (Buf), "%s", LocalLabelName (Label));
+                xsprintf (Buf, sizeof (Buf), "%s", LocalLabelName (Label));
             }
             break;
 
         case CF_EXTERNAL:
             /* External label */
             if (Offs) {
-               xsprintf (Buf, sizeof (Buf), "_%s%+ld", (char*) Label, Offs);
+                xsprintf (Buf, sizeof (Buf), "_%s%+ld", (char*) Label, Offs);
             } else {
-               xsprintf (Buf, sizeof (Buf), "_%s", (char*) Label);
+                xsprintf (Buf, sizeof (Buf), "_%s", (char*) Label);
             }
             break;
 
         case CF_ABSOLUTE:
             /* Absolute address */
-            xsprintf (Buf, sizeof (Buf), "$%04X", (int)((Label+Offs) & 0xFFFF));
+            xsprintf (Buf, sizeof (Buf), "$%04X", (unsigned)((Label+Offs) & 0xFFFF));
             break;
 
         case CF_REGVAR:
@@ -132,7 +139,7 @@ static const char* GetLabelName (unsigned Flags, unsigned long Label, long Offs)
 
 
 /*****************************************************************************/
-/*                           Pre- and postamble                             */
+/*                            Pre- and postamble                             */
 /*****************************************************************************/
 
 
@@ -140,19 +147,14 @@ static const char* GetLabelName (unsigned Flags, unsigned long Label, long Offs)
 void g_preamble (void)
 /* Generate the assembler code preamble */
 {
-    /* Create a new (global) segment list and remember it */
-    PushSegments (0);
-    GS = CS;
-
     /* Identify the compiler version */
     AddTextLine (";");
-    AddTextLine ("; File generated by cc65 v %u.%u.%u",
-                VER_MAJOR, VER_MINOR, VER_PATCH);
+    AddTextLine ("; File generated by cc65 v %s", GetVersionAsString ());
     AddTextLine (";");
 
     /* Insert some object file options */
-    AddTextLine ("\t.fopt\t\tcompiler,\"cc65 v %u.%u.%u\"",
-                   VER_MAJOR, VER_MINOR, VER_PATCH);
+    AddTextLine ("\t.fopt\t\tcompiler,\"cc65 v %s\"",
+                 GetVersionAsString ());
 
     /* If we're producing code for some other CPU, switch the command set */
     switch (CPU) {
@@ -177,8 +179,9 @@ void g_preamble (void)
     /* Tell the assembler if we want to generate debug info */
     AddTextLine ("\t.debuginfo\t%s", (DebugInfo != 0)? "on" : "off");
 
-    /* Import the stack pointer for direct auto variable access */
-    AddTextLine ("\t.importzp\tsp, sreg, regsave, regbank, tmp1, ptr1, ptr2");
+    /* Import zero page variables */
+    AddTextLine ("\t.importzp\tsp, sreg, regsave, regbank");
+    AddTextLine ("\t.importzp\ttmp1, tmp2, tmp3, tmp4, ptr1, ptr2, ptr3, ptr4");
 
     /* Define long branch macros */
     AddTextLine ("\t.macpack\tlongbranch");
@@ -191,8 +194,8 @@ void g_fileinfo (const char* Name, unsigned long Size, unsigned long MTime)
 {
     if (DebugInfo) {
         /* We have to place this into the global text segment, so it will
-         * appear before all .dbg line statements.
-         */
+        ** appear before all .dbg line statements.
+        */
         TS_AddLine (GS->Text, "\t.dbg\t\tfile, \"%s\", %lu, %lu", Name, Size, MTime);
     }
 }
@@ -200,7 +203,7 @@ void g_fileinfo (const char* Name, unsigned long Size, unsigned long MTime)
 
 
 /*****************************************************************************/
-/*                             Segment support                              */
+/*                              Segment support                              */
 /*****************************************************************************/
 
 
@@ -248,7 +251,7 @@ void g_segname (segment_t Seg)
 
 
 /*****************************************************************************/
-/*                                          Code                                    */
+/*                                   Code                                    */
 /*****************************************************************************/
 
 
@@ -256,7 +259,7 @@ void g_segname (segment_t Seg)
 unsigned sizeofarg (unsigned flags)
 /* Return the size of a function argument type that is encoded in flags */
 {
-    switch (flags & CF_TYPE) {
+    switch (flags & CF_TYPEMASK) {
 
         case CF_CHAR:
             return (flags & CF_FORCECHAR)? 1 : 2;
@@ -297,15 +300,15 @@ int push (unsigned flags)
 
 static unsigned MakeByteOffs (unsigned Flags, unsigned Offs)
 /* The value in Offs is an offset to an address in a/x. Make sure, an object
- * of the type given in Flags can be loaded or stored into this address by
- * adding part of the offset to the address in ax, so that the remaining
- * offset fits into an index register. Return the remaining offset.
- */
+** of the type given in Flags can be loaded or stored into this address by
+** adding part of the offset to the address in ax, so that the remaining
+** offset fits into an index register. Return the remaining offset.
+*/
 {
     /* If the offset is too large for a byte register, add the high byte
-     * of the offset to the primary. Beware: We need a special correction
-     * if the offset in the low byte will overflow in the operation.
-     */
+    ** of the offset to the primary. Beware: We need a special correction
+    ** if the offset in the low byte will overflow in the operation.
+    */
     unsigned O = Offs & ~0xFFU;
     if ((Offs & 0xFF) > 256 - sizeofarg (Flags)) {
         /* We need to add the low byte also */
@@ -325,7 +328,7 @@ static unsigned MakeByteOffs (unsigned Flags, unsigned Offs)
 
 
 /*****************************************************************************/
-/*                     Functions handling local labels                      */
+/*                      Functions handling local labels                      */
 /*****************************************************************************/
 
 
@@ -346,8 +349,26 @@ void g_defdatalabel (unsigned label)
 
 
 
+void g_aliasdatalabel (unsigned label, unsigned baselabel, long offs)
+/* Define label as a local alias for baselabel+offs */
+{
+    /* We need an intermediate buffer here since LocalLabelName uses a
+    ** static buffer which changes with each call.
+    */
+    StrBuf L = AUTO_STRBUF_INITIALIZER;
+    SB_AppendStr (&L, LocalLabelName (label));
+    SB_Terminate (&L);
+    AddDataLine ("%s\t:=\t%s+%ld",
+                 SB_GetConstBuf (&L),
+                 LocalLabelName (baselabel),
+                 offs);
+    SB_Done (&L);
+}
+
+
+
 /*****************************************************************************/
-/*                    Functions handling global labels                      */
+/*                     Functions handling global labels                      */
 /*****************************************************************************/
 
 
@@ -386,11 +407,9 @@ void g_defimport (const char* Name, int ZP)
 
 
 void g_importstartup (void)
-/* Forced import of the startup segment */
+/* Forced import of the startup module */
 {
-#if 0
-    AddTextLine ("\t.forceimport\t__STARTUP_RUN__");
-#endif
+    AddTextLine ("\t.forceimport\t__STARTUP__");
 }
 
 
@@ -404,46 +423,16 @@ void g_importmainargs (void)
 
 
 /*****************************************************************************/
-/*                   Load functions for various registers                   */
-/*****************************************************************************/
-
-
-
-static void ldaconst (unsigned val)
-/* Load a with a constant */
-{
-    AddCodeLine ("lda #$%02X", val & 0xFF);
-}
-
-
-
-static void ldxconst (unsigned val)
-/* Load x with a constant */
-{
-    AddCodeLine ("ldx #$%02X", val & 0xFF);
-}
-
-
-
-static void ldyconst (unsigned val)
-/* Load y with a constant */
-{
-    AddCodeLine ("ldy #$%02X", val & 0xFF);
-}
-
-
-
-/*****************************************************************************/
-/*                          Function entry and exit                         */
+/*                          Function entry and exit                          */
 /*****************************************************************************/
 
 
 
 /* Remember the argument size of a function. The variable is set by g_enter
- * and used by g_leave. If the functions gets its argument size by the caller
- * (variable param list or function without prototype), g_enter will set the
- * value to -1.
- */
+** and used by g_leave. If the function gets its argument size by the caller
+** (variable param list or function without prototype), g_enter will set the
+** value to -1.
+*/
 static int funcargs;
 
 
@@ -465,39 +454,30 @@ void g_leave (void)
 /* Function epilogue */
 {
     /* How many bytes of locals do we have to drop? */
-    unsigned k = (unsigned) -StackPtr;
+    unsigned ToDrop = (unsigned) -StackPtr;
 
     /* If we didn't have a variable argument list, don't call leave */
     if (funcargs >= 0) {
 
-        /* Drop stackframe if needed. We can only drop 255 bytes at a time. */
-        k += funcargs;
-        while (k > 0) {
-            unsigned ToDrop = (k > 255)? 255 : k;
-            if (ToDrop <= 8) {
-                AddCodeLine ("jsr incsp%d", k);
-            } else {
-                       ldyconst (ToDrop);
-                       AddCodeLine ("jsr addysp");
-            }
-            k -= ToDrop;
-        }
+        /* Drop stackframe if needed */
+        g_drop (ToDrop + funcargs);
 
-    } else {
+    } else if (StackPtr != 0) {
 
-        if (k == 0) {
-            /* Nothing to drop */
+        /* We've a stack frame to drop */
+        if (ToDrop > 255) {
+            g_drop (ToDrop);            /* Inlines the code */
             AddCodeLine ("jsr leave");
         } else {
-            /* We've a stack frame to drop */
-            while (k > 255) {
-                ldyconst (255);
-                AddCodeLine ("jsr addysp");
-                k -= 255;
-            }
-            ldyconst (k);
+            AddCodeLine ("ldy #$%02X", ToDrop);
             AddCodeLine ("jsr leavey");
         }
+
+    } else {
+
+        /* Nothing to drop */
+        AddCodeLine ("jsr leave");
+
     }
 
     /* Add the final rts */
@@ -507,7 +487,7 @@ void g_leave (void)
 
 
 /*****************************************************************************/
-/*                                   Register variables                             */
+/*                            Register variables                             */
 /*****************************************************************************/
 
 
@@ -520,14 +500,13 @@ void g_swap_regvars (int StackOffs, int RegOffs, unsigned Bytes)
     CheckLocalOffs (StackOffs);
 
     /* Generate code */
+    AddCodeLine ("ldy #$%02X", StackOffs & 0xFF);
     if (Bytes == 1) {
 
         if (IS_Get (&CodeSizeFactor) < 165) {
-            ldyconst (StackOffs);
-            ldxconst (RegOffs);
+            AddCodeLine ("ldx #$%02X", RegOffs & 0xFF);
             AddCodeLine ("jsr regswap1");
         } else {
-            ldyconst (StackOffs);
             AddCodeLine ("lda (sp),y");
             AddCodeLine ("ldx regbank%+d", RegOffs);
             AddCodeLine ("sta regbank%+d", RegOffs);
@@ -537,15 +516,13 @@ void g_swap_regvars (int StackOffs, int RegOffs, unsigned Bytes)
 
     } else if (Bytes == 2) {
 
-        ldyconst (StackOffs);
-        ldxconst (RegOffs);
+        AddCodeLine ("ldx #$%02X", RegOffs & 0xFF);
         AddCodeLine ("jsr regswap2");
 
     } else {
 
-        ldyconst (StackOffs);
-        ldxconst (RegOffs);
-        ldaconst (Bytes);
+        AddCodeLine ("ldx #$%02X", RegOffs & 0xFF);
+        AddCodeLine ("lda #$%02X", Bytes & 0xFF);
         AddCodeLine ("jsr regswap");
     }
 }
@@ -572,8 +549,8 @@ void g_save_regvars (int RegOffs, unsigned Bytes)
         /* More than two bytes - loop */
         unsigned Label = GetLocalLabel ();
         g_space (Bytes);
-        ldyconst (Bytes - 1);
-        ldxconst (Bytes);
+        AddCodeLine ("ldy #$%02X", (unsigned char) (Bytes - 1));
+        AddCodeLine ("ldx #$%02X", (unsigned char) Bytes);
         g_defcodelabel (Label);
         AddCodeLine ("lda regbank%+d,x", RegOffs-1);
         AddCodeLine ("sta (sp),y");
@@ -599,13 +576,13 @@ void g_restore_regvars (int StackOffs, int RegOffs, unsigned Bytes)
     /* Don't loop for up to two bytes */
     if (Bytes == 1) {
 
-        ldyconst (StackOffs);
+        AddCodeLine ("ldy #$%02X", StackOffs);
         AddCodeLine ("lda (sp),y");
         AddCodeLine ("sta regbank%+d", RegOffs);
 
     } else if (Bytes == 2) {
 
-        ldyconst (StackOffs);
+        AddCodeLine ("ldy #$%02X", StackOffs);
         AddCodeLine ("lda (sp),y");
         AddCodeLine ("sta regbank%+d", RegOffs);
         AddCodeLine ("iny");
@@ -614,7 +591,7 @@ void g_restore_regvars (int StackOffs, int RegOffs, unsigned Bytes)
 
     } else if (Bytes == 3 && IS_Get (&CodeSizeFactor) >= 133) {
 
-        ldyconst (StackOffs);
+        AddCodeLine ("ldy #$%02X", StackOffs);
         AddCodeLine ("lda (sp),y");
         AddCodeLine ("sta regbank%+d", RegOffs);
         AddCodeLine ("iny");
@@ -627,11 +604,11 @@ void g_restore_regvars (int StackOffs, int RegOffs, unsigned Bytes)
     } else if (StackOffs <= RegOffs) {
 
         /* More bytes, but the relation between the register offset in the
-         * register bank and the stack offset allows us to generate short
-         * code that uses just one index register.
-         */
+        ** register bank and the stack offset allows us to generate short
+        ** code that uses just one index register.
+        */
         unsigned Label = GetLocalLabel ();
-        ldyconst (StackOffs);
+        AddCodeLine ("ldy #$%02X", StackOffs);
         g_defcodelabel (Label);
         AddCodeLine ("lda (sp),y");
         AddCodeLine ("sta regbank%+d,y", RegOffs - StackOffs);
@@ -641,13 +618,13 @@ void g_restore_regvars (int StackOffs, int RegOffs, unsigned Bytes)
 
     } else {
 
-        /* Ok, this is the generic code. We need to save X because the
-         * caller will only save A.
-         */
+        /* OK, this is the generic code. We need to save X because the
+        ** caller will only save A.
+        */
         unsigned Label = GetLocalLabel ();
         AddCodeLine ("stx tmp1");
-        ldyconst (StackOffs + Bytes - 1);
-        ldxconst (Bytes - 1);
+        AddCodeLine ("ldy #$%02X", (unsigned char) (StackOffs + Bytes - 1));
+        AddCodeLine ("ldx #$%02X", (unsigned char) (Bytes - 1));
         g_defcodelabel (Label);
         AddCodeLine ("lda (sp),y");
         AddCodeLine ("sta regbank%+d,x", RegOffs);
@@ -662,7 +639,7 @@ void g_restore_regvars (int StackOffs, int RegOffs, unsigned Bytes)
 
 
 /*****************************************************************************/
-/*                           Fetching memory cells                          */
+/*                           Fetching memory cells                           */
 /*****************************************************************************/
 
 
@@ -677,17 +654,17 @@ void g_getimmed (unsigned Flags, unsigned long Val, long Offs)
     if ((Flags & CF_CONST) != 0) {
 
         /* Numeric constant */
-        switch (Flags & CF_TYPE) {
+        switch (Flags & CF_TYPEMASK) {
 
             case CF_CHAR:
                 if ((Flags & CF_FORCECHAR) != 0) {
-                    ldaconst (Val);
+                    AddCodeLine ("lda #$%02X", (unsigned char) Val);
                     break;
                 }
                 /* FALL THROUGH */
             case CF_INT:
-                ldxconst ((Val >> 8) & 0xFF);
-                ldaconst (Val & 0xFF);
+                AddCodeLine ("ldx #$%02X", (unsigned char) (Val >> 8));
+                AddCodeLine ("lda #$%02X", (unsigned char) Val);
                 break;
 
             case CF_LONG:
@@ -711,7 +688,7 @@ void g_getimmed (unsigned Flags, unsigned long Val, long Offs)
                     AddCodeLine ("stx sreg+1");
                     Done |= 0x08;
                 }
-               if ((Done & 0x04) == 0 && B1 != B3) {
+                if ((Done & 0x04) == 0 && B1 != B3) {
                     AddCodeLine ("lda #$%02X", B3);
                     AddCodeLine ("sta sreg");
                     Done |= 0x04;
@@ -753,28 +730,28 @@ void g_getimmed (unsigned Flags, unsigned long Val, long Offs)
 
 
 
-void g_getstatic (unsigned flags, unsigned long label, long offs)
+void g_getstatic (unsigned flags, uintptr_t label, long offs)
 /* Fetch an static memory cell into the primary register */
 {
     /* Create the correct label name */
     const char* lbuf = GetLabelName (flags, label, offs);
 
     /* Check the size and generate the correct load operation */
-    switch (flags & CF_TYPE) {
+    switch (flags & CF_TYPEMASK) {
 
         case CF_CHAR:
             if ((flags & CF_FORCECHAR) || (flags & CF_TEST)) {
-                AddCodeLine ("lda %s", lbuf);  /* load A from the label */
+                AddCodeLine ("lda %s", lbuf);   /* load A from the label */
             } else {
-               ldxconst (0);
-               AddCodeLine ("lda %s", lbuf);   /* load A from the label */
-               if (!(flags & CF_UNSIGNED)) {
-                   /* Must sign extend */
+                AddCodeLine ("ldx #$00");
+                AddCodeLine ("lda %s", lbuf);   /* load A from the label */
+                if (!(flags & CF_UNSIGNED)) {
+                    /* Must sign extend */
                     unsigned L = GetLocalLabel ();
-                   AddCodeLine ("bpl %s", LocalLabelName (L));
-                   AddCodeLine ("dex");
+                    AddCodeLine ("bpl %s", LocalLabelName (L));
+                    AddCodeLine ("dex");
                     g_defcodelabel (L);
-               }
+                }
             }
             break;
 
@@ -789,13 +766,13 @@ void g_getstatic (unsigned flags, unsigned long label, long offs)
 
         case CF_LONG:
             if (flags & CF_TEST) {
-               AddCodeLine ("lda %s+3", lbuf);
+                AddCodeLine ("lda %s+3", lbuf);
                 AddCodeLine ("ora %s+2", lbuf);
                 AddCodeLine ("ora %s+1", lbuf);
                 AddCodeLine ("ora %s+0", lbuf);
             } else {
-               AddCodeLine ("lda %s+3", lbuf);
-               AddCodeLine ("sta sreg+1");
+                AddCodeLine ("lda %s+3", lbuf);
+                AddCodeLine ("sta sreg+1");
                 AddCodeLine ("lda %s+2", lbuf);
                 AddCodeLine ("sta sreg");
                 AddCodeLine ("ldx %s+1", lbuf);
@@ -811,145 +788,171 @@ void g_getstatic (unsigned flags, unsigned long label, long offs)
 
 
 
-void g_getlocal (unsigned flags, int offs)
+void g_getlocal (unsigned Flags, int Offs)
 /* Fetch specified local object (local var). */
 {
-    offs -= StackPtr;
-    CheckLocalOffs (offs);
-    switch (flags & CF_TYPE) {
+    Offs -= StackPtr;
+    switch (Flags & CF_TYPEMASK) {
 
         case CF_CHAR:
-            if ((flags & CF_FORCECHAR) || (flags & CF_TEST)) {
-                ldyconst (offs);
+            CheckLocalOffs (Offs);
+            if ((Flags & CF_FORCECHAR) || (Flags & CF_TEST)) {
+                AddCodeLine ("ldy #$%02X", Offs);
                 AddCodeLine ("lda (sp),y");
             } else {
-                ldyconst (offs);
+                AddCodeLine ("ldy #$%02X", Offs);
                 AddCodeLine ("ldx #$00");
                 AddCodeLine ("lda (sp),y");
-               if ((flags & CF_UNSIGNED) == 0) {
+                if ((Flags & CF_UNSIGNED) == 0) {
                     unsigned L = GetLocalLabel();
-                   AddCodeLine ("bpl %s", LocalLabelName (L));
-                   AddCodeLine ("dex");
+                    AddCodeLine ("bpl %s", LocalLabelName (L));
+                    AddCodeLine ("dex");
                     g_defcodelabel (L);
-               }
+                }
             }
             break;
 
         case CF_INT:
-            CheckLocalOffs (offs + 1);
-            if (flags & CF_TEST) {
-               ldyconst (offs + 1);
-               AddCodeLine ("lda (sp),y");
+            CheckLocalOffs (Offs + 1);
+            AddCodeLine ("ldy #$%02X", (unsigned char) (Offs+1));
+            if (Flags & CF_TEST) {
+                AddCodeLine ("lda (sp),y");
                 AddCodeLine ("dey");
                 AddCodeLine ("ora (sp),y");
             } else {
-                ldyconst (offs+1);
                 AddCodeLine ("jsr ldaxysp");
             }
             break;
 
         case CF_LONG:
-            ldyconst (offs+3);
+            CheckLocalOffs (Offs + 3);
+            AddCodeLine ("ldy #$%02X", (unsigned char) (Offs+3));
             AddCodeLine ("jsr ldeaxysp");
-            if (flags & CF_TEST) {
-               g_test (flags);
+            if (Flags & CF_TEST) {
+                g_test (Flags);
             }
             break;
 
         default:
-            typeerror (flags);
+            typeerror (Flags);
     }
 }
 
 
 
-void g_getind (unsigned flags, unsigned offs)
+void g_getind (unsigned Flags, unsigned Offs)
 /* Fetch the specified object type indirect through the primary register
- * into the primary register
- */
+** into the primary register
+*/
 {
     /* If the offset is greater than 255, add the part that is > 255 to
-     * the primary. This way we get an easy addition and use the low byte
-     * as the offset
-     */
-    offs = MakeByteOffs (flags, offs);
+    ** the primary. This way we get an easy addition and use the low byte
+    ** as the offset
+    */
+    Offs = MakeByteOffs (Flags, Offs);
 
     /* Handle the indirect fetch */
-    switch (flags & CF_TYPE) {
+    switch (Flags & CF_TYPEMASK) {
 
         case CF_CHAR:
             /* Character sized */
-            if (flags & CF_UNSIGNED) {
-                ldyconst (offs);
+            AddCodeLine ("ldy #$%02X", Offs);
+            if (Flags & CF_UNSIGNED) {
                 AddCodeLine ("jsr ldauidx");
             } else {
-                ldyconst (offs);
                 AddCodeLine ("jsr ldaidx");
             }
             break;
 
         case CF_INT:
-            if (flags & CF_TEST) {
-               ldyconst (offs);
+            if (Flags & CF_TEST) {
+                AddCodeLine ("ldy #$%02X", Offs);
                 AddCodeLine ("sta ptr1");
                 AddCodeLine ("stx ptr1+1");
                 AddCodeLine ("lda (ptr1),y");
                 AddCodeLine ("iny");
                 AddCodeLine ("ora (ptr1),y");
             } else {
-                ldyconst (offs+1);
+                AddCodeLine ("ldy #$%02X", Offs+1);
                 AddCodeLine ("jsr ldaxidx");
             }
             break;
 
         case CF_LONG:
-            ldyconst (offs+3);
+            AddCodeLine ("ldy #$%02X", Offs+3);
             AddCodeLine ("jsr ldeaxidx");
-            if (flags & CF_TEST) {
-                g_test (flags);
+            if (Flags & CF_TEST) {
+                g_test (Flags);
             }
             break;
 
         default:
-            typeerror (flags);
+            typeerror (Flags);
 
     }
 }
 
 
 
-void g_leasp (int offs)
+void g_leasp (int Offs)
 /* Fetch the address of the specified symbol into the primary register */
 {
+    unsigned char Lo, Hi;
+
     /* Calculate the offset relative to sp */
-    offs -= StackPtr;
+    Offs -= StackPtr;
 
-    /* For value 0 we do direct code */
-    if (offs == 0) {
-        AddCodeLine ("lda sp");
-        AddCodeLine ("ldx sp+1");
-    } else {
-        if (IS_Get (&CodeSizeFactor) < 300) {
-            ldaconst (offs);                   /* Load A with offset value */
-            AddCodeLine ("jsr leaasp");        /* Load effective address */
+    /* Get low and high byte */
+    Lo = (unsigned char) Offs;
+    Hi = (unsigned char) (Offs >> 8);
+
+    /* Generate code */
+    if (Lo == 0) {
+        if (Hi <= 3) {
+            AddCodeLine ("lda sp");
+            AddCodeLine ("ldx sp+1");
+            while (Hi--) {
+                AddCodeLine ("inx");
+            }
+        } else {
+            AddCodeLine ("lda sp+1");
+            AddCodeLine ("clc");
+            AddCodeLine ("adc #$%02X", Hi);
+            AddCodeLine ("tax");
+            AddCodeLine ("lda sp");
+        }
+    } else if (Hi == 0) {
+        /* 8 bit offset */
+        if (IS_Get (&CodeSizeFactor) < 200) {
+            /* 8 bit offset with subroutine call */
+            AddCodeLine ("lda #$%02X", Lo);
+            AddCodeLine ("jsr leaa0sp");
         } else {
+            /* 8 bit offset inlined */
             unsigned L = GetLocalLabel ();
-            if ((CPUIsets[CPU] & CPU_ISET_65SC02) != 0 && offs == 1) {
-               AddCodeLine ("lda sp");
-               AddCodeLine ("ldx sp+1");
-               AddCodeLine ("ina");
-               AddCodeLine ("bne %s", LocalLabelName (L));
-               AddCodeLine ("inx");
-            } else {
-               ldaconst (offs);
-               AddCodeLine ("clc");
-               AddCodeLine ("ldx sp+1");
-               AddCodeLine ("adc sp");
-               AddCodeLine ("bcc %s", LocalLabelName (L));
-               AddCodeLine ("inx");
-            }
+            AddCodeLine ("lda sp");
+            AddCodeLine ("ldx sp+1");
+            AddCodeLine ("clc");
+            AddCodeLine ("adc #$%02X", Lo);
+            AddCodeLine ("bcc %s", LocalLabelName (L));
+            AddCodeLine ("inx");
             g_defcodelabel (L);
         }
+    } else if (IS_Get (&CodeSizeFactor) < 170) {
+        /* Full 16 bit offset with subroutine call */
+        AddCodeLine ("lda #$%02X", Lo);
+        AddCodeLine ("ldx #$%02X", Hi);
+        AddCodeLine ("jsr leaaxsp");
+    } else {
+        /* Full 16 bit offset inlined */
+        AddCodeLine ("lda sp");
+        AddCodeLine ("clc");
+        AddCodeLine ("adc #$%02X", Lo);
+        AddCodeLine ("pha");
+        AddCodeLine ("lda sp+1");
+        AddCodeLine ("adc #$%02X", Hi);
+        AddCodeLine ("tax");
+        AddCodeLine ("pla");
     }
 }
 
@@ -957,8 +960,8 @@ void g_leasp (int offs)
 
 void g_leavariadic (int Offs)
 /* Fetch the address of a parameter in a variadic function into the primary
- * register
- */
+** register
+*/
 {
     unsigned ArgSizeOffs;
 
@@ -966,14 +969,14 @@ void g_leavariadic (int Offs)
     Offs -= StackPtr;
 
     /* Get the offset of the parameter which is stored at sp+0 on function
-     * entry and check if this offset is reachable with a byte offset.
-     */
+    ** entry and check if this offset is reachable with a byte offset.
+    */
     CHECK (StackPtr <= 0);
     ArgSizeOffs = -StackPtr;
     CheckLocalOffs (ArgSizeOffs);
 
     /* Get the size of all parameters. */
-    ldyconst (ArgSizeOffs);
+    AddCodeLine ("ldy #$%02X", ArgSizeOffs);
     AddCodeLine ("lda (sp),y");
 
     /* Add the value of the stackpointer */
@@ -986,7 +989,8 @@ void g_leavariadic (int Offs)
         AddCodeLine ("inx");
         g_defcodelabel (L);
     } else {
-        AddCodeLine ("jsr leaasp");
+        AddCodeLine ("ldx #$00");
+        AddCodeLine ("jsr leaaxsp");
     }
 
     /* Add the offset to the primary */
@@ -1000,19 +1004,19 @@ void g_leavariadic (int Offs)
 
 
 /*****************************************************************************/
-/*                            Store into memory                             */
+/*                             Store into memory                             */
 /*****************************************************************************/
 
 
 
-void g_putstatic (unsigned flags, unsigned long label, long offs)
+void g_putstatic (unsigned flags, uintptr_t label, long offs)
 /* Store the primary register into the specified static memory cell */
 {
     /* Create the correct label name */
     const char* lbuf = GetLabelName (flags, label, offs);
 
     /* Check the size and generate the correct store operation */
-    switch (flags & CF_TYPE) {
+    switch (flags & CF_TYPEMASK) {
 
         case CF_CHAR:
             AddCodeLine ("sta %s", lbuf);
@@ -1045,19 +1049,19 @@ void g_putlocal (unsigned Flags, int Offs, long Val)
 {
     Offs -= StackPtr;
     CheckLocalOffs (Offs);
-    switch (Flags & CF_TYPE) {
+    switch (Flags & CF_TYPEMASK) {
 
         case CF_CHAR:
             if (Flags & CF_CONST) {
-               AddCodeLine ("lda #$%02X", (unsigned char) Val);
+                AddCodeLine ("lda #$%02X", (unsigned char) Val);
             }
-            ldyconst (Offs);
+            AddCodeLine ("ldy #$%02X", Offs);
             AddCodeLine ("sta (sp),y");
             break;
 
         case CF_INT:
             if (Flags & CF_CONST) {
-                ldyconst (Offs+1);
+                AddCodeLine ("ldy #$%02X", Offs+1);
                 AddCodeLine ("lda #$%02X", (unsigned char) (Val >> 8));
                 AddCodeLine ("sta (sp),y");
                 if ((Flags & CF_NOKEEP) == 0) {
@@ -1074,11 +1078,10 @@ void g_putlocal (unsigned Flags, int Offs, long Val)
                 }
                 AddCodeLine ("sta (sp),y");
             } else {
+                AddCodeLine ("ldy #$%02X", Offs);
                 if ((Flags & CF_NOKEEP) == 0 || IS_Get (&CodeSizeFactor) < 160) {
-                    ldyconst (Offs);
                     AddCodeLine ("jsr staxysp");
                 } else {
-                    ldyconst (Offs);
                     AddCodeLine ("sta (sp),y");
                     AddCodeLine ("iny");
                     AddCodeLine ("txa");
@@ -1089,9 +1092,9 @@ void g_putlocal (unsigned Flags, int Offs, long Val)
 
         case CF_LONG:
             if (Flags & CF_CONST) {
-               g_getimmed (Flags, Val, 0);
+                g_getimmed (Flags, Val, 0);
             }
-            ldyconst (Offs);
+            AddCodeLine ("ldy #$%02X", Offs);
             AddCodeLine ("jsr steaxysp");
             break;
 
@@ -1105,14 +1108,14 @@ void g_putlocal (unsigned Flags, int Offs, long Val)
 
 void g_putind (unsigned Flags, unsigned Offs)
 /* Store the specified object type in the primary register at the address
- * on the top of the stack
- */
+** on the top of the stack
+*/
 {
     /* We can handle offsets below $100 directly, larger offsets must be added
-     * to the address. Since a/x is in use, best code is achieved by adding
-     * just the high byte. Be sure to check if the low byte will overflow while
-     * while storing.
-     */
+    ** to the address. Since a/x is in use, best code is achieved by adding
+    ** just the high byte. Be sure to check if the low byte will overflow while
+    ** while storing.
+    */
     if ((Offs & 0xFF) > 256 - sizeofarg (Flags | CF_FORCECHAR)) {
 
         /* Overflow - we need to add the low byte also */
@@ -1147,20 +1150,18 @@ void g_putind (unsigned Flags, unsigned Offs)
     }
 
     /* Check the size and determine operation */
-    switch (Flags & CF_TYPE) {
+    AddCodeLine ("ldy #$%02X", Offs);
+    switch (Flags & CF_TYPEMASK) {
 
         case CF_CHAR:
-            ldyconst (Offs);
             AddCodeLine ("jsr staspidx");
             break;
 
         case CF_INT:
-            ldyconst (Offs);
             AddCodeLine ("jsr staxspidx");
             break;
 
         case CF_LONG:
-            ldyconst (Offs);
             AddCodeLine ("jsr steaxspidx");
             break;
 
@@ -1176,7 +1177,7 @@ void g_putind (unsigned Flags, unsigned Offs)
 
 
 /*****************************************************************************/
-/*                    type conversion and similiar stuff                    */
+/*                    type conversion and similiar stuff                     */
 /*****************************************************************************/
 
 
@@ -1184,7 +1185,7 @@ void g_putind (unsigned Flags, unsigned Offs)
 void g_toslong (unsigned flags)
 /* Make sure, the value on TOS is a long. Convert if necessary */
 {
-    switch (flags & CF_TYPE) {
+    switch (flags & CF_TYPEMASK) {
 
         case CF_CHAR:
         case CF_INT:
@@ -1209,7 +1210,7 @@ void g_toslong (unsigned flags)
 void g_tosint (unsigned flags)
 /* Make sure, the value on TOS is an int. Convert if necessary */
 {
-    switch (flags & CF_TYPE) {
+    switch (flags & CF_TYPEMASK) {
 
         case CF_CHAR:
         case CF_INT:
@@ -1227,26 +1228,34 @@ void g_tosint (unsigned flags)
 
 
 
-void g_regint (unsigned Flags)
-/* Make sure, the value in the primary register an int. Convert if necessary */
+static void g_regchar (unsigned Flags)
+/* Make sure, the value in the primary register is in the range of char. Truncate if necessary */
 {
     unsigned L;
 
-    switch (Flags & CF_TYPE) {
+    AddCodeLine ("ldx #$00");
+
+    if ((Flags & CF_UNSIGNED) == 0) {
+        /* Sign extend */
+        L = GetLocalLabel();
+        AddCodeLine ("cmp #$80");
+        AddCodeLine ("bcc %s", LocalLabelName (L));
+        AddCodeLine ("dex");
+        g_defcodelabel (L);
+    }
+}
+
+
+
+void g_regint (unsigned Flags)
+/* Make sure, the value in the primary register an int. Convert if necessary */
+{
+    switch (Flags & CF_TYPEMASK) {
 
         case CF_CHAR:
             if (Flags & CF_FORCECHAR) {
                 /* Conversion is from char */
-                if (Flags & CF_UNSIGNED) {
-                    AddCodeLine ("ldx #$00");
-                } else {
-                    L = GetLocalLabel();
-                    AddCodeLine ("ldx #$00");
-                    AddCodeLine ("cmp #$80");
-                    AddCodeLine ("bcc %s", LocalLabelName (L));
-                    AddCodeLine ("dex");
-                    g_defcodelabel (L);
-                }
+                g_regchar (Flags);
             }
             /* FALLTHROUGH */
 
@@ -1264,9 +1273,7 @@ void g_regint (unsigned Flags)
 void g_reglong (unsigned Flags)
 /* Make sure, the value in the primary register a long. Convert if necessary */
 {
-    unsigned L;
-
-    switch (Flags & CF_TYPE) {
+    switch (Flags & CF_TYPEMASK) {
 
         case CF_CHAR:
             if (Flags & CF_FORCECHAR) {
@@ -1281,12 +1288,7 @@ void g_reglong (unsigned Flags)
                     }
                 } else {
                     if (IS_Get (&CodeSizeFactor) >= 366) {
-                        L = GetLocalLabel();
-                        AddCodeLine ("ldx #$00");
-                        AddCodeLine ("cmp #$80");
-                        AddCodeLine ("bcc %s", LocalLabelName (L));
-                        AddCodeLine ("dex");
-                        g_defcodelabel (L);
+                        g_regchar (Flags);
                         AddCodeLine ("stx sreg");
                         AddCodeLine ("stx sreg+1");
                     } else {
@@ -1298,15 +1300,15 @@ void g_reglong (unsigned Flags)
 
         case CF_INT:
             if (Flags & CF_UNSIGNED) {
-               if (IS_Get (&CodeSizeFactor) >= 200) {
-                   ldyconst (0);
-                   AddCodeLine ("sty sreg");
-                   AddCodeLine ("sty sreg+1");
-               } else {
-                           AddCodeLine ("jsr axulong");
-               }
+                if (IS_Get (&CodeSizeFactor) >= 200) {
+                    AddCodeLine ("ldy #$00");
+                    AddCodeLine ("sty sreg");
+                    AddCodeLine ("sty sreg+1");
+                } else {
+                    AddCodeLine ("jsr axulong");
+                }
             } else {
-               AddCodeLine ("jsr axlong");
+                AddCodeLine ("jsr axlong");
             }
             break;
 
@@ -1322,23 +1324,23 @@ void g_reglong (unsigned Flags)
 
 unsigned g_typeadjust (unsigned lhs, unsigned rhs)
 /* Adjust the integer operands before doing a binary operation. lhs is a flags
- * value, that corresponds to the value on TOS, rhs corresponds to the value
- * in (e)ax. The return value is the the flags value for the resulting type.
- */
+** value, that corresponds to the value on TOS, rhs corresponds to the value
+** in (e)ax. The return value is the the flags value for the resulting type.
+*/
 {
     unsigned ltype, rtype;
     unsigned result;
 
     /* Get the type spec from the flags */
-    ltype = lhs & CF_TYPE;
-    rtype = rhs & CF_TYPE;
+    ltype = lhs & CF_TYPEMASK;
+    rtype = rhs & CF_TYPEMASK;
 
     /* Check if a conversion is needed */
     if (ltype == CF_LONG && rtype != CF_LONG && (rhs & CF_CONST) == 0) {
         /* We must promote the primary register to long */
         g_reglong (rhs);
         /* Get the new rhs type */
-        rhs = (rhs & ~CF_TYPE) | CF_LONG;
+        rhs = (rhs & ~CF_TYPEMASK) | CF_LONG;
         rtype = CF_LONG;
     } else if (ltype != CF_LONG && (lhs & CF_CONST) == 0 && rtype == CF_LONG) {
         /* We must promote the lhs to long */
@@ -1348,16 +1350,16 @@ unsigned g_typeadjust (unsigned lhs, unsigned rhs)
             g_toslong (lhs);
         }
         /* Get the new rhs type */
-        lhs = (lhs & ~CF_TYPE) | CF_LONG;
+        lhs = (lhs & ~CF_TYPEMASK) | CF_LONG;
         ltype = CF_LONG;
     }
 
     /* Determine the result type for the operation:
-     *  - The result is const if both operands are const.
-     *  - The result is unsigned if one of the operands is unsigned.
-     *  - The result is long if one of the operands is long.
-     *  - Otherwise the result is int sized.
-     */
+    **  - The result is const if both operands are const.
+    **  - The result is unsigned if one of the operands is unsigned.
+    **  - The result is long if one of the operands is long.
+    **  - Otherwise the result is int sized.
+    */
     result = (lhs & CF_CONST) & (rhs & CF_CONST);
     result |= (lhs & CF_UNSIGNED) | (rhs & CF_UNSIGNED);
     if (rtype == CF_LONG || ltype == CF_LONG) {
@@ -1372,38 +1374,45 @@ unsigned g_typeadjust (unsigned lhs, unsigned rhs)
 
 unsigned g_typecast (unsigned lhs, unsigned rhs)
 /* Cast the value in the primary register to the operand size that is flagged
- * by the lhs value. Return the result value.
- */
+** by the lhs value. Return the result value.
+*/
 {
-    unsigned ltype, rtype;
-
-    /* Get the type spec from the flags */
-    ltype = lhs & CF_TYPE;
-    rtype = rhs & CF_TYPE;
-
     /* Check if a conversion is needed */
     if ((rhs & CF_CONST) == 0) {
-        if (ltype == CF_LONG && rtype != CF_LONG) {
-            /* We must promote the primary register to long */
-            g_reglong (rhs);
-        } else if (ltype == CF_INT && rtype != CF_INT) {
-            /* We must promote the primary register to int */
-            g_regint (rhs);
+        switch (lhs & CF_TYPEMASK) {
+
+            case CF_LONG:
+                /* We must promote the primary register to long */
+                g_reglong (rhs);
+                break;
+
+            case CF_INT:
+                /* We must promote the primary register to int */
+                g_regint (rhs);
+                break;
+
+            case CF_CHAR:
+                /* We must truncate the primary register to char */
+                g_regchar (lhs);
+                break;
+
+            default:
+                typeerror (lhs);
         }
     }
 
     /* Do not need any other action. If the left type is int, and the primary
-     * register is long, it will be automagically truncated. If the right hand
-     * side is const, it is not located in the primary register and handled by
-     * the expression parser code.
-     */
+    ** register is long, it will be automagically truncated. If the right hand
+    ** side is const, it is not located in the primary register and handled by
+    ** the expression parser code.
+    */
 
     /* Result is const if the right hand side was const */
     lhs |= (rhs & CF_CONST);
 
     /* The resulting type is that of the left hand side (that's why you called
-     * this function :-)
-     */
+    ** this function :-)
+    */
     return lhs;
 }
 
@@ -1411,10 +1420,10 @@ unsigned g_typecast (unsigned lhs, unsigned rhs)
 
 void g_scale (unsigned flags, long val)
 /* Scale the value in the primary register by the given value. If val is positive,
- * scale up, is val is negative, scale down. This function is used to scale
- * the operands or results of pointer arithmetic by the size of the type, the
- * pointer points to.
- */
+** scale up, is val is negative, scale down. This function is used to scale
+** the operands or results of pointer arithmetic by the size of the type, the
+** pointer points to.
+*/
 {
     int p2;
 
@@ -1427,41 +1436,32 @@ void g_scale (unsigned flags, long val)
         if ((p2 = PowerOf2 (val)) > 0 && p2 <= 4) {
 
             /* Factor is 2, 4, 8 and 16, use special function */
-            switch (flags & CF_TYPE) {
+            switch (flags & CF_TYPEMASK) {
 
                 case CF_CHAR:
                     if (flags & CF_FORCECHAR) {
-                       while (p2--) {
-                           AddCodeLine ("asl a");
-                       }
-                       break;
-                   }
-                   /* FALLTHROUGH */
-
-               case CF_INT:
-                    if (IS_Get (&CodeSizeFactor) >= (p2+1)*130) {
-                       AddCodeLine ("stx tmp1");
-                       while (p2--) {
-                           AddCodeLine ("asl a");
-                           AddCodeLine ("rol tmp1");
-                       }
-                       AddCodeLine ("ldx tmp1");
-                   } else {
-                       if (flags & CF_UNSIGNED) {
-                           AddCodeLine ("jsr shlax%d", p2);
-                       } else {
-                           AddCodeLine ("jsr aslax%d", p2);
-                       }
-                   }
-                   break;
-
-               case CF_LONG:
-                   if (flags & CF_UNSIGNED) {
-                       AddCodeLine ("jsr shleax%d", p2);
-                   } else {
-                       AddCodeLine ("jsr asleax%d", p2);
-                   }
-                   break;
+                        while (p2--) {
+                            AddCodeLine ("asl a");
+                        }
+                        break;
+                    }
+                    /* FALLTHROUGH */
+
+                case CF_INT:
+                    if (flags & CF_UNSIGNED) {
+                        AddCodeLine ("jsr shlax%d", p2);
+                    } else {
+                        AddCodeLine ("jsr aslax%d", p2);
+                    }
+                    break;
+
+                case CF_LONG:
+                    if (flags & CF_UNSIGNED) {
+                        AddCodeLine ("jsr shleax%d", p2);
+                    } else {
+                        AddCodeLine ("jsr asleax%d", p2);
+                    }
+                    break;
 
                 default:
                     typeerror (flags);
@@ -1482,17 +1482,17 @@ void g_scale (unsigned flags, long val)
         if ((p2 = PowerOf2 (val)) > 0 && p2 <= 4) {
 
             /* Factor is 2, 4, 8 and 16 use special function */
-            switch (flags & CF_TYPE) {
+            switch (flags & CF_TYPEMASK) {
 
                 case CF_CHAR:
                     if (flags & CF_FORCECHAR) {
                         if (flags & CF_UNSIGNED) {
                             while (p2--) {
-                               AddCodeLine ("lsr a");
+                                AddCodeLine ("lsr a");
                             }
                             break;
                         } else if (p2 <= 2) {
-                           AddCodeLine ("cmp #$80");
+                            AddCodeLine ("cmp #$80");
                             AddCodeLine ("ror a");
                             break;
                         }
@@ -1501,36 +1501,17 @@ void g_scale (unsigned flags, long val)
 
                 case CF_INT:
                     if (flags & CF_UNSIGNED) {
-                        if (IS_Get (&CodeSizeFactor) >= (p2+1)*130) {
-                            AddCodeLine ("stx tmp1");
-                            while (p2--) {
-                               AddCodeLine ("lsr tmp1");
-                                AddCodeLine ("ror a");
-                            }
-                            AddCodeLine ("ldx tmp1");
-                        } else {
-                            AddCodeLine ("jsr lsrax%d", p2);
-                        }
+                        AddCodeLine ("jsr lsrax%d", p2);
                     } else {
-                        if (IS_Get (&CodeSizeFactor) >= (p2+1)*150) {
-                            AddCodeLine ("stx tmp1");
-                            while (p2--) {
-                               AddCodeLine ("cpx #$80");
-                               AddCodeLine ("ror tmp1");
-                               AddCodeLine ("ror a");
-                            }
-                            AddCodeLine ("ldx tmp1");
-                        } else {
-                            AddCodeLine ("jsr asrax%d", p2);
-                       }
+                        AddCodeLine ("jsr asrax%d", p2);
                     }
                     break;
 
                 case CF_LONG:
                     if (flags & CF_UNSIGNED) {
-                       AddCodeLine ("jsr lsreax%d", p2);
+                        AddCodeLine ("jsr lsreax%d", p2);
                     } else {
-                               AddCodeLine ("jsr asreax%d", p2);
+                        AddCodeLine ("jsr asreax%d", p2);
                     }
                     break;
 
@@ -1551,7 +1532,7 @@ void g_scale (unsigned flags, long val)
 
 
 /*****************************************************************************/
-/*             Adds and subs of variables fix a fixed address               */
+/*              Adds and subs of variables fix a fixed address               */
 /*****************************************************************************/
 
 
@@ -1565,7 +1546,7 @@ void g_addlocal (unsigned flags, int offs)
     offs -= StackPtr;
     CheckLocalOffs (offs);
 
-    switch (flags & CF_TYPE) {
+    switch (flags & CF_TYPEMASK) {
 
         case CF_CHAR:
             L = GetLocalLabel();
@@ -1604,7 +1585,7 @@ void g_addlocal (unsigned flags, int offs)
 
 
 
-void g_addstatic (unsigned flags, unsigned long label, long offs)
+void g_addstatic (unsigned flags, uintptr_t label, long offs)
 /* Add a static variable to ax */
 {
     unsigned L;
@@ -1612,7 +1593,7 @@ void g_addstatic (unsigned flags, unsigned long label, long offs)
     /* Create the correct label name */
     const char* lbuf = GetLabelName (flags, label, offs);
 
-    switch (flags & CF_TYPE) {
+    switch (flags & CF_TYPEMASK) {
 
         case CF_CHAR:
             L = GetLocalLabel();
@@ -1649,42 +1630,42 @@ void g_addstatic (unsigned flags, unsigned long label, long offs)
 
 
 /*****************************************************************************/
-/*                          Special op= functions                           */
+/*                           Special op= functions                           */
 /*****************************************************************************/
 
 
 
-void g_addeqstatic (unsigned flags, unsigned long label, long offs,
-                   unsigned long val)
+void g_addeqstatic (unsigned flags, uintptr_t label, long offs,
+                    unsigned long val)
 /* Emit += for a static variable */
 {
     /* Create the correct label name */
     const char* lbuf = GetLabelName (flags, label, offs);
 
     /* Check the size and determine operation */
-    switch (flags & CF_TYPE) {
+    switch (flags & CF_TYPEMASK) {
 
         case CF_CHAR:
             if (flags & CF_FORCECHAR) {
-                       AddCodeLine ("ldx #$00");
-               if (flags & CF_CONST) {
-                   if (val == 1) {
-                       AddCodeLine ("inc %s", lbuf);
-                       AddCodeLine ("lda %s", lbuf);
-                   } else {
-                               AddCodeLine ("lda #$%02X", (int)(val & 0xFF));
-                       AddCodeLine ("clc");
-                       AddCodeLine ("adc %s", lbuf);
-                       AddCodeLine ("sta %s", lbuf);
-                   }
-                       } else {
-                   AddCodeLine ("clc");
-                   AddCodeLine ("adc %s", lbuf);
-                   AddCodeLine ("sta %s", lbuf);
-               }
-               if ((flags & CF_UNSIGNED) == 0) {
+                AddCodeLine ("ldx #$00");
+                if (flags & CF_CONST) {
+                    if (val == 1) {
+                        AddCodeLine ("inc %s", lbuf);
+                        AddCodeLine ("lda %s", lbuf);
+                    } else {
+                        AddCodeLine ("lda #$%02X", (int)(val & 0xFF));
+                        AddCodeLine ("clc");
+                        AddCodeLine ("adc %s", lbuf);
+                        AddCodeLine ("sta %s", lbuf);
+                    }
+                } else {
+                    AddCodeLine ("clc");
+                    AddCodeLine ("adc %s", lbuf);
+                    AddCodeLine ("sta %s", lbuf);
+                }
+                if ((flags & CF_UNSIGNED) == 0) {
                     unsigned L = GetLocalLabel();
-                   AddCodeLine ("bpl %s", LocalLabelName (L));
+                    AddCodeLine ("bpl %s", LocalLabelName (L));
                     AddCodeLine ("dex");
                     g_defcodelabel (L);
                 }
@@ -1700,25 +1681,25 @@ void g_addeqstatic (unsigned flags, unsigned long label, long offs,
                     AddCodeLine ("bne %s", LocalLabelName (L));
                     AddCodeLine ("inc %s+1", lbuf);
                     g_defcodelabel (L);
-                    AddCodeLine ("lda %s", lbuf);              /* Hmmm... */
+                    AddCodeLine ("lda %s", lbuf);               /* Hmmm... */
                     AddCodeLine ("ldx %s+1", lbuf);
                 } else {
-                           AddCodeLine ("lda #$%02X", (int)(val & 0xFF));
+                    AddCodeLine ("lda #$%02X", (int)(val & 0xFF));
                     AddCodeLine ("clc");
                     AddCodeLine ("adc %s", lbuf);
                     AddCodeLine ("sta %s", lbuf);
                     if (val < 0x100) {
-                               unsigned L = GetLocalLabel ();
-                               AddCodeLine ("bcc %s", LocalLabelName (L));
-                               AddCodeLine ("inc %s+1", lbuf);
-                               g_defcodelabel (L);
-                               AddCodeLine ("ldx %s+1", lbuf);
+                        unsigned L = GetLocalLabel ();
+                        AddCodeLine ("bcc %s", LocalLabelName (L));
+                        AddCodeLine ("inc %s+1", lbuf);
+                        g_defcodelabel (L);
+                        AddCodeLine ("ldx %s+1", lbuf);
                     } else {
-                               AddCodeLine ("lda #$%02X", (unsigned char)(val >> 8));
-                               AddCodeLine ("adc %s+1", lbuf);
-                               AddCodeLine ("sta %s+1", lbuf);
-                               AddCodeLine ("tax");
-                               AddCodeLine ("lda %s", lbuf);
+                        AddCodeLine ("lda #$%02X", (unsigned char)(val >> 8));
+                        AddCodeLine ("adc %s+1", lbuf);
+                        AddCodeLine ("sta %s+1", lbuf);
+                        AddCodeLine ("tax");
+                        AddCodeLine ("lda %s", lbuf);
                     }
                 }
             } else {
@@ -1727,9 +1708,9 @@ void g_addeqstatic (unsigned flags, unsigned long label, long offs,
                 AddCodeLine ("sta %s", lbuf);
                 AddCodeLine ("txa");
                 AddCodeLine ("adc %s+1", lbuf);
-               AddCodeLine ("sta %s+1", lbuf);
-               AddCodeLine ("tax");
-               AddCodeLine ("lda %s", lbuf);
+                AddCodeLine ("sta %s+1", lbuf);
+                AddCodeLine ("tax");
+                AddCodeLine ("lda %s", lbuf);
             }
             break;
 
@@ -1743,7 +1724,7 @@ void g_addeqstatic (unsigned flags, unsigned long label, long offs,
                         AddCodeLine ("jsr laddeq1");
                     } else {
                         AddCodeLine ("lda #$%02X", (int)(val & 0xFF));
-                       AddCodeLine ("jsr laddeqa");
+                        AddCodeLine ("jsr laddeqa");
                     }
                 } else {
                     g_getstatic (flags, label, offs);
@@ -1765,19 +1746,19 @@ void g_addeqstatic (unsigned flags, unsigned long label, long offs,
 
 
 
-void g_addeqlocal (unsigned flags, int offs, unsigned long val)
+void g_addeqlocal (unsigned flags, int Offs, unsigned long val)
 /* Emit += for a local variable */
 {
     /* Calculate the true offset, check it, load it into Y */
-    offs -= StackPtr;
-    CheckLocalOffs (offs);
+    Offs -= StackPtr;
+    CheckLocalOffs (Offs);
 
     /* Check the size and determine operation */
-    switch (flags & CF_TYPE) {
+    switch (flags & CF_TYPEMASK) {
 
         case CF_CHAR:
             if (flags & CF_FORCECHAR) {
-                ldyconst (offs);
+                AddCodeLine ("ldy #$%02X", Offs);
                 AddCodeLine ("ldx #$00");
                 if (flags & CF_CONST) {
                     AddCodeLine ("clc");
@@ -1789,18 +1770,18 @@ void g_addeqlocal (unsigned flags, int offs, unsigned long val)
                     AddCodeLine ("adc (sp),y");
                     AddCodeLine ("sta (sp),y");
                 }
-               if ((flags & CF_UNSIGNED) == 0) {
+                if ((flags & CF_UNSIGNED) == 0) {
                     unsigned L = GetLocalLabel();
-                   AddCodeLine ("bpl %s", LocalLabelName (L));
-                   AddCodeLine ("dex");
+                    AddCodeLine ("bpl %s", LocalLabelName (L));
+                    AddCodeLine ("dex");
                     g_defcodelabel (L);
-               }
-               break;
+                }
+                break;
             }
             /* FALLTHROUGH */
 
         case CF_INT:
-            ldyconst (offs);
+            AddCodeLine ("ldy #$%02X", Offs);
             if (flags & CF_CONST) {
                 if (IS_Get (&CodeSizeFactor) >= 400) {
                     AddCodeLine ("clc");
@@ -1819,15 +1800,15 @@ void g_addeqlocal (unsigned flags, int offs, unsigned long val)
                     AddCodeLine ("jsr addeqysp");
                 }
             } else {
-               AddCodeLine ("jsr addeqysp");
+                AddCodeLine ("jsr addeqysp");
             }
             break;
 
         case CF_LONG:
             if (flags & CF_CONST) {
-               g_getimmed (flags, val, 0);
+                g_getimmed (flags, val, 0);
             }
-            ldyconst (offs);
+            AddCodeLine ("ldy #$%02X", Offs);
             AddCodeLine ("jsr laddeqysp");
             break;
 
@@ -1842,13 +1823,13 @@ void g_addeqind (unsigned flags, unsigned offs, unsigned long val)
 /* Emit += for the location with address in ax */
 {
     /* If the offset is too large for a byte register, add the high byte
-     * of the offset to the primary. Beware: We need a special correction
-     * if the offset in the low byte will overflow in the operation.
-     */
+    ** of the offset to the primary. Beware: We need a special correction
+    ** if the offset in the low byte will overflow in the operation.
+    */
     offs = MakeByteOffs (flags, offs);
 
     /* Check the size and determine operation */
-    switch (flags & CF_TYPE) {
+    switch (flags & CF_TYPEMASK) {
 
         case CF_CHAR:
             AddCodeLine ("sta ptr1");
@@ -1862,32 +1843,12 @@ void g_addeqind (unsigned flags, unsigned offs, unsigned long val)
             break;
 
         case CF_INT:
-            if (IS_Get (&CodeSizeFactor) >= 200) {
-                /* Lots of code, use only if size is not important */
-                       AddCodeLine ("sta ptr1");
-                AddCodeLine ("stx ptr1+1");
-                AddCodeLine ("ldy #$%02X", offs);
-                AddCodeLine ("lda #$%02X", (int)(val & 0xFF));
-                AddCodeLine ("clc");
-                AddCodeLine ("adc (ptr1),y");
-                AddCodeLine ("sta (ptr1),y");
-                AddCodeLine ("pha");
-                AddCodeLine ("iny");
-                AddCodeLine ("lda #$%02X", (unsigned char)(val >> 8));
-                AddCodeLine ("adc (ptr1),y");
-                AddCodeLine ("sta (ptr1),y");
-                AddCodeLine ("tax");
-                AddCodeLine ("pla");
-                break;
-            }
-            /* FALL THROUGH */
-
         case CF_LONG:
-            AddCodeLine ("jsr pushax");        /* Push the address */
-            push (CF_PTR);                     /* Correct the internal sp */
-            g_getind (flags, offs);            /* Fetch the value */
-            g_inc (flags, val);                        /* Increment value in primary */
-            g_putind (flags, offs);            /* Store the value back */
+            AddCodeLine ("jsr pushax");         /* Push the address */
+            push (CF_PTR);                      /* Correct the internal sp */
+            g_getind (flags, offs);             /* Fetch the value */
+            g_inc (flags, val);                 /* Increment value in primary */
+            g_putind (flags, offs);             /* Store the value back */
             break;
 
         default:
@@ -1897,7 +1858,7 @@ void g_addeqind (unsigned flags, unsigned offs, unsigned long val)
 
 
 
-void g_subeqstatic (unsigned flags, unsigned long label, long offs,
+void g_subeqstatic (unsigned flags, uintptr_t label, long offs,
                     unsigned long val)
 /* Emit -= for a static variable */
 {
@@ -1905,46 +1866,46 @@ void g_subeqstatic (unsigned flags, unsigned long label, long offs,
     const char* lbuf = GetLabelName (flags, label, offs);
 
     /* Check the size and determine operation */
-    switch (flags & CF_TYPE) {
+    switch (flags & CF_TYPEMASK) {
 
         case CF_CHAR:
             if (flags & CF_FORCECHAR) {
                 AddCodeLine ("ldx #$00");
-               if (flags & CF_CONST) {
+                if (flags & CF_CONST) {
                     if (val == 1) {
                         AddCodeLine ("dec %s", lbuf);
                         AddCodeLine ("lda %s", lbuf);
                     } else {
-                               AddCodeLine ("sec");
-                       AddCodeLine ("lda %s", lbuf);
-                       AddCodeLine ("sbc #$%02X", (int)(val & 0xFF));
-                       AddCodeLine ("sta %s", lbuf);
+                        AddCodeLine ("lda %s", lbuf);
+                        AddCodeLine ("sec");
+                        AddCodeLine ("sbc #$%02X", (int)(val & 0xFF));
+                        AddCodeLine ("sta %s", lbuf);
                     }
-               } else {
+                } else {
                     AddCodeLine ("eor #$FF");
                     AddCodeLine ("sec");
-                           AddCodeLine ("adc %s", lbuf);
+                    AddCodeLine ("adc %s", lbuf);
                     AddCodeLine ("sta %s", lbuf);
-               }
+                }
                 if ((flags & CF_UNSIGNED) == 0) {
                     unsigned L = GetLocalLabel();
                     AddCodeLine ("bpl %s", LocalLabelName (L));
                     AddCodeLine ("dex");
                     g_defcodelabel (L);
-               }
-               break;
+                }
+                break;
             }
             /* FALLTHROUGH */
 
         case CF_INT:
-            AddCodeLine ("sec");
             if (flags & CF_CONST) {
-                       AddCodeLine ("lda %s", lbuf);
-               AddCodeLine ("sbc #$%02X", (unsigned char)val);
-               AddCodeLine ("sta %s", lbuf);
-               if (val < 0x100) {
-                   unsigned L = GetLocalLabel ();
-                   AddCodeLine ("bcs %s", LocalLabelName (L));
+                AddCodeLine ("lda %s", lbuf);
+                AddCodeLine ("sec");
+                AddCodeLine ("sbc #$%02X", (unsigned char)val);
+                AddCodeLine ("sta %s", lbuf);
+                if (val < 0x100) {
+                    unsigned L = GetLocalLabel ();
+                    AddCodeLine ("bcs %s", LocalLabelName (L));
                     AddCodeLine ("dec %s+1", lbuf);
                     g_defcodelabel (L);
                     AddCodeLine ("ldx %s+1", lbuf);
@@ -1957,11 +1918,12 @@ void g_subeqstatic (unsigned flags, unsigned long label, long offs,
                 }
             } else {
                 AddCodeLine ("eor #$FF");
-                       AddCodeLine ("adc %s", lbuf);
+                AddCodeLine ("sec");
+                AddCodeLine ("adc %s", lbuf);
                 AddCodeLine ("sta %s", lbuf);
                 AddCodeLine ("txa");
                 AddCodeLine ("eor #$FF");
-                       AddCodeLine ("adc %s+1", lbuf);
+                AddCodeLine ("adc %s+1", lbuf);
                 AddCodeLine ("sta %s+1", lbuf);
                 AddCodeLine ("tax");
                 AddCodeLine ("lda %s", lbuf);
@@ -1996,52 +1958,53 @@ void g_subeqstatic (unsigned flags, unsigned long label, long offs,
 
 
 
-void g_subeqlocal (unsigned flags, int offs, unsigned long val)
+void g_subeqlocal (unsigned flags, int Offs, unsigned long val)
 /* Emit -= for a local variable */
 {
     /* Calculate the true offset, check it, load it into Y */
-    offs -= StackPtr;
-    CheckLocalOffs (offs);
+    Offs -= StackPtr;
+    CheckLocalOffs (Offs);
 
     /* Check the size and determine operation */
-    switch (flags & CF_TYPE) {
+    switch (flags & CF_TYPEMASK) {
 
         case CF_CHAR:
             if (flags & CF_FORCECHAR) {
-               ldyconst (offs);
+                AddCodeLine ("ldy #$%02X", Offs);
                 AddCodeLine ("ldx #$00");
-               AddCodeLine ("sec");
                 if (flags & CF_CONST) {
                     AddCodeLine ("lda (sp),y");
+                    AddCodeLine ("sec");
                     AddCodeLine ("sbc #$%02X", (unsigned char)val);
                 } else {
                     AddCodeLine ("eor #$FF");
-                   AddCodeLine ("adc (sp),y");
+                    AddCodeLine ("sec");
+                    AddCodeLine ("adc (sp),y");
                 }
-               AddCodeLine ("sta (sp),y");
+                AddCodeLine ("sta (sp),y");
                 if ((flags & CF_UNSIGNED) == 0) {
-                   unsigned L = GetLocalLabel();
-                           AddCodeLine ("bpl %s", LocalLabelName (L));
-                   AddCodeLine ("dex");
-                   g_defcodelabel (L);
-               }
-               break;
+                    unsigned L = GetLocalLabel();
+                    AddCodeLine ("bpl %s", LocalLabelName (L));
+                    AddCodeLine ("dex");
+                    g_defcodelabel (L);
+                }
+                break;
             }
             /* FALLTHROUGH */
 
         case CF_INT:
             if (flags & CF_CONST) {
-               g_getimmed (flags, val, 0);
+                g_getimmed (flags, val, 0);
             }
-            ldyconst (offs);
+            AddCodeLine ("ldy #$%02X", Offs);
             AddCodeLine ("jsr subeqysp");
             break;
 
         case CF_LONG:
             if (flags & CF_CONST) {
-               g_getimmed (flags, val, 0);
+                g_getimmed (flags, val, 0);
             }
-            ldyconst (offs);
+            AddCodeLine ("ldy #$%02X", Offs);
             AddCodeLine ("jsr lsubeqysp");
             break;
 
@@ -2056,13 +2019,13 @@ void g_subeqind (unsigned flags, unsigned offs, unsigned long val)
 /* Emit -= for the location with address in ax */
 {
     /* If the offset is too large for a byte register, add the high byte
-     * of the offset to the primary. Beware: We need a special correction
-     * if the offset in the low byte will overflow in the operation.
-     */
+    ** of the offset to the primary. Beware: We need a special correction
+    ** if the offset in the low byte will overflow in the operation.
+    */
     offs = MakeByteOffs (flags, offs);
 
     /* Check the size and determine operation */
-    switch (flags & CF_TYPE) {
+    switch (flags & CF_TYPEMASK) {
 
         case CF_CHAR:
             AddCodeLine ("sta ptr1");
@@ -2076,32 +2039,12 @@ void g_subeqind (unsigned flags, unsigned offs, unsigned long val)
             break;
 
         case CF_INT:
-            if (IS_Get (&CodeSizeFactor) >= 200) {
-                /* Lots of code, use only if size is not important */
-                AddCodeLine ("sta ptr1");
-                       AddCodeLine ("stx ptr1+1");
-                AddCodeLine ("ldy #$%02X", offs);
-                AddCodeLine ("lda (ptr1),y");
-                AddCodeLine ("sec");
-                AddCodeLine ("sbc #$%02X", (unsigned char)val);
-                AddCodeLine ("sta (ptr1),y");
-                AddCodeLine ("pha");
-                AddCodeLine ("iny");
-                AddCodeLine ("lda (ptr1),y");
-                AddCodeLine ("sbc #$%02X", (unsigned char)(val >> 8));
-                AddCodeLine ("sta (ptr1),y");
-               AddCodeLine ("tax");
-                AddCodeLine ("pla");
-                break;
-            }
-            /* FALL THROUGH */
-
         case CF_LONG:
-            AddCodeLine ("jsr pushax");        /* Push the address */
-            push (CF_PTR);                     /* Correct the internal sp */
-            g_getind (flags, offs);            /* Fetch the value */
-            g_dec (flags, val);                        /* Increment value in primary */
-            g_putind (flags, offs);            /* Store the value back */
+            AddCodeLine ("jsr pushax");         /* Push the address */
+            push (CF_PTR);                      /* Correct the internal sp */
+            g_getind (flags, offs);             /* Fetch the value */
+            g_dec (flags, val);                 /* Increment value in primary */
+            g_putind (flags, offs);             /* Store the value back */
             break;
 
         default:
@@ -2112,7 +2055,7 @@ void g_subeqind (unsigned flags, unsigned offs, unsigned long val)
 
 
 /*****************************************************************************/
-/*                 Add a variable address to the value in ax                */
+/*                 Add a variable address to the value in ax                 */
 /*****************************************************************************/
 
 
@@ -2151,7 +2094,7 @@ void g_addaddr_local (unsigned flags attribute ((unused)), int offs)
 
 
 
-void g_addaddr_static (unsigned flags, unsigned long label, long offs)
+void g_addaddr_static (unsigned flags, uintptr_t label, long offs)
 /* Add the address of a static variable to ax */
 {
     /* Create the correct label name */
@@ -2170,7 +2113,7 @@ void g_addaddr_static (unsigned flags, unsigned long label, long offs)
 
 
 /*****************************************************************************/
-/*                                                                          */
+/*                                                                           */
 /*****************************************************************************/
 
 
@@ -2179,11 +2122,11 @@ void g_save (unsigned flags)
 /* Copy primary register to hold register. */
 {
     /* Check the size and determine operation */
-    switch (flags & CF_TYPE) {
+    switch (flags & CF_TYPEMASK) {
 
         case CF_CHAR:
             if (flags & CF_FORCECHAR) {
-               AddCodeLine ("pha");
+                AddCodeLine ("pha");
                 break;
             }
             /* FALLTHROUGH */
@@ -2208,12 +2151,12 @@ void g_restore (unsigned flags)
 /* Copy hold register to primary. */
 {
     /* Check the size and determine operation */
-    switch (flags & CF_TYPE) {
+    switch (flags & CF_TYPEMASK) {
 
         case CF_CHAR:
             if (flags & CF_FORCECHAR) {
-                       AddCodeLine ("pla");
-               break;
+                AddCodeLine ("pla");
+                break;
             }
             /* FALLTHROUGH */
 
@@ -2235,18 +2178,18 @@ void g_restore (unsigned flags)
 
 void g_cmp (unsigned flags, unsigned long val)
 /* Immidiate compare. The primary register will not be changed, Z flag
- * will be set.
- */
+** will be set.
+*/
 {
     unsigned L;
 
     /* Check the size and determine operation */
-    switch (flags & CF_TYPE) {
+    switch (flags & CF_TYPEMASK) {
 
         case CF_CHAR:
             if (flags & CF_FORCECHAR) {
-                       AddCodeLine ("cmp #$%02X", (unsigned char)val);
-                       break;
+                AddCodeLine ("cmp #$%02X", (unsigned char)val);
+                break;
             }
             /* FALLTHROUGH */
 
@@ -2269,58 +2212,33 @@ void g_cmp (unsigned flags, unsigned long val)
 
 
 
-static void oper (unsigned flags, unsigned long val, char** subs)
-/* Encode a binary operation. subs is a pointer to four groups of three
- * strings:
- *      0-2    --> Operate on ints
- *      3-5    --> Operate on unsigneds
- *      6-8    --> Operate on longs
- *      9-11   --> Operate on unsigned longs
- *
- * The first subroutine names in each string group is used to encode an
- * operation with a zero constant, the second to encode an operation with
- * a 8 bit constant, and the third is used in all other cases.
- */
+static void oper (unsigned Flags, unsigned long Val, const char* const* Subs)
+/* Encode a binary operation. subs is a pointer to four strings:
+**      0       --> Operate on ints
+**      1       --> Operate on unsigneds
+**      2       --> Operate on longs
+**      3       --> Operate on unsigned longs
+*/
 {
-    unsigned offs;
-
     /* Determine the offset into the array */
-    offs = (flags & CF_UNSIGNED)? 3 : 0;
-    switch (flags & CF_TYPE) {
-        case CF_CHAR:
-        case CF_INT:
-            break;
-
-        case CF_LONG:
-            offs += 6;
-            break;
+    if (Flags & CF_UNSIGNED) {
+        ++Subs;
+    }
+    if ((Flags & CF_TYPEMASK) == CF_LONG) {
+        Subs += 2;
+    }
 
-        default:
-            typeerror (flags);
+    /* Load the value if it is not already in the primary */
+    if (Flags & CF_CONST) {
+        /* Load value */
+        g_getimmed (Flags, Val, 0);
     }
 
-    /* Encode the operation */
-    if (flags & CF_CONST) {
-        /* Constant value given */
-        if (val == 0 && subs [offs+0]) {
-            /* Special case: constant with value zero */
-            AddCodeLine ("jsr %s", subs [offs+0]);
-        } else if (val < 0x100 && subs [offs+1]) {
-            /* Special case: constant with high byte zero */
-            ldaconst (val);            /* Load low byte */
-            AddCodeLine ("jsr %s", subs [offs+1]);
-        } else {
-            /* Others: arbitrary constant value */
-            g_getimmed (flags, val, 0);                /* Load value */
-            AddCodeLine ("jsr %s", subs [offs+2]);
-        }
-    } else {
-        /* Value not constant (is already in (e)ax) */
-        AddCodeLine ("jsr %s", subs [offs+2]);
-    }
+    /* Output the operation */
+    AddCodeLine ("jsr %s", *Subs);
 
     /* The operation will pop it's argument */
-    pop (flags);
+    pop (Flags);
 }
 
 
@@ -2328,7 +2246,7 @@ static void oper (unsigned flags, unsigned long val, char** subs)
 void g_test (unsigned flags)
 /* Test the value in the primary and set the condition codes */
 {
-    switch (flags & CF_TYPE) {
+    switch (flags & CF_TYPEMASK) {
 
         case CF_CHAR:
             if (flags & CF_FORCECHAR) {
@@ -2344,9 +2262,9 @@ void g_test (unsigned flags)
 
         case CF_LONG:
             if (flags & CF_UNSIGNED) {
-               AddCodeLine ("jsr utsteax");
+                AddCodeLine ("jsr utsteax");
             } else {
-               AddCodeLine ("jsr tsteax");
+                AddCodeLine ("jsr tsteax");
             }
             break;
 
@@ -2361,13 +2279,13 @@ void g_test (unsigned flags)
 void g_push (unsigned flags, unsigned long val)
 /* Push the primary register or a constant value onto the stack */
 {
-    if (flags & CF_CONST && (flags & CF_TYPE) != CF_LONG) {
+    if (flags & CF_CONST && (flags & CF_TYPEMASK) != CF_LONG) {
 
         /* We have a constant 8 or 16 bit value */
-        if ((flags & CF_TYPE) == CF_CHAR && (flags & CF_FORCECHAR)) {
+        if ((flags & CF_TYPEMASK) == CF_CHAR && (flags & CF_FORCECHAR)) {
 
             /* Handle as 8 bit value */
-            ldaconst (val);
+            AddCodeLine ("lda #$%02X", (unsigned char) val);
             AddCodeLine ("jsr pusha");
 
         } else {
@@ -2386,7 +2304,7 @@ void g_push (unsigned flags, unsigned long val)
         }
 
         /* Push the primary register */
-        switch (flags & CF_TYPE) {
+        switch (flags & CF_TYPEMASK) {
 
             case CF_CHAR:
                 if (flags & CF_FORCECHAR) {
@@ -2400,7 +2318,7 @@ void g_push (unsigned flags, unsigned long val)
                 break;
 
             case CF_LONG:
-               AddCodeLine ("jsr pusheax");
+                AddCodeLine ("jsr pusheax");
                 break;
 
             default:
@@ -2418,10 +2336,10 @@ void g_push (unsigned flags, unsigned long val)
 
 void g_swap (unsigned flags)
 /* Swap the primary register and the top of the stack. flags give the type
- * of *both* values (must have same size).
- */
+** of *both* values (must have same size).
+*/
 {
-    switch (flags & CF_TYPE) {
+    switch (flags & CF_TYPEMASK) {
 
         case CF_CHAR:
         case CF_INT:
@@ -2445,7 +2363,7 @@ void g_call (unsigned Flags, const char* Label, unsigned ArgSize)
 {
     if ((Flags & CF_FIXARGC) == 0) {
         /* Pass the argument count */
-        ldyconst (ArgSize);
+        AddCodeLine ("ldy #$%02X", ArgSize);
     }
     AddCodeLine ("jsr _%s", Label);
     StackPtr += ArgSize;                /* callee pops args */
@@ -2460,7 +2378,7 @@ void g_callind (unsigned Flags, unsigned ArgSize, int Offs)
         /* Address is in a/x */
         if ((Flags & CF_FIXARGC) == 0) {
             /* Pass arg count */
-            ldyconst (ArgSize);
+            AddCodeLine ("ldy #$%02X", ArgSize);
         }
         AddCodeLine ("jsr callax");
     } else {
@@ -2507,27 +2425,70 @@ void g_falsejump (unsigned flags attribute ((unused)), unsigned label)
 }
 
 
+void g_lateadjustSP (unsigned label)
+/* Adjust stack based on non-immediate data */
+{
+    AddCodeLine ("pha");
+    AddCodeLine ("lda %s", LocalLabelName (label));
+    AddCodeLine ("clc");
+    AddCodeLine ("adc sp");
+    AddCodeLine ("sta sp");
+    AddCodeLine ("lda %s+1", LocalLabelName (label));
+    AddCodeLine ("adc sp+1");
+    AddCodeLine ("sta sp+1");
+    AddCodeLine ("pla");
+}
 
-static void mod_internal (int k, char* verb1, char* verb2)
+void g_drop (unsigned Space)
+/* Drop space allocated on the stack */
 {
-    if (k <= 8) {
-        AddCodeLine ("jsr %ssp%c", verb1, k + '0');
-    } else {
-        CheckLocalOffs (k);
-        ldyconst (k);
-        AddCodeLine ("jsr %ssp", verb2);
+    if (Space > 255) {
+        /* Inline the code since calling addysp repeatedly is quite some
+        ** overhead.
+        */
+        AddCodeLine ("pha");
+        AddCodeLine ("lda #$%02X", (unsigned char) Space);
+        AddCodeLine ("clc");
+        AddCodeLine ("adc sp");
+        AddCodeLine ("sta sp");
+        AddCodeLine ("lda #$%02X", (unsigned char) (Space >> 8));
+        AddCodeLine ("adc sp+1");
+        AddCodeLine ("sta sp+1");
+        AddCodeLine ("pla");
+    } else if (Space > 8) {
+        AddCodeLine ("ldy #$%02X", Space);
+        AddCodeLine ("jsr addysp");
+    } else if (Space != 0) {
+        AddCodeLine ("jsr incsp%u", Space);
     }
 }
 
 
 
-void g_space (int space)
+void g_space (int Space)
 /* Create or drop space on the stack */
 {
-    if (space < 0) {
-        mod_internal (-space, "inc", "addy");
-    } else if (space > 0) {
-        mod_internal (space, "dec", "suby");
+    if (Space < 0) {
+        /* This is actually a drop operation */
+        g_drop (-Space);
+    } else if (Space > 255) {
+        /* Inline the code since calling subysp repeatedly is quite some
+        ** overhead.
+        */
+        AddCodeLine ("pha");
+        AddCodeLine ("lda sp");
+        AddCodeLine ("sec");
+        AddCodeLine ("sbc #$%02X", (unsigned char) Space);
+        AddCodeLine ("sta sp");
+        AddCodeLine ("lda sp+1");
+        AddCodeLine ("sbc #$%02X", (unsigned char) (Space >> 8));
+        AddCodeLine ("sta sp+1");
+        AddCodeLine ("pla");
+    } else if (Space > 8) {
+        AddCodeLine ("ldy #$%02X", Space);
+        AddCodeLine ("jsr subysp");
+    } else if (Space != 0) {
+        AddCodeLine ("jsr decsp%u", Space);
     }
 }
 
@@ -2552,15 +2513,12 @@ void g_stackcheck (void)
 void g_add (unsigned flags, unsigned long val)
 /* Primary = TOS + Primary */
 {
-    static char* ops [12] = {
-        0,             "tosadda0",     "tosaddax",
-        0,             "tosadda0",     "tosaddax",
-        0,             0,              "tosaddeax",
-        0,             0,              "tosaddeax",
+    static const char* const ops[4] = {
+        "tosaddax", "tosaddax", "tosaddeax", "tosaddeax"
     };
 
     if (flags & CF_CONST) {
-        flags &= ~CF_FORCECHAR;        /* Handle chars as ints */
+        flags &= ~CF_FORCECHAR; /* Handle chars as ints */
         g_push (flags & ~CF_CONST, 0);
     }
     oper (flags, val, ops);
@@ -2571,15 +2529,12 @@ void g_add (unsigned flags, unsigned long val)
 void g_sub (unsigned flags, unsigned long val)
 /* Primary = TOS - Primary */
 {
-    static char* ops [12] = {
-        0,             "tossuba0",     "tossubax",
-        0,             "tossuba0",     "tossubax",
-        0,             0,              "tossubeax",
-        0,             0,              "tossubeax",
+    static const char* const ops[4] = {
+        "tossubax", "tossubax", "tossubeax", "tossubeax"
     };
 
     if (flags & CF_CONST) {
-        flags &= ~CF_FORCECHAR;        /* Handle chars as ints */
+        flags &= ~CF_FORCECHAR; /* Handle chars as ints */
         g_push (flags & ~CF_CONST, 0);
     }
     oper (flags, val, ops);
@@ -2590,11 +2545,8 @@ void g_sub (unsigned flags, unsigned long val)
 void g_rsub (unsigned flags, unsigned long val)
 /* Primary = Primary - TOS */
 {
-    static char* ops [12] = {
-        0,             "tosrsuba0",    "tosrsubax",
-        0,             "tosrsuba0",    "tosrsubax",
-        0,             0,              "tosrsubeax",
-        0,             0,              "tosrsubeax",
+    static const char* const ops[4] = {
+        "tosrsubax", "tosrsubax", "tosrsubeax", "tosrsubeax"
     };
     oper (flags, val, ops);
 }
@@ -2604,11 +2556,8 @@ void g_rsub (unsigned flags, unsigned long val)
 void g_mul (unsigned flags, unsigned long val)
 /* Primary = TOS * Primary */
 {
-    static char* ops [12] = {
-        0,             "tosmula0",     "tosmulax",
-        0,             "tosumula0",    "tosumulax",
-        0,             0,              "tosmuleax",
-        0,             0,              "tosumuleax",
+    static const char* const ops[4] = {
+        "tosmulax", "tosumulax", "tosmuleax", "tosumuleax"
     };
 
     int p2;
@@ -2621,48 +2570,48 @@ void g_mul (unsigned flags, unsigned long val)
     }
 
     /* If the right hand side is const, the lhs is not on stack but still
-     * in the primary register.
-     */
+    ** in the primary register.
+    */
     if (flags & CF_CONST) {
 
-        switch (flags & CF_TYPE) {
+        switch (flags & CF_TYPEMASK) {
 
             case CF_CHAR:
                 if (flags & CF_FORCECHAR) {
                     /* Handle some special cases */
                     switch (val) {
 
-                       case 3:
-                           AddCodeLine ("sta tmp1");
-                           AddCodeLine ("asl a");
-                           AddCodeLine ("clc");
-                           AddCodeLine ("adc tmp1");
-                           return;
-
-                       case 5:
-                           AddCodeLine ("sta tmp1");
-                           AddCodeLine ("asl a");
-                           AddCodeLine ("asl a");
-                           AddCodeLine ("clc");
-                           AddCodeLine ("adc tmp1");
-                           return;
-
-                       case 6:
-                           AddCodeLine ("sta tmp1");
-                           AddCodeLine ("asl a");
-                           AddCodeLine ("clc");
-                           AddCodeLine ("adc tmp1");
+                        case 3:
+                            AddCodeLine ("sta tmp1");
+                            AddCodeLine ("asl a");
+                            AddCodeLine ("clc");
+                            AddCodeLine ("adc tmp1");
+                            return;
+
+                        case 5:
+                            AddCodeLine ("sta tmp1");
+                            AddCodeLine ("asl a");
+                            AddCodeLine ("asl a");
+                            AddCodeLine ("clc");
+                            AddCodeLine ("adc tmp1");
+                            return;
+
+                        case 6:
+                            AddCodeLine ("sta tmp1");
+                            AddCodeLine ("asl a");
+                            AddCodeLine ("clc");
+                            AddCodeLine ("adc tmp1");
+                            AddCodeLine ("asl a");
+                            return;
+
+                        case 10:
+                            AddCodeLine ("sta tmp1");
+                            AddCodeLine ("asl a");
                             AddCodeLine ("asl a");
-                           return;
-
-                       case 10:
-                           AddCodeLine ("sta tmp1");
-                           AddCodeLine ("asl a");
-                           AddCodeLine ("asl a");
-                           AddCodeLine ("clc");
-                           AddCodeLine ("adc tmp1");
-                           AddCodeLine ("asl a");
-                           return;
+                            AddCodeLine ("clc");
+                            AddCodeLine ("adc tmp1");
+                            AddCodeLine ("asl a");
+                            return;
                     }
                 }
                 /* FALLTHROUGH */
@@ -2698,9 +2647,9 @@ void g_mul (unsigned flags, unsigned long val)
         }
 
         /* If we go here, we didn't emit code. Push the lhs on stack and fall
-         * into the normal, non-optimized stuff.
-         */
-        flags &= ~CF_FORCECHAR;        /* Handle chars as ints */
+        ** into the normal, non-optimized stuff.
+        */
+        flags &= ~CF_FORCECHAR; /* Handle chars as ints */
         g_push (flags & ~CF_CONST, 0);
 
     }
@@ -2714,11 +2663,8 @@ void g_mul (unsigned flags, unsigned long val)
 void g_div (unsigned flags, unsigned long val)
 /* Primary = TOS / Primary */
 {
-    static char* ops [12] = {
-        0,             "tosdiva0",     "tosdivax",
-        0,             "tosudiva0",    "tosudivax",
-        0,             0,              "tosdiveax",
-        0,             0,              "tosudiveax",
+    static const char* const ops[4] = {
+        "tosdivax", "tosudivax", "tosdiveax", "tosudiveax"
     };
 
     /* Do strength reduction if the value is constant and a power of two */
@@ -2730,7 +2676,7 @@ void g_div (unsigned flags, unsigned long val)
         /* Generate a division */
         if (flags & CF_CONST) {
             /* lhs is not on stack */
-            flags &= ~CF_FORCECHAR;    /* Handle chars as ints */
+            flags &= ~CF_FORCECHAR;     /* Handle chars as ints */
             g_push (flags & ~CF_CONST, 0);
         }
         oper (flags, val, ops);
@@ -2742,11 +2688,8 @@ void g_div (unsigned flags, unsigned long val)
 void g_mod (unsigned flags, unsigned long val)
 /* Primary = TOS % Primary */
 {
-    static char* ops [12] = {
-        0,             "tosmoda0",     "tosmodax",
-        0,             "tosumoda0",    "tosumodax",
-        0,             0,              "tosmodeax",
-        0,             0,              "tosumodeax",
+    static const char* const ops[4] = {
+        "tosmodax", "tosumodax", "tosmodeax", "tosumodeax"
     };
     int p2;
 
@@ -2758,7 +2701,7 @@ void g_mod (unsigned flags, unsigned long val)
         /* Do it the hard way... */
         if (flags & CF_CONST) {
             /* lhs is not on stack */
-            flags &= ~CF_FORCECHAR;    /* Handle chars as ints */
+            flags &= ~CF_FORCECHAR;     /* Handle chars as ints */
             g_push (flags & ~CF_CONST, 0);
         }
         oper (flags, val, ops);
@@ -2770,24 +2713,21 @@ void g_mod (unsigned flags, unsigned long val)
 void g_or (unsigned flags, unsigned long val)
 /* Primary = TOS | Primary */
 {
-    static char* ops [12] = {
-        0,             "tosora0",      "tosorax",
-        0,             "tosora0",      "tosorax",
-        0,             0,              "tosoreax",
-        0,             0,              "tosoreax",
+    static const char* const ops[4] = {
+        "tosorax", "tosorax", "tosoreax", "tosoreax"
     };
 
     /* If the right hand side is const, the lhs is not on stack but still
-     * in the primary register.
-     */
+    ** in the primary register.
+    */
     if (flags & CF_CONST) {
 
-        switch (flags & CF_TYPE) {
+        switch (flags & CF_TYPEMASK) {
 
             case CF_CHAR:
                 if (flags & CF_FORCECHAR) {
                     if ((val & 0xFF) != 0) {
-                               AddCodeLine ("ora #$%02X", (unsigned char)val);
+                        AddCodeLine ("ora #$%02X", (unsigned char)val);
                     }
                     return;
                 }
@@ -2802,7 +2742,7 @@ void g_or (unsigned flags, unsigned long val)
                     if ((val & 0xFF) != 0) {
                         AddCodeLine ("ora #$%02X", (unsigned char)val);
                     }
-                    ldxconst (0xFF);
+                    AddCodeLine ("ldx #$FF");
                 } else if (val != 0) {
                     AddCodeLine ("ora #$%02X", (unsigned char)val);
                     AddCodeLine ("pha");
@@ -2827,9 +2767,9 @@ void g_or (unsigned flags, unsigned long val)
         }
 
         /* If we go here, we didn't emit code. Push the lhs on stack and fall
-         * into the normal, non-optimized stuff. Note: The standard stuff will
-         * always work with ints.
-         */
+        ** into the normal, non-optimized stuff. Note: The standard stuff will
+        ** always work with ints.
+        */
         flags &= ~CF_FORCECHAR;
         g_push (flags & ~CF_CONST, 0);
     }
@@ -2843,25 +2783,22 @@ void g_or (unsigned flags, unsigned long val)
 void g_xor (unsigned flags, unsigned long val)
 /* Primary = TOS ^ Primary */
 {
-    static char* ops [12] = {
-        0,             "tosxora0",     "tosxorax",
-        0,             "tosxora0",     "tosxorax",
-        0,             0,              "tosxoreax",
-        0,             0,              "tosxoreax",
+    static const char* const ops[4] = {
+        "tosxorax", "tosxorax", "tosxoreax", "tosxoreax"
     };
 
 
     /* If the right hand side is const, the lhs is not on stack but still
-     * in the primary register.
-     */
+    ** in the primary register.
+    */
     if (flags & CF_CONST) {
 
-        switch (flags & CF_TYPE) {
+        switch (flags & CF_TYPEMASK) {
 
             case CF_CHAR:
                 if (flags & CF_FORCECHAR) {
                     if ((val & 0xFF) != 0) {
-                               AddCodeLine ("eor #$%02X", (unsigned char)val);
+                        AddCodeLine ("eor #$%02X", (unsigned char)val);
                     }
                     return;
                 }
@@ -2870,14 +2807,14 @@ void g_xor (unsigned flags, unsigned long val)
             case CF_INT:
                 if (val <= 0xFF) {
                     if (val != 0) {
-                       AddCodeLine ("eor #$%02X", (unsigned char)val);
+                        AddCodeLine ("eor #$%02X", (unsigned char)val);
                     }
-                       } else if (val != 0) {
+                } else if (val != 0) {
                     if ((val & 0xFF) != 0) {
                         AddCodeLine ("eor #$%02X", (unsigned char)val);
                     }
                     AddCodeLine ("pha");
-                   AddCodeLine ("txa");
+                    AddCodeLine ("txa");
                     AddCodeLine ("eor #$%02X", (unsigned char)(val >> 8));
                     AddCodeLine ("tax");
                     AddCodeLine ("pla");
@@ -2887,7 +2824,7 @@ void g_xor (unsigned flags, unsigned long val)
             case CF_LONG:
                 if (val <= 0xFF) {
                     if (val != 0) {
-                               AddCodeLine ("eor #$%02X", (unsigned char)val);
+                        AddCodeLine ("eor #$%02X", (unsigned char)val);
                     }
                     return;
                 }
@@ -2898,9 +2835,9 @@ void g_xor (unsigned flags, unsigned long val)
         }
 
         /* If we go here, we didn't emit code. Push the lhs on stack and fall
-         * into the normal, non-optimized stuff. Note: The standard stuff will
-         * always work with ints.
-         */
+        ** into the normal, non-optimized stuff. Note: The standard stuff will
+        ** always work with ints.
+        */
         flags &= ~CF_FORCECHAR;
         g_push (flags & ~CF_CONST, 0);
     }
@@ -2914,68 +2851,71 @@ void g_xor (unsigned flags, unsigned long val)
 void g_and (unsigned Flags, unsigned long Val)
 /* Primary = TOS & Primary */
 {
-    static char* ops [12] = {
-        0,             "tosanda0",     "tosandax",
-        0,             "tosanda0",     "tosandax",
-        0,             0,              "tosandeax",
-        0,             0,              "tosandeax",
+    static const char* const ops[4] = {
+        "tosandax", "tosandax", "tosandeax", "tosandeax"
     };
 
     /* If the right hand side is const, the lhs is not on stack but still
-     * in the primary register.
-     */
+    ** in the primary register.
+    */
     if (Flags & CF_CONST) {
 
-        switch (Flags & CF_TYPE) {
+        switch (Flags & CF_TYPEMASK) {
 
             case CF_CHAR:
                 if (Flags & CF_FORCECHAR) {
-                    if ((Val & 0xFF) != 0xFF) {
+                    if ((Val & 0xFF) == 0x00) {
+                        AddCodeLine ("lda #$00");
+                    } else if ((Val & 0xFF) != 0xFF) {
                         AddCodeLine ("and #$%02X", (unsigned char)Val);
                     }
-                           return;
-                       }
-                       /* FALLTHROUGH */
+                    return;
+                }
+                /* FALLTHROUGH */
             case CF_INT:
-                       if ((Val & 0xFFFF) != 0xFFFF) {
-                           if (Val <= 0xFF) {
-                               ldxconst (0);
-                               if (Val == 0) {
-                                   ldaconst (0);
-                               } else if (Val != 0xFF) {
-                                   AddCodeLine ("and #$%02X", (unsigned char)Val);
-                               }
-                           } else if ((Val & 0xFF00) == 0xFF00) {
-                               AddCodeLine ("and #$%02X", (unsigned char)Val);
-                           } else if ((Val & 0x00FF) == 0x0000) {
-                               AddCodeLine ("txa");
-                               AddCodeLine ("and #$%02X", (unsigned char)(Val >> 8));
-                       AddCodeLine ("tax");
-                       ldaconst (0);
+                if ((Val & 0xFFFF) != 0xFFFF) {
+                    if (Val <= 0xFF) {
+                        AddCodeLine ("ldx #$00");
+                        if (Val == 0) {
+                            AddCodeLine ("lda #$00");
+                        } else if (Val != 0xFF) {
+                            AddCodeLine ("and #$%02X", (unsigned char)Val);
+                        }
+                    } else if ((Val & 0xFFFF) == 0xFF00) {
+                        AddCodeLine ("lda #$00");
+                    } else if ((Val & 0xFF00) == 0xFF00) {
+                        AddCodeLine ("and #$%02X", (unsigned char)Val);
+                    } else if ((Val & 0x00FF) == 0x0000) {
+                        AddCodeLine ("txa");
+                        AddCodeLine ("and #$%02X", (unsigned char)(Val >> 8));
+                        AddCodeLine ("tax");
+                        AddCodeLine ("lda #$00");
                     } else {
-                       AddCodeLine ("tay");
-                       AddCodeLine ("txa");
-                       AddCodeLine ("and #$%02X", (unsigned char)(Val >> 8));
-                       AddCodeLine ("tax");
-                       AddCodeLine ("tya");
-                       if ((Val & 0x00FF) != 0x00FF) {
-                                   AddCodeLine ("and #$%02X", (unsigned char)Val);
-                       }
+                        AddCodeLine ("tay");
+                        AddCodeLine ("txa");
+                        AddCodeLine ("and #$%02X", (unsigned char)(Val >> 8));
+                        AddCodeLine ("tax");
+                        AddCodeLine ("tya");
+                        if ((Val & 0x00FF) == 0x0000) {
+                            AddCodeLine ("lda #$00");
+                        } else if ((Val & 0x00FF) != 0x00FF) {
+                            AddCodeLine ("and #$%02X", (unsigned char)Val);
+                        }
                     }
                 }
                 return;
 
             case CF_LONG:
                 if (Val <= 0xFF) {
-                    ldxconst (0);
+                    AddCodeLine ("ldx #$00");
                     AddCodeLine ("stx sreg+1");
-                   AddCodeLine ("stx sreg");
+                    AddCodeLine ("stx sreg");
                     if ((Val & 0xFF) != 0xFF) {
-                        AddCodeLine ("and #$%02X", (unsigned char)Val);
+                         AddCodeLine ("and #$%02X", (unsigned char)Val);
                     }
                     return;
                 } else if (Val == 0xFF00) {
-                    ldaconst (0);
+                    AddCodeLine ("lda #$00");
                     AddCodeLine ("sta sreg+1");
                     AddCodeLine ("sta sreg");
                     return;
@@ -2987,9 +2927,9 @@ void g_and (unsigned Flags, unsigned long Val)
         }
 
         /* If we go here, we didn't emit code. Push the lhs on stack and fall
-         * into the normal, non-optimized stuff. Note: The standard stuff will
-         * always work with ints.
-         */
+        ** into the normal, non-optimized stuff. Note: The standard stuff will
+        ** always work with ints.
+        */
         Flags &= ~CF_FORCECHAR;
         g_push (Flags & ~CF_CONST, 0);
     }
@@ -3003,83 +2943,124 @@ void g_and (unsigned Flags, unsigned long Val)
 void g_asr (unsigned flags, unsigned long val)
 /* Primary = TOS >> Primary */
 {
-    static char* ops [12] = {
-        0,             0,              "tosasrax",
-        0,             0,              "tosshrax",
-        0,             0,              "tosasreax",
-        0,             0,              "tosshreax",
+    static const char* const ops[4] = {
+        "tosasrax", "tosshrax", "tosasreax", "tosshreax"
     };
 
     /* If the right hand side is const, the lhs is not on stack but still
-     * in the primary register.
-     */
+    ** in the primary register.
+    */
     if (flags & CF_CONST) {
 
-        switch (flags & CF_TYPE) {
+        switch (flags & CF_TYPEMASK) {
 
             case CF_CHAR:
             case CF_INT:
-                       if (val >= 8 && (flags & CF_UNSIGNED)) {
-                    AddCodeLine ("txa");
-                    ldxconst (0);
+                val &= 0x0F;
+                if (val >= 8) {
+                    if (flags & CF_UNSIGNED) {
+                        AddCodeLine ("txa");
+                        AddCodeLine ("ldx #$00");
+                    } else {
+                        unsigned L = GetLocalLabel();
+                        AddCodeLine ("cpx #$80");   /* Sign bit into carry */
+                        AddCodeLine ("txa");
+                        AddCodeLine ("ldx #$00");
+                        AddCodeLine ("bcc %s", LocalLabelName (L));
+                        AddCodeLine ("dex");        /* Make $FF */
+                        g_defcodelabel (L);
+                    }
                     val -= 8;
                 }
-                if (val == 0) {
-                    /* Done */
-                    return;
-                } else if (val >= 1 && val <= 4) {
+                if (val >= 4) {
                     if (flags & CF_UNSIGNED) {
-                               AddCodeLine ("jsr shrax%ld", val);
+                        AddCodeLine ("jsr shrax4");
                     } else {
-                               AddCodeLine ("jsr asrax%ld", val);
+                        AddCodeLine ("jsr asrax4");
                     }
-                    return;
-                       }
-                break;
-
-            case CF_LONG:
-                if (val == 0) {
-                    /* Nothing to do */
-                    return;
-                } else if (val >= 1 && val <= 4) {
+                    val -= 4;
+                }
+                if (val > 0) {
                     if (flags & CF_UNSIGNED) {
-                               AddCodeLine ("jsr shreax%ld", val);
+                        AddCodeLine ("jsr shrax%ld", val);
                     } else {
-                               AddCodeLine ("jsr asreax%ld", val);
+                        AddCodeLine ("jsr asrax%ld", val);
                     }
-                    return;
-                } else if (val == 8 && (flags & CF_UNSIGNED)) {
-                    AddCodeLine ("txa");
-                    AddCodeLine ("ldx sreg");
-                    AddCodeLine ("ldy sreg+1");
-                    AddCodeLine ("sty sreg");
-                    AddCodeLine ("ldy #$00");
-                    AddCodeLine ("sty sreg+1");
-                    return;
-                } else if (val == 16) {
+                }
+                return;
+
+            case CF_LONG:
+                val &= 0x1F;
+                if (val >= 24) {
+                    AddCodeLine ("ldx #$00");
+                    AddCodeLine ("lda sreg+1");
+                    if ((flags & CF_UNSIGNED) == 0) {
+                        unsigned L = GetLocalLabel();
+                        AddCodeLine ("bpl %s", LocalLabelName (L));
+                        AddCodeLine ("dex");
+                        g_defcodelabel (L);
+                    }
+                    AddCodeLine ("stx sreg");
+                    AddCodeLine ("stx sreg+1");
+                    val -= 24;
+                }
+                if (val >= 16) {
                     AddCodeLine ("ldy #$00");
                     AddCodeLine ("ldx sreg+1");
                     if ((flags & CF_UNSIGNED) == 0) {
-                               unsigned L = GetLocalLabel();
+                        unsigned L = GetLocalLabel();
                         AddCodeLine ("bpl %s", LocalLabelName (L));
                         AddCodeLine ("dey");
-                               g_defcodelabel (L);
+                        g_defcodelabel (L);
                     }
                     AddCodeLine ("lda sreg");
                     AddCodeLine ("sty sreg+1");
                     AddCodeLine ("sty sreg");
-                   return;
+                    val -= 16;
                 }
-                break;
+                if (val >= 8) {
+                    AddCodeLine ("txa");
+                    AddCodeLine ("ldx sreg");
+                    AddCodeLine ("ldy sreg+1");
+                    AddCodeLine ("sty sreg");
+                    if ((flags & CF_UNSIGNED) == 0) {
+                        unsigned L = GetLocalLabel();
+                        AddCodeLine ("cpy #$80");
+                        AddCodeLine ("ldy #$00");
+                        AddCodeLine ("bcc %s", LocalLabelName (L));
+                        AddCodeLine ("dey");
+                        g_defcodelabel (L);
+                    } else {
+                        AddCodeLine ("ldy #$00");
+                    }
+                    AddCodeLine ("sty sreg+1");
+                    val -= 8;
+                }
+                if (val >= 4) {
+                    if (flags & CF_UNSIGNED) {
+                        AddCodeLine ("jsr shreax4");
+                    } else {
+                        AddCodeLine ("jsr asreax4");
+                    }
+                    val -= 4;
+                }
+                if (val > 0) {
+                    if (flags & CF_UNSIGNED) {
+                        AddCodeLine ("jsr shreax%ld", val);
+                    } else {
+                        AddCodeLine ("jsr asreax%ld", val);
+                    }
+                }
+                return;
 
             default:
                 typeerror (flags);
         }
 
         /* If we go here, we didn't emit code. Push the lhs on stack and fall
-         * into the normal, non-optimized stuff. Note: The standard stuff will
-         * always work with ints.
-         */
+        ** into the normal, non-optimized stuff. Note: The standard stuff will
+        ** always work with ints.
+        */
         flags &= ~CF_FORCECHAR;
         g_push (flags & ~CF_CONST, 0);
     }
@@ -3093,85 +3074,92 @@ void g_asr (unsigned flags, unsigned long val)
 void g_asl (unsigned flags, unsigned long val)
 /* Primary = TOS << Primary */
 {
-    static char* ops [12] = {
-        0,                     0,              "tosaslax",
-        0,             0,              "tosshlax",
-        0,             0,              "tosasleax",
-        0,             0,              "tosshleax",
+    static const char* const ops[4] = {
+        "tosaslax", "tosshlax", "tosasleax", "tosshleax"
     };
 
 
     /* If the right hand side is const, the lhs is not on stack but still
-     * in the primary register.
-     */
+    ** in the primary register.
+    */
     if (flags & CF_CONST) {
 
-        switch (flags & CF_TYPE) {
+        switch (flags & CF_TYPEMASK) {
 
             case CF_CHAR:
             case CF_INT:
+                val &= 0x0F;
                 if (val >= 8) {
                     AddCodeLine ("tax");
                     AddCodeLine ("lda #$00");
                     val -= 8;
                 }
-                if (val == 0) {
-                    /* Done */
-                    return;
-                } else if (val >= 1 && val <= 4) {
-                    if (IS_Get (&CodeSizeFactor) >= (long) (val+1)*130) {
-                       AddCodeLine ("stx tmp1");
-                       while (val--) {
-                           AddCodeLine ("asl a");
-                           AddCodeLine ("rol tmp1");
-                       }
-                       AddCodeLine ("ldx tmp1");
-                   } else {
-                       if (flags & CF_UNSIGNED) {
-                           AddCodeLine ("jsr shlax%ld", val);
-                       } else {
-                           AddCodeLine ("jsr aslax%ld", val);
-                       }
-                   }
-                    return;
+                if (val >= 4) {
+                    if (flags & CF_UNSIGNED) {
+                        AddCodeLine ("jsr shlax4");
+                    } else {
+                        AddCodeLine ("jsr aslax4");
+                    }
+                    val -= 4;
                 }
-                break;
-
-            case CF_LONG:
-                if (val == 0) {
-                    /* Nothing to do */
-                    return;
-                } else if (val >= 1 && val <= 4) {
+                if (val > 0) {
                     if (flags & CF_UNSIGNED) {
-                               AddCodeLine ("jsr shleax%ld", val);
+                        AddCodeLine ("jsr shlax%ld", val);
                     } else {
-                               AddCodeLine ("jsr asleax%ld", val);
+                        AddCodeLine ("jsr aslax%ld", val);
                     }
-                    return;
-                } else if (val == 8) {
-                    AddCodeLine ("ldy sreg");
-                    AddCodeLine ("sty sreg+1");
-                    AddCodeLine ("stx sreg");
-                    AddCodeLine ("tax");
+                }
+                return;
+
+            case CF_LONG:
+                val &= 0x1F;
+                if (val >= 24) {
+                    AddCodeLine ("sta sreg+1");
                     AddCodeLine ("lda #$00");
-                    return;
-                } else if (val == 16) {
+                    AddCodeLine ("tax");
+                    AddCodeLine ("sta sreg");
+                    val -= 24;
+                }
+                if (val >= 16) {
                     AddCodeLine ("stx sreg+1");
                     AddCodeLine ("sta sreg");
                     AddCodeLine ("lda #$00");
                     AddCodeLine ("tax");
-                    return;
+                    val -= 16;
                 }
-                break;
+                if (val >= 8) {
+                    AddCodeLine ("ldy sreg");
+                    AddCodeLine ("sty sreg+1");
+                    AddCodeLine ("stx sreg");
+                    AddCodeLine ("tax");
+                    AddCodeLine ("lda #$00");
+                    val -= 8;
+                }
+                if (val > 4) {
+                    if (flags & CF_UNSIGNED) {
+                        AddCodeLine ("jsr shleax4");
+                    } else {
+                        AddCodeLine ("jsr asleax4");
+                    }
+                    val -= 4;
+                }
+                if (val > 0) {
+                    if (flags & CF_UNSIGNED) {
+                        AddCodeLine ("jsr shleax%ld", val);
+                    } else {
+                        AddCodeLine ("jsr asleax%ld", val);
+                    }
+                }
+                return;
 
             default:
                 typeerror (flags);
         }
 
         /* If we go here, we didn't emit code. Push the lhs on stack and fall
-         * into the normal, non-optimized stuff. Note: The standard stuff will
-         * always work with ints.
-         */
+        ** into the normal, non-optimized stuff. Note: The standard stuff will
+        ** always work with ints.
+        */
         flags &= ~CF_FORCECHAR;
         g_push (flags & ~CF_CONST, 0);
     }
@@ -3185,7 +3173,7 @@ void g_asl (unsigned flags, unsigned long val)
 void g_neg (unsigned Flags)
 /* Primary = -Primary */
 {
-    switch (Flags & CF_TYPE) {
+    switch (Flags & CF_TYPEMASK) {
 
         case CF_CHAR:
             if (Flags & CF_FORCECHAR) {
@@ -3214,7 +3202,7 @@ void g_neg (unsigned Flags)
 void g_bneg (unsigned flags)
 /* Primary = !Primary */
 {
-    switch (flags & CF_TYPE) {
+    switch (flags & CF_TYPEMASK) {
 
         case CF_CHAR:
             AddCodeLine ("jsr bnega");
@@ -3238,7 +3226,7 @@ void g_bneg (unsigned flags)
 void g_com (unsigned Flags)
 /* Primary = ~Primary */
 {
-    switch (Flags & CF_TYPE) {
+    switch (Flags & CF_TYPEMASK) {
 
         case CF_CHAR:
             if (Flags & CF_FORCECHAR) {
@@ -3272,15 +3260,15 @@ void g_inc (unsigned flags, unsigned long val)
 
     /* Generate code for the supported types */
     flags &= ~CF_CONST;
-    switch (flags & CF_TYPE) {
+    switch (flags & CF_TYPEMASK) {
 
         case CF_CHAR:
             if (flags & CF_FORCECHAR) {
                 if ((CPUIsets[CPU] & CPU_ISET_65SC02) != 0 && val <= 2) {
                     while (val--) {
-                               AddCodeLine ("ina");
+                        AddCodeLine ("ina");
                     }
-               } else {
+                } else {
                     AddCodeLine ("clc");
                     AddCodeLine ("adc #$%02X", (unsigned char)val);
                 }
@@ -3300,7 +3288,7 @@ void g_inc (unsigned flags, unsigned long val)
                 if (val <= 8) {
                     AddCodeLine ("jsr incax%lu", val);
                 } else if (val <= 255) {
-                    ldyconst (val);
+                    AddCodeLine ("ldy #$%02X", (unsigned char) val);
                     AddCodeLine ("jsr incaxy");
                 } else {
                     g_add (flags | CF_CONST, val);
@@ -3309,29 +3297,34 @@ void g_inc (unsigned flags, unsigned long val)
                 /* Inline the code */
                 if (val <= 0x300) {
                     if ((val & 0xFF) != 0) {
-                               unsigned L = GetLocalLabel();
-                               AddCodeLine ("clc");
-                               AddCodeLine ("adc #$%02X", (unsigned char) val);
-                               AddCodeLine ("bcc %s", LocalLabelName (L));
-                               AddCodeLine ("inx");
-                               g_defcodelabel (L);
+                        unsigned L = GetLocalLabel();
+                        AddCodeLine ("clc");
+                        AddCodeLine ("adc #$%02X", (unsigned char) val);
+                        AddCodeLine ("bcc %s", LocalLabelName (L));
+                        AddCodeLine ("inx");
+                        g_defcodelabel (L);
                     }
                     if (val >= 0x100) {
-                               AddCodeLine ("inx");
+                        AddCodeLine ("inx");
                     }
                     if (val >= 0x200) {
-                               AddCodeLine ("inx");
+                        AddCodeLine ("inx");
                     }
                     if (val >= 0x300) {
-                               AddCodeLine ("inx");
+                        AddCodeLine ("inx");
                     }
-                } else {
+                } else if ((val & 0xFF) != 0) {
                     AddCodeLine ("clc");
-                    if ((val & 0xFF) != 0) {
-                       AddCodeLine ("adc #$%02X", (unsigned char) val);
-                    }
+                    AddCodeLine ("adc #$%02X", (unsigned char) val);
+                    AddCodeLine ("pha");
+                    AddCodeLine ("txa");
+                    AddCodeLine ("adc #$%02X", (unsigned char) (val >> 8));
+                    AddCodeLine ("tax");
+                    AddCodeLine ("pla");
+                } else {
                     AddCodeLine ("pha");
                     AddCodeLine ("txa");
+                    AddCodeLine ("clc");
                     AddCodeLine ("adc #$%02X", (unsigned char) (val >> 8));
                     AddCodeLine ("tax");
                     AddCodeLine ("pla");
@@ -3341,7 +3334,7 @@ void g_inc (unsigned flags, unsigned long val)
 
         case CF_LONG:
             if (val <= 255) {
-                ldyconst (val);
+                AddCodeLine ("ldy #$%02X", (unsigned char) val);
                 AddCodeLine ("jsr inceaxy");
             } else {
                 g_add (flags | CF_CONST, val);
@@ -3366,17 +3359,17 @@ void g_dec (unsigned flags, unsigned long val)
 
     /* Generate code for the supported types */
     flags &= ~CF_CONST;
-    switch (flags & CF_TYPE) {
+    switch (flags & CF_TYPEMASK) {
 
         case CF_CHAR:
             if (flags & CF_FORCECHAR) {
                 if ((CPUIsets[CPU] & CPU_ISET_65SC02) != 0 && val <= 2) {
                     while (val--) {
-                       AddCodeLine ("dea");
+                        AddCodeLine ("dea");
                     }
                 } else {
                     AddCodeLine ("sec");
-                   AddCodeLine ("sbc #$%02X", (unsigned char)val);
+                    AddCodeLine ("sbc #$%02X", (unsigned char)val);
                 }
                 break;
             }
@@ -3388,7 +3381,7 @@ void g_dec (unsigned flags, unsigned long val)
                 if (val <= 8) {
                     AddCodeLine ("jsr decax%d", (int) val);
                 } else if (val <= 255) {
-                    ldyconst (val);
+                    AddCodeLine ("ldy #$%02X", (unsigned char) val);
                     AddCodeLine ("jsr decaxy");
                 } else {
                     g_sub (flags | CF_CONST, val);
@@ -3398,35 +3391,42 @@ void g_dec (unsigned flags, unsigned long val)
                 if (val < 0x300) {
                     if ((val & 0xFF) != 0) {
                         unsigned L = GetLocalLabel();
-                               AddCodeLine ("sec");
-                               AddCodeLine ("sbc #$%02X", (unsigned char) val);
-                               AddCodeLine ("bcs %s", LocalLabelName (L));
-                               AddCodeLine ("dex");
+                        AddCodeLine ("sec");
+                        AddCodeLine ("sbc #$%02X", (unsigned char) val);
+                        AddCodeLine ("bcs %s", LocalLabelName (L));
+                        AddCodeLine ("dex");
                         g_defcodelabel (L);
                     }
                     if (val >= 0x100) {
-                               AddCodeLine ("dex");
+                        AddCodeLine ("dex");
                     }
                     if (val >= 0x200) {
-                               AddCodeLine ("dex");
+                        AddCodeLine ("dex");
                     }
                 } else {
-                    AddCodeLine ("sec");
                     if ((val & 0xFF) != 0) {
-                       AddCodeLine ("sbc #$%02X", (unsigned char) val);
+                        AddCodeLine ("sec");
+                        AddCodeLine ("sbc #$%02X", (unsigned char) val);
+                        AddCodeLine ("pha");
+                        AddCodeLine ("txa");
+                        AddCodeLine ("sbc #$%02X", (unsigned char) (val >> 8));
+                        AddCodeLine ("tax");
+                        AddCodeLine ("pla");
+                    } else {
+                        AddCodeLine ("pha");
+                        AddCodeLine ("txa");
+                        AddCodeLine ("sec");
+                        AddCodeLine ("sbc #$%02X", (unsigned char) (val >> 8));
+                        AddCodeLine ("tax");
+                        AddCodeLine ("pla");
                     }
-                    AddCodeLine ("pha");
-                    AddCodeLine ("txa");
-                    AddCodeLine ("sbc #$%02X", (unsigned char) (val >> 8));
-                    AddCodeLine ("tax");
-                    AddCodeLine ("pla");
                 }
             }
             break;
 
         case CF_LONG:
             if (val <= 255) {
-                ldyconst (val);
+                AddCodeLine ("ldy #$%02X", (unsigned char) val);
                 AddCodeLine ("jsr deceaxy");
             } else {
                 g_sub (flags | CF_CONST, val);
@@ -3442,31 +3442,28 @@ void g_dec (unsigned flags, unsigned long val)
 
 
 /*
- * Following are the conditional operators. They compare the TOS against
- * the primary and put a literal 1 in the primary if the condition is
- * true, otherwise they clear the primary register
- */
+** Following are the conditional operators. They compare the TOS against
+** the primary and put a literal 1 in the primary if the condition is
+** true, otherwise they clear the primary register
+*/
 
 
 
 void g_eq (unsigned flags, unsigned long val)
 /* Test for equal */
 {
-    static char* ops [12] = {
-        "toseq00",     "toseqa0",      "toseqax",
-        "toseq00",     "toseqa0",      "toseqax",
-        0,             0,              "toseqeax",
-        0,             0,              "toseqeax",
+    static const char* const ops[4] = {
+        "toseqax", "toseqax", "toseqeax", "toseqeax"
     };
 
     unsigned L;
 
     /* If the right hand side is const, the lhs is not on stack but still
-     * in the primary register.
-     */
+    ** in the primary register.
+    */
     if (flags & CF_CONST) {
 
-        switch (flags & CF_TYPE) {
+        switch (flags & CF_TYPEMASK) {
 
             case CF_CHAR:
                 if (flags & CF_FORCECHAR) {
@@ -3479,7 +3476,7 @@ void g_eq (unsigned flags, unsigned long val)
             case CF_INT:
                 L = GetLocalLabel();
                 AddCodeLine ("cpx #$%02X", (unsigned char)(val >> 8));
-                       AddCodeLine ("bne %s", LocalLabelName (L));
+                AddCodeLine ("bne %s", LocalLabelName (L));
                 AddCodeLine ("cmp #$%02X", (unsigned char)val);
                 g_defcodelabel (L);
                 AddCodeLine ("jsr booleq");
@@ -3493,9 +3490,9 @@ void g_eq (unsigned flags, unsigned long val)
         }
 
         /* If we go here, we didn't emit code. Push the lhs on stack and fall
-         * into the normal, non-optimized stuff. Note: The standard stuff will
-         * always work with ints.
-         */
+        ** into the normal, non-optimized stuff. Note: The standard stuff will
+        ** always work with ints.
+        */
         flags &= ~CF_FORCECHAR;
         g_push (flags & ~CF_CONST, 0);
     }
@@ -3509,21 +3506,18 @@ void g_eq (unsigned flags, unsigned long val)
 void g_ne (unsigned flags, unsigned long val)
 /* Test for not equal */
 {
-    static char* ops [12] = {
-        "tosne00",     "tosnea0",      "tosneax",
-        "tosne00",     "tosnea0",      "tosneax",
-        0,             0,              "tosneeax",
-        0,             0,              "tosneeax",
+    static const char* const ops[4] = {
+        "tosneax", "tosneax", "tosneeax", "tosneeax"
     };
 
     unsigned L;
 
     /* If the right hand side is const, the lhs is not on stack but still
-     * in the primary register.
-     */
+    ** in the primary register.
+    */
     if (flags & CF_CONST) {
 
-        switch (flags & CF_TYPE) {
+        switch (flags & CF_TYPEMASK) {
 
             case CF_CHAR:
                 if (flags & CF_FORCECHAR) {
@@ -3550,9 +3544,9 @@ void g_ne (unsigned flags, unsigned long val)
         }
 
         /* If we go here, we didn't emit code. Push the lhs on stack and fall
-         * into the normal, non-optimized stuff. Note: The standard stuff will
-         * always work with ints.
-         */
+        ** into the normal, non-optimized stuff. Note: The standard stuff will
+        ** always work with ints.
+        */
         flags &= ~CF_FORCECHAR;
         g_push (flags & ~CF_CONST, 0);
     }
@@ -3566,22 +3560,21 @@ void g_ne (unsigned flags, unsigned long val)
 void g_lt (unsigned flags, unsigned long val)
 /* Test for less than */
 {
-    static char* ops [12] = {
-        "toslt00",     "toslta0",      "tosltax",
-        "tosult00",    "tosulta0",     "tosultax",
-        0,             0,              "toslteax",
-        0,             0,              "tosulteax",
+    static const char* const ops[4] = {
+        "tosltax", "tosultax", "toslteax", "tosulteax"
     };
 
+    unsigned Label;
+
     /* If the right hand side is const, the lhs is not on stack but still
-     * in the primary register.
-     */
+    ** in the primary register.
+    */
     if (flags & CF_CONST) {
 
         /* Because the handling of the overflow flag is too complex for
-         * inlining, we can handle only unsigned compares, and signed
-         * compares against zero here.
-         */
+        ** inlining, we can handle only unsigned compares, and signed
+        ** compares against zero here.
+        */
         if (flags & CF_UNSIGNED) {
 
             /* Give a warning in some special cases */
@@ -3592,7 +3585,7 @@ void g_lt (unsigned flags, unsigned long val)
             }
 
             /* Look at the type */
-            switch (flags & CF_TYPE) {
+            switch (flags & CF_TYPEMASK) {
 
                 case CF_CHAR:
                     if (flags & CF_FORCECHAR) {
@@ -3632,38 +3625,90 @@ void g_lt (unsigned flags, unsigned long val)
 
         } else if (val == 0) {
 
-            /* Look at the type */
-            switch (flags & CF_TYPE) {
+            /* A signed compare against zero must only look at the sign bit */
+            switch (flags & CF_TYPEMASK) {
 
                 case CF_CHAR:
                     if (flags & CF_FORCECHAR) {
-                        AddCodeLine ("tax");
-                        AddCodeLine ("jsr boollt");
+                        AddCodeLine ("asl a");          /* Bit 7 -> carry */
+                        AddCodeLine ("lda #$00");
+                        AddCodeLine ("ldx #$00");
+                        AddCodeLine ("rol a");
                         return;
                     }
                     /* FALLTHROUGH */
 
                 case CF_INT:
                     /* Just check the high byte */
-                    AddCodeLine ("txa");
-                    AddCodeLine ("jsr boollt");
+                    AddCodeLine ("cpx #$80");           /* Bit 7 -> carry */
+                    AddCodeLine ("lda #$00");
+                    AddCodeLine ("ldx #$00");
+                    AddCodeLine ("rol a");
                     return;
 
                 case CF_LONG:
                     /* Just check the high byte */
                     AddCodeLine ("lda sreg+1");
-                    AddCodeLine ("jsr boollt");
+                    AddCodeLine ("asl a");              /* Bit 7 -> carry */
+                    AddCodeLine ("lda #$00");
+                    AddCodeLine ("ldx #$00");
+                    AddCodeLine ("rol a");
                     return;
 
                 default:
                     typeerror (flags);
             }
+
+        } else {
+
+            /* Signed compare against a constant != zero */
+            switch (flags & CF_TYPEMASK) {
+
+                case CF_CHAR:
+                    if (flags & CF_FORCECHAR) {
+                        Label = GetLocalLabel ();
+                        AddCodeLine ("sec");
+                        AddCodeLine ("sbc #$%02X", (unsigned char)val);
+                        AddCodeLine ("bvc %s", LocalLabelName (Label));
+                        AddCodeLine ("eor #$80");
+                        g_defcodelabel (Label);
+                        AddCodeLine ("asl a");          /* Bit 7 -> carry */
+                        AddCodeLine ("lda #$00");
+                        AddCodeLine ("ldx #$00");
+                        AddCodeLine ("rol a");
+                        return;
+                    }
+                    /* FALLTHROUGH */
+
+                case CF_INT:
+                    /* Do a subtraction */
+                    Label = GetLocalLabel ();
+                    AddCodeLine ("cmp #$%02X", (unsigned char)val);
+                    AddCodeLine ("txa");
+                    AddCodeLine ("sbc #$%02X", (unsigned char)(val >> 8));
+                    AddCodeLine ("bvc %s", LocalLabelName (Label));
+                    AddCodeLine ("eor #$80");
+                    g_defcodelabel (Label);
+                    AddCodeLine ("asl a");          /* Bit 7 -> carry */
+                    AddCodeLine ("lda #$00");
+                    AddCodeLine ("ldx #$00");
+                    AddCodeLine ("rol a");
+                    return;
+
+                case CF_LONG:
+                    /* This one is too costly */
+                    break;
+
+                default:
+                    typeerror (flags);
+            }
+
         }
 
         /* If we go here, we didn't emit code. Push the lhs on stack and fall
-         * into the normal, non-optimized stuff. Note: The standard stuff will
-         * always work with ints.
-         */
+        ** into the normal, non-optimized stuff. Note: The standard stuff will
+        ** always work with ints.
+        */
         flags &= ~CF_FORCECHAR;
         g_push (flags & ~CF_CONST, 0);
     }
@@ -3677,48 +3722,45 @@ void g_lt (unsigned flags, unsigned long val)
 void g_le (unsigned flags, unsigned long val)
 /* Test for less than or equal to */
 {
-    static char* ops [12] = {
-        "tosle00",     "toslea0",      "tosleax",
-        "tosule00",    "tosulea0",     "tosuleax",
-        0,             0,              "tosleeax",
-        0,             0,              "tosuleeax",
+    static const char* const ops[4] = {
+        "tosleax", "tosuleax", "tosleeax", "tosuleeax"
     };
 
 
     /* If the right hand side is const, the lhs is not on stack but still
-     * in the primary register.
-     */
+    ** in the primary register.
+    */
     if (flags & CF_CONST) {
 
         /* Look at the type */
-        switch (flags & CF_TYPE) {
+        switch (flags & CF_TYPEMASK) {
 
             case CF_CHAR:
                 if (flags & CF_FORCECHAR) {
                     if (flags & CF_UNSIGNED) {
                         /* Unsigned compare */
-                       if (val < 0xFF) {
-                           /* Use < instead of <= because the former gives
-                            * better code on the 6502 than the latter.
-                            */
+                        if (val < 0xFF) {
+                            /* Use < instead of <= because the former gives
+                            ** better code on the 6502 than the latter.
+                            */
                             g_lt (flags, val+1);
-                       } else {
+                        } else {
                             /* Always true */
                             Warning ("Condition is always true");
                             AddCodeLine ("jsr return1");
-                       }
+                        }
                     } else {
                         /* Signed compare */
-                       if ((long) val < 0x7F) {
-                           /* Use < instead of <= because the former gives
-                            * better code on the 6502 than the latter.
-                            */
+                        if ((long) val < 0x7F) {
+                            /* Use < instead of <= because the former gives
+                            ** better code on the 6502 than the latter.
+                            */
                             g_lt (flags, val+1);
-                       } else {
+                        } else {
                             /* Always true */
                             Warning ("Condition is always true");
                             AddCodeLine ("jsr return1");
-                       }
+                        }
                     }
                     return;
                 }
@@ -3728,10 +3770,10 @@ void g_le (unsigned flags, unsigned long val)
                 if (flags & CF_UNSIGNED) {
                     /* Unsigned compare */
                     if (val < 0xFFFF) {
-                       /* Use < instead of <= because the former gives
-                        * better code on the 6502 than the latter.
-                        */
-                       g_lt (flags, val+1);
+                        /* Use < instead of <= because the former gives
+                        ** better code on the 6502 than the latter.
+                        */
+                        g_lt (flags, val+1);
                     } else {
                         /* Always true */
                         Warning ("Condition is always true");
@@ -3740,7 +3782,7 @@ void g_le (unsigned flags, unsigned long val)
                 } else {
                     /* Signed compare */
                     if ((long) val < 0x7FFF) {
-                               g_lt (flags, val+1);
+                        g_lt (flags, val+1);
                     } else {
                         /* Always true */
                         Warning ("Condition is always true");
@@ -3753,10 +3795,10 @@ void g_le (unsigned flags, unsigned long val)
                 if (flags & CF_UNSIGNED) {
                     /* Unsigned compare */
                     if (val < 0xFFFFFFFF) {
-                       /* Use < instead of <= because the former gives
-                        * better code on the 6502 than the latter.
-                        */
-                       g_lt (flags, val+1);
+                        /* Use < instead of <= because the former gives
+                        ** better code on the 6502 than the latter.
+                        */
+                        g_lt (flags, val+1);
                     } else {
                         /* Always true */
                         Warning ("Condition is always true");
@@ -3765,7 +3807,7 @@ void g_le (unsigned flags, unsigned long val)
                 } else {
                     /* Signed compare */
                     if ((long) val < 0x7FFFFFFF) {
-                               g_lt (flags, val+1);
+                        g_lt (flags, val+1);
                     } else {
                         /* Always true */
                         Warning ("Condition is always true");
@@ -3779,9 +3821,9 @@ void g_le (unsigned flags, unsigned long val)
         }
 
         /* If we go here, we didn't emit code. Push the lhs on stack and fall
-         * into the normal, non-optimized stuff. Note: The standard stuff will
-         * always work with ints.
-         */
+        ** into the normal, non-optimized stuff. Note: The standard stuff will
+        ** always work with ints.
+        */
         flags &= ~CF_FORCECHAR;
         g_push (flags & ~CF_CONST, 0);
     }
@@ -3795,46 +3837,43 @@ void g_le (unsigned flags, unsigned long val)
 void g_gt (unsigned flags, unsigned long val)
 /* Test for greater than */
 {
-    static char* ops [12] = {
-        "tosgt00",     "tosgta0",      "tosgtax",
-        "tosugt00",    "tosugta0",     "tosugtax",
-        0,             0,              "tosgteax",
-        0,             0,              "tosugteax",
+    static const char* const ops[4] = {
+        "tosgtax", "tosugtax", "tosgteax", "tosugteax"
     };
 
 
     /* If the right hand side is const, the lhs is not on stack but still
-     * in the primary register.
-     */
+    ** in the primary register.
+    */
     if (flags & CF_CONST) {
 
         /* Look at the type */
-        switch (flags & CF_TYPE) {
+        switch (flags & CF_TYPEMASK) {
 
             case CF_CHAR:
                 if (flags & CF_FORCECHAR) {
                     if (flags & CF_UNSIGNED) {
-                               if (val == 0) {
-                           /* If we have a compare > 0, we will replace it by
-                            * != 0 here, since both are identical but the
-                             * latter is easier to optimize.
-                            */
+                        if (val == 0) {
+                            /* If we have a compare > 0, we will replace it by
+                            ** != 0 here, since both are identical but the
+                            ** latter is easier to optimize.
+                            */
                             g_ne (flags, val);
-                       } else if (val < 0xFF) {
-                           /* Use >= instead of > because the former gives
-                            * better code on the 6502 than the latter.
-                            */
+                        } else if (val < 0xFF) {
+                            /* Use >= instead of > because the former gives
+                            ** better code on the 6502 than the latter.
+                            */
                             g_ge (flags, val+1);
-                       } else {
+                        } else {
                             /* Never true */
                             Warning ("Condition is never true");
                             AddCodeLine ("jsr return0");
                         }
                     } else {
-                       if ((long) val < 0x7F) {
-                           /* Use >= instead of > because the former gives
-                            * better code on the 6502 than the latter.
-                            */
+                        if ((long) val < 0x7F) {
+                            /* Use >= instead of > because the former gives
+                            ** better code on the 6502 than the latter.
+                            */
                             g_ge (flags, val+1);
                         } else {
                             /* Never true */
@@ -3849,23 +3888,23 @@ void g_gt (unsigned flags, unsigned long val)
             case CF_INT:
                 if (flags & CF_UNSIGNED) {
                     /* Unsigned compare */
-                   if (val == 0) {
-                       /* If we have a compare > 0, we will replace it by
-                        * != 0 here, since both are identical but the latter
-                        * is easier to optimize.
-                        */
+                    if (val == 0) {
+                        /* If we have a compare > 0, we will replace it by
+                        ** != 0 here, since both are identical but the latter
+                        ** is easier to optimize.
+                        */
                         g_ne (flags, val);
-                   } else if (val < 0xFFFF) {
-                       /* Use >= instead of > because the former gives better
-                        * code on the 6502 than the latter.
-                        */
-                       g_ge (flags, val+1);
+                    } else if (val < 0xFFFF) {
+                        /* Use >= instead of > because the former gives better
+                        ** code on the 6502 than the latter.
+                        */
+                        g_ge (flags, val+1);
                     } else {
                         /* Never true */
                         Warning ("Condition is never true");
                         AddCodeLine ("jsr return0");
-                   }
-                       } else {
+                    }
+                } else {
                     /* Signed compare */
                     if ((long) val < 0x7FFF) {
                         g_ge (flags, val+1);
@@ -3873,30 +3912,30 @@ void g_gt (unsigned flags, unsigned long val)
                         /* Never true */
                         Warning ("Condition is never true");
                         AddCodeLine ("jsr return0");
-                   }
+                    }
                 }
                 return;
 
             case CF_LONG:
                 if (flags & CF_UNSIGNED) {
                     /* Unsigned compare */
-                   if (val == 0) {
-                       /* If we have a compare > 0, we will replace it by
-                        * != 0 here, since both are identical but the latter
-                        * is easier to optimize.
-                        */
+                    if (val == 0) {
+                        /* If we have a compare > 0, we will replace it by
+                        ** != 0 here, since both are identical but the latter
+                        ** is easier to optimize.
+                        */
                         g_ne (flags, val);
-                   } else if (val < 0xFFFFFFFF) {
-                       /* Use >= instead of > because the former gives better
-                        * code on the 6502 than the latter.
-                        */
-                       g_ge (flags, val+1);
+                    } else if (val < 0xFFFFFFFF) {
+                        /* Use >= instead of > because the former gives better
+                        ** code on the 6502 than the latter.
+                        */
+                        g_ge (flags, val+1);
                     } else {
                         /* Never true */
                         Warning ("Condition is never true");
                         AddCodeLine ("jsr return0");
-                   }
-                       } else {
+                    }
+                } else {
                     /* Signed compare */
                     if ((long) val < 0x7FFFFFFF) {
                         g_ge (flags, val+1);
@@ -3904,18 +3943,18 @@ void g_gt (unsigned flags, unsigned long val)
                         /* Never true */
                         Warning ("Condition is never true");
                         AddCodeLine ("jsr return0");
-                   }
+                    }
                 }
                 return;
 
             default:
-               typeerror (flags);
+                typeerror (flags);
         }
 
         /* If we go here, we didn't emit code. Push the lhs on stack and fall
-         * into the normal, non-optimized stuff. Note: The standard stuff will
-         * always work with ints.
-         */
+        ** into the normal, non-optimized stuff. Note: The standard stuff will
+        ** always work with ints.
+        */
         flags &= ~CF_FORCECHAR;
         g_push (flags & ~CF_CONST, 0);
     }
@@ -3929,23 +3968,22 @@ void g_gt (unsigned flags, unsigned long val)
 void g_ge (unsigned flags, unsigned long val)
 /* Test for greater than or equal to */
 {
-    static char* ops [12] = {
-        "tosge00",     "tosgea0",      "tosgeax",
-        "tosuge00",    "tosugea0",     "tosugeax",
-        0,             0,              "tosgeeax",
-        0,             0,              "tosugeeax",
+    static const char* const ops[4] = {
+        "tosgeax", "tosugeax", "tosgeeax", "tosugeeax"
     };
 
+    unsigned Label;
+
 
     /* If the right hand side is const, the lhs is not on stack but still
-     * in the primary register.
-     */
+    ** in the primary register.
+    */
     if (flags & CF_CONST) {
 
         /* Because the handling of the overflow flag is too complex for
-         * inlining, we can handle only unsigned compares, and signed
-         * compares against zero here.
-         */
+        ** inlining, we can handle only unsigned compares, and signed
+        ** compares against zero here.
+        */
         if (flags & CF_UNSIGNED) {
 
             /* Give a warning in some special cases */
@@ -3956,30 +3994,31 @@ void g_ge (unsigned flags, unsigned long val)
             }
 
             /* Look at the type */
-            switch (flags & CF_TYPE) {
+            switch (flags & CF_TYPEMASK) {
 
                 case CF_CHAR:
                     if (flags & CF_FORCECHAR) {
+                        /* Do a subtraction. Condition is true if carry set */
                         AddCodeLine ("cmp #$%02X", (unsigned char)val);
-                        AddCodeLine ("jsr booluge");
+                        AddCodeLine ("lda #$00");
+                        AddCodeLine ("ldx #$00");
+                        AddCodeLine ("rol a");
                         return;
                     }
                     /* FALLTHROUGH */
 
                 case CF_INT:
-                    /* If the low byte is zero, we must only test the high byte */
-                    AddCodeLine ("cpx #$%02X", (unsigned char)(val >> 8));
-                    if ((val & 0xFF) != 0) {
-                        unsigned L = GetLocalLabel();
-                        AddCodeLine ("bne %s", LocalLabelName (L));
-                        AddCodeLine ("cmp #$%02X", (unsigned char)val);
-                        g_defcodelabel (L);
-                    }
-                    AddCodeLine ("jsr booluge");
+                    /* Do a subtraction. Condition is true if carry set */
+                    AddCodeLine ("cmp #$%02X", (unsigned char)val);
+                    AddCodeLine ("txa");
+                    AddCodeLine ("sbc #$%02X", (unsigned char)(val >> 8));
+                    AddCodeLine ("lda #$00");
+                    AddCodeLine ("ldx #$00");
+                    AddCodeLine ("rol a");
                     return;
 
                 case CF_LONG:
-                    /* Do a subtraction */
+                    /* Do a subtraction. Condition is true if carry set */
                     AddCodeLine ("cmp #$%02X", (unsigned char)val);
                     AddCodeLine ("txa");
                     AddCodeLine ("sbc #$%02X", (unsigned char)(val >> 8));
@@ -3987,7 +4026,9 @@ void g_ge (unsigned flags, unsigned long val)
                     AddCodeLine ("sbc #$%02X", (unsigned char)(val >> 16));
                     AddCodeLine ("lda sreg+1");
                     AddCodeLine ("sbc #$%02X", (unsigned char)(val >> 24));
-                    AddCodeLine ("jsr booluge");
+                    AddCodeLine ("lda #$00");
+                    AddCodeLine ("ldx #$00");
+                    AddCodeLine ("rol a");
                     return;
 
                 default:
@@ -3996,8 +4037,8 @@ void g_ge (unsigned flags, unsigned long val)
 
         } else if (val == 0) {
 
-            /* Look at the type */
-            switch (flags & CF_TYPE) {
+            /* A signed compare against zero must only look at the sign bit */
+            switch (flags & CF_TYPEMASK) {
 
                 case CF_CHAR:
                     if (flags & CF_FORCECHAR) {
@@ -4022,13 +4063,56 @@ void g_ge (unsigned flags, unsigned long val)
                 default:
                     typeerror (flags);
             }
-        }
 
+        } else {
+
+            /* Signed compare against a constant != zero */
+            switch (flags & CF_TYPEMASK) {
+
+                case CF_CHAR:
+                    if (flags & CF_FORCECHAR) {
+                        Label = GetLocalLabel ();
+                        AddCodeLine ("sec");
+                        AddCodeLine ("sbc #$%02X", (unsigned char)val);
+                        AddCodeLine ("bvs %s", LocalLabelName (Label));
+                        AddCodeLine ("eor #$80");
+                        g_defcodelabel (Label);
+                        AddCodeLine ("asl a");          /* Bit 7 -> carry */
+                        AddCodeLine ("lda #$00");
+                        AddCodeLine ("ldx #$00");
+                        AddCodeLine ("rol a");
+                        return;
+                    }
+                    /* FALLTHROUGH */
+
+                case CF_INT:
+                    /* Do a subtraction */
+                    Label = GetLocalLabel ();
+                    AddCodeLine ("cmp #$%02X", (unsigned char)val);
+                    AddCodeLine ("txa");
+                    AddCodeLine ("sbc #$%02X", (unsigned char)(val >> 8));
+                    AddCodeLine ("bvs %s", LocalLabelName (Label));
+                    AddCodeLine ("eor #$80");
+                    g_defcodelabel (Label);
+                    AddCodeLine ("asl a");          /* Bit 7 -> carry */
+                    AddCodeLine ("lda #$00");
+                    AddCodeLine ("ldx #$00");
+                    AddCodeLine ("rol a");
+                    return;
+
+                case CF_LONG:
+                    /* This one is too costly */
+                    break;
+
+                default:
+                    typeerror (flags);
+            }
+        }
 
         /* If we go here, we didn't emit code. Push the lhs on stack and fall
-         * into the normal, non-optimized stuff. Note: The standard stuff will
-         * always work with ints.
-         */
+        ** into the normal, non-optimized stuff. Note: The standard stuff will
+        ** always work with ints.
+        */
         flags &= ~CF_FORCECHAR;
         g_push (flags & ~CF_CONST, 0);
     }
@@ -4040,7 +4124,7 @@ void g_ge (unsigned flags, unsigned long val)
 
 
 /*****************************************************************************/
-/*                         Allocating static storage                        */
+/*                         Allocating static storage                         */
 /*****************************************************************************/
 
 
@@ -4059,10 +4143,10 @@ void g_defdata (unsigned flags, unsigned long val, long offs)
     if (flags & CF_CONST) {
 
         /* Numeric constant */
-        switch (flags & CF_TYPE) {
+        switch (flags & CF_TYPEMASK) {
 
             case CF_CHAR:
-               AddDataLine ("\t.byte\t$%02lX", val & 0xFF);
+                AddDataLine ("\t.byte\t$%02lX", val & 0xFF);
                 break;
 
             case CF_INT:
@@ -4122,7 +4206,7 @@ void g_defbytes (const void* Bytes, unsigned Count)
         } while (Chunk);
 
         /* Output the line */
-        AddDataLine (Buf);
+        AddDataLine ("%s", Buf);
     }
 }
 
@@ -4143,7 +4227,7 @@ void g_initregister (unsigned Label, unsigned Reg, unsigned Size)
 {
     /* Register variables do always have less than 128 bytes */
     unsigned CodeLabel = GetLocalLabel ();
-    ldxconst (Size-1);
+    AddCodeLine ("ldx #$%02X", (unsigned char) (Size - 1));
     g_defcodelabel (CodeLabel);
     AddCodeLine ("lda %s,x", GetLabelName (CF_STATIC, Label, 0));
     AddCodeLine ("sta %s,x", GetLabelName (CF_REGVAR, Reg, 0));
@@ -4160,19 +4244,19 @@ void g_initauto (unsigned Label, unsigned Size)
 
     CheckLocalOffs (Size);
     if (Size <= 128) {
-        ldyconst (Size-1);
+        AddCodeLine ("ldy #$%02X", Size-1);
         g_defcodelabel (CodeLabel);
         AddCodeLine ("lda %s,y", GetLabelName (CF_STATIC, Label, 0));
         AddCodeLine ("sta (sp),y");
         AddCodeLine ("dey");
         AddCodeLine ("bpl %s", LocalLabelName (CodeLabel));
     } else if (Size <= 256) {
-        ldyconst (0);
+        AddCodeLine ("ldy #$00");
         g_defcodelabel (CodeLabel);
         AddCodeLine ("lda %s,y", GetLabelName (CF_STATIC, Label, 0));
         AddCodeLine ("sta (sp),y");
         AddCodeLine ("iny");
-        AddCodeLine ("cpy #$%02X", (unsigned char) Size);
+        AddCmpCodeIfSizeNot256 ("cpy #$%02X", Size);
         AddCodeLine ("bne %s", LocalLabelName (CodeLabel));
     }
 }
@@ -4184,7 +4268,7 @@ void g_initstatic (unsigned InitLabel, unsigned VarLabel, unsigned Size)
 {
     if (Size <= 128) {
         unsigned CodeLabel = GetLocalLabel ();
-        ldyconst (Size-1);
+        AddCodeLine ("ldy #$%02X", Size-1);
         g_defcodelabel (CodeLabel);
         AddCodeLine ("lda %s,y", GetLabelName (CF_STATIC, InitLabel, 0));
         AddCodeLine ("sta %s,y", GetLabelName (CF_STATIC, VarLabel, 0));
@@ -4192,21 +4276,21 @@ void g_initstatic (unsigned InitLabel, unsigned VarLabel, unsigned Size)
         AddCodeLine ("bpl %s", LocalLabelName (CodeLabel));
     } else if (Size <= 256) {
         unsigned CodeLabel = GetLocalLabel ();
-        ldyconst (0);
+        AddCodeLine ("ldy #$00");
         g_defcodelabel (CodeLabel);
         AddCodeLine ("lda %s,y", GetLabelName (CF_STATIC, InitLabel, 0));
         AddCodeLine ("sta %s,y", GetLabelName (CF_STATIC, VarLabel, 0));
         AddCodeLine ("iny");
-        AddCodeLine ("cpy #$%02X", (unsigned char) Size);
+        AddCmpCodeIfSizeNot256 ("cpy #$%02X", Size);
         AddCodeLine ("bne %s", LocalLabelName (CodeLabel));
     } else {
-        /* Use the easy way here: memcpy */
+        /* Use the easy way here: memcpy() */
         g_getimmed (CF_STATIC, VarLabel, 0);
         AddCodeLine ("jsr pushax");
         g_getimmed (CF_STATIC, InitLabel, 0);
         AddCodeLine ("jsr pushax");
         g_getimmed (CF_INT | CF_UNSIGNED | CF_CONST, Size, 0);
-        AddCodeLine ("jsr %s", GetLabelName (CF_EXTERNAL, (unsigned long) "memcpy", 0));
+        AddCodeLine ("jsr %s", GetLabelName (CF_EXTERNAL, (uintptr_t) "memcpy", 0));
     }
 }
 
@@ -4270,11 +4354,11 @@ void g_switch (Collection* Nodes, unsigned DefaultLabel, unsigned Depth)
 
             /* Determine the next label */
             if (I == CollCount (Nodes) - 1) {
-               /* Last node means not found */
+                /* Last node means not found */
                 g_truejump (0, DefaultLabel);
             } else {
-               /* Jump to the next check */
-               NextLabel = GetLocalLabel ();
+                /* Jump to the next check */
+                NextLabel = GetLocalLabel ();
                 g_truejump (0, NextLabel);
             }
 
@@ -4291,7 +4375,7 @@ void g_switch (Collection* Nodes, unsigned DefaultLabel, unsigned Depth)
 
 
 /*****************************************************************************/
-/*                       User supplied assembler code                       */
+/*                       User supplied assembler code                        */
 /*****************************************************************************/
 
 
@@ -4299,8 +4383,5 @@ void g_switch (Collection* Nodes, unsigned DefaultLabel, unsigned Depth)
 void g_asmcode (struct StrBuf* B)
 /* Output one line of assembler code. */
 {
-    AddCodeLine ("%.*s", SB_GetLen (B), SB_GetConstBuf (B));
+    AddCodeLine ("%.*s", (int) SB_GetLen (B), SB_GetConstBuf (B));
 }
-
-
-