Source to src/filesys.c
/*
* UAE - The Un*x Amiga Emulator
*
* Unix file system handler for AmigaDOS
*
* Copyright 1996 Ed Hanway
* Copyright 1996, 1997 Bernd Schmidt
*
* Version 0.4: 970308
*
* Based on example code (c) 1988 The Software Distillery
* and published in Transactor for the Amiga, Volume 2, Issues 2-5.
* (May - August 1989)
*
* Known limitations:
* Does not support ACTION_INHIBIT (big deal).
* Does not support several 2.0+ packet types.
* Does not support removable volumes.
* May not return the correct error code in some cases.
* Does not check for sane values passed by AmigaDOS. May crash the emulation
* if passed garbage values.
* Could do tighter checks on malloc return values.
* Will probably fail spectacularly in some cases if the filesystem is
* modified at the same time by another process while UAE is running.
*/
#include "sysconfig.h"
#include "sysdeps.h"
#include "threaddep/thread.h"
#include "options.h"
#include "uae.h"
#include "memory.h"
#include "custom.h"
#include "events.h"
#include "newcpu.h"
#include "filesys.h"
#include "traps.h"
#include "autoconf.h"
#include "fsusage.h"
#include "native2amiga.h"
#include "scsidev.h"
#include "fsdb.h"
/* Count the number of FS packets waiting to be serviced, for the benefit
* of the idle on STOP code. */
int active_fs_packets;
uae_sem_t packet_counter_sem;
#define TRACING_ENABLED 0
#if TRACING_ENABLED
#define TRACE(x) do { write_log x; } while(0)
#define DUMPLOCK(u,x) dumplock(u,x)
#else
#define TRACE(x)
#define DUMPLOCK(u,x)
#endif
static long dos_errno(void)
{
int e = errno;
switch (e) {
case ENOMEM: return ERROR_NO_FREE_STORE;
case EEXIST: return ERROR_OBJECT_EXISTS;
case EACCES: return ERROR_WRITE_PROTECTED;
case ENOENT: return ERROR_OBJECT_NOT_AROUND;
case ENOTDIR: return ERROR_OBJECT_WRONG_TYPE;
case ENOSPC: return ERROR_DISK_IS_FULL;
case EBUSY: return ERROR_OBJECT_IN_USE;
case EISDIR: return ERROR_OBJECT_WRONG_TYPE;
#if defined(ETXTBSY)
case ETXTBSY: return ERROR_OBJECT_IN_USE;
#endif
#if defined(EROFS)
case EROFS: return ERROR_DISK_WRITE_PROTECTED;
#endif
#if defined(ENOTEMPTY)
#if ENOTEMPTY != EEXIST
case ENOTEMPTY: return ERROR_DIRECTORY_NOT_EMPTY;
#endif
#endif
default:
TRACE(("Unimplemented error %s\n", strerror(e)));
return ERROR_NOT_IMPLEMENTED;
}
}
/*
* This _should_ be no issue for us, but let's rather use a guaranteed
* thread safe function if we have one.
* This used to be broken in glibc versions <= 2.0.1 (I think). I hope
* no one is using this these days.
* Michael Krause says it's also broken in libc5. ARRRGHHHHHH!!!!
*/
#if 0 && defined HAVE_READDIR_R
static struct dirent *my_readdir (DIR *dirstream, struct dirent *space)
{
struct dirent *loc;
if (readdir_r (dirstream, space, &loc) == 0) {
/* Success */
return loc;
}
return 0;
}
#else
#define my_readdir(dirstream, space) readdir(dirstream)
#endif
uaecptr filesys_initcode;
static uae_u32 fsdevname, filesys_configdev;
#define FS_STARTUP 0
#define FS_GO_DOWN 1
#define DEVNAMES_PER_HDF 32
typedef struct {
char *devname; /* device name, e.g. UAE0: */
uaecptr devname_amiga;
uaecptr startup;
char *volname; /* volume name, e.g. CDROM, WORK, etc. */
char *rootdir; /* root unix directory */
int readonly; /* disallow write access? */
int bootpri;
int devno;
struct hardfiledata hf;
/* Threading stuff */
smp_comm_pipe *unit_pipe, *back_pipe;
uae_thread_id tid;
struct _unit *volatile self;
/* Reset handling */
uae_sem_t reset_sync_sem;
int reset_state;
/* RDB stuff */
uaecptr rdb_devname_amiga[DEVNAMES_PER_HDF];
int rdb_lowcyl;
int rdb_highcyl;
int rdb_cylblocks;
uae_u8 *rdb_filesysstore;
int rdb_filesyssize;
char *filesysdir;
} UnitInfo;
#define MAX_UNITS 20
struct uaedev_mount_info {
int num_units;
UnitInfo ui[MAX_UNITS];
};
static struct uaedev_mount_info *current_mountinfo;
int nr_units (struct uaedev_mount_info *mountinfo)
{
return mountinfo->num_units;
}
int hardfile_fs_type (struct uaedev_mount_info *mountinfo, int unit_no)
{
if (mountinfo->ui[unit_no].volname)
return FILESYS_VIRTUAL;
if (mountinfo->ui[unit_no].hf.secspertrack == 0) {
#if 0
if (mountinfo->ui[unit_no].hf.flags & 1)
return FILESYS_HARDDRIVE;
#endif
return FILESYS_HARDFILE_RDB;
}
return FILESYS_HARDFILE;
}
static void close_filesys_unit (UnitInfo *uip)
{
if (uip->hf.fd != 0)
fclose (uip->hf.fd);
if (uip->volname != 0)
free (uip->volname);
if (uip->devname != 0)
free (uip->devname);
if (uip->rootdir != 0)
free (uip->rootdir);
if (uip->unit_pipe)
free (uip->unit_pipe);
if (uip->back_pipe)
free (uip->back_pipe);
uip->unit_pipe = 0;
uip->back_pipe = 0;
uip->hf.fd = 0;
uip->volname = 0;
uip->devname = 0;
uip->rootdir = 0;
}
char *get_filesys_unit (struct uaedev_mount_info *mountinfo, int nr,
char **devname, char **volname, char **rootdir,
int *readonly, int *secspertrack, int *surfaces, int *reserved,
int *cylinders, int *size, int *blocksize, int *bootpri)
{
UnitInfo *uip = mountinfo->ui + nr;
if (nr >= mountinfo->num_units)
return "No slot allocated for this unit";
*volname = uip->volname ? my_strdup (uip->volname) : 0;
*devname = uip->devname ? my_strdup (uip->devname) : 0;
*rootdir = uip->rootdir ? my_strdup (uip->rootdir) : 0;
*readonly = uip->readonly;
*secspertrack = uip->hf.secspertrack;
*surfaces = uip->hf.surfaces;
*reserved = uip->hf.reservedblocks;
*size = uip->hf.size;
*cylinders = uip->hf.nrcyls;
*blocksize = uip->hf.blocksize;
*bootpri = uip->bootpri;
return 0;
}
static char *set_filesys_unit_1 (struct uaedev_mount_info *mountinfo, int nr,
const char *devname, const char *volname, const char *rootdir,
int readonly, int secspertrack, int surfaces, int reserved,
int blocksize, int bootpri)
{
UnitInfo *ui = mountinfo->ui + nr;
if (nr >= mountinfo->num_units)
return "No slot allocated for this unit";
ui->hf.fd = 0;
ui->devname = 0;
ui->volname = 0;
ui->rootdir = 0;
ui->unit_pipe = 0;
ui->back_pipe = 0;
if (volname != 0) {
ui->volname = my_strdup (volname);
ui->hf.fd = 0;
} else {
ui->volname = 0;
ui->hf.fd = fopen (rootdir, "r+b");
if (ui->hf.fd == 0) {
readonly = 1;
ui->hf.fd = fopen (rootdir, "rb");
}
if (ui->hf.fd == 0)
return "Hardfile not found";
if ((secspertrack || surfaces || reserved)
&& (secspertrack < 1 || secspertrack > 32767
|| surfaces < 1 || surfaces > 1023
|| reserved < 0 || reserved > 1023
|| (blocksize & (blocksize - 1)) != 0))
{
return "Bad hardfile geometry";
}
fseek (ui->hf.fd, 0, SEEK_END);
ui->hf.size = ftell (ui->hf.fd);
ui->hf.secspertrack = secspertrack;
ui->hf.surfaces = surfaces;
ui->hf.reservedblocks = reserved;
ui->hf.nrcyls = (secspertrack * surfaces
? (ui->hf.size / blocksize) / (secspertrack * surfaces)
: 0);
ui->hf.blocksize = blocksize;
}
ui->self = 0;
ui->reset_state = FS_STARTUP;
ui->rootdir = my_strdup (rootdir);
if (devname != 0 && strlen (devname) != 0)
ui->devname = my_strdup (devname);
ui->readonly = readonly;
if (bootpri < -128)
bootpri = -128;
if (bootpri > 127)
bootpri = 127;
ui->bootpri = bootpri;
return 0;
}
char *set_filesys_unit (struct uaedev_mount_info *mountinfo, int nr,
const char *devname, const char *volname, const char *rootdir,
int readonly, int secspertrack, int surfaces, int reserved,
int blocksize, int bootpri)
{
UnitInfo ui = mountinfo->ui[nr];
char *result = set_filesys_unit_1 (mountinfo, nr, devname, volname, rootdir,
readonly, secspertrack, surfaces, reserved,
blocksize, bootpri);
if (result)
mountinfo->ui[nr] = ui;
else
close_filesys_unit (&ui);
return result;
}
char *add_filesys_unit (struct uaedev_mount_info *mountinfo,
const char *devname, const char *volname, const char *rootdir,
int readonly, int secspertrack, int surfaces, int reserved,
int blocksize, int bootpri)
{
char *retval;
int nr = mountinfo->num_units;
UnitInfo *uip = mountinfo->ui + nr;
if (nr >= MAX_UNITS)
return "Maximum number of file systems mounted";
mountinfo->num_units++;
retval = set_filesys_unit_1 (mountinfo, nr, devname, volname, rootdir, readonly,
secspertrack, surfaces, reserved, blocksize,
bootpri);
if (retval)
mountinfo->num_units--;
return retval;
}
int kill_filesys_unit (struct uaedev_mount_info *mountinfo, int nr)
{
UnitInfo *uip = mountinfo->ui;
if (nr >= mountinfo->num_units || nr < 0)
return -1;
close_filesys_unit (mountinfo->ui + nr);
mountinfo->num_units--;
for (; nr < mountinfo->num_units; nr++) {
uip[nr] = uip[nr+1];
}
return 0;
}
int move_filesys_unit (struct uaedev_mount_info *mountinfo, int nr, int to)
{
UnitInfo tmpui;
UnitInfo *uip = mountinfo->ui;
if (nr >= mountinfo->num_units || nr < 0
|| to >= mountinfo->num_units || to < 0
|| to == nr)
return -1;
tmpui = uip[nr];
if (to > nr) {
int i;
for (i = nr; i < to; i++)
uip[i] = uip[i + 1];
} else {
int i;
for (i = nr; i > to; i--)
uip[i] = uip[i - 1];
}
uip[to] = tmpui;
return 0;
}
int sprintf_filesys_unit (struct uaedev_mount_info *mountinfo, char *buffer, int num)
{
UnitInfo *uip = mountinfo->ui;
if (num >= mountinfo->num_units)
return -1;
if (uip[num].volname != 0)
sprintf (buffer, "(DH%d:) Filesystem, %s: %s %s", num, uip[num].volname,
uip[num].rootdir, uip[num].readonly ? "ro" : "");
else
sprintf (buffer, "(DH%d:) Hardfile, \"%s\", size %d bytes", num,
uip[num].rootdir, uip[num].hf.size);
return 0;
}
void write_filesys_config (struct uaedev_mount_info *mountinfo,
const char *unexpanded, const char *default_path, FILE *f)
{
UnitInfo *uip = mountinfo->ui;
int i;
for (i = 0; i < mountinfo->num_units; i++) {
char *str;
str = cfgfile_subst_path (default_path, unexpanded, uip[i].rootdir);
if (hardfile_fs_type (mountinfo, i) == FILESYS_VIRTUAL) {
fprintf (f, "filesystem2=%s,%s:%s:%s,%d\n", uip[i].readonly ? "ro" : "rw",
"", uip[i].volname, str, uip[i].bootpri);
fprintf (f, "filesystem=%s,%s:%s\n", uip[i].readonly ? "ro" : "rw",
uip[i].volname, str);
} else {
fprintf (f, "hardfile2=%s,%s:%s,%d,%d,%d,%d,%d,%s,%s\n",
uip[i].readonly ? "ro" : "rw",
"", str,
uip[i].hf.secspertrack,
uip[i].hf.surfaces, uip[i].hf.reservedblocks,
uip[i].hf.blocksize, uip[i].bootpri, "", "uae");
fprintf (f, "hardfile=%s,%d,%d,%d,%d,%s\n",
uip[i].readonly ? "ro" : "rw", uip[i].hf.secspertrack,
uip[i].hf.surfaces, uip[i].hf.reservedblocks,
uip[i].hf.blocksize, str);
}
free (str);
}
}
struct uaedev_mount_info *alloc_mountinfo (void)
{
struct uaedev_mount_info *info;
info = (struct uaedev_mount_info *)malloc (sizeof *info);
/* memset (info, 0xaa, sizeof *info);*/
info->num_units = 0;
return info;
}
struct uaedev_mount_info *dup_mountinfo (struct uaedev_mount_info *mip)
{
int i;
struct uaedev_mount_info *i2 = alloc_mountinfo ();
memcpy (i2, mip, sizeof *i2);
for (i = 0; i < i2->num_units; i++) {
UnitInfo *uip = i2->ui + i;
if (uip->volname)
uip->volname = my_strdup (uip->volname);
if (uip->rootdir)
uip->rootdir = my_strdup (uip->rootdir);
if (uip->hf.fd)
uip->hf.fd = fdopen ( dup (fileno (uip->hf.fd)), uip->readonly ? "rb" : "r+b");
}
return i2;
}
void free_mountinfo (struct uaedev_mount_info *mip)
{
int i;
for (i = 0; i < mip->num_units; i++)
close_filesys_unit (mip->ui + i);
free (mip);
}
struct hardfiledata *get_hardfile_data (int nr)
{
UnitInfo *uip = current_mountinfo->ui;
if (nr < 0 || nr >= current_mountinfo->num_units || uip[nr].volname != 0)
return 0;
return &uip[nr].hf;
}
/* minimal AmigaDOS definitions */
/* field offsets in DosPacket */
#define dp_Type 8
#define dp_Res1 12
#define dp_Res2 16
#define dp_Arg1 20
#define dp_Arg2 24
#define dp_Arg3 28
#define dp_Arg4 32
/* result codes */
#define DOS_TRUE (0xffffffffUL)
#define DOS_FALSE (0L)
/* Passed as type to Lock() */
#define SHARED_LOCK -2 /* File is readable by others */
#define ACCESS_READ -2 /* Synonym */
#define EXCLUSIVE_LOCK -1 /* No other access allowed */
#define ACCESS_WRITE -1 /* Synonym */
/* packet types */
#define ACTION_CURRENT_VOLUME 7
#define ACTION_LOCATE_OBJECT 8
#define ACTION_RENAME_DISK 9
#define ACTION_FREE_LOCK 15
#define ACTION_DELETE_OBJECT 16
#define ACTION_RENAME_OBJECT 17
#define ACTION_COPY_DIR 19
#define ACTION_SET_PROTECT 21
#define ACTION_CREATE_DIR 22
#define ACTION_EXAMINE_OBJECT 23
#define ACTION_EXAMINE_NEXT 24
#define ACTION_DISK_INFO 25
#define ACTION_INFO 26
#define ACTION_FLUSH 27
#define ACTION_SET_COMMENT 28
#define ACTION_PARENT 29
#define ACTION_SET_DATE 34
#define ACTION_FIND_WRITE 1004
#define ACTION_FIND_INPUT 1005
#define ACTION_FIND_OUTPUT 1006
#define ACTION_END 1007
#define ACTION_SEEK 1008
#define ACTION_IS_FILESYSTEM 1027
#define ACTION_READ 'R'
#define ACTION_WRITE 'W'
/* 2.0+ packet types */
#define ACTION_INHIBIT 31
#define ACTION_SET_FILE_SIZE 1022
#define ACTION_LOCK_RECORD 2008
#define ACTION_FREE_RECORD 2009
#define ACTION_SAME_LOCK 40
#define ACTION_CHANGE_MODE 1028
#define ACTION_FH_FROM_LOCK 1026
#define ACTION_COPY_DIR_FH 1030
#define ACTION_PARENT_FH 1031
#define ACTION_EXAMINE_FH 1034
#define ACTION_EXAMINE_ALL 1033
#define ACTION_MAKE_LINK 1021
#define ACTION_READ_LINK 1024
#define ACTION_FORMAT 1020
#define ACTION_IS_FILESYSTEM 1027
#define ACTION_ADD_NOTIFY 4097
#define ACTION_REMOVE_NOTIFY 4098
#define DISK_TYPE 0x444f5301 /* DOS\1 */
typedef struct {
uae_u32 uniq;
/* The directory we're going through. */
a_inode *aino;
/* The file we're going to look up next. */
a_inode *curr_file;
} ExamineKey;
typedef struct key {
struct key *next;
a_inode *aino;
uae_u32 uniq;
int fd;
off_t file_pos;
} Key;
/* Since ACTION_EXAMINE_NEXT is so braindamaged, we have to keep
* some of these around
*/
#define EXKEYS 100
#define MAX_AINO_HASH 128
/* handler state info */
typedef struct _unit {
struct _unit *next;
/* Amiga stuff */
uaecptr dosbase;
uaecptr volume;
uaecptr port; /* Our port */
uaecptr locklist;
/* Native stuff */
uae_s32 unit; /* unit number */
UnitInfo ui; /* unit startup info */
char tmpbuf3[256];
/* Dummy message processing */
uaecptr dummy_message;
volatile unsigned int cmds_sent;
volatile unsigned int cmds_complete;
volatile unsigned int cmds_acked;
/* ExKeys */
ExamineKey examine_keys[EXKEYS];
int next_exkey;
unsigned long total_locked_ainos;
/* Keys */
struct key *keys;
uae_u32 key_uniq;
uae_u32 a_uniq;
a_inode rootnode;
unsigned long aino_cache_size;
a_inode *aino_hash[MAX_AINO_HASH];
unsigned long nr_cache_hits;
unsigned long nr_cache_lookups;
} Unit;
typedef uae_u8 *dpacket;
#define PUT_PCK_RES1(p,v) do { do_put_mem_long ((uae_u32 *)((p) + dp_Res1), (v)); } while (0)
#define PUT_PCK_RES2(p,v) do { do_put_mem_long ((uae_u32 *)((p) + dp_Res2), (v)); } while (0)
#define GET_PCK_TYPE(p) ((uae_s32)(do_get_mem_long ((uae_u32 *)((p) + dp_Type))))
#define GET_PCK_RES1(p) ((uae_s32)(do_get_mem_long ((uae_u32 *)((p) + dp_Res1))))
#define GET_PCK_RES2(p) ((uae_s32)(do_get_mem_long ((uae_u32 *)((p) + dp_Res2))))
#define GET_PCK_ARG1(p) ((uae_s32)(do_get_mem_long ((uae_u32 *)((p) + dp_Arg1))))
#define GET_PCK_ARG2(p) ((uae_s32)(do_get_mem_long ((uae_u32 *)((p) + dp_Arg2))))
#define GET_PCK_ARG3(p) ((uae_s32)(do_get_mem_long ((uae_u32 *)((p) + dp_Arg3))))
#define GET_PCK_ARG4(p) ((uae_s32)(do_get_mem_long ((uae_u32 *)((p) + dp_Arg4))))
static char *bstr1 (uaecptr addr)
{
static char buf[256];
int i;
int n = get_byte (addr);
addr++;
for (i = 0; i < n; i++, addr++)
buf[i] = get_byte (addr);
buf[i] = 0;
return buf;
}
static char *bstr (Unit *unit, uaecptr addr)
{
int i;
int n = get_byte (addr);
addr++;
for (i = 0; i < n; i++, addr++)
unit->tmpbuf3[i] = get_byte (addr);
unit->tmpbuf3[i] = 0;
return unit->tmpbuf3;
}
static char *bstr_cut (Unit *unit, uaecptr addr)
{
char *p = unit->tmpbuf3;
int i, colon_seen = 0;
int n = get_byte (addr);
addr++;
for (i = 0; i < n; i++, addr++) {
uae_u8 c = get_byte (addr);
unit->tmpbuf3[i] = c;
if (c == '/' || (c == ':' && colon_seen++ == 0))
p = unit->tmpbuf3 + i + 1;
}
unit->tmpbuf3[i] = 0;
return p;
}
static Unit *units = 0;
static int unit_num = 0;
static Unit*
find_unit (uaecptr port)
{
Unit* u;
for (u = units; u; u = u->next)
if (u->port == port)
break;
return u;
}
static void prepare_for_open (char *name)
{
#if 0
struct stat statbuf;
int mode;
if (-1 == stat (name, &statbuf))
return;
mode = statbuf.st_mode;
mode |= S_IRUSR;
mode |= S_IWUSR;
mode |= S_IXUSR;
chmod (name, mode);
#endif
}
static void de_recycle_aino (Unit *unit, a_inode *aino)
{
if (aino->next == 0 || aino == &unit->rootnode)
return;
aino->next->prev = aino->prev;
aino->prev->next = aino->next;
aino->next = aino->prev = 0;
unit->aino_cache_size--;
}
static void dispose_aino (Unit *unit, a_inode **aip, a_inode *aino)
{
int hash = aino->uniq % MAX_AINO_HASH;
if (unit->aino_hash[hash] == aino)
unit->aino_hash[hash] = 0;
if (aino->dirty && aino->parent)
fsdb_dir_writeback (aino->parent);
*aip = aino->sibling;
if (aino->comment)
free (aino->comment);
free (aino->nname);
free (aino->aname);
free (aino);
}
static void recycle_aino (Unit *unit, a_inode *new_aino)
{
if (new_aino->dir || new_aino->shlock > 0
|| new_aino->elock || new_aino == &unit->rootnode)
/* Still in use */
return;
TRACE (("Recycling; cache size %d, total_locked %d\n",
unit->aino_cache_size, unit->total_locked_ainos));
if (unit->aino_cache_size > 500 + unit->total_locked_ainos) {
/* Reap a few. */
int i = 0;
while (i < 50) {
a_inode *parent = unit->rootnode.prev->parent;
a_inode **aip, *old_prev;
aip = &parent->child;
if (! parent->locked_children) {
for (;;) {
a_inode *aino = *aip;
if (aino == 0)
break;
/* Not recyclable if next == 0 (i.e., not chained into
recyclable list), or if parent directory is being
ExNext()ed. */
if (aino->next == 0) {
aip = &aino->sibling;
} else {
if (aino->shlock > 0 || aino->elock)
write_log ("panic: freeing locked a_inode!\n");
de_recycle_aino (unit, aino);
dispose_aino (unit, aip, aino);
i++;
}
}
}
/* In the previous loop, we went through all children of one
parent. Re-arrange the recycled list so that we'll find a
different parent the next time around. */
old_prev = unit->rootnode.prev;
do {
unit->rootnode.next->prev = unit->rootnode.prev;
unit->rootnode.prev->next = unit->rootnode.next;
unit->rootnode.next = unit->rootnode.prev;
unit->rootnode.prev = unit->rootnode.prev->prev;
unit->rootnode.prev->next = unit->rootnode.next->prev = &unit->rootnode;
} while (unit->rootnode.prev != old_prev
&& unit->rootnode.prev->parent == parent);
}
#if 0
{
char buffer[40];
sprintf (buffer, "%d ainos reaped.\n", i);
TRACE ((buffer));
}
#endif
}
/* Chain it into circular list. */
new_aino->next = unit->rootnode.next;
new_aino->prev = &unit->rootnode;
new_aino->prev->next = new_aino;
new_aino->next->prev = new_aino;
unit->aino_cache_size++;
}
static void update_child_names (Unit *unit, a_inode *a, a_inode *parent)
{
int l0 = strlen (parent->nname) + 2;
while (a != 0) {
char *name_start;
char *new_name;
char dirsep[2] = { FSDB_DIR_SEPARATOR, '\0' };
a->parent = parent;
name_start = strrchr (a->nname, FSDB_DIR_SEPARATOR);
if (name_start == 0) {
write_log ("malformed file name");
}
name_start++;
new_name = (char *)xmalloc (strlen (name_start) + l0);
strcpy (new_name, parent->nname);
strcat (new_name, dirsep);
strcat (new_name, name_start);
free (a->nname);
a->nname = new_name;
if (a->child)
update_child_names (unit, a->child, a);
a = a->sibling;
}
}
static void move_aino_children (Unit *unit, a_inode *from, a_inode *to)
{
to->child = from->child;
from->child = 0;
update_child_names (unit, to->child, to);
}
static void delete_aino (Unit *unit, a_inode *aino)
{
a_inode **aip;
TRACE(("deleting aino %x\n", aino->uniq));
aino->dirty = 1;
aino->deleted = 1;
de_recycle_aino (unit, aino);
/* If any ExKeys are currently pointing at us, advance them. */
if (aino->parent->exnext_count > 0) {
int i;
TRACE(("entering exkey validation\n"));
for (i = 0; i < EXKEYS; i++) {
ExamineKey *k = unit->examine_keys + i;
if (k->uniq == 0)
continue;
if (k->aino == aino->parent) {
TRACE(("Same parent found for %d\n", i));
if (k->curr_file == aino) {
k->curr_file = aino->sibling;
TRACE(("Advancing curr_file\n"));
}
}
}
}
aip = &aino->parent->child;
while (*aip != aino && *aip != 0)
aip = &(*aip)->sibling;
if (*aip != aino) {
write_log ("Couldn't delete aino.\n");
return;
}
dispose_aino (unit, aip, aino);
}
static a_inode *lookup_sub (a_inode *dir, uae_u32 uniq)
{
a_inode **cp = &dir->child;
a_inode *c, *retval;
for (;;) {
c = *cp;
if (c == 0)
return 0;
if (c->uniq == uniq) {
retval = c;
break;
}
if (c->dir) {
a_inode *a = lookup_sub (c, uniq);
if (a != 0) {
retval = a;
break;
}
}
cp = &c->sibling;
}
if (! dir->locked_children) {
/* Move to the front to speed up repeated lookups. Don't do this if
an ExNext is going on in this directory, or we'll terminally
confuse it. */
*cp = c->sibling;
c->sibling = dir->child;
dir->child = c;
}
return retval;
}
static a_inode *lookup_aino (Unit *unit, uae_u32 uniq)
{
a_inode *a;
int hash = uniq % MAX_AINO_HASH;
if (uniq == 0)
return &unit->rootnode;
a = unit->aino_hash[hash];
if (a == 0 || a->uniq != uniq)
a = lookup_sub (&unit->rootnode, uniq);
else
unit->nr_cache_hits++;
unit->nr_cache_lookups++;
unit->aino_hash[hash] = a;
return a;
}
char *build_nname (const char *d, const char *n)
{
char dsep[2] = { FSDB_DIR_SEPARATOR, '\0' };
char *p = (char *) xmalloc (strlen (d) + strlen (n) + 2);
strcpy (p, d);
strcat (p, dsep);
strcat (p, n);
return p;
}
char *build_aname (const char *d, const char *n)
{
char *p = (char *) xmalloc (strlen (d) + strlen (n) + 2);
strcpy (p, d);
strcat (p, "/");
strcat (p, n);
return p;
}
/* This gets called to translate an Amiga name that some program used to
* a name that we can use on the native filesystem. */
static char *get_nname (Unit *unit, a_inode *base, char *rel,
char **modified_rel)
{
char *found;
char *p = 0;
*modified_rel = 0;
/* If we have a mapping of some other aname to "rel", we must pretend
* it does not exist.
* This can happen for example if an Amiga program creates a
* file called ".". We can't represent this in our filesystem,
* so we create a special file "uae_xxx" and record the mapping
* aname "." -> nname "uae_xxx" in the database. Then, the Amiga
* program looks up "uae_xxx" (yes, it's contrived). The filesystem
* should not make the uae_xxx file visible to the Amiga side. */
if (fsdb_used_as_nname (base, rel))
return 0;
/* A file called "." (or whatever else is invalid on this filesystem)
* does not exist, as far as the Amiga side is concerned. */
if (fsdb_name_invalid (rel))
return 0;
/* See if we have a file that has the same name as the aname we are
* looking for. */
found = fsdb_search_dir (base->nname, rel);
if (found == 0)
return found;
if (found == rel)
return build_nname (base->nname, rel);
*modified_rel = found;
return build_nname (base->nname, found);
}
static char *create_nname (Unit *unit, a_inode *base, char *rel)
{
char *p;
/* We are trying to create a file called REL. */
/* If the name is used otherwise in the directory (or globally), we
* need a new unique nname. */
if (fsdb_name_invalid (rel) || fsdb_used_as_nname (base, rel)) {
oh_dear:
p = fsdb_create_unique_nname (base, rel);
return p;
}
p = build_nname (base->nname, rel);
/* Delete this code once we know everything works. */
if (access (p, R_OK) >= 0 || errno != ENOENT) {
write_log ("Filesystem in trouble... please report.\n");
free (p);
goto oh_dear;
}
return p;
}
/*
* This gets called if an ACTION_EXAMINE_NEXT happens and we hit an object
* for which we know the name on the native filesystem, but no corresponding
* Amiga filesystem name.
* @@@ For DOS filesystems, it might make sense to declare the new name
* "weak", so that it can get overriden by a subsequent call to get_nname().
* That way, if someone does "dir :" and there is a file "foobar.inf", and
* someone else tries to open "foobar.info", get_nname() could maybe made to
* figure out that this is supposed to be the file "foobar.inf".
* DOS sucks...
*/
static char *get_aname (Unit *unit, a_inode *base, char *rel)
{
return my_strdup (rel);
}
static void init_child_aino (Unit *unit, a_inode *base, a_inode *aino)
{
aino->uniq = ++unit->a_uniq;
if (unit->a_uniq == 0xFFFFFFFF) {
write_log ("Running out of a_inodes (prepare for big trouble)!\n");
}
aino->shlock = 0;
aino->elock = 0;
aino->dirty = 0;
aino->deleted = 0;
/* For directories - this one isn't being ExNext()ed yet. */
aino->locked_children = 0;
aino->exnext_count = 0;
/* But the parent might be. */
if (base->exnext_count) {
unit->total_locked_ainos++;
base->locked_children++;
}
/* Update tree structure */
aino->parent = base;
aino->child = 0;
aino->sibling = base->child;
base->child = aino;
aino->next = aino->prev = 0;
}
static a_inode *new_child_aino (Unit *unit, a_inode *base, char *rel)
{
char *modified_rel;
char *nn;
a_inode *aino;
TRACE(("new_child_aino %s, %s\n", base->aname, rel));
aino = fsdb_lookup_aino_aname (base, rel);
if (aino == 0) {
nn = get_nname (unit, base, rel, &modified_rel);
if (nn == 0)
return 0;
aino = (a_inode *) xcalloc (sizeof (a_inode), 1);
if (aino == 0)
return 0;
aino->aname = modified_rel ? modified_rel : my_strdup (rel);
aino->nname = nn;
aino->comment = 0;
aino->has_dbentry = 0;
fsdb_fill_file_attrs (aino);
if (aino->dir)
fsdb_clean_dir (aino);
}
init_child_aino (unit, base, aino);
recycle_aino (unit, aino);
TRACE(("created aino %x, lookup, amigaos_mode %d\n", aino->uniq, aino->amigaos_mode));
return aino;
}
static a_inode *create_child_aino (Unit *unit, a_inode *base, char *rel, int isdir)
{
a_inode *aino = (a_inode *) xcalloc (sizeof (a_inode), 1);
if (aino == 0)
return 0;
aino->aname = my_strdup (rel);
aino->nname = create_nname (unit, base, rel);
init_child_aino (unit, base, aino);
aino->amigaos_mode = 0;
aino->dir = isdir;
aino->comment = 0;
aino->has_dbentry = 0;
aino->dirty = 1;
recycle_aino (unit, aino);
TRACE(("created aino %x, create\n", aino->uniq));
return aino;
}
static a_inode *lookup_child_aino (Unit *unit, a_inode *base, char *rel, uae_u32 *err)
{
a_inode *c = base->child;
int l0 = strlen (rel);
if (base->dir == 0) {
*err = ERROR_OBJECT_WRONG_TYPE;
return 0;
}
while (c != 0) {
int l1 = strlen (c->aname);
if (l0 <= l1 && same_aname (rel, c->aname + l1 - l0)
&& (l0 == l1 || c->aname[l1-l0-1] == '/'))
break;
c = c->sibling;
}
if (c != 0)
return c;
c = new_child_aino (unit, base, rel);
if (c == 0)
*err = ERROR_OBJECT_NOT_AROUND;
return c;
}
/* Different version because for this one, REL is an nname. */
static a_inode *lookup_child_aino_for_exnext (Unit *unit, a_inode *base, char *rel, uae_u32 *err)
{
a_inode *c = base->child;
int l0 = strlen (rel);
*err = 0;
while (c != 0) {
int l1 = strlen (c->nname);
/* Note: using strcmp here. */
if (l0 <= l1 && strcmp (rel, c->nname + l1 - l0) == 0
&& (l0 == l1 || c->nname[l1-l0-1] == FSDB_DIR_SEPARATOR))
break;
c = c->sibling;
}
if (c != 0)
return c;
c = fsdb_lookup_aino_nname (base, rel);
if (c == 0) {
c = (a_inode *)malloc (sizeof (a_inode));
if (c == 0) {
*err = ERROR_NO_FREE_STORE;
return 0;
}
c->nname = build_nname (base->nname, rel);
c->aname = get_aname (unit, base, rel);
c->comment = 0;
c->has_dbentry = 0;
fsdb_fill_file_attrs (c);
if (c->dir)
fsdb_clean_dir (c);
}
init_child_aino (unit, base, c);
recycle_aino (unit, c);
TRACE(("created aino %x, exnext\n", c->uniq));
return c;
}
static a_inode *get_aino (Unit *unit, a_inode *base, const char *rel, uae_u32 *err)
{
char *tmp;
char *p;
a_inode *curr;
int i;
*err = 0;
TRACE(("get_path(%s,%s)\n", base->aname, rel));
/* root-relative path? */
for (i = 0; rel[i] && rel[i] != '/' && rel[i] != ':'; i++)
;
if (':' == rel[i])
rel += i+1;
tmp = my_strdup (rel);
p = tmp;
curr = base;
while (*p) {
/* start with a slash? go up a level. */
if (*p == '/') {
if (curr->parent != 0)
curr = curr->parent;
p++;
} else {
a_inode *next;
char *component_end;
component_end = strchr (p, '/');
if (component_end != 0)
*component_end = '\0';
next = lookup_child_aino (unit, curr, p, err);
if (next == 0) {
/* if only last component not found, return parent dir. */
if (*err != ERROR_OBJECT_NOT_AROUND || component_end != 0)
curr = 0;
/* ? what error is appropriate? */
break;
}
curr = next;
if (component_end)
p = component_end+1;
else
break;
}
}
free (tmp);
return curr;
}
static uae_u32 startup_handler (TrapContext *dummy)
{
/* Just got the startup packet. It's in A4. DosBase is in A2,
* our allocated volume structure is in D6, A5 is a pointer to
* our port. */
uaecptr rootnode = get_long (m68k_areg (regs, 2) + 34);
uaecptr dos_info = get_long (rootnode + 24) << 2;
uaecptr pkt = m68k_dreg (regs, 3);
uaecptr arg2 = get_long (pkt + dp_Arg2);
int i, namelen;
char* devname = bstr1 (get_long (pkt + dp_Arg1) << 2);
char* s;
Unit *unit;
UnitInfo *uinfo;
/* find UnitInfo with correct device name */
s = strchr (devname, ':');
if (s)
*s = '\0';
for (i = 0; i < current_mountinfo->num_units; i++) {
/* Hardfile volume name? */
if (current_mountinfo->ui[i].volname == 0)
continue;
if (current_mountinfo->ui[i].startup == arg2)
break;
}
if (i == current_mountinfo->num_units
|| access (current_mountinfo->ui[i].rootdir, R_OK) != 0)
{
write_log ("Failed attempt to mount device\n", devname);
put_long (pkt + dp_Res1, DOS_FALSE);
put_long (pkt + dp_Res2, ERROR_DEVICE_NOT_MOUNTED);
return 1;
}
uinfo = current_mountinfo->ui + i;
unit = (Unit *) xcalloc (sizeof (Unit), 1);
unit->next = units;
units = unit;
uinfo->self = unit;
unit->volume = 0;
unit->port = m68k_areg (regs, 5);
unit->unit = unit_num++;
unit->ui.devname = uinfo->devname;
unit->ui.volname = my_strdup (uinfo->volname); /* might free later for rename */
unit->ui.rootdir = uinfo->rootdir;
unit->ui.readonly = uinfo->readonly;
unit->ui.bootpri = uinfo->bootpri;
unit->ui.unit_pipe = uinfo->unit_pipe;
unit->ui.back_pipe = uinfo->back_pipe;
unit->cmds_complete = 0;
unit->cmds_sent = 0;
unit->cmds_acked = 0;
for (i = 0; i < EXKEYS; i++) {
unit->examine_keys[i].aino = 0;
unit->examine_keys[i].curr_file = 0;
unit->examine_keys[i].uniq = 0;
}
unit->total_locked_ainos = 0;
unit->next_exkey = 1;
unit->keys = 0;
unit->a_uniq = unit->key_uniq = 0;
unit->rootnode.aname = uinfo->volname;
unit->rootnode.nname = uinfo->rootdir;
unit->rootnode.sibling = 0;
unit->rootnode.next = unit->rootnode.prev = &unit->rootnode;
unit->rootnode.uniq = 0;
unit->rootnode.parent = 0;
unit->rootnode.child = 0;
unit->rootnode.dir = 1;
unit->rootnode.amigaos_mode = 0;
unit->rootnode.shlock = 0;
unit->rootnode.elock = 0;
unit->rootnode.comment = 0;
unit->rootnode.has_dbentry = 0;
unit->aino_cache_size = 0;
for (i = 0; i < MAX_AINO_HASH; i++)
unit->aino_hash[i] = 0;
/* write_comm_pipe_int (unit->ui.unit_pipe, -1, 1);*/
TRACE(("**** STARTUP volume %s\n", unit->ui.volname));
/* fill in our process in the device node */
put_long ((get_long (pkt + dp_Arg3) << 2) + 8, unit->port);
unit->dosbase = m68k_areg (regs, 2);
/* make new volume */
unit->volume = m68k_areg (regs, 3) + 32;
#ifdef UAE_FILESYS_THREADS
unit->locklist = m68k_areg (regs, 3) + 8;
#else
unit->locklist = m68k_areg (regs, 3);
#endif
unit->dummy_message = m68k_areg (regs, 3) + 12;
put_long (unit->dummy_message + 10, 0);
put_long (unit->volume + 4, 2); /* Type = dt_volume */
put_long (unit->volume + 12, 0); /* Lock */
put_long (unit->volume + 16, 3800); /* Creation Date */
put_long (unit->volume + 20, 0);
put_long (unit->volume + 24, 0);
put_long (unit->volume + 28, 0); /* lock list */
put_long (unit->volume + 40, (unit->volume + 44) >> 2); /* Name */
namelen = strlen (unit->ui.volname);
put_byte (unit->volume + 44, namelen);
for (i = 0; i < namelen; i++)
put_byte (unit->volume + 45 + i, unit->ui.volname[i]);
/* link into DOS list */
put_long (unit->volume, get_long (dos_info + 4));
put_long (dos_info + 4, unit->volume >> 2);
put_long (unit->volume + 8, unit->port);
put_long (unit->volume + 32, DISK_TYPE);
put_long (pkt + dp_Res1, DOS_TRUE);
fsdb_clean_dir (&unit->rootnode);
return 0;
}
static void
do_info (Unit *unit, dpacket packet, uaecptr info)
{
struct fs_usage fsu;
if (get_fs_usage (unit->ui.rootdir, 0, &fsu) != 0) {
PUT_PCK_RES1 (packet, DOS_FALSE);
PUT_PCK_RES2 (packet, dos_errno ());
return;
}
fsu.fsu_blocks >>= 1;
fsu.fsu_bavail >>= 1;
put_long (info, 0); /* errors */
put_long (info + 4, unit->unit); /* unit number */
put_long (info + 8, unit->ui.readonly ? 80 : 82); /* state */
put_long (info + 12, fsu.fsu_blocks ); /* numblocks */
put_long (info + 16, fsu.fsu_blocks - fsu.fsu_bavail); /* inuse */
put_long (info + 20, 1024); /* bytesperblock */
put_long (info + 24, DISK_TYPE); /* disk type */
put_long (info + 28, unit->volume >> 2); /* volume node */
put_long (info + 32, 0); /* inuse */
PUT_PCK_RES1 (packet, DOS_TRUE);
}
static void
action_disk_info (Unit *unit, dpacket packet)
{
TRACE(("ACTION_DISK_INFO\n"));
do_info(unit, packet, GET_PCK_ARG1 (packet) << 2);
}
static void
action_info (Unit *unit, dpacket packet)
{
TRACE(("ACTION_INFO\n"));
do_info(unit, packet, GET_PCK_ARG2 (packet) << 2);
}
static void free_key (Unit *unit, Key *k)
{
Key *k1;
Key *prev = 0;
for (k1 = unit->keys; k1; k1 = k1->next) {
if (k == k1) {
if (prev)
prev->next = k->next;
else
unit->keys = k->next;
break;
}
prev = k1;
}
if (k->fd >= 0)
close (k->fd);
free(k);
}
static Key *lookup_key (Unit *unit, uae_u32 uniq)
{
Key *k;
/* It's hardly worthwhile to optimize this - most of the time there are
* only one or zero keys. */
for (k = unit->keys; k; k = k->next) {
if (uniq == k->uniq)
return k;
}
write_log ("Error: couldn't find key!\n");
#if 0
exit(1);
/* NOTREACHED */
#endif
/* There isn't much hope we will recover. Unix would kill the process,
* AmigaOS gets killed by it. */
write_log ("Better reset that Amiga - the system is messed up.\n");
return 0;
}
static Key *new_key (Unit *unit)
{
Key *k = (Key *) xmalloc(sizeof(Key));
k->uniq = ++unit->key_uniq;
k->fd = -1;
k->file_pos = 0;
k->next = unit->keys;
unit->keys = k;
return k;
}
static void
dumplock (Unit *unit, uaecptr lock)
{
a_inode *a;
TRACE(("LOCK: 0x%lx", lock));
if (!lock) {
TRACE(("\n"));
return;
}
TRACE(("{ next=0x%lx, mode=%ld, handler=0x%lx, volume=0x%lx, aino %lx ",
get_long (lock) << 2, get_long (lock+8),
get_long (lock+12), get_long (lock+16),
get_long (lock + 4)));
a = lookup_aino (unit, get_long (lock + 4));
if (a == 0) {
TRACE(("not found!"));
} else {
TRACE(("%s", a->nname));
}
TRACE((" }\n"));
}
static a_inode *find_aino (Unit *unit, uaecptr lock, const char *name, uae_u32 *err)
{
a_inode *a;
if (lock) {
a_inode *olda = lookup_aino (unit, get_long (lock + 4));
if (olda == 0) {
/* That's the best we can hope to do. */
a = get_aino (unit, &unit->rootnode, name, err);
} else {
TRACE(("aino: 0x%08lx", (unsigned long int)olda->uniq));
TRACE((" \"%s\"\n", olda->nname));
a = get_aino (unit, olda, name, err);
}
} else {
a = get_aino (unit, &unit->rootnode, name, err);
}
if (a) {
TRACE(("aino=\"%s\"\n", a->nname));
}
return a;
}
static uaecptr make_lock (Unit *unit, uae_u32 uniq, long mode)
{
/* allocate lock from the list kept by the assembly code */
uaecptr lock;
lock = get_long (unit->locklist);
put_long (unit->locklist, get_long (lock));
lock += 4;
put_long (lock + 4, uniq);
put_long (lock + 8, mode);
put_long (lock + 12, unit->port);
put_long (lock + 16, unit->volume >> 2);
/* prepend to lock chain */
put_long (lock, get_long (unit->volume + 28));
put_long (unit->volume + 28, lock >> 2);
DUMPLOCK(unit, lock);
return lock;
}
static void free_lock (Unit *unit, uaecptr lock)
{
if (! lock)
return;
if (lock == get_long (unit->volume + 28) << 2) {
put_long (unit->volume + 28, get_long (lock));
} else {
uaecptr current = get_long (unit->volume + 28);
uaecptr next = 0;
while (current) {
next = get_long (current << 2);
if (lock == next << 2)
break;
current = next;
}
put_long (current << 2, get_long (lock));
}
lock -= 4;
put_long (lock, get_long (unit->locklist));
put_long (unit->locklist, lock);
}
static void
action_lock (Unit *unit, dpacket packet)
{
uaecptr lock = GET_PCK_ARG1 (packet) << 2;
uaecptr name = GET_PCK_ARG2 (packet) << 2;
long mode = GET_PCK_ARG3 (packet);
a_inode *a;
uae_u32 err;
if (mode != SHARED_LOCK && mode != EXCLUSIVE_LOCK) {
TRACE(("Bad mode.\n"));
mode = SHARED_LOCK;
}
TRACE(("ACTION_LOCK(0x%lx, \"%s\", %d)\n", lock, bstr (unit, name), mode));
DUMPLOCK(unit, lock);
a = find_aino (unit, lock, bstr (unit, name), &err);
if (err == 0 && (a->elock || (mode != SHARED_LOCK && a->shlock > 0))) {
err = ERROR_OBJECT_IN_USE;
}
/* Lock() doesn't do access checks. */
if (err != 0) {
PUT_PCK_RES1 (packet, DOS_FALSE);
PUT_PCK_RES2 (packet, err);
return;
}
if (mode == SHARED_LOCK)
a->shlock++;
else
a->elock = 1;
de_recycle_aino (unit, a);
PUT_PCK_RES1 (packet, make_lock (unit, a->uniq, mode) >> 2);
}
static void action_free_lock (Unit *unit, dpacket packet)
{
uaecptr lock = GET_PCK_ARG1 (packet) << 2;
a_inode *a;
TRACE(("ACTION_FREE_LOCK(0x%lx)\n", lock));
DUMPLOCK(unit, lock);
a = lookup_aino (unit, get_long (lock + 4));
if (a == 0) {
PUT_PCK_RES1 (packet, DOS_FALSE);
PUT_PCK_RES2 (packet, ERROR_OBJECT_NOT_AROUND);
return;
}
if (a->elock)
a->elock = 0;
else
a->shlock--;
recycle_aino (unit, a);
free_lock(unit, lock);
PUT_PCK_RES1 (packet, DOS_TRUE);
}
static void
action_dup_lock (Unit *unit, dpacket packet)
{
uaecptr lock = GET_PCK_ARG1 (packet) << 2;
a_inode *a;
TRACE(("ACTION_DUP_LOCK(0x%lx)\n", lock));
DUMPLOCK(unit, lock);
if (!lock) {
PUT_PCK_RES1 (packet, 0);
return;
}
a = lookup_aino (unit, get_long (lock + 4));
if (a == 0) {
PUT_PCK_RES1 (packet, DOS_FALSE);
PUT_PCK_RES2 (packet, ERROR_OBJECT_NOT_AROUND);
return;
}
/* DupLock()ing exclusive locks isn't possible, says the Autodoc, but
* at least the RAM-Handler seems to allow it. Let's see what happens
* if we don't. */
if (a->elock) {
PUT_PCK_RES1 (packet, DOS_FALSE);
PUT_PCK_RES2 (packet, ERROR_OBJECT_IN_USE);
return;
}
a->shlock++;
de_recycle_aino (unit, a);
PUT_PCK_RES1 (packet, make_lock (unit, a->uniq, -2) >> 2);
}
/* convert time_t to/from AmigaDOS time */
const int secs_per_day = 24 * 60 * 60;
const int diff = (8 * 365 + 2) * (24 * 60 * 60);
static void
get_time (time_t t, long* days, long* mins, long* ticks)
{
/* time_t is secs since 1-1-1970 */
/* days since 1-1-1978 */
/* mins since midnight */
/* ticks past minute @ 50Hz */
t -= diff;
*days = t / secs_per_day;
t -= *days * secs_per_day;
*mins = t / 60;
t -= *mins * 60;
*ticks = t * 50;
}
static time_t
put_time (long days, long mins, long ticks)
{
time_t t;
t = ticks / 50;
t += mins * 60;
t += days * secs_per_day;
t += diff;
return t;
}
static void free_exkey (Unit *unit, ExamineKey *ek)
{
if (--ek->aino->exnext_count == 0) {
TRACE (("Freeing ExKey and reducing total_locked from %d by %d\n",
unit->total_locked_ainos, ek->aino->locked_children));
unit->total_locked_ainos -= ek->aino->locked_children;
ek->aino->locked_children = 0;
}
ek->aino = 0;
ek->uniq = 0;
}
static ExamineKey *lookup_exkey (Unit *unit, uae_u32 uniq)
{
ExamineKey *ek;
int i;
ek = unit->examine_keys;
for (i = 0; i < EXKEYS; i++, ek++) {
/* Did we find a free one? */
if (ek->uniq == uniq)
return ek;
}
write_log ("Houston, we have a BIG problem.\n");
return 0;
}
/* This is so sick... who invented ACTION_EXAMINE_NEXT? What did he THINK??? */
static ExamineKey *new_exkey (Unit *unit, a_inode *aino)
{
uae_u32 uniq;
uae_u32 oldest = 0xFFFFFFFE;
ExamineKey *ek, *oldest_ek = 0;
int i;
ek = unit->examine_keys;
for (i = 0; i < EXKEYS; i++, ek++) {
/* Did we find a free one? */
if (ek->aino == 0)
continue;
if (ek->uniq < oldest)
oldest = (oldest_ek = ek)->uniq;
}
ek = unit->examine_keys;
for (i = 0; i < EXKEYS; i++, ek++) {
/* Did we find a free one? */
if (ek->aino == 0)
goto found;
}
/* This message should usually be harmless. */
write_log ("Houston, we have a problem.\n");
free_exkey (unit, oldest_ek);
ek = oldest_ek;
found:
uniq = unit->next_exkey;
if (uniq >= 0xFFFFFFFE) {
/* Things will probably go wrong, but most likely the Amiga will crash
* before this happens because of something else. */
uniq = 1;
}
unit->next_exkey = uniq+1;
ek->aino = aino;
ek->curr_file = 0;
ek->uniq = uniq;
return ek;
}
static void move_exkeys (Unit *unit, a_inode *from, a_inode *to)
{
int i;
unsigned long tmp = 0;
for (i = 0; i < EXKEYS; i++) {
ExamineKey *k = unit->examine_keys + i;
if (k->uniq == 0)
continue;
if (k->aino == from) {
k->aino = to;
tmp++;
}
}
if (tmp != from->exnext_count)
write_log ("filesys.c: Bug in ExNext bookkeeping. BAD.\n");
to->exnext_count = from->exnext_count;
to->locked_children = from->locked_children;
from->exnext_count = 0;
from->locked_children = 0;
}
static void
get_fileinfo (Unit *unit, dpacket packet, uaecptr info, a_inode *aino)
{
struct stat statbuf;
long days, mins, ticks;
int i, n;
char *x;
/* No error checks - this had better work. */
stat (aino->nname, &statbuf);
if (aino->parent == 0) {
x = unit->ui.volname;
put_long (info + 4, 1);
put_long (info + 120, 1);
} else {
/* AmigaOS docs say these have to contain the same value. */
put_long (info + 4, aino->dir ? 2 : -3);
put_long (info + 120, aino->dir ? 2 : -3);
x = aino->aname;
}
TRACE(("name=\"%s\"\n", x));
n = strlen (x);
if (n > 106)
n = 106;
i = 8;
put_byte (info + i, n); i++;
while (n--)
put_byte (info + i, *x), i++, x++;
while (i < 108)
put_byte (info + i, 0), i++;
put_long (info + 116, aino->amigaos_mode);
put_long (info + 124, statbuf.st_size);
#ifdef HAVE_ST_BLOCKS
put_long (info + 128, statbuf.st_blocks);
#else
put_long (info + 128, statbuf.st_size / 512 + 1);
#endif
get_time (statbuf.st_mtime, &days, &mins, &ticks);
put_long (info + 132, days);
put_long (info + 136, mins);
put_long (info + 140, ticks);
if (aino->comment == 0)
put_long (info + 144, 0);
else {
TRACE(("comment=\"%s\"\n", aino->comment));
i = 144;
x = aino->comment;
if (! x)
x = "";
n = strlen (x);
if (n > 78)
n = 78;
put_byte (info + i, n); i++;
while (n--)
put_byte (info + i, *x), i++, x++;
while (i < 224)
put_byte (info + i, 0), i++;
}
PUT_PCK_RES1 (packet, DOS_TRUE);
}
static void action_examine_object (Unit *unit, dpacket packet)
{
uaecptr lock = GET_PCK_ARG1 (packet) << 2;
uaecptr info = GET_PCK_ARG2 (packet) << 2;
a_inode *aino = 0;
TRACE(("ACTION_EXAMINE_OBJECT(0x%lx,0x%lx)\n", lock, info));
DUMPLOCK(unit, lock);
if (lock != 0)
aino = lookup_aino (unit, get_long (lock + 4));
if (aino == 0)
aino = &unit->rootnode;
get_fileinfo (unit, packet, info, aino);
if (aino->dir) {
put_long (info, 0xFFFFFFFF);
} else
put_long (info, 0);
}
/* Read a directory's contents, create a_inodes for each file, and
mark them as locked in memory so that recycle_aino will not reap
them.
We do this to avoid problems with the host OS: we don't want to
leave the directory open on the host side until all ExNext()s have
finished - they may never finish! */
static void populate_directory (Unit *unit, a_inode *base)
{
DIR *d = opendir (base->nname);
a_inode *aino;
for (aino = base->child; aino; aino = aino->sibling) {
base->locked_children++;
unit->total_locked_ainos++;
}
TRACE(("Populating directory, child %p, locked_children %d\n",
base->child, base->locked_children));
for (;;) {
struct dirent de_space;
struct dirent *de;
uae_u32 err;
/* Find next file that belongs to the Amiga fs (skipping things
like "..", "." etc. */
do {
de = my_readdir (d, &de_space);
} while (de && fsdb_name_invalid (de->d_name));
if (! de)
break;
/* This calls init_child_aino, which will notice that the parent is
being ExNext()ed, and it will increment the locked counts. */
aino = lookup_child_aino_for_exnext (unit, base, de->d_name, &err);
}
closedir (d);
}
static void do_examine (Unit *unit, dpacket packet, ExamineKey *ek, uaecptr info)
{
if (ek->curr_file == 0)
goto no_more_entries;
get_fileinfo (unit, packet, info, ek->curr_file);
ek->curr_file = ek->curr_file->sibling;
TRACE (("curr_file set to %p %s\n", ek->curr_file,
ek->curr_file ? ek->curr_file->aname : "NULL"));
return;
no_more_entries:
TRACE(("no more entries\n"));
free_exkey (unit, ek);
PUT_PCK_RES1 (packet, DOS_FALSE);
PUT_PCK_RES2 (packet, ERROR_NO_MORE_ENTRIES);
}
static void action_examine_next (Unit *unit, dpacket packet)
{
uaecptr lock = GET_PCK_ARG1 (packet) << 2;
uaecptr info = GET_PCK_ARG2 (packet) << 2;
a_inode *aino = 0;
ExamineKey *ek;
uae_u32 uniq;
TRACE(("ACTION_EXAMINE_NEXT(0x%lx,0x%lx)\n", lock, info));
DUMPLOCK(unit, lock);
if (lock != 0)
aino = lookup_aino (unit, get_long (lock + 4));
if (aino == 0)
aino = &unit->rootnode;
uniq = get_long (info);
if (uniq == 0) {
write_log ("ExNext called for a file! (Houston?)\n");
goto no_more_entries;
} else if (uniq == 0xFFFFFFFE)
goto no_more_entries;
else if (uniq == 0xFFFFFFFF) {
TRACE(("Creating new ExKey\n"));
ek = new_exkey (unit, aino);
if (ek) {
if (aino->exnext_count++ == 0)
populate_directory (unit, aino);
}
ek->curr_file = aino->child;
TRACE(("Initial curr_file: %p %s\n", ek->curr_file,
ek->curr_file ? ek->curr_file->aname : "NULL"));
} else {
TRACE(("Looking up ExKey\n"));
ek = lookup_exkey (unit, get_long (info));
}
if (ek == 0) {
write_log ("Couldn't find a matching ExKey. Prepare for trouble.\n");
goto no_more_entries;
}
put_long (info, ek->uniq);
do_examine (unit, packet, ek, info);
return;
no_more_entries:
PUT_PCK_RES1 (packet, DOS_FALSE);
PUT_PCK_RES2 (packet, ERROR_NO_MORE_ENTRIES);
}
static void do_find (Unit *unit, dpacket packet, int mode, int create, int fallback)
{
uaecptr fh = GET_PCK_ARG1 (packet) << 2;
uaecptr lock = GET_PCK_ARG2 (packet) << 2;
uaecptr name = GET_PCK_ARG3 (packet) << 2;
a_inode *aino;
Key *k;
int fd;
uae_u32 err;
mode_t openmode;
int aino_created = 0;
TRACE(("ACTION_FIND_*(0x%lx,0x%lx,\"%s\",%d)\n", fh, lock, bstr (unit, name), mode));
DUMPLOCK(unit, lock);
aino = find_aino (unit, lock, bstr (unit, name), &err);
if (aino == 0 || (err != 0 && err != ERROR_OBJECT_NOT_AROUND)) {
/* Whatever it is, we can't handle it. */
PUT_PCK_RES1 (packet, DOS_FALSE);
PUT_PCK_RES2 (packet, err);
return;
}
if (err == 0) {
/* Object exists. */
if (aino->dir) {
PUT_PCK_RES1 (packet, DOS_FALSE);
PUT_PCK_RES2 (packet, ERROR_OBJECT_WRONG_TYPE);
return;
}
if (aino->elock || (create == 2 && aino->shlock > 0)) {
PUT_PCK_RES1 (packet, DOS_FALSE);
PUT_PCK_RES2 (packet, ERROR_OBJECT_IN_USE);
return;
}
if (create == 2 && (aino->amigaos_mode & A_FIBF_DELETE) != 0) {
PUT_PCK_RES1 (packet, DOS_FALSE);
PUT_PCK_RES2 (packet, ERROR_DELETE_PROTECTED);
return;
}
if (create != 2) {
if ((((mode & aino->amigaos_mode) & A_FIBF_WRITE) != 0 || unit->ui.readonly)
&& fallback)
{
mode &= ~A_FIBF_WRITE;
}
/* Kick 1.3 doesn't check read and write access bits - maybe it would be
* simpler just not to do that either. */
if ((mode & A_FIBF_WRITE) != 0 && unit->ui.readonly) {
PUT_PCK_RES1 (packet, DOS_FALSE);
PUT_PCK_RES2 (packet, ERROR_DISK_WRITE_PROTECTED);
return;
}
if (((mode & aino->amigaos_mode) & A_FIBF_WRITE) != 0
|| mode == 0)
{
PUT_PCK_RES1 (packet, DOS_FALSE);
PUT_PCK_RES2 (packet, ERROR_WRITE_PROTECTED);
return;
}
if (((mode & aino->amigaos_mode) & A_FIBF_READ) != 0) {
PUT_PCK_RES1 (packet, DOS_FALSE);
PUT_PCK_RES2 (packet, ERROR_READ_PROTECTED);
return;
}
}
} else if (create == 0) {
PUT_PCK_RES1 (packet, DOS_FALSE);
PUT_PCK_RES2 (packet, err);
return;
} else {
/* Object does not exist. aino points to containing directory. */
aino = create_child_aino (unit, aino, my_strdup (bstr_cut (unit, name)), 0);
if (aino == 0) {
PUT_PCK_RES1 (packet, DOS_FALSE);
PUT_PCK_RES2 (packet, ERROR_DISK_IS_FULL); /* best we can do */
return;
}
aino_created = 1;
}
prepare_for_open (aino->nname);
openmode = (((mode & A_FIBF_READ) == 0 ? O_WRONLY
: (mode & A_FIBF_WRITE) == 0 ? O_RDONLY
: O_RDWR)
| (create ? O_CREAT : 0)
| (create == 2 ? O_TRUNC : 0));
fd = open (aino->nname, openmode | O_BINARY, 0777);
if (fd < 0) {
if (aino_created)
delete_aino (unit, aino);
PUT_PCK_RES1 (packet, DOS_FALSE);
PUT_PCK_RES2 (packet, dos_errno ());
return;
}
k = new_key (unit);
k->fd = fd;
k->aino = aino;
put_long (fh+36, k->uniq);
if (create == 2)
aino->elock = 1;
else
aino->shlock++;
de_recycle_aino (unit, aino);
PUT_PCK_RES1 (packet, DOS_TRUE);
}
static void
action_fh_from_lock (Unit *unit, dpacket packet)
{
uaecptr fh = GET_PCK_ARG1 (packet) << 2;
uaecptr lock = GET_PCK_ARG2 (packet) << 2;
a_inode *aino;
Key *k;
int fd;
mode_t openmode;
int mode;
TRACE(("ACTION_FH_FROM_LOCK(0x%lx,0x%lx)\n",fh,lock));
DUMPLOCK(unit,lock);
if (!lock) {
PUT_PCK_RES1 (packet, DOS_FALSE);
PUT_PCK_RES2 (packet, 0);
return;
}
aino = lookup_aino (unit, get_long (lock + 4));
if (aino == 0)
aino = &unit->rootnode;
mode = aino->amigaos_mode; /* Use same mode for opened filehandle as existing Lock() */
prepare_for_open (aino->nname);
TRACE ((" mode is %d\n", mode));
openmode = (((mode & A_FIBF_READ) ? O_WRONLY
: (mode & A_FIBF_WRITE) ? O_RDONLY
: O_RDWR));
/* the files on CD really can have the write-bit set. */
if (unit->ui.readonly)
openmode = O_RDONLY;
fd = open (aino->nname, openmode | O_BINARY, 0777);
if (fd < 0) {
PUT_PCK_RES1 (packet, DOS_FALSE);
PUT_PCK_RES2 (packet, dos_errno());
return;
}
k = new_key (unit);
k->fd = fd;
k->aino = aino;
put_long (fh+36, k->uniq);
/* I don't think I need to play with shlock count here, because I'm
opening from an existing lock ??? */
/* Is this right? I don't completely understand how this works. Do I
also need to free_lock() my lock, since nobody else is going to? */
de_recycle_aino (unit, aino);
PUT_PCK_RES1 (packet, DOS_TRUE);
/* PUT_PCK_RES2 (packet, k->uniq); - this shouldn't be necessary, try without it */
}
static void
action_find_input (Unit *unit, dpacket packet)
{
do_find(unit, packet, A_FIBF_READ|A_FIBF_WRITE, 0, 1);
}
static void
action_find_output (Unit *unit, dpacket packet)
{
if (unit->ui.readonly) {
PUT_PCK_RES1 (packet, DOS_FALSE);
PUT_PCK_RES2 (packet, ERROR_DISK_WRITE_PROTECTED);
return;
}
do_find(unit, packet, A_FIBF_READ|A_FIBF_WRITE, 2, 0);
}
static void
action_find_write (Unit *unit, dpacket packet)
{
if (unit->ui.readonly) {
PUT_PCK_RES1 (packet, DOS_FALSE);
PUT_PCK_RES2 (packet, ERROR_DISK_WRITE_PROTECTED);
return;
}
do_find(unit, packet, A_FIBF_READ|A_FIBF_WRITE, 1, 0);
}
static void
action_end (Unit *unit, dpacket packet)
{
Key *k;
TRACE(("ACTION_END(0x%lx)\n", GET_PCK_ARG1 (packet)));
k = lookup_key (unit, GET_PCK_ARG1 (packet));
if (k != 0) {
if (k->aino->elock)
k->aino->elock = 0;
else
k->aino->shlock--;
recycle_aino (unit, k->aino);
free_key (unit, k);
}
PUT_PCK_RES1 (packet, DOS_TRUE);
PUT_PCK_RES2 (packet, 0);
}
static void
action_read (Unit *unit, dpacket packet)
{
Key *k = lookup_key (unit, GET_PCK_ARG1 (packet));
uaecptr addr = GET_PCK_ARG2 (packet);
long size = (uae_s32)GET_PCK_ARG3 (packet);
int actual;
if (k == 0) {
PUT_PCK_RES1 (packet, DOS_FALSE);
/* PUT_PCK_RES2 (packet, EINVAL); */
return;
}
TRACE(("ACTION_READ(%s,0x%lx,%ld)\n",k->aino->nname,addr,size));
#ifdef RELY_ON_LOADSEG_DETECTION
/* HACK HACK HACK HACK
* Try to detect a LoadSeg() */
if (k->file_pos == 0 && size >= 4) {
unsigned char buf[4];
off_t currpos = lseek(k->fd, 0, SEEK_CUR);
read(k->fd, buf, 4);
lseek(k->fd, currpos, SEEK_SET);
if (buf[0] == 0 && buf[1] == 0 && buf[2] == 3 && buf[3] == 0xF3)
possible_loadseg();
}
#endif
if (valid_address (addr, size)) {
uae_u8 *realpt;
realpt = get_real_address (addr);
actual = read(k->fd, realpt, size);
if (actual == 0) {
PUT_PCK_RES1 (packet, 0);
PUT_PCK_RES2 (packet, 0);
} else if (actual < 0) {
PUT_PCK_RES1 (packet, 0);
PUT_PCK_RES2 (packet, dos_errno());
} else {
PUT_PCK_RES1 (packet, actual);
k->file_pos += actual;
}
} else {
char *buf;
write_log ("unixfs warning: Bad pointer passed for read: %08x\n", addr);
/* ugh this is inefficient but easy */
buf = (char *)malloc(size);
if (!buf) {
PUT_PCK_RES1 (packet, -1);
PUT_PCK_RES2 (packet, ERROR_NO_FREE_STORE);
return;
}
actual = read(k->fd, buf, size);
if (actual < 0) {
PUT_PCK_RES1 (packet, 0);
PUT_PCK_RES2 (packet, dos_errno());
} else {
int i;
PUT_PCK_RES1 (packet, actual);
for (i = 0; i < actual; i++)
put_byte (addr + i, buf[i]);
k->file_pos += actual;
}
free (buf);
}
}
static void
action_write (Unit *unit, dpacket packet)
{
Key *k = lookup_key (unit, GET_PCK_ARG1 (packet));
uaecptr addr = GET_PCK_ARG2 (packet);
long size = GET_PCK_ARG3 (packet);
char *buf;
int i;
if (k == 0) {
PUT_PCK_RES1 (packet, DOS_FALSE);
/* PUT_PCK_RES2 (packet, EINVAL); */
return;
}
TRACE(("ACTION_WRITE(%s,0x%lx,%ld)\n",k->aino->nname,addr,size));
if (unit->ui.readonly) {
PUT_PCK_RES1 (packet, DOS_FALSE);
PUT_PCK_RES2 (packet, ERROR_DISK_WRITE_PROTECTED);
return;
}
/* ugh this is inefficient but easy */
buf = (char *)malloc(size);
if (!buf) {
PUT_PCK_RES1 (packet, -1);
PUT_PCK_RES2 (packet, ERROR_NO_FREE_STORE);
return;
}
for (i = 0; i < size; i++)
buf[i] = get_byte (addr + i);
PUT_PCK_RES1 (packet, write(k->fd, buf, size));
if (GET_PCK_RES1 (packet) != size)
PUT_PCK_RES2 (packet, dos_errno ());
if (GET_PCK_RES1 (packet) >= 0)
k->file_pos += GET_PCK_RES1 (packet);
free (buf);
}
static void
action_seek (Unit *unit, dpacket packet)
{
Key *k = lookup_key (unit, GET_PCK_ARG1 (packet));
long pos = (uae_s32)GET_PCK_ARG2 (packet);
long mode = (uae_s32)GET_PCK_ARG3 (packet);
off_t res;
long old;
int whence = SEEK_CUR;
if (k == 0) {
PUT_PCK_RES1 (packet, -1);
PUT_PCK_RES2 (packet, ERROR_INVALID_LOCK);
return;
}
if (mode > 0) whence = SEEK_END;
if (mode < 0) whence = SEEK_SET;
TRACE(("ACTION_SEEK(%s,%d,%d)\n", k->aino->nname, pos, mode));
old = lseek (k->fd, 0, SEEK_CUR);
{
uae_s32 temppos;
long filesize = lseek (k->fd, 0, SEEK_END);
lseek (k->fd, old, SEEK_SET);
if (whence == SEEK_CUR) temppos = old + pos;
if (whence == SEEK_SET) temppos = pos;
if (whence == SEEK_END) temppos = filesize + pos;
if (filesize < temppos) {
res = -1;
PUT_PCK_RES1 (packet,res);
PUT_PCK_RES2 (packet, ERROR_SEEK_ERROR);
return;
}
}
res = lseek (k->fd, pos, whence);
if (-1 == res) {
PUT_PCK_RES1 (packet, res);
PUT_PCK_RES2 (packet, ERROR_SEEK_ERROR);
} else
PUT_PCK_RES1 (packet, old);
k->file_pos = res;
}
static void
action_set_protect (Unit *unit, dpacket packet)
{
uaecptr lock = GET_PCK_ARG2 (packet) << 2;
uaecptr name = GET_PCK_ARG3 (packet) << 2;
uae_u32 mask = GET_PCK_ARG4 (packet);
a_inode *a;
uae_u32 err;
TRACE(("ACTION_SET_PROTECT(0x%lx,\"%s\",0x%lx)\n", lock, bstr (unit, name), mask));
if (unit->ui.readonly) {
PUT_PCK_RES1 (packet, DOS_FALSE);
PUT_PCK_RES2 (packet, ERROR_DISK_WRITE_PROTECTED);
return;
}
a = find_aino (unit, lock, bstr (unit, name), &err);
if (err != 0) {
PUT_PCK_RES1 (packet, DOS_FALSE);
PUT_PCK_RES2 (packet, err);
return;
}
err = fsdb_set_file_attrs (a, mask);
if (err != 0) {
PUT_PCK_RES1 (packet, DOS_FALSE);
PUT_PCK_RES2 (packet, err);
} else {
PUT_PCK_RES1 (packet, DOS_TRUE);
}
}
static void action_set_comment (Unit * unit, dpacket packet)
{
uaecptr lock = GET_PCK_ARG2 (packet) << 2;
uaecptr name = GET_PCK_ARG3 (packet) << 2;
uaecptr comment = GET_PCK_ARG4 (packet) << 2;
char *commented;
a_inode *a;
uae_u32 err;
long res1, res2;
if (unit->ui.readonly) {
PUT_PCK_RES1 (packet, DOS_FALSE);
PUT_PCK_RES2 (packet, ERROR_DISK_WRITE_PROTECTED);
return;
}
commented = bstr (unit, comment);
commented = strlen (commented) > 0 ? my_strdup (commented) : 0;
TRACE (("ACTION_SET_COMMENT(0x%lx,\"%s\")\n", lock, commented));
a = find_aino (unit, lock, bstr (unit, name), &err);
if (err != 0) {
PUT_PCK_RES1 (packet, DOS_FALSE);
PUT_PCK_RES2 (packet, err);
maybe_free_and_out:
if (commented)
free (commented);
return;
}
PUT_PCK_RES1 (packet, DOS_TRUE);
PUT_PCK_RES2 (packet, 0);
if (a->comment == 0 && commented == 0)
goto maybe_free_and_out;
if (a->comment != 0 && commented != 0 && strcmp (a->comment, commented) == 0)
goto maybe_free_and_out;
if (a->comment)
free (a->comment);
a->comment = commented;
a->dirty = 1;
}
static void
action_same_lock (Unit *unit, dpacket packet)
{
uaecptr lock1 = GET_PCK_ARG1 (packet) << 2;
uaecptr lock2 = GET_PCK_ARG2 (packet) << 2;
TRACE(("ACTION_SAME_LOCK(0x%lx,0x%lx)\n",lock1,lock2));
DUMPLOCK(unit, lock1); DUMPLOCK(unit, lock2);
if (!lock1 || !lock2) {
PUT_PCK_RES1 (packet, lock1 == lock2 ? DOS_TRUE : DOS_FALSE);
} else {
PUT_PCK_RES1 (packet, get_long (lock1 + 4) == get_long (lock2 + 4) ? DOS_TRUE : DOS_FALSE);
}
}
static void
action_change_mode (Unit *unit, dpacket packet)
{
#define CHANGE_LOCK 0
#define CHANGE_FH 1
/* will be CHANGE_FH or CHANGE_LOCK value */
long type = GET_PCK_ARG1 (packet);
/* either a file-handle or lock */
uaecptr object = GET_PCK_ARG2 (packet) << 2;
/* will be EXCLUSIVE_LOCK/SHARED_LOCK if CHANGE_LOCK,
* or MODE_OLDFILE/MODE_NEWFILE/MODE_READWRITE if CHANGE_FH */
long mode = GET_PCK_ARG3 (packet);
unsigned long uniq;
a_inode *a = NULL, *olda = NULL;
uae_u32 err = 0;
TRACE(("ACTION_CHANGE_MODE(0x%lx,%d,%d)\n",object,type,mode));
if (! object
|| (type != CHANGE_FH && type != CHANGE_LOCK))
{
PUT_PCK_RES1 (packet, DOS_FALSE);
PUT_PCK_RES2 (packet, ERROR_INVALID_LOCK);
return;
}
/* @@@ Brian: shouldn't this be good enough to support
CHANGE_FH? */
if (type == CHANGE_FH)
mode = (mode == 1006 ? -1 : -2);
if (type == CHANGE_LOCK)
uniq = get_long (object + 4);
else {
Key *k = lookup_key (unit, object);
if (!k) {
PUT_PCK_RES1 (packet, DOS_FALSE);
PUT_PCK_RES2 (packet, ERROR_OBJECT_NOT_AROUND);
return;
}
uniq = k->aino->uniq;
}
a = lookup_aino (unit, uniq);
if (! a)
err = ERROR_INVALID_LOCK;
else {
if (mode == -1) {
if (a->shlock > 1)
err = ERROR_OBJECT_IN_USE;
else {
a->shlock = 0;
a->elock = 1;
}
} else { /* Must be SHARED_LOCK == -2 */
a->elock = 0;
a->shlock++;
}
}
if (err) {
PUT_PCK_RES1 (packet, DOS_FALSE);
PUT_PCK_RES2 (packet, err);
return;
} else {
de_recycle_aino (unit, a);
PUT_PCK_RES1 (packet, DOS_TRUE);
}
}
static void
action_parent_common (Unit *unit, dpacket packet, unsigned long uniq)
{
a_inode *olda = lookup_aino (unit, uniq);
if (olda == 0) {
PUT_PCK_RES1 (packet, DOS_FALSE);
PUT_PCK_RES2 (packet, ERROR_INVALID_LOCK);
return;
}
if (olda->parent == 0) {
PUT_PCK_RES1 (packet, 0);
PUT_PCK_RES2 (packet, 0);
return;
}
if (olda->parent->elock) {
PUT_PCK_RES1 (packet, DOS_FALSE);
PUT_PCK_RES2 (packet, ERROR_OBJECT_IN_USE);
return;
}
olda->parent->shlock++;
de_recycle_aino (unit, olda->parent);
PUT_PCK_RES1 (packet, make_lock (unit, olda->parent->uniq, -2) >> 2);
}
static void
action_parent_fh (Unit *unit, dpacket packet)
{
Key *k = lookup_key (unit, GET_PCK_ARG1 (packet));
if (!k) {
PUT_PCK_RES1 (packet, DOS_FALSE);
PUT_PCK_RES2 (packet, ERROR_OBJECT_NOT_AROUND);
return;
}
action_parent_common (unit, packet, k->aino->uniq);
}
static void
action_parent (Unit *unit, dpacket packet)
{
uaecptr lock = GET_PCK_ARG1 (packet) << 2;
TRACE(("ACTION_PARENT(0x%lx)\n",lock));
if (!lock) {
PUT_PCK_RES1 (packet, 0);
PUT_PCK_RES2 (packet, 0);
return;
}
action_parent_common (unit, packet, get_long (lock + 4));
}
static void
action_create_dir (Unit *unit, dpacket packet)
{
uaecptr lock = GET_PCK_ARG1 (packet) << 2;
uaecptr name = GET_PCK_ARG2 (packet) << 2;
a_inode *aino;
uae_u32 err;
TRACE(("ACTION_CREATE_DIR(0x%lx,\"%s\")\n", lock, bstr (unit, name)));
if (unit->ui.readonly) {
PUT_PCK_RES1 (packet, DOS_FALSE);
PUT_PCK_RES2 (packet, ERROR_DISK_WRITE_PROTECTED);
return;
}
aino = find_aino (unit, lock, bstr (unit, name), &err);
if (aino == 0 || (err != 0 && err != ERROR_OBJECT_NOT_AROUND)) {
PUT_PCK_RES1 (packet, DOS_FALSE);
PUT_PCK_RES2 (packet, err);
return;
}
if (err == 0) {
/* Object exists. */
PUT_PCK_RES1 (packet, DOS_FALSE);
PUT_PCK_RES2 (packet, ERROR_OBJECT_EXISTS);
return;
}
/* Object does not exist. aino points to containing directory. */
aino = create_child_aino (unit, aino, my_strdup (bstr_cut (unit, name)), 1);
if (aino == 0) {
PUT_PCK_RES1 (packet, DOS_FALSE);
PUT_PCK_RES2 (packet, ERROR_DISK_IS_FULL); /* best we can do */
return;
}
if (mkdir (aino->nname, 0777) == -1) {
PUT_PCK_RES1 (packet, DOS_FALSE);
PUT_PCK_RES2 (packet, dos_errno());
return;
}
aino->shlock = 1;
de_recycle_aino (unit, aino);
PUT_PCK_RES1 (packet, make_lock (unit, aino->uniq, -2) >> 2);
}
static void
action_examine_fh (Unit *unit, dpacket packet)
{
Key *k;
a_inode *aino = 0;
uaecptr info = GET_PCK_ARG2 (packet) << 2;
TRACE(("ACTION_EXAMINE_FH(0x%lx,0x%lx)\n",
GET_PCK_ARG1 (packet), GET_PCK_ARG2 (packet) ));
k = lookup_key (unit, GET_PCK_ARG1 (packet));
if (k != 0)
aino = k->aino;
if (aino == 0)
aino = &unit->rootnode;
get_fileinfo (unit, packet, info, aino);
if (aino->dir)
put_long (info, 0xFFFFFFFF);
else
put_long (info, 0);
}
/* For a nice example of just how contradictory documentation can be, see the
* Autodoc for DOS:SetFileSize and the Packets.txt description of this packet...
* This implementation tries to mimic the behaviour of the Kick 3.1 ramdisk
* (which seems to match the Autodoc description). */
static void
action_set_file_size (Unit *unit, dpacket packet)
{
Key *k, *k1;
off_t offset = GET_PCK_ARG2 (packet);
long mode = (uae_s32)GET_PCK_ARG3 (packet);
int whence = SEEK_CUR;
if (mode > 0) whence = SEEK_END;
if (mode < 0) whence = SEEK_SET;
TRACE(("ACTION_SET_FILE_SIZE(0x%lx, %d, 0x%x)\n", GET_PCK_ARG1 (packet), offset, mode));
k = lookup_key (unit, GET_PCK_ARG1 (packet));
if (k == 0) {
PUT_PCK_RES1 (packet, DOS_TRUE);
PUT_PCK_RES2 (packet, ERROR_OBJECT_NOT_AROUND);
return;
}
/* If any open files have file pointers beyond this size, truncate only
* so far that these pointers do not become invalid. */
for (k1 = unit->keys; k1; k1 = k1->next) {
if (k != k1 && k->aino == k1->aino) {
if (k1->file_pos > offset)
offset = k1->file_pos;
}
}
/* Write one then truncate: that should give the right size in all cases. */
offset = lseek (k->fd, offset, whence);
write (k->fd, /* whatever */(char *)&k1, 1);
if (k->file_pos > offset)
k->file_pos = offset;
lseek (k->fd, k->file_pos, SEEK_SET);
/* Brian: no bug here; the file _must_ be one byte too large after writing
The write is supposed to guarantee that the file can't be smaller than
the requested size, the truncate guarantees that it can't be larger.
If we were to write one byte earlier we'd clobber file data. */
if (truncate (k->aino->nname, offset) == -1) {
PUT_PCK_RES1 (packet, DOS_FALSE);
PUT_PCK_RES2 (packet, dos_errno ());
return;
}
PUT_PCK_RES1 (packet, offset);
PUT_PCK_RES2 (packet, 0);
}
static void
action_delete_object (Unit *unit, dpacket packet)
{
uaecptr lock = GET_PCK_ARG1 (packet) << 2;
uaecptr name = GET_PCK_ARG2 (packet) << 2;
a_inode *a;
uae_u32 err;
TRACE(("ACTION_DELETE_OBJECT(0x%lx,\"%s\")\n", lock, bstr (unit, name)));
if (unit->ui.readonly) {
PUT_PCK_RES1 (packet, DOS_FALSE);
PUT_PCK_RES2 (packet, ERROR_DISK_WRITE_PROTECTED);
return;
}
a = find_aino (unit, lock, bstr (unit, name), &err);
if (err != 0) {
PUT_PCK_RES1 (packet, DOS_FALSE);
PUT_PCK_RES2 (packet, err);
return;
}
if (a->amigaos_mode & A_FIBF_DELETE) {
PUT_PCK_RES1 (packet, DOS_FALSE);
PUT_PCK_RES2 (packet, ERROR_DELETE_PROTECTED);
return;
}
if (a->shlock > 0 || a->elock) {
PUT_PCK_RES1 (packet, DOS_FALSE);
PUT_PCK_RES2 (packet, ERROR_OBJECT_IN_USE);
return;
}
if (a->dir) {
/* This should take care of removing the fsdb if no files remain. */
fsdb_dir_writeback (a);
if (rmdir (a->nname) == -1) {
PUT_PCK_RES1 (packet, DOS_FALSE);
PUT_PCK_RES2 (packet, dos_errno());
return;
}
} else {
if (unlink (a->nname) == -1) {
PUT_PCK_RES1 (packet, DOS_FALSE);
PUT_PCK_RES2 (packet, dos_errno());
return;
}
}
if (a->child != 0) {
write_log ("Serious error in action_delete_object.\n");
} else {
delete_aino (unit, a);
}
PUT_PCK_RES1 (packet, DOS_TRUE);
}
static void
action_set_date (Unit *unit, dpacket packet)
{
uaecptr lock = GET_PCK_ARG2 (packet) << 2;
uaecptr name = GET_PCK_ARG3 (packet) << 2;
uaecptr date = GET_PCK_ARG4 (packet);
a_inode *a;
struct utimbuf ut;
uae_u32 err;
TRACE(("ACTION_SET_DATE(0x%lx,\"%s\")\n", lock, bstr (unit, name)));
if (unit->ui.readonly) {
PUT_PCK_RES1 (packet, DOS_FALSE);
PUT_PCK_RES2 (packet, ERROR_DISK_WRITE_PROTECTED);
return;
}
ut.actime = ut.modtime = put_time(get_long (date), get_long (date + 4),
get_long (date + 8));
a = find_aino (unit, lock, bstr (unit, name), &err);
if (err == 0 && utime (a->nname, &ut) == -1)
err = dos_errno ();
if (err != 0) {
PUT_PCK_RES1 (packet, DOS_FALSE);
PUT_PCK_RES2 (packet, err);
} else
PUT_PCK_RES1 (packet, DOS_TRUE);
}
static void
action_rename_object (Unit *unit, dpacket packet)
{
uaecptr lock1 = GET_PCK_ARG1 (packet) << 2;
uaecptr name1 = GET_PCK_ARG2 (packet) << 2;
uaecptr lock2 = GET_PCK_ARG3 (packet) << 2;
uaecptr name2 = GET_PCK_ARG4 (packet) << 2;
a_inode *a1, *a2;
uae_u32 err1, err2;
TRACE(("ACTION_RENAME_OBJECT(0x%lx,\"%s\",", lock1, bstr (unit, name1)));
TRACE(("0x%lx,\"%s\")\n", lock2, bstr (unit, name2)));
if (unit->ui.readonly) {
PUT_PCK_RES1 (packet, DOS_FALSE);
PUT_PCK_RES2 (packet, ERROR_DISK_WRITE_PROTECTED);
return;
}
a1 = find_aino (unit, lock1, bstr (unit, name1), &err1);
if (err1 != 0) {
PUT_PCK_RES1 (packet, DOS_FALSE);
PUT_PCK_RES2 (packet, err1);
return;
}
/* See whether the other name already exists in the filesystem. */
a2 = find_aino (unit, lock2, bstr (unit, name2), &err2);
if (a2 == a1) {
/* Renaming to the same name, but possibly different case. */
if (strcmp (a1->aname, bstr_cut (unit, name2)) == 0) {
/* Exact match -> do nothing. */
PUT_PCK_RES1 (packet, DOS_TRUE);
return;
}
a2 = a2->parent;
} else if (a2 == 0 || err2 != ERROR_OBJECT_NOT_AROUND) {
PUT_PCK_RES1 (packet, DOS_FALSE);
PUT_PCK_RES2 (packet, err2 == 0 ? ERROR_OBJECT_EXISTS : err2);
return;
}
a2 = create_child_aino (unit, a2, bstr_cut (unit, name2), a1->dir);
if (a2 == 0) {
PUT_PCK_RES1 (packet, DOS_FALSE);
PUT_PCK_RES2 (packet, ERROR_DISK_IS_FULL); /* best we can do */
return;
}
/* @@@ what should we do if there are locks on a1? */
if (-1 == rename (a1->nname, a2->nname)) {
PUT_PCK_RES1 (packet, DOS_FALSE);
PUT_PCK_RES2 (packet, dos_errno ());
return;
}
a2->comment = a1->comment;
a1->comment = 0;
a2->amigaos_mode = a1->amigaos_mode;
a2->uniq = a1->uniq;
move_exkeys (unit, a1, a2);
move_aino_children (unit, a1, a2);
delete_aino (unit, a1);
PUT_PCK_RES1 (packet, DOS_TRUE);
}
static void
action_current_volume (Unit *unit, dpacket packet)
{
PUT_PCK_RES1 (packet, unit->volume >> 2);
}
static void
action_rename_disk (Unit *unit, dpacket packet)
{
uaecptr name = GET_PCK_ARG1 (packet) << 2;
int i;
int namelen;
TRACE(("ACTION_RENAME_DISK(\"%s\")\n", bstr (unit, name)));
if (unit->ui.readonly) {
PUT_PCK_RES1 (packet, DOS_FALSE);
PUT_PCK_RES2 (packet, ERROR_DISK_WRITE_PROTECTED);
return;
}
/* get volume name */
namelen = get_byte (name); name++;
free (unit->ui.volname);
unit->ui.volname = (char *) xmalloc (namelen + 1);
for (i = 0; i < namelen; i++, name++)
unit->ui.volname[i] = get_byte (name);
unit->ui.volname[i] = 0;
put_byte (unit->volume + 44, namelen);
for (i = 0; i < namelen; i++)
put_byte (unit->volume + 45 + i, unit->ui.volname[i]);
PUT_PCK_RES1 (packet, DOS_TRUE);
}
static void
action_is_filesystem (Unit *unit, dpacket packet)
{
PUT_PCK_RES1 (packet, DOS_TRUE);
}
static void
action_flush (Unit *unit, dpacket packet)
{
/* sync(); */ /* pretty drastic, eh */
PUT_PCK_RES1 (packet, DOS_TRUE);
}
/* We don't want multiple interrupts to be active at the same time. I don't
* know whether AmigaOS takes care of that, but this does. */
static uae_sem_t singlethread_int_sem;
static uae_u32 exter_int_helper (TrapContext *dummy)
{
UnitInfo *uip = current_mountinfo->ui;
uaecptr port;
static int unit_no;
switch (m68k_dreg (regs, 0)) {
case 0:
/* Determine whether a given EXTER interrupt is for us. */
if (uae_int_requested) {
if (uae_sem_trywait (&singlethread_int_sem) != 0)
/* Pretend it isn't for us. We might get it again later. */
return 0;
/* Clear the interrupt flag _before_ we do any processing.
* That way, we can get too many interrupts, but never not
* enough. */
uae_int_requested = 0;
unit_no = 0;
return 1;
}
return 0;
case 1:
/* Release a message_lock. This is called as soon as the message is
* received by the assembly code. We use the opportunity to check
* whether we have some locks that we can give back to the assembler
* code.
* Note that this is called from the main loop, unlike the other cases
* in this switch statement which are called from the interrupt handler.
*/
#ifdef UAE_FILESYS_THREADS
{
Unit *unit = find_unit (m68k_areg (regs, 5));
unit->cmds_complete = unit->cmds_acked;
while (comm_pipe_has_data (unit->ui.back_pipe)) {
uaecptr locks, lockend;
locks = read_comm_pipe_int_blocking (unit->ui.back_pipe);
lockend = locks;
while (get_long (lockend) != 0)
lockend = get_long (lockend);
put_long (lockend, get_long (m68k_areg (regs, 3)));
put_long (m68k_areg (regs, 3), locks);
}
}
#else
write_log ("exter_int_helper should not be called with arg 1!\n");
#endif
break;
case 2:
/* Find work that needs to be done:
* return d0 = 0: none
* d0 = 1: PutMsg(), port in a0, message in a1
* d0 = 2: Signal(), task in a1, signal set in d1
* d0 = 3: ReplyMsg(), message in a1 */
#ifdef SUPPORT_THREADS
/* First, check signals/messages */
while (comm_pipe_has_data (&native2amiga_pending)) {
switch (read_comm_pipe_int_blocking (&native2amiga_pending)) {
case 0: /* Signal() */
m68k_areg (regs, 1) = read_comm_pipe_u32_blocking (&native2amiga_pending);
m68k_dreg (regs, 1) = read_comm_pipe_u32_blocking (&native2amiga_pending);
return 2;
case 1: /* PutMsg() */
m68k_areg (regs, 0) = read_comm_pipe_u32_blocking (&native2amiga_pending);
m68k_areg (regs, 1) = read_comm_pipe_u32_blocking (&native2amiga_pending);
return 1;
case 2: /* ReplyMsg() */
m68k_areg (regs, 1) = read_comm_pipe_u32_blocking (&native2amiga_pending);
return 3;
default:
write_log ("exter_int_helper: unknown native action\n");
break;
}
}
#endif
/* Find some unit that needs a message sent, and return its port,
* or zero if all are done.
* Take care not to dereference self for units that didn't have their
* startup packet sent. */
for (;;) {
if (unit_no >= current_mountinfo->num_units)
return 0;
if (uip[unit_no].self != 0
&& uip[unit_no].self->cmds_acked == uip[unit_no].self->cmds_complete
&& uip[unit_no].self->cmds_acked != uip[unit_no].self->cmds_sent)
break;
unit_no++;
}
uip[unit_no].self->cmds_acked = uip[unit_no].self->cmds_sent;
port = uip[unit_no].self->port;
if (port) {
m68k_areg (regs, 0) = port;
m68k_areg (regs, 1) = find_unit (port)->dummy_message;
unit_no++;
return 1;
}
break;
case 4:
/* Exit the interrupt, and release the single-threading lock. */
uae_sem_post (&singlethread_int_sem);
break;
default:
write_log ("Shouldn't happen in exter_int_helper.\n");
break;
}
return 0;
}
static int handle_packet (Unit *unit, dpacket pck)
{
uae_s32 type = GET_PCK_TYPE (pck);
PUT_PCK_RES2 (pck, 0);
switch (type) {
case ACTION_LOCATE_OBJECT: action_lock (unit, pck); break;
case ACTION_FREE_LOCK: action_free_lock (unit, pck); break;
case ACTION_COPY_DIR: action_dup_lock (unit, pck); break;
case ACTION_DISK_INFO: action_disk_info (unit, pck); break;
case ACTION_INFO: action_info (unit, pck); break;
case ACTION_EXAMINE_OBJECT: action_examine_object (unit, pck); break;
case ACTION_EXAMINE_NEXT: action_examine_next (unit, pck); break;
case ACTION_FIND_INPUT: action_find_input (unit, pck); break;
case ACTION_FIND_WRITE: action_find_write (unit, pck); break;
case ACTION_FIND_OUTPUT: action_find_output (unit, pck); break;
case ACTION_END: action_end (unit, pck); break;
case ACTION_READ: action_read (unit, pck); break;
case ACTION_WRITE: action_write (unit, pck); break;
case ACTION_SEEK: action_seek (unit, pck); break;
case ACTION_SET_PROTECT: action_set_protect (unit, pck); break;
case ACTION_SET_COMMENT: action_set_comment (unit, pck); break;
case ACTION_SAME_LOCK: action_same_lock (unit, pck); break;
case ACTION_PARENT: action_parent (unit, pck); break;
case ACTION_CREATE_DIR: action_create_dir (unit, pck); break;
case ACTION_DELETE_OBJECT: action_delete_object (unit, pck); break;
case ACTION_RENAME_OBJECT: action_rename_object (unit, pck); break;
case ACTION_SET_DATE: action_set_date (unit, pck); break;
case ACTION_CURRENT_VOLUME: action_current_volume (unit, pck); break;
case ACTION_RENAME_DISK: action_rename_disk (unit, pck); break;
case ACTION_IS_FILESYSTEM: action_is_filesystem (unit, pck); break;
case ACTION_FLUSH: action_flush (unit, pck); break;
/* 2.0+ packet types */
case ACTION_SET_FILE_SIZE: action_set_file_size (unit, pck); break;
case ACTION_EXAMINE_FH: action_examine_fh (unit, pck); break;
case ACTION_FH_FROM_LOCK: action_fh_from_lock (unit, pck); break;
case ACTION_CHANGE_MODE: action_change_mode (unit, pck); break;
case ACTION_PARENT_FH: action_parent_fh (unit, pck); break;
/* unsupported packets */
case ACTION_LOCK_RECORD:
case ACTION_FREE_RECORD:
case ACTION_COPY_DIR_FH:
case ACTION_EXAMINE_ALL:
case ACTION_MAKE_LINK:
case ACTION_READ_LINK:
case ACTION_FORMAT:
case ACTION_ADD_NOTIFY:
case ACTION_REMOVE_NOTIFY:
default:
TRACE(("*** UNSUPPORTED PACKET %ld\n", type));
return 0;
}
return 1;
}
#ifdef UAE_FILESYS_THREADS
static void *filesys_thread (void *unit_v)
{
UnitInfo *ui = (UnitInfo *)unit_v;
for (;;) {
uae_u8 *pck;
uae_u8 *msg;
uae_u32 morelocks;
int i;
pck = (uae_u8 *)read_comm_pipe_pvoid_blocking (ui->unit_pipe);
msg = (uae_u8 *)read_comm_pipe_pvoid_blocking (ui->unit_pipe);
morelocks = (uae_u32)read_comm_pipe_int_blocking (ui->unit_pipe);
if (ui->reset_state == FS_GO_DOWN) {
if (pck != 0)
continue;
/* Death message received. */
uae_sem_post (&ui->reset_sync_sem);
/* Die. */
return 0;
}
put_long (get_long (morelocks), get_long (ui->self->locklist));
put_long (ui->self->locklist, morelocks);
if (! handle_packet (ui->self, pck)) {
PUT_PCK_RES1 (pck, DOS_FALSE);
PUT_PCK_RES2 (pck, ERROR_ACTION_NOT_KNOWN);
}
/* Mark the packet as processed for the list scan in the assembly code. */
do_put_mem_long ((uae_u32 *)(msg + 4), -1);
/* Acquire the message lock, so that we know we can safely send the
* message. */
ui->self->cmds_sent++;
/* The message is sent by our interrupt handler, so make sure an interrupt
* happens. */
uae_int_requested = 1;
uae_sem_wait (&packet_counter_sem);
active_fs_packets--;
uae_sem_post (&packet_counter_sem);
/* Send back the locks. */
if (get_long (ui->self->locklist) != 0)
write_comm_pipe_int (ui->back_pipe, (int)(get_long (ui->self->locklist)), 0);
put_long (ui->self->locklist, 0);
}
return 0;
}
#endif
/* Talk about spaghetti code... */
static uae_u32 filesys_handler (TrapContext *dummy)
{
Unit *unit = find_unit (m68k_areg (regs, 5));
uaecptr packet_addr = m68k_dreg (regs, 3);
uaecptr message_addr = m68k_areg (regs, 4);
uae_u8 *pck;
uae_u8 *msg;
if (! valid_address (packet_addr, 36) || ! valid_address (message_addr, 14)) {
write_log ("Bad address passed for packet.\n");
goto error2;
}
pck = get_real_address (packet_addr);
msg = get_real_address (message_addr);
#if 0
if (unit->reset_state == FS_GO_DOWN)
/* You might as well queue it, if you live long enough */
return 1;
#endif
do_put_mem_long ((uae_u32 *)(msg + 4), -1);
if (!unit || !unit->volume) {
write_log ("Filesystem was not initialized.\n");
goto error;
}
#ifdef UAE_FILESYS_THREADS
{
/* Get two more locks and hand them over to the other thread. */
uae_u32 morelocks;
morelocks = get_long (m68k_areg (regs, 3));
put_long (m68k_areg (regs, 3), get_long (get_long (morelocks)));
put_long (get_long (morelocks), 0);
uae_sem_wait (&packet_counter_sem);
active_fs_packets++;
uae_sem_post (&packet_counter_sem);
/* The packet wasn't processed yet. */
do_put_mem_long ((uae_u32 *)(msg + 4), 0);
write_comm_pipe_pvoid (unit->ui.unit_pipe, (void *)pck, 0);
write_comm_pipe_pvoid (unit->ui.unit_pipe, (void *)msg, 0);
write_comm_pipe_int (unit->ui.unit_pipe, (int)morelocks, 1);
/* Don't reply yet. */
return 1;
}
#endif
if (! handle_packet (unit, pck)) {
error:
PUT_PCK_RES1 (pck, DOS_FALSE);
PUT_PCK_RES2 (pck, ERROR_ACTION_NOT_KNOWN);
}
TRACE(("reply: %8lx, %ld\n", GET_PCK_RES1 (pck), GET_PCK_RES2 (pck)));
error2:
return 0;
}
static int current_deviceno = 0;
static int current_cdrom = 0;
static void reset_uaedevices (void)
{
current_deviceno = 0;
current_cdrom = 0;
}
static void init_filesys_diagentry (void)
{
do_put_mem_long ((uae_u32 *)(filesysory + 0x2100), EXPANSION_explibname);
do_put_mem_long ((uae_u32 *)(filesysory + 0x2104), filesys_configdev);
do_put_mem_long ((uae_u32 *)(filesysory + 0x2108), EXPANSION_doslibname);
do_put_mem_long ((uae_u32 *)(filesysory + 0x210c), current_mountinfo->num_units);
}
void filesys_start_threads (void)
{
UnitInfo *uip;
int i;
current_mountinfo = dup_mountinfo (currprefs.mountinfo);
reset_uaedevices ();
uip = current_mountinfo->ui;
for (i = 0; i < current_mountinfo->num_units; i++) {
UnitInfo *ui = &uip[i];
ui->unit_pipe = 0;
#ifdef UAE_FILESYS_THREADS
if (hardfile_fs_type (current_mountinfo, i) == FILESYS_VIRTUAL) {
uip[i].unit_pipe = (smp_comm_pipe *)xmalloc (sizeof (smp_comm_pipe));
uip[i].back_pipe = (smp_comm_pipe *)xmalloc (sizeof (smp_comm_pipe));
init_comm_pipe (uip[i].unit_pipe, 50, 3);
init_comm_pipe (uip[i].back_pipe, 50, 1);
uae_start_thread (filesys_thread, (void *)(uip + i), &uip[i].tid);
}
#endif
}
}
void filesys_reset (void)
{
Unit *u, *u1;
/* We get called once from customreset at the beginning of the program
* before filesys_start_threads has been called. Survive that. */
if (current_mountinfo == 0)
return;
for (u = units; u; u = u1) {
u1 = u->next;
free (u);
}
unit_num = 0;
units = 0;
free_mountinfo (current_mountinfo);
current_mountinfo = 0;
}
static void free_all_ainos (Unit *u, a_inode *parent)
{
a_inode *a;
while (a = parent->child) {
free_all_ainos (u, a);
dispose_aino (u, &parent->child, a);
}
}
void filesys_prepare_reset (void)
{
UnitInfo *uip = current_mountinfo->ui;
Unit *u;
int i;
#ifdef UAE_FILESYS_THREADS
for (i = 0; i < current_mountinfo->num_units; i++) {
if (uip[i].unit_pipe != 0) {
uae_sem_init (&uip[i].reset_sync_sem, 0, 0);
uip[i].reset_state = FS_GO_DOWN;
/* send death message */
write_comm_pipe_int (uip[i].unit_pipe, 0, 0);
write_comm_pipe_int (uip[i].unit_pipe, 0, 0);
write_comm_pipe_int (uip[i].unit_pipe, 0, 1);
uae_sem_wait (&uip[i].reset_sync_sem);
}
}
#endif
u = units;
while (u != 0) {
free_all_ainos (u, &u->rootnode);
u->rootnode.next = u->rootnode.prev = &u->rootnode;
u->aino_cache_size = 0;
u = u->next;
}
}
static uae_u32 filesys_diagentry (TrapContext *dummy)
{
uaecptr resaddr = m68k_areg (regs, 2) + 0x10;
uaecptr start = resaddr;
uaecptr residents, tmp;
TRACE (("filesystem: diagentry called\n"));
filesys_configdev = m68k_areg (regs, 3);
init_filesys_diagentry ();
uae_sem_init (&singlethread_int_sem, 0, 1);
if (ROM_hardfile_resid != 0) {
/* Build a struct Resident. This will set up and initialize
* the uae.device */
put_word (resaddr + 0x0, 0x4AFC);
put_long (resaddr + 0x2, resaddr);
put_long (resaddr + 0x6, resaddr + 0x1A); /* Continue scan here */
put_word (resaddr + 0xA, 0x8101); /* RTF_AUTOINIT|RTF_COLDSTART; Version 1 */
put_word (resaddr + 0xC, 0x0305); /* NT_DEVICE; pri 05 */
put_long (resaddr + 0xE, ROM_hardfile_resname);
put_long (resaddr + 0x12, ROM_hardfile_resid);
put_long (resaddr + 0x16, ROM_hardfile_init); /* calls filesys_init */
}
resaddr += 0x1A;
tmp = resaddr;
/* The good thing about this function is that it always gets called
* when we boot. So we could put all sorts of stuff that wants to be done
* here.
* We can simply add more Resident structures here. Although the Amiga OS
* only knows about the one at address DiagArea + 0x10, we scan for other
* Resident structures and call InitResident() for them at the end of the
* diag entry. */
resaddr = scsidev_startup(resaddr);
native2amiga_startup();
/* scan for Residents and return pointer to array of them */
residents = resaddr;
while (tmp < residents && tmp > start) {
if (get_word (tmp) == 0x4AFC &&
get_long (tmp + 0x2) == tmp) {
put_word (resaddr, 0x227C); /* movea.l #tmp,a1 */
put_long (resaddr + 2, tmp);
put_word (resaddr + 6, 0x7200); /* moveq.l #0,d1 */
put_long (resaddr + 8, 0x4EAEFF9A); /* jsr -$66(a6) ; InitResident */
resaddr += 12;
tmp = get_long (tmp + 0x6);
} else {
tmp++;
}
}
/* call setup_exter */
put_word (resaddr + 0, 0x2079);
put_long (resaddr + 2, RTAREA_BASE + 28 + 4); /* move.l RTAREA_BASE+32,a0 */
put_word (resaddr + 6, 0xd1fc);
put_long (resaddr + 8, RTAREA_BASE + 8 + 4); /* add.l #RTAREA_BASE+12,a0 */
put_word (resaddr + 12, 0x4e90); /* jsr (a0) */
put_word (resaddr + 14, 0x7001); /* moveq.l #1,d0 */
put_word (resaddr + 16, RTS);
m68k_areg (regs, 0) = residents;
return 1;
}
/* don't forget filesys.asm! */
#define PP_MAXSIZE 4 * 96
#define PP_FSSIZE 400
#define PP_FSPTR 404
#define PP_FSRES 408
#define PP_EXPLIB 412
#define PP_FSHDSTART 416
static uae_u32 REGPARAM2 filesys_dev_bootfilesys (TrapContext *dummy)
{
uaecptr devicenode = m68k_areg (regs, 3);
uaecptr parmpacket = m68k_areg (regs, 1);
uaecptr fsres = get_long (parmpacket + PP_FSRES);
uaecptr fsnode;
uae_u32 dostype, dostype2;
int no = m68k_dreg (regs, 6);
int unit_no = no & 65535;
int type = hardfile_fs_type (current_mountinfo, unit_no);
if (type == FILESYS_VIRTUAL)
return 0;
dostype = get_long (parmpacket + 80);
fsnode = get_long (fsres + 18);
while (get_long (fsnode)) {
dostype2 = get_long (fsnode + 14);
if (dostype2 == dostype) {
if (get_long (fsnode + 22) & (1 << 7)) {
put_long (devicenode + 32, get_long (fsnode + 54)); /* dn_SegList */
put_long (devicenode + 36, -1); /* dn_GlobalVec */
}
return 1;
}
fsnode = get_long (fsnode);
}
return 0;
}
/* Remember a pointer AmigaOS gave us so we can later use it to identify
* which unit a given startup message belongs to. */
static uae_u32 filesys_dev_remember (TrapContext *dummy)
{
int no = m68k_dreg (regs, 6);
int unit_no = no & 65535;
int sub_no = no >> 16;
UnitInfo *uip = ¤t_mountinfo->ui[unit_no];
uaecptr devicenode = m68k_areg (regs, 3);
uaecptr parmpacket = m68k_areg (regs, 1);
/* copy filesystem loaded from RDB */
if (get_long (parmpacket + PP_FSPTR)) {
int i;
for (i = 0; i < uip->rdb_filesyssize; i++)
put_byte (get_long (parmpacket + PP_FSPTR) + i, uip->rdb_filesysstore[i]);
free (uip->rdb_filesysstore);
uip->rdb_filesysstore = 0;
uip->rdb_filesyssize = 0;
}
if ((uae_s32)m68k_dreg (regs, 3) >= 0)
uip->startup = get_long (devicenode + 28);
return devicenode;
}
static int legalrdbblock (UnitInfo *uip, int block)
{
if (block <= 0)
return 0;
if (block >= uip->hf.size / uip->hf.blocksize)
return 0;
return 1;
}
static int rl (uae_u8 *p)
{
return (p[0] << 24) | (p[1] << 16) | (p[2] << 8) | (p[3]);
}
static int rdb_checksum (char *id, uae_u8 *p, int block)
{
uae_u32 sum = 0;
int i, blocksize;
if (memcmp (id, p, 4))
return 0;
blocksize = rl (p + 4);
if (blocksize < 1 || blocksize * 4 > FILESYS_MAX_BLOCKSIZE)
return 0;
for (i = 0; i < blocksize; i++)
sum += rl (p + i * 4);
sum = -sum;
if (sum) {
write_log ("RDB: block %d ('%s') checksum error\n", block, id);
return 0;
}
return 1;
}
static void dump_partinfo (char *name, int num, uaecptr pp, int partblock)
{
uae_u32 dostype = get_long (pp + 80);
write_log ("RDB: '%s' dostype=%08.8X. PartBlock=%d\n", name, dostype, partblock);
write_log ("BlockSize: %d, Surfaces: %d, SectorsPerBlock %d\n",
get_long (pp + 20) * 4, get_long (pp + 28), get_long (pp + 32));
write_log ("SectorsPerTrack: %d, Reserved: %d, LowCyl %d, HighCyl %d\n",
get_long (pp + 36), get_long (pp + 40), get_long (pp + 52), get_long (pp + 56));
write_log ("Buffers: %d, BufMemType: %08.8x, MaxTransfer: %08.8x, BootPri: %d\n",
get_long (pp + 60), get_long (pp + 64), get_long (pp + 68), get_long (pp + 76));
}
static char *device_dupfix (uaecptr expbase, char *devname)
{
uaecptr bnode, dnode, name;
int len, i, modified;
char dname[256], newname[256];
strcpy (newname, devname);
modified = 1;
while (modified) {
modified = 0;
bnode = get_long (expbase + 74); /* expansion.library bootnode list */
while (get_long (bnode)) {
dnode = get_long (bnode + 16); /* device node */
name = get_long (dnode + 40) << 2; /* device name BSTR */
len = get_byte (name);
for (i = 0; i < len; i++)
dname[i] = get_byte (name + 1 + i);
dname[len] = 0;
for (;;) {
if (strcasecmp (newname, dname) == 0) {
if (strlen (newname) > 2 && newname[strlen (newname) - 2] == '_') {
newname[strlen (newname) - 1]++;
} else {
strcat (newname, "_0");
}
modified = 1;
} else {
break;
}
}
bnode = get_long (bnode);
}
}
return my_strdup (newname);
}
#define rdbmnt write_log ("Mounting uaehf.device %d (%d) (size=%llu):\n", unit_no, partnum, hfd->size);
static int rdb_mount (UnitInfo *uip, int unit_no, int partnum, uaecptr parmpacket)
{
int lastblock = 63, blocksize, readblocksize, badblock, driveinitblock;
uae_u8 bufrdb[FILESYS_MAX_BLOCKSIZE], *buf = 0;
uae_u8 *fsmem = 0;
int rdblock, partblock, fileblock, lsegblock, i;
uae_u32 flags;
struct hardfiledata *hfd = &uip->hf;
uae_u32 dostype;
uaecptr fsres, fsnode;
int err = 0;
int oldversion, oldrevision;
int newversion, newrevision;
if (hfd->blocksize == 0) {
rdbmnt
write_log ("failed, blocksize == 0\n");
return -1;
}
if (lastblock * hfd->blocksize > hfd->size) {
rdbmnt
write_log ("failed, too small (%d*%d > %I64u)\n", lastblock, hfd->blocksize, hfd->size);
return -2;
}
for (rdblock = 0; rdblock < lastblock; rdblock++) {
hdf_read (hfd, bufrdb, rdblock * hfd->blocksize, hfd->blocksize);
if (rdb_checksum ("RDSK", bufrdb, rdblock))
break;
hdf_read (hfd, bufrdb, rdblock * hfd->blocksize, hfd->blocksize);
if (!memcmp ("RDSK", bufrdb, 4)) {
bufrdb[0xdc] = 0;
bufrdb[0xdd] = 0;
bufrdb[0xde] = 0;
bufrdb[0xdf] = 0;
#if 0
if (rdb_checksum ("RDSK", bufrdb, rdblock)) {
write_log ("Windows 95/98/ME trashed RDB detected, fixing..\n");
hdf_write (hfd, bufrdb, rdblock * hfd->blocksize, hfd->blocksize);
break;
}
#endif
}
}
if (rdblock == lastblock) {
rdbmnt
write_log ("failed, no RDB detected\n");
return -2;
}
blocksize = rl (bufrdb + 16);
readblocksize = blocksize > hfd->blocksize ? blocksize : hfd->blocksize;
badblock = rl (bufrdb + 24);
if (badblock != -1) {
rdbmnt
write_log ("RDB: badblock list is not yet supported. Contact the author.\n");
return -2;
}
driveinitblock = rl (bufrdb + 36);
if (driveinitblock != -1) {
rdbmnt
write_log ("RDB: driveinit is not yet supported. Contact the author.\n");
return -2;
}
hfd->cylinders = rl (bufrdb + 64);
hfd->sectors = rl (bufrdb + 68);
hfd->heads = rl (bufrdb + 72);
fileblock = rl (bufrdb + 32);
if (partnum == 0) {
write_log ("RDB: RDSK detected at %d, FSHD=%d, C=%d S=%d H=%d\n",
rdblock, fileblock, hfd->cylinders, hfd->sectors, hfd->heads);
}
buf = (uae_u8*)xmalloc (readblocksize);
for (i = 0; i <= partnum; i++) {
if (i == 0)
partblock = rl (bufrdb + 28);
else
partblock = rl (buf + 4 * 4);
if (!legalrdbblock (uip, partblock)) {
err = -2;
goto error;
}
memset (buf, 0, readblocksize);
hdf_read (hfd, buf, partblock * hfd->blocksize, readblocksize);
if (!rdb_checksum ("PART", buf, partblock)) {
err = -2;
goto error;
}
}
rdbmnt
flags = rl (buf + 20);
if (flags & 2) { /* do not mount */
err = -1;
write_log ("RDB: Automount disabled, not mounting\n");
goto error;
}
if (!(flags & 1)) /* not bootable */
m68k_dreg (regs, 7) = 0;
buf[37 + buf[36]] = 0; /* zero terminate BSTR */
uip->rdb_devname_amiga[partnum]
= ds (device_dupfix (get_long (parmpacket + PP_EXPLIB), buf + 37));
put_long (parmpacket, uip->rdb_devname_amiga[partnum]); /* name */
put_long (parmpacket + 4, ROM_hardfile_resname);
put_long (parmpacket + 8, uip->devno);
put_long (parmpacket + 12, 0); /* Device flags */
for (i = 0; i < PP_MAXSIZE; i++)
put_byte (parmpacket + 16 + i, buf[128 + i]);
dump_partinfo (buf + 37, uip->devno, parmpacket, partblock);
dostype = get_long (parmpacket + 80);
if (dostype == 0) {
write_log ("RDB: mount failed, dostype=0\n");
err = -1;
goto error;
}
if (hfd->cylinders * hfd->sectors * hfd->heads * blocksize > hfd->size)
write_log ("RDB: WARNING: end of partition > size of disk!\n");
err = 2;
/* load custom filesystems if needed */
if (fileblock == -1 || !legalrdbblock (uip, fileblock))
goto error;
fsres = get_long (parmpacket + PP_FSRES);
if (!fsres) {
write_log ("RDB: FileSystem.resource not found, this shouldn't happen!\n");
goto error;
}
fsnode = get_long (fsres + 18);
while (get_long (fsnode)) {
if (get_long (fsnode + 14) == dostype)
break;
fsnode = get_long (fsnode);
}
oldversion = oldrevision = -1;
if (get_long (fsnode)) {
oldversion = get_word (fsnode + 18);
oldrevision = get_word (fsnode + 20);
} else {
fsnode = 0;
}
for (;;) {
if (fileblock == -1) {
if (!fsnode)
write_log ("RDB: required FS %08.8X not in FileSystem.resource or in RDB\n", dostype);
goto error;
}
if (!legalrdbblock (uip, fileblock)) {
write_log ("RDB: corrupt FSHD pointer %d\n", fileblock);
goto error;
}
memset (buf, 0, readblocksize);
hdf_read (hfd, buf, fileblock * hfd->blocksize, readblocksize);
if (!rdb_checksum ("FSHD", buf, fileblock)) {
write_log ("RDB: checksum error in FSHD block %d\n", fileblock);
goto error;
}
fileblock = rl (buf + 16);
if ((dostype >> 8) == (rl (buf + 32) >> 8))
break;
}
newversion = (buf[36] << 8) | buf[37];
newrevision = (buf[38] << 8) | buf[39];
write_log ("RDB: RDB filesystem %08.8X version %d.%d\n", dostype, newversion, newrevision);
if (fsnode) {
write_log ("RDB: %08.8X in FileSystem.resouce version %d.%d\n", dostype, oldversion, oldrevision);
}
if (newversion * 65536 + newrevision <= oldversion * 65536 + oldrevision && oldversion >= 0) {
write_log ("RDB: FS in FileSystem.resource is newer or same, ignoring RDB filesystem\n");
goto error;
}
for (i = 0; i < 140; i++)
put_byte (parmpacket + PP_FSHDSTART + i, buf[32 + i]);
put_long (parmpacket + PP_FSHDSTART, dostype);
/* we found required FSHD block */
fsmem = (uae_u8*)xmalloc (262144);
lsegblock = rl (buf + 72);
i = 0;
for (;;) {
int pb = lsegblock;
if (!legalrdbblock (uip, lsegblock))
goto error;
memset (buf, 0, readblocksize);
hdf_read (hfd, buf, lsegblock * hfd->blocksize, readblocksize);
if (!rdb_checksum ("LSEG", buf, lsegblock))
goto error;
lsegblock = rl (buf + 16);
if (lsegblock == pb)
goto error;
memcpy (fsmem + i * (blocksize - 20), buf + 20, blocksize - 20);
i++;
if (lsegblock == -1)
break;
}
write_log ("RDB: Filesystem loaded, %d bytes\n", i * (blocksize - 20));
put_long (parmpacket + PP_FSSIZE, i * (blocksize - 20)); /* RDB filesystem size hack */
uip->rdb_filesysstore = fsmem;
uip->rdb_filesyssize = i * (blocksize - 20);
free (buf);
return 2;
error:
free (buf);
free (fsmem);
return err;
}
static void get_new_device (int type, uaecptr parmpacket, char **devname,
uaecptr *devname_amiga, int unit_no)
{
char buffer[80];
sprintf (buffer, "DH%d", unit_no);
if (*devname == NULL)
*devname = device_dupfix (get_long (parmpacket + PP_EXPLIB), buffer);
*devname_amiga = ds (*devname);
if (type == FILESYS_VIRTUAL)
write_log ("FS: mounted virtual unit %s\n", buffer);
else
write_log ("FS: mounted HDF unit %s\n", buffer);
}
/* Fill in per-unit fields of a parampacket */
static uae_u32 filesys_dev_storeinfo (TrapContext *dummy)
{
UnitInfo *uip = current_mountinfo->ui;
int no = m68k_dreg (regs, 6);
int unit_no = no & 65535;
int sub_no = no >> 16;
int type = hardfile_fs_type (current_mountinfo, unit_no);
uaecptr parmpacket = m68k_areg (regs, 0);
if (type == FILESYS_HARDFILE_RDB || type == FILESYS_HARDDRIVE) {
/* RDB hardfile */
uip[unit_no].devno = unit_no;
return rdb_mount (&uip[unit_no], unit_no, sub_no, parmpacket);
}
/* No subunits for normal filesystems and hardfiles. */
if (sub_no)
return -2;
get_new_device (type, parmpacket, &uip[unit_no].devname, &uip[unit_no].devname_amiga, unit_no);
uip[unit_no].devno = unit_no;
put_long (parmpacket, uip[unit_no].devname_amiga);
put_long (parmpacket + 4, type != FILESYS_VIRTUAL ? ROM_hardfile_resname : fsdevname);
put_long (parmpacket + 8, uip[unit_no].devno);
put_long (parmpacket + 12, 0); /* Device flags */
put_long (parmpacket + 16, 16); /* Env. size */
put_long (parmpacket + 20, uip[unit_no].hf.blocksize >> 2); /* longwords per block */
put_long (parmpacket + 24, 0); /* unused */
put_long (parmpacket + 28, uip[unit_no].hf.surfaces); /* heads */
put_long (parmpacket + 32, 1); /* sectors per block */
put_long (parmpacket + 36, uip[unit_no].hf.secspertrack); /* sectors per track */
put_long (parmpacket + 40, uip[unit_no].hf.reservedblocks); /* reserved blocks */
put_long (parmpacket + 44, 0); /* unused */
put_long (parmpacket + 48, 0); /* interleave */
put_long (parmpacket + 52, 0); /* lowCyl */
put_long (parmpacket + 56, uip[unit_no].hf.nrcyls - 1); /* hiCyl */
put_long (parmpacket + 60, 50); /* Number of buffers */
put_long (parmpacket + 64, 0); /* Buffer mem type */
put_long (parmpacket + 68, 0x7FFFFFFF); /* largest transfer */
put_long (parmpacket + 72, ~1); /* addMask (?) */
put_long (parmpacket + 76, uip[unit_no].bootpri); /* bootPri */
put_long (parmpacket + 80, 0x444f5300); /* DOS\0 */
put_long (parmpacket + 84, 0); /* pad */
return type;
}
void filesys_install (void)
{
int i;
uaecptr loop;
TRACE (("Installing filesystem\n"));
ROM_filesys_resname = ds ("UAEunixfs.resource");
ROM_filesys_resid = ds ("UAE unixfs 0.4");
fsdevname = ds ("uae.device"); /* does not really exist */
ROM_filesys_diagentry = here();
calltrap (deftrap(filesys_diagentry));
dw (0x4ED0); /* JMP (a0) - jump to code that inits Residents */
loop = here ();
org (RTAREA_BASE + 0xFF18);
calltrap (deftrap (filesys_dev_bootfilesys));
dw (RTS);
/* Special trap for the assembly make_dev routine */
org (RTAREA_BASE + 0xFF20);
calltrap (deftrap (filesys_dev_remember));
dw (RTS);
org (RTAREA_BASE + 0xFF28);
calltrap (deftrap (filesys_dev_storeinfo));
dw (RTS);
org (RTAREA_BASE + 0xFF30);
calltrap (deftrap (filesys_handler));
dw (RTS);
org (RTAREA_BASE + 0xFF40);
calltrap (deftrap (startup_handler));
dw (RTS);
org (RTAREA_BASE + 0xFF50);
calltrap (deftrap (exter_int_helper));
dw (RTS);
org (loop);
uae_sem_init (&packet_counter_sem, 0, 0);
}
void filesys_install_code (void)
{
align(4);
/* The last offset comes from the code itself, look for it near the top. */
EXPANSION_bootcode = here () + 8 + 0x1c + 4;
/* Ouch. Make sure this is _always_ a multiple of two bytes. */
filesys_initcode = here() + 8 + 0x30 + 4;
#include "filesys_bootrom.c"
}