/* * C64 P00 file format support. * * 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 */ #include "SidTuneCfg.h" #include "SidTune.h" #include "SidTuneTools.h" #define X00_ID_LEN 8 #define X00_NAME_LEN 17 // File format from PC64. PC64 automatically generates // the filename from the cbm name (16 to 8 conversion) // but we only need to worry about that when writing files // should we want pc64 compatibility. The extension numbers // are just an index to try to avoid repeats. Name conversion // works by creating an initial filename from alphanumeric // and ' ', '-' characters only with the later two being // converted to '_'. Then it parses the filename // from end to start removing characters stopping as soon // as the filename becomes <= 8. The removal of characters // occurs in three passes, the first removes all '_', then // vowels and finally numerics. If the filename is still // greater than 8 it is trucated. struct X00Header struct X00Header { char id[X00_ID_LEN]; // C64File char name[X00_NAME_LEN]; // C64 name uint8_t length; // Rel files only (Bytes/Record), // should be 0 for all other types }; typedef enum { X00_UNKNOWN, X00_DEL, X00_SEQ, X00_PRG, X00_USR, X00_REL } X00Format; static const char _sidtune_id[] = "C64File"; static const char _sidtune_format_del[] = "Unsupported tape image file (DEL)"; static const char _sidtune_format_seq[] = "Unsupported tape image file (SEQ)"; static const char _sidtune_format_prg[] = "Tape image file (PRG)"; static const char _sidtune_format_usr[] = "Unsupported USR file (USR)"; static const char _sidtune_format_rel[] = "Unsupported tape image file (REL)"; static const char _sidtune_truncated[] = "ERROR: File is most likely truncated"; SidTune::LoadStatus SidTune::X00_fileSupport(const char *fileName, Buffer_sidtt& dataBuf) { const char *ext = SidTuneTools::fileExtOfPath(const_cast(fileName)); const char *format = 0; const X00Header *pHeader = reinterpret_cast(dataBuf.get()); uint_least32_t bufLen = dataBuf.len (); // Combined extension & magic field identification if (strlen (ext) != 4) return LOAD_NOT_MINE; if (!isdigit(ext[2]) || !isdigit(ext[3])) return LOAD_NOT_MINE; X00Format type = X00_UNKNOWN; switch (toupper(ext[1])) { case 'D': type = X00_DEL; format = _sidtune_format_del; break; case 'S': type = X00_SEQ; format = _sidtune_format_seq; break; case 'P': type = X00_PRG; format = _sidtune_format_prg; break; case 'U': type = X00_USR; format = _sidtune_format_usr; break; case 'R': type = X00_REL; format = _sidtune_format_rel; break; } if (type == X00_UNKNOWN) return LOAD_NOT_MINE; // Verify the file is what we think it is if (bufLen < X00_ID_LEN) return LOAD_NOT_MINE; else if (strcmp (pHeader->id, _sidtune_id)) return LOAD_NOT_MINE; info.formatString = format; // File types current supported if (type != X00_PRG) return LOAD_ERROR; if (bufLen < sizeof(X00Header)+2) { info.formatString = _sidtune_truncated; return LOAD_ERROR; } { // Decode file name SmartPtr_sidtt spPet((const uint8_t*)pHeader->name,X00_NAME_LEN); convertPetsciiToAscii(spPet,infoString[0]); } // Automatic settings fileOffset = sizeof(X00Header); info.songs = 1; info.startSong = 1; info.compatibility = SIDTUNE_COMPATIBILITY_BASIC; info.numberOfInfoStrings = 1; info.infoString[0] = infoString[0]; // Create the speed/clock setting table. convertOldStyleSpeedToTables(~0, info.clockSpeed); return LOAD_OK; }