Blame src/plugin_common/tags.c

Packit 8f7830
/* plugin_common - Routines common to several plugins
Packit 8f7830
 * Copyright (C) 2002-2009  Josh Coalson
Packit 8f7830
 * Copyright (C) 2011-2016  Xiph.Org Foundation
Packit 8f7830
 *
Packit 8f7830
 * This library is free software; you can redistribute it and/or
Packit 8f7830
 * modify it under the terms of the GNU Lesser General Public
Packit 8f7830
 * License as published by the Free Software Foundation; either
Packit 8f7830
 * version 2.1 of the License, or (at your option) any later version.
Packit 8f7830
 *
Packit 8f7830
 * This library is distributed in the hope that it will be useful,
Packit 8f7830
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit 8f7830
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
Packit 8f7830
 * Lesser General Public License for more details.
Packit 8f7830
 *
Packit 8f7830
 * You should have received a copy of the GNU Lesser General Public
Packit 8f7830
 * License along with this library; if not, write to the Free Software
Packit 8f7830
 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA
Packit 8f7830
 */
Packit 8f7830
Packit 8f7830
#ifdef HAVE_CONFIG_H
Packit 8f7830
#  include <config.h>
Packit 8f7830
#endif
Packit 8f7830
Packit 8f7830
#include <stdio.h>
Packit 8f7830
#include <string.h>
Packit 8f7830
#include <stdlib.h>
Packit 8f7830
Packit 8f7830
#include "tags.h"
Packit 8f7830
#include "FLAC/assert.h"
Packit 8f7830
#include "FLAC/metadata.h"
Packit 8f7830
#include "share/alloc.h"
Packit 8f7830
Packit 8f7830
#ifndef FLaC__INLINE
Packit 8f7830
#define FLaC__INLINE
Packit 8f7830
#endif
Packit 8f7830
Packit 8f7830
Packit 8f7830
static FLaC__INLINE size_t local__wide_strlen(const FLAC__uint16 *s)
Packit 8f7830
{
Packit 8f7830
	size_t n = 0;
Packit 8f7830
	while(*s++)
Packit 8f7830
		n++;
Packit 8f7830
	return n;
Packit 8f7830
}
Packit 8f7830
Packit 8f7830
/*
Packit 8f7830
 * also disallows non-shortest-form encodings, c.f.
Packit 8f7830
 *   http://www.unicode.org/versions/corrigendum1.html
Packit 8f7830
 * and a more clear explanation at the end of this section:
Packit 8f7830
 *   http://www.cl.cam.ac.uk/~mgk25/unicode.html#utf-8
Packit 8f7830
 */
