//===--- GenArray.cpp - Swift IR Generation for Array Types -----------===// // // This source file is part of the Swift.org open source project // // Copyright (c) 2014 - 2015 Apple Inc. and the Swift project authors // Licensed under Apache License v2.0 with Runtime Library Exception // // See http://swift.org/LICENSE.txt for license information // See http://swift.org/CONTRIBUTORS.txt for the list of Swift project authors // //===----------------------------------------------------------------------===// // // This file implements IR generation for array types in Swift. This // includes creating the IR type as well as performing primitive array // access operations. // //===----------------------------------------------------------------------===// #include "swift/AST/Types.h" #include "swift/AST/Decl.h" #include "swift/AST/Expr.h" #include "llvm/DerivedTypes.h" #include "Address.h" #include "Explosion.h" #include "GenFunc.h" #include "GenHeap.h" #include "GenType.h" #include "IRGenModule.h" #include "GenArray.h" using namespace swift; using namespace irgen; namespace { class ArrayTypeInfo : public TypeInfo { public: ArrayTypeInfo() : TypeInfo(nullptr, Size(0), Alignment(0), IsPOD) {} unsigned getExplosionSize(ExplosionKind kind) const { return 1; } void getSchema(ExplosionSchema &schema) const { // FIXME } void load(IRGenFunction &IGF, Address addr, Explosion &explosion) const { // FIXME } void loadAsTake(IRGenFunction &IGF, Address addr, Explosion &e) const { // FIXME } void assign(IRGenFunction &IGF, Explosion &explosion, Address addr) const { // FIXME } void initialize(IRGenFunction &IGF, Explosion &explosion, Address addr) const { // FIXME } void reexplode(IRGenFunction &IGF, Explosion &src, Explosion &dest) const { // FIXME } void copy(IRGenFunction &IGF, Explosion &src, Explosion &dest) const { // FIXME } void manage(IRGenFunction &IGF, Explosion &src, Explosion &dest) const { // FIXME } void destroy(IRGenFunction &IGF, Address addr) const { // FIXME } }; } const TypeInfo * TypeConverter::convertArrayType(IRGenModule &IGM, ArrayType *T) { // FIXME return new ArrayTypeInfo(); } /// Emit an array allocation expression. void swift::irgen::emitNewArrayExpr(IRGenFunction &IGF, NewArrayExpr *E, Explosion &out) { // First things first: emit the array bound. Sema ensures that this // is always a builtin type. llvm::Value *length; { Explosion bounds(ExplosionKind::Maximal); IGF.emitRValue(E->getBounds()[0].Value, bounds); length = bounds.claimUnmanagedNext(); } // Emit the allocation. const TypeInfo &elementTI = IGF.getFragileTypeInfo(E->getElementType()); Expr *init = nullptr; ArrayHeapLayout layout(IGF.IGM, elementTI); Address begin; ManagedValue alloc = layout.emitAlloc(IGF, length, begin, init, "new-array"); // Eventually the data address will be well-typed, but for now it // always needs to be an i8*. llvm::Value *dataAddr = begin.getAddress(); dataAddr = IGF.Builder.CreateBitCast(dataAddr, IGF.IGM.Int8PtrTy); // Emit the callee. std::vector args; Callee callee = emitCallee(IGF, E->getInjectionFunction(), out.getKind(), /*uncurry*/ 0, args); // The injection function takes this tuple: // (Builtin.RawPointer, Builtin.ObjectPointer, typeof(length)) Explosion injectionArg(callee.getExplosionLevel()); injectionArg.addUnmanaged(dataAddr); injectionArg.add(alloc); injectionArg.addUnmanaged(length); args.push_back(Arg::forUnowned(injectionArg)); const TypeInfo &sliceTI = IGF.getFragileTypeInfo(E->getType()); emitCall(IGF, callee, args, sliceTI, out); }