// Copyright 2015 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.
package report
import (
"fmt"
"testing"
"github.com/google/syzkaller/pkg/mgrconfig"
"github.com/google/syzkaller/pkg/symbolizer"
)
func TestLinuxIgnores(t *testing.T) {
cfg := &mgrconfig.Config{
TargetOS: "linux",
}
reporter, err := NewReporter(cfg)
if err != nil {
t.Fatal(err)
}
cfg.Ignores = []string{"BUG: bug3"}
reporter1, err := NewReporter(cfg)
if err != nil {
t.Fatal(err)
}
cfg.Ignores = []string{"BUG: bug3", "BUG: bug1"}
reporter2, err := NewReporter(cfg)
if err != nil {
t.Fatal(err)
}
cfg.Ignores = []string{"BUG: bug3", "BUG: bug1", "BUG: bug2"}
reporter3, err := NewReporter(cfg)
if err != nil {
t.Fatal(err)
}
const log = `
[ 0.000000] BUG: bug1
[ 0.000000] BUG: bug2
`
if !reporter.ContainsCrash([]byte(log)) {
t.Fatalf("no crash")
}
if rep := reporter.Parse([]byte(log)); rep.Title != "BUG: bug1" {
t.Fatalf("want `BUG: bug1`, found `%v`", rep.Title)
}
if !reporter1.ContainsCrash([]byte(log)) {
t.Fatalf("no crash")
}
if rep := reporter1.Parse([]byte(log)); rep.Title != "BUG: bug1" {
t.Fatalf("want `BUG: bug1`, found `%v`", rep.Title)
}
if !reporter2.ContainsCrash([]byte(log)) {
t.Fatalf("no crash")
}
if rep := reporter2.Parse([]byte(log)); rep.Title != "BUG: bug2" {
t.Fatalf("want `BUG: bug2`, found `%v`", rep.Title)
}
if reporter3.ContainsCrash([]byte(log)) {
t.Fatalf("found crash, should be ignored")
}
if rep := reporter3.Parse([]byte(log)); rep != nil {
t.Fatalf("found `%v`, should be ignored", rep.Title)
}
}
func TestLinuxSymbolizeLine(t *testing.T) {
tests := []struct {
line string
result string
}{
// Normal symbolization.
{
"[ 2713.153531] [<ffffffff82d1b1d9>] foo+0x101/0x185\n",
"[ 2713.153531] [<ffffffff82d1b1d9>] foo+0x101/0x185 foo.c:555\n",
},
{
"RIP: 0010:[<ffffffff8188c0e6>] [<ffffffff8188c0e6>] foo+0x101/0x185\n",
"RIP: 0010:[<ffffffff8188c0e6>] [<ffffffff8188c0e6>] foo+0x101/0x185 foo.c:555\n",
},
// Strip "./" file prefix.
{
"[ 2713.153531] [<ffffffff82d1b1d9>] foo+0x111/0x185\n",
"[ 2713.153531] [<ffffffff82d1b1d9>] foo+0x111/0x185 foo.h:111\n",
},
// Needs symbolization, but symbolizer returns nothing.
{
"[ 2713.153531] [<ffffffff82d1b1d9>] foo+0x121/0x185\n",
"[ 2713.153531] [<ffffffff82d1b1d9>] foo+0x121/0x185\n",
},
// Needs symbolization, but symbolizer returns error.
{
"[ 2713.153531] [<ffffffff82d1b1d9>] foo+0x131/0x185\n",
"[ 2713.153531] [<ffffffff82d1b1d9>] foo+0x131/0x185\n",
},
// Needs symbolization, but symbol is missing.
{
"[ 2713.153531] [<ffffffff82d1b1d9>] bar+0x131/0x185\n",
"[ 2713.153531] [<ffffffff82d1b1d9>] bar+0x131/0x185\n",
},
// Bad offset.
{
"[ 2713.153531] [<ffffffff82d1b1d9>] bar+0xffffffffffffffffffff/0x185\n",
"[ 2713.153531] [<ffffffff82d1b1d9>] bar+0xffffffffffffffffffff/0x185\n",
},
// Should not be symbolized.
{
"WARNING: CPU: 2 PID: 2636 at ipc/shm.c:162 foo+0x101/0x185\n",
"WARNING: CPU: 2 PID: 2636 at ipc/shm.c:162 foo+0x101/0x185 foo.c:555\n",
},
// Tricky function name.
{
" [<ffffffff84e5bea0>] do_ipv6_setsockopt.isra.7.part.3+0x101/0x2830 \n",
" [<ffffffff84e5bea0>] do_ipv6_setsockopt.isra.7.part.3+0x101/0x2830 net.c:111 \n",
},
// Old KASAN frame format (with tab).
{
"[ 50.419727] baz+0x101/0x200\n",
"[ 50.419727] baz+0x101/0x200 baz.c:100\n",
},
// Inlined frames.
{
" [<ffffffff84e5bea0>] foo+0x141/0x185\n",
" [<ffffffff84e5bea0>] inlined1 net.c:111 [inline]\n" +
" [<ffffffff84e5bea0>] inlined2 mm.c:222 [inline]\n" +
" [<ffffffff84e5bea0>] foo+0x141/0x185 kasan.c:333\n",
},
// Several symbols with the same name.
{
"[<ffffffff82d1b1d9>] baz+0x101/0x200\n",
"[<ffffffff82d1b1d9>] baz+0x101/0x200 baz.c:100\n",
},
}
symbols := map[string][]symbolizer.Symbol{
"foo": {
{Addr: 0x1000000, Size: 0x190},
},
"do_ipv6_setsockopt.isra.7.part.3": {
{Addr: 0x2000000, Size: 0x2830},
},
"baz": {
{Addr: 0x3000000, Size: 0x100},
{Addr: 0x4000000, Size: 0x200},
{Addr: 0x5000000, Size: 0x300},
},
}
symb := func(bin string, pc uint64) ([]symbolizer.Frame, error) {
if bin != "vmlinux" {
return nil, fmt.Errorf("unknown pc 0x%x", pc)
}
switch pc {
case 0x1000100:
return []symbolizer.Frame{
{
File: "/linux/foo.c",
Line: 555,
},
}, nil
case 0x1000110:
return []symbolizer.Frame{
{
File: "/linux/./foo.h",
Line: 111,
},
}, nil
case 0x1000120:
return nil, nil
case 0x1000130:
return nil, fmt.Errorf("unknown pc 0x%x", pc)
case 0x2000100:
return []symbolizer.Frame{
{
File: "/linux/net.c",
Line: 111,
},
}, nil
case 0x1000140:
return []symbolizer.Frame{
{
Func: "inlined1",
File: "/linux/net.c",
Line: 111,
Inline: true,
},
{
Func: "inlined2",
File: "/linux/mm.c",
Line: 222,
Inline: true,
},
{
Func: "noninlined3",
File: "/linux/kasan.c",
Line: 333,
Inline: false,
},
}, nil
case 0x4000100:
return []symbolizer.Frame{
{
File: "/linux/baz.c",
Line: 100,
},
}, nil
default:
return nil, fmt.Errorf("unknown pc 0x%x", pc)
}
}
for i, test := range tests {
t.Run(fmt.Sprint(i), func(t *testing.T) {
result := symbolizeLine(symb, symbols, "vmlinux", "/linux/", []byte(test.line))
if test.result != string(result) {
t.Errorf("want %q\n\t get %q", test.result, string(result))
}
})
}
}