Packit 8f7830
static size_t local__utf8len(const FLAC__byte *utf8)
Packit 8f7830
{
Packit 8f7830
	FLAC__ASSERT(0 != utf8);
Packit 8f7830
	if ((utf8[0] & 0x80) == 0) {
Packit 8f7830
		return 1;
Packit 8f7830
	}
Packit 8f7830
	else if ((utf8[0] & 0xE0) == 0xC0 && (utf8[1] & 0xC0) == 0x80) {
Packit 8f7830
		if ((utf8[0] & 0xFE) == 0xC0) /* overlong sequence check */
Packit 8f7830
			return 0;
Packit 8f7830
		return 2;
Packit 8f7830
	}
Packit 8f7830
	else if ((utf8[0] & 0xF0) == 0xE0 && (utf8[1] & 0xC0) == 0x80 && (utf8[2] & 0xC0) == 0x80) {
Packit 8f7830
		if (utf8[0] == 0xE0 && (utf8[1] & 0xE0) == 0x80) /* overlong sequence check */
Packit 8f7830
			return 0;
Packit 8f7830
		/* illegal surrogates check (U+D800...U+DFFF and U+FFFE...U+FFFF) */
Packit 8f7830
		if (utf8[0] == 0xED && (utf8[1] & 0xE0) == 0xA0) /* D800-DFFF */
Packit 8f7830
			return 0;
Packit 8f7830
		if (utf8[0] == 0xEF && utf8[1] == 0xBF && (utf8[2] & 0xFE) == 0xBE) /* FFFE-FFFF */
Packit 8f7830
			return 0;
Packit 8f7830
		return 3;
Packit 8f7830
	}
Packit 8f7830
	else if ((utf8[0] & 0xF8) == 0xF0 && (utf8[1] & 0xC0) == 0x80 && (utf8[2] & 0xC0) == 0x80 && (utf8[3] & 0xC0) == 0x80) {
Packit 8f7830
		if (utf8[0] == 0xF0 && (utf8[1] & 0xF0) == 0x80) /* overlong sequence check */
Packit 8f7830
			return 0;
Packit 8f7830
		return 4;
Packit 8f7830
	}
Packit 8f7830
	else if ((utf8[0] & 0xFC) == 0xF8 && (utf8[1] & 0xC0) == 0x80 && (utf8[2] & 0xC0) == 0x80 && (utf8[3] & 0xC0) == 0x80 && (utf8[4] & 0xC0) == 0x80) {
Packit 8f7830
		if (utf8[0] == 0xF8 && (utf8[1] & 0xF8) == 0x80) /* overlong sequence check */
Packit 8f7830
			return 0;
Packit 8f7830
		return 5;
Packit 8f7830
	}
Packit 8f7830
	else if ((utf8[0] & 0xFE) == 0xFC && (utf8[1] & 0xC0) == 0x80 && (utf8[2] & 0xC0) == 0x80 && (utf8[3] & 0xC0) == 0x80 && (utf8[4] & 0xC0) == 0x80 && (utf8[5] & 0xC0) == 0x80) {
Packit 8f7830
		if (utf8[0] == 0xFC && (utf8[1] & 0xFC) == 0x80) /* overlong sequence check */
Packit 8f7830
			return 0;
Packit 8f7830
		return 6;
Packit 8f7830
	}
Packit 8f7830
	else {
Packit 8f7830
		return 0;
Packit 8f7830
	}
Packit 8f7830
}
Packit 8f7830
Packit 8f7830
Packit 8f7830
static size_t local__utf8_to_ucs2(const FLAC__byte *utf8, FLAC__uint16 *ucs2)
Packit 8f7830
{
Packit 8f7830
	const size_t len = local__utf8len(utf8);
Packit 8f7830
Packit 8f7830
	FLAC__ASSERT(0 != ucs2);
Packit 8f7830
Packit 8f7830
	if (len == 1)
Packit 8f7830
		*ucs2 = *utf8;
Packit 8f7830
	else if (len == 2)
Packit 8f7830
		*ucs2 = (*utf8 & 0x3F)<<6 | (*(utf8+1) & 0x3F);
Packit 8f7830
	else if (len == 3)
Packit 8f7830
		*ucs2 = (*utf8 & 0x1F)<<12 | (*(utf8+1) & 0x3F)<<6 | (*(utf8+2) & 0x3F);
Packit 8f7830
	else
Packit 8f7830
		*ucs2 = '?';
Packit 8f7830
Packit 8f7830
	return len;
Packit 8f7830
}
Packit 8f7830
Packit 8f7830
static FLAC__uint16 *local__convert_utf8_to_ucs2(const char *src, unsigned length)
Packit 8f7830
{
Packit 8f7830
	FLAC__uint16 *out;
Packit 8f7830
	size_t chars = 0;
Packit 8f7830
Packit 8f7830
	FLAC__ASSERT(0 != src);
Packit 8f7830
Packit 8f7830
	/* calculate length */
Packit 8f7830
	{
Packit 8f7830
		const unsigned char *s, *end;
Packit 8f7830
		for (s=(const unsigned char *)src, end=s+length; s
Packit 8f7830
			const unsigned n = local__utf8len(s);
Packit 8f7830
			if (n == 0)
Packit 8f7830
				return 0;
Packit 8f7830
			s += n;
Packit 8f7830
		}
Packit 8f7830
		FLAC__ASSERT(s == end);
Packit 8f7830
	}
Packit 8f7830
Packit 8f7830
	/* allocate */
Packit 8f7830
	out = safe_malloc_mul_2op_(chars, /*times*/sizeof(FLAC__uint16));
Packit 8f7830
	if (0 == out) {
Packit 8f7830
		FLAC__ASSERT(0);
Packit 8f7830
		return 0;
Packit 8f7830
	}
Packit 8f7830
Packit 8f7830
	/* convert */
Packit 8f7830
	{
Packit 8f7830
		const unsigned char *s = (const unsigned char *)src;
Packit 8f7830
		FLAC__uint16 *u = out;
Packit 8f7830
		for ( ; chars; chars--)
Packit 8f7830
			s += local__utf8_to_ucs2(s, u++);
Packit 8f7830
	}
Packit 8f7830
Packit 8f7830
	return out;
Packit 8f7830
}
Packit 8f7830
Packit 8f7830
static FLaC__INLINE size_t local__ucs2len(FLAC__uint16 ucs2)
Packit 8f7830
{
Packit 8f7830
	if (ucs2 < 0x0080)
Packit 8f7830
		return 1;
Packit 8f7830
	else if (ucs2 < 0x0800)
Packit 8f7830
		return 2;
Packit 8f7830
	else
Packit 8f7830
		return 3;
Packit 8f7830
}
Packit 8f7830
Packit 8f7830
static size_t local__ucs2_to_utf8(FLAC__uint16 ucs2, FLAC__byte *utf8)
Packit 8f7830
{
Packit 8f7830
	if (ucs2 < 0x080) {
Packit 8f7830
		utf8[0] = (FLAC__byte)ucs2;
Packit 8f7830
		return 1;
Packit 8f7830
	}
Packit 8f7830
	else if (ucs2 < 0x800) {
Packit 8f7830
		utf8[0] = 0xc0 | (ucs2 >> 6);
Packit 8f7830
		utf8[1] = 0x80 | (ucs2 & 0x3f);
Packit 8f7830
		return 2;
Packit 8f7830
	}
Packit 8f7830
	else {
Packit 8f7830
		utf8[0] = 0xe0 | (ucs2 >> 12);
Packit 8f7830
		utf8[1] = 0x80 | ((ucs2 >> 6) & 0x3f);
Packit 8f7830
		utf8[2] = 0x80 | (ucs2 & 0x3f);
Packit 8f7830
		return 3;
Packit 8f7830
	}
Packit 8f7830
}
Packit 8f7830
Packit 8f7830
static char *local__convert_ucs2_to_utf8(const FLAC__uint16 *src, unsigned length)
Packit 8f7830
{
Packit 8f7830
	char *out;
Packit 8f7830
	size_t len = 0, n;
Packit 8f7830
Packit 8f7830
	FLAC__ASSERT(0 != src);
Packit 8f7830
Packit 8f7830
	/* calculate length */
Packit 8f7830
	{
Packit 8f7830
		unsigned i;
Packit 8f7830
		for (i = 0; i < length; i++) {
Packit 8f7830
			n = local__ucs2len(src[i]);
Packit 8f7830
			if(len + n < len) /* overflow check */
Packit 8f7830
				return 0;
Packit 8f7830
			len += n;
Packit 8f7830
		}
Packit 8f7830
	}
Packit 8f7830
Packit 8f7830
	/* allocate */
Packit 8f7830
	out = safe_malloc_mul_2op_(len, /*times*/sizeof(char));
Packit 8f7830
	if (0 == out)
Packit 8f7830
		return 0;
Packit 8f7830
Packit 8f7830
	/* convert */
Packit 8f7830
	{
Packit 8f7830
		unsigned char *u = (unsigned char *)out;
Packit 8f7830
		for ( ; *src; src++)
Packit 8f7830
			u += local__ucs2_to_utf8(*src, u);
Packit 8f7830
		local__ucs2_to_utf8(*src, u);
Packit 8f7830
	}
Packit 8f7830
Packit 8f7830
	return out;
Packit 8f7830
}
Packit 8f7830
Packit 8f7830
Packit 8f7830
FLAC__bool FLAC_plugin__tags_get(const char *filename, FLAC__StreamMetadata **tags)
Packit 8f7830
{
Packit 8f7830
	if(!FLAC__metadata_get_tags(filename, tags))
Packit 8f7830
		if(0 == (*tags = FLAC__metadata_object_new(FLAC__METADATA_TYPE_VORBIS_COMMENT)))
Packit 8f7830
			return false;
Packit 8f7830
	return true;
Packit 8f7830
}
Packit 8f7830
Packit 8f7830
FLAC__bool FLAC_plugin__tags_set(const char *filename, const FLAC__StreamMetadata *tags)
Packit 8f7830
{
Packit 8f7830
	FLAC__Metadata_Chain *chain;
Packit 8f7830
	FLAC__Metadata_Iterator *iterator;
Packit 8f7830
	FLAC__StreamMetadata *block;
Packit 8f7830
	FLAC__bool got_vorbis_comments = false;
Packit 8f7830
	FLAC__bool ok;
Packit 8f7830
Packit 8f7830
	if(0 == (chain = FLAC__metadata_chain_new()))
Packit 8f7830
		return false;
Packit 8f7830
Packit 8f7830
	if(!FLAC__metadata_chain_read(chain, filename)) {
Packit 8f7830
		FLAC__metadata_chain_delete(chain);
Packit 8f7830
		return false;
Packit 8f7830
	}
Packit 8f7830
Packit 8f7830
	if(0 == (iterator = FLAC__metadata_iterator_new())) {
Packit 8f7830
		FLAC__metadata_chain_delete(chain);
Packit 8f7830
		return false;
Packit 8f7830
	}
Packit 8f7830
Packit 8f7830
	FLAC__metadata_iterator_init(iterator, chain);
Packit 8f7830
Packit 8f7830
	do {
Packit 8f7830
		if(FLAC__metadata_iterator_get_block_type(iterator) == FLAC__METADATA_TYPE_VORBIS_COMMENT)
Packit 8f7830
			got_vorbis_comments = true;
Packit 8f7830
	} while(!got_vorbis_comments && FLAC__metadata_iterator_next(iterator));
Packit 8f7830
Packit 8f7830
	if(0 == (block = FLAC__metadata_object_clone(tags))) {
Packit 8f7830
		FLAC__metadata_chain_delete(chain);
Packit 8f7830
		FLAC__metadata_iterator_delete(iterator);
Packit 8f7830
		return false;
Packit 8f7830
	}
Packit 8f7830
Packit 8f7830
	if(got_vorbis_comments)
Packit 8f7830
		ok = FLAC__metadata_iterator_set_block(iterator, block);
Packit 8f7830
	else
Packit 8f7830
		ok = FLAC__metadata_iterator_insert_block_after(iterator, block);
Packit 8f7830
Packit 8f7830
	FLAC__metadata_iterator_delete(iterator);
Packit 8f7830
Packit 8f7830
	if(ok) {
Packit 8f7830
		FLAC__metadata_chain_sort_padding(chain);
Packit 8f7830
		ok = FLAC__metadata_chain_write(chain, /*use_padding=*/true, /*preserve_file_stats=*/true);
Packit 8f7830
	}
Packit 8f7830
Packit 8f7830
	FLAC__metadata_chain_delete(chain);
Packit 8f7830
Packit 8f7830
	return ok;
Packit 8f7830
}
Packit 8f7830
Packit 8f7830
void FLAC_plugin__tags_destroy(FLAC__StreamMetadata **tags)
Packit 8f7830
{
Packit 8f7830
	FLAC__metadata_object_delete(*tags);
Packit 8f7830
	*tags = 0;
Packit 8f7830
}
Packit 8f7830
Packit 8f7830
const char *FLAC_plugin__tags_get_tag_utf8(const FLAC__StreamMetadata *tags, const char *name)
Packit 8f7830
{
Packit 8f7830
	const int i = FLAC__metadata_object_vorbiscomment_find_entry_from(tags, /*offset=*/0, name);
Packit 8f7830
	return (i < 0? 0 : strchr((const char *)tags->data.vorbis_comment.comments[i].entry, '=')+1);
Packit 8f7830
}
Packit 8f7830
Packit 8f7830
FLAC__uint16 *FLAC_plugin__tags_get_tag_ucs2(const FLAC__StreamMetadata *tags, const char *name)
Packit 8f7830
{
Packit 8f7830
	const char *utf8 = FLAC_plugin__tags_get_tag_utf8(tags, name);
Packit 8f7830
	if(0 == utf8)
Packit 8f7830
		return 0;
Packit 8f7830
	return local__convert_utf8_to_ucs2(utf8, strlen(utf8)+1); /* +1 for terminating null */
Packit 8f7830
}
Packit 8f7830
Packit 8f7830
int FLAC_plugin__tags_delete_tag(FLAC__StreamMetadata *tags, const char *name)
Packit 8f7830
{
Packit 8f7830
	return FLAC__metadata_object_vorbiscomment_remove_entries_matching(tags, name);
Packit 8f7830
}
Packit 8f7830
Packit 8f7830
int FLAC_plugin__tags_delete_all(FLAC__StreamMetadata *tags)
Packit 8f7830
{
Packit 8f7830
	int n = (int)tags->data.vorbis_comment.num_comments;
Packit 8f7830
	if(n > 0) {
Packit 8f7830
		if(!FLAC__metadata_object_vorbiscomment_resize_comments(tags, 0))
Packit 8f7830
			n = -1;
Packit 8f7830
	}
Packit 8f7830
	return n;
Packit 8f7830
}
Packit 8f7830
Packit 8f7830
FLAC__bool FLAC_plugin__tags_add_tag_utf8(FLAC__StreamMetadata *tags, const char *name, const char *value, const char *separator)
Packit 8f7830
{
Packit 8f7830
	int i;
Packit 8f7830
Packit 8f7830
	FLAC__ASSERT(0 != tags);
Packit 8f7830
	FLAC__ASSERT(0 != name);
Packit 8f7830
	FLAC__ASSERT(0 != value);
Packit 8f7830
Packit 8f7830
	if(separator && (i = FLAC__metadata_object_vorbiscomment_find_entry_from(tags, /*offset=*/0, name)) >= 0) {
Packit 8f7830
		FLAC__StreamMetadata_VorbisComment_Entry *entry = tags->data.vorbis_comment.comments+i;
Packit 8f7830
		const size_t value_len = strlen(value);
Packit 8f7830
		const size_t separator_len = strlen(separator);
Packit 8f7830
		FLAC__byte *new_entry;
Packit 8f7830
		if(0 == (new_entry = safe_realloc_add_4op_(entry->entry, entry->length, /*+*/value_len, /*+*/separator_len, /*+*/1)))
Packit 8f7830
			return false;
Packit 8f7830
		memcpy(new_entry+entry->length, separator, separator_len);
Packit 8f7830
		entry->length += separator_len;
Packit 8f7830
		memcpy(new_entry+entry->length, value, value_len);
Packit 8f7830
		entry->length += value_len;
Packit 8f7830
		new_entry[entry->length] = '\0';
Packit 8f7830
		entry->entry = new_entry;
Packit 8f7830
	}
Packit 8f7830
	else {
Packit 8f7830
		FLAC__StreamMetadata_VorbisComment_Entry entry;
Packit 8f7830
		if(!FLAC__metadata_object_vorbiscomment_entry_from_name_value_pair(&entry, name, value))
Packit 8f7830
			return false;
Packit 8f7830
		FLAC__metadata_object_vorbiscomment_append_comment(tags, entry, /*copy=*/false);
Packit 8f7830
	}
Packit 8f7830
	return true;
Packit 8f7830
}
Packit 8f7830
Packit 8f7830
FLAC__bool FLAC_plugin__tags_set_tag_ucs2(FLAC__StreamMetadata *tags, const char *name, const FLAC__uint16 *value, FLAC__bool replace_all)
Packit 8f7830
{
Packit 8f7830
	FLAC__StreamMetadata_VorbisComment_Entry entry;
Packit 8f7830
Packit 8f7830
	FLAC__ASSERT(0 != tags);
Packit 8f7830
	FLAC__ASSERT(0 != name);
Packit 8f7830
	FLAC__ASSERT(0 != value);
Packit 8f7830
Packit 8f7830
	{
Packit 8f7830
		char *utf8 = local__convert_ucs2_to_utf8(value, local__wide_strlen(value)+1); /* +1 for the terminating null */
Packit 8f7830
		if(0 == utf8)
Packit 8f7830
			return false;
Packit 8f7830
		if(!FLAC__metadata_object_vorbiscomment_entry_from_name_value_pair(&entry, name, utf8)) {
Packit 8f7830
			free(utf8);
Packit 8f7830
			return false;
Packit 8f7830
		}
Packit 8f7830
		free(utf8);
Packit 8f7830
	}
Packit 8f7830
	if(!FLAC__metadata_object_vorbiscomment_replace_comment(tags, entry, replace_all, /*copy=*/false))
Packit 8f7830
		return false;
Packit 8f7830
	return true;
Packit 8f7830
}