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

org.qbicc.plugin.llvm.LLVMModuleGenerator Maven / Gradle / Ivy

package org.qbicc.plugin.llvm;

import java.io.BufferedWriter;
import java.io.IOException;
import java.nio.file.Path;
import java.util.List;
import java.util.Map;

import io.smallrye.common.constraint.Assert;
import org.qbicc.context.CompilationContext;
import org.qbicc.context.Location;
import org.qbicc.facts.Facts;
import org.qbicc.facts.core.ExecutableReachabilityFacts;
import org.qbicc.graph.literal.Literal;
import org.qbicc.graph.literal.LiteralFactory;
import org.qbicc.machine.arch.Platform;
import org.qbicc.machine.llvm.FunctionAttributes;
import org.qbicc.machine.llvm.FunctionDefinition;
import org.qbicc.machine.llvm.Global;
import org.qbicc.machine.llvm.LLValue;
import org.qbicc.machine.llvm.Linkage;
import org.qbicc.machine.llvm.Module;
import org.qbicc.machine.llvm.ModuleFlagBehavior;
import org.qbicc.machine.llvm.RuntimePreemption;
import org.qbicc.machine.llvm.ThreadLocalStorageModel;
import org.qbicc.machine.llvm.Types;
import org.qbicc.machine.llvm.Values;
import org.qbicc.object.Data;
import org.qbicc.object.DataDeclaration;
import org.qbicc.object.Declaration;
import org.qbicc.object.Function;
import org.qbicc.object.FunctionDeclaration;
import org.qbicc.object.GlobalXtor;
import org.qbicc.object.ModuleSection;
import org.qbicc.object.ProgramModule;
import org.qbicc.object.SectionObject;
import org.qbicc.object.Segment;
import org.qbicc.object.ThreadLocalMode;
import org.qbicc.type.ArrayType;
import org.qbicc.type.CompoundType;
import org.qbicc.type.FunctionType;
import org.qbicc.type.PointerType;
import org.qbicc.type.TypeSystem;
import org.qbicc.type.UnsignedIntegerType;
import org.qbicc.type.ValueType;
import org.qbicc.type.VariadicType;
import org.qbicc.type.definition.MethodBody;
import org.qbicc.type.definition.element.ConstructorElement;
import org.qbicc.type.definition.element.ExecutableElement;
import org.qbicc.type.definition.element.MemberElement;
import org.qbicc.type.definition.element.MethodElement;

final class LLVMModuleGenerator {
    private final CompilationContext context;
    private final LLVMConfiguration config;
    private final int picLevel;
    private final int pieLevel;

    LLVMModuleGenerator(CompilationContext context, LLVMConfiguration config) {
        this.context = context;
        this.config = config;
        if (config.isPie()) {
            this.picLevel = 2;
            this.pieLevel = 2;
        } else {
            this.picLevel = 0;
            this.pieLevel = 0;
        }
    }

