// SPDX-FileCopyrightText: 2019-2023 Connor McLaughlin // SPDX-License-Identifier: (GPL-3.0 OR CC-BY-NC-ND-4.0) #include "cue_parser.h" #include "common/error.h" #include "common/log.h" #include "common/string_util.h" #include Log_SetChannel(CueParser); namespace CueParser { static bool TokenMatch(std::string_view s1, const char* token); } bool CueParser::TokenMatch(std::string_view s1, const char* token) { const size_t token_len = std::strlen(token); if (s1.length() != token_len) return false; return (StringUtil::Strncasecmp(s1.data(), token, token_len) == 0); } CueParser::File::File() = default; CueParser::File::~File() = default; const CueParser::Track* CueParser::File::GetTrack(u32 n) const { for (const auto& it : m_tracks) { if (it.number == n) return ⁢ } return nullptr; } CueParser::Track* CueParser::File::GetMutableTrack(u32 n) { for (auto& it : m_tracks) { if (it.number == n) return ⁢ } return nullptr; } bool CueParser::File::Parse(std::FILE* fp, Error* error) { char line[1024]; u32 line_number = 1; while (std::fgets(line, sizeof(line), fp)) { if (!ParseLine(line, line_number, error)) return false; line_number++; } if (!CompleteLastTrack(line_number, error)) return false; if (!SetTrackLengths(line_number, error)) return false; return true; } void CueParser::File::SetError(u32 line_number, Error* error, const char* format, ...) { std::va_list ap; SmallString str; va_start(ap, format); str.vsprintf(format, ap); va_end(ap); ERROR_LOG("Cue parse error at line {}: {}", line_number, str.c_str()); Error::SetString(error, fmt::format("Cue parse error at line {}: {}", line_number, str)); } std::string_view CueParser::File::GetToken(const char*& line) { std::string_view ret; const char* start = line; while (std::isspace(*start) && *start != '\0') start++; if (*start == '\0') return ret; const char* end; const bool quoted = *start == '\"'; if (quoted) { start++; end = start; while (*end != '\"' && *end != '\0') end++; if (*end != '\"') return ret; ret = std::string_view(start, static_cast(end - start)); // eat closing " end++; } else { end = start; while (!std::isspace(*end) && *end != '\0') end++; ret = std::string_view(start, static_cast(end - start)); } line = end; return ret; } std::optional CueParser::File::GetMSF(std::string_view token) { static const s32 max_values[] = {std::numeric_limits::max(), 60, 75}; u32 parts[3] = {}; u32 part = 0; u32 start = 0; for (;;) { while (start < token.length() && token[start] < '0' && token[start] <= '9') start++; if (start == token.length()) return std::nullopt; u32 end = start; while (end < token.length() && token[end] >= '0' && token[end] <= '9') end++; const std::optional value = StringUtil::FromChars(token.substr(start, end - start)); if (!value.has_value() || value.value() < 0 || value.value() > max_values[part]) return std::nullopt; parts[part] = static_cast(value.value()); part++; if (part == 3) break; while (end < token.length() && std::isspace(token[end])) end++; if (end == token.length() || token[end] != ':') return std::nullopt; start = end + 1; } MSF ret; ret.minute = static_cast(parts[0]); ret.second = static_cast(parts[1]); ret.frame = static_cast(parts[2]); return ret; } bool CueParser::File::ParseLine(const char* line, u32 line_number, Error* error) { const std::string_view command(GetToken(line)); if (command.empty()) return true; if (TokenMatch(command, "REM")) { // comment, eat it return true; } if (TokenMatch(command, "FILE")) return HandleFileCommand(line, line_number, error); else if (TokenMatch(command, "TRACK")) return HandleTrackCommand(line, line_number, error); else if (TokenMatch(command, "INDEX")) return HandleIndexCommand(line, line_number, error); else if (TokenMatch(command, "PREGAP")) return HandlePregapCommand(line, line_number, error); else if (TokenMatch(command, "FLAGS")) return HandleFlagCommand(line, line_number, error); if (TokenMatch(command, "POSTGAP")) { WARNING_LOG("Ignoring '{}' command", command); return true; } // stuff we definitely ignore if (TokenMatch(command, "CATALOG") || TokenMatch(command, "CDTEXTFILE") || TokenMatch(command, "ISRC") || TokenMatch(command, "TRACK_ISRC") || TokenMatch(command, "TITLE") || TokenMatch(command, "PERFORMER") || TokenMatch(command, "SONGWRITER") || TokenMatch(command, "COMPOSER") || TokenMatch(command, "ARRANGER") || TokenMatch(command, "MESSAGE") || TokenMatch(command, "DISC_ID") || TokenMatch(command, "GENRE") || TokenMatch(command, "TOC_INFO1") || TokenMatch(command, "TOC_INFO2") || TokenMatch(command, "UPC_EAN") || TokenMatch(command, "SIZE_INFO")) { return true; } SetError(line_number, error, "Invalid command '%*s'", static_cast(command.size()), command.data()); return false; } bool CueParser::File::HandleFileCommand(const char* line, u32 line_number, Error* error) { const std::string_view filename(GetToken(line)); const std::string_view mode(GetToken(line)); if (filename.empty()) { SetError(line_number, error, "Missing filename"); return false; } if (!TokenMatch(mode, "BINARY")) { SetError(line_number, error, "Only BINARY modes are supported"); return false; } m_current_file = filename; DEBUG_LOG("File '{}'", filename); return true; } bool CueParser::File::HandleTrackCommand(const char* line, u32 line_number, Error* error) { if (!CompleteLastTrack(line_number, error)) return false; if (!m_current_file.has_value()) { SetError(line_number, error, "Starting a track declaration without a file set"); return false; } const std::string_view track_number_str(GetToken(line)); if (track_number_str.empty()) { SetError(line_number, error, "Missing track number"); return false; } const std::optional track_number = StringUtil::FromChars(track_number_str); if (track_number.value_or(0) < MIN_TRACK_NUMBER || track_number.value_or(0) > MAX_TRACK_NUMBER) { SetError(line_number, error, "Invalid track number %d", track_number.value_or(0)); return false; } const std::string_view mode_str = GetToken(line); TrackMode mode; if (TokenMatch(mode_str, "AUDIO")) mode = TrackMode::Audio; else if (TokenMatch(mode_str, "MODE1/2048")) mode = TrackMode::Mode1; else if (TokenMatch(mode_str, "MODE1/2352")) mode = TrackMode::Mode1Raw; else if (TokenMatch(mode_str, "MODE2/2336")) mode = TrackMode::Mode2; else if (TokenMatch(mode_str, "MODE2/2048")) mode = TrackMode::Mode2Form1; else if (TokenMatch(mode_str, "MODE2/2342")) mode = TrackMode::Mode2Form2; else if (TokenMatch(mode_str, "MODE2/2332")) mode = TrackMode::Mode2FormMix; else if (TokenMatch(mode_str, "MODE2/2352")) mode = TrackMode::Mode2Raw; else { SetError(line_number, error, "Invalid mode: '%*s'", static_cast(mode_str.length()), mode_str.data()); return false; } m_current_track = Track(); m_current_track->number = static_cast(track_number.value()); m_current_track->file = m_current_file.value(); m_current_track->mode = mode; return true; } bool CueParser::File::HandleIndexCommand(const char* line, u32 line_number, Error* error) { if (!m_current_track.has_value()) { SetError(line_number, error, "Setting index without track"); return false; } const std::string_view index_number_str(GetToken(line)); if (index_number_str.empty()) { SetError(line_number, error, "Missing index number"); return false; } const std::optional index_number = StringUtil::FromChars(index_number_str); if (index_number.value_or(-1) < MIN_INDEX_NUMBER || index_number.value_or(-1) > MAX_INDEX_NUMBER) { SetError(line_number, error, "Invalid index number %d", index_number.value_or(-1)); return false; } if (m_current_track->GetIndex(static_cast(index_number.value())) != nullptr) { SetError(line_number, error, "Duplicate index %d", index_number.value()); return false; } const std::string_view msf_str(GetToken(line)); if (msf_str.empty()) { SetError(line_number, error, "Missing index location"); return false; } const std::optional msf(GetMSF(msf_str)); if (!msf.has_value()) { SetError(line_number, error, "Invalid index location '%*s'", static_cast(msf_str.size()), msf_str.data()); return false; } m_current_track->indices.emplace_back(static_cast(index_number.value()), msf.value()); return true; } bool CueParser::File::HandlePregapCommand(const char* line, u32 line_number, Error* error) { if (!m_current_track.has_value()) { SetError(line_number, error, "Setting pregap without track"); return false; } if (m_current_track->zero_pregap.has_value()) { SetError(line_number, error, "Pregap already specified for track %u", m_current_track->number); return false; } const std::string_view msf_str(GetToken(line)); if (msf_str.empty()) { SetError(line_number, error, "Missing pregap location"); return false; } const std::optional msf(GetMSF(msf_str)); if (!msf.has_value()) { SetError(line_number, error, "Invalid pregap location '%*s'", static_cast(msf_str.size()), msf_str.data()); return false; } m_current_track->zero_pregap = std::move(msf); return true; } bool CueParser::File::HandleFlagCommand(const char* line, u32 line_number, Error* error) { if (!m_current_track.has_value()) { SetError(line_number, error, "Flags command outside of track"); return false; } for (;;) { const std::string_view token(GetToken(line)); if (token.empty()) break; if (TokenMatch(token, "PRE")) m_current_track->SetFlag(TrackFlag::PreEmphasis); else if (TokenMatch(token, "DCP")) m_current_track->SetFlag(TrackFlag::CopyPermitted); else if (TokenMatch(token, "4CH")) m_current_track->SetFlag(TrackFlag::FourChannelAudio); else if (TokenMatch(token, "SCMS")) m_current_track->SetFlag(TrackFlag::SerialCopyManagement); else WARNING_LOG("Unknown track flag '{}'", token); } return true; } bool CueParser::File::CompleteLastTrack(u32 line_number, Error* error) { if (!m_current_track.has_value()) return true; const MSF* index1 = m_current_track->GetIndex(1); if (!index1) { SetError(line_number, error, "Track %u is missing index 1", m_current_track->number); return false; } // check indices for (const auto& [index_number, index_msf] : m_current_track->indices) { if (index_number == 0) continue; const MSF* prev_index = m_current_track->GetIndex(index_number - 1); if (prev_index && *prev_index > index_msf) { SetError(line_number, error, "Index %u is after index %u in track %u", index_number - 1, index_number, m_current_track->number); return false; } } const MSF* index0 = m_current_track->GetIndex(0); if (index0 && m_current_track->zero_pregap.has_value()) { WARNING_LOG("Zero pregap and index 0 specified in track {}, ignoring zero pregap", m_current_track->number); m_current_track->zero_pregap.reset(); } m_current_track->start = *index1; m_tracks.push_back(std::move(m_current_track.value())); m_current_track.reset(); return true; } bool CueParser::File::SetTrackLengths(u32 line_number, Error* error) { for (const Track& track : m_tracks) { if (track.number > 1) { // set the length of the previous track based on this track's start, if they're the same file Track* previous_track = GetMutableTrack(track.number - 1); if (previous_track && previous_track->file == track.file) { if (previous_track->start > track.start) { SetError(line_number, error, "Track %u start greater than track %u start", previous_track->number, track.number); return false; } // Use index 0, otherwise index 1. const MSF* start_index = track.GetIndex(0); if (!start_index) start_index = track.GetIndex(1); previous_track->length = MSF::FromLBA(start_index->ToLBA() - previous_track->start.ToLBA()); } } } return true; } const CueParser::MSF* CueParser::Track::GetIndex(u32 n) const { for (const auto& it : indices) { if (it.first == n) return &it.second; } return nullptr; }