2007-05-30 21:56:52 +00:00
|
|
|
/* ScummVM - Graphic Adventure Engine
|
|
|
|
*
|
|
|
|
* ScummVM is the legal property of its developers, whose names
|
|
|
|
* are too numerous to list here. Please refer to the COPYRIGHT
|
|
|
|
* file distributed with this source distribution.
|
2003-05-26 22:03:04 +00:00
|
|
|
*
|
|
|
|
* 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
|
2005-10-18 01:30:26 +00:00
|
|
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
2003-05-26 22:03:04 +00:00
|
|
|
*
|
2006-02-11 09:55:41 +00:00
|
|
|
* $URL$
|
|
|
|
* $Id$
|
2003-05-26 22:03:04 +00:00
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
2007-02-19 17:48:19 +00:00
|
|
|
#ifndef SCUMM_PLAYER_V2_H
|
|
|
|
#define SCUMM_PLAYER_V2_H
|
2003-05-26 22:03:04 +00:00
|
|
|
|
|
|
|
#include "common/scummsys.h"
|
2005-01-10 22:06:49 +00:00
|
|
|
#include "common/mutex.h"
|
2003-09-07 16:16:19 +00:00
|
|
|
#include "scumm/music.h"
|
2004-11-27 16:12:11 +00:00
|
|
|
#include "sound/audiostream.h"
|
2007-02-20 18:50:17 +00:00
|
|
|
#include "sound/mixer.h"
|
2003-10-03 18:33:57 +00:00
|
|
|
|
|
|
|
namespace Scumm {
|
|
|
|
|
|
|
|
class ScummEngine;
|
|
|
|
|
2006-07-23 16:13:26 +00:00
|
|
|
#include "common/pack-start.h" // START STRUCT PACKING
|
2003-05-26 22:03:04 +00:00
|
|
|
|
|
|
|
struct channel_data {
|
|
|
|
uint16 time_left; // 00
|
|
|
|
uint16 next_cmd; // 02
|
|
|
|
uint16 base_freq; // 04
|
|
|
|
uint16 freq_delta; // 06
|
|
|
|
uint16 freq; // 08
|
|
|
|
uint16 volume; // 10
|
|
|
|
uint16 volume_delta; // 12
|
|
|
|
uint16 tempo; // 14
|
|
|
|
uint16 inter_note_pause; // 16
|
|
|
|
uint16 transpose; // 18
|
|
|
|
uint16 note_length; // 20
|
|
|
|
uint16 hull_curve; // 22
|
|
|
|
uint16 hull_offset; // 24
|
|
|
|
uint16 hull_counter; // 26
|
|
|
|
uint16 freqmod_table; // 28
|
|
|
|
uint16 freqmod_offset; // 30
|
|
|
|
uint16 freqmod_incr; // 32
|
|
|
|
uint16 freqmod_multiplier; // 34
|
|
|
|
uint16 freqmod_modulo; // 36
|
2003-06-01 16:42:29 +00:00
|
|
|
uint16 unknown[4]; // 38 - 44
|
|
|
|
uint16 music_timer; // 46
|
2003-05-26 22:03:04 +00:00
|
|
|
uint16 music_script_nr; // 48
|
2007-07-01 18:18:43 +00:00
|
|
|
} PACKED_STRUCT;
|
2003-05-26 22:03:04 +00:00
|
|
|
|
2006-07-23 16:13:26 +00:00
|
|
|
#include "common/pack-end.h" // END STRUCT PACKING
|
2003-06-15 01:00:43 +00:00
|
|
|
|
|
|
|
|
2003-10-03 23:34:06 +00:00
|
|
|
/**
|
2003-10-05 17:32:38 +00:00
|
|
|
* Scumm V2 PC-Speaker MIDI driver.
|
|
|
|
* This simulates the pc speaker sound, which is driven by the 8253 (square
|
|
|
|
* wave generator) and a low-band filter.
|
2003-10-03 23:34:06 +00:00
|
|
|
*/
|
2006-04-29 22:33:31 +00:00
|
|
|
class Player_V2 : public Audio::AudioStream, public MusicEngine {
|
2003-05-26 22:03:04 +00:00
|
|
|
public:
|
2006-10-21 12:44:10 +00:00
|
|
|
Player_V2(ScummEngine *scumm, Audio::Mixer *mixer, bool pcjr);
|
2003-08-14 02:28:19 +00:00
|
|
|
virtual ~Player_V2();
|
2003-05-26 22:03:04 +00:00
|
|
|
|
2004-11-28 21:24:02 +00:00
|
|
|
virtual void setMusicVolume(int vol);
|
2003-10-03 13:44:12 +00:00
|
|
|
virtual void startSound(int sound);
|
|
|
|
virtual void stopSound(int sound);
|
2003-08-14 02:28:19 +00:00
|
|
|
virtual void stopAllSounds();
|
|
|
|
virtual int getMusicTimer() const;
|
2003-10-03 13:44:12 +00:00
|
|
|
virtual int getSoundStatus(int sound) const;
|
2003-05-26 22:03:04 +00:00
|
|
|
|
2004-11-27 16:18:50 +00:00
|
|
|
// AudioStream API
|
2004-11-27 16:12:11 +00:00
|
|
|
int readBuffer(int16 *buffer, const int numSamples) {
|
|
|
|
do_mix(buffer, numSamples / 2);
|
|
|
|
return numSamples;
|
|
|
|
}
|
|
|
|
bool isStereo() const { return true; }
|
|
|
|
bool endOfData() const { return false; }
|
|
|
|
int getRate() const { return _sample_rate; }
|
|
|
|
|
2003-08-14 02:28:19 +00:00
|
|
|
protected:
|
2003-06-27 00:10:17 +00:00
|
|
|
bool _isV3Game;
|
2005-05-10 23:48:48 +00:00
|
|
|
Audio::Mixer *_mixer;
|
2007-02-20 18:50:17 +00:00
|
|
|
Audio::SoundHandle _soundHandle;
|
2004-01-08 20:37:26 +00:00
|
|
|
ScummEngine *_vm;
|
2003-05-31 16:54:46 +00:00
|
|
|
|
|
|
|
bool _pcjr;
|
2003-06-06 21:56:37 +00:00
|
|
|
int _header_len;
|
2003-05-31 16:54:46 +00:00
|
|
|
|
2003-08-05 17:52:47 +00:00
|
|
|
uint32 _sample_rate;
|
|
|
|
uint32 _next_tick;
|
|
|
|
uint32 _tick_len;
|
2003-05-31 16:54:46 +00:00
|
|
|
unsigned int _update_step;
|
|
|
|
unsigned int _decay;
|
2003-09-30 17:10:03 +00:00
|
|
|
int _level;
|
2003-05-31 16:54:46 +00:00
|
|
|
unsigned int _RNG;
|
|
|
|
unsigned int _volumetable[16];
|
2003-05-26 22:03:04 +00:00
|
|
|
|
2003-05-31 16:54:46 +00:00
|
|
|
int _timer_count[4];
|
|
|
|
int _timer_output;
|
|
|
|
|
2003-08-14 19:41:40 +00:00
|
|
|
int _current_nr;
|
|
|
|
byte *_current_data;
|
|
|
|
int _next_nr;
|
|
|
|
byte *_next_data;
|
|
|
|
byte *_retaddr;
|
2003-05-26 22:03:04 +00:00
|
|
|
|
2003-09-19 14:43:57 +00:00
|
|
|
private:
|
2003-10-03 23:34:06 +00:00
|
|
|
union ChannelInfo {
|
|
|
|
channel_data d;
|
|
|
|
uint16 array[sizeof(channel_data)/2];
|
|
|
|
};
|
2005-07-30 21:11:48 +00:00
|
|
|
|
2003-09-19 14:43:57 +00:00
|
|
|
int _music_timer;
|
|
|
|
int _music_timer_ctr;
|
|
|
|
int _ticks_per_music_timer;
|
|
|
|
|
|
|
|
const uint16 *_freqs_table;
|
|
|
|
|
2005-01-28 22:05:51 +00:00
|
|
|
Common::Mutex _mutex;
|
2003-09-19 14:43:57 +00:00
|
|
|
ChannelInfo _channels[5];
|
|
|
|
|
|
|
|
protected:
|
2005-01-10 22:06:49 +00:00
|
|
|
void mutex_up();
|
|
|
|
void mutex_down();
|
2003-05-31 16:54:46 +00:00
|
|
|
|
2003-09-19 14:43:57 +00:00
|
|
|
virtual void nextTick();
|
2003-08-14 02:28:19 +00:00
|
|
|
virtual void clear_channel(int i);
|
|
|
|
virtual void chainSound(int nr, byte *data);
|
|
|
|
virtual void chainNextSound();
|
2003-05-26 22:03:04 +00:00
|
|
|
|
2003-09-19 14:43:57 +00:00
|
|
|
virtual void generateSpkSamples(int16 *data, uint len);
|
|
|
|
virtual void generatePCjrSamples(int16 *data, uint len);
|
2003-05-31 16:54:46 +00:00
|
|
|
|
2003-08-05 17:52:47 +00:00
|
|
|
void lowPassFilter(int16 *data, uint len);
|
2003-05-31 16:54:46 +00:00
|
|
|
void squareGenerator(int channel, int freq, int vol,
|
2003-11-08 21:59:32 +00:00
|
|
|
int noiseFeedback, int16 *sample, uint len);
|
|
|
|
|
2003-09-19 14:43:57 +00:00
|
|
|
private:
|
|
|
|
void do_mix(int16 *buf, uint len);
|
|
|
|
|
|
|
|
void set_pcjr(bool pcjr);
|
|
|
|
void execute_cmd(ChannelInfo *channel);
|
|
|
|
void next_freqs(ChannelInfo *channel);
|
2003-05-26 22:03:04 +00:00
|
|
|
};
|
|
|
|
|
2009-06-06 18:21:49 +00:00
|
|
|
/**
|
|
|
|
* Scumm V2 CMS/Gameblaster MIDI driver.
|
|
|
|
*/
|
|
|
|
class Player_V2CMS : public Audio::AudioStream, public MusicEngine {
|
|
|
|
public:
|
|
|
|
Player_V2CMS(ScummEngine *scumm, Audio::Mixer *mixer);
|
|
|
|
virtual ~Player_V2CMS();
|
|
|
|
|
|
|
|
virtual void setMusicVolume(int vol);
|
|
|
|
virtual void startSound(int sound);
|
|
|
|
virtual void stopSound(int sound);
|
|
|
|
virtual void stopAllSounds();
|
|
|
|
virtual int getMusicTimer() const;
|
|
|
|
virtual int getSoundStatus(int sound) const;
|
|
|
|
|
|
|
|
// AudioStream API
|
|
|
|
int readBuffer(int16 *buffer, const int numSamples);
|
|
|
|
bool isStereo() const { return true; }
|
|
|
|
bool endOfData() const { return false; }
|
|
|
|
int getRate() const { return _sample_rate; }
|
|
|
|
|
|
|
|
protected:
|
|
|
|
|
|
|
|
#include "common/pack-start.h" // START STRUCT PACKING
|
|
|
|
struct Voice {
|
|
|
|
byte attack;
|
|
|
|
byte decay;
|
|
|
|
byte sustain;
|
|
|
|
byte release;
|
|
|
|
byte octadd;
|
|
|
|
int16 vibrato;
|
|
|
|
int16 vibrato2;
|
|
|
|
int16 noise;
|
|
|
|
} PACKED_STRUCT;
|
|
|
|
|
|
|
|
struct Voice2 {
|
|
|
|
byte *amplitudeOutput;
|
|
|
|
byte *freqOutput;
|
|
|
|
byte *octaveOutput;
|
|
|
|
|
2009-06-06 18:23:24 +00:00
|
|
|
uint8 channel;
|
2009-06-06 18:21:49 +00:00
|
|
|
int8 sustainLevel;
|
|
|
|
int8 attackRate;
|
2009-06-06 18:23:24 +00:00
|
|
|
uint8 maxAmpl;
|
2009-06-06 18:21:49 +00:00
|
|
|
int8 decayRate;
|
|
|
|
int8 sustainRate;
|
|
|
|
int8 releaseRate;
|
|
|
|
int8 releaseTime;
|
|
|
|
int8 vibratoRate;
|
|
|
|
int8 vibratoDepth;
|
|
|
|
|
|
|
|
int8 curVibratoRate;
|
|
|
|
int8 curVibratoUnk;
|
|
|
|
|
|
|
|
int8 unkVibratoRate;
|
|
|
|
int8 unkVibratoDepth;
|
|
|
|
|
|
|
|
int8 unkRate;
|
|
|
|
int8 unkCount;
|
|
|
|
|
|
|
|
int nextProcessState;
|
|
|
|
int8 curVolume;
|
|
|
|
int8 curOctave;
|
|
|
|
int8 curFreq;
|
|
|
|
|
|
|
|
int8 octaveAdd;
|
|
|
|
|
|
|
|
int8 playingNote;
|
|
|
|
Voice2 *nextVoice;
|
|
|
|
|
|
|
|
byte chanNumber;
|
|
|
|
} PACKED_STRUCT;
|
|
|
|
|
|
|
|
struct MusicChip {
|
|
|
|
byte ampl[4];
|
|
|
|
byte freq[4];
|
|
|
|
byte octave[2];
|
|
|
|
} PACKED_STRUCT;
|
|
|
|
#include "common/pack-end.h" // END STRUCT PACKING
|
|
|
|
|
|
|
|
Voice _cmsVoicesBase[16];
|
|
|
|
Voice2 _cmsVoices[8];
|
|
|
|
MusicChip _cmsChips[2];
|
|
|
|
|
2009-07-14 19:32:25 +00:00
|
|
|
int8 _tempo;
|
|
|
|
int8 _tempoSum;
|
2009-06-06 18:21:49 +00:00
|
|
|
byte _looping;
|
|
|
|
byte _octaveMask;
|
|
|
|
int16 _midiDelay;
|
|
|
|
Voice2 *_midiChannel[16];
|
|
|
|
byte _midiChannelUse[16];
|
|
|
|
byte *_midiData;
|
|
|
|
byte *_midiSongBegin;
|
|
|
|
|
|
|
|
int _loadedMidiSong;
|
|
|
|
|
|
|
|
byte _lastMidiCommand;
|
|
|
|
uint _outputTableReady;
|
|
|
|
byte _clkFrequenz;
|
|
|
|
byte _restart;
|
|
|
|
byte _curSno;
|
|
|
|
|
|
|
|
void loadMidiData(byte *data, int sound);
|
|
|
|
void play();
|
|
|
|
|
|
|
|
void processChannel(Voice2 *channel);
|
|
|
|
void processRelease(Voice2 *channel);
|
|
|
|
void processAttack(Voice2 *channel);
|
|
|
|
void processDecay(Voice2 *channel);
|
|
|
|
void processSustain(Voice2 *channel);
|
|
|
|
void processVibrato(Voice2 *channel);
|
|
|
|
|
|
|
|
void playMusicChips(const MusicChip *table);
|
|
|
|
void playNote(byte *&data);
|
|
|
|
void clearNote(byte *&data);
|
|
|
|
void offAllChannels();
|
|
|
|
void playVoice();
|
|
|
|
void processMidiData(uint ticks);
|
|
|
|
|
|
|
|
Voice2 *getFreeVoice();
|
|
|
|
Voice2 *getPlayVoice(byte param);
|
|
|
|
|
|
|
|
// from Player_V2
|
|
|
|
protected:
|
|
|
|
bool _isV3Game;
|
|
|
|
Audio::Mixer *_mixer;
|
|
|
|
Audio::SoundHandle _soundHandle;
|
|
|
|
ScummEngine *_vm;
|
|
|
|
|
|
|
|
int _header_len;
|
|
|
|
|
|
|
|
uint32 _sample_rate;
|
|
|
|
uint32 _next_tick;
|
|
|
|
uint32 _tick_len;
|
|
|
|
|
|
|
|
int _timer_count[4];
|
|
|
|
int _timer_output;
|
|
|
|
|
|
|
|
int _current_nr;
|
|
|
|
byte *_current_data;
|
|
|
|
int _next_nr;
|
|
|
|
byte *_next_data;
|
|
|
|
byte *_retaddr;
|
|
|
|
|
|
|
|
private:
|
|
|
|
union ChannelInfo {
|
|
|
|
channel_data d;
|
|
|
|
uint16 array[sizeof(channel_data)/2];
|
|
|
|
};
|
|
|
|
|
|
|
|
int _music_timer;
|
|
|
|
int _music_timer_ctr;
|
|
|
|
int _ticks_per_music_timer;
|
|
|
|
|
|
|
|
Common::Mutex _mutex;
|
|
|
|
ChannelInfo _channels[5];
|
|
|
|
|
|
|
|
protected:
|
|
|
|
void mutex_up();
|
|
|
|
void mutex_down();
|
|
|
|
|
|
|
|
virtual void nextTick();
|
|
|
|
virtual void clear_channel(int i);
|
|
|
|
virtual void chainSound(int nr, byte *data);
|
|
|
|
virtual void chainNextSound();
|
|
|
|
|
|
|
|
private:
|
|
|
|
void do_mix(int16 *buf, uint len);
|
|
|
|
|
|
|
|
void execute_cmd(ChannelInfo *channel);
|
|
|
|
void next_freqs(ChannelInfo *channel);
|
|
|
|
};
|
|
|
|
|
2003-10-03 18:33:57 +00:00
|
|
|
} // End of namespace Scumm
|
|
|
|
|
2005-07-30 21:11:48 +00:00
|
|
|
#endif
|