6502 Macro Assembler in a single c++ file using the struse single file text parsing library. Supports most syntaxes. x65 was recently named Asm6502 but was renamed because Asm6502 is too generic, x65 has no particular meaning.
In order to minimize the documentation and make this page shorter I've moved the old documentation here.
The up to date documentation is here.
x65 can assemble 6502, 65C02 and 65816 source and build executables for c64, Apple II or just raw binary.
Noteworthy features:
- Code with sections, object files and linking or single file fixed address, or mix it up with fixed address sections in object files.
- Assembler listing with cycle counting for code review.
- Export multiple binaries with a single link operation.
- C style scoping within '{' and '}' with local and pool labels respecting scopes.
- Conditional assembly with if/ifdef/else etc.
- Recursible macro support
- Assembler directives representing a variety of features.
- Local labels can be defined in a number of ways, such as leading period (.label) or leading at-sign (@label) or terminating dollar sign (label$), and leading colon for merlin (:label).
- String Symbols system allows building user expressions and macros during assembly.
- Reassignment of symbols and labels by default.
- No indentation required for instructions, meaning that labels can't be mnemonics, macros or directives (merlin requires indentation).
- Supporting the syntax of other 6502 assemblers (Merlin syntax requires command line argument, -endm adds support for sources using macro/endmacro and repeat/endrepeat combos rather than scoeps).
- Apple II GS executable output (relocatable executable).
- Code
- Linking
- Comments
- Labels
- String Symbols
- Directives
- Macros
- Expressions
- List File with Cycle Count
x65.cpp requires struse.h which is a single file text parsing library that can be retrieved from https://github.com/Sakrac/struse.
- 6502 opcodes
- 6502 cheat sheet
- 6502 opcode grid
- Codebase64 CPU section
- 6502 illegal opcodes
- 65816 opcodes
Please note that releases have moved the Github releases
x65 is the assembler
x65macro.i is a 6502 include file that defines a number of standard macros that can assign values, move values, copy values and loop constructs, see x65.txt for details.
dump_x65 is a tool to inspect the contents of .x65 object files generated by x65 to track down linking issues
x65dsasm is a tool to disassemble assembled binary code for review, it will perform a basic analysis and assign labels where appropriate and treats unreferenced bytes as data rather than code. It can also export assemblable code from a binary.
This project would not be completed without the direct or indirect support of great people, some which I can currently remember:
- Marc dePeo, helping me uncover the strange and unique world of Merlin's assembler syntax (and working together with me on True Crime NY gameplay code and more)
- Che Lalic, explaining the murky bits of 65816 (and a Ninja on SNES NBA Hangtime and other projects)
- John Brooks, sharing the Rastan Apple II source code so I could test 65816 and figure out a number of issues with my initial linker, and encouraging the implementation of Apple II GS OS executable file format / OMF export (and helping out with Playstation All-Stars)
- Brutal Deluxe for releasing the excellent OMF Analyzer tool and the source, which was a significant help generating Apple II GS OS executables.
- The C64 demo scene for sharing a great deal of 6502 programming resources and overall inspiration.
- Jordan Mechner, sharing the Prince of Persia Apple II source code so I could test out a significant part of the assembler and the Merlin syntax mode
- Bill Budge, sharing the Pinball Construction Set Apple II source code, although at the point I tried it, all of it just assembled without any assembler issues at all.
Looking for help testing various features of the assembler, I have a large number of tests that pass without fail but there are so many ways for assemblers to break. Primarily tested with personal archive of sources written for Kick assmebler, DASM, TASM, XASM, etc. and passing most of Apple II Prince of Persia and Pinball Construction set.
TODO
- irp (indefinite repeat)
FIXED
- Source Debug output file including linkable object files, C64Debugger format
- Adding MERGE directive, Label Pools rewrite, TEXT data can be indexed from a string symbol
- Label Pools were destroyed after each scope so they did not work in include files which defeated their purpose. Label pools are now persistent through scopes.
- Labels reserved from label pools now distinguish between global and local. Use [.!@$] as a prefix to reserve a local label from a label pool (previously always local)
- Merlin macro parameters are not required on the MAC line, scope braces ('{', '}') can be used in the first column in Merlin.
- First line of a Merlin macro was sometimes ignored, two sequential subtractions were ignored in expressions.
- Pushing source contexts (macro, rept, include etc.) will always increment the scope depth.
- Fixed REPT / LUP to not destroy local symbols in the scope it was used in while also destroying local symbols within the repeating block correctly
- Switched over to inttypes.h from built-in types since the word unsigned was used a little too much in the code
- Removed the disassembler and put it into its own project x65dsasm
- LUP/REPT directives clean up local symbols each iteration to avoid crossing over an iteration with branches to local labels.
- Fixed Merlin MAC directive which is a little different from normal assembler macros
- Labels can start with numbers and values will only be interpreted as decimal numbers if terminated by a character that is not an alphabetic character or underscore
- INCSYM failed with local labels, this is now properly handled. (fixed again..)
- INCBIN and IMPORT BINARY always failed (force 0 bytes length)
- Using more than 16 bytes of Pool labels was flawed as byte 15 and 16 would be the same address.
- Fixed STRUCT directive (failed if contained line was empty).
- Adding x65macro.i
- Vice symbols will generate breakpoints whenever label 'debugbreak' is encountered
- Evaluating '==' was broken
- Macros can have dots in their names
- Handling double negative in expressions (--35 == 35)
- Macros works within conditionals (if/else/endif, etc)
- String symbols broke late evaluation resulting in garbage references, this has been fixed
- Added string symbols
- Resolved the DirectPage_Stack section vs. Zeropage section for Apple II GS/OS executables.
- OMF export for Apple II GS/OS executables
Revisions:
- 10 - String Symbols
- 9 - Apple II GS OS executable
- 8 - Fish food / Linking tested and passed with external project (Apple II gs Rastan)
- 7 - 65816 Support
- 6 - 65C02 support
- 5 - Merlin syntax
- 4 - Object files, relative sections and linking
- 3 - 6502 full support
- 2 - Moved file out of struse samples
- 1 - Built a sample of a simple assembler within struse