2016-02-21 13:13:52 +00:00
|
|
|
// Copyright 2016 Citra Emulator Project
|
|
|
|
// Licensed under GPLv2 or any later version
|
|
|
|
// Refer to the license.txt file included.
|
|
|
|
|
2016-04-27 13:53:23 +01:00
|
|
|
#include <memory>
|
|
|
|
#include <string>
|
2016-09-20 23:52:38 -07:00
|
|
|
#include "audio_core/audio_core.h"
|
2016-02-21 13:13:52 +00:00
|
|
|
#include "audio_core/hle/dsp.h"
|
2016-04-24 10:21:10 +01:00
|
|
|
#include "audio_core/hle/pipe.h"
|
2016-04-27 13:53:23 +01:00
|
|
|
#include "audio_core/null_sink.h"
|
|
|
|
#include "audio_core/sink.h"
|
|
|
|
#include "audio_core/sink_details.h"
|
2016-02-21 13:13:52 +00:00
|
|
|
#include "core/core_timing.h"
|
|
|
|
#include "core/hle/kernel/vm_manager.h"
|
|
|
|
#include "core/hle/service/dsp_dsp.h"
|
|
|
|
|
|
|
|
namespace AudioCore {
|
|
|
|
|
|
|
|
// Audio Ticks occur about every 5 miliseconds.
|
|
|
|
static int tick_event; ///< CoreTiming event
|
|
|
|
static constexpr u64 audio_frame_ticks = 1310252ull; ///< Units: ARM11 cycles
|
|
|
|
|
|
|
|
static void AudioTickCallback(u64 /*userdata*/, int cycles_late) {
|
|
|
|
if (DSP::HLE::Tick()) {
|
2016-04-24 10:21:10 +01:00
|
|
|
// TODO(merry): Signal all the other interrupts as appropriate.
|
|
|
|
DSP_DSP::SignalPipeInterrupt(DSP::HLE::DspPipe::Audio);
|
2016-04-24 13:36:44 +01:00
|
|
|
// HACK(merry): Added to prevent regressions. Will remove soon.
|
|
|
|
DSP_DSP::SignalPipeInterrupt(DSP::HLE::DspPipe::Binary);
|
2016-02-21 13:13:52 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Reschedule recurrent event
|
|
|
|
CoreTiming::ScheduleEvent(audio_frame_ticks - cycles_late, tick_event);
|
|
|
|
}
|
|
|
|
|
|
|
|
void Init() {
|
|
|
|
DSP::HLE::Init();
|
|
|
|
|
|
|
|
tick_event = CoreTiming::RegisterEvent("AudioCore::tick_event", AudioTickCallback);
|
|
|
|
CoreTiming::ScheduleEvent(audio_frame_ticks, tick_event);
|
|
|
|
}
|
|
|
|
|
|
|
|
void AddAddressSpace(Kernel::VMManager& address_space) {
|
2016-09-18 09:38:01 +09:00
|
|
|
auto r0_vma = address_space
|
|
|
|
.MapBackingMemory(DSP::HLE::region0_base,
|
|
|
|
reinterpret_cast<u8*>(&DSP::HLE::g_regions[0]),
|
|
|
|
sizeof(DSP::HLE::SharedMemory), Kernel::MemoryState::IO)
|
|
|
|
.MoveFrom();
|
2016-02-21 13:13:52 +00:00
|
|
|
address_space.Reprotect(r0_vma, Kernel::VMAPermission::ReadWrite);
|
|
|
|
|
2016-09-18 09:38:01 +09:00
|
|
|
auto r1_vma = address_space
|
|
|
|
.MapBackingMemory(DSP::HLE::region1_base,
|
|
|
|
reinterpret_cast<u8*>(&DSP::HLE::g_regions[1]),
|
|
|
|
sizeof(DSP::HLE::SharedMemory), Kernel::MemoryState::IO)
|
|
|
|
.MoveFrom();
|
2016-02-21 13:13:52 +00:00
|
|
|
address_space.Reprotect(r1_vma, Kernel::VMAPermission::ReadWrite);
|
|
|
|
}
|
|
|
|
|
2016-04-27 13:53:23 +01:00
|
|
|
void SelectSink(std::string sink_id) {
|
2016-09-18 09:38:01 +09:00
|
|
|
auto iter =
|
|
|
|
std::find_if(g_sink_details.begin(), g_sink_details.end(),
|
|
|
|
[sink_id](const auto& sink_detail) { return sink_detail.id == sink_id; });
|
2016-04-27 13:53:23 +01:00
|
|
|
|
2016-12-10 01:06:53 +00:00
|
|
|
if (sink_id == "auto" || iter == g_sink_details.end()) {
|
|
|
|
if (sink_id != "auto") {
|
|
|
|
LOG_ERROR(Audio, "AudioCore::SelectSink given invalid sink_id %s", sink_id.c_str());
|
|
|
|
}
|
|
|
|
// Auto-select.
|
|
|
|
// g_sink_details is ordered in terms of desirability, with the best choice at the front.
|
|
|
|
iter = g_sink_details.begin();
|
2016-04-27 13:53:23 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
DSP::HLE::SetSink(iter->factory());
|
|
|
|
}
|
|
|
|
|
2016-08-31 16:56:30 +01:00
|
|
|
void EnableStretching(bool enable) {
|
|
|
|
DSP::HLE::EnableStretching(enable);
|
|
|
|
}
|
|
|
|
|
2016-02-21 13:13:52 +00:00
|
|
|
void Shutdown() {
|
|
|
|
CoreTiming::UnscheduleEvent(tick_event, 0);
|
|
|
|
DSP::HLE::Shutdown();
|
|
|
|
}
|
|
|
|
|
2016-04-27 13:53:23 +01:00
|
|
|
} // namespace AudioCore
|