00001
00002
00003
00004
00005
00006
00007
00008
00009
00010
00011
00012
00013
00014
00015
00016
00017
00018
00019 #include "JackNetManager.h"
00020 #include "JackArgParser.h"
00021 #include "JackTime.h"
00022
00023 using namespace std;
00024
00025 namespace Jack
00026 {
00027
00028
00029 JackNetMaster::JackNetMaster ( JackNetSocket& socket, session_params_t& params, const char* multicast_ip)
00030 : JackNetMasterInterface ( params, socket, multicast_ip )
00031 {
00032 jack_log ( "JackNetMaster::JackNetMaster" );
00033
00034
00035 fClientName = const_cast<char*> ( fParams.fName );
00036 fJackClient = NULL;
00037 fSendTransportData.fState = -1;
00038 fReturnTransportData.fState = -1;
00039 fLastTransportState = -1;
00040 uint port_index;
00041
00042
00043 fAudioCapturePorts = new jack_port_t* [fParams.fSendAudioChannels];
00044 for ( port_index = 0; port_index < fParams.fSendAudioChannels; port_index++ )
00045 fAudioCapturePorts[port_index] = NULL;
00046 fAudioPlaybackPorts = new jack_port_t* [fParams.fReturnAudioChannels];
00047 for ( port_index = 0; port_index < fParams.fReturnAudioChannels; port_index++ )
00048 fAudioPlaybackPorts[port_index] = NULL;
00049
00050 fMidiCapturePorts = new jack_port_t* [fParams.fSendMidiChannels];
00051 for ( port_index = 0; port_index < fParams.fSendMidiChannels; port_index++ )
00052 fMidiCapturePorts[port_index] = NULL;
00053 fMidiPlaybackPorts = new jack_port_t* [fParams.fReturnMidiChannels];
00054 for ( port_index = 0; port_index < fParams.fReturnMidiChannels; port_index++ )
00055 fMidiPlaybackPorts[port_index] = NULL;
00056
00057
00058 #ifdef JACK_MONITOR
00059 fPeriodUsecs = ( int ) ( 1000000.f * ( ( float ) fParams.fPeriodSize / ( float ) fParams.fSampleRate ) );
00060 string plot_name;
00061 plot_name = string ( fParams.fName );
00062 plot_name += string ( "_master" );
00063 plot_name += string ( ( fParams.fSlaveSyncMode ) ? "_sync" : "_async" );
00064 switch ( fParams.fNetworkMode )
00065 {
00066 case 's' :
00067 plot_name += string ( "_slow" );
00068 break;
00069 case 'n' :
00070 plot_name += string ( "_normal" );
00071 break;
00072 case 'f' :
00073 plot_name += string ( "_fast" );
00074 break;
00075 }
00076 fNetTimeMon = new JackGnuPlotMonitor<float> ( 128, 4, plot_name );
00077 string net_time_mon_fields[] =
00078 {
00079 string ( "sync send" ),
00080 string ( "end of send" ),
00081 string ( "sync recv" ),
00082 string ( "end of cycle" )
00083 };
00084 string net_time_mon_options[] =
00085 {
00086 string ( "set xlabel \"audio cycles\"" ),
00087 string ( "set ylabel \"% of audio cycle\"" )
00088 };
00089 fNetTimeMon->SetPlotFile ( net_time_mon_options, 2, net_time_mon_fields, 4 );
00090 #endif
00091 }
00092
00093 JackNetMaster::~JackNetMaster()
00094 {
00095 jack_log ( "JackNetMaster::~JackNetMaster, ID %u.", fParams.fID );
00096
00097 if ( fJackClient )
00098 {
00099 jack_deactivate ( fJackClient );
00100 FreePorts();
00101 jack_client_close ( fJackClient );
00102 }
00103 delete[] fAudioCapturePorts;
00104 delete[] fAudioPlaybackPorts;
00105 delete[] fMidiCapturePorts;
00106 delete[] fMidiPlaybackPorts;
00107 #ifdef JACK_MONITOR
00108 fNetTimeMon->Save();
00109 delete fNetTimeMon;
00110 #endif
00111 }
00112
00113 bool JackNetMaster::Init(bool auto_connect)
00114 {
00115
00116 if ( !JackNetMasterInterface::Init() )
00117 return false;
00118
00119
00120 SetParams();
00121
00122
00123 jack_status_t status;
00124 if ( ( fJackClient = jack_client_open ( fClientName, JackNullOption, &status, NULL ) ) == NULL )
00125 {
00126 jack_error ( "Can't open a new jack client." );
00127 return false;
00128 }
00129
00130 if (jack_set_process_callback(fJackClient, SetProcess, this ) < 0)
00131 goto fail;
00132
00133 if (jack_set_buffer_size_callback(fJackClient, SetBufferSize, this) < 0)
00134 goto fail;
00135
00136 if ( AllocPorts() != 0 )
00137 {
00138 jack_error ( "Can't allocate jack ports." );
00139 goto fail;
00140 }
00141
00142
00143 fRunning = true;
00144
00145
00146 if ( jack_activate ( fJackClient ) != 0 )
00147 {
00148 jack_error ( "Can't activate jack client." );
00149 goto fail;
00150 }
00151
00152 if (auto_connect)
00153 ConnectPorts();
00154 jack_info ( "New NetMaster started." );
00155 return true;
00156
00157 fail:
00158 FreePorts();
00159 jack_client_close ( fJackClient );
00160 fJackClient = NULL;
00161 return false;
00162 }
00163
00164
00165 int JackNetMaster::AllocPorts()
00166 {
00167 uint i;
00168 char name[24];
00169 jack_nframes_t port_latency = jack_get_buffer_size ( fJackClient );
00170
00171 jack_log ( "JackNetMaster::AllocPorts" );
00172
00173
00174 for ( i = 0; i < fParams.fSendAudioChannels; i++ )
00175 {
00176 sprintf ( name, "to_slave_%d", i+1 );
00177 if ( ( fAudioCapturePorts[i] = jack_port_register ( fJackClient, name, JACK_DEFAULT_AUDIO_TYPE, JackPortIsInput | JackPortIsTerminal, 0 ) ) == NULL )
00178 return -1;
00179
00180 jack_port_set_latency ( fAudioCapturePorts[i], 0 );
00181 }
00182
00183 for ( i = 0; i < fParams.fReturnAudioChannels; i++ )
00184 {
00185 sprintf ( name, "from_slave_%d", i+1 );
00186 if ( ( fAudioPlaybackPorts[i] = jack_port_register ( fJackClient, name, JACK_DEFAULT_AUDIO_TYPE, JackPortIsOutput | JackPortIsTerminal, 0 ) ) == NULL )
00187 return -1;
00188
00189 switch ( fParams.fNetworkMode )
00190 {
00191 case 'f' :
00192 jack_port_set_latency ( fAudioPlaybackPorts[i], ( fParams.fSlaveSyncMode ) ? 0 : port_latency );
00193 break;
00194 case 'n' :
00195 jack_port_set_latency ( fAudioPlaybackPorts[i], port_latency + ( fParams.fSlaveSyncMode ) ? 0 : port_latency );
00196 break;
00197 case 's' :
00198 jack_port_set_latency ( fAudioPlaybackPorts[i], 2 * port_latency + ( fParams.fSlaveSyncMode ) ? 0 : port_latency );
00199 break;
00200 }
00201 }
00202
00203
00204
00205 for ( i = 0; i < fParams.fSendMidiChannels; i++ )
00206 {
00207 sprintf ( name, "midi_to_slave_%d", i+1 );
00208 if ( ( fMidiCapturePorts[i] = jack_port_register ( fJackClient, name, JACK_DEFAULT_MIDI_TYPE, JackPortIsInput | JackPortIsTerminal, 0 ) ) == NULL )
00209 return -1;
00210
00211 jack_port_set_latency ( fMidiCapturePorts[i], 0 );
00212 }
00213 for ( i = 0; i < fParams.fReturnMidiChannels; i++ )
00214 {
00215 sprintf ( name, "midi_from_slave_%d", i+1 );
00216 if ( ( fMidiPlaybackPorts[i] = jack_port_register ( fJackClient, name, JACK_DEFAULT_MIDI_TYPE, JackPortIsOutput | JackPortIsTerminal, 0 ) ) == NULL )
00217 return -1;
00218
00219 switch ( fParams.fNetworkMode )
00220 {
00221 case 'f' :
00222 jack_port_set_latency ( fMidiPlaybackPorts[i], ( fParams.fSlaveSyncMode ) ? 0 : port_latency );
00223 break;
00224 case 'n' :
00225 jack_port_set_latency ( fMidiPlaybackPorts[i], port_latency + ( fParams.fSlaveSyncMode ) ? 0 : port_latency );
00226 break;
00227 case 's' :
00228 jack_port_set_latency ( fMidiPlaybackPorts[i], 2 * port_latency + ( fParams.fSlaveSyncMode ) ? 0 : port_latency );
00229 break;
00230 }
00231 }
00232 return 0;
00233 }
00234
00235 void JackNetMaster::ConnectPorts()
00236 {
00237 const char **ports;
00238
00239 ports = jack_get_ports(fJackClient, NULL, NULL, JackPortIsPhysical | JackPortIsOutput);
00240 if (ports != NULL) {
00241 for (unsigned int i = 0; i < fParams.fSendAudioChannels && ports[i]; i++) {
00242 jack_connect(fJackClient, ports[i], jack_port_name(fAudioCapturePorts[i]));
00243 }
00244 free(ports);
00245 }
00246
00247 ports = jack_get_ports(fJackClient, NULL, NULL, JackPortIsPhysical | JackPortIsInput);
00248 if (ports != NULL) {
00249 for (unsigned int i = 0; i < fParams.fReturnAudioChannels && ports[i]; i++) {
00250 jack_connect(fJackClient, jack_port_name(fAudioPlaybackPorts[i]), ports[i]);
00251 }
00252 free(ports);
00253 }
00254 }
00255
00256 void JackNetMaster::FreePorts()
00257 {
00258 jack_log ( "JackNetMaster::FreePorts, ID %u", fParams.fID );
00259
00260 uint port_index;
00261 for ( port_index = 0; port_index < fParams.fSendAudioChannels; port_index++ )
00262 if ( fAudioCapturePorts[port_index] )
00263 jack_port_unregister ( fJackClient, fAudioCapturePorts[port_index] );
00264 for ( port_index = 0; port_index < fParams.fReturnAudioChannels; port_index++ )
00265 if ( fAudioPlaybackPorts[port_index] )
00266 jack_port_unregister ( fJackClient, fAudioPlaybackPorts[port_index] );
00267 for ( port_index = 0; port_index < fParams.fSendMidiChannels; port_index++ )
00268 if ( fMidiCapturePorts[port_index] )
00269 jack_port_unregister ( fJackClient, fMidiCapturePorts[port_index] );
00270 for ( port_index = 0; port_index < fParams.fReturnMidiChannels; port_index++ )
00271 if ( fMidiPlaybackPorts[port_index] )
00272 jack_port_unregister ( fJackClient, fMidiPlaybackPorts[port_index] );
00273 }
00274
00275
00276 void JackNetMaster::EncodeTransportData()
00277 {
00278
00279
00280 fSendTransportData.fTimebaseMaster = NO_CHANGE;
00281
00282
00283 fSendTransportData.fState = static_cast<uint> ( jack_transport_query ( fJackClient, &fSendTransportData.fPosition ) );
00284
00285
00286 fSendTransportData.fNewState = ( ( fSendTransportData.fState != fLastTransportState ) &&
00287 ( fSendTransportData.fState != fReturnTransportData.fState ) );
00288 if ( fSendTransportData.fNewState )
00289 jack_info ( "Sending '%s' to '%s' frame = %ld", GetTransportState ( fSendTransportData.fState ), fParams.fName, fSendTransportData.fPosition.frame );
00290 fLastTransportState = fSendTransportData.fState;
00291 }
00292
00293 void JackNetMaster::DecodeTransportData()
00294 {
00295
00296 if ( fReturnTransportData.fTimebaseMaster != NO_CHANGE )
00297 {
00298 int timebase = 0;
00299 switch ( fReturnTransportData.fTimebaseMaster )
00300 {
00301 case RELEASE_TIMEBASEMASTER :
00302 timebase = jack_release_timebase ( fJackClient );
00303 if ( timebase < 0 )
00304 jack_error ( "Can't release timebase master." );
00305 else
00306 jack_info ( "'%s' isn't the timebase master anymore.", fParams.fName );
00307 break;
00308 case TIMEBASEMASTER :
00309 timebase = jack_set_timebase_callback ( fJackClient, 0, SetTimebaseCallback, this );
00310 if ( timebase < 0 )
00311 jack_error ( "Can't set a new timebase master." );
00312 else
00313 jack_info ( "'%s' is the new timebase master.", fParams.fName );
00314 break;
00315 case CONDITIONAL_TIMEBASEMASTER :
00316 timebase = jack_set_timebase_callback ( fJackClient, 1, SetTimebaseCallback, this );
00317 if ( timebase != EBUSY )
00318 {
00319 if ( timebase < 0 )
00320 jack_error ( "Can't set a new timebase master." );
00321 else
00322 jack_info ( "'%s' is the new timebase master.", fParams.fName );
00323 }
00324 break;
00325 }
00326 }
00327
00328
00329 if ( fReturnTransportData.fNewState && ( fReturnTransportData.fState != jack_transport_query ( fJackClient, NULL ) ) )
00330 {
00331 switch ( fReturnTransportData.fState )
00332 {
00333 case JackTransportStopped :
00334 jack_transport_stop ( fJackClient );
00335 jack_info ( "'%s' stops transport.", fParams.fName );
00336 break;
00337 case JackTransportStarting :
00338 if ( jack_transport_reposition ( fJackClient, &fReturnTransportData.fPosition ) == EINVAL )
00339 jack_error ( "Can't set new position." );
00340 jack_transport_start ( fJackClient );
00341 jack_info ( "'%s' starts transport frame = %d frame = %d", fParams.fName, fReturnTransportData.fPosition.frame);
00342 break;
00343 case JackTransportNetStarting :
00344 jack_info ( "'%s' is ready to roll..", fParams.fName );
00345 break;
00346 case JackTransportRolling :
00347 jack_info ( "'%s' is rolling.", fParams.fName );
00348 break;
00349 }
00350 }
00351 }
00352
00353 void JackNetMaster::SetTimebaseCallback ( jack_transport_state_t state, jack_nframes_t nframes, jack_position_t* pos, int new_pos, void* arg )
00354 {
00355 static_cast<JackNetMaster*> ( arg )->TimebaseCallback ( pos );
00356 }
00357
00358 void JackNetMaster::TimebaseCallback ( jack_position_t* pos )
00359 {
00360 pos->bar = fReturnTransportData.fPosition.bar;
00361 pos->beat = fReturnTransportData.fPosition.beat;
00362 pos->tick = fReturnTransportData.fPosition.tick;
00363 pos->bar_start_tick = fReturnTransportData.fPosition.bar_start_tick;
00364 pos->beats_per_bar = fReturnTransportData.fPosition.beats_per_bar;
00365 pos->beat_type = fReturnTransportData.fPosition.beat_type;
00366 pos->ticks_per_beat = fReturnTransportData.fPosition.ticks_per_beat;
00367 pos->beats_per_minute = fReturnTransportData.fPosition.beats_per_minute;
00368 }
00369
00370
00371
00372 bool JackNetMaster::IsSlaveReadyToRoll()
00373 {
00374 return ( fReturnTransportData.fState == JackTransportNetStarting );
00375 }
00376
00377 int JackNetMaster::SetBufferSize (jack_nframes_t nframes, void* arg)
00378 {
00379 jack_error("Cannot handle bufer size change, so proxy will be removed...");
00380 static_cast<JackNetMaster*> ( arg )->Exit();
00381 return 0;
00382 }
00383
00384
00385 int JackNetMaster::SetProcess ( jack_nframes_t nframes, void* arg )
00386 {
00387 return static_cast<JackNetMaster*> ( arg )->Process();
00388 }
00389
00390 int JackNetMaster::Process()
00391 {
00392 if ( !fRunning )
00393 return 0;
00394
00395 uint port_index;
00396 int res = 0;
00397
00398 #ifdef JACK_MONITOR
00399 jack_time_t begin_time = GetMicroSeconds();
00400 fNetTimeMon->New();
00401 #endif
00402
00403
00404 for ( port_index = 0; port_index < fParams.fSendMidiChannels; port_index++ )
00405 fNetMidiCaptureBuffer->SetBuffer ( port_index, static_cast<JackMidiBuffer*> ( jack_port_get_buffer ( fMidiCapturePorts[port_index],
00406 fParams.fPeriodSize ) ) );
00407 for ( port_index = 0; port_index < fParams.fSendAudioChannels; port_index++ )
00408 fNetAudioCaptureBuffer->SetBuffer ( port_index, static_cast<sample_t*> ( jack_port_get_buffer ( fAudioCapturePorts[port_index],
00409 fParams.fPeriodSize ) ) );
00410 for ( port_index = 0; port_index < fParams.fReturnMidiChannels; port_index++ )
00411 fNetMidiPlaybackBuffer->SetBuffer ( port_index, static_cast<JackMidiBuffer*> ( jack_port_get_buffer ( fMidiPlaybackPorts[port_index],
00412 fParams.fPeriodSize ) ) );
00413 for ( port_index = 0; port_index < fParams.fReturnAudioChannels; port_index++ )
00414 fNetAudioPlaybackBuffer->SetBuffer ( port_index, static_cast<sample_t*> ( jack_port_get_buffer ( fAudioPlaybackPorts[port_index],
00415 fParams.fPeriodSize ) ) );
00416
00417 if (IsSynched()) {
00418
00419
00420 EncodeSyncPacket();
00421
00422
00423 if ( SyncSend() == SOCKET_ERROR )
00424 return SOCKET_ERROR;
00425
00426 #ifdef JACK_MONITOR
00427 fNetTimeMon->Add ( ( ( ( float ) (GetMicroSeconds() - begin_time ) ) / ( float ) fPeriodUsecs ) * 100.f );
00428 #endif
00429
00430
00431 if ( DataSend() == SOCKET_ERROR )
00432 return SOCKET_ERROR;
00433
00434 #ifdef JACK_MONITOR
00435 fNetTimeMon->Add ( ( ( ( float ) (GetMicroSeconds() - begin_time ) ) / ( float ) fPeriodUsecs ) * 100.f );
00436 #endif
00437
00438 } else {
00439 jack_error("Connection is not synched, skip cycle...");
00440 }
00441
00442
00443 res = SyncRecv();
00444 if ( ( res == 0 ) || ( res == SOCKET_ERROR ) )
00445 return res;
00446
00447 #ifdef JACK_MONITOR
00448 fNetTimeMon->Add ( ( ( ( float ) (GetMicroSeconds() - begin_time ) ) / ( float ) fPeriodUsecs ) * 100.f );
00449 #endif
00450
00451
00452 DecodeSyncPacket();
00453
00454
00455 res = DataRecv();
00456 if ( ( res == 0 ) || ( res == SOCKET_ERROR ) )
00457 return res;
00458
00459 #ifdef JACK_MONITOR
00460 fNetTimeMon->AddLast ( ( ( ( float ) (GetMicroSeconds() - begin_time ) ) / ( float ) fPeriodUsecs ) * 100.f );
00461 #endif
00462 return 0;
00463 }
00464
00465
00466
00467 JackNetMasterManager::JackNetMasterManager ( jack_client_t* client, const JSList* params ) : fSocket()
00468 {
00469 jack_log ( "JackNetMasterManager::JackNetMasterManager" );
00470
00471 fManagerClient = client;
00472 fManagerName = jack_get_client_name ( fManagerClient );
00473 strcpy(fMulticastIP, DEFAULT_MULTICAST_IP);
00474 fSocket.SetPort ( DEFAULT_PORT );
00475 fGlobalID = 0;
00476 fRunning = true;
00477 fAutoConnect = false;
00478
00479 const JSList* node;
00480 const jack_driver_param_t* param;
00481 for ( node = params; node; node = jack_slist_next ( node ) )
00482 {
00483 param = ( const jack_driver_param_t* ) node->data;
00484 switch ( param->character )
00485 {
00486 case 'a' :
00487 if (strlen (param->value.str) < 32)
00488 strcpy(fMulticastIP, param->value.str);
00489 else
00490 jack_error("Can't use multicast address %s, using default %s", param->value.ui, DEFAULT_MULTICAST_IP);
00491 break;
00492
00493 case 'p':
00494 fSocket.SetPort ( param->value.ui );
00495 break;
00496
00497 case 'c':
00498 fAutoConnect = param->value.i;
00499 break;
00500 }
00501 }
00502
00503
00504 jack_set_sync_callback ( fManagerClient, SetSyncCallback, this );
00505
00506
00507 if ( jack_activate ( fManagerClient ) != 0 )
00508 jack_error ( "Can't activate the network manager client, transport disabled." );
00509
00510
00511 if ( jack_client_create_thread ( fManagerClient, &fManagerThread, 0, 0, NetManagerThread, this ) )
00512 jack_error ( "Can't create the network manager control thread." );
00513 }
00514
00515 JackNetMasterManager::~JackNetMasterManager()
00516 {
00517 jack_log ( "JackNetMasterManager::~JackNetMasterManager" );
00518 jack_info ( "Exiting net manager..." );
00519 fRunning = false;
00520 jack_client_kill_thread ( fManagerClient, fManagerThread );
00521 master_list_t::iterator it;
00522 for ( it = fMasterList.begin(); it != fMasterList.end(); it++ )
00523 delete ( *it );
00524 fSocket.Close();
00525 SocketAPIEnd();
00526 }
00527
00528 int JackNetMasterManager::SetSyncCallback ( jack_transport_state_t state, jack_position_t* pos, void* arg )
00529 {
00530 return static_cast<JackNetMasterManager*> ( arg )->SyncCallback ( state, pos );
00531 }
00532
00533 int JackNetMasterManager::SyncCallback ( jack_transport_state_t state, jack_position_t* pos )
00534 {
00535
00536 int ret = 1;
00537 master_list_it_t it;
00538 for ( it = fMasterList.begin(); it != fMasterList.end(); it++ )
00539 if ( ! ( *it )->IsSlaveReadyToRoll() )
00540 ret = 0;
00541 jack_log ( "JackNetMasterManager::SyncCallback returns '%s'", ( ret ) ? "true" : "false" );
00542 return ret;
00543 }
00544
00545 void* JackNetMasterManager::NetManagerThread ( void* arg )
00546 {
00547 JackNetMasterManager* master_manager = static_cast<JackNetMasterManager*> ( arg );
00548 jack_info ( "Starting Jack Network Manager." );
00549 jack_info ( "Listening on '%s:%d'", master_manager->fMulticastIP, master_manager->fSocket.GetPort() );
00550 master_manager->Run();
00551 return NULL;
00552 }
00553
00554 void JackNetMasterManager::Run()
00555 {
00556 jack_log ( "JackNetMasterManager::Run" );
00557
00558 int attempt = 0;
00559
00560
00561 session_params_t host_params;
00562 int rx_bytes = 0;
00563 JackNetMaster* net_master;
00564
00565
00566 if ( SocketAPIInit() < 0 )
00567 {
00568 jack_error ( "Can't init Socket API, exiting..." );
00569 return;
00570 }
00571
00572
00573 if ( fSocket.NewSocket() == SOCKET_ERROR )
00574 {
00575 jack_error ( "Can't create the network management input socket : %s", StrError ( NET_ERROR_CODE ) );
00576 return;
00577 }
00578
00579
00580 if ( fSocket.Bind() == SOCKET_ERROR )
00581 {
00582 jack_error ( "Can't bind the network manager socket : %s", StrError ( NET_ERROR_CODE ) );
00583 fSocket.Close();
00584 return;
00585 }
00586
00587
00588 if ( fSocket.JoinMCastGroup ( fMulticastIP ) == SOCKET_ERROR )
00589 jack_error ( "Can't join multicast group : %s", StrError ( NET_ERROR_CODE ) );
00590
00591
00592 if ( fSocket.SetLocalLoop() == SOCKET_ERROR )
00593 jack_error ( "Can't set local loop : %s", StrError ( NET_ERROR_CODE ) );
00594
00595
00596 if ( fSocket.SetTimeOut ( 2000000 ) == SOCKET_ERROR )
00597 jack_error ( "Can't set timeout : %s", StrError ( NET_ERROR_CODE ) );
00598
00599 jack_info ( "Waiting for a slave..." );
00600
00601
00602 do
00603 {
00604 session_params_t net_params;
00605 rx_bytes = fSocket.CatchHost ( &net_params, sizeof ( session_params_t ), 0 );
00606 SessionParamsNToH(&net_params, &host_params);
00607 if ( ( rx_bytes == SOCKET_ERROR ) && ( fSocket.GetError() != NET_NO_DATA ) )
00608 {
00609 jack_error ( "Error in receive : %s", StrError ( NET_ERROR_CODE ) );
00610 if ( ++attempt == 10 )
00611 {
00612 jack_error ( "Can't receive on the socket, exiting net manager." );
00613 return;
00614 }
00615 }
00616 if ( rx_bytes == sizeof ( session_params_t ) )
00617 {
00618 switch ( GetPacketType ( &host_params ) )
00619 {
00620 case SLAVE_AVAILABLE:
00621 if ( ( net_master = InitMaster ( host_params ) ) )
00622 SessionParamsDisplay ( &net_master->fParams );
00623 else
00624 jack_error ( "Can't init new net master..." );
00625 jack_info ( "Waiting for a slave..." );
00626 break;
00627 case KILL_MASTER:
00628 if ( KillMaster ( &host_params ) )
00629 jack_info ( "Waiting for a slave..." );
00630 break;
00631 default:
00632 break;
00633 }
00634 }
00635 }
00636 while ( fRunning );
00637 }
00638
00639 JackNetMaster* JackNetMasterManager::InitMaster ( session_params_t& params )
00640 {
00641 jack_log ( "JackNetMasterManager::InitMaster, Slave : %s", params.fName );
00642
00643
00644 if (params.fProtocolVersion != MASTER_PROTOCOL) {
00645 jack_error ( "Error : slave is running with a different protocol %s", params.fName );
00646 return NULL;
00647 }
00648
00649
00650 fSocket.GetName ( params.fMasterNetName );
00651 params.fID = ++fGlobalID;
00652 params.fSampleRate = jack_get_sample_rate ( fManagerClient );
00653 params.fPeriodSize = jack_get_buffer_size ( fManagerClient );
00654 params.fBitdepth = 0;
00655 SetSlaveName ( params );
00656
00657
00658 JackNetMaster* master = new JackNetMaster(fSocket, params, fMulticastIP);
00659 if ( master->Init(fAutoConnect) )
00660 {
00661 fMasterList.push_back ( master );
00662 return master;
00663 }
00664 delete master;
00665 return NULL;
00666 }
00667
00668 void JackNetMasterManager::SetSlaveName ( session_params_t& params )
00669 {
00670 jack_log ( "JackNetMasterManager::SetSlaveName" );
00671
00672 master_list_it_t it;
00673 for ( it = fMasterList.begin(); it != fMasterList.end(); it++ )
00674 if ( strcmp ( ( *it )->fParams.fName, params.fName ) == 0 )
00675 sprintf ( params.fName, "%s-%u", params.fName, params.fID );
00676 }
00677
00678 master_list_it_t JackNetMasterManager::FindMaster ( uint32_t id )
00679 {
00680 jack_log ( "JackNetMasterManager::FindMaster, ID %u.", id );
00681
00682 master_list_it_t it;
00683 for ( it = fMasterList.begin(); it != fMasterList.end(); it++ )
00684 if ( ( *it )->fParams.fID == id )
00685 return it;
00686 return it;
00687 }
00688
00689 int JackNetMasterManager::KillMaster ( session_params_t* params )
00690 {
00691 jack_log ( "JackNetMasterManager::KillMaster, ID %u.", params->fID );
00692
00693 master_list_it_t master = FindMaster ( params->fID );
00694 if ( master != fMasterList.end() )
00695 {
00696 fMasterList.erase ( master );
00697 delete *master;
00698 return 1;
00699 }
00700 return 0;
00701 }
00702 }
00703
00704 static Jack::JackNetMasterManager* master_manager = NULL;
00705
00706 #ifdef __cplusplus
00707 extern "C"
00708 {
00709 #endif
00710
00711 SERVER_EXPORT jack_driver_desc_t* jack_get_descriptor()
00712 {
00713 jack_driver_desc_t *desc;
00714 desc = ( jack_driver_desc_t* ) calloc ( 1, sizeof ( jack_driver_desc_t ) );
00715
00716 strcpy ( desc->name, "netmanager" );
00717 strcpy ( desc->desc, "netjack multi-cast master component" );
00718
00719 desc->nparams = 3;
00720 desc->params = ( jack_driver_param_desc_t* ) calloc ( desc->nparams, sizeof ( jack_driver_param_desc_t ) );
00721
00722 int i = 0;
00723 strcpy ( desc->params[i].name, "multicast_ip" );
00724 desc->params[i].character = 'a';
00725 desc->params[i].type = JackDriverParamString;
00726 strcpy ( desc->params[i].value.str, DEFAULT_MULTICAST_IP );
00727 strcpy ( desc->params[i].short_desc, "Multicast Address" );
00728 strcpy ( desc->params[i].long_desc, desc->params[i].short_desc );
00729
00730 i++;
00731 strcpy ( desc->params[i].name, "udp_net_port" );
00732 desc->params[i].character = 'p';
00733 desc->params[i].type = JackDriverParamInt;
00734 desc->params[i].value.i = DEFAULT_PORT;
00735 strcpy ( desc->params[i].short_desc, "UDP port" );
00736 strcpy ( desc->params[i].long_desc, desc->params[i].short_desc );
00737
00738 i++;
00739 strcpy ( desc->params[i].name, "auto_connect" );
00740 desc->params[i].character = 'c';
00741 desc->params[i].type = JackDriverParamBool;
00742 desc->params[i].value.i = false;
00743 strcpy ( desc->params[i].short_desc, "Auto connect netmaster to system ports" );
00744 strcpy ( desc->params[i].long_desc, desc->params[i].short_desc );
00745
00746 return desc;
00747 }
00748
00749 SERVER_EXPORT int jack_internal_initialize ( jack_client_t* jack_client, const JSList* params )
00750 {
00751 if ( master_manager )
00752 {
00753 jack_error ( "Master Manager already loaded" );
00754 return 1;
00755 }
00756 else
00757 {
00758 jack_log ( "Loading Master Manager" );
00759 master_manager = new Jack::JackNetMasterManager ( jack_client, params );
00760 return ( master_manager ) ? 0 : 1;
00761 }
00762 }
00763
00764 SERVER_EXPORT int jack_initialize ( jack_client_t* jack_client, const char* load_init )
00765 {
00766 JSList* params = NULL;
00767 bool parse_params = true;
00768 int res = 1;
00769 jack_driver_desc_t* desc = jack_get_descriptor();
00770
00771 Jack::JackArgParser parser ( load_init );
00772 if ( parser.GetArgc() > 0 )
00773 parse_params = parser.ParseParams ( desc, ¶ms );
00774
00775 if (parse_params) {
00776 res = jack_internal_initialize ( jack_client, params );
00777 parser.FreeParams ( params );
00778 }
00779 return res;
00780 }
00781
00782 SERVER_EXPORT void jack_finish ( void* arg )
00783 {
00784 if ( master_manager )
00785 {
00786 jack_log ( "Unloading Master Manager" );
00787 delete master_manager;
00788 master_manager = NULL;
00789 }
00790 }
00791 #ifdef __cplusplus
00792 }
00793 #endif