mirror of
https://github.com/libretro/scummvm.git
synced 2025-04-04 07:41:58 +00:00
synced with scummvm to 2011-Apr-13
This commit is contained in:
parent
f0a4299aef
commit
0640dcf2c7
@ -1,3 +1,9 @@
|
|||||||
|
NOTE: Only certain parts of the Residual project are under the GNU LGPL.
|
||||||
|
The majority of the files are under the GNU GPL. See the headers of the
|
||||||
|
individual files to find out the exact license.
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
GNU LESSER GENERAL PUBLIC LICENSE
|
GNU LESSER GENERAL PUBLIC LICENSE
|
||||||
Version 2.1, February 1999
|
Version 2.1, February 1999
|
||||||
|
|
||||||
|
13
Makefile
13
Makefile
@ -26,7 +26,7 @@ ifeq "$(HAVE_GCC)" "1"
|
|||||||
CXXFLAGS+= -Wno-long-long -Wno-multichar -Wno-unknown-pragmas -Wno-reorder
|
CXXFLAGS+= -Wno-long-long -Wno-multichar -Wno-unknown-pragmas -Wno-reorder
|
||||||
# Enable even more warnings...
|
# Enable even more warnings...
|
||||||
CXXFLAGS+= -Wpointer-arith -Wcast-qual
|
CXXFLAGS+= -Wpointer-arith -Wcast-qual
|
||||||
CXXFLAGS+= -Wshadow -Wimplicit -Wnon-virtual-dtor -Wwrite-strings
|
CXXFLAGS+= -Wshadow -Wnon-virtual-dtor -Wwrite-strings
|
||||||
|
|
||||||
# Currently we disable this gcc flag, since it will also warn in cases,
|
# Currently we disable this gcc flag, since it will also warn in cases,
|
||||||
# where using GCC_PRINTF (means: __attribute__((format(printf, x, y))))
|
# where using GCC_PRINTF (means: __attribute__((format(printf, x, y))))
|
||||||
@ -48,9 +48,12 @@ ifeq "$(HAVE_CLANG)" "1"
|
|||||||
CXXFLAGS+= -Wno-conversion -Wno-shorten-64-to-32 -Wno-sign-compare -Wno-four-char-constants
|
CXXFLAGS+= -Wno-conversion -Wno-shorten-64-to-32 -Wno-sign-compare -Wno-four-char-constants
|
||||||
endif
|
endif
|
||||||
|
|
||||||
# Warn if global constructors are used. Only available in GCC with LLVM backend
|
ifeq "$(HAVE_ICC)" "1"
|
||||||
# (and maybe clang?), hence off by default.
|
# Disable some warnings:
|
||||||
#CXXFLAGS+= -Wglobal-constructors
|
# 161: unrecognized #pragma
|
||||||
|
# 1899: multicharacter character literal (potential portability problem)
|
||||||
|
CXXFLAGS+= -diag-disable 161,1899
|
||||||
|
endif
|
||||||
|
|
||||||
#######################################################################
|
#######################################################################
|
||||||
# Default commands - put the necessary replacements in config.mk #
|
# Default commands - put the necessary replacements in config.mk #
|
||||||
@ -69,7 +72,7 @@ ZIP ?= zip -q
|
|||||||
# Misc stuff - you should never have to edit this #
|
# Misc stuff - you should never have to edit this #
|
||||||
#######################################################################
|
#######################################################################
|
||||||
|
|
||||||
EXECUTABLE := residual$(EXEEXT)
|
EXECUTABLE := $(EXEPRE)residual$(EXEEXT)
|
||||||
|
|
||||||
include $(srcdir)/Makefile.common
|
include $(srcdir)/Makefile.common
|
||||||
|
|
||||||
|
@ -16,22 +16,22 @@ all: $(EXECUTABLE) plugins
|
|||||||
######################################################################
|
######################################################################
|
||||||
|
|
||||||
PLUGINS :=
|
PLUGINS :=
|
||||||
MODULES := base $(MODULES)
|
MODULES := devtools base $(MODULES)
|
||||||
|
|
||||||
-include $(srcdir)/engines/engines.mk
|
-include $(srcdir)/engines/engines.mk
|
||||||
|
|
||||||
# After the game specific modules follow the shared modules
|
# After the game specific modules follow the shared modules
|
||||||
MODULES += \
|
MODULES += \
|
||||||
gui \
|
gui \
|
||||||
sound \
|
|
||||||
backends \
|
backends \
|
||||||
engines \
|
engines \
|
||||||
graphics \
|
graphics \
|
||||||
|
audio \
|
||||||
common \
|
common \
|
||||||
po
|
po
|
||||||
|
|
||||||
ifdef USE_MT32EMU
|
ifdef USE_MT32EMU
|
||||||
MODULES += sound/softsynth/mt32
|
MODULES += audio/softsynth/mt32
|
||||||
endif
|
endif
|
||||||
|
|
||||||
######################################################################
|
######################################################################
|
||||||
@ -54,6 +54,10 @@ DEPFILES =
|
|||||||
# the build date in gScummVMBuildDate is correct.
|
# the build date in gScummVMBuildDate is correct.
|
||||||
base/version.o: $(filter-out base/libbase.a,$(OBJS))
|
base/version.o: $(filter-out base/libbase.a,$(OBJS))
|
||||||
|
|
||||||
|
ifdef USE_ELF_LOADER
|
||||||
|
backends/plugins/elf/version.o: $(filter-out base/libbase.a,$(filter-out backends/libbackends.a,$(OBJS)))
|
||||||
|
endif
|
||||||
|
|
||||||
# Replace regular output with quiet messages
|
# Replace regular output with quiet messages
|
||||||
ifneq ($(findstring $(MAKEFLAGS),s),s)
|
ifneq ($(findstring $(MAKEFLAGS),s),s)
|
||||||
ifneq ($(VERBOSE_BUILD),1)
|
ifneq ($(VERBOSE_BUILD),1)
|
||||||
@ -158,34 +162,36 @@ VER_EXTRA = $(shell echo $(VERSION) | cut -d. -f 3 | cut -c2-)
|
|||||||
|
|
||||||
|
|
||||||
######################################################################
|
######################################################################
|
||||||
# Get Subversion's working copy information
|
# Get git's working copy information
|
||||||
######################################################################
|
######################################################################
|
||||||
|
|
||||||
ifneq ($(shell svn info $(srcdir) 1>/dev/null 2>&1 || echo "error"),error)
|
ifneq ($(shell cd $(srcdir); git rev-parse --verify HEAD 1>/dev/null 2>&1 || echo "error"),error)
|
||||||
SVNROOT := $(srcdir)
|
GITROOT := $(srcdir)
|
||||||
ifeq ($(origin VER_SVNREV), undefined)
|
ifeq ($(origin VER_REV), undefined)
|
||||||
|
# Are there uncommitted changes? (describe --dirty is only available since 1.6.6)
|
||||||
|
VER_DIRTY := $(shell cd $(srcdir); git update-index --refresh --unmerged 1>/dev/null 2>&1; git diff-index --quiet HEAD || echo "-dirty")
|
||||||
# Get the working copy base revision
|
# Get the working copy base revision
|
||||||
VER_SVNREV := $(shell LANG=C svn info $(SVNROOT) | grep "^Revision" | cut -d ' ' -f 2)
|
VER_REV := $(shell cd $(srcdir); git describe --match desc/\* | cut -d '-' -f 2-)$(VER_DIRTY)
|
||||||
endif
|
endif
|
||||||
else
|
else
|
||||||
SVNROOT := https://resisual.svn.sourceforge.net/svnroot/residual/residual/trunk/
|
GITROOT := git://github.com/residual/residual.git
|
||||||
endif
|
endif
|
||||||
|
|
||||||
# Define the Subversion revision if available, either autodetected or
|
# Define the Subversion revision if available, either autodetected or
|
||||||
# specified by the user, but only for base/version.cpp.
|
# specified by the user, but only for base/version.cpp.
|
||||||
ifneq ($(origin VER_SVNREV), undefined)
|
ifneq ($(origin VER_REV), undefined)
|
||||||
CXXFLAGS+= -DRESIDUAL_SVN_REVISION=\"$(VER_SVNREV)\"
|
base/version.o: CXXFLAGS:=$(CXXFLAGS) -DRESIDUAL_REVISION=\"$(VER_REV)\"
|
||||||
endif
|
endif
|
||||||
|
|
||||||
######################################################################
|
######################################################################
|
||||||
# Distribution settings
|
# Distribution settings
|
||||||
######################################################################
|
######################################################################
|
||||||
|
|
||||||
ifeq ($(VER_EXTRA),svn)
|
ifeq ($(VER_EXTRA),git)
|
||||||
ifeq ($(origin VER_SVNREV), undefined)
|
ifeq ($(origin VER_REV), undefined)
|
||||||
DISTVERSION = $(shell date '+%Y-%m-%d')
|
DISTVERSION = $(shell date '+%Y-%m-%d')
|
||||||
else
|
else
|
||||||
DISTVERSION = svn$(VER_SVNREV)
|
DISTVERSION = git$(VER_REV)
|
||||||
endif
|
endif
|
||||||
else
|
else
|
||||||
DISTVERSION = $(VERSION)
|
DISTVERSION = $(VERSION)
|
||||||
@ -195,14 +201,15 @@ DISTNAME := residual-$(DISTVERSION)
|
|||||||
DISTDIR := dist
|
DISTDIR := dist
|
||||||
VERFILE := $(DISTDIR)/$(DISTNAME)/base/internal_version.h
|
VERFILE := $(DISTDIR)/$(DISTNAME)/base/internal_version.h
|
||||||
|
|
||||||
|
# TODO git via $(GITROOT)
|
||||||
$(VERFILE): $(srcdir)/base/internal_version.h
|
$(VERFILE): $(srcdir)/base/internal_version.h
|
||||||
@$(RM_REC) $(DISTDIR)
|
@$(RM_REC) $(DISTDIR)
|
||||||
@$(MKDIR) $(DISTDIR)
|
@$(MKDIR) $(DISTDIR)
|
||||||
svn export $(SVNROOT) $(DISTDIR)/$(DISTNAME)
|
svn export $(SVNROOT) $(DISTDIR)/$(DISTNAME)
|
||||||
ifneq ($(origin VER_SVNREV), undefined)
|
ifneq ($(origin VER_REV), undefined)
|
||||||
@# Use the current SVN revision as a default for the snapshot sources
|
@# Use the current SVN revision as a default for the snapshot sources
|
||||||
@svn cat $(SVNROOT)/base/internal_version.h | sed -e \
|
@svn cat $(SVNROOT)/base/internal_version.h | sed -e \
|
||||||
"s/^#define RESIDUAL_SVN_REVISION$$/#define RESIDUAL_SVN_REVISION \"$(VER_SVNREV)\"/g" \
|
"s/^#define RESIDUAL_REVISION$$/#define RESIDUAL_REVISION \"$(VER_REV)\"/g" \
|
||||||
> $(VERFILE)
|
> $(VERFILE)
|
||||||
endif
|
endif
|
||||||
|
|
||||||
|
@ -29,12 +29,12 @@
|
|||||||
#include "common/queue.h"
|
#include "common/queue.h"
|
||||||
#include "common/util.h"
|
#include "common/util.h"
|
||||||
|
|
||||||
#include "sound/audiostream.h"
|
#include "audio/audiostream.h"
|
||||||
#include "sound/decoders/flac.h"
|
#include "audio/decoders/flac.h"
|
||||||
#include "sound/mixer.h"
|
#include "audio/mixer.h"
|
||||||
#include "sound/decoders/mp3.h"
|
#include "audio/decoders/mp3.h"
|
||||||
#include "sound/decoders/raw.h"
|
#include "audio/decoders/raw.h"
|
||||||
#include "sound/decoders/vorbis.h"
|
#include "audio/decoders/vorbis.h"
|
||||||
|
|
||||||
|
|
||||||
namespace Audio {
|
namespace Audio {
|
||||||
@ -228,8 +228,8 @@ int SubLoopingAudioStream::readBuffer(int16 *buffer, const int numSamples) {
|
|||||||
SubSeekableAudioStream::SubSeekableAudioStream(SeekableAudioStream *parent, const Timestamp start, const Timestamp end, DisposeAfterUse::Flag disposeAfterUse)
|
SubSeekableAudioStream::SubSeekableAudioStream(SeekableAudioStream *parent, const Timestamp start, const Timestamp end, DisposeAfterUse::Flag disposeAfterUse)
|
||||||
: _parent(parent), _disposeAfterUse(disposeAfterUse),
|
: _parent(parent), _disposeAfterUse(disposeAfterUse),
|
||||||
_start(convertTimeToStreamPos(start, getRate(), isStereo())),
|
_start(convertTimeToStreamPos(start, getRate(), isStereo())),
|
||||||
_pos(0, getRate() * (isStereo() ? 2 : 1)),
|
_pos(0, getRate() * (isStereo() ? 2 : 1)),
|
||||||
_length(convertTimeToStreamPos(end - start, getRate(), isStereo())) {
|
_length(convertTimeToStreamPos(end, getRate(), isStereo()) - _start) {
|
||||||
|
|
||||||
assert(_length.totalNumberOfFrames() % (isStereo() ? 2 : 1) == 0);
|
assert(_length.totalNumberOfFrames() % (isStereo() ? 2 : 1) == 0);
|
||||||
_parent->seek(_start);
|
_parent->seek(_start);
|
||||||
@ -393,7 +393,7 @@ Timestamp convertTimeToStreamPos(const Timestamp &where, int rate, bool isStereo
|
|||||||
//
|
//
|
||||||
// An example is when converting the timestamp 500ms to a 11025 Hz based
|
// An example is when converting the timestamp 500ms to a 11025 Hz based
|
||||||
// stream. It would have an internal frame counter of 5512.5. Now when
|
// stream. It would have an internal frame counter of 5512.5. Now when
|
||||||
// doing calculations at frame precision, this might lead to unexpected
|
// doing calculations at frame precision, this might lead to unexpected
|
||||||
// results: The frame difference between a timestamp 1000ms and the above
|
// results: The frame difference between a timestamp 1000ms and the above
|
||||||
// mentioned timestamp (both with 11025 as framerate) would be 5512,
|
// mentioned timestamp (both with 11025 as framerate) would be 5512,
|
||||||
// instead of 5513, which is what a frame-precision based code would expect.
|
// instead of 5513, which is what a frame-precision based code would expect.
|
@ -30,7 +30,7 @@
|
|||||||
#include "common/sys.h"
|
#include "common/sys.h"
|
||||||
#include "common/types.h"
|
#include "common/types.h"
|
||||||
|
|
||||||
#include "sound/timestamp.h"
|
#include "audio/timestamp.h"
|
||||||
|
|
||||||
namespace Audio {
|
namespace Audio {
|
||||||
|
|
||||||
@ -276,11 +276,6 @@ private:
|
|||||||
* The same caveats apply to SubSeekableAudioStream as do to SeekableAudioStream.
|
* The same caveats apply to SubSeekableAudioStream as do to SeekableAudioStream.
|
||||||
*
|
*
|
||||||
* Manipulating the parent stream directly /will/ mess up a substream.
|
* Manipulating the parent stream directly /will/ mess up a substream.
|
||||||
*
|
|
||||||
* IMPORTANT:
|
|
||||||
* Note for engine authors. This object is currently under inspection. In case
|
|
||||||
* we need to revise the looping API we might drop this. So if you really need
|
|
||||||
* something like this object, please drop a mail to LordHoto.
|
|
||||||
*/
|
*/
|
||||||
class SubSeekableAudioStream : public SeekableAudioStream {
|
class SubSeekableAudioStream : public SeekableAudioStream {
|
||||||
public:
|
public:
|
@ -23,7 +23,10 @@
|
|||||||
*
|
*
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#include "sound/decoders/flac.h"
|
// Disable symbol overrides for FILE as that is used in FLAC headers
|
||||||
|
#define FORBIDDEN_SYMBOL_EXCEPTION_FILE
|
||||||
|
|
||||||
|
#include "audio/decoders/flac.h"
|
||||||
|
|
||||||
#ifdef USE_FLAC
|
#ifdef USE_FLAC
|
||||||
|
|
||||||
@ -31,8 +34,7 @@
|
|||||||
#include "common/stream.h"
|
#include "common/stream.h"
|
||||||
#include "common/util.h"
|
#include "common/util.h"
|
||||||
|
|
||||||
#include "sound/audiostream.h"
|
#include "audio/audiostream.h"
|
||||||
#include "sound/audiocd.h"
|
|
||||||
|
|
||||||
#define FLAC__NO_DLL // that MS-magic gave me headaches - just link the library you like
|
#define FLAC__NO_DLL // that MS-magic gave me headaches - just link the library you like
|
||||||
#include <FLAC/export.h>
|
#include <FLAC/export.h>
|
||||||
@ -140,7 +142,7 @@ public:
|
|||||||
bool seek(const Timestamp &where);
|
bool seek(const Timestamp &where);
|
||||||
Timestamp getLength() const { return _length; }
|
Timestamp getLength() const { return _length; }
|
||||||
|
|
||||||
bool isStreamDecoderReady() const { return getStreamDecoderState() == FLAC__STREAM_DECODER_SEARCH_FOR_FRAME_SYNC ; }
|
bool isStreamDecoderReady() const { return getStreamDecoderState() == FLAC__STREAM_DECODER_SEARCH_FOR_FRAME_SYNC; }
|
||||||
protected:
|
protected:
|
||||||
uint getChannels() const { return MIN<uint>(_streaminfo.channels, MAX_OUTPUT_CHANNELS); }
|
uint getChannels() const { return MIN<uint>(_streaminfo.channels, MAX_OUTPUT_CHANNELS); }
|
||||||
|
|
||||||
@ -303,7 +305,7 @@ int FLACStream::readBuffer(int16 *buffer, const int numSamples) {
|
|||||||
const uint numChannels = getChannels();
|
const uint numChannels = getChannels();
|
||||||
|
|
||||||
if (numChannels == 0) {
|
if (numChannels == 0) {
|
||||||
warning("FLACStream: Stream not sucessfully initialised, cant playback");
|
warning("FLACStream: Stream not successfully initialised, cant playback");
|
||||||
return -1; // streaminfo wasnt read!
|
return -1; // streaminfo wasnt read!
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -553,7 +555,7 @@ void FLACStream::convertBuffersGeneric(SampleType* bufDestination, const FLAC__i
|
|||||||
|
|
||||||
for (; numSamples > 0; numSamples -= numChannels) {
|
for (; numSamples > 0; numSamples -= numChannels) {
|
||||||
for (uint i = 0; i < numChannels; ++i)
|
for (uint i = 0; i < numChannels; ++i)
|
||||||
*bufDestination++ = static_cast<SampleType>(*(inChannels[i]++) >> kPower) ;
|
*bufDestination++ = static_cast<SampleType>(*(inChannels[i]++) >> kPower);
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
for (; numSamples > 0; numSamples -= numChannels) {
|
for (; numSamples > 0; numSamples -= numChannels) {
|
@ -27,10 +27,12 @@
|
|||||||
* @file
|
* @file
|
||||||
* Sound decoder used in engines:
|
* Sound decoder used in engines:
|
||||||
* - agos
|
* - agos
|
||||||
|
* - draci
|
||||||
* - kyra
|
* - kyra
|
||||||
* - m4
|
* - m4
|
||||||
* - queen
|
* - queen
|
||||||
* - saga
|
* - saga
|
||||||
|
* - sci
|
||||||
* - scumm
|
* - scumm
|
||||||
* - sword1
|
* - sword1
|
||||||
* - sword2
|
* - sword2
|
@ -23,7 +23,7 @@
|
|||||||
*
|
*
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#include "sound/decoders/mp3.h"
|
#include "audio/decoders/mp3.h"
|
||||||
|
|
||||||
#ifdef USE_MAD
|
#ifdef USE_MAD
|
||||||
|
|
||||||
@ -31,8 +31,7 @@
|
|||||||
#include "common/stream.h"
|
#include "common/stream.h"
|
||||||
#include "common/util.h"
|
#include "common/util.h"
|
||||||
|
|
||||||
#include "sound/audiocd.h"
|
#include "audio/audiostream.h"
|
||||||
#include "sound/audiostream.h"
|
|
||||||
|
|
||||||
#include <mad.h>
|
#include <mad.h>
|
||||||
|
|
@ -27,10 +27,13 @@
|
|||||||
* @file
|
* @file
|
||||||
* Sound decoder used in engines:
|
* Sound decoder used in engines:
|
||||||
* - agos
|
* - agos
|
||||||
|
* - draci
|
||||||
* - kyra
|
* - kyra
|
||||||
* - m4
|
* - m4
|
||||||
|
* - mohawk
|
||||||
* - queen
|
* - queen
|
||||||
* - saga
|
* - saga
|
||||||
|
* - sci
|
||||||
* - scumm
|
* - scumm
|
||||||
* - sword1
|
* - sword1
|
||||||
* - sword2
|
* - sword2
|
@ -24,11 +24,11 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
#include "common/endian.h"
|
#include "common/endian.h"
|
||||||
#include "common/stream.h"
|
#include "common/memstream.h"
|
||||||
|
|
||||||
#include "sound/audiostream.h"
|
#include "audio/audiostream.h"
|
||||||
#include "sound/mixer.h"
|
#include "audio/mixer.h"
|
||||||
#include "sound/decoders/raw.h"
|
#include "audio/decoders/raw.h"
|
||||||
|
|
||||||
namespace Audio {
|
namespace Audio {
|
||||||
|
|
@ -23,7 +23,12 @@
|
|||||||
*
|
*
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#include "sound/decoders/vorbis.h"
|
// Disable symbol overrides for FILE and fseek as those are used in the
|
||||||
|
// Vorbis headers.
|
||||||
|
#define FORBIDDEN_SYMBOL_EXCEPTION_FILE
|
||||||
|
#define FORBIDDEN_SYMBOL_EXCEPTION_fseek
|
||||||
|
|
||||||
|
#include "audio/decoders/vorbis.h"
|
||||||
|
|
||||||
#ifdef USE_VORBIS
|
#ifdef USE_VORBIS
|
||||||
|
|
||||||
@ -31,8 +36,7 @@
|
|||||||
#include "common/stream.h"
|
#include "common/stream.h"
|
||||||
#include "common/util.h"
|
#include "common/util.h"
|
||||||
|
|
||||||
#include "sound/audiostream.h"
|
#include "audio/audiostream.h"
|
||||||
#include "sound/audiocd.h"
|
|
||||||
|
|
||||||
#ifdef USE_TREMOR
|
#ifdef USE_TREMOR
|
||||||
#if defined(__GP32__) // custom libtremor locations
|
#if defined(__GP32__) // custom libtremor locations
|
@ -27,10 +27,12 @@
|
|||||||
* @file
|
* @file
|
||||||
* Sound decoder used in engines:
|
* Sound decoder used in engines:
|
||||||
* - agos
|
* - agos
|
||||||
|
* - draci
|
||||||
* - kyra
|
* - kyra
|
||||||
* - m4
|
* - m4
|
||||||
* - queen
|
* - queen
|
||||||
* - saga
|
* - saga
|
||||||
|
* - sci
|
||||||
* - scumm
|
* - scumm
|
||||||
* - sword1
|
* - sword1
|
||||||
* - sword2
|
* - sword2
|
@ -22,10 +22,10 @@
|
|||||||
* $Id$
|
* $Id$
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#include "sound/fmopl.h"
|
#include "audio/fmopl.h"
|
||||||
|
|
||||||
#include "sound/softsynth/opl/dosbox.h"
|
#include "audio/softsynth/opl/dosbox.h"
|
||||||
#include "sound/softsynth/opl/mame.h"
|
#include "audio/softsynth/opl/mame.h"
|
||||||
|
|
||||||
#include "common/config-manager.h"
|
#include "common/config-manager.h"
|
||||||
#include "common/translation.h"
|
#include "common/translation.h"
|
@ -28,8 +28,8 @@
|
|||||||
#include "common/str.h"
|
#include "common/str.h"
|
||||||
#include "common/system.h"
|
#include "common/system.h"
|
||||||
#include "common/util.h"
|
#include "common/util.h"
|
||||||
#include "sound/mididrv.h"
|
#include "audio/mididrv.h"
|
||||||
#include "sound/musicplugin.h"
|
#include "audio/musicplugin.h"
|
||||||
#include "common/translation.h"
|
#include "common/translation.h"
|
||||||
|
|
||||||
const byte MidiDriver::_mt32ToGm[128] = {
|
const byte MidiDriver::_mt32ToGm[128] = {
|
||||||
@ -162,7 +162,7 @@ MidiDriver::DeviceHandle MidiDriver::detectDevice(int flags) {
|
|||||||
case MT_AMIGA:
|
case MT_AMIGA:
|
||||||
if (flags & MDT_AMIGA)
|
if (flags & MDT_AMIGA)
|
||||||
return hdl;
|
return hdl;
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case MT_APPLEIIGS:
|
case MT_APPLEIIGS:
|
||||||
if (flags & MDT_APPLEIIGS)
|
if (flags & MDT_APPLEIIGS)
|
||||||
@ -209,35 +209,40 @@ MidiDriver::DeviceHandle MidiDriver::detectDevice(int flags) {
|
|||||||
hdl = getDeviceHandle("auto");
|
hdl = getDeviceHandle("auto");
|
||||||
|
|
||||||
const MusicType type = getMusicType(hdl);
|
const MusicType type = getMusicType(hdl);
|
||||||
if (type != MT_AUTO && type != MT_INVALID) {
|
|
||||||
if (flags & MDT_PREFER_MT32)
|
|
||||||
// If we have a preferred MT32 device we disable the gm/mt32 mapping (more about this in mididrv.h)
|
|
||||||
_forceTypeMT32 = true;
|
|
||||||
|
|
||||||
return hdl;
|
// If have a "Don't use GM/MT-32" setting we skip this part and jump
|
||||||
}
|
// to AdLib, PC Speaker etc. detection right away.
|
||||||
|
if (type != MT_NULL) {
|
||||||
|
if (type != MT_AUTO && type != MT_INVALID) {
|
||||||
|
if (flags & MDT_PREFER_MT32)
|
||||||
|
// If we have a preferred MT32 device we disable the gm/mt32 mapping (more about this in mididrv.h)
|
||||||
|
_forceTypeMT32 = true;
|
||||||
|
|
||||||
// If we have no specific device selected (neither in the scummvm nor in the game domain)
|
return hdl;
|
||||||
// and no preferred MT32 or GM device selected we arrive here.
|
}
|
||||||
// If MT32 is preferred we try for the first available device with music type 'MT_MT32' (usually the mt32 emulator)
|
|
||||||
if (flags & MDT_PREFER_MT32) {
|
// If we have no specific device selected (neither in the scummvm nor in the game domain)
|
||||||
|
// and no preferred MT32 or GM device selected we arrive here.
|
||||||
|
// If MT32 is preferred we try for the first available device with music type 'MT_MT32' (usually the mt32 emulator)
|
||||||
|
if (flags & MDT_PREFER_MT32) {
|
||||||
|
for (MusicPlugin::List::const_iterator m = p.begin(); m != p.end(); ++m) {
|
||||||
|
MusicDevices i = (**m)->getDevices();
|
||||||
|
for (MusicDevices::iterator d = i.begin(); d != i.end(); ++d) {
|
||||||
|
if (d->getMusicType() == MT_MT32)
|
||||||
|
return d->getHandle();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Now we default to the first available device with music type 'MT_GM'
|
||||||
for (MusicPlugin::List::const_iterator m = p.begin(); m != p.end(); ++m) {
|
for (MusicPlugin::List::const_iterator m = p.begin(); m != p.end(); ++m) {
|
||||||
MusicDevices i = (**m)->getDevices();
|
MusicDevices i = (**m)->getDevices();
|
||||||
for (MusicDevices::iterator d = i.begin(); d != i.end(); ++d) {
|
for (MusicDevices::iterator d = i.begin(); d != i.end(); ++d) {
|
||||||
if (d->getMusicType() == MT_MT32)
|
if (d->getMusicType() == MT_GM || d->getMusicType() == MT_GS)
|
||||||
return d->getHandle();
|
return d->getHandle();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Now we default to the first available device with music type 'MT_GM'
|
|
||||||
for (MusicPlugin::List::const_iterator m = p.begin(); m != p.end(); ++m) {
|
|
||||||
MusicDevices i = (**m)->getDevices();
|
|
||||||
for (MusicDevices::iterator d = i.begin(); d != i.end(); ++d) {
|
|
||||||
if (d->getMusicType() == MT_GM || d->getMusicType() == MT_GS)
|
|
||||||
return d->getHandle();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
MusicType tp = MT_AUTO;
|
MusicType tp = MT_AUTO;
|
||||||
@ -247,10 +252,10 @@ MidiDriver::DeviceHandle MidiDriver::detectDevice(int flags) {
|
|||||||
tp = MT_PC98;
|
tp = MT_PC98;
|
||||||
else if (flags & MDT_ADLIB)
|
else if (flags & MDT_ADLIB)
|
||||||
tp = MT_ADLIB;
|
tp = MT_ADLIB;
|
||||||
else if (flags & MDT_PCSPK)
|
|
||||||
tp = MT_PCSPK;
|
|
||||||
else if (flags & MDT_PCJR)
|
else if (flags & MDT_PCJR)
|
||||||
tp = MT_PCJR;
|
tp = MT_PCJR;
|
||||||
|
else if (flags & MDT_PCSPK)
|
||||||
|
tp = MT_PCSPK;
|
||||||
else if (flags & MDT_C64)
|
else if (flags & MDT_C64)
|
||||||
tp = MT_C64;
|
tp = MT_C64;
|
||||||
else if (flags & MDT_AMIGA)
|
else if (flags & MDT_AMIGA)
|
@ -37,15 +37,6 @@ namespace Audio {
|
|||||||
}
|
}
|
||||||
namespace Common { class String; }
|
namespace Common { class String; }
|
||||||
|
|
||||||
/**
|
|
||||||
* Music Driver Types, used to uniquely identify each music driver.
|
|
||||||
*
|
|
||||||
* The pseudo drivers are listed first, then all native drivers,
|
|
||||||
* then all other MIDI drivers, and finally the non-MIDI drivers.
|
|
||||||
*
|
|
||||||
* @todo Rename MidiDriverType to MusicDriverType
|
|
||||||
*/
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Music types that music drivers can implement and engines can rely on.
|
* Music types that music drivers can implement and engines can rely on.
|
||||||
*/
|
*/
|
||||||
@ -71,16 +62,21 @@ enum MusicType {
|
|||||||
* A set of flags to be passed to detectDevice() which can be used to
|
* A set of flags to be passed to detectDevice() which can be used to
|
||||||
* specify what kind of music driver is preferred / accepted.
|
* specify what kind of music driver is preferred / accepted.
|
||||||
*
|
*
|
||||||
* The flags (except for MDT_PREFER_MT32 and MDT_PREFER_GM) indicate whether a given driver
|
* The flags (except for MDT_PREFER_MT32 and MDT_PREFER_GM) indicate whether a
|
||||||
* type is acceptable. E.g. the TOWNS music driver could be returned by
|
* given driver type is acceptable. E.g. the TOWNS music driver could be
|
||||||
* detectDevice if and only if MDT_TOWNS is specified.
|
* returned by detectDevice if and only if MDT_TOWNS is specified.
|
||||||
|
*
|
||||||
|
* MDT_PREFER_MT32 and MDT_PREFER_GM indicate the MIDI device type to use when
|
||||||
|
* no device is selected in the music options, or when the MIDI device selected
|
||||||
|
* does not match the requirements of a game engine. With these flags, more
|
||||||
|
* priority is given to an MT-32 device, or a GM device respectively.
|
||||||
*
|
*
|
||||||
* @todo Rename MidiDriverFlags to MusicDriverFlags
|
* @todo Rename MidiDriverFlags to MusicDriverFlags
|
||||||
*/
|
*/
|
||||||
enum MidiDriverFlags {
|
enum MidiDriverFlags {
|
||||||
MDT_NONE = 0,
|
MDT_NONE = 0,
|
||||||
MDT_PCSPK = 1 << 0, // PC Speaker: Maps to MD_PCSPK and MD_PCJR
|
MDT_PCSPK = 1 << 0, // PC Speaker: Maps to MT_PCSPK and MT_PCJR
|
||||||
MDT_CMS = 1 << 1, // Creative Music System / Gameblaster: Maps to MD_CMS
|
MDT_CMS = 1 << 1, // Creative Music System / Gameblaster: Maps to MT_CMS
|
||||||
MDT_PCJR = 1 << 2, // Tandy/PC Junior driver
|
MDT_PCJR = 1 << 2, // Tandy/PC Junior driver
|
||||||
MDT_ADLIB = 1 << 3, // AdLib: Maps to MT_ADLIB
|
MDT_ADLIB = 1 << 3, // AdLib: Maps to MT_ADLIB
|
||||||
MDT_C64 = 1 << 4,
|
MDT_C64 = 1 << 4,
|
||||||
@ -94,19 +90,53 @@ enum MidiDriverFlags {
|
|||||||
};
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Abstract description of a MIDI driver. Used by the config file and command
|
* TODO: Document this, give it a better name.
|
||||||
* line parsing code, and also to be able to give the user a list of available
|
|
||||||
* drivers.
|
|
||||||
*
|
|
||||||
* @todo Rename MidiDriverType to MusicDriverType
|
|
||||||
*/
|
*/
|
||||||
|
class MidiDriver_BASE {
|
||||||
|
public:
|
||||||
|
virtual ~MidiDriver_BASE() { }
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Output a packed midi command to the midi stream.
|
||||||
|
* The 'lowest' byte (i.e. b & 0xFF) is the status
|
||||||
|
* code, then come (if used) the first and second
|
||||||
|
* opcode.
|
||||||
|
*/
|
||||||
|
virtual void send(uint32 b) = 0;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Output a midi command to the midi stream. Convenience wrapper
|
||||||
|
* around the usual 'packed' send method.
|
||||||
|
*
|
||||||
|
* Do NOT use this for sysEx transmission; instead, use the sysEx()
|
||||||
|
* method below.
|
||||||
|
*/
|
||||||
|
void send(byte status, byte firstOp, byte secondOp) {
|
||||||
|
send(status | ((uint32)firstOp << 8) | ((uint32)secondOp << 16));
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Transmit a sysEx to the midi device.
|
||||||
|
*
|
||||||
|
* The given msg MUST NOT contain the usual SysEx frame, i.e.
|
||||||
|
* do NOT include the leading 0xF0 and the trailing 0xF7.
|
||||||
|
*
|
||||||
|
* Furthermore, the maximal supported length of a SysEx
|
||||||
|
* is 264 bytes. Passing longer buffers can lead to
|
||||||
|
* undefined behavior (most likely, a crash).
|
||||||
|
*/
|
||||||
|
virtual void sysEx(const byte *msg, uint16 length) { }
|
||||||
|
|
||||||
|
// TODO: Document this.
|
||||||
|
virtual void metaEvent(byte type, byte *data, uint16 length) { }
|
||||||
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Abstract MIDI Driver Class
|
* Abstract MIDI Driver Class
|
||||||
*
|
*
|
||||||
* @todo Rename MidiDriver to MusicDriver
|
* @todo Rename MidiDriver to MusicDriver
|
||||||
*/
|
*/
|
||||||
class MidiDriver {
|
class MidiDriver : public MidiDriver_BASE {
|
||||||
public:
|
public:
|
||||||
/**
|
/**
|
||||||
* The device handle.
|
* The device handle.
|
||||||
@ -130,7 +160,7 @@ public:
|
|||||||
|
|
||||||
/** Returns device handle based on the present devices and the flags parameter. */
|
/** Returns device handle based on the present devices and the flags parameter. */
|
||||||
static DeviceHandle detectDevice(int flags);
|
static DeviceHandle detectDevice(int flags);
|
||||||
|
|
||||||
/** Find the music driver matching the given driver name/description. */
|
/** Find the music driver matching the given driver name/description. */
|
||||||
static DeviceHandle getDeviceHandle(const Common::String &identifier);
|
static DeviceHandle getDeviceHandle(const Common::String &identifier);
|
||||||
|
|
||||||
@ -177,28 +207,14 @@ public:
|
|||||||
*/
|
*/
|
||||||
virtual int open() = 0;
|
virtual int open() = 0;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Check whether the midi driver has already been opened.
|
||||||
|
*/
|
||||||
|
virtual bool isOpen() const = 0;
|
||||||
|
|
||||||
/** Close the midi driver. */
|
/** Close the midi driver. */
|
||||||
virtual void close() = 0;
|
virtual void close() = 0;
|
||||||
|
|
||||||
/**
|
|
||||||
* Output a packed midi command to the midi stream.
|
|
||||||
* The 'lowest' byte (i.e. b & 0xFF) is the status
|
|
||||||
* code, then come (if used) the first and second
|
|
||||||
* opcode.
|
|
||||||
*/
|
|
||||||
virtual void send(uint32 b) = 0;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Output a midi command to the midi stream. Convenience wrapper
|
|
||||||
* around the usual 'packed' send method.
|
|
||||||
*
|
|
||||||
* Do NOT use this for sysEx transmission; instead, use the sysEx()
|
|
||||||
* method below.
|
|
||||||
*/
|
|
||||||
void send(byte status, byte firstOp, byte secondOp) {
|
|
||||||
send(status | ((uint32)firstOp << 8) | ((uint32)secondOp << 16));
|
|
||||||
}
|
|
||||||
|
|
||||||
/** Get or set a property. */
|
/** Get or set a property. */
|
||||||
virtual uint32 property(int prop, uint32 param) { return 0; }
|
virtual uint32 property(int prop, uint32 param) { return 0; }
|
||||||
|
|
||||||
@ -225,22 +241,8 @@ public:
|
|||||||
*/
|
*/
|
||||||
void sendGMReset();
|
void sendGMReset();
|
||||||
|
|
||||||
/**
|
|
||||||
* Transmit a sysEx to the midi device.
|
|
||||||
*
|
|
||||||
* The given msg MUST NOT contain the usual SysEx frame, i.e.
|
|
||||||
* do NOT include the leading 0xF0 and the trailing 0xF7.
|
|
||||||
*
|
|
||||||
* Furthermore, the maximal supported length of a SysEx
|
|
||||||
* is 264 bytes. Passing longer buffers can lead to
|
|
||||||
* undefined behavior (most likely, a crash).
|
|
||||||
*/
|
|
||||||
virtual void sysEx(const byte *msg, uint16 length) { }
|
|
||||||
|
|
||||||
virtual void sysEx_customInstrument(byte channel, uint32 type, const byte *instr) { }
|
virtual void sysEx_customInstrument(byte channel, uint32 type, const byte *instr) { }
|
||||||
|
|
||||||
virtual void metaEvent(byte type, byte *data, uint16 length) { }
|
|
||||||
|
|
||||||
// Timing functions - MidiDriver now operates timers
|
// Timing functions - MidiDriver now operates timers
|
||||||
virtual void setTimerCallback(void *timer_param, Common::TimerManager::TimerProc timer_proc) = 0;
|
virtual void setTimerCallback(void *timer_param, Common::TimerManager::TimerProc timer_proc) = 0;
|
||||||
|
|
@ -23,8 +23,8 @@
|
|||||||
*
|
*
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#include "sound/midiparser.h"
|
#include "audio/midiparser.h"
|
||||||
#include "sound/mididrv.h"
|
#include "audio/mididrv.h"
|
||||||
#include "common/util.h"
|
#include "common/util.h"
|
||||||
|
|
||||||
//////////////////////////////////////////////////
|
//////////////////////////////////////////////////
|
||||||
@ -43,6 +43,7 @@ _psec_per_tick(5208), // 500000 / 96
|
|||||||
_autoLoop(false),
|
_autoLoop(false),
|
||||||
_smartJump(false),
|
_smartJump(false),
|
||||||
_centerPitchWheelOnUnload(false),
|
_centerPitchWheelOnUnload(false),
|
||||||
|
_sendSustainOffOnNotesOff(false),
|
||||||
_num_tracks(0),
|
_num_tracks(0),
|
||||||
_active_track(255),
|
_active_track(255),
|
||||||
_abort_parse(0) {
|
_abort_parse(0) {
|
||||||
@ -64,6 +65,9 @@ void MidiParser::property(int prop, int value) {
|
|||||||
case mpCenterPitchWheelOnUnload:
|
case mpCenterPitchWheelOnUnload:
|
||||||
_centerPitchWheelOnUnload = (value != 0);
|
_centerPitchWheelOnUnload = (value != 0);
|
||||||
break;
|
break;
|
||||||
|
case mpSendSustainOffOnNotesOff:
|
||||||
|
_sendSustainOffOnNotesOff = (value != 0);
|
||||||
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -281,6 +285,8 @@ void MidiParser::allNotesOff() {
|
|||||||
|
|
||||||
for (i = 0; i < 16; ++i) {
|
for (i = 0; i < 16; ++i) {
|
||||||
sendToDriver(0xB0 | i, 0x7b, 0); // All notes off
|
sendToDriver(0xB0 | i, 0x7b, 0); // All notes off
|
||||||
|
if (_sendSustainOffOnNotesOff)
|
||||||
|
sendToDriver(0xB0 | i, 0x40, 0); // Also send a sustain off event (bug #3116608)
|
||||||
}
|
}
|
||||||
|
|
||||||
memset(_active_notes, 0, sizeof(_active_notes));
|
memset(_active_notes, 0, sizeof(_active_notes));
|
||||||
@ -344,7 +350,7 @@ void MidiParser::hangAllActiveNotes() {
|
|||||||
if (_next_event.command() == 0x8) {
|
if (_next_event.command() == 0x8) {
|
||||||
if (temp_active[_next_event.basic.param1] & (1 << _next_event.channel())) {
|
if (temp_active[_next_event.basic.param1] & (1 << _next_event.channel())) {
|
||||||
hangingNote(_next_event.channel(), _next_event.basic.param1, (advance_tick - _position._last_event_tick) * _psec_per_tick, false);
|
hangingNote(_next_event.channel(), _next_event.basic.param1, (advance_tick - _position._last_event_tick) * _psec_per_tick, false);
|
||||||
temp_active[_next_event.basic.param1] &= ~ (1 << _next_event.channel());
|
temp_active[_next_event.basic.param1] &= ~(1 << _next_event.channel());
|
||||||
}
|
}
|
||||||
} else if (_next_event.event == 0xFF && _next_event.ext.type == 0x2F) {
|
} else if (_next_event.event == 0xFF && _next_event.ext.type == 0x2F) {
|
||||||
// warning("MidiParser::hangAllActiveNotes(): Hit End of Track with active notes left");
|
// warning("MidiParser::hangAllActiveNotes(): Hit End of Track with active notes left");
|
||||||
@ -361,7 +367,7 @@ void MidiParser::hangAllActiveNotes() {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
bool MidiParser::jumpToTick(uint32 tick, bool fireEvents, bool stopNotes) {
|
bool MidiParser::jumpToTick(uint32 tick, bool fireEvents, bool stopNotes, bool dontSendNoteOn) {
|
||||||
if (_active_track >= _num_tracks)
|
if (_active_track >= _num_tracks)
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
@ -402,8 +408,17 @@ bool MidiParser::jumpToTick(uint32 tick, bool fireEvents, bool stopNotes) {
|
|||||||
_driver->sysEx(info.ext.data, (uint16)info.length-1);
|
_driver->sysEx(info.ext.data, (uint16)info.length-1);
|
||||||
else
|
else
|
||||||
_driver->sysEx(info.ext.data, (uint16)info.length);
|
_driver->sysEx(info.ext.data, (uint16)info.length);
|
||||||
} else
|
} else {
|
||||||
sendToDriver(info.event, info.basic.param1, info.basic.param2);
|
// The note on sending code is used by the SCUMM engine. Other engine using this code
|
||||||
|
// (such as SCI) have issues with this, as all the notes sent can be heard when a song
|
||||||
|
// is fast-forwarded. Thus, if the engine requests it, don't send note on events.
|
||||||
|
if (info.command() == 0x9 && dontSendNoteOn) {
|
||||||
|
// Don't send note on; doing so creates a "warble" with some instruments on the MT-32.
|
||||||
|
// Refer to patch #3117577
|
||||||
|
} else {
|
||||||
|
sendToDriver(info.event, info.basic.param1, info.basic.param2);
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
parseNextEvent(_next_event);
|
parseNextEvent(_next_event);
|
@ -32,7 +32,7 @@
|
|||||||
#include "common/endian.h"
|
#include "common/endian.h"
|
||||||
|
|
||||||
class MidiParser;
|
class MidiParser;
|
||||||
class MidiDriver;
|
class MidiDriver_BASE;
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
@ -273,7 +273,7 @@ protected:
|
|||||||
///< Used for "Smart Jump" and MIDI formats that do not include explicit Note Off events.
|
///< Used for "Smart Jump" and MIDI formats that do not include explicit Note Off events.
|
||||||
byte _hanging_notes_count; ///< Count of hanging notes, used to optimize expiration.
|
byte _hanging_notes_count; ///< Count of hanging notes, used to optimize expiration.
|
||||||
|
|
||||||
MidiDriver *_driver; ///< The device to which all events will be transmitted.
|
MidiDriver_BASE *_driver; ///< The device to which all events will be transmitted.
|
||||||
uint32 _timer_rate; ///< The time in microseconds between onTimer() calls. Obtained from the MidiDriver.
|
uint32 _timer_rate; ///< The time in microseconds between onTimer() calls. Obtained from the MidiDriver.
|
||||||
uint32 _ppqn; ///< Pulses Per Quarter Note. (We refer to "pulses" as "ticks".)
|
uint32 _ppqn; ///< Pulses Per Quarter Note. (We refer to "pulses" as "ticks".)
|
||||||
uint32 _tempo; ///< Microseconds per quarter note.
|
uint32 _tempo; ///< Microseconds per quarter note.
|
||||||
@ -281,7 +281,7 @@ protected:
|
|||||||
bool _autoLoop; ///< For lightweight clients that don't provide their own flow control.
|
bool _autoLoop; ///< For lightweight clients that don't provide their own flow control.
|
||||||
bool _smartJump; ///< Support smart expiration of hanging notes when jumping
|
bool _smartJump; ///< Support smart expiration of hanging notes when jumping
|
||||||
bool _centerPitchWheelOnUnload; ///< Center the pitch wheels when unloading a song
|
bool _centerPitchWheelOnUnload; ///< Center the pitch wheels when unloading a song
|
||||||
|
bool _sendSustainOffOnNotesOff; ///< Send a sustain off on a notes off event, stopping hanging notes
|
||||||
byte *_tracks[120]; ///< Multi-track MIDI formats are supported, up to 120 tracks.
|
byte *_tracks[120]; ///< Multi-track MIDI formats are supported, up to 120 tracks.
|
||||||
byte _num_tracks; ///< Count of total tracks for multi-track MIDI formats. 1 for single-track formats.
|
byte _num_tracks; ///< Count of total tracks for multi-track MIDI formats. 1 for single-track formats.
|
||||||
byte _active_track; ///< Keeps track of the currently active track, in multi-track formats.
|
byte _active_track; ///< Keeps track of the currently active track, in multi-track formats.
|
||||||
@ -361,7 +361,13 @@ public:
|
|||||||
* Center the pitch wheels when unloading music in preparation
|
* Center the pitch wheels when unloading music in preparation
|
||||||
* for the next piece of music.
|
* for the next piece of music.
|
||||||
*/
|
*/
|
||||||
mpCenterPitchWheelOnUnload = 4
|
mpCenterPitchWheelOnUnload = 4,
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Sends a sustain off event when a notes off event is triggered.
|
||||||
|
* Stops hanging notes.
|
||||||
|
*/
|
||||||
|
mpSendSustainOffOnNotesOff = 5
|
||||||
};
|
};
|
||||||
|
|
||||||
public:
|
public:
|
||||||
@ -374,7 +380,7 @@ public:
|
|||||||
virtual void unloadMusic();
|
virtual void unloadMusic();
|
||||||
virtual void property(int prop, int value);
|
virtual void property(int prop, int value);
|
||||||
|
|
||||||
void setMidiDriver(MidiDriver *driver) { _driver = driver; }
|
void setMidiDriver(MidiDriver_BASE *driver) { _driver = driver; }
|
||||||
void setTimerRate(uint32 rate) { _timer_rate = rate; }
|
void setTimerRate(uint32 rate) { _timer_rate = rate; }
|
||||||
void setTempo(uint32 tempo);
|
void setTempo(uint32 tempo);
|
||||||
void onTimer();
|
void onTimer();
|
||||||
@ -383,7 +389,7 @@ public:
|
|||||||
void stopPlaying();
|
void stopPlaying();
|
||||||
|
|
||||||
bool setTrack(int track);
|
bool setTrack(int track);
|
||||||
bool jumpToTick(uint32 tick, bool fireEvents = false, bool stopNotes = true);
|
bool jumpToTick(uint32 tick, bool fireEvents = false, bool stopNotes = true, bool dontSendNoteOn = false);
|
||||||
|
|
||||||
uint32 getPPQN() { return _ppqn; }
|
uint32 getPPQN() { return _ppqn; }
|
||||||
virtual uint32 getTick() { return _position._play_tick; }
|
virtual uint32 getTick() { return _position._play_tick; }
|
@ -26,10 +26,10 @@
|
|||||||
#include "common/util.h"
|
#include "common/util.h"
|
||||||
#include "common/system.h"
|
#include "common/system.h"
|
||||||
|
|
||||||
#include "sound/mixer_intern.h"
|
#include "audio/mixer_intern.h"
|
||||||
#include "sound/rate.h"
|
#include "audio/rate.h"
|
||||||
#include "sound/audiostream.h"
|
#include "audio/audiostream.h"
|
||||||
#include "sound/timestamp.h"
|
#include "audio/timestamp.h"
|
||||||
|
|
||||||
|
|
||||||
namespace Audio {
|
namespace Audio {
|
||||||
@ -54,8 +54,9 @@ public:
|
|||||||
* @param len number of sample *pairs*. So a value of
|
* @param len number of sample *pairs*. So a value of
|
||||||
* 10 means that the buffer contains twice 10 sample, each
|
* 10 means that the buffer contains twice 10 sample, each
|
||||||
* 16 bits, for a total of 40 bytes.
|
* 16 bits, for a total of 40 bytes.
|
||||||
|
* @return number of sample pairs processed (which can still be silence!)
|
||||||
*/
|
*/
|
||||||
void mix(int16 *data, uint len);
|
int mix(int16 *data, uint len);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Queries whether the channel is still playing or not.
|
* Queries whether the channel is still playing or not.
|
||||||
@ -257,7 +258,7 @@ void MixerImpl::playStream(
|
|||||||
insertChannel(handle, chan);
|
insertChannel(handle, chan);
|
||||||
}
|
}
|
||||||
|
|
||||||
void MixerImpl::mixCallback(byte *samples, uint len) {
|
int MixerImpl::mixCallback(byte *samples, uint len) {
|
||||||
assert(samples);
|
assert(samples);
|
||||||
|
|
||||||
Common::StackLock lock(_mutex);
|
Common::StackLock lock(_mutex);
|
||||||
@ -272,14 +273,21 @@ void MixerImpl::mixCallback(byte *samples, uint len) {
|
|||||||
memset(buf, 0, 2 * len * sizeof(int16));
|
memset(buf, 0, 2 * len * sizeof(int16));
|
||||||
|
|
||||||
// mix all channels
|
// mix all channels
|
||||||
|
int res = 0, tmp;
|
||||||
for (int i = 0; i != NUM_CHANNELS; i++)
|
for (int i = 0; i != NUM_CHANNELS; i++)
|
||||||
if (_channels[i]) {
|
if (_channels[i]) {
|
||||||
if (_channels[i]->isFinished()) {
|
if (_channels[i]->isFinished()) {
|
||||||
delete _channels[i];
|
delete _channels[i];
|
||||||
_channels[i] = 0;
|
_channels[i] = 0;
|
||||||
} else if (!_channels[i]->isPaused())
|
} else if (!_channels[i]->isPaused()) {
|
||||||
_channels[i]->mix(buf, len);
|
tmp = _channels[i]->mix(buf, len);
|
||||||
|
|
||||||
|
if (tmp > res)
|
||||||
|
res = tmp;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
return res;
|
||||||
}
|
}
|
||||||
|
|
||||||
void MixerImpl::stopAll() {
|
void MixerImpl::stopAll() {
|
||||||
@ -538,19 +546,23 @@ Timestamp Channel::getElapsedTime() {
|
|||||||
return ts;
|
return ts;
|
||||||
}
|
}
|
||||||
|
|
||||||
void Channel::mix(int16 *data, uint len) {
|
int Channel::mix(int16 *data, uint len) {
|
||||||
assert(_stream);
|
assert(_stream);
|
||||||
|
|
||||||
|
int res = 0;
|
||||||
|
|
||||||
if (_stream->endOfData()) {
|
if (_stream->endOfData()) {
|
||||||
// TODO: call drain method
|
// TODO: call drain method
|
||||||
} else {
|
} else {
|
||||||
assert(_converter);
|
assert(_converter);
|
||||||
|
|
||||||
_samplesConsumed = _samplesDecoded;
|
_samplesConsumed = _samplesDecoded;
|
||||||
_mixerTimeStamp = g_system->getMillis();
|
_mixerTimeStamp = g_system->getMillis();
|
||||||
_pauseTime = 0;
|
_pauseTime = 0;
|
||||||
_samplesDecoded += _converter->flow(*_stream, data, len, _volL, _volR);
|
res = _converter->flow(*_stream, data, len, _volL, _volR);
|
||||||
|
_samplesDecoded += res;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
return res;
|
||||||
}
|
}
|
||||||
|
|
||||||
} // End of namespace Audio
|
} // End of namespace Audio
|
@ -28,8 +28,9 @@
|
|||||||
|
|
||||||
#include "common/types.h"
|
#include "common/types.h"
|
||||||
#include "common/mutex.h"
|
#include "common/mutex.h"
|
||||||
|
#include "common/noncopyable.h"
|
||||||
|
|
||||||
#include "sound/timestamp.h"
|
#include "audio/timestamp.h"
|
||||||
|
|
||||||
class OSystem;
|
class OSystem;
|
||||||
|
|
||||||
@ -37,8 +38,6 @@ class OSystem;
|
|||||||
namespace Audio {
|
namespace Audio {
|
||||||
|
|
||||||
class AudioStream;
|
class AudioStream;
|
||||||
class RewindableAudioStream;
|
|
||||||
class SeekableAudioStream;
|
|
||||||
class Channel;
|
class Channel;
|
||||||
class Mixer;
|
class Mixer;
|
||||||
class MixerImpl;
|
class MixerImpl;
|
||||||
@ -61,7 +60,7 @@ public:
|
|||||||
* The main audio mixer handles mixing of an arbitrary number of
|
* The main audio mixer handles mixing of an arbitrary number of
|
||||||
* audio streams (in the form of AudioStream instances).
|
* audio streams (in the form of AudioStream instances).
|
||||||
*/
|
*/
|
||||||
class Mixer {
|
class Mixer : Common::NonCopyable {
|
||||||
public:
|
public:
|
||||||
enum SoundType {
|
enum SoundType {
|
||||||
kPlainSoundType = 0,
|
kPlainSoundType = 0,
|
@ -28,7 +28,7 @@
|
|||||||
|
|
||||||
#include "common/sys.h"
|
#include "common/sys.h"
|
||||||
#include "common/mutex.h"
|
#include "common/mutex.h"
|
||||||
#include "sound/mixer.h"
|
#include "audio/mixer.h"
|
||||||
|
|
||||||
namespace Audio {
|
namespace Audio {
|
||||||
|
|
||||||
@ -118,8 +118,10 @@ public:
|
|||||||
* The mixer callback function, to be called at regular intervals by
|
* The mixer callback function, to be called at regular intervals by
|
||||||
* the backend (e.g. from an audio mixing thread). All the actual mixing
|
* the backend (e.g. from an audio mixing thread). All the actual mixing
|
||||||
* work is done from here.
|
* work is done from here.
|
||||||
|
*
|
||||||
|
* @return number of sample pairs processed (which can still be silence!)
|
||||||
*/
|
*/
|
||||||
void mixCallback(byte *samples, uint len);
|
int mixCallback(byte *samples, uint len);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Set the internal 'is ready' flag of the mixer.
|
* Set the internal 'is ready' flag of the mixer.
|
@ -1,7 +1,6 @@
|
|||||||
MODULE := sound
|
MODULE := audio
|
||||||
|
|
||||||
MODULE_OBJS := \
|
MODULE_OBJS := \
|
||||||
audiocd.o \
|
|
||||||
audiostream.o \
|
audiostream.o \
|
||||||
fmopl.o \
|
fmopl.o \
|
||||||
mididrv.o \
|
mididrv.o \
|
@ -22,7 +22,7 @@
|
|||||||
* $Id$
|
* $Id$
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#include "sound/mpu401.h"
|
#include "audio/mpu401.h"
|
||||||
#include "common/system.h"
|
#include "common/system.h"
|
||||||
#include "common/timer.h"
|
#include "common/timer.h"
|
||||||
#include "common/util.h" // for ARRAYSIZE
|
#include "common/util.h" // for ARRAYSIZE
|
||||||
@ -101,12 +101,18 @@ MidiDriver_MPU401::MidiDriver_MPU401() :
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
MidiDriver_MPU401::~MidiDriver_MPU401() {
|
||||||
|
}
|
||||||
|
|
||||||
void MidiDriver_MPU401::close() {
|
void MidiDriver_MPU401::close() {
|
||||||
if (_timer_proc)
|
if (_timer_proc) {
|
||||||
g_system->getTimerManager()->removeTimerProc(_timer_proc);
|
g_system->getTimerManager()->removeTimerProc(_timer_proc);
|
||||||
_timer_proc = 0;
|
_timer_proc = 0;
|
||||||
for (int i = 0; i < 16; ++i)
|
}
|
||||||
send(0x7B << 8 | 0xB0 | i);
|
if (isOpen()) {
|
||||||
|
for (int i = 0; i < 16; ++i)
|
||||||
|
send(0x7B << 8 | 0xB0 | i);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
uint32 MidiDriver_MPU401::property(int prop, uint32 param) {
|
uint32 MidiDriver_MPU401::property(int prop, uint32 param) {
|
@ -26,7 +26,7 @@
|
|||||||
#ifndef SOUND_MPU401_H
|
#ifndef SOUND_MPU401_H
|
||||||
#define SOUND_MPU401_H
|
#define SOUND_MPU401_H
|
||||||
|
|
||||||
#include "sound/mididrv.h"
|
#include "audio/mididrv.h"
|
||||||
|
|
||||||
////////////////////////////////////////
|
////////////////////////////////////////
|
||||||
//
|
//
|
||||||
@ -46,22 +46,22 @@ private:
|
|||||||
public:
|
public:
|
||||||
MidiDriver *device();
|
MidiDriver *device();
|
||||||
byte getNumber() { return _channel; }
|
byte getNumber() { return _channel; }
|
||||||
void release() { _allocated = false; }
|
virtual void release() { _allocated = false; }
|
||||||
|
|
||||||
void send(uint32 b);
|
virtual void send(uint32 b);
|
||||||
|
|
||||||
// Regular messages
|
// Regular messages
|
||||||
void noteOff(byte note);
|
virtual void noteOff(byte note);
|
||||||
void noteOn(byte note, byte velocity);
|
virtual void noteOn(byte note, byte velocity);
|
||||||
void programChange(byte program);
|
virtual void programChange(byte program);
|
||||||
void pitchBend(int16 bend);
|
virtual void pitchBend(int16 bend);
|
||||||
|
|
||||||
// Control Change messages
|
// Control Change messages
|
||||||
void controlChange(byte control, byte value);
|
virtual void controlChange(byte control, byte value);
|
||||||
void pitchBendFactor(byte value);
|
virtual void pitchBendFactor(byte value);
|
||||||
|
|
||||||
// SysEx messages
|
// SysEx messages
|
||||||
void sysEx_customInstrument(uint32 type, const byte *instr);
|
virtual void sysEx_customInstrument(uint32 type, const byte *instr);
|
||||||
|
|
||||||
// Only to be called by the owner
|
// Only to be called by the owner
|
||||||
void init(MidiDriver *owner, byte channel);
|
void init(MidiDriver *owner, byte channel);
|
||||||
@ -78,14 +78,15 @@ private:
|
|||||||
|
|
||||||
public:
|
public:
|
||||||
MidiDriver_MPU401();
|
MidiDriver_MPU401();
|
||||||
|
virtual ~MidiDriver_MPU401();
|
||||||
|
|
||||||
virtual void close();
|
virtual void close();
|
||||||
void setTimerCallback(void *timer_param, Common::TimerManager::TimerProc timer_proc);
|
virtual void setTimerCallback(void *timer_param, Common::TimerManager::TimerProc timer_proc);
|
||||||
uint32 getBaseTempo(void) { return 10000; }
|
virtual uint32 getBaseTempo(void) { return 10000; }
|
||||||
uint32 property(int prop, uint32 param);
|
virtual uint32 property(int prop, uint32 param);
|
||||||
|
|
||||||
MidiChannel *allocateChannel();
|
virtual MidiChannel *allocateChannel();
|
||||||
MidiChannel *getPercussionChannel() { return &_midi_channels[9]; }
|
virtual MidiChannel *getPercussionChannel() { return &_midi_channels[9]; }
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
@ -23,7 +23,7 @@
|
|||||||
*
|
*
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#include "sound/musicplugin.h"
|
#include "audio/musicplugin.h"
|
||||||
#include "common/hash-str.h"
|
#include "common/hash-str.h"
|
||||||
#include "common/translation.h"
|
#include "common/translation.h"
|
||||||
|
|
@ -26,7 +26,7 @@
|
|||||||
#define SOUND_MUSICPLUGIN_H
|
#define SOUND_MUSICPLUGIN_H
|
||||||
|
|
||||||
#include "base/plugins.h"
|
#include "base/plugins.h"
|
||||||
#include "sound/mididrv.h"
|
#include "audio/mididrv.h"
|
||||||
#include "common/list.h"
|
#include "common/list.h"
|
||||||
|
|
||||||
class MusicPluginObject;
|
class MusicPluginObject;
|
@ -22,7 +22,7 @@
|
|||||||
* $Id$
|
* $Id$
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#include "sound/null.h"
|
#include "audio/null.h"
|
||||||
|
|
||||||
Common::Error NullMusicPlugin::createInstance(MidiDriver **mididriver, MidiDriver::DeviceHandle) const {
|
Common::Error NullMusicPlugin::createInstance(MidiDriver **mididriver, MidiDriver::DeviceHandle) const {
|
||||||
*mididriver = new MidiDriver_NULL();
|
*mididriver = new MidiDriver_NULL();
|
@ -25,14 +25,15 @@
|
|||||||
#ifndef SOUND_NULL_H
|
#ifndef SOUND_NULL_H
|
||||||
#define SOUND_NULL_H
|
#define SOUND_NULL_H
|
||||||
|
|
||||||
#include "sound/musicplugin.h"
|
#include "audio/musicplugin.h"
|
||||||
#include "sound/mpu401.h"
|
#include "audio/mpu401.h"
|
||||||
#include "common/translation.h"
|
#include "common/translation.h"
|
||||||
|
|
||||||
/* NULL driver */
|
/* NULL driver */
|
||||||
class MidiDriver_NULL : public MidiDriver_MPU401 {
|
class MidiDriver_NULL : public MidiDriver_MPU401 {
|
||||||
public:
|
public:
|
||||||
int open() { return 0; }
|
int open() { return 0; }
|
||||||
|
bool isOpen() const { return true; }
|
||||||
void send(uint32 b) { }
|
void send(uint32 b) { }
|
||||||
};
|
};
|
||||||
|
|
@ -31,9 +31,9 @@
|
|||||||
* improvements over the original code were made.
|
* improvements over the original code were made.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#include "sound/audiostream.h"
|
#include "audio/audiostream.h"
|
||||||
#include "sound/rate.h"
|
#include "audio/rate.h"
|
||||||
#include "sound/mixer.h"
|
#include "audio/mixer.h"
|
||||||
#include "common/frac.h"
|
#include "common/frac.h"
|
||||||
#include "common/util.h"
|
#include "common/util.h"
|
||||||
|
|
@ -44,9 +44,9 @@
|
|||||||
* other improvments over the original code were made.
|
* other improvments over the original code were made.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#include "sound/audiostream.h"
|
#include "audio/audiostream.h"
|
||||||
#include "sound/rate.h"
|
#include "audio/rate.h"
|
||||||
#include "sound/mixer.h"
|
#include "audio/mixer.h"
|
||||||
#include "common/util.h"
|
#include "common/util.h"
|
||||||
|
|
||||||
//#define DEBUG_RATECONV
|
//#define DEBUG_RATECONV
|
||||||
@ -60,7 +60,7 @@ namespace Audio {
|
|||||||
* ARM routine we call doesn't respect those definitions.
|
* ARM routine we call doesn't respect those definitions.
|
||||||
*/
|
*/
|
||||||
#define FRAC_BITS 16
|
#define FRAC_BITS 16
|
||||||
#define FRAC_ONE (1<<FRAC_BITS)
|
#define FRAC_ONE (1 << FRAC_BITS)
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The size of the intermediate input cache. Bigger values may increase
|
* The size of the intermediate input cache. Bigger values may increase
|
||||||
@ -138,68 +138,66 @@ extern "C" {
|
|||||||
}
|
}
|
||||||
|
|
||||||
extern "C" st_sample_t *ARM_SimpleRate_M(
|
extern "C" st_sample_t *ARM_SimpleRate_M(
|
||||||
AudioStream &input,
|
AudioStream &input,
|
||||||
int (*fn)(Audio::AudioStream&,int16*,int),
|
int (*fn)(Audio::AudioStream&,int16*,int),
|
||||||
SimpleRateDetails *sr,
|
SimpleRateDetails *sr,
|
||||||
st_sample_t *obuf,
|
st_sample_t *obuf,
|
||||||
st_size_t osamp,
|
st_size_t osamp,
|
||||||
st_volume_t vol_l,
|
st_volume_t vol_l,
|
||||||
st_volume_t vol_r);
|
st_volume_t vol_r);
|
||||||
|
|
||||||
extern "C" st_sample_t *ARM_SimpleRate_S(
|
extern "C" st_sample_t *ARM_SimpleRate_S(
|
||||||
AudioStream &input,
|
AudioStream &input,
|
||||||
int (*fn)(Audio::AudioStream&,int16*,int),
|
int (*fn)(Audio::AudioStream&,int16*,int),
|
||||||
SimpleRateDetails *sr,
|
SimpleRateDetails *sr,
|
||||||
st_sample_t *obuf,
|
st_sample_t *obuf,
|
||||||
st_size_t osamp,
|
st_size_t osamp,
|
||||||
st_volume_t vol_l,
|
st_volume_t vol_l,
|
||||||
st_volume_t vol_r);
|
st_volume_t vol_r);
|
||||||
|
|
||||||
extern "C" st_sample_t *ARM_SimpleRate_R(
|
extern "C" st_sample_t *ARM_SimpleRate_R(
|
||||||
AudioStream &input,
|
AudioStream &input,
|
||||||
int (*fn)(Audio::AudioStream&,int16*,int),
|
int (*fn)(Audio::AudioStream&,int16*,int),
|
||||||
SimpleRateDetails *sr,
|
SimpleRateDetails *sr,
|
||||||
st_sample_t *obuf,
|
st_sample_t *obuf,
|
||||||
st_size_t osamp,
|
st_size_t osamp,
|
||||||
st_volume_t vol_l,
|
st_volume_t vol_l,
|
||||||
st_volume_t vol_r);
|
st_volume_t vol_r);
|
||||||
|
|
||||||
extern "C" int SimpleRate_readFudge(Audio::AudioStream &input,
|
extern "C" int SimpleRate_readFudge(Audio::AudioStream &input, int16 *a, int b)
|
||||||
int16 *a, int b)
|
|
||||||
{
|
{
|
||||||
#ifdef DEBUG_RATECONV
|
#ifdef DEBUG_RATECONV
|
||||||
fprintf(stderr, "Reading ptr=%x n%d\n", a, b);
|
debug("Reading ptr=%x n%d", a, b);
|
||||||
fflush(stderr);
|
|
||||||
#endif
|
#endif
|
||||||
return input.readBuffer(a, b);
|
return input.readBuffer(a, b);
|
||||||
}
|
}
|
||||||
|
|
||||||
template<bool stereo, bool reverseStereo>
|
template<bool stereo, bool reverseStereo>
|
||||||
int SimpleRateConverter<stereo, reverseStereo>::flow(AudioStream &input, st_sample_t *obuf, st_size_t osamp, st_volume_t vol_l, st_volume_t vol_r) {
|
int SimpleRateConverter<stereo, reverseStereo>::flow(AudioStream &input, st_sample_t *obuf, st_size_t osamp, st_volume_t vol_l, st_volume_t vol_r) {
|
||||||
|
|
||||||
#ifdef DEBUG_RATECONV
|
#ifdef DEBUG_RATECONV
|
||||||
fprintf(stderr, "Simple st=%d rev=%d\n", stereo, reverseStereo);
|
debug("Simple st=%d rev=%d", stereo, reverseStereo);
|
||||||
fflush(stderr);
|
|
||||||
#endif
|
#endif
|
||||||
st_sample_t *ostart = obuf;
|
st_sample_t *ostart = obuf;
|
||||||
|
|
||||||
if (!stereo) {
|
if (!stereo) {
|
||||||
obuf = ARM_SimpleRate_M(input,
|
obuf = ARM_SimpleRate_M(input,
|
||||||
&SimpleRate_readFudge,
|
&SimpleRate_readFudge,
|
||||||
&sr,
|
&sr,
|
||||||
obuf, osamp, vol_l, vol_r);
|
obuf, osamp, vol_l, vol_r);
|
||||||
} else if (reverseStereo) {
|
} else if (reverseStereo) {
|
||||||
obuf = ARM_SimpleRate_R(input,
|
obuf = ARM_SimpleRate_R(input,
|
||||||
&SimpleRate_readFudge,
|
&SimpleRate_readFudge,
|
||||||
&sr,
|
&sr,
|
||||||
obuf, osamp, vol_l, vol_r);
|
obuf, osamp, vol_l, vol_r);
|
||||||
} else {
|
} else {
|
||||||
obuf = ARM_SimpleRate_S(input,
|
obuf = ARM_SimpleRate_S(input,
|
||||||
&SimpleRate_readFudge,
|
&SimpleRate_readFudge,
|
||||||
&sr,
|
&sr,
|
||||||
obuf, osamp, vol_l, vol_r);
|
obuf, osamp, vol_l, vol_r);
|
||||||
}
|
}
|
||||||
return (obuf-ostart)/2;
|
|
||||||
|
return (obuf - ostart) / 2;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -242,31 +240,31 @@ extern "C" {
|
|||||||
}
|
}
|
||||||
|
|
||||||
extern "C" st_sample_t *ARM_LinearRate_M(
|
extern "C" st_sample_t *ARM_LinearRate_M(
|
||||||
AudioStream &input,
|
AudioStream &input,
|
||||||
int (*fn)(Audio::AudioStream&,int16*,int),
|
int (*fn)(Audio::AudioStream&,int16*,int),
|
||||||
LinearRateDetails *lr,
|
LinearRateDetails *lr,
|
||||||
st_sample_t *obuf,
|
st_sample_t *obuf,
|
||||||
st_size_t osamp,
|
st_size_t osamp,
|
||||||
st_volume_t vol_l,
|
st_volume_t vol_l,
|
||||||
st_volume_t vol_r);
|
st_volume_t vol_r);
|
||||||
|
|
||||||
extern "C" st_sample_t *ARM_LinearRate_S(
|
extern "C" st_sample_t *ARM_LinearRate_S(
|
||||||
AudioStream &input,
|
AudioStream &input,
|
||||||
int (*fn)(Audio::AudioStream&,int16*,int),
|
int (*fn)(Audio::AudioStream&,int16*,int),
|
||||||
LinearRateDetails *lr,
|
LinearRateDetails *lr,
|
||||||
st_sample_t *obuf,
|
st_sample_t *obuf,
|
||||||
st_size_t osamp,
|
st_size_t osamp,
|
||||||
st_volume_t vol_l,
|
st_volume_t vol_l,
|
||||||
st_volume_t vol_r);
|
st_volume_t vol_r);
|
||||||
|
|
||||||
extern "C" st_sample_t *ARM_LinearRate_R(
|
extern "C" st_sample_t *ARM_LinearRate_R(
|
||||||
AudioStream &input,
|
AudioStream &input,
|
||||||
int (*fn)(Audio::AudioStream&,int16*,int),
|
int (*fn)(Audio::AudioStream&,int16*,int),
|
||||||
LinearRateDetails *lr,
|
LinearRateDetails *lr,
|
||||||
st_sample_t *obuf,
|
st_sample_t *obuf,
|
||||||
st_size_t osamp,
|
st_size_t osamp,
|
||||||
st_volume_t vol_l,
|
st_volume_t vol_l,
|
||||||
st_volume_t vol_r);
|
st_volume_t vol_r);
|
||||||
|
|
||||||
template<bool stereo, bool reverseStereo>
|
template<bool stereo, bool reverseStereo>
|
||||||
class LinearRateConverter : public RateConverter {
|
class LinearRateConverter : public RateConverter {
|
||||||
@ -318,28 +316,33 @@ template<bool stereo, bool reverseStereo>
|
|||||||
int LinearRateConverter<stereo, reverseStereo>::flow(AudioStream &input, st_sample_t *obuf, st_size_t osamp, st_volume_t vol_l, st_volume_t vol_r) {
|
int LinearRateConverter<stereo, reverseStereo>::flow(AudioStream &input, st_sample_t *obuf, st_size_t osamp, st_volume_t vol_l, st_volume_t vol_r) {
|
||||||
|
|
||||||
#ifdef DEBUG_RATECONV
|
#ifdef DEBUG_RATECONV
|
||||||
fprintf(stderr, "Linear st=%d rev=%d\n", stereo, reverseStereo);
|
debug("Linear st=%d rev=%d", stereo, reverseStereo);
|
||||||
fflush(stderr);
|
|
||||||
#endif
|
#endif
|
||||||
st_sample_t *ostart = obuf;
|
st_sample_t *ostart = obuf;
|
||||||
|
|
||||||
|
if (vol_l > 0xff)
|
||||||
|
vol_l = 0xff;
|
||||||
|
|
||||||
|
if (vol_r > 0xff)
|
||||||
|
vol_r = 0xff;
|
||||||
|
|
||||||
if (!stereo) {
|
if (!stereo) {
|
||||||
obuf = ARM_LinearRate_M(input,
|
obuf = ARM_LinearRate_M(input,
|
||||||
&SimpleRate_readFudge,
|
&SimpleRate_readFudge,
|
||||||
&lr,
|
&lr,
|
||||||
obuf, osamp, vol_l, vol_r);
|
obuf, osamp, vol_l, vol_r);
|
||||||
} else if (reverseStereo) {
|
} else if (reverseStereo) {
|
||||||
obuf = ARM_LinearRate_R(input,
|
obuf = ARM_LinearRate_R(input,
|
||||||
&SimpleRate_readFudge,
|
&SimpleRate_readFudge,
|
||||||
&lr,
|
&lr,
|
||||||
obuf, osamp, vol_l, vol_r);
|
obuf, osamp, vol_l, vol_r);
|
||||||
} else {
|
} else {
|
||||||
obuf = ARM_LinearRate_S(input,
|
obuf = ARM_LinearRate_S(input,
|
||||||
&SimpleRate_readFudge,
|
&SimpleRate_readFudge,
|
||||||
&lr,
|
&lr,
|
||||||
obuf, osamp, vol_l, vol_r);
|
obuf, osamp, vol_l, vol_r);
|
||||||
}
|
}
|
||||||
return (obuf-ostart)/2;
|
return (obuf - ostart) / 2;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -358,31 +361,32 @@ extern "C" {
|
|||||||
}
|
}
|
||||||
|
|
||||||
extern "C" st_sample_t *ARM_CopyRate_M(
|
extern "C" st_sample_t *ARM_CopyRate_M(
|
||||||
st_size_t len,
|
st_size_t len,
|
||||||
st_sample_t *obuf,
|
st_sample_t *obuf,
|
||||||
st_volume_t vol_l,
|
st_volume_t vol_l,
|
||||||
st_volume_t vol_r,
|
st_volume_t vol_r,
|
||||||
st_sample_t *_buffer);
|
st_sample_t *_buffer);
|
||||||
|
|
||||||
extern "C" st_sample_t *ARM_CopyRate_S(
|
extern "C" st_sample_t *ARM_CopyRate_S(
|
||||||
st_size_t len,
|
st_size_t len,
|
||||||
st_sample_t *obuf,
|
st_sample_t *obuf,
|
||||||
st_volume_t vol_l,
|
st_volume_t vol_l,
|
||||||
st_volume_t vol_r,
|
st_volume_t vol_r,
|
||||||
st_sample_t *_buffer);
|
st_sample_t *_buffer);
|
||||||
|
|
||||||
extern "C" st_sample_t *ARM_CopyRate_R(
|
extern "C" st_sample_t *ARM_CopyRate_R(
|
||||||
st_size_t len,
|
st_size_t len,
|
||||||
st_sample_t *obuf,
|
st_sample_t *obuf,
|
||||||
st_volume_t vol_l,
|
st_volume_t vol_l,
|
||||||
st_volume_t vol_r,
|
st_volume_t vol_r,
|
||||||
st_sample_t *_buffer);
|
st_sample_t *_buffer);
|
||||||
|
|
||||||
|
|
||||||
template<bool stereo, bool reverseStereo>
|
template<bool stereo, bool reverseStereo>
|
||||||
class CopyRateConverter : public RateConverter {
|
class CopyRateConverter : public RateConverter {
|
||||||
st_sample_t *_buffer;
|
st_sample_t *_buffer;
|
||||||
st_size_t _bufferSize;
|
st_size_t _bufferSize;
|
||||||
|
|
||||||
public:
|
public:
|
||||||
CopyRateConverter() : _buffer(0), _bufferSize(0) {}
|
CopyRateConverter() : _buffer(0), _bufferSize(0) {}
|
||||||
~CopyRateConverter() {
|
~CopyRateConverter() {
|
||||||
@ -393,11 +397,10 @@ public:
|
|||||||
assert(input.isStereo() == stereo);
|
assert(input.isStereo() == stereo);
|
||||||
|
|
||||||
#ifdef DEBUG_RATECONV
|
#ifdef DEBUG_RATECONV
|
||||||
fprintf(stderr, "Copy st=%d rev=%d\n", stereo, reverseStereo);
|
debug("Copy st=%d rev=%d", stereo, reverseStereo);
|
||||||
fflush(stderr);
|
|
||||||
#endif
|
#endif
|
||||||
st_size_t len;
|
st_size_t len;
|
||||||
st_sample_t *ostart = obuf;
|
st_sample_t *ostart = obuf;
|
||||||
|
|
||||||
if (stereo)
|
if (stereo)
|
||||||
osamp *= 2;
|
osamp *= 2;
|
||||||
@ -422,8 +425,9 @@ fflush(stderr);
|
|||||||
else
|
else
|
||||||
obuf = ARM_CopyRate_M(len, obuf, vol_l, vol_r, _buffer);
|
obuf = ARM_CopyRate_M(len, obuf, vol_l, vol_r, _buffer);
|
||||||
|
|
||||||
return (obuf-ostart)/2;
|
return (obuf - ostart) / 2;
|
||||||
}
|
}
|
||||||
|
|
||||||
virtual int drain(st_sample_t *obuf, st_size_t osamp, st_volume_t vol) {
|
virtual int drain(st_sample_t *obuf, st_size_t osamp, st_volume_t vol) {
|
||||||
return (ST_SUCCESS);
|
return (ST_SUCCESS);
|
||||||
}
|
}
|
||||||
@ -467,3 +471,5 @@ RateConverter *makeRateConverter(st_rate_t inrate, st_rate_t outrate, bool stere
|
|||||||
}
|
}
|
||||||
|
|
||||||
} // End of namespace Audio
|
} // End of namespace Audio
|
||||||
|
|
||||||
|
|
@ -1,3 +1,4 @@
|
|||||||
|
|
||||||
/* Residual - A 3D game interpreter
|
/* Residual - A 3D game interpreter
|
||||||
*
|
*
|
||||||
* Residual is the legal property of its developers, whose names
|
* Residual is the legal property of its developers, whose names
|
||||||
@ -22,11 +23,11 @@
|
|||||||
* $Id$
|
* $Id$
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#include "sound/softsynth/emumidi.h"
|
#include "audio/softsynth/emumidi.h"
|
||||||
#include "common/debug.h"
|
#include "common/debug.h"
|
||||||
#include "common/util.h"
|
#include "common/util.h"
|
||||||
#include "sound/fmopl.h"
|
#include "audio/fmopl.h"
|
||||||
#include "sound/musicplugin.h"
|
#include "audio/musicplugin.h"
|
||||||
#include "common/translation.h"
|
#include "common/translation.h"
|
||||||
|
|
||||||
#ifdef DEBUG_ADLIB
|
#ifdef DEBUG_ADLIB
|
||||||
@ -118,7 +119,7 @@ public:
|
|||||||
byte getNumber() { return _channel; }
|
byte getNumber() { return _channel; }
|
||||||
void release() { _allocated = false; }
|
void release() { _allocated = false; }
|
||||||
|
|
||||||
void send (uint32 b);
|
void send(uint32 b);
|
||||||
|
|
||||||
// Regular messages
|
// Regular messages
|
||||||
void noteOff(byte note);
|
void noteOff(byte note);
|
@ -23,7 +23,7 @@
|
|||||||
*
|
*
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#include "sound/null.h"
|
#include "audio/null.h"
|
||||||
|
|
||||||
// Plugin interface
|
// Plugin interface
|
||||||
// (This can only create a null driver since apple II gs support seeems not to be implemented
|
// (This can only create a null driver since apple II gs support seeems not to be implemented
|
@ -22,8 +22,8 @@
|
|||||||
* $Id$
|
* $Id$
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#include "sound/softsynth/cms.h"
|
#include "audio/softsynth/cms.h"
|
||||||
#include "sound/null.h"
|
#include "audio/null.h"
|
||||||
|
|
||||||
#include "common/textconsole.h"
|
#include "common/textconsole.h"
|
||||||
#include "common/translation.h"
|
#include "common/translation.h"
|
||||||
@ -86,35 +86,39 @@ static const int amplitude_lookup[16] = {
|
|||||||
|
|
||||||
void CMSEmulator::portWrite(int port, int val) {
|
void CMSEmulator::portWrite(int port, int val) {
|
||||||
switch (port) {
|
switch (port) {
|
||||||
case 0x220:
|
case 0x220:
|
||||||
portWriteIntern(0, 1, val);
|
portWriteIntern(0, 1, val);
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case 0x221:
|
case 0x221:
|
||||||
_saa1099[0].selected_reg = val & 0x1f;
|
_saa1099[0].selected_reg = val & 0x1f;
|
||||||
if (_saa1099[0].selected_reg == 0x18 || _saa1099[0].selected_reg == 0x19) {
|
if (_saa1099[0].selected_reg == 0x18 || _saa1099[0].selected_reg == 0x19) {
|
||||||
/* clock the envelope channels */
|
/* clock the envelope channels */
|
||||||
if (_saa1099[0].env_clock[0]) envelope(0, 0);
|
if (_saa1099[0].env_clock[0])
|
||||||
if (_saa1099[0].env_clock[1]) envelope(0, 1);
|
envelope(0, 0);
|
||||||
}
|
if (_saa1099[0].env_clock[1])
|
||||||
break;
|
envelope(0, 1);
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
|
||||||
case 0x222:
|
case 0x222:
|
||||||
portWriteIntern(1, 1, val);
|
portWriteIntern(1, 1, val);
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case 0x223:
|
case 0x223:
|
||||||
_saa1099[1].selected_reg = val & 0x1f;
|
_saa1099[1].selected_reg = val & 0x1f;
|
||||||
if (_saa1099[1].selected_reg == 0x18 || _saa1099[1].selected_reg == 0x19) {
|
if (_saa1099[1].selected_reg == 0x18 || _saa1099[1].selected_reg == 0x19) {
|
||||||
/* clock the envelope channels */
|
/* clock the envelope channels */
|
||||||
if (_saa1099[1].env_clock[0]) envelope(1, 0);
|
if (_saa1099[1].env_clock[0])
|
||||||
if (_saa1099[1].env_clock[1]) envelope(1, 1);
|
envelope(1, 0);
|
||||||
}
|
if (_saa1099[1].env_clock[1])
|
||||||
break;
|
envelope(1, 1);
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
|
||||||
default:
|
default:
|
||||||
warning("CMSEmulator got port: 0x%X", port);
|
warning("CMSEmulator got port: 0x%X", port);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -177,10 +181,10 @@ void CMSEmulator::update(int chip, int16 *buffer, int length) {
|
|||||||
|
|
||||||
for (ch = 0; ch < 2; ch++) {
|
for (ch = 0; ch < 2; ch++) {
|
||||||
switch (saa->noise_params[ch]) {
|
switch (saa->noise_params[ch]) {
|
||||||
case 0: saa->noise[ch].freq = 31250.0 * 2; break;
|
case 0: saa->noise[ch].freq = 31250.0 * 2; break;
|
||||||
case 1: saa->noise[ch].freq = 15625.0 * 2; break;
|
case 1: saa->noise[ch].freq = 15625.0 * 2; break;
|
||||||
case 2: saa->noise[ch].freq = 7812.5 * 2; break;
|
case 2: saa->noise[ch].freq = 7812.5 * 2; break;
|
||||||
case 3: saa->noise[ch].freq = saa->channels[ch * 3].freq; break;
|
case 3: saa->noise[ch].freq = saa->channels[ch * 3].freq; break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -254,95 +258,95 @@ void CMSEmulator::portWriteIntern(int chip, int offset, int data) {
|
|||||||
int ch;
|
int ch;
|
||||||
|
|
||||||
switch (reg) {
|
switch (reg) {
|
||||||
/* channel i amplitude */
|
/* channel i amplitude */
|
||||||
case 0x00:
|
case 0x00:
|
||||||
case 0x01:
|
case 0x01:
|
||||||
case 0x02:
|
case 0x02:
|
||||||
case 0x03:
|
case 0x03:
|
||||||
case 0x04:
|
case 0x04:
|
||||||
case 0x05:
|
case 0x05:
|
||||||
ch = reg & 7;
|
ch = reg & 7;
|
||||||
saa->channels[ch].amplitude[LEFT] = amplitude_lookup[data & 0x0f];
|
saa->channels[ch].amplitude[LEFT] = amplitude_lookup[data & 0x0f];
|
||||||
saa->channels[ch].amplitude[RIGHT] = amplitude_lookup[(data >> 4) & 0x0f];
|
saa->channels[ch].amplitude[RIGHT] = amplitude_lookup[(data >> 4) & 0x0f];
|
||||||
break;
|
break;
|
||||||
|
|
||||||
/* channel i frequency */
|
/* channel i frequency */
|
||||||
case 0x08:
|
case 0x08:
|
||||||
case 0x09:
|
case 0x09:
|
||||||
case 0x0a:
|
case 0x0a:
|
||||||
case 0x0b:
|
case 0x0b:
|
||||||
case 0x0c:
|
case 0x0c:
|
||||||
case 0x0d:
|
case 0x0d:
|
||||||
ch = reg & 7;
|
ch = reg & 7;
|
||||||
saa->channels[ch].frequency = data & 0xff;
|
saa->channels[ch].frequency = data & 0xff;
|
||||||
break;
|
break;
|
||||||
|
|
||||||
/* channel i octave */
|
/* channel i octave */
|
||||||
case 0x10:
|
case 0x10:
|
||||||
case 0x11:
|
case 0x11:
|
||||||
case 0x12:
|
case 0x12:
|
||||||
ch = (reg - 0x10) << 1;
|
ch = (reg - 0x10) << 1;
|
||||||
saa->channels[ch + 0].octave = data & 0x07;
|
saa->channels[ch + 0].octave = data & 0x07;
|
||||||
saa->channels[ch + 1].octave = (data >> 4) & 0x07;
|
saa->channels[ch + 1].octave = (data >> 4) & 0x07;
|
||||||
break;
|
break;
|
||||||
|
|
||||||
/* channel i frequency enable */
|
/* channel i frequency enable */
|
||||||
case 0x14:
|
case 0x14:
|
||||||
saa->channels[0].freq_enable = data & 0x01;
|
saa->channels[0].freq_enable = data & 0x01;
|
||||||
saa->channels[1].freq_enable = data & 0x02;
|
saa->channels[1].freq_enable = data & 0x02;
|
||||||
saa->channels[2].freq_enable = data & 0x04;
|
saa->channels[2].freq_enable = data & 0x04;
|
||||||
saa->channels[3].freq_enable = data & 0x08;
|
saa->channels[3].freq_enable = data & 0x08;
|
||||||
saa->channels[4].freq_enable = data & 0x10;
|
saa->channels[4].freq_enable = data & 0x10;
|
||||||
saa->channels[5].freq_enable = data & 0x20;
|
saa->channels[5].freq_enable = data & 0x20;
|
||||||
break;
|
break;
|
||||||
|
|
||||||
/* channel i noise enable */
|
/* channel i noise enable */
|
||||||
case 0x15:
|
case 0x15:
|
||||||
saa->channels[0].noise_enable = data & 0x01;
|
saa->channels[0].noise_enable = data & 0x01;
|
||||||
saa->channels[1].noise_enable = data & 0x02;
|
saa->channels[1].noise_enable = data & 0x02;
|
||||||
saa->channels[2].noise_enable = data & 0x04;
|
saa->channels[2].noise_enable = data & 0x04;
|
||||||
saa->channels[3].noise_enable = data & 0x08;
|
saa->channels[3].noise_enable = data & 0x08;
|
||||||
saa->channels[4].noise_enable = data & 0x10;
|
saa->channels[4].noise_enable = data & 0x10;
|
||||||
saa->channels[5].noise_enable = data & 0x20;
|
saa->channels[5].noise_enable = data & 0x20;
|
||||||
break;
|
break;
|
||||||
|
|
||||||
/* noise generators parameters */
|
/* noise generators parameters */
|
||||||
case 0x16:
|
case 0x16:
|
||||||
saa->noise_params[0] = data & 0x03;
|
saa->noise_params[0] = data & 0x03;
|
||||||
saa->noise_params[1] = (data >> 4) & 0x03;
|
saa->noise_params[1] = (data >> 4) & 0x03;
|
||||||
break;
|
break;
|
||||||
|
|
||||||
/* envelope generators parameters */
|
/* envelope generators parameters */
|
||||||
case 0x18:
|
case 0x18:
|
||||||
case 0x19:
|
case 0x19:
|
||||||
ch = reg - 0x18;
|
ch = reg - 0x18;
|
||||||
saa->env_reverse_right[ch] = data & 0x01;
|
saa->env_reverse_right[ch] = data & 0x01;
|
||||||
saa->env_mode[ch] = (data >> 1) & 0x07;
|
saa->env_mode[ch] = (data >> 1) & 0x07;
|
||||||
saa->env_bits[ch] = data & 0x10;
|
saa->env_bits[ch] = data & 0x10;
|
||||||
saa->env_clock[ch] = data & 0x20;
|
saa->env_clock[ch] = data & 0x20;
|
||||||
saa->env_enable[ch] = data & 0x80;
|
saa->env_enable[ch] = data & 0x80;
|
||||||
/* reset the envelope */
|
/* reset the envelope */
|
||||||
saa->env_step[ch] = 0;
|
saa->env_step[ch] = 0;
|
||||||
break;
|
break;
|
||||||
|
|
||||||
/* channels enable & reset generators */
|
/* channels enable & reset generators */
|
||||||
case 0x1c:
|
case 0x1c:
|
||||||
saa->all_ch_enable = data & 0x01;
|
saa->all_ch_enable = data & 0x01;
|
||||||
saa->sync_state = data & 0x02;
|
saa->sync_state = data & 0x02;
|
||||||
if (data & 0x02) {
|
if (data & 0x02) {
|
||||||
int i;
|
int i;
|
||||||
/* Synch & Reset generators */
|
/* Synch & Reset generators */
|
||||||
for (i = 0; i < 6; i++) {
|
for (i = 0; i < 6; i++) {
|
||||||
saa->channels[i].level = 0;
|
saa->channels[i].level = 0;
|
||||||
saa->channels[i].counter = 0.0;
|
saa->channels[i].counter = 0.0;
|
||||||
}
|
|
||||||
}
|
}
|
||||||
break;
|
}
|
||||||
|
break;
|
||||||
|
|
||||||
default:
|
default:
|
||||||
// The CMS allows all registers to be written, so we just output some debug
|
// The CMS allows all registers to be written, so we just output some debug
|
||||||
// message here
|
// message here
|
||||||
debug(5, "CMS Unkown write to reg %x with %x",reg, data);
|
debug(5, "CMS Unknown write to reg %x with %x",reg, data);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
@ -25,9 +25,9 @@
|
|||||||
#ifndef SOUND_SOFTSYNTH_EMUMIDI_H
|
#ifndef SOUND_SOFTSYNTH_EMUMIDI_H
|
||||||
#define SOUND_SOFTSYNTH_EMUMIDI_H
|
#define SOUND_SOFTSYNTH_EMUMIDI_H
|
||||||
|
|
||||||
#include "sound/audiostream.h"
|
#include "audio/audiostream.h"
|
||||||
#include "sound/mididrv.h"
|
#include "audio/mididrv.h"
|
||||||
#include "sound/mixer.h"
|
#include "audio/mixer.h"
|
||||||
|
|
||||||
#define FIXP_SHIFT 16
|
#define FIXP_SHIFT 16
|
||||||
|
|
||||||
@ -45,25 +45,24 @@ private:
|
|||||||
int _samplesPerTick;
|
int _samplesPerTick;
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
|
int _baseFreq;
|
||||||
|
|
||||||
virtual void generateSamples(int16 *buf, int len) = 0;
|
virtual void generateSamples(int16 *buf, int len) = 0;
|
||||||
virtual void onTimer() {}
|
virtual void onTimer() {}
|
||||||
|
|
||||||
int _baseFreq;
|
|
||||||
|
|
||||||
public:
|
public:
|
||||||
MidiDriver_Emulated(Audio::Mixer *mixer) : _mixer(mixer) {
|
MidiDriver_Emulated(Audio::Mixer *mixer) :
|
||||||
_isOpen = false;
|
_mixer(mixer),
|
||||||
|
_isOpen(false),
|
||||||
_timerProc = 0;
|
_timerProc(0),
|
||||||
_timerParam = 0;
|
_timerParam(0),
|
||||||
|
_nextTick(0),
|
||||||
_nextTick = 0;
|
_samplesPerTick(0),
|
||||||
_samplesPerTick = 0;
|
_baseFreq(250) {
|
||||||
|
|
||||||
_baseFreq = 250;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
int open() {
|
// MidiDriver API
|
||||||
|
virtual int open() {
|
||||||
_isOpen = true;
|
_isOpen = true;
|
||||||
|
|
||||||
int d = getRate() / _baseFreq;
|
int d = getRate() / _baseFreq;
|
||||||
@ -73,19 +72,23 @@ public:
|
|||||||
// but less prone to arithmetic overflow.
|
// but less prone to arithmetic overflow.
|
||||||
|
|
||||||
_samplesPerTick = (d << FIXP_SHIFT) + (r << FIXP_SHIFT) / _baseFreq;
|
_samplesPerTick = (d << FIXP_SHIFT) + (r << FIXP_SHIFT) / _baseFreq;
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
void setTimerCallback(void *timer_param, Common::TimerManager::TimerProc timer_proc) {
|
bool isOpen() const { return _isOpen; }
|
||||||
|
|
||||||
|
virtual void setTimerCallback(void *timer_param, Common::TimerManager::TimerProc timer_proc) {
|
||||||
_timerProc = timer_proc;
|
_timerProc = timer_proc;
|
||||||
_timerParam = timer_param;
|
_timerParam = timer_param;
|
||||||
}
|
}
|
||||||
|
|
||||||
uint32 getBaseTempo() { return 1000000 / _baseFreq; }
|
virtual uint32 getBaseTempo() {
|
||||||
|
return 1000000 / _baseFreq;
|
||||||
|
}
|
||||||
|
|
||||||
// AudioStream API
|
// AudioStream API
|
||||||
int readBuffer(int16 *data, const int numSamples) {
|
virtual int readBuffer(int16 *data, const int numSamples) {
|
||||||
const int stereoFactor = isStereo() ? 2 : 1;
|
const int stereoFactor = isStereo() ? 2 : 1;
|
||||||
int len = numSamples / stereoFactor;
|
int len = numSamples / stereoFactor;
|
||||||
int step;
|
int step;
|
||||||
@ -101,16 +104,22 @@ public:
|
|||||||
if (!(_nextTick >> FIXP_SHIFT)) {
|
if (!(_nextTick >> FIXP_SHIFT)) {
|
||||||
if (_timerProc)
|
if (_timerProc)
|
||||||
(*_timerProc)(_timerParam);
|
(*_timerProc)(_timerParam);
|
||||||
|
|
||||||
onTimer();
|
onTimer();
|
||||||
|
|
||||||
_nextTick += _samplesPerTick;
|
_nextTick += _samplesPerTick;
|
||||||
}
|
}
|
||||||
|
|
||||||
data += step * stereoFactor;
|
data += step * stereoFactor;
|
||||||
len -= step;
|
len -= step;
|
||||||
} while (len);
|
} while (len);
|
||||||
|
|
||||||
return numSamples;
|
return numSamples;
|
||||||
}
|
}
|
||||||
bool endOfData() const { return false; }
|
|
||||||
|
virtual bool endOfData() const {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
#endif
|
#endif
|
@ -27,9 +27,9 @@
|
|||||||
#ifdef USE_FLUIDSYNTH
|
#ifdef USE_FLUIDSYNTH
|
||||||
|
|
||||||
#include "common/config-manager.h"
|
#include "common/config-manager.h"
|
||||||
#include "sound/musicplugin.h"
|
#include "audio/musicplugin.h"
|
||||||
#include "sound/mpu401.h"
|
#include "audio/mpu401.h"
|
||||||
#include "sound/softsynth/emumidi.h"
|
#include "audio/softsynth/emumidi.h"
|
||||||
|
|
||||||
#include <fluidsynth.h>
|
#include <fluidsynth.h>
|
||||||
|
|
||||||
@ -40,7 +40,6 @@ private:
|
|||||||
fluid_synth_t *_synth;
|
fluid_synth_t *_synth;
|
||||||
int _soundFont;
|
int _soundFont;
|
||||||
int _outputRate;
|
int _outputRate;
|
||||||
Audio::SoundHandle _handle;
|
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
// Because GCC complains about casting from const to non-const...
|
// Because GCC complains about casting from const to non-const...
|
||||||
@ -144,7 +143,7 @@ int MidiDriver_FluidSynth::open() {
|
|||||||
MidiDriver_Emulated::open();
|
MidiDriver_Emulated::open();
|
||||||
|
|
||||||
// The MT-32 emulator uses kSFXSoundType here. I don't know why.
|
// The MT-32 emulator uses kSFXSoundType here. I don't know why.
|
||||||
_mixer->playStream(Audio::Mixer::kMusicSoundType, &_handle, this, -1, Audio::Mixer::kMaxChannelVolume, 0, DisposeAfterUse::NO, true);
|
_mixer->playStream(Audio::Mixer::kMusicSoundType, &_mixerSoundHandle, this, -1, Audio::Mixer::kMaxChannelVolume, 0, DisposeAfterUse::NO, true);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -153,7 +152,7 @@ void MidiDriver_FluidSynth::close() {
|
|||||||
return;
|
return;
|
||||||
_isOpen = false;
|
_isOpen = false;
|
||||||
|
|
||||||
_mixer->stopHandle(_handle);
|
_mixer->stopHandle(_mixerSoundHandle);
|
||||||
|
|
||||||
if (_soundFont != -1)
|
if (_soundFont != -1)
|
||||||
fluid_synth_sfunload(_synth, _soundFont, 1);
|
fluid_synth_sfunload(_synth, _soundFont, 1);
|
@ -26,11 +26,11 @@
|
|||||||
|
|
||||||
#ifdef USE_MT32EMU
|
#ifdef USE_MT32EMU
|
||||||
|
|
||||||
#include "sound/softsynth/mt32/mt32emu.h"
|
#include "audio/softsynth/mt32/mt32emu.h"
|
||||||
|
|
||||||
#include "sound/softsynth/emumidi.h"
|
#include "audio/softsynth/emumidi.h"
|
||||||
#include "sound/musicplugin.h"
|
#include "audio/musicplugin.h"
|
||||||
#include "sound/mpu401.h"
|
#include "audio/mpu401.h"
|
||||||
|
|
||||||
#include "common/config-manager.h"
|
#include "common/config-manager.h"
|
||||||
#include "common/debug.h"
|
#include "common/debug.h"
|
||||||
@ -51,7 +51,6 @@ class MidiChannel_MT32 : public MidiChannel_MPU401 {
|
|||||||
|
|
||||||
class MidiDriver_MT32 : public MidiDriver_Emulated {
|
class MidiDriver_MT32 : public MidiDriver_Emulated {
|
||||||
private:
|
private:
|
||||||
Audio::SoundHandle _handle;
|
|
||||||
MidiChannel_MT32 _midiChannels[16];
|
MidiChannel_MT32 _midiChannels[16];
|
||||||
uint16 _channelMask;
|
uint16 _channelMask;
|
||||||
MT32Emu::Synth *_synth;
|
MT32Emu::Synth *_synth;
|
||||||
@ -149,7 +148,7 @@ static void drawProgress(float progress) {
|
|||||||
Common::Rect r(x, y, x + w, y + h);
|
Common::Rect r(x, y, x + w, y + h);
|
||||||
|
|
||||||
uint32 col;
|
uint32 col;
|
||||||
|
|
||||||
if (screenFormat.bytesPerPixel > 1)
|
if (screenFormat.bytesPerPixel > 1)
|
||||||
col = screenFormat.RGBToColor(0, 171, 0);
|
col = screenFormat.RGBToColor(0, 171, 0);
|
||||||
else
|
else
|
||||||
@ -230,8 +229,7 @@ static void MT32_PrintDebug(void *userData, const char *fmt, va_list list) {
|
|||||||
static int MT32_Report(void *userData, MT32Emu::ReportType type, const void *reportData) {
|
static int MT32_Report(void *userData, MT32Emu::ReportType type, const void *reportData) {
|
||||||
switch (type) {
|
switch (type) {
|
||||||
case MT32Emu::ReportType_lcdMessage:
|
case MT32Emu::ReportType_lcdMessage:
|
||||||
// FIXME implament in Residual
|
g_system->displayMessageOnOSD((const char *)reportData);
|
||||||
// g_system->displayMessageOnOSD((const char *)reportData);
|
|
||||||
break;
|
break;
|
||||||
case MT32Emu::ReportType_errorControlROM:
|
case MT32Emu::ReportType_errorControlROM:
|
||||||
error("Failed to load MT32_CONTROL.ROM");
|
error("Failed to load MT32_CONTROL.ROM");
|
||||||
@ -317,9 +315,9 @@ int MidiDriver_MT32::open() {
|
|||||||
|
|
||||||
if (screenFormat.bytesPerPixel == 1) {
|
if (screenFormat.bytesPerPixel == 1) {
|
||||||
const byte dummy_palette[] = {
|
const byte dummy_palette[] = {
|
||||||
0, 0, 0, 0, // background
|
0, 0, 0, // background
|
||||||
0, 171, 0, 0, // border, font
|
0, 171, 0, // border, font
|
||||||
171, 0, 0, 0 // fill
|
171, 0, 0 // fill
|
||||||
};
|
};
|
||||||
|
|
||||||
// TODO implement in Residual
|
// TODO implement in Residual
|
||||||
@ -340,7 +338,7 @@ int MidiDriver_MT32::open() {
|
|||||||
|
|
||||||
g_system->updateScreen();*/
|
g_system->updateScreen();*/
|
||||||
|
|
||||||
_mixer->playStream(Audio::Mixer::kSFXSoundType, &_handle, this, -1, Audio::Mixer::kMaxChannelVolume, 0, DisposeAfterUse::NO, true);
|
_mixer->playStream(Audio::Mixer::kSFXSoundType, &_mixerSoundHandle, this, -1, Audio::Mixer::kMaxChannelVolume, 0, DisposeAfterUse::NO, true);
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
@ -382,7 +380,7 @@ void MidiDriver_MT32::close() {
|
|||||||
// Detach the player callback handler
|
// Detach the player callback handler
|
||||||
setTimerCallback(NULL, NULL);
|
setTimerCallback(NULL, NULL);
|
||||||
// Detach the mixer callback handler
|
// Detach the mixer callback handler
|
||||||
_mixer->stopHandle(_handle);
|
_mixer->stopHandle(_mixerSoundHandle);
|
||||||
|
|
||||||
_synth->close();
|
_synth->close();
|
||||||
delete _synth;
|
delete _synth;
|
@ -29,7 +29,7 @@
|
|||||||
// http://www.dreampoint.co.uk
|
// http://www.dreampoint.co.uk
|
||||||
// This code is public domain
|
// This code is public domain
|
||||||
|
|
||||||
#include "sound/softsynth/mt32/freeverb.h"
|
#include "audio/softsynth/mt32/freeverb.h"
|
||||||
|
|
||||||
comb::comb() {
|
comb::comb() {
|
||||||
filterstore = 0;
|
filterstore = 0;
|
||||||
@ -125,12 +125,12 @@ revmodel::revmodel() {
|
|||||||
allpassR[2].setfeedback(0.5f);
|
allpassR[2].setfeedback(0.5f);
|
||||||
allpassL[3].setfeedback(0.5f);
|
allpassL[3].setfeedback(0.5f);
|
||||||
allpassR[3].setfeedback(0.5f);
|
allpassR[3].setfeedback(0.5f);
|
||||||
|
setmode(initialmode);
|
||||||
setwet(initialwet);
|
setwet(initialwet);
|
||||||
setroomsize(initialroom);
|
setroomsize(initialroom);
|
||||||
setdry(initialdry);
|
setdry(initialdry);
|
||||||
setdamp(initialdamp);
|
setdamp(initialdamp);
|
||||||
setwidth(initialwidth);
|
setwidth(initialwidth);
|
||||||
setmode(initialmode);
|
|
||||||
|
|
||||||
// Buffer will be full of rubbish - so we MUST mute them
|
// Buffer will be full of rubbish - so we MUST mute them
|
||||||
mute();
|
mute();
|
@ -1,4 +1,4 @@
|
|||||||
MODULE := sound/softsynth/mt32
|
MODULE := audio/softsynth/mt32
|
||||||
|
|
||||||
MODULE_OBJS := \
|
MODULE_OBJS := \
|
||||||
mt32_file.o \
|
mt32_file.o \
|
70
audio/softsynth/mt32/mt32_file.cpp
Normal file
70
audio/softsynth/mt32/mt32_file.cpp
Normal file
@ -0,0 +1,70 @@
|
|||||||
|
/* Copyright (c) 2003-2005 Various contributors
|
||||||
|
*
|
||||||
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
|
* of this software and associated documentation files (the "Software"), to
|
||||||
|
* deal in the Software without restriction, including without limitation the
|
||||||
|
* rights to use, copy, modify, merge, publish, distribute, sublicense, and/or
|
||||||
|
* sell copies of the Software, and to permit persons to whom the Software is
|
||||||
|
* furnished to do so, subject to the following conditions:
|
||||||
|
*
|
||||||
|
* The above copyright notice and this permission notice shall be included in
|
||||||
|
* all copies or substantial portions of the Software.
|
||||||
|
*
|
||||||
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||||
|
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||||
|
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||||
|
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||||
|
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
|
||||||
|
* FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS
|
||||||
|
* IN THE SOFTWARE.
|
||||||
|
*/
|
||||||
|
|
||||||
|
|
||||||
|
#include "mt32emu.h"
|
||||||
|
|
||||||
|
namespace MT32Emu {
|
||||||
|
|
||||||
|
bool File::readBit16u(Bit16u *in) {
|
||||||
|
Bit8u b[2];
|
||||||
|
if (read(&b[0], 2) != 2)
|
||||||
|
return false;
|
||||||
|
*in = ((b[0] << 8) | b[1]);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool File::readBit32u(Bit32u *in) {
|
||||||
|
Bit8u b[4];
|
||||||
|
if (read(&b[0], 4) != 4)
|
||||||
|
return false;
|
||||||
|
*in = ((b[0] << 24) | (b[1] << 16) | (b[2] << 8) | b[3]);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool File::writeBit16u(Bit16u out) {
|
||||||
|
if (!writeBit8u((Bit8u)((out & 0xFF00) >> 8))) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
if (!writeBit8u((Bit8u)(out & 0x00FF))) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool File::writeBit32u(Bit32u out) {
|
||||||
|
if (!writeBit8u((Bit8u)((out & 0xFF000000) >> 24))) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
if (!writeBit8u((Bit8u)((out & 0x00FF0000) >> 16))) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
if (!writeBit8u((Bit8u)((out & 0x0000FF00) >> 8))) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
if (!writeBit8u((Bit8u)(out & 0x000000FF))) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
} // End of namespace MT32Emu
|
||||||
|
|
@ -22,7 +22,7 @@
|
|||||||
#ifndef MT32EMU_FILE_H
|
#ifndef MT32EMU_FILE_H
|
||||||
#define MT32EMU_FILE_H
|
#define MT32EMU_FILE_H
|
||||||
|
|
||||||
#include <stdio.h>
|
#include "common/sys.h"
|
||||||
|
|
||||||
namespace MT32Emu {
|
namespace MT32Emu {
|
||||||
|
|
||||||
@ -47,19 +47,6 @@ public:
|
|||||||
virtual bool isEOF() = 0;
|
virtual bool isEOF() = 0;
|
||||||
};
|
};
|
||||||
|
|
||||||
class ANSIFile: public File {
|
} // End of namespace MT32Emu
|
||||||
private:
|
|
||||||
FILE *fp;
|
|
||||||
public:
|
|
||||||
bool open(const char *filename, OpenMode mode);
|
|
||||||
void close();
|
|
||||||
size_t read(void *in, size_t size);
|
|
||||||
bool readBit8u(Bit8u *in);
|
|
||||||
size_t write(const void *out, size_t size);
|
|
||||||
bool writeBit8u(Bit8u out);
|
|
||||||
bool isEOF();
|
|
||||||
};
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
#endif
|
#endif
|
@ -283,6 +283,7 @@ void Part::cacheTimbre(PatchCache cache[4], const TimbreParam *timbre) {
|
|||||||
backupCacheToPartials(cache);
|
backupCacheToPartials(cache);
|
||||||
int partialCount = 0;
|
int partialCount = 0;
|
||||||
for (int t = 0; t < 4; t++) {
|
for (int t = 0; t < 4; t++) {
|
||||||
|
cache[t].PCMPartial = false;
|
||||||
if (((timbre->common.pmute >> t) & 0x1) == 1) {
|
if (((timbre->common.pmute >> t) & 0x1) == 1) {
|
||||||
cache[t].playPartial = true;
|
cache[t].playPartial = true;
|
||||||
partialCount++;
|
partialCount++;
|
@ -118,7 +118,7 @@ void Partial::initKeyFollow(int key) {
|
|||||||
#endif
|
#endif
|
||||||
#if MT32EMU_ACCURATENOTES == 1
|
#if MT32EMU_ACCURATENOTES == 1
|
||||||
noteVal = newPitch;
|
noteVal = newPitch;
|
||||||
synth->printDebug("key=%d, pitch=%f, pitchKeyfollow=%f, pitchShift=%f, newPitch=%f", key, patchCache->pitch, patchCache->pitchKeyfollow, patchCache->pitchShift, newPitch);
|
synth->printDebug("key=%d, pitch=%f, pitchKeyfollow=%f, pitchShift=%f, newPitch=%f", key, (double)patchCache->pitch, (double)patchCache->pitchKeyfollow, (double)patchCache->pitchShift, (double)newPitch);
|
||||||
#else
|
#else
|
||||||
float newPitchInt;
|
float newPitchInt;
|
||||||
float newPitchFract = modff(newPitch, &newPitchInt);
|
float newPitchFract = modff(newPitch, &newPitchInt);
|
||||||
@ -128,7 +128,7 @@ void Partial::initKeyFollow(int key) {
|
|||||||
}
|
}
|
||||||
noteVal = (int)newPitchInt;
|
noteVal = (int)newPitchInt;
|
||||||
fineShift = (int)(powf(2.0f, newPitchFract / 12.0f) * 4096.0f);
|
fineShift = (int)(powf(2.0f, newPitchFract / 12.0f) * 4096.0f);
|
||||||
synth->printDebug("key=%d, pitch=%f, pitchKeyfollow=%f, pitchShift=%f, newPitch=%f, noteVal=%d, fineShift=%d", key, patchCache->pitch, patchCache->pitchKeyfollow, patchCache->pitchShift, newPitch, noteVal, fineShift);
|
synth->printDebug("key=%d, pitch=%f, pitchKeyfollow=%f, pitchShift=%f, newPitch=%f, noteVal=%d, fineShift=%d", key, (double)patchCache->pitch, (double)patchCache->pitchKeyfollow, (double)patchCache->pitchShift, (double)newPitch, noteVal, fineShift);
|
||||||
#endif
|
#endif
|
||||||
// FIXME:KG: Raise/lower by octaves until in the supported range.
|
// FIXME:KG: Raise/lower by octaves until in the supported range.
|
||||||
while (noteVal > HIGHEST_NOTE) // FIXME:KG: see tables.cpp: >108?
|
while (noteVal > HIGHEST_NOTE) // FIXME:KG: see tables.cpp: >108?
|
||||||
@ -451,7 +451,7 @@ void Partial::setBend(float factor) {
|
|||||||
// FIXME:KG: Bend should be influenced by pitch key-follow too, according to docs.
|
// FIXME:KG: Bend should be influenced by pitch key-follow too, according to docs.
|
||||||
float bendSemitones = factor * patchCache->benderRange; // -24 .. 24
|
float bendSemitones = factor * patchCache->benderRange; // -24 .. 24
|
||||||
float mult = powf(2.0f, bendSemitones / 12.0f);
|
float mult = powf(2.0f, bendSemitones / 12.0f);
|
||||||
synth->printDebug("setBend(): factor=%f, benderRange=%f, semitones=%f, mult=%f\n", factor, patchCache->benderRange, bendSemitones, mult);
|
synth->printDebug("setBend(): factor=%f, benderRange=%f, semitones=%f, mult=%f\n", (double)factor, (double)patchCache->benderRange, (double)bendSemitones, (double)mult);
|
||||||
bendShift = (int)(mult * 4096.0f);
|
bendShift = (int)(mult * 4096.0f);
|
||||||
}
|
}
|
||||||
|
|
@ -162,21 +162,11 @@ void Synth::initReverb(Bit8u newRevMode, Bit8u newRevTime, Bit8u newRevLevel) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
File *Synth::openFile(const char *filename, File::OpenMode mode) {
|
File *Synth::openFile(const char *filename, File::OpenMode mode) {
|
||||||
if (myProp.openFile != NULL) {
|
// It should never happen that openFile is NULL in our use case.
|
||||||
return myProp.openFile(myProp.userData, filename, mode);
|
// Just to cover the case where something is horrible wrong we
|
||||||
}
|
// use an assert here.
|
||||||
char pathBuf[2048];
|
assert(myProp.openFile != NULL);
|
||||||
if (myProp.baseDir != NULL) {
|
return myProp.openFile(myProp.userData, filename, mode);
|
||||||
strcpy(&pathBuf[0], myProp.baseDir);
|
|
||||||
strcat(&pathBuf[0], filename);
|
|
||||||
filename = pathBuf;
|
|
||||||
}
|
|
||||||
ANSIFile *file = new ANSIFile();
|
|
||||||
if (!file->open(filename, mode)) {
|
|
||||||
delete file;
|
|
||||||
return NULL;
|
|
||||||
}
|
|
||||||
return file;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void Synth::closeFile(File *file) {
|
void Synth::closeFile(File *file) {
|
||||||
@ -326,7 +316,7 @@ bool Synth::initPCMList(Bit16u mapAddress, Bit16u count) {
|
|||||||
// The number below is confirmed to a reasonable degree of accuracy on CM-32L
|
// The number below is confirmed to a reasonable degree of accuracy on CM-32L
|
||||||
double STANDARDFREQ = 442.0;
|
double STANDARDFREQ = 442.0;
|
||||||
float rTune = (float)(STANDARDFREQ * pow(2.0, (0x5000 - rTuneOffset) / 4056.0 - 9.0 / 12.0));
|
float rTune = (float)(STANDARDFREQ * pow(2.0, (0x5000 - rTuneOffset) / 4056.0 - 9.0 / 12.0));
|
||||||
//printDebug("%f,%d,%d", pTune, tps[i].pitchCoarse, tps[i].pitchFine);
|
//printDebug("%f,%d,%d", (double)pTune, tps[i].pitchCoarse, tps[i].pitchFine);
|
||||||
if (rAddr + rLen > pcmROMSize) {
|
if (rAddr + rLen > pcmROMSize) {
|
||||||
printDebug("Control ROM error: Wave map entry %d points to invalid PCM address 0x%04X, length 0x%04X", i, rAddr, rLen);
|
printDebug("Control ROM error: Wave map entry %d points to invalid PCM address 0x%04X, length 0x%04X", i, rAddr, rLen);
|
||||||
return false;
|
return false;
|
||||||
@ -347,7 +337,8 @@ bool Synth::initRhythmTimbre(int timbreNum, const Bit8u *mem, unsigned int memLe
|
|||||||
memcpy(&timbre->common, mem, 14);
|
memcpy(&timbre->common, mem, 14);
|
||||||
unsigned int memPos = 14;
|
unsigned int memPos = 14;
|
||||||
char drumname[11];
|
char drumname[11];
|
||||||
Common::strlcpy(drumname, timbre->common.name, 11);
|
memset(drumname, 0, 11);
|
||||||
|
memcpy(drumname, timbre->common.name, 10);
|
||||||
for (int t = 0; t < 4; t++) {
|
for (int t = 0; t < 4; t++) {
|
||||||
if (((timbre->common.pmute >> t) & 0x1) == 0x1) {
|
if (((timbre->common.pmute >> t) & 0x1) == 0x1) {
|
||||||
if (memPos + 58 >= memLen) {
|
if (memPos + 58 >= memLen) {
|
||||||
@ -1030,7 +1021,7 @@ bool Synth::refreshSystem() {
|
|||||||
// The LAPC-I documentation claims a range of 427.5Hz-452.6Hz (similar to what we have here)
|
// The LAPC-I documentation claims a range of 427.5Hz-452.6Hz (similar to what we have here)
|
||||||
// The MT-32 documentation claims a range of 432.1Hz-457.6Hz
|
// The MT-32 documentation claims a range of 432.1Hz-457.6Hz
|
||||||
masterTune = 440.0f * powf(2.0f, (mt32ram.system.masterTune - 64.0f) / (128.0f * 12.0f));
|
masterTune = 440.0f * powf(2.0f, (mt32ram.system.masterTune - 64.0f) / (128.0f * 12.0f));
|
||||||
printDebug(" Master Tune: %f", masterTune);
|
printDebug(" Master Tune: %f", (double)masterTune);
|
||||||
printDebug(" Reverb: mode=%d, time=%d, level=%d", mt32ram.system.reverbMode, mt32ram.system.reverbTime, mt32ram.system.reverbLevel);
|
printDebug(" Reverb: mode=%d, time=%d, level=%d", mt32ram.system.reverbMode, mt32ram.system.reverbTime, mt32ram.system.reverbLevel);
|
||||||
report(ReportType_newReverbMode, &mt32ram.system.reverbMode);
|
report(ReportType_newReverbMode, &mt32ram.system.reverbMode);
|
||||||
report(ReportType_newReverbTime, &mt32ram.system.reverbTime);
|
report(ReportType_newReverbTime, &mt32ram.system.reverbTime);
|
@ -329,7 +329,7 @@ void Tables::initMT32ConstantTables(Synth *synth) {
|
|||||||
tdist = (lf - 25.0f) / 25.0f;
|
tdist = (lf - 25.0f) / 25.0f;
|
||||||
padjtable[lf] = 1.0f - tdist;
|
padjtable[lf] = 1.0f - tdist;
|
||||||
}
|
}
|
||||||
//synth->printDebug("lf %d = padj %f", lf, padjtable[lf]);
|
//synth->printDebug("lf %d = padj %f", lf, (double)padjtable[lf]);
|
||||||
}
|
}
|
||||||
|
|
||||||
float lfp, depf, finalval, tlf;
|
float lfp, depf, finalval, tlf;
|
||||||
@ -352,7 +352,7 @@ void Tables::initMT32ConstantTables(Synth *synth) {
|
|||||||
pval = (int)finalval;
|
pval = (int)finalval;
|
||||||
|
|
||||||
pitchEnvVal[lf][depat] = pval;
|
pitchEnvVal[lf][depat] = pval;
|
||||||
//synth->printDebug("lf %d depat %d pval %d tlf %f lfp %f", lf,depat,pval, tlf, lfp);
|
//synth->printDebug("lf %d depat %d pval %d tlf %f lfp %f", lf,depat,pval, (double)tlf, (double)lfp);
|
||||||
} else {
|
} else {
|
||||||
pitchEnvVal[lf][depat] = 4096;
|
pitchEnvVal[lf][depat] = 4096;
|
||||||
//synth->printDebug("lf %d depat %d pval 4096", lf, depat);
|
//synth->printDebug("lf %d depat %d pval 4096", lf, depat);
|
||||||
@ -402,7 +402,7 @@ void Tables::initMT32ConstantTables(Synth *synth) {
|
|||||||
tvaBiasMult[lf][distval] = (int)(dval * 256.0f);
|
tvaBiasMult[lf][distval] = (int)(dval * 256.0f);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
//synth->printDebug("Ampbias lf %d distval %d = %f (%x) %f", lf, distval, dval, tvaBiasMult[lf][distval],amplog);
|
//synth->printDebug("Ampbias lf %d distval %d = %f (%x) %f", lf, distval, (double)dval, tvaBiasMult[lf][distval],(double)amplog);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -430,7 +430,7 @@ void Tables::initMT32ConstantTables(Synth *synth) {
|
|||||||
tvfBiasMult[lf][distval] = (int)(dval * 256.0f);
|
tvfBiasMult[lf][distval] = (int)(dval * 256.0f);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
//synth->printDebug("Fbias lf %d distval %d = %f (%x) %f", lf, distval, dval, tvfBiasMult[lf][distval],amplog);
|
//synth->printDebug("Fbias lf %d distval %d = %f (%x) %f", lf, distval, (double)dval, tvfBiasMult[lf][distval],(double)amplog);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -471,7 +471,7 @@ static void initDep(KeyLookup *keyLookup, float f) {
|
|||||||
keyLookup->envTimeMult[dep] = (int)(ff * 256.0f);
|
keyLookup->envTimeMult[dep] = (int)(ff * 256.0f);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
//synth->printDebug("F %f d1 %x d2 %x d3 %x d4 %x d5 %x", f, noteLookup->fildepTable[0], noteLookup->fildepTable[1], noteLookup->fildepTable[2], noteLookup->fildepTable[3], noteLookup->fildepTable[4]);
|
//synth->printDebug("F %f d1 %x d2 %x d3 %x d4 %x d5 %x", (double)f, noteLookup->fildepTable[0], noteLookup->fildepTable[1], noteLookup->fildepTable[2], noteLookup->fildepTable[3], noteLookup->fildepTable[4]);
|
||||||
}
|
}
|
||||||
|
|
||||||
Bit16s Tables::clampWF(Synth *synth, const char *n, float ampVal, double input) {
|
Bit16s Tables::clampWF(Synth *synth, const char *n, float ampVal, double input) {
|
||||||
@ -586,8 +586,8 @@ File *Tables::initNote(Synth *synth, NoteLookup *noteLookup, float note, float r
|
|||||||
|
|
||||||
initSaw(noteLookup, noteLookup->div2);
|
initSaw(noteLookup, noteLookup->div2);
|
||||||
|
|
||||||
//synth->printDebug("Note %f; freq=%f, div=%f", note, freq, rate / freq);
|
//synth->printDebug("Note %f; freq=%f, div=%f", (double)note, (double)freq, (double) rate / freq);
|
||||||
file = initWave(synth, noteLookup, (const float)WGAMP, div2, file);
|
file = initWave(synth, noteLookup, WGAMP, div2, file);
|
||||||
|
|
||||||
// Create the pitch tables
|
// Create the pitch tables
|
||||||
if (noteLookup->wavTable == NULL)
|
if (noteLookup->wavTable == NULL)
|
||||||
@ -730,7 +730,7 @@ Tables::Tables() {
|
|||||||
|
|
||||||
bool Tables::init(Synth *synth, PCMWaveEntry *pcmWaves, float sampleRate, float masterTune) {
|
bool Tables::init(Synth *synth, PCMWaveEntry *pcmWaves, float sampleRate, float masterTune) {
|
||||||
if (sampleRate <= 0.0f) {
|
if (sampleRate <= 0.0f) {
|
||||||
synth->printDebug("Bad sampleRate (%f <= 0.0f)", sampleRate);
|
synth->printDebug("Bad sampleRate (%f <= 0.0f)", (double)sampleRate);
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
if (initialisedSampleRate == 0.0f) {
|
if (initialisedSampleRate == 0.0f) {
|
@ -41,10 +41,6 @@
|
|||||||
namespace OPL {
|
namespace OPL {
|
||||||
namespace DOSBox {
|
namespace DOSBox {
|
||||||
|
|
||||||
#ifndef PI
|
|
||||||
#define PI 3.14159265358979323846
|
|
||||||
#endif
|
|
||||||
|
|
||||||
namespace DBOPL {
|
namespace DBOPL {
|
||||||
|
|
||||||
#define OPLRATE ((double)(14318180.0 / 288.0))
|
#define OPLRATE ((double)(14318180.0 / 288.0))
|
||||||
@ -224,7 +220,7 @@ static inline Bits MakeVolume( Bitu wave, Bitu volume ) {
|
|||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
return (sig >> exp);
|
return (sig >> exp);
|
||||||
};
|
}
|
||||||
|
|
||||||
static Bits DB_FASTCALL WaveForm0( Bitu i, Bitu volume ) {
|
static Bits DB_FASTCALL WaveForm0( Bitu i, Bitu volume ) {
|
||||||
Bits neg = 0 - (( i >> 9) & 1);//Create ~0 or 0
|
Bits neg = 0 - (( i >> 9) & 1);//Create ~0 or 0
|
||||||
@ -418,7 +414,7 @@ Bits Operator::TemplateVolume( ) {
|
|||||||
}
|
}
|
||||||
//In sustain phase, but not sustaining, do regular release
|
//In sustain phase, but not sustaining, do regular release
|
||||||
case RELEASE:
|
case RELEASE:
|
||||||
vol += RateForward( releaseAdd );;
|
vol += RateForward( releaseAdd );
|
||||||
if ( GCC_UNLIKELY(vol >= ENV_MAX) ) {
|
if ( GCC_UNLIKELY(vol >= ENV_MAX) ) {
|
||||||
volume = ENV_MAX;
|
volume = ENV_MAX;
|
||||||
SetState( OFF );
|
SetState( OFF );
|
||||||
@ -576,7 +572,7 @@ INLINE Bits Operator::GetWave( Bitu index, Bitu vol ) {
|
|||||||
return (waveBase[ index & waveMask ] * MulTable[ vol >> ENV_EXTRA ]) >> MUL_SH;
|
return (waveBase[ index & waveMask ] * MulTable[ vol >> ENV_EXTRA ]) >> MUL_SH;
|
||||||
#elif ( DBOPL_WAVE == WAVE_TABLELOG )
|
#elif ( DBOPL_WAVE == WAVE_TABLELOG )
|
||||||
Bit32s wave = waveBase[ index & waveMask ];
|
Bit32s wave = waveBase[ index & waveMask ];
|
||||||
Bit32u total = ( wave & 0x7fff ) + vol << ( 3 - ENV_EXTRA );
|
Bit32u total = ( wave & 0x7fff ) + ( vol << ( 3 - ENV_EXTRA ) );
|
||||||
Bit32s sig = ExpTable[ total & 0xff ];
|
Bit32s sig = ExpTable[ total & 0xff ];
|
||||||
Bit32u exp = total >> 8;
|
Bit32u exp = total >> 8;
|
||||||
Bit32s neg = wave >> 16;
|
Bit32s neg = wave >> 16;
|
||||||
@ -1392,7 +1388,7 @@ void InitTables( void ) {
|
|||||||
//Add 0.5 for the trunc rounding of the integer cast
|
//Add 0.5 for the trunc rounding of the integer cast
|
||||||
//Do a PI sinetable instead of the original 0.5 PI
|
//Do a PI sinetable instead of the original 0.5 PI
|
||||||
for ( int i = 0; i < 512; i++ ) {
|
for ( int i = 0; i < 512; i++ ) {
|
||||||
SinTable[i] = (Bit16s)( 0.5 - log10( sin( (i + 0.5) * (PI / 512.0) ) ) / log10(2.0)*256 );
|
SinTable[i] = (Bit16s)( 0.5 - log10( sin( (i + 0.5) * (M_PI / 512.0) ) ) / log10(2.0)*256 );
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
#if ( DBOPL_WAVE == WAVE_TABLEMUL )
|
#if ( DBOPL_WAVE == WAVE_TABLEMUL )
|
||||||
@ -1406,7 +1402,7 @@ void InitTables( void ) {
|
|||||||
|
|
||||||
//Sine Wave Base
|
//Sine Wave Base
|
||||||
for ( int i = 0; i < 512; i++ ) {
|
for ( int i = 0; i < 512; i++ ) {
|
||||||
WaveTable[ 0x0200 + i ] = (Bit16s)(sin( (i + 0.5) * (PI / 512.0) ) * 4084);
|
WaveTable[ 0x0200 + i ] = (Bit16s)(sin( (i + 0.5) * (M_PI / 512.0) ) * 4084);
|
||||||
WaveTable[ 0x0000 + i ] = -WaveTable[ 0x200 + i ];
|
WaveTable[ 0x0000 + i ] = -WaveTable[ 0x200 + i ];
|
||||||
}
|
}
|
||||||
//Exponential wave
|
//Exponential wave
|
||||||
@ -1418,7 +1414,7 @@ void InitTables( void ) {
|
|||||||
#if ( DBOPL_WAVE == WAVE_TABLELOG )
|
#if ( DBOPL_WAVE == WAVE_TABLELOG )
|
||||||
//Sine Wave Base
|
//Sine Wave Base
|
||||||
for ( int i = 0; i < 512; i++ ) {
|
for ( int i = 0; i < 512; i++ ) {
|
||||||
WaveTable[ 0x0200 + i ] = (Bit16s)( 0.5 - log10( sin( (i + 0.5) * (PI / 512.0) ) ) / log10(2.0)*256 );
|
WaveTable[ 0x0200 + i ] = (Bit16s)( 0.5 - log10( sin( (i + 0.5) * (M_PI / 512.0) ) ) / log10(2.0)*256 );
|
||||||
WaveTable[ 0x0000 + i ] = ((Bit16s)0x8000) | WaveTable[ 0x200 + i];
|
WaveTable[ 0x0000 + i ] = ((Bit16s)0x8000) | WaveTable[ 0x200 + i];
|
||||||
}
|
}
|
||||||
//Exponential wave
|
//Exponential wave
|
@ -34,7 +34,7 @@
|
|||||||
|
|
||||||
#ifndef DISABLE_DOSBOX_OPL
|
#ifndef DISABLE_DOSBOX_OPL
|
||||||
|
|
||||||
#include "sound/fmopl.h"
|
#include "audio/fmopl.h"
|
||||||
|
|
||||||
namespace OPL {
|
namespace OPL {
|
||||||
namespace DOSBox {
|
namespace DOSBox {
|
@ -694,7 +694,7 @@ static int OPLOpenTable(void) {
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
/* make total level table */
|
/* make total level table */
|
||||||
for (t = 0; t < EG_ENT - 1 ; t++) {
|
for (t = 0; t < EG_ENT - 1; t++) {
|
||||||
rate = ((1 << TL_BITS) - 1) / pow(10.0, EG_STEP * t / 20); /* dB -> voltage */
|
rate = ((1 << TL_BITS) - 1) / pow(10.0, EG_STEP * t / 20); /* dB -> voltage */
|
||||||
TL_TABLE[ t] = (int)rate;
|
TL_TABLE[ t] = (int)rate;
|
||||||
TL_TABLE[TL_MAX + t] = -TL_TABLE[t];
|
TL_TABLE[TL_MAX + t] = -TL_TABLE[t];
|
||||||
@ -708,7 +708,7 @@ static int OPLOpenTable(void) {
|
|||||||
/* degree 0 = degree 180 = off */
|
/* degree 0 = degree 180 = off */
|
||||||
SIN_TABLE[0] = SIN_TABLE[SIN_ENT /2 ] = &TL_TABLE[EG_ENT - 1];
|
SIN_TABLE[0] = SIN_TABLE[SIN_ENT /2 ] = &TL_TABLE[EG_ENT - 1];
|
||||||
for (s = 1;s <= SIN_ENT / 4; s++) {
|
for (s = 1;s <= SIN_ENT / 4; s++) {
|
||||||
pom = sin(2 * LOCAL_PI * s / SIN_ENT); /* sin */
|
pom = sin(2 * M_PI * s / SIN_ENT); /* sin */
|
||||||
pom = 20 * log10(1 / pom); /* decibel */
|
pom = 20 * log10(1 / pom); /* decibel */
|
||||||
j = int(pom / EG_STEP); /* TL_TABLE steps */
|
j = int(pom / EG_STEP); /* TL_TABLE steps */
|
||||||
|
|
||||||
@ -739,14 +739,14 @@ static int OPLOpenTable(void) {
|
|||||||
ENV_CURVE[EG_OFF >> ENV_BITS]= EG_ENT - 1;
|
ENV_CURVE[EG_OFF >> ENV_BITS]= EG_ENT - 1;
|
||||||
/* make LFO ams table */
|
/* make LFO ams table */
|
||||||
for (i=0; i < AMS_ENT; i++) {
|
for (i=0; i < AMS_ENT; i++) {
|
||||||
pom = (1.0 + sin(2 * LOCAL_PI * i / AMS_ENT)) / 2; /* sin */
|
pom = (1.0 + sin(2 * M_PI * i / AMS_ENT)) / 2; /* sin */
|
||||||
AMS_TABLE[i] = (int)((1.0 / EG_STEP) * pom); /* 1dB */
|
AMS_TABLE[i] = (int)((1.0 / EG_STEP) * pom); /* 1dB */
|
||||||
AMS_TABLE[AMS_ENT + i] = (int)((4.8 / EG_STEP) * pom); /* 4.8dB */
|
AMS_TABLE[AMS_ENT + i] = (int)((4.8 / EG_STEP) * pom); /* 4.8dB */
|
||||||
}
|
}
|
||||||
/* make LFO vibrate table */
|
/* make LFO vibrate table */
|
||||||
for (i=0; i < VIB_ENT; i++) {
|
for (i=0; i < VIB_ENT; i++) {
|
||||||
/* 100cent = 1seminote = 6% ?? */
|
/* 100cent = 1seminote = 6% ?? */
|
||||||
pom = (double)VIB_RATE * 0.06 * sin(2 * LOCAL_PI * i / VIB_ENT); /* +-100sect step */
|
pom = (double)VIB_RATE * 0.06 * sin(2 * M_PI * i / VIB_ENT); /* +-100sect step */
|
||||||
VIB_TABLE[i] = (int)(VIB_RATE + (pom * 0.07)); /* +- 7cent */
|
VIB_TABLE[i] = (int)(VIB_RATE + (pom * 0.07)); /* +- 7cent */
|
||||||
VIB_TABLE[VIB_ENT + i] = (int)(VIB_RATE + (pom * 0.14)); /* +-14cent */
|
VIB_TABLE[VIB_ENT + i] = (int)(VIB_RATE + (pom * 0.14)); /* +-14cent */
|
||||||
}
|
}
|
||||||
@ -1082,10 +1082,10 @@ void OPLResetChip(FM_OPL *OPL) {
|
|||||||
for (i = 0xff; i >= 0x20; i--)
|
for (i = 0xff; i >= 0x20; i--)
|
||||||
OPLWriteReg(OPL,i,0);
|
OPLWriteReg(OPL,i,0);
|
||||||
/* reset OPerator parameter */
|
/* reset OPerator parameter */
|
||||||
for (c = 0; c < OPL->max_ch ;c++ ) {
|
for (c = 0; c < OPL->max_ch; c++) {
|
||||||
OPL_CH *CH = &OPL->P_CH[c];
|
OPL_CH *CH = &OPL->P_CH[c];
|
||||||
/* OPL->P_CH[c].PAN = OPN_CENTER; */
|
/* OPL->P_CH[c].PAN = OPN_CENTER; */
|
||||||
for (s = 0; s < 2; s++ ) {
|
for (s = 0; s < 2; s++) {
|
||||||
/* wave table */
|
/* wave table */
|
||||||
CH->SLOT[s].wavetable = &SIN_TABLE[0];
|
CH->SLOT[s].wavetable = &SIN_TABLE[0];
|
||||||
/* CH->SLOT[s].evm = ENV_MOD_RR; */
|
/* CH->SLOT[s].evm = ENV_MOD_RR; */
|
@ -33,7 +33,7 @@
|
|||||||
#include "common/util.h"
|
#include "common/util.h"
|
||||||
#include "common/random.h"
|
#include "common/random.h"
|
||||||
|
|
||||||
#include "sound/fmopl.h"
|
#include "audio/fmopl.h"
|
||||||
|
|
||||||
namespace OPL {
|
namespace OPL {
|
||||||
namespace MAME {
|
namespace MAME {
|
@ -23,8 +23,8 @@
|
|||||||
*
|
*
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#include "sound/softsynth/pcspk.h"
|
#include "audio/softsynth/pcspk.h"
|
||||||
#include "sound/null.h"
|
#include "audio/null.h"
|
||||||
|
|
||||||
namespace Audio {
|
namespace Audio {
|
||||||
|
|
||||||
@ -109,7 +109,7 @@ int8 PCSpeaker::generateSine(uint32 x, uint32 oscLength) {
|
|||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
// TODO: Maybe using a look-up-table would be better?
|
// TODO: Maybe using a look-up-table would be better?
|
||||||
return CLIP<int16>((int16) (128 * sin(2.0 * LOCAL_PI * x / oscLength)), -128, 127);
|
return CLIP<int16>((int16) (128 * sin(2.0 * M_PI * x / oscLength)), -128, 127);
|
||||||
}
|
}
|
||||||
|
|
||||||
int8 PCSpeaker::generateSaw(uint32 x, uint32 oscLength) {
|
int8 PCSpeaker::generateSaw(uint32 x, uint32 oscLength) {
|
@ -25,8 +25,8 @@
|
|||||||
#ifndef SOUND_SOFTSYNTH_PCSPK_H
|
#ifndef SOUND_SOFTSYNTH_PCSPK_H
|
||||||
#define SOUND_SOFTSYNTH_PCSPK_H
|
#define SOUND_SOFTSYNTH_PCSPK_H
|
||||||
|
|
||||||
#include "sound/audiostream.h"
|
#include "audio/audiostream.h"
|
||||||
#include "sound/mixer.h"
|
#include "audio/mixer.h"
|
||||||
#include "common/mutex.h"
|
#include "common/mutex.h"
|
||||||
|
|
||||||
namespace Audio {
|
namespace Audio {
|
@ -31,7 +31,7 @@
|
|||||||
#ifndef DISABLE_SID
|
#ifndef DISABLE_SID
|
||||||
|
|
||||||
#include "sid.h"
|
#include "sid.h"
|
||||||
#include "sound/null.h"
|
#include "audio/null.h"
|
||||||
#include <math.h>
|
#include <math.h>
|
||||||
|
|
||||||
namespace Resid {
|
namespace Resid {
|
||||||
@ -1423,4 +1423,34 @@ int SID::clock(cycle_count& delta_t, short* buf, int n, int interleave) {
|
|||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Plugin interface
|
||||||
|
// (This can only create a null driver since C64 audio support is not part of the
|
||||||
|
// midi driver architecture. But we need the plugin for the options menu in the launcher
|
||||||
|
// and for MidiDriver::detectDevice() which is more or less used by all engines.)
|
||||||
|
|
||||||
|
class C64MusicPlugin : public NullMusicPlugin {
|
||||||
|
public:
|
||||||
|
const char *getName() const {
|
||||||
|
return _s("C64 Audio Emulator");
|
||||||
|
}
|
||||||
|
|
||||||
|
const char *getId() const {
|
||||||
|
return "C64";
|
||||||
|
}
|
||||||
|
|
||||||
|
MusicDevices getDevices() const;
|
||||||
|
};
|
||||||
|
|
||||||
|
MusicDevices C64MusicPlugin::getDevices() const {
|
||||||
|
MusicDevices devices;
|
||||||
|
devices.push_back(MusicDevice(this, "", MT_C64));
|
||||||
|
return devices;
|
||||||
|
}
|
||||||
|
|
||||||
|
//#if PLUGIN_ENABLED_DYNAMIC(C64)
|
||||||
|
//REGISTER_PLUGIN_DYNAMIC(C64, PLUGIN_TYPE_MUSIC, C64MusicPlugin);
|
||||||
|
//#else
|
||||||
|
REGISTER_PLUGIN_STATIC(C64, PLUGIN_TYPE_MUSIC, C64MusicPlugin);
|
||||||
|
//#endif
|
||||||
|
|
||||||
#endif
|
#endif
|
@ -24,9 +24,9 @@
|
|||||||
|
|
||||||
#include <math.h>
|
#include <math.h>
|
||||||
|
|
||||||
#include "sound/softsynth/ym2612.h"
|
#include "audio/softsynth/ym2612.h"
|
||||||
#include "common/util.h"
|
#include "common/util.h"
|
||||||
#include "sound/musicplugin.h"
|
#include "audio/musicplugin.h"
|
||||||
#include "common/translation.h"
|
#include "common/translation.h"
|
||||||
|
|
||||||
////////////////////////////////////////
|
////////////////////////////////////////
|
||||||
@ -677,14 +677,14 @@ void MidiDriver_YM2612::createLookupTables() {
|
|||||||
int i;
|
int i;
|
||||||
sintbl = new int [2048];
|
sintbl = new int [2048];
|
||||||
for (i = 0; i < 2048; i++)
|
for (i = 0; i < 2048; i++)
|
||||||
sintbl[i] = (int)(0xffff * sin(i/2048.0*2.0*LOCAL_PI));
|
sintbl[i] = (int)(0xffff * sin(i/2048.0 * 2.0 * M_PI));
|
||||||
}
|
}
|
||||||
|
|
||||||
{
|
{
|
||||||
int i;
|
int i;
|
||||||
powtbl = new int [1025];
|
powtbl = new int [1025];
|
||||||
for (i = 0; i <= 1024; i++)
|
for (i = 0; i <= 1024; i++)
|
||||||
powtbl[i] = (int)(0x10000 * pow(2.0, (i-512)/512.0));
|
powtbl[i] = (int)(0x10000 * pow(2.0, (i - 512) / 512.0));
|
||||||
}
|
}
|
||||||
|
|
||||||
{
|
{
|
@ -27,7 +27,7 @@
|
|||||||
|
|
||||||
#include "common/sys.h"
|
#include "common/sys.h"
|
||||||
|
|
||||||
#include "sound/softsynth/emumidi.h"
|
#include "audio/softsynth/emumidi.h"
|
||||||
|
|
||||||
////////////////////////////////////////
|
////////////////////////////////////////
|
||||||
//
|
//
|
@ -23,7 +23,7 @@
|
|||||||
*
|
*
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#include "sound/timestamp.h"
|
#include "audio/timestamp.h"
|
||||||
#include "common/algorithm.h"
|
#include "common/algorithm.h"
|
||||||
|
|
||||||
namespace Audio {
|
namespace Audio {
|
||||||
@ -148,21 +148,9 @@ Timestamp Timestamp::addMsecs(int ms) const {
|
|||||||
}
|
}
|
||||||
|
|
||||||
void Timestamp::addIntern(const Timestamp &ts) {
|
void Timestamp::addIntern(const Timestamp &ts) {
|
||||||
|
assert(_framerate == ts._framerate);
|
||||||
_secs += ts._secs;
|
_secs += ts._secs;
|
||||||
|
_numFrames += ts._numFrames;
|
||||||
if (_framerate == ts._framerate) {
|
|
||||||
_numFrames += ts._numFrames;
|
|
||||||
} else {
|
|
||||||
// We need to multiply by the quotient of the two framerates.
|
|
||||||
// We cancel the GCD in this fraction to reduce the risk of
|
|
||||||
// overflows.
|
|
||||||
const uint g = Common::gcd(_framerate, ts._framerate);
|
|
||||||
const uint p = _framerate / g;
|
|
||||||
const uint q = ts._framerate / g;
|
|
||||||
|
|
||||||
_framerate *= q;
|
|
||||||
_numFrames = _numFrames * q + ts._numFrames * p;
|
|
||||||
}
|
|
||||||
|
|
||||||
normalize();
|
normalize();
|
||||||
}
|
}
|
||||||
@ -187,24 +175,6 @@ Timestamp Timestamp::operator-(const Timestamp &ts) const {
|
|||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
|
||||||
Timestamp &Timestamp::operator+=(const Timestamp &ts) {
|
|
||||||
addIntern(ts);
|
|
||||||
return *this;
|
|
||||||
}
|
|
||||||
|
|
||||||
Timestamp &Timestamp::operator-=(const Timestamp &ts) {
|
|
||||||
addIntern(-ts);
|
|
||||||
return *this;
|
|
||||||
}
|
|
||||||
*/
|
|
||||||
|
|
||||||
/*
|
|
||||||
int Timestamp::frameDiff(const Timestamp &ts) const {
|
|
||||||
return (*this - ts).totalNumberOfFrames();
|
|
||||||
}
|
|
||||||
*/
|
|
||||||
|
|
||||||
int Timestamp::frameDiff(const Timestamp &ts) const {
|
int Timestamp::frameDiff(const Timestamp &ts) const {
|
||||||
|
|
||||||
int delta = 0;
|
int delta = 0;
|
@ -31,55 +31,62 @@
|
|||||||
namespace Audio {
|
namespace Audio {
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Timestamps allow measuring times with a sub-millisecond granularity,
|
* Timestamps allow specifying points in time and measuring time intervals
|
||||||
* and without rounding losses. This is achieved by measuring frames
|
* with a sub-millisecond granularity.
|
||||||
* instead of milliseconds: Specify an initial time in milliseconds
|
*
|
||||||
* plus framerate (in frames per second).
|
* When dealing with audio and video decoding, it is often necessary to
|
||||||
|
* measure time (intervals) in terms of frames, relative to a fixed
|
||||||
|
* frame rate (that is, a fixed number of frames per seconds). For
|
||||||
|
* example, in a typical video there are 24 frames per second, and in a
|
||||||
|
* typical sound there are 44100 frames (i.e. samples for mono sound
|
||||||
|
* and pairs of samples for stereo) per second.
|
||||||
|
*
|
||||||
|
* At the same time, the system clock provided by ScummVM measures time
|
||||||
|
* in milliseconds. For syncing purposes and other reasons, it is often
|
||||||
|
* necessary to convert between and compare time measures given on the
|
||||||
|
* one hand as a frame count, and on the other hand as a number of
|
||||||
|
* milliseconds.
|
||||||
|
*
|
||||||
|
* If handled carelessly, this can introduce rounding errors that
|
||||||
|
* quickly accumulate, resulting in user noticeable disturbance, such as
|
||||||
|
* audio and video running out of sync. E.g. a typical approach is to
|
||||||
|
* measure all time in milliseconds. But with a frame rate of 24 frames
|
||||||
|
* per second, one frame is 41.66666... milliseconds long. On the other
|
||||||
|
* hand, if measuring in frames, then similar rounding issue occur when
|
||||||
|
* converting from milliseconds to frames.
|
||||||
|
*
|
||||||
|
* One solution is to use floating point arithmetic to compute with
|
||||||
|
* fractional frames resp. (milli)seconds. This has other undesirable
|
||||||
|
* side effects; foremost, some platforms ScummVM runs on still have
|
||||||
|
* only limited (and slow) floating point support.
|
||||||
|
*
|
||||||
|
* This class provides an alternate solution: It stores time in terms of
|
||||||
|
* frames, but with a twist: Client code can specify arbitrary
|
||||||
|
* (integral) framerates; but internally, Timestamp modifies the
|
||||||
|
* framerate to be a multiple of 1000. This way, both numbers of frames
|
||||||
|
* (relative to the original framerate) as well as milliseconds can be
|
||||||
|
* represented as integers. This change is completely hidden from the
|
||||||
|
* user, however.
|
||||||
|
*
|
||||||
|
* A Timestamp can be converted to a frame count or milliseconds at
|
||||||
|
* virtually no cost. Likewise, it is posible to compute the difference
|
||||||
|
* between two Timestamps in milliseconds or number of frames.
|
||||||
|
* Timestamps can be easily compared using regular comparison operators,
|
||||||
|
* resulting in nicely readable code; this is even possible for
|
||||||
|
* timestamps that are specified using different framerates.
|
||||||
|
* Client code can modify Timestamps by adding a number of frames
|
||||||
|
* to it, or adding a number of milliseconds. Adding negative amounts is
|
||||||
|
* also allowed, and a Timestamp can even represent a "negative time"
|
||||||
|
* (mainly useful when using the Timestamp to store a time interval).
|
||||||
*/
|
*/
|
||||||
class Timestamp {
|
class Timestamp {
|
||||||
protected:
|
|
||||||
/**
|
|
||||||
* The seconds part of this timestamp.
|
|
||||||
* The total time in seconds represented by this timestamp can be
|
|
||||||
* computed as follows:
|
|
||||||
* _secs + (double)_numFrames / _framerate
|
|
||||||
*/
|
|
||||||
int _secs;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* The number of frames which together with _secs encodes
|
|
||||||
* the timestamp. The total number of frames represented
|
|
||||||
* by this timestamp is computed as follows:
|
|
||||||
* _numFrames + _secs * _framerate
|
|
||||||
*
|
|
||||||
* This is always a value greater or equal to zero.
|
|
||||||
* The only reason this is an int and not an uint is to
|
|
||||||
* allow intermediate negative values.
|
|
||||||
*/
|
|
||||||
int _numFrames;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* The internal framerate, i.e. the number of frames per second.
|
|
||||||
* This is computed as the least common multiple of the framerate
|
|
||||||
* specified by the client code, and 1000.
|
|
||||||
* This way, we ensure that we can store both frames and
|
|
||||||
* milliseconds without any rounding losses.
|
|
||||||
*/
|
|
||||||
uint _framerate;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Factor by which the original framerate specified by the client
|
|
||||||
* code was multipled to obtain the internal _framerate value.
|
|
||||||
*/
|
|
||||||
uint _framerateFactor;
|
|
||||||
|
|
||||||
public:
|
public:
|
||||||
/**
|
/**
|
||||||
* Set up a timestamp with a given time and framerate.
|
* Set up a timestamp with a given time and framerate.
|
||||||
* @param msecs starting time in milliseconds
|
* @param msecs starting time in milliseconds
|
||||||
* @param framerate number of frames per second (must be > 0)
|
* @param framerate number of frames per second (must be > 0)
|
||||||
*/
|
*/
|
||||||
Timestamp(uint msecs, uint framerate);
|
Timestamp(uint msecs = 0, uint framerate = 1);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Set up a timestamp with a given time, frames and framerate.
|
* Set up a timestamp with a given time, frames and framerate.
|
||||||
@ -126,11 +133,17 @@ public:
|
|||||||
// unary minus
|
// unary minus
|
||||||
Timestamp operator-() const;
|
Timestamp operator-() const;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Compute the sum of two timestamps. This is only
|
||||||
|
* allowed if they use the same framerate.
|
||||||
|
*/
|
||||||
Timestamp operator+(const Timestamp &ts) const;
|
Timestamp operator+(const Timestamp &ts) const;
|
||||||
Timestamp operator-(const Timestamp &ts) const;
|
|
||||||
|
|
||||||
// Timestamp &operator+=(const Timestamp &ts);
|
/**
|
||||||
// Timestamp &operator-=(const Timestamp &ts);
|
* Compute the difference between two timestamps. This is only
|
||||||
|
* allowed if they use the same framerate.
|
||||||
|
*/
|
||||||
|
Timestamp operator-(const Timestamp &ts) const;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Computes the number of frames between this timestamp and ts.
|
* Computes the number of frames between this timestamp and ts.
|
||||||
@ -176,7 +189,6 @@ public:
|
|||||||
inline uint framerate() const { return _framerate / _framerateFactor; }
|
inline uint framerate() const { return _framerate / _framerateFactor; }
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Compare this timestamp to another one and return
|
* Compare this timestamp to another one and return
|
||||||
* a value similar to strcmp.
|
* a value similar to strcmp.
|
||||||
@ -193,6 +205,44 @@ protected:
|
|||||||
* Add another timestamp to this one and normalize the result.
|
* Add another timestamp to this one and normalize the result.
|
||||||
*/
|
*/
|
||||||
void addIntern(const Timestamp &ts);
|
void addIntern(const Timestamp &ts);
|
||||||
|
|
||||||
|
protected:
|
||||||
|
/**
|
||||||
|
* The seconds part of this timestamp.
|
||||||
|
* The total time in seconds represented by this timestamp can be
|
||||||
|
* computed as follows:
|
||||||
|
* _secs + (double)_numFrames / _framerate
|
||||||
|
*/
|
||||||
|
int _secs;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The number of frames which together with _secs encodes the
|
||||||
|
* timestamp. The total number of *internal* frames represented
|
||||||
|
* by this timestamp can be computed as follows:
|
||||||
|
* _numFrames + _secs * _framerate
|
||||||
|
* To obtain the number of frames with respect to the original
|
||||||
|
* framerate, this value has to be divided by _framerateFactor.
|
||||||
|
*
|
||||||
|
* This is always a value greater or equal to zero.
|
||||||
|
* The only reason this is an int and not an uint is to
|
||||||
|
* allow intermediate negative values.
|
||||||
|
*/
|
||||||
|
int _numFrames;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The internal framerate, i.e. the number of frames per second.
|
||||||
|
* This is computed as the least common multiple of the framerate
|
||||||
|
* specified by the client code, and 1000.
|
||||||
|
* This way, we ensure that we can store both frames and
|
||||||
|
* milliseconds without any rounding losses.
|
||||||
|
*/
|
||||||
|
uint _framerate;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Factor by which the original framerate specified by the client
|
||||||
|
* code was multipled to obtain the internal _framerate value.
|
||||||
|
*/
|
||||||
|
uint _framerateFactor;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
148
backends/audiocd/audiocd.h
Normal file
148
backends/audiocd/audiocd.h
Normal file
@ -0,0 +1,148 @@
|
|||||||
|
/* Residual - A 3D game interpreter
|
||||||
|
*
|
||||||
|
* Residual is the legal property of its developers, whose names
|
||||||
|
* are too numerous to list here. Please refer to the AUTHORS
|
||||||
|
* file distributed with this source distribution.
|
||||||
|
*
|
||||||
|
* 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., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
||||||
|
*
|
||||||
|
* $URL$
|
||||||
|
* $Id$
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef BACKENDS_AUDIOCD_ABSTRACT_H
|
||||||
|
#define BACKENDS_AUDIOCD_ABSTRACT_H
|
||||||
|
|
||||||
|
#include "common/sys.h"
|
||||||
|
#include "common/noncopyable.h"
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Abstract Audio CD manager class. Subclasses implement the actual
|
||||||
|
* functionality.
|
||||||
|
*/
|
||||||
|
class AudioCDManager : Common::NonCopyable {
|
||||||
|
public:
|
||||||
|
virtual ~AudioCDManager() {}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* A structure containing the current playback information
|
||||||
|
*/
|
||||||
|
struct Status {
|
||||||
|
bool playing;
|
||||||
|
int track;
|
||||||
|
int start;
|
||||||
|
int duration;
|
||||||
|
int numLoops;
|
||||||
|
int volume;
|
||||||
|
int balance;
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @name Emulated playback functions
|
||||||
|
* Engines should call these functions. Not all platforms
|
||||||
|
* support cd playback, and these functions should try to
|
||||||
|
* emulate it.
|
||||||
|
*/
|
||||||
|
//@{
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Start audio CD playback
|
||||||
|
* @param track the track to play.
|
||||||
|
* @param num_loops how often playback should be repeated (-1 = infinitely often).
|
||||||
|
* @param start_frame the frame at which playback should start (75 frames = 1 second).
|
||||||
|
* @param duration the number of frames to play.
|
||||||
|
* @param only_emulate determines if the track should be emulated only
|
||||||
|
*/
|
||||||
|
virtual void play(int track, int numLoops, int startFrame, int duration, bool only_emulate = false) = 0;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get if audio is being played.
|
||||||
|
* @return true if CD or emulated audio is playing
|
||||||
|
*/
|
||||||
|
virtual bool isPlaying() const = 0;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Set the audio volume
|
||||||
|
*/
|
||||||
|
virtual void setVolume(byte volume) = 0;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Set the speakers balance
|
||||||
|
*/
|
||||||
|
virtual void setBalance(int8 balance) = 0;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Stop CD or emulated audio playback.
|
||||||
|
*/
|
||||||
|
virtual void stop() = 0;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Update CD or emulated audio status.
|
||||||
|
*/
|
||||||
|
virtual void update() = 0;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get the playback status.
|
||||||
|
* @return a Status struct with playback data.
|
||||||
|
*/
|
||||||
|
virtual Status getStatus() const = 0;
|
||||||
|
|
||||||
|
//@}
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @name Real CD audio methods
|
||||||
|
* These functions should be called from the emulated
|
||||||
|
* ones if they can't emulate the audio playback.
|
||||||
|
*/
|
||||||
|
//@{
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Initialise the specified CD drive for audio playback.
|
||||||
|
* @param drive the drive id
|
||||||
|
* @return true if the CD drive was inited succesfully
|
||||||
|
*/
|
||||||
|
virtual bool openCD(int drive) = 0;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Poll CD status.
|
||||||
|
* @return true if CD audio is playing
|
||||||
|
*/
|
||||||
|
virtual bool pollCD() const = 0;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Start CD audio playback.
|
||||||
|
* @param track the track to play.
|
||||||
|
* @param num_loops how often playback should be repeated (-1 = infinitely often).
|
||||||
|
* @param start_frame the frame at which playback should start (75 frames = 1 second).
|
||||||
|
* @param duration the number of frames to play.
|
||||||
|
*/
|
||||||
|
virtual void playCD(int track, int num_loops, int start_frame, int duration) = 0;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Stop CD audio playback.
|
||||||
|
*/
|
||||||
|
virtual void stopCD() = 0;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Update CD audio status.
|
||||||
|
*/
|
||||||
|
virtual void updateCD() = 0;
|
||||||
|
|
||||||
|
//@}
|
||||||
|
};
|
||||||
|
|
||||||
|
#endif
|
@ -23,33 +23,24 @@
|
|||||||
*
|
*
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#include "sound/audiocd.h"
|
#include "backends/audiocd/default/default-audiocd.h"
|
||||||
#include "sound/audiostream.h"
|
#include "audio/audiostream.h"
|
||||||
#include "sound/decoders/mp3.h"
|
|
||||||
#include "sound/decoders/vorbis.h"
|
|
||||||
#include "sound/decoders/flac.h"
|
|
||||||
#include "engines/engine.h"
|
|
||||||
#include "common/util.h"
|
|
||||||
#include "common/system.h"
|
#include "common/system.h"
|
||||||
|
|
||||||
DECLARE_SINGLETON(Audio::AudioCDManager)
|
DefaultAudioCDManager::DefaultAudioCDManager() {
|
||||||
|
|
||||||
namespace Audio {
|
|
||||||
|
|
||||||
AudioCDManager::AudioCDManager() {
|
|
||||||
_cd.playing = false;
|
_cd.playing = false;
|
||||||
_cd.track = 0;
|
_cd.track = 0;
|
||||||
_cd.start = 0;
|
_cd.start = 0;
|
||||||
_cd.duration = 0;
|
_cd.duration = 0;
|
||||||
_cd.numLoops = 0;
|
_cd.numLoops = 0;
|
||||||
_cd.volume = Mixer::kMaxChannelVolume;
|
_cd.volume = Audio::Mixer::kMaxChannelVolume;
|
||||||
_cd.balance = 0;
|
_cd.balance = 0;
|
||||||
_mixer = g_system->getMixer();
|
_mixer = g_system->getMixer();
|
||||||
_emulating = false;
|
_emulating = false;
|
||||||
assert(_mixer);
|
assert(_mixer);
|
||||||
}
|
}
|
||||||
|
|
||||||
void AudioCDManager::play(int track, int numLoops, int startFrame, int duration, bool only_emulate) {
|
void DefaultAudioCDManager::play(int track, int numLoops, int startFrame, int duration, bool only_emulate) {
|
||||||
if (numLoops != 0 || startFrame != 0) {
|
if (numLoops != 0 || startFrame != 0) {
|
||||||
_cd.track = track;
|
_cd.track = track;
|
||||||
_cd.numLoops = numLoops;
|
_cd.numLoops = numLoops;
|
||||||
@ -65,14 +56,14 @@ void AudioCDManager::play(int track, int numLoops, int startFrame, int duration,
|
|||||||
Audio::SeekableAudioStream *stream = 0;
|
Audio::SeekableAudioStream *stream = 0;
|
||||||
|
|
||||||
for (int i = 0; !stream && i < 2; ++i)
|
for (int i = 0; !stream && i < 2; ++i)
|
||||||
stream = SeekableAudioStream::openStreamFile(trackName[i]);
|
stream = Audio::SeekableAudioStream::openStreamFile(trackName[i]);
|
||||||
|
|
||||||
// Stop any currently playing emulated track
|
// Stop any currently playing emulated track
|
||||||
_mixer->stopHandle(_handle);
|
_mixer->stopHandle(_handle);
|
||||||
|
|
||||||
if (stream != 0) {
|
if (stream != 0) {
|
||||||
Timestamp start = Timestamp(0, startFrame, 75);
|
Audio::Timestamp start = Audio::Timestamp(0, startFrame, 75);
|
||||||
Timestamp end = duration ? Timestamp(0, startFrame + duration, 75) : stream->getLength();
|
Audio::Timestamp end = duration ? Audio::Timestamp(0, startFrame + duration, 75) : stream->getLength();
|
||||||
|
|
||||||
/*
|
/*
|
||||||
FIXME: Seems numLoops == 0 and numLoops == 1 both indicate a single repetition,
|
FIXME: Seems numLoops == 0 and numLoops == 1 both indicate a single repetition,
|
||||||
@ -80,39 +71,38 @@ void AudioCDManager::play(int track, int numLoops, int startFrame, int duration,
|
|||||||
repetitions. Finally, -1 means infinitely many
|
repetitions. Finally, -1 means infinitely many
|
||||||
*/
|
*/
|
||||||
_emulating = true;
|
_emulating = true;
|
||||||
_mixer->playStream(Mixer::kMusicSoundType, &_handle,
|
_mixer->playStream(Audio::Mixer::kMusicSoundType, &_handle,
|
||||||
makeLoopingAudioStream(stream, start, end, (numLoops < 1) ? numLoops + 1 : numLoops), -1, _cd.volume, _cd.balance);
|
Audio::makeLoopingAudioStream(stream, start, end, (numLoops < 1) ? numLoops + 1 : numLoops), -1, _cd.volume, _cd.balance);
|
||||||
|
|
||||||
} else {
|
} else {
|
||||||
_emulating = false;
|
_emulating = false;
|
||||||
if (!only_emulate)
|
if (!only_emulate)
|
||||||
g_system->playCD(track, numLoops, startFrame, duration);
|
playCD(track, numLoops, startFrame, duration);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void AudioCDManager::stop() {
|
void DefaultAudioCDManager::stop() {
|
||||||
if (_emulating) {
|
if (_emulating) {
|
||||||
// Audio CD emulation
|
// Audio CD emulation
|
||||||
_mixer->stopHandle(_handle);
|
_mixer->stopHandle(_handle);
|
||||||
_emulating = false;
|
_emulating = false;
|
||||||
} else {
|
} else {
|
||||||
// Real Audio CD
|
// Real Audio CD
|
||||||
g_system->stopCD();
|
stopCD();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
bool AudioCDManager::isPlaying() const {
|
bool DefaultAudioCDManager::isPlaying() const {
|
||||||
if (_emulating) {
|
if (_emulating) {
|
||||||
// Audio CD emulation
|
// Audio CD emulation
|
||||||
return _mixer->isSoundHandleActive(_handle);
|
return _mixer->isSoundHandleActive(_handle);
|
||||||
} else {
|
} else {
|
||||||
// Real Audio CD
|
// Real Audio CD
|
||||||
return g_system->pollCD();
|
return pollCD();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void AudioCDManager::setVolume(byte volume) {
|
void DefaultAudioCDManager::setVolume(byte volume) {
|
||||||
_cd.volume = volume;
|
_cd.volume = volume;
|
||||||
if (_emulating) {
|
if (_emulating) {
|
||||||
// Audio CD emulation
|
// Audio CD emulation
|
||||||
@ -120,7 +110,7 @@ void AudioCDManager::setVolume(byte volume) {
|
|||||||
_mixer->setChannelVolume(_handle, _cd.volume);
|
_mixer->setChannelVolume(_handle, _cd.volume);
|
||||||
} else {
|
} else {
|
||||||
// Real Audio CD
|
// Real Audio CD
|
||||||
|
|
||||||
// Unfortunately I can't implement this atm
|
// Unfortunately I can't implement this atm
|
||||||
// since SDL doesn't seem to offer an interface method for this.
|
// since SDL doesn't seem to offer an interface method for this.
|
||||||
|
|
||||||
@ -128,7 +118,7 @@ void AudioCDManager::setVolume(byte volume) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void AudioCDManager::setBalance(int8 balance) {
|
void DefaultAudioCDManager::setBalance(int8 balance) {
|
||||||
_cd.balance = balance;
|
_cd.balance = balance;
|
||||||
if (_emulating) {
|
if (_emulating) {
|
||||||
// Audio CD emulation
|
// Audio CD emulation
|
||||||
@ -136,7 +126,7 @@ void AudioCDManager::setBalance(int8 balance) {
|
|||||||
_mixer->setChannelBalance(_handle, _cd.balance);
|
_mixer->setChannelBalance(_handle, _cd.balance);
|
||||||
} else {
|
} else {
|
||||||
// Real Audio CD
|
// Real Audio CD
|
||||||
|
|
||||||
// Unfortunately I can't implement this atm
|
// Unfortunately I can't implement this atm
|
||||||
// since SDL doesn't seem to offer an interface method for this.
|
// since SDL doesn't seem to offer an interface method for this.
|
||||||
|
|
||||||
@ -144,7 +134,7 @@ void AudioCDManager::setBalance(int8 balance) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void AudioCDManager::updateCD() {
|
void DefaultAudioCDManager::update() {
|
||||||
if (_emulating) {
|
if (_emulating) {
|
||||||
// Check whether the audio track stopped playback
|
// Check whether the audio track stopped playback
|
||||||
if (!_mixer->isSoundHandleActive(_handle)) {
|
if (!_mixer->isSoundHandleActive(_handle)) {
|
||||||
@ -156,16 +146,12 @@ void AudioCDManager::updateCD() {
|
|||||||
_emulating = false;
|
_emulating = false;
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
g_system->updateCD();
|
updateCD();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
AudioCDManager::Status AudioCDManager::getStatus() const {
|
DefaultAudioCDManager::Status DefaultAudioCDManager::getStatus() const {
|
||||||
// TODO: This could be improved for "real" CD playback.
|
|
||||||
// But to do that, we would have to extend the OSystem interface.
|
|
||||||
Status info = _cd;
|
Status info = _cd;
|
||||||
info.playing = isPlaying();
|
info.playing = isPlaying();
|
||||||
return info;
|
return info;
|
||||||
}
|
}
|
||||||
|
|
||||||
} // End of namespace Audio
|
|
62
backends/audiocd/default/default-audiocd.h
Normal file
62
backends/audiocd/default/default-audiocd.h
Normal file
@ -0,0 +1,62 @@
|
|||||||
|
/* Residual - A 3D game interpreter
|
||||||
|
*
|
||||||
|
* Residual is the legal property of its developers, whose names
|
||||||
|
* are too numerous to list here. Please refer to the AUTHORS
|
||||||
|
* file distributed with this source distribution.
|
||||||
|
*
|
||||||
|
* 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., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
||||||
|
*
|
||||||
|
* $URL$
|
||||||
|
* $Id$
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef BACKENDS_AUDIOCD_DEFAULT_H
|
||||||
|
#define BACKENDS_AUDIOCD_DEFAULT_H
|
||||||
|
|
||||||
|
#include "backends/audiocd/audiocd.h"
|
||||||
|
#include "audio/mixer.h"
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The default audio cd manager. Implements emulation of audio cd playback.
|
||||||
|
*/
|
||||||
|
class DefaultAudioCDManager : public AudioCDManager {
|
||||||
|
public:
|
||||||
|
DefaultAudioCDManager();
|
||||||
|
virtual ~DefaultAudioCDManager() {}
|
||||||
|
|
||||||
|
void play(int track, int numLoops, int startFrame, int duration, bool only_emulate = false);
|
||||||
|
void stop();
|
||||||
|
bool isPlaying() const;
|
||||||
|
void setVolume(byte volume);
|
||||||
|
void setBalance(int8 balance);
|
||||||
|
void update();
|
||||||
|
virtual Status getStatus() const; // Subclasses should override for better status results
|
||||||
|
|
||||||
|
virtual bool openCD(int drive) { return false; }
|
||||||
|
virtual void updateCD() {}
|
||||||
|
virtual bool pollCD() const { return false; }
|
||||||
|
virtual void playCD(int track, int num_loops, int start_frame, int duration) {}
|
||||||
|
virtual void stopCD() {}
|
||||||
|
|
||||||
|
protected:
|
||||||
|
Audio::SoundHandle _handle;
|
||||||
|
bool _emulating;
|
||||||
|
|
||||||
|
Status _cd;
|
||||||
|
Audio::Mixer *_mixer;
|
||||||
|
};
|
||||||
|
|
||||||
|
#endif
|
138
backends/audiocd/sdl/sdl-audiocd.cpp
Normal file
138
backends/audiocd/sdl/sdl-audiocd.cpp
Normal file
@ -0,0 +1,138 @@
|
|||||||
|
/* Residual - A 3D game interpreter
|
||||||
|
*
|
||||||
|
* Residual is the legal property of its developers, whose names
|
||||||
|
* are too numerous to list here. Please refer to the AUTHORS
|
||||||
|
* file distributed with this source distribution.
|
||||||
|
*
|
||||||
|
* 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., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
||||||
|
*
|
||||||
|
* $URL$
|
||||||
|
* $Id$
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include "common/sys.h"
|
||||||
|
|
||||||
|
#if defined(SDL_BACKEND)
|
||||||
|
|
||||||
|
#include "backends/audiocd/sdl/sdl-audiocd.h"
|
||||||
|
|
||||||
|
SdlAudioCDManager::SdlAudioCDManager()
|
||||||
|
:
|
||||||
|
_cdrom(0),
|
||||||
|
_cdTrack(0),
|
||||||
|
_cdNumLoops(0),
|
||||||
|
_cdStartFrame(0),
|
||||||
|
_cdDuration(0),
|
||||||
|
_cdEndTime(0),
|
||||||
|
_cdStopTime(0) {
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
SdlAudioCDManager::~SdlAudioCDManager() {
|
||||||
|
if (_cdrom) {
|
||||||
|
SDL_CDStop(_cdrom);
|
||||||
|
SDL_CDClose(_cdrom);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
bool SdlAudioCDManager::openCD(int drive) {
|
||||||
|
if (SDL_InitSubSystem(SDL_INIT_CDROM) == -1)
|
||||||
|
_cdrom = NULL;
|
||||||
|
else {
|
||||||
|
_cdrom = SDL_CDOpen(drive);
|
||||||
|
// Did it open? Check if _cdrom is NULL
|
||||||
|
if (!_cdrom) {
|
||||||
|
warning("Couldn't open drive: %s", SDL_GetError());
|
||||||
|
} else {
|
||||||
|
_cdNumLoops = 0;
|
||||||
|
_cdStopTime = 0;
|
||||||
|
_cdEndTime = 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return (_cdrom != NULL);
|
||||||
|
}
|
||||||
|
|
||||||
|
void SdlAudioCDManager::stopCD() {
|
||||||
|
// Stop CD Audio in 1/10th of a second
|
||||||
|
_cdStopTime = SDL_GetTicks() + 100;
|
||||||
|
_cdNumLoops = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
void SdlAudioCDManager::playCD(int track, int num_loops, int start_frame, int duration) {
|
||||||
|
if (!num_loops && !start_frame)
|
||||||
|
return;
|
||||||
|
|
||||||
|
if (!_cdrom)
|
||||||
|
return;
|
||||||
|
|
||||||
|
if (duration > 0)
|
||||||
|
duration += 5;
|
||||||
|
|
||||||
|
_cdTrack = track;
|
||||||
|
_cdNumLoops = num_loops;
|
||||||
|
_cdStartFrame = start_frame;
|
||||||
|
|
||||||
|
SDL_CDStatus(_cdrom);
|
||||||
|
if (start_frame == 0 && duration == 0)
|
||||||
|
SDL_CDPlayTracks(_cdrom, track, 0, 1, 0);
|
||||||
|
else
|
||||||
|
SDL_CDPlayTracks(_cdrom, track, start_frame, 0, duration);
|
||||||
|
_cdDuration = duration;
|
||||||
|
_cdStopTime = 0;
|
||||||
|
_cdEndTime = SDL_GetTicks() + _cdrom->track[track].length * 1000 / CD_FPS;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool SdlAudioCDManager::pollCD() const {
|
||||||
|
if (!_cdrom)
|
||||||
|
return false;
|
||||||
|
|
||||||
|
return (_cdNumLoops != 0 && (SDL_GetTicks() < _cdEndTime || SDL_CDStatus(_cdrom) == CD_PLAYING));
|
||||||
|
}
|
||||||
|
|
||||||
|
void SdlAudioCDManager::updateCD() {
|
||||||
|
if (!_cdrom)
|
||||||
|
return;
|
||||||
|
|
||||||
|
if (_cdStopTime != 0 && SDL_GetTicks() >= _cdStopTime) {
|
||||||
|
SDL_CDStop(_cdrom);
|
||||||
|
_cdNumLoops = 0;
|
||||||
|
_cdStopTime = 0;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (_cdNumLoops == 0 || SDL_GetTicks() < _cdEndTime)
|
||||||
|
return;
|
||||||
|
|
||||||
|
if (_cdNumLoops != 1 && SDL_CDStatus(_cdrom) != CD_STOPPED) {
|
||||||
|
// Wait another second for it to be done
|
||||||
|
_cdEndTime += 1000;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (_cdNumLoops > 0)
|
||||||
|
_cdNumLoops--;
|
||||||
|
|
||||||
|
if (_cdNumLoops != 0) {
|
||||||
|
if (_cdStartFrame == 0 && _cdDuration == 0)
|
||||||
|
SDL_CDPlayTracks(_cdrom, _cdTrack, 0, 1, 0);
|
||||||
|
else
|
||||||
|
SDL_CDPlayTracks(_cdrom, _cdTrack, _cdStartFrame, 0, _cdDuration);
|
||||||
|
_cdEndTime = SDL_GetTicks() + _cdrom->track[_cdTrack].length * 1000 / CD_FPS;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#endif
|
53
backends/audiocd/sdl/sdl-audiocd.h
Normal file
53
backends/audiocd/sdl/sdl-audiocd.h
Normal file
@ -0,0 +1,53 @@
|
|||||||
|
/* Residual - A 3D game interpreter
|
||||||
|
*
|
||||||
|
* Residual is the legal property of its developers, whose names
|
||||||
|
* are too numerous to list here. Please refer to the AUTHORS
|
||||||
|
* file distributed with this source distribution.
|
||||||
|
*
|
||||||
|
* 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., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
||||||
|
*
|
||||||
|
* $URL$
|
||||||
|
* $Id$
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef BACKENDS_AUDIOCD_SDL_H
|
||||||
|
#define BACKENDS_AUDIOCD_SDL_H
|
||||||
|
|
||||||
|
#include "backends/audiocd/default/default-audiocd.h"
|
||||||
|
|
||||||
|
#include "backends/platform/sdl/sdl-sys.h"
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The SDL audio cd manager. Implements real audio cd playback.
|
||||||
|
*/
|
||||||
|
class SdlAudioCDManager : public DefaultAudioCDManager {
|
||||||
|
public:
|
||||||
|
SdlAudioCDManager();
|
||||||
|
virtual ~SdlAudioCDManager();
|
||||||
|
|
||||||
|
protected:
|
||||||
|
virtual bool openCD(int drive);
|
||||||
|
virtual void updateCD();
|
||||||
|
virtual bool pollCD() const;
|
||||||
|
virtual void playCD(int track, int num_loops, int start_frame, int duration);
|
||||||
|
virtual void stopCD();
|
||||||
|
|
||||||
|
SDL_CD *_cdrom;
|
||||||
|
int _cdTrack, _cdNumLoops, _cdStartFrame, _cdDuration;
|
||||||
|
uint32 _cdEndTime, _cdStopTime;
|
||||||
|
};
|
||||||
|
|
||||||
|
#endif
|
@ -1,80 +0,0 @@
|
|||||||
/* Residual - A 3D game interpreter
|
|
||||||
*
|
|
||||||
* Residual is the legal property of its developers, whose names
|
|
||||||
* are too numerous to list here. Please refer to the AUTHORS
|
|
||||||
* file distributed with this source distribution.
|
|
||||||
*
|
|
||||||
* 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., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
|
||||||
*
|
|
||||||
* $URL$
|
|
||||||
* $Id$
|
|
||||||
*
|
|
||||||
*/
|
|
||||||
|
|
||||||
#include "backends/base-backend.h"
|
|
||||||
#include "backends/events/default/default-events.h"
|
|
||||||
#include "gui/message.h"
|
|
||||||
|
|
||||||
void BaseBackend::displayMessageOnOSD(const char *msg) {
|
|
||||||
// Display the message for 1.5 seconds
|
|
||||||
GUI::TimedMessageDialog dialog(msg, 1500);
|
|
||||||
dialog.runModal();
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
static Common::EventManager *s_eventManager = 0;
|
|
||||||
|
|
||||||
Common::EventManager *BaseBackend::getEventManager() {
|
|
||||||
// FIXME/TODO: Eventually this method should be turned into an abstract one,
|
|
||||||
// to force backends to implement this conciously (even if they
|
|
||||||
// end up returning the default event manager anyway).
|
|
||||||
if (!s_eventManager)
|
|
||||||
s_eventManager = new DefaultEventManager(this);
|
|
||||||
return s_eventManager;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
/*
|
|
||||||
FIXME: Maybe we should push the default config file loading/saving code below
|
|
||||||
out to all the backends?
|
|
||||||
*/
|
|
||||||
|
|
||||||
|
|
||||||
#if defined(UNIX)
|
|
||||||
#if defined(SAMSUNGTV)
|
|
||||||
#define DEFAULT_CONFIG_FILE "/dtv/usb/sda1/.scummvmrc"
|
|
||||||
#else
|
|
||||||
#define DEFAULT_CONFIG_FILE ".residualrc"
|
|
||||||
#endif
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#if !defined(UNIX)
|
|
||||||
#define DEFAULT_CONFIG_FILE "scummvm.ini"
|
|
||||||
#endif
|
|
||||||
|
|
||||||
Common::SeekableReadStream *BaseBackend::createConfigReadStream() {
|
|
||||||
Common::FSNode file(DEFAULT_CONFIG_FILE);
|
|
||||||
return file.createReadStream();
|
|
||||||
}
|
|
||||||
|
|
||||||
Common::WriteStream *BaseBackend::createConfigWriteStream() {
|
|
||||||
#ifdef __DC__
|
|
||||||
return 0;
|
|
||||||
#else
|
|
||||||
Common::FSNode file(DEFAULT_CONFIG_FILE);
|
|
||||||
return file.createWriteStream();
|
|
||||||
#endif
|
|
||||||
}
|
|
@ -23,6 +23,8 @@
|
|||||||
*
|
*
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
#include "common/sys.h"
|
||||||
|
|
||||||
#if !defined(DISABLE_DEFAULT_EVENTMANAGER)
|
#if !defined(DISABLE_DEFAULT_EVENTMANAGER)
|
||||||
|
|
||||||
#include "common/system.h"
|
#include "common/system.h"
|
||||||
|
@ -23,9 +23,14 @@
|
|||||||
*
|
*
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
#include "common/sys.h"
|
||||||
|
|
||||||
|
#if defined(SDL_BACKEND)
|
||||||
|
|
||||||
|
#include "backends/events/sdl/sdl-events.h"
|
||||||
#include "backends/platform/sdl/sdl.h"
|
#include "backends/platform/sdl/sdl.h"
|
||||||
#include "common/util.h"
|
#include "backends/graphics/graphics.h"
|
||||||
#include "common/events.h"
|
#include "common/config-manager.h"
|
||||||
|
|
||||||
// FIXME move joystick defines out and replace with confile file options
|
// FIXME move joystick defines out and replace with confile file options
|
||||||
// we should really allow users to map any key to a joystick button
|
// we should really allow users to map any key to a joystick button
|
||||||
@ -46,10 +51,33 @@
|
|||||||
#define JOY_BUT_SPACE 4
|
#define JOY_BUT_SPACE 4
|
||||||
#define JOY_BUT_F5 5
|
#define JOY_BUT_F5 5
|
||||||
|
|
||||||
|
SdlEventSource::SdlEventSource()
|
||||||
|
: _scrollLock(false), _joystick(0), _lastScreenID(0), EventSource() {
|
||||||
|
// Reset mouse state
|
||||||
|
memset(&_km, 0, sizeof(_km));
|
||||||
|
/* Residual doesn't support this
|
||||||
|
int joystick_num = ConfMan.getInt("joystick_num");
|
||||||
|
if (joystick_num > -1) {
|
||||||
|
// Initialize SDL joystick subsystem
|
||||||
|
if (SDL_InitSubSystem(SDL_INIT_JOYSTICK) == -1) {
|
||||||
|
error("Could not initialize SDL: %s", SDL_GetError());
|
||||||
|
}
|
||||||
|
|
||||||
|
// Enable joystick
|
||||||
|
if (SDL_NumJoysticks() > 0) {
|
||||||
|
printf("Using joystick: %s\n", SDL_JoystickName(0));
|
||||||
|
_joystick = SDL_JoystickOpen(joystick_num);
|
||||||
|
}
|
||||||
|
}*/
|
||||||
|
}
|
||||||
|
|
||||||
|
SdlEventSource::~SdlEventSource() {
|
||||||
|
/* Residual doesn't support this
|
||||||
|
if (_joystick)
|
||||||
|
SDL_JoystickClose(_joystick);*/
|
||||||
|
}
|
||||||
|
|
||||||
static int mapKey(SDLKey key, SDLMod mod, Uint16 unicode) {
|
int SdlEventSource::mapKey(SDLKey key, SDLMod mod, Uint16 unicode) {
|
||||||
if (key >= SDLK_F1 && key <= SDLK_F9) {
|
if (key >= SDLK_F1 && key <= SDLK_F9) {
|
||||||
return key - SDLK_F1 + Common::ASCII_F1;
|
return key - SDLK_F1 + Common::ASCII_F1;
|
||||||
} else if (key >= SDLK_KP0 && key <= SDLK_KP9) {
|
} else if (key >= SDLK_KP0 && key <= SDLK_KP9) {
|
||||||
@ -66,25 +94,17 @@ static int mapKey(SDLKey key, SDLMod mod, Uint16 unicode) {
|
|||||||
return key;
|
return key;
|
||||||
}
|
}
|
||||||
|
|
||||||
void OSystem_SDL::fillMouseEvent(Common::Event &event, int x, int y) {
|
void SdlEventSource::fillMouseEvent(Common::Event &event, int x, int y) {
|
||||||
event.mouse.x = x;
|
event.mouse.x = x;
|
||||||
event.mouse.y = y;
|
event.mouse.y = y;
|
||||||
|
|
||||||
// Update the "keyboard mouse" coords
|
// Update the "keyboard mouse" coords
|
||||||
_km.x = x;
|
_km.x = x;
|
||||||
_km.y = y;
|
_km.y = y;
|
||||||
/* Residual doesn't support this
|
|
||||||
// Adjust for the screen scaling
|
|
||||||
if (!_overlayVisible) {
|
|
||||||
event.mouse.x /= _videoMode.scaleFactor;
|
|
||||||
event.mouse.y /= _videoMode.scaleFactor;
|
|
||||||
if (_videoMode.aspectRatioCorrection)
|
|
||||||
event.mouse.y = aspect2Real(event.mouse.y);
|
|
||||||
}*/
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void OSystem_SDL::handleKbdMouse() {
|
void SdlEventSource::handleKbdMouse() {
|
||||||
uint32 curTime = getMillis();
|
uint32 curTime = g_system->getMillis();
|
||||||
if (curTime >= _km.last_time + _km.delay_time) {
|
if (curTime >= _km.last_time + _km.delay_time) {
|
||||||
_km.last_time = curTime;
|
_km.last_time = curTime;
|
||||||
if (_km.x_down_count == 1) {
|
if (_km.x_down_count == 1) {
|
||||||
@ -153,7 +173,7 @@ void OSystem_SDL::handleKbdMouse() {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static void SDLModToOSystemKeyFlags(SDLMod mod, Common::Event &event) {
|
void SdlEventSource::SDLModToOSystemKeyFlags(SDLMod mod, Common::Event &event) {
|
||||||
|
|
||||||
event.kbd.flags = 0;
|
event.kbd.flags = 0;
|
||||||
|
|
||||||
@ -178,19 +198,19 @@ static void SDLModToOSystemKeyFlags(SDLMod mod, Common::Event &event) {
|
|||||||
event.kbd.flags |= Common::KBD_CAPS;
|
event.kbd.flags |= Common::KBD_CAPS;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool OSystem_SDL::pollEvent(Common::Event &event) {
|
bool SdlEventSource::pollEvent(Common::Event &event) {
|
||||||
SDL_Event ev;
|
|
||||||
ev.type = SDL_NOEVENT;
|
|
||||||
|
|
||||||
handleKbdMouse();
|
handleKbdMouse();
|
||||||
/* Residual doesn't support this
|
/* Residual doesn't support this
|
||||||
// If the screen mode changed, send an Common::EVENT_SCREEN_CHANGED
|
// If the screen changed, send an Common::EVENT_SCREEN_CHANGED
|
||||||
if (_modeChanged) {
|
int screenID = ((OSystem_SDL *)g_system)->getGraphicsManager()->getScreenChangeID();
|
||||||
_modeChanged = false;
|
if (screenID != _lastScreenID) {
|
||||||
|
_lastScreenID = screenID;
|
||||||
event.type = Common::EVENT_SCREEN_CHANGED;
|
event.type = Common::EVENT_SCREEN_CHANGED;
|
||||||
return true;
|
return true;
|
||||||
}
|
}*/
|
||||||
*/
|
|
||||||
|
SDL_Event ev;
|
||||||
|
ev.type = SDL_NOEVENT;
|
||||||
while (SDL_PollEvent(&ev)) {
|
while (SDL_PollEvent(&ev)) {
|
||||||
preprocessEvents(&ev);
|
preprocessEvents(&ev);
|
||||||
if (dispatchSDLEvent(ev, event))
|
if (dispatchSDLEvent(ev, event))
|
||||||
@ -199,7 +219,7 @@ bool OSystem_SDL::pollEvent(Common::Event &event) {
|
|||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool OSystem_SDL::dispatchSDLEvent(SDL_Event &ev, Common::Event &event) {
|
bool SdlEventSource::dispatchSDLEvent(SDL_Event &ev, Common::Event &event) {
|
||||||
switch (ev.type) {
|
switch (ev.type) {
|
||||||
case SDL_KEYDOWN:
|
case SDL_KEYDOWN:
|
||||||
return handleKeyDown(ev, event);
|
return handleKeyDown(ev, event);
|
||||||
@ -219,9 +239,16 @@ bool OSystem_SDL::dispatchSDLEvent(SDL_Event &ev, Common::Event &event) {
|
|||||||
return handleJoyAxisMotion(ev, event);
|
return handleJoyAxisMotion(ev, event);
|
||||||
|
|
||||||
case SDL_VIDEOEXPOSE:
|
case SDL_VIDEOEXPOSE:
|
||||||
/* Residual doesn't support this */
|
// HACK: Send a fake event, handled by SdlGraphicsManager
|
||||||
//_forceFull = true;
|
event.type = (Common::EventType)OSystem_SDL::kSdlEventExpose;
|
||||||
break;
|
return true;
|
||||||
|
|
||||||
|
case SDL_VIDEORESIZE:
|
||||||
|
// HACK: Send a fake event, handled by OpenGLSdlGraphicsManager
|
||||||
|
event.type = (Common::EventType)OSystem_SDL::kSdlEventResize;
|
||||||
|
event.mouse.x = ev.resize.w;
|
||||||
|
event.mouse.y = ev.resize.h;
|
||||||
|
return true;
|
||||||
|
|
||||||
case SDL_QUIT:
|
case SDL_QUIT:
|
||||||
event.type = Common::EVENT_QUIT;
|
event.type = Common::EVENT_QUIT;
|
||||||
@ -233,7 +260,7 @@ bool OSystem_SDL::dispatchSDLEvent(SDL_Event &ev, Common::Event &event) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
bool OSystem_SDL::handleKeyDown(SDL_Event &ev, Common::Event &event) {
|
bool SdlEventSource::handleKeyDown(SDL_Event &ev, Common::Event &event) {
|
||||||
|
|
||||||
SDLModToOSystemKeyFlags(SDL_GetModState(), event);
|
SDLModToOSystemKeyFlags(SDL_GetModState(), event);
|
||||||
|
|
||||||
@ -244,42 +271,6 @@ bool OSystem_SDL::handleKeyDown(SDL_Event &ev, Common::Event &event) {
|
|||||||
if (_scrollLock)
|
if (_scrollLock)
|
||||||
event.kbd.flags |= Common::KBD_SCRL;
|
event.kbd.flags |= Common::KBD_SCRL;
|
||||||
|
|
||||||
// Alt-Return and Alt-Enter toggle full screen mode
|
|
||||||
/* Residual doesn't support this
|
|
||||||
if (event.kbd.hasFlags(Common::KBD_ALT) && (ev.key.keysym.sym == SDLK_RETURN || ev.key.keysym.sym == SDLK_KP_ENTER)) {
|
|
||||||
beginGFXTransaction();
|
|
||||||
setFullscreenMode(!_videoMode.fullscreen);
|
|
||||||
endGFXTransaction();
|
|
||||||
#ifdef USE_OSD
|
|
||||||
if (_videoMode.fullscreen)
|
|
||||||
displayMessageOnOSD("Fullscreen mode");
|
|
||||||
else
|
|
||||||
displayMessageOnOSD("Windowed mode");
|
|
||||||
#endif
|
|
||||||
|
|
||||||
return false;
|
|
||||||
}*/
|
|
||||||
|
|
||||||
/* // Alt-S: Create a screenshot
|
|
||||||
if (event.kbd.hasFlags(Common::KBD_ALT) && ev.key.keysym.sym == 's') {
|
|
||||||
char filename[20];
|
|
||||||
|
|
||||||
for (int n = 0;; n++) {
|
|
||||||
SDL_RWops *file;
|
|
||||||
|
|
||||||
sprintf(filename, "residual%05d.bmp", n);
|
|
||||||
file = SDL_RWFromFile(filename, "r");
|
|
||||||
if (!file)
|
|
||||||
break;
|
|
||||||
SDL_RWclose(file);
|
|
||||||
}
|
|
||||||
if (saveScreenshot(filename))
|
|
||||||
printf("Saved '%s'\n", filename);
|
|
||||||
else
|
|
||||||
printf("Could not save screenshot!\n");
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
*/
|
|
||||||
// Ctrl-m toggles mouse capture
|
// Ctrl-m toggles mouse capture
|
||||||
if (event.kbd.hasFlags(Common::KBD_CTRL) && ev.key.keysym.sym == 'm') {
|
if (event.kbd.hasFlags(Common::KBD_CTRL) && ev.key.keysym.sym == 'm') {
|
||||||
toggleMouseGrab();
|
toggleMouseGrab();
|
||||||
@ -287,13 +278,13 @@ bool OSystem_SDL::handleKeyDown(SDL_Event &ev, Common::Event &event) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
#if defined(MACOSX)
|
#if defined(MACOSX)
|
||||||
// On Macintosh', Cmd-Q quits
|
// On Macintosh, Cmd-Q quits
|
||||||
if ((ev.key.keysym.mod & KMOD_META) && ev.key.keysym.sym == 'q') {
|
if ((ev.key.keysym.mod & KMOD_META) && ev.key.keysym.sym == 'q') {
|
||||||
event.type = Common::EVENT_QUIT;
|
event.type = Common::EVENT_QUIT;
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
#elif defined(UNIX)
|
#elif defined(UNIX)
|
||||||
// On other unices, Control-Q quits
|
// On other *nix systems, Control-Q quits
|
||||||
if ((ev.key.keysym.mod & KMOD_CTRL) && ev.key.keysym.sym == 'q') {
|
if ((ev.key.keysym.mod & KMOD_CTRL) && ev.key.keysym.sym == 'q') {
|
||||||
event.type = Common::EVENT_QUIT;
|
event.type = Common::EVENT_QUIT;
|
||||||
return true;
|
return true;
|
||||||
@ -306,16 +297,11 @@ bool OSystem_SDL::handleKeyDown(SDL_Event &ev, Common::Event &event) {
|
|||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
// Ctrl-u toggles mute
|
||||||
if ((ev.key.keysym.mod & KMOD_CTRL) && ev.key.keysym.sym == 'u') {
|
if ((ev.key.keysym.mod & KMOD_CTRL) && ev.key.keysym.sym == 'u') {
|
||||||
event.type = Common::EVENT_MUTE;
|
event.type = Common::EVENT_MUTE;
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
/* Residual doesn't support this
|
|
||||||
// Ctrl-Alt-<key> will change the GFX mode
|
|
||||||
if ((event.kbd.flags & (Common::KBD_CTRL|Common::KBD_ALT)) == (Common::KBD_CTRL|Common::KBD_ALT)) {
|
|
||||||
if (handleScalerHotkeys(ev.key))
|
|
||||||
return false;
|
|
||||||
}*/
|
|
||||||
|
|
||||||
if (remapKey(ev, event))
|
if (remapKey(ev, event))
|
||||||
return true;
|
return true;
|
||||||
@ -327,38 +313,65 @@ bool OSystem_SDL::handleKeyDown(SDL_Event &ev, Common::Event &event) {
|
|||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool OSystem_SDL::handleKeyUp(SDL_Event &ev, Common::Event &event) {
|
bool SdlEventSource::handleKeyUp(SDL_Event &ev, Common::Event &event) {
|
||||||
if (remapKey(ev, event))
|
if (remapKey(ev, event))
|
||||||
return true;
|
return true;
|
||||||
|
|
||||||
|
SDLMod mod = SDL_GetModState();
|
||||||
|
|
||||||
|
// Check if this is an event handled by handleKeyDown(), and stop if it is
|
||||||
|
|
||||||
|
// Check if the Ctrl key is down, so that we can trap cases where the
|
||||||
|
// user has the Ctrl key down, and has just released a special key
|
||||||
|
if (mod & KMOD_CTRL) {
|
||||||
|
if (ev.key.keysym.sym == 'm' || // Ctrl-m toggles mouse capture
|
||||||
|
#if defined(MACOSX)
|
||||||
|
// Meta - Q, handled below
|
||||||
|
#elif defined(UNIX)
|
||||||
|
ev.key.keysym.sym == 'q' || // On other *nix systems, Control-Q quits
|
||||||
|
#else
|
||||||
|
ev.key.keysym.sym == 'z' || // Ctrl-z quit
|
||||||
|
#endif
|
||||||
|
ev.key.keysym.sym == 'u') // Ctrl-u toggles mute
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Same for other keys (Meta and Alt)
|
||||||
|
#if defined(MACOSX)
|
||||||
|
if ((mod & KMOD_META) && ev.key.keysym.sym == 'q')
|
||||||
|
return false; // On Macintosh, Cmd-Q quits
|
||||||
|
#elif defined(UNIX)
|
||||||
|
// Control Q has already been handled above
|
||||||
|
#else
|
||||||
|
if ((mod & KMOD_ALT) && ev.key.keysym.sym == 'x')
|
||||||
|
return false; // Alt-x quit
|
||||||
|
#endif
|
||||||
|
|
||||||
|
// If we reached here, this isn't an event handled by handleKeyDown(), thus
|
||||||
|
// continue normally
|
||||||
|
|
||||||
event.type = Common::EVENT_KEYUP;
|
event.type = Common::EVENT_KEYUP;
|
||||||
event.kbd.keycode = (Common::KeyCode)ev.key.keysym.sym;
|
event.kbd.keycode = (Common::KeyCode)ev.key.keysym.sym;
|
||||||
event.kbd.ascii = mapKey(ev.key.keysym.sym, ev.key.keysym.mod, ev.key.keysym.unicode);
|
event.kbd.ascii = mapKey(ev.key.keysym.sym, ev.key.keysym.mod, ev.key.keysym.unicode);
|
||||||
|
|
||||||
// Ctrl-Alt-<key> will change the GFX mode
|
// Ctrl-Alt-<key> will change the GFX mode
|
||||||
SDLModToOSystemKeyFlags(SDL_GetModState(), event);
|
SDLModToOSystemKeyFlags(mod, event);
|
||||||
|
|
||||||
// Set the scroll lock sticky flag
|
// Set the scroll lock sticky flag
|
||||||
if (_scrollLock)
|
if (_scrollLock)
|
||||||
event.kbd.flags |= Common::KBD_SCRL;
|
event.kbd.flags |= Common::KBD_SCRL;
|
||||||
/* Residual doesn't support this
|
|
||||||
if (isScalerHotkey(event))
|
|
||||||
// Swallow these key up events
|
|
||||||
return false;*/
|
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool OSystem_SDL::handleMouseMotion(SDL_Event &ev, Common::Event &event) {
|
bool SdlEventSource::handleMouseMotion(SDL_Event &ev, Common::Event &event) {
|
||||||
event.type = Common::EVENT_MOUSEMOVE;
|
event.type = Common::EVENT_MOUSEMOVE;
|
||||||
fillMouseEvent(event, ev.motion.x, ev.motion.y);
|
fillMouseEvent(event, ev.motion.x, ev.motion.y);
|
||||||
|
|
||||||
/* Residual doesn't support this */
|
|
||||||
//setMousePos(event.mouse.x, event.mouse.y);
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool OSystem_SDL::handleMouseButtonDown(SDL_Event &ev, Common::Event &event) {
|
bool SdlEventSource::handleMouseButtonDown(SDL_Event &ev, Common::Event &event) {
|
||||||
if (ev.button.button == SDL_BUTTON_LEFT)
|
if (ev.button.button == SDL_BUTTON_LEFT)
|
||||||
event.type = Common::EVENT_LBUTTONDOWN;
|
event.type = Common::EVENT_LBUTTONDOWN;
|
||||||
else if (ev.button.button == SDL_BUTTON_RIGHT)
|
else if (ev.button.button == SDL_BUTTON_RIGHT)
|
||||||
@ -381,7 +394,7 @@ bool OSystem_SDL::handleMouseButtonDown(SDL_Event &ev, Common::Event &event) {
|
|||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool OSystem_SDL::handleMouseButtonUp(SDL_Event &ev, Common::Event &event) {
|
bool SdlEventSource::handleMouseButtonUp(SDL_Event &ev, Common::Event &event) {
|
||||||
if (ev.button.button == SDL_BUTTON_LEFT)
|
if (ev.button.button == SDL_BUTTON_LEFT)
|
||||||
event.type = Common::EVENT_LBUTTONUP;
|
event.type = Common::EVENT_LBUTTONUP;
|
||||||
else if (ev.button.button == SDL_BUTTON_RIGHT)
|
else if (ev.button.button == SDL_BUTTON_RIGHT)
|
||||||
@ -397,7 +410,7 @@ bool OSystem_SDL::handleMouseButtonUp(SDL_Event &ev, Common::Event &event) {
|
|||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool OSystem_SDL::handleJoyButtonDown(SDL_Event &ev, Common::Event &event) {
|
bool SdlEventSource::handleJoyButtonDown(SDL_Event &ev, Common::Event &event) {
|
||||||
if (ev.jbutton.button == JOY_BUT_LMOUSE) {
|
if (ev.jbutton.button == JOY_BUT_LMOUSE) {
|
||||||
event.type = Common::EVENT_LBUTTONDOWN;
|
event.type = Common::EVENT_LBUTTONDOWN;
|
||||||
fillMouseEvent(event, _km.x, _km.y);
|
fillMouseEvent(event, _km.x, _km.y);
|
||||||
@ -428,7 +441,7 @@ bool OSystem_SDL::handleJoyButtonDown(SDL_Event &ev, Common::Event &event) {
|
|||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool OSystem_SDL::handleJoyButtonUp(SDL_Event &ev, Common::Event &event) {
|
bool SdlEventSource::handleJoyButtonUp(SDL_Event &ev, Common::Event &event) {
|
||||||
if (ev.jbutton.button == JOY_BUT_LMOUSE) {
|
if (ev.jbutton.button == JOY_BUT_LMOUSE) {
|
||||||
event.type = Common::EVENT_LBUTTONUP;
|
event.type = Common::EVENT_LBUTTONUP;
|
||||||
fillMouseEvent(event, _km.x, _km.y);
|
fillMouseEvent(event, _km.x, _km.y);
|
||||||
@ -459,7 +472,7 @@ bool OSystem_SDL::handleJoyButtonUp(SDL_Event &ev, Common::Event &event) {
|
|||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool OSystem_SDL::handleJoyAxisMotion(SDL_Event &ev, Common::Event &event) {
|
bool SdlEventSource::handleJoyAxisMotion(SDL_Event &ev, Common::Event &event) {
|
||||||
int axis = ev.jaxis.value;
|
int axis = ev.jaxis.value;
|
||||||
if ( axis > JOY_DEADZONE) {
|
if ( axis > JOY_DEADZONE) {
|
||||||
axis -= JOY_DEADZONE;
|
axis -= JOY_DEADZONE;
|
||||||
@ -472,7 +485,7 @@ bool OSystem_SDL::handleJoyAxisMotion(SDL_Event &ev, Common::Event &event) {
|
|||||||
|
|
||||||
if ( ev.jaxis.axis == JOY_XAXIS) {
|
if ( ev.jaxis.axis == JOY_XAXIS) {
|
||||||
#ifdef JOY_ANALOG
|
#ifdef JOY_ANALOG
|
||||||
_km.x_vel = axis/2000;
|
_km.x_vel = axis / 2000;
|
||||||
_km.x_down_count = 0;
|
_km.x_down_count = 0;
|
||||||
#else
|
#else
|
||||||
if (axis != 0) {
|
if (axis != 0) {
|
||||||
@ -507,7 +520,7 @@ bool OSystem_SDL::handleJoyAxisMotion(SDL_Event &ev, Common::Event &event) {
|
|||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool OSystem_SDL::remapKey(SDL_Event &ev, Common::Event &event) {
|
bool SdlEventSource::remapKey(SDL_Event &ev, Common::Event &event) {
|
||||||
#ifdef LINUPY
|
#ifdef LINUPY
|
||||||
// On Yopy map the End button to quit
|
// On Yopy map the End button to quit
|
||||||
if ((ev.key.keysym.sym == 293)) {
|
if ((ev.key.keysym.sym == 293)) {
|
||||||
@ -574,3 +587,19 @@ bool OSystem_SDL::remapKey(SDL_Event &ev, Common::Event &event) {
|
|||||||
#endif
|
#endif
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void SdlEventSource::toggleMouseGrab() {
|
||||||
|
if (SDL_WM_GrabInput(SDL_GRAB_QUERY) == SDL_GRAB_OFF)
|
||||||
|
SDL_WM_GrabInput(SDL_GRAB_ON);
|
||||||
|
else
|
||||||
|
SDL_WM_GrabInput(SDL_GRAB_OFF);
|
||||||
|
}
|
||||||
|
|
||||||
|
void SdlEventSource::resetKeyboadEmulation(int16 x_max, int16 y_max) {
|
||||||
|
_km.x_max = x_max;
|
||||||
|
_km.y_max = y_max;
|
||||||
|
_km.delay_time = 25;
|
||||||
|
_km.last_time = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
#endif
|
135
backends/events/sdl/sdl-events.h
Normal file
135
backends/events/sdl/sdl-events.h
Normal file
@ -0,0 +1,135 @@
|
|||||||
|
/* Residual - A 3D game interpreter
|
||||||
|
*
|
||||||
|
* Residual is the legal property of its developers, whose names
|
||||||
|
* are too numerous to list here. Please refer to the AUTHORS
|
||||||
|
* file distributed with this source distribution.
|
||||||
|
*
|
||||||
|
* 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., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
||||||
|
*
|
||||||
|
* $URL$
|
||||||
|
* $Id$
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
|
||||||
|
#if !defined(BACKEND_EVENTS_SDL_H) && !defined(DISABLE_DEFAULT_EVENTMANAGER)
|
||||||
|
#define BACKEND_EVENTS_SDL_H
|
||||||
|
|
||||||
|
#include "backends/events/default/default-events.h"
|
||||||
|
|
||||||
|
#include "backends/platform/sdl/sdl-sys.h"
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The SDL event source.
|
||||||
|
*/
|
||||||
|
class SdlEventSource : public Common::EventSource {
|
||||||
|
public:
|
||||||
|
SdlEventSource();
|
||||||
|
virtual ~SdlEventSource();
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Gets and processes SDL events.
|
||||||
|
*/
|
||||||
|
virtual bool pollEvent(Common::Event &event);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Resets keyboard emulation after a video screen change
|
||||||
|
*/
|
||||||
|
virtual void resetKeyboadEmulation(int16 x_max, int16 y_max);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Toggles mouse input grab
|
||||||
|
*/
|
||||||
|
virtual void toggleMouseGrab();
|
||||||
|
|
||||||
|
protected:
|
||||||
|
/** @name Keyboard mouse emulation
|
||||||
|
* Disabled by fingolfin 2004-12-18.
|
||||||
|
* I am keeping the rest of the code in for now, since the joystick
|
||||||
|
* code (or rather, "hack") uses it, too.
|
||||||
|
*/
|
||||||
|
//@{
|
||||||
|
|
||||||
|
struct KbdMouse {
|
||||||
|
int16 x, y, x_vel, y_vel, x_max, y_max, x_down_count, y_down_count;
|
||||||
|
uint32 last_time, delay_time, x_down_time, y_down_time;
|
||||||
|
};
|
||||||
|
KbdMouse _km;
|
||||||
|
|
||||||
|
//@}
|
||||||
|
|
||||||
|
/** Scroll lock state - since SDL doesn't track it */
|
||||||
|
bool _scrollLock;
|
||||||
|
|
||||||
|
/** Joystick */
|
||||||
|
SDL_Joystick *_joystick;
|
||||||
|
|
||||||
|
/** Last screen id for checking if it was modified */
|
||||||
|
int _lastScreenID;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Pre process an event before it is dispatched.
|
||||||
|
*/
|
||||||
|
virtual void preprocessEvents(SDL_Event *event) {}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Dispatchs SDL events for each handler.
|
||||||
|
*/
|
||||||
|
virtual bool dispatchSDLEvent(SDL_Event &ev, Common::Event &event);
|
||||||
|
|
||||||
|
|
||||||
|
/** @name Event Handlers
|
||||||
|
* Handlers for specific SDL events, called by SdlEventSource::dispatchSDLEvent().
|
||||||
|
* This way, if a managers inherits fromt this SDL events manager, it can
|
||||||
|
* change the behavior of only a single event, without having to override all
|
||||||
|
* of SdlEventSource::dispatchSDLEvent().
|
||||||
|
*/
|
||||||
|
//@{
|
||||||
|
|
||||||
|
virtual bool handleKeyDown(SDL_Event &ev, Common::Event &event);
|
||||||
|
virtual bool handleKeyUp(SDL_Event &ev, Common::Event &event);
|
||||||
|
virtual bool handleMouseMotion(SDL_Event &ev, Common::Event &event);
|
||||||
|
virtual bool handleMouseButtonDown(SDL_Event &ev, Common::Event &event);
|
||||||
|
virtual bool handleMouseButtonUp(SDL_Event &ev, Common::Event &event);
|
||||||
|
virtual bool handleJoyButtonDown(SDL_Event &ev, Common::Event &event);
|
||||||
|
virtual bool handleJoyButtonUp(SDL_Event &ev, Common::Event &event);
|
||||||
|
virtual bool handleJoyAxisMotion(SDL_Event &ev, Common::Event &event);
|
||||||
|
virtual void handleKbdMouse();
|
||||||
|
|
||||||
|
//@}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Assigns the mouse coords to the mouse event
|
||||||
|
*/
|
||||||
|
virtual void fillMouseEvent(Common::Event &event, int x, int y);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Remaps key events. This allows platforms to configure
|
||||||
|
* their custom keys.
|
||||||
|
*/
|
||||||
|
virtual bool remapKey(SDL_Event &ev, Common::Event &event);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Maps the ASCII value of key
|
||||||
|
*/
|
||||||
|
virtual int mapKey(SDLKey key, SDLMod mod, Uint16 unicode);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Configures the key modifiers flags status
|
||||||
|
*/
|
||||||
|
virtual void SDLModToOSystemKeyFlags(SDLMod mod, Common::Event &event);
|
||||||
|
};
|
||||||
|
|
||||||
|
#endif
|
@ -23,6 +23,9 @@
|
|||||||
*
|
*
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
// Disable symbol overrides so that we can use FILE, fopen etc.
|
||||||
|
#define FORBIDDEN_SYMBOL_ALLOW_ALL
|
||||||
|
|
||||||
#include "backends/fs/stdiostream.h"
|
#include "backends/fs/stdiostream.h"
|
||||||
|
|
||||||
StdioStream::StdioStream(void *handle) : _handle(handle) {
|
StdioStream::StdioStream(void *handle) : _handle(handle) {
|
||||||
|
@ -22,7 +22,11 @@
|
|||||||
* $Id$
|
* $Id$
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
// Disable symbol overrides so that we can use system headers.
|
||||||
|
#define FORBIDDEN_SYMBOL_ALLOW_ALL
|
||||||
|
|
||||||
#if defined(WIN32)
|
#if defined(WIN32)
|
||||||
|
|
||||||
#include "backends/fs/windows/windows-fs-factory.h"
|
#include "backends/fs/windows/windows-fs-factory.h"
|
||||||
#include "backends/fs/windows/windows-fs.cpp"
|
#include "backends/fs/windows/windows-fs.cpp"
|
||||||
|
|
||||||
|
67
backends/graphics/graphics.h
Normal file
67
backends/graphics/graphics.h
Normal file
@ -0,0 +1,67 @@
|
|||||||
|
/* Residual - A 3D game interpreter
|
||||||
|
*
|
||||||
|
* Residual is the legal property of its developers, whose names
|
||||||
|
* are too numerous to list here. Please refer to the AUTHORS
|
||||||
|
* file distributed with this source distribution.
|
||||||
|
*
|
||||||
|
* 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., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
||||||
|
*
|
||||||
|
* $URL$
|
||||||
|
* $Id$
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef BACKENDS_GRAPHICS_ABSTRACT_H
|
||||||
|
#define BACKENDS_GRAPHICS_ABSTRACT_H
|
||||||
|
|
||||||
|
#include "common/system.h"
|
||||||
|
#include "common/noncopyable.h"
|
||||||
|
#include "common/keyboard.h"
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Abstract class for graphics manager. Subclasses
|
||||||
|
* implement the real functionality.
|
||||||
|
*/
|
||||||
|
class GraphicsManager : Common::NonCopyable {
|
||||||
|
public:
|
||||||
|
virtual ~GraphicsManager() {}
|
||||||
|
|
||||||
|
virtual bool hasFeature(OSystem::Feature f) = 0;
|
||||||
|
virtual void setFeatureState(OSystem::Feature f, bool enable) = 0;
|
||||||
|
virtual bool getFeatureState(OSystem::Feature f) = 0;
|
||||||
|
|
||||||
|
virtual void launcherInitSize(uint w, uint h) = 0;
|
||||||
|
virtual byte *setupScreen(int screenW, int screenH, bool fullscreen, bool accel3d) = 0;
|
||||||
|
virtual int16 getHeight() = 0;
|
||||||
|
virtual int16 getWidth() = 0;
|
||||||
|
virtual void updateScreen() = 0;
|
||||||
|
|
||||||
|
virtual void showOverlay() = 0;
|
||||||
|
virtual void hideOverlay() = 0;
|
||||||
|
virtual Graphics::PixelFormat getOverlayFormat() const = 0;
|
||||||
|
virtual void clearOverlay() = 0;
|
||||||
|
virtual void grabOverlay(OverlayColor *buf, int pitch) = 0;
|
||||||
|
virtual void copyRectToOverlay(const OverlayColor *buf, int pitch, int x, int y, int w, int h)= 0;
|
||||||
|
virtual int16 getOverlayHeight() = 0;
|
||||||
|
virtual int16 getOverlayWidth() = 0;
|
||||||
|
|
||||||
|
virtual bool showMouse(bool visible) = 0;
|
||||||
|
virtual void warpMouse(int x, int y) = 0;
|
||||||
|
virtual void setMouseCursor(const byte *buf, uint w, uint h, int hotspotX, int hotspotY, uint32 keycolor, int cursorTargetScale = 1, const Graphics::PixelFormat *format = NULL) = 0;
|
||||||
|
|
||||||
|
virtual void displayMessageOnOSD(const char *msg) {}
|
||||||
|
};
|
||||||
|
|
||||||
|
#endif
|
@ -23,7 +23,14 @@
|
|||||||
*
|
*
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
#include "common/sys.h"
|
||||||
|
|
||||||
|
#if defined(SDL_BACKEND)
|
||||||
|
|
||||||
|
#include "backends/graphics/sdl/sdl-graphics.h"
|
||||||
|
#include "backends/events/sdl/sdl-events.h"
|
||||||
#include "backends/platform/sdl/sdl.h"
|
#include "backends/platform/sdl/sdl.h"
|
||||||
|
#include "common/config-manager.h"
|
||||||
#include "common/mutex.h"
|
#include "common/mutex.h"
|
||||||
#include "common/translation.h"
|
#include "common/translation.h"
|
||||||
#include "common/util.h"
|
#include "common/util.h"
|
||||||
@ -32,17 +39,70 @@
|
|||||||
#endif
|
#endif
|
||||||
#include "graphics/font.h"
|
#include "graphics/font.h"
|
||||||
#include "graphics/fontman.h"
|
#include "graphics/fontman.h"
|
||||||
#include "graphics/scaler.h"
|
|
||||||
#include "graphics/surface.h"
|
#include "graphics/surface.h"
|
||||||
|
|
||||||
|
SdlGraphicsManager::SdlGraphicsManager(SdlEventSource *sdlEventSource)
|
||||||
|
:
|
||||||
|
_sdlEventSource(sdlEventSource),
|
||||||
|
_screen(0),
|
||||||
|
_overlayVisible(false),
|
||||||
|
_overlayscreen(0),
|
||||||
|
_overlayWidth(0), _overlayHeight(0),
|
||||||
|
_overlayDirty(true), _overlayNumTex(0)
|
||||||
|
#ifdef USE_OPENGL
|
||||||
|
, _overlayTexIds(0) {
|
||||||
|
#else
|
||||||
|
{
|
||||||
|
#endif
|
||||||
|
if (SDL_InitSubSystem(SDL_INIT_VIDEO) == -1) {
|
||||||
|
error("Could not initialize SDL: %s", SDL_GetError());
|
||||||
|
}
|
||||||
|
|
||||||
|
// This is also called in initSDL(), but initializing graphics
|
||||||
|
// may reset it.
|
||||||
|
SDL_EnableUNICODE(1);
|
||||||
|
|
||||||
void OSystem_SDL::launcherInitSize(uint w, uint h) {
|
#ifdef _WIN32_WCE
|
||||||
|
if (ConfMan.hasKey("use_GDI") && ConfMan.getBool("use_GDI")) {
|
||||||
|
SDL_VideoInit("windib", 0);
|
||||||
|
sdlFlags ^= SDL_INIT_VIDEO;
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
|
||||||
|
SdlGraphicsManager::~SdlGraphicsManager() {
|
||||||
|
// Unregister the event observer
|
||||||
|
if (g_system->getEventManager()->getEventDispatcher() != NULL)
|
||||||
|
g_system->getEventManager()->getEventDispatcher()->unregisterObserver(this);
|
||||||
|
}
|
||||||
|
|
||||||
|
void SdlGraphicsManager::initEventObserver() {
|
||||||
|
// Register the graphics manager as a event observer
|
||||||
|
g_system->getEventManager()->getEventDispatcher()->registerObserver(this, 10, false);
|
||||||
|
}
|
||||||
|
|
||||||
|
bool SdlGraphicsManager::hasFeature(OSystem::Feature f) {
|
||||||
|
return
|
||||||
|
#ifdef USE_OPENGL
|
||||||
|
(f == OSystem::kFeatureOpenGL);
|
||||||
|
#else
|
||||||
|
false;
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
|
||||||
|
void SdlGraphicsManager::setFeatureState(OSystem::Feature f, bool enable) {
|
||||||
|
}
|
||||||
|
|
||||||
|
bool SdlGraphicsManager::getFeatureState(OSystem::Feature f) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
void SdlGraphicsManager::launcherInitSize(uint w, uint h) {
|
||||||
closeOverlay();
|
closeOverlay();
|
||||||
setupScreen(w, h, false, false);
|
setupScreen(w, h, false, false);
|
||||||
}
|
}
|
||||||
|
|
||||||
byte *OSystem_SDL::setupScreen(int screenW, int screenH, bool fullscreen, bool accel3d) {
|
byte *SdlGraphicsManager::setupScreen(int screenW, int screenH, bool fullscreen, bool accel3d) {
|
||||||
uint32 sdlflags;
|
uint32 sdlflags;
|
||||||
int bpp;
|
int bpp;
|
||||||
|
|
||||||
@ -161,7 +221,7 @@ byte *OSystem_SDL::setupScreen(int screenW, int screenH, bool fullscreen, bool a
|
|||||||
|
|
||||||
#define BITMAP_TEXTURE_SIZE 256
|
#define BITMAP_TEXTURE_SIZE 256
|
||||||
|
|
||||||
void OSystem_SDL::updateScreen() {
|
void SdlGraphicsManager::updateScreen() {
|
||||||
#ifdef USE_OPENGL
|
#ifdef USE_OPENGL
|
||||||
if (_opengl) {
|
if (_opengl) {
|
||||||
if (_overlayVisible) {
|
if (_overlayVisible) {
|
||||||
@ -265,11 +325,11 @@ void OSystem_SDL::updateScreen() {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
int16 OSystem_SDL::getHeight() {
|
int16 SdlGraphicsManager::getHeight() {
|
||||||
return _screen->h;
|
return _screen->h;
|
||||||
}
|
}
|
||||||
|
|
||||||
int16 OSystem_SDL::getWidth() {
|
int16 SdlGraphicsManager::getWidth() {
|
||||||
return _screen->w;
|
return _screen->w;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -278,7 +338,7 @@ int16 OSystem_SDL::getWidth() {
|
|||||||
#pragma mark --- Overlays ---
|
#pragma mark --- Overlays ---
|
||||||
#pragma mark -
|
#pragma mark -
|
||||||
|
|
||||||
void OSystem_SDL::showOverlay() {
|
void SdlGraphicsManager::showOverlay() {
|
||||||
if (_overlayVisible)
|
if (_overlayVisible)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
@ -287,7 +347,8 @@ void OSystem_SDL::showOverlay() {
|
|||||||
clearOverlay();
|
clearOverlay();
|
||||||
}
|
}
|
||||||
|
|
||||||
void OSystem_SDL::hideOverlay() {
|
void SdlGraphicsManager::hideOverlay() {
|
||||||
|
|
||||||
if (!_overlayVisible)
|
if (!_overlayVisible)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
@ -296,7 +357,8 @@ void OSystem_SDL::hideOverlay() {
|
|||||||
clearOverlay();
|
clearOverlay();
|
||||||
}
|
}
|
||||||
|
|
||||||
void OSystem_SDL::clearOverlay() {
|
void SdlGraphicsManager::clearOverlay() {
|
||||||
|
|
||||||
if (!_overlayVisible)
|
if (!_overlayVisible)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
@ -324,7 +386,7 @@ void OSystem_SDL::clearOverlay() {
|
|||||||
_overlayDirty = true;
|
_overlayDirty = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
void OSystem_SDL::grabOverlay(OverlayColor *buf, int pitch) {
|
void SdlGraphicsManager::grabOverlay(OverlayColor *buf, int pitch) {
|
||||||
if (_overlayscreen == NULL)
|
if (_overlayscreen == NULL)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
@ -342,7 +404,7 @@ void OSystem_SDL::grabOverlay(OverlayColor *buf, int pitch) {
|
|||||||
SDL_UnlockSurface(_overlayscreen);
|
SDL_UnlockSurface(_overlayscreen);
|
||||||
}
|
}
|
||||||
|
|
||||||
void OSystem_SDL::copyRectToOverlay(const OverlayColor *buf, int pitch, int x, int y, int w, int h) {
|
void SdlGraphicsManager::copyRectToOverlay(const OverlayColor *buf, int pitch, int x, int y, int w, int h) {
|
||||||
if (_overlayscreen == NULL)
|
if (_overlayscreen == NULL)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
@ -383,16 +445,7 @@ void OSystem_SDL::copyRectToOverlay(const OverlayColor *buf, int pitch, int x, i
|
|||||||
SDL_UnlockSurface(_overlayscreen);
|
SDL_UnlockSurface(_overlayscreen);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void SdlGraphicsManager::closeOverlay() {
|
||||||
#pragma mark -
|
|
||||||
#pragma mark --- Mouse ---
|
|
||||||
#pragma mark -
|
|
||||||
|
|
||||||
bool OSystem_SDL::showMouse(bool visible) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
void OSystem_SDL::closeOverlay() {
|
|
||||||
if (_overlayscreen) {
|
if (_overlayscreen) {
|
||||||
SDL_FreeSurface(_overlayscreen);
|
SDL_FreeSurface(_overlayscreen);
|
||||||
_overlayscreen = NULL;
|
_overlayscreen = NULL;
|
||||||
@ -406,111 +459,20 @@ void OSystem_SDL::closeOverlay() {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void OSystem_SDL::warpMouse(int x, int y) {
|
#pragma mark -
|
||||||
|
#pragma mark --- Mouse ---
|
||||||
|
#pragma mark -
|
||||||
|
|
||||||
|
bool SdlGraphicsManager::showMouse(bool visible) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
void SdlGraphicsManager::warpMouse(int x, int y) {
|
||||||
SDL_WarpMouse(x, y);
|
SDL_WarpMouse(x, y);
|
||||||
/* Residual doesn't support this
|
|
||||||
int y1 = y;
|
|
||||||
|
|
||||||
// Don't change mouse position, when mouse is outside of our window (in case of windowed mode)
|
|
||||||
if (!(SDL_GetAppState( ) & SDL_APPMOUSEFOCUS))
|
|
||||||
return;
|
|
||||||
|
|
||||||
if (_videoMode.aspectRatioCorrection && !_overlayVisible)
|
|
||||||
y1 = real2Aspect(y);
|
|
||||||
|
|
||||||
if (_mouseCurState.x != x || _mouseCurState.y != y) {
|
|
||||||
if (!_overlayVisible)
|
|
||||||
SDL_WarpMouse(x * _videoMode.scaleFactor, y1 * _videoMode.scaleFactor);
|
|
||||||
else
|
|
||||||
SDL_WarpMouse(x, y1);
|
|
||||||
|
|
||||||
// SDL_WarpMouse() generates a mouse movement event, so
|
|
||||||
// setMousePos() would be called eventually. However, the
|
|
||||||
// cannon script in CoMI calls this function twice each time
|
|
||||||
// the cannon is reloaded. Unless we update the mouse position
|
|
||||||
// immediately the second call is ignored, causing the cannon
|
|
||||||
// to change its aim.
|
|
||||||
|
|
||||||
setMousePos(x, y);
|
|
||||||
}*/
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void OSystem_SDL::setMouseCursor(const byte *buf, uint w, uint h, int hotspot_x, int hotspot_y, uint32 keycolor, int cursorTargetScale, const Graphics::PixelFormat *format) {
|
bool SdlGraphicsManager::notifyEvent(const Common::Event &event) {
|
||||||
#ifdef USE_RGB_COLOR
|
return false;
|
||||||
if (!format)
|
}
|
||||||
_cursorFormat = Graphics::PixelFormat::createFormatCLUT8();
|
|
||||||
else if (format->bytesPerPixel <= _screenFormat.bytesPerPixel)
|
|
||||||
_cursorFormat = *format;
|
|
||||||
|
|
||||||
if (_cursorFormat.bytesPerPixel < 4)
|
|
||||||
assert(keycolor < (uint)(1 << (_cursorFormat.bytesPerPixel << 3)));
|
|
||||||
#else
|
|
||||||
assert(keycolor <= 0xFF);
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
if (w == 0 || h == 0)
|
|
||||||
return;
|
|
||||||
|
|
||||||
/* Residual doesn't support this
|
|
||||||
_mouseCurState.hotX = hotspot_x;
|
|
||||||
_mouseCurState.hotY = hotspot_y;
|
|
||||||
|
|
||||||
_mouseKeyColor = keycolor;
|
|
||||||
|
|
||||||
_cursorTargetScale = cursorTargetScale;
|
|
||||||
|
|
||||||
_mouseTextureId = new GLuint[1];
|
|
||||||
glGenTextures(1, _mouseTextureId);
|
|
||||||
glBindTexture(GL_TEXTURE_2D, _overlayTexIds[i]);
|
|
||||||
glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_NEAREST);
|
|
||||||
glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_NEAREST);
|
|
||||||
glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_S, GL_CLAMP);
|
|
||||||
glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_T, GL_CLAMP);
|
|
||||||
glTexImage2D(GL_TEXTURE_2D, 0, GL_RGB, BITMAP_TEXTURE_SIZE, BITMAP_TEXTURE_SIZE, 0, GL_RGB, GL_UNSIGNED_SHORT_5_6_5, NULL);
|
|
||||||
|
|
||||||
glPixelStorei(GL_UNPACK_ALIGNMENT, 2);
|
|
||||||
glPixelStorei(GL_UNPACK_ROW_LENGTH, _overlayWidth);
|
|
||||||
glBindTexture(GL_TEXTURE_2D, _overlayTexIds[curTexIdx]);
|
|
||||||
glTexSubImage2D(GL_TEXTURE_2D, 0, 0, 0, _mouseWidth, _mouseHeight, GL_RGB, GL_UNSIGNED_SHORT_5_6_5, (byte *)buf + (y * 2 * _mouseWidth) + (2 * x));
|
|
||||||
glPixelStorei(GL_UNPACK_ALIGNMENT, 4);
|
|
||||||
glPixelStorei(GL_UNPACK_ROW_LENGTH, 0);
|
|
||||||
|
|
||||||
if (_mouseCurState.w != (int)w || _mouseCurState.h != (int)h) {
|
|
||||||
_mouseCurState.w = w;
|
|
||||||
_mouseCurState.h = h;
|
|
||||||
|
|
||||||
if (_mouseOrigSurface)
|
|
||||||
SDL_FreeSurface(_mouseOrigSurface);
|
|
||||||
|
|
||||||
_mouseOrigSurface = SDL_CreateRGBSurface(SDL_SWSURFACE | SDL_RLEACCEL | SDL_SRCCOLORKEY | SDL_SRCALPHA,
|
|
||||||
_mouseCurState.w,
|
|
||||||
_mouseCurState.h,
|
|
||||||
16,
|
|
||||||
_hwscreen->format->Rmask,
|
|
||||||
_hwscreen->format->Gmask,
|
|
||||||
_hwscreen->format->Bmask,
|
|
||||||
_hwscreen->format->Amask);
|
|
||||||
|
|
||||||
if (_mouseOrigSurface == NULL)
|
|
||||||
error("allocating _mouseOrigSurface failed");
|
|
||||||
SDL_SetColorKey(_mouseOrigSurface, SDL_RLEACCEL | SDL_SRCCOLORKEY | SDL_SRCALPHA, kMouseColorKey);
|
|
||||||
}
|
|
||||||
|
|
||||||
free(_mouseData);
|
|
||||||
#ifdef USE_RGB_COLOR
|
|
||||||
_mouseData = (byte *)malloc(w * h * _cursorFormat.bytesPerPixel);
|
|
||||||
memcpy(_mouseData, buf, w * h * _cursorFormat.bytesPerPixel);
|
|
||||||
#else
|
|
||||||
_mouseData = (byte *)malloc(w * h);
|
|
||||||
memcpy(_mouseData, buf, w * h);
|
|
||||||
#endif
|
|
||||||
|
|
||||||
blitCursor();*/
|
|
||||||
}
|
|
||||||
|
|
||||||
void OSystem_SDL::toggleMouseGrab() {
|
|
||||||
if (SDL_WM_GrabInput(SDL_GRAB_QUERY) == SDL_GRAB_OFF)
|
|
||||||
SDL_WM_GrabInput(SDL_GRAB_ON);
|
|
||||||
else
|
|
||||||
SDL_WM_GrabInput(SDL_GRAB_OFF);
|
|
||||||
}
|
|
111
backends/graphics/sdl/sdl-graphics.h
Normal file
111
backends/graphics/sdl/sdl-graphics.h
Normal file
@ -0,0 +1,111 @@
|
|||||||
|
/* Residual - A 3D game interpreter
|
||||||
|
*
|
||||||
|
* Residual is the legal property of its developers, whose names
|
||||||
|
* are too numerous to list here. Please refer to the AUTHORS
|
||||||
|
* file distributed with this source distribution.
|
||||||
|
*
|
||||||
|
* 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., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
||||||
|
*
|
||||||
|
* $URL$
|
||||||
|
* $Id$
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef BACKENDS_GRAPHICS_SDL_H
|
||||||
|
#define BACKENDS_GRAPHICS_SDL_H
|
||||||
|
|
||||||
|
#ifdef USE_OPENGL
|
||||||
|
#include <SDL_opengl.h>
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#include "backends/graphics/graphics.h"
|
||||||
|
#include "graphics/scaler.h"
|
||||||
|
#include "common/events.h"
|
||||||
|
#include "common/system.h"
|
||||||
|
|
||||||
|
#include "backends/events/sdl/sdl-events.h"
|
||||||
|
|
||||||
|
#include "backends/platform/sdl/sdl-sys.h"
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* SDL graphics manager
|
||||||
|
*/
|
||||||
|
class SdlGraphicsManager : public GraphicsManager, public Common::EventObserver {
|
||||||
|
public:
|
||||||
|
SdlGraphicsManager(SdlEventSource *sdlEventSource);
|
||||||
|
virtual ~SdlGraphicsManager();
|
||||||
|
|
||||||
|
virtual void initEventObserver();
|
||||||
|
|
||||||
|
virtual bool hasFeature(OSystem::Feature f);
|
||||||
|
virtual void setFeatureState(OSystem::Feature f, bool enable);
|
||||||
|
virtual bool getFeatureState(OSystem::Feature f);
|
||||||
|
|
||||||
|
virtual void launcherInitSize(uint w, uint h);
|
||||||
|
byte *setupScreen(int screenW, int screenH, bool fullscreen, bool accel3d);
|
||||||
|
virtual int16 getHeight();
|
||||||
|
virtual int16 getWidth();
|
||||||
|
|
||||||
|
public:
|
||||||
|
virtual void updateScreen();
|
||||||
|
|
||||||
|
virtual void showOverlay();
|
||||||
|
virtual void hideOverlay();
|
||||||
|
virtual Graphics::PixelFormat getOverlayFormat() const { return _overlayFormat; }
|
||||||
|
virtual void clearOverlay();
|
||||||
|
virtual void grabOverlay(OverlayColor *buf, int pitch);
|
||||||
|
virtual void copyRectToOverlay(const OverlayColor *buf, int pitch, int x, int y, int w, int h);
|
||||||
|
virtual int16 getOverlayHeight() { return _overlayHeight; }
|
||||||
|
virtual int16 getOverlayWidth() { return _overlayWidth; }
|
||||||
|
void closeOverlay();
|
||||||
|
|
||||||
|
virtual bool showMouse(bool visible);
|
||||||
|
virtual void warpMouse(int x, int y);
|
||||||
|
void setMouseCursor(const byte *buf, uint w, uint h, int hotspotX, int hotspotY, uint32 keycolor, int cursorTargetScale = 1, const Graphics::PixelFormat *format = NULL) {}
|
||||||
|
|
||||||
|
#ifdef USE_OSD
|
||||||
|
virtual void displayMessageOnOSD(const char *msg);
|
||||||
|
#endif
|
||||||
|
|
||||||
|
// Override from Common::EventObserver
|
||||||
|
bool notifyEvent(const Common::Event &event);
|
||||||
|
|
||||||
|
protected:
|
||||||
|
SdlEventSource *_sdlEventSource;
|
||||||
|
|
||||||
|
SDL_Surface *_screen;
|
||||||
|
|
||||||
|
#ifdef USE_OPENGL
|
||||||
|
bool _opengl;
|
||||||
|
#endif
|
||||||
|
bool _fullscreen;
|
||||||
|
|
||||||
|
// overlay
|
||||||
|
SDL_Surface *_overlayscreen;
|
||||||
|
bool _overlayVisible;
|
||||||
|
Graphics::PixelFormat _overlayFormat;
|
||||||
|
int _overlayWidth, _overlayHeight;
|
||||||
|
bool _overlayDirty;
|
||||||
|
#ifdef USE_OPENGL
|
||||||
|
int _overlayNumTex;
|
||||||
|
GLuint *_overlayTexIds;
|
||||||
|
#endif
|
||||||
|
|
||||||
|
/** Force full redraw on next updateScreen */
|
||||||
|
bool _forceFull;
|
||||||
|
};
|
||||||
|
|
||||||
|
#endif
|
@ -26,9 +26,9 @@
|
|||||||
|
|
||||||
#ifdef ENABLE_KEYMAPPER
|
#ifdef ENABLE_KEYMAPPER
|
||||||
|
|
||||||
#include "gui/GuiManager.h"
|
#include "gui/gui-manager.h"
|
||||||
#include "gui/PopUpWidget.h"
|
#include "gui/widgets/popup.h"
|
||||||
#include "gui/ScrollBarWidget.h"
|
#include "gui/widgets/scrollbar.h"
|
||||||
#include "gui/ThemeEval.h"
|
#include "gui/ThemeEval.h"
|
||||||
|
|
||||||
#include "common/translation.h"
|
#include "common/translation.h"
|
||||||
|
108
backends/log/log.cpp
Normal file
108
backends/log/log.cpp
Normal file
@ -0,0 +1,108 @@
|
|||||||
|
/* Residual - A 3D game interpreter
|
||||||
|
*
|
||||||
|
* Residual is the legal property of its developers, whose names
|
||||||
|
* are too numerous to list here. Please refer to the AUTHORS
|
||||||
|
* file distributed with this source distribution.
|
||||||
|
*
|
||||||
|
* 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., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
||||||
|
*
|
||||||
|
* $URL$
|
||||||
|
* $Id$
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include "backends/log/log.h"
|
||||||
|
|
||||||
|
#include "common/stream.h"
|
||||||
|
#include "common/str.h"
|
||||||
|
#include "common/system.h"
|
||||||
|
|
||||||
|
#include "base/version.h"
|
||||||
|
|
||||||
|
namespace Backends {
|
||||||
|
namespace Log {
|
||||||
|
|
||||||
|
Log::Log(OSystem *system)
|
||||||
|
: _system(system), _stream(0), _startOfLine(true) {
|
||||||
|
assert(system);
|
||||||
|
}
|
||||||
|
|
||||||
|
void Log::open(Common::WriteStream *stream) {
|
||||||
|
// Close the previous log
|
||||||
|
close();
|
||||||
|
|
||||||
|
_stream = stream;
|
||||||
|
|
||||||
|
// Output information about the ScummVM version at the start of the log
|
||||||
|
// file
|
||||||
|
print(gResidualFullVersion);
|
||||||
|
print("\n");
|
||||||
|
print(gResidualFeatures);
|
||||||
|
print("\n");
|
||||||
|
print("--- Log opened.\n");
|
||||||
|
_startOfLine = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
void Log::close() {
|
||||||
|
if (_stream) {
|
||||||
|
// Output a message to indicate that the log was closed successfully
|
||||||
|
print("--- Log closed successfully.\n");
|
||||||
|
|
||||||
|
delete _stream;
|
||||||
|
_stream = 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void Log::print(const char *message, const bool printTime) {
|
||||||
|
if (!_stream)
|
||||||
|
return;
|
||||||
|
|
||||||
|
while (*message) {
|
||||||
|
if (_startOfLine) {
|
||||||
|
_startOfLine = false;
|
||||||
|
if (printTime)
|
||||||
|
printTimeStamp();
|
||||||
|
}
|
||||||
|
|
||||||
|
const char *msgStart = message;
|
||||||
|
// scan for end of line/string
|
||||||
|
while (*message && *message != '\n')
|
||||||
|
++message;
|
||||||
|
|
||||||
|
if (*message == '\n') {
|
||||||
|
++message;
|
||||||
|
_startOfLine = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
// TODO: It might be wise to check for write errors and/or incomplete
|
||||||
|
// writes here, since losing certain bits of the log is not nice.
|
||||||
|
_stream->write(msgStart, message - msgStart);
|
||||||
|
}
|
||||||
|
|
||||||
|
_stream->flush();
|
||||||
|
}
|
||||||
|
|
||||||
|
void Log::printTimeStamp() {
|
||||||
|
TimeDate date;
|
||||||
|
_system->getTimeAndDate(date);
|
||||||
|
|
||||||
|
_stream->writeString(Common::String::format("[%d-%02d-%02d %02d:%02d:%02d] ",
|
||||||
|
date.tm_year + 1900, date.tm_mon, date.tm_mday,
|
||||||
|
date.tm_hour, date.tm_min, date.tm_sec));
|
||||||
|
}
|
||||||
|
|
||||||
|
} // End of namespace Log
|
||||||
|
} // End of namespace Backends
|
||||||
|
|
132
backends/log/log.h
Normal file
132
backends/log/log.h
Normal file
@ -0,0 +1,132 @@
|
|||||||
|
/* Residual - A 3D game interpreter
|
||||||
|
*
|
||||||
|
* Residual is the legal property of its developers, whose names
|
||||||
|
* are too numerous to list here. Please refer to the AUTHORS
|
||||||
|
* file distributed with this source distribution.
|
||||||
|
*
|
||||||
|
* 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., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
||||||
|
*
|
||||||
|
* $URL$
|
||||||
|
* $Id$
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef BACKENDS_LOG_LOG_H
|
||||||
|
#define BACKENDS_LOG_LOG_H
|
||||||
|
|
||||||
|
#include "common/sys.h"
|
||||||
|
|
||||||
|
class OSystem;
|
||||||
|
|
||||||
|
namespace Common {
|
||||||
|
class WriteStream;
|
||||||
|
} // End of namespace Common
|
||||||
|
|
||||||
|
namespace Backends {
|
||||||
|
namespace Log {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Log file writer.
|
||||||
|
*
|
||||||
|
* This can be used by the backends to implement file logging functionality.
|
||||||
|
*/
|
||||||
|
class Log {
|
||||||
|
public:
|
||||||
|
/**
|
||||||
|
* Constructor for the logger object.
|
||||||
|
*
|
||||||
|
* @param system The OSystem instance to use. Must be non-null.
|
||||||
|
*/
|
||||||
|
Log(OSystem *system);
|
||||||
|
~Log() { close(); }
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Opens a new log file.
|
||||||
|
*
|
||||||
|
* The previous log, which was handled by this logger, will be closed
|
||||||
|
* before the new stream is associated.
|
||||||
|
*
|
||||||
|
* The current implemention will always call flush after data is written
|
||||||
|
* to the log file. It might thus be wise to pass an unbuffered write
|
||||||
|
* stream here to avoid unnecessary overhead.
|
||||||
|
* @see Common::WriteStream::flush
|
||||||
|
*
|
||||||
|
* Calling open with stream being 0 is valid and will result in the same
|
||||||
|
* behavior as calling close, but it may have additional overhead.
|
||||||
|
* @see close
|
||||||
|
*
|
||||||
|
* This function will output information about the ScummVM version and
|
||||||
|
* the features built into ScummVM automatically. It will also add a short
|
||||||
|
* notice to indicate that the log was opened successfully.
|
||||||
|
*
|
||||||
|
* @param stream Stream where to output the log contents.
|
||||||
|
* Note that the stream will be deleted by the logger.
|
||||||
|
*/
|
||||||
|
void open(Common::WriteStream *stream);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Closes the current log file.
|
||||||
|
*
|
||||||
|
* This function will output a line saying that the log was closed
|
||||||
|
* successfully. This can be used to check whether a log is incomplete
|
||||||
|
* because of whatever reasons.
|
||||||
|
*/
|
||||||
|
void close();
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Prints a message to the log stream.
|
||||||
|
*
|
||||||
|
* This has optional support to output a timestamp on every new line.
|
||||||
|
* The timestamp will look like: "[YYYY-MM-DD HH:MM:SS] ".
|
||||||
|
* Printing of a timestamp is done by default.
|
||||||
|
*
|
||||||
|
* It might be noteworthy that this function does not append a new line
|
||||||
|
* to the given message.
|
||||||
|
*
|
||||||
|
* In case no stream is associated with this logger, this function will
|
||||||
|
* quit immediatly.
|
||||||
|
*
|
||||||
|
* @param message The message to write.
|
||||||
|
* @param printTimeOnNewline Whether to print a timestamp on the start of
|
||||||
|
* a new line.
|
||||||
|
*/
|
||||||
|
void print(const char *message, const bool printTimeOnNewline = true);
|
||||||
|
private:
|
||||||
|
/**
|
||||||
|
* Prints a time stamp in the form: "[YYYY-MM-DD HH:MM:SS] ".
|
||||||
|
*/
|
||||||
|
void printTimeStamp();
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The OSystem instance used to query data like the time.
|
||||||
|
*/
|
||||||
|
OSystem *_system;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Where to write the output too.
|
||||||
|
*/
|
||||||
|
Common::WriteStream *_stream;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Whether we are at the start of a line.
|
||||||
|
*/
|
||||||
|
bool _startOfLine;
|
||||||
|
};
|
||||||
|
|
||||||
|
} // End of namespace Log
|
||||||
|
} // End of namespace Backends
|
||||||
|
|
||||||
|
#endif
|
||||||
|
|
@ -22,14 +22,17 @@
|
|||||||
* $Id$
|
* $Id$
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
// Disable symbol overrides so that we can use system headers.
|
||||||
|
#define FORBIDDEN_SYMBOL_ALLOW_ALL
|
||||||
|
|
||||||
#include "common/sys.h"
|
#include "common/sys.h"
|
||||||
|
|
||||||
#if defined(USE_ALSA)
|
#if defined(USE_ALSA)
|
||||||
|
|
||||||
#include "common/config-manager.h"
|
#include "common/config-manager.h"
|
||||||
#include "common/util.h"
|
#include "common/util.h"
|
||||||
#include "sound/musicplugin.h"
|
#include "audio/musicplugin.h"
|
||||||
#include "sound/mpu401.h"
|
#include "audio/mpu401.h"
|
||||||
|
|
||||||
#include <alsa/asoundlib.h>
|
#include <alsa/asoundlib.h>
|
||||||
|
|
||||||
@ -41,7 +44,7 @@
|
|||||||
|
|
||||||
#if SND_LIB_MAJOR >= 1 || SND_LIB_MINOR >= 6
|
#if SND_LIB_MAJOR >= 1 || SND_LIB_MINOR >= 6
|
||||||
#define snd_seq_flush_output(x) snd_seq_drain_output(x)
|
#define snd_seq_flush_output(x) snd_seq_drain_output(x)
|
||||||
#define snd_seq_set_client_group(x,name) /*nop */
|
#define snd_seq_set_client_group(x,name) /*nop */
|
||||||
#define my_snd_seq_open(seqp) snd_seq_open(seqp, "hw", SND_SEQ_OPEN_DUPLEX, 0)
|
#define my_snd_seq_open(seqp) snd_seq_open(seqp, "hw", SND_SEQ_OPEN_DUPLEX, 0)
|
||||||
#else
|
#else
|
||||||
/* SND_SEQ_OPEN_OUT causes oops on early version of ALSA */
|
/* SND_SEQ_OPEN_OUT causes oops on early version of ALSA */
|
||||||
@ -50,9 +53,8 @@
|
|||||||
|
|
||||||
#define perm_ok(pinfo,bits) ((snd_seq_port_info_get_capability(pinfo) & (bits)) == (bits))
|
#define perm_ok(pinfo,bits) ((snd_seq_port_info_get_capability(pinfo) & (bits)) == (bits))
|
||||||
|
|
||||||
static int check_permission(snd_seq_port_info_t *pinfo)
|
static int check_permission(snd_seq_port_info_t *pinfo) {
|
||||||
{
|
if (perm_ok(pinfo, SND_SEQ_PORT_CAP_WRITE | SND_SEQ_PORT_CAP_SUBS_WRITE)) {
|
||||||
if (perm_ok(pinfo, SND_SEQ_PORT_CAP_WRITE|SND_SEQ_PORT_CAP_SUBS_WRITE)) {
|
|
||||||
if (!(snd_seq_port_info_get_capability(pinfo) & SND_SEQ_PORT_CAP_NO_EXPORT))
|
if (!(snd_seq_port_info_get_capability(pinfo) & SND_SEQ_PORT_CAP_NO_EXPORT))
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
@ -65,10 +67,11 @@ static int check_permission(snd_seq_port_info_t *pinfo)
|
|||||||
|
|
||||||
#define ADDR_DELIM ".:"
|
#define ADDR_DELIM ".:"
|
||||||
|
|
||||||
class MidiDriver_ALSA:public MidiDriver_MPU401 {
|
class MidiDriver_ALSA : public MidiDriver_MPU401 {
|
||||||
public:
|
public:
|
||||||
MidiDriver_ALSA(int client, int port);
|
MidiDriver_ALSA(int client, int port);
|
||||||
int open();
|
int open();
|
||||||
|
bool isOpen() const { return _isOpen; }
|
||||||
void close();
|
void close();
|
||||||
void send(uint32 b);
|
void send(uint32 b);
|
||||||
void sysEx(const byte *msg, uint16 length);
|
void sysEx(const byte *msg, uint16 length);
|
||||||
@ -80,11 +83,12 @@ private:
|
|||||||
snd_seq_t *seq_handle;
|
snd_seq_t *seq_handle;
|
||||||
int seq_client, seq_port;
|
int seq_client, seq_port;
|
||||||
int my_client, my_port;
|
int my_client, my_port;
|
||||||
|
// The volume controller value of the first MIDI channel
|
||||||
|
int8 _channel0Volume;
|
||||||
};
|
};
|
||||||
|
|
||||||
MidiDriver_ALSA::MidiDriver_ALSA(int client, int port)
|
MidiDriver_ALSA::MidiDriver_ALSA(int client, int port)
|
||||||
: _isOpen(false), seq_handle(0), seq_client(client), seq_port(port), my_client(0), my_port(0)
|
: _isOpen(false), seq_handle(0), seq_client(client), seq_port(port), my_client(0), my_port(0), _channel0Volume(127) {
|
||||||
{
|
|
||||||
memset(&ev, 0, sizeof(ev));
|
memset(&ev, 0, sizeof(ev));
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -111,7 +115,7 @@ int MidiDriver_ALSA::open() {
|
|||||||
// with those capabilities.
|
// with those capabilities.
|
||||||
|
|
||||||
my_port = snd_seq_create_simple_port(seq_handle, "RESIDUAL port 0", 0,
|
my_port = snd_seq_create_simple_port(seq_handle, "RESIDUAL port 0", 0,
|
||||||
SND_SEQ_PORT_TYPE_MIDI_GENERIC | SND_SEQ_PORT_TYPE_APPLICATION);
|
SND_SEQ_PORT_TYPE_MIDI_GENERIC | SND_SEQ_PORT_TYPE_APPLICATION);
|
||||||
|
|
||||||
if (my_port < 0) {
|
if (my_port < 0) {
|
||||||
snd_seq_close(seq_handle);
|
snd_seq_close(seq_handle);
|
||||||
@ -205,24 +209,41 @@ void MidiDriver_ALSA::send(uint32 b) {
|
|||||||
case 0xB0:
|
case 0xB0:
|
||||||
/* is it this simple ? Wow... */
|
/* is it this simple ? Wow... */
|
||||||
snd_seq_ev_set_controller(&ev, chanID, midiCmd[1], midiCmd[2]);
|
snd_seq_ev_set_controller(&ev, chanID, midiCmd[1], midiCmd[2]);
|
||||||
|
|
||||||
|
// We save the volume of the first MIDI channel here to utilize it in
|
||||||
|
// our workaround for broken USB-MIDI cables.
|
||||||
|
if (chanID == 0 && midiCmd[1] == 0x07)
|
||||||
|
_channel0Volume = midiCmd[2];
|
||||||
|
|
||||||
send_event(1);
|
send_event(1);
|
||||||
break;
|
break;
|
||||||
case 0xC0:
|
case 0xC0:
|
||||||
snd_seq_ev_set_pgmchange(&ev, chanID, midiCmd[1]);
|
snd_seq_ev_set_pgmchange(&ev, chanID, midiCmd[1]);
|
||||||
send_event(0);
|
send_event(0);
|
||||||
|
|
||||||
|
// Send a volume change command to work around a firmware bug in common
|
||||||
|
// USB-MIDI cables. If the first MIDI command in a USB packet is a
|
||||||
|
// Cx or Dx command, the second command in the packet is dropped
|
||||||
|
// somewhere.
|
||||||
|
send(0x07B0 | (_channel0Volume << 16));
|
||||||
break;
|
break;
|
||||||
case 0xD0:
|
case 0xD0:
|
||||||
snd_seq_ev_set_chanpress(&ev, chanID, midiCmd[1]);
|
snd_seq_ev_set_chanpress(&ev, chanID, midiCmd[1]);
|
||||||
send_event(1);
|
send_event(1);
|
||||||
|
|
||||||
|
// Send a volume change command to work around a firmware bug in common
|
||||||
|
// USB-MIDI cables. If the first MIDI command in a USB packet is a
|
||||||
|
// Cx or Dx command, the second command in the packet is dropped
|
||||||
|
// somewhere.
|
||||||
|
send(0x07B0 | (_channel0Volume << 16));
|
||||||
break;
|
break;
|
||||||
case 0xE0:{
|
case 0xE0: {
|
||||||
// long theBend = ((((long)midiCmd[1] + (long)(midiCmd[2] << 7))) - 0x2000) / 4;
|
// long theBend = ((((long)midiCmd[1] + (long)(midiCmd[2] << 7))) - 0x2000) / 4;
|
||||||
// snd_seq_ev_set_pitchbend(&ev, chanID, theBend);
|
// snd_seq_ev_set_pitchbend(&ev, chanID, theBend);
|
||||||
long theBend = ((long)midiCmd[1] + (long)(midiCmd[2] << 7)) - 0x2000;
|
long theBend = ((long)midiCmd[1] + (long)(midiCmd[2] << 7)) - 0x2000;
|
||||||
snd_seq_ev_set_pitchbend(&ev, chanID, theBend);
|
snd_seq_ev_set_pitchbend(&ev, chanID, theBend);
|
||||||
send_event(1);
|
send_event(1);
|
||||||
}
|
} break;
|
||||||
break;
|
|
||||||
|
|
||||||
default:
|
default:
|
||||||
warning("Unknown MIDI Command: %08x", (int)b);
|
warning("Unknown MIDI Command: %08x", (int)b);
|
||||||
@ -277,6 +298,9 @@ typedef Common::List<AlsaDevice> AlsaDevices;
|
|||||||
|
|
||||||
AlsaDevice::AlsaDevice(Common::String name, MusicType mt, int client)
|
AlsaDevice::AlsaDevice(Common::String name, MusicType mt, int client)
|
||||||
: _name(name), _type(mt), _client(client) {
|
: _name(name), _type(mt), _client(client) {
|
||||||
|
// Make sure we do not get any trailing spaces to avoid problems when
|
||||||
|
// storing the name in the configuration file.
|
||||||
|
_name.trim();
|
||||||
}
|
}
|
||||||
|
|
||||||
Common::String AlsaDevice::getName() {
|
Common::String AlsaDevice::getName() {
|
||||||
|
@ -22,14 +22,17 @@
|
|||||||
* $Id$
|
* $Id$
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
// Disable symbol overrides so that we can use system headers.
|
||||||
|
#define FORBIDDEN_SYMBOL_ALLOW_ALL
|
||||||
|
|
||||||
#include "common/sys.h"
|
#include "common/sys.h"
|
||||||
|
|
||||||
#if defined(__amigaos4__)
|
#if defined(__amigaos4__)
|
||||||
|
|
||||||
#include "common/endian.h"
|
#include "common/endian.h"
|
||||||
#include "common/util.h"
|
#include "common/util.h"
|
||||||
#include "sound/musicplugin.h"
|
#include "audio/musicplugin.h"
|
||||||
#include "sound/mpu401.h"
|
#include "audio/mpu401.h"
|
||||||
|
|
||||||
#include <proto/camd.h>
|
#include <proto/camd.h>
|
||||||
#include <proto/exec.h>
|
#include <proto/exec.h>
|
||||||
@ -43,6 +46,7 @@ class MidiDriver_CAMD : public MidiDriver_MPU401 {
|
|||||||
public:
|
public:
|
||||||
MidiDriver_CAMD();
|
MidiDriver_CAMD();
|
||||||
int open();
|
int open();
|
||||||
|
bool isOpen() const { return _isOpen; }
|
||||||
void close();
|
void close();
|
||||||
void send(uint32 b);
|
void send(uint32 b);
|
||||||
void sysEx(const byte *msg, uint16 length);
|
void sysEx(const byte *msg, uint16 length);
|
||||||
@ -79,7 +83,7 @@ int MidiDriver_CAMD::open() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
struct MidiNode *midi_node;
|
struct MidiNode *midi_node;
|
||||||
midi_node = _ICamd->CreateMidi(MIDI_MsgQueue, 0L, MIDI_SysExSize, 4096L, MIDI_Name, "scummvm", TAG_END);
|
midi_node = _ICamd->CreateMidi(MIDI_MsgQueue, 0L, MIDI_SysExSize, 4096L, MIDI_Name, "residual", TAG_END);
|
||||||
if (!midi_node) {
|
if (!midi_node) {
|
||||||
closeAll();
|
closeAll();
|
||||||
error("Could not create CAMD MIDI node");
|
error("Could not create CAMD MIDI node");
|
||||||
|
@ -22,6 +22,11 @@
|
|||||||
* $Id$
|
* $Id$
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
// Disable symbol overrides so that we can use system headers.
|
||||||
|
#define FORBIDDEN_SYMBOL_ALLOW_ALL
|
||||||
|
|
||||||
|
#include "common/sys.h"
|
||||||
|
|
||||||
#ifdef MACOSX
|
#ifdef MACOSX
|
||||||
|
|
||||||
// HACK to disable deprecated warnings under Mac OS X 10.5.
|
// HACK to disable deprecated warnings under Mac OS X 10.5.
|
||||||
@ -37,8 +42,8 @@
|
|||||||
|
|
||||||
#include "common/config-manager.h"
|
#include "common/config-manager.h"
|
||||||
#include "common/util.h"
|
#include "common/util.h"
|
||||||
#include "sound/musicplugin.h"
|
#include "audio/musicplugin.h"
|
||||||
#include "sound/mpu401.h"
|
#include "audio/mpu401.h"
|
||||||
|
|
||||||
#include <CoreServices/CoreServices.h>
|
#include <CoreServices/CoreServices.h>
|
||||||
#include <AudioToolbox/AUGraph.h>
|
#include <AudioToolbox/AUGraph.h>
|
||||||
@ -67,7 +72,9 @@ do { \
|
|||||||
class MidiDriver_CORE : public MidiDriver_MPU401 {
|
class MidiDriver_CORE : public MidiDriver_MPU401 {
|
||||||
public:
|
public:
|
||||||
MidiDriver_CORE();
|
MidiDriver_CORE();
|
||||||
|
~MidiDriver_CORE();
|
||||||
int open();
|
int open();
|
||||||
|
bool isOpen() const { return _auGraph != 0; }
|
||||||
void close();
|
void close();
|
||||||
void send(uint32 b);
|
void send(uint32 b);
|
||||||
void sysEx(const byte *msg, uint16 length);
|
void sysEx(const byte *msg, uint16 length);
|
||||||
@ -81,10 +88,18 @@ MidiDriver_CORE::MidiDriver_CORE()
|
|||||||
: _auGraph(0) {
|
: _auGraph(0) {
|
||||||
}
|
}
|
||||||
|
|
||||||
|
MidiDriver_CORE::~MidiDriver_CORE() {
|
||||||
|
if (_auGraph) {
|
||||||
|
AUGraphStop(_auGraph);
|
||||||
|
DisposeAUGraph(_auGraph);
|
||||||
|
_auGraph = 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
int MidiDriver_CORE::open() {
|
int MidiDriver_CORE::open() {
|
||||||
OSStatus err = 0;
|
OSStatus err = 0;
|
||||||
|
|
||||||
if (_auGraph)
|
if (isOpen())
|
||||||
return MERR_ALREADY_OPEN;
|
return MERR_ALREADY_OPEN;
|
||||||
|
|
||||||
// Open the Music Device.
|
// Open the Music Device.
|
||||||
@ -171,7 +186,6 @@ bail:
|
|||||||
|
|
||||||
void MidiDriver_CORE::close() {
|
void MidiDriver_CORE::close() {
|
||||||
MidiDriver_MPU401::close();
|
MidiDriver_MPU401::close();
|
||||||
|
|
||||||
if (_auGraph) {
|
if (_auGraph) {
|
||||||
AUGraphStop(_auGraph);
|
AUGraphStop(_auGraph);
|
||||||
DisposeAUGraph(_auGraph);
|
DisposeAUGraph(_auGraph);
|
||||||
@ -180,7 +194,7 @@ void MidiDriver_CORE::close() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
void MidiDriver_CORE::send(uint32 b) {
|
void MidiDriver_CORE::send(uint32 b) {
|
||||||
assert(_auGraph != NULL);
|
assert(isOpen());
|
||||||
|
|
||||||
byte status_byte = (b & 0x000000FF);
|
byte status_byte = (b & 0x000000FF);
|
||||||
byte first_byte = (b & 0x0000FF00) >> 8;
|
byte first_byte = (b & 0x0000FF00) >> 8;
|
||||||
@ -193,7 +207,7 @@ void MidiDriver_CORE::sysEx(const byte *msg, uint16 length) {
|
|||||||
unsigned char buf[266];
|
unsigned char buf[266];
|
||||||
|
|
||||||
assert(length + 2 <= ARRAYSIZE(buf));
|
assert(length + 2 <= ARRAYSIZE(buf));
|
||||||
assert(_auGraph != NULL);
|
assert(isOpen());
|
||||||
|
|
||||||
// Add SysEx frame
|
// Add SysEx frame
|
||||||
buf[0] = 0xF0;
|
buf[0] = 0xF0;
|
||||||
|
@ -22,12 +22,17 @@
|
|||||||
* $Id$
|
* $Id$
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
// Disable symbol overrides so that we can use system headers.
|
||||||
|
#define FORBIDDEN_SYMBOL_ALLOW_ALL
|
||||||
|
|
||||||
|
#include "common/sys.h"
|
||||||
|
|
||||||
#ifdef MACOSX
|
#ifdef MACOSX
|
||||||
|
|
||||||
#include "common/config-manager.h"
|
#include "common/config-manager.h"
|
||||||
#include "common/util.h"
|
#include "common/util.h"
|
||||||
#include "sound/musicplugin.h"
|
#include "audio/musicplugin.h"
|
||||||
#include "sound/mpu401.h"
|
#include "audio/mpu401.h"
|
||||||
|
|
||||||
#include <CoreMIDI/CoreMIDI.h>
|
#include <CoreMIDI/CoreMIDI.h>
|
||||||
|
|
||||||
@ -51,6 +56,7 @@ public:
|
|||||||
MidiDriver_CoreMIDI();
|
MidiDriver_CoreMIDI();
|
||||||
~MidiDriver_CoreMIDI();
|
~MidiDriver_CoreMIDI();
|
||||||
int open();
|
int open();
|
||||||
|
bool isOpen() const { return mOutPort != 0 && mDest != 0; }
|
||||||
void close();
|
void close();
|
||||||
void send(uint32 b);
|
void send(uint32 b);
|
||||||
void sysEx(const byte *msg, uint16 length);
|
void sysEx(const byte *msg, uint16 length);
|
||||||
@ -65,7 +71,7 @@ MidiDriver_CoreMIDI::MidiDriver_CoreMIDI()
|
|||||||
: mClient(0), mOutPort(0), mDest(0) {
|
: mClient(0), mOutPort(0), mDest(0) {
|
||||||
|
|
||||||
OSStatus err;
|
OSStatus err;
|
||||||
err = MIDIClientCreate(CFSTR("ScummVM MIDI Driver for OS X"), NULL, NULL, &mClient);
|
err = MIDIClientCreate(CFSTR("Residual MIDI Driver for OS X"), NULL, NULL, &mClient);
|
||||||
}
|
}
|
||||||
|
|
||||||
MidiDriver_CoreMIDI::~MidiDriver_CoreMIDI() {
|
MidiDriver_CoreMIDI::~MidiDriver_CoreMIDI() {
|
||||||
@ -75,7 +81,7 @@ MidiDriver_CoreMIDI::~MidiDriver_CoreMIDI() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
int MidiDriver_CoreMIDI::open() {
|
int MidiDriver_CoreMIDI::open() {
|
||||||
if (mDest)
|
if (isOpen())
|
||||||
return MERR_ALREADY_OPEN;
|
return MERR_ALREADY_OPEN;
|
||||||
|
|
||||||
OSStatus err = noErr;
|
OSStatus err = noErr;
|
||||||
@ -86,7 +92,7 @@ int MidiDriver_CoreMIDI::open() {
|
|||||||
if (dests > 0 && mClient) {
|
if (dests > 0 && mClient) {
|
||||||
mDest = MIDIGetDestination(0);
|
mDest = MIDIGetDestination(0);
|
||||||
err = MIDIOutputPortCreate( mClient,
|
err = MIDIOutputPortCreate( mClient,
|
||||||
CFSTR("scummvm_output_port"),
|
CFSTR("residual_output_port"),
|
||||||
&mOutPort);
|
&mOutPort);
|
||||||
} else {
|
} else {
|
||||||
return MERR_DEVICE_NOT_AVAILABLE;
|
return MERR_DEVICE_NOT_AVAILABLE;
|
||||||
@ -101,7 +107,7 @@ int MidiDriver_CoreMIDI::open() {
|
|||||||
void MidiDriver_CoreMIDI::close() {
|
void MidiDriver_CoreMIDI::close() {
|
||||||
MidiDriver_MPU401::close();
|
MidiDriver_MPU401::close();
|
||||||
|
|
||||||
if (mOutPort && mDest) {
|
if (isOpen()) {
|
||||||
MIDIPortDispose(mOutPort);
|
MIDIPortDispose(mOutPort);
|
||||||
mOutPort = 0;
|
mOutPort = 0;
|
||||||
mDest = 0;
|
mDest = 0;
|
||||||
@ -109,8 +115,7 @@ void MidiDriver_CoreMIDI::close() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
void MidiDriver_CoreMIDI::send(uint32 b) {
|
void MidiDriver_CoreMIDI::send(uint32 b) {
|
||||||
assert(mOutPort != 0);
|
assert(isOpen());
|
||||||
assert(mDest != 0);
|
|
||||||
|
|
||||||
// Extract the MIDI data
|
// Extract the MIDI data
|
||||||
byte status_byte = (b & 0x000000FF);
|
byte status_byte = (b & 0x000000FF);
|
||||||
@ -153,8 +158,7 @@ void MidiDriver_CoreMIDI::send(uint32 b) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
void MidiDriver_CoreMIDI::sysEx(const byte *msg, uint16 length) {
|
void MidiDriver_CoreMIDI::sysEx(const byte *msg, uint16 length) {
|
||||||
assert(mOutPort != 0);
|
assert(isOpen());
|
||||||
assert(mDest != 0);
|
|
||||||
|
|
||||||
byte buf[384];
|
byte buf[384];
|
||||||
MIDIPacketList *packetList = (MIDIPacketList *)buf;
|
MIDIPacketList *packetList = (MIDIPacketList *)buf;
|
||||||
|
@ -27,13 +27,18 @@
|
|||||||
* some code liberated from seq.cpp and coremidi.cpp
|
* some code liberated from seq.cpp and coremidi.cpp
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
// Disable symbol overrides so that we can use system headers.
|
||||||
|
#define FORBIDDEN_SYMBOL_ALLOW_ALL
|
||||||
|
|
||||||
|
#include "common/sys.h"
|
||||||
|
|
||||||
#if defined(IRIX)
|
#if defined(IRIX)
|
||||||
|
|
||||||
#include "common/sys.h"
|
#include "common/sys.h"
|
||||||
#include "common/util.h"
|
#include "common/util.h"
|
||||||
#include "common/config-manager.h"
|
#include "common/config-manager.h"
|
||||||
#include "sound/musicplugin.h"
|
#include "audio/musicplugin.h"
|
||||||
#include "sound/mpu401.h"
|
#include "audio/mpu401.h"
|
||||||
|
|
||||||
#include <dmedia/midi.h>
|
#include <dmedia/midi.h>
|
||||||
#include <sys/types.h>
|
#include <sys/types.h>
|
||||||
@ -52,6 +57,7 @@ class MidiDriver_DMEDIA : public MidiDriver_MPU401 {
|
|||||||
public:
|
public:
|
||||||
MidiDriver_DMEDIA();
|
MidiDriver_DMEDIA();
|
||||||
int open();
|
int open();
|
||||||
|
bool isOpen() const { return _isOpen; }
|
||||||
void close();
|
void close();
|
||||||
void send(uint32 b);
|
void send(uint32 b);
|
||||||
void sysEx(const byte *msg, uint16 length);
|
void sysEx(const byte *msg, uint16 length);
|
||||||
@ -87,8 +93,8 @@ int MidiDriver_DMEDIA::open() {
|
|||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (getenv("SCUMMVM_MIDIPORT")) {
|
if (getenv("RESIDUAL_MIDIPORT")) {
|
||||||
_deviceNum = atoi(getenv("SCUMMVM_MIDIPORT"));
|
_deviceNum = atoi(getenv("RESIDUAL_MIDIPORT"));
|
||||||
_midiportName = mdGetName(_deviceNum);
|
_midiportName = mdGetName(_deviceNum);
|
||||||
} else {
|
} else {
|
||||||
var = ConfMan.get("dmedia_port").c_str();
|
var = ConfMan.get("dmedia_port").c_str();
|
||||||
@ -174,7 +180,7 @@ void MidiDriver_DMEDIA::sysEx (const byte *msg, uint16 length) {
|
|||||||
memcpy(buf, msg, length);
|
memcpy(buf, msg, length);
|
||||||
buf[length] = MD_EOX;
|
buf[length] = MD_EOX;
|
||||||
event.sysexmsg = buf;
|
event.sysexmsg = buf;
|
||||||
event.msglen = length;
|
event.msglen = length;
|
||||||
event.msg[0] = MD_SYSEX;
|
event.msg[0] = MD_SYSEX;
|
||||||
event.msg[1] = 0;
|
event.msg[1] = 0;
|
||||||
event.msg[2] = 0;
|
event.msg[2] = 0;
|
||||||
|
@ -28,16 +28,20 @@
|
|||||||
* both the QuickTime support and (vkeybd http://www.alsa-project.org/~iwai/alsa.html)
|
* both the QuickTime support and (vkeybd http://www.alsa-project.org/~iwai/alsa.html)
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
// Disable symbol overrides so that we can use system headers.
|
||||||
|
#define FORBIDDEN_SYMBOL_ALLOW_ALL
|
||||||
|
|
||||||
#include "common/sys.h"
|
#include "common/sys.h"
|
||||||
|
|
||||||
#if defined(USE_SEQ_MIDI)
|
#if defined(USE_SEQ_MIDI)
|
||||||
|
|
||||||
#include "common/util.h"
|
#include "common/util.h"
|
||||||
#include "sound/musicplugin.h"
|
#include "audio/musicplugin.h"
|
||||||
#include "sound/mpu401.h"
|
#include "audio/mpu401.h"
|
||||||
|
|
||||||
#include <fcntl.h>
|
#include <fcntl.h>
|
||||||
#include <unistd.h>
|
#include <unistd.h>
|
||||||
|
#include <errno.h>
|
||||||
|
|
||||||
////////////////////////////////////////
|
////////////////////////////////////////
|
||||||
//
|
//
|
||||||
@ -51,6 +55,7 @@ class MidiDriver_SEQ : public MidiDriver_MPU401 {
|
|||||||
public:
|
public:
|
||||||
MidiDriver_SEQ();
|
MidiDriver_SEQ();
|
||||||
int open();
|
int open();
|
||||||
|
bool isOpen() const { return _isOpen; }
|
||||||
void close();
|
void close();
|
||||||
void send(uint32 b);
|
void send(uint32 b);
|
||||||
void sysEx(const byte *msg, uint16 length);
|
void sysEx(const byte *msg, uint16 length);
|
||||||
@ -75,14 +80,14 @@ int MidiDriver_SEQ::open() {
|
|||||||
_isOpen = true;
|
_isOpen = true;
|
||||||
device = 0;
|
device = 0;
|
||||||
|
|
||||||
device_name = getenv("SCUMMVM_MIDI");
|
device_name = getenv("RESIDUAL_MIDI");
|
||||||
|
|
||||||
if (device_name == NULL) {
|
if (device_name == NULL) {
|
||||||
warning("SCUMMVM_MIDI environment variable not set, using /dev/sequencer");
|
warning("RESIDUAL_MIDI environment variable not set, using /dev/sequencer");
|
||||||
device_name = dev_seq;
|
device_name = dev_seq;
|
||||||
}
|
}
|
||||||
|
|
||||||
device = (::open((device_name), O_RDWR, 0));
|
device = ::open((device_name), O_RDWR, 0);
|
||||||
|
|
||||||
if ((device_name == NULL) || (device < 0)) {
|
if ((device_name == NULL) || (device < 0)) {
|
||||||
if (device_name == NULL)
|
if (device_name == NULL)
|
||||||
@ -95,8 +100,8 @@ int MidiDriver_SEQ::open() {
|
|||||||
error("Cannot open /dev/null to dump midi output");
|
error("Cannot open /dev/null to dump midi output");
|
||||||
}
|
}
|
||||||
|
|
||||||
if (getenv("SCUMMVM_MIDIPORT"))
|
if (getenv("RESIDUAL_MIDIPORT"))
|
||||||
_device_num = atoi(getenv("SCUMMVM_MIDIPORT"));
|
_device_num = atoi(getenv("RESIDUAL_MIDIPORT"));
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -144,10 +149,11 @@ void MidiDriver_SEQ::send(uint32 b) {
|
|||||||
warning("MidiDriver_SEQ::send: unknown : %08x", (int)b);
|
warning("MidiDriver_SEQ::send: unknown : %08x", (int)b);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
(void)write(device, buf, position);
|
if (write(device, buf, position) == -1)
|
||||||
|
warning("MidiDriver_SEQ::send: write failed (%s)", strerror(errno));
|
||||||
}
|
}
|
||||||
|
|
||||||
void MidiDriver_SEQ::sysEx (const byte *msg, uint16 length) {
|
void MidiDriver_SEQ::sysEx(const byte *msg, uint16 length) {
|
||||||
unsigned char buf [266*4];
|
unsigned char buf [266*4];
|
||||||
int position = 0;
|
int position = 0;
|
||||||
const byte *chr = msg;
|
const byte *chr = msg;
|
||||||
@ -169,7 +175,8 @@ void MidiDriver_SEQ::sysEx (const byte *msg, uint16 length) {
|
|||||||
buf[position++] = _device_num;
|
buf[position++] = _device_num;
|
||||||
buf[position++] = 0;
|
buf[position++] = 0;
|
||||||
|
|
||||||
(void)write(device, buf, position);
|
if (write(device, buf, position) == -1)
|
||||||
|
warning("MidiDriver_SEQ::send: write failed (%s)", strerror(errno));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -34,17 +34,23 @@
|
|||||||
* cycles. I might change so sysex messages are sent the other way later.
|
* cycles. I might change so sysex messages are sent the other way later.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
// Disable symbol overrides so that we can use system headers.
|
||||||
|
#define FORBIDDEN_SYMBOL_ALLOW_ALL
|
||||||
|
|
||||||
|
#include "common/sys.h"
|
||||||
|
|
||||||
#if defined __MINT__
|
#if defined __MINT__
|
||||||
|
|
||||||
#include <osbind.h>
|
#include <osbind.h>
|
||||||
#include "sound/mpu401.h"
|
#include "audio/mpu401.h"
|
||||||
#include "common/util.h"
|
#include "common/util.h"
|
||||||
#include "sound/musicplugin.h"
|
#include "audio/musicplugin.h"
|
||||||
|
|
||||||
class MidiDriver_STMIDI : public MidiDriver_MPU401 {
|
class MidiDriver_STMIDI : public MidiDriver_MPU401 {
|
||||||
public:
|
public:
|
||||||
MidiDriver_STMIDI() : _isOpen (false) { }
|
MidiDriver_STMIDI() : _isOpen (false) { }
|
||||||
int open();
|
int open();
|
||||||
|
bool isOpen() const { return _isOpen; }
|
||||||
void close();
|
void close();
|
||||||
void send(uint32 b);
|
void send(uint32 b);
|
||||||
void sysEx(const byte *msg, uint16 length);
|
void sysEx(const byte *msg, uint16 length);
|
||||||
@ -54,7 +60,7 @@ private:
|
|||||||
};
|
};
|
||||||
|
|
||||||
int MidiDriver_STMIDI::open() {
|
int MidiDriver_STMIDI::open() {
|
||||||
if ((_isOpen) && (!Bcostat(4)))
|
if (_isOpen && (!Bcostat(4)))
|
||||||
return MERR_ALREADY_OPEN;
|
return MERR_ALREADY_OPEN;
|
||||||
warning("ST Midi Port Open");
|
warning("ST Midi Port Open");
|
||||||
_isOpen = true;
|
_isOpen = true;
|
||||||
@ -118,36 +124,36 @@ void MidiDriver_STMIDI::sysEx (const byte *msg, uint16 length) {
|
|||||||
|
|
||||||
class StMidiMusicPlugin : public MusicPluginObject {
|
class StMidiMusicPlugin : public MusicPluginObject {
|
||||||
public:
|
public:
|
||||||
const char *getName() const {
|
const char *getName() const {
|
||||||
return "STMIDI";
|
return "STMIDI";
|
||||||
}
|
}
|
||||||
|
|
||||||
const char *getId() const {
|
const char *getId() const {
|
||||||
return "stmidi";
|
return "stmidi";
|
||||||
}
|
}
|
||||||
|
|
||||||
MusicDevices getDevices() const;
|
MusicDevices getDevices() const;
|
||||||
Common::Error createInstance(MidiDriver **mididriver, MidiDriver::DeviceHandle = 0) const;
|
Common::Error createInstance(MidiDriver **mididriver, MidiDriver::DeviceHandle = 0) const;
|
||||||
};
|
};
|
||||||
|
|
||||||
MusicDevices StMidiMusicPlugin::getDevices() const {
|
MusicDevices StMidiMusicPlugin::getDevices() const {
|
||||||
MusicDevices devices;
|
MusicDevices devices;
|
||||||
// TODO: Return a different music type depending on the configuration
|
// TODO: Return a different music type depending on the configuration
|
||||||
// TODO: List the available devices
|
// TODO: List the available devices
|
||||||
devices.push_back(MusicDevice(this, "", MT_GM));
|
devices.push_back(MusicDevice(this, "", MT_GM));
|
||||||
return devices;
|
return devices;
|
||||||
}
|
}
|
||||||
|
|
||||||
Common::Error StMidiMusicPlugin::createInstance(MidiDriver **mididriver, MidiDriver::DeviceHandle) const {
|
Common::Error StMidiMusicPlugin::createInstance(MidiDriver **mididriver, MidiDriver::DeviceHandle) const {
|
||||||
*mididriver = new MidiDriver_STMIDI();
|
*mididriver = new MidiDriver_STMIDI();
|
||||||
|
|
||||||
return Common::kNoError;
|
return Common::kNoError;
|
||||||
}
|
}
|
||||||
|
|
||||||
//#if PLUGIN_ENABLED_DYNAMIC(STMIDI)
|
//#if PLUGIN_ENABLED_DYNAMIC(STMIDI)
|
||||||
//REGISTER_PLUGIN_DYNAMIC(STMIDI, PLUGIN_TYPE_MUSIC, StMidiMusicPlugin);
|
//REGISTER_PLUGIN_DYNAMIC(STMIDI, PLUGIN_TYPE_MUSIC, StMidiMusicPlugin);
|
||||||
//#else
|
//#else
|
||||||
REGISTER_PLUGIN_STATIC(STMIDI, PLUGIN_TYPE_MUSIC, StMidiMusicPlugin);
|
REGISTER_PLUGIN_STATIC(STMIDI, PLUGIN_TYPE_MUSIC, StMidiMusicPlugin);
|
||||||
//#endif
|
//#endif
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
@ -34,13 +34,18 @@
|
|||||||
*
|
*
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#if defined (UNIX)
|
// Disable symbol overrides so that we can use system headers.
|
||||||
|
#define FORBIDDEN_SYMBOL_ALLOW_ALL
|
||||||
|
|
||||||
|
#include "common/sys.h"
|
||||||
|
|
||||||
|
#if defined(USE_TIMIDITY)
|
||||||
|
|
||||||
#include "common/util.h"
|
#include "common/util.h"
|
||||||
#include "common/endian.h"
|
#include "common/endian.h"
|
||||||
#include "common/str.h"
|
#include "common/str.h"
|
||||||
#include "sound/musicplugin.h"
|
#include "audio/musicplugin.h"
|
||||||
#include "sound/mpu401.h"
|
#include "audio/mpu401.h"
|
||||||
|
|
||||||
#include <fcntl.h>
|
#include <fcntl.h>
|
||||||
#include <unistd.h>
|
#include <unistd.h>
|
||||||
@ -87,49 +92,50 @@ class MidiDriver_TIMIDITY : public MidiDriver_MPU401 {
|
|||||||
public:
|
public:
|
||||||
MidiDriver_TIMIDITY();
|
MidiDriver_TIMIDITY();
|
||||||
|
|
||||||
int open();
|
int open();
|
||||||
void close();
|
bool isOpen() const { return _isOpen; }
|
||||||
void send(uint32 b);
|
void close();
|
||||||
void sysEx(const byte *msg, uint16 length);
|
void send(uint32 b);
|
||||||
|
void sysEx(const byte *msg, uint16 length);
|
||||||
|
|
||||||
private:
|
private:
|
||||||
/* standart routine to extract ip address from a string */
|
/* standart routine to extract ip address from a string */
|
||||||
in_addr_t host_to_addr(const char* address);
|
in_addr_t host_to_addr(const char* address);
|
||||||
|
|
||||||
/* creates a tcp connection to TiMidity server, returns filedesc (like open()) */
|
/* creates a tcp connection to TiMidity server, returns filedesc (like open()) */
|
||||||
int connect_to_server(const char* hostname, unsigned short tcp_port);
|
int connect_to_server(const char* hostname, unsigned short tcp_port);
|
||||||
|
|
||||||
/* send command to the server; printf-like; returns reply string */
|
/* send command to the server; printf-like; returns reply string */
|
||||||
char *timidity_ctl_command(const char *fmt, ...) GCC_PRINTF(2, 3);
|
char *timidity_ctl_command(const char *fmt, ...) GCC_PRINTF(2, 3);
|
||||||
|
|
||||||
/* timidity data socket-related stuff */
|
/* timidity data socket-related stuff */
|
||||||
void timidity_meta_seq(int p1, int p2, int p3);
|
void timidity_meta_seq(int p1, int p2, int p3);
|
||||||
int timidity_sync(int centsec);
|
int timidity_sync(int centsec);
|
||||||
int timidity_eot();
|
int timidity_eot();
|
||||||
|
|
||||||
/* write() analogue for any midi data */
|
/* write() analogue for any midi data */
|
||||||
void timidity_write_data(const void *buf, size_t nbytes);
|
void timidity_write_data(const void *buf, size_t nbytes);
|
||||||
|
|
||||||
/* get single line of server reply on control connection */
|
/* get single line of server reply on control connection */
|
||||||
int fdgets(char *buff, size_t buff_size);
|
int fdgets(char *buff, size_t buff_size);
|
||||||
|
|
||||||
/* teardown connection to server */
|
/* teardown connection to server */
|
||||||
void teardown();
|
void teardown();
|
||||||
|
|
||||||
/* close (if needed) and nullify both control and data filedescs */
|
/* close (if needed) and nullify both control and data filedescs */
|
||||||
void close_all();
|
void close_all();
|
||||||
|
|
||||||
private:
|
private:
|
||||||
bool _isOpen;
|
bool _isOpen;
|
||||||
int _device_num;
|
int _device_num;
|
||||||
|
|
||||||
int _control_fd;
|
int _control_fd;
|
||||||
int _data_fd;
|
int _data_fd;
|
||||||
|
|
||||||
/* buffer for partial data read from _control_fd - from timidity-io.c, see fdgets() */
|
/* buffer for partial data read from _control_fd - from timidity-io.c, see fdgets() */
|
||||||
char _controlbuffer[BUFSIZ];
|
char _controlbuffer[BUFSIZ];
|
||||||
int _controlbuffer_count; /* beginning of read pointer */
|
int _controlbuffer_count; /* beginning of read pointer */
|
||||||
int _controlbuffer_size; /* end of read pointer */
|
int _controlbuffer_size; /* end of read pointer */
|
||||||
};
|
};
|
||||||
|
|
||||||
MidiDriver_TIMIDITY::MidiDriver_TIMIDITY() {
|
MidiDriver_TIMIDITY::MidiDriver_TIMIDITY() {
|
||||||
@ -144,9 +150,9 @@ MidiDriver_TIMIDITY::MidiDriver_TIMIDITY() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
int MidiDriver_TIMIDITY::open() {
|
int MidiDriver_TIMIDITY::open() {
|
||||||
char *res;
|
char *res;
|
||||||
char timidity_host[MAXHOSTNAMELEN];
|
char timidity_host[MAXHOSTNAMELEN];
|
||||||
int timidity_port, data_port, i;
|
int timidity_port, data_port, i;
|
||||||
|
|
||||||
/* count ourselves open */
|
/* count ourselves open */
|
||||||
if (_isOpen)
|
if (_isOpen)
|
||||||
@ -228,8 +234,8 @@ int MidiDriver_TIMIDITY::open() {
|
|||||||
/*
|
/*
|
||||||
* From seq.cpp
|
* From seq.cpp
|
||||||
*/
|
*/
|
||||||
if (getenv("SCUMMVM_MIDIPORT"))
|
if (getenv("RESIDUAL_MIDIPORT"))
|
||||||
_device_num = atoi(getenv("SCUMMVM_MIDIPORT"));
|
_device_num = atoi(getenv("RESIDUAL_MIDIPORT"));
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
@ -332,7 +338,11 @@ char *MidiDriver_TIMIDITY::timidity_ctl_command(const char *fmt, ...) {
|
|||||||
buff[len++] = '\n';
|
buff[len++] = '\n';
|
||||||
|
|
||||||
/* write command to control socket */
|
/* write command to control socket */
|
||||||
(void)write(_control_fd, buff, len);
|
if (write(_control_fd, buff, len) == -1) {
|
||||||
|
warning("TiMidity: CONTROL WRITE FAILED (%s)", strerror(errno));
|
||||||
|
// TODO: Disable output?
|
||||||
|
//close_all();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
while (1) {
|
while (1) {
|
||||||
@ -418,10 +428,9 @@ void MidiDriver_TIMIDITY::timidity_write_data(const void *buf, size_t nbytes) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
int MidiDriver_TIMIDITY::fdgets(char *buff, size_t buff_size) {
|
int MidiDriver_TIMIDITY::fdgets(char *buff, size_t buff_size) {
|
||||||
int n, len, count, size;
|
int n, count, size;
|
||||||
char *buff_endp = buff + buff_size - 1, *pbuff, *beg;
|
char *buff_endp = buff + buff_size - 1, *pbuff, *beg;
|
||||||
|
|
||||||
len = 0;
|
|
||||||
count = _controlbuffer_count;
|
count = _controlbuffer_count;
|
||||||
size = _controlbuffer_size;
|
size = _controlbuffer_size;
|
||||||
pbuff = _controlbuffer;
|
pbuff = _controlbuffer;
|
||||||
|
@ -22,6 +22,11 @@
|
|||||||
* $Id$
|
* $Id$
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
// Disable symbol overrides so that we can use system headers.
|
||||||
|
#define FORBIDDEN_SYMBOL_ALLOW_ALL
|
||||||
|
|
||||||
|
#include "common/sys.h"
|
||||||
|
|
||||||
#if defined(WIN32) && !defined(_WIN32_WCE)
|
#if defined(WIN32) && !defined(_WIN32_WCE)
|
||||||
|
|
||||||
#define WIN32_LEAN_AND_MEAN
|
#define WIN32_LEAN_AND_MEAN
|
||||||
@ -29,8 +34,8 @@
|
|||||||
// winnt.h defines ARRAYSIZE, but we want our own one...
|
// winnt.h defines ARRAYSIZE, but we want our own one...
|
||||||
#undef ARRAYSIZE
|
#undef ARRAYSIZE
|
||||||
|
|
||||||
#include "sound/musicplugin.h"
|
#include "audio/musicplugin.h"
|
||||||
#include "sound/mpu401.h"
|
#include "audio/mpu401.h"
|
||||||
#include "common/config-manager.h"
|
#include "common/config-manager.h"
|
||||||
#include "common/translation.h"
|
#include "common/translation.h"
|
||||||
|
|
||||||
@ -56,6 +61,7 @@ private:
|
|||||||
public:
|
public:
|
||||||
MidiDriver_WIN(int deviceIndex) : _isOpen(false), _device(deviceIndex) { }
|
MidiDriver_WIN(int deviceIndex) : _isOpen(false), _device(deviceIndex) { }
|
||||||
int open();
|
int open();
|
||||||
|
bool isOpen() const { return _isOpen; }
|
||||||
void close();
|
void close();
|
||||||
void send(uint32 b);
|
void send(uint32 b);
|
||||||
void sysEx(const byte *msg, uint16 length);
|
void sysEx(const byte *msg, uint16 length);
|
||||||
@ -88,6 +94,8 @@ void MidiDriver_WIN::close() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
void MidiDriver_WIN::send(uint32 b) {
|
void MidiDriver_WIN::send(uint32 b) {
|
||||||
|
assert(_isOpen);
|
||||||
|
|
||||||
union {
|
union {
|
||||||
DWORD dwData;
|
DWORD dwData;
|
||||||
BYTE bData[4];
|
BYTE bData[4];
|
||||||
|
Some files were not shown because too many files have changed in this diff Show More
Loading…
x
Reference in New Issue
Block a user