Files
swift-mirror/lib/IRGen/GenStruct.cpp
John McCall 38b34b7307 Pass #1 at localizing assumptions about fixed layout and
handling non-fixed layouts.

This uncovered a bug where we weren't rounding up the header
size to the element alignment when allocating an array of archetypes.

Writing up a detailed test case for *that* revealed that we were
never initializing the length field of heap arrays.  Fixing that
caused a bunch of tests to crash trying to release stuff.  So...
I've left this in a workaround state right now because I have to
catch a plane.

Swift SVN r4804
2013-04-18 07:58:21 +00:00

195 lines
6.5 KiB
C++

//===--- GenStruct.cpp - Swift IR Generation For 'struct' 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 struct types.
//
//===----------------------------------------------------------------------===//
#include "GenStruct.h"
#include "swift/AST/Types.h"
#include "swift/AST/Decl.h"
#include "swift/AST/Expr.h"
#include "swift/AST/Pattern.h"
#include "swift/Basic/Optional.h"
#include "llvm/IR/DerivedTypes.h"
#include "llvm/IR/Function.h"
#include "GenInit.h"
#include "GenMeta.h"
#include "GenSequential.h"
#include "GenType.h"
#include "IRGenFunction.h"
#include "IRGenModule.h"
#include "Linking.h"
#include "Explosion.h"
using namespace swift;
using namespace irgen;
namespace {
class StructFieldInfo : public SequentialField<StructFieldInfo> {
public:
StructFieldInfo(VarDecl *field, const TypeInfo &type)
: SequentialField(type), Field(field) {}
/// The field.
VarDecl *Field;
StringRef getFieldName() const {
return Field->getName().str();
}
};
/// Layout information for struct types.
class StructTypeInfo : // FIXME: FixedTypeInfo as the base class is a lie.
public SequentialTypeInfo<StructTypeInfo, FixedTypeInfo, StructFieldInfo> {
public:
StructTypeInfo(llvm::Type *T, unsigned numFields)
: SequentialTypeInfo(T, numFields) {
}
};
class StructTypeBuilder :
public SequentialTypeBuilder<StructTypeBuilder, StructTypeInfo, VarDecl*> {
llvm::StructType *StructTy;
public:
StructTypeBuilder(IRGenModule &IGM, llvm::StructType *structTy) :
SequentialTypeBuilder(IGM), StructTy(structTy) {
}
StructTypeInfo *construct(void *buffer, ArrayRef<VarDecl*> fields) {
return ::new(buffer) StructTypeInfo(StructTy, fields.size());
}
StructFieldInfo getFieldInfo(VarDecl *field, const TypeInfo &fieldTI) {
return StructFieldInfo(field, fieldTI);
}
Type getType(VarDecl *field) { return field->getType(); }
void performLayout(ArrayRef<const TypeInfo *> fieldTypes) {
StructLayout layout(IGM, LayoutKind::NonHeapObject,
LayoutStrategy::Optimal, fieldTypes, StructTy);
recordLayout(layout, StructTy);
}
};
} // end anonymous namespace.
OwnedAddress irgen::projectPhysicalStructMemberAddress(IRGenFunction &IGF,
OwnedAddress base,
CanType baseType,
unsigned fieldIndex) {
assert((baseType->is<StructType>() || baseType->is<BoundGenericStructType>())
&& "not a struct type");
auto &baseTI = IGF.getFragileTypeInfo(baseType).as<StructTypeInfo>();
auto &fieldI = baseTI.getFields()[fieldIndex];
Address project = fieldI.projectAddress(IGF, base);
return OwnedAddress(project, base.getOwner());
}
void irgen::projectPhysicalStructMemberFromExplosion(IRGenFunction &IGF,
CanType baseType,
Explosion &base,
unsigned fieldNo,
Explosion &out) {
auto &baseTI = IGF.getFragileTypeInfo(baseType).as<StructTypeInfo>();
auto &fieldI = baseTI.getFields()[fieldNo];
// If the field requires no storage, there's nothing to do.
if (fieldI.isEmpty()) {
return IGF.emitFakeExplosion(fieldI.getTypeInfo(), out);
}
// Otherwise, project from the base.
auto fieldRange = fieldI.getProjectionRange(out.getKind());
ArrayRef<ManagedValue> element = base.getRange(fieldRange.first,
fieldRange.second);
out.add(element);
}
/// emitStructDecl - Emit all the declarations associated with this struct type.
void IRGenModule::emitStructDecl(StructDecl *st) {
emitStructMetadata(*this, st);
// FIXME: This is mostly copy-paste from emitExtension;
// figure out how to refactor!
for (Decl *member : st->getMembers()) {
switch (member->getKind()) {
case DeclKind::Import:
case DeclKind::TopLevelCode:
case DeclKind::Protocol:
case DeclKind::Extension:
case DeclKind::Destructor:
case DeclKind::OneOfElement:
case DeclKind::InfixOperator:
case DeclKind::PrefixOperator:
case DeclKind::PostfixOperator:
llvm_unreachable("decl not allowed in struct!");
// We can have meaningful initializers for variables, but
// we can't handle them yet. For the moment, just ignore them.
case DeclKind::PatternBinding:
continue;
case DeclKind::Subscript:
// Getter/setter will be handled separately.
continue;
case DeclKind::TypeAlias:
continue;
case DeclKind::OneOf:
emitOneOfDecl(cast<OneOfDecl>(member));
continue;
case DeclKind::Struct:
emitStructDecl(cast<StructDecl>(member));
continue;
case DeclKind::Class:
emitClassDecl(cast<ClassDecl>(member));
continue;
case DeclKind::Var:
if (cast<VarDecl>(member)->isProperty())
// Getter/setter will be handled separately.
continue;
// FIXME: Will need an implementation here for resilience
continue;
case DeclKind::Func:
// Methods are emitted as SIL Functions already.
continue;
case DeclKind::Constructor: {
// Constructors get emitted as SIL Functions.
continue;
}
}
llvm_unreachable("bad extension member kind");
}
}
const TypeInfo *TypeConverter::convertStructType(StructDecl *D) {
StructTypeBuilder builder(IGM, IGM.createNominalType(D));
// Collect all the fields from the type.
SmallVector<VarDecl*, 8> fields;
for (Decl *D : D->getMembers())
if (VarDecl *VD = dyn_cast<VarDecl>(D))
if (!VD->isProperty())
fields.push_back(VD);
// Allocate the TypeInfo and register it as a forward-declaration.
// We do this before we look at any of the child types.
auto structTI = builder.create(fields);
auto typesMapKey = D->getDeclaredType().getPointer();
Types.insert(std::make_pair(typesMapKey, structTI));
// Complete the type and return it.
return builder.complete(fields);
}