view foosdk/sdk/foobar2000/SDK/mem_block_container.h @ 1:20d02a178406 default tip

*: check in everything else yay
author Paper <paper@tflc.us>
date Mon, 05 Jan 2026 02:15:46 -0500
parents
children
line wrap: on
line source

#pragma once

#include "filesystem.h"

//! Generic interface for a memory block; used by various other interfaces to return memory blocks while allowing caller to allocate.
class NOVTABLE mem_block_container {
public:
	virtual const void * get_ptr() const = 0;
	virtual void * get_ptr() = 0;
	virtual t_size get_size() const = 0;
	virtual void set_size(t_size p_size) = 0;
	
	void from_stream(stream_reader * p_stream,t_size p_bytes,abort_callback & p_abort);

	void set(const void * p_buffer,t_size p_size);
	void set(const mem_block_container & source) {copy(source);}
	template<typename t_source> void set(const t_source & source) {
		PFC_STATIC_ASSERT( sizeof(source[0]) == 1 );
		set(source.get_ptr(), source.get_size());
	}

	inline void copy(const mem_block_container & p_source) {set(p_source.get_ptr(),p_source.get_size());}
	inline void reset() {set_size(0);}

	const mem_block_container & operator=(const mem_block_container & p_source) {copy(p_source);return *this;}

	void resize(size_t v) { set_size(v); }
	size_t size() const { return get_size(); }
protected:
	mem_block_container() {}
	~mem_block_container() {}
};

//! mem_block_container implementation.
template<template<typename> class t_alloc = pfc::alloc_standard>
class mem_block_container_impl_t : public mem_block_container {
public:
	const void* get_ptr() const override { return m_data.get_ptr(); }
	void* get_ptr() override { return m_data.get_ptr(); }
	t_size get_size() const override { return m_data.get_size(); }
	void set_size(t_size p_size) override { m_data.set_size(p_size); }
private:
	pfc::array_t<t_uint8, t_alloc> m_data;
};

class mem_block_container_impl : public mem_block_container {
public:
	const void * get_ptr() const override {return m_data.ptr();}
	void * get_ptr() override {return m_data.ptr();}
	t_size get_size() const override {return m_data.size();}
	void set_size(t_size p_size) override {m_data.resize(p_size);}

	pfc::mem_block m_data;
};


template<unsigned alignBytes = 16> class mem_block_container_aligned_impl : public mem_block_container {
public:
	const void * get_ptr() const {return m_data.get_ptr();}
	void * get_ptr() {return m_data.get_ptr();}
	t_size get_size() const {return m_data.get_size();}
	void set_size(t_size p_size) {m_data.set_size(p_size);}
private:
	pfc::mem_block_aligned<16> m_data;
};

template<unsigned alignBytes = 16> class mem_block_container_aligned_incremental_impl : public mem_block_container {
public:
	mem_block_container_aligned_incremental_impl() : m_size() {}
	const void * get_ptr() const {return m_data.get_ptr();}
	void * get_ptr() {return m_data.get_ptr();}
	t_size get_size() const {return m_size;}
	void set_size(t_size p_size) {
		if (m_data.size() < p_size) {
			m_data.resize( pfc::multiply_guarded<size_t>(p_size, 3) / 2 );
		}
		m_size = p_size;
	}
private:
	pfc::mem_block_aligned<16> m_data;
	size_t m_size;
};

class mem_block_container_temp_impl : public mem_block_container {
public:
	mem_block_container_temp_impl(void * p_buffer,t_size p_size) : m_size(0), m_buffer_size(p_size), m_buffer(p_buffer) {}
	const void * get_ptr() const {return m_buffer;}
	void * get_ptr() {return m_buffer;}
	t_size get_size() const {return m_size;}
	void set_size(t_size p_size) {if (p_size > m_buffer_size) throw pfc::exception_overflow(); m_size = p_size;}
private:
	t_size m_size,m_buffer_size;
	void * m_buffer;
};

template<typename t_ref>
class mem_block_container_ref_impl : public mem_block_container {
public:
	mem_block_container_ref_impl(t_ref & ref) : m_ref(ref) {
		PFC_STATIC_ASSERT( sizeof(ref[0]) == 1 );
	}
	const void * get_ptr() const {return m_ref.get_ptr();}
	void * get_ptr() {return m_ref.get_ptr();}
	t_size get_size() const {return m_ref.get_size();}
	void set_size(t_size p_size) {m_ref.set_size(p_size);}
private:
	t_ref & m_ref;
};