All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.robovm.compiler.ClassCompiler Maven / Gradle / Ivy

The newest version!
/*
 * Copyright (C) 2012 RoboVM AB
 *
 * This program is free software; you can redistribute it and/or
 * modify it under the terms of the GNU General Public License
 * as published by the Free Software Foundation; either version 2
 * of the License, or (at your option) any later version.
 *
 * This program is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 * GNU General Public License for more details.
 *
 * You should have received a copy of the GNU General Public License
 * along with this program.  If not, see .
 */
package org.robovm.compiler;

import org.apache.commons.io.FileUtils;
import org.apache.commons.io.IOUtils;
import org.apache.commons.lang3.tuple.Triple;
import org.robovm.compiler.clazz.Clazz;
import org.robovm.compiler.clazz.ClazzInfo;
import org.robovm.compiler.clazz.Dependency;
import org.robovm.compiler.clazz.MethodInfo;
import org.robovm.compiler.config.Arch;
import org.robovm.compiler.config.Config;
import org.robovm.compiler.config.OS;
import org.robovm.compiler.llvm.Alias;
import org.robovm.compiler.llvm.AliasRef;
import org.robovm.compiler.llvm.And;
import org.robovm.compiler.llvm.ArrayConstant;
import org.robovm.compiler.llvm.ArrayConstantBuilder;
import org.robovm.compiler.llvm.ArrayType;
import org.robovm.compiler.llvm.Bitcast;
import org.robovm.compiler.llvm.Br;
import org.robovm.compiler.llvm.ByteArrayConstant;
import org.robovm.compiler.llvm.Constant;
import org.robovm.compiler.llvm.ConstantBitcast;
import org.robovm.compiler.llvm.Fence;
import org.robovm.compiler.llvm.Function;
import org.robovm.compiler.llvm.FunctionDeclaration;
import org.robovm.compiler.llvm.FunctionRef;
import org.robovm.compiler.llvm.Getelementptr;
import org.robovm.compiler.llvm.Global;
import org.robovm.compiler.llvm.GlobalRef;
import org.robovm.compiler.llvm.Icmp;
import org.robovm.compiler.llvm.IntegerConstant;
import org.robovm.compiler.llvm.Label;
import org.robovm.compiler.llvm.Linkage;
import org.robovm.compiler.llvm.Load;
import org.robovm.compiler.llvm.NullConstant;
import org.robovm.compiler.llvm.Ordering;
import org.robovm.compiler.llvm.PointerType;
import org.robovm.compiler.llvm.Ret;
import org.robovm.compiler.llvm.Store;
import org.robovm.compiler.llvm.StructureConstant;
import org.robovm.compiler.llvm.StructureConstantBuilder;
import org.robovm.compiler.llvm.StructureType;
import org.robovm.compiler.llvm.Type;
import org.robovm.compiler.llvm.Value;
import org.robovm.compiler.llvm.Variable;
import org.robovm.compiler.llvm.VariableRef;
import org.robovm.compiler.plugin.CompilerPlugin;
import org.robovm.debugger.debuginfo.DebuggerDebugObjectFileInfo;
import org.robovm.compiler.plugin.objc.ObjCMemberPlugin;
import org.robovm.compiler.trampoline.Checkcast;
import org.robovm.compiler.trampoline.Instanceof;
import org.robovm.compiler.trampoline.Invoke;
import org.robovm.compiler.trampoline.Invokeinterface;
import org.robovm.compiler.trampoline.Invokevirtual;
import org.robovm.compiler.trampoline.Trampoline;
import org.robovm.compiler.util.io.HfsCompressor;
import org.robovm.llvm.Context;
import org.robovm.llvm.LineInfo;
import org.robovm.llvm.Module;
import org.robovm.llvm.ObjectFile;
import org.robovm.llvm.PassManager;
import org.robovm.llvm.PassManagerBuilder;
import org.robovm.llvm.Symbol;
import org.robovm.llvm.Target;
import org.robovm.llvm.TargetMachine;
import org.robovm.llvm.binding.Attribute;
import org.robovm.llvm.binding.CodeGenFileType;
import org.robovm.llvm.binding.CodeGenOptLevel;
import org.robovm.llvm.binding.RelocMode;
import soot.BooleanType;
import soot.ByteType;
import soot.CharType;
import soot.DoubleType;
import soot.FloatType;
import soot.IntType;
import soot.LongType;
import soot.Modifier;
import soot.PrimType;
import soot.RefLikeType;
import soot.ShortType;
import soot.SootClass;
import soot.SootField;
import soot.SootMethod;
import soot.VoidType;
import soot.jimple.Jimple;
import soot.jimple.JimpleBody;
import soot.jimple.internal.JReturnVoidStmt;
import soot.tagkit.ConstantValueTag;
import soot.tagkit.Tag;

import java.io.BufferedReader;
import java.io.BufferedWriter;
import java.io.ByteArrayInputStream;
import java.io.ByteArrayOutputStream;
import java.io.File;
import java.io.IOException;
import java.io.InputStream;
import java.io.InputStreamReader;
import java.io.OutputStream;
import java.io.OutputStreamWriter;
import java.nio.charset.StandardCharsets;
import java.util.ArrayList;
import java.util.Collections;
import java.util.Comparator;
import java.util.HashMap;
import java.util.HashSet;
import java.util.List;
import java.util.Map;
import java.util.Set;
import java.util.concurrent.Executor;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.RejectedExecutionException;
import java.util.regex.Matcher;
import java.util.regex.Pattern;

import static org.robovm.compiler.Annotations.*;
import static org.robovm.compiler.Functions.*;
import static org.robovm.compiler.StructMemberMethodCompiler.STRUCT_ATTRIBUTES_METHOD;
import static org.robovm.compiler.Types.*;
import static org.robovm.compiler.llvm.Type.*;

/**
 *
 * @version $Id$
 */
public class ClassCompiler {
    private static final int DUMMY_METHOD_SIZE = 0x01abcdef;
    public static final int CI_PUBLIC = 0x1;
    public static final int CI_FINAL = 0x2;
    public static final int CI_INTERFACE = 0x4;
    public static final int CI_ABSTRACT = 0x8;
    public static final int CI_SYNTHETIC = 0x10;
    public static final int CI_ANNOTATION = 0x20;
    public static final int CI_ENUM = 0x40;
    public static final int CI_ATTRIBUTES = 0x80;
    public static final int CI_ERROR = 0x100;
    public static final int CI_INITIALIZED = 0x200;
    public static final int CI_FINALIZABLE = 0x400;

    public static final int CI_ERROR_TYPE_NONE = 0x0;
    public static final int CI_ERROR_TYPE_NO_CLASS_DEF_FOUND = 0x1;
    public static final int CI_ERROR_TYPE_ILLEGAL_ACCESS = 0x2;
    public static final int CI_ERROR_TYPE_INCOMPATIBLE_CLASS_CHANGE = 0x3;
    
    public static final int FI_ACCESS_MASK = 0x3;
    public static final int FI_PUBLIC = 0x1;
    public static final int FI_PRIVATE = 0x2;
    public static final int FI_PROTECTED = 0x3;
    public static final int FI_STATIC = 0x4;
    public static final int FI_FINAL = 0x8;
    public static final int FI_VOLATILE = 0x10;
    public static final int FI_TRANSIENT = 0x20;
    public static final int FI_SYNTHETIC = 0x40;
    public static final int FI_ENUM = 0x80;
    public static final int FI_ATTRIBUTES = 0x100;
    
    public static final int MI_ACCESS_MASK = 0x3;
    public static final int MI_PUBLIC = 0x1;
    public static final int MI_PRIVATE = 0x2;
    public static final int MI_PROTECTED = 0x3;
    public static final int MI_STATIC = 0x4;
    public static final int MI_FINAL = 0x8;
    public static final int MI_SYNCHRONIZED = 0x10;
    public static final int MI_BRIDGE = 0x20;
    public static final int MI_VARARGS = 0x40;
    public static final int MI_NATIVE = 0x80;
    public static final int MI_ABSTRACT = 0x100;
    public static final int MI_STRICT = 0x200;
    public static final int MI_SYNTHETIC = 0x400;
    public static final int MI_ATTRIBUTES = 0x800;
    public static final int MI_BRO_BRIDGE = 0x1000;
    public static final int MI_BRO_CALLBACK = 0x2000;
    public static final int MI_COMPACT_DESC = 0x4000;
    
    public static final int DESC_B = 1;
    public static final int DESC_C = 2;
    public static final int DESC_D = 3;
    public static final int DESC_F = 4;
    public static final int DESC_I = 5;
    public static final int DESC_J = 6;
    public static final int DESC_S = 7;
    public static final int DESC_Z = 8;
    public static final int DESC_V = 9;
    
    private SootClass sootClass;
    
    private ModuleBuilder mb;
    private Map> trampolines;
    private Set catches;
    /**
     * Contains the class fields of the class being compiled.
     */
    private List classFields;
    /**
     * Contains the instance fields of the class being compiled.
     */
    private List instanceFields;
    
    private StructureType classType;
    private StructureType instanceType;
    
