/*************************************************************************/ /* cp_loader_it_samples.cpp */ /*************************************************************************/ /* This file is part of: */ /* GODOT ENGINE */ /* http://www.godotengine.org */ /*************************************************************************/ /* Copyright (c) 2007-2014 Juan Linietsky, Ariel Manzur. */ /* */ /* Permission is hereby granted, free of charge, to any person obtaining */ /* a copy of this software and associated documentation files (the */ /* "Software"), to deal in the Software without restriction, including */ /* without limitation the rights to use, copy, modify, merge, publish, */ /* distribute, sublicense, and/or sell copies of the Software, and to */ /* permit persons to whom the Software is furnished to do so, subject to */ /* the following conditions: */ /* */ /* The above copyright notice and this permission notice shall be */ /* included in all copies or substantial portions of the Software. */ /* */ /* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, */ /* EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF */ /* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.*/ /* IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY */ /* CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, */ /* TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE */ /* SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */ /*************************************************************************/ #include "cp_loader_it.h" #include "cp_sample.h" struct AuxSampleData { uint32_t fileofs; uint32_t c5spd; uint32_t length; uint32_t loop_begin; uint32_t loop_end; bool loop_enabled; bool pingpong_enabled; bool is16bit; bool stereo; bool exists; bool compressed; }; enum IT_Sample_Flags { IT_SAMPLE_EXISTS=1, IT_SAMPLE_16BITS=2, IT_SAMPLE_STEREO=4, IT_SAMPLE_COMPRESSED=8, IT_SAMPLE_LOOPED=16, IT_SAMPLE_SUSTAIN_LOOPED=32, IT_SAMPLE_LOOP_IS_PINGPONG=64, IT_SAMPLE_SUSTAIN_LOOP_IS_PINGPONG=128 }; CPLoader::Error CPLoader_IT::load_sample(CPSample *p_sample) { AuxSampleData aux_sample_data; char aux_header[4]; file->get_byte_array((uint8_t*)aux_header,4); if ( aux_header[0]!='I' || aux_header[1]!='M' || aux_header[2]!='P' || aux_header[3]!='S') { //CP_PRINTERR("IT CPLoader CPSample: Failed Identifier"); return FILE_UNRECOGNIZED; } // Ignore deprecated 8.3 filename for (int i=0;i<12;i++) file->get_byte(); file->get_byte(); //ignore zerobyte p_sample->set_global_volume( file->get_byte() ); /* SAMPLE FLAGS */ uint8_t flags=file->get_byte(); aux_sample_data.loop_enabled=flags&IT_SAMPLE_LOOPED; aux_sample_data.pingpong_enabled=flags&IT_SAMPLE_LOOP_IS_PINGPONG; aux_sample_data.is16bit=flags&IT_SAMPLE_16BITS; aux_sample_data.exists=flags&IT_SAMPLE_EXISTS; aux_sample_data.stereo=flags&IT_SAMPLE_STEREO; aux_sample_data.compressed=flags&IT_SAMPLE_COMPRESSED; p_sample->set_default_volume(file->get_byte()); /* SAMPLE NAME */ char aux_name[26]; file->get_byte_array((uint8_t*)aux_name,26); p_sample->set_name(aux_name); // ?? uint8_t convert_flag=file->get_byte(); // PAN uint8_t pan=file->get_byte(); p_sample->set_pan( pan&0x7F ); p_sample->set_pan_enabled( pan & 0x80 ); aux_sample_data.length=file->get_dword(); aux_sample_data.loop_begin= file->get_dword(); aux_sample_data.loop_end= file->get_dword(); aux_sample_data.c5spd=file->get_dword(); /*p_sample->data.set_sustain_loop_begin=*/file->get_dword(); /*p_sample->data.sustain_loop_end=*/file->get_dword(); aux_sample_data.fileofs=file->get_dword(); p_sample->set_vibrato_speed( file->get_byte() ); p_sample->set_vibrato_depth( file->get_byte() ); p_sample->set_vibrato_rate( file->get_byte() ); switch( file->get_byte() ) { /* Vibrato Wave: 0=sine, 1=rampdown, 2=square, 3=random */ case 0: p_sample->set_vibrato_type( CPSample::VIBRATO_SINE ); break; case 1: p_sample->set_vibrato_type( CPSample::VIBRATO_SAW ); break; case 2: p_sample->set_vibrato_type( CPSample::VIBRATO_SQUARE ); break; case 3: p_sample->set_vibrato_type( CPSample::VIBRATO_RANDOM ); break; default: p_sample->set_vibrato_type( CPSample::VIBRATO_SINE ); break; } //printf("Name %s - Flags: fileofs :%i - c5spd %i - len %i 16b %i - data?: %i\n",p_sample->get_name(),aux_sample_data.fileofs,aux_sample_data.c5spd, aux_sample_data.length, aux_sample_data.is16bit,aux_sample_data.exists); CPSample_ID samp_id; if (aux_sample_data.exists) { samp_id=load_sample_data(aux_sample_data); CPSampleManager::get_singleton()->set_c5_freq(samp_id,aux_sample_data.c5spd); CPSampleManager::get_singleton()->set_loop_begin( samp_id,aux_sample_data.loop_begin ); CPSampleManager::get_singleton()->set_loop_end( samp_id,aux_sample_data.loop_end ); CPSample_Loop_Type loop_type=aux_sample_data.loop_enabled?( aux_sample_data.pingpong_enabled? CP_LOOP_BIDI: CP_LOOP_FORWARD):CP_LOOP_NONE; CPSampleManager::get_singleton()->set_loop_end( samp_id,aux_sample_data.loop_end ); CPSampleManager::get_singleton()->set_loop_type( samp_id, loop_type); } //printf("Loaded id is null?: %i\n",samp_id.is_null()); p_sample->set_sample_data(samp_id); if (!samp_id.is_null()) { // printf("Loaded ID: stereo: %i len %i 16bit %i\n",CPSampleManager::get_singleton()->is_stereo(samp_id), CPSampleManager::get_singleton()->get_size( samp_id), CPSampleManager::get_singleton()->is_16bits( samp_id) ); } CP_ERR_COND_V( file->eof_reached(),FILE_CORRUPTED ); CP_ERR_COND_V( file->get_error(),FILE_CORRUPTED ); return FILE_OK; } CPSample_ID CPLoader_IT::load_sample_data(AuxSampleData& p_sample_data) { int aux_sample_properties = (p_sample_data.is16bit?IT_SAMPLE_16BITS:0)|(p_sample_data.compressed?IT_SAMPLE_COMPRESSED:0)|(p_sample_data.stereo?IT_SAMPLE_STEREO:0); file->seek(p_sample_data.fileofs); CPSampleManager *sm=CPSampleManager::get_singleton(); CPSample_ID id; switch (aux_sample_properties) { case (0): // 8 bits, mono case (IT_SAMPLE_16BITS): // 16 bits mono case (IT_SAMPLE_STEREO): // 8 bits stereo case (IT_SAMPLE_16BITS|IT_SAMPLE_STEREO): { // 16 bits mono id=sm->create(p_sample_data.is16bit,p_sample_data.stereo,p_sample_data.length); if (id.is_null()) break; sm->lock_data(id); int16_t *ptr16 = (int16_t*)sm->get_data(id); int8_t *ptr8=(int8_t*)ptr16; int chans=p_sample_data.stereo?2:1; if (p_sample_data.is16bit) { for (int c=0;c<chans;c++) { for (int i=0;i<p_sample_data.length;i++) { ptr16[i*chans+c]=file->get_word(); } } } else { for (int c=0;c<chans;c++) { for (int i=0;i<p_sample_data.length;i++) { ptr8[i*chans+c]=file->get_byte(); } } } sm->unlock_data(id); } break; case (IT_SAMPLE_COMPRESSED): { // 8 bits compressed id=sm->create(false,false,p_sample_data.length); if (id.is_null()) break; sm->lock_data(id); if ( load_sample_8bits_IT_compressed((void*)sm->get_data( id),p_sample_data.length) ) { sm->unlock_data(id); sm->destroy(id); break; } sm->unlock_data(id); } break; case (IT_SAMPLE_16BITS|IT_SAMPLE_COMPRESSED): { // 16 bits compressed id=sm->create(true,false,p_sample_data.length); if (id.is_null()) break; sm->lock_data(id); if ( load_sample_16bits_IT_compressed((void*)sm->get_data(id),p_sample_data.length) ) { sm->unlock_data(id); sm->destroy(id); break; } sm->unlock_data(id); } break; default: { // I dont know how to handle stereo compressed, does that exist? } break; } return id; } CPLoader::Error CPLoader_IT::load_samples() { for (int i=0;i<header.smpnum;i++) { //seek to sample file->seek(0xC0+header.ordnum+header.insnum*4+i*4); uint32_t final_location=file->get_dword(); file->seek( final_location ); Error err=load_sample(song->get_sample(i)); CP_ERR_COND_V(err,err); } if (file->eof_reached() || file->get_error()) return FILE_CORRUPTED; return FILE_OK; } /* * NOTICE * NOTICE * NOTICE * NOTICE * NOTICE * NOTICE * NOTICE * NOTICE * NOTICE * NOTICE * NOTICE -The following sample decompression code is based on xmp's code.(http://xmp.helllabs.org) which is based in openCP code. * NOTICE * NOTICE * NOTICE * NOTICE * NOTICE * NOTICE * NOTICE * NOTICE * NOTICE * NOTICE * NOTICE */ uint32_t CPLoader_IT::read_n_bits_from_IT_compressed_block (uint8_t p_bits_to_read) { uint32_t aux_return_value; uint32_t val; uint8_t *buffer=(uint8_t*)source_position; if ( p_bits_to_read <= source_remaining_bits ) { val=buffer[3]; val<<=8; val|=buffer[2]; val<<=8; val|=buffer[1]; val<<=8; val|=buffer[0]; aux_return_value = val & ((1 << p_bits_to_read) - 1); val >>= p_bits_to_read; source_remaining_bits -= p_bits_to_read; buffer[3]=val>>24; buffer[2]=(val>>16)&0xFF; buffer[1]=(val>>8)&0xFF; buffer[0]=(val)&0xFF; } else { aux_return_value=buffer[3]; aux_return_value<<=8; aux_return_value|=buffer[2]; aux_return_value<<=8; aux_return_value|=buffer[1]; aux_return_value<<=8; aux_return_value|=buffer[0]; uint32_t nbits = p_bits_to_read - source_remaining_bits; source_position++; buffer+=4; val=buffer[3]; val<<=8; val|=buffer[2]; val<<=8; val|=buffer[1]; val<<=8; val|=buffer[0]; aux_return_value |= ((val & ((1 << nbits) - 1)) << source_remaining_bits); val >>= nbits; source_remaining_bits = 32 - nbits; buffer[3]=val>>24; buffer[2]=(val>>16)&0xFF; buffer[1]=(val>>8)&0xFF; buffer[0]=(val)&0xFF; } return aux_return_value; } bool CPLoader_IT::read_IT_compressed_block (bool p_16bits) { uint16_t size; size=file->get_word(); if (file->eof_reached() || file->get_error()) return true; pat_data = (uint8_t*)CP_ALLOC( 4* ((size >> 2) + 2) ); if (!pat_data) return true; source_buffer=(uint32_t*)pat_data; file->get_byte_array((uint8_t*)source_buffer,size); if (file->eof_reached() || file->get_error()) { free_IT_compressed_block(); return true; } source_position = source_buffer; source_remaining_bits = 32; return false; } void CPLoader_IT::free_IT_compressed_block () { if (pat_data) { CP_FREE(pat_data); pat_data=NULL; } } bool CPLoader_IT::load_sample_8bits_IT_compressed(void *p_dest_buffer,int p_buffsize) { int8_t *dest_buffer; /* destination buffer which will be returned */ uint16_t block_length; /* length of compressed data block in samples */ uint16_t block_position; /* position in block */ uint8_t bit_width; /* actual "bit width" */ uint16_t aux_value; /* value read from file to be processed */ int8_t d1, d2; /* integrator buffers (d2 for it2.15) */ int8_t *dest_position; /* position in output buffer */ int8_t v; /* sample value */ bool it215; // is this an it215 module? dest_buffer = (int8_t *) p_dest_buffer; if (dest_buffer==NULL) return true; for (int i=0;i<p_buffsize;i++) dest_buffer[i]=0; dest_position = dest_buffer; it215=(header.cmwt==0x215); /* now unpack data till the dest buffer is full */ while (p_buffsize) { /* read a new block of compressed data and reset variables */ if ( read_IT_compressed_block(false) ) { CP_PRINTERR("Out of memory decompressing IT CPSample"); return true; } block_length = (p_buffsize < 0x8000) ? p_buffsize : 0x8000; block_position = 0; bit_width = 9; /* start with width of 9 bits */ d1 = d2 = 0; /* reset integrator buffers */ /* now uncompress the data block */ while ( block_position < block_length ) { aux_value = read_n_bits_from_IT_compressed_block(bit_width); /* read bits */ if ( bit_width < 7 ) { /* method 1 (1-6 bits) */ if ( aux_value == (1 << (bit_width - 1)) ) { /* check for "100..." */ aux_value = read_n_bits_from_IT_compressed_block(3) + 1; /* yes -> read new width; */ bit_width = (aux_value < bit_width) ? aux_value : aux_value + 1; /* and expand it */ continue; /* ... next value */ } } else if ( bit_width < 9 ) { /* method 2 (7-8 bits) */ uint8_t border = (0xFF >> (9 - bit_width)) - 4; /* lower border for width chg */ if ( aux_value > border && aux_value <= (border + 8) ) { aux_value -= border; /* convert width to 1-8 */ bit_width = (aux_value < bit_width) ? aux_value : aux_value + 1; /* and expand it */ continue; /* ... next value */ } } else if ( bit_width == 9 ) { /* method 3 (9 bits) */ if ( aux_value & 0x100 ) { /* bit 8 set? */ bit_width = (aux_value + 1) & 0xff; /* new width... */ continue; /* ... and next value */ } } else { /* illegal width, abort */ free_IT_compressed_block(); CP_PRINTERR("CPSample has illegal BitWidth "); return true; } /* now expand value to signed byte */ if ( bit_width < 8 ) { uint8_t tmp_shift = 8 - bit_width; v=(aux_value << tmp_shift); v>>=tmp_shift; } else v = (int8_t) aux_value; /* integrate upon the sample values */ d1 += v; d2 += d1; /* ... and store it into the buffer */ *(dest_position++) = it215 ? d2 : d1; block_position++; } /* now subtract block lenght from total length and go on */ free_IT_compressed_block(); p_buffsize -= block_length; } return false; } bool CPLoader_IT::load_sample_16bits_IT_compressed(void *p_dest_buffer,int p_buffsize) { int16_t *dest_buffer; /* destination buffer which will be returned */ uint16_t block_length; /* length of compressed data block in samples */ uint16_t block_position; /* position in block */ uint8_t bit_width; /* actual "bit width" */ uint32_t aux_value; /* value read from file to be processed */ int16_t d1, d2; /* integrator buffers (d2 for it2.15) */ int16_t *dest_position; /* position in output buffer */ int16_t v; /* sample value */ bool it215; // is this an it215 module? dest_buffer = (int16_t *) p_dest_buffer; if (dest_buffer==NULL) return true; for (int i=0;i<p_buffsize;i++) dest_buffer[i]=0; dest_position = dest_buffer; it215=(header.cmwt==0x215); while (p_buffsize) { /* read a new block of compressed data and reset variables */ if ( read_IT_compressed_block(true) ) { return true; } block_length = (p_buffsize < 0x4000) ? p_buffsize : 0x4000; block_position = 0; bit_width = 17; /* start with width of 9 bits */ d1 = d2 = 0; /* reset integrator buffers */ while ( block_position < block_length ) { aux_value = read_n_bits_from_IT_compressed_block(bit_width); /* read bits */ if ( bit_width < 7 ) { /* method 1 (1-6 bits) */ if ( (signed)aux_value == (1 << (bit_width - 1)) ) { /* check for "100..." */ aux_value = read_n_bits_from_IT_compressed_block(4) + 1; /* yes -> read new width; */ bit_width = (aux_value < bit_width) ? aux_value : aux_value + 1; /* and expand it */ continue; /* ... next value */ } } else if ( bit_width < 17 ) { uint16_t border = (0xFFFF >> (17 - bit_width)) - 8; if ( (int)aux_value > (int)border && (int)aux_value <= ((int)border + 16) ) { aux_value -= border; /* convert width to 1-8 */ bit_width = (aux_value < bit_width) ? aux_value : aux_value + 1; /* and expand it */ continue; /* ... next value */ } } else if ( bit_width == 17 ) { if ( aux_value & 0x10000 ) { /* bit 8 set? */ bit_width = (aux_value + 1) & 0xff; /* new width... */ continue; /* ... and next value */ } } else { /* illegal width, abort */ CP_PRINTERR("CPSample has illegal BitWidth "); free_IT_compressed_block(); return true; } /* now expand value to signed byte */ if ( bit_width < 16 ) { uint8_t tmp_shift = 16 - bit_width; v=(aux_value << tmp_shift); v>>=tmp_shift; } else v = (int16_t) aux_value; /* integrate upon the sample values */ d1 += v; d2 += d1; /* ... and store it into the buffer */ *(dest_position++) = it215 ? d2 : d1; block_position++; } /* now subtract block lenght from total length and go on */ free_IT_compressed_block(); p_buffsize -= block_length; } return false; }