HlsCodecStringHelpers.cs 3.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125
  1. using System;
  2. using System.Globalization;
  3. using System.Text;
  4. namespace Jellyfin.Api.Helpers
  5. {
  6. /// <summary>
  7. /// Hls Codec string helpers.
  8. /// </summary>
  9. public static class HlsCodecStringHelpers
  10. {
  11. /// <summary>
  12. /// Gets a MP3 codec string.
  13. /// </summary>
  14. /// <returns>MP3 codec string.</returns>
  15. public static string GetMP3String()
  16. {
  17. return "mp4a.40.34";
  18. }
  19. /// <summary>
  20. /// Gets an AAC codec string.
  21. /// </summary>
  22. /// <param name="profile">AAC profile.</param>
  23. /// <returns>AAC codec string.</returns>
  24. public static string GetAACString(string? profile)
  25. {
  26. StringBuilder result = new StringBuilder("mp4a", 9);
  27. if (string.Equals(profile, "HE", StringComparison.OrdinalIgnoreCase))
  28. {
  29. result.Append(".40.5");
  30. }
  31. else
  32. {
  33. // Default to LC if profile is invalid
  34. result.Append(".40.2");
  35. }
  36. return result.ToString();
  37. }
  38. /// <summary>
  39. /// Gets a H.264 codec string.
  40. /// </summary>
  41. /// <param name="profile">H.264 profile.</param>
  42. /// <param name="level">H.264 level.</param>
  43. /// <returns>H.264 string.</returns>
  44. public static string GetH264String(string? profile, int level)
  45. {
  46. StringBuilder result = new StringBuilder("avc1", 11);
  47. if (string.Equals(profile, "high", StringComparison.OrdinalIgnoreCase))
  48. {
  49. result.Append(".6400");
  50. }
  51. else if (string.Equals(profile, "main", StringComparison.OrdinalIgnoreCase))
  52. {
  53. result.Append(".4D40");
  54. }
  55. else if (string.Equals(profile, "baseline", StringComparison.OrdinalIgnoreCase))
  56. {
  57. result.Append(".42E0");
  58. }
  59. else
  60. {
  61. // Default to constrained baseline if profile is invalid
  62. result.Append(".4240");
  63. }
  64. string levelHex = level.ToString("X2", CultureInfo.InvariantCulture);
  65. result.Append(levelHex);
  66. return result.ToString();
  67. }
  68. /// <summary>
  69. /// Gets a H.265 codec string.
  70. /// </summary>
  71. /// <param name="profile">H.265 profile.</param>
  72. /// <param name="level">H.265 level.</param>
  73. /// <returns>H.265 string.</returns>
  74. public static string GetH265String(string? profile, int level)
  75. {
  76. // The h265 syntax is a bit of a mystery at the time this comment was written.
  77. // This is what I've found through various sources:
  78. // FORMAT: [codecTag].[profile].[constraint?].L[level * 30].[UNKNOWN]
  79. StringBuilder result = new StringBuilder("hev1", 16);
  80. if (string.Equals(profile, "main10", StringComparison.OrdinalIgnoreCase))
  81. {
  82. result.Append(".2.6");
  83. }
  84. else
  85. {
  86. // Default to main if profile is invalid
  87. result.Append(".1.6");
  88. }
  89. result.Append(".L")
  90. .Append(level * 3)
  91. .Append(".B0");
  92. return result.ToString();
  93. }
  94. /// <summary>
  95. /// Gets an AC-3 codec string.
  96. /// </summary>
  97. /// <returns>AC-3 codec string.</returns>
  98. public static string GetAC3String()
  99. {
  100. return "mp4a.a5";
  101. }
  102. /// <summary>
  103. /// Gets an E-AC-3 codec string.
  104. /// </summary>
  105. /// <returns>E-AC-3 codec string.</returns>
  106. public static string GetEAC3String()
  107. {
  108. return "mp4a.a6";
  109. }
  110. }
  111. }