samplebuilder_test.go 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602
  1. // SPDX-FileCopyrightText: 2023 The Pion community <https://pion.ly>
  2. // SPDX-License-Identifier: MIT
  3. package samplebuilder
  4. import (
  5. "fmt"
  6. "testing"
  7. "time"
  8. "github.com/pion/rtp"
  9. "github.com/pion/webrtc/v3/pkg/media"
  10. "github.com/stretchr/testify/assert"
  11. )
  12. type sampleBuilderTest struct {
  13. message string
  14. packets []*rtp.Packet
  15. withHeadChecker bool
  16. headBytes []byte
  17. samples []*media.Sample
  18. maxLate uint16
  19. maxLateTimestamp uint32
  20. }
  21. type fakeDepacketizer struct {
  22. headChecker bool
  23. headBytes []byte
  24. }
  25. func (f *fakeDepacketizer) Unmarshal(r []byte) ([]byte, error) {
  26. return r, nil
  27. }
  28. func (f *fakeDepacketizer) IsPartitionHead(payload []byte) bool {
  29. if !f.headChecker {
  30. // simulates a bug in the 3.0 version
  31. // the tests should be fixed to not assume the bug
  32. return true
  33. }
  34. for _, b := range f.headBytes {
  35. if payload[0] == b {
  36. return true
  37. }
  38. }
  39. return false
  40. }
  41. func (f *fakeDepacketizer) IsPartitionTail(marker bool, _ []byte) bool {
  42. return marker
  43. }
  44. func TestSampleBuilder(t *testing.T) {
  45. testData := []sampleBuilderTest{
  46. {
  47. message: "SampleBuilder shouldn't emit anything if only one RTP packet has been pushed",
  48. packets: []*rtp.Packet{
  49. {Header: rtp.Header{SequenceNumber: 5000, Timestamp: 5}, Payload: []byte{0x01}},
  50. },
  51. samples: []*media.Sample{},
  52. maxLate: 50,
  53. maxLateTimestamp: 0,
  54. },
  55. {
  56. message: "SampleBuilder shouldn't emit anything if only one RTP packet has been pushed even if the market bit is set",
  57. packets: []*rtp.Packet{
  58. {Header: rtp.Header{SequenceNumber: 5000, Timestamp: 5, Marker: true}, Payload: []byte{0x01}},
  59. },
  60. samples: []*media.Sample{},
  61. maxLate: 50,
  62. maxLateTimestamp: 0,
  63. },
  64. {
  65. message: "SampleBuilder should emit two packets, we had three packets with unique timestamps",
  66. packets: []*rtp.Packet{
  67. {Header: rtp.Header{SequenceNumber: 5000, Timestamp: 5}, Payload: []byte{0x01}},
  68. {Header: rtp.Header{SequenceNumber: 5001, Timestamp: 6}, Payload: []byte{0x02}},
  69. {Header: rtp.Header{SequenceNumber: 5002, Timestamp: 7}, Payload: []byte{0x03}},
  70. },
  71. samples: []*media.Sample{
  72. {Data: []byte{0x01}, Duration: time.Second, PacketTimestamp: 5},
  73. {Data: []byte{0x02}, Duration: time.Second, PacketTimestamp: 6},
  74. },
  75. maxLate: 50,
  76. maxLateTimestamp: 0,
  77. },
  78. {
  79. message: "SampleBuilder should emit one packet, we had a packet end of sequence marker and run out of space",
  80. packets: []*rtp.Packet{
  81. {Header: rtp.Header{SequenceNumber: 5000, Timestamp: 5, Marker: true}, Payload: []byte{0x01}},
  82. {Header: rtp.Header{SequenceNumber: 5002, Timestamp: 7}, Payload: []byte{0x02}},
  83. {Header: rtp.Header{SequenceNumber: 5004, Timestamp: 9}, Payload: []byte{0x03}},
  84. {Header: rtp.Header{SequenceNumber: 5006, Timestamp: 11}, Payload: []byte{0x04}},
  85. {Header: rtp.Header{SequenceNumber: 5008, Timestamp: 13}, Payload: []byte{0x05}},
  86. {Header: rtp.Header{SequenceNumber: 5010, Timestamp: 15}, Payload: []byte{0x06}},
  87. {Header: rtp.Header{SequenceNumber: 5012, Timestamp: 17}, Payload: []byte{0x07}},
  88. },
  89. samples: []*media.Sample{
  90. {Data: []byte{0x01}, Duration: time.Second * 2, PacketTimestamp: 5},
  91. },
  92. maxLate: 5,
  93. maxLateTimestamp: 0,
  94. },
  95. {
  96. message: "SampleBuilder shouldn't emit any packet, we do not have a valid end of sequence and run out of space",
  97. packets: []*rtp.Packet{
  98. {Header: rtp.Header{SequenceNumber: 5000, Timestamp: 5}, Payload: []byte{0x01}},
  99. {Header: rtp.Header{SequenceNumber: 5002, Timestamp: 7}, Payload: []byte{0x02}},
  100. {Header: rtp.Header{SequenceNumber: 5004, Timestamp: 9}, Payload: []byte{0x03}},
  101. {Header: rtp.Header{SequenceNumber: 5006, Timestamp: 11}, Payload: []byte{0x04}},
  102. {Header: rtp.Header{SequenceNumber: 5008, Timestamp: 13}, Payload: []byte{0x05}},
  103. {Header: rtp.Header{SequenceNumber: 5010, Timestamp: 15}, Payload: []byte{0x06}},
  104. {Header: rtp.Header{SequenceNumber: 5012, Timestamp: 17}, Payload: []byte{0x07}},
  105. },
  106. samples: []*media.Sample{},
  107. maxLate: 5,
  108. maxLateTimestamp: 0,
  109. },
  110. {
  111. message: "SampleBuilder should emit one packet, we had a packet end of sequence marker and run out of space",
  112. packets: []*rtp.Packet{
  113. {Header: rtp.Header{SequenceNumber: 5000, Timestamp: 5, Marker: true}, Payload: []byte{0x01}},
  114. {Header: rtp.Header{SequenceNumber: 5002, Timestamp: 7, Marker: true}, Payload: []byte{0x02}},
  115. {Header: rtp.Header{SequenceNumber: 5004, Timestamp: 9}, Payload: []byte{0x03}},
  116. {Header: rtp.Header{SequenceNumber: 5006, Timestamp: 11}, Payload: []byte{0x04}},
  117. {Header: rtp.Header{SequenceNumber: 5008, Timestamp: 13}, Payload: []byte{0x05}},
  118. {Header: rtp.Header{SequenceNumber: 5010, Timestamp: 15}, Payload: []byte{0x06}},
  119. {Header: rtp.Header{SequenceNumber: 5012, Timestamp: 17}, Payload: []byte{0x07}},
  120. },
  121. samples: []*media.Sample{
  122. {Data: []byte{0x01}, Duration: time.Second * 2, PacketTimestamp: 5},
  123. {Data: []byte{0x02}, Duration: time.Second * 2, PacketTimestamp: 7, PrevDroppedPackets: 1},
  124. },
  125. maxLate: 5,
  126. maxLateTimestamp: 0,
  127. },
  128. {
  129. message: "SampleBuilder should emit one packet, we had two packets but two with duplicate timestamps",
  130. packets: []*rtp.Packet{
  131. {Header: rtp.Header{SequenceNumber: 5000, Timestamp: 5}, Payload: []byte{0x01}},
  132. {Header: rtp.Header{SequenceNumber: 5001, Timestamp: 6}, Payload: []byte{0x02}},
  133. {Header: rtp.Header{SequenceNumber: 5002, Timestamp: 6}, Payload: []byte{0x03}},
  134. {Header: rtp.Header{SequenceNumber: 5003, Timestamp: 7}, Payload: []byte{0x04}},
  135. },
  136. samples: []*media.Sample{
  137. {Data: []byte{0x01}, Duration: time.Second, PacketTimestamp: 5},
  138. {Data: []byte{0x02, 0x03}, Duration: time.Second, PacketTimestamp: 6},
  139. },
  140. maxLate: 50,
  141. maxLateTimestamp: 0,
  142. },
  143. {
  144. message: "SampleBuilder shouldn't emit a packet because we have a gap before a valid one",
  145. packets: []*rtp.Packet{
  146. {Header: rtp.Header{SequenceNumber: 5000, Timestamp: 5}, Payload: []byte{0x01}},
  147. {Header: rtp.Header{SequenceNumber: 5007, Timestamp: 6}, Payload: []byte{0x02}},
  148. {Header: rtp.Header{SequenceNumber: 5008, Timestamp: 7}, Payload: []byte{0x03}},
  149. },
  150. samples: []*media.Sample{},
  151. maxLate: 50,
  152. maxLateTimestamp: 0,
  153. },
  154. {
  155. message: "SampleBuilder shouldn't emit a packet after a gap as there are gaps and have not reached maxLate yet",
  156. packets: []*rtp.Packet{
  157. {Header: rtp.Header{SequenceNumber: 5000, Timestamp: 5}, Payload: []byte{0x01}},
  158. {Header: rtp.Header{SequenceNumber: 5007, Timestamp: 6}, Payload: []byte{0x02}},
  159. {Header: rtp.Header{SequenceNumber: 5008, Timestamp: 7}, Payload: []byte{0x03}},
  160. },
  161. withHeadChecker: true,
  162. headBytes: []byte{0x02},
  163. samples: []*media.Sample{},
  164. maxLate: 50,
  165. maxLateTimestamp: 0,
  166. },
  167. {
  168. message: "SampleBuilder shouldn't emit a packet after a gap if PartitionHeadChecker doesn't assume it head",
  169. packets: []*rtp.Packet{
  170. {Header: rtp.Header{SequenceNumber: 5000, Timestamp: 5}, Payload: []byte{0x01}},
  171. {Header: rtp.Header{SequenceNumber: 5007, Timestamp: 6}, Payload: []byte{0x02}},
  172. {Header: rtp.Header{SequenceNumber: 5008, Timestamp: 7}, Payload: []byte{0x03}},
  173. },
  174. withHeadChecker: true,
  175. headBytes: []byte{},
  176. samples: []*media.Sample{},
  177. maxLate: 50,
  178. maxLateTimestamp: 0,
  179. },
  180. {
  181. message: "SampleBuilder should emit multiple valid packets",
  182. packets: []*rtp.Packet{
  183. {Header: rtp.Header{SequenceNumber: 5000, Timestamp: 1}, Payload: []byte{0x01}},
  184. {Header: rtp.Header{SequenceNumber: 5001, Timestamp: 2}, Payload: []byte{0x02}},
  185. {Header: rtp.Header{SequenceNumber: 5002, Timestamp: 3}, Payload: []byte{0x03}},
  186. {Header: rtp.Header{SequenceNumber: 5003, Timestamp: 4}, Payload: []byte{0x04}},
  187. {Header: rtp.Header{SequenceNumber: 5004, Timestamp: 5}, Payload: []byte{0x05}},
  188. {Header: rtp.Header{SequenceNumber: 5005, Timestamp: 6}, Payload: []byte{0x06}},
  189. },
  190. samples: []*media.Sample{
  191. {Data: []byte{0x01}, Duration: time.Second, PacketTimestamp: 1},
  192. {Data: []byte{0x02}, Duration: time.Second, PacketTimestamp: 2},
  193. {Data: []byte{0x03}, Duration: time.Second, PacketTimestamp: 3},
  194. {Data: []byte{0x04}, Duration: time.Second, PacketTimestamp: 4},
  195. {Data: []byte{0x05}, Duration: time.Second, PacketTimestamp: 5},
  196. },
  197. maxLate: 50,
  198. maxLateTimestamp: 0,
  199. },
  200. {
  201. message: "SampleBuilder should skip time stamps too old",
  202. packets: []*rtp.Packet{
  203. {Header: rtp.Header{SequenceNumber: 5000, Timestamp: 1}, Payload: []byte{0x01}},
  204. {Header: rtp.Header{SequenceNumber: 5001, Timestamp: 2}, Payload: []byte{0x02}},
  205. {Header: rtp.Header{SequenceNumber: 5002, Timestamp: 3}, Payload: []byte{0x03}},
  206. {Header: rtp.Header{SequenceNumber: 5013, Timestamp: 4000}, Payload: []byte{0x04}},
  207. {Header: rtp.Header{SequenceNumber: 5014, Timestamp: 4000}, Payload: []byte{0x05}},
  208. {Header: rtp.Header{SequenceNumber: 5015, Timestamp: 4002}, Payload: []byte{0x06}},
  209. {Header: rtp.Header{SequenceNumber: 5016, Timestamp: 7000}, Payload: []byte{0x04}},
  210. {Header: rtp.Header{SequenceNumber: 5017, Timestamp: 7001}, Payload: []byte{0x05}},
  211. },
  212. samples: []*media.Sample{
  213. {Data: []byte{0x04, 0x05}, Duration: time.Second * time.Duration(2), PacketTimestamp: 4000, PrevDroppedPackets: 13},
  214. },
  215. withHeadChecker: true,
  216. headBytes: []byte{0x04},
  217. maxLate: 50,
  218. maxLateTimestamp: 2000,
  219. },
  220. }
  221. t.Run("Pop", func(t *testing.T) {
  222. assert := assert.New(t)
  223. for _, t := range testData {
  224. var opts []Option
  225. if t.maxLateTimestamp != 0 {
  226. opts = append(opts, WithMaxTimeDelay(
  227. time.Millisecond*time.Duration(int64(t.maxLateTimestamp)),
  228. ))
  229. }
  230. d := &fakeDepacketizer{
  231. headChecker: t.withHeadChecker,
  232. headBytes: t.headBytes,
  233. }
  234. s := New(t.maxLate, d, 1, opts...)
  235. samples := []*media.Sample{}
  236. for _, p := range t.packets {
  237. s.Push(p)
  238. }
  239. for sample := s.Pop(); sample != nil; sample = s.Pop() {
  240. samples = append(samples, sample)
  241. }
  242. assert.Equal(t.samples, samples, t.message)
  243. }
  244. })
  245. }
  246. // SampleBuilder should respect maxLate if we popped successfully but then have a gap larger then maxLate
  247. func TestSampleBuilderMaxLate(t *testing.T) {
  248. assert := assert.New(t)
  249. s := New(50, &fakeDepacketizer{}, 1)
  250. s.Push(&rtp.Packet{Header: rtp.Header{SequenceNumber: 0, Timestamp: 1}, Payload: []byte{0x01}})
  251. s.Push(&rtp.Packet{Header: rtp.Header{SequenceNumber: 1, Timestamp: 2}, Payload: []byte{0x01}})
  252. s.Push(&rtp.Packet{Header: rtp.Header{SequenceNumber: 2, Timestamp: 3}, Payload: []byte{0x01}})
  253. assert.Equal(&media.Sample{Data: []byte{0x01}, Duration: time.Second, PacketTimestamp: 1}, s.Pop(), "Failed to build samples before gap")
  254. s.Push(&rtp.Packet{Header: rtp.Header{SequenceNumber: 5000, Timestamp: 500}, Payload: []byte{0x02}})
  255. s.Push(&rtp.Packet{Header: rtp.Header{SequenceNumber: 5001, Timestamp: 501}, Payload: []byte{0x02}})
  256. s.Push(&rtp.Packet{Header: rtp.Header{SequenceNumber: 5002, Timestamp: 502}, Payload: []byte{0x02}})
  257. assert.Equal(&media.Sample{Data: []byte{0x01}, Duration: time.Second, PacketTimestamp: 2}, s.Pop(), "Failed to build samples after large gap")
  258. assert.Equal((*media.Sample)(nil), s.Pop(), "Failed to build samples after large gap")
  259. s.Push(&rtp.Packet{Header: rtp.Header{SequenceNumber: 6000, Timestamp: 600}, Payload: []byte{0x03}})
  260. assert.Equal(&media.Sample{Data: []byte{0x02}, Duration: time.Second, PacketTimestamp: 500, PrevDroppedPackets: 4998}, s.Pop(), "Failed to build samples after large gap")
  261. assert.Equal(&media.Sample{Data: []byte{0x02}, Duration: time.Second, PacketTimestamp: 501}, s.Pop(), "Failed to build samples after large gap")
  262. }
  263. func TestSeqnumDistance(t *testing.T) {
  264. testData := []struct {
  265. x uint16
  266. y uint16
  267. d uint16
  268. }{
  269. {0x0001, 0x0003, 0x0002},
  270. {0x0003, 0x0001, 0x0002},
  271. {0xFFF3, 0xFFF1, 0x0002},
  272. {0xFFF1, 0xFFF3, 0x0002},
  273. {0xFFFF, 0x0001, 0x0002},
  274. {0x0001, 0xFFFF, 0x0002},
  275. }
  276. for _, data := range testData {
  277. if ret := seqnumDistance(data.x, data.y); ret != data.d {
  278. t.Errorf("seqnumDistance(%d, %d) returned %d which must be %d",
  279. data.x, data.y, ret, data.d)
  280. }
  281. }
  282. }
  283. func TestSampleBuilderCleanReference(t *testing.T) {
  284. for _, seqStart := range []uint16{
  285. 0,
  286. 0xFFF8, // check upper boundary
  287. 0xFFFE, // check upper boundary
  288. } {
  289. seqStart := seqStart
  290. t.Run(fmt.Sprintf("From%d", seqStart), func(t *testing.T) {
  291. s := New(10, &fakeDepacketizer{}, 1)
  292. s.Push(&rtp.Packet{Header: rtp.Header{SequenceNumber: 0 + seqStart, Timestamp: 0}, Payload: []byte{0x01}})
  293. s.Push(&rtp.Packet{Header: rtp.Header{SequenceNumber: 1 + seqStart, Timestamp: 0}, Payload: []byte{0x02}})
  294. s.Push(&rtp.Packet{Header: rtp.Header{SequenceNumber: 2 + seqStart, Timestamp: 0}, Payload: []byte{0x03}})
  295. pkt4 := &rtp.Packet{Header: rtp.Header{SequenceNumber: 14 + seqStart, Timestamp: 120}, Payload: []byte{0x04}}
  296. s.Push(pkt4)
  297. pkt5 := &rtp.Packet{Header: rtp.Header{SequenceNumber: 12 + seqStart, Timestamp: 120}, Payload: []byte{0x05}}
  298. s.Push(pkt5)
  299. for i := 0; i < 3; i++ {
  300. if s.buffer[(i+int(seqStart))%0x10000] != nil {
  301. t.Errorf("Old packet (%d) is not unreferenced (maxLate: 10, pushed: 12)", i)
  302. }
  303. }
  304. if s.buffer[(14+int(seqStart))%0x10000] != pkt4 {
  305. t.Error("New packet must be referenced after jump")
  306. }
  307. if s.buffer[(12+int(seqStart))%0x10000] != pkt5 {
  308. t.Error("New packet must be referenced after jump")
  309. }
  310. })
  311. }
  312. }
  313. func TestSampleBuilderPushMaxZero(t *testing.T) {
  314. // Test packets released via 'maxLate' of zero.
  315. pkts := []rtp.Packet{
  316. {Header: rtp.Header{SequenceNumber: 0, Timestamp: 0, Marker: true}, Payload: []byte{0x01}},
  317. }
  318. d := &fakeDepacketizer{
  319. headChecker: true,
  320. headBytes: []byte{0x01},
  321. }
  322. s := New(0, d, 1)
  323. s.Push(&pkts[0])
  324. if sample := s.Pop(); sample == nil {
  325. t.Error("Should expect a popped sample")
  326. }
  327. }
  328. func TestSampleBuilderWithPacketReleaseHandler(t *testing.T) {
  329. var released []*rtp.Packet
  330. fakePacketReleaseHandler := func(p *rtp.Packet) {
  331. released = append(released, p)
  332. }
  333. // Test packets released via 'maxLate'.
  334. pkts := []rtp.Packet{
  335. {Header: rtp.Header{SequenceNumber: 0, Timestamp: 0}, Payload: []byte{0x01}},
  336. {Header: rtp.Header{SequenceNumber: 11, Timestamp: 120}, Payload: []byte{0x02}},
  337. {Header: rtp.Header{SequenceNumber: 12, Timestamp: 121}, Payload: []byte{0x03}},
  338. {Header: rtp.Header{SequenceNumber: 13, Timestamp: 122}, Payload: []byte{0x04}},
  339. {Header: rtp.Header{SequenceNumber: 21, Timestamp: 200}, Payload: []byte{0x05}},
  340. }
  341. s := New(10, &fakeDepacketizer{}, 1, WithPacketReleaseHandler(fakePacketReleaseHandler))
  342. s.Push(&pkts[0])
  343. s.Push(&pkts[1])
  344. if len(released) == 0 {
  345. t.Errorf("Old packet is not released")
  346. }
  347. if len(released) > 0 && released[0].SequenceNumber != pkts[0].SequenceNumber {
  348. t.Errorf("Unexpected packet released by maxLate")
  349. }
  350. // Test packets released after samples built.
  351. s.Push(&pkts[2])
  352. s.Push(&pkts[3])
  353. s.Push(&pkts[4])
  354. if s.Pop() == nil {
  355. t.Errorf("Should have some sample here.")
  356. }
  357. if len(released) < 3 {
  358. t.Errorf("packet built with sample is not released")
  359. }
  360. if len(released) >= 2 && released[2].SequenceNumber != pkts[2].SequenceNumber {
  361. t.Errorf("Unexpected packet released by samples built")
  362. }
  363. }
  364. func TestSampleBuilderWithPacketHeadHandler(t *testing.T) {
  365. packets := []*rtp.Packet{
  366. {Header: rtp.Header{SequenceNumber: 5000, Timestamp: 5}, Payload: []byte{0x01}},
  367. {Header: rtp.Header{SequenceNumber: 5001, Timestamp: 5}, Payload: []byte{0x02}},
  368. {Header: rtp.Header{SequenceNumber: 5002, Timestamp: 6}, Payload: []byte{0x01}},
  369. {Header: rtp.Header{SequenceNumber: 5003, Timestamp: 6}, Payload: []byte{0x02}},
  370. {Header: rtp.Header{SequenceNumber: 5004, Timestamp: 7}, Payload: []byte{0x01}},
  371. }
  372. headCount := 0
  373. s := New(10, &fakeDepacketizer{}, 1, WithPacketHeadHandler(func(headPacket interface{}) interface{} {
  374. headCount++
  375. return true
  376. }))
  377. for _, pkt := range packets {
  378. s.Push(pkt)
  379. }
  380. for {
  381. sample := s.Pop()
  382. if sample == nil {
  383. break
  384. }
  385. assert.NotNil(t, sample.Metadata, "sample metadata shouldn't be nil")
  386. assert.Equal(t, true, sample.Metadata, "sample metadata should've been set to true")
  387. }
  388. assert.Equal(t, 2, headCount, "two sample heads should have been inspected")
  389. }
  390. func TestPopWithTimestamp(t *testing.T) {
  391. t.Run("Crash on nil", func(t *testing.T) {
  392. s := New(0, &fakeDepacketizer{}, 1)
  393. sample, timestamp := s.PopWithTimestamp()
  394. assert.Nil(t, sample)
  395. assert.Equal(t, uint32(0), timestamp)
  396. })
  397. }
  398. type truePartitionHeadChecker struct{}
  399. func (f *truePartitionHeadChecker) IsPartitionHead([]byte) bool {
  400. return true
  401. }
  402. func TestSampleBuilderData(t *testing.T) {
  403. s := New(10, &fakeDepacketizer{}, 1,
  404. WithPartitionHeadChecker(&truePartitionHeadChecker{}),
  405. )
  406. j := 0
  407. for i := 0; i < 0x20000; i++ {
  408. p := rtp.Packet{
  409. Header: rtp.Header{
  410. SequenceNumber: uint16(i),
  411. Timestamp: uint32(i + 42),
  412. },
  413. Payload: []byte{byte(i)},
  414. }
  415. s.Push(&p)
  416. for {
  417. sample, ts := s.PopWithTimestamp()
  418. if sample == nil {
  419. break
  420. }
  421. assert.Equal(t, ts, uint32(j+42), "timestamp")
  422. assert.Equal(t, len(sample.Data), 1, "data length")
  423. assert.Equal(t, byte(j), sample.Data[0], "data")
  424. j++
  425. }
  426. }
  427. // only the last packet should be dropped
  428. assert.Equal(t, j, 0x1FFFF)
  429. }
  430. func BenchmarkSampleBuilderSequential(b *testing.B) {
  431. s := New(100, &fakeDepacketizer{}, 1)
  432. b.ResetTimer()
  433. j := 0
  434. for i := 0; i < b.N; i++ {
  435. p := rtp.Packet{
  436. Header: rtp.Header{
  437. SequenceNumber: uint16(i),
  438. Timestamp: uint32(i + 42),
  439. },
  440. Payload: make([]byte, 50),
  441. }
  442. s.Push(&p)
  443. for {
  444. s := s.Pop()
  445. if s == nil {
  446. break
  447. }
  448. j++
  449. }
  450. }
  451. if b.N > 200 && j < b.N-100 {
  452. b.Errorf("Got %v (N=%v)", j, b.N)
  453. }
  454. }
  455. func BenchmarkSampleBuilderLoss(b *testing.B) {
  456. s := New(100, &fakeDepacketizer{}, 1)
  457. b.ResetTimer()
  458. j := 0
  459. for i := 0; i < b.N; i++ {
  460. if i%13 == 0 {
  461. continue
  462. }
  463. p := rtp.Packet{
  464. Header: rtp.Header{
  465. SequenceNumber: uint16(i),
  466. Timestamp: uint32(i + 42),
  467. },
  468. Payload: make([]byte, 50),
  469. }
  470. s.Push(&p)
  471. for {
  472. s := s.Pop()
  473. if s == nil {
  474. break
  475. }
  476. j++
  477. }
  478. }
  479. if b.N > 200 && j < b.N/2-100 {
  480. b.Errorf("Got %v (N=%v)", j, b.N)
  481. }
  482. }
  483. func BenchmarkSampleBuilderReordered(b *testing.B) {
  484. s := New(100, &fakeDepacketizer{}, 1)
  485. b.ResetTimer()
  486. j := 0
  487. for i := 0; i < b.N; i++ {
  488. p := rtp.Packet{
  489. Header: rtp.Header{
  490. SequenceNumber: uint16(i ^ 3),
  491. Timestamp: uint32((i ^ 3) + 42),
  492. },
  493. Payload: make([]byte, 50),
  494. }
  495. s.Push(&p)
  496. for {
  497. s := s.Pop()
  498. if s == nil {
  499. break
  500. }
  501. j++
  502. }
  503. }
  504. if b.N > 2 && j < b.N-5 && j > b.N {
  505. b.Errorf("Got %v (N=%v)", j, b.N)
  506. }
  507. }
  508. func BenchmarkSampleBuilderFragmented(b *testing.B) {
  509. s := New(100, &fakeDepacketizer{}, 1)
  510. b.ResetTimer()
  511. j := 0
  512. for i := 0; i < b.N; i++ {
  513. p := rtp.Packet{
  514. Header: rtp.Header{
  515. SequenceNumber: uint16(i),
  516. Timestamp: uint32(i/2 + 42),
  517. },
  518. Payload: make([]byte, 50),
  519. }
  520. s.Push(&p)
  521. for {
  522. s := s.Pop()
  523. if s == nil {
  524. break
  525. }
  526. j++
  527. }
  528. }
  529. if b.N > 200 && j < b.N/2-100 {
  530. b.Errorf("Got %v (N=%v)", j, b.N)
  531. }
  532. }
  533. func BenchmarkSampleBuilderFragmentedLoss(b *testing.B) {
  534. s := New(100, &fakeDepacketizer{}, 1)
  535. b.ResetTimer()
  536. j := 0
  537. for i := 0; i < b.N; i++ {
  538. if i%13 == 0 {
  539. continue
  540. }
  541. p := rtp.Packet{
  542. Header: rtp.Header{
  543. SequenceNumber: uint16(i),
  544. Timestamp: uint32(i/2 + 42),
  545. },
  546. Payload: make([]byte, 50),
  547. }
  548. s.Push(&p)
  549. for {
  550. s := s.Pop()
  551. if s == nil {
  552. break
  553. }
  554. j++
  555. }
  556. }
  557. if b.N > 200 && j < b.N/3-100 {
  558. b.Errorf("Got %v (N=%v)", j, b.N)
  559. }
  560. }