-
Notifications
You must be signed in to change notification settings - Fork 1.2k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Port to node-addon-api (and N-API) and remove NAN, libuv, and v8 #2235
Conversation
@chearon I was thinking about this a little. Might it make sense to create an ArrayBuffer on the JS side so that the parameters to all the moveto and lineto and other API calls do not result in any C++ calls at all, but get recorded in the ArrayBuffer instead? Then, when the ArrayBuffer gets full or the user issues a call to render, one call is made to C++ which iterates over the ArrayBuffer and executes all the moveto/lineto/etc. API calls stored therein. The size of the ArrayBuffer can be tuned by the user as part of initialization. We can pick a sensible default if the user chooses not to provide a value. Most importantly, this would not be visible to the users at all, because we'd start the execution of the batched API calls whenever the user issued a render. It might affect debugging though, in which case we'd document that, for debugging, the ArrayBuffer size must be zero. or one, which would result in today's behaviour. |
|
||
v8::Local<v8::Object> buf = Nan::NewBuffer((char *)dest->buffer, dest->bufsize).ToLocalChecked(); | ||
Napi::Object buf = Napi::Buffer<char>::New(env, (char *)dest->buffer, dest->bufsize); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Wrapping an existing pointer in a nodejs Buffer are not compatible with recent versions of electron.
See https://github.com/nodejs/node-addon-api/blob/main/doc/external_buffer.md
It looks like you might have some other occurrences of this too
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Thanks for pointing that out. So I guess we can't claim Electron 21+ support yet. I'm going to consider that out of scope for this PR since we used external memory before these changes, but we should support Electron too without impacting Node performance. I wonder if Node will ever make the same change that Electron did too...
I'm so curious to see the use case for node-canvas in Electron. I thought you would just use <canvas>
...
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
but we should support Electron too without impacting Node performance.
There is an alternate function Napi::Buffer<char>::NewOrCopy(
designed for this case. Depending on whether you use dest
again after this call, it may not require any more changes than that.
I'm so curious to see the use case for node-canvas in Electron. I thought you would just use ...
That would depend on whether you want to use it on the backend or frontend of the application, but I agree it does sound like an usual case.
I have a couple of applications which use electron for packaging and to get a tray icon, nothing more. They are also possible to run in plain nodejs. It might technically be possible to use a canvas element, but it wouldnt be nice.
I found a library for a usb device which uses canvas for generating the images to display. I think that one is quite likely to be in electron instead of nodejs
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Actually there is a HUGE use case in electron. It is why myself and many have been praying for node-canvas adoption of N-API ;) as mentioned above a big use case is because in electron you split everything into renderer and main threads. In main, where you do all your node stuff, you cannot use <canvas>
and hence the need for node-canvas. Passing stuff between main and renderer is a pain (via IPC) so it is very useful to have node-canvas in main thread.
Lastly, and most important for me, Mozilla pdf-js relies on node-canvas and I use it for many things in my electron app. Without node-canvas napi compatibility I am not able to use pdf-js in newer versions of electron.
Does that make sense? Any questions?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Yeah we are also relying on node-canvas to handle rendering the contents of the canvas on the renderer side as a PDF or SVG in the main process. Node-canvas support for rendering to PDF/SVG is a great plus in this case, and would not be possible to do in the rendered thread currently (I think).
Basically we use node-canvas to support exporting SVG and PDF versions of the canvas scene, which is a huge plus being able to export vector versions of whatever is drawn in the canvas!
Hoping that this would work with Electron versions 21+, but honestly if this already works with Electron version 20, that's a huge improvement already, so thank you for your work!
I happened to have tried that a while ago (super sloppily, just for benchmarking) in f2e073c. That gave a 3.25x speedup in the NAN version and should give a larger speedup in the NAPI version because of the higher overhead. I'm not sure how it would actually work without explicit calls to render, since you can mix However, even if it's not batched, copying the values to a TypedArray from JS helps: 2.3 Mops/sec -> 3.0 Mops/sec for crude diffdiff --git a/lib/context2d.js b/lib/context2d.js
index 103ec63..082df97 100644
--- a/lib/context2d.js
+++ b/lib/context2d.js
@@ -9,3 +9,12 @@
const bindings = require('./bindings')
module.exports = bindings.CanvasRenderingContext2d
+
+bindings.CanvasRenderingContext2d.prototype.lineTo = function(x, y) {
+ if (isFinite(x) && isFinite(y)) {
+ const ab = this._argsBuffer;
+ ab[0] = x;
+ ab[1] = y;
+ this._lineTo();
+ }
+}
diff --git a/src/CanvasRenderingContext2d.cc b/src/CanvasRenderingContext2d.cc
index e1fb4b9..64c0c03 100644
--- a/src/CanvasRenderingContext2d.cc
+++ b/src/CanvasRenderingContext2d.cc
@@ -126,7 +126,7 @@ Context2d::Initialize(Napi::Env& env, Napi::Object& exports) {
InstanceMethod<&Context2d::RoundRect>("roundRect"),
InstanceMethod<&Context2d::MeasureText>("measureText"),
InstanceMethod<&Context2d::MoveTo>("moveTo"),
- InstanceMethod<&Context2d::LineTo>("lineTo"),
+ InstanceMethod<&Context2d::LineTo>("_lineTo"),
InstanceMethod<&Context2d::BezierCurveTo>("bezierCurveTo"),
InstanceMethod<&Context2d::QuadraticCurveTo>("quadraticCurveTo"),
InstanceMethod<&Context2d::BeginPath>("beginPath"),
@@ -233,6 +233,10 @@ Context2d::Context2d(const Napi::CallbackInfo& info) : Napi::ObjectWrap<Context2
states.emplace();
state = &states.top();
pango_layout_set_font_description(_layout, state->fontDescription);
+
+ _argsArray = Napi::Float64Array::New(env, 9);
+ info.This().As<Napi::Object>().Set("_argsBuffer", _argsArray);
+ args = _argsArray.Data();
}
/*
@@ -2572,10 +2576,6 @@ Context2d::setTextPath(double x, double y) {
void
Context2d::LineTo(const Napi::CallbackInfo& info) {
- double args[2];
- if(!checkArgs(info, args, 2))
- return;
-
cairo_line_to(context(), args[0], args[1]);
}
diff --git a/src/CanvasRenderingContext2d.h b/src/CanvasRenderingContext2d.h
index 745106e..385e06d 100644
--- a/src/CanvasRenderingContext2d.h
+++ b/src/CanvasRenderingContext2d.h
@@ -220,4 +220,6 @@ class Context2d : public Napi::ObjectWrap<Context2d> {
cairo_t *_context = nullptr;
cairo_path_t *_path;
PangoLayout *_layout = nullptr;
+ Napi::Float64Array _argsArray;
+ double* args = nullptr;
}; Tedious to do for everything, and we'd want to hide stuff better. |
@zbjornson awesome result! Sounds like it might be worth the effort, if it improves perf beyond just cancelling out the overhead incurred from switching to Node-API. |
Buffering all of the path calls is super interesting but I do agree that the flushing of the path is the downside. It has to happen in at least
I am kind of bummed about this being less pretty but that's a good speedup and it's easy. I wish |
I agree wholeheartedly. It may not be pretty, but node-canvas is definitely a performance-sensitive add-on.
I'll take a look at node-addon-api to see if we can provide variants which accept a napi_callback, thus avoiding the extra construction and additional level of indirection. We cannot avoid the unwrap though, because we need to retrieve the pointer to the native instance. We did, after all, design node-addon-api to mix and match with plain Node-API. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Got through half of the files so far. Looks great, NAPI is so much nicer to work with.
src/Canvas.cc
Outdated
, buf | ||
, Nan::New<Number>(len) }; | ||
async.runInAsyncScope(Nan::GetCurrentContext()->Global(), streaminfo->fn, sizeof argv / sizeof *argv, argv); | ||
Napi::Object buf = Napi::Buffer<uint8_t>::New(env, (uint8_t *)(data), len); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Here and everywhere there's Napi::Object xx = Napi::Buffer...
, including the return type of Canvas::ToBuffer
, should be
Napi::Object buf = Napi::Buffer<uint8_t>::New(env, (uint8_t *)(data), len); | |
auto buf = Napi::Buffer<uint8_t>::New(env, (uint8_t *)(data), len); |
or Napi::Buffer<uint8_t>
to avoid casting to the superclass.
(Avoids an MSVC warning https://learn.microsoft.com/en-us/cpp/code-quality/C26437.)
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Napi::Value
should work too if I understand correctly.
I don't know our stance on auto
but I prefer seeing the types most of the time. Maybe I should set up type checking in my editor.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Napi::Value
is still an ancestor class. Only auto
or Napi::Buffer<uint8_t>
avoid the implicit cast.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
The link above is about slicing, which is slightly different than an implicit cast, but I see what you're saying. But wouldn't there be a slice happening every time you return a more derived type in a node-addon-api callback, which must have a return type of Napi::Value
? It is designed to be used this way, so I'm not sure if it's really an issue. I don't see that MSVC warning on CI either.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Implicit cast to a super class, right. It's not actually an issue because the only members in the derived class are retrieved from V8 when needed. Fine to ignore.
NAN_METHOD(Pattern::New) { | ||
if (!info.IsConstructCall()) { | ||
return Nan::ThrowTypeError("Class constructors cannot be invoked without 'new'"); | ||
Pattern::Pattern(const Napi::CallbackInfo& info) : ObjectWrap<Pattern>(info), env(info.Env()) { |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
A crash happens with new canvas.CanvasPattern()
or ctx.createPattern()
.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I'm not able to reproduce any crash. I get Uncaught TypeError: Image or Canvas expected
for both of those, which is correct. I also tried passing a real image, and verified that is painting correctly.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
> canvas = createCanvas(1,1)
[Canvas 1x1]
> canvas.getContext("2d").createPattern()
# Exception thrown at 0x0000000068DF09A6 (libcairo-2.dll) in node.exe: 0xC0000005: Access violation reading location 0xFFFFFFFFFFFFFFFF
Turns out a crash happens with the same code in master also, but with a different cause. Can deal with later.
I actually don't think we'd gain too much from being able to provide a If needed, we may want to expose the number of statically allocated arguments as a preprocessor directive: https://github.com/nodejs/node-addon-api/blob/358b2d3b4f4cf0b3bdfc9897be888494ee36ae1d/napi.h#L1875 |
@chearon actually, it looks like napi_unwrap takes the cake: http://gabrielschulhof.github.io/misc/templated.instance.method.perf.breakdown.svg. That is not something easily changed, unfortunately. |
@gabrielschulhof thank you for looking into it. I was suggesting a direct call to an I think we should merge this and do Path2D after we're sure performance is a problem. @zbjornson huge thanks for the detailed look. I think I got everything. The |
Cool, I'll try to get through the remaining review this weekend. |
Hey, just wondering, any update on this getting merged ? :) |
Any update on plans for when this will be "officially" released? Thanks much for the hard work! |
I'm satisfied with it as-is, but would rather @zbjornson or @LinusU give an approval before merging. And anyone else that wants to review/test would be helping a lot! @Sakari369 you can use this branch temporarily, it passes all of our tests. |
Thank you for the info, will test it out! 🙏 |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Great, just a few minor comments.
if (!env.IsExceptionPending()) { | ||
Napi::Error::New(env, GENERIC_FACE_ERROR).ThrowAsJavaScriptException(); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
NAPI checks if an exception is pending and ignores subsequent calls to ThrowAsJavaScriptException()
. You could delete all of these checks.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
NODE_API_SWALLOW_UNTHROWABLE_EXCEPTIONS
is not enabled currently, so these are needed. I'll look into adding it though, I didn't know about that.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Good eyes. Fine by me if you want to keep the checks since the fatal exception is meant to reveal bugs.
Also - this fixes a few WPT tests (
|
I do get a different number of failures (123 on |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Added a comment with the fix for the one new failing WPT test.
The others went away after I pulled and rebuilt. The value of Math.PI/2
passed from JS was somehow very slightly different in the NAN and N-API versions. I went back to the earlier commits in this PR and couldn't reproduce it. 😞
42) WPT: path-objects
2d.path.arc.shape.3:
AssertionError [ERR_ASSERTION]: Red channel of the pixel at (1, 48)
+ expected - actual
-242
+0
at _assertPixel (test\wpt\generated\path-objects.js:57:9)
at Context.<anonymous> (test\wpt\generated\path-objects.js:1439:3)
at processImmediate (node:internal/timers:466:21)
This branch before pull/rebuild:
master:
-1.5707963267948966 master
... something I didn't write down that ended in 7
With the fix below, 3 new WPT tests are passing in this branch. Nice work.
this makes the lineTo benchmark (lineTo executes a very small number of operations, so it mostly measures the js<->C++ barrier) run about 50% faster
Hey fantastic work on this PR, really glad to see this working with electron again. In testing things out with pdfjs, I discovered the I wanted to bring it up here for further consideration. Thanks! |
Oh, it should be assignable since it's assignable in the browser. If you/someone wants to make a PR I can review it, or I can get around to it sometime soon. |
Gotcha @chearon, good to hear! I would love to help but I have just about no experience with with C++ - unfortunately the best I can do at the moment is complain 😄 Thanks again! |
@abrenoch good catch! Given my main use of node-canvas is for usage with pdfjs, this is key for me as well :) thanks! |
@chearon as an FYI, I do not know if this is expected or not, but the maintainer of pdf-to-img mentioned here k-yle/pdf-to-img#228 (comment) that they were not able to compile v3 pre-release and provided the error they received. Probably worth taking a look just to make sure it isn't something needed to be fixed before v3 release. For quick viewing of the error here, I took a screenshot as well (link to comment above): |
Hi, thanks for all the hard work on this ticket! I'm very excited about v3, would anyone be able to provide an estimated timeline? Thanks in advance! |
Update on prebuilds:
The tip commits on the master and prebuilds branches switch to prebuild-install, but the build failed for the rsvg reason above: https://github.com/Automattic/node-canvas/actions/runs/7352791664/job/20017985691 -- so it remains untested until that's fixed. If we don't want to switch to prebuild-install, feel free to force-push over my commits. I might have time to keep working on this tomorrow. Input welcome on approaches. |
Supporting librsvg alone is as much of a pain as building to every version of node was before N-API. It has so many |
That shrinks it from 73.5 MB to 40-50 MB. The AWS Lambda ZIP upload size limit is 50 MB; Netlify I think is the same because it uploads to Lambda; Netlify Edge is 20 MB; Good Cloud Functions 2nd gen has no limit. So, maybe/it depends. |
If you look at the "Make Bundle" section of the prebuild action, it looks like we are distributing If Linux and Windows get down to a number like that, I think it would be worth distributing in the NPM tarball and either remove SVG from prebuilds or start shipping it as a JS or WASM dependency. (I would rather remove it from prebuilds than distribute it any way, which is kind of insane: that's an entire layout engine with a CSS engine, XML parser, etc. I think there are good technical reasons not to do so, but I don't expect users to care when they see SVG support "removed"). |
Please when can we expect this to be released? |
Had some time to work on this finally, but eesh are prebuilds miserable to maintain. Lots of new build breaks:
I only got one of those fixed, which at least made the Windows prebuild work. Any help on the other issues would be welcome. |
@zbjornson Can we make a release until this, for now... And whenever we get to fix the issue for linux and Mac, you can make one more release |
@kotasudhakar no, sorry. I haven't gotten back to the actual prebuild distribution part yet (the switch away from node-pre-gyp). |
Curious where things stand today? We are looking at implementing Mozilla pdfjs but it relies on node-canvas and therefore we are not able to use with electron until the napi version is finished. Thanks for all the hard work! |
I hope to pick up where @zbjornson left off on prebuilds, but it's the last thing I want to do in my free time. We could release under a beta tag without prebuilds for electron/other users in the meantime. |
@chearon I hear ya! I wish I could help but I just have no experience in this domain. Doesn't electron require the prebuilds though? Or is that essentially what electron-rebuild does where it runs that instead of using a prebuild? Sorry for my lack of knowledge in this area. |
Progress today:
Linux and Windows now build, but we still have the installation part to do. I might work on that later today. Also want to look into making prebuilds from the 2.x NAN branch for newer Node.js versions. I am completely new to homebrew, so hoping someone familiar with it can work on the Mac build. I've been fumbling around with the tip commits in https://github.com/Automattic/node-canvas/commits/prebuilds/. |
Alright, v3.0.0-rc2 is available for testing, with prebuilds available on Linux x64 and Windows x64. https://github.com/Automattic/node-canvas/releases/tag/v3.0.0-rc2 |
macOS x64 prebuilds are up and ready for testing. I didn't realize that
|
I've finished the conversion to N-API after a month of working on this. This should help a lot with prebuilds (my main motivation tbh) enable compatibility with Bun and Electron, etc.
Because this uses APIs from N-API v6, I've bumped the minimum node version to v10.20.0.
Changes
node-addon-api
uses C++ strings, I replaced some C strings with C++CanvasRenderingContext2dInit
andCanvasPatternInit
(sends JS values to C++) are now simplybindings.setParseFont
andbindings.setDOMMatrix
Drive-bys
maybe.Check()
calls, which crash the whole process if it is empty, withUnwrapTo
andUnwrapOr
, so that in these rare cases node-canvas doesn't crash. In some instances, like when we're reading our ownImageData
, I left the force-unwraps.Performance
node-addon-api
(C++ wrapper) andnapi
(ABI stable C library) incur two new layers of overhead when going between JS and C++. I've measured this to be an extra 0.2 µs per call function on my (2012) laptop. Users that call functions millions of times, for example, lots oflineTo
s, will see performance degradation after this PR. In thelineTo
case, we may be able to make a fasterPath2D
API if we pack the paths into a byte array.Future
node-addon-api
has the ability to throw C++ exceptions when a JS exception happens. It is far easier to correctly handle errors with this enabled (for example, we wouldn't need theCHECK_RECEIVER
code) and we should think about turning that on. For this PR, I considered that change out of scope, so the green code still uses Maybes.font_face_list
is still global).Fixes #923