From 2fcb37d0e956107438875df9dc2253453c885ceb Mon Sep 17 00:00:00 2001 From: Volker Fischer Date: Sun, 8 Mar 2009 21:38:38 +0000 Subject: [PATCH] some more JACK code, this should be a first working version -> not yet tested --- linux/sound.cpp | 151 +++++++++++++++++++++++++++++++++++++++++----- linux/sound.h | 18 ++++-- windows/sound.cpp | 2 +- 3 files changed, 150 insertions(+), 21 deletions(-) diff --git a/linux/sound.cpp b/linux/sound.cpp index 04f7d4b0..33dab110 100755 --- a/linux/sound.cpp +++ b/linux/sound.cpp @@ -15,7 +15,7 @@ # if USE_JACK CSound::CSound ( void (*fpNewCallback) ( CVector& psData, void* arg ), void* arg ) : - CSoundBase ( false, fpNewCallback, arg ) + CSoundBase ( true, fpNewCallback, arg ) { jack_status_t JackStatus; @@ -29,41 +29,160 @@ CSound::CSound ( void (*fpNewCallback) ( CVector& psData, void* arg ), // tell the JACK server to call "process()" whenever // there is work to be done jack_set_process_callback ( pJackClient, process, this ); + +// TEST check sample rate, if not correct, just fire error +if ( jack_get_sample_rate ( pJackClient ) != SND_CRD_SAMPLE_RATE ) +{ + throw CGenErr ( "Jack server sample rate is different from " + "required one" ); } + // create four ports + input_port_left = jack_port_register ( pJackClient, "input left", + JACK_DEFAULT_AUDIO_TYPE, JackPortIsInput, 0 ); + input_port_right = jack_port_register ( pJackClient, "input right", + JACK_DEFAULT_AUDIO_TYPE, JackPortIsInput, 0 ); + output_port_left = jack_port_register ( pJackClient, "output left", + JACK_DEFAULT_AUDIO_TYPE, JackPortIsOutput, 0 ); + output_port_right = jack_port_register ( pJackClient, "output right", + JACK_DEFAULT_AUDIO_TYPE, JackPortIsOutput, 0 ); +} + +void CSound::Start() +{ + const char** ports; + + // tell the JACK server that we are ready to roll + if ( jack_activate ( pJackClient ) ) + { + throw CGenErr ( "Cannot activate client" ); + } + + // connect the ports, note: you cannot do this before + // the client is activated, because we cannot allow + // connections to be made to clients that are not + // running + if ( ( ports = jack_get_ports ( pJackClient, NULL, NULL, + JackPortIsPhysical | JackPortIsOutput ) ) == NULL ) + { + throw CGenErr ( "Cannot find any physical capture ports" ); + } + + if ( !ports[1] ) + { + throw CGenErr ( "Cannot find enough physical capture ports" ); + } + + if ( jack_connect ( pJackClient, ports[0], jack_port_name ( input_port_left ) ) ) + { + throw CGenErr ( "Cannot connect input ports" ); + } + if ( jack_connect ( pJackClient, ports[1], jack_port_name ( input_port_right ) ) ) + { + throw CGenErr ( "Cannot connect input ports" ); + } + + free ( ports ); + + if ( ( ports = jack_get_ports ( pJackClient, NULL, NULL, + JackPortIsPhysical | JackPortIsInput ) ) == NULL ) + { + throw CGenErr ( "Cannot find any physical playback ports" ); + } + + if ( !ports[1] ) + { + throw CGenErr ( "Cannot find enough physical playback ports" ); + } + + if ( jack_connect ( pJackClient, jack_port_name ( output_port_left ), ports[0] ) ) + { + throw CGenErr ( "Cannot connect output ports" ); + } + if ( jack_connect ( pJackClient, jack_port_name ( output_port_right ), ports[1] ) ) + { + throw CGenErr ( "Cannot connect output ports" ); + } + + free ( ports ); + + // call base class + CSoundBase::Start(); +} + +void CSound::Stop() +{ + // deactivate client + jack_deactivate ( pJackClient ); + + // call base class + CSoundBase::Stop(); +} int CSound::Init ( const int iNewPrefMonoBufferSize ) { // TEST -return iNewPrefMonoBufferSize; +iJACKBufferSizeMono = jack_get_buffer_size ( pJackClient ); + + // init base clasee + CSoundBase::Init ( iJACKBufferSizeMono ); + + // set internal buffer size value and calculate stereo buffer size + iJACKBufferSizeStero = 2 * iJACKBufferSizeMono; + + // create memory for intermediate audio buffer + vecsTmpAudioSndCrdStereo.Init ( iJACKBufferSizeStero ); + +// TEST +return iJACKBufferSizeMono; } -bool CSound::Read ( CVector& psData ) -{ - - -} - -bool CSound::Write ( CVector& psData ) -{ - - - -} // JACK callbacks -------------------------------------------------------------- int CSound::process ( jack_nframes_t nframes, void* arg ) { + int i; CSound* pSound = reinterpret_cast ( arg ); + // get input data + jack_default_audio_sample_t* in_left = + (jack_default_audio_sample_t*) jack_port_get_buffer ( pSound->input_port_left, nframes ); + + jack_default_audio_sample_t* in_right = + (jack_default_audio_sample_t*) jack_port_get_buffer ( pSound->input_port_right, nframes ); + + // copy input data + for ( i = 0; i < pSound->iJACKBufferSizeMono; i++ ) + { + +// TODO better conversion from float to short + + pSound->vecsTmpAudioSndCrdStereo[2 * i] = (short) in_left[i]; + pSound->vecsTmpAudioSndCrdStereo[2 * i + 1] = (short) in_right[i]; + } + + // call processing callback function + pSound->Callback ( pSound->vecsTmpAudioSndCrdStereo ); + + // put output data + jack_default_audio_sample_t* out_left = + (jack_default_audio_sample_t*) jack_port_get_buffer ( pSound->output_port_left, nframes ); + + jack_default_audio_sample_t* out_right = + (jack_default_audio_sample_t*) jack_port_get_buffer ( pSound->output_port_right, nframes ); + + // copy output data + for ( i = 0; i < pSound->iJACKBufferSizeMono; i++ ) + { + out_left[i] = pSound->vecsTmpAudioSndCrdStereo[2 * i]; + out_right[i] = pSound->vecsTmpAudioSndCrdStereo[2 * i + 1]; + } return 0; // zero on success, non-zero on error } - - # else // Wave in ********************************************************************* void CSound::InitRecording() diff --git a/linux/sound.h b/linux/sound.h index 94d37354..5a9fe246 100755 --- a/linux/sound.h +++ b/linux/sound.h @@ -55,20 +55,30 @@ public: CSound ( void (*fpNewCallback) ( CVector& psData, void* arg ), void* arg ); virtual ~CSound() {} + virtual int Init ( const int iNewPrefMonoBufferSize ); + virtual void Start(); + virtual void Stop(); + // not implemented yet, always return one device and default string int GetNumDev() { return 1; } std::string GetDeviceName ( const int iDiD ) { return "wave mapper"; } std::string SetDev ( const int iNewDev ) { return ""; } // dummy int GetDev() { return 0; } - virtual int Init ( const int iNewPrefMonoBufferSize ); - virtual bool Read ( CVector& psData ); - virtual bool Write ( CVector& psData ); + // these variables should be protected but cannot since we want + // to access them from the callback function + CVector vecsTmpAudioSndCrdStereo; + int iJACKBufferSizeMono; + int iJACKBufferSizeStero; + + jack_port_t* input_port_left; + jack_port_t* input_port_right; + jack_port_t* output_port_left; + jack_port_t* output_port_right; protected: // callback static int process ( jack_nframes_t nframes, void* arg ); - jack_client_t* pJackClient; }; # else diff --git a/windows/sound.cpp b/windows/sound.cpp index ba0be53b..fc406cea 100755 --- a/windows/sound.cpp +++ b/windows/sound.cpp @@ -489,7 +489,7 @@ void CSound::bufferSwitch ( long index, ASIOBool processNow ) } // call processing callback function - pSound->Callback( vecsTmpAudioSndCrdStereo ); + pSound->Callback ( vecsTmpAudioSndCrdStereo ); // perform the processing for input and output for ( int i = 0; i < 2 * NUM_IN_OUT_CHANNELS; i++ ) // stereo