
Home / Autodocs / dos.library
NAME
- ExAll
-
Examine an entire directory (V36)
SYNOPSIS
continue = ExAll(lock, buffer, size, type, control)
D0 D1 D2 D3 D4 D5
BOOL ExAll(BPTR,STRPTR,LONG,LONG,struct ExAllControl *)
FUNCTION
Examines an entire directory, returning information on all of its directory entries in a compact manner, possibly more efficiently than the
Examine() ..
ExNext() loop could.
'lock' must be on a directory. 'size' is the size of the buffer supplied. The 'buffer' will be filled with (partial) ExAllData structures, as specified by the 'type' argument.
'type' is a value from those shown below that determines which information is to be stored in the 'buffer'. Each higher value adds a new thing to the list as described in the table below:
- ED_NAME
-
File name
- ED_TYPE
-
Type of directory entry
- ED_SIZE
-
Size in bytes
- ED_PROTECTION
-
Protection bits
- ED_DATE
-
3 longwords of date (= sizeof(struct DateStamp))
- ED_COMMENT
-
Comment (will be NULL if no comment)
- ED_OWNER
-
owner user-id and group-id (if supported) (V39)
Thus, ED_NAME gives only filenames, and ED_OWNER gives everything.
- NOTE:
-
V37 dos.library, when doing ExAll() emulation, and V37 file systems will return an error if passed type=ED_OWNER. If ExAll() fails and IoErr() returns ERROR_BAD_NUMBER, retry with type=ED_COMMENT to get everything but owner info.
All file systems supporting ExAll() must support type parameters through ED_COMMENT, and must check type and then stop & return ERROR_BAD_NUMBER if they do not support the type.
The ExAllData->ead_Next member gives a pointer to the next entry in the buffer. The final entry will have NULL in ead_Next.
The 'control' argument is required so that the file system can keep track if more than one call to ExAll() is required. This happens when there are more names in a directory than will fit into the buffer.
- NOTE:
-
The control structure (ExAllControl) MUST be allocated by AllocDosObject(DOS_EXALLCONTROL, NULL) !!!
The format of the control structure is as follows:
- eac_Entries
-
This member tells the calling application how many entries are in the buffer after calling ExAll().
- Note:
-
make sure your code handles the 0 entries case, including 0 entries with continue non-zero.
- eac_LastKey
-
This member ABSOLUTELY MUST be initialised to 0 before calling ExAll() for the first time. Any other value will lead to undefined behaviour and will cause nasty things to happen. If ExAll() returns non-zero, then this member should not be touched before making the second and subsequent calls to ExAll(). Whenever ExAll() returns non-zero, there are more calls required before all names have been received.
As soon as a FALSE return is received, then ExAll() has completed. If IoErr() returns ERROR_NO_MORE_ENTRIES, and otherwise, it returns the error that occurred, similar to the ExNext() function.
- eac_MatchString
-
If this member is NULL then all file names will be returned. If this member is non-NULL then it is interpreted as a pointer to a string that is used to pattern-match all file names before accepting them and putting them into the buffer. The default AmigaDOS caseless pattern match function is used. This string MUST have been parsed by ParsePatternNoCase() before you call ExAll() for the first time!
- eac_MatchFunc
-
This member contains a pointer to a Hook for a function to decide if the entry will be included in the returned list of entries. The entry is filled out first, and then passed to the Hook. If no eac_MatchFunc is to be called, then this entry should be NULL. The Hook is called with the following parameters (as is standard for Hooks):
accept = MatchFunc( hookptr, exalldata, typeptr )
D0 A0 A1 A2
(A0 = pointer to 'struct Hook', A1 = pointer to filled-in
'struct ExAllData', A2 = pointer to LONG of type).
MatchFunc() should return FALSE if the entry is not to be accepted, otherwise return TRUE.
Note that dos.library will emulate ExAll() using
Examine() and
ExNext() if the file system in question does not support the features which enable ExAll() to be used.
INPUTS
- lock
-
Lock on directory to be examined. NOTE: Not all file systems will accept a NULL Lock here.
- buffer
-
Buffer for data returned (MUST be at least word-aligned, preferably long-word aligned).
- size
-
Size in bytes of 'buffer'.
- type
-
Type of data to be returned.
- control
-
Control data structure (see notes above). MUST have been allocated by AllocDosObject(DOS_EXALLCONTROL, NULL)!
RESULT
- continue
-
Whether ExAll() is done. If FALSE is returned, either ExAll() has completed (with the IoErr() function returning ERROR_NO_MORE_ENTRIES), or an error occurred (check IoErr()). If non-zero is returned, you MUST keep calling ExAll() again until it returns FALSE.
WARNING
It is never safe to use ExAll() with a NULL
Lock!
If you call ExAll() once, you should either continue calling it until it returns FALSE, or (requires Kickstart 3.0 or better), call
ExAllEnd() in order to bring the scanning operation to a conclusion. Some file systems, e.g. network file systems, may still be busy processing directory records when you stop calling ExAll().
ExAllEnd() could be the fastest way to make them stop, releasing any resources currently tied up with preparing directory records which will not be needed. In any case, by the time you call
UnLock() on the
Lock you passed to ExAll() the file system should be aware that directory scanning has come to an end.
Filling the buffer with ExAllData records will take its time, and the larger the buffer, the more time the file system will spend on filling it. This may result in the data provided being stale by the time the ExAll() function returns. Directory entries may have been removed, replaced or changed by the time you are ready to process them. This is certainly true for
Examine()/
ExNext() style directory scanning, too, but the much larger buffer size used by ExAll() makes it much more likely for the ExAllData records to go out of sync with the directory contents. Be prepared to handle this.
NOTES
The use of ExAll() is discouraged, both from the file system implementor's point of view as well as from the ExAll() user's point of view. The promise of ExAll()'s advantages over the use of
Examine()/
ExNext() must be weighed against the challenge of using ExAll() correctly and certainly of the challenge of implementing it correctly in a file system, assuming that it is implemented correctly to begin with. Support for ExAll() in file systems used to be rare and tended to be poorly-tested.
As experience shows, file system implementors struggle with correctly handling all the corner cases, leading to unpredictable behaviour on both the file system side and the ExAll() user's side. This is especially true for the Amiga operating system itself which never shipped with fully feature-complete and consistent ExAll() support in all its file systems until AmigaOS 3.2.2 (2023). Similar challenges exist for the ExAll() user's side in following the complex protocol documented in the example code (see section EXAMPLE below). There is very little room for error and very little to gain by using ExAll(), compared to the use of the
Examine() ..
ExNext() loop.
If you have a choice, try the
Examine() ..
ExNext() approach first before you decide to give ExAll() a chance. ExAll() is the perhaps most complex dos.library function and the complexity of using it correctly and successfully require particularly careful design, implementation and verification. You may want to invest this kind of effort elsewhere.
The advantage of ExAll() is in that most of the work being done happens inside the file system Process itself. Unlike calling
Examine() and then
ExNext() over and over again, ExAll() does not have to incur a DosPacket exchange delay for every directory entry. That said, the file system still has to produce the directory entry records by following the metadata structures on the storage medium, which, e.g. for the FFS, still involves plenty of individual disk blocks to be read and processed. The DirCache mode of the FFS features an optimized directory layout which lends itself to filling the ExAll() buffer very efficiently and quickly. However, few file systems are able to provide directory records quite so efficiently and may suffer from other drawbacks, which is certainly the case for the DirCache mode of the FFS.
ExAll() will fall back onto emulating the directory scanning and filtering if the file system does not feature built-in support for it, using
Examine() ..
ExNext(). Worst case, the file system will only be marginally faster in filling the ExAllData buffer than your own
Examine() ..
ExNext() calls. Part of the complexity of what the ExAll() function accomplishes is in dealing with the emulation and what it entails. You may want to avoid this.
EXAMPLE
eac =
AllocDosObject(DOS_EXALLCONTROL, NULL);
if (eac == NULL) ...
...
eac->eac_LastKey = 0;
do {
more = ExAll(lock, EAData, sizeof(EAData), ED_FOO, eac);
if (more == FALSE &&
IoErr() != ERROR_NO_MORE_ENTRIES) {
/* ExAll() failed abnormally */
break;
}
if (eac->eac_Entries == 0) {
/* ExAll() failed normally with no entries */
continue; /* ("more" is *usually* FALSE) */
}
ead = (struct ExAllData *)EAData;
do {
/* use ead here */
...
/* get next ead */
ead = ead->ed_Next;
} while (ead);
} while (more);
...
FreeDosObject(DOS_EXALLCONTROL,eac);
BUGS
In V36, there were problems with ExAll (particularily with eac_MatchString, and ed_Next with the ram-disk and the emulation of it in dos.library for handlers that do not support the packet. It is advised you only use this under V37 and later.
The V39 DirCache variant of the FFS had a bug in ExAll that required to first
Examine() the
Lock before passing it to ExAll. This was fixed in V40, with a patch in SetPatch.
The V40 version passed an invalid argument to the dos.library error report handler in case of trouble, potentially causing crashes or other mischiefs. This was fixed in V47, with a patch in SetPatch V45.
See note above regarding extensions to the Type parameter.
SEE ALSO