Files
swift-mirror/lib/IRGen/GenInit.cpp
John McCall abb146fa0c Teach StackAddress to directly express the kind of allocation that was
performed, and simplify and improve the handling of allocation throughout
IRGen.

The improvements are that we used to use dynamic allocas in a few
places that clearly could and should use static allocas.
2026-03-06 03:15:28 -05:00

105 lines
3.7 KiB
C++

//===--- GenInit.cpp - IR Generation for Initialization -------------------===//
//
// This source file is part of the Swift.org open source project
//
// Copyright (c) 2014 - 2017 Apple Inc. and the Swift project authors
// Licensed under Apache License v2.0 with Runtime Library Exception
//
// See https://swift.org/LICENSE.txt for license information
// See https://swift.org/CONTRIBUTORS.txt for the list of Swift project authors
//
//===----------------------------------------------------------------------===//
//
// This file implements IR generation for the initialization of
// local and global variables.
//
//
//===----------------------------------------------------------------------===//
#include "swift/AST/Pattern.h"
#include "swift/Basic/Assertions.h"
#include "swift/SIL/SILDeclRef.h"
#include "swift/SIL/SILGlobalVariable.h"
#include "swift/IRGen/Linking.h"
#include "llvm/IR/GlobalVariable.h"
#include "llvm/IR/Module.h"
#include "DebugTypeInfo.h"
#include "GenTuple.h"
#include "IRGenDebugInfo.h"
#include "IRGenFunction.h"
#include "IRGenModule.h"
#include "FixedTypeInfo.h"
#include "Temporary.h"
using namespace swift;
using namespace irgen;
/// Emit a global variable.
void IRGenModule::emitSILGlobalVariable(SILGlobalVariable *var) {
auto &ti = getTypeInfo(var->getLoweredType());
auto expansion = getResilienceExpansionForLayout(var);
// If the variable is empty in all resilience domains that can access this
// variable directly, don't actually emit it; just return undef.
if (ti.isKnownEmpty(expansion)) {
if (DebugInfo && var->getDecl()) {
auto DbgTy = DebugTypeInfo::getGlobal(var, *this);
DebugInfo->emitGlobalVariableDeclaration(nullptr, var->getDecl()->getName().str(),
"", DbgTy,
var->getLinkage() != SILLinkage::Public &&
var->getLinkage() != SILLinkage::Package,
SILLocation(var->getDecl()));
}
return;
}
/// Create the global variable.
getAddrOfSILGlobalVariable(var, ti,
var->isDefinition() ? ForDefinition : NotForDefinition);
}
StackAddress
FixedTypeInfo::allocateStack(IRGenFunction &IGF, SILType T, const Twine &name,
StackAllocationIsNested_t isNested) const {
// If the type is known to be empty, don't actually allocate anything.
if (isKnownEmpty(ResilienceExpansion::Maximal)) {
auto addr = getUndefAddress();
return StackAddress(addr, StackAddress::StaticAlloca, nullptr);
}
return IGF.emitStaticAlloca(getStorageType(), getFixedSize(),
getFixedAlignment(), name);
}
void FixedTypeInfo::destroyStack(IRGenFunction &IGF, StackAddress addr,
SILType T, bool isOutlined) const {
destroy(IGF, addr.getAddress(), T, isOutlined);
FixedTypeInfo::deallocateStack(IGF, addr, T);
}
void FixedTypeInfo::deallocateStack(IRGenFunction &IGF, StackAddress addr,
SILType T) const {
if (isKnownEmpty(ResilienceExpansion::Maximal))
return;
IGF.Builder.CreateLifetimeEnd(addr.getAddress(), getFixedSize());
}
void TemporarySet::destroyAll(IRGenFunction &IGF) const {
assert(!hasBeenCleared() && "destroying a set that's been cleared?");
// Deallocate all the temporaries.
for (auto &temporary : llvm::reverse(Stack)) {
temporary.destroy(IGF);
}
}
void Temporary::destroy(IRGenFunction &IGF) const {
auto &ti = IGF.getTypeInfo(Type);
if (Type.isSensitive()) {
llvm::Value *size = ti.getSize(IGF, Type);
IGF.emitClearSensitive(Addr.getAddress(), size);
}
ti.deallocateStack(IGF, Addr, Type);
}