1 // basic_test.cc -- a test case for gold
2 
3 // Copyright (C) 2006-2014 Free Software Foundation, Inc.
4 // Written by Ian Lance Taylor <iant@google.com>.
5 
6 // This file is part of gold.
7 
8 // This program is free software; you can redistribute it and/or modify
9 // it under the terms of the GNU General Public License as published by
10 // the Free Software Foundation; either version 3 of the License, or
11 // (at your option) any later version.
12 
13 // This program is distributed in the hope that it will be useful,
14 // but WITHOUT ANY WARRANTY; without even the implied warranty of
15 // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
16 // GNU General Public License for more details.
17 
18 // You should have received a copy of the GNU General Public License
19 // along with this program; if not, write to the Free Software
20 // Foundation, Inc., 51 Franklin Street - Fifth Floor, Boston,
21 // MA 02110-1301, USA.
22 
23 // The goal of this program is to produce as many different types of
24 // relocations as we can in a stand-alone program that does not use
25 // TLS.  This program is compiled without optimization.
26 
27 // 1  Code reference to global data.
28 // 2  Code reference to static data.
29 // 3  Code reference to BSS data.
30 // 4  Code reference to offset within global data.
31 // 5  Code reference to offset within static data.
32 // 6  Code reference to offset within BSS data.
33 // 7  Switch statement with a table of destinations.
34 // 8  Taking the address of a label (a gcc extension).
35 // 9  Taking the address of a nested function (a gcc extension).
36 // 10 Data reference to global data.
37 // 11 Data reference to static data.
38 // 12 Data reference to BSS data.
39 // 13 Data reference to offset within global data.
40 // 14 Data reference to offset within static data.
41 // 15 Data reference to offset within BSS data.
42 // 16 Virtual table.
43 // 17 Inline function.
44 // 18 Call through pointer to method.
45 // 19 Initialize variable to pointer to method.
46 // 20 Global constructor and destructor.
47 
48 // 1 Code reference to global data.
49 int t1 = 11;
50 
51 // 2 Code reference to static data.
52 static int t2 = 22;
53 
54 // 3 Code reference to BSS data (initialized after program starts, to
55 // 33).
56 int t3;
57 
58 // 4 Code reference to offset within global data.
59 char t4[] = "Hello, world";
60 
61 // 5 Code reference to offset within static data.
62 static char t5[] = "Hello, world";
63 
64 // 6 Code reference to offset within BSS data (initialized after
65 // program starts, to contents of t4).
66 char t6[13];
67 
68 // Test cases 1 through 6.
69 
70 bool
t1_6()71 t1_6()
72 {
73   return (t1 == 11
74 	  && t2 == 22
75 	  && t3 == 33
76 	  && t4[5] == ','
77 	  && t5[7] == 'w'
78 	  && t6[9] == 'r');
79 }
80 
81 // 7  Switch statement with a table of destinations.
82 
83 int
t7(int i)84 t7(int i)
85 {
86   switch (i)
87     {
88     case 0:
89       return 12;
90     case 1:
91       return 34;
92     case 2:
93       return 56;
94     case 3:
95       return 78;
96     case 4:
97       return 90;
98     case 5:
99       return 13;
100     case 6:
101       return 0;
102     case 7:
103       return 57;
104     case 8:
105       return 79;
106     case 9:
107       return 81;
108     default:
109       return 144;
110     }
111 }
112 
113 // 8  Taking the address of a label (a gcc extension).
114 
115 int
t8(int i)116 t8(int i)
117 {
118   for (int j = 0; j < 10; ++j)
119     {
120       void* p;
121       if (i + j > 6)
122 	p = &&lab1;
123       else
124 	p = &&lab2;
125       if (j == 7)
126 	goto *p;
127     }
128   return 15;
129  lab1:
130   return 0;
131  lab2:
132   return 12;
133 }
134 
135 // 9  Taking the address of a nested function (a gcc extension).
136 // Disabled because this is only supported in C, not C++.
137 
138 int
t9a(int (* pfn)(int))139 t9a(int (*pfn)(int))
140 {
141   return (*pfn)(10) - 10;
142 }
143 
144 int
t9(int i)145 t9(int i)
146 {
147 #if 0
148   int
149   t9c(int j)
150   {
151     return i + j;
152   }
153   return t9a(&t9c);
154 #else
155   return i;
156 #endif
157 }
158 
159 // 10 Data reference to global data.
160 int* t10 = &t1;
161 
162 // 11 Data reference to static data.
163 int* t11 = &t2;
164 
165 // 12 Data reference to BSS data.
166 int* t12 = &t3;
167 
168 // 13 Data reference to offset within global data.
169 char* t13 = &t4[6];
170 
171 // 14 Data reference to offset within static data.
172 char* t14 = &t5[8];
173 
174 // 15 Data reference to offset within BSS data.
175 char* t15 = &t6[10];
176 
177 // Test cases 10 through 15.
178 
179 bool
t10_15()180 t10_15()
181 {
182   return (*t10 == 11
183 	  && *t11 == 22
184 	  && *t12 == 33
185 	  && *t13 == ' '
186 	  && *t14 == 'o'
187 	  && *t15 == 'l');
188 }
189 
190 // 16 Virtual table.
191 
192 class t16a
193 {
194  public:
195   virtual
~t16a()196   ~t16a()
197   { }
198   virtual int
t()199   t()
200   { return 83; }
201 };
202 
203 class t16b : public t16a
204 {
205  public:
206   virtual int
t()207   t()
208   { return 92; }
209 };
210 
211 t16b t16v;
212 
213 bool
t16()214 t16()
215 {
216   return t16v.t() == 92;
217 }
218 
219 // 17 Inline function.
220 
221 inline int
t17a()222 t17a()
223 {
224   return 74;
225 }
226 
227 bool
t17()228 t17()
229 {
230   return t17a() == 74;
231 }
232 
233 // 18 Call through pointer to method.
234 
235 class t18a
236 {
237  public:
238   int
ta()239   ta()
240   { return 65; }
241 
242   int
tb()243   tb()
244   { return 90; }
245 };
246 
247 t18a t18v;
248 
249 int
t18f(int (t18a::* p)())250 t18f(int (t18a::* p)())
251 {
252   return (t18v.*p)();
253 }
254 
255 bool
t18()256 t18()
257 {
258   return t18f(&t18a::ta) == 65;
259 }
260 
261 // 19 Initialize variable to pointer to method.
262 
263 int (t18a::* t19v)() = &t18a::tb;
264 
265 bool
t19()266 t19()
267 {
268   return (t18v.*t19v)() == 90;
269 }
270 
271 // 20 Global constructor and destructor.
272 
273 class t20a
274 {
275  public:
t20a()276   t20a()
277     : i(96)
278   { }
~t20a()279   ~t20a()
280   { }
281   int
get() const282   get() const
283   { return this->i; }
284  private:
285   int i;
286 };
287 
288 t20a t20v;
289 
290 bool
t20()291 t20()
292 {
293   return t20v.get() == 96;
294 }
295 
296 // Main function.  Initialize variables and call test functions.
297 
298 int
main()299 main()
300 {
301   t3 = 33;
302   for (int i = 0; i < 13; ++i)
303     t6[i] = t4[i];
304 
305   if (t1_6()
306       && t7(6) == 0
307       && t8(0) == 0
308       && t9(5) == 5
309       && t10_15()
310       && t16()
311       && t17()
312       && t18()
313       && t19()
314       && t20())
315     return 0;
316   else
317     return 1;
318 }
319