mirror of
https://github.com/vgmstream/vgmstream.git
synced 2024-12-19 09:55:53 +01:00
686d77debc
Separate header variations since every other dev is going to make their own, it was getting kind of messy.
121 lines
3.9 KiB
C
121 lines
3.9 KiB
C
#include "meta.h"
|
|
#include "../coding/coding.h"
|
|
#include "../layout/layout.h"
|
|
|
|
static STREAMFILE* setup_opus_ppp_streamfile(STREAMFILE *streamFile, off_t subfile_offset, size_t subfile_size, const char* fake_ext);
|
|
|
|
/* .AT9 Opus - from Penny-Punching Princess (Switch) */
|
|
VGMSTREAM * init_vgmstream_opus_ppp(STREAMFILE *streamFile) {
|
|
VGMSTREAM * vgmstream = NULL;
|
|
off_t segment_offset;
|
|
int loop_flag, channel_count;
|
|
int i;
|
|
|
|
segmented_layout_data *data = NULL;
|
|
int segment_count, loop_start_segment = 0, loop_end_segment = 0;
|
|
int num_samples = 0, loop_start_sample = 0, loop_end_sample = 0;
|
|
|
|
|
|
/* checks */
|
|
if (!check_extensions(streamFile, "at9"))
|
|
goto fail;
|
|
if (read_32bitBE(0x00,streamFile) != 0x09000000) /* file type? DSPs had 08 */
|
|
goto fail;
|
|
if (read_32bitLE(0x04,streamFile) + 0x1c != get_streamfile_size(streamFile))
|
|
goto fail;
|
|
/* 0x08(2): sample rate, 0x0a(2): loop flag?, 0x0c: num_samples (slightly smaller than added samples) */
|
|
|
|
segment_count = 3; /* intro/loop/end */
|
|
loop_start_segment = 1;
|
|
loop_end_segment = 1;
|
|
loop_flag = (segment_count > 0);
|
|
|
|
/* init layout */
|
|
data = init_layout_segmented(segment_count);
|
|
if (!data) goto fail;
|
|
|
|
/* open each segment subfile */
|
|
segment_offset = 0x1c;
|
|
for (i = 0; i < segment_count; i++) {
|
|
STREAMFILE* temp_streamFile;
|
|
size_t segment_size = read_32bitLE(0x10+0x04*i,streamFile);
|
|
|
|
if (!segment_size)
|
|
goto fail;
|
|
|
|
temp_streamFile = setup_opus_ppp_streamfile(streamFile, segment_offset,segment_size, "opus");
|
|
if (!temp_streamFile) goto fail;
|
|
|
|
data->segments[i] = init_vgmstream_opus_std(temp_streamFile);
|
|
close_streamfile(temp_streamFile);
|
|
if (!data->segments[i]) goto fail;
|
|
|
|
segment_offset += segment_size;
|
|
|
|
//todo there are some trailing samples that must be removed for smooth loops, start skip seems ok
|
|
data->segments[i]->num_samples -= 374; //not correct for all files, no idea how to calculate
|
|
|
|
/* get looping and samples */
|
|
if (loop_flag && loop_start_segment == i)
|
|
loop_start_sample = num_samples;
|
|
|
|
num_samples += data->segments[i]->num_samples;
|
|
|
|
if (loop_flag && loop_end_segment == i)
|
|
loop_end_sample = num_samples;
|
|
}
|
|
|
|
/* setup segmented VGMSTREAMs */
|
|
if (!setup_layout_segmented(data))
|
|
goto fail;
|
|
|
|
|
|
channel_count = data->segments[0]->channels;
|
|
|
|
/* build the VGMSTREAM */
|
|
vgmstream = allocate_vgmstream(channel_count,loop_flag);
|
|
if (!vgmstream) goto fail;
|
|
|
|
vgmstream->sample_rate = (uint16_t)read_16bitLE(0x08,streamFile);
|
|
vgmstream->num_samples = num_samples;
|
|
vgmstream->loop_start_sample = loop_start_sample;
|
|
vgmstream->loop_end_sample = loop_end_sample;
|
|
|
|
vgmstream->meta_type = meta_OPUS_PPP;
|
|
vgmstream->coding_type = data->segments[0]->coding_type;
|
|
vgmstream->layout_type = layout_segmented;
|
|
|
|
vgmstream->layout_data = data;
|
|
data->loop_segment = loop_start_segment;
|
|
|
|
return vgmstream;
|
|
|
|
fail:
|
|
close_vgmstream(vgmstream);
|
|
free_layout_segmented(data);
|
|
return NULL;
|
|
}
|
|
|
|
static STREAMFILE* setup_opus_ppp_streamfile(STREAMFILE *streamFile, off_t subfile_offset, size_t subfile_size, const char* fake_ext) {
|
|
STREAMFILE *temp_streamFile = NULL, *new_streamFile = NULL;
|
|
|
|
/* setup subfile */
|
|
new_streamFile = open_wrap_streamfile(streamFile);
|
|
if (!new_streamFile) goto fail;
|
|
temp_streamFile = new_streamFile;
|
|
|
|
new_streamFile = open_clamp_streamfile(temp_streamFile, subfile_offset,subfile_size);
|
|
if (!new_streamFile) goto fail;
|
|
temp_streamFile = new_streamFile;
|
|
|
|
new_streamFile = open_fakename_streamfile(temp_streamFile, NULL,fake_ext);
|
|
if (!new_streamFile) goto fail;
|
|
temp_streamFile = new_streamFile;
|
|
|
|
return temp_streamFile;
|
|
|
|
fail:
|
|
close_streamfile(temp_streamFile);
|
|
return NULL;
|
|
}
|