214 lines
5.6 KiB
Go
214 lines
5.6 KiB
Go
|
// Imported from https://code.google.com/p/go/source/browse/cmd/cover/profile.go?repo=tools&r=c10a9dd5e0b0a859a8385b6f004584cb083a3934
|
||
|
|
||
|
// Copyright 2013 The Go 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 main
|
||
|
|
||
|
import (
|
||
|
"bufio"
|
||
|
"fmt"
|
||
|
"io"
|
||
|
"math"
|
||
|
"regexp"
|
||
|
"sort"
|
||
|
"strconv"
|
||
|
"strings"
|
||
|
)
|
||
|
|
||
|
// Profile represents the profiling data for a specific file.
|
||
|
type Profile struct {
|
||
|
FileName string
|
||
|
Mode string
|
||
|
Blocks []ProfileBlock
|
||
|
}
|
||
|
|
||
|
// ProfileBlock represents a single block of profiling data.
|
||
|
type ProfileBlock struct {
|
||
|
StartLine, StartCol int
|
||
|
EndLine, EndCol int
|
||
|
NumStmt, Count int
|
||
|
}
|
||
|
|
||
|
type byFileName []*Profile
|
||
|
|
||
|
func (p byFileName) Len() int { return len(p) }
|
||
|
func (p byFileName) Less(i, j int) bool { return p[i].FileName < p[j].FileName }
|
||
|
func (p byFileName) Swap(i, j int) { p[i], p[j] = p[j], p[i] }
|
||
|
|
||
|
// ParseProfiles parses profile data from the given Reader and returns a
|
||
|
// Profile for each file.
|
||
|
func ParseProfiles(in io.Reader, ignore *Ignore) ([]*Profile, error) {
|
||
|
files := make(map[string]*Profile)
|
||
|
// First line is "mode: foo", where foo is "set", "count", or "atomic".
|
||
|
// Rest of file is in the format
|
||
|
// encoding/base64/base64.go:34.44,37.40 3 1
|
||
|
// where the fields are: name.go:line.column,line.column numberOfStatements count
|
||
|
s := bufio.NewScanner(in)
|
||
|
mode := ""
|
||
|
for s.Scan() {
|
||
|
line := s.Text()
|
||
|
if mode == "" {
|
||
|
const p = "mode: "
|
||
|
if !strings.HasPrefix(line, p) || line == p {
|
||
|
return nil, fmt.Errorf("bad mode line: %v", line)
|
||
|
}
|
||
|
mode = line[len(p):]
|
||
|
continue
|
||
|
}
|
||
|
m := lineRe.FindStringSubmatch(line)
|
||
|
if m == nil {
|
||
|
continue
|
||
|
}
|
||
|
fn := m[1]
|
||
|
if ignore.Match(fn, nil) {
|
||
|
continue
|
||
|
}
|
||
|
p := files[fn]
|
||
|
if p == nil {
|
||
|
p = &Profile{
|
||
|
FileName: fn,
|
||
|
Mode: mode,
|
||
|
}
|
||
|
files[fn] = p
|
||
|
}
|
||
|
p.Blocks = append(p.Blocks, ProfileBlock{
|
||
|
StartLine: toInt(m[2]),
|
||
|
StartCol: toInt(m[3]),
|
||
|
EndLine: toInt(m[4]),
|
||
|
EndCol: toInt(m[5]),
|
||
|
NumStmt: toInt(m[6]),
|
||
|
Count: toInt(m[7]),
|
||
|
})
|
||
|
}
|
||
|
if err := s.Err(); err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
|
||
|
for _, p := range files {
|
||
|
sort.Sort(blocksByStart(p.Blocks))
|
||
|
// Merge samples from the same location.
|
||
|
j := 1
|
||
|
for i := 1; i < len(p.Blocks); i++ {
|
||
|
b := p.Blocks[i]
|
||
|
last := p.Blocks[j-1]
|
||
|
if b.StartLine == last.StartLine &&
|
||
|
b.StartCol == last.StartCol &&
|
||
|
b.EndLine == last.EndLine &&
|
||
|
b.EndCol == last.EndCol {
|
||
|
if b.NumStmt != last.NumStmt {
|
||
|
return nil, fmt.Errorf("inconsistent NumStmt: changed from %d to %d", last.NumStmt, b.NumStmt)
|
||
|
}
|
||
|
if mode == "set" {
|
||
|
p.Blocks[j-1].Count |= b.Count
|
||
|
} else {
|
||
|
p.Blocks[j-1].Count += b.Count
|
||
|
}
|
||
|
continue
|
||
|
}
|
||
|
p.Blocks[j] = b
|
||
|
j++
|
||
|
}
|
||
|
p.Blocks = p.Blocks[:j]
|
||
|
}
|
||
|
|
||
|
// Generate a sorted slice.
|
||
|
profiles := make([]*Profile, 0, len(files))
|
||
|
for _, profile := range files {
|
||
|
profiles = append(profiles, profile)
|
||
|
}
|
||
|
sort.Sort(byFileName(profiles))
|
||
|
return profiles, nil
|
||
|
}
|
||
|
|
||
|
type blocksByStart []ProfileBlock
|
||
|
|
||
|
func (b blocksByStart) Len() int { return len(b) }
|
||
|
func (b blocksByStart) Swap(i, j int) { b[i], b[j] = b[j], b[i] }
|
||
|
func (b blocksByStart) Less(i, j int) bool {
|
||
|
bi, bj := b[i], b[j]
|
||
|
return bi.StartLine < bj.StartLine || bi.StartLine == bj.StartLine && bi.StartCol < bj.StartCol
|
||
|
}
|
||
|
|
||
|
var lineRe = regexp.MustCompile(`^(.+):([0-9]+).([0-9]+),([0-9]+).([0-9]+) ([0-9]+) ([0-9]+)$`)
|
||
|
|
||
|
func toInt(s string) int {
|
||
|
i, err := strconv.Atoi(s)
|
||
|
if err != nil {
|
||
|
panic(err)
|
||
|
}
|
||
|
return i
|
||
|
}
|
||
|
|
||
|
// Boundary represents the position in a source file of the beginning or end of a
|
||
|
// block as reported by the coverage profile. In HTML mode, it will correspond to
|
||
|
// the opening or closing of a <span> tag and will be used to colorize the source
|
||
|
type Boundary struct {
|
||
|
Offset int // Location as a byte offset in the source file.
|
||
|
Start bool // Is this the start of a block?
|
||
|
Count int // Event count from the cover profile.
|
||
|
Norm float64 // Count normalized to [0..1].
|
||
|
}
|
||
|
|
||
|
// Boundaries returns a Profile as a set of Boundary objects within the provided src.
|
||
|
func (p *Profile) Boundaries(src []byte) []Boundary {
|
||
|
var boundaries []Boundary
|
||
|
|
||
|
// Find maximum count.
|
||
|
max := 0
|
||
|
for _, b := range p.Blocks {
|
||
|
if b.Count > max {
|
||
|
max = b.Count
|
||
|
}
|
||
|
}
|
||
|
// Divisor for normalization.
|
||
|
divisor := math.Log(float64(max))
|
||
|
|
||
|
// boundary returns a Boundary, populating the Norm field with a normalized Count.
|
||
|
boundary := func(offset int, start bool, count int) Boundary {
|
||
|
b := Boundary{Offset: offset, Start: start, Count: count}
|
||
|
if !start || count == 0 {
|
||
|
return b
|
||
|
}
|
||
|
if max <= 1 {
|
||
|
b.Norm = 0.8 // Profile is in"set" mode; we want a heat map. Use cov8 in the CSS.
|
||
|
} else if count > 0 {
|
||
|
b.Norm = math.Log(float64(count)) / divisor
|
||
|
}
|
||
|
return b
|
||
|
}
|
||
|
|
||
|
line, col := 1, 2 // TODO: Why is this 2?
|
||
|
for si, bi := 0, 0; si < len(src) && bi < len(p.Blocks); {
|
||
|
b := p.Blocks[bi]
|
||
|
if b.StartLine == line && b.StartCol == col {
|
||
|
boundaries = append(boundaries, boundary(si, true, b.Count))
|
||
|
}
|
||
|
if b.EndLine == line && b.EndCol == col {
|
||
|
boundaries = append(boundaries, boundary(si, false, 0))
|
||
|
bi++
|
||
|
continue // Don't advance through src; maybe the next block starts here.
|
||
|
}
|
||
|
if src[si] == '\n' {
|
||
|
line++
|
||
|
col = 0
|
||
|
}
|
||
|
col++
|
||
|
si++
|
||
|
}
|
||
|
sort.Sort(boundariesByPos(boundaries))
|
||
|
return boundaries
|
||
|
}
|
||
|
|
||
|
type boundariesByPos []Boundary
|
||
|
|
||
|
func (b boundariesByPos) Len() int { return len(b) }
|
||
|
func (b boundariesByPos) Swap(i, j int) { b[i], b[j] = b[j], b[i] }
|
||
|
func (b boundariesByPos) Less(i, j int) bool {
|
||
|
if b[i].Offset == b[j].Offset {
|
||
|
return !b[i].Start && b[j].Start
|
||
|
}
|
||
|
return b[i].Offset < b[j].Offset
|
||
|
}
|