capi_track.cpp 5.6 KB

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