• R/O
  • HTTP
  • SSH
  • HTTPS

Tags
Keine Tags

Frequently used words (click to add to your profile)

javac++androidlinuxc#windowsobjective-ccocoa誰得qtpythonphprubygameguibathyscaphec計画中(planning stage)翻訳omegatframeworktwitterdomtestvb.netdirectxゲームエンジンbtronarduinopreviewer

Motorola M6800 (6800) Exorciser / SWTPC emulator plus 6801 instruction set emulation


File Info

Rev. 92ef8a291a6240887b19ec9117ea774aba59ea9a
Größe 36,058 Bytes
Zeit 2022-09-03 00:57:21
Autor Joel Matthew Rees
Log Message

Fixing the embarrassing CPX bug -- C flag, etc.

Content

/*	MDOS diskette access
 *	Copyright
 *		(C) 2011 Joseph H. Allen
 *
 * This is free software; you can redistribute it and/or modify it under the 
 * terms of the GNU General Public License as published by the Free Software 
 * Foundation; either version 1, or (at your option) any later version.  
 *
 * It is distributed in the hope that it will be useful, but WITHOUT ANY 
 * WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS 
 * FOR A PARTICULAR PURPOSE.  See the GNU General Public License for more 
 * details.  
 * 
 * You should have received a copy of the GNU General Public License along with 
 * this software; see the file COPYING.  If not, write to the Free Software Foundation, 
 * 675 Mass Ave, Cambridge, MA 02139, USA.
 */

/* MDOS disk access */

#include <stdio.h>
#include <stdlib.h>
#include <string.h>

#define SECTOR_SIZE 128
#define CLUSTER_SIZE (SECTOR_SIZE * 4)

/* Specific sectors */

#define SECTOR_ID 0

#define SECTOR_CAT 1	/* Allocation bitmap */
/* Bit 7 of byte 0 is first cluster.  1 means allocated. */

#define SECTOR_LCAT 2	/* Bad block bitmap */

#define SECTOR_DIR 3
#define SECTOR_DIR_SIZE 20
#define SECTOR_BOOT_0 23
#define SECTOR_BOOT_1 24

struct rib {
	unsigned char sdw[114];
	unsigned char blank[3];
	unsigned char last_size;	/* 117 */
	unsigned char size_high;	/* 118 */
	unsigned char size_low;		/* 119 */
	unsigned char addr_high;	/* 120 */
	unsigned char addr_low;		/* 121 */
	unsigned char pc_high;		/* 122 */
	unsigned char pc_low;		/* 123 */
	unsigned char blank_1[4];
};

struct dirent {
	unsigned char name[8];
	unsigned char suffix[2];
	unsigned char sector_high;
	unsigned char sector_low;
	unsigned char attr_high;
	unsigned char attr_low;
	unsigned char blank[2];
};

FILE *disk;

void getsect(unsigned char *buf, int sect)
{
        fseek(disk, sect * SECTOR_SIZE, SEEK_SET);
        fread((char *)buf, SECTOR_SIZE, 1, disk);
}

void putsect(unsigned char *buf, int sect)
{
        fseek(disk, sect * SECTOR_SIZE, SEEK_SET);
        fwrite((char *)buf, SECTOR_SIZE, 1, disk);
}

int lower(int c)
{
        if (c >= 'A' && c <= 'Z')
                return c - 'A' + 'a';
        else
                return c;
}

struct name
{
        char *name;

        /* From RIB */
        int last_size;
        int size;
        int pc;
        int load;

        /* Calculated from RIB */
        int sects;

        int system;
        int write_protect;
        int delete_protect;
        int compr;
        int cont;
        int type;
        int is_cm; /* If suffix is .CM */
        int sector; /* Sector no. of RIB */
};

struct name *names[(SECTOR_DIR_SIZE * SECTOR_SIZE) / sizeof(struct dirent)];
int name_n;

int comp(struct name **l, struct name **r)
{
        return strcmp((*l)->name, (*r)->name);
}

