DecoderAPI.cxx 2.21 KB
Newer Older
1
/*
Max Kellermann's avatar
Max Kellermann committed
2
 * Copyright 2003-2021 The Music Player Daemon Project
3
 * http://www.musicpd.org
4 5 6 7 8 9 10 11 12 13
 *
 * 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 "DecoderAPI.hxx"
Max Kellermann's avatar
Max Kellermann committed
21
#include "input/InputStream.hxx"
22
#include "Log.hxx"
23

24
#include <cassert>
25

26
size_t
27
decoder_read(DecoderClient *client,
28
	     InputStream &is,
29
	     void *buffer, size_t length) noexcept
30
{
31
	assert(buffer != nullptr);
Max Kellermann's avatar
Max Kellermann committed
32

33
	/* XXX don't allow client==nullptr */
34 35
	if (client != nullptr)
		return client->Read(is, buffer, length);
36

37 38
	try {
		return is.LockRead(buffer, length);
39 40
	} catch (...) {
		LogError(std::current_exception());
41
		return 0;
42
	}
Max Kellermann's avatar
Max Kellermann committed
43 44
}

45 46 47 48
size_t
decoder_read_much(DecoderClient *client, InputStream &is,
		  void *_buffer, size_t size) noexcept
{
Rosen Penev's avatar
Rosen Penev committed
49
	auto buffer = (uint8_t *)_buffer;
50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65

	size_t total = 0;

	while (size > 0 && !is.LockIsEOF()) {
		size_t nbytes = decoder_read(client, is, buffer, size);
		if (nbytes == 0)
			return false;

		total += nbytes;
		buffer += nbytes;
		size -= nbytes;
	}

	return total;
}

66
bool
67
decoder_read_full(DecoderClient *client, InputStream &is,
68
		  void *_buffer, size_t size) noexcept
69
{
Rosen Penev's avatar
Rosen Penev committed
70
	auto buffer = (uint8_t *)_buffer;
71 72

	while (size > 0) {
73
		size_t nbytes = decoder_read(client, is, buffer, size);
74 75 76 77 78 79 80 81 82 83
		if (nbytes == 0)
			return false;

		buffer += nbytes;
		size -= nbytes;
	}

	return true;
}

84
bool
85
decoder_skip(DecoderClient *client, InputStream &is, size_t size) noexcept
86 87 88
{
	while (size > 0) {
		char buffer[1024];
89
		size_t nbytes = decoder_read(client, is, buffer,
90 91 92 93 94 95 96 97 98
					     std::min(sizeof(buffer), size));
		if (nbytes == 0)
			return false;

		size -= nbytes;
	}

	return true;
}