    private final Config config;
    private final MethodCompiler javaMethodCompiler;
    private final BroMethodCompiler bridgeMethodCompiler;
    private final CallbackMethodCompiler callbackMethodCompiler;
    private final NativeMethodCompiler nativeMethodCompiler;
    private final StructMemberMethodCompiler structMemberMethodCompiler;
    private final GlobalValueMethodCompiler globalValueMethodCompiler;
    private final AttributesEncoder attributesEncoder;
    private final TrampolineCompiler trampolineResolver;
    private final ObjCMemberPlugin.MethodCompiler objcMethodCompiler;

    private final ByteArrayOutputStream output = new ByteArrayOutputStream(4 * 1024 * 1024);
    
    public ClassCompiler(Config config) {
        this.config = config;
        this.javaMethodCompiler = new MethodCompiler(config);
        this.bridgeMethodCompiler = new BridgeMethodCompiler(config);
        this.callbackMethodCompiler = new CallbackMethodCompiler(config);
        this.nativeMethodCompiler = new NativeMethodCompiler(config);
        this.structMemberMethodCompiler = new StructMemberMethodCompiler(config);
        this.globalValueMethodCompiler = new GlobalValueMethodCompiler(config);
        this.attributesEncoder = new AttributesEncoder();
        this.trampolineResolver = new TrampolineCompiler(config);
        this.objcMethodCompiler = new ObjCMemberPlugin.MethodCompiler(config);
    }
    
    public boolean mustCompile(Clazz clazz) {
        File oFile = config.getOFile(clazz);
        if (!oFile.exists() || oFile.lastModified() < clazz.lastModified() || oFile.length() == 0) {
            return true;
        }
        
        ClazzInfo ci = clazz.getClazzInfo();
        if (ci == null) {
            return true;
        }
        
        Set dependencies = ci.getAllDependencies();
        for (Dependency dep : dependencies) {
            Clazz depClazz = config.getClazzes().load(dep.getClassName());
            if (depClazz == null) {
                if (dep.getPath() != null) {
                    // depClazz was available the last time clazz was compiled but is now gone
                    return true;
                }
            } else {
                if (dep.getPath() == null) {
                    // depClazz was not available the last time clazz was compiled but is now available
                    return true;
                }
                if (!dep.getPath().equals(depClazz.getPath().getFile().getAbsolutePath())) {
                    // depClazz was located in another place the last time clazz was built
                    return true;
                }
                if (depClazz.isInBootClasspath() != dep.isInBootClasspath()) {
                    // depClazz has moved to/from the bootclasspath since the last time clazz was built
                    return true;
                }
                if (depClazz.lastModified() > oFile.lastModified()) {
                    // depClazz has been changed since the last time clazz was built 
                    return true;
                }
            }
        }
        
        // No class or interface has zero dependencies (we always add java.lang.Object as a dependency)
        // If dependencies is empty it probably means that an error occurred while reading the
        // serialized dependencies. By returning true here in that case the class will be recompiled
        // and the dependencies regenerated.
        return dependencies.isEmpty();
    }
    
    public void compile(Clazz clazz, Executor executor, ClassCompilerListener listener) throws IOException {
        reset();        
        
        Arch arch = config.getArch();
        OS os = config.getOs();

        try {
            config.getLogger().info("Compiling %s (%s %s %s)", clazz, os, arch,
                    config.isDebug() ? "debug" : "release");
            output.reset();
            compile(clazz, output);
        } catch (Throwable t) {
            if (t instanceof IOException) {
                throw (IOException) t;
            }
            if (t instanceof RuntimeException) {
                throw (RuntimeException) t;
            }
            throw new RuntimeException(t);
        }

        List cCode = new ArrayList<>();
        cCode.addAll(bridgeMethodCompiler.getCWrapperFunctions());
        cCode.addAll(callbackMethodCompiler.getCWrapperFunctions());
        
        scheduleMachineCodeGeneration(executor, listener, config, clazz, output.toByteArray(), cCode);
    }

    private static void scheduleMachineCodeGeneration(Executor executor, final ClassCompilerListener listener,
            final Config config, final Clazz clazz, final byte[] llData, final List cCode) {
        
        Runnable task = new Runnable() {
            @Override
            public void run() {
                try {
                    generateMachineCode(config, clazz, llData, cCode);
                    listener.success(clazz);
                } catch (Throwable t) {
                    listener.failure(clazz, t);
                }
            }
        };
        
        try {
            executor.execute(task);
        } catch (RejectedExecutionException e) {
            if (!(executor instanceof ExecutorService) || !((ExecutorService) executor).isShutdown()) {
                // The task was rejected, probably because all workers are busy. Run it in this thread instead.
                task.run();
            }
        }
    }
    
    private static void generateMachineCode(Config config, Clazz clazz, byte[] llData, List cCode) throws IOException {

        if (config.isDumpIntermediates()) {
            File llFile = config.getLlFile(clazz);
            llFile.getParentFile().mkdirs();
            FileUtils.writeByteArrayToFile(llFile, llData);
            File cFile = config.getCFile(clazz);
            if (cCode.isEmpty()) {
                cFile.delete();
            } else {
                FileUtils.writeLines(cFile, "ascii", cCode);
            }
        }

        File oFile = config.getOFile(clazz);
        try (Context context = new Context()) {
            try (Module module = Module.parseIR(context, llData, clazz.getClassName())) {
                
                if (!cCode.isEmpty()) {
                    int size = 0;
                    for (String s : cCode) {
                        size += s.length();
                    }
                    StringBuilder sb = new StringBuilder(size);
                    for (String s : cCode) {
                        sb.append(s);
                    }
                    try (Module m2 = Module.parseClangString(context, sb.toString(), clazz.getClassName() + ".c", config.getClangTriple())) {
                        module.link(m2);
                        for (org.robovm.llvm.Function f1 : m2.getFunctions()) {
                            String name = f1.getName();
                            org.robovm.llvm.Function f2 = module.getFunctionByName(name);
                            if (Symbols.isBridgeCSymbol(name) || Symbols.isCallbackCSymbol(name) || Symbols.isCallbackInnerCSymbol(name)) {
                                f2.setLinkage(org.robovm.llvm.binding.Linkage.PrivateLinkage);
                                if (Symbols.isCallbackInnerCSymbol(name)) {
                                    // TODO: We should also always inline the bridge functions but for some reason
                                    // that makes the RoboVM tests hang indefinitely.
                                    f2.removeAttribute(Attribute.NoInlineAttribute);
                                    f2.addAttribute(Attribute.AlwaysInlineAttribute);
                                }
                            }
                        }
                    }
                }
                
                try (PassManager passManager = createPassManager(config)) {
                    passManager.run(module);
                }

                if (config.isDumpIntermediates()) {
                    File bcFile = config.getBcFile(clazz);
                    bcFile.getParentFile().mkdirs();
                    module.writeBitcode(bcFile);
                }

                String triple = config.getTriple();
                Target target = Target.lookupTarget(triple);
                try (TargetMachine targetMachine = target.createTargetMachine(triple,
                        config.getArch().getLlvmCpu(), null, 
                        config.isDebug()? CodeGenOptLevel.CodeGenLevelNone: null,
                        RelocMode.RelocPIC, null)) {
                    targetMachine.setAsmVerbosityDefault(true);
                    targetMachine.setFunctionSections(true);
                    targetMachine.setDataSections(true);
                    targetMachine.getOptions().setNoFramePointerElim(true);
                    targetMachine.getOptions().setPositionIndependentExecutable(!config.isDebug()); // NOTE: Doesn't have any effect on x86. See #503.

                    ByteArrayOutputStream output = new ByteArrayOutputStream(256 * 1024);
                    targetMachine.emit(module, output, CodeGenFileType.AssemblyFile);

                    byte[] asm = output.toByteArray();
                    output.reset();
                    patchAsmWithFunctionSizes(config, clazz, new ByteArrayInputStream(asm), output);
                    asm = output.toByteArray();

                    if (config.isDumpIntermediates()) {
                        File sFile = config.getSFile(clazz);
                        sFile.getParentFile().mkdirs();
                        FileUtils.writeByteArrayToFile(sFile, asm);
                    }

                    oFile.getParentFile().mkdirs();
                    ByteArrayOutputStream oFileBytes = new ByteArrayOutputStream();
                    targetMachine.assemble(asm, clazz.getClassName(), oFileBytes);                                                                                               
                    new HfsCompressor().compress(oFile, oFileBytes.toByteArray(), config);

                    ModuleBuilder linesMb;
                    ModuleBuilder debugInfoMb = null;
                    try (ObjectFile objectFile = ObjectFile.load(oFile)) {
                        // notify plugins
                        for (CompilerPlugin plugin : config.getCompilerPlugins()) {
                            plugin.afterObjectFile(config, clazz, oFile, objectFile);
                        }

                        /*
                         * Read out line number info from the .o file if any and
                         * assemble into a separate .o file.
                         */
                        linesMb = buildLineNumberData(config, clazz, objectFile);

                        /*
                         * read out debug info binary data amd assemble into a separate .o file
                         */
                        if (config.isDebug()) {
                            debugInfoMb = buildDebugInfoData(config, clazz, objectFile);
                        }
                    }

                    if (linesMb != null) {
                        File linesLlFile = config.isDumpIntermediates() ? config.getLinesLlFile(clazz) : null;
                        File linesOFile = config.getLinesOFile(clazz);
                        createObjectFileFromData(config, context, targetMachine, linesMb, clazz.getClassName() + ".lines",
                                linesLlFile, linesOFile);
                    } else {
                        // Make sure there's no stale lines.o file lingering
                        File linesOFile = config.getLinesOFile(clazz);
                        if (linesOFile.exists()) {
                            linesOFile.delete();
                        }
                    }

                    if (debugInfoMb != null) {
                        File debugInfoLlFile = config.isDumpIntermediates() ? config.getDebugInfoLlFile(clazz) : null;
                        File debugInfoOFile = config.getDebugInfoOFile(clazz);
                        createObjectFileFromData(config, context, targetMachine, debugInfoMb, clazz.getClassName() + ".debuginfo",
                                debugInfoLlFile, debugInfoOFile);
                    } else {
                        // Make sure there's no stale debuginfo.o file lingering
                        File debugInfoOFile = config.getDebugInfoOFile(clazz);
                        if (debugInfoOFile.exists()) {
                            debugInfoOFile.delete();
                        }
                    }
                }
            }
        } catch (Throwable t) {
            if (oFile.exists()) {
                oFile.delete();
            }
            if (t instanceof IOException) {
                throw (IOException) t;
            }
            if (t instanceof RuntimeException) {
                throw (RuntimeException) t;
            }
            if (t instanceof Error) {
                throw (Error) t;
            }
            throw new CompilerException(t);
        }
    }

