blob: 7ff64350df818db9a6c5c51d81baffce763df744 [file] [log] [blame]
Russ Cox7d507dc2013-12-08 22:49:37 -05001// 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#include <u.h>
33#include <libc.h>
34#include <bio.h>
35#include <link.h>
36
37static int
38yy_isalpha(int c)
39{
40 return c >= 0 && c <= 0xFF && isalpha(c);
41}
42
Russ Coxa9f6db52013-12-16 12:51:58 -050043static struct {
44 char *name;
45 int val;
46} headers[] = {
47 "darwin", Hdarwin,
48 "dragonfly", Hdragonfly,
49 "elf", Helf,
50 "freebsd", Hfreebsd,
51 "linux", Hlinux,
52 "netbsd", Hnetbsd,
53 "openbsd", Hopenbsd,
54 "plan9", Hplan9,
Shenghou Ma6ebf59b2014-02-09 16:45:38 -050055 "solaris", Hsolaris,
Russ Coxa9f6db52013-12-16 12:51:58 -050056 "windows", Hwindows,
57 "windowsgui", Hwindows,
58 0, 0
59};
60
61int
62headtype(char *name)
63{
64 int i;
65
66 for(i=0; headers[i].name; i++)
67 if(strcmp(name, headers[i].name) == 0)
68 return headers[i].val;
69 return -1;
70}
71
72char*
73headstr(int v)
74{
75 static char buf[20];
76 int i;
77
78 for(i=0; headers[i].name; i++)
79 if(v == headers[i].val)
80 return headers[i].name;
81 snprint(buf, sizeof buf, "%d", v);
82 return buf;
83}
84
Russ Cox7d507dc2013-12-08 22:49:37 -050085Link*
86linknew(LinkArch *arch)
87{
88 Link *ctxt;
89 char *p;
90 char buf[1024];
91
92 nuxiinit();
93
94 ctxt = emallocz(sizeof *ctxt);
95 ctxt->arch = arch;
96 ctxt->version = HistVersion;
97
Russ Cox7d507dc2013-12-08 22:49:37 -050098 p = getgoarch();
99 if(strncmp(p, arch->name, strlen(arch->name)) != 0)
100 sysfatal("invalid goarch %s (want %s or derivative)", p, arch->name);
101
102 if(getwd(buf, sizeof buf) == 0)
103 strcpy(buf, "/???");
104 if(yy_isalpha(buf[0]) && buf[1] == ':') {
105 // On Windows.
106 ctxt->windows = 1;
107
108 // Canonicalize path by converting \ to / (Windows accepts both).
109 for(p=buf; *p; p++)
110 if(*p == '\\')
111 *p = '/';
112 }
113 ctxt->pathname = strdup(buf);
Russ Coxa9f6db52013-12-16 12:51:58 -0500114
115 ctxt->headtype = headtype(getgoos());
116 if(ctxt->headtype < 0)
117 sysfatal("unknown goos %s", getgoos());
118
119 // Record thread-local storage offset.
120 switch(ctxt->headtype) {
121 default:
122 sysfatal("unknown thread-local storage offset for %s", headstr(ctxt->headtype));
123 case Hplan9:
124 ctxt->tlsoffset = -2*ctxt->arch->ptrsize;
125 break;
126 case Hwindows:
127 break;
128 case Hlinux:
129 case Hfreebsd:
130 case Hnetbsd:
131 case Hopenbsd:
132 case Hdragonfly:
Shenghou Ma6ebf59b2014-02-09 16:45:38 -0500133 case Hsolaris:
Russ Coxa9f6db52013-12-16 12:51:58 -0500134 /*
135 * ELF uses TLS offset negative from FS.
136 * Translate 0(FS) and 8(FS) into -16(FS) and -8(FS).
137 * Known to low-level assembly in package runtime and runtime/cgo.
138 */
139 ctxt->tlsoffset = -2*ctxt->arch->ptrsize;
140 break;
141
142 case Hdarwin:
143 /*
144 * OS X system constants - offset from 0(GS) to our TLS.
145 * Explained in ../../pkg/runtime/cgo/gcc_darwin_*.c.
146 */
147 switch(ctxt->arch->thechar) {
148 default:
149 sysfatal("unknown thread-local storage offset for darwin/%s", ctxt->arch->name);
150 case '6':
151 ctxt->tlsoffset = 0x8a0;
152 break;
153 case '8':
154 ctxt->tlsoffset = 0x468;
155 break;
156 }
157 break;
158 }
159
160 // On arm, record goarm.
161 if(ctxt->arch->thechar == '5') {
162 p = getgoarm();
163 if(p != nil)
164 ctxt->goarm = atoi(p);
165 else
166 ctxt->goarm = 6;
167 }
Russ Cox7d507dc2013-12-08 22:49:37 -0500168
169 return ctxt;
170}
171
172LSym*
173linknewsym(Link *ctxt, char *symb, int v)
174{
175 LSym *s;
176 int l;
177
178 l = strlen(symb) + 1;
179 s = malloc(sizeof(*s));
180 memset(s, 0, sizeof(*s));
181
182 s->dynid = -1;
183 s->plt = -1;
184 s->got = -1;
185 s->name = malloc(l + 1);
186 memmove(s->name, symb, l);
187 s->name[l] = '\0';
188
189 s->type = 0;
190 s->version = v;
191 s->value = 0;
192 s->sig = 0;
193 s->size = 0;
194 ctxt->nsymbol++;
195
196 s->allsym = ctxt->allsym;
197 ctxt->allsym = s;
198
199 return s;
200}
201
202static LSym*
203_lookup(Link *ctxt, char *symb, int v, int creat)
204{
205 LSym *s;
206 char *p;
207 uint32 h;
208 int c;
209
210 h = v;
211 for(p=symb; c = *p; p++)
212 h = h+h+h + c;
213 h &= 0xffffff;
214 h %= LINKHASH;
215 for(s = ctxt->hash[h]; s != nil; s = s->hash)
216 if(strcmp(s->name, symb) == 0)
217 return s;
218 if(!creat)
219 return nil;
220
221 s = linknewsym(ctxt, symb, v);
222 s->extname = s->name;
223 s->hash = ctxt->hash[h];
224 ctxt->hash[h] = s;
225
226 return s;
227}
228
229LSym*
230linklookup(Link *ctxt, char *name, int v)
231{
232 return _lookup(ctxt, name, v, 1);
233}
234
235// read-only lookup
236LSym*
237linkrlookup(Link *ctxt, char *name, int v)
238{
239 return _lookup(ctxt, name, v, 0);
240}
241
242int
243linksymfmt(Fmt *f)
244{
245 LSym *s;
246
247 s = va_arg(f->args, LSym*);
248 if(s == nil)
249 return fmtstrcpy(f, "<nil>");
250
251 return fmtstrcpy(f, s->name);
252}