dlist = LockDosList(flags)
D0 D1
struct DosList *LockDosList(ULONG)
Locks the DOS device list in preparation to walk the list. If the list is busy then this function will not return until it is available.
This function "nests": you can call it multiple times, and then must unlock it the same number of times. The 'dlist' returned is NOT a valid entry: it is a special value.
Note that for Kickstart 1.3 compatibility, LockDosList() also calls Forbid(). This was originally intended to be removed in Kickstart 2.0 (V36) but was never done.
The pointer returned by LockDosList() is NOT an actual DosList pointer - you should use on of the other DosEntry calls to get actual pointers to DosList structures (such as
NextDosEntry()), passing the value returned by LockDosList() as the 'dlist' value.
LockDosList() and
AttemptLockDosList() are built around SignalSemaphores. LDF_READ locking allows for shared access to be performed whereas LDF_WRITE locking enforces exclusive access.
Note for file system and handler writers: you should never call this function with LDF_WRITE, since it can deadlock you (if someone has it read-locked and they're trying to send you a packet). Use
AttemptLockDosList() instead, and effectively busy-wait with delays for the list to be available. The other option is that you can spawn a Process to add the entry safely.
The name "DosList" stands for the collection of AmigaDOS devices which are stored in a list that requires special functions to access, modify or extend it with new entries:
- AttemptLockDosList ()
-
see next
- LockDosList ()
-
These functions provide exclusive access to the DosList.
You must call one of these before you may call the FindDosEntry(), NextDosEntry() and RemDosEntry() functions.
When calling AttemptLockDosList() make sure that the attempt has succeeded before you proceed!
- UnLockDosList ()
-
This is the counterpart to the AttemptLockDosList() and LockDosList() functions. Call it when you no longer need the DosList to be locked.
- MakeDosEntry ()
-
This function creates a new DosList entry, to be added with the AddDosEntry() function.
- FreeDosEntry ()
-
This function frees a DosList entry which was created by MakeDosEntry() and which must have first been removed from the DosList, if it had been added in the first place.
- AddDosEntry ()
-
This function will add a new DosEntry, provided it does not conflict with an existing entry.
- RemDosEntry ()
-
This is the counterpart to the AddDosEntry() function.
- NextDosEntry ()
-
This function will find the next DosList entry which matches the entry types you want to access.
- FindDosEntry ()
-
This function will search the DosList for an entry whose name and entry type matches your requirements.
These special functions are provided for convenience and also to insure the consistency of the DosList, e.g. devices and assigns must have unique names.
The DosList entries are represented by a common data structure known as 'struct DosList' which is defined in the <dos/dosextens.h> header file. This common data structure encapsulates three different types of entries:
-
A device, such as "df0:" or "RAM:" (DLT_DEVICE)
-
A volume, such as "Work:" (DLT_VOLUME)
-
An assign, such as "S:" or "LIBS:" (DLT_DIRECTORY, DLT_LATE or DLT_NONBINDING)
The common DosList data structure was introduced with the Native Developer Kit (NDK) for Workbench and Kickstart 1.3. Prior to that, there were three data structures which covered this ground which you might still find in legacy code. You should not use these in modern code and make use of 'struct DosList' instead.
For reference, these are the (only) data structures which were available during the Kickstart 1.1-1.2 era:
DLT_DEVICE
struct DevInfo, in <libraries/dosextens.h>
struct DeviceNode, in <libraries/filehandler.h> starting
with the Kickstart/Workbench 1.2
developer disk set
Note that both data structures are equivalent and differ only in the data structure name and the names of the individual data structure members.
DLT_VOLUME
DLT_DIRECTORY
struct DeviceList, in <libraries/dosextens.h>
Note that directly modifying the DosList contents is strongly discouraged. Use the dos.library (V36) functions which were provided for this purpose:
AssignAdd(),
AssignLate(),
AssignLock() and
AssignPath()
These are for adding and removing assigns as well as for extending them through
AssignAdd().
MakeDosEntry() and
FreeDosEntry()
These are for creating and freeing DosList entries which represent devices (DLT_DEVICE) and volumes (DLT_VOLUME).
AddDosEntry() and
RemDosEntry()
These are for adding and removing DosList entries.
If you must examine DosList entries, here follows a description of the individual entry types.
Every DosList entry has the same size as the 'struct DevInfo' structure. Depending upon the entry type, not all of the data structure will be used. For example, a DLT_DIRECTORY type entry makes use only of the dol_Next, dol_Type, dol_Task, dol_
Lock and dol_Name structure members, while a DLT_DEVICE type entry does not use the dol_
Lock entry, it makes use of the dol_Handler, dol_StackSize, dol_Priority, dol_Startup, dol_SegList and dol_GlobVec members.
- Note:
-
The following data structure descriptions omit the structure members which are not used by the respective directory entry type.
DLT_DEVICE (volume)
- dol_Next (BPTR)
-
Pointer to the next DosList entry or NULL.
- dol_Type (LONG)
-
Will be set to DLT_DEVICE.
- dol_Task (struct MsgPort *)
-
NULL if the device is not active yet and otherwise contains the address of the file system Process MsgPort.
- dol_misc.dol_handler.dol_Handler (BSTR)
-
Name of the load file (e.g. "L:FastFileSystem") if the file system must be loaded from disk in order to activate the file system. If this is NULL, then the dol_misc.dol_handler.dol_SegList must contain a pointer to the seglist of the file system.
- dol_misc.dol_handler.dol_StackSize (LONG)
-
When the file system Process is created, this specifies the size of the stack provided for it. Note that this value has a different meaning depending upon whether the dol_misc.dol_handler.dol_GlobVec value is 0 or -1 (see the documentation for dol_GlobVec for more details).
For a dol_GlobVec value of 0, the dol_StackSize must be specified as the number of 32-bit words to use, e.g. a value of 600 32-bit words yields a stack size of 4 * 600 = 2400 bytes.
For a dol_GlobVec value of -1, the dol_StackSize must be specified as the number of bytes to use.
- dol_misc.dol_handler.dol_Priority (LONG)
-
This specifies the Task priority to be used by the file system Process to be created and must be in the range -128..127.
- dol_misc.dol_handler.dol_Startup (LONG, BSTR or BPTR)
-
The dol_Startup member may contain an integer value, a BCPL pointer to a BCPL string or a BCPL pointer to a 'struct FileSysStartupMsg' (see <dos/filehandler.h). These values are mutually exclusive.
Some handlers, such as port-handler, expect a small integer value (0, 1 or 2) which selects the respective mode of operation. For Port-Handler, 0 would make it open "serial.device", 1 would open "parallel.device" and 2 would open "printer.device".
Some handlers expect a string which conveys operating or configuration parameters which are unsuitable for passing them as part of the Open() name. For example, the startup string (as specified in a mount file) may tell the file system which network file server to connect to and which authentication information to provide.
File systems which make use of mass storage devices, such as "trackdisk.device" or "scsi.device", expect a BCPL pointer to a 'struct FileSysStartupMsg' which contains details pertaining to which device driver and which unit and flags to use when opening. Also provided are the disk or partition parameters as part of the associated storage medium in the form of the 'struct DosEnvec' (see <dos/filehandler.h>).
For more information on how to deal with the dol_Startup contents, and how to find out which kind of information to find here without guessing, see the MakeDosEntry() documentation.
- dol_misc.dol_handler.dol_GlobVec (LONG)
-
This value indicates how the operating system should set up a handler or file system Process and how to provide the configuration information it should use. The value makes a difference with regard to what the dol_StackSize value should mean.
For a dol_GlobVec value of 0, the operating system will prepare for a BCPL style handler or file system Process, which receives its startup DosPacket as a parameter in a designated CPU register. The operating system supports this option for compatibility only. Do not use this.
For a dol_GlobVec value of -1, the operating system will prepare for a 'C' or assembly language style handler or file system Process, which receives its startup DosPacket via PutMsg(), sent to the Process.pr_MsgPort.
- Note:
-
The purpose and meaning of the dol_GlobVec value is more complex than this brief introduction may reveal. Please restrict yourself to the use of the values of either 0 or -1.
- dol_Name (BPTR)
-
The name of the handler or file system device as a BCPL pointer to a BCPL string. Starting with dos.library V36 (Kickstart 2.0) the BCPL string will be terminated by a NUL character. This enables it to be used as a regular 'C' string.
DLT_DIRECTORY (assign)
- dol_Next (BPTR)
-
Pointer to the next DosList entry or NULL.
- dol_Type (LONG)
-
Will be set to DLT_DIRECTORY.
- dol_Task (struct MsgPort *)
-
Address of the Process MsgPort of the file system which created the Lock stored in dol_Lock.
- dol_Lock (BPTR)
-
Lock on the directory or file which this assign is bound to.
- dol_misc.dol_assign.dol_List (struct AssignList *, V36)
-
Kickstart 2.0 introduced the option to have an assign bound not to just a single file or directory, but a list of these. Most useful as a list of directories, this is what makes it possible to have the "LIBS:" assign refer to "SYS:LIBS" and "SYS:Classes", with OpenLibrary() checking each directory in turn for the library or BOOPSI class library to be opened.
The AssignList nodes are stored in a singly-linked list, with dol_List pointing to the first node, if any:
struct AssignList {
struct AssignList *al_Next; BPTR al_
Lock;
};
- dol_Name (BPTR)
-
The name of the assign as a BCPL pointer to a BCPL string. Starting with dos.library V36 (Kickstart 2.0) the BCPL string will be terminated by a NUL character. This enables it to be used as a regular 'C' string.
DLT_VOLUME (volume)
- dol_Next (BPTR)
-
Pointer to the next DosList entry or NULL.
- dol_Type (LONG)
-
Will be set to DLT_VOLUME.
- dol_Task (struct MsgPort *)
-
Address of the Process MsgPort of the file system which this volume is associated with.
- dol_VolumeDate (struct DateStamp)
-
Date and time when the storage medium associated with this volume was initialized.
- dol_VolumeList (BPTR)
-
If the storage medium which this volume is associated with was removed or access to this medium was inhibited, the file system will store a BCPL pointer to the first FileLock associated with the volume in this member. Otherwise, it will be NULL.
- dol_DiskType (LONG)
-
A file system type signature, such as ID_DOS_DISK (see <dos/dos.h>), which identifies the type of file system which makes use of this volume. For historical reasons, most file systems will store ID_DOS_DISK here, making the contents of this data structure member an unreliable indication of the kind of file system involved.
- dol_Name (BPTR)
-
The name of the volume as a BCPL pointer to a BCPL string. Starting with dos.library V36 (Kickstart 2.0) the BCPL string will be terminated by a NUL character. This enables it to be used as a regular 'C' string.
DLT_LATE (late-binding assign, V36)
- dol_Next (BPTR)
-
Pointer to the next DosList entry or NULL.
- dol_Type (LONG)
-
Will be set to DLT_LATE.
- dol_misc.dol_assign.dol_AssignName (TEXT *)
-
Path name which the assign should be bound to when
it is first accessed. If the access attempt is successful, the late-binding assign will be transformed into the standard DLT_DIRECTORY type form.
- dol_Name (BPTR)
-
The name of the assign as a BCPL pointer to a BCPL string. Starting with dos.library V36 (Kickstart 2.0) the BCPL string will be terminated by a NUL character. This enables it to be used as a regular 'C' string.
DLT_NONBINDING (non-binding assign, V36)
- dol_Next (BPTR)
-
Pointer to the next DosList entry or NULL.
- dol_Type (LONG)
-
Will be set to DLT_NONBINDING.
- dol_misc.dol_assign.dol_AssignName (TEXT *)
-
Path name which the assign should reference whenever it is accessed. When accessed, dos.library will attempt to obtain a shared Lock on the path name and then resolve the path which follows the assign name relative to that Lock. The Lock will be automatically unlocked when it is no longer needed.
- dol_Name (BPTR)
-
The name of the assign as a BCPL pointer to a BCPL string. Starting with dos.library V36 (Kickstart 2.0) the BCPL string will be terminated by a NUL character. This enables it to be used as a regular 'C' string.
As an example, here's how you can search for all volumes of a
specific name and do something with them:
2.0 way:
dl = LockDosList(LDF_VOLUMES|LDF_READ);
while ((dl =
FindDosEntry(dl,name,LDF_VOLUMES)) != NULL)
{
Add to list of volumes to process or break out of
the while loop.
(You could try using it here, but I advise
against it for compatibility reasons if you
are planning on continuing to examine the list.)
}
process list of volumes saved above, or current entry if
you're only interested in the first one of that name.
UnLockDosList(LDF_VOLUMES|LDF_READ);
/* must not use dl after this! */
1.3/2.0 way:
if (version >= 36)
dl = LockDosList(LDF_VOLUMES|LDF_READ);
else {
Forbid();
/* tricky! note dol_Next is at offset 0! */
dl = (struct DosList *)&(
(struct DosInfo *)BADDR(DOSBase->dl_Root.rn_
Info)
)->di_DeviceList);
}
while (version >= 36 ?
dl =
FindDosEntry(dl,name,LDF_VOLUMES) :
dl = yourfindentry(dl,name,DLT_VOLUME))
{
Add to list of volumes to process, or break out of
the while loop.
Do NOT lock foo1/foo2 here if you will continue
to examine the list - it breaks the forbid
and the list may change on you.
}
process list of volumes saved above, or current entry if
you're only interested in the first one of that name.
if (version >= 36)
UnLockDosList(LDF_VOLUMES|LDF_READ);
else
/* must not use dl after this! */
...
struct DosList *
yourfindentry (struct DosList *dl, STRPTRname, type)
{
/* tricky - depends on dol_Next being at offset 0,
and the initial ptr being a ptr to di_DeviceList! */
while ((dl = dl->dol_Next) != NULL)
{
if (dl->dol_Type == type &&
Stricmp(name, (STRRPTR)(BADDR(dl->dol_Name))+1) == 0)
{
}
}
return dl;
}