2018-12-19 16:12:57 +00:00
|
|
|
// Copyright 2018 Citra Emulator Project
|
|
|
|
// Licensed under GPLv2 or any later version
|
|
|
|
// Refer to the license.txt file included.
|
|
|
|
|
|
|
|
#include "audio_core/hle/wmf_decoder.h"
|
|
|
|
#include "audio_core/hle/wmf_decoder_utils.h"
|
|
|
|
|
|
|
|
namespace AudioCore::HLE {
|
|
|
|
|
|
|
|
class WMFDecoder::Impl {
|
|
|
|
public:
|
|
|
|
explicit Impl(Memory::MemorySystem& memory);
|
|
|
|
~Impl();
|
|
|
|
std::optional<BinaryResponse> ProcessRequest(const BinaryRequest& request);
|
|
|
|
|
|
|
|
private:
|
|
|
|
std::optional<BinaryResponse> Initalize(const BinaryRequest& request);
|
|
|
|
|
|
|
|
void Clear();
|
|
|
|
|
|
|
|
std::optional<BinaryResponse> Decode(const BinaryRequest& request);
|
|
|
|
|
2019-01-29 05:23:57 +00:00
|
|
|
MFOutputState DecodingLoop(ADTSData adts_header, std::array<std::vector<u8>, 2>& out_streams);
|
2018-12-19 16:12:57 +00:00
|
|
|
|
|
|
|
bool initalized = false;
|
|
|
|
bool selected = false;
|
|
|
|
|
|
|
|
Memory::MemorySystem& memory;
|
|
|
|
|
2019-01-14 21:27:43 +00:00
|
|
|
unique_mfptr<IMFTransform> transform;
|
2018-12-19 16:12:57 +00:00
|
|
|
DWORD in_stream_id = 0;
|
|
|
|
DWORD out_stream_id = 0;
|
|
|
|
};
|
|
|
|
|
|
|
|
WMFDecoder::Impl::Impl(Memory::MemorySystem& memory) : memory(memory) {
|
2019-01-06 04:53:24 +00:00
|
|
|
MFCoInit();
|
2018-12-19 16:12:57 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
WMFDecoder::Impl::~Impl() = default;
|
|
|
|
|
|
|
|
std::optional<BinaryResponse> WMFDecoder::Impl::ProcessRequest(const BinaryRequest& request) {
|
|
|
|
if (request.codec != DecoderCodec::AAC) {
|
|
|
|
LOG_ERROR(Audio_DSP, "Got unknown codec {}", static_cast<u16>(request.codec));
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
|
|
|
|
switch (request.cmd) {
|
|
|
|
case DecoderCommand::Init: {
|
2019-01-06 04:53:24 +00:00
|
|
|
LOG_INFO(Audio_DSP, "WMFDecoder initializing");
|
2018-12-19 16:12:57 +00:00
|
|
|
return Initalize(request);
|
|
|
|
}
|
|
|
|
case DecoderCommand::Decode: {
|
|
|
|
return Decode(request);
|
|
|
|
}
|
|
|
|
case DecoderCommand::Unknown: {
|
|
|
|
BinaryResponse response;
|
|
|
|
std::memcpy(&response, &request, sizeof(response));
|
|
|
|
response.unknown1 = 0x0;
|
|
|
|
return response;
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
LOG_ERROR(Audio_DSP, "Got unknown binary request: {}", static_cast<u16>(request.cmd));
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
std::optional<BinaryResponse> WMFDecoder::Impl::Initalize(const BinaryRequest& request) {
|
|
|
|
if (initalized) {
|
|
|
|
Clear();
|
|
|
|
}
|
|
|
|
|
|
|
|
BinaryResponse response;
|
2019-01-07 23:57:20 +00:00
|
|
|
IMFTransform* tmp = nullptr;
|
2018-12-19 16:12:57 +00:00
|
|
|
std::memcpy(&response, &request, sizeof(response));
|
|
|
|
response.unknown1 = 0x0;
|
|
|
|
|
2019-01-07 23:57:20 +00:00
|
|
|
if (!MFDecoderInit(&tmp)) {
|
2018-12-19 16:12:57 +00:00
|
|
|
LOG_CRITICAL(Audio_DSP, "Can't init decoder");
|
|
|
|
return response;
|
|
|
|
}
|
2019-01-07 23:57:20 +00:00
|
|
|
transform.reset(tmp);
|
2018-12-19 16:12:57 +00:00
|
|
|
|
|
|
|
HRESULT hr = transform->GetStreamIDs(1, &in_stream_id, 1, &out_stream_id);
|
|
|
|
if (hr == E_NOTIMPL) {
|
|
|
|
// if not implemented, it means this MFT does not assign stream ID for you
|
|
|
|
in_stream_id = 0;
|
|
|
|
out_stream_id = 0;
|
|
|
|
} else if (FAILED(hr)) {
|
|
|
|
ReportError("Decoder failed to initialize the stream ID", hr);
|
|
|
|
return response;
|
|
|
|
}
|
|
|
|
|
|
|
|
initalized = true;
|
|
|
|
return response;
|
|
|
|
}
|
|
|
|
|
|
|
|
void WMFDecoder::Impl::Clear() {
|
|
|
|
if (initalized) {
|
2019-01-07 23:57:20 +00:00
|
|
|
MFFlush(transform.get());
|
|
|
|
MFDeInit(transform.get());
|
2018-12-19 16:12:57 +00:00
|
|
|
}
|
|
|
|
initalized = false;
|
|
|
|
selected = false;
|
|
|
|
}
|
|
|
|
|
2019-01-29 05:23:57 +00:00
|
|
|
MFOutputState WMFDecoder::Impl::DecodingLoop(ADTSData adts_header,
|
2018-12-19 16:12:57 +00:00
|
|
|
std::array<std::vector<u8>, 2>& out_streams) {
|
2019-01-06 05:28:56 +00:00
|
|
|
MFOutputState output_status = OK;
|
2019-01-06 04:53:24 +00:00
|
|
|
char* output_buffer = nullptr;
|
2018-12-19 16:12:57 +00:00
|
|
|
DWORD output_len = 0;
|
2019-01-14 21:27:43 +00:00
|
|
|
DWORD tmp = 0;
|
|
|
|
// IMFSample* output_tmp = nullptr;
|
|
|
|
IMFMediaBuffer* mdbuf = nullptr;
|
|
|
|
unique_mfptr<IMFSample> output;
|
2018-12-19 16:12:57 +00:00
|
|
|
|
|
|
|
while (true) {
|
2019-01-14 21:27:43 +00:00
|
|
|
auto [output_status, output] = ReceiveSample(transform.get(), out_stream_id);
|
2018-12-19 16:12:57 +00:00
|
|
|
|
|
|
|
// 0 -> okay; 3 -> okay but more data available (buffer too small)
|
2019-01-06 05:28:56 +00:00
|
|
|
if (output_status == OK || output_status == HAVE_MORE_DATA) {
|
2019-01-14 21:27:43 +00:00
|
|
|
CopySampleToBuffer(output.get(), (void**)&output_buffer, &output_len);
|
2018-12-19 16:12:57 +00:00
|
|
|
|
|
|
|
// the following was taken from ffmpeg version of the decoder
|
|
|
|
f32 val_f32;
|
|
|
|
for (size_t i = 0; i < output_len;) {
|
|
|
|
for (std::size_t channel = 0; channel < adts_header.channels; channel++) {
|
|
|
|
std::memcpy(&val_f32, output_buffer + i, sizeof(val_f32));
|
|
|
|
s16 val = static_cast<s16>(0x7FFF * val_f32);
|
|
|
|
out_streams[channel].push_back(val & 0xFF);
|
|
|
|
out_streams[channel].push_back(val >> 8);
|
|
|
|
i += sizeof(val_f32);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (output_buffer)
|
|
|
|
free(output_buffer);
|
|
|
|
}
|
|
|
|
|
|
|
|
// in case of "ok" only, just return quickly
|
2019-01-06 05:28:56 +00:00
|
|
|
if (output_status == OK)
|
2019-01-29 05:23:57 +00:00
|
|
|
return OK;
|
2018-12-19 16:12:57 +00:00
|
|
|
|
|
|
|
// for status = 2, reset MF
|
2019-01-06 05:28:56 +00:00
|
|
|
if (output_status == NEED_RECONFIG) {
|
2018-12-19 16:12:57 +00:00
|
|
|
Clear();
|
2019-01-29 05:23:57 +00:00
|
|
|
return FATAL_ERROR;
|
2018-12-19 16:12:57 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// for status = 3, try again with new buffer
|
2019-01-06 05:28:56 +00:00
|
|
|
if (output_status == HAVE_MORE_DATA)
|
2018-12-19 16:12:57 +00:00
|
|
|
continue;
|
|
|
|
|
2019-01-06 20:29:21 +00:00
|
|
|
if (output_status == NEED_MORE_INPUT) // according to MS document, this is not an error (?!)
|
2019-01-29 05:23:57 +00:00
|
|
|
return NEED_MORE_INPUT;
|
2019-01-06 20:29:21 +00:00
|
|
|
|
2019-01-29 05:23:57 +00:00
|
|
|
return FATAL_ERROR; // return on other status
|
2018-12-19 16:12:57 +00:00
|
|
|
}
|
|
|
|
|
2019-01-29 05:23:57 +00:00
|
|
|
return FATAL_ERROR;
|
2018-12-19 16:12:57 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
std::optional<BinaryResponse> WMFDecoder::Impl::Decode(const BinaryRequest& request) {
|
|
|
|
BinaryResponse response;
|
|
|
|
response.codec = request.codec;
|
|
|
|
response.cmd = request.cmd;
|
|
|
|
response.size = request.size;
|
|
|
|
response.num_channels = 2;
|
|
|
|
response.num_samples = 1024;
|
|
|
|
|
|
|
|
if (!initalized) {
|
|
|
|
LOG_DEBUG(Audio_DSP, "Decoder not initalized");
|
|
|
|
// This is a hack to continue games that are not compiled with the aac codec
|
|
|
|
return response;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (request.src_addr < Memory::FCRAM_PADDR ||
|
|
|
|
request.src_addr + request.size > Memory::FCRAM_PADDR + Memory::FCRAM_SIZE) {
|
|
|
|
LOG_ERROR(Audio_DSP, "Got out of bounds src_addr {:08x}", request.src_addr);
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
u8* data = memory.GetFCRAMPointer(request.src_addr - Memory::FCRAM_PADDR);
|
|
|
|
|
|
|
|
std::array<std::vector<u8>, 2> out_streams;
|
2019-01-14 21:27:43 +00:00
|
|
|
unique_mfptr<IMFSample> sample;
|
2018-12-19 16:12:57 +00:00
|
|
|
ADTSData adts_header;
|
|
|
|
char* aac_tag = (char*)calloc(1, 14);
|
|
|
|
int input_status = 0;
|
|
|
|
|
2019-01-06 04:53:24 +00:00
|
|
|
if (DetectMediaType((char*)data, request.size, &adts_header, &aac_tag) != 0) {
|
2018-12-19 16:12:57 +00:00
|
|
|
LOG_ERROR(Audio_DSP, "Unable to deduce decoding parameters from ADTS stream");
|
|
|
|
return response;
|
|
|
|
}
|
|
|
|
|
2019-01-06 04:05:12 +00:00
|
|
|
response.num_channels = adts_header.channels;
|
|
|
|
|
2018-12-19 16:12:57 +00:00
|
|
|
if (!selected) {
|
|
|
|
LOG_DEBUG(Audio_DSP, "New ADTS stream: channels = {}, sample rate = {}",
|
|
|
|
adts_header.channels, adts_header.samplerate);
|
2019-01-07 23:57:20 +00:00
|
|
|
SelectInputMediaType(transform.get(), in_stream_id, adts_header, (UINT8*)aac_tag, 14);
|
|
|
|
SelectOutputMediaType(transform.get(), out_stream_id);
|
|
|
|
SendSample(transform.get(), in_stream_id, nullptr);
|
2018-12-19 16:12:57 +00:00
|
|
|
// cache the result from detect_mediatype and call select_*_mediatype only once
|
|
|
|
// This could increase performance very slightly
|
|
|
|
transform->ProcessMessage(MFT_MESSAGE_NOTIFY_BEGIN_STREAMING, 0);
|
|
|
|
selected = true;
|
|
|
|
}
|
|
|
|
|
2019-01-29 05:23:57 +00:00
|
|
|
sample = CreateSample((void*)data, request.size, 1, 0);
|
2018-12-19 16:12:57 +00:00
|
|
|
sample->SetUINT32(MFSampleExtension_CleanPoint, 1);
|
|
|
|
|
|
|
|
while (true) {
|
2019-01-14 21:27:43 +00:00
|
|
|
input_status = SendSample(transform.get(), in_stream_id, sample.get());
|
2018-12-19 16:12:57 +00:00
|
|
|
|
2019-01-29 05:23:57 +00:00
|
|
|
if (DecodingLoop(adts_header, out_streams) == FATAL_ERROR) {
|
2019-01-06 04:53:24 +00:00
|
|
|
// if the decode issues are caused by MFT not accepting new samples, try again
|
2018-12-19 16:12:57 +00:00
|
|
|
// NOTICE: you are required to check the output even if you already knew/guessed
|
|
|
|
// MFT didn't accept the input sample
|
|
|
|
if (input_status == 1) {
|
|
|
|
// try again
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2019-01-06 20:29:21 +00:00
|
|
|
LOG_ERROR(Audio_DSP, "Errors occurred when receiving output");
|
2018-12-19 16:12:57 +00:00
|
|
|
return response;
|
|
|
|
}
|
|
|
|
|
|
|
|
break; // jump out of the loop if at least we don't have obvious issues
|
|
|
|
}
|
|
|
|
|
|
|
|
if (out_streams[0].size() != 0) {
|
|
|
|
if (request.dst_addr_ch0 < Memory::FCRAM_PADDR ||
|
|
|
|
request.dst_addr_ch0 + out_streams[0].size() >
|
|
|
|
Memory::FCRAM_PADDR + Memory::FCRAM_SIZE) {
|
|
|
|
LOG_ERROR(Audio_DSP, "Got out of bounds dst_addr_ch0 {:08x}", request.dst_addr_ch0);
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
std::memcpy(memory.GetFCRAMPointer(request.dst_addr_ch0 - Memory::FCRAM_PADDR),
|
|
|
|
out_streams[0].data(), out_streams[0].size());
|
|
|
|
}
|
|
|
|
|
|
|
|
if (out_streams[1].size() != 0) {
|
|
|
|
if (request.dst_addr_ch1 < Memory::FCRAM_PADDR ||
|
|
|
|
request.dst_addr_ch1 + out_streams[1].size() >
|
|
|
|
Memory::FCRAM_PADDR + Memory::FCRAM_SIZE) {
|
|
|
|
LOG_ERROR(Audio_DSP, "Got out of bounds dst_addr_ch1 {:08x}", request.dst_addr_ch1);
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
std::memcpy(memory.GetFCRAMPointer(request.dst_addr_ch1 - Memory::FCRAM_PADDR),
|
|
|
|
out_streams[1].data(), out_streams[1].size());
|
|
|
|
}
|
|
|
|
|
|
|
|
return response;
|
|
|
|
}
|
|
|
|
|
|
|
|
WMFDecoder::WMFDecoder(Memory::MemorySystem& memory) : impl(std::make_unique<Impl>(memory)) {}
|
|
|
|
|
|
|
|
WMFDecoder::~WMFDecoder() = default;
|
|
|
|
|
|
|
|
std::optional<BinaryResponse> WMFDecoder::ProcessRequest(const BinaryRequest& request) {
|
|
|
|
return impl->ProcessRequest(request);
|
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace AudioCore::HLE
|