#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; }