VorbisComment.cpp 8.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189
  1. /*
  2. * Copyright (c) 2023, kleines Filmröllchen <filmroellchen@serenityos.org>
  3. *
  4. * SPDX-License-Identifier: BSD-2-Clause
  5. */
  6. #include "VorbisComment.h"
  7. #include <AK/Endian.h>
  8. #include <AK/MemoryStream.h>
  9. #include <AK/Span.h>
  10. #include <AK/String.h>
  11. namespace Audio {
  12. static StringView vorbis_field_for_role(Person::Role role)
  13. {
  14. static HashMap<Person::Role, StringView> role_map {
  15. { Person::Role::Artist, "ARTIST"sv },
  16. { Person::Role::Performer, "PERFORMER"sv },
  17. { Person::Role::Lyricist, "LYRICIST"sv },
  18. { Person::Role::Conductor, "CONDUCTOR"sv },
  19. { Person::Role::Publisher, "PUBLISHER"sv },
  20. { Person::Role::Engineer, "ENCODED-BY"sv },
  21. { Person::Role::Composer, "COMPOSER"sv },
  22. };
  23. return role_map.get(role).value();
  24. }
  25. // "Content vector format"
  26. static ErrorOr<void> read_vorbis_field(Metadata& metadata_to_write_into, String const& unparsed_user_comment)
  27. {
  28. // Technically the field name has to be ASCII, but we just accept all UTF-8.
  29. auto field_name_and_contents = TRY(unparsed_user_comment.split_limit('=', 2));
  30. if (field_name_and_contents.size() != 2)
  31. return Error::from_string_view("User comment does not contain '='"sv);
  32. auto contents = field_name_and_contents.take_last();
  33. auto field_name = TRY(field_name_and_contents.take_first().to_uppercase());
  34. // Some of these are taken from https://age.hobba.nl/audio/tag_frame_reference.html
  35. if (field_name == "TITLE"sv) {
  36. if (metadata_to_write_into.title.has_value())
  37. TRY(metadata_to_write_into.add_miscellaneous(field_name, contents));
  38. else
  39. metadata_to_write_into.title = contents;
  40. } else if (field_name == "VERSION"sv) {
  41. if (metadata_to_write_into.subtitle.has_value())
  42. TRY(metadata_to_write_into.add_miscellaneous(field_name, contents));
  43. else
  44. metadata_to_write_into.subtitle = contents;
  45. } else if (field_name == "ALBUM"sv) {
  46. if (metadata_to_write_into.album.has_value())
  47. TRY(metadata_to_write_into.add_miscellaneous(field_name, contents));
  48. else
  49. metadata_to_write_into.album = contents;
  50. } else if (field_name == "COPYRIGHT"sv) {
  51. if (metadata_to_write_into.copyright.has_value())
  52. TRY(metadata_to_write_into.add_miscellaneous(field_name, contents));
  53. else
  54. metadata_to_write_into.copyright = contents;
  55. } else if (field_name == "ISRC"sv) {
  56. if (metadata_to_write_into.isrc.has_value())
  57. TRY(metadata_to_write_into.add_miscellaneous(field_name, contents));
  58. else
  59. metadata_to_write_into.isrc = contents;
  60. } else if (field_name == "GENRE"sv) {
  61. if (metadata_to_write_into.genre.has_value())
  62. TRY(metadata_to_write_into.add_miscellaneous(field_name, contents));
  63. else
  64. metadata_to_write_into.genre = contents;
  65. } else if (field_name == "COMMENT"sv) {
  66. if (metadata_to_write_into.comment.has_value())
  67. TRY(metadata_to_write_into.add_miscellaneous(field_name, contents));
  68. else
  69. metadata_to_write_into.comment = contents;
  70. } else if (field_name == "TRACKNUMBER"sv) {
  71. if (metadata_to_write_into.track_number.has_value())
  72. TRY(metadata_to_write_into.add_miscellaneous(field_name, contents));
  73. else if (auto maybe_number = contents.to_number<unsigned>(); maybe_number.has_value())
  74. metadata_to_write_into.track_number = maybe_number.release_value();
  75. else
  76. TRY(metadata_to_write_into.add_miscellaneous(field_name, contents));
  77. } else if (field_name == "DATE"sv) {
  78. if (metadata_to_write_into.unparsed_time.has_value())
  79. TRY(metadata_to_write_into.add_miscellaneous(field_name, contents));
  80. else
  81. metadata_to_write_into.unparsed_time = contents;
  82. } else if (field_name == vorbis_field_for_role(Person::Role::Performer)) {
  83. TRY(metadata_to_write_into.add_person(Person::Role::Performer, contents));
  84. } else if (field_name == vorbis_field_for_role(Person::Role::Artist)) {
  85. TRY(metadata_to_write_into.add_person(Person::Role::Artist, contents));
  86. } else if (field_name == vorbis_field_for_role(Person::Role::Composer)) {
  87. TRY(metadata_to_write_into.add_person(Person::Role::Composer, contents));
  88. } else if (field_name == vorbis_field_for_role(Person::Role::Conductor)) {
  89. TRY(metadata_to_write_into.add_person(Person::Role::Conductor, contents));
  90. } else if (field_name == vorbis_field_for_role(Person::Role::Lyricist)) {
  91. TRY(metadata_to_write_into.add_person(Person::Role::Lyricist, contents));
  92. } else if (field_name == "ORGANIZATION"sv) {
  93. TRY(metadata_to_write_into.add_person(Person::Role::Publisher, contents));
  94. } else if (field_name == vorbis_field_for_role(Person::Role::Publisher)) {
  95. TRY(metadata_to_write_into.add_person(Person::Role::Publisher, contents));
  96. } else if (field_name == vorbis_field_for_role(Person::Role::Engineer)) {
  97. TRY(metadata_to_write_into.add_person(Person::Role::Engineer, contents));
  98. } else {
  99. TRY(metadata_to_write_into.add_miscellaneous(field_name, contents));
  100. }
  101. return {};
  102. }
  103. ErrorOr<Metadata, LoaderError> load_vorbis_comment(ByteBuffer const& vorbis_comment)
  104. {
  105. FixedMemoryStream stream { vorbis_comment };
  106. auto vendor_length = TRY(stream.read_value<LittleEndian<u32>>());
  107. Vector<u8> raw_vendor_string;
  108. TRY(raw_vendor_string.try_resize(vendor_length));
  109. TRY(stream.read_until_filled(raw_vendor_string));
  110. auto vendor_string = TRY(String::from_utf8(StringView { raw_vendor_string.span() }));
  111. Metadata metadata;
  112. metadata.encoder = move(vendor_string);
  113. auto user_comment_count = TRY(stream.read_value<LittleEndian<u32>>());
  114. for (size_t i = 0; i < user_comment_count; ++i) {
  115. auto user_comment_length = TRY(stream.read_value<LittleEndian<u32>>());
  116. Vector<u8> raw_user_comment;
  117. TRY(raw_user_comment.try_resize(user_comment_length));
  118. TRY(stream.read_until_filled(raw_user_comment));
  119. auto unparsed_user_comment = TRY(String::from_utf8(StringView { raw_user_comment.span() }));
  120. TRY(read_vorbis_field(metadata, unparsed_user_comment));
  121. }
  122. return metadata;
  123. }
  124. struct VorbisCommentPair {
  125. String field_name;
  126. String contents;
  127. };
  128. static ErrorOr<Vector<VorbisCommentPair>> make_vorbis_user_comments(Metadata const& metadata)
  129. {
  130. Vector<VorbisCommentPair> user_comments;
  131. auto add_if_present = [&](auto field_name, auto const& value) -> ErrorOr<void> {
  132. if (value.has_value())
  133. TRY(user_comments.try_append(VorbisCommentPair { field_name, TRY(String::formatted("{}", value.value())) }));
  134. return {};
  135. };
  136. TRY(add_if_present("TITLE"_string, metadata.title));
  137. TRY(add_if_present("VERSION"_string, metadata.subtitle));
  138. TRY(add_if_present("ALBUM"_string, metadata.album));
  139. TRY(add_if_present("COPYRIGHT"_string, metadata.copyright));
  140. TRY(add_if_present("ISRC"_string, metadata.isrc));
  141. TRY(add_if_present("GENRE"_string, metadata.genre));
  142. TRY(add_if_present("COMMENT"_string, metadata.comment));
  143. TRY(add_if_present("TRACKNUMBER"_string, metadata.track_number));
  144. TRY(add_if_present("DATE"_string, metadata.unparsed_time));
  145. for (auto const& person : metadata.people)
  146. TRY(user_comments.try_append(VorbisCommentPair { TRY(String::from_utf8(vorbis_field_for_role(person.role))), person.name }));
  147. for (auto const& field : metadata.miscellaneous) {
  148. for (auto const& value : field.value)
  149. TRY(user_comments.try_append(VorbisCommentPair { field.key, value }));
  150. }
  151. return user_comments;
  152. }
  153. ErrorOr<void> write_vorbis_comment(Metadata const& metadata, Stream& target)
  154. {
  155. auto encoder = metadata.encoder.value_or({}).bytes();
  156. TRY(target.write_value<LittleEndian<u32>>(encoder.size()));
  157. TRY(target.write_until_depleted(encoder));
  158. auto vorbis_user_comments = TRY(make_vorbis_user_comments(metadata));
  159. TRY(target.write_value<LittleEndian<u32>>(vorbis_user_comments.size()));
  160. for (auto const& field : vorbis_user_comments) {
  161. auto const serialized_field = TRY(String::formatted("{}={}", field.field_name, field.contents));
  162. TRY(target.write_value<LittleEndian<u32>>(serialized_field.bytes().size()));
  163. TRY(target.write_until_depleted(serialized_field.bytes()));
  164. }
  165. return {};
  166. }
  167. }