2009-04-06 21:14:55 +02:00
|
|
|
// Copyright (c) 2007-2009 Fredrik Mellbin
|
2008-09-01 23:16:13 +02:00
|
|
|
//
|
|
|
|
// Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
|
|
// of this software and associated documentation files (the "Software"), to deal
|
|
|
|
// in the Software without restriction, including without limitation the rights
|
|
|
|
// to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
|
|
// copies of the Software, and to permit persons to whom the Software is
|
|
|
|
// furnished to do so, subject to the following conditions:
|
|
|
|
//
|
|
|
|
// The above copyright notice and this permission notice shall be included in
|
|
|
|
// all copies or substantial portions of the Software.
|
|
|
|
//
|
|
|
|
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
|
|
// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
|
|
// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
|
|
// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
|
|
// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
|
|
// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
|
|
|
// THE SOFTWARE.
|
|
|
|
|
|
|
|
#include "wave64writer.h"
|
2008-09-14 20:21:59 +02:00
|
|
|
#include <string.h>
|
2008-09-01 23:16:13 +02:00
|
|
|
|
2008-09-08 18:11:09 +02:00
|
|
|
#define WAVE_FORMAT_IEEE_FLOAT 0x0003
|
|
|
|
#define WAVE_FORMAT_PCM 1
|
|
|
|
|
2008-09-01 23:16:13 +02:00
|
|
|
static const uint8_t GuidRIFF[16]={
|
|
|
|
// {66666972-912E-11CF-A5D6-28DB04C10000}
|
|
|
|
0x72, 0x69, 0x66, 0x66, 0x2E, 0x91, 0xCF, 0x11, 0xA5, 0xD6, 0x28, 0xDB, 0x04, 0xC1, 0x00, 0x00
|
|
|
|
};
|
|
|
|
|
|
|
|
static const uint8_t GuidWAVE[16]={
|
|
|
|
// {65766177-ACF3-11D3-8CD1-00C04F8EDB8A}
|
|
|
|
0x77, 0x61, 0x76, 0x65, 0xF3, 0xAC, 0xD3, 0x11, 0x8C, 0xD1, 0x00, 0xC0, 0x4F, 0x8E, 0xDB, 0x8A
|
|
|
|
};
|
|
|
|
|
|
|
|
static const uint8_t Guidfmt[16]={
|
|
|
|
// {20746D66-ACF3-11D3-8CD1-00C04F8EDB8A}
|
|
|
|
0x66, 0x6D, 0x74, 0x20, 0xF3, 0xAC, 0xD3, 0x11, 0x8C, 0xD1, 0x00, 0xC0, 0x4F, 0x8E, 0xDB, 0x8A
|
|
|
|
};
|
|
|
|
|
|
|
|
static const uint8_t Guiddata[16]={
|
|
|
|
// {61746164-ACF3-11D3-8CD1-00C04F8EDB8A}
|
|
|
|
0x64, 0x61, 0x74, 0x61, 0xF3, 0xAC, 0xD3, 0x11, 0x8C, 0xD1, 0x00, 0xC0, 0x4F, 0x8E, 0xDB, 0x8A
|
|
|
|
};
|
|
|
|
|
2009-07-14 00:27:40 +02:00
|
|
|
Wave64Writer::Wave64Writer(const char *Filename, uint16_t BitsPerSample, uint16_t Channels, uint32_t SamplesPerSec, bool IsFloat) : WavFile(Filename, std::ios::out | std::ios::binary | std::ios::trunc) {
|
2008-09-01 23:16:13 +02:00
|
|
|
BytesWritten = 0;
|
|
|
|
this->BitsPerSample = BitsPerSample;
|
|
|
|
this->Channels = Channels;
|
|
|
|
this->SamplesPerSec = SamplesPerSec;
|
|
|
|
this->IsFloat = IsFloat;
|
|
|
|
|
2009-07-14 00:27:40 +02:00
|
|
|
if (!WavFile.is_open())
|
2008-09-08 18:11:09 +02:00
|
|
|
throw "Failed to open destination file for writing";
|
2008-09-01 23:16:13 +02:00
|
|
|
|
|
|
|
WriteHeader(true, IsFloat);
|
|
|
|
}
|
|
|
|
|
|
|
|
Wave64Writer::~Wave64Writer() {
|
|
|
|
WriteHeader(false, IsFloat);
|
2009-07-14 00:27:40 +02:00
|
|
|
WavFile.close();
|
2008-09-01 23:16:13 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void Wave64Writer::WriteHeader(bool Initial, bool IsFloat) {
|
2008-09-08 18:11:09 +02:00
|
|
|
FFMS_WAVEFORMATEX WFEX;
|
2008-09-01 23:16:13 +02:00
|
|
|
if (IsFloat)
|
|
|
|
WFEX.wFormatTag = WAVE_FORMAT_IEEE_FLOAT;
|
|
|
|
else
|
|
|
|
WFEX.wFormatTag = WAVE_FORMAT_PCM;
|
|
|
|
WFEX.nChannels = Channels;
|
|
|
|
WFEX.nSamplesPerSec = SamplesPerSec;
|
|
|
|
WFEX.nAvgBytesPerSec = (BitsPerSample * Channels * SamplesPerSec) / 8;
|
|
|
|
WFEX.nBlockAlign = (BitsPerSample * Channels) / 8;
|
|
|
|
WFEX.wBitsPerSample = BitsPerSample;
|
|
|
|
WFEX.cbSize = 0;
|
|
|
|
|
|
|
|
uint64_t Header[14];
|
|
|
|
|
|
|
|
memset(Header, 0, sizeof(Header));
|
|
|
|
|
|
|
|
memcpy(Header + 0, GuidRIFF, 16);
|
|
|
|
if (Initial) {
|
2008-09-14 20:21:59 +02:00
|
|
|
Header[2] = 0x7F00000000000000ull;
|
2008-09-01 23:16:13 +02:00
|
|
|
} else {
|
|
|
|
Header[2] = BytesWritten + sizeof(Header);
|
|
|
|
}
|
|
|
|
|
|
|
|
memcpy(Header + 3, GuidWAVE, 16);
|
|
|
|
memcpy(Header + 5, Guidfmt, 16);
|
|
|
|
|
2008-09-23 22:54:22 +02:00
|
|
|
Header[7] = 48;
|
2008-09-01 23:16:13 +02:00
|
|
|
|
|
|
|
memcpy(Header + 8, &WFEX, sizeof(WFEX));
|
|
|
|
memcpy(Header + 11, Guiddata, 16);
|
|
|
|
|
|
|
|
if (Initial)
|
2008-09-14 20:21:59 +02:00
|
|
|
Header[13] = 0x7E00000000000000ull;
|
2008-09-01 23:16:13 +02:00
|
|
|
else
|
|
|
|
Header[13] = BytesWritten + 24;
|
|
|
|
|
2009-07-14 00:27:40 +02:00
|
|
|
std::streampos CPos = WavFile.tellp();
|
|
|
|
WavFile.seekp(0, std::ios::beg);
|
|
|
|
WavFile.write(reinterpret_cast<const char *>(Header), sizeof(Header));
|
2008-09-01 23:16:13 +02:00
|
|
|
if (!Initial)
|
2009-07-14 00:27:40 +02:00
|
|
|
WavFile.seekp(CPos, std::ios::beg);
|
2008-09-01 23:16:13 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void Wave64Writer::WriteData(void *Data, std::streamsize Length) {
|
2009-07-14 00:27:40 +02:00
|
|
|
WavFile.write(reinterpret_cast<char *>(Data), Length);
|
2008-09-01 23:16:13 +02:00
|
|
|
BytesWritten += Length;
|
|
|
|
}
|