aboutsummaryrefslogtreecommitdiffstats
path: root/pkg/build/linux_test.go
blob: 8f4e2d73c9882315d3209043720c3d9c0ceda2bf (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
// Copyright 2019 syzkaller project authors. All rights reserved.
// Use of this source code is governed by Apache 2 LICENSE that can be found in the LICENSE file.

//go:build linux

package build

import (
	"bytes"
	"context"
	"fmt"
	"os"
	"strings"
	"testing"
	"text/template"

	"github.com/google/syzkaller/pkg/debugtracer"
	"github.com/google/syzkaller/pkg/osutil"
	"golang.org/x/sync/errgroup"
)

func TestElfBinarySignature(t *testing.T) {
	t.Parallel()
	enumerateFlags(t, nil, []string{"-g", "-O1", "-O2", "-no-pie", "-static"})
}

func TestQueryLinuxCompiler(t *testing.T) {
	const goodDir = "./testdata/linux_compiler_ok"
	const expectedCompiler = "gcc (Debian 10.2.1-6+build2) 10.2.1 20210110, GNU ld (GNU Binutils for Debian) 2.35.2"
	ret, err := queryLinuxCompiler(goodDir)
	if err != nil {
		t.Fatalf("error: %v", err)
	}
	if ret != expectedCompiler {
		t.Fatalf("got: %T, expected: %T", ret, expectedCompiler)
	}
	const badDir = "./testingData/non_existing_folder"
	_, err = queryLinuxCompiler(badDir)
	if err == nil {
		t.Fatalf("expected an error, got none")
	}
}

func enumerateFlags(t *testing.T, flags, allFlags []string) {
	if len(allFlags) != 0 {
		enumerateFlags(t, flags, allFlags[1:])
		enumerateFlags(t, append(flags, allFlags[0]), allFlags[1:])
		return
	}
	t.Run(strings.Join(flags, "-"), func(t *testing.T) {
		t.Parallel()
		sign1, sign2, sign3 := "", "", ""
		g, _ := errgroup.WithContext(context.Background())
		g.Go(func() error {
			var err error
			sign1, err = sign(t, flags, false, false)
			return err
		})
		g.Go(func() error {
			var err error
			sign2, err = sign(t, flags, false, true)
			return err
		})
		g.Go(func() error {
			var err error
			sign3, err = sign(t, flags, true, false)
			return err
		})
		if err := g.Wait(); err != nil {
			t.Error(err)
		}
		if sign1 != sign2 {
			t.Errorf("signature has changed after a comment-only change")
		}
		if sign1 == sign3 {
			t.Errorf("signature has not changed after a change")
		}
	})
}

func sign(t *testing.T, flags []string, changed, comment bool) (string, error) {
	buf := new(bytes.Buffer)
	if err := srcTemplate.Execute(buf, SrcParams{Changed: changed, Comment: comment}); err != nil {
		return "", fmt.Errorf("template exec failed: %w", err)
	}
	src := buf.Bytes()
	bin, err := osutil.TempFile("syz-build-test")
	if err != nil {
		return "", fmt.Errorf("temp file creation error: %w", err)
	}
	defer os.Remove(bin)
	cmd := osutil.Command("gcc", append(flags, "-pthread", "-o", bin, "-x", "c", "-")...)
	cmd.Stdin = buf
	out, err := cmd.CombinedOutput()
	if err != nil {
		return "", fmt.Errorf("compiler failed: %w\n%s\n\n%s", err, src, out)
	}
	sign, err := elfBinarySignature(bin, &debugtracer.TestTracer{T: t})
	if err != nil {
		return "", fmt.Errorf("signature creation failed: %w", err)
	}
	return sign, nil
}

type SrcParams struct {
	Changed bool
	Comment bool
}

var srcTemplate = template.Must(template.New("").Parse(`
#include <stdio.h>
#include <pthread.h>

int main() {
	int x = {{if .Changed}}0{{else}}1{{end}};
	{{if .Comment}}
	// Some comment goes here.
	// It affects line numbers in debug info.
	{{end}}
	printf("%d %p\n", x, pthread_create);
}
`))