int calc_sects(struct rib *r)
{
        int total = 0; /* Count rib */
        int x;
        for (x = 0; x != sizeof(r->sdw); x += 2) {
                int sdw = ((r->sdw[x] << 8) + r->sdw[x + 1]);
                if (sdw & 0x8000) {
                        /*
                        printf("Total = %d\n", total * 4);
                        printf("Logical last = %d\n", (sdw & 0x7FFF));
                        */
                        return (sdw & 0x7FFF) + 1; /* Size in sectors */
                        break;
                }
                total += ((sdw >> 10) & 0x1F) + 1;
        }
        return total * 4; /* clusters to sectors */
}

void dump_rib(struct rib *r)
{
        int x;
        printf("Dumping rib:\n");
        for (x = 0; x != sizeof(r->sdw); x += 2) {
                int sdw = ((r->sdw[x] << 8) + r->sdw[x + 1]);
                if (sdw & 0x8000) {
                        printf("End mark found: last logical sector is %d\n", (sdw & 0x7FFF));
                        return;
                } else {
                        int clust = (sdw & 0x3FF);
                        int sect = clust * 4;
                        int siz = ((sdw >> 10) & 0x1F) + 1;
                        int len = (((sdw >> 10) & 0x1F) + 1) * 4;
                        printf("Segment start=%d, size=%d\n", clust, siz);
                }
        }
}

void read_file(int sector, int type, FILE *f)
{
        unsigned char rib_buf[SECTOR_SIZE];
        struct rib *r = (struct rib *)rib_buf;
        int x;
        int ign = 0;
        int secno = 0;
        unsigned char last_size;
        unsigned short size;

        getsect(rib_buf, sector);

        size = (r->size_high << 8) + r->size_low;
        last_size = r->last_size;

        /* printf("Size is %d sectors, %d bytes in last, total=%d bytes\n", size, last_size,
               (size - 1) * SECTOR_SIZE + last_size); */

        if (type == 2)
                fprintf(stderr,"Load=$%4.4x, Start=$%4.4x\n",(r->addr_high << 8) + r->addr_low,
                        (r->pc_high << 8) + r->pc_low);

        for (x = 0; x != sizeof(r->sdw); x += 2) {
                int sdw = ((r->sdw[x] << 8) + r->sdw[x + 1]);
                if (sdw & 0x8000) {
                        return;
                } else {
                        int clust = (sdw & 0x3FF);
                        int sect = clust * 4;
                        int len = (((sdw >> 10) & 0x1F) + 1) * 4;
                        int x;
                        /* printf("Clust = %d, Sect=%d, len=%d\n", clust, sect, len); */
                        for (x = 0; x != len; ++x) {
                                unsigned char buf[SECTOR_SIZE];
                                int n;
                                if (sect + x != sector) {
                                        /* printf("Sector %d:\n", sect + x); */
                                        getsect(buf, sect + x);
                                        if (type == 5) {
                                                /* Convert line end of ASCII file to UNIX */
                                                /* ASCII ends when we get a NUL which is not right after
                                                 * a CR or LF */
                                                for (n = 0; n != SECTOR_SIZE; ++n) {
                                                        int c = buf[n];
                                                        if (c & 0x80) {
                                                                int z;
                                                                for (z = 0; z != (c & 0x7F); ++z)
                                                                        fputc(' ', f);
                                                                ign = 0;
                                                        } else if (c == 13) {
                                                                fputc('\n', f);
                                                                ign = 1;
                                                        } else if (c == 10) {
                                                                /* putchar('J'); */
                                                                ign = 1;
                                                        } else if (c == 0) {
                                                                if (!ign)
                                                                        return;
                                                                else
                                                                        /* putchar('@') */;
                                                                ign = 0;
                                                        } else {
                                                                ign = 0;
                                                                fputc(c, f);
                                                        }
                                                }
                                                /* printf("EOS\n"); */
                                        } else if (type == 2) {
                                                /* Binary image file.  Exact size is in RIB */
                                                if (secno + 1 == size) {
                                                        /* printf("Sector %d, size=%d\n", secno, last_size); */
                                                        /* Last sector */
                                                        fwrite(buf, last_size, 1, f);
                                                        return;
                                                } else {
                                                        /* printf("Sector %d, size=128\n", secno); */
                                                        fwrite(buf, SECTOR_SIZE, 1, f);
                                                        secno++;
                                                }
                                        } else {
                                                /* Unknown file.  Send all sectors out. */
                                                fwrite(buf, SECTOR_SIZE, 1, f);
                                        }
                                }
                        }
                }
        }
}

