ref: 98f6464a85a41fb0c4801df589c825011506fb76
dir: /src/it/readamf.c/
/* _______ ____ __ ___ ___ * \ _ \ \ / \ / \ \ / / ' ' ' * | | \ \ | | || | \/ | . . * | | | | | | || ||\ /| | * | | | | | | || || \/ | | ' ' ' * | | | | | | || || | | . . * | |_/ / \ \__// || | | * /_______/ynamic \____/niversal /__\ /____\usic /| . . ibliotheque * / \ * / . \ * readamf.c - Code to read a DSMI AMF module from / / \ \ * an open file. | < / \_ * | \/ /\ / * By Chris Moeller. \_ / > / * | \ / / * | ' / * \__/ */ #include <stdlib.h> #include <string.h> #include <math.h> #include "dumb.h" #include "internal/it.h" static void it_amf_process_track( IT_ENTRY *entry_table, unsigned char *track, int rows, int channels ) { int last_instrument = 0; int tracksize = track[ 0 ] + ( track[ 1 ] << 8 ) + ( track[ 2 ] << 16 ); track += 3; while ( tracksize-- ) { unsigned int row = track[ 0 ]; unsigned int command = track[ 1 ]; unsigned int argument = track[ 2 ]; IT_ENTRY * entry = entry_table + row * channels; if ( row >= ( unsigned int ) rows ) break; if ( command < 0x7F ) { entry->mask |= IT_ENTRY_NOTE | IT_ENTRY_INSTRUMENT | IT_ENTRY_VOLPAN; entry->note = command; if ( ! entry->instrument ) entry->instrument = last_instrument; entry->volpan = argument; } else if ( command == 0x7F ) { signed char row_delta = ( signed char ) argument; int row_source = ( int ) row + ( int ) row_delta; if ( row_source >= 0 && row_source < ( int ) rows ) { *entry = entry_table[ row_source * channels ]; } } else if ( command == 0x80 ) { entry->mask |= IT_ENTRY_INSTRUMENT; last_instrument = argument + 1; entry->instrument = last_instrument; } else if ( command == 0x83 ) { entry->mask |= IT_ENTRY_VOLPAN; entry->volpan = argument; } else { unsigned int effect = command & 0x7F; unsigned int effectvalue = argument; switch (effect) { case 0x01: effect = IT_SET_SPEED; break; case 0x02: effect = IT_VOLUME_SLIDE; case 0x0A: if ( effect == 0x0A ) effect = IT_VOLSLIDE_TONEPORTA; case 0x0B: if ( effect == 0x0B ) effect = IT_VOLSLIDE_VIBRATO; if ( effectvalue & 0x80 ) effectvalue = ( -( signed char ) effectvalue ) & 0x0F; else effectvalue = ( effectvalue & 0x0F ) << 4; break; case 0x04: if ( effectvalue & 0x80 ) { effect = IT_PORTAMENTO_UP; effectvalue = ( -( signed char ) effectvalue ) & 0x7F; } else { effect = IT_PORTAMENTO_DOWN; } break; case 0x06: effect = IT_TONE_PORTAMENTO; break; case 0x07: effect = IT_TREMOR; break; case 0x08: effect = IT_ARPEGGIO; break; case 0x09: effect = IT_VIBRATO; break; case 0x0C: effect = IT_BREAK_TO_ROW; break; case 0x0D: effect = IT_JUMP_TO_ORDER; break; case 0x0F: effect = IT_RETRIGGER_NOTE; break; case 0x10: effect = IT_SET_SAMPLE_OFFSET; break; case 0x11: if ( effectvalue ) { effect = IT_VOLUME_SLIDE; if ( effectvalue & 0x80 ) effectvalue = 0xF0 | ( ( -( signed char ) effectvalue ) & 0x0F ); else effectvalue = 0x0F | ( ( effectvalue & 0x0F ) << 4 ); } else effect = 0; break; case 0x12: case 0x16: if ( effectvalue ) { int mask = ( effect == 0x16 ) ? 0xE0 : 0xF0; effect = ( effectvalue & 0x80 ) ? IT_PORTAMENTO_UP : IT_PORTAMENTO_DOWN; if ( effectvalue & 0x80 ) effectvalue = mask | ( ( -( signed char ) effectvalue ) & 0x0F ); else effectvalue = mask | ( effectvalue & 0x0F ); } else effect = 0; break; case 0x13: effect = IT_S; effectvalue = EFFECT_VALUE( IT_S_NOTE_DELAY, effectvalue & 0x0F ); break; case 0x14: effect = IT_S; effectvalue = EFFECT_VALUE( IT_S_DELAYED_NOTE_CUT, effectvalue & 0x0F ); break; case 0x15: effect = IT_SET_SONG_TEMPO; break; case 0x17: effectvalue = ( effectvalue + 64 ) & 0x7F; if ( entry->mask & IT_ENTRY_EFFECT ) { if ( !( entry->mask & IT_ENTRY_VOLPAN ) ) { entry->mask |= IT_ENTRY_VOLPAN; entry->volpan = ( effectvalue / 2 ) + 128; } effect = 0; } else { effect = IT_SET_PANNING; } break; default: effect = effectvalue = 0; } if ( effect ) { entry->mask |= IT_ENTRY_EFFECT; entry->effect = effect; entry->effectvalue = effectvalue; } } track += 3; } } static int it_amf_process_pattern( IT_PATTERN *pattern, IT_ENTRY *entry_table, int rows, int channels ) { int i, j; int n_entries = rows; IT_ENTRY * entry; pattern->n_rows = rows; for ( i = 0, j = channels * rows; i < j; i++ ) { if ( entry_table[ i ].mask ) { n_entries++; } } pattern->n_entries = n_entries; pattern->entry = entry = malloc( n_entries * sizeof( IT_ENTRY ) ); if ( !entry ) { return -1; } for ( i = 0; i < rows; i++ ) { for ( j = 0; j < channels; j++ ) { if ( entry_table[ i * channels + j ].mask ) { *entry = entry_table[ i * channels + j ]; entry->channel = j; entry++; } } IT_SET_END_ROW( entry ); entry++; } return 0; } static int it_amf_read_sample_header( IT_SAMPLE *sample, DUMBFILE *f, int * offset, int ver ) { int exists; exists = dumbfile_getc( f ); dumbfile_getnc( (char *) sample->name, 32, f ); sample->name[32] = 0; dumbfile_getnc( (char *) sample->filename, 13, f ); sample->filename[13] = 0; *offset = (int)dumbfile_igetl( f ); sample->length = dumbfile_igetl( f ); sample->C5_speed = dumbfile_igetw( f ); sample->default_volume = dumbfile_getc( f ); sample->global_volume = 64; if ( sample->default_volume > 64 ) sample->default_volume = 64; if ( ver >= 11 ) { sample->loop_start = dumbfile_igetl( f ); sample->loop_end = dumbfile_igetl( f ); } else { sample->loop_start = dumbfile_igetw( f ); sample->loop_end = sample->length; } if ( sample->length <= 0 ) { sample->flags = 0; return 0; } sample->flags = exists == 1 ? IT_SAMPLE_EXISTS : 0; sample->default_pan = 0; sample->finetune = 0; if ( sample->loop_end > sample->loop_start + 2 && sample->loop_end <= sample->length ) sample->flags |= IT_SAMPLE_LOOP; sample->vibrato_speed = 0; sample->vibrato_depth = 0; sample->vibrato_rate = 0; sample->vibrato_waveform = 0; // do we have to set _all_ these? sample->max_resampling_quality = -1; return dumbfile_error(f); } static int it_amf_read_sample_data( IT_SAMPLE *sample, DUMBFILE *f ) { int i, read_length = 0; sample->data = malloc( sample->length ); if ( !sample->data ) return -1; if ( sample->length ) read_length = (int)dumbfile_getnc( sample->data, sample->length, f ); for ( i = 0; i < read_length; i++ ) { ( ( signed char * ) sample->data )[ i ] ^= 0x80; } for ( i = read_length; i < sample->length; i++ ) { ( ( signed char * ) sample->data )[ i ] = 0; } return 0; /* Sometimes the last sample is truncated :( */ } static DUMB_IT_SIGDATA *it_amf_load_sigdata(DUMBFILE *f, int * version) { DUMB_IT_SIGDATA *sigdata; int i, j, ver, ntracks, realntracks, nchannels; int maxsampleseekpos = 0; int sampleseekpos[256]; unsigned short *orderstotracks; unsigned short *trackmap; unsigned int tracksize[256]; unsigned char **track; static const char sig[] = "AMF"; char signature [3]; if ( dumbfile_getnc( signature, 3, f ) != 3 || memcmp( signature, sig, 3 ) ) { return NULL; } *version = ver = dumbfile_getc( f ); if ( ver < 10 || ver > 14) { return NULL; } sigdata = malloc(sizeof(*sigdata)); if (!sigdata) { return NULL; } dumbfile_getnc( (char *) sigdata->name, 32, f ); sigdata->name[ 32 ] = 0; sigdata->n_samples = dumbfile_getc( f ); sigdata->n_orders = dumbfile_getc( f ); ntracks = dumbfile_igetw( f ); nchannels = dumbfile_getc( f ); if ( dumbfile_error( f ) || sigdata->n_samples < 1 || sigdata->n_samples > 255 || sigdata->n_orders < 1 || sigdata->n_orders > 255 || ! ntracks || nchannels < 1 || nchannels > 32 ) { free( sigdata ); return NULL; } sigdata->n_pchannels = nchannels; memset( sigdata->channel_volume, 64, DUMB_IT_N_CHANNELS ); if ( ver >= 11 ) { int nchannels = ( ver >= 13 ) ? 32 : 16; for ( i = 0; i < nchannels; i++ ) { signed char panpos = dumbfile_getc( f ); int pan = ( panpos + 64 ) / 2; if ( pan < 0 ) pan = 0; else if ( pan > 64 ) pan = IT_SURROUND; sigdata->channel_pan[ i ] = pan; } } else { int sep = 32 * dumb_it_default_panning_separation / 100; for ( i = 0; i < 16; i++ ) { sigdata->channel_pan[ i ] = ( dumbfile_getc( f ) & 1 ) ? 32 - sep : 32 + sep; } } sigdata->tempo = 125; sigdata->speed = 6; if ( ver >= 13 ) { i = dumbfile_getc( f ); if ( i >= 32 ) sigdata->tempo = i; i = dumbfile_getc( f ); if ( i <= 32 ) sigdata->speed = i; } sigdata->order = malloc( sigdata->n_orders ); if ( !sigdata->order ) { free( sigdata ); return NULL; } orderstotracks = malloc( sigdata->n_orders * nchannels * sizeof( unsigned short ) ); if ( !orderstotracks ) { free( sigdata->order ); free( sigdata ); return NULL; } for ( i = 0; i < sigdata->n_orders; i++ ) { sigdata->order[ i ] = i; tracksize[ i ] = 64; if ( ver >= 14 ) { tracksize[ i ] = dumbfile_igetw( f ); } for ( j = 0; j < nchannels; j++ ) { orderstotracks[ i * nchannels + j ] = dumbfile_igetw( f ); } } if ( dumbfile_error( f ) ) { free( orderstotracks ); free( sigdata->order ); free( sigdata ); return NULL; } sigdata->sample = malloc( sigdata->n_samples * sizeof( *sigdata->sample ) ); if ( !sigdata->sample ) { free( orderstotracks ); free( sigdata->order ); free( sigdata ); return NULL; } sigdata->restart_position = 0; sigdata->song_message = NULL; sigdata->instrument = NULL; sigdata->pattern = NULL; sigdata->midi = NULL; sigdata->checkpoint = NULL; sigdata->n_instruments = 0; for ( i = 0; i < sigdata->n_samples; ++i ) sigdata->sample[i].data = NULL; for ( i = 0; i < sigdata->n_samples; ++i ) { int offset; if ( it_amf_read_sample_header( &sigdata->sample[i], f, &offset, ver ) ) { goto error_ott; } sampleseekpos[ i ] = offset; if ( offset > maxsampleseekpos ) maxsampleseekpos = offset; } sigdata->n_patterns = sigdata->n_orders; sigdata->pattern = malloc( sigdata->n_patterns * sizeof( *sigdata->pattern ) ); if ( !sigdata->pattern ) { goto error_ott; } for (i = 0; i < sigdata->n_patterns; ++i) sigdata->pattern[i].entry = NULL; trackmap = malloc( ntracks * sizeof( unsigned short ) ); if ( !trackmap ) { goto error_ott; } if ( dumbfile_getnc( ( char * ) trackmap, ntracks * sizeof( unsigned short ), f ) != (long)(ntracks * sizeof( unsigned short )) ) { goto error_tm; } realntracks = 0; for ( i = 0; i < ntracks; i++ ) { if ( trackmap[ i ] > realntracks ) realntracks = trackmap[ i ]; } track = calloc( realntracks, sizeof( unsigned char * ) ); if ( !track ) { goto error_tm; } for ( i = 0; i < realntracks; i++ ) { int tracksize = dumbfile_igetw( f ); tracksize += dumbfile_getc( f ) << 16; track[ i ] = malloc( tracksize * 3 + 3 ); if ( !track[ i ] ) { goto error_all; } track[ i ][ 0 ] = tracksize & 255; track[ i ][ 1 ] = ( tracksize >> 8 ) & 255; track[ i ][ 2 ] = ( tracksize >> 16 ) & 255; if ( dumbfile_getnc( (char *) track[ i ] + 3, tracksize * 3, f ) != tracksize * 3 ) { goto error_all; } } for ( i = 1; i <= maxsampleseekpos; i++ ) { for ( j = 0; j < sigdata->n_samples; j++ ) { if ( sampleseekpos[ j ] == i ) { if ( it_amf_read_sample_data( &sigdata->sample[ j ], f ) ) { goto error_all; } break; } } } /* Process tracks into patterns */ for ( i = 0; i < sigdata->n_patterns; i++ ) { IT_ENTRY * entry_table = calloc( tracksize[ i ] * nchannels, sizeof( IT_ENTRY ) ); if ( !entry_table ) { goto error_all; } for ( j = 0; j < nchannels; j++ ) { int ntrack = orderstotracks[ i * nchannels + j ]; if ( ntrack && ntrack <= ntracks ) { int realtrack = trackmap[ ntrack - 1 ]; if ( realtrack ) { realtrack--; if ( realtrack < realntracks && track[ realtrack ] ) { it_amf_process_track( entry_table + j, track[ realtrack ], tracksize[ i ], nchannels ); } } } } if ( it_amf_process_pattern( &sigdata->pattern[ i ], entry_table, tracksize[ i ], nchannels ) ) { free( entry_table ); goto error_all; } free( entry_table ); } /* Now let's initialise the remaining variables, and we're done! */ sigdata->flags = IT_OLD_EFFECTS | IT_COMPATIBLE_GXX | IT_STEREO | IT_WAS_AN_S3M; sigdata->global_volume = 128; sigdata->mixing_volume = 48; sigdata->pan_separation = 128; _dumb_it_fix_invalid_orders(sigdata); for ( i = 0; i < realntracks; i++ ) { if ( track[ i ] ) { free( track[ i ] ); } } free( track ); free( trackmap ); free( orderstotracks ); return sigdata; error_all: for ( i = 0; i < realntracks; i++ ) { if ( track[ i ] ) { free( track[ i ] ); } } free( track ); error_tm: free( trackmap ); error_ott: free( orderstotracks ); _dumb_it_unload_sigdata( sigdata ); return NULL; } DUH *dumb_read_amf_quick(DUMBFILE *f) { sigdata_t *sigdata; DUH_SIGTYPE_DESC *descptr = &_dumb_sigtype_it; int version; sigdata = it_amf_load_sigdata(f, &version); if (!sigdata) return NULL; { const char *tag[2][2]; char ver_string[14]; tag[0][0] = "TITLE"; tag[0][1] = (const char *)(((DUMB_IT_SIGDATA *)sigdata)->name); tag[1][0] = "FORMAT"; memcpy( ver_string, "DSMI AMF v", 10 ); ver_string[10] = '0' + version / 10; ver_string[11] = '.'; ver_string[12] = '0' + version % 10; ver_string[13] = 0; tag[1][1] = ver_string; return make_duh(-1, 2, (const char *const (*)[2])tag, 1, &descptr, &sigdata); } }