diff --git a/.gitmodules b/.gitmodules index bd40f6179..5757ad53b 100644 --- a/.gitmodules +++ b/.gitmodules @@ -19,3 +19,6 @@ [submodule "build_tools"] path = build_tools url = https://github.com/xenia-project/build-tools.git +[submodule "third_party/libav"] + path = third_party/libav + url = https://github.com/xenia-project/libav.git diff --git a/premake5.lua b/premake5.lua index 75999667d..47ea4af6c 100644 --- a/premake5.lua +++ b/premake5.lua @@ -8,7 +8,6 @@ includedirs({ ".", "src", "third_party", - "build_tools/third_party/gflags/src", }) defines({ @@ -146,6 +145,15 @@ solution("xenia") end configurations({"Checked", "Debug", "Release"}) + -- Include third party files first so they don't have to deal with gflags. + include("third_party/capstone.lua") + include("third_party/elemental-forms") + include("third_party/glew.lua") + include("third_party/imgui.lua") + include("third_party/libav.lua") + include("third_party/xxhash.lua") + include("build_tools/third_party/gflags.lua") + include("src/xenia") include("src/xenia/app") include("src/xenia/apu") @@ -169,10 +177,3 @@ solution("xenia") include("src/xenia/hid/winkey") include("src/xenia/hid/xinput") end - - include("third_party/capstone.lua") - include("third_party/elemental-forms") - include("third_party/glew.lua") - include("third_party/imgui.lua") - include("third_party/xxhash.lua") - include("build_tools/third_party/gflags.lua") diff --git a/src/xenia/app/premake5.lua b/src/xenia/app/premake5.lua index 1de6feb17..a7746755e 100644 --- a/src/xenia/app/premake5.lua +++ b/src/xenia/app/premake5.lua @@ -32,6 +32,7 @@ project("xenia-app") }) includedirs({ project_root.."/third_party/elemental-forms/src", + project_root.."/build_tools/third_party/gflags/src", }) local_platform_files() files({ @@ -52,18 +53,6 @@ project("xenia-app") "xenia-hid-winkey", "xenia-hid-xinput", }) - filter("configurations:Checked") - local libav_root = "../third_party/libav-xma-bin/lib/Debug" - linkoptions({ - libav_root.."/libavcodec.a", - libav_root.."/libavutil.a", - }) - filter("configurations:Debug or Release") - local libav_root = "../third_party/libav-xma-bin/lib/Release" - linkoptions({ - libav_root.."/libavcodec.a", - libav_root.."/libavutil.a", - }) filter("platforms:Windows") -- Only create the .user file if it doesn't already exist. diff --git a/src/xenia/apu/debug_visualizers.natvis b/src/xenia/apu/debug_visualizers.natvis new file mode 100644 index 000000000..095fce6ad --- /dev/null +++ b/src/xenia/apu/debug_visualizers.natvis @@ -0,0 +1,9 @@ + + + + + + id={id_}, allocated={is_allocated_}, enabled={is_enabled_} + + + diff --git a/src/xenia/apu/nop/premake5.lua b/src/xenia/apu/nop/premake5.lua index 068234586..276404fb1 100644 --- a/src/xenia/apu/nop/premake5.lua +++ b/src/xenia/apu/nop/premake5.lua @@ -13,5 +13,6 @@ project("xenia-apu-nop") defines({ }) includedirs({ + project_root.."/build_tools/third_party/gflags/src", }) local_platform_files() diff --git a/src/xenia/apu/premake5.lua b/src/xenia/apu/premake5.lua index 68ed44140..647bf02d6 100644 --- a/src/xenia/apu/premake5.lua +++ b/src/xenia/apu/premake5.lua @@ -7,14 +7,15 @@ project("xenia-apu") kind("StaticLib") language("C++") links({ - "libavcodec.a", - "libavutil.a", + "libavcodec", + "libavutil", "xenia-base", }) defines({ }) includedirs({ - project_root.."/third_party/libav-xma-bin/include/", + project_root.."/third_party/libav/include/", + project_root.."/build_tools/third_party/gflags/src", }) local_platform_files() diff --git a/src/xenia/apu/xaudio2/premake5.lua b/src/xenia/apu/xaudio2/premake5.lua index 7054e7419..466268609 100644 --- a/src/xenia/apu/xaudio2/premake5.lua +++ b/src/xenia/apu/xaudio2/premake5.lua @@ -12,5 +12,6 @@ project("xenia-apu-xaudio2") defines({ }) includedirs({ + project_root.."/build_tools/third_party/gflags/src", }) local_platform_files() diff --git a/src/xenia/apu/xma_context.cc b/src/xenia/apu/xma_context.cc index 7fbe08f17..333366220 100644 --- a/src/xenia/apu/xma_context.cc +++ b/src/xenia/apu/xma_context.cc @@ -13,13 +13,18 @@ #include #include "xenia/apu/xma_decoder.h" +#include "xenia/apu/xma_helpers.h" +#include "xenia/base/bit_stream.h" #include "xenia/base/logging.h" #include "xenia/base/ring_buffer.h" #include "xenia/profiling.h" extern "C" { #include "libavcodec/avcodec.h" +#include "libavcodec/xma2dec.h" #include "libavutil/channel_layout.h" + +extern AVCodec ff_xma2_decoder; } // extern "C" // Credits for most of this code goes to: @@ -50,14 +55,8 @@ int XmaContext::Setup(uint32_t id, Memory* memory, uint32_t guest_ptr) { memory_ = memory; guest_ptr_ = guest_ptr; - static bool avcodec_initialized = false; - if (!avcodec_initialized) { - avcodec_register_all(); - avcodec_initialized = true; - } - // Allocate important stuff. - codec_ = avcodec_find_decoder(AV_CODEC_ID_WMAPRO); + codec_ = &ff_xma2_decoder; if (!codec_) { return 1; } @@ -89,9 +88,11 @@ int XmaContext::Setup(uint32_t id, Memory* memory, uint32_t guest_ptr) { context_->extradata_size = sizeof(extra_data_); context_->extradata = reinterpret_cast(&extra_data_); + partial_frame_buffer_.resize(2048); + // Current frame stuff whatever // samples per frame * 2 max channels * output bytes - current_frame_ = new uint8_t[kSamplesPerFrame * 2 * 2]; + current_frame_ = new uint8_t[kSamplesPerFrame * kBytesPerSample * 2]; current_frame_pos_ = 0; frame_samples_size_ = 0; @@ -100,11 +101,11 @@ int XmaContext::Setup(uint32_t id, Memory* memory, uint32_t guest_ptr) { } void XmaContext::Work() { + std::lock_guard lock(lock_); if (!is_allocated() || !is_enabled()) { return; } - std::lock_guard lock(lock_); set_is_enabled(false); auto context_ptr = memory()->TranslateVirtual(guest_ptr()); @@ -119,11 +120,11 @@ void XmaContext::Enable() { auto context_ptr = memory()->TranslateVirtual(guest_ptr()); XMA_CONTEXT_DATA data(context_ptr); - XELOGAPU( - "XmaContext: kicking context %d (%d/%d bytes)", id(), - (data.input_buffer_read_offset & ~0x7FF) / 8, - (data.input_buffer_0_packet_count + data.input_buffer_1_packet_count) * - kBytesPerPacket); + XELOGAPU("XmaContext: kicking context %d (buffer %d %d/%d bits)", id(), + data.current_buffer, data.input_buffer_read_offset, + (data.current_buffer == 0 ? data.input_buffer_0_packet_count + : data.input_buffer_1_packet_count) * + kBytesPerPacket * 8); data.Store(context_ptr); @@ -145,8 +146,6 @@ void XmaContext::Clear() { std::lock_guard lock(lock_); XELOGAPU("XmaContext: reset context %d", id()); - DiscardPacket(); - auto context_ptr = memory()->TranslateVirtual(guest_ptr()); XMA_CONTEXT_DATA data(context_ptr); @@ -174,8 +173,6 @@ void XmaContext::Release() { set_is_allocated(false); auto context_ptr = memory()->TranslateVirtual(guest_ptr()); std::memset(context_ptr, 0, sizeof(XMA_CONTEXT_DATA)); // Zero it. - - DiscardPacket(); } int XmaContext::GetSampleRate(int id) { @@ -193,6 +190,83 @@ int XmaContext::GetSampleRate(int id) { return 0; } +size_t XmaContext::SavePartial(uint8_t* packet, uint32_t frame_offset_bits, + size_t frame_size_bits, bool append) { + uint8_t* buff = partial_frame_buffer_.data(); + + BitStream stream(packet, 2048 * 8); + stream.SetOffset(frame_offset_bits); + + if (!append) { + // Reset the buffer. + // TODO: Probably not necessary. + std::memset(buff, 0, partial_frame_buffer_.size()); + + size_t copy_bits = (2048 * 8) - frame_offset_bits; + size_t copy_offset = stream.Copy(buff, copy_bits); + partial_frame_offset_bits_ = copy_bits; + partial_frame_start_offset_bits_ = copy_offset; + + return copy_bits; + } else { + size_t copy_bits = frame_size_bits - partial_frame_offset_bits_; + size_t copy_offset = stream.Copy( + buff + + ((partial_frame_offset_bits_ + partial_frame_start_offset_bits_) / + 8), + copy_bits); + + partial_frame_offset_bits_ += copy_bits; + + return copy_bits; + } +} + +bool XmaContext::ValidFrameOffset(uint8_t* block, size_t size_bytes, + size_t frame_offset_bits) { + uint32_t packet_num = + GetFramePacketNumber(block, size_bytes, frame_offset_bits); + uint8_t* packet = block + (packet_num * kBytesPerPacket); + size_t relative_offset_bits = frame_offset_bits % (kBytesPerPacket * 8); + + uint32_t first_frame_offset = xma::GetPacketFrameOffset(packet); + if (first_frame_offset == -1) { + // Packet only contains a partial frame, so no frames can start here. + return false; + } + + BitStream stream(packet, kBytesPerPacket * 8); + stream.SetOffset(first_frame_offset); + while (true) { + if (stream.offset_bits() == relative_offset_bits) { + return true; + } + + if (stream.BitsRemaining() < 15) { + // Not enough room for another frame header. + return false; + } + + uint64_t size = stream.Read(15); + if ((size - 15) > stream.BitsRemaining()) { + // Last frame. + return false; + } else if (size == 0x7FFF) { + // Invalid frame (and last of this packet) + return false; + } + + stream.Advance(size - 16); + + // Read the trailing bit to see if frames follow + if (stream.Read(1) == 0) { + break; + } + } + + return false; +} + void XmaContext::DecodePackets(XMA_CONTEXT_DATA* data) { SCOPE_profile_cpu_f("apu"); @@ -206,27 +280,56 @@ void XmaContext::DecodePackets(XMA_CONTEXT_DATA* data) { // 32bit header (big endian) // Frames are the smallest thing the SPUs can decode. - // They usually can span packets (libav handles this) + // They can and usually will span packets. // Sample rates (data.sample_rate): - // 0 - 24 kHz ? + // 0 - 24 kHz // 1 - 32 kHz - // 2 - 44.1 kHz ? - // 3 - 48 kHz ? + // 2 - 44.1 kHz + // 3 - 48 kHz // SPUs also support stereo decoding. (data.is_stereo) - // Quick die if there's no data. - if (!data->input_buffer_0_valid && !data->input_buffer_1_valid) { - return; - } - // Check the output buffer - we cannot decode anything else if it's // unavailable. if (!data->output_buffer_valid) { return; } + assert_zero(data->unk_dword_9); + + // XAudio Loops + // loop_count: + // - XAUDIO2_MAX_LOOP_COUNT = 254 + // - XAUDIO2_LOOP_INFINITE = 255 + // loop_start/loop_end are bit offsets to a specific frame + + // Translate pointers for future use. + // Sometimes the game will use rolling input buffers. If they do, we cannot + // assume they form a complete block! In addition, the buffers DO NOT have + // to be contiguous! + uint8_t* in0 = data->input_buffer_0_valid + ? memory()->TranslatePhysical(data->input_buffer_0_ptr) + : nullptr; + uint8_t* in1 = data->input_buffer_1_valid + ? memory()->TranslatePhysical(data->input_buffer_1_ptr) + : nullptr; + uint8_t* current_input_buffer = data->current_buffer ? in1 : in0; + + XELOGAPU("Processing context %d (offset %d, buffer %d, ptr %.8X)", id(), + data->input_buffer_read_offset, data->current_buffer, + current_input_buffer); + + size_t input_buffer_0_size = + data->input_buffer_0_packet_count * kBytesPerPacket; + size_t input_buffer_1_size = + data->input_buffer_1_packet_count * kBytesPerPacket; + size_t input_total_size = input_buffer_0_size + input_buffer_1_size; + + size_t current_input_size = + data->current_buffer ? input_buffer_1_size : input_buffer_0_size; + size_t current_input_packet_count = current_input_size / kBytesPerPacket; + // Output buffers are in raw PCM samples, 256 bytes per block. // Output buffer is a ring buffer. We need to write from the write offset // to the read offset. @@ -243,162 +346,310 @@ void XmaContext::DecodePackets(XMA_CONTEXT_DATA* data) { output_rb.set_write_offset(output_write_offset); size_t output_remaining_bytes = output_rb.write_count(); + bool output_written = false; // Decode until we can't write any more data. while (output_remaining_bytes > 0) { - // This'll copy audio samples into the output buffer. - // The samples need to be 2 bytes long! - // Copies one frame at a time, so keep calling this until size == 0. - int written_bytes = 0; - int decode_attempts_remaining = 3; + int num_channels = data->is_stereo ? 2 : 1; - uint8_t work_buffer[kOutputMaxSizeBytes]; - while (decode_attempts_remaining) { - size_t read_bytes = 0; - written_bytes = - DecodePacket(work_buffer, 0, output_remaining_bytes, &read_bytes); - if (written_bytes >= 0) { - // assert_true((written_bytes % 256) == 0); - auto written_bytes_rb = output_rb.Write(work_buffer, written_bytes); - assert_true(written_bytes == written_bytes_rb); + // Check if we have part of a frame waiting (and the game hasn't jumped + // around) + if (current_frame_pos_ && + last_input_read_pos_ == data->input_buffer_read_offset) { + size_t to_write = std::min( + output_remaining_bytes, + ((size_t)kBytesPerFrame * num_channels - current_frame_pos_)); + output_rb.Write(current_frame_ + current_frame_pos_, to_write); + output_written = true; + XELOGAPU("XmaContext %d: wrote out %d bytes of left-over samples", id(), + to_write); - // Ok. - break; - } else if (read_bytes % 2048 == 0) { - // Sometimes the decoder will fail on a packet. I think it's - // looking for cross-packet frames and failing. If you run it again - // on the same packet it'll work though. - --decode_attempts_remaining; - } else { - // Failed in the middle of a packet, do not retry! - decode_attempts_remaining = 0; - break; + current_frame_pos_ += to_write; + if (current_frame_pos_ >= kBytesPerFrame * num_channels) { + current_frame_pos_ = 0; } - } - if (!decode_attempts_remaining) { - XELOGAPU("XmaContext: libav failed to decode packet (returned %.8X)", - -written_bytes); - - // Failed out. - if (data->input_buffer_0_valid || data->input_buffer_1_valid) { - // There's new data available - maybe we'll be ok if we decode it? - written_bytes = 0; - DiscardPacket(); - } else { - // No data and hosed - bail. - break; - } - } - - data->output_buffer_write_offset = output_rb.write_offset() / 256; - output_remaining_bytes -= written_bytes; - - // If we need more data and the input buffers have it, grab it. - if (written_bytes) { - // Haven't finished with current packet. + data->output_buffer_write_offset = output_rb.write_offset() / 256; + output_remaining_bytes -= to_write; continue; - } else if (data->input_buffer_0_valid || data->input_buffer_1_valid) { - // Done with previous packet, so grab a new one. - int ret = StartPacket(data); - if (ret <= 0) { - // No more data (but may have prepared a packet) + } + + if (!data->input_buffer_0_valid && !data->input_buffer_1_valid) { + // Out of data. + break; + } + + if (data->input_buffer_read_offset == 0) { + // Invalid offset. Go ahead and set it. + uint32_t offset = xma::GetPacketFrameOffset(current_input_buffer); + if (offset == -1) { + // No more frames. + if (data->current_buffer == 0) { + data->input_buffer_0_valid = 0; + data->input_buffer_read_offset = 0; + data->current_buffer++; + } else if (data->current_buffer == 1) { + data->input_buffer_1_valid = 0; + data->input_buffer_read_offset = 0; + data->current_buffer--; + } + + // Die if we have no partial saved. + if (!partial_frame_saved_) { + return; + } + } else { + data->input_buffer_read_offset = offset; + } + } + + if (!ValidFrameOffset(current_input_buffer, current_input_size, + data->input_buffer_read_offset)) { + XELOGAPU("XmaContext %d: Invalid read offset %d!", id(), + data->input_buffer_read_offset); + if (data->current_buffer == 0) { + data->current_buffer = 1; + data->input_buffer_0_valid = 0; + } else if (data->current_buffer == 1) { + data->current_buffer = 0; + data->input_buffer_1_valid = 0; + } + + data->input_buffer_read_offset = 0; + return; + } + + // Check if we need to save a partial frame. + if (data->input_buffer_read_offset != 0 && !partial_frame_saved_ && + GetFramePacketNumber(current_input_buffer, current_input_size, + data->input_buffer_read_offset) == + current_input_packet_count - 1) { + BitStream stream(current_input_buffer, current_input_size * 8); + stream.SetOffset(data->input_buffer_read_offset); + + if (stream.BitsRemaining() >= 15) { + uint64_t frame_size = stream.Read(15); + if (data->input_buffer_read_offset + frame_size >= + current_input_size * 8 && + frame_size != 0x7FFF) { + uint32_t rel_offset = data->input_buffer_read_offset % (2048 * 8); + + // Frame is cut off! Save and exit. + partial_frame_saved_ = true; + partial_frame_size_known_ = true; + partial_frame_total_size_bits_ = frame_size; + SavePartial( + current_input_buffer + (current_input_packet_count - 1) * 2048, + rel_offset, frame_size, false); + } + } else { + // Header cut in half :/ + uint32_t rel_offset = data->input_buffer_read_offset % (2048 * 8); + + partial_frame_saved_ = true; + partial_frame_size_known_ = false; + SavePartial( + current_input_buffer + (current_input_packet_count - 1) * 2048, + rel_offset, 0, false); + } + + if (partial_frame_saved_) { + XELOGAPU("XmaContext %d: saved a partial frame", id()); + + if (data->current_buffer == 0) { + data->input_buffer_0_valid = 0; + data->input_buffer_read_offset = 0; + data->current_buffer++; + } else if (data->current_buffer == 1) { + data->input_buffer_1_valid = 0; + data->input_buffer_read_offset = 0; + data->current_buffer--; + } + + return; + } + } + + if (partial_frame_saved_ && !partial_frame_size_known_) { + // Append the rest of the header. + size_t offset = SavePartial(current_input_buffer, 32, 15, true); + + // Read the frame size. + BitStream stream(partial_frame_buffer_.data(), + 15 + partial_frame_start_offset_bits_); + stream.SetOffset(partial_frame_start_offset_bits_); + + uint64_t size = stream.Read(15); + partial_frame_size_known_ = true; + partial_frame_total_size_bits_ = size; + + // Now append the rest of the frame. + SavePartial(current_input_buffer, 32 + (uint32_t)offset, size, true); + } else if (partial_frame_saved_) { + // Append the rest of the frame. + SavePartial(current_input_buffer, 32, partial_frame_total_size_bits_, + true); + } + + // Prepare the decoder. Reinitialize if any parameters have changed. + PrepareDecoder(current_input_buffer, current_input_size, data->sample_rate, + num_channels); + + bool partial = false; + size_t bit_offset = data->input_buffer_read_offset; + if (partial_frame_saved_) { + XELOGAPU("XmaContext %d: processing saved partial frame", id()); + packet_->data = partial_frame_buffer_.data(); + packet_->size = (int)partial_frame_buffer_.size(); + + bit_offset = partial_frame_start_offset_bits_; + partial = true; + partial_frame_saved_ = false; + } else { + packet_->data = current_input_buffer; + packet_->size = (int)current_input_size; + } + + int invalid_frame = 0; // invalid frame? + int got_frame = 0; // successfully decoded a frame? + int frame_size = 0; + int len = + xma2_decode_frame(context_, packet_, decoded_frame_, &got_frame, + &invalid_frame, &frame_size, !partial, bit_offset); + if (!partial && len == 0) { + // Got the last frame of a packet. Advance the read offset to the next + // packet. + uint32_t packet_number = + GetFramePacketNumber(current_input_buffer, current_input_size, + data->input_buffer_read_offset); + if (packet_number == current_input_packet_count - 1) { + // Last packet. + if (data->current_buffer == 0) { + data->input_buffer_0_valid = 0; + data->input_buffer_read_offset = 0; + data->current_buffer = 1; + } else if (data->current_buffer == 1) { + data->input_buffer_1_valid = 0; + data->input_buffer_read_offset = 0; + data->current_buffer = 0; + } + } else { + // Advance the read offset. + packet_number++; + uint8_t* packet = current_input_buffer + (packet_number * 2048); + uint32_t first_frame_offset = xma::GetPacketFrameOffset(packet); + if (first_frame_offset == -1) { + // Invalid packet (only contained a frame partial). Out of input. + if (data->current_buffer == 0) { + data->input_buffer_0_valid = 0; + data->current_buffer = 1; + } else if (data->current_buffer == 1) { + data->input_buffer_1_valid = 0; + data->current_buffer = 0; + } + + data->input_buffer_read_offset = 0; + } else { + data->input_buffer_read_offset = + packet_number * 2048 * 8 + first_frame_offset; + } + } + } + + if (got_frame) { + // Valid frame. + // Check and see if we need to loop back to any spot. + if (data->loop_count > 0 && + data->input_buffer_read_offset == data->loop_end) { + // Loop back to the beginning. + data->input_buffer_read_offset = data->loop_start; + if (data->loop_count < 255) { + data->loop_count--; + } + } else if (!partial && len > 0) { + data->input_buffer_read_offset += len; + } + } else if (len < 0) { + // Did not get frame + XELOGAPU("libav failed to decode a frame!"); + if (frame_size && frame_size != 0x7FFF) { + data->input_buffer_read_offset += frame_size; + } else { data->input_buffer_0_valid = 0; data->input_buffer_1_valid = 0; } - } else { - // Decoder is out of data and there's no more to give. - break; + return; + } + + last_input_read_pos_ = data->input_buffer_read_offset; + + if (got_frame) { + // Successfully decoded a frame. + // Copy to the output buffer. + size_t written_bytes = 0; + + // Validity checks. + assert(decoded_frame_->nb_samples <= kSamplesPerFrame); + assert(context_->sample_fmt == AV_SAMPLE_FMT_FLTP); + + // Check the returned buffer size. + assert(av_samples_get_buffer_size(NULL, context_->channels, + decoded_frame_->nb_samples, + context_->sample_fmt, 1) == + context_->channels * decoded_frame_->nb_samples * sizeof(float)); + + // Convert the frame. + ConvertFrame((const uint8_t**)decoded_frame_->data, context_->channels, + decoded_frame_->nb_samples, current_frame_); + current_frame_pos_ = 0; + + if (output_remaining_bytes < kBytesPerFrame * num_channels) { + // Output buffer isn't big enough to store the entire frame! Write out a + // part of it. + current_frame_pos_ = output_remaining_bytes; + output_rb.Write(current_frame_, output_remaining_bytes); + + written_bytes = output_remaining_bytes; + } else { + output_rb.Write(current_frame_, kBytesPerFrame * num_channels); + + written_bytes = kBytesPerFrame * num_channels; + } + + output_written = true; + output_remaining_bytes -= written_bytes; + data->output_buffer_write_offset = output_rb.write_offset() / 256; } } // The game will kick us again with a new output buffer later. - data->output_buffer_valid = 0; + // It's important that we only invalidate this if we actually wrote to it!! + if (output_written) { + data->output_buffer_valid = 0; + } } -int XmaContext::StartPacket(XMA_CONTEXT_DATA* data) { - // Translate pointers for future use. - uint8_t* in0 = data->input_buffer_0_valid - ? memory()->TranslatePhysical(data->input_buffer_0_ptr) - : nullptr; - uint8_t* in1 = data->input_buffer_1_valid - ? memory()->TranslatePhysical(data->input_buffer_1_ptr) - : nullptr; - - int sample_rate = GetSampleRate(data->sample_rate); - int channels = data->is_stereo ? 2 : 1; - - // See if we've finished with the input. - // Block count is in packets, so expand by packet size. - uint32_t input_size_0_bytes = data->input_buffer_0_valid - ? (data->input_buffer_0_packet_count) * 2048 - : 0; - uint32_t input_size_1_bytes = data->input_buffer_1_valid - ? (data->input_buffer_1_packet_count) * 2048 - : 0; - - // Total input size - uint32_t input_size_bytes = input_size_0_bytes + input_size_1_bytes; - - // Input read offset is in bits. Typically starts at 32 (4 bytes). - // "Sequence" offset - used internally for WMA Pro decoder. - // Just the read offset. - uint32_t seq_offset_bytes = (data->input_buffer_read_offset & ~0x7FF) / 8; - uint32_t input_remaining_bytes = input_size_bytes - seq_offset_bytes; - - if (seq_offset_bytes < input_size_bytes) { - // Setup input offset and input buffer. - uint32_t input_offset_bytes = seq_offset_bytes; - auto input_buffer = in0; - - if (seq_offset_bytes >= input_size_0_bytes && input_size_1_bytes) { - // Size overlap, select input buffer 1. - // TODO(DrChat): This needs testing. - input_offset_bytes -= input_size_0_bytes; - input_buffer = in1; - } - - // Still have data to read. - auto packet = input_buffer + input_offset_bytes; - assert_true(input_offset_bytes % 2048 == 0); - PreparePacket(packet, seq_offset_bytes, kBytesPerPacket, sample_rate, - channels); - data->input_buffer_read_offset += kBytesPerPacket * 8; - - input_remaining_bytes -= kBytesPerPacket; - if (input_remaining_bytes <= 0) { - // Used the last of the data but prepared a packet. - return 0; - } - } else { - // No more data available and no packet prepared. +uint32_t XmaContext::GetFramePacketNumber(uint8_t* block, size_t size, + size_t bit_offset) { + size *= 8; + if (bit_offset >= size) { + // Not good :( + assert_always(); return -1; } - return input_remaining_bytes; + size_t byte_offset = bit_offset >> 3; + size_t packet_number = byte_offset / kBytesPerPacket; + + return (uint32_t)packet_number; } -int XmaContext::PreparePacket(uint8_t* input, size_t seq_offset, size_t size, - int sample_rate, int channels) { - if (size != kBytesPerPacket) { - // Invalid packet size! - assert_always(); - return 1; - } - if (packet_->size > 0 || current_frame_pos_ != frame_samples_size_) { - // Haven't finished parsing another packet. - return 1; - } +int XmaContext::PrepareDecoder(uint8_t* block, size_t size, int sample_rate, + int channels) { + // Sanity check: Packet metadata is always 1 for XMA2/0 for XMA + assert_true((block[2] & 0x7) == 1 || (block[2] & 0x7) == 0); - std::memcpy(packet_data_, input, size); - - // Modify the packet header so it's WMAPro compatible. - auto int_packet_data = reinterpret_cast(packet_data_); - *int_packet_data = - (((seq_offset & 0x7800) | 0x400) >> 7) | (*int_packet_data & 0xFFFEFF08); - - packet_->data = packet_data_; - packet_->size = kBytesPerPacket; + sample_rate = GetSampleRate(sample_rate); // Re-initialize the context with new sample rate and channels. if (context_->sample_rate != sample_rate || context_->channels != channels) { @@ -417,110 +668,35 @@ int XmaContext::PreparePacket(uint8_t* input, size_t seq_offset, size_t size, } } + av_frame_unref(decoded_frame_); + return 0; } -void XmaContext::DiscardPacket() { - if (packet_->size > 0 || current_frame_pos_ != frame_samples_size_) { - packet_->data = 0; - packet_->size = 0; - current_frame_pos_ = frame_samples_size_; - } -} +bool XmaContext::ConvertFrame(const uint8_t** samples, int num_channels, + int num_samples, uint8_t* output_buffer) { + // Loop through every sample, convert and drop it into the output array. + // If more than one channel, we need to interleave the samples from each + // channel next to each other. + // TODO: This can definitely be optimized with AVX/SSE intrinsics! + uint32_t o = 0; + for (int i = 0; i < num_samples; i++) { + for (int j = 0; j < num_channels; j++) { + // Select the appropriate array based on the current channel. + auto sample_array = reinterpret_cast(samples[j]); -int XmaContext::DecodePacket(uint8_t* output, size_t output_offset, - size_t output_size, size_t* read_bytes) { - size_t to_copy = 0; - size_t original_offset = output_offset; - if (read_bytes) { - *read_bytes = 0; - } + // Raw sample should be within [-1, 1]. + // Clamp it, just in case. + float raw_sample = xe::saturate(sample_array[i]); - // We're holding onto an already-decoded frame. Copy it out. - if (current_frame_pos_ != frame_samples_size_) { - to_copy = std::min(output_size, frame_samples_size_ - current_frame_pos_); - memcpy(output + output_offset, current_frame_ + current_frame_pos_, - to_copy); - - current_frame_pos_ += to_copy; - output_size -= to_copy; - output_offset += to_copy; - } - - while (output_size > 0 && packet_->size > 0) { - int got_frame = 0; - - // Decode the current frame. - int len = - avcodec_decode_audio4(context_, decoded_frame_, &got_frame, packet_); - if (len < 0) { - // Error in codec (bad sample rate or something). - return len; - } - - if (read_bytes) { - *read_bytes += len; - } - - // Offset by decoded length. - packet_->size -= len; - packet_->data += len; - packet_->dts = packet_->pts = AV_NOPTS_VALUE; - - // Successfully decoded a frame. - if (got_frame) { - // Validity checks. - if (decoded_frame_->nb_samples > kSamplesPerFrame) { - return -2; - } else if (context_->sample_fmt != AV_SAMPLE_FMT_FLTP) { - return -3; - } - - // Check the returned buffer size. - if (av_samples_get_buffer_size(NULL, context_->channels, - decoded_frame_->nb_samples, - context_->sample_fmt, 1) != - context_->channels * decoded_frame_->nb_samples * sizeof(float)) { - return -4; - } - - // Loop through every sample, convert and drop it into the output array. - // If more than one channel, the game wants the samples from each channel - // interleaved next to each other. - uint32_t o = 0; - for (int i = 0; i < decoded_frame_->nb_samples; i++) { - for (int j = 0; j < context_->channels; j++) { - // Select the appropriate array based on the current channel. - auto sample_array = reinterpret_cast(decoded_frame_->data[j]); - - // Raw sample should be within [-1, 1]. - // Clamp it, just in case. - float raw_sample = xe::saturate(sample_array[i]); - - // Convert the sample and output it in big endian. - float scaled_sample = raw_sample * ((1 << 15) - 1); - int sample = static_cast(scaled_sample); - xe::store_and_swap(¤t_frame_[o++ * 2], - sample & 0xFFFF); - } - } - current_frame_pos_ = 0; - - // Total size of the frame's samples. - // Magic number 2 is sizeof an output sample. - frame_samples_size_ = context_->channels * decoded_frame_->nb_samples * 2; - - to_copy = std::min(output_size, (size_t)(frame_samples_size_)); - std::memcpy(output + output_offset, current_frame_, to_copy); - - current_frame_pos_ += to_copy; - output_size -= to_copy; - output_offset += to_copy; + // Convert the sample and output it in big endian. + float scaled_sample = raw_sample * ((1 << 15) - 1); + int sample = static_cast(scaled_sample); + xe::store_and_swap(&output_buffer[o++ * 2], sample & 0xFFFF); } } - // Return number of bytes written. - return static_cast(output_offset - original_offset); + return true; } } // namespace apu diff --git a/src/xenia/apu/xma_context.h b/src/xenia/apu/xma_context.h index 06e2299b8..96d059188 100644 --- a/src/xenia/apu/xma_context.h +++ b/src/xenia/apu/xma_context.h @@ -60,8 +60,8 @@ struct XMA_CONTEXT_DATA { uint32_t input_buffer_1_packet_count : 12; // XMASetInputBuffer1, number of // 2KB packets. Max 4095 packets. // These packets form a block. - uint32_t loop_subframe_end : 2; // +12bit, XMASetLoopData - uint32_t unk_dword_1_a : 3; // ? might be loop_subframe_skip + uint32_t loop_subframe_start : 2; // +12bit, XMASetLoopData + uint32_t loop_subframe_end : 3; // +14bit, XMASetLoopData uint32_t loop_subframe_skip : 3; // +17bit, XMASetLoopData might be // subframe_decode_count uint32_t subframe_decode_count : 4; // +20bit might be subframe_skip_count @@ -77,10 +77,12 @@ struct XMA_CONTEXT_DATA { // DWORD 3 uint32_t loop_start : 26; // XMASetLoopData LoopStartOffset + // frame offset in bits uint32_t unk_dword_3 : 6; // ? ParserErrorStatus/ParserErrorSet(?) // DWORD 4 uint32_t loop_end : 26; // XMASetLoopData LoopEndOffset + // frame offset in bits uint32_t packet_metadata : 5; // XMAGetPacketMetadata uint32_t current_buffer : 1; // ? @@ -91,7 +93,7 @@ struct XMA_CONTEXT_DATA { // DWORD 7 uint32_t output_buffer_ptr; // physical address // DWORD 8 - uint32_t overlap_add_ptr; // PtrOverlapAdd(?) + uint32_t work_buffer_ptr; // PtrOverlapAdd(?) // DWORD 9 // +0bit, XMAGetOutputBufferReadOffset AKA WriteBufferOffsetRead @@ -133,6 +135,7 @@ class XmaContext { static const uint32_t kBytesPerSample = 2; static const uint32_t kSamplesPerFrame = 512; static const uint32_t kSamplesPerSubframe = 128; + static const uint32_t kBytesPerFrame = kSamplesPerFrame * kBytesPerSample; static const uint32_t kBytesPerSubframe = kSamplesPerSubframe * kBytesPerSample; @@ -164,7 +167,18 @@ class XmaContext { private: static int GetSampleRate(int id); + size_t SavePartial(uint8_t* packet, uint32_t frame_offset_bits, + size_t frame_size_bits, bool append); + bool ValidFrameOffset(uint8_t* block, size_t size_bytes, + size_t frame_offset_bits); void DecodePackets(XMA_CONTEXT_DATA* data); + uint32_t GetFramePacketNumber(uint8_t* block, size_t size, size_t bit_offset); + int PrepareDecoder(uint8_t* block, size_t size, int sample_rate, + int channels); + + bool ConvertFrame(const uint8_t** samples, int num_channels, int num_samples, + uint8_t* output_buffer); + int StartPacket(XMA_CONTEXT_DATA* data); int PreparePacket(uint8_t* input, size_t seq_offset, size_t size, @@ -189,11 +203,19 @@ class XmaContext { AVPacket* packet_ = nullptr; WmaProExtraData extra_data_; + bool partial_frame_saved_ = false; + bool partial_frame_size_known_ = false; + size_t partial_frame_total_size_bits_ = 0; + size_t partial_frame_start_offset_bits_ = 0; + size_t partial_frame_offset_bits_ = 0; // blah internal don't use this + std::vector partial_frame_buffer_; + + // If we didn't finish writing a frame to the output buffer, this is the + // offset. size_t current_frame_pos_ = 0; + uint32_t last_input_read_pos_ = 0; // Last seen read buffer pos uint8_t* current_frame_ = nullptr; uint32_t frame_samples_size_ = 0; - - uint8_t packet_data_[kBytesPerPacket]; }; } // namespace apu diff --git a/src/xenia/apu/xma_decoder.cc b/src/xenia/apu/xma_decoder.cc index d8bd47ef0..a33968ef9 100644 --- a/src/xenia/apu/xma_decoder.cc +++ b/src/xenia/apu/xma_decoder.cc @@ -7,8 +7,11 @@ ****************************************************************************** */ -#include "xenia/apu/xma_context.h" #include "xenia/apu/xma_decoder.h" + +#include + +#include "xenia/apu/xma_context.h" #include "xenia/base/logging.h" #include "xenia/base/math.h" #include "xenia/base/ring_buffer.h" @@ -46,6 +49,8 @@ extern "C" { // do this, it's likely they are either passing the context to XAudio or // using the XMA* functions. +DEFINE_bool(libav_verbose, false, "Verbose libav output (debug and above)"); + namespace xe { namespace apu { @@ -55,7 +60,32 @@ XmaDecoder::XmaDecoder(cpu::Processor* processor) XmaDecoder::~XmaDecoder() = default; void av_log_callback(void* avcl, int level, const char* fmt, va_list va) { - xe::LogLineVarargs('A', fmt, va); + if (!FLAGS_libav_verbose && level > AV_LOG_WARNING) { + return; + } + + char level_char = '?'; + switch (level) { + case AV_LOG_ERROR: + level_char = '!'; + break; + case AV_LOG_WARNING: + level_char = 'w'; + break; + case AV_LOG_INFO: + level_char = 'i'; + break; + case AV_LOG_VERBOSE: + level_char = 'v'; + break; + case AV_LOG_DEBUG: + level_char = 'd'; + break; + } + + StringBuffer buff; + buff.AppendVarargs(fmt, va); + xe::LogLineFormat(level_char, "libav: %s", buff.GetString()); } X_STATUS XmaDecoder::Setup(kernel::KernelState* kernel_state) { @@ -104,6 +134,11 @@ void XmaDecoder::WorkerThreadMain() { for (uint32_t n = 0; n < kContextCount; n++) { XmaContext& context = contexts_[n]; context.Work(); + + // TODO: Need thread safety to do this. + // Probably not too important though. + // registers_.current_context = n; + // registers_.next_context = (n + 1) % kContextCount; } } } @@ -182,7 +217,6 @@ uint32_t XmaDecoder::ReadRegister(uint32_t addr) { // number registers_.current_context = registers_.next_context; registers_.next_context = (registers_.next_context + 1) % kContextCount; - value = registers_.current_context; } value = xe::byte_swap(value); diff --git a/src/xenia/apu/xma_helpers.h b/src/xenia/apu/xma_helpers.h new file mode 100644 index 000000000..c22c70c25 --- /dev/null +++ b/src/xenia/apu/xma_helpers.h @@ -0,0 +1,47 @@ +/** + ****************************************************************************** + * Xenia : Xbox 360 Emulator Research Project * + ****************************************************************************** + * Copyright 2015 Ben Vanik. All rights reserved. * + * Released under the BSD license - see LICENSE in the root for more details. * + ****************************************************************************** + */ + +// This file contains some functions used to help parse XMA data. + +#ifndef XENIA_APU_XMA_HELPERS_H_ +#define XENIA_APU_XMA_HELPERS_H_ + +#include + +namespace xe { +namespace apu { +namespace xma { + +// Get number of frames that /begin/ in this packet. +uint32_t GetPacketFrameCount(uint8_t* packet) { + return (uint8_t)(packet[0] >> 2); +} + +// Get the first frame offset in bits +uint32_t GetPacketFrameOffset(uint8_t* packet) { + uint32_t val = (uint16_t)(((packet[0] & 0x3) << 13) | (packet[1] << 5) | + (packet[2] >> 3)); + if (val == 0x7FFF) { + return -1; + } else { + return val + 32; + } +} + +uint32_t GetPacketMetadata(uint8_t* packet) { + return (uint8_t)(packet[2] & 0x7); +} + +uint32_t GetPacketSkipCount(uint8_t* packet) { return (uint8_t)(packet[3]); } + +} // namespace xma +} // namespace apu +} // namespace xe + +#endif // XENIA_APU_XMA_HELPERS_H_ diff --git a/src/xenia/base/bit_stream.cc b/src/xenia/base/bit_stream.cc new file mode 100644 index 000000000..2a073109e --- /dev/null +++ b/src/xenia/base/bit_stream.cc @@ -0,0 +1,143 @@ +/** + ****************************************************************************** + * Xenia : Xbox 360 Emulator Research Project * + ****************************************************************************** + * Copyright 2015 Ben Vanik. All rights reserved. * + * Released under the BSD license - see LICENSE in the root for more details. * + ****************************************************************************** + */ + +#include "xenia/base/bit_stream.h" + +#include + +#include "xenia/base/assert.h" +#include "xenia/base/byte_order.h" + +namespace xe { +BitStream::BitStream(uint8_t* buffer, size_t size_in_bits) + : buffer_(buffer), size_bits_(size_in_bits) {} + +BitStream::~BitStream() {} + +void BitStream::SetOffset(size_t offset_bits) { + assert_false(offset_bits > size_bits_); + offset_bits_ = std::min(offset_bits, size_bits_); +} + +size_t BitStream::BitsRemaining() { return size_bits_ - offset_bits_; } + +uint64_t BitStream::Peek(size_t num_bits) { + // FYI: The reason we can't copy more than 57 bits is: + // 57 = 7 * 8 + 1 - that can only span a maximum of 8 bytes. + // We can't read in 9 bytes (easily), so we limit it. + assert_false(num_bits > 57); + assert_false(offset_bits_ + num_bits > size_bits_); + + size_t offset_bytes = offset_bits_ >> 3; + size_t rel_offset_bits = offset_bits_ - (offset_bytes << 3); + + // offset --> + // ..[junk]..| target bits |....[junk]............. + uint64_t bits = *(uint64_t*)(buffer_ + offset_bytes); + + // We need the data in little endian. + // TODO: Have a flag specifying endianness of data? + bits = xe::byte_swap(bits); + + // Shift right + // .....[junk]........| target bits | + bits >>= 64 - (rel_offset_bits + num_bits); + + // AND with mask + // ...................| target bits | + bits &= (1 << num_bits) - 1; + + return bits; +} + +uint64_t BitStream::Read(size_t num_bits) { + uint64_t val = Peek(num_bits); + Advance(num_bits); + + return val; +} + +// TODO: This is totally not tested! +bool BitStream::Write(uint64_t val, size_t num_bits) { + assert_false(num_bits > 57); + assert_false(offset_bits_ + num_bits >= size_bits_); + + size_t offset_bytes = offset_bits_ >> 3; + size_t rel_offset_bits = offset_bits_ - (offset_bytes << 3); + + // Construct a mask + uint64_t mask = (1 << num_bits) - 1; + mask <<= 64 - (rel_offset_bits + num_bits); + mask = ~mask; + + // Shift the value left into position. + val <<= 64 - (rel_offset_bits + num_bits); + + // offset -----> + // ....[junk]...| target bits w/ junk |....[junk]...... + uint64_t bits = *(uint64_t*)(buffer_ + offset_bytes); + + // AND with mask + // ....[junk]...| target bits (0) |........[junk]...... + bits &= mask; + + // OR with val + // ....[junk]...| target bits (val) |......[junk]...... + bits |= val; + + // Store into the bitstream. + *(uint64_t*)(buffer_ + offset_bytes) = bits; + + // Advance the bitstream forward. + Advance(num_bits); + + return true; +} + +size_t BitStream::Copy(uint8_t* dest_buffer, size_t num_bits) { + size_t offset_bytes = offset_bits_ >> 3; + size_t rel_offset_bits = offset_bits_ - (offset_bytes << 3); + size_t bits_left = num_bits; + size_t out_offset_bytes = 0; + + // First: Copy the first few bits up to a byte boundary. + if (rel_offset_bits) { + uint64_t bits = Peek(8 - rel_offset_bits); + dest_buffer[out_offset_bytes] |= (uint8_t)bits; + + bits_left -= 8 - rel_offset_bits; + Advance(8 - rel_offset_bits); + out_offset_bytes++; + } + + // Second: Use memcpy for the bytes left. + if (bits_left >= 8) { + std::memcpy(dest_buffer + out_offset_bytes, + buffer_ + offset_bytes + out_offset_bytes, bits_left / 8); + out_offset_bytes += (bits_left / 8); + Advance((bits_left / 8) * 8); + bits_left -= (bits_left / 8) * 8; + } + + // Third: Copy the last few bits. + if (bits_left) { + uint64_t bits = Peek(bits_left); + bits <<= 8 - bits_left; + + dest_buffer[out_offset_bytes] |= (uint8_t)bits; + Advance(bits_left); + } + + // Return the bit offset to the copied bits. + return rel_offset_bits; +} + +void BitStream::Advance(size_t num_bits) { SetOffset(offset_bits_ + num_bits); } + +} // namespace xe \ No newline at end of file diff --git a/src/xenia/base/bit_stream.h b/src/xenia/base/bit_stream.h new file mode 100644 index 000000000..ad84379d4 --- /dev/null +++ b/src/xenia/base/bit_stream.h @@ -0,0 +1,44 @@ +/** + ****************************************************************************** + * Xenia : Xbox 360 Emulator Research Project * + ****************************************************************************** + * Copyright 2015 Ben Vanik. All rights reserved. * + * Released under the BSD license - see LICENSE in the root for more details. * + ****************************************************************************** + */ + +#ifndef XENIA_BASE_BIT_STREAM_H_ +#define XENIA_BASE_BIT_STREAM_H_ + +#include + +namespace xe { +class BitStream { + public: + BitStream(uint8_t* buffer, size_t size_in_bits); + ~BitStream(); + + const uint8_t* buffer() const { return buffer_; } + uint8_t* buffer() { return buffer_; } + size_t offset_bits() const { return offset_bits_; } + size_t size_bits() const { return size_bits_; } + + void Advance(size_t num_bits); + void SetOffset(size_t offset_bits); + size_t BitsRemaining(); + + // Note: num_bits MUST be in the range 0-57 (inclusive) + uint64_t Peek(size_t num_bits); + uint64_t Read(size_t num_bits); + bool Write(uint64_t val, size_t num_bits); // TODO: Not tested! + + size_t Copy(uint8_t* dest_buffer, size_t num_bits); + + private: + uint8_t* buffer_ = nullptr; + size_t offset_bits_ = 0; + size_t size_bits_ = 0; +}; +} // namespace xe + +#endif // XENIA_BASE_BIT_STREAM_H_ \ No newline at end of file diff --git a/src/xenia/base/premake5.lua b/src/xenia/base/premake5.lua index 155c68613..21ed7720e 100644 --- a/src/xenia/base/premake5.lua +++ b/src/xenia/base/premake5.lua @@ -8,6 +8,7 @@ project("xenia-base") defines({ }) includedirs({ + project_root.."/build_tools/third_party/gflags/src", }) local_platform_files() removefiles({"main_*.cc"}) @@ -17,6 +18,7 @@ project("xenia-base") test_suite("xenia-base-tests", project_root, ".", { includedirs = { + project_root.."/build_tools/third_party/gflags/src", }, links = { "xenia-base", diff --git a/src/xenia/cpu/backend/x64/premake5.lua b/src/xenia/cpu/backend/x64/premake5.lua index 795946cc9..1ad85ce55 100644 --- a/src/xenia/cpu/backend/x64/premake5.lua +++ b/src/xenia/cpu/backend/x64/premake5.lua @@ -21,5 +21,6 @@ project("xenia-cpu-backend-x64") }) includedirs({ project_root.."/third_party/capstone/include", + project_root.."/build_tools/third_party/gflags/src", }) local_platform_files() diff --git a/src/xenia/cpu/frontend/testing/premake5.lua b/src/xenia/cpu/frontend/testing/premake5.lua index afce34fba..74f0a39fc 100644 --- a/src/xenia/cpu/frontend/testing/premake5.lua +++ b/src/xenia/cpu/frontend/testing/premake5.lua @@ -24,6 +24,9 @@ project("xenia-cpu-frontend-tests") files({ "*.s", }) + includedirs({ + project_root.."/build_tools/third_party/gflags/src", + }) filter("files:*.s") flags({"ExcludeFromBuild"}) filter("platforms:Windows") diff --git a/src/xenia/cpu/premake5.lua b/src/xenia/cpu/premake5.lua index 14c731837..15f6a00a2 100644 --- a/src/xenia/cpu/premake5.lua +++ b/src/xenia/cpu/premake5.lua @@ -11,6 +11,7 @@ project("xenia-cpu") }) includedirs({ project_root.."/third_party/llvm/include", + project_root.."/build_tools/third_party/gflags/src", }) local_platform_files() local_platform_files("backend") diff --git a/src/xenia/cpu/testing/premake5.lua b/src/xenia/cpu/testing/premake5.lua index bbef37b84..fa3df2df2 100644 --- a/src/xenia/cpu/testing/premake5.lua +++ b/src/xenia/cpu/testing/premake5.lua @@ -3,6 +3,7 @@ include(project_root.."/build_tools") test_suite("xenia-cpu-tests", project_root, ".", { includedirs = { + project_root.."/build_tools/third_party/gflags/src", }, links = { "xenia-base", diff --git a/src/xenia/debug/premake5.lua b/src/xenia/debug/premake5.lua index 0a2e8da88..ad6670fc5 100644 --- a/src/xenia/debug/premake5.lua +++ b/src/xenia/debug/premake5.lua @@ -13,6 +13,7 @@ project("xenia-debug") defines({ }) includedirs({ + project_root.."/build_tools/third_party/gflags/src", }) local_platform_files() recursive_platform_files("proto") diff --git a/src/xenia/debug/ui/premake5.lua b/src/xenia/debug/ui/premake5.lua index 508d4ccb8..e7f24d51c 100644 --- a/src/xenia/debug/ui/premake5.lua +++ b/src/xenia/debug/ui/premake5.lua @@ -34,6 +34,7 @@ project("xenia-debug-ui") }) includedirs({ project_root.."/third_party/elemental-forms/src", + project_root.."/build_tools/third_party/gflags/src", }) recursive_platform_files() files({ @@ -48,19 +49,6 @@ project("xenia-debug-ui") project_root.."/third_party/elemental-forms", }) - filter("configurations:Checked") - local libav_root = "../third_party/libav-xma-bin/lib/Debug" - linkoptions({ - libav_root.."/libavcodec.a", - libav_root.."/libavutil.a", - }) - filter("configurations:Debug or Release") - local libav_root = "../third_party/libav-xma-bin/lib/Release" - linkoptions({ - libav_root.."/libavcodec.a", - libav_root.."/libavutil.a", - }) - filter("platforms:Windows") debugdir(project_root) debugargs({ diff --git a/src/xenia/gpu/gl4/premake5.lua b/src/xenia/gpu/gl4/premake5.lua index 643896d3d..435e3542f 100644 --- a/src/xenia/gpu/gl4/premake5.lua +++ b/src/xenia/gpu/gl4/premake5.lua @@ -21,6 +21,7 @@ project("xenia-gpu-gl4") }) includedirs({ project_root.."/third_party/elemental-forms/src", + project_root.."/build_tools/third_party/gflags/src", }) local_platform_files() @@ -62,25 +63,13 @@ project("xenia-gpu-gl4-trace-viewer") }) includedirs({ project_root.."/third_party/elemental-forms/src", + project_root.."/build_tools/third_party/gflags/src", }) files({ "trace_viewer_main.cc", "../../base/main_"..platform_suffix..".cc", }) - filter("configurations:Checked") - local libav_root = "../third_party/libav-xma-bin/lib/Debug" - linkoptions({ - libav_root.."/libavcodec.a", - libav_root.."/libavutil.a", - }) - filter("configurations:Debug or Release") - local libav_root = "../third_party/libav-xma-bin/lib/Release" - linkoptions({ - libav_root.."/libavcodec.a", - libav_root.."/libavutil.a", - }) - filter("platforms:Windows") debugdir(project_root) debugargs({ diff --git a/src/xenia/gpu/premake5.lua b/src/xenia/gpu/premake5.lua index 66d919f30..521a04869 100644 --- a/src/xenia/gpu/premake5.lua +++ b/src/xenia/gpu/premake5.lua @@ -16,5 +16,6 @@ project("xenia-gpu") }) includedirs({ project_root.."/third_party/elemental-forms/src", + project_root.."/build_tools/third_party/gflags/src", }) local_platform_files() diff --git a/src/xenia/hid/nop/premake5.lua b/src/xenia/hid/nop/premake5.lua index 7fec94cc3..baf417d5f 100644 --- a/src/xenia/hid/nop/premake5.lua +++ b/src/xenia/hid/nop/premake5.lua @@ -13,5 +13,6 @@ project("xenia-hid-nop") defines({ }) includedirs({ + project_root.."/build_tools/third_party/gflags/src", }) local_platform_files() diff --git a/src/xenia/hid/premake5.lua b/src/xenia/hid/premake5.lua index 45f3f7bcd..4904d9efd 100644 --- a/src/xenia/hid/premake5.lua +++ b/src/xenia/hid/premake5.lua @@ -12,5 +12,6 @@ project("xenia-hid") defines({ }) includedirs({ + project_root.."/build_tools/third_party/gflags/src", }) local_platform_files() diff --git a/src/xenia/hid/winkey/premake5.lua b/src/xenia/hid/winkey/premake5.lua index b30122d2a..44736b50d 100644 --- a/src/xenia/hid/winkey/premake5.lua +++ b/src/xenia/hid/winkey/premake5.lua @@ -13,6 +13,7 @@ project("xenia-hid-winkey") defines({ }) includedirs({ + project_root.."/build_tools/third_party/gflags/src", project_root.."/third_party/elemental-forms/src", }) local_platform_files() diff --git a/src/xenia/hid/xinput/premake5.lua b/src/xenia/hid/xinput/premake5.lua index adb3f758c..568171b6f 100644 --- a/src/xenia/hid/xinput/premake5.lua +++ b/src/xenia/hid/xinput/premake5.lua @@ -13,5 +13,6 @@ project("xenia-hid-xinput") defines({ }) includedirs({ + project_root.."/build_tools/third_party/gflags/src", }) local_platform_files() diff --git a/src/xenia/kernel/premake5.lua b/src/xenia/kernel/premake5.lua index 509d39a72..e26ea9ccb 100644 --- a/src/xenia/kernel/premake5.lua +++ b/src/xenia/kernel/premake5.lua @@ -20,6 +20,7 @@ project("xenia-kernel") }) includedirs({ project_root.."/third_party/elemental-forms/src", + project_root.."/build_tools/third_party/gflags/src", }) recursive_platform_files() files({ diff --git a/src/xenia/kernel/xboxkrnl_audio_xma.cc b/src/xenia/kernel/xboxkrnl_audio_xma.cc index a7fec8cb7..e664e7ca4 100644 --- a/src/xenia/kernel/xboxkrnl_audio_xma.cc +++ b/src/xenia/kernel/xboxkrnl_audio_xma.cc @@ -171,7 +171,7 @@ SHIM_CALL XMASetLoopData_shim(PPCContext* ppc_context, context.loop_end = loop_data->loop_end; context.loop_count = loop_data->loop_count; context.loop_subframe_end = loop_data->loop_subframe_end; - context.loop_subframe_skip = loop_data->loop_subframe_end; + context.loop_subframe_skip = loop_data->loop_subframe_skip; context.Store(SHIM_MEM_ADDR(context_ptr)); diff --git a/src/xenia/premake5.lua b/src/xenia/premake5.lua index 9fc178e71..f5d11a6fc 100644 --- a/src/xenia/premake5.lua +++ b/src/xenia/premake5.lua @@ -14,5 +14,6 @@ project("xenia-core") }) includedirs({ project_root.."/third_party/elemental-forms/src", + project_root.."/build_tools/third_party/gflags/src", }) files({"*.h", "*.cc"}) diff --git a/src/xenia/ui/gl/premake5.lua b/src/xenia/ui/gl/premake5.lua index 4d773ce9a..dab3c9a42 100644 --- a/src/xenia/ui/gl/premake5.lua +++ b/src/xenia/ui/gl/premake5.lua @@ -18,5 +18,6 @@ project("xenia-ui-gl") }) includedirs({ project_root.."/third_party/elemental-forms/src", + project_root.."/build_tools/third_party/gflags/src", }) local_platform_files() diff --git a/src/xenia/ui/premake5.lua b/src/xenia/ui/premake5.lua index 9e4fed1ae..43697e222 100644 --- a/src/xenia/ui/premake5.lua +++ b/src/xenia/ui/premake5.lua @@ -14,5 +14,6 @@ project("xenia-ui") }) includedirs({ project_root.."/third_party/elemental-forms/src", + project_root.."/build_tools/third_party/gflags/src", }) local_platform_files() diff --git a/src/xenia/vfs/premake5.lua b/src/xenia/vfs/premake5.lua index b5659dcf8..dc2331f56 100644 --- a/src/xenia/vfs/premake5.lua +++ b/src/xenia/vfs/premake5.lua @@ -12,5 +12,6 @@ project("xenia-vfs") defines({ }) includedirs({ + project_root.."/build_tools/third_party/gflags/src", }) recursive_platform_files() diff --git a/third_party/libav b/third_party/libav new file mode 160000 index 000000000..8be22f03d --- /dev/null +++ b/third_party/libav @@ -0,0 +1 @@ +Subproject commit 8be22f03d7e3c1663a66cc09375f840a7fc9a365 diff --git a/third_party/libav.lua b/third_party/libav.lua new file mode 100644 index 000000000..3341a696c --- /dev/null +++ b/third_party/libav.lua @@ -0,0 +1,3 @@ +group("third_party") + include("libav/libavcodec/premake5.lua") + include("libav/libavutil/premake5.lua")