blob: b40adf2442d136c09a89cd739a83439c9c7ffe12 [file] [log] [blame]
/*
* Copyright 2016 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.dev.jjs.ast.Context;
import com.google.gwt.dev.jjs.ast.JMethod;
import com.google.gwt.dev.jjs.ast.JMethodCall;
import com.google.gwt.dev.jjs.ast.JModVisitor;
import com.google.gwt.dev.jjs.ast.JProgram;
import com.google.gwt.dev.jjs.ast.RuntimeConstants;
import com.google.gwt.thirdparty.guava.common.collect.ImmutableSet;
import com.google.gwt.thirdparty.guava.common.collect.Iterables;
import com.google.gwt.thirdparty.guava.common.collect.Sets;
import java.util.Set;
/**
* Replaces direct calls to native methods that override methods from java.lang.Object to directly
* call them through java.lang.Object. This makes sure that the calls are routed through the
* trampoline.
*/
public class ReplaceCallsToNativeJavaLangObjectOverrides {
public static void exec(final JProgram program) {
final Set<JMethod> overridableJavaLangObjectMethods = ImmutableSet.of(
program.getIndexedMethodOrNull(RuntimeConstants.OBJECT_EQUALS),
program.getIndexedMethodOrNull(RuntimeConstants.OBJECT_HASHCODE),
program.getIndexedMethodOrNull(RuntimeConstants.OBJECT_TO_STRING));
new JModVisitor() {
@Override
public void endVisit(JMethodCall x, Context ctx) {
JMethod targetMethod = x.getTarget();
if (!targetMethod.isJsNative()) {
return;
}
JMethod overridenMethod = Iterables.getOnlyElement(
Sets.intersection(targetMethod.getOverriddenMethods(),
overridableJavaLangObjectMethods),
null);
if (overridenMethod == null) {
return;
}
ctx.replaceMe(
new JMethodCall(x.getSourceInfo(), x.getInstance(), overridenMethod, x.getArgs()));
}
}.accept(program);
}
}