2013-10-17 14:00:43 +00:00
|
|
|
// Copyright (c) 2013- 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
|
|
|
|
// the Free Software Foundation, version 2.0 or later versions.
|
|
|
|
|
|
|
|
// 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/.
|
|
|
|
|
|
|
|
#pragma once
|
|
|
|
|
2013-10-20 06:41:34 +00:00
|
|
|
#include <cmath>
|
|
|
|
|
2013-10-17 14:00:43 +00:00
|
|
|
#include "base/basictypes.h"
|
2014-03-23 18:45:08 +00:00
|
|
|
#include "Core/HW/MediaEngine.h"
|
|
|
|
|
|
|
|
#ifdef USE_FFMPEG
|
|
|
|
|
|
|
|
extern "C" {
|
|
|
|
#include <libavformat/avformat.h>
|
|
|
|
#include <libswresample/swresample.h>
|
|
|
|
#include <libavutil/samplefmt.h>
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif // USE_FFMPEG
|
2013-10-17 14:00:43 +00:00
|
|
|
|
|
|
|
// Wraps FFMPEG in a nice interface that's drop-in compatible with
|
|
|
|
// the old one. Decodes packet by packet - does NOT demux. That's done by
|
|
|
|
// MpegDemux. Only decodes Atrac3+, not regular Atrac3.
|
2013-10-18 12:12:10 +00:00
|
|
|
|
2013-10-17 14:00:43 +00:00
|
|
|
// Based on http://ffmpeg.org/doxygen/trunk/doc_2examples_2decoding_encoding_8c-example.html#_a13
|
|
|
|
|
2013-10-18 12:12:10 +00:00
|
|
|
// Ideally, Maxim's Atrac3+ decoder would be available as a standalone library
|
2013-10-17 14:00:43 +00:00
|
|
|
// that we could link, as that would be totally sufficient for the use case here.
|
|
|
|
// However, it will be maintained as a part of FFMPEG so that's the way we'll go
|
|
|
|
// for simplicity and sanity.
|
|
|
|
|
2014-03-23 18:45:08 +00:00
|
|
|
struct SimpleAudio {
|
|
|
|
public:
|
|
|
|
SimpleAudio(int);
|
|
|
|
SimpleAudio(u32, int);
|
|
|
|
~SimpleAudio();
|
|
|
|
|
|
|
|
bool Decode(void* inbuf, int inbytes, uint8_t *outbuf, int *outbytes);
|
|
|
|
bool IsOK() const { return codec_ != 0; }
|
2014-04-11 13:09:31 +00:00
|
|
|
int getOutSamples();
|
|
|
|
int getSourcePos();
|
2014-03-23 18:45:08 +00:00
|
|
|
|
|
|
|
u32 ctxPtr;
|
|
|
|
int audioType;
|
2014-04-11 13:09:31 +00:00
|
|
|
int outSamples; // output samples per frame
|
|
|
|
int srcPos; // source position after decode
|
2014-03-23 18:45:08 +00:00
|
|
|
|
|
|
|
private:
|
|
|
|
#ifdef USE_FFMPEG
|
|
|
|
AVFrame *frame_;
|
|
|
|
AVCodec *codec_;
|
|
|
|
AVCodecContext *codecCtx_;
|
|
|
|
SwrContext *swrCtx_;
|
|
|
|
AVCodecID audioCodecId; // AV_CODEC_ID_XXX
|
|
|
|
|
|
|
|
bool GetAudioCodecID(int audioType); // Get audioCodecId from audioType
|
|
|
|
#endif // USE_FFMPEG
|
|
|
|
};
|
|
|
|
|
2014-04-11 13:09:31 +00:00
|
|
|
// audioType
|
2014-03-23 18:45:08 +00:00
|
|
|
enum {
|
|
|
|
PSP_CODEC_AT3PLUS = 0x00001000,
|
|
|
|
PSP_CODEC_AT3 = 0x00001001,
|
|
|
|
PSP_CODEC_MP3 = 0x00001002,
|
|
|
|
PSP_CODEC_AAC = 0x00001003,
|
|
|
|
};
|
|
|
|
|
|
|
|
static const char *const codecNames[4] = {
|
|
|
|
"AT3+", "AT3", "MP3", "AAC",
|
|
|
|
};
|
|
|
|
|
|
|
|
void AudioClose(SimpleAudio **ctx);
|
|
|
|
static const char *GetCodecName(int codec) {
|
|
|
|
if (codec >= PSP_CODEC_AT3PLUS && codec <= PSP_CODEC_AAC) {
|
|
|
|
return codecNames[codec - PSP_CODEC_AT3PLUS];
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
return "(unk)";
|
|
|
|
}
|
|
|
|
};
|
|
|
|
bool isValidCodec(int codec);
|
2014-04-11 13:09:31 +00:00
|
|
|
|
|
|
|
|
|
|
|
class AuCtx{
|
|
|
|
public:
|
|
|
|
// Au source informations
|
|
|
|
u64 startPos;
|
|
|
|
u64 endPos;
|
|
|
|
u32 AuBuf;
|
|
|
|
u32 AuBufSize;
|
|
|
|
u32 PCMBuf;
|
|
|
|
u32 PCMBufSize;
|
|
|
|
int freq;
|
|
|
|
int BitRate;
|
|
|
|
int SamplingRate;
|
|
|
|
int Channels;
|
|
|
|
|
|
|
|
// audio settings
|
|
|
|
u32 SumDecodedSamples;
|
|
|
|
int LoopNum;
|
|
|
|
u32 MaxOutputSample;
|
|
|
|
|
|
|
|
// Au decoder
|
|
|
|
SimpleAudio *decoder;
|
|
|
|
|
|
|
|
// Au type
|
|
|
|
int audioType;
|
|
|
|
|
|
|
|
// buffers informations
|
|
|
|
int AuBufAvailable; // the available buffer of AuBuf to be able to recharge data
|
|
|
|
int readPos; // read position in audio source file
|
|
|
|
int writePos; // write position in AuBuf, i.e. the size of bytes decoded in AuBuf.
|
|
|
|
|
|
|
|
AuCtx() :decoder(NULL){};
|
|
|
|
~AuCtx(){
|
|
|
|
if (decoder){
|
|
|
|
AudioClose(&decoder);
|
|
|
|
decoder = NULL;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
u32 sceAuExit();
|
|
|
|
u32 sceAuDecode(u32 pcmAddr);
|
|
|
|
u32 sceAuGetLoopNum();
|
|
|
|
u32 sceAuSetLoopNum(int loop);
|
|
|
|
int sceAuCheckStreamDataNeeded();
|
|
|
|
u32 sceAuNotifyAddStreamData(int size);
|
|
|
|
u32 sceAuGetInfoToAddStreamData(u32 buff, u32 size, u32 srcPos);
|
|
|
|
u32 sceAuGetMaxOutputSample();
|
|
|
|
u32 sceAuGetSumDecodedSample();
|
|
|
|
u32 sceAuResetPlayPosition();
|
|
|
|
int sceAuGetChannelNum();
|
|
|
|
int sceAuGetBitRate();
|
|
|
|
int sceAuGetSamplingRate();
|
|
|
|
u32 sceAuResetPlayPositionByFrame(int position);
|
|
|
|
|
|
|
|
void DoState(PointerWrap &p) {
|
|
|
|
auto s = p.Section("AuContext", 1);
|
|
|
|
if (!s)
|
|
|
|
return;
|
|
|
|
|
|
|
|
p.Do(startPos);
|
|
|
|
p.Do(endPos);
|
|
|
|
p.Do(AuBuf);
|
|
|
|
p.Do(AuBufSize);
|
|
|
|
p.Do(PCMBuf);
|
|
|
|
p.Do(PCMBufSize);
|
|
|
|
p.Do(freq);
|
|
|
|
p.Do(SumDecodedSamples);
|
|
|
|
p.Do(LoopNum);
|
|
|
|
p.Do(Channels);
|
|
|
|
p.Do(MaxOutputSample);
|
|
|
|
p.Do(AuBufAvailable);
|
|
|
|
p.Do(readPos);
|
|
|
|
p.Do(writePos);
|
|
|
|
p.Do(audioType);
|
|
|
|
p.Do(BitRate);
|
|
|
|
p.Do(SamplingRate);
|
|
|
|
|
|
|
|
if (p.mode == p.MODE_READ){
|
|
|
|
decoder = new SimpleAudio(audioType);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|