/* Count free clusters */

int amount_free(unsigned char *cat)
{
        int total = 0;
        int x;
        for (x = 0; x != (SECTOR_SIZE * 8); ++x) {
                if (!(cat[x >> 3] & (1 << (7 - (x & 7)))))
                        ++total;
        }
        return total;
}

/* Free command */

int do_free(void)
{
        int amount;
        unsigned char cat[SECTOR_SIZE];
        getsect(cat, SECTOR_CAT);
        amount = amount_free(cat);
        printf("%d free sectors, %d free bytes\n", amount * 4, amount * 512);
        return 0;
}

/* Mark a segment as clear or free */

void mark_space(unsigned char *cat, int start, int size, int alloc)
{
        int x;
        for (x = start; x != start + size; ++x) {
                if (alloc)
                        cat[x >> 3] |= (1 << (7 - (x & 7)));
                else
                        cat[x >> 3] &= ~(1 << (7 - (x & 7)));
        }
}

/* Allocate n clusters.  Returns with amount actually allocated and starting cluster in *start */

int find_space(unsigned char *cat, int n, int *start)
{
        int x;
        int bestfirst = -1;
        int bestsize = 0;
        int curfirst = -1;
        int cursize;
        printf("Find space %d\n", n);
        /* Find first contiguous block which fits requested size */
        /* Note longest contiguous block along the way */
        for (x = 0; x != SECTOR_SIZE * 8; ++x) {
                if ((1 << (7 - (x & 7)) & cat[x >> 3])) {
                        /* This cluster is allocated */
                        if (curfirst != -1) {
                                /* We're in a segment, note it */
                                printf("Found segment: start=%d, size=%d\n", curfirst, cursize);
                                if (cursize > bestsize) {
                                        bestsize = cursize;
                                        bestfirst = curfirst;
                                }
                                curfirst = -1;
                                cursize = 0;
                        } else {
                                /* Not in segment, do nothing */
                        }
                } else {
                        /* This cluster is free */
                        if (curfirst != -1) {
                                /* We're in a segment: extend it */
                                ++cursize;
                                if (cursize == n || cursize == 32) goto done;
                        } else {
                                /* We're not in a segment, start one */
                                curfirst = x;
                                cursize = 1;
                                if (cursize == n) goto done;
                        }
                }
        }
        done:
        /* In a segment? Finalize it. */
        if (curfirst != -1) {
                printf("Found segment: start=%d, size=%d\n", curfirst, cursize);
                if (cursize > bestsize) {
                        bestfirst = curfirst;
                        bestsize = cursize;
                }
        }
        *start = bestfirst;
        return bestsize;
}

/* Allocate space, build RIB */

int alloc_space(unsigned char *cat, struct rib *r, int sects)
{
        int req = (sects + 3) / 4; /* Number of clusters to allocate */
        int seg = 0; /* Segment pointer in RIB */
        int sdw;
        /* printf("Amount avail = %d clusters.\n", amount_free(cat)); */
        while (req) {
                int start = -1;
                int found;
                if (seg == 114) {
                        printf("Too many segments\n");
                        return -1;
                }
                found = find_space(cat, req, &start);
                /* printf("Segment=%d Size=%d\n", start, found); */
                if (found) {
                        mark_space(cat, start, found, 1);

                        /* Add segment to rib */
                        sdw = start + ((found - 1) << 10);
                        r->sdw[seg] = (sdw >> 8);
                        r->sdw[seg + 1] = sdw;
                        seg += 2;
                }

                req -= found;
                if (!req) {
                        /* printf("Done\n"); */
                } else if (!found) {
                        printf("Not enough space\n");
                        return -1;
                }
        }
        if (seg == 114) {
                printf("Too many segments\n");
                return -1;
        }
        /* Add end mark */
        sdw = 0x8000 + sects - 2; /* Last sector does not count rib */
        r->sdw[seg] = (sdw >> 8);
        r->sdw[seg + 1] = sdw;
        /* Need to write back cat here */
        /* Success */
        return 0;
}

