926bd8ca82
Bug: skia:10883 Change-Id: I42d7b2926aaab2b76adf69162153a6a65998380d Reviewed-on: https://skia-review.googlesource.com/c/skia/+/373742 Reviewed-by: Weston Tracey <westont@google.com> Commit-Queue: Eric Boren <borenet@google.com>
228 lines
5.7 KiB
Go
228 lines
5.7 KiB
Go
// Copyright 2020 The Chromium 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 gen_tasks_logic
|
|
|
|
import (
|
|
"log"
|
|
"os/exec"
|
|
"path/filepath"
|
|
"regexp"
|
|
"sort"
|
|
"strings"
|
|
|
|
"go.skia.org/infra/go/cas/rbe"
|
|
"go.skia.org/infra/task_scheduler/go/specs"
|
|
)
|
|
|
|
const (
|
|
// If a parent path contains more than this many immediate child paths (ie.
|
|
// files and dirs which are directly inside it as opposed to indirect
|
|
// descendants), we will include the parent in the isolate file instead of
|
|
// the children. This results in a simpler CasSpec which should need to be
|
|
// changed less often.
|
|
combinePathsThreshold = 3
|
|
)
|
|
|
|
var (
|
|
// Any files in Git which match these patterns will be included, either
|
|
// directly or indirectly via a parent dir.
|
|
pathRegexes = []*regexp.Regexp{
|
|
regexp.MustCompile(`.*\.c$`),
|
|
regexp.MustCompile(`.*\.cc$`),
|
|
regexp.MustCompile(`.*\.cpp$`),
|
|
regexp.MustCompile(`.*\.gn$`),
|
|
regexp.MustCompile(`.*\.gni$`),
|
|
regexp.MustCompile(`.*\.h$`),
|
|
regexp.MustCompile(`.*\.mm$`),
|
|
regexp.MustCompile(`.*\.storyboard$`),
|
|
}
|
|
|
|
// These paths are always added to the inclusion list. Note that they may
|
|
// not appear in the CasSpec if they are included indirectly via a parent
|
|
// dir.
|
|
explicitPaths = []string{
|
|
".clang-format",
|
|
".clang-tidy",
|
|
"bin/fetch-clang-format",
|
|
"bin/fetch-gn",
|
|
"buildtools",
|
|
"infra/bots/assets/android_ndk_darwin/VERSION",
|
|
"infra/bots/assets/android_ndk_linux/VERSION",
|
|
"infra/bots/assets/android_ndk_windows/VERSION",
|
|
"infra/bots/assets/cast_toolchain/VERSION",
|
|
"infra/bots/assets/clang_linux/VERSION",
|
|
"infra/bots/assets/clang_win/VERSION",
|
|
"infra/bots/run_recipe.py",
|
|
"infra/canvaskit",
|
|
"infra/pathkit",
|
|
"resources",
|
|
"third_party/externals",
|
|
}
|
|
)
|
|
|
|
// getAllCheckedInPaths returns every path checked in to the repo.
|
|
func getAllCheckedInPaths() []string {
|
|
cmd := exec.Command("git", "ls-files")
|
|
// Use '../skia' to get to the Skia checkout, in case this is used by
|
|
// another repo.
|
|
cmd.Dir = filepath.Join(CheckoutRoot(), "..", "skia")
|
|
output, err := cmd.CombinedOutput()
|
|
if err != nil {
|
|
log.Fatal(err)
|
|
}
|
|
split := strings.Split(string(output), "\n")
|
|
rv := make([]string, 0, len(split))
|
|
for _, line := range split {
|
|
if line != "" {
|
|
rv = append(rv, line)
|
|
}
|
|
}
|
|
return rv
|
|
}
|
|
|
|
// getRelevantPaths returns all paths needed by compile tasks.
|
|
func getRelevantPaths() []string {
|
|
rv := []string{}
|
|
for _, path := range getAllCheckedInPaths() {
|
|
for _, regex := range pathRegexes {
|
|
if regex.MatchString(path) {
|
|
rv = append(rv, path)
|
|
break
|
|
}
|
|
}
|
|
}
|
|
return append(rv, explicitPaths...)
|
|
}
|
|
|
|
// node is a single node in a directory tree of task inputs.
|
|
type node struct {
|
|
children map[string]*node
|
|
name string
|
|
isLeaf bool
|
|
}
|
|
|
|
// newNode returns a node instance.
|
|
func newNode(name string) *node {
|
|
return &node{
|
|
children: map[string]*node{},
|
|
name: name,
|
|
isLeaf: false,
|
|
}
|
|
}
|
|
|
|
// isRoot returns true iff this is the root node.
|
|
func (n *node) isRoot() bool {
|
|
return n.name == ""
|
|
}
|
|
|
|
// add the given entry (given as a slice of path components) to the node.
|
|
func (n *node) add(entry []string) {
|
|
// Remove the first element if we're not the root node.
|
|
if !n.isRoot() {
|
|
if entry[0] != n.name {
|
|
log.Fatalf("Failed to compute compile CAS inputs; attempting to add entry %v to node %q", entry, n.name)
|
|
}
|
|
entry = entry[1:]
|
|
|
|
// If the entry is now empty, this node is a leaf.
|
|
if len(entry) == 0 {
|
|
n.isLeaf = true
|
|
return
|
|
}
|
|
}
|
|
|
|
// Add a child node.
|
|
if !n.isLeaf {
|
|
name := entry[0]
|
|
child, ok := n.children[name]
|
|
if !ok {
|
|
child = newNode(name)
|
|
n.children[name] = child
|
|
}
|
|
child.add(entry)
|
|
|
|
// If we have more than combinePathsThreshold immediate children,
|
|
// combine them into this node.
|
|
immediateChilden := 0
|
|
for _, child := range n.children {
|
|
if child.isLeaf {
|
|
immediateChilden++
|
|
}
|
|
if !n.isRoot() && immediateChilden >= combinePathsThreshold {
|
|
n.isLeaf = true
|
|
n.children = map[string]*node{}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// entries returns the entries represented by this node and its children.
|
|
// Will not return children in the following cases:
|
|
// - This Node is a leaf, ie. it represents an entry which was explicitly
|
|
// inserted into the Tree, as opposed to only part of a path to other
|
|
// entries.
|
|
// - This Node has immediate children exceeding combinePathsThreshold and
|
|
// thus has been upgraded to a leaf node.
|
|
func (n *node) entries() [][]string {
|
|
if n.isLeaf {
|
|
return [][]string{{n.name}}
|
|
}
|
|
rv := [][]string{}
|
|
for _, child := range n.children {
|
|
for _, entry := range child.entries() {
|
|
if !n.isRoot() {
|
|
entry = append([]string{n.name}, entry...)
|
|
}
|
|
rv = append(rv, entry)
|
|
}
|
|
}
|
|
return rv
|
|
}
|
|
|
|
// tree represents a directory tree of task inputs.
|
|
type tree struct {
|
|
root *node
|
|
}
|
|
|
|
// newTree returns a tree instance.
|
|
func newTree() *tree {
|
|
return &tree{
|
|
root: newNode(""),
|
|
}
|
|
}
|
|
|
|
// add the given path to the tree. Entries may be combined as defined by
|
|
// combinePathsThreshold.
|
|
func (t *tree) add(p string) {
|
|
split := strings.Split(p, "/")
|
|
t.root.add(split)
|
|
}
|
|
|
|
// entries returns all entries in the tree. Entries may be combined as defined
|
|
// by combinePathsThreshold.
|
|
func (t *tree) entries() []string {
|
|
entries := t.root.entries()
|
|
rv := make([]string, 0, len(entries))
|
|
for _, entry := range entries {
|
|
rv = append(rv, strings.Join(append([]string{"skia"}, entry...), "/"))
|
|
}
|
|
sort.Strings(rv)
|
|
return rv
|
|
}
|
|
|
|
// generateCompileCAS creates the CasSpec used for tasks which build Skia.
|
|
func generateCompileCAS(b *specs.TasksCfgBuilder) {
|
|
t := newTree()
|
|
for _, path := range getRelevantPaths() {
|
|
t.add(path)
|
|
}
|
|
spec := &specs.CasSpec{
|
|
Root: "..",
|
|
Paths: t.entries(),
|
|
Excludes: []string{rbe.ExcludeGitDir},
|
|
}
|
|
b.MustAddCasSpec(CAS_COMPILE, spec)
|
|
}
|