    public void processProgramModule(final ProgramModule programModule, BufferedWriter writer, Path irFile) {
        final Module module = Module.newModule();
        TypeSystem ts = context.getTypeSystem();
        module.dataLayout()
            .byteOrder(ts.getEndianness())
            .pointerSize(ts.getPointerSize() * 8)
            .pointerAlign(ts.getPointerAlignment() * 8)
            .refSize(ts.getReferenceSize() * 8)
            .refAlign(ts.getReferenceAlignment() * 8)
            .int8Align(ts.getUnsignedInteger8Type().getAlign() * 8)
            .int16Align(ts.getUnsignedInteger16Type().getAlign() * 8)
            .int32Align(ts.getUnsignedInteger32Type().getAlign() * 8)
            .int64Align(ts.getUnsignedInteger64Type().getAlign() * 8)
            .float32Align(ts.getFloat32Type().getAlign() * 8)
            .float64Align(ts.getFloat64Type().getAlign() * 8)
            ;
        module.sourceFileName(irFile.toString());
        final LLVMModuleNodeVisitor moduleVisitor = new LLVMModuleNodeVisitor(this, module, context, config);
        final LLVMModuleDebugInfo debugInfo = new LLVMModuleDebugInfo(programModule, module, context);

        if (picLevel != 0) {
            module.addFlag(ModuleFlagBehavior.Max, "PIC Level", Types.i32, Values.intConstant(picLevel));
        }

        if (pieLevel != 0) {
            module.addFlag(ModuleFlagBehavior.Max, "PIE Level", Types.i32, Values.intConstant(pieLevel));
        }
        final Platform platform = context.getPlatform();

        // declare debug function here
        org.qbicc.machine.llvm.Function decl = module.declare("llvm.dbg.value");
        decl.returns(Types.void_);
        decl.param(Types.metadata).param(Types.metadata).param(Types.metadata);

        // declare global ctors and dtors
        processXtors(programModule.constructors(), "llvm.global_ctors", module, moduleVisitor);
        processXtors(programModule.destructors(), "llvm.global_dtors", module, moduleVisitor);

        for (Declaration item : programModule.declarations()) {
            String name = item.getName();
            Linkage linkage = map(item.getLinkage());
            if (item instanceof FunctionDeclaration fn) {
                decl = module.declare(name).linkage(linkage);
                FunctionType fnType = fn.getValueType();
                decl.returns(moduleVisitor.map(fnType.getReturnType()));
                int cnt = fnType.getParameterCount();
                for (int i = 0; i < cnt; i++) {
                    ValueType type = fnType.getParameterType(i);
                    if (type instanceof VariadicType) {
                        if (i < cnt - 1) {
                            throw new IllegalStateException("Variadic type as non-final parameter type");
                        }
                        decl.variadic();
                    } else {
                        decl.param(moduleVisitor.map(type));
                    }
                }
            } else if (item instanceof DataDeclaration) {
                Global obj = module.global(moduleVisitor.map(item.getValueType())).linkage(Linkage.EXTERNAL);
                ThreadLocalMode tlm = item.getThreadLocalMode();
                if (tlm != null) {
                    obj.threadLocal(map(tlm));
                }
                obj.asGlobal(item.getName());
            }
        }
        for (ModuleSection section : programModule.sections()) {
            String sectionName = section.getName();
            final Segment segment = section.getSection().getSegment();
            for (SectionObject item : section.contents()) {
                String name = item.getName();
                Linkage linkage = map(item.getLinkage());
                if (item instanceof Function fn) {
                    ExecutableElement element = fn.getOriginalElement();
                    if (! Facts.get(context).hadFact(element, ExecutableReachabilityFacts.IS_INVOKED)) {
                        // not reachable; do not emit
                        continue;
                    }
                    MethodBody body = fn.getBody();
                    boolean isExact = item == context.getExactFunction(element);
                    if (body == null) {
                        context.error("Function `%s` has no body", name);
                        continue;
                    }
                    FunctionDefinition functionDefinition = module.define(name).linkage(linkage);
                    LLValue topSubprogram;

                    if (element instanceof MethodElement me) {
                        functionDefinition.comment(me.getEnclosingType().getInternalName()+"."+me.getName()+" "+me.getDescriptor());
                    } else if (element instanceof ConstructorElement ce) {
                        functionDefinition.comment(ce.getEnclosingType().getInternalName()+". "+ce.getDescriptor());
                    }

                    if (isExact) {
                        topSubprogram = debugInfo.getDebugInfoForFunction(element).getSubprogram();
                        functionDefinition.meta("dbg", topSubprogram);
                    } else {
                        topSubprogram = debugInfo.createThunkSubprogram(fn).asRef();
                        functionDefinition.meta("dbg", topSubprogram);
                    }
                    functionDefinition.attribute(FunctionAttributes.framePointer("non-leaf"));
                    functionDefinition.attribute(FunctionAttributes.uwtable);
                    if (config.isStatepointEnabled()) {
                        functionDefinition.gc("statepoint-example");
                    }
                    if (fn.isNoReturn()) {
                        functionDefinition.attribute(FunctionAttributes.noreturn);
                    }

                    LLVMNodeVisitor nodeVisitor = new LLVMNodeVisitor(context, module, debugInfo, topSubprogram, moduleVisitor, fn, functionDefinition);
                    if (! sectionName.equals(CompilationContext.IMPLICIT_SECTION_NAME)) {
                        functionDefinition.section(platform.formatSectionName(segment.toString(), segment.toString(), sectionName));
                    }

                    nodeVisitor.execute();
                } else if (item instanceof Data data) {
                    Literal value = (Literal) data.getValue();
                    Global obj = module.global(moduleVisitor.map(data.getValueType()));
                    if (value != null) {
                        obj.value(moduleVisitor.map(value));
                    } else {
                        obj.value(Values.zeroinitializer);
                    }
                    obj.alignment(data.getValueType().getAlign());
                    obj.linkage(linkage);
                    ThreadLocalMode tlm = data.getThreadLocalMode();
                    if (tlm != null) {
                        obj.threadLocal(map(tlm));
                    }
                    if (data.isDsoLocal()) {
                        obj.preemption(RuntimePreemption.LOCAL);
                    }
                    if (! sectionName.equals(CompilationContext.IMPLICIT_SECTION_NAME)) {
                        obj.section(platform.formatSectionName(segment.toString(), segment.toString(), sectionName));
                    }
                    MemberElement element = data.getOriginalElement();
                    if (element != null) {
                        obj.meta("dbg", debugInfo.getDebugInfoForGlobal(data, element));
                    }
                    obj.asGlobal(data.getName());
                } else {
                    throw new IllegalStateException();
                }
            }
        }
        try {
            module.writeTo(writer);
        } catch (IOException e) {
            context.error(Location.builder().setClassInternalName(programModule.getTypeDefinition().getInternalName()).build(), "Failed to emit LLVM output: %s", e.toString());
        }
    }

