From a3919a0bb40a44953e0ce8a478feceb80bd9a299 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Szymon=20=C5=81opaciuk?= Date: Thu, 7 Feb 2019 21:24:59 +0000 Subject: [PATCH 1/2] Add support for using AutQuit7 MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit This allows the user to choose whether to use AutoQuit or AutQuit7 (which supports System 7). When the path to AutQuit7 is specified, it will be configured according to the instructions at https://www.gryphel.com/c/minivmac/extras/autquit7/ by putting an alias to AutQuit7 in the "Startup Items" folder. That will cause the application named "app" in the folder of AutQuit7 (top level, no change) to start on boot, and will shut down the OS when quit. Signed-off-by: Szymon Łopaciuk --- LaunchAPPL/Client/MiniVMac.cc | 67 +++++++++++++++++++++++++++++++---- LaunchAPPL/Client/MiniVMac.h | 39 ++++++++++++++++++++ 2 files changed, 99 insertions(+), 7 deletions(-) diff --git a/LaunchAPPL/Client/MiniVMac.cc b/LaunchAPPL/Client/MiniVMac.cc index 3f88cd8ea6..0d353e5459 100644 --- a/LaunchAPPL/Client/MiniVMac.cc +++ b/LaunchAPPL/Client/MiniVMac.cc @@ -2,6 +2,7 @@ #include "Launcher.h" #include "Utilities.h" #include "BinaryIO.h" +#include "ResourceFile.h" extern "C" { #include "hfs.h" @@ -35,6 +36,7 @@ class MiniVMacLauncher : public Launcher hfsvol *vol; void CopySystemFile(const std::string& fn, bool required); + void MakeAlias(const std::string& dest, const std::string& src); fs::path ConvertImage(const fs::path& path); public: MiniVMacLauncher(po::variables_map& options); @@ -126,8 +128,10 @@ MiniVMacLauncher::MiniVMacLauncher(po::variables_map &options) vmacDir = fs::absolute( options["minivmac-dir"].as() ); vmacPath = fs::absolute( options["minivmac-path"].as(), vmacDir ); + bool usesAutQuit7 = options.count("autquit7-image"); + systemImage = fs::absolute(options["system-image"].as(), vmacDir); - fs::path autoquitImage = fs::absolute(options["autoquit-image"].as(), vmacDir); + fs::path autoquitImage = fs::absolute(options[usesAutQuit7 ? "autquit7-image" : "autoquit-image"].as(), vmacDir); systemImage = ConvertImage(systemImage); autoquitImage = ConvertImage(autoquitImage); @@ -147,8 +151,9 @@ MiniVMacLauncher::MiniVMacLauncher(po::variables_map &options) hfs_format(imagePath.string().c_str(), 0, 0, "SysAndApp", 0, NULL); { - bootblock1[0x1A] = 8; - memcpy(&bootblock1[0x1B],"AutoQuit", 8); + std::string finderName = std::string(usesAutQuit7 ? "Finder" : "AutoQuit"); + bootblock1[0x1A] = finderName.size(); + memcpy(&bootblock1[0x1B], finderName.c_str(), finderName.size()); bootblock1[0x5A] = 3; memcpy(&bootblock1[0x5B],"App", 3); @@ -176,6 +181,11 @@ MiniVMacLauncher::MiniVMacLauncher(po::variables_map &options) CopySystemFile(systemFileName, true); CopySystemFile("MacsBug", false); + if (usesAutQuit7) + { + CopySystemFile("Finder", true); + } + { std::ostringstream rsrcOut; app.resources.writeFork(rsrcOut); @@ -190,18 +200,27 @@ MiniVMacLauncher::MiniVMacLauncher(po::variables_map &options) hfs_close(file); } - hfs_umount(sysvol); + hfs_umount(sysvol); sysvol = NULL; sysvol = hfs_mount(autoquitImage.string().c_str(),0, HFS_MODE_RDONLY); if(!sysvol) throw std::runtime_error("Cannot open disk image: " + autoquitImage.string()); assert(sysvol); - CopySystemFile("AutoQuit", true); + if (usesAutQuit7) + { + CopySystemFile("AutQuit7", true); + MakeAlias("AutQuit7 alias", "AutQuit7"); + hfs_mkdir(vol, "Startup Items"); + hfs_rename(vol, "AutQuit7 alias", "Startup Items"); + } + else + { + CopySystemFile("AutoQuit", true); + } { hfsfile *file = hfs_create(vol, "out", "TEXT", "MPS "); hfs_close(file); } - hfs_umount(sysvol); sysvol = NULL; hfs_umount(vol); vol = NULL; @@ -326,6 +345,39 @@ void MiniVMacLauncher::CopySystemFile(const std::string &fn, bool required) hfs_close(out); } +void MiniVMacLauncher::MakeAlias(const std::string& dest, const std::string& src) +{ + hfsdirent ent; + hfsvolent vent; + + hfs_stat(vol, src.c_str(), &ent); + hfs_vstat(vol, &vent); + + AliasData alias; + alias.size = sizeof(AliasData); + memcpy(&(alias.volumeName), vent.name, 27); + alias.volumeNameSize = strlen(vent.name); + alias.volumeCreationDate = vent.crdate; + alias.parentDirID = ent.parid; + memcpy(&(alias.fileName), ent.name, 63); + alias.fileNameSize = strlen(ent.name); + alias.fileNum = ent.cnid; + alias.fileCreationDate = ent.crdate; + memcpy(&(alias.typeCode), ent.u.file.type, 4); + memcpy(&(alias.creatorCode), ent.u.file.creator, 4); + + std::ostringstream roalias; + Resources res; + res.addResource(Resource("alis", 0, std::string((char*)&alias, sizeof(AliasData)))); + res.writeFork(roalias); + std::string ralias = roalias.str(); + + hfsfile *falias = hfs_create(vol, dest.c_str(), "adrp", ent.u.file.creator); + hfs_setfork(falias, 1); + hfs_write(falias, ralias.data(), ralias.size()); + hfs_close(falias); +} + bool MiniVMacLauncher::Go(int timeout) { @@ -361,6 +413,7 @@ void MiniVMac::GetOptions(options_description &desc) ("minivmac-rom", po::value()->default_value("./vMac.ROM"),"minivmac ROM file") ("system-image", po::value(),"path to disk image with system") ("autoquit-image", po::value(),"path to autoquit disk image, available from the minivmac web site") + ("autquit7-image", po::value(),"path to autquit7 disk image, available from the minivmac web site") ; } @@ -370,7 +423,7 @@ bool MiniVMac::CheckOptions(variables_map &options) && options.count("minivmac-dir") != 0 && options.count("minivmac-rom") != 0 && options.count("system-image") != 0 - && options.count("autoquit-image") != 0; + && (options.count("autoquit-image") + options.count("autquit7-image")) == 1; } std::unique_ptr MiniVMac::MakeLauncher(variables_map &options) diff --git a/LaunchAPPL/Client/MiniVMac.h b/LaunchAPPL/Client/MiniVMac.h index 0f83aa4ad8..b98c722c7c 100644 --- a/LaunchAPPL/Client/MiniVMac.h +++ b/LaunchAPPL/Client/MiniVMac.h @@ -3,6 +3,45 @@ #include "LaunchMethod.h" + +/* Adapted from http://sebastien.kirche.free.fr/python_stuff/MacOS-aliases.txt */ +typedef struct +{ + int16_t type; /* type of data */ + int16_t size; /* length of variable length data */ + char data[]; /* actual data */ +} VarData; + +typedef struct +{ + /* Type Code: alis */ + char userType[4] = {0, 0, 0, 0}; /* for application use, can be zeros */ + uint16_t size; /* alias record size, including variable length data */ + int16_t version = 2; /* alias version, current 2 */ + int16_t type = 0; /* file = 0, directory = 1 */ + char volumeNameSize; + char volumeName[27]; /* volume name */ + uint32_t volumeCreationDate; /* volume creation date, seconds since 1904 */ + uint16_t volumeSig = 0x4244; /*BD*/ /* volume signature MFS = RW, HFS = BD */ + int16_t volumeType = 5; /* [HD] = 0, Foreign = 1, Floppy: 400K, 800K, 1400K = 2-4, OtherEjectable = 5 */ + int32_t parentDirID; /* parent directory ID, 0 for relative searches */ + char fileNameSize; + char fileName[63]; /* file or directory name */ + int32_t fileNum; /* file number or directory ID */ + uint32_t fileCreationDate; /* file or directory creation date, seconds since 1904 */ + char typeCode[4]; /* file type */ + char creatorCode[4]; /* file's creator */ + int16_t nlvlFrom = 0; /* next level up from alias, used in relative searches */ + int16_t nlvlTo = 0; /* next level down to target, ditto */ + uint32_t volumeAttr = 0; /* various flags (locked, ejectable), see link above */ + int16_t volumeFSID = 0; /* file system ID for the volume, 0 for MFS, HFS */ + int16_t unused = 0; + uint32_t unused1 = 0; + uint32_t unused2 = 0; + VarData vdata[]; /* variable length data, see link above */ +} AliasData; + + class MiniVMac : public LaunchMethod { public: From 89d4565f0d239f8a10ab580b50c20eadba550219 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Szymon=20=C5=81opaciuk?= Date: Sun, 10 Feb 2019 01:56:37 +0000 Subject: [PATCH 2/2] Autodetect System version for correct AutoQuit MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit Reads the System resource to find out the System Software version for running correct AutoQuit. Includes other minor changes. Signed-off-by: Szymon Łopaciuk --- LaunchAPPL/Client/MiniVMac.cc | 115 +++++++++++++++++++++++++--------- LaunchAPPL/Client/MiniVMac.h | 39 ------------ 2 files changed, 85 insertions(+), 69 deletions(-) diff --git a/LaunchAPPL/Client/MiniVMac.cc b/LaunchAPPL/Client/MiniVMac.cc index 0d353e5459..ca530e9143 100644 --- a/LaunchAPPL/Client/MiniVMac.cc +++ b/LaunchAPPL/Client/MiniVMac.cc @@ -25,6 +25,45 @@ using std::vector; namespace po = boost::program_options; namespace fs = boost::filesystem; + +/* Adapted from http://sebastien.kirche.free.fr/python_stuff/MacOS-aliases.txt */ +typedef struct +{ + int16_t type; /* type of data */ + int16_t size; /* length of variable length data */ + char data[]; /* actual data */ +} VarData; + +typedef struct +{ + /* Type Code: alis */ + char userType[4] = {0, 0, 0, 0}; /* for application use, can be zeros */ + uint16_t size; /* alias record size, including variable length data */ + int16_t version = 2; /* alias version, current 2 */ + int16_t type = 0; /* file = 0, directory = 1 */ + char volumeNameSize; + char volumeName[27]; /* volume name */ + uint32_t volumeCreationDate; /* volume creation date, seconds since 1904 */ + uint16_t volumeSig = 0x4244; /*BD*/ /* volume signature MFS = RW, HFS = BD */ + int16_t volumeType = 5; /* [HD] = 0, Foreign = 1, Floppy: 400K, 800K, 1400K = 2-4, OtherEjectable = 5 */ + int32_t parentDirID; /* parent directory ID, 0 for relative searches */ + char fileNameSize; + char fileName[63]; /* file or directory name */ + int32_t fileNum; /* file number or directory ID */ + uint32_t fileCreationDate; /* file or directory creation date, seconds since 1904 */ + char typeCode[4]; /* file type */ + char creatorCode[4]; /* file's creator */ + int16_t nlvlFrom = 0; /* next level up from alias, used in relative searches */ + int16_t nlvlTo = 0; /* next level down to target, ditto */ + uint32_t volumeAttr = 0; /* various flags (locked, ejectable), see link above */ + int16_t volumeFSID = 0; /* file system ID for the volume, 0 for MFS, HFS */ + int16_t unused = 0; + uint32_t unused1 = 0; + uint32_t unused2 = 0; + VarData vdata[]; /* variable length data, see link above */ +} AliasData; + + class MiniVMacLauncher : public Launcher { fs::path imagePath; @@ -36,6 +75,7 @@ class MiniVMacLauncher : public Launcher hfsvol *vol; void CopySystemFile(const std::string& fn, bool required); + uint16_t GetSystemVersion(const std::string& systemFileName); void MakeAlias(const std::string& dest, const std::string& src); fs::path ConvertImage(const fs::path& path); public: @@ -128,13 +168,8 @@ MiniVMacLauncher::MiniVMacLauncher(po::variables_map &options) vmacDir = fs::absolute( options["minivmac-dir"].as() ); vmacPath = fs::absolute( options["minivmac-path"].as(), vmacDir ); - bool usesAutQuit7 = options.count("autquit7-image"); - systemImage = fs::absolute(options["system-image"].as(), vmacDir); - fs::path autoquitImage = fs::absolute(options[usesAutQuit7 ? "autquit7-image" : "autoquit-image"].as(), vmacDir); - systemImage = ConvertImage(systemImage); - autoquitImage = ConvertImage(autoquitImage); std::vector bootblock1(1024); fs::ifstream(systemImage).read((char*) bootblock1.data(), 1024); @@ -142,35 +177,38 @@ MiniVMacLauncher::MiniVMacLauncher(po::variables_map &options) if(bootblock1[0] != 'L' || bootblock1[1] != 'K' || bootblock1[0xA] > 15) throw std::runtime_error("Not a bootable Mac disk image: " + systemImage.string()); - string systemFileName(bootblock1.begin() + 0xB, bootblock1.begin() + 0xB + bootblock1[0xA]); - - - int size = 5000*1024; - - fs::ofstream(imagePath, std::ios::binary | std::ios::trunc).seekp(size-1).put(0); - hfs_format(imagePath.string().c_str(), 0, 0, "SysAndApp", 0, NULL); - - { - std::string finderName = std::string(usesAutQuit7 ? "Finder" : "AutoQuit"); - bootblock1[0x1A] = finderName.size(); - memcpy(&bootblock1[0x1B], finderName.c_str(), finderName.size()); - bootblock1[0x5A] = 3; - memcpy(&bootblock1[0x5B],"App", 3); - - fs::fstream(imagePath, std::ios::in | std::ios::out | std::ios::binary) - .write((const char*) bootblock1.data(), 1024); - } - - - vol = hfs_mount(imagePath.string().c_str(), 0, HFS_MODE_RDWR); - assert(vol); - sysvol = hfs_mount(systemImage.string().c_str(),0, HFS_MODE_RDONLY); assert(sysvol); hfsvolent ent; hfs_vstat(sysvol, &ent); hfs_setcwd(sysvol, ent.blessed); + string systemFileName(bootblock1.begin() + 0xB, bootblock1.begin() + 0xB + bootblock1[0xA]); + uint16_t sysver = GetSystemVersion(systemFileName); + + bool usesAutQuit7 = (sysver >= 0x700); + fs::path autoquitImage = fs::absolute(options[usesAutQuit7 ? "autquit7-image" : "autoquit-image"].as(), vmacDir); + autoquitImage = ConvertImage(autoquitImage); + + int size = 5000*1024; + + fs::ofstream(imagePath, std::ios::binary | std::ios::trunc).seekp(size-1).put(0); + hfs_format(imagePath.string().c_str(), 0, 0, "SysAndApp", 0, NULL); + + if(!usesAutQuit7) + { + std::string finderName = std::string(usesAutQuit7 ? "Finder" : "AutoQuit"); + bootblock1[0x1A] = finderName.size(); + memcpy(&bootblock1[0x1B], finderName.c_str(), finderName.size()); + bootblock1[0x5A] = 3; + memcpy(&bootblock1[0x5B],"App", 3); + } + fs::fstream(imagePath, std::ios::in | std::ios::out | std::ios::binary) + .write((const char*) bootblock1.data(), 1024); + + + vol = hfs_mount(imagePath.string().c_str(), 0, HFS_MODE_RDWR); + assert(vol); hfs_vstat(vol, &ent); ent.blessed = hfs_getcwd(vol); @@ -200,7 +238,7 @@ MiniVMacLauncher::MiniVMacLauncher(po::variables_map &options) hfs_close(file); } - hfs_umount(sysvol); sysvol = NULL; + hfs_umount(sysvol); sysvol = hfs_mount(autoquitImage.string().c_str(),0, HFS_MODE_RDONLY); if(!sysvol) throw std::runtime_error("Cannot open disk image: " + autoquitImage.string()); @@ -379,6 +417,22 @@ void MiniVMacLauncher::MakeAlias(const std::string& dest, const std::string& src } +uint16_t MiniVMacLauncher::GetSystemVersion(const std::string& systemFileName) +{ + hfsdirent fileent; + hfs_stat(sysvol, systemFileName.c_str(), &fileent); + hfsfile* system = hfs_open(sysvol, systemFileName.c_str()); + std::vector buffer(fileent.u.file.rsize); + hfs_setfork(system, 1); + hfs_read(system, buffer.data(), fileent.u.file.rsize); + hfs_close(system); + std::istringstream systemResStream(std::string((char*)buffer.data(), buffer.size())); + Resources systemRes(systemResStream); + Resource vers = systemRes.resources[ResRef('vers', 1)]; + return (uint16_t)vers.getData()[0] << 8 | vers.getData()[1]; +} + + bool MiniVMacLauncher::Go(int timeout) { fs::current_path(tempDir); @@ -423,7 +477,8 @@ bool MiniVMac::CheckOptions(variables_map &options) && options.count("minivmac-dir") != 0 && options.count("minivmac-rom") != 0 && options.count("system-image") != 0 - && (options.count("autoquit-image") + options.count("autquit7-image")) == 1; + && options.count("autoquit-image") != 0 + && options.count("autquit7-image") != 0; } std::unique_ptr MiniVMac::MakeLauncher(variables_map &options) diff --git a/LaunchAPPL/Client/MiniVMac.h b/LaunchAPPL/Client/MiniVMac.h index b98c722c7c..0f83aa4ad8 100644 --- a/LaunchAPPL/Client/MiniVMac.h +++ b/LaunchAPPL/Client/MiniVMac.h @@ -3,45 +3,6 @@ #include "LaunchMethod.h" - -/* Adapted from http://sebastien.kirche.free.fr/python_stuff/MacOS-aliases.txt */ -typedef struct -{ - int16_t type; /* type of data */ - int16_t size; /* length of variable length data */ - char data[]; /* actual data */ -} VarData; - -typedef struct -{ - /* Type Code: alis */ - char userType[4] = {0, 0, 0, 0}; /* for application use, can be zeros */ - uint16_t size; /* alias record size, including variable length data */ - int16_t version = 2; /* alias version, current 2 */ - int16_t type = 0; /* file = 0, directory = 1 */ - char volumeNameSize; - char volumeName[27]; /* volume name */ - uint32_t volumeCreationDate; /* volume creation date, seconds since 1904 */ - uint16_t volumeSig = 0x4244; /*BD*/ /* volume signature MFS = RW, HFS = BD */ - int16_t volumeType = 5; /* [HD] = 0, Foreign = 1, Floppy: 400K, 800K, 1400K = 2-4, OtherEjectable = 5 */ - int32_t parentDirID; /* parent directory ID, 0 for relative searches */ - char fileNameSize; - char fileName[63]; /* file or directory name */ - int32_t fileNum; /* file number or directory ID */ - uint32_t fileCreationDate; /* file or directory creation date, seconds since 1904 */ - char typeCode[4]; /* file type */ - char creatorCode[4]; /* file's creator */ - int16_t nlvlFrom = 0; /* next level up from alias, used in relative searches */ - int16_t nlvlTo = 0; /* next level down to target, ditto */ - uint32_t volumeAttr = 0; /* various flags (locked, ejectable), see link above */ - int16_t volumeFSID = 0; /* file system ID for the volume, 0 for MFS, HFS */ - int16_t unused = 0; - uint32_t unused1 = 0; - uint32_t unused2 = 0; - VarData vdata[]; /* variable length data, see link above */ -} AliasData; - - class MiniVMac : public LaunchMethod { public: