#include "meta.h" #include /* .sli+ogg/opus - KiriKiri engine / WaveLoopManager loop points loader [Fate/Stay Night (PC), World End Economica (PC)] */ VGMSTREAM * init_vgmstream_sli_ogg(STREAMFILE *streamFile) { VGMSTREAM * vgmstream = NULL; STREAMFILE * streamData = NULL; int32_t loop_start = -1, loop_length = -1; int32_t loop_from = -1, loop_to = -1; /* checks */ if (!check_extensions(streamFile, "sli")) goto fail; { /* try with file.ogg/opus.sli=header and file.ogg/opus=data */ char basename[PATH_LIMIT]; get_streamfile_basename(streamFile,basename,PATH_LIMIT); streamData = open_streamfile_by_filename(streamFile, basename); if (!streamData) goto fail; } /* let the real initer do the parsing */ if (check_extensions(streamData, "ogg")) { /* Fate/Stay Night (PC) */ #ifdef VGM_USE_VORBIS vgmstream = init_vgmstream_ogg_vorbis(streamData); if (!vgmstream) goto fail; vgmstream->meta_type = meta_OGG_SLI; #else goto fail; #endif } else if (check_extensions(streamData, "opus")) { /* Sabbat of the Witch (PC) */ #ifdef VGM_USE_FFMPEG vgmstream = init_vgmstream_ffmpeg(streamData); if (!vgmstream) goto fail; /* FFmpeg's Opus encoder delay is borked but no need to fix: * somehow sli+opus use 0 in the OpusHead (to simplify looping?) */ vgmstream->meta_type = meta_OPUS_SLI; #else goto fail; #endif } else { goto fail; } /* find loop text */ { char line[PATH_LIMIT]; size_t bytes_read; off_t sli_offset; int line_ok; sli_offset = 0; while ((loop_start == -1 || loop_length == -1) && sli_offset < get_streamfile_size(streamFile)) { char *endptr, *foundptr; bytes_read = read_line(line, sizeof(line), sli_offset, streamFile, &line_ok); if (!line_ok) goto fail; if (memcmp("LoopStart=",line,10)==0 && line[10] != '\0') { loop_start = strtol(line+10,&endptr,10); if (*endptr != '\0') { loop_start = -1; /* if it didn't parse cleanly */ } } else if (memcmp("LoopLength=",line,11)==0 && line[11] != '\0') { loop_length = strtol(line+11,&endptr,10); if (*endptr != '\0') { loop_length = -1; /* if it didn't parse cleanly */ } } /* a completely different format (2.0?), also with .sli extension and can be handled similarly */ if ((foundptr = strstr(line,"To=")) != NULL && isdigit(foundptr[3])) { loop_to = strtol(foundptr+3,&endptr,10); if (*endptr != ';') { loop_to = -1; } } if ((foundptr = strstr(line,"From=")) != NULL && isdigit(foundptr[5])) { loop_from = strtol(foundptr+5,&endptr,10); if (*endptr != ';') { loop_from = -1; } } sli_offset += bytes_read; } } if (loop_start != -1 && loop_length != -1) { /* v1 */ vgmstream_force_loop(vgmstream,1,loop_start, loop_start+loop_length); } else if (loop_from != -1 && loop_to != -1) { /* v2 */ vgmstream_force_loop(vgmstream,1,loop_to, loop_from); } else { goto fail; /* if there's no loop points the .sli wasn't valid */ } close_streamfile(streamData); return vgmstream; fail: close_streamfile(streamData); close_vgmstream(vgmstream); return NULL; }