blob: db6b691b2cc7563d88bbb86cc3cf8ce659a9db90 [file] [log] [blame]
// Copyright 2021 The Dawn & Tint Authors
//
// Redistribution and use in source and binary forms, with or without
// modification, are permitted provided that the following conditions are met:
//
// 1. Redistributions of source code must retain the above copyright notice, this
// list of conditions and the following disclaimer.
//
// 2. Redistributions in binary form must reproduce the above copyright notice,
// this list of conditions and the following disclaimer in the documentation
// and/or other materials provided with the distribution.
//
// 3. Neither the name of the copyright holder nor the names of its
// contributors may be used to endorse or promote products derived from
// this software without specific prior written permission.
//
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
// AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
// IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
// DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE
// FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
// DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
// SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
// CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
// OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
// tests runs tint against a number of test shaders checking for expected behavior
package main
import (
"bufio"
"context"
"crypto/sha256"
"encoding/json"
"flag"
"fmt"
"io"
"os"
"os/exec"
"path"
"path/filepath"
"regexp"
"runtime"
"sort"
"strings"
"time"
"unicode/utf8"
"dawn.googlesource.com/dawn/tools/src/container"
"dawn.googlesource.com/dawn/tools/src/fileutils"
"dawn.googlesource.com/dawn/tools/src/glob"
"dawn.googlesource.com/dawn/tools/src/match"
"dawn.googlesource.com/dawn/tools/src/oswrapper"
"dawn.googlesource.com/dawn/tools/src/transform"
"github.com/fatih/color"
"github.com/sergi/go-diff/diffmatchpatch"
"golang.org/x/term"
)
type outputFormat string
const (
testTimeout = 2 * time.Minute
glsl = outputFormat("glsl")
hlslFXC = outputFormat("hlsl-fxc")
hlslDXC = outputFormat("hlsl-dxc")
msl = outputFormat("msl")
spvasm = outputFormat("spvasm")
wgsl = outputFormat("wgsl")
)
// allOutputFormats holds all the supported outputFormats
var allOutputFormats = []outputFormat{wgsl, spvasm, msl, hlslDXC, hlslFXC, glsl}
// The default non-flag arguments to the command
var defaultArgs = []string{"test/tint"}
// The globs automatically appended if a glob argument is a directory
var directoryGlobs = []string{
"**.wgsl",
"**.spvasm",
"**.spv",
}
// Directories we don't generate expected PASS result files for.
// These directories contain large corpora of tests for which the generated code
// is uninteresting.
// These paths use unix-style slashes and do not contain the '/test/tint' prefix.
func dirsWithNoPassExpectations(fsReader oswrapper.FilesystemReader) []string {
dawnRoot := fileutils.DawnRoot(fsReader)
return []string{
filepath.ToSlash(dawnRoot) + "/test/tint/benchmark/",
filepath.ToSlash(dawnRoot) + "/test/tint/unittest/",
filepath.ToSlash(dawnRoot) + "/test/tint/vk-gl-cts/",
}
}
func main() {
if err := run(oswrapper.GetRealOSWrapper()); err != nil {
fmt.Println(err)
os.Exit(1)
}
}
func showUsage() {
fmt.Printf(`
tests runs tint against a number of test shaders checking for expected behavior
usage:
tests [flags...] [globs...]
[globs] a list of project-root relative file globs, directory or file paths
of test cases.
A file path will be added to the test list.
A directory will automatically expand to the globs:
%v
Globs will include all test files that match the glob, but exclude
those that match the --ignore flag.
If omitted, defaults to: %v
optional flags:`,
transform.SliceNoErr(directoryGlobs, func(in string) string { return fmt.Sprintf("'<dir>/%v'", in) }),
transform.SliceNoErr(defaultArgs, func(in string) string { return fmt.Sprintf("'%v'", in) }))
flag.PrintDefaults()
fmt.Println(``)
os.Exit(1)
}
// TODO(crbug.com/344014313): Add unittests once fileutils is converted to
// support dependency injection. This should also be broken up into multiple
// functions.
func run(fsReaderWriter oswrapper.FilesystemReaderWriter) error {
terminalWidth, _, err := term.GetSize(int(os.Stdout.Fd()))
if err != nil {
terminalWidth = 0
}
var formatList, ignore, dxcPath, fxcPath, tintPath, xcrunPath string
var maxTableWidth int
var server, useIrReader bool
numCPU := runtime.NumCPU()
verbose, generateExpected, generateSkip := false, false, false
flag.StringVar(&formatList, "format", "all", "comma separated list of formats to emit. Possible values are: all, wgsl, spvasm, msl, hlsl, hlsl-dxc, hlsl-fxc, glsl")
flag.StringVar(&ignore, "ignore", "**.expected.*", "files to ignore in globs")
flag.StringVar(&dxcPath, "dxcompiler", "", "path to DXC DLL for validating HLSL output")
flag.StringVar(&fxcPath, "fxc", "", "path to FXC DLL for validating HLSL output")
flag.StringVar(&tintPath, "tint", defaultTintPath(fsReaderWriter), "path to the tint executable")
flag.StringVar(&xcrunPath, "xcrun", "", "path to xcrun executable for validating MSL output")
flag.BoolVar(&verbose, "verbose", false, "print all run tests, including rows that all pass")
flag.BoolVar(&generateExpected, "generate-expected", false, "create or update all expected outputs")
flag.BoolVar(&generateSkip, "generate-skip", false, "create or update all expected outputs that fail with SKIP")
flag.BoolVar(&server, "server", true, "run Tint in server mode")
flag.BoolVar(&useIrReader, "use-ir-reader", true, "force use of IR SPIR-V Reader")
flag.IntVar(&numCPU, "j", numCPU, "maximum number of concurrent threads to run tests")
flag.IntVar(&maxTableWidth, "table-width", terminalWidth, "maximum width of the results table")
flag.Usage = showUsage
flag.Parse()
// Check the executable can be found and actually is executable
if !fileutils.IsExe(tintPath, fsReaderWriter) {
fmt.Fprintln(os.Stderr, "tint executable not found, please specify with --tint")
showUsage()
}
// Apply default args, if not provided
args := flag.Args()
if len(args) == 0 {
args = defaultArgs
}
filePredicate := func(s string) bool { return true }
if m, err := match.New(ignore); err == nil {
filePredicate = func(s string) bool { return !m(s) }
} else {
return fmt.Errorf("failed to parse --ignore: %w", err)
}
// Transform args to globs, find the rootPath directory
absFiles := []string{}
rootPath := ""
globs := []string{}
for _, arg := range args {
if len(arg) > 1 && arg[0:2] == "--" {
return fmt.Errorf("unexpected flag after globs: %s", arg)
}
// Make absolute
if !filepath.IsAbs(arg) {
arg = filepath.Join(fileutils.DawnRoot(fsReaderWriter), arg)
}
switch {
case fileutils.IsDir(arg, fsReaderWriter):
// Argument is to a directory, expand out to N globs
for _, glob := range directoryGlobs {
globs = append(globs, path.Join(arg, glob))
}
case fileutils.IsFile(arg, fsReaderWriter):
// Argument is a file, append to absFiles
absFiles = append(absFiles, arg)
default:
globs = append(globs, arg)
}
if rootPath == "" {
rootPath = filepath.Dir(arg)
} else {
rootPath = fileutils.CommonRootDir(rootPath, arg)
}
}
// Remove any absFiles that should be ignored
absFiles = transform.Filter(absFiles, filePredicate)
// Glob the absFiles to test
for _, g := range globs {
globFiles, err := glob.Glob(g, fsReaderWriter)
if err != nil {
return fmt.Errorf("Failed to glob files: %w", err)
}
filtered := transform.Filter(globFiles, filePredicate)
normalized := transform.SliceNoErr(filtered, filepath.ToSlash)
absFiles = append(absFiles, normalized...)
}
// Ensure the files are sorted (globbing should do this, but why not)
sort.Strings(absFiles)
// Parse --format into a list of outputFormat
formats := []outputFormat{}
if formatList == "all" {
formats = allOutputFormats
} else {
for _, f := range strings.Split(formatList, ",") {
parsed, err := parseOutputFormat(strings.TrimSpace(f))
if err != nil {
return err
}
formats = append(formats, parsed...)
}
}
defaultMSLExe := "xcrun"
if runtime.GOOS == "windows" {
defaultMSLExe = "metal.exe"
}
defaultDXCDll := "libdxcompiler.so"
if runtime.GOOS == "windows" {
defaultDXCDll = "dxcompiler.dll"
} else if runtime.GOOS == "darwin" {
defaultDXCDll = "libdxcompiler.dylib"
}
toolchainHash := sha256.New()
// If explicit verification compilers have been specified, check they exist.
// Otherwise, look on PATH for them, but don't error if they cannot be found.
for _, tool := range []struct {
name string
lang string
path *string
}{
{defaultDXCDll, "hlsl-dxc", &dxcPath},
{"d3dcompiler_47.dll", "hlsl-fxc", &fxcPath},
{defaultMSLExe, "msl", &xcrunPath},
} {
if *tool.path == "" {
// Look first in the directory of the tint executable
p, err := exec.LookPath(filepath.Join(filepath.Dir(tintPath), tool.name))
if err == nil && fileutils.IsExe(p, fsReaderWriter) {
*tool.path = p
} else {
// Look in PATH
p, err := exec.LookPath(tool.name)
if err == nil && fileutils.IsExe(p, fsReaderWriter) {
*tool.path = p
}
}
} else if !fileutils.IsExe(*tool.path, fsReaderWriter) {
return fmt.Errorf("%v not found at '%v'", tool.name, *tool.path)
}
color.Set(color.FgCyan)
fmt.Printf("%-8s", tool.lang)
color.Unset()
fmt.Printf(" validation ")
if *tool.path != "" {
fmt.Printf("ENABLED (%s)", *tool.path)
} else {
color.Set(color.FgRed)
fmt.Printf("DISABLED")
}
color.Unset()
fmt.Println()
toolchainHash.Write([]byte(tool.name))
if s, err := fsReaderWriter.Stat(*tool.path); err == nil {
toolchainHash.Write([]byte(s.ModTime().String()))
toolchainHash.Write([]byte(fmt.Sprint(s.Size())))
}
}
fmt.Println()
validationCache := loadValidationCache(
fmt.Sprintf("%x", toolchainHash.Sum(nil)), verbose, fsReaderWriter)
defer saveValidationCache(validationCache, fsReaderWriter)
// Build the list of results.
// These hold the chans used to report the job results.
results := make([]map[outputFormat]chan status, len(absFiles))
for i := range absFiles {
fileResults := map[outputFormat]chan status{}
for _, format := range formats {
fileResults[format] = make(chan status, 1)
}
results[i] = fileResults
}
pendingJobs := make(chan job, 256)
// Spawn numCPU job runners...
runCfg := runConfig{
tintPath: tintPath,
dxcPath: dxcPath,
fxcPath: fxcPath,
xcrunPath: xcrunPath,
generateExpected: generateExpected,
generateSkip: generateSkip,
validationCache: validationCache,
server: server,
}
for cpu := 0; cpu < numCPU; cpu++ {
go func() {
tintServer := tintServerState{}
for job := range pendingJobs {
job.run(runCfg, fsReaderWriter, &tintServer)
}
}()
}
// Issue the jobs...
go func() {
for i, file := range absFiles { // For each test file...
flags, err := parseFlags(file, fsReaderWriter)
if err != nil {
fmt.Println(file+" error:", err)
continue
}
for _, format := range formats { // For each output format...
j := job{
file: file,
format: format,
result: results[i][format],
}
for _, f := range flags {
if f.formats.Contains(format) {
j.flags = append(j.flags, f.flags...)
}
}
pendingJobs <- j
}
}
close(pendingJobs)
}()
type failure struct {
file string
format outputFormat
err error
}
type stats struct {
numTests, numPass, numSkip, numFail, numValidTests, numInvalid int
timeTaken time.Duration
}
// Statistics per output format
statsByFmt := map[outputFormat]*stats{}
for _, format := range formats {
statsByFmt[format] = &stats{}
}
// Make file paths relative to rootPath, if possible
relFiles := transform.GoSliceNoErr(absFiles, func(path string) string {
path = filepath.ToSlash(path) // Normalize
if rel, err := filepath.Rel(rootPath, path); err == nil {
return rel
}
return path
})
// Print the table of file x format and gather per-format stats
failures := []failure{}
filenameColumnWidth := maxStringLen(relFiles)
// Calculate the table width
tableWidth := filenameColumnWidth + 3
for _, format := range formats {
tableWidth += formatWidth(format) + 3
}
// Reduce filename column width if too big
if tableWidth > maxTableWidth {
filenameColumnWidth -= tableWidth - maxTableWidth
if filenameColumnWidth < 20 {
filenameColumnWidth = 20
}
}
red := color.New(color.FgRed)
green := color.New(color.FgGreen)
yellow := color.New(color.FgYellow)
cyan := color.New(color.FgCyan)
printFormatsHeader := func() {
fmt.Print(strings.Repeat(" ", filenameColumnWidth))
fmt.Print(" ┃ ")
for _, format := range formats {
cyan.Print(alignCenter(format, formatWidth(format)))
fmt.Print(" │ ")
}
fmt.Println()
}
printHorizontalLine := func() {
fmt.Print(strings.Repeat("━", filenameColumnWidth))
fmt.Print("━╋━")
for _, format := range formats {
fmt.Print(strings.Repeat("━", formatWidth(format)))
fmt.Print("━┿━")
}
fmt.Println()
}
fmt.Println()
printFormatsHeader()
printHorizontalLine()
newKnownGood := knownGoodHashes{}
for i, file := range relFiles {
absFile := absFiles[i]
results := results[i]
row := &strings.Builder{}
rowAllPassed := true
filenameLength := utf8.RuneCountInString(file)
shortFile := file
if filenameLength > filenameColumnWidth {
shortFile = "..." + file[filenameLength-filenameColumnWidth+3:]
}
fmt.Fprint(row, alignRight(shortFile, filenameColumnWidth))
fmt.Fprint(row, " ┃ ")
for _, format := range formats {
columnWidth := formatWidth(format)
result := <-results[format]
// Update the known-good hashes
newKnownGood[fileAndFormat{absFile, format}] = result.passHashes
// Update stats
stats := statsByFmt[format]
stats.numTests++
stats.timeTaken += result.timeTaken
if err := result.err; err != nil {
failures = append(failures, failure{
file: file, format: format, err: err,
})
}
switch result.code {
case pass:
green.Fprint(row, alignCenter("PASS", columnWidth))
stats.numPass++
case fail:
red.Fprint(row, alignCenter("FAIL", columnWidth))
rowAllPassed = false
stats.numFail++
case skip:
yellow.Fprint(row, alignCenter("SKIP", columnWidth))
rowAllPassed = false
stats.numSkip++
case invalid:
yellow.Fprint(row, alignCenter("INVALID", columnWidth))
rowAllPassed = false
stats.numInvalid++
default:
fmt.Fprint(row, alignCenter(result.code, columnWidth))
rowAllPassed = false
}
fmt.Fprint(row, " │ ")
}
if verbose || !rowAllPassed {
fmt.Fprintln(color.Output, row)
}
}
// Update the validation cache known-good hashes.
// This has to be done after all the results have been collected to avoid
// concurrent access on the map.
for ff, hashes := range newKnownGood {
if len(newKnownGood) > 0 {
validationCache.knownGood[ff] = hashes
} else {
delete(validationCache.knownGood, ff)
}
}
printHorizontalLine()
printFormatsHeader()
printHorizontalLine()
printStat := func(col *color.Color, name string, num func(*stats) int) {
row := &strings.Builder{}
anyNonZero := false
for _, format := range formats {
columnWidth := formatWidth(format)
count := num(statsByFmt[format])
if count > 0 {
col.Fprint(row, alignLeft(count, columnWidth))
anyNonZero = true
} else {
fmt.Fprint(row, alignLeft(count, columnWidth))
}
fmt.Fprint(row, " │ ")
}
if !anyNonZero {
return
}
col.Print(alignRight(name, filenameColumnWidth))
fmt.Print(" ┃ ")
fmt.Fprintln(color.Output, row)
col.Print(strings.Repeat(" ", filenameColumnWidth))
fmt.Print(" ┃ ")
for _, format := range formats {
columnWidth := formatWidth(format)
stats := statsByFmt[format]
count := num(stats)
percent := percentage(count, stats.numTests)
if count > 0 {
col.Print(alignRight(percent, columnWidth))
} else {
fmt.Print(alignRight(percent, columnWidth))
}
fmt.Print(" │ ")
}
fmt.Println()
}
printStat(green, "PASS", func(s *stats) int { return s.numPass })
printStat(yellow, "SKIP", func(s *stats) int { return s.numSkip })
printStat(yellow, "INVALID", func(s *stats) int { return s.numInvalid })
printStat(red, "FAIL", func(s *stats) int { return s.numFail })
cyan.Print(alignRight("TIME", filenameColumnWidth))
fmt.Print(" ┃ ")
for _, format := range formats {
timeTaken := printDuration(statsByFmt[format].timeTaken)
cyan.Print(alignLeft(timeTaken, formatWidth(format)))
fmt.Print(" │ ")
}
fmt.Println()
for _, f := range failures {
color.Set(color.FgBlue)
fmt.Printf("%s ", f.file)
color.Set(color.FgCyan)
fmt.Printf("%s ", f.format)
color.Set(color.FgRed)
fmt.Println("FAIL")
color.Unset()
fmt.Println(indent(f.err.Error(), 4))
}
if len(failures) > 0 {
fmt.Println()
}
allStats := stats{}
for _, format := range formats {
stats := statsByFmt[format]
allStats.numTests += stats.numTests
allStats.numPass += stats.numPass
allStats.numSkip += stats.numSkip
allStats.numInvalid += stats.numInvalid
allStats.numFail += stats.numFail
}
// Remove the invalid tests from the test count
allStats.numValidTests = allStats.numTests - allStats.numInvalid
fmt.Printf("%d tests run, ", allStats.numTests)
fmt.Printf("%d valid tests", allStats.numValidTests)
if allStats.numPass > 0 {
fmt.Printf(", ")
color.Set(color.FgGreen)
fmt.Printf("%d tests pass", allStats.numPass)
color.Unset()
} else {
fmt.Printf(", %d tests pass", allStats.numPass)
}
if allStats.numSkip > 0 {
fmt.Printf(", ")
color.Set(color.FgYellow)
fmt.Printf("%d tests skipped", allStats.numSkip)
color.Unset()
} else {
fmt.Printf(", %d tests skipped", allStats.numSkip)
}
if allStats.numInvalid > 0 {
fmt.Printf(", ")
color.Set(color.FgYellow)
fmt.Printf("%d invalid tests", allStats.numInvalid)
color.Unset()
} else {
fmt.Printf(", %d invalid tests", allStats.numInvalid)
}
if allStats.numFail > 0 {
fmt.Printf(", ")
color.Set(color.FgRed)
fmt.Printf("%d tests failed", allStats.numFail)
color.Unset()
} else {
fmt.Printf(", %d tests failed", allStats.numFail)
}
fmt.Println()
fmt.Println()
if allStats.numFail > 0 {
return fmt.Errorf("%v tests failed", allStats.numFail)
}
return nil
}
// Structures to hold the results of the tests
type statusCode string
const (
fail statusCode = "FAIL"
pass statusCode = "PASS"
skip statusCode = "SKIP"
invalid statusCode = "INVALID"
)
type status struct {
code statusCode
err error
timeTaken time.Duration
passHashes []string
}
type job struct {
file string
flags []string
format outputFormat
result chan status
}
type runConfig struct {
tintPath string
dxcPath string
fxcPath string
xcrunPath string
generateExpected bool
generateSkip bool
validationCache validationCache
server bool
}
type tintServerState struct {
cmd *exec.Cmd
stdin io.WriteCloser
stdout, stderr io.ReadCloser
}
// Skips the path portion of FXC warning/error strings, matching the rest.
// We use this to scrub the path to avoid generating needless diffs.
var reFXCErrorStringHash = regexp.MustCompile(`(?:.*?)(\(.*?\): (?:warning|error).*)`)
// TODO(crbug.com/344014313): Split this up into multiple functions and add
// unittest coverage for functions that can be tested.
func (j job) run(cfg runConfig, fsReaderWriter oswrapper.FilesystemReaderWriter, tintServer *tintServerState) {
j.result <- func() status {
// expectedFilePath is the path to the expected output file for the given test
expectedFilePath := j.file + ".expected."
// Only attempt to generate WGSL for SPVASM input files
if strings.HasSuffix(j.file, ".spvasm") && j.format != wgsl {
return status{code: skip, timeTaken: 0}
}
switch j.format {
case hlslDXC:
expectedFilePath += "dxc.hlsl"
case hlslFXC:
expectedFilePath += "fxc.hlsl"
default:
expectedFilePath += string(j.format)
}
// Is there an expected output file? If so, load it.
expected, expectedFileExists := "", false
if content, err := fsReaderWriter.ReadFile(expectedFilePath); err == nil {
expected = string(content)
expectedFileExists = true
}
isSkipTest := false
if strings.HasPrefix(expected, "SKIP") { // Special SKIP token
isSkipTest = true
}
isSkipInvalidTest := false
if strings.HasPrefix(expected, "SKIP: INVALID") { // Special invalid test case token
isSkipInvalidTest = true
}
isSkipTimeoutTest := false
if strings.HasPrefix(expected, "SKIP: TIMEOUT") { // Special timeout test case token
isSkipTimeoutTest = true
}
// If the test is known to fail and we are not regenerating expectations, just skip the test.
if isSkipTest && !cfg.generateExpected && !cfg.generateSkip {
if isSkipInvalidTest {
return status{code: invalid, timeTaken: 0}
} else {
return status{code: skip, timeTaken: 0}
}
}
expected = strings.ReplaceAll(expected, "\r\n", "\n")
outputFormat := strings.Split(string(j.format), "-")[0] // 'hlsl-fxc' -> 'hlsl', etc.
if j.format == hlslFXC {
// Emit HLSL specifically for FXC
outputFormat += "-fxc"
}
args := []string{
j.file,
"--format", outputFormat,
"--print-hash",
}
// Append any skip-hashes, if they're found.
if j.format != "wgsl" { // Don't skip 'wgsl' as this 'toolchain' is ever changing.
if skipHashes := cfg.validationCache.knownGood[fileAndFormat{j.file, j.format}]; len(skipHashes) > 0 {
args = append(args, "--skip-hash", strings.Join(skipHashes, ","))
}
}
// Can we validate?
validate := false
switch j.format {
case wgsl:
args = append(args, "--validate") // wgsl validation uses Tint, so is always available
validate = true
case spvasm, glsl:
args = append(args, "--validate") // spirv-val and glslang are statically linked, always available
validate = true
case hlslDXC:
if cfg.dxcPath != "" {
args = append(args, "--dxc", cfg.dxcPath)
validate = true
}
case hlslFXC:
if cfg.fxcPath != "" {
args = append(args, "--fxc", cfg.fxcPath)
validate = true
}
case msl:
if cfg.xcrunPath != "" {
args = append(args, "--xcrun", cfg.xcrunPath)
validate = true
}
default:
panic("unknown format: " + j.format)
}
// Invoke the compiler...
ok := false
var out string
args = append(args, j.flags...)
timedOut := false
start := time.Now()
if cfg.server {
ok, out, timedOut = invokeWithServer(cfg.tintPath, tintServer, args...)
} else {
ok, out, timedOut = invokeWithoutServer(cfg.tintPath, args...)
}
timeTaken := time.Since(start)
out = strings.ReplaceAll(out, "\r\n", "\n")
out = strings.ReplaceAll(out, filepath.ToSlash(fileutils.DawnRoot(fsReaderWriter)), "<dawn>")
out, hashes := extractValidationHashes(out)
matched := expected == "" || expected == out
canEmitPassExpectationFile := true
for _, noPass := range dirsWithNoPassExpectations(fsReaderWriter) {
if strings.HasPrefix(filepath.ToSlash(j.file), noPass) {
canEmitPassExpectationFile = false
break
}
}
saveExpectedFile := func(path string, content string) error {
return fsReaderWriter.WriteFile(path, []byte(content), 0666)
}
// Do not update expected if test is marked as SKIP: TIMEOUT
if ok && cfg.generateExpected && !isSkipTimeoutTest && (validate || !isSkipTest) {
// User requested to update PASS expectations, and test passed.
if canEmitPassExpectationFile {
saveExpectedFile(expectedFilePath, out)
} else if expectedFileExists {
// Test lives in a directory where we do not want to save PASS
// files, and there already exists an expectation file. Test has
// likely started passing. Delete the old expectation.
fsReaderWriter.Remove(expectedFilePath)
}
matched = true // test passed and matched expectations
}
var skip_str string = "FAILED"
if isSkipInvalidTest {
skip_str = "INVALID"
}
if timedOut {
skip_str = "TIMEOUT"
}
passed := ok && (matched || isSkipTimeoutTest)
if !passed {
if j.format == hlslFXC {
out = reFXCErrorStringHash.ReplaceAllString(out, `<scrubbed_path>${1}`)
}
}
switch {
case passed:
// Test passed
return status{code: pass, timeTaken: timeTaken, passHashes: hashes}
// --- Below this point the test has failed ---
case isSkipTest:
// Do not update expected if timeout test actually timed out.
if cfg.generateSkip && !(isSkipTimeoutTest && timedOut) {
saveExpectedFile(expectedFilePath, "SKIP: "+skip_str+"\n\n"+out)
}
if isSkipInvalidTest {
return status{code: invalid, timeTaken: timeTaken}
} else {
return status{code: skip, timeTaken: timeTaken}
}
case !ok:
// Compiler returned non-zero exit code
if cfg.generateSkip {
saveExpectedFile(expectedFilePath, "SKIP: "+skip_str+"\n\n"+out)
}
err := fmt.Errorf("%s", out)
return status{code: fail, err: err, timeTaken: timeTaken}
default:
// Compiler returned zero exit code, or output was not as expected
if cfg.generateSkip {
saveExpectedFile(expectedFilePath, "SKIP: "+skip_str+"\n\n"+out)
}
// Expected output did not match
dmp := diffmatchpatch.New()
diff := dmp.DiffPrettyText(dmp.DiffMain(expected, out, true))
err := fmt.Errorf(`Output was not as expected
--------------------------------------------------------------------------------
-- Expected: --
--------------------------------------------------------------------------------
%s
--------------------------------------------------------------------------------
-- Got: --
--------------------------------------------------------------------------------
%s
--------------------------------------------------------------------------------
-- Diff: --
--------------------------------------------------------------------------------
%s`,
expected, out, diff)
return status{code: fail, err: err, timeTaken: timeTaken}
}
}()
}
var reValidationHash = regexp.MustCompile(`<<HASH: ([^>]*)>>\n`)
// Parses and returns the validation hashes emitted by tint, or an empty string
// if the hash wasn't found, along with the input string with the validation
// hashes removed.
func extractValidationHashes(in string) (out string, hashes []string) {
matches := reValidationHash.FindAllStringSubmatch(in, -1)
if matches == nil {
return in, nil
}
out = in
for _, match := range matches {
out = strings.ReplaceAll(out, match[0], "")
hashes = append(hashes, match[1])
}
return out, hashes
}
// indent returns the string 's' indented with 'n' whitespace characters
func indent(s string, n int) string {
tab := strings.Repeat(" ", n)
return tab + strings.ReplaceAll(s, "\n", "\n"+tab)
}
// alignLeft returns the string of 'val' padded so that it is aligned left in
// a column of the given width
func alignLeft(val interface{}, width int) string {
s := fmt.Sprint(val)
padding := width - utf8.RuneCountInString(s)
if padding < 0 {
return s
}
return s + strings.Repeat(" ", padding)
}
// alignCenter returns the string of 'val' padded so that it is centered in a
// column of the given width.
func alignCenter(val interface{}, width int) string {
s := fmt.Sprint(val)
padding := width - utf8.RuneCountInString(s)
if padding < 0 {
return s
}
return strings.Repeat(" ", padding/2) + s + strings.Repeat(" ", (padding+1)/2)
}
// alignRight returns the string of 'val' padded so that it is aligned right in
// a column of the given width
func alignRight(val interface{}, width int) string {
s := fmt.Sprint(val)
padding := width - utf8.RuneCountInString(s)
if padding < 0 {
return s
}
return strings.Repeat(" ", padding) + s
}
// maxStringLen returns the maximum number of runes found in all the strings in
// 'l'
func maxStringLen(l []string) int {
max := 0
for _, s := range l {
if c := utf8.RuneCountInString(s); c > max {
max = c
}
}
return max
}
// formatWidth returns the width in runes for the outputFormat column 'b'
func formatWidth(b outputFormat) int {
const min = 6
c := utf8.RuneCountInString(string(b))
if c < min {
return min
}
return c
}
// percentage returns the percentage of n out of total as a string
func percentage(n, total int) string {
if total == 0 {
return "-"
}
f := float64(n) / float64(total)
return fmt.Sprintf("%.1f%c", f*100.0, '%')
}
// invokeWithServer runs a test with the provided arguments using Tint in server mode
func invokeWithServer(tintPath string, tintServer *tintServerState, args ...string) (ok bool, output string, timedOut bool) {
// Start the Tint server if it is not currently running.
if tintServer.cmd == nil {
tintServer.cmd = exec.CommandContext(context.Background(), tintPath, "--server")
var err error
// Set up pipes for stdin, stdout, and stderr.
tintServer.stdin, err = tintServer.cmd.StdinPipe()
if err != nil {
return false, err.Error(), false
}
tintServer.stdout, err = tintServer.cmd.StdoutPipe()
if err != nil {
return false, err.Error(), false
}
tintServer.stderr, err = tintServer.cmd.StderrPipe()
if err != nil {
return false, err.Error(), false
}
// Start the server.
if err = tintServer.cmd.Start(); err != nil {
return false, err.Error(), false
}
}
ctx, cancel := context.WithTimeout(context.Background(), testTimeout)
defer cancel()
// Send the test to the server inside a goroutine so that we can timeout if needed.
type testResult struct {
output string
ok bool
}
result := make(chan testResult, 1)
go func() {
// Send the test arguments to the Tint server.
_, inErr := tintServer.stdin.Write([]byte("\"" + strings.Join(args, "\" \"") + "\"\n"))
// Read from stdout and stderr until the next null character.
// Perform these as two asynchronous operations to prevent buffering of large output from
// blocking progress.
type readResult struct {
str string
err error
}
read := func(stream io.ReadCloser) chan readResult {
resultChannel := make(chan readResult, 1)
go func() {
reader := bufio.NewReader(stream)
result, err := reader.ReadString(0)
resultChannel <- readResult{strings.TrimSuffix(result, "\x00"), err}
}()
return resultChannel
}
stdoutChannel := read(tintServer.stdout)
stderrChannel := read(tintServer.stderr)
// Read the results from the channels.
stdoutResult := <-stdoutChannel
stderrResult := <-stderrChannel
stdoutStr, outErr := stdoutResult.str, stdoutResult.err
stderrStr, errErr := stderrResult.str, stderrResult.err
str := stderrStr + stdoutStr
result <- testResult{
output: str,
ok: inErr == nil && outErr == nil && errErr == nil,
}
}()
select {
case <-ctx.Done():
// The test timed out, so we need to kill the server.
// Manually close the pipes too so that the goroutine exits.
tintServer.stdout.Close()
tintServer.stderr.Close()
tintServer.cmd.Process.Kill()
tintServer.cmd = nil
return false, fmt.Sprintf("test timed out after %v", testTimeout), true
case result := <-result:
// If any IO operations failed then the server likely exited or crashed.
if !result.ok {
tintServer.stdin.Close()
err := tintServer.cmd.Wait()
tintServer.cmd = nil
if result.output != "" {
result.output += fmt.Sprintf("\ntint executable returned error: %v\n", err.Error())
return false, result.output, false
}
return false, err.Error(), false
}
return true, result.output, false
}
}
// invoke runs the Tint executable with the provided arguments.
func invokeWithoutServer(tintPath string, args ...string) (ok bool, output string, timedOut bool) {
ctx, cancel := context.WithTimeout(context.Background(), testTimeout)
defer cancel()
cmd := exec.CommandContext(ctx, tintPath, args...)
out, err := cmd.CombinedOutput()
str := string(out)
if err != nil {
if ctx.Err() == context.DeadlineExceeded {
return false, fmt.Sprintf("test timed out after %v", testTimeout), true
}
if str != "" {
str += fmt.Sprintf("\ntint executable returned error: %v\n", err.Error())
return false, str, false
}
return false, err.Error(), false
}
return true, str, false
}
var reFlags = regexp.MustCompile(`^\s*(?:\/\/|;)\s*(\[[\w-]+\])?\s*flags:(.*)`)
// cmdLineFlags are the flags that apply to the given formats, parsed by parseFlags
type cmdLineFlags struct {
formats container.Set[outputFormat]
flags []string
}
// parseFlags looks for a `// flags:` or `// [format] flags:` header at the start of the file with
// the given path, returning each of the parsed flags
func parseFlags(path string, fsReader oswrapper.FilesystemReader) ([]cmdLineFlags, error) {
inputFile, err := fsReader.Open(path)
if err != nil {
return nil, err
}
defer inputFile.Close()
out := []cmdLineFlags{}
scanner := bufio.NewScanner(inputFile)
scanner.Split(bufio.ScanLines)
for scanner.Scan() {
content := scanner.Text()
m := reFlags.FindStringSubmatch(content)
if len(m) == 3 {
formats := allOutputFormats
if m[1] != "" {
matchedFormat := strings.Trim(m[1], "[]")
fmts, err := parseOutputFormat(matchedFormat)
if err != nil {
return nil, err
}
formats = fmts
}
out = append(out, cmdLineFlags{
formats: container.NewSet(formats...),
flags: strings.Split(m[2], " "),
})
continue
}
if len(content) > 0 && !strings.HasPrefix(content, "//") {
break
}
}
return out, nil
}
// parseOutputFormat parses the outputFormat(s) from s, and
// returns an array of the single parsed format.
func parseOutputFormat(s string) ([]outputFormat, error) {
switch s {
case "wgsl":
return []outputFormat{wgsl}, nil
case "spvasm":
return []outputFormat{spvasm}, nil
case "msl":
return []outputFormat{msl}, nil
case "hlsl":
return []outputFormat{hlslDXC, hlslFXC}, nil
case "hlsl-dxc":
return []outputFormat{hlslDXC}, nil
case "hlsl-fxc":
return []outputFormat{hlslFXC}, nil
case "glsl":
return []outputFormat{glsl}, nil
default:
return nil, fmt.Errorf("unknown format '%s'", s)
}
}
func printDuration(d time.Duration) string {
sec := int(d.Seconds())
min := int(sec) / 60
hour := min / 60
sec -= min * 60
min -= hour * 60
sb := &strings.Builder{}
if hour > 0 {
fmt.Fprintf(sb, "%dh", hour)
}
if min > 0 {
fmt.Fprintf(sb, "%dm", min)
}
if sec > 0 {
fmt.Fprintf(sb, "%ds", sec)
}
return sb.String()
}
// fileAndFormat is a pair of test file path and output format.
type fileAndFormat struct {
file string
format outputFormat
}
// Used to optimize end-to-end testing of tint
type validationCache struct {
// A hash of all the validation toolchains in use.
toolchainHash string
// A map of fileAndFormat to known-good (validated) output hashes.
knownGood knownGoodHashes
}
// A map of fileAndFormat to known-good (validated) output hashes.
type knownGoodHashes map[fileAndFormat][]string
// ValidationCacheFile is the serialized form of a known-good validation.cache file
type ValidationCacheFile struct {
ToolchainHash string
KnownGood []ValidationCacheFileKnownGood
}
// ValidationCacheFileKnownGood holds a single record for a known-to-pass test output, given the
// target format and tool hashes.
type ValidationCacheFileKnownGood struct {
File string
Format outputFormat
Hashes []string
}
func validationCachePath(fsReader oswrapper.FilesystemReader) string {
return filepath.Join(fileutils.DawnRoot(fsReader), "test", "tint", "validation.cache")
}
// TODO(crbug.com/344014313): Add unittest coverage.
// loadValidationCache attempts to load the validation cache.
// Returns an empty cache if the file could not be loaded, or if toolchains have changed.
func loadValidationCache(
toolchainHash string, verbose bool, fsReader oswrapper.FilesystemReader) validationCache {
out := validationCache{
toolchainHash: toolchainHash,
knownGood: knownGoodHashes{},
}
file, err := fsReader.Open(validationCachePath(fsReader))
if err != nil {
if verbose {
fmt.Println(err)
}
return out
}
defer file.Close()
content := ValidationCacheFile{}
if err := json.NewDecoder(file).Decode(&content); err != nil {
if verbose {
fmt.Println(err)
}
return out
}
if content.ToolchainHash != toolchainHash {
color.Set(color.FgYellow)
fmt.Println("Toolchains have changed - clearing validation cache")
color.Unset()
return out
}
for _, knownGood := range content.KnownGood {
out.knownGood[fileAndFormat{knownGood.File, knownGood.Format}] = knownGood.Hashes
}
return out
}
// TODO(crbug.com/344014313): Add unittest coverage.
// saveValidationCache saves the validation cache file.
func saveValidationCache(vc validationCache, fsReaderWriter oswrapper.FilesystemReaderWriter) {
out := ValidationCacheFile{
ToolchainHash: vc.toolchainHash,
KnownGood: make([]ValidationCacheFileKnownGood, 0, len(vc.knownGood)),
}
for ff, hashes := range vc.knownGood {
out.KnownGood = append(out.KnownGood, ValidationCacheFileKnownGood{
File: ff.file,
Format: ff.format,
Hashes: hashes,
})
}
sort.Slice(out.KnownGood, func(i, j int) bool {
switch {
case out.KnownGood[i].File < out.KnownGood[j].File:
return true
case out.KnownGood[i].File > out.KnownGood[j].File:
return false
case out.KnownGood[i].Format < out.KnownGood[j].Format:
return true
case out.KnownGood[i].Format > out.KnownGood[j].Format:
return false
}
return false
})
file, err := fsReaderWriter.Create(validationCachePath(fsReaderWriter))
if err != nil {
fmt.Printf("WARNING: failed to save the validation cache file: %v\n", err)
}
defer file.Close()
enc := json.NewEncoder(file)
enc.SetIndent("", " ")
if err := enc.Encode(&out); err != nil {
fmt.Printf("WARNING: failed to encode the validation cache file: %v\n", err)
}
}
// defaultTintPath returns the default path to the tint executable
func defaultTintPath(fsReader oswrapper.FilesystemReader) string {
return filepath.Join(fileutils.DawnRoot(fsReader), "out", "active", "tint"+fileutils.ExeExt)
}