2012-11-01 15:19:01 +00:00
|
|
|
// Copyright (c) 2012- PPSSPP Project.
|
|
|
|
|
|
|
|
// This program is free software: you can redistribute it and/or modify
|
|
|
|
// it under the terms of the GNU General Public License as published by
|
2012-11-04 22:01:49 +00:00
|
|
|
// the Free Software Foundation, version 2.0 or later versions.
|
2012-11-01 15:19:01 +00:00
|
|
|
|
|
|
|
// This program is distributed in the hope that it will be useful,
|
|
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
// GNU General Public License 2.0 for more details.
|
|
|
|
|
|
|
|
// A copy of the GPL 2.0 should have been included with the program.
|
|
|
|
// If not, see http://www.gnu.org/licenses/
|
|
|
|
|
|
|
|
// Official git repository and contact information can be found at
|
|
|
|
// https://github.com/hrydgard/ppsspp and http://www.ppsspp.org/.
|
|
|
|
|
|
|
|
#include "__sceAudio.h"
|
|
|
|
#include "sceAudio.h"
|
|
|
|
#include "sceKernel.h"
|
|
|
|
#include "sceKernelThread.h"
|
|
|
|
#include "StdMutex.h"
|
|
|
|
#include "CommonTypes.h"
|
|
|
|
#include "../CoreTiming.h"
|
|
|
|
#include "../MemMap.h"
|
|
|
|
#include "../Host.h"
|
2012-11-17 13:20:04 +00:00
|
|
|
#include "../Config.h"
|
|
|
|
#include "FixedSizeQueue.h"
|
|
|
|
#include "Common/Thread.h"
|
2012-11-01 15:19:01 +00:00
|
|
|
|
|
|
|
std::recursive_mutex section;
|
|
|
|
|
|
|
|
int eventAudioUpdate = -1;
|
2012-11-17 13:20:04 +00:00
|
|
|
int eventHostAudioUpdate = -1;
|
2012-12-17 18:38:04 +00:00
|
|
|
int mixFrequency = 44100;
|
2012-12-18 10:40:14 +00:00
|
|
|
|
2012-12-17 18:38:04 +00:00
|
|
|
const int hwSampleRate = 44100;
|
2012-12-18 10:40:14 +00:00
|
|
|
const int hwBlockSize = 60;
|
|
|
|
const int hostAttemptBlockSize = 256;
|
2012-11-17 13:20:04 +00:00
|
|
|
const int audioIntervalUs = (int)(1000000ULL * hwBlockSize / hwSampleRate);
|
|
|
|
const int audioHostIntervalUs = (int)(1000000ULL * hostAttemptBlockSize / hwSampleRate);
|
|
|
|
|
|
|
|
// High and low watermarks, basically.
|
2012-12-17 19:43:43 +00:00
|
|
|
const int chanQueueMaxSizeFactor = 4;
|
2012-12-16 05:11:53 +00:00
|
|
|
const int chanQueueMinSizeFactor = 1;
|
2012-11-17 13:20:04 +00:00
|
|
|
|
2012-12-18 10:40:14 +00:00
|
|
|
FixedSizeQueue<s16, hostAttemptBlockSize * 16> outAudioQueue;
|
2012-11-01 15:19:01 +00:00
|
|
|
|
|
|
|
|
|
|
|
void hleAudioUpdate(u64 userdata, int cyclesLate)
|
|
|
|
{
|
|
|
|
__AudioUpdate();
|
|
|
|
|
2012-12-18 10:40:14 +00:00
|
|
|
CoreTiming::ScheduleEvent(usToCycles(audioIntervalUs) - cyclesLate, eventAudioUpdate, 0);
|
2012-11-17 13:20:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void hleHostAudioUpdate(u64 userdata, int cyclesLate)
|
|
|
|
{
|
|
|
|
host->UpdateSound();
|
2012-12-18 10:40:14 +00:00
|
|
|
CoreTiming::ScheduleEvent(usToCycles(audioHostIntervalUs) - cyclesLate, eventHostAudioUpdate, 0);
|
2012-11-01 15:19:01 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void __AudioInit()
|
|
|
|
{
|
2012-12-17 18:38:04 +00:00
|
|
|
mixFrequency = 44100;
|
2012-11-17 13:20:04 +00:00
|
|
|
|
2012-11-01 15:19:01 +00:00
|
|
|
eventAudioUpdate = CoreTiming::RegisterEvent("AudioUpdate", &hleAudioUpdate);
|
2012-11-17 13:20:04 +00:00
|
|
|
eventHostAudioUpdate = CoreTiming::RegisterEvent("AudioUpdateHost", &hleHostAudioUpdate);
|
2012-11-01 15:19:01 +00:00
|
|
|
|
2012-11-17 13:20:04 +00:00
|
|
|
CoreTiming::ScheduleEvent(usToCycles(audioIntervalUs), eventAudioUpdate, 0);
|
|
|
|
CoreTiming::ScheduleEvent(usToCycles(audioHostIntervalUs), eventHostAudioUpdate, 0);
|
2012-11-01 15:19:01 +00:00
|
|
|
for (int i = 0; i < 8; i++)
|
|
|
|
chans[i].clear();
|
|
|
|
}
|
|
|
|
|
|
|
|
void __AudioShutdown()
|
|
|
|
{
|
2012-11-17 13:20:04 +00:00
|
|
|
for (int i = 0; i < 8; i++)
|
|
|
|
chans[i].clear();
|
2012-11-01 15:19:01 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
u32 __AudioEnqueue(AudioChannel &chan, int chanNum, bool blocking)
|
|
|
|
{
|
|
|
|
section.lock();
|
|
|
|
if (chan.sampleAddress == 0)
|
|
|
|
return SCE_ERROR_AUDIO_NOT_OUTPUT;
|
2012-11-17 13:20:04 +00:00
|
|
|
if (chan.sampleQueue.size() > chan.sampleCount*2*chanQueueMaxSizeFactor) {
|
2012-11-01 15:19:01 +00:00
|
|
|
// Block!
|
|
|
|
if (blocking) {
|
2012-11-17 13:20:04 +00:00
|
|
|
chan.waitingThread = __KernelGetCurThread();
|
|
|
|
// WARNING: This changes currentThread so must grab waitingThread before (line above).
|
2012-11-01 15:19:01 +00:00
|
|
|
__KernelWaitCurThread(WAITTYPE_AUDIOCHANNEL, (SceUID)chanNum, 0, 0, false);
|
2012-12-17 19:15:23 +00:00
|
|
|
// Fall through to the sample queueing, don't want to lose the samples even though
|
|
|
|
// we're getting full.
|
2012-11-01 15:19:01 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2012-11-17 13:20:04 +00:00
|
|
|
chan.waitingThread = 0;
|
2012-11-01 15:19:01 +00:00
|
|
|
return SCE_ERROR_AUDIO_CHANNEL_BUSY;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (chan.format == PSP_AUDIO_FORMAT_STEREO)
|
|
|
|
{
|
|
|
|
for (u32 i = 0; i < chan.sampleCount * 2; i++)
|
|
|
|
{
|
|
|
|
chan.sampleQueue.push((s16)Memory::Read_U16(chan.sampleAddress + 2 * i));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (chan.format == PSP_AUDIO_FORMAT_MONO)
|
|
|
|
{
|
|
|
|
for (u32 i = 0; i < chan.sampleCount; i++)
|
|
|
|
{
|
|
|
|
// Expand to stereo
|
|
|
|
s16 sample = (s16)Memory::Read_U16(chan.sampleAddress + 2 * i);
|
|
|
|
chan.sampleQueue.push(sample);
|
|
|
|
chan.sampleQueue.push(sample);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
section.unlock();
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2012-11-17 13:20:04 +00:00
|
|
|
// Mix samples from the various audio channels into a single sample queue.
|
|
|
|
// This single sample queue is where __AudioMix should read from. If the sample queue is full, we should
|
|
|
|
// just sleep the main emulator thread a little.
|
|
|
|
void __AudioUpdate()
|
2012-11-01 15:19:01 +00:00
|
|
|
{
|
2012-11-23 09:33:19 +00:00
|
|
|
// Audio throttle doesn't really work on the PSP since the mixing intervals are so closely tied
|
|
|
|
// to the CPU. Much better to throttle the frame rate on frame display and just throw away audio
|
|
|
|
// if the buffer somehow gets full.
|
2012-11-01 15:19:01 +00:00
|
|
|
|
2012-11-17 13:20:04 +00:00
|
|
|
s32 mixBuffer[hwBlockSize * 2];
|
|
|
|
memset(mixBuffer, 0, sizeof(mixBuffer));
|
2012-11-01 15:19:01 +00:00
|
|
|
|
2012-12-23 16:55:54 +00:00
|
|
|
for (int i = 0; i < PSP_AUDIO_CHANNEL_MAX; i++)
|
2012-11-01 15:19:01 +00:00
|
|
|
{
|
2012-11-17 13:20:04 +00:00
|
|
|
if (!chans[i].reserved)
|
|
|
|
continue;
|
|
|
|
if (!chans[i].sampleQueue.size()) {
|
2012-12-18 10:40:14 +00:00
|
|
|
// ERROR_LOG(HLE, "No queued samples, skipping channel %i", i);
|
2012-11-17 13:20:04 +00:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (int s = 0; s < hwBlockSize; s++)
|
2012-11-01 15:19:01 +00:00
|
|
|
{
|
|
|
|
if (chans[i].sampleQueue.size() >= 2)
|
|
|
|
{
|
2012-11-17 13:20:04 +00:00
|
|
|
s16 sampleL = chans[i].sampleQueue.front();
|
2012-11-01 15:19:01 +00:00
|
|
|
chans[i].sampleQueue.pop();
|
2012-12-17 18:38:04 +00:00
|
|
|
s16 sampleR = chans[i].sampleQueue.front();
|
2012-11-01 15:19:01 +00:00
|
|
|
chans[i].sampleQueue.pop();
|
2012-11-17 13:20:04 +00:00
|
|
|
mixBuffer[s * 2] += sampleL;
|
|
|
|
mixBuffer[s * 2 + 1] += sampleR;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
ERROR_LOG(HLE, "channel %i buffer underrun at %i of %i", i, s, hwBlockSize);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2012-11-01 15:19:01 +00:00
|
|
|
|
2012-11-17 13:20:04 +00:00
|
|
|
if (chans[i].sampleQueue.size() < chans[i].sampleCount * 2 * chanQueueMinSizeFactor)
|
|
|
|
{
|
|
|
|
// Ask the thread to send more samples until next time, queue is being drained.
|
|
|
|
if (chans[i].waitingThread) {
|
|
|
|
SceUID waitingThread = chans[i].waitingThread;
|
|
|
|
chans[i].waitingThread = 0;
|
|
|
|
// DEBUG_LOG(HLE, "Woke thread %i for some buffer filling", waitingThread);
|
|
|
|
__KernelResumeThreadFromWait(waitingThread);
|
2012-11-01 15:19:01 +00:00
|
|
|
}
|
|
|
|
}
|
2012-11-17 13:20:04 +00:00
|
|
|
}
|
|
|
|
|
2012-11-23 09:33:19 +00:00
|
|
|
section.lock();
|
|
|
|
|
2012-12-18 10:40:14 +00:00
|
|
|
if (g_Config.bEnableSound) {
|
|
|
|
if (outAudioQueue.room() >= hwBlockSize * 2) {
|
|
|
|
// Push the mixed samples onto the output audio queue.
|
|
|
|
for (int i = 0; i < hwBlockSize; i++) {
|
|
|
|
s32 sampleL = mixBuffer[i * 2] >> 2; // TODO - what factor?
|
|
|
|
s32 sampleR = mixBuffer[i * 2 + 1] >> 2;
|
2012-11-17 13:20:04 +00:00
|
|
|
|
2012-12-18 10:40:14 +00:00
|
|
|
outAudioQueue.push((s16)sampleL);
|
|
|
|
outAudioQueue.push((s16)sampleR);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// This happens quite a lot. There's still something slightly off
|
|
|
|
// about the amount of audio we produce.
|
2012-12-21 22:02:35 +00:00
|
|
|
DEBUG_LOG(HLE, "Audio outbuffer overrun! room = %i / %i", outAudioQueue.room(), (u32)outAudioQueue.capacity());
|
2012-11-17 13:20:04 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
section.unlock();
|
|
|
|
}
|
|
|
|
|
|
|
|
void __AudioSetOutputFrequency(int freq)
|
|
|
|
{
|
2012-12-16 20:20:16 +00:00
|
|
|
WARN_LOG(HLE, "Switching audio frequency to %i", freq);
|
2012-11-17 13:20:04 +00:00
|
|
|
mixFrequency = freq;
|
|
|
|
}
|
|
|
|
|
|
|
|
// numFrames is number of stereo frames.
|
|
|
|
int __AudioMix(short *outstereo, int numFrames)
|
|
|
|
{
|
|
|
|
// TODO: if mixFrequency != the actual output frequency, resample!
|
|
|
|
|
|
|
|
section.lock();
|
|
|
|
int underrun = -1;
|
|
|
|
s16 sampleL = 0;
|
|
|
|
s16 sampleR = 0;
|
2012-11-17 18:56:28 +00:00
|
|
|
bool anythingToPlay = false;
|
|
|
|
for (int i = 0; i < numFrames; i++) {
|
2012-11-17 13:20:04 +00:00
|
|
|
if (outAudioQueue.size() >= 2)
|
2012-11-01 15:19:01 +00:00
|
|
|
{
|
2012-11-17 13:20:04 +00:00
|
|
|
sampleL = outAudioQueue.front();
|
|
|
|
outAudioQueue.pop();
|
|
|
|
sampleR = outAudioQueue.front();
|
|
|
|
outAudioQueue.pop();
|
|
|
|
outstereo[i * 2] = sampleL;
|
|
|
|
outstereo[i * 2 + 1] = sampleR;
|
2012-11-17 18:56:28 +00:00
|
|
|
anythingToPlay = true;
|
2012-11-17 13:20:04 +00:00
|
|
|
} else {
|
2012-11-23 09:33:19 +00:00
|
|
|
if (underrun == -1) underrun = i;
|
2012-11-17 13:20:04 +00:00
|
|
|
outstereo[i * 2] = sampleL; // repeat last sample, can reduce clicking
|
|
|
|
outstereo[i * 2 + 1] = sampleR; // repeat last sample, can reduce clicking
|
2012-11-01 15:19:01 +00:00
|
|
|
}
|
|
|
|
}
|
2012-11-17 18:56:28 +00:00
|
|
|
if (anythingToPlay && underrun >= 0) {
|
2012-12-18 10:40:14 +00:00
|
|
|
DEBUG_LOG(HLE, "audio out buffer UNDERRUN at %i of %i", underrun, numFrames);
|
2012-11-17 13:20:04 +00:00
|
|
|
} else {
|
|
|
|
// DEBUG_LOG(HLE, "No underrun, mixed %i samples fine", numFrames);
|
|
|
|
}
|
2012-11-01 15:19:01 +00:00
|
|
|
section.unlock();
|
2012-11-17 13:20:04 +00:00
|
|
|
return numFrames;
|
2012-11-01 15:19:01 +00:00
|
|
|
}
|