2008-12-26 20:36:47 +00:00
|
|
|
/*
|
|
|
|
* AppleCommander - An Apple ][ image utility.
|
2022-01-19 03:55:20 +00:00
|
|
|
* Copyright (C) 2002-2022 by Robert Greene
|
2008-12-26 20:36:47 +00:00
|
|
|
* robgreene at users.sourceforge.net
|
|
|
|
*
|
|
|
|
* This program 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 2 of the License, or (at your
|
|
|
|
* option) any later version.
|
|
|
|
*
|
|
|
|
* This program 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 program; if not, write to the Free Software Foundation, Inc.,
|
|
|
|
* 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
|
|
|
*/
|
|
|
|
package com.webcodepro.applecommander.storage.os.gutenberg;
|
|
|
|
|
|
|
|
import java.util.ArrayList;
|
|
|
|
import java.util.List;
|
|
|
|
|
|
|
|
import com.webcodepro.applecommander.storage.DirectoryEntry;
|
|
|
|
import com.webcodepro.applecommander.storage.DiskFullException;
|
|
|
|
import com.webcodepro.applecommander.storage.FileEntry;
|
|
|
|
import com.webcodepro.applecommander.storage.FormattedDisk;
|
|
|
|
import com.webcodepro.applecommander.storage.StorageBundle;
|
|
|
|
import com.webcodepro.applecommander.storage.physical.ImageOrder;
|
|
|
|
import com.webcodepro.applecommander.util.AppleUtil;
|
|
|
|
import com.webcodepro.applecommander.util.TextBundle;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Manages a disk that is in Gutenberg Word Processor format.
|
|
|
|
* <p>
|
|
|
|
* Date created: Dec 17, 2008 04:29:23 PM
|
|
|
|
* @author David Schmidt
|
|
|
|
*/
|
|
|
|
public class GutenbergFormatDisk extends FormattedDisk {
|
|
|
|
private TextBundle textBundle = StorageBundle.getInstance();
|
|
|
|
/**
|
|
|
|
* Indicates the index of the track in the location array.
|
|
|
|
*/
|
|
|
|
public static final int TRACK_LOCATION_INDEX = 0;
|
|
|
|
/**
|
|
|
|
* Indicates the index of the sector in the location array.
|
|
|
|
*/
|
|
|
|
public static final int SECTOR_LOCATION_INDEX = 1;
|
|
|
|
/**
|
|
|
|
* The catalog track.
|
|
|
|
*/
|
|
|
|
public static final int CATALOG_TRACK = 17;
|
|
|
|
/**
|
|
|
|
* The VTOC sector.
|
|
|
|
*/
|
|
|
|
public static final int VTOC_SECTOR = 7;
|
|
|
|
/**
|
|
|
|
* The standard track/sector pairs in a track/sector list.
|
|
|
|
*/
|
|
|
|
public static final int TRACK_SECTOR_PAIRS = 122;
|
|
|
|
/**
|
|
|
|
* The list of filetypes available.
|
|
|
|
*/
|
|
|
|
private static final String[] filetypes = {
|
|
|
|
"T" //$NON-NLS-1$
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Use this inner interface for managing the disk usage data.
|
2019-10-07 00:40:19 +00:00
|
|
|
* This off-loads format-specific implementation to the implementing class.
|
2008-12-26 20:36:47 +00:00
|
|
|
*/
|
|
|
|
private class WPDiskUsage implements DiskUsage {
|
|
|
|
private int[] location = null;
|
|
|
|
public boolean hasNext() {
|
|
|
|
return location == null
|
|
|
|
|| (location[TRACK_LOCATION_INDEX] < getTracks()
|
|
|
|
&& location[SECTOR_LOCATION_INDEX] < getSectors());
|
|
|
|
}
|
|
|
|
public void next() {
|
|
|
|
if (location == null) {
|
|
|
|
location = new int[2];
|
|
|
|
} else {
|
|
|
|
location[SECTOR_LOCATION_INDEX]++;
|
|
|
|
if (location[SECTOR_LOCATION_INDEX] >= getSectors()) {
|
|
|
|
location[SECTOR_LOCATION_INDEX] = 0;
|
|
|
|
location[TRACK_LOCATION_INDEX]++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
/**
|
|
|
|
* Get the free setting for the bitmap at the current location.
|
|
|
|
* I don't think there is a map stored on disk, however.
|
|
|
|
*/
|
|
|
|
public boolean isFree() {
|
|
|
|
if (location == null || location.length != 2) {
|
|
|
|
throw new IllegalArgumentException(StorageBundle.getInstance()
|
|
|
|
.get("DosFormatDisk.InvalidDimensionError")); //$NON-NLS-1$
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
public boolean isUsed() {
|
|
|
|
return !isFree();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Constructor for GutenbergFormatDisk.
|
|
|
|
*/
|
|
|
|
public GutenbergFormatDisk(String filename, ImageOrder imageOrder) {
|
|
|
|
super(filename, imageOrder);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Create a GutenbergFormatDisk. All DOS disk images are expected to
|
|
|
|
* be 140K in size.
|
|
|
|
*/
|
|
|
|
public static GutenbergFormatDisk[] create(String filename, ImageOrder imageOrder) {
|
|
|
|
GutenbergFormatDisk disk = new GutenbergFormatDisk(filename, imageOrder);
|
|
|
|
disk.format();
|
|
|
|
return new GutenbergFormatDisk[] { disk };
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Identify the operating system format of this disk as Gutenberg.
|
|
|
|
* @see com.webcodepro.applecommander.storage.FormattedDisk#getFormat()
|
|
|
|
*/
|
|
|
|
public String getFormat() {
|
|
|
|
return textBundle.get("Gutenberg"); //$NON-NLS-1$
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Retrieve a list of files.
|
|
|
|
* @see com.webcodepro.applecommander.storage.FormattedDisk#getFiles()
|
|
|
|
*/
|
2017-11-09 21:03:03 +00:00
|
|
|
public List<FileEntry> getFiles() {
|
|
|
|
List<FileEntry> list = new ArrayList<>();
|
2008-12-26 20:36:47 +00:00
|
|
|
int track = CATALOG_TRACK;
|
|
|
|
int sector = VTOC_SECTOR;
|
|
|
|
while (track < 40) { // iterate through all catalog sectors
|
|
|
|
byte[] catalogSector = readSector(track, sector);
|
2015-01-10 04:33:32 +00:00
|
|
|
int offset = 0x10; // First entry is 0x10 deep, except for the first one
|
2008-12-26 20:36:47 +00:00
|
|
|
while (offset < 0xff) { // iterate through all entries
|
2015-01-10 04:33:32 +00:00
|
|
|
if ((catalogSector[offset] != -96) && ((track != CATALOG_TRACK) || (track == CATALOG_TRACK && offset > 0x10))) {
|
2008-12-26 20:36:47 +00:00
|
|
|
list.add(new GutenbergFileEntry(this, track, sector, offset));
|
|
|
|
}
|
|
|
|
offset+= GutenbergFileEntry.FILE_DESCRIPTIVE_ENTRY_LENGTH;
|
|
|
|
}
|
|
|
|
track = AppleUtil.getUnsignedByte(catalogSector[4]); // Pull in the next catalog sector
|
|
|
|
sector = AppleUtil.getUnsignedByte(catalogSector[5]);
|
|
|
|
}
|
|
|
|
return list;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Create a FileEntry.
|
|
|
|
*/
|
|
|
|
public FileEntry createFile() throws DiskFullException {
|
|
|
|
byte[] vtoc = readVtoc();
|
|
|
|
int track = AppleUtil.getUnsignedByte(vtoc[1]);
|
|
|
|
int sector = AppleUtil.getUnsignedByte(vtoc[2]);
|
|
|
|
while (sector != 0) { // bug fix: iterate through all catalog _sectors_
|
|
|
|
byte[] catalogSector = readSector(track, sector);
|
|
|
|
int offset = 0x0b;
|
|
|
|
while (offset < 0xff) { // iterate through all entries
|
|
|
|
int value = AppleUtil.getUnsignedByte(catalogSector[offset]);
|
|
|
|
if (value == 0 || value == 0xff) {
|
|
|
|
return new GutenbergFileEntry(this, track, sector, offset);
|
|
|
|
}
|
|
|
|
offset+= GutenbergFileEntry.FILE_DESCRIPTIVE_ENTRY_LENGTH;
|
|
|
|
}
|
|
|
|
track = catalogSector[1];
|
|
|
|
sector = catalogSector[2];
|
|
|
|
}
|
2018-01-08 13:59:24 +00:00
|
|
|
throw new DiskFullException(
|
|
|
|
textBundle.get("DosFormatDisk.NoMoreSpaceError") //$NON-NLS-1$
|
|
|
|
, this.getFilename());
|
2008-12-26 20:36:47 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Identify if additional directories can be created. This
|
|
|
|
* may indicate that directories are not available to this
|
|
|
|
* operating system or simply that the disk image is "locked"
|
|
|
|
* to writing.
|
|
|
|
*/
|
|
|
|
public boolean canCreateDirectories() {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Indicates if this disk image can create a file.
|
2019-10-07 00:40:19 +00:00
|
|
|
* If not, the reason may be as simple as it has not been implemented
|
2008-12-26 20:36:47 +00:00
|
|
|
* to something specific about the disk.
|
|
|
|
*/
|
|
|
|
public boolean canCreateFile() {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Compute the amount of freespace available on the disk.
|
|
|
|
* This algorithm completely ignores tracks and sectors by
|
|
|
|
* running through the entire bitmap stored on the VTOC.
|
|
|
|
* @see com.webcodepro.applecommander.storage.FormattedDisk#getFreeSpace()
|
|
|
|
*/
|
|
|
|
public int getFreeSpace() {
|
|
|
|
return getFreeSectors() * SECTOR_SIZE;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Comput the number of free sectors available on the disk.
|
|
|
|
*/
|
|
|
|
public int getFreeSectors() {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Return the amount of used space in bytes.
|
|
|
|
* @see com.webcodepro.applecommander.storage.FormattedDisk#getUsedSpace()
|
|
|
|
*/
|
|
|
|
public int getUsedSpace() {
|
|
|
|
return APPLE_140KB_DISK;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Compute the number of used sectors on the disk.
|
|
|
|
*/
|
|
|
|
public int getUsedSectors() {
|
|
|
|
return getTotalSectors() - getFreeSectors();
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Compute the total number of sectors available on the disk.
|
|
|
|
*/
|
|
|
|
public int getTotalSectors() {
|
|
|
|
int tracks = getTracks();
|
|
|
|
int sectors = getSectors();
|
|
|
|
return tracks * sectors;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Return the WP disk name.
|
|
|
|
* @see com.webcodepro.applecommander.storage.FormattedDisk#getDiskName()
|
|
|
|
*/
|
|
|
|
public String getDiskName() {
|
|
|
|
// Pull the disk name out...
|
|
|
|
return AppleUtil.getString(readVtoc(), 6, 9).trim();
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Return the VTOC (Volume Table Of Contents).
|
|
|
|
*/
|
|
|
|
protected byte[] readVtoc() {
|
|
|
|
return readSector(CATALOG_TRACK, VTOC_SECTOR);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Save the VTOC (Volume Table Of Contents) to disk.
|
|
|
|
*/
|
|
|
|
protected void writeVtoc(byte[] vtoc) {
|
|
|
|
writeSector(CATALOG_TRACK, VTOC_SECTOR, vtoc);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get the disk usage iterator.
|
|
|
|
*/
|
|
|
|
public DiskUsage getDiskUsage() {
|
|
|
|
return new WPDiskUsage();
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get the number of tracks on this disk.
|
|
|
|
*/
|
|
|
|
public int getTracks() {
|
|
|
|
byte[] vtoc = readVtoc();
|
|
|
|
return AppleUtil.getUnsignedByte(vtoc[0x34]);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get the number of sectors on this disk.
|
|
|
|
*/
|
|
|
|
public int getSectors() {
|
|
|
|
byte[] vtoc = readVtoc();
|
|
|
|
return AppleUtil.getUnsignedByte(vtoc[0x35]);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get suggested dimensions for display of bitmap. For DOS 3.3, that information
|
|
|
|
* is stored in the VTOC, and that information is fairly important.
|
|
|
|
* @see com.webcodepro.applecommander.storage.FormattedDisk#getBitmapDimensions()
|
|
|
|
*/
|
|
|
|
public int[] getBitmapDimensions() {
|
|
|
|
int tracks = getTracks();
|
|
|
|
int sectors = getSectors();
|
|
|
|
return new int[] { tracks, sectors };
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get the length of the bitmap.
|
|
|
|
*/
|
|
|
|
public int getBitmapLength() {
|
|
|
|
return getTotalSectors();
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get the labels to use in the bitmap.
|
|
|
|
*/
|
|
|
|
public String[] getBitmapLabels() {
|
|
|
|
return new String[] { textBundle.get("DosFormatDisk.Track"), textBundle.get("DosFormatDisk.Sector") }; //$NON-NLS-1$ //$NON-NLS-2$
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get WP-specific disk information.
|
|
|
|
*/
|
2017-11-09 21:03:03 +00:00
|
|
|
public List<DiskInformation> getDiskInformation() {
|
|
|
|
List<DiskInformation> list = super.getDiskInformation();
|
2008-12-26 20:36:47 +00:00
|
|
|
return list;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get the standard file column header information.
|
|
|
|
* This default implementation is intended only for standard mode.
|
|
|
|
*/
|
2017-11-09 21:03:03 +00:00
|
|
|
public List<FileColumnHeader> getFileColumnHeaders(int displayMode) {
|
|
|
|
List<FileColumnHeader> list = new ArrayList<>();
|
2008-12-26 20:36:47 +00:00
|
|
|
switch (displayMode) {
|
|
|
|
case FILE_DISPLAY_NATIVE:
|
2020-11-21 20:59:45 +00:00
|
|
|
list.add(new FileColumnHeader(" ", 1, FileColumnHeader.ALIGN_CENTER, "locked"));
|
|
|
|
list.add(new FileColumnHeader(textBundle.get("DosFormatDisk.Type"), 1,
|
|
|
|
FileColumnHeader.ALIGN_CENTER, "type"));
|
|
|
|
list.add(new FileColumnHeader(textBundle.get("DosFormatDisk.SizeInSectors"), 3,
|
|
|
|
FileColumnHeader.ALIGN_RIGHT, "sectors"));
|
|
|
|
list.add(new FileColumnHeader(textBundle.get("Name"), 30,
|
|
|
|
FileColumnHeader.ALIGN_LEFT, "name"));
|
2008-12-26 20:36:47 +00:00
|
|
|
break;
|
|
|
|
case FILE_DISPLAY_DETAIL:
|
2020-11-21 20:59:45 +00:00
|
|
|
list.add(new FileColumnHeader(" ", 1, FileColumnHeader.ALIGN_CENTER, "locked"));
|
|
|
|
list.add(new FileColumnHeader(textBundle.get("DosFormatDisk.Type"), 1,
|
|
|
|
FileColumnHeader.ALIGN_CENTER, "type"));
|
|
|
|
list.add(new FileColumnHeader(textBundle.get("Name"), 30,
|
|
|
|
FileColumnHeader.ALIGN_LEFT, "name"));
|
|
|
|
list.add(new FileColumnHeader(textBundle.get("SizeInBytes"), 6,
|
|
|
|
FileColumnHeader.ALIGN_RIGHT, "size"));
|
|
|
|
list.add(new FileColumnHeader(textBundle.get("DosFormatDisk.SizeInSectors"), 3,
|
|
|
|
FileColumnHeader.ALIGN_RIGHT, "sectors"));
|
|
|
|
list.add(new FileColumnHeader(textBundle.get("DeletedQ"), 7,
|
|
|
|
FileColumnHeader.ALIGN_CENTER, "deleted"));
|
|
|
|
list.add(new FileColumnHeader(textBundle.get("DosFormatDisk.TrackAndSectorList"), 7,
|
|
|
|
FileColumnHeader.ALIGN_CENTER, "trackAndSectorList"));
|
2008-12-26 20:36:47 +00:00
|
|
|
break;
|
|
|
|
default: // FILE_DISPLAY_STANDARD
|
|
|
|
list.addAll(super.getFileColumnHeaders(displayMode));
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
return list;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Indicates if this disk format supports "deleted" files.
|
|
|
|
*/
|
|
|
|
public boolean supportsDeletedFiles() {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Indicates if this disk image can read data from a file.
|
|
|
|
*/
|
|
|
|
public boolean canReadFileData() {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Indicates if this disk image can write data to a file.
|
|
|
|
*/
|
|
|
|
public boolean canWriteFileData() {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Identify if this disk format as not capable of having directories.
|
|
|
|
* @see com.webcodepro.applecommander.storage.FormattedDisk#canHaveDirectories()
|
|
|
|
*/
|
|
|
|
public boolean canHaveDirectories() {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Indicates if this disk image can delete a file.
|
|
|
|
*/
|
|
|
|
public boolean canDeleteFile() {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get the data associated with the specified FileEntry.
|
|
|
|
*/
|
|
|
|
public byte[] getFileData(FileEntry fileEntry) {
|
|
|
|
if ( !(fileEntry instanceof GutenbergFileEntry)) {
|
|
|
|
throw new IllegalArgumentException(textBundle.get("DosFormatDisk.InvalidFileEntryError")); //$NON-NLS-1$
|
|
|
|
}
|
|
|
|
GutenbergFileEntry wpEntry = (GutenbergFileEntry) fileEntry;
|
|
|
|
// Size is calculated by sectors used - not actual size - as size varies
|
|
|
|
// on filetype, etc.
|
|
|
|
int filesize = wpEntry.getSectorsUsed();
|
|
|
|
byte[] fileData = null;
|
|
|
|
if (filesize > 0) {
|
|
|
|
fileData = new byte[(wpEntry.getSectorsUsed()) * SECTOR_SIZE];
|
|
|
|
} else {
|
|
|
|
fileData = new byte[0];
|
|
|
|
// don't need to load it - also bypass potential issues
|
|
|
|
return fileData;
|
|
|
|
}
|
|
|
|
int track = wpEntry.getTrack();
|
|
|
|
int sector = wpEntry.getSector();
|
|
|
|
int offset = 0;
|
|
|
|
while (track < 128) {
|
|
|
|
byte[] sectorData = readSector(track,sector);
|
|
|
|
track = AppleUtil.getUnsignedByte(sectorData[0x04]);
|
|
|
|
sector = AppleUtil.getUnsignedByte(sectorData[0x05]);
|
|
|
|
System.arraycopy(sectorData, 6, fileData, offset, sectorData.length-6);
|
|
|
|
offset+= sectorData.length-6;
|
|
|
|
}
|
|
|
|
return fileData;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Writes the raw bytes into the file. This bypasses any special formatting
|
|
|
|
* of the data (such as prepending the data with a length and/or an address).
|
|
|
|
* Typically, the FileEntry.setFileData method should be used.
|
|
|
|
*/
|
|
|
|
public void setFileData(FileEntry fileEntry, byte[] fileData) throws DiskFullException {
|
|
|
|
setFileData((GutenbergFileEntry)fileEntry, fileData);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Set the data associated with the specified GutenbergFileEntry into sectors
|
|
|
|
* on the disk.
|
|
|
|
*/
|
|
|
|
protected void setFileData(GutenbergFileEntry fileEntry, byte[] data) throws DiskFullException {
|
|
|
|
// compute free space and see if the data will fit!
|
|
|
|
int numberOfDataSectors = (data.length + SECTOR_SIZE - 1) / SECTOR_SIZE;
|
|
|
|
int numberOfSectors = numberOfDataSectors +
|
|
|
|
(numberOfDataSectors + TRACK_SECTOR_PAIRS - 1) / TRACK_SECTOR_PAIRS;
|
|
|
|
if (numberOfSectors > getFreeSectors() + fileEntry.getSectorsUsed()) {
|
|
|
|
throw new DiskFullException(
|
|
|
|
textBundle.format("DosFormatDisk.NotEnoughSectorsError", //$NON-NLS-1$
|
2018-01-08 13:59:24 +00:00
|
|
|
numberOfSectors, getFreeSectors())
|
|
|
|
, this.getFilename());
|
2008-12-26 20:36:47 +00:00
|
|
|
}
|
|
|
|
// free "old" data and just rewrite stuff...
|
|
|
|
// freeSectors(fileEntry); (not going to work...)
|
|
|
|
byte[] vtoc = readVtoc();
|
|
|
|
int track = fileEntry.getTrack();
|
|
|
|
int sector = fileEntry.getSector();
|
|
|
|
if (track == 0 || track == 255) {
|
|
|
|
track = 1;
|
|
|
|
sector = 0;
|
|
|
|
while (true) {
|
|
|
|
if (isSectorFree(track,sector,vtoc)) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
sector++;
|
|
|
|
if (sector >= getSectors()) {
|
|
|
|
track++;
|
|
|
|
sector = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
fileEntry.setTrack(track);
|
|
|
|
fileEntry.setSector(sector);
|
|
|
|
}
|
|
|
|
setSectorUsed(track, sector, vtoc);
|
|
|
|
byte[] trackSectorList = new byte[SECTOR_SIZE];
|
|
|
|
int offset = 0;
|
|
|
|
int trackSectorOffset = 0x0c;
|
|
|
|
int totalSectors = 0;
|
|
|
|
int t=1; // initial search for space
|
|
|
|
int s=0;
|
|
|
|
while (offset < data.length) {
|
|
|
|
// locate next free sector
|
|
|
|
while (true) {
|
|
|
|
if (isSectorFree(t,s,vtoc)) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
s++;
|
|
|
|
if (s >= getSectors()) {
|
|
|
|
t++;
|
|
|
|
s = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
setSectorUsed(t,s,vtoc);
|
|
|
|
if (trackSectorOffset >= 0x100) {
|
|
|
|
// filled up the first track/sector list - save it
|
|
|
|
trackSectorList[0x01] = (byte) t;
|
|
|
|
trackSectorList[0x02] = (byte) s;
|
|
|
|
writeSector(track, sector, trackSectorList);
|
|
|
|
trackSectorList = new byte[SECTOR_SIZE];
|
|
|
|
trackSectorOffset = 0x0c;
|
|
|
|
track = t;
|
|
|
|
sector = s;
|
|
|
|
} else {
|
|
|
|
// write out a data sector
|
|
|
|
trackSectorList[trackSectorOffset] = (byte) t;
|
|
|
|
trackSectorList[trackSectorOffset+1] = (byte) s;
|
|
|
|
trackSectorOffset+= 2;
|
|
|
|
byte[] sectorData = new byte[SECTOR_SIZE];
|
|
|
|
int length = Math.min(SECTOR_SIZE, data.length - offset);
|
|
|
|
System.arraycopy(data, offset, sectorData, 0, length);
|
|
|
|
writeSector(t,s,sectorData);
|
|
|
|
offset+= SECTOR_SIZE;
|
|
|
|
}
|
|
|
|
totalSectors++;
|
|
|
|
}
|
|
|
|
writeSector(track, sector, trackSectorList); // last T/S list
|
|
|
|
totalSectors++;
|
|
|
|
fileEntry.setSectorsUsed(totalSectors);
|
|
|
|
writeVtoc(vtoc);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Format the disk as DOS 3.3.
|
|
|
|
* @see com.webcodepro.applecommander.storage.FormattedDisk#format()
|
|
|
|
*/
|
|
|
|
public void format() {
|
|
|
|
getImageOrder().format();
|
|
|
|
format(15, 35, 16);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Format the disk as DOS 3.3 given the dymanic parameters.
|
|
|
|
* (Used for UniDOS and OzDOS.)
|
|
|
|
*/
|
|
|
|
protected void format(int firstCatalogSector, int tracksPerDisk,
|
|
|
|
int sectorsPerTrack) {
|
|
|
|
|
|
|
|
writeBootCode();
|
|
|
|
// create catalog sectors
|
|
|
|
byte[] data = new byte[SECTOR_SIZE];
|
|
|
|
for (int sector=firstCatalogSector; sector > 0; sector--) {
|
|
|
|
if (sector > 1) {
|
|
|
|
data[0x01] = CATALOG_TRACK;
|
|
|
|
data[0x02] = (byte)(sector-1);
|
|
|
|
} else {
|
|
|
|
data[0x01] = 0;
|
|
|
|
data[0x02] = 0;
|
|
|
|
}
|
|
|
|
writeSector(CATALOG_TRACK, sector, data);
|
|
|
|
}
|
|
|
|
// create VTOC
|
|
|
|
data[0x01] = CATALOG_TRACK; // track# of first catalog sector
|
|
|
|
data[0x02] = (byte)firstCatalogSector; // sector# of first catalog sector
|
|
|
|
data[0x03] = 3; // DOS 3.3 formatted
|
|
|
|
data[0x06] = (byte)254; // DISK VOLUME#
|
|
|
|
data[0x27] = TRACK_SECTOR_PAIRS;// maximum # of T/S pairs in a sector
|
|
|
|
data[0x30] = CATALOG_TRACK+1; // last track where sectors allocated
|
|
|
|
data[0x31] = 1; // direction of allocation
|
|
|
|
data[0x34] = (byte)tracksPerDisk; // tracks per disk
|
|
|
|
data[0x35] = (byte)sectorsPerTrack;// sectors per track
|
|
|
|
data[0x37] = 1; // 36/37 are # of bytes per sector
|
|
|
|
for (int track=0; track<tracksPerDisk; track++) {
|
|
|
|
for (int sector=0; sector<sectorsPerTrack; sector++) {
|
|
|
|
if (track == 0 || track == CATALOG_TRACK) {
|
|
|
|
setSectorUsed(track, sector, data);
|
|
|
|
} else {
|
|
|
|
setSectorFree(track, sector, data);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
writeVtoc(data);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Indicates if a specific track/sector is free.
|
|
|
|
*/
|
|
|
|
public boolean isSectorFree(int track, int sector, byte[] vtoc) {
|
|
|
|
checkRange(track, sector);
|
|
|
|
byte byt = vtoc[getFreeMapByte(track, sector)];
|
|
|
|
return AppleUtil.isBitSet(byt, getFreeMapBit(sector));
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Indicates if a specific track/sector is used.
|
|
|
|
*/
|
|
|
|
public boolean isSectorUsed(int track, int sector, byte[] vtoc) {
|
|
|
|
return !isSectorFree(track, sector, vtoc);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Sets the track/sector indicator to free.
|
|
|
|
*/
|
|
|
|
public void setSectorFree(int track, int sector, byte[] vtoc) {
|
|
|
|
checkRange(track, sector);
|
|
|
|
int offset = getFreeMapByte(track, sector);
|
|
|
|
byte byt = vtoc[offset];
|
|
|
|
byt = AppleUtil.setBit(byt, getFreeMapBit(sector));
|
|
|
|
vtoc[offset] = byt;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Sets the track/sector indicator to used.
|
|
|
|
*/
|
|
|
|
public void setSectorUsed(int track, int sector, byte[] vtoc) {
|
|
|
|
checkRange(track, sector);
|
|
|
|
int offset = getFreeMapByte(track, sector);
|
|
|
|
byte byt = vtoc[offset];
|
|
|
|
byt = AppleUtil.clearBit(byt, getFreeMapBit(sector));
|
|
|
|
vtoc[offset] = byt;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Compute the VTOC byte for the T/S map.
|
|
|
|
*/
|
|
|
|
protected int getFreeMapByte(int track, int sector) {
|
|
|
|
int trackOffset = track * 4;
|
|
|
|
int sectorOffset = 1 - ((sector & 0x8) >> 3);
|
|
|
|
return 0x38 + trackOffset + sectorOffset;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Compute the VTOC bit for the T/S map.
|
|
|
|
*/
|
|
|
|
protected int getFreeMapBit(int sector) {
|
|
|
|
int bit = sector & 0x7;
|
|
|
|
return bit;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Validate track/sector range. This just validates the
|
|
|
|
* maximum values allowable for track and sector.
|
|
|
|
*/
|
|
|
|
protected void checkRange(int track, int sector) {
|
|
|
|
if (track > 50 || sector > 32) {
|
|
|
|
throw new IllegalArgumentException(
|
|
|
|
textBundle.format("DosFormatDisk.InvalidTrackAndSectorCombinationError", //$NON-NLS-1$
|
|
|
|
track, sector));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns the logical disk number. Returns a 0 to indicate no numbering.
|
|
|
|
*/
|
|
|
|
public int getLogicalDiskNumber() {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns a valid filename for the given filename.
|
|
|
|
*/
|
|
|
|
public String getSuggestedFilename(String filename) {
|
|
|
|
int len = Math.min(filename.length(), 12);
|
|
|
|
return filename.toUpperCase().substring(0, len).trim();
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns a valid filetype for the given filename. The most simple
|
|
|
|
* format will just assume a filetype of binary. This method is
|
|
|
|
* available for the interface to make an intelligent first guess
|
|
|
|
* as to the filetype.
|
|
|
|
*/
|
|
|
|
public String getSuggestedFiletype(String filename) {
|
|
|
|
return "T"; //$NON-NLS-1$
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns a list of possible file types. Since the filetype is
|
|
|
|
* specific to each operating system, a simple String is used.
|
|
|
|
*/
|
|
|
|
public String[] getFiletypes() {
|
|
|
|
return filetypes;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Indicates if this filetype requires an address component.
|
|
|
|
* For DOS, only the Binary type needs an address.
|
|
|
|
*/
|
|
|
|
public boolean needsAddress(String filetype) {
|
|
|
|
return "B".equals(filetype); //$NON-NLS-1$
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Indicates if this FormattedDisk supports a disk map.
|
|
|
|
*/
|
|
|
|
public boolean supportsDiskMap() {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Change to a different ImageOrder. Remains in DOS 3.3 format but
|
2019-10-07 00:40:19 +00:00
|
|
|
* the underlying order can change.
|
2008-12-26 20:36:47 +00:00
|
|
|
* @see ImageOrder
|
|
|
|
*/
|
|
|
|
public void changeImageOrder(ImageOrder imageOrder) {
|
|
|
|
AppleUtil.changeImageOrderByTrackAndSector(getImageOrder(), imageOrder);
|
|
|
|
setImageOrder(imageOrder);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Create a new DirectoryEntry.
|
2018-03-10 04:49:40 +00:00
|
|
|
* @see com.webcodepro.applecommander.storage.DirectoryEntry#createDirectory(String)
|
2012-08-08 03:10:57 +00:00
|
|
|
*/
|
|
|
|
public DirectoryEntry createDirectory(String name) throws DiskFullException {
|
|
|
|
throw new UnsupportedOperationException(textBundle.get("DirectoryCreationNotSupported")); //$NON-NLS-1$
|
|
|
|
}
|
2008-12-26 20:36:47 +00:00
|
|
|
}
|