Home | History | Annotate | Download | only in test
      1 // Copyright 2014 The Go Authors.  All rights reserved.
      2 // Use of this source code is governed by a BSD-style
      3 // license that can be found in the LICENSE file.
      4 
      5 package cgotest
      6 
      7 // Test that we have no more than one build ID.  In the past we used
      8 // to generate a separate build ID for each package using cgo, and the
      9 // linker concatenated them all.  We don't want that--we only want
     10 // one.
     11 
     12 import (
     13 	"bytes"
     14 	"debug/elf"
     15 	"os"
     16 	"testing"
     17 )
     18 
     19 func testBuildID(t *testing.T) {
     20 	f, err := elf.Open("/proc/self/exe")
     21 	if err != nil {
     22 		if os.IsNotExist(err) {
     23 			t.Skip("no /proc/self/exe")
     24 		}
     25 		t.Fatalf("opening /proc/self/exe: ", err)
     26 	}
     27 	defer f.Close()
     28 
     29 	c := 0
     30 	for i, s := range f.Sections {
     31 		if s.Type != elf.SHT_NOTE {
     32 			continue
     33 		}
     34 
     35 		d, err := s.Data()
     36 		if err != nil {
     37 			t.Logf("reading data of note section %d: %v", i, err)
     38 			continue
     39 		}
     40 
     41 		for len(d) > 0 {
     42 
     43 			// ELF standards differ as to the sizes in
     44 			// note sections.  Both the GNU linker and
     45 			// gold always generate 32-bit sizes, so that
     46 			// is what we assume here.
     47 
     48 			if len(d) < 12 {
     49 				t.Logf("note section %d too short (%d < 12)", i, len(d))
     50 				continue
     51 			}
     52 
     53 			namesz := f.ByteOrder.Uint32(d)
     54 			descsz := f.ByteOrder.Uint32(d[4:])
     55 			typ := f.ByteOrder.Uint32(d[8:])
     56 
     57 			an := (namesz + 3) &^ 3
     58 			ad := (descsz + 3) &^ 3
     59 
     60 			if int(12+an+ad) > len(d) {
     61 				t.Logf("note section %d too short for header (%d < 12 + align(%d,4) + align(%d,4))", i, len(d), namesz, descsz)
     62 				continue
     63 			}
     64 
     65 			// 3 == NT_GNU_BUILD_ID
     66 			if typ == 3 && namesz == 4 && bytes.Equal(d[12:16], []byte("GNU\000")) {
     67 				c++
     68 			}
     69 
     70 			d = d[12+an+ad:]
     71 		}
     72 	}
     73 
     74 	if c > 1 {
     75 		t.Errorf("found %d build ID notes", c)
     76 	}
     77 }
     78