mirror of
https://github.com/apple/swift.git
synced 2025-12-14 20:36:38 +01:00
69 lines
2.6 KiB
Swift
69 lines
2.6 KiB
Swift
//===--- SimplifyInitEnumDataAddr.swift -----------------------------------===//
|
|
//
|
|
// This source file is part of the Swift.org open source project
|
|
//
|
|
// Copyright (c) 2014 - 2023 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
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
import SIL
|
|
|
|
extension InitEnumDataAddrInst : OnoneSimplifiable {
|
|
func simplify(_ context: SimplifyContext) {
|
|
|
|
// Optimize the sequence
|
|
// ```
|
|
// %1 = init_enum_data_addr %enum_addr, #someCaseWithPayload
|
|
// ...
|
|
// store %payload to %1
|
|
// inject_enum_addr %enum_addr, #someCaseWithPayload
|
|
// ```
|
|
// to
|
|
// ```
|
|
// %1 = enum $E, #someCaseWithPayload, %payload
|
|
// store %1 to %enum_addr
|
|
// ```
|
|
// This store and inject instructions must appear in consecutive order.
|
|
// But usually this is the case, because it's generated this way by SILGen.
|
|
// We also check that between the init and the store, no instruction writes to memory.
|
|
//
|
|
if let store = self.uses.singleUse?.instruction as? StoreInst,
|
|
store.destination == self,
|
|
let inject = store.next as? InjectEnumAddrInst,
|
|
inject.enum == self.enum,
|
|
inject.enum.type.isLoadable(in: parentFunction),
|
|
!anyInstructionMayWriteToMemory(between: self, and: store) {
|
|
|
|
assert(self.caseIndex == inject.caseIndex, "mismatching case indices when creating an enum")
|
|
|
|
let builder = Builder(before: store, context)
|
|
let enumInst = builder.createEnum(caseIndex: self.caseIndex, payload: store.source, enumType: self.enum.type.objectType)
|
|
let storeOwnership = StoreInst.StoreOwnership(for: self.enum.type, in: parentFunction, initialize: true)
|
|
builder.createStore(source: enumInst, destination: self.enum, ownership: storeOwnership)
|
|
context.erase(instruction: store)
|
|
context.erase(instruction: inject)
|
|
context.erase(instruction: self)
|
|
}
|
|
}
|
|
}
|
|
|
|
// Returns false if `first` and `last` are in the same basic block and no instructions between them write to memory. True otherwise.
|
|
private func anyInstructionMayWriteToMemory(between first: Instruction, and last: Instruction) -> Bool {
|
|
var nextInstruction = first.next
|
|
while let i = nextInstruction {
|
|
if i == last {
|
|
return false
|
|
}
|
|
if i.mayWriteToMemory {
|
|
return true
|
|
}
|
|
nextInstruction = i.next
|
|
}
|
|
// End of basic block, and we did not find `last`
|
|
return true
|
|
}
|