    private void processXtors(final List xtors, final String xtorName, Module module, LLVMModuleNodeVisitor moduleVisitor) {
        if (! xtors.isEmpty()) {
            TypeSystem ts = context.getTypeSystem();
            LiteralFactory lf = context.getLiteralFactory();
            int xtorSize = 0;
            // this special type has a fixed size and layout
            UnsignedIntegerType u32 = ts.getUnsignedInteger32Type();
            CompoundType.Member priorityMember = ts.getUnalignedCompoundTypeMember("priority", u32, 0);
            xtorSize += u32.getSize();
            PointerType voidFnPtrType = ts.getFunctionType(ts.getVoidType()).getPointer();
            CompoundType.Member fnMember = ts.getUnalignedCompoundTypeMember("fn", voidFnPtrType, xtorSize);
            xtorSize += voidFnPtrType.getSize();
            PointerType u8ptr = ts.getUnsignedInteger8Type().getPointer();
            CompoundType.Member dataMember = ts.getUnalignedCompoundTypeMember("data", u8ptr, xtorSize);
            xtorSize += u8ptr.getSize();
            CompoundType xtorType = ts.getCompoundType(CompoundType.Tag.NONE, "xtor_t", xtorSize, 1, () -> List.of(
                priorityMember, fnMember, dataMember
            ));
            // special global
            ArrayType arrayType = ts.getArrayType(xtorType, xtors.size());
            Global global_ctors = module.global(moduleVisitor.map(arrayType));
            global_ctors.appending();
            global_ctors.value(moduleVisitor.map(lf.literalOf(
                arrayType, xtors.stream().map(g -> lf.literalOf(xtorType, Map.of(
                    priorityMember, lf.literalOf(g.getPriority()),
                    fnMember, lf.bitcastLiteral(lf.literalOf(g.getFunction()), voidFnPtrType),
                    dataMember, lf.nullLiteralOfType(u8ptr)
                ))).toList()
            )));
            global_ctors.asGlobal(xtorName);
        }
    }

    public int getLlvmMajor() {
        return config.getMajorVersion();
    }

    Linkage map(org.qbicc.object.Linkage linkage) {
        switch (linkage) {
            case COMMON: return Linkage.COMMON;
            case INTERNAL: return Linkage.INTERNAL;
            case PRIVATE: return Linkage.PRIVATE;
            case WEAK: return Linkage.EXTERN_WEAK;
            case EXTERNAL: return Linkage.EXTERNAL;
            default: throw Assert.impossibleSwitchCase(linkage);
        }
    }

    ThreadLocalStorageModel map(ThreadLocalMode mode) {
        switch (mode) {
            case GENERAL_DYNAMIC: return ThreadLocalStorageModel.GENERAL_DYNAMIC;
            case LOCAL_DYNAMIC: return ThreadLocalStorageModel.LOCAL_DYNAMIC;
            case INITIAL_EXEC: return ThreadLocalStorageModel.INITIAL_EXEC;
            case LOCAL_EXEC: return ThreadLocalStorageModel.LOCAL_EXEC;
            default: throw Assert.impossibleSwitchCase(mode);
        }
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy