blob: cae7e4aafe71343af954adddcfc45ab51fc40d90 [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[] = {
Russ Coxebce7942014-07-02 15:41:29 -040047 {"darwin", Hdarwin},
48 {"dragonfly", Hdragonfly},
49 {"elf", Helf},
50 {"freebsd", Hfreebsd},
51 {"linux", Hlinux},
Russ Coxa99369f2015-01-28 22:39:57 -050052 {"android", Hlinux}, // must be after "linux" entry or else headstr(Hlinux) == "android"
Russ Coxebce7942014-07-02 15:41:29 -040053 {"nacl", Hnacl},
54 {"netbsd", Hnetbsd},
55 {"openbsd", Hopenbsd},
56 {"plan9", Hplan9},
57 {"solaris", Hsolaris},
58 {"windows", Hwindows},
59 {"windowsgui", Hwindows},
Russ Coxa9f6db52013-12-16 12:51:58 -050060};
61
62int
63headtype(char *name)
64{
65 int i;
66
Russ Cox3be0a0e2015-01-12 15:17:10 -050067 for(i=0; i < nelem(headers); i++)
Russ Coxa9f6db52013-12-16 12:51:58 -050068 if(strcmp(name, headers[i].name) == 0)
69 return headers[i].val;
70 return -1;
71}
72
73char*
74headstr(int v)
75{
76 static char buf[20];
77 int i;
78
Russ Cox3be0a0e2015-01-12 15:17:10 -050079 for(i=0; i < nelem(headers); i++)
Russ Coxa9f6db52013-12-16 12:51:58 -050080 if(v == headers[i].val)
81 return headers[i].name;
82 snprint(buf, sizeof buf, "%d", v);
83 return buf;
84}
85
Russ Cox7d507dc2013-12-08 22:49:37 -050086Link*
87linknew(LinkArch *arch)
88{
89 Link *ctxt;
90 char *p;
91 char buf[1024];
92
Russ Coxdba9eb32015-01-16 19:22:58 -050093 linksetexp();
Shenghou Ma08ee2662014-08-06 00:25:41 -040094 nuxiinit(arch);
Russ Cox7d507dc2013-12-08 22:49:37 -050095
96 ctxt = emallocz(sizeof *ctxt);
97 ctxt->arch = arch;
98 ctxt->version = HistVersion;
Russ Coxe97b3ab2014-04-15 20:46:46 -040099 ctxt->goroot = getgoroot();
100 ctxt->goroot_final = getenv("GOROOT_FINAL");
101 if(ctxt->goroot_final != nil && ctxt->goroot_final[0] == '\0')
102 ctxt->goroot_final = nil;
Russ Cox7d507dc2013-12-08 22:49:37 -0500103
Russ Cox7d507dc2013-12-08 22:49:37 -0500104 p = getgoarch();
Russ Coxd9c6ae62014-02-27 20:37:00 -0500105 if(strcmp(p, arch->name) != 0)
106 sysfatal("invalid goarch %s (want %s)", p, arch->name);
Russ Cox7d507dc2013-12-08 22:49:37 -0500107
108 if(getwd(buf, sizeof buf) == 0)
109 strcpy(buf, "/???");
110 if(yy_isalpha(buf[0]) && buf[1] == ':') {
111 // On Windows.
112 ctxt->windows = 1;
113
114 // Canonicalize path by converting \ to / (Windows accepts both).
115 for(p=buf; *p; p++)
116 if(*p == '\\')
117 *p = '/';
118 }
119 ctxt->pathname = strdup(buf);
Russ Coxa9f6db52013-12-16 12:51:58 -0500120
121 ctxt->headtype = headtype(getgoos());
122 if(ctxt->headtype < 0)
123 sysfatal("unknown goos %s", getgoos());
124
125 // Record thread-local storage offset.
Russ Cox90093f02014-04-15 13:45:39 -0400126 // TODO(rsc): Move tlsoffset back into the linker.
Russ Coxa9f6db52013-12-16 12:51:58 -0500127 switch(ctxt->headtype) {
128 default:
129 sysfatal("unknown thread-local storage offset for %s", headstr(ctxt->headtype));
130 case Hplan9:
Russ Coxa9f6db52013-12-16 12:51:58 -0500131 case Hwindows:
132 break;
133 case Hlinux:
134 case Hfreebsd:
135 case Hnetbsd:
136 case Hopenbsd:
137 case Hdragonfly:
Shenghou Ma6ebf59b2014-02-09 16:45:38 -0500138 case Hsolaris:
Russ Coxa9f6db52013-12-16 12:51:58 -0500139 /*
140 * ELF uses TLS offset negative from FS.
141 * Translate 0(FS) and 8(FS) into -16(FS) and -8(FS).
142 * Known to low-level assembly in package runtime and runtime/cgo.
143 */
144 ctxt->tlsoffset = -2*ctxt->arch->ptrsize;
145 break;
Rémy Oudompheng88f84b32014-03-14 21:33:55 +0100146
Russ Coxd9c6ae62014-02-27 20:37:00 -0500147 case Hnacl:
Rémy Oudompheng88f84b32014-03-14 21:33:55 +0100148 switch(ctxt->arch->thechar) {
149 default:
150 sysfatal("unknown thread-local storage offset for nacl/%s", ctxt->arch->name);
Shenghou Ma10837152014-12-26 00:15:07 -0500151 case '5':
152 ctxt->tlsoffset = 0;
153 break;
Rémy Oudompheng88f84b32014-03-14 21:33:55 +0100154 case '6':
155 ctxt->tlsoffset = 0;
156 break;
157 case '8':
158 ctxt->tlsoffset = -8;
159 break;
160 }
Russ Coxd9c6ae62014-02-27 20:37:00 -0500161 break;
Russ Coxa9f6db52013-12-16 12:51:58 -0500162
163 case Hdarwin:
164 /*
165 * OS X system constants - offset from 0(GS) to our TLS.
Russ Cox220a6de2014-09-08 00:06:45 -0400166 * Explained in ../../runtime/cgo/gcc_darwin_*.c.
Russ Coxa9f6db52013-12-16 12:51:58 -0500167 */
168 switch(ctxt->arch->thechar) {
169 default:
170 sysfatal("unknown thread-local storage offset for darwin/%s", ctxt->arch->name);
171 case '6':
172 ctxt->tlsoffset = 0x8a0;
173 break;
174 case '8':
175 ctxt->tlsoffset = 0x468;
176 break;
Shenghou Ma10837152014-12-26 00:15:07 -0500177 case '5':
178 ctxt->tlsoffset = 0; // dummy value, not needed
179 break;
Russ Coxa9f6db52013-12-16 12:51:58 -0500180 }
181 break;
182 }
183
184 // On arm, record goarm.
185 if(ctxt->arch->thechar == '5') {
186 p = getgoarm();
187 if(p != nil)
188 ctxt->goarm = atoi(p);
189 else
190 ctxt->goarm = 6;
191 }
Russ Cox7d507dc2013-12-08 22:49:37 -0500192
193 return ctxt;
194}
195
196LSym*
197linknewsym(Link *ctxt, char *symb, int v)
198{
199 LSym *s;
Russ Cox7d507dc2013-12-08 22:49:37 -0500200
Russ Cox7d507dc2013-12-08 22:49:37 -0500201 s = malloc(sizeof(*s));
202 memset(s, 0, sizeof(*s));
203
204 s->dynid = -1;
205 s->plt = -1;
206 s->got = -1;
Russ Cox8e0e09a2014-07-21 17:38:48 -0400207 s->name = estrdup(symb);
Russ Cox7d507dc2013-12-08 22:49:37 -0500208 s->type = 0;
209 s->version = v;
210 s->value = 0;
211 s->sig = 0;
212 s->size = 0;
213 ctxt->nsymbol++;
214
215 s->allsym = ctxt->allsym;
216 ctxt->allsym = s;
217
218 return s;
219}
220
221static LSym*
222_lookup(Link *ctxt, char *symb, int v, int creat)
223{
224 LSym *s;
225 char *p;
226 uint32 h;
227 int c;
228
229 h = v;
230 for(p=symb; c = *p; p++)
231 h = h+h+h + c;
232 h &= 0xffffff;
233 h %= LINKHASH;
234 for(s = ctxt->hash[h]; s != nil; s = s->hash)
Russ Cox468cf822014-04-16 11:53:14 -0400235 if(s->version == v && strcmp(s->name, symb) == 0)
Russ Cox7d507dc2013-12-08 22:49:37 -0500236 return s;
237 if(!creat)
238 return nil;
239
240 s = linknewsym(ctxt, symb, v);
241 s->extname = s->name;
242 s->hash = ctxt->hash[h];
243 ctxt->hash[h] = s;
244
245 return s;
246}
247
248LSym*
249linklookup(Link *ctxt, char *name, int v)
250{
251 return _lookup(ctxt, name, v, 1);
252}
253
254// read-only lookup
255LSym*
256linkrlookup(Link *ctxt, char *name, int v)
257{
258 return _lookup(ctxt, name, v, 0);
259}
260
261int
262linksymfmt(Fmt *f)
263{
264 LSym *s;
265
266 s = va_arg(f->args, LSym*);
267 if(s == nil)
268 return fmtstrcpy(f, "<nil>");
269
270 return fmtstrcpy(f, s->name);
271}