Home | History | Annotate | Download | only in patch
      1 test: add -target flag.
      2 
      3 --- test/run.go
      4 +++ test/run.go
      5 @@ -34,19 +34,19 @@ import (
      6  
      7  var (
      8  	verbose        = flag.Bool("v", false, "verbose. if set, parallelism is set to 1.")
      9  	keep           = flag.Bool("k", false, "keep. keep temporary directory.")
     10  	numParallel    = flag.Int("n", runtime.NumCPU(), "number of parallel tests to run")
     11  	summary        = flag.Bool("summary", false, "show summary of results")
     12  	showSkips      = flag.Bool("show_skips", false, "show skipped tests")
     13  	runSkips       = flag.Bool("run_skips", false, "run skipped tests (ignore skip and build tags)")
     14 -	linkshared     = flag.Bool("linkshared", false, "")
     15  	updateErrors   = flag.Bool("update_errors", false, "update error messages in test file based on compiler output")
     16  	runoutputLimit = flag.Int("l", defaultRunOutputLimit(), "number of parallel runoutput tests to run")
     17 +	target         = flag.String("target", "", "if non empty, use 'go_target' to compile test files and 'go_target_exec' to run the binaries")
     18  
     19  	shard  = flag.Int("shard", 0, "shard index to run. Only applicable if -shards is non-zero.")
     20  	shards = flag.Int("shards", 0, "number of shards. If 0, all tests are run. This is used by the continuous build.")
     21  )
     22  
     23  var (
     24  	goos, goarch string
     25  
     26 @@ -189,48 +189,49 @@ func goFiles(dir string) []string {
     27  	}
     28  	sort.Strings(names)
     29  	return names
     30  }
     31  
     32  type runCmd func(...string) ([]byte, error)
     33  
     34  func compileFile(runcmd runCmd, longname string, flags []string) (out []byte, err error) {
     35 -	cmd := []string{"go", "tool", "compile", "-e"}
     36 +	cmd := []string{findGoCmd(), "tool", "compile", "-e"}
     37  	cmd = append(cmd, flags...)
     38 -	if *linkshared {
     39 -		cmd = append(cmd, "-dynlink", "-installsuffix=dynlink")
     40 -	}
     41  	cmd = append(cmd, longname)
     42  	return runcmd(cmd...)
     43  }
     44  
     45  func compileInDir(runcmd runCmd, dir string, flags []string, names ...string) (out []byte, err error) {
     46 -	cmd := []string{"go", "tool", "compile", "-e", "-D", ".", "-I", "."}
     47 +	cmd := []string{findGoCmd(), "tool", "compile", "-e", "-D", ".", "-I", "."}
     48  	cmd = append(cmd, flags...)
     49 -	if *linkshared {
     50 -		cmd = append(cmd, "-dynlink", "-installsuffix=dynlink")
     51 -	}
     52  	for _, name := range names {
     53  		cmd = append(cmd, filepath.Join(dir, name))
     54  	}
     55  	return runcmd(cmd...)
     56  }
     57  
     58  func linkFile(runcmd runCmd, goname string) (err error) {
     59  	pfile := strings.Replace(goname, ".go", ".o", -1)
     60 -	cmd := []string{"go", "tool", "link", "-w", "-o", "a.exe", "-L", "."}
     61 -	if *linkshared {
     62 -		cmd = append(cmd, "-linkshared", "-installsuffix=dynlink")
     63 -	}
     64 -	cmd = append(cmd, pfile)
     65 -	_, err = runcmd(cmd...)
     66 +	_, err = runcmd(findGoCmd(), "tool", "link", "-w", "-o", "a.exe", "-L", ".", pfile)
     67  	return
     68  }
     69  
     70 +func goRun(runcmd runCmd, flags []string, goname string, args ...string) (out []byte, err error) {
     71 +	cmd := []string{findGoCmd(), "run", goGcflags()}
     72 +	if len(findExecCmd()) > 0 {
     73 +		cmd = append(cmd, "-exec")
     74 +		cmd = append(cmd, findExecCmd()...)
     75 +	}
     76 +	cmd = append(cmd, flags...)
     77 +	cmd = append(cmd, goname)
     78 +	cmd = append(cmd, args...)
     79 +	return runcmd(cmd...)
     80 +}
     81 +
     82  // skipError describes why a test was skipped.
     83  type skipError string
     84  
     85  func (s skipError) Error() string { return string(s) }
     86  
     87  func check(err error) {
     88  	if err != nil {
     89  		log.Fatal(err)
     90 @@ -590,18 +591,17 @@ func (t *test) run() {
     91  
     92  	long := filepath.Join(cwd, t.goFileName())
     93  	switch action {
     94  	default:
     95  		t.err = fmt.Errorf("unimplemented action %q", action)
     96  
     97  	case "errorcheck":
     98  		// TODO(gri) remove need for -C (disable printing of columns in error messages)
     99 -		cmdline := []string{"go", "tool", "compile", "-C", "-e", "-o", "a.o"}
    100 -		// No need to add -dynlink even if linkshared if we're just checking for errors...
    101 +		cmdline := []string{findGoCmd(), "tool", "compile", "-C", "-e", "-o", "a.o"}
    102  		cmdline = append(cmdline, flags...)
    103  		cmdline = append(cmdline, long)
    104  		out, err := runcmd(cmdline...)
    105  		if wantError {
    106  			if err == nil {
    107  				t.err = fmt.Errorf("compilation succeeded unexpectedly\n%s", out)
    108  				return
    109  			}
    110 @@ -704,17 +704,17 @@ func (t *test) run() {
    111  				}
    112  				if strings.Replace(string(out), "\r\n", "\n", -1) != t.expectedOutput() {
    113  					t.err = fmt.Errorf("incorrect output\n%s", out)
    114  				}
    115  			}
    116  		}
    117  
    118  	case "build":
    119 -		_, err := runcmd("go", "build", goGcflags(), "-o", "a.exe", long)
    120 +		_, err := runcmd(findGoCmd(), "build", goGcflags(), "-o", "a.exe", long)
    121  		if err != nil {
    122  			t.err = err
    123  		}
    124  
    125  	case "builddir":
    126  		// Build an executable from all the .go and .s files in a subdirectory.
    127  		useTmp = true
    128  		longdir := filepath.Join(cwd, t.goDirName())
    129 @@ -730,177 +730,132 @@ func (t *test) run() {
    130  			case ".go":
    131  				gos = append(gos, file)
    132  			case ".s":
    133  				asms = append(asms, file)
    134  			}
    135  
    136  		}
    137  		var objs []string
    138 -		cmd := []string{"go", "tool", "compile", "-e", "-D", ".", "-I", ".", "-o", "go.o"}
    139 +		cmd := []string{findGoCmd(), "tool", "compile", "-e", "-D", ".", "-I", ".", "-o", "go.o"}
    140  		if len(asms) > 0 {
    141  			cmd = append(cmd, "-asmhdr", "go_asm.h")
    142  		}
    143  		for _, file := range gos {
    144  			cmd = append(cmd, filepath.Join(longdir, file.Name()))
    145  		}
    146  		_, err := runcmd(cmd...)
    147  		if err != nil {
    148  			t.err = err
    149  			break
    150  		}
    151  		objs = append(objs, "go.o")
    152  		if len(asms) > 0 {
    153 -			cmd = []string{"go", "tool", "asm", "-e", "-I", ".", "-o", "asm.o"}
    154 +			cmd = []string{findGoCmd(), "tool", "asm", "-e", "-I", ".", "-o", "asm.o"}
    155  			for _, file := range asms {
    156  				cmd = append(cmd, filepath.Join(longdir, file.Name()))
    157  			}
    158  			_, err = runcmd(cmd...)
    159  			if err != nil {
    160  				t.err = err
    161  				break
    162  			}
    163  			objs = append(objs, "asm.o")
    164  		}
    165 -		cmd = []string{"go", "tool", "pack", "c", "all.a"}
    166 +		cmd = []string{findGoCmd(), "tool", "pack", "c", "all.a"}
    167  		cmd = append(cmd, objs...)
    168  		_, err = runcmd(cmd...)
    169  		if err != nil {
    170  			t.err = err
    171  			break
    172  		}
    173 -		cmd = []string{"go", "tool", "link", "all.a"}
    174 +		cmd = []string{findGoCmd(), "tool", "link", "all.a"}
    175  		_, err = runcmd(cmd...)
    176  		if err != nil {
    177  			t.err = err
    178  			break
    179  		}
    180  
    181  	case "buildrun": // build binary, then run binary, instead of go run. Useful for timeout tests where failure mode is infinite loop.
    182  		// TODO: not supported on NaCl
    183  		useTmp = true
    184 -		cmd := []string{"go", "build", goGcflags(), "-o", "a.exe"}
    185 -		if *linkshared {
    186 -			cmd = append(cmd, "-linkshared")
    187 -		}
    188 +		cmd := []string{findGoCmd(), "build", goGcflags(), "-o", "a.exe"}
    189  		longdirgofile := filepath.Join(filepath.Join(cwd, t.dir), t.gofile)
    190  		cmd = append(cmd, flags...)
    191  		cmd = append(cmd, longdirgofile)
    192  		out, err := runcmd(cmd...)
    193  		if err != nil {
    194  			t.err = err
    195  			return
    196  		}
    197 -		cmd = []string{"./a.exe"}
    198 +		cmd = []string{}
    199 +		if len(findExecCmd()) > 0 {
    200 +			cmd = append(cmd, findExecCmd()...)
    201 +		}
    202 +		cmd = append(cmd, "./a.exe")
    203  		out, err = runcmd(append(cmd, args...)...)
    204  		if err != nil {
    205  			t.err = err
    206  			return
    207  		}
    208  
    209  		if strings.Replace(string(out), "\r\n", "\n", -1) != t.expectedOutput() {
    210  			t.err = fmt.Errorf("incorrect output\n%s", out)
    211  		}
    212  
    213  	case "run":
    214  		useTmp = false
    215 -		var out []byte
    216 -		var err error
    217 -		if len(flags)+len(args) == 0 && goGcflags() == "" && !*linkshared {
    218 -			// If we're not using special go command flags,
    219 -			// skip all the go command machinery.
    220 -			// This avoids any time the go command would
    221 -			// spend checking whether, for example, the installed
    222 -			// package runtime is up to date.
    223 -			// Because we run lots of trivial test programs,
    224 -			// the time adds up.
    225 -			pkg := filepath.Join(t.tempDir, "pkg.a")
    226 -			if _, err := runcmd("go", "tool", "compile", "-o", pkg, t.goFileName()); err != nil {
    227 -				t.err = err
    228 -				return
    229 -			}
    230 -			exe := filepath.Join(t.tempDir, "test.exe")
    231 -			cmd := []string{"go", "tool", "link", "-s", "-w"}
    232 -			cmd = append(cmd, "-o", exe, pkg)
    233 -			if _, err := runcmd(cmd...); err != nil {
    234 -				t.err = err
    235 -				return
    236 -			}
    237 -			out, err = runcmd(append([]string{exe}, args...)...)
    238 -		} else {
    239 -			cmd := []string{"go", "run", goGcflags()}
    240 -			if *linkshared {
    241 -				cmd = append(cmd, "-linkshared")
    242 -			}
    243 -			cmd = append(cmd, flags...)
    244 -			cmd = append(cmd, t.goFileName())
    245 -			out, err = runcmd(append(cmd, args...)...)
    246 -		}
    247 +		out, err := goRun(runcmd, flags, t.goFileName(), args...)
    248  		if err != nil {
    249  			t.err = err
    250  			return
    251  		}
    252  		if strings.Replace(string(out), "\r\n", "\n", -1) != t.expectedOutput() {
    253  			t.err = fmt.Errorf("incorrect output\n%s", out)
    254  		}
    255  
    256  	case "runoutput":
    257  		rungatec <- true
    258  		defer func() {
    259  			<-rungatec
    260  		}()
    261  		useTmp = false
    262 -		cmd := []string{"go", "run", goGcflags()}
    263 -		if *linkshared {
    264 -			cmd = append(cmd, "-linkshared")
    265 -		}
    266 -		cmd = append(cmd, t.goFileName())
    267 -		out, err := runcmd(append(cmd, args...)...)
    268 +		out, err := goRun(runcmd, nil, t.goFileName(), args...)
    269  		if err != nil {
    270  			t.err = err
    271  			return
    272  		}
    273  		tfile := filepath.Join(t.tempDir, "tmp__.go")
    274  		if err := ioutil.WriteFile(tfile, out, 0666); err != nil {
    275  			t.err = fmt.Errorf("write tempfile:%s", err)
    276  			return
    277  		}
    278 -		cmd = []string{"go", "run", goGcflags()}
    279 -		if *linkshared {
    280 -			cmd = append(cmd, "-linkshared")
    281 -		}
    282 -		cmd = append(cmd, tfile)
    283 -		out, err = runcmd(cmd...)
    284 +		out, err = goRun(runcmd, nil, tfile)
    285  		if err != nil {
    286  			t.err = err
    287  			return
    288  		}
    289  		if string(out) != t.expectedOutput() {
    290  			t.err = fmt.Errorf("incorrect output\n%s", out)
    291  		}
    292  
    293  	case "errorcheckoutput":
    294  		useTmp = false
    295 -		cmd := []string{"go", "run", goGcflags()}
    296 -		if *linkshared {
    297 -			cmd = append(cmd, "-linkshared")
    298 -		}
    299 -		cmd = append(cmd, t.goFileName())
    300 -		out, err := runcmd(append(cmd, args...)...)
    301 +		out, err := goRun(runcmd, nil, t.goFileName(), args...)
    302  		if err != nil {
    303  			t.err = err
    304  			return
    305  		}
    306  		tfile := filepath.Join(t.tempDir, "tmp__.go")
    307  		err = ioutil.WriteFile(tfile, out, 0666)
    308  		if err != nil {
    309  			t.err = fmt.Errorf("write tempfile:%s", err)
    310  			return
    311  		}
    312 -		cmdline := []string{"go", "tool", "compile", "-e", "-o", "a.o"}
    313 +		cmdline := []string{findGoCmd(), "tool", "compile", "-e", "-o", "a.o"}
    314  		cmdline = append(cmdline, flags...)
    315  		cmdline = append(cmdline, tfile)
    316  		out, err = runcmd(cmdline...)
    317  		if wantError {
    318  			if err == nil {
    319  				t.err = fmt.Errorf("compilation succeeded unexpectedly\n%s", out)
    320  				return
    321  			}
    322 @@ -917,26 +872,37 @@ func (t *test) run() {
    323  
    324  var execCmd []string
    325  
    326  func findExecCmd() []string {
    327  	if execCmd != nil {
    328  		return execCmd
    329  	}
    330  	execCmd = []string{} // avoid work the second time
    331 +	if *target != "" {
    332 +		execCmd = []string{"go_" + *target + "_exec"}
    333 +		return execCmd
    334 +	}
    335  	if goos == runtime.GOOS && goarch == runtime.GOARCH {
    336  		return execCmd
    337  	}
    338  	path, err := exec.LookPath(fmt.Sprintf("go_%s_%s_exec", goos, goarch))
    339  	if err == nil {
    340  		execCmd = []string{path}
    341  	}
    342  	return execCmd
    343  }
    344  
    345 +func findGoCmd() string {
    346 +	if *target != "" {
    347 +		return "go_" + *target
    348 +	}
    349 +	return "go"
    350 +}
    351 +
    352  func (t *test) String() string {
    353  	return filepath.Join(t.dir, t.gofile)
    354  }
    355  
    356  func (t *test) makeTempDir() {
    357  	var err error
    358  	t.tempDir, err = ioutil.TempDir("", "")
    359  	check(err)
    360