/*
* Copyright (c) 2017, 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.nfi.test;
import com.oracle.truffle.api.CallTarget;
import com.oracle.truffle.api.interop.TruffleObject;
import com.oracle.truffle.api.interop.java.JavaInterop;
import com.oracle.truffle.nfi.test.interop.NullObject;
import com.oracle.truffle.nfi.test.interop.TestCallback;
import com.oracle.truffle.tck.TruffleRunner;
import com.oracle.truffle.tck.TruffleRunner.Inject;
import java.util.ArrayList;
import java.util.Collection;
import static org.hamcrest.CoreMatchers.instanceOf;
import static org.hamcrest.CoreMatchers.is;
import org.junit.Assert;
import org.junit.Assume;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.junit.runners.Parameterized;
import org.junit.runners.Parameterized.Parameter;
import org.junit.runners.Parameterized.Parameters;
@RunWith(Parameterized.class)
@Parameterized.UseParametersRunnerFactory(TruffleRunner.ParametersFactory.class)
public class NullNFITest extends NFITest {
public enum NullMode {
NATIVE_NULL, // TruffleObject with isNull==true translates to native NULL pointer
BOXED_NULL, // TruffleObject with isNull==true is passed as-is
NULL_FORBIDDEN // passing NULL to native code is forbidden
}
@Parameters(name = "{0}")
public static Collection<Object[]> data() {
ArrayList<Object[]> ret = new ArrayList<>();
ret.add(new Object[]{"pointer", NullMode.NATIVE_NULL});
ret.add(new Object[]{"string", NullMode.NATIVE_NULL});
ret.add(new Object[]{"object", NullMode.BOXED_NULL});
ret.add(new Object[]{"():void", NullMode.NULL_FORBIDDEN});
return ret;
}
@Parameter(0) public String type;
@Parameter(1) public NullMode nullMode;
public class TestNullRetNode extends SendExecuteNode {
public TestNullRetNode() {
super("return_null", String.format("() : %s", type), 0);
}
}
@Test
public void testNullRet(@Inject(TestNullRetNode.class) CallTarget callTarget) {
Object ret = callTarget.call();
Assert.assertThat("return value", ret, is(instanceOf(TruffleObject.class)));
TruffleObject obj = (TruffleObject) ret;
Assert.assertTrue("isNull", JavaInterop.isNull(obj));
}
private String getExpected() {
switch (nullMode) {
case NATIVE_NULL:
return "null";
case BOXED_NULL:
return "non-null";
case NULL_FORBIDDEN:
Assume.assumeTrue("can't pass NULL to native", false);
break;
default:
Assert.fail();
}
return null;
}
public class TestNullArgNode extends SendExecuteNode {
public TestNullArgNode() {
super("null_arg", String.format("(%s) : string", type), 1);
}
}
@Test
public void testNullArg(@Inject(TestNullArgNode.class) CallTarget callTarget) {
String expected = getExpected();
Object ret = callTarget.call(new NullObject());
Assert.assertThat("return value", ret, is(instanceOf(TruffleObject.class)));
TruffleObject obj = (TruffleObject) ret;
Assert.assertTrue("isBoxed", JavaInterop.isBoxed(obj));
Assert.assertEquals("return value", expected, JavaInterop.unbox(obj));
}
public class TestNullCallbackArgNode extends SendExecuteNode {
public TestNullCallbackArgNode() {
super("callback_null_arg", String.format("((%s):void) : void", type), 1);
}
}
@Test
public void testNullCallbackArg(@Inject(TestNullCallbackArgNode.class) CallTarget callTarget) {
TruffleObject nullCallback = new TestCallback(1, (args) -> {
Assert.assertThat("callback argument", args[0], is(instanceOf(TruffleObject.class)));
Assert.assertTrue("isNull", JavaInterop.isNull((TruffleObject) args[0]));
return null;
});
callTarget.call(nullCallback);
}
public class TestNullCallbackRetNode extends SendExecuteNode {
public TestNullCallbackRetNode() {
super("callback_null_ret", String.format("(():%s) : string", type), 1);
}
}
@Test
public void testNullCallbackRet(@Inject(TestNullCallbackRetNode.class) CallTarget callTarget) {
String expected = getExpected();
TruffleObject nullCallback = new TestCallback(0, (args) -> {
return new NullObject();
});
Object ret = callTarget.call(nullCallback);
Assert.assertThat("return value", ret, is(instanceOf(TruffleObject.class)));
TruffleObject obj = (TruffleObject) ret;
Assert.assertTrue("isBoxed", JavaInterop.isBoxed(obj));
Assert.assertEquals("return value", expected, JavaInterop.unbox(obj));
}
}