ApeTag.cxx 2.8 KB
Newer Older
1
/*
Max Kellermann's avatar
Max Kellermann committed
2
 * Copyright 2003-2017 The Music Player Daemon Project
3 4 5 6 7 8 9 10 11 12 13
 * http://www.musicpd.org
 *
 * This program is free software; you can redistribute it and/or modify
 * it under the terms of the GNU General Public License as published by
 * the Free Software Foundation; either version 2 of the License, or
 * (at your option) any later version.
 *
 * This program is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 * GNU General Public License for more details.
14 15 16 17
 *
 * You should have received a copy of the GNU General Public License along
 * with this program; if not, write to the Free Software Foundation, Inc.,
 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
18 19
 */

20
#include "config.h"
Max Kellermann's avatar
Max Kellermann committed
21 22
#include "ApeTag.hxx"
#include "ApeLoader.hxx"
23
#include "ParseName.hxx"
24 25
#include "Table.hxx"
#include "Handler.hxx"
26
#include "util/StringView.hxx"
27

28
#include <string>
29 30 31

#include <string.h>

32
static constexpr struct tag_table ape_tags[] = {
33
	{ "year", TAG_DATE },
Max Kellermann's avatar
Max Kellermann committed
34
	{ nullptr, TAG_NUM_OF_ITEM_TYPES }
35 36
};

37
static TagType
38 39
tag_ape_name_parse(const char *name)
{
40
	TagType type = tag_table_lookup_i(ape_tags, name);
41 42 43 44
	if (type == TAG_NUM_OF_ITEM_TYPES)
		type = tag_name_parse_i(name);

	return type;
45 46
}

47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72
/**
 * Invoke the given callback for each string inside the given range.
 * The strings are separated by null bytes, but the last one may not
 * be terminated.
 */
template<typename C>
static void
ForEachValue(const char *value, const char *end, C &&callback)
{
	while (true) {
		const char *n = (const char *)memchr(value, 0, end - value);
		if (n == nullptr)
			break;

		if (n > value)
			callback(value);

		value = n + 1;
	}

	if (value < end) {
		const std::string value2(value, end);
		callback(value2.c_str());
	}
}

73 74 75 76
/**
 * @return true if the item was recognized
 */
static bool
77
tag_ape_import_item(unsigned long flags,
78
		    const char *key, StringView value,
79
		    TagHandler &handler) noexcept
80 81 82
{
	/* we only care about utf-8 text tags */
	if ((flags & (0x3 << 1)) != 0)
83
		return false;
84

85 86
	const auto begin = value.begin();
	const auto end = value.end();
87

88 89 90
	if (handler.WantPair())
		ForEachValue(begin, end, [&handler, key](const char *_value) {
				handler.OnPair(key, _value);
91
			});
92

93
	TagType type = tag_ape_name_parse(key);
94
	if (type == TAG_NUM_OF_ITEM_TYPES)
95
		return false;
96

97 98
	ForEachValue(begin, end, [&handler, type](const char *_value) {
			handler.OnTag(type, _value);
99
		});
100

101
	return true;
102 103
}

104
bool
105
tag_ape_scan2(InputStream &is, TagHandler &handler) noexcept
106 107 108
{
	bool recognized = false;

109
	auto callback = [&handler, &recognized]
110 111
		(unsigned long flags, const char *key,
		 StringView value) {
112
		recognized |= tag_ape_import_item(flags, key, value, handler);
113 114 115 116 117
		return true;
	};

	return tag_ape_scan(is, callback) && recognized;
}