DsdLib.cxx 3.45 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 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"
Max Kellermann's avatar
Max Kellermann committed
29
#include "input/InputStream.hxx"
30
#include "tag/TagId3.hxx"
31
#include "util/Error.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
bool
42
DsdId::Equals(const char *s) const
43
{
44
	assert(s != nullptr);
45
	assert(strlen(s) == sizeof(value));
46

47
	return memcmp(value, s, sizeof(value)) == 0;
48 49 50
}

bool
51
dsdlib_read(Decoder *decoder, InputStream &is,
52 53 54 55 56 57 58 59 60 61
	    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
62
dsdlib_skip_to(Decoder *decoder, InputStream &is,
63
	       int64_t offset)
64
{
65 66
	if (is.IsSeekable())
		return is.Seek(offset, SEEK_SET, IgnoreError());
67

68
	if (is.GetOffset() > offset)
69 70 71
		return false;

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

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

82
	assert(is.GetOffset() == offset);
83 84 85 86 87 88 89
	return true;
}

/**
 * Skip some bytes from the #input_stream.
 */
bool
90
dsdlib_skip(Decoder *decoder, InputStream &is,
91
	    int64_t delta)
92 93 94 95 96 97
{
	assert(delta >= 0);

	if (delta == 0)
		return true;

98 99
	if (is.IsSeekable())
		return is.Seek(delta, SEEK_CUR, IgnoreError());
100 101 102 103

	char buffer[8192];
	while (delta > 0) {
		size_t length = sizeof(buffer);
104
		if ((int64_t)length > delta)
105 106 107 108 109 110 111 112 113 114 115 116
			length = delta;

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

		delta -= nbytes;
	}

	return true;
}

117 118
#ifdef HAVE_ID3TAG
void
119
dsdlib_tag_id3(InputStream &is,
120
	       const struct tag_handler *handler,
121
	       void *handler_ctx, int64_t tagoffset)
122 123 124 125 126 127
{
	assert(tagoffset >= 0);

	if (tagoffset == 0)
		return;

128
	if (!dsdlib_skip_to(nullptr, is, tagoffset))
129 130
		return;

131
	struct id3_tag *id3_tag = nullptr;
132 133 134
	id3_length_t count;

	/* Prevent broken files causing problems */
135 136
	const auto size = is.GetSize();
	const auto offset = is.GetOffset();
137
	if (offset >= size)
138 139
		return;

140
	count = size - offset;
141 142 143 144 145 146 147 148 149

	/* 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;

150
	if (!dsdlib_read(nullptr, is, dsdid3data, count))
151 152 153
		return;

	id3_tag = id3_tag_parse(dsdid3data, count);
154
	if (id3_tag == nullptr)
155 156 157 158 159 160 161 162 163
		return;

	scan_id3_tag(id3_tag, handler, handler_ctx);

	id3_tag_delete(id3_tag);

	return;
}
#endif