2020-08-29 23:55:26 +02:00
; Prog8 definitions for floating point handling on the CommanderX16
2023-12-26 23:37:59 +01:00
%option enable_floats, no_symbol_prefixing, ignore_unused
2024-02-07 22:36:43 +01:00
%import shared_floats_functions
2020-08-29 23:55:26 +02:00
2020-09-20 23:49:36 +02:00
floats {
2024-04-15 00:52:09 +02:00
; ---- this block contains C-64 compatible floating point related functions ----
2020-08-29 23:55:26 +02:00
2022-03-27 19:34:49 +02:00
; ---- ROM float functions (same as on C128 except base page) ----
2020-08-29 23:55:26 +02:00
; note: the fac1 and fac2 are working registers and take 6 bytes each,
; floats in memory (and rom) are stored in 5-byte MFLPT packed format.
; note: fac1/2 might get clobbered even if not mentioned in the function's name.
; note: for subtraction and division, the left operand is in fac2, the right operand in fac1.
2024-04-15 00:52:09 +02:00
romsub $fe00 = AYINT() clobbers(A,X,Y) ; fac1-> signed word in 'facmo' and 'faclo', MSB FIRST. (might throw ILLEGAL QUANTITY) See "basic.sym" kernal symbol file for their memory locations.
2020-08-29 23:55:26 +02:00
; GIVAYF: signed word in Y/A (note different lsb/msb order) -> float in fac1
2020-09-20 23:49:36 +02:00
; there is also floats.GIVUAYFAY - unsigned word in A/Y (lo/hi) to fac1
2020-08-29 23:55:26 +02:00
; (tip: use GIVAYFAY to use A/Y input; lo/hi switched to normal order)
romsub $fe03 = GIVAYF(ubyte lo @ Y, ubyte hi @ A) clobbers(A,X,Y)
2022-03-27 19:34:49 +02:00
romsub $fe06 = FOUT() clobbers(X) -> uword @ AY ; fac1 -> string, address returned in AY
2024-01-06 00:04:15 +01:00
romsub $fe09 = VAL_1(uword string @XY, ubyte length @A) clobbers(A,X,Y) -> float @FAC1 ; convert ASCII string in XY and length in A, to floating point in FAC1. WARNING: only implemented in ROM 47+. Safer to use floats.parse() instead.
2022-03-27 19:34:49 +02:00
2022-12-03 19:56:54 +01:00
; GETADR: fac1 -> unsigned word in Y/A (might throw ILLEGAL QUANTITY) (result also in $14/15)
2020-08-29 23:55:26 +02:00
; (tip: use GETADRAY to get A/Y output; lo/hi switched to normal little endian order)
2022-03-27 19:34:49 +02:00
romsub $fe0c = GETADR() clobbers(X) -> ubyte @ Y, ubyte @ A
romsub $fe0f = FLOATC() clobbers(A,X,Y) ; convert address to floating point
romsub $fe12 = FSUB(uword mflpt @ AY) clobbers(A,X,Y) ; fac1 = mflpt from A/Y - fac1
2022-03-31 18:35:47 +02:00
romsub $fe15 = FSUBT() clobbers(A,X,Y) ; fac1 = fac2-fac1 mind the order of the operands
2022-03-27 19:34:49 +02:00
romsub $fe18 = FADD(uword mflpt @ AY) clobbers(A,X,Y) ; fac1 += mflpt value from A/Y
2022-03-31 18:35:47 +02:00
romsub $fe1b = FADDT() clobbers(A,X,Y) ; fac1 += fac2
2022-03-27 19:34:49 +02:00
romsub $fe1e = FMULT(uword mflpt @ AY) clobbers(A,X,Y) ; fac1 *= mflpt value from A/Y
2022-03-31 18:35:47 +02:00
romsub $fe21 = FMULTT() clobbers(A,X,Y) ; fac1 *= fac2
2023-03-29 23:46:44 +02:00
romsub $fe24 = FDIV(uword mflpt @ AY) clobbers(A,X,Y) ; fac1 = mflpt in A/Y / fac1
romsub $fe27 = FDIVT() clobbers(A,X,Y) ; fac1 = fac2/fac1 mind the order of the operands
2022-03-27 19:34:49 +02:00
romsub $fe2a = LOG() clobbers(A,X,Y) ; fac1 = LN(fac1) (natural log)
2024-04-15 00:52:09 +02:00
romsub $fe2d = INT() clobbers(A,X,Y) ; INT() truncates, use FADDH first to integer round instead of trunc
2022-03-27 19:34:49 +02:00
romsub $fe30 = SQR() clobbers(A,X,Y) ; fac1 = SQRT(fac1)
romsub $fe33 = NEGOP() clobbers(A) ; switch the sign of fac1 (fac1 = -fac1)
romsub $fe36 = FPWR(uword mflpt @ AY) clobbers(A,X,Y) ; fac1 = fac2 ** float in A/Y
2022-03-31 18:35:47 +02:00
romsub $fe39 = FPWRT() clobbers(A,X,Y) ; fac1 = fac2 ** fac1
2022-03-27 19:34:49 +02:00
romsub $fe3c = EXP() clobbers(A,X,Y) ; fac1 = EXP(fac1) (e ** fac1)
romsub $fe3f = COS() clobbers(A,X,Y) ; fac1 = COS(fac1)
romsub $fe42 = SIN() clobbers(A,X,Y) ; fac1 = SIN(fac1)
romsub $fe45 = TAN() clobbers(A,X,Y) ; fac1 = TAN(fac1)
romsub $fe48 = ATN() clobbers(A,X,Y) ; fac1 = ATN(fac1)
2024-04-15 00:52:09 +02:00
romsub $fe4b = ROUND() clobbers(A,X,Y) ; round least significant bit of fac1
2022-03-27 19:34:49 +02:00
romsub $fe4e = ABS() clobbers(A,X,Y) ; fac1 = ABS(fac1)
romsub $fe51 = SIGN() clobbers(X,Y) -> ubyte @ A ; SIGN(fac1) to A, $ff, $0, $1 for negative, zero, positive
romsub $fe54 = FCOMP(uword mflpt @ AY) clobbers(X,Y) -> ubyte @ A ; A = compare fac1 to mflpt in A/Y, 0=equal 1=fac1 is greater, 255=fac1 is less than
2022-10-22 13:33:35 +02:00
romsub $fe57 = RND_0() clobbers(A,X,Y) ; fac1 = RND(fac1) float random number generator NOTE: incompatible with C64's RND routine
2022-11-06 22:41:23 +01:00
romsub $fe57 = RND() clobbers(A,X,Y) ; alias for RND_0
2022-03-27 19:34:49 +02:00
romsub $fe5a = CONUPK(uword mflpt @ AY) clobbers(A,X,Y) ; load mflpt value from memory in A/Y into fac2
romsub $fe5d = ROMUPK(uword mflpt @ AY) clobbers(A,X,Y) ; load mflpt value from memory in current bank in A/Y into fac2
romsub $fe60 = MOVFRM(uword mflpt @ AY) clobbers(A,X,Y) ; load mflpt value from memory in A/Y into fac1 (use MOVFM instead)
romsub $fe63 = MOVFM(uword mflpt @ AY) clobbers(A,X,Y) ; load mflpt value from memory in A/Y into fac1
romsub $fe66 = MOVMF(uword mflpt @ XY) clobbers(A,X,Y) ; store fac1 to memory X/Y as 5-byte mflpt
romsub $fe69 = MOVFA() clobbers(A,X) ; copy fac2 to fac1
2024-04-15 00:52:09 +02:00
romsub $fe6c = MOVAF() clobbers(A,X) ; copy fac1 to fac2 (rounded the least significant bit)
2022-03-27 19:34:49 +02:00
; X16 additions
2024-04-15 00:52:09 +02:00
romsub $fe6f = FADDH() clobbers(A,X,Y) ; fac1 += 0.5, for integer rounding- call this before INT
2022-12-03 19:56:54 +01:00
romsub $fe72 = ZEROFC() clobbers(A,X,Y) ; fac1 = 0
2023-12-05 22:54:35 +01:00
romsub $fe75 = NORMAL() clobbers(A,X,Y) ; normalize fac1
romsub $fe78 = NEGFAC() clobbers(A) ; switch the sign of fac1 (fac1 = -fac1) (doesn't work, juse use NEGOP() instead!)
2022-12-03 19:56:54 +01:00
romsub $fe7b = MUL10() clobbers(A,X,Y) ; fac1 *= 10
2023-12-05 22:54:35 +01:00
romsub $fe7e = DIV10() clobbers(A,X,Y) ; fac1 /= 10 , CAUTION: result is always positive! Have to restore sign manually!
2022-12-03 19:56:54 +01:00
romsub $fe81 = MOVEF() clobbers(A,X) ; copy fac1 to fac2
romsub $fe84 = SGN() clobbers(A,X,Y) ; fac1 = SGN(fac1), result of SIGN (-1, 0 or 1)
2023-03-20 23:25:28 +01:00
romsub $fe87 = FLOAT() clobbers(A,X,Y) ; FAC = (s8).A
2022-12-03 19:56:54 +01:00
romsub $fe8a = FLOATS() clobbers(A,X,Y) ; FAC = (s16)facho+1:facho
romsub $fe8d = QINT() clobbers(A,X,Y) ; facho:facho+1:facho+2:facho+3 = u32(FAC)
romsub $fe90 = FINLOG(byte value @A) clobbers (A, X, Y) ; fac1 += signed byte in A
2022-03-31 18:35:47 +02:00
2022-03-27 19:34:49 +02:00
asmsub FREADSA (byte value @A) clobbers(A,X,Y) {
; ---- 8 bit signed A -> float in fac1
%asm {{
tay
bpl +
lda #$ff
jmp GIVAYF
+ lda #0
jmp GIVAYF
}}
}
2020-08-29 23:55:26 +02:00
asmsub GIVUAYFAY (uword value @ AY) clobbers(A,X,Y) {
; ---- unsigned 16 bit word in A/Y (lo/hi) to fac1
%asm {{
2024-04-15 00:52:09 +02:00
sty $c4 ; facmo
sta $c5 ; facmo+1
2022-03-27 19:34:49 +02:00
ldx #$90
sec
2023-07-15 15:18:26 +02:00
jmp FLOATC
2020-08-29 23:55:26 +02:00
}}
}
asmsub GIVAYFAY (uword value @ AY) clobbers(A,X,Y) {
; ---- signed 16 bit word in A/Y (lo/hi) to float in fac1
%asm {{
2020-12-28 00:19:58 +01:00
sta P8ZP_SCRATCH_B1
2020-08-29 23:55:26 +02:00
tya
2020-12-28 00:19:58 +01:00
ldy P8ZP_SCRATCH_B1
2020-08-29 23:55:26 +02:00
jmp GIVAYF ; this uses the inverse order, Y/A
}}
}
asmsub GETADRAY () clobbers(X) -> uword @ AY {
; ---- fac1 to unsigned word in A/Y
%asm {{
jsr GETADR ; this uses the inverse order, Y/A
sta P8ZP_SCRATCH_B1
tya
ldy P8ZP_SCRATCH_B1
rts
}}
}
2021-03-08 23:21:52 +01:00
asmsub FREADUY (ubyte value @Y) {
; -- 8 bit unsigned Y -> float in fac1
2022-03-27 19:34:49 +02:00
%asm {{
2023-03-20 23:25:28 +01:00
lda #0
jmp GIVAYF
2022-03-27 19:34:49 +02:00
}}
}
2023-12-29 03:12:44 +01:00
asmsub parse(str value @AY) -> float @FAC1 {
2024-03-30 22:29:13 +01:00
; -- Parse a string value of a number to float in FAC1.
; Requires kernal R47 or newer! (depends on val_1 working)
2023-08-16 14:47:20 +02:00
%asm {{
2023-11-30 23:07:25 +01:00
ldx VAL_1
cpx #$4c ; is there an implementation in VAL_1? (test for JMP)
2024-03-30 22:29:13 +01:00
bne _borked ; no, print error message
2023-11-30 23:07:25 +01:00
pha ; yes, count the length and call rom VAL_1.
phy
jsr prog8_lib.strlen
tya
ply
plx
jmp VAL_1
2024-03-30 22:29:13 +01:00
_borked
2023-11-27 23:58:28 +01:00
ldy #0
- lda _msg,y
beq +
jsr cbm.CHROUT
iny
bne -
+ jmp sys.exit
2024-03-30 22:29:13 +01:00
_msg .text 13,"?rom 47+ required for val1",13,0
2023-08-16 14:47:20 +02:00
}}
}
2022-10-22 13:33:35 +02:00
2024-04-15 00:52:09 +02:00
&uword AYINT_facmo = $c6 ; $c6/$c7 contain result of AYINT (See "basic.sym" kernal symbol file)
2022-10-22 13:33:35 +02:00
2023-12-29 03:12:44 +01:00
sub rnd() -> float {
2022-10-22 13:33:35 +02:00
%asm {{
lda #1
2023-07-15 15:18:26 +02:00
jmp RND_0
2022-10-22 13:33:35 +02:00
}}
}
2023-12-05 22:54:35 +01:00
asmsub normalize(float value @FAC1) -> float @ FAC1 {
%asm {{
jmp floats.NORMAL
}}
}
2021-05-16 00:14:57 +02:00
%asminclude "library:c64/floats.asm"
%asminclude "library:c64/floats_funcs.asm"
2020-08-29 23:55:26 +02:00
2022-11-06 22:41:23 +01:00
2020-08-29 23:55:26 +02:00
}