org.robovm.compiler.util.generic.GenericSignatureParser Maven / Gradle / Ivy
/*
* Copyright (C) 2008 The Android Open Source Project
*
* 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 org.robovm.compiler.util.generic;
import java.lang.reflect.GenericSignatureFormatError;
import soot.BooleanType;
import soot.ByteType;
import soot.CharType;
import soot.DoubleType;
import soot.FloatType;
import soot.IntType;
import soot.LongType;
import soot.ShortType;
import soot.VoidType;
import soot.tagkit.SignatureTag;
/**
* Implements a parser for the generics signature attribute.
* Uses a top-down, recursive descent parsing approach for the following grammar:
*
* ClassSignature ::=
* OptFormalTypeParams SuperclassSignature {SuperinterfaceSignature}.
* SuperclassSignature ::= ClassTypeSignature.
* SuperinterfaceSignature ::= ClassTypeSignature.
*
* OptFormalTypeParams ::=
* ["<" FormalTypeParameter {FormalTypeParameter} ">"].
*
* FormalTypeParameter ::= Ident ClassBound {InterfaceBound}.
* ClassBound ::= ":" [FieldTypeSignature].
* InterfaceBound ::= ":" FieldTypeSignature.
*
* FieldTypeSignature ::=
* ClassTypeSignature | ArrayTypeSignature | TypeVariableSignature.
* ArrayTypeSignature ::= "[" TypSignature.
*
* ClassTypeSignature ::=
* "L" {Ident "/"} Ident OptTypeArguments {"." Ident OptTypeArguments} ";".
*
* OptTypeArguments ::= "<" TypeArgument {TypeArgument} ">".
*
* TypeArgument ::= ([WildcardIndicator] FieldTypeSignature) | "*".
* WildcardIndicator ::= "+" | "-".
*
* TypeVariableSignature ::= "T" Ident ";".
*
* TypSignature ::= FieldTypeSignature | BaseType.
* BaseType ::= "B" | "C" | "D" | "F" | "I" | "J" | "S" | "Z".
*
* MethodTypeSignature ::=
* OptFormalTypeParams "(" {TypeSignature} ")" ReturnType {ThrowsSignature}.
* ThrowsSignature ::= ("^" ClassTypeSignature) | ("^" TypeVariableSignature).
*
* ReturnType ::= TypSignature | VoidDescriptor.
* VoidDescriptor ::= "V".
*
*
*
* RoboVM note: This is a port of the GenericSignatureParser class from Android
* class library. It has been adapted to work with Soot types instead of
* java.lang.reflect.Type implementations.
*/
public class GenericSignatureParser {
public ListOfTypes exceptionTypes;
public ListOfTypes parameterTypes;
public TypeVariable[] formalTypeParameters;
public Type returnType;
public Type fieldType;
public ListOfTypes interfaceTypes;
public Type superclassType;
GenericDeclaration genericDecl;
/*
* Parser:
*/
char symbol; // 0: eof; else valid term symbol or first char of identifier.
String identifier;
/*
* Scanner:
* eof is private to the scan methods
* and it's set only when a scan is issued at the end of the buffer.
*/
private boolean eof;
char[] buffer;
int pos;
public GenericSignatureParser() {
}
void setInput(GenericDeclaration genericDecl, SignatureTag input) {
if (input != null) {
this.genericDecl = genericDecl;
this.buffer = input.getSignature().toCharArray();
this.eof = false;
scanSymbol();
}
else {
this.eof = true;
}
}
/**
* Parses the generic signature of a class and creates the data structure
* representing the signature.
*
* @param genericDecl the GenericDeclaration calling this method
* @param signature the generic signature of the class
*/
public void parseForClass(GenericDeclaration genericDecl,
SignatureTag signature) {
setInput(genericDecl, signature);
if (!eof) {
parseClassSignature();
} else {
if(genericDecl instanceof SootClassType) {
SootClassType c = (SootClassType) genericDecl;
this.formalTypeParameters = ListOfVariables.EMPTY;
this.superclassType = c.getSuperclass();
this.interfaceTypes = new ListOfTypes(c.getInterfaces());
} else {
this.formalTypeParameters = ListOfVariables.EMPTY;
this.superclassType = new SootClassType("java.lang.Object");
this.interfaceTypes = ListOfTypes.EMPTY;
}
}
}
/**
* Parses the generic signature of a method and creates the data structure
* representing the signature.
*
* @param genericDecl the GenericDeclaration calling this method
* @param signature the generic signature of the class
*/
public void parseForMethod(GenericDeclaration genericDecl,
SignatureTag signature, SootClassType[] rawExceptionTypes) {
setInput(genericDecl, signature);
if (!eof) {
parseMethodTypeSignature(rawExceptionTypes);
} else {
if(genericDecl instanceof SootMethodType) {
SootMethodType m = (SootMethodType) genericDecl;
this.formalTypeParameters = ListOfVariables.EMPTY;
this.parameterTypes = new ListOfTypes(m.getParameterTypes());
this.exceptionTypes = new ListOfTypes(m.getExceptionTypes());
this.returnType = m.getReturnType();
} else {
this.formalTypeParameters = ListOfVariables.EMPTY;
this.parameterTypes = ListOfTypes.EMPTY;
this.exceptionTypes = ListOfTypes.EMPTY;
this.returnType = new SootTypeType(VoidType.v());
}
}
}
/**
* Parses the generic signature of a constructor and creates the data
* structure representing the signature.
*
* @param genericDecl the GenericDeclaration calling this method
* @param signature the generic signature of the class
*/
public void parseForConstructor(GenericDeclaration genericDecl,
SignatureTag signature, SootClassType[] rawExceptionTypes) {
setInput(genericDecl, signature);
if (!eof) {
parseMethodTypeSignature(rawExceptionTypes);
} else {
if(genericDecl instanceof SootConstructorType) {
SootConstructorType c = (SootConstructorType) genericDecl;
this.formalTypeParameters = ListOfVariables.EMPTY;
this.parameterTypes = new ListOfTypes(c.getParameterTypes());
this.exceptionTypes = new ListOfTypes(c.getExceptionTypes());
} else {
this.formalTypeParameters = ListOfVariables.EMPTY;
this.parameterTypes = ListOfTypes.EMPTY;
this.exceptionTypes = ListOfTypes.EMPTY;
}
}
}
/**
* Parses the generic signature of a field and creates the data structure
* representing the signature.
*
* @param genericDecl the GenericDeclaration calling this method
* @param signature the generic signature of the class
*/
public void parseForField(GenericDeclaration genericDecl,
SignatureTag signature) {
setInput(genericDecl, signature);
if (!eof) {
this.fieldType = parseFieldTypeSignature();
}
}
//
// Parser:
//
void parseClassSignature() {
// ClassSignature ::=
// OptFormalTypeParameters SuperclassSignature {SuperinterfaceSignature}.
parseOptFormalTypeParameters();
// SuperclassSignature ::= ClassTypeSignature.
this.superclassType = parseClassTypeSignature();
interfaceTypes = new ListOfTypes(16);
while (symbol > 0) {
// SuperinterfaceSignature ::= ClassTypeSignature.
interfaceTypes.add(parseClassTypeSignature());
}
}
void parseOptFormalTypeParameters() {
// OptFormalTypeParameters ::=
// ["<" FormalTypeParameter {FormalTypeParameter} ">"].
ListOfVariables typeParams = new ListOfVariables();
if (symbol == '<') {
scanSymbol();
typeParams.add(parseFormalTypeParameter());
while ((symbol != '>') && (symbol > 0)) {
typeParams.add(parseFormalTypeParameter());
}
expect('>');
}
this.formalTypeParameters = typeParams.getArray();
}
ImplForVariable parseFormalTypeParameter() {
// FormalTypeParameter ::= Ident ClassBound {InterfaceBound}.
scanIdentifier();
String name = identifier.intern(); // FIXME: is this o.k.?
ListOfTypes bounds = new ListOfTypes(8);
// ClassBound ::= ":" [FieldTypeSignature].
expect(':');
if (symbol == 'L' || symbol == '[' || symbol == 'T') {
bounds.add(parseFieldTypeSignature());
}
while (symbol == ':') {
// InterfaceBound ::= ":" FieldTypeSignature.
scanSymbol();
bounds.add(parseFieldTypeSignature());
}
return new ImplForVariable(genericDecl, name, bounds);
}
Type parseFieldTypeSignature() {
// FieldTypeSignature ::= ClassTypeSignature | ArrayTypeSignature
// | TypeVariableSignature.
switch (symbol) {
case 'L':
return parseClassTypeSignature();
case '[':
// ArrayTypeSignature ::= "[" TypSignature.
scanSymbol();
return new ImplForArray(parseTypeSignature());
case 'T':
return parseTypeVariableSignature();
default:
throw new GenericSignatureFormatError();
}
}
Type parseClassTypeSignature() {
// ClassTypeSignature ::= "L" {Ident "/"} Ident
// OptTypeArguments {"." Ident OptTypeArguments} ";".
expect('L');
StringBuilder qualIdent = new StringBuilder();
scanIdentifier();
while (symbol == '/') {
scanSymbol();
qualIdent.append(identifier).append(".");
scanIdentifier();
}
qualIdent.append(this.identifier);
ListOfTypes typeArgs = parseOptTypeArguments();
ImplForType parentType =
new ImplForType(null, qualIdent.toString(), typeArgs);
ImplForType type = parentType;
while (symbol == '.') {
// Deal with Member Classes:
scanSymbol();
scanIdentifier();
qualIdent.append("$").append(identifier); // FIXME: is "$" correct?
typeArgs = parseOptTypeArguments();
type = new ImplForType(parentType, qualIdent.toString(), typeArgs);
}
expect(';');
return type;
}
ListOfTypes parseOptTypeArguments() {
// OptTypeArguments ::= "<" TypeArgument {TypeArgument} ">".
ListOfTypes typeArgs = new ListOfTypes(8);
if (symbol == '<') {
scanSymbol();
typeArgs.add(parseTypeArgument());
while ((symbol != '>') && (symbol > 0)) {
typeArgs.add(parseTypeArgument());
}
expect('>');
}
return typeArgs;
}
Type parseTypeArgument() {
// TypeArgument ::= (["+" | "-"] FieldTypeSignature) | "*".
ListOfTypes extendsBound = new ListOfTypes(1);
ListOfTypes superBound = new ListOfTypes(1);
if (symbol == '*') {
scanSymbol();
extendsBound.add(new SootClassType("java.lang.Object"));
return new ImplForWildcard(extendsBound, superBound);
}
else if (symbol == '+') {
scanSymbol();
extendsBound.add(parseFieldTypeSignature());
return new ImplForWildcard(extendsBound, superBound);
}
else if (symbol == '-') {
scanSymbol();
superBound.add(parseFieldTypeSignature());
extendsBound.add(new SootClassType("java.lang.Object"));
return new ImplForWildcard(extendsBound, superBound);
}
else {
return parseFieldTypeSignature();
}
}
ImplForVariable parseTypeVariableSignature() {
// TypeVariableSignature ::= "T" Ident ";".
expect('T');
scanIdentifier();
expect(';');
// Reference to type variable:
// Note: we don't know the declaring GenericDeclaration yet.
return new ImplForVariable(genericDecl, identifier);
}
Type parseTypeSignature() {
switch (symbol) {
case 'B': scanSymbol(); return new SootTypeType(ByteType.v());
case 'C': scanSymbol(); return new SootTypeType(CharType.v());
case 'D': scanSymbol(); return new SootTypeType(DoubleType.v());
case 'F': scanSymbol(); return new SootTypeType(FloatType.v());
case 'I': scanSymbol(); return new SootTypeType(IntType.v());
case 'J': scanSymbol(); return new SootTypeType(LongType.v());
case 'S': scanSymbol(); return new SootTypeType(ShortType.v());
case 'Z': scanSymbol(); return new SootTypeType(BooleanType.v());
default:
// Not an elementary type, but a FieldTypeSignature.
return parseFieldTypeSignature();
}
}
/**
* @param rawExceptionTypes the non-generic exceptions. This is necessary
* because the signature may omit the exceptions when none are generic.
* May be null for methods that declare no exceptions.
*/
void parseMethodTypeSignature(SootClassType[] rawExceptionTypes) {
// MethodTypeSignature ::= [FormalTypeParameters]
// "(" {TypeSignature} ")" ReturnType {ThrowsSignature}.
parseOptFormalTypeParameters();
parameterTypes = new ListOfTypes(16);
expect('(');
while (symbol != ')' && (symbol > 0)) {
parameterTypes.add(parseTypeSignature());
}
expect(')');
returnType = parseReturnType();
if (symbol == '^') {
exceptionTypes = new ListOfTypes(8);
do {
scanSymbol();
// ThrowsSignature ::= ("^" ClassTypeSignature) |
// ("^" TypeVariableSignature).
if (symbol == 'T') {
exceptionTypes.add(parseTypeVariableSignature());
} else {
exceptionTypes.add(parseClassTypeSignature());
}
} while (symbol == '^');
} else if (rawExceptionTypes != null) {
exceptionTypes = new ListOfTypes(rawExceptionTypes);
} else {
exceptionTypes = new ListOfTypes(0);
}
}
Type parseReturnType() {
// ReturnType ::= TypeSignature | "V".
if (symbol != 'V') { return parseTypeSignature(); }
else { scanSymbol(); return new SootTypeType(VoidType.v()); }
}
//
// Scanner:
//
void scanSymbol() {
if (!eof) {
if (pos < buffer.length) {
symbol = buffer[pos];
pos++;
} else {
symbol = 0;
eof = true;
}
} else {
throw new GenericSignatureFormatError();
}
}
void expect(char c) {
if (symbol == c) {
scanSymbol();
} else {
throw new GenericSignatureFormatError();
}
}
boolean isStopSymbol(char ch) {
switch (ch) {
case ':':
case '/':
case ';':
case '<':
case '.':
return true;
}
return false;
}
// PRE: symbol is the first char of the identifier.
// POST: symbol = the next symbol AFTER the identifier.
void scanIdentifier() {
if (!eof) {
StringBuilder identBuf = new StringBuilder(32);
if (!isStopSymbol(symbol)) {
identBuf.append(symbol);
do {
char ch = buffer[pos];
if ((ch >= 'a') && (ch <= 'z') || (ch >= 'A') && (ch <= 'Z')
|| !isStopSymbol(ch)) {
identBuf.append(buffer[pos]);
pos++;
} else {
identifier = identBuf.toString();
scanSymbol();
return;
}
} while (pos != buffer.length);
identifier = identBuf.toString();
symbol = 0;
eof = true;
} else {
// Ident starts with incorrect char.
symbol = 0;
eof = true;
throw new GenericSignatureFormatError();
}
} else {
throw new GenericSignatureFormatError();
}
}
}