/* Convert logical sector number to physical sector number */

int lsn_to_psn(struct rib *r, int lsn)
{
        int x;
        for (x = 0; x != sizeof(r->sdw); x += 2) {
                int sdw = ((r->sdw[x] << 8) + r->sdw[x + 1]);
                if (sdw & 0x8000) {
                        printf("End mark found?  Internal error.\n");
                        exit(-1);
                } else {
                        int clust = (sdw & 0x3FF);
                        int sect = clust * 4;
                        int siz = ((sdw >> 10) & 0x1F) + 1;
                        int len = siz * 4;
                        if (lsn < len) {
                                return sect + lsn;
                        } else {
                                lsn -= len;
                        }
                }
        }
}

/* Write a file */

int write_file(unsigned char *cat, char *buf, int sects)
{
        int x;
        int rib_sect;
        struct rib r[1];

        memset(r, 0, SECTOR_SIZE);
        if (alloc_space(cat, r, sects + 1)) {
                return -1;
        }

        putsect((unsigned char *)r, rib_sect = lsn_to_psn(r, 0));
        for (x = 0; x != sects; ++x) {
                printf("File to sector %x\n", lsn_to_psn(r, x + 1));
                putsect(buf + SECTOR_SIZE * x, lsn_to_psn(r, x + 1));
        }

        printf("Rib sector = %x\n", rib_sect);

        return rib_sect;
}

/* File name hash function */

int hash(unsigned char *str)
{
	int x;
	unsigned b;
	unsigned a;
	b = 0;
	for (x = 0; x != 10; ++x) {
		a = *str++;
		if (a >= 0x25)
			a -= 0x25;
		else
			a = 0;
		b = (b & 0xFF) + a + (b >> 8);
		b = (((b << 1) + (b >> 8)) & 0x1FF);
	};
	b = (((b >> 1) + (b << 8)) & 0x1FF);
	a = b;
	a = (((a >> 1) + (a << 8)) & 0x1FF);
	a = (((a >> 1) + (a << 8)) & 0x1FF);
	a = (((a >> 1) + (a << 8)) & 0x1FF);
	a = (((a >> 1) + (a << 8)) & 0x1FF);
	a = (a & 0xFF) + (b & 0xFF);
	b = a;
	b &= 0x1f;
	if (b >= 20) {
		b -= 20;
		if (b < 10) {
			b = (b << 1) + (a & 1);
		}
	}
	return b;
}

/* Write directory entry */

