/** * encoding: UTF-8 * This file is part of reSID, a MOS6581 SID emulator engine. * Copyright (C) 2004 Dag Lem <resid@nimrod.no> * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA * * @author Ken Händel * */ package resid; import resid.ISIDDefs.chip_model; public class Voice { protected WaveformGenerator wave = new WaveformGenerator(); protected EnvelopeGenerator envelope = new EnvelopeGenerator(); protected boolean muted; /** * Waveform D/A zero level. */ protected int /* sound_sample */wave_zero; /** * Multiplying D/A DC offset. */ protected int /* sound_sample */voice_DC; // ---------------------------------------------------------------------------- // Inline functions. // The following functions are defined inline because they are called every // time a sample is calculated. // ---------------------------------------------------------------------------- /** * Amplitude modulated waveform output. Ideal range [-2048*255, 2047*255]. */ public int /* sound_sample */output() { if (!muted) { // Multiply oscillator output with envelope output. return ((wave.output() - wave_zero) * envelope.output() + voice_DC); } else { return 0; } } // ---------------------------------------------------------------------------- // END Inline functions. // ---------------------------------------------------------------------------- /** * Constructor. */ public Voice() { muted = false; set_chip_model(chip_model.MOS6581); } /** * Set chip model. * * @param model */ public void set_chip_model(chip_model model) { wave.set_chip_model(model); if (model == chip_model.MOS6581) { // The waveform D/A converter introduces a DC offset in the signal // to the envelope multiplying D/A converter. The "zero" level of // the waveform D/A converter can be found as follows: // // Measure the "zero" voltage of voice 3 on the SID audio output // pin, routing only voice 3 to the mixer ($d417 = $0b, $d418 = // $0f, all other registers zeroed). // // Then set the sustain level for voice 3 to maximum and search for // the waveform output value yielding the same voltage as found // above. This is done by trying out different waveform output // values until the correct value is found, e.g. with the following // program: // // lda #$08 // sta $d412 // lda #$0b // sta $d417 // lda #$0f // sta $d418 // lda #$f0 // sta $d414 // lda #$21 // sta $d412 // lda #$01 // sta $d40e // // ldx #$00 // lda #$38 ; Tweak this to find the "zero" level // l cmp $d41b // bne l // stx $d40e ; Stop frequency counter - freeze waveform output // brk // // The waveform output range is 0x000 to 0xfff, so the "zero" // level should ideally have been 0x800. In the measured chip, the // waveform output "zero" level was found to be 0x380 (i.e. $d41b // = 0x38) at 5.94V. wave_zero = 0x380; // The envelope multiplying D/A converter introduces another DC // offset. This is isolated by the following measurements: // // * The "zero" output level of the mixer at full volume is 5.44V. // * Routing one voice to the mixer at full volume yields // 6.75V at maximum voice output (wave = 0xfff, sustain = 0xf) // 5.94V at "zero" voice output (wave = any, sustain = 0x0) // 5.70V at minimum voice output (wave = 0x000, sustain = 0xf) // * The DC offset of one voice is (5.94V - 5.44V) = 0.50V // * The dynamic range of one voice is |6.75V - 5.70V| = 1.05V // * The DC offset is thus 0.50V/1.05V ~ 1/2 of the dynamic range. // // Note that by removing the DC offset, we get the following ranges // for // one voice: // y > 0: (6.75V - 5.44V) - 0.50V = 0.81V // y < 0: (5.70V - 5.44V) - 0.50V = -0.24V // The scaling of the voice amplitude is not symmetric about y = 0; // this follows from the DC level in the waveform output. voice_DC = 0x800 * 0xff; } else { // No DC offsets in the MOS8580. wave_zero = 0x800; voice_DC = 0; } } /** * Set sync source. */ public void set_sync_source(Voice source) { wave.set_sync_source(source.wave); } /** * Register functions. * * @param control */ public void writeCONTROL_REG(int /* reg8 */control) { wave.writeCONTROL_REG(control); envelope.writeCONTROL_REG(control); } /** * SID reset. */ public void reset() { wave.reset(); envelope.reset(); } /** * Voice mute. * * @param enable */ public void mute(boolean enable) { // enable = true (means voice is muted) muted = enable; } }