2121 lines
70 KiB
C++
2121 lines
70 KiB
C++
#include "ehs/io/audio/Audio.h"
|
|
#include "ehs/io/File.h"
|
|
#include "ehs/io/RIFF.h"
|
|
|
|
namespace ehs
|
|
{
|
|
Array<AudioCodec> Audio::codecs;
|
|
|
|
bool Audio::HasCodec(const UInt_64 hashExt)
|
|
{
|
|
for (UInt_64 i = 0; i < codecs.Size(); ++i)
|
|
if (codecs[i].GetHashExt() == hashExt)
|
|
return true;
|
|
|
|
return false;
|
|
}
|
|
|
|
bool Audio::HasCodec(const Str_8& ext)
|
|
{
|
|
return HasCodec(ext.Hash_64());
|
|
}
|
|
|
|
bool Audio::AddCodec(AudioCodec codec)
|
|
{
|
|
if (HasCodec(codec.GetHashExt()))
|
|
return false;
|
|
|
|
codecs.Push(std::move(codec));
|
|
|
|
return true;
|
|
}
|
|
|
|
const AudioCodec* Audio::GetCodec(const UInt_64 hashExt)
|
|
{
|
|
for (UInt_64 i = 0; i < codecs.Size(); ++i)
|
|
if (codecs[i].GetHashExt() == hashExt)
|
|
return &codecs[i];
|
|
|
|
return nullptr;
|
|
}
|
|
|
|
const AudioCodec* Audio::GetCodec(const Str_8& ext)
|
|
{
|
|
return GetCodec(ext.Hash_64());
|
|
}
|
|
|
|
Audio::~Audio()
|
|
{
|
|
delete[] data;
|
|
delete[] peak;
|
|
}
|
|
|
|
Audio::Audio()
|
|
: sampleRate(0), dataType(DataType::FLOAT), byteDepth(0), channels(0), frames(0), length(0.0f), data(nullptr),
|
|
peak(nullptr)
|
|
{
|
|
AddType("Audio");
|
|
}
|
|
|
|
Audio::Audio(const Str_8& filePath)
|
|
: Resource(File::ParseName_8(filePath)), sampleRate(0), dataType(DataType::FLOAT), byteDepth(0), channels(0),
|
|
frames(0), length(0.0f), data(nullptr), peak(nullptr)
|
|
{
|
|
AddType("Audio");
|
|
|
|
File file(filePath, Mode::READ, Disposition::OPEN);
|
|
Str_8 ext = file.GetExtension();
|
|
|
|
const AudioCodec* codec = GetCodec(ext);
|
|
if (!codec)
|
|
{
|
|
EHS_LOG_INT(LogType::ERR, 0, "Codec not found for file extension, \"" + ext + "\".");
|
|
return;
|
|
}
|
|
|
|
Serializer<UInt_64> data = file.ReadSerializer_64(codec->GetEndianness(), file.Size());
|
|
|
|
file.Release();
|
|
|
|
codec->Decode(data, this);
|
|
}
|
|
|
|
Audio::Audio(const Str_8& filePath, DataType type)
|
|
: Resource(File::ParseName_8(filePath)), sampleRate(0), dataType(DataType::FLOAT), byteDepth(0), channels(0),
|
|
frames(0), length(0.0f), data(nullptr), peak(nullptr)
|
|
{
|
|
AddType("Audio");
|
|
|
|
File file(filePath, Mode::READ, Disposition::OPEN);
|
|
Str_8 ext = file.GetExtension();
|
|
|
|
const AudioCodec* codec = GetCodec(ext);
|
|
if (!codec)
|
|
{
|
|
EHS_LOG_INT(LogType::ERR, 0, "Codec not found for file extension, \"" + ext + "\".");
|
|
return;
|
|
}
|
|
|
|
Serializer<UInt_64> data = file.ReadSerializer_64(codec->GetEndianness(), file.Size());
|
|
|
|
file.Release();
|
|
|
|
codec->Decode(data, this);
|
|
|
|
ToDataType(type);
|
|
}
|
|
|
|
Audio::Audio(Str_8 id, const UInt_64 sampleRate, const DataType dataType, const UInt_8 channels, const UInt_64 frames, const Byte* const data)
|
|
: Resource((Str_8&&)id), dataType(dataType), byteDepth(ToByteDepth(dataType)), sampleRate(sampleRate),
|
|
channels(channels), frames(frames), length((float)frames / (float)sampleRate),
|
|
data(new Byte[GetSize()]), peak(new Byte[byteDepth])
|
|
{
|
|
Util::Copy(this->data, data, GetSize());
|
|
|
|
AddType("Audio");
|
|
}
|
|
|
|
Audio::Audio(Str_8 id, const UInt_64 sampleRate, const DataType dataType, const UInt_8 channels, const Serializer<UInt_64>& data)
|
|
: Resource((Str_8&&)id), sampleRate(sampleRate), dataType(dataType), byteDepth(ToByteDepth(dataType)),
|
|
channels(channels), frames(data.Size() / channels / byteDepth), length((float)frames / (float)sampleRate),
|
|
data(new Byte[data.Size()]), peak(new Byte[byteDepth])
|
|
{
|
|
Util::Copy(this->data, data, data.Size());
|
|
|
|
AddType("Audio");
|
|
}
|
|
|
|
Audio::Audio(Str_8 id, const UInt_64 sampleRate, const DataType dataType, const UInt_8 channels, const Vector<Byte>& data)
|
|
: Resource((Str_8&&)id), sampleRate(sampleRate), dataType(dataType), byteDepth(ToByteDepth(dataType)),
|
|
channels(channels), frames(data.Size() / channels / byteDepth), length((float)frames / (float)sampleRate),
|
|
data(new Byte[data.Size()]), peak(new Byte[byteDepth])
|
|
{
|
|
Util::Copy(this->data, data, data.Size());
|
|
|
|
AddType("Audio");
|
|
}
|
|
|
|
Audio::Audio(Str_8 id, const UInt_64 sampleRate, const DataType dataType, const UInt_8 channels, const Array<Byte>& data)
|
|
: Resource((Str_8&&)id), sampleRate(sampleRate), dataType(dataType), byteDepth(ToByteDepth(dataType)),
|
|
channels(channels), frames(data.Size() / channels / byteDepth), length((float)frames / (float)sampleRate),
|
|
data(new Byte[data.Size()]), peak(new Byte[byteDepth])
|
|
{
|
|
Util::Copy(this->data, data, data.Size());
|
|
|
|
AddType("Audio");
|
|
}
|
|
|
|
Audio::Audio(Str_8 id, const UInt_64 sampleRate, const DataType dataType, const UInt_8 channels, const UInt_64 frames)
|
|
: Resource((Str_8&&)id), sampleRate(sampleRate), dataType(dataType), byteDepth(ToByteDepth(dataType)),
|
|
channels(channels), frames(frames), length((float)frames / (float)sampleRate),
|
|
data(new Byte[GetSize()]), peak(new Byte[byteDepth])
|
|
{
|
|
AddType("Audio");
|
|
}
|
|
|
|
Audio::Audio(Audio&& audio) noexcept
|
|
: Resource((Resource&&)audio), sampleRate(audio.sampleRate), dataType(audio.dataType),
|
|
byteDepth(audio.byteDepth), channels(audio.channels), frames(audio.frames), length(audio.length),
|
|
data(audio.data), peak(audio.peak)
|
|
{
|
|
audio.sampleRate = 0;
|
|
audio.dataType = DataType::FLOAT;
|
|
audio.byteDepth = ToByteDepth(audio.dataType);
|
|
audio.channels = 0;
|
|
audio.frames = 0;
|
|
audio.length = 0.0f;
|
|
audio.data = nullptr;
|
|
audio.peak = nullptr;
|
|
}
|
|
|
|
Audio::Audio(const Audio& audio)
|
|
: Resource(audio), sampleRate(audio.sampleRate), dataType(audio.dataType), byteDepth(audio.byteDepth),
|
|
channels(audio.channels), frames(audio.frames), length(audio.length), data(new Byte[GetSize()]),
|
|
peak(new Byte[audio.byteDepth])
|
|
{
|
|
Util::Copy(data, audio.data, GetSize());
|
|
}
|
|
|
|
Audio& Audio::operator=(Audio&& audio) noexcept
|
|
{
|
|
if (this == &audio)
|
|
return *this;
|
|
|
|
Resource::operator=((Resource&&)audio);
|
|
|
|
sampleRate = audio.sampleRate;
|
|
dataType = audio.dataType;
|
|
byteDepth = audio.byteDepth;
|
|
channels = audio.channels;
|
|
frames = audio.frames;
|
|
length = audio.length;
|
|
delete[] data;
|
|
data = audio.data;
|
|
delete[] peak;
|
|
peak = audio.peak;
|
|
|
|
audio.sampleRate = 0;
|
|
audio.dataType = DataType::FLOAT;
|
|
audio.byteDepth = ToByteDepth(audio.dataType);
|
|
audio.channels = 0;
|
|
audio.frames = 0;
|
|
audio.length = 0.0f;
|
|
audio.data = nullptr;
|
|
audio.peak = nullptr;
|
|
|
|
return *this;
|
|
}
|
|
|
|
Audio& Audio::operator=(const Audio& audio)
|
|
{
|
|
if (this == &audio)
|
|
return *this;
|
|
|
|
Resource::operator=(audio);
|
|
|
|
sampleRate = audio.sampleRate;
|
|
dataType = audio.dataType;
|
|
byteDepth = audio.byteDepth;
|
|
channels = audio.channels;
|
|
length = audio.length;
|
|
frames = audio.frames;
|
|
|
|
delete[] data;
|
|
data = new Byte[audio.GetSize()];
|
|
Util::Copy(data, audio.data, audio.GetSize());
|
|
|
|
delete[] peak;
|
|
peak = new Byte[audio.byteDepth];
|
|
Util::Copy(peak, audio.peak, audio.byteDepth);
|
|
|
|
return *this;
|
|
}
|
|
|
|
Audio::operator const Byte*() const
|
|
{
|
|
return data;
|
|
}
|
|
|
|
Audio::operator Byte*()
|
|
{
|
|
return data;
|
|
}
|
|
|
|
void Audio::Release()
|
|
{
|
|
sampleRate = 0;
|
|
dataType = DataType::FLOAT;
|
|
byteDepth = 0;
|
|
channels = 0;
|
|
frames = 0;
|
|
length = 0.0f;
|
|
delete[] data;
|
|
data = nullptr;
|
|
delete[] peak;
|
|
peak = nullptr;
|
|
}
|
|
|
|
UInt_64 Audio::GetSampleRate() const
|
|
{
|
|
return sampleRate;
|
|
}
|
|
|
|
DataType Audio::GetDataType() const
|
|
{
|
|
return dataType;
|
|
}
|
|
|
|
UInt_8 Audio::GetByteDepth() const
|
|
{
|
|
return byteDepth;
|
|
}
|
|
|
|
UInt_8 Audio::GetBitDepth() const
|
|
{
|
|
return byteDepth * 8;
|
|
}
|
|
|
|
UInt_8 Audio::GetChannels() const
|
|
{
|
|
return channels;
|
|
}
|
|
|
|
UInt_64 Audio::GetFrameCount() const
|
|
{
|
|
return frames;
|
|
}
|
|
|
|
UInt_64 Audio::GetSampleCount() const
|
|
{
|
|
return channels * frames;
|
|
}
|
|
|
|
UInt_8 Audio::GetFrameSize() const
|
|
{
|
|
return byteDepth * channels;
|
|
}
|
|
|
|
UInt_64 Audio::GetSize() const
|
|
{
|
|
return byteDepth * channels * frames;
|
|
}
|
|
|
|
float Audio::GetLength() const
|
|
{
|
|
return length;
|
|
}
|
|
|
|
Byte* Audio::GetFrame(UInt_64 frameIndex) const
|
|
{
|
|
return &data[frameIndex * GetFrameSize()];
|
|
}
|
|
|
|
Array<Byte> Audio::FrameAsMono(const UInt_64 frameIndex) const
|
|
{
|
|
Array<Byte> result(byteDepth);
|
|
ToMono(1, result, frameIndex);
|
|
return result;
|
|
}
|
|
|
|
Array<Byte> Audio::FrameAsStereo(const UInt_64 frameIndex) const
|
|
{
|
|
Array<Byte> result;
|
|
|
|
if (channels == 1)
|
|
{
|
|
result.Resize(byteDepth * channels);
|
|
Mono_to_Stereo(1, result, frameIndex);
|
|
}
|
|
else if (channels == 2)
|
|
{
|
|
return {&data[byteDepth * channels * frameIndex], (UInt_64)(byteDepth * channels)};
|
|
}
|
|
else if (channels == 6)
|
|
{
|
|
result.Resize(byteDepth * channels);
|
|
Five_One_to_Stereo(1, result, frameIndex);
|
|
}
|
|
else if (channels == 8)
|
|
{
|
|
result.Resize(byteDepth * channels);
|
|
Seven_One_to_Stereo(1, result, frameIndex);
|
|
}
|
|
else
|
|
{
|
|
EHS_LOG_INT(LogType::ERR, 0, "Conversion from " + Str_8::FromNum(channels) + " channels, to 2 channels is unsupported.");
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
Array<Byte> Audio::FrameAsFive_One(const UInt_64 frameIndex) const
|
|
{
|
|
Array<Byte> result;
|
|
|
|
if (channels == 1)
|
|
{
|
|
result.Resize(byteDepth * 6 );
|
|
Mono_to_Five_One(1, result, frameIndex);
|
|
}
|
|
else if (channels == 2)
|
|
{
|
|
result.Resize(byteDepth * 6);
|
|
Stereo_to_Five_One(1, result, frameIndex);
|
|
}
|
|
else if (channels == 6)
|
|
{
|
|
return {&data[byteDepth * channels * frameIndex], (UInt_64)(byteDepth * channels)};
|
|
}
|
|
else if (channels == 8)
|
|
{
|
|
result.Resize(byteDepth * 6);
|
|
Seven_One_to_Five_One(1, result, frameIndex);
|
|
}
|
|
else
|
|
{
|
|
EHS_LOG_INT(LogType::ERR, 0, "Conversion from " + Str_8::FromNum(channels) + " channels, to 6 channels is unsupported.");
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
Array<Byte> Audio::FrameAsSeven_One(const UInt_64 frameIndex) const
|
|
{
|
|
Array<Byte> result;
|
|
|
|
if (channels == 1)
|
|
{
|
|
result.Resize(byteDepth * 8);
|
|
Mono_to_Seven_One(1, result, frameIndex);
|
|
}
|
|
else if (channels == 2)
|
|
{
|
|
result.Resize(byteDepth * 8);
|
|
Stereo_to_Seven_One(1, result, frameIndex);
|
|
}
|
|
else if (channels == 6)
|
|
{
|
|
result.Resize(byteDepth * 8);
|
|
Five_One_to_Seven_One(1, result, frameIndex);
|
|
}
|
|
else if (channels == 8)
|
|
{
|
|
return {&data[byteDepth * channels * frameIndex], (UInt_64)(byteDepth * channels)};
|
|
}
|
|
else
|
|
{
|
|
EHS_LOG_INT(LogType::ERR, 0, "Conversion from " + Str_8::FromNum(channels) + " channels, to 8 channels is unsupported.");
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
SInt_8 Audio::SampleAsSInt_8(const UInt_64 sampleIndex) const
|
|
{
|
|
switch (dataType)
|
|
{
|
|
case DataType::SINT_8:
|
|
return ((SInt_8 *)data)[sampleIndex];
|
|
case DataType::SINT_16:
|
|
return (SInt_8)((double)((SInt_16 *)data)[sampleIndex] / (double)EHS_SINT_16_MAX * (double)EHS_SINT_8_MAX);
|
|
case DataType::FLOAT:
|
|
return (SInt_8)(((double *)data)[sampleIndex] * (double)EHS_SINT_8_MAX);
|
|
case DataType::SINT_32:
|
|
return (SInt_8)((double)((SInt_32 *)data)[sampleIndex] / (double)EHS_SINT_32_MAX * (double)EHS_SINT_8_MAX);
|
|
case DataType::SINT_64:
|
|
return (SInt_8)((double)((SInt_64 *)data)[sampleIndex] / (double)EHS_SINT_64_MAX * (double)EHS_SINT_8_MAX);
|
|
default:
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
SInt_16 Audio::SampleAsSInt_16(const UInt_64 sampleIndex) const
|
|
{
|
|
switch (dataType)
|
|
{
|
|
case DataType::SINT_8:
|
|
return (SInt_16)((double)((SInt_8*)data)[sampleIndex] / (double)EHS_SINT_8_MAX * (double)EHS_SINT_16_MAX);
|
|
case DataType::SINT_16:
|
|
return ((SInt_16 *)data)[sampleIndex];
|
|
case DataType::FLOAT:
|
|
return (SInt_16)(((double *)data)[sampleIndex] * (double)EHS_SINT_16_MAX);
|
|
case DataType::SINT_32:
|
|
return (SInt_16)((double)((SInt_32 *)data)[sampleIndex] / (double)EHS_SINT_32_MAX * (double)EHS_SINT_16_MAX);
|
|
case DataType::SINT_64:
|
|
return (SInt_16)((double)((SInt_64 *)data)[sampleIndex] / (double)EHS_SINT_64_MAX * (double)EHS_SINT_16_MAX);
|
|
default:
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
float Audio::SampleAsFloat(const UInt_64 sampleIndex) const
|
|
{
|
|
switch (dataType)
|
|
{
|
|
case DataType::SINT_8:
|
|
return (float)((SInt_8*)data)[sampleIndex] / (float)EHS_SINT_8_MAX;
|
|
case DataType::SINT_16:
|
|
return (float)((SInt_16*)data)[sampleIndex] / (float)EHS_SINT_16_MAX;
|
|
case DataType::FLOAT:
|
|
return ((float*)data)[sampleIndex];
|
|
case DataType::SINT_32:
|
|
return (float)((SInt_32*)data)[sampleIndex] / (float)EHS_SINT_32_MAX;
|
|
case DataType::SINT_64:
|
|
return (float)((SInt_64*)data)[sampleIndex] / (float)EHS_SINT_64_MAX;
|
|
default:
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
SInt_32 Audio::SampleAsSInt_32(const UInt_64 sampleIndex) const
|
|
{
|
|
switch (dataType)
|
|
{
|
|
case DataType::SINT_8:
|
|
return (SInt_32)((double)((SInt_8 *)data)[sampleIndex] / (double)EHS_SINT_8_MAX * (double)EHS_SINT_32_MAX);
|
|
case DataType::SINT_16:
|
|
return (SInt_32)((double)((SInt_16 *)data)[sampleIndex] / (double)EHS_SINT_16_MAX * (double)EHS_SINT_32_MAX);
|
|
case DataType::FLOAT:
|
|
return (SInt_32)(((double *)data)[sampleIndex] * (double)EHS_SINT_32_MAX);
|
|
case DataType::SINT_32:
|
|
return ((SInt_32 *)data)[sampleIndex];
|
|
case DataType::SINT_64:
|
|
return (SInt_32)((double)((SInt_64 *)data)[sampleIndex] / (double)EHS_SINT_64_MAX * (double)EHS_SINT_32_MAX);
|
|
default:
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
SInt_64 Audio::SampleAsSInt_64(const UInt_64 sampleIndex) const
|
|
{
|
|
switch (dataType)
|
|
{
|
|
case DataType::SINT_8:
|
|
return (SInt_64)((double)((SInt_8 *)data)[sampleIndex] / (double)EHS_SINT_8_MAX * (double)EHS_SINT_64_MAX);
|
|
case DataType::SINT_16:
|
|
return (SInt_64)((double)((SInt_16 *)data)[sampleIndex] / (double)EHS_SINT_16_MAX * (double)EHS_SINT_64_MAX);
|
|
case DataType::FLOAT:
|
|
return (SInt_64)(((double *)data)[sampleIndex] * (double)EHS_SINT_64_MAX);
|
|
case DataType::SINT_32:
|
|
return (SInt_64)((double)((SInt_32 *)data)[sampleIndex] / (double)EHS_SINT_32_MAX * (double)EHS_SINT_64_MAX);
|
|
case DataType::SINT_64:
|
|
return ((SInt_64 *)data)[sampleIndex];
|
|
default:
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
SInt_8 Audio::PeakAsSInt_8() const
|
|
{
|
|
switch (dataType)
|
|
{
|
|
case DataType::SINT_8:
|
|
return *(SInt_8*)peak;
|
|
case DataType::SINT_16:
|
|
return (SInt_8)((double)*(SInt_16 *)peak / (double)EHS_SINT_16_MAX * (double)EHS_SINT_8_MAX);
|
|
case DataType::FLOAT:
|
|
return (SInt_8)(*(double *)peak * (double)EHS_SINT_8_MAX);
|
|
case DataType::SINT_32:
|
|
return (SInt_8)((double)*(SInt_32 *)peak / (double)EHS_SINT_32_MAX * (double)EHS_SINT_8_MAX);
|
|
case DataType::SINT_64:
|
|
return (SInt_8)((double)*(SInt_64 *)peak / (double)EHS_SINT_64_MAX * (double)EHS_SINT_8_MAX);
|
|
default:
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
SInt_16 Audio::PeakAsSInt_16() const
|
|
{
|
|
switch (dataType)
|
|
{
|
|
case DataType::SINT_8:
|
|
return (SInt_16)((double)*(SInt_8 *)peak / (double)EHS_SINT_8_MAX * (double)EHS_SINT_16_MAX);
|
|
case DataType::SINT_16:
|
|
return *(SInt_16*)peak;
|
|
case DataType::FLOAT:
|
|
return (SInt_16)(*(double*)peak * (double)EHS_SINT_16_MAX);
|
|
case DataType::SINT_32:
|
|
return (SInt_16)((double)*(SInt_32 *)peak / (double)EHS_SINT_32_MAX * (double)EHS_SINT_16_MAX);
|
|
case DataType::SINT_64:
|
|
return (SInt_16)((double)*(SInt_64 *)peak / (double)EHS_SINT_64_MAX * (double)EHS_SINT_16_MAX);
|
|
default:
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
float Audio::PeakAsFloat() const
|
|
{
|
|
switch (dataType)
|
|
{
|
|
case DataType::SINT_8:
|
|
return (float)*(SInt_8*)peak / (float)EHS_SINT_8_MAX;
|
|
case DataType::SINT_16:
|
|
return (float)*(SInt_16*)peak / (float)EHS_SINT_16_MAX;
|
|
case DataType::FLOAT:
|
|
return *(float*)peak;
|
|
case DataType::SINT_32:
|
|
return (float)*(SInt_32*)peak / (float)EHS_SINT_32_MAX;
|
|
case DataType::SINT_64:
|
|
return (float)*(SInt_64*)peak / (float)EHS_SINT_64_MAX;
|
|
default:
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
SInt_32 Audio::PeakAsSInt_32() const
|
|
{
|
|
switch (dataType)
|
|
{
|
|
case DataType::SINT_8:
|
|
return (SInt_32)((double)*(SInt_8 *)peak / (double)EHS_SINT_8_MAX * (double)EHS_SINT_32_MAX);
|
|
case DataType::SINT_16:
|
|
return (SInt_32)((double)*(SInt_16 *)peak / (double)EHS_SINT_16_MAX * (double)EHS_SINT_32_MAX);
|
|
case DataType::FLOAT:
|
|
return (SInt_32)(*(double *)peak * (double)EHS_SINT_32_MAX);
|
|
case DataType::SINT_32:
|
|
return *(SInt_32 *)peak;
|
|
case DataType::SINT_64:
|
|
return (SInt_32)((double)*(SInt_64 *)peak / (double)EHS_SINT_64_MAX * (double)EHS_SINT_32_MAX);
|
|
default:
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
SInt_64 Audio::PeakAsSInt_64() const
|
|
{
|
|
switch (dataType)
|
|
{
|
|
case DataType::SINT_8:
|
|
return (SInt_64)((double)*(SInt_8 *)peak / (double)EHS_SINT_8_MAX * (double)EHS_SINT_64_MAX);
|
|
case DataType::SINT_16:
|
|
return (SInt_64)((double)*(SInt_16 *)peak / (double)EHS_SINT_16_MAX * (double)EHS_SINT_64_MAX);
|
|
case DataType::FLOAT:
|
|
return (SInt_64)(*(double *)peak * (double)EHS_SINT_64_MAX);
|
|
case DataType::SINT_32:
|
|
return (SInt_64)((double)*(SInt_32 *)peak / (double)EHS_SINT_32_MAX * (double)EHS_SINT_64_MAX);
|
|
case DataType::SINT_64:
|
|
return *(SInt_64 *)peak;
|
|
default:
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
void Audio::SetPeak(const UInt_64 size, const Byte* newPeak)
|
|
{
|
|
Util::Copy(peak, newPeak, size);
|
|
}
|
|
|
|
const Byte* Audio::GetPeak() const
|
|
{
|
|
return peak;
|
|
}
|
|
|
|
void Audio::ToDataType(const DataType newDataType)
|
|
{
|
|
if (dataType == newDataType)
|
|
return;
|
|
|
|
UInt_8 newByteDepth = ToByteDepth(newDataType);
|
|
|
|
Byte* newData = new Byte[GetSampleCount() * newByteDepth];
|
|
Byte* newPeak = new Byte[newByteDepth];
|
|
|
|
if (dataType == DataType::SINT_16 && newDataType == DataType::SINT_8)
|
|
SInt_16_to_SInt_8(newData, newPeak);
|
|
else if (dataType == DataType::FLOAT && newDataType == DataType::SINT_8)
|
|
Float_to_SInt_8(newData, newPeak);
|
|
else if (dataType == DataType::SINT_32 && newDataType == DataType::SINT_8)
|
|
SInt_32_to_SInt_8(newData, newPeak);
|
|
else if (dataType == DataType::SINT_64 && newDataType == DataType::SINT_8)
|
|
SInt_64_to_SInt_8(newData, newPeak);
|
|
else if (dataType == DataType::SINT_8 && newDataType == DataType::SINT_16)
|
|
SInt_8_to_SInt_16(newData, newPeak);
|
|
else if (dataType == DataType::FLOAT && newDataType == DataType::SINT_16)
|
|
Float_to_SInt_16(newData, newPeak);
|
|
else if (dataType == DataType::SINT_32 && newDataType == DataType::SINT_16)
|
|
SInt_32_to_SInt_16(newData, newPeak);
|
|
else if (dataType == DataType::SINT_64 && newDataType == DataType::SINT_16)
|
|
SInt_64_to_SInt_16(newData, newPeak);
|
|
else if (dataType == DataType::SINT_8 && newDataType == DataType::FLOAT)
|
|
SInt_8_to_Float(newData, newPeak);
|
|
else if (dataType == DataType::SINT_16 && newDataType == DataType::FLOAT)
|
|
SInt_16_to_Float(newData, newPeak);
|
|
else if (dataType == DataType::SINT_32 && newDataType == DataType::FLOAT)
|
|
SInt_32_to_Float(newData, newPeak);
|
|
else if (dataType == DataType::SINT_64 && newDataType == DataType::FLOAT)
|
|
SInt_64_to_Float(newData, newPeak);
|
|
else if (dataType == DataType::SINT_8 && newDataType == DataType::SINT_32)
|
|
SInt_8_to_SInt_32(newData, newPeak);
|
|
else if (dataType == DataType::SINT_16 && newDataType == DataType::SINT_32)
|
|
SInt_16_to_SInt_32(newData, newPeak);
|
|
else if (dataType == DataType::FLOAT && newDataType == DataType::SINT_32)
|
|
Float_to_SInt_32(newData, newPeak);
|
|
else if (dataType == DataType::SINT_64 && newDataType == DataType::SINT_32)
|
|
SInt_64_to_SInt_32(newData, newPeak);
|
|
else if (dataType == DataType::SINT_8 && newDataType == DataType::SINT_64)
|
|
SInt_8_to_SInt_64(newData, newPeak);
|
|
else if (dataType == DataType::SINT_16 && newDataType == DataType::SINT_64)
|
|
SInt_16_to_SInt_64(newData, newPeak);
|
|
else if (dataType == DataType::FLOAT && newDataType == DataType::SINT_64)
|
|
Float_to_SInt_64(newData, newPeak);
|
|
else if (dataType == DataType::SINT_32 && newDataType == DataType::SINT_64)
|
|
SInt_32_to_SInt_64(newData, newPeak);
|
|
|
|
dataType = newDataType;
|
|
byteDepth = ToByteDepth(newDataType);
|
|
|
|
delete[] data;
|
|
data = newData;
|
|
delete[] peak;
|
|
peak = newPeak;
|
|
}
|
|
|
|
Audio Audio::GetAsDataType(const DataType newDataType) const
|
|
{
|
|
if (dataType == newDataType)
|
|
return *this;
|
|
|
|
Audio result;
|
|
result.sampleRate = sampleRate;
|
|
result.dataType = newDataType;
|
|
result.byteDepth = ToByteDepth(newDataType);
|
|
result.channels = channels;
|
|
result.frames = frames;
|
|
result.length = length;
|
|
|
|
result.data = new Byte[result.GetSize()];
|
|
result.peak = new Byte[result.byteDepth];
|
|
|
|
if (dataType == DataType::SINT_16 && newDataType == DataType::SINT_8)
|
|
SInt_16_to_SInt_8(result.data, result.peak);
|
|
else if (dataType == DataType::FLOAT && newDataType == DataType::SINT_8)
|
|
Float_to_SInt_8(result.data, result.peak);
|
|
else if (dataType == DataType::SINT_32 && newDataType == DataType::SINT_8)
|
|
SInt_32_to_SInt_8(result.data, result.peak);
|
|
else if (dataType == DataType::SINT_64 && newDataType == DataType::SINT_8)
|
|
SInt_64_to_SInt_8(result.data, result.peak);
|
|
else if (dataType == DataType::SINT_8 && newDataType == DataType::SINT_16)
|
|
SInt_8_to_SInt_16(result.data, result.peak);
|
|
else if (dataType == DataType::FLOAT && newDataType == DataType::SINT_16)
|
|
Float_to_SInt_16(result.data, result.peak);
|
|
else if (dataType == DataType::SINT_32 && newDataType == DataType::SINT_16)
|
|
SInt_32_to_SInt_16(result.data, result.peak);
|
|
else if (dataType == DataType::SINT_64 && newDataType == DataType::SINT_16)
|
|
SInt_64_to_SInt_16(result.data, result.peak);
|
|
else if (dataType == DataType::SINT_8 && newDataType == DataType::FLOAT)
|
|
SInt_8_to_Float(result.data, result.peak);
|
|
else if (dataType == DataType::SINT_16 && newDataType == DataType::FLOAT)
|
|
SInt_16_to_Float(result.data, result.peak);
|
|
else if (dataType == DataType::SINT_32 && newDataType == DataType::FLOAT)
|
|
SInt_32_to_Float(result.data, result.peak);
|
|
else if (dataType == DataType::SINT_64 && newDataType == DataType::FLOAT)
|
|
SInt_64_to_Float(result.data, result.peak);
|
|
else if (dataType == DataType::SINT_8 && newDataType == DataType::SINT_32)
|
|
SInt_8_to_SInt_32(result.data, result.peak);
|
|
else if (dataType == DataType::SINT_16 && newDataType == DataType::SINT_32)
|
|
SInt_16_to_SInt_32(result.data, result.peak);
|
|
else if (dataType == DataType::FLOAT && newDataType == DataType::SINT_32)
|
|
Float_to_SInt_32(result.data, result.peak);
|
|
else if (dataType == DataType::SINT_64 && newDataType == DataType::SINT_32)
|
|
SInt_64_to_SInt_32(result.data, result.peak);
|
|
else if (dataType == DataType::SINT_8 && newDataType == DataType::SINT_64)
|
|
SInt_8_to_SInt_64(result.data, result.peak);
|
|
else if (dataType == DataType::SINT_16 && newDataType == DataType::SINT_64)
|
|
SInt_16_to_SInt_64(result.data, result.peak);
|
|
else if (dataType == DataType::FLOAT && newDataType == DataType::SINT_64)
|
|
Float_to_SInt_64(result.data, result.peak);
|
|
else if (dataType == DataType::SINT_32 && newDataType == DataType::SINT_64)
|
|
SInt_32_to_SInt_64(result.data, result.peak);
|
|
|
|
return result;
|
|
}
|
|
|
|
void Audio::ToChannels(const UInt_8 newChannels)
|
|
{
|
|
if (!data || newChannels == channels)
|
|
return;
|
|
|
|
Byte* result;
|
|
|
|
if (newChannels == 1)
|
|
{
|
|
result = new Byte[byteDepth * newChannels * frames];
|
|
ToMono(frames, result, 0);
|
|
}
|
|
else if (channels == 1 && newChannels == 2)
|
|
{
|
|
result = new Byte[byteDepth * newChannels * frames];
|
|
Mono_to_Stereo(frames, result, 0);
|
|
}
|
|
else if (channels == 6 && newChannels == 2)
|
|
{
|
|
result = new Byte[byteDepth * newChannels * frames];
|
|
Five_One_to_Stereo(frames, result, 0);
|
|
}
|
|
else if (channels == 8 && newChannels == 2)
|
|
{
|
|
result = new Byte[byteDepth * newChannels * frames];
|
|
Seven_One_to_Stereo(frames, result, 0);
|
|
}
|
|
else if (channels == 1 && newChannels == 6)
|
|
{
|
|
result = new Byte[byteDepth * newChannels * frames];
|
|
Mono_to_Five_One(frames, result, 0);
|
|
}
|
|
else if (channels == 2 && newChannels == 6)
|
|
{
|
|
result = new Byte[byteDepth * newChannels * frames];
|
|
Stereo_to_Five_One(frames, result, 0);
|
|
}
|
|
else if (channels == 8 && newChannels == 6)
|
|
{
|
|
result = new Byte[byteDepth * newChannels * frames];
|
|
Seven_One_to_Five_One(frames, result, 0);
|
|
}
|
|
else if (channels == 1 && newChannels == 8)
|
|
{
|
|
result = new Byte[byteDepth * newChannels * frames];
|
|
Mono_to_Seven_One(frames, result, 0);
|
|
}
|
|
else if (channels == 2 && newChannels == 8)
|
|
{
|
|
result = new Byte[byteDepth * newChannels * frames];
|
|
Stereo_to_Seven_One(frames, result, 0);
|
|
}
|
|
else if (channels == 6 && newChannels == 8)
|
|
{
|
|
result = new Byte[byteDepth * newChannels * frames];
|
|
Five_One_to_Seven_One(frames, result, 0);
|
|
}
|
|
else
|
|
{
|
|
EHS_LOG_INT(LogType::ERR, 0, "Conversion from " + Str_8::FromNum(channels) + " channels, to " +
|
|
Str_8::FromNum(newChannels) + " channels is unsupported.");
|
|
return;
|
|
}
|
|
|
|
channels = newChannels;
|
|
delete[] data;
|
|
data = result;
|
|
}
|
|
|
|
Audio Audio::GetAsChannels(const UInt_8 newChannels) const
|
|
{
|
|
Audio result;
|
|
|
|
if (!data || newChannels == channels)
|
|
return result;
|
|
|
|
if (newChannels == 1)
|
|
{
|
|
result.data = new Byte[byteDepth * newChannels * frames];
|
|
ToMono(frames, result.data, 0);
|
|
|
|
result.peak = new Byte[byteDepth];
|
|
Util::Copy(result.peak, peak, byteDepth);
|
|
}
|
|
else if (channels == 1 && newChannels == 2)
|
|
{
|
|
result.data = new Byte[byteDepth * newChannels * frames];
|
|
Mono_to_Stereo(frames, result.data, 0);
|
|
|
|
result.peak = new Byte[byteDepth];
|
|
Util::Copy(result.peak, peak, byteDepth);
|
|
}
|
|
else if (channels == 6 && newChannels == 2)
|
|
{
|
|
result.data = new Byte[byteDepth * newChannels * frames];
|
|
Five_One_to_Stereo(frames, result.data, 0);
|
|
|
|
result.peak = new Byte[byteDepth];
|
|
Util::Copy(result.peak, peak, byteDepth);
|
|
}
|
|
else if (channels == 8 && newChannels == 2)
|
|
{
|
|
result.data = new Byte[byteDepth * newChannels * frames];
|
|
Seven_One_to_Stereo(frames, result.data, 0);
|
|
|
|
result.peak = new Byte[byteDepth];
|
|
Util::Copy(result.peak, peak, byteDepth);
|
|
}
|
|
else if (channels == 1 && newChannels == 6)
|
|
{
|
|
result.data = new Byte[byteDepth * newChannels * frames];
|
|
Mono_to_Five_One(frames, result.data, 0);
|
|
|
|
result.peak = new Byte[byteDepth];
|
|
Util::Copy(result.peak, peak, byteDepth);
|
|
}
|
|
else if (channels == 2 && newChannels == 6)
|
|
{
|
|
result.data = new Byte[byteDepth * newChannels * frames];
|
|
Stereo_to_Five_One(frames, result.data, 0);
|
|
|
|
result.peak = new Byte[byteDepth];
|
|
Util::Copy(result.peak, peak, byteDepth);
|
|
}
|
|
else if (channels == 8 && newChannels == 6)
|
|
{
|
|
result.data = new Byte[byteDepth * newChannels * frames];
|
|
Seven_One_to_Five_One(frames, result.data, 0);
|
|
|
|
result.peak = new Byte[byteDepth];
|
|
Util::Copy(result.peak, peak, byteDepth);
|
|
}
|
|
else if (channels == 1 && newChannels == 8)
|
|
{
|
|
result.data = new Byte[byteDepth * newChannels * frames];
|
|
Mono_to_Seven_One(frames, result.data, 0);
|
|
|
|
result.peak = new Byte[byteDepth];
|
|
Util::Copy(result.peak, peak, byteDepth);
|
|
}
|
|
else if (channels == 2 && newChannels == 8)
|
|
{
|
|
result.data = new Byte[byteDepth * newChannels * frames];
|
|
Stereo_to_Seven_One(frames, result.data, 0);
|
|
|
|
result.peak = new Byte[byteDepth];
|
|
Util::Copy(result.peak, peak, byteDepth);
|
|
}
|
|
else if (channels == 6 && newChannels == 8)
|
|
{
|
|
result.data = new Byte[byteDepth * newChannels * frames];
|
|
Five_One_to_Seven_One(frames, result.data, 0);
|
|
|
|
result.peak = new Byte[byteDepth];
|
|
Util::Copy(result.peak, peak, byteDepth);
|
|
}
|
|
else
|
|
{
|
|
EHS_LOG_INT(LogType::ERR, 0, "Conversion from " + Str_8::FromNum(channels) + " channels, to " +
|
|
Str_8::FromNum(newChannels) + " channels is unsupported.");
|
|
|
|
return result;
|
|
}
|
|
|
|
result.sampleRate = sampleRate;
|
|
result.dataType = dataType;
|
|
result.byteDepth = byteDepth;
|
|
result.channels = newChannels;
|
|
result.frames = frames;
|
|
result.length = (float)frames / (float)sampleRate;
|
|
|
|
return result;
|
|
}
|
|
|
|
bool Audio::Export(const Str_8& filePath) const
|
|
{
|
|
Str_8 ext = File::ParseExt_8(filePath);
|
|
|
|
const AudioCodec* codec = GetCodec(ext);
|
|
if (!codec)
|
|
{
|
|
EHS_LOG_INT(LogType::ERR, 0, "Codec not found for file extension, \"" + ext + "\".");
|
|
return false;
|
|
}
|
|
|
|
Serializer<UInt_64> result;
|
|
if (!codec->Encode(result, this))
|
|
return false;
|
|
|
|
File file(filePath, Mode::WRITE, Disposition::CREATE_PERSISTENT);
|
|
file.WriteSerializer_64(result);
|
|
|
|
return true;
|
|
}
|
|
|
|
void Audio::ToMono(const UInt_64 newFrameCount, Byte* newData, const UInt_64 frameOffset) const
|
|
{
|
|
switch (dataType)
|
|
{
|
|
case DataType::SINT_8:
|
|
{
|
|
for (UInt_64 i = 0; i < newFrameCount; ++i)
|
|
{
|
|
UInt_64 sampleOffset = channels * (frameOffset + i);
|
|
((SInt_8*)newData)[i] = (((SInt_8*)data)[sampleOffset] + ((SInt_8*)data)[sampleOffset + 1]) / (PeakAsSInt_8() * 2);
|
|
}
|
|
return;
|
|
}
|
|
case DataType::SINT_16:
|
|
{
|
|
for (UInt_64 i = 0; i < newFrameCount; ++i)
|
|
{
|
|
UInt_64 sampleOffset = channels * (frameOffset + i);
|
|
((SInt_16*)newData)[i] = (((SInt_16*)data)[sampleOffset] + ((SInt_16*)data)[sampleOffset + 1]) / (PeakAsSInt_16() * 2);
|
|
}
|
|
return;
|
|
}
|
|
case DataType::FLOAT:
|
|
{
|
|
for (UInt_64 i = 0; i < newFrameCount; ++i)
|
|
{
|
|
UInt_64 sampleOffset = channels * (frameOffset + i);
|
|
((float*)newData)[i] = (((float*)data)[sampleOffset] + ((float*)data)[sampleOffset + 1]) / (PeakAsFloat() * 2.0f);
|
|
}
|
|
return;
|
|
}
|
|
case DataType::SINT_32:
|
|
{
|
|
for (UInt_64 i = 0; i < newFrameCount; ++i)
|
|
{
|
|
UInt_64 sampleOffset = channels * (frameOffset + i);
|
|
((SInt_32*)newData)[i] = (((SInt_32*)data)[sampleOffset] + ((SInt_32*)data)[sampleOffset + 1]) / (PeakAsSInt_32() * 2);
|
|
}
|
|
return;
|
|
}
|
|
case DataType::SINT_64:
|
|
{
|
|
for (UInt_64 i = 0; i < newFrameCount; ++i)
|
|
{
|
|
UInt_64 sampleOffset = channels * (frameOffset + i);
|
|
((SInt_64*)newData)[i] = (((SInt_64*)data)[sampleOffset] + ((SInt_64*)data)[sampleOffset + 1]) / (PeakAsSInt_64() * 2);
|
|
}
|
|
return;
|
|
}
|
|
default:
|
|
return;
|
|
}
|
|
}
|
|
|
|
void Audio::Mono_to_Stereo(const UInt_64 newFrameCount, Byte* newData, const UInt_64 frameOffset) const
|
|
{
|
|
switch (dataType)
|
|
{
|
|
case DataType::SINT_8:
|
|
{
|
|
for (UInt_64 i = 0; i < newFrameCount; ++i)
|
|
{
|
|
UInt_64 sampleOffsetA = 2 * i;
|
|
UInt_64 sampleOffsetB = frameOffset + i;
|
|
|
|
((SInt_8*)newData)[sampleOffsetA] = ((SInt_8*)data)[sampleOffsetB];
|
|
((SInt_8*)newData)[sampleOffsetA + 1] = ((SInt_8*)data)[sampleOffsetB];
|
|
}
|
|
return;
|
|
}
|
|
case DataType::SINT_16:
|
|
{
|
|
for (UInt_64 i = 0; i < newFrameCount; ++i)
|
|
{
|
|
UInt_64 sampleOffsetA = 2 * i;
|
|
UInt_64 sampleOffsetB = frameOffset + i;
|
|
|
|
((SInt_16*)newData)[sampleOffsetA] = ((SInt_16*)data)[sampleOffsetB];
|
|
((SInt_16*)newData)[sampleOffsetA + 1] = ((SInt_16*)data)[sampleOffsetB];
|
|
}
|
|
return;
|
|
}
|
|
case DataType::FLOAT:
|
|
{
|
|
for (UInt_64 i = 0; i < newFrameCount; ++i)
|
|
{
|
|
UInt_64 sampleOffsetA = 2 * i;
|
|
UInt_64 sampleOffsetB = frameOffset + i;
|
|
|
|
((float*)newData)[sampleOffsetA] = ((float*)data)[sampleOffsetB];
|
|
((float*)newData)[sampleOffsetA + 1] = ((float*)data)[sampleOffsetB];
|
|
}
|
|
return;
|
|
}
|
|
case DataType::SINT_32:
|
|
{
|
|
for (UInt_64 i = 0; i < newFrameCount; ++i)
|
|
{
|
|
UInt_64 sampleOffsetA = 2 * i;
|
|
UInt_64 sampleOffsetB = frameOffset + i;
|
|
|
|
((SInt_32*)newData)[sampleOffsetA] = ((SInt_32*)data)[sampleOffsetB];
|
|
((SInt_32*)newData)[sampleOffsetA + 1] = ((SInt_32*)data)[sampleOffsetB];
|
|
}
|
|
return;
|
|
}
|
|
case DataType::SINT_64:
|
|
{
|
|
for (UInt_64 i = 0; i < newFrameCount; ++i)
|
|
{
|
|
UInt_64 sampleOffsetA = 2 * i;
|
|
UInt_64 sampleOffsetB = frameOffset + i;
|
|
|
|
((SInt_64*)newData)[sampleOffsetA] = ((SInt_64*)data)[sampleOffsetB];
|
|
((SInt_64*)newData)[sampleOffsetA + 1] = ((SInt_64*)data)[sampleOffsetB];
|
|
}
|
|
return;
|
|
}
|
|
default:
|
|
return;
|
|
}
|
|
}
|
|
|
|
void Audio::Five_One_to_Stereo(const UInt_64 newFrameCount, Byte* newData, const UInt_64 frameOffset) const
|
|
{
|
|
switch (dataType)
|
|
{
|
|
case DataType::SINT_8:
|
|
{
|
|
for (UInt_64 i = 0; i < newFrameCount; ++i)
|
|
{
|
|
UInt_64 sampleOffsetA = 2 * i;
|
|
UInt_64 sampleOffsetB = 6 * (frameOffset + i);
|
|
|
|
((SInt_8*)newData)[sampleOffsetA] = ((SInt_8*)data)[sampleOffsetB];
|
|
((SInt_8*)newData)[sampleOffsetA + 1] = ((SInt_8*)data)[sampleOffsetB + 1];
|
|
}
|
|
return;
|
|
}
|
|
case DataType::SINT_16:
|
|
{
|
|
for (UInt_64 i = 0; i < newFrameCount; ++i)
|
|
{
|
|
UInt_64 sampleOffsetA = 2 * i;
|
|
UInt_64 sampleOffsetB = 6 * (frameOffset + i);
|
|
|
|
((SInt_16*)newData)[sampleOffsetA] = ((SInt_16*)data)[sampleOffsetB];
|
|
((SInt_16*)newData)[sampleOffsetA + 1] = ((SInt_16*)data)[sampleOffsetB + 1];
|
|
}
|
|
return;
|
|
}
|
|
case DataType::FLOAT:
|
|
{
|
|
for (UInt_64 i = 0; i < newFrameCount; ++i)
|
|
{
|
|
UInt_64 sampleOffsetA = 2 * i;
|
|
UInt_64 sampleOffsetB = 6 * (frameOffset + i);
|
|
|
|
((float*)newData)[sampleOffsetA] = ((float*)data)[sampleOffsetB];
|
|
((float*)newData)[sampleOffsetA + 1] = ((float*)data)[sampleOffsetB + 1];
|
|
}
|
|
return;
|
|
}
|
|
case DataType::SINT_32:
|
|
{
|
|
for (UInt_64 i = 0; i < newFrameCount; ++i)
|
|
{
|
|
UInt_64 sampleOffsetA = 2 * i;
|
|
UInt_64 sampleOffsetB = 6 * (frameOffset + i);
|
|
|
|
((SInt_32*)newData)[sampleOffsetA] = ((SInt_32*)data)[sampleOffsetB];
|
|
((SInt_32*)newData)[sampleOffsetA + 1] = ((SInt_32*)data)[sampleOffsetB + 1];
|
|
}
|
|
return;
|
|
}
|
|
case DataType::SINT_64:
|
|
{
|
|
for (UInt_64 i = 0; i < newFrameCount; ++i)
|
|
{
|
|
UInt_64 sampleOffsetA = 2 * i;
|
|
UInt_64 sampleOffsetB = 6 * (frameOffset + i);
|
|
|
|
((SInt_64*)newData)[sampleOffsetA] = ((SInt_64*)data)[sampleOffsetB];
|
|
((SInt_64*)newData)[sampleOffsetA + 1] = ((SInt_64*)data)[sampleOffsetB + 1];
|
|
}
|
|
return;
|
|
}
|
|
default:
|
|
return;
|
|
}
|
|
}
|
|
|
|
void Audio::Seven_One_to_Stereo(const UInt_64 newFrameCount, Byte* newData, const UInt_64 frameOffset) const
|
|
{
|
|
switch (dataType)
|
|
{
|
|
case DataType::SINT_8:
|
|
{
|
|
for (UInt_64 i = 0; i < newFrameCount; ++i)
|
|
{
|
|
UInt_64 sampleOffsetA = 2 * i;
|
|
UInt_64 sampleOffsetB = 8 * (frameOffset + i);
|
|
|
|
((SInt_8*)newData)[sampleOffsetA] = ((SInt_8*)data)[sampleOffsetB];
|
|
((SInt_8*)newData)[sampleOffsetA + 1] = ((SInt_8*)data)[sampleOffsetB + 1];
|
|
}
|
|
return;
|
|
}
|
|
case DataType::SINT_16:
|
|
{
|
|
for (UInt_64 i = 0; i < newFrameCount; ++i)
|
|
{
|
|
UInt_64 sampleOffsetA = 2 * i;
|
|
UInt_64 sampleOffsetB = 8 * (frameOffset + i);
|
|
|
|
((SInt_16*)newData)[sampleOffsetA] = ((SInt_16*)data)[sampleOffsetB];
|
|
((SInt_16*)newData)[sampleOffsetA + 1] = ((SInt_16*)data)[sampleOffsetB + 1];
|
|
}
|
|
return;
|
|
}
|
|
case DataType::FLOAT:
|
|
{
|
|
for (UInt_64 i = 0; i < newFrameCount; ++i)
|
|
{
|
|
UInt_64 sampleOffsetA = 2 * i;
|
|
UInt_64 sampleOffsetB = 8 * (frameOffset + i);
|
|
|
|
((float*)newData)[sampleOffsetA] = ((float*)data)[sampleOffsetB];
|
|
((float*)newData)[sampleOffsetA + 1] = ((float*)data)[sampleOffsetB + 1];
|
|
}
|
|
return;
|
|
}
|
|
case DataType::SINT_32:
|
|
{
|
|
for (UInt_64 i = 0; i < newFrameCount; ++i)
|
|
{
|
|
UInt_64 sampleOffsetA = 2 * i;
|
|
UInt_64 sampleOffsetB = 8 * (frameOffset + i);
|
|
|
|
((SInt_32*)newData)[sampleOffsetA] = ((SInt_32*)data)[sampleOffsetB];
|
|
((SInt_32*)newData)[sampleOffsetA + 1] = ((SInt_32*)data)[sampleOffsetB + 1];
|
|
}
|
|
return;
|
|
}
|
|
case DataType::SINT_64:
|
|
{
|
|
for (UInt_64 i = 0; i < newFrameCount; ++i)
|
|
{
|
|
UInt_64 sampleOffsetA = 2 * i;
|
|
UInt_64 sampleOffsetB = 8 * (frameOffset + i);
|
|
|
|
((SInt_64*)newData)[sampleOffsetA] = ((SInt_64*)data)[sampleOffsetB];
|
|
((SInt_64*)newData)[sampleOffsetA + 1] = ((SInt_64*)data)[sampleOffsetB + 1];
|
|
}
|
|
return;
|
|
}
|
|
default:
|
|
return;
|
|
}
|
|
}
|
|
|
|
void Audio::Mono_to_Five_One(const UInt_64 newFrameCount, Byte* newData, const UInt_64 frameOffset) const
|
|
{
|
|
switch (dataType)
|
|
{
|
|
case DataType::SINT_8:
|
|
{
|
|
for (UInt_64 i = 0; i < newFrameCount; ++i)
|
|
{
|
|
UInt_64 sampleOffsetA = 6 * i;
|
|
UInt_64 sampleOffsetB = frameOffset + i;
|
|
|
|
((SInt_8*)newData)[sampleOffsetA] = ((SInt_8*)data)[sampleOffsetB];
|
|
((SInt_8*)newData)[sampleOffsetA + 1] = ((SInt_8*)data)[sampleOffsetB];
|
|
((SInt_8*)newData)[sampleOffsetA + 2] = ((SInt_8*)data)[sampleOffsetB];
|
|
((SInt_8*)newData)[sampleOffsetA + 3] = ((SInt_8*)data)[sampleOffsetB];
|
|
((SInt_8*)newData)[sampleOffsetA + 4] = ((SInt_8*)data)[sampleOffsetB];
|
|
((SInt_8*)newData)[sampleOffsetA + 5] = ((SInt_8*)data)[sampleOffsetB];
|
|
}
|
|
return;
|
|
}
|
|
case DataType::SINT_16:
|
|
{
|
|
for (UInt_64 i = 0; i < newFrameCount; ++i)
|
|
{
|
|
UInt_64 sampleOffsetA = 6 * i;
|
|
UInt_64 sampleOffsetB = frameOffset + i;
|
|
|
|
((SInt_16*)newData)[sampleOffsetA] = ((SInt_16*)data)[sampleOffsetB];
|
|
((SInt_16*)newData)[sampleOffsetA + 1] = ((SInt_16*)data)[sampleOffsetB];
|
|
((SInt_16*)newData)[sampleOffsetA + 2] = ((SInt_16*)data)[sampleOffsetB];
|
|
((SInt_16*)newData)[sampleOffsetA + 3] = ((SInt_16*)data)[sampleOffsetB];
|
|
((SInt_16*)newData)[sampleOffsetA + 4] = ((SInt_16*)data)[sampleOffsetB];
|
|
((SInt_16*)newData)[sampleOffsetA + 5] = ((SInt_16*)data)[sampleOffsetB];
|
|
}
|
|
return;
|
|
}
|
|
case DataType::FLOAT:
|
|
{
|
|
for (UInt_64 i = 0; i < newFrameCount; ++i)
|
|
{
|
|
UInt_64 sampleOffsetA = 6 * i;
|
|
UInt_64 sampleOffsetB = frameOffset + i;
|
|
|
|
((float*)newData)[sampleOffsetA] = ((float*)data)[sampleOffsetB];
|
|
((float*)newData)[sampleOffsetA + 1] = ((float*)data)[sampleOffsetB];
|
|
((float*)newData)[sampleOffsetA + 2] = ((float*)data)[sampleOffsetB];
|
|
((float*)newData)[sampleOffsetA + 3] = ((float*)data)[sampleOffsetB];
|
|
((float*)newData)[sampleOffsetA + 4] = ((float*)data)[sampleOffsetB];
|
|
((float*)newData)[sampleOffsetA + 5] = ((float*)data)[sampleOffsetB];
|
|
}
|
|
return;
|
|
}
|
|
case DataType::SINT_32:
|
|
{
|
|
for (UInt_64 i = 0; i < newFrameCount; ++i)
|
|
{
|
|
UInt_64 sampleOffsetA = 6 * i;
|
|
UInt_64 sampleOffsetB = frameOffset + i;
|
|
|
|
((SInt_32*)newData)[sampleOffsetA] = ((SInt_32*)data)[sampleOffsetB];
|
|
((SInt_32*)newData)[sampleOffsetA + 1] = ((SInt_32*)data)[sampleOffsetB];
|
|
((SInt_32*)newData)[sampleOffsetA + 2] = ((SInt_32*)data)[sampleOffsetB];
|
|
((SInt_32*)newData)[sampleOffsetA + 3] = ((SInt_32*)data)[sampleOffsetB];
|
|
((SInt_32*)newData)[sampleOffsetA + 4] = ((SInt_32*)data)[sampleOffsetB];
|
|
((SInt_32*)newData)[sampleOffsetA + 5] = ((SInt_32*)data)[sampleOffsetB];
|
|
}
|
|
return;
|
|
}
|
|
case DataType::SINT_64:
|
|
{
|
|
for (UInt_64 i = 0; i < newFrameCount; ++i)
|
|
{
|
|
UInt_64 sampleOffsetA = 6 * i;
|
|
UInt_64 sampleOffsetB = frameOffset + i;
|
|
|
|
((SInt_64*)newData)[sampleOffsetA] = ((SInt_64*)data)[sampleOffsetB];
|
|
((SInt_64*)newData)[sampleOffsetA + 1] = ((SInt_64*)data)[sampleOffsetB];
|
|
((SInt_64*)newData)[sampleOffsetA + 2] = ((SInt_64*)data)[sampleOffsetB];
|
|
((SInt_64*)newData)[sampleOffsetA + 3] = ((SInt_64*)data)[sampleOffsetB];
|
|
((SInt_64*)newData)[sampleOffsetA + 4] = ((SInt_64*)data)[sampleOffsetB];
|
|
((SInt_64*)newData)[sampleOffsetA + 5] = ((SInt_64*)data)[sampleOffsetB];
|
|
}
|
|
return;
|
|
}
|
|
default:
|
|
return;
|
|
}
|
|
}
|
|
|
|
void Audio::Stereo_to_Five_One(const UInt_64 newFrameCount, Byte* newData, const UInt_64 frameOffset) const
|
|
{
|
|
switch (dataType)
|
|
{
|
|
case DataType::SINT_8:
|
|
{
|
|
for (UInt_64 i = 0; i < newFrameCount; ++i)
|
|
{
|
|
UInt_64 sampleOffsetA = 6 * i;
|
|
UInt_64 sampleOffsetB = 2 * (frameOffset + i);
|
|
|
|
((SInt_8*)newData)[sampleOffsetA] = ((SInt_8*)data)[sampleOffsetB];
|
|
((SInt_8*)newData)[sampleOffsetA + 1] = ((SInt_8*)data)[sampleOffsetB + 1];
|
|
((SInt_8*)newData)[sampleOffsetA + 2] = (((SInt_8*)data)[sampleOffsetB] + ((SInt_8*)data)[sampleOffsetB + 1]) / (PeakAsSInt_8() * 2);
|
|
((SInt_8*)newData)[sampleOffsetA + 3] = ((SInt_8*)newData)[sampleOffsetA + 2];
|
|
((SInt_8*)newData)[sampleOffsetA + 4] = ((SInt_8*)data)[sampleOffsetB];
|
|
((SInt_8*)newData)[sampleOffsetA + 5] = ((SInt_8*)data)[sampleOffsetB + 1];
|
|
}
|
|
return;
|
|
}
|
|
case DataType::SINT_16:
|
|
{
|
|
for (UInt_64 i = 0; i < newFrameCount; ++i)
|
|
{
|
|
UInt_64 sampleOffsetA = 6 * i;
|
|
UInt_64 sampleOffsetB = 2 * (frameOffset + i);
|
|
|
|
((SInt_16*)newData)[sampleOffsetA] = ((SInt_16*)data)[sampleOffsetB];
|
|
((SInt_16*)newData)[sampleOffsetA + 1] = ((SInt_16*)data)[sampleOffsetB + 1];
|
|
((SInt_16*)newData)[sampleOffsetA + 2] = (((SInt_16*)data)[sampleOffsetB] + ((SInt_16*)data)[sampleOffsetB + 1]) / (PeakAsSInt_16() * 2);
|
|
((SInt_16*)newData)[sampleOffsetA + 3] = ((SInt_16*)newData)[sampleOffsetA + 2];
|
|
((SInt_16*)newData)[sampleOffsetA + 4] = ((SInt_16*)data)[sampleOffsetB];
|
|
((SInt_16*)newData)[sampleOffsetA + 5] = ((SInt_16*)data)[sampleOffsetB + 1];
|
|
}
|
|
return;
|
|
}
|
|
case DataType::FLOAT:
|
|
{
|
|
for (UInt_64 i = 0; i < newFrameCount; ++i)
|
|
{
|
|
UInt_64 sampleOffsetA = 6 * i;
|
|
UInt_64 sampleOffsetB = 2 * (frameOffset + i);
|
|
|
|
((float*)newData)[sampleOffsetA] = ((float*)data)[sampleOffsetB];
|
|
((float*)newData)[sampleOffsetA + 1] = ((float*)data)[sampleOffsetB + 1];
|
|
((float*)newData)[sampleOffsetA + 2] = (((float*)data)[sampleOffsetB] + ((float*)data)[sampleOffsetB + 1]) / (PeakAsFloat() * 2.0f);
|
|
((float*)newData)[sampleOffsetA + 3] = ((float*)newData)[sampleOffsetA + 2];
|
|
((float*)newData)[sampleOffsetA + 4] = ((float*)data)[sampleOffsetB];
|
|
((float*)newData)[sampleOffsetA + 5] = ((float*)data)[sampleOffsetB + 1];
|
|
}
|
|
return;
|
|
}
|
|
case DataType::SINT_32:
|
|
{
|
|
for (UInt_64 i = 0; i < newFrameCount; ++i)
|
|
{
|
|
UInt_64 sampleOffsetA = 6 * i;
|
|
UInt_64 sampleOffsetB = 2 * (frameOffset + i);
|
|
|
|
((SInt_32*)newData)[sampleOffsetA] = ((SInt_32*)data)[sampleOffsetB];
|
|
((SInt_32*)newData)[sampleOffsetA + 1] = ((SInt_32*)data)[sampleOffsetB + 1];
|
|
((SInt_32*)newData)[sampleOffsetA + 2] = (((SInt_32*)data)[sampleOffsetB] + ((SInt_32*)data)[sampleOffsetB + 1]) / (PeakAsSInt_32() * 2);
|
|
((SInt_32*)newData)[sampleOffsetA + 3] = ((SInt_32*)newData)[sampleOffsetA + 2];
|
|
((SInt_32*)newData)[sampleOffsetA + 4] = ((SInt_32*)data)[sampleOffsetB];
|
|
((SInt_32*)newData)[sampleOffsetA + 5] = ((SInt_32*)data)[sampleOffsetB + 1];
|
|
}
|
|
return;
|
|
}
|
|
case DataType::SINT_64:
|
|
{
|
|
for (UInt_64 i = 0; i < newFrameCount; ++i)
|
|
{
|
|
UInt_64 sampleOffsetA = 6 * i;
|
|
UInt_64 sampleOffsetB = 2 * (frameOffset + i);
|
|
|
|
((SInt_64*)newData)[sampleOffsetA] = ((SInt_64*)data)[sampleOffsetB];
|
|
((SInt_64*)newData)[sampleOffsetA + 1] = ((SInt_64*)data)[sampleOffsetB + 1];
|
|
((SInt_64*)newData)[sampleOffsetA + 2] = (((SInt_64*)data)[sampleOffsetB] + ((SInt_64*)data)[sampleOffsetB + 1]) / (PeakAsSInt_64() * 2);
|
|
((SInt_64*)newData)[sampleOffsetA + 3] = ((SInt_64*)newData)[sampleOffsetA + 2];
|
|
((SInt_64*)newData)[sampleOffsetA + 4] = ((SInt_64*)data)[sampleOffsetB];
|
|
((SInt_64*)newData)[sampleOffsetA + 5] = ((SInt_64*)data)[sampleOffsetB + 1];
|
|
}
|
|
return;
|
|
}
|
|
default:
|
|
return;
|
|
}
|
|
}
|
|
|
|
void Audio::Seven_One_to_Five_One(const UInt_64 newFrameCount, Byte* newData, const UInt_64 frameOffset) const
|
|
{
|
|
switch (dataType)
|
|
{
|
|
case DataType::SINT_8:
|
|
{
|
|
for (UInt_64 i = 0; i < newFrameCount; ++i)
|
|
{
|
|
UInt_64 sampleOffsetA = 6 * i;
|
|
UInt_64 sampleOffsetB = 8 * (frameOffset + i);
|
|
|
|
((SInt_8*)newData)[sampleOffsetA] = ((SInt_8*)data)[sampleOffsetB];
|
|
((SInt_8*)newData)[sampleOffsetA + 1] = ((SInt_8*)data)[sampleOffsetB + 1];
|
|
((SInt_8*)newData)[sampleOffsetA + 2] = ((SInt_8*)data)[sampleOffsetB + 2];
|
|
((SInt_8*)newData)[sampleOffsetA + 3] = ((SInt_8*)data)[sampleOffsetB + 3];
|
|
((SInt_8*)newData)[sampleOffsetA + 4] = ((SInt_8*)data)[sampleOffsetB + 4];
|
|
((SInt_8*)newData)[sampleOffsetA + 5] = ((SInt_8*)data)[sampleOffsetB + 5];
|
|
}
|
|
return;
|
|
}
|
|
case DataType::SINT_16:
|
|
{
|
|
for (UInt_64 i = 0; i < newFrameCount; ++i)
|
|
{
|
|
UInt_64 sampleOffsetA = 6 * i;
|
|
UInt_64 sampleOffsetB = 8 * (frameOffset + i);
|
|
|
|
((SInt_16*)newData)[sampleOffsetA] = ((SInt_16*)data)[sampleOffsetB];
|
|
((SInt_16*)newData)[sampleOffsetA + 1] = ((SInt_16*)data)[sampleOffsetB + 1];
|
|
((SInt_16*)newData)[sampleOffsetA + 2] = ((SInt_16*)data)[sampleOffsetB + 2];
|
|
((SInt_16*)newData)[sampleOffsetA + 3] = ((SInt_16*)data)[sampleOffsetB + 3];
|
|
((SInt_16*)newData)[sampleOffsetA + 4] = ((SInt_16*)data)[sampleOffsetB + 4];
|
|
((SInt_16*)newData)[sampleOffsetA + 5] = ((SInt_16*)data)[sampleOffsetB + 5];
|
|
}
|
|
return;
|
|
}
|
|
case DataType::FLOAT:
|
|
{
|
|
for (UInt_64 i = 0; i < newFrameCount; ++i)
|
|
{
|
|
UInt_64 sampleOffsetA = 6 * i;
|
|
UInt_64 sampleOffsetB = 8 * (frameOffset + i);
|
|
|
|
((float*)newData)[sampleOffsetA] = ((float*)data)[sampleOffsetB];
|
|
((float*)newData)[sampleOffsetA + 1] = ((float*)data)[sampleOffsetB + 1];
|
|
((float*)newData)[sampleOffsetA + 2] = ((float*)data)[sampleOffsetB + 2];
|
|
((float*)newData)[sampleOffsetA + 3] = ((float*)data)[sampleOffsetB + 3];
|
|
((float*)newData)[sampleOffsetA + 4] = ((float*)data)[sampleOffsetB + 4];
|
|
((float*)newData)[sampleOffsetA + 5] = ((float*)data)[sampleOffsetB + 5];
|
|
}
|
|
return;
|
|
}
|
|
case DataType::SINT_32:
|
|
{
|
|
for (UInt_64 i = 0; i < newFrameCount; ++i)
|
|
{
|
|
UInt_64 sampleOffsetA = 6 * i;
|
|
UInt_64 sampleOffsetB = 8 * (frameOffset + i);
|
|
|
|
((SInt_32*)newData)[sampleOffsetA] = ((SInt_32*)data)[sampleOffsetB];
|
|
((SInt_32*)newData)[sampleOffsetA + 1] = ((SInt_32*)data)[sampleOffsetB + 1];
|
|
((SInt_32*)newData)[sampleOffsetA + 2] = ((SInt_32*)data)[sampleOffsetB + 2];
|
|
((SInt_32*)newData)[sampleOffsetA + 3] = ((SInt_32*)data)[sampleOffsetB + 3];
|
|
((SInt_32*)newData)[sampleOffsetA + 4] = ((SInt_32*)data)[sampleOffsetB + 4];
|
|
((SInt_32*)newData)[sampleOffsetA + 5] = ((SInt_32*)data)[sampleOffsetB + 5];
|
|
}
|
|
return;
|
|
}
|
|
case DataType::SINT_64:
|
|
{
|
|
for (UInt_64 i = 0; i < newFrameCount; ++i)
|
|
{
|
|
UInt_64 sampleOffsetA = 6 * i;
|
|
UInt_64 sampleOffsetB = 8 * (frameOffset + i);
|
|
|
|
((SInt_64*)newData)[sampleOffsetA] = ((SInt_64*)data)[sampleOffsetB];
|
|
((SInt_64*)newData)[sampleOffsetA + 1] = ((SInt_64*)data)[sampleOffsetB + 1];
|
|
((SInt_64*)newData)[sampleOffsetA + 2] = ((SInt_64*)data)[sampleOffsetB + 2];
|
|
((SInt_64*)newData)[sampleOffsetA + 3] = ((SInt_64*)data)[sampleOffsetB + 3];
|
|
((SInt_64*)newData)[sampleOffsetA + 4] = ((SInt_64*)data)[sampleOffsetB + 4];
|
|
((SInt_64*)newData)[sampleOffsetA + 5] = ((SInt_64*)data)[sampleOffsetB + 5];
|
|
}
|
|
return;
|
|
}
|
|
default:
|
|
return;
|
|
}
|
|
}
|
|
|
|
void Audio::Mono_to_Seven_One(const UInt_64 newFrameCount, Byte* newData, const UInt_64 frameOffset) const
|
|
{
|
|
switch (dataType)
|
|
{
|
|
case DataType::SINT_8:
|
|
{
|
|
for (UInt_64 i = 0; i < newFrameCount; ++i)
|
|
{
|
|
UInt_64 sampleOffsetA = 8 * i;
|
|
UInt_64 sampleOffsetB = frameOffset + i;
|
|
|
|
((SInt_8*)newData)[sampleOffsetA] = ((SInt_8*)data)[sampleOffsetB];
|
|
((SInt_8*)newData)[sampleOffsetA + 1] = ((SInt_8*)data)[sampleOffsetB];
|
|
((SInt_8*)newData)[sampleOffsetA + 2] = ((SInt_8*)data)[sampleOffsetB];
|
|
((SInt_8*)newData)[sampleOffsetA + 3] = ((SInt_8*)data)[sampleOffsetB];
|
|
((SInt_8*)newData)[sampleOffsetA + 4] = ((SInt_8*)data)[sampleOffsetB];
|
|
((SInt_8*)newData)[sampleOffsetA + 5] = ((SInt_8*)data)[sampleOffsetB];
|
|
((SInt_8*)newData)[sampleOffsetA + 6] = ((SInt_8*)data)[sampleOffsetB];
|
|
((SInt_8*)newData)[sampleOffsetA + 7] = ((SInt_8*)data)[sampleOffsetB];
|
|
}
|
|
return;
|
|
}
|
|
case DataType::SINT_16:
|
|
{
|
|
for (UInt_64 i = 0; i < newFrameCount; ++i)
|
|
{
|
|
UInt_64 sampleOffsetA = 8 * i;
|
|
UInt_64 sampleOffsetB = frameOffset + i;
|
|
|
|
((SInt_16*)newData)[sampleOffsetA] = ((SInt_16*)data)[sampleOffsetB];
|
|
((SInt_16*)newData)[sampleOffsetA + 1] = ((SInt_16*)data)[sampleOffsetB];
|
|
((SInt_16*)newData)[sampleOffsetA + 2] = ((SInt_16*)data)[sampleOffsetB];
|
|
((SInt_16*)newData)[sampleOffsetA + 3] = ((SInt_16*)data)[sampleOffsetB];
|
|
((SInt_16*)newData)[sampleOffsetA + 4] = ((SInt_16*)data)[sampleOffsetB];
|
|
((SInt_16*)newData)[sampleOffsetA + 5] = ((SInt_16*)data)[sampleOffsetB];
|
|
((SInt_16*)newData)[sampleOffsetA + 6] = ((SInt_16*)data)[sampleOffsetB];
|
|
((SInt_16*)newData)[sampleOffsetA + 7] = ((SInt_16*)data)[sampleOffsetB];
|
|
}
|
|
return;
|
|
}
|
|
case DataType::FLOAT:
|
|
{
|
|
for (UInt_64 i = 0; i < newFrameCount; ++i)
|
|
{
|
|
UInt_64 sampleOffsetA = 8 * i;
|
|
UInt_64 sampleOffsetB = frameOffset + i;
|
|
|
|
((float*)newData)[sampleOffsetA] = ((float*)data)[sampleOffsetB];
|
|
((float*)newData)[sampleOffsetA + 1] = ((float*)data)[sampleOffsetB];
|
|
((float*)newData)[sampleOffsetA + 2] = ((float*)data)[sampleOffsetB];
|
|
((float*)newData)[sampleOffsetA + 3] = ((float*)data)[sampleOffsetB];
|
|
((float*)newData)[sampleOffsetA + 4] = ((float*)data)[sampleOffsetB];
|
|
((float*)newData)[sampleOffsetA + 5] = ((float*)data)[sampleOffsetB];
|
|
((float*)newData)[sampleOffsetA + 6] = ((float*)data)[sampleOffsetB];
|
|
((float*)newData)[sampleOffsetA + 7] = ((float*)data)[sampleOffsetB];
|
|
}
|
|
return;
|
|
}
|
|
case DataType::SINT_32:
|
|
{
|
|
for (UInt_64 i = 0; i < newFrameCount; ++i)
|
|
{
|
|
UInt_64 sampleOffsetA = 8 * i;
|
|
UInt_64 sampleOffsetB = frameOffset + i;
|
|
|
|
((SInt_32*)newData)[sampleOffsetA] = ((SInt_32*)data)[sampleOffsetB];
|
|
((SInt_32*)newData)[sampleOffsetA + 1] = ((SInt_32*)data)[sampleOffsetB];
|
|
((SInt_32*)newData)[sampleOffsetA + 2] = ((SInt_32*)data)[sampleOffsetB];
|
|
((SInt_32*)newData)[sampleOffsetA + 3] = ((SInt_32*)data)[sampleOffsetB];
|
|
((SInt_32*)newData)[sampleOffsetA + 4] = ((SInt_32*)data)[sampleOffsetB];
|
|
((SInt_32*)newData)[sampleOffsetA + 5] = ((SInt_32*)data)[sampleOffsetB];
|
|
((SInt_32*)newData)[sampleOffsetA + 6] = ((SInt_32*)data)[sampleOffsetB];
|
|
((SInt_32*)newData)[sampleOffsetA + 7] = ((SInt_32*)data)[sampleOffsetB];
|
|
}
|
|
return;
|
|
}
|
|
case DataType::SINT_64:
|
|
{
|
|
for (UInt_64 i = 0; i < newFrameCount; ++i)
|
|
{
|
|
UInt_64 sampleOffsetA = 8 * i;
|
|
UInt_64 sampleOffsetB = frameOffset + i;
|
|
|
|
((SInt_64*)newData)[sampleOffsetA] = ((SInt_64*)data)[sampleOffsetB];
|
|
((SInt_64*)newData)[sampleOffsetA + 1] = ((SInt_64*)data)[sampleOffsetB];
|
|
((SInt_64*)newData)[sampleOffsetA + 2] = ((SInt_64*)data)[sampleOffsetB];
|
|
((SInt_64*)newData)[sampleOffsetA + 3] = ((SInt_64*)data)[sampleOffsetB];
|
|
((SInt_64*)newData)[sampleOffsetA + 4] = ((SInt_64*)data)[sampleOffsetB];
|
|
((SInt_64*)newData)[sampleOffsetA + 5] = ((SInt_64*)data)[sampleOffsetB];
|
|
((SInt_64*)newData)[sampleOffsetA + 6] = ((SInt_64*)data)[sampleOffsetB];
|
|
((SInt_64*)newData)[sampleOffsetA + 7] = ((SInt_64*)data)[sampleOffsetB];
|
|
}
|
|
return;
|
|
}
|
|
default:
|
|
return;
|
|
}
|
|
}
|
|
|
|
void Audio::Stereo_to_Seven_One(const UInt_64 newFrameCount, Byte* newData, const UInt_64 frameOffset) const
|
|
{
|
|
switch (dataType)
|
|
{
|
|
case DataType::SINT_8:
|
|
{
|
|
for (UInt_64 i = 0; i < newFrameCount; ++i)
|
|
{
|
|
UInt_64 sampleOffsetA = 8 * i;
|
|
UInt_64 sampleOffsetB = 2 * (frameOffset + i);
|
|
|
|
((SInt_8*)newData)[sampleOffsetA] = ((SInt_8*)data)[sampleOffsetB];
|
|
((SInt_8*)newData)[sampleOffsetA + 1] = ((SInt_8*)data)[sampleOffsetB + 1];
|
|
((SInt_8*)newData)[sampleOffsetA + 2] = (((SInt_8*)data)[sampleOffsetB] + ((SInt_8*)data)[sampleOffsetB + 1]) / (PeakAsSInt_8() * 2);
|
|
((SInt_8*)newData)[sampleOffsetA + 3] = ((SInt_8*)newData)[sampleOffsetA + 2];
|
|
((SInt_8*)newData)[sampleOffsetA + 4] = ((SInt_8*)data)[sampleOffsetB];
|
|
((SInt_8*)newData)[sampleOffsetA + 5] = ((SInt_8*)data)[sampleOffsetB + 1];
|
|
((SInt_8*)newData)[sampleOffsetA + 6] = ((SInt_8*)data)[sampleOffsetB];
|
|
((SInt_8*)newData)[sampleOffsetA + 7] = ((SInt_8*)data)[sampleOffsetB + 1];
|
|
}
|
|
return;
|
|
}
|
|
case DataType::SINT_16:
|
|
{
|
|
for (UInt_64 i = 0; i < newFrameCount; ++i)
|
|
{
|
|
UInt_64 sampleOffsetA = 8 * i;
|
|
UInt_64 sampleOffsetB = 2 * (frameOffset + i);
|
|
|
|
((SInt_16*)newData)[sampleOffsetA] = ((SInt_16*)data)[sampleOffsetB];
|
|
((SInt_16*)newData)[sampleOffsetA + 1] = ((SInt_16*)data)[sampleOffsetB + 1];
|
|
((SInt_16*)newData)[sampleOffsetA + 2] = (((SInt_16*)data)[sampleOffsetB] + ((SInt_16*)data)[sampleOffsetB + 1]) / (PeakAsSInt_16() * 2);
|
|
((SInt_16*)newData)[sampleOffsetA + 3] = ((SInt_16*)newData)[sampleOffsetA + 2];
|
|
((SInt_16*)newData)[sampleOffsetA + 4] = ((SInt_16*)data)[sampleOffsetB];
|
|
((SInt_16*)newData)[sampleOffsetA + 5] = ((SInt_16*)data)[sampleOffsetB + 1];
|
|
((SInt_16*)newData)[sampleOffsetA + 6] = ((SInt_16*)data)[sampleOffsetB];
|
|
((SInt_16*)newData)[sampleOffsetA + 7] = ((SInt_16*)data)[sampleOffsetB + 1];
|
|
}
|
|
return;
|
|
}
|
|
case DataType::FLOAT:
|
|
{
|
|
for (UInt_64 i = 0; i < newFrameCount; ++i)
|
|
{
|
|
UInt_64 sampleOffsetA = 8 * i;
|
|
UInt_64 sampleOffsetB = 2 * (frameOffset + i);
|
|
|
|
((float*)newData)[sampleOffsetA] = ((float*)data)[sampleOffsetB];
|
|
((float*)newData)[sampleOffsetA + 1] = ((float*)data)[sampleOffsetB + 1];
|
|
((float*)newData)[sampleOffsetA + 2] = (((float*)data)[sampleOffsetB] + ((float*)data)[sampleOffsetB + 1]) / (PeakAsFloat() * 2.0f);
|
|
((float*)newData)[sampleOffsetA + 3] = ((float*)newData)[sampleOffsetA + 2];
|
|
((float*)newData)[sampleOffsetA + 4] = ((float*)data)[sampleOffsetB];
|
|
((float*)newData)[sampleOffsetA + 5] = ((float*)data)[sampleOffsetB + 1];
|
|
((float*)newData)[sampleOffsetA + 6] = ((float*)data)[sampleOffsetB];
|
|
((float*)newData)[sampleOffsetA + 7] = ((float*)data)[sampleOffsetB + 1];
|
|
}
|
|
return;
|
|
}
|
|
case DataType::SINT_32:
|
|
{
|
|
for (UInt_64 i = 0; i < newFrameCount; ++i)
|
|
{
|
|
UInt_64 sampleOffsetA = 8 * i;
|
|
UInt_64 sampleOffsetB = 2 * (frameOffset + i);
|
|
|
|
((SInt_32*)newData)[sampleOffsetA] = ((SInt_32*)data)[sampleOffsetB];
|
|
((SInt_32*)newData)[sampleOffsetA + 1] = ((SInt_32*)data)[sampleOffsetB + 1];
|
|
((SInt_32*)newData)[sampleOffsetA + 2] = (((SInt_32*)data)[sampleOffsetB] + ((SInt_32*)data)[sampleOffsetB + 1]) / (PeakAsSInt_32() * 2);
|
|
((SInt_32*)newData)[sampleOffsetA + 3] = ((SInt_32*)newData)[sampleOffsetA + 2];
|
|
((SInt_32*)newData)[sampleOffsetA + 4] = ((SInt_32*)data)[sampleOffsetB];
|
|
((SInt_32*)newData)[sampleOffsetA + 5] = ((SInt_32*)data)[sampleOffsetB + 1];
|
|
((SInt_32*)newData)[sampleOffsetA + 6] = ((SInt_32*)data)[sampleOffsetB];
|
|
((SInt_32*)newData)[sampleOffsetA + 7] = ((SInt_32*)data)[sampleOffsetB + 1];
|
|
}
|
|
return;
|
|
}
|
|
case DataType::SINT_64:
|
|
{
|
|
for (UInt_64 i = 0; i < newFrameCount; ++i)
|
|
{
|
|
UInt_64 sampleOffsetA = 8 * i;
|
|
UInt_64 sampleOffsetB = 2 * (frameOffset + i);
|
|
|
|
((SInt_64*)newData)[sampleOffsetA] = ((SInt_64*)data)[sampleOffsetB];
|
|
((SInt_64*)newData)[sampleOffsetA + 1] = ((SInt_64*)data)[sampleOffsetB + 1];
|
|
((SInt_64*)newData)[sampleOffsetA + 2] = (((SInt_64*)data)[sampleOffsetB] + ((SInt_64*)data)[sampleOffsetB + 1]) / (PeakAsSInt_64() * 2);
|
|
((SInt_64*)newData)[sampleOffsetA + 3] = ((SInt_64*)newData)[sampleOffsetA + 2];
|
|
((SInt_64*)newData)[sampleOffsetA + 4] = ((SInt_64*)data)[sampleOffsetB];
|
|
((SInt_64*)newData)[sampleOffsetA + 5] = ((SInt_64*)data)[sampleOffsetB + 1];
|
|
((SInt_64*)newData)[sampleOffsetA + 6] = ((SInt_64*)data)[sampleOffsetB];
|
|
((SInt_64*)newData)[sampleOffsetA + 7] = ((SInt_64*)data)[sampleOffsetB + 1];
|
|
}
|
|
return;
|
|
}
|
|
default:
|
|
return;
|
|
}
|
|
}
|
|
|
|
void Audio::Five_One_to_Seven_One(const UInt_64 newFrameCount, Byte* newData, const UInt_64 frameOffset) const
|
|
{
|
|
switch (dataType)
|
|
{
|
|
case DataType::SINT_8:
|
|
{
|
|
for (UInt_64 i = 0; i < newFrameCount; ++i)
|
|
{
|
|
UInt_64 sampleOffsetA = 8 * i;
|
|
UInt_64 sampleOffsetB = 6 * (frameOffset + i);
|
|
|
|
((SInt_8*)newData)[sampleOffsetA] = ((SInt_8*)data)[sampleOffsetB];
|
|
((SInt_8*)newData)[sampleOffsetA + 1] = ((SInt_8*)data)[sampleOffsetB + 1];
|
|
((SInt_8*)newData)[sampleOffsetA + 2] = ((SInt_8*)data)[sampleOffsetB + 2];
|
|
((SInt_8*)newData)[sampleOffsetA + 3] = ((SInt_8*)data)[sampleOffsetB + 3];
|
|
((SInt_8*)newData)[sampleOffsetA + 4] = ((SInt_8*)data)[sampleOffsetB + 4];
|
|
((SInt_8*)newData)[sampleOffsetA + 5] = ((SInt_8*)data)[sampleOffsetB + 5];
|
|
((SInt_8*)newData)[sampleOffsetA + 6] = ((SInt_8*)data)[sampleOffsetB];
|
|
((SInt_8*)newData)[sampleOffsetA + 7] = ((SInt_8*)data)[sampleOffsetB + 1];
|
|
}
|
|
return;
|
|
}
|
|
case DataType::SINT_16:
|
|
{
|
|
for (UInt_64 i = 0; i < newFrameCount; ++i)
|
|
{
|
|
UInt_64 sampleOffsetA = 8 * i;
|
|
UInt_64 sampleOffsetB = 6 * (frameOffset + i);
|
|
|
|
((SInt_16*)newData)[sampleOffsetA] = ((SInt_16*)data)[sampleOffsetB];
|
|
((SInt_16*)newData)[sampleOffsetA + 1] = ((SInt_16*)data)[sampleOffsetB + 1];
|
|
((SInt_16*)newData)[sampleOffsetA + 2] = ((SInt_16*)data)[sampleOffsetB + 2];
|
|
((SInt_16*)newData)[sampleOffsetA + 3] = ((SInt_16*)data)[sampleOffsetB + 3];
|
|
((SInt_16*)newData)[sampleOffsetA + 4] = ((SInt_16*)data)[sampleOffsetB + 4];
|
|
((SInt_16*)newData)[sampleOffsetA + 5] = ((SInt_16*)data)[sampleOffsetB + 5];
|
|
((SInt_16*)newData)[sampleOffsetA + 6] = ((SInt_16*)data)[sampleOffsetB];
|
|
((SInt_16*)newData)[sampleOffsetA + 7] = ((SInt_16*)data)[sampleOffsetB + 1];
|
|
}
|
|
return;
|
|
}
|
|
case DataType::FLOAT:
|
|
{
|
|
for (UInt_64 i = 0; i < newFrameCount; ++i)
|
|
{
|
|
UInt_64 sampleOffsetA = 8 * i;
|
|
UInt_64 sampleOffsetB = 6 * (frameOffset + i);
|
|
|
|
((float*)newData)[sampleOffsetA] = ((float*)data)[sampleOffsetB];
|
|
((float*)newData)[sampleOffsetA + 1] = ((float*)data)[sampleOffsetB + 1];
|
|
((float*)newData)[sampleOffsetA + 2] = ((float*)data)[sampleOffsetB + 2];
|
|
((float*)newData)[sampleOffsetA + 3] = ((float*)data)[sampleOffsetB + 3];
|
|
((float*)newData)[sampleOffsetA + 4] = ((float*)data)[sampleOffsetB + 4];
|
|
((float*)newData)[sampleOffsetA + 5] = ((float*)data)[sampleOffsetB + 5];
|
|
((float*)newData)[sampleOffsetA + 6] = ((float*)data)[sampleOffsetB];
|
|
((float*)newData)[sampleOffsetA + 7] = ((float*)data)[sampleOffsetB + 1];
|
|
}
|
|
return;
|
|
}
|
|
case DataType::SINT_32:
|
|
{
|
|
for (UInt_64 i = 0; i < newFrameCount; ++i)
|
|
{
|
|
UInt_64 sampleOffsetA = 8 * i;
|
|
UInt_64 sampleOffsetB = 6 * (frameOffset + i);
|
|
|
|
((SInt_32*)newData)[sampleOffsetA] = ((SInt_32*)data)[sampleOffsetB];
|
|
((SInt_32*)newData)[sampleOffsetA + 1] = ((SInt_32*)data)[sampleOffsetB + 1];
|
|
((SInt_32*)newData)[sampleOffsetA + 2] = ((SInt_32*)data)[sampleOffsetB + 2];
|
|
((SInt_32*)newData)[sampleOffsetA + 3] = ((SInt_32*)data)[sampleOffsetB + 3];
|
|
((SInt_32*)newData)[sampleOffsetA + 4] = ((SInt_32*)data)[sampleOffsetB + 4];
|
|
((SInt_32*)newData)[sampleOffsetA + 5] = ((SInt_32*)data)[sampleOffsetB + 5];
|
|
((SInt_32*)newData)[sampleOffsetA + 6] = ((SInt_32*)data)[sampleOffsetB];
|
|
((SInt_32*)newData)[sampleOffsetA + 7] = ((SInt_32*)data)[sampleOffsetB + 1];
|
|
}
|
|
return;
|
|
}
|
|
case DataType::SINT_64:
|
|
{
|
|
for (UInt_64 i = 0; i < newFrameCount; ++i)
|
|
{
|
|
UInt_64 sampleOffsetA = 8 * i;
|
|
UInt_64 sampleOffsetB = 6 * (frameOffset + i);
|
|
|
|
((SInt_64*)newData)[sampleOffsetA] = ((SInt_64*)data)[sampleOffsetB];
|
|
((SInt_64*)newData)[sampleOffsetA + 1] = ((SInt_64*)data)[sampleOffsetB + 1];
|
|
((SInt_64*)newData)[sampleOffsetA + 2] = ((SInt_64*)data)[sampleOffsetB + 2];
|
|
((SInt_64*)newData)[sampleOffsetA + 3] = ((SInt_64*)data)[sampleOffsetB + 3];
|
|
((SInt_64*)newData)[sampleOffsetA + 4] = ((SInt_64*)data)[sampleOffsetB + 4];
|
|
((SInt_64*)newData)[sampleOffsetA + 5] = ((SInt_64*)data)[sampleOffsetB + 5];
|
|
((SInt_64*)newData)[sampleOffsetA + 6] = ((SInt_64*)data)[sampleOffsetB];
|
|
((SInt_64*)newData)[sampleOffsetA + 7] = ((SInt_64*)data)[sampleOffsetB + 1];
|
|
}
|
|
return;
|
|
}
|
|
default:
|
|
return;
|
|
}
|
|
}
|
|
|
|
void Audio::SInt_16_to_SInt_8(Byte* newData, Byte* newPeak) const
|
|
{
|
|
for (UInt_64 a = 0; a < GetSampleCount(); ++a)
|
|
newData[a] = (Byte)((double)((SInt_16*)data)[a] / (double)EHS_SINT_16_MAX * (double)EHS_SINT_8_MAX);
|
|
|
|
*(SInt_8*)newPeak = (SInt_8)((double)*(SInt_16*)peak / (double)EHS_SINT_16_MAX * (double)EHS_SINT_8_MAX);
|
|
}
|
|
|
|
void Audio::Float_to_SInt_8(Byte* newData, Byte* newPeak) const
|
|
{
|
|
for (UInt_64 a = 0; a < GetSampleCount(); ++a)
|
|
((SInt_8*)newData)[a] = (SInt_8)((double)((float*)data)[a] * (double)EHS_SINT_8_MAX);
|
|
|
|
*(SInt_8*)newPeak = (SInt_8)((double)(*(float*)peak) * (double)EHS_SINT_8_MAX);
|
|
}
|
|
|
|
void Audio::SInt_32_to_SInt_8(Byte* newData, Byte* newPeak) const
|
|
{
|
|
for (UInt_64 a = 0; a < GetSampleCount(); ++a)
|
|
newData[a] = (Byte)((double)((SInt_32*)data)[a] / (double)EHS_SINT_32_MAX * (double)EHS_SINT_8_MAX);
|
|
|
|
*(SInt_8*)newPeak = (SInt_8)((double)*(SInt_32*)peak / (double)EHS_SINT_32_MAX * (double)EHS_SINT_8_MAX);
|
|
}
|
|
|
|
void Audio::SInt_64_to_SInt_8(Byte* newData, Byte* newPeak) const
|
|
{
|
|
for (UInt_64 a = 0; a < GetSampleCount(); ++a)
|
|
newData[a] = (Byte)((double)((SInt_64*)data)[a] / (double)EHS_SINT_64_MAX * (double)EHS_SINT_8_MAX);
|
|
|
|
*(SInt_8*)newPeak = (SInt_8)((double)*(SInt_64*)peak / (double)EHS_SINT_64_MAX * (double)EHS_SINT_8_MAX);
|
|
}
|
|
|
|
void Audio::SInt_8_to_SInt_16(Byte* newData, Byte* newPeak) const
|
|
{
|
|
for (UInt_64 a = 0; a < GetSampleCount(); ++a)
|
|
((SInt_16*)newData)[a] = (SInt_16)((double)(SInt_8)data[a] / (double)EHS_SINT_8_MAX * (double)EHS_SINT_16_MAX);
|
|
|
|
*(SInt_16*)newPeak = (SInt_16)((double)*(SInt_8*)peak / (double)EHS_SINT_8_MAX * (double)EHS_SINT_16_MAX);
|
|
}
|
|
|
|
void Audio::Float_to_SInt_16(Byte* newData, Byte* newPeak) const
|
|
{
|
|
for (UInt_64 a = 0; a < GetSampleCount(); ++a)
|
|
((SInt_16*)newData)[a] = (SInt_16)((double)((float*)data)[a] * (double)EHS_SINT_16_MAX);
|
|
|
|
*(SInt_16*)newPeak = (SInt_16)((double)(*(float*)peak) * (double)EHS_SINT_16_MAX);
|
|
}
|
|
|
|
void Audio::SInt_32_to_SInt_16(Byte* newData, Byte* newPeak) const
|
|
{
|
|
for (UInt_64 a = 0; a < GetSampleCount(); ++a)
|
|
((SInt_16*)newData)[a] = (SInt_16)((double)((SInt_32*)data)[a] / (double)EHS_SINT_32_MAX * (double)EHS_SINT_16_MAX);
|
|
|
|
*(SInt_16*)newPeak = (SInt_16)((double)*(SInt_32*)peak / (double)EHS_SINT_32_MAX * (double)EHS_SINT_16_MAX);
|
|
}
|
|
|
|
void Audio::SInt_64_to_SInt_16(Byte* newData, Byte* newPeak) const
|
|
{
|
|
for (UInt_64 a = 0; a < GetSampleCount(); ++a)
|
|
((SInt_16*)newData)[a] = (SInt_16)((double)((SInt_64*)data)[a] / (double)EHS_SINT_64_MAX * (double)EHS_SINT_16_MAX);
|
|
|
|
*(SInt_16*)newPeak = (SInt_16)((double)*(SInt_64*)peak / (double)EHS_SINT_64_MAX * (double)EHS_SINT_16_MAX);
|
|
}
|
|
|
|
void Audio::SInt_8_to_Float(Byte* newData, Byte* newPeak) const
|
|
{
|
|
for (UInt_64 a = 0; a < GetSampleCount(); ++a)
|
|
((float*)newData)[a] = (float)((double)(((SInt_8*)data)[a]) / (double)EHS_SINT_8_MAX);
|
|
|
|
*(float*)newPeak = (float)((double)*(SInt_8*)peak / (double)EHS_SINT_8_MAX);
|
|
}
|
|
|
|
void Audio::SInt_16_to_Float(Byte* newData, Byte* newPeak) const
|
|
{
|
|
for (UInt_64 a = 0; a < GetSampleCount(); ++a)
|
|
((float*)newData)[a] = (float)((double)(((SInt_16*)data)[a]) / (double)EHS_SINT_16_MAX);
|
|
|
|
*(float*)newPeak = (float)((double)*(SInt_16*)peak / (double)EHS_SINT_16_MAX);
|
|
}
|
|
|
|
void Audio::SInt_32_to_Float(Byte* newData, Byte* newPeak) const
|
|
{
|
|
for (UInt_64 a = 0; a < GetSampleCount(); ++a)
|
|
((float*)newData)[a] = (float)((double)(((SInt_32*)data)[a]) / (double)EHS_SINT_32_MAX);
|
|
|
|
*(float*)newPeak = (float)((double)*(SInt_32*)peak / (double)EHS_SINT_32_MAX);
|
|
}
|
|
|
|
void Audio::SInt_64_to_Float(Byte* newData, Byte* newPeak) const
|
|
{
|
|
for (UInt_64 a = 0; a < GetSampleCount(); ++a)
|
|
((float*)newData)[a] = (float)((double)(((SInt_64*)data)[a]) / (double)EHS_SINT_64_MAX);
|
|
|
|
*(float*)newPeak = (float)((double)*(SInt_64*)peak / (double)EHS_SINT_64_MAX);
|
|
}
|
|
|
|
void Audio::SInt_8_to_SInt_32(Byte* newData, Byte* newPeak) const
|
|
{
|
|
for (UInt_64 a = 0; a < GetSampleCount(); ++a)
|
|
((SInt_32*)newData)[a] = (SInt_32)((double)(SInt_8)data[a] / (double)EHS_SINT_8_MAX * (double)EHS_SINT_32_MAX);
|
|
|
|
*(SInt_32*)newPeak = (SInt_32)((double)*(SInt_8*)peak / (double)EHS_SINT_8_MAX * (double)EHS_SINT_32_MAX);
|
|
}
|
|
|
|
void Audio::SInt_16_to_SInt_32(Byte* newData, Byte* newPeak) const
|
|
{
|
|
for (UInt_64 a = 0; a < GetSampleCount(); ++a)
|
|
((SInt_32*)newData)[a] = (SInt_32)((double)((SInt_16*)data)[a] / (double)EHS_SINT_16_MAX * (double)EHS_SINT_32_MAX);
|
|
|
|
*(SInt_32*)newPeak = (SInt_32)((double)*(SInt_16*)peak / (double)EHS_SINT_16_MAX * (double)EHS_SINT_32_MAX);
|
|
}
|
|
|
|
void Audio::Float_to_SInt_32(Byte* newData, Byte* newPeak) const
|
|
{
|
|
for (UInt_64 a = 0; a < GetSampleCount(); ++a)
|
|
((SInt_32*)newData)[a] = (SInt_32)(((double*)data)[a] * (double)EHS_SINT_32_MAX);
|
|
|
|
*(SInt_32*)newPeak = (SInt_32)(*(double*)peak * (double)EHS_SINT_32_MAX);
|
|
}
|
|
|
|
void Audio::SInt_64_to_SInt_32(Byte* newData, Byte* newPeak) const
|
|
{
|
|
for (UInt_64 a = 0; a < GetSampleCount(); ++a)
|
|
((SInt_32*)newData)[a] = (SInt_32)((double)((SInt_64*)data)[a] / (double)EHS_SINT_64_MAX * (double)EHS_SINT_32_MAX);
|
|
|
|
*(SInt_32*)newPeak = (SInt_32)((double)*(SInt_64*)peak / (double)EHS_SINT_64_MAX * (double)EHS_SINT_32_MAX);
|
|
}
|
|
|
|
void Audio::SInt_8_to_SInt_64(Byte* newData, Byte* newPeak) const
|
|
{
|
|
for (UInt_64 a = 0; a < GetSampleCount(); ++a)
|
|
((SInt_64*)newData)[a] = (SInt_64)((double)(SInt_8)data[a] / (double)EHS_SINT_8_MAX * (double)EHS_SINT_64_MAX);
|
|
|
|
*(SInt_64*)newPeak = (SInt_64)((double)*(SInt_8*)peak / (double)EHS_SINT_8_MAX * (double)EHS_SINT_64_MAX);
|
|
}
|
|
|
|
void Audio::SInt_16_to_SInt_64(Byte* newData, Byte* newPeak) const
|
|
{
|
|
for (UInt_64 a = 0; a < GetSampleCount(); ++a)
|
|
((SInt_64*)newData)[a] = (SInt_64)((double)((SInt_16*)data)[a] / (double)EHS_SINT_16_MAX * (double)EHS_SINT_64_MAX);
|
|
|
|
*(SInt_64*)newPeak = (SInt_64)((double)*(SInt_16*)peak / (double)EHS_SINT_16_MAX * (double)EHS_SINT_64_MAX);
|
|
}
|
|
|
|
void Audio::Float_to_SInt_64(Byte* newData, Byte* newPeak) const
|
|
{
|
|
for (UInt_64 a = 0; a < GetSampleCount(); ++a)
|
|
((SInt_64*)newData)[a] = (SInt_64)(((double*)data)[a] * (double)EHS_SINT_64_MAX);
|
|
|
|
*(SInt_64*)newPeak = (SInt_64)(*(double*)peak * (double)EHS_SINT_64_MAX);
|
|
}
|
|
|
|
void Audio::SInt_32_to_SInt_64(Byte* newData, Byte* newPeak) const
|
|
{
|
|
for (UInt_64 a = 0; a < GetSampleCount(); ++a)
|
|
((SInt_64*)newData)[a] = (SInt_64)((double)((SInt_32*)data)[a] / (double)EHS_SINT_32_MAX * (double)EHS_SINT_64_MAX);
|
|
|
|
*(SInt_64*)newPeak = (SInt_64)((double)*(SInt_32*)peak / (double)EHS_SINT_32_MAX * (double)EHS_SINT_64_MAX);
|
|
}
|
|
|
|
bool EncodeEHA(const ehs::AudioCodec* const codec, ehs::Serializer<ehs::UInt_64>& out, const ehs::Audio* const in)
|
|
{
|
|
Serializer<UInt_64> result(codec->GetEndianness());
|
|
result.WriteVersion({1, 0, 0});
|
|
result.Write(in->GetSampleRate());
|
|
result.Write(in->GetDataType());
|
|
result.Write(in->GetByteDepth());
|
|
result.Write(in->GetChannels());
|
|
result.Write(in->GetFrameCount());
|
|
|
|
UInt_64 size = in->GetSize();
|
|
UInt_8 byteDepth = in->GetByteDepth();
|
|
|
|
result.Resize(result.Size() + size + byteDepth);
|
|
Util::Copy(&result[result.GetOffset()], &in[0], size);
|
|
result.SetOffset(result.GetOffset() + size);
|
|
|
|
Util::Copy(&result[result.GetOffset()], in->GetPeak(), byteDepth);
|
|
|
|
return true;
|
|
}
|
|
|
|
bool DecodeEHA(const ehs::AudioCodec* const codec, ehs::Serializer<ehs::UInt_64>& in, ehs::Audio* const out)
|
|
{
|
|
Version version = in.ReadVersion();
|
|
if (version != Version(1, 0, 0))
|
|
{
|
|
EHS_LOG_INT(LogType::ERR, 0, "Incompatible audio file version.");
|
|
return false;
|
|
}
|
|
|
|
UInt_64 sampleRate = in.Read<UInt_64>();
|
|
DataType dataType = in.Read<DataType>();
|
|
UInt_8 byteDepth = in.Read<UInt_8>();
|
|
UInt_8 channels = in.Read<UInt_8>();
|
|
UInt_64 frames = in.Read<UInt_64>();
|
|
|
|
*out = Audio(out->GetId(), sampleRate, dataType, channels, frames);
|
|
|
|
UInt_64 size = out->GetSize();
|
|
Util::Copy(&(*out)[0], &in[in.GetOffset()], size);
|
|
in.SetOffset(in.GetOffset() + size);
|
|
|
|
out->SetPeak(byteDepth, &in[in.GetOffset()]);
|
|
|
|
return true;
|
|
}
|
|
|
|
bool DecodeWAV(const ehs::AudioCodec* const codec, ehs::Serializer<ehs::UInt_64>& in, ehs::Audio* out)
|
|
{
|
|
RIFF riff(in);
|
|
|
|
if (riff.GetType() != "WAVE")
|
|
{
|
|
EHS_LOG_INT(LogType::ERR, 0, "Data is not in WAVE format.");
|
|
return false;
|
|
}
|
|
|
|
RIFF_Chunk fmt = riff.GetChunk("fmt ");
|
|
if (!fmt.IsValid())
|
|
{
|
|
EHS_LOG_INT(LogType::ERR, 1, "Wave does not have a format chunk.");
|
|
return false;
|
|
}
|
|
|
|
Serializer<> fmtSer = fmt.GetData();
|
|
|
|
RIFF_Chunk dChunk = riff.GetChunk("data");
|
|
if (!dChunk.IsValid())
|
|
{
|
|
EHS_LOG_INT(LogType::ERR, 2, "Wave does not have a data chunk.");
|
|
return false;
|
|
}
|
|
|
|
UInt_16 compression = fmtSer.Read<UInt_16>();
|
|
if (compression == 0x2)
|
|
{
|
|
EHS_LOG_INT(LogType::ERR, 3, "Microsoft ADPCM compression unsupported.");
|
|
return false;
|
|
}
|
|
else if (compression == 0x6)
|
|
{
|
|
EHS_LOG_INT(LogType::ERR, 4, "ITU G.711 a-law compression unsupported.");
|
|
return false;
|
|
}
|
|
else if (compression == 0x7)
|
|
{
|
|
EHS_LOG_INT(LogType::ERR, 5, "ITU G.711 µ-law compression unsupported.");
|
|
return false;
|
|
}
|
|
else if (compression == 0x11)
|
|
{
|
|
EHS_LOG_INT(LogType::ERR, 6, "IMA ADPCM compression unsupported.");
|
|
return false;
|
|
}
|
|
else if (compression == 0x16)
|
|
{
|
|
EHS_LOG_INT(LogType::ERR, 7, "TU G.723 ADPCM (Yamaha) compression unsupported.");
|
|
return false;
|
|
}
|
|
else if (compression == 0x31)
|
|
{
|
|
EHS_LOG_INT(LogType::ERR, 8, "GSM 6.10 compression unsupported.");
|
|
return false;
|
|
}
|
|
else if (compression == 0x40)
|
|
{
|
|
EHS_LOG_INT(LogType::ERR, 9, "ITU G.721 ADPCM compression unsupported.");
|
|
return false;
|
|
}
|
|
else if (compression == 0x50)
|
|
{
|
|
EHS_LOG_INT(LogType::ERR, 10, "MPEG compression unsupported.");
|
|
return false;
|
|
}
|
|
else if (compression == 0xFFFF)
|
|
{
|
|
EHS_LOG_INT(LogType::ERR, 11, "Experimental compression unsupported.");
|
|
return false;
|
|
}
|
|
else if (compression != 0x1 && compression != 0x3)
|
|
{
|
|
EHS_LOG_INT(LogType::ERR, 12, "Wave has unknown compression of " + Str_8::FromNum(compression) + ".");
|
|
return false;
|
|
}
|
|
|
|
UInt_16 channels = fmtSer.Read<UInt_16>();
|
|
UInt_32 sampleRate = fmtSer.Read<UInt_32>();
|
|
fmtSer.SetOffset(fmtSer.GetOffset() + 6);
|
|
UInt_8 byteDepth = (UInt_8)(fmtSer.Read<UInt_16>() / 8);
|
|
|
|
DataType dataType;
|
|
if (byteDepth == 1)
|
|
dataType = DataType::SINT_8;
|
|
else if (byteDepth == 2)
|
|
dataType = DataType::SINT_16;
|
|
else if (byteDepth == 3)
|
|
dataType = DataType::SINT_24;
|
|
else if (byteDepth == 4 && compression == 0x3)
|
|
dataType = DataType::FLOAT;
|
|
else if (byteDepth == 4)
|
|
dataType = DataType::SINT_32;
|
|
else if (byteDepth == 8)
|
|
dataType = DataType::SINT_64;
|
|
else
|
|
return false;
|
|
|
|
UInt_64 size = dChunk.GetData().Size();
|
|
UInt_64 frames = size / byteDepth / channels;
|
|
|
|
*out = Audio(out->GetId(), sampleRate, dataType, channels, frames);
|
|
|
|
Serializer<> dataSer = dChunk.GetData();
|
|
|
|
for (UInt_32 i = 0; i < dataSer.Size(); i += byteDepth)
|
|
{
|
|
if (byteDepth == 1)
|
|
{
|
|
*(SInt_8*)&(*out)[i] = dataSer.Read<SInt_8>();
|
|
if ((*out)[i] > *(SInt_8*)out->GetPeak())
|
|
out->SetPeak(sizeof(SInt_8), &(*out)[i]);
|
|
}
|
|
else if (byteDepth == 2)
|
|
{
|
|
*(SInt_16*)&(*out)[i] = dataSer.Read<SInt_16>();
|
|
if (*(SInt_16*)&(*out)[i] > *(SInt_16*)out->GetPeak())
|
|
out->SetPeak(sizeof(SInt_16), &(*out)[i]);
|
|
}
|
|
else if (byteDepth == 3)
|
|
{
|
|
*(SInt_16*)&(*out)[i + 1] = dataSer.Read<SInt_16>();
|
|
(*out)[i] = dataSer.Read<Byte>();
|
|
|
|
SInt_32 signal = 0;
|
|
signal |= (*out)[i];
|
|
signal |= (*out)[i + 1] << 8;
|
|
signal |= (*out)[i + 2] << 16;
|
|
|
|
SInt_32 peak = 0;
|
|
peak |= out->GetPeak()[0];
|
|
peak |= out->GetPeak()[1] << 8;
|
|
peak |= out->GetPeak()[2] << 16;
|
|
|
|
if (signal > peak)
|
|
out->SetPeak(3, &(*out)[i]);
|
|
}
|
|
else if (byteDepth == 4 && compression == 0x3)
|
|
{
|
|
*(float*)&(*out)[i] = dataSer.Read<float>();
|
|
if (*(float*)&(*out)[i] > *(float*)out->GetPeak())
|
|
out->SetPeak(sizeof(float), &(*out)[i]);
|
|
}
|
|
else if (byteDepth == 4)
|
|
{
|
|
*(SInt_32*)&(*out)[i] = dataSer.Read<SInt_32>();
|
|
if (*(SInt_32*)&(*out)[i] > *(SInt_32*)out->GetPeak())
|
|
out->SetPeak(sizeof(SInt_32), &(*out)[i]);
|
|
}
|
|
else if (byteDepth == 8)
|
|
{
|
|
*(SInt_64*)&(*out)[i] = dataSer.Read<SInt_64>();
|
|
if (*(SInt_64*)&(*out)[i] > *(SInt_64*)out->GetPeak())
|
|
out->SetPeak(sizeof(SInt_64), &(*out)[i]);
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
} |