Files
swift-mirror/lib/IRGen/GenArray.cpp
Eli Friedman 1341f19da8 IRGen for varargs tuples.
Swift SVN r1883
2012-05-17 03:22:49 +00:00

141 lines
4.2 KiB
C++

//===--- 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();
}
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");
emitArrayInjectionCall(IGF, alloc, begin, E->getType(),
E->getInjectionFunction(), length, out);
}
void swift::irgen::emitArrayInjectionCall(IRGenFunction &IGF, ManagedValue alloc,
Address begin, Type sliceTy,
Expr *injectionFn, llvm::Value *length,
Explosion &out) {
// Emit the allocation.
// 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.
llvm::SmallVector<Arg, 4> args;
Callee callee = emitCallee(IGF, injectionFn,
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(sliceTy);
emitCall(IGF, callee, args, sliceTI, out);
}