3020 ;jrf 32 ;jumps over the next 32 bytes
<32 bytes of masked sprite data>
1F2802 ;ldir keymask,2 ;copies two bytes to the RAM location at keyMask (these are ports keymask and sMask)
0303 ;.db 3,3
1F0008 ;ldir sLSB0,8 ;copies 8 bytes to the first two sets of sprite data ports
02000303 ;.dw 2 \ .db 3,3 ;location of the sprite, coordinates
12005535 ;.dw 18 \ .db 85,53 ;location of the second sprite, coordinates
3302 ;jrb $ ;jumps back two bytes to the start of this instruction. Causes an infinite loop.
This is a pretty neat idea. I've put some thought into doing something similar myself, but on a computer.I've started programming a virtual CPU, too!
What made you decide to leave out registers?the only reason to have registers in a processor is to make things faster, so you don't have to be accessing ram constantly.
#include "jade.inc"
.org $00
jrf(start)
sprite:
.db $FF
.db $FF
.db $FF
.db $FF
.db $FF
.db $FF
.db $FF
.db $FF
start:
ldirc(keyMask,3)
.db 3,3,1
ldirc(sprite0,5)
.dw $100+sprite \ .db 0,0,2
loop:
cpc(key1,kClear)
jrfz(quit)
jrb(loop)
quit:
ldc(status,$80)
;##_key 0_##
key0 = 29h
kDown = $FE
kLeft = $FD
kRight = $FB
kUp = $F7
The sprite doesn't really seem to move, however. Sometimes it will jump to the middle of the top row, but never moves up or down. I've also gotten a couple crashes so far, some from mistyping the hex ;) Can you think of an easier way to transfer programs apart from typing them by hand? I tried prefixing the quote with the " token and ending it with ->Str0, but when running the basic interpreter complains about some of the compiled tokens and it won't store the entire string into Str0. Another idea might be a little [On]+[Clear] interrupt to forceably quit a program.;saferam1 =86ECh
saferam1 =9872h
but then i double checked one of my own projects and saw this:;saferam1 =$86EC ;saveSScreen=768
;saferam2 =$8A3A ;statVars=531
;saferam3 =$8508 ;textShadow=128
;saferam4 =$8478 ;OPs=66
;saferam5 =$84D3 ;iMathPtrs=10
;saferamp =$9872 ;appBackUpScreen=768
;saferamp2 =$8251 ;bootTemp=52
;gbuf =$9340 ;768
;graphVariables =$8E67 ;1179 : 494+126(financeVars)+108(smallEditRAM)+157(plus de graphs)+294(tableStuff)
so... $86EC it is ;)
ldirc(keyMask,4)
.db 3,1,1,1
The last '1' says to update the next sprite at each instruction. Keep in mind that it cycles through each sprite one at a time, so Sprite0 gets updated every 8th instruction. With a delay of 256, there was time for 2048 instructions to execute and pressing down looped through 4 instructions in your code meaning Y could get incremented 512 times before updating the sprite."[JADETEST":Asm(prgmJADE
And that will execute the program named JADETEST as a JADE program. Use '5' instead of '[' to execute an appvar.Loop:
cpc(key0,kDown)
incz(sY0)
cpc(key0,kUp)
decz(sY0)
cpc(key0,kLeft)
incz(sX0)
cpc(key0,kRight)
decz(sX0)
cpc(key1,kClear)
ldcz(status,$80)
jrb(Loop)
And that would be the code to move the coordinates of the sprite, which is 20 bytes compared to 50 bytes. bits(const,addr) ;will perform bitwise AND logic, but doesn't change (addr), only flags. Useful to see if one of the bits is set.
inv(addr) ;invert the bits
rotl(addr) ;rotate the bits left
rotr(addr) ;rotate the bits right
shftl(addr) ;shift the bits left
shftr(addr) ;shift the bits right
Here is my proposed outline for a new instruction set: 0 1 2 3 4 5 6 7 8 9 A B C D E F
0 lda() adda() adca() suba() sbca() xora() ora() anda() cpa() inc() rotl() shftl() pusha() pop() inv() ldira()
1 ldc() addc() adcc() subc() sbcc() xorc() orc() andc() cpc() dec() rotr() shftr() pushc() ex() bits() ldirc()
2 ret() setz() setc() togz() togc() jp1() jp2() jrf() jrb() call1() call2() callf() callb()
3
;Duplicate for execution on the c flag condition
4x,5x,6x,7x
;Duplicate for execution on the z flag condition
8x,9x,Ax,Bx
;???
Cx,Dx,Ex,Fx
There will then be room for 19 more instructions and possibly 64 more depending on what is done for the last set. As a quick note,these instructions are not included:resz() resc()
These are equivalent to:togzz() togcc()
For example, togzz() will only toggle the z flag if the z flag is set,so it only resets the z flag.#include "jade.inc"
.org $00
jrf(start)
sprite:
.db %00111100
.db %01100110
.db %11000011
.db %11100111
.db $FF
.db $FF
.db $FF
.db $FF
start:
ldirc(sprite0,5)
.dw $100+sprite \ .db 0,0,2
ldirc(keyMask,2)
.db 3,1
Loop:
orc(status,1) ;set bit 0 of the status port
KeyLoop:
cpc(key1,kClear) ;Check if clear is pressed
ldcz(status,$80) ;Turn off Jade if the z flag is set by writing 80h to the status port
cpc(key0,-1) ;test if no keys are being pressed
jrbz(KeyLoop)
callf(SpriteWait) ;In case not enough time has passed for the sprite to be drawn
ldc(sUpdate,0) ;Acknowledge the sprite drawing, allowing it to redraw the sprite (XORing twice results in no change)
callf(SpriteWait) ;We need to wait until the sprite is drawn before updating coordinates
cpc(key0,kDown)
incz(sY0)
cpc(key0,kUp)
decz(sY0)
cpc(key0,kLeft)
incz(sX0)
cpc(key0,kRight)
decz(sX0)
ldc(sUpdate,0) ;Acknowledge any sprites drawn, allowing the updated coordinates to be displayed
jrb(Loop)
SpriteWait:
cpa(sUpdate,sMask) ;Check if all of the sprites are drawn
retz()
jrb(SpriteWait)
I really like the idea of the conditional codes. It's something i really like about ARM assembly, pretty much any instruction can take conditions. So you can do things like addc r1,r2 or even cmpz r1,r2. You can actually also add shifts, so you can do addc r1,r2,r3 LSL 4, which adds r2 + [r3 shifted to the left 4 bits] and stores the result in r1, but only if the carry flag is set (that's one instruction!) :PAwesome, that sounds so cool o.o
Right now, what does sMask do? I assumed that a sprite wouldn't be processed unless you turned on that sprite's corresponding sMask bit. If i set the sprite delay to 1 and only have one sprite set in the sMask, it'll still process the other 7 sprites before coming back to the one i've enabled?Yes, that is how sMask works. Processing is rather quick if the sprite is not to be drawn, though.
Also, a double-buffer might work really well here since with 8 sprites not a lot of the screen should be changing at once:Hehe, I have a double buffer routine ready, but in the current version I just use the horribly slow OS routine _GrBufCpy. The way I plan to implement it should update the screen slightly faster than the code linked to, only because I keep the habit of setting the X coordinate at the beginning of the program (the one that is 0 to 63). Since my routine returns with it being the same, there is no need to keep setting that coordinate.
http://wikiti.brandonw.net/index.php?title=Z80_Routines:Graphic:Fastcopy#Double-buffered_copy
I think your other ideas are also good, in particular updating the LCD after each sprite is drawn will probably cause quite a bit of a slowdown. Waiting until all sprites have been drawn before updating is probably a good idea. I was also wondering if bit-instructions were going to be added :D:) Currently, you can set bits using orc() and reset bits with andc() and toggle with xorc(), but testing bits is more complicated without modifying RAM, so that is why I wanted to add the bits() command.
#include 'Jade.inc'
.org $00
jrf(start)
sprite:
.db %00111100
.db %01100110
.db %11000011
.db %11100111
.db $FF
.db $FF
.db $FF
.db $FF
start:
ldirc(sLSB0,5)
.dw $100+sprite \ .db 0,0,2
ldirc(keyMask,2)
.db 3,1 ;
Loop:
orc(status,2) ;set bit 1 of the status port to draw the sprites
orc(status,3) ;set bit 0 and 1 of the status port to update the LCD, then redraw the sprites
KeyLoop:
cpc(key1,jkClear) ;Check if clear is pressed
ldcz(status,$80) ;Turn off Jade if the z flag is set by writing 80h to the status port
cpc(key0,-1) ;test if no keys are being pressed
jrbz(KeyLoop)
cpc(key0,jkDown)
incz(sY0)
cpc(key0,jkUp)
decz(sY0)
cpc(key0,jkLeft)
incz(sX0)
cpc(key0,jkRight)
decz(sX0)
jrb(Loop)
Now I need to try to document everything .__.#include "ti83plus.inc"
#include "Jade.inc"
.org 0
scratchmath equ 252
scratchmath1 equ 253
ballxinc equ 254
ballyinc equ 255
Start:
ldirc(sLSB0,25)
.dw $100+ball \ .db 3,3,2 ;load the sprite data for Sprite0. The 2 is for XOR logic.
.dw $100+paddle \ .db 0,24,2 ;load the sprite data for Sprite1. The 2 is for XOR logic.
.dw $100+paddle \ .db 0,32,2 ;load the sprite data for Sprite2. The 2 is for XOR logic.
.dw $100+paddle \ .db 94,24,2 ;load the sprite data for Sprite3. The 2 is for XOR logic.
.dw $100+paddle \ .db 94,32,2 ;load the sprite data for Sprite4. The 2 is for XOR logic.
ldirc(keyMask,2)
.db %00010110,31 ;the second byte is for sMask (the sprite mask)
ldc(ballxinc,1)
ldc(ballyinc,1)
Loop:
orc(status,2) ;tell the status port to draw the sprites
orc(status,3) ;tell the status port to update the LCD, then draw the sprites
KeyLoop:
bits(key1,40h) ;check for the Clear key being pressed
orcz(status,80h) ;sets bit 7 of the status port if the z flag is set. This turns off Jade.
bits(key2,2) ;check if player 2 down key is being pressed
togz()
jrfz(TestP2Up)
cpc(sY3,48)
jrfz(TestP2Up)
addc(sY3,2)
addc(sY4,2)
TestP2Up:
bits(key2,8) ;check if player 2 down key is being pressed
togz()
jrfz(TestP1Down)
cpc(sY3,0)
jrfz(TestP1Down)
subc(sY3,2)
subc(sY4,2)
TestP1Down:
bits(key4,2) ;check if player 2 down key is being pressed
togz()
jrfz(TestP1Up)
cpc(sY1,48)
jrfz(TestP1Up)
addc(sY1,2)
addc(sY2,2)
TestP1Up:
bits(key4,8) ;check if player 2 down key is being pressed
togz()
jrfz(MoveBall)
cpc(sY1,0)
jrfz(MoveBall)
subc(sY1,2)
subc(sY2,2)
MoveBall:
adda(sX0,ballxinc)
callfz(TestP1Collision)
cpc(sX0,89)
callfz(TestP2Collision)
adda(sY0,ballyinc)
jrfz(NegYinc)
cpc(sY0,60)
togz()
jrbz(Loop)
NegYinc:
inv(ballyinc)
inc(ballyinc)
jrb(loop)
TestP1Collision:
lda(scratchmath,sY1)
jrf(CheckBounds)
TestP2Collision:
lda(scratchmath,sY3)
CheckBounds:
lda(scratchmath1,scratchmath)
subc(scratchmath,3)
ldcc(scratchmath,0)
suba(scratchmath,sY0)
jrfc(CheckOtherBound)
LoseCode:
ldc(status,80h)
CheckOtherBound:
addc(scratchmath1,15)
suba(scratchmath1,sY0)
jrbc(LoseCode)
NegXinc:
inv(ballxinc)
inc(ballxinc)
ret()
ball:
.db 18h
.db 24h
.db 24h
.db 18h
.db 0,0,0,0
paddle:
.db $C0
.db $C0
.db $C0
.db $C0
.db $C0
.db $C0
.db $C0
.db $C0
Use [3] and [9] to move the paddle of player 2 and [1] and [7] for Player 1. [Clear] exits.1 byte for the file info and type
bit 0:1 for the type
bit 2 for the save state size (128 bytes or 256)
bit 3 set if there is a description
bit 4 set if there is an icon
1 bytes for the name length
n bytes for the name
2 bytes for the miscellaneous data size
n bytes for miscellaneous data such as description and icon
2 bytes for the ROM size
n bytes for ROM data
What do you mean draw a string? Just print text? I also tested out AsmDream but i just can't get used to its syntax, i think for now i'm going to stick with the computer ;)Yes, that is what I mean about the text. And yeah, the syntax does take some time to get used to XD. I wonder if there is a way to get Mimas to work nicely for it.
EDIT : Also, what instructions can currently be used with c/z flags? To make defining things easier, we could do something like:Every instruction should be able to be used with the c and z flag, I just didn't finish including all of them in the .inc file yet.
lda = 0
adda = 1
adca = 2
;... etc.
c = 64
z = 128
;...
#define lda(addr1 , addr2) .db lda,addr1,addr2
#define ldac(addr1 , addr2) .db lda+c,addr1,addr2
#define ldaz(addr1 , addr2) .db lda+z,addr1,addr2
#define adda(addr1 , addr2) .db adda,addr1,addr2
#define addac(addr1 , addr2) .db adda+c,addr1,addr2
#define addaz(addr1 , addr2) .db adda+z,addr1,addr2
#define adca(addr1 , addr2) .db adca,addr1,addr2
#define adcac(addr1 , addr2) .db adca+c,addr1,addr2
#define adcaz(addr1 , addr2) .db adca+z,addr1,addr2
;etc.
EDIT2 : So i've been trying to figure out how to get a simple tilemap working but i can't think of how to do it without registers :/ Here's what i've got so far, a fake tilemap setup:That looks really nice! What do you mean that you cannot get it to work without registers? Do you mean you cannot get it to work without using RAM? That is the only way that I can think of doing it. I plan to look at your code and try my own hand at it, too, because that does look nice.
(http://www.mirari.fr/LOcY)
spriteX = stackbase-1
spriteY = stackbase-2
tilenum = stackbase-3
.org 0
ldc(keyMask,255)
ldirc(160,96)
;tilemap data
;drawn by column
.db 1,1,1,1,1,1,1,1
.db 1,0,2,2,2,2,0,1
.db 1,0,0,2,2,0,0,1
.db 1,0,0,0,0,0,0,1
.db 1,0,0,0,0,0,0,1
.db 1,0,0,0,0,0,0,1
.db 1,0,0,0,0,0,0,1
.db 1,0,0,0,0,0,0,1
.db 1,0,0,0,0,0,0,1
.db 1,0,0,2,2,0,0,1
.db 1,0,2,2,2,2,0,1
.db 1,1,1,1,1,1,1,1
ldirc(sMSB0,4)
.db 1,96,0,2
ldc(sMask,1)
TileMap:
ldc(sY0,64)
subc(sX0,8)
jrfc(Start)
TileMapLoop:
subc(sY0,8)
jrbc(TileMap)
pop(tilenum)
rotl(tilenum)
rotl(tilenum)
rotl(tilenum)
addc(tilenum,tiles)
lda(sLSB0,tilenum)
orc(status,2)
jrb(TileMapLoop)
tiles:
.db 0,0,0,0,0,0,0,0
.db 3Ch,42h,81h,81h,81h,81h,42h,3Ch
.db 3Ch,7Eh,$FF,$FF,$FF,$FF,7Eh,3Ch
Start:
ldc(stackptr,0)
orc(status,1)
ldc(status,80h)
ld bc,0701h
ld a,$FE ;test the keys
ld hl,saveSScreen+key0
KeyUpdateLoop:
out (1),a
rlca
nop
ini
jr nz,KeyUpdateLoop
I replaced the byte with something else, explained later. For now, the next update is indirection: ind2()
lda(addr1,indirect)
ind2() says that the second argument passed to the processor will be read as indirection. How that may get complicated is with something like this: ind2()
inc(addr1)
inc(indirect)
The first increment works as normal, but the second one has indirection. Internally, ind() instructions rotate a bit into an indirection counter (up to 8 bits). I did this on purpose, instead of simply setting bit 0 or bit 1 accordingly. It is just as easy to rotate the bits in, so why give the extreme coder the ability to use all 8 bits? With this in mind, you can make both arguments in an instruction read as indirection: ind1()
ind1()
lda(indirect,indirect)
The way that works, is first a bit is rotated in, then another bit is rotated in, leaving the lower 2 bits of the indirection byte set, so the next two addresses will be computed as indirection. Even better is that the indirection byte is what has replaced keyMask, so you can control indirection a little more and save bytes in some cases.Ok buddy, i'll check which data you need to pass and tell you that soon =]Okay, thanks!
What if you did ind2() \ ind1()? :P That's a cool idea, though, and definitely something that will make writing more complicated games easier :)Yes, that is valid, and it would cause the next argument to be indirection, then the one after that is normal, then the argument after that is indirection. The way this could be useful is when passing arguments to calls where you can force a call to use indirection. Though in that case, something like ind1() \ ind2() might be better.
jp Start ;go to the normal start of the app
jp Compile ;jump to the compiling code
If your compiling code is a subroutine that ends with an RET, then people can simply use 'call 4083h' to compile a program.