2015-12-16 20:30:58 +00:00
|
|
|
/*
|
|
|
|
fm_transmitter - use Raspberry Pi as FM transmitter
|
|
|
|
|
|
|
|
Copyright (c) 2015, Marcin Kondej
|
|
|
|
All rights reserved.
|
|
|
|
|
|
|
|
See https://github.com/markondej/fm_transmitter
|
|
|
|
|
|
|
|
Redistribution and use in source and binary forms, with or without modification, are
|
|
|
|
permitted provided that the following conditions are met:
|
|
|
|
|
|
|
|
1. Redistributions of source code must retain the above copyright notice, this list
|
|
|
|
of conditions and the following disclaimer.
|
|
|
|
|
|
|
|
2. 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.
|
|
|
|
|
|
|
|
3. Neither the name of the copyright holder nor the names of its contributors may be
|
|
|
|
used to endorse or promote products derived from this software without specific
|
|
|
|
prior written permission.
|
|
|
|
|
|
|
|
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 COPYRIGHT HOLDER 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.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "wave_reader.h"
|
|
|
|
#include "error_reporter.h"
|
2016-12-07 00:21:44 +00:00
|
|
|
#include <iostream>
|
2015-12-16 20:30:58 +00:00
|
|
|
#include <sstream>
|
|
|
|
#include <string.h>
|
|
|
|
|
|
|
|
using std::ostringstream;
|
|
|
|
using std::exception;
|
2016-12-07 00:21:44 +00:00
|
|
|
using std::cin;
|
2015-12-16 20:30:58 +00:00
|
|
|
|
2016-12-06 16:07:30 +00:00
|
|
|
WaveReader::WaveReader(string filename, bool &forceStop) :
|
2016-12-07 00:21:44 +00:00
|
|
|
filename(filename), fileSize(0)
|
2015-12-16 20:30:58 +00:00
|
|
|
{
|
2016-12-07 00:21:44 +00:00
|
|
|
char* headerData = (char*)((void*)&header);
|
2015-12-16 20:30:58 +00:00
|
|
|
vector<char>* data;
|
|
|
|
unsigned bytesToRead, headerOffset;
|
|
|
|
ostringstream oss;
|
|
|
|
|
2016-12-07 00:21:44 +00:00
|
|
|
if (!filename.empty()) {
|
|
|
|
ifs.open(filename.c_str(), ifstream::binary);
|
2015-12-16 20:30:58 +00:00
|
|
|
|
2016-12-07 00:21:44 +00:00
|
|
|
if (!ifs.is_open()) {
|
|
|
|
oss << "Cannot open " << filename << ", file does not exist";
|
|
|
|
throw ErrorReporter(oss.str());
|
|
|
|
}
|
2015-12-16 20:30:58 +00:00
|
|
|
|
2016-12-07 00:21:44 +00:00
|
|
|
is = &ifs;
|
|
|
|
|
|
|
|
is->seekg(0, is->end);
|
|
|
|
fileSize = is->tellg();
|
|
|
|
is->seekg(0, is->beg);
|
|
|
|
} else {
|
|
|
|
is = &cin;
|
2016-12-06 16:07:30 +00:00
|
|
|
}
|
2015-12-16 20:30:58 +00:00
|
|
|
|
|
|
|
try {
|
|
|
|
bytesToRead = sizeof(PCMWaveHeader::chunkID) + sizeof(PCMWaveHeader::chunkSize) + sizeof(PCMWaveHeader::format);
|
2016-12-06 16:07:30 +00:00
|
|
|
data = readData(bytesToRead, forceStop, true);
|
2016-12-07 00:21:44 +00:00
|
|
|
if (forceStop) {
|
|
|
|
return;
|
|
|
|
}
|
2015-12-16 20:30:58 +00:00
|
|
|
memcpy(headerData, &(*data)[0], bytesToRead);
|
|
|
|
headerOffset = bytesToRead;
|
|
|
|
delete data;
|
|
|
|
|
|
|
|
if ((string(header.chunkID, 4) != string("RIFF")) || (string(header.format, 4) != string("WAVE"))) {
|
2016-12-07 00:21:44 +00:00
|
|
|
oss << "Error while opening " << getFilename() << ", WAVE file expected";
|
2015-12-16 20:30:58 +00:00
|
|
|
throw ErrorReporter(oss.str());
|
|
|
|
}
|
|
|
|
|
|
|
|
bytesToRead = sizeof(PCMWaveHeader::subchunk1ID) + sizeof(PCMWaveHeader::subchunk1Size);
|
2016-12-06 16:07:30 +00:00
|
|
|
data = readData(bytesToRead, forceStop, true);
|
2016-12-07 00:21:44 +00:00
|
|
|
if (forceStop) {
|
|
|
|
return;
|
|
|
|
}
|
2015-12-16 20:30:58 +00:00
|
|
|
memcpy(&headerData[headerOffset], &(*data)[0], bytesToRead);
|
|
|
|
headerOffset += bytesToRead;
|
|
|
|
delete data;
|
|
|
|
|
|
|
|
unsigned subchunk1MinSize = sizeof(PCMWaveHeader) - headerOffset - sizeof(PCMWaveHeader::subchunk2ID) - sizeof(PCMWaveHeader::subchunk2Size);
|
|
|
|
if ((string(header.subchunk1ID, 4) != string("fmt ")) || (header.subchunk1Size < subchunk1MinSize)) {
|
2016-12-07 00:21:44 +00:00
|
|
|
oss << "Error while opening " << getFilename() << ", data corrupted";
|
2015-12-16 20:30:58 +00:00
|
|
|
throw ErrorReporter(oss.str());
|
|
|
|
}
|
|
|
|
|
2016-12-06 16:07:30 +00:00
|
|
|
data = readData(header.subchunk1Size, forceStop, true);
|
2016-12-07 00:21:44 +00:00
|
|
|
if (forceStop) {
|
|
|
|
return;
|
|
|
|
}
|
2015-12-16 20:30:58 +00:00
|
|
|
memcpy(&headerData[headerOffset], &(*data)[0], subchunk1MinSize);
|
|
|
|
headerOffset += subchunk1MinSize;
|
|
|
|
delete data;
|
|
|
|
|
|
|
|
if ((header.audioFormat != WAVE_FORMAT_PCM) ||
|
|
|
|
(header.byteRate != (header.bitsPerSample >> 3) * header.channels * header.sampleRate) ||
|
|
|
|
(header.blockAlign != (header.bitsPerSample >> 3) * header.channels) ||
|
|
|
|
(((header.bitsPerSample >> 3) != 1) && ((header.bitsPerSample >> 3) != 2))) {
|
2016-12-07 00:21:44 +00:00
|
|
|
oss << "Error while opening " << getFilename() << ", unsupported WAVE format";
|
2015-12-16 20:30:58 +00:00
|
|
|
throw ErrorReporter(oss.str());
|
|
|
|
}
|
|
|
|
|
|
|
|
bytesToRead = sizeof(PCMWaveHeader::subchunk2ID) + sizeof(PCMWaveHeader::subchunk2Size);
|
2016-12-06 16:07:30 +00:00
|
|
|
data = readData(bytesToRead, forceStop, true);
|
2016-12-07 00:21:44 +00:00
|
|
|
if (forceStop) {
|
|
|
|
return;
|
|
|
|
}
|
2015-12-16 20:30:58 +00:00
|
|
|
memcpy(&headerData[headerOffset], &(*data)[0], bytesToRead);
|
|
|
|
headerOffset += bytesToRead;
|
|
|
|
delete data;
|
|
|
|
|
|
|
|
if ((string(header.subchunk2ID, 4) != string("data")) || (header.subchunk2Size + ifs.tellg() < fileSize)) {
|
2016-12-07 00:21:44 +00:00
|
|
|
oss << "Error while opening " << getFilename() << ", data corrupted";
|
2015-12-16 20:30:58 +00:00
|
|
|
throw ErrorReporter(oss.str());
|
|
|
|
}
|
|
|
|
} catch (ErrorReporter &error) {
|
2016-12-07 00:21:44 +00:00
|
|
|
if (!filename.empty()) {
|
2016-12-06 16:07:30 +00:00
|
|
|
ifs.close();
|
|
|
|
}
|
2015-12-16 20:30:58 +00:00
|
|
|
throw error;
|
|
|
|
}
|
|
|
|
|
2016-12-07 00:21:44 +00:00
|
|
|
if (!filename.empty()) {
|
|
|
|
dataOffset = is->tellg();
|
|
|
|
}
|
2015-12-16 20:30:58 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
WaveReader::~WaveReader()
|
|
|
|
{
|
2016-12-07 00:21:44 +00:00
|
|
|
if (!filename.empty()) {
|
2016-12-06 16:07:30 +00:00
|
|
|
ifs.close();
|
|
|
|
}
|
2015-12-16 20:30:58 +00:00
|
|
|
}
|
|
|
|
|
2016-12-07 00:21:44 +00:00
|
|
|
vector<char>* WaveReader::readData(unsigned bytesToRead, bool &forceStop, bool closeFileOnException)
|
2015-12-16 20:30:58 +00:00
|
|
|
{
|
2016-12-07 00:21:44 +00:00
|
|
|
unsigned bytesRead = 0;
|
2016-12-06 16:07:30 +00:00
|
|
|
vector<char>* data = new vector<char>();
|
|
|
|
data->resize(bytesToRead);
|
|
|
|
|
2016-12-07 00:21:44 +00:00
|
|
|
while ((bytesRead < bytesToRead) && !forceStop) {
|
|
|
|
if (is->eof()) {
|
|
|
|
delete data;
|
|
|
|
|
|
|
|
if (closeFileOnException && !filename.empty()) {
|
|
|
|
ifs.close();
|
|
|
|
}
|
|
|
|
|
|
|
|
ostringstream oss;
|
|
|
|
oss << "Error while reading " << getFilename() << ", data corrupted";
|
|
|
|
throw ErrorReporter(oss.str());
|
|
|
|
}
|
|
|
|
bytesRead += is->readsome(&(*data)[0], bytesToRead - bytesRead);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (forceStop) {
|
|
|
|
delete data;
|
|
|
|
data = NULL;
|
2015-12-16 20:30:58 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return data;
|
|
|
|
}
|
|
|
|
|
2016-12-07 00:21:44 +00:00
|
|
|
vector<float>* WaveReader::getFrames(unsigned frameCount, unsigned frameOffset, bool &forceStop) {
|
2015-12-16 20:30:58 +00:00
|
|
|
unsigned bytesToRead, bytesLeft, bytesPerFrame, offset;
|
|
|
|
vector<float>* frames = new vector<float>();
|
|
|
|
vector<char>* data;
|
|
|
|
|
|
|
|
bytesPerFrame = (header.bitsPerSample >> 3) * header.channels;
|
|
|
|
bytesToRead = frameCount * bytesPerFrame;
|
|
|
|
|
2016-12-07 00:21:44 +00:00
|
|
|
if (!filename.empty()) {
|
|
|
|
bytesLeft = header.subchunk2Size - frameOffset * bytesPerFrame;
|
|
|
|
if (bytesToRead > bytesLeft) {
|
|
|
|
bytesToRead = bytesLeft - bytesLeft % bytesPerFrame;
|
|
|
|
frameCount = bytesToRead / bytesPerFrame;
|
|
|
|
}
|
|
|
|
is->seekg(dataOffset + frameOffset * bytesPerFrame);
|
2015-12-16 20:30:58 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
try {
|
2016-12-07 00:21:44 +00:00
|
|
|
data = readData(bytesToRead, forceStop, false);
|
2015-12-16 20:30:58 +00:00
|
|
|
} catch (ErrorReporter &error) {
|
|
|
|
delete frames;
|
|
|
|
throw error;
|
|
|
|
}
|
|
|
|
for (unsigned i = 0; i < frameCount; i++) {
|
|
|
|
offset = bytesPerFrame * i;
|
|
|
|
if (header.channels != 1) {
|
|
|
|
if (header.bitsPerSample != 8) {
|
|
|
|
frames->push_back(((int)(signed char)(*data)[offset + 1] + (int)(signed char)(*data)[offset + 3]) / (float)0x100);
|
|
|
|
} else {
|
|
|
|
frames->push_back(((int)(*data)[offset] + (int)(*data)[offset + 1]) / (float)0x100 - 1.0f);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if (header.bitsPerSample != 8) {
|
|
|
|
frames->push_back((signed char)(*data)[offset + 1] / (float)0x80);
|
|
|
|
} else {
|
|
|
|
frames->push_back((*data)[offset] / (float)0x80 - 1.0f);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
delete data;
|
|
|
|
|
|
|
|
return frames;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool WaveReader::isEnd(unsigned frameOffset)
|
|
|
|
{
|
|
|
|
return header.subchunk2Size <= frameOffset * (header.bitsPerSample >> 3) * header.channels;
|
|
|
|
}
|
|
|
|
|
2016-12-07 00:21:44 +00:00
|
|
|
string WaveReader::getFilename()
|
|
|
|
{
|
|
|
|
return filename.empty() ? "STDIN" : filename;
|
|
|
|
}
|
|
|
|
|
2015-12-16 20:30:58 +00:00
|
|
|
AudioFormat* WaveReader::getFormat()
|
|
|
|
{
|
|
|
|
AudioFormat* format = new AudioFormat;
|
|
|
|
format->channels = header.channels;
|
|
|
|
format->sampleRate = header.sampleRate;
|
|
|
|
format->bitsPerSample = header.bitsPerSample;
|
|
|
|
return format;
|
|
|
|
}
|