Three code examples are provided which show how to create a DLT_DEVICE
type DosList entry such as may be used for the "PRT:" device using
the "L:Port-Handler", for a simple device which expects a string
in the dol_Startup member and for a "RAD:" file system which makes
use of the ramdrive.device storage device.
The fourth example shows how you would figure out if the
DosList->dol_misc.dol_handler.dol_Startup member refers to a
BCPL string, a BCPL pointer to a 'struct FileSysStartupMsg' or
none of these two.
1. "PRT:" device and "L:Port-Handler"
This is an example of a handler which uses the dos.library V34
BCPL legacy interface which requires that the dol_GlobVec
member is filled in. Do not use this, it is just to show how
it would be done. Note that dol_Startup is set to the number 2,
which would tell Port-Handler to open "printer.device"; the value
0 would have it open "serial.device" and the value 1 would have
led to "parallel.device" being opened.
---------------------------------------------------------------------
#include <dos/dosextens.h>
#include <string.h>
extern struct DosLibrary * DOSBase;
const TEXT handler_name[] = "L:Port-Handler";
UBYTE * bcpl_handler_name = NULL;
struct DosList * dol = NULL;
LONG success = FALSE;
int len;
/* Allocate memory for a NUL-terminated BCPL string.
* CAUTION: Must not be longer than 254 characters!
*/
len = strlen(handler_name);
if (len <= 254)
bcpl_handler_name = AllocVec(1 + len + 1, MEMF_ANY|MEMF_PUBLIC);
if (bcpl_handler_name != NULL)
{
/* Note: BCPL string length must include the NUL-termination
* for compatibility with expansion.library/MakeDosNode.
*/
bcpl_handler_name[0] = len + 1;
strcpy(&bcpl_handler_name[1], handler_name);
dol = MakeDosEntry("PRT", DLT_DEVICE);
if (dol != NULL)
{
dol->dol_misc.dol_handler.dol_Handler = MKBADDR(bcpl_handler_name);
dol->dol_misc.dol_handler.dol_StackSize = 1024;
dol->dol_misc.dol_handler.dol_Startup = 2;
dol->dol_misc.dol_handler.dol_GlobVec = MKBADDR(DOSBase->dl_GV);
success =
AddDosEntry(dol);
}
}
if (success == FALSE)
{
FreeDosEntry(dol);
FreeVec(bcpl_handler_name);
}
---------------------------------------------------------------------
2. Simple "EXAMPLE:" device with startup string
This is a very simple example which shows how you set up and
install a NUL-terminated BCPL string in the dol_Startup member,
if it is needed. This is not a complete example of how a
handler would be set up.
---------------------------------------------------------------------
#include <dos/dosextens.h>
#include <string.h>
const TEXT startup[] = "quiet";
UBYTE * bcpl_startup = NULL;
struct DosList * dol = NULL;
LONG success = FALSE;
int len;
/* Allocate memory for a NUL-terminated BCPL string.
* CAUTION: Must not be longer than 254 characters!
*/
len = strlen(startup);
if (len <= 254)
bcpl_startup = AllocVec(1 + len + 1, MEMF_ANY|MEMF_PUBLIC);
if (bcpl_startup != NULL)
{
/* Note: BCPL string length must include the NUL-termination
* for compatibility with expansion.library/MakeDosNode.
*/
bcpl_startup[0] = len + 1;
strcpy(&bcpl_startup[1], startup);
dol = MakeDosEntry("EXAMPLE", DLT_DEVICE);
if (dol != NULL)
{
/* Note that a real handler or file system would set up
* more of the DosList, e.g. the name of the handler on
* disk or the seglist already in memory.
*/
dol->dol_misc.dol_handler.dol_Startup = MKBADDR(bcpl_startup);
success =
AddDosEntry(dol);
}
}
if (success == FALSE)
{
FreeDosEntry(dol);
FreeVec(bcpl_startup);
}
---------------------------------------------------------------------
3. "RAD:" file system which makes use of the ramdrive.device
This example shows in detail how you might mount a file system
which uses a mass storage device driver after dos.library has
already booted the system. This is different from how you would
set up a partition as part of an Autoboot hard disk controller.
In that case, you would use the expansion.library functions
MakeDosNode() and AddBootNode(). MakeDosNode() performs all the
steps shown in the following example, save for the default file
system seglist access, because dos.library is not operational at
the time an Autoboot hard disk controller will set up the
partition information.
Please note that this setup procedure which fills in dol_GlobVec=0
and then initializes dol_SegList from the DOSBase default file system
segment list has a very limited use. It basically works only for the
default file system. For real use, you would adopt an entry found in
the FileSystem.resource list which matches the DOS Type you expect.
This is the same approach which an Autoboot hard disk controller
makes use of.
Finally, how do you activate a file system which has been mounted?
Call
DeviceProc(), which will look up the DosList entry for the
file system device and start the file system Process, if this has
not been done yet. In the example below, you would accomplish this
by calling
DeviceProc("RAD:"), after having successfully added the
new DosList entry.
---------------------------------------------------------------------
#include <dos/dosextens.h>
#include <dos/filehandler.h>
#include <string.h>
extern struct DosLibrary * DOSBase;
const TEXT device_name[] = "ramdrive.device";
const LONG unit_number = 0;
const LONG opendevice_flags = 0;
UBYTE * bcpl_device_name = NULL;
struct DosEnvec * de;
struct FileSysStartupMsg * fssm;
struct DosList * dol;
LONG success = FALSE;
int len;
/* Allocate memory for a NUL-terminated BCPL string.
* CAUTION: Must not be longer than 254 characters!
*/
len = strlen(device_name);
if (len <= 254)
bcpl_device_name = AllocVec(1 + len + 1, MEMF_ANY|MEMF_PUBLIC);
/* Allocate memory for the DOS environment vector, which is a
* variable length table in which each slot contains a specific
* parameter which describes the layout, features and other
* properties of a mass storage device.
*
* In this example, we allocate memory for a table which
* contains tables entries DE_TABLESIZE..DE_DOSTYPE. See
* <dos/filehandler.h> for a full description.
*/
de = AllocVec((1+DE_DOSTYPE) * sizeof(LONG),
MEMF_ANY|MEMF_PUBLIC|MEMF_CLEAR);
/* Allocate memory for the FileSysStartupMsg structure which
* ties the device name, the unit number, the OpenDevice() flags
* needed to open that device unit and the DOS environment
* vector together. In spite of its name, the FileSysStartupMsg
* structure is not sent as a message or DosPacket. Its address
* will go into the DosList->dol_misc.dol_handler.dol_Startup member.
*/
fssm = AllocVec(sizeof(*fssm), MEMF_ANY|MEMF_PUBLIC|MEMF_CLEAR);
if (bcpl_device_name != NULL && de != NULL && fssm != NULL)
{
/* Initialize the device name, which must be a NUL-terminated
* BCPL string. The NUL-termination must be part of the
* string length for compatibility with
* expansion.library/MakeDosNode.
*/
bcpl_device_name[0] = len + 1;
strcpy(&bcpl_device_name[1], device_name);
/* Fill in the FileSysStartupMsg, which provides the device
* name, the device unit number and the OpenDevice() flags,
* as well as the DOS environment vector.
*/
fssm->fssm_Device = MKBADDR(bcpl_device_name);
fssm->fssm_Unit = unit_number;
fssm->fssm_Flags = opendevice_flags;
fssm->fssm_Environ = MKBADDR(de);
/* Fill in the parameters for an Amiga floppy disk drive.
* This would describe an 80 track disk drive, with 11
* sectors per track and two heads. A sector is 512 bytes
* in size, and each block consists of 1 sector only.
*
* This example assumes that all DosEnvec fields are
* set to 0 which are not filled in by the following code.
*/
de->de_TableSize = DE_DOSTYPE;
de->de_SizeBlock = 512 / sizeof(LONG);
de->de_Surfaces = 2;
de->de_SectorPerBlock = 1;
de->de_BlocksPerTrack = 11;
de->de_Reserved = 2;
de->de_LowCyl = 0;
de->de_HighCyl = 80;
de->de_NumBuffers = 5;
de->de_BufMemType = MEMF_ANY|MEMF_PUBLIC;
de->de_MaxTransfer = 0x200000;
de->de_Mask = 0x7FFFFFFE;
de->de_BootPri = -128;
de->de_DosType = ID_DOS_DISK;
/* Finally, create the device DosList, which will have to
* be filled in, before it can be used.
*/
dol = MakeDosEntry("RAD", DLT_DEVICE);
if (dol != NULL)
{
/* The following steps configure the "RAD:" device similar
* to how "df0:" is set up. This means that it will use
* the default Amiga ROM file system using 2400 bytes of
* stack and the BCPL legacy file system interface.
*/
dol->dol_misc.dol_handler.dol_StackSize = 2400 / sizeof(LONG);
dol->dol_misc.dol_handler.dol_Priority = 10;
/* Note: You should fill in the seglist or provide the name of the
* handler to load from disk. Otherwise, dos.library will
* assign the default ROM file system to your device which
* only works if you also set the de_DosType to one of the
* ID_DOS_DISK, etc. types.
*
* In this example we deliberately assign the default
* ROM file system to the device dol_SegList member,
* by picking its seglist up through the DOSLibrary base.
* Just to show how it could be done if you also use
* a dol_GlobVec value of 0.
*/
dol->dol_misc.dol_handler.dol_SegList =
((struct DosLibrary *)DOSBase)->dl_Root->rn_FileHandlerSegment;
/* Setting dol_GlobVec to 0 selects the BCPL legacy file
* system interface, which also says that the dol_StackSize
* value must be given as the number of 32 bit words, not
* as the number of bytes.
*/
dol->dol_misc.dol_handler.dol_GlobVec = 0;
/* Finally, make this file system ready to start
* once its device node has been added successfully.
*/
dol->dol_misc.dol_handler.dol_Startup = MKBADDR(fssm);
success =
AddDosEntry(dol);
}
}
if (success == FALSE)
{
FreeDosEntry(dol);
FreeVec(de);
FreeVec(fssm);
FreeVec(bcpl_device_name);
}
---------------------------------------------------------------------
4. How to tell what DosList->dol_misc.dol_handler.dol_Startup means
and why this is relevant in the first place
A handler or file system may need to know how and by which means it
should do its job. For example, the Amiga ROM default file system
can be used with a variety of mass storage device drivers which
implement the same command set: the file system just needs to know
which device driver, device unit and partition configuration to use.
The Port-Handler needs to know which specific device driver should
be accessed when you open the SER:, PAR: or PRT: AmigaDOS devices.
This kind of information is provided through the DosList
dol_misc.dol_handler.dol_Startup member:
- A "simple" handler such as the original Workbench 1.1 Port-Handler
expects to find an integer value in dol_Startup, which must be
one of 0, 1 or 2 (for serial.device, parallel.device or
printer.device).
- A handler may also expect to find a pointer to a BCPL string in
- A file system which makes use of a mass storage device driver will
expect to find a pointer to a FileSysStartupMsg in dol_Startup,
which in turn references further information, such as the name of
the device driver and the partition parameters.
Regardless of what information is provided for the file system or
handler, it will need to be validated before it can be put to use.
This is the main reason why you should be able to identify the
kind and format of information expected. This knowledge is useful
for debugging and diagnostics, too.
Identifying an integer value stored in dol_Startup is comparatively
easy, but a BCPL pointer to a BCPL string or to a FileSysStartupMsg
is harder to identify reliably as what it is. The following example
code attempts to break down the value into three categories, with
"unknown" (dol_startup_unknown) as the fourth option if no other
category makes sense. Please note that if dol_startup_bstr is returned
you should inspect what appears to be a BCPL string and validate
its contents. For example, check if the string length is > 0 and
if all the characters of the string make sense.
---------------------------------------------------------------------
#include <dos/dosextens.h>
#include <dos/filehandler.h>
/* This is a simple test to check if a BPTR may point to a valid
* RAM address. The test involves looking at the two most
* significant bits of the BPTR, which must both be 0.
*/
#define IS_VALID_BPTR_ADDRESS(ptr) (((ptr) & 0xC0000000) == 0)
/* Which kind of information may be stored in the DosList
* dol->dol_misc.dol_handler.dol_Startup member? Note that the
* dol_startup_bstr type may require additional scrutiny.
*/
enum dol_startup_t
{
dol_startup_unknown, /* Uncertain or not known */
dol_startup_integer, /* A negative or positive integer */
dol_startup_bstr, /* BCPL pointer to BCPL string */
dol_startup_fssm, /* BCPL pointer to FileSysStartupMsg */
};
enum dol_startup_t
get_dol_startup_type(const struct DosList * dol)
{
LONG startup;
const void * address;
const struct FileSysStartupMsg * fssm;
const UBYTE * device_name;
const struct DosEnvec * de;
/* The DosList entry address should be a valid RAM address. */
if (dol == NULL || TypeOfMem(dol) == 0)
return dol_startup_unknown;
startup = (LONG)dol->dol_misc.dol_handler.dol_Startup;
/* Is this a small integer? A BCPL pointer can never
* be mistaken for a negative integer because its two
* most significant bits will always be 0.
*/
if (startup <= 2)
return dol_startup_integer;
/* Could this be a BCPL pointer? */
if (IS_VALID_BPTR_ADDRESS(startup) == FALSE)
return dol_startup_unknown; /* No, it is not. */
/* This data structure was allocated from RAM, and
* its address should be located in a known RAM
* address range.
*/
address = BADDR(startup);
if (TypeOfMem(address) == 0)
return dol_startup_unknown;
/* If this is a pointer to a FileSysStartupMsg, then the
* fssm_Device and fssm_Environ members should contain valid
* RAM addresses.
*/
fssm = address;
if (IS_VALID_BPTR_ADDRESS(fssm->fssm_Device) &&
IS_VALID_BPTR_ADDRESS(fssm->fssm_Environ))
{
device_name = BADDR(fssm->fssm_Device);
de = BADDR(fssm->fssm_Environ);
if (TypeOfMem(device_name) != 0 && TypeOfMem(de) != 0)
{
/* The environment table size should be > 0. The
* minimum size is 11, which covers all table
* entries up to and including DE_NUMBUFFERS.
*/
if (de->de_TableSize >= DE_NUMBUFFERS)
}
}
/* This may be a BPTR to a BCPL string, but further
* validation may be needed.
*/
return dol_startup_bstr;
}
---------------------------------------------------------------------