// Read from file, collecting header for __.PKGDEF.
// The header is from the beginning of the file until a line
// containing just "!". The first line must begin with "go object ".
+ //
+ // Note: It's possible for "\n!\n" to appear within the binary
+ // package export data format. To avoid truncating the package
+ // definition prematurely (issue 21703), we keep keep track of
+ // how many "$$" delimiters we've seen.
+
rbuf := bufio.NewReader(f)
var wbuf bytes.Buffer
+ markers := 0
for {
line, err := rbuf.ReadBytes('\n')
if err != nil {
if wbuf.Len() == 0 && !bytes.HasPrefix(line, []byte("go object ")) {
return nil, errors.New("not a Go object file")
}
- if bytes.Equal(line, []byte("!\n")) {
+ if markers%2 == 0 && bytes.Equal(line, []byte("!\n")) {
break
}
+ if bytes.HasPrefix(line, []byte("$$")) {
+ markers++
+ }
wbuf.Write(line)
}
return wbuf.Bytes(), nil
}
}
+// Test that "\n!\n" inside export data doesn't result in a truncated
+// package definition when creating a .a archive from a .o Go object.
+func TestIssue21703(t *testing.T) {
+ testenv.MustHaveGoBuild(t)
+
+ dir := tmpDir(t)
+ defer os.RemoveAll(dir)
+
+ const aSrc = `package a; const X = "\n!\n"`
+ err := ioutil.WriteFile(filepath.Join(dir, "a.go"), []byte(aSrc), 0666)
+ if err != nil {
+ t.Fatal(err)
+ }
+
+ const bSrc = `package b; import _ "a"`
+ err = ioutil.WriteFile(filepath.Join(dir, "b.go"), []byte(bSrc), 0666)
+ if err != nil {
+ t.Fatal(err)
+ }
+
+ run := func(args ...string) string {
+ return doRun(t, dir, args...)
+ }
+
+ goBin := testenv.GoToolPath(t)
+ run(goBin, "build", "cmd/pack") // writes pack binary to dir
+ run(goBin, "tool", "compile", "a.go")
+ run("./pack", "c", "a.a", "a.o")
+ run(goBin, "tool", "compile", "-I", ".", "b.go")
+}
+
// doRun runs a program in a directory and returns the output.
func doRun(t *testing.T, dir string, args ...string) string {
cmd := exec.Command(args[0], args[1:]...)