github.com/miolini/go@v0.0.0-20160405192216-fca68c8cb408/src/cmd/internal/obj/data.go (about) 1 // Derived from Inferno utils/6l/obj.c and utils/6l/span.c 2 // http://code.google.com/p/inferno-os/source/browse/utils/6l/obj.c 3 // http://code.google.com/p/inferno-os/source/browse/utils/6l/span.c 4 // 5 // Copyright © 1994-1999 Lucent Technologies Inc. All rights reserved. 6 // Portions Copyright © 1995-1997 C H Forsyth (forsyth@terzarima.net) 7 // Portions Copyright © 1997-1999 Vita Nuova Limited 8 // Portions Copyright © 2000-2007 Vita Nuova Holdings Limited (www.vitanuova.com) 9 // Portions Copyright © 2004,2006 Bruce Ellis 10 // Portions Copyright © 2005-2007 C H Forsyth (forsyth@terzarima.net) 11 // Revisions Copyright © 2000-2007 Lucent Technologies Inc. and others 12 // Portions Copyright © 2009 The Go Authors. All rights reserved. 13 // 14 // Permission is hereby granted, free of charge, to any person obtaining a copy 15 // of this software and associated documentation files (the "Software"), to deal 16 // in the Software without restriction, including without limitation the rights 17 // to use, copy, modify, merge, publish, distribute, sublicense, and/or sell 18 // copies of the Software, and to permit persons to whom the Software is 19 // furnished to do so, subject to the following conditions: 20 // 21 // The above copyright notice and this permission notice shall be included in 22 // all copies or substantial portions of the Software. 23 // 24 // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR 25 // IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, 26 // FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE 27 // AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER 28 // LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, 29 // OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN 30 // THE SOFTWARE. 31 32 package obj 33 34 import ( 35 "log" 36 "math" 37 ) 38 39 // Grow increases the length of s.P to lsiz. 40 func (s *LSym) Grow(lsiz int64) { 41 siz := int(lsiz) 42 if int64(siz) != lsiz { 43 log.Fatalf("LSym.Grow size %d too long", lsiz) 44 } 45 if len(s.P) >= siz { 46 return 47 } 48 // TODO(dfc) append cap-len at once, rather than 49 // one byte at a time. 50 for cap(s.P) < siz { 51 s.P = append(s.P[:cap(s.P)], 0) 52 } 53 s.P = s.P[:siz] 54 } 55 56 // GrowCap increases the capacity of s.P to c. 57 func (s *LSym) GrowCap(c int64) { 58 if int64(cap(s.P)) >= c { 59 return 60 } 61 if s.P == nil { 62 s.P = make([]byte, 0, c) 63 return 64 } 65 b := make([]byte, len(s.P), c) 66 copy(b, s.P) 67 s.P = b 68 } 69 70 // prepwrite prepares to write data of size siz into s at offset off. 71 func (s *LSym) prepwrite(ctxt *Link, off int64, siz int) { 72 if off < 0 || siz < 0 || off >= 1<<30 { 73 log.Fatalf("prepwrite: bad off=%d siz=%d", off, siz) 74 } 75 if s.Type == SBSS || s.Type == STLSBSS { 76 ctxt.Diag("cannot supply data for BSS var") 77 } 78 s.Grow(off + int64(siz)) 79 } 80 81 // WriteFloat32 writes f into s at offset off. 82 func (s *LSym) WriteFloat32(ctxt *Link, off int64, f float32) { 83 s.prepwrite(ctxt, off, 4) 84 ctxt.Arch.ByteOrder.PutUint32(s.P[off:], math.Float32bits(f)) 85 } 86 87 // WriteFloat64 writes f into s at offset off. 88 func (s *LSym) WriteFloat64(ctxt *Link, off int64, f float64) { 89 s.prepwrite(ctxt, off, 8) 90 ctxt.Arch.ByteOrder.PutUint64(s.P[off:], math.Float64bits(f)) 91 } 92 93 // WriteInt writes an integer i of size siz into s at offset off. 94 func (s *LSym) WriteInt(ctxt *Link, off int64, siz int, i int64) { 95 s.prepwrite(ctxt, off, siz) 96 switch siz { 97 default: 98 ctxt.Diag("WriteInt: bad integer size: %d", siz) 99 case 1: 100 s.P[off] = byte(i) 101 case 2: 102 ctxt.Arch.ByteOrder.PutUint16(s.P[off:], uint16(i)) 103 case 4: 104 ctxt.Arch.ByteOrder.PutUint32(s.P[off:], uint32(i)) 105 case 8: 106 ctxt.Arch.ByteOrder.PutUint64(s.P[off:], uint64(i)) 107 } 108 } 109 110 // WriteAddr writes an address of size siz into s at offset off. 111 // rsym and roff specify the relocation for the address. 112 func (s *LSym) WriteAddr(ctxt *Link, off int64, siz int, rsym *LSym, roff int64) { 113 if siz != ctxt.Arch.Ptrsize { 114 ctxt.Diag("WriteAddr: bad address size: %d", siz) 115 } 116 s.prepwrite(ctxt, off, siz) 117 r := Addrel(s) 118 r.Off = int32(off) 119 r.Siz = uint8(siz) 120 r.Sym = rsym 121 r.Type = R_ADDR 122 r.Add = roff 123 } 124 125 // WriteString writes a string of size siz into s at offset off. 126 func (s *LSym) WriteString(ctxt *Link, off int64, siz int, str string) { 127 if siz < len(str) { 128 ctxt.Diag("WriteString: bad string size: %d < %d", siz, len(str)) 129 } 130 s.prepwrite(ctxt, off, siz) 131 copy(s.P[off:off+int64(siz)], str) 132 } 133 134 func Addrel(s *LSym) *Reloc { 135 s.R = append(s.R, Reloc{}) 136 return &s.R[len(s.R)-1] 137 } 138 139 func Setuintxx(ctxt *Link, s *LSym, off int64, v uint64, wid int64) int64 { 140 if s.Type == 0 { 141 s.Type = SDATA 142 } 143 if s.Size < off+wid { 144 s.Size = off + wid 145 s.Grow(s.Size) 146 } 147 148 switch wid { 149 case 1: 150 s.P[off] = uint8(v) 151 case 2: 152 ctxt.Arch.ByteOrder.PutUint16(s.P[off:], uint16(v)) 153 case 4: 154 ctxt.Arch.ByteOrder.PutUint32(s.P[off:], uint32(v)) 155 case 8: 156 ctxt.Arch.ByteOrder.PutUint64(s.P[off:], uint64(v)) 157 } 158 159 return off + wid 160 }