github.com/zebozhuang/go@v0.0.0-20200207033046-f8a98f6f5c5d/src/cmd/link/internal/ld/objfile.go (about) 1 // Copyright 2013 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 ld 6 7 // Reading of Go object files. 8 9 import ( 10 "bufio" 11 "bytes" 12 "cmd/internal/bio" 13 "cmd/internal/dwarf" 14 "cmd/internal/objabi" 15 "crypto/sha1" 16 "encoding/base64" 17 "io" 18 "log" 19 "strconv" 20 "strings" 21 ) 22 23 const ( 24 startmagic = "\x00\x00go19ld" 25 endmagic = "\xff\xffgo19ld" 26 ) 27 28 var emptyPkg = []byte(`"".`) 29 30 // objReader reads Go object files. 31 type objReader struct { 32 rd *bufio.Reader 33 ctxt *Link 34 lib *Library 35 pn string 36 dupSym *Symbol 37 localSymVersion int 38 39 // rdBuf is used by readString and readSymName as scratch for reading strings. 40 rdBuf []byte 41 42 // List of symbol references for the file being read. 43 refs []*Symbol 44 data []byte 45 reloc []Reloc 46 pcdata []Pcdata 47 autom []Auto 48 funcdata []*Symbol 49 funcdataoff []int64 50 file []*Symbol 51 } 52 53 func LoadObjFile(ctxt *Link, f *bio.Reader, lib *Library, length int64, pn string) { 54 55 start := f.Offset() 56 r := &objReader{ 57 rd: f.Reader, 58 lib: lib, 59 ctxt: ctxt, 60 pn: pn, 61 dupSym: &Symbol{Name: ".dup"}, 62 localSymVersion: ctxt.Syms.IncVersion(), 63 } 64 r.loadObjFile() 65 if f.Offset() != start+length { 66 log.Fatalf("%s: unexpected end at %d, want %d", pn, f.Offset(), start+length) 67 } 68 } 69 70 func (r *objReader) loadObjFile() { 71 pkg := objabi.PathToPrefix(r.lib.Pkg) 72 73 // Magic header 74 var buf [8]uint8 75 r.readFull(buf[:]) 76 if string(buf[:]) != startmagic { 77 log.Fatalf("%s: invalid file start %x %x %x %x %x %x %x %x", r.pn, buf[0], buf[1], buf[2], buf[3], buf[4], buf[5], buf[6], buf[7]) 78 } 79 80 // Version 81 c, err := r.rd.ReadByte() 82 if err != nil || c != 1 { 83 log.Fatalf("%s: invalid file version number %d", r.pn, c) 84 } 85 86 // Autolib 87 for { 88 lib := r.readString() 89 if lib == "" { 90 break 91 } 92 l := addlib(r.ctxt, pkg, r.pn, lib) 93 if l != nil { 94 r.lib.imports = append(r.lib.imports, l) 95 } 96 } 97 98 // Symbol references 99 r.refs = []*Symbol{nil} // zeroth ref is nil 100 for { 101 c, err := r.rd.Peek(1) 102 if err != nil { 103 log.Fatalf("%s: peeking: %v", r.pn, err) 104 } 105 if c[0] == 0xff { 106 r.rd.ReadByte() 107 break 108 } 109 r.readRef() 110 } 111 112 // Lengths 113 r.readSlices() 114 115 // Data section 116 r.readFull(r.data) 117 118 // Defined symbols 119 for { 120 c, err := r.rd.Peek(1) 121 if err != nil { 122 log.Fatalf("%s: peeking: %v", r.pn, err) 123 } 124 if c[0] == 0xff { 125 break 126 } 127 r.readSym() 128 } 129 130 // Magic footer 131 buf = [8]uint8{} 132 r.readFull(buf[:]) 133 if string(buf[:]) != endmagic { 134 log.Fatalf("%s: invalid file end", r.pn) 135 } 136 } 137 138 func (r *objReader) readSlices() { 139 n := r.readInt() 140 r.data = make([]byte, n) 141 n = r.readInt() 142 r.reloc = make([]Reloc, n) 143 n = r.readInt() 144 r.pcdata = make([]Pcdata, n) 145 n = r.readInt() 146 r.autom = make([]Auto, n) 147 n = r.readInt() 148 r.funcdata = make([]*Symbol, n) 149 r.funcdataoff = make([]int64, n) 150 n = r.readInt() 151 r.file = make([]*Symbol, n) 152 } 153 154 // Symbols are prefixed so their content doesn't get confused with the magic footer. 155 const symPrefix = 0xfe 156 157 func (r *objReader) readSym() { 158 var c byte 159 var err error 160 if c, err = r.rd.ReadByte(); c != symPrefix || err != nil { 161 log.Fatalln("readSym out of sync") 162 } 163 if c, err = r.rd.ReadByte(); err != nil { 164 log.Fatalln("error reading input: ", err) 165 } 166 t := abiSymKindToSymKind[c] 167 s := r.readSymIndex() 168 flags := r.readInt() 169 dupok := flags&1 != 0 170 local := flags&2 != 0 171 makeTypelink := flags&4 != 0 172 size := r.readInt() 173 typ := r.readSymIndex() 174 data := r.readData() 175 nreloc := r.readInt() 176 pkg := objabi.PathToPrefix(r.lib.Pkg) 177 isdup := false 178 179 var dup *Symbol 180 if s.Type != 0 && s.Type != SXREF { 181 if (t == SDATA || t == SBSS || t == SNOPTRBSS) && len(data) == 0 && nreloc == 0 { 182 if s.Size < int64(size) { 183 s.Size = int64(size) 184 } 185 if typ != nil && s.Gotype == nil { 186 s.Gotype = typ 187 } 188 return 189 } 190 191 if (s.Type == SDATA || s.Type == SBSS || s.Type == SNOPTRBSS) && len(s.P) == 0 && len(s.R) == 0 { 192 goto overwrite 193 } 194 if s.Type != SBSS && s.Type != SNOPTRBSS && !dupok && !s.Attr.DuplicateOK() { 195 log.Fatalf("duplicate symbol %s (types %d and %d) in %s and %s", s.Name, s.Type, t, s.File, r.pn) 196 } 197 if len(s.P) > 0 { 198 dup = s 199 s = r.dupSym 200 isdup = true 201 } 202 } 203 204 overwrite: 205 s.File = pkg 206 if dupok { 207 s.Attr |= AttrDuplicateOK 208 } 209 if t == SXREF { 210 log.Fatalf("bad sxref") 211 } 212 if t == 0 { 213 log.Fatalf("missing type for %s in %s", s.Name, r.pn) 214 } 215 if t == SBSS && (s.Type == SRODATA || s.Type == SNOPTRBSS) { 216 t = s.Type 217 } 218 s.Type = t 219 if s.Size < int64(size) { 220 s.Size = int64(size) 221 } 222 s.Attr.Set(AttrLocal, local) 223 s.Attr.Set(AttrMakeTypelink, makeTypelink) 224 if typ != nil { 225 s.Gotype = typ 226 } 227 if isdup && typ != nil { // if bss sym defined multiple times, take type from any one def 228 dup.Gotype = typ 229 } 230 s.P = data 231 if nreloc > 0 { 232 s.R = r.reloc[:nreloc:nreloc] 233 if !isdup { 234 r.reloc = r.reloc[nreloc:] 235 } 236 237 for i := 0; i < nreloc; i++ { 238 s.R[i] = Reloc{ 239 Off: r.readInt32(), 240 Siz: r.readUint8(), 241 Type: objabi.RelocType(r.readInt32()), 242 Add: r.readInt64(), 243 Sym: r.readSymIndex(), 244 } 245 } 246 } 247 248 if s.Type == STEXT { 249 s.FuncInfo = new(FuncInfo) 250 pc := s.FuncInfo 251 252 pc.Args = r.readInt32() 253 pc.Locals = r.readInt32() 254 if r.readUint8() != 0 { 255 s.Attr |= AttrNoSplit 256 } 257 flags := r.readInt() 258 if flags&(1<<2) != 0 { 259 s.Attr |= AttrReflectMethod 260 } 261 if flags&(1<<3) != 0 { 262 s.Attr |= AttrShared 263 } 264 n := r.readInt() 265 pc.Autom = r.autom[:n:n] 266 if !isdup { 267 r.autom = r.autom[n:] 268 } 269 270 for i := 0; i < n; i++ { 271 pc.Autom[i] = Auto{ 272 Asym: r.readSymIndex(), 273 Aoffset: r.readInt32(), 274 Name: r.readInt16(), 275 Gotype: r.readSymIndex(), 276 } 277 } 278 279 pc.Pcsp.P = r.readData() 280 pc.Pcfile.P = r.readData() 281 pc.Pcline.P = r.readData() 282 pc.Pcinline.P = r.readData() 283 n = r.readInt() 284 pc.Pcdata = r.pcdata[:n:n] 285 if !isdup { 286 r.pcdata = r.pcdata[n:] 287 } 288 for i := 0; i < n; i++ { 289 pc.Pcdata[i].P = r.readData() 290 } 291 n = r.readInt() 292 pc.Funcdata = r.funcdata[:n:n] 293 pc.Funcdataoff = r.funcdataoff[:n:n] 294 if !isdup { 295 r.funcdata = r.funcdata[n:] 296 r.funcdataoff = r.funcdataoff[n:] 297 } 298 for i := 0; i < n; i++ { 299 pc.Funcdata[i] = r.readSymIndex() 300 } 301 for i := 0; i < n; i++ { 302 pc.Funcdataoff[i] = r.readInt64() 303 } 304 n = r.readInt() 305 pc.File = r.file[:n:n] 306 if !isdup { 307 r.file = r.file[n:] 308 } 309 for i := 0; i < n; i++ { 310 pc.File[i] = r.readSymIndex() 311 } 312 n = r.readInt() 313 pc.InlTree = make([]InlinedCall, n) 314 for i := 0; i < n; i++ { 315 pc.InlTree[i].Parent = r.readInt32() 316 pc.InlTree[i].File = r.readSymIndex() 317 pc.InlTree[i].Line = r.readInt32() 318 pc.InlTree[i].Func = r.readSymIndex() 319 } 320 321 if !dupok { 322 if s.Attr.OnList() { 323 log.Fatalf("symbol %s listed multiple times", s.Name) 324 } 325 s.Attr |= AttrOnList 326 r.lib.textp = append(r.lib.textp, s) 327 } else { 328 // there may ba a dup in another package 329 // put into a temp list and add to text later 330 if !isdup { 331 r.lib.dupTextSyms = append(r.lib.dupTextSyms, s) 332 } else { 333 r.lib.dupTextSyms = append(r.lib.dupTextSyms, dup) 334 } 335 } 336 } 337 if s.Type == SDWARFINFO { 338 r.patchDWARFName(s) 339 } 340 } 341 342 func (r *objReader) patchDWARFName(s *Symbol) { 343 // This is kind of ugly. Really the package name should not 344 // even be included here. 345 if s.Size < 1 || s.P[0] != dwarf.DW_ABRV_FUNCTION { 346 return 347 } 348 e := bytes.IndexByte(s.P, 0) 349 if e == -1 { 350 return 351 } 352 p := bytes.Index(s.P[:e], emptyPkg) 353 if p == -1 { 354 return 355 } 356 pkgprefix := []byte(objabi.PathToPrefix(r.lib.Pkg) + ".") 357 patched := bytes.Replace(s.P[:e], emptyPkg, pkgprefix, -1) 358 359 s.P = append(patched, s.P[e:]...) 360 delta := int64(len(s.P)) - s.Size 361 s.Size = int64(len(s.P)) 362 for i := range s.R { 363 r := &s.R[i] 364 if r.Off > int32(e) { 365 r.Off += int32(delta) 366 } 367 } 368 } 369 370 func (r *objReader) readFull(b []byte) { 371 _, err := io.ReadFull(r.rd, b) 372 if err != nil { 373 log.Fatalf("%s: error reading %s", r.pn, err) 374 } 375 } 376 377 func (r *objReader) readRef() { 378 if c, err := r.rd.ReadByte(); c != symPrefix || err != nil { 379 log.Fatalf("readSym out of sync") 380 } 381 name := r.readSymName() 382 v := r.readInt() 383 if v != 0 && v != 1 { 384 log.Fatalf("invalid symbol version for %q: %d", name, v) 385 } 386 if v == 1 { 387 v = r.localSymVersion 388 } 389 s := r.ctxt.Syms.Lookup(name, v) 390 r.refs = append(r.refs, s) 391 392 if s == nil || v != 0 { 393 return 394 } 395 if s.Name[0] == '$' && len(s.Name) > 5 && s.Type == 0 && len(s.P) == 0 { 396 x, err := strconv.ParseUint(s.Name[5:], 16, 64) 397 if err != nil { 398 log.Panicf("failed to parse $-symbol %s: %v", s.Name, err) 399 } 400 s.Type = SRODATA 401 s.Attr |= AttrLocal 402 switch s.Name[:5] { 403 case "$f32.": 404 if uint64(uint32(x)) != x { 405 log.Panicf("$-symbol %s too large: %d", s.Name, x) 406 } 407 Adduint32(r.ctxt, s, uint32(x)) 408 case "$f64.", "$i64.": 409 Adduint64(r.ctxt, s, x) 410 default: 411 log.Panicf("unrecognized $-symbol: %s", s.Name) 412 } 413 s.Attr.Set(AttrReachable, false) 414 } 415 if strings.HasPrefix(s.Name, "runtime.gcbits.") { 416 s.Attr |= AttrLocal 417 } 418 } 419 420 func (r *objReader) readInt64() int64 { 421 uv := uint64(0) 422 for shift := uint(0); ; shift += 7 { 423 if shift >= 64 { 424 log.Fatalf("corrupt input") 425 } 426 c, err := r.rd.ReadByte() 427 if err != nil { 428 log.Fatalln("error reading input: ", err) 429 } 430 uv |= uint64(c&0x7F) << shift 431 if c&0x80 == 0 { 432 break 433 } 434 } 435 436 return int64(uv>>1) ^ (int64(uv<<63) >> 63) 437 } 438 439 func (r *objReader) readInt() int { 440 n := r.readInt64() 441 if int64(int(n)) != n { 442 log.Panicf("%v out of range for int", n) 443 } 444 return int(n) 445 } 446 447 func (r *objReader) readInt32() int32 { 448 n := r.readInt64() 449 if int64(int32(n)) != n { 450 log.Panicf("%v out of range for int32", n) 451 } 452 return int32(n) 453 } 454 455 func (r *objReader) readInt16() int16 { 456 n := r.readInt64() 457 if int64(int16(n)) != n { 458 log.Panicf("%v out of range for int16", n) 459 } 460 return int16(n) 461 } 462 463 func (r *objReader) readUint8() uint8 { 464 n := r.readInt64() 465 if int64(uint8(n)) != n { 466 log.Panicf("%v out of range for uint8", n) 467 } 468 return uint8(n) 469 } 470 471 func (r *objReader) readString() string { 472 n := r.readInt() 473 if cap(r.rdBuf) < n { 474 r.rdBuf = make([]byte, 2*n) 475 } 476 r.readFull(r.rdBuf[:n]) 477 return string(r.rdBuf[:n]) 478 } 479 480 func (r *objReader) readData() []byte { 481 n := r.readInt() 482 p := r.data[:n:n] 483 r.data = r.data[n:] 484 return p 485 } 486 487 // readSymName reads a symbol name, replacing all "". with pkg. 488 func (r *objReader) readSymName() string { 489 pkg := objabi.PathToPrefix(r.lib.Pkg) 490 n := r.readInt() 491 if n == 0 { 492 r.readInt64() 493 return "" 494 } 495 if cap(r.rdBuf) < n { 496 r.rdBuf = make([]byte, 2*n) 497 } 498 origName, err := r.rd.Peek(n) 499 if err == bufio.ErrBufferFull { 500 // Long symbol names are rare but exist. One source is type 501 // symbols for types with long string forms. See #15104. 502 origName = make([]byte, n) 503 r.readFull(origName) 504 } else if err != nil { 505 log.Fatalf("%s: error reading symbol: %v", r.pn, err) 506 } 507 adjName := r.rdBuf[:0] 508 for { 509 i := bytes.Index(origName, emptyPkg) 510 if i == -1 { 511 s := string(append(adjName, origName...)) 512 // Read past the peeked origName, now that we're done with it, 513 // using the rfBuf (also no longer used) as the scratch space. 514 // TODO: use bufio.Reader.Discard if available instead? 515 if err == nil { 516 r.readFull(r.rdBuf[:n]) 517 } 518 r.rdBuf = adjName[:0] // in case 2*n wasn't enough 519 520 if Buildmode == BuildmodeShared || *FlagLinkshared { 521 // These types are included in the symbol 522 // table when dynamically linking. To keep 523 // binary size down, we replace the names 524 // with SHA-1 prefixes. 525 // 526 // Keep the type.. prefix, which parts of the 527 // linker (like the DWARF generator) know means 528 // the symbol is not decodable. 529 // 530 // Leave type.runtime. symbols alone, because 531 // other parts of the linker manipulates them, 532 // and also symbols whose names would not be 533 // shortened by this process. 534 if len(s) > 14 && strings.HasPrefix(s, "type.") && !strings.HasPrefix(s, "type.runtime.") { 535 hash := sha1.Sum([]byte(s)) 536 prefix := "type." 537 if s[5] == '.' { 538 prefix = "type.." 539 } 540 s = prefix + base64.StdEncoding.EncodeToString(hash[:6]) 541 } 542 } 543 return s 544 } 545 adjName = append(adjName, origName[:i]...) 546 adjName = append(adjName, pkg...) 547 adjName = append(adjName, '.') 548 origName = origName[i+len(emptyPkg):] 549 } 550 } 551 552 // Reads the index of a symbol reference and resolves it to a symbol 553 func (r *objReader) readSymIndex() *Symbol { 554 i := r.readInt() 555 return r.refs[i] 556 }