    private static void createObjectFileFromData(Config config, Context context, TargetMachine targetMachine, ModuleBuilder mb,
                                                 String dataName, File llFile, File oFile) throws IOException, InterruptedException {
        byte[] data = mb.build().toString().getBytes(StandardCharsets.UTF_8);
        if (llFile != null) {
            llFile.getParentFile().mkdirs();
            FileUtils.writeByteArrayToFile(llFile, data);
        }
        try (Module module = Module.parseIR(context, data, dataName)) {
            ByteArrayOutputStream bytes = new ByteArrayOutputStream();
            targetMachine.emit(module, bytes, CodeGenFileType.ObjectFile);
            new HfsCompressor().compress(oFile, bytes.toByteArray(), config);
        }

    }

    private static ModuleBuilder buildLineNumberData(Config config, Clazz clazz, ObjectFile objectFile) {
        ModuleBuilder linesMb = null;
        String symbolPrefix = config.getOs().getFamily() == OS.Family.darwin ? "_" : "";
        symbolPrefix += Symbols.EXTERNAL_SYMBOL_PREFIX;

        for (Symbol symbol : objectFile.getSymbols()) {
            if (symbol.getSize() > 0 && symbol.getName().startsWith(symbolPrefix)) {
                List lineInfos = objectFile.getLineInfos(symbol);
                if (!lineInfos.isEmpty()) {
                    Collections.sort(lineInfos, new Comparator() {
                        public int compare(LineInfo o1, LineInfo o2) {
                            return Long.compare(o1.getAddress(), o2.getAddress());
                        }
                    });

                    // The base address of the method which will be used to calculate offsets into the method
                    long baseAddress = symbol.getAddress();
                    // The first line number in the method. All other line numbers in the table will be deltas against this.
                    int firstLineNumber = lineInfos.get(0).getLineNumber();
                    // Calculate the max address and line number offsets
                    long maxAddressOffset = 0;
                    long maxLineOffset = 0;
                    for (LineInfo lineInfo : lineInfos) {
                        maxAddressOffset = Math.max(maxAddressOffset, lineInfo.getAddress() - baseAddress);
                        maxLineOffset = Math.max(maxLineOffset, lineInfo.getLineNumber() - firstLineNumber);
                    }

                    // Calculate the number of bytes needed to represent the highest offsets.
                    // Either 1, 2 or 4 bytes will be used.
                    int addressOffsetSize = (maxAddressOffset & ~0xff) == 0 ? 1 : ((maxAddressOffset & ~0xffff) == 0 ? 2 : 4);
                    int lineOffsetSize = (maxLineOffset & ~0xff) == 0 ? 1 : ((maxLineOffset & ~0xffff) == 0 ? 2 : 4);

                    // The size of the address offsets table. We skip the first LineInfo as its offset is always 0.
                    int addressOffsetTableSize = addressOffsetSize * (lineInfos.size() - 1);
                    // Pad size of address offset table to make sure line offsets are aligned properly
                    int addressOffsetPadding = (lineOffsetSize - (addressOffsetTableSize & (lineOffsetSize - 1))) & (lineOffsetSize - 1);
                    addressOffsetTableSize += addressOffsetPadding;

                    // The first 32 bits of the line number info contains the number of line numbers
                    // minus the first. The 4 most significant bits are used to store the number of
                    // bytes needed by each entry in each table.
                    int flags = 0;
                    flags = addressOffsetSize - 1;
                    flags <<= 2;
                    flags |= lineOffsetSize - 1;
                    flags <<= 28;
                    flags |= (lineInfos.size() - 1) & 0x0fffffff;

                    StructureConstantBuilder builder = new StructureConstantBuilder();
                    builder
                            .add(new IntegerConstant(flags))
                            .add(new IntegerConstant(firstLineNumber));

                    for (LineInfo lineInfo : lineInfos.subList(1, lineInfos.size())) {
                        if (addressOffsetSize == 1) {
                            builder.add(new IntegerConstant((byte) (lineInfo.getAddress() - baseAddress)));
                        } else if (addressOffsetSize == 2) {
                            builder.add(new IntegerConstant((short) (lineInfo.getAddress() - baseAddress)));
                        } else {
                            builder.add(new IntegerConstant((int) (lineInfo.getAddress() - baseAddress)));
                        }
                    }

                    // Padding
                    for (int i = 0; i < addressOffsetPadding; i++) {
                        builder.add(new IntegerConstant((byte) 0));
                    }

                    for (LineInfo lineInfo : lineInfos.subList(1, lineInfos.size())) {
                        if (lineOffsetSize == 1) {
                            builder.add(new IntegerConstant((byte) (lineInfo.getLineNumber() - firstLineNumber)));
                        } else if (lineOffsetSize == 2) {
                            builder.add(new IntegerConstant((short) (lineInfo.getLineNumber() - firstLineNumber)));
                        } else {
                            builder.add(new IntegerConstant((int) (lineInfo.getLineNumber() - firstLineNumber)));
                        }
                    }

                    // Extract the method's name and descriptor from the symbol and
                    // build the linetable symbol name.
                    String methodName = symbol.getName().substring(symbol.getName().lastIndexOf('.') + 1);
                    methodName = methodName.substring(0, methodName.indexOf('('));
                    String methodDesc = symbol.getName().substring(symbol.getName().lastIndexOf('('));
                    String linetableSymbol = Symbols.linetableSymbol(clazz.getInternalName(), methodName, methodDesc);
                    if (linesMb == null) {
                        linesMb = new ModuleBuilder();
                    }
                    linesMb.addGlobal(new Global(linetableSymbol, builder.build(), true));
                }
            }
        }

        // emit bitcode section for line number object file
        if (linesMb != null)
            emitBitcodeSection(config, linesMb);

        return linesMb;
    }

    private static ModuleBuilder buildDebugInfoData(Config config, Clazz clazz, ObjectFile objectFile) throws IOException {
        ModuleBuilder debugInfoMb = null;

        DebuggerDebugObjectFileInfo.RawData debugInfo = clazz.getAttachment(DebuggerDebugObjectFileInfo.RawData.class);
        if (debugInfo != null) {
            // read all binary data to file
            byte[] debugDataBytes = DebuggerDebugObjectFileInfo.dumpDebugInfo(debugInfo);
            String debugInfoSymbol = Symbols.debugInfoSymbol(clazz.getInternalName());
            debugInfoMb = new ModuleBuilder();
            Global debugInfoSymbolGlobal = new Global(debugInfoSymbol, new ByteArrayConstant(debugDataBytes), true);
            debugInfoMb.addGlobal(debugInfoSymbolGlobal);

            // list to keep weak values (such as debugger ones) to be added to llvm.used
            List usedGlobalValues = new ArrayList<>();
            usedGlobalValues.add(new ConstantBitcast(debugInfoSymbolGlobal.ref(), Type.I8_PTR));

            // also need to add reference to @llvm.used otherwise linker will drop out this data as not used
            // dkimitsa: always add it as array even if there is only one element as llvm.used is array of pointers
            // and debug version of llvm will crash on assertion while validating type
            Global usedGlobal;
            ArrayConstant usedValuesArr = new ArrayConstant(new ArrayType(usedGlobalValues.size(), Type.I8_PTR),
                    usedGlobalValues.toArray(new Value[0]));
            usedGlobal = new Global("llvm.used", Linkage.appending, usedValuesArr, false, "llvm.metadata");
            debugInfoMb.addGlobal(usedGlobal);

            if (config.isDumpIntermediates()) {
                File f = new File(config.getDebugInfoOFile(clazz).getAbsolutePath() + ".dump");
                DebuggerDebugObjectFileInfo.dumpDebugInfoAsText(debugInfo, f);
            }
        }

        return debugInfoMb;
    }

    private static PassManager createPassManager(Config config) {
        PassManager passManager = new PassManager();
        
        if (config.isDebug()) {
            // Minimal passes. Just inline functions marked 'alwaysinline'.
            passManager.addAlwaysInlinerPass();
        } else {
            try (PassManagerBuilder builder = new PassManagerBuilder()) {
                builder.setSetOptLevel(2);
                builder.setDisableTailCalls(true);
                builder.useAlwaysInliner(true);
                builder.populateModulePassManager(passManager);
            }
        }
        
        return passManager;
    }
    
