e53c34770a
- Use cc65 C library _filetype for writing instead of fixed 'p'. - Support for fake O_TRUNC through SCRATCH.
439 lines
8.6 KiB
ArmAsm
439 lines
8.6 KiB
ArmAsm
;
|
|
; Copyright (c) 2010, Kajtar Zsolt <soci@c64.rulez.org>
|
|
; All rights reserved.
|
|
;
|
|
; Redistribution and use in source and binary forms, with or without
|
|
; modification, are permitted provided that the following conditions
|
|
; are met:
|
|
; 1. Redistributions of source code must retain the above copyright
|
|
; notice, this list of conditions and the following disclaimer.
|
|
; 2. Redistributions in binary form must reproduce the above copyright
|
|
; notice, this list of conditions and the following disclaimer in the
|
|
; documentation and/or other materials provided with the distribution.
|
|
; 3. Neither the name of the Institute nor the names of its contributors
|
|
; may be used to endorse or promote products derived from this software
|
|
; without specific prior written permission.
|
|
;
|
|
; THIS SOFTWARE IS PROVIDED BY THE INSTITUTE AND CONTRIBUTORS ``AS IS'' AND
|
|
; ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
|
; IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
|
|
; ARE DISCLAIMED. IN NO EVENT SHALL THE INSTITUTE OR CONTRIBUTORS BE LIABLE
|
|
; FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
|
|
; DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
|
|
; OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
|
|
; HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
|
|
; LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
|
|
; OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
|
|
; SUCH DAMAGE.
|
|
;
|
|
; This file is part of the Contiki operating system.
|
|
;
|
|
; Author: Kajtar Zsolt <soci@c64.rulez.org>
|
|
;
|
|
;---------------------------------------------------------------------
|
|
.define F_IDE64 1 ; support IDE64, 100 byte only
|
|
|
|
.constructor init_pfs
|
|
.destructor done_pfs
|
|
.importzp ptr1, ptr2, ptr3, sp
|
|
.import __curunit, __filetype, popax, addysp, subysp
|
|
.export pfs_rwcommon, pfs_rwsetflags, pfs_rwcommonend
|
|
.if F_IDE64
|
|
.export ide64_rwprepare, ide64_rwfinish
|
|
.endif
|
|
.export _pfs_open, _pfs_read, _pfs_close
|
|
;---------------------------------------------------------------------
|
|
F_EOF = $80
|
|
F_NBLK = $40
|
|
F_OPEN = $20
|
|
F_MAXLEN = 80 ;max filename length
|
|
ST = $90 ;status
|
|
FN = $BB ;filename
|
|
FNL = $B7 ;filenamelength
|
|
LF = $B8 ;logical file number
|
|
SA = $B9 ;secondary address
|
|
OPEN = $FFC0
|
|
CLOSE = $FFC3
|
|
CHKIN = $FFC6
|
|
CHKOUT = $FFC9
|
|
CLRCHN = $FFCC
|
|
CHRIN = $FFCF
|
|
CHROUT = $FFD2
|
|
SETLFS = $FFBA
|
|
SETNAM = $FFBD
|
|
CLALL = $FFE7
|
|
WRITE = $DEF1
|
|
READ = $DEF4
|
|
;---------------------------------------------------------------------
|
|
.bss
|
|
|
|
cmdc: .res 1
|
|
flags: .res 10
|
|
;---------------------------------------------------------------------
|
|
.data
|
|
|
|
illchr: .byte $3A, $2A, $3F, $3D ;illegal chars
|
|
pw: .byte $2C
|
|
filet: .byte $50, $2C, $57 ;,p,w
|
|
;---------------------------------------------------------------------
|
|
.segment "INIT"
|
|
|
|
init_pfs:
|
|
ldy #F_MAXLEN+8
|
|
jsr subysp ;allocate
|
|
lda #0
|
|
sta FNL ;no name
|
|
ldy #15-1
|
|
jsr open2 ;open command channel
|
|
sta cmdc
|
|
rts
|
|
;---------------------------------------------------------------------
|
|
.code
|
|
|
|
error3: jmp error
|
|
|
|
_pfs_open:
|
|
sta ptr2
|
|
; Pop and store name
|
|
jsr popax
|
|
sta FN
|
|
stx FN+1 ;filename (kernal)
|
|
ldy #F_MAXLEN+8
|
|
jsr subysp ;allocate name
|
|
ldy #255
|
|
sty ptr3
|
|
sty ptr1
|
|
@L10: iny
|
|
cpy #F_MAXLEN
|
|
bcs error3 ;too long...
|
|
ldx #4 ;4+1 (comma)
|
|
@L12: cmp illchr,x
|
|
beq error3 ;illegal char?
|
|
dex
|
|
bpl @L12
|
|
cmp #$2F
|
|
bne @L11
|
|
sty ptr1 ;last slash
|
|
@L11: lda (FN),y
|
|
bne @L10
|
|
sty FNL
|
|
|
|
tay
|
|
tax
|
|
lda #$30 ;this partition
|
|
sta (sp),y
|
|
iny
|
|
inc ptr1
|
|
beq nopath
|
|
lda #$2F
|
|
@L13: sta (sp),y
|
|
iny
|
|
lda (FN,x)
|
|
inc FN
|
|
bne @L14
|
|
inc FN+1
|
|
@L14: cpy ptr1
|
|
bcc @L13
|
|
lda #$2F
|
|
sta (sp),y
|
|
iny
|
|
nopath: lda #$3A
|
|
@L16: sta (sp),y
|
|
iny
|
|
lda (FN,x)
|
|
inc FN
|
|
bne @L15
|
|
inc FN+1
|
|
@L15: ora #0
|
|
bne @L16
|
|
lsr ptr2
|
|
bcs ro ;read only
|
|
lda __filetype
|
|
sta filet ;set filetype
|
|
ldx #252
|
|
@L20: lda pw-252,x
|
|
sta (sp),y ;write
|
|
iny
|
|
inx
|
|
bne @L20
|
|
ro: tya ;name length (kernal)
|
|
ldx sp
|
|
ldy sp+1
|
|
jsr SETNAM
|
|
|
|
lda #0 ;file number
|
|
tay ;secondary address
|
|
open2: sta ptr2
|
|
sty ptr2+1
|
|
|
|
next: inc ptr2 ;next file number
|
|
ldx ptr2 ;file number
|
|
cpx #11
|
|
bcs error ;no more files
|
|
lda flags-1,x
|
|
bne next ;already used
|
|
lda ptr2+1
|
|
bne nextsa
|
|
inx
|
|
stx ptr2+1
|
|
nextsa: inc ptr2+1 ;next channel
|
|
retr: lda ptr2 ;file number
|
|
ldx __curunit
|
|
ldy ptr2+1 ;secondary address
|
|
jsr SETLFS
|
|
jsr OPEN ;open
|
|
bcs oerr
|
|
ldx cmdc
|
|
beq opok ;error channel open
|
|
jsr CHKIN
|
|
bcs error
|
|
jsr CHRIN
|
|
pha
|
|
jsr CHRIN
|
|
sta ptr1
|
|
@L4: jsr CHRIN
|
|
lda ST
|
|
beq @L4
|
|
jsr CLRCHN
|
|
pla
|
|
tax
|
|
lsr
|
|
cmp #$18 ;no serious error
|
|
beq opok
|
|
txa
|
|
pha
|
|
lda ptr2
|
|
jsr CLOSE ;close
|
|
pla
|
|
ldx ptr1
|
|
cmp #$37 ;no channel?
|
|
bne nnoc
|
|
cpx #$30
|
|
bne error ;not no channel
|
|
lda ptr2+1
|
|
cmp #14
|
|
bcc nextsa ;try next channel
|
|
bcs error ;give up
|
|
|
|
opok: ldx ptr2
|
|
lda #F_OPEN
|
|
sta flags-1,x
|
|
txa ;ok, return file number
|
|
ldx #0
|
|
ret: ldy #F_MAXLEN+8 ; free filename
|
|
jmp addysp
|
|
|
|
oerr: dec ptr2+1
|
|
cmp #2 ;already open,
|
|
beq next ;retry with next
|
|
|
|
error: lda #$FF
|
|
tax ;failed
|
|
bne ret
|
|
|
|
nnoc: inc ptr3
|
|
bne error ;no retry
|
|
cmp #$36
|
|
bne error ;no exists
|
|
cpx #$33
|
|
bne error
|
|
ldx cmdc
|
|
jsr CHKOUT
|
|
bcs error
|
|
lda FNL
|
|
sec
|
|
sbc #5
|
|
tax
|
|
lda #$53 ;scratch
|
|
jsr CHROUT
|
|
ldy #1
|
|
@L4: lda (FN),y
|
|
iny
|
|
jsr CHROUT
|
|
dex
|
|
bne @L4
|
|
lda #$3D
|
|
jsr CHROUT
|
|
iny
|
|
lda (FN),y
|
|
jsr CHROUT
|
|
lda #$0d
|
|
jsr CHROUT
|
|
jsr CLRCHN
|
|
jmp retr
|
|
|
|
.proc _pfs_read
|
|
jsr pfs_rwcommon ; pop params, check handle
|
|
beq error2 ; not open
|
|
|
|
bmi eof
|
|
.if F_IDE64
|
|
asl
|
|
bmi nblk ; no block operation
|
|
|
|
jsr CHKIN
|
|
bcs error2
|
|
|
|
; check support
|
|
jsr ide64_rwprepare
|
|
bcs norm
|
|
|
|
; read
|
|
jsr READ
|
|
bcs nosup
|
|
jmp ide64_rwfinish
|
|
|
|
nosup: lda #F_NBLK
|
|
jsr pfs_rwsetflags
|
|
.endif
|
|
|
|
; Valid lfn. Make it the input file
|
|
nblk: jsr CHKIN
|
|
bcs error2
|
|
|
|
; Decrement the count
|
|
norm: ldy #0
|
|
@L3: inc ptr1
|
|
bne @L0
|
|
inc ptr1+1
|
|
beq done ; branch always
|
|
|
|
; Read the next byte
|
|
@L0: jsr CHRIN
|
|
tax ; save the input byte
|
|
|
|
lda ST ; read the IEEE status
|
|
cmp #1 ; save it
|
|
and #%10111111 ; check anything but the EOI bit
|
|
bne error5 ; assume device not present
|
|
|
|
; Store the byte just read
|
|
txa
|
|
sta (ptr2),y
|
|
inc ptr2
|
|
bne @L1
|
|
inc ptr2+1 ; *buf++ = A;
|
|
|
|
; Get the status again and check the EOI bit
|
|
@L1: bcc @L3 ; loop if no end of file
|
|
|
|
; Set the EOI flag and bail out
|
|
lda #F_EOF
|
|
jsr pfs_rwsetflags
|
|
|
|
; Read done, close the input channel
|
|
done: jsr CLRCHN ; clrchn
|
|
|
|
; Return the number of chars read
|
|
eof: jmp pfs_rwcommonend
|
|
.endproc
|
|
|
|
; Error entry, file is not open
|
|
done_pfs:
|
|
ldx #10
|
|
@L2: ldy flags-1,x ; file open?
|
|
beq @L1
|
|
txa
|
|
jsr _pfs_close
|
|
@L1: dex
|
|
bne @L2
|
|
rts
|
|
|
|
error5: jsr CLRCHN ; clrchn
|
|
|
|
error2: ldx #255
|
|
txa
|
|
rts
|
|
|
|
_pfs_close:
|
|
pha
|
|
jsr CLOSE ;close
|
|
pla
|
|
tax
|
|
lda #0
|
|
sta flags-1,x
|
|
rts
|
|
|
|
.proc pfs_rwcommon
|
|
eor #$FF
|
|
sta ptr1
|
|
txa
|
|
eor #$FF
|
|
sta ptr1+1 ; remember -count-1
|
|
|
|
jsr popax ; get buf
|
|
sta ptr2
|
|
stx ptr2+1
|
|
|
|
sta ptr3
|
|
stx ptr3+1 ; for length
|
|
|
|
jsr popax ; get the handle
|
|
sta LF
|
|
lda #0
|
|
beq pfs_rwsetflags
|
|
.endproc
|
|
|
|
.if F_IDE64
|
|
.proc ide64_rwprepare
|
|
sec
|
|
lda ptr1+1
|
|
eor #255
|
|
beq small ; too small, not worth it
|
|
tay
|
|
lda ptr1 ; setup registers
|
|
eor #255
|
|
tax
|
|
lda $031B
|
|
eor #$DE
|
|
bne noide ; open vector set?
|
|
lda $DE60
|
|
eor #$49
|
|
bne noide ; check identification
|
|
lda $DE61
|
|
eor #$44
|
|
bne noide
|
|
lda $DE62
|
|
eor #$45
|
|
bne noide
|
|
clc
|
|
lda #ptr2
|
|
small: rts
|
|
|
|
noide: lda #F_NBLK
|
|
bne pfs_rwsetflags
|
|
.endproc
|
|
.endif
|
|
|
|
.proc pfs_rwsetflags
|
|
ldx LF
|
|
ora flags-1,x
|
|
sta flags-1,x
|
|
rts
|
|
.endproc
|
|
|
|
.if F_IDE64
|
|
.proc ide64_rwfinish
|
|
txa
|
|
pha
|
|
tya
|
|
pha
|
|
jsr CLRCHN
|
|
pla
|
|
tax
|
|
pla
|
|
rts
|
|
.endproc
|
|
.endif
|
|
|
|
.proc pfs_rwcommonend
|
|
lda ptr2
|
|
sec
|
|
sbc ptr3
|
|
pha
|
|
lda ptr2+1
|
|
sbc ptr3+1
|
|
tax
|
|
pla
|
|
rts
|
|
.endproc
|