int write_dir(char *name, int rib_sect, int attr)
{
        struct dirent d[1];
        int x;
        int hsh;
        int count;
        unsigned char dir_buf[SECTOR_SIZE];
        /* Copy file name into directory entry */
        x = 0;
        while (*name && *name != '.' && x < 8) {
                if (*name >= 'a' && *name <= 'z')
                        d->name[x++] = *name++ - 'a' + 'A';
                else
                        d->name[x++] = *name++;
        }
        while (x < 8) {
                d->name[x++] = ' ';
        }
        x = 0;
        while (*name && *name != '.')
                ++name;
        if (*name == '.') {
                ++name;
                while (*name && x < 2) {
                        if (*name >= 'a' && *name <= 'z')
                                d->suffix[x++] = *name++ - 'a' + 'A';
                        else
                                d->suffix[x++] = *name++;
                }
        } else {
                d->name[x++] = 'S';
                d->name[x++] = 'A';
        }
        while (x < 2) {
                d->suffix[x++] = ' ';
        }
        d->sector_high = (rib_sect >> 8);
        d->sector_low = rib_sect;
        d->attr_high = (attr >> 8);
        d->attr_low = attr;
        d->blank[0] = 0;
        d->blank[1] = 0;
        
        /* Compute hash of directory entry */
        hsh = hash(d->name);
        /* printf("hash=%d\n", hsh); */


        for (count = 0; count != 20; ++count) {
                int y;
                getsect(dir_buf, hsh + SECTOR_DIR);
                printf("Trying directory sector %d\n", hsh + SECTOR_DIR);
                for (y = 0; y != SECTOR_SIZE; y += 16) {
                        if (dir_buf[y] == 0x00 || dir_buf[y] == 0xFF) {
                                printf("OK, using entry %d\n", y);
                                memcpy(dir_buf + y, (char *)d, sizeof(struct dirent));
                                putsect(dir_buf, hsh + SECTOR_DIR);
                                return 0;
                        }
                }
                if (++hsh == SECTOR_DIR_SIZE)
                        hsh = 0;
        }

        printf("Out of directory entries\n");
        return -1;
}

/* Delete file */

int del_file(int rib_sector)
{
        int x;
        unsigned char rib[SECTOR_SIZE];
        struct rib *r = (struct rib *)rib;
        unsigned char cat[SECTOR_SIZE];
        getsect(rib, rib_sector);
        getsect(cat, SECTOR_CAT);
        for (x = 0; x != sizeof(r->sdw); x += 2) {
                int sdw = ((r->sdw[x] << 8) + r->sdw[x + 1]);
                if (sdw & 0x8000) {
                        break;
                } else {
                        int clust = (sdw & 0x3FF);
                        int siz = ((sdw >> 10) & 0x1F) + 1;
                        mark_space(cat, clust, siz, 0);
                }
        }
        putsect(cat, SECTOR_CAT);
        return 0;
}

/* Defined below in this file: (JMR20201103) */
int find_file(char *filename, int *type, int del);

/* Delete file name */

int rm(char *name)
{
        int type;
        int rib = find_file(name, &type, 1);
        if (rib != -1) {
                if (del_file(rib)) {
                        printf("Error deleting file '%s'\n", name);
                        return -1;
                } else {
                }	return 0;
        } else {
                printf("File '%s' not found\n", name);
                return -1;
        }
}

/* Put a file on the disk */

int put_file(char *local_name, char *mdos_name)
{
        FILE *f = fopen(local_name, "r");
        long size;
        long up;
        long x;
        unsigned char *buf;
        unsigned char cat[SECTOR_SIZE];
        int rib_sect;
        if (!f) {
                printf("Couldn't open '%s'\n", local_name);
                return -1;
        }
        if (fseek(f, 0, SEEK_END)) {
                printf("Couldn't get file size of '%s'\n", local_name);
                fclose(f);
                return -1;
        }
        size = ftell(f);
        if (size < 0)  {
                printf("Couldn't get file size of '%s'\n", local_name);
                fclose(f);
                return -1;
        }
        rewind(f);
        up = ((size + SECTOR_SIZE - 1) & ~(long)(SECTOR_SIZE - 1));
        buf = (unsigned char *)malloc(up);
        if (size != fread(buf, 1, size, f)) {
                printf("Couldn't read file '%s'\n", local_name);
                fclose(f);
                free(buf);
                return -1;
        }
        fclose(f);
        /* Convert UNIX line endings to MDOS */
        for (x = 0; x != size; ++x)
                if (buf[x] == '\n')
                        buf[x] = '\r';
        /* Fill with NULs to end of sector */
        while (x != up)
                buf[x++] = 0;

        /* Delete existing file */
        rm(mdos_name);

        /* Get cat... */
        getsect(cat, SECTOR_CAT);

        /* Allocate space and write file */
        rib_sect = write_file(cat, buf, up / SECTOR_SIZE);

        if (rib_sect == -1) {
                printf("Couldn't write file\n");
                return -1;
        }

        if (write_dir(mdos_name, rib_sect, 0x0500)) {
                printf("Couldn't write directory entry\n");
                return -1;
        }

        /* Success! */
        putsect(cat, SECTOR_CAT);
        return 0;
}

