mirror of
https://github.com/vgmstream/vgmstream.git
synced 2025-01-20 00:12:45 +01:00
124 lines
3.3 KiB
C
124 lines
3.3 KiB
C
#include "coding.h"
|
|
#include "relic_decoder_lib.h"
|
|
|
|
//TODO: fix looping
|
|
|
|
struct relic_codec_data {
|
|
relic_handle_t* handle;
|
|
int channels;
|
|
int frame_size;
|
|
|
|
int32_t samples_discard;
|
|
int32_t samples_consumed;
|
|
int32_t samples_filled;
|
|
};
|
|
|
|
|
|
relic_codec_data* init_relic(int channels, int bitrate, int codec_rate) {
|
|
relic_codec_data* data = NULL;
|
|
|
|
data = calloc(1, sizeof(relic_codec_data));
|
|
if (!data) goto fail;
|
|
|
|
data->handle = relic_init(channels, bitrate, codec_rate);
|
|
if (!data->handle) goto fail;
|
|
|
|
data->channels = channels;
|
|
data->frame_size = relic_get_frame_size(data->handle);
|
|
|
|
return data;
|
|
fail:
|
|
free_relic(data);
|
|
return NULL;
|
|
}
|
|
|
|
static int decode_frame_next(VGMSTREAMCHANNEL* stream, relic_codec_data* data) {
|
|
int ch;
|
|
int bytes;
|
|
int ok;
|
|
uint8_t buf[RELIC_BUFFER_SIZE];
|
|
|
|
for (ch = 0; ch < data->channels; ch++) {
|
|
bytes = read_streamfile(buf, stream->offset, data->frame_size, stream->streamfile);
|
|
if (bytes != data->frame_size) goto fail;
|
|
stream->offset += data->frame_size;
|
|
|
|
ok = relic_decode_frame(data->handle, buf, ch);
|
|
if (!ok) goto fail;
|
|
}
|
|
|
|
data->samples_consumed = 0;
|
|
data->samples_filled = RELIC_SAMPLES_PER_FRAME;
|
|
return 1;
|
|
fail:
|
|
return 0;
|
|
}
|
|
|
|
void decode_relic(VGMSTREAMCHANNEL* stream, relic_codec_data* data, sample_t* outbuf, int32_t samples_to_do) {
|
|
|
|
while (samples_to_do > 0) {
|
|
|
|
if (data->samples_consumed < data->samples_filled) {
|
|
/* consume samples */
|
|
int samples_to_get = (data->samples_filled - data->samples_consumed);
|
|
|
|
if (data->samples_discard) {
|
|
/* discard samples for looping */
|
|
if (samples_to_get > data->samples_discard)
|
|
samples_to_get = data->samples_discard;
|
|
data->samples_discard -= samples_to_get;
|
|
}
|
|
else {
|
|
/* get max samples and copy */
|
|
if (samples_to_get > samples_to_do)
|
|
samples_to_get = samples_to_do;
|
|
|
|
relic_get_pcm16(data->handle, outbuf, samples_to_get, data->samples_consumed);
|
|
|
|
samples_to_do -= samples_to_get;
|
|
outbuf += samples_to_get * data->channels;
|
|
}
|
|
|
|
/* mark consumed samples */
|
|
data->samples_consumed += samples_to_get;
|
|
}
|
|
else {
|
|
int ok = decode_frame_next(stream, data);
|
|
if (!ok) goto decode_fail;
|
|
}
|
|
}
|
|
return;
|
|
|
|
decode_fail:
|
|
/* on error just put some 0 samples */
|
|
VGM_LOG("RELIC: decode fail, missing %i samples\n", samples_to_do);
|
|
memset(outbuf, 0, samples_to_do * data->channels * sizeof(sample));
|
|
}
|
|
|
|
void reset_relic(relic_codec_data* data) {
|
|
if (!data) return;
|
|
|
|
relic_reset(data->handle);
|
|
data->samples_filled = 0;
|
|
data->samples_consumed = 0;
|
|
data->samples_discard = 0;
|
|
}
|
|
|
|
void seek_relic(relic_codec_data* data, int32_t num_sample) {
|
|
if (!data) return;
|
|
|
|
reset_relic(data);
|
|
data->samples_discard = num_sample;
|
|
}
|
|
|
|
void free_relic(relic_codec_data* data) {
|
|
if (!data) return;
|
|
|
|
relic_free(data->handle);
|
|
free(data);
|
|
}
|
|
|
|
int32_t relic_bytes_to_samples(size_t bytes, int channels, int bitrate) {
|
|
return bytes / channels / (bitrate / 8) * RELIC_SAMPLES_PER_FRAME;
|
|
}
|