+++ /dev/null
-// Copyright 2015 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"
- "bytes"
- "fmt"
- "io"
- "log"
- "os"
- "regexp"
- "strconv"
- "strings"
-)
-
-var (
- hexDumpRE = regexp.MustCompile(`^\t(0x[0-9a-f]{4,})(( ([0-9a-f]{2}| )){16}) [ -\x7F]{1,16}\n`)
- listingRE = regexp.MustCompile(`^\t(0x[0-9a-f]{4,}) ([0-9]{4,}) \(.*:[0-9]+\)\t`)
-)
-
-// okdiffs lists regular expressions for lines to consider minor mismatches.
-// If one of these regexps matches both of a pair of unequal lines, the mismatch
-// is reported but not treated as the one worth looking for.
-// For example, differences in the TEXT line are typically frame size
-// changes due to optimization decisions made in the body of the function.
-// Better to keep looking for the actual difference.
-// Similarly, forward jumps might have different offsets due to a
-// change in instruction encoding later on.
-// Better to find that change.
-var okdiffs = []*regexp.Regexp{
- regexp.MustCompile(`\) TEXT[ ].*,\$`),
- regexp.MustCompile(`\) WORD[ ].*,\$`),
- regexp.MustCompile(`\) (B|BR|JMP) `),
- regexp.MustCompile(`\) FUNCDATA `),
- regexp.MustCompile(`\\(z|x00)`),
- regexp.MustCompile(`\$\([0-9]\.[0-9]+e[+\-][0-9]+\)`),
- regexp.MustCompile(`size=.*value=.*args=.*locals=`),
-}
-
-func compareLogs(outfile string) string {
- f1, err := os.Open(outfile + ".log")
- if err != nil {
- log.Fatal(err)
- }
- defer f1.Close()
-
- f2, err := os.Open(outfile + ".stash.log")
- if err != nil {
- log.Fatal(err)
- }
- defer f2.Close()
-
- b1 := bufio.NewReader(f1)
- b2 := bufio.NewReader(f2)
-
- offset := int64(0)
- textOffset := offset
- textLineno := 0
- lineno := 0
- var line1, line2 string
- var prefix bytes.Buffer
-Reading:
- for {
- var err1, err2 error
- line1, err1 = b1.ReadString('\n')
- line2, err2 = b2.ReadString('\n')
- if strings.Contains(line1, ")\tTEXT\t") {
- textOffset = offset
- textLineno = lineno
- }
- offset += int64(len(line1))
- lineno++
- if err1 == io.EOF && err2 == io.EOF {
- return "no differences in debugging output"
- }
-
- if lineno == 1 || line1 == line2 && err1 == nil && err2 == nil {
- continue
- }
- // Lines are inconsistent. Worth stopping?
- for _, re := range okdiffs {
- if re.MatchString(line1) && re.MatchString(line2) {
- fmt.Fprintf(&prefix, "inconsistent log line:\n%s:%d:\n\t%s\n%s:%d:\n\t%s\n\n",
- f1.Name(), lineno, strings.TrimSuffix(line1, "\n"),
- f2.Name(), lineno, strings.TrimSuffix(line2, "\n"))
- continue Reading
- }
- }
-
- if err1 != nil {
- line1 = err1.Error()
- }
- if err2 != nil {
- line2 = err2.Error()
- }
- break
- }
-
- msg := fmt.Sprintf("inconsistent log line:\n%s:%d:\n\t%s\n%s:%d:\n\t%s",
- f1.Name(), lineno, strings.TrimSuffix(line1, "\n"),
- f2.Name(), lineno, strings.TrimSuffix(line2, "\n"))
-
- if m := hexDumpRE.FindStringSubmatch(line1); m != nil {
- target, err := strconv.ParseUint(m[1], 0, 64)
- if err != nil {
- goto Skip
- }
-
- m2 := hexDumpRE.FindStringSubmatch(line2)
- if m2 == nil {
- goto Skip
- }
-
- fields1 := strings.Fields(m[2])
- fields2 := strings.Fields(m2[2])
- i := 0
- for i < len(fields1) && i < len(fields2) && fields1[i] == fields2[i] {
- i++
- }
- target += uint64(i)
-
- f1.Seek(textOffset, 0)
- b1 = bufio.NewReader(f1)
- last := ""
- lineno := textLineno
- limitAddr := uint64(0)
- lastAddr := uint64(0)
- for {
- line1, err1 := b1.ReadString('\n')
- if err1 != nil {
- break
- }
- lineno++
- if m := listingRE.FindStringSubmatch(line1); m != nil {
- addr, _ := strconv.ParseUint(m[1], 0, 64)
- if addr > target {
- limitAddr = addr
- break
- }
- last = line1
- lastAddr = addr
- } else if hexDumpRE.FindStringSubmatch(line1) != nil {
- break
- }
- }
- if last != "" {
- msg = fmt.Sprintf("assembly instruction at %#04x-%#04x:\n%s:%d\n\t%s\n\n%s",
- lastAddr, limitAddr, f1.Name(), lineno-1, strings.TrimSuffix(last, "\n"), msg)
- }
- }
-Skip:
-
- return prefix.String() + msg
-}