2022-02-11 09:07:11 +11:00
|
|
|
/* SPDX-License-Identifier: GPL-2.0-or-later */
|
2007-06-25 19:50:25 +00:00
|
|
|
|
2019-02-18 08:08:12 +11:00
|
|
|
/** \file
|
|
|
|
|
* \ingroup imbdds
|
2011-02-27 20:23:21 +00:00
|
|
|
*/
|
|
|
|
|
|
2007-06-25 19:50:25 +00:00
|
|
|
#include <Stream.h>
|
|
|
|
|
|
2020-12-04 11:28:09 +01:00
|
|
|
#include <cstdio> /* printf */
|
|
|
|
|
#include <cstring> /* memcpy */
|
2007-06-25 19:50:25 +00:00
|
|
|
|
2015-01-09 04:23:01 +11:00
|
|
|
static const char *msg_error_seek = "DDS: trying to seek beyond end of stream (corrupt file?)";
|
|
|
|
|
static const char *msg_error_read = "DDS: trying to read beyond end of stream (corrupt file?)";
|
|
|
|
|
|
2022-03-16 11:58:22 +11:00
|
|
|
inline bool is_read_within_bounds(const Stream &mem, unsigned int count)
|
2022-01-10 14:26:57 +01:00
|
|
|
{
|
|
|
|
|
if (mem.pos >= mem.size) {
|
|
|
|
|
/* No more data remained in the memory buffer. */
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
2022-03-16 11:58:22 +11:00
|
|
|
if (count > mem.size - mem.pos) {
|
2022-01-10 14:26:57 +01:00
|
|
|
/* Reading past the memory bounds. */
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
2007-06-25 19:50:25 +00:00
|
|
|
unsigned int Stream::seek(unsigned int p)
|
|
|
|
|
{
|
|
|
|
|
if (p > size) {
|
2019-04-16 03:39:29 +02:00
|
|
|
set_failed(msg_error_seek);
|
2007-06-25 19:50:25 +00:00
|
|
|
}
|
|
|
|
|
else {
|
|
|
|
|
pos = p;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return pos;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
unsigned int mem_read(Stream &mem, unsigned long long &i)
|
|
|
|
|
{
|
2022-01-10 14:26:57 +01:00
|
|
|
if (!is_read_within_bounds(mem, 8)) {
|
2019-04-16 03:39:29 +02:00
|
|
|
mem.set_failed(msg_error_seek);
|
2020-08-08 11:02:11 +10:00
|
|
|
return 0;
|
2013-01-19 06:12:25 +00:00
|
|
|
}
|
2022-05-13 09:24:28 +10:00
|
|
|
memcpy(&i, mem.mem + mem.pos, 8); /* TODO: make sure little endian. */
|
2007-06-25 19:50:25 +00:00
|
|
|
mem.pos += 8;
|
2020-08-08 11:02:11 +10:00
|
|
|
return 8;
|
2007-06-25 19:50:25 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
unsigned int mem_read(Stream &mem, unsigned int &i)
|
|
|
|
|
{
|
2022-01-10 14:26:57 +01:00
|
|
|
if (!is_read_within_bounds(mem, 4)) {
|
2019-04-16 03:39:29 +02:00
|
|
|
mem.set_failed(msg_error_read);
|
2020-08-08 11:02:11 +10:00
|
|
|
return 0;
|
2013-01-19 06:12:25 +00:00
|
|
|
}
|
2022-05-13 09:24:28 +10:00
|
|
|
memcpy(&i, mem.mem + mem.pos, 4); /* TODO: make sure little endian. */
|
2007-06-25 19:50:25 +00:00
|
|
|
mem.pos += 4;
|
2020-08-08 11:02:11 +10:00
|
|
|
return 4;
|
2007-06-25 19:50:25 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
unsigned int mem_read(Stream &mem, unsigned short &i)
|
|
|
|
|
{
|
2022-01-10 14:26:57 +01:00
|
|
|
if (!is_read_within_bounds(mem, 2)) {
|
2019-04-16 03:39:29 +02:00
|
|
|
mem.set_failed(msg_error_read);
|
2020-08-08 11:02:11 +10:00
|
|
|
return 0;
|
2013-01-19 06:12:25 +00:00
|
|
|
}
|
2022-05-13 09:24:28 +10:00
|
|
|
memcpy(&i, mem.mem + mem.pos, 2); /* TODO: make sure little endian. */
|
2007-06-25 19:50:25 +00:00
|
|
|
mem.pos += 2;
|
2020-08-08 11:02:11 +10:00
|
|
|
return 2;
|
2007-06-25 19:50:25 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
unsigned int mem_read(Stream &mem, unsigned char &i)
|
|
|
|
|
{
|
2022-01-10 14:26:57 +01:00
|
|
|
if (!is_read_within_bounds(mem, 1)) {
|
2019-04-16 03:39:29 +02:00
|
|
|
mem.set_failed(msg_error_read);
|
2020-08-08 11:02:11 +10:00
|
|
|
return 0;
|
2013-01-19 06:12:25 +00:00
|
|
|
}
|
2007-06-25 19:50:25 +00:00
|
|
|
i = (mem.mem + mem.pos)[0];
|
|
|
|
|
mem.pos += 1;
|
2020-08-08 11:02:11 +10:00
|
|
|
return 1;
|
2007-06-25 19:50:25 +00:00
|
|
|
}
|
|
|
|
|
|
2022-03-16 11:58:22 +11:00
|
|
|
unsigned int mem_read(Stream &mem, unsigned char *i, unsigned int count)
|
2007-10-12 16:09:59 +00:00
|
|
|
{
|
2022-03-16 11:58:22 +11:00
|
|
|
if (!is_read_within_bounds(mem, count)) {
|
2019-04-16 03:39:29 +02:00
|
|
|
mem.set_failed(msg_error_read);
|
2020-08-08 11:02:11 +10:00
|
|
|
return 0;
|
2013-01-19 06:12:25 +00:00
|
|
|
}
|
2022-03-16 11:58:22 +11:00
|
|
|
memcpy(i, mem.mem + mem.pos, count);
|
|
|
|
|
mem.pos += count;
|
|
|
|
|
return count;
|
2007-10-12 16:09:59 +00:00
|
|
|
}
|
2019-04-16 03:39:29 +02:00
|
|
|
|
|
|
|
|
void Stream::set_failed(const char *msg)
|
|
|
|
|
{
|
|
|
|
|
if (!failed) {
|
2019-04-16 05:51:25 +02:00
|
|
|
puts(msg);
|
2019-04-16 03:39:29 +02:00
|
|
|
failed = true;
|
|
|
|
|
}
|
|
|
|
|
}
|