Skip to content
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

Add JSRT weak reference APIs #2948

Merged
merged 6 commits into from
May 11, 2017
Merged

Conversation

jasongin
Copy link
Contributor

@jasongin jasongin commented May 9, 2017

  • Add a new typedef: JsWeakRef
  • Add two new APIs: JsCreateWeakReference(), JsGetWeakReferenceValue()
  • Add a native test for the new APIs

The weak reference APIs are required for N-API in Node-ChakraCore. See nodejs/node-chakracore#238

@msftclas
Copy link

msftclas commented May 9, 2017

@jasongin,
Thanks for your contribution as a Microsoft full-time employee or intern. You do not need to sign a CLA.
Thanks,
Microsoft Pull Request Bot

VALIDATE_JSREF(weakRef);

return GlobalAPIWrapper_NoRecord([&]() -> JsErrorCode {
Memory::RecyclerWeakReference<char>* recyclerWeakReference = reinterpret_cast<Memory::RecyclerWeakReference<char>*>(weakRef);
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Can we add a break on this line to make it easier to read?

@@ -2890,6 +2890,40 @@ CHAKRA_API JsSetPromiseContinuationCallback(_In_ JsPromiseContinuationCallback p
/*allowInObjectBeforeCollectCallback*/true);
}

CHAKRA_API JsCreateWeakReference(
_In_ JsValueRef value,
_Out_ JsWeakRef* weakRef)
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

weakRef [](start = 21, length = 7)

PARAM_NOT_NULL check for out parameters in both APIs

return GlobalAPIWrapper_NoRecord([&]() -> JsErrorCode {
Memory::RecyclerWeakReference<char>* recyclerWeakReference = reinterpret_cast<Memory::RecyclerWeakReference<char>*>(weakRef);
*value = reinterpret_cast<JsValueRef>(recyclerWeakReference->Get());
return JsNoError;
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Does it make sense to return a specific error code when the reference was already released? Having it return an invalid reference seems ok, I just wasn't sure whether this was consistent with other APIs.

/cc @liminzhu

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I would not consider it a failure when the weak-referenced value is not available, so I don't think it should return an error code then.

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

That's true and consistent with other libraries, there just aren't many APIs in JSRT where the out param can be JS_INVALID_REFERENCE after a successful call. I think the documentation is pretty clear, so that should cover it.

@@ -2890,6 +2890,40 @@ CHAKRA_API JsSetPromiseContinuationCallback(_In_ JsPromiseContinuationCallback p
/*allowInObjectBeforeCollectCallback*/true);
}

CHAKRA_API JsCreateWeakReference(
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

JsCreateWeakReference [](start = 11, length = 21)

I think this API should not be callable during collection callbacks as this allocates memory.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I'll add a check for that.

JsWeakRef weakRef;
FAIL_CHECK(JsCreateWeakReference(valueRef, &weakRef));

FAIL_CHECK(JsGetWeakReferenceValue(weakRef, &valueRef));
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Would it make sense to fetch the value to a new reference, and check that valueRef == weakRefValue?


Recycler* recycler = threadContext->GetRecycler();
Memory::RecyclerWeakReference<char>* recyclerWeakReference =
recycler->CreateWeakReferenceHandle<char>(reinterpret_cast<char*>(value));
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

CreateWeakReferenceHandle [](start = 22, length = 25)

Should it be ok to use FindOrCreateWeakReferenceHandle instead of CreateWeakReferenceHandle?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I'm not sure. I think @boingoing actually wrote this code originally. Taylor can you comment on this?

@@ -2890,6 +2890,40 @@ CHAKRA_API JsSetPromiseContinuationCallback(_In_ JsPromiseContinuationCallback p
/*allowInObjectBeforeCollectCallback*/true);
}

CHAKRA_API JsCreateWeakReference(
_In_ JsValueRef value,
_Out_ JsWeakRef* weakRef)
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

weakRef [](start = 21, length = 7)

Initialize weakRef to nullptr


CHAKRA_API JsGetWeakReferenceValue(
_In_ JsWeakRef weakRef,
_Out_ JsValueRef* value)
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

value [](start = 22, length = 5)

Initialize value to JS_INVALID_REFERENCE

{
VALIDATE_JSREF(weakRef);

return GlobalAPIWrapper_NoRecord([&]() -> JsErrorCode {
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Is GC deterministic? What would happen in a TTD scenario when the object backing a weak reference is collected at different points?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I don't know enough about how TTD works to answer that. I'd appreciate input from someone else about how weak references should interact with TTD.

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

@mrkmarron can hopefully offer an opinion when he's got the time

@MSLaguana
Copy link
Contributor

MSLaguana commented May 9, 2017

Looks like the OSX and Ubuntu static tests failed your freshly added test; after invoking the GC the reference wasn't invalidated.

@MSLaguana
Copy link
Contributor

@dotnet-bot test Ubuntu ubuntu_linux_release please

@jasongin
Copy link
Contributor Author

jasongin commented May 9, 2017

Looks like the OSX static test failed your freshly added test; after invoking the GC the reference wasn't invalidated.

I see ubuntu_linux_debug_static failed the same way. I ran the test (and it passed) on Ubuntu with a release build. Maybe the GC behavior is different with a debug build? I'll try to repro.

@jasongin
Copy link
Contributor Author

jasongin commented May 9, 2017

I confirmed the test passes with a Release build and fails with a Debug build. I assume because the GC has different configuration in a debug build. I don't know yet what to do about that.

@jasongin
Copy link
Contributor Author

jasongin commented May 9, 2017

Is the test even in the right place? I just noticed there are other (more complete?) JSRT tests at bin/NativeTests/JsRTApiTest.cpp. But it looks like those are Windows-only? Should I move my test over to there?

@kunalspathak
Copy link
Contributor

@jasongin - That should be the right place to add jsrt native test.

@jasongin
Copy link
Contributor Author

When I put the test into bin/NativeTests/JsRTApiTest.cpp, it passes on Windows with both debug and release builds. But, those tests are Windows-only, is that OK?

Why is there native test code in 2 places, separated for Windows and non-Windows platforms?

@kunalspathak
Copy link
Contributor

@akroshg - do you know why?

@akroshg
Copy link
Contributor

akroshg commented May 10, 2017

Why not add test in core\bin\NativeTests ?

@jasongin
Copy link
Contributor Author

@akroshg bin/NativeTests is what I just mentioned above. But those tests are Windows-only, aren't they? At least I don't see any way to build them on non-windows platforms. There is only a .vcxproj file.

I want to ensure the new APIs work on all supported platforms. Originally in this PR I had put tests under then test/native-tests directory, which seems a more obvious place to find tests than a bin directory.

@akroshg
Copy link
Contributor

akroshg commented May 10, 2017

They were written in different branches. the windows only depends on the catch framework and during that time the cross-plat was not picked up. However there were compilation and other sort of problems happened when we directly use the windows to crossplat which needed to be looked at. (It just we haven't got time to consolidate them yet).


In reply to: 300545940 [](ancestors = 300545940)

@akroshg
Copy link
Contributor

akroshg commented May 10, 2017

@ yes they are for windows only (although no effort made to consolidate them).

NativeTest were put along with ch under the bin directory. When I did that I put there so that all application (test or product) will be found under bin folder. But it seems like it can be confusing.


In reply to: 300549055 [](ancestors = 300549055)

@jasongin
Copy link
Contributor Author

I pushed a couple commits to this PR, to address feedback above and move the test to bin/NativeTests.

Remaining issues are the question about FindOrCreateWeakReferenceHandle and concern about interaction with TTD

@agarwal-sandeep
Copy link
Collaborator

:shipit:

@liminzhu
Copy link
Collaborator

Do APIs get shipped to Chakra when in ChakraCommon.h? If that's the case might want to consider just putting the APIs in ChakraCore first in ChakraCore.h. Also curious how is this used in Node? Got kind of lost on nodejs/node-chakracore#238.

@jasongin
Copy link
Contributor Author

Also curious how is this used in Node?

This is necessary to implement the reference functions in the new Node.js addon API in Node-ChakraCore. Those reference APIs are used by native addons that hold a reference to JavaScript values that have associated native data, such as a typed array with external data, or a native class instance that is "wrapped" in a JavaScript object. The weak references are useful because often the native code only needs to access the value as long as some JS code is still holding it.

In Node-V8, those APIs are implemented using a v8::Persistent, which supports weak-reference functionality via v8::PersistentBase::SetWeak().

@liminzhu
Copy link
Collaborator

@jasongin exactly what I look for, thanks! Do you imagine an additional finalizer param for JsCreateWeakReference called upon invalidating the weak reference useful, similar to what v8::PersistentBase::SetWeak() does?

@jasongin
Copy link
Contributor Author

Do you imagine an additional finalizer param for JsCreateWeakReference called upon invalidating the weak reference useful, similar to what v8::PersistentBase::SetWeak() does?

That would be redundant because JsSetObjectBeforeCollectCallback already provides that functionality.

@jasongin
Copy link
Contributor Author

Do APIs get shipped to Chakra when in ChakraCommon.h? If that's the case might want to consider just putting the APIs in ChakraCore first in ChakraCore.h

What does everyone think? While these APIs should work fine in Chakra, I don't know any reason they will be really needed there. (There's no plan to implement N-API for Node-Chakra AFAIK.)

@liminzhu
Copy link
Collaborator

@jasongin my argument - NAPI is still experimental and in case we need to change the new JSRT APIs for w/e reason it's a lot easier when they're just in ChakraCore.

I want them in Chakra eventually ofc. In general I don't like API differences between Chakra and ChakraCore unless there's legitimate reason not to ship something to Chakra, but there's no need to rush for the reason you mentioned. My mental model is to 'stage' experimental APIs in ChakraCore and once they mature we push them in Chakra.

 - Move new APIs from ChakraCommon.h to ChakraCore.h, conditional on CHAKRACOREBUILD_
 - Call FindOrCreateWeakReferenceHandle() instead of CreateWeakReferenceHandle()
@jasongin
Copy link
Contributor Author

I pushed another commit:

  • Moved new APIs from ChakraCommon.h to ChakraCore.h, and made them conditional on CHAKRACOREBUILD_
  • Using FindOrCreateWeakReferenceHandle() instead of CreateWeakReferenceHandle()

@jasongin
Copy link
Contributor Author

I think the TTD issue should not be blocking for now. These changes can only impact TTD if the new APIs are called, and in the short term that's only likely to be done via N-API in Node-ChakraCore, and N-API is still officially experimental.

@@ -103,6 +103,8 @@
JsGetRuntime
JsIdle
JsSetPromiseContinuationCallback
JsCreateWeakReference
JsGetWeakReferenceValue
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Since you moved the headers to ChakraCore.h, you should also move these lines below the #ifndef below as well, I believe.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Fixed!

Copy link
Contributor

@boingoing boingoing left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

LGTM 👍

I think FindOrCreateWeakReferenceHandle is the right thing to do here. Not sure about the TTD impact, though.

@jasongin jasongin merged commit ca71319 into chakra-core:release/2.0 May 11, 2017
@jasongin jasongin deleted the weakref branch May 11, 2017 06:02
dilijev added a commit that referenced this pull request May 11, 2017
Merge pull request #2948 from jasongin:weakref

 - Add a new typedef: `JsWeakRef`
 - Add two new APIs: `JsCreateWeakReference()`, `JsGetWeakReferenceValue()`
 - Add a native test for the new APIs

The weak reference APIs are required for N-API in Node-ChakraCore. See nodejs/node-chakracore#238
chakrabot pushed a commit that referenced this pull request May 11, 2017
Merge pull request #2948 from jasongin:weakref

 - Add a new typedef: `JsWeakRef`
 - Add two new APIs: `JsCreateWeakReference()`, `JsGetWeakReferenceValue()`
 - Add a native test for the new APIs

The weak reference APIs are required for N-API in Node-ChakraCore. See nodejs/node-chakracore#238
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

10 participants