llgo/ssa: DeferAlways/DeferInCond/DeferInLoop

This commit is contained in:
xushiwei
2024-06-03 02:43:04 +08:00
parent 6ac6fb0192
commit cfca98512a
4 changed files with 63 additions and 22 deletions

View File

@@ -803,8 +803,10 @@ type DoAction int
const (
Call DoAction = iota
Defer
Go
DeferAlways // defer statement executes always
DeferInCond // defer statement executes in a conditional block
DeferInLoop // defer statement executes in a loop block
)
// Do call a function with an action.
@@ -812,10 +814,10 @@ func (b Builder) Do(da DoAction, fn Expr, args ...Expr) (ret Expr) {
switch da {
case Call:
return b.Call(fn, args...)
case Defer:
b.Defer(fn, args...)
case Go:
b.Go(fn, args...)
default:
b.Defer(da, fn, args...)
}
return
}

View File

@@ -216,24 +216,37 @@ func (b Builder) getDefer() *aDefer {
}
// Defer emits a defer instruction.
func (b Builder) Defer(fn Expr, args ...Expr) {
func (b Builder) Defer(kind DoAction, fn Expr, args ...Expr) {
if debugInstr {
logCall("Defer", fn, args)
}
prog := b.Prog
self := b.getDefer()
next := self.nextBit
self.nextBit++
bits := b.Load(self.bitsPtr)
nextbit := prog.Val(uintptr(1 << next))
b.Store(self.bitsPtr, b.BinOp(token.OR, bits, nextbit))
var prog Program
var nextbit Expr
var self = b.getDefer()
switch kind {
case DeferInCond:
prog = b.Prog
next := self.nextBit
self.nextBit++
bits := b.Load(self.bitsPtr)
nextbit = prog.Val(uintptr(1 << next))
b.Store(self.bitsPtr, b.BinOp(token.OR, bits, nextbit))
case DeferAlways:
// nothing to do
default:
panic("todo: DeferInLoop is not supported")
}
self.stmts = append(self.stmts, func(bits Expr) {
zero := prog.Val(uintptr(0))
has := b.BinOp(token.NEQ, b.BinOp(token.AND, bits, nextbit), zero)
b.IfThen(has, func() {
switch kind {
case DeferInCond:
zero := prog.Val(uintptr(0))
has := b.BinOp(token.NEQ, b.BinOp(token.AND, bits, nextbit), zero)
b.IfThen(has, func() {
b.Call(fn, args...)
})
case DeferAlways:
b.Call(fn, args...)
})
}
})
}