/*
* Copyright (c) 2016, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
*
* This code is free software; you can redistribute it and/or modify it
* under the terms of the GNU General Public License version 2 only, as
* published by the Free Software Foundation. Oracle designates this
* particular file as subject to the "Classpath" exception as provided
* by Oracle in the LICENSE file that accompanied this code.
*
* This code is distributed in the hope that it will be useful, but WITHOUT
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
* version 2 for more details (a copy is included in the LICENSE file that
* accompanied this code).
*
* You should have received a copy of the GNU General Public License version
* 2 along with this work; if not, write to the Free Software Foundation,
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
*
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
* or visit www.oracle.com if you need additional information or have any
* questions.
*/
package com.oracle.truffle.api.instrumentation.test;
import java.util.ArrayList;
import java.util.List;
import java.util.concurrent.ExecutionException;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
import java.util.concurrent.Future;
import java.util.concurrent.atomic.AtomicBoolean;
import java.util.concurrent.atomic.AtomicInteger;
import java.util.concurrent.atomic.AtomicReference;
import org.junit.Assert;
import org.junit.Test;
import com.oracle.truffle.api.frame.VirtualFrame;
import com.oracle.truffle.api.instrumentation.EventContext;
import com.oracle.truffle.api.instrumentation.ExecutionEventListener;
import com.oracle.truffle.api.instrumentation.Instrumenter;
import com.oracle.truffle.api.instrumentation.LoadSourceEvent;
import com.oracle.truffle.api.instrumentation.LoadSourceListener;
import com.oracle.truffle.api.instrumentation.LoadSourceSectionEvent;
import com.oracle.truffle.api.instrumentation.LoadSourceSectionListener;
import com.oracle.truffle.api.instrumentation.SourceSectionFilter;
import com.oracle.truffle.api.instrumentation.TruffleInstrument;
import com.oracle.truffle.api.instrumentation.TruffleInstrument.Registration;
import com.oracle.truffle.api.source.Source;
import com.oracle.truffle.api.vm.PolyglotEngine;
public class InstrumentationMultiThreadingTest {
@Test
public void testAsyncAttachement() throws InterruptedException, ExecutionException {
// unfortunately we don't support multiple evals yet
int nEvals = 1;
int nInstruments = 10;
for (int j = 0; j < 5; j++) {
ExecutorService executorService = Executors.newFixedThreadPool(nEvals + nInstruments);
List<Future<?>> futures = new ArrayList<>();
final AtomicBoolean terminated = new AtomicBoolean(false);
final AtomicReference<PolyglotEngine> engineRef = new AtomicReference<>();
for (int i = 0; i < nEvals; i++) {
futures.add(executorService.submit(new Runnable() {
public void run() {
final PolyglotEngine engine = PolyglotEngine.newBuilder().build();
engineRef.set(engine);
while (!terminated.get()) {
try {
engine.eval(Source.newBuilder("ROOT(BLOCK(STATEMENT(EXPRESSION, EXPRESSION), STATEMENT(EXPRESSION)))").name("asyncTest").mimeType(
InstrumentationTestLanguage.MIME_TYPE).build());
} catch (Throwable e) {
throw new RuntimeException(e);
}
}
}
}));
}
for (int i = 0; i < nInstruments; i++) {
final int index = i;
futures.add(executorService.submit(new Runnable() {
public void run() {
while (!terminated.get()) {
PolyglotEngine engine = engineRef.get();
if (engine != null) {
engine.getRuntime().getInstruments().get("testAsyncAttachement1").setEnabled(index % 2 == 0);
engine.getRuntime().getInstruments().get("testAsyncAttachement2").setEnabled(index % 2 == 1);
}
}
}
}));
}
Thread.sleep(1000);
terminated.set(true);
for (Future<?> future : futures) {
future.get();
}
engineRef.get().getRuntime().getInstruments().get("testAsyncAttachement1").setEnabled(false);
engineRef.get().getRuntime().getInstruments().get("testAsyncAttachement2").setEnabled(false);
}
Assert.assertEquals(0, createDisposeCount.get());
}
@Registration(id = "testAsyncAttachement1")
public static class TestAsyncAttachement1 extends TruffleInstrument {
@Override
protected void onCreate(Env env) {
createDummyBindings(env.getInstrumenter());
createDisposeCount.incrementAndGet();
}
@Override
protected void onDispose(Env env) {
createDisposeCount.decrementAndGet();
}
}
static final AtomicInteger createDisposeCount = new AtomicInteger(0);
@Registration(id = "testAsyncAttachement2")
public static class TestAsyncAttachement2 extends TruffleInstrument {
@Override
protected void onCreate(Env env) {
createDummyBindings(env.getInstrumenter());
createDisposeCount.incrementAndGet();
}
@Override
protected void onDispose(Env env) {
createDisposeCount.decrementAndGet();
}
}
private static void createDummyBindings(Instrumenter instrumenter) {
ExecutionEventListener dummyListener = new ExecutionEventListener() {
public void onReturnValue(EventContext context, VirtualFrame frame, Object result) {
}
public void onReturnExceptional(EventContext context, VirtualFrame frame, Throwable exception) {
}
public void onEnter(EventContext context, VirtualFrame frame) {
}
};
instrumenter.attachListener(SourceSectionFilter.newBuilder().tagIs(InstrumentationTestLanguage.EXPRESSION).build(), dummyListener);
instrumenter.attachListener(SourceSectionFilter.newBuilder().tagIs(InstrumentationTestLanguage.STATEMENT).build(), dummyListener);
instrumenter.attachLoadSourceListener(SourceSectionFilter.ANY, new LoadSourceListener() {
public void onLoad(LoadSourceEvent event) {
}
}, true);
instrumenter.attachLoadSourceSectionListener(SourceSectionFilter.newBuilder().tagIs(InstrumentationTestLanguage.EXPRESSION).build(), new LoadSourceSectionListener() {
public void onLoad(LoadSourceSectionEvent event) {
}
}, true);
instrumenter.attachLoadSourceSectionListener(SourceSectionFilter.newBuilder().tagIs(InstrumentationTestLanguage.STATEMENT).build(), new LoadSourceSectionListener() {
public void onLoad(LoadSourceSectionEvent event) {
}
}, true);
}
}