/* Directory listing */

char *typestr[]=
{
        "0 User defined",
        "1 Unknown",
        "2 Image",
        "3 Object",
        "4 Unknown",
        "5 ASCII",
        "6 Unknown",
        "7 ASCII coverted"
};

void mdos_dir(int all, int full, int single, int only_ascii)
{
        unsigned char buf[SECTOR_SIZE];
        unsigned char rib_buf[SECTOR_SIZE];
        int x, y;
        int rows;
        int cols = (80 / 13);
        for (x = SECTOR_DIR; x != SECTOR_DIR + SECTOR_DIR_SIZE; ++x) {
                int y;
                getsect(buf, x);
                for (y = 0; y != SECTOR_SIZE; y += sizeof(struct dirent)) {
                        struct dirent *d = (struct dirent *)(buf + y);
                        if (d->name[0] && d->name[0] != 0xFF) {
                                struct name *nam;
                                struct rib *rib = (struct rib *)rib_buf;
                                char s[50];
                                int p = 0;
                                int i;
                                for (i = 0; i != sizeof(d->name); i++) {
                                        s[p++] = lower(d->name[i]);
                                }
                                while (p && s[p - 1] == ' ') --p;
                                s[p++] = '.';
                                for (i = 0; i != sizeof(d->suffix); i++) {
                                        s[p++] = lower(d->suffix[i]);
                                }
                                while (p && s[p - 1] == ' ') --p;
                                s[p] = 0;
                                nam = (struct name *)malloc(sizeof(struct name));
                                nam->name = strdup(s);
                                nam->type = (d->attr_high & 7);
                                if (d->suffix[0] == 'C' && d->suffix[1] == 'M')
                                        nam->is_cm = 1;
                                else
                                        nam->is_cm = 0;
                                if (d->attr_high & 0x80)
                                        nam->write_protect = 1;
                                else
                                        nam->write_protect = 0;
                                if (d->attr_high & 0x40)
                                        nam->delete_protect = 1;
                                else
                                        nam->delete_protect = 0;
                                if (d->attr_high & 0x20)
                                        nam->system = 1;
                                else
                                        nam->system = 0;
                                if (d->attr_high & 0x10)
                                        nam->cont = 1;
                                else
                                        nam->cont = 0;
                                if (d->attr_high & 0x08)
                                        nam->compr = 1;
                                else
                                        nam->compr = 0;

                                nam->sector = (d->sector_high << 8) + d->sector_low;

                                /* Get RIB */
                                getsect(rib_buf, nam->sector);

                                /* Only valid for loadable files */
                                nam->size = (rib->size_high << 8) + rib->size_low;
                                nam->pc = (rib->pc_high << 8) + rib->pc_low;
                                nam->load = (rib->addr_high << 8) + rib->addr_low;

                                nam->last_size = rib->last_size;
                                nam->sects = calc_sects(rib); /* Size in sectors */

#if 0
                                printf("\nName=%s\n", nam->name);
                                printf("Size in sectors=%d, %d bytes\n", nam->sects, nam->sects * SECTOR_SIZE);
                                printf("load size=%d sectors, %d bytes\n", nam->size, (nam->size - 1) * SECTOR_SIZE + nam->last_size);
                                printf("Initial pc=%x\n", nam->pc);
                                printf("Load addr=%x\n", nam->load);
                                printf("Last_size=%d\n", nam->last_size);
#endif

                                if ((all || !nam->system) && (!only_ascii || nam->type == 5))
                                        names[name_n++] = nam;
                        }
                }
        }
        qsort(names, name_n, sizeof(struct name *), (int (*)(const void *, const void *))comp);

        if (full) {
                int totals = 0;
                printf("\n");
                for (x = 0; x != name_n; ++x) {
                        if (names[x]->type == 2) {
                                printf("-r%c%c%c%c%c%c %6d %-13s (%s) Load=$%4.4x, Jump=$%4.4x\n",
                                       (names[x]->write_protect ? '-' : 'w'),
                                       (names[x]->is_cm ? 'x' : '-'),
                                       (names[x]->system ? 's' : '-'),
                                       (names[x]->delete_protect ? 'd' : '-'),
                                       (names[x]->cont ? 'c' : '-'),
                                       (names[x]->compr ? 'z' : '-'),
                                       (names[x]->size - 1) * SECTOR_SIZE + names[x]->last_size, names[x]->name,
                                       typestr[names[x]->type], names[x]->load, names[x]->pc);
                                totals += names[x]->sects;
                        } else {
                                printf("-r%c%c%c%c%c%c %6d %-13s (%s)\n",
                                       (names[x]->write_protect ? '-' : 'w'),
                                       (names[x]->is_cm ? 'x' : '-'),
                                       (names[x]->system ? 's' : '-'),
                                       (names[x]->delete_protect ? 'd' : '-'),
                                       (names[x]->cont ? 'c' : '-'),
                                       (names[x]->compr ? 'z' : '-'),
                                       names[x]->sects * SECTOR_SIZE, names[x]->name,
                                       typestr[names[x]->type]);
                                totals += names[x]->sects;
                        }
                }
                printf("\n%d entries\n", name_n);
                printf("\n%d sectors, %d bytes\n", totals, totals * SECTOR_SIZE);
                printf("\n");
                do_free();
                printf("\n");
        } else if (single) {
                int x;
                for (x = 0; x != name_n; ++x) {
                        printf("%s\n", names[x]->name);
                }
        } else {

                /* Rows of 12 names each ordered like ls */

                rows = (name_n + cols - 1) / cols;

                for (y = 0; y != rows; ++y) {
                        for (x = 0; x != cols; ++x) {
                                int n = y + x * rows;
                                /* printf("%11d  ", n); */
                                if (n < name_n)
                                        printf("%-11s  ", names[n]->name);
                                else
                                        printf("             ");
                        }
                        printf("\n");
                }
        }
}

