kickc/src/test/ref/sinus-basic.asm

318 lines
6.6 KiB
NASM

/// @file
/// A lightweight library for printing on the C64.
///
/// Printing with this library is done by calling print_ function for each element
// Commodore 64 PRG executable file
.file [name="sinus-basic.prg", type="prg", segments="Program"]
.segmentdef Program [segments="Basic, Code, Data"]
.segmentdef Basic [start=$0801]
.segmentdef Code [start=$80d]
.segmentdef Data [startAfter="Code"]
.segment Basic
:BasicUpstart(main)
// Zeropage addresses used to hold lo/hi-bytes of addresses of float numbers in MEM
.label memLo = $fe
.label memHi = $ff
.label print_screen = $400
.label print_line_cursor = 2
.label print_char_cursor = 4
.segment Code
main: {
.label f_2pi = $e2e5
.label i = 8
// setFAC(1275)
lda #<$4fb
sta.z setFAC.prepareMEM1_mem
lda #>$4fb
sta.z setFAC.prepareMEM1_mem+1
jsr setFAC
// divFACby10()
jsr divFACby10
// setMEMtoFAC(f_127)
lda #<f_127
sta.z setMEMtoFAC.mem
lda #>f_127
sta.z setMEMtoFAC.mem+1
jsr setMEMtoFAC
lda #<print_screen
sta.z print_line_cursor
lda #>print_screen
sta.z print_line_cursor+1
lda #<print_screen
sta.z print_char_cursor
lda #>print_screen
sta.z print_char_cursor+1
lda #1
sta.z i
__b1:
// setFAC((word)i)
lda.z i
sta.z setFAC.w
lda #0
sta.z setFAC.w+1
jsr setFAC
// mulFACbyMEM(f_2pi)
lda #<f_2pi
sta.z mulFACbyMEM.mem
lda #>f_2pi
sta.z mulFACbyMEM.mem+1
jsr mulFACbyMEM
// setMEMtoFAC(f_i)
lda #<f_i
sta.z setMEMtoFAC.mem
lda #>f_i
sta.z setMEMtoFAC.mem+1
jsr setMEMtoFAC
// setFAC(25)
lda #<$19
sta.z setFAC.prepareMEM1_mem
lda #>$19
sta.z setFAC.prepareMEM1_mem+1
jsr setFAC
// divMEMbyFAC(f_i)
jsr divMEMbyFAC
// sinFAC()
jsr sinFAC
// mulFACbyMEM(f_127)
lda #<f_127
sta.z mulFACbyMEM.mem
lda #>f_127
sta.z mulFACbyMEM.mem+1
jsr mulFACbyMEM
// addMEMtoFAC(f_127)
jsr addMEMtoFAC
// getFAC()
jsr getFAC
// print_uint(getFAC())
jsr print_uint
// print_ln()
jsr print_ln
// for(byte i : 1..25)
inc.z i
lda #$1a
cmp.z i
bne __b15
// }
rts
__b15:
lda.z print_line_cursor
sta.z print_char_cursor
lda.z print_line_cursor+1
sta.z print_char_cursor+1
jmp __b1
.segment Data
f_i: .byte 0, 0, 0, 0, 0
f_127: .byte 0, 0, 0, 0, 0
}
.segment Code
// FAC = unsigned int
// Set the FAC (floating point accumulator) to the integer value of a 16bit unsigned int
// void setFAC(__zp(6) unsigned int w)
setFAC: {
.label prepareMEM1_mem = 6
.label w = 6
// BYTE0(mem)
lda.z prepareMEM1_mem
// *memLo = BYTE0(mem)
sta.z memLo
// BYTE1(mem)
lda.z prepareMEM1_mem+1
// *memHi = BYTE1(mem)
sta.z memHi
// asm
// Load unsigned int register Y,A into FAC (floating point accumulator)
ldy memLo
jsr $b391
// }
rts
}
// FAC = FAC/10
// Set FAC to FAC divided by 10
divFACby10: {
// asm
jsr $bafe
// }
rts
}
// MEM = FAC
// Stores the value of the FAC to memory
// Stores 5 chars (means it is necessary to allocate 5 chars to avoid clobbering other data using eg. char[] mem = {0, 0, 0, 0, 0};)
// void setMEMtoFAC(__zp(6) char *mem)
setMEMtoFAC: {
.label mem = 6
// BYTE0(mem)
lda.z mem
// *memLo = BYTE0(mem)
sta.z memLo
// BYTE1(mem)
lda.z mem+1
// *memHi = BYTE1(mem)
sta.z memHi
// asm
ldx memLo
tay
jsr $bbd4
// }
rts
}
// FAC = MEM*FAC
// Set FAC to MEM (float saved in memory) multiplied by FAC (float accumulator)
// Reads 5 chars from memory
// void mulFACbyMEM(__zp(6) char *mem)
mulFACbyMEM: {
.label mem = 6
// BYTE0(mem)
lda.z mem
// *memLo = BYTE0(mem)
sta.z memLo
// BYTE1(mem)
lda.z mem+1
// *memHi = BYTE1(mem)
sta.z memHi
// asm
lda memLo
ldy memHi
jsr $ba28
// }
rts
}
// FAC = MEM/FAC
// Set FAC to MEM (float saved in memory) divided by FAC (float accumulator)
// Reads 5 chars from memory
// void divMEMbyFAC(char *mem)
divMEMbyFAC: {
.const prepareMEM1_mem = main.f_i
// *memLo = BYTE0(mem)
lda #<prepareMEM1_mem
sta.z memLo
// *memHi = BYTE1(mem)
lda #>prepareMEM1_mem
sta.z memHi
// asm
lda memLo
ldy memHi
jsr $bb0f
// }
rts
}
// FAC = sin(FAC)
// Set FAC to sine of the FAC - sin(FAC)
// Sine is calculated on radians (0-2*PI)
sinFAC: {
// asm
jsr $e26b
// }
rts
}
// FAC = MEM+FAC
// Set FAC to MEM (float saved in memory) plus FAC (float accumulator)
// Reads 5 chars from memory
// void addMEMtoFAC(char *mem)
addMEMtoFAC: {
.const prepareMEM1_mem = main.f_127
// *memLo = BYTE0(mem)
lda #<prepareMEM1_mem
sta.z memLo
// *memHi = BYTE1(mem)
lda #>prepareMEM1_mem
sta.z memHi
// asm
lda memLo
ldy memHi
jsr $b867
// }
rts
}
// unsigned int = FAC
// Get the value of the FAC (floating point accumulator) as an integer 16bit unsigned int
// Destroys the value in the FAC in the process
getFAC: {
.label return = 6
// asm
// Load FAC (floating point accumulator) integer part into unsigned int register Y,A
jsr $b1aa
sty memLo
sta memHi
// unsigned int w = MAKEWORD( *memHi, *memLo )
sta.z return+1
tya
sta.z return
// }
rts
}
// Print a unsigned int as HEX
// void print_uint(__zp(6) unsigned int w)
print_uint: {
.label w = 6
// print_uchar(BYTE1(w))
ldx.z w+1
jsr print_uchar
// print_uchar(BYTE0(w))
ldx.z w
jsr print_uchar
// }
rts
}
// Print a newline
print_ln: {
__b1:
// print_line_cursor + 0x28
lda #$28
clc
adc.z print_line_cursor
sta.z print_line_cursor
bcc !+
inc.z print_line_cursor+1
!:
// while (print_line_cursor<print_char_cursor)
lda.z print_line_cursor+1
cmp.z print_char_cursor+1
bcc __b1
bne !+
lda.z print_line_cursor
cmp.z print_char_cursor
bcc __b1
!:
// }
rts
}
// Print a char as HEX
// void print_uchar(__register(X) char b)
print_uchar: {
// b>>4
txa
lsr
lsr
lsr
lsr
// print_char(print_hextab[b>>4])
tay
lda print_hextab,y
// Table of hexadecimal digits
jsr print_char
// b&0xf
lda #$f
axs #0
// print_char(print_hextab[b&0xf])
lda print_hextab,x
jsr print_char
// }
rts
}
// Print a single char
// void print_char(__register(A) char ch)
print_char: {
// *(print_char_cursor++) = ch
ldy #0
sta (print_char_cursor),y
// *(print_char_cursor++) = ch;
inc.z print_char_cursor
bne !+
inc.z print_char_cursor+1
!:
// }
rts
}
.segment Data
print_hextab: .text "0123456789abcdef"