/** @file
Parser: memory reference counting classes decls.
Copyright (c) 2001-2026 Art. Lebedev Studio (https://www.artlebedev.com)
Authors: Konstantin Morshnev <moko@design.ru>, Alexandr Petrosian <paf@design.ru>
*/
#ifndef PA_MEMORY_H
#define PA_MEMORY_H
#define IDENT_PA_MEMORY_H "$Id: pa_memory.h,v 1.47 2026/04/25 13:38:46 moko Exp $"
// include
#include "pa_config_includes.h"
#include <new>
// define destructors use for Array, Hash and VMethodFrame
#define USE_DESTRUCTORS
// std::basic_stringstream used in ^table.csv-string[] is compatible with delete usage check only under Debian 9/10, FreeBSD 12
// #define CHECK_DELETE_USAGE
// forwards
void *pa_fail_alloc(const char* what, size_t size);
// inlines
inline void *pa_malloc(size_t size) {
if(void *result=GC_MALLOC(size))
return result;
return pa_fail_alloc("allocate", size);
}
inline void *pa_malloc_atomic(size_t size) {
if(void *result=GC_MALLOC_ATOMIC(size))
return result;
return pa_fail_alloc("allocate clean", size);
}
/// length may be zero, and this is normal
inline char *pa_strdup(const char* auto_variable_never_null, size_t known_length) {
if(char *result=static_cast<char*>(GC_MALLOC_ATOMIC(known_length+1))) {
memcpy(result, auto_variable_never_null, known_length);
result[known_length]=0;
return result;
}
return static_cast<char*>(pa_fail_alloc("allocate clean", known_length+1));
}
inline char *pa_strdup(const char* auto_variable_never_null) {
return pa_strdup(auto_variable_never_null, strlen(auto_variable_never_null));
}
inline void pa_free(void *ptr) {
GC_FREE(ptr);
}
inline void *pa_realloc(void *ptr, size_t size) {
if(void *result=GC_REALLOC(ptr, size))
return result;
return pa_fail_alloc("reallocate to", size);
}
/// memory allocation/dallocation goes via pa_malloc/pa_free.
class PA_Allocated {
public:
/// the sole: instances allocated using our functions
static void *operator new(size_t size) {
return pa_malloc(size);
}
static void operator delete(void *ptr) {
pa_free(ptr);
}
static void *operator new[](size_t size) {
return pa_malloc(size);
}
static void operator delete[](void *ptr) {
pa_free(ptr);
}
};
// new(PointerFreeGC)/new(PointerGC) should be used to allocate types not inherited from PA_Allocated
#define PointerFreeGC (true)
#define PointerGC (false)
inline void *operator new[] (size_t size, bool pointer_free) {
return pointer_free ? pa_malloc_atomic(size) : pa_malloc(size);
}
inline void *operator new (size_t size, bool pointer_free) {
return pointer_free ? pa_malloc_atomic(size) : pa_malloc(size);
}
/// Base for all Parser classes
typedef PA_Allocated PA_Object;
// defines
#define override
#define rethrow throw
#if defined(_MSC_VER) || (__cplusplus>=201103L)
#define PA_THROW(what)
#else
#define PA_THROW(what) throw(what)
#endif
#if defined(_MSC_VER) || defined(FREEBSD1X) || defined(__APPLE__)
// no checks for FreeBSD1X.X and OS X due to https://bugs.llvm.org/show_bug.cgi?id=40161 bug
#define PA_CHECK_REGULAR_ALLOC_UNAVAILABLE
#endif
// #define PA_CHECK_REGULAR_ALLOC_UNAVAILABLE
#if !defined(PA_CHECK_REGULAR_ALLOC_UNAVAILABLE)
// regular new/delete are disabled from accidental use
void *new_disabled();
void delete_disabled();
inline void *operator new[] (std::size_t) PA_THROW(std::bad_alloc){ return new_disabled(); }
inline void operator delete[](void *) throw(){ delete_disabled(); }
inline void operator delete[](void *, std::size_t) throw(){ delete_disabled(); }
inline void *operator new(std::size_t) PA_THROW(std::bad_alloc){ return new_disabled(); }
#ifdef CHECK_DELETE_USAGE
inline void operator delete(void *) throw(){ delete_disabled(); }
#endif
#ifndef PA_DEBUG_DISABLE_GC
// other regular allocators as disabled from accidental use as well
void *calloc_disabled();
void *malloc_disabled();
void *realloc_disabled();
void free_disabled();
char *strdup_disabled();
inline void *calloc(size_t) { return calloc_disabled(); }
inline void *malloc(size_t) { return malloc_disabled(); }
inline void *realloc(void *, size_t) { return realloc_disabled(); }
inline void free(void *) { free_disabled(); }
inline char *strdup(const char*, size_t){ return strdup_disabled(); }
#endif // PA_DEBUG_DISABLE_GC
#endif // _MSC_VER
void pa_gc_collect(bool forced=false);
void pa_gc_set_free_space_divisor(int);
#endif
E-mail: