1
0
mirror of https://github.com/KarolS/millfork.git synced 2025-04-04 22:29:32 +00:00

Merge pull request #100 from retrac0/apple2

ProDOS library + some small Apple II terminal fixes
This commit is contained in:
Karol Stasiak 2021-02-18 01:33:47 +01:00 committed by GitHub
commit 3d57421959
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
5 changed files with 328 additions and 8 deletions

View File

@ -1,10 +1,14 @@
[< back to index](../doc_index.md)
### A note about Apple II
## Apple II
Apple II variants other than II+/IIe/Enhanced IIe are untested;
this includes the original II, IIc and IIc+, but also later compatible computers (Apple III and IIgs).
They may or may not work.
### Model support
The current platform configuration for the Apple II targets the original Apple II with an NMOS processor.
Simple programs have been tested on the Apple II, II+, IIe and enhanced IIe. The IIc, IIc+ are untested.
The IIgs may work in compatibility mode, but this is untested. The Apple III is untested.
### Running your program
The compiler output is a raw machine code file, which then has to be put on a disk.
You can do it using [CiderPress](http://a2ciderpress.com/),
@ -13,6 +17,12 @@ or some other tool.
The file has to be loaded from $0C00. An example how to put such a file onto a disk using AppleCommander:
java -jar AppleCommander-1.3.5.jar -p disk_image.dsk FILENAME B 0xc00 < compiler_output.a2
java -jar AppleCommander.jar -p disk_image.dsk FILENAME B 0xc00 < compiler_output.a2
When you have placed your file on disk, boot the disk and enter this at the BASIC prompt:
] BRUN FILENAME
This has been successfully tested under DOS 3.3 and [ProDOS 2.4](https://prodos8.com/), on an Apple II+ and Apple IIe.
Creating a bootable disk is beyond the scope of this document.

View File

@ -0,0 +1,37 @@
// simple demonstration of ProDOS library routines that will write
// the Disk II ROM (assuming it is in slot 6) to the file DISKII.ROM
import stdio
import apple2_prodos
void main() {
// ProDOS requires a 1K aligned page for an open file as an IO buffer
array iobuf [1024] align(256)
// you have to explicitly create a file if it doesn't exist yet
// this can fail (so you should check prodos_error
// 06 is general binary type
prodos_create("DISKII.ROM"p, $06)
if prodos_error != 0 {
// prodos error call will be returned in prodos_error so you know
// what went wrong. 0 = no error
putstrz("{n}couldn't create file"z)
panic()
}
// ProDOS file handle
byte fp
fp = prodos_open("DISKII.ROM"p, iobuf)
// should check here again for error, and after all calls
// write the disk controller ROM to the file
prodos_write(fp, $c600, 256)
// closing frees the handle and io buffer
prodos_close(fp)
putstrz("{n}DONE"z)
while true { }
}

View File

@ -11,7 +11,7 @@ asm void bell() @$FBE4 extern
asm void putchar(byte register(a) char) @$FDED extern
asm void new_line() @$FC62 extern
asm pointer readline() {
jsr $FD6A
jsr $FD6F
ldx #$ff
__readline_loop:
inx

273
include/apple2_prodos.mfk Normal file
View File

@ -0,0 +1,273 @@
const byte PRODOS_ALLOC_INTERRUPT = $40
const byte PRODOS_DEALLOC_INTERRUPT = $41
const byte PRODOS_READ_BLOCK = $80
const byte PRODOS_WRITE_BLOCK = $81
const byte PRODOS_GET_TIME = $82
const byte PRODOS_CREATE = $c0
const byte PRODOS_DESTROY = $c1
const byte PRODOS_RENAME = $c2
const byte PRODOS_SET_FILE_INFO = $c3
const byte PRODOS_GET_FILE_INFO = $c4
const byte PRODOS_ON_LINE = $c5
const byte PRODOS_SET_PREFIX = $c6
const byte PRODOS_GET_PREFIX = $c7
const byte PRODOS_OPEN = $c8
const byte PRODOS_NEWLINE = $c9
const byte PRODOS_READ = $ca
const byte PRODOS_WRITE = $cb
const byte PRODOS_CLOSE = $cc
const byte PRODOS_FLUSH = $cd
const byte PRODOS_SET_MARK = $ce
const byte PRODOS_GET_MARK = $cf
const byte PRODOS_SET_EOF = $d0
const byte PRODOS_GET_EOF = $d1
const byte PRODOS_SET_BUF = $d2
const byte PRODOS_GET_BUF = $d3
// ProDOS MLI parameter lists
struct read_block_plist {
byte param_count
byte unit_num
pointer data_buffer
word block_num
}
struct write_block_plist {
byte param_count
byte unit_num
pointer data_buffer
word block_num
}
struct close_plist {
byte param_count
byte ref_num
}
struct flush_plist {
byte param_count
byte ref_num
}
struct create_plist {
byte param_count
pointer pathname
byte access
byte file_type
word aux_type
byte storage_type
word create_time
word create_date
}
struct destroy_plist {
byte param_count
pointer pathname
}
struct open_plist {
byte param_count
pointer pathname
pointer io_buffer
byte ref_num
}
struct newline_plist {
byte param_count
byte ref_num
byte enable_mask
byte newline_char
}
struct read_plist {
byte param_count
byte ref_num
pointer data_buffer
word request_count
word trans_count
}
struct rename_plist {
byte param_count
pointer pathname
pointer new_pathname
}
struct write_plist {
byte param_count
byte ref_num
pointer data_buffer
word request_count
word trans_count
}
struct get_prefix_plist {
byte param_count
pointer data_buffer
}
struct set_prefix_plist {
byte param_count
pointer data_buffer
}
// we'll just reuse the same area for all plists
union prodos_plist {
read_block_plist read_block
write_block_plist write_block
create_plist create
destroy_plist destroy
rename_plist rename
open_plist open
newline_plist newline
read_plist read
write_plist write
close_plist close
flush_plist flush
get_prefix_plist get_prefix
set_prefix_plist set_prefix
}
prodos_plist plist
byte prodos_error
// Millfork doesn't support self-modification in the assembler yet, so this
// code is placed in the following array:
//
// jsr $bf00
// $00
// plist.addr
// rts
//
// We modify mli_trampoline[3] ($00 on the second line) to set the specific
// ProDOS call before we jsr to $bf00
//
// TODO: can we just jmp to bf00 and save the extra rts ?
array mli_trampoline = [ $20, 0, $bf, 0, plist.addr.lo, plist.addr.hi, $60 ]
asm void prodos_mli_call(byte register(a) pdcall) {
sta mli_trampoline+3
jsr mli_trampoline
sta prodos_error
rts
}
void prodos_read_block(byte unum, pointer dbuf, word bnum) {
plist.read_block.param_count = 3
plist.read_block.unit_num = unum
plist.read_block.data_buffer = dbuf
plist.read_block.block_num = bnum
prodos_mli_call(PRODOS_READ_BLOCK)
}
void prodos_write_block(byte unum, pointer dbuf, word bnum) {
plist.write_block.param_count = 3
plist.write_block.unit_num = unum
plist.write_block.data_buffer = dbuf
plist.write_block.block_num = bnum
prodos_mli_call(PRODOS_WRITE_BLOCK)
}
void prodos_close(byte rnum) {
plist.close.param_count = 1
plist.close.ref_num = rnum
prodos_mli_call(PRODOS_CLOSE)
}
void prodos_flush(byte fp) {
plist.flush.param_count = 1
plist.flush.ref_num = fp
prodos_mli_call(PRODOS_FLUSH)
}
void prodos_get_prefix(pointer fnbuf) {
plist.get_prefix.param_count = 1
plist.get_prefix.data_buffer = fnbuf
prodos_mli_call(PRODOS_GET_PREFIX)
}
void prodos_set_prefix(pointer fnbuf) {
plist.set_prefix.param_count = 1
plist.set_prefix.data_buffer = fnbuf
prodos_mli_call(PRODOS_SET_PREFIX)
}
void prodos_create(pointer fn, byte ftype) {
plist.create.param_count = 7
plist.create.pathname = fn
plist.create.access = $c3
plist.create.file_type = ftype
plist.create.aux_type = $0
plist.create.storage_type = 1
plist.create.create_time = 0
plist.create.create_date = 0
prodos_mli_call(PRODOS_CREATE)
}
void prodos_destroy (pointer fn) {
plist.destroy.param_count = 0
plist.destroy.pathname = fn
prodos_mli_call(PRODOS_DESTROY)
}
void prodos_rename(pointer fn, pointer newfn) {
plist.rename.param_count = 2
plist.rename.pathname = fn
plist.rename.new_pathname = newfn
prodos_mli_call(PRODOS_RENAME)
}
// returns file handle if no error
byte prodos_open (pointer fn, pointer b) {
plist.open.param_count = 3
plist.open.pathname = fn
plist.open.io_buffer = b
prodos_mli_call(PRODOS_OPEN)
return plist.open.ref_num
}
void prodos_newline(byte fp, byte mask, byte nlchar) {
plist.newline.param_count = 3
plist.newline.ref_num = fp
plist.newline.enable_mask = mask
plist.newline.newline_char = nlchar
prodos_mli_call(PRODOS_NEWLINE)
}
void prodos_read(byte rnum, pointer dbuf, word rcnt) {
plist.read.param_count = 4
plist.read.ref_num = rnum
plist.read.data_buffer = dbuf
plist.read.request_count = rcnt
prodos_mli_call(PRODOS_READ)
}
void prodos_write(byte rnum, pointer dbuf, word rcnt) {
plist.write.param_count = 4
plist.write.ref_num = rnum
plist.write.data_buffer = dbuf
plist.write.request_count = rcnt
prodos_mli_call(PRODOS_WRITE)
}

View File

@ -10,4 +10,4 @@ a-z=C1
{q}=02
{apos}=07
{nbsp}=40
{n}=8D