mirror of
https://github.com/catseye/SixtyPical.git
synced 2024-11-25 07:32:16 +00:00
A 6502-oriented low-level programming language supporting advanced static analysis
6502-assemblyabstract-interpretationeffect-systemexperimental-languageflow-typinglow-level-programmingstatic-analysissymbolic-execution
bin | ||
doc | ||
eg | ||
src/sixtypical | ||
tests | ||
.gitignore | ||
.hgignore | ||
.hgtags | ||
HISTORY.md | ||
LICENSE | ||
loadngo.sh | ||
README.md | ||
test.sh |
SixtyPical
Version 0.13. Work-in-progress, everything is subject to change.
SixtyPical is a very low-level programming language, similar to 6502 assembly, with static analysis through abstract interpretation.
In practice, this means it catches things like
- you forgot to clear carry before adding something to the accumulator
- a subroutine that you call trashes a register you thought was preserved
- you tried to write the address of something that was not a routine, to a jump vector
and suchlike. It also provides some convenient operations and abstractions based on common machine-language programming idioms, such as
- copying values from one register to another (via a third register when there are no underlying instructions that directly support it)
- explicit tail calls
- indirect subroutine calls
The reference implementation can analyze and compile SixtyPical programs to 6502 machine code.
Documentation
- Design Goals
- SixtyPical specification
- SixtyPical revision history
- Literate test suite for SixtyPical syntax
- Literate test suite for SixtyPical execution
- Literate test suite for SixtyPical analysis
- Literate test suite for SixtyPical compilation
- 6502 Opcodes used/not used in SixtyPical
TODO
Save registers on stack
This preserves them, so that, semantically, they can be used later even though they are trashed inside the block.
Re-order routines and optimize tail-calls to fallthroughs
Not because it saves 3 bytes, but because it's a neat trick. Doing it optimally is probably NP-complete. But doing it adeuqately is probably not that hard.
And at some point...
- Confirm that
and
can be used to restrict the range of table reads/writes. low
andhigh
address operators - to turnword
type intobyte
.const
s that can be used in defining the size of tables, etc.- Tests, and implementation, ensuring a routine can be assigned to a vector of "wider" type
- Related: can we simply view a (small) part of a buffer as a byte table? If not, why not?
- Check that the buffer being read or written to through pointer, appears in approporiate inputs or outputs set. (Associate each pointer with the buffer it points into.)
static
pointers -- currently not possible because pointers must be zero-page, thus@
, thus uninitialized.- Question the value of the "consistent initialization" principle for
if
statement analysis. interrupt
routines -- to indicate that "the supervisor" has stored values on the stack, so we can trash them.- Error messages that include the line number of the source code.
- Add absolute addressing in shl/shr, absolute-indexed for add, sub, etc.
- Automatic tail-call optimization (could be tricky, w/constraints?)
- Possibly
ld x, [ptr] + y
, possiblyst x, [ptr] + y
. - Maybe even
copy [ptra] + y, [ptrb] + y
, which can be compiled to indirect LDA then indirect STA!