#include "vorbis_custom_decoder.h" #ifdef VGM_USE_VORBIS #include /* **************************************************************************** */ /* DEFS */ /* **************************************************************************** */ static int get_page_info(STREAMFILE *streamFile, off_t page_offset, off_t *out_packet_offset, size_t *out_packet_size, int *out_page_packets, int target_packet); static int build_header(uint8_t * buf, size_t bufsize, STREAMFILE *streamFile, off_t packet_offset, size_t packet_size); /* **************************************************************************** */ /* EXTERNAL API */ /* **************************************************************************** */ /** * SK just replaces the id 0x4F676753 ("OggS") by 0x11534B10 (\11"SK"\10), and the word "vorbis" by "SK" * in init packets (for obfuscation, surely). So essentially we are parsing regular Ogg here. * * A simpler way to implement this would be in ogg_vorbis_file with read callbacks (pretend this is proof of concept). */ int vorbis_custom_setup_init_sk(STREAMFILE *streamFile, off_t start_offset, vorbis_custom_codec_data *data) { off_t offset = start_offset; off_t id_offset = 0, comment_offset = 0, setup_offset = 0; size_t id_size = 0, comment_size = 0, setup_size = 0; int page_packets; /* rebuild header packets, they are standard except the "vorbis" keyword is replaced by "SK" */ /* first page has the id packet */ if (!get_page_info(streamFile, offset, &id_offset, &id_size, &page_packets, 0)) goto fail; if (page_packets != 1) goto fail; offset = id_offset + id_size; /* second page has the comment and setup packets */ if (!get_page_info(streamFile, offset, &comment_offset, &comment_size, &page_packets, 0)) goto fail; if (page_packets != 2) goto fail; if (!get_page_info(streamFile, offset, &setup_offset, &setup_size, &page_packets, 1)) goto fail; if (page_packets != 2) goto fail; offset = comment_offset + comment_size + setup_size; /* init with all offsets found */ data->op.bytes = build_header(data->buffer, data->buffer_size, streamFile, id_offset, id_size); if (!data->op.bytes) goto fail; if (vorbis_synthesis_headerin(&data->vi, &data->vc, &data->op) != 0) goto fail; /* parse identification header */ data->op.bytes = build_header(data->buffer, data->buffer_size, streamFile, comment_offset, comment_size); if (!data->op.bytes) goto fail; if (vorbis_synthesis_headerin(&data->vi, &data->vc, &data->op) !=0 ) goto fail; /* parse comment header */ data->op.bytes = build_header(data->buffer, data->buffer_size, streamFile, setup_offset, setup_size); if (!data->op.bytes) goto fail; if (vorbis_synthesis_headerin(&data->vi, &data->vc, &data->op) != 0) goto fail; /* parse setup header */ /* data starts after triad */ data->config.data_start_offset = offset; return 1; fail: VGM_LOG("SK Vorbis: failed to setup init packets @ around 0x%08lx\n", offset); return 0; } int vorbis_custom_parse_packet_sk(VGMSTREAMCHANNEL *stream, vorbis_custom_codec_data *data) { off_t packet_offset = 0; size_t packet_size = 0; int page_packets; int res; /* read OggS/SK page and get current packet */ res = get_page_info(stream->streamfile, stream->offset, &packet_offset, &packet_size, &page_packets, data->current_packet); data->current_packet++; if (!res || packet_size > data->buffer_size) { VGM_LOG("FSB Vorbis: wrong packet (0x%x) @ %lx\n", packet_size, stream->offset); goto fail; } /* read raw block */ data->op.bytes = read_streamfile(data->buffer, packet_offset, packet_size, stream->streamfile); if (data->op.bytes != packet_size) { VGM_LOG("SK Vorbis: read error, 0x%x packet size vs 0x%lx bytes @ %lx\n", packet_size, data->op.bytes, packet_offset); goto fail; /* wrong packet? */ } if (data->current_packet >= page_packets) { /* processed all packets in page, go to next */ if (!get_page_info(stream->streamfile, stream->offset, &packet_offset, &packet_size, &page_packets, -1)) goto fail; stream->offset = packet_offset + packet_size; data->current_packet = 0; } return 1; fail: VGM_LOG("SK Vorbis: failed to parse packet @ around 0x%08lx\n", stream->offset); return 0; } /* **************************************************************************** */ /* INTERNAL HELPERS */ /* **************************************************************************** */ /** * Get packet info from an Ogg page, from segment/packet N (-1 = all segments) * * Page format: * 0x00(4): capture pattern ("OggS") * 0x01(1): stream structure version * 0x05(1): header type flag * 0x06(8): absolute granule position * 0x0e(4): stream serial number * 0x12(4): page sequence number * 0x16(4): page checksum * 0x1a(1): page segments (total bytes in segment table) * 0x1b(n): segment table (N bytes, 1 packet is sum of sizes until != 0xFF) * 0x--(n): data * Reference: https://xiph.org/ogg/doc/framing.html */ static int get_page_info(STREAMFILE *streamFile, off_t page_offset, off_t *out_packet_offset, size_t *out_packet_size, int *out_page_packets, int target_packet) { off_t table_offset, current_packet_offset, target_packet_offset = 0; size_t total_packets_size = 0, current_packet_size = 0, target_packet_size = 0; int page_packets = 0; uint8_t segments; int i; if (read_32bitBE(page_offset+0x00, streamFile) != 0x11534B10) /* \11"SK"\10 */ goto fail; /* not a valid page */ /* No point on validating other stuff, but they look legal enough (CRC too it seems) */ segments = (uint8_t)read_8bit(page_offset+0x1a, streamFile); table_offset = page_offset + 0x1b; current_packet_offset = page_offset + 0x1b + segments; /* first packet starts after segments */ /* process segments */ for (i = 0; i < segments; i++) { uint8_t segment_size = (uint8_t)read_8bit(table_offset, streamFile); total_packets_size += segment_size; current_packet_size += segment_size; table_offset += 0x01; if (segment_size != 0xFF) { /* packet complete */ page_packets++; if (target_packet+1 == page_packets) { target_packet_offset = current_packet_offset; target_packet_size = current_packet_size; } /* keep reading to fill page_packets */ current_packet_offset += current_packet_size; /* move to next packet */ current_packet_size = 0; } } /* < 0 is accepted and returns first offset and all packets sizes */ if (target_packet+1 > page_packets) goto fail; if (target_packet < 0) { target_packet_offset = page_offset + 0x1b + segments; /* first */ target_packet_size = total_packets_size; } if (out_packet_offset) *out_packet_offset = target_packet_offset; if (out_packet_size) *out_packet_size = target_packet_size; if (out_page_packets) *out_page_packets = page_packets; return 1; fail: VGM_LOG("SK Vorbis: failed to read page @ 0x%08lx\n", page_offset); return 0; } /* rebuild a "SK" header packet to a "vorbis" one */ static int build_header(uint8_t * buf, size_t bufsize, STREAMFILE *streamFile, off_t packet_offset, size_t packet_size) { int bytes; if (0x07+packet_size-0x03 > bufsize) return 0; put_8bit (buf+0x00, read_8bit(packet_offset,streamFile)); /* packet_type */ memcpy (buf+0x01, "vorbis", 6); /* id */ bytes = read_streamfile(buf+0x07,packet_offset+0x03, packet_size-0x03,streamFile); /* copy rest (all except id+"SK") */ if (packet_size-0x03 != bytes) { VGM_LOG("SK Vorbis: packet (size 0x%x) not copied correctly (bytes=%x) @ 0x%lx\n", packet_size, bytes, packet_offset); return 0; } return 0x07+packet_size-0x03; } #endif