llgo/ssa: DeferAlways/DeferInCond/DeferInLoop
This commit is contained in:
@@ -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
|
||||
}
|
||||
|
||||
@@ -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...)
|
||||
})
|
||||
}
|
||||
})
|
||||
}
|
||||
|
||||
|
||||
Reference in New Issue
Block a user