prog8/il65/emit/incrdecr.py

278 lines
12 KiB
Python
Raw Normal View History

2018-01-14 14:18:50 +00:00
"""
Programming Language for 6502/6510 microprocessors, codename 'Sick'
This is the code generator for the in-place incr and decr instructions.
2018-01-28 20:58:16 +00:00
Incrementing or decrementing variables by a small value 0..255 (for integers)
2018-01-14 23:20:36 +00:00
is quite frequent and this generates assembly code tweaked for this case.
2018-01-14 14:18:50 +00:00
Written by Irmen de Jong (irmen@razorvine.net) - license: GNU GPL 3.0
"""
from typing import Callable
2018-02-03 00:44:14 +00:00
from ..plyparse import Scope, VarType, VarDef, Register, IncrDecr, SymbolName, Dereference, LiteralValue, datatype_of
2018-01-14 14:18:50 +00:00
from ..datatypes import DataType, REGISTER_BYTES
2018-02-03 00:44:14 +00:00
from . import CodeError, preserving_registers, to_hex
2018-01-14 14:18:50 +00:00
2018-02-02 21:42:09 +00:00
def generate_incrdecr(out: Callable, stmt: IncrDecr, scope: Scope, floats_enabled: bool) -> None:
2018-01-14 14:18:50 +00:00
assert isinstance(stmt.howmuch, (int, float)) and stmt.howmuch >= 0
assert stmt.operator in ("++", "--")
2018-01-28 20:58:16 +00:00
if stmt.howmuch == 0:
return
2018-01-23 20:20:01 +00:00
target = stmt.target # one of Register/SymbolName/Dereference, or a VarDef
if isinstance(target, SymbolName):
2018-01-16 00:47:55 +00:00
symdef = scope.lookup(target.name)
if isinstance(symdef, VarDef):
2018-01-23 20:20:01 +00:00
target = symdef # type: ignore
else:
raise CodeError("cannot incr/decr this", symdef)
2018-02-03 00:44:14 +00:00
2018-01-14 14:18:50 +00:00
if stmt.howmuch > 255:
if datatype_of(target, scope) != DataType.FLOAT:
raise CodeError("only supports integer incr/decr by up to 255 for now")
howmuch_str = str(stmt.howmuch)
if isinstance(target, Register):
reg = target.name
2018-01-14 14:18:50 +00:00
# note: these operations below are all checked to be ok
if stmt.operator == "++":
if reg == 'A':
# a += 1..255
out("\vclc")
out("\vadc #" + howmuch_str)
2018-01-14 14:18:50 +00:00
elif reg in REGISTER_BYTES:
if stmt.howmuch == 1:
2018-01-14 14:18:50 +00:00
# x/y += 1
out("\vin{:s}".format(reg.lower()))
2018-01-14 14:18:50 +00:00
else:
# x/y += 2..255
with preserving_registers({'A'}, scope, out):
out("\vt{:s}a".format(reg.lower()))
out("\vclc")
out("\vadc #" + howmuch_str)
out("\vta{:s}".format(reg.lower()))
2018-01-14 14:18:50 +00:00
elif reg == "AX":
# AX += 1..255
out("\vclc")
out("\vadc #" + howmuch_str)
out("\vbcc +")
out("\vinx")
2018-01-14 14:18:50 +00:00
out("+")
elif reg == "AY":
# AY += 1..255
out("\vclc")
out("\vadc # " + howmuch_str)
out("\vbcc +")
out("\viny")
2018-01-14 14:18:50 +00:00
out("+")
elif reg == "XY":
if stmt.howmuch == 1:
2018-01-14 14:18:50 +00:00
# XY += 1
out("\vinx")
out("\vbne +")
out("\viny")
2018-01-14 14:18:50 +00:00
out("+")
else:
# XY += 2..255
with preserving_registers({'A'}, scope, out):
out("\vtxa")
out("\vclc")
out("\vadc #" + howmuch_str)
out("\vtax")
out("\vbcc +")
out("\viny")
2018-01-14 14:18:50 +00:00
out("+")
else:
raise CodeError("invalid incr register: " + reg)
else:
if reg == 'A':
# a -= 1..255
out("\vsec")
out("\vsbc #" + howmuch_str)
2018-01-14 14:18:50 +00:00
elif reg in REGISTER_BYTES:
if stmt.howmuch == 1:
2018-01-14 14:18:50 +00:00
# x/y -= 1
out("\vde{:s}".format(reg.lower()))
2018-01-14 14:18:50 +00:00
else:
# x/y -= 2..255
with preserving_registers({'A'}, scope, out):
out("\vt{:s}a".format(reg.lower()))
out("\vsec")
out("\vsbc #" + howmuch_str)
out("\vta{:s}".format(reg.lower()))
2018-01-14 14:18:50 +00:00
elif reg == "AX":
# AX -= 1..255
out("\vsec")
out("\vsbc #" + howmuch_str)
out("\vbcs +")
out("\vdex")
2018-01-14 14:18:50 +00:00
out("+")
elif reg == "AY":
# AY -= 1..255
out("\vsec")
out("\vsbc #" + howmuch_str)
out("\vbcs +")
out("\vdey")
2018-01-14 14:18:50 +00:00
out("+")
elif reg == "XY":
if stmt.howmuch == 1:
2018-01-14 14:18:50 +00:00
# XY -= 1
out("\vcpx #0")
out("\vbne +")
out("\vdey")
2018-01-14 14:18:50 +00:00
out("+\t\tdex")
else:
# XY -= 2..255
with preserving_registers({'A'}, scope, out):
out("\vtxa")
out("\vsec")
out("\vsbc #" + howmuch_str)
out("\vtax")
out("\vbcs +")
out("\vdey")
2018-01-14 14:18:50 +00:00
out("+")
else:
raise CodeError("invalid decr register: " + reg)
elif isinstance(target, VarDef):
if target.vartype == VarType.CONST:
raise CodeError("cannot modify a constant", target)
what_str = target.name
if target.datatype == DataType.BYTE:
if stmt.howmuch == 1:
out("\v{:s} {:s}".format("inc" if stmt.operator == "++" else "dec", what_str))
2018-01-14 14:18:50 +00:00
else:
with preserving_registers({'A'}, scope, out):
out("\vlda " + what_str)
2018-01-14 14:18:50 +00:00
if stmt.operator == "++":
out("\vclc")
out("\vadc #" + howmuch_str)
2018-01-14 14:18:50 +00:00
else:
out("\vsec")
out("\vsbc #" + howmuch_str)
out("\vsta " + what_str)
elif target.datatype == DataType.WORD:
if stmt.howmuch == 1:
2018-01-14 14:18:50 +00:00
# mem.word +=/-= 1
if stmt.operator == "++":
out("\vinc " + what_str)
out("\vbne +")
out("\vinc {:s}+1".format(what_str))
2018-01-14 14:18:50 +00:00
out("+")
else:
with preserving_registers({'A'}, scope, out):
out("\vlda " + what_str)
out("\vbne +")
out("\vdec {:s}+1".format(what_str))
2018-01-14 14:18:50 +00:00
out("+\t\tdec " + what_str)
else:
# mem.word +=/-= 2..255
if stmt.operator == "++":
with preserving_registers({'A'}, scope, out):
out("\vclc")
out("\vlda " + what_str)
out("\vadc #" + howmuch_str)
out("\vsta " + what_str)
out("\vbcc +")
out("\vinc {:s}+1".format(what_str))
2018-01-14 14:18:50 +00:00
out("+")
else:
with preserving_registers({'A'}, scope, out):
out("\vsec")
out("\vlda " + what_str)
out("\vsbc #" + howmuch_str)
out("\vsta " + what_str)
out("\vbcs +")
out("\vdec {:s}+1".format(what_str))
2018-01-14 14:18:50 +00:00
out("+")
elif target.datatype == DataType.FLOAT:
2018-02-02 21:42:09 +00:00
if not floats_enabled:
raise CodeError("floating point numbers not enabled via option")
if stmt.howmuch == 1.0:
2018-01-14 14:18:50 +00:00
# special case for +/-1
with preserving_registers({'A', 'X', 'Y'}, scope, out, loads_a_within=True):
out("\vldx #<" + what_str)
out("\vldy #>" + what_str)
2018-01-14 14:18:50 +00:00
if stmt.operator == "++":
out("\vjsr c64flt.float_add_one")
2018-01-14 14:18:50 +00:00
else:
out("\vjsr c64flt.float_sub_one")
2018-02-01 22:14:12 +00:00
elif stmt.howmuch != 0:
float_name = scope.define_float_constant(stmt.howmuch)
with preserving_registers({'A', 'X', 'Y'}, scope, out, loads_a_within=True):
2018-02-01 22:14:12 +00:00
out("\vlda #<" + float_name)
out("\vsta c64.SCRATCH_ZPWORD1")
2018-02-01 22:14:12 +00:00
out("\vlda #>" + float_name)
out("\vsta c64.SCRATCH_ZPWORD1+1")
out("\vldx #<" + what_str)
out("\vldy #>" + what_str)
2018-01-14 14:18:50 +00:00
if stmt.operator == "++":
out("\vjsr c64flt.float_add_SW1_to_XY")
2018-01-14 14:18:50 +00:00
else:
out("\vjsr c64flt.float_sub_SW1_from_XY")
2018-01-14 14:18:50 +00:00
else:
raise CodeError("cannot in/decrement memory of type " + str(target.datatype), stmt.howmuch)
2018-02-02 21:42:09 +00:00
elif isinstance(target, Dereference):
2018-02-03 00:44:14 +00:00
if isinstance(target.operand, (LiteralValue, SymbolName)):
if isinstance(target.operand, LiteralValue):
what = to_hex(target.operand.value)
else:
symdef = target.my_scope().lookup(target.operand.name)
if isinstance(symdef, VarDef) and symdef.vartype == VarType.MEMORY:
what = to_hex(symdef.value.value) # type: ignore
else:
what = target.operand.name
if stmt.howmuch == 1:
if target.datatype == DataType.FLOAT:
if not floats_enabled:
raise CodeError("floating point numbers not enabled via option")
with preserving_registers({'A', 'X', 'Y'}, scope, out, loads_a_within=True):
out("\vldx " + what)
out("\vldy {:s}+1".format(what))
if stmt.operator == "++":
out("\vjsr c64flt.float_add_one")
else:
out("\vjsr c64flt.float_sub_one")
else:
with preserving_registers({'A', 'Y'}, scope, out, loads_a_within=True):
out("\vlda " + what)
out("\vsta c64.SCRATCH_ZPWORD1")
out("\vlda {:s}+1".format(what))
out("\vsta c64.SCRATCH_ZPWORD1+1")
if target.datatype == DataType.BYTE:
if stmt.operator == "++":
out("\vjsr il65_lib.incr_deref_byte")
else:
out("\vjsr il65_lib.decr_deref_byte")
elif target.datatype == DataType.WORD:
if stmt.operator == "++":
out("\vjsr il65_lib.incr_deref_word")
else:
out("\vjsr il65_lib.decr_deref_word")
else:
raise CodeError("cannot inc/decrement dereferenced literal of type " + str(target.datatype), stmt)
else:
raise CodeError("can't inc/dec this by something else as 1 right now", stmt) # XXX
elif isinstance(target.operand, Register):
if target.operand.datatype == DataType.BYTE:
raise CodeError("can't dereference just a single register, need combined register", target)
reg = target.operand.name
if stmt.howmuch == 1:
if stmt.operator == "++":
out("\vclc")
else:
out("\vsec")
if target.datatype == DataType.BYTE:
with preserving_registers({'A', 'Y'}, scope, out, loads_a_within=True):
out("\vjsr il65_lib.incrdecr_deref_byte_reg_" + reg)
else:
with preserving_registers({'A', 'Y'}, scope, out, loads_a_within=True):
out("\vjsr il65_lib.incrdecr_deref_word_reg_" + reg)
else:
raise CodeError("can't inc/dec this by something else as 1 right now", stmt) # XXX
2018-02-02 21:42:09 +00:00
else:
2018-02-03 00:44:14 +00:00
raise TypeError("invalid dereference target type", target)
2018-01-14 14:18:50 +00:00
else:
2018-02-03 00:44:14 +00:00
raise CodeError("cannot inc/decrement", target) # @todo support more