[XAM] XMP: Implement playback

This commit is contained in:
Hendi 2024-10-15 00:56:13 +02:00 committed by Radosław Gliński
parent 5a76cac218
commit 764f230dd9
3 changed files with 295 additions and 10 deletions

View File

@ -16,9 +16,15 @@ project("xenia-kernel")
"xenia-cpu",
"xenia-hid",
"xenia-vfs",
"libavcodec",
"libavformat",
"libavutil",
})
defines({
})
includedirs({
project_root.."/third_party/FFmpeg/",
})
recursive_platform_files()
files({
"debug_visualizers.natvis",

View File

@ -12,8 +12,29 @@
#include "xenia/base/logging.h"
#include "xenia/base/threading.h"
#include "xenia/emulator.h"
#include "xenia/xbox.h"
#include "xenia/apu/audio_driver.h"
#include "xenia/apu/audio_system.h"
extern "C" {
#if XE_COMPILER_MSVC
#pragma warning(push)
#pragma warning(disable : 4101 4244 5033)
#endif
#include "third_party/FFmpeg/libavcodec/avcodec.h"
#include "third_party/FFmpeg/libavformat/avformat.h"
#include "third_party/FFmpeg/libavutil/opt.h"
#if XE_COMPILER_MSVC
#pragma warning(pop)
#endif
} // extern "C"
DEFINE_bool(enable_xmp, true, "Enables Music Player playback.", "APU");
DEFINE_int32(xmp_default_volume, 70,
"Default music volume if game doesn't set it [0-100].", "APU");
namespace xe {
namespace kernel {
namespace xam {
@ -26,11 +47,247 @@ XmpApp::XmpApp(KernelState* kernel_state)
playback_mode_(PlaybackMode::kUnknown),
repeat_mode_(RepeatMode::kUnknown),
unknown_flags_(0),
volume_(1.0f),
volume_(cvars::xmp_default_volume / 100.0f),
active_playlist_(nullptr),
active_song_index_(0),
next_playlist_handle_(1),
next_song_handle_(1) {}
next_song_handle_(1) {
if (cvars::enable_xmp) {
worker_running_ = true;
worker_thread_ = threading::Thread::Create({}, [&] { WorkerThreadMain(); });
worker_thread_->set_name("Music Player");
}
}
struct VFSContext {
xe::vfs::File* file;
size_t byte_offset;
};
static int xenia_vfs_read(void* opaque, uint8_t* buf, int buf_size) {
auto ctx = static_cast<VFSContext*>(opaque);
size_t bytes_read;
X_STATUS status =
ctx->file->ReadSync(buf, buf_size, ctx->byte_offset, &bytes_read);
if (XFAILED(status)) {
return status == X_STATUS_END_OF_FILE ? AVERROR_EOF : status;
}
ctx->byte_offset += bytes_read;
return static_cast<int>(bytes_read);
}
bool XmpApp::PlayFile(std::string_view filename) {
auto playlist = active_playlist_;
const int buffer_size = 8192;
uint8_t* buffer = reinterpret_cast<uint8_t*>(av_malloc(buffer_size));
VFSContext vfs_ctx = {};
xe::vfs::FileAction file_action;
X_STATUS status = kernel_state_->file_system()->OpenFile(
nullptr, filename, xe::vfs::FileDisposition::kOpen,
xe::vfs::FileAccess::kGenericRead, false, true, &vfs_ctx.file,
&file_action);
if (XFAILED(status)) {
XELOGE("Opening {} failed with status {:X}", filename, status);
return false;
}
AVIOContext* avio_ctx = avio_alloc_context(buffer, buffer_size, 0, &vfs_ctx,
xenia_vfs_read, nullptr, nullptr);
AVFormatContext* formatContext = avformat_alloc_context();
formatContext->pb = avio_ctx;
int ret;
if ((ret = avformat_open_input(&formatContext, nullptr, nullptr, nullptr)) !=
0) {
XELOGE("ffmpeg: Could not open WMA file: {:x}", ret);
av_freep(&avio_ctx->buffer);
avio_context_free(&avio_ctx);
return false;
}
if (avformat_find_stream_info(formatContext, nullptr) < 0) {
XELOGE("ffmpeg: Could not find stream info");
avformat_close_input(&formatContext);
av_freep(&avio_ctx->buffer);
avio_context_free(&avio_ctx);
return false;
}
AVCodec* codec = nullptr;
int streamIndex =
av_find_best_stream(formatContext, AVMEDIA_TYPE_AUDIO, -1, -1, &codec, 0);
if (streamIndex < 0) {
XELOGE("ffmpeg: Could not find audio stream");
avformat_close_input(&formatContext);
av_freep(&avio_ctx->buffer);
avio_context_free(&avio_ctx);
return false;
}
AVStream* audioStream = formatContext->streams[streamIndex];
AVCodecContext* codecContext = avcodec_alloc_context3(codec);
avcodec_parameters_to_context(codecContext, audioStream->codecpar);
if (audioStream->codecpar->format != AV_SAMPLE_FMT_FLTP &&
audioStream->codecpar->format != AV_SAMPLE_FMT_FLT) {
XELOGE("Audio stream has unexpected sample format {:d}",
audioStream->codecpar->format);
avcodec_free_context(&codecContext);
avformat_close_input(&formatContext);
av_freep(&avio_ctx->buffer);
avio_context_free(&avio_ctx);
return false;
}
if (avcodec_open2(codecContext, codec, nullptr) < 0) {
XELOGE("ffmpeg: Could not open codec");
avcodec_free_context(&codecContext);
avformat_close_input(&formatContext);
av_freep(&avio_ctx->buffer);
avio_context_free(&avio_ctx);
return false;
}
auto driverReady = xe::threading::Semaphore::Create(64, 64);
{
std::unique_lock<std::mutex> guard(driver_mutex_);
driver_ = kernel_state_->emulator()->audio_system()->CreateDriver(
driverReady.get(), codecContext->sample_rate, codecContext->channels,
false);
if (!driver_->Initialize()) {
XELOGE("Driver initialization failed!");
driver_->Shutdown();
driver_ = nullptr;
avcodec_free_context(&codecContext);
avformat_close_input(&formatContext);
av_freep(&avio_ctx->buffer);
avio_context_free(&avio_ctx);
return false;
}
}
if (volume_ == 0.0f) {
// Some games set volume to 0 on startup and then never call SetVolume
// again...
volume_ = cvars::xmp_default_volume / 100.0f;
}
driver_->SetVolume(volume_);
AVPacket* packet = av_packet_alloc();
AVFrame* frame = av_frame_alloc();
std::vector<float> frameBuffer;
// Read frames, decode & send to audio driver
while (av_read_frame(formatContext, packet) >= 0) {
if (active_playlist_ != playlist) {
frameBuffer.clear();
break;
}
if (packet->stream_index == streamIndex) {
int ret = avcodec_send_packet(codecContext, packet);
if (ret < 0) {
XELOGE("Error sending packet for decoding: {:X}", ret);
break;
}
while (ret >= 0) {
ret = avcodec_receive_frame(codecContext, frame);
if (ret == AVERROR(EAGAIN) || ret == AVERROR_EOF) break;
if (ret < 0) {
XELOGW("Error during decoding: {:X}", ret);
break;
}
// If the frame is planar, convert it to interleaved
if (frame->format == AV_SAMPLE_FMT_FLTP) {
for (int sample = 0; sample < frame->nb_samples; sample++) {
for (int ch = 0; ch < codecContext->channels; ch++) {
float sampleValue =
reinterpret_cast<float*>(frame->data[ch])[sample];
frameBuffer.push_back(sampleValue);
}
}
} else if (frame->format == AV_SAMPLE_FMT_FLT) {
int frameSizeFloats = frame->nb_samples * codecContext->channels;
float* frameData = reinterpret_cast<float*>(frame->data[0]);
frameBuffer.insert(frameBuffer.end(), frameData,
frameData + frameSizeFloats);
}
while (frameBuffer.size() >= xe::apu::AudioDriver::kFrameSamplesMax) {
xe::threading::Wait(driverReady.get(), true);
if (active_playlist_ != playlist) {
frameBuffer.clear();
break;
}
driver_->SubmitFrame(frameBuffer.data());
frameBuffer.erase(
frameBuffer.begin(),
frameBuffer.begin() + xe::apu::AudioDriver::kFrameSamplesMax);
}
}
}
av_packet_unref(packet);
}
if (!frameBuffer.empty()) {
while (frameBuffer.size() < xe::apu::AudioDriver::kFrameSamplesMax) {
frameBuffer.push_back(0.0f);
}
xe::threading::Wait(driverReady.get(), true);
driver_->SubmitFrame(frameBuffer.data());
}
av_frame_free(&frame);
av_packet_free(&packet);
avcodec_free_context(&codecContext);
avformat_close_input(&formatContext);
av_freep(&avio_ctx->buffer);
avio_context_free(&avio_ctx);
{
std::unique_lock<std::mutex> guard(driver_mutex_);
driver_->Shutdown();
driver_ = nullptr;
}
if (state_ == State::kPlaying && active_playlist_ == playlist) {
active_playlist_ = nullptr;
state_ = State::kIdle;
OnStateChanged();
}
return true;
}
void XmpApp::WorkerThreadMain() {
while (worker_running_) {
if (state_ != State::kPlaying) {
resume_fence_.Wait();
}
auto playlist = active_playlist_;
if (!playlist) {
continue;
}
auto utf8_path = xe::path_to_utf8(playlist->songs[0].get()->file_path);
XELOGI("Playing file {}", utf8_path);
if (!PlayFile(utf8_path)) {
XELOGE("Playback failed");
xe::threading::Sleep(std::chrono::minutes(1));
}
}
}
X_HRESULT XmpApp::XMPGetStatus(uint32_t state_ptr) {
if (!XThread::GetCurrentThread()->main_thread()) {
@ -39,7 +296,7 @@ X_HRESULT XmpApp::XMPGetStatus(uint32_t state_ptr) {
xe::threading::Sleep(std::chrono::milliseconds(1));
}
XELOGD("XMPGetStatus({:08X})", state_ptr);
XELOGD("XMPGetStatus({:08X}) -> {:d}", state_ptr, (uint32_t)state_);
xe::store_and_swap<uint32_t>(memory_->TranslateVirtual(state_ptr),
static_cast<uint32_t>(state_));
return X_E_SUCCESS;
@ -132,16 +389,10 @@ X_HRESULT XmpApp::XMPPlayTitlePlaylist(uint32_t playlist_handle,
playlist = it->second;
}
if (playback_client_ == PlaybackClient::kSystem) {
XELOGW("XMPPlayTitlePlaylist: System playback is enabled!");
return X_E_SUCCESS;
}
// Start playlist?
XELOGW("Playlist playback not supported");
active_playlist_ = playlist;
active_song_index_ = 0;
state_ = State::kPlaying;
resume_fence_.Signal();
OnStateChanged();
kernel_state_->BroadcastNotification(kNotificationXmpPlaybackBehaviorChanged,
1);
@ -152,6 +403,11 @@ X_HRESULT XmpApp::XMPContinue() {
XELOGD("XMPContinue()");
if (state_ == State::kPaused) {
state_ = State::kPlaying;
resume_fence_.Signal();
{
std::unique_lock<std::mutex> guard(driver_mutex_);
if (driver_ != nullptr) driver_->Resume();
}
}
OnStateChanged();
return X_E_SUCCESS;
@ -170,6 +426,10 @@ X_HRESULT XmpApp::XMPStop(uint32_t unk) {
X_HRESULT XmpApp::XMPPause() {
XELOGD("XMPPause()");
if (state_ == State::kPlaying) {
{
std::unique_lock<std::mutex> guard(driver_mutex_);
if (driver_ != nullptr) driver_->Pause();
}
state_ = State::kPaused;
}
OnStateChanged();
@ -184,6 +444,7 @@ X_HRESULT XmpApp::XMPNext() {
state_ = State::kPlaying;
active_song_index_ =
(active_song_index_ + 1) % active_playlist_->songs.size();
resume_fence_.Signal();
OnStateChanged();
return X_E_SUCCESS;
}
@ -199,6 +460,7 @@ X_HRESULT XmpApp::XMPPrevious() {
} else {
--active_song_index_;
}
resume_fence_.Signal();
OnStateChanged();
return X_E_SUCCESS;
}
@ -309,6 +571,10 @@ X_HRESULT XmpApp::DispatchMessageSync(uint32_t message, uint32_t buffer_ptr,
assert_true(args->xmp_client == 0x00000002);
XELOGD("XMPSetVolume({:g})", float(args->value));
volume_ = args->value;
{
std::unique_lock<std::mutex> guard(driver_mutex_);
if (driver_ != nullptr) driver_->SetVolume(volume_);
}
return X_E_SUCCESS;
}
case 0x0007000D: {

View File

@ -11,6 +11,7 @@
#define XENIA_KERNEL_XAM_APPS_XMP_APP_H_
#include <memory>
#include <mutex>
#include <string>
#include <unordered_map>
#include <vector>
@ -19,6 +20,8 @@
#include "xenia/kernel/kernel_state.h"
#include "xenia/kernel/xam/app_manager.h"
#include "xenia/apu/audio_driver.h"
namespace xe {
namespace kernel {
namespace xam {
@ -93,6 +96,8 @@ class XmpApp : public App {
private:
void OnStateChanged();
void WorkerThreadMain();
bool PlayFile(std::string_view filename);
State state_;
PlaybackClient playback_client_;
@ -107,6 +112,14 @@ class XmpApp : public App {
std::unordered_map<uint32_t, Playlist*> playlists_;
uint32_t next_playlist_handle_;
uint32_t next_song_handle_;
std::atomic<bool> worker_running_ = {false};
std::unique_ptr<xe::threading::Thread> worker_thread_;
bool paused_ = true;
xe::threading::Fence resume_fence_; // Signaled when resume requested.
std::mutex driver_mutex_ = {};
xe::apu::AudioDriver* driver_ = nullptr;
};
} // namespace apps