/** ****************************************************************************** * Xenia : Xbox 360 Emulator Research Project * ****************************************************************************** * Copyright 2013 Ben Vanik. All rights reserved. * * Released under the BSD license - see LICENSE in the root for more details. * ****************************************************************************** */ #include "xenia/apu/audio_system.h" #include "xenia/apu/apu_flags.h" #include "xenia/apu/audio_driver.h" #include "xenia/apu/xma_decoder.h" #include "xenia/base/byte_stream.h" #include "xenia/base/logging.h" #include "xenia/base/math.h" #include "xenia/base/profiling.h" #include "xenia/base/ring_buffer.h" #include "xenia/base/string_buffer.h" #include "xenia/base/threading.h" #include "xenia/cpu/thread_state.h" // As with normal Microsoft, there are like twelve different ways to access // the audio APIs. Early games use XMA*() methods almost exclusively to touch // decoders. Later games use XAudio*() and direct memory writes to the XMA // structures (as opposed to the XMA* calls), meaning that we have to support // both. // // For ease of implementation, most audio related processing is handled in // AudioSystem, and the functions here call off to it. // The XMA*() functions just manipulate the audio system in the guest context // and let the normal AudioSystem handling take it, to prevent duplicate // implementations. They can be found in xboxkrnl_audio_xma.cc namespace xe { namespace apu { AudioSystem::AudioSystem(cpu::Processor* processor) : memory_(processor->memory()), processor_(processor), worker_running_(false) { std::memset(clients_, 0, sizeof(clients_)); for (size_t i = 0; i < kMaximumClientCount; ++i) { client_semaphores_[i] = xe::threading::Semaphore::Create(0, kMaximumQueuedFrames); wait_handles_[i] = client_semaphores_[i].get(); } shutdown_event_ = xe::threading::Event::CreateAutoResetEvent(false); wait_handles_[kMaximumClientCount] = shutdown_event_.get(); xma_decoder_ = std::make_unique(processor_); resume_event_ = xe::threading::Event::CreateAutoResetEvent(false); } AudioSystem::~AudioSystem() { if (xma_decoder_) { xma_decoder_->Shutdown(); } } X_STATUS AudioSystem::Setup(kernel::KernelState* kernel_state) { X_STATUS result = xma_decoder_->Setup(kernel_state); if (result) { return result; } worker_running_ = true; worker_thread_ = kernel::object_ref( new kernel::XHostThread(kernel_state, 128 * 1024, 0, [this]() { WorkerThreadMain(); return 0; })); // As we run audio callbacks the debugger must be able to suspend us. worker_thread_->set_can_debugger_suspend(true); worker_thread_->set_name("Audio Worker"); worker_thread_->Create(); return X_STATUS_SUCCESS; } void AudioSystem::WorkerThreadMain() { // Initialize driver and ringbuffer. Initialize(); // Main run loop. while (worker_running_) { // These handles signify the number of submitted samples. Once we reach // 64 samples, we wait until our audio backend releases a semaphore // (signaling a sample has finished playing) auto result = xe::threading::WaitAny(wait_handles_, xe::countof(wait_handles_), true); if (result.first == xe::threading::WaitResult::kFailed) { // TODO: Assert? continue; } if (result.first == threading::WaitResult::kSuccess && result.second == kMaximumClientCount) { // Shutdown event signaled. if (paused_) { pause_fence_.Signal(); threading::Wait(resume_event_.get(), false); } continue; } // Number of clients pumped bool pumped = false; if (result.first == xe::threading::WaitResult::kSuccess) { auto index = result.second; auto global_lock = global_critical_region_.Acquire(); uint32_t client_callback = clients_[index].callback; uint32_t client_callback_arg = clients_[index].wrapped_callback_arg; global_lock.unlock(); if (client_callback) { SCOPE_profile_cpu_i("apu", "xe::apu::AudioSystem->client_callback"); uint64_t args[] = {client_callback_arg}; processor_->Execute(worker_thread_->thread_state(), client_callback, args, xe::countof(args)); } pumped = true; } if (!worker_running_) { break; } if (!pumped) { SCOPE_profile_cpu_i("apu", "Sleep"); xe::threading::Sleep(std::chrono::milliseconds(500)); } } worker_running_ = false; // TODO(benvanik): call module API to kill? } int AudioSystem::FindFreeClient() { for (int i = 0; i < kMaximumClientCount; i++) { auto& client = clients_[i]; if (!client.in_use) { return i; } } return -1; } void AudioSystem::Initialize() {} void AudioSystem::Shutdown() { worker_running_ = false; shutdown_event_->Set(); if (worker_thread_) { worker_thread_->Wait(0, 0, 0, nullptr); worker_thread_.reset(); } } X_STATUS AudioSystem::RegisterClient(uint32_t callback, uint32_t callback_arg, size_t* out_index) { auto global_lock = global_critical_region_.Acquire(); auto index = FindFreeClient(); assert_true(index >= 0); auto client_semaphore = client_semaphores_[index].get(); auto ret = client_semaphore->Release(kMaximumQueuedFrames, nullptr); assert_true(ret); AudioDriver* driver; auto result = CreateDriver(index, client_semaphore, &driver); if (XFAILED(result)) { return result; } assert_not_null(driver); uint32_t ptr = memory()->SystemHeapAlloc(0x4); xe::store_and_swap(memory()->TranslateVirtual(ptr), callback_arg); clients_[index] = {driver, callback, callback_arg, ptr, true}; if (out_index) { *out_index = index; } return X_STATUS_SUCCESS; } void AudioSystem::SubmitFrame(size_t index, uint32_t samples_ptr) { SCOPE_profile_cpu_f("apu"); auto global_lock = global_critical_region_.Acquire(); assert_true(index < kMaximumClientCount); assert_true(clients_[index].driver != NULL); (clients_[index].driver)->SubmitFrame(samples_ptr); } void AudioSystem::UnregisterClient(size_t index) { SCOPE_profile_cpu_f("apu"); auto global_lock = global_critical_region_.Acquire(); assert_true(index < kMaximumClientCount); DestroyDriver(clients_[index].driver); memory()->SystemHeapFree(clients_[index].wrapped_callback_arg); clients_[index] = {0}; // Drain the semaphore of its count. auto client_semaphore = client_semaphores_[index].get(); xe::threading::WaitResult wait_result; do { wait_result = xe::threading::Wait(client_semaphore, false, std::chrono::milliseconds(0)); } while (wait_result == xe::threading::WaitResult::kSuccess); assert_true(wait_result == xe::threading::WaitResult::kTimeout); } bool AudioSystem::Save(ByteStream* stream) { stream->Write('XAUD'); // Count the number of used clients first. // Any gaps should be handled gracefully. uint32_t used_clients = 0; for (int i = 0; i < kMaximumClientCount; i++) { if (clients_[i].in_use) { used_clients++; } } stream->Write(used_clients); for (uint32_t i = 0; i < kMaximumClientCount; i++) { auto& client = clients_[i]; if (!client.in_use) { continue; } stream->Write(i); stream->Write(client.callback); stream->Write(client.callback_arg); stream->Write(client.wrapped_callback_arg); } return true; } bool AudioSystem::Restore(ByteStream* stream) { if (stream->Read() != 'XAUD') { XELOGE("AudioSystem::Restore - Invalid magic value!"); return false; } uint32_t num_clients = stream->Read(); for (uint32_t i = 0; i < num_clients; i++) { auto id = stream->Read(); assert_true(id < kMaximumClientCount); auto& client = clients_[id]; // Reset the semaphore and recreate the driver ourselves. if (client.driver) { UnregisterClient(id); } client.callback = stream->Read(); client.callback_arg = stream->Read(); client.wrapped_callback_arg = stream->Read(); client.in_use = true; auto client_semaphore = client_semaphores_[id].get(); auto ret = client_semaphore->Release(kMaximumQueuedFrames, nullptr); assert_true(ret); AudioDriver* driver = nullptr; auto status = CreateDriver(id, client_semaphore, &driver); if (XFAILED(status)) { XELOGE( "AudioSystem::Restore - Call to CreateDriver failed with status %.8X", status); return false; } assert_not_null(driver); client.driver = driver; } return true; } void AudioSystem::Pause() { if (paused_) { return; } paused_ = true; // Kind of a hack, but it works. shutdown_event_->Set(); pause_fence_.Wait(); xma_decoder_->Pause(); } void AudioSystem::Resume() { if (!paused_) { return; } paused_ = false; resume_event_->Set(); xma_decoder_->Resume(); } } // namespace apu } // namespace xe