DsdLib.cxx 3.8 KB
Newer Older
1
/*
2
 * Copyright (C) 2003-2013 The Music Player Daemon Project
3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26
 * 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.
 *
 * 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.
 */

/* \file
 *
 * This file contains functions used by the DSF and DSDIFF decoders.
 *
 */

#include "config.h"
27
#include "DsdLib.hxx"
28
#include "DecoderAPI.hxx"
29
#include "util/bit_reverse.h"
30
#include "TagHandler.hxx"
Max Kellermann's avatar
Max Kellermann committed
31
#include "TagId3.hxx"
32 33

#include <unistd.h>
Max Kellermann's avatar
Max Kellermann committed
34
#include <string.h>
35 36
#include <stdio.h> /* for SEEK_SET, SEEK_CUR */

37 38 39 40
#ifdef HAVE_ID3TAG
#include <id3tag.h>
#endif

41 42 43
bool
dsdlib_id_equals(const struct dsdlib_id *id, const char *s)
{
44 45
	assert(id != nullptr);
	assert(s != nullptr);
46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65
	assert(strlen(s) == sizeof(id->value));

	return memcmp(id->value, s, sizeof(id->value)) == 0;
}

bool
dsdlib_read(struct decoder *decoder, struct input_stream *is,
	    void *data, size_t length)
{
	size_t nbytes = decoder_read(decoder, is, data, length);
	return nbytes == length;
}

/**
 * Skip the #input_stream to the specified offset.
 */
bool
dsdlib_skip_to(struct decoder *decoder, struct input_stream *is,
	       goffset offset)
{
66
	if (input_stream_is_seekable(is))
67
		return input_stream_seek(is, offset, SEEK_SET, nullptr);
68

69
	if (input_stream_get_offset(is) > offset)
70 71 72
		return false;

	char buffer[8192];
73
	while (input_stream_get_offset(is) < offset) {
74
		size_t length = sizeof(buffer);
75 76
		if (offset - input_stream_get_offset(is) < (goffset)length)
			length = offset - input_stream_get_offset(is);
77 78 79 80 81 82

		size_t nbytes = decoder_read(decoder, is, buffer, length);
		if (nbytes == 0)
			return false;
	}

83
	assert(input_stream_get_offset(is) == offset);
84 85 86 87 88 89 90 91 92 93 94 95 96 97 98
	return true;
}

/**
 * Skip some bytes from the #input_stream.
 */
bool
dsdlib_skip(struct decoder *decoder, struct input_stream *is,
	    goffset delta)
{
	assert(delta >= 0);

	if (delta == 0)
		return true;

99
	if (input_stream_is_seekable(is))
100
		return input_stream_seek(is, delta, SEEK_CUR, nullptr);
101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117

	char buffer[8192];
	while (delta > 0) {
		size_t length = sizeof(buffer);
		if ((goffset)length > delta)
			length = delta;

		size_t nbytes = decoder_read(decoder, is, buffer, length);
		if (nbytes == 0)
			return false;

		delta -= nbytes;
	}

	return true;
}

118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133
/**
 * Add tags from ID3 tag. All tags commonly found in the ID3 tags of
 * DSF and DSDIFF files are imported
 */

#ifdef HAVE_ID3TAG
void
dsdlib_tag_id3(struct input_stream *is,
	       const struct tag_handler *handler,
	       void *handler_ctx, goffset tagoffset)
{
	assert(tagoffset >= 0);

	if (tagoffset == 0)
		return;

134
	if (!dsdlib_skip_to(nullptr, is, tagoffset))
135 136
		return;

137
	struct id3_tag *id3_tag = nullptr;
138 139 140
	id3_length_t count;

	/* Prevent broken files causing problems */
141 142 143
	const goffset size = input_stream_get_size(is);
	const goffset offset = input_stream_get_offset(is);
	if (offset >= size)
144 145
		return;

146
	count = size - offset;
147 148 149 150 151 152 153 154 155

	/* Check and limit id3 tag size to prevent a stack overflow */
	if (count == 0 || count > 4096)
		return;

	id3_byte_t dsdid3[count];
	id3_byte_t *dsdid3data;
	dsdid3data = dsdid3;

156
	if (!dsdlib_read(nullptr, is, dsdid3data, count))
157 158 159
		return;

	id3_tag = id3_tag_parse(dsdid3data, count);
160
	if (id3_tag == nullptr)
161 162 163 164 165 166 167 168 169
		return;

	scan_id3_tag(id3_tag, handler, handler_ctx);

	id3_tag_delete(id3_tag);

	return;
}
#endif