diff --git a/psx/octoshock/bizhawk/octoshock.vcxproj b/psx/octoshock/bizhawk/octoshock.vcxproj
index 7c271354c3..17d4399f4d 100644
--- a/psx/octoshock/bizhawk/octoshock.vcxproj
+++ b/psx/octoshock/bizhawk/octoshock.vcxproj
@@ -12,17 +12,14 @@
-
-
-
-
-
-
+
+ All
+
@@ -52,8 +49,6 @@
-
-
diff --git a/psx/octoshock/bizhawk/octoshock.vcxproj.filters b/psx/octoshock/bizhawk/octoshock.vcxproj.filters
index 9f00eb2a33..72d05a57b3 100644
--- a/psx/octoshock/bizhawk/octoshock.vcxproj.filters
+++ b/psx/octoshock/bizhawk/octoshock.vcxproj.filters
@@ -59,18 +59,6 @@
psx
-
- cdrom
-
-
- cdrom
-
-
- cdrom
-
-
- cdrom
-
psx\input
@@ -108,12 +96,6 @@
video
-
- cdrom
-
-
- cdrom
-
emuware
@@ -129,6 +111,9 @@
psx
+
+ cdrom
+
@@ -176,15 +161,6 @@
psx
-
- cdrom
-
-
- cdrom
-
-
- cdrom
-
psx\input
@@ -236,6 +212,9 @@
psx
+
+ cdrom
+
diff --git a/psx/octoshock/cdrom/CDAccess.cpp b/psx/octoshock/cdrom/CDAccess.cpp
deleted file mode 100644
index 9d932a2491..0000000000
--- a/psx/octoshock/cdrom/CDAccess.cpp
+++ /dev/null
@@ -1,58 +0,0 @@
-/* Mednafen - Multi-system Emulator
- *
- * 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 "../mednafen.h"
-#include "CDAccess.h"
-#include "CDAccess_Image.h"
-#include "CDAccess_CCD.h"
-
-#ifdef HAVE_LIBCDIO
-#include "CDAccess_Physical.h"
-#endif
-
-using namespace CDUtility;
-
-CDAccess::CDAccess()
-{
-
-}
-
-CDAccess::~CDAccess()
-{
-
-}
-
-CDAccess *cdaccess_open_image(const char *path, bool image_memcache)
-{
- CDAccess *ret = NULL;
-
- if(strlen(path) >= 4 && !strcasecmp(path + strlen(path) - 4, ".ccd"))
- ret = new CDAccess_CCD(path, image_memcache);
- else
- ret = new CDAccess_Image(path, image_memcache);
-
- return ret;
-}
-
-CDAccess *cdaccess_open_phys(const char *devicename)
-{
- #ifdef HAVE_LIBCDIO
- return new CDAccess_Physical(devicename);
- #else
- throw MDFN_Error(0, _("Physical CD access support not compiled in."));
- #endif
-}
diff --git a/psx/octoshock/cdrom/CDAccess.h b/psx/octoshock/cdrom/CDAccess.h
deleted file mode 100644
index 5fbdda19b8..0000000000
--- a/psx/octoshock/cdrom/CDAccess.h
+++ /dev/null
@@ -1,31 +0,0 @@
-#ifndef __MDFN_CDROMFILE_H
-#define __MDFN_CDROMFILE_H
-
-#include
-
-#include "CDUtility.h"
-
-class CDAccess
-{
- public:
-
- CDAccess();
- virtual ~CDAccess();
-
- virtual void Read_Raw_Sector(uint8 *buf, int32 lba) = 0;
-
- virtual void Read_TOC(CDUtility::TOC *toc) = 0;
-
- virtual bool Is_Physical(void) throw() = 0;
-
- virtual void Eject(bool eject_status) = 0; // Eject a disc if it's physical, otherwise NOP. Returns true on success(or NOP), false on error
-
- private:
- CDAccess(const CDAccess&); // No copy constructor.
- CDAccess& operator=(const CDAccess&); // No assignment operator.
-};
-
-CDAccess *cdaccess_open_image(const char *path, bool image_memcache);
-CDAccess *cdaccess_open_phys(const char *devicename);
-
-#endif
diff --git a/psx/octoshock/cdrom/CDUtility.cpp b/psx/octoshock/cdrom/CDUtility.cpp
index 4a403f542a..7b8de3f0d7 100644
--- a/psx/octoshock/cdrom/CDUtility.cpp
+++ b/psx/octoshock/cdrom/CDUtility.cpp
@@ -22,8 +22,6 @@
#include "octoshock.h"
#include "CDUtility.h"
-#include "dvdisaster.h"
-#include "lec.h"
// Kill_LEC_Correct();
@@ -31,6 +29,10 @@
namespace CDUtility
{
+ void CDUtility_Init()
+ {
+ }
+
// lookup table for crc calculation
static uint16 subq_crctab[256] =
{
@@ -65,260 +67,71 @@ static uint16 subq_crctab[256] =
0x2E93, 0x3EB2, 0x0ED1, 0x1EF0
};
-
-static uint8 scramble_table[2352 - 12];
-
-static bool CDUtility_Inited = false;
-
-static void InitScrambleTable(void)
-{
- unsigned cv = 1;
-
- for(unsigned i = 12; i < 2352; i++)
- {
- unsigned char z = 0;
-
- for(int b = 0; b < 8; b++)
- {
- z |= (cv & 1) << b;
-
- int feedback = ((cv >> 1) & 1) ^ (cv & 1);
- cv = (cv >> 1) | (feedback << 14);
- }
-
- scramble_table[i - 12] = z;
- }
-
- //for(int i = 0; i < 2352 - 12; i++)
- // printf("0x%02x, ", scramble_table[i]);
-}
-
-void CDUtility_Init(void)
-{
- if(!CDUtility_Inited)
- {
- Init_LEC_Correct();
-
- InitScrambleTable();
-
- CDUtility_Inited = true;
- }
-}
-
-void encode_mode0_sector(uint32 aba, uint8 *sector_data)
-{
- CDUtility_Init();
-
- lec_encode_mode0_sector(aba, sector_data);
-}
-
-void encode_mode1_sector(uint32 aba, uint8 *sector_data)
-{
- CDUtility_Init();
-
- lec_encode_mode1_sector(aba, sector_data);
-}
-
-void encode_mode2_sector(uint32 aba, uint8 *sector_data)
-{
- CDUtility_Init();
-
- lec_encode_mode2_sector(aba, sector_data);
-}
-
-void encode_mode2_form1_sector(uint32 aba, uint8 *sector_data)
-{
- CDUtility_Init();
-
- lec_encode_mode2_form1_sector(aba, sector_data);
-}
-
-void encode_mode2_form2_sector(uint32 aba, uint8 *sector_data)
-{
- CDUtility_Init();
-
- lec_encode_mode2_form2_sector(aba, sector_data);
-}
-
-bool edc_check(const uint8 *sector_data, bool xa)
-{
- CDUtility_Init();
-
- return(CheckEDC(sector_data, xa));
-}
-
-bool edc_lec_check_and_correct(uint8 *sector_data, bool xa)
-{
- CDUtility_Init();
-
- return(ValidateRawSector(sector_data, xa));
-}
-
-
-bool subq_check_checksum(const uint8 *SubQBuf)
-{
- uint16 crc = 0;
- uint16 stored_crc = 0;
-
- stored_crc = SubQBuf[0xA] << 8;
- stored_crc |= SubQBuf[0xB];
-
- for(int i = 0; i < 0xA; i++)
- crc = subq_crctab[(crc >> 8) ^ SubQBuf[i]] ^ (crc << 8);
-
- crc = ~crc;
-
- return(crc == stored_crc);
-}
-
-void subq_generate_checksum(uint8 *buf)
-{
- uint16 crc = 0;
-
- for(int i = 0; i < 0xA; i++)
- crc = subq_crctab[(crc >> 8) ^ buf[i]] ^ (crc << 8);
-
- // Checksum
- buf[0xa] = ~(crc >> 8);
- buf[0xb] = ~(crc);
-}
-
-void subq_deinterleave(const uint8 *SubPWBuf, uint8 *qbuf)
-{
- memset(qbuf, 0, 0xC);
-
- for(int i = 0; i < 96; i++)
- {
- qbuf[i >> 3] |= ((SubPWBuf[i] >> 6) & 0x1) << (7 - (i & 0x7));
- }
-}
-
-
-// Deinterleaves 96 bytes of subchannel P-W data from 96 bytes of interleaved subchannel PW data.
-void subpw_deinterleave(const uint8 *in_buf, uint8 *out_buf)
-{
- assert(in_buf != out_buf);
-
- memset(out_buf, 0, 96);
-
- for(unsigned ch = 0; ch < 8; ch++)
- {
- for(unsigned i = 0; i < 96; i++)
- {
- out_buf[(ch * 12) + (i >> 3)] |= ((in_buf[i] >> (7 - ch)) & 0x1) << (7 - (i & 0x7));
- }
- }
-
-}
-
-// Interleaves 96 bytes of subchannel P-W data from 96 bytes of uninterleaved subchannel PW data.
-void subpw_interleave(const uint8 *in_buf, uint8 *out_buf)
-{
- assert(in_buf != out_buf);
-
- for(unsigned d = 0; d < 12; d++)
- {
- for(unsigned bitpoodle = 0; bitpoodle < 8; bitpoodle++)
- {
- uint8 rawb = 0;
-
- for(unsigned ch = 0; ch < 8; ch++)
- {
- rawb |= ((in_buf[ch * 12 + d] >> (7 - bitpoodle)) & 1) << (7 - ch);
- }
- out_buf[(d << 3) + bitpoodle] = rawb;
- }
- }
-}
-
-// NOTES ON LEADOUT AREA SYNTHESIS
-//
-// I'm not trusting that the "control" field for the TOC leadout entry will always be set properly, so | the control fields for the last track entry
-// and the leadout entry together before extracting the D2 bit. Audio track->data leadout is fairly benign though maybe noisy(especially if we ever implement
-// data scrambling properly), but data track->audio leadout could break things in an insidious manner for the more accurate drive emulation code).
-//
-void subpw_synth_leadout_lba(const TOC& toc, const int32 lba, uint8* SubPWBuf)
-{
- uint8 buf[0xC];
- uint32 lba_relative;
- uint32 ma, sa, fa;
- uint32 m, s, f;
-
- lba_relative = lba - toc.tracks[100].lba;
-
- f = (lba_relative % 75);
- s = ((lba_relative / 75) % 60);
- m = (lba_relative / 75 / 60);
-
- fa = (lba + 150) % 75;
- sa = ((lba + 150) / 75) % 60;
- ma = ((lba + 150) / 75 / 60);
-
- uint8 adr = 0x1; // Q channel data encodes position
- uint8 control = (toc.tracks[toc.last_track].control & 0x4) | toc.tracks[100].control;
-
- memset(buf, 0, 0xC);
- buf[0] = (adr << 0) | (control << 4);
- buf[1] = 0xAA;
- buf[2] = 0x01;
-
- // Track relative MSF address
- buf[3] = U8_to_BCD(m);
- buf[4] = U8_to_BCD(s);
- buf[5] = U8_to_BCD(f);
-
- buf[6] = 0; // Zerroooo
-
- // Absolute MSF address
- buf[7] = U8_to_BCD(ma);
- buf[8] = U8_to_BCD(sa);
- buf[9] = U8_to_BCD(fa);
-
- subq_generate_checksum(buf);
-
- for(int i = 0; i < 96; i++)
- SubPWBuf[i] = (((buf[i >> 3] >> (7 - (i & 0x7))) & 1) ? 0x40 : 0x00) | 0x80;
-}
-
-void synth_leadout_sector_lba(const uint8 mode, const TOC& toc, const int32 lba, uint8* out_buf)
-{
- memset(out_buf, 0, 2352 + 96);
- subpw_synth_leadout_lba(toc, lba, out_buf + 2352);
-
- if((toc.tracks[toc.last_track].control | toc.tracks[100].control) & 0x4)
- {
- switch(mode)
- {
- default:
- encode_mode0_sector(LBA_to_ABA(lba), out_buf);
- break;
-
- case 0x01:
- encode_mode1_sector(LBA_to_ABA(lba), out_buf);
- break;
-
- case 0x02:
- out_buf[18] = 0x20;
- encode_mode2_form2_sector(LBA_to_ABA(lba), out_buf);
- break;
- }
- }
-}
-
-#if 0
-bool subq_extrapolate(const uint8 *subq_input, int32 position_delta, uint8 *subq_output)
-{
- assert(subq_check_checksum(subq_input));
-
-
- subq_generate_checksum(subq_output);
-}
-#endif
-
-void scrambleize_data_sector(uint8 *sector_data)
-{
- for(unsigned i = 12; i < 2352; i++)
- sector_data[i] ^= scramble_table[i - 12];
-}
-
-}
+
+bool subq_check_checksum(const uint8 *SubQBuf)
+{
+ uint16 crc = 0;
+ uint16 stored_crc = 0;
+
+ stored_crc = SubQBuf[0xA] << 8;
+ stored_crc |= SubQBuf[0xB];
+
+ for(int i = 0; i < 0xA; i++)
+ crc = subq_crctab[(crc >> 8) ^ SubQBuf[i]] ^ (crc << 8);
+
+ crc = ~crc;
+
+ return(crc == stored_crc);
+}
+
+void subq_generate_checksum(uint8 *buf)
+{
+ uint16 crc = 0;
+
+ for(int i = 0; i < 0xA; i++)
+ crc = subq_crctab[(crc >> 8) ^ buf[i]] ^ (crc << 8);
+
+ // Checksum
+ buf[0xa] = ~(crc >> 8);
+ buf[0xb] = ~(crc);
+}
+
+
+// Deinterleaves 96 bytes of subchannel P-W data from 96 bytes of interleaved subchannel PW data.
+void subpw_deinterleave(const uint8 *in_buf, uint8 *out_buf)
+{
+ assert(in_buf != out_buf);
+
+ memset(out_buf, 0, 96);
+
+ for(unsigned ch = 0; ch < 8; ch++)
+ {
+ for(unsigned i = 0; i < 96; i++)
+ {
+ out_buf[(ch * 12) + (i >> 3)] |= ((in_buf[i] >> (7 - ch)) & 0x1) << (7 - (i & 0x7));
+ }
+ }
+
+}
+
+// Interleaves 96 bytes of subchannel P-W data from 96 bytes of uninterleaved subchannel PW data.
+void subpw_interleave(const uint8 *in_buf, uint8 *out_buf)
+{
+ assert(in_buf != out_buf);
+
+ for(unsigned d = 0; d < 12; d++)
+ {
+ for(unsigned bitpoodle = 0; bitpoodle < 8; bitpoodle++)
+ {
+ uint8 rawb = 0;
+
+ for(unsigned ch = 0; ch < 8; ch++)
+ {
+ rawb |= ((in_buf[ch * 12 + d] >> (7 - bitpoodle)) & 1) << (7 - ch);
+ }
+ out_buf[(d << 3) + bitpoodle] = rawb;
+ }
+ }
+}
+
+}
\ No newline at end of file
diff --git a/psx/octoshock/cdrom/CDUtility.h b/psx/octoshock/cdrom/CDUtility.h
index 896d145459..2cdbb756b6 100644
--- a/psx/octoshock/cdrom/CDUtility.h
+++ b/psx/octoshock/cdrom/CDUtility.h
@@ -152,52 +152,6 @@ namespace CDUtility
return( ((num / 10) << 4) + (num % 10) );
}
- // should always perform the conversion, even if the bcd number is invalid.
- static INLINE bool BCD_to_U8_check(uint8 bcd_number, uint8 *out_number)
- {
- *out_number = BCD_to_U8(bcd_number);
-
- if(!BCD_is_valid(bcd_number))
- return(false);
-
- return(true);
- }
-
- //
- // Sector data encoding functions(to full 2352 bytes raw sector).
- //
- // sector_data must be able to contain at least 2352 bytes.
- void encode_mode0_sector(uint32 aba, uint8 *sector_data);
- void encode_mode1_sector(uint32 aba, uint8 *sector_data); // 2048 bytes of user data at offset 16
- void encode_mode2_sector(uint32 aba, uint8 *sector_data); // 2336 bytes of user data at offset 16
- void encode_mode2_form1_sector(uint32 aba, uint8 *sector_data); // 2048+8 bytes of user data at offset 16
- void encode_mode2_form2_sector(uint32 aba, uint8 *sector_data); // 2324+8 bytes of user data at offset 16
-
-
- // out_buf must be able to contain 2352+96 bytes.
- // "mode" is only used if(toc.tracks[100].control & 0x4)
- void synth_leadout_sector_lba(const uint8 mode, const TOC& toc, const int32 lba, uint8* out_buf);
-
- //
- // User data error detection and correction
- //
-
- // Check EDC of a mode 1 or mode 2 form 1 sector.
- // Returns "true" if checksum is ok(matches).
- // Returns "false" if checksum mismatch.
- // sector_data should contain 2352 bytes of raw sector data.
- bool edc_check(const uint8 *sector_data, bool xa);
-
- // Check EDC and L-EC data of a mode 1 or mode 2 form 1 sector, and correct bit errors if any exist.
- // Returns "true" if errors weren't detected, or they were corrected succesfully.
- // Returns "false" if errors couldn't be corrected.
- // sector_data should contain 2352 bytes of raw sector data.
- bool edc_lec_check_and_correct(uint8 *sector_data, bool xa);
-
- //
- // Subchannel(Q in particular) functions
- //
-
// Returns false on checksum mismatch, true on match.
bool subq_check_checksum(const uint8 *subq_buf);
@@ -205,23 +159,12 @@ namespace CDUtility
// in subq_buf.
void subq_generate_checksum(uint8 *subq_buf);
- // Deinterleaves 12 bytes of subchannel Q data from 96 bytes of interleaved subchannel PW data.
- void subq_deinterleave(const uint8 *subpw_buf, uint8 *subq_buf);
-
// Deinterleaves 96 bytes of subchannel P-W data from 96 bytes of interleaved subchannel PW data.
void subpw_deinterleave(const uint8 *in_buf, uint8 *out_buf);
// Interleaves 96 bytes of subchannel P-W data from 96 bytes of uninterleaved subchannel PW data.
void subpw_interleave(const uint8 *in_buf, uint8 *out_buf);
- // Extrapolates Q subchannel current position data from subq_input, with frame/sector delta position_delta, and writes to subq_output.
- // Only valid for ADR_CURPOS.
- // subq_input must pass subq_check_checksum().
- // TODO
- //void subq_extrapolate(const uint8 *subq_input, int32 position_delta, uint8 *subq_output);
-
- // (De)Scrambles data sector.
- void scrambleize_data_sector(uint8 *sector_data);
}
#endif
diff --git a/psx/octoshock/cdrom/SimpleFIFO.cpp b/psx/octoshock/cdrom/SimpleFIFO.cpp
deleted file mode 100644
index f0a7f5fa14..0000000000
--- a/psx/octoshock/cdrom/SimpleFIFO.cpp
+++ /dev/null
@@ -1,3 +0,0 @@
-#include "../mednafen.h"
-#include "SimpleFIFO.h"
-
diff --git a/psx/octoshock/cdrom/cdromif.cpp b/psx/octoshock/cdrom/cdromif.cpp
deleted file mode 100644
index 7fddb300ca..0000000000
--- a/psx/octoshock/cdrom/cdromif.cpp
+++ /dev/null
@@ -1,915 +0,0 @@
-/* Mednafen - Multi-system Emulator
- *
- * 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 "../mednafen.h"
-#include
-#include
-#include
-#include "cdromif.h"
-#include "CDAccess.h"
-#include "../general.h"
-
-#include
-
-using namespace CDUtility;
-
-enum
-{
- // Status/Error messages
- CDIF_MSG_DONE = 0, // Read -> emu. args: No args.
- CDIF_MSG_INFO, // Read -> emu. args: str_message
- CDIF_MSG_FATAL_ERROR, // Read -> emu. args: *TODO ARGS*
-
- //
- // Command messages.
- //
- CDIF_MSG_DIEDIEDIE, // Emu -> read
-
- CDIF_MSG_READ_SECTOR, /* Emu -> read
- args[0] = lba
- */
-
- CDIF_MSG_EJECT, // Emu -> read, args[0]; 0=insert, 1=eject
-};
-
-class CDIF_Message
-{
- public:
-
- CDIF_Message();
- CDIF_Message(unsigned int message_, uint32 arg0 = 0, uint32 arg1 = 0, uint32 arg2 = 0, uint32 arg3 = 0);
- CDIF_Message(unsigned int message_, const std::string &str);
- ~CDIF_Message();
-
- unsigned int message;
- uint32 args[4];
- void *parg;
- std::string str_message;
-};
-
-class CDIF_Queue
-{
- public:
-
- CDIF_Queue();
- ~CDIF_Queue();
-
- bool Read(CDIF_Message *message, bool blocking = TRUE);
-
- void Write(const CDIF_Message &message);
-
- private:
- std::queue ze_queue;
- MDFN_Mutex *ze_mutex;
- MDFN_Cond *ze_cond;
-};
-
-
-typedef struct
-{
- bool valid;
- bool error;
- uint32 lba;
- uint8 data[2352 + 96];
-} CDIF_Sector_Buffer;
-
-// TODO: prohibit copy constructor
-class CDIF_MT : public CDIF
-{
- public:
-
- CDIF_MT(CDAccess *cda);
- virtual ~CDIF_MT();
-
- virtual void HintReadSector(uint32 lba);
- virtual bool ReadRawSector(uint8 *buf, uint32 lba);
-
- // Return true if operation succeeded or it was a NOP(either due to not being implemented, or the current status matches eject_status).
- // Returns false on failure(usually drive error of some kind; not completely fatal, can try again).
- virtual bool Eject(bool eject_status);
-
- // FIXME: Semi-private:
- int ReadThreadStart(void);
-
- private:
-
- CDAccess *disc_cdaccess;
-
- MDFN_Thread *CDReadThread;
-
- // Queue for messages to the read thread.
- CDIF_Queue ReadThreadQueue;
-
- // Queue for messages to the emu thread.
- CDIF_Queue EmuThreadQueue;
-
-
- enum { SBSize = 256 };
- CDIF_Sector_Buffer SectorBuffers[SBSize];
-
- uint32 SBWritePos;
-
- MDFN_Mutex *SBMutex;
- MDFN_Cond *SBCond;
-
-
- //
- // Read-thread-only:
- //
- void RT_EjectDisc(bool eject_status, bool skip_actual_eject = false);
-
- uint32 ra_lba;
- int ra_count;
- uint32 last_read_lba;
-};
-
-
-// TODO: prohibit copy constructor
-class CDIF_ST : public CDIF
-{
- public:
-
- CDIF_ST(CDAccess *cda);
- virtual ~CDIF_ST();
-
- virtual void HintReadSector(uint32 lba);
- virtual bool ReadRawSector(uint8 *buf, uint32 lba);
- virtual bool Eject(bool eject_status);
-
- private:
- CDAccess *disc_cdaccess;
-};
-
-CDIF::CDIF() : UnrecoverableError(false), is_phys_cache(false), DiscEjected(false)
-{
-
-}
-
-CDIF::~CDIF()
-{
-
-}
-
-
-CDIF_Message::CDIF_Message()
-{
- message = 0;
-
- memset(args, 0, sizeof(args));
-}
-
-CDIF_Message::CDIF_Message(unsigned int message_, uint32 arg0, uint32 arg1, uint32 arg2, uint32 arg3)
-{
- message = message_;
- args[0] = arg0;
- args[1] = arg1;
- args[2] = arg2;
- args[3] = arg3;
-}
-
-CDIF_Message::CDIF_Message(unsigned int message_, const std::string &str)
-{
- message = message_;
- str_message = str;
-}
-
-CDIF_Message::~CDIF_Message()
-{
-
-}
-
-CDIF_Queue::CDIF_Queue()
-{
- ze_mutex = MDFND_CreateMutex();
- ze_cond = MDFND_CreateCond();
-}
-
-CDIF_Queue::~CDIF_Queue()
-{
- MDFND_DestroyMutex(ze_mutex);
- MDFND_DestroyCond(ze_cond);
-}
-
-// Returns FALSE if message not read, TRUE if it was read. Will always return TRUE if "blocking" is set.
-// Will throw MDFN_Error if the read message code is CDIF_MSG_FATAL_ERROR
-bool CDIF_Queue::Read(CDIF_Message *message, bool blocking)
-{
- bool ret = true;
-
- //
- //
- //
- MDFND_LockMutex(ze_mutex);
-
- if(blocking)
- {
- while(ze_queue.size() == 0) // while, not just if.
- {
- MDFND_WaitCond(ze_cond, ze_mutex);
- }
- }
-
- if(ze_queue.size() == 0)
- ret = false;
- else
- {
- *message = ze_queue.front();
- ze_queue.pop();
- }
-
- MDFND_UnlockMutex(ze_mutex);
- //
- //
- //
-
- if(ret && message->message == CDIF_MSG_FATAL_ERROR)
- throw MDFN_Error(0, "%s", message->str_message.c_str());
-
- return(ret);
-}
-
-void CDIF_Queue::Write(const CDIF_Message &message)
-{
- MDFND_LockMutex(ze_mutex);
-
- try
- {
- ze_queue.push(message);
- }
- catch(...)
- {
- fprintf(stderr, "\n\nCDIF_Message queue push failed!!! (We now return you to your regularly unscheduled lockup)\n\n");
- }
-
- MDFND_SignalCond(ze_cond); // Signal while the mutex is held to prevent icky race conditions.
-
- MDFND_UnlockMutex(ze_mutex);
-}
-
-
-void CDIF_MT::RT_EjectDisc(bool eject_status, bool skip_actual_eject)
-{
- if(eject_status != DiscEjected)
- {
- if(!skip_actual_eject)
- disc_cdaccess->Eject(eject_status);
-
- // Set after ->Eject(), since it might throw an exception.
- DiscEjected = -1; // For if TOC reading fails or there's something horribly wrong with the disc.
-
- if(!eject_status) // Re-read the TOC
- {
- disc_cdaccess->Read_TOC(&disc_toc);
-
- if(disc_toc.first_track < 1 || disc_toc.last_track > 99 || disc_toc.first_track > disc_toc.last_track)
- {
- throw(MDFN_Error(0, _("TOC first(%d)/last(%d) track numbers bad."), disc_toc.first_track, disc_toc.last_track));
- }
- }
- DiscEjected = eject_status;
-
- SBWritePos = 0;
- ra_lba = 0;
- ra_count = 0;
- last_read_lba = ~0U;
- memset(SectorBuffers, 0, SBSize * sizeof(CDIF_Sector_Buffer));
- }
-}
-
-struct RTS_Args
-{
- CDIF_MT *cdif_ptr;
-};
-
-static int ReadThreadStart_C(void *v_arg)
-{
- RTS_Args *args = (RTS_Args *)v_arg;
-
- return args->cdif_ptr->ReadThreadStart();
-}
-
-int CDIF_MT::ReadThreadStart()
-{
- bool Running = TRUE;
-
- DiscEjected = true;
- SBWritePos = 0;
- ra_lba = 0;
- ra_count = 0;
- last_read_lba = ~0U;
-
- try
- {
- RT_EjectDisc(false, true);
- }
- catch(std::exception &e)
- {
- EmuThreadQueue.Write(CDIF_Message(CDIF_MSG_FATAL_ERROR, std::string(e.what())));
- return(0);
- }
-
- is_phys_cache = disc_cdaccess->Is_Physical();
-
- EmuThreadQueue.Write(CDIF_Message(CDIF_MSG_DONE));
-
- while(Running)
- {
- CDIF_Message msg;
-
- // Only do a blocking-wait for a message if we don't have any sectors to read-ahead.
- // MDFN_DispMessage("%d %d %d\n", last_read_lba, ra_lba, ra_count);
- if(ReadThreadQueue.Read(&msg, ra_count ? FALSE : TRUE))
- {
- switch(msg.message)
- {
- case CDIF_MSG_DIEDIEDIE:
- Running = FALSE;
- break;
-
- case CDIF_MSG_EJECT:
- try
- {
- RT_EjectDisc(msg.args[0]);
- EmuThreadQueue.Write(CDIF_Message(CDIF_MSG_DONE));
- }
- catch(std::exception &e)
- {
- EmuThreadQueue.Write(CDIF_Message(CDIF_MSG_FATAL_ERROR, std::string(e.what())));
- }
- break;
-
- case CDIF_MSG_READ_SECTOR:
- {
- static const int max_ra = 16;
- static const int initial_ra = 1;
- static const int speedmult_ra = 2;
- uint32 new_lba = msg.args[0];
-
- assert((unsigned int)max_ra < (SBSize / 4));
-
- if(last_read_lba != ~0U && new_lba == (last_read_lba + 1))
- {
- int how_far_ahead = ra_lba - new_lba;
-
- if(how_far_ahead <= max_ra)
- ra_count = std::min(speedmult_ra, 1 + max_ra - how_far_ahead);
- else
- ra_count++;
- }
- else if(new_lba != last_read_lba)
- {
- ra_lba = new_lba;
- ra_count = initial_ra;
- }
-
- last_read_lba = new_lba;
- }
- break;
- }
- }
-
- // Don't read >= the "end" of the disc, silly snake. Slither.
- if(ra_count && ra_lba == disc_toc.tracks[100].lba)
- {
- ra_count = 0;
- //printf("Ephemeral scarabs: %d!\n", ra_lba);
- }
-
- if(ra_count)
- {
- uint8 tmpbuf[2352 + 96];
- bool error_condition = false;
-
- try
- {
- disc_cdaccess->Read_Raw_Sector(tmpbuf, ra_lba);
- }
- catch(std::exception &e)
- {
- MDFN_PrintError(_("Sector %u read error: %s"), ra_lba, e.what());
- memset(tmpbuf, 0, sizeof(tmpbuf));
- error_condition = true;
- }
-
- //
- //
- MDFND_LockMutex(SBMutex);
-
- SectorBuffers[SBWritePos].lba = ra_lba;
- memcpy(SectorBuffers[SBWritePos].data, tmpbuf, 2352 + 96);
- SectorBuffers[SBWritePos].valid = TRUE;
- SectorBuffers[SBWritePos].error = error_condition;
- SBWritePos = (SBWritePos + 1) % SBSize;
-
- MDFND_SignalCond(SBCond);
-
- MDFND_UnlockMutex(SBMutex);
- //
- //
-
- ra_lba++;
- ra_count--;
- }
- }
-
- return(1);
-}
-
-CDIF_MT::CDIF_MT(CDAccess *cda) : disc_cdaccess(cda), CDReadThread(NULL), SBMutex(NULL), SBCond(NULL)
-{
- try
- {
- CDIF_Message msg;
- RTS_Args s;
-
- if(!(SBMutex = MDFND_CreateMutex()))
- throw MDFN_Error(0, _("Error creating CD read thread mutex."));
-
- if(!(SBCond = MDFND_CreateCond()))
- throw MDFN_Error(0, _("Error creating CD read thread condition variable."));
-
- UnrecoverableError = false;
-
- s.cdif_ptr = this;
-
- if(!(CDReadThread = MDFND_CreateThread(ReadThreadStart_C, &s)))
- throw MDFN_Error(0, _("Error creating CD read thread."));
-
- EmuThreadQueue.Read(&msg);
- }
- catch(...)
- {
- if(CDReadThread)
- {
- MDFND_WaitThread(CDReadThread, NULL);
- CDReadThread = NULL;
- }
-
- if(SBMutex)
- {
- MDFND_DestroyMutex(SBMutex);
- SBMutex = NULL;
- }
-
- if(SBCond)
- {
- MDFND_DestroyCond(SBCond);
- SBCond = NULL;
- }
-
- if(disc_cdaccess)
- {
- delete disc_cdaccess;
- disc_cdaccess = NULL;
- }
-
- throw;
- }
-}
-
-
-CDIF_MT::~CDIF_MT()
-{
- bool thread_deaded_failed = false;
-
- try
- {
- ReadThreadQueue.Write(CDIF_Message(CDIF_MSG_DIEDIEDIE));
- }
- catch(std::exception &e)
- {
- MDFND_PrintError(e.what());
- thread_deaded_failed = true;
- }
-
- if(!thread_deaded_failed)
- MDFND_WaitThread(CDReadThread, NULL);
-
- if(SBMutex)
- {
- MDFND_DestroyMutex(SBMutex);
- SBMutex = NULL;
- }
-
- if(SBCond)
- {
- MDFND_DestroyCond(SBCond);
- SBCond = NULL;
- }
-
- if(disc_cdaccess)
- {
- delete disc_cdaccess;
- disc_cdaccess = NULL;
- }
-}
-
-bool CDIF::ValidateRawSector(uint8 *buf)
-{
- int mode = buf[12 + 3];
-
- if(mode != 0x1 && mode != 0x2)
- return(false);
-
- if(!edc_lec_check_and_correct(buf, mode == 2))
- return(false);
-
- return(true);
-}
-
-bool CDIF_MT::ReadRawSector(uint8 *buf, uint32 lba)
-{
- bool found = FALSE;
- bool error_condition = false;
-
- if(UnrecoverableError)
- {
- memset(buf, 0, 2352 + 96);
- return(false);
- }
-
- // This shouldn't happen, the emulated-system-specific CDROM emulation code should make sure the emulated program doesn't try
- // to read past the last "real" sector of the disc.
- if(lba >= disc_toc.tracks[100].lba)
- {
- printf("Attempt to read LBA %d, >= LBA %d\n", lba, disc_toc.tracks[100].lba);
- return(FALSE);
- }
-
- ReadThreadQueue.Write(CDIF_Message(CDIF_MSG_READ_SECTOR, lba));
-
- //
- //
- //
- MDFND_LockMutex(SBMutex);
-
- do
- {
- for(int i = 0; i < SBSize; i++)
- {
- if(SectorBuffers[i].valid && SectorBuffers[i].lba == lba)
- {
- error_condition = SectorBuffers[i].error;
- memcpy(buf, SectorBuffers[i].data, 2352 + 96);
- found = TRUE;
- }
- }
-
- if(!found)
- {
- //int32 swt = MDFND_GetTime();
- MDFND_WaitCond(SBCond, SBMutex);
- //printf("SB Waited: %d\n", MDFND_GetTime() - swt);
- }
- } while(!found);
-
- MDFND_UnlockMutex(SBMutex);
- //
- //
- //
-
-
- return(!error_condition);
-}
-
-void CDIF_MT::HintReadSector(uint32 lba)
-{
- if(UnrecoverableError)
- return;
-
- ReadThreadQueue.Write(CDIF_Message(CDIF_MSG_READ_SECTOR, lba));
-}
-
-int CDIF::ReadSector(uint8* pBuf, uint32 lba, uint32 nSectors)
-{
- int ret = 0;
-
- if(UnrecoverableError)
- return(false);
-
- while(nSectors--)
- {
- uint8 tmpbuf[2352 + 96];
-
- if(!ReadRawSector(tmpbuf, lba))
- {
- puts("CDIF Raw Read error");
- return(FALSE);
- }
-
- if(!ValidateRawSector(tmpbuf))
- {
- MDFN_DispMessage(_("Uncorrectable data at sector %d"), lba);
- MDFN_PrintError(_("Uncorrectable data at sector %d"), lba);
- return(false);
- }
-
- const int mode = tmpbuf[12 + 3];
-
- if(!ret)
- ret = mode;
-
- if(mode == 1)
- {
- memcpy(pBuf, &tmpbuf[12 + 4], 2048);
- }
- else if(mode == 2)
- {
- memcpy(pBuf, &tmpbuf[12 + 4 + 8], 2048);
- }
- else
- {
- printf("CDIF_ReadSector() invalid sector type at LBA=%u\n", (unsigned int)lba);
- return(false);
- }
-
- pBuf += 2048;
- lba++;
- }
-
- return(ret);
-}
-
-bool CDIF_MT::Eject(bool eject_status)
-{
- if(UnrecoverableError)
- return(false);
-
- try
- {
- CDIF_Message msg;
-
- ReadThreadQueue.Write(CDIF_Message(CDIF_MSG_EJECT, eject_status));
- EmuThreadQueue.Read(&msg);
- }
- catch(std::exception &e)
- {
- MDFN_PrintError(_("Error on eject/insert attempt: %s"), e.what());
- return(false);
- }
-
- return(true);
-}
-
-//
-//
-// Single-threaded implementation follows.
-//
-//
-
-CDIF_ST::CDIF_ST(CDAccess *cda) : disc_cdaccess(cda)
-{
- //puts("***WARNING USING SINGLE-THREADED CD READER***");
-
- is_phys_cache = disc_cdaccess->Is_Physical();
- UnrecoverableError = false;
- DiscEjected = false;
-
- disc_cdaccess->Read_TOC(&disc_toc);
-
- if(disc_toc.first_track < 1 || disc_toc.last_track > 99 || disc_toc.first_track > disc_toc.last_track)
- {
- throw(MDFN_Error(0, _("TOC first(%d)/last(%d) track numbers bad."), disc_toc.first_track, disc_toc.last_track));
- }
-}
-
-CDIF_ST::~CDIF_ST()
-{
- if(disc_cdaccess)
- {
- delete disc_cdaccess;
- disc_cdaccess = NULL;
- }
-}
-
-void CDIF_ST::HintReadSector(uint32 lba)
-{
- // TODO: disc_cdaccess seek hint? (probably not, would require asynchronousitycamel)
-}
-
-bool CDIF_ST::ReadRawSector(uint8 *buf, uint32 lba)
-{
- if(UnrecoverableError)
- {
- memset(buf, 0, 2352 + 96);
- return(false);
- }
-
- try
- {
- disc_cdaccess->Read_Raw_Sector(buf, lba);
- }
- catch(std::exception &e)
- {
- MDFN_PrintError(_("Sector %u read error: %s"), lba, e.what());
- memset(buf, 0, 2352 + 96);
- return(false);
- }
-
- return(true);
-}
-
-bool CDIF_ST::Eject(bool eject_status)
-{
- if(UnrecoverableError)
- return(false);
-
- try
- {
- if(eject_status != DiscEjected)
- {
- disc_cdaccess->Eject(eject_status);
-
- // Set after ->Eject(), since it might throw an exception.
- DiscEjected = -1; // For if TOC reading fails or there's something horribly wrong with the disc.
-
- if(!eject_status) // Re-read the TOC
- {
- disc_cdaccess->Read_TOC(&disc_toc);
-
- if(disc_toc.first_track < 1 || disc_toc.last_track > 99 || disc_toc.first_track > disc_toc.last_track)
- {
- throw(MDFN_Error(0, _("TOC first(%d)/last(%d) track numbers bad."), disc_toc.first_track, disc_toc.last_track));
- }
- }
- DiscEjected = eject_status;
- }
- }
- catch(std::exception &e)
- {
- MDFN_PrintError("%s", e.what());
- return(false);
- }
-
- return(true);
-}
-
-
-class CDIF_Stream_Thing : public Stream
-{
- public:
-
- CDIF_Stream_Thing(CDIF *cdintf_arg, uint32 lba_arg, uint32 sector_count_arg);
- ~CDIF_Stream_Thing();
-
- virtual uint64 attributes(void);
- virtual uint8 *map(void);
- virtual void unmap(void);
-
- virtual uint64 read(void *data, uint64 count, bool error_on_eos = true);
- virtual void write(const void *data, uint64 count);
-
- virtual void seek(int64 offset, int whence);
- virtual int64 tell(void);
- virtual int64 size(void);
- virtual void close(void);
-
- private:
- CDIF *cdintf;
- const uint32 start_lba;
- const uint32 sector_count;
- int64 position;
-};
-
-CDIF_Stream_Thing::CDIF_Stream_Thing(CDIF *cdintf_arg, uint32 start_lba_arg, uint32 sector_count_arg) : cdintf(cdintf_arg), start_lba(start_lba_arg), sector_count(sector_count_arg)
-{
-
-}
-
-CDIF_Stream_Thing::~CDIF_Stream_Thing()
-{
-
-}
-
-uint64 CDIF_Stream_Thing::attributes(void)
-{
- return(ATTRIBUTE_READABLE | ATTRIBUTE_SEEKABLE);
-}
-
-uint8 *CDIF_Stream_Thing::map(void)
-{
- return NULL;
-}
-
-void CDIF_Stream_Thing::unmap(void)
-{
-
-}
-
-uint64 CDIF_Stream_Thing::read(void *data, uint64 count, bool error_on_eos)
-{
- if(count > (((uint64)sector_count * 2048) - position))
- {
- if(error_on_eos)
- {
- throw MDFN_Error(0, "EOF");
- }
-
- count = ((uint64)sector_count * 2048) - position;
- }
-
- if(!count)
- return(0);
-
- for(uint64 rp = position; rp < (position + count); rp = (rp &~ 2047) + 2048)
- {
- uint8 buf[2048];
-
- if(!cdintf->ReadSector(buf, start_lba + (rp / 2048), 1))
- {
- throw MDFN_Error(ErrnoHolder(EIO));
- }
-
- //::printf("Meow: %08llx -- %08llx\n", count, (rp - position) + std::min(2048 - (rp & 2047), count - (rp - position)));
- memcpy((uint8*)data + (rp - position), buf + (rp & 2047), std::min(2048 - (rp & 2047), count - (rp - position)));
- }
-
- position += count;
-
- return count;
-}
-
-void CDIF_Stream_Thing::write(const void *data, uint64 count)
-{
- throw MDFN_Error(ErrnoHolder(EBADF));
-}
-
-void CDIF_Stream_Thing::seek(int64 offset, int whence)
-{
- int64 new_position;
-
- switch(whence)
- {
- default:
- throw MDFN_Error(ErrnoHolder(EINVAL));
- break;
-
- case SEEK_SET:
- new_position = offset;
- break;
-
- case SEEK_CUR:
- new_position = position + offset;
- break;
-
- case SEEK_END:
- new_position = ((int64)sector_count * 2048) + offset;
- break;
- }
-
- if(new_position < 0 || new_position > ((int64)sector_count * 2048))
- throw MDFN_Error(ErrnoHolder(EINVAL));
-
- position = new_position;
-}
-
-int64 CDIF_Stream_Thing::tell(void)
-{
- return position;
-}
-
-int64 CDIF_Stream_Thing::size(void)
-{
- return(sector_count * 2048);
-}
-
-void CDIF_Stream_Thing::close(void)
-{
-
-}
-
-
-Stream *CDIF::MakeStream(uint32 lba, uint32 sector_count)
-{
- return new CDIF_Stream_Thing(this, lba, sector_count);
-}
-
-
-CDIF *CDIF_Open(const char *path, const bool is_device, bool image_memcache)
-{
- if(is_device)
- return new CDIF_MT(cdaccess_open_phys(path));
- else
- {
- CDAccess *cda = cdaccess_open_image(path, image_memcache);
-
- if(!image_memcache)
- return new CDIF_MT(cda);
- else
- return new CDIF_ST(cda);
- }
-}
diff --git a/psx/octoshock/cdrom/cdromif.h b/psx/octoshock/cdrom/cdromif.h
deleted file mode 100644
index 8b0a0bbcd4..0000000000
--- a/psx/octoshock/cdrom/cdromif.h
+++ /dev/null
@@ -1,70 +0,0 @@
-/* Mednafen - Multi-system Emulator
- *
- * 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
- */
-
-#ifndef __MDFN_CDROM_CDROMIF_H
-#define __MDFN_CDROM_CDROMIF_H
-
-#include "CDUtility.h"
-#include "../Stream.h"
-
-#include
-
-typedef CDUtility::TOC CD_TOC;
-
-class CDIF
-{
- public:
-
- CDIF();
- virtual ~CDIF();
-
- inline void ReadTOC(CDUtility::TOC *read_target)
- {
- *read_target = disc_toc;
- }
-
- virtual void HintReadSector(uint32 lba) = 0;
- virtual bool ReadRawSector(uint8 *buf, uint32 lba) = 0;
-
- // Call for mode 1 or mode 2 form 1 only.
- bool ValidateRawSector(uint8 *buf);
-
- // Utility/Wrapped functions
- // Reads mode 1 and mode2 form 1 sectors(2048 bytes per sector returned)
- // Will return the type(1, 2) of the first sector read to the buffer supplied, 0 on error
- int ReadSector(uint8* pBuf, uint32 lba, uint32 nSectors);
-
- // Return true if operation succeeded or it was a NOP(either due to not being implemented, or the current status matches eject_status).
- // Returns false on failure(usually drive error of some kind; not completely fatal, can try again).
- virtual bool Eject(bool eject_status) = 0;
-
- inline bool IsPhysical(void) { return(is_phys_cache); }
-
- // For Mode 1, or Mode 2 Form 1.
- // No reference counting or whatever is done, so if you destroy the CDIF object before you destroy the returned Stream, things will go BOOM.
- Stream *MakeStream(uint32 lba, uint32 sector_count);
-
- protected:
- bool UnrecoverableError;
- bool is_phys_cache;
- CDUtility::TOC disc_toc;
- int DiscEjected; // 0 = inserted, 1 = ejected, -1 = DRAGONS ATE THE DISC. NOM NOM NOM.
-};
-
-CDIF *CDIF_Open(const char *path, const bool is_device, bool image_memcache);
-
-#endif
diff --git a/psx/octoshock/cdrom/crc32.cpp b/psx/octoshock/cdrom/crc32.cpp
deleted file mode 100644
index 9b3b2c8e6b..0000000000
--- a/psx/octoshock/cdrom/crc32.cpp
+++ /dev/null
@@ -1,130 +0,0 @@
-/* dvdisaster: Additional error correction for optical media.
- * Copyright (C) 2004-2007 Carsten Gnoerlich.
- * Project home page: http://www.dvdisaster.com
- * Email: carsten@dvdisaster.com -or- cgnoerlich@fsfe.org
- *
- * CRC32 code based upon public domain code by Ross Williams (see notes below)
- *
- * 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,
- * or direct your browser at http://www.gnu.org.
- */
-
-#include "dvdisaster.h"
-
-/***
- *** EDC checksum used in CDROM sectors
- ***/
-
-/*****************************************************************/
-/* */
-/* CRC LOOKUP TABLE */
-/* ================ */
-/* The following CRC lookup table was generated automagically */
-/* by the Rocksoft^tm Model CRC Algorithm Table Generation */
-/* Program V1.0 using the following model parameters: */
-/* */
-/* Width : 4 bytes. */
-/* Poly : 0x8001801BL */
-/* Reverse : TRUE. */
-/* */
-/* For more information on the Rocksoft^tm Model CRC Algorithm, */
-/* see the document titled "A Painless Guide to CRC Error */
-/* Detection Algorithms" by Ross Williams */
-/* (ross@guest.adelaide.edu.au.). This document is likely to be */
-/* in the FTP archive "ftp.adelaide.edu.au/pub/rocksoft". */
-/* */
-/*****************************************************************/
-
-unsigned long edctable[256] =
-{
- 0x00000000L, 0x90910101L, 0x91210201L, 0x01B00300L,
- 0x92410401L, 0x02D00500L, 0x03600600L, 0x93F10701L,
- 0x94810801L, 0x04100900L, 0x05A00A00L, 0x95310B01L,
- 0x06C00C00L, 0x96510D01L, 0x97E10E01L, 0x07700F00L,
- 0x99011001L, 0x09901100L, 0x08201200L, 0x98B11301L,
- 0x0B401400L, 0x9BD11501L, 0x9A611601L, 0x0AF01700L,
- 0x0D801800L, 0x9D111901L, 0x9CA11A01L, 0x0C301B00L,
- 0x9FC11C01L, 0x0F501D00L, 0x0EE01E00L, 0x9E711F01L,
- 0x82012001L, 0x12902100L, 0x13202200L, 0x83B12301L,
- 0x10402400L, 0x80D12501L, 0x81612601L, 0x11F02700L,
- 0x16802800L, 0x86112901L, 0x87A12A01L, 0x17302B00L,
- 0x84C12C01L, 0x14502D00L, 0x15E02E00L, 0x85712F01L,
- 0x1B003000L, 0x8B913101L, 0x8A213201L, 0x1AB03300L,
- 0x89413401L, 0x19D03500L, 0x18603600L, 0x88F13701L,
- 0x8F813801L, 0x1F103900L, 0x1EA03A00L, 0x8E313B01L,
- 0x1DC03C00L, 0x8D513D01L, 0x8CE13E01L, 0x1C703F00L,
- 0xB4014001L, 0x24904100L, 0x25204200L, 0xB5B14301L,
- 0x26404400L, 0xB6D14501L, 0xB7614601L, 0x27F04700L,
- 0x20804800L, 0xB0114901L, 0xB1A14A01L, 0x21304B00L,
- 0xB2C14C01L, 0x22504D00L, 0x23E04E00L, 0xB3714F01L,
- 0x2D005000L, 0xBD915101L, 0xBC215201L, 0x2CB05300L,
- 0xBF415401L, 0x2FD05500L, 0x2E605600L, 0xBEF15701L,
- 0xB9815801L, 0x29105900L, 0x28A05A00L, 0xB8315B01L,
- 0x2BC05C00L, 0xBB515D01L, 0xBAE15E01L, 0x2A705F00L,
- 0x36006000L, 0xA6916101L, 0xA7216201L, 0x37B06300L,
- 0xA4416401L, 0x34D06500L, 0x35606600L, 0xA5F16701L,
- 0xA2816801L, 0x32106900L, 0x33A06A00L, 0xA3316B01L,
- 0x30C06C00L, 0xA0516D01L, 0xA1E16E01L, 0x31706F00L,
- 0xAF017001L, 0x3F907100L, 0x3E207200L, 0xAEB17301L,
- 0x3D407400L, 0xADD17501L, 0xAC617601L, 0x3CF07700L,
- 0x3B807800L, 0xAB117901L, 0xAAA17A01L, 0x3A307B00L,
- 0xA9C17C01L, 0x39507D00L, 0x38E07E00L, 0xA8717F01L,
- 0xD8018001L, 0x48908100L, 0x49208200L, 0xD9B18301L,
- 0x4A408400L, 0xDAD18501L, 0xDB618601L, 0x4BF08700L,
- 0x4C808800L, 0xDC118901L, 0xDDA18A01L, 0x4D308B00L,
- 0xDEC18C01L, 0x4E508D00L, 0x4FE08E00L, 0xDF718F01L,
- 0x41009000L, 0xD1919101L, 0xD0219201L, 0x40B09300L,
- 0xD3419401L, 0x43D09500L, 0x42609600L, 0xD2F19701L,
- 0xD5819801L, 0x45109900L, 0x44A09A00L, 0xD4319B01L,
- 0x47C09C00L, 0xD7519D01L, 0xD6E19E01L, 0x46709F00L,
- 0x5A00A000L, 0xCA91A101L, 0xCB21A201L, 0x5BB0A300L,
- 0xC841A401L, 0x58D0A500L, 0x5960A600L, 0xC9F1A701L,
- 0xCE81A801L, 0x5E10A900L, 0x5FA0AA00L, 0xCF31AB01L,
- 0x5CC0AC00L, 0xCC51AD01L, 0xCDE1AE01L, 0x5D70AF00L,
- 0xC301B001L, 0x5390B100L, 0x5220B200L, 0xC2B1B301L,
- 0x5140B400L, 0xC1D1B501L, 0xC061B601L, 0x50F0B700L,
- 0x5780B800L, 0xC711B901L, 0xC6A1BA01L, 0x5630BB00L,
- 0xC5C1BC01L, 0x5550BD00L, 0x54E0BE00L, 0xC471BF01L,
- 0x6C00C000L, 0xFC91C101L, 0xFD21C201L, 0x6DB0C300L,
- 0xFE41C401L, 0x6ED0C500L, 0x6F60C600L, 0xFFF1C701L,
- 0xF881C801L, 0x6810C900L, 0x69A0CA00L, 0xF931CB01L,
- 0x6AC0CC00L, 0xFA51CD01L, 0xFBE1CE01L, 0x6B70CF00L,
- 0xF501D001L, 0x6590D100L, 0x6420D200L, 0xF4B1D301L,
- 0x6740D400L, 0xF7D1D501L, 0xF661D601L, 0x66F0D700L,
- 0x6180D800L, 0xF111D901L, 0xF0A1DA01L, 0x6030DB00L,
- 0xF3C1DC01L, 0x6350DD00L, 0x62E0DE00L, 0xF271DF01L,
- 0xEE01E001L, 0x7E90E100L, 0x7F20E200L, 0xEFB1E301L,
- 0x7C40E400L, 0xECD1E501L, 0xED61E601L, 0x7DF0E700L,
- 0x7A80E800L, 0xEA11E901L, 0xEBA1EA01L, 0x7B30EB00L,
- 0xE8C1EC01L, 0x7850ED00L, 0x79E0EE00L, 0xE971EF01L,
- 0x7700F000L, 0xE791F101L, 0xE621F201L, 0x76B0F300L,
- 0xE541F401L, 0x75D0F500L, 0x7460F600L, 0xE4F1F701L,
- 0xE381F801L, 0x7310F900L, 0x72A0FA00L, 0xE231FB01L,
- 0x71C0FC00L, 0xE151FD01L, 0xE0E1FE01L, 0x7070FF00L
-};
-
-/*
- * CDROM EDC calculation
- */
-
-uint32 EDCCrc32(const unsigned char *data, int len)
-{
- uint32 crc = 0;
-
- while(len--)
- crc = edctable[(crc ^ *data++) & 0xFF] ^ (crc >> 8);
-
- return crc;
-}
diff --git a/psx/octoshock/cdrom/dvdisaster.h b/psx/octoshock/cdrom/dvdisaster.h
deleted file mode 100644
index a3e79de6d2..0000000000
--- a/psx/octoshock/cdrom/dvdisaster.h
+++ /dev/null
@@ -1,172 +0,0 @@
-/* dvdisaster: Additional error correction for optical media.
- * Copyright (C) 2004-2007 Carsten Gnoerlich.
- * Project home page: http://www.dvdisaster.com
- * Email: carsten@dvdisaster.com -or- cgnoerlich@fsfe.org
- *
- * 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,
- * or direct your browser at http://www.gnu.org.
- */
-
-#ifndef DVDISASTER_H
-#define DVDISASTER_H
-
-/* "Dare to be gorgeous and unique.
- * But don't ever be cryptic or otherwise unfathomable.
- * Make it unforgettably great."
- *
- * From "A Final Note on Style",
- * Amiga Intuition Reference Manual, 1986, p. 231
- */
-
-/***
- *** I'm too lazy to mess with #include dependencies.
- *** Everything #includeable is rolled up herein...
- */
-
-#include "octoshock.h"
-
-#include
-#include
-#include
-#include
-#include
-#include
-#include
-#include
-#include
-#include
-#include
-
-
-/***
- *** dvdisaster.c
- ***/
-
-void PrepareDeadSector(void);
-
-void CreateEcc(void);
-void FixEcc(void);
-void Verify(void);
-
-uint32 EDCCrc32(const unsigned char*, int);
-
-/***
- *** galois.c
- ***
- * This is currently the hardcoded GF(2**8).
- * int32 gives abundant space for the GF.
- * Squeezing it down to uint8 won't probably gain much,
- * so we implement this defensively here.
- *
- * Note that some performance critical stuff needs to
- * be #included from galois-inlines.h
- */
-
-/* Galois field parameters for 8bit symbol Reed-Solomon code */
-
-#define GF_SYMBOLSIZE 8
-#define GF_FIELDSIZE (1<= GF_FIELDMAX)
- {
- x -= GF_FIELDMAX;
- x = (x >> GF_SYMBOLSIZE) + (x & GF_FIELDMAX);
- }
-
- return x;
-}
diff --git a/psx/octoshock/cdrom/galois.cpp b/psx/octoshock/cdrom/galois.cpp
deleted file mode 100644
index 2792cfc341..0000000000
--- a/psx/octoshock/cdrom/galois.cpp
+++ /dev/null
@@ -1,156 +0,0 @@
-/* dvdisaster: Additional error correction for optical media.
- * Copyright (C) 2004-2007 Carsten Gnoerlich.
- * Project home page: http://www.dvdisaster.com
- * Email: carsten@dvdisaster.com -or- cgnoerlich@fsfe.org
- *
- * The Reed-Solomon error correction draws a lot of inspiration - and even code -
- * from Phil Karn's excellent Reed-Solomon library: http://www.ka9q.net/code/fec/
- *
- * 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,
- * or direct your browser at http://www.gnu.org.
- */
-
-#include "dvdisaster.h"
-
-#include "galois-inlines.h"
-
-/***
- *** Galois field arithmetic.
- ***
- * Calculations are done over the extension field GF(2**n).
- * Be careful not to overgeneralize these arithmetics;
- * they only work for the case of GF(p**n) with p being prime.
- */
-
-/* Initialize the Galois field tables */
-
-
-GaloisTables* CreateGaloisTables(int32 gf_generator)
-{
- GaloisTables *gt = (GaloisTables *)calloc(1, sizeof(GaloisTables));
- int32 b,log;
-
- /* Allocate the tables.
- The encoder uses a special version of alpha_to which has the mod_fieldmax()
- folded into the table. */
-
- gt->gfGenerator = gf_generator;
-
- gt->indexOf = (int32 *)calloc(GF_FIELDSIZE, sizeof(int32));
- gt->alphaTo = (int32 *)calloc(GF_FIELDSIZE, sizeof(int32));
- gt->encAlphaTo = (int32 *)calloc(2*GF_FIELDSIZE, sizeof(int32));
-
- /* create the log/ilog values */
-
- for(b=1, log=0; logindexOf[b] = log;
- gt->alphaTo[log] = b;
- b = b << 1;
- if(b & GF_FIELDSIZE)
- b = b ^ gf_generator;
- }
-
- if(b!=1)
- {
- printf("Failed to create the Galois field log tables!\n");
- exit(1);
- }
-
- /* we're even closed using infinity (makes things easier) */
-
- gt->indexOf[0] = GF_ALPHA0; /* log(0) = inf */
- gt->alphaTo[GF_ALPHA0] = 0; /* and the other way around */
-
- for(b=0; b<2*GF_FIELDSIZE; b++)
- gt->encAlphaTo[b] = gt->alphaTo[mod_fieldmax(b)];
-
- return gt;
-}
-
-void FreeGaloisTables(GaloisTables *gt)
-{
- if(gt->indexOf) free(gt->indexOf);
- if(gt->alphaTo) free(gt->alphaTo);
- if(gt->encAlphaTo) free(gt->encAlphaTo);
-
- free(gt);
-}
-
-/***
- *** Create the the Reed-Solomon generator polynomial
- *** and some auxiliary data structures.
- */
-
-ReedSolomonTables *CreateReedSolomonTables(GaloisTables *gt,
- int32 first_consecutive_root,
- int32 prim_elem,
- int nroots_in)
-{ ReedSolomonTables *rt = (ReedSolomonTables *)calloc(1, sizeof(ReedSolomonTables));
- int32 i,j,root;
-
- rt->gfTables = gt;
- rt->fcr = first_consecutive_root;
- rt->primElem = prim_elem;
- rt->nroots = nroots_in;
- rt->ndata = GF_FIELDMAX - rt->nroots;
-
- rt->gpoly = (int32 *)calloc((rt->nroots+1), sizeof(int32));
-
- /* Create the RS code generator polynomial */
-
- rt->gpoly[0] = 1;
-
- for(i=0, root=first_consecutive_root*prim_elem; inroots; i++, root+=prim_elem)
- { rt->gpoly[i+1] = 1;
-
- /* Multiply gpoly by alpha**(root+x) */
-
- for(j=i; j>0; j--)
- {
- if(rt->gpoly[j] != 0)
- rt->gpoly[j] = rt->gpoly[j-1] ^ gt->alphaTo[mod_fieldmax(gt->indexOf[rt->gpoly[j]] + root)];
- else
- rt->gpoly[j] = rt->gpoly[j-1];
- }
-
- rt->gpoly[0] = gt->alphaTo[mod_fieldmax(gt->indexOf[rt->gpoly[0]] + root)];
- }
-
- /* Store the polynomials index for faster encoding */
-
- for(i=0; i<=rt->nroots; i++)
- rt->gpoly[i] = gt->indexOf[rt->gpoly[i]];
-
-#if 0
- /* for the precalculated unrolled loops only */
-
- for(i=gt->nroots-1; i>0; i--)
- PrintCLI(
- " par_idx[((++spk)&%d)] ^= enc_alpha_to[feedback + %3d];\n",
- nroots-1,gt->gpoly[i]);
-
- PrintCLI(" par_idx[sp] = enc_alpha_to[feedback + %3d];\n",
- gt->gpoly[0]);
-#endif
-
- return rt;
-}
-
-void FreeReedSolomonTables(ReedSolomonTables *rt)
-{
- if(rt->gpoly) free(rt->gpoly);
-
- free(rt);
-}
diff --git a/psx/octoshock/cdrom/l-ec.cpp b/psx/octoshock/cdrom/l-ec.cpp
deleted file mode 100644
index 5c035ce4ab..0000000000
--- a/psx/octoshock/cdrom/l-ec.cpp
+++ /dev/null
@@ -1,478 +0,0 @@
-/* dvdisaster: Additional error correction for optical media.
- * Copyright (C) 2004-2007 Carsten Gnoerlich.
- * Project home page: http://www.dvdisaster.com
- * Email: carsten@dvdisaster.com -or- cgnoerlich@fsfe.org
- *
- * The Reed-Solomon error correction draws a lot of inspiration - and even code -
- * from Phil Karn's excellent Reed-Solomon library: http://www.ka9q.net/code/fec/
- *
- * 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,
- * or direct your browser at http://www.gnu.org.
- */
-
-#include "dvdisaster.h"
-
-#include "galois-inlines.h"
-
-#define MIN(a, b) (((a) < (b)) ? (a) : (b))
-
-/***
- *** Mapping between cd frame and parity vectors
- ***/
-
-/*
- * Mapping of frame bytes to P/Q Vectors
- */
-
-int PToByteIndex(int p, int i)
-{ return 12 + p + i*86;
-}
-
-void ByteIndexToP(int b, int *p, int *i)
-{ *p = (b-12)%86;
- *i = (b-12)/86;
-}
-
-int QToByteIndex(int q, int i)
-{ int offset = 12 + (q & 1);
-
- if(i == 43) return 2248+q;
- if(i == 44) return 2300+q;
-
- q&=~1;
- return offset + (q*43 + i*88) % 2236;
-}
-
-void ByteIndexToQ(int b, int *q, int *i)
-{ int x,y,offset;
-
- if(b >= 2300)
- { *i = 44;
- *q = (b-2300);
- return;
- }
-
- if(b >= 2248)
- { *i = 43;
- *q = (b-2248);
- return;
- }
-
- offset = b&1;
- b = (b-12)/2;
- x = b/43;
- y = (b-(x*43))%26;
- *i = b-(x*43);
- *q = 2*((x+26-y)%26)+offset;
-}
-
-/*
- * There are 86 vectors of P-parity, yielding a RS(26,24) code.
- */
-
-void GetPVector(unsigned char *frame, unsigned char *data, int n)
-{ int i;
- int w_idx = n+12;
-
- for(i=0; i<26; i++, w_idx+=86)
- data[i] = frame[w_idx];
-}
-
-void SetPVector(unsigned char *frame, unsigned char *data, int n)
-{ int i;
- int w_idx = n+12;
-
- for(i=0; i<26; i++, w_idx+=86)
- frame[w_idx] = data[i];
-}
-
-void FillPVector(unsigned char *frame, unsigned char data, int n)
-{ int i;
- int w_idx = n+12;
-
- for(i=0; i<26; i++, w_idx+=86)
- frame[w_idx] = data;
-}
-
-void OrPVector(unsigned char *frame, unsigned char value, int n)
-{ int i;
- int w_idx = n+12;
-
- for(i=0; i<26; i++, w_idx+=86)
- frame[w_idx] |= value;
-}
-
-void AndPVector(unsigned char *frame, unsigned char value, int n)
-{ int i;
- int w_idx = n+12;
-
- for(i=0; i<26; i++, w_idx+=86)
- frame[w_idx] &= value;
-}
-
-/*
- * There are 52 vectors of Q-parity, yielding a RS(45,43) code.
- */
-
-void GetQVector(unsigned char *frame, unsigned char *data, int n)
-{ int offset = 12 + (n & 1);
- int w_idx = (n&~1) * 43;
- int i;
-
- for(i=0; i<43; i++, w_idx+=88)
- data[i] = frame[(w_idx % 2236) + offset];
-
- data[43] = frame[2248 + n];
- data[44] = frame[2300 + n];
-}
-
-void SetQVector(unsigned char *frame, unsigned char *data, int n)
-{ int offset = 12 + (n & 1);
- int w_idx = (n&~1) * 43;
- int i;
-
- for(i=0; i<43; i++, w_idx+=88)
- frame[(w_idx % 2236) + offset] = data[i];
-
- frame[2248 + n] = data[43];
- frame[2300 + n] = data[44];
-}
-
-void FillQVector(unsigned char *frame, unsigned char data, int n)
-{ int offset = 12 + (n & 1);
- int w_idx = (n&~1) * 43;
- int i;
-
- for(i=0; i<43; i++, w_idx+=88)
- frame[(w_idx % 2236) + offset] = data;
-
- frame[2248 + n] = data;
- frame[2300 + n] = data;
-}
-
-void OrQVector(unsigned char *frame, unsigned char data, int n)
-{ int offset = 12 + (n & 1);
- int w_idx = (n&~1) * 43;
- int i;
-
- for(i=0; i<43; i++, w_idx+=88)
- frame[(w_idx % 2236) + offset] |= data;
-
- frame[2248 + n] |= data;
- frame[2300 + n] |= data;
-}
-
-void AndQVector(unsigned char *frame, unsigned char data, int n)
-{ int offset = 12 + (n & 1);
- int w_idx = (n&~1) * 43;
- int i;
-
- for(i=0; i<43; i++, w_idx+=88)
- frame[(w_idx % 2236) + offset] &= data;
-
- frame[2248 + n] &= data;
- frame[2300 + n] &= data;
-}
-
-/***
- *** C2 error counting
- ***/
-
-int CountC2Errors(unsigned char *frame)
-{ int i,count = 0;
- frame += 2352;
-
- for(i=0; i<294; i++, frame++)
- { if(*frame & 0x01) count++;
- if(*frame & 0x02) count++;
- if(*frame & 0x04) count++;
- if(*frame & 0x08) count++;
- if(*frame & 0x10) count++;
- if(*frame & 0x20) count++;
- if(*frame & 0x40) count++;
- if(*frame & 0x80) count++;
- }
-
- return count;
-}
-
-/***
- *** L-EC error correction for CD raw data sectors
- ***/
-
-/*
- * These could be used from ReedSolomonTables,
- * but hardcoding them is faster.
- */
-
-#define NROOTS 2
-#define LEC_FIRST_ROOT 0 //GF_ALPHA0
-#define LEC_PRIM_ELEM 1
-#define LEC_PRIMTH_ROOT 1
-
-/*
- * Calculate the error syndrome
- */
-
-int DecodePQ(ReedSolomonTables *rt, unsigned char *data, int padding,
- int *erasure_list, int erasure_count)
-{ GaloisTables *gt = rt->gfTables;
- int syndrome[NROOTS];
- int lambda[NROOTS+1];
- int omega[NROOTS+1];
- int b[NROOTS+1];
- int reg[NROOTS+1];
- int root[NROOTS];
- int loc[NROOTS];
- int syn_error;
- int deg_lambda,lambda_roots;
- int deg_omega;
- int shortened_size = GF_FIELDMAX - padding;
- int corrected = 0;
- int i,j,k;
- int r,el;
-
- /*** Form the syndromes: Evaluate data(x) at roots of g(x) */
-
- for(i=0; ialphaTo[mod_fieldmax(gt->indexOf[syndrome[i]]
- + (LEC_FIRST_ROOT+i)*LEC_PRIM_ELEM)];
-
- /*** Convert syndrome to index form, check for nonzero condition. */
-
- syn_error = 0;
- for(i=0; iindexOf[syndrome[i]];
- }
-
- /*** If the syndrome is zero, everything is fine. */
-
- if(!syn_error)
- return 0;
-
- /*** Initialize lambda to be the erasure locator polynomial */
-
- lambda[0] = 1;
- lambda[1] = lambda[2] = 0;
-
- erasure_list[0] += padding;
- erasure_list[1] += padding;
-
- if(erasure_count > 2) /* sanity check */
- erasure_count = 0;
-
- if(erasure_count > 0)
- { lambda[1] = gt->alphaTo[mod_fieldmax(LEC_PRIM_ELEM*(GF_FIELDMAX-1-erasure_list[0]))];
-
- for(i=1; i0; j--)
- { int tmp = gt->indexOf[lambda[j-1]];
-
- if(tmp != GF_ALPHA0)
- lambda[j] ^= gt->alphaTo[mod_fieldmax(u + tmp)];
- }
- }
- }
-
- for(i=0; iindexOf[lambda[i]];
-
- /*** Berlekamp-Massey algorithm to determine error+erasure locator polynomial */
-
- r = erasure_count; /* r is the step number */
- el = erasure_count;
-
- /* Compute discrepancy at the r-th step in poly-form */
-
- while(++r <= NROOTS)
- { int discr_r = 0;
-
- for(i=0; ialphaTo[mod_fieldmax(gt->indexOf[lambda[i]] + syndrome[r-i-1])];
-
- discr_r = gt->indexOf[discr_r];
-
- if(discr_r == GF_ALPHA0)
- { /* B(x) = x*B(x) */
- memmove(b+1, b, NROOTS*sizeof(b[0]));
- b[0] = GF_ALPHA0;
- }
- else
- { int t[NROOTS+1];
-
- /* T(x) = lambda(x) - discr_r*x*b(x) */
- t[0] = lambda[0];
- for(i=0; ialphaTo[mod_fieldmax(discr_r + b[i])];
- else t[i+1] = lambda[i+1];
- }
-
- if(2*el <= r+erasure_count-1)
- { el = r + erasure_count - el;
-
- /* B(x) <-- inv(discr_r) * lambda(x) */
- for(i=0; i<=NROOTS; i++)
- b[i] = (lambda[i] == 0) ? GF_ALPHA0
- : mod_fieldmax(gt->indexOf[lambda[i]] - discr_r + GF_FIELDMAX);
- }
- else
- { /* 2 lines below: B(x) <-- x*B(x) */
- memmove(b+1, b, NROOTS*sizeof(b[0]));
- b[0] = GF_ALPHA0;
- }
-
- memcpy(lambda, t, (NROOTS+1)*sizeof(t[0]));
- }
- }
-
- /*** Convert lambda to index form and compute deg(lambda(x)) */
-
- deg_lambda = 0;
- for(i=0; iindexOf[lambda[i]];
- if(lambda[i] != GF_ALPHA0)
- deg_lambda = i;
- }
-
- /*** Find roots of the error+erasure locator polynomial by Chien search */
-
- memcpy(reg+1, lambda+1, NROOTS*sizeof(reg[0]));
- lambda_roots = 0; /* Number of roots of lambda(x) */
-
- for(i=1, k=LEC_PRIMTH_ROOT-1; i<=GF_FIELDMAX; i++, k=mod_fieldmax(k+LEC_PRIMTH_ROOT))
- { int q=1; /* lambda[0] is always 0 */
-
- for(j=deg_lambda; j>0; j--)
- { if(reg[j] != GF_ALPHA0)
- { reg[j] = mod_fieldmax(reg[j] + j);
- q ^= gt->alphaTo[reg[j]];
- }
- }
-
- if(q != 0) continue; /* Not a root */
-
- /* store root in index-form and the error location number */
-
- root[lambda_roots] = i;
- loc[lambda_roots] = k;
-
- /* If we've already found max possible roots, abort the search to save time */
-
- if(++lambda_roots == deg_lambda) break;
- }
-
- /* deg(lambda) unequal to number of roots => uncorrectable error detected
- This is not reliable for very small numbers of roots, e.g. nroots = 2 */
-
- if(deg_lambda != lambda_roots)
- { return -1;
- }
-
- /* Compute err+eras evaluator poly omega(x) = syn(x)*lambda(x)
- (modulo x**nroots). in index form. Also find deg(omega). */
-
- deg_omega = deg_lambda-1;
-
- for(i=0; i<=deg_omega; i++)
- { int tmp = 0;
-
- for(j=i; j>=0; j--)
- { if((syndrome[i - j] != GF_ALPHA0) && (lambda[j] != GF_ALPHA0))
- tmp ^= gt->alphaTo[mod_fieldmax(syndrome[i - j] + lambda[j])];
- }
-
- omega[i] = gt->indexOf[tmp];
- }
-
- /* Compute error values in poly-form.
- num1 = omega(inv(X(l))),
- num2 = inv(X(l))**(FIRST_ROOT-1) and
- den = lambda_pr(inv(X(l))) all in poly-form. */
-
- for(j=lambda_roots-1; j>=0; j--)
- { int num1 = 0;
- int num2;
- int den;
- int location = loc[j];
-
- for(i=deg_omega; i>=0; i--)
- { if(omega[i] != GF_ALPHA0)
- num1 ^= gt->alphaTo[mod_fieldmax(omega[i] + i * root[j])];
- }
-
- num2 = gt->alphaTo[mod_fieldmax(root[j] * (LEC_FIRST_ROOT - 1) + GF_FIELDMAX)];
- den = 0;
-
- /* lambda[i+1] for i even is the formal derivative lambda_pr of lambda[i] */
-
- for(i=MIN(deg_lambda, NROOTS-1) & ~1; i>=0; i-=2)
- { if(lambda[i+1] != GF_ALPHA0)
- den ^= gt->alphaTo[mod_fieldmax(lambda[i+1] + i * root[j])];
- }
-
- /* Apply error to data */
-
- if(num1 != 0 && location >= padding)
- {
- corrected++;
- data[location-padding] ^= gt->alphaTo[mod_fieldmax(gt->indexOf[num1] + gt->indexOf[num2]
- + GF_FIELDMAX - gt->indexOf[den])];
-
- /* If no erasures were given, at most one error was corrected.
- Return its position in erasure_list[0]. */
-
- if(!erasure_count)
- erasure_list[0] = location-padding;
- }
-#if 1
- else return -3;
-#endif
- }
-
- /*** Form the syndromes: Evaluate data(x) at roots of g(x) */
-
- for(i=0; ialphaTo[mod_fieldmax(gt->indexOf[syndrome[i]]
- + (LEC_FIRST_ROOT+i)*LEC_PRIM_ELEM)];
- }
-
- /*** Convert syndrome to index form, check for nonzero condition. */
-#if 1
- for(i=0; i
- *
- * 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., 675 Mass Ave, Cambridge, MA 02139, USA.
- */
-
-#ifdef HAVE_CONFIG_H
-#include
-#endif
-
-#include
-#include
-
-#include "lec.h"
-
-#define GF8_PRIM_POLY 0x11d /* x^8 + x^4 + x^3 + x^2 + 1 */
-
-#define EDC_POLY 0x8001801b /* (x^16 + x^15 + x^2 + 1) (x^16 + x^2 + x + 1) */
-
-#define LEC_HEADER_OFFSET 12
-#define LEC_DATA_OFFSET 16
-#define LEC_MODE1_DATA_LEN 2048
-#define LEC_MODE1_EDC_OFFSET 2064
-#define LEC_MODE1_INTERMEDIATE_OFFSET 2068
-#define LEC_MODE1_P_PARITY_OFFSET 2076
-#define LEC_MODE1_Q_PARITY_OFFSET 2248
-#define LEC_MODE2_FORM1_DATA_LEN (2048+8)
-#define LEC_MODE2_FORM1_EDC_OFFSET 2072
-#define LEC_MODE2_FORM2_DATA_LEN (2324+8)
-#define LEC_MODE2_FORM2_EDC_OFFSET 2348
-
-
-typedef u_int8_t gf8_t;
-
-static u_int8_t GF8_LOG[256];
-static gf8_t GF8_ILOG[256];
-
-static const class Gf8_Q_Coeffs_Results_01 {
-private:
- u_int16_t table[43][256];
-public:
- Gf8_Q_Coeffs_Results_01();
- ~Gf8_Q_Coeffs_Results_01() {}
- const u_int16_t *operator[] (int i) const { return &table[i][0]; }
- operator const u_int16_t *() const { return &table[0][0]; }
-} CF8_Q_COEFFS_RESULTS_01;
-
-static const class CrcTable {
-private:
- u_int32_t table[256];
-public:
- CrcTable();
- ~CrcTable() {}
- u_int32_t operator[](int i) const { return table[i]; }
- operator const u_int32_t *() const { return table; }
-} CRCTABLE;
-
-static const class ScrambleTable {
-private:
- u_int8_t table[2340];
-public:
- ScrambleTable();
- ~ScrambleTable() {}
- u_int8_t operator[](int i) const { return table[i]; }
- operator const u_int8_t *() const { return table; }
-} SCRAMBLE_TABLE;
-
-/* Creates the logarithm and inverse logarithm table that is required
- * for performing multiplication in the GF(8) domain.
- */
-static void gf8_create_log_tables()
-{
- u_int8_t log;
- u_int16_t b;
-
- for (b = 0; b <= 255; b++) {
- GF8_LOG[b] = 0;
- GF8_ILOG[b] = 0;
- }
-
- b = 1;
-
- for (log = 0; log < 255; log++) {
- GF8_LOG[(u_int8_t)b] = log;
- GF8_ILOG[log] = (u_int8_t)b;
-
- b <<= 1;
-
- if ((b & 0x100) != 0)
- b ^= GF8_PRIM_POLY;
- }
-}
-
-/* Addition in the GF(8) domain: just the XOR of the values.
- */
-#define gf8_add(a, b) (a) ^ (b)
-
-
-/* Multiplication in the GF(8) domain: add the logarithms (modulo 255)
- * and return the inverse logarithm. Not used!
- */
-#if 0
-static gf8_t gf8_mult(gf8_t a, gf8_t b)
-{
- int16_t sum;
-
- if (a == 0 || b == 0)
- return 0;
-
- sum = GF8_LOG[a] + GF8_LOG[b];
-
- if (sum >= 255)
- sum -= 255;
-
- return GF8_ILOG[sum];
-}
-#endif
-
-/* Division in the GF(8) domain: Like multiplication but logarithms a
- * subtracted.
- */
-static gf8_t gf8_div(gf8_t a, gf8_t b)
-{
- int16_t sum;
-
- assert(b != 0);
-
- if (a == 0)
- return 0;
-
- sum = GF8_LOG[a] - GF8_LOG[b];
-
- if (sum < 0)
- sum += 255;
-
- return GF8_ILOG[sum];
-}
-
-Gf8_Q_Coeffs_Results_01::Gf8_Q_Coeffs_Results_01()
-{
- int i, j;
- u_int16_t c;
- gf8_t GF8_COEFFS_HELP[2][45];
- u_int8_t GF8_Q_COEFFS[2][45];
-
-
- gf8_create_log_tables();
-
- /* build matrix H:
- * 1 1 ... 1 1
- * a^44 a^43 ... a^1 a^0
- *
- *
- */
-
- for (j = 0; j < 45; j++) {
- GF8_COEFFS_HELP[0][j] = 1; /* e0 */
- GF8_COEFFS_HELP[1][j] = GF8_ILOG[44-j]; /* e1 */
- }
-
-
- /* resolve equation system for parity byte 0 and 1 */
-
- /* e1' = e1 + e0 */
- for (j = 0; j < 45; j++) {
- GF8_Q_COEFFS[1][j] = gf8_add(GF8_COEFFS_HELP[1][j],
- GF8_COEFFS_HELP[0][j]);
- }
-
- /* e1'' = e1' / (a^1 + 1) */
- for (j = 0; j < 45; j++) {
- GF8_Q_COEFFS[1][j] = gf8_div(GF8_Q_COEFFS[1][j], GF8_Q_COEFFS[1][43]);
- }
-
- /* e0' = e0 + e1 / a^1 */
- for (j = 0; j < 45; j++) {
- GF8_Q_COEFFS[0][j] = gf8_add(GF8_COEFFS_HELP[0][j],
- gf8_div(GF8_COEFFS_HELP[1][j],
- GF8_ILOG[1]));
- }
-
- /* e0'' = e0' / (1 + 1 / a^1) */
- for (j = 0; j < 45; j++) {
- GF8_Q_COEFFS[0][j] = gf8_div(GF8_Q_COEFFS[0][j], GF8_Q_COEFFS[0][44]);
- }
-
- /*
- * Compute the products of 0..255 with all of the Q coefficients in
- * advance. When building the scalar product between the data vectors
- * and the P/Q vectors the individual products can be looked up in
- * this table
- *
- * The P parity coefficients are just a subset of the Q coefficients so
- * that we do not need to create a separate table for them.
- */
-
- for (j = 0; j < 43; j++) {
-
- table[j][0] = 0;
-
- for (i = 1; i < 256; i++) {
- c = GF8_LOG[i] + GF8_LOG[GF8_Q_COEFFS[0][j]];
- if (c >= 255) c -= 255;
- table[j][i] = GF8_ILOG[c];
-
- c = GF8_LOG[i] + GF8_LOG[GF8_Q_COEFFS[1][j]];
- if (c >= 255) c -= 255;
- table[j][i] |= GF8_ILOG[c]<<8;
- }
- }
-}
-
-/* Reverses the bits in 'd'. 'bits' defines the bit width of 'd'.
- */
-static u_int32_t mirror_bits(u_int32_t d, int bits)
-{
- int i;
- u_int32_t r = 0;
-
- for (i = 0; i < bits; i++) {
- r <<= 1;
-
- if ((d & 0x1) != 0)
- r |= 0x1;
-
- d >>= 1;
- }
-
- return r;
-}
-
-/* Build the CRC lookup table for EDC_POLY poly. The CRC is 32 bit wide
- * and reversed (i.e. the bit stream is divided by the EDC_POLY with the
- * LSB first order).
- */
-CrcTable::CrcTable ()
-{
- u_int32_t i, j;
- u_int32_t r;
-
- for (i = 0; i < 256; i++) {
- r = mirror_bits(i, 8);
-
- r <<= 24;
-
- for (j = 0; j < 8; j++) {
- if ((r & 0x80000000) != 0) {
- r <<= 1;
- r ^= EDC_POLY;
- }
- else {
- r <<= 1;
- }
- }
-
- r = mirror_bits(r, 32);
-
- table[i] = r;
- }
-}
-
-/* Calculates the CRC of given data with given lengths based on the
- * table lookup algorithm.
- */
-static u_int32_t calc_edc(u_int8_t *data, int len)
-{
- u_int32_t crc = 0;
-
- while (len--) {
- crc = CRCTABLE[(int)(crc ^ *data++) & 0xff] ^ (crc >> 8);
- }
-
- return crc;
-}
-
-/* Build the scramble table as defined in the yellow book. The bytes
- 12 to 2351 of a sector will be XORed with the data of this table.
- */
-ScrambleTable::ScrambleTable()
-{
- u_int16_t i, j;
- u_int16_t reg = 1;
- u_int8_t d;
-
- for (i = 0; i < 2340; i++) {
- d = 0;
-
- for (j = 0; j < 8; j++) {
- d >>= 1;
-
- if ((reg & 0x1) != 0)
- d |= 0x80;
-
- if ((reg & 0x1) != ((reg >> 1) & 0x1)) {
- reg >>= 1;
- reg |= 0x4000; /* 15-bit register */
- }
- else {
- reg >>= 1;
- }
- }
-
- table[i] = d;
- }
-}
-
-/* Calc EDC for a MODE 1 sector
- */
-static void calc_mode1_edc(u_int8_t *sector)
-{
- u_int32_t crc = calc_edc(sector, LEC_MODE1_DATA_LEN + 16);
-
- sector[LEC_MODE1_EDC_OFFSET] = crc & 0xffL;
- sector[LEC_MODE1_EDC_OFFSET + 1] = (crc >> 8) & 0xffL;
- sector[LEC_MODE1_EDC_OFFSET + 2] = (crc >> 16) & 0xffL;
- sector[LEC_MODE1_EDC_OFFSET + 3] = (crc >> 24) & 0xffL;
-}
-
-/* Calc EDC for a XA form 1 sector
- */
-static void calc_mode2_form1_edc(u_int8_t *sector)
-{
- u_int32_t crc = calc_edc(sector + LEC_DATA_OFFSET,
- LEC_MODE2_FORM1_DATA_LEN);
-
- sector[LEC_MODE2_FORM1_EDC_OFFSET] = crc & 0xffL;
- sector[LEC_MODE2_FORM1_EDC_OFFSET + 1] = (crc >> 8) & 0xffL;
- sector[LEC_MODE2_FORM1_EDC_OFFSET + 2] = (crc >> 16) & 0xffL;
- sector[LEC_MODE2_FORM1_EDC_OFFSET + 3] = (crc >> 24) & 0xffL;
-}
-
-/* Calc EDC for a XA form 2 sector
- */
-static void calc_mode2_form2_edc(u_int8_t *sector)
-{
- u_int32_t crc = calc_edc(sector + LEC_DATA_OFFSET,
- LEC_MODE2_FORM2_DATA_LEN);
-
- sector[LEC_MODE2_FORM2_EDC_OFFSET] = crc & 0xffL;
- sector[LEC_MODE2_FORM2_EDC_OFFSET + 1] = (crc >> 8) & 0xffL;
- sector[LEC_MODE2_FORM2_EDC_OFFSET + 2] = (crc >> 16) & 0xffL;
- sector[LEC_MODE2_FORM2_EDC_OFFSET + 3] = (crc >> 24) & 0xffL;
-}
-
-/* Writes the sync pattern to the given sector.
- */
-static void set_sync_pattern(u_int8_t *sector)
-{
- sector[0] = 0;
-
- sector[1] = sector[2] = sector[3] = sector[4] = sector[5] =
- sector[6] = sector[7] = sector[8] = sector[9] = sector[10] = 0xff;
-
- sector[11] = 0;
-}
-
-
-static u_int8_t bin2bcd(u_int8_t b)
-{
- return (((b/10) << 4) & 0xf0) | ((b%10) & 0x0f);
-}
-
-/* Builds the sector header.
- */
-static void set_sector_header(u_int8_t mode, u_int32_t adr, u_int8_t *sector)
-{
- sector[LEC_HEADER_OFFSET] = bin2bcd(adr / (60*75));
- sector[LEC_HEADER_OFFSET + 1] = bin2bcd((adr / 75) % 60);
- sector[LEC_HEADER_OFFSET + 2] = bin2bcd(adr % 75);
- sector[LEC_HEADER_OFFSET + 3] = mode;
-}
-
-/* Calculate the P parities for the sector.
- * The 43 P vectors of length 24 are combined with the GF8_P_COEFFS.
- */
-static void calc_P_parity(u_int8_t *sector)
-{
- int i, j;
- u_int16_t p01_msb, p01_lsb;
- u_int8_t *p_lsb_start;
- u_int8_t *p_lsb;
- u_int8_t *p0, *p1;
- u_int8_t d0,d1;
-
- p_lsb_start = sector + LEC_HEADER_OFFSET;
-
- p1 = sector + LEC_MODE1_P_PARITY_OFFSET;
- p0 = sector + LEC_MODE1_P_PARITY_OFFSET + 2 * 43;
-
- for (i = 0; i <= 42; i++) {
- p_lsb = p_lsb_start;
-
- p01_lsb = p01_msb = 0;
-
- for (j = 19; j <= 42; j++) {
- d0 = *p_lsb;
- d1 = *(p_lsb+1);
-
- p01_lsb ^= CF8_Q_COEFFS_RESULTS_01[j][d0];
- p01_msb ^= CF8_Q_COEFFS_RESULTS_01[j][d1];
-
- p_lsb += 2 * 43;
- }
-
- *p0 = p01_lsb;
- *(p0 + 1) = p01_msb;
-
- *p1 = p01_lsb>>8;
- *(p1 + 1) = p01_msb>>8;
-
- p0 += 2;
- p1 += 2;
-
- p_lsb_start += 2;
- }
-}
-
-/* Calculate the Q parities for the sector.
- * The 26 Q vectors of length 43 are combined with the GF8_Q_COEFFS.
- */
-static void calc_Q_parity(u_int8_t *sector)
-{
- int i, j;
- u_int16_t q01_lsb, q01_msb;
- u_int8_t *q_lsb_start;
- u_int8_t *q_lsb;
- u_int8_t *q0, *q1, *q_start;
- u_int8_t d0,d1;
-
- q_lsb_start = sector + LEC_HEADER_OFFSET;
-
- q_start = sector + LEC_MODE1_Q_PARITY_OFFSET;
- q1 = sector + LEC_MODE1_Q_PARITY_OFFSET;
- q0 = sector + LEC_MODE1_Q_PARITY_OFFSET + 2 * 26;
-
- for (i = 0; i <= 25; i++) {
- q_lsb = q_lsb_start;
-
- q01_lsb = q01_msb = 0;
-
- for (j = 0; j <= 42; j++) {
- d0 = *q_lsb;
- d1 = *(q_lsb+1);
-
- q01_lsb ^= CF8_Q_COEFFS_RESULTS_01[j][d0];
- q01_msb ^= CF8_Q_COEFFS_RESULTS_01[j][d1];
-
- q_lsb += 2 * 44;
-
- if (q_lsb >= q_start) {
- q_lsb -= 2 * 1118;
- }
- }
-
- *q0 = q01_lsb;
- *(q0 + 1) = q01_msb;
-
- *q1 = q01_lsb>>8;
- *(q1 + 1) = q01_msb>>8;
-
- q0 += 2;
- q1 += 2;
-
- q_lsb_start += 2 * 43;
- }
-}
-
-/* Encodes a MODE 0 sector.
- * 'adr' is the current physical sector address
- * 'sector' must be 2352 byte wide
- */
-void lec_encode_mode0_sector(u_int32_t adr, u_int8_t *sector)
-{
- u_int16_t i;
-
- set_sync_pattern(sector);
- set_sector_header(0, adr, sector);
-
- sector += 16;
-
- for (i = 0; i < 2336; i++)
- *sector++ = 0;
-}
-
-/* Encodes a MODE 1 sector.
- * 'adr' is the current physical sector address
- * 'sector' must be 2352 byte wide containing 2048 bytes user data at
- * offset 16
- */
-void lec_encode_mode1_sector(u_int32_t adr, u_int8_t *sector)
-{
- set_sync_pattern(sector);
- set_sector_header(1, adr, sector);
-
- calc_mode1_edc(sector);
-
- /* clear the intermediate field */
- sector[LEC_MODE1_INTERMEDIATE_OFFSET] =
- sector[LEC_MODE1_INTERMEDIATE_OFFSET + 1] =
- sector[LEC_MODE1_INTERMEDIATE_OFFSET + 2] =
- sector[LEC_MODE1_INTERMEDIATE_OFFSET + 3] =
- sector[LEC_MODE1_INTERMEDIATE_OFFSET + 4] =
- sector[LEC_MODE1_INTERMEDIATE_OFFSET + 5] =
- sector[LEC_MODE1_INTERMEDIATE_OFFSET + 6] =
- sector[LEC_MODE1_INTERMEDIATE_OFFSET + 7] = 0;
-
- calc_P_parity(sector);
- calc_Q_parity(sector);
-}
-
-/* Encodes a MODE 2 sector.
- * 'adr' is the current physical sector address
- * 'sector' must be 2352 byte wide containing 2336 bytes user data at
- * offset 16
- */
-void lec_encode_mode2_sector(u_int32_t adr, u_int8_t *sector)
-{
- set_sync_pattern(sector);
- set_sector_header(2, adr, sector);
-}
-
-/* Encodes a XA form 1 sector.
- * 'adr' is the current physical sector address
- * 'sector' must be 2352 byte wide containing 2048+8 bytes user data at
- * offset 16
- */
-void lec_encode_mode2_form1_sector(u_int32_t adr, u_int8_t *sector)
-{
- set_sync_pattern(sector);
-
- calc_mode2_form1_edc(sector);
-
- /* P/Q partiy must not contain the sector header so clear it */
- sector[LEC_HEADER_OFFSET] =
- sector[LEC_HEADER_OFFSET + 1] =
- sector[LEC_HEADER_OFFSET + 2] =
- sector[LEC_HEADER_OFFSET + 3] = 0;
-
- calc_P_parity(sector);
- calc_Q_parity(sector);
-
- /* finally add the sector header */
- set_sector_header(2, adr, sector);
-}
-
-/* Encodes a XA form 2 sector.
- * 'adr' is the current physical sector address
- * 'sector' must be 2352 byte wide containing 2324+8 bytes user data at
- * offset 16
- */
-void lec_encode_mode2_form2_sector(u_int32_t adr, u_int8_t *sector)
-{
- set_sync_pattern(sector);
-
- calc_mode2_form2_edc(sector);
-
- set_sector_header(2, adr, sector);
-}
-
-/* Scrambles and byte swaps an encoded sector.
- * 'sector' must be 2352 byte wide.
- */
-void lec_scramble(u_int8_t *sector)
-{
- u_int16_t i;
- const u_int8_t *stable = SCRAMBLE_TABLE;
- u_int8_t *p = sector;
- u_int8_t tmp;
-
-
- for (i = 0; i < 6; i++) {
- /* just swap bytes of sector sync */
- tmp = *p;
- *p = *(p + 1);
- p++;
- *p++ = tmp;
- }
- for (;i < (2352 / 2); i++) {
- /* scramble and swap bytes */
- tmp = *p ^ *stable++;
- *p = *(p + 1) ^ *stable++;
- p++;
- *p++ = tmp;
- }
-}
-
-#if 0
-#include
-#include
-#include
-#include
-
-int main(int argc, char **argv)
-{
- char *infile;
- char *outfile;
- int fd_in, fd_out;
- u_int8_t buffer1[2352];
- u_int8_t buffer2[2352];
- u_int32_t lba;
- int i;
-
-#if 0
- for (i = 0; i < 2048; i++)
- buffer1[i + 16] = 234;
-
- lba = 150;
-
- for (i = 0; i < 100000; i++) {
- lec_encode_mode1_sector(lba, buffer1);
- lec_scramble(buffer2);
- lba++;
- }
-
-#else
-
- if (argc != 3)
- return 1;
-
- infile = argv[1];
- outfile = argv[2];
-
-
- if ((fd_in = open(infile, O_RDONLY)) < 0) {
- perror("Cannot open input file");
- return 1;
- }
-
- if ((fd_out = open(outfile, O_WRONLY|O_CREAT|O_TRUNC, 0666)) < 0) {
- perror("Cannot open output file");
- return 1;
- }
-
- lba = 150;
-
- do {
- if (read(fd_in, buffer1, 2352) != 2352)
- break;
-
- switch (*(buffer1 + 12 + 3)) {
- case 1:
- memcpy(buffer2 + 16, buffer1 + 16, 2048);
-
- lec_encode_mode1_sector(lba, buffer2);
- break;
-
- case 2:
- if ((*(buffer1 + 12 + 4 + 2) & 0x20) != 0) {
- /* form 2 sector */
- memcpy(buffer2 + 16, buffer1 + 16, 2324 + 8);
- lec_encode_mode2_form2_sector(lba, buffer2);
- }
- else {
- /* form 1 sector */
- memcpy(buffer2 + 16, buffer1 + 16, 2048 + 8);
- lec_encode_mode2_form1_sector(lba, buffer2);
- }
- break;
- }
-
- if (memcmp(buffer1, buffer2, 2352) != 0) {
- printf("Verify error at lba %ld\n", lba);
- }
-
- lec_scramble(buffer2);
- write(fd_out, buffer2, 2352);
-
- lba++;
- } while (1);
-
- close(fd_in);
- close(fd_out);
-
-#endif
-
- return 0;
-}
-#endif
diff --git a/psx/octoshock/cdrom/lec.h b/psx/octoshock/cdrom/lec.h
deleted file mode 100644
index c5e874c3f3..0000000000
--- a/psx/octoshock/cdrom/lec.h
+++ /dev/null
@@ -1,77 +0,0 @@
-/* cdrdao - write audio CD-Rs in disc-at-once mode
- *
- * Copyright (C) 1998-2002 Andreas Mueller
- *
- * 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., 675 Mass Ave, Cambridge, MA 02139, USA.
- */
-
-#ifndef __LEC_H__
-#define __LEC_H__
-
-#ifdef HAVE_CONFIG_H
-#include
-#endif
-
-#include
-#include
-
-typedef uint32_t u_int32_t;
-typedef uint16_t u_int16_t;
-typedef uint8_t u_int8_t;
-
-#ifndef TRUE
-#define TRUE 1
-#endif
-
-/* Encodes a MODE 0 sector.
- * 'adr' is the current physical sector address
- * 'sector' must be 2352 byte wide
- */
-void lec_encode_mode0_sector(u_int32_t adr, u_int8_t *sector);
-
-/* Encodes a MODE 1 sector.
- * 'adr' is the current physical sector address
- * 'sector' must be 2352 byte wide containing 2048 bytes user data at
- * offset 16
- */
-void lec_encode_mode1_sector(u_int32_t adr, u_int8_t *sector);
-
-/* Encodes a MODE 2 sector.
- * 'adr' is the current physical sector address
- * 'sector' must be 2352 byte wide containing 2336 bytes user data at
- * offset 16
- */
-void lec_encode_mode2_sector(u_int32_t adr, u_int8_t *sector);
-
-/* Encodes a XA form 1 sector.
- * 'adr' is the current physical sector address
- * 'sector' must be 2352 byte wide containing 2048+8 bytes user data at
- * offset 16
- */
-void lec_encode_mode2_form1_sector(u_int32_t adr, u_int8_t *sector);
-
-/* Encodes a XA form 2 sector.
- * 'adr' is the current physical sector address
- * 'sector' must be 2352 byte wide containing 2324+8 bytes user data at
- * offset 16
- */
-void lec_encode_mode2_form2_sector(u_int32_t adr, u_int8_t *sector);
-
-/* Scrambles and byte swaps an encoded sector.
- * 'sector' must be 2352 byte wide.
- */
-void lec_scramble(u_int8_t *sector);
-
-#endif
diff --git a/psx/octoshock/cdrom/recover-raw.cpp b/psx/octoshock/cdrom/recover-raw.cpp
deleted file mode 100644
index 78be2e2a54..0000000000
--- a/psx/octoshock/cdrom/recover-raw.cpp
+++ /dev/null
@@ -1,203 +0,0 @@
-/* dvdisaster: Additional error correction for optical media.
- * Copyright (C) 2004-2007 Carsten Gnoerlich.
- * Project home page: http://www.dvdisaster.com
- * Email: carsten@dvdisaster.com -or- cgnoerlich@fsfe.org
- *
- * 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,
- * or direct your browser at http://www.gnu.org.
- */
-
-#include "dvdisaster.h"
-
-static GaloisTables *gt = NULL; /* for L-EC Reed-Solomon */
-static ReedSolomonTables *rt = NULL;
-
-bool Init_LEC_Correct(void)
-{
- gt = CreateGaloisTables(0x11d);
- rt = CreateReedSolomonTables(gt, 0, 1, 10);
-
- return(1);
-}
-
-void Kill_LEC_Correct(void)
-{
- FreeGaloisTables(gt);
- FreeReedSolomonTables(rt);
-}
-
-/***
- *** CD level CRC calculation
- ***/
-
-/*
- * Test raw sector against its 32bit CRC.
- * Returns TRUE if frame is good.
- */
-
-int CheckEDC(const unsigned char *cd_frame, bool xa_mode)
-{
- unsigned int expected_crc, real_crc;
- unsigned int crc_base = xa_mode ? 2072 : 2064;
-
- expected_crc = cd_frame[crc_base + 0] << 0;
- expected_crc |= cd_frame[crc_base + 1] << 8;
- expected_crc |= cd_frame[crc_base + 2] << 16;
- expected_crc |= cd_frame[crc_base + 3] << 24;
-
- if(xa_mode)
- real_crc = EDCCrc32(cd_frame+16, 2056);
- else
- real_crc = EDCCrc32(cd_frame, 2064);
-
- if(expected_crc == real_crc)
- return(1);
- else
- {
- //printf("Bad EDC CRC: Calculated: %08x, Recorded: %08x\n", real_crc, expected_crc);
- return(0);
- }
-}
-
-/***
- *** A very simple L-EC error correction.
- ***
- * Perform just one pass over the Q and P vectors to see if everything
- * is okay respectively correct minor errors. This is pretty much the
- * same stuff the drive is supposed to do in the final L-EC stage.
- */
-
-static int simple_lec(unsigned char *frame)
-{
- unsigned char byte_state[2352];
- unsigned char p_vector[P_VECTOR_SIZE];
- unsigned char q_vector[Q_VECTOR_SIZE];
- unsigned char p_state[P_VECTOR_SIZE];
- int erasures[Q_VECTOR_SIZE], erasure_count;
- int ignore[2];
- int p_failures, q_failures;
- int p_corrected, q_corrected;
- int p,q;
-
- /* Setup */
-
- memset(byte_state, 0, 2352);
-
- p_failures = q_failures = 0;
- p_corrected = q_corrected = 0;
-
- /* Perform Q-Parity error correction */
-
- for(q=0; q 2)
- { GetPVector(byte_state, p_state, p);
- erasure_count = 0;
-
- for(i=0; i 0 && erasure_count <= 2)
- { GetPVector(frame, p_vector, p);
- err = DecodePQ(rt, p_vector, P_PADDING, erasures, erasure_count);
- }
- }
-
- /* See what we've got */
-
- if(err < 0) /* Uncorrectable. */
- { p_failures++;
- }
- else /* Correctable. */
- { if(err == 1 || err == 2) /* Store back corrected vector */
- { SetPVector(frame, p_vector, p);
- p_corrected++;
- }
- }
- }
-
- /* Sum up */
-
- if(q_failures || p_failures || q_corrected || p_corrected)
- {
- return 1;
- }
-
- return 0;
-}
-
-/***
- *** Validate CD raw sector
- ***/
-
-int ValidateRawSector(unsigned char *frame, bool xaMode)
-{
- int lec_did_sth = FALSE_0;
-
- /* Do simple L-EC.
- It seems that drives stop their internal L-EC as soon as the
- EDC is okay, so we may see uncorrected errors in the parity bytes.
- Since we are also interested in the user data only and doing the
- L-EC is expensive, we skip our L-EC as well when the EDC is fine. */
-
- if(!CheckEDC(frame, xaMode))
- {
- lec_did_sth = simple_lec(frame);
- }
- /* Test internal sector checksum again */
-
- if(!CheckEDC(frame, xaMode))
- {
- /* EDC failure in RAW sector */
- return FALSE_0;
- }
-
- return TRUE_1;
-}
-
diff --git a/psx/octoshock/psx/cdc.cpp b/psx/octoshock/psx/cdc.cpp
index bf1b1838fd..eb2a287f42 100644
--- a/psx/octoshock/psx/cdc.cpp
+++ b/psx/octoshock/psx/cdc.cpp
@@ -51,8 +51,6 @@
#include "psx.h"
#include "cdc.h"
#include "spu.h"
-#include "cdrom/CDUtility.h"
-#include "cdrom/cdromif.h"
#include "endian.h"
using namespace CDUtility;
@@ -176,7 +174,7 @@ void PS_CDC::SoftReset(void)
SectorPipe_Pos = SectorPipe_In = 0;
SectorsRead = 0;
- memset(SubQBuf, 0, sizeof(SubQBuf));
+ //memset(SubQBuf, 0, sizeof(SubQBuf));
memset(SubQBuf_Safe, 0, sizeof(SubQBuf_Safe));
SubQChecksumOK = false;
@@ -266,7 +264,7 @@ SYNCFUNC(PS_CDC)
NSS(SectorPipe_Pos);
NSS(SectorPipe_In);
- NSS(SubQBuf);
+ //NSS(SubQBuf);
NSS(SubQBuf_Safe);
NSS(SubQChecksumOK);
@@ -425,12 +423,12 @@ bool PS_CDC::DecodeSubQ(uint8 *subpw)
if((tmp_q[0] & 0xF) == 1)
{
- memcpy(SubQBuf, tmp_q, 0xC);
+ memcpy(SubQBuf_Safe, tmp_q, 0xC);
SubQChecksumOK = subq_check_checksum(tmp_q);
if(SubQChecksumOK)
{
- memcpy(SubQBuf_Safe, tmp_q, 0xC);
+ // memcpy(SubQBuf_Safe, tmp_q, 0xC);
return(true);
}
}
@@ -860,18 +858,17 @@ void PS_CDC::HandlePlayRead(void)
{
PSX_WARNING("[CDC] In leadout area: %u", CurSector);
- //
- // Synthesis is a bit of a kludge... :/
- //
- synth_leadout_sector_lba(0x02, toc, CurSector, read_buf);
- DecodeSubQ(read_buf + 2352);
+ // " Synthesis is a bit of a kludge " but we've taken it out of here
+ //synth_leadout_sector_lba(0x02, toc, CurSector, read_buf);
+ Cur_disc->ReadLBA2448(CurSector,read_buf); // FIXME: error out on error.
}
else
{
Cur_disc->ReadLBA2448(CurSector,read_buf); // FIXME: error out on error.
- DecodeSubQ(read_buf + 2352);
}
+ DecodeSubQ(read_buf + 2352);
+
if(SubQBuf_Safe[1] == 0xAA && (DriveStatus == DS_PLAYING || (!(SubQBuf_Safe[0] & 0x40) && (Mode & MODE_CDDA))))
{
@@ -975,10 +972,12 @@ void PS_CDC::HandlePlayRead(void)
// maybe if(!(Mode & 0x30)) too?
if(!(buf[12 + 6] & 0x20))
{
+ #ifdef LEC_CHECK
if(!edc_lec_check_and_correct(buf, true))
{
printf("Bad sector? - %d", CurSector);
}
+ #endif
}
if(!(Mode & 0x30) && (buf[12 + 6] & 0x20))
diff --git a/psx/octoshock/psx/cdc.h b/psx/octoshock/psx/cdc.h
index 511aa89073..7ab715b2ae 100644
--- a/psx/octoshock/psx/cdc.h
+++ b/psx/octoshock/psx/cdc.h
@@ -92,7 +92,7 @@ class PS_CDC
uint8 SectorPipe_Pos;
uint8 SectorPipe_In;
- uint8 SubQBuf[0xC];
+ //uint8 SubQBuf[0xC];
uint8 SubQBuf_Safe[0xC];
bool SubQChecksumOK;
diff --git a/psx/octoshock/psx/psx.cpp b/psx/octoshock/psx/psx.cpp
index c5e6c1d69e..0018535642 100644
--- a/psx/octoshock/psx/psx.cpp
+++ b/psx/octoshock/psx/psx.cpp
@@ -22,8 +22,8 @@
#include "timer.h"
#include "sio.h"
#include "cdc.h"
+#include "Stream.h"
#include "spu.h"
-#include "cdrom/cdromif.h"
#include "error.h"
#include "endian.h"
#include "emuware/EW_state.h"
@@ -2270,6 +2270,8 @@ s32 ShockDiscRef::InternalReadLBA2448(s32 lba, void* dst2448, bool needSubcode)
if(needSubcode && mSuppliesDeinterleavedSubcode)
{
+ //presently, CDC consumes deinterleaved subcode.
+ //perhaps this could be optimized in the future
u8 tmp[96];
CDUtility::subpw_interleave((u8*)dst2448+2352,tmp);
memcpy((u8*)dst2448+2352,tmp,96);
diff --git a/psx/octoshock/psx/psx.h b/psx/octoshock/psx/psx.h
index f8d79d000d..3e404e8454 100644
--- a/psx/octoshock/psx/psx.h
+++ b/psx/octoshock/psx/psx.h
@@ -1,7 +1,6 @@
#pragma once
#include "emuware/emuware.h"
-#include "cdrom/cdromif.h"
#include "video/surface.h"
#include "masmem.h"
#include "endian.h"