aboutsummaryrefslogtreecommitdiff
path: root/Src/replicant/nsid3v2/frame_id.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'Src/replicant/nsid3v2/frame_id.cpp')
-rw-r--r--Src/replicant/nsid3v2/frame_id.cpp167
1 files changed, 167 insertions, 0 deletions
diff --git a/Src/replicant/nsid3v2/frame_id.cpp b/Src/replicant/nsid3v2/frame_id.cpp
new file mode 100644
index 00000000..807d8cd6
--- /dev/null
+++ b/Src/replicant/nsid3v2/frame_id.cpp
@@ -0,0 +1,167 @@
+#include "nsid3v2.h"
+#include "nsid3v2/header.h"
+#include "nsid3v2/tag.h"
+#include "nsid3v2/frame_utils.h"
+#include "nu/ByteReader.h"
+#include "nu/ByteWriter.h"
+#include "nx/nxstring.h"
+
+struct ParsedID
+{
+ ParsedString owner;
+ const void *identifier_data;
+ size_t identifier_byte_length;
+};
+
+static int ParseID(const void *data, size_t data_len, ParsedID &parsed)
+{
+ int ret;
+ if (data_len < 1)
+ return NErr_Insufficient;
+
+ bytereader_value_t byte_reader;
+ bytereader_init(&byte_reader, data, data_len);
+
+ /* owner is always latin-1 */
+ ret = ParseNullTerminatedString(&byte_reader, 0, parsed.owner);
+ if (ret != NErr_Success)
+ return ret;
+ parsed.identifier_data = bytereader_pointer(&byte_reader);
+ parsed.identifier_byte_length = bytereader_size(&byte_reader);
+ return NErr_Success;
+}
+
+int NSID3v2_Tag_ID_Find(const nsid3v2_tag_t t, const char *owner, nsid3v2_frame_t *out_frame, int text_flags)
+{
+ const ID3v2::Tag *tag = (const ID3v2::Tag *)t;
+ if (!tag)
+ return NErr_Empty;
+ const ID3v2::Frame *frame = tag->FindFirstFrame(NSID3V2_FRAME_ID);
+ while (frame)
+ {
+ const void *data;
+ size_t data_len;
+ ParsedID parsed;
+ if (frame->GetData(&data, &data_len) == NErr_Success && data_len > 0 && ParseID(data, data_len, parsed) == NErr_Success && (!owner || DescriptionMatches(parsed.owner, owner, text_flags)))
+ {
+ *out_frame = (nsid3v2_frame_t)frame;
+ return NErr_Success;
+ }
+ frame = tag->FindNextFrame(frame);
+ }
+
+ return NErr_Empty;
+}
+
+int NSID3v2_Frame_ID_Get(nsid3v2_frame_t f, nx_string_t *owner, const void **id_data, size_t *length, int text_flags)
+{
+ const ID3v2::Frame *frame = (const ID3v2::Frame *)f;
+ if (frame)
+ {
+ const void *data;
+ size_t data_len;
+ ParsedID parsed;
+ if (frame->GetData(&data, &data_len) == NErr_Success && data_len > 0 && ParseID(data, data_len, parsed) == NErr_Success)
+ {
+ if (owner)
+ {
+ int ret = NXStringCreateFromParsedString(owner, parsed.owner, text_flags);
+ if (ret != NErr_Success)
+ return ret;
+ }
+
+ *id_data = parsed.identifier_data;
+ *length = parsed.identifier_byte_length;
+
+ return NErr_Success;
+ }
+
+ }
+ return NErr_Empty;
+}
+
+int NSID3v2_Tag_ID_Get(const nsid3v2_tag_t t, const char *owner, const void **id_data, size_t *length, int text_flags)
+{
+ const ID3v2::Tag *tag = (const ID3v2::Tag *)t;
+ if (!tag)
+ return NErr_Empty;
+
+ ID3v2::Frame *frame = tag->FindFirstFrame(NSID3V2_FRAME_ID);
+ while (frame)
+ {
+ const void *data;
+ size_t data_len;
+ ParsedID parsed;
+ if (frame->GetData(&data, &data_len) == NErr_Success && data_len > 0 && ParseID(data, data_len, parsed) == NErr_Success && (!owner || DescriptionMatches(parsed.owner, owner, text_flags)))
+ {
+ *id_data = parsed.identifier_data;
+ *length = parsed.identifier_byte_length;
+
+ return NErr_Success;
+ }
+
+ frame = tag->FindNextFrame(frame);
+ }
+ return NErr_Empty;
+}
+
+
+
+/* ---------------- Setters ---------------- */
+int NSID3v2_Frame_ID_Set(nsid3v2_frame_t f, const char *owner, const void *id_data, size_t length, int text_flags)
+{
+ ID3v2::Frame *frame = (ID3v2::Frame *)f;
+ if (frame)
+ {
+ size_t owner_length=owner?strlen(owner):0;
+
+ /* TODO: overflow check */
+ size_t total_size = owner_length + 1 + length;
+
+ void *data;
+ size_t data_len;
+ int ret = frame->NewData(total_size, &data, &data_len);
+ if (ret != NErr_Success)
+ return ret;
+
+ bytewriter_s byte_writer;
+ bytewriter_init(&byte_writer, data, data_len);
+ bytewriter_write_n(&byte_writer, owner, owner_length);
+ bytewriter_write_u8(&byte_writer, 0); // write null terminator separately, in case owner is NULL
+ bytewriter_write_n(&byte_writer, id_data, length);
+
+ return NErr_Success;
+ }
+ return NErr_Empty;
+}
+
+int NSID3v2_Tag_ID_Set(nsid3v2_tag_t t, const char *owner, const void *id_data, size_t length, int text_flags)
+{
+ ID3v2::Tag *tag = (ID3v2::Tag *)t;
+ if (!tag)
+ return NErr_Empty;
+
+ ID3v2::Frame *frame = tag->FindFirstFrame(NSID3V2_FRAME_ID);
+ while (frame)
+ {
+ const void *data;
+ size_t data_len;
+ ParsedID parsed;
+ if (frame->GetData(&data, &data_len) == NErr_Success && data_len > 0 && ParseID(data, data_len, parsed) == NErr_Success && (!owner || DescriptionMatches(parsed.owner, owner, text_flags)))
+ {
+ break;
+ }
+
+ frame = tag->FindNextFrame(frame);
+ }
+
+ if (!frame)
+ {
+ frame = tag->NewFrame(NSID3V2_FRAME_ID, 0);
+ if (!frame)
+ return NErr_OutOfMemory;
+ tag->AddFrame(frame);
+ }
+
+ return NSID3v2_Frame_ID_Set((nsid3v2_frame_t)frame, owner, id_data, length, text_flags);
+}