ApeTag.cxx 3.06 KB
Newer Older
1
/*
Max Kellermann's avatar
Max Kellermann committed
2
 * Copyright (C) 2003-2014 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"
Max Kellermann's avatar
Max Kellermann committed
23
#include "Tag.hxx"
24
#include "TagTable.hxx"
25
#include "TagHandler.hxx"
26
#include "fs/Path.hxx"
27

28
#include <string>
29 30 31

#include <string.h>

32
const struct tag_table ape_tags[] = {
33 34
	{ "album artist", TAG_ALBUM_ARTIST },
	{ "year", TAG_DATE },
Max Kellermann's avatar
Max Kellermann committed
35
	{ nullptr, TAG_NUM_OF_ITEM_TYPES }
36 37
};

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

	return type;
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 73
/**
 * 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());
	}
}

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

86 87 88 89 90 91 92
	const char *const end = value + value_length;

	if (handler->pair != nullptr)
		ForEachValue(value, end, [handler, handler_ctx,
					  key](const char *_value) {
				handler->pair(key, _value, handler_ctx);
			});
93

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

98 99
	ForEachValue(value, end, [handler, handler_ctx,
				    type](const char *_value) {
100
			tag_handler_invoke_tag(handler, handler_ctx,
101 102
					       type, _value);
		});
103

104
	return true;
105 106
}

107
bool
108
tag_ape_scan2(Path path_fs,
109 110
	      const struct tag_handler *handler, void *handler_ctx)
{
Max Kellermann's avatar
Max Kellermann committed
111 112 113 114 115 116 117 118 119 120
	bool recognized = false;

	auto callback = [handler, handler_ctx, &recognized]
		(unsigned long flags, const char *key,
		 const char *value,
		 size_t value_length) {
		recognized |= tag_ape_import_item(flags, key, value,
						  value_length,
						  handler, handler_ctx);
		return true;
121 122
	};

Max Kellermann's avatar
Max Kellermann committed
123
	return tag_ape_scan(path_fs, callback) && recognized;
124
}