forked from mia/Aegisub
91d73ec8ea
1. cd aegisub/ 2. svn mv *cpp *h src/ 3. svn mv Makefile.am MatroskaParser.c auto4_perldata.inc bitmaps boost \ changelog.txt config gl include libosxutil libresrc md5.c msvc mythes.cxx \ mythes.hxx res.rc src/ 4. cd .. 5. svn mv FFmpegSource2/ INSTALL Makefile.am README acinclude.m4 \ autogen.sh automation/ bin build configure.in desktop dummy.txt lib \ libass/ m4macros/ packages/ po/ scripts/ universalchardet/ aegisub/ 6. mkdir -p docs/wiki_convert 7. svn add docs/wiki_convert 8. cd docs 9. svn mv aegisub_convert_docs.pl convert.bat output wiki_convert/ * See r2749 for full description. Originally committed to SVN as r2752.
112 lines
3.6 KiB
C++
112 lines
3.6 KiB
C++
// Copyright (c) 2007-2008 Fredrik Mellbin
|
|
//
|
|
// 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"
|
|
#include <string.h>
|
|
|
|
#define WAVE_FORMAT_IEEE_FLOAT 0x0003
|
|
#define WAVE_FORMAT_PCM 1
|
|
|
|
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
|
|
};
|
|
|
|
Wave64Writer::Wave64Writer(const char *Filename, uint16_t BitsPerSample, uint16_t Channels, uint32_t SamplesPerSec, bool IsFloat) : std::ofstream(Filename, std::ios::out | std::ios::binary | std::ios::trunc) {
|
|
BytesWritten = 0;
|
|
this->BitsPerSample = BitsPerSample;
|
|
this->Channels = Channels;
|
|
this->SamplesPerSec = SamplesPerSec;
|
|
this->IsFloat = IsFloat;
|
|
|
|
if (!is_open())
|
|
throw "Failed to open destination file for writing";
|
|
|
|
WriteHeader(true, IsFloat);
|
|
}
|
|
|
|
Wave64Writer::~Wave64Writer() {
|
|
WriteHeader(false, IsFloat);
|
|
close();
|
|
}
|
|
|
|
void Wave64Writer::WriteHeader(bool Initial, bool IsFloat) {
|
|
FFMS_WAVEFORMATEX WFEX;
|
|
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) {
|
|
Header[2] = 0x7F00000000000000ull;
|
|
} else {
|
|
Header[2] = BytesWritten + sizeof(Header);
|
|
}
|
|
|
|
memcpy(Header + 3, GuidWAVE, 16);
|
|
memcpy(Header + 5, Guidfmt, 16);
|
|
|
|
Header[7] = 48;
|
|
|
|
memcpy(Header + 8, &WFEX, sizeof(WFEX));
|
|
memcpy(Header + 11, Guiddata, 16);
|
|
|
|
if (Initial)
|
|
Header[13] = 0x7E00000000000000ull;
|
|
else
|
|
Header[13] = BytesWritten + 24;
|
|
|
|
std::ios::streampos CPos = tellp();
|
|
seekp(0, std::ios::beg);
|
|
write(reinterpret_cast<const char *>(Header), sizeof(Header));
|
|
if (!Initial)
|
|
seekp(CPos, std::ios::beg);
|
|
}
|
|
|
|
void Wave64Writer::WriteData(void *Data, std::streamsize Length) {
|
|
write(reinterpret_cast<char *>(Data), Length);
|
|
BytesWritten += Length;
|
|
}
|