capi_track.cpp 6.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236
  1. /**
  2. * Copyright (c) 2020 Paul-Louis Ageneau
  3. *
  4. * This library is free software; you can redistribute it and/or
  5. * modify it under the terms of the GNU Lesser General Public
  6. * License as published by the Free Software Foundation; either
  7. * version 2.1 of the License, or (at your option) any later version.
  8. *
  9. * This library is distributed in the hope that it will be useful,
  10. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  11. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
  12. * Lesser General Public License for more details.
  13. *
  14. * You should have received a copy of the GNU Lesser General Public
  15. * License along with this library; if not, write to the Free Software
  16. * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
  17. */
  18. #include <rtc/rtc.h>
  19. #include <cstdio>
  20. #include <cstdlib>
  21. #include <cstring>
  22. #ifdef _WIN32
  23. #include <windows.h>
  24. static void sleep(unsigned int secs) { Sleep(secs * 1000); }
  25. #else
  26. #include <unistd.h> // for sleep
  27. #endif
  28. typedef struct {
  29. rtcState state;
  30. rtcGatheringState gatheringState;
  31. int pc;
  32. int tr;
  33. bool connected;
  34. } Peer;
  35. static Peer *peer1 = NULL;
  36. static Peer *peer2 = NULL;
  37. static const char *mediaDescription = "video 9 UDP/TLS/RTP/SAVPF\r\n"
  38. "a=mid:video\r\n"
  39. "a=sendonly\r\n";
  40. static void RTC_API descriptionCallback(int pc, const char *sdp, const char *type, void *ptr) {
  41. Peer *peer = (Peer *)ptr;
  42. printf("Description %d:\n%s\n", peer == peer1 ? 1 : 2, sdp);
  43. Peer *other = peer == peer1 ? peer2 : peer1;
  44. rtcSetRemoteDescription(other->pc, sdp, type);
  45. }
  46. static void RTC_API candidateCallback(int pc, const char *cand, const char *mid, void *ptr) {
  47. Peer *peer = (Peer *)ptr;
  48. printf("Candidate %d: %s\n", peer == peer1 ? 1 : 2, cand);
  49. Peer *other = peer == peer1 ? peer2 : peer1;
  50. rtcAddRemoteCandidate(other->pc, cand, mid);
  51. }
  52. static void RTC_API stateChangeCallback(int pc, rtcState state, void *ptr) {
  53. Peer *peer = (Peer *)ptr;
  54. peer->state = state;
  55. printf("State %d: %d\n", peer == peer1 ? 1 : 2, (int)state);
  56. }
  57. static void RTC_API gatheringStateCallback(int pc, rtcGatheringState state, void *ptr) {
  58. Peer *peer = (Peer *)ptr;
  59. peer->gatheringState = state;
  60. printf("Gathering state %d: %d\n", peer == peer1 ? 1 : 2, (int)state);
  61. }
  62. static void RTC_API openCallback(int id, void *ptr) {
  63. Peer *peer = (Peer *)ptr;
  64. peer->connected = true;
  65. printf("Track %d: Open\n", peer == peer1 ? 1 : 2);
  66. }
  67. static void RTC_API closedCallback(int id, void *ptr) {
  68. Peer *peer = (Peer *)ptr;
  69. peer->connected = false;
  70. printf("Track %d: Closed\n", peer == peer1 ? 1 : 2);
  71. }
  72. static void RTC_API trackCallback(int pc, int tr, void *ptr) {
  73. Peer *peer = (Peer *)ptr;
  74. char buffer[1024];
  75. if (rtcGetTrackDescription(tr, buffer, 1024) < 0) {
  76. fprintf(stderr, "rtcGetTrackDescription failed\n");
  77. return;
  78. }
  79. printf("Track %d: Received with media description: \n%s\n", peer == peer1 ? 1 : 2, buffer);
  80. char mid[256];
  81. if (rtcGetTrackMid(tr, mid, 256) < 0 || strcmp(mid, "video") != 0) {
  82. fprintf(stderr, "rtcGetTrackMid failed\n");
  83. return;
  84. }
  85. // Description is reversed here
  86. rtcDirection direction;
  87. if (rtcGetTrackDirection(tr, &direction) < 0 || direction != RTC_DIRECTION_RECVONLY) {
  88. fprintf(stderr, "rtcGetTrackDirection failed\n");
  89. return;
  90. }
  91. peer->tr = tr;
  92. rtcSetOpenCallback(tr, openCallback);
  93. rtcSetClosedCallback(tr, closedCallback);
  94. }
  95. static Peer *createPeer(const rtcConfiguration *config) {
  96. Peer *peer = (Peer *)malloc(sizeof(Peer));
  97. if (!peer)
  98. return nullptr;
  99. memset(peer, 0, sizeof(Peer));
  100. // Create peer connection
  101. peer->pc = rtcCreatePeerConnection(config);
  102. if (peer->pc < 0) {
  103. fprintf(stderr, "PeerConnection creation failed\n");
  104. free(peer);
  105. return nullptr;
  106. }
  107. rtcSetUserPointer(peer->pc, peer);
  108. rtcSetTrackCallback(peer->pc, trackCallback);
  109. rtcSetLocalDescriptionCallback(peer->pc, descriptionCallback);
  110. rtcSetLocalCandidateCallback(peer->pc, candidateCallback);
  111. rtcSetStateChangeCallback(peer->pc, stateChangeCallback);
  112. rtcSetGatheringStateChangeCallback(peer->pc, gatheringStateCallback);
  113. peer->tr = -1;
  114. return peer;
  115. }
  116. static void deletePeer(Peer *peer) {
  117. if (peer) {
  118. if (peer->tr >= 0)
  119. rtcDeleteTrack(peer->tr);
  120. if (peer->pc >= 0)
  121. rtcDeletePeerConnection(peer->pc);
  122. free(peer);
  123. }
  124. }
  125. int test_capi_track_main() {
  126. int attempts;
  127. rtcInitLogger(RTC_LOG_DEBUG, nullptr);
  128. // Create peer 1
  129. rtcConfiguration config1;
  130. memset(&config1, 0, sizeof(config1));
  131. // STUN server example
  132. // const char *iceServers[1] = {"stun:stun.l.google.com:19302"};
  133. // config1.iceServers = iceServers;
  134. // config1.iceServersCount = 1;
  135. peer1 = createPeer(&config1);
  136. if (!peer1)
  137. goto error;
  138. // Create peer 2
  139. rtcConfiguration config2;
  140. memset(&config2, 0, sizeof(config2));
  141. // STUN server example
  142. // config2.iceServers = iceServers;
  143. // config2.iceServersCount = 1;
  144. // Port range example
  145. config2.portRangeBegin = 5000;
  146. config2.portRangeEnd = 6000;
  147. peer2 = createPeer(&config2);
  148. if (!peer2)
  149. goto error;
  150. // Peer 1: Create track
  151. peer1->tr = rtcAddTrack(peer1->pc, mediaDescription);
  152. rtcSetOpenCallback(peer1->tr, openCallback);
  153. rtcSetClosedCallback(peer1->tr, closedCallback);
  154. char mid[256];
  155. if (rtcGetTrackMid(peer1->tr, mid, 256) < 0 || strcmp(mid, "video") != 0) {
  156. fprintf(stderr, "rtcGetTrackMid failed\n");
  157. goto error;
  158. }
  159. rtcDirection direction;
  160. if (rtcGetTrackDirection(peer1->tr, &direction) < 0 || direction != RTC_DIRECTION_SENDONLY) {
  161. fprintf(stderr, "rtcGetTrackDirection failed\n");
  162. goto error;
  163. }
  164. // Initiate the handshake
  165. rtcSetLocalDescription(peer1->pc, NULL);
  166. attempts = 10;
  167. while ((!peer2->connected || !peer1->connected) && attempts--)
  168. sleep(1);
  169. if (peer1->state != RTC_CONNECTED || peer2->state != RTC_CONNECTED) {
  170. fprintf(stderr, "PeerConnection is not connected\n");
  171. goto error;
  172. }
  173. if (!peer1->connected || !peer2->connected) {
  174. fprintf(stderr, "Track is not connected\n");
  175. goto error;
  176. }
  177. deletePeer(peer1);
  178. sleep(1);
  179. deletePeer(peer2);
  180. sleep(1);
  181. printf("Success\n");
  182. return 0;
  183. error:
  184. deletePeer(peer1);
  185. deletePeer(peer2);
  186. return -1;
  187. }
  188. #include <stdexcept>
  189. void test_capi_track() {
  190. if (test_capi_track_main())
  191. throw std::runtime_error("Connection failed");
  192. }