Commit 92cbf82f authored by Alan Donovan's avatar Alan Donovan

exp/ssa: add dedicated Panic instruction.

By avoiding the need for self-loops following calls to panic,
we reduce the number of basic blocks considerably.

R=gri
CC=golang-dev, iant
https://golang.org/cl/7403043
parent 0ad88a48
...@@ -62,6 +62,7 @@ var ( ...@@ -62,6 +62,7 @@ var (
tInvalid = types.Typ[types.Invalid] tInvalid = types.Typ[types.Invalid]
tUntypedNil = types.Typ[types.UntypedNil] tUntypedNil = types.Typ[types.UntypedNil]
tRangeIter = &types.Basic{Name: "iter"} // the type of all "range" iterators tRangeIter = &types.Basic{Name: "iter"} // the type of all "range" iterators
tEface = new(types.Interface)
// The result type of a "select". // The result type of a "select".
tSelect = &types.Result{Values: []*types.Var{ tSelect = &types.Result{Values: []*types.Var{
...@@ -512,6 +513,11 @@ func (b *Builder) builtin(fn *Function, name string, args []ast.Expr, typ types. ...@@ -512,6 +513,11 @@ func (b *Builder) builtin(fn *Function, name string, args []ast.Expr, typ types.
return intLiteral(at.Len) return intLiteral(at.Len)
} }
// Otherwise treat as normal. // Otherwise treat as normal.
case "panic":
fn.emit(&Panic{X: emitConv(fn, b.expr(fn, args[0]), tEface)})
fn.currentBlock = fn.newBasicBlock("unreachable")
return vFalse // any non-nil Value will do
} }
return nil // treat all others as a regular function call return nil // treat all others as a regular function call
} }
...@@ -774,32 +780,20 @@ func (b *Builder) expr(fn *Function, e ast.Expr) Value { ...@@ -774,32 +780,20 @@ func (b *Builder) expr(fn *Function, e ast.Expr) Value {
// Type conversion, e.g. string(x) or big.Int(x) // Type conversion, e.g. string(x) or big.Int(x)
return emitConv(fn, b.expr(fn, e.Args[0]), typ) return emitConv(fn, b.expr(fn, e.Args[0]), typ)
} }
// Call to "intrinsic" built-ins, e.g. new, make. // Call to "intrinsic" built-ins, e.g. new, make, panic.
wasPanic := false
if id, ok := e.Fun.(*ast.Ident); ok { if id, ok := e.Fun.(*ast.Ident); ok {
obj := b.obj(id) obj := b.obj(id)
if _, ok := fn.Prog.Builtins[obj]; ok { if _, ok := fn.Prog.Builtins[obj]; ok {
if v := b.builtin(fn, id.Name, e.Args, typ); v != nil { if v := b.builtin(fn, id.Name, e.Args, typ); v != nil {
return v return v
} }
wasPanic = id.Name == "panic"
} }
} }
// Regular function call. // Regular function call.
var v Call var v Call
b.setCall(fn, e, &v.CallCommon) b.setCall(fn, e, &v.CallCommon)
v.setType(typ) v.setType(typ)
fn.emit(&v) return fn.emit(&v)
// Compile panic as if followed by for{} so that its
// successor is unreachable.
// TODO(adonovan): consider a dedicated Panic instruction
// (in which case, don't forget Go and Defer).
if wasPanic {
emitSelfLoop(fn)
fn.currentBlock = fn.newBasicBlock("unreachable")
}
return &v
case *ast.UnaryExpr: case *ast.UnaryExpr:
switch e.Op { switch e.Op {
...@@ -1161,7 +1155,7 @@ func (b *Builder) setCall(fn *Function, e *ast.CallExpr, c *CallCommon) { ...@@ -1161,7 +1155,7 @@ func (b *Builder) setCall(fn *Function, e *ast.CallExpr, c *CallCommon) {
bptypes = append(bptypes, nil) // map bptypes = append(bptypes, nil) // map
bptypes = append(bptypes, nil) // key bptypes = append(bptypes, nil) // key
case "print", "println": // print{,ln}(any, ...any) case "print", "println": // print{,ln}(any, ...any)
vt = new(types.Interface) // variadic vt = tEface // variadic
if !c.HasEllipsis { if !c.HasEllipsis {
args, varargs = args[:1], args[1:] args, varargs = args[:1], args[1:]
} }
...@@ -1188,7 +1182,7 @@ func (b *Builder) setCall(fn *Function, e *ast.CallExpr, c *CallCommon) { ...@@ -1188,7 +1182,7 @@ func (b *Builder) setCall(fn *Function, e *ast.CallExpr, c *CallCommon) {
} }
bptypes = append(bptypes, argType, argType) bptypes = append(bptypes, argType, argType)
case "panic": case "panic":
bptypes = append(bptypes, new(types.Interface)) bptypes = append(bptypes, tEface)
case "recover": case "recover":
// no-op // no-op
default: default:
...@@ -2257,14 +2251,14 @@ start: ...@@ -2257,14 +2251,14 @@ start:
case *ast.GoStmt: case *ast.GoStmt:
// The "intrinsics" new/make/len/cap are forbidden here. // The "intrinsics" new/make/len/cap are forbidden here.
// panic() is not forbidden, but is not (yet) an intrinsic. // panic is treated like an ordinary function call.
var v Go var v Go
b.setCall(fn, s.Call, &v.CallCommon) b.setCall(fn, s.Call, &v.CallCommon)
fn.emit(&v) fn.emit(&v)
case *ast.DeferStmt: case *ast.DeferStmt:
// The "intrinsics" new/make/len/cap are forbidden here. // The "intrinsics" new/make/len/cap are forbidden here.
// panic() is not forbidden, but is not (yet) an intrinsic. // panic is treated like an ordinary function call.
var v Defer var v Defer
b.setCall(fn, s.Call, &v.CallCommon) b.setCall(fn, s.Call, &v.CallCommon)
fn.emit(&v) fn.emit(&v)
......
...@@ -247,15 +247,3 @@ func emitTailCall(f *Function, call *Call) { ...@@ -247,15 +247,3 @@ func emitTailCall(f *Function, call *Call) {
f.emit(&ret) f.emit(&ret)
f.currentBlock = nil f.currentBlock = nil
} }
// emitSelfLoop emits to f a self-loop.
// This is a defensive measure to ensure control-flow integrity.
// It should never be reachable.
// Postcondition: f.currentBlock is nil.
//
func emitSelfLoop(f *Function) {
loop := f.newBasicBlock("selfloop")
emitJump(f, loop)
f.currentBlock = loop
emitJump(f, loop)
}
...@@ -171,6 +171,9 @@ func visitInstr(fr *frame, instr ssa.Instruction) continuation { ...@@ -171,6 +171,9 @@ func visitInstr(fr *frame, instr ssa.Instruction) continuation {
} }
return kReturn return kReturn
case *ssa.Panic:
panic(targetPanic{fr.get(instr.X)})
case *ssa.Send: case *ssa.Send:
fr.get(instr.Chan).(chan value) <- copyVal(fr.get(instr.X)) fr.get(instr.Chan).(chan value) <- copyVal(fr.get(instr.X))
...@@ -475,7 +478,7 @@ func callSSA(i *interpreter, caller *frame, callpos token.Pos, fn *ssa.Function, ...@@ -475,7 +478,7 @@ func callSSA(i *interpreter, caller *frame, callpos token.Pos, fn *ssa.Function,
for { for {
if i.mode&EnableTracing != 0 { if i.mode&EnableTracing != 0 {
fmt.Fprintf(os.Stderr, ".%s:\n", fr.block.Name) fmt.Fprintf(os.Stderr, ".%s:\n", fr.block)
} }
block: block:
for _, instr = range fr.block.Instrs { for _, instr = range fr.block.Instrs {
......
...@@ -16,6 +16,9 @@ type targetPanic struct { ...@@ -16,6 +16,9 @@ type targetPanic struct {
v value v value
} }
// If the target program calls exit, the interpreter panics with this type.
type exitPanic int
// literalValue returns the value of the literal with the // literalValue returns the value of the literal with the
// dynamic type tag appropriate for l.Type(). // dynamic type tag appropriate for l.Type().
func literalValue(l *ssa.Literal) value { func literalValue(l *ssa.Literal) value {
...@@ -974,6 +977,8 @@ func callBuiltin(caller *frame, callpos token.Pos, fn *ssa.Builtin, args []value ...@@ -974,6 +977,8 @@ func callBuiltin(caller *frame, callpos token.Pos, fn *ssa.Builtin, args []value
} }
case "panic": case "panic":
// ssa.Panic handles most cases; this is only for "go
// panic" or "defer panic".
panic(targetPanic{args[0]}) panic(targetPanic{args[0]})
case "recover": case "recover":
......
...@@ -282,6 +282,10 @@ func (s *Go) String() string { ...@@ -282,6 +282,10 @@ func (s *Go) String() string {
return printCall(&s.CallCommon, "go ", s) return printCall(&s.CallCommon, "go ", s)
} }
func (s *Panic) String() string {
return "panic " + relName(s.X, s)
}
func (s *Ret) String() string { func (s *Ret) String() string {
var b bytes.Buffer var b bytes.Buffer
b.WriteString("ret") b.WriteString("ret")
......
...@@ -96,7 +96,7 @@ func findDuplicate(blocks []*BasicBlock) *BasicBlock { ...@@ -96,7 +96,7 @@ func findDuplicate(blocks []*BasicBlock) *BasicBlock {
func (s *sanity) checkInstr(idx int, instr Instruction) { func (s *sanity) checkInstr(idx int, instr Instruction) {
switch instr := instr.(type) { switch instr := instr.(type) {
case *If, *Jump, *Ret: case *If, *Jump, *Ret, *Panic:
s.errorf("control flow instruction not at end of block") s.errorf("control flow instruction not at end of block")
case *Phi: case *Phi:
if idx == 0 { if idx == 0 {
...@@ -192,6 +192,12 @@ func (s *sanity) checkFinalInstr(idx int, instr Instruction) { ...@@ -192,6 +192,12 @@ func (s *sanity) checkFinalInstr(idx int, instr Instruction) {
} }
// TODO(adonovan): check number and types of results // TODO(adonovan): check number and types of results
case *Panic:
if nsuccs := len(s.block.Succs); nsuccs != 0 {
s.errorf("Panic-terminated block has %d successors; expected none", nsuccs)
return
}
default: default:
s.errorf("non-control flow instruction at end of block") s.errorf("non-control flow instruction at end of block")
} }
......
...@@ -248,7 +248,7 @@ type Function struct { ...@@ -248,7 +248,7 @@ type Function struct {
// An SSA basic block. // An SSA basic block.
// //
// The final element of Instrs is always an explicit transfer of // The final element of Instrs is always an explicit transfer of
// control (If, Jump or Ret). // control (If, Jump, Ret or Panic).
// //
// A block may contain no Instructions only if it is unreachable, // A block may contain no Instructions only if it is unreachable,
// i.e. Preds is nil. Empty blocks are typically pruned. // i.e. Preds is nil. Empty blocks are typically pruned.
...@@ -842,6 +842,22 @@ type Ret struct { ...@@ -842,6 +842,22 @@ type Ret struct {
Results []Value Results []Value
} }
// Panic initiates a panic with value X.
//
// A Panic instruction must be the last instruction of its containing
// BasicBlock, which must have no successors.
//
// NB: 'go panic(x)' and 'defer panic(x)' do not use this instruction;
// they are treated as calls to a built-in function.
//
// Example printed form:
// panic t0
//
type Panic struct {
anInstruction
X Value // an interface{}
}
// Go creates a new goroutine and calls the specified function // Go creates a new goroutine and calls the specified function
// within it. // within it.
// //
...@@ -1125,6 +1141,7 @@ func (*MakeMap) ImplementsInstruction() {} ...@@ -1125,6 +1141,7 @@ func (*MakeMap) ImplementsInstruction() {}
func (*MakeSlice) ImplementsInstruction() {} func (*MakeSlice) ImplementsInstruction() {}
func (*MapUpdate) ImplementsInstruction() {} func (*MapUpdate) ImplementsInstruction() {}
func (*Next) ImplementsInstruction() {} func (*Next) ImplementsInstruction() {}
func (*Panic) ImplementsInstruction() {}
func (*Phi) ImplementsInstruction() {} func (*Phi) ImplementsInstruction() {}
func (*Range) ImplementsInstruction() {} func (*Range) ImplementsInstruction() {}
func (*Ret) ImplementsInstruction() {} func (*Ret) ImplementsInstruction() {}
...@@ -1227,6 +1244,10 @@ func (v *Next) Operands(rands []*Value) []*Value { ...@@ -1227,6 +1244,10 @@ func (v *Next) Operands(rands []*Value) []*Value {
return append(rands, &v.Iter) return append(rands, &v.Iter)
} }
func (s *Panic) Operands(rands []*Value) []*Value {
return append(rands, &s.X)
}
func (v *Phi) Operands(rands []*Value) []*Value { func (v *Phi) Operands(rands []*Value) []*Value {
for i := range v.Edges { for i := range v.Edges {
rands = append(rands, &v.Edges[i]) rands = append(rands, &v.Edges[i])
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment