blob: 48cd201419dfdf5a75d3f161b1a4a0d0f7bf24df [file] [log] [blame]
/*
* Copyright 2014 Google Inc.
*
* Licensed under the Apache License, Version 2.0 (the "License"); you may not
* use this file except in compliance with the License. You may obtain a copy of
* the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
* WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
* License for the specific language governing permissions and limitations under
* the License.
*/
package com.google.gwt.dev.jjs.impl;
import com.google.gwt.core.ext.TreeLogger;
import com.google.gwt.core.ext.UnableToCompleteException;
import com.google.gwt.dev.jjs.ast.JMethod;
import com.google.gwt.dev.jjs.ast.JProgram;
/**
* Tests for the JsniRestrictionChecker.
*/
public class JsniRestrictionCheckerTest extends OptimizerTestBase {
public void testInvocationInDevirtualizedTypesSucceeds() throws Exception {
addSnippetClassDecl(
"static class Buggy {",
" interface IBar {",
" void bar();",
" }",
" static final class Foo ",
" extends com.google.gwt.core.client.JavaScriptObject implements IBar {",
" protected Foo() { };",
" void foo() { };",
" public void bar() { };",
" static void staticFoo() { };",
" }",
" native void jsniMethod(Object o) /*-{",
" @java.lang.Double::new(D)();",
" @java.lang.Boolean::new(Z)();",
" new Object().@java.lang.Number::doubleValue()();",
" new Object().@java.lang.Double::doubleValue()();",
" new Object().@Buggy.Foo::foo()();",
" new Object().@Buggy.IBar::bar()();",
" @Buggy.Foo::staticFoo()();",
" }-*/;",
"}");
assertCompileSucceeds("new Buggy().jsniMethod(null);");
}
public void testReferenceToDevirtualizedInstanceMethodFails() throws Exception {
addSnippetImport("com.google.gwt.core.client.JavaScriptObject");
addSnippetClassDecl(
"static class Buggy {",
" native void jsniMethod(Object o) /*-{",
" var a = new Object().@java.lang.Double::doubleValue();",
" }-*/;",
"}");
assertCompileFails("new Buggy().jsniMethod(null);",
"Line 6: Method 'double Double.doubleValue()' is implemented by devirtualized "
+ "type 'Double' JSO and can only be used in calls within a JSNI method body.");
}
public void testReferenceToTrampolineWarns() throws Exception {
addSnippetImport("com.google.gwt.core.client.JavaScriptObject");
addSnippetClassDecl(
"static class Buggy {",
" native void jsniMethod(Object o) /*-{",
" var a = new Object().@java.lang.Number::doubleValue();",
" var a = new Object().@java.lang.CharSequence::charAt(I);",
" var a = \"Hello\".@java.lang.Object::toString();",
" }-*/;",
"}");
assertCompileSucceeds("new Buggy().jsniMethod(null);",
"Line 6: Unsafe reference to method 'double Number.doubleValue()'. "
+ "Instance methods from 'Number' should not be called on Boolean, Double, String, "
+ "Array or JSO instances from within a JSNI method body.",
"Line 7: Unsafe reference to method 'char CharSequence.charAt(int)'. Instance methods from"
+ " 'CharSequence' should not be called on Boolean, Double, String, Array or JSO"
+ " instances from within a JSNI method body.",
"Line 8: Unsafe reference to method 'String Object.toString()'. Instance methods from "
+ "'Object' should not be called on Boolean, Double, String, Array or JSO instances "
+ "from within a JSNI method body.");
}
public void testStaticJsoDispatchSucceeds() throws Exception {
addSnippetImport("com.google.gwt.core.client.JavaScriptObject");
addSnippetClassDecl(
"static class Buggy {",
" static final class Foo extends com.google.gwt.core.client.JavaScriptObject {",
" protected Foo() { };",
" static void foo() { };",
" }",
" native void jsniMeth(Object o) /*-{",
" @Buggy.Foo::foo()();",
" }-*/;",
"}");
assertCompileSucceeds("new Buggy().jsniMeth(null);");
}
public void testJsoInterfaceDispatchFails() throws Exception {
addSnippetImport("com.google.gwt.core.client.JavaScriptObject");
addSnippetClassDecl(
"static class Buggy {",
" interface IFoo {",
" void foo();",
" }",
" static final class Foo extends JavaScriptObject implements IFoo {",
" protected Foo() { };",
" public void foo() { };",
" }",
" native void jsniMethod(Object o) /*-{",
" var a = new Object().@Buggy.IFoo::foo();",
" }-*/;",
"}");
assertCompileFails("new Buggy().jsniMethod(null);",
"Line 13: Method 'void EntryPoint.Buggy.IFoo.foo()' is implemented by a JSO and can only "
+ "be used in calls within a JSNI method body.");
}
public void testNonstaticJsoDispatchFails() throws Exception {
addSnippetImport("com.google.gwt.core.client.JavaScriptObject");
addSnippetClassDecl(
"static class Buggy {",
" native void jsniMethod(Object o) /*-{",
" var a = new Object().@com.google.gwt.core.client.JavaScriptObject::toString();",
" }-*/;",
"}");
assertCompileFails("new Buggy().jsniMethod(null);",
"Line 6: Method 'String JavaScriptObject.toString()' is implemented by a JSO and can "
+ "only be used in calls within a JSNI method body.");
}
public void testNonstaticJsoSubclassDispatchFails() throws Exception {
addSnippetImport("com.google.gwt.core.client.JavaScriptObject");
addSnippetClassDecl(
"static class Buggy {",
" static final class Foo extends com.google.gwt.core.client.JavaScriptObject {",
" protected Foo() { };",
" void foo() { };",
" }",
" native void jsniMethod(Object o) /*-{",
" var a = new Object().@Buggy.Foo::foo();",
" }-*/;",
"}");
assertCompileFails("new Buggy().jsniMethod(null);",
"Line 10: Method 'void EntryPoint.Buggy.Foo.foo()' is implemented by a JSO and can "
+ "only be used in calls within a JSNI method body.");
}
public void testStringInstanceMethodCallFail() throws Exception {
addSnippetClassDecl(
"static class Buggy {",
" static String foo;",
" native void jsniMethod(Object o) /*-{",
" var a = \"Hello\".@java.lang.String::length();",
" }-*/;",
"}");
assertCompileFails("new Buggy().jsniMethod(null);",
"Line 6: Method 'int String.length()' is implemented by devirtualized type 'String' "
+ "JSO and can only be used in calls within a JSNI method body.");
}
public void testStringStaticMethodCallSucceeds() throws Exception {
addSnippetClassDecl(
"static class Buggy {",
" static String foo;",
" native void jsniMethod(Object o) /*-{",
" var a = @java.lang.String::valueOf(Z);",
" }-*/;",
"}");
assertCompileSucceeds("new Buggy().jsniMethod(null);");
}
@Override
protected boolean doOptimizeMethod(TreeLogger logger, JProgram program, JMethod method)
throws UnableToCompleteException {
JsniRestrictionChecker.exec(logger, program);
return false;
}
}