mirror of https://github.com/microsoft/MS-DOS.git
526 lines
18 KiB
NASM
526 lines
18 KiB
NASM
|
TITLE COPYRPOC ;Procedures called by COPY
|
|||
|
|
|||
|
INCLUDE COMSW.ASM
|
|||
|
|
|||
|
.xlist
|
|||
|
.xcref
|
|||
|
INCLUDE DOSSYM.ASM
|
|||
|
INCLUDE DEVSYM.ASM
|
|||
|
INCLUDE COMSEG.ASM
|
|||
|
.list
|
|||
|
.cref
|
|||
|
|
|||
|
INCLUDE COMEQU.ASM
|
|||
|
|
|||
|
DATARES SEGMENT PUBLIC
|
|||
|
DATARES ENDS
|
|||
|
|
|||
|
TRANDATA SEGMENT PUBLIC
|
|||
|
|
|||
|
EXTRN OVERWR:BYTE,FULDIR:BYTE,LOSTERR:BYTE
|
|||
|
EXTRN DEVWMES:BYTE,INBDEV:BYTE,NOSPACE:BYTE
|
|||
|
|
|||
|
TRANDATA ENDS
|
|||
|
|
|||
|
TRANSPACE SEGMENT PUBLIC
|
|||
|
|
|||
|
EXTRN CFLAG:BYTE,NXTADD:WORD,DESTCLOSED:BYTE
|
|||
|
EXTRN PLUS:BYTE,BINARY:BYTE,ASCII:BYTE,FILECNT:WORD
|
|||
|
EXTRN WRITTEN:BYTE,CONCAT:BYTE,DESTBUF:BYTE,SRCBUF:BYTE
|
|||
|
EXTRN SDIRBUF:BYTE,DIRBUF:BYTE,DESTFCB:BYTE,MELCOPY:BYTE
|
|||
|
EXTRN FIRSTDEST:BYTE,DESTISDIR:BYTE,DESTSWITCH:WORD
|
|||
|
EXTRN DESTTAIL:WORD,DESTINFO:BYTE,INEXACT:BYTE
|
|||
|
EXTRN DESTVARS:BYTE,SRCINFO:BYTE,RDEOF:BYTE
|
|||
|
EXTRN USERDIR1:BYTE,NOWRITE:BYTE
|
|||
|
EXTRN SRCHAND:WORD,CPDATE:WORD,CPTIME:WORD
|
|||
|
EXTRN SRCISDEV:BYTE,BYTCNT:WORD,TPA:WORD,TERMREAD:BYTE
|
|||
|
EXTRN DESTHAND:WORD,DESTISDEV:BYTE,DIRCHAR:BYTE
|
|||
|
|
|||
|
TRANSPACE ENDS
|
|||
|
|
|||
|
TRANCODE SEGMENT PUBLIC BYTE
|
|||
|
|
|||
|
PUBLIC SEARCH,SEARCHNEXT,DOCOPY,CLOSEDEST,FLSHFIL,SETASC
|
|||
|
PUBLIC BUILDNAME,COPERR
|
|||
|
|
|||
|
EXTRN PRINT:NEAR,BUILDPATH:NEAR,RESTUDIR1:NEAR
|
|||
|
EXTRN COMPNAME:NEAR,ENDCOPY:NEAR
|
|||
|
|
|||
|
ASSUME CS:TRANGROUP,DS:TRANGROUP,ES:TRANGROUP,SS:NOTHING
|
|||
|
|
|||
|
|
|||
|
SEARCHNEXT:
|
|||
|
MOV AH,DIR_SEARCH_NEXT
|
|||
|
TEST [SRCINFO],2
|
|||
|
JNZ SEARCH ; Do serach-next if ambig
|
|||
|
OR AH,AH ; Reset zero flag
|
|||
|
return
|
|||
|
SEARCH:
|
|||
|
PUSH AX
|
|||
|
MOV AH,SET_DMA
|
|||
|
MOV DX,OFFSET TRANGROUP:DIRBUF
|
|||
|
INT int_command ; Put result of search in DIRBUF
|
|||
|
POP AX ; Restore search first/next command
|
|||
|
MOV DX,FCB
|
|||
|
INT int_command ; Do the search
|
|||
|
OR AL,AL
|
|||
|
return
|
|||
|
|
|||
|
DOCOPY:
|
|||
|
mov [RDEOF],0 ; No EOF yet
|
|||
|
mov dx,offset trangroup:SRCBUF
|
|||
|
mov ax,OPEN SHL 8
|
|||
|
INT int_command
|
|||
|
retc ; If open fails, ignore
|
|||
|
mov bx,ax ; Save handle
|
|||
|
mov [SRCHAND],bx ; Save handle
|
|||
|
mov ax,(FILE_TIMES SHL 8)
|
|||
|
INT int_command
|
|||
|
mov [CPDATE],dx ; Save DATE
|
|||
|
mov [CPTIME],cx ; Save TIME
|
|||
|
mov ax,(IOCTL SHL 8)
|
|||
|
INT int_command ; Get device stuff
|
|||
|
and dl,devid_ISDEV
|
|||
|
mov [SRCISDEV],dl ; Set source info
|
|||
|
jz COPYLP ; Source not a device
|
|||
|
cmp [BINARY],0
|
|||
|
jz COPYLP ; ASCII device OK
|
|||
|
mov dx,offset trangroup:INBDEV ; Cannot do binary input
|
|||
|
jmp COPERR
|
|||
|
|
|||
|
COPYLP:
|
|||
|
mov bx,[SRCHAND]
|
|||
|
mov cx,[BYTCNT]
|
|||
|
mov dx,[NXTADD]
|
|||
|
sub cx,dx ; Compute available space
|
|||
|
jnz GOTROOM
|
|||
|
call FLSHFIL
|
|||
|
CMP [TERMREAD],0
|
|||
|
JNZ CLOSESRC ; Give up
|
|||
|
mov cx,[BYTCNT]
|
|||
|
GOTROOM:
|
|||
|
push ds
|
|||
|
mov ds,[TPA]
|
|||
|
ASSUME DS:NOTHING
|
|||
|
mov ah,READ
|
|||
|
INT int_command
|
|||
|
pop ds
|
|||
|
ASSUME DS:TRANGROUP
|
|||
|
jc CLOSESRC ; Give up if error
|
|||
|
mov cx,ax ; Get count
|
|||
|
jcxz CLOSESRC ; No more to read
|
|||
|
cmp [SRCISDEV],0
|
|||
|
jnz NOTESTA ; Is a device, ASCII mode
|
|||
|
cmp [ASCII],0
|
|||
|
jz BINREAD
|
|||
|
NOTESTA:
|
|||
|
MOV DX,CX
|
|||
|
MOV DI,[NXTADD]
|
|||
|
MOV AL,1AH
|
|||
|
PUSH ES
|
|||
|
MOV ES,[TPA]
|
|||
|
REPNE SCASB ; Scan for EOF
|
|||
|
POP ES
|
|||
|
JNZ USEALL
|
|||
|
INC [RDEOF]
|
|||
|
INC CX
|
|||
|
USEALL:
|
|||
|
SUB DX,CX
|
|||
|
MOV CX,DX
|
|||
|
BINREAD:
|
|||
|
ADD CX,[NXTADD]
|
|||
|
MOV [NXTADD],CX
|
|||
|
CMP CX,[BYTCNT] ; Is buffer full?
|
|||
|
JB TESTDEV ; If not, we may have found EOF
|
|||
|
CALL FLSHFIL
|
|||
|
CMP [TERMREAD],0
|
|||
|
JNZ CLOSESRC ; Give up
|
|||
|
JMP SHORT COPYLP
|
|||
|
|
|||
|
TESTDEV:
|
|||
|
cmp [SRCISDEV],0
|
|||
|
JZ CLOSESRC ; If file then EOF
|
|||
|
CMP [RDEOF],0
|
|||
|
JZ COPYLP ; On device, go till ^Z
|
|||
|
CLOSESRC:
|
|||
|
mov bx,[SRCHAND]
|
|||
|
mov ah,CLOSE
|
|||
|
INT int_command
|
|||
|
return
|
|||
|
|
|||
|
CLOSEDEST:
|
|||
|
cmp [DESTCLOSED],0
|
|||
|
retnz ; Don't double close
|
|||
|
MOV AL,BYTE PTR [DESTSWITCH]
|
|||
|
CALL SETASC ; Check for B or A switch on destination
|
|||
|
JZ BINCLOS
|
|||
|
MOV BX,[NXTADD]
|
|||
|
CMP BX,[BYTCNT] ; Is memory full?
|
|||
|
JNZ PUTZ
|
|||
|
call TRYFLUSH ; Make room for one lousy byte
|
|||
|
jz NOCONC
|
|||
|
CONCHNG: ; Concat flag changed on us
|
|||
|
stc
|
|||
|
return
|
|||
|
NOCONC:
|
|||
|
XOR BX,BX
|
|||
|
PUTZ:
|
|||
|
PUSH DS
|
|||
|
MOV DS,[TPA]
|
|||
|
MOV WORD PTR [BX],1AH ; Add End-of-file mark (Ctrl-Z)
|
|||
|
POP DS
|
|||
|
INC [NXTADD]
|
|||
|
MOV [NOWRITE],0 ; Make sure our ^Z gets written
|
|||
|
MOV AL,[WRITTEN]
|
|||
|
XOR AH,AH
|
|||
|
ADD AX,[NXTADD]
|
|||
|
JC BINCLOS ; > 1
|
|||
|
CMP AX,1
|
|||
|
JZ FORGETIT ; WRITTEN = 0 NXTADD = 1 (the ^Z)
|
|||
|
BINCLOS:
|
|||
|
call TRYFLUSH
|
|||
|
jnz CONCHNG
|
|||
|
cmp [WRITTEN],0
|
|||
|
jz FORGETIT ; Never wrote nothin
|
|||
|
MOV BX,[DESTHAND]
|
|||
|
MOV CX,[CPTIME]
|
|||
|
MOV DX,[CPDATE]
|
|||
|
CMP [INEXACT],0 ; Copy not exact?
|
|||
|
JZ DODCLOSE ; If no, copy date & time
|
|||
|
MOV AH,GET_TIME
|
|||
|
INT int_command
|
|||
|
SHL CL,1
|
|||
|
SHL CL,1 ; Left justify min in CL
|
|||
|
SHL CX,1
|
|||
|
SHL CX,1
|
|||
|
SHL CX,1 ; hours to high 5 bits, min to 5-10
|
|||
|
SHR DH,1 ; Divide seconds by 2 (now 5 bits)
|
|||
|
OR CL,DH ; And stick into low 5 bits of CX
|
|||
|
PUSH CX ; Save packed time
|
|||
|
MOV AH,GET_DATE
|
|||
|
INT int_command
|
|||
|
SUB CX,1980
|
|||
|
XCHG CH,CL
|
|||
|
SHL CX,1 ; Year to high 7 bits
|
|||
|
SHL DH,1 ; Month to high 3 bits
|
|||
|
SHL DH,1
|
|||
|
SHL DH,1
|
|||
|
SHL DH,1
|
|||
|
SHL DH,1 ; Most sig bit of month in carry
|
|||
|
ADC CH,0 ; Put that bit next to year
|
|||
|
OR DL,DH ; Or low three of month into day
|
|||
|
MOV DH,CH ; Get year and high bit of month
|
|||
|
POP CX ; Get time back
|
|||
|
DODCLOSE:
|
|||
|
MOV AX,(FILE_TIMES SHL 8) OR 1
|
|||
|
INT int_command ; Set date and time
|
|||
|
MOV AH,CLOSE
|
|||
|
INT int_command
|
|||
|
INC [FILECNT]
|
|||
|
INC [DESTCLOSED]
|
|||
|
RET50:
|
|||
|
CLC
|
|||
|
return
|
|||
|
|
|||
|
FORGETIT:
|
|||
|
MOV BX,[DESTHAND]
|
|||
|
CALL DODCLOSE ; Close the dest
|
|||
|
MOV DX,OFFSET TRANGROUP:DESTBUF
|
|||
|
MOV AH,UNLINK
|
|||
|
INT int_command ; And delete it
|
|||
|
MOV [FILECNT],0 ; No files transferred
|
|||
|
JMP RET50
|
|||
|
|
|||
|
TRYFLUSH:
|
|||
|
mov al,[CONCAT]
|
|||
|
push ax
|
|||
|
call FLSHFIL
|
|||
|
pop ax
|
|||
|
cmp al,[CONCAT]
|
|||
|
return
|
|||
|
|
|||
|
FLSHFIL:
|
|||
|
; Write out any data remaining in memory.
|
|||
|
; Inputs:
|
|||
|
; [NXTADD] = No. of bytes to write
|
|||
|
; [CFLAG] <>0 if file has been created
|
|||
|
; Outputs:
|
|||
|
; [NXTADD] = 0
|
|||
|
|
|||
|
MOV [TERMREAD],0
|
|||
|
cmp [CFLAG],0
|
|||
|
JZ NOTEXISTS
|
|||
|
JMP EXISTS
|
|||
|
NOTEXISTS:
|
|||
|
call BUILDDEST ; Find out all about the destination
|
|||
|
CALL COMPNAME ; Source and dest. the same?
|
|||
|
JNZ PROCDEST ; If not, go ahead
|
|||
|
CMP [SRCISDEV],0
|
|||
|
JNZ PROCDEST ; Same name on device OK
|
|||
|
CMP [CONCAT],0 ; Concatenation?
|
|||
|
MOV DX,OFFSET TRANGROUP:OVERWR
|
|||
|
JZ COPERRJ ; If not, overwrite error
|
|||
|
MOV [NOWRITE],1 ; Flag not writting (just seeking)
|
|||
|
PROCDEST:
|
|||
|
mov ax,(OPEN SHL 8) OR 1
|
|||
|
CMP [NOWRITE],0
|
|||
|
JNZ DODESTOPEN ; Don't actually create if NOWRITE set
|
|||
|
mov ah,CREAT
|
|||
|
xor cx,cx
|
|||
|
DODESTOPEN:
|
|||
|
mov dx,offset trangroup:DESTBUF
|
|||
|
INT int_command
|
|||
|
MOV DX,OFFSET TRANGROUP:FULDIR
|
|||
|
JC COPERRJ
|
|||
|
mov [DESTHAND],ax ; Save handle
|
|||
|
mov [CFLAG],1 ; Destination now exists
|
|||
|
mov bx,ax
|
|||
|
mov ax,(IOCTL SHL 8)
|
|||
|
INT int_command ; Get device stuff
|
|||
|
mov [DESTISDEV],dl ; Set dest info
|
|||
|
test dl,devid_ISDEV
|
|||
|
jz EXISTS ; Dest not a device
|
|||
|
mov al,BYTE PTR [DESTSWITCH]
|
|||
|
AND AL,ASWITCH+BSWITCH
|
|||
|
JNZ TESTBOTH
|
|||
|
MOV AL,[ASCII] ; Neither set, use current setting
|
|||
|
OR AL,[BINARY]
|
|||
|
JZ EXSETA ; Neither set, default to ASCII
|
|||
|
TESTBOTH:
|
|||
|
JPE EXISTS ; Both are set, ignore
|
|||
|
test AL,BSWITCH
|
|||
|
jz EXISTS ; Leave in cooked mode
|
|||
|
mov ax,(IOCTL SHL 8) OR 1
|
|||
|
xor dh,dh
|
|||
|
or dl,devid_RAW
|
|||
|
mov [DESTISDEV],dl ; New value
|
|||
|
INT int_command ; Set device to RAW mode
|
|||
|
jmp short EXISTS
|
|||
|
|
|||
|
COPERRJ:
|
|||
|
jmp SHORT COPERR
|
|||
|
|
|||
|
EXSETA:
|
|||
|
; What we read in may have been in binary mode, flag zapped write OK
|
|||
|
mov [ASCII],ASWITCH ; Set ASCII mode
|
|||
|
or [INEXACT],ASWITCH ; ASCII -> INEXACT
|
|||
|
EXISTS:
|
|||
|
cmp [NOWRITE],0
|
|||
|
jnz NOCHECKING ; If nowrite don't bother with name check
|
|||
|
CALL COMPNAME ; Source and dest. the same?
|
|||
|
JNZ NOCHECKING ; If not, go ahead
|
|||
|
CMP [SRCISDEV],0
|
|||
|
JNZ NOCHECKING ; Same name on device OK
|
|||
|
; At this point we know in append (would have gotten overwrite error on first
|
|||
|
; destination create otherwise), and user trying to specify destination which
|
|||
|
; has been scribbled already (if dest had been named first, NOWRITE would
|
|||
|
; be set).
|
|||
|
MOV DX,OFFSET TRANGROUP:LOSTERR ; Tell him he's not going to get it
|
|||
|
CALL PRINT
|
|||
|
MOV [NXTADD],0 ; Set return
|
|||
|
INC [TERMREAD] ; Tell Read to give up
|
|||
|
RET60:
|
|||
|
return
|
|||
|
|
|||
|
NOCHECKING:
|
|||
|
mov bx,[DESTHAND] ; Get handle
|
|||
|
XOR CX,CX
|
|||
|
XCHG CX,[NXTADD]
|
|||
|
JCXZ RET60 ; If Nothing to write, forget it
|
|||
|
INC [WRITTEN] ; Flag that we wrote something
|
|||
|
CMP [NOWRITE],0 ; If NOWRITE set, just seek CX bytes
|
|||
|
JNZ SEEKEND
|
|||
|
XOR DX,DX
|
|||
|
PUSH DS
|
|||
|
MOV DS,[TPA]
|
|||
|
ASSUME DS:NOTHING
|
|||
|
MOV AH,WRITE
|
|||
|
INT int_command
|
|||
|
POP DS
|
|||
|
ASSUME DS:TRANGROUP
|
|||
|
MOV DX,OFFSET TRANGROUP:NOSPACE
|
|||
|
JC COPERR ; Failure
|
|||
|
sub cx,ax
|
|||
|
retz ; Wrote all supposed to
|
|||
|
test [DESTISDEV],devid_ISDEV
|
|||
|
jz COPERR ; Is a file, error
|
|||
|
test [DESTISDEV],devid_RAW
|
|||
|
jnz DEVWRTERR ; Is a raw device, error
|
|||
|
cmp [INEXACT],0
|
|||
|
retnz ; INEXACT so OK
|
|||
|
dec cx
|
|||
|
retz ; Wrote one byte less (the ^Z)
|
|||
|
DEVWRTERR:
|
|||
|
MOV DX,OFFSET TRANGROUP:DEVWMES
|
|||
|
COPERR:
|
|||
|
CALL PRINT
|
|||
|
inc [DESTCLOSED]
|
|||
|
cmp [CFLAG],0
|
|||
|
jz ENDCOPYJ ; Never actually got it open
|
|||
|
MOV bx,[DESTHAND]
|
|||
|
MOV AH,CLOSE ; Close the file
|
|||
|
INT int_command
|
|||
|
MOV DX,OFFSET TRANGROUP:DESTBUF
|
|||
|
MOV AH,UNLINK
|
|||
|
INT int_command ; And delete it
|
|||
|
MOV [CFLAG],0
|
|||
|
ENDCOPYJ:
|
|||
|
JMP ENDCOPY
|
|||
|
|
|||
|
|
|||
|
SEEKEND:
|
|||
|
xor dx,dx ; Zero high half of offset
|
|||
|
xchg dx,cx ; cx:dx is seek location
|
|||
|
mov ax,(LSEEK SHL 8) OR 1
|
|||
|
INT int_command ; Seek ahead in the file
|
|||
|
cmp [RDEOF],0
|
|||
|
retz
|
|||
|
; If a ^Z has been read we must set the file size to the current
|
|||
|
; file pointer location
|
|||
|
MOV AH,WRITE
|
|||
|
INT int_command ; CX is zero, truncates file
|
|||
|
return
|
|||
|
|
|||
|
SETASC:
|
|||
|
; Given switch vector in AX,
|
|||
|
; Set ASCII switch if A is set
|
|||
|
; Clear ASCII switch if B is set
|
|||
|
; BINARY set if B specified
|
|||
|
; Leave ASCII unchanged if neither or both are set
|
|||
|
; Also sets INEXACT if ASCII is ever set. AL = ASCII on exit, flags set
|
|||
|
AND AL,ASWITCH+BSWITCH
|
|||
|
JPE LOADSW ; PE means both or neither are set
|
|||
|
PUSH AX
|
|||
|
AND AL,BSWITCH
|
|||
|
MOV [BINARY],AL
|
|||
|
POP AX
|
|||
|
AND AL,ASWITCH
|
|||
|
MOV [ASCII],AL
|
|||
|
OR [INEXACT],AL
|
|||
|
LOADSW:
|
|||
|
MOV AL,[ASCII]
|
|||
|
OR AL,AL
|
|||
|
return
|
|||
|
|
|||
|
BUILDDEST:
|
|||
|
cmp [DESTISDIR],-1
|
|||
|
jnz KNOWABOUTDEST ; Already done the figuring
|
|||
|
MOV DI,OFFSET TRANGROUP:USERDIR1
|
|||
|
mov bp,offset trangroup:DESTVARS
|
|||
|
call BUILDPATH
|
|||
|
call RESTUDIR1
|
|||
|
|
|||
|
; Now know all about the destination
|
|||
|
|
|||
|
KNOWABOUTDEST:
|
|||
|
xor al,al
|
|||
|
xchg al,[FIRSTDEST]
|
|||
|
or al,al
|
|||
|
jnz FIRSTDST
|
|||
|
jmp NOTFIRSTDEST
|
|||
|
FIRSTDST:
|
|||
|
mov si,[DESTTAIL] ; Create an FCB of the original DEST
|
|||
|
mov di,offset trangroup:DESTFCB
|
|||
|
mov ax,PARSE_FILE_DESCRIPTOR SHL 8
|
|||
|
INT int_command
|
|||
|
mov ax,word ptr [DESTBUF] ; Get drive
|
|||
|
cmp ah,':'
|
|||
|
jz DRVSPEC4
|
|||
|
mov al,'@'
|
|||
|
DRVSPEC4:
|
|||
|
MOV CL,[ASCII] ; Save current ASCII setting
|
|||
|
sub al,'@'
|
|||
|
mov [DESTFCB],al
|
|||
|
mov al,[DESTINFO]
|
|||
|
mov ah,[SRCINFO]
|
|||
|
and ax,0202H
|
|||
|
or al,al
|
|||
|
jz NOTMELCOPY
|
|||
|
cmp al,ah
|
|||
|
jnz NOTMELCOPY
|
|||
|
cmp [PLUS],0
|
|||
|
jz NOTMELCOPY
|
|||
|
inc [MELCOPY] ; ambig source, ambig dest, and pluses
|
|||
|
xor al,al
|
|||
|
jmp short SETCONC
|
|||
|
|
|||
|
NOTMELCOPY:
|
|||
|
xor al,2 ; al=2 if unambig dest, =0 if ambig dest
|
|||
|
and al,ah
|
|||
|
shr al,1 ; al=1 if unambig dest AND ambig sorce
|
|||
|
; Implies concatination
|
|||
|
SETCONC:
|
|||
|
or al,[PLUS] ; al=1 if concat
|
|||
|
mov [CONCAT],al
|
|||
|
shl al,1
|
|||
|
shl al,1
|
|||
|
mov [INEXACT],al ; Concat -> inexact copy
|
|||
|
cmp [BINARY],0
|
|||
|
jnz NOTFIRSTDEST ; Binary explicitly given, all OK
|
|||
|
mov [ASCII],al ; Concat -> ASCII
|
|||
|
or cl,cl
|
|||
|
jnz NOTFIRSTDEST ; ASCII flag set before, DATA read correctly
|
|||
|
or al,al
|
|||
|
JZ NOTFIRSTDEST ; ASCII flag did not change states
|
|||
|
; At this point there may already be binary read data in the read buffer.
|
|||
|
; We need to find the first ^Z (if there is one) and trim the amount
|
|||
|
; of data in the buffer correctly.
|
|||
|
MOV CX,[NXTADD]
|
|||
|
JCXZ NOTFIRSTDEST ; No data, everything OK
|
|||
|
MOV AL,1AH
|
|||
|
PUSH ES
|
|||
|
XOR DI,DI
|
|||
|
MOV ES,[TPA]
|
|||
|
REPNE SCASB ; Scan for EOF
|
|||
|
POP ES
|
|||
|
JNZ NOTFIRSTDEST ; No ^Z in buffer, everything OK
|
|||
|
DEC DI ; Point at ^Z
|
|||
|
MOV [NXTADD],DI ; New buffer
|
|||
|
NOTFIRSTDEST:
|
|||
|
mov bx,offset trangroup:DIRBUF+1 ; Source of replacement chars
|
|||
|
cmp [CONCAT],0
|
|||
|
jz GOTCHRSRC ; Not a concat
|
|||
|
mov bx,offset trangroup:SDIRBUF+1 ; Source of replacement chars
|
|||
|
GOTCHRSRC:
|
|||
|
mov si,offset trangroup:DESTFCB+1 ; Original dest name
|
|||
|
mov di,[DESTTAIL] ; Where to put result
|
|||
|
|
|||
|
BUILDNAME:
|
|||
|
mov cx,8
|
|||
|
BUILDMAIN:
|
|||
|
lodsb
|
|||
|
cmp al,"?"
|
|||
|
jnz NOTAMBIG
|
|||
|
mov al,byte ptr [BX]
|
|||
|
NOTAMBIG:
|
|||
|
cmp al,' '
|
|||
|
jz NOSTORE
|
|||
|
stosb
|
|||
|
NOSTORE:
|
|||
|
inc bx
|
|||
|
loop BUILDMAIN
|
|||
|
mov cl,3
|
|||
|
cmp byte ptr [SI],' '
|
|||
|
jz ENDDEST ; No extension
|
|||
|
mov al,'.'
|
|||
|
stosb
|
|||
|
BUILDEXT:
|
|||
|
lodsb
|
|||
|
cmp al,"?"
|
|||
|
jnz NOTAMBIGE
|
|||
|
mov al,byte ptr [BX]
|
|||
|
NOTAMBIGE:
|
|||
|
cmp al,' '
|
|||
|
jz NOSTOREE
|
|||
|
stosb
|
|||
|
NOSTOREE:
|
|||
|
inc bx
|
|||
|
loop BUILDEXT
|
|||
|
ENDDEST:
|
|||
|
xor al,al
|
|||
|
stosb ; NUL terminate
|
|||
|
return
|
|||
|
|
|||
|
TRANCODE ENDS
|
|||
|
END
|
|||
|
|