    private static void patchAsmWithFunctionSizes(Config config, Clazz clazz, InputStream inStream, OutputStream outStream) throws IOException {
        String labelPrefix = config.getOs().getFamily() == OS.Family.darwin ? "_" : "";
        String localLabelPrefix = config.getOs().getFamily() == OS.Family.darwin ? "L" : ".L";
        
        Set functionNames = new HashSet();
        for (SootMethod method : clazz.getSootClass().getMethods()) {
            if (!method.isAbstract()) {
                String name = labelPrefix + Symbols.methodSymbol(method);
                functionNames.add(name);
            }
        }
        
        String infoStructLabel = labelPrefix + Symbols.infoStructSymbol(clazz.getInternalName());
        Pattern methodImplPattern = Pattern.compile("\\s*\\.(?:quad|long)\\s+\"?(" 
                + Pattern.quote(labelPrefix + Symbols.methodSymbolPrefix(clazz.getInternalName())) 
                + "[^\\s\"]+)\"?.*");
        
        BufferedReader in = null;
        BufferedWriter out = null;
        try {
            in = new BufferedReader(new InputStreamReader(inStream, StandardCharsets.UTF_8));
            out = new BufferedWriter(new OutputStreamWriter(outStream, StandardCharsets.UTF_8));
            String line = null;
            String currentFunction = null;
            while ((line = in.readLine()) != null) {
                if (currentFunction == null) {
                    out.write(line);
                    out.write('\n');
                    int colon = line.indexOf(':');
                    if (colon != -1) {
                        String label = line.substring(0, colon);
                        if (label.startsWith("\"") && label.endsWith("\"")) {
                            label = label.substring(1, label.length() - 1);
                        }
                        if (functionNames.contains(label)) {
                            currentFunction = label;
                        } else if (label.equals(infoStructLabel)) {
                            break;
                        }
                    }
                } else if (line.trim().equals(".cfi_endproc") || line.trim().startsWith(".section") || line.trim().startsWith(".globl")) {
                    out.write("\"");
                    out.write(localLabelPrefix);
                    out.write(currentFunction);
                    out.write("_end\":\n\n");
                    currentFunction = null;
                    out.write(line);
                    out.write('\n');
                } else {
                    out.write(line);
                    out.write('\n');
                }
            }
            
            while ((line = in.readLine()) != null) {
                out.write(line);
                out.write('\n');
                Matcher matcher = methodImplPattern.matcher(line);
                if (matcher.matches()) {
                    String functionName = matcher.group(1);
                    if (functionNames.contains(functionName)) {
                        line = in.readLine();
                        if (line.contains(String.valueOf(DUMMY_METHOD_SIZE))) {
                            out.write("\t.long\t");
                            out.write("\"" + localLabelPrefix + functionName + "_end\" - \"" + functionName + "\"");
                            out.write('\n');
                        } else {
                            out.write(line);
                            out.write('\n');
                        }
                    }
                }
            }
        } finally {
            IOUtils.closeQuietly(in);
            IOUtils.closeQuietly(out);
        }
    }
    
    private void reset() {
        output.reset();
        sootClass = null;
        mb = null;
        trampolines = null;
        catches = null;
        classFields = null;
        instanceFields = null;
        classType = null;
        instanceType = null;
    }
    
