2024-02-29 14:50:31 +00:00
|
|
|
// SPDX-FileCopyrightText: 2019-2024 Connor McLaughlin <stenzek@gmail.com>
|
2022-12-04 11:03:45 +00:00
|
|
|
// SPDX-License-Identifier: (GPL-3.0 OR CC-BY-NC-ND-4.0)
|
|
|
|
|
2021-07-24 08:38:54 +00:00
|
|
|
#include "assert.h"
|
|
|
|
#include "cd_image.h"
|
2023-11-18 06:21:51 +00:00
|
|
|
|
2024-03-23 03:05:54 +00:00
|
|
|
// TODO: Remove me..
|
|
|
|
#include "core/host.h"
|
|
|
|
|
2024-02-29 14:50:31 +00:00
|
|
|
#include "common/assert.h"
|
2022-07-08 12:43:38 +00:00
|
|
|
#include "common/error.h"
|
|
|
|
#include "common/log.h"
|
2024-05-19 12:47:01 +00:00
|
|
|
#include "common/path.h"
|
2024-02-29 14:50:31 +00:00
|
|
|
#include "common/small_string.h"
|
2022-07-08 12:43:38 +00:00
|
|
|
#include "common/string_util.h"
|
2023-11-18 06:21:51 +00:00
|
|
|
|
2021-07-24 08:38:54 +00:00
|
|
|
#include <algorithm>
|
|
|
|
#include <cerrno>
|
|
|
|
#include <cinttypes>
|
|
|
|
#include <cmath>
|
2024-02-29 14:50:31 +00:00
|
|
|
#include <optional>
|
|
|
|
#include <span>
|
2023-11-18 06:21:51 +00:00
|
|
|
|
2021-07-24 08:38:54 +00:00
|
|
|
Log_SetChannel(CDImageDevice);
|
|
|
|
|
2024-02-29 14:50:31 +00:00
|
|
|
// Common code
|
|
|
|
[[maybe_unused]] static constexpr u32 MAX_TRACK_NUMBER = 99;
|
|
|
|
[[maybe_unused]] static constexpr u32 SCSI_CMD_LENGTH = 12;
|
|
|
|
|
|
|
|
enum class SCSIReadMode : u8
|
|
|
|
{
|
|
|
|
None,
|
|
|
|
Raw,
|
|
|
|
Full,
|
|
|
|
SubQOnly,
|
|
|
|
};
|
|
|
|
|
|
|
|
[[maybe_unused]] static void FillSCSIReadCommand(u8 cmd[SCSI_CMD_LENGTH], u32 sector_number, SCSIReadMode mode)
|
|
|
|
{
|
|
|
|
cmd[0] = 0xBE; // READ CD
|
|
|
|
cmd[1] = 0x00; // sector type
|
|
|
|
cmd[2] = Truncate8(sector_number >> 24); // Starting LBA
|
|
|
|
cmd[3] = Truncate8(sector_number >> 16);
|
|
|
|
cmd[4] = Truncate8(sector_number >> 8);
|
|
|
|
cmd[5] = Truncate8(sector_number);
|
|
|
|
cmd[6] = 0x00; // Transfer Count
|
|
|
|
cmd[7] = 0x00;
|
|
|
|
cmd[8] = 0x01;
|
|
|
|
cmd[9] = (1 << 7) | // include sync
|
|
|
|
(0b11 << 5) | // include header codes
|
|
|
|
(1 << 4) | // include user data
|
|
|
|
(1 << 3) | // edc/ecc
|
|
|
|
(0 << 2); // don't include C2 data
|
|
|
|
|
|
|
|
if (mode == SCSIReadMode::None || mode == SCSIReadMode::Raw)
|
|
|
|
cmd[10] = 0b000;
|
|
|
|
else if (mode == SCSIReadMode::Full)
|
|
|
|
cmd[10] = 0b001;
|
|
|
|
else // if (mode == SCSIReadMode::SubQOnly)
|
|
|
|
cmd[10] = 0b010;
|
|
|
|
cmd[11] = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
[[maybe_unused]] static void FillSCSISetSpeedCommand(u8 cmd[SCSI_CMD_LENGTH], u32 speed_multiplier)
|
|
|
|
{
|
|
|
|
DebugAssert(speed_multiplier > 0);
|
|
|
|
|
|
|
|
cmd[0] = 0xDA; // SET CD-ROM SPEED
|
|
|
|
cmd[1] = 0x00;
|
|
|
|
cmd[2] = Truncate8(speed_multiplier - 1);
|
|
|
|
cmd[3] = 0x00;
|
|
|
|
cmd[4] = 0x00;
|
|
|
|
cmd[5] = 0x00;
|
|
|
|
cmd[6] = 0x00;
|
|
|
|
cmd[7] = 0x00;
|
|
|
|
cmd[8] = 0x00;
|
|
|
|
cmd[9] = 0x00;
|
|
|
|
cmd[10] = 0x00;
|
|
|
|
cmd[11] = 0x00;
|
|
|
|
}
|
|
|
|
|
|
|
|
[[maybe_unused]] static constexpr u32 SCSIReadCommandOutputSize(SCSIReadMode mode)
|
|
|
|
{
|
|
|
|
switch (mode)
|
|
|
|
{
|
|
|
|
case SCSIReadMode::None:
|
|
|
|
case SCSIReadMode::Raw:
|
|
|
|
return CDImage::RAW_SECTOR_SIZE;
|
|
|
|
case SCSIReadMode::Full:
|
|
|
|
return CDImage::RAW_SECTOR_SIZE + CDImage::ALL_SUBCODE_SIZE;
|
|
|
|
case SCSIReadMode::SubQOnly:
|
|
|
|
return CDImage::RAW_SECTOR_SIZE + CDImage::SUBCHANNEL_BYTES_PER_FRAME;
|
|
|
|
default:
|
|
|
|
UnreachableCode();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-03-23 03:05:54 +00:00
|
|
|
[[maybe_unused]] static bool VerifySCSIReadData(std::span<const u8> buffer, SCSIReadMode mode,
|
|
|
|
CDImage::LBA expected_sector)
|
2024-02-29 14:50:31 +00:00
|
|
|
{
|
|
|
|
const u32 expected_size = SCSIReadCommandOutputSize(mode);
|
|
|
|
if (buffer.size() != expected_size)
|
|
|
|
{
|
|
|
|
Log_ErrorFmt("SCSI returned {} bytes, expected {}", buffer.size(), expected_size);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2024-03-23 03:05:54 +00:00
|
|
|
const CDImage::Position expected_pos = CDImage::Position::FromLBA(expected_sector);
|
|
|
|
|
2024-02-29 14:50:31 +00:00
|
|
|
if (mode == SCSIReadMode::Full)
|
|
|
|
{
|
|
|
|
// Validate subcode.
|
|
|
|
u8 deinterleaved_subcode[CDImage::ALL_SUBCODE_SIZE];
|
|
|
|
CDImage::SubChannelQ subq;
|
|
|
|
CDImage::DeinterleaveSubcode(buffer.data() + CDImage::RAW_SECTOR_SIZE, deinterleaved_subcode);
|
|
|
|
std::memcpy(&subq, &deinterleaved_subcode[CDImage::SUBCHANNEL_BYTES_PER_FRAME], sizeof(subq));
|
2024-03-23 03:05:54 +00:00
|
|
|
|
|
|
|
Log_DevFmt("SCSI full subcode read returned [{}] for {:02d}:{:02d}:{:02d}",
|
|
|
|
StringUtil::EncodeHex(subq.data.data(), static_cast<int>(subq.data.size())), expected_pos.minute,
|
|
|
|
expected_pos.second, expected_pos.frame);
|
|
|
|
|
2024-02-29 14:50:31 +00:00
|
|
|
if (!subq.IsCRCValid())
|
|
|
|
{
|
|
|
|
Log_WarningFmt("SCSI full subcode read returned invalid SubQ CRC (got {:02X} expected {:02X})", subq.crc,
|
|
|
|
CDImage::SubChannelQ::ComputeCRC(subq.data));
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2024-03-23 03:05:54 +00:00
|
|
|
const CDImage::Position got_pos =
|
|
|
|
CDImage::Position::FromBCD(subq.absolute_minute_bcd, subq.absolute_second_bcd, subq.absolute_frame_bcd);
|
|
|
|
if (expected_pos != got_pos)
|
|
|
|
{
|
|
|
|
Log_WarningFmt(
|
|
|
|
"SCSI full subcode read returned invalid MSF (got {:02x}:{:02x}:{:02x}, expected {:02d}:{:02d}:{:02d})",
|
|
|
|
subq.absolute_minute_bcd, subq.absolute_second_bcd, subq.absolute_frame_bcd, expected_pos.minute,
|
|
|
|
expected_pos.second, expected_pos.frame);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2024-02-29 14:50:31 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
else if (mode == SCSIReadMode::SubQOnly)
|
|
|
|
{
|
|
|
|
CDImage::SubChannelQ subq;
|
|
|
|
std::memcpy(&subq, buffer.data() + CDImage::RAW_SECTOR_SIZE, sizeof(subq));
|
2024-03-23 03:05:54 +00:00
|
|
|
Log_DevFmt("SCSI subq read returned [{}] for {:02d}:{:02d}:{:02d}",
|
|
|
|
StringUtil::EncodeHex(subq.data.data(), static_cast<int>(subq.data.size())), expected_pos.minute,
|
|
|
|
expected_pos.second, expected_pos.frame);
|
|
|
|
|
2024-02-29 14:50:31 +00:00
|
|
|
if (!subq.IsCRCValid())
|
|
|
|
{
|
|
|
|
Log_WarningFmt("SCSI subq read returned invalid SubQ CRC (got {:02X} expected {:02X})", subq.crc,
|
|
|
|
CDImage::SubChannelQ::ComputeCRC(subq.data));
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2024-03-23 03:05:54 +00:00
|
|
|
const CDImage::Position got_pos =
|
|
|
|
CDImage::Position::FromBCD(subq.absolute_minute_bcd, subq.absolute_second_bcd, subq.absolute_frame_bcd);
|
|
|
|
if (expected_pos != got_pos)
|
|
|
|
{
|
|
|
|
Log_WarningFmt("SCSI subq read returned invalid MSF (got {:02x}:{:02x}:{:02x}, expected {:02d}:{:02d}:{:02d})",
|
|
|
|
subq.absolute_minute_bcd, subq.absolute_second_bcd, subq.absolute_frame_bcd, expected_pos.minute,
|
|
|
|
expected_pos.second, expected_pos.frame);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2024-02-29 14:50:31 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
else // if (mode == SCSIReadMode::None || mode == SCSIReadMode::Raw)
|
|
|
|
{
|
|
|
|
// I guess we could check the sector sync data too...
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-03-23 03:05:54 +00:00
|
|
|
[[maybe_unused]] static bool ShouldTryReadingSubcode()
|
|
|
|
{
|
|
|
|
return !Host::GetBaseBoolSettingValue("CDROM", "IgnoreHostSubcode", false);
|
|
|
|
}
|
|
|
|
|
2023-09-30 05:44:14 +00:00
|
|
|
#if defined(_WIN32)
|
|
|
|
|
2023-11-05 10:36:28 +00:00
|
|
|
// The include order here is critical.
|
|
|
|
// clang-format off
|
|
|
|
#include "common/windows_headers.h"
|
|
|
|
#include <winioctl.h>
|
|
|
|
#include <ntddcdrm.h>
|
|
|
|
#include <ntddscsi.h>
|
|
|
|
// clang-format on
|
|
|
|
|
2021-07-24 08:38:54 +00:00
|
|
|
static u32 BEToU32(const u8* val)
|
|
|
|
{
|
|
|
|
return (static_cast<u32>(val[0]) << 24) | (static_cast<u32>(val[1]) << 16) | (static_cast<u32>(val[2]) << 8) |
|
|
|
|
static_cast<u32>(val[3]);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void U16ToBE(u8* beval, u16 leval)
|
|
|
|
{
|
|
|
|
beval[0] = static_cast<u8>(leval >> 8);
|
|
|
|
beval[1] = static_cast<u8>(leval);
|
|
|
|
}
|
|
|
|
|
2023-11-18 06:21:51 +00:00
|
|
|
namespace {
|
|
|
|
|
2021-07-24 08:38:54 +00:00
|
|
|
class CDImageDeviceWin32 : public CDImage
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
CDImageDeviceWin32();
|
|
|
|
~CDImageDeviceWin32() override;
|
|
|
|
|
2023-08-19 13:40:36 +00:00
|
|
|
bool Open(const char* filename, Error* error);
|
2021-07-24 08:38:54 +00:00
|
|
|
|
|
|
|
bool ReadSubChannelQ(SubChannelQ* subq, const Index& index, LBA lba_in_index) override;
|
|
|
|
bool HasNonStandardSubchannel() const override;
|
|
|
|
|
|
|
|
protected:
|
|
|
|
bool ReadSectorFromIndex(void* buffer, const Index& index, LBA lba_in_index) override;
|
|
|
|
|
|
|
|
private:
|
2024-02-29 14:50:31 +00:00
|
|
|
std::optional<u32> DoSCSICommand(u8 cmd[SCSI_CMD_LENGTH], std::span<u8> out_buffer);
|
|
|
|
std::optional<u32> DoSCSIRead(LBA lba, SCSIReadMode read_mode);
|
|
|
|
bool DoRawRead(LBA lba);
|
|
|
|
bool DoSetSpeed(u32 speed_multiplier);
|
2021-07-24 08:38:54 +00:00
|
|
|
|
2024-02-29 14:50:31 +00:00
|
|
|
bool ReadSectorToBuffer(LBA lba);
|
|
|
|
bool DetermineReadMode(bool try_sptd);
|
2021-07-24 08:38:54 +00:00
|
|
|
|
|
|
|
HANDLE m_hDevice = INVALID_HANDLE_VALUE;
|
|
|
|
|
2024-02-29 14:50:31 +00:00
|
|
|
u32 m_current_lba = ~static_cast<LBA>(0);
|
2021-07-24 08:38:54 +00:00
|
|
|
|
2024-02-29 14:50:31 +00:00
|
|
|
SCSIReadMode m_scsi_read_mode = SCSIReadMode::None;
|
|
|
|
bool m_has_valid_subcode = false;
|
2021-07-24 08:38:54 +00:00
|
|
|
|
|
|
|
std::array<u8, CD_RAW_SECTOR_WITH_SUBCODE_SIZE> m_buffer;
|
|
|
|
};
|
|
|
|
|
2023-11-18 06:21:51 +00:00
|
|
|
} // namespace
|
|
|
|
|
2021-07-24 08:38:54 +00:00
|
|
|
CDImageDeviceWin32::CDImageDeviceWin32() = default;
|
|
|
|
|
|
|
|
CDImageDeviceWin32::~CDImageDeviceWin32()
|
|
|
|
{
|
|
|
|
if (m_hDevice != INVALID_HANDLE_VALUE)
|
|
|
|
CloseHandle(m_hDevice);
|
|
|
|
}
|
|
|
|
|
2023-08-19 13:40:36 +00:00
|
|
|
bool CDImageDeviceWin32::Open(const char* filename, Error* error)
|
2021-07-24 08:38:54 +00:00
|
|
|
{
|
2024-02-29 14:50:31 +00:00
|
|
|
bool try_sptd = true;
|
|
|
|
|
2021-07-24 08:38:54 +00:00
|
|
|
m_filename = filename;
|
|
|
|
m_hDevice = CreateFile(filename, GENERIC_READ | GENERIC_WRITE, FILE_SHARE_READ | FILE_SHARE_WRITE, nullptr,
|
|
|
|
OPEN_EXISTING, 0, NULL);
|
|
|
|
if (m_hDevice == INVALID_HANDLE_VALUE)
|
|
|
|
{
|
|
|
|
m_hDevice = CreateFile(filename, GENERIC_READ, FILE_SHARE_READ, nullptr, OPEN_EXISTING, 0, NULL);
|
|
|
|
if (m_hDevice != INVALID_HANDLE_VALUE)
|
|
|
|
{
|
2024-02-29 14:50:31 +00:00
|
|
|
Log_WarningFmt("Could not open '{}' as read/write, can't use SPTD", filename);
|
|
|
|
try_sptd = false;
|
2021-07-24 08:38:54 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2024-02-29 14:50:31 +00:00
|
|
|
Log_ErrorFmt("CreateFile('{}') failed: %08X", filename, GetLastError());
|
2021-07-24 08:38:54 +00:00
|
|
|
if (error)
|
|
|
|
error->SetWin32(GetLastError());
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Set it to 4x speed. A good balance between readahead and spinning up way too high.
|
2023-09-03 04:30:26 +00:00
|
|
|
static constexpr u32 READ_SPEED_MULTIPLIER = 8;
|
|
|
|
static constexpr u32 READ_SPEED_KBS = (DATA_SECTOR_SIZE * FRAMES_PER_SECOND * READ_SPEED_MULTIPLIER) / 1024;
|
2021-07-24 08:38:54 +00:00
|
|
|
CDROM_SET_SPEED set_speed = {CdromSetSpeed, READ_SPEED_KBS, 0, CdromDefaultRotation};
|
|
|
|
if (!DeviceIoControl(m_hDevice, IOCTL_CDROM_SET_SPEED, &set_speed, sizeof(set_speed), nullptr, 0, nullptr, nullptr))
|
2024-02-29 14:50:31 +00:00
|
|
|
Log_WarningFmt("DeviceIoControl(IOCTL_CDROM_SET_SPEED) failed: {:08X}", GetLastError());
|
2021-07-24 08:38:54 +00:00
|
|
|
|
|
|
|
CDROM_READ_TOC_EX read_toc_ex = {};
|
|
|
|
read_toc_ex.Format = CDROM_READ_TOC_EX_FORMAT_TOC;
|
|
|
|
read_toc_ex.Msf = 0;
|
|
|
|
read_toc_ex.SessionTrack = 1;
|
|
|
|
|
|
|
|
CDROM_TOC toc = {};
|
|
|
|
U16ToBE(toc.Length, sizeof(toc) - sizeof(UCHAR) * 2);
|
|
|
|
|
|
|
|
DWORD bytes_returned;
|
|
|
|
if (!DeviceIoControl(m_hDevice, IOCTL_CDROM_READ_TOC_EX, &read_toc_ex, sizeof(read_toc_ex), &toc, sizeof(toc),
|
|
|
|
&bytes_returned, nullptr) ||
|
|
|
|
toc.LastTrack < toc.FirstTrack)
|
|
|
|
{
|
|
|
|
Log_ErrorPrintf("DeviceIoCtl(IOCTL_CDROM_READ_TOC_EX) failed: %08X", GetLastError());
|
|
|
|
if (error)
|
|
|
|
error->SetWin32(GetLastError());
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
DWORD last_track_address = 0;
|
|
|
|
LBA disc_lba = 0;
|
|
|
|
Log_DevPrintf("FirstTrack=%u, LastTrack=%u", toc.FirstTrack, toc.LastTrack);
|
|
|
|
|
|
|
|
const u32 num_tracks_to_check = (toc.LastTrack - toc.FirstTrack) + 1 + 1;
|
|
|
|
for (u32 track_index = 0; track_index < num_tracks_to_check; track_index++)
|
|
|
|
{
|
|
|
|
const TRACK_DATA& td = toc.TrackData[track_index];
|
|
|
|
const u8 track_num = td.TrackNumber;
|
|
|
|
const DWORD track_address = BEToU32(td.Address);
|
|
|
|
Log_DevPrintf(" [%u]: Num=%02X, Address=%u", track_index, track_num, track_address);
|
|
|
|
|
|
|
|
// fill in the previous track's length
|
|
|
|
if (!m_tracks.empty())
|
|
|
|
{
|
|
|
|
if (track_num < m_tracks.back().track_number)
|
|
|
|
{
|
|
|
|
Log_ErrorPrintf("Invalid TOC, track %u less than %u", track_num, m_tracks.back().track_number);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
const LBA previous_track_length = static_cast<LBA>(track_address - last_track_address);
|
|
|
|
m_tracks.back().length += previous_track_length;
|
|
|
|
m_indices.back().length += previous_track_length;
|
|
|
|
disc_lba += previous_track_length;
|
|
|
|
}
|
|
|
|
|
2021-07-29 04:00:44 +00:00
|
|
|
last_track_address = track_address;
|
2021-07-24 08:38:54 +00:00
|
|
|
if (track_num == LEAD_OUT_TRACK_NUMBER)
|
|
|
|
{
|
|
|
|
AddLeadOutIndex();
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
// precompute subchannel q flags for the whole track
|
|
|
|
SubChannelQ::Control control{};
|
|
|
|
control.bits = td.Adr | (td.Control << 4);
|
|
|
|
|
|
|
|
const LBA track_lba = static_cast<LBA>(track_address);
|
|
|
|
const TrackMode track_mode = control.data ? CDImage::TrackMode::Mode2Raw : CDImage::TrackMode::Audio;
|
|
|
|
|
|
|
|
// TODO: How the hell do we handle pregaps here?
|
2024-03-25 13:15:35 +00:00
|
|
|
const u32 pregap_frames = (track_index == 0) ? (FRAMES_PER_SECOND * 2) : 0;
|
2021-07-24 08:38:54 +00:00
|
|
|
if (pregap_frames > 0)
|
|
|
|
{
|
|
|
|
Index pregap_index = {};
|
|
|
|
pregap_index.start_lba_on_disc = disc_lba;
|
|
|
|
pregap_index.start_lba_in_track = static_cast<LBA>(-static_cast<s32>(pregap_frames));
|
|
|
|
pregap_index.length = pregap_frames;
|
|
|
|
pregap_index.track_number = track_num;
|
|
|
|
pregap_index.index_number = 0;
|
|
|
|
pregap_index.mode = track_mode;
|
2023-11-05 10:36:28 +00:00
|
|
|
pregap_index.submode = CDImage::SubchannelMode::None;
|
2021-07-24 08:38:54 +00:00
|
|
|
pregap_index.control.bits = control.bits;
|
|
|
|
pregap_index.is_pregap = true;
|
|
|
|
m_indices.push_back(pregap_index);
|
|
|
|
disc_lba += pregap_frames;
|
|
|
|
}
|
|
|
|
|
|
|
|
// index 1, will be filled in next iteration
|
|
|
|
if (track_num <= MAX_TRACK_NUMBER)
|
|
|
|
{
|
|
|
|
// add the track itself
|
2023-11-05 10:36:28 +00:00
|
|
|
m_tracks.push_back(
|
|
|
|
Track{track_num, disc_lba, static_cast<u32>(m_indices.size()), 0, track_mode, SubchannelMode::None, control});
|
2021-07-24 08:38:54 +00:00
|
|
|
|
|
|
|
Index index1;
|
|
|
|
index1.start_lba_on_disc = disc_lba;
|
|
|
|
index1.start_lba_in_track = 0;
|
|
|
|
index1.length = 0;
|
|
|
|
index1.track_number = track_num;
|
|
|
|
index1.index_number = 1;
|
|
|
|
index1.file_index = 0;
|
2024-02-29 14:50:31 +00:00
|
|
|
index1.file_sector_size = RAW_SECTOR_SIZE;
|
|
|
|
index1.file_offset = static_cast<u64>(track_lba);
|
2021-07-24 08:38:54 +00:00
|
|
|
index1.mode = track_mode;
|
2023-11-05 10:36:28 +00:00
|
|
|
index1.submode = CDImage::SubchannelMode::None;
|
2021-07-24 08:38:54 +00:00
|
|
|
index1.control.bits = control.bits;
|
|
|
|
index1.is_pregap = false;
|
|
|
|
m_indices.push_back(index1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (m_tracks.empty())
|
|
|
|
{
|
|
|
|
Log_ErrorPrintf("File '%s' contains no tracks", filename);
|
2023-08-19 13:40:36 +00:00
|
|
|
Error::SetString(error, fmt::format("File '{}' contains no tracks", filename));
|
2021-07-24 08:38:54 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
m_lba_count = disc_lba;
|
|
|
|
|
|
|
|
Log_DevPrintf("%u tracks, %u indices, %u lbas", static_cast<u32>(m_tracks.size()), static_cast<u32>(m_indices.size()),
|
|
|
|
static_cast<u32>(m_lba_count));
|
|
|
|
for (u32 i = 0; i < m_tracks.size(); i++)
|
|
|
|
{
|
2021-11-22 13:58:08 +00:00
|
|
|
Log_DevPrintf(" Track %u: Start %u, length %u, mode %u, control 0x%02X", static_cast<u32>(m_tracks[i].track_number),
|
|
|
|
static_cast<u32>(m_tracks[i].start_lba), static_cast<u32>(m_tracks[i].length),
|
2021-07-24 08:38:54 +00:00
|
|
|
static_cast<u32>(m_tracks[i].mode), static_cast<u32>(m_tracks[i].control.bits));
|
|
|
|
}
|
|
|
|
for (u32 i = 0; i < m_indices.size(); i++)
|
|
|
|
{
|
|
|
|
Log_DevPrintf(" Index %u: Track %u, Index %u, Start %u, length %u, file sector size %u, file offset %" PRIu64, i,
|
|
|
|
static_cast<u32>(m_indices[i].track_number), static_cast<u32>(m_indices[i].index_number),
|
|
|
|
static_cast<u32>(m_indices[i].start_lba_on_disc), static_cast<u32>(m_indices[i].length),
|
|
|
|
static_cast<u32>(m_indices[i].file_sector_size), m_indices[i].file_offset);
|
|
|
|
}
|
|
|
|
|
2024-02-29 14:50:31 +00:00
|
|
|
if (!DetermineReadMode(try_sptd))
|
2021-07-24 08:38:54 +00:00
|
|
|
{
|
|
|
|
Log_ErrorPrintf("Could not determine read mode");
|
2023-08-19 13:40:36 +00:00
|
|
|
Error::SetString(error, "Could not determine read mode");
|
2021-07-24 08:38:54 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return Seek(1, Position{0, 0, 0});
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CDImageDeviceWin32::ReadSubChannelQ(SubChannelQ* subq, const Index& index, LBA lba_in_index)
|
|
|
|
{
|
2024-02-29 14:50:31 +00:00
|
|
|
if (index.file_sector_size == 0 || !m_has_valid_subcode)
|
2021-07-24 08:38:54 +00:00
|
|
|
return CDImage::ReadSubChannelQ(subq, index, lba_in_index);
|
|
|
|
|
2024-02-29 14:50:31 +00:00
|
|
|
const LBA offset = static_cast<LBA>(index.file_offset) + lba_in_index;
|
|
|
|
if (m_current_lba != offset && !ReadSectorToBuffer(offset))
|
2021-07-24 08:38:54 +00:00
|
|
|
return false;
|
|
|
|
|
2024-03-25 13:15:35 +00:00
|
|
|
if (m_scsi_read_mode == SCSIReadMode::SubQOnly)
|
|
|
|
{
|
|
|
|
// copy out subq
|
|
|
|
std::memcpy(subq->data.data(), m_buffer.data() + RAW_SECTOR_SIZE, SUBCHANNEL_BYTES_PER_FRAME);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
else // if (m_scsi_read_mode == SCSIReadMode::Full || m_scsi_read_mode == SCSIReadMode::None)
|
|
|
|
{
|
|
|
|
// need to deinterleave the subcode
|
|
|
|
u8 deinterleaved_subcode[ALL_SUBCODE_SIZE];
|
|
|
|
DeinterleaveSubcode(m_buffer.data() + RAW_SECTOR_SIZE, deinterleaved_subcode);
|
|
|
|
|
|
|
|
// P, Q, ...
|
|
|
|
std::memcpy(subq->data.data(), deinterleaved_subcode + SUBCHANNEL_BYTES_PER_FRAME, SUBCHANNEL_BYTES_PER_FRAME);
|
|
|
|
return true;
|
|
|
|
}
|
2021-07-24 08:38:54 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
bool CDImageDeviceWin32::HasNonStandardSubchannel() const
|
|
|
|
{
|
2024-02-29 14:50:31 +00:00
|
|
|
return m_has_valid_subcode;
|
2021-07-24 08:38:54 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
bool CDImageDeviceWin32::ReadSectorFromIndex(void* buffer, const Index& index, LBA lba_in_index)
|
|
|
|
{
|
|
|
|
if (index.file_sector_size == 0)
|
|
|
|
return false;
|
|
|
|
|
2024-02-29 14:50:31 +00:00
|
|
|
const LBA offset = static_cast<LBA>(index.file_offset) + lba_in_index;
|
|
|
|
if (m_current_lba != offset && !ReadSectorToBuffer(offset))
|
2021-07-24 08:38:54 +00:00
|
|
|
return false;
|
|
|
|
|
|
|
|
std::memcpy(buffer, m_buffer.data(), RAW_SECTOR_SIZE);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2024-02-29 14:50:31 +00:00
|
|
|
std::optional<u32> CDImageDeviceWin32::DoSCSICommand(u8 cmd[SCSI_CMD_LENGTH], std::span<u8> out_buffer)
|
2021-07-24 08:38:54 +00:00
|
|
|
{
|
2024-02-29 14:50:31 +00:00
|
|
|
struct SPTDBuffer
|
|
|
|
{
|
|
|
|
SCSI_PASS_THROUGH_DIRECT cmd;
|
|
|
|
u8 sense[20];
|
|
|
|
};
|
|
|
|
SPTDBuffer sptd = {};
|
|
|
|
sptd.cmd.Length = sizeof(sptd.cmd);
|
|
|
|
sptd.cmd.CdbLength = SCSI_CMD_LENGTH;
|
|
|
|
sptd.cmd.SenseInfoLength = sizeof(sptd.sense);
|
|
|
|
sptd.cmd.DataIn = out_buffer.empty() ? SCSI_IOCTL_DATA_UNSPECIFIED : SCSI_IOCTL_DATA_IN;
|
|
|
|
sptd.cmd.DataTransferLength = static_cast<u32>(out_buffer.size());
|
|
|
|
sptd.cmd.TimeOutValue = 10;
|
2024-04-28 04:25:34 +00:00
|
|
|
sptd.cmd.SenseInfoOffset = OFFSETOF(SPTDBuffer, sense);
|
2024-02-29 14:50:31 +00:00
|
|
|
sptd.cmd.DataBuffer = out_buffer.empty() ? nullptr : out_buffer.data();
|
|
|
|
std::memcpy(sptd.cmd.Cdb, cmd, SCSI_CMD_LENGTH);
|
2021-07-24 08:38:54 +00:00
|
|
|
|
2024-02-29 14:50:31 +00:00
|
|
|
DWORD bytes_returned;
|
|
|
|
if (!DeviceIoControl(m_hDevice, IOCTL_SCSI_PASS_THROUGH_DIRECT, &sptd, sizeof(sptd), &sptd, sizeof(sptd),
|
|
|
|
&bytes_returned, nullptr))
|
|
|
|
{
|
|
|
|
Log_ErrorFmt("DeviceIoControl() for SCSI 0x{:02X} failed: {}", cmd[0], GetLastError());
|
|
|
|
return std::nullopt;
|
|
|
|
}
|
2021-07-24 08:38:54 +00:00
|
|
|
|
2024-02-29 14:50:31 +00:00
|
|
|
if (sptd.cmd.ScsiStatus != 0)
|
|
|
|
{
|
|
|
|
Log_ErrorFmt("SCSI command 0x{:02X} failed: {}", cmd[0], sptd.cmd.ScsiStatus);
|
|
|
|
return std::nullopt;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (sptd.cmd.DataTransferLength != out_buffer.size())
|
|
|
|
Log_WarningFmt("Only read {} of {} bytes", sptd.cmd.DataTransferLength, out_buffer.size());
|
|
|
|
|
|
|
|
return sptd.cmd.DataTransferLength;
|
2021-07-24 08:38:54 +00:00
|
|
|
}
|
|
|
|
|
2024-02-29 14:50:31 +00:00
|
|
|
std::optional<u32> CDImageDeviceWin32::DoSCSIRead(LBA lba, SCSIReadMode read_mode)
|
2021-07-24 08:38:54 +00:00
|
|
|
{
|
2024-02-29 14:50:31 +00:00
|
|
|
u8 cmd[SCSI_CMD_LENGTH];
|
|
|
|
FillSCSIReadCommand(cmd, lba, read_mode);
|
|
|
|
|
|
|
|
const u32 size = SCSIReadCommandOutputSize(read_mode);
|
|
|
|
return DoSCSICommand(cmd, std::span<u8>(m_buffer.data(), size));
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CDImageDeviceWin32::DoSetSpeed(u32 speed_multiplier)
|
|
|
|
{
|
|
|
|
u8 cmd[SCSI_CMD_LENGTH];
|
|
|
|
FillSCSISetSpeedCommand(cmd, speed_multiplier);
|
|
|
|
return DoSCSICommand(cmd, {}).has_value();
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CDImageDeviceWin32::DoRawRead(LBA lba)
|
|
|
|
{
|
|
|
|
const DWORD expected_size = RAW_SECTOR_SIZE + ALL_SUBCODE_SIZE;
|
|
|
|
|
|
|
|
RAW_READ_INFO rri;
|
|
|
|
rri.DiskOffset.QuadPart = static_cast<u64>(lba) * 2048;
|
|
|
|
rri.SectorCount = 1;
|
|
|
|
rri.TrackMode = RawWithSubCode;
|
|
|
|
|
|
|
|
DWORD bytes_returned;
|
|
|
|
if (!DeviceIoControl(m_hDevice, IOCTL_CDROM_RAW_READ, &rri, sizeof(rri), m_buffer.data(),
|
|
|
|
static_cast<DWORD>(m_buffer.size()), &bytes_returned, nullptr))
|
2021-07-24 08:38:54 +00:00
|
|
|
{
|
2024-02-29 14:50:31 +00:00
|
|
|
Log_ErrorFmt("DeviceIoControl(IOCTL_CDROM_RAW_READ) for LBA {} failed: {:08X}", lba, GetLastError());
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (bytes_returned != expected_size)
|
|
|
|
Log_WarningFmt("Only read {} of {} bytes", bytes_returned, expected_size);
|
2021-07-24 08:38:54 +00:00
|
|
|
|
2024-02-29 14:50:31 +00:00
|
|
|
return true;
|
|
|
|
}
|
2021-07-24 08:38:54 +00:00
|
|
|
|
2024-02-29 14:50:31 +00:00
|
|
|
bool CDImageDeviceWin32::ReadSectorToBuffer(LBA lba)
|
|
|
|
{
|
|
|
|
if (m_scsi_read_mode != SCSIReadMode::None)
|
|
|
|
{
|
|
|
|
const std::optional<u32> size = DoSCSIRead(lba, m_scsi_read_mode);
|
|
|
|
const u32 expected_size = SCSIReadCommandOutputSize(m_scsi_read_mode);
|
|
|
|
if (size.value_or(0) != expected_size)
|
2021-07-24 08:38:54 +00:00
|
|
|
{
|
2024-02-29 14:50:31 +00:00
|
|
|
Log_ErrorFmt("Read of LBA {} failed: only got {} of {} bytes", lba, size.value(), expected_size);
|
2021-07-24 08:38:54 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2024-02-29 14:50:31 +00:00
|
|
|
if (!DoRawRead(lba))
|
2021-07-24 08:38:54 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2024-02-29 14:50:31 +00:00
|
|
|
m_current_lba = lba;
|
2021-07-24 08:38:54 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2024-02-29 14:50:31 +00:00
|
|
|
bool CDImageDeviceWin32::DetermineReadMode(bool try_sptd)
|
2021-07-24 08:38:54 +00:00
|
|
|
{
|
|
|
|
// Prefer raw reads if we can use them
|
2024-03-25 13:15:35 +00:00
|
|
|
const Index& first_index = m_indices[m_tracks[0].first_index];
|
|
|
|
const LBA track_1_lba = static_cast<LBA>(first_index.file_offset);
|
|
|
|
const LBA track_1_subq_lba = first_index.start_lba_on_disc;
|
2024-03-23 03:05:54 +00:00
|
|
|
const bool check_subcode = ShouldTryReadingSubcode();
|
2021-07-24 08:38:54 +00:00
|
|
|
|
2024-03-25 13:15:35 +00:00
|
|
|
if (try_sptd)
|
2021-07-24 08:38:54 +00:00
|
|
|
{
|
2024-03-25 13:15:35 +00:00
|
|
|
std::optional<u32> transfer_size;
|
|
|
|
|
|
|
|
Log_DevPrint("Trying SCSI read with full subcode...");
|
|
|
|
if (check_subcode && (transfer_size = DoSCSIRead(track_1_lba, SCSIReadMode::Full)).has_value())
|
2021-07-24 08:38:54 +00:00
|
|
|
{
|
2024-03-25 13:15:35 +00:00
|
|
|
if (VerifySCSIReadData(std::span<u8>(m_buffer.data(), transfer_size.value()), SCSIReadMode::Full,
|
|
|
|
track_1_subq_lba))
|
|
|
|
{
|
|
|
|
Log_VerbosePrint("Using SCSI reads with subcode");
|
|
|
|
m_scsi_read_mode = SCSIReadMode::Full;
|
|
|
|
m_has_valid_subcode = true;
|
|
|
|
return true;
|
|
|
|
}
|
2021-07-24 08:38:54 +00:00
|
|
|
}
|
|
|
|
|
2024-03-25 13:15:35 +00:00
|
|
|
Log_WarningPrint("Full subcode failed, trying SCSI read with only subq...");
|
|
|
|
if (check_subcode && (transfer_size = DoSCSIRead(track_1_lba, SCSIReadMode::SubQOnly)).has_value())
|
2024-02-29 14:50:31 +00:00
|
|
|
{
|
2024-03-25 13:15:35 +00:00
|
|
|
if (VerifySCSIReadData(std::span<u8>(m_buffer.data(), transfer_size.value()), SCSIReadMode::SubQOnly,
|
|
|
|
track_1_subq_lba))
|
|
|
|
{
|
|
|
|
Log_VerbosePrint("Using SCSI reads with subq only");
|
|
|
|
m_scsi_read_mode = SCSIReadMode::SubQOnly;
|
|
|
|
m_has_valid_subcode = true;
|
|
|
|
return true;
|
|
|
|
}
|
2024-02-29 14:50:31 +00:00
|
|
|
}
|
2021-07-24 08:38:54 +00:00
|
|
|
|
2024-03-25 13:15:35 +00:00
|
|
|
// As a last ditch effort, try SCSI without subcode.
|
|
|
|
Log_WarningPrint("Subq only failed failed, trying SCSI without subcode...");
|
|
|
|
if ((transfer_size = DoSCSIRead(track_1_lba, SCSIReadMode::Raw)).has_value())
|
2021-07-24 08:38:54 +00:00
|
|
|
{
|
2024-03-25 13:15:35 +00:00
|
|
|
if (VerifySCSIReadData(std::span<u8>(m_buffer.data(), transfer_size.value()), SCSIReadMode::Raw,
|
|
|
|
track_1_subq_lba))
|
|
|
|
{
|
|
|
|
Log_WarningPrint("Using SCSI raw reads, libcrypt games will not run correctly");
|
|
|
|
m_scsi_read_mode = SCSIReadMode::Raw;
|
|
|
|
m_has_valid_subcode = false;
|
|
|
|
return true;
|
|
|
|
}
|
2021-07-24 08:38:54 +00:00
|
|
|
}
|
2024-02-29 14:50:31 +00:00
|
|
|
}
|
|
|
|
|
2024-03-25 13:15:35 +00:00
|
|
|
Log_WarningPrint("SCSI reads failed, trying raw read...");
|
|
|
|
if (DoRawRead(track_1_lba))
|
2024-02-29 14:50:31 +00:00
|
|
|
{
|
2024-03-25 13:15:35 +00:00
|
|
|
// verify subcode
|
|
|
|
if (VerifySCSIReadData(std::span<u8>(m_buffer.data(), SCSIReadCommandOutputSize(SCSIReadMode::Full)),
|
|
|
|
SCSIReadMode::Full, track_1_subq_lba))
|
2021-07-24 08:38:54 +00:00
|
|
|
{
|
2024-03-25 13:15:35 +00:00
|
|
|
Log_VerbosePrint("Using raw reads with full subcode");
|
|
|
|
m_scsi_read_mode = SCSIReadMode::None;
|
|
|
|
m_has_valid_subcode = true;
|
2024-02-29 14:50:31 +00:00
|
|
|
return true;
|
2021-07-24 08:38:54 +00:00
|
|
|
}
|
|
|
|
|
2024-02-29 14:50:31 +00:00
|
|
|
Log_WarningPrint("Using raw reads without subcode, libcrypt games will not run correctly");
|
|
|
|
m_scsi_read_mode = SCSIReadMode::None;
|
|
|
|
m_has_valid_subcode = false;
|
2021-07-24 08:38:54 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2024-02-29 14:50:31 +00:00
|
|
|
Log_ErrorPrint("No read modes were successful, cannot use device.");
|
2021-07-24 08:38:54 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2023-08-19 13:40:36 +00:00
|
|
|
std::unique_ptr<CDImage> CDImage::OpenDeviceImage(const char* filename, Error* error)
|
2021-07-24 08:38:54 +00:00
|
|
|
{
|
|
|
|
std::unique_ptr<CDImageDeviceWin32> image = std::make_unique<CDImageDeviceWin32>();
|
|
|
|
if (!image->Open(filename, error))
|
|
|
|
return {};
|
|
|
|
|
|
|
|
return image;
|
|
|
|
}
|
|
|
|
|
|
|
|
std::vector<std::pair<std::string, std::string>> CDImage::GetDeviceList()
|
|
|
|
{
|
|
|
|
std::vector<std::pair<std::string, std::string>> ret;
|
|
|
|
|
|
|
|
char buf[256];
|
|
|
|
if (GetLogicalDriveStringsA(sizeof(buf), buf) != 0)
|
|
|
|
{
|
|
|
|
const char* ptr = buf;
|
|
|
|
while (*ptr != '\0')
|
|
|
|
{
|
|
|
|
std::size_t len = std::strlen(ptr);
|
|
|
|
const DWORD type = GetDriveTypeA(ptr);
|
|
|
|
if (type != DRIVE_CDROM)
|
|
|
|
{
|
|
|
|
ptr += len + 1u;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Drop the trailing slash.
|
|
|
|
const std::size_t append_len = (ptr[len - 1] == '\\') ? (len - 1) : len;
|
|
|
|
|
|
|
|
std::string path;
|
|
|
|
path.append("\\\\.\\");
|
|
|
|
path.append(ptr, append_len);
|
|
|
|
|
|
|
|
std::string name(ptr, append_len);
|
|
|
|
|
|
|
|
ret.emplace_back(std::move(path), std::move(name));
|
|
|
|
|
|
|
|
ptr += len + 1u;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CDImage::IsDeviceName(const char* filename)
|
|
|
|
{
|
2023-12-13 11:15:07 +00:00
|
|
|
return std::string_view(filename).starts_with("\\\\.\\");
|
2021-07-24 08:38:54 +00:00
|
|
|
}
|
|
|
|
|
2024-02-29 14:50:31 +00:00
|
|
|
#elif defined(__linux__) && !defined(__ANDROID__)
|
|
|
|
|
|
|
|
#include <fcntl.h>
|
|
|
|
#include <libudev.h>
|
|
|
|
#include <linux/cdrom.h>
|
|
|
|
#include <scsi/sg.h>
|
|
|
|
#include <sys/ioctl.h>
|
|
|
|
#include <unistd.h>
|
|
|
|
|
|
|
|
namespace {
|
|
|
|
|
|
|
|
class CDImageDeviceLinux : public CDImage
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
CDImageDeviceLinux();
|
|
|
|
~CDImageDeviceLinux() override;
|
|
|
|
|
|
|
|
bool Open(const char* filename, Error* error);
|
|
|
|
|
|
|
|
bool ReadSubChannelQ(SubChannelQ* subq, const Index& index, LBA lba_in_index) override;
|
|
|
|
bool HasNonStandardSubchannel() const override;
|
|
|
|
|
|
|
|
protected:
|
|
|
|
bool ReadSectorFromIndex(void* buffer, const Index& index, LBA lba_in_index) override;
|
|
|
|
|
|
|
|
private:
|
|
|
|
// Raw reads use an offset of 00:02:00
|
|
|
|
static constexpr LBA RAW_READ_OFFSET = 2 * FRAMES_PER_SECOND;
|
|
|
|
|
|
|
|
bool ReadSectorToBuffer(LBA lba);
|
|
|
|
bool DetermineReadMode(Error* error);
|
|
|
|
|
|
|
|
std::optional<u32> DoSCSICommand(u8 cmd[SCSI_CMD_LENGTH], std::span<u8> out_buffer);
|
|
|
|
std::optional<u32> DoSCSIRead(LBA lba, SCSIReadMode read_mode);
|
|
|
|
bool DoRawRead(LBA lba);
|
|
|
|
bool DoSetSpeed(u32 speed_multiplier);
|
|
|
|
|
|
|
|
int m_fd = -1;
|
|
|
|
LBA m_current_lba = ~static_cast<LBA>(0);
|
|
|
|
|
|
|
|
SCSIReadMode m_scsi_read_mode = SCSIReadMode::None;
|
|
|
|
|
|
|
|
std::array<u8, RAW_SECTOR_SIZE + ALL_SUBCODE_SIZE> m_buffer;
|
|
|
|
};
|
|
|
|
|
|
|
|
} // namespace
|
|
|
|
|
|
|
|
CDImageDeviceLinux::CDImageDeviceLinux() = default;
|
|
|
|
|
|
|
|
CDImageDeviceLinux::~CDImageDeviceLinux()
|
|
|
|
{
|
|
|
|
if (m_fd >= 0)
|
|
|
|
close(m_fd);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CDImageDeviceLinux::Open(const char* filename, Error* error)
|
|
|
|
{
|
|
|
|
m_filename = filename;
|
|
|
|
|
|
|
|
m_fd = open(filename, O_RDONLY);
|
|
|
|
if (m_fd < 0)
|
|
|
|
{
|
|
|
|
Error::SetErrno(error, "Failed to open device: ", errno);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Set it to 4x speed. A good balance between readahead and spinning up way too high.
|
|
|
|
const int read_speed = 4;
|
|
|
|
if (!DoSetSpeed(read_speed) && ioctl(m_fd, CDROM_SELECT_SPEED, &read_speed) != 0)
|
|
|
|
Log_WarningFmt("ioctl(CDROM_SELECT_SPEED) failed: {}", errno);
|
|
|
|
|
|
|
|
// Read ToC
|
|
|
|
cdrom_tochdr toc_hdr = {};
|
|
|
|
if (ioctl(m_fd, CDROMREADTOCHDR, &toc_hdr) != 0)
|
|
|
|
{
|
|
|
|
Error::SetErrno(error, "ioctl(CDROMREADTOCHDR) failed: ", errno);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
Log_DevFmt("FirstTrack={}, LastTrack={}", toc_hdr.cdth_trk0, toc_hdr.cdth_trk1);
|
|
|
|
if (toc_hdr.cdth_trk1 < toc_hdr.cdth_trk0)
|
|
|
|
{
|
|
|
|
Error::SetStringFmt(error, "Last track {} is before first track {}", toc_hdr.cdth_trk1, toc_hdr.cdth_trk0);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
cdrom_tocentry toc_ent = {};
|
|
|
|
toc_ent.cdte_format = CDROM_LBA;
|
|
|
|
|
|
|
|
LBA disc_lba = 0;
|
|
|
|
int last_track_lba = 0;
|
|
|
|
const u32 num_tracks_to_check = (toc_hdr.cdth_trk1 - toc_hdr.cdth_trk0) + 1;
|
|
|
|
for (u32 track_index = 0; track_index < num_tracks_to_check; track_index++)
|
|
|
|
{
|
|
|
|
const u32 track_num = toc_hdr.cdth_trk0 + track_index;
|
|
|
|
|
|
|
|
toc_ent.cdte_track = static_cast<u8>(track_num);
|
|
|
|
if (ioctl(m_fd, CDROMREADTOCENTRY, &toc_ent) < 0)
|
|
|
|
{
|
|
|
|
Error::SetErrno(error, "ioctl(CDROMREADTOCENTRY) failed: ", errno);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
Log_DevFmt(" [{}]: Num={}, LBA={}", track_index, track_num, toc_ent.cdte_addr.lba);
|
|
|
|
|
|
|
|
// fill in the previous track's length
|
|
|
|
if (!m_tracks.empty())
|
|
|
|
{
|
|
|
|
if (track_num < m_tracks.back().track_number)
|
|
|
|
{
|
|
|
|
Log_ErrorPrintf("Invalid TOC, track %u less than %u", track_num, m_tracks.back().track_number);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
const LBA previous_track_length = static_cast<LBA>(toc_ent.cdte_addr.lba - last_track_lba);
|
|
|
|
m_tracks.back().length += previous_track_length;
|
|
|
|
m_indices.back().length += previous_track_length;
|
|
|
|
disc_lba += previous_track_length;
|
|
|
|
}
|
|
|
|
|
|
|
|
last_track_lba = toc_ent.cdte_addr.lba;
|
|
|
|
|
|
|
|
// precompute subchannel q flags for the whole track
|
|
|
|
SubChannelQ::Control control{};
|
|
|
|
control.bits = toc_ent.cdte_adr | (toc_ent.cdte_ctrl << 4);
|
|
|
|
|
|
|
|
const LBA track_lba = static_cast<LBA>(toc_ent.cdte_addr.lba);
|
|
|
|
const TrackMode track_mode = control.data ? CDImage::TrackMode::Mode2Raw : CDImage::TrackMode::Audio;
|
|
|
|
|
|
|
|
// TODO: How the hell do we handle pregaps here?
|
2024-03-25 13:15:35 +00:00
|
|
|
const u32 pregap_frames = (track_index == 0) ? 150 : 0;
|
2024-02-29 14:50:31 +00:00
|
|
|
if (pregap_frames > 0)
|
|
|
|
{
|
|
|
|
Index pregap_index = {};
|
|
|
|
pregap_index.start_lba_on_disc = disc_lba;
|
|
|
|
pregap_index.start_lba_in_track = static_cast<LBA>(-static_cast<s32>(pregap_frames));
|
|
|
|
pregap_index.length = pregap_frames;
|
|
|
|
pregap_index.track_number = track_num;
|
|
|
|
pregap_index.index_number = 0;
|
|
|
|
pregap_index.mode = track_mode;
|
|
|
|
pregap_index.submode = CDImage::SubchannelMode::None;
|
|
|
|
pregap_index.control.bits = control.bits;
|
|
|
|
pregap_index.is_pregap = true;
|
|
|
|
m_indices.push_back(pregap_index);
|
|
|
|
disc_lba += pregap_frames;
|
|
|
|
}
|
|
|
|
|
|
|
|
// index 1, will be filled in next iteration
|
|
|
|
if (track_num <= MAX_TRACK_NUMBER)
|
|
|
|
{
|
|
|
|
// add the track itself
|
|
|
|
m_tracks.push_back(
|
|
|
|
Track{track_num, disc_lba, static_cast<u32>(m_indices.size()), 0, track_mode, SubchannelMode::None, control});
|
|
|
|
|
|
|
|
Index index1;
|
|
|
|
index1.start_lba_on_disc = disc_lba;
|
|
|
|
index1.start_lba_in_track = 0;
|
|
|
|
index1.length = 0;
|
|
|
|
index1.track_number = track_num;
|
|
|
|
index1.index_number = 1;
|
|
|
|
index1.file_index = 0;
|
|
|
|
index1.file_sector_size = RAW_SECTOR_SIZE;
|
|
|
|
index1.file_offset = static_cast<u64>(track_lba);
|
|
|
|
index1.mode = track_mode;
|
|
|
|
index1.submode = CDImage::SubchannelMode::None;
|
|
|
|
index1.control.bits = control.bits;
|
|
|
|
index1.is_pregap = false;
|
|
|
|
m_indices.push_back(index1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (m_tracks.empty())
|
|
|
|
{
|
|
|
|
Log_ErrorPrintf("File '%s' contains no tracks", filename);
|
|
|
|
Error::SetString(error, fmt::format("File '{}' contains no tracks", filename));
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Read lead-out.
|
|
|
|
toc_ent.cdte_track = 0xAA;
|
|
|
|
if (ioctl(m_fd, CDROMREADTOCENTRY, &toc_ent) < 0)
|
|
|
|
{
|
|
|
|
Error::SetErrno(error, "ioctl(CDROMREADTOCENTRY) for lead-out failed: ", errno);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (toc_ent.cdte_addr.lba < last_track_lba)
|
|
|
|
{
|
|
|
|
Error::SetStringFmt(error, "Lead-out LBA {} is less than last track {}", toc_ent.cdte_addr.lba, last_track_lba);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Fill last track length from lead-out.
|
|
|
|
const LBA previous_track_length = static_cast<LBA>(toc_ent.cdte_addr.lba - last_track_lba);
|
|
|
|
m_tracks.back().length += previous_track_length;
|
|
|
|
m_indices.back().length += previous_track_length;
|
|
|
|
disc_lba += previous_track_length;
|
|
|
|
|
|
|
|
// And add the lead-out itself.
|
|
|
|
AddLeadOutIndex();
|
|
|
|
|
|
|
|
m_lba_count = disc_lba;
|
|
|
|
|
|
|
|
Log_DevPrintf("%u tracks, %u indices, %u lbas", static_cast<u32>(m_tracks.size()), static_cast<u32>(m_indices.size()),
|
|
|
|
static_cast<u32>(m_lba_count));
|
|
|
|
for (u32 i = 0; i < m_tracks.size(); i++)
|
|
|
|
{
|
|
|
|
Log_DevPrintf(" Track %u: Start %u, length %u, mode %u, control 0x%02X", static_cast<u32>(m_tracks[i].track_number),
|
|
|
|
static_cast<u32>(m_tracks[i].start_lba), static_cast<u32>(m_tracks[i].length),
|
|
|
|
static_cast<u32>(m_tracks[i].mode), static_cast<u32>(m_tracks[i].control.bits));
|
|
|
|
}
|
|
|
|
for (u32 i = 0; i < m_indices.size(); i++)
|
|
|
|
{
|
|
|
|
Log_DevPrintf(" Index %u: Track %u, Index %u, Start %u, length %u, file sector size %u, file offset %" PRIu64, i,
|
|
|
|
static_cast<u32>(m_indices[i].track_number), static_cast<u32>(m_indices[i].index_number),
|
|
|
|
static_cast<u32>(m_indices[i].start_lba_on_disc), static_cast<u32>(m_indices[i].length),
|
|
|
|
static_cast<u32>(m_indices[i].file_sector_size), m_indices[i].file_offset);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!DetermineReadMode(error))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
return Seek(1, Position{0, 0, 0});
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CDImageDeviceLinux::ReadSubChannelQ(SubChannelQ* subq, const Index& index, LBA lba_in_index)
|
|
|
|
{
|
|
|
|
if (index.file_sector_size == 0 || m_scsi_read_mode < SCSIReadMode::Full)
|
|
|
|
return CDImage::ReadSubChannelQ(subq, index, lba_in_index);
|
|
|
|
|
|
|
|
const LBA disc_lba = static_cast<LBA>(index.file_offset) + lba_in_index;
|
|
|
|
if (m_current_lba != disc_lba && !ReadSectorToBuffer(disc_lba))
|
|
|
|
return false;
|
|
|
|
|
2024-03-25 13:15:35 +00:00
|
|
|
if (m_scsi_read_mode == SCSIReadMode::SubQOnly)
|
|
|
|
{
|
|
|
|
// copy out subq
|
|
|
|
std::memcpy(subq->data.data(), m_buffer.data() + RAW_SECTOR_SIZE, SUBCHANNEL_BYTES_PER_FRAME);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
else // if (m_scsi_read_mode == SCSIReadMode::Full)
|
|
|
|
{
|
|
|
|
// need to deinterleave the subcode
|
|
|
|
u8 deinterleaved_subcode[ALL_SUBCODE_SIZE];
|
|
|
|
DeinterleaveSubcode(m_buffer.data() + RAW_SECTOR_SIZE, deinterleaved_subcode);
|
|
|
|
|
|
|
|
// P, Q, ...
|
|
|
|
std::memcpy(subq->data.data(), deinterleaved_subcode + SUBCHANNEL_BYTES_PER_FRAME, SUBCHANNEL_BYTES_PER_FRAME);
|
|
|
|
return true;
|
|
|
|
}
|
2024-02-29 14:50:31 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
bool CDImageDeviceLinux::HasNonStandardSubchannel() const
|
|
|
|
{
|
|
|
|
// Can only read subchannel through SPTD.
|
|
|
|
return m_scsi_read_mode >= SCSIReadMode::Full;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CDImageDeviceLinux::ReadSectorFromIndex(void* buffer, const Index& index, LBA lba_in_index)
|
|
|
|
{
|
|
|
|
if (index.file_sector_size == 0)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
const LBA disc_lba = static_cast<LBA>(index.file_offset) + lba_in_index;
|
|
|
|
if (m_current_lba != disc_lba && !ReadSectorToBuffer(disc_lba))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
std::memcpy(buffer, m_buffer.data(), RAW_SECTOR_SIZE);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
std::optional<u32> CDImageDeviceLinux::DoSCSICommand(u8 cmd[SCSI_CMD_LENGTH], std::span<u8> out_buffer)
|
|
|
|
{
|
|
|
|
sg_io_hdr_t hdr;
|
|
|
|
std::memset(&hdr, 0, sizeof(hdr));
|
|
|
|
hdr.cmd_len = SCSI_CMD_LENGTH;
|
|
|
|
hdr.interface_id = 'S';
|
|
|
|
hdr.dxfer_direction = out_buffer.empty() ? SG_DXFER_NONE : SG_DXFER_FROM_DEV;
|
|
|
|
hdr.mx_sb_len = 0;
|
|
|
|
hdr.dxfer_len = static_cast<u32>(out_buffer.size());
|
|
|
|
hdr.dxferp = out_buffer.empty() ? nullptr : out_buffer.data();
|
|
|
|
hdr.cmdp = cmd;
|
|
|
|
hdr.timeout = 10000; // milliseconds
|
|
|
|
|
|
|
|
if (ioctl(m_fd, SG_IO, &hdr) != 0)
|
|
|
|
{
|
|
|
|
Log_ErrorFmt("ioctl(SG_IO) for command {:02X} failed: {}", cmd[0], errno);
|
|
|
|
return std::nullopt;
|
|
|
|
}
|
|
|
|
else if (hdr.status != 0)
|
|
|
|
{
|
|
|
|
Log_ErrorFmt("SCSI command {:02X} failed with status {}", cmd[0], hdr.status);
|
|
|
|
return std::nullopt;
|
|
|
|
}
|
|
|
|
|
|
|
|
return hdr.dxfer_len;
|
|
|
|
}
|
|
|
|
|
|
|
|
std::optional<u32> CDImageDeviceLinux::DoSCSIRead(LBA lba, SCSIReadMode read_mode)
|
|
|
|
{
|
|
|
|
u8 cmd[SCSI_CMD_LENGTH];
|
|
|
|
FillSCSIReadCommand(cmd, lba, read_mode);
|
|
|
|
|
|
|
|
const u32 size = SCSIReadCommandOutputSize(read_mode);
|
|
|
|
return DoSCSICommand(cmd, std::span<u8>(m_buffer.data(), size));
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CDImageDeviceLinux::DoSetSpeed(u32 speed_multiplier)
|
|
|
|
{
|
|
|
|
u8 cmd[SCSI_CMD_LENGTH];
|
|
|
|
FillSCSISetSpeedCommand(cmd, speed_multiplier);
|
|
|
|
return DoSCSICommand(cmd, {}).has_value();
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CDImageDeviceLinux::DoRawRead(LBA lba)
|
|
|
|
{
|
|
|
|
const Position msf = Position::FromLBA(lba + RAW_READ_OFFSET);
|
|
|
|
std::memcpy(m_buffer.data(), &msf, sizeof(msf));
|
|
|
|
if (ioctl(m_fd, CDROMREADRAW, m_buffer.data()) != 0)
|
|
|
|
{
|
|
|
|
Log_ErrorFmt("CDROMREADRAW for LBA {} (MSF {}:{}:{}) failed: {}", lba, msf.minute, msf.second, msf.frame, errno);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CDImageDeviceLinux::ReadSectorToBuffer(LBA lba)
|
|
|
|
{
|
|
|
|
if (m_scsi_read_mode != SCSIReadMode::None)
|
|
|
|
{
|
|
|
|
const std::optional<u32> size = DoSCSIRead(lba, m_scsi_read_mode);
|
|
|
|
const u32 expected_size = SCSIReadCommandOutputSize(m_scsi_read_mode);
|
|
|
|
if (size.value_or(0) != expected_size)
|
|
|
|
{
|
|
|
|
Log_ErrorFmt("Read of LBA {} failed: only got {} of {} bytes", lba, size.value(), expected_size);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (!DoRawRead(lba))
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
m_current_lba = lba;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CDImageDeviceLinux::DetermineReadMode(Error* error)
|
|
|
|
{
|
|
|
|
const LBA track_1_lba = static_cast<LBA>(m_indices[m_tracks[0].first_index].file_offset);
|
2024-03-23 03:05:54 +00:00
|
|
|
const LBA track_1_subq_lba = track_1_lba + FRAMES_PER_SECOND * 2;
|
|
|
|
const bool check_subcode = ShouldTryReadingSubcode();
|
2024-02-29 14:50:31 +00:00
|
|
|
std::optional<u32> transfer_size;
|
|
|
|
|
|
|
|
Log_DevPrint("Trying SCSI read with full subcode...");
|
2024-03-23 03:05:54 +00:00
|
|
|
if (check_subcode && (transfer_size = DoSCSIRead(track_1_lba, SCSIReadMode::Full)).has_value())
|
2024-02-29 14:50:31 +00:00
|
|
|
{
|
2024-03-23 03:05:54 +00:00
|
|
|
if (VerifySCSIReadData(std::span<u8>(m_buffer.data(), transfer_size.value()), SCSIReadMode::Full, track_1_subq_lba))
|
2024-02-29 14:50:31 +00:00
|
|
|
{
|
|
|
|
Log_VerbosePrint("Using SCSI reads with subcode");
|
|
|
|
m_scsi_read_mode = SCSIReadMode::Full;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Log_WarningPrint("Full subcode failed, trying SCSI read with only subq...");
|
2024-03-23 03:05:54 +00:00
|
|
|
if (check_subcode && (transfer_size = DoSCSIRead(track_1_lba, SCSIReadMode::SubQOnly)).has_value())
|
2024-02-29 14:50:31 +00:00
|
|
|
{
|
2024-03-23 03:05:54 +00:00
|
|
|
if (VerifySCSIReadData(std::span<u8>(m_buffer.data(), transfer_size.value()), SCSIReadMode::SubQOnly,
|
|
|
|
track_1_subq_lba))
|
2024-02-29 14:50:31 +00:00
|
|
|
{
|
|
|
|
Log_VerbosePrint("Using SCSI reads with subq only");
|
|
|
|
m_scsi_read_mode = SCSIReadMode::SubQOnly;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Log_WarningPrint("SCSI subcode reads failed, trying CDROMREADRAW...");
|
|
|
|
if (DoRawRead(track_1_lba))
|
|
|
|
{
|
|
|
|
Log_WarningPrint("Using CDROMREADRAW, libcrypt games will not run correctly");
|
|
|
|
m_scsi_read_mode = SCSIReadMode::None;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
// As a last ditch effort, try SCSI without subcode.
|
|
|
|
Log_WarningPrint("CDROMREADRAW failed, trying SCSI without subcode...");
|
|
|
|
if ((transfer_size = DoSCSIRead(track_1_lba, SCSIReadMode::Raw)).has_value())
|
|
|
|
{
|
2024-03-23 03:05:54 +00:00
|
|
|
if (VerifySCSIReadData(std::span<u8>(m_buffer.data(), transfer_size.value()), SCSIReadMode::Raw, track_1_subq_lba))
|
2024-02-29 14:50:31 +00:00
|
|
|
{
|
|
|
|
Log_WarningPrint("Using SCSI raw reads, libcrypt games will not run correctly");
|
|
|
|
m_scsi_read_mode = SCSIReadMode::Raw;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Log_ErrorPrint("No read modes were successful, cannot use device.");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
std::unique_ptr<CDImage> CDImage::OpenDeviceImage(const char* filename, Error* error)
|
|
|
|
{
|
|
|
|
std::unique_ptr<CDImageDeviceLinux> image = std::make_unique<CDImageDeviceLinux>();
|
|
|
|
if (!image->Open(filename, error))
|
|
|
|
return {};
|
|
|
|
|
|
|
|
return image;
|
|
|
|
}
|
|
|
|
|
|
|
|
std::vector<std::pair<std::string, std::string>> CDImage::GetDeviceList()
|
|
|
|
{
|
|
|
|
std::vector<std::pair<std::string, std::string>> ret;
|
|
|
|
|
|
|
|
// borrowed from PCSX2
|
|
|
|
udev* udev_context = udev_new();
|
|
|
|
if (udev_context)
|
|
|
|
{
|
|
|
|
udev_enumerate* enumerate = udev_enumerate_new(udev_context);
|
|
|
|
if (enumerate)
|
|
|
|
{
|
|
|
|
udev_enumerate_add_match_subsystem(enumerate, "block");
|
|
|
|
udev_enumerate_add_match_property(enumerate, "ID_CDROM", "1");
|
|
|
|
udev_enumerate_scan_devices(enumerate);
|
|
|
|
udev_list_entry* devices = udev_enumerate_get_list_entry(enumerate);
|
|
|
|
|
|
|
|
udev_list_entry* dev_list_entry;
|
|
|
|
udev_list_entry_foreach(dev_list_entry, devices)
|
|
|
|
{
|
|
|
|
const char* path = udev_list_entry_get_name(dev_list_entry);
|
|
|
|
udev_device* device = udev_device_new_from_syspath(udev_context, path);
|
|
|
|
const char* devnode = udev_device_get_devnode(device);
|
|
|
|
if (devnode)
|
|
|
|
ret.emplace_back(devnode, devnode);
|
|
|
|
udev_device_unref(device);
|
|
|
|
}
|
|
|
|
udev_enumerate_unref(enumerate);
|
|
|
|
}
|
|
|
|
udev_unref(udev_context);
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CDImage::IsDeviceName(const char* filename)
|
|
|
|
{
|
|
|
|
if (!std::string_view(filename).starts_with("/dev"))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
const int fd = open(filename, O_RDONLY | O_NONBLOCK);
|
|
|
|
if (fd < 0)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
const bool is_cdrom = (ioctl(fd, CDROM_GET_CAPABILITY, 0) >= 0);
|
|
|
|
close(fd);
|
|
|
|
return is_cdrom;
|
|
|
|
}
|
|
|
|
|
2024-05-19 12:47:01 +00:00
|
|
|
#elif defined(__APPLE__)
|
|
|
|
|
|
|
|
#include <CoreFoundation/CoreFoundation.h>
|
|
|
|
#include <IOKit/IOBSD.h>
|
|
|
|
#include <IOKit/IOKitLib.h>
|
|
|
|
#include <IOKit/storage/IOCDMedia.h>
|
|
|
|
#include <IOKit/storage/IOCDMediaBSDClient.h>
|
|
|
|
#include <IOKit/storage/IODVDMedia.h>
|
|
|
|
#include <IOKit/storage/IODVDMediaBSDClient.h>
|
|
|
|
#include <IOKit/storage/IOMedia.h>
|
|
|
|
#include <fcntl.h>
|
|
|
|
#include <sys/ioctl.h>
|
|
|
|
#include <unistd.h>
|
|
|
|
|
|
|
|
namespace {
|
|
|
|
|
|
|
|
class CDImageDeviceMacOS : public CDImage
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
CDImageDeviceMacOS();
|
|
|
|
~CDImageDeviceMacOS() override;
|
|
|
|
|
|
|
|
bool Open(const char* filename, Error* error);
|
|
|
|
|
|
|
|
bool ReadSubChannelQ(SubChannelQ* subq, const Index& index, LBA lba_in_index) override;
|
|
|
|
bool HasNonStandardSubchannel() const override;
|
|
|
|
|
|
|
|
protected:
|
|
|
|
bool ReadSectorFromIndex(void* buffer, const Index& index, LBA lba_in_index) override;
|
|
|
|
|
|
|
|
private:
|
|
|
|
// Raw reads should subtract 00:02:00.
|
|
|
|
static constexpr u32 RAW_READ_OFFSET = 2 * FRAMES_PER_SECOND;
|
|
|
|
|
|
|
|
bool ReadSectorToBuffer(LBA lba);
|
|
|
|
bool DetermineReadMode(Error* error);
|
|
|
|
|
|
|
|
bool DoSetSpeed(u32 speed_multiplier);
|
|
|
|
|
|
|
|
int m_fd = -1;
|
|
|
|
LBA m_current_lba = ~static_cast<LBA>(0);
|
|
|
|
|
|
|
|
SCSIReadMode m_read_mode = SCSIReadMode::None;
|
|
|
|
|
|
|
|
std::array<u8, RAW_SECTOR_SIZE + ALL_SUBCODE_SIZE> m_buffer;
|
|
|
|
};
|
|
|
|
|
|
|
|
} // namespace
|
|
|
|
|
|
|
|
static io_service_t GetDeviceMediaService(std::string_view devname)
|
|
|
|
{
|
|
|
|
std::string_view filename = Path::GetFileName(devname);
|
|
|
|
if (filename.starts_with("r"))
|
|
|
|
filename = filename.substr(1);
|
|
|
|
if (filename.empty())
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
TinyString rdevname(filename);
|
|
|
|
io_iterator_t iterator;
|
|
|
|
kern_return_t ret = IOServiceGetMatchingServices(0, IOBSDNameMatching(0, 0, rdevname.c_str()), &iterator);
|
|
|
|
if (ret != KERN_SUCCESS)
|
|
|
|
{
|
|
|
|
Log_ErrorFmt("IOServiceGetMatchingService() returned {}", ret);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
// search up the heirarchy
|
|
|
|
for (;;)
|
|
|
|
{
|
|
|
|
io_service_t service = IOIteratorNext(iterator);
|
|
|
|
IOObjectRelease(iterator);
|
|
|
|
|
|
|
|
if (IOObjectConformsTo(service, kIOCDMediaClass) || IOObjectConformsTo(service, kIODVDMediaClass))
|
|
|
|
{
|
|
|
|
return service;
|
|
|
|
}
|
|
|
|
|
|
|
|
ret = IORegistryEntryGetParentIterator(service, kIOServicePlane, &iterator);
|
|
|
|
IOObjectRelease(service);
|
|
|
|
if (ret != KERN_SUCCESS)
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
CDImageDeviceMacOS::CDImageDeviceMacOS() = default;
|
|
|
|
|
|
|
|
CDImageDeviceMacOS::~CDImageDeviceMacOS()
|
|
|
|
{
|
|
|
|
if (m_fd >= 0)
|
|
|
|
close(m_fd);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CDImageDeviceMacOS::Open(const char* filename, Error* error)
|
|
|
|
{
|
|
|
|
m_filename = filename;
|
|
|
|
|
|
|
|
m_fd = open(filename, O_RDONLY);
|
|
|
|
if (m_fd < 0)
|
|
|
|
{
|
|
|
|
Error::SetErrno(error, "Failed to open device: ", errno);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
constexpr int read_speed = 8;
|
|
|
|
DoSetSpeed(read_speed);
|
|
|
|
|
|
|
|
// Read ToC
|
|
|
|
static constexpr u32 TOC_BUFFER_SIZE = 2048;
|
|
|
|
std::unique_ptr<CDTOC, void (*)(void*)> toc(static_cast<CDTOC*>(std::malloc(TOC_BUFFER_SIZE)), std::free);
|
|
|
|
dk_cd_read_toc_t toc_read = {};
|
|
|
|
toc_read.format = kCDTOCFormatTOC;
|
|
|
|
toc_read.formatAsTime = true;
|
|
|
|
toc_read.buffer = toc.get();
|
|
|
|
toc_read.bufferLength = TOC_BUFFER_SIZE;
|
|
|
|
if (ioctl(m_fd, DKIOCCDREADTOC, &toc_read) != 0)
|
|
|
|
{
|
|
|
|
Error::SetErrno(error, "ioctl(DKIOCCDREADTOC) failed: ", errno);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
const u32 desc_count = CDTOCGetDescriptorCount(toc.get());
|
|
|
|
Log_DevFmt("sessionFirst={}, sessionLast={}, count={}", toc->sessionFirst, toc->sessionLast, desc_count);
|
|
|
|
if (toc->sessionLast < toc->sessionFirst)
|
|
|
|
{
|
|
|
|
Error::SetStringFmt(error, "Last track {} is before first track {}", toc->sessionLast, toc->sessionFirst);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// find track range
|
|
|
|
u32 leadout_index = desc_count;
|
|
|
|
u32 first_track = MAX_TRACK_NUMBER;
|
|
|
|
u32 last_track = 1;
|
|
|
|
for (u32 i = 0; i < desc_count; i++)
|
|
|
|
{
|
|
|
|
const CDTOCDescriptor& desc = toc->descriptors[i];
|
|
|
|
Log_DevFmt(" [{}]: Num={}, Point=0x{:02X} ADR={} MSF={}:{}:{}", i, desc.tno, desc.point, desc.adr, desc.p.minute,
|
|
|
|
desc.p.second, desc.p.frame);
|
|
|
|
|
|
|
|
// Why does MacOS use 0xA2 instead of 0xAA for leadout??
|
|
|
|
if (desc.point == 0xA2)
|
|
|
|
{
|
|
|
|
leadout_index = i;
|
|
|
|
}
|
|
|
|
else if (desc.point >= 1 && desc.point <= MAX_TRACK_NUMBER)
|
|
|
|
{
|
|
|
|
first_track = std::min<u32>(first_track, desc.point);
|
|
|
|
last_track = std::max<u32>(last_track, desc.point);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (leadout_index == desc_count)
|
|
|
|
{
|
|
|
|
Error::SetStringView(error, "Lead-out track not found.");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
LBA disc_lba = 0;
|
|
|
|
LBA last_track_lba = 0;
|
|
|
|
for (u32 track_num = first_track; track_num <= last_track; track_num++)
|
|
|
|
{
|
|
|
|
u32 toc_index;
|
|
|
|
for (toc_index = 0; toc_index < desc_count; toc_index++)
|
|
|
|
{
|
|
|
|
const CDTOCDescriptor& desc = toc->descriptors[toc_index];
|
|
|
|
if (desc.point == track_num)
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
if (toc_index == desc_count)
|
|
|
|
{
|
|
|
|
Error::SetStringFmt(error, "Track {} not found in TOC", track_num);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
const CDTOCDescriptor& desc = toc->descriptors[toc_index];
|
|
|
|
const u32 track_lba = Position{desc.p.minute, desc.p.second, desc.p.frame}.ToLBA();
|
|
|
|
|
|
|
|
// fill in the previous track's length
|
|
|
|
if (!m_tracks.empty())
|
|
|
|
{
|
|
|
|
const LBA previous_track_length = track_lba - last_track_lba;
|
|
|
|
m_tracks.back().length += previous_track_length;
|
|
|
|
m_indices.back().length += previous_track_length;
|
|
|
|
disc_lba += previous_track_length;
|
|
|
|
}
|
|
|
|
|
|
|
|
last_track_lba = track_lba;
|
|
|
|
|
|
|
|
// precompute subchannel q flags for the whole track
|
|
|
|
SubChannelQ::Control control{};
|
|
|
|
control.bits = desc.adr | (desc.control << 4);
|
|
|
|
|
|
|
|
const TrackMode track_mode = control.data ? CDImage::TrackMode::Mode2Raw : CDImage::TrackMode::Audio;
|
|
|
|
|
|
|
|
// TODO: How the hell do we handle pregaps here?
|
|
|
|
const u32 pregap_frames = (track_num == 1) ? 150 : 0;
|
|
|
|
if (pregap_frames > 0)
|
|
|
|
{
|
|
|
|
Index pregap_index = {};
|
|
|
|
pregap_index.start_lba_on_disc = disc_lba;
|
|
|
|
pregap_index.start_lba_in_track = static_cast<LBA>(-static_cast<s32>(pregap_frames));
|
|
|
|
pregap_index.length = pregap_frames;
|
|
|
|
pregap_index.track_number = track_num;
|
|
|
|
pregap_index.index_number = 0;
|
|
|
|
pregap_index.mode = track_mode;
|
|
|
|
pregap_index.submode = CDImage::SubchannelMode::None;
|
|
|
|
pregap_index.control.bits = control.bits;
|
|
|
|
pregap_index.is_pregap = true;
|
|
|
|
m_indices.push_back(pregap_index);
|
|
|
|
disc_lba += pregap_frames;
|
|
|
|
}
|
|
|
|
|
|
|
|
// index 1, will be filled in next iteration
|
|
|
|
if (track_num <= MAX_TRACK_NUMBER)
|
|
|
|
{
|
|
|
|
// add the track itself
|
|
|
|
m_tracks.push_back(
|
|
|
|
Track{track_num, disc_lba, static_cast<u32>(m_indices.size()), 0, track_mode, SubchannelMode::None, control});
|
|
|
|
|
|
|
|
Index index1;
|
|
|
|
index1.start_lba_on_disc = disc_lba;
|
|
|
|
index1.start_lba_in_track = 0;
|
|
|
|
index1.length = 0;
|
|
|
|
index1.track_number = track_num;
|
|
|
|
index1.index_number = 1;
|
|
|
|
index1.file_index = 0;
|
|
|
|
index1.file_sector_size = RAW_SECTOR_SIZE;
|
|
|
|
index1.file_offset = static_cast<u64>(track_lba);
|
|
|
|
index1.mode = track_mode;
|
|
|
|
index1.submode = CDImage::SubchannelMode::None;
|
|
|
|
index1.control.bits = control.bits;
|
|
|
|
index1.is_pregap = false;
|
|
|
|
m_indices.push_back(index1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (m_tracks.empty())
|
|
|
|
{
|
|
|
|
Log_ErrorPrintf("File '%s' contains no tracks", filename);
|
|
|
|
Error::SetString(error, fmt::format("File '{}' contains no tracks", filename));
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Fill last track length from lead-out.
|
|
|
|
const CDTOCDescriptor& leadout_desc = toc->descriptors[leadout_index];
|
|
|
|
const u32 leadout_lba = Position{leadout_desc.p.minute, leadout_desc.p.second, leadout_desc.p.frame}.ToLBA();
|
|
|
|
const LBA previous_track_length = static_cast<LBA>(leadout_lba - last_track_lba);
|
|
|
|
m_tracks.back().length += previous_track_length;
|
|
|
|
m_indices.back().length += previous_track_length;
|
|
|
|
disc_lba += previous_track_length;
|
|
|
|
|
|
|
|
// And add the lead-out itself.
|
|
|
|
AddLeadOutIndex();
|
|
|
|
|
|
|
|
m_lba_count = disc_lba;
|
|
|
|
|
|
|
|
Log_DevPrintf("%u tracks, %u indices, %u lbas", static_cast<u32>(m_tracks.size()), static_cast<u32>(m_indices.size()),
|
|
|
|
static_cast<u32>(m_lba_count));
|
|
|
|
for (u32 i = 0; i < m_tracks.size(); i++)
|
|
|
|
{
|
|
|
|
Log_DevPrintf(" Track %u: Start %u, length %u, mode %u, control 0x%02X", static_cast<u32>(m_tracks[i].track_number),
|
|
|
|
static_cast<u32>(m_tracks[i].start_lba), static_cast<u32>(m_tracks[i].length),
|
|
|
|
static_cast<u32>(m_tracks[i].mode), static_cast<u32>(m_tracks[i].control.bits));
|
|
|
|
}
|
|
|
|
for (u32 i = 0; i < m_indices.size(); i++)
|
|
|
|
{
|
|
|
|
Log_DevPrintf(" Index %u: Track %u, Index %u, Start %u, length %u, file sector size %u, file offset %" PRIu64, i,
|
|
|
|
static_cast<u32>(m_indices[i].track_number), static_cast<u32>(m_indices[i].index_number),
|
|
|
|
static_cast<u32>(m_indices[i].start_lba_on_disc), static_cast<u32>(m_indices[i].length),
|
|
|
|
static_cast<u32>(m_indices[i].file_sector_size), m_indices[i].file_offset);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!DetermineReadMode(error))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
return Seek(1, Position{0, 0, 0});
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CDImageDeviceMacOS::ReadSubChannelQ(SubChannelQ* subq, const Index& index, LBA lba_in_index)
|
|
|
|
{
|
|
|
|
if (index.file_sector_size == 0 || m_read_mode < SCSIReadMode::Full)
|
|
|
|
return CDImage::ReadSubChannelQ(subq, index, lba_in_index);
|
|
|
|
|
|
|
|
const LBA disc_lba = static_cast<LBA>(index.file_offset) + lba_in_index;
|
|
|
|
if (m_current_lba != disc_lba && !ReadSectorToBuffer(disc_lba))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
if (m_read_mode == SCSIReadMode::SubQOnly)
|
|
|
|
{
|
|
|
|
// copy out subq
|
|
|
|
std::memcpy(subq->data.data(), m_buffer.data() + RAW_SECTOR_SIZE, SUBCHANNEL_BYTES_PER_FRAME);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
else // if (m_scsi_read_mode == SCSIReadMode::Full)
|
|
|
|
{
|
|
|
|
// need to deinterleave the subcode
|
|
|
|
u8 deinterleaved_subcode[ALL_SUBCODE_SIZE];
|
|
|
|
DeinterleaveSubcode(m_buffer.data() + RAW_SECTOR_SIZE, deinterleaved_subcode);
|
|
|
|
|
|
|
|
// P, Q, ...
|
|
|
|
std::memcpy(subq->data.data(), deinterleaved_subcode + SUBCHANNEL_BYTES_PER_FRAME, SUBCHANNEL_BYTES_PER_FRAME);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CDImageDeviceMacOS::HasNonStandardSubchannel() const
|
|
|
|
{
|
|
|
|
// Can only read subchannel through SPTD.
|
|
|
|
return m_read_mode >= SCSIReadMode::Full;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CDImageDeviceMacOS::ReadSectorFromIndex(void* buffer, const Index& index, LBA lba_in_index)
|
|
|
|
{
|
|
|
|
if (index.file_sector_size == 0)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
const LBA disc_lba = static_cast<LBA>(index.file_offset) + lba_in_index;
|
|
|
|
if (m_current_lba != disc_lba && !ReadSectorToBuffer(disc_lba))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
std::memcpy(buffer, m_buffer.data(), RAW_SECTOR_SIZE);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CDImageDeviceMacOS::DoSetSpeed(u32 speed_multiplier)
|
|
|
|
{
|
|
|
|
const u16 speed = static_cast<u16>((FRAMES_PER_SECOND * RAW_SECTOR_SIZE * speed_multiplier) / 1024);
|
|
|
|
if (ioctl(m_fd, DKIOCCDSETSPEED, &speed) != 0)
|
|
|
|
{
|
|
|
|
Log_ErrorFmt("DKIOCCDSETSPEED for speed {} failed: {}", speed, errno);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CDImageDeviceMacOS::ReadSectorToBuffer(LBA lba)
|
|
|
|
{
|
|
|
|
if (lba < RAW_READ_OFFSET)
|
|
|
|
{
|
|
|
|
Log_ErrorFmt("Out of bounds LBA {}", lba);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
const u32 sector_size =
|
|
|
|
RAW_SECTOR_SIZE + ((m_read_mode == SCSIReadMode::Full) ?
|
|
|
|
ALL_SUBCODE_SIZE :
|
|
|
|
((m_read_mode == SCSIReadMode::SubQOnly) ? SUBCHANNEL_BYTES_PER_FRAME : 0));
|
|
|
|
dk_cd_read_t desc = {};
|
|
|
|
desc.sectorArea =
|
|
|
|
kCDSectorAreaSync | kCDSectorAreaHeader | kCDSectorAreaSubHeader | kCDSectorAreaUser | kCDSectorAreaAuxiliary |
|
|
|
|
((m_read_mode == SCSIReadMode::Full) ? kCDSectorAreaSubChannel :
|
|
|
|
((m_read_mode == SCSIReadMode::SubQOnly) ? kCDSectorAreaSubChannelQ : 0));
|
|
|
|
desc.sectorType = kCDSectorTypeUnknown;
|
|
|
|
desc.offset = static_cast<u64>(lba - RAW_READ_OFFSET) * sector_size;
|
|
|
|
desc.buffer = m_buffer.data();
|
|
|
|
desc.bufferLength = sector_size;
|
|
|
|
if (ioctl(m_fd, DKIOCCDREAD, &desc) != 0)
|
|
|
|
{
|
|
|
|
const Position msf = Position::FromLBA(lba);
|
|
|
|
Log_ErrorFmt("DKIOCCDREAD for LBA {} (MSF {}:{}:{}) failed: {}", lba, msf.minute, msf.second, msf.frame, errno);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
m_current_lba = lba;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CDImageDeviceMacOS::DetermineReadMode(Error* error)
|
|
|
|
{
|
|
|
|
const LBA track_1_lba = static_cast<LBA>(m_indices[m_tracks[0].first_index].file_offset);
|
|
|
|
const bool check_subcode = ShouldTryReadingSubcode();
|
|
|
|
|
|
|
|
Log_DevPrint("Trying read with full subcode...");
|
|
|
|
m_read_mode = SCSIReadMode::Full;
|
|
|
|
m_current_lba = m_lba_count;
|
|
|
|
if (check_subcode && ReadSectorToBuffer(track_1_lba))
|
|
|
|
{
|
|
|
|
if (VerifySCSIReadData(std::span<u8>(m_buffer.data(), RAW_SECTOR_SIZE + ALL_SUBCODE_SIZE), SCSIReadMode::Full,
|
|
|
|
track_1_lba))
|
|
|
|
{
|
|
|
|
Log_VerbosePrint("Using reads with subcode");
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#if 0
|
|
|
|
// This seems to lock up on my drive... :/
|
|
|
|
Log_WarningPrint("Full subcode failed, trying SCSI read with only subq...");
|
|
|
|
m_read_mode = SCSIReadMode::SubQOnly;
|
|
|
|
m_current_lba = m_lba_count;
|
|
|
|
if (check_subcode && ReadSectorToBuffer(track_1_lba))
|
|
|
|
{
|
|
|
|
if (VerifySCSIReadData(std::span<u8>(m_buffer.data(), RAW_SECTOR_SIZE + SUBCHANNEL_BYTES_PER_FRAME),
|
|
|
|
SCSIReadMode::SubQOnly, track_1_lba))
|
|
|
|
{
|
|
|
|
Log_VerbosePrint("Using reads with subq only");
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
Log_WarningPrint("SCSI reads failed, trying without subcode...");
|
|
|
|
m_read_mode = SCSIReadMode::Raw;
|
|
|
|
m_current_lba = m_lba_count;
|
|
|
|
if (ReadSectorToBuffer(track_1_lba))
|
|
|
|
{
|
|
|
|
Log_WarningPrint("Using non-subcode reads, libcrypt games will not run correctly");
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
Log_ErrorPrint("No read modes were successful, cannot use device.");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
std::unique_ptr<CDImage> CDImage::OpenDeviceImage(const char* filename, Error* error)
|
|
|
|
{
|
|
|
|
std::unique_ptr<CDImageDeviceMacOS> image = std::make_unique<CDImageDeviceMacOS>();
|
|
|
|
if (!image->Open(filename, error))
|
|
|
|
return {};
|
|
|
|
|
|
|
|
return image;
|
|
|
|
}
|
|
|
|
|
|
|
|
std::vector<std::pair<std::string, std::string>> CDImage::GetDeviceList()
|
|
|
|
{
|
|
|
|
std::vector<std::pair<std::string, std::string>> ret;
|
|
|
|
|
|
|
|
// borrowed from PCSX2
|
|
|
|
auto append_list = [&ret](const char* classes_name) {
|
|
|
|
CFMutableDictionaryRef classes = IOServiceMatching(kIOCDMediaClass);
|
|
|
|
if (!classes)
|
|
|
|
return;
|
|
|
|
|
|
|
|
CFDictionarySetValue(classes, CFSTR(kIOMediaEjectableKey), kCFBooleanTrue);
|
|
|
|
|
|
|
|
io_iterator_t iter;
|
|
|
|
kern_return_t result = IOServiceGetMatchingServices(0, classes, &iter);
|
|
|
|
if (result != KERN_SUCCESS)
|
|
|
|
{
|
|
|
|
CFRelease(classes);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
while (io_object_t media = IOIteratorNext(iter))
|
|
|
|
{
|
|
|
|
CFTypeRef path = IORegistryEntryCreateCFProperty(media, CFSTR(kIOBSDNameKey), kCFAllocatorDefault, 0);
|
|
|
|
if (path)
|
|
|
|
{
|
|
|
|
char buf[PATH_MAX];
|
|
|
|
if (CFStringGetCString((CFStringRef)path, buf, sizeof(buf), kCFStringEncodingUTF8))
|
|
|
|
{
|
|
|
|
if (std::none_of(ret.begin(), ret.end(), [&buf](const auto& it) { return it.second == buf; }))
|
|
|
|
ret.emplace_back(fmt::format("/dev/r{}", buf), buf);
|
|
|
|
}
|
|
|
|
CFRelease(path);
|
|
|
|
IOObjectRelease(media);
|
|
|
|
}
|
|
|
|
IOObjectRelease(media);
|
|
|
|
}
|
|
|
|
|
|
|
|
IOObjectRelease(iter);
|
|
|
|
};
|
|
|
|
|
|
|
|
append_list(kIOCDMediaClass);
|
|
|
|
append_list(kIODVDMediaClass);
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CDImage::IsDeviceName(const char* filename)
|
|
|
|
{
|
|
|
|
if (!std::string_view(filename).starts_with("/dev"))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
io_service_t service = GetDeviceMediaService(filename);
|
|
|
|
const bool valid = (service != 0);
|
|
|
|
if (valid)
|
|
|
|
IOObjectRelease(service);
|
|
|
|
|
|
|
|
return valid;
|
|
|
|
}
|
|
|
|
|
2021-07-24 08:38:54 +00:00
|
|
|
#else
|
|
|
|
|
2023-08-19 13:40:36 +00:00
|
|
|
std::unique_ptr<CDImage> CDImage::OpenDeviceImage(const char* filename, Error* error)
|
2021-07-24 08:38:54 +00:00
|
|
|
{
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
|
|
|
|
std::vector<std::pair<std::string, std::string>> CDImage::GetDeviceList()
|
|
|
|
{
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CDImage::IsDeviceName(const char* filename)
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2023-09-30 05:44:14 +00:00
|
|
|
#endif
|