| 
							- /*
 - Copyright (C) 2006 Grame
 - 
 - Portable Audio I/O Library for ASIO Drivers
 - Author: Stephane Letz
 - Based on the Open Source API proposed by Ross Bencina
 - Copyright (c) 2000-2002 Stephane Letz, Phil Burk, Ross Bencina
 - 
 - 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., 675 Mass Ave, Cambridge, MA 02139, USA.
 - 
 - */
 - 
 - #include "pa_asio.h"
 - #include "JackDriverLoader.h"
 - #include "driver_interface.h"
 - 
 - #include "JackASIODriver.h"
 - #include "JackEngineControl.h"
 - #include "JackGraphManager.h"
 - #include "JackError.h"
 - #include "JackClientControl.h"
 - #include "JackGlobals.h"
 - #include <iostream>
 - 
 - 
 - #include <windows.h>
 - #include <mmsystem.h>
 - 
 - #include "asiosys.h"
 - #include "asio.h"
 - #include "asiodrivers.h"
 - #include "iasiothiscallresolver.h"
 - 
 - 
 - /* external references */
 - extern AsioDrivers* asioDrivers ;
 - bool loadAsioDriver(char *name);
 - 
 - 
 - namespace Jack
 - {
 - 
 - 
 - /*
 -     load the asio driver named by <driverName> and return statistics about
 -     the driver in info. If no error occurred, the driver will remain open
 -     and must be closed by the called by calling ASIOExit() - if an error
 -     is returned the driver will already be closed.
 - */
 - static PaError LoadAsioDriver( const char *driverName,
 -         PaAsioDriverInfo *driverInfo, void *systemSpecific )
 - {
 -     PaError result = paNoError;
 -     ASIOError asioError;
 -     int asioIsInitialized = 0;
 - 
 -     if( !loadAsioDriver(const_cast<char*>(driverName)))
 -     {
 -         result = paUnanticipatedHostError;
 -         PA_ASIO_SET_LAST_HOST_ERROR( 0, "Failed to load ASIO driver" );
 -         goto error;
 -     }
 - 
 -     memset( &driverInfo->asioDriverInfo, 0, sizeof(ASIODriverInfo) );
 -     driverInfo->asioDriverInfo.asioVersion = 2;
 -     driverInfo->asioDriverInfo.sysRef = systemSpecific;
 -     if( (asioError = ASIOInit( &driverInfo->asioDriverInfo )) != ASE_OK )
 -     {
 -         result = paUnanticipatedHostError;
 -         PA_ASIO_SET_LAST_ASIO_ERROR( asioError );
 -         goto error;
 -     }
 -     else
 -     {
 -         asioIsInitialized = 1;
 -     }
 - 
 -     if( (asioError = ASIOGetChannels(&driverInfo->inputChannelCount,
 -             &driverInfo->outputChannelCount)) != ASE_OK )
 -     {
 -         result = paUnanticipatedHostError;
 -         PA_ASIO_SET_LAST_ASIO_ERROR( asioError );
 -         goto error;
 -     }
 - 
 -     if( (asioError = ASIOGetBufferSize(&driverInfo->bufferMinSize,
 -             &driverInfo->bufferMaxSize, &driverInfo->bufferPreferredSize,
 -             &driverInfo->bufferGranularity)) != ASE_OK )
 -     {
 -         result = paUnanticipatedHostError;
 -         PA_ASIO_SET_LAST_ASIO_ERROR( asioError );
 -         goto error;
 -     }
 - 
 -     if( ASIOOutputReady() == ASE_OK )
 -         driverInfo->postOutput = true;
 -     else
 -         driverInfo->postOutput = false;
 - 
 -     return result;
 - 
 - error:
 -     if( asioIsInitialized )
 -         ASIOExit();
 - 
 -     return result;
 - }
 - 
 - 
 - int JackASIODriver::bufferSwitch(long index, ASIOBool directProcess)
 - {
 -     JackASIODriver* driver = (JackASIODriver*)userData;
 - 	
 - 	// the actual processing callback.
 -     // Beware that this is normally in a seperate thread, hence be sure that
 -     // you take care about thread synchronization. This is omitted here for
 -     // simplicity.
 - 
 -     // as this is a "back door" into the bufferSwitchTimeInfo a timeInfo needs
 -     // to be created though it will only set the timeInfo.samplePosition and
 -     // timeInfo.systemTime fields and the according flags
 - 
 -     ASIOTime  timeInfo;
 -     memset(&timeInfo, 0, sizeof(timeInfo));
 - 
 -     // get the time stamp of the buffer, not necessary if no
 -     // synchronization to other media is required
 -     if( ASIOGetSamplePosition(&timeInfo.timeInfo.samplePosition, &timeInfo.timeInfo.systemTime) == ASE_OK)
 -             timeInfo.timeInfo.flags = kSystemTimeValid | kSamplePositionValid;
 - 	
 - 	
 -     driver->fLastWaitUst = GetMicroSeconds(); // Take callback date here
 - 	driver->fInputBuffer = (float**)inputBuffer;
 - 	driver->fOutputBuffer = (float**)outputBuffer;
 - 	
 - 	 // Call the real callback
 -     bufferSwitchTimeInfo( &timeInfo, index, directProcess );
 - 	
 - 	return driver->Process();
 - }
 - 
 - int JackASIODriver::Read()
 - {
 - 	return 0;
 - }
 - 
 - int JackASIODriver::Write()
 - {
 -     return 0;
 - }
 - 
 - 
 - int JackASIODriver::Initialize( PaUtilHostApiRepresentation **hostApi, PaHostApiIndex hostApiIndex )
 - {
 -     PaError result = paNoError;
 -     int i, driverCount;
 -     PaAsioHostApiRepresentation *asioHostApi;
 -     PaAsioDeviceInfo *deviceInfoArray;
 -     char **names;
 -     PaAsioDriverInfo paAsioDriverInfo;
 - 
 -     asioHostApi = (PaAsioHostApiRepresentation*)PaUtil_AllocateMemory( sizeof(PaAsioHostApiRepresentation) );
 -     if( !asioHostApi )
 -     {
 -         result = paInsufficientMemory;
 -         goto error;
 -     }
 - 
 -     asioHostApi->allocations = PaUtil_CreateAllocationGroup();
 -     if( !asioHostApi->allocations )
 -     {
 -         result = paInsufficientMemory;
 -         goto error;
 -     }
 - 
 -     asioHostApi->systemSpecific = 0;
 -     asioHostApi->openAsioDeviceIndex = paNoDevice;
 - 
 -     *hostApi = &asioHostApi->inheritedHostApiRep;
 -     (*hostApi)->info.structVersion = 1;
 - 
 -     (*hostApi)->info.type = paASIO;
 -     (*hostApi)->info.name = "ASIO";
 -     (*hostApi)->info.deviceCount = 0;
 - 
 -     #ifdef WINDOWS
 -         /* use desktop window as system specific ptr */
 -         asioHostApi->systemSpecific = GetDesktopWindow();
 -         CoInitialize(NULL);
 -     #endif
 - 
 -     /* MUST BE CHECKED : to force fragments loading on Mac */
 -     loadAsioDriver( "dummy" ); 
 - 
 -     /* driverCount is the number of installed drivers - not necessarily
 -         the number of installed physical devices. */
 -     #if MAC
 -         driverCount = asioDrivers->getNumFragments();
 -     #elif WINDOWS
 -         driverCount = asioDrivers->asioGetNumDev();
 -     #endif
 - 
 -     if( driverCount > 0 )
 -     {
 -         names = GetAsioDriverNames( asioHostApi->allocations, driverCount );
 -         if( !names )
 -         {
 -             result = paInsufficientMemory;
 -             goto error;
 -         }
 - 
 - 
 -         /* allocate enough space for all drivers, even if some aren't installed */
 - 
 -         (*hostApi)->deviceInfos = (PaDeviceInfo**)PaUtil_GroupAllocateMemory(
 -                 asioHostApi->allocations, sizeof(PaDeviceInfo*) * driverCount );
 -         if( !(*hostApi)->deviceInfos )
 -         {
 -             result = paInsufficientMemory;
 -             goto error;
 -         }
 - 
 -         /* allocate all device info structs in a contiguous block */
 -         deviceInfoArray = (PaAsioDeviceInfo*)PaUtil_GroupAllocateMemory(
 -                 asioHostApi->allocations, sizeof(PaAsioDeviceInfo) * driverCount );
 -         if( !deviceInfoArray )
 -         {
 -             result = paInsufficientMemory;
 -             goto error;
 -         }
 - 
 -         for( i=0; i < driverCount; ++i )
 -         {
 - 
 -             PA_DEBUG(("ASIO names[%d]:%s\n",i,names[i]));
 - 
 -             // Since portaudio opens ALL ASIO drivers, and no one else does that,
 -             // we face fact that some drivers were not meant for it, drivers which act
 -             // like shells on top of REAL drivers, for instance.
 -             // so we get duplicate handles, locks and other problems.
 -             // so lets NOT try to load any such wrappers. 
 -             // The ones i [davidv] know of so far are:
 - 
 -             if (   strcmp (names[i],"ASIO DirectX Full Duplex Driver") == 0
 -                 || strcmp (names[i],"ASIO Multimedia Driver")          == 0
 -                 || strncmp(names[i],"Premiere",8)                      == 0   //"Premiere Elements Windows Sound 1.0"
 -                 || strncmp(names[i],"Adobe",5)                         == 0 ) //"Adobe Default Windows Sound 1.5"
 -             {
 -                 PA_DEBUG(("BLACKLISTED!!!\n"));
 -                 continue;
 -             }
 - 
 - 
 -             /* Attempt to load the asio driver... */
 -             if( LoadAsioDriver( names[i], &paAsioDriverInfo, asioHostApi->systemSpecific ) == paNoError )
 -             {
 -                 PaAsioDeviceInfo *asioDeviceInfo = &deviceInfoArray[ (*hostApi)->info.deviceCount ];
 -                 PaDeviceInfo *deviceInfo = &asioDeviceInfo->commonDeviceInfo;
 - 
 -                 deviceInfo->structVersion = 2;
 -                 deviceInfo->hostApi = hostApiIndex;
 - 
 -                 deviceInfo->name = names[i];
 -                 PA_DEBUG(("PaAsio_Initialize: drv:%d name = %s\n",  i,deviceInfo->name));
 -                 PA_DEBUG(("PaAsio_Initialize: drv:%d inputChannels       = %d\n", i, paAsioDriverInfo.inputChannelCount));
 -                 PA_DEBUG(("PaAsio_Initialize: drv:%d outputChannels      = %d\n", i, paAsioDriverInfo.outputChannelCount));
 -                 PA_DEBUG(("PaAsio_Initialize: drv:%d bufferMinSize       = %d\n", i, paAsioDriverInfo.bufferMinSize));
 -                 PA_DEBUG(("PaAsio_Initialize: drv:%d bufferMaxSize       = %d\n", i, paAsioDriverInfo.bufferMaxSize));
 -                 PA_DEBUG(("PaAsio_Initialize: drv:%d bufferPreferredSize = %d\n", i, paAsioDriverInfo.bufferPreferredSize));
 -                 PA_DEBUG(("PaAsio_Initialize: drv:%d bufferGranularity   = %d\n", i, paAsioDriverInfo.bufferGranularity));
 - 
 -                 deviceInfo->maxInputChannels  = paAsioDriverInfo.inputChannelCount;
 -                 deviceInfo->maxOutputChannels = paAsioDriverInfo.outputChannelCount;
 - 
 -                 deviceInfo->defaultSampleRate = 0.;
 -                 bool foundDefaultSampleRate = false;
 -                 for( int j=0; j < PA_DEFAULTSAMPLERATESEARCHORDER_COUNT_; ++j )
 -                 {
 -                     ASIOError asioError = ASIOCanSampleRate( defaultSampleRateSearchOrder_[j] );
 -                     if( asioError != ASE_NoClock && asioError != ASE_NotPresent )
 -                     {
 -                         deviceInfo->defaultSampleRate = defaultSampleRateSearchOrder_[j];
 -                         foundDefaultSampleRate = true;
 -                         break;
 -                     }
 -                 }
 - 
 -                 PA_DEBUG(("PaAsio_Initialize: drv:%d defaultSampleRate = %f\n", i, deviceInfo->defaultSampleRate));
 - 
 -                 if( foundDefaultSampleRate ){
 - 
 -                     /* calculate default latency values from bufferPreferredSize
 -                         for default low latency, and bufferPreferredSize * 3
 -                         for default high latency.
 -                         use the default sample rate to convert from samples to
 -                         seconds. Without knowing what sample rate the user will
 -                         use this is the best we can do.
 -                     */
 - 
 -                     double defaultLowLatency =
 -                             paAsioDriverInfo.bufferPreferredSize / deviceInfo->defaultSampleRate;
 - 
 -                     deviceInfo->defaultLowInputLatency = defaultLowLatency;
 -                     deviceInfo->defaultLowOutputLatency = defaultLowLatency;
 - 
 -                     long defaultHighLatencyBufferSize =
 -                             paAsioDriverInfo.bufferPreferredSize * 3;
 - 
 -                     if( defaultHighLatencyBufferSize > paAsioDriverInfo.bufferMaxSize )
 -                         defaultHighLatencyBufferSize = paAsioDriverInfo.bufferMaxSize;
 - 
 -                     double defaultHighLatency =
 -                             defaultHighLatencyBufferSize / deviceInfo->defaultSampleRate;
 - 
 -                     if( defaultHighLatency < defaultLowLatency )
 -                         defaultHighLatency = defaultLowLatency; /* just incase the driver returns something strange */ 
 -                             
 -                     deviceInfo->defaultHighInputLatency = defaultHighLatency;
 -                     deviceInfo->defaultHighOutputLatency = defaultHighLatency;
 -                     
 -                 }else{
 - 
 -                     deviceInfo->defaultLowInputLatency = 0.;
 -                     deviceInfo->defaultLowOutputLatency = 0.;
 -                     deviceInfo->defaultHighInputLatency = 0.;
 -                     deviceInfo->defaultHighOutputLatency = 0.;
 -                 }
 - 
 -                 PA_DEBUG(("PaAsio_Initialize: drv:%d defaultLowInputLatency = %f\n", i, deviceInfo->defaultLowInputLatency));
 -                 PA_DEBUG(("PaAsio_Initialize: drv:%d defaultLowOutputLatency = %f\n", i, deviceInfo->defaultLowOutputLatency));
 -                 PA_DEBUG(("PaAsio_Initialize: drv:%d defaultHighInputLatency = %f\n", i, deviceInfo->defaultHighInputLatency));
 -                 PA_DEBUG(("PaAsio_Initialize: drv:%d defaultHighOutputLatency = %f\n", i, deviceInfo->defaultHighOutputLatency));
 - 
 -                 asioDeviceInfo->minBufferSize = paAsioDriverInfo.bufferMinSize;
 -                 asioDeviceInfo->maxBufferSize = paAsioDriverInfo.bufferMaxSize;
 -                 asioDeviceInfo->preferredBufferSize = paAsioDriverInfo.bufferPreferredSize;
 -                 asioDeviceInfo->bufferGranularity = paAsioDriverInfo.bufferGranularity;
 - 
 - 
 -                 asioDeviceInfo->asioChannelInfos = (ASIOChannelInfo*)PaUtil_GroupAllocateMemory(
 -                         asioHostApi->allocations,
 -                         sizeof(ASIOChannelInfo) * (deviceInfo->maxInputChannels
 -                                 + deviceInfo->maxOutputChannels) );
 -                 if( !asioDeviceInfo->asioChannelInfos )
 -                 {
 -                     result = paInsufficientMemory;
 -                     goto error;
 -                 }
 - 
 -                 int a;
 - 
 -                 for( a=0; a < deviceInfo->maxInputChannels; ++a ){
 -                     asioDeviceInfo->asioChannelInfos[a].channel = a;
 -                     asioDeviceInfo->asioChannelInfos[a].isInput = ASIOTrue;
 -                     ASIOError asioError = ASIOGetChannelInfo( &asioDeviceInfo->asioChannelInfos[a] );
 -                     if( asioError != ASE_OK )
 -                     {
 -                         result = paUnanticipatedHostError;
 -                         PA_ASIO_SET_LAST_ASIO_ERROR( asioError );
 -                         goto error;
 -                     }
 -                 }
 - 
 -                 for( a=0; a < deviceInfo->maxOutputChannels; ++a ){
 -                     int b = deviceInfo->maxInputChannels + a;
 -                     asioDeviceInfo->asioChannelInfos[b].channel = a;
 -                     asioDeviceInfo->asioChannelInfos[b].isInput = ASIOFalse;
 -                     ASIOError asioError = ASIOGetChannelInfo( &asioDeviceInfo->asioChannelInfos[b] );
 -                     if( asioError != ASE_OK )
 -                     {
 -                         result = paUnanticipatedHostError;
 -                         PA_ASIO_SET_LAST_ASIO_ERROR( asioError );
 -                         goto error;
 -                     }
 -                 }
 - 
 - 
 -                 /* unload the driver */
 -                 ASIOExit();
 - 
 -                 (*hostApi)->deviceInfos[ (*hostApi)->info.deviceCount ] = deviceInfo;
 -                 ++(*hostApi)->info.deviceCount;
 -             }
 -         }
 -     }
 - 
 -     if( (*hostApi)->info.deviceCount > 0 )
 -     {
 -         (*hostApi)->info.defaultInputDevice = 0;
 -         (*hostApi)->info.defaultOutputDevice = 0;
 -     }
 -     else
 -     {
 -         (*hostApi)->info.defaultInputDevice = paNoDevice;
 -         (*hostApi)->info.defaultOutputDevice = paNoDevice;
 -     }
 - 
 - 
 -     (*hostApi)->Terminate = Terminate;
 -     (*hostApi)->OpenStream = OpenStream;
 -     (*hostApi)->IsFormatSupported = IsFormatSupported;
 - 
 -     PaUtil_InitializeStreamInterface( &asioHostApi->callbackStreamInterface, CloseStream, StartStream,
 -                                       StopStream, AbortStream, IsStreamStopped, IsStreamActive,
 -                                       GetStreamTime, GetStreamCpuLoad,
 -                                       PaUtil_DummyRead, PaUtil_DummyWrite,
 -                                       PaUtil_DummyGetReadAvailable, PaUtil_DummyGetWriteAvailable );
 - 
 -     PaUtil_InitializeStreamInterface( &asioHostApi->blockingStreamInterface, CloseStream, StartStream,
 -                                       StopStream, AbortStream, IsStreamStopped, IsStreamActive,
 -                                       GetStreamTime, PaUtil_DummyGetCpuLoad,
 -                                       ReadStream, WriteStream, GetStreamReadAvailable, GetStreamWriteAvailable );
 - 
 -     return result;
 - 
 - error:
 -     if( asioHostApi )
 -     {
 -         if( asioHostApi->allocations )
 -         {
 -             PaUtil_FreeAllAllocations( asioHostApi->allocations );
 -             PaUtil_DestroyAllocationGroup( asioHostApi->allocations );
 -         }
 - 
 -         PaUtil_FreeMemory( asioHostApi );
 -     }
 -     return result;
 - }
 - 
 - 
 - 
 - void JackASIODriverTerminate( struct PaUtilHostApiRepresentation *hostApi )
 - {
 -     PaAsioHostApiRepresentation *asioHostApi = (PaAsioHostApiRepresentation*)hostApi;
 - 
 -     /*
 -         IMPLEMENT ME:
 -             - clean up any resources not handled by the allocation group
 -     */
 - 
 -     if( asioHostApi->allocations )
 -     {
 -         PaUtil_FreeAllAllocations( asioHostApi->allocations );
 -         PaUtil_DestroyAllocationGroup( asioHostApi->allocations );
 -     }
 - 
 -     PaUtil_FreeMemory( asioHostApi );
 - }
 - 
 - 
 - int JackASIODriver::Open(jack_nframes_t nframes,
 -                               jack_nframes_t samplerate,
 -                               int capturing,
 -                               int playing,
 -                               int inchannels,
 -                               int outchannels,
 - 							  bool monitor,
 -                               const char* capture_driver_uid,
 - 							  const char* playback_driver_uid,
 - 							  jack_nframes_t capture_latency,
 - 							  jack_nframes_t playback_latency)
 - {
 - 	return 0;
 - 
 - error:
 -    return -1;
 - }
 - 
 - int JackASIODriver::Close()
 - {
 -    	return 0;
 - }
 - 
 - int JackASIODriver::Start()
 - {
 -     JackLog("JackASIODriver::Start\n");
 - 	return 0;
 - }
 - 
 - int JackASIODriver::Stop()
 - {
 - 	JackLog("JackASIODriver::Stop\n");
 -     return 0;
 - }
 - 
 - int JackASIODriver::SetBufferSize(jack_nframes_t nframes)
 - {
 - 	return 0;
 - }
 - 
 - void JackASIODriver::PrintState()
 - {
 - 	int i;
 -     std::cout << "JackASIODriver state" << std::endl;
 - 
 -     jack_port_id_t port_index;
 - 
 -     std::cout << "Input ports" << std::endl;
 - 
 - 	/*
 -     for (i = 0; i < fPlaybackChannels; i++) {
 -         port_index = fCapturePortList[i];
 -         JackPort* port = fGraphManager->GetPort(port_index);
 -         std::cout << port->GetName() << std::endl;
 -         if (fGraphManager->GetConnectionsNum(port_index)) {}
 -     }
 - 
 -     std::cout << "Output ports" << std::endl;
 - 
 -     for (i = 0; i < fCaptureChannels; i++) {
 -         port_index = fPlaybackPortList[i];
 -         JackPort* port = fGraphManager->GetPort(port_index);
 -         std::cout << port->GetName() << std::endl;
 -         if (fGraphManager->GetConnectionsNum(port_index)) {}
 -     }
 - 	*/
 - }
 - 
 - } // end of namespace
 - 
 - #ifdef __cplusplus
 - extern "C"
 - {
 - #endif
 - 
 - #include "JackExports.h"
 - 
 -     EXPORT jack_driver_desc_t* driver_get_descriptor() {
 -         jack_driver_desc_t *desc;
 -         unsigned int i;
 -         desc = (jack_driver_desc_t*)calloc(1, sizeof(jack_driver_desc_t));
 - 
 -         strcpy(desc->name, "ASIO");
 - 
 -         desc->nparams = 13;
 -         desc->params = (jack_driver_param_desc_t*)calloc(desc->nparams, sizeof(jack_driver_param_desc_t));
 - 
 -         i = 0;
 -         strcpy(desc->params[i].name, "channels");
 -         desc->params[i].character = 'c';
 -         desc->params[i].type = JackDriverParamInt;
 -         desc->params[i].value.ui = 0;
 -         strcpy(desc->params[i].short_desc, "Maximum number of channels");
 -         strcpy(desc->params[i].long_desc, desc->params[i].short_desc);
 - 
 -         i++;
 -         strcpy(desc->params[i].name, "inchannels");
 -         desc->params[i].character = 'i';
 -         desc->params[i].type = JackDriverParamInt;
 -         desc->params[i].value.ui = 0;
 -         strcpy(desc->params[i].short_desc, "Maximum number of input channels");
 -         strcpy(desc->params[i].long_desc, desc->params[i].short_desc);
 - 
 -         i++;
 -         strcpy(desc->params[i].name, "outchannels");
 -         desc->params[i].character = 'o';
 -         desc->params[i].type = JackDriverParamInt;
 -         desc->params[i].value.ui = 0;
 -         strcpy(desc->params[i].short_desc, "Maximum number of output channels");
 -         strcpy(desc->params[i].long_desc, desc->params[i].short_desc);
 - 
 -         i++;
 -         strcpy(desc->params[i].name, "capture");
 -         desc->params[i].character = 'C';
 -         desc->params[i].type = JackDriverParamString;
 -         strcpy(desc->params[i].value.str, "will take default PortAudio input device");
 -         strcpy(desc->params[i].short_desc, "Provide capture ports. Optionally set PortAudio device name");
 -         strcpy(desc->params[i].long_desc, desc->params[i].short_desc);
 - 
 -         i++;
 -         strcpy(desc->params[i].name, "playback");
 -         desc->params[i].character = 'P';
 -         desc->params[i].type = JackDriverParamString;
 -         strcpy(desc->params[i].value.str, "will take default PortAudio output device");
 -         strcpy(desc->params[i].short_desc, "Provide playback ports. Optionally set PortAudio device name");
 -         strcpy(desc->params[i].long_desc, desc->params[i].short_desc);
 - 		
 - 		i++;
 -         strcpy (desc->params[i].name, "monitor");
 -         desc->params[i].character = 'm';
 -         desc->params[i].type = JackDriverParamBool;
 -         desc->params[i].value.i = 0;
 -         strcpy(desc->params[i].short_desc, "Provide monitor ports for the output");
 -         strcpy(desc->params[i].long_desc, desc->params[i].short_desc);
 - 
 -         i++;
 -         strcpy(desc->params[i].name, "duplex");
 -         desc->params[i].character = 'D';
 -         desc->params[i].type = JackDriverParamBool;
 -         desc->params[i].value.i = TRUE;
 -         strcpy(desc->params[i].short_desc, "Provide both capture and playback ports");
 -         strcpy(desc->params[i].long_desc, desc->params[i].short_desc);
 - 
 -         i++;
 -         strcpy(desc->params[i].name, "rate");
 -         desc->params[i].character = 'r';
 -         desc->params[i].type = JackDriverParamUInt;
 -         desc->params[i].value.ui = 44100U;
 -         strcpy(desc->params[i].short_desc, "Sample rate");
 -         strcpy(desc->params[i].long_desc, desc->params[i].short_desc);
 - 
 -         i++;
 -         strcpy(desc->params[i].name, "period");
 -         desc->params[i].character = 'p';
 -         desc->params[i].type = JackDriverParamUInt;
 -         desc->params[i].value.ui = 128U;
 -         strcpy(desc->params[i].short_desc, "Frames per period");
 -         strcpy(desc->params[i].long_desc, desc->params[i].short_desc);
 - 
 -         i++;
 -         strcpy(desc->params[i].name, "device");
 -         desc->params[i].character = 'd';
 -         desc->params[i].type = JackDriverParamString;
 -         desc->params[i].value.ui = 128U;
 - 		strcpy(desc->params[i].value.str, "will take default CoreAudio device name");
 -         strcpy(desc->params[i].short_desc, "CoreAudio device name");
 -         strcpy(desc->params[i].long_desc, desc->params[i].short_desc);
 - 
 -         i++;
 -         strcpy(desc->params[i].name, "input-latency");
 -         desc->params[i].character = 'I';
 -         desc->params[i].type = JackDriverParamUInt;
 -         desc->params[i].value.i = 0;
 -         strcpy(desc->params[i].short_desc, "Extra input latency");
 -         strcpy(desc->params[i].long_desc, desc->params[i].short_desc);
 - 
 -         i++;
 -         strcpy(desc->params[i].name, "output-latency");
 -         desc->params[i].character = 'O';
 -         desc->params[i].type = JackDriverParamUInt;
 -         desc->params[i].value.i = 0;
 -         strcpy(desc->params[i].short_desc, "Extra output latency");
 -         strcpy(desc->params[i].long_desc, desc->params[i].short_desc);
 - 		
 - 		i++;
 - 		strcpy(desc->params[i].name, "list-devices");
 - 		desc->params[i].character  = 'l';
 - 		desc->params[i].type = JackDriverParamBool;
 - 		desc->params[i].value.i  = TRUE;
 - 		strcpy(desc->params[i].short_desc, "Display available CoreAudio devices");
 - 		strcpy(desc->params[i].long_desc, desc->params[i].short_desc);
 - 
 -         return desc;
 -     }
 - 
 -     EXPORT Jack::JackDriverClientInterface* driver_initialize(Jack::JackEngine* engine, Jack::JackSynchro** table, const JSList* params) 
 - 	{
 -         jack_nframes_t srate = 44100;
 -         jack_nframes_t frames_per_interrupt = 512;
 -         int capture = FALSE;
 -         int playback = FALSE;
 -         int chan_in = 0;
 -         int chan_out = 0;
 - 		bool monitor = false;	
 -     	char* capture_pcm_name = "";
 - 		char* playback_pcm_name = "";
 -         const JSList *node;
 -         const jack_driver_param_t *param;
 -         jack_nframes_t systemic_input_latency = 0;
 -         jack_nframes_t systemic_output_latency = 0;
 - 
 - 		for (node = params; node; node = jack_slist_next(node)) {
 -             param = (const jack_driver_param_t *) node->data;
 - 
 -             switch (param->character) {
 - 
 -                 case 'd':
 -                     capture_pcm_name = strdup(param->value.str);
 - 					playback_pcm_name = strdup(param->value.str);
 -                     break;
 - 
 -                 case 'D':
 -                     capture = TRUE;
 -                     playback = TRUE;
 -                     break;
 - 
 -                 case 'c':
 -                     chan_in = chan_out = (int) param->value.ui;
 -                     break;
 - 
 -                 case 'i':
 -                     chan_in = (int) param->value.ui;
 -                     break;
 - 
 -                 case 'o':
 -                     chan_out = (int) param->value.ui;
 -                     break;
 - 
 -                 case 'C':
 - 					capture = TRUE;
 - 					if (strcmp(param->value.str, "none") != 0) {
 -                         capture_pcm_name = strdup(param->value.str);
 -                     }
 -                     break;
 - 
 -                 case 'P':
 - 					playback = TRUE;
 - 					if (strcmp(param->value.str, "none") != 0) {
 -                         playback_pcm_name = strdup(param->value.str);
 -                     }
 -                     break;
 - 				
 - 				case 'm':
 -                     monitor = param->value.i;
 -                     break;
 - 
 -                 case 'r':
 -                     srate = param->value.ui;
 -                     break;
 - 
 -                 case 'p':
 -                     frames_per_interrupt = (unsigned int) param->value.ui;
 -                     break;
 - 
 -                 case 'I':
 -                     systemic_input_latency = param->value.ui;
 -                     break;
 - 
 -                 case 'O':
 -                     systemic_output_latency = param->value.ui;
 -                     break;
 - 				
 - 				case 'l':
 - 					Jack::DisplayDeviceNames();
 - 					break;
 -             }
 -         }
 - 
 -         // duplex is the default 
 -         if (!capture && !playback) {
 -             capture = TRUE;
 -             playback = TRUE;
 -         }
 - 
 -         Jack::JackDriverClientInterface* driver = new Jack::JackASIODriver("ASIO", engine, table);
 - 		if (driver->Open(frames_per_interrupt, srate, capture, playback, chan_in, chan_out, monitor, capture_pcm_name, playback_pcm_name, systemic_input_latency, systemic_output_latency) == 0) {
 - 			return driver;
 -         } else {
 -             delete driver;
 -             return NULL;
 -         }
 -     }
 - 
 - #ifdef __cplusplus
 - }
 - #endif
 
 
  |