1 |
/* PCSX2 - PS2 Emulator for PCs |
2 |
* Copyright (C) 2002-2010 PCSX2 Dev Team |
3 |
* |
4 |
* PCSX2 is free software: you can redistribute it and/or modify it under the terms |
5 |
* of the GNU Lesser General Public License as published by the Free Software Found- |
6 |
* ation, either version 3 of the License, or (at your option) any later version. |
7 |
* |
8 |
* PCSX2 is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; |
9 |
* without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR |
10 |
* PURPOSE. See the GNU General Public License for more details. |
11 |
* |
12 |
* You should have received a copy of the GNU General Public License along with PCSX2. |
13 |
* If not, see <http://www.gnu.org/licenses/>. |
14 |
*/ |
15 |
|
16 |
|
17 |
/* |
18 |
* Original code from libcdvd by Hiryu & Sjeep (C) 2002 |
19 |
* Modified by Florin for PCSX2 emu |
20 |
* Fixed CdRead by linuzappz |
21 |
*/ |
22 |
|
23 |
#include "PrecompiledHeader.h" |
24 |
|
25 |
#include <stdio.h> |
26 |
#include <stdlib.h> |
27 |
#include <fcntl.h> |
28 |
|
29 |
#include "CDVDisoReader.h" |
30 |
|
31 |
static u8 *pbuffer; |
32 |
static u8 cdbuffer[2352] = {0}; |
33 |
static isoFile *iso = NULL; |
34 |
|
35 |
static int psize, cdtype; |
36 |
|
37 |
static s32 layer1start = -1; |
38 |
|
39 |
void CALLBACK ISOclose() |
40 |
{ |
41 |
isoClose(iso); |
42 |
iso = NULL; |
43 |
} |
44 |
|
45 |
s32 CALLBACK ISOopen(const char* pTitle) |
46 |
{ |
47 |
ISOclose(); // just in case |
48 |
|
49 |
if( (pTitle == NULL) || (pTitle[0] == 0) ) |
50 |
{ |
51 |
Console.Error( "CDVDiso Error: No filename specified." ); |
52 |
return -1; |
53 |
} |
54 |
|
55 |
iso = isoOpen(pTitle); |
56 |
if (iso == NULL) |
57 |
{ |
58 |
Console.Error( "CDVDiso Error: Failed loading %s", pTitle ); |
59 |
return -1; |
60 |
} |
61 |
|
62 |
switch (iso->type) |
63 |
{ |
64 |
case ISOTYPE_DVD: |
65 |
cdtype = CDVD_TYPE_PS2DVD; |
66 |
break; |
67 |
case ISOTYPE_AUDIO: |
68 |
cdtype = CDVD_TYPE_CDDA; |
69 |
break; |
70 |
default: |
71 |
cdtype = CDVD_TYPE_PS2CD; |
72 |
break; |
73 |
} |
74 |
|
75 |
layer1start = -1; |
76 |
|
77 |
return 0; |
78 |
} |
79 |
|
80 |
s32 CALLBACK ISOreadSubQ(u32 lsn, cdvdSubQ* subq) |
81 |
{ |
82 |
// fake it |
83 |
u8 min, sec, frm; |
84 |
subq->ctrl = 4; |
85 |
subq->mode = 1; |
86 |
subq->trackNum = itob(1); |
87 |
subq->trackIndex = itob(1); |
88 |
|
89 |
lba_to_msf(lsn, &min, &sec, &frm); |
90 |
subq->trackM = itob(min); |
91 |
subq->trackS = itob(sec); |
92 |
subq->trackF = itob(frm); |
93 |
|
94 |
subq->pad = 0; |
95 |
|
96 |
lba_to_msf(lsn + (2*75), &min, &sec, &frm); |
97 |
subq->discM = itob(min); |
98 |
subq->discS = itob(sec); |
99 |
subq->discF = itob(frm); |
100 |
|
101 |
return 0; |
102 |
} |
103 |
|
104 |
s32 CALLBACK ISOgetTN(cdvdTN *Buffer) |
105 |
{ |
106 |
Buffer->strack = 1; |
107 |
Buffer->etrack = 1; |
108 |
|
109 |
return 0; |
110 |
} |
111 |
|
112 |
s32 CALLBACK ISOgetTD(u8 Track, cdvdTD *Buffer) |
113 |
{ |
114 |
if (Track == 0) |
115 |
{ |
116 |
Buffer->lsn = iso->blocks; |
117 |
} |
118 |
else |
119 |
{ |
120 |
Buffer->type = CDVD_MODE1_TRACK; |
121 |
Buffer->lsn = 0; |
122 |
} |
123 |
|
124 |
return 0; |
125 |
} |
126 |
|
127 |
#include "gui/App.h" |
128 |
#include "Utilities/HashMap.h" |
129 |
|
130 |
static bool testForPartitionInfo( const u8 (&tempbuffer)[CD_FRAMESIZE_RAW] ) |
131 |
{ |
132 |
const int off = iso->blockofs; |
133 |
|
134 |
// test for: CD001 |
135 |
return ( |
136 |
(tempbuffer[off+1] == 0x43) && |
137 |
(tempbuffer[off+2] == 0x44) && |
138 |
(tempbuffer[off+3] == 0x30) && |
139 |
(tempbuffer[off+4] == 0x30) && |
140 |
(tempbuffer[off+5] == 0x31) |
141 |
); |
142 |
} |
143 |
|
144 |
static bool FindLayer1Start() |
145 |
{ |
146 |
if( (layer1start != -1) || (iso->blocks < 0x230540) ) return true; |
147 |
|
148 |
Console.WriteLn("CDVDiso: searching for layer1..."); |
149 |
|
150 |
int blockresult = -1; |
151 |
|
152 |
// Check the ini file cache first: |
153 |
// Cache is stored in LayerBreakCache.ini, and is associated by hex-encoded hash key of the |
154 |
// complete filename/path of the iso file. :) |
155 |
|
156 |
wxString layerCacheFile( Path::Combine(GetSettingsFolder().ToString(), L"LayerBreakCache.ini") ); |
157 |
wxFileConfig layerCacheIni( wxEmptyString, wxEmptyString, layerCacheFile, wxEmptyString, wxCONFIG_USE_RELATIVE_PATH ); |
158 |
|
159 |
wxString cacheKey; |
160 |
cacheKey.Printf( L"%X", HashTools::Hash( iso->filename, strlen( iso->filename ) ) ); |
161 |
|
162 |
blockresult = layerCacheIni.Read( cacheKey, -1 ); |
163 |
if( blockresult != -1 ) |
164 |
{ |
165 |
u8 tempbuffer[CD_FRAMESIZE_RAW]; |
166 |
isoReadBlock(iso, tempbuffer, blockresult); |
167 |
|
168 |
if( testForPartitionInfo( tempbuffer ) ) |
169 |
{ |
170 |
Console.WriteLn( "CDVDiso: loaded second layer from settings cache, sector=0x%8.8x", blockresult ); |
171 |
layer1start = blockresult; |
172 |
} |
173 |
else |
174 |
{ |
175 |
Console.Warning( "CDVDiso: second layer info in the settings cache appears to be obsolete or invalid." ); |
176 |
} |
177 |
} |
178 |
else |
179 |
{ |
180 |
DevCon.WriteLn( "CDVDiso: no cached info for second layer found." ); |
181 |
} |
182 |
|
183 |
if( layer1start == -1 ) |
184 |
{ |
185 |
|
186 |
// Layer sizes are arbitrary, and either layer could be the smaller (GoW and Rogue Galaxy |
187 |
// both have Layer1 larger than Layer0 for example), so we have to brute-force the search |
188 |
// from some arbitrary start position. |
189 |
// |
190 |
// Method: Inside->out. We start at the middle of the image and work our way out toward |
191 |
// both the beginning and end of the image at the same time. Most images have the layer |
192 |
// break quite close to the middle of the image, so this should be pretty fast in most cases. |
193 |
|
194 |
// [TODO] Layer searching can be slow, especially for compressed disc images, so it would |
195 |
// be quite courteous to pop up a status dialog bar that lets the user know that it's |
196 |
// thinking. Since we're not on the GUI thread, we'll need to establish some messages |
197 |
// to create the window and pass progress increments back to it. |
198 |
|
199 |
|
200 |
uint midsector = (iso->blocks / 2) & ~0xf; |
201 |
uint deviation = 0; |
202 |
|
203 |
while( (layer1start == -1) && (deviation < midsector-16) ) |
204 |
{ |
205 |
u8 tempbuffer[CD_FRAMESIZE_RAW]; |
206 |
isoReadBlock(iso, tempbuffer, midsector-deviation); |
207 |
|
208 |
if(testForPartitionInfo( tempbuffer )) |
209 |
layer1start = midsector-deviation; |
210 |
else |
211 |
{ |
212 |
isoReadBlock(iso, tempbuffer, midsector+deviation); |
213 |
if( testForPartitionInfo( tempbuffer ) ) |
214 |
layer1start = midsector+deviation; |
215 |
} |
216 |
|
217 |
if( layer1start != -1 ) |
218 |
{ |
219 |
if( !pxAssertDev( tempbuffer[iso->blockofs] == 0x01, "Layer1-Detect: CD001 tag found, but the partition type is invalid." ) ) |
220 |
{ |
221 |
Console.Error( "CDVDiso: Invalid partition type on layer 1!? (type=0x%x)", tempbuffer[iso->blockofs] ); |
222 |
} |
223 |
} |
224 |
deviation += 16; |
225 |
} |
226 |
|
227 |
if( layer1start == -1 ) |
228 |
{ |
229 |
Console.Warning("CDVDiso: Couldn't find second layer... ignoring"); |
230 |
return false; |
231 |
} |
232 |
else |
233 |
{ |
234 |
Console.WriteLn("CDVDiso: second layer found at sector 0x%8.8x", layer1start); |
235 |
|
236 |
// Save layer information to configuration: |
237 |
|
238 |
layerCacheIni.Write( cacheKey, layer1start ); |
239 |
} |
240 |
} |
241 |
return true; |
242 |
} |
243 |
|
244 |
// Should return 0 if no error occurred, or -1 if layer detection FAILED. |
245 |
s32 CALLBACK ISOgetDualInfo(s32* dualType, u32* _layer1start) |
246 |
{ |
247 |
if( !FindLayer1Start() ) return -1; |
248 |
|
249 |
if(layer1start<0) |
250 |
{ |
251 |
*dualType = 0; |
252 |
*_layer1start = iso->blocks; |
253 |
} |
254 |
else |
255 |
{ |
256 |
*dualType = 1; |
257 |
*_layer1start = layer1start; |
258 |
} |
259 |
return 0; |
260 |
} |
261 |
|
262 |
s32 CALLBACK ISOgetDiskType() |
263 |
{ |
264 |
return cdtype; |
265 |
} |
266 |
|
267 |
s32 CALLBACK ISOgetTOC(void* toc) |
268 |
{ |
269 |
u8 type = ISOgetDiskType(); |
270 |
u8* tocBuff = (u8*)toc; |
271 |
|
272 |
//CDVD_LOG("CDVDgetTOC\n"); |
273 |
|
274 |
if (type == CDVD_TYPE_DVDV || type == CDVD_TYPE_PS2DVD) |
275 |
{ |
276 |
// get dvd structure format |
277 |
// scsi command 0x43 |
278 |
memset(tocBuff, 0, 2048); |
279 |
|
280 |
FindLayer1Start(); |
281 |
|
282 |
if (layer1start < 0) |
283 |
{ |
284 |
// fake it |
285 |
tocBuff[ 0] = 0x04; |
286 |
tocBuff[ 1] = 0x02; |
287 |
tocBuff[ 2] = 0xF2; |
288 |
tocBuff[ 3] = 0x00; |
289 |
tocBuff[ 4] = 0x86; |
290 |
tocBuff[ 5] = 0x72; |
291 |
|
292 |
tocBuff[16] = 0x00; |
293 |
tocBuff[17] = 0x03; |
294 |
tocBuff[18] = 0x00; |
295 |
tocBuff[19] = 0x00; |
296 |
return 0; |
297 |
} |
298 |
else |
299 |
{ |
300 |
// dual sided |
301 |
tocBuff[ 0] = 0x24; |
302 |
tocBuff[ 1] = 0x02; |
303 |
tocBuff[ 2] = 0xF2; |
304 |
tocBuff[ 3] = 0x00; |
305 |
tocBuff[ 4] = 0x41; |
306 |
tocBuff[ 5] = 0x95; |
307 |
|
308 |
tocBuff[14] = 0x60; // dual sided, ptp |
309 |
|
310 |
tocBuff[16] = 0x00; |
311 |
tocBuff[17] = 0x03; |
312 |
tocBuff[18] = 0x00; |
313 |
tocBuff[19] = 0x00; |
314 |
|
315 |
s32 l1s = layer1start + 0x30000 - 1; |
316 |
tocBuff[20] = (l1s >> 24); |
317 |
tocBuff[21] = (l1s >> 16) & 0xff; |
318 |
tocBuff[22] = (l1s >> 8) & 0xff; |
319 |
tocBuff[23] = (l1s >> 0) & 0xff; |
320 |
} |
321 |
} |
322 |
else if ((type == CDVD_TYPE_CDDA) || (type == CDVD_TYPE_PS2CDDA) || |
323 |
(type == CDVD_TYPE_PS2CD) || (type == CDVD_TYPE_PSCDDA) || (type == CDVD_TYPE_PSCD)) |
324 |
{ |
325 |
// cd toc |
326 |
// (could be replaced by 1 command that reads the full toc) |
327 |
u8 min, sec, frm; |
328 |
s32 i, err; |
329 |
cdvdTN diskInfo; |
330 |
cdvdTD trackInfo; |
331 |
memset(tocBuff, 0, 1024); |
332 |
if (ISOgetTN(&diskInfo) == -1) |
333 |
{ |
334 |
diskInfo.etrack = 0; |
335 |
diskInfo.strack = 1; |
336 |
} |
337 |
if (ISOgetTD(0, &trackInfo) == -1) trackInfo.lsn = 0; |
338 |
|
339 |
tocBuff[0] = 0x41; |
340 |
tocBuff[1] = 0x00; |
341 |
|
342 |
//Number of FirstTrack |
343 |
tocBuff[2] = 0xA0; |
344 |
tocBuff[7] = itob(diskInfo.strack); |
345 |
|
346 |
//Number of LastTrack |
347 |
tocBuff[12] = 0xA1; |
348 |
tocBuff[17] = itob(diskInfo.etrack); |
349 |
|
350 |
//DiskLength |
351 |
lba_to_msf(trackInfo.lsn, &min, &sec, &frm); |
352 |
tocBuff[22] = 0xA2; |
353 |
tocBuff[27] = itob(min); |
354 |
tocBuff[28] = itob(sec); |
355 |
|
356 |
for (i = diskInfo.strack; i <= diskInfo.etrack; i++) |
357 |
{ |
358 |
err = ISOgetTD(i, &trackInfo); |
359 |
lba_to_msf(trackInfo.lsn, &min, &sec, &frm); |
360 |
tocBuff[i*10+30] = trackInfo.type; |
361 |
tocBuff[i*10+32] = err == -1 ? 0 : itob(i); //number |
362 |
tocBuff[i*10+37] = itob(min); |
363 |
tocBuff[i*10+38] = itob(sec); |
364 |
tocBuff[i*10+39] = itob(frm); |
365 |
} |
366 |
} |
367 |
else |
368 |
return -1; |
369 |
|
370 |
return 0; |
371 |
} |
372 |
|
373 |
s32 CALLBACK ISOreadSector(u8* tempbuffer, u32 lsn, int mode) |
374 |
{ |
375 |
int _lsn = lsn; |
376 |
|
377 |
if (_lsn < 0) lsn = iso->blocks + _lsn; |
378 |
if (lsn > iso->blocks) return -1; |
379 |
|
380 |
if(mode == CDVD_MODE_2352) |
381 |
{ |
382 |
isoReadBlock(iso, tempbuffer, lsn); |
383 |
return 0; |
384 |
} |
385 |
|
386 |
isoReadBlock(iso, cdbuffer, lsn); |
387 |
|
388 |
pbuffer = cdbuffer; |
389 |
|
390 |
switch (mode) |
391 |
{ |
392 |
case CDVD_MODE_2352: |
393 |
psize = 2352; |
394 |
break; |
395 |
case CDVD_MODE_2340: |
396 |
pbuffer += 12; |
397 |
psize = 2340; |
398 |
break; |
399 |
case CDVD_MODE_2328: |
400 |
pbuffer += 24; |
401 |
psize = 2328; |
402 |
break; |
403 |
case CDVD_MODE_2048: |
404 |
pbuffer += 24; |
405 |
psize = 2048; |
406 |
break; |
407 |
} |
408 |
|
409 |
// version 3 blockdumps have no pbuffer header, so lets reset back to the |
410 |
// original pointer. :) |
411 |
if( iso->flags & ISOFLAGS_BLOCKDUMP_V3 ) |
412 |
pbuffer = cdbuffer; |
413 |
|
414 |
memcpy_fast(tempbuffer,pbuffer,psize); |
415 |
|
416 |
return 0; |
417 |
} |
418 |
|
419 |
s32 CALLBACK ISOreadTrack(u32 lsn, int mode) |
420 |
{ |
421 |
int _lsn = lsn; |
422 |
|
423 |
if (_lsn < 0) lsn = iso->blocks + _lsn; |
424 |
if (lsn > iso->blocks) return -1; |
425 |
|
426 |
isoReadBlock(iso, cdbuffer, lsn); |
427 |
pbuffer = cdbuffer; |
428 |
|
429 |
switch (mode) |
430 |
{ |
431 |
case CDVD_MODE_2352: |
432 |
psize = 2352; |
433 |
break; |
434 |
case CDVD_MODE_2340: |
435 |
pbuffer += 12; |
436 |
psize = 2340; |
437 |
break; |
438 |
case CDVD_MODE_2328: |
439 |
pbuffer += 24; |
440 |
psize = 2328; |
441 |
break; |
442 |
case CDVD_MODE_2048: |
443 |
pbuffer += 24; |
444 |
psize = 2048; |
445 |
break; |
446 |
} |
447 |
|
448 |
// version 3 blockdumps have no pbuffer header, so lets reset back to the |
449 |
// original pointer. :) |
450 |
if( iso->flags & ISOFLAGS_BLOCKDUMP_V3 ) |
451 |
pbuffer = cdbuffer; |
452 |
|
453 |
return 0; |
454 |
} |
455 |
|
456 |
s32 CALLBACK ISOgetBuffer2(u8* buffer) |
457 |
{ |
458 |
memcpy_fast(buffer,pbuffer,psize); |
459 |
return 0; |
460 |
} |
461 |
|
462 |
u8* CALLBACK ISOgetBuffer() |
463 |
{ |
464 |
return pbuffer; |
465 |
} |
466 |
|
467 |
s32 CALLBACK ISOgetTrayStatus() |
468 |
{ |
469 |
return CDVD_TRAY_CLOSE; |
470 |
} |
471 |
|
472 |
s32 CALLBACK ISOctrlTrayOpen() |
473 |
{ |
474 |
return 0; |
475 |
} |
476 |
s32 CALLBACK ISOctrlTrayClose() |
477 |
{ |
478 |
return 0; |
479 |
} |
480 |
|
481 |
s32 CALLBACK ISOdummyS32() |
482 |
{ |
483 |
return 0; |
484 |
} |
485 |
|
486 |
void CALLBACK ISOnewDiskCB(void(* /* callback */)()) |
487 |
{ |
488 |
} |
489 |
|
490 |
CDVD_API CDVDapi_Iso = |
491 |
{ |
492 |
ISOclose, |
493 |
|
494 |
ISOopen, |
495 |
ISOreadTrack, |
496 |
ISOgetBuffer, // emu shouldn't use this one. |
497 |
ISOreadSubQ, |
498 |
ISOgetTN, |
499 |
ISOgetTD, |
500 |
ISOgetTOC, |
501 |
ISOgetDiskType, |
502 |
ISOdummyS32, // trayStatus |
503 |
ISOdummyS32, // trayOpen |
504 |
ISOdummyS32, // trayClose |
505 |
|
506 |
ISOnewDiskCB, |
507 |
|
508 |
ISOreadSector, |
509 |
ISOgetBuffer2, |
510 |
ISOgetDualInfo, |
511 |
}; |