Add ability to turn flat, non-GS .SHK files into proper disk images

This commit is contained in:
2012-07-31 21:06:01 +00:00
parent 77aa498bee
commit 437270ea95
6 changed files with 134 additions and 69 deletions

View File

@ -179,15 +179,10 @@ public class Disk {
int diskSize = 0; int diskSize = 0;
byte[] diskImage = null; byte[] diskImage = null;
if (isSDK()) { if (isSDK() || isSHK()) {
// If we have an SDK, unpack it and send along the byte array // If we have an SDK, unpack it and send along the byte array
diskImage = com.webcodepro.shrinkit.Utilities.unpackSDKFile(filename);
diskSize = diskImage.length;
} else if (isSHK()) {
// If we have an SHK, unpack it and send along the byte array
diskImage = com.webcodepro.shrinkit.Utilities.unpackSHKFile(filename); diskImage = com.webcodepro.shrinkit.Utilities.unpackSHKFile(filename);
throw new IOException("SHK unpacking is not implemented yet."); // TODO - remove me diskSize = diskImage.length;
//TODO - diskSize = diskImage.length;
} else { } else {
File file = new File(filename); File file = new File(filename);
diskSize = (int) file.length(); diskSize = (int) file.length();
@ -791,6 +786,29 @@ public class Disk {
setImageOrder(pdo); setImageOrder(pdo);
} }
/**
* Find the standard sized disk that will fit the requested number of bytes.
* @returns int size of the disk if it will satisfy the request, -1 otherwise
*/
public static int sizeToFit(long bytes) {
if (bytes < APPLE_140KB_DISK) {
return APPLE_140KB_DISK;
} else if (bytes < APPLE_800KB_DISK) {
return APPLE_800KB_DISK;
} else if (bytes < APPLE_5MB_HARDDISK) {
return APPLE_5MB_HARDDISK;
} else if (bytes < APPLE_10MB_HARDDISK) {
return APPLE_10MB_HARDDISK;
} else if (bytes < APPLE_20MB_HARDDISK) {
return APPLE_20MB_HARDDISK;
} else if (bytes < APPLE_32MB_HARDDISK) {
return APPLE_20MB_HARDDISK;
} else if (bytes < APPLE_32MB_HARDDISK) {
return APPLE_32MB_HARDDISK;
}
return -1;
}
/** /**
* Change ImageOrder from source order to target order by copying sector by * Change ImageOrder from source order to target order by copying sector by
* sector. * sector.

View File

@ -173,7 +173,7 @@ public class ProdosFileEntry extends ProdosCommonEntry implements FileEntry {
} }
/** /**
* Set the filetype. * Set the filetype based on a string value.
*/ */
public void setFiletype(String filetype) { public void setFiletype(String filetype) {
byte[] entry = readFileEntry(); byte[] entry = readFileEntry();
@ -181,6 +181,15 @@ public class ProdosFileEntry extends ProdosCommonEntry implements FileEntry {
writeFileEntry(entry); writeFileEntry(entry);
} }
/**
* Set the filetype based on a long - thunk into a byte
*/
public void setFiletype(long fileType) {
byte[] entry = readFileEntry();
entry[0x10] = (byte)fileType;
writeFileEntry(entry);
}
/** /**
* Indicate if this is an AppleWorks file. * Indicate if this is an AppleWorks file.
* Intended to force upper/lowercase into the filename. * Intended to force upper/lowercase into the filename.
@ -261,6 +270,15 @@ public class ProdosFileEntry extends ProdosCommonEntry implements FileEntry {
return AppleUtil.getWordValue(readFileEntry(), 0x1f); return AppleUtil.getWordValue(readFileEntry(), 0x1f);
} }
/**
* Set the auxiliary type for this file.
*/
public void setAuxiliaryType(int auxiliaryType) {
byte[] entry = readFileEntry();
setAuxiliaryType(entry, auxiliaryType);
writeFileEntry(entry);
}
/** /**
* Set the auxiliary type for this file. * Set the auxiliary type for this file.
*/ */

View File

@ -539,7 +539,7 @@ public class ac {
* Unshrink the ShrinkIt data depending on what kind it is: * Unshrink the ShrinkIt data depending on what kind it is:
* *
* SDK disk image - unpack it to a disk image * SDK disk image - unpack it to a disk image
* ShrinkIt file bundle [future] - unpack files onto a disk image of reqeusted size * ShrinkIt file bundle [future] - unpack files onto a disk image of requested size
*/ */
static void unshrink(String shrinkName, String imageName, int imageSize) static void unshrink(String shrinkName, String imageName, int imageSize)
throws IOException { throws IOException {

View File

@ -37,6 +37,7 @@ public class HeaderBlock {
private byte[] attribBytes; private byte[] attribBytes;
private String filename; private String filename;
private String rawFilename; private String rawFilename;
private long headerSize = 0;
private List<ThreadRecord> threads = new ArrayList<ThreadRecord>(); private List<ThreadRecord> threads = new ArrayList<ThreadRecord>();
/** /**
@ -88,7 +89,10 @@ public class HeaderBlock {
*/ */
public void readThreads(LittleEndianByteInputStream bs) throws IOException { public void readThreads(LittleEndianByteInputStream bs) throws IOException {
for (long l=0; l<totalThreads; l++) threads.add(new ThreadRecord(this, bs)); for (long l=0; l<totalThreads; l++) threads.add(new ThreadRecord(this, bs));
for (ThreadRecord r : threads) r.readThreadData(bs); for (ThreadRecord r : threads) {
r.readThreadData(bs);
headerSize += r.getThreadEof();
}
} }
/** /**
@ -242,4 +246,7 @@ public class HeaderBlock {
public void setThreadRecords(List<ThreadRecord> threads) { public void setThreadRecords(List<ThreadRecord> threads) {
this.threads = threads; this.threads = threads;
} }
public long getHeaderSize() {
return headerSize;
}
} }

View File

@ -15,6 +15,7 @@ import com.webcodepro.shrinkit.io.LittleEndianByteInputStream;
public class NuFileArchive { public class NuFileArchive {
private MasterHeaderBlock master; private MasterHeaderBlock master;
private List<HeaderBlock> headers; private List<HeaderBlock> headers;
private long totalSize = 0;
/** /**
* Need to enumerate some basic sub-types of archives. * Need to enumerate some basic sub-types of archives.
@ -34,13 +35,20 @@ public class NuFileArchive {
HeaderBlock header = new HeaderBlock(bs); HeaderBlock header = new HeaderBlock(bs);
header.readThreads(bs); header.readThreads(bs);
headers.add(header); headers.add(header);
totalSize += header.getHeaderSize();
} }
} }
/**
* @return long size in bytes of the archive
*/
public long getArchiveSize() {
return totalSize;
}
public MasterHeaderBlock getMasterHeaderBlock() { public MasterHeaderBlock getMasterHeaderBlock() {
return master; return master;
} }
public List<HeaderBlock> getHeaderBlocks() { public List<HeaderBlock> getHeaderBlocks() {
return headers; return headers;
} }}
}

View File

@ -23,8 +23,17 @@ import java.io.File;
import java.io.FileInputStream; import java.io.FileInputStream;
import java.io.IOException; import java.io.IOException;
import java.io.InputStream; import java.io.InputStream;
import java.util.Date;
import com.webcodepro.applecommander.storage.Disk;
import com.webcodepro.applecommander.storage.FileEntry;
import com.webcodepro.applecommander.storage.FormattedDisk;
import com.webcodepro.applecommander.storage.StorageBundle; import com.webcodepro.applecommander.storage.StorageBundle;
import com.webcodepro.applecommander.storage.os.prodos.ProdosFileEntry;
import com.webcodepro.applecommander.storage.os.prodos.ProdosFormatDisk;
import com.webcodepro.applecommander.storage.physical.ByteArrayImageLayout;
import com.webcodepro.applecommander.storage.physical.ImageOrder;
import com.webcodepro.applecommander.storage.physical.ProdosOrder;
import com.webcodepro.applecommander.util.TextBundle; import com.webcodepro.applecommander.util.TextBundle;
import com.webcodepro.shrinkit.io.LittleEndianByteInputStream; import com.webcodepro.shrinkit.io.LittleEndianByteInputStream;
@ -36,50 +45,9 @@ import com.webcodepro.shrinkit.io.LittleEndianByteInputStream;
public class Utilities public class Utilities
{ {
/** /**
* Interpret a SDK NuFile/NuFX/Shrinkit archive as a full disk image. * Interpret a NuFile/NuFX/Shrinkit archive as a full disk image.
* * Note that a disk within a shk (Disk Disintegrator Deluxe 5.0_D1.SHK) should
* @return byte[] buffer containing full disk of data; null if unable to read * be interpreted directly as that disk image.
* @throws IllegalArgumentException if the filename is not able to be read
* @throws IOException the file has some malformed-ness about it
*/
public static byte[] unpackSDKFile(String fileName) throws IOException {
TextBundle textBundle = StorageBundle.getInstance();
byte buffer[] = null;
ThreadRecord dataThread = null;
File file = new File(fileName);
if (file.isDirectory() || !file.canRead()) {
throw new IllegalArgumentException(textBundle.format("NotAFile", fileName, 1)); //$NON-NLS-1$
}
InputStream is = new FileInputStream(file);
NuFileArchive a = new NuFileArchive(is);
for (HeaderBlock b : a.getHeaderBlocks()) {
for (ThreadRecord r : b.getThreadRecords()) {
try
{
if (r.getThreadKind() == ThreadKind.DISK_IMAGE)
{
dataThread = r;
}
}
catch (Exception ex)
{
System.out.println(ex);
}
}
if (null != dataThread) {
dataThread.readThreadData(new LittleEndianByteInputStream(dataThread.getRawInputStream()));
InputStream fis = dataThread.getInputStream();
int dmgLen = (int)(dataThread.getThreadEof());
buffer = new byte[dmgLen];
fis.read(buffer,0,dmgLen);
fis.close();
}
}
return buffer;
}
/**
* Interpret a SHK NuFile/NuFX/Shrinkit archive as a full disk image.
* *
* @return byte[] buffer containing full disk of data; null if unable to read * @return byte[] buffer containing full disk of data; null if unable to read
* @throws IllegalArgumentException if the filename is not able to be read * @throws IllegalArgumentException if the filename is not able to be read
@ -87,21 +55,49 @@ public class Utilities
*/ */
public static byte[] unpackSHKFile(String fileName) throws IOException { public static byte[] unpackSHKFile(String fileName) throws IOException {
TextBundle textBundle = StorageBundle.getInstance(); TextBundle textBundle = StorageBundle.getInstance();
byte buffer[] = null; byte dmgBuffer[] = null;
ThreadRecord dataThread = null;
File file = new File(fileName); File file = new File(fileName);
if (file.isDirectory() || !file.canRead()) { if (file.isDirectory() || !file.canRead()) {
throw new IllegalArgumentException(textBundle.format("NotAFile", fileName, 1)); //$NON-NLS-1$ throw new IllegalArgumentException(textBundle.format("NotAFile", fileName, 1)); //$NON-NLS-1$
} }
InputStream is = new FileInputStream(file); InputStream is = new FileInputStream(file);
NuFileArchive a = new NuFileArchive(is); NuFileArchive a = new NuFileArchive(is);
int newDiskSize = Disk.sizeToFit(a.getArchiveSize());
ByteArrayImageLayout layout = new ByteArrayImageLayout(newDiskSize);
ImageOrder imageOrder = new ProdosOrder(layout);
FormattedDisk[] disks = ProdosFormatDisk.create(fileName, "APPLECOMMANDER", imageOrder);
ProdosFormatDisk pdDisk = (ProdosFormatDisk)disks[0];
for (HeaderBlock b : a.getHeaderBlocks()) { for (HeaderBlock b : a.getHeaderBlocks()) {
ProdosFileEntry newFile = null;
for (ThreadRecord r : b.getThreadRecords()) { for (ThreadRecord r : b.getThreadRecords()) {
try try
{ {
if (r.getThreadKind() == ThreadKind.DISK_IMAGE) switch (r.getThreadKind()) {
{ case ASCII_TEXT:
dataThread = r; break;
case ALLOCATED_SPACE:
break;
case APPLE_IIGS_ICON:
break;
case CREATE_DIRECTORY:
break;
case DATA_FORK:
// This is a normal-ish file
newFile = (ProdosFileEntry) pdDisk.createFile();
if (newFile != null) {
newFile.setFileData(readThread(r));
}
break;
case DISK_IMAGE:
dmgBuffer = readThread(r);
break;
case RESOURCE_FORK:
break;
case FILENAME:
break;
default:
// Hmmm, this should not occur - but let us not fret about it.
break;
} }
} }
catch (Exception ex) catch (Exception ex)
@ -109,15 +105,33 @@ public class Utilities
System.out.println(ex); System.out.println(ex);
} }
} }
if (null != dataThread) { if (newFile != null) {
dataThread.readThreadData(new LittleEndianByteInputStream(dataThread.getRawInputStream())); newFile.setFilename(b.getFilename());
InputStream fis = dataThread.getInputStream(); newFile.setFiletype(b.getFileType());
int dmgLen = (int)(dataThread.getThreadEof()); newFile.setAuxiliaryType((int)b.getExtraType());
buffer = new byte[dmgLen]; newFile.setCreationDate(b.getCreateWhen());
fis.read(buffer,0,dmgLen); newFile.setLastModificationDate(b.getModWhen());
newFile = null;
}
}
if (dmgBuffer != null)
return dmgBuffer;
else
return imageOrder.readBytes(0,newDiskSize);
}
/**
* readThread
*
* Reads the data from a thread
* @returns byte[] buffer
*/
public static byte[] readThread(ThreadRecord thread) throws IOException {
thread.readThreadData(new LittleEndianByteInputStream(thread.getRawInputStream()));
InputStream fis = thread.getInputStream();
byte[] buffer = new byte[(int)(thread.getThreadEof())];
fis.read(buffer,0,buffer.length);
fis.close(); fis.close();
}
}
return buffer; return buffer;
} }
} }