You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
3472 lines
85 KiB
3472 lines
85 KiB
page ,132
|
|
title DIR Internal Command
|
|
;/*
|
|
; * Microsoft Confidential
|
|
; * Copyright (C) Microsoft Corporation 1991
|
|
; * All Rights Reserved.
|
|
; */
|
|
|
|
;*** DIR.ASM - DIR internal command
|
|
|
|
comment % =================================================================
|
|
|
|
This module replaces TCMD1A.ASM. The old module was titled
|
|
"PART4 COMMAND Transient routines".
|
|
|
|
From residual documentation, I surmise that TCMD.ASM originally
|
|
contained the internal commands DIR, PAUSE, ERASE, TYPE, VOL, and
|
|
VER. The file seems to have been successively split:
|
|
|
|
TCMD -> TCMD1,TCMD2 -> TCMD1A,TCMD1B,TCMD2A,TCMD2B
|
|
|
|
TCMD1A.ASM contained only the DIR command.
|
|
|
|
Usage:
|
|
------
|
|
|
|
DIR <filespec> /w /p /b /s /l /o<sortorder> /a<attriblist>
|
|
|
|
DIR /?
|
|
|
|
|
|
<filespec> may include any or none of: drive; directory path;
|
|
wildcarded filename. If drive or directory path are
|
|
omitted, the current defaults are used. If the
|
|
file name or extension is omitted, wildcards are
|
|
assumed.
|
|
|
|
/w Wide listing format. Files are displayed in compressed
|
|
'name.ext' format. Subdirectory files are enclosed in
|
|
brackets, '[dirname]'.
|
|
|
|
/p Paged, or prompted listing. A screenful is displayed
|
|
at a time. The name of the directory being listed appears
|
|
at the top of each page.
|
|
|
|
Bugbug: pages nead to be uniform length..?
|
|
|
|
/b Bare listing format. Turns off /w or /p. Files are
|
|
listed in compressed 'name.ext' format, one per line,
|
|
without additional information. Good for making batch
|
|
files or for piping. When used with /s, complete
|
|
pathnames are listed.
|
|
|
|
/s Descend subdirectory tree. Performs command on current
|
|
or specified directory, then for each subdirectory below
|
|
that directory. Directory header and footer is displayed
|
|
for each directory where matching files are found, unless
|
|
used with /b. /b suppresses headers and footers.
|
|
|
|
Tree is explored depth first, alphabetically within the
|
|
same level.
|
|
|
|
Bugbug: hidden directories aren't searched.
|
|
|
|
/l Display file names, extensions and paths in lowercase. ;M010
|
|
|
|
/o Sort order. /o alone sorts by default order (dirs-first, name,
|
|
extension). A sort order may be specified after /o. Any of
|
|
the following characters may be used: nedsg (name, extension,
|
|
date/time, size, group-dirs-first). Placing a '-' before any
|
|
letter causes a downward sort on that field. E.g., /oe-d
|
|
means sort first by extension in alphabetical order, then
|
|
within each extension sort by date and time in reverse chronological
|
|
order.
|
|
|
|
/a Attribute selection. Without /a, hidden and system files
|
|
are suppressed from the listing. With /a alone, all files
|
|
are listed. An attribute list may follow /a, consisting of
|
|
any of the following characters: hsdar (hidden, system,
|
|
directory, archive, read-only). A '-' before any letter
|
|
means 'not' that attribute. E.g., /ar-d means files that
|
|
are marked read-only and are not directory files. Note
|
|
that hidden or system files may be included in the listing.
|
|
They are suppressed without /a but are treated like any other
|
|
attribute with /a.
|
|
|
|
/? Help listing. Display DIR useage information. ;M008;Handled externally
|
|
|
|
/h has been removed. ;M008
|
|
|
|
DIRCMD An environment variable named DIRCMD is parsed before the
|
|
DIR command line. Any command line options may be specified
|
|
in DIRCMD, and become defaults. /? will be ignored in DIRCMD.
|
|
A filespec may be specified in DIRCMD and will be used unless
|
|
a filespec is specified on the command line. Any switch
|
|
specified in DIRCMD may be overridden on the command line.
|
|
If the original DIR default action is desired for a particular
|
|
switch, the switch letter may be preceded by a '-' on the
|
|
command line. E.g.,
|
|
|
|
/-w use long listing format
|
|
/-p don't page the listing
|
|
/-b don't use bare format
|
|
/-s don't descend subdirectory tree
|
|
/-o display files in disk order
|
|
/-a suppress hidden and system files
|
|
|
|
|
|
Notes:
|
|
------
|
|
|
|
For sorted listings, file entries are loaded into the TPA buffer, which
|
|
is usually about 64K in size. This allows sorts of up to 3000 files at
|
|
a time. Each entry takes up 21 bytes in the buffer (see EntryStruc below).
|
|
The byte after the last entry is 0FFh. The first byte of each entry is
|
|
a flag byte which is made zero when the entry is loaded, and made one
|
|
when the entry is used.
|
|
|
|
|
|
Revision History
|
|
================
|
|
M01 md 7/13/90 Use ROM BIOS data area to obtain screen height
|
|
in the absence of ANSI.SYS
|
|
|
|
M007 sa 8/1/90 Allow /p/b combination
|
|
|
|
M008 sa 8/1/90 Remove /h parameter. Eliminate code used
|
|
to internally handle /? message.
|
|
|
|
M010 sa 8/5/90 Add support for /l (lowercase) option.
|
|
|
|
M011 sa 8/5/90 Patch up bug where MS-DOS does not load the
|
|
first FCB with the drive number when the drive
|
|
letter in the command line is preceded by a
|
|
switch. Now dir manually loads the drive
|
|
number after parsing.
|
|
|
|
M018 md 8/12/90 Increment the screen height by 1 when obtained
|
|
from the ROM BIOS.
|
|
|
|
M023 sa 8/31/90 Prevent DIR from failing if it encounters
|
|
a subdirectory having len(pathname)>MAXPATH.
|
|
Just skip over that subdirectory.
|
|
|
|
M028 dbo 9/24/90 When country=US, sort by strict character
|
|
byte value, rather than collating table.
|
|
This to match MS-DOS Shell's sort order.
|
|
|
|
========================================================================= %
|
|
|
|
|
|
|
|
|
|
;*** SYMBOLS & MACROS
|
|
|
|
.xlist
|
|
.xcref
|
|
|
|
include comsw.asm ; get COMMAND version switches
|
|
include dossym.inc ; get DOS basic symbol set
|
|
include syscall.inc ; get DOS call names
|
|
include doscntry.inc ; get extended country info symbols
|
|
include bpb.inc
|
|
include filemode.inc
|
|
include find.inc
|
|
include comseg.asm ; define segment order
|
|
include comequ.asm ; get equates for COMMAND routines
|
|
include ioctl.inc ; get symbols for ioctl's
|
|
include rombios.inc ; get ROM BIOS data definition
|
|
|
|
.list
|
|
.cref
|
|
|
|
;M008;NUM_DIR_SWS reduced by 2 for /h,/? not used
|
|
;M010;NUM_DIR_SWS increased by 2 for /l,/-l
|
|
NUM_DIR_SWS equ 14 ; # of dir switch synonyms in Dir_Sw_Ptrs list
|
|
|
|
;M010;'lcase' replaces removed 'help' in OptionRec
|
|
OptionRec record inmem:1,lcase:1,bare:1,subd:1,pagd:1,wide:1
|
|
; on/off bit record for /l, /b, /s, /p, /w options
|
|
; (order is hard-coded; see OnOffSw)
|
|
; Inmem is set when entries are loaded in memory.
|
|
|
|
NUM_ATTR_LTRS equ 6 ; length of attribute letter list
|
|
NUM_ORDER_LTRS equ 5 ; length of sort order letter list
|
|
|
|
ResultBuffer struc ; structure of parse result buffer
|
|
ValueType db ?
|
|
ValueTag db ?
|
|
SynPtr dw ?
|
|
ValuePtr dd ?
|
|
ResultBuffer ends
|
|
|
|
ErrorRec record baddir:1,dev:1
|
|
; Error bits are:
|
|
; Invalid directory format
|
|
; File is device
|
|
|
|
EntryStruc struc ; our private directory entry structure
|
|
used db ? ; =0 until entry used, then =1
|
|
filename db 8 dup (?) ; filename
|
|
fileext db 3 dup (?) ; extension
|
|
fileattr db ? ; file attributes
|
|
filetime dw ? ; file time
|
|
filedate dw ? ; file date
|
|
filesize dd ? ; file size
|
|
EntryStruc ends
|
|
|
|
shove macro val ; hose-bag 8086 doesn't push immediate
|
|
mov ax,val ; invisible, dangerous use of AX!
|
|
push ax
|
|
endm
|
|
|
|
|
|
|
|
|
|
;*** DATA
|
|
|
|
DATARES segment public byte
|
|
|
|
extrn Append_Flag:byte ; true when APPEND needs to be reset
|
|
extrn Append_State:word ; state to reset APPEND to
|
|
|
|
DATARES ends
|
|
|
|
|
|
TRANDATA segment public byte
|
|
|
|
extrn AttrLtrs:byte ; list of attribute letters
|
|
extrn BadCd_Ptr:word ; "invalid directory" msg block
|
|
extrn Bytes_Ptr:word ; "%1 bytes" msg block
|
|
extrn BytMes_Ptr:word ; "%1 bytes free" msg block
|
|
extrn DirCont_Ptr:word ; "(continuing %1)" msg block
|
|
extrn DirDat_Yr:word ; year field of date msg block
|
|
extrn DirDat_Mo_Day:word ; month/day field of date msg block
|
|
extrn DirDatTim_Ptr:word ; date/time msg block
|
|
extrn DirEnvVar:byte ; DIR environment variable name
|
|
extrn DirHead_Ptr:word ; directory header message block
|
|
extrn DirMes_Ptr:word ; "%1 File(s)" msg block
|
|
extrn DirTim_Hr_Min:word ; time field of msg block
|
|
extrn Dir_Sw_Ptrs:word ; list of DIR switch synonym ptrs
|
|
extrn Disp_File_Size_Ptr:word ; file size message block
|
|
extrn DMes_Ptr:word ; <DIR> message block
|
|
extrn ErrParsEnv_Ptr:word ; "(Error occurred in env.." msg blk
|
|
extrn Extend_Buf_Ptr:word ; extended error message block
|
|
extrn Extend_Buf_Sub:byte ; # substitions in message block
|
|
extrn Msg_Disp_Class:byte ; message display class
|
|
extrn OrderLtrs:byte ; list of sort order letters
|
|
extrn Parse_Dir:byte ; DIR parse block
|
|
extrn String_Buf_Ptr:word ; message block ptr to string
|
|
extrn Tab_Ptr:word ; tab output block
|
|
extrn Total_Ptr:word ; "Total:" msg block
|
|
|
|
TRANDATA ends
|
|
|
|
|
|
TRANSPACE segment public byte
|
|
|
|
extrn AttrSelect:byte ; attribute select states -
|
|
extrn AttrSpecified:byte ; attribute mask -
|
|
|
|
; Attribute conditions are recorded in two steps.
|
|
; AttrSpecified indicates which attributes are to be checked.
|
|
; AttrSelect indicates which state the specified attributes
|
|
; must be in for a file to be included in the listing.
|
|
; Attributes not indicated in AttrSpecified are ignored when
|
|
; deciding which files to include.
|
|
|
|
extrn Bits:word ; some option flags (see OptionRec)
|
|
extrn BwdBuf:byte ; 'build working dir string' buf
|
|
extrn BytCnt:word ; # bytes in TPA
|
|
extrn Bytes_Free:word ; #bytes free for BytMes_Ptr msg block
|
|
extrn CharBuf:byte ; character string buffer
|
|
extrn ComSw:word ; error bits (see ErrorRec)
|
|
extrn CountryPtrInfo:byte ; buffer for collating table ptr
|
|
extrn CountryPtrId:byte ; info ID for collating table ptr
|
|
extrn CountryPtr:dword ; collating table ptr
|
|
extrn CurDrv:byte ; current drive # (0-based)
|
|
extrn DestBuf:byte ; null-terminated sort codes -
|
|
|
|
; Sort order is specified as a series of 0 to 5 sort code
|
|
; bytes, followed by a zero byte.
|
|
; Codes are 1=name, 2=extension, 3=date&time, 4=size, and
|
|
; 5=filetype (subdir or not).
|
|
; Bit 7 of code is set for a downwards sort.
|
|
|
|
extrn DestIsDir:byte ; indicator of delim char's in path
|
|
extrn DestTail:word ; ptr to filename in pathname
|
|
extrn Dir_Num:word ; #files for DirMes_Ptr msg block
|
|
extrn DirBuf:byte ; DTA buffer for DOS calls
|
|
extrn DirFlag:byte ; signal to PathCrunch routine
|
|
extrn Display_Ioctl:word ; display info block for IOCTL call
|
|
extrn EndDestBuf:byte ; end of DestBuf (sort order codes)
|
|
extrn File_Size_High:word ; field for file size message block
|
|
extrn File_Size_Low:word ; field for file size message block
|
|
extrn FileCnt:word ; file count in a single directory
|
|
extrn FileCntTotal:dword ; file count in all directories
|
|
extrn FileSiz:dword ; file sizes in a single directory
|
|
extrn FileSizTotal:dword ; file sizes in all directories
|
|
extrn InternatVars:byte ; buffer for international info
|
|
extrn LeftOnLine:byte ; entries left on current display line
|
|
extrn LeftOnPage:word ; lines left on page
|
|
extrn LinPerPag:word ; lines/page entry in Display_Ioctl
|
|
extrn Msg_Numb:word ; extended error code
|
|
extrn OldCtrlCHandler:dword ; old int 23 vector
|
|
extrn Parse1_Syn:word ; ptr to matched synonym
|
|
extrn PathCnt:word ; length of pathname (see PathPos)
|
|
extrn PathPos:word ; ptr to pathname buffer (see SrcBuf)
|
|
extrn PerLine:byte ; # entries per line
|
|
extrn ResSeg:word ; RESGROUP seg addr
|
|
extrn ScanBuf:byte ; buffer for environment value and
|
|
; subdirectory names
|
|
extrn SrcBuf:byte ; pathname buffer
|
|
extrn String_Ptr_2:word ; message substitution string ptr
|
|
extrn Tpa:word ; TPA buffer seg addr
|
|
|
|
TRANSPACE ends
|
|
|
|
|
|
|
|
|
|
;*** PRINCIPAL ROUTINES
|
|
|
|
TRANCODE segment public byte
|
|
|
|
extrn CError:near ; COMMAND error recycle point
|
|
|
|
public Catalog ; our entry point
|
|
|
|
break <DIR (Catalog) principal routines>
|
|
|
|
assume cs:TRANGROUP,ds:TRANGROUP,es:nothing,ss:TRANGROUP
|
|
|
|
; Bugbug: Each routine should start with it's own ASSUME.
|
|
|
|
|
|
|
|
|
|
;*** Catalog - DIR command main routine
|
|
;
|
|
; ENTRY FCB #1 in PSP has drive# from cmd-line or default
|
|
; Cmd-line tail text is at 81h, terminated by 0Dh
|
|
; CS, DS, ES, SS = TRANGROUP seg addr
|
|
; Tpa = TPA buffer seg addr
|
|
; BytCnt = # bytes in TPA buffer
|
|
;
|
|
; EXIT nothing
|
|
;
|
|
; USED AX,BX,CX,DX,SI,DI,BP
|
|
;
|
|
; ERROR EXITS
|
|
;
|
|
; Errors are handled by setting up error message pointers
|
|
; for Std_EPrintf and jumping to CError. Syntax errors in
|
|
; the environment variable, however, are handled by printing
|
|
; an error message and continuing.
|
|
;
|
|
; EFFECTS
|
|
;
|
|
; Directory listing is displayed (on standard output).
|
|
; APPEND is disabled. HeadFix routine is expected to
|
|
; restore APPEND state.
|
|
; Working directory may be changed. The user's default
|
|
; directory is saved and flagged for restoration by RestUDir
|
|
; during COMMAND cycle.
|
|
; Lots of variables may be changed in TRANSPACE segment.
|
|
;
|
|
; NOTES
|
|
;
|
|
; ES = TRANGROUP seg addr except when used to address the
|
|
; the TPA buffer, where directory entries are loaded from disk.
|
|
|
|
Catalog proc
|
|
|
|
call SetDefaults
|
|
call ParseEnvironment
|
|
call ParseCmdLine
|
|
jnc @F ; no parse error
|
|
jmp catErr ; error msg is set up
|
|
@@: call SetOptions
|
|
call SetCollatingTable
|
|
|
|
; Drive # to operate on has already been placed in FCB by
|
|
; COMMAND preprocessing. OkVolArg & PathCrunch depend on that.
|
|
|
|
test Bits,mask bare
|
|
jnz @F ; don't display volume info for /b
|
|
invoke OkVolArg ; find & display volume info
|
|
sub LeftOnPage,2 ; record display lines used by volume info
|
|
jmp short catCrunch
|
|
|
|
; OkVolArg side effects:
|
|
; APPEND is disabled;
|
|
; DTA established at DirBuf;
|
|
; Filename fields in FCB are wildcarded.
|
|
|
|
@@:
|
|
; OkVolArg wasn't executed, so we have to do these ourselves.
|
|
|
|
invoke DisAppend ; disable APPEND
|
|
|
|
mov dx,offset TRANGROUP:DirBuf
|
|
mov ah,Set_DMA
|
|
int 21h ; set DTA
|
|
|
|
mov di,FCB ; ES:DI = ptr to FCB
|
|
inc di ; ES:DI = ptr to filename field of FCB
|
|
mov al,'?' ; AL = wildcard character
|
|
mov cx,11
|
|
rep stosb ; wildcard filename field
|
|
|
|
catCrunch:
|
|
call CrunchPath ; crunch pathname to get directory and filename
|
|
jc catRecErr ; handle recorded or extended error
|
|
|
|
; User's directory has been saved, we've changed to specified directory.
|
|
; ComSw = error bits for later use
|
|
; FCB contains parsed filename
|
|
|
|
cmp ComSw,0
|
|
jne catRecErr ; handle recorded error
|
|
|
|
call InstallCtrlC ; install control-C handler
|
|
|
|
call ZeroTotals ; zero grand totals
|
|
call ListDir ; list main directory
|
|
jc catExtErr
|
|
|
|
test Bits,mask subd
|
|
jz @F ; subdirectories option not set
|
|
call ListSubds ; list subdirectories
|
|
jc catExtErr
|
|
@@:
|
|
; Check if any files were found.
|
|
|
|
test Bits,mask bare
|
|
jnz catRet ; don't bother for bare format
|
|
|
|
mov ax,word ptr FileCntTotal
|
|
or ax,ax
|
|
jz catNoFiles ; no files found
|
|
|
|
call DisplayTotals ; display trailing grand totals
|
|
jmp short catRet ; all done
|
|
|
|
catRecErr:
|
|
|
|
; ComSw may have error bit set. If not, do extended error.
|
|
|
|
test ComSw,mask dev
|
|
jnz catNoFiles ; filename is device, respond 'file not found'
|
|
|
|
test ComSw,mask baddir
|
|
jz catExtErr ; no ComSw error bits, must be extended error
|
|
mov dx,offset TRANGROUP:BadCd_Ptr ; invalid directory
|
|
jmp short catErr
|
|
|
|
catNoFiles:
|
|
|
|
; Display header and force 'file not found' message.
|
|
|
|
call DisplayHeader
|
|
mov ax,ERROR_FILE_NOT_FOUND
|
|
mov Msg_Disp_Class,EXT_MSG_CLASS
|
|
mov dx,offset TRANGROUP:Extend_Buf_ptr
|
|
mov Extend_Buf_ptr,ax
|
|
jmp short catErr
|
|
|
|
catExtErr:
|
|
|
|
; DOS has returned an error status. Get the extended error#, and
|
|
; set up an error message, changing 'No more files' error
|
|
; to 'File not found' error.
|
|
|
|
invoke Set_Ext_Error_Msg
|
|
cmp Extend_Buf_Ptr,ERROR_NO_MORE_FILES
|
|
jne @F
|
|
mov Extend_Buf_Ptr,ERROR_FILE_NOT_FOUND
|
|
@@:
|
|
|
|
; Error exit. Error message information has been set up
|
|
; for Std_EPrintf.
|
|
|
|
catErr: jmp CError ; go to COMMAND error recycle point
|
|
|
|
catRet: ret
|
|
|
|
Catalog endp
|
|
|
|
|
|
|
|
|
|
;*** SetDefaults - set default pathname, options
|
|
;
|
|
; ENTRY DS = TRANGROUP seg addr
|
|
;
|
|
; EXIT nothing
|
|
;
|
|
; USED AX,DI
|
|
;
|
|
; EFFECTS
|
|
; SrcBuf = '*',EOL - default pathname
|
|
; PathPos = ptr to pathname
|
|
; PathCnt = length of pathname
|
|
|
|
|
|
|
|
SetDefaults proc
|
|
|
|
mov di,offset TRANGROUP:SrcBuf ; DI = ptr to pathname buffer
|
|
mov PathPos,di ; PathPos = ptr to pathname
|
|
mov al,STAR
|
|
stosb
|
|
mov al,END_OF_LINE_IN
|
|
stosb ; SrcBuf = '*',0Dh
|
|
mov PathCnt,1 ; PathCnt = pathname length
|
|
|
|
xor ax,ax ; AX = 0
|
|
mov ComSw,ax ; = no error
|
|
mov Bits,ax ; = options off
|
|
mov DestBuf,al ; = no sort
|
|
mov AttrSpecified,ATTR_HIDDEN+ATTR_SYSTEM
|
|
mov AttrSelect,al ; exclude hidden, system files
|
|
|
|
ret
|
|
|
|
SetDefaults endp
|
|
|
|
|
|
|
|
|
|
;*** ParseEnvironment - find and parse our environment variable
|
|
;
|
|
; Find our environment variable and parse it. If a parse
|
|
; error occurs, issue an error message. The parse results
|
|
; up to the error will still have effect. Always leave
|
|
; the option variables in a useable state.
|
|
;
|
|
; ENTRY DS = TRANGROUP seg addr
|
|
;
|
|
; EXIT nothing
|
|
;
|
|
; USED AX,BX,CX,DX,SI,DI
|
|
;
|
|
; EFFECTS
|
|
;
|
|
; Bits may contain new option settings.
|
|
; DestBuf may contain new series of sort codes.
|
|
; AttrSpecified, AttrSelect may contain new attribute conditions.
|
|
; SrcBuf may contain a new default pathname/filespec.
|
|
; PathPos, PathCnt updated for new pathname.
|
|
;
|
|
; If a parse error occurred, an error message will be issued.
|
|
|
|
ParseEnvironment proc
|
|
|
|
call GetEnvValue ; get environment variable value
|
|
jc peRet ; name not found in environment
|
|
|
|
; SI = ptr to value of environment variable, in TRANGROUP seg
|
|
|
|
call ParseLine ; parse environment value
|
|
cmp ax,END_OF_LINE
|
|
je peRet ; successful completion
|
|
|
|
; Some kind of parse error occurred.
|
|
; We're set up for a Std_EPrintf call.
|
|
|
|
invoke Std_EPrintf ; display the parse error
|
|
mov Msg_Disp_Class,UTIL_MSG_CLASS ; restore default msg class
|
|
|
|
mov dx,offset TRANGROUP:ErrParsEnv_Ptr
|
|
invoke Printf_Crlf ; "(Error occurred in environment.."
|
|
|
|
;M008;Internal handling of /? removed
|
|
;peOk: and Bits,not mask help ; disallow /h in environment variable
|
|
|
|
peRet: ret
|
|
|
|
ParseEnvironment endp
|
|
|
|
|
|
|
|
|
|
;*** ParseCmdLine - parse and record command line parameters
|
|
;
|
|
; ENTRY PSP offset 81h is beginning of cmd line buffer
|
|
; DS, ES, CS = TRANGROUP seg addr
|
|
;
|
|
; EXIT CY = set if parse error occurred
|
|
;
|
|
; If parse error occurred, we're set up for Std_EPrintf call:
|
|
; AX = system parser error code
|
|
; DX = ptr to message block
|
|
;
|
|
; USED AX,BX,CX,DX,SI,DI
|
|
;
|
|
; EFFECTS
|
|
;
|
|
; Bits may contain new option settings.
|
|
; DestBuf may contain new series of sort codes.
|
|
; AttrSpecified, AttrSelect may contain new attribute conditions.
|
|
; SrcBuf may contain a new default pathname/filespec.
|
|
; PathPos, PathCnt updated for new pathname.
|
|
;
|
|
; If parse error occurred, we're set up for Std_EPrintf call:
|
|
; Msg_Disp_Class = parse error class
|
|
; Byte after last parameter in text is zeroed to make ASCIIZ string
|
|
; Message block (see DX) is set up for parse error message
|
|
|
|
ParseCmdLine proc
|
|
|
|
mov si,81h ; SI = ptr to cmd-line tail text
|
|
call ParseLine ; parse cmd line tail
|
|
cmp AX,END_OF_LINE
|
|
je pcOk ; parse completed successfully
|
|
|
|
; A parse error occurred. We're all set up for message output.
|
|
|
|
stc ; return failure
|
|
jmp short pcRet
|
|
|
|
pcOk: clc ; return success
|
|
|
|
pcRet: ret
|
|
|
|
ParseCmdLine endp
|
|
|
|
|
|
|
|
|
|
;*** SetCollatingTable - set up character collating table for sorting
|
|
;
|
|
; If country is other than USA, try to get a collating table
|
|
; for character sorting. For USA, use straight byte values.
|
|
; This is so DIR behaves like the MS-DOS Shell, which sorts
|
|
; by straight byte values in the USA for better performance.
|
|
;
|
|
; ENTRY ES = TRANGROUP seg addr
|
|
;
|
|
; EXIT nothing
|
|
;
|
|
; USED AX,BX,CX,DX,DI
|
|
;
|
|
; EFFECTS
|
|
;
|
|
; If collating table is set -
|
|
; CountryPtrId = 6.
|
|
; CountryPtr points to collating table.
|
|
;
|
|
; Otherwise -
|
|
; CountryPtrId = 0.
|
|
|
|
SetCollatingTable proc
|
|
|
|
; Begin modification M028
|
|
|
|
mov dx,offset TRANGROUP:InternatVars
|
|
; DS:DX = ptr to international info buffer
|
|
mov ax,INTERNATIONAL shl 8
|
|
; AX = 'Get current country info'
|
|
int 21h ; call DOS
|
|
jc scNoTable ; error - so don't collate
|
|
|
|
; BX = country code
|
|
|
|
cmp bx,1
|
|
je scNoTable ; we're in USA, don't collate
|
|
|
|
; End modification M028
|
|
|
|
;* Country code is other than USA. Try to get a collating table.
|
|
|
|
mov ax,(GETEXTCNTRY shl 8) + SETCOLLATE
|
|
; AH = 'Get Extended Country Info'
|
|
; AL = 'Get Pointer to Collating Table'
|
|
mov bx,-1 ; BX = code page of interest = CON
|
|
mov cx,5 ; CX = length of info buffer
|
|
mov dx,bx ; DX = country ID = default
|
|
mov di,offset TRANGROUP:CountryPtrInfo
|
|
; ES:DI = ptr to info buffer
|
|
int 21h ; call DOS
|
|
jnc scRet ; success
|
|
|
|
;* Set CountryPtrId = 0 to signal no collating table.
|
|
|
|
scNoTable: ;M028
|
|
mov CountryPtrId,0
|
|
|
|
scRet: ret
|
|
|
|
SetCollatingTable endp
|
|
|
|
|
|
|
|
|
|
;*** SetOptions - check and set options
|
|
;
|
|
; ENTRY nothing
|
|
;
|
|
; EXIT nothing
|
|
;
|
|
; USED AX,BX,CX,DX
|
|
;
|
|
; EFFECTS
|
|
;
|
|
; Bits may contain modified option settings.
|
|
; Display_Ioctl table, including LinPerPag variable, is filled in.
|
|
; LeftOnPage is initialized to # lines till end of page is handled.
|
|
; PerLine is set according to /w presence.
|
|
|
|
SetOptions proc
|
|
|
|
; If bare listing requested, cancel wide listings.
|
|
|
|
test Bits,mask bare
|
|
jz @F
|
|
and Bits,not mask wide ;M007;Allow /p with /b
|
|
|
|
@@:
|
|
|
|
; Set # lines per display page.
|
|
|
|
;M01 Obtain screen height from ROM BIOS data area
|
|
;
|
|
;M01 mov LinPerPag,LINESPERPAGE ; default value
|
|
|
|
ifndef JAPAN
|
|
push ds
|
|
MOV AX,ROMBIOS_DATA ; Get ROM Data segment
|
|
MOV DS,AX ;
|
|
Assume DS:ROMBIOS_DATA
|
|
|
|
MOV al,CRT_Rows ; Get max rows
|
|
pop ds ;
|
|
Assume DS:Trangroup
|
|
|
|
or al,al ; If zero specified
|
|
jnz @F ;
|
|
endif
|
|
MOV al,LINESPERPAGE ; assume 24 rows
|
|
|
|
@@:
|
|
xor ah,ah
|
|
ifndef JAPAN
|
|
inc al ; height + 1 ;M018
|
|
endif
|
|
mov LinPerPag,ax ; set the rows now
|
|
|
|
; Now the console driver can change the rows if it knows better (M01 end)
|
|
|
|
mov ax,(IOCTL shl 8)+GENERIC_IOCTL_HANDLE ; IOCTL for handles
|
|
mov bx,STDOUT ; handle #
|
|
mov ch,IOC_SC ; screen
|
|
mov cl,GET_GENERIC ; get display info
|
|
mov dx,offset TRANGROUP:Display_Ioctl ; info block
|
|
int 21h ; call DOS
|
|
|
|
mov ax,LinPerPag ; AX = # lines per page
|
|
mov LeftOnPage,ax ; initialize # lines left on page
|
|
|
|
; Set # entries per line.
|
|
|
|
mov PerLine,NORMPERLIN ; # entries per line without /w
|
|
test Bits,mask wide
|
|
jz @F
|
|
mov PerLine,WIDEPERLIN ; # entries per line with /w
|
|
@@:
|
|
;M011;start;The following code checks if a drive
|
|
;letter has been parsed into SrcBuf, and if
|
|
;so, the correct drive number is loaded into
|
|
;the first FCB, at offset 5C.
|
|
|
|
cmp TRANGROUP:[SrcBuf+1],COLON_CHAR ; is this a drive letter?
|
|
jne soRet
|
|
mov al,TRANGROUP:[SrcBuf] ; load drive letter into al
|
|
and al,not 20h ; capitalize ASCII drive letter (LowerCase-32)-->UpperCase
|
|
sub al,'@' ; convert to 1-based number (1=A)
|
|
mov ds:FCB,al ; store in first FCB
|
|
;M011;end
|
|
soRet: ret
|
|
|
|
SetOptions endp
|
|
|
|
|
|
|
|
|
|
;*** CrunchPath - analyze supplied or default pathname
|
|
;
|
|
; ENTRY PathPos = ptr to pathname buffer
|
|
; PathCnt = length of pathname, not incl trailing delimiter
|
|
; Pathname in buffer must end in delimiter (like CR) and
|
|
; must have space for another char after the delimiter.
|
|
;
|
|
; EXIT CY = clear if no error
|
|
; We are changed to directory found in pathname
|
|
; Previous directory ready to be restored via RestUDir
|
|
; FCB filename fields contain filename (possibly w/ wildcards)
|
|
;
|
|
; If error occurred,
|
|
; CY = set
|
|
; ComSw = error bits (see ErrorRec)
|
|
; If ComSw not set,
|
|
; Ready for DOS Get Extended Error call
|
|
|
|
|
|
CrunchPath proc
|
|
|
|
call FileIsDevice
|
|
jne @F ; not a device, skip ahead
|
|
or ComSw,mask dev ; signal file is device
|
|
jmp short cpErr ; return error
|
|
@@:
|
|
push PathPos ; save ptr to pathname
|
|
mov DirFlag,-1 ; tell PathCrunch not to parse file into FCB
|
|
invoke PathCrunch ; change to directory in pathname
|
|
mov DirFlag,0 ; reset our little flag
|
|
pop si ; SI = ptr to pathname
|
|
jc cpNoDir ; didn't find directory path
|
|
jz cpRet ; found directory path w/ no filename
|
|
; - leave wildcard default in FCB and return
|
|
|
|
;* We found a directory, and there was a filename attached.
|
|
; DestTail = ptr to ASCIIZ filename
|
|
|
|
mov si,DestTail ; SI = ptr to filename
|
|
jmp short cpFile ; go parse the file into FCB
|
|
|
|
;* PathCrunch failed to find a directory in the pathname.
|
|
;
|
|
; Msg_Numb = error code
|
|
; DestIsDir = nonzero if path delimiter char's occur in pathname
|
|
; SI = ptr to pathname (now an ASCIIZ string)
|
|
|
|
cpNoDir:
|
|
mov ax,Msg_Numb ; AX = error code from PathCrunch
|
|
or ax,ax
|
|
jnz cpErr ; error occurred - return it
|
|
cmp DestIsDir,0
|
|
je cpMaybe ; no path delimiters seen, maybe it's a file
|
|
or ComSw,mask baddir ; signal invalid directory name
|
|
jmp short cpErr ; return error
|
|
|
|
cpMaybe:
|
|
|
|
; SI = ptr to pathname
|
|
|
|
cmp byte ptr [si+1],COLON_CHAR
|
|
jnz @F ; no drive specifier, skip ahead
|
|
lodsw ; SI = ptr past drive specifier "d:"
|
|
@@: cmp [si],".."
|
|
jne cpFile ; if not "..", treat as a file
|
|
cmp byte ptr [si+2],0
|
|
jne cpFile ; or if there's more after "..", treat as file
|
|
or ComSw,mask baddir ; signal invalid directory
|
|
jmp short cpErr ; return error
|
|
|
|
; The preceding code was taken from the old DIR routine.
|
|
; It's garbage, I'm afraid. It's meant to check for ".."
|
|
; occurring when we're at the root directory. Too bad it
|
|
; doesn't handle problems with "..\..", etc.
|
|
|
|
|
|
; We're ready to parse a filename into the FCB.
|
|
; SI = ptr to ASCIIZ filename
|
|
|
|
cpFile: mov di,FCB ; DI = ptr to FCB
|
|
mov ax,(PARSE_FILE_DESCRIPTOR shl 8) or 0Eh
|
|
; wildcards already in FCB used as defaults
|
|
int 21h
|
|
clc ; return success
|
|
jmp short cpRet
|
|
|
|
cpErr: stc ; return error
|
|
|
|
cpRet: ret
|
|
|
|
CrunchPath endp
|
|
|
|
|
|
|
|
|
|
;*** InstallCtrlC - install our private control-C handler
|
|
;
|
|
; Put our control-c handler in front of command.com's default
|
|
; handler, to make sure the user's default directory gets restored.
|
|
; This shouldn't be necessary, but, for now, there are situations
|
|
; where the TDATA segment is left in a modified state when a
|
|
; control-c occurs. This means that the transient will be
|
|
; reloaded, and the user's directory cannot be restored.
|
|
;
|
|
; Bugbug: fix the wider problem? Involves message services. Ugly.
|
|
;
|
|
; ENTRY nothing
|
|
;
|
|
; EXIT nothing
|
|
;
|
|
; USED AX,BX,DX
|
|
;
|
|
; EFFECTS
|
|
;
|
|
; CtrlCHandler address placed in int 23 vector.
|
|
;
|
|
; NOTE
|
|
;
|
|
; Command.com's basic control-c handler will be restored
|
|
; to the int 23 vector by the HeadFix routine, after DIR finishes.
|
|
|
|
InstallCtrlC proc
|
|
|
|
push es ; preserve ES
|
|
mov ax,(GET_INTERRUPT_VECTOR shl 8) + 23h
|
|
int 21h
|
|
mov word ptr OldCtrlCHandler,bx ; save old int 23 vector
|
|
mov word ptr OldCtrlCHandler+2,es
|
|
pop es ; restore ES
|
|
|
|
mov dx,offset TRANGROUP:CtrlCHandler ; DS:DX = ptr to CtrlCHandler
|
|
mov ax,(SET_INTERRUPT_VECTOR shl 8) + 23h
|
|
int 21h
|
|
ret
|
|
|
|
InstallCtrlC endp
|
|
|
|
|
|
|
|
|
|
;*** ListSubds - search and list files in subdirectories
|
|
;
|
|
; ENTRY Current directory (on selected drive) is top of subdir tree
|
|
; FCB is still set up for file searches
|
|
; Bits, AttrSpecified, AttrSelect, DestBuf all still set up
|
|
;
|
|
; EXIT CY = clear if no error
|
|
; FileCnt = # files found & displayed
|
|
; FileSiz = total size of files found
|
|
;
|
|
; If error,
|
|
; CY = set
|
|
; Ready for DOS Get Extended Error call
|
|
;
|
|
; USED AX,BX,CX,DX,SI,DI,BP
|
|
;
|
|
; EFFECTS
|
|
;
|
|
; FileCntTotal, FileSizTotal are updated.
|
|
; Subdirectories may be listed on standard output device.
|
|
;
|
|
; NOTES
|
|
;
|
|
; ListSubds seeds the recursive entry point lsNode with a ptr
|
|
; to a buffer where we'll stack up subdirectory filenames.
|
|
; Each name is stored ASCIIZ.
|
|
|
|
ListSubds proc
|
|
|
|
invoke SetRest1 ; make sure user's dir gets restored
|
|
|
|
mov bx,offset TRANGROUP:ScanBuf ; BX = ptr to child name buffer
|
|
|
|
lsNode:
|
|
mov byte ptr ds:[bx],0 ; start with null child name
|
|
lsLoop:
|
|
call FindNextChild ; search for next subdirectory
|
|
jc lsErr ; search failed - examine error
|
|
|
|
mov dx,bx ; DX = ptr to child's name
|
|
call ChangeDir ; enter child directory
|
|
|
|
; M023;start
|
|
jnc @F ; check for error
|
|
cmp ax,ERROR_PATH_NOT_FOUND ; error due to len(pathname)>MAXPATH?
|
|
je lsLoop ; yes, skip over this subdirectory
|
|
jmp SHORT lsRet ; no, other error: DIR must fail
|
|
; M023;end
|
|
|
|
@@: push bx
|
|
call ListDir ; list the directory
|
|
pop bx
|
|
|
|
; Note we're ignoring errors returned here.
|
|
|
|
mov di,bx ; DI = ptr to child's name
|
|
mov cx,13 ; CX = max name length w/ null
|
|
xor al,al ; AL = zero byte to look for
|
|
repne scasb ; DI = ptr to next name pos'n in buf
|
|
push bx ; save ptr to child's name
|
|
mov bx,di ; BX = ptr to next name pos'n in buf
|
|
call lsNode ; recurse from new node
|
|
pop bx ; BX = ptr to child's name
|
|
pushf ; save error condition
|
|
|
|
shove 0
|
|
shove ".."
|
|
mov dx,sp ; DX = ptr to "..",0 on stack
|
|
call ChangeDir ; return to parent directory
|
|
pop ax ; restore stack
|
|
pop ax
|
|
|
|
popf ; restore error condition from child
|
|
jc lsRet ; return error
|
|
|
|
jmp lsLoop ; look for more children
|
|
lsErr:
|
|
invoke Get_Ext_Error_Number ; AX = extended error code
|
|
cmp ax,ERROR_FILE_NOT_FOUND
|
|
je lsRet ; file not found, we're ok
|
|
cmp ax,ERROR_NO_MORE_FILES
|
|
je lsRet ; no more files, we're ok
|
|
stc ; return other errors
|
|
|
|
lsRet: ret
|
|
|
|
ListSubds endp
|
|
|
|
|
|
|
|
|
|
break <DIR support routines>
|
|
|
|
;*** SUPPORT ROUTINES
|
|
|
|
|
|
|
|
|
|
;*** CheckChild - check potential subdirectory name for FindNextChild
|
|
;
|
|
; ENTRY DirBuf contains DOS Find-buffer with potential child
|
|
; BX = ptr to last child's name
|
|
; BP = ptr to temp child's name
|
|
;
|
|
; EXIT nothing
|
|
;
|
|
; USED AX,CX,SI,DI
|
|
;
|
|
; EFFECTS
|
|
;
|
|
; Filename pointed to by BP may be changed.
|
|
;
|
|
; NOTES
|
|
;
|
|
; Potential filename replaces temp filename if:
|
|
; it's a subdirectory file;
|
|
; it doesn't start with a '.';
|
|
; it's alphanumerically greater than last child's name;
|
|
; and it's alphanumerically less than temp name.
|
|
|
|
CheckChild proc
|
|
|
|
test DirBuf.find_buf_attr,ATTR_DIRECTORY
|
|
jz ccRet ; not a subdirectory file- return
|
|
|
|
cmp DirBuf.find_buf_pname,'.'
|
|
je ccRet ; starts with a dot- return
|
|
|
|
mov si,offset TRANGROUP:DirBuf+find_buf_pname
|
|
mov di,bx
|
|
call CmpAscz ; compare candidate to last child's name
|
|
jna ccRet ; it's not above it- return
|
|
|
|
mov si,offset TRANGROUP:DirBuf+find_buf_pname
|
|
mov di,bp
|
|
call CmpAscz ; compare candidate to temp name
|
|
jnb ccRet ; it's not below it- return
|
|
|
|
; New kid is alright. Copy to temp.
|
|
|
|
mov si,offset TRANGROUP:DirBuf+find_buf_pname
|
|
mov di,bp
|
|
mov cx,13
|
|
rep movsb
|
|
|
|
ccRet: ret
|
|
|
|
CheckChild endp
|
|
|
|
|
|
|
|
|
|
;*** CmpEntry - compare one directory entry to another in sort order
|
|
;
|
|
; Compare one directory entry against another according to
|
|
; the sort codes in DestBuf. One or more comparisons
|
|
; may be made of file name, extension, time/date, and
|
|
; size. Comparisons may be made for upward or downward
|
|
; sort order.
|
|
;
|
|
; ENTRY ES:BX = ptr to entry to compare
|
|
; ES:BP = ptr to entry to be compared against
|
|
; DestBuf contains sort codes (see DestBuf)
|
|
; DS = TRANGROUP seg addr
|
|
;
|
|
; EXIT BX = unchanged
|
|
; BP = unchanged
|
|
; Condition flags set for same, above, or below
|
|
; comparing BX entry against BP entry.
|
|
; 'Same, above, below' translate to 'same, after, before'.
|
|
;
|
|
; USED: AX,CX,DX,SI,DI
|
|
|
|
CmpEntry proc
|
|
|
|
mov si,offset TRANGROUP:DestBuf ; (DS:SI) = ptr to sort codes
|
|
ceLoop:
|
|
xor ax,ax ; AX = 0
|
|
mov al,[si] ; AL = sort code
|
|
or al,al
|
|
jz ceDone ; sort code is zero, we're done
|
|
inc si ; DS:SI = ptr to next sort code
|
|
push si ; save ptr to next sort code
|
|
dec al
|
|
sal al,1 ; AX = index into cmp call table
|
|
; CY set for downward sort order
|
|
mov si,ax ; SI = index into cmp call table
|
|
mov ax,cs:FieldCmps[si] ; AX = addr of compare routine
|
|
jc ceDn ; downwards sort - go swap entries
|
|
call ax ; do upwards sort
|
|
jmp short @F
|
|
ceDn:
|
|
xchg bx,bp ; swap entry ptrs for downward sort order
|
|
call ax ; do sort
|
|
xchg bx,bp ; swap ptrs back
|
|
@@:
|
|
pop si ; SI = ptr to next sort code
|
|
je ceLoop ; compare showed no difference, keep trying
|
|
|
|
ceDone:
|
|
|
|
; Get here either from unequal compare or sort code = 0.
|
|
; In the latter case, condition codes indicate equality,
|
|
; which is correct.
|
|
|
|
ret
|
|
|
|
FieldCmps label word ; call table of entry comparisons
|
|
dw CmpName
|
|
dw CmpExt
|
|
dw CmpTime
|
|
dw CmpSize
|
|
dw CmpType
|
|
|
|
CmpEntry endp
|
|
|
|
|
|
|
|
|
|
;*** CmpName - compare file name of two entries
|
|
;*** CmpExt - compare extension of two entries
|
|
;
|
|
; ENTRY ES:BX = ptr to one entry
|
|
; ES:BP = ptr to another entry
|
|
;
|
|
; EXIT BX = unchanged
|
|
; BP = unchanged
|
|
; Condition flags set for same, above, or below
|
|
; comparing BX entry to BP entry.
|
|
;
|
|
; USED: AX,CX,DX,SI,DI
|
|
|
|
CmpName proc
|
|
|
|
mov si,bx ; ES:SI = ptr to BX entry
|
|
mov di,bp ; ES:DI = ptr to BP entry
|
|
add si,filename ; ES:SI = ptr to BX name
|
|
add di,filename ; ES:DI = ptr to BP name
|
|
mov cx,size filename; CX = length of name
|
|
jmp short CmpStr
|
|
|
|
CmpExt: mov si,bx ; ES:SI = ptr to BX entry
|
|
mov di,bp ; ES:DI = ptr to BP entry
|
|
add si,fileext ; ES:SI = ptr to BX extension
|
|
add di,fileext ; ES:DI = ptr to BP extension
|
|
mov cx,size fileext ; CX = length of extension field
|
|
|
|
; Bugbug: use symbol for subfunction code.
|
|
|
|
CmpStr: cmp CountryPtrId,6
|
|
jne cnNoCollTable ; no collating table available
|
|
|
|
;* Compare strings using collating table.
|
|
;
|
|
; ES:SI = ptr to 1st string
|
|
; ES:DI = ptr to 2nd string
|
|
; CX = length
|
|
|
|
push bp ; preserve BP
|
|
push bx ; preserve BX
|
|
push ds ; preserve DS
|
|
lds bx,CountryPtr ; DS:BX = ptr to collating table
|
|
assume ds:NOTHING
|
|
mov bp,ds:[bx] ; BP = size of collating table
|
|
inc bx
|
|
inc bx ; DS:BX = ptr to collating values
|
|
; DS:[BX]-2 = size of table
|
|
xor ax,ax ; AX = 0 for starters
|
|
|
|
; Bugbug: Investigate removing collating table length checks.
|
|
|
|
cnNextChar:
|
|
mov al,es:[di] ; AL = AX = char from 2nd string
|
|
inc di ; ES:DI = ptr to next char 2nd string
|
|
cmp ax,bp ; compare to collating table length
|
|
jae @F ; char not in table
|
|
xlat
|
|
@@: ; AL = AX = collating value
|
|
mov dx,ax ; DX = collating value from 2nd string
|
|
lods byte ptr es:[si] ; AL = AX = char from 1st string
|
|
; ES:SI = ptr to next char 1st string
|
|
cmp ax,bp ; compare to collating table length
|
|
jae @F ; char not in table
|
|
xlat
|
|
@@: ; AL = AX = collating value
|
|
cmp ax,dx ; compare collating values
|
|
|
|
ifdef DBCS ; DBCS tail byte must not use
|
|
; collating table
|
|
jnz cnNot_Same
|
|
mov al,es:[di-1] ; get previous 2nd string character
|
|
invoke testkanj
|
|
jz cnDo_Next ; if it was not DBCS lead byte
|
|
mov al,es:[di] ; get tail byte from 2nd string
|
|
cmp es:[si],al ; compare with 1st strings tail byte
|
|
jnz cnNot_Same
|
|
inc si ; pass tail byte
|
|
inc di
|
|
dec cx
|
|
cnDo_Next:
|
|
loop cnNextChar
|
|
cnNot_Same:
|
|
|
|
else ; Not DBCS
|
|
|
|
loope cnNextChar ; until unequal or no more left
|
|
endif
|
|
|
|
pop ds ; restore DS
|
|
assume ds:TRANGROUP
|
|
pop bx ; restore BX
|
|
pop bp ; restore BP
|
|
ret
|
|
|
|
;* If no collating table is available, simply compare raw ASCII values.
|
|
; Don't we wish we could just do this all the time? Sigh.
|
|
|
|
cnNoCollTable:
|
|
rep cmps byte ptr es:[si],[di]
|
|
ret
|
|
|
|
CmpName endp
|
|
|
|
|
|
|
|
|
|
;*** CmpTime - compare entries by date/time
|
|
;
|
|
; ENTRY ES:BX = ptr to one entry
|
|
; ES:BP = ptr to another entry
|
|
;
|
|
; EXIT BX = unchanged
|
|
; BP = unchanged
|
|
; Condition flags set for same, above, or below
|
|
; comparing BX entry to BP entry.
|
|
;
|
|
; USED: CX,SI,DI
|
|
;
|
|
; NOTE Filetime and filedate fields in our private entry
|
|
; structure must be adjacent and in that order.
|
|
|
|
CmpTime proc
|
|
|
|
mov si,bx
|
|
mov di,bp
|
|
add si,filedate + size filedate - 1
|
|
add di,filedate + size filedate - 1
|
|
mov cx,size filetime + size filedate
|
|
std
|
|
repe cmps byte ptr es:[si],[di]
|
|
cld
|
|
ret
|
|
|
|
CmpTime endp
|
|
|
|
|
|
|
|
|
|
;*** CmpSize - compare entries by size
|
|
;
|
|
; ENTRY ES:BX = ptr to one entry
|
|
; ES:BP = ptr to another entry
|
|
;
|
|
; EXIT BX = unchanged
|
|
; BP = unchanged
|
|
; Condition flags set for same, above, or below
|
|
; comparing BX entry to BP entry.
|
|
;
|
|
; USED: CX,SI,DI
|
|
|
|
CmpSize proc
|
|
|
|
mov si,bx
|
|
mov di,bp
|
|
add si,filesize + size filesize - 1
|
|
add di,filesize + size filesize - 1
|
|
mov cx,size filesize
|
|
std
|
|
repe cmps byte ptr es:[si],[di]
|
|
cld
|
|
ret
|
|
|
|
CmpSize endp
|
|
|
|
|
|
|
|
|
|
;*** CmpType - compare entries by file type (subdirectory or not)
|
|
;
|
|
; ENTRY ES:BX = ptr to one entry
|
|
; ES:BP = ptr to another entry
|
|
;
|
|
; EXIT BX = unchanged
|
|
; BP = unchanged
|
|
; Condition flags set for same, above, or below
|
|
; comparing BX entry to BP entry.
|
|
;
|
|
; USED: AX
|
|
|
|
CmpType proc
|
|
|
|
mov al,es:[bx].fileattr
|
|
mov ah,es:[bp].fileattr
|
|
and ax,(ATTR_DIRECTORY shl 8) + ATTR_DIRECTORY
|
|
cmp ah,al
|
|
ret
|
|
|
|
CmpType endp
|
|
|
|
|
|
|
|
|
|
|
|
;*** DefaultAttr - set default attribute conditions
|
|
;
|
|
; ENTRY nothing
|
|
;
|
|
; EXIT CY clear
|
|
;
|
|
; USED
|
|
;
|
|
; EFFECTS
|
|
;
|
|
; AttrSpecified, AttrSelect are updated with new attribute conditions.
|
|
|
|
DefaultAttr proc
|
|
|
|
mov AttrSpecified,ATTR_HIDDEN+ATTR_SYSTEM ; specify H and S
|
|
mov AttrSelect,0 ; H and S must be off
|
|
clc ; return success
|
|
ret
|
|
|
|
DefaultAttr endp
|
|
|
|
|
|
|
|
|
|
;*** DisplayTotals - display grand total stats
|
|
;
|
|
; If we searched subdirectories, display the total # files found
|
|
; and total size of files found.
|
|
; Display disk space remaining.
|
|
;
|
|
; ENTRY FileCntTotal, FileSizTotal contain correct values
|
|
; Bits contains setting of /s
|
|
; FCB contains drive #
|
|
;
|
|
; EXIT nothing
|
|
;
|
|
; USES AX,DX
|
|
; FileSiz
|
|
|
|
DisplayTotals proc
|
|
|
|
test Bits,mask subd
|
|
jz dtFree ; no subdirectories- do bytes free
|
|
|
|
invoke Crlf2 ; start on new line
|
|
call UseLine
|
|
|
|
mov dx,offset TRANGROUP:Total_Ptr
|
|
invoke Std_Printf ; "Total:",cr,lf
|
|
call UseLine
|
|
|
|
mov ax,word ptr FileCntTotal ; AX = # files found mod 64K
|
|
mov si,offset TRANGROUP:FileSizTotal
|
|
mov di,offset TRANGROUP:FileSiz
|
|
movsw
|
|
movsw ; move total size to size variable
|
|
call DisplayCntSiz ; display file count & size
|
|
dtFree:
|
|
mov ah,GET_DRIVE_FREESPACE ; AH = DOS Get Free Space function
|
|
mov dl,byte ptr ds:FCB ; DL = drive#
|
|
int 21h ; call DOS
|
|
cmp ax,-1 ; check 'invalid drive' return code
|
|
jz dtRet ; can't get drive space - return
|
|
mul cx
|
|
mul bx
|
|
mov Bytes_Free,ax
|
|
mov Bytes_Free+2,dx
|
|
mov dx,offset TRANGROUP:BytMes_Ptr
|
|
invoke Std_Printf ; "nnn bytes free",cr,lf
|
|
call UseLine
|
|
|
|
dtRet: ret
|
|
|
|
DisplayTotals endp
|
|
|
|
|
|
|
|
|
|
;*** FileIsDevice - see if file looks like a device
|
|
;
|
|
; ENTRY PathPos = ptr to pathname
|
|
; PathCnt = length of pathname w/o terminating char
|
|
; DirBuf is DOS DTA
|
|
;
|
|
; EXIT ZR = set if file looks like a device
|
|
;
|
|
; USED AX,BX,CX,DX,DI
|
|
;
|
|
; EFFECTS
|
|
;
|
|
; DTA buffer holds results of Find First function
|
|
;
|
|
; NOTES
|
|
;
|
|
; We try to flag devices in two ways. First, we try
|
|
; the DOS Find First function. It returns attribute bit 6
|
|
; set on a successful find if it identifies a device name.
|
|
; Unfortunately, it returns 'path not found' for a device
|
|
; name terminated with colon, such as "CON:". So, we look
|
|
; for any colon in the pathname after the 2nd character,
|
|
; and flag the pathname as a device if we find one.
|
|
|
|
FileIsDevice proc
|
|
|
|
mov dx,PathPos ; DX = ptr to pathname
|
|
|
|
mov di,dx
|
|
add di,PathCnt ; DI = ptr to byte after pathname
|
|
xor bl,bl ; BL = NUL to terminate pathname with
|
|
xchg bl,byte ptr [di] ; BL = saved pathname terminating char
|
|
|
|
xor cx,cx ; CX = attribute mask (normal search)
|
|
mov ah,FIND_FIRST ; AH = DOS Find First function code
|
|
int 21h ; call DOS
|
|
xchg bl,byte ptr [di] ; restore pathname terminating char
|
|
jc piCol ; didn't find a dir entry, check for colon
|
|
|
|
; Found a dir entry, see if Find First thinks it's a device.
|
|
|
|
test byte ptr DirBuf.Find_Buf_Attr,ATTR_DEVICE
|
|
jz piCol ; device attribute not set, look for colon
|
|
xor cx,cx ; it's a device, return ZR flag
|
|
jmp short piRet
|
|
|
|
; Device attribute not returned by Find First function. But
|
|
; let's check for a colon anywhere in the pathname after the
|
|
; second byte.
|
|
;
|
|
; DI = ptr to byte after pathname
|
|
|
|
piCol: dec di ; DI = ptr to last char in pathname
|
|
mov al,COLON_CHAR ; AL = colon char to search for
|
|
mov cx,PathCnt ; CX = # chars to scan
|
|
dec cx
|
|
dec cx ; ignore 1st two chars of pathname
|
|
or cx,cx
|
|
js piRet ; if < 2 chars in pathname, just return
|
|
or di,di ; clear ZR in case CX = 0
|
|
std ; scan downward
|
|
repne scasb
|
|
cld ; restore default upward direction
|
|
|
|
; After scanning, the ZR flag is set to indicate presence of a colon.
|
|
|
|
piRet: ret
|
|
|
|
FileIsDevice endp
|
|
|
|
|
|
|
|
|
|
;*** FindFirst - find first directory entry to display
|
|
;*** FindNext - find next directory entry to display
|
|
;
|
|
; ENTRY Bits<inmem> = set if entries are loaded in TPA
|
|
; AttrSpecified, AttrSelect are set
|
|
;
|
|
; EXIT CY = clear if successful
|
|
; BX = offset in TPA buffer of directory entry found
|
|
;
|
|
; If unsuccessful,
|
|
; CY = set
|
|
; AX = DOS error code
|
|
; DOS Get Extended Error call will get error code
|
|
;
|
|
; NOTE: if entries were loaded into TPA, AX contains
|
|
; ERROR_NO_MORE_FILES when no more entries are available,
|
|
; but DOS Get Extended Error call WON'T return the correct
|
|
; error. That's ok, because we'll see the value in AX
|
|
; and recognize it as a non-error condition.
|
|
;
|
|
; USED AX,CX,DX,SI,DI
|
|
;
|
|
; EFFECTS
|
|
;
|
|
; Entries in memory may be marked as output.
|
|
; If not sorted, entry is loaded at TPA.
|
|
;
|
|
; NOTES
|
|
;
|
|
; If we don't find a qualifying file, we return after the final
|
|
; DOS Find File call. A DOS Get Extended Error call will then
|
|
; indicate an appropriate condition.
|
|
|
|
FindFirst proc
|
|
|
|
mov ax,offset TRANGROUP:GetFirst
|
|
jmp short ffFindEntry
|
|
|
|
FindNext:
|
|
mov ax,offset TRANGROUP:GetNext
|
|
|
|
; AX = address of correct disk get routine to use.
|
|
|
|
ffFindEntry:
|
|
push es ; save TRANGROUP seg addr
|
|
test Bits,mask inmem
|
|
jz ffDisk ; entries not in memory, search disk
|
|
|
|
; Entries are loaded in memory to sort out. Find the first one.
|
|
; There will always be one, or LoadEntries would've failed.
|
|
|
|
call FindInMem ; find first entry in TPA
|
|
jmp short ffRet ; return what TPA search returns
|
|
|
|
; Get entry from disk.
|
|
|
|
ffDisk:
|
|
call ax ; get entry from disk
|
|
jc ffGetErr ; get & return error
|
|
mov es,Tpa ; ES = seg addr of TPA
|
|
xor di,di ; ES:DI = ptr to TPA
|
|
mov bx,di ; BX = offset of entry in TPA
|
|
call LoadEntry ; load entry to TPA
|
|
clc ; return success
|
|
jmp short ffRet
|
|
|
|
ffGetErr:
|
|
invoke Get_Ext_Error_Number ; AX = DOS error code
|
|
stc
|
|
|
|
ffRet: pop es ; ES = TRANGROUP seg addr again
|
|
ret
|
|
|
|
FindFirst endp
|
|
|
|
|
|
|
|
|
|
;*** FindInMem - find next directory entry in TPA buffer
|
|
;
|
|
; ENTRY TPA is loaded (see LoadEntries)
|
|
;
|
|
; EXIT BX = offset in TPA of entry found
|
|
;
|
|
; If no more files,
|
|
; CY = set
|
|
; AX = DOS 'no more files' error code
|
|
;
|
|
; USED AX,BX,CX,DX,SI,DI,BP,ES
|
|
;
|
|
; EFFECTS
|
|
;
|
|
; Entry found is flagged as 'used' (see EntryStruc).
|
|
|
|
FindInMem proc
|
|
|
|
mov es,Tpa ; ES = TPA seg addr
|
|
xor bx,bx ; ES:BX = ptr to 1st entry in TPA
|
|
cld ; make sure default string direction is up
|
|
|
|
call FindOneInMem ; locate an entry
|
|
jc fiNoMore ; none left, set up 'no more files' error
|
|
|
|
; BX = ptr to entry in TPA
|
|
|
|
fiBest:
|
|
mov bp,bx ; BP = ptr to best entry so far
|
|
fiNext:
|
|
call FindNextInMem ; locate next entry
|
|
jc fiFound ; no more, best entry so far wins
|
|
|
|
; BX = ptr to next entry
|
|
|
|
call CmpEntry ; compare it to best found so far (BP)
|
|
jnb fiNext ; it's not better, go look at next one
|
|
jmp fiBest ; it's better, go mark it as best so far
|
|
|
|
fiNoMore:
|
|
|
|
; No more entries available in TPA. Set up 'no more files' error.
|
|
|
|
mov ax,ERROR_NO_MORE_FILES ; AX = 'no more files' error code
|
|
stc ; return error
|
|
jmp short fiRet
|
|
|
|
fiFound:
|
|
mov bx,bp ; BX = ptr to best entry found
|
|
mov byte ptr es:[bx],1 ; mark entry 'used'
|
|
clc ; return success
|
|
fiRet: ret
|
|
|
|
FindInMem endp
|
|
|
|
|
|
|
|
|
|
;*** FindNextChild - find next subdirectory in current directory
|
|
;
|
|
; ENTRY BX = ptr to last child found, ASCIIZ filename
|
|
; DirBuf is established DTA
|
|
;
|
|
; EXIT BX = ptr (same addr) to next child found, ASCIIZ filename
|
|
;
|
|
; If failure,
|
|
; CY = set
|
|
; DOS Get Extended Error call will get error
|
|
;
|
|
; USED AX,CX,DX,SI,DI,BP
|
|
;
|
|
; EFFECTS
|
|
;
|
|
; DirBuf is used for find first/next calls.
|
|
;
|
|
; NOTES
|
|
;
|
|
; We keep on checking files until DOS returns an error. If
|
|
; the error is 'no more files' and the temp filename is not
|
|
; the initial high tag, copy the temp to the child's name spot
|
|
; and return success. Otherwise, send the error back to caller.
|
|
;
|
|
; This routine depends on DS,ES,CS, & SS all being equal.
|
|
|
|
FindNextChild proc
|
|
|
|
sub sp,12 ; make temp filename buf on stack
|
|
shove 00FFh ; temp filename = high tag
|
|
mov bp,sp ; BP = ptr to temp filename buf
|
|
shove "*"
|
|
shove ".*"
|
|
call GetDriveLtr ; AX = "d:"
|
|
push ax
|
|
mov dx,sp ; DX = ptr to "d:*.*",0 on stack
|
|
|
|
; See that the stack is restored properly at the end of this proc.
|
|
|
|
mov cx,ATTR_DIRECTORY ; CX = attributes for file search
|
|
mov ah,FIND_FIRST
|
|
int 21h ; DOS- Find First matching file
|
|
jc fcRet ; return error
|
|
|
|
call CheckChild ; check child against last, temp
|
|
|
|
fcNext: mov cx,ATTR_DIRECTORY ; CX = attributes for file search
|
|
mov ah,FIND_NEXT
|
|
int 21h ; DOS- Find Next matching file
|
|
jc fcErr ; examine error
|
|
|
|
call CheckChild ; check child against last, temp
|
|
jmp fcNext ; go find another child
|
|
|
|
fcErr:
|
|
invoke Get_Ext_Error_Number ; AX = extended error code
|
|
cmp ax,ERROR_NO_MORE_FILES ; no more files?
|
|
jne short fcNope ; some other error- return it
|
|
|
|
; We ran out of files. See if we qualified at least one.
|
|
|
|
cmp byte ptr [bp],0FFh
|
|
je fcNope ; temp filename is unused- no child
|
|
|
|
; Move temp filename to child name position.
|
|
|
|
mov si,bp ; SI = ptr to temp filename
|
|
mov di,bx ; DI = ptr to child name pos'n
|
|
fcMove: lodsb ; AL = next byte of filename
|
|
stosb ; store byte
|
|
or al,al
|
|
jz fcRet ; byte was zero, return success (CY clear)
|
|
jmp fcMove ; go move another byte
|
|
|
|
fcNope: stc ; return error
|
|
fcRet: lahf
|
|
add sp,20 ; restore stack
|
|
sahf
|
|
ret
|
|
|
|
FindNextChild endp
|
|
|
|
|
|
|
|
|
|
|
|
;*** FindOneInMem - find the first available entry in TPA
|
|
;*** FindNextInMem - find the next available entry in TPA
|
|
;
|
|
; ENTRY ES = TPA seg addr
|
|
; BX = ptr to entry in TPA
|
|
;
|
|
; EXIT BX = ptr to entry found
|
|
; CY = set if no more entries available in TPA
|
|
;
|
|
; USED AL
|
|
|
|
FindOneInMem proc
|
|
|
|
mov al,es:[bx] ; examine 'used' byte of starting entry
|
|
cmp al,1
|
|
je FindNextInMem ; entry has already been used
|
|
cmp al,0FFh
|
|
je foNoMore ; 0FFh, we're at the end of the list
|
|
|
|
; BX = ptr to entry that hasn't been output yet.
|
|
|
|
clc ; return success
|
|
ret
|
|
|
|
FindNextInMem:
|
|
add bx,size EntryStruc ; BX = ptr to next entry
|
|
jmp FindOneInMem ; go look at it
|
|
|
|
foNoMore:
|
|
stc ; ran out of entries, return failure
|
|
ret
|
|
|
|
FindOneInMem endp
|
|
|
|
|
|
|
|
|
|
;*** GetEnvValue - get value of our environment variable
|
|
;
|
|
; ENTRY DS, ES = TRANGROUP seg addr
|
|
;
|
|
; EXIT CY = set if environment variable not in environment
|
|
;
|
|
; Otherwise:
|
|
; SI = ptr to environment variable asciiz value in TRANGROUP
|
|
;
|
|
; USED AX,BX,CX,DX,DI
|
|
; (We assume the (almost) worst, since we don't know about
|
|
; Find_Name_In_Environment.)
|
|
;
|
|
; EFFECTS
|
|
;
|
|
; ScanBuf is loaded with value text
|
|
|
|
GetEnvValue proc
|
|
|
|
push es ; save ES
|
|
mov si,offset TRANGROUP:DirEnvVar ; DS:SI = ptr to variable name
|
|
invoke Find_Name_In_Environment
|
|
jc geRet ; name not found in environment
|
|
|
|
; ES:DI = ptr to value of environment variable
|
|
; We're assuming DS, CS, and SS are unchanged.
|
|
|
|
push ds
|
|
push es
|
|
pop ds
|
|
pop es
|
|
|
|
assume ds:nothing
|
|
|
|
; DS = seg addr of environment variable value (in environment segment)
|
|
; ES = TRANGROUP seg addr
|
|
|
|
mov si,di ; DS:SI = ptr to value string
|
|
mov di,offset TRANGROUP:ScanBuf ; ES:DI = ptr to dest buffer
|
|
@@: lodsb
|
|
or al,al
|
|
stosb
|
|
loopnz @B ; move the string, including trailing null
|
|
|
|
push es
|
|
pop ds ; DS = TRANGROUP seg addr again
|
|
assume ds:TRANGROUP
|
|
|
|
mov si,offset TRANGROUP:ScanBuf ; SI = ptr to var value
|
|
geRet: pop es ; restore ES
|
|
ret
|
|
|
|
GetEnvValue endp
|
|
|
|
|
|
|
|
|
|
;*** GetFirst - get first directory entry from disk
|
|
;
|
|
; ENTRY DOS DTA established at DirBuf
|
|
; FCB contains drive # and filename
|
|
; Current directory (on selected drive) is the one to search
|
|
; AttrSpecified & AttrSelect masks set
|
|
;
|
|
; EXIT CY = clear if success
|
|
; DirBuf contains extended FCB for file found
|
|
;
|
|
; If unsuccessful,
|
|
; CY = set
|
|
; Ready for DOS Get Extended Error call
|
|
;
|
|
; USED AX,DX
|
|
;
|
|
; EFFECTS
|
|
;
|
|
; FCB-7 = 0FFh to mark extended FCB
|
|
; FCB-1 = attribute mask to find all files
|
|
; These fields should remain unmodified for GetNext calls.
|
|
;
|
|
;
|
|
;*** GetNext - get next directory entry from disk
|
|
;
|
|
; ENTRY As for GetFirst, plus
|
|
; FCB-7 set up as extended FCB w/ find-all attribute byte
|
|
;
|
|
; EXIT As for GetFirst
|
|
;
|
|
; USED AX,DX
|
|
|
|
GetFirst proc
|
|
|
|
mov byte ptr ds:FCB-7,0FFh ; signal extended FCB
|
|
mov byte ptr ds:FCB-1,ATTR_ALL
|
|
; find any file
|
|
mov dx,FCB-7 ; DX = ptr to extended FCB
|
|
mov ah,DIR_SEARCH_FIRST ; AH = DOS Find First function code
|
|
int 21h ; call DOS
|
|
shl al,1 ; CY = set if error
|
|
jc gfRet ; return error
|
|
jmp short gfFound ; go look at attr's
|
|
|
|
GetNext:
|
|
mov dx,FCB-7 ; DX = ptr to extended FCB
|
|
mov ah,DIR_SEARCH_NEXT ; AH = DOS Find Next function code
|
|
int 21h ; call DOS
|
|
shl al,1 ; CY = set if error
|
|
jc gfRet ; return error
|
|
|
|
;* Found an entry. Check attributes.
|
|
|
|
gfFound:
|
|
mov al,[DirBuf+8].dir_attr ; AL = file attributes
|
|
mov ah,AttrSpecified ; AH = mask of pertinent attr's
|
|
and al,ah ; AL = pertinent attr's of file
|
|
and ah,AttrSelect ; AH = attr settings to match
|
|
cmp al,ah
|
|
jne GetNext ; attr's don't match, look for another
|
|
|
|
gfRet: ret
|
|
|
|
GetFirst endp
|
|
|
|
|
|
|
|
|
|
;*** ListDir - search for and list files in the current directory
|
|
;
|
|
; List header, files, and trailer for current directory on selected
|
|
; drive. Header & trailer are listed if at least one file is found.
|
|
; If no qualifying files are found, no display output occurs.
|
|
;
|
|
; ENTRY Current directory (on selected drive) is the one to be listed
|
|
; FCB contains selected drive # and filename spec
|
|
; Option bits, attribute masks, and sort codes set up
|
|
;
|
|
; EXIT CY = clear if no error
|
|
; FileCnt = # files found & displayed
|
|
;
|
|
; If error,
|
|
; CY = set
|
|
; Ready for DOS Get Extended Error call
|
|
;
|
|
; USED AX,BX,CX,DX,SI,DI,BP
|
|
; FileSiz
|
|
;
|
|
; EFFECTS
|
|
;
|
|
; FileCntTotal, FileSizTotal are updated.
|
|
; Files found are listed. A directory header and trailer are
|
|
; displayed only if files are found.
|
|
|
|
ListDir proc
|
|
|
|
xor ax,ax
|
|
mov FileCnt,ax ; zero file count
|
|
mov word ptr FileSiz,ax ; zero file size accumulator
|
|
mov word ptr FileSiz+2,ax
|
|
|
|
cmp DestBuf,0 ; check for sort code
|
|
je @F ; no sort
|
|
call LoadEntries ; load entries for sorted listing
|
|
jnc @F ; no error - continue
|
|
invoke Get_Ext_Error_Number ; AX = DOS error code
|
|
stc
|
|
jmp short ldErr ; return error
|
|
@@:
|
|
call FindFirst ; find first file
|
|
jc ldErr ; not found, return error
|
|
|
|
; BX = offset in TPA buffer of entry found
|
|
|
|
call DisplayHeader ; if at least one file, display header
|
|
call DisplayFile ; display the file entry
|
|
ldNext:
|
|
call FindNext ; find another file
|
|
jc ldErr ; not found
|
|
call DisplayFile ; display entry
|
|
jmp ldNext ; go find another one
|
|
|
|
ldErr:
|
|
cmp ax,ERROR_FILE_NOT_FOUND
|
|
je ldDone ; file not found, we're done
|
|
cmp ax,ERROR_NO_MORE_FILES
|
|
je ldDone ; no more files, we're done
|
|
stc
|
|
jmp short ldRet
|
|
|
|
ldDone:
|
|
cmp FileCnt,0
|
|
je @F ; no files found, just return
|
|
call DisplayTrailer ; display trailing info
|
|
@@: clc ; return success
|
|
|
|
ldRet: ret
|
|
|
|
ListDir endp
|
|
|
|
|
|
|
|
|
|
;*** LoadEntries - attempt to load entries from current directory
|
|
;
|
|
; Load all qualifying directory entries from the current directory
|
|
; into the TPA. If an error is returned by FindFirst/FindNext calls
|
|
; other than 'no more files', return to caller with carry flag set.
|
|
; If we run out of buffer space, display a message that we haven't
|
|
; enough memory to sort this directory, but return without error.
|
|
; Other routines know whether or not entries have been loaded by
|
|
; the 'inmem' flag bit, which we set here.
|
|
;
|
|
; The TPA is usually 64K - 512 bytes long. At 20 bytes per entry,
|
|
; this allows sorting over 3000 entries in a directory.
|
|
;
|
|
; ENTRY Tpa = buffer seg addr
|
|
; BytCnt = buffer length, in bytes
|
|
; Current directory (on selected drive) is the one to load
|
|
; FCB contains drive # and filespec
|
|
; Bits, AttrSpecified, AttrSelect, & DestBuf (sort codes) are set
|
|
;
|
|
; EXIT CY = set if error
|
|
; If error, DOS Get Extended Error will get error info
|
|
;
|
|
; USED AX,CX,DX,SI,DI
|
|
;
|
|
; EFFECTS
|
|
;
|
|
; Inmem bit of Bits = set if load succeeded.
|
|
; Tpa buffer contains directory entries.
|
|
; Byte after last entry = 0FFh.
|
|
|
|
LoadEntries proc
|
|
|
|
push es ; save TRANGROUP seg addr
|
|
mov es,Tpa ; ES = TPA seg addr
|
|
xor di,di ; ES:DI = destination ptr
|
|
and Bits,not mask inmem ; signal entries not loaded
|
|
|
|
call GetFirst ; look for first file
|
|
jc leRet ; return any error
|
|
call LoadEntry ; load entry into TPA
|
|
|
|
leNext: call GetNext ; get another file
|
|
jc leLoaded ; assume any error is no more files
|
|
mov ax,BytCnt ; AX = size of TPA
|
|
sub ax,di ; AX = bytes left in TPA
|
|
cmp ax,size EntryStruc+2 ; insist on entry size + 2 bytes
|
|
jb leOk ; not enough memory left, give up
|
|
call LoadEntry ; load entry into TPA
|
|
jmp leNext ; go get another file
|
|
|
|
leLoaded:
|
|
mov byte ptr es:[di],0FFh ; mark end of entry list
|
|
or Bits,mask inmem ; signal entries loaded in memory
|
|
leOk: clc ; return no error
|
|
|
|
leRet: pop es ; ES = TRANGROUP seg addr again
|
|
ret
|
|
|
|
LoadEntries endp
|
|
|
|
|
|
|
|
|
|
;*** LoadEntry - load directory entry from DirBuf ext'd FCB
|
|
;
|
|
; ENTRY ES:DI = ptr to load point in TPA
|
|
; DirBuf contains extended FCB of entry to load
|
|
;
|
|
; EXIT ES:DI = ptr to next byte available in TPA
|
|
;
|
|
; USED AX,CX,SI
|
|
;
|
|
; NOTES
|
|
;
|
|
; I could've used symbolic offsets and sizes of fields from
|
|
; the dir_entry struc to do this, but this is time-critical,
|
|
; so I hard-wired the structure of the DOS 4.x returned FCB,
|
|
; as well as our private directory entry structure.
|
|
;
|
|
; We force a zero size for subdirectory files. A zero size is
|
|
; ordinarily returned for subdirectories, but with Novell
|
|
; Netware 286 or 386 loaded, we can't depend on it. Bug #1594.
|
|
|
|
LoadEntry proc
|
|
|
|
mov si,offset TRANGROUP:Dirbuf+8 ; DS:SI = ptr to filename
|
|
xor al,al ; AL = 0
|
|
stosb ; 'used' byte = false
|
|
mov cx,11
|
|
rep movsb ; transfer filename & extension
|
|
lodsb ; AL = attrib byte
|
|
stosb ; store attrib byte
|
|
add si,dir_time-dir_attr-1 ; skip to time field
|
|
movsw ; transfer time
|
|
movsw ; transfer date
|
|
inc si ; skip alloc unit
|
|
inc si
|
|
|
|
and al,ATTR_DIRECTORY
|
|
jnz leSetDirSize ; force zero size for subdir
|
|
|
|
movsw
|
|
movsw ; transfer size
|
|
ret
|
|
|
|
leSetDirSize:
|
|
xor ax,ax
|
|
stosw
|
|
stosw ; store zero size
|
|
ret
|
|
|
|
LoadEntry endp
|
|
|
|
|
|
|
|
|
|
;*** NoOrder - turn sorting off
|
|
;
|
|
; ENTRY nothing
|
|
;
|
|
; EXIT CY clear
|
|
;
|
|
; USED AX
|
|
;
|
|
; EFFECTS
|
|
;
|
|
; DestBuf is updated with sort code bytes. See DestBuf description.
|
|
|
|
NoOrder proc
|
|
|
|
mov DestBuf,0 ; no sort
|
|
clc ; no error
|
|
ret
|
|
|
|
NoOrder endp
|
|
|
|
|
|
|
|
|
|
;*** OnOffSw - record occurence of on/off option switch
|
|
;
|
|
; ENTRY DI = index into word list of switches
|
|
;
|
|
; EXIT CY clear
|
|
;
|
|
; USED AX,CX
|
|
;
|
|
; EFFECTS
|
|
;
|
|
; Bits modified to indicate option state.
|
|
|
|
OnOffSw proc
|
|
|
|
mov cx,di ; CX = index into word list of options
|
|
shr cx,1
|
|
shr cx,1 ; CX = bit position of option
|
|
mov ax,1
|
|
shl ax,cl ; AX = bit mask of option
|
|
test di,2 ; check if it is a negated option
|
|
jz @F ; it's negated
|
|
or Bits,ax ; turn option on
|
|
jmp short ooRet
|
|
|
|
@@: not ax ; AX = complemented bit mask of option
|
|
and Bits,ax ; turn option off
|
|
|
|
ooRet: clc ; always return success
|
|
ret
|
|
|
|
OnOffSw endp
|
|
|
|
|
|
|
|
|
|
;*** ParseAttr - parse and record /A option
|
|
;
|
|
; ENTRY BX = ptr to system parser result buffer for /A occurence
|
|
;
|
|
; EXIT CY = set if error occurs parsing attribute conditions
|
|
;
|
|
; For parse error, we set up for Std_EPrintf call:
|
|
; AX = parse error code, like system parser
|
|
; DX = ptr to message block
|
|
;
|
|
; USED AX,CX,DX,DI
|
|
;
|
|
; EFFECTS
|
|
;
|
|
; AttrSpecified, AttrSelect are updated with new attribute conditions.
|
|
; If parse error occurs, attribute conditions parsed so far hold.
|
|
;
|
|
; For parse error, we set up for Std_EPrintf call:
|
|
; Msg_Disp_Class = parse error message class
|
|
; Message block (see DX) is set up for parse error message
|
|
|
|
ParseAttr proc
|
|
|
|
push si ; save SI
|
|
mov AttrSpecified,0 ; cancel all attribute conditions
|
|
|
|
; Each /A invocation starts by assuming all files are to be listed.
|
|
|
|
mov si,word ptr [bx].ValuePtr ; SI = ptr to string after /A
|
|
|
|
paLoop: mov dx,1 ; DX = 1 (for un-negated attribute)
|
|
|
|
lodsb ; AL = next char in string
|
|
or al,al
|
|
jz paOk ; it's terminating null, we're done
|
|
cmp al,'-'
|
|
jne @F ; not '-', go look for letter
|
|
dec dx ; DX = 0 (for negated attribute)
|
|
lodsb ; AL = next char
|
|
@@:
|
|
mov di,offset TRANGROUP:AttrLtrs ; DI = ptr to attrib letter list
|
|
mov cx,NUM_ATTR_LTRS ; CX = length of attrib letter list
|
|
repne scasb ; look for our letter in the list
|
|
jne paErr ; not found, return error
|
|
|
|
not cx
|
|
add cx,NUM_ATTR_LTRS ; CX = attrib bit #, 0-5
|
|
|
|
; Note that we rely on AttrLtrs to be in the attribute bit order,
|
|
; starting from bit 0.
|
|
|
|
; Record this attribute bit in AttrSpecified.
|
|
|
|
mov al,1
|
|
shl al,cl ; AL = mask for our bit
|
|
or AttrSpecified,al ; set it in the 'specified' mask
|
|
|
|
; Record the selected state for this attribute in AttrSelect.
|
|
; DX = 0 or 1, the selected state for this attribute.
|
|
|
|
not al ; AL = mask for all other bits
|
|
and AttrSelect,al ; clear our bit
|
|
shl dl,cl ; DL = our bit state in position
|
|
or AttrSelect,dl ; set selected attr state
|
|
jmp paLoop ; go look at next char
|
|
|
|
; The attribute letter string is invalid.
|
|
|
|
paErr:
|
|
call SetupParamError ; set message up for Std_EPrintf
|
|
stc ; return error
|
|
jmp short paRet
|
|
|
|
paOk: clc ; return success
|
|
|
|
paRet: pop si ; restore SI
|
|
ret
|
|
|
|
ParseAttr endp
|
|
|
|
|
|
|
|
|
|
;*** ParseLine - parse a line of text
|
|
;
|
|
; Parse text until an EOL (CR or NUL) is found, or until a parse
|
|
; error occurs.
|
|
;
|
|
; ENTRY DS:SI = ptr to text
|
|
; CS, DS, ES = TRANGROUP seg addr
|
|
;
|
|
; EXIT AX = last return code from system parser
|
|
; CX = # positional parameters (pathnames) found - 0 or 1
|
|
;
|
|
; If parse error occurred, we're set up for Std_EPrintf call:
|
|
; DX = ptr to message block
|
|
;
|
|
; USED BX,CX,DX,SI,DI
|
|
;
|
|
; EFFECTS
|
|
;
|
|
; Bits may contain new option settings.
|
|
; DestBuf may contain new series of sort codes.
|
|
; AttrSpecified, AttrSelect may contain new attribute conditions.
|
|
; SrcBuf may contain a new default pathname/filespec.
|
|
; PathPos, PathCnt updated for new pathname.
|
|
;
|
|
; If parse error occurred, we're set up for Std_EPrintf call:
|
|
; Msg_Disp_Class = parse error class
|
|
; Byte after last parameter in text is zeroed to make ASCIIZ string
|
|
; Message block (see DX) is set up for parse error message
|
|
|
|
ParseLine proc
|
|
|
|
mov di,offset TRANGROUP:Parse_Dir ; ES:DI = ptr to parse block
|
|
xor cx,cx ; CX = # positionals found
|
|
plPars:
|
|
invoke Parse_With_Msg ; call parser
|
|
cmp ax,END_OF_LINE
|
|
je plRet ; EOL encountered, return
|
|
cmp ax,RESULT_NO_ERROR
|
|
jne plRet ; parse error occurred, return
|
|
|
|
; Parse call succeeded. We have a filespec or a switch.
|
|
; DX = ptr to result buffer
|
|
|
|
mov bx,dx ; BX = ptr to parse result buffer
|
|
cmp byte ptr [bx],RESULT_FILESPEC
|
|
je plFil ; we have a filespec
|
|
|
|
call ParseSwitch ; else we have a switch
|
|
jc plRet ; error parsing switch, return
|
|
jmp plPars ; parse more
|
|
|
|
plFil: call CopyPathname ; copy pathname into our buffer
|
|
jmp plPars ; parse more
|
|
|
|
plRet: ret
|
|
|
|
ParseLine endp
|
|
|
|
|
|
|
|
|
|
;*** ParseOrder - parse and record /O option
|
|
;
|
|
; ENTRY BX = ptr to system parser result buffer for /O occurence
|
|
;
|
|
; EXIT CY = set if error occurs parsing order
|
|
;
|
|
; For parse error, we set up for Std_EPrintf call:
|
|
; AX = parse error code, like system parser
|
|
; DX = ptr to message block
|
|
;
|
|
; USED AX,CX,DX,DI
|
|
;
|
|
; EFFECTS
|
|
;
|
|
; DestBuf is updated with sort code bytes. See DestBuf description.
|
|
;
|
|
; For parse error, we set up for Std_EPrintf call:
|
|
; Msg_Disp_Class = parse error message class
|
|
; Message block (see DX) is set up for parse error message
|
|
|
|
ParseOrder proc
|
|
|
|
push si ; save SI
|
|
push bx ; save ptr to result buffer
|
|
|
|
mov si,word ptr [bx].ValuePtr ; SI = ptr to order letters
|
|
mov bx,offset TRANGROUP:DestBuf ; BX = ptr to sort code buffer
|
|
mov al,[si] ; AL = 1st char of order string
|
|
or al,al
|
|
jnz poLtr ; not NUL, go parse letters
|
|
|
|
; We have /O alone. Set standard sort order.
|
|
; Note hardwired dependency on character order in OrderLtrs.
|
|
|
|
mov byte ptr [bx],5 ; sort 1st by group (subdirs 1st)
|
|
inc bx
|
|
mov byte ptr [bx],1 ; then by name
|
|
inc bx
|
|
mov byte ptr [bx],2 ; then by extension
|
|
inc bx
|
|
jmp short poOk ; return success
|
|
|
|
; We have /O<something>. Parse sort order letters.
|
|
|
|
poLtr: xor dl,dl ; DL = 0 (upward sort)
|
|
lodsb ; AL = next sort order letter
|
|
or al,al
|
|
jz poOk ; NUL found, return success
|
|
|
|
cmp al,'-'
|
|
jne @F ; not '-', go look for letter
|
|
mov dl,80h ; DL = downward sort mask
|
|
lodsb ; AL = next char
|
|
@@:
|
|
mov di,offset TRANGROUP:OrderLtrs ; DI = ptr to list of letters
|
|
mov cx,NUM_ORDER_LTRS ; CX = length of list
|
|
repne scasb ; look for our letter in the list
|
|
jne poErr ; not found, return error
|
|
|
|
neg cx
|
|
add cx,NUM_ORDER_LTRS ; CL = sort order code, 1-5
|
|
or cl,dl ; CL = sort code with up/dn bit
|
|
mov [bx],cl ; store sort order code in buffer
|
|
inc bx ; BX = ptr to next spot in buffer
|
|
cmp bx,offset TRANGROUP:EndDestBuf
|
|
jae poErr ; too many letters
|
|
|
|
jmp poLtr ; go look at next char
|
|
|
|
; The sort order string is invalid.
|
|
|
|
poErr: pop bx ; BX = ptr to result buffer
|
|
call SetupParamError ; set message up for Std_EPrintf
|
|
stc ; return failure
|
|
jmp short poRet
|
|
|
|
poOk: mov byte ptr [bx],0 ; mark end of sort code list
|
|
pop bx ; BX = ptr to result buffer
|
|
clc ; return success
|
|
|
|
poRet: pop si ; restore SI
|
|
ret
|
|
|
|
ParseOrder endp
|
|
|
|
|
|
|
|
|
|
;*** ParseSwitch - parse a switch
|
|
;
|
|
; ENTRY BX = ptr to parse result buffer after system parser processed
|
|
; a switch
|
|
;
|
|
; EXIT CY = set if parse error occurred
|
|
;
|
|
; If parse error occurred, we're set up for Std_EPrintf call:
|
|
; AX = parse error code, like system parser
|
|
; DX = ptr to message block
|
|
;
|
|
; USED AX,BX,DX
|
|
;
|
|
; EFFECTS
|
|
;
|
|
; Bits may contain new option settings.
|
|
; DestBuf may contain new series of sort codes.
|
|
; AttrSpecified, AttrSelect may contain new attribute conditions.
|
|
;
|
|
; If parse error occurred, we're set up for Std_EPrintf call:
|
|
; Msg_Disp_Class = parse error class
|
|
; Byte after last parameter in text is zeroed to make ASCIIZ string
|
|
; Message block (see DX) is set up for parse error message
|
|
|
|
ParseSwitch proc
|
|
|
|
push cx ; save CX
|
|
push di ; save DI
|
|
|
|
mov ax,[bx].SynPtr ; AX = synonym ptr
|
|
mov di,offset TRANGROUP:Dir_Sw_Ptrs
|
|
; ES:DI = ptr to list of synonym ptrs
|
|
mov cx,NUM_DIR_SWS ; CX = # of dir switches in list
|
|
cld ; scan direction = upward
|
|
repne scasw ; locate synonym ptr in list
|
|
sub di,offset TRANGROUP:Dir_Sw_Ptrs + 2
|
|
|
|
; DI = index into word list of synonym ptrs
|
|
|
|
call cs:SwHandlers[di] ; use same index into call table
|
|
|
|
pop di ; restore DI
|
|
pop cx ; restore CX
|
|
|
|
ret
|
|
|
|
; Order in this table must correspond to order in Dir_Sw_Ptrs list.
|
|
; Simple on/off switches must occur first in both lists, and must be
|
|
; in order of option bits in Bits, starting with bit 0.
|
|
|
|
SwHandlers label word
|
|
dw OnOffSw ; /-W
|
|
dw OnOffSw ; /W
|
|
dw OnOffSw ; /-P
|
|
dw OnOffSw ; /P
|
|
dw OnOffSw ; /-S
|
|
dw OnOffSw ; /S
|
|
dw OnOffSw ; /-B
|
|
dw OnOffSw ; /B
|
|
dw OnOffSw ; /-L ;M010
|
|
dw OnOffSw ; /L ;M010
|
|
dw NoOrder ; /-O
|
|
dw ParseOrder ; /O
|
|
dw DefaultAttr ; /-A
|
|
dw ParseAttr ; /A
|
|
|
|
ParseSwitch endp
|
|
|
|
|
|
|
|
break <DIR utility routines>
|
|
|
|
;*** UTILITY ROUTINES
|
|
|
|
|
|
|
|
|
|
;*** ChangeDir - change directory on target drive
|
|
;
|
|
; ENTRY FCB contains drive #
|
|
; DS:DX = ptr to ASCIIZ string w/o drive specifier
|
|
;
|
|
; EXIT Changed current directory on drive
|
|
;
|
|
; If error,
|
|
; CY = set
|
|
; DOS Get Extended Error call will get error
|
|
;
|
|
; USED AX,DX,SI,DI
|
|
;
|
|
; EFFECTS
|
|
;
|
|
; DirBuf is used to build "d:string".
|
|
|
|
ChangeDir proc
|
|
|
|
mov di,offset TRANGROUP:DirBuf
|
|
call GetDriveLtr ; AX = "d:"
|
|
stosw ; put drive specifier in buffer
|
|
mov si,dx ; SI = ptr to argument string
|
|
cdLoop:
|
|
lodsb
|
|
stosb ; move byte to buffer
|
|
or al,al
|
|
jne cdLoop ; continue until null transferred
|
|
|
|
mov dx,offset TRANGROUP:DirBuf ; DX = ptr to "d:string"
|
|
mov ah,CHDIR
|
|
int 21h ; change directory
|
|
ret ; return what CHDIR returns
|
|
|
|
ChangeDir endp
|
|
|
|
|
|
|
|
|
|
;*** CmpAscz - compare two ASCIIZ strings alphanumerically
|
|
;
|
|
; ENTRY DS:SI = ptr to one ASCIIZ string
|
|
; ES:DI = ptr to another ASCIIZ string
|
|
;
|
|
; EXIT flags set after REPE CMPSB
|
|
;
|
|
; USED AL,CX,SI,DI
|
|
;
|
|
; NOTES
|
|
;
|
|
; Maximum run of comparison is length of DS:SI string.
|
|
; This ensures that two identical strings followed by
|
|
; random characters will compare correctly.
|
|
|
|
CmpAscz proc
|
|
|
|
push di
|
|
|
|
mov di,si
|
|
xor al,al
|
|
mov cx,0FFFFh
|
|
repne scasb
|
|
not cx
|
|
|
|
pop di
|
|
repe cmpsb
|
|
ret
|
|
|
|
CmpAscz endp
|
|
|
|
|
|
|
|
|
|
;*** CopyPathname - copy pathname to our buffer
|
|
;
|
|
; ENTRY BX = ptr to parse result buffer after system parser processed
|
|
; a filespec
|
|
;
|
|
; EXIT nothing
|
|
;
|
|
; USED AX
|
|
;
|
|
; EFFECTS
|
|
;
|
|
; SrcBuf may contain a new pathname/filespec.
|
|
; PathPos, PathCnt updated for new pathname.
|
|
|
|
CopyPathname proc
|
|
|
|
push si
|
|
lds si,dword ptr [bx].ValuePtr ; load far ptr from result buffer
|
|
invoke Move_To_SrcBuf ; copy pathname to SrcBuf
|
|
pop si
|
|
ret
|
|
|
|
CopyPathname endp
|
|
|
|
|
|
|
|
;*** CountFile - update counters with current file
|
|
;
|
|
; ENTRY BX = offset of entry in TPA buffer
|
|
;
|
|
; EXIT nothing
|
|
;
|
|
; USED AX,DX
|
|
;
|
|
; EFFECTS
|
|
;
|
|
; FileCnt, FileCntTotal, FileSiz, FileSizTotal are updated.
|
|
|
|
CountFile proc
|
|
|
|
push es ; save TRANGROUP seg addr
|
|
mov es,Tpa ; ES = TPA seg addr
|
|
|
|
inc FileCnt ; # files this directory
|
|
inc word ptr FileCntTotal ; # files total
|
|
jnz @F
|
|
inc word ptr FileCntTotal+2
|
|
@@:
|
|
mov ax,word ptr es:[bx].filesize ; AX = low word of file size
|
|
mov dx,word ptr es:[bx].filesize+2 ; DX = high word of file size
|
|
add word ptr FileSiz,ax
|
|
adc word ptr FileSiz+2,dx ; size of this directory
|
|
add word ptr FileSizTotal,ax
|
|
adc word ptr FileSizTotal+2,dx ; total size of files listed
|
|
|
|
pop es ; ES = TRANGROUP seg addr again
|
|
ret
|
|
|
|
CountFile endp
|
|
|
|
|
|
|
|
|
|
;*** DisplayBare - display filename in bare format
|
|
;
|
|
; ENTRY BX = offset of entry in TPA buffer
|
|
;
|
|
; EXIT DX = # char's displayed, including dot
|
|
;
|
|
; USED AX,CX,SI,DI
|
|
;
|
|
; EFFECTS
|
|
;
|
|
; Filename is displayed in name.ext format, followed by cr/lf.
|
|
; If /s is on, complete pathname is displayed.
|
|
;
|
|
; NOTE
|
|
;
|
|
; Directory pseudofiles . and .. and suppressed in bare listing.
|
|
|
|
DisplayBare proc
|
|
|
|
; Suppress . and .. files from bare listing.
|
|
|
|
mov cx,ds ; CX = saved TRANGROUP seg addr
|
|
mov ds,Tpa ; DS:BX = ptr to file entry
|
|
assume ds:NOTHING
|
|
cmp ds:[bx].filename,'.' ; check 1st char of filename
|
|
mov ds,cx ; DS = TRANGROUP seg addr again
|
|
assume ds:TRANGROUP
|
|
je dbRet ; it's . or .. - don't display
|
|
|
|
test Bits,mask subd
|
|
jz dbNameExt ; not /s - display filename only
|
|
|
|
invoke Build_Dir_String
|
|
mov di,offset TRANGROUP:BwdBuf ; ES:DI = ptr to dir string
|
|
|
|
test Bits,mask lcase ;M010;check for lowercase option
|
|
jz @F ;M010;lowercase not needed
|
|
mov si,di ;M010;DS:SI --> ASCIIZ string in BwdBuf
|
|
call LowercaseString ;M010;path string is in BwdBuf
|
|
|
|
@@: xor al,al ; AL = 0
|
|
mov cx,0FFFFh
|
|
cld
|
|
repne scasb ; ES:DI = ptr to byte after null
|
|
dec di ; ES:DI = ptr to null byte
|
|
|
|
ifdef DBCS
|
|
push si
|
|
push di
|
|
mov si,offset TRANGROUP:BwdBuf
|
|
dec di
|
|
call CheckDBCSTailByte
|
|
pop di
|
|
pop si
|
|
jz dbTailByte ; if last char is double byte
|
|
endif
|
|
|
|
cmp byte ptr es:[di-1],'\'
|
|
je @F ; already terminated w/ '\'
|
|
|
|
ifdef DBCS
|
|
dbTailByte:
|
|
endif
|
|
|
|
mov ax,'\' ; AX = '\',0
|
|
stosw ; add to dir string
|
|
@@:
|
|
mov String_Ptr_2,offset TRANGROUP:BwdBuf
|
|
mov dx,offset TRANGROUP:String_Buf_Ptr
|
|
invoke Std_Printf ; display device & directory path
|
|
|
|
dbNameExt:
|
|
call DisplayDotForm ; display name.ext
|
|
invoke CrLf2 ; display cr/lf
|
|
call UseLine ;M007;Allow /p with /b
|
|
dbRet: ret
|
|
|
|
DisplayBare endp
|
|
|
|
|
|
|
|
|
|
;*** DisplayDotForm - display filename in compressed dot format
|
|
;
|
|
; Display name.ext, with no cr/lf's. Dot is displayed only
|
|
; if the filename has a nonblank extension.
|
|
;
|
|
; ENTRY BX = offset of entry in TPA buffer
|
|
;
|
|
; EXIT DX = # char's displayed, including dot
|
|
;
|
|
; USED AX,CX,SI,DI
|
|
;
|
|
; EFFECTS
|
|
;
|
|
; Filename is displayed in name.ext format.
|
|
;
|
|
; NOTE
|
|
;
|
|
; We allow for bogus filenames that have blanks embedded
|
|
; in the name or extension.
|
|
|
|
; Bugbug: might be a good performance gain if we buffered
|
|
; up the output and used DOS function 9.
|
|
|
|
DisplayDotForm proc
|
|
|
|
push ds ; save TRANGROUP seg addr
|
|
push es ; save ES
|
|
mov ax,cs:Tpa ; AX = TPA seg addr
|
|
mov ds,ax ; DS:BX = ptr to entry
|
|
assume ds:nothing
|
|
mov es,ax ; ES:BX = ptr to entry
|
|
|
|
mov di,bx ; ES:DI = ptr to entry
|
|
add di,filename + size filename - 1
|
|
; ES:DI = ptr to last char in name field
|
|
mov cx,size filename ; CX = length of name field
|
|
mov al,' '
|
|
std ; scan down
|
|
repe scasb ; scan for nonblank
|
|
|
|
; Assume file name has at least one character.
|
|
|
|
inc cx ; CX = # chars in name
|
|
mov dx,cx ; DX = # chars to be displayed
|
|
|
|
mov si,bx ; DS:SI = ptr to entry
|
|
add si,filename ; DS:SI = ptr to name
|
|
|
|
NextNameChar:
|
|
cld
|
|
lodsb ; AL = next char
|
|
|
|
ifdef DBCS
|
|
invoke testkanj
|
|
jz @f ; if this is not lead byte
|
|
invoke Print_Char ; display lead byte
|
|
dec cx
|
|
jz ExtChar ; if this is end
|
|
lodsb ; get tail byte
|
|
jmp short NameChar10 ; display tail byte
|
|
@@:
|
|
endif
|
|
|
|
test Bits,mask lcase ;M010;check for lowercase option
|
|
jz @F ;M010;lowercase not required
|
|
call LowerCase ;M010;filename char is in AL
|
|
|
|
ifdef DBCS
|
|
NameChar10:
|
|
endif
|
|
|
|
@@: invoke Print_Char ; display it
|
|
loop NextNameChar
|
|
|
|
ifdef DBCS
|
|
ExtChar:
|
|
endif
|
|
|
|
; Now do extension.
|
|
|
|
mov di,bx ; ES:DI = ptr to entry
|
|
add di,fileext + size fileext - 1
|
|
; ES:DI = ptr to last char in ext field
|
|
mov cx,size fileext ; CX = length of ext field
|
|
mov al,' '
|
|
std ; scan down
|
|
repe scasb ; scan for nonblank
|
|
je ddDone ; no nonblank chars in ext
|
|
|
|
inc cx ; CX = # chars in ext
|
|
add dx,cx ; DX = total # chars to be displayed
|
|
inc dx ; including dot
|
|
|
|
mov al,'.'
|
|
invoke Print_Char
|
|
mov si,bx ; DS:SI = ptr to entry
|
|
add si,fileext ; DS:SI = ptr to ext
|
|
|
|
NextExtChar:
|
|
cld
|
|
lodsb ; AL = next char
|
|
|
|
ifdef DBCS
|
|
invoke testkanj
|
|
jz @f ; if this is not lead byte
|
|
invoke Print_Char ; display lead byte
|
|
dec cx
|
|
jz ddDone ; if this is end
|
|
lodsb ; get tail byte
|
|
jmp short ExtChar10 ; display tail byte
|
|
@@:
|
|
endif
|
|
|
|
test CS:Bits,mask lcase ;M010;check for lowercase option
|
|
jz @F ;M010;lowercase not required
|
|
call LowerCase ;M010;fileext char is in AL
|
|
|
|
ifdef DBCS
|
|
ExtChar10:
|
|
endif
|
|
|
|
@@: invoke Print_Char ; display it
|
|
loop NextExtChar
|
|
|
|
ddDone: pop es ; restore ES
|
|
pop ds ; DS = TRANGROUP seg addr again
|
|
assume ds:TRANGROUP
|
|
cld ; leave direction flag = up
|
|
ret
|
|
|
|
DisplayDotForm endp
|
|
|
|
|
|
|
|
|
|
;*** DisplayFile - display file entry, update counters
|
|
;
|
|
; ENTRY BX = offset of entry in TPA buffer
|
|
; Bits contains /w, /p settings
|
|
;
|
|
; EXIT nothing
|
|
;
|
|
; USED AX,CX,DX,SI,DI,BP
|
|
;
|
|
; EFFECTS
|
|
;
|
|
; Entry is displayed.
|
|
; If not /b,
|
|
; Cursor is left at end of entry on screen.
|
|
; FileCnt, FileCntTotal, FileSiz, FileSizTotal are updated.
|
|
; If /b,
|
|
; Cursor is left at beginning of next line.
|
|
; Cnt's and Siz's aren't updated.
|
|
|
|
DisplayFile proc
|
|
|
|
test Bits,mask bare
|
|
jz dfNorm ; not /b - do normal display
|
|
|
|
call DisplayBare ; display file in bare format
|
|
jmp short dfRet
|
|
|
|
dfNorm: call DisplayNext ; pos'n cursor for next entry
|
|
test Bits,mask wide
|
|
jz dfFull ; full format
|
|
call DisplayWide ; wide format
|
|
jmp short dfCnt
|
|
|
|
dfFull: call DisplayName ; display filename & extension
|
|
call DisplayTheRest ; display size, date, time
|
|
|
|
dfCnt: call CountFile ; update file counters
|
|
dfRet: ret
|
|
|
|
DisplayFile endp
|
|
|
|
|
|
|
|
|
|
;*** DisplayHeader - display directory header of working directory
|
|
;
|
|
; ENTRY Current directory (on selected drive) is the one to display
|
|
; LeftOnPage = # lines left on display page
|
|
;
|
|
; EXIT nothing
|
|
;
|
|
; ERROR EXIT
|
|
;
|
|
; Build_Dir_String will exit through CError with "Invalid drive
|
|
; specification" if there's a problem obtaining the current
|
|
; directory pathname.
|
|
;
|
|
; USED AX,DX,SI,DI
|
|
;
|
|
; EFFECTS
|
|
;
|
|
; BwdBuf (which is really the same buffer as DirBuf, which
|
|
; we are using for the DTA) contains the directory string.
|
|
; LeftOnPage is adjusted.
|
|
|
|
DisplayHeader proc
|
|
|
|
test Bits,mask bare
|
|
jnz dhRet ; /b - don't display header
|
|
|
|
test Bits,mask subd
|
|
jz dhNorm ; not /s
|
|
|
|
; For subdirectory listings, put a blank line before the header.
|
|
|
|
invoke Crlf2 ; start with a blank line
|
|
call UseLine
|
|
jmp short dhCom
|
|
dhNorm:
|
|
mov al,BLANK ; if not /s, precede by a blank
|
|
invoke Print_Char ; print a leading blank
|
|
dhCom:
|
|
invoke Build_Dir_String
|
|
mov dx,offset TRANGROUP:DirHead_ptr
|
|
invoke Std_Printf ; print header & cr/lf
|
|
call UseLine
|
|
invoke Crlf2 ; another cr/lf
|
|
call UseLine
|
|
dhRet: ret
|
|
|
|
DisplayHeader endp
|
|
|
|
|
|
|
|
|
|
;*** DisplayName - display file name & extension
|
|
;
|
|
; ENTRY BX = offset of entry in TPA buffer
|
|
;
|
|
; EXIT nothing
|
|
;
|
|
; USED AX,CX,DX,SI,DI
|
|
;
|
|
; EFFECTS
|
|
;
|
|
; Filename & extension are displayed in spread format.
|
|
; Cursor is left at end of extension.
|
|
|
|
DisplayName proc
|
|
|
|
push ds ; save TRANGROUP seg addr
|
|
mov ds,Tpa ; DS:BX = ptr to entry
|
|
assume ds:nothing
|
|
mov si,bx ; DS:SI = ptr to entry
|
|
add si,filename ; DS:SI = ptr to filename
|
|
mov di,offset TRANGROUP:CharBuf ; ES:DI = ptr to CharBuf
|
|
|
|
mov cx,8
|
|
cld
|
|
rep movsb ; move filename to CharBuf
|
|
mov al,' '
|
|
stosb ; add a blank
|
|
mov cx,3
|
|
rep movsb ; add extension
|
|
xor al,al
|
|
stosb ; add a NULL
|
|
|
|
pop ds ; DS = TRANGROUP seg addr again
|
|
assume ds:TRANGROUP
|
|
|
|
test Bits,mask lcase ;M010;check for lowercase option
|
|
jz @F ;M010;lowercase not required
|
|
mov si,offset TRANGROUP:CharBuf ;M010;DS:SI --> ASCIIZ string
|
|
call LowercaseString ;M010;filename.ext string is in CharBuf
|
|
|
|
@@: mov String_Ptr_2,offset TRANGROUP:CharBuf
|
|
mov dx,offset TRANGROUP:String_Buf_Ptr
|
|
invoke Std_Printf ; print filename & extension
|
|
ret
|
|
|
|
DisplayName endp
|
|
|
|
|
|
|
|
|
|
;*** DisplayNext - move display cursor to next entry position
|
|
;
|
|
; ENTRY LeftOnLine = # entries can still be printed on this line
|
|
; LeftOnPage = # lines can still be printed for this page
|
|
; FileCnt = # files in this dir displayed before this one
|
|
; Bits contains /w setting
|
|
;
|
|
; EXIT nothing
|
|
;
|
|
; USED AX,DX
|
|
;
|
|
; EFFECTS
|
|
;
|
|
; LeftOnLine will be updated to reflect the entry about to be
|
|
; displayed.
|
|
; LeftOnPage may be updated.
|
|
|
|
DisplayNext proc
|
|
|
|
cmp FileCnt,0
|
|
je dn1st ; 1st file in directory
|
|
cmp LeftOnLine,0
|
|
jng dnEol ; no more room on this line
|
|
|
|
; We are in wide mode (LeftOnLine is always 0 otherwise) and
|
|
; we still have room for more on this line.
|
|
; Tab to next position.
|
|
|
|
mov dx,offset TRANGROUP:Tab_Ptr
|
|
invoke Std_Printf
|
|
jmp short dnDone
|
|
|
|
dnEol:
|
|
|
|
; Start this entry on a new line.
|
|
|
|
invoke Crlf2 ; start on new line
|
|
call UseLine
|
|
dn1st: mov al,PerLine
|
|
mov LeftOnLine,al ; reset # entries left on line
|
|
|
|
dnDone: dec LeftOnLine ; reflect the entry about to be displayed
|
|
ret
|
|
|
|
DisplayNext endp
|
|
|
|
|
|
|
|
|
|
;*** DisplayTheRest - display file size/dir, date, time
|
|
;
|
|
; ENTRY BX = offset of entry in TPA buffer
|
|
; Display cursor is at end of file extension
|
|
;
|
|
; EXIT nothing
|
|
;
|
|
; USED AX,CX,DX,SI,DI,BP
|
|
;
|
|
; EFFECTS
|
|
;
|
|
; File size, date, & time are displayed.
|
|
|
|
DisplayTheRest proc
|
|
|
|
push es ; save TRANGROUP seg addr
|
|
mov es,Tpa ; ES = TPA seg addr
|
|
mov bp,bx ; BP = offset of entry in TPA
|
|
test es:[bp].fileattr,ATTR_DIRECTORY
|
|
jz drNonDir ; not a directory file
|
|
|
|
; For a directory file, display <DIR> instead of size.
|
|
|
|
mov dx,offset TRANGROUP:DMes_Ptr
|
|
invoke Std_Printf
|
|
jmp short drCom ; skip to common fields
|
|
|
|
drNonDir:
|
|
|
|
; For a non-directory file, display file size.
|
|
|
|
mov dx,word ptr es:[bp].filesize
|
|
mov File_Size_Low,dx
|
|
mov dx,word ptr es:[bp].filesize+2
|
|
mov File_Size_High,dx
|
|
mov dx,offset TRANGROUP:Disp_File_Size_Ptr
|
|
invoke Std_Printf
|
|
|
|
drCom:
|
|
|
|
; For all files, display date & time.
|
|
|
|
mov ax,es:[bp].filedate ; AX = date word
|
|
or ax,ax ; test for null date (DOS 1.x)
|
|
jz drDone ; no date, skip date/time display
|
|
mov bx,ax ; BX = date word
|
|
and ax,1Fh ; AX = day of month
|
|
mov dl,al ; DL = day of month
|
|
mov ax,bx ; AX = date word
|
|
mov cl,5
|
|
shr ax,cl ; shift day out
|
|
and al,0Fh ; AL = month
|
|
mov dh,al ; DH = month
|
|
mov cl,bh
|
|
shr cl,1 ; CL = year - 1980
|
|
xor ch,ch ; CX = year - 1980
|
|
add cx,80 ; CX = 2-digit year
|
|
cmp cl,100
|
|
jb @F ; not year 2000 yet, skip ahead
|
|
sub cl,100 ; adjust for 21st century
|
|
@@: xchg dh,dl ; DX = month/day
|
|
mov DirDat_Yr,cx ; move year to msg block
|
|
mov DirDat_Mo_Day,dx ; move month/day to msg block
|
|
mov cx,es:[bp].filetime ; CX = file time
|
|
jcxz drPrint ; no time field - go print
|
|
shr cx,1
|
|
shr cx,1
|
|
shr cx,1 ; CH = hours
|
|
shr cl,1
|
|
shr cl,1 ; CL = minutes
|
|
xchg ch,cl ; CX = hr/min
|
|
mov DirTim_Hr_Min,cx ; move time to msg block
|
|
drPrint:mov dx,offset TRANGROUP:DirDatTim_Ptr
|
|
invoke Std_Printf ; print date & time
|
|
|
|
drDone: pop es ; ES = TRANGROUP seg addr again
|
|
mov bx,bp ; BX = offset of entry in TPA again
|
|
ret
|
|
|
|
DisplayTheRest endp
|
|
|
|
|
|
|
|
|
|
;*** DisplayTrailer - display trailing lines for directory listing
|
|
;
|
|
; ENTRY LeftOnPage = # lines left on display page
|
|
; FileCnt = # files listed
|
|
; FileSiz = total size of files listed
|
|
;
|
|
; EXIT nothing
|
|
;
|
|
; USED
|
|
;
|
|
; EFFECTS
|
|
;
|
|
; Trailing info lines are displayed
|
|
|
|
DisplayTrailer proc
|
|
|
|
test Bits,mask bare
|
|
jnz dtrRet ; /b - don't display trailer
|
|
|
|
invoke Crlf2 ; start on new line
|
|
call UseLine
|
|
mov ax,FileCnt ; AX = # files found
|
|
|
|
DisplayCntSiz:
|
|
|
|
; DisplayTotals uses this entry point.
|
|
;
|
|
; AX = # files
|
|
; FileSiz = dword total size of files
|
|
|
|
mov Dir_Num,ax ; load # files
|
|
mov dx,offset TRANGROUP:DirMes_Ptr ; DX = ptr to message block
|
|
invoke Std_Printf ; "nnn File(s)"
|
|
|
|
mov dx,offset TRANGROUP:Bytes_Ptr
|
|
invoke Std_Printf ; "nnn bytes",cr,lf
|
|
call UseLine
|
|
|
|
dtrRet: ret
|
|
|
|
DisplayTrailer endp
|
|
|
|
|
|
|
|
|
|
;*** DisplayWide - display filename in wide format
|
|
;
|
|
; ENTRY BX = offset of entry in TPA buffer
|
|
;
|
|
; EXIT nothing
|
|
;
|
|
; USED AX,CX,DX,SI,DI
|
|
;
|
|
; EFFECTS
|
|
;
|
|
; Name.ext is displayed. Cursor left at end of field (padded
|
|
; with blanks). Subdirectory files are displayed as [name.ext].
|
|
|
|
DisplayWide proc
|
|
|
|
push ds ; save TRANGROUP seg addr
|
|
mov ds,Tpa ; DS:BX = ptr to entry
|
|
assume ds:nothing
|
|
|
|
test ds:[bx].fileattr,ATTR_DIRECTORY
|
|
jz @F ; not a subdirectory file
|
|
mov al,'['
|
|
invoke Print_Char ; prefix subdirectory
|
|
|
|
@@: call DisplayDotForm ; display name.ext
|
|
|
|
; DX = # chars displayed in name.ext
|
|
|
|
test ds:[bx].fileattr,ATTR_DIRECTORY
|
|
jz @F ; not a subdirectory file
|
|
mov al,']'
|
|
invoke Print_Char ; postfix subdirectory
|
|
@@:
|
|
; Pad field with blanks.
|
|
|
|
mov cx,size filename + size fileext + 1
|
|
; CX = field size
|
|
sub cx,dx ; CX = # pad char's
|
|
jcxz dwDone
|
|
mov al,' '
|
|
|
|
@@: invoke Print_Char
|
|
loop @B
|
|
|
|
dwDone: pop ds ; DS = TRANGROUP seg addr again
|
|
assume ds:TRANGROUP
|
|
ret
|
|
|
|
DisplayWide endp
|
|
|
|
|
|
|
|
|
|
;*** EndPage - end the current display page
|
|
;
|
|
; ENTRY LeftOnPage = # lines left on display page
|
|
; Current directory (on selected drive) is the one being listed
|
|
; Bits contains /p setting
|
|
;
|
|
; EXIT LeftOnPage = # lines left for next page
|
|
;
|
|
; USED AX,DX
|
|
;
|
|
; EFFECTS
|
|
;
|
|
; Pause is invoked to display a message and wait for a keystroke.
|
|
; BwdBuf (same as DirBuf) used to build directory string.
|
|
|
|
EndPage proc
|
|
|
|
test Bits,mask pagd
|
|
jz epNew ; paged display isn't enabled
|
|
|
|
push bx ; save BX
|
|
push cx ; save CX
|
|
|
|
invoke Pause ; "Press any key to continue..."
|
|
|
|
invoke Build_Dir_String
|
|
mov dx,offset TRANGROUP:DirCont_Ptr
|
|
invoke Printf_Crlf ; "(continuing <dir>)", cr/lf
|
|
|
|
pop cx ; restore CX
|
|
pop bx ; restore BX
|
|
|
|
epNew: mov ax,LinPerPag ; AX = # lines per page
|
|
dec ax ; AX = # lines till next EndPage
|
|
mov LeftOnPage,ax ; LeftOnPage = countdown variable
|
|
|
|
ret
|
|
|
|
EndPage endp
|
|
|
|
|
|
|
|
|
|
;*** GetDriveLtr - get target drive letter
|
|
;
|
|
; ENTRY FCB contains drive #
|
|
;
|
|
; EXIT AX = "d:"
|
|
;
|
|
; USED nothing
|
|
|
|
GetDriveLtr proc
|
|
|
|
mov al,ds:Fcb ; AL = target drive #
|
|
or al,al
|
|
jnz @F ; not current drive default, skip ahead
|
|
mov al,ds:CurDrv ; AL = current drive #
|
|
inc al ; AL = 1-based drive #
|
|
@@: add al,'A'-1 ; AL = target drive letter
|
|
mov ah,':' ; AX = "d:"
|
|
ret
|
|
|
|
GetDriveLtr endp
|
|
|
|
|
|
|
|
|
|
|
|
;*** SetupParamError - set up for Std_EPrintf parameter parse error message
|
|
;
|
|
; Do for our /O and /A string parsers what Parse_With_Msg does
|
|
; for system parser calls. Set up a message substitution block,
|
|
; etc. for invalid value strings. I copied the procedure from
|
|
; Setup_Parse_Error_Msg.
|
|
;
|
|
; ENTRY BX = ptr to system parser result buffer (contains ptr to str)
|
|
;
|
|
;
|
|
; EXIT AX = system parser error return code for bad param format
|
|
; DX = ptr to message description block for Std_EPrintf
|
|
;
|
|
; USED SI
|
|
;
|
|
; EFFECTS
|
|
;
|
|
; Msg_Disp_Class = parse error message class
|
|
; Message block (see DX) is set up for parse error message
|
|
|
|
SetupParamError proc
|
|
|
|
mov ax,9 ; parse error #
|
|
mov Msg_Disp_Class,PARSE_MSG_CLASS
|
|
mov Extend_Buf_Ptr,ax
|
|
mov si,word ptr [bx].ValuePtr
|
|
mov String_Ptr_2,si
|
|
mov Extend_Buf_Sub,ONE_SUBST
|
|
mov dx,offset TRANGROUP:Extend_Buf_Ptr
|
|
ret
|
|
|
|
SetupParamError endp
|
|
|
|
|
|
|
|
|
|
;*** UseLine - use a display line, start a new page if none left
|
|
;
|
|
; ENTRY nothing
|
|
;
|
|
; EXIT nothing
|
|
;
|
|
; USED flags
|
|
|
|
UseLine proc
|
|
|
|
dec LeftOnPage
|
|
ifndef NEC_98
|
|
cmp LeftOnPage,2
|
|
else ;NEC_98
|
|
cmp LeftOnPage,1 ;NEC04 Canged Page Line (23 to 24)
|
|
endif ;NEC_98
|
|
ja ulRet
|
|
call EndPage
|
|
ulRet: ret
|
|
|
|
UseLine endp
|
|
|
|
|
|
|
|
|
|
;*** ZeroTotals - zero grand total file count, size
|
|
;
|
|
; ENTRY nothing
|
|
;
|
|
; EXIT nothing
|
|
;
|
|
; USED AX
|
|
;
|
|
; EFFECTS
|
|
;
|
|
; FileCntTotal & FileSizTotal are zeroed.
|
|
;
|
|
; NOTES
|
|
;
|
|
; FileCntTotal & FileSizTotal must be juxtaposed, in that order.
|
|
|
|
ZeroTotals proc
|
|
|
|
mov di,offset TRANGROUP:FileCntTotal
|
|
mov cx,size FileCntTotal+size FileSizTotal
|
|
xor al,al
|
|
rep stosb
|
|
ret
|
|
|
|
ZeroTotals endp
|
|
|
|
|
|
|
|
|
|
;*** CtrlCHandler - our own control-c handler
|
|
;
|
|
; Make sure user's default directory gets restored. See notes
|
|
; at InstallCtrlCHandler.
|
|
;
|
|
; ENTRY control-c
|
|
;
|
|
; EXIT to OldCtrlCHandler
|
|
;
|
|
; USED DS,flags
|
|
;
|
|
; EFFECTS
|
|
;
|
|
; Restore user's default directory.
|
|
;
|
|
; NOTES
|
|
;
|
|
; This handler is only installed after calling PathCrunch,
|
|
; which sets UserDir1, so the restoration will work.
|
|
;
|
|
; The original control-c vector will be restored, whether
|
|
; or not this one is invoked, in the HeadFix routine.
|
|
|
|
CtrlCHandler proc far
|
|
|
|
;SR;
|
|
; Save all registers used: ds, dx, ax. I know ax is being used by the
|
|
;CtrlC handler, am not sure about ds & dx. Save them to be safe
|
|
;
|
|
push ds
|
|
push cs
|
|
pop ds ; DS = TRANGROUP seg addr
|
|
push ax
|
|
push dx
|
|
invoke RestUDir ; restore user's default directory
|
|
pop dx
|
|
pop ax
|
|
pop ds
|
|
jmp cs:OldCtrlCHandler ; go to previous int 23 handler
|
|
|
|
CtrlCHandler endp
|
|
|
|
|
|
;M010;start
|
|
;*** LowerCase - convert ASCII character in AL to lowercase
|
|
;
|
|
; ENTRY AL = character to be displayed
|
|
;
|
|
; EXIT AL is lowercase
|
|
;
|
|
; USED nothing
|
|
|
|
LowerCase proc
|
|
|
|
assume ds:NOTHING,es:NOTHING
|
|
|
|
cmp al,'A' ; ensure AL is in range 'A'-'Z'
|
|
jb lcRet
|
|
cmp al,'Z'
|
|
ja lcRet
|
|
|
|
or al,20h ; convert to ASCII lowercase (UpperCase+32)-->LowerCase
|
|
|
|
lcRet: ret
|
|
|
|
LowerCase endp
|
|
|
|
|
|
|
|
|
|
;*** LowercaseString - convert ASCIIZ string at DS:SI to lowercase
|
|
;
|
|
; ENTRY DS:SI points to start of ASCIIZ string
|
|
; ES = DS
|
|
;
|
|
; EXIT nothing
|
|
;
|
|
; USED AL,SI
|
|
|
|
LowercaseString proc
|
|
|
|
assume ds:NOTHING,es:NOTHING
|
|
|
|
push di ; save di
|
|
mov di,si ; ES:DI --> ASCIIZ string
|
|
cld
|
|
|
|
NextChar:
|
|
lodsb ; get character from string into al
|
|
or al,al ; are we at end of string?
|
|
jz EndOfString
|
|
|
|
ifdef DBCS
|
|
invoke testkanj
|
|
jz @f ; if this is not lead byte
|
|
stosb ; store lead byte
|
|
lodsb ; get tail byte
|
|
or al,al
|
|
jz EndOfString ; if end
|
|
stosb ; store tail byte
|
|
jmp short NextChar
|
|
@@:
|
|
endif
|
|
|
|
call LowerCase ; convert character to lowercase
|
|
stosb ; store character back into buffer
|
|
jmp SHORT NextChar ; repeat until end of string
|
|
|
|
EndOfString:
|
|
pop di ; restore di
|
|
ret
|
|
|
|
LowercaseString endp
|
|
;M010;end
|
|
|
|
ifdef DBCS
|
|
;
|
|
; Check if the character position is at Tail Byte of DBCS
|
|
;
|
|
; input: ds:si = start address of the string
|
|
; ds:di = character position to check
|
|
; output: ZF = 1 if at Tail Byte
|
|
;
|
|
CheckDBCSTailByte proc near
|
|
push ax
|
|
push cx
|
|
push di
|
|
mov cx,di ; save character position
|
|
cdtb_check:
|
|
cmp di,si
|
|
jz cdtb_next ; if at the top
|
|
dec di ; go back
|
|
mov al,[di] ; get character
|
|
invoke testkanj
|
|
jnz cdtb_check ; if DBCS lead byte do next
|
|
inc di ; adjust
|
|
cdtb_next:
|
|
sub cx,di ; if the length is odd then
|
|
xor cl,1 ; the character position is
|
|
test cl,1 ; at the tail byte
|
|
pop di
|
|
pop cx
|
|
pop ax
|
|
ret
|
|
CheckDBCSTailByte endp
|
|
endif
|
|
|
|
|
|
TRANCODE ends
|
|
|
|
end
|
|
|