input/CdioParanoia: C++ wrappers for libcdio types

This commit is contained in:
Max Kellermann 2019-01-20 21:45:50 +01:00
parent 48ec09ab1e
commit 4ba9357a9c
2 changed files with 175 additions and 24 deletions

View File

@ -23,6 +23,7 @@
#include "config.h"
#include "CdioParanoiaInputPlugin.hxx"
#include "lib/cdio/Paranoia.hxx"
#include "../InputStream.hxx"
#include "../InputPlugin.hxx"
#include "util/TruncateString.hxx"
@ -41,19 +42,12 @@
#include <stdlib.h>
#include <assert.h>
#include <cdio/version.h>
#if LIBCDIO_VERSION_NUM >= 90
#include <cdio/paranoia/paranoia.h>
#else
#include <cdio/paranoia.h>
#endif
#include <cdio/cd_types.h>
class CdioParanoiaInputStream final : public InputStream {
cdrom_drive_t *const drv;
CdIo_t *const cdio;
cdrom_paranoia_t *const para;
CdromParanoia para;
const lsn_t lsn_from, lsn_to;
int lsn_relofs;
@ -67,18 +61,17 @@ class CdioParanoiaInputStream final : public InputStream {
bool reverse_endian,
lsn_t _lsn_from, lsn_t _lsn_to)
:InputStream(_uri, _mutex),
drv(_drv), cdio(_cdio), para(cdio_paranoia_init(drv)),
drv(_drv), cdio(_cdio), para(drv),
lsn_from(_lsn_from), lsn_to(_lsn_to),
lsn_relofs(0),
buffer_lsn(-1)
{
/* Set reading mode for full paranoia, but allow
skipping sectors. */
paranoia_modeset(para,
PARANOIA_MODE_FULL^PARANOIA_MODE_NEVERSKIP);
para.SetMode(PARANOIA_MODE_FULL^PARANOIA_MODE_NEVERSKIP);
/* seek to beginning of the track */
cdio_paranoia_seek(para, lsn_from, SEEK_SET);
para.Seek(lsn_from);
seekable = true;
size = (lsn_to - lsn_from + 1) * CDIO_CD_FRAMESIZE_RAW;
@ -91,7 +84,7 @@ class CdioParanoiaInputStream final : public InputStream {
}
~CdioParanoiaInputStream() {
cdio_paranoia_free(para);
para = {};
cdio_cddap_close_no_free_cdio(drv);
cdio_destroy(cdio);
}
@ -278,7 +271,7 @@ CdioParanoiaInputStream::Seek(offset_type new_offset)
{
const ScopeUnlock unlock(mutex);
cdio_paranoia_seek(para, lsn_from + lsn_relofs, SEEK_SET);
para.Seek(lsn_from + lsn_relofs);
}
}
@ -298,8 +291,9 @@ CdioParanoiaInputStream::Read(void *ptr, size_t length)
if (lsn_relofs != buffer_lsn) {
const ScopeUnlock unlock(mutex);
rbuf = cdio_paranoia_read(para, nullptr);
try {
rbuf = para.Read().data;
} catch (...) {
char *s_err = cdio_cddap_errors(drv);
if (s_err) {
FormatError(cdio_domain,
@ -310,8 +304,9 @@ CdioParanoiaInputStream::Read(void *ptr, size_t length)
free(s_err);
#endif
}
if (!rbuf)
throw std::runtime_error("paranoia read error");
throw;
}
//store current buffer
memcpy(buffer, rbuf, CDIO_CD_FRAMESIZE_RAW);

156
src/lib/cdio/Paranoia.hxx Normal file
View File

@ -0,0 +1,156 @@
/*
* Copyright 2019 Max Kellermann <max.kellermann@gmail.com>
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
*
* - Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
*
* - Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the
* distribution.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
* ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
* LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
* FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE
* FOUNDATION OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
* INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
* (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
* SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
* HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
* STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
* ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED
* OF THE POSSIBILITY OF SUCH DAMAGE.
*/
#ifndef CDIO_PARANOIA_HXX
#define CDIO_PARANOIA_HXX
#include "util/ConstBuffer.hxx"
#include "util/Compiler.h"
#include <cdio/version.h>
#if LIBCDIO_VERSION_NUM >= 90
#include <cdio/paranoia/paranoia.h>
#else
#include <cdio/paranoia.h>
#endif
#include <stdexcept>
#include <utility>
class CdromDrive {
cdrom_drive_t *drv = nullptr;
public:
CdromDrive() = default;
explicit CdromDrive(CdIo_t *cdio)
:drv(cdio_cddap_identify_cdio(cdio, 1, nullptr))
{
if (drv == nullptr)
throw std::runtime_error("Failed to identify audio CD");
cdda_verbose_set(drv, CDDA_MESSAGE_FORGETIT,
CDDA_MESSAGE_FORGETIT);
}
~CdromDrive() noexcept {
if (drv != nullptr)
cdio_cddap_close_no_free_cdio(drv);
}
CdromDrive(CdromDrive &&src) noexcept
:drv(std::exchange(src.drv, nullptr)) {}
CdromDrive &operator=(CdromDrive &&src) noexcept {
using std::swap;
swap(drv, src.drv);
return *this;
}
auto get() const noexcept {
return drv;
}
void Open() {
if (cdio_cddap_open(drv) != 0)
throw std::runtime_error("Failed to open disc");
}
gcc_pure
bool IsAudioTrack(track_t i) const noexcept {
return cdio_cddap_track_audiop(drv, i);
}
auto GetTrackSectorRange(track_t i) const {
auto first = cdio_cddap_track_firstsector(drv, i);
auto last = cdio_cddap_track_lastsector(drv, i);
if (first < 0 || last < 0)
throw std::runtime_error("Invalid track number");
return std::make_pair(first, last);
}
gcc_pure
unsigned GetTrackCount() const noexcept {
return cdio_cddap_tracks(drv);
}
unsigned GetTrackChannels(track_t i) const {
auto value = cdio_cddap_track_channels(drv, i);
if (value < 0)
throw std::runtime_error("cdio_cddap_track_channels() failed");
return unsigned(value);
}
};
class CdromParanoia {
cdrom_paranoia_t *paranoia = nullptr;
public:
CdromParanoia() = default;
explicit CdromParanoia(cdrom_drive_t *drv) noexcept
:paranoia(cdio_paranoia_init(drv)) {}
~CdromParanoia() noexcept {
if (paranoia != nullptr)
cdio_paranoia_free(paranoia);
}
CdromParanoia(CdromParanoia &&src) noexcept
:paranoia(std::exchange(src.paranoia, nullptr)) {}
CdromParanoia &operator=(CdromParanoia &&src) noexcept {
using std::swap;
swap(paranoia, src.paranoia);
return *this;
}
auto get() const noexcept {
return paranoia;
}
void SetMode(int mode_flags) noexcept {
paranoia_modeset(paranoia, mode_flags);
}
void Seek(int32_t seek, int whence=SEEK_SET) {
if (cdio_paranoia_seek(paranoia, seek, whence) < 0)
throw std::runtime_error("Failed to seek disc");
}
ConstBuffer<int16_t> Read() {
const int16_t *data = cdio_paranoia_read(paranoia, nullptr);
if (data == nullptr)
throw std::runtime_error("Read from audio CD failed");
return {data, CDIO_CD_FRAMESIZE_RAW / sizeof(int16_t)};
}
};
#endif