blob: 56e8fba0e35d03160f4600dc047454690f079031 [file] [log] [blame]
/*
* Copyright 2008 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.JCastOperation;
import com.google.gwt.dev.jjs.ast.JDeclaredType;
import com.google.gwt.dev.jjs.ast.JExpression;
import com.google.gwt.dev.jjs.ast.JMethod;
import com.google.gwt.dev.jjs.ast.JMethodCall;
import com.google.gwt.dev.jjs.ast.JPrimitiveType;
import com.google.gwt.dev.jjs.ast.JProgram;
import java.util.Collection;
import java.util.LinkedHashMap;
import java.util.Map;
/**
* Utilities for managing autoboxing of Java primitive types.
*/
public class AutoboxUtils {
private static final JPrimitiveType[] TYPES = {
JPrimitiveType.BOOLEAN, JPrimitiveType.BYTE, JPrimitiveType.CHAR, JPrimitiveType.SHORT,
JPrimitiveType.INT, JPrimitiveType.LONG, JPrimitiveType.FLOAT, JPrimitiveType.DOUBLE};
private final Map<JPrimitiveType, JMethod> boxMethods =
new LinkedHashMap<JPrimitiveType, JMethod>();
private final Map<JDeclaredType, JMethod> unboxMethods =
new LinkedHashMap<JDeclaredType, JMethod>();
public AutoboxUtils(JProgram program) {
for (JPrimitiveType primType : TYPES) {
JDeclaredType wrapperType = program.getFromTypeMap(primType.getWrapperTypeName());
String boxSig =
"valueOf(" + primType.getJsniSignatureName() + ")" + wrapperType.getJsniSignatureName();
String unboxSig = primType.getName() + "Value()" + primType.getJsniSignatureName();
for (JMethod method : wrapperType.getMethods()) {
if (method.isStatic()) {
if (method.getSignature().equals(boxSig)) {
boxMethods.put(primType, method);
}
} else {
if (method.getSignature().equals(unboxSig)) {
unboxMethods.put(wrapperType, method);
}
}
}
}
assert boxMethods.size() == TYPES.length;
assert unboxMethods.size() == TYPES.length;
}
/**
* Box the expression <code>toBox</code> into the wrapper type corresponding
* to <code>primitiveType</code>. If <code>toBox</code> is not already of type
* <code>primitiveType</code>, then a cast may be necessary.
*/
public JExpression box(JExpression toBox, JPrimitiveType primitiveType) {
// Add a cast to toBox if need be
if (toBox.getType() != primitiveType) {
toBox = new JCastOperation(toBox.getSourceInfo(), primitiveType, toBox);
}
JMethod method = boxMethods.get(primitiveType);
assert method != null;
JMethodCall call = new JMethodCall(toBox.getSourceInfo(), null, method);
call.addArg(toBox);
return call;
}
public Collection<JMethod> getBoxMethods() {
return boxMethods.values();
}
public Collection<JMethod> getUnboxMethods() {
return unboxMethods.values();
}
/**
* If <code>x</code> is an unbox expression, then return the expression that
* is being unboxed by it. Otherwise, return <code>null</code>.
*/
public JExpression undoUnbox(JExpression arg) {
if (arg instanceof JMethodCall) {
JMethodCall argMethodCall = (JMethodCall) arg;
assert unboxMethods.values().contains(argMethodCall.getTarget());
return argMethodCall.getInstance();
}
return null;
}
}