    private void compile(Clazz clazz, OutputStream out) throws IOException {
        ClazzInfo ci = clazz.resetClazzInfo();

        mb = new ModuleBuilder();

        for (CompilerPlugin compilerPlugin : config.getCompilerPlugins()) {
            compilerPlugin.helloClass(config, clazz);
        }

        for (CompilerPlugin compilerPlugin : config.getCompilerPlugins()) {
            compilerPlugin.beforeClass(config, clazz, mb);
        }

        // dkimitsa: intentionally moved reset section after compiler plugin beforeClass invocation.
        // one reason for this is structMemberMethodCompiler. it tries to resolve structure type
        // and find out all marshallers but some of them (like ObjCBlock ones) are generated only
        // in beforeClass. so to have all marshallers and generated classes in place, reset section
        // is moved here. it shall have no affect as these compilers are not used by plugins itself
        javaMethodCompiler.reset(clazz);
        bridgeMethodCompiler.reset(clazz);
        callbackMethodCompiler.reset(clazz);
        nativeMethodCompiler.reset(clazz);
        structMemberMethodCompiler.reset(clazz);
        globalValueMethodCompiler.reset(clazz);


        sootClass = clazz.getSootClass();
        trampolines = new HashMap<>();
        catches = new HashSet();
        classFields = getClassFields(config.getOs(), config.getArch(),sootClass);
        instanceFields = getInstanceFields(config.getOs(), config.getArch(),sootClass);
        classType = getClassType(mb, config.getOs(), config.getArch(),sootClass);
        instanceType = getInstanceType(config.getOs(), config.getArch(),sootClass);
        
        attributesEncoder.encode(mb, sootClass);
        
        // Add a  method if the class has ConstantValueTags but no .
        // This has to be done before createInfoStruct() is called otherwise the
        // ClassInfoHeader->initializer value will become NULL and constant static fields
        // will never be initialized.
        if (!sootClass.declaresMethodByName("") && hasConstantValueTags(classFields)) {
            SootMethod clinit = new SootMethod("", Collections.EMPTY_LIST, VoidType.v(), Modifier.STATIC);
            JimpleBody body = Jimple.v().newBody(clinit);
            clinit.setActiveBody(body);
            body.getUnits().add(new JReturnVoidStmt());
            this.sootClass.addMethod(clinit);
        }

        if (isStruct(sootClass)) {
            SootMethod _sizeOf = new SootMethod("_sizeOf", Collections.EMPTY_LIST, IntType.v(), Modifier.PROTECTED | Modifier.NATIVE);
            sootClass.addMethod(_sizeOf);
            SootMethod sizeOf = new SootMethod("sizeOf", Collections.EMPTY_LIST, IntType.v(), Modifier.PUBLIC | Modifier.STATIC | Modifier.NATIVE);
            sootClass.addMethod(sizeOf);

            // method that will provide meta flags for runtime to help finding out if stret is required
            SootMethod stretMeta = new SootMethod(STRUCT_ATTRIBUTES_METHOD, Collections.EMPTY_LIST, IntType.v(), Modifier.PUBLIC | Modifier.STATIC | Modifier.NATIVE);
            sootClass.addMethod(stretMeta);
            // method that returns offset of struct member
            SootMethod offset = new SootMethod("offsetOf", Collections.singletonList(IntType.v()), IntType.v(), Modifier.PUBLIC | Modifier.STATIC | Modifier.NATIVE);
            sootClass.addMethod(offset);
        }
        
        mb.addInclude(getClass().getClassLoader().getResource(String.format("header-%s-%s.ll", config.getOs().getFamily(), config.getArch().getCpuArch())));
        mb.addInclude(getClass().getClassLoader().getResource("header.ll"));

        mb.addFunction(createLdcClass());
        mb.addFunction(createLdcClassWrapper());
        Function allocator = createAllocator();
        mb.addFunction(allocator);
        mb.addFunction(createClassInitWrapperFunction(allocator.ref()));
        
        for (SootField f : sootClass.getFields()) {
            Function getter = createFieldGetter(f, classFields, classType, instanceFields, instanceType);
            Function setter = createFieldSetter(f, classFields, classType, instanceFields, instanceType);
            mb.addFunction(getter);
            mb.addFunction(setter);
            // generate cinit wrapper even for private fields
            // as with java11 JEP181 there will be no access$$$ wrappers
            // to access these fields from inner classes
            if (f.isStatic() /* && !f.isPrivate() */ ) {
                mb.addFunction(createClassInitWrapperFunction(getter.ref()));
                if (!f.isFinal()) {
                    mb.addFunction(createClassInitWrapperFunction(setter.ref()));
                }
            }
        }

        // After this point no changes to methods/fields may be done by CompilerPlugins.
        ci.initClassInfo();

        // when Java18 used as compiler JDK-8272564 changes will be applied this will affect invocation of
        // java.lang.Object methods on interface receivers.
        // as per changes invokevirtual is replaced with invokeinterface and these methods
        // has to be resolved as per https://docs.oracle.com/javase/specs/jvms/se17/html/jvms-5.html#jvms-5.4.3.4 item 3
        // in RoboVM case this means that final methods of "java.lang.Object" has to be available for lookup,
        // thus changes added to include [lookup] wrappers for final methods in this case
        // otherwise it will fail during linking with message(s):
        // Undefined symbols for architecture arm64:"[j]java.lang.Object.notifyAll()V[lookup]"
        boolean isJavaLangObject = sootClass.getName().equals("java.lang.Object");
        for (SootMethod method : sootClass.getMethods()) {
            
            for (CompilerPlugin compilerPlugin : config.getCompilerPlugins()) {
                compilerPlugin.beforeMethod(config, clazz, method, mb);
            }
            
            String name = method.getName();
            Function function = null;
            if (hasBridgeAnnotation(method)) {
                // javac generates JVM synthetic bridge methods for covariant return
                // and copies @Bridge annotations as well. Don't try to compile bridge methods
                // as these are not a subject for RoboVM bridge compiler and it will fail on them
                if (!isJvmSyntheticBridgeMethod(method))
                    function = bridgeMethod(method);
                else
                    function = method(method);
            } else if (hasGlobalValueAnnotation(method)) {
                function = globalValueMethod(method);
            } else if (isStruct(sootClass) && ("_sizeOf".equals(name) || "sizeOf".equals(name) ||
                    "offsetOf".equals(name) || STRUCT_ATTRIBUTES_METHOD.equals(name) || hasStructMemberAnnotation(method))) {
                function = structMember(method);
            } else if (method.isNative()) {
                function = nativeMethod(method);
            } else if (objcMethodCompiler.willCompile(method)) {
                function = objcPublishMethod(method);
            } else if (!method.isAbstract()) {
                function = method(method);
            }
            if (hasCallbackAnnotation(method)) {
                callbackMethod(method);
            }
            if (!name.equals("") && !name.equals("") 
                    && !method.isPrivate() && !method.isStatic() 
                    && (isJavaLangObject || !Modifier.isFinal(method.getModifiers()))
                    && !Modifier.isFinal(sootClass.getModifiers())) {
                
                createLookupFunction(method);
            }
            if (method.isStatic() && !name.equals("")) {
                String fnName = method.isSynchronized() 
                        ? Symbols.synchronizedWrapperSymbol(method) 
                        : Symbols.methodSymbol(method);
                FunctionRef fn = new FunctionRef(fnName, getFunctionType(method));
                mb.addFunction(createClassInitWrapperFunction(fn));
            }
            
            for (CompilerPlugin compilerPlugin : config.getCompilerPlugins()) {
                if (function != null) {
                    compilerPlugin.afterMethod(config, clazz, method, mb, function);
                }
            }
        }
        
        for (Trampoline trampoline : trampolines.keySet()) {
            Set deps = new HashSet();
            Set> mDeps = new HashSet<>();
            trampolineResolver.compile(mb, clazz, trampoline, deps, mDeps);
            for (SootMethod m : trampolines.get(trampoline)) {
                MethodInfo mi = ci.getMethod(m.getName(), getDescriptor(m));
                mi.addClassDependencies(deps, false);
                mi.addInvokeMethodDependencies(mDeps, false);
            }
        }
        
        /*
         * Add method dependencies from overriding methods to the overridden
         * super method(s). These will be reversed by the DependencyGraph to
         * create edges from the super/interface method to the overriding
         * method.
         */
        Map> overriddenMethods = getOverriddenMethods(this.sootClass);
        for (SootMethod from : overriddenMethods.keySet()) {
            MethodInfo mi = ci.getMethod(from.getName(), getDescriptor(from));
            for (SootMethod to : overriddenMethods.get(from)) {
                mi.addSuperMethodDependency(getInternalName(to.getDeclaringClass()), to.getName(), getDescriptor(to), false);
            }
        }
        
        /*
         * Edge case. A method in a superclass might satisfy an interface method
         * in the interfaces implemented by this class. See e.g. the abstract
         * class HashMap$HashIterator which doesn't implement Iterator but has
         * the hasNext() and other methods. We add a dependency from the current
         * class to the super method to ensure it's included if the current
         * class is linked in.
         */
        if (sootClass.hasSuperclass()) {
            for (SootClass interfaze : getImmediateInterfaces(sootClass)) {
                for (SootMethod m : interfaze.getMethods()) {
                    if (!m.isStatic()) {
                        try {
                            this.sootClass.getMethod(m.getName(), m.getParameterTypes());
                        } catch (RuntimeException e) {
                            /*
                             * Not found. Find the implementation in
                             * superclasses.
                             */
                            SootMethod superMethod = null;
                            for (SootClass sc = sootClass.getSuperclass(); sc.hasSuperclass(); sc = sc.getSuperclass()) {
                                try {
                                    SootMethod candidate = sc.getMethod(m.getName(), m.getParameterTypes());
                                    if (!candidate.isStatic()) {
                                        superMethod = candidate;
                                        break;
                                    }
                                } catch (RuntimeException e2) {
                                    // Not found.
                                }
                            }

                            if (superMethod != null) {
                                ci.addSuperMethodDependency(getInternalName(superMethod.getDeclaringClass()),
                                        superMethod.getName(), getDescriptor(superMethod), false);
                            }
                        }
                    }
                }
            }
        }

        Global classInfoStruct = null;
        try {
            if (!sootClass.isInterface()) {
                config.getVTableCache().get(sootClass);
            }
            classInfoStruct = new Global(Symbols.infoStructSymbol(clazz.getInternalName()), Linkage.weak, createClassInfoStruct());
        } catch (IllegalArgumentException e) {
            // VTable throws this if any of the superclasses of the class is actually an interface.
            // Shouldn't happen frequently but the DRLVM test suite has some tests for this.
            // The Linker will take care of making sure the class cannot be loaded at runtime.
            classInfoStruct = new Global(Symbols.infoStructSymbol(clazz.getInternalName()), I8_PTR, true);
        }
        mb.addGlobal(classInfoStruct);
        /*
         * Emit an internal i8* alias for the info struct which MethodCompiler
         * can use when referencing this info struct in exception landing pads
         * in methods in the same class. See #1007.
         */
        mb.addAlias(new Alias(classInfoStruct.getName() + "_i8ptr", Linkage._private, new ConstantBitcast(
                classInfoStruct.ref(), I8_PTR)));
        
        Function infoFn = FunctionBuilder.infoStruct(sootClass);
        infoFn.add(new Ret(new ConstantBitcast(classInfoStruct.ref(), I8_PTR_PTR)));
        mb.addFunction(infoFn);
        
        for (CompilerPlugin compilerPlugin : config.getCompilerPlugins()) {
            compilerPlugin.afterClass(config, clazz, mb);
        }

        // emit bitcode section for class
        emitBitcodeSection(config, mb);

        OutputStreamWriter writer = new OutputStreamWriter(out, StandardCharsets.UTF_8);
        mb.build().write(writer);
        writer.flush();

        ci.setCatchNames(catches);
        
        ci.addClassDependency("java/lang/Object", false); // Make sure no class or interface has zero dependencies
        if (sootClass.hasSuperclass() && !sootClass.isInterface()) {
            ci.addClassDependency(getInternalName(sootClass.getSuperclass()), false);
        }
        for (SootClass iface : sootClass.getInterfaces()) {
            ci.addClassDependency(getInternalName(iface), false);
        }
        for (SootField f : sootClass.getFields()) {
            addClassDependencyIfNeeded(clazz, f.getType(), false);
        }
        for (SootMethod m : sootClass.getMethods()) {
            MethodInfo mi = ci.getMethod(m.getName(), getDescriptor(m));
            addClassDependencyIfNeeded(clazz, mi, m.getReturnType(), false);
            @SuppressWarnings("unchecked")
            List paramTypes = (List) m.getParameterTypes();
            for (soot.Type type : paramTypes) {
                addClassDependencyIfNeeded(clazz, mi, type, false);
            }
        }
        ci.addClassDependencies(attributesEncoder.getDependencies(), false);
        ci.addClassDependencies(catches, false);
        
        for (Trampoline t : trampolines.keySet()) {
            if (t instanceof Checkcast) {
                ci.addCheckcast(t.getTarget());
            } else if (t instanceof Instanceof) {
                ci.addInstanceof(t.getTarget());
            } else if (t instanceof Invokevirtual || t instanceof Invokeinterface) {
                ci.addInvoke(t.getTarget() + "." + ((Invoke) t).getMethodName() + ((Invoke) t).getMethodDesc());
            }
        }
        clazz.saveClazzInfo();
    }

    /**
     * adds `__LLVM,__asm` section to module builder to mark it as object without bitcode build from assembly.
     */
    static void emitBitcodeSection(Config config, ModuleBuilder mb) {
        // for release build add bitcode section
        if (mb != null && config.shouldEmitBitcode()) {
            // can't just add section, adding zero constants to create one
            mb.addGlobal(new Global("robovm.bitcode", Linkage.appending, new IntegerConstant(0), true, "__LLVM,__asm"));
        }
    }

    private boolean isJvmSyntheticBridgeMethod(SootMethod method) {
        final int BRIDGE = 0x00000040;
        final int SYNTHETIC = 0x00001000;

        return !method.isAbstract() &&
                !method.isNative() &&
                (method.getModifiers() & BRIDGE) == BRIDGE &&
                (method.getModifiers() & SYNTHETIC) == SYNTHETIC;
    }

    private static void addClassDependencyIfNeeded(Clazz clazz, soot.Type type, boolean weak) {
        if (type instanceof RefLikeType) {
            addClassDependencyIfNeeded(clazz, getDescriptor(type), weak);
        }
    }

    private static void addClassDependencyIfNeeded(Clazz clazz, String desc, boolean weak) {
        if (!isPrimitive(desc) && (!isArray(desc) || !isPrimitiveBaseType(desc))) {
            String internalName = isArray(desc) ? getBaseType(desc) : getInternalNameFromDescriptor(desc);
            if (!clazz.getInternalName().equals(internalName)) {
                clazz.getClazzInfo().addClassDependency(internalName, weak);
            }
        }
    }

    private static void addClassDependencyIfNeeded(Clazz clazz, MethodInfo mi, soot.Type type, boolean weak) {
        if (type instanceof RefLikeType) {
            addClassDependencyIfNeeded(clazz, mi, getDescriptor(type), weak);
        }
    }
    
