ketmar memory management patch to handle deprecated delete

This commit is contained in:
Adam D. Ruppe 2018-03-01 11:30:25 -05:00
parent 6ed8ab8a4f
commit 04b8b4b08f
2 changed files with 30 additions and 6 deletions

25
color.d
View File

@ -837,6 +837,16 @@ interface MemoryImage {
}
}
// ***This method is deliberately not publicly documented.***
// What it does is unconditionally frees internal image storage, without any sanity checks.
// If you will do this, make sure that you have no references to image data left (like
// slices of [data] array, for example). Those references will become invalid, and WILL
// lead to Undefined Behavior.
// tl;dr: IF YOU HAVE *ANY* QUESTIONS REGARDING THIS COMMENT, DON'T USE THIS!
// Note to implementors: it is safe to simply do nothing in this method.
// Also, it should be safe to call this method twice or more.
void clearInternal () nothrow @system @nogc;
/// Convenient alias for `fromImage`
alias fromImageFile = fromImage;
}
@ -850,6 +860,14 @@ class IndexedImage : MemoryImage {
/// the data as indexes into the palette. Stored left to right, top to bottom, no padding.
ubyte[] data;
override void clearInternal () nothrow @system @nogc {
import core.memory : GC;
// it is safe to call [GC.free] with `null` pointer.
GC.free(palette.ptr); palette = null;
GC.free(data.ptr); data = null;
_width = _height = 0;
}
/// .
override int width() const {
return _width;
@ -988,6 +1006,13 @@ class TrueColorImage : MemoryImage {
int _width;
int _height;
override void clearInternal () nothrow @system @nogc {
import core.memory : GC;
// it is safe to call [GC.free] with `null` pointer.
GC.free(imageData.bytes.ptr); imageData.bytes = null;
_width = _height = 0;
}
/// .
override TrueColorImage clone() const {
auto n = new TrueColorImage(width, height);

View File

@ -2968,20 +2968,19 @@ public NVGImage createImage() (NVGContext ctx, const(char)[] filename, const(NVG
static if (NanoVegaHasArsdImage) {
import arsd.image;
// do we have new arsd API to load images?
static if (!is(typeof(MemoryImage.fromImageFile))) {
static if (!is(typeof(MemoryImage.fromImageFile)) || !is(typeof(MemoryImage.clearInternal))) {
static assert(0, "Sorry, your ARSD is too old. Please, update it.");
}
try {
auto oimg = MemoryImage.fromImageFile(filename);
if (auto img = cast(TrueColorImage)oimg) {
oimg = null;
scope(exit) { delete img.imageData.bytes; delete img; }
scope(exit) oimg.clearInternal();
return ctx.createImageRGBA(img.width, img.height, img.imageData.bytes[], imageFlagsList);
} else {
TrueColorImage img = oimg.getAsTrueColorImage;
if (auto xi = cast(IndexedImage)oimg) { delete xi.palette; delete xi.data; delete xi; }
scope(exit) img.clearInternal();
oimg.clearInternal(); // drop original image, as `getAsTrueColorImage()` MUST create a new one here
oimg = null;
scope(exit) { delete img.imageData.bytes; delete img; }
return ctx.createImageRGBA(img.width, img.height, img.imageData.bytes[], imageFlagsList);
}
} catch (Exception) {}
@ -3013,7 +3012,7 @@ static if (NanoVegaHasArsdImage) {
return ctx.createImageRGBA(tc.width, tc.height, tc.imageData.bytes[], imageFlagsList);
} else {
auto tc = img.getAsTrueColorImage;
scope(exit) { delete tc.imageData.bytes; delete tc; }
scope(exit) tc.clearInternal(); // here, it is guaranteed that `tc` is newly allocated image, so it is safe to kill it
return ctx.createImageRGBA(tc.width, tc.height, tc.imageData.bytes[], imageFlagsList);
}
}