|  | // 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 | 
|  | } |