DsdLib.cxx 3.5 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 "InputStream.hxx"
30
#include "util/bit_reverse.h"
31
#include "tag/TagHandler.hxx"
32
#include "tag/TagId3.hxx"
33
#include "util/Error.hxx"
34 35

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

39 40 41 42
#ifdef HAVE_ID3TAG
#include <id3tag.h>
#endif

43
bool
44
DsdId::Equals(const char *s) const
45
{
46
	assert(s != nullptr);
47
	assert(strlen(s) == sizeof(value));
48

49
	return memcmp(value, s, sizeof(value)) == 0;
50 51 52
}

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

70
	if (is.GetOffset() > offset)
71 72 73
		return false;

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

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

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

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

	if (delta == 0)
		return true;

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

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

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

		delta -= nbytes;
	}

	return true;
}

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

	if (tagoffset == 0)
		return;

130
	if (!dsdlib_skip_to(nullptr, is, tagoffset))
131 132
		return;

133
	struct id3_tag *id3_tag = nullptr;
134 135 136
	id3_length_t count;

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

142
	count = size - offset;
143 144 145 146 147 148 149 150 151

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

152
	if (!dsdlib_read(nullptr, is, dsdid3data, count))
153 154 155
		return;

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

	scan_id3_tag(id3_tag, handler, handler_ctx);

	id3_tag_delete(id3_tag);

	return;
}
#endif