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.

220 lines
6.1KB

  1. /*
  2. Copyright (C) 2004-2008 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 "JackPosixSemaphore.h"
  16. #include "JackTools.h"
  17. #include "JackConstants.h"
  18. #include "JackError.h"
  19. #include <fcntl.h>
  20. #include <stdio.h>
  21. #include <sys/time.h>
  22. namespace Jack
  23. {
  24. void JackPosixSemaphore::BuildName(const char* client_name, const char* server_name, char* res, int size)
  25. {
  26. char ext_client_name[SYNC_MAX_NAME_SIZE + 1];
  27. JackTools::RewriteName(client_name, ext_client_name);
  28. snprintf(res, size, "jack_sem.%d_%s_%s", JackTools::GetUID(), server_name, ext_client_name);
  29. }
  30. bool JackPosixSemaphore::Signal()
  31. {
  32. int res;
  33. if (!fSemaphore) {
  34. jack_error("JackPosixSemaphore::Signal name = %s already deallocated!!", fName);
  35. return false;
  36. }
  37. if (fFlush)
  38. return true;
  39. if ((res = sem_post(fSemaphore)) != 0) {
  40. jack_error("JackPosixSemaphore::Signal name = %s err = %s", fName, strerror(errno));
  41. }
  42. return (res == 0);
  43. }
  44. bool JackPosixSemaphore::SignalAll()
  45. {
  46. int res;
  47. if (!fSemaphore) {
  48. jack_error("JackPosixSemaphore::SignalAll name = %s already deallocated!!", fName);
  49. return false;
  50. }
  51. if (fFlush)
  52. return true;
  53. if ((res = sem_post(fSemaphore)) != 0) {
  54. jack_error("JackPosixSemaphore::SignalAll name = %s err = %s", fName, strerror(errno));
  55. }
  56. return (res == 0);
  57. }
  58. /*
  59. bool JackPosixSemaphore::Wait()
  60. {
  61. int res;
  62. if (!fSemaphore) {
  63. jack_error("JackPosixSemaphore::Wait name = %s already deallocated!!", fName);
  64. return false;
  65. }
  66. if ((res = sem_wait(fSemaphore)) != 0) {
  67. jack_error("JackPosixSemaphore::Wait name = %s err = %s", fName, strerror(errno));
  68. }
  69. return (res == 0);
  70. }
  71. */
  72. bool JackPosixSemaphore::Wait()
  73. {
  74. int res;
  75. while ((res = sem_wait(fSemaphore) < 0)) {
  76. jack_error("JackPosixSemaphore::Wait name = %s err = %s", fName, strerror(errno));
  77. if (errno != EINTR)
  78. break;
  79. }
  80. return (res == 0);
  81. }
  82. #if (_POSIX_C_SOURCE >= 200112L || _XOPEN_SOURCE >= 600) // glibc feature test
  83. bool JackPosixSemaphore::TimedWait(long usec)
  84. {
  85. int res;
  86. struct timeval now;
  87. timespec time;
  88. if (!fSemaphore) {
  89. jack_error("JackPosixSemaphore::TimedWait name = %s already deallocated!!", fName);
  90. return false;
  91. }
  92. gettimeofday(&now, 0);
  93. time.tv_sec = now.tv_sec + usec / 1000000;
  94. long tv_usec = (now.tv_usec + (usec % 1000000));
  95. time.tv_sec += tv_usec / 1000000;
  96. time.tv_nsec = (tv_usec % 1000000) * 1000;
  97. while ((res = sem_timedwait(fSemaphore, &time)) < 0) {
  98. jack_error("JackPosixSemaphore::TimedWait err = %s", strerror(errno));
  99. jack_log("JackPosixSemaphore::TimedWait now : %ld %ld ", now.tv_sec, now.tv_usec);
  100. jack_log("JackPosixSemaphore::TimedWait next : %ld %ld ", time.tv_sec, time.tv_nsec/1000);
  101. if (errno != EINTR)
  102. break;
  103. }
  104. return (res == 0);
  105. }
  106. #else
  107. #warning "JackPosixSemaphore::TimedWait is not supported: Jack in SYNC mode with JackPosixSemaphore will not run properly !!"
  108. bool JackPosixSemaphore::TimedWait(long usec)
  109. {
  110. return Wait();
  111. }
  112. #endif
  113. // Server side : publish the semaphore in the global namespace
  114. bool JackPosixSemaphore::Allocate(const char* name, const char* server_name, int value)
  115. {
  116. BuildName(name, server_name, fName, sizeof(fName));
  117. jack_log("JackPosixSemaphore::Allocate name = %s val = %ld", fName, value);
  118. if ((fSemaphore = sem_open(fName, O_CREAT, 0777, value)) == (sem_t*)SEM_FAILED) {
  119. jack_error("Allocate: can't check in named semaphore name = %s err = %s", fName, strerror(errno));
  120. return false;
  121. } else {
  122. return true;
  123. }
  124. }
  125. // Client side : get the published semaphore from server
  126. bool JackPosixSemaphore::ConnectInput(const char* name, const char* server_name)
  127. {
  128. BuildName(name, server_name, fName, sizeof(fName));
  129. jack_log("JackPosixSemaphore::Connect name = %s", fName);
  130. // Temporary...
  131. if (fSemaphore) {
  132. jack_log("Already connected name = %s", name);
  133. return true;
  134. }
  135. if ((fSemaphore = sem_open(fName, O_CREAT)) == (sem_t*)SEM_FAILED) {
  136. jack_error("Connect: can't connect named semaphore name = %s err = %s", fName, strerror(errno));
  137. return false;
  138. } else {
  139. int val = 0;
  140. sem_getvalue(fSemaphore, &val);
  141. jack_log("JackPosixSemaphore::Connect sem_getvalue %ld", val);
  142. return true;
  143. }
  144. }
  145. bool JackPosixSemaphore::Connect(const char* name, const char* server_name)
  146. {
  147. return ConnectInput(name, server_name);
  148. }
  149. bool JackPosixSemaphore::ConnectOutput(const char* name, const char* server_name)
  150. {
  151. return ConnectInput(name, server_name);
  152. }
  153. bool JackPosixSemaphore::Disconnect()
  154. {
  155. if (fSemaphore) {
  156. jack_log("JackPosixSemaphore::Disconnect name = %s", fName);
  157. if (sem_close(fSemaphore) != 0) {
  158. jack_error("Disconnect: can't disconnect named semaphore name = %s err = %s", fName, strerror(errno));
  159. return false;
  160. } else {
  161. fSemaphore = NULL;
  162. return true;
  163. }
  164. } else {
  165. return true;
  166. }
  167. }
  168. // Server side : destroy the semaphore
  169. void JackPosixSemaphore::Destroy()
  170. {
  171. if (fSemaphore != NULL) {
  172. jack_log("JackPosixSemaphore::Destroy name = %s", fName);
  173. sem_unlink(fName);
  174. if (sem_close(fSemaphore) != 0) {
  175. jack_error("Destroy: can't destroy semaphore name = %s err = %s", fName, strerror(errno));
  176. }
  177. fSemaphore = NULL;
  178. } else {
  179. jack_error("JackPosixSemaphore::Destroy semaphore == NULL");
  180. }
  181. }
  182. } // end of namespace