commit
aee205101b
|
@ -22,7 +22,7 @@ extern "C" {
|
||||||
namespace xe {
|
namespace xe {
|
||||||
namespace apu {
|
namespace apu {
|
||||||
|
|
||||||
AudioDecoder::AudioDecoder() : offset_(0), codec_(nullptr), context_(nullptr),
|
AudioDecoder::AudioDecoder() : codec_(nullptr), context_(nullptr),
|
||||||
decoded_frame_(nullptr), packet_(nullptr) {}
|
decoded_frame_(nullptr), packet_(nullptr) {}
|
||||||
|
|
||||||
AudioDecoder::~AudioDecoder() {
|
AudioDecoder::~AudioDecoder() {
|
||||||
|
@ -100,7 +100,7 @@ int AudioDecoder::Initialize(int bits) {
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
int AudioDecoder::PreparePacket(uint8_t* input, size_t size,
|
int AudioDecoder::PreparePacket(uint8_t* input, size_t seq_offset, size_t size,
|
||||||
int sample_rate, int channels) {
|
int sample_rate, int channels) {
|
||||||
if (size != XMAContextData::kBytesPerBlock) {
|
if (size != XMAContextData::kBytesPerBlock) {
|
||||||
// Invalid packet size!
|
// Invalid packet size!
|
||||||
|
@ -115,9 +115,8 @@ int AudioDecoder::PreparePacket(uint8_t* input, size_t size,
|
||||||
std::memcpy(packet_data_, input, size);
|
std::memcpy(packet_data_, input, size);
|
||||||
|
|
||||||
// Modify the packet header so it's WMAPro compatible
|
// Modify the packet header so it's WMAPro compatible
|
||||||
*((int *)packet_data_) = (((offset_ & 0x7800) | 0x400) >> 7) |
|
*((int *)packet_data_) = (((seq_offset & 0x7800) | 0x400) >> 7) |
|
||||||
(*((int*)packet_data_) & 0xFFFEFF08);
|
(*((int*)packet_data_) & 0xFFFEFF08);
|
||||||
offset_ += XMAContextData::kBytesPerBlock; // Sequence number
|
|
||||||
|
|
||||||
packet_->data = packet_data_;
|
packet_->data = packet_data_;
|
||||||
packet_->size = XMAContextData::kBytesPerBlock;
|
packet_->size = XMAContextData::kBytesPerBlock;
|
||||||
|
@ -201,15 +200,12 @@ int AudioDecoder::DecodePacket(uint8_t* output, size_t output_offset, size_t out
|
||||||
for (int i = 0; i < decoded_frame_->nb_samples; i++) {
|
for (int i = 0; i < decoded_frame_->nb_samples; i++) {
|
||||||
// Raw sample should be within [-1, 1]
|
// Raw sample should be within [-1, 1]
|
||||||
float fRawSample = sample_array[i];
|
float fRawSample = sample_array[i];
|
||||||
float fScaledSample = fRawSample * (1 << (bits_ - 1));
|
|
||||||
|
|
||||||
// Clamp the sample in range
|
// Clamp it, just in case.
|
||||||
int64_t range = (1 << (bits_ - 1));
|
fRawSample = std::min( 1.f, fRawSample);
|
||||||
if (fScaledSample > (range - 1)) {
|
fRawSample = std::max(-1.f, fRawSample);
|
||||||
fScaledSample = (float)range;
|
|
||||||
} else if (fScaledSample < (-range + 1)) {
|
float fScaledSample = fRawSample * ((1 << bits_) - 1);
|
||||||
fScaledSample = (float)-range;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Convert the sample and output it in big endian
|
// Convert the sample and output it in big endian
|
||||||
int sample = (int)fScaledSample;
|
int sample = (int)fScaledSample;
|
||||||
|
|
|
@ -38,7 +38,8 @@ class AudioDecoder {
|
||||||
|
|
||||||
int Initialize(int bits);
|
int Initialize(int bits);
|
||||||
|
|
||||||
int PreparePacket(uint8_t* input, size_t size, int sample_rate, int channels);
|
int PreparePacket(uint8_t* input, size_t seq_offset, size_t size,
|
||||||
|
int sample_rate, int channels);
|
||||||
void DiscardPacket();
|
void DiscardPacket();
|
||||||
|
|
||||||
int DecodePacket(uint8_t* output, size_t offset, size_t size);
|
int DecodePacket(uint8_t* output, size_t offset, size_t size);
|
||||||
|
@ -54,7 +55,6 @@ class AudioDecoder {
|
||||||
size_t current_frame_pos_;
|
size_t current_frame_pos_;
|
||||||
uint8_t* current_frame_;
|
uint8_t* current_frame_;
|
||||||
uint32_t frame_samples_size_;
|
uint32_t frame_samples_size_;
|
||||||
int offset_;
|
|
||||||
|
|
||||||
uint8_t packet_data_[XMAContextData::kBytesPerBlock];
|
uint8_t packet_data_[XMAContextData::kBytesPerBlock];
|
||||||
};
|
};
|
||||||
|
|
|
@ -91,15 +91,18 @@ X_STATUS AudioSystem::Setup() {
|
||||||
for (int i = kXmaContextCount - 1; i >= 0; --i) {
|
for (int i = kXmaContextCount - 1; i >= 0; --i) {
|
||||||
uint32_t ptr = registers_.xma_context_array_ptr + i * kXmaContextSize;
|
uint32_t ptr = registers_.xma_context_array_ptr + i * kXmaContextSize;
|
||||||
|
|
||||||
|
XMAContext& context = xma_context_array_[i];
|
||||||
|
|
||||||
// Initialize it
|
// Initialize it
|
||||||
xma_context_array_[i].guest_ptr = ptr;
|
context.guest_ptr = ptr;
|
||||||
xma_context_array_[i].in_use = false;
|
context.in_use = false;
|
||||||
|
context.kicked = false;
|
||||||
|
|
||||||
// Create a new decoder per context
|
// Create a new decoder per context
|
||||||
// Needed because some data needs to be persisted across calls
|
// Needed because some data needs to be persisted across calls
|
||||||
// TODO: Need to destroy this on class destruction
|
// TODO: Need to destroy this on class destruction
|
||||||
xma_context_array_[i].decoder = new AudioDecoder();
|
context.decoder = new AudioDecoder();
|
||||||
xma_context_array_[i].decoder->Initialize(16);
|
context.decoder->Initialize(16);
|
||||||
}
|
}
|
||||||
registers_.next_context = 1;
|
registers_.next_context = 1;
|
||||||
|
|
||||||
|
@ -192,12 +195,15 @@ void AudioSystem::DecoderThreadMain() {
|
||||||
// Okay, let's loop through XMA contexts to find ones we need to decode!
|
// Okay, let's loop through XMA contexts to find ones we need to decode!
|
||||||
for (uint32_t n = 0; n < kXmaContextCount; n++) {
|
for (uint32_t n = 0; n < kXmaContextCount; n++) {
|
||||||
XMAContext& context = xma_context_array_[n];
|
XMAContext& context = xma_context_array_[n];
|
||||||
if (context.in_use) {
|
if (context.in_use && context.kicked) {
|
||||||
context.lock.lock();
|
context.lock.lock();
|
||||||
|
context.kicked = false;
|
||||||
|
|
||||||
auto context_ptr = memory()->TranslateVirtual(context.guest_ptr);
|
auto context_ptr = memory()->TranslateVirtual(context.guest_ptr);
|
||||||
XMAContextData data(context_ptr);
|
XMAContextData data(context_ptr);
|
||||||
ProcessXmaContext(context, data);
|
ProcessXmaContext(context, data);
|
||||||
data.Store(context_ptr);
|
data.Store(context_ptr);
|
||||||
|
|
||||||
context.lock.unlock();
|
context.lock.unlock();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -324,11 +330,6 @@ void AudioSystem::UnregisterClient(size_t index) {
|
||||||
}
|
}
|
||||||
|
|
||||||
void AudioSystem::ProcessXmaContext(XMAContext& context, XMAContextData& data) {
|
void AudioSystem::ProcessXmaContext(XMAContext& context, XMAContextData& data) {
|
||||||
if (!context.in_use) {
|
|
||||||
// Skip unused contexts.
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
SCOPE_profile_cpu_f("apu");
|
SCOPE_profile_cpu_f("apu");
|
||||||
|
|
||||||
// Translate pointers for future use.
|
// Translate pointers for future use.
|
||||||
|
@ -340,6 +341,8 @@ void AudioSystem::ProcessXmaContext(XMAContext& context, XMAContextData& data) {
|
||||||
: nullptr;
|
: nullptr;
|
||||||
uint8_t* out = memory()->TranslatePhysical(data.output_buffer_ptr);
|
uint8_t* out = memory()->TranslatePhysical(data.output_buffer_ptr);
|
||||||
|
|
||||||
|
assert(!in1);
|
||||||
|
|
||||||
// What I see:
|
// What I see:
|
||||||
// XMA outputs 2 bytes per sample
|
// XMA outputs 2 bytes per sample
|
||||||
// 512 samples per frame (128 per subframe)
|
// 512 samples per frame (128 per subframe)
|
||||||
|
@ -362,7 +365,7 @@ void AudioSystem::ProcessXmaContext(XMAContext& context, XMAContextData& data) {
|
||||||
while (data.output_buffer_valid) {
|
while (data.output_buffer_valid) {
|
||||||
// Check the output buffer - we cannot decode anything else if it's
|
// Check the output buffer - we cannot decode anything else if it's
|
||||||
// unavailable.
|
// unavailable.
|
||||||
// Output buffers are in frames.
|
// Output buffers are in raw PCM samples, 256 bytes per block.
|
||||||
uint32_t output_size_bytes = data.output_buffer_block_count * 256;
|
uint32_t output_size_bytes = data.output_buffer_block_count * 256;
|
||||||
uint32_t output_offset_bytes = data.output_buffer_write_offset * 256;
|
uint32_t output_offset_bytes = data.output_buffer_write_offset * 256;
|
||||||
uint32_t output_remaining_bytes = output_size_bytes - output_offset_bytes;
|
uint32_t output_remaining_bytes = output_size_bytes - output_offset_bytes;
|
||||||
|
@ -423,24 +426,39 @@ void AudioSystem::ProcessXmaContext(XMAContext& context, XMAContextData& data) {
|
||||||
}
|
}
|
||||||
int channels = data.is_stereo ? 2 : 1;
|
int channels = data.is_stereo ? 2 : 1;
|
||||||
|
|
||||||
// See if we've finished with the input
|
// See if we've finished with the input.
|
||||||
// TODO - Probably need to move this, I think it might skip the very
|
// Block count is in packets, so expand by packet size.
|
||||||
// last packet (see the call to PreparePacket)
|
uint32_t input_size_0_bytes =
|
||||||
// Block count is in frames, so expand by
|
(data.input_buffer_0_block_count) * 2048;
|
||||||
// samples_per_frame*bytes_per_sample*bits_per_byte.
|
uint32_t input_size_1_bytes =
|
||||||
uint32_t input_size_bytes =
|
(data.input_buffer_1_block_count) * 2048;
|
||||||
(data.input_buffer_0_block_count + data.input_buffer_1_block_count) *
|
|
||||||
2048;
|
// Total input size
|
||||||
|
uint32_t input_size_bytes = input_size_0_bytes + input_size_1_bytes;
|
||||||
|
|
||||||
// Input read offset is in bits. Typically starts at 32 (4 bytes).
|
// Input read offset is in bits. Typically starts at 32 (4 bytes).
|
||||||
uint32_t input_offset_bytes =
|
// "Sequence" offset - used internally for WMA Pro decoder.
|
||||||
|
// Just the read offset.
|
||||||
|
uint32_t seq_offset_bytes =
|
||||||
(data.input_buffer_read_offset & ~0x7FF) / 8;
|
(data.input_buffer_read_offset & ~0x7FF) / 8;
|
||||||
if (input_offset_bytes < input_size_bytes) {
|
|
||||||
|
if (seq_offset_bytes < input_size_bytes) {
|
||||||
|
// Setup input offset and input buffer.
|
||||||
|
uint32_t input_offset_bytes = seq_offset_bytes;
|
||||||
|
auto input_buffer = in0;
|
||||||
|
|
||||||
|
if (seq_offset_bytes >= input_size_0_bytes) {
|
||||||
|
// Size overlap, select input buffer 1.
|
||||||
|
input_offset_bytes -= input_size_0_bytes;
|
||||||
|
input_buffer = in1;
|
||||||
|
}
|
||||||
|
|
||||||
// Still have data to read.
|
// Still have data to read.
|
||||||
// TODO: Select input buffer 1 if necessary.
|
auto packet = input_buffer + input_offset_bytes;
|
||||||
auto packet = in0 + input_offset_bytes;
|
context.decoder->PreparePacket(packet, seq_offset_bytes, 2048,
|
||||||
context.decoder->PreparePacket(packet, 2048, sample_rate, channels);
|
sample_rate, channels);
|
||||||
data.input_buffer_read_offset += 2048 * 8;
|
data.input_buffer_read_offset += 2048 * 8;
|
||||||
if (input_offset_bytes + 2048 >= input_size_bytes) {
|
if (seq_offset_bytes + 2048 >= input_size_bytes) {
|
||||||
// Used the last of the data.
|
// Used the last of the data.
|
||||||
data.input_buffer_0_valid = 0;
|
data.input_buffer_0_valid = 0;
|
||||||
data.input_buffer_1_valid = 0;
|
data.input_buffer_1_valid = 0;
|
||||||
|
@ -515,8 +533,9 @@ void AudioSystem::WriteRegister(uint32_t addr, uint64_t value) {
|
||||||
|
|
||||||
XELOGAPU(
|
XELOGAPU(
|
||||||
"AudioSystem: kicking context %d (%d/%d bytes)", context_id,
|
"AudioSystem: kicking context %d (%d/%d bytes)", context_id,
|
||||||
data.input_buffer_read_offset / 8,
|
(data.input_buffer_read_offset & ~0x7FF) / 8,
|
||||||
data.input_buffer_0_block_count * XMAContextData::kBytesPerBlock);
|
(data.input_buffer_0_block_count + data.input_buffer_1_block_count)
|
||||||
|
* XMAContextData::kBytesPerBlock);
|
||||||
|
|
||||||
// Reset valid flags so our audio decoder knows to process this one.
|
// Reset valid flags so our audio decoder knows to process this one.
|
||||||
data.input_buffer_0_valid = data.input_buffer_0_ptr != 0;
|
data.input_buffer_0_valid = data.input_buffer_0_ptr != 0;
|
||||||
|
@ -524,6 +543,8 @@ void AudioSystem::WriteRegister(uint32_t addr, uint64_t value) {
|
||||||
data.output_buffer_write_offset = 0;
|
data.output_buffer_write_offset = 0;
|
||||||
|
|
||||||
data.Store(context_ptr);
|
data.Store(context_ptr);
|
||||||
|
|
||||||
|
context.kicked = true;
|
||||||
context.lock.unlock();
|
context.lock.unlock();
|
||||||
}
|
}
|
||||||
value >>= 1;
|
value >>= 1;
|
||||||
|
|
|
@ -200,6 +200,7 @@ class AudioSystem {
|
||||||
uint32_t guest_ptr;
|
uint32_t guest_ptr;
|
||||||
xe::mutex lock;
|
xe::mutex lock;
|
||||||
bool in_use;
|
bool in_use;
|
||||||
|
bool kicked;
|
||||||
|
|
||||||
AudioDecoder* decoder;
|
AudioDecoder* decoder;
|
||||||
};
|
};
|
||||||
|
|
Loading…
Reference in New Issue