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.

2486 lines
76KB

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