    private static void addClassDependencyIfNeeded(Clazz clazz, MethodInfo mi, String desc, boolean weak) {
        if (!isPrimitive(desc) && (!isArray(desc) || !isPrimitiveBaseType(desc))) {
            String internalName = isArray(desc) ? getBaseType(desc) : getInternalNameFromDescriptor(desc);
            if (!clazz.getInternalName().equals(internalName)) {
                mi.addClassDependency(internalName, weak);
            }
        }
    }

    private Map> getOverriddenMethods(SootClass sc) {
        Map> result = new HashMap<>();
        for (SootMethod m : sc.getMethods()) {
            if (!m.isStatic() && !m.isPrivate() && !m.getName().equals("")) {
                result.put(m, new HashSet());
            }
        }
        findOverriddenSuperMethods(sc, sc, result);
        return result;
    }

    private void findOverriddenSuperMethods(SootClass childClass, SootClass sc, Map> result) {
        if (sc != childClass) {
            for (SootMethod superMethod : sc.getMethods()) {
                if (superMethod.isStatic() || superMethod.isPrivate() || superMethod.getName().equals("")) {
                    // Not overridable
                    continue;
                }
                if (!superMethod.isPublic() && !superMethod.isProtected()) {
                    // Package private. Must be in same package as the child class.
                    if (!childClass.getPackageName().equals(sc.getPackageName())) {
                        continue;
                    }
                }
    
                for (SootMethod childMethod : result.keySet()) {
                    if (childMethod.getName().equals(superMethod.getName())
                            && childMethod.getParameterTypes().equals(superMethod.getParameterTypes())) {
                        result.get(childMethod).add(superMethod);
                    }
                }
            }
        }
        for (SootClass interfaze : sc.getInterfaces()) {
            findOverriddenSuperMethods(childClass, interfaze, result);
        }
        if (sc.hasSuperclass()) {
            findOverriddenSuperMethods(childClass, sc.getSuperclass(), result);
        }
    }
    
    private Set getImmediateInterfaces(SootClass sc) {
        HashSet result = new HashSet<>();
        result.addAll(sc.getInterfaces());
        for (SootClass interfaze : sc.getInterfaces()) {
            result.addAll(getImmediateInterfaces(interfaze));
        }
        return result;
    }
    
    private void createLookupFunction(SootMethod m) {
        Function function = FunctionBuilder.lookup(m, true);
        mb.addFunction(function);

        Variable reserved0 = function.newVariable(I8_PTR_PTR);
        function.add(new Getelementptr(reserved0, function.getParameterRef(0), 0, 4));
        Variable reserved1 = function.newVariable(I8_PTR_PTR);
        function.add(new Getelementptr(reserved1, function.getParameterRef(0), 0, 5));
        function.add(new Store(getString(m.getName()), reserved0.ref()));
        function.add(new Store(getString(getDescriptor(m)), reserved1.ref()));
        
        if (!sootClass.isInterface()) {
            int vtableIndex = 0;
            try {
                VTable vtable = config.getVTableCache().get(sootClass);
                vtableIndex = vtable.getEntry(m).getIndex();
            } catch (IllegalArgumentException e) {
                // VTable throws this if any of the superclasses of the class is actually an interface.
                // Shouldn't happen frequently but the DRLVM test suite has some tests for this.
                // Use 0 as vtableIndex since this lookup function will never be called anyway.
            }
            Value classPtr = call(function, OBJECT_CLASS, function.getParameterRef(1));
            Value vtablePtr = call(function, CLASS_VITABLE, classPtr);
            Variable funcPtrPtr = function.newVariable(I8_PTR_PTR);
            function.add(new Getelementptr(funcPtrPtr, vtablePtr, 0, 1, vtableIndex));
            Variable funcPtr = function.newVariable(I8_PTR);
            function.add(new Load(funcPtr, funcPtrPtr.ref()));
            Variable f = function.newVariable(function.getType());
            function.add(new Bitcast(f, funcPtr.ref(), f.getType()));
            Value result = tailcall(function, f.ref(), function.getParameterRefs());
            function.add(new Ret(result));
        } else {
            ITable itable = config.getITableCache().get(sootClass);
            ITable.Entry entry = itable.getEntry(m);
            List args = new ArrayList();
            args.add(function.getParameterRef(0));
            args.add(getInfoStruct(function, sootClass));
            args.add(function.getParameterRef(1));
            args.add(new IntegerConstant(entry.getIndex()));
            Value fptr = call(function, BC_LOOKUP_INTERFACE_METHOD_IMPL, args);
            Variable f = function.newVariable(function.getType());
            function.add(new Bitcast(f, fptr, f.getType()));
            Value result = tailcall(function, f.ref(), function.getParameterRefs());
            function.add(new Ret(result));
        }
    }
    
    private Constant createVTableStruct() {
        VTable vtable = config.getVTableCache().get(sootClass);
        String name = Symbols.vtableSymbol(getInternalName(sootClass));
        for (VTable.Entry entry : vtable.getEntries()) {
            FunctionRef fref = entry.getFunctionRef();
            if (fref != null && !mb.hasSymbol(fref.getName())) {
                mb.addFunctionDeclaration(new FunctionDeclaration(fref));
            }
        }
        Global vtableStruct = new Global(name, Linkage._private, vtable.getStruct(), true);
        mb.addGlobal(vtableStruct);
        return new ConstantBitcast(vtableStruct.ref(), I8_PTR);
    }
    
    private Constant createITableStruct() {
        ITable itable = config.getITableCache().get(sootClass);
        String name = Symbols.itableSymbol(getInternalName(sootClass));
        Global itableStruct = new Global(name, Linkage._private, itable.getStruct(), true);
        mb.addGlobal(itableStruct);
        return new ConstantBitcast(itableStruct.ref(), I8_PTR);
    }
    
    private Constant createITablesStruct() {
        if (!sootClass.isInterface()) {
            HashSet interfaces = new HashSet();
            collectInterfaces(sootClass, interfaces);
            List tables = new ArrayList();
            int i = 0;
            for (SootClass ifs : interfaces) {
                ITable itable = config.getITableCache().get(ifs);
                if (itable.size() > 0) {
                    String name = Symbols.itableSymbol(getInternalName(sootClass), i++);
                    String typeInfoName = Symbols.typeInfoSymbol(getInternalName(ifs));
                    if (!mb.hasSymbol(typeInfoName)) {
                        mb.addGlobal(new Global(typeInfoName, Linkage.external, I8_PTR, true));
                    }
                    Global itableStruct = new Global(name, Linkage._private,
                            new StructureConstantBuilder()
                                .add(mb.getGlobalRef(typeInfoName))
                                .add(itable.getStruct(mb, sootClass)).build(), true);
                    mb.addGlobal(itableStruct);
                    tables.add(new ConstantBitcast(itableStruct.ref(), I8_PTR));
                }
            }
            
            if (tables.isEmpty()) {
                return new NullConstant(I8_PTR);
            } else {
                Global itablesStruct = new Global(Symbols.itablesSymbol(getInternalName(sootClass)), Linkage._private,
                        new StructureConstantBuilder()
                            .add(new IntegerConstant((short) tables.size()))
                            .add(tables.get(0)) // cache value must never be null
                            .add(new ArrayConstantBuilder(I8_PTR).add(tables).build())
                            .build());
                mb.addGlobal(itablesStruct);
                return new ConstantBitcast(itablesStruct.ref(), I8_PTR);
            }
            
        } else {
            return new NullConstant(I8_PTR);
        }
    }
    
