Hamazing/source/framework/musicplayers/raspberry_casket.asm

4196 lines
128 KiB
NASM
Raw Permalink Normal View History

2023-05-22 16:17:35 +02:00
;--------------------------------------------------------------------
; Raspberry Casket Player V1.1 (28-Dec-2022)
; ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
;
; Provided by Chris 'platon42' Hodges <chrisly@platon42.de>
;
; Latest: https://github.com/chrisly42/PretrackerRaspberryCasket
;
; Rewritten by platon42/Desire based on a resourced, binary identical
; version of the original Pretracker V1.0 replayer binary provided
; by hitchhikr (thanks!), originally written in C by Pink/Abyss.
;
; This version is the hard work of reverse engineering all the
; offsets, removing all the C compiler crud, removing dead and
; surplus code (maybe artefacts from earlier ideas that did nothing),
; optimizing the code where possible. This resulted in both reduced
; size of the replayer, faster sample calculation and speeding the
; tick routine up significantly.
;
; I also added a few optional features that come in handy, such as
; song-end detection and precalc progress support.
;
; It took me more than a month and it was not fun.
;
; Also: Open source. It's 2022, keeping the code closed is just not
; part of the demoscene spirit (anymore?), at least for a replayer.
;
; Also note that this is not the final state of the source code.
; I could go over many places still and try to rework them.
; But I wanted the code to be out in public.
;
; Verification
; ~~~~~~~~~~~~
; The replayer has been verified on about 60 Pretracker tunes to
; create an identical internal state for each tick and identical
; samples (if certain optimizations switches are disabled).
;
; I might have introduced bugs though. If you find some problems,
; please let me know under chrisly@platon42.de. Thank you.
;
; Usage
; ~~~~~
; The new replayer comes as a drop-in binary replacement if you wish.
; In this case you will get faster sample generation (about 12%
; faster on 68000) and about 45% less CPU time spent. However, you
; won't get stuff as song-end detection and precalc progress this way.
; This mode uses the old CPU DMA wait that takes away 8 raster lines.
;
; If you want to get rid of the unnecessary waiting, you can switch
; to a copper driven audio control. If you want to use the top portion
; of the copperlist for this, you probably need to double buffer it.
; Otherwise, you could also position the copperlist at the end of
; the display and use single buffering if you call the tick routine
; during the vertical blank.
;
; Please use the documented sizes for the MySong and MyPlayer data
; structures, which are the symbols sv_SIZEOF and pv_SIZEOF
; respectively (about 2K and 12K with volume table).
;
; The source needs two common include files to compile (custom.i and
; dmabits.i). You should leave assembler optimizations enabled.
;
; (0. If you're using copper list mode, call pre_PrepareCopperlist.)
;
; 1. Call pre_SongInit with
; - a pointer to MySong (mv_SIZEOF) in a1 and
; - the music data in a2.
; It will return the amount of sample memory needed in d0.
;
; 2. Then call pre_PlayerInit with
; - a pointer to MyPlayer (pv_SIZEOF) in a0
; - a pointer to chip memory sample buffer in a1
; - the pointer to MySong in a2
; - a pointer to a longword for progress information or null in a3
; This will create the samples, too.
;
; 3. After that, regularly call pre_PlayerTick with MyPlayer in a0
; and optionally the copperlist in a1 if you're using that mode).
;
; Size
; ~~~~
; The original C compiled code was... just bad. The new binary is
; about 1/3 of the original one.
;
; The code has been also optimized in a way that it compresses better.
; The original code compressed with Blueberry's Shrinkler goes from
; 18052 bytes down to 9023 bytes.
;
; Raspberry Casket, depending on the features compiled in, is about
; 6216 bytes and goes down to ~4348 bytes (in isolation).
;
; So this means that the optimization is not just "on the outside".
;
; Timing
; ~~~~~~
; Sample generation is a bit faster (I guess around 10-15%), but most
; of the time is spent on muls operations, so this is the limiting
; factor.
;
; Raspberry Casket is about twice as fast as the old replayer for playback.
;
; Unfortunately, the replayer is still pretty slow and has high
; jitter compared to other standard music replayers.
; This means it may take up to 32 raster lines (13-18 on average)
; which is significant more than a standard Protracker replayer
; (the original one could take about 60 raster lines worst case and
; about 34 on average!).
;
; Watch out for Presto, the LightSpeedPlayer variant that should
; solve this problem.
;
; Changelog see https://github.com/chrisly42/PretrackerRaspberryCasket#Changelog
;--------------------------------------------------------------------
; Here come the various options for you to configure.
;
; To create an optimized drop-in replacement for the old V1.0 binary:
; PRETRACKER_SUPPORT_V1_5 = 0
; PRETRACKER_PARANOIA_MODE = 0
; PRETRACKER_DUBIOUS_PITCH_SHIFT_FOR_DELAYED_TRACK = 0
; PRETRACKER_KEEP_JUMP_TABLE = 1
; PRETRACKER_SONG_END_DETECTION = 0
; PRETRACKER_PROGRESS_SUPPORT = 0
; PRETRACKER_FASTER_CODE = 1
; PRETRACKER_VOLUME_TABLE = 1
; PRETRACKER_BUGFIX_CODE = 1
; PRETRACKER_DONT_TRASH_REGS = 1
; PRETRACKER_COPPER_OUTPUT = 0
; This player was based on the V1.0 binary of the pretracker replayer
; and thus can only play files created with versions up to V1.0.
; Subsongs and sound effects are not supported. There is usually no
; use for this in an intro anyway.
;
; Enabling this switch will patch the song runtime to make it backward
; compatible with this replayer. Song data will be modified in memory.
; I would encourage you to still use V1.0 as the files saved with
; Pretracker V1.5 are unnecessarily bigger (about 458 bytes) and might
; lose some instrument names it seems.
IFND PRETRACKER_SUPPORT_V1_5
PRETRACKER_SUPPORT_V1_5 = 0
ENDC
; The original binary had a lot of extra checks that I would consider
; paranoia. For example it handled songs without instruments, waves
; or patterns.
; It contains code that I think is unreachable and a relic of
; being used for the actual tracker itself (releasing notes after
; a delay, looping patterns instead of advancing song).
; Or for combinations of song stopping (F00) and note delay in the
; same step, which no sane musician would use.
;
; This switch gets rid of the code. If you find a song that doesn't
; work with this disabled, please let me know.
IFND PRETRACKER_PARANOIA_MODE
PRETRACKER_PARANOIA_MODE = 0
ENDC
; There is some pitch shifting applied during track delay by
; a maximum of one period -- depending on the delay value.
; I guess this might add some phasing to the sound (which would be
; only audible in a mono mix, I would think), but it comes with an
; extra multiplication per channel.
; If you really want that, enable this switch.
IFND PRETRACKER_DUBIOUS_PITCH_SHIFT_FOR_DELAYED_TRACK
PRETRACKER_DUBIOUS_PITCH_SHIFT_FOR_DELAYED_TRACK = 0
ENDC
; The binary comes with a jump table that has three official entry
; points. As you have the source, you could just bsr to the three
; functions (pre_SongInit, pre_PlayerInit, pre_PlayerTick) directly.
; If you want a drop-in (binary) replacement instead, enable this
; switch to get the jump table back.
IFND PRETRACKER_KEEP_JUMP_TABLE
PRETRACKER_KEEP_JUMP_TABLE = 0
ENDC
; The original Pretracker replayer does not come with a song-end
; detection. If you want to have that (e.g. for a music disk), enable
; this switch and check for pv_songend_detected_b (relative to the
; MyPlayer structure) which goes to true when song-end is reached.
IFND PRETRACKER_SONG_END_DETECTION
PRETRACKER_SONG_END_DETECTION = 0
ENDC
; Do you want to have information on the sample generation progress
; during the call to pre_PlayerInit? Then enable this and call w
; pre_PlayerInit with a pointer to a longword in a3.
; Please make sure yourself that the initial value is zero.
; It will be incremented by the number of samples (in bytes)
; for each waveform done. You can get the total number of samples
; from the (previously undocumented) return value of pre_SongInit.
IFND PRETRACKER_PROGRESS_SUPPORT
PRETRACKER_PROGRESS_SUPPORT = 0
ENDC
; Use slightly faster code and smaller for sample generation that
; might be off by 1 (rounding error) compared to the original code.
IFND PRETRACKER_FASTER_CODE
PRETRACKER_FASTER_CODE = 1
ENDC
; Use tables for volume calculation instead of multiplication.
; Is slightly faster on 68000 (about 11%), but probably has no
; benefit for 68020+ and needs about 3.5 KB more RAM (MyPlayer)
; and 40 bytes of code.
IFND PRETRACKER_VOLUME_TABLE
PRETRACKER_VOLUME_TABLE = 1
ENDC
; I found some obvious bugs in the code. This switch enables bugfixes,
; but the sound might not be exactly the same as in the tracker /
; original player (e.g. there is a bug in the volume envelope that
; will cause a pause in the decay curve that is not supposed to
; happen).
IFND PRETRACKER_BUGFIX_CODE
PRETRACKER_BUGFIX_CODE = 1
ENDC
; You want to take care of what registers may be trashed by these
; routines because you have your own ABI? Good!
; Setting this to 0 will remove the register saving/restoring on
; the stack. All registers then may be trashed.
; Otherwise, d2-d7/a2-a6 are preserved as with the AmigaOS standard.
IFND PRETRACKER_DONT_TRASH_REGS
PRETRACKER_DONT_TRASH_REGS = 1
ENDC
; Enable output to copperlist instead of audio registers and DMA wait.
; This gives your slightly more CPU time and less jitter (maybe 7
; rasterlines).
;
; When enabled, provide a pointer to a copperlist in a1 on calling
; pre_PlayerTick. It will update the necessary fields in the
; copperlist. To initially generate a copperlist (or two), use the
; pre_PrepareCopperlist subroutine. It will write out 37 copper
; commands (WAITs and MOVEs) starting for a given rasterline (if
; you're below rasterline 255, make sure you have the necessary
; wait for this case yourself!).
;
; There are two major reasonable ways to use this:
; - Your intro does no fancy copper shenanigans:
; You can reserve the space of 37 longwords inside your normal
; copperlist. The position should be after rasterline ~35.
; - You do a lot of copper stuff during the frame:
; Create the pretracker copperlist to be at the very end of the
; frame (rasterline 300 is a good spot). Make sure your custom
; copper code has a static jump to the 37 copper instructions
; at its end and terminate the copperlist correctly (hint:
; move.l d1,(a0) after bsr pre_PrepareCopperlist will terminate
; the copperlist ($fffffffe).
IFND PRETRACKER_COPPER_OUTPUT
PRETRACKER_COPPER_OUTPUT = 0 ; 0 = standard CPU wait, 1 = Copperlist
ENDC
; Pretracker song format description:
;
; $0000 4: PRT<version> ($19 (V0.x), $1b (V1.0), $1e (V1.5))
; $0004 4: File offset to position data (POSD)
; $0008 4: File offset to pattern data (PATT)
; $000C 4: File offset to instruments (INST)
; $0010 4: File offset to waves (WAVE)
; $0014 20: Songname
; $0028 20: Author
; $003C 1: Restart position for song (<=V1.0)
; $003D 1: Number of patterns (<=V1.0)
; $003E 1: Songlength in patterns (<=V1.0)
; $003F 1: Number of steps per pattern (<=V1.0)
; $0040 1: Number of instruments (<=V1.0), $40 for V1.5
; $0041 1: Number of waves
; $0042 24: Wave generation ordering (>=V1.0)
; $005a 1: Number of subsongs (>=V1.5)
; [...]
; Position data (POSD):
; - Per subsong (>=V1.5)
; - 1: Restart pos
; - 1: #patterns
; - 1: #numsteps
; - 1: songlength
; - 4: relative pattern offset in pattern data
; - Positions x 4 x [Pattern number (byte), pitch shift (signed byte)]
;
; Pattern data (PATT):
; Each pattern line consists of three bytes:
; - 1 Bit : Bit 4 of Inst Number
; - 1 Bit : ARP instead of effect
; - 6 Bits: Pitch ($01 is C-0, $3d is NOTE OFF)
;
; - 4 Bits: Bit 0-3 of Inst Number
; - 4 Bits: Effect command
; - 8 Bits: Effect data
; - Patterns x steps x 3 bytes
;
; Unknown data after pattern data: (12 10 00 00 00 00)
;
; Instrument definitions (INST):
; - 32 (!) x Null terminated string (or 23 chars max) (for V1.5 this is 64 strings)
; - For each instrument: Instrument Info (ININ)
; - $00 1: Vibrato Delay
; - $01 1: Vibrato Depth
; - $02 1: Vibrato Speed (-1)
; - $03 1: ADSR Attack
; - $04 1: ADSR Decay
; - $05 1: ADSR Sustain
; - $06 1: ADSR Release
; - $07 1: Number of Inst Pattern steps
;
; - For each instrument:
; - 1 Byte: number of steps
; - 3 Bytes x number of steps: Inst Pattern (IPTT)
;
; Inst pattern data (IPTT):
; Each pattern line consists of three bytes:
; - 1 Bit : Next note stitched to this one
; - 1 Bit : Fixed Pitch Note
; - 6 Bits : Pitch ($01 is C-0)
; - 4 Bits : unused?
; - 12 Bits: Effect
; - Patterns x steps x 3 bytes
;
; Wave definitions (WAVE):
; - 24 (!) x Null terminated string (or 23 chars max)
; - Optional padding to even address, if necessary
; - For each wave:
; - 42 Bytes: Wave info structure (see definition below)
; ----------------------------------------
; Some constants for clarity
MAX_VOLUME = $40
MAX_SPEED = $2f
MAX_WAVES = 24
MAX_INSTRUMENTS = 32
MAX_TRACK_DELAY = 32
NOTE_OFF_PITCH = $3d
NOTES_IN_OCTAVE = 12
NUM_CHANNELS = 4 ; yes, you can reduce the number of channels if you want
; ----------------------------------------
; Pretracker file structures
; Pattern data (PATT and IPTT)
pdb_pitch_ctrl = 0
pdb_inst_effect = 1 ; for normal pattern data
pdb_effect_cmd = 1 ; for inst pattern
pdb_effect_data = 2
; Pattern pos data (POSD)
ppd_pat_num = 0
ppd_pat_shift = 1
; Instrument Info (ININ)
ii_vibrato_delay = 0
ii_vibrato_depth = 1
ii_vibrato_speed = 2
ii_adsr_attack = 3
ii_adsr_decay = 4
ii_adsr_sustain = 5
ii_adsr_release = 6
ii_pattern_steps = 7
ii_SIZEOF = 8
; Wave Info (WAVE)
wi_loop_start_w = $00
wi_loop_end_w = $02
wi_subloop_len_w = $04
wi_allow_9xx_b = $06
wi_subloop_wait_b = $07
wi_subloop_step_w = $08
wi_chipram_w = $0a
wi_loop_offset_w = $0c
wi_chord_note1_b = $0e
wi_chord_note2_b = $0f
wi_chord_note3_b = $10
wi_chord_shift_b = $11
wi_osc_unknown_b = $12 ; always $00? (unused in code?)
wi_osc_phase_spd_b = $13
wi_flags_b = $14 ; bit 0/1: osc type, bit 2: needs extra octaves, bit 3: boost, bit 4: pitch linear, bit 5: vol fast
wi_osc_phase_min_b = $15
wi_osc_phase_max_b = $16
wi_osc_basenote_b = $17
wi_osc_gain_b = $18
wi_sam_len_b = $19 ; in multiples of 128, zero-based (0 == 128)
wi_mix_wave_b = $1a
wi_vol_attack_b = $1b
wi_vol_delay_b = $1c
wi_vol_decay_b = $1d
wi_vol_sustain_b = $1e
wi_flt_type_b = $1f ; 1=lowpass, 2=highpass, 3=bandpass, 4=notch
wi_flt_resonance_b = $20
wi_pitch_ramp_b = $21
wi_flt_start_b = $22
wi_flt_min_b = $23
wi_flt_max_b = $24
wi_flt_speed_b = $25
wi_mod_params_l = $26
wi_mod_wetness_b = $26
wi_mod_length_b = $27
wi_mod_predelay_b = $28
wi_mod_density_b = $29 ; (1-7), unisono (bits 3/4) and post bit 5
wi_SIZEOF = $2a
; ----------------------------------------
; Unpacked Instrument Info (addition to player for faster processing)
rsreset
uii_vibrato_delay rs.w 1
uii_vibrato_depth rs.w 1
uii_vibrato_speed rs.w 1
uii_adsr_release rs.b 1
rs.b 1 ; dummy
uii_adsr_attack rs.w 1
uii_adsr_decay rs.w 1
uii_adsr_sustain rs.w 1
uii_pattern_steps rs.b 1
rs.b 1 ; padding
uii_SIZEOF rs.b 0
; ----------------------------------------
; MySong offsets
rsreset
sv_waveinfo_table rs.l MAX_WAVES ; 24 pointers to wave infos to avoid mulu
sv_inst_patterns_table rs.l MAX_INSTRUMENTS ; 32 pointers to pattern data
; --- 127 byte displacement limit ---
sv_wavelength_table rs.l MAX_WAVES ; 24 longwords to sample lengths (standard octave) (NEW)
sv_wavetotal_table rs.l MAX_WAVES ; 24 longwords to sample lengths for all octaves (NEW)
sv_wavegen_order_table rs.b MAX_WAVES ; 24 bytes
sv_num_waves_b rs.b 1
sv_num_steps_b rs.b 1
sv_patterns_ptr rs.l 1
sv_curr_pat_pos_w rs.w 1 ; only byte used FIXME why is this part of MySong? Should be in Player
sv_pat_pos_len_w rs.w 1 ; only byte used
sv_pat_restart_pos_w rs.w 1 ; only byte used
sv_pos_data_adr rs.l 1
sv_waveinfo_ptr rs.l 1 ; base pointer of wave info
sv_pattern_table rs.l 256
sv_inst_infos_table rs.b MAX_INSTRUMENTS*uii_SIZEOF
sv_SIZEOF rs.b 0
; ----------------------------------------
; channel output data (part of pcd structure below)
rsreset
ocd_sam_ptr rs.l 1 ; 0
ocd_length rs.w 1 ; 4
ocd_loop_offset rs.w 1 ; 6
ocd_period rs.w 1 ; 8
ocd_volume rs.b 1 ; 10
ocd_trigger rs.b 1 ; 11 needs to be after volume
ocd_unused rs.l 1 ; 12 unused, but makes the structure an even 16 bytes
ocd_SIZEOF rs.b 0
; channel structure (part of pv structure)
rsreset
; DO NOT CHANGE ORDER -- OPTIMIZED CLEARING
pcd_pat_portamento_dest_w rs.w 1 ; portamento destination pitch
pcd_pat_pitch_slide_w rs.w 1
pcd_pat_vol_ramp_speed_b rs.b 1
pcd_pat_2nd_inst_num4_b rs.b 1
pcd_pat_2nd_inst_delay_b rs.b 1
pcd_wave_offset_b rs.b 1
pcd_inst_pitch_slide_w rs.w 1
pcd_inst_sel_arp_note_w rs.w 1
pcd_inst_note_pitch_w rs.w 1
pcd_inst_curr_port_pitch_w rs.w 1
pcd_inst_line_ticks_b rs.b 1
pcd_inst_pitch_pinned_b rs.b 1
pcd_inst_vol_slide_b rs.b 1
pcd_inst_step_pos_b rs.b 1
pcd_inst_wave_num_w rs.w 1 ; current wave number (1 based) (lower byte used)
pcd_track_delay_offset_b rs.b 1 ; $ff = no track delay
pcd_inst_speed_stop_b rs.b 1 ; speed byte, $ff stops processing
pcd_inst_pitch_w rs.w 1
pcd_inst_vol_w rs.w 1
pcd_loaded_inst_vol_b rs.b 1
pcd_pat_vol_b rs.b 1 ; Multiplied with volume of instrument.
; DO NOT CHANGE ORDER -- OPTIMIZED CLEARING END
pcd_arp_notes_l rs.b 0
pcd_arp_note_1_b rs.b 1
pcd_arp_note_2_b rs.b 1
pcd_arp_note_3_b rs.b 1
rs.b 1 ; gets cleared
pcd_last_trigger_pos_w rs.w 1 ; I think this makes sure that we don't retrigger the note on same pos
pcd_pat_portamento_speed_b rs.b 1
pcd_pat_adsr_rel_delay_b rs.b 1 ; counts down until adsr release. Seems unused?
pcd_note_off_delay_b rs.b 1 ; time before note is released ($ff = disabled)
pcd_inst_pattern_steps_b rs.b 1 ; number of steps in instrument pattern
pcd_note_delay_b rs.b 1 ; $ff = no note delay
pcd_track_delay_steps_b rs.b 1 ; $00 = no track delay, $ff = stop track delay (this is for the next channel!)
pcd_track_delay_vol16_b rs.b 1
pcd_track_init_delay_b rs.b 1 ; number of frames to ignore the delay
pcd_inst_num4_w rs.w 1 ; current instrument number * 4
;pcd_inst_new_step_w rs.w 1 ; seems to be unused
pcd_inst_subloop_wait_w rs.w 1
pcd_inst_loop_offset_w rs.w 1
pcd_inst_info_ptr rs.l 1 ; pointer to currently active instrument
pcd_waveinfo_ptr rs.l 1 ; pointer to currently active waveinfo
pcd_channel_mask_b rs.b 1
pcd_channel_num_b rs.b 1
pcd_adsr_phase_w rs.w 1 ; 0=attack, 1=decay, 2=sustain, 3=release ! do not change order
pcd_adsr_volume_w rs.w 1 ; 0 for restart / $400 (word only) ! do not change order
pcd_adsr_phase_speed_b rs.b 1
pcd_inst_ping_pong_dir_b rs.b 1 ; direction of ping-pong (-1 == $00 / +1 = $ff)
pcd_adsr_pos_w rs.w 1 ; pos in adsr curve
pcd_adsr_vol64_w rs.w 1 ; some adsr volume
pcd_new_inst_num_b rs.b 1 ; load new instrument (number) ! do not change order
rs.b 1 ; gets cleared
pcd_vibrato_pos_w rs.w 1 ;
pcd_vibrato_delay_w rs.w 1 ; is a byte value ! do not change order
pcd_vibrato_depth_w rs.w 1 ; is a byte value ! do not change order
pcd_vibrato_speed_w rs.w 1 ; is a byte value ! do not change order
pcd_adsr_release_b rs.b 1 ; is a byte value ! do not change order
rs.b 1 ; padding will be overwritten!
pcd_out_base rs.b ocd_SIZEOF
pcd_track_delay_buffer rs.b MAX_TRACK_DELAY*ocd_SIZEOF
pcd_SIZEOF rs.b 0
pcd_out_ptr_l = pcd_out_base+ocd_sam_ptr
pcd_out_len_w = pcd_out_base+ocd_length
pcd_out_lof_w = pcd_out_base+ocd_loop_offset
pcd_out_per_w = pcd_out_base+ocd_period
pcd_out_vol_b = pcd_out_base+ocd_volume
pcd_out_trg_b = pcd_out_base+ocd_trigger
pcd_out_unused_l = pcd_out_base+ocd_unused ; copied for track delay, but not used?
rsreset
owb_saw_waves rs.b 128
owb_sqr_waves rs.b 128
owb_tri_waves rs.b 128
owb_wave_length rs.b 1
owb_SIZEOF rs.b 0
; ----------------------------------------
; MyPlayer global variables (not bound to channel)
rsreset
; DO NOT CHANGE ORDER -- OPTIMIZED INIT
pv_pat_curr_row_b rs.b 1 ; current step
pv_next_pat_row_b rs.b 1
pv_next_pat_pos_b rs.b 1
pv_pat_speed_even_b rs.b 1 ; even shuffle speed
pv_pat_speed_odd_b rs.b 1 ; odd shuffle speed
pv_pat_line_ticks_b rs.b 1
pv_pat_stopped_b rs.b 1 ; 0 = stop, $ff = run
pv_songend_detected_b rs.b 1
; DO NOT CHANGE ORDER -- OPTIMIZED INIT END
pv_loop_pattern_b rs.b 1 ; repeat current pattern, do not advance
rs.b 1 ; padding
pv_trigger_mask_w rs.w 1
pv_my_song rs.l 1
pv_sample_buffer_ptr rs.l 1 ; pointer to start of sample buffer
pv_copperlist_ptr rs.l 1
pv_wave_sample_table rs.l MAX_WAVES ; 24 pointers to sample starts
pv_period_table rs.w 16*NOTES_IN_OCTAVE*3
; --- 127 byte displacement limit ---
pv_channeldata rs.b NUM_CHANNELS*pcd_SIZEOF
IFNE PRETRACKER_VOLUME_TABLE
pv_osc_buffers rs.b 0 ; reuse space of volume_table, which is bigger than NOTES_IN_OCTAVE*owb_SIZEOF
pv_volume_table rs.b (MAX_VOLUME+1)*MAX_VOLUME*2
ELSE
pv_osc_buffers rs.b NOTES_IN_OCTAVE*owb_SIZEOF
ENDC
pv_precalc_sample_size rs.l 1
pv_precalc_progress_ptr rs.l 1
pv_wg_wave_ord_num_w rs.w 1
IFNE PRETRACKER_PARANOIA_MODE ; same wave for mixing cannot be selected in Pretracker
pv_wg_curr_wave_num_b rs.b 1
rs.b 1
ENDC
pv_wg_curr_sample_ptr rs.l 1
pv_wg_curr_samend_ptr rs.l 1
pv_wg_curr_sample_len_w rs.w 1
pv_wg_chord_note_num_b rs.b 1 ; don't change order
pv_wg_unisono_run_b rs.b 1 ; don't change order
pv_wg_chord_flag_w rs.w 1
pv_wg_chord_pitches rs.l 1
pv_wg_osc_speed_l rs.l 1
pv_wg_flt_taps rs.w 4
pv_SIZEOF rs.b 0
;--------------------------------------------------------------------
include "../includes/hardware/custom.i"
include "../includes/hardware/dmabits.i"
CLIPTO8BIT MACRO
cmpi.w #-$80,\1
bge.s .nominclip\@
moveq.l #-$80,\1
.nominclip\@
cmpi.w #$7F,\1
ble.s .nomaxclip\@
moveq.l #$7F,\1
.nomaxclip\@
ENDM
CLIPORTRUNC8BIT MACRO
beq.s .unboosted\@
asr.l #6,\1
cmpi.w #-$80,\1
bge.s .nominclip\@
moveq.l #-$80,\1
.nominclip\@
cmpi.w #$7F,\1
ble.s .nomaxclip\@
moveq.l #$7F,\1
bra.s .nomaxclip\@
.unboosted\@
asr.l #8,\1
.nomaxclip\@
ENDM
CLIPTO8BITAFTERADD MACRO
bvc.s .noclip\@
spl \1
eor.b #$7f,\1
.noclip\@
ENDM
;--------------------------------------------------------------------
; Code starts here
IFNE PRETRACKER_KEEP_JUMP_TABLE
pre_FuncTable
dc.l pre_SongInit-pre_FuncTable
dc.l pre_PlayerInit-pre_FuncTable
dc.l pre_PlayerTick-pre_FuncTable
ENDC
;dc.b '$VER: Raspberry Casket 1.1',0
;even
IFNE PRETRACKER_COPPER_OUTPUT
;********************************************************************
;--------------------------------------------------------------------
; pre_PrepareCopperlist - initialize copperlist for replaying
;
; a0.l = copperlist (37 longwords for 4 channels, 5+8*NUM_CHANNELS)
; d0.w = rasterline (<239 or >=256)
; out: a0 = copperlist ptr after init
;********************************************************************
pre_PrepareCopperlist:
IFNE PRETRACKER_DONT_TRASH_REGS
movem.l d2-d3/d6/d7,-(sp)
ENDC
moveq.l #-2,d1
lsl.w #8,d0
move.b #$07,d0
move.w d0,(a0)+
move.w d1,(a0)+
move.l #(dmacon<<16)|DMAF_AUDIO,(a0)+
add.w #$500,d0
move.w d0,(a0)+
move.w d1,(a0)+
; writing 5*4 = 20 words
move.w #aud0+ac_ptr,d2
moveq.l #0,d3
moveq.l #NUM_CHANNELS-1,d7
.chloop moveq.l #5-1,d6
.dloop move.w d2,(a0)+
move.w d3,(a0)+
addq.w #2,d2
dbra d6,.dloop
addq.w #ac_SIZEOF-ac_dat,d2
dbra d7,.chloop
move.l #(dmacon<<16)|DMAF_SETCLR,(a0)+
add.w #$500,d0
move.w d0,(a0)+
move.w d1,(a0)+
; writing 2*4 = 12 words
move.w #aud0+ac_ptr,d2
moveq.l #NUM_CHANNELS-1,d7
.chloop2
moveq.l #3-1,d6
.dloop2 move.w d2,(a0)+
move.w d3,(a0)+
addq.w #2,d2
dbra d6,.dloop2
add.w #ac_SIZEOF-ac_per,d2
dbra d7,.chloop2
IFNE PRETRACKER_DONT_TRASH_REGS
movem.l (sp)+,d2-d3/d6/d7
ENDC
rts
ENDC
;********************************************************************
;--------------------------------------------------------------------
; SongInit - initialize data structure belonging to a song
;
; In:
; - a0: MyPlayer structure (unused)
; - a1: MySong structure (must be sv_SIZEOF bytes!)
; - a2: Pretracker song data
; Out:
; - d0: chipmemory (bytes) required for samples or 0 on error
;********************************************************************
pre_SongInit:
IFNE PRETRACKER_DONT_TRASH_REGS
movem.l d2/d7/a2-a5,-(sp)
ENDC
moveq.l #0,d0
move.l $0000(a2),d1
move.b d1,d2
clr.b d1
cmpi.l #$50525400,d1 ; "PRE"-Text
bne .error
moveq.l #MAX_INSTRUMENTS-1,d7 ; notice there's one extra name (available in 1.5, but not usable)!
IFNE PRETRACKER_SUPPORT_V1_5
cmpi.b #$1e,d2
bgt .error
bne.s .nopatchv15
move.l $005c(a2),d0 ; make song backward compatible
ror.w #8,d0
move.l d0,$003c(a2)
addq.l #8,$0004(a2) ; skip over first pattern data
moveq.l #2*MAX_INSTRUMENTS-1,d7 ; v1.5 has 32 slots (the other ones used for sfx)
.nopatchv15
ELSE
cmpi.b #$1b,d2
bgt .error
ENDC
move.l a1,a0
move.w #sv_SIZEOF,d0
bsr pre_MemClr
move.b $003c(a2),sv_pat_restart_pos_w+1(a1) ; song restart pos
move.b $003e(a2),sv_pat_pos_len_w+1(a1) ; songlength in pattern positions
move.b $003f(a2),sv_num_steps_b(a1) ; number of steps!
move.b $0041(a2),sv_num_waves_b(a1) ; number of instruments
move.l $0004(a2),d0
add.l a2,d0
move.l d0,sv_pos_data_adr(a1) ; address to position data (POSD)
move.l $0008(a2),d0
add.l a2,d0
move.l d0,sv_patterns_ptr(a1) ; address to pattern data (PATT)
move.l $000c(a2),d0 ; offset into instrument names
lea (a2,d0.l),a0 ; instrument names
.instrnamesloop
moveq.l #23-1,d0 ; max 23 chars
.inststrloop
tst.b (a0)+
dbeq d0,.inststrloop
dbra d7,.instrnamesloop
moveq.l #0,d7
move.b $0040(a2),d7 ; number of instruments
IFNE PRETRACKER_PARANOIA_MODE
beq.s .noinstsskip
ENDC
move.l d7,d0
lsl.w #3,d0
add.l a0,d0 ; skip 8 bytes of info per instrument (ININ)
lea sv_inst_patterns_table(a1),a3
lea sv_inst_infos_table(a1),a4
IFNE PRETRACKER_SUPPORT_V1_5
cmp.w #MAX_INSTRUMENTS,d7
ble.s .notruncto32
moveq.l #MAX_INSTRUMENTS,d7
.notruncto32
ENDC
subq.w #1,d7
.instinfoloop
move.l d0,(a3)+
moveq.l #0,d1
move.b (a0)+,d1 ; ii_vibrato_delay
lea pre_vib_delay_table(pc),a5
move.b (a5,d1.w),d1
addq.w #1,d1
move.w d1,uii_vibrato_delay(a4)
moveq.l #0,d1
move.b (a0)+,d1 ; ii_vibrato_depth
move.b pre_vib_depth_table-pre_vib_delay_table(a5,d1.w),uii_vibrato_depth+1(a4)
move.b (a0)+,d1 ; ii_vibrato_speed
move.b pre_vib_speed_table-pre_vib_delay_table(a5,d1.w),d1
muls uii_vibrato_depth(a4),d1 ; bake in this strange vibrato stuff
asr.w #4,d1
move.w d1,uii_vibrato_speed(a4)
moveq.l #0,d1
move.b (a0)+,d1 ; ii_adsr_attack
add.w d1,d1
move.w pre_fast_roll_off_16-pre_vib_delay_table(a5,d1.w),d1
move.w d1,uii_adsr_attack(a4)
moveq.l #0,d1
move.b (a0)+,d1 ; ii_adsr_decay
move.b pre_ramp_up_16-pre_vib_delay_table(a5,d1.w),uii_adsr_decay+1(a4)
move.b (a0)+,d1 ; ii_adsr_sustain
; what is this? a patch?
cmp.b #15,d1
bne.s .dont_patch_sustain
moveq.l #16,d1
.dont_patch_sustain
lsl.w #6,d1
move.w d1,uii_adsr_sustain(a4)
moveq.l #0,d1
move.b (a0)+,d1 ; ii_adsr_release
move.b pre_ramp_up_16-pre_vib_delay_table(a5,d1.w),uii_adsr_release(a4)
move.b (a0)+,d1 ; ii_pattern_steps
move.b d1,uii_pattern_steps(a4)
add.l d1,d0
add.l d1,d0
add.l d1,d0 ; calc next start address
lea uii_SIZEOF(a4),a4
dbra d7,.instinfoloop
.noinstsskip
move.l $0010(a2),d0 ; offset into wave names
lea (a2,d0.l),a0
moveq.l #MAX_WAVES-1,d7
.wavenamesloop
moveq.l #23-1,d0 ; max 23 chars
.wavestrloop
tst.b (a0)+
dbeq d0,.wavestrloop
dbra d7,.wavenamesloop
move.l a0,d0
lsr.w #1,d0
bcc.s .addressiseven
addq.l #1,a0 ; make address even
.addressiseven
move.l a0,sv_waveinfo_ptr(a1)
lea sv_wavegen_order_table(a1),a0
cmpi.b #$19,d2 ; check if version is higher than 19
bhi.s .haswaveorderinfo
moveq.l #0,d0 ; fill 24 bytes with default order of waves?
moveq.l #MAX_WAVES-1,d7
.fillcount
move.b d0,(a0)+
addq.b #1,d0
dbra d7,.fillcount
bra.s .contafterworkaround
.haswaveorderinfo
moveq.l #(MAX_WAVES/4)-1,d7
lea $0042(a2),a2 ; offset into wave ordering
.memcpyloop
move.l (a2)+,(a0)+
dbra d7,.memcpyloop
.contafterworkaround
moveq.l #2,d0 ; at least empty sample
moveq.l #0,d7
move.b sv_num_waves_b(a1),d7 ; has instruments?
IFNE PRETRACKER_PARANOIA_MODE
beq.s .hasnoinstruments
ENDC
move.l sv_waveinfo_ptr(a1),a3
subq.w #1,d7
.wavetableloop
moveq.l #0,d1
move.b wi_sam_len_b(a3),d1
addq.w #1,d1
lsl.w #7,d1
move.l d1,sv_wavelength_table-sv_waveinfo_table(a1)
btst #2,wi_flags_b(a3)
beq.s .onlythreeocts
mulu #15,d1
lsr.l #3,d1 ; * 1.875
.onlythreeocts
move.l d1,sv_wavetotal_table-sv_waveinfo_table(a1)
move.l a3,(a1)+
add.l d1,d0
lea wi_SIZEOF(a3),a3
dbra d7,.wavetableloop
; d0 will contain the size of the samples
.hasnoinstruments
.error
.exit
IFNE PRETRACKER_DONT_TRASH_REGS
movem.l (sp)+,d2/d7/a2-a5
ENDC
rts
;********************************************************************
;--------------------------------------------------------------------
; PlayerInit - initialize player and calculate samples
;
; In:
; - a0: MyPlayer (must have size of pv_SIZEOF, will be initialized)
; - a1: sample buffer
; - a2: MySong (must have been filled with SongInit before!)
; - a3: pointer to a longword for the progress of samples bytes generated (or null)
;********************************************************************
pre_PlayerInit:
IFNE PRETRACKER_DONT_TRASH_REGS
movem.l d2-d7/a2-a6,-(sp)
ENDC
move.l a0,a4
move.l a2,a6
move.w #pv_SIZEOF,d0
bsr pre_MemClr ; keeps a1 unchanged!
; ----------------------------------------
; proposed register assignment:
; a0 = sample output / scratch
; a1 = scratch
; a3 = waveinfo
; a4 = MyPlayer
; a6 = MySong
move.l a2,pv_my_song(a4)
IFNE PRETRACKER_PROGRESS_SUPPORT
move.l a3,pv_precalc_progress_ptr(a4)
ENDC
move.l a1,pv_sample_buffer_ptr(a4)
IFNE PRETRACKER_PARANOIA_MODE
beq.s .hasnosamplebuffer ; PARANOIA
ENDC
clr.w (a1)+ ; empty sample
moveq.l #0,d7
move.b sv_num_waves_b(a6),d7
IFNE PRETRACKER_PARANOIA_MODE
beq.s .hasnosamplebuffer ; PARANOIA
ENDC
lea pv_wave_sample_table(a4),a0
lea sv_wavetotal_table(a6),a3
subq.w #1,d7
.samplestartsloop
move.l a1,(a0)+ ; write sample start pointer
adda.l (a3)+,a1
dbra d7,.samplestartsloop
.hasnosamplebuffer
; ----------------------------------------
lea pre_period_table(pc),a0
lea pv_period_table(a4),a1
.calcperiodtable
; fill the missing entries in the period table by interpolating
moveq.l #3*NOTES_IN_OCTAVE-1,d7
.periodtableloop
move.w (a0)+,d0
move.w (a0),d1
sub.w d0,d1
swap d0
clr.w d0
swap d1
clr.w d1
asr.l #4,d1
moveq.l #16-1,d6
.perfineipolloop
swap d0
move.w d0,(a1)+
swap d0
add.l d1,d0
dbra d6,.perfineipolloop
dbra d7,.periodtableloop
; ----------------------------------------
moveq.l #0,d0
move.b sv_num_steps_b(a6),d0
move.w d0,d1
add.w d0,d0
add.w d1,d0 ; *3 bytes per pattern line
move.l sv_patterns_ptr(a6),a3
lea sv_pattern_table(a6),a0
move.w #255-1,d7 ; FIXME we should use the number of patterns instead?
.pattableloop
move.l a3,(a0)+
add.w d0,a3
dbra d7,.pattableloop
; ----------------------------------------
move.l #$00ffff06,pv_pat_curr_row_b(a4) ; pattern frame = 0, line = $ff, pattern pos = $ff, speed_even = 0
move.l #$06060100,pv_pat_speed_odd_b(a4) ; and pv_pat_line_ticks_b, pv_pat_stopped_b, pv_songend_detected_b
move.l sv_waveinfo_ptr(a6),a1
lea pv_channeldata(a4),a0
moveq.l #NUM_CHANNELS-1,d7
moveq.l #0,d0
.chaninitloop2
move.b #MAX_VOLUME,pcd_pat_vol_b(a0)
st pcd_track_delay_offset_b(a0)
move.l a1,pcd_waveinfo_ptr(a0)
move.w #3,pcd_adsr_phase_w(a0)
move.l pv_sample_buffer_ptr(a4),pcd_out_ptr_l(a0)
move.w #2,pcd_out_len_w(a0)
move.w #$7B,pcd_out_per_w(a0)
move.b d0,pcd_channel_num_b(a0)
bset d0,pcd_channel_mask_b(a0)
addq.b #1,d0
lea pcd_SIZEOF(a0),a0
dbra d7,.chaninitloop2
; ----------------------------------------
lea pre_log12_table(pc),a0 ; 128, 121, 114, 107, 102, 96, 90, 85, 80, 76, 72, 67
lea pv_osc_buffers+owb_sqr_waves(a4),a1
moveq.l #NOTES_IN_OCTAVE-1,d7
.noteloop
moveq.l #0,d6
move.l d6,a3 ; tabpos
move.b (a0)+,d6 ; period
move.b d6,owb_wave_length-owb_sqr_waves(a1)
move.l #$ff00,d5
divu d6,d5 ; frac increment
move.w d6,d4
lsr.w #1,d4 ; half-period
move.w d4,d3
lsr.w #1,d3 ; quarter-period
moveq.l #0,d0 ; acc
lea (a1,d6.w),a2
lea owb_tri_waves-owb_sqr_waves(a2),a2
suba.w d3,a2
.notewaveloop
move.w d0,d2
lsr.w #8,d2
moveq.l #$7f,d1
sub.b d2,d1
move.b d1,owb_saw_waves-owb_sqr_waves(a1,a3.w)
add.b d2,d2
cmp.w a3,d3 ; tabpos == negquarter
bne.s .nowrapback
suba.w d6,a2 ; go back to start of period
.nowrapback
cmp.w d0,a3
ble.s .otherhalf
moveq.l #$7f,d1
sub.b d2,d1
move.b d1,(a2)+
bra.s .clip80
.otherhalf
add.b #$80,d2
move.b d2,(a2)+
moveq.l #$7f,d2
cmp.w a3,d4
bne.s .noclip80
.clip80 moveq.l #-$80,d2
.noclip80
move.b d2,owb_sqr_waves-owb_sqr_waves(a1,a3.w)
add.w d5,d0 ; increment acc by frac
addq.w #1,a3 ; increment pos
cmp.w a3,d6
bne.s .notewaveloop
lea owb_SIZEOF(a1),a1
dbra d7,.noteloop
; ----------------------------------------
.audioinit
bset #1,$BFE001 ; filter off
IFNE PRETRACKER_PARANOIA_MODE
tst.b sv_num_waves_b(a6)
beq .earlyexit ; PARANOIA
ENDC
; ----------------------------------------
; proposed register assignment:
; a0 = sample output
; a1 = scratch
; a3 = songinfo
; a4 = MyPlayer
; a6 = MySong / waveinfo
.wavegenloop
;movea.l pv_my_song(a4),a6
lea sv_wavegen_order_table(a6),a1
move.w pv_wg_wave_ord_num_w(a4),d0 ; counter through all 24 waves
moveq.l #0,d1
move.b (a1,d0.w),d1 ; apply wave order redirection
IFNE PRETRACKER_PARANOIA_MODE ; same wave for mixing cannot be selected in Pretracker
move.b d1,pv_wg_curr_wave_num_b(a4)
ENDC
lsl.w #2,d1
move.l pv_wave_sample_table(a4,d1.w),a0
move.l a0,pv_wg_curr_sample_ptr(a4)
add.w #sv_wavelength_table,d1
adda.w d1,a6
move.l sv_wavelength_table-sv_wavelength_table(a6),d0
move.w d0,pv_wg_curr_sample_len_w(a4)
IFNE PRETRACKER_PROGRESS_SUPPORT
move.l sv_wavetotal_table-sv_wavelength_table(a6),pv_precalc_sample_size(a4)
ENDC
move.l sv_waveinfo_table-sv_wavelength_table(a6),a3
; clear sample data (a0 and d0 from above)
bsr pre_MemClr
move.l a0,pv_wg_curr_samend_ptr(a4)
; read out chord information
lea wi_chord_note1_b(a3),a1
move.b (a1)+,d1
move.b (a1)+,d2
move.b (a1)+,d3
moveq.l #0,d4
move.b d1,d4
or.b d2,d4
or.b d3,d4
seq d4
neg.b d4
move.w d4,pv_wg_chord_flag_w(a4) ; has chord flag (0/1)
move.b wi_osc_basenote_b(a3),d0
add.b d0,d1
add.b d0,d2
add.b d0,d3
lea pv_wg_chord_pitches(a4),a1
move.b d0,(a1)+
move.b d1,(a1)+
move.b d2,(a1)+
move.b d3,(a1)+
suba.l a5,a5
clr.w pv_wg_chord_note_num_b(a4) ; and pv_wg_chord_note_num_b
.wavegen_chordloop
lea pv_wg_chord_pitches(a4),a1
moveq.l #0,d1
move.b pv_wg_chord_note_num_b(a4),d1 ; chord note counter
move.b (a1,d1.w),d0 ; get chord note
ext.w d0
tst.w d1
beq.s .base_note_is_never_skipped ; is base note?
cmp.b wi_osc_basenote_b(a3),d0
beq .wave_gen_tone_done ; skip chord notes that are same as base note
.base_note_is_never_skipped
moveq.l #0,d1
moveq.l #NOTES_IN_OCTAVE,d2
move.w d0,d1
move.w d0,a2 ; save base note, used later (much later, noise generator)!
add.w #NOTES_IN_OCTAVE*NOTES_IN_OCTAVE,d1 ; make sure we don't run into negative modulo
divu d2,d1
swap d1
move.w d1,d0 ; note within octave
swap d1
sub.w d2,d1 ; restore octave
mulu #owb_SIZEOF,d0
lea (a4,d0.w),a1
lea pv_osc_buffers+owb_saw_waves(a1),a6
moveq.l #3,d0
and.b wi_flags_b(a3),d0
beq.s .osc_selected
lea owb_tri_waves-owb_saw_waves(a6),a6
subq.b #1,d0
beq.s .osc_selected
lea owb_sqr_waves-owb_tri_waves(a6),a6
subq.b #1,d0
beq.s .osc_selected
suba.l a6,a6 ; noise selected
.osc_selected
move.l #$8000,d6
move.w d1,d0 ; check octave shift
bgt.s .shiftleft
beq.s .contshift
;.shiftright
move.w d0,d3
neg.w d3
asr.l d3,d6
bra.s .contshift
.shiftleft
lsl.l d0,d6
.contshift
; ----------------------------------------
; pitch ramp
move.b wi_pitch_ramp_b(a3),d3
ext.w d3
ext.l d3
btst #4,wi_flags_b(a3) ; pitch linear flag
beq.s .pitch_not_linear
tst.b d3
bgt.s .pitch_ramp_positive
lsl.l d0,d3
add.l d3,d3
bra.s .pitch_ramp_cont
.pitch_not_linear
tst.b d3
ble.s .pitch_ramp_cont
.pitch_ramp_positive
muls d3,d3
.pitch_ramp_cont
move.l d3,d2
lsl.l #8,d2
lsl.l #2,d2
moveq.l #0,d7
move.b pv_osc_buffers+owb_wave_length(a1),d7 ; get period
moveq.l #15,d5
lsl.l d5,d7
sub.w d0,d5 ; 15-octave
lsl.w #3,d5
moveq.l #0,d3
move.b wi_osc_phase_min_b(a3),d3
mulu d5,d3
lsl.l #6,d3
moveq.l #0,d0
move.b wi_osc_phase_max_b(a3),d0
mulu d5,d0
lsl.l #6,d0
move.l d0,a5
moveq.l #0,d5
move.b wi_osc_phase_spd_b(a3),d5
lsl.l #8,d5
lsl.l #3,d5
cmp.l d3,d0
bge.s .osc_with_positive_phase_speed
neg.l d5
movea.l d3,a5
bra.s .osc_continue
.osc_with_positive_phase_speed
move.l d3,d0
.osc_continue
move.l d0,pv_wg_osc_speed_l(a4)
; d0 = d6 * chord_shift * chordnum + d6 * phase_min = d6 * (chord_shift * chordnum + phase_min)
moveq.l #0,d4
move.b wi_chord_shift_b(a3),d4
move.w pv_wg_chord_flag_w(a4),d0
add.b pv_wg_chord_note_num_b(a4),d0
mulu d0,d4
moveq.l #0,d0
move.b wi_osc_phase_min_b(a3),d0
add.w d0,d4
move.l d6,d0
lsr.l #4,d0
lsl.l #4,d4
mulu d4,d0
cmp.l d7,d0
ble.s .lbC002516
.lbC002510
sub.l d7,d0
cmp.l d7,d0
bgt.s .lbC002510
.lbC002516
move.l a6,d4
bne .no_noise
; ----------------------------------------
; d1 = octave
; d5 = osc phase speed
; a2 = base note
.gen_noise
IFNE PRETRACKER_PARANOIA_MODE
tst.w pv_wg_curr_sample_len_w(a4)
beq .wave_gen_tone_done
ENDC
move.l #$8000,d5
move.l d5,a5
tst.w d1
bge.s .gen_noise_positive_octave
.gen_noise_negative_octave
moveq.l #NOTES_IN_OCTAVE,d1
move.l a2,d3
neg.l d3
move.l d3,d0
divu d1,d0
IFNE PRETRACKER_PARANOIA_MODE
bvs.s .divisionoverflow
ENDC
addq.w #1,d0
lsr.w d0,d5
moveq.l #0,d0
move.w d5,d0
divu d1,d0
moveq.l #0,d4
move.w d0,d4
IFNE PRETRACKER_PARANOIA_MODE
bra.s .returnfromoverflow
.divisionoverflow
move.l #$AAA,d4 ; some dummy value, I would expect
.returnfromoverflow
ENDC
moveq.l #0,d0
move.w d3,d0
.cheap_mod12
sub.w d1,d0
bpl.s .cheap_mod12
neg.w d0
mulu d4,d0
add.l d0,d5
.gen_noise_positive_octave
moveq.l #0,d0
move.b wi_osc_phase_min_b(a3),d0
moveq.l #0,d1
move.b wi_chord_shift_b(a3),d1
add.w d1,d0
addq.w #1,d0
movea.l d5,a1
movea.l pv_wg_curr_sample_ptr(a4),a0
moveq.l #0,d1
moveq.l #0,d6
moveq.l #0,d3
move.b wi_osc_gain_b(a3),d3
.gen_noise_outerloop
move.w d0,d1 ; random noise generator
lsl.w #8,d1
lsl.w #5,d1
eor.w d1,d0
move.w d0,d1
lsr.w #8,d1
lsr.w #1,d1
eor.w d1,d0
move.w d0,d1
lsl.w #7,d1
eor.w d1,d0
cmpa.l #$8000,a5 ; if symmetrical
beq.s .gen_noise_centered
; what does this do? (a5 - $8000) (a5 +$7fff)&$8000
move.l a5,d4
addi.l #$FFFF8000,d4
move.l a5,d1
addi.l #$FFFF7FFF,d1
andi.w #$8000,d1
movea.l d4,a5
suba.l d1,a5
.gen_noise_centered
move.b d0,d1
ext.w d1
move.w d3,d4
muls d1,d4
move.l d4,d1
asr.l #7,d1
CLIPTO8BIT d1
add.b (a0),d1
.gen_noise_innerloop
move.b d1,(a0)+
adda.l a1,a5
tst.l d2
beq.s .lbC002BAC
add.l d2,d6
move.l d6,d4
asr.l #8,d4
asr.l #2,d4
add.l d5,d4
movea.l d4,a1
btst #4,wi_flags_b(a3) ; pitch linear flag
beq.s .noise_nonlinear_pitch
move.l d2,d4
asr.l #7,d4
sub.l d4,d2
.noise_nonlinear_pitch
cmpa.w #$1FF,a1
bgt.s .lbC002BAC
moveq.l #0,d2
move.l d2,a5
;suba.l a5,a5
movea.w #$200,a1
.lbC002BAC
cmpa.l pv_wg_curr_samend_ptr(a4),a0
beq .wave_gen_tone_done
cmpa.w #$7FFF,a5
ble.s .gen_noise_innerloop
bra .gen_noise_outerloop
.no_noise
move.l d6,d1
tst.b pv_wg_unisono_run_b(a4)
beq.s .is_not_in_unisono
moveq.l #3<<3,d1
and.b wi_mod_density_b(a3),d1
lsr.b #3,d1
moveq.l #9,d4
sub.b d1,d4
move.l d6,d1
asr.l d4,d1
add.l d6,d1
.is_not_in_unisono
IFNE PRETRACKER_PARANOIA_MODE
tst.w pv_wg_curr_sample_len_w(a4)
beq .wave_gen_tone_done
ENDC
; ----------------------------------------
; chord gen
; in: d0/d1/d2/d3/d5/d7
; in: a0
move.l d3,a1
movea.l pv_wg_curr_sample_ptr(a4),a0
suba.l a2,a2
.chordtoneloop
move.l d0,d4
sub.l a1,d4
bpl.s .noclip_osc_phase
moveq.l #0,d4
.noclip_osc_phase
asr.l #8,d4
asr.l #7,d4
move.b (a6,d4.w),d4 ; fetch precalced sample
ext.w d4
moveq.l #0,d3
move.b wi_osc_gain_b(a3),d3
muls d4,d3
asr.w #7,d3
move.b (a0),d4
ext.w d4
add.w d3,d4
CLIPTO8BIT d4
move.b d4,(a0)+
add.l d1,d0
cmp.l d7,d0
blt.s .lbC0025A2
sub.l d7,d0
add.l d5,a1
cmp.l a5,a1
blt.s .lbC00259A
neg.l d5
move.l a5,a1
.lbC00259A
cmp.l pv_wg_osc_speed_l(a4),a1
bgt.s .lbC0025A2
neg.l d5
move.l pv_wg_osc_speed_l(a4),a1
.lbC0025A2
tst.l d2
beq.s .chordtone_done
adda.l d2,a2
move.l a2,d1
asr.l #8,d1
asr.l #2,d1
add.l d6,d1
btst #4,wi_flags_b(a3) ; pitch linear flag
beq.s .no_linear_pitch
move.l d2,d4
asr.l #7,d4
sub.l d4,d2
.no_linear_pitch
cmp.l d7,d1
bcs.s .chordtone_done
moveq.l #0,d2
moveq.l #0,d1
.chordtone_done
cmpa.l pv_wg_curr_samend_ptr(a4),a0
bne.s .chordtoneloop
.wave_gen_tone_done
addq.b #1,pv_wg_chord_note_num_b(a4)
cmp.b #4,pv_wg_chord_note_num_b(a4)
bne .wavegen_chordloop
moveq.l #3<<3,d0
and.b wi_mod_density_b(a3),d0 ; unisono
beq.s .chords_done
move.l a6,d1
beq.s .chords_done
tst.b pv_wg_unisono_run_b(a4)
bne.s .chords_done
move.w #$0001,pv_wg_chord_note_num_b(a4) ; sets also pv_wg_unisono_run_b
bra .wavegen_chordloop
.chords_done
; ----------------------------------------
; filters
; proposed register assignment:
; a0 = sample output
; a1 = end of filter chunk
; a2 = filter func
; d7/a6 = filter start
; a4 = MyPlayer
; a5 = unused
; a3 = waveinfo
; d3/d4/d5/d6 = filter taps
; d0/d1/d7 = scratch
moveq.l #0,d0
move.b wi_flt_type_b(a3),d0
beq .filter_done
IFNE PRETRACKER_PARANOIA_MODE
tst.w pv_wg_curr_sample_len_w(a4)
beq .filter_done
ENDC
add.w d0,d0
lea .filterfunc_jmptable(pc),a2
add.w -1*2(a2,d0.w),a2
moveq.l #0,d4 ; filter tap values
moveq.l #0,d5 ; filter tap values
movem.l d4-d5,pv_wg_flt_taps(a4)
lea wi_flt_start_b(a3),a0
moveq.l #0,d0
move.b (a0)+,d0 ; wi_flt_start_b
lsl.w #8,d0
move.b (a0)+,d4 ; wi_flt_min_b
lsl.w #8,d4 ; flt_min*256
move.b (a0)+,d5 ; wi_flt_max_b
lsl.w #8,d5 ; flt_max*256
move.b (a0)+,d3 ; wi_flt_speed_b
ext.w d3
ext.l d3 ; flt_speed*128
lsl.l #7,d3
movea.l pv_wg_curr_sample_ptr(a4),a0
.entry_to_filter_loop
move.l d0,a6
move.l d3,d1 ; flt_speed_b*128
adda.l d1,a6 ; suppress M68kUnexpectedConditionalInstruction
bgt.s .filter_speed_pos
.filter_speed_neg
move.l d4,d1 ; flt_min*256
cmp.l d1,d0
blt.s .lbC002790
cmp.l d1,a6
bgt.s .lbC002936
move.l d1,a6
cmp.l d5,d1 ; flt_max*256
beq.s .filter_load_min
neg.l d3 ; flt_speed_b*128
bra.s .filter_load_min
.filterfunc_jmptable
dc.w .lowpassfilter-.filterfunc_jmptable
dc.w .highpassfilter-.filterfunc_jmptable
dc.w .bandpassfilter-.filterfunc_jmptable
dc.w .notchfilter-.filterfunc_jmptable
.lbC002790
tst.l d0
blt.s .lbC002936
move.l a6,d7
bgt.s .lbC002936
neg.l d3 ; flt_speed_b*128
move.w #$FF,d2
move.w d2,d1
sub.l a6,a6
bra.s .filter_cont
.filter_speed_pos
cmp.l d5,d0 ; flt_max*256
bgt.s .lbC002D2A
cmp.l d5,a6 ; flt_max*256
blt.s .lbC002936
move.l d4,d2 ; flt_min*256
cmp.l d5,d2 ; flt_max*256
beq.s .filter_load_max_no_flip
neg.l d3 ; flt_speed_b*128
move.l d5,a6 ; flt_max*256
bra.s .filter_load_max
.lbC002D2A
cmpi.l #$FF00,d0
bgt.s .lbC002936
cmp.l #$FEFF,a6
ble.s .lbC002936
neg.l d3 ; flt_speed_b*128
moveq.l #0,d2
move.l #$FF00,a6
bra.s .filter_cont
.lbC002936
move.w a6,d2
lsr.w #8,d2
not.b d2
bra.s .filter_cont
.filter_load_max_no_flip
movea.l d2,a6
.filter_load_max
moveq.l #0,d2
move.b wi_flt_max_b(a3),d2
not.b d2
bra.s .filter_cont
.filter_load_min
moveq.l #0,d2
move.b wi_flt_min_b(a3),d2
not.b d2
.filter_cont
btst #0,wi_flt_type_b(a3)
bne.s .not_notch_or_highpass
.highpass_or_notch ; entered for 2 or 4
not.b d2
.not_notch_or_highpass ; entered for 1 or 3
move.w d2,d0
add.w d0,d0
moveq.l #0,d7
move.b wi_flt_resonance_b(a3),d7
beq.s .filter_no_resonance
move.w d2,d1
ext.l d1
lsl.l #8,d1
moveq.l #$B6/2,d0
sub.w d7,d0
add.w d0,d0
cmpi.w #$36,d0
bge.s .filter_no_clip_resonance
moveq.l #$36,d0
.filter_no_clip_resonance
divu d0,d1
move.w d2,d0
add.w d1,d0
.filter_no_resonance
lea $40(a0),a1 ; end of sample chunk
movem.l d3-d5,-(sp)
movem.w pv_wg_flt_taps(a4),d3-d6
; d0/d2 relevant for inner loop
.filter_innerloop
move.b (a0),d1
ext.w d1
move.w d4,d7
sub.w d5,d7
muls d0,d7
asr.l #8,d7
sub.w d4,d7
add.w d1,d7
muls d2,d7
asr.l #8,d7
add.w d7,d4
move.w d4,d7
sub.w d5,d7
muls d2,d7
asr.l #8,d7
add.w d7,d5
move.w d5,d7
sub.w d6,d7
muls d2,d7
asr.l #8,d7
add.w d7,d6
move.w d6,d7
sub.w d3,d7
muls d2,d7
asr.l #8,d7
add.w d7,d3
move.w d3,d7
jmp (a2)
.highpassfilter
sub.w d1,d7
bra.s .filterclipresult
.bandpassfilter
sub.w d4,d7
sub.w d5,d7
sub.w d6,d7
asr.w #1,d7
bra.s .filterclipresult
.notchfilter
sub.w d4,d7
neg.w d7
.lowpassfilter
.filterclipresult
CLIPTO8BIT d7
.filter_outputbyte
move.b d7,(a0)+
cmpa.l a0,a1
bne.s .filter_innerloop
.filterloop_end_test
movem.w d3-d6,pv_wg_flt_taps(a4)
movem.l (sp)+,d3-d5
cmpa.l pv_wg_curr_samend_ptr(a4),a0
bhs.s .filter_done
move.l a6,d0
bra .entry_to_filter_loop
.filter_done
; ----------------------------------------
; Optional Pre-Modulator
btst #5,wi_mod_density_b(a3) ; post bit
bne.s .nopremodulator
bsr pre_Modulator
.nopremodulator
; ----------------------------------------
; start with volume envelope
; a0 = output sample buffer
; d0 = scratch (e.g. sample)
; d1 = increment for attack phase
; d3 = current volume for attack and decay phases
; d4 = remaining sample length - 1
; a3 = wave info
.vol_do_envelope
movea.l pv_wg_curr_sample_ptr(a4),a0 ; load buffer pointer
move.w pv_wg_curr_sample_len_w(a4),d4 ; load length
IFNE PRETRACKER_PARANOIA_MODE
beq .vol_envelope_finished ; paranoia
ENDC
subq.w #1,d4 ; we use length-1, <0 is end
moveq.l #2,d1 ; turns into $20000 through swap
moveq.l #0,d0
move.b wi_vol_attack_b(a3),d0
bne.s .has_attack_volume
cmpi.b #$FF,wi_vol_sustain_b(a3)
beq .vol_envelope_finished
; no attack but not full sustain -> go to delay
;move.l #$100<<16,d3
bra.s .vol_skip_attack
.vol_avoid_overflow_with_1
moveq.l #1,d1
bra.s .cont_vol_envelope
.has_attack_volume
moveq.l #0,d3
cmp.w #1,d0
beq.s .cont_vol_envelope
cmp.w #2,d0
beq.s .vol_avoid_overflow_with_1
swap d1 ; turn into $20000
divu d0,d1
swap d1
clr.w d1
; swap is done below
.cont_vol_envelope
swap d1 ; move to high word (should be max $20000 then)
btst #5,wi_flags_b(a3) ; vol fast flag
beq.s .vol_no_fast
lsl.l #4,d1 ; multiply speed by 16
.vol_no_fast
add.l d1,d3 ; increase volume
cmpi.l #$FFFFFF,d3
ble.s .vol_do_attack ; first step overshooting?
.vol_skip_attack
btst #3,wi_flags_b(a3) ; boost flag
bne.s .vol_delay_boosted
bra.s .vol_delay_normal
.vol_do_attack
btst #3,wi_flags_b(a3) ; boost flag
bne.s .vol_attack_boosted
; ----------------------------------------
; attack phase with volume boosted
.vol_attack_normal
.vol_attack_normal_loop
move.b (a0),d0
ext.w d0
swap d3
muls d3,d0
swap d3
asr.w #8,d0
move.b d0,(a0)+
subq.w #1,d4
bmi .vol_envelope_finished
add.l d1,d3 ; increase volume
cmpi.l #$FFFFFF,d3
ble.s .vol_attack_normal_loop
; ----------------------------------------
; delay phase (normal)
.vol_delay_normal ; moved this label two inst up, didn't make sense there
moveq.l #0,d0
move.b wi_vol_delay_b(a3),d0
lsl.w #4,d0
IFNE PRETRACKER_FASTER_CODE
; skip the delay -- we don't change the volume for this section
addq.w #1,d0
sub.w d0,d4
bmi .vol_envelope_finished
lea 1(a0,d0.w),a0
ELSE
lea 2(a0,d0.w),a1
move.w #$FF,d3 ; FIXME I don't think that this is quite right. Shouldn't the max volume NOT change the value?
.vol_delay_normal_loop
move.b (a0),d0
IFNE 1
ext.w d0
muls d3,d0
asr.w #8,d0
ELSE
; this should be the same as above (*(256-1))
spl d3
add.b d3,d0
ENDC
move.b d0,(a0)+
cmpa.l a1,a0
dbeq d4,.vol_delay_normal_loop
bne .vol_envelope_finished
ENDC
bra.s .vol_delay_end_reached
; ----------------------------------------
; attack with volume boosted
.vol_attack_boosted
.vol_attack_boosted_loop
move.b (a0),d0
ext.w d0
swap d3
muls d3,d0
swap d3
asr.w #6,d0
CLIPTO8BIT d0
move.b d0,(a0)+
subq.w #1,d4
bmi .vol_envelope_finished
add.l d1,d3
cmpi.l #$FFFFFF,d3
ble.s .vol_attack_boosted_loop
; ----------------------------------------
; delay with max volume boosted
.vol_delay_boosted
moveq.l #0,d0
move.b wi_vol_delay_b(a3),d0
lsl.w #4,d0
lea 2(a0,d0.w),a1
IFNE PRETRACKER_FASTER_CODE
.vol_delay_boosted_loop
move.b (a0),d0
add.b d0,d0
CLIPTO8BITAFTERADD d0
add.b d0,d0
CLIPTO8BITAFTERADD d0
ELSE
move.w #$FF,d3 ; FIXME I don't think that this is quite right. It should be $100 to boost by full volume
.vol_delay_boosted_loop
move.b (a0),d0
ext.w d0
muls d3,d0
asr.w #6,d0
CLIPTO8BIT d0
ENDC
move.b d0,(a0)+
cmpa.l a1,a0
dbeq d4,.vol_delay_boosted_loop
bne .vol_envelope_finished
.vol_delay_end_reached
subq.w #1,d4
; ----------------------------------------
; decay phase
; d0 = scratch
; d1 = current volume decrement
; d2 = table index boundary
; d3 = 16:16 decay pos
; d4 = sample length counter
; d5 = volume
; d6 = scratch
; d7 = decay increment
; a0 = sample pointer
; a1 = (current) roll off table pointer (points to upper bound)
; a2 = lower bound
.vol_do_decay
moveq.l #0,d3
move.b wi_vol_decay_b(a3),d3
beq .vol_do_sustain
move.w d3,d7
mulu d7,d7
lsr.w #2,d7
add.w d3,d7 ; d7 = (d3^2)/4+d3 (<= 16511)
btst #5,wi_flags_b(a3) ; vol fast flag
beq.s .vol_decay_not_fast
moveq.l #0,d3 ; will cause a5=$400,a2=$200,d2=0, decay value has no effect on the ramp used
.vol_decay_not_fast
lsl.w #8,d3
lsl.l #4,d3
move.l d3,d2
swap d2
lea pre_roll_off_table(pc),a1
add.w d2,d2
adda.w d2,a1
move.w (a1)+,a2 ; first index in table
lsr.w #1,d2 ; update next boundary
moveq.l #0,d1 ; current volume decrement
moveq.l #0,d5
not.w d5 ; set maximum volume
.vol_decay_normal_loop
add.l d7,d3 ; increment position in decay
swap d3
cmp.w #$8e,d3 ; pos in table where it makes no sense to do lerp anymore
bhi.s .vol_keep_voldec
cmp.w d2,d3
bls.s .vol_do_lerp
.vol_lerp_next_section
move.w (a1),a2
IFNE PRETRACKER_BUGFIX_CODE
lea pre_roll_off_table+2(pc),a1 ; Take the right boundary value
ELSE
lea pre_roll_off_table(pc),a1 ; This will set a wrong boundary and thus plateau the slope decay speed
ENDC
add.w d3,a1
add.w d3,a1
move.w d3,d2 ; update next boundary
.vol_do_lerp ; ((lowerbound-upperbound)*(d3<<8))>>8 + upperbound
move.w a2,d1
move.w (a1),d0
sub.w d1,d0 ; delta between lower and upper bound (negative value)
beq.s .vol_skip_lerp
swap d3
move.w d3,d1
lsr.w #8,d1
muls d0,d1
asr.l #8,d1
add.w a2,d1
swap d3
.vol_keep_voldec
.vol_skip_lerp
swap d3
sub.w d1,d5
bls.s .vol_do_sustain
move.w d5,d6
lsr.w #8,d6
cmp.b wi_vol_sustain_b(a3),d6
bls.s .vol_do_sustain
move.b (a0),d0
ext.w d0
muls d6,d0
btst #3,wi_flags_b(a3) ; boost flag
CLIPORTRUNC8BIT d0
move.b d0,(a0)+
dbra d4,.vol_decay_normal_loop
bra.s .vol_envelope_finished
; ----------------------------------------
; sustain phase
.vol_do_sustain
moveq.l #0,d3
move.b wi_vol_sustain_b(a3),d3
beq.s .vol_sustain_silence
btst #3,wi_flags_b(a3) ; boost flag
beq.s .vol_sustain_normal
.vol_sustain_boosted
.vol_sustain_boosted_loop
move.b (a0),d0
ext.w d0
muls d3,d0
asr.w #6,d0
CLIPTO8BIT d0
move.b d0,(a0)+
dbra d4,.vol_sustain_boosted_loop
bra.s .vol_envelope_finished
.vol_sustain_silence
moveq.l #0,d0
.vol_sustain_silence_loop
move.b d0,(a0)+
dbra d4,.vol_sustain_silence_loop
bra.s .vol_envelope_finished
.vol_sustain_normal
IFNE PRETRACKER_FASTER_CODE
cmp.b #$ff,d3
beq.s .vol_envelope_finished
ENDC
.vol_sustain_normal_loop
move.b (a0),d0
ext.w d0
muls d3,d0
asr.w #8,d0
move.b d0,(a0)+
dbra d4,.vol_sustain_normal_loop
.vol_envelope_finished
; ----------------------------------------
; Optional Post-Modulator
btst #5,wi_mod_density_b(a3) ; post bit
beq.s .nopostmodulator
bsr pre_Modulator
.nopostmodulator
; ----------------------------------------
; wave mixing (removed some code here that was doing nothing as result
; because below higher octaves code would overwrite it anyway).
movea.l pv_my_song(a4),a6
moveq.l #0,d0
move.b wi_mix_wave_b(a3),d0
beq.s .mix_no_wave_mixing ; no mixing selected
subq.b #1,d0
IFNE PRETRACKER_PARANOIA_MODE ; same wave for mixing cannot be selected in Pretracker
cmp.b pv_wg_curr_wave_num_b(a4),d0
beq .mix_no_wave_mixing ; same wave number!
ENDC
lsl.w #2,d0
move.l pv_wave_sample_table(a4,d0.w),a1
add.w #sv_wavelength_table,d0
move.l (a6,d0.w),d3
move.w pv_wg_curr_sample_len_w(a4),d4 ; length of the sample to mix to
cmp.w d3,d4
ble.s .mix_picked_shorter
move.w d3,d4
.mix_picked_shorter
move.l pv_wg_curr_sample_ptr(a4),a0
subq.w #1,d4
.mix_mixloop
move.b (a0),d0
add.b (a1)+,d0
CLIPTO8BITAFTERADD d0
move.b d0,(a0)+
dbra d4,.mix_mixloop
.mix_no_wave_mixing
; ----------------------------------------
; create higher octaves (this has been massively shortened)
btst #2,wi_flags_b(a3)
beq.s .oct_has_no_extra_octaves
moveq.l #0,d4
move.b wi_sam_len_b(a3),d4 ; length of the sample
IFNE PRETRACKER_PARANOIA_MODE
beq.s .oct_has_no_extra_octaves
ENDC
addq.w #1,d4
lsl.w #7,d4
movea.l pv_wg_curr_sample_ptr(a4),a1
lea (a1,d4.l),a0 ; needs to be .l due to 32678 max length
mulu #7,d4
lsr.l #3,d4 ; * 0.875
subq.w #1,d4
.oct_downsample_loop
move.b (a1),(a0)+
addq.l #2,a1
dbra d4,.oct_downsample_loop
.oct_has_no_extra_octaves
; ----------------------------------------
IFNE PRETRACKER_PROGRESS_SUPPORT
move.l pv_precalc_progress_ptr(a4),d0
beq.s .no_progress_out
move.l d0,a0
move.l pv_precalc_sample_size(a4),d0
add.l d0,(a0)
.no_progress_out
ENDC
move.b sv_num_waves_b(a6),d0
addq.b #1,pv_wg_wave_ord_num_w+1(a4)
cmp.b pv_wg_wave_ord_num_w+1(a4),d0
bgt .wavegenloop
; ----------------------------------------
IFNE PRETRACKER_VOLUME_TABLE
lea pv_volume_table(a4),a0
moveq.l #(MAX_VOLUME+1)-1,d7
moveq.l #0,d0
.vol_outerloop
moveq.l #MAX_VOLUME*2-1,d6
moveq.l #0,d1
.vol_innerloop
move.w d1,d2
lsr.w #6,d2
move.b d2,(a0)+
add.w d0,d1
dbra d6,.vol_innerloop
addq.w #1,d0
dbra d7,.vol_outerloop
ENDC
; ----------------------------------------
.earlyexit
IFNE PRETRACKER_DONT_TRASH_REGS
movem.l (sp)+,d2-d7/a2-a6
ENDC
rts
;--------------------------------------------------------------------
; a3: waveinfo
;
; d6: wetness (word)
; uses all data registers and a0-a2 (a3 unchanged)
pre_Modulator:
tst.b wi_mod_wetness_b(a3)
beq.s .earlyexit
moveq.l #7,d0
and.b wi_mod_density_b(a3),d0
bne.s .has_density
.earlyexit
rts
.has_density
move.l pv_wg_curr_sample_ptr(a4),a0
move.w pv_wg_curr_sample_len_w(a4),d4
IFNE PRETRACKER_PARANOIA_MODE
bne.s .not_zero
rts
.not_zero
ENDC
moveq.l #0,d6
move.b wi_mod_wetness_b(a3),d6
moveq.l #0,d7
lea pre_modulator_ramp_8(pc),a1
lea pre_ramp_up_down_32(pc),a2
.loop moveq.l #0,d5
move.b wi_mod_length_b(a3),d5
mulu (a1)+,d5 ; result is a long value
moveq.l #0,d1
move.b wi_mod_predelay_b(a3),d1
btst #5,wi_mod_density_b(a3) ; post-modulator?
bne.s .factor1_256
; factor 1/4 and 64
lsr.l #2,d5
lsl.l #6,d1
bra.s .cont
.factor1_256
lsl.w #8,d1
.cont add.l d1,d5 ; sum up length and predelay
moveq.l #0,d2
moveq.l #0,d3
.innerloop
add.w d7,d3
addq.w #8,d3
moveq.l #0,d1
move.w d3,d1
lsr.w #8,d1
lsr.w #3,d1
add.w d1,d1
move.w (a2,d1.w),d1 ; 4 bit value
add.l d5,d1
lsr.l #6,d1
move.w d2,d0
sub.w d1,d0
bmi.s .isneg
move.b (a0,d0.w),d1
ext.w d1
btst #0,d7
beq.s .keep_dc
neg.w d1
.keep_dc
muls d6,d1
asr.w #8,d1
add.b (a0,d2.w),d1
CLIPTO8BITAFTERADD d1
move.b d1,(a0,d2.w)
.isneg
addq.w #1,d2
cmp.w d4,d2
bcs.s .innerloop
.restartloop
addq.w #1,d7
moveq.l #7,d0
and.b wi_mod_density_b(a3),d0
cmp.w d0,d7
bcs.s .loop
rts
;--------------------------------------------------------------------
pre_MemClr
lsr.w #1,d0
subq.w #1,d0
bmi.s .skipmemclr
moveq.l #0,d1
.fillmemloop
move.w d1,(a0)+
dbra d0,.fillmemloop
.skipmemclr
rts
;********************************************************************
;--------------------------------------------------------------------
; PlayerTick - Play one frame of music (called every VBL)
;
; In:
; - a0: MyPlayer
; - a1: copperlist (if enabled)
;********************************************************************
pre_PlayerTick:
IFNE PRETRACKER_DONT_TRASH_REGS
movem.l d2-d7/a2-a6,-(sp)
ENDC
move.l a0,a4
IFNE PRETRACKER_COPPER_OUTPUT
move.l a1,pv_copperlist_ptr(a4)
ENDC
movea.l pv_my_song(a4),a6
tst.b pv_pat_stopped_b(a4)
beq .inst_pattern_processing ; don't process if music has been stopped
; ----------------------------------------
; processes the current pattern position
; registers used:
; d0: pitch shift (lower part)
; d1: scratch
; d2: effect cmd
; d3: pitch_ctrl
; d4: inst number
; d5: effect cmd
; d6: unused (flag later)
; d7: pitch
; a0: pattern data pointer
; a1: short-term scratch
; a2: unused
; a3: unused
; a4: pv
; a5: channel struct
; a6: mysong struct
.process_pattern
lea pv_channeldata(a4),a5 ; start with first channel
.pre_pat_chan_loop
IFNE PRETRACKER_PARANOIA_MODE
; I think this is something leftover from the tracker itself.
; Nothing sets pcd_pat_adsr_rel_delay_b from inside the player.
; It is used as a counter to release a note (ADSR) after a given time.
; It's not the same as the instrument ADSR release (see pcd_note_off_delay_b)
move.b pcd_pat_adsr_rel_delay_b(a5),d1
ble.s .handle_2nd_instrument
subq.b #1,d1
move.b d1,pcd_pat_adsr_rel_delay_b(a5)
bne.s .handle_2nd_instrument
move.w pcd_adsr_volume_w(a5),d3
lsr.w #6,d3
move.w d3,pcd_adsr_vol64_w(a5)
moveq.l #16,d4
move.w d4,pcd_adsr_pos_w(a5)
sub.w d3,d4
lsr.w #1,d4
add.b pcd_adsr_release_b(a5),d4
move.b d4,pcd_adsr_phase_speed_b(a5)
move.w #3,pcd_adsr_phase_w(a5)
ENDC
; ----------------------------------------
.handle_2nd_instrument
moveq.l #0,d1
move.b pcd_pat_2nd_inst_num4_b(a5),d1
beq.s .handle_current_instrument
move.b pcd_pat_2nd_inst_delay_b(a5),d3
beq.s .trigger_2nd_instrument
subq.b #1,d3
move.b d3,pcd_pat_2nd_inst_delay_b(a5)
bra.s .handle_current_instrument
.trigger_2nd_instrument
move.b d1,pcd_new_inst_num_b(a5)
move.w d1,pcd_inst_num4_w(a5)
add.w d1,d1
add.w d1,d1
lea sv_inst_infos_table-uii_SIZEOF(a6),a1
add.w d1,a1
move.l a1,pcd_inst_info_ptr(a5) ; loads 2nd instrument
move.b uii_pattern_steps(a1),pcd_inst_pattern_steps_b(a5)
moveq.l #0,d1
move.l a5,a1
move.l d1,(a1)+ ; pcd_pat_portamento_dest_w and pcd_pat_pitch_slide_w
move.l d1,(a1)+ ; pcd_pat_vol_ramp_speed_b, pcd_pat_2nd_inst_num4_b, pcd_pat_2nd_inst_delay_b, pcd_wave_offset_b
move.l d1,(a1)+ ; pcd_inst_pitch_slide_w and pcd_inst_sel_arp_note_w
move.w d1,(a1)+ ; pcd_inst_note_pitch_w
addq.w #2,a1
move.l d1,(a1)+ ; pcd_inst_line_ticks_b, pcd_inst_pitch_pinned_b, pcd_inst_vol_slide_b, pcd_inst_step_pos_b
subq.b #1,d1
move.w d1,(a1)+ ; pcd_inst_wave_num_w
move.l #$ff010010,(a1)+ ; pcd_track_delay_offset_b, pcd_inst_speed_stop_b, pcd_inst_pitch_w
move.l #(MAX_VOLUME<<16)|(MAX_VOLUME<<8)|MAX_VOLUME,(a1)+ ; pcd_inst_vol_w / pcd_loaded_inst_vol_b / pcd_pat_vol_b
bra.s .continue_with_inst
; ----------------------------------------
.handle_current_instrument
; ----------------------------------------
; handle portamento
move.w pcd_pat_portamento_dest_w(a5),d3
beq.s .no_portamento_active
move.w pcd_inst_curr_port_pitch_w(a5),d2
moveq.l #0,d1
move.b pcd_pat_portamento_speed_b(a5),d1
cmp.w d3,d2
bge.s .do_portamento_down
add.w d1,d2
cmp.w d3,d2
bgt.s .portamento_note_reached
bra.s .update_portamento_value
.do_portamento_down
sub.w d1,d2
cmp.w d3,d2
bge.s .update_portamento_value
.portamento_note_reached
clr.w pcd_pat_portamento_dest_w(a5)
move.w d3,d2
.update_portamento_value
move.w d2,pcd_inst_curr_port_pitch_w(a5)
.no_portamento_active
; ----------------------------------------
; handle volume ramping
move.b pcd_pat_vol_ramp_speed_b(a5),d1
beq.s .no_vol_ramping_active
add.b pcd_pat_vol_b(a5),d1
bpl.s .noclip_pat_vol_min
moveq.l #0,d1
.noclip_pat_vol_min
cmpi.b #MAX_VOLUME,d1
ble.s .noclip_pat_vol_max
moveq.l #MAX_VOLUME,d1
.noclip_pat_vol_max
move.b d1,pcd_pat_vol_b(a5)
.no_vol_ramping_active
; ----------------------------------------
; enters with channel number in d0
.continue_with_inst
; handle delayed note and note off first
moveq.l #0,d4
move.b pcd_note_delay_b(a5),d4
blt .pat_play_cont
beq.s .no_note_delay
subq.b #1,d4
beq.s .note_delay_end_reached
move.b d4,pcd_note_delay_b(a5) ; note still delayed
IFNE PRETRACKER_BUGFIX_CODE
bra .pat_play_cont ; I believe that with activated track delay, we must jump here
ELSE
bra .pat_channels_loop_test
ENDC
.note_delay_end_reached
st pcd_note_delay_b(a5) ; release note delay
.no_note_delay
moveq.l #0,d5
move.b pcd_channel_num_b(a5),d5
move.w sv_curr_pat_pos_w(a6),d2
add.w d2,d2
add.w d2,d2 ; *4
add.w d5,d2
add.w d2,d2 ; 8*pos+2*chan
movea.l sv_pos_data_adr(a6),a1
adda.w d2,a1
;move.l a1,d2
;cmpa.w #0,a1
;beq .pat_play_other ; this is probably never happening!
moveq.l #0,d2
move.b pv_pat_curr_row_b(a4),d2
cmp.b sv_num_steps_b(a6),d2
bcc .pat_play_cont
move.b ppd_pat_num(a1),d5
beq .pat_play_cont
add.w d5,d5
add.w d5,d5
add.w #sv_pattern_table,d5
move.l -4(a6,d5.w),a0
IFNE PRETRACKER_PARANOIA_MODE
move.l a0,d5 ; move to data register due to cc's
beq .pat_play_cont
ENDC
move.b ppd_pat_shift(a1),d0 ; pattern pitch shift (signed)
ext.w d0
add.w d2,a0
add.w d2,d2
add.w d2,a0 ; pattern data
moveq.l #15,d2
move.b pdb_inst_effect(a0),d4 ; instrument and command byte
and.w d4,d2
lsr.w #4,d4 ; instrument nr bits 0-4
moveq.l #0,d5
move.b pdb_effect_data(a0),d5
cmpi.b #$e,d2
bne.s .pat_exy_cmd_cont
; handle $exy commands
tst.b pcd_note_delay_b(a5)
bne.s .pat_exy_cmd_cont ; ignore if already running note delay
move.l d5,d3
moveq.l #15,d1
and.w d3,d1
lsr.w #4,d3
sub.w #$d,d3
bne.s .pat_is_not_ed_cmd
; note delay in x sub steps
IFNE PRETRACKER_PARANOIA_MODE ; who does this kind of stuff?
tst.b pv_pat_speed_even_b(a4)
beq.s .pat_exy_cmd_cont
ENDC
move.b d1,pcd_note_delay_b(a5)
IFNE PRETRACKER_BUGFIX_CODE
bra .pat_play_cont ; I believe that with activated track delay, we must jump here
ELSE
bra .pat_channels_loop_test
ENDC
.pat_is_not_ed_cmd
addq.b #$d-$a,d3
bne.s .pat_exy_cmd_cont
; note off in x sub steps
move.b d1,pcd_note_off_delay_b(a5)
.pat_exy_cmd_cont
st pcd_note_delay_b(a5)
; ----------------------------------------
; read out pattern editor data
moveq.l #0,d6 ; clear arp flag
move.b pdb_pitch_ctrl(a0),d3 ; pitch and control byte
bpl.s .noselinst16plus
add.w #16,d4 ; add high bit of instrument number
.noselinst16plus
moveq.l #$3f,d7
and.w d3,d7 ; pitch
add.w d4,d4
add.w d4,d4
beq.s .no_new_note ; if no instrument
tst.w d7
bne.s .no_new_note ; if it has pitch
; only change of instrument, not pitch
move.b pcd_loaded_inst_vol_b(a5),pcd_pat_vol_b(a5)
cmp.w pcd_inst_num4_w(a5),d4
bne.s .no_new_note
; attack!
move.l d6,pcd_adsr_phase_w(a5) ; and pcd_adsr_volume_w
;clr.w pcd_adsr_volume_w(a5)
;move.b #1,pcd_adsr_trigger_b(a5) ; never read
.no_new_note
; d2 = effect cmd, d3 = pitch_ctrl, d4 = inst number, d5 = effect data, d7 = pitch
andi.b #$40,d3 ; ARP bit
bne.s .has_arp_note
.no_arp_note
tst.b d2
bne.s .arp_processing_done
; 0xx: play second instrument
tst.b d5
beq.s .no_effect
moveq.l #15,d1 ; FIXME seems like it only supports the lower 15 instruments
and.b d5,d1
add.w d1,d1
add.w d1,d1
tst.b d7
bne.s .play_2nd_inst_without_trigger
.play_2nd_inst_with_no_pitch
addq.w #1,d0 ; pattern pitch shift
lsl.w #4,d0
tst.b d1
beq.s .clear_note_arp
moveq.l #0,d2
move.w d4,d3
move.w d1,d4
bra.s .trigger_new_instrument
.play_2nd_inst_without_trigger
move.w d4,d3 ; 1st instrument num
move.w d1,d4 ; 2nd instrument num
bra.s .arp_processing_done
.clear_note_arp
clr.l pcd_arp_notes_l(a5)
move.b d7,d2
move.b d4,d3
bra .clear_portamento
.has_arp_note
move.b d2,d3
or.b d5,d3
beq.s .all_arp_notes_zero ; if we go there, d3 MUST be 0 already
move.b d2,pcd_arp_note_1_b(a5)
move.b d5,d2
lsr.b #4,d2
move.b d2,pcd_arp_note_2_b(a5)
moveq.l #15,d2
and.b d5,d2
move.b d2,pcd_arp_note_3_b(a5)
.all_arp_notes_zero
moveq.l #1,d6
.no_effect
moveq.l #0,d2 ; make sure we don't get a random command here
moveq.l #0,d3
; ----------------------------------------
; d2 = effect cmd, d3 = alt inst number (or 0), d4 = inst number, d5 = effect cmd, d6 = ARP flag, d7 = pitch
.arp_processing_done
cmpi.b #NOTE_OFF_PITCH,d7
bne.s .has_no_note_off
.release_note
; release note
; FIXME we have the identical code (different regs) three times (one is inactive)
move.w pcd_adsr_volume_w(a5),d4
asr.w #6,d4
move.w d4,pcd_adsr_vol64_w(a5)
moveq.l #16,d7
move.w d7,pcd_adsr_pos_w(a5)
sub.w d4,d7
lsr.w #1,d7
add.b pcd_adsr_release_b(a5),d7
move.b d7,pcd_adsr_phase_speed_b(a5)
move.w #3,pcd_adsr_phase_w(a5)
bra .start_patt_effect_handling
.has_inst_number
add.w d7,d0 ; pattern pitch shift?
lsl.w #4,d0
cmpi.b #3,d2 ; is command portamento?
beq.s .pat_set_portamento
.trigger_new_instrument
move.w d4,d1
move.b d1,pcd_new_inst_num_b(a5)
move.w d1,pcd_inst_num4_w(a5)
add.w d1,d1
add.w d1,d1
lea sv_inst_infos_table-uii_SIZEOF(a6),a1
add.w d1,a1
move.l a1,pcd_inst_info_ptr(a5)
move.b uii_pattern_steps(a1),pcd_inst_pattern_steps_b(a5)
moveq.l #0,d1
move.l a5,a1
move.l d1,(a1)+ ; pcd_pat_portamento_dest_w and pcd_pat_pitch_slide_w
move.l d1,(a1)+ ; pcd_pat_vol_ramp_speed_b, pcd_pat_2nd_inst_num4_b, pcd_pat_2nd_inst_delay_b, pcd_wave_offset_b
move.l d1,(a1)+ ; pcd_inst_pitch_slide_w and pcd_inst_sel_arp_note_w
move.l d1,(a1)+ ; pcd_inst_note_pitch_w and pcd_inst_curr_port_pitch_w
move.l d1,(a1)+ ; pcd_inst_line_ticks_b, pcd_inst_pitch_pinned_b, pcd_inst_vol_slide_b, pcd_inst_step_pos_b
subq.b #1,d1
move.w d1,(a1)+ ; pcd_inst_wave_num_w
move.l #$ff010010,(a1)+ ; pcd_track_delay_offset_b, pcd_inst_speed_stop_b, pcd_inst_pitch_w
move.l #(MAX_VOLUME<<16)|(MAX_VOLUME<<8)|MAX_VOLUME,(a1)+ ; pcd_inst_vol_w / pcd_loaded_inst_vol_b / pcd_pat_vol_b
bra.s .cont_after_inst_trigger
.has_no_note_off
tst.b d7
beq.s .start_patt_effect_handling
tst.b d4
bne.s .has_inst_number
add.w d7,d0 ; pattern pitch shift
lsl.w #4,d0
.cont_after_inst_trigger
tst.b d6 ; has ARP?
bne.s .lbC001852
bra.s .lbC00193C
.pat_set_portamento
tst.b d6
bne.s .pat_new_portamento
.lbC00193C
clr.l pcd_arp_notes_l(a5)
moveq.l #0,d6
.lbC001852
cmpi.b #3,d2 ; is command portamento?
beq.s .pat_new_portamento
.clear_portamento
move.w #$10,pcd_inst_pitch_w(a5)
move.w d0,pcd_inst_curr_port_pitch_w(a5)
clr.w pcd_pat_portamento_dest_w(a5)
bra.s .start_patt_effect_handling
.pat_new_portamento
add.w #$10,d0 ; pattern pitch shift
move.w d0,pcd_pat_portamento_dest_w(a5)
move.w pcd_inst_pitch_w(a5),d1
add.w d1,pcd_inst_curr_port_pitch_w(a5)
clr.w pcd_inst_pitch_w(a5)
tst.b d5
beq.s .pat_keep_old_portamento
move.b d5,pcd_pat_portamento_speed_b(a5)
.pat_keep_old_portamento
; ----------------------------------------
.start_patt_effect_handling
; FIXME can we move this code to avoid blocking the two registers d3/d5
tst.b d3
beq.s .has_no_second_inst
move.b d3,pcd_pat_2nd_inst_num4_b(a5)
move.b d5,d3
lsr.b #4,d3
move.b d3,pcd_pat_2nd_inst_delay_b(a5)
.has_no_second_inst
clr.b pcd_pat_vol_ramp_speed_b(a5)
clr.w pcd_pat_pitch_slide_w(a5)
tst.b d6
bne .pat_play_cont
add.w d2,d2
move.w .pattern_command_jmptable(pc,d2.w),d2
jmp .pattern_command_jmptable(pc,d2.w)
.pattern_command_jmptable
dc.w .pat_play_nop-.pattern_command_jmptable
dc.w .pat_slide_up-.pattern_command_jmptable
dc.w .pat_slide_down-.pattern_command_jmptable
dc.w .pat_play_nop-.pattern_command_jmptable
dc.w .pat_set_vibrato-.pattern_command_jmptable
dc.w .pat_set_track_delay-.pattern_command_jmptable
dc.w .pat_play_nop-.pattern_command_jmptable
dc.w .pat_play_nop-.pattern_command_jmptable
dc.w .pat_play_nop-.pattern_command_jmptable
dc.w .pat_set_wave_offset-.pattern_command_jmptable
dc.w .pat_volume_ramp-.pattern_command_jmptable
dc.w .pat_pos_jump-.pattern_command_jmptable
dc.w .pat_set_volume-.pattern_command_jmptable
dc.w .pat_pat_break-.pattern_command_jmptable
dc.w .pat_play_nop-.pattern_command_jmptable
dc.w .pat_set_speed-.pattern_command_jmptable
; d5 = command parameter data
; ----------------------------------------
.pat_set_speed
lea pv_pat_speed_even_b(a4),a1
cmpi.b #MAX_SPEED,d5
bhs.s .pat_set_speed_shuffle
move.b d5,(a1)+ ; pv_pat_speed_even_b
move.b d5,(a1)+ ; pv_pat_speed_odd_b
move.b d5,(a1)+ ; pv_pat_line_ticks_b
sne (a1)+ ; pv_pat_stopped_b
IFNE PRETRACKER_SONG_END_DETECTION
seq (a1)+ ; pv_songend_detected_b
ENDC
bra .pat_play_cont
.pat_set_speed_shuffle
moveq.l #15,d2
and.w d5,d2 ; odd speed
lsr.w #4,d5 ; even speed
move.b d5,(a1)+ ; pv_pat_speed_even_b
move.b d2,(a1)+ ; pv_pat_speed_odd_b
btst #0,pv_pat_curr_row_b(a4)
beq.s .pat_shuffle_on_even
move.b d2,d5 ; toggle speed to odd row
.pat_shuffle_on_even
move.b d5,(a1)+ ; pv_pat_line_ticks_b
bra .pat_play_cont
; ----------------------------------------
.pat_set_vibrato
clr.w pcd_vibrato_pos_w(a5)
move.w #1,pcd_vibrato_delay_w(a5)
moveq.l #15,d2
and.w d5,d2
lea pre_vib_depth_table(pc),a1
move.b (a1,d2.w),pcd_vibrato_depth_w+1(a5)
lsr.b #4,d5
move.b pre_vib_speed_table-pre_vib_depth_table(a1,d5.w),d2
muls pcd_vibrato_depth_w(a5),d2
asr.w #4,d2
move.w d2,pcd_vibrato_speed_w(a5)
bra.s .pat_play_cont
; ----------------------------------------
.pat_set_track_delay
cmp.b #NUM_CHANNELS-1,pcd_channel_num_b(a5)
beq.s .pat_play_cont ; we are at channel 3 -- track delay not available here
lea pcd_SIZEOF+pcd_track_delay_buffer+ocd_volume(a5),a1
moveq.l #MAX_TRACK_DELAY-1,d2
.clr_track_delay_buffer_loop
clr.b (a1) ; ocd_volume
lea ocd_SIZEOF(a1),a1
dbra d2,.clr_track_delay_buffer_loop
tst.b d5
bne.s .pat_track_delay_set
st pcd_track_delay_steps_b(a5)
bra.s .handle_track_delay
; ----------------------------------------
.pat_volume_ramp
tst.b d5
beq.s .pat_play_cont
moveq.l #15,d3
and.b d5,d3
beq.s .pat_vol_ramp_up
; NOTE: Changed behaviour: using d3 instead of d5
; if both lower and upper were specified, this
; probably led to a drastic decrease of volume.
neg.b d3
move.b d3,pcd_pat_vol_ramp_speed_b(a5)
bra.s .pat_play_cont
.pat_vol_ramp_up
lsr.b #4,d5
move.b d5,pcd_pat_vol_ramp_speed_b(a5)
bra.s .pat_play_cont
.pat_track_delay_set
moveq.l #15,d2
and.b d5,d2
add.b d2,d2
move.b d2,pcd_track_delay_steps_b(a5)
; subq.b #1,d2
; move.b d2,pcd_SIZEOF+pcd_track_init_delay_b(a5)
lsr.b #4,d5
move.b d5,pcd_track_delay_vol16_b(a5)
bra.s .pat_play_cont
; ----------------------------------------
.pat_slide_down
neg.w d5
.pat_slide_up
move.w d5,pcd_pat_pitch_slide_w(a5)
bra.s .pat_play_cont
; ----------------------------------------
.pat_set_wave_offset
move.b d5,pcd_wave_offset_b(a5)
bra.s .pat_play_cont
; ----------------------------------------
.pat_pos_jump
move.b d5,pv_next_pat_pos_b(a4)
bra.s .pat_play_cont
; ----------------------------------------
.pat_pat_break
move.b d5,pv_next_pat_row_b(a4)
bra.s .pat_play_cont
; ----------------------------------------
.pat_set_volume
cmpi.b #MAX_VOLUME,d5
bls.s .pat_set_volume_nomax
moveq.l #MAX_VOLUME,d5
.pat_set_volume_nomax
move.b d5,pcd_pat_vol_b(a5)
; ----------------------------------------
.pat_play_nop
.pat_play_cont
move.b pcd_track_delay_steps_b(a5),d2 ; FIXME this is a mess
beq.s .pat_channels_loop_test
.handle_track_delay
lea pcd_SIZEOF(a5),a5
cmp.b #NUM_CHANNELS-2,pcd_channel_num_b-pcd_SIZEOF(a5) ; FIXME find out why we need this
bge.s .pat_channels_loop_end
.pat_channels_loop_test
lea pcd_SIZEOF(a5),a5
cmp.b #NUM_CHANNELS-1,pcd_channel_num_b-pcd_SIZEOF(a5)
bne .pre_pat_chan_loop
.pat_channels_loop_end
; end of pattern loop
; ----------------------------------------
; Pattern advancing and pattern break and jump handling. Song looping and song-end detection.
.pattern_advancing
subq.b #1,pv_pat_line_ticks_b(a4)
bne .no_pattern_advance
; clear note delay info
moveq.l #0,d0
REPT NUM_CHANNELS
move.b d0,pv_channeldata+pcd_note_delay_b+REPTN*pcd_SIZEOF(a4)
ENDR
move.b sv_num_steps_b(a6),d1 ; number of steps in pattern
move.b pv_pat_curr_row_b(a4),d0
addq.b #1,d0 ; normal step increment
move.w sv_curr_pat_pos_w(a6),d3 ; current song position
move.b pv_next_pat_row_b(a4),d2 ; $ff means no pattern break
bmi.s .no_pattern_break
st pv_next_pat_row_b(a4) ; processed break, set to $ff
move.b d2,d0
IFNE 0 ; PRETRACKER_BUGFIX_CODE ; currently disabled to keep old behaviour
moveq.l #0,d2 ; clear mask
ENDC
cmp.b d1,d0
blo.s .has_legal_break_pos
move.b d1,d0 ; limit to last step
subq.b #1,d0
bra.s .has_legal_break_pos
.no_pattern_break
cmp.b d1,d0
blo.s .pattern_end_not_reached
moveq.l #0,d0
IFNE PRETRACKER_PARANOIA_MODE
move.b pv_loop_pattern_b(a4),d0 ; keep same pattern rolling?
bne.s .pattern_end_not_reached
ENDC
.has_legal_break_pos
addq.w #1,d3 ; pattern break will increment song pos -- if there is no new pattern position
.pattern_end_not_reached
move.b pv_next_pat_pos_b(a4),d4
bmi.s .no_new_position ; has a new pattern position
st pv_next_pat_pos_b(a4)
IFNE PRETRACKER_SONG_END_DETECTION
cmp.b d4,d3
bgt.s .no_backjump
st pv_songend_detected_b(a4) ; detect jumping back
.no_backjump
ENDC
move.b d4,d3 ; load new position
IFNE 0 ; PRETRACKER_BUGFIX_CODE ; currently disabled to keep old behaviour
not.b d2
and.b d2,d0 ; if we had NO pattern break, we will clear d0
ELSE
moveq.l #0,d0
ENDC
.no_new_position
cmp.w sv_pat_pos_len_w(a6),d3
blt.s .no_restart_song
move.w sv_pat_restart_pos_w(a6),d3
IFNE PRETRACKER_SONG_END_DETECTION
st pv_songend_detected_b(a4)
ENDC
.no_restart_song
move.b d0,pv_pat_curr_row_b(a4)
move.w d3,sv_curr_pat_pos_w(a6)
move.b pv_pat_speed_even_b(a4),d1
lsr.b #1,d0
bcc.s .set_speed_even
move.b pv_pat_speed_odd_b(a4),d1
.set_speed_even
move.b d1,pv_pat_line_ticks_b(a4)
.no_pattern_advance
; ----------------------------------------
; processes the instrument pattern for each running instrument
; registers used:
; d0: pitch
; d1: volume
; d2: inst num
; d3: scratch
; a0: pattern data pointer
; a1: scratch
; a2: instrument info
; a3: wave info
; a4: pv
; a5: channel struct
; a6: mysong struct
.inst_pattern_processing
lea pv_channeldata(a4),a5
.inst_chan_loop
move.l pcd_waveinfo_ptr(a5),a3
move.l pcd_inst_info_ptr(a5),a2
move.l a2,d3
beq .inst_no_inst_active
; calculate pitch -- funny that there is no min check (seems to happen later though)
move.w pcd_inst_pitch_slide_w(a5),d0
add.w pcd_pat_pitch_slide_w(a5),d0
beq.s .inst_no_pitch_slides_active
add.w pcd_inst_pitch_w(a5),d0
cmp.w #(3*NOTES_IN_OCTAVE)<<4,d0
ble.s .inst_noclip_pitch_max
move.w #(3*NOTES_IN_OCTAVE)<<4,d0
.inst_noclip_pitch_max
move.w d0,pcd_inst_pitch_w(a5)
.inst_no_pitch_slides_active
move.b pcd_inst_vol_slide_b(a5),d1
beq.s .inst_no_vol_slide_active
add.b pcd_inst_vol_w+1(a5),d1
bpl.s .inst_noclip_vol_zero
moveq.l #0,d1
.inst_noclip_vol_zero
cmp.b #MAX_VOLUME,d1
ble.s .inst_noclip_vol_max
moveq.l #MAX_VOLUME,d1
.inst_noclip_vol_max
move.b d1,pcd_inst_vol_w+1(a5)
.inst_no_vol_slide_active
move.b pcd_inst_line_ticks_b(a5),d2
bne .inst_still_ticking
moveq.l #0,d0
move.w d0,pcd_inst_pitch_slide_w(a5)
move.b d0,pcd_inst_vol_slide_b(a5)
; IFNE PRETRACKER_PARANOIA_MODE ; new step is never written
; move.w pcd_inst_new_step_w(a5),d1
; blt.s .inst_no_new_step_pos
; cmpi.w #$20,d1
; ble.s .inst_good_new_step_pos
; moveq.l #$20,d1
;.inst_good_new_step_pos
; move.b d1,pcd_inst_step_pos_b(a5)
; move.w #$ffff,pcd_inst_new_step_w(a5)
;.inst_no_new_step_pos
; ENDC
move.b pcd_inst_step_pos_b(a5),d0
cmp.b pcd_inst_pattern_steps_b(a5),d0
bhs .inst_pat_loop_exit
moveq.l #-1,d4
moveq.l #0,d7
moveq.l #0,d3 ; flag for stitching -- if set, must not trigger new note
; enters with d4 = -1, meaning no first note pos yet
move.w pcd_inst_num4_w(a5),d1
movea.l sv_inst_patterns_table-4(a6,d1.w),a0
add.w d0,a0
add.w d0,a0
add.w d0,a0
.inst_pat_loop
IFNE PRETRACKER_PARANOIA_MODE
moveq.l #0,d2 ; default to not stitched
move.b (a0)+,d1 ; pdb_pitch_ctrl get pitch byte
move.b d1,d6
bpl.s .inst_note_is_not_stitched ; means that note is stitched
tst.w d4
bpl.s .inst_no_update_first_note
move.w d0,d4 ; position of first note before stitching
.inst_no_update_first_note
moveq.l #1,d2 ; next note will be fetched immediately
.inst_note_is_not_stitched
ELSE
move.b (a0)+,d1 ; pdb_pitch_ctrl get pitch byte
move.b d1,d6
smi d2 ; note stitched?
;neg.b d2
ENDC
tst.b d3
bne.s .skippitchloading
andi.w #$3F,d1
beq.s .skippitchloading ; no new note
subq.w #1,d1
lsl.w #4,d1
move.w d1,pcd_inst_note_pitch_w(a5)
and.w #1<<6,d6
sne pcd_inst_pitch_pinned_b(a5)
IFEQ PRETRACKER_FASTER_CODE
neg.b pcd_inst_pitch_pinned_b(a5) ; only to be state binary compatible
ENDC
.skippitchloading
moveq.l #15,d6
and.b (a0)+,d6 ; pdb_effect_cmd command number
add.w d6,d6
moveq.l #0,d5
move.b (a0)+,d5 ; pdb_effect_data command parameter byte
move.w .inst_command_jmptable(pc,d6.w),d3
jmp .inst_command_jmptable(pc,d3.w)
.inst_command_jmptable
dc.w .inst_select_wave-.inst_command_jmptable
dc.w .inst_slide_up-.inst_command_jmptable
dc.w .inst_slide_down-.inst_command_jmptable
dc.w .inst_adsr-.inst_command_jmptable
dc.w .inst_select_wave-.inst_command_jmptable
dc.w .inst_nop-.inst_command_jmptable
dc.w .inst_nop-.inst_command_jmptable
dc.w .inst_nop-.inst_command_jmptable
dc.w .inst_nop-.inst_command_jmptable
dc.w .inst_nop-.inst_command_jmptable
dc.w .inst_vol_slide-.inst_command_jmptable
dc.w .inst_jump_to_step-.inst_command_jmptable
dc.w .inst_set_volume-.inst_command_jmptable
dc.w .inst_nop-.inst_command_jmptable
dc.w .inst_use_pat_arp-.inst_command_jmptable
dc.w .inst_set_speed-.inst_command_jmptable
; d0 = current step / next step
; d5 = command parameter data / scratch
; d2 = note stitched flag
; d6 = scratch
; ----------------------------------------
.inst_select_wave
subq.w #1,d5
cmp.w #MAX_WAVES,d5
bhs .inst_cmd_cont_next
add.w d5,d5
add.w d5,d5
cmp.w pcd_inst_wave_num_w(a5),d5
beq .inst_cmd_cont_next
move.w d5,pcd_inst_wave_num_w(a5)
move.l sv_waveinfo_table(a6,d5.w),a3
move.l a3,pcd_waveinfo_ptr(a5)
move.b pcd_channel_mask_b(a5),d3
move.b d3,pcd_out_trg_b(a5)
or.b d3,pv_trigger_mask_w+1(a4)
move.l pv_wave_sample_table(a4,d5.w),a1
tst.w d6
bne.s .inst_select_wave_nosync
moveq.l #0,d3
move.w wi_loop_offset_w(a3),d6 ; is unlikely 32768
move.w wi_subloop_len_w(a3),d5
bne.s .inst_set_wave_has_subloop
adda.w d6,a1 ; add loop offset
move.w wi_chipram_w(a3),d5
sub.w d6,d5
;cmp.w #1,d5 ; not necessary as increases in steps of 2
bhi.s .inst_set_wave_has_min_length
moveq.l #2,d5
.inst_set_wave_has_min_length
move.w d5,pcd_out_len_w(a5)
moveq.l #-1,d3
.inst_set_wave_has_subloop
move.w d3,pcd_out_lof_w(a5)
move.l a1,pcd_out_ptr_l(a5)
st pcd_inst_ping_pong_dir_b(a5)
moveq.l #0,d5
move.b wi_subloop_wait_b(a3),d5
addq.w #1,d5
move.w d5,pcd_inst_subloop_wait_w(a5)
move.w d6,pcd_inst_loop_offset_w(a5)
bra .inst_cmd_cont_next
; ----------------------------------------
.inst_select_wave_nosync
move.w wi_chipram_w(a3),d5
moveq.l #0,d3
tst.w wi_subloop_len_w(a3)
bne.s .inst_set_wave_ns_has_subloop
move.w wi_loop_offset_w(a3),d6
adda.w d6,a1
sub.w d6,d5
;cmp.w #1,d5 ; not necessary as increases in steps of 2
bhi.s .inst_set_wave_ns_has_min_length
moveq.l #2,d5
.inst_set_wave_ns_has_min_length
move.w d5,pcd_out_len_w(a5)
moveq.l #-1,d3
.inst_set_wave_ns_has_subloop
move.w d3,pcd_out_lof_w(a5)
move.l a1,pcd_out_ptr_l(a5)
move.w pcd_inst_loop_offset_w(a5),d6
cmp.w d6,d5
bhs.s .inst_set_wave_ns_keep_pp
st pcd_inst_ping_pong_dir_b(a5)
.inst_set_wave_ns_keep_pp
moveq.l #0,d5
move.w d5,pcd_inst_subloop_wait_w(a5)
sub.w wi_subloop_step_w(a3),d6
move.w d6,pcd_inst_loop_offset_w(a5)
bra .inst_cmd_cont_next
; ----------------------------------------
.inst_slide_down
neg.w d5
.inst_slide_up
move.w d5,pcd_inst_pitch_slide_w(a5)
bra .inst_cmd_cont_next
; ----------------------------------------
.inst_adsr
subq.w #1,d5
beq.s .inst_adsr_release
subq.w #1,d5
bne .inst_cmd_cont_next
; d5.l is zero
move.w d5,pcd_adsr_phase_w(a5)
move.w d5,pcd_adsr_volume_w(a5)
bra .inst_cmd_cont_next
.inst_adsr_release
move.w pcd_adsr_volume_w(a5),d5
asr.w #6,d5
move.w d5,pcd_adsr_vol64_w(a5)
moveq.l #16,d6
move.w d6,pcd_adsr_pos_w(a5)
sub.w d5,d6
lsr.w #1,d6
add.b pcd_adsr_release_b(a5),d6
move.b d6,pcd_adsr_phase_speed_b(a5)
move.w #3,pcd_adsr_phase_w(a5)
bra.s .inst_cmd_cont_next
; ----------------------------------------
.inst_vol_slide
moveq.l #15,d3
and.w d5,d3
beq.s .inst_vol_slide_up
; NOTE: Changed behaviour: using d3 instead of d5
; if both lower and upper were specified, this
; probably led to a drastic decrease of volume.
neg.w d3
move.b d3,pcd_inst_vol_slide_b(a5)
bra.s .inst_cmd_cont_next
.inst_vol_slide_up
lsr.b #4,d5
move.b d5,pcd_inst_vol_slide_b(a5)
bra.s .inst_cmd_cont_next
; ----------------------------------------
.inst_jump_to_step
cmp.w d0,d5
bge.s .inst_cmd_cont_next ; only backward jumps allowed (?)
IFNE PRETRACKER_PARANOIA_MODE
; this stuff is PARANOIA
tst.b d4
bmi.s .inst_jump_to_step_doit ; we did not have a stitched note before
cmp.b d3,d4
ble .inst_cmd_cont_next ; we are jumping back to the stitched note, ignore
.inst_jump_to_step_doit
ENDC
move.w d5,d0
tst.b d7 ; check if we had jumped before
bne.s .inst_we_were_stitched
IFNE PRETRACKER_PARANOIA_MODE
moveq.l #-1,d4 ; mark as no first stitch pos
ENDC
move.w pcd_inst_num4_w(a5),d1
movea.l sv_inst_patterns_table-4(a6,d1.w),a0
add.w d5,a0
add.w d5,d5
add.w d5,a0
bra.s .inst_fetch_next
.inst_we_were_stitched
move.b pcd_inst_speed_stop_b(a5),d2
bra.s .inst_cont_from_nasty_double_jump
; ----------------------------------------
.inst_set_volume
cmp.w #MAX_VOLUME,d5
ble.s .inst_set_volume_nomax
moveq.l #MAX_VOLUME,d5
.inst_set_volume_nomax
move.w d5,pcd_inst_vol_w(a5)
bra.s .inst_cmd_cont_next
; ----------------------------------------
.inst_use_pat_arp
moveq.l #3,d3
and.w d5,d3
beq.s .inst_use_pat_arp_play_base
lsr.w #4,d5
beq.s .inst_use_pat_arp_skip_empty
subq.w #1,d5
bne.s .inst_cmd_cont_next ; illegal high nibble (only 0/1 allowed)
; pick arp note
move.b pcd_arp_notes_l-1(a5,d3.w),d3
; play base note
.inst_use_pat_arp_set
lsl.w #4,d3
.inst_use_pat_arp_play_base
move.w d3,pcd_inst_sel_arp_note_w(a5)
bra.s .inst_cmd_cont_next
.inst_use_pat_arp_skip_empty
; pick arp note, if it's 0, skip it
move.b pcd_arp_notes_l-1(a5,d3.w),d3
bne.s .inst_use_pat_arp_set
addq.w #1,d0
bra.s .inst_fetch_next
; ----------------------------------------
.inst_set_speed
tst.w d5
bne.s .inst_set_speed_nostop
st d5
.inst_set_speed_nostop
move.b d5,pcd_inst_speed_stop_b(a5)
; ----------------------------------------
.inst_nop
.inst_cmd_cont_next
addq.w #1,d0
tst.b d2
beq.s .inst_pat_loop_exit2
; d2 != 0 in this case, hence d3 will be set
.inst_fetch_next
moveq.l #1,d7 ; mark that we are in at least next iteration
move.b d2,d3 ; mark stitching
cmp.b pcd_inst_pattern_steps_b(a5),d0
blo .inst_pat_loop
.inst_pat_loop_exit
st d2
.inst_pat_loop_exit2
add.b pcd_inst_speed_stop_b(a5),d2
.inst_cont_from_nasty_double_jump
move.b d2,pcd_inst_line_ticks_b(a5)
move.b d0,pcd_inst_step_pos_b(a5) ; update inst step pos
.inst_still_ticking
tst.b pcd_inst_wave_num_w+1(a5)
bpl.s .inst_wave_selected
.inst_no_wave_selected
moveq.l #0,d3
move.w d3,pcd_inst_wave_num_w(a5) ; FIXME why set to wave 0? just asking, because this is zero based
move.l sv_waveinfo_ptr(a6),a3
move.l a3,pcd_waveinfo_ptr(a5)
move.b pcd_channel_mask_b(a5),d0
move.b d0,pcd_out_trg_b(a5)
or.b d0,pv_trigger_mask_w+1(a4)
tst.w wi_subloop_len_w(a3)
bne.s .inst_nowave_has_subloop
move.w wi_loop_offset_w(a3),d5
move.l pv_wave_sample_table(a4),a1
adda.w d5,a1
move.l a1,pcd_out_ptr_l(a5)
move.w wi_chipram_w(a3),d0
move.w d0,d6
subq.w #1,d6
cmp.w d4,d6
ble.s .lbC00118E
sub.w d5,d0
bra.s .lbC000F48
.lbC00118E
moveq.l #2,d0
.lbC000F48
move.w d0,pcd_out_len_w(a5)
subq.w #1,d3
bra.s .inst_set_no_lof
.inst_nowave_has_subloop
move.l pv_wave_sample_table(a4),pcd_out_ptr_l(a5)
move.w wi_loop_offset_w(a3),d4
.inst_set_no_lof
move.w d3,pcd_out_lof_w(a5)
moveq.l #0,d0
move.b wi_subloop_wait_b(a3),d0
addq.w #1,d0
move.w d0,pcd_inst_subloop_wait_w(a5)
move.w d4,pcd_inst_loop_offset_w(a5)
st pcd_inst_ping_pong_dir_b(a5)
.inst_wave_selected
cmpi.b #$FF,d2
beq.s .inst_pat_loop_exit3
subq.b #1,d2
move.b d2,pcd_inst_line_ticks_b(a5)
.inst_no_inst_active
.inst_pat_loop_exit3
; ----------------------------------------
; expects d2 = inst num
; a5 = channel
move.w pcd_inst_vol_w(a5),d1
tst.b pcd_new_inst_num_b(a5)
bne.s .load_instrument
.dont_load_instrument
move.w pcd_adsr_volume_w(a5),d2
move.w pcd_adsr_phase_w(a5),d4
beq.s .adsr_attack
subq.w #1,d4
move.w d4,d3
beq.s .adsr_decay_and_release ; we destinguish via d3 == 0 -> decay
subq.w #1,d4
beq .adsr_sustain
.adsr_release
move.w pcd_adsr_pos_w(a5),d4
add.w pcd_adsr_vol64_w(a5),d4
move.w d4,pcd_adsr_pos_w(a5)
sub.w #16,d4
blt.s .adsr_done
move.w d4,pcd_adsr_pos_w(a5)
; same code for both release and decay
.adsr_decay_and_release
moveq.l #0,d4
move.b pcd_adsr_phase_speed_b(a5),d4
cmpi.b #$8f,d4
bhs.s .adsr_absurd_slow_release
move.b d4,d5
addq.b #1,d5
add.w d4,d4
lea pre_roll_off_table(pc),a1
move.w (a1,d4.w),d4
bra.s .adsr_release_cont
.adsr_absurd_slow_release
moveq.l #2,d4 ; FIXME I guess this should be 1, if I look at the roll-off table
moveq.l #-$71,d5 ; same as $8f, we only need the byte
.adsr_release_cont
move.b d5,pcd_adsr_phase_speed_b(a5)
tst.w d3
beq.s .adsr_is_actually_decay
sub.w d4,d2
bpl.s .adsr_done
moveq.l #0,d2
bra.s .adsr_done
.adsr_is_actually_decay
sub.w d4,d2
cmp.w uii_adsr_sustain(a2),d2
bgt.s .adsr_done
.adsr_enter_sustain
move.w #2,pcd_adsr_phase_w(a5)
move.w uii_adsr_sustain(a2),d2
bra.s .adsr_done
.load_instrument
move.b d1,pcd_loaded_inst_vol_b(a5)
move.l uii_vibrato_delay(a2),pcd_vibrato_delay_w(a5) ; and uii_vibrato_depth
;move.w uii_vibrato_delay(a2),pcd_vibrato_delay_w(a5)
;move.w uii_vibrato_depth(a2),pcd_vibrato_depth_w(a5)
move.l uii_vibrato_speed(a2),pcd_vibrato_speed_w(a5) ; and uii_adsr_release
;move.w uii_vibrato_speed(a2),pcd_vibrato_speed_w(a5)
;move.b uii_adsr_release(a2),pcd_adsr_release_b(a5)
moveq.l #0,d2
move.l d2,pcd_adsr_phase_w(a5) ; and pcd_adsr_volume_w
;move.w d2,pcd_adsr_phase_w(a5)
;move.w d2,pcd_adsr_volume_w(a5)
move.l d2,pcd_new_inst_num_b(a5) ; and pcd_vibrato_pos_w
.adsr_attack
add.w uii_adsr_attack(a2),d2
cmpi.w #MAX_VOLUME<<4,d2
blt.s .adsr_done
.adsr_do_decay
move.w #MAX_VOLUME<<4,d2
move.w #1,pcd_adsr_phase_w(a5)
move.b uii_adsr_decay+1(a2),pcd_adsr_phase_speed_b(a5)
.adsr_sustain
.adsr_done
move.w d2,pcd_adsr_volume_w(a5)
; handle note cut-off command (EAx command)
move.b pcd_note_off_delay_b(a5),d4
beq.s .dont_release_note
subq.b #1,d4
move.b d4,pcd_note_off_delay_b(a5)
bne.s .dont_release_note
; cut off note
clr.w pcd_adsr_volume_w(a5)
move.w #3,pcd_adsr_phase_w(a5)
.dont_release_note
; ----------------------------------------
; calculate final volume output = inst_vol * ADSR volume * pattern volume
IFNE PRETRACKER_VOLUME_TABLE
lea pv_volume_table(a4),a1
lsl.w #3,d2
and.w #127<<7,d2
or.b d1,d2
move.b (a1,d2.w),d1
lsl.w #7,d1
or.b pcd_pat_vol_b(a5),d1
move.b (a1,d1.w),pcd_out_vol_b(a5)
ELSE
lsr.w #4,d2
mulu d2,d1
lsr.w #6,d1
moveq.l #0,d2
move.b pcd_pat_vol_b(a5),d2
mulu d1,d2
lsr.w #6,d2
move.b d2,pcd_out_vol_b(a5)
ENDC
; ----------------------------------------
; wave loop pos advancing and pattern wave offset command handling
move.w wi_subloop_len_w(a3),d3
beq .wave_has_no_subloop
move.w d3,pcd_out_len_w(a5) ; FIXME can we move this to wave loading?
move.w wi_subloop_step_w(a3),d2
moveq.l #0,d1
move.b pcd_wave_offset_b(a5),d1
beq.s .wave_with_subloop_but_no_wave_offset
tst.b wi_allow_9xx_b(a3)
beq.s .wave_with_subloop_but_no_wave_offset
; update loop offset from pattern
lsl.w #7,d1
clr.b pcd_wave_offset_b(a5)
; keep current direction of ping-pong unchanged
move.b pcd_inst_ping_pong_dir_b(a5),d4
bpl.s .wave_move_one_step_ahead
sub.w d2,d1 ; go in reverse direction one step?
bra.s .wave_submove_cont
.wave_move_one_step_ahead
add.w d2,d1 ; go in reverse direction one step?
bra.s .wave_submove_cont
.wave_with_subloop_but_no_wave_offset
subq.w #1,pcd_inst_subloop_wait_w(a5)
bgt.s .wave_subloop_wait
; subloop moves!
move.b pcd_inst_ping_pong_dir_b(a5),d4
move.w pcd_inst_loop_offset_w(a5),d1
.wave_submove_cont
; reset subloop wait
moveq.l #0,d5
move.b wi_subloop_wait_b(a3),d5 ; FIXME why is this not increased by one here?
move.w d5,pcd_inst_subloop_wait_w(a5)
tst.b d4
bpl.s .wave_is_moving_backwards
add.w d2,d1 ; increment offset in forward direction one step
move.w d1,d4
add.w d3,d4 ; calculate next start of loop (curr start)
move.w wi_loop_end_w(a3),d2
cmp.w d1,d2
bhs.s .is_not_past_loop_end ; are we starting playback past the end of the loop
move.w wi_chipram_w(a3),d2 ; use sample end
.is_not_past_loop_end
sub.w d4,d2 ; space left = (max(loop end, sample end) - curr start)
bhi.s .wave_new_loop_pos_fits ; max(loop end, sample end) > curr start?
add.w d2,d1 ; place offset at last possible place (loop_end - loop size)
clr.b pcd_inst_ping_pong_dir_b(a5) ; mark going backwards
bra.s .wave_loop_dir_changed
.wave_is_moving_backwards
sub.w d2,d1 ; decrement offset in backward direction one step
move.w wi_loop_start_w(a3),d4
move.w d4,d2
sub.w d1,d2 ; calc how many bytes we are part front
bmi.s .wave_new_loop_pos_fits
.wave_hit_front_of_loop
move.w d4,d1
st pcd_inst_ping_pong_dir_b(a5)
.wave_loop_dir_changed
tst.w d2
bne.s .wave_new_loop_pos_fits ; perfect fit for last loop
; partial fit only
subq.w #1,pcd_inst_subloop_wait_w(a5) ; why, oh why?
.wave_new_loop_pos_fits
move.w d1,d4
move.w d4,pcd_inst_loop_offset_w(a5)
bra.s .done_lof_calc
.wave_subloop_wait
move.w pcd_inst_loop_offset_w(a5),d4
.done_lof_calc
move.w d4,pcd_out_lof_w(a5)
move.w pcd_inst_wave_num_w(a5),d1 ; FIXME can we move this to wave loading?
move.l pv_wave_sample_table(a4,d1.w),pcd_out_ptr_l(a5)
bra.s .loop_handling_done
.wave_has_no_subloop
move.b pcd_wave_offset_b(a5),d1
beq.s .loop_handling_done
tst.b wi_allow_9xx_b(a3)
beq.s .loop_handling_done
; apply offset from pattern for sample without subloop
lsl.w #8,d1 ; clear upper byte
lsr.w #1,d1
clr.b pcd_wave_offset_b(a5)
move.b pcd_channel_mask_b(a5),d2 ; trigger output
move.b d2,pcd_out_trg_b(a5)
or.b d2,pv_trigger_mask_w+1(a4)
move.w wi_chipram_w(a3),d2
sub.w d1,d2
bgt.s .waveoffset_is_not_past_end
moveq.l #2,d2
IFNE PRETRACKER_BUGFIX_CODE
; FIXME actually we should set the start address to the empty sample
; FIXME (or at least to the beginning of the sample?) to avoid an audible glitch
moveq.l #0,d1
ENDC
.waveoffset_is_not_past_end
move.w d2,pcd_out_len_w(a5)
move.w pcd_inst_wave_num_w(a5),d2
add.l pv_wave_sample_table(a4,d2.w),d1
move.l d1,pcd_out_ptr_l(a5)
;move.w #$FFFF,pcd_out_lof_w(a5) ; this is already set to -1, no need to set it again
.loop_handling_done
; ----------------------------------------
; pitch handling
move.w pcd_inst_pitch_w(a5),d0
sub.w #$10,d0
tst.b pcd_inst_pitch_pinned_b(a5)
bne.s .pitch_pinned
.pitch_not_pinned
add.w pcd_inst_sel_arp_note_w(a5),d0
add.w pcd_inst_curr_port_pitch_w(a5),d0
sub.w #$10,d0
.pitch_pinned
add.w pcd_inst_note_pitch_w(a5),d0
; ----------------------------------------
; vibrato processing
move.b pcd_vibrato_delay_w+1(a5),d1
beq.s .vibrato_already_active
subq.b #1,d1
move.b d1,pcd_vibrato_delay_w+1(a5)
bne.s .vibrato_still_delayed
.vibrato_already_active
move.w pcd_vibrato_speed_w(a5),d2
beq.s .vibrato_disabled ; no speed -- skip stuff
move.w pcd_vibrato_depth_w(a5),d4
move.w d2,d1
add.w pcd_vibrato_pos_w(a5),d1
cmp.w d1,d4
blt.s .vibrato_flipit
neg.w d4
cmp.w d1,d4
ble.s .vibrato_cont
.vibrato_flipit
neg.w d2
move.w d2,pcd_vibrato_speed_w(a5)
move.w d4,d1
.vibrato_cont
move.w d1,pcd_vibrato_pos_w(a5)
asr.w #3,d1
add.w d1,d0
.vibrato_disabled
.vibrato_still_delayed
; ----------------------------------------
; select right sample corresponding to current pitch
move.w pcd_out_len_w(a5),d3
cmpi.w #$219,d0
ble .noclippitchhigh
move.w #$231,d6 ; That's probably B-3+1, mapping to period $71 (although $7c is the last safe value)
btst #2,wi_flags_b(a3)
beq .noclippitchlow2
; select high pitch version of the sample
moveq.l #0,d2
move.w wi_chipram_w(a3),d2
move.w d0,d5
sub.w #$219,d5
lsr.w #6,d5
lea pre_octave_select_table(pc),a1
moveq.l #0,d1
move.b (a1,d5.w),d1
move.w pcd_out_lof_w(a5),d7
addq.w #1,d7 ; compare to $ffff
beq.s .has_no_loop_offset
subq.w #1,d7
lsr.w d1,d7
move.w d7,pcd_out_lof_w(a5)
lsr.w d1,d3
move.w d3,pcd_out_len_w(a5)
bra.s .lbC0005C0
.has_no_loop_offset
tst.b pcd_out_trg_b(a5)
beq.s .no_retrigger_new
move.w pcd_inst_wave_num_w(a5),d7
movea.l pv_wave_sample_table(a4,d7.w),a3
move.l pcd_out_ptr_l(a5),d4
sub.l a3,d4
move.w d3,d6
add.w d4,d6
move.w d2,d7
sub.w d6,d7
cmp.w d7,d3
bcc.s .fix_sample_loop
moveq.l #2,d3
bra.s .cont_after_one_shot
.fix_sample_loop
add.w d6,d3
sub.w d2,d3
lsr.w d1,d3
.cont_after_one_shot
move.w d3,pcd_out_len_w(a5)
lsr.w d1,d4
add.l a3,d4
move.l d4,pcd_out_ptr_l(a5)
.lbC0005C0
subq.b #1,d1
bmi.s .is_normal_octave
; find offset in sample buffer for the right octave
moveq.l #0,d4
.movetoloopposloop
add.l d2,d4
lsr.w #1,d2
dbra d1,.movetoloopposloop
add.l d4,pcd_out_ptr_l(a5)
.no_retrigger_new
.is_normal_octave
moveq.l #0,d1
move.b pre_octave_note_offset_table-pre_octave_select_table(a1,d5.w),d1
add.w d1,d1
add.w d1,d1
sub.w d1,d0
cmpi.w #$231,d0
ble.s .noclippitchhigh
move.w #$231,d0
.noclippitchhigh
tst.w d0
bge.s .noclippitchlow
moveq.l #0,d0
.noclippitchlow
move.w d0,d6
.noclippitchlow2
tst.b pcd_out_trg_b(a5)
beq.s .wasnottriggered
moveq.l #0,d0
move.w pcd_out_lof_w(a5),d0
addq.w #1,d0 ; compare to $ffff
beq.s .hasnoloop2
subq.w #1,d0
add.l d0,pcd_out_ptr_l(a5)
clr.w pcd_out_lof_w(a5)
.wasnottriggered
.hasnoloop2
cmp.w pcd_last_trigger_pos_w(a5),d3
beq.s .hassamesampos
move.w d3,pcd_last_trigger_pos_w(a5)
move.b pcd_channel_mask_b(a5),d3
move.b d3,pcd_out_trg_b(a5)
or.b d3,pv_trigger_mask_w+1(a4)
.hassamesampos
add.w d6,d6
move.w pv_period_table(a4,d6.w),pcd_out_per_w(a5)
; ----------------------------------------
; track delay handling
move.b pcd_track_delay_steps_b(a5),d3
beq .incrementchannel ; no track delay
cmp.b #NUM_CHANNELS-1,pcd_channel_num_b(a5) ; last channel processed?
beq .updatechannels ; no track delay for last channel
moveq.l #MAX_TRACK_DELAY-1,d0 ; load from last buffer
; handle track delay
cmpi.b #$FF,d3
beq.s .clear_track_delay
; advance and wrap offset
move.b pcd_SIZEOF+pcd_track_delay_offset_b(a5),d1
addq.b #1,d1
and.w d0,d1
move.b d1,pcd_SIZEOF+pcd_track_delay_offset_b(a5)
move.w d1,d2
lea pcd_SIZEOF(a5),a3
lsl.w #4,d2
lea pcd_track_delay_buffer(a3,d2.w),a3
lea pcd_out_base(a5),a1
move.l (a1)+,(a3)+ ; ocd_sam_ptr
move.l (a1)+,(a3)+ ; ocd_length/ocd_loop_offset
move.l (a1)+,(a3)+ ; ocd_period/ocd_volume/ocd_trigger
;move.l (a1)+,(a3)+ ; this is never used
move.b -(a3),d2
add.b d2,d2 ; increment channel
bne.s .copy_trigger_for_delayed_channel
tst.b pcd_SIZEOF+pcd_track_delay_steps_b(a5)
bne.s .dont_trigger_track_delay_first_note
move.b d2,(a3) ; trigger note (ocd_trigger)
.copy_trigger_for_delayed_channel
or.b d2,pv_trigger_mask_w+1(a4)
.dont_trigger_track_delay_first_note
IFNE PRETRACKER_VOLUME_TABLE
lea pv_volume_table(a4),a1
move.b pcd_track_delay_vol16_b(a5),-(sp)
move.w (sp)+,d4
clr.b d4
add.w d4,d4
move.b -(a3),d4 ; ocd_volume
move.b (a1,d4.w),(a3)+
ELSE
moveq.l #0,d4
move.b -(a3),d4 ; ocd_volume
move.b pcd_track_delay_vol16_b(a5),d2
ext.w d2
mulu d4,d2 ; apply track delay volume
lsr.w #4,d2
move.b d2,(a3)+ ; fix volume
ENDC
move.b d3,pcd_SIZEOF+pcd_track_delay_steps_b(a5)
sub.b d3,d1
and.w d1,d0
IFNE PRETRACKER_DUBIOUS_PITCH_SHIFT_FOR_DELAYED_TRACK
moveq.l #7,d4
and.w d4,d1
lea pre_minus4plus4_table(pc),a1
move.b (a1,d1.w),d1
ext.w d1
ENDC
bra.s .load_track_data_from_buffer
.clear_track_delay
moveq.l #0,d1
move.b d1,pcd_track_delay_steps_b(a5)
move.b d1,pcd_SIZEOF+pcd_pat_vol_b(a5)
move.b d1,pcd_SIZEOF+pcd_track_delay_steps_b(a5)
st pcd_SIZEOF+pcd_track_delay_offset_b(a5)
.load_track_data_from_buffer
lea pcd_SIZEOF(a5),a5 ; skip the channel we applied track delay to
lsl.w #4,d0
lea pcd_track_delay_buffer(a5,d0.w),a1
lea pcd_out_base(a5),a3
move.l (a1)+,(a3)+ ; ocd_sam_ptr
move.l (a1)+,(a3)+ ; ocd_length/ocd_loop_offset
move.l (a1)+,(a3)+ ; ocd_period/ocd_volume/ocd_trigger
;move.l (a1)+,(a3)+ ; this is never used
;clr.b ocd_volume-ocd_unused(a1) ; does not seem to bother
IFNE PRETRACKER_DUBIOUS_PITCH_SHIFT_FOR_DELAYED_TRACK
; FIXME this seems odd! Why modulate the period by the distance?
move.w pcd_out_base+ocd_period(a5),d2
move.w d1,d0
muls d2,d0
lsl.l #4,d0
swap d0
add.w d0,d2
move.w d2,pcd_out_base+ocd_period(a5)
ENDC
.incrementchannel
lea pcd_SIZEOF(a5),a5
cmp.b #NUM_CHANNELS-1,pcd_channel_num_b-pcd_SIZEOF(a5)
bne .inst_chan_loop
; ----------------------------------------
.updatechannels
; so this changed a lot from the original routine
move.w pv_trigger_mask_w(a4),d2
IFNE PRETRACKER_COPPER_OUTPUT
move.l pv_copperlist_ptr(a4),d0
beq .skipcopperlist
move.l d0,a5
move.b d2,1*4+3(a5) ; dmacon
move.b d2,(1+1+1+5*NUM_CHANNELS)*4+3(a5) ; dmacon after wait, dmacon, wait, 20 writes
lea pv_channeldata+pcd_out_base(a4),a0
move.l pv_sample_buffer_ptr(a4),d3
moveq.l #0,d5
move.w d5,pv_trigger_mask_w(a4)
moveq.l #-1,d1
lea 3*4+2(a5),a1
lea (1+1+1+5*NUM_CHANNELS+1+1)*4+2(a5),a2 ; wait, dmacon, wait, 20 writes, dmacon, wait
moveq.l #NUM_CHANNELS-1,d7
.checkchan
moveq.l #0,d2
move.w ocd_loop_offset(a0),d2
cmp.w d1,d2
bne.s .is_looping_sample
tst.b ocd_trigger(a0)
beq.s .one_shot_clear_loop
move.b d5,ocd_trigger(a0)
move.l ocd_sam_ptr(a0),d0
move.l d3,d6 ; set loop start
move.w ocd_length(a0),d4
lsr.w #1,d4
move.w d4,2*4(a1) ; ac_len
moveq.l #1,d4
bra.s .setptrvolper
.one_shot_clear_loop
move.l d3,d6
move.l d3,d0
moveq.l #1,d4
move.w d4,2*4(a1) ; ac_len
bra.s .setptrvolper
.is_looping_sample
move.l ocd_sam_ptr(a0),d0
add.l d2,d0
move.l d0,d6
move.w ocd_length(a0),d4
lsr.w #1,d4
move.w d4,2*4(a1) ; ac_len
tst.b ocd_trigger(a0)
beq.s .setptrvolper
move.b d5,ocd_trigger(a0)
sub.l d2,d6
.setptrvolper
move.w d0,1*4(a1) ; ac_ptr (lo)
swap d0
move.w d0,(a1) ; ac_ptr (hi)
move.w ocd_period(a0),3*4(a1)
move.b ocd_volume(a0),4*4+1(a1)
move.w d6,1*4(a2) ; ac_ptr (lo)
swap d6
move.w d6,(a2) ; ac_ptr (hi)
move.w d4,2*4(a2) ; ac_len
lea pcd_SIZEOF(a0),a0
lea 5*4(a1),a1
lea 3*4(a2),a2
dbra d7,.checkchan
.skipcopperlist
ELSE
lea $dff000,a5
; turn channels off and remember raster position
move.w d2,dmacon(a5) ; turn dma channels off
move.w vhposr(a5),d5 ; I know this only works for the lower 256 rasterlines...
add.w #4<<8,d5 ; target rasterpos
; in the meanwhile we can update both channels that
; - need triggering by setting the new start and length
; - need updating of loop offset only
; update volume and period in any case
lea pv_channeldata+pcd_out_base(a4),a0
lea aud0(a5),a1
move.l pv_sample_buffer_ptr(a4),d3
moveq.l #0,d6
move.w d6,pv_trigger_mask_w(a4)
moveq.l #-1,d1
moveq.l #NUM_CHANNELS-1,d7
.checkchan
tst.b ocd_trigger(a0)
beq.s .updateloop
move.b d6,ocd_trigger(a0)
; set start and length of the new sample
move.w ocd_length(a0),d0
lsr.w #1,d0
move.w d0,ac_len(a1)
move.l ocd_sam_ptr(a0),ac_ptr(a1)
bra.s .setvolperchan
.updateloop
; just update loop offset if looping
moveq.l #0,d0
move.w ocd_loop_offset(a0),d0
cmp.w d1,d0
beq.s .setvolperchan
add.l ocd_sam_ptr(a0),d0
move.l d0,ac_ptr(a1)
.setvolperchan
move.b ocd_volume(a0),ac_vol+1(a1)
move.w ocd_period(a0),ac_per(a1)
lea pcd_SIZEOF(a0),a0
lea ac_SIZEOF(a1),a1
dbra d7,.checkchan
tst.w d2
beq.s .skiprasterwait ; if no channel needed triggering, we are done!
or.w #DMAF_SETCLR,d2
.rasterwait1
cmp.w vhposr(a5),d5
bgt.s .rasterwait1
move.w d2,dmacon(a5) ; enable triggered channels
add.w #4<<8,d5 ; target rasterpos
.rasterwait2
cmp.w vhposr(a5),d5
bgt.s .rasterwait2
lea pv_channeldata+pcd_out_base(a4),a0
lea aud(a5),a1
.chanloop
lsr.b #1,d2
bcc.s .nosetloopchan
moveq.l #0,d0
move.w ocd_loop_offset(a0),d0
cmp.w d1,d0
beq.s .setchan_no_loop2
add.l ocd_sam_ptr(a0),d0
bra.s .keepchanrunning2
.setchan_no_loop2
move.l d3,d0
move.w #1,ac_len(a1)
.keepchanrunning2
move.l d0,ac_ptr(a1)
.nosetloopchan
lea pcd_SIZEOF(a0),a0
lea ac_SIZEOF(a1),a1
tst.b d2
bne.s .chanloop
.skiprasterwait
ENDC
IFNE PRETRACKER_DONT_TRASH_REGS
movem.l (sp)+,d2-d7/a2-a6
ENDC
rts
;--------------------------------------------------------------------
; table data currently about 594 bytes
; I assume this is a log table for freq distances within an octave
pre_log12_table:
dc.b $400000/$8000,$400000/$871D,$400000/$8F2F,$400000/$97B7,$400000/$9FC4,$400000/$A9DE
dc.b $400000/$B505,$400000/$BF49,$400000/$CB31,$400000/$D645,$400000/$E215,$400000/$F1A0
; linear then steep quadratic slope
pre_vib_speed_table:
dc.b 2,3,4,5,6,7,8,9,10,11,12,13,14,20,40,80
; linear (a bit wonky), then a bit quadratic, then steep
pre_vib_depth_table:
dc.b 0,8,9,10,11,12,13,14,18,20,28,40,50,70,160,255
; linear then almost quadratic
pre_vib_delay_table:
dc.b 0,4,8,10,12,14,16,18,20,24,32,40,56,96,150,255
pre_ramp_up_16:
dc.b 0,1,3,6,7,9,10,11,12,13,14,16,19,35,55,143
pre_fast_roll_off_16:
dc.w $400,$200,$80,$64,$50,$40,$30,$20,$10,14,12,10,8
dc.w 4,2,1
pre_roll_off_table:
dc.w $400,$200,$180,$140,$100,$C0,$A0,$80,$78,$74,$6E
dc.w $69,$64,$5A,$46,$40,$38,$30,$28,$20,$1F,$1E,$1D
dc.w $1C,$1B,$1A,$19,$18,$17,$16,$15,$14,$13,$12,$11
dc.w $10,15,14,13,13,12,12,11,11,10,10,9,9,8,8,8,8,7,7
dc.w 7,7,6,6,6,6,5,5,5,5,4,4,4,4,4,4,4,4,4,4,3,4,4,3,4
dc.w 4,3,4,3,4,3,4,3,4,3,3,3,3,3,3,3,3,3,2,3,3,3,2,3,3
dc.w 2,3,3,2,3,3,2,3,2,3,2,3,2,3,2,3,2,2,2,2,2,2,2,2,1
dc.w 2,1,2,1,2,1,2,1,1,2,1,1,1,2,1
pre_octave_note_offset_table:
dc.b 1*NOTES_IN_OCTAVE*4,1*NOTES_IN_OCTAVE*4,1*NOTES_IN_OCTAVE*4
dc.b 2*NOTES_IN_OCTAVE*4,2*NOTES_IN_OCTAVE*4,2*NOTES_IN_OCTAVE*4
dc.b 3*NOTES_IN_OCTAVE*4,3*NOTES_IN_OCTAVE*4,3*NOTES_IN_OCTAVE*4
dc.b 3*NOTES_IN_OCTAVE*4,3*NOTES_IN_OCTAVE*4,3*NOTES_IN_OCTAVE*4
dc.b 3*NOTES_IN_OCTAVE*4,3*NOTES_IN_OCTAVE*4,3*NOTES_IN_OCTAVE*4
IFNE PRETRACKER_PARANOIA_MODE
dc.b 3*NOTES_IN_OCTAVE*4,3*NOTES_IN_OCTAVE*4,3*NOTES_IN_OCTAVE*4
dc.b 3*NOTES_IN_OCTAVE*4,3*NOTES_IN_OCTAVE*4,3*NOTES_IN_OCTAVE*4
dc.b 3*NOTES_IN_OCTAVE*4,3*NOTES_IN_OCTAVE*4,3*NOTES_IN_OCTAVE*4
ELSE
dc.b 3*NOTES_IN_OCTAVE*4
ENDC
even
; based on pitch $219 (537), covers up to pitch 0x819 (2073).
; In practice, I was unable to get higher than $5ff
pre_octave_select_table:
dc.b 1,1,1
dc.b 2,2,2
dc.b 3,3,3
dc.b 3,3,3
dc.b 3,3,3
dc.b 3,3,3
IFNE PRETRACKER_PARANOIA_MODE
dc.b 3,3,3
dc.b 3,3,3
ELSE
dc.b 3
ENDC
even
IFNE PRETRACKER_DUBIOUS_PITCH_SHIFT_FOR_DELAYED_TRACK
; -4,-3,-1,1,2,3,4,0
pre_minus4plus4_table:
dc.b $FC,$FB,$FF,1,2,3,4,0
ENDC
pre_ramp_up_down_32:
dc.w 0*32,1*32,2*32,3*32,4*32,5*32,6*32,7*32,8*32,9*32,10*32,11*32,12*32,13*32,14*32,15*32
dc.w 15*32,14*32,13*32,12*32,11*32,10*32,9*32,8*32,7*32,6*32,5*32,4*32,3*32,2*32,1*32,0*32
pre_modulator_ramp_8:
;dc.w 77,293,539,1079,1337,1877,2431,3031 ; the 1079 value is strange (938 better?)
dc.w $4D,$125,$21B,$437,$539,$755,$96D,$BD7
pre_period_table:
dc.w $350,$320,$2F2,$2C8,$2A0,$279,$256,$236,$216,$1F8,$1DC,$1C0
dc.w $1A8,$190,$179,$164,$151,$13E,$12C,$11B,$10B,$FC,$EE,$E0
dc.w $D4,$C8,$BD,$B2,$A8,$9F,$96,$8D,$86,$7E,$78,$71
dc.w $71