mirror of
https://github.com/alfishe/amiga-bootcamp.git
synced 2026-06-13 00:26:28 +00:00
Phase 1: enrich 07_dos and 10_devices (highest FPGA priority)
07_dos: - file_io.md: 108→240+ lines — buffered I/O (FRead/FWrite/SetVBuf), access mode comparison, FileHandle struct with offsets, standard handles, Printf %ld warning, FileInfoBlock, practical patterns (copy file, get size, load to RAM), error code table - filesystem.md: 91→270+ lines — full disk geometry (ADF/HDF), all 8 DOS\x filesystem IDs, root block byte-level layout, file header layout with reverse-order pointer quirk, OFS vs FFS data blocks with efficiency numbers, bitmap blocks, extension blocks, checksum algorithm, Python ADF reader - locks_examine.md: 113→270+ lines — lock semantics diagram, FileLock struct with handler discovery, ExAll bulk scan, practical patterns (atomic write, path resolution, volume info), 4 antipatterns (leaked locks, exclusive too long, unchecked IoErr, DupLock), pattern matching 10_devices: - audio.md: 73→240+ lines — hardware architecture diagram, channel registers with offsets, period/frequency table, priority allocation, double-buffering, audio interrupts, AM/PM modulation, direct HW - timer.md: 80→230+ lines — CIA timer hardware, all 5 units with decision flowchart, non-blocking delays, signal-based waiting, time arithmetic, ReadEClock, periodic game loop pattern, pitfalls - trackdisk.md: 82→210+ lines — MFM encoding, track format, disk geometry, read/write/motor, change notification, track caching, direct hardware access, FPGA timing implications - keyboard.md: 58→220+ lines — CIA-A serial handshake protocol with sequence diagram, bit rotation quirk, complete key code map, key matrix bitmap, reset sequence, FPGA notes
This commit is contained in:
parent
aeaea88d75
commit
da9e7d3b63
7 changed files with 1599 additions and 301 deletions
|
|
@ -1,126 +1,382 @@
|
|||
[← Home](../README.md) · [AmigaDOS](README.md)
|
||||
|
||||
# Locks and Examine — Lock, UnLock, Examine, ExNext, ExAll
|
||||
# Locks, Examine, and Directory Scanning
|
||||
|
||||
## Overview
|
||||
|
||||
AmigaDOS uses **locks** to reference files and directories. A lock is a BPTR to a `FileLock` structure. Locks provide exclusive or shared access and are used for directory scanning, attribute reading, and path resolution.
|
||||
A **Lock** is AmigaDOS's equivalent of a Unix file descriptor for directory operations. It represents a reference to a filesystem object (file or directory) and is the primary mechanism for navigating the filesystem, examining metadata, and preventing concurrent modifications. Understanding lock semantics — especially the shared/exclusive model — is critical for writing robust AmigaOS applications.
|
||||
|
||||
---
|
||||
|
||||
## Lock Types
|
||||
|
||||
```c
|
||||
/* dos/dos.h */
|
||||
#define SHARED_LOCK -2 /* read-only; multiple readers allowed */
|
||||
#define ACCESS_READ SHARED_LOCK
|
||||
#define EXCLUSIVE_LOCK -1 /* read/write; only one holder */
|
||||
#define ACCESS_WRITE EXCLUSIVE_LOCK
|
||||
```
|
||||
|
||||
---
|
||||
|
||||
## Core Functions
|
||||
|
||||
| LVO | Function | Registers | Returns |
|
||||
| Mode | Constant | Dec | Meaning |
|
||||
|---|---|---|---|
|
||||
| −84 | `Lock(name, mode)` | D1=name, D2=mode | D0=lock BPTR (0=fail) |
|
||||
| −90 | `UnLock(lock)` | D1=lock | — |
|
||||
| −96 | `DupLock(lock)` | D1=lock | D0=new lock |
|
||||
| −102 | `Examine(lock, fib)` | D1=lock, D2=fib | D0=BOOL |
|
||||
| −108 | `ExNext(lock, fib)` | D1=lock, D2=fib | D0=BOOL |
|
||||
| −78 | `CurrentDir(lock)` | D1=lock | D0=old lock |
|
||||
| −654 | `ExAll(lock, buf, size, type, ctrl)` | D1–D5 | D0=BOOL |
|
||||
|
||||
---
|
||||
|
||||
## struct FileInfoBlock
|
||||
| Shared | `SHARED_LOCK` / `ACCESS_READ` | −2 | Multiple readers allowed simultaneously |
|
||||
| Exclusive | `EXCLUSIVE_LOCK` / `ACCESS_WRITE` | −1 | Only one holder; blocks all other locks |
|
||||
|
||||
```c
|
||||
/* dos/dos.h — NDK39 */
|
||||
struct FileInfoBlock {
|
||||
LONG fib_DiskKey; /* handler-private key */
|
||||
LONG fib_DirEntryType; /* >0 = directory, <0 = file */
|
||||
char fib_FileName[108]; /* null-terminated name */
|
||||
LONG fib_Protection; /* rwed bits */
|
||||
LONG fib_EntryType; /* same as DirEntryType */
|
||||
LONG fib_Size; /* file size in bytes */
|
||||
LONG fib_NumBlocks; /* blocks used */
|
||||
struct DateStamp fib_Date; /* modification date */
|
||||
char fib_Comment[80]; /* file comment string */
|
||||
UWORD fib_OwnerUID;
|
||||
UWORD fib_OwnerGID;
|
||||
char fib_Reserved[32];
|
||||
};
|
||||
```
|
||||
/* Obtain a shared lock: */
|
||||
BPTR lock = Lock("SYS:Libs", SHARED_LOCK);
|
||||
|
||||
> [!IMPORTANT]
|
||||
> `FileInfoBlock` must be longword-aligned. Use `AllocDosObject(DOS_FIB, NULL)` on OS 2.0+ or `AllocMem(sizeof(struct FileInfoBlock), MEMF_PUBLIC)`.
|
||||
/* Obtain an exclusive lock: */
|
||||
BPTR lock = Lock("RAM:temp.dat", EXCLUSIVE_LOCK);
|
||||
|
||||
---
|
||||
|
||||
## Protection Bits
|
||||
|
||||
```c
|
||||
/* dos/dos.h */
|
||||
#define FIBF_SCRIPT (1<<6) /* s — script (executable script) */
|
||||
#define FIBF_PURE (1<<5) /* p — pure (re-entrant) */
|
||||
#define FIBF_ARCHIVE (1<<4) /* a — archived */
|
||||
#define FIBF_READ (1<<3) /* r — readable (0=allowed, 1=denied!) */
|
||||
#define FIBF_WRITE (1<<2) /* w — writable */
|
||||
#define FIBF_EXECUTE (1<<1) /* e — executable */
|
||||
#define FIBF_DELETE (1<<0) /* d — deletable */
|
||||
```
|
||||
|
||||
> [!WARNING]
|
||||
> Amiga protection bits are **inverted** from Unix: bit SET means access is **denied**.
|
||||
|
||||
---
|
||||
|
||||
## Directory Scanning
|
||||
|
||||
```c
|
||||
BPTR lock = Lock("SYS:", SHARED_LOCK);
|
||||
struct FileInfoBlock *fib = AllocDosObject(DOS_FIB, NULL);
|
||||
|
||||
if (Examine(lock, fib)) { /* read dir's own info */
|
||||
while (ExNext(lock, fib)) { /* iterate entries */
|
||||
Printf("%-30s %8ld %s\n",
|
||||
fib->fib_FileName,
|
||||
fib->fib_Size,
|
||||
fib->fib_DirEntryType > 0 ? "(dir)" : "");
|
||||
}
|
||||
/* ExNext returns FALSE when done; IoErr() == ERROR_NO_MORE_ENTRIES */
|
||||
}
|
||||
|
||||
FreeDosObject(DOS_FIB, fib);
|
||||
/* Always unlock when done: */
|
||||
UnLock(lock);
|
||||
```
|
||||
|
||||
### Lock Semantics
|
||||
|
||||
```mermaid
|
||||
graph TD
|
||||
subgraph "Shared Lock Active"
|
||||
S1["Process A: SHARED"] --> OK1["Process B: SHARED → OK"]
|
||||
S1 --> FAIL1["Process C: EXCLUSIVE → FAILS"]
|
||||
end
|
||||
|
||||
subgraph "Exclusive Lock Active"
|
||||
E1["Process A: EXCLUSIVE"] --> FAIL2["Process B: SHARED → FAILS"]
|
||||
E1 --> FAIL3["Process C: EXCLUSIVE → FAILS"]
|
||||
end
|
||||
```
|
||||
|
||||
| Scenario | Result | Error Code |
|
||||
|---|---|---|
|
||||
| Shared + Shared | Both succeed | — |
|
||||
| Shared + Exclusive | Exclusive fails | `ERROR_OBJECT_IN_USE` (202) |
|
||||
| Exclusive + Shared | Shared fails | `ERROR_OBJECT_IN_USE` (202) |
|
||||
| Exclusive + Exclusive | Second fails | `ERROR_OBJECT_IN_USE` (202) |
|
||||
|
||||
> [!IMPORTANT]
|
||||
> `Lock()` is **non-blocking** — it returns immediately with 0 and sets `IoErr()` if the lock cannot be obtained. There is no "wait for lock" mechanism in AmigaDOS. You must poll or retry.
|
||||
|
||||
---
|
||||
|
||||
## ExAll (OS 2.0+) — Bulk Scan
|
||||
## Lock Internals — struct FileLock
|
||||
|
||||
```c
|
||||
/* dos/dosextens.h */
|
||||
struct FileLock {
|
||||
BPTR fl_Link; /* BPTR to next lock (handler's list) */
|
||||
LONG fl_Key; /* block number on disk (handler-specific) */
|
||||
LONG fl_Access; /* SHARED_LOCK or EXCLUSIVE_LOCK */
|
||||
struct MsgPort *fl_Task; /* handler's MsgPort — THIS IS HOW YOU
|
||||
FIND THE FILE HANDLER */
|
||||
BPTR fl_Volume; /* BPTR to DosList volume node */
|
||||
};
|
||||
```
|
||||
|
||||
### Finding a File's Handler Process
|
||||
|
||||
This is a common need — e.g., to send custom packets:
|
||||
|
||||
```c
|
||||
/* Get the handler MsgPort from any lock: */
|
||||
struct FileLock *fl = (struct FileLock *)BADDR(lock);
|
||||
struct MsgPort *handler = fl->fl_Task;
|
||||
/* Now you can PutMsg(handler, &packet) */
|
||||
|
||||
/* From a filename (without an existing lock): */
|
||||
struct DevProc *dp = GetDeviceProc("DF0:myfile", NULL);
|
||||
if (dp) {
|
||||
struct MsgPort *handler = dp->dvp_Port;
|
||||
/* ... send packets ... */
|
||||
FreeDeviceProc(dp);
|
||||
}
|
||||
|
||||
/* From a file handle: */
|
||||
struct FileHandle *fhp = (struct FileHandle *)BADDR(filehandle);
|
||||
struct MsgPort *handler = fhp->fh_Type;
|
||||
```
|
||||
|
||||
---
|
||||
|
||||
## Examining Files and Directories
|
||||
|
||||
### Examine a Single Object
|
||||
|
||||
```c
|
||||
struct FileInfoBlock *fib = AllocDosObject(DOS_FIB, NULL);
|
||||
BPTR lock = Lock("SYS:C/Dir", SHARED_LOCK);
|
||||
|
||||
if (lock && Examine(lock, fib))
|
||||
{
|
||||
Printf("Name: %s\n", fib->fib_FileName);
|
||||
Printf("Size: %ld bytes\n", fib->fib_Size);
|
||||
Printf("Type: %s\n",
|
||||
fib->fib_DirEntryType > 0 ? "Directory" : "File");
|
||||
|
||||
/* Protection bits: */
|
||||
Printf("Protect: %08lx\n", fib->fib_Protection);
|
||||
/* Bits are ACTIVE-LOW: bit set = permission DENIED */
|
||||
/* FIBF_READ=8, FIBF_WRITE=4, FIBF_EXECUTE=2, FIBF_DELETE=1 */
|
||||
/* Bits 4-7: hold/script/pure/archive (active-HIGH) */
|
||||
}
|
||||
|
||||
UnLock(lock);
|
||||
FreeDosObject(DOS_FIB, fib);
|
||||
```
|
||||
|
||||
> [!WARNING]
|
||||
> **Protection bits are inverted** for RWED: bit set means the permission is **denied**, not granted. This is opposite to Unix. `fib_Protection = 0` means "all permissions granted".
|
||||
|
||||
### Scan a Directory (ExNext Loop)
|
||||
|
||||
```c
|
||||
/* Classic directory enumeration pattern: */
|
||||
BPTR dirLock = Lock("SYS:Libs", SHARED_LOCK);
|
||||
struct FileInfoBlock *fib = AllocDosObject(DOS_FIB, NULL);
|
||||
|
||||
if (dirLock && Examine(dirLock, fib)) /* examine the dir itself first */
|
||||
{
|
||||
while (ExNext(dirLock, fib))
|
||||
{
|
||||
Printf(" %s (%ld bytes)\n",
|
||||
fib->fib_FileName, fib->fib_Size);
|
||||
}
|
||||
/* ExNext returns FALSE when done — check error: */
|
||||
if (IoErr() != ERROR_NO_MORE_ENTRIES)
|
||||
PrintFault(IoErr(), "ExNext error");
|
||||
}
|
||||
|
||||
UnLock(dirLock);
|
||||
FreeDosObject(DOS_FIB, fib);
|
||||
```
|
||||
|
||||
### ExAll — Bulk Directory Scan (OS 2.0+)
|
||||
|
||||
`ExAll` is significantly faster than `ExNext` for large directories — it batches results:
|
||||
|
||||
```c
|
||||
#define BUFSIZE 4096
|
||||
APTR buf = AllocVec(BUFSIZE, MEMF_ANY);
|
||||
struct ExAllControl *eac = AllocDosObject(DOS_EXALLCONTROL, NULL);
|
||||
UBYTE buf[4096];
|
||||
eac->eac_LastKey = 0;
|
||||
|
||||
BPTR dirLock = Lock("SYS:Libs", SHARED_LOCK);
|
||||
BOOL more;
|
||||
|
||||
eac->eac_LastKey = 0;
|
||||
do {
|
||||
more = ExAll(lock, buf, sizeof(buf), ED_NAME, eac);
|
||||
more = ExAll(dirLock, buf, BUFSIZE, ED_SIZE, eac);
|
||||
|
||||
if (!more && IoErr() != ERROR_NO_MORE_ENTRIES)
|
||||
{
|
||||
PrintFault(IoErr(), "ExAll error");
|
||||
break;
|
||||
}
|
||||
|
||||
struct ExAllData *ead = (struct ExAllData *)buf;
|
||||
while (ead) {
|
||||
Printf("%s\n", ead->ed_Name);
|
||||
while (ead)
|
||||
{
|
||||
Printf(" %s (%ld bytes)\n", ead->ed_Name, ead->ed_Size);
|
||||
ead = ead->ed_Next;
|
||||
}
|
||||
} while (more);
|
||||
|
||||
FreeDosObject(DOS_EXALLCONTROL, eac);
|
||||
FreeVec(buf);
|
||||
UnLock(dirLock);
|
||||
```
|
||||
|
||||
---
|
||||
|
||||
## Practical Patterns
|
||||
|
||||
### Safe File Replacement (Atomic Write)
|
||||
|
||||
```c
|
||||
/* Anti-pattern: overwriting directly — corruption on crash */
|
||||
/* WRONG: */
|
||||
fh = Open("config.prefs", MODE_NEWFILE); /* truncates original! */
|
||||
Write(fh, data, size); /* crash here = lost config */
|
||||
Close(fh);
|
||||
|
||||
/* Correct: write-to-temp then rename */
|
||||
fh = Open("config.prefs.tmp", MODE_NEWFILE);
|
||||
Write(fh, data, size);
|
||||
Close(fh);
|
||||
|
||||
/* Atomic swap: */
|
||||
DeleteFile("config.prefs.bak"); /* remove old backup */
|
||||
Rename("config.prefs", "config.prefs.bak"); /* backup current */
|
||||
Rename("config.prefs.tmp", "config.prefs"); /* install new */
|
||||
```
|
||||
|
||||
### Check If File Exists (Without Opening)
|
||||
|
||||
```c
|
||||
/* Use Lock — lighter than Open: */
|
||||
BPTR lock = Lock("SYS:Libs/68040.library", SHARED_LOCK);
|
||||
if (lock) {
|
||||
UnLock(lock);
|
||||
/* file exists */
|
||||
} else {
|
||||
/* file doesn't exist (or permission denied — check IoErr()) */
|
||||
}
|
||||
```
|
||||
|
||||
### Get Parent Directory
|
||||
|
||||
```c
|
||||
BPTR fileLock = Lock("SYS:Libs/68040.library", SHARED_LOCK);
|
||||
BPTR parentLock = ParentDir(fileLock);
|
||||
if (parentLock) {
|
||||
/* parentLock = lock on SYS:Libs */
|
||||
NameFromLock(parentLock, buf, sizeof(buf));
|
||||
Printf("Parent: %s\n", buf); /* "SYS:Libs" */
|
||||
UnLock(parentLock);
|
||||
}
|
||||
UnLock(fileLock);
|
||||
```
|
||||
|
||||
### Resolve Full Path from Lock
|
||||
|
||||
```c
|
||||
/* Get the full filesystem path of any lock: */
|
||||
BPTR lock = Lock("PROGDIR:data", SHARED_LOCK);
|
||||
char fullpath[256];
|
||||
if (NameFromLock(lock, fullpath, sizeof(fullpath)))
|
||||
Printf("Full path: %s\n", fullpath);
|
||||
/* e.g. "DH0:Games/MyGame/data" */
|
||||
UnLock(lock);
|
||||
```
|
||||
|
||||
### Determine Volume / Device Name
|
||||
|
||||
```c
|
||||
/* Find which volume a lock belongs to: */
|
||||
struct InfoData *id = AllocVec(sizeof(struct InfoData), MEMF_ANY);
|
||||
BPTR lock = Lock("SYS:", SHARED_LOCK);
|
||||
if (Info(lock, id))
|
||||
{
|
||||
struct DosList *vol = (struct DosList *)BADDR(id->id_VolumeNode);
|
||||
UBYTE *bname = (UBYTE *)BADDR(vol->dol_Name);
|
||||
Printf("Volume: %.*s\n", bname[0], &bname[1]); /* BSTR */
|
||||
Printf("Disk type: $%08lx\n", id->id_DiskType); /* 'DOS\1'=FFS */
|
||||
Printf("Used: %ld / %ld blocks\n",
|
||||
id->id_NumBlocksUsed, id->id_NumBlocks);
|
||||
}
|
||||
UnLock(lock);
|
||||
FreeVec(id);
|
||||
```
|
||||
|
||||
---
|
||||
|
||||
## Common Antipatterns
|
||||
|
||||
### ❌ Leaking Locks
|
||||
|
||||
```c
|
||||
/* WRONG: lock is never released → filesystem handler keeps reference
|
||||
forever, preventing volume ejection */
|
||||
void bad_function(void) {
|
||||
BPTR lock = Lock("DF0:file", SHARED_LOCK);
|
||||
if (!lock) return;
|
||||
/* ... forgot UnLock(lock) ... */
|
||||
}
|
||||
|
||||
/* CORRECT: always pair Lock/UnLock */
|
||||
void good_function(void) {
|
||||
BPTR lock = Lock("DF0:file", SHARED_LOCK);
|
||||
if (!lock) return;
|
||||
/* ... do work ... */
|
||||
UnLock(lock); /* ALWAYS release */
|
||||
}
|
||||
```
|
||||
|
||||
> Leaked locks are the #1 cause of "please insert volume X" requesters that never go away. The filesystem handler keeps the volume "in use" because an outstanding lock exists.
|
||||
|
||||
### ❌ Holding Exclusive Lock Too Long
|
||||
|
||||
```c
|
||||
/* WRONG: other processes blocked for entire read operation */
|
||||
BPTR lock = Lock("shared_data.dat", EXCLUSIVE_LOCK);
|
||||
fh = OpenFromLock(lock); /* lock consumed by OpenFromLock */
|
||||
/* ... lengthy processing ... */
|
||||
Close(fh);
|
||||
|
||||
/* BETTER: use shared lock, only exclusive for the write phase */
|
||||
BPTR lock = Lock("shared_data.dat", SHARED_LOCK);
|
||||
fh = OpenFromLock(lock);
|
||||
Read(fh, buf, size); /* shared — others can read too */
|
||||
Close(fh);
|
||||
/* Now open exclusively just for writing: */
|
||||
fh = Open("shared_data.dat", MODE_READWRITE);
|
||||
Write(fh, newdata, newsize);
|
||||
Close(fh);
|
||||
```
|
||||
|
||||
### ❌ Not Checking IoErr() After Lock Failure
|
||||
|
||||
```c
|
||||
/* WRONG: can't distinguish "file not found" from "disk error" */
|
||||
BPTR lock = Lock("DF0:important", SHARED_LOCK);
|
||||
if (!lock) Printf("Not found!\n"); /* might be disk error! */
|
||||
|
||||
/* CORRECT: */
|
||||
BPTR lock = Lock("DF0:important", SHARED_LOCK);
|
||||
if (!lock) {
|
||||
LONG err = IoErr();
|
||||
switch (err) {
|
||||
case ERROR_OBJECT_NOT_FOUND: Printf("Not found\n"); break;
|
||||
case ERROR_OBJECT_IN_USE: Printf("Locked by another process\n"); break;
|
||||
case ERROR_DISK_NOT_VALIDATED: Printf("Disk not ready\n"); break;
|
||||
default: PrintFault(err, "Lock failed"); break;
|
||||
}
|
||||
}
|
||||
```
|
||||
|
||||
### ❌ Using DupLock on Exclusive Locks
|
||||
|
||||
```c
|
||||
/* WRONG: DupLock on an exclusive lock creates a SHARED copy.
|
||||
This breaks the exclusivity guarantee! */
|
||||
BPTR excl = Lock("data.dat", EXCLUSIVE_LOCK);
|
||||
BPTR dup = DupLock(excl); /* dup is SHARED — exclusivity lost */
|
||||
|
||||
/* DupLock always creates a shared lock, regardless of the source. */
|
||||
```
|
||||
|
||||
---
|
||||
|
||||
## Pattern Matching (Directory Wildcards)
|
||||
|
||||
```c
|
||||
/* Match files against AmigaDOS wildcard patterns: */
|
||||
/* #? = any string (like Unix *) */
|
||||
/* ? = any single char */
|
||||
/* ~ = NOT */
|
||||
/* | = OR */
|
||||
/* ( ) = grouping */
|
||||
|
||||
char tokenBuf[256];
|
||||
LONG tok = ParsePattern("#?.library", tokenBuf, sizeof(tokenBuf));
|
||||
if (tok >= 0) {
|
||||
/* tok > 0 means pattern contains wildcards */
|
||||
/* tok = 0 means literal string (no wildcards) */
|
||||
|
||||
while (ExNext(dirLock, fib)) {
|
||||
if (MatchPattern(tokenBuf, fib->fib_FileName))
|
||||
Printf(" Match: %s\n", fib->fib_FileName);
|
||||
}
|
||||
}
|
||||
```
|
||||
|
||||
| AmigaDOS Pattern | Unix Equivalent | Meaning |
|
||||
|---|---|---|
|
||||
| `#?` | `*` | Any string |
|
||||
| `?` | `?` | Any single character |
|
||||
| `#?.library` | `*.library` | All library files |
|
||||
| `~(#?.info)` | (no equiv) | Everything except .info files |
|
||||
| `(a|b)#?` | `{a,b}*` | Starting with a or b |
|
||||
|
||||
---
|
||||
|
||||
## References
|
||||
|
||||
- NDK39: `dos/dos.h`, `dos/dosextens.h`, `dos/exall.h`
|
||||
- ADCD 2.1: `Lock`, `UnLock`, `Examine`, `ExNext`, `ExAll`
|
||||
- ADCD 2.1: `Lock`, `UnLock`, `Examine`, `ExNext`, `ExAll`, `NameFromLock`
|
||||
- Ralph Babel: *The AmigaDOS Manual* — lock semantics chapter
|
||||
- See also: [packet_system.md](packet_system.md) — how locks translate to handler packets
|
||||
- See also: [file_io.md](file_io.md) — file operations using handles
|
||||
|
|
|
|||
Loading…
Add table
Add a link
Reference in a new issue