from __future__ import with_statement | |
import threading | |
import logging | |
import time | |
from ctypes import * | |
from JetUtils import OsWindows | |
# stream state | |
EAS_STATE_READY = 0 | |
EAS_STATE_PLAY = 1 | |
EAS_STATE_STOPPING = 2 | |
EAS_STATE_PAUSING = 3 | |
EAS_STATE_STOPPED = 4 | |
EAS_STATE_PAUSED = 5 | |
EAS_STATE_OPEN = 6 | |
EAS_STATE_ERROR = 7 | |
EAS_STATE_EMPTY = 8 | |
# EAS error codes | |
EAS_SUCCESS = 0 | |
EAS_FAILURE = -1 | |
EAS_ERROR_INVALID_MODULE = -2 | |
EAS_ERROR_MALLOC_FAILED = -3 | |
EAS_ERROR_FILE_POS = -4 | |
EAS_ERROR_INVALID_FILE_MODE = -5 | |
EAS_ERROR_FILE_SEEK = -6 | |
EAS_ERROR_FILE_LENGTH = -7 | |
EAS_ERROR_NOT_IMPLEMENTED = -8 | |
EAS_ERROR_CLOSE_FAILED = -9 | |
EAS_ERROR_FILE_OPEN_FAILED = -10 | |
EAS_ERROR_INVALID_HANDLE = -11 | |
EAS_ERROR_NO_MIX_BUFFER = -12 | |
EAS_ERROR_PARAMETER_RANGE = -13 | |
EAS_ERROR_MAX_FILES_OPEN = -14 | |
EAS_ERROR_UNRECOGNIZED_FORMAT = -15 | |
EAS_BUFFER_SIZE_MISMATCH = -16 | |
EAS_ERROR_FILE_FORMAT = -17 | |
EAS_ERROR_SMF_NOT_INITIALIZED = -18 | |
EAS_ERROR_LOCATE_BEYOND_END = -19 | |
EAS_ERROR_INVALID_PCM_TYPE = -20 | |
EAS_ERROR_MAX_PCM_STREAMS = -21 | |
EAS_ERROR_NO_VOICE_ALLOCATED = -22 | |
EAS_ERROR_INVALID_CHANNEL = -23 | |
EAS_ERROR_ALREADY_STOPPED = -24 | |
EAS_ERROR_FILE_READ_FAILED = -25 | |
EAS_ERROR_HANDLE_INTEGRITY = -26 | |
EAS_ERROR_MAX_STREAMS_OPEN = -27 | |
EAS_ERROR_INVALID_PARAMETER = -28 | |
EAS_ERROR_FEATURE_NOT_AVAILABLE = -29 | |
EAS_ERROR_SOUND_LIBRARY = -30 | |
EAS_ERROR_NOT_VALID_IN_THIS_STATE = -31 | |
EAS_ERROR_NO_VIRTUAL_SYNTHESIZER = -32 | |
EAS_ERROR_FILE_ALREADY_OPEN = -33 | |
EAS_ERROR_FILE_ALREADY_CLOSED = -34 | |
EAS_ERROR_INCOMPATIBLE_VERSION = -35 | |
EAS_ERROR_QUEUE_IS_FULL = -36 | |
EAS_ERROR_QUEUE_IS_EMPTY = -37 | |
EAS_ERROR_FEATURE_ALREADY_ACTIVE = -38 | |
# special result codes | |
EAS_EOF = 3 | |
EAS_STREAM_BUFFERING = 4 | |
# buffer full error returned from Render | |
EAS_BUFFER_FULL = 5 | |
# file types | |
file_types = ( | |
'Unknown', | |
'SMF Type 0 (.mid)', | |
'SMF Type 1 (.mid)', | |
'SMAF - Unknown type (.mmf)', | |
'SMAF MA-2 (.mmf)', | |
'SMAF MA-3 (.mmf)', | |
'SMAF MA-5 (.mmf)', | |
'CMX/QualComm (.pmd)', | |
'MFi (NTT/DoCoMo i-mode)', | |
'OTA/Nokia (.ott)', | |
'iMelody (.imy)', | |
'RTX/RTTTL (.rtx)', | |
'XMF Type 0 (.xmf)', | |
'XMF Type 1 (.xmf)', | |
'WAVE/PCM (.wav)', | |
'WAVE/IMA-ADPCM (.wav)', | |
'MMAPI Tone Control (.js)' | |
) | |
stream_states = ( | |
'Ready', | |
'Play', | |
'Stopping', | |
'Stopped', | |
'Pausing', | |
'Paused', | |
'Open', | |
'Error', | |
'Empty' | |
) | |
# iMode play modes | |
IMODE_PLAY_ALL = 0 | |
IMODE_PLAY_PARTIAL = 1 | |
# callback type for metadata | |
EAS_METADATA_CBFUNC = CFUNCTYPE(c_int, c_int, c_char_p, c_ulong) | |
# callbacks for external audio | |
EAS_EXT_PRG_CHG_FUNC = CFUNCTYPE(c_int, c_void_p, c_void_p) | |
EAS_EXT_EVENT_FUNC = CFUNCTYPE(c_int, c_void_p, c_void_p) | |
# callback for aux mixer decoder | |
EAS_DECODER_FUNC = CFUNCTYPE(c_void_p, c_void_p, c_int, c_int) | |
# DLL path | |
if OsWindows(): | |
EAS_DLL_PATH = "EASDLL.dll" | |
else: | |
EAS_DLL_PATH = "libEASLIb.dylib" | |
eas_dll = None | |
# logger | |
eas_logger = None | |
#--------------------------------------------------------------- | |
# InitEASModule | |
#--------------------------------------------------------------- | |
def InitEASModule (dll_path=None): | |
global eas_dll | |
global eas_logger | |
# initialize logger | |
if eas_logger is None: | |
eas_logger = logging.getLogger('EAS') | |
# initialize path to DLL | |
if dll_path is None: | |
dll_path=EAS_DLL_PATH | |
# intialize DLL | |
if eas_dll is None: | |
eas_dll = cdll.LoadLibrary(dll_path) | |
#--------------------------------------------------------------- | |
# S_JET_CONFIG | |
#--------------------------------------------------------------- | |
class S_JET_CONFIG (Structure): | |
_fields_ = [('appLowNote', c_ubyte)] | |
#--------------------------------------------------------------- | |
# S_EXT_AUDIO_PRG_CHG | |
#--------------------------------------------------------------- | |
class S_EXT_AUDIO_PRG_CHG (Structure): | |
_fields_ = [('bank', c_ushort), | |
('program', c_ubyte), | |
('channel', c_ubyte)] | |
#--------------------------------------------------------------- | |
# S_EXT_AUDIO_EVENT | |
#--------------------------------------------------------------- | |
class S_EXT_AUDIO_EVENT (Structure): | |
_fields_ = [('channel', c_ubyte), | |
('note', c_ubyte), | |
('velocity', c_ubyte), | |
('noteOn', c_ubyte)] | |
#--------------------------------------------------------------- | |
# S_MIDI_CONTROLLERS | |
#--------------------------------------------------------------- | |
class S_MIDI_CONTROLLERS (Structure): | |
_fields_ = [('modWheel', c_ubyte), | |
('volume', c_ubyte), | |
('pan', c_ubyte), | |
('expression', c_ubyte), | |
('channelPressure', c_ubyte)] | |
#--------------------------------------------------------------- | |
# WAVEFORMAT | |
#--------------------------------------------------------------- | |
class WAVEFORMAT (Structure): | |
_fields_ = [('wFormatTag', c_ushort), | |
('nChannels', c_ushort), | |
('nSamplesPerSec', c_ulong), | |
('nAvgBytesPerSec', c_ulong), | |
('nBlockAlign', c_ushort), | |
('wBitsPerSample', c_ushort)] | |
#--------------------------------------------------------------- | |
# EAS_Exception | |
#--------------------------------------------------------------- | |
class EAS_Exception (Exception): | |
def __init__ (self, result_code, msg, function=None): | |
self.msg = msg | |
self.result_code = result_code | |
self.function = function | |
def __str__ (self): | |
return self.msg | |
#--------------------------------------------------------------- | |
# Log callback function | |
#--------------------------------------------------------------- | |
# map EAS severity levels to the Python logging module | |
severity_mapping = (logging.CRITICAL, logging.CRITICAL, logging.ERROR, logging.WARNING, logging.INFO, logging.DEBUG) | |
LOG_FUNC_TYPE = CFUNCTYPE(c_int, c_int, c_char_p) | |
def Log (level, msg): | |
eas_logger.log(severity_mapping[level], msg) | |
return level | |
LogCallback = LOG_FUNC_TYPE(Log) | |
#--------------------------------------------------------------- | |
# EAS_Stream | |
#--------------------------------------------------------------- | |
class EAS_Stream (object): | |
def __init__ (self, handle, eas): | |
eas_logger.debug('EAS_Stream.__init__') | |
self.handle = handle | |
self.eas = eas | |
def SetVolume (self, volume): | |
"""Set the stream volume""" | |
eas_logger.debug('Call EAS_SetVolume: volume=%d' % volume) | |
with self.eas.lock: | |
result = eas_dll.EAS_SetVolume(self.eas.handle, self.handle, volume) | |
if result: | |
raise EAS_Exception(result, 'EAS_SetVolume error %d on file %s' % (result, self.path), 'EAS_SetVolume') | |
def GetVolume (self): | |
"""Get the stream volume.""" | |
eas_logger.debug('Call EAS_GetVolume') | |
with self.eas.lock: | |
volume = eas_dll.EAS_GetVolume(self.eas.handle, self.handle) | |
if volume < 0: | |
raise EAS_Exception(volume, 'EAS_GetVolume error %d on file %s' % (volume, self.path), 'EAS_GetVolume') | |
eas_logger.debug('EAS_GetVolume: volume=%d' % volume) | |
return volume | |
def SetPriority (self, priority): | |
"""Set the stream priority""" | |
eas_logger.debug('Call EAS_SetPriority: priority=%d' % priority) | |
with self.eas.lock: | |
result = eas_dll.EAS_SetPriority(self.eas.handle, self.handle, priority) | |
if result: | |
raise EAS_Exception(result, 'EAS_SetPriority error %d on file %s' % (result, self.path), 'EAS_SetPriority') | |
def GetPriority (self): | |
"""Get the stream priority.""" | |
eas_logger.debug('Call EAS_GetPriority') | |
priority = c_int(0) | |
with self.eas.lock: | |
result = eas_dll.EAS_GetPriority(self.eas.handle, self.handle, byref(priority)) | |
if result: | |
raise EAS_Exception(result, 'EAS_GetPriority error %d on file %s' % (result, self.path), 'EAS_GetPriority') | |
eas_logger.debug('EAS_GetPriority: priority=%d' % priority.value) | |
return priority.value | |
def SetTransposition (self, transposition): | |
"""Set the transposition of a stream.""" | |
eas_logger.debug('Call EAS_SetTransposition: transposition=%d' % transposition) | |
with self.eas.lock: | |
result = eas_dll.EAS_SetTransposition(self.eas.handle, self.handle, transposition) | |
if result: | |
raise EAS_Exception(result, 'EAS_SetTransposition error %d on file %s' % (result, self.path), 'EAS_SetTransposition') | |
def SetPolyphony (self, polyphony): | |
"""Set the polyphony of a stream.""" | |
eas_logger.debug('Call EAS_SetPolyphony: polyphony=%d' % polyphony) | |
with self.eas.lock: | |
result = eas_dll.EAS_SetPolyphony(self.eas.handle, self.handle, polyphony) | |
if result: | |
raise EAS_Exception(result, 'EAS_SetPolyphony error %d on file %s' % (result, self.path), 'EAS_SetPolyphony') | |
def GetPolyphony (self): | |
"""Get the polyphony of a stream.""" | |
eas_logger.debug('Call EAS_GetPolyphony') | |
polyphony = c_int(0) | |
with self.eas.lock: | |
result = eas_dll.EAS_GetPolyphony(self.eas.handle, self.handle, byref(polyphony)) | |
if result: | |
raise EAS_Exception(result, 'EAS_GetPolyphony error %d on file %s' % (result, self.path), 'EAS_GetPolyphony') | |
eas_logger.debug('EAS_SetPolyphony: polyphony=%d' % polyphony.value) | |
return polyphony.value | |
def SelectLib (self, test_lib=False): | |
eas_logger.debug('Call EAS_SelectLib: test_lib=%s' % test_lib) | |
with self.eas.lock: | |
result = eas_dll.EAS_SelectLib(self.eas.handle, self.handle, test_lib) | |
if result: | |
raise EAS_Exception(result, 'EAS_SelectLib error %d on file %s' % (result, self.path), 'EAS_SelectLib') | |
def LoadDLSCollection (self, path): | |
eas_logger.debug('Call EAS_LoadDLSCollection: lib_path=%d' % path) | |
with self.eas.lock: | |
result = eas_dll.EAS_LoadDLSCollection(self.eas.handle, self.handle, path) | |
if result: | |
raise EAS_Exception(result, 'EAS_LoadDLSCollection error %d on file %s lib %s' % (result, self.path, path), 'EAS_LoadDLSCollection') | |
def RegExtAudioCallback (self, user_data, prog_chg_func, event_func): | |
"""Register an external audio callback.""" | |
eas_logger.debug('Call EAS_RegExtAudioCallback') | |
if prog_chg_func is not None: | |
prog_chg_func = EAS_EXT_PRG_CHG_FUNC(prog_chg_func) | |
else: | |
prog_chg_func = 0 | |
if event_func is not None: | |
event_func = EAS_EXT_EVENT_FUNC(event_func) | |
else: | |
event_func = 0 | |
with self.eas.lock: | |
result = eas_dll.EAS_RegExtAudioCallback(self.eas.handle, self.handle, user_data, prog_chg_func, event_func) | |
if result: | |
raise EAS_Exception(result, 'EAS_RegExtAudioCallback error %d on file %s' % (result, self.path), 'EAS_RegExtAudioCallback') | |
def SetPlayMode (self, play_mode): | |
"""Set play mode on a stream.""" | |
eas_logger.debug('Call EAS_SetPlayMode: play_mode=%d' % play_mode) | |
with self.eas.lock: | |
result = eas_dll.EAS_SetPlayMode(self.eas.handle, self.handle, play_mode) | |
if result: | |
raise EAS_Exception(result, 'EAS_SetPlayMode error %d on file %s' % (result, self.path), 'EAS_SetPlayMode') | |
""" | |
EAS_PUBLIC EAS_RESULT EAS_GetMIDIControllers (EAS_DATA_HANDLE pEASData, EAS_HANDLE streamHandle, EAS_U8 channel, S_MIDI_CONTROLLERS *pControl); | |
""" | |
#--------------------------------------------------------------- | |
# EAS_File | |
#--------------------------------------------------------------- | |
class EAS_File (EAS_Stream): | |
def __init__ (self, path, handle, eas): | |
EAS_Stream.__init__(self, handle, eas) | |
eas_logger.debug('EAS_File.__init__') | |
self.path = path | |
self.prepared = False | |
def Prepare (self): | |
"""Prepare an audio file for playback""" | |
if self.prepared: | |
eas_logger.warning('Prepare already called on file %s' % self.path) | |
else: | |
with self.eas.lock: | |
eas_logger.debug('Call EAS_Prepare for file: %s' % self.path) | |
result = eas_dll.EAS_Prepare(self.eas.handle, self.handle) | |
if result: | |
raise EAS_Exception(result, 'EAS_Prepare error %d on file %s' % (result, self.path), 'EAS_Prepare') | |
self.prepared = True | |
def State (self): | |
"""Get stream state.""" | |
with self.eas.lock: | |
eas_logger.debug('Call EAS_State for file: %s' % self.path) | |
state = c_long(-1) | |
result = eas_dll.EAS_State(self.eas.handle, self.handle, byref(state)) | |
if result: | |
raise EAS_Exception(result, 'EAS_State error %d on file %s' % (result, self.path), 'EAS_State') | |
eas_logger.debug('EAS_State: file=%s, state=%s' % (self.path, stream_states[state.value])) | |
return state.value | |
def Close (self): | |
"""Close audio file.""" | |
if hasattr(self, 'handle'): | |
with self.eas.lock: | |
eas_logger.debug('Call EAS_CloseFile for file: %s' % self.path) | |
result = eas_dll.EAS_CloseFile(self.eas.handle, self.handle) | |
if result: | |
raise EAS_Exception(result, 'EAS_CloseFile error %d on file %s' % (result, self.path), 'EAS_CloseFile') | |
# remove file from the EAS object | |
self.eas.audio_streams.remove(self) | |
# clean up references | |
del self.handle | |
del self.eas | |
del self.path | |
def Pause (self): | |
"""Pause a stream.""" | |
eas_logger.debug('Call EAS_Pause') | |
with self.eas.lock: | |
result = eas_dll.EAS_Pause(self.eas.handle, self.handle) | |
if result: | |
raise EAS_Exception(result, 'EAS_Pause error %d on file %s' % (result, self.path), 'EAS_Pause') | |
def Resume (self): | |
"""Resume a stream.""" | |
eas_logger.debug('Call EAS_Resume') | |
with self.eas.lock: | |
result = eas_dll.EAS_Resume(self.eas.handle, self.handle) | |
if result: | |
raise EAS_Exception(result, 'EAS_Resume error %d on file %s' % (result, self.path), 'EAS_Resume') | |
def Locate (self, secs, offset=False): | |
"""Set the playback position of a stream in seconds.""" | |
eas_logger.debug('Call EAS_Locate: location=%.3f, relative=%s' % (secs, offset)) | |
with self.eas.lock: | |
result = eas_dll.EAS_Locate(self.eas.handle, self.handle, int(secs * 1000 + 0.5), offset) | |
if result: | |
raise EAS_Exception(result, 'EAS_Locate error %d on file %s' % (result, self.path), 'EAS_Locate') | |
def GetLocation (self): | |
"""Get the stream location in seconds.""" | |
eas_logger.debug('Call EAS_GetLocation') | |
msecs = c_int(0) | |
with self.eas.lock: | |
result = eas_dll.EAS_GetLocation(self.eas.handle, self.handle, byref(msecs)) | |
if result: | |
raise EAS_Exception(result, 'EAS_GetLocation error %d on file %s' % (result, self.path), 'EAS_GetLocation') | |
msecs = float(msecs.value) / 1000 | |
eas_logger.debug('EAS_GetLocation: location=%.3f' % msecs) | |
return msecs | |
def GetFileType (self): | |
"""Get the file type.""" | |
eas_logger.debug('Call EAS_GetFileType') | |
file_type = c_int(0) | |
with self.eas.lock: | |
result = eas_dll.EAS_GetFileType(self.eas.handle, self.handle, byref(file_type)) | |
if result: | |
raise EAS_Exception(result, 'EAS_GetFileType error %d on file %s' % (result, self.path), 'EAS_GetFileType') | |
file_type = file_type.value | |
if file_type < len(file_types): | |
file_desc = file_types[file_type] | |
else: | |
file_desc = 'Unrecognized type %d' % file_type | |
eas_logger.debug('EAS_GetFileType: type=%d, desc=%s' % (file_type, file_desc)) | |
return (file_type, file_desc) | |
def SetRepeat (self, count): | |
"""Set the repeat count of a stream.""" | |
eas_logger.debug('Call EAS_SetRepeat: count=%d' % count) | |
with self.eas.lock: | |
result = eas_dll.EAS_SetRepeat(self.eas.handle, self.handle, count) | |
if result: | |
raise EAS_Exception(result, 'EAS_SetRepeat error %d on file %s' % (result, self.path), 'EAS_SetRepeat') | |
def GetRepeat (self): | |
"""Get the repeat count of a stream.""" | |
eas_logger.debug('Call EAS_GetRepeat') | |
count = c_int(0) | |
with self.eas.lock: | |
result = eas_dll.EAS_GetRepeat(self.eas.handle, self.handle, byref(count)) | |
if result: | |
raise EAS_Exception(result, 'EAS_GetRepeat error %d on file %s' % (result, self.path), 'EAS_GetRepeat') | |
eas_logger.debug('EAS_GetRepeat: count=%d' % count.value) | |
return count.value | |
def SetPlaybackRate (self, rate): | |
"""Set the playback rate of a stream.""" | |
eas_logger.debug('Call EAS_SetPlaybackRate') | |
with self.eas.lock: | |
result = eas_dll.EAS_SetPlaybackRate(self.eas.handle, self.handle, rate) | |
if result: | |
raise EAS_Exception(result, 'EAS_SetPlaybackRate error %d on file %s' % (result, self.path), 'EAS_SetPlaybackRate') | |
def ParseMetaData (self): | |
"""Parse the metadata in a file.""" | |
eas_logger.debug('Call EAS_ParseMetaData') | |
length = c_int(0) | |
with self.eas.lock: | |
result = eas_dll.EAS_ParseMetaData(self.eas.handle, self.handle, byref(length)) | |
if result: | |
raise EAS_Exception(result, 'EAS_ParseMetaData error %d on file %s' % (result, self.path), 'EAS_ParseMetaData') | |
return float(length.value) / 1000.0 | |
def RegisterMetaDataCallback (self, func, buf, buf_size, user_data): | |
"""Register a metadata callback.""" | |
eas_logger.debug('Call EAS_RegisterMetaDataCallback') | |
with self.eas.lock: | |
if func is not None: | |
callback = EAS_METADATA_CBFUNC(func) | |
else: | |
callback = 0 | |
result = eas_dll.EAS_RegisterMetaDataCallback(self.eas.handle, self.handle, callback, buf, buf_size, user_data) | |
if result: | |
raise EAS_Exception(result, 'EAS_RegisterMetaDataCallback error %d on file %s' % (result, self.path), 'EAS_RegisterMetaDataCallback') | |
def GetWaveFmtChunk (self): | |
"""Get the file type.""" | |
eas_logger.debug('Call EAS_GetWaveFmtChunk') | |
wave_fmt_chunk = c_void_p(0) | |
with self.eas.lock: | |
result = eas_dll.EAS_GetWaveFmtChunk(self.eas.handle, self.handle, byref(wave_fmt_chunk)) | |
if result: | |
raise EAS_Exception(result, 'EAS_GetWaveFmtChunk error %d on file %s' % (result, self.path), 'EAS_GetWaveFmtChunk') | |
return cast(wave_fmt_chunk, POINTER(WAVEFORMAT)).contents | |
def Play (self, max_time=None): | |
"""Plays the file to the end or max_time.""" | |
eas_logger.debug('EAS_File.Play') | |
if not self.prepared: | |
self.Prepare() | |
if max_time is not None: | |
max_time += self.eas.GetRenderTime() | |
while self.State() not in (EAS_STATE_STOPPED, EAS_STATE_ERROR, EAS_STATE_EMPTY): | |
self.eas.Render() | |
if max_time is not None: | |
if self.eas.GetRenderTime() >= max_time: | |
eas_logger.info('Max render time exceeded - stopping playback') | |
self.Pause() | |
self.eas.Render() | |
break | |
#--------------------------------------------------------------- | |
# EAS_MIDIStream | |
#--------------------------------------------------------------- | |
class EAS_MIDIStream (EAS_Stream): | |
def Write(self, data): | |
"""Write data to MIDI stream.""" | |
with self.eas.lock: | |
result = eas_dll.EAS_WriteMIDIStream(self.eas.handle, self.handle, data, len(data)) | |
if result: | |
raise EAS_Exception(result, 'EAS_WriteMIDIStream error %d' % result, 'EAS_WriteMIDIStream') | |
def Close (self): | |
"""Close MIDI stream.""" | |
if hasattr(self, 'handle'): | |
with self.eas.lock: | |
eas_logger.debug('Call EAS_CloseMIDIStream') | |
result = eas_dll.EAS_CloseMIDIStream(self.eas.handle, self.handle) | |
if result: | |
raise EAS_Exception(result, 'EAS_CloseFile error %d' % result, 'EAS_CloseMIDIStream') | |
# remove file from the EAS object | |
self.eas.audio_streams.remove(self) | |
# clean up references | |
del self.handle | |
del self.eas | |
#--------------------------------------------------------------- | |
# EAS_Config | |
#--------------------------------------------------------------- | |
class EAS_Config (Structure): | |
_fields_ = [('libVersion', c_ulong), | |
('checkedVersion', c_int), | |
('maxVoices', c_long), | |
('numChannels', c_long), | |
('sampleRate', c_long), | |
('mixBufferSize', c_long), | |
('filterEnabled', c_int), | |
('buildTimeStamp', c_ulong), | |
('buildGUID', c_char_p)] | |
#--------------------------------------------------------------- | |
# EAS | |
#--------------------------------------------------------------- | |
class EAS (object): | |
def __init__ (self, handle=None, dll_path=None, log_file=None): | |
if eas_dll is None: | |
InitEASModule(dll_path) | |
if log_file is not None: | |
eas_logger.addHandler(log_file) | |
eas_logger.debug('EAS.__init__') | |
self.Init(handle) | |
def __del__ (self): | |
eas_logger.debug('EAS.__del__') | |
self.Shutdown() | |
def Init (self, handle=None): | |
"""Initializes the EAS Library.""" | |
eas_logger.debug('EAS.Init') | |
# if we are already initialized, shutdown first | |
if hasattr(self, 'handle'): | |
eas_logger.debug('EAS.Init called with library already initalized') | |
self.ShutDown() | |
# setup the logging function | |
eas_dll.SetLogCallback(LogCallback) | |
# create some members | |
self.handle = c_void_p(0) | |
self.audio_streams = [] | |
self.output_streams = [] | |
self.aux_mixer = None | |
# create a sync lock | |
self.lock = threading.RLock() | |
with self.lock: | |
# set log callback | |
# get library configuration | |
self.Config() | |
# initialize library | |
if handle is None: | |
self.do_shutdown = True | |
eas_logger.debug('Call EAS_Init') | |
result = eas_dll.EAS_Init(byref(self.handle)) | |
if result: | |
raise EAS_Exception(result, 'EAS_Init error %d' % result, 'EAS_Init') | |
else: | |
self.do_shutdown = False | |
self.handle = handle | |
# allocate audio buffer for rendering | |
AudioBufferType = c_ubyte * (2 * self.config.mixBufferSize * self.config.numChannels) | |
self.audio_buffer = AudioBufferType() | |
self.buf_size = self.config.mixBufferSize | |
def Config (self): | |
"""Retrieves the EAS library configuration""" | |
if not hasattr(self, 'config'): | |
eas_logger.debug('Call EAS_Config') | |
eas_dll.EAS_Config.restype = POINTER(EAS_Config) | |
self.config = eas_dll.EAS_Config()[0] | |
eas_logger.debug("libVersion=%08x, maxVoices=%d, numChannels=%d, sampleRate = %d, mixBufferSize=%d" % | |
(self.config.libVersion, self.config.maxVoices, self.config.numChannels, self.config.sampleRate, self.config.mixBufferSize)) | |
def Shutdown (self): | |
"""Shuts down the EAS library""" | |
eas_logger.debug('EAS.Shutdown') | |
if hasattr(self, 'handle'): | |
with self.lock: | |
# close audio streams | |
audio_streams = self.audio_streams | |
for f in audio_streams: | |
eas_logger.warning('Stream was not closed before EAS_Shutdown') | |
f.Close() | |
# close output streams | |
output_streams = self.output_streams | |
for s in output_streams: | |
s.close() | |
# shutdown library | |
if self.do_shutdown: | |
eas_logger.debug('Call EAS_Shutdown') | |
result = eas_dll.EAS_Shutdown(self.handle) | |
if result: | |
raise EAS_Exception(result, 'EAS_Shutdown error %d' % result, 'EAS_Shutdown') | |
del self.handle | |
def OpenFile (self, path): | |
"""Opens an audio file to be played by the EAS library and | |
returns an EAS_File object | |
Arguments: | |
path - path to audio file | |
Returns: | |
EAS_File | |
""" | |
with self.lock: | |
eas_logger.debug('Call EAS_OpenFile for file: %s' % path) | |
stream_handle = c_void_p(0) | |
result = eas_dll.EAS_OpenFile(self.handle, path, byref(stream_handle)) | |
if result: | |
raise EAS_Exception(result, 'EAS_OpenFile error %d on file %s' % (result, path), 'EAS_OpenFile') | |
# create file object and save in list | |
stream = EAS_File(path, stream_handle, self) | |
self.audio_streams.append(stream) | |
return stream | |
def OpenMIDIStream (self, stream=None): | |
"""Opens a MIDI stream. | |
Arguments: | |
stream - open stream object. If None, a new synth | |
is created. | |
Returns: | |
EAS_MIDIStream | |
""" | |
with self.lock: | |
eas_logger.debug('Call EAS_OpenMIDIStream') | |
stream_handle = c_void_p(0) | |
if stream.handle is not None: | |
result = eas_dll.EAS_OpenMIDIStream(self.handle, byref(stream_handle), stream.handle) | |
else: | |
result = eas_dll.EAS_OpenMIDIStream(self.handle, byref(stream_handle), 0) | |
if result: | |
raise EAS_Exception(result, 'EAS_OpenMIDIStream error %d' % result, 'EAS_OpenMIDIStream') | |
# create stream object and save in list | |
stream = EAS_MIDIStream(stream_handle, self) | |
self.audio_streams.append(stream) | |
return stream | |
def OpenToneControlStream (self, path): | |
"""Opens an MMAPI tone control file to be played by the EAS | |
library and returns an EAS_File object | |
Arguments: | |
path - path to audio file | |
Returns: | |
EAS_File | |
""" | |
with self.lock: | |
eas_logger.debug('Call EAS_MMAPIToneControl for file: %s' % path) | |
stream_handle = c_void_p(0) | |
result = eas_dll.EAS_MMAPIToneControl(self.handle, path, byref(stream_handle)) | |
if result: | |
raise EAS_Exception(result, 'EAS_MMAPIToneControl error %d on file %s' % (result, path), 'EAS_OpenToneControlStream') | |
# create file object and save in list | |
stream = EAS_File(path, stream_handle, self) | |
self.audio_streams.append(stream) | |
return stream | |
def Attach (self, stream): | |
"""Attach a file or output device to the EAS output. | |
The stream object must support the following methods as | |
defined in the Python wave module: | |
close() | |
setparams() | |
writeframesraw() | |
Arguments: | |
stream - open wave object | |
""" | |
self.output_streams.append(stream) | |
stream.setparams((self.config.numChannels, 2, self.config.sampleRate, 0, 'NONE', None)) | |
def Detach (self, stream): | |
"""Detach a file or output device from the EAS output. See | |
EAS.Attach for more details. It is the responsibility of | |
the caller to close the wave file or stream. | |
Arguments: | |
stream - open and attached wave object | |
""" | |
self.output_streams.remove(stream) | |
def StartWave (self, dev_num=0, sampleRate=None, maxBufSize=None): | |
"""Route the audio output to the indicated wave device. Note | |
that this can cause EASDLL.EAS_RenderWaveOut to return an | |
error code if all the output buffers are full. In this case, | |
the render thread should sleep a bit and try again. | |
Unfortunately, due to the nature of the MMSYSTEM interface, | |
there is no simple way to suspend the render thread. | |
""" | |
if sampleRate == None: | |
sampleRate = self.config.sampleRate | |
if maxBufSize == None: | |
maxBufSize = self.config.mixBufferSize | |
with self.lock: | |
result = eas_dll.OpenWaveOutDevice(dev_num, sampleRate, maxBufSize) | |
if result: | |
raise EAS_Exception(result, 'OpenWaveOutDevice error %d' % result, 'OpenWaveOutDevice') | |
def StopWave (self): | |
"""Stop routing audio output to the audio device.""" | |
with self.lock: | |
result = eas_dll.CloseWaveOutDevice() | |
if result: | |
raise EAS_Exception(result, 'CloseWaveOutDevice error %d' % result, 'CloseWaveOutDevice') | |
def Render (self, count=None, secs=None): | |
"""Calls EAS_Render to render audio. | |
Arguments | |
count - number of buffers to render | |
secs - number of seconds to render | |
If both count and secs are None, render a single buffer. | |
""" | |
# determine number of buffers to render | |
if count is None: | |
if secs is not None: | |
count = int(secs * float(self.config.sampleRate) / float(self.buf_size) + 0.5) | |
else: | |
count = 1 | |
# render buffers | |
eas_logger.debug('rendering %d buffers' % count) | |
samplesRendered = c_long(0) | |
with self.lock: | |
for c in range(count): | |
# render a buffer of audio | |
eas_logger.debug('rendering buffer') | |
while 1: | |
if self.aux_mixer is None: | |
result = eas_dll.EAS_RenderWaveOut(self.handle, byref(self.audio_buffer), self.buf_size, byref(samplesRendered)) | |
else: | |
result = eas_dll.EAS_RenderAuxMixer(self.handle, byref(self.audio_buffer), byref(samplesRendered)) | |
if result == 0: | |
break; | |
if result == EAS_BUFFER_FULL: | |
time.sleep(0.01) | |
else: | |
raise EAS_Exception(result, 'EAS_Render error %d' % result, 'EAS_Render') | |
# output to attached streams | |
for s in self.output_streams: | |
s.writeframesraw(self.audio_buffer) | |
def GetRenderTime (self): | |
"""Get the render time in seconds.""" | |
eas_logger.debug('Call EAS_GetRenderTime') | |
msecs = c_int(0) | |
with self.lock: | |
result = eas_dll.EAS_GetRenderTime(self.handle, byref(msecs)) | |
if result: | |
raise EAS_Exception(result, 'EAS_GetRenderTime error %d' % result, 'EAS_GetRenderTime') | |
msecs = float(msecs.value) / 1000 | |
eas_logger.debug('EAS_GetRenderTime: time=%.3f' % msecs) | |
return msecs | |
def SetVolume (self, volume): | |
"""Set the master volume""" | |
eas_logger.debug('Call EAS_SetVolume: volume=%d' % volume) | |
with self.lock: | |
result = eas_dll.EAS_SetVolume(self.handle, 0, volume) | |
if result: | |
raise EAS_Exception(result, 'EAS_SetVolume error %d' % result, 'EAS_SetVolume') | |
def GetVolume (self): | |
"""Get the stream volume.""" | |
eas_logger.debug('Call EAS_GetVolume') | |
volume = c_int(0) | |
with self.lock: | |
result = eas_dll.EAS_GetVolume(self.handle, 0, byref(volume)) | |
if result: | |
raise EAS_Exception(result, 'EAS_GetVolume error %d' % result, 'EAS_GetVolume') | |
eas_logger.debug('EAS_GetVolume: volume=%d' % volume.value) | |
return volume.value | |
def SetPolyphony (self, polyphony, synth_num=0): | |
"""Set the polyphony of a synth.""" | |
eas_logger.debug('Call EAS_SetSynthPolyphony: synth_num=%d, polyphony=%d' % (synth_num, polyphony)) | |
with self.lock: | |
result = eas_dll.EAS_SetSynthPolyphony(self.handle, synth_num, polyphony) | |
if result: | |
raise EAS_Exception(result, 'EAS_SetSynthPolyphony error %d on synth %d' % (result, synth_num), 'EAS_SetPolyphony') | |
def GetPolyphony (self, synth_num=0): | |
"""Get the polyphony of a synth.""" | |
eas_logger.debug('Call EAS_GetSynthPolyphony: synth_num=%d' % synth_num) | |
polyphony = c_int(0) | |
with self.lock: | |
result = eas_dll.EAS_GetSynthPolyphony(self.handle, synth_num, byref(polyphony)) | |
if result: | |
raise EAS_Exception(result, 'EAS_GetSynthPolyphony error %d on synth %d' % (result, synth_num), 'EAS_GetPolyphony') | |
eas_logger.debug('Call EAS_GetSynthPolyphony: synth_num=%d, polyphony=%d' % (synth_num, polyphony.value)) | |
return polyphony.value | |
def SetMaxLoad (self, max_load): | |
"""Set the maximum parser load.""" | |
eas_logger.debug('Call EAS_SetMaxLoad: max_load=%d' % max_load) | |
with self.lock: | |
result = eas_dll.EAS_SetMaxLoad(self.handle, max_load) | |
if result: | |
raise EAS_Exception(result, 'EAS_SetMaxLoad error %d' % result, 'EAS_SetMaxLoad') | |
def SetParameter (self, module, param, value): | |
"""Set a module parameter.""" | |
eas_logger.debug('Call EAS_SetParameter: module=%d, param=%d, value=%d' % (module, param, value)) | |
with self.lock: | |
result = eas_dll.EAS_SetParameter(self.handle, module, param, value) | |
if result: | |
raise EAS_Exception(result, 'EAS_SetParameter error %d (param=%d, value=%d)' % (result, param, value), 'EAS_SetParameter') | |
def GetParameter (self, module, param): | |
"""Get the polyphony of a synth.""" | |
eas_logger.debug('Call EAS_GetParameter: module=%d, param=%d' % (module, param)) | |
value = c_int(0) | |
with self.lock: | |
result = eas_dll.EAS_GetParameter(self.handle, module, param, byref(value)) | |
if result: | |
raise EAS_Exception(result, 'EAS_SetParameter error %d (param=%d)' % (result, param), 'EAS_GetParameter') | |
eas_logger.debug('Call EAS_SetParameter: module=%d, param=%d, value=%d' % (module, param, value.value)) | |
return value.value | |
def SelectLib (self, test_lib=False): | |
eas_logger.debug('Call EAS_SelectLib: test_lib=%s' % test_lib) | |
easdll = cdll.LoadLibrary('EASDLL') | |
with self.lock: | |
result = eas_dll.EAS_SelectLib(self.handle, 0, test_lib) | |
if result: | |
raise EAS_Exception(result, 'EAS_SelectLib error %d' % result, 'EAS_SelectLib') | |
def LoadDLSCollection (self, path): | |
eas_logger.debug('Call EAS_LoadDLSCollection: lib_path=%s' % path) | |
with self.lock: | |
result = eas_dll.EAS_LoadDLSCollection(self.handle, 0, path) | |
if result: | |
raise EAS_Exception(result, 'EAS_LoadDLSCollection error %d lib %s' % (result, path), 'EAS_LoadDLSCollection') | |
def SetAuxMixerHook (self, aux_mixer): | |
# if aux mixer has bigger buffer, re-allocate buffer | |
if (aux_mixer is not None) and (aux_mixer.buf_size > self.config.mixBufferSize): | |
buf_size = aux_mixer.buf_size | |
else: | |
buf_size = self.config.mixBufferSize | |
# allocate audio buffer for rendering | |
AudioBufferType = c_ubyte * (2 * buf_size * self.config.numChannels) | |
self.audio_buffer = AudioBufferType() | |
self.buf_size = buf_size | |
self.aux_mixer = aux_mixer | |
def SetDebugLevel (self, level=3): | |
"""Sets the EAS debug level.""" | |
with self.lock: | |
eas_logger.debug('Call EAS_SetDebugLevel') | |
eas_dll.EAS_DLLSetDebugLevel(self.handle, level) | |
#--------------------------------------------------------------- | |
# EASAuxMixer | |
#--------------------------------------------------------------- | |
class EASAuxMixer (object): | |
def __init__ (self, eas=None, num_streams=3, sample_rate=44100, max_sample_rate=44100): | |
eas_logger.debug('EASAuxMixer.__init__') | |
self.Init(eas, num_streams, sample_rate, max_sample_rate) | |
def __del__ (self): | |
eas_logger.debug('EASAuxMixer.__del__') | |
self.Shutdown() | |
def Init (self, eas=None, num_streams=3, sample_rate=44100, max_sample_rate=44100): | |
"""Initializes the EAS Auxilliary Mixer.""" | |
eas_logger.debug('EASAuxMixer.Init') | |
if hasattr(self, 'eas'): | |
raise EAS_Exception(-1, 'EASAuxMixer already initialized', 'EASAuxMixer.Init') | |
# initialize EAS, if necessary | |
if eas is None: | |
eas_logger.debug('No EAS handle --- initializing EAS') | |
eas = EAS() | |
self.alloc_eas = True | |
else: | |
self.alloc_eas = False | |
self.eas = eas | |
# initialize library | |
eas_logger.debug('Call EAS_InitAuxMixer') | |
buf_size = c_int(0) | |
result = eas_dll.EAS_InitAuxMixer(eas.handle, num_streams, sample_rate, max_sample_rate, byref(buf_size)) | |
if result: | |
raise EAS_Exception(result, 'EAS_InitAuxMixer error %d' % result, 'EAS_InitAuxMixer') | |
self.buf_size = buf_size.value | |
self.streams = [] | |
eas.SetAuxMixerHook(self) | |
def Shutdown (self): | |
"""Shuts down the EAS Auxilliary Mixer""" | |
eas_logger.debug('EASAuxMixer.Shutdown') | |
if not hasattr(self, 'eas'): | |
return | |
with self.eas.lock: | |
if len(self.streams): | |
eas_logger.warning('Stream was not closed before EAS_ShutdownAuxMixer') | |
for stream in self.streams: | |
self.CloseStream(stream) | |
self.eas.SetAuxMixerHook(None) | |
# shutdown library | |
eas_logger.debug('Call EAS_ShutdownAuxMixer') | |
result = eas_dll.EAS_ShutdownAuxMixer(self.eas.handle) | |
if result: | |
raise EAS_Exception(result, 'EAS_ShutdownAuxMixer error %d' % result, 'EAS_ShutdownAuxMixer') | |
# if we created the EAS reference here, shut it down | |
if self.alloc_eas: | |
self.eas.Shutdown() | |
self.alloc_eas = False | |
del self.eas | |
def OpenStream (self, decoder_func, inst_data, sample_rate, num_channels): | |
"""Opens an audio file to be played by the JET library and | |
returns a JET_File object | |
Arguments: | |
callback - callback function to decode more audio | |
""" | |
with self.eas.lock: | |
eas_logger.debug('Call EAS_OpenAudioStream') | |
decoder_func = EAS_DECODER_FUNC(decoder_func) | |
stream_handle = c_void_p(0) | |
result = eas_dll.EAS_OpenAudioStream(self.eas.handle, decoder_func, inst_data, sample_rate, num_channels, stream_handle) | |
if result: | |
raise EAS_Exception(result, 'EAS_OpenAudioStream error %d on file %s' % (result, path), 'EAS_OpenAudioStream') | |
self.streams.add(stream_handle) | |
return stream_handle | |
def CloseStream (self, stream_handle): | |
"""Closes an open audio stream.""" | |
with self.eas.lock: | |
eas_logger.debug('Call EAS_CloseAudioStream') | |
result = eas_dll.JET_CloseFile(self.eas.handle, stream_handle) | |
if result: | |
raise EAS_Exception(result, 'EAS_CloseAudioStream error %d' % result, 'EAS_CloseAudioStream') | |
#--------------------------------------------------------------- | |
# JET_Status | |
#--------------------------------------------------------------- | |
class JET_Status (Structure): | |
_fields_ = [('currentUserID', c_int), | |
('segmentRepeatCount', c_int), | |
('numQueuedSegments', c_int), | |
('paused', c_int), | |
('location', c_long), | |
('currentPlayingSegment', c_int), | |
('currentQueuedSegment', c_int), | |
] | |
#--------------------------------------------------------------- | |
# JET_File | |
#--------------------------------------------------------------- | |
class JET_File (object): | |
def __init__ (self, handle, jet): | |
eas_logger.debug('JET_File.__init__') | |
self.handle = handle | |
self.jet = jet | |
#--------------------------------------------------------------- | |
# JET | |
#--------------------------------------------------------------- | |
class JET (object): | |
def __init__ (self, eas=None): | |
# eas_logger.debug('JET.__init__') | |
self.Init(eas) | |
def __del__ (self): | |
eas_logger.debug('JET.__del__') | |
self.Shutdown() | |
def Init (self, eas=None, config=None): | |
"""Initializes the JET Library.""" | |
# eas_logger.debug('JET.Init') | |
if hasattr(self, 'eas'): | |
raise EAS_Exception(-1, 'JET library already initialized', 'Jet.Init') | |
# create some members | |
if eas is None: | |
# eas_logger.debug('No EAS handle --- initializing EAS') | |
eas = EAS() | |
self.alloc_eas = True | |
else: | |
self.alloc_eas = False | |
self.eas = eas | |
self.fileOpen = False | |
# handle configuration | |
if config is None: | |
config_handle = c_void_p(0) | |
config_size = 0 | |
else: | |
jet_config = S_JET_CONFIG() | |
jet_config.appLowNote = config.appLowNote | |
config_handle = c_void_p(jet_config) | |
config_size = jet_config.sizeof() | |
# initialize library | |
# eas_logger.debug('Call JET_Init') | |
result = eas_dll.JET_Init(eas.handle, config_handle, config_size) | |
if result: | |
raise EAS_Exception(result, 'JET_Init error %d' % result, 'JET_Init') | |
def Shutdown (self): | |
"""Shuts down the JET library""" | |
eas_logger.debug('JET.Shutdown') | |
if not hasattr(self, 'eas'): | |
return | |
with self.eas.lock: | |
if self.fileOpen: | |
eas_logger.warning('Stream was not closed before JET_Shutdown') | |
self.CloseFile() | |
# shutdown library | |
eas_logger.debug('Call JET_Shutdown') | |
result = eas_dll.JET_Shutdown(self.eas.handle) | |
if result: | |
raise EAS_Exception(result, 'JET_Shutdown error %d' % result, 'JET_Shutdown') | |
# if we created the EAS reference here, shut it down | |
if self.alloc_eas: | |
self.eas.Shutdown() | |
self.alloc_eas = False | |
del self.eas | |
def OpenFile (self, path): | |
"""Opens an audio file to be played by the JET library and | |
returns a JET_File object | |
Arguments: | |
path - path to audio file | |
""" | |
with self.eas.lock: | |
eas_logger.debug('Call JET_OpenFile for file: %s' % path) | |
result = eas_dll.JET_OpenFile(self.eas.handle, path) | |
if result: | |
raise EAS_Exception(result, 'JET_OpenFile error %d on file %s' % (result, path), 'JET_OpenFile') | |
def CloseFile (self): | |
"""Closes an open audio file.""" | |
with self.eas.lock: | |
eas_logger.debug('Call JET_CloseFile') | |
result = eas_dll.JET_CloseFile(self.eas.handle) | |
if result: | |
raise EAS_Exception(result, 'JET_CloseFile error %d' % result, 'JET_CloseFile') | |
def QueueSegment (self, userID, seg_num, dls_num=-1, repeat=0, tranpose=0, mute_flags=0): | |
"""Queue a segment for playback. | |
Arguments: | |
seg_num - segment number to queue | |
repeat - repeat count (-1=repeat forever, 0=no repeat, 1+ = play n+1 times) | |
tranpose - transpose amount (+/-12) | |
""" | |
with self.eas.lock: | |
eas_logger.debug('Call JET_QueueSegment') | |
result = eas_dll.JET_QueueSegment(self.eas.handle, seg_num, dls_num, repeat, tranpose, mute_flags, userID) | |
if result: | |
raise EAS_Exception(result, 'JET_QueueSegment error %d' % result, 'JET_QueueSegment') | |
def Clear_Queue(self): | |
"""Kills the queue.""" | |
with self.eas.lock: | |
eas_logger.debug('Call JET_Clear_Queue') | |
result = eas_dll.JET_Clear_Queue(self.eas.handle) | |
if result: | |
raise EAS_Exception(result, 'JET_Clear_Queue error %d' % result, 'JET_Clear_Queue') | |
def GetAppEvent(self): | |
"""Gets an App event.""" | |
with self.eas.lock: | |
eas_logger.debug('Call JET_GetEvent') | |
result = eas_dll.JET_GetEvent(self.eas.handle, 0, 0) | |
return result | |
def Play(self): | |
"""Starts JET playback.""" | |
with self.eas.lock: | |
eas_logger.debug('Call JET_Play') | |
result = eas_dll.JET_Play(self.eas.handle) | |
if result: | |
raise EAS_Exception(result, 'JET_Play error %d' % result, 'JET_Play') | |
def Pause(self): | |
"""Pauses JET playback.""" | |
with self.eas.lock: | |
eas_logger.debug('Call JET_Pause') | |
result = eas_dll.JET_Pause(self.eas.handle) | |
if result: | |
raise EAS_Exception(result, 'JET_Pause error %d' % result, 'JET_Pause') | |
def Render (self, count=None, secs=None): | |
"""Calls EAS_Render to render audio. | |
Arguments | |
count - number of buffers to render | |
secs - number of seconds to render | |
If both count and secs are None, render a single buffer. | |
""" | |
# calls JET.Render | |
with self.eas.lock: | |
self.eas.Render(count, secs) | |
def Status (self): | |
"""Get JET status.""" | |
with self.eas.lock: | |
eas_logger.debug('Call JET_Status') | |
status = JET_Status() | |
result = eas_dll.JET_Status(self.eas.handle, byref(status)) | |
if result: | |
raise EAS_Exception(result, 'JET_Status error %d' % result, 'JET_Status') | |
eas_logger.debug("currentUserID=%d, repeatCount=%d, numQueuedSegments=%d, paused=%d" % | |
(status.currentUserID, status.segmentRepeatCount, status.numQueuedSegments, status.paused)) | |
return status | |
def SetVolume (self, volume): | |
"""Set the JET volume""" | |
eas_logger.debug('Call JET_SetVolume') | |
with self.eas.lock: | |
result = eas_dll.JET_SetVolume(self.eas.handle, volume) | |
if result: | |
raise EAS_Exception(result, 'JET_SetVolume error %d' % result, 'JET_SetVolume') | |
def SetTransposition (self, transposition): | |
"""Set the transposition of a stream.""" | |
eas_logger.debug('Call JET_SetTransposition') | |
with self.eas.lock: | |
result = eas_dll.JET_SetTransposition(self.eas.handle, transposition) | |
if result: | |
raise EAS_Exception(result, 'JET_SetTransposition error %d' % result, 'JET_SetTransposition') | |
def TriggerClip (self, clipID): | |
"""Trigger a clip in the current segment.""" | |
eas_logger.debug('Call JET_TriggerClip') | |
with self.eas.lock: | |
result = eas_dll.JET_TriggerClip(self.eas.handle, clipID) | |
if result: | |
raise EAS_Exception(result, 'JET_SetTransposition error %d' % result, 'JET_TriggerClip') | |
def SetMuteFlag (self, track_num, mute, sync=True): | |
"""Trigger a clip in the current segment.""" | |
eas_logger.debug('Call JET_SetMuteFlag') | |
with self.eas.lock: | |
result = eas_dll.JET_SetMuteFlag(self.eas.handle, track_num, mute, sync) | |
if result: | |
raise EAS_Exception(result, 'JET_SetMuteFlag error %d' % result, 'JET_SetMuteFlag') | |
def SetMuteFlags (self, mute_flags, sync=True): | |
"""Trigger a clip in the current segment.""" | |
eas_logger.debug('Call JET_SetMuteFlags') | |
with self.eas.lock: | |
result = eas_dll.JET_SetMuteFlags(self.eas.handle, mute_flags, sync) | |
if result: | |
raise EAS_Exception(result, 'JET_SetMuteFlag error %d' % result, 'JET_SetMuteFlags') | |