C syntax compiler optimized for the 6502 microprocessor
Go to file
Curtis F Kaylor f1b362e126 Added and debugged opcode address modes 2022-08-07 15:31:02 -04:00
apple1 Removed obsolete header file apple1/oldclude/apple1.h 2018-01-28 15:02:49 -05:00
apple2 Add/update Apple ][ test programs 2018-03-03 13:28:22 -05:00
apps Added and debugged opcode address modes 2022-08-07 15:31:02 -04:00
art Added artwork 2018-03-17 21:03:43 -04:00
bin Added customized py65 monitor 2018-03-03 21:50:18 -05:00
cc02src Added command line argument parsing to asm02.c02 2021-09-25 15:56:12 -04:00
doc Added isalud() to ctype module 2022-07-09 18:25:34 -04:00
include Added isalud() to ctype module 2022-07-09 18:25:34 -04:00
lib6502 Removed -a option from run6502 2021-09-25 14:50:03 -04:00
py65 Moved block test program from py65/ to test/ 2020-10-15 20:19:20 -04:00
src Added -I option to c02 compiler 2021-12-18 19:38:15 -05:00
test Added isalud() to ctype module 2022-07-09 18:25:34 -04:00
util Updated and tested programs in test/ directory 2018-08-19 18:24:02 -04:00
vcs Added Atari 2600 C02 test programs vcs/*.c02 2018-01-28 14:44:10 -05:00
vic20 Modified vic20/*.bat to use A02 assembler 2020-09-08 13:02:43 -04:00
work Removed work/notes.txt 2019-11-20 01:19:38 -05:00
x16 Completed, tested, debugged, and documented module bitlib 2020-10-04 22:17:09 -04:00
.gitignore Added .gitignore to repository 2019-11-20 01:22:54 -05:00
LICENSE.md Added Mozilla License 3.0 license file 2018-08-05 17:17:40 -04:00
Makefile Added A02 Assembler to Makefile 2019-11-15 14:30:43 -05:00
README.md Added integers and triple return values to readme.md 2019-11-15 13:47:33 -05:00
a02.c Modified intlib to use system vars intacc, intarg, and intovr 2022-04-04 19:47:16 -04:00
a02.h Modified intlib to use system vars intacc, intarg, and intovr 2022-04-04 19:47:16 -04:00
a02.sh Added Makefile and a02.sh shell script 2018-02-13 10:19:24 -05:00
c02.bat Added DOS batch, Bash script, and Notepad++ language files 2018-01-28 13:43:17 -05:00
c02.lang Added DOS batch, Bash script, and Notepad++ language files 2018-01-28 13:43:17 -05:00
c02.ppj Added A02 Assembler to Makefile 2019-11-15 14:30:43 -05:00
c02.sh Changed to c02.sh 2018-06-27 20:28:17 -04:00
ca02.sh Updated c64.a02, vic.a02, x16.a02 2019-11-20 01:36:43 -05:00
clean02.bat Removed *.sym from clean02.bat 2020-10-04 11:05:31 -04:00
clean02.sh Added cleanup scripts clean02.bat and clean02.sh 2018-01-28 14:57:18 -05:00
test.asm Initial commit of c02.c and test.asm 2017-03-28 00:09:18 -04:00

README.md

C02 6502 Compiler

C02 is a simple C-syntax language designed to generate highly optimized code for the 6502 microprocessor. The C02 specification is a highly specific subset of the C standard with some modifications and extensions

The compiler generates assembly language code, currently targeted to the DASM assembler.

See the top-level documentation for more information.

Building the Compiler

In Linux, use gcc and the Makefile.

In Windows use the Pelles C IDE and the file ./C02.ppj.

Compiling C02 programs

At a command line, cd into the appropriate directory, e.g. py65 or vic20.

Execute the c02 compiler from the parent directory using the c02 source file (without the .c02 extension) as an argument.

For example, to compile the program conds.c02, in the py65 directory, use the command

../c02 conds

in Linux, or

..\c02 conds

in Windows.

Some of the subdirectories contain a c02.bat file, which will compile the .c02 program, then run dasm to assemble the code. However, the path to dasm is hardcoded, so you will likely need to change it.

The file c02.sh provides the same functionality in Linux, but it may not be in a working state.

Syntax Examples

/* Directives */
#include <header.h02>   //Include header from standard library
#include "inc/hdr.h02"  //Include header from local directory
#pragma origin 8192     //Set start address of object code
#pragma zeropage $80    //Set start address of zero page variables 

/* Constants */
#define TRUE = $FF ; //Constants
#define FALSE = 0
enum {BLACK, WHITE, RED, CYAN, PURPLE, GREEN, BLUE, YELLOW}; 

/* Structures */
struct record {char name[8]; char index;}; //Struct Definition
struct record rec;                         //Struct Declaration

/* Variables and Array Declarations */
char b, c, d, e, f, g, h;    //8-bit Variables 
int i, j;                    //16-bit Variables
zeropage char p, q;          //8-bit Variables in Page 0
zeropage int u, v;           //16-bit Variables in Page 0
const char nine = 9;         //Const 8-bit variable set to decimal literal
const char maxsiz = $FF;     //Const 8-bit variable set to hexadecimal literal
const char flag = %01010101; //Const 8-bit variable set to binary literal
const char debug = #TRUE;    //Const 8-bit variable set to constant
const int k = $1234;         //Const 16-bit variable set to hexadecimal literal
char r[7];                   //8 byte Array (decimal dimension)
aligned char m[$FF];         //256 byte array aligned to page boundary
const char n = {1,2,3};      //Const array set to literal list
const char s = "string";     //Const array set to string literal
const char t = {"one", 1);   //Const array set to mixed list

/* Functions Declarations */
void myfunc(); //Forward declaration of function
char fnc(c) { /*function body */}     //One 8-bit Parameter
char fnd(c,d) { /*function body */}   //Two 8-bit Parameters
char fne(c,d,e) { /*function body */} //Three 8-bit Parameters
char fni(i) { /*function body */}     //One 16-bit Parameter
char fnj(c,i) { /*function body */}   //8-bit and 16-bit Parameters

/* Returning from a Function */
return c, d, e;    //Return up to three 8-bit values
return c,j;        //Return an 8-bit an 16-bit value
return i;          //Return a 16-bit value
return;            //No explicit return values

/* Assignments */
hmove; s80vid;               //Implicit Assignments
x = 0; y = a; a = 1;         //Register Assignments
b = c + d - e & f | g ^ h;   //Assignment and Expression
r[f] = m[g+1] & t[h-1];      //Array Indexing
r[j] = r[a] + s[x] + t[y];   //Arrays Indexed by Register
d = (e>f) ? d[e] : e[f];     //Shortcut If
b<< ;c[d]>>; x++; y--;       //Post-Operations

/* Function Calls */
b = abs(c); d = min(e,f), plot(b,c,d); //Up to Three Char Arguments
b = div(c+d,e)) - f; c = mult(d+e, f); //Expression in First Arg Only 
j = swap(i); j = ishift(b, i);         //Pass Int or Char with Int
puts("string"); fputs(f, &s);          //Passing Strings and Arrays
setdst(&r); b = strcpy(&s);            //Using Multiple String Arguments
proc(@record, &record);                //Pass Length and Address of Struct
c = getc(); setptr(?,g,h);             //No Args and Skipped Arguments
b,c = scnpos(); d,e,f = get3d();       //Plural Assignments
push b,c; mult(); pop p;               //Pass Arguments via Stack
iprint(); inline "Hello World";        //Pass Inline String Argument
irect(); inline 10,10,100,100;         //Pass Inline Char Arguments
icpstr(); inline &r, &s;               //Pass Inline Address Arguments

/* Control Structures */
if (c = 27) goto end;
if (b) e = div(f,g) else puts("Division by 0!");
if (b==0 || b>10 && b<20) fprint(n,"input %d in range"); 
c = 'A' ; while (c <= 'Z') { putc(c); c++; }
while() { c=rdkey; if (c=0) continue; putchr(c); if (c=13) break; }
do c = rdkey(); while (c=0);
do {c = getchr(); putchr(c);} while (c<>13)
for (c='A'; c<='Z'; c++) putc(c);
for (i=strlen(s)-1;i:+;i--) putc(s[i]);
for (i=0;c>0;i++) { c=getc(); s[i]=c }
select (getc()) {
  case $0D: putln("The Enter key");
  case #ESCKEY: if (#NOESC) break; goto escape; 
  case 'A','a': putln ("The letter A");
  default: putln("some other key");
}
end: //Label