    private StructureConstant createClassInfoStruct() {
        int flags = 0;
        
        if (Modifier.isPublic(sootClass.getModifiers())) {
            flags |= CI_PUBLIC;
        }
        if (Modifier.isFinal(sootClass.getModifiers())) {
            flags |= CI_FINAL;
        }
        if (Modifier.isInterface(sootClass.getModifiers())) {
            flags |= CI_INTERFACE;
        }
        if (Modifier.isAbstract(sootClass.getModifiers())) {
            flags |= CI_ABSTRACT;
        }
        if ((sootClass.getModifiers() & 0x1000) > 0) {
            flags |= CI_SYNTHETIC;
        }
        if (Modifier.isAnnotation(sootClass.getModifiers())) {
            flags |= CI_ANNOTATION;
        }
        if (Modifier.isEnum(sootClass.getModifiers())) {
            flags |= CI_ENUM;
        }
        if (attributesEncoder.classHasAttributes()) {
            flags |= CI_ATTRIBUTES;
        }
        if (hasFinalizer(sootClass)) {
            flags |= CI_FINALIZABLE;
        }
        
        // Create the ClassInfoHeader structure.
        StructureConstantBuilder header = new StructureConstantBuilder();
        header.add(new NullConstant(I8_PTR)); // Points to the runtime Class struct
        header.add(new IntegerConstant(flags));
        header.add(getString(getInternalName(sootClass)));
        if (sootClass.declaresMethod("", Collections.emptyList(), VoidType.v())) {
            SootMethod method = sootClass.getMethod("", Collections.emptyList(), VoidType.v());
            header.add(new FunctionRef(Symbols.methodSymbol(method), getFunctionType(method)));            
        } else {
            header.add(new NullConstant(I8_PTR));
        }
        mb.addGlobal(new Global(Symbols.typeInfoSymbol(getInternalName(sootClass)), Linkage.external, I8_PTR, true));
        header.add(new GlobalRef(Symbols.typeInfoSymbol(getInternalName(sootClass)), I8_PTR)); // TypeInfo* generated by Linker

        if (!sootClass.isInterface()) {
            header.add(createVTableStruct());
        } else {
            header.add(createITableStruct());
        }
        header.add(createITablesStruct());
        
        header.add(sizeof(classType));
        header.add(sizeof(instanceType));
        if (!instanceFields.isEmpty()) {
            header.add(offsetof(instanceType, 1, 1));
        } else {
            header.add(sizeof(instanceType));
        }
        header.add(new IntegerConstant((short) countReferences(classFields)));
        header.add(new IntegerConstant((short) countReferences(instanceFields)));

        StructureConstantBuilder body = new StructureConstantBuilder();
        body.add(new IntegerConstant((short) sootClass.getInterfaceCount()));
        body.add(new IntegerConstant((short) sootClass.getFieldCount()));
        body.add(new IntegerConstant((short) sootClass.getMethodCount()));
        
        if (!sootClass.isInterface()) {
            body.add(getStringOrNull(sootClass.hasSuperclass() ? getInternalName(sootClass.getSuperclass()) : null));
        }

        if (attributesEncoder.classHasAttributes()) {
            body.add(new ConstantBitcast(attributesEncoder.getClassAttributes().ref(), I8_PTR));
        }
        
        for (SootClass s : sootClass.getInterfaces()) {
            body.add(getString(getInternalName(s)));
        }
        
        for (SootField f : sootClass.getFields()) {
            flags = 0;
            soot.Type t = f.getType();
            if (t instanceof PrimType) {
                if (t.equals(BooleanType.v())) {
                    flags |= DESC_Z;
                } else if (t.equals(ByteType.v())) {
                    flags |= DESC_B;
                } else if (t.equals(ShortType.v())) {
                    flags |= DESC_S;
                } else if (t.equals(CharType.v())) {
                    flags |= DESC_C;
                } else if (t.equals(IntType.v())) {
                    flags |= DESC_I;
                } else if (t.equals(LongType.v())) {
                    flags |= DESC_J;
                } else if (t.equals(FloatType.v())) {
                    flags |= DESC_F;
                } else if (t.equals(DoubleType.v())) {
                    flags |= DESC_D;
                }
                flags <<= 12;
            }
            if (Modifier.isPublic(f.getModifiers())) {
                flags |= FI_PUBLIC;
            } else if (Modifier.isPrivate(f.getModifiers())) {
                flags |= FI_PRIVATE;
            } else if (Modifier.isProtected(f.getModifiers())) {
                flags |= FI_PROTECTED;
            }
            if (Modifier.isStatic(f.getModifiers())) {
                flags |= FI_STATIC;
            }
            if (Modifier.isFinal(f.getModifiers())) {
                flags |= FI_FINAL;
            }
            if (Modifier.isVolatile(f.getModifiers())) {
                flags |= FI_VOLATILE;
            }
            if (Modifier.isTransient(f.getModifiers())) {
                flags |= FI_TRANSIENT;
            }
            if ((f.getModifiers() & 0x1000) > 0) {
                flags |= FI_SYNTHETIC;
            }
            if (Modifier.isEnum(f.getModifiers())) {
                flags |= FI_ENUM;
            }
            if (attributesEncoder.fieldHasAttributes(f)) {
                flags |= FI_ATTRIBUTES;
            }
            body.add(new IntegerConstant((short) flags));
            body.add(getString(f.getName()));
            if (!(t instanceof PrimType)) {
                body.add(getString(getDescriptor(f)));
            }
            if (f.isStatic()) {
                int index = classFields.indexOf(f);
                body.add(offsetof(classType, 1, index, 1));
            } else {
                int index = instanceFields.indexOf(f);
                body.add(offsetof(instanceType, 1, 1 + index, 1));
            }
            if (attributesEncoder.fieldHasAttributes(f)) {
                body.add(new ConstantBitcast(attributesEncoder.getFieldAttributes(f).ref(), I8_PTR));
            }
        }
        
        VTable vtable = !sootClass.isInterface() ? config.getVTableCache().get(sootClass) : null;
        ITable itable = sootClass.isInterface() ? config.getITableCache().get(sootClass) : null;;

        for (SootMethod m : sootClass.getMethods()) {
            soot.Type t = m.getReturnType();
            flags = 0;
            if (Modifier.isPublic(m.getModifiers())) {
                flags |= MI_PUBLIC;
            } else if (Modifier.isPrivate(m.getModifiers())) {
                flags |= MI_PRIVATE;
            } else if (Modifier.isProtected(m.getModifiers())) {
                flags |= MI_PROTECTED;
            }
            if (Modifier.isStatic(m.getModifiers())) {
                flags |= MI_STATIC;
            }
            if (Modifier.isFinal(m.getModifiers())) {
                flags |= MI_FINAL;
            }
            if (Modifier.isSynchronized(m.getModifiers())) {
                flags |= MI_SYNCHRONIZED;
            }
            if ((m.getModifiers() & 0x0040) > 0) {
                flags |= MI_BRIDGE;
            }
            if ((m.getModifiers() & 0x0080) > 0) {
                flags |= MI_VARARGS;
            }
            if (Modifier.isNative(m.getModifiers())) {
                if (!isStruct(sootClass) && !hasStructMemberAnnotation(m)) {
                    flags |= MI_NATIVE;
                }
            }
            if (Modifier.isAbstract(m.getModifiers())) {
                flags |= MI_ABSTRACT;
            }
            if (Modifier.isStrictFP(m.getModifiers())) {
                flags |= MI_STRICT;
            }
            if ((m.getModifiers() & 0x1000) > 0) {
                flags |= MI_SYNTHETIC;
            }
            if (attributesEncoder.methodHasAttributes(m)) {
                flags |= MI_ATTRIBUTES;
            }
            if ((hasBridgeAnnotation(m) && !isJvmSyntheticBridgeMethod(m)) || hasGlobalValueAnnotation(m)) {
                flags |= MI_BRO_BRIDGE;
            }
            if (hasCallbackAnnotation(m)) {
                flags |= MI_BRO_CALLBACK;
            }
            if ((t instanceof PrimType || t == VoidType.v()) && m.getParameterCount() == 0) {
                flags |= MI_COMPACT_DESC;
            }
            body.add(new IntegerConstant((short) flags));            

            Constant viTableIndex = new IntegerConstant((short) -1);
            if (vtable != null) {
                VTable.Entry entry = vtable.getEntry(m);
                if (entry != null) {
                    viTableIndex = new IntegerConstant((short) entry.getIndex());
                }
            } else {
                ITable.Entry entry = itable.getEntry(m);
                if (entry != null) {
                    viTableIndex = new IntegerConstant((short) entry.getIndex());
                }
            }
            body.add(viTableIndex);            
            
            body.add(getString(m.getName()));
            
            if ((flags & MI_COMPACT_DESC) > 0) {
                int desc = 0;
                if (t.equals(BooleanType.v())) {
                    desc = DESC_Z;
                } else if (t.equals(ByteType.v())) {
                    desc = DESC_B;
                } else if (t.equals(ShortType.v())) {
                    desc = DESC_S;
                } else if (t.equals(CharType.v())) {
                    desc = DESC_C;
                } else if (t.equals(IntType.v())) {
                    desc = DESC_I;
                } else if (t.equals(LongType.v())) {
                    desc = DESC_J;
                } else if (t.equals(FloatType.v())) {
                    desc = DESC_F;
                } else if (t.equals(DoubleType.v())) {
                    desc = DESC_D;
                } else if (t.equals(VoidType.v())) {
                    desc = DESC_V;
                }
                body.add(new IntegerConstant((byte) desc));            
            } else {
                body.add(getString(getDescriptor(m)));
            }
            if (attributesEncoder.methodHasAttributes(m)) {
                body.add(new ConstantBitcast(attributesEncoder.getMethodAttributes(m).ref(), I8_PTR));
            }
            if (!m.isAbstract()) {
                body.add(new ConstantBitcast(new FunctionRef(Symbols.methodSymbol(m), getFunctionType(m)), I8_PTR));
                body.add(new IntegerConstant(DUMMY_METHOD_SIZE)); // Size of function. This value will be modified later by patching the .s file.
                if (m.isSynchronized()) {
                    body.add(new ConstantBitcast(new FunctionRef(Symbols.synchronizedWrapperSymbol(m), getFunctionType(m)), I8_PTR));
                }
                if ((flags & MI_NATIVE) == 0) {
                    // Cannot use m.isNative() in the condition above since methods which are native in the
                    // Java class file may have been changed to non-native by the RoboVM compiler 
                    // (e.g. @StructMember methods). The native code which parses the info structs will see 
                    // the method as non-native.

                    // Add a weak linetable pointer which points to a -1 value which will be interpreted as 0 linenumbers in the table
                    Global linetableGlobal = new Global(Symbols.linetableSymbol(m), Linkage.weak, new IntegerConstant(-1));
                    mb.addGlobal(linetableGlobal);
                    body.add(linetableGlobal.ref());
                }
            }
            if (hasBridgeAnnotation(m) && !isJvmSyntheticBridgeMethod(m)) {
                if (!readBooleanElem(getAnnotation(m, BRIDGE), "dynamic", false)) {
                    body.add(new GlobalRef(Symbols.bridgePtrSymbol(m), I8_PTR));
                } else {
                    body.add(new NullConstant(I8_PTR));
                }
            } else if (hasGlobalValueAnnotation(m)) {
                body.add(new GlobalRef(Symbols.globalValuePtrSymbol(m), I8_PTR));
            }
            if (hasCallbackAnnotation(m)) {
                body.add(new AliasRef(Symbols.callbackPtrSymbol(m), I8_PTR));
            }
        }
        
        // Return the struct {header, body}. To be compatible with the C code in classinfo.c 
        // it is important that the header is padded the same as in C so that the body starts
        // after sizeof(ClassInfoHeader) bytes.
        StructureConstant infoStruct = new StructureConstantBuilder().add(header.build()).add(body.build()).build(INFO_STRUCT_ALIAS);
        mb.addType(infoStruct.getType());
        return infoStruct;
    }

