mirror of git://gcc.gnu.org/git/gcc.git
				
				
				
			
		
			
				
	
	
		
			95 lines
		
	
	
		
			2.6 KiB
		
	
	
	
		
			C
		
	
	
	
			
		
		
	
	
			95 lines
		
	
	
		
			2.6 KiB
		
	
	
	
		
			C
		
	
	
	
| /* go-deferred-recover.c -- support for a deferred recover function.
 | |
| 
 | |
|    Copyright 2010 The Go Authors. All rights reserved.
 | |
|    Use of this source code is governed by a BSD-style
 | |
|    license that can be found in the LICENSE file.  */
 | |
| 
 | |
| #include <stddef.h>
 | |
| 
 | |
| #include "runtime.h"
 | |
| #include "go-panic.h"
 | |
| #include "go-defer.h"
 | |
| 
 | |
| /* This is called when a call to recover is deferred.  That is,
 | |
|    something like
 | |
|      defer recover()
 | |
| 
 | |
|    We need to handle this specially.  In 6g/8g, the recover function
 | |
|    looks up the stack frame.  In particular, that means that a
 | |
|    deferred recover will not recover a panic thrown in the same
 | |
|    function that defers the recover.  It will only recover a panic
 | |
|    thrown in a function that defers the deferred call to recover.
 | |
| 
 | |
|    In other words:
 | |
| 
 | |
|    func f1() {
 | |
| 	defer recover()	// does not stop panic
 | |
| 	panic(0)
 | |
|    }
 | |
| 
 | |
|    func f2() {
 | |
| 	defer func() {
 | |
| 		defer recover()	// stops panic(0)
 | |
| 	}()
 | |
| 	panic(0)
 | |
|    }
 | |
| 
 | |
|    func f3() {
 | |
| 	defer func() {
 | |
| 		defer recover()	// does not stop panic
 | |
| 		panic(0)
 | |
| 	}()
 | |
| 	panic(1)
 | |
|    }
 | |
| 
 | |
|    func f4() {
 | |
| 	defer func() {
 | |
| 		defer func() {
 | |
| 			defer recover()	// stops panic(0)
 | |
| 		}()
 | |
| 		panic(0)
 | |
| 	}()
 | |
| 	panic(1)
 | |
|    }
 | |
| 
 | |
|    The interesting case here is f3.  As can be seen from f2, the
 | |
|    deferred recover could pick up panic(1).  However, this does not
 | |
|    happen because it is blocked by the panic(0).
 | |
| 
 | |
|    When a function calls recover, then when we invoke it we pass a
 | |
|    hidden parameter indicating whether it should recover something.
 | |
|    This parameter is set based on whether the function is being
 | |
|    invoked directly from defer.  The parameter winds up determining
 | |
|    whether __go_recover or __go_deferred_recover is called at all.
 | |
| 
 | |
|    In the case of a deferred recover, the hidden parameter which
 | |
|    controls the call is actually the one set up for the function which
 | |
|    runs the defer recover() statement.  That is the right thing in all
 | |
|    the cases above except for f3.  In f3 the function is permitted to
 | |
|    call recover, but the deferred recover call is not.  We address
 | |
|    that here by checking for that specific case before calling
 | |
|    recover.  If this function was deferred when there is already a
 | |
|    panic on the panic stack, then we can only recover that panic, not
 | |
|    any other.
 | |
| 
 | |
|    Note that we can get away with using a special function here
 | |
|    because you are not permitted to take the address of a predeclared
 | |
|    function like recover.  */
 | |
| 
 | |
| struct __go_empty_interface
 | |
| __go_deferred_recover ()
 | |
| {
 | |
|   G *g;
 | |
| 
 | |
|   g = runtime_g ();
 | |
|   if (g->defer == NULL || g->defer->__panic != g->panic)
 | |
|     {
 | |
|       struct __go_empty_interface ret;
 | |
| 
 | |
|       ret.__type_descriptor = NULL;
 | |
|       ret.__object = NULL;
 | |
|       return ret;
 | |
|     }
 | |
|   return __go_recover ();
 | |
| }
 |