audiolevelextension.go 1.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263
  1. // SPDX-FileCopyrightText: 2023 The Pion community <https://pion.ly>
  2. // SPDX-License-Identifier: MIT
  3. package rtp
  4. import (
  5. "errors"
  6. )
  7. const (
  8. // audioLevelExtensionSize One byte header size
  9. audioLevelExtensionSize = 1
  10. )
  11. var errAudioLevelOverflow = errors.New("audio level overflow")
  12. // AudioLevelExtension is a extension payload format described in
  13. // https://tools.ietf.org/html/rfc6464
  14. //
  15. // Implementation based on:
  16. // https://chromium.googlesource.com/external/webrtc/+/e2a017725570ead5946a4ca8235af27470ca0df9/webrtc/modules/rtp_rtcp/source/rtp_header_extensions.cc#49
  17. //
  18. // One byte format:
  19. // 0 1
  20. // 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5
  21. // +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
  22. // | ID | len=0 |V| level |
  23. // +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
  24. //
  25. // Two byte format:
  26. // 0 1 2 3
  27. // 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1
  28. // +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
  29. // | ID | len=1 |V| level | 0 (pad) |
  30. // +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
  31. type AudioLevelExtension struct {
  32. Level uint8
  33. Voice bool
  34. }
  35. // Marshal serializes the members to buffer
  36. func (a AudioLevelExtension) Marshal() ([]byte, error) {
  37. if a.Level > 127 {
  38. return nil, errAudioLevelOverflow
  39. }
  40. voice := uint8(0x00)
  41. if a.Voice {
  42. voice = 0x80
  43. }
  44. buf := make([]byte, audioLevelExtensionSize)
  45. buf[0] = voice | a.Level
  46. return buf, nil
  47. }
  48. // Unmarshal parses the passed byte slice and stores the result in the members
  49. func (a *AudioLevelExtension) Unmarshal(rawData []byte) error {
  50. if len(rawData) < audioLevelExtensionSize {
  51. return errTooSmall
  52. }
  53. a.Level = rawData[0] & 0x7F
  54. a.Voice = rawData[0]&0x80 != 0
  55. return nil
  56. }