/*========================================================================== * * Copyright (C) 1999 Microsoft Corporation. All Rights Reserved. * * File: dvsndt.h * Content: definition of CSoundTarget class * * History: * Date By Reason * ==== == ====== * 09/02/99 rodtoll Created * 09/08/99 rodtoll Updated to provide lockup detection * 09/14/99 rodtoll Added WriteAheadSilence() * 09/20/99 rodtoll Added handlers for buffer loss * 11/12/99 rodtoll Updated to use new abstractions for playback (allows use * of waveOut with this class). * 01/24/2000 rodtoll Fix: Bug #129427 - Destroying transport before calling Delete3DSound * 01/27/2000 rodtoll Bug #129934 - Update SoundTargets to take DSBUFFERDESC * 02/17/2000 rodtoll Bug #133691 - Choppy audio - queue was not adapting * Added instrumentation * 04/17/2000 rodtoll Fix: Bug #32215 - Session Lost after resuming from hibernation * 06/21/2000 rodtoll Fix: Bug #35767 - Must implement ability for dsound effects on voice buffers * Added new constructor/init that takes pre-built buffers * 07/09/2000 rodtoll Added signature bytes * ***************************************************************************/ #ifndef __DVSNDT_H #define __DVSNDT_H // CSoundTarget // // This class represents a single mixer target within the DirectPlayClient system. Normally there // is only a single mixer target ("main") for all incoming audio. However, using the CreateUserBuffer and // DeleteUserBuffer APIs the developer can specify that they wish to seperate a group or a player // from the main mixer target. In this manner they can control the 3d spatialization of the group/player's // incoming audio stream. // // This class handles all the details related to a mixer target. It encapsulates the mixing of single // or multiple source audio frames and then commiting them to the corresponding directsound buffer. It also // handles timing errors in the directsoundbuffer. // // For example, if the directsoundbuffer stops running, it will attempt to reset the buffer. // // If the directsoundbuffer slows down (because of high CPU), it moves the read pointer forward. In short // it ensures that there is always 1 or 2 frames of mixed audio present in the buffer in advance of the // read pointer. // // In addition the class provides reference counting to prevent premature deletion of the class. If you // wish to take a reference to the class, call AddRef and you MUST then call Release when you are done. // // Do not destroy the object directly. When the last reference to the object is released the object will // destroy itself. // // This class is not multithread safe (except for AddRef and Release). Only one thread should be // accessing it. // #define VSIG_SOUNDTARGET 'TNSV' #define VSIG_SOUNDTARGET_FREE 'TNS_' volatile struct CSoundTarget { public: CSoundTarget( DVID dvidTarget, CAudioPlaybackDevice *lpads, LPDSBUFFERDESC lpdsBufferDesc, DWORD dwPriority, DWORD dwFlags, DWORD dwFrameSize ); CSoundTarget( DVID dvidTarget, CAudioPlaybackDevice *lpads, CAudioPlaybackBuffer *lpdsBuffer, LPDSBUFFERDESC lpdsBufferDesc, DWORD dwPriority, DWORD dwFlags, DWORD dwFrameSize ); CSoundTarget( DVID dvidTarget, CAudioPlaybackDevice *lpads, LPDIRECTSOUNDBUFFER lpdsBuffer, BOOL fEightBit, DWORD dwPriority, DWORD dwFlags, DWORD dwFrameSize ); ~CSoundTarget(); HRESULT StartMix(); HRESULT MixInSingle( LPBYTE lpbBuffer ); HRESULT MixIn( const BYTE* lpbBuffer ); HRESULT Commit(); inline HRESULT GetInitResult() const { return m_hrInitResult; }; LPDIRECTSOUND3DBUFFER Get3DBuffer(); inline CAudioPlaybackBuffer *GetBuffer() { return m_lpAudioPlaybackBuffer; }; CSoundTarget *m_lpstNext; // Next entry in the list inline DVID GetTarget() const { return m_dvidTarget; }; inline LONG GetRefCount() const { return m_lRefCount; }; LONG AddRef(); LONG Release(); void GetStats( PlaybackStats *statPlayback ); HRESULT GetCurrentLead( PDWORD pdwLead ); protected: HRESULT Initialize( DVID dvidTarget, CAudioPlaybackBuffer *lpdsBuffer, BOOL fEightBit, DWORD dwPriority, DWORD dwFlags, DWORD dwFrameSize ); HRESULT RestoreLostBuffer(); HRESULT AdjustWritePtr(); HRESULT WriteAheadSilence(); void Stats_Init(); void Stats_Begin(); void Stats_End(); public: DWORD m_dwSignature; protected: CAudioPlaybackBuffer *m_lpAudioPlaybackBuffer; LPDIRECTSOUND3DBUFFER m_lpds3dBuffer; // 3d buffer interface for this sound target DWORD m_dwNextWritePos; // Next byte position in the directsound buffer we'll // be writing to. DWORD m_dwBufferSize; // Size of directsound buffer in bytes BOOL m_bGroup; // Does this buffer represent a group? LPLONG m_lpMixBuffer; // High resolution mixing buffer BOOL m_bCommited; // Has the data in the latest mix been commited to the buffer BOOL m_fEightBit; // Is the buffer 8-bit? HRESULT m_hrInitResult; // Contains result of the initializatio of this object DWORD m_dwMixSize; // # of samples per frame DWORD m_dwFrameSize; // Size of a frame in bytes BOOL m_fMixed; // Set to TRUE as soon as a single source has been mixed // Set to FALSE whenever the mix is commited. LONG m_lRefCount; // Reference count on the object DWORD m_dwLastWritePos; // Byte position that the last write occured at. DWORD m_dwNumResets; // # of times buffer has been reset BOOL m_fLastFramePushed; // Did the last frame push the read pointer forward? DWORD m_dwNumSinceMove; // How many frames since no movement was detected? DWORD m_dwLastWriteTime; // GetTickCount() at last call to AdjustWritePtr BOOL m_fIgnoreFrame; // This frame will cross the boundary of where the write pointer is. // Do NOT write to the buffer DWORD m_dwWritePos; DNCRITICAL_SECTION m_csGuard; // Guard for reference counts DWORD m_dwPlayFlags; DWORD m_dwPriority; DVID m_dvidTarget; // DVID this buffer is for. // DVID_REMAINING for the global one PlaybackStats m_statPlay; // Playback Statistics LPVOID m_lpDummy; }; #endif