    private Function compileMethod(AbstractMethodCompiler methodCompiler, SootMethod method) {
        Function fn = methodCompiler.compile(mb, method);
        for (Trampoline t : methodCompiler.getTrampolines()) {
            List l = trampolines.get(t);
            if (l == null) {
                l = new ArrayList<>();
                trampolines.put(t, l);
            }
            l.add(method);
        }
        catches.addAll(methodCompiler.getCatches());
        return fn;
    }

    private Function nativeMethod(SootMethod method) {
        return compileMethod(nativeMethodCompiler, method);
    }

    private Function bridgeMethod(SootMethod method) {
        return compileMethod(bridgeMethodCompiler, method);
    }
    
    private Function callbackMethod(SootMethod method) {
        return compileMethod(callbackMethodCompiler, method);
    }
    
    private Function structMember(SootMethod method) {
        return compileMethod(structMemberMethodCompiler, method);
    }

    private Function globalValueMethod(SootMethod method) {
        return compileMethod(globalValueMethodCompiler, method);
    }

    private Function objcPublishMethod(SootMethod method) {
        return compileMethod(objcMethodCompiler, method);
    }

    private Function method(SootMethod method) {
        return compileMethod(javaMethodCompiler, method);
    }
    
    private Function createAllocator() {
        Function fn = FunctionBuilder.allocator(sootClass);
        Value info = getInfoStruct(fn, sootClass);        
        Value result = call(fn, BC_ALLOCATE, fn.getParameterRef(0), info);
        fn.add(new Ret(result));
        return fn;
    }

    private Function createLdcClass() {
        Function fn = FunctionBuilder.ldcInternal(sootClass);
        Value info = getInfoStruct(fn, sootClass);
        Value result = call(fn, BC_LDC_CLASS, fn.getParameterRef(0), info);
        fn.add(new Ret(result));
        return fn;
    }
    
    private Function createLdcClassWrapper() {
        Function fn = FunctionBuilder.ldcExternal(sootClass);
        Value info = getInfoStruct(fn, sootClass);
        Value result = call(fn, LDC_CLASS_WRAPPER, fn.getParameterRef(0), info);
        fn.add(new Ret(result));
        return fn;
    }
    
    static Function createFieldGetter(SootField field, List classFields, 
            StructureType classType, List instanceFields, StructureType instanceType) {
        
        Function fn = FunctionBuilder.getter(field);
        return createFieldGetter(fn, field, classFields, classType, instanceFields, instanceType);
    }
    
    static Function createFieldGetter(Function fn, SootField field, List classFields, 
            StructureType classType, List instanceFields, StructureType instanceType) {
        
        Value fieldPtr = null;
        if (field.isStatic()) {
            fieldPtr = getClassFieldPtr(fn, field, classFields, classType);
        } else {
            fieldPtr = getInstanceFieldPtr(fn, fn.getParameterRef(1), field, instanceFields, instanceType);
        }
        Variable result = fn.newVariable(getType(field.getType()));
        if (Modifier.isVolatile(field.getModifiers())) {
            fn.add(new Fence(Ordering.seq_cst));
            if (LongType.v().equals(field.getType())) {
                fn.add(new Load(result, fieldPtr, false, Ordering.monotonic, 8));
            } else {
                fn.add(new Load(result, fieldPtr));
            }
        } else {
            fn.add(new Load(result, fieldPtr));
        }
        fn.add(new Ret(new VariableRef(result)));
        return fn;
    }
    
    static Function createFieldSetter(SootField field, List classFields, 
            StructureType classType, List instanceFields, StructureType instanceType) {
        
        Function fn = FunctionBuilder.setter(field);
        return createFieldSetter(fn, field, classFields, classType, instanceFields, instanceType);
    }
    
    static Function createFieldSetter(Function fn, SootField field, List classFields, 
            StructureType classType, List instanceFields, StructureType instanceType) {

        Value fieldPtr = null;
        Value value = null;
        if (field.isStatic()) {
            fieldPtr = getClassFieldPtr(fn, field, classFields, classType);
            value = fn.getParameterRef(1);
        } else {
            fieldPtr = getInstanceFieldPtr(fn, fn.getParameterRef(1), field, instanceFields, instanceType);
            value = fn.getParameterRef(2);
        }
        if (Modifier.isVolatile(field.getModifiers()) || !field.isStatic() && Modifier.isFinal(field.getModifiers())) {
            if (LongType.v().equals(field.getType())) {
                fn.add(new Store(value, fieldPtr, false, Ordering.monotonic, 8));
            } else {
                fn.add(new Store(value, fieldPtr));
            }
            fn.add(new Fence(Ordering.seq_cst));
        } else {
            fn.add(new Store(value, fieldPtr));
        }
        fn.add(new Ret());
        return fn;
    }
    
    private Function createClassInitWrapperFunction(FunctionRef targetFn) {
        Function fn = FunctionBuilder.clinitWrapper(targetFn);
        Value info = getInfoStruct(fn, sootClass);
        Variable infoHeader = fn.newVariable(new PointerType(new StructureType(I8_PTR, I32)));
        fn.add(new Bitcast(infoHeader, info, infoHeader.getType()));
        Variable infoHeaderFlags = fn.newVariable(new PointerType(I32));
        fn.add(new Getelementptr(infoHeaderFlags, infoHeader.ref(), 0, 1));
        Variable flags = fn.newVariable(I32);
        fn.add(new Load(flags, infoHeaderFlags.ref()));
        Variable initializedFlag = fn.newVariable(I32);
        fn.add(new And(initializedFlag, flags.ref(), new IntegerConstant(CI_INITIALIZED)));
        Variable initialized = fn.newVariable(I1);
        fn.add(new Icmp(initialized, Icmp.Condition.eq, initializedFlag.ref(), new IntegerConstant(CI_INITIALIZED)));
        Label trueLabel = new Label();
        Label falseLabel = new Label();
        fn.add(new Br(initialized.ref(), fn.newBasicBlockRef(trueLabel), fn.newBasicBlockRef(falseLabel)));
        fn.newBasicBlock(trueLabel);
        Value result = call(fn, targetFn, fn.getParameterRefs());
        fn.add(new Ret(result));
        fn.newBasicBlock(falseLabel);
        call(fn, BC_INITIALIZE_CLASS, fn.getParameterRef(0), info);
        fn.add(new Br(fn.newBasicBlockRef(trueLabel)));
        return fn;
    }

    private static int countReferences(List l) {
        int count = 0;
        for (SootField f : l) {
            if (f.getType() instanceof RefLikeType) {
                count++;
            }
        }
        return count;
    }
    
    private static void collectInterfaces(SootClass clazz, Set interfaces) {
        if (clazz.hasSuperclass()) {
            collectInterfaces(clazz.getSuperclass(), interfaces);
        }
        if (clazz.isInterface()) {
            interfaces.add(clazz);
        }
        for (SootClass sc : clazz.getInterfaces()) {
            collectInterfaces(sc, interfaces);
        }
    }
    
    private static boolean hasConstantValueTags(List classFields) {
        for (SootField field : classFields) {
            for (Tag tag : field.getTags()) {
                if (tag instanceof ConstantValueTag) {
                    return true;
                }
            }
        }
        return false;
    }
    
    private static boolean hasFinalizer(SootClass clazz) {
        // Don't search interfaces or java.lang.Object
        if (clazz.isInterface() || !clazz.hasSuperclass()) {
            return false;
        }
        return clazz.declaresMethod("finalize", Collections.emptyList(), VoidType.v());
    }
    
    private Constant getString(String string) {
        return mb.getString(string);
    }
    
    private Constant getStringOrNull(String string) {
        return mb.getStringOrNull(string);
    }
    
    static Value getClassFieldPtr(Function f, SootField field, List classFields, 
            StructureType classType) {
        
        Value info = getInfoStruct(f, field.getDeclaringClass());        
        Variable base = f.newVariable(I8_PTR);
        f.add(new Load(base, info));
        return getFieldPtr(f, new VariableRef(base), offsetof(classType, 1, 
                classFields.indexOf(field), 1), getType(field.getType()));
    }

    static Value getInfoStruct(Function f, SootClass sootClass) {
        return call(f, FunctionBuilder.infoStruct(sootClass).ref());
    }
    
    static Value getInstanceFieldPtr(Function f, Value base, SootField field, 
            List instanceFields, StructureType instanceType) {
        return getFieldPtr(f, base, offsetof(instanceType, 1, 
                1 + instanceFields.indexOf(field), 1), getType(field.getType()));
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy