2017-12-20 18:44:32 +00:00
|
|
|
// Copyright 2017 Citra Emulator Project
|
|
|
|
// Licensed under GPLv2 or any later version
|
|
|
|
// Refer to the license.txt file included.
|
|
|
|
|
|
|
|
#pragma once
|
|
|
|
|
|
|
|
#include <memory>
|
2023-07-07 01:52:40 +03:00
|
|
|
#include <span>
|
2020-01-19 22:49:22 +00:00
|
|
|
#include <boost/serialization/access.hpp>
|
2017-12-20 18:44:32 +00:00
|
|
|
#include "audio_core/audio_types.h"
|
|
|
|
#include "audio_core/time_stretch.h"
|
|
|
|
#include "common/common_types.h"
|
2018-09-08 21:07:28 +01:00
|
|
|
#include "common/ring_buffer.h"
|
2017-12-20 18:44:32 +00:00
|
|
|
#include "core/memory.h"
|
|
|
|
|
2019-02-17 23:41:48 -05:00
|
|
|
namespace Service::DSP {
|
2023-10-04 19:14:59 +05:30
|
|
|
enum class InterruptType : u32;
|
2019-02-17 23:41:48 -05:00
|
|
|
} // namespace Service::DSP
|
2018-07-24 15:54:33 -04:00
|
|
|
|
2017-12-20 18:44:32 +00:00
|
|
|
namespace AudioCore {
|
|
|
|
|
|
|
|
class Sink;
|
2023-05-01 12:17:45 -07:00
|
|
|
enum class SinkType : u32;
|
2017-12-20 18:44:32 +00:00
|
|
|
|
|
|
|
class DspInterface {
|
|
|
|
public:
|
|
|
|
DspInterface();
|
|
|
|
virtual ~DspInterface();
|
|
|
|
|
|
|
|
DspInterface(const DspInterface&) = delete;
|
|
|
|
DspInterface(DspInterface&&) = delete;
|
|
|
|
DspInterface& operator=(const DspInterface&) = delete;
|
|
|
|
DspInterface& operator=(DspInterface&&) = delete;
|
|
|
|
|
2018-12-06 07:18:26 -05:00
|
|
|
/**
|
|
|
|
* Reads data from one of three DSP registers
|
|
|
|
* @note this function blocks until the data is available
|
|
|
|
* @param register_number the index of the register to read
|
|
|
|
* @returns the value of the register
|
|
|
|
*/
|
|
|
|
virtual u16 RecvData(u32 register_number) = 0;
|
2017-12-20 18:44:32 +00:00
|
|
|
|
2018-12-06 07:28:47 -05:00
|
|
|
/**
|
|
|
|
* Checks whether data is ready in one of three DSP registers
|
|
|
|
* @param register_number the index of the register to check
|
|
|
|
* @returns true if data is ready
|
|
|
|
*/
|
|
|
|
virtual bool RecvDataIsReady(u32 register_number) const = 0;
|
|
|
|
|
2018-12-06 07:35:07 -05:00
|
|
|
/**
|
|
|
|
* Sets the DSP semaphore register
|
|
|
|
* @param semaphore_value the value set to the semaphore register
|
|
|
|
*/
|
|
|
|
virtual void SetSemaphore(u16 semaphore_value) = 0;
|
|
|
|
|
2017-12-20 18:44:32 +00:00
|
|
|
/**
|
|
|
|
* Reads `length` bytes from the DSP pipe identified with `pipe_number`.
|
|
|
|
* @note Can read up to the maximum value of a u16 in bytes (65,535).
|
|
|
|
* @note IF an error is encoutered with either an invalid `pipe_number` or `length` value, an
|
|
|
|
* empty vector will be returned.
|
|
|
|
* @note IF `length` is set to 0, an empty vector will be returned.
|
|
|
|
* @note IF `length` is greater than the amount of data available, this function will only read
|
|
|
|
* the available amount.
|
|
|
|
* @param pipe_number a `DspPipe`
|
|
|
|
* @param length the number of bytes to read. The max is 65,535 (max of u16).
|
|
|
|
* @returns a vector of bytes from the specified pipe. On error, will be empty.
|
|
|
|
*/
|
2023-07-04 21:00:24 -07:00
|
|
|
virtual std::vector<u8> PipeRead(DspPipe pipe_number, std::size_t length) = 0;
|
2017-12-20 18:44:32 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* How much data is left in pipe
|
|
|
|
* @param pipe_number The Pipe ID
|
|
|
|
* @return The amount of data remaning in the pipe. This is the maximum length PipeRead will
|
|
|
|
* return.
|
|
|
|
*/
|
2018-09-06 16:03:28 -04:00
|
|
|
virtual std::size_t GetPipeReadableSize(DspPipe pipe_number) const = 0;
|
2017-12-20 18:44:32 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Write to a DSP pipe.
|
|
|
|
* @param pipe_number The Pipe ID
|
|
|
|
* @param buffer The data to write to the pipe.
|
|
|
|
*/
|
2023-07-07 01:52:40 +03:00
|
|
|
virtual void PipeWrite(DspPipe pipe_number, std::span<const u8> buffer) = 0;
|
2017-12-20 18:44:32 +00:00
|
|
|
|
|
|
|
/// Returns a reference to the array backing DSP memory
|
|
|
|
virtual std::array<u8, Memory::DSP_RAM_SIZE>& GetDspMemory() = 0;
|
|
|
|
|
2023-10-04 19:14:59 +05:30
|
|
|
/// Sets the handler for the interrupts we trigger
|
|
|
|
virtual void SetInterruptHandler(
|
|
|
|
std::function<void(Service::DSP::InterruptType type, DspPipe pipe)> handler) = 0;
|
2018-07-24 15:54:33 -04:00
|
|
|
|
2018-12-06 08:35:21 -05:00
|
|
|
/// Loads the DSP program
|
2023-07-07 01:52:40 +03:00
|
|
|
virtual void LoadComponent(std::span<const u8> buffer) = 0;
|
2018-12-06 08:35:21 -05:00
|
|
|
|
2018-12-06 09:50:55 -05:00
|
|
|
/// Unloads the DSP program
|
|
|
|
virtual void UnloadComponent() = 0;
|
|
|
|
|
2023-05-01 12:17:45 -07:00
|
|
|
/// Select the sink to use based on sink type.
|
|
|
|
void SetSink(SinkType sink_type, std::string_view audio_device);
|
2017-12-20 18:44:32 +00:00
|
|
|
/// Get the current sink
|
|
|
|
Sink& GetSink();
|
|
|
|
/// Enable/Disable audio stretching.
|
|
|
|
void EnableStretching(bool enable);
|
|
|
|
|
|
|
|
protected:
|
2020-01-28 22:19:36 +08:00
|
|
|
void OutputFrame(StereoFrame16 frame);
|
2018-12-06 12:19:58 -05:00
|
|
|
void OutputSample(std::array<s16, 2> sample);
|
2017-12-20 18:44:32 +00:00
|
|
|
|
|
|
|
private:
|
|
|
|
void FlushResidualStretcherAudio();
|
2018-09-08 21:07:28 +01:00
|
|
|
void OutputCallback(s16* buffer, std::size_t num_frames);
|
2017-12-20 18:44:32 +00:00
|
|
|
|
2018-09-08 21:28:19 +01:00
|
|
|
std::atomic<bool> perform_time_stretching = false;
|
|
|
|
std::atomic<bool> flushing_time_stretcher = false;
|
2018-09-08 21:07:28 +01:00
|
|
|
Common::RingBuffer<s16, 0x2000, 2> fifo;
|
|
|
|
std::array<s16, 2> last_frame{};
|
2017-12-20 18:44:32 +00:00
|
|
|
TimeStretcher time_stretcher;
|
2021-09-21 01:10:36 +08:00
|
|
|
std::unique_ptr<Sink> sink;
|
2020-01-19 22:49:22 +00:00
|
|
|
|
|
|
|
template <class Archive>
|
|
|
|
void serialize(Archive& ar, const unsigned int) {}
|
|
|
|
friend class boost::serialization::access;
|
2017-12-20 18:44:32 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
} // namespace AudioCore
|