| 
									
										
										
										
											2014-05-20 12:10:19 -04:00
										 |  |  | // +build !nacl | 
					
						
							| 
									
										
										
										
											2015-02-03 10:52:18 -08:00
										 |  |  | // run | 
					
						
							| 
									
										
										
										
											2014-05-20 12:10:19 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-04-16 22:08:00 -04:00
										 |  |  | // Copyright 2014 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. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | package main | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import ( | 
					
						
							|  |  |  | 	"bytes" | 
					
						
							|  |  |  | 	"fmt" | 
					
						
							|  |  |  | 	"io/ioutil" | 
					
						
							| 
									
										
										
										
											2014-09-16 17:39:55 -04:00
										 |  |  | 	"log" | 
					
						
							| 
									
										
										
										
											2014-04-16 22:08:00 -04:00
										 |  |  | 	"os" | 
					
						
							|  |  |  | 	"os/exec" | 
					
						
							|  |  |  | 	"path/filepath" | 
					
						
							|  |  |  | 	"regexp" | 
					
						
							|  |  |  | 	"runtime" | 
					
						
							|  |  |  | 	"strconv" | 
					
						
							|  |  |  | 	"strings" | 
					
						
							|  |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | var tests = ` | 
					
						
							|  |  |  | # These are test cases for the linker analysis that detects chains of | 
					
						
							|  |  |  | # nosplit functions that would cause a stack overflow. | 
					
						
							|  |  |  | # | 
					
						
							|  |  |  | # Lines beginning with # are comments. | 
					
						
							|  |  |  | # | 
					
						
							|  |  |  | # Each test case describes a sequence of functions, one per line. | 
					
						
							|  |  |  | # Each function definition is the function name, then the frame size, | 
					
						
							|  |  |  | # then optionally the keyword 'nosplit', then the body of the function. | 
					
						
							|  |  |  | # The body is assembly code, with some shorthands. | 
					
						
							|  |  |  | # The shorthand 'call x' stands for CALL x(SB). | 
					
						
							|  |  |  | # The shorthand 'callind' stands for 'CALL R0', where R0 is a register. | 
					
						
							|  |  |  | # Each test case must define a function named main, and it must be first. | 
					
						
							|  |  |  | # That is, a line beginning "main " indicates the start of a new test case. | 
					
						
							|  |  |  | # Within a stanza, ; can be used instead of \n to separate lines. | 
					
						
							|  |  |  | # | 
					
						
							|  |  |  | # After the function definition, the test case ends with an optional | 
					
						
							|  |  |  | # REJECT line, specifying the architectures on which the case should | 
					
						
							|  |  |  | # be rejected. "REJECT" without any architectures means reject on all architectures. | 
					
						
							|  |  |  | # The linker should accept the test case on systems not explicitly rejected. | 
					
						
							|  |  |  | # | 
					
						
							|  |  |  | # 64-bit systems do not attempt to execute test cases with frame sizes | 
					
						
							|  |  |  | # that are only 32-bit aligned. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | # Ordinary function should work | 
					
						
							|  |  |  | main 0 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | # Large frame marked nosplit is always wrong. | 
					
						
							|  |  |  | main 10000 nosplit | 
					
						
							|  |  |  | REJECT | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | # Calling a large frame is okay. | 
					
						
							|  |  |  | main 0 call big | 
					
						
							|  |  |  | big 10000 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | # But not if the frame is nosplit. | 
					
						
							|  |  |  | main 0 call big | 
					
						
							|  |  |  | big 10000 nosplit | 
					
						
							|  |  |  | REJECT | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | # Recursion is okay. | 
					
						
							|  |  |  | main 0 call main | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | # Recursive nosplit runs out of space. | 
					
						
							|  |  |  | main 0 nosplit call main | 
					
						
							|  |  |  | REJECT | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | # Chains of ordinary functions okay. | 
					
						
							|  |  |  | main 0 call f1 | 
					
						
							|  |  |  | f1 80 call f2 | 
					
						
							|  |  |  | f2 80 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | # Chains of nosplit must fit in the stack limit, 128 bytes. | 
					
						
							|  |  |  | main 0 call f1 | 
					
						
							|  |  |  | f1 80 nosplit call f2 | 
					
						
							|  |  |  | f2 80 nosplit | 
					
						
							|  |  |  | REJECT | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | # Larger chains. | 
					
						
							|  |  |  | main 0 call f1 | 
					
						
							|  |  |  | f1 16 call f2 | 
					
						
							|  |  |  | f2 16 call f3 | 
					
						
							|  |  |  | f3 16 call f4 | 
					
						
							|  |  |  | f4 16 call f5 | 
					
						
							|  |  |  | f5 16 call f6 | 
					
						
							|  |  |  | f6 16 call f7 | 
					
						
							|  |  |  | f7 16 call f8 | 
					
						
							|  |  |  | f8 16 call end | 
					
						
							|  |  |  | end 1000 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | main 0 call f1 | 
					
						
							|  |  |  | f1 16 nosplit call f2 | 
					
						
							|  |  |  | f2 16 nosplit call f3 | 
					
						
							|  |  |  | f3 16 nosplit call f4 | 
					
						
							|  |  |  | f4 16 nosplit call f5 | 
					
						
							|  |  |  | f5 16 nosplit call f6 | 
					
						
							|  |  |  | f6 16 nosplit call f7 | 
					
						
							|  |  |  | f7 16 nosplit call f8 | 
					
						
							|  |  |  | f8 16 nosplit call end | 
					
						
							|  |  |  | end 1000 | 
					
						
							|  |  |  | REJECT | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | # Test cases near the 128-byte limit. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | # Ordinary stack split frame is always okay. | 
					
						
							|  |  |  | main 112 | 
					
						
							|  |  |  | main 116 | 
					
						
							|  |  |  | main 120 | 
					
						
							|  |  |  | main 124 | 
					
						
							|  |  |  | main 128 | 
					
						
							|  |  |  | main 132 | 
					
						
							|  |  |  | main 136 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | # A nosplit leaf can use the whole 128-CallSize bytes available on entry. | 
					
						
							|  |  |  | main 112 nosplit | 
					
						
							|  |  |  | main 116 nosplit | 
					
						
							|  |  |  | main 120 nosplit | 
					
						
							|  |  |  | main 124 nosplit | 
					
						
							|  |  |  | main 128 nosplit; REJECT | 
					
						
							|  |  |  | main 132 nosplit; REJECT | 
					
						
							|  |  |  | main 136 nosplit; REJECT | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | # Calling a nosplit function from a nosplit function requires | 
					
						
							|  |  |  | # having room for the saved caller PC and the called frame. | 
					
						
							|  |  |  | # Because ARM doesn't save LR in the leaf, it gets an extra 4 bytes. | 
					
						
							| 
									
										
										
										
											2014-12-05 19:13:20 -05:00
										 |  |  | # Because ppc64 doesn't save LR in the leaf, it gets an extra 8 bytes. | 
					
						
							| 
									
										
										
										
											2014-04-16 22:08:00 -04:00
										 |  |  | main 112 nosplit call f; f 0 nosplit | 
					
						
							| 
									
										
										
										
											2014-08-14 15:29:37 -04:00
										 |  |  | main 116 nosplit call f; f 0 nosplit | 
					
						
							| 
									
										
										
										
											2014-04-16 22:08:00 -04:00
										 |  |  | main 120 nosplit call f; f 0 nosplit; REJECT amd64 | 
					
						
							|  |  |  | main 124 nosplit call f; f 0 nosplit; REJECT amd64 386 | 
					
						
							|  |  |  | main 128 nosplit call f; f 0 nosplit; REJECT | 
					
						
							|  |  |  | main 132 nosplit call f; f 0 nosplit; REJECT | 
					
						
							|  |  |  | main 136 nosplit call f; f 0 nosplit; REJECT | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | # Calling a splitting function from a nosplit function requires | 
					
						
							|  |  |  | # having room for the saved caller PC of the call but also the | 
					
						
							| 
									
										
										
										
											2014-08-14 15:29:37 -04:00
										 |  |  | # saved caller PC for the call to morestack. | 
					
						
							| 
									
										
										
										
											2014-12-05 19:13:20 -05:00
										 |  |  | # Again the ARM and ppc64 work in less space. | 
					
						
							| 
									
										
										
										
											2014-04-16 22:08:00 -04:00
										 |  |  | main 104 nosplit call f; f 0 call f | 
					
						
							|  |  |  | main 108 nosplit call f; f 0 call f | 
					
						
							|  |  |  | main 112 nosplit call f; f 0 call f; REJECT amd64 | 
					
						
							|  |  |  | main 116 nosplit call f; f 0 call f; REJECT amd64 | 
					
						
							|  |  |  | main 120 nosplit call f; f 0 call f; REJECT amd64 386 | 
					
						
							|  |  |  | main 124 nosplit call f; f 0 call f; REJECT amd64 386 | 
					
						
							|  |  |  | main 128 nosplit call f; f 0 call f; REJECT | 
					
						
							|  |  |  | main 132 nosplit call f; f 0 call f; REJECT | 
					
						
							|  |  |  | main 136 nosplit call f; f 0 call f; REJECT | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | # Indirect calls are assumed to be splitting functions. | 
					
						
							|  |  |  | main 104 nosplit callind | 
					
						
							|  |  |  | main 108 nosplit callind | 
					
						
							|  |  |  | main 112 nosplit callind; REJECT amd64 | 
					
						
							|  |  |  | main 116 nosplit callind; REJECT amd64 | 
					
						
							|  |  |  | main 120 nosplit callind; REJECT amd64 386 | 
					
						
							|  |  |  | main 124 nosplit callind; REJECT amd64 386 | 
					
						
							|  |  |  | main 128 nosplit callind; REJECT | 
					
						
							|  |  |  | main 132 nosplit callind; REJECT | 
					
						
							|  |  |  | main 136 nosplit callind; REJECT | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | # Issue 7623 | 
					
						
							|  |  |  | main 0 call f; f 112 | 
					
						
							|  |  |  | main 0 call f; f 116 | 
					
						
							|  |  |  | main 0 call f; f 120 | 
					
						
							|  |  |  | main 0 call f; f 124 | 
					
						
							|  |  |  | main 0 call f; f 128 | 
					
						
							|  |  |  | main 0 call f; f 132 | 
					
						
							|  |  |  | main 0 call f; f 136 | 
					
						
							|  |  |  | ` | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | var ( | 
					
						
							|  |  |  | 	commentRE = regexp.MustCompile(`(?m)^#.*`) | 
					
						
							|  |  |  | 	rejectRE  = regexp.MustCompile(`(?s)\A(.+?)((\n|; *)REJECT(.*))?\z`) | 
					
						
							|  |  |  | 	lineRE    = regexp.MustCompile(`(\w+) (\d+)( nosplit)?(.*)`) | 
					
						
							|  |  |  | 	callRE    = regexp.MustCompile(`\bcall (\w+)\b`) | 
					
						
							|  |  |  | 	callindRE = regexp.MustCompile(`\bcallind\b`) | 
					
						
							|  |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func main() { | 
					
						
							|  |  |  | 	goarch := os.Getenv("GOARCH") | 
					
						
							|  |  |  | 	if goarch == "" { | 
					
						
							|  |  |  | 		goarch = runtime.GOARCH | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-21 13:28:13 -04:00
										 |  |  | 	version, err := exec.Command("go", "tool", "compile", "-V").Output() | 
					
						
							| 
									
										
										
										
											2015-01-14 11:09:50 -05:00
										 |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		bug() | 
					
						
							| 
									
										
										
										
											2015-05-27 12:33:43 -07:00
										 |  |  | 		fmt.Printf("running go tool compile -V: %v\n", err) | 
					
						
							| 
									
										
										
										
											2015-01-14 11:09:50 -05:00
										 |  |  | 		return | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if strings.Contains(string(version), "framepointer") { | 
					
						
							|  |  |  | 		// Skip this test if GOEXPERIMENT=framepointer | 
					
						
							|  |  |  | 		return | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-04-16 22:08:00 -04:00
										 |  |  | 	dir, err := ioutil.TempDir("", "go-test-nosplit") | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		bug() | 
					
						
							|  |  |  | 		fmt.Printf("creating temp dir: %v\n", err) | 
					
						
							|  |  |  | 		return | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	defer os.RemoveAll(dir) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	tests = strings.Replace(tests, "\t", " ", -1) | 
					
						
							|  |  |  | 	tests = commentRE.ReplaceAllString(tests, "") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	nok := 0 | 
					
						
							|  |  |  | 	nfail := 0 | 
					
						
							|  |  |  | TestCases: | 
					
						
							|  |  |  | 	for len(tests) > 0 { | 
					
						
							|  |  |  | 		var stanza string | 
					
						
							|  |  |  | 		i := strings.Index(tests, "\nmain ") | 
					
						
							|  |  |  | 		if i < 0 { | 
					
						
							|  |  |  | 			stanza, tests = tests, "" | 
					
						
							|  |  |  | 		} else { | 
					
						
							|  |  |  | 			stanza, tests = tests[:i], tests[i+1:] | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		m := rejectRE.FindStringSubmatch(stanza) | 
					
						
							|  |  |  | 		if m == nil { | 
					
						
							|  |  |  | 			bug() | 
					
						
							|  |  |  | 			fmt.Printf("invalid stanza:\n\t%s\n", indent(stanza)) | 
					
						
							|  |  |  | 			continue | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		lines := strings.TrimSpace(m[1]) | 
					
						
							|  |  |  | 		reject := false | 
					
						
							|  |  |  | 		if m[2] != "" { | 
					
						
							|  |  |  | 			if strings.TrimSpace(m[4]) == "" { | 
					
						
							|  |  |  | 				reject = true | 
					
						
							|  |  |  | 			} else { | 
					
						
							|  |  |  | 				for _, rej := range strings.Fields(m[4]) { | 
					
						
							|  |  |  | 					if rej == goarch { | 
					
						
							|  |  |  | 						reject = true | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		if lines == "" && !reject { | 
					
						
							|  |  |  | 			continue | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-09-16 17:39:55 -04:00
										 |  |  | 		var gobuf bytes.Buffer | 
					
						
							|  |  |  | 		fmt.Fprintf(&gobuf, "package main\n") | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-04-16 22:08:00 -04:00
										 |  |  | 		var buf bytes.Buffer | 
					
						
							| 
									
										
										
										
											2014-08-14 13:59:58 -04:00
										 |  |  | 		ptrSize := 4 | 
					
						
							|  |  |  | 		switch goarch { | 
					
						
							| 
									
										
										
										
											2014-12-05 19:13:20 -05:00
										 |  |  | 		case "ppc64", "ppc64le": | 
					
						
							| 
									
										
										
										
											2014-08-14 13:59:58 -04:00
										 |  |  | 			ptrSize = 8 | 
					
						
							| 
									
										
										
										
											2015-06-06 15:24:18 -04:00
										 |  |  | 			fmt.Fprintf(&buf, "#define CALL BL\n#define REGISTER (CTR)\n") | 
					
						
							| 
									
										
										
										
											2014-08-14 13:59:58 -04:00
										 |  |  | 		case "arm": | 
					
						
							| 
									
										
										
										
											2014-04-16 22:08:00 -04:00
										 |  |  | 			fmt.Fprintf(&buf, "#define CALL BL\n#define REGISTER (R0)\n") | 
					
						
							| 
									
										
										
										
											2015-03-08 14:27:51 +01:00
										 |  |  | 		case "arm64": | 
					
						
							|  |  |  | 			ptrSize = 8 | 
					
						
							|  |  |  | 			fmt.Fprintf(&buf, "#define CALL BL\n#define REGISTER (R0)\n") | 
					
						
							| 
									
										
										
										
											2014-08-14 13:59:58 -04:00
										 |  |  | 		case "amd64": | 
					
						
							|  |  |  | 			ptrSize = 8 | 
					
						
							|  |  |  | 			fmt.Fprintf(&buf, "#define REGISTER AX\n") | 
					
						
							|  |  |  | 		default: | 
					
						
							| 
									
										
										
										
											2014-04-16 22:08:00 -04:00
										 |  |  | 			fmt.Fprintf(&buf, "#define REGISTER AX\n") | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		for _, line := range strings.Split(lines, "\n") { | 
					
						
							|  |  |  | 			line = strings.TrimSpace(line) | 
					
						
							|  |  |  | 			if line == "" { | 
					
						
							|  |  |  | 				continue | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2014-08-27 14:08:26 -04:00
										 |  |  | 			for i, subline := range strings.Split(line, ";") { | 
					
						
							| 
									
										
										
										
											2014-04-16 22:08:00 -04:00
										 |  |  | 				subline = strings.TrimSpace(subline) | 
					
						
							|  |  |  | 				if subline == "" { | 
					
						
							|  |  |  | 					continue | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 				m := lineRE.FindStringSubmatch(subline) | 
					
						
							|  |  |  | 				if m == nil { | 
					
						
							|  |  |  | 					bug() | 
					
						
							|  |  |  | 					fmt.Printf("invalid function line: %s\n", subline) | 
					
						
							|  |  |  | 					continue TestCases | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 				name := m[1] | 
					
						
							|  |  |  | 				size, _ := strconv.Atoi(m[2]) | 
					
						
							| 
									
										
										
										
											2014-08-27 14:08:26 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-05 19:50:09 -05:00
										 |  |  | 				// The limit was originally 128 but is now 512. | 
					
						
							| 
									
										
										
										
											2014-08-27 14:08:26 -04:00
										 |  |  | 				// Instead of rewriting the test cases above, adjust | 
					
						
							| 
									
										
										
										
											2015-05-01 09:36:18 -07:00
										 |  |  | 				// the first stack frame to use up the extra bytes. | 
					
						
							| 
									
										
										
										
											2014-08-27 14:08:26 -04:00
										 |  |  | 				if i == 0 { | 
					
						
							| 
									
										
										
										
											2014-12-05 19:50:09 -05:00
										 |  |  | 					size += 512 - 128 | 
					
						
							| 
									
										
										
										
											2015-05-01 09:36:18 -07:00
										 |  |  | 					// Noopt builds have a larger stackguard. | 
					
						
							|  |  |  | 					// See ../cmd/dist/buildruntime.go:stackGuardMultiplier | 
					
						
							|  |  |  | 					for _, s := range strings.Split(os.Getenv("GO_GCFLAGS"), " ") { | 
					
						
							|  |  |  | 						if s == "-N" { | 
					
						
							|  |  |  | 							size += 640 | 
					
						
							|  |  |  | 						} | 
					
						
							|  |  |  | 					} | 
					
						
							| 
									
										
										
										
											2014-08-27 14:08:26 -04:00
										 |  |  | 				} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-29 15:15:03 -04:00
										 |  |  | 				if size%ptrSize == 4 || goarch == "arm64" && size != 0 && (size+8)%16 != 0 { | 
					
						
							| 
									
										
										
										
											2014-04-16 22:08:00 -04:00
										 |  |  | 					continue TestCases | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 				nosplit := m[3] | 
					
						
							|  |  |  | 				body := m[4] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				if nosplit != "" { | 
					
						
							|  |  |  | 					nosplit = ",7" | 
					
						
							|  |  |  | 				} else { | 
					
						
							|  |  |  | 					nosplit = ",0" | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 				body = callRE.ReplaceAllString(body, "CALL ·$1(SB);") | 
					
						
							|  |  |  | 				body = callindRE.ReplaceAllString(body, "CALL REGISTER;") | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-09-16 17:39:55 -04:00
										 |  |  | 				fmt.Fprintf(&gobuf, "func %s()\n", name) | 
					
						
							| 
									
										
										
										
											2014-04-16 22:08:00 -04:00
										 |  |  | 				fmt.Fprintf(&buf, "TEXT ·%s(SB)%s,$%d-0\n\t%s\n\tRET\n\n", name, nosplit, size, body) | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-09-16 17:39:55 -04:00
										 |  |  | 		if err := ioutil.WriteFile(filepath.Join(dir, "asm.s"), buf.Bytes(), 0666); err != nil { | 
					
						
							|  |  |  | 			log.Fatal(err) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		if err := ioutil.WriteFile(filepath.Join(dir, "main.go"), gobuf.Bytes(), 0666); err != nil { | 
					
						
							|  |  |  | 			log.Fatal(err) | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2014-04-16 22:08:00 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		cmd := exec.Command("go", "build") | 
					
						
							|  |  |  | 		cmd.Dir = dir | 
					
						
							|  |  |  | 		output, err := cmd.CombinedOutput() | 
					
						
							|  |  |  | 		if err == nil { | 
					
						
							|  |  |  | 			nok++ | 
					
						
							|  |  |  | 			if reject { | 
					
						
							|  |  |  | 				bug() | 
					
						
							|  |  |  | 				fmt.Printf("accepted incorrectly:\n\t%s\n", indent(strings.TrimSpace(stanza))) | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} else { | 
					
						
							|  |  |  | 			nfail++ | 
					
						
							|  |  |  | 			if !reject { | 
					
						
							|  |  |  | 				bug() | 
					
						
							|  |  |  | 				fmt.Printf("rejected incorrectly:\n\t%s\n", indent(strings.TrimSpace(stanza))) | 
					
						
							|  |  |  | 				fmt.Printf("\n\tlinker output:\n\t%s\n", indent(string(output))) | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if !bugged && (nok == 0 || nfail == 0) { | 
					
						
							|  |  |  | 		bug() | 
					
						
							|  |  |  | 		fmt.Printf("not enough test cases run\n") | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func indent(s string) string { | 
					
						
							|  |  |  | 	return strings.Replace(s, "\n", "\n\t", -1) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | var bugged = false | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func bug() { | 
					
						
							|  |  |  | 	if !bugged { | 
					
						
							|  |  |  | 		bugged = true | 
					
						
							|  |  |  | 		fmt.Printf("BUG\n") | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } |