446 lines
9.3 KiB
C
446 lines
9.3 KiB
C
/*
|
|
Audio Overload SDK - QSF file engine
|
|
|
|
Copyright (c) 2007, R. Belmont and Richard Bannister.
|
|
|
|
All rights reserved.
|
|
|
|
Redistribution and use in source and binary forms, with or without modification, are permitted provided that the following conditions are met:
|
|
|
|
* Redistributions of source code must retain the above copyright notice, this list of conditions and the following disclaimer.
|
|
* Redistributions in binary form must reproduce the above copyright notice, this list of conditions and the following disclaimer in the documentation and/or other materials provided with the distribution.
|
|
* Neither the names of R. Belmont and Richard Bannister nor the names of its contributors may be used to endorse or promote products derived from this software without specific prior written permission.
|
|
|
|
THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
|
"AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
|
LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
|
|
A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR
|
|
CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
|
|
EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
|
|
PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
|
|
PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
|
|
LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
|
|
NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
|
|
SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
*/
|
|
|
|
//
|
|
// eng_qsf.c
|
|
// by R. Belmont
|
|
//
|
|
|
|
/*
|
|
The program section of a QSF file, once decompressed, contains a series of
|
|
data blocks of the form:
|
|
|
|
3 bytes - ASCII section name tag
|
|
4 bytes - Starting offset (LSB-first)
|
|
4 bytes - Length (N) (LSB-first)
|
|
N bytes - Data
|
|
|
|
The data is then loaded to the given starting offset in the section described
|
|
by the ASCII tag.
|
|
|
|
The following sections are defined:
|
|
|
|
"KEY" - Kabuki decryption key. This section should be 11 bytes and contain
|
|
the following:
|
|
4 bytes - swap_key1 (MSB-first)
|
|
4 bytes - swap_key2 (MSB-first)
|
|
2 bytes - addr_key (MSB-first)
|
|
1 bytes - xor_key
|
|
"Z80" - Z80 program ROM.
|
|
"SMP" - QSound sample ROM.
|
|
|
|
If the KEY section isn't given or both swap_keys are zero, then it is assumed
|
|
that no encryption is used.
|
|
*/
|
|
|
|
#include <stdio.h>
|
|
#include <string.h>
|
|
#include <stdlib.h>
|
|
|
|
#include "ao.h"
|
|
#include "qsound.h"
|
|
#include "z80.h"
|
|
|
|
#include "corlett.h"
|
|
|
|
#define DEBUG_LOADER (0)
|
|
|
|
// timer rate is 285 Hz
|
|
static int32 samples_per_tick = 44100/285;
|
|
static int32 samples_to_next_tick = 44100/285;
|
|
|
|
static corlett_t *c = NULL;
|
|
static char qsfby[256];
|
|
static uint32 skey1, skey2;
|
|
static uint16 akey;
|
|
static uint8 xkey;
|
|
static int32 uses_kabuki = 0;
|
|
|
|
static char *Z80ROM, *QSamples;
|
|
static char RAM[0x1000], RAM2[0x1000];
|
|
static int32 cur_bank;
|
|
|
|
static struct QSound_interface qsintf =
|
|
{
|
|
QSOUND_CLOCK,
|
|
NULL
|
|
};
|
|
|
|
extern void cps1_decode(unsigned char *rom, int swap_key1,int swap_key2,int addr_key,int xor_key);
|
|
|
|
static void qsf_walktags(uint8 *buffer, uint8 *end)
|
|
{
|
|
uint8 *cbuf = buffer;
|
|
uint32 offset, length;
|
|
|
|
while (cbuf < end)
|
|
{
|
|
#if DEBUG_LOADER
|
|
printf("cbuf: %08x end: %08x\n", (uint32)cbuf, (uint32)end);
|
|
#endif
|
|
offset = cbuf[3] | cbuf[4]<<8 | cbuf[5]<<16 | cbuf[6]<<24;
|
|
length = cbuf[7] | cbuf[8]<<8 | cbuf[9]<<16 | cbuf[10]<<24;
|
|
|
|
#if DEBUG_LOADER
|
|
printf("Tag: %c%c%c @ %08x, length %08x\n", cbuf[0], cbuf[1], cbuf[2], offset, length);
|
|
#endif
|
|
|
|
switch (cbuf[0])
|
|
{
|
|
case 'Z':
|
|
memcpy(&Z80ROM[offset], &cbuf[11], length);
|
|
break;
|
|
|
|
case 'S':
|
|
memcpy(&QSamples[offset], &cbuf[11], length);
|
|
break;
|
|
|
|
case 'K':
|
|
skey1 = cbuf[11]<<24 | cbuf[12]<<16 | cbuf[13]<<8 | cbuf[14];
|
|
skey2 = cbuf[15]<<24 | cbuf[16]<<16 | cbuf[17]<<8 | cbuf[18];
|
|
akey = cbuf[19]<<8 | cbuf[20];
|
|
xkey = cbuf[20];
|
|
break;
|
|
|
|
default:
|
|
printf("ERROR: Unknown QSF tag!\n");
|
|
break;
|
|
}
|
|
|
|
cbuf += 11;
|
|
cbuf += length;
|
|
}
|
|
}
|
|
|
|
static int32 qsf_irq_cb(int param)
|
|
{
|
|
return 0x000000ff; // RST_38
|
|
}
|
|
|
|
int32 qsf_start(uint8 *buffer, uint32 length)
|
|
{
|
|
uint8 *file, *lib_decoded, *lib_raw_file;
|
|
uint64 file_len, lib_len, lib_raw_length;
|
|
corlett_t *lib;
|
|
|
|
z80_init();
|
|
|
|
Z80ROM = malloc(512*1024);
|
|
QSamples = malloc(8*1024*1024);
|
|
|
|
skey1 = skey2 = 0;
|
|
akey = 0;
|
|
xkey = 0;
|
|
cur_bank = 0;
|
|
|
|
memset(RAM, 0, 0x1000);
|
|
memset(RAM2, 0, 0x1000);
|
|
|
|
// Decode the current QSF
|
|
if (corlett_decode(buffer, length, &file, &file_len, &c) != AO_SUCCESS)
|
|
{
|
|
return AO_FAIL;
|
|
}
|
|
|
|
// Get the library file
|
|
if (c->lib[0] != 0)
|
|
{
|
|
uint64 tmp_length;
|
|
|
|
#if DEBUG_LOADER
|
|
printf("Loading library: %s\n", c->lib);
|
|
#endif
|
|
if (ao_get_lib(c->lib, &lib_raw_file, &tmp_length) != AO_SUCCESS)
|
|
{
|
|
return AO_FAIL;
|
|
}
|
|
lib_raw_length = tmp_length;
|
|
|
|
if (corlett_decode(lib_raw_file, lib_raw_length, &lib_decoded, &lib_len, &lib) != AO_SUCCESS)
|
|
{
|
|
free(lib_raw_file);
|
|
return AO_FAIL;
|
|
}
|
|
|
|
// Free up raw file
|
|
free(lib_raw_file);
|
|
|
|
// use the contents
|
|
qsf_walktags(lib_decoded, lib_decoded+lib_len);
|
|
|
|
// Dispose the corlett structure for the lib - we don't use it
|
|
free(lib);
|
|
free(lib_decoded);
|
|
}
|
|
|
|
// now patch the file into RAM OVER the libraries
|
|
qsf_walktags(file, file+file_len);
|
|
|
|
free(file);
|
|
|
|
if ((skey1 != 0) && (skey2 != 0))
|
|
{
|
|
#if DEBUG_LOADER
|
|
printf("Decoding Kabuki: skey1 %08x skey2 %08x akey %04x xkey %02x\n", skey1, skey2, akey, xkey);
|
|
#endif
|
|
|
|
uses_kabuki = 1;
|
|
cps1_decode((unsigned char *)Z80ROM, skey1, skey2, akey, xkey);
|
|
}
|
|
|
|
// set qsfby tag
|
|
strcpy(qsfby, "n/a");
|
|
if (c)
|
|
{
|
|
int i;
|
|
for (i = 0; i < MAX_UNKNOWN_TAGS; i++)
|
|
{
|
|
if (!strcasecmp(c->tag_name[i], "qsfby"))
|
|
{
|
|
strcpy(qsfby, c->tag_data[i]);
|
|
}
|
|
}
|
|
}
|
|
|
|
z80_reset(NULL);
|
|
z80_set_irq_callback(qsf_irq_cb);
|
|
qsintf.sample_rom = QSamples;
|
|
qsound_sh_start(&qsintf);
|
|
|
|
return AO_SUCCESS;
|
|
}
|
|
|
|
static void timer_tick(void)
|
|
{
|
|
z80_set_irq_line(0, ASSERT_LINE);
|
|
z80_set_irq_line(0, CLEAR_LINE);
|
|
}
|
|
|
|
int32 qsf_gen(int16 *buffer, uint32 samples)
|
|
{
|
|
int16 output[44100/30], output2[44100/30];
|
|
int16 *stereo[2];
|
|
int16 *outp = buffer;
|
|
int32 i, opos, tickinc, loops;
|
|
|
|
// our largest possible step is samples_per_tick or samples, whichever is smaller
|
|
if (samples_to_next_tick > samples)
|
|
{
|
|
tickinc = samples;
|
|
}
|
|
else
|
|
{
|
|
tickinc = samples_to_next_tick;
|
|
}
|
|
|
|
loops = samples / tickinc;
|
|
opos = 0;
|
|
|
|
for (i = 0; i < loops; i++)
|
|
{
|
|
z80_execute((8000000/44100)*tickinc);
|
|
stereo[0] = &output[opos];
|
|
stereo[1] = &output2[opos];
|
|
qsound_update(0, stereo, tickinc);
|
|
|
|
opos += tickinc;
|
|
samples_to_next_tick -= tickinc;
|
|
|
|
if (samples_to_next_tick <= 0)
|
|
{
|
|
timer_tick();
|
|
samples_to_next_tick = samples_per_tick;
|
|
}
|
|
}
|
|
|
|
// are there "leftovers"?
|
|
if (opos < samples)
|
|
{
|
|
z80_execute((8000000/44100)*(samples-opos));
|
|
stereo[0] = &output[opos];
|
|
stereo[1] = &output2[opos];
|
|
qsound_update(0, stereo, (samples-opos));
|
|
|
|
samples_to_next_tick -= (samples-opos);
|
|
|
|
if (samples_to_next_tick <= 0)
|
|
{
|
|
timer_tick();
|
|
samples_to_next_tick = samples_per_tick;
|
|
}
|
|
}
|
|
|
|
for (i = 0; i < samples; i++)
|
|
{
|
|
*outp++ = output[i];
|
|
*outp++ = output2[i];
|
|
}
|
|
|
|
return AO_SUCCESS;
|
|
}
|
|
|
|
int32 qsf_stop(void)
|
|
{
|
|
z80_exit();
|
|
free(Z80ROM);
|
|
free(QSamples);
|
|
free(c);
|
|
|
|
return AO_SUCCESS;
|
|
}
|
|
|
|
int32 qsf_command(int32 command, int32 parameter)
|
|
{
|
|
switch (command)
|
|
{
|
|
case COMMAND_RESTART:
|
|
return AO_SUCCESS;
|
|
|
|
}
|
|
return AO_FAIL;
|
|
}
|
|
|
|
int32 qsf_fill_info(ao_display_info *info)
|
|
{
|
|
if (c == NULL)
|
|
return AO_FAIL;
|
|
|
|
strcpy(info->title[1], "Name: ");
|
|
sprintf(info->info[1], "%s", c->inf_title);
|
|
|
|
strcpy(info->title[2], "Game: ");
|
|
sprintf(info->info[2], "%s", c->inf_game);
|
|
|
|
strcpy(info->title[3], "Artist: ");
|
|
sprintf(info->info[3], "%s", c->inf_artist);
|
|
|
|
strcpy(info->title[4], "Copyright: ");
|
|
sprintf(info->info[4], "%s", c->inf_copy);
|
|
|
|
strcpy(info->title[5], "Year: ");
|
|
sprintf(info->info[5], "%s", c->inf_year);
|
|
|
|
strcpy(info->title[6], "Length: ");
|
|
sprintf(info->info[6], "%s", c->inf_length);
|
|
|
|
strcpy(info->title[7], "Fade: ");
|
|
sprintf(info->info[7], "%s", c->inf_fade);
|
|
|
|
strcpy(info->title[8], "Ripper: ");
|
|
sprintf(info->info[8], "%s", qsfby);
|
|
|
|
return AO_SUCCESS;
|
|
}
|
|
|
|
uint8 qsf_memory_read(uint16 addr)
|
|
{
|
|
if (addr < 0x8000)
|
|
{
|
|
return Z80ROM[addr];
|
|
}
|
|
else if (addr < 0xc000)
|
|
{
|
|
return Z80ROM[(addr - 0x8000) + cur_bank];
|
|
}
|
|
else if (addr <= 0xcfff)
|
|
{
|
|
return RAM[addr - 0xc000];
|
|
}
|
|
else if (addr == 0xd007)
|
|
{
|
|
return qsound_status_r();
|
|
}
|
|
else if (addr >= 0xf000)
|
|
{
|
|
return RAM2[addr-0xf000];
|
|
}
|
|
}
|
|
|
|
uint8 qsf_memory_readop(uint16 addr)
|
|
{
|
|
if (!uses_kabuki)
|
|
{
|
|
return qsf_memory_read(addr);
|
|
}
|
|
|
|
if (addr < 0x8000)
|
|
{
|
|
return Z80ROM[addr + (256*1024)];
|
|
}
|
|
|
|
return qsf_memory_read(addr);
|
|
}
|
|
|
|
uint8 qsf_memory_readport(uint16 addr)
|
|
{
|
|
return Z80ROM[0x11];
|
|
}
|
|
|
|
void qsf_memory_write(uint16 addr, uint8 byte)
|
|
{
|
|
if (addr >= 0xc000 && addr <= 0xcfff)
|
|
{
|
|
|
|
RAM[addr-0xc000] = byte;
|
|
return;
|
|
}
|
|
else if (addr == 0xd000)
|
|
{
|
|
qsound_data_h_w(byte);
|
|
return;
|
|
}
|
|
else if (addr == 0xd001)
|
|
{
|
|
qsound_data_l_w(byte);
|
|
return;
|
|
}
|
|
else if (addr == 0xd002)
|
|
{
|
|
qsound_cmd_w(byte);
|
|
return;
|
|
}
|
|
else if (addr == 0xd003)
|
|
{
|
|
cur_bank = (0x8000 + (byte & 0xf) * 0x4000);
|
|
if (cur_bank > (256*1024))
|
|
{
|
|
cur_bank = 0;
|
|
}
|
|
// printf("Z80 bank to %x (%x)\n", cur_bank, byte);
|
|
return;
|
|
}
|
|
else if (addr >= 0xf000)
|
|
{
|
|
RAM2[addr-0xf000] = byte;
|
|
return;
|
|
}
|
|
}
|
|
|
|
void qsf_memory_writeport(uint16 addr, uint8 byte)
|
|
{
|
|
printf("Unk port %x @ %x\n", byte, addr);
|
|
}
|
|
|