2016-11-20 20:16:41 +00:00
|
|
|
/*
|
|
|
|
* malloc.c: safe wrappers around malloc, realloc, free, strdup
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
2017-04-29 22:21:56 +00:00
|
|
|
#include "src/puzzles.h"
|
2016-11-20 20:16:41 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* smalloc should guarantee to return a useful pointer - Halibut
|
|
|
|
* can do nothing except die when it's out of memory anyway.
|
|
|
|
*/
|
|
|
|
|
|
|
|
int allocs = 0;
|
|
|
|
int frees = 0;
|
|
|
|
|
2017-10-29 16:42:34 +00:00
|
|
|
/* We don't load as an overlay anymore, so the audiobuf should always
|
|
|
|
* be available. */
|
|
|
|
bool audiobuf_available = true;
|
2017-01-13 23:45:04 +00:00
|
|
|
|
|
|
|
static bool grab_audiobuf(void)
|
|
|
|
{
|
|
|
|
if(!audiobuf_available)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
if(rb->audio_status())
|
|
|
|
rb->audio_stop();
|
|
|
|
|
2017-10-29 16:42:34 +00:00
|
|
|
size_t sz;
|
|
|
|
|
2017-01-13 23:45:04 +00:00
|
|
|
void *audiobuf = rb->plugin_get_audio_buffer(&sz);
|
|
|
|
extern char *giant_buffer;
|
|
|
|
|
2017-10-29 16:42:34 +00:00
|
|
|
#if 0
|
|
|
|
/* Try aligning if tlsf crashes in add_new_area(). This is
|
|
|
|
* disabled now since things seem to work without it. */
|
|
|
|
void *old = audiobuf;
|
|
|
|
|
|
|
|
/* I'm sorry. */
|
|
|
|
audiobuf = (void*)((int) audiobuf & ~0xff);
|
|
|
|
audiobuf += 0x100;
|
|
|
|
|
|
|
|
sz -= audiobuf - old;
|
|
|
|
#endif
|
|
|
|
|
2017-01-13 23:45:04 +00:00
|
|
|
add_new_area(audiobuf, sz, giant_buffer);
|
|
|
|
audiobuf_available = false;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2016-11-20 20:16:41 +00:00
|
|
|
void *smalloc(size_t size) {
|
|
|
|
void *p;
|
|
|
|
p = malloc(size);
|
2017-11-22 00:28:16 +00:00
|
|
|
//LOGF("allocs: %d", ++allocs);
|
2016-11-20 20:16:41 +00:00
|
|
|
if (!p)
|
2017-01-13 23:45:04 +00:00
|
|
|
{
|
|
|
|
if(grab_audiobuf())
|
|
|
|
return smalloc(size);
|
|
|
|
fatal("out of memory");
|
|
|
|
}
|
2016-11-20 20:16:41 +00:00
|
|
|
return p;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* sfree should guaranteeably deal gracefully with freeing NULL
|
|
|
|
*/
|
|
|
|
void sfree(void *p) {
|
|
|
|
if (p) {
|
|
|
|
++frees;
|
2017-11-22 00:28:16 +00:00
|
|
|
//LOGF("frees: %d, total outstanding: %d", frees, allocs - frees);
|
2017-01-13 23:45:04 +00:00
|
|
|
free(p);
|
2016-11-20 20:16:41 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* srealloc should guaranteeably be able to realloc NULL
|
|
|
|
*/
|
|
|
|
void *srealloc(void *p, size_t size) {
|
|
|
|
void *q;
|
|
|
|
if (p) {
|
|
|
|
q = realloc(p, size);
|
|
|
|
} else {
|
2017-11-22 00:28:16 +00:00
|
|
|
//LOGF("allocs: %d", ++allocs);
|
2016-11-20 20:16:41 +00:00
|
|
|
q = malloc(size);
|
|
|
|
}
|
|
|
|
if (!q)
|
2017-01-13 23:45:04 +00:00
|
|
|
{
|
|
|
|
if(grab_audiobuf())
|
|
|
|
return srealloc(p, size);
|
|
|
|
fatal("out of memory");
|
|
|
|
}
|
2016-11-20 20:16:41 +00:00
|
|
|
return q;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* dupstr is like strdup, but with the never-return-NULL property
|
|
|
|
* of smalloc (and also reliably defined in all environments :-)
|
|
|
|
*/
|
|
|
|
char *dupstr(const char *s) {
|
|
|
|
char *r = smalloc(1+strlen(s));
|
|
|
|
strcpy(r,s);
|
|
|
|
return r;
|
|
|
|
}
|