/* Find a file, return sector number of its rib */

int find_file(char *filename, int *type, int del)
{
        unsigned char buf[SECTOR_SIZE];
        unsigned char rib_buf[SECTOR_SIZE];
        int x, y;
        for (x = SECTOR_DIR; x != SECTOR_DIR + SECTOR_DIR_SIZE; ++x) {
                int y;
                getsect(buf, x);
                for (y = 0; y != SECTOR_SIZE; y += sizeof(struct dirent)) {
                        struct dirent *d = (struct dirent *)(buf + y);
                        if (d->name[0] && d->name[0] != 0xFF) {
                                struct name *nam;
                                struct rib *rib = (struct rib *)rib_buf;
                                char s[50];
                                int p = 0;
                                int i;
                                for (i = 0; i != sizeof(d->name); i++) {
                                        s[p++] = lower(d->name[i]);
                                }
                                while (p && s[p - 1] == ' ') --p;
                                s[p++] = '.';
                                for (i = 0; i != sizeof(d->suffix); i++) {
                                        s[p++] = lower(d->suffix[i]);
                                }
                                while (p && s[p - 1] == ' ') --p;
                                s[p] = 0;
                                if (!strcmp(s, filename)) {
                                        if (type)
                                                *type = (d->attr_high & 7);
                                        if (del) {
                                                d->name[0] = 0xFF;
                                                d->name[1] = 0xFF;
                                                putsect(buf, x);
                                        }
                                        return (d->sector_high << 8) + d->sector_low;
                                }
                        }
                }
        }
        return -1;
}

/* cat a file */

void cat(char *name)
{
        int type;
        int sector = find_file(name, &type, 0);
        if (sector == -1) {
                printf("File '%s' not found\n", name);
                exit(-1);
        } else {
                /* printf("Found file.  Sector of rib is %d\n", sector); */
                read_file(sector, type, stdout);
        }
}

/* get a file from the disk */

int get_file(char *mdos_name, char *local_name)
{
        int type;
        int sector = find_file(mdos_name, &type, 0);
        if (sector == -1) {
                printf("File '%s' not found\n", mdos_name);
                return -1;
        } else {
                FILE *f = fopen(local_name, "w");
                if (!f) {
                        printf("Couldn't open local file '%s'\n", local_name);
                        return -1;
                }
                /* printf("Found file.  Sector of rib is %d\n", sector); */
                read_file(sector, type, f);
                if (fclose(f)) {
                        printf("Couldn't close local file '%s'\n", local_name);
                        return -1;
                }
                return -1;
        }
}

int main(int argc, char *argv[])
{
        int all = 0;
        int full = 0;
        int single = 0;
        int only_ascii = 0;
	int x;
	char *disk_name;
	x = 1;
	if (x == argc) {
                printf("\nEXORciser MDOS diskette access\n");
                printf("\n");
                printf("Syntax: mdos path-to-diskette command args\n");
                printf("\n");
                printf("  Commands:\n");
                printf("      ls [-la1A]                    Directory listing\n");
                printf("                  -l for long\n");
                printf("                  -a to show system files\n");
                printf("                  -1 to show a single name per line\n");
                printf("                  -A show only ASCII files\n");
                printf("      cat mdos-name                 Type file to console\n");
                printf("      get mdos-name [local-name]    Copy file from diskette to local-name\n");
                printf("      put local-name [mdos-name]    Copy file to diskette to mdos-name\n");
                printf("      free                          Print amount of free space\n");
                printf("      rm mdos-name                  Delete a file\n");
                printf("\n");
                return -1;
	}
	disk_name = argv[x++];
	disk = fopen(disk_name, "r+");
	if (!disk) {
	        printf("Couldn't open '%s'\n", disk_name);
	        return -1;
	}

	/* Directory options */
	dir:
	while (x != argc && argv[x][0] == '-') {
	        int y;
	        for (y = 1;argv[x][y];++y) {
	                int opt = argv[x][y];
	                switch (opt) {
	                        case 'l': full = 1; break;
	                        case 'a': all = 1; break;
	                        case '1': single = 1; break;
	                        case 'A': only_ascii = 1; break;
	                        default: printf("Unknown option '%c'\n", opt); return -1;
	                }
	        }
	        ++x;
	}

	if (x == argc) {
	        /* Just print a directory listing */
	        mdos_dir(all, full, single, only_ascii);
	        return 0;
        } else if (!strcmp(argv[x], "ls")) {
                ++x;
                goto dir;
        } else if (!strcmp(argv[x], "free")) {
                return do_free();
	} else if (!strcmp(argv[x], "cat")) {
	        ++x;
	        if (x == argc) {
	                printf("Missing file name to cat\n");
	                return -1;
	        } else {
	                cat(argv[x++]);
	                return 0;
	        }
	} else if (!strcmp(argv[x], "get")) {
                char *local_name;
                char *mdos_name;
                ++x;
                if (x == argc) {
                        printf("Missing file name to get\n");
                        return -1;
                }
                mdos_name = argv[x];
                local_name = mdos_name;
                if (x + 1 != argc)
                        local_name = argv[++x];
                return get_file(mdos_name, local_name);
        } else if (!strcmp(argv[x], "put")) {
                char *local_name;
                char *mdos_name;
                ++x;
                if (x == argc) {
                        printf("Missing file name to put\n");
                        return -1;
                }
                local_name = argv[x];
                mdos_name = local_name;
                if (x + 1 != argc)
                        mdos_name = argv[++x];
                return put_file(local_name, mdos_name);
        } else if (!strcmp(argv[x], "rm")) {
                char *name;
                ++x;
                if (x == argc) {
                        printf("Missing name to delete\n");
                        return -1;
                } else {
                        name = argv[x];
                }
                return rm(name);
	} else {
	        printf("Unknown command '%s'\n", argv[x]);
	        return -1;
	}
	return 0;
}