Add bank .rsd RedSpark [M&L Dream Team (3DS), Imabikisou (Wii)]

This commit is contained in:
bnnm 2024-01-07 16:47:46 +01:00
parent ce1b9539fb
commit 08e3ae4872
2 changed files with 335 additions and 123 deletions

View File

@ -1,149 +1,357 @@
#include "meta.h" #include "meta.h"
#include "../coding/coding.h" #include "../util/endianness.h"
#include "../util.h"
static uint32_t rotlwi(uint32_t x, uint32_t r) { #define MAX_CHANNELS 4
return (x << r) | (x >> (32-r)); typedef struct {
} bool loop_flag;
int channels;
int sample_rate;
int32_t num_samples;
int32_t loop_start;
int32_t loop_end;
int16_t coefs[MAX_CHANNELS][16];
static uint32_t find_key(uint32_t firstword) { int total_subsongs;
uint32_t expected = 0x52656453;
return firstword ^ expected;
}
/* RSD - RedSpark (MadWorld) uint32_t stream_offset;
RS3D - RedSpark (Mario & Luigi: Dream Team I fi*/ uint32_t stream_size;
VGMSTREAM * init_vgmstream_redspark(STREAMFILE *streamFile) {
VGMSTREAM * vgmstream = NULL;
char filename[PATH_LIMIT];
off_t start_offset;
int loop_flag;
int channel_count;
int dt_flag = 0;
uint32_t key;
enum {encsize = 0x1000};
uint8_t buf[encsize];
int32_t(*get_32bit)(const uint8_t *p) = NULL;
int16_t(*get_16bit)(const uint8_t *p) = NULL;
get_16bit = get_16bitBE;
get_32bit = get_32bitBE;
/* check extension, case insensitive */ bool dummy;
streamFile->get_name(streamFile,filename,sizeof(filename)); } redspark_header_t;
if (strcasecmp("rsd", filename_extension(filename))) goto fail;
/* decrypt into buffer */
{
uint32_t data;
int i;
if (read_streamfile(buf,0,encsize,streamFile)!=encsize) goto fail;
if (memcmp(&buf[0], "RedSpark", 8)) { //Check to see if already decrypted
key = find_key(get_32bitBE(&buf[0]));
data = get_32bitBE(&buf[0]) ^ key;
put_32bitBE(&buf[0], data);
key = rotlwi(key, 11);
for (i = 4; i < encsize; i += 4) { static bool parse_header(redspark_header_t* h, STREAMFILE* sf, bool is_new);
key = rotlwi(key, 3) + key;
data = get_32bitBE(&buf[i]) ^ key;
put_32bitBE(&buf[i], data);
}
}
else {
get_16bit = get_16bitLE;
get_32bit = get_32bitLE;
dt_flag = 1;
for (i = 4; i < encsize; i += 4) { /* RedSpark - Games with audio by RedSpark Ltd. (Minoru Akao) [MadWorld (Wii), Imabikisou (Wii), Mario & Luigi: Dream Team (3DS)] */
data = get_32bitBE(&buf[i]); VGMSTREAM* init_vgmstream_redspark(STREAMFILE* sf) {
put_32bitBE(&buf[i], data); VGMSTREAM* vgmstream = NULL;
} bool is_new;
}
/* checks*/
uint32_t head_id = read_u32be(0x00, sf);
if (head_id == get_id32be("RedS")) {
is_new = true; /* M&L */
}
else if (head_id > 0x15800000 && head_id < 0x1B800000) {
is_new = false; /* others: header is encrypted but in predictable ranges to fail faster (will do extra checks later) */
}
else {
return NULL;
} }
/* check header */ if (!check_extensions(sf, "rsd"))
if (memcmp(&buf[0],"RedSpark",8)) return NULL;
goto fail;
loop_flag = (buf[0x4f] != 0); redspark_header_t h = {0};
channel_count = buf[0x4e]; if (!parse_header(&h, sf, is_new))
return NULL;
/* make sure loop info is the only two cue points */
if (loop_flag && buf[0x4f] != 2) goto fail;
/* build the VGMSTREAM */ /* build the VGMSTREAM */
vgmstream = allocate_vgmstream(channel_count,loop_flag); vgmstream = allocate_vgmstream(h.channels, h.loop_flag);
if (!vgmstream) goto fail; if (!vgmstream) goto fail;
/* fill in the vital statistics */
start_offset = 0x1000;
vgmstream->channels = channel_count;
vgmstream->sample_rate = get_32bit(&buf[0x3c]);
vgmstream->coding_type = coding_NGC_DSP;
if (dt_flag)
vgmstream->num_samples = get_32bit(&buf[0x40]);
else
vgmstream->num_samples = get_32bit(&buf[0x40])*14;
if (loop_flag) {
off_t start = 0x54;
start += channel_count*8;
if (dt_flag) {
vgmstream->loop_start_sample = get_32bit(&buf[start+4]);
vgmstream->loop_end_sample = (get_32bit(&buf[start+0xc]));
}
else {
vgmstream->loop_start_sample = get_32bit(&buf[start+4])*14;
vgmstream->loop_end_sample = (get_32bit(&buf[start+0xc])+1)*14;
}
if (vgmstream->loop_end_sample > vgmstream->num_samples) {
vgmstream->loop_end_sample = vgmstream->num_samples;
}
}
if (channel_count >= 2) {
vgmstream->layout_type = layout_interleave;
vgmstream->interleave_block_size = 8;
} else {
vgmstream->layout_type = layout_none;
}
vgmstream->meta_type = meta_REDSPARK; vgmstream->meta_type = meta_REDSPARK;
vgmstream->sample_rate = h.sample_rate;
vgmstream->num_samples = h.num_samples;
vgmstream->loop_start_sample = h.loop_start;
vgmstream->loop_end_sample = h.loop_end;
vgmstream->num_streams = h.total_subsongs;
vgmstream->stream_size = h.stream_size;
{ vgmstream->coding_type = coding_NGC_DSP;
off_t start = 0x54; vgmstream->layout_type = layout_interleave;
int i,j; vgmstream->interleave_block_size = 0x08;
start += channel_count * 8; for (int ch = 0; ch < h.channels; ch++) {
if (loop_flag) { for (int i = 0; i < 16; i++) {
start += 16; vgmstream->ch[ch].adpcm_coef[i] = h.coefs[ch][i];
}
for (j = 0; j < channel_count; j++) {
for (i=0;i<16;i++) {
vgmstream->ch[j].adpcm_coef[i] =
get_16bit(&buf[start+0x2e*j+i*2]);
}
} }
} }
/* open the file for reading */ if (h.dummy) {
{ vgmstream->num_samples = h.sample_rate;
int i; vgmstream->coding_type = coding_SILENCE;
STREAMFILE * file;
file = streamFile->open(streamFile,filename,STREAMFILE_DEFAULT_BUFFER_SIZE);
if (!file) goto fail;
for (i=0;i<channel_count;i++) {
vgmstream->ch[i].streamfile = file;
vgmstream->ch[i].channel_start_offset=
vgmstream->ch[i].offset=start_offset + i*vgmstream->interleave_block_size;
}
} }
if (!vgmstream_open_stream(vgmstream, sf, h.stream_offset))
goto fail;
return vgmstream; return vgmstream;
/* clean up anything we may have opened */
fail: fail:
if (vgmstream) close_vgmstream(vgmstream); close_vgmstream(vgmstream);
return NULL; return NULL;
} }
static uint32_t rotlwi(uint32_t x, uint32_t r) {
return (x << r) | (x >> (32 - r));
}
static uint32_t decrypt_chunk(uint8_t* buf, int buf_size, uint32_t key) {
uint32_t data;
int pos = 0;
if (key == 0) {
/* initial key seems to only vary slightly between files though (doesn't seem to depend on size/name) */
key = get_u32be(buf + 0x00) ^ get_id32be("RedS");
data = get_u32be(buf + 0x00) ^ key;
put_u32be(&buf[0], data);
key = rotlwi(key, 11);
pos = 0x04;
}
for (int i = pos; i < buf_size; i += 0x04) {
key = rotlwi(key, 3) + key;
data = get_u32be(buf + i) ^ key;
put_u32be(buf + i, data);
}
return key; /* to resume decrypting if needed */
}
#define HEADER_MAX 0x2800 /* seen 0x2420 in one bank */
/* header is encrypted except in M&L 3DS so decrypt + handle in buffer; format
* base 0x30 header + subheader with tables/headers depending on type */
static bool parse_header(redspark_header_t* h, STREAMFILE* sf, bool is_new) {
uint8_t buf[HEADER_MAX];
/* LE on 3DS */
get_u16_t get_u16 = !is_new ? get_u16be : get_u16le;
get_u32_t get_u32 = !is_new ? get_u32be : get_u32le;
uint32_t curr_key = 0x00;
int target_subsong = sf->stream_index;
/* base header:
00 "RedSpark"
08 chunk size (usually data size, smaller for subspark)
0c type/chunk? 00010000=streams, 00000000=bus/se
10 ? (low number, usually same for multiple files)
14 file id or null
18 data offset
1c bank flag
1e 0909=stream, 0404=stream (MW) or RedSpark-within aka 'subspark' (Imabikisou), 0000=bus/se config (no audio)
20 data size (usually file size, or header + subheader size for subsparks)
24-30 null/reserved
*/
int base_size = 0x30;
if (read_streamfile(buf, 0x00, base_size,sf) != base_size)
return false;
if (!is_new)
curr_key = decrypt_chunk(buf, 0x30, curr_key);
/* get base header */
if (get_u64be(buf + 0x00) != get_id64be("RedSpark"))
return false;
uint32_t data_offset = get_u32(buf + 0x18);
uint32_t data_size = get_u32(buf + 0x20);
int bank_flag = get_u16(buf + 0x1c);
int type = get_u16(buf + 0x1e);
if (data_offset >= HEADER_MAX)
return false;
/* get subheader and prepare offsets */
int redspark_pos;
int sub_offset, sub_size;
int head_pos;
if (data_offset == 0x30) {
/* 'subspark', seen in a few sfx in Imabikisou (earlier version of banks) */
/* at data offset (not encrypted):
00 always 017F0100 (LE?)
04-10: null
10 file id
13 string? size
14 encrypted string? (ends with 0)
at subchunk_size:
another RedSpark with bank subflag
*/
/* base sub-RedSpark + reset flags */
redspark_pos = data_size;
if (read_streamfile(buf + redspark_pos, data_size, base_size, sf) != base_size)
return false;
if (!is_new)
curr_key = decrypt_chunk(buf + redspark_pos, base_size, 0); /* new header */
/* setup rest of header (handled below) */
uint32_t subdata_offset = get_u32(buf + redspark_pos + 0x18);
bank_flag = get_u16(buf + redspark_pos + 0x1c);
type = get_u16(buf + redspark_pos + 0x1e);
data_offset = redspark_pos + subdata_offset;
if (data_offset >= HEADER_MAX)
return false;
sub_offset = redspark_pos + base_size;
sub_size = data_offset - sub_offset;
head_pos = sub_offset;
}
else {
redspark_pos = 0x00;
sub_offset = base_size;
sub_size = data_offset - sub_offset;
head_pos = sub_offset;
}
/* read + decrypt rest of header */
if (read_streamfile(buf + sub_offset, sub_offset, sub_size, sf) != sub_size)
return false;
if (!is_new)
decrypt_chunk(buf + sub_offset, sub_size, curr_key);
//VGM_LOGB(buf, data_offset, 0);
/* bus/se config */
if (type == 0x0000) {
/* 30 number of entries (after data offset)
per entry
00 channel offset (after all entries)
per channel at the above offset
00 channel config (similar to channel config in streams but extended) */
vgm_logi("RedSpark: file has no audio\n");
return false;
}
/* main info */
uint32_t coef_offset;
if (bank_flag) {
/* bank with N subsongs (seen in M&L sfx packs and in subsparks) */
/* 00 data size
0c entries (only 1 is possible)
0e entries again
20+ table
per entry:
00 absolute offset to header
per header
00 null?
04 stream size
08 sample rate (null in subspark)
0c nibbles (old) or samples (new), (null in subspark so uses loops)
10 stream offset (from data_offset)
14 loop end (num samples if non-looped)
18 loop start of -1 it not looped
20 config?
24 config?
28 coefs + hists
5c channel config?
*/
h->total_subsongs = get_u16(buf + head_pos + 0x0c);
if (!check_subsongs(&target_subsong, h->total_subsongs))
return false;
int target_pos = head_pos + 0x20 + (target_subsong - 1) * 0x04;
if (target_pos + 0x04 >= HEADER_MAX)
return false;
target_pos = get_u32(buf + target_pos) + redspark_pos;
if (target_pos + 0x70 >= HEADER_MAX)
return false;
h->stream_size = get_u32(buf + target_pos + 0x04);
h->sample_rate = get_u32(buf + target_pos + 0x08);
h->num_samples = get_u32(buf + target_pos + 0x0c);
h->stream_offset = get_u32(buf + target_pos + 0x10) + data_offset;
h->loop_end = get_u32(buf + target_pos + 0x14);
h->loop_start = get_u32(buf + target_pos + 0x18);
coef_offset = target_pos + 0x28;
h->channels = 1;
h->loop_flag = (h->loop_start != -1); /* TODO: many files sound kind of odd */
if (h->num_samples == 0)
h->num_samples = h->loop_end;
if (h->sample_rate == 0)
h->sample_rate = 32000;
/* empty entry */
if (h->stream_size == 0)
h->dummy = true;
}
else {
/* stream */
/* 00 data size (after data offset)
04 null
08 null
0c sample rate
10 frames (old) or samples (new)
14 some chunk? (0x10000, 0xc000)
18 null
1c null
1e channels (usually 1-2, sometimes 4 in MW)
1f num loop cues (2 == loop)
20 cues again?
21 volume?
22 null
24+ variable
per channel
00 config per channel, size 0x08 (number/panning/volume/etc?)
per loop point
00 chunk size?
04 value
per channel
00 dsp coefs + hists (size 0x2E)
if cues:
00 offset?
04 null
per cue:
00 name size
01 string ("Loop Start" / "Loop End")
*/
h->sample_rate = get_u32(buf + head_pos + 0x0c);
h->num_samples = get_u32(buf + head_pos + 0x10);
h->channels = get_u8(buf + head_pos + 0x1e);
int loop_cues = get_u8(buf + head_pos + 0x1f);
head_pos += 0x24;
/* just in case to avoid bad reads outside buf */
if (h->channels > MAX_CHANNELS)
return false;
head_pos += h->channels * 0x08;
h->loop_flag = (loop_cues != 0);
if (h->loop_flag) {
/* only two cue points */
if (loop_cues != 0 && loop_cues != 2)
return false;
h->loop_start = get_u32(buf + head_pos + 0x04);
h->loop_end = get_u32(buf + head_pos + 0x0c);
head_pos += 0x10;
}
coef_offset = head_pos;
h->stream_offset = data_offset;
h->stream_size = data_size;
}
/* coefs from decrypted buf (could read hist but it's 0 in DSPs) */
for (int ch = 0; ch < h->channels; ch++) {
for (int i = 0; i < 16; i++) {
h->coefs[ch][i] = get_u16(buf + coef_offset + 0x2e * ch + i * 2);
}
}
/* fixes */
if (!is_new) {
h->loop_end += 1;
if (bank_flag) {
h->num_samples /= 2 * 0x8;
h->loop_start /= 2 * 0x8;
h->loop_end /= 2 * 0x8;
}
h->num_samples *= 14;
h->loop_start *= 14;
h->loop_end *= 14;
if (h->loop_end > h->num_samples) /* needed for some files */
h->loop_end = h->num_samples;
}
/* new + bank may need +1 */
return true;
}

View File

@ -13,7 +13,11 @@ typedef uint16_t (*read_u16_t)(off_t, STREAMFILE*);
typedef int16_t (*read_s16_t)(off_t, STREAMFILE*); typedef int16_t (*read_s16_t)(off_t, STREAMFILE*);
typedef float (*read_f32_t)(off_t, STREAMFILE*); typedef float (*read_f32_t)(off_t, STREAMFILE*);
typedef int16_t (*get_s16_t)(const uint8_t*); typedef uint16_t (*get_u16_t)(const uint8_t*);
typedef int16_t (*get_s16_t)(const uint8_t*);
typedef uint32_t (*get_u32_t)(const uint8_t*);
typedef int32_t (*get_s32_t)(const uint8_t*);
/* guess byte endianness from a given value, return true if big endian and false if little endian */ /* guess byte endianness from a given value, return true if big endian and false if little endian */
static inline int guess_endian16(off_t offset, STREAMFILE* sf) { static inline int guess_endian16(off_t offset, STREAMFILE* sf) {