The JUCE cross-platform C++ framework, with DISTRHO/KXStudio specific changes
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

2450 lines
74KB

  1. /*
  2. ==============================================================================
  3. This file is part of the JUCE library - "Jules' Utility Class Extensions"
  4. Copyright 2004-7 by Raw Material Software ltd.
  5. ------------------------------------------------------------------------------
  6. JUCE can be redistributed and/or modified under the terms of the
  7. GNU General Public License, as published by the Free Software Foundation;
  8. either version 2 of the License, or (at your option) any later version.
  9. JUCE is distributed in the hope that it will be useful,
  10. but WITHOUT ANY WARRANTY; without even the implied warranty of
  11. MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  12. GNU General Public License for more details.
  13. You should have received a copy of the GNU General Public License
  14. along with JUCE; if not, visit www.gnu.org/licenses or write to the
  15. Free Software Foundation, Inc., 59 Temple Place, Suite 330,
  16. Boston, MA 02111-1307 USA
  17. ------------------------------------------------------------------------------
  18. If you'd like to release a closed-source product which uses JUCE, commercial
  19. licenses are also available: visit www.rawmaterialsoftware.com/juce for
  20. more information.
  21. ==============================================================================
  22. */
  23. #ifdef _MSC_VER
  24. #pragma warning (disable: 4514)
  25. #pragma warning (push)
  26. #endif
  27. #include "win32_headers.h"
  28. #include <stddef.h>
  29. #include "../../../juce_Config.h"
  30. #if JUCE_USE_CDBURNER
  31. // you'll need the Platform SDK for these headers - if you don't have it and don't
  32. // need to use CD-burning, then you might just want to disable the JUCE_USE_CDBURNER
  33. // flag in juce_Config.h to avoid these includes.
  34. #include <imapi.h>
  35. #include <imapierror.h>
  36. #endif
  37. #include "../../../src/juce_core/basics/juce_StandardHeader.h"
  38. BEGIN_JUCE_NAMESPACE
  39. #include "../../../src/juce_appframework/audio/audio_file_formats/juce_AudioCDReader.h"
  40. #include "../../../src/juce_appframework/audio/audio_file_formats/juce_AudioCDBurner.h"
  41. #include "../../../src/juce_appframework/events/juce_Timer.h"
  42. #include "../../../src/juce_appframework/application/juce_DeletedAtShutdown.h"
  43. #include "../../../src/juce_appframework/audio/dsp/juce_AudioDataConverters.h"
  44. #ifdef _MSC_VER
  45. #pragma warning (pop)
  46. #endif
  47. //***************************************************************************
  48. // %%% TARGET STATUS VALUES %%%
  49. //***************************************************************************
  50. #define STATUS_GOOD 0x00 // Status Good
  51. #define STATUS_CHKCOND 0x02 // Check Condition
  52. #define STATUS_CONDMET 0x04 // Condition Met
  53. #define STATUS_BUSY 0x08 // Busy
  54. #define STATUS_INTERM 0x10 // Intermediate
  55. #define STATUS_INTCDMET 0x14 // Intermediate-condition met
  56. #define STATUS_RESCONF 0x18 // Reservation conflict
  57. #define STATUS_COMTERM 0x22 // Command Terminated
  58. #define STATUS_QFULL 0x28 // Queue full
  59. //***************************************************************************
  60. // %%% SCSI MISCELLANEOUS EQUATES %%%
  61. //***************************************************************************
  62. #define MAXLUN 7 // Maximum Logical Unit Id
  63. #define MAXTARG 7 // Maximum Target Id
  64. #define MAX_SCSI_LUNS 64 // Maximum Number of SCSI LUNs
  65. #define MAX_NUM_HA 8 // Maximum Number of SCSI HA's
  66. //***************************************************************************
  67. // %%% Commands for all Device Types %%%
  68. //***************************************************************************
  69. #define SCSI_CHANGE_DEF 0x40 // Change Definition (Optional)
  70. #define SCSI_COMPARE 0x39 // Compare (O)
  71. #define SCSI_COPY 0x18 // Copy (O)
  72. #define SCSI_COP_VERIFY 0x3A // Copy and Verify (O)
  73. #define SCSI_INQUIRY 0x12 // Inquiry (MANDATORY)
  74. #define SCSI_LOG_SELECT 0x4C // Log Select (O)
  75. #define SCSI_LOG_SENSE 0x4D // Log Sense (O)
  76. #define SCSI_MODE_SEL6 0x15 // Mode Select 6-byte (Device Specific)
  77. #define SCSI_MODE_SEL10 0x55 // Mode Select 10-byte (Device Specific)
  78. #define SCSI_MODE_SEN6 0x1A // Mode Sense 6-byte (Device Specific)
  79. #define SCSI_MODE_SEN10 0x5A // Mode Sense 10-byte (Device Specific)
  80. #define SCSI_READ_BUFF 0x3C // Read Buffer (O)
  81. #define SCSI_REQ_SENSE 0x03 // Request Sense (MANDATORY)
  82. #define SCSI_SEND_DIAG 0x1D // Send Diagnostic (O)
  83. #define SCSI_TST_U_RDY 0x00 // Test Unit Ready (MANDATORY)
  84. #define SCSI_WRITE_BUFF 0x3B // Write Buffer (O)
  85. //***************************************************************************
  86. // %%% Commands Unique to Direct Access Devices %%%
  87. //***************************************************************************
  88. #define SCSI_COMPARE 0x39 // Compare (O)
  89. #define SCSI_FORMAT 0x04 // Format Unit (MANDATORY)
  90. #define SCSI_LCK_UN_CAC 0x36 // Lock Unlock Cache (O)
  91. #define SCSI_PREFETCH 0x34 // Prefetch (O)
  92. #define SCSI_MED_REMOVL 0x1E // Prevent/Allow medium Removal (O)
  93. #define SCSI_READ6 0x08 // Read 6-byte (MANDATORY)
  94. #define SCSI_READ10 0x28 // Read 10-byte (MANDATORY)
  95. #define SCSI_RD_CAPAC 0x25 // Read Capacity (MANDATORY)
  96. #define SCSI_RD_DEFECT 0x37 // Read Defect Data (O)
  97. #define SCSI_READ_LONG 0x3E // Read Long (O)
  98. #define SCSI_REASS_BLK 0x07 // Reassign Blocks (O)
  99. #define SCSI_RCV_DIAG 0x1C // Receive Diagnostic Results (O)
  100. #define SCSI_RELEASE 0x17 // Release Unit (MANDATORY)
  101. #define SCSI_REZERO 0x01 // Rezero Unit (O)
  102. #define SCSI_SRCH_DAT_E 0x31 // Search Data Equal (O)
  103. #define SCSI_SRCH_DAT_H 0x30 // Search Data High (O)
  104. #define SCSI_SRCH_DAT_L 0x32 // Search Data Low (O)
  105. #define SCSI_SEEK6 0x0B // Seek 6-Byte (O)
  106. #define SCSI_SEEK10 0x2B // Seek 10-Byte (O)
  107. #define SCSI_SEND_DIAG 0x1D // Send Diagnostics (MANDATORY)
  108. #define SCSI_SET_LIMIT 0x33 // Set Limits (O)
  109. #define SCSI_START_STP 0x1B // Start/Stop Unit (O)
  110. #define SCSI_SYNC_CACHE 0x35 // Synchronize Cache (O)
  111. #define SCSI_VERIFY 0x2F // Verify (O)
  112. #define SCSI_WRITE6 0x0A // Write 6-Byte (MANDATORY)
  113. #define SCSI_WRITE10 0x2A // Write 10-Byte (MANDATORY)
  114. #define SCSI_WRT_VERIFY 0x2E // Write and Verify (O)
  115. #define SCSI_WRITE_LONG 0x3F // Write Long (O)
  116. #define SCSI_WRITE_SAME 0x41 // Write Same (O)
  117. //***************************************************************************
  118. // %%% Commands Unique to Sequential Access Devices %%%
  119. //***************************************************************************
  120. #define SCSI_ERASE 0x19 // Erase (MANDATORY)
  121. #define SCSI_LOAD_UN 0x1b // Load/Unload (O)
  122. #define SCSI_LOCATE 0x2B // Locate (O)
  123. #define SCSI_RD_BLK_LIM 0x05 // Read Block Limits (MANDATORY)
  124. #define SCSI_READ_POS 0x34 // Read Position (O)
  125. #define SCSI_READ_REV 0x0F // Read Reverse (O)
  126. #define SCSI_REC_BF_DAT 0x14 // Recover Buffer Data (O)
  127. #define SCSI_RESERVE 0x16 // Reserve Unit (MANDATORY)
  128. #define SCSI_REWIND 0x01 // Rewind (MANDATORY)
  129. #define SCSI_SPACE 0x11 // Space (MANDATORY)
  130. #define SCSI_VERIFY_T 0x13 // Verify (Tape) (O)
  131. #define SCSI_WRT_FILE 0x10 // Write Filemarks (MANDATORY)
  132. //***************************************************************************
  133. // %%% Commands Unique to Printer Devices %%%
  134. //***************************************************************************
  135. #define SCSI_PRINT 0x0A // Print (MANDATORY)
  136. #define SCSI_SLEW_PNT 0x0B // Slew and Print (O)
  137. #define SCSI_STOP_PNT 0x1B // Stop Print (O)
  138. #define SCSI_SYNC_BUFF 0x10 // Synchronize Buffer (O)
  139. //***************************************************************************
  140. // %%% Commands Unique to Processor Devices %%%
  141. //***************************************************************************
  142. #define SCSI_RECEIVE 0x08 // Receive (O)
  143. #define SCSI_SEND 0x0A // Send (O)
  144. //***************************************************************************
  145. // %%% Commands Unique to Write-Once Devices %%%
  146. //***************************************************************************
  147. #define SCSI_MEDIUM_SCN 0x38 // Medium Scan (O)
  148. #define SCSI_SRCHDATE10 0x31 // Search Data Equal 10-Byte (O)
  149. #define SCSI_SRCHDATE12 0xB1 // Search Data Equal 12-Byte (O)
  150. #define SCSI_SRCHDATH10 0x30 // Search Data High 10-Byte (O)
  151. #define SCSI_SRCHDATH12 0xB0 // Search Data High 12-Byte (O)
  152. #define SCSI_SRCHDATL10 0x32 // Search Data Low 10-Byte (O)
  153. #define SCSI_SRCHDATL12 0xB2 // Search Data Low 12-Byte (O)
  154. #define SCSI_SET_LIM_10 0x33 // Set Limits 10-Byte (O)
  155. #define SCSI_SET_LIM_12 0xB3 // Set Limits 10-Byte (O)
  156. #define SCSI_VERIFY10 0x2F // Verify 10-Byte (O)
  157. #define SCSI_VERIFY12 0xAF // Verify 12-Byte (O)
  158. #define SCSI_WRITE12 0xAA // Write 12-Byte (O)
  159. #define SCSI_WRT_VER10 0x2E // Write and Verify 10-Byte (O)
  160. #define SCSI_WRT_VER12 0xAE // Write and Verify 12-Byte (O)
  161. //***************************************************************************
  162. // %%% Commands Unique to CD-ROM Devices %%%
  163. //***************************************************************************
  164. #define SCSI_PLAYAUD_10 0x45 // Play Audio 10-Byte (O)
  165. #define SCSI_PLAYAUD_12 0xA5 // Play Audio 12-Byte 12-Byte (O)
  166. #define SCSI_PLAYAUDMSF 0x47 // Play Audio MSF (O)
  167. #define SCSI_PLAYA_TKIN 0x48 // Play Audio Track/Index (O)
  168. #define SCSI_PLYTKREL10 0x49 // Play Track Relative 10-Byte (O)
  169. #define SCSI_PLYTKREL12 0xA9 // Play Track Relative 12-Byte (O)
  170. #define SCSI_READCDCAP 0x25 // Read CD-ROM Capacity (MANDATORY)
  171. #define SCSI_READHEADER 0x44 // Read Header (O)
  172. #define SCSI_SUBCHANNEL 0x42 // Read Subchannel (O)
  173. #define SCSI_READ_TOC 0x43 // Read TOC (O)
  174. //***************************************************************************
  175. // %%% Commands Unique to Scanner Devices %%%
  176. //***************************************************************************
  177. #define SCSI_GETDBSTAT 0x34 // Get Data Buffer Status (O)
  178. #define SCSI_GETWINDOW 0x25 // Get Window (O)
  179. #define SCSI_OBJECTPOS 0x31 // Object Postion (O)
  180. #define SCSI_SCAN 0x1B // Scan (O)
  181. #define SCSI_SETWINDOW 0x24 // Set Window (MANDATORY)
  182. //***************************************************************************
  183. // %%% Commands Unique to Optical Memory Devices %%%
  184. //***************************************************************************
  185. #define SCSI_UpdateBlk 0x3D // Update Block (O)
  186. //***************************************************************************
  187. // %%% Commands Unique to Medium Changer Devices %%%
  188. //***************************************************************************
  189. #define SCSI_EXCHMEDIUM 0xA6 // Exchange Medium (O)
  190. #define SCSI_INITELSTAT 0x07 // Initialize Element Status (O)
  191. #define SCSI_POSTOELEM 0x2B // Position to Element (O)
  192. #define SCSI_REQ_VE_ADD 0xB5 // Request Volume Element Address (O)
  193. #define SCSI_SENDVOLTAG 0xB6 // Send Volume Tag (O)
  194. //***************************************************************************
  195. // %%% Commands Unique to Communication Devices %%%
  196. //***************************************************************************
  197. #define SCSI_GET_MSG_6 0x08 // Get Message 6-Byte (MANDATORY)
  198. #define SCSI_GET_MSG_10 0x28 // Get Message 10-Byte (O)
  199. #define SCSI_GET_MSG_12 0xA8 // Get Message 12-Byte (O)
  200. #define SCSI_SND_MSG_6 0x0A // Send Message 6-Byte (MANDATORY)
  201. #define SCSI_SND_MSG_10 0x2A // Send Message 10-Byte (O)
  202. #define SCSI_SND_MSG_12 0xAA // Send Message 12-Byte (O)
  203. //***************************************************************************
  204. // %%% Request Sense Data Format %%%
  205. //***************************************************************************
  206. typedef struct {
  207. BYTE ErrorCode; // Error Code (70H or 71H)
  208. BYTE SegmentNum; // Number of current segment descriptor
  209. BYTE SenseKey; // Sense Key(See bit definitions too)
  210. BYTE InfoByte0; // Information MSB
  211. BYTE InfoByte1; // Information MID
  212. BYTE InfoByte2; // Information MID
  213. BYTE InfoByte3; // Information LSB
  214. BYTE AddSenLen; // Additional Sense Length
  215. BYTE ComSpecInf0; // Command Specific Information MSB
  216. BYTE ComSpecInf1; // Command Specific Information MID
  217. BYTE ComSpecInf2; // Command Specific Information MID
  218. BYTE ComSpecInf3; // Command Specific Information LSB
  219. BYTE AddSenseCode; // Additional Sense Code
  220. BYTE AddSenQual; // Additional Sense Code Qualifier
  221. BYTE FieldRepUCode; // Field Replaceable Unit Code
  222. BYTE SenKeySpec15; // Sense Key Specific 15th byte
  223. BYTE SenKeySpec16; // Sense Key Specific 16th byte
  224. BYTE SenKeySpec17; // Sense Key Specific 17th byte
  225. BYTE AddSenseBytes; // Additional Sense Bytes
  226. } SENSE_DATA_FMT;
  227. //***************************************************************************
  228. // %%% REQUEST SENSE ERROR CODE %%%
  229. //***************************************************************************
  230. #define SERROR_CURRENT 0x70 // Current Errors
  231. #define SERROR_DEFERED 0x71 // Deferred Errors
  232. //***************************************************************************
  233. // %%% REQUEST SENSE BIT DEFINITIONS %%%
  234. //***************************************************************************
  235. #define SENSE_VALID 0x80 // Byte 0 Bit 7
  236. #define SENSE_FILEMRK 0x80 // Byte 2 Bit 7
  237. #define SENSE_EOM 0x40 // Byte 2 Bit 6
  238. #define SENSE_ILI 0x20 // Byte 2 Bit 5
  239. //***************************************************************************
  240. // %%% REQUEST SENSE SENSE KEY DEFINITIONS %%%
  241. //***************************************************************************
  242. #define KEY_NOSENSE 0x00 // No Sense
  243. #define KEY_RECERROR 0x01 // Recovered Error
  244. #define KEY_NOTREADY 0x02 // Not Ready
  245. #define KEY_MEDIUMERR 0x03 // Medium Error
  246. #define KEY_HARDERROR 0x04 // Hardware Error
  247. #define KEY_ILLGLREQ 0x05 // Illegal Request
  248. #define KEY_UNITATT 0x06 // Unit Attention
  249. #define KEY_DATAPROT 0x07 // Data Protect
  250. #define KEY_BLANKCHK 0x08 // Blank Check
  251. #define KEY_VENDSPEC 0x09 // Vendor Specific
  252. #define KEY_COPYABORT 0x0A // Copy Abort
  253. #define KEY_EQUAL 0x0C // Equal (Search)
  254. #define KEY_VOLOVRFLW 0x0D // Volume Overflow
  255. #define KEY_MISCOMP 0x0E // Miscompare (Search)
  256. #define KEY_RESERVED 0x0F // Reserved
  257. //***************************************************************************
  258. // %%% PERIPHERAL DEVICE TYPE DEFINITIONS %%%
  259. //***************************************************************************
  260. #define DTYPE_DASD 0x00 // Disk Device
  261. #define DTYPE_SEQD 0x01 // Tape Device
  262. #define DTYPE_PRNT 0x02 // Printer
  263. #define DTYPE_PROC 0x03 // Processor
  264. #define DTYPE_WORM 0x04 // Write-once read-multiple
  265. #define DTYPE_CROM 0x05 // CD-ROM device
  266. #define DTYPE_SCAN 0x06 // Scanner device
  267. #define DTYPE_OPTI 0x07 // Optical memory device
  268. #define DTYPE_JUKE 0x08 // Medium Changer device
  269. #define DTYPE_COMM 0x09 // Communications device
  270. #define DTYPE_RESL 0x0A // Reserved (low)
  271. #define DTYPE_RESH 0x1E // Reserved (high)
  272. #define DTYPE_UNKNOWN 0x1F // Unknown or no device type
  273. //***************************************************************************
  274. // %%% ANSI APPROVED VERSION DEFINITIONS %%%
  275. //***************************************************************************
  276. #define ANSI_MAYBE 0x0 // Device may or may not be ANSI approved stand
  277. #define ANSI_SCSI1 0x1 // Device complies to ANSI X3.131-1986 (SCSI-1)
  278. #define ANSI_SCSI2 0x2 // Device complies to SCSI-2
  279. #define ANSI_RESLO 0x3 // Reserved (low)
  280. #define ANSI_RESHI 0x7 // Reserved (high)
  281. typedef struct
  282. {
  283. USHORT Length;
  284. UCHAR ScsiStatus;
  285. UCHAR PathId;
  286. UCHAR TargetId;
  287. UCHAR Lun;
  288. UCHAR CdbLength;
  289. UCHAR SenseInfoLength;
  290. UCHAR DataIn;
  291. ULONG DataTransferLength;
  292. ULONG TimeOutValue;
  293. ULONG DataBufferOffset;
  294. ULONG SenseInfoOffset;
  295. UCHAR Cdb[16];
  296. } SCSI_PASS_THROUGH, *PSCSI_PASS_THROUGH;
  297. typedef struct
  298. {
  299. USHORT Length;
  300. UCHAR ScsiStatus;
  301. UCHAR PathId;
  302. UCHAR TargetId;
  303. UCHAR Lun;
  304. UCHAR CdbLength;
  305. UCHAR SenseInfoLength;
  306. UCHAR DataIn;
  307. ULONG DataTransferLength;
  308. ULONG TimeOutValue;
  309. PVOID DataBuffer;
  310. ULONG SenseInfoOffset;
  311. UCHAR Cdb[16];
  312. } SCSI_PASS_THROUGH_DIRECT, *PSCSI_PASS_THROUGH_DIRECT;
  313. typedef struct
  314. {
  315. SCSI_PASS_THROUGH_DIRECT spt;
  316. ULONG Filler;
  317. UCHAR ucSenseBuf[32];
  318. } SCSI_PASS_THROUGH_DIRECT_WITH_BUFFER, *PSCSI_PASS_THROUGH_DIRECT_WITH_BUFFER;
  319. typedef struct
  320. {
  321. ULONG Length;
  322. UCHAR PortNumber;
  323. UCHAR PathId;
  324. UCHAR TargetId;
  325. UCHAR Lun;
  326. } SCSI_ADDRESS, *PSCSI_ADDRESS;
  327. #define METHOD_BUFFERED 0
  328. #define METHOD_IN_DIRECT 1
  329. #define METHOD_OUT_DIRECT 2
  330. #define METHOD_NEITHER 3
  331. #define FILE_ANY_ACCESS 0
  332. #ifndef FILE_READ_ACCESS
  333. #define FILE_READ_ACCESS (0x0001)
  334. #endif
  335. #ifndef FILE_WRITE_ACCESS
  336. #define FILE_WRITE_ACCESS (0x0002)
  337. #endif
  338. #define IOCTL_SCSI_BASE 0x00000004
  339. #define SCSI_IOCTL_DATA_OUT 0
  340. #define SCSI_IOCTL_DATA_IN 1
  341. #define SCSI_IOCTL_DATA_UNSPECIFIED 2
  342. #define CTL_CODE2( DevType, Function, Method, Access ) ( \
  343. ((DevType) << 16) | ((Access) << 14) | ((Function) << 2) | (Method) \
  344. )
  345. #define IOCTL_SCSI_PASS_THROUGH CTL_CODE2( IOCTL_SCSI_BASE, 0x0401, METHOD_BUFFERED, FILE_READ_ACCESS | FILE_WRITE_ACCESS )
  346. #define IOCTL_SCSI_GET_CAPABILITIES CTL_CODE2( IOCTL_SCSI_BASE, 0x0404, METHOD_BUFFERED, FILE_ANY_ACCESS)
  347. #define IOCTL_SCSI_PASS_THROUGH_DIRECT CTL_CODE2( IOCTL_SCSI_BASE, 0x0405, METHOD_BUFFERED, FILE_READ_ACCESS | FILE_WRITE_ACCESS )
  348. #define IOCTL_SCSI_GET_ADDRESS CTL_CODE2( IOCTL_SCSI_BASE, 0x0406, METHOD_BUFFERED, FILE_ANY_ACCESS )
  349. #define SENSE_LEN 14
  350. #define SRB_DIR_SCSI 0x00
  351. #define SRB_POSTING 0x01
  352. #define SRB_ENABLE_RESIDUAL_COUNT 0x04
  353. #define SRB_DIR_IN 0x08
  354. #define SRB_DIR_OUT 0x10
  355. #define SRB_EVENT_NOTIFY 0x40
  356. #define RESIDUAL_COUNT_SUPPORTED 0x02
  357. #define MAX_SRB_TIMEOUT 1080001u
  358. #define DEFAULT_SRB_TIMEOUT 1080001u
  359. #define SC_HA_INQUIRY 0x00
  360. #define SC_GET_DEV_TYPE 0x01
  361. #define SC_EXEC_SCSI_CMD 0x02
  362. #define SC_ABORT_SRB 0x03
  363. #define SC_RESET_DEV 0x04
  364. #define SC_SET_HA_PARMS 0x05
  365. #define SC_GET_DISK_INFO 0x06
  366. #define SC_RESCAN_SCSI_BUS 0x07
  367. #define SC_GETSET_TIMEOUTS 0x08
  368. #define SS_PENDING 0x00
  369. #define SS_COMP 0x01
  370. #define SS_ABORTED 0x02
  371. #define SS_ABORT_FAIL 0x03
  372. #define SS_ERR 0x04
  373. #define SS_INVALID_CMD 0x80
  374. #define SS_INVALID_HA 0x81
  375. #define SS_NO_DEVICE 0x82
  376. #define SS_INVALID_SRB 0xE0
  377. #define SS_OLD_MANAGER 0xE1
  378. #define SS_BUFFER_ALIGN 0xE1
  379. #define SS_ILLEGAL_MODE 0xE2
  380. #define SS_NO_ASPI 0xE3
  381. #define SS_FAILED_INIT 0xE4
  382. #define SS_ASPI_IS_BUSY 0xE5
  383. #define SS_BUFFER_TO_BIG 0xE6
  384. #define SS_BUFFER_TOO_BIG 0xE6
  385. #define SS_MISMATCHED_COMPONENTS 0xE7
  386. #define SS_NO_ADAPTERS 0xE8
  387. #define SS_INSUFFICIENT_RESOURCES 0xE9
  388. #define SS_ASPI_IS_SHUTDOWN 0xEA
  389. #define SS_BAD_INSTALL 0xEB
  390. #define HASTAT_OK 0x00
  391. #define HASTAT_SEL_TO 0x11
  392. #define HASTAT_DO_DU 0x12
  393. #define HASTAT_BUS_FREE 0x13
  394. #define HASTAT_PHASE_ERR 0x14
  395. #define HASTAT_TIMEOUT 0x09
  396. #define HASTAT_COMMAND_TIMEOUT 0x0B
  397. #define HASTAT_MESSAGE_REJECT 0x0D
  398. #define HASTAT_BUS_RESET 0x0E
  399. #define HASTAT_PARITY_ERROR 0x0F
  400. #define HASTAT_REQUEST_SENSE_FAILED 0x10
  401. #define PACKED
  402. #pragma pack(1)
  403. typedef struct
  404. {
  405. BYTE SRB_Cmd;
  406. BYTE SRB_Status;
  407. BYTE SRB_HaID;
  408. BYTE SRB_Flags;
  409. DWORD SRB_Hdr_Rsvd;
  410. BYTE HA_Count;
  411. BYTE HA_SCSI_ID;
  412. BYTE HA_ManagerId[16];
  413. BYTE HA_Identifier[16];
  414. BYTE HA_Unique[16];
  415. WORD HA_Rsvd1;
  416. BYTE pad[20];
  417. } PACKED SRB_HAInquiry, *PSRB_HAInquiry, FAR *LPSRB_HAInquiry;
  418. typedef struct
  419. {
  420. BYTE SRB_Cmd;
  421. BYTE SRB_Status;
  422. BYTE SRB_HaID;
  423. BYTE SRB_Flags;
  424. DWORD SRB_Hdr_Rsvd;
  425. BYTE SRB_Target;
  426. BYTE SRB_Lun;
  427. BYTE SRB_DeviceType;
  428. BYTE SRB_Rsvd1;
  429. BYTE pad[68];
  430. } PACKED SRB_GDEVBlock, *PSRB_GDEVBlock, FAR *LPSRB_GDEVBlock;
  431. typedef struct
  432. {
  433. BYTE SRB_Cmd;
  434. BYTE SRB_Status;
  435. BYTE SRB_HaID;
  436. BYTE SRB_Flags;
  437. DWORD SRB_Hdr_Rsvd;
  438. BYTE SRB_Target;
  439. BYTE SRB_Lun;
  440. WORD SRB_Rsvd1;
  441. DWORD SRB_BufLen;
  442. BYTE FAR *SRB_BufPointer;
  443. BYTE SRB_SenseLen;
  444. BYTE SRB_CDBLen;
  445. BYTE SRB_HaStat;
  446. BYTE SRB_TargStat;
  447. VOID FAR *SRB_PostProc;
  448. BYTE SRB_Rsvd2[20];
  449. BYTE CDBByte[16];
  450. BYTE SenseArea[SENSE_LEN+2];
  451. } PACKED SRB_ExecSCSICmd, *PSRB_ExecSCSICmd, FAR *LPSRB_ExecSCSICmd;
  452. typedef struct
  453. {
  454. BYTE SRB_Cmd;
  455. BYTE SRB_Status;
  456. BYTE SRB_HaId;
  457. BYTE SRB_Flags;
  458. DWORD SRB_Hdr_Rsvd;
  459. } PACKED SRB, *PSRB, FAR *LPSRB;
  460. #pragma pack()
  461. //==============================================================================
  462. struct CDDeviceInfo
  463. {
  464. char vendor[9];
  465. char productId[17];
  466. char rev[5];
  467. char vendorSpec[21];
  468. BYTE ha;
  469. BYTE tgt;
  470. BYTE lun;
  471. char scsiDriveLetter; // will be 0 if not using scsi
  472. };
  473. //==============================================================================
  474. class CDReadBuffer
  475. {
  476. public:
  477. int startFrame;
  478. int numFrames;
  479. int dataStartOffset;
  480. int dataLength;
  481. BYTE* buffer;
  482. int bufferSize;
  483. int index;
  484. bool wantsIndex;
  485. //==============================================================================
  486. CDReadBuffer (const int numberOfFrames)
  487. : startFrame (0),
  488. numFrames (0),
  489. dataStartOffset (0),
  490. dataLength (0),
  491. index (0),
  492. wantsIndex (false)
  493. {
  494. bufferSize = 2352 * numberOfFrames;
  495. buffer = (BYTE*) malloc (bufferSize);
  496. }
  497. ~CDReadBuffer()
  498. {
  499. free (buffer);
  500. }
  501. bool isZero() const
  502. {
  503. BYTE* p = buffer + dataStartOffset;
  504. for (int i = dataLength; --i >= 0;)
  505. if (*p++ != 0)
  506. return false;
  507. return true;
  508. }
  509. };
  510. class CDDeviceHandle;
  511. class CDController
  512. {
  513. public:
  514. CDController();
  515. virtual ~CDController();
  516. virtual bool read (CDReadBuffer* t) = 0;
  517. virtual void shutDown();
  518. bool readAudio (CDReadBuffer* t, CDReadBuffer* overlapBuffer = 0);
  519. int getLastIndex();
  520. public:
  521. bool initialised;
  522. CDDeviceHandle* deviceInfo;
  523. int framesToCheck, framesOverlap;
  524. void prepare (SRB_ExecSCSICmd& s);
  525. void perform (SRB_ExecSCSICmd& s);
  526. void setPaused (bool paused);
  527. };
  528. //==============================================================================
  529. #pragma pack(1)
  530. struct TOCTRACK
  531. {
  532. BYTE rsvd;
  533. BYTE ADR;
  534. BYTE trackNumber;
  535. BYTE rsvd2;
  536. BYTE addr[4];
  537. };
  538. struct TOC
  539. {
  540. WORD tocLen;
  541. BYTE firstTrack;
  542. BYTE lastTrack;
  543. TOCTRACK tracks[100];
  544. };
  545. #pragma pack()
  546. enum
  547. {
  548. READTYPE_ANY = 0,
  549. READTYPE_ATAPI1 = 1,
  550. READTYPE_ATAPI2 = 2,
  551. READTYPE_READ6 = 3,
  552. READTYPE_READ10 = 4,
  553. READTYPE_READ_D8 = 5,
  554. READTYPE_READ_D4 = 6,
  555. READTYPE_READ_D4_1 = 7,
  556. READTYPE_READ10_2 = 8
  557. };
  558. //==============================================================================
  559. class CDDeviceHandle
  560. {
  561. public:
  562. CDDeviceHandle (const CDDeviceInfo* const device)
  563. : scsiHandle (0),
  564. readType (READTYPE_ANY),
  565. controller (0)
  566. {
  567. memcpy (&info, device, sizeof (info));
  568. }
  569. ~CDDeviceHandle()
  570. {
  571. if (controller != 0)
  572. {
  573. controller->shutDown();
  574. delete controller;
  575. }
  576. if (scsiHandle != 0)
  577. CloseHandle (scsiHandle);
  578. }
  579. bool readTOC (TOC* lpToc, bool useMSF);
  580. bool readAudio (CDReadBuffer* buffer, CDReadBuffer* overlapBuffer = 0);
  581. void openDrawer (bool shouldBeOpen);
  582. CDDeviceInfo info;
  583. HANDLE scsiHandle;
  584. BYTE readType;
  585. private:
  586. CDController* controller;
  587. bool testController (const int readType,
  588. CDController* const newController,
  589. CDReadBuffer* const bufferToUse);
  590. };
  591. //==============================================================================
  592. DWORD (*fGetASPI32SupportInfo)(void);
  593. DWORD (*fSendASPI32Command)(LPSRB);
  594. //==============================================================================
  595. static HINSTANCE winAspiLib = 0;
  596. static bool usingScsi = false;
  597. static bool initialised = false;
  598. static bool InitialiseCDRipper()
  599. {
  600. if (! initialised)
  601. {
  602. initialised = true;
  603. OSVERSIONINFO info;
  604. info.dwOSVersionInfoSize = sizeof (info);
  605. GetVersionEx (&info);
  606. usingScsi = (info.dwPlatformId == VER_PLATFORM_WIN32_NT) && (info.dwMajorVersion > 4);
  607. if (! usingScsi)
  608. {
  609. fGetASPI32SupportInfo = 0;
  610. fSendASPI32Command = 0;
  611. winAspiLib = LoadLibrary (_T("WNASPI32.DLL"));
  612. if (winAspiLib != 0)
  613. {
  614. fGetASPI32SupportInfo = (DWORD(*)(void)) GetProcAddress (winAspiLib, "GetASPI32SupportInfo");
  615. fSendASPI32Command = (DWORD(*)(LPSRB)) GetProcAddress (winAspiLib, "SendASPI32Command");
  616. if (fGetASPI32SupportInfo == 0 || fSendASPI32Command == 0)
  617. return false;
  618. }
  619. else
  620. {
  621. usingScsi = true;
  622. }
  623. }
  624. }
  625. return true;
  626. }
  627. static void DeinitialiseCDRipper()
  628. {
  629. if (winAspiLib != 0)
  630. {
  631. fGetASPI32SupportInfo = 0;
  632. fSendASPI32Command = 0;
  633. FreeLibrary (winAspiLib);
  634. winAspiLib = 0;
  635. }
  636. initialised = false;
  637. }
  638. //==============================================================================
  639. static HANDLE CreateSCSIDeviceHandle (char driveLetter)
  640. {
  641. TCHAR devicePath[8];
  642. devicePath[0] = '\\';
  643. devicePath[1] = '\\';
  644. devicePath[2] = '.';
  645. devicePath[3] = '\\';
  646. devicePath[4] = driveLetter;
  647. devicePath[5] = ':';
  648. devicePath[6] = 0;
  649. OSVERSIONINFO info;
  650. info.dwOSVersionInfoSize = sizeof (info);
  651. GetVersionEx (&info);
  652. DWORD flags = GENERIC_READ;
  653. if ((info.dwPlatformId == VER_PLATFORM_WIN32_NT) && (info.dwMajorVersion > 4))
  654. flags = GENERIC_READ | GENERIC_WRITE;
  655. HANDLE h = CreateFile (devicePath, flags, FILE_SHARE_WRITE | FILE_SHARE_READ, 0, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, 0);
  656. if (h == INVALID_HANDLE_VALUE)
  657. {
  658. flags ^= GENERIC_WRITE;
  659. h = CreateFile (devicePath, flags, FILE_SHARE_WRITE | FILE_SHARE_READ, 0, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, 0);
  660. }
  661. return h;
  662. }
  663. static DWORD performScsiPassThroughCommand (const LPSRB_ExecSCSICmd srb,
  664. const char driveLetter,
  665. HANDLE& deviceHandle,
  666. const bool retryOnFailure = true)
  667. {
  668. SCSI_PASS_THROUGH_DIRECT_WITH_BUFFER s;
  669. zerostruct (s);
  670. s.spt.Length = sizeof (SCSI_PASS_THROUGH);
  671. s.spt.CdbLength = srb->SRB_CDBLen;
  672. s.spt.DataIn = (BYTE) ((srb->SRB_Flags & SRB_DIR_IN)
  673. ? SCSI_IOCTL_DATA_IN
  674. : ((srb->SRB_Flags & SRB_DIR_OUT)
  675. ? SCSI_IOCTL_DATA_OUT
  676. : SCSI_IOCTL_DATA_UNSPECIFIED));
  677. s.spt.DataTransferLength = srb->SRB_BufLen;
  678. s.spt.TimeOutValue = 5;
  679. s.spt.DataBuffer = srb->SRB_BufPointer;
  680. s.spt.SenseInfoOffset = offsetof (SCSI_PASS_THROUGH_DIRECT_WITH_BUFFER, ucSenseBuf);
  681. memcpy (s.spt.Cdb, srb->CDBByte, srb->SRB_CDBLen);
  682. srb->SRB_Status = SS_ERR;
  683. srb->SRB_TargStat = 0x0004;
  684. DWORD bytesReturned = 0;
  685. if (DeviceIoControl (deviceHandle, IOCTL_SCSI_PASS_THROUGH_DIRECT,
  686. &s, sizeof (s),
  687. &s, sizeof (s),
  688. &bytesReturned, 0) != 0)
  689. {
  690. srb->SRB_Status = SS_COMP;
  691. }
  692. else if (retryOnFailure)
  693. {
  694. const DWORD error = GetLastError();
  695. if ((error == ERROR_MEDIA_CHANGED) || (error == ERROR_INVALID_HANDLE))
  696. {
  697. if (error != ERROR_INVALID_HANDLE)
  698. CloseHandle (deviceHandle);
  699. deviceHandle = CreateSCSIDeviceHandle (driveLetter);
  700. return performScsiPassThroughCommand (srb, driveLetter, deviceHandle, false);
  701. }
  702. }
  703. return srb->SRB_Status;
  704. }
  705. //==============================================================================
  706. // Controller types..
  707. class ControllerType1 : public CDController
  708. {
  709. public:
  710. ControllerType1() {}
  711. ~ControllerType1() {}
  712. bool read (CDReadBuffer* rb)
  713. {
  714. if (rb->numFrames * 2352 > rb->bufferSize)
  715. return false;
  716. SRB_ExecSCSICmd s;
  717. prepare (s);
  718. s.SRB_Flags = SRB_DIR_IN | SRB_EVENT_NOTIFY;
  719. s.SRB_BufLen = rb->bufferSize;
  720. s.SRB_BufPointer = rb->buffer;
  721. s.SRB_CDBLen = 12;
  722. s.CDBByte[0] = 0xBE;
  723. s.CDBByte[3] = (BYTE)((rb->startFrame >> 16) & 0xFF);
  724. s.CDBByte[4] = (BYTE)((rb->startFrame >> 8) & 0xFF);
  725. s.CDBByte[5] = (BYTE)(rb->startFrame & 0xFF);
  726. s.CDBByte[8] = (BYTE)(rb->numFrames & 0xFF);
  727. s.CDBByte[9] = (BYTE)((deviceInfo->readType == READTYPE_ATAPI1) ? 0x10 : 0xF0);
  728. perform (s);
  729. if (s.SRB_Status != SS_COMP)
  730. return false;
  731. rb->dataLength = rb->numFrames * 2352;
  732. rb->dataStartOffset = 0;
  733. return true;
  734. }
  735. };
  736. //==============================================================================
  737. class ControllerType2 : public CDController
  738. {
  739. public:
  740. ControllerType2() {}
  741. ~ControllerType2() {}
  742. void shutDown()
  743. {
  744. if (initialised)
  745. {
  746. BYTE bufPointer[] = { 0, 0, 0, 8, 83, 0, 0, 0, 0, 0, 8, 0 };
  747. SRB_ExecSCSICmd s;
  748. prepare (s);
  749. s.SRB_Flags = SRB_EVENT_NOTIFY | SRB_ENABLE_RESIDUAL_COUNT;
  750. s.SRB_BufLen = 0x0C;
  751. s.SRB_BufPointer = bufPointer;
  752. s.SRB_CDBLen = 6;
  753. s.CDBByte[0] = 0x15;
  754. s.CDBByte[4] = 0x0C;
  755. perform (s);
  756. }
  757. }
  758. bool init()
  759. {
  760. SRB_ExecSCSICmd s;
  761. s.SRB_Status = SS_ERR;
  762. if (deviceInfo->readType == READTYPE_READ10_2)
  763. {
  764. BYTE bufPointer1[] = { 0, 0, 0, 8, 0, 0, 0, 0, 0, 0, 9, 48, 35, 6, 0, 0, 0, 0, 0, 128 };
  765. BYTE bufPointer2[] = { 0, 0, 0, 8, 0, 0, 0, 0, 0, 0, 9, 48, 1, 6, 32, 7, 0, 0, 0, 0 };
  766. for (int i = 0; i < 2; ++i)
  767. {
  768. prepare (s);
  769. s.SRB_Flags = SRB_EVENT_NOTIFY;
  770. s.SRB_BufLen = 0x14;
  771. s.SRB_BufPointer = (i == 0) ? bufPointer1 : bufPointer2;
  772. s.SRB_CDBLen = 6;
  773. s.CDBByte[0] = 0x15;
  774. s.CDBByte[1] = 0x10;
  775. s.CDBByte[4] = 0x14;
  776. perform (s);
  777. if (s.SRB_Status != SS_COMP)
  778. return false;
  779. }
  780. }
  781. else
  782. {
  783. BYTE bufPointer[] = { 0, 0, 0, 8, 0, 0, 0, 0, 0, 0, 9, 48 };
  784. prepare (s);
  785. s.SRB_Flags = SRB_EVENT_NOTIFY;
  786. s.SRB_BufLen = 0x0C;
  787. s.SRB_BufPointer = bufPointer;
  788. s.SRB_CDBLen = 6;
  789. s.CDBByte[0] = 0x15;
  790. s.CDBByte[4] = 0x0C;
  791. perform (s);
  792. }
  793. return s.SRB_Status == SS_COMP;
  794. }
  795. bool read (CDReadBuffer* rb)
  796. {
  797. if (rb->numFrames * 2352 > rb->bufferSize)
  798. return false;
  799. if (!initialised)
  800. {
  801. initialised = init();
  802. if (!initialised)
  803. return false;
  804. }
  805. SRB_ExecSCSICmd s;
  806. prepare (s);
  807. s.SRB_Flags = SRB_DIR_IN | SRB_EVENT_NOTIFY;
  808. s.SRB_BufLen = rb->bufferSize;
  809. s.SRB_BufPointer = rb->buffer;
  810. s.SRB_CDBLen = 10;
  811. s.CDBByte[0] = 0x28;
  812. s.CDBByte[1] = (BYTE)(deviceInfo->info.lun << 5);
  813. s.CDBByte[3] = (BYTE)((rb->startFrame >> 16) & 0xFF);
  814. s.CDBByte[4] = (BYTE)((rb->startFrame >> 8) & 0xFF);
  815. s.CDBByte[5] = (BYTE)(rb->startFrame & 0xFF);
  816. s.CDBByte[8] = (BYTE)(rb->numFrames & 0xFF);
  817. perform (s);
  818. if (s.SRB_Status != SS_COMP)
  819. return false;
  820. rb->dataLength = rb->numFrames * 2352;
  821. rb->dataStartOffset = 0;
  822. return true;
  823. }
  824. };
  825. //==============================================================================
  826. class ControllerType3 : public CDController
  827. {
  828. public:
  829. ControllerType3() {}
  830. ~ControllerType3() {}
  831. bool read (CDReadBuffer* rb)
  832. {
  833. if (rb->numFrames * 2352 > rb->bufferSize)
  834. return false;
  835. if (!initialised)
  836. {
  837. setPaused (false);
  838. initialised = true;
  839. }
  840. SRB_ExecSCSICmd s;
  841. prepare (s);
  842. s.SRB_Flags = SRB_DIR_IN | SRB_EVENT_NOTIFY;
  843. s.SRB_BufLen = rb->numFrames * 2352;
  844. s.SRB_BufPointer = rb->buffer;
  845. s.SRB_CDBLen = 12;
  846. s.CDBByte[0] = 0xD8;
  847. s.CDBByte[3] = (BYTE)((rb->startFrame >> 16) & 0xFF);
  848. s.CDBByte[4] = (BYTE)((rb->startFrame >> 8) & 0xFF);
  849. s.CDBByte[5] = (BYTE)(rb->startFrame & 0xFF);
  850. s.CDBByte[9] = (BYTE)(rb->numFrames & 0xFF);
  851. perform (s);
  852. if (s.SRB_Status != SS_COMP)
  853. return false;
  854. rb->dataLength = rb->numFrames * 2352;
  855. rb->dataStartOffset = 0;
  856. return true;
  857. }
  858. };
  859. //==============================================================================
  860. class ControllerType4 : public CDController
  861. {
  862. public:
  863. ControllerType4() {}
  864. ~ControllerType4() {}
  865. bool selectD4Mode()
  866. {
  867. BYTE bufPointer[12] = { 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 9, 48 };
  868. SRB_ExecSCSICmd s;
  869. prepare (s);
  870. s.SRB_Flags = SRB_EVENT_NOTIFY;
  871. s.SRB_CDBLen = 6;
  872. s.SRB_BufLen = 12;
  873. s.SRB_BufPointer = bufPointer;
  874. s.CDBByte[0] = 0x15;
  875. s.CDBByte[1] = 0x10;
  876. s.CDBByte[4] = 0x08;
  877. perform (s);
  878. return s.SRB_Status == SS_COMP;
  879. }
  880. bool read (CDReadBuffer* rb)
  881. {
  882. if (rb->numFrames * 2352 > rb->bufferSize)
  883. return false;
  884. if (!initialised)
  885. {
  886. setPaused (true);
  887. if (deviceInfo->readType == READTYPE_READ_D4_1)
  888. selectD4Mode();
  889. initialised = true;
  890. }
  891. SRB_ExecSCSICmd s;
  892. prepare (s);
  893. s.SRB_Flags = SRB_DIR_IN | SRB_EVENT_NOTIFY;
  894. s.SRB_BufLen = rb->bufferSize;
  895. s.SRB_BufPointer = rb->buffer;
  896. s.SRB_CDBLen = 10;
  897. s.CDBByte[0] = 0xD4;
  898. s.CDBByte[3] = (BYTE)((rb->startFrame >> 16) & 0xFF);
  899. s.CDBByte[4] = (BYTE)((rb->startFrame >> 8) & 0xFF);
  900. s.CDBByte[5] = (BYTE)(rb->startFrame & 0xFF);
  901. s.CDBByte[8] = (BYTE)(rb->numFrames & 0xFF);
  902. perform (s);
  903. if (s.SRB_Status != SS_COMP)
  904. return false;
  905. rb->dataLength = rb->numFrames * 2352;
  906. rb->dataStartOffset = 0;
  907. return true;
  908. }
  909. };
  910. //==============================================================================
  911. CDController::CDController() : initialised (false)
  912. {
  913. }
  914. CDController::~CDController()
  915. {
  916. }
  917. void CDController::prepare (SRB_ExecSCSICmd& s)
  918. {
  919. zerostruct (s);
  920. s.SRB_Cmd = SC_EXEC_SCSI_CMD;
  921. s.SRB_HaID = deviceInfo->info.ha;
  922. s.SRB_Target = deviceInfo->info.tgt;
  923. s.SRB_Lun = deviceInfo->info.lun;
  924. s.SRB_SenseLen = SENSE_LEN;
  925. }
  926. void CDController::perform (SRB_ExecSCSICmd& s)
  927. {
  928. HANDLE event = CreateEvent (0, TRUE, FALSE, 0);
  929. s.SRB_PostProc = (void*)event;
  930. ResetEvent (event);
  931. DWORD status = (usingScsi) ? performScsiPassThroughCommand ((LPSRB_ExecSCSICmd)&s,
  932. deviceInfo->info.scsiDriveLetter,
  933. deviceInfo->scsiHandle)
  934. : fSendASPI32Command ((LPSRB)&s);
  935. if (status == SS_PENDING)
  936. WaitForSingleObject (event, 4000);
  937. CloseHandle (event);
  938. }
  939. void CDController::setPaused (bool paused)
  940. {
  941. SRB_ExecSCSICmd s;
  942. prepare (s);
  943. s.SRB_Flags = SRB_EVENT_NOTIFY;
  944. s.SRB_CDBLen = 10;
  945. s.CDBByte[0] = 0x4B;
  946. s.CDBByte[8] = (BYTE) (paused ? 0 : 1);
  947. perform (s);
  948. }
  949. void CDController::shutDown()
  950. {
  951. }
  952. bool CDController::readAudio (CDReadBuffer* rb, CDReadBuffer* overlapBuffer)
  953. {
  954. if (overlapBuffer != 0)
  955. {
  956. const bool canDoJitter = (overlapBuffer->bufferSize >= 2352 * framesToCheck);
  957. const bool doJitter = canDoJitter && ! overlapBuffer->isZero();
  958. if (doJitter
  959. && overlapBuffer->startFrame > 0
  960. && overlapBuffer->numFrames > 0
  961. && overlapBuffer->dataLength > 0)
  962. {
  963. const int numFrames = rb->numFrames;
  964. if (overlapBuffer->startFrame == (rb->startFrame - framesToCheck))
  965. {
  966. rb->startFrame -= framesOverlap;
  967. if (framesToCheck < framesOverlap
  968. && numFrames + framesOverlap <= rb->bufferSize / 2352)
  969. rb->numFrames += framesOverlap;
  970. }
  971. else
  972. {
  973. overlapBuffer->dataLength = 0;
  974. overlapBuffer->startFrame = 0;
  975. overlapBuffer->numFrames = 0;
  976. }
  977. }
  978. if (! read (rb))
  979. return false;
  980. if (doJitter)
  981. {
  982. const int checkLen = framesToCheck * 2352;
  983. const int maxToCheck = rb->dataLength - checkLen;
  984. if (overlapBuffer->dataLength == 0 || overlapBuffer->isZero())
  985. return true;
  986. BYTE* const p = overlapBuffer->buffer + overlapBuffer->dataStartOffset;
  987. bool found = false;
  988. for (int i = 0; i < maxToCheck; ++i)
  989. {
  990. if (!memcmp (p, rb->buffer + i, checkLen))
  991. {
  992. i += checkLen;
  993. rb->dataStartOffset = i;
  994. rb->dataLength -= i;
  995. rb->startFrame = overlapBuffer->startFrame + framesToCheck;
  996. found = true;
  997. break;
  998. }
  999. }
  1000. rb->numFrames = rb->dataLength / 2352;
  1001. rb->dataLength = 2352 * rb->numFrames;
  1002. if (!found)
  1003. return false;
  1004. }
  1005. if (canDoJitter)
  1006. {
  1007. memcpy (overlapBuffer->buffer,
  1008. rb->buffer + rb->dataStartOffset + 2352 * (rb->numFrames - framesToCheck),
  1009. 2352 * framesToCheck);
  1010. overlapBuffer->startFrame = rb->startFrame + rb->numFrames - framesToCheck;
  1011. overlapBuffer->numFrames = framesToCheck;
  1012. overlapBuffer->dataLength = 2352 * framesToCheck;
  1013. overlapBuffer->dataStartOffset = 0;
  1014. }
  1015. else
  1016. {
  1017. overlapBuffer->startFrame = 0;
  1018. overlapBuffer->numFrames = 0;
  1019. overlapBuffer->dataLength = 0;
  1020. }
  1021. return true;
  1022. }
  1023. else
  1024. {
  1025. return read (rb);
  1026. }
  1027. }
  1028. int CDController::getLastIndex()
  1029. {
  1030. char qdata[100];
  1031. SRB_ExecSCSICmd s;
  1032. prepare (s);
  1033. s.SRB_Flags = SRB_DIR_IN | SRB_EVENT_NOTIFY;
  1034. s.SRB_BufLen = sizeof (qdata);
  1035. s.SRB_BufPointer = (BYTE*)qdata;
  1036. s.SRB_CDBLen = 12;
  1037. s.CDBByte[0] = 0x42;
  1038. s.CDBByte[1] = (BYTE)(deviceInfo->info.lun << 5);
  1039. s.CDBByte[2] = 64;
  1040. s.CDBByte[3] = 1; // get current position
  1041. s.CDBByte[7] = 0;
  1042. s.CDBByte[8] = (BYTE)sizeof (qdata);
  1043. perform (s);
  1044. if (s.SRB_Status == SS_COMP)
  1045. return qdata[7];
  1046. return 0;
  1047. }
  1048. //==============================================================================
  1049. bool CDDeviceHandle::readTOC (TOC* lpToc, bool useMSF)
  1050. {
  1051. HANDLE event = CreateEvent (0, TRUE, FALSE, 0);
  1052. SRB_ExecSCSICmd s;
  1053. zerostruct (s);
  1054. s.SRB_Cmd = SC_EXEC_SCSI_CMD;
  1055. s.SRB_HaID = info.ha;
  1056. s.SRB_Target = info.tgt;
  1057. s.SRB_Lun = info.lun;
  1058. s.SRB_Flags = SRB_DIR_IN | SRB_EVENT_NOTIFY;
  1059. s.SRB_BufLen = 0x324;
  1060. s.SRB_BufPointer = (BYTE*)lpToc;
  1061. s.SRB_SenseLen = 0x0E;
  1062. s.SRB_CDBLen = 0x0A;
  1063. s.SRB_PostProc = (void*)event;
  1064. s.CDBByte[0] = 0x43;
  1065. s.CDBByte[1] = (BYTE)(useMSF ? 0x02 : 0x00);
  1066. s.CDBByte[7] = 0x03;
  1067. s.CDBByte[8] = 0x24;
  1068. ResetEvent (event);
  1069. DWORD status = (usingScsi) ? performScsiPassThroughCommand ((LPSRB_ExecSCSICmd)&s, info.scsiDriveLetter, scsiHandle)
  1070. : fSendASPI32Command ((LPSRB)&s);
  1071. if (status == SS_PENDING)
  1072. WaitForSingleObject (event, 4000);
  1073. CloseHandle (event);
  1074. return (s.SRB_Status == SS_COMP);
  1075. }
  1076. bool CDDeviceHandle::readAudio (CDReadBuffer* const buffer,
  1077. CDReadBuffer* const overlapBuffer)
  1078. {
  1079. if (controller == 0)
  1080. {
  1081. testController (READTYPE_ATAPI2, new ControllerType1(), buffer)
  1082. || testController (READTYPE_ATAPI1, new ControllerType1(), buffer)
  1083. || testController (READTYPE_READ10_2, new ControllerType2(), buffer)
  1084. || testController (READTYPE_READ10, new ControllerType2(), buffer)
  1085. || testController (READTYPE_READ_D8, new ControllerType3(), buffer)
  1086. || testController (READTYPE_READ_D4, new ControllerType4(), buffer)
  1087. || testController (READTYPE_READ_D4_1, new ControllerType4(), buffer);
  1088. }
  1089. buffer->index = 0;
  1090. if ((controller != 0)
  1091. && controller->readAudio (buffer, overlapBuffer))
  1092. {
  1093. if (buffer->wantsIndex)
  1094. buffer->index = controller->getLastIndex();
  1095. return true;
  1096. }
  1097. return false;
  1098. }
  1099. void CDDeviceHandle::openDrawer (bool shouldBeOpen)
  1100. {
  1101. if (shouldBeOpen)
  1102. {
  1103. if (controller != 0)
  1104. {
  1105. controller->shutDown();
  1106. delete controller;
  1107. controller = 0;
  1108. }
  1109. if (scsiHandle != 0)
  1110. {
  1111. CloseHandle (scsiHandle);
  1112. scsiHandle = 0;
  1113. }
  1114. }
  1115. SRB_ExecSCSICmd s;
  1116. zerostruct (s);
  1117. s.SRB_Cmd = SC_EXEC_SCSI_CMD;
  1118. s.SRB_HaID = info.ha;
  1119. s.SRB_Target = info.tgt;
  1120. s.SRB_Lun = info.lun;
  1121. s.SRB_SenseLen = SENSE_LEN;
  1122. s.SRB_Flags = SRB_DIR_IN | SRB_EVENT_NOTIFY;
  1123. s.SRB_BufLen = 0;
  1124. s.SRB_BufPointer = 0;
  1125. s.SRB_CDBLen = 12;
  1126. s.CDBByte[0] = 0x1b;
  1127. s.CDBByte[1] = (BYTE)(info.lun << 5);
  1128. s.CDBByte[4] = (BYTE)((shouldBeOpen) ? 2 : 3);
  1129. HANDLE event = CreateEvent (0, TRUE, FALSE, 0);
  1130. s.SRB_PostProc = (void*)event;
  1131. ResetEvent (event);
  1132. DWORD status = (usingScsi) ? performScsiPassThroughCommand ((LPSRB_ExecSCSICmd)&s, info.scsiDriveLetter, scsiHandle)
  1133. : fSendASPI32Command ((LPSRB)&s);
  1134. if (status == SS_PENDING)
  1135. WaitForSingleObject (event, 4000);
  1136. CloseHandle (event);
  1137. }
  1138. bool CDDeviceHandle::testController (const int type,
  1139. CDController* const newController,
  1140. CDReadBuffer* const rb)
  1141. {
  1142. controller = newController;
  1143. readType = (BYTE)type;
  1144. controller->deviceInfo = this;
  1145. controller->framesToCheck = 1;
  1146. controller->framesOverlap = 3;
  1147. bool passed = false;
  1148. memset (rb->buffer, 0xcd, rb->bufferSize);
  1149. if (controller->read (rb))
  1150. {
  1151. passed = true;
  1152. int* p = (int*) (rb->buffer + rb->dataStartOffset);
  1153. int wrong = 0;
  1154. for (int i = rb->dataLength / 4; --i >= 0;)
  1155. {
  1156. if (*p++ == (int) 0xcdcdcdcd)
  1157. {
  1158. if (++wrong == 4)
  1159. {
  1160. passed = false;
  1161. break;
  1162. }
  1163. }
  1164. else
  1165. {
  1166. wrong = 0;
  1167. }
  1168. }
  1169. }
  1170. if (! passed)
  1171. {
  1172. controller->shutDown();
  1173. delete controller;
  1174. controller = 0;
  1175. }
  1176. return passed;
  1177. }
  1178. //==============================================================================
  1179. static void GetAspiDeviceInfo (CDDeviceInfo* dev, BYTE ha, BYTE tgt, BYTE lun)
  1180. {
  1181. HANDLE event = CreateEvent (0, TRUE, FALSE, 0);
  1182. const int bufSize = 128;
  1183. BYTE buffer[bufSize];
  1184. zeromem (buffer, bufSize);
  1185. SRB_ExecSCSICmd s;
  1186. zerostruct (s);
  1187. s.SRB_Cmd = SC_EXEC_SCSI_CMD;
  1188. s.SRB_HaID = ha;
  1189. s.SRB_Target = tgt;
  1190. s.SRB_Lun = lun;
  1191. s.SRB_Flags = SRB_DIR_IN | SRB_EVENT_NOTIFY;
  1192. s.SRB_BufLen = bufSize;
  1193. s.SRB_BufPointer = buffer;
  1194. s.SRB_SenseLen = SENSE_LEN;
  1195. s.SRB_CDBLen = 6;
  1196. s.SRB_PostProc = (void*)event;
  1197. s.CDBByte[0] = SCSI_INQUIRY;
  1198. s.CDBByte[4] = 100;
  1199. ResetEvent (event);
  1200. if (fSendASPI32Command ((LPSRB)&s) == SS_PENDING)
  1201. WaitForSingleObject (event, 4000);
  1202. CloseHandle (event);
  1203. if (s.SRB_Status == SS_COMP)
  1204. {
  1205. memcpy (dev->vendor, &buffer[8], 8);
  1206. memcpy (dev->productId, &buffer[16], 16);
  1207. memcpy (dev->rev, &buffer[32], 4);
  1208. memcpy (dev->vendorSpec, &buffer[36], 20);
  1209. }
  1210. }
  1211. static int FindCDDevices (CDDeviceInfo* const list,
  1212. int maxItems)
  1213. {
  1214. int count = 0;
  1215. if (usingScsi)
  1216. {
  1217. for (char driveLetter = 'b'; driveLetter <= 'z'; ++driveLetter)
  1218. {
  1219. TCHAR drivePath[8];
  1220. drivePath[0] = driveLetter;
  1221. drivePath[1] = ':';
  1222. drivePath[2] = '\\';
  1223. drivePath[3] = 0;
  1224. if (GetDriveType (drivePath) == DRIVE_CDROM)
  1225. {
  1226. HANDLE h = CreateSCSIDeviceHandle (driveLetter);
  1227. if (h != INVALID_HANDLE_VALUE)
  1228. {
  1229. BYTE buffer[100], passThroughStruct[1024];
  1230. zeromem (buffer, sizeof (buffer));
  1231. zeromem (passThroughStruct, sizeof (passThroughStruct));
  1232. PSCSI_PASS_THROUGH_DIRECT_WITH_BUFFER p = (PSCSI_PASS_THROUGH_DIRECT_WITH_BUFFER)passThroughStruct;
  1233. p->spt.Length = sizeof (SCSI_PASS_THROUGH);
  1234. p->spt.CdbLength = 6;
  1235. p->spt.SenseInfoLength = 24;
  1236. p->spt.DataIn = SCSI_IOCTL_DATA_IN;
  1237. p->spt.DataTransferLength = 100;
  1238. p->spt.TimeOutValue = 2;
  1239. p->spt.DataBuffer = buffer;
  1240. p->spt.SenseInfoOffset = offsetof (SCSI_PASS_THROUGH_DIRECT_WITH_BUFFER, ucSenseBuf);
  1241. p->spt.Cdb[0] = 0x12;
  1242. p->spt.Cdb[4] = 100;
  1243. DWORD bytesReturned = 0;
  1244. if (DeviceIoControl (h, IOCTL_SCSI_PASS_THROUGH_DIRECT,
  1245. p, sizeof (SCSI_PASS_THROUGH_DIRECT_WITH_BUFFER),
  1246. p, sizeof (SCSI_PASS_THROUGH_DIRECT_WITH_BUFFER),
  1247. &bytesReturned, 0) != 0)
  1248. {
  1249. zeromem (&list[count], sizeof (CDDeviceInfo));
  1250. list[count].scsiDriveLetter = driveLetter;
  1251. memcpy (list[count].vendor, &buffer[8], 8);
  1252. memcpy (list[count].productId, &buffer[16], 16);
  1253. memcpy (list[count].rev, &buffer[32], 4);
  1254. memcpy (list[count].vendorSpec, &buffer[36], 20);
  1255. zeromem (passThroughStruct, sizeof (passThroughStruct));
  1256. PSCSI_ADDRESS scsiAddr = (PSCSI_ADDRESS)passThroughStruct;
  1257. scsiAddr->Length = sizeof (SCSI_ADDRESS);
  1258. if (DeviceIoControl (h, IOCTL_SCSI_GET_ADDRESS,
  1259. 0, 0, scsiAddr, sizeof (SCSI_ADDRESS),
  1260. &bytesReturned, 0) != 0)
  1261. {
  1262. list[count].ha = scsiAddr->PortNumber;
  1263. list[count].tgt = scsiAddr->TargetId;
  1264. list[count].lun = scsiAddr->Lun;
  1265. ++count;
  1266. }
  1267. }
  1268. CloseHandle (h);
  1269. }
  1270. }
  1271. }
  1272. }
  1273. else
  1274. {
  1275. const DWORD d = fGetASPI32SupportInfo();
  1276. BYTE status = HIBYTE (LOWORD (d));
  1277. if (status != SS_COMP || status == SS_NO_ADAPTERS)
  1278. return 0;
  1279. const int numAdapters = LOBYTE (LOWORD (d));
  1280. for (BYTE ha = 0; ha < numAdapters; ++ha)
  1281. {
  1282. SRB_HAInquiry s;
  1283. zerostruct (s);
  1284. s.SRB_Cmd = SC_HA_INQUIRY;
  1285. s.SRB_HaID = ha;
  1286. fSendASPI32Command ((LPSRB)&s);
  1287. if (s.SRB_Status == SS_COMP)
  1288. {
  1289. maxItems = (int)s.HA_Unique[3];
  1290. if (maxItems == 0)
  1291. maxItems = 8;
  1292. for (BYTE tgt = 0; tgt < maxItems; ++tgt)
  1293. {
  1294. for (BYTE lun = 0; lun < 8; ++lun)
  1295. {
  1296. SRB_GDEVBlock sb;
  1297. zerostruct (sb);
  1298. sb.SRB_Cmd = SC_GET_DEV_TYPE;
  1299. sb.SRB_HaID = ha;
  1300. sb.SRB_Target = tgt;
  1301. sb.SRB_Lun = lun;
  1302. fSendASPI32Command ((LPSRB) &sb);
  1303. if (sb.SRB_Status == SS_COMP
  1304. && sb.SRB_DeviceType == DTYPE_CROM)
  1305. {
  1306. zeromem (&list[count], sizeof (CDDeviceInfo));
  1307. list[count].ha = ha;
  1308. list[count].tgt = tgt;
  1309. list[count].lun = lun;
  1310. GetAspiDeviceInfo (&(list[count]), ha, tgt, lun);
  1311. ++count;
  1312. }
  1313. }
  1314. }
  1315. }
  1316. }
  1317. }
  1318. return count;
  1319. }
  1320. //==============================================================================
  1321. static int ripperUsers = 0;
  1322. static bool initialisedOk = false;
  1323. class DeinitialiseTimer : private Timer,
  1324. private DeletedAtShutdown
  1325. {
  1326. DeinitialiseTimer (const DeinitialiseTimer&);
  1327. const DeinitialiseTimer& operator= (const DeinitialiseTimer&);
  1328. public:
  1329. DeinitialiseTimer()
  1330. {
  1331. startTimer (4000);
  1332. }
  1333. ~DeinitialiseTimer()
  1334. {
  1335. if (--ripperUsers == 0)
  1336. DeinitialiseCDRipper();
  1337. }
  1338. void timerCallback()
  1339. {
  1340. delete this;
  1341. }
  1342. juce_UseDebuggingNewOperator
  1343. };
  1344. static void incUserCount()
  1345. {
  1346. if (ripperUsers++ == 0)
  1347. initialisedOk = InitialiseCDRipper();
  1348. }
  1349. static void decUserCount()
  1350. {
  1351. new DeinitialiseTimer();
  1352. }
  1353. //==============================================================================
  1354. struct CDDeviceWrapper
  1355. {
  1356. CDDeviceHandle* cdH;
  1357. CDReadBuffer* overlapBuffer;
  1358. bool jitter;
  1359. };
  1360. //==============================================================================
  1361. static int getAddressOf (const TOCTRACK* const t)
  1362. {
  1363. return (((DWORD)t->addr[0]) << 24) + (((DWORD)t->addr[1]) << 16) +
  1364. (((DWORD)t->addr[2]) << 8) + ((DWORD)t->addr[3]);
  1365. }
  1366. static int getMSFAddressOf (const TOCTRACK* const t)
  1367. {
  1368. return 60 * t->addr[1] + t->addr[2];
  1369. }
  1370. static const int samplesPerFrame = 44100 / 75;
  1371. static const int bytesPerFrame = samplesPerFrame * 4;
  1372. //==============================================================================
  1373. const StringArray AudioCDReader::getAvailableCDNames()
  1374. {
  1375. StringArray results;
  1376. incUserCount();
  1377. if (initialisedOk)
  1378. {
  1379. CDDeviceInfo list[8];
  1380. const int num = FindCDDevices (list, 8);
  1381. decUserCount();
  1382. for (int i = 0; i < num; ++i)
  1383. {
  1384. String s;
  1385. if (list[i].scsiDriveLetter > 0)
  1386. s << String::charToString (list[i].scsiDriveLetter).toUpperCase() << T(": ");
  1387. s << String (list[i].vendor).trim()
  1388. << T(" ") << String (list[i].productId).trim()
  1389. << T(" ") << String (list[i].rev).trim();
  1390. results.add (s);
  1391. }
  1392. }
  1393. return results;
  1394. }
  1395. static CDDeviceHandle* openHandle (const CDDeviceInfo* const device)
  1396. {
  1397. SRB_GDEVBlock s;
  1398. zerostruct (s);
  1399. s.SRB_Cmd = SC_GET_DEV_TYPE;
  1400. s.SRB_HaID = device->ha;
  1401. s.SRB_Target = device->tgt;
  1402. s.SRB_Lun = device->lun;
  1403. if (usingScsi)
  1404. {
  1405. HANDLE h = CreateSCSIDeviceHandle (device->scsiDriveLetter);
  1406. if (h != INVALID_HANDLE_VALUE)
  1407. {
  1408. CDDeviceHandle* cdh = new CDDeviceHandle (device);
  1409. cdh->scsiHandle = h;
  1410. return cdh;
  1411. }
  1412. }
  1413. else
  1414. {
  1415. if (fSendASPI32Command ((LPSRB)&s) == SS_COMP
  1416. && s.SRB_DeviceType == DTYPE_CROM)
  1417. {
  1418. return new CDDeviceHandle (device);
  1419. }
  1420. }
  1421. return 0;
  1422. }
  1423. AudioCDReader* AudioCDReader::createReaderForCD (const int deviceIndex)
  1424. {
  1425. incUserCount();
  1426. if (initialisedOk)
  1427. {
  1428. CDDeviceInfo list[8];
  1429. const int num = FindCDDevices (list, 8);
  1430. if (((unsigned int) deviceIndex) < (unsigned int) num)
  1431. {
  1432. CDDeviceHandle* const handle = openHandle (&(list[deviceIndex]));
  1433. if (handle != 0)
  1434. {
  1435. CDDeviceWrapper* const d = new CDDeviceWrapper();
  1436. d->cdH = handle;
  1437. d->overlapBuffer = new CDReadBuffer(3);
  1438. return new AudioCDReader (d);
  1439. }
  1440. }
  1441. }
  1442. decUserCount();
  1443. return 0;
  1444. }
  1445. AudioCDReader::AudioCDReader (void* handle_)
  1446. : AudioFormatReader (0, T("CD Audio")),
  1447. handle (handle_),
  1448. indexingEnabled (false),
  1449. lastIndex (0),
  1450. firstFrameInBuffer (0),
  1451. samplesInBuffer (0)
  1452. {
  1453. jassert (handle_ != 0);
  1454. refreshTrackLengths();
  1455. sampleRate = 44100.0;
  1456. bitsPerSample = 16;
  1457. lengthInSamples = getPositionOfTrackStart (numTracks);
  1458. numChannels = 2;
  1459. usesFloatingPointData = false;
  1460. buffer.setSize (4 * bytesPerFrame, true);
  1461. }
  1462. AudioCDReader::~AudioCDReader()
  1463. {
  1464. CDDeviceWrapper* const device = (CDDeviceWrapper*)handle;
  1465. delete device->cdH;
  1466. delete device->overlapBuffer;
  1467. delete device;
  1468. decUserCount();
  1469. }
  1470. bool AudioCDReader::read (int** destSamples,
  1471. int64 startSampleInFile,
  1472. int numSamples)
  1473. {
  1474. CDDeviceWrapper* const device = (CDDeviceWrapper*)handle;
  1475. bool ok = true;
  1476. int offset = 0;
  1477. if (startSampleInFile < 0)
  1478. {
  1479. int* l = destSamples[0];
  1480. int* r = destSamples[1];
  1481. numSamples += (int) startSampleInFile;
  1482. offset -= (int) startSampleInFile;
  1483. while (++startSampleInFile <= 0)
  1484. {
  1485. *l++ = 0;
  1486. if (r != 0)
  1487. *r++ = 0;
  1488. }
  1489. }
  1490. while (numSamples > 0)
  1491. {
  1492. const int bufferStartSample = firstFrameInBuffer * samplesPerFrame;
  1493. const int bufferEndSample = bufferStartSample + samplesInBuffer;
  1494. if (startSampleInFile >= bufferStartSample
  1495. && startSampleInFile < bufferEndSample)
  1496. {
  1497. const int toDo = (int) jmin ((int64) numSamples, bufferEndSample - startSampleInFile);
  1498. int* const l = destSamples[0] + offset;
  1499. int* const r = destSamples[1] + offset;
  1500. const short* src = (const short*) buffer.getData();
  1501. src += 2 * (startSampleInFile - bufferStartSample);
  1502. for (int i = 0; i < toDo; ++i)
  1503. {
  1504. l[i] = src [i << 1] << 16;
  1505. if (r != 0)
  1506. r[i] = src [(i << 1) + 1] << 16;
  1507. }
  1508. offset += toDo;
  1509. startSampleInFile += toDo;
  1510. numSamples -= toDo;
  1511. }
  1512. else
  1513. {
  1514. const int framesInBuffer = buffer.getSize() / bytesPerFrame;
  1515. const int frameNeeded = (int) (startSampleInFile / samplesPerFrame);
  1516. if (firstFrameInBuffer + framesInBuffer != frameNeeded)
  1517. {
  1518. device->overlapBuffer->dataLength = 0;
  1519. device->overlapBuffer->startFrame = 0;
  1520. device->overlapBuffer->numFrames = 0;
  1521. device->jitter = false;
  1522. }
  1523. firstFrameInBuffer = frameNeeded;
  1524. lastIndex = 0;
  1525. CDReadBuffer readBuffer (framesInBuffer + 4);
  1526. readBuffer.wantsIndex = indexingEnabled;
  1527. int i;
  1528. for (i = 5; --i >= 0;)
  1529. {
  1530. readBuffer.startFrame = frameNeeded;
  1531. readBuffer.numFrames = framesInBuffer;
  1532. if (device->cdH->readAudio (&readBuffer, (device->jitter) ? device->overlapBuffer : 0))
  1533. break;
  1534. else
  1535. device->overlapBuffer->dataLength = 0;
  1536. }
  1537. if (i >= 0)
  1538. {
  1539. memcpy ((char*) buffer.getData(),
  1540. readBuffer.buffer + readBuffer.dataStartOffset,
  1541. readBuffer.dataLength);
  1542. samplesInBuffer = readBuffer.dataLength >> 2;
  1543. lastIndex = readBuffer.index;
  1544. }
  1545. else
  1546. {
  1547. int* l = destSamples[0] + offset;
  1548. int* r = destSamples[1] + offset;
  1549. while (--numSamples >= 0)
  1550. {
  1551. *l++ = 0;
  1552. if (r != 0)
  1553. *r++ = 0;
  1554. }
  1555. // sometimes the read fails for just the very last couple of blocks, so
  1556. // we'll ignore and errors in the last half-second of the disk..
  1557. ok = startSampleInFile > (trackStarts [numTracks] - 20000);
  1558. break;
  1559. }
  1560. }
  1561. }
  1562. return ok;
  1563. }
  1564. bool AudioCDReader::isCDStillPresent() const
  1565. {
  1566. TOC toc;
  1567. zerostruct (toc);
  1568. return ((CDDeviceWrapper*)handle)->cdH->readTOC (&toc, false);
  1569. }
  1570. int AudioCDReader::getNumTracks() const
  1571. {
  1572. return numTracks;
  1573. }
  1574. int AudioCDReader::getPositionOfTrackStart (int trackNum) const
  1575. {
  1576. return (trackNum >= 0 && trackNum <= numTracks) ? trackStarts [trackNum] * samplesPerFrame
  1577. : 0;
  1578. }
  1579. void AudioCDReader::refreshTrackLengths()
  1580. {
  1581. zeromem (trackStarts, sizeof (trackStarts));
  1582. zeromem (audioTracks, sizeof (audioTracks));
  1583. TOC toc;
  1584. zerostruct (toc);
  1585. if (((CDDeviceWrapper*)handle)->cdH->readTOC (&toc, false))
  1586. {
  1587. numTracks = 1 + toc.lastTrack - toc.firstTrack;
  1588. for (int i = 0; i <= numTracks; ++i)
  1589. {
  1590. trackStarts[i] = getAddressOf (&toc.tracks[i]);
  1591. audioTracks[i] = ((toc.tracks[i].ADR & 4) == 0);
  1592. }
  1593. }
  1594. else
  1595. {
  1596. numTracks = 0;
  1597. }
  1598. }
  1599. bool AudioCDReader::isTrackAudio (int trackNum) const
  1600. {
  1601. return (trackNum >= 0 && trackNum <= numTracks) ? audioTracks [trackNum]
  1602. : false;
  1603. }
  1604. void AudioCDReader::enableIndexScanning (bool b)
  1605. {
  1606. indexingEnabled = b;
  1607. }
  1608. int AudioCDReader::getLastIndex() const
  1609. {
  1610. return lastIndex;
  1611. }
  1612. const int framesPerIndexRead = 4;
  1613. int AudioCDReader::getIndexAt (int samplePos)
  1614. {
  1615. CDDeviceWrapper* const device = (CDDeviceWrapper*) handle;
  1616. const int frameNeeded = samplePos / samplesPerFrame;
  1617. device->overlapBuffer->dataLength = 0;
  1618. device->overlapBuffer->startFrame = 0;
  1619. device->overlapBuffer->numFrames = 0;
  1620. device->jitter = false;
  1621. firstFrameInBuffer = 0;
  1622. lastIndex = 0;
  1623. CDReadBuffer readBuffer (4 + framesPerIndexRead);
  1624. readBuffer.wantsIndex = true;
  1625. int i;
  1626. for (i = 5; --i >= 0;)
  1627. {
  1628. readBuffer.startFrame = frameNeeded;
  1629. readBuffer.numFrames = framesPerIndexRead;
  1630. if (device->cdH->readAudio (&readBuffer, (false) ? device->overlapBuffer : 0))
  1631. break;
  1632. }
  1633. if (i >= 0)
  1634. return readBuffer.index;
  1635. return -1;
  1636. }
  1637. const Array <int> AudioCDReader::findIndexesInTrack (const int trackNumber)
  1638. {
  1639. Array <int> indexes;
  1640. const int trackStart = getPositionOfTrackStart (trackNumber);
  1641. const int trackEnd = getPositionOfTrackStart (trackNumber + 1);
  1642. bool needToScan = true;
  1643. if (trackEnd - trackStart > 20 * 44100)
  1644. {
  1645. // check the end of the track for indexes before scanning the whole thing
  1646. needToScan = false;
  1647. int pos = jmax (trackStart, trackEnd - 44100 * 5);
  1648. bool seenAnIndex = false;
  1649. while (pos <= trackEnd - samplesPerFrame)
  1650. {
  1651. const int index = getIndexAt (pos);
  1652. if (index == 0)
  1653. {
  1654. // lead-out, so skip back a bit if we've not found any indexes yet..
  1655. if (seenAnIndex)
  1656. break;
  1657. pos -= 44100 * 5;
  1658. if (pos < trackStart)
  1659. break;
  1660. }
  1661. else
  1662. {
  1663. if (index > 0)
  1664. seenAnIndex = true;
  1665. if (index > 1)
  1666. {
  1667. needToScan = true;
  1668. break;
  1669. }
  1670. pos += samplesPerFrame * framesPerIndexRead;
  1671. }
  1672. }
  1673. }
  1674. if (needToScan)
  1675. {
  1676. CDDeviceWrapper* const device = (CDDeviceWrapper*) handle;
  1677. int pos = trackStart;
  1678. int last = -1;
  1679. while (pos < trackEnd - samplesPerFrame * 10)
  1680. {
  1681. const int frameNeeded = pos / samplesPerFrame;
  1682. device->overlapBuffer->dataLength = 0;
  1683. device->overlapBuffer->startFrame = 0;
  1684. device->overlapBuffer->numFrames = 0;
  1685. device->jitter = false;
  1686. firstFrameInBuffer = 0;
  1687. CDReadBuffer readBuffer (4);
  1688. readBuffer.wantsIndex = true;
  1689. int i;
  1690. for (i = 5; --i >= 0;)
  1691. {
  1692. readBuffer.startFrame = frameNeeded;
  1693. readBuffer.numFrames = framesPerIndexRead;
  1694. if (device->cdH->readAudio (&readBuffer, (false) ? device->overlapBuffer : 0))
  1695. break;
  1696. }
  1697. if (i < 0)
  1698. break;
  1699. if (readBuffer.index > last && readBuffer.index > 1)
  1700. {
  1701. last = readBuffer.index;
  1702. indexes.add (pos);
  1703. }
  1704. pos += samplesPerFrame * framesPerIndexRead;
  1705. }
  1706. indexes.removeValue (trackStart);
  1707. }
  1708. return indexes;
  1709. }
  1710. int AudioCDReader::getCDDBId()
  1711. {
  1712. refreshTrackLengths();
  1713. if (numTracks > 0)
  1714. {
  1715. TOC toc;
  1716. zerostruct (toc);
  1717. if (((CDDeviceWrapper*) handle)->cdH->readTOC (&toc, true))
  1718. {
  1719. int n = 0;
  1720. for (int i = numTracks; --i >= 0;)
  1721. {
  1722. int j = getMSFAddressOf (&toc.tracks[i]);
  1723. while (j > 0)
  1724. {
  1725. n += (j % 10);
  1726. j /= 10;
  1727. }
  1728. }
  1729. if (n != 0)
  1730. {
  1731. const int t = getMSFAddressOf (&toc.tracks[numTracks])
  1732. - getMSFAddressOf (&toc.tracks[0]);
  1733. return ((n % 0xff) << 24) | (t << 8) | numTracks;
  1734. }
  1735. }
  1736. }
  1737. return 0;
  1738. }
  1739. void AudioCDReader::ejectDisk()
  1740. {
  1741. ((CDDeviceWrapper*) handle)->cdH->openDrawer (true);
  1742. }
  1743. #if JUCE_USE_CDBURNER
  1744. //==============================================================================
  1745. static IDiscRecorder* enumCDBurners (StringArray* list, int indexToOpen, IDiscMaster** master)
  1746. {
  1747. CoInitialize (0);
  1748. IDiscMaster* dm;
  1749. IDiscRecorder* result = 0;
  1750. if (SUCCEEDED (CoCreateInstance (CLSID_MSDiscMasterObj, 0,
  1751. CLSCTX_INPROC_SERVER | CLSCTX_LOCAL_SERVER,
  1752. IID_IDiscMaster,
  1753. (void**) &dm)))
  1754. {
  1755. if (SUCCEEDED (dm->Open()))
  1756. {
  1757. IEnumDiscRecorders* drEnum = 0;
  1758. if (SUCCEEDED (dm->EnumDiscRecorders (&drEnum)))
  1759. {
  1760. IDiscRecorder* dr = 0;
  1761. DWORD dummy;
  1762. int index = 0;
  1763. while (drEnum->Next (1, &dr, &dummy) == S_OK)
  1764. {
  1765. if (indexToOpen == index)
  1766. {
  1767. result = dr;
  1768. break;
  1769. }
  1770. else if (list != 0)
  1771. {
  1772. BSTR path;
  1773. if (SUCCEEDED (dr->GetPath (&path)))
  1774. list->add ((const WCHAR*) path);
  1775. }
  1776. ++index;
  1777. dr->Release();
  1778. }
  1779. drEnum->Release();
  1780. }
  1781. /*if (redbookFormat != 0)
  1782. {
  1783. IEnumDiscMasterFormats* mfEnum;
  1784. if (SUCCEEDED (dm->EnumDiscMasterFormats (&mfEnum)))
  1785. {
  1786. IID formatIID;
  1787. DWORD dummy;
  1788. while (mfEnum->Next (1, &formatIID, &dummy) == S_OK)
  1789. {
  1790. }
  1791. mfEnum->Release();
  1792. }
  1793. redbookFormat
  1794. }*/
  1795. if (master == 0)
  1796. dm->Close();
  1797. }
  1798. if (master != 0)
  1799. *master = dm;
  1800. else
  1801. dm->Release();
  1802. }
  1803. return result;
  1804. }
  1805. const StringArray AudioCDBurner::findAvailableDevices()
  1806. {
  1807. StringArray devs;
  1808. enumCDBurners (&devs, -1, 0);
  1809. return devs;
  1810. }
  1811. AudioCDBurner* AudioCDBurner::openDevice (const int deviceIndex)
  1812. {
  1813. AudioCDBurner* b = new AudioCDBurner (deviceIndex);
  1814. if (b->internal == 0)
  1815. deleteAndZero (b);
  1816. return b;
  1817. }
  1818. class CDBurnerInfo : public IDiscMasterProgressEvents
  1819. {
  1820. public:
  1821. CDBurnerInfo()
  1822. : refCount (1),
  1823. progress (0),
  1824. shouldCancel (false),
  1825. listener (0)
  1826. {
  1827. }
  1828. ~CDBurnerInfo()
  1829. {
  1830. }
  1831. HRESULT __stdcall QueryInterface (REFIID id, void __RPC_FAR* __RPC_FAR* result)
  1832. {
  1833. if (result == 0)
  1834. return E_POINTER;
  1835. if (id == IID_IUnknown || id == IID_IDiscMasterProgressEvents)
  1836. {
  1837. AddRef();
  1838. *result = this;
  1839. return S_OK;
  1840. }
  1841. *result = 0;
  1842. return E_NOINTERFACE;
  1843. }
  1844. ULONG __stdcall AddRef() { return ++refCount; }
  1845. ULONG __stdcall Release() { jassert (refCount > 0); const int r = --refCount; if (r == 0) delete this; return r; }
  1846. HRESULT __stdcall QueryCancel (boolean* pbCancel)
  1847. {
  1848. if (listener != 0 && ! shouldCancel)
  1849. shouldCancel = listener->audioCDBurnProgress (progress);
  1850. *pbCancel = shouldCancel;
  1851. return S_OK;
  1852. }
  1853. HRESULT __stdcall NotifyBlockProgress (long nCompleted, long nTotal)
  1854. {
  1855. progress = nCompleted / (float) nTotal;
  1856. shouldCancel = listener != 0 && listener->audioCDBurnProgress (progress);
  1857. return E_NOTIMPL;
  1858. }
  1859. HRESULT __stdcall NotifyPnPActivity (void) { return E_NOTIMPL; }
  1860. HRESULT __stdcall NotifyAddProgress (long /*nCompletedSteps*/, long /*nTotalSteps*/) { return E_NOTIMPL; }
  1861. HRESULT __stdcall NotifyTrackProgress (long /*nCurrentTrack*/, long /*nTotalTracks*/) { return E_NOTIMPL; }
  1862. HRESULT __stdcall NotifyPreparingBurn (long /*nEstimatedSeconds*/) { return E_NOTIMPL; }
  1863. HRESULT __stdcall NotifyClosingDisc (long /*nEstimatedSeconds*/) { return E_NOTIMPL; }
  1864. HRESULT __stdcall NotifyBurnComplete (HRESULT /*status*/) { return E_NOTIMPL; }
  1865. HRESULT __stdcall NotifyEraseComplete (HRESULT /*status*/) { return E_NOTIMPL; }
  1866. IDiscMaster* discMaster;
  1867. IDiscRecorder* discRecorder;
  1868. IRedbookDiscMaster* redbook;
  1869. AudioCDBurner::BurnProgressListener* listener;
  1870. float progress;
  1871. bool shouldCancel;
  1872. private:
  1873. int refCount;
  1874. };
  1875. AudioCDBurner::AudioCDBurner (const int deviceIndex)
  1876. : internal (0)
  1877. {
  1878. IDiscMaster* discMaster;
  1879. IDiscRecorder* dr = enumCDBurners (0, deviceIndex, &discMaster);
  1880. if (dr != 0)
  1881. {
  1882. IRedbookDiscMaster* redbook;
  1883. HRESULT hr = discMaster->SetActiveDiscMasterFormat (IID_IRedbookDiscMaster, (void**) &redbook);
  1884. hr = discMaster->SetActiveDiscRecorder (dr);
  1885. CDBurnerInfo* const info = new CDBurnerInfo();
  1886. internal = info;
  1887. info->discMaster = discMaster;
  1888. info->discRecorder = dr;
  1889. info->redbook = redbook;
  1890. }
  1891. }
  1892. AudioCDBurner::~AudioCDBurner()
  1893. {
  1894. CDBurnerInfo* const info = (CDBurnerInfo*) internal;
  1895. if (info != 0)
  1896. {
  1897. info->discRecorder->Close();
  1898. info->redbook->Release();
  1899. info->discRecorder->Release();
  1900. info->discMaster->Release();
  1901. info->Release();
  1902. }
  1903. }
  1904. bool AudioCDBurner::isDiskPresent() const
  1905. {
  1906. CDBurnerInfo* const info = (CDBurnerInfo*) internal;
  1907. HRESULT hr = info->discRecorder->OpenExclusive();
  1908. long type, flags;
  1909. hr = info->discRecorder->QueryMediaType (&type, &flags);
  1910. info->discRecorder->Close();
  1911. return hr == S_OK && type != 0 && (flags & MEDIA_WRITABLE) != 0;
  1912. }
  1913. int AudioCDBurner::getNumAvailableAudioBlocks() const
  1914. {
  1915. CDBurnerInfo* const info = (CDBurnerInfo*) internal;
  1916. long blocksFree = 0;
  1917. info->redbook->GetAvailableAudioTrackBlocks (&blocksFree);
  1918. return blocksFree;
  1919. }
  1920. const String AudioCDBurner::burn (AudioCDBurner::BurnProgressListener* listener,
  1921. const bool ejectDiscAfterwards,
  1922. const bool performFakeBurnForTesting)
  1923. {
  1924. CDBurnerInfo* const info = (CDBurnerInfo*) internal;
  1925. info->listener = listener;
  1926. info->progress = 0;
  1927. info->shouldCancel = false;
  1928. UINT_PTR cookie;
  1929. HRESULT hr = info->discMaster->ProgressAdvise (info, &cookie);
  1930. hr = info->discMaster->RecordDisc (performFakeBurnForTesting,
  1931. ejectDiscAfterwards);
  1932. String error;
  1933. if (hr != S_OK)
  1934. {
  1935. const char* e = "Couldn't open or write to the CD device";
  1936. if (hr == IMAPI_E_USERABORT)
  1937. e = "User cancelled the write operation";
  1938. else if (hr == IMAPI_E_MEDIUM_NOTPRESENT || hr == IMAPI_E_TRACKOPEN)
  1939. e = "No Disk present";
  1940. error = e;
  1941. }
  1942. info->discMaster->ProgressUnadvise (cookie);
  1943. info->listener = 0;
  1944. return error;
  1945. }
  1946. bool AudioCDBurner::addAudioTrack (AudioSource* source, int numSamples)
  1947. {
  1948. if (source == 0)
  1949. return false;
  1950. CDBurnerInfo* const info = (CDBurnerInfo*) internal;
  1951. long bytesPerBlock;
  1952. HRESULT hr = info->redbook->GetAudioBlockSize (&bytesPerBlock);
  1953. const int samplesPerBlock = bytesPerBlock / 4;
  1954. bool ok = true;
  1955. hr = info->redbook->CreateAudioTrack ((long) numSamples / (bytesPerBlock * 4));
  1956. byte* const buffer = (byte*) juce_malloc (bytesPerBlock);
  1957. AudioSampleBuffer sourceBuffer (2, samplesPerBlock);
  1958. int samplesDone = 0;
  1959. source->prepareToPlay (samplesPerBlock, 44100.0);
  1960. while (ok)
  1961. {
  1962. {
  1963. AudioSourceChannelInfo info;
  1964. info.buffer = &sourceBuffer;
  1965. info.numSamples = samplesPerBlock;
  1966. info.startSample = 0;
  1967. sourceBuffer.clear();
  1968. source->getNextAudioBlock (info);
  1969. }
  1970. zeromem (buffer, bytesPerBlock);
  1971. AudioDataConverters::convertFloatToInt16LE (sourceBuffer.getSampleData (0, 0),
  1972. buffer, samplesPerBlock, 4);
  1973. AudioDataConverters::convertFloatToInt16LE (sourceBuffer.getSampleData (1, 0),
  1974. buffer + 2, samplesPerBlock, 4);
  1975. hr = info->redbook->AddAudioTrackBlocks (buffer, bytesPerBlock);
  1976. if (hr != S_OK)
  1977. ok = false;
  1978. samplesDone += samplesPerBlock;
  1979. if (samplesDone >= numSamples)
  1980. break;
  1981. }
  1982. juce_free (buffer);
  1983. hr = info->redbook->CloseAudioTrack();
  1984. delete source;
  1985. return ok && hr == S_OK;
  1986. }
  1987. #endif
  1988. END_JUCE_NAMESPACE