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

Implement custom notification on weak reference garbage-collection #425

Open
wants to merge 1 commit into
base: master
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
19 changes: 17 additions & 2 deletions src/main/java/com/eclipsesource/v8/V8.java
Original file line number Diff line number Diff line change
Expand Up @@ -43,7 +43,7 @@ public class V8 extends V8Object {
private volatile static int runtimeCounter = 0;
private static String v8Flags = null;
private static boolean initialized = false;
protected Map<Long, V8Value> v8WeakReferences = new HashMap<Long, V8Value>();
protected Map<Long, WeakRefEntry> v8WeakReferences = new HashMap<Long, WeakRefEntry>();

private Map<String, Object> data = null;
private final V8Locker locker;
Expand All @@ -70,6 +70,16 @@ private class MethodDescriptor {
boolean includeReceiver;
}

static class WeakRefEntry {
V8Value value;
WeakReferenceHandler handler;

WeakRefEntry(V8Value value, WeakReferenceHandler handler) {
this.value = value;
this.handler = handler;
}
}

private synchronized static void load(final String tmpDirectory) {
try {
LibraryLoader.loadLibrary(tmpDirectory);
Expand Down Expand Up @@ -843,8 +853,13 @@ protected void disposeMethodID(final long methodID) {
}

protected void weakReferenceReleased(final long objectID) {
V8Value v8Value = v8WeakReferences.get(objectID);
WeakRefEntry entry = v8WeakReferences.get(objectID);
V8Value v8Value = entry.value;
WeakReferenceHandler handler = entry.handler;
if (v8Value != null) {
if (handler != null) {
handler.v8WeakReferenceCollected(v8Value);
}
v8WeakReferences.remove(objectID);
try {
v8Value.close();
Expand Down
24 changes: 22 additions & 2 deletions src/main/java/com/eclipsesource/v8/V8Value.java
Original file line number Diff line number Diff line change
Expand Up @@ -212,16 +212,36 @@ public V8Value twin() {
* the object will be closed, so this should only be used if there is no
* other way to track object usage.
*
* @param handler A handler that will be notified when the value weakly-
* referenced by this value is reclaimed by the V8
* garbage collector.
* @return The receiver.
*/
public V8Value setWeak() {
public V8Value setWeak(WeakReferenceHandler handler) {
v8.checkThread();
v8.checkReleased();
v8.v8WeakReferences.put(getHandle(), this);
v8.v8WeakReferences.put(getHandle(), new V8.WeakRefEntry(this, handler));
v8.setWeak(v8.getV8RuntimePtr(), getHandle());
return this;
}

/**
* Sets the V8Value as weak reference. A weak reference will eventually
* be closed when no more references exist to this object. Once setWeak
* is called, you should check if {@link V8Value#isReleased()} is true
* before invoking any methods on this object.
*
* If any other references exist to this object, the object will not be
* reclaimed. Even if no reference exist, V8 does not give any guarantee
* the object will be closed, so this should only be used if there is no
* other way to track object usage.
*
* @return The receiver.
*/
public V8Value setWeak() {
return setWeak(null);
}

/**
* If {@link V8Value#setWeak()} has been called on this Object, this method
* will return true. Otherwise it will return false.
Expand Down
5 changes: 5 additions & 0 deletions src/main/java/com/eclipsesource/v8/WeakReferenceHandler.java
Original file line number Diff line number Diff line change
@@ -0,0 +1,5 @@
package com.eclipsesource.v8;

public interface WeakReferenceHandler {
void v8WeakReferenceCollected(V8Value weakRef);
}
79 changes: 79 additions & 0 deletions src/test/java/com/eclipsesource/v8/V8WeakReferenceHandlerTest.java
Original file line number Diff line number Diff line change
@@ -0,0 +1,79 @@
package com.eclipsesource.v8;

import org.junit.After;
import org.junit.Assert;
import org.junit.Before;
import org.junit.Test;

public class V8WeakReferenceHandlerTest {
private V8 v8;

@Before
public void setup() {
V8.setFlags("--expose_gc");
v8 = V8.createV8Runtime();
}

@After
public void tearDown() {
try {
v8.close();
if (V8.getActiveRuntimes() != 0) {
throw new IllegalStateException("V8Runtimes not properly released");
}
} catch (IllegalStateException e) {
System.out.println(e.getMessage());
}
}

static class WeakReferenceObserver implements WeakReferenceHandler {
boolean collected = false;
V8Value ref;

WeakReferenceObserver(V8Value ref) {
this.ref = ref;
}

@Override
public void v8WeakReferenceCollected(V8Value weakRef) {
if (weakRef == ref) {
collected = true;
} else {
throw new IllegalStateException("v8WeakReferenceCollected() called with an unexpected reference");
}
}
}

@Test
public void testReferenceCollectedCalled() {
final V8Object o = v8.executeObjectScript(
"let a = [];" +
"a.push({foo: \"bar\"});"+
"a[0];");

WeakReferenceObserver observer = new WeakReferenceObserver(o);

o.setWeak(observer);

Assert.assertFalse(observer.collected);
v8.executeVoidScript("a = null");
v8.executeVoidScript("gc()");
Assert.assertTrue(observer.collected);
}

@Test
public void testReferenceCollectedNotCalled() {
final V8Object o = v8.executeObjectScript(
"let a = [];" +
"a.push({foo: \"bar\"});"+
"a[0];");

WeakReferenceObserver observer = new WeakReferenceObserver(o);

o.setWeak(observer);

Assert.assertFalse(observer.collected);
v8.executeVoidScript("gc()");
Assert.assertFalse(observer.collected);
}
}