// Copyright 2000-2020 JetBrains s.r.o. Use of this source code is governed by the Apache 2.0 license that can be found in the LICENSE file. package org.jetbrains.java.decompiler.struct.gen.generics; import org.jetbrains.java.decompiler.code.CodeConstants; import org.jetbrains.java.decompiler.main.DecompilerContext; import org.jetbrains.java.decompiler.main.extern.IFernflowerLogger; import org.jetbrains.java.decompiler.util.TextUtil; import java.util.ArrayList; import java.util.List; public final class GenericMain { private static final String[] typeNames = { "byte", "char", "double", "float", "int", "long", "short", "boolean", }; public static GenericClassDescriptor parseClassSignature(String signature) { String original = signature; try { GenericClassDescriptor descriptor = new GenericClassDescriptor(); signature = parseFormalParameters(signature, descriptor.fparameters, descriptor.fbounds); String superCl = GenericType.getNextType(signature); descriptor.superclass = new GenericType(superCl); signature = signature.substring(superCl.length()); while (signature.length() > 0) { String superIf = GenericType.getNextType(signature); descriptor.superinterfaces.add(new GenericType(superIf)); signature = signature.substring(superIf.length()); } return descriptor; } catch (RuntimeException e) { DecompilerContext.getLogger().writeMessage("Invalid signature: " + original, IFernflowerLogger.Severity.WARN); return null; } } public static GenericFieldDescriptor parseFieldSignature(String signature) { try { return new GenericFieldDescriptor(new GenericType(signature)); } catch (RuntimeException e) { DecompilerContext.getLogger().writeMessage("Invalid signature: " + signature, IFernflowerLogger.Severity.WARN); return null; } } public static GenericMethodDescriptor parseMethodSignature(String signature) { String original = signature; try { List typeParameters = new ArrayList<>(); List> typeParameterBounds = new ArrayList<>(); signature = parseFormalParameters(signature, typeParameters, typeParameterBounds); int to = signature.indexOf(")"); String parameters = signature.substring(1, to); signature = signature.substring(to + 1); List parameterTypes = new ArrayList<>(); while (parameters.length() > 0) { String par = GenericType.getNextType(parameters); parameterTypes.add(new GenericType(par)); parameters = parameters.substring(par.length()); } String ret = GenericType.getNextType(signature); GenericType returnType = new GenericType(ret); signature = signature.substring(ret.length()); List exceptionTypes = new ArrayList<>(); if (signature.length() > 0) { String[] exceptions = signature.split("\\^"); for (int i = 1; i < exceptions.length; i++) { exceptionTypes.add(new GenericType(exceptions[i])); } } return new GenericMethodDescriptor(typeParameters, typeParameterBounds, parameterTypes, returnType, exceptionTypes); } catch (RuntimeException e) { DecompilerContext.getLogger().writeMessage("Invalid signature: " + original, IFernflowerLogger.Severity.WARN); return null; } } private static String parseFormalParameters(String signature, List parameters, List> bounds) { if (signature.charAt(0) != '<') { return signature; } int counter = 1; int index = 1; loop: while (index < signature.length()) { switch (signature.charAt(index)) { case '<': counter++; break; case '>': counter--; if (counter == 0) { break loop; } } index++; } String value = signature.substring(1, index); signature = signature.substring(index + 1); while (value.length() > 0) { int to = value.indexOf(":"); String param = value.substring(0, to); value = value.substring(to + 1); List lstBounds = new ArrayList<>(); while (true) { if (value.charAt(0) == ':') { // empty superclass, skip value = value.substring(1); } String bound = GenericType.getNextType(value); lstBounds.add(new GenericType(bound)); value = value.substring(bound.length()); if (value.length() == 0 || value.charAt(0) != ':') { break; } else { value = value.substring(1); } } parameters.add(param); bounds.add(lstBounds); } return signature; } public static String getGenericCastTypeName(GenericType type) { StringBuilder s = new StringBuilder(getTypeName(type)); TextUtil.append(s, "[]", type.arrayDim); return s.toString(); } private static String getTypeName(GenericType type) { int tp = type.type; if (tp <= CodeConstants.TYPE_BOOLEAN) { return typeNames[tp]; } else if (tp == CodeConstants.TYPE_VOID) { return "void"; } else if (tp == CodeConstants.TYPE_GENVAR) { return type.value; } else if (tp == CodeConstants.TYPE_OBJECT) { StringBuilder buffer = new StringBuilder(); appendClassName(type, buffer); return buffer.toString(); } throw new RuntimeException("Invalid type: " + type); } private static void appendClassName(GenericType type, StringBuilder buffer) { List enclosingClasses = type.getEnclosingClasses(); if (enclosingClasses.isEmpty()) { String name = type.value.replace('/', '.'); buffer.append(DecompilerContext.getImportCollector().getShortName(name)); } else { for (GenericType tp : enclosingClasses) { if (buffer.length() == 0) { buffer.append(DecompilerContext.getImportCollector().getShortName(tp.value.replace('/', '.'))); } else { buffer.append(tp.value); } appendTypeArguments(tp, buffer); buffer.append('.'); } buffer.append(type.value); } appendTypeArguments(type, buffer); } private static void appendTypeArguments(GenericType type, StringBuilder buffer) { if (!type.getArguments().isEmpty()) { buffer.append('<'); for (int i = 0; i < type.getArguments().size(); i++) { if (i > 0) { buffer.append(", "); } int wildcard = type.getWildcards().get(i); switch (wildcard) { case GenericType.WILDCARD_UNBOUND: buffer.append('?'); break; case GenericType.WILDCARD_EXTENDS: buffer.append("? extends "); break; case GenericType.WILDCARD_SUPER: buffer.append("? super "); break; } GenericType genPar = type.getArguments().get(i); if (genPar != null) { buffer.append(getGenericCastTypeName(genPar)); } } buffer.append(">"); } } }