Skip to content

Commit

Permalink
Fix CImg memcpy overflow caused by N_CHANNELS 4
Browse files Browse the repository at this point in the history
  • Loading branch information
weareu authored and kant2002 committed Oct 2, 2015
1 parent 00d7e99 commit b89363b
Show file tree
Hide file tree
Showing 7 changed files with 31 additions and 15 deletions.
5 changes: 3 additions & 2 deletions lib/Image.js
Original file line number Diff line number Diff line change
Expand Up @@ -2,8 +2,9 @@

var lwip_image = require('../build/Release/lwip_image');

function Image(pixelsBuf, width, height, trans, metadata) {
this.__lwip = new lwip_image.LwipImage(pixelsBuf, width, height);
function Image(pixelsBuf, width, height, trans, metadata, channels) {
if (!channels) { channels = 4; } //4 assumed if none provided
this.__lwip = new lwip_image.LwipImage(pixelsBuf, width, height, channels);
this.__locked = false;
this.__trans = trans;
this.__metadata = metadata;
Expand Down
4 changes: 2 additions & 2 deletions lib/obtain.js
Original file line number Diff line number Diff line change
Expand Up @@ -40,12 +40,12 @@
if (numChannels !== parseInt(numChannels) || numChannels < 1 || numChannels > 4)
throw Error("Buffer size does not match width and height");
if (numChannels !== 4) source = util.makeRgbaBuffer(source, numChannels);
callback(null, new Image(source, type.width, type.height, (numChannels % 2 === 0)));
callback(null, new Image(source, type.width, type.height, (numChannels % 2 === 0), null, numChannels));
} else if (typeof type === 'string') {
// it's an encoded image
opener = getOpener(type);
opener(source, function(err, pixelsBuf, width, height, channels, trans, metadata) {
callback(err, err ? null : new Image(pixelsBuf, width, height, trans, metadata));
callback(err, err ? null : new Image(pixelsBuf, width, height, trans, metadata, channels));
});
} else throw Error('Invalid type');
} else throw Error("Invalid source");
Expand Down
3 changes: 2 additions & 1 deletion src/image/crop_worker.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -14,7 +14,8 @@ CropWorker::~CropWorker() {}

void CropWorker::Execute () {
try {
_cimg->crop(_left, _top, 0, 0, _right, _bottom, 0, N_CHANNELS - 1);
int channels = _cimg->spectrum();
_cimg->crop(_left, _top, 0, 0, _right, _bottom, 0, channels - 1);
} catch (CImgException e) {
SetErrorMessage("Unable to crop image");
return;
Expand Down
25 changes: 19 additions & 6 deletions src/image/image.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -29,9 +29,9 @@ void LwipImage::Init(Handle<Object> exports) {
);
}

LwipImage::LwipImage(unsigned char * data, size_t width, size_t height) {
LwipImage::LwipImage(unsigned char * data, size_t width, size_t height, int channels) {
// TODO: CImg constructor may throw an exception. handle it in LwipImage::New.
_cimg = new CImg<unsigned char>(data, width, height, 1, N_CHANNELS, false);
_cimg = new CImg<unsigned char>(data, width, height, 1, channels, false);
}

LwipImage::~LwipImage() {
Expand All @@ -57,11 +57,24 @@ NAN_METHOD(LwipImage::New) {
// info[0] - pixels buffer
// info[1,2] - width and height
Local<Object> pixBuff = info[0].As<Object>();
size_t width = info[1]->NumberValue();
size_t height = info[2]->NumberValue();
unsigned char * pixels = (unsigned char *)Buffer::Data(pixBuff);
size_t width = Nan::To<uint32_t>(info[1]).FromJust();
size_t height = Nan::To<uint32_t>(info[2]).FromJust();
uint32_t channels = Nan::To<uint32_t>(info[3]).FromJust();

unsigned char * pixels = (unsigned char *)Buffer::Data(pixBuff);
size_t len = Buffer::Length(pixBuff);

//CImg expects the size of the buffer to match width*height as it will memcpy this much from buffer.
//If it doesnt' match buffer length then throw so that CImg does not read past our buffer.
int expectedSize = sizeof(unsigned char) * width * height * 1 * channels;
if (expectedSize != len)
{
Nan::ThrowError(Nan::New<String>("Invalid image size, channels or buffer provided.").ToLocalChecked());
return;
}

// TODO: handle CImg exception
LwipImage * obj = new LwipImage(pixels, width, height);
LwipImage * obj = new LwipImage(pixels, width, height, channels);
obj->Wrap(info.This());
info.GetReturnValue().Set(info.This());
}
Expand Down
5 changes: 2 additions & 3 deletions src/image/image.h
Original file line number Diff line number Diff line change
Expand Up @@ -4,8 +4,6 @@
#define cimg_display 0
#define cimg_verbosity 0

#define N_CHANNELS 4

#include <string>
#include <cmath>
#include <node.h>
Expand Down Expand Up @@ -42,10 +40,11 @@ class LwipImage : public node::ObjectWrap {
static NAN_METHOD(paste);
static NAN_METHOD(width);
static NAN_METHOD(height);
static NAN_METHOD(channels);
static NAN_METHOD(getPixel);
static NAN_METHOD(buffer);
static NAN_METHOD(setPixel);
LwipImage(unsigned char * data, size_t width, size_t height);
LwipImage(unsigned char * data, size_t width, size_t height, int channels);
~LwipImage();
private:
static Nan::Persistent<v8::FunctionTemplate> constructor;
Expand Down
1 change: 1 addition & 0 deletions src/image/init.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -2,6 +2,7 @@

// create an init function for our node module
void InitAll(Handle<Object> exports) {
Nan::HandleScope scope;
LwipImage::Init(exports);
}

Expand Down
3 changes: 2 additions & 1 deletion src/image/pad_worker.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -18,13 +18,14 @@ PadWorker::~PadWorker() {}

void PadWorker::Execute () {
CImg<unsigned char> * res;
int channels = _cimg->spectrum();
size_t oldwidth = _cimg->width(),
oldheight = _cimg->height(),
newwidth = oldwidth + _left + _right,
newheight = oldheight + _top + _bottom;
if (oldwidth != newwidth || oldheight != newheight) {
try {
res = new CImg<unsigned char>(newwidth, newheight, 1, N_CHANNELS);
res = new CImg<unsigned char>(newwidth, newheight, 1, channels);
} catch (CImgException e) {
SetErrorMessage("Out of memory");
return;
Expand Down

0 comments on commit b89363b

Please sign in to comment.