tagparser/wav/waveaudiostream.cpp

148 lines
4.6 KiB
C++
Raw Normal View History

2015-09-06 19:57:33 +02:00
#include "./waveaudiostream.h"
2015-04-22 19:22:01 +02:00
2015-09-22 01:37:23 +02:00
#include "../mpegaudio/mpegaudioframestream.h"
2015-09-06 19:57:33 +02:00
#include "../exceptions.h"
#include "../mediaformat.h"
2015-04-22 19:22:01 +02:00
2015-09-23 00:03:05 +02:00
#include <c++utilities/conversion/stringconversion.h>
2018-03-07 01:17:50 +01:00
#include <c++utilities/io/binaryreader.h>
2015-06-12 02:35:50 +02:00
2015-04-22 19:22:01 +02:00
using namespace std;
2015-09-22 01:37:23 +02:00
using namespace ChronoUtilities;
2015-04-22 19:22:01 +02:00
namespace TagParser {
2015-04-22 19:22:01 +02:00
2015-06-12 02:35:50 +02:00
/*!
* \class TagParser::WaveFormatHeader
2015-06-12 02:35:50 +02:00
* \brief The WaveFormatHeader class parses the WAVEFORMATEX structure defined by MS.
*/
/*!
2015-09-22 01:37:23 +02:00
* \brief Parses the WAVE "fmt " header segment using the specified \a reader.
2015-06-12 02:35:50 +02:00
* \remarks Reads 16 bytes from the associated stream.
*/
void WaveFormatHeader::parse(IoUtilities::BinaryReader &reader)
{
formatTag = reader.readUInt16LE();
channelCount = reader.readUInt16LE();
sampleRate = reader.readUInt32LE();
bytesPerSecond = reader.readUInt32LE();
chunkSize = reader.readUInt16LE();
bitsPerSample = reader.readUInt16LE();
}
/*!
* \brief Returns the media format denoted by the format tag.
*/
MediaFormat WaveFormatHeader::format() const
{
2018-03-07 01:17:50 +01:00
switch (formatTag) {
case 0x0001u:
return GeneralMediaFormat::Pcm;
case 0x0050u:
return MediaFormat(GeneralMediaFormat::Mpeg1Audio, SubFormats::Mpeg1Layer2);
case 0x0055u:
return MediaFormat(GeneralMediaFormat::Mpeg1Audio, SubFormats::Mpeg1Layer3);
default:
return GeneralMediaFormat::Unknown;
2015-06-12 02:35:50 +02:00
}
}
2015-04-22 19:22:01 +02:00
/*!
* \class TagParser::WaveAudioStream
* \brief Implementation of TagParser::AbstractTrack for the
2015-04-22 19:22:01 +02:00
* RIFF WAVE container format.
*/
/*!
* \brief Constructs a new track for the \a stream at the specified \a startOffset.
*/
2018-03-07 01:17:50 +01:00
WaveAudioStream::WaveAudioStream(iostream &stream, uint64 startOffset)
: AbstractTrack(stream, startOffset)
, m_dataOffset(0)
2015-04-22 19:22:01 +02:00
{
m_mediaType = MediaType::Audio;
2015-04-22 19:22:01 +02:00
}
/*!
* \brief Destroys the track.
*/
WaveAudioStream::~WaveAudioStream()
2018-03-07 01:17:50 +01:00
{
}
2015-04-22 19:22:01 +02:00
TrackType WaveAudioStream::type() const
{
return TrackType::WaveAudioStream;
}
2015-06-12 02:35:50 +02:00
/*!
* \brief Adds the information from the specified \a waveHeader to the specified \a track.
*/
void WaveAudioStream::addInfo(const WaveFormatHeader &waveHeader, AbstractTrack &track)
{
track.m_format = waveHeader.format();
2015-09-23 00:03:05 +02:00
track.m_formatId = ConversionUtilities::numberToString(waveHeader.formatTag);
2015-06-12 02:35:50 +02:00
track.m_channelCount = waveHeader.channelCount;
track.m_samplingFrequency = waveHeader.sampleRate;
2015-06-12 02:35:50 +02:00
track.m_bytesPerSecond = waveHeader.bytesPerSecond;
track.m_chunkSize = waveHeader.chunkSize;
track.m_bitsPerSample = waveHeader.bitsPerSample;
track.m_bitrate = waveHeader.bitrate();
}
void WaveAudioStream::internalParseHeader(Diagnostics &diag)
2015-04-22 19:22:01 +02:00
{
2015-09-22 01:37:23 +02:00
const string context("parsing RIFF/WAVE header");
2018-03-07 01:17:50 +01:00
if (!m_istream) {
2015-04-22 19:22:01 +02:00
throw NoDataFoundException();
}
2018-03-07 01:17:50 +01:00
if (m_reader.readUInt32BE() != 0x52494646u) {
2018-03-07 01:11:42 +01:00
throw NoDataFoundException();
}
m_istream->seekg(static_cast<streamoff>(m_startOffset + 8));
2018-03-07 01:17:50 +01:00
if (m_reader.readUInt32BE() != 0x57415645u) {
2018-03-07 01:11:42 +01:00
throw NoDataFoundException();
}
2018-03-07 01:17:50 +01:00
while (!m_dataOffset) {
2018-03-07 01:11:42 +01:00
uint32 segmentId = m_reader.readUInt32BE();
uint32 restHeaderLen = m_reader.readUInt32LE();
2018-03-07 01:17:50 +01:00
switch (segmentId) {
2018-03-07 01:11:42 +01:00
case 0x666D7420u:
2018-03-07 01:17:50 +01:00
if (restHeaderLen >= 16u) {
2018-03-07 01:11:42 +01:00
WaveFormatHeader waveHeader;
waveHeader.parse(m_reader);
addInfo(waveHeader, *this);
restHeaderLen -= 16u;
} else {
diag.emplace_back(DiagLevel::Warning, "\"fmt \" segment is truncated.", context);
2015-04-22 19:22:01 +02:00
}
2018-03-07 01:11:42 +01:00
break;
case 0x64617461u:
m_dataOffset = static_cast<uint64>(m_istream->tellg());
2018-03-07 01:11:42 +01:00
m_size = restHeaderLen;
m_sampleCount = m_size / m_chunkSize;
m_duration = TimeSpan::fromSeconds(static_cast<double>(m_sampleCount) / static_cast<double>(m_samplingFrequency));
break;
2018-03-07 01:17:50 +01:00
default:;
2015-04-22 19:22:01 +02:00
}
2018-03-07 01:11:42 +01:00
m_istream->seekg(restHeaderLen, ios_base::cur);
2015-04-22 19:22:01 +02:00
}
2018-03-07 01:17:50 +01:00
if (m_format.general != GeneralMediaFormat::Mpeg1Audio || !m_dataOffset) {
2018-03-07 01:11:42 +01:00
return;
2015-09-22 01:37:23 +02:00
}
m_istream->seekg(static_cast<streamoff>(m_dataOffset));
2018-03-07 01:11:42 +01:00
MpegAudioFrame frame;
frame.parseHeader(m_reader);
MpegAudioFrameStream::addInfo(frame, *this);
m_bitrate = frame.isXingFramefieldPresent()
2018-03-07 01:17:50 +01:00
? ((static_cast<double>(m_size) * 8.0)
/ (static_cast<double>(frame.xingFrameCount() * frame.sampleCount()) / static_cast<double>(frame.samplingFrequency())) / 1024.0)
: frame.bitrate();
m_bytesPerSecond = static_cast<uint32>(m_bitrate * 125);
2018-03-07 01:11:42 +01:00
m_duration = TimeSpan::fromSeconds(static_cast<double>(m_size) / (m_bitrate * 128.0));
2015-04-22 19:22:01 +02:00
}
2018-03-07 01:17:50 +01:00
} // namespace TagParser