textfiles/apple/format.nufx.txt

1847 lines
46 KiB
Plaintext

-------------------------------------------------------------------------------
| The New Archive Standard for the Apple II -- "NuFX" -- NuFile eXchange |
-------------------------------------------------------------------------------
NuFX Documentation - 12/2/88 by Andy Nicholas
Final Revision Three - 2/3/89
Please distribute as widely as possible
Send questions/comments to:
Paper Bag Productions CSNET: nicholaA@moravian.edu
c/o Andy Nicholas InterNET: nicholaA%batman.moravian.edu@relay.cs.net
Box 435 nicholaA@batman.moravian.edu.csnet
Moravian College ProLine: andyn@pro-sol.cts.com [619-670-5379]
Bethlehem, PA 18018 AppleLink PE: ShrinkIt
NuFX Documentation 2/3/89 - Page 2 -
Preface
-------
The first copy of this proposal that was circulated was preliminary
revision 10. Since then I have had many requests for a more flexible
solution to the archival problem. To do so meant the removal of such
features as the alignment to 128-byte boundaries of all data blocks and
so forth, and the re-definition of the header block so that it is less rigid
in its format (extensible).
To do this makes it at least a factor more complex for programs which can
extract records from NuFX archives on-the-fly (such as telecommunications
programs). While this may be seen as a hardship by some, I believe the
additional power provided by all formats after revision 10 justifies the
additional complexity. Standalone utility programs should have no problems
adjusting to the new format.
This final revision of the documentation fixes the master_header's size
at 48 bytes, eliminates the archive_create_program and archive_mod_program
fields, increases the size of the access field to a longword, and introduces
a new name for the archive, NFX, or NuFX, for "NuFile eXchange."
I am also almost finished an archive program which places files or disks
into NuFX archives using dynamic LZW compression, ShrinkIt. Any reference
to ShrinkIt in the text is a reference to this program.
Final Rev 2 corrects the all the algorithms to include the length of the
thread list in their calculations, includes some notes on how to properly
include threads and directories, and standardizes on the use of "NuFX" to
describe the archive name so as to distinguish it from Sun Microsystems'
"NFS" network product.
Final Rev 3 changes some of the terminology used, changes the date labels
to xx_when instead of xxx_date_time, assigns the resource_fork of a file
to be stored as a thread_kind in a data_thread instead of having it's own
thread available. An attempt at further clarity has been undertaken.
andy (2/3/89)
NuFX Documentation 2/3/89 - Page 3 -
History
-------
The Apple II community has always been sorely lacking a well-defined method
for archiving files. NuFX is an attempt to rectify the situation by providing
for a flexible, consistent standard for archiving files, disks, and other
computer medium.
The Binary II standard, authored by Gary B. Little, for placing multiple files
within a single file has been rendered obsolete by the (now) recent release of
GS/OS(tm) which provides access to multiple filing systems. Since GS/OS can
(or will) use HFS files, Binary II does NOT provide for:
o filenames larger than 64 characters
(GS/OS can create 8000 character filenames)
o a convenient way to add to, remove from, and other ways work on an archive.
o including HFS-style files which contain resource forks.
o including of entire disk images.
o including messages along with a file.
o a convenient way to represent that a file is compressed or encrypted by
a specific application. (an excess secondary header must be attached to
the beginning of files, such is the case with SQueezed files)
o a true archive standard. Binary II's original intent was to make transfer
of Apple II files from local machines to large information services like
The Source, Delphi, CompuServe, and GEnie, possible. Otherwise, a file's
attribute information would be lost. Binary II is now being stretched
beyond what it was originally meant to do.
o no support for multiple data threads or structures.
Adding all of these features to the existing Binary II standard would not only
be nerve-racking, but nearly impossible without violating the existing standard
and causing a great deal of confusion (ie, "did you say it was Binary II, first
revision or the second one?") Although Binary II is flexible, it is simply
unable to address all of these concerns without alienating existing Binary II
extraction programs.
So, to provide some differentiation between standards and provide a better
functioning format, I have defined a new standard called "NuFX" (NuFile
eXchange for the Apple II). NuFX fixes the problems that Apple IIgs(tm)
users would soon be experiencing as other filing systems become available for
GS/OS(tm). I am trying to stop a set of problems before they have a chance to
develop. NuFX provides all of the features of Binary II, but doesn't stop
there... it goes farther to allow the user the ultimate in flexibility,
usefulness and performance.
NuFX Documentation 2/3/89 - Page 4 -
Implementation
--------------
The basic structure of a NuFX archive is as follows:
[First record] [Next Record]
+---------------------------------------------------------------------------+
| Master Header | Header | Data . . . . . . . . | Header | Data . . . . |
+---------------------------------------------------------------------------+
A single master header block contains values which describe the entire archive
(those of you who are into structured programming can consider them archive
globals). Each of the succeeding header blocks contain only information about
the record they precede (consider them archive locals).
Each header block may be followed by a series of "threads." Each thread may
be a portion of data, a message, the resource part of an extended file, a
control sequence for a NuFX utility program, or almost any sort of
sequential data. The number of threads is described as a longword (32-bit
word), so it is also possible to properly archive and store the data
portions of sparse files.
NuFX Documentation 2/3/89 - Page 5 -
Master header block contents:
All word and double-word values are byte-reversed.
Offset Length Content
------ ------ ---------------------------------------------------------------
+0 1 $4E Master ID Byte #1
+1 1 $F5 Master ID Byte #2
+2 1 $46 Master ID Byte #3 spells the word "NuFile" in
+3 1 $E9 Master ID Byte #4 alternating ASCII (high, low) for
+4 1 $6C Master ID Byte #5 uniqueness.
+5 1 $E5 Master ID Byte #6
+6 2 master_crc
16-bit CRC of the remaining fields in this block.
(bytes +8 through +47)
Any programs which modify the master header block *MUST*
recalculate the CRC for the master header.
+8 4 total_records
Total number of records in this archive file.
It is possible to chain multiple records (Files or
Disks) together. It is also possible to chain
different types of records together (Files and
Disks mixed).
NuFX Documentation 2/3/89 - Page 6 -
+12 8 archive_create_when
The date and time on which this archive was initially
created. This field should never be changed once initially
written.
The format of this field is as follows:
+12 second - 0 through 59
+13 minute - 0 through 59
+14 hour - 0 through 23
+15 current Year minus 1900
+16 day - 0 through 30
+17 month - 0 through 11, with 0=January
+18 filler byte - reserved must be null (00).
+19 weekDay - 1 through 7, with 1=Sunday
The format of this field is identical to that described
in the _ReadTimeHex ($0D03) call described on page 14-14
of the Apple_IIgs_Toolbox_Reference:_Volume_1.
If the date is not known, or is unable to be calculated, this
field should be set to null (00). If the weekDay is not
known, or is unable to be calculated, this field should be
set to null (00).
+20 8 archive_mod_when
The date of the last modification to this archive. This field
should be changed every time a change is made to any of the
records in the archive.
The format of this field is as follows:
+20 second - 0 through 59
+21 minute - 0 through 59
+22 hour - 0 through 23
+23 current Year minus 1900
+24 day - 0 through 30
+25 month - 0 through 11, with 0=January
+26 filler byte - reserved, must be null (00).
+27 weekDay - 1 through 7, with 1=Sunday
The format of this field is identical to that described
in the _ReadTimeHex ($0D03) call described on page 14-14
of the Apple_IIgs_Toolbox_Reference:_Volume_1.
If the date is not known, or is unable to be calculated, this
field should be set to null (00). If the weekDay is not
known, or is unable to be calculated, this field should be
set to null (00).
NuFX Documentation 2/3/89 - Page 7 -
+28
.
. *** RESERVED, MUST BE SET TO NULL (00) ***
. Do NOT use any of these fields.
.
+47
NuFX Documentation 2/3/89 - Page 8 -
The following header block must precede each record within the NuFX archive.
The cyclic redundancy check (CRC) has been provided to detect archives which
have possibly been corrupted. The only time the CRC should be included in
in a block is for the master header and for each of the regular header blocks.
The CRC functions to ensure reliability and record integrity.
Header Block contents:
All word and double-word values are byte-reversed.
Offset Length Content
------ ------ ---------------------------------------------------------------
+0 1 $4E - Header ID Byte #1
+1 1 $F5 - Header ID Byte #2 Spells "NuFX" in alternating ascii
+2 1 $46 - Header ID Byte #3 (high/low) for uniqueness.
+3 1 $D8 - Header ID Byte #4
+4 2 header_crc
16-bit CRC of the remaining fields of this block.
(bytes 6 through the end of the attributes, filename, and
any threads.)
This field is used to verify the integrity of the rest of the
block.
Programs which make NuFX archives *MUST* include this
in every header. It is up to the discretion of the extracting
program to check the validity of these bytes. Any programs
which might modify the header of a particular record *MUST*
recalculate the CRC for the header block.
+6 2 attrib_count
This field describes the length of the attribute section of each
header in bytes. This count measures the distance in bytes
from the first field (offset +0) to and including the
filename_length field. By convention, the filename_length
field will always be the last 2 bytes of the attribute
section regardless of what has preceded it.
NuFX Documentation 2/3/89 - Page 9 -
+8 2 version_number
Minimum NuFX version number needed for extraction.
(Currently $0000)
This field is used to detect the possible existence of
other as-of-yet undefined fields and features. Utility
programs should check this value to be certain that they
are capable of extracting a record with this minimum version.
+10 4 total_threads
The number of thread sub-records which should be expected
immediately following the end of the file/path name. This
field is extremely important because it contains the
information about the length of the last 1/3 of the header.
+14 2 file_sys_id
Native file system identifier:
$0000 reserved
$0001 ProDOS/SOS
$0002 DOS 3.3
$0003 DOS 3.2
$0004 Apple II Pascal
$0005 Macintosh(tm) (HFS)
$0006 Macintosh (MFS)
$0007 LISA(tm) file system
$0008 Apple CP/M
$0009 reserved, do not use
$000A MS-DOS
$000B High-Sierra/ISO 9660
$000C
.
. reserved
.
$FFFF
Disk: if the file system of a disk is not known, then
this field should be set to null (0000).
NuFX Documentation 2/3/89 - Page 10 -
+16 2 file_sys_info
Information about the current filing system:
[$00xx]
Native file system separator. Under Prodos, the "/" ($2F)
character is used to separate paths. Under HFS, the
":" ($3A) character is used to separate paths. Under
MS-DOS, the "\" ($5C) character is used to separate paths.
The low byte of this word is used to store the file system's
separator.
The primary reason for including this field is that the
receiving file system (say, to Prodos 8 from GS/OS running an
HFS File System Translator) must know how to parse a valid
file/path name from the filename field for the receiving file
system.
[$xx00]
Sparse byte. If the high-byte of this word is $01, then
the image which follows is a sparse file (and the threads
should have been filled in properly to indicate this). If
a more "normal" image follows, this byte will be null (00).
NuFX Documentation 2/3/89 - Page 11 -
+18 4 access [0000 0000 0000 0000 DRB00IWR]
bits 31-8 reserved, must be zero
bit 7 D=0, destroy disabled
D=1, destroy enabled
bit 6 R=0, rename disabled
R=1, rename enabled
bit 5 B=0, backup not needed
B=1, backup needed
bits 4-3 reserved, must be zero
bit 2 I=0, file is visible
I=1, file is invisible
bit 1 W=0, write disabled
W=1, write enabled
bit 0 R=0, read disabled
R=1, read enabled
Disk: this field should be set to null (00).
+22 4 file_type
Disk: this field should be set to null (00).
+26 4 extra_type
ProDOS aux_type or HFS creator_type
Disk: this field *MUST* be set to the total number of blocks
on the device. This information must be present so that
the extracting program can place the record on the proper
type of device.
+30 2 storage_type
$0 - $3 = standard file
$5 = extended (gs/os) file
$d = subdirectory
Disk: file_sys_block_size
This should only be used if a disk is being archived. The block
size used by the device should be placed in this field. For
example, under Prodos, this field will be 512, while HFS
would set it to 524.
NuFX Documentation 2/3/89 - Page 12 -
+32 8 create_when
The date and time on which this record was initially created.
If the creation date and time is available from a disk device,
this information should be included.
The format of this field is as follows:
+32 second - 0 through 59
+33 minute - 0 through 59
+34 hour - 0 through 23
+35 current Year minus 1900
+36 day - 0 through 30
+37 month - 0 through 11, with 0=January
+38 filler byte - reserved, must be set to null (00).
+39 weekDay - 1 through 7, with 1=Sunday
The format of this field is identical to that described
in the _ReadTimeHex ($0D03) call described on page 14-14
of the Apple_IIgs_Toolbox_Reference:_Volume_1.
If the date is not known, or is unable to be calculated, this
field should be set to null (00). If the weekDay is not
known, or is unable to be calculated, this field should be
set to null (00).
+40 8 mod_when
The date and time on which this record was last modified.
If the modification date is available from a disk device,
this information should be included.
The format of this field is as follows:
+40 second - 0 through 59
+41 minute - 0 through 59
+42 hour - 0 through 23
+43 current Year minus 1900
+44 day - 0 through 30
+45 month - 0 through 11, with 0=January
+46 filler byte - reserved, must be set to null (00).
+47 weekDay - 1 through 7, with 1=Sunday
The format of this field is identical to that described
in the _ReadTimeHex ($0D03) call described on page 14-14
of the Apple_IIgs_Toolbox_Reference:_Volume_1.
If the date is not known, or is unable to be calculated, this
field should be set to null (00). If the weekDay is not
known, or is unable to be calculated, this field should be
set to null (00).
NuFX Documentation 2/3/89 - Page 13 -
+48 8 archive_when
The date and time on which this record was placed in this
archive.
The format of this field is as follows:
+48 second - 0 through 59
+49 minute - 0 through 59
+50 hour - 0 through 23
+51 current Year minus 1900
+52 day - 0 through 30
+53 month - 0 through 11, with 0=January
+54 filler byte - reserved, must be set to null (00).
+55 weekDay - 1 through 7, with 1=Sunday
The format of this field is identical to that described
in the _ReadTimeHex ($0D03) call described on page 14-14
of the Apple_IIgs_Toolbox_Reference:_Volume_1.
If the date is not known, or is unable to be calculated, this
field should be set to null (00). If the weekDay is not
known, or is unable to be calculated, this field should be
set to null (00).
Any other attributes which are needed may be added at the discretion of the
NuFX application programmer. The attrib_count field should be modified
accordingly.
NuFX Documentation 2/3/89 - Page 14 -
attrib_count-2 filename_length (xx bytes)
Length of filename. Under Prodos, this will not exceed 64
characters. If HFS or another filing system is used, this
field may exceed 64 characters. This is the last field
considered included in the attributes section. To allow the
inclusion of future additional parameters in the attributes
section, NuFX utility programs should rely on the
attribs_count field to find the filename_length field.
-------------------------
End of attributes section
xx Bytes Filename or partial pathname if applicable.
If this is a disk which is being archived, then the
volume_name should be included in this field. If a volume
name is included in this field, a separator ("/" or ":")
should *NOT* be included in, or precede the name. If a volume
name is not available, then this field should be set to nulls.
(00's)
If a partial pathname is specified, the directories to which
the current pathname refers need not have preceded this
particular record. The extraction program must test each
referenced directory individually. If the directory in
question does not exist, the extracting program should create
it.
Any utility which extracts files from a NuFX archive *MUST NOT*
assume that this field will be in a format it is able to
handle. In particular, extraction programs should check for
mixed case text in a file/path name and do whatever
conversions are necessary to parse a legal file/path name.
In general, assume nothing.
-----------------------
End of filename section
NuFX Documentation 2/3/89 - Page 15 -
Threads
-------
Thread records are 16 byte records which immediately follow the filename and
describe the types of data structures which are included with a given record.
The number of thread records is described in the attribute section by a
longword, total_threads.
Each thread record should be checked for the type of information that a given
utility program can extract. If a utility is incapable of extracting a
particular thread, that thread should be skipped. If a utility finds a
redundancy in a thread_record, it must decide whether to skip the record or to
do something with that particular thread (ie, if a utility finds 2
message_threads it can either ignore the second thread or display it. Likewise,
if a utility finds 2 resource threads, it can either overwrite the first thread
which was extracted, or warn the user and skip the errant thread).
A thread record can be represented as follows:
Offset Length Content
------ ------ ---------------------------------------------------------------
+0 2 thread_class
+2 2 thread_format
+4 2 thread_kind
+6 2 reserved
+8 4 thread_eof
+12 4 comp_thread_eof
"thread_class" describes the classification of the thread
---------------------------------------------------------
$0000 = message_thread
$0001 = control_thread
$0002 = data_thread
$0003 = sparse_thread
NuFX Documentation 2/3/89 - Page 16 -
"thread_format" is the format of the data within the thread.
------------------------------------------------------------
$0000 = Uncompressed [Not application specific]
$0001 = SQueezed (SQ/USQ) [Not application specific]
$0002 = Dynamic LZW [ShrinkIt]
$0003
.
. RESERVED, contact the author
.
$FFFF
"thread_kind" describes the kind of data which is contained in the thread
-------------------------------------------------------------------------
if thread_class
$0000 = message_thread
thread_kind $0000 = ASCII text
$xxxx = all others undefined
$0001 = control_thread
thread_kind $0000 = create directory
$xxxx = all others undefined
$0002 = data_thread
thread_kind $0000 = data_fork of file
thread_kind $0001 = disk image
thread_kind $0002 = resource_fork of file
$xxxx = all others undefined
"thread_eof" is the length of the uncompressed thread
-----------------------------------------------------
"comp_thread_eof" is the length of the compressed thread
--------------------------------------------------------
Current ideas for messages include static pictures, sounds, sound & pictures,
animations, and possibly executable files. I encourage writers of NuFX
utility programs to be able to handle messages of the lowest common
denominator, ASCII text.
NuFX Documentation 2/3/89 - Page 17 -
Finding the start of the thread list
------------------------------------
The beginning of the thread records can be found by using the following
algorithm:
Threads := (mark at beginning of header) + (attrib_count) +
(filename_length)
Finding the end of the thread list
----------------------------------
The end of the thread records can be found by applying the following
algorithm:
endOfThreads := (mark at beginning of header) + (attrib_count) +
(filename_length) + (16 * total_threads)
Finding the start of a data_thread
----------------------------------
The beginning of a data_thread can be found using the following algorithm:
Data Mark := (mark at beginning of header) + (attrib_count) +
(filename_length) + (16 * total_threads) +
(comp_thread_eof of all threads in the thread list which
are not data prior to finding a data_thread)
Finding the start of a resource_thread
--------------------------------------
The beginning of a resource_thread can be found using the following algorithm:
Resource Mark := (mark at beginning of header) + (attrib_count) +
(filename_length) + (16 * total_threads) +
(comp_thread_eof of all the threads in the thread list
which are not resources prior to finding a
resource_thread)
Finding the next record
-----------------------
The next record can be found using the following algorithm:
Next Mark := (mark at beginning of header) + (attrib_count) +
(filename_length) + (16 * total_threads) +
(comp_thread_eof of each thread)
NuFX Documentation 2/3/89 - Page 18 -
Misc notes on threads
---------------------
There must *ALWAYS* be at least 1 thread attached to each record, whether
the thread has any physical length or not. Phantom files and directories will
have both the thread_eof and comp_thread_eof fields set to null (00).
If a control_thread indicates that a directory should be created on the
destination device, the path to be created must take the form of a prodos
partial pathname. That is, the path must *NOT* be preceded with a volume
name. ie, /STUFF/SUBDIR is an invalid path, while SUBDIR/ANOTHERSUB is
a legal path.
If a control_thread indicates that a directory is to be created, *ALL*
the subdirectories that are contained in the pathname must be created.
Control_threads will be eventually used to control the execution of utility
programs by allowing for directory creation, renaming, deleting, moving,
or modifying files. A form of scripting language will eventually be available
to allow utility programs to perform these actions automatically.
Control_threads will allow extraction programs to perform operations akin to
Apple's GS/OS installer program, allowing updates to program sets dependent
upon such things as date of creation or modification, version number, etc.
NuFX Documentation 2/3/89 - Page 19 -
Normal Files
------------
Normal Prodos files (sub_types $01,$02,$03) should be handled in the
following manner: the data portion of the file will occupy the first
data_thread.
Sample header block for a normal file record:
00 01 02 03 04 05 06 07 08 09 0A 0B 0C 0D 0E 0F
+-------------------------------------------------+ created 01:10:00
00 | 4E F5 46 D8 55 34 3A 00 00 00 01 00 00 00 01 00 | 10/22/88
10 | 2f 00 00 00 c3 00 04 00 00 00 00 00 00 00 01 00 | saturday
20 | 00 0a 01 58 16 0a 00 07 00 10 0b 58 11 0b 00 05 |
30 | 00 0c 01 28 16 0a 00 07 05 00 53 54 55 46 46 02 | last mod 11:16:00
40 | 00 02 00 00 00 00 00 00 20 00 00 00 10 00 00 | 11/17/88
+-------------------------------------------------+ thursday
archived 01:12:00
header_id = "NuFX" 10/22/88
header_crc = $3455 saturday
attrib_count = $003A (58 bytes in attrib section)
version = $0000
total_threads = $00000001
file_sys_id = $0001 (Prodos)
file_sys_info = $002f (not sparse, / = separator)
access = $000000C3 (full access, not invis)
filetype = $00000004 (Prodos BIN)
aux_type = $00000000
storage_type = $0001 (Prodos sapling file)
create_when = 00 0a 01 58 16 0a 00 07
mod_when = 00 10 0b 58 11 0b 00 05
archive_when = 00 0c 01 58 16 0a 00 07
filename_length = $0005
filename = "STUFF"
thread_class = $0002 (data_thread)
thread_format = $0002 (compressed with ShrinkIt)
thread_kind = $0000 (file)
reserved = $0000
thread_eof = $00002000
comp_thread_eof = $00001000 (file is 50% of original size)
NuFX Documentation 2/3/89 - Page 20 -
Extended Files
--------------
Sample header block for an extended file record:
00 01 02 03 04 05 06 07 08 09 0A 0B 0C 0D 0E 0F
+-------------------------------------------------+ created 01:10:00
00 | 4E F5 46 D8 65 78 3A 00 00 00 02 00 00 00 01 00 | 10/22/88
10 | 2f 00 00 00 c3 00 b3 00 00 00 00 00 00 00 05 00 | saturday
20 | 00 0a 01 58 16 0a 00 07 00 10 0b 58 11 0b 00 05 |
30 | 00 0c 01 58 16 0a 00 07 09 00 45 58 54 2e 53 54 | last mod 11:16:00
40 | 55 46 46 02 00 02 00 00 00 00 00 00 20 00 00 00 | 11/17/88
50 | 08 00 00 02 00 02 00 00 00 00 00 00 10 00 00 00 | thursday
60 | 08 00 00 |
+-------------------------------------------------+ archived 01:12:00
10/22/88
header_id = "NuFX" saturday
header_crc = $7865
attrib_count = $003A (58 bytes in attrib section)
version = $0000
total_threads = $00000002
file_sys_id = $0001 (Prodos)
file_sys_info = $002f (not sparse, / = separator)
access = $000000C3 (full access, not invis)
filetype = $000000B3 (Prodos S16)
aux_type = $00000000
storage_type = $0005 (extended file)
create_when = 00 0a 01 58 16 0a 00 07
mod_when = 00 10 0b 58 11 0b 00 05
archive_when = 00 0c 01 58 16 0a 00 07
filename_length = $0009
filename = "EXT.STUFF"
thread_class = $0002 (data_thread)
thread_format = $0002 (compressed by ShrinkIt)
thread_kind = $0000 (file)
reserved = $0000
thread_length = $00002000
comp_thread_length = $00000800 (data_fork is 25% of original size)
thread_class = $0002 (data_thread)
thread_format = $0002 (compressed by ShrinkIt)
thread_kind = $0002 (resource_fork)
reserved = $0000
thread_eof = $00001000
comp_thread_eof = $00000800 (resource is 50% of original size)
NuFX Documentation 2/3/89 - Page 21 -
Disks
-----
If the file system of a particular disk is not known, the file_sys_id field
should be set to null, the volume name should also be set to null, and all
the other fields pertaining only to files should be set to null.
If the file system of a particular disk *IS* known, as many of the fields
as possible should be filled with the correct information. Fields which do not
pertain to an archived disk should remain set to null.
If an entire disk is added to the archive without some form of compression
(ie, record_format = uncompressed), then the blocks which comprise the disk
image *MUST* be added sequentially from the first through the last block.
Since there will be no character included in the data stream to mark the
end/beginning of a block, extraction programs should rely on the
file_sys_block_size field to determine how many bytes to read from the record
to properly fill a block.
NuFX Documentation 2/3/89 - Page 22 -
Sample header block for a disk record:
00 01 02 03 04 05 06 07 08 09 0A 0B 0C 0D 0E 0F
+-------------------------------------------------+ created 01:10:00
00 | 4E F5 46 D8 67 05 3A 00 00 00 01 00 00 00 01 00 | 10/22/88
10 | 2f 00 00 00 00 00 00 00 00 00 40 06 00 00 00 02 | saturday
20 | 00 0a 01 58 16 0a 00 07 00 10 0b 58 11 0b 00 05 |
30 | 00 0c 01 58 16 0a 00 07 04 00 44 49 53 4B 02 00 | last mod 11:16:00
40 | 02 00 01 00 00 00 00 00 00 00 51 45 07 00 | 11/17/88
+-------------------------------------------------+ thursday
archived 01:12:00
header_id = "NuFX" 10/22/88
header_crc = $0567 saturday
attrib_count = $003A (58 bytes in attrib section)
version = $0000
total_threads = $00000001 (one thread)
file_sys_id = $0001 (Prodos)
file_sys_info = $002f (not sparse, / = separator)
access = $00000000 (none)
filetype = $00000000 (none)
aux_type/creator_type = $00000640 (1600 blocks on device -- 3.5" disk)
storage_type = $0200 (block size = 512 bytes)
create_when = 00 0a 01 58 16 0a 00 07
mod_when = 00 10 0b 58 11 0b 00 05
archive_when = 00 0c 01 58 16 0a 00 07
filename_length = $0004
filename = "DISK"
thread_class = $0002 (data_thread)
thread_format = $0002 (compressed with ShrinkIt)
thread_kind = $0001 (disk image)
reserved = $0000
thread_eof = $00000000 (unknown size before compressing)
comp_thread_eof = $00074551 (size after compression)
NuFX Documentation 2/3/89 - Page 23 -
Directories
-----------
Directories are handled almost the same way that normal files are handled with
the exception that there will be no data in the thread which follows the entry.
A thread_record *MUST* exist to inform a utility that a directory is to be
created through the use of the proper control_thread value.
Directories do not necessarily have to precede a record which references a
directory. ie, if a record contains STUFF/MY.STUFF, the directory "STUFF"
need not exist for the extracting program to properly extract the record. The
extracting program must check to see if each of the directories referenced
exist, and if they do not exist, create them. While this method places a great
burden on the abilities of the extraction program, it avoids the anomalies
associated with the deletion of directories within an archive.
NuFX Documentation 2/3/89 - Page 24 -
Cyclic Redundancy Checks (CRC's)
--------------------------------
Many people are not aware of how to calculate a CRC, so to provide this
function, I am providing source code to a very fast routine which does the crc
calculation. The routine "makeLookup" needs to be called only once. After
this, the routine "doByte" should be called repeatedly with each new byte in
succession to generate the cumulative CRC for the block. The CRC word should
be reset to null (0000) before beginning each new CRC.
This is the same CRC calculation which is done for CRC/Xmodem, and Ymodem. The
code is easily portable to a 16-bit environment like the Apple IIgs. The only
detrimental factor with this routine is that it requires 512 bytes of main
memory to operate. If you can spare the space, this is one of the fastest
routines I know to generate a CRC-16 on a 6502-type machine.
*-------------------------------
* fast crc routine based on table lookups by
* Andy Nicholas - 03/30/88 - merlin 'c02 - easily portable to nmos 6502 also.
* easily portable into orca/m format, just snip and save.
xc turn 65c02 opcodes on
*-------------------------------
* routine to make the lookup tables
*-------------------------------
makeLookup
LDX #0 zero first page
zeroLoop STZ crclo,x zero crc lo bytes
STZ crchi,x zero crc hi bytes
INX
BNE zeroLoop
*-------------------------------
* the following is the normal bitwise computation
* tweeked a little to work in the table-maker
docrc
LDX #0 number to do crc for
fetch TXA
EOR crchi,x add byte into high
STA crchi,x of crc
LDY #8 do 8 bits
loop ASL crclo,x shift current crc-16 left
ROL crchi,x
BCC loop1
NuFX Documentation 2/3/89 - Page 25 -
* if previous high bit wasn't set, then don't add crc
* polynomial ($1021) into the cumulative crc. else add it.
LDA crchi,x add hi part of crc poly into
EOR #$10 cumulative crc hi
STA crchi,x
LDA crclo,x add lo part of crc poly into
EOR #$21 cumulative crc lo
STA crclo,x
loop1 DEY do next bit
BNE loop done? nope, loop
INX do next number in series (0-255)
BNE fetch didn't roll over, so fetch more
RTS done
crclo ds 256 space for low byte of crc table
crchi ds 256 space for high bytes of crc table
*-------------------------------
* do a crc on 1 byte/fast
* on initial entry, CRC should be initialized to 0000
* on entry, A = byte to be included in CRC
* on exit, CRC = new CRC
*-------------------------------
doByte
EOR crc+1 add byte into crc hi byte
TAX to make offset into tables
LDA crc get previous lo byte back
EOR crchi,x add it to the proper table entry
STA crc+1 save it
LDA crclo,x get new lo byte
STA crc save it back
RTS all done
crc dw 0000 cumulative crc for all data
NuFX Documentation 2/3/89 - Page 26 -
Possible Block Combinations
---------------------------
The blocks *MUST* occur in the following fashion:
Master Header block containing N entries
Header block
threads (message, control, data, or resource)
.
.
.
Next Header block (notice no second Master Header block)
threads (message, control, data, or resource)
.
.
.
Nth Header Block
threads (message, control, data, or resource)
NuFX Documentation 2/3/89 - Page 27 -
Known NuFX utility programs:
---------------------------
Name Author Description Current Ver
------------- -------------- ------------------------------------ -----------
ShrinkIt Andy Nicholas Compresses files/disks, provides 0.95
archive and file utilities.
NuList Andy Nicholas Lists contents of NuFX files for 1.1
the GBBS "Pro" (tm) BBS, online.
NuFX Documentation 2/3/89 - Page 28 -
Legal Stuff
-----------
Apple, Apple IIGS, AppleLink, GS/OS, Macintosh, and Lisa are registered
trademarks of Apple Computer, Inc.
GBBS "Pro" is a registered trademark of L&L Productions.
About the Author
----------------
I am currently a Junior attending Moravian College in Bethlehem, Pennsylvania,
majoring in Computer Science.
Any comments or suggestions you have about NuFX are more than welcome, or if
you wish to request that any of the fields be assigned your own value, or if
you would like to inform me of a NuFX utility you have written, you can contact
me at:
Paper Bag Productions
c/o Andy Nicholas
Box 435
Moravian College
Bethlehem, PA 18018
CSNET : nicholaA@moravian.edu
InterNET : nicholaA%batman.moravian.edu@relay.cs.net
nicholaA@batman.moravian.edu.csnet
ProLine : andyn@pro-sol.cts.com [619-670-5379]
AppleLink PE: ShrinkIt
I would like to thank the following people for their help and input during the
design phase of the NuFX proposal:
Jason Blochowiak, Morgan Davis, Don Elton, Dave Lyons, Jon Davidson,
Vince Cooper, Lance Taylor-Warren, Floyd Zink, Kent Dickey, John Brooks,
Doug Brandon, Todd South, Larry Hawkins, Kevin Keller