jack2 codebase
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

1164 lines
39KB

  1. /*
  2. Copyright (C) 2009-2011 Grame
  3. This program is free software; you can redistribute it and/or modify
  4. it under the terms of the GNU Lesser General Public License as published by
  5. the Free Software Foundation; either version 2.1 of the License, or
  6. (at your option) any later version.
  7. This program is distributed in the hope that it will be useful,
  8. but WITHOUT ANY WARRANTY; without even the implied warranty of
  9. MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  10. GNU Lesser General Public License for more details.
  11. You should have received a copy of the GNU Lesser General Public License
  12. along with this program; if not, write to the Free Software
  13. Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
  14. */
  15. #include <assert.h>
  16. #include <stdarg.h>
  17. #include "JackNetInterface.h"
  18. #include "JackAudioAdapterInterface.h"
  19. #ifdef __cplusplus
  20. extern "C"
  21. {
  22. #endif
  23. // NetJack common API
  24. #define MASTER_NAME_SIZE 256
  25. enum JackNetEncoder {
  26. JackFloatEncoder = 0,
  27. JackIntEncoder = 1,
  28. JackCeltEncoder = 2,
  29. JackOpusEncoder = 3
  30. };
  31. typedef struct {
  32. int audio_input;
  33. int audio_output;
  34. int midi_input;
  35. int midi_output;
  36. int mtu;
  37. int time_out; // in millisecond, -1 means in infinite
  38. int encoder; // one of JackNetEncoder
  39. int kbps; // KB per second for CELT encoder
  40. int latency; // network cycles
  41. } jack_slave_t;
  42. typedef struct {
  43. int audio_input;
  44. int audio_output;
  45. int midi_input;
  46. int midi_output;
  47. jack_nframes_t buffer_size;
  48. jack_nframes_t sample_rate;
  49. char master_name[MASTER_NAME_SIZE];
  50. int time_out;
  51. } jack_master_t;
  52. // NetJack slave API
  53. typedef struct _jack_net_slave jack_net_slave_t;
  54. typedef int (* JackNetSlaveProcessCallback) (jack_nframes_t buffer_size,
  55. int audio_input,
  56. float** audio_input_buffer,
  57. int midi_input,
  58. void** midi_input_buffer,
  59. int audio_output,
  60. float** audio_output_buffer,
  61. int midi_output,
  62. void** midi_output_buffer,
  63. void* data);
  64. typedef int (*JackNetSlaveBufferSizeCallback) (jack_nframes_t nframes, void *arg);
  65. typedef int (*JackNetSlaveSampleRateCallback) (jack_nframes_t nframes, void *arg);
  66. typedef void (*JackNetSlaveShutdownCallback) (void* arg);
  67. typedef int (*JackNetSlaveRestartCallback) (void* arg);
  68. typedef void (*JackNetSlaveErrorCallback) (int error_code, void* arg);
  69. LIB_EXPORT jack_net_slave_t* jack_net_slave_open(const char* ip, int port, const char* name, jack_slave_t* request, jack_master_t* result);
  70. LIB_EXPORT int jack_net_slave_close(jack_net_slave_t* net);
  71. LIB_EXPORT int jack_net_slave_activate(jack_net_slave_t* net);
  72. LIB_EXPORT int jack_net_slave_deactivate(jack_net_slave_t* net);
  73. LIB_EXPORT int jack_net_slave_is_active(jack_net_slave_t* net);
  74. LIB_EXPORT int jack_set_net_slave_process_callback(jack_net_slave_t* net, JackNetSlaveProcessCallback net_callback, void *arg);
  75. LIB_EXPORT int jack_set_net_slave_buffer_size_callback(jack_net_slave_t* net, JackNetSlaveBufferSizeCallback bufsize_callback, void *arg);
  76. LIB_EXPORT int jack_set_net_slave_sample_rate_callback(jack_net_slave_t* net, JackNetSlaveSampleRateCallback samplerate_callback, void *arg);
  77. LIB_EXPORT int jack_set_net_slave_shutdown_callback(jack_net_slave_t* net, JackNetSlaveShutdownCallback shutdown_callback, void *arg);
  78. LIB_EXPORT int jack_set_net_slave_restart_callback(jack_net_slave_t* net, JackNetSlaveRestartCallback restart_callback, void *arg);
  79. LIB_EXPORT int jack_set_net_slave_error_callback(jack_net_slave_t* net, JackNetSlaveErrorCallback error_callback, void *arg);
  80. // NetJack master API
  81. typedef struct _jack_net_master jack_net_master_t;
  82. LIB_EXPORT jack_net_master_t* jack_net_master_open(const char* ip, int port, const char* name, jack_master_t* request, jack_slave_t* result);
  83. LIB_EXPORT int jack_net_master_close(jack_net_master_t* net);
  84. LIB_EXPORT int jack_net_master_recv(jack_net_master_t* net, int audio_input, float** audio_input_buffer, int midi_input, void** midi_input_buffer);
  85. LIB_EXPORT int jack_net_master_send(jack_net_master_t* net, int audio_output, float** audio_output_buffer, int midi_output, void** midi_output_buffer);
  86. // NetJack adapter API
  87. typedef struct _jack_adapter jack_adapter_t;
  88. LIB_EXPORT jack_adapter_t* jack_create_adapter(int input, int output,
  89. jack_nframes_t host_buffer_size,
  90. jack_nframes_t host_sample_rate,
  91. jack_nframes_t adapted_buffer_size,
  92. jack_nframes_t adapted_sample_rate);
  93. LIB_EXPORT int jack_destroy_adapter(jack_adapter_t* adapter);
  94. LIB_EXPORT void jack_flush_adapter(jack_adapter_t* adapter);
  95. LIB_EXPORT int jack_adapter_push_and_pull(jack_adapter_t* adapter, float** input, float** output, unsigned int frames);
  96. LIB_EXPORT int jack_adapter_pull_and_push(jack_adapter_t* adapter, float** input, float** output, unsigned int frames);
  97. #define LOG_LEVEL_INFO 1
  98. #define LOG_LEVEL_ERROR 2
  99. LIB_EXPORT void jack_error(const char *fmt, ...);
  100. LIB_EXPORT void jack_info(const char *fmt, ...);
  101. LIB_EXPORT void jack_log(const char *fmt, ...);
  102. #ifdef __cplusplus
  103. }
  104. #endif
  105. namespace Jack
  106. {
  107. struct JackNetExtMaster : public JackNetMasterInterface {
  108. jack_master_t fRequest;
  109. int fPacketTimeOut;
  110. JackNetExtMaster(const char* ip,
  111. int port,
  112. const char* name,
  113. jack_master_t* request)
  114. {
  115. fRunning = true;
  116. assert(strlen(ip) < 32);
  117. strcpy(fMulticastIP, ip);
  118. fSocket.SetPort(port);
  119. fRequest.buffer_size = request->buffer_size;
  120. fRequest.sample_rate = request->sample_rate;
  121. fRequest.audio_input = request->audio_input;
  122. fRequest.audio_output = request->audio_output;
  123. fRequest.time_out = request->time_out;
  124. }
  125. virtual ~JackNetExtMaster()
  126. {}
  127. int Open(jack_slave_t* result)
  128. {
  129. // Init socket API (win32)
  130. if (SocketAPIInit() < 0) {
  131. jack_error("Can't init Socket API, exiting...");
  132. return -1;
  133. }
  134. // Request socket
  135. if (fSocket.NewSocket() == SOCKET_ERROR) {
  136. jack_error("Can't create the network management input socket : %s", StrError(NET_ERROR_CODE));
  137. return -1;
  138. }
  139. // Bind the socket to the local port
  140. if (fSocket.Bind() == SOCKET_ERROR) {
  141. jack_error("Can't bind the network manager socket : %s", StrError(NET_ERROR_CODE));
  142. fSocket.Close();
  143. return -1;
  144. }
  145. // Join multicast group
  146. if (fSocket.JoinMCastGroup(fMulticastIP) == SOCKET_ERROR) {
  147. jack_error("Can't join multicast group : %s", StrError(NET_ERROR_CODE));
  148. }
  149. // Local loop
  150. if (fSocket.SetLocalLoop() == SOCKET_ERROR) {
  151. jack_error("Can't set local loop : %s", StrError(NET_ERROR_CODE));
  152. }
  153. // Set a timeout on the multicast receive (the thread can now be cancelled)
  154. if (fSocket.SetTimeOut(MANAGER_INIT_TIMEOUT) == SOCKET_ERROR) {
  155. jack_error("Can't set timeout : %s", StrError(NET_ERROR_CODE));
  156. }
  157. // Main loop, wait for data, deal with it and wait again
  158. int attempt = 0;
  159. int rx_bytes = 0;
  160. int try_count = (fRequest.time_out > 0) ? int((1000000.f * float(fRequest.time_out)) / float(MANAGER_INIT_TIMEOUT)) : INT_MAX;
  161. do
  162. {
  163. session_params_t net_params;
  164. rx_bytes = fSocket.CatchHost(&net_params, sizeof(session_params_t), 0);
  165. SessionParamsNToH(&net_params, &fParams);
  166. if ((rx_bytes == SOCKET_ERROR) && (fSocket.GetError() != NET_NO_DATA)) {
  167. jack_error("Error in receive : %s", StrError(NET_ERROR_CODE));
  168. if (++attempt == 10) {
  169. jack_error("Can't receive on the socket, exiting net manager" );
  170. goto error;
  171. }
  172. }
  173. if (rx_bytes == sizeof(session_params_t )) {
  174. switch (GetPacketType(&fParams)) {
  175. case SLAVE_AVAILABLE:
  176. if (InitMaster(result) == 0) {
  177. SessionParamsDisplay(&fParams);
  178. fRunning = false;
  179. } else {
  180. jack_error("Can't init new net master...");
  181. goto error;
  182. }
  183. jack_info("Waiting for a slave...");
  184. break;
  185. case KILL_MASTER:
  186. break;
  187. default:
  188. break;
  189. }
  190. }
  191. }
  192. while (fRunning && (--try_count > 0));
  193. if (try_count == 0) {
  194. jack_error("Time out error in connect");
  195. return -1;
  196. }
  197. // Set result parameters
  198. result->audio_input = fParams.fSendAudioChannels;
  199. result->audio_output = fParams.fReturnAudioChannels;
  200. result->midi_input = fParams.fSendMidiChannels;
  201. result->midi_output = fParams.fReturnMidiChannels;
  202. result->mtu = fParams.fMtu;
  203. result->latency = fParams.fNetworkLatency;
  204. return 0;
  205. error:
  206. fSocket.Close();
  207. return -1;
  208. }
  209. int InitMaster(jack_slave_t* result)
  210. {
  211. // Check MASTER <==> SLAVE network protocol coherency
  212. if (fParams.fProtocolVersion != NETWORK_PROTOCOL) {
  213. jack_error("Error : slave '%s' is running with a different protocol %d != %d", fParams.fName, fParams.fProtocolVersion, NETWORK_PROTOCOL);
  214. return -1;
  215. }
  216. // Settings
  217. fSocket.GetName(fParams.fMasterNetName);
  218. fParams.fID = 1;
  219. fParams.fPeriodSize = fRequest.buffer_size;
  220. fParams.fSampleRate = fRequest.sample_rate;
  221. if (fRequest.audio_input == -1) {
  222. if (fParams.fSendAudioChannels == -1) {
  223. jack_error("Error : master and slave use -1 for wanted inputs...");
  224. return -1;
  225. } else {
  226. result->audio_input = fParams.fSendAudioChannels;
  227. jack_info("Takes slave %d inputs", fParams.fSendAudioChannels);
  228. }
  229. } else if (fParams.fSendAudioChannels == -1) {
  230. fParams.fSendAudioChannels = fRequest.audio_input;
  231. jack_info("Takes master %d inputs", fRequest.audio_input);
  232. } else if (fParams.fSendAudioChannels != fRequest.audio_input) {
  233. jack_error("Error : master wants %d inputs and slave wants %d inputs...", fRequest.audio_input, fParams.fSendAudioChannels);
  234. return -1;
  235. }
  236. if (fRequest.audio_output == -1) {
  237. if (fParams.fReturnAudioChannels == -1) {
  238. jack_error("Error : master and slave use -1 for wanted outputs...");
  239. return -1;
  240. } else {
  241. result->audio_output = fParams.fReturnAudioChannels;
  242. jack_info("Takes slave %d outputs", fParams.fReturnAudioChannels);
  243. }
  244. } else if (fParams.fReturnAudioChannels == -1) {
  245. fParams.fReturnAudioChannels = fRequest.audio_output;
  246. jack_info("Takes master %d outputs", fRequest.audio_output);
  247. } else if (fParams.fReturnAudioChannels != fRequest.audio_output) {
  248. jack_error("Error : master wants %d outputs and slave wants %d outputs...", fRequest.audio_output, fParams.fReturnAudioChannels);
  249. return -1;
  250. }
  251. // Close request socket
  252. fSocket.Close();
  253. /// Network init
  254. if (!JackNetMasterInterface::Init()) {
  255. return -1;
  256. }
  257. // Set global parameters
  258. if (!SetParams()) {
  259. return -1;
  260. }
  261. return 0;
  262. }
  263. int Close()
  264. {
  265. fSocket.Close();
  266. return 0;
  267. }
  268. int Read(int audio_input, float** audio_input_buffer, int midi_input, void** midi_input_buffer)
  269. {
  270. try {
  271. assert(audio_input == fParams.fReturnAudioChannels);
  272. for (int audio_port_index = 0; audio_port_index < audio_input; audio_port_index++) {
  273. fNetAudioPlaybackBuffer->SetBuffer(audio_port_index, audio_input_buffer[audio_port_index]);
  274. }
  275. for (int midi_port_index = 0; midi_port_index < midi_input; midi_port_index++) {
  276. fNetMidiPlaybackBuffer->SetBuffer(midi_port_index, ((JackMidiBuffer**)midi_input_buffer)[midi_port_index]);
  277. }
  278. int res = SyncRecv();
  279. switch (res) {
  280. case NET_SYNCHING:
  281. // Data will not be received, so cleanup buffers...
  282. for (int audio_port_index = 0; audio_port_index < audio_input; audio_port_index++) {
  283. memset(audio_input_buffer[audio_port_index], 0, sizeof(float) * fParams.fPeriodSize);
  284. }
  285. return res;
  286. case SOCKET_ERROR:
  287. return res;
  288. case SYNC_PACKET_ERROR:
  289. // since sync packet is incorrect, don't decode it and continue with data
  290. break;
  291. default:
  292. // decode sync
  293. DecodeSyncPacket();
  294. break;
  295. }
  296. return DataRecv();
  297. } catch (JackNetException& e) {
  298. jack_error("Lost connection");
  299. return -1;
  300. }
  301. }
  302. int Write(int audio_output, float** audio_output_buffer, int midi_output, void** midi_output_buffer)
  303. {
  304. try {
  305. assert(audio_output == fParams.fSendAudioChannels);
  306. for (int audio_port_index = 0; audio_port_index < audio_output; audio_port_index++) {
  307. fNetAudioCaptureBuffer->SetBuffer(audio_port_index, audio_output_buffer[audio_port_index]);
  308. }
  309. for (int midi_port_index = 0; midi_port_index < midi_output; midi_port_index++) {
  310. fNetMidiCaptureBuffer->SetBuffer(midi_port_index, ((JackMidiBuffer**)midi_output_buffer)[midi_port_index]);
  311. }
  312. EncodeSyncPacket();
  313. // send sync
  314. if (SyncSend() == SOCKET_ERROR) {
  315. return SOCKET_ERROR;
  316. }
  317. // send data
  318. if (DataSend() == SOCKET_ERROR) {
  319. return SOCKET_ERROR;
  320. }
  321. return 0;
  322. } catch (JackNetException& e) {
  323. jack_error("Lost connection");
  324. return -1;
  325. }
  326. }
  327. // Transport
  328. void EncodeTransportData()
  329. {}
  330. void DecodeTransportData()
  331. {}
  332. };
  333. struct JackNetExtSlave : public JackNetSlaveInterface, public JackRunnableInterface {
  334. // Data buffers
  335. float** fAudioCaptureBuffer;
  336. float** fAudioPlaybackBuffer;
  337. JackMidiBuffer** fMidiCaptureBuffer;
  338. JackMidiBuffer** fMidiPlaybackBuffer;
  339. JackThread fThread;
  340. JackNetSlaveProcessCallback fProcessCallback;
  341. void* fProcessArg;
  342. JackNetSlaveShutdownCallback fShutdownCallback;
  343. void* fShutdownArg;
  344. JackNetSlaveRestartCallback fRestartCallback;
  345. void* fRestartArg;
  346. JackNetSlaveErrorCallback fErrorCallback;
  347. void* fErrorArg;
  348. JackNetSlaveBufferSizeCallback fBufferSizeCallback;
  349. void* fBufferSizeArg;
  350. JackNetSlaveSampleRateCallback fSampleRateCallback;
  351. void* fSampleRateArg;
  352. int fConnectTimeOut;
  353. JackNetExtSlave(const char* ip,
  354. int port,
  355. const char* name,
  356. jack_slave_t* request)
  357. :fThread(this),
  358. fProcessCallback(NULL),fProcessArg(NULL),
  359. fShutdownCallback(NULL), fShutdownArg(NULL),
  360. fRestartCallback(NULL), fRestartArg(NULL),
  361. fErrorCallback(NULL), fErrorArg(NULL),
  362. fBufferSizeCallback(NULL), fBufferSizeArg(NULL),
  363. fSampleRateCallback(NULL), fSampleRateArg(NULL)
  364. {
  365. char host_name[JACK_CLIENT_NAME_SIZE];
  366. // Request parameters
  367. assert(strlen(ip) < 32);
  368. strcpy(fMulticastIP, ip);
  369. fParams.fMtu = request->mtu;
  370. fParams.fTransportSync = 0;
  371. fParams.fSendAudioChannels = request->audio_input;
  372. fParams.fReturnAudioChannels = request->audio_output;
  373. fParams.fSendMidiChannels = request->midi_input;
  374. fParams.fReturnMidiChannels = request->midi_output;
  375. fParams.fNetworkLatency = request->latency;
  376. fParams.fSampleEncoder = request->encoder;
  377. fParams.fKBps = request->kbps;
  378. fParams.fSlaveSyncMode = 1;
  379. fConnectTimeOut = request->time_out;
  380. // Create name with hostname and client name
  381. GetHostName(host_name, JACK_CLIENT_NAME_SIZE);
  382. snprintf(fParams.fName, JACK_CLIENT_NAME_SIZE, "%s_%s", host_name, name);
  383. fSocket.GetName(fParams.fSlaveNetName);
  384. // Set the socket parameters
  385. fSocket.SetPort(port);
  386. fSocket.SetAddress(fMulticastIP, port);
  387. fAudioCaptureBuffer = NULL;
  388. fAudioPlaybackBuffer = NULL;
  389. fMidiCaptureBuffer = NULL;
  390. fMidiPlaybackBuffer = NULL;
  391. }
  392. virtual ~JackNetExtSlave()
  393. {}
  394. void AllocPorts()
  395. {
  396. // Set buffers
  397. if (fParams.fSendAudioChannels > 0) {
  398. fAudioCaptureBuffer = new float*[fParams.fSendAudioChannels];
  399. for (int audio_port_index = 0; audio_port_index < fParams.fSendAudioChannels; audio_port_index++) {
  400. fAudioCaptureBuffer[audio_port_index] = new float[fParams.fPeriodSize];
  401. memset(fAudioCaptureBuffer[audio_port_index], 0, sizeof(float) * fParams.fPeriodSize);
  402. fNetAudioCaptureBuffer->SetBuffer(audio_port_index, fAudioCaptureBuffer[audio_port_index]);
  403. }
  404. }
  405. if (fParams.fSendMidiChannels > 0) {
  406. fMidiCaptureBuffer = new JackMidiBuffer*[fParams.fSendMidiChannels];
  407. for (int midi_port_index = 0; midi_port_index < fParams.fSendMidiChannels; midi_port_index++) {
  408. fMidiCaptureBuffer[midi_port_index] = (JackMidiBuffer*)new float[fParams.fPeriodSize];
  409. memset(fMidiCaptureBuffer[midi_port_index], 0, sizeof(float) * fParams.fPeriodSize);
  410. fNetMidiCaptureBuffer->SetBuffer(midi_port_index, fMidiCaptureBuffer[midi_port_index]);
  411. }
  412. }
  413. if (fParams.fReturnAudioChannels > 0) {
  414. fAudioPlaybackBuffer = new float*[fParams.fReturnAudioChannels];
  415. for (int audio_port_index = 0; audio_port_index < fParams.fReturnAudioChannels; audio_port_index++) {
  416. fAudioPlaybackBuffer[audio_port_index] = new float[fParams.fPeriodSize];
  417. memset(fAudioPlaybackBuffer[audio_port_index], 0, sizeof(float) * fParams.fPeriodSize);
  418. fNetAudioPlaybackBuffer->SetBuffer(audio_port_index, fAudioPlaybackBuffer[audio_port_index]);
  419. }
  420. }
  421. if (fParams.fReturnMidiChannels > 0) {
  422. fMidiPlaybackBuffer = new JackMidiBuffer*[fParams.fReturnMidiChannels];
  423. for (int midi_port_index = 0; midi_port_index < fParams.fReturnMidiChannels; midi_port_index++) {
  424. fMidiPlaybackBuffer[midi_port_index] = (JackMidiBuffer*)new float[fParams.fPeriodSize];
  425. memset(fMidiPlaybackBuffer[midi_port_index], 0, sizeof(float) * fParams.fPeriodSize);
  426. fNetMidiPlaybackBuffer->SetBuffer(midi_port_index, fMidiPlaybackBuffer[midi_port_index]);
  427. }
  428. }
  429. }
  430. void FreePorts()
  431. {
  432. if (fAudioCaptureBuffer) {
  433. for (int audio_port_index = 0; audio_port_index < fParams.fSendAudioChannels; audio_port_index++) {
  434. delete[] fAudioCaptureBuffer[audio_port_index];
  435. }
  436. delete[] fAudioCaptureBuffer;
  437. fAudioCaptureBuffer = NULL;
  438. }
  439. if (fMidiCaptureBuffer) {
  440. for (int midi_port_index = 0; midi_port_index < fParams.fSendMidiChannels; midi_port_index++) {
  441. delete[] fMidiCaptureBuffer[midi_port_index];
  442. }
  443. delete[] fMidiCaptureBuffer;
  444. fMidiCaptureBuffer = NULL;
  445. }
  446. if (fAudioPlaybackBuffer) {
  447. for (int audio_port_index = 0; audio_port_index < fParams.fReturnAudioChannels; audio_port_index++) {
  448. delete[] fAudioPlaybackBuffer[audio_port_index];
  449. }
  450. delete[] fAudioPlaybackBuffer;
  451. fAudioPlaybackBuffer = NULL;
  452. }
  453. if (fMidiPlaybackBuffer) {
  454. for (int midi_port_index = 0; midi_port_index < fParams.fReturnMidiChannels; midi_port_index++) {
  455. delete[] (fMidiPlaybackBuffer[midi_port_index]);
  456. }
  457. delete[] fMidiPlaybackBuffer;
  458. fMidiPlaybackBuffer = NULL;
  459. }
  460. }
  461. int Open(jack_master_t* result)
  462. {
  463. // Check CELT encoder parameters
  464. if ((fParams.fSampleEncoder == JackCeltEncoder) && (fParams.fKBps == 0)) {
  465. jack_error("CELT encoder with 0 for kps...");
  466. return -1;
  467. }
  468. if ((fParams.fSampleEncoder == JackOpusEncoder) && (fParams.fKBps == 0)) {
  469. jack_error("Opus encoder with 0 for kps...");
  470. return -1;
  471. }
  472. // Check latency
  473. if (fParams.fNetworkLatency > NETWORK_MAX_LATENCY) {
  474. jack_error("Error : network latency is limited to %d", NETWORK_MAX_LATENCY);
  475. return -1;
  476. }
  477. // Init network connection
  478. if (!JackNetSlaveInterface::InitConnection(fConnectTimeOut)) {
  479. jack_error("Initing network fails...");
  480. return -1;
  481. }
  482. // Finish connection...
  483. if (!JackNetSlaveInterface::InitRendering()) {
  484. jack_error("Starting network fails...");
  485. return -1;
  486. }
  487. // Then set global parameters
  488. if (!SetParams()) {
  489. jack_error("SetParams error...");
  490. return -1;
  491. }
  492. // Set result
  493. if (result != NULL) {
  494. result->buffer_size = fParams.fPeriodSize;
  495. result->sample_rate = fParams.fSampleRate;
  496. result->audio_input = fParams.fSendAudioChannels;
  497. result->audio_output = fParams.fReturnAudioChannels;
  498. result->midi_input = fParams.fSendMidiChannels;
  499. result->midi_output = fParams.fReturnMidiChannels;
  500. strcpy(result->master_name, fParams.fMasterNetName);
  501. }
  502. AllocPorts();
  503. return 0;
  504. }
  505. int Restart()
  506. {
  507. // Do it until client possibly decides to stop trying to connect...
  508. while (true) {
  509. // If restart cb is set, then call it
  510. if (fRestartCallback) {
  511. if (fRestartCallback(fRestartArg) != 0) {
  512. return -1;
  513. }
  514. // Otherwise if shutdown cb is set, then call it
  515. } else if (fShutdownCallback) {
  516. fShutdownCallback(fShutdownArg);
  517. }
  518. // Init network connection
  519. if (!JackNetSlaveInterface::InitConnection(fConnectTimeOut)) {
  520. jack_error("Initing network fails after time_out, retry...");
  521. } else {
  522. break;
  523. }
  524. }
  525. // Finish connection
  526. if (!JackNetSlaveInterface::InitRendering()) {
  527. jack_error("Starting network fails...");
  528. return -1;
  529. }
  530. // Then set global parameters
  531. if (!SetParams()) {
  532. jack_error("SetParams error...");
  533. return -1;
  534. }
  535. // We need to notify possibly new buffer size and sample rate (see Execute)
  536. if (fBufferSizeCallback) {
  537. if (fBufferSizeCallback(fParams.fPeriodSize, fBufferSizeArg) != 0) {
  538. jack_error("New buffer size = %d cannot be used...", fParams.fPeriodSize);
  539. return -1;
  540. }
  541. }
  542. if (fSampleRateCallback) {
  543. if (fSampleRateCallback(fParams.fSampleRate, fSampleRateArg) != 0) {
  544. jack_error("New sample rate = %d cannot be used...", fParams.fSampleRate);
  545. return -1;
  546. }
  547. }
  548. AllocPorts();
  549. return 0;
  550. }
  551. int Close()
  552. {
  553. fSocket.Close();
  554. FreePorts();
  555. return 0;
  556. }
  557. // Transport
  558. void EncodeTransportData()
  559. {}
  560. void DecodeTransportData()
  561. {}
  562. bool Init()
  563. {
  564. // Will do "something" on OSX only...
  565. UInt64 period, constraint;
  566. period = constraint = UInt64(1000000000.f * (float(fParams.fPeriodSize) / float(fParams.fSampleRate)));
  567. UInt64 computation = JackTools::ComputationMicroSec(fParams.fPeriodSize) * 1000;
  568. fThread.SetParams(period, computation, constraint);
  569. return (fThread.AcquireSelfRealTime(80) == 0); // TODO: get a value from the server
  570. }
  571. bool IsRunning()
  572. {
  573. return (fThread.GetStatus() == JackThread::kRunning);
  574. }
  575. bool Execute()
  576. {
  577. try {
  578. /*
  579. Fist cycle use an INT_MAX time out, so that connection
  580. is considered established (with PACKET_TIMEOUT later on)
  581. when the first cycle has been done.
  582. */
  583. DummyProcess();
  584. // keep running even in case of error
  585. while (fThread.GetStatus() == JackThread::kRunning) {
  586. if (Process() == SOCKET_ERROR) {
  587. return false;
  588. }
  589. }
  590. return false;
  591. } catch (JackNetException& e) {
  592. // otherwise just restart...
  593. e.PrintMessage();
  594. jack_info("NetSlave is restarted");
  595. fThread.DropRealTime();
  596. fThread.SetStatus(JackThread::kIniting);
  597. FreePorts();
  598. if (Restart() == 0 && Init()) {
  599. fThread.SetStatus(JackThread::kRunning);
  600. return true;
  601. } else {
  602. return false;
  603. }
  604. }
  605. }
  606. int Read()
  607. {
  608. // receive sync (launch the cycle)
  609. switch (SyncRecv()) {
  610. case SOCKET_ERROR:
  611. return SOCKET_ERROR;
  612. case SYNC_PACKET_ERROR:
  613. // since sync packet is incorrect, don't decode it and continue with data
  614. if (fErrorCallback) {
  615. fErrorCallback(SYNC_PACKET_ERROR, fErrorArg);
  616. }
  617. break;
  618. default:
  619. // decode sync
  620. DecodeSyncPacket();
  621. break;
  622. }
  623. int res = DataRecv();
  624. if (res == DATA_PACKET_ERROR && fErrorCallback) {
  625. fErrorCallback(DATA_PACKET_ERROR, fErrorArg);
  626. }
  627. return res;
  628. }
  629. int Write()
  630. {
  631. EncodeSyncPacket();
  632. if (SyncSend() == SOCKET_ERROR) {
  633. return SOCKET_ERROR;
  634. }
  635. return DataSend();
  636. }
  637. void DummyProcess()
  638. {
  639. // First cycle with INT_MAX time out
  640. SetPacketTimeOut(INT_MAX);
  641. // One cycle
  642. Process();
  643. // Then use PACKET_TIMEOUT for next cycles
  644. SetPacketTimeOut(PACKET_TIMEOUT);
  645. }
  646. int Process()
  647. {
  648. // Read data from the network, throw JackNetException in case of network error...
  649. if (Read() == SOCKET_ERROR) {
  650. return SOCKET_ERROR;
  651. }
  652. fProcessCallback(fParams.fPeriodSize,
  653. fParams.fSendAudioChannels,
  654. fAudioCaptureBuffer,
  655. fParams.fSendMidiChannels,
  656. (void**)fMidiCaptureBuffer,
  657. fParams.fReturnAudioChannels,
  658. fAudioPlaybackBuffer,
  659. fParams.fReturnMidiChannels,
  660. (void**)fMidiPlaybackBuffer,
  661. fProcessArg);
  662. // Then write data to network, throw JackNetException in case of network error...
  663. if (Write() == SOCKET_ERROR) {
  664. return SOCKET_ERROR;
  665. }
  666. return 0;
  667. }
  668. int Start()
  669. {
  670. return (fProcessCallback == 0) ? -1 : fThread.StartSync();
  671. }
  672. int Stop()
  673. {
  674. return (fProcessCallback == 0) ? -1 : fThread.Kill();
  675. }
  676. // Callback
  677. int SetProcessCallback(JackNetSlaveProcessCallback net_callback, void *arg)
  678. {
  679. if (fThread.GetStatus() == JackThread::kRunning) {
  680. return -1;
  681. } else {
  682. fProcessCallback = net_callback;
  683. fProcessArg = arg;
  684. return 0;
  685. }
  686. }
  687. int SetShutdownCallback(JackNetSlaveShutdownCallback shutdown_callback, void *arg)
  688. {
  689. if (fThread.GetStatus() == JackThread::kRunning) {
  690. return -1;
  691. } else {
  692. fShutdownCallback = shutdown_callback;
  693. fShutdownArg = arg;
  694. return 0;
  695. }
  696. }
  697. int SetRestartCallback(JackNetSlaveRestartCallback restart_callback, void *arg)
  698. {
  699. if (fThread.GetStatus() == JackThread::kRunning) {
  700. return -1;
  701. } else {
  702. fRestartCallback = restart_callback;
  703. fRestartArg = arg;
  704. return 0;
  705. }
  706. }
  707. int SetErrorCallback(JackNetSlaveErrorCallback error_callback, void *arg)
  708. {
  709. if (fThread.GetStatus() == JackThread::kRunning) {
  710. return -1;
  711. } else {
  712. fErrorCallback = error_callback;
  713. fErrorArg = arg;
  714. return 0;
  715. }
  716. }
  717. int SetBufferSizeCallback(JackNetSlaveBufferSizeCallback bufsize_callback, void *arg)
  718. {
  719. if (fThread.GetStatus() == JackThread::kRunning) {
  720. return -1;
  721. } else {
  722. fBufferSizeCallback = bufsize_callback;
  723. fBufferSizeArg = arg;
  724. return 0;
  725. }
  726. }
  727. int SetSampleRateCallback(JackNetSlaveSampleRateCallback samplerate_callback, void *arg)
  728. {
  729. if (fThread.GetStatus() == JackThread::kRunning) {
  730. return -1;
  731. } else {
  732. fSampleRateCallback = samplerate_callback;
  733. fSampleRateArg = arg;
  734. return 0;
  735. }
  736. }
  737. };
  738. struct JackNetAdapter : public JackAudioAdapterInterface {
  739. JackNetAdapter(int input, int output,
  740. jack_nframes_t host_buffer_size,
  741. jack_nframes_t host_sample_rate,
  742. jack_nframes_t adapted_buffer_size,
  743. jack_nframes_t adapted_sample_rate)
  744. :JackAudioAdapterInterface(host_buffer_size, host_sample_rate, adapted_buffer_size, adapted_sample_rate)
  745. {
  746. fCaptureChannels = input;
  747. fPlaybackChannels = output;
  748. Create();
  749. }
  750. void Create()
  751. {
  752. //ringbuffers
  753. if (fCaptureChannels > 0) {
  754. fCaptureRingBuffer = new JackResampler*[fCaptureChannels];
  755. }
  756. if (fPlaybackChannels > 0) {
  757. fPlaybackRingBuffer = new JackResampler*[fPlaybackChannels];
  758. }
  759. if (fAdaptative) {
  760. AdaptRingBufferSize();
  761. jack_info("Ringbuffer automatic adaptative mode size = %d frames", fRingbufferCurSize);
  762. } else {
  763. if (fRingbufferCurSize > DEFAULT_RB_SIZE) {
  764. fRingbufferCurSize = DEFAULT_RB_SIZE;
  765. }
  766. jack_info("Fixed ringbuffer size = %d frames", fRingbufferCurSize);
  767. }
  768. for (int i = 0; i < fCaptureChannels; i++ ) {
  769. fCaptureRingBuffer[i] = new JackResampler();
  770. fCaptureRingBuffer[i]->Reset(fRingbufferCurSize);
  771. }
  772. for (int i = 0; i < fPlaybackChannels; i++ ) {
  773. fPlaybackRingBuffer[i] = new JackResampler();
  774. fPlaybackRingBuffer[i]->Reset(fRingbufferCurSize);
  775. }
  776. if (fCaptureChannels > 0) {
  777. jack_log("ReadSpace = %ld", fCaptureRingBuffer[0]->ReadSpace());
  778. }
  779. if (fPlaybackChannels > 0) {
  780. jack_log("WriteSpace = %ld", fPlaybackRingBuffer[0]->WriteSpace());
  781. }
  782. }
  783. virtual ~JackNetAdapter()
  784. {
  785. Destroy();
  786. }
  787. void Flush()
  788. {
  789. for (int i = 0; i < fCaptureChannels; i++ ) {
  790. fCaptureRingBuffer[i]->Reset(fRingbufferCurSize);
  791. }
  792. for (int i = 0; i < fPlaybackChannels; i++ ) {
  793. fPlaybackRingBuffer[i]->Reset(fRingbufferCurSize);
  794. }
  795. }
  796. };
  797. } // end of namespace
  798. using namespace Jack;
  799. LIB_EXPORT jack_net_slave_t* jack_net_slave_open(const char* ip, int port, const char* name, jack_slave_t* request, jack_master_t* result)
  800. {
  801. JackNetExtSlave* slave = new JackNetExtSlave(ip, port, name, request);
  802. if (slave->Open(result) == 0) {
  803. return (jack_net_slave_t*)slave;
  804. } else {
  805. delete slave;
  806. return NULL;
  807. }
  808. }
  809. LIB_EXPORT int jack_net_slave_close(jack_net_slave_t* net)
  810. {
  811. JackNetExtSlave* slave = (JackNetExtSlave*)net;
  812. slave->Close();
  813. delete slave;
  814. return 0;
  815. }
  816. LIB_EXPORT int jack_set_net_slave_process_callback(jack_net_slave_t* net, JackNetSlaveProcessCallback net_callback, void *arg)
  817. {
  818. JackNetExtSlave* slave = (JackNetExtSlave*)net;
  819. return slave->SetProcessCallback(net_callback, arg);
  820. }
  821. LIB_EXPORT int jack_net_slave_activate(jack_net_slave_t* net)
  822. {
  823. JackNetExtSlave* slave = (JackNetExtSlave*)net;
  824. return slave->Start();
  825. }
  826. LIB_EXPORT int jack_net_slave_deactivate(jack_net_slave_t* net)
  827. {
  828. JackNetExtSlave* slave = (JackNetExtSlave*)net;
  829. return slave->Stop();
  830. }
  831. LIB_EXPORT int jack_net_slave_is_active(jack_net_slave_t* net)
  832. {
  833. JackNetExtSlave* slave = (JackNetExtSlave*)net;
  834. return slave->IsRunning();
  835. }
  836. LIB_EXPORT int jack_set_net_slave_buffer_size_callback(jack_net_slave_t *net, JackNetSlaveBufferSizeCallback bufsize_callback, void *arg)
  837. {
  838. JackNetExtSlave* slave = (JackNetExtSlave*)net;
  839. return slave->SetBufferSizeCallback(bufsize_callback, arg);
  840. }
  841. LIB_EXPORT int jack_set_net_slave_sample_rate_callback(jack_net_slave_t *net, JackNetSlaveSampleRateCallback samplerate_callback, void *arg)
  842. {
  843. JackNetExtSlave* slave = (JackNetExtSlave*)net;
  844. return slave->SetSampleRateCallback(samplerate_callback, arg);
  845. }
  846. LIB_EXPORT int jack_set_net_slave_shutdown_callback(jack_net_slave_t *net, JackNetSlaveShutdownCallback shutdown_callback, void *arg)
  847. {
  848. JackNetExtSlave* slave = (JackNetExtSlave*)net;
  849. return slave->SetShutdownCallback(shutdown_callback, arg);
  850. }
  851. LIB_EXPORT int jack_set_net_slave_restart_callback(jack_net_slave_t *net, JackNetSlaveRestartCallback restart_callback, void *arg)
  852. {
  853. JackNetExtSlave* slave = (JackNetExtSlave*)net;
  854. return slave->SetRestartCallback(restart_callback, arg);
  855. }
  856. LIB_EXPORT int jack_set_net_slave_error_callback(jack_net_slave_t *net, JackNetSlaveErrorCallback error_callback, void *arg)
  857. {
  858. JackNetExtSlave* slave = (JackNetExtSlave*)net;
  859. return slave->SetErrorCallback(error_callback, arg);
  860. }
  861. // Master API
  862. LIB_EXPORT jack_net_master_t* jack_net_master_open(const char* ip, int port, const char* name, jack_master_t* request, jack_slave_t* result)
  863. {
  864. JackNetExtMaster* master = new JackNetExtMaster(ip, port, name, request);
  865. if (master->Open(result) == 0) {
  866. return (jack_net_master_t*)master;
  867. } else {
  868. delete master;
  869. return NULL;
  870. }
  871. }
  872. LIB_EXPORT int jack_net_master_close(jack_net_master_t* net)
  873. {
  874. JackNetExtMaster* master = (JackNetExtMaster*)net;
  875. master->Close();
  876. delete master;
  877. return 0;
  878. }
  879. LIB_EXPORT int jack_net_master_recv(jack_net_master_t* net, int audio_input, float** audio_input_buffer, int midi_input, void** midi_input_buffer)
  880. {
  881. JackNetExtMaster* master = (JackNetExtMaster*)net;
  882. return master->Read(audio_input, audio_input_buffer, midi_input, midi_input_buffer);
  883. }
  884. LIB_EXPORT int jack_net_master_send(jack_net_master_t* net, int audio_output, float** audio_output_buffer, int midi_output, void** midi_output_buffer)
  885. {
  886. JackNetExtMaster* master = (JackNetExtMaster*)net;
  887. return master->Write(audio_output, audio_output_buffer, midi_output, midi_output_buffer);
  888. }
  889. // Adapter API
  890. LIB_EXPORT jack_adapter_t* jack_create_adapter(int input, int output,
  891. jack_nframes_t host_buffer_size,
  892. jack_nframes_t host_sample_rate,
  893. jack_nframes_t adapted_buffer_size,
  894. jack_nframes_t adapted_sample_rate)
  895. {
  896. try {
  897. return (jack_adapter_t*)new JackNetAdapter(input, output, host_buffer_size, host_sample_rate, adapted_buffer_size, adapted_sample_rate);
  898. } catch (...) {
  899. return NULL;
  900. }
  901. }
  902. LIB_EXPORT int jack_destroy_adapter(jack_adapter_t* adapter)
  903. {
  904. delete((JackNetAdapter*)adapter);
  905. return 0;
  906. }
  907. LIB_EXPORT void jack_flush_adapter(jack_adapter_t* adapter)
  908. {
  909. JackNetAdapter* slave = (JackNetAdapter*)adapter;
  910. slave->Flush();
  911. }
  912. LIB_EXPORT int jack_adapter_push_and_pull(jack_adapter_t* adapter, float** input, float** output, unsigned int frames)
  913. {
  914. JackNetAdapter* slave = (JackNetAdapter*)adapter;
  915. return slave->PushAndPull(input, output, frames);
  916. }
  917. LIB_EXPORT int jack_adapter_pull_and_push(jack_adapter_t* adapter, float** input, float** output, unsigned int frames)
  918. {
  919. JackNetAdapter* slave = (JackNetAdapter*)adapter;
  920. return slave->PullAndPush(input, output, frames);
  921. }
  922. static void jack_format_and_log(int level, const char *prefix, const char *fmt, va_list ap)
  923. {
  924. static const char* netjack_log = getenv("JACK_NETJACK_LOG");
  925. static bool is_netjack_log = (netjack_log) ? atoi(netjack_log) : 0;
  926. if (is_netjack_log) {
  927. char buffer[300];
  928. size_t len;
  929. if (prefix != NULL) {
  930. len = strlen(prefix);
  931. memcpy(buffer, prefix, len);
  932. } else {
  933. len = 0;
  934. }
  935. vsnprintf(buffer + len, sizeof(buffer) - len, fmt, ap);
  936. printf("%s", buffer);
  937. printf("\n");
  938. }
  939. }
  940. LIB_EXPORT void jack_error(const char *fmt, ...)
  941. {
  942. va_list ap;
  943. va_start(ap, fmt);
  944. jack_format_and_log(LOG_LEVEL_INFO, "Jack: ", fmt, ap);
  945. va_end(ap);
  946. }
  947. LIB_EXPORT void jack_info(const char *fmt, ...)
  948. {
  949. va_list ap;
  950. va_start(ap, fmt);
  951. jack_format_and_log(LOG_LEVEL_INFO, "Jack: ", fmt, ap);
  952. va_end(ap);
  953. }
  954. LIB_EXPORT void jack_log(const char *fmt, ...)
  955. {
  956. va_list ap;
  957. va_start(ap, fmt);
  958. jack_format_and_log(LOG_LEVEL_INFO, "Jack: ", fmt, ap);
  959. va_end(ap);
  960. }