annotate dmd/StructDeclaration.d @ 130:60bb0fe4563e

dmdfe 2.037 first main iteration
author Eldar Insafutdinov <e.insafutdinov@gmail.com>
date Thu, 09 Sep 2010 22:51:44 +0100
parents 010eb8f0e18d
children 206db751bd4c
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
rev   line source
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
1 module dmd.StructDeclaration;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
2
114
e28b18c23469 added a module dmd.common for commonly used stuff
Trass3r
parents: 103
diff changeset
3 import dmd.common;
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
4 import dmd.AggregateDeclaration;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
5 import dmd.FuncDeclaration;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
6 import dmd.DeclarationExp;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
7 import dmd.VoidInitializer;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
8 import dmd.Initializer;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
9 import dmd.ExpInitializer;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
10 import dmd.TOK;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
11 import dmd.Statement;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
12 import dmd.VarExp;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
13 import dmd.CompoundStatement;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
14 import dmd.AssignExp;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
15 import dmd.DotVarExp;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
16 import dmd.AddrExp;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
17 import dmd.CastExp;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
18 import dmd.PostBlitDeclaration;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
19 import dmd.Lexer;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
20 import dmd.ExpStatement;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
21 import dmd.DotIdExp;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
22 import dmd.TypeSArray;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
23 import dmd.ThisExp;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
24 import dmd.ThisDeclaration;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
25 import dmd.TypeFunction;
130
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
26 import dmd.Parameter;
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
27 import dmd.Id;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
28 import dmd.TY;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
29 import dmd.LINK;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
30 import dmd.Type;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
31 import dmd.DsymbolTable;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
32 import dmd.ArrayTypes;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
33 import dmd.Loc;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
34 import dmd.STC;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
35 import dmd.Identifier;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
36 import dmd.TemplateInstance;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
37 import dmd.Dsymbol;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
38 import dmd.Scope;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
39 import dmd.OutBuffer;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
40 import dmd.HdrGenState;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
41 import dmd.PROT;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
42 import dmd.TypeStruct;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
43 import dmd.expression.Util;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
44 import dmd.Expression;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
45 import dmd.IdentifierExp;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
46 import dmd.PtrExp;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
47 import dmd.CallExp;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
48 import dmd.ReturnStatement;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
49 import dmd.ScopeDsymbol;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
50 import dmd.Module;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
51 import dmd.VarDeclaration;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
52 import dmd.InvariantDeclaration;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
53 import dmd.NewDeclaration;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
54 import dmd.DeleteDeclaration;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
55 import dmd.Global;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
56
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
57 import dmd.backend.dt_t;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
58 import dmd.backend.Util;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
59 import dmd.backend.SC;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
60 import dmd.backend.DT;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
61 import dmd.backend.FL;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
62 import dmd.backend.glue;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
63
129
010eb8f0e18d further work on dmd test suite
korDen
parents: 126
diff changeset
64 import std.stdio;
010eb8f0e18d further work on dmd test suite
korDen
parents: 126
diff changeset
65
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
66 class StructDeclaration : AggregateDeclaration
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
67 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
68 bool zeroInit; // true if initialize with 0 fill
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
69
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
70 version (DMDV2) {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
71 int hasIdentityAssign; // !=0 if has identity opAssign
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
72 FuncDeclaration cpctor; // generated copy-constructor, if any
130
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
73 FuncDeclaration eq; // bool opEquals(ref const T), if any
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
74
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
75 FuncDeclarations postblits; // Array of postblit functions
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
76 FuncDeclaration postblit; // aggregate postblit
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
77 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
78
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
79 this(Loc loc, Identifier id)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
80 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
81 super(loc, id);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
82
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
83 // For forward references
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
84 type = new TypeStruct(this);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
85
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
86 postblits = new FuncDeclarations(); ///
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
87 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
88
72
2e2a5c3f943a reduced warnings by adding override to the methods
Trass3r
parents: 49
diff changeset
89 override Dsymbol syntaxCopy(Dsymbol s)
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
90 {
49
0aa7d1437ada AttribDeclaration.oneMember
korDen
parents: 13
diff changeset
91 StructDeclaration sd;
0aa7d1437ada AttribDeclaration.oneMember
korDen
parents: 13
diff changeset
92
0aa7d1437ada AttribDeclaration.oneMember
korDen
parents: 13
diff changeset
93 if (s)
0aa7d1437ada AttribDeclaration.oneMember
korDen
parents: 13
diff changeset
94 sd = cast(StructDeclaration)s;
0aa7d1437ada AttribDeclaration.oneMember
korDen
parents: 13
diff changeset
95 else
0aa7d1437ada AttribDeclaration.oneMember
korDen
parents: 13
diff changeset
96 sd = new StructDeclaration(loc, ident);
0aa7d1437ada AttribDeclaration.oneMember
korDen
parents: 13
diff changeset
97 ScopeDsymbol.syntaxCopy(sd);
0aa7d1437ada AttribDeclaration.oneMember
korDen
parents: 13
diff changeset
98 return sd;
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
99 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
100
72
2e2a5c3f943a reduced warnings by adding override to the methods
Trass3r
parents: 49
diff changeset
101 override void semantic(Scope sc)
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
102 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
103 Scope sc2;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
104
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
105 //printf("+StructDeclaration.semantic(this=%p, '%s')\n", this, toChars());
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
106
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
107 //static int count; if (++count == 20) *(char*)0=0;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
108
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
109 assert(type);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
110 if (!members) // if forward reference
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
111 return;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
112
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
113 if (symtab)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
114 { if (!scope_)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
115 return; // semantic() already completed
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
116 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
117 else
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
118 symtab = new DsymbolTable();
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
119
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
120 Scope scx = null;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
121 if (scope_)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
122 { sc = scope_;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
123 scx = scope_; // save so we don't make redundant copies
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
124 scope_ = null;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
125 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
126
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
127 parent = sc.parent;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
128 type = type.semantic(loc, sc);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
129 version (STRUCTTHISREF) {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
130 handle = type;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
131 } else {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
132 handle = type.pointerTo();
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
133 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
134 structalign = sc.structalign;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
135 protection = sc.protection;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
136 storage_class |= sc.stc;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
137 if (sc.stc & STC.STCdeprecated)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
138 isdeprecated = 1;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
139 assert(!isAnonymous());
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
140 if (sc.stc & STC.STCabstract)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
141 error("structs, unions cannot be abstract");
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
142 version (DMDV2) {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
143 if (storage_class & STC.STCimmutable)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
144 type = type.invariantOf();
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
145 else if (storage_class & STC.STCconst)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
146 type = type.constOf();
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
147 else if (storage_class & STC.STCshared)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
148 type = type.sharedOf();
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
149 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
150
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
151 if (sizeok == 0) // if not already done the addMember step
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
152 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
153 int hasfunctions = 0;
77
ad4792a1cfd6 more D-ification container accessing
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 74
diff changeset
154 foreach(Dsymbol s; members)
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
155 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
156 //printf("adding member '%s' to '%s'\n", s.toChars(), this.toChars());
13
427f8aa74d28 On the road to make Phobos compilable
korDen
parents: 0
diff changeset
157 s.addMember(sc, this, true);
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
158 if (s.isFuncDeclaration())
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
159 hasfunctions = 1;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
160 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
161
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
162 // If nested struct, add in hidden 'this' pointer to outer scope
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
163 if (hasfunctions && !(storage_class & STC.STCstatic))
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
164 { Dsymbol s = toParent2();
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
165 if (s)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
166 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
167 AggregateDeclaration ad = s.isAggregateDeclaration();
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
168 FuncDeclaration fd = s.isFuncDeclaration();
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
169
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
170 TemplateInstance ti;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
171 if (ad && (ti = ad.parent.isTemplateInstance()) !is null && ti.isnested || fd)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
172 { isnested = true;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
173 Type t;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
174 if (ad)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
175 t = ad.handle;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
176 else if (fd)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
177 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
178 AggregateDeclaration add = fd.isMember2();
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
179 if (add)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
180 t = add.handle;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
181 else
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
182 t = Type.tvoidptr;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
183 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
184 else
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
185 assert(0);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
186 if (t.ty == TY.Tstruct)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
187 t = Type.tvoidptr; // t should not be a ref type
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
188 assert(!vthis);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
189 vthis = new ThisDeclaration(loc, t);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
190 //vthis.storage_class |= STC.STCref;
74
7e0d548de9e6 Switch Arrays of Dsymbols to the new templated Vector type
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 72
diff changeset
191 members.push(vthis);
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
192 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
193 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
194 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
195 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
196
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
197 sizeok = 0;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
198 sc2 = sc.push(this);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
199 sc2.stc &= storage_class & STC.STC_TYPECTOR;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
200 sc2.parent = this;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
201 if (isUnionDeclaration())
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
202 sc2.inunion = 1;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
203 sc2.protection = PROT.PROTpublic;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
204 sc2.explicitProtection = 0;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
205
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
206 int members_dim = members.dim;
77
ad4792a1cfd6 more D-ification container accessing
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 74
diff changeset
207 foreach(Dsymbol s; members)
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
208 {
129
010eb8f0e18d further work on dmd test suite
korDen
parents: 126
diff changeset
209 s.semantic(sc2);
010eb8f0e18d further work on dmd test suite
korDen
parents: 126
diff changeset
210 if (isUnionDeclaration())
010eb8f0e18d further work on dmd test suite
korDen
parents: 126
diff changeset
211 sc2.offset = 0;
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
212 static if (false) {
129
010eb8f0e18d further work on dmd test suite
korDen
parents: 126
diff changeset
213 if (sizeok == 2)
010eb8f0e18d further work on dmd test suite
korDen
parents: 126
diff changeset
214 { //printf("forward reference\n");
010eb8f0e18d further work on dmd test suite
korDen
parents: 126
diff changeset
215 break;
010eb8f0e18d further work on dmd test suite
korDen
parents: 126
diff changeset
216 }
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
217 }
129
010eb8f0e18d further work on dmd test suite
korDen
parents: 126
diff changeset
218 if (auto d = s.isDeclaration())
010eb8f0e18d further work on dmd test suite
korDen
parents: 126
diff changeset
219 {
010eb8f0e18d further work on dmd test suite
korDen
parents: 126
diff changeset
220 if (auto t = d.type) {
010eb8f0e18d further work on dmd test suite
korDen
parents: 126
diff changeset
221 if (t.toBasetype().ty == TY.Tstruct) {
010eb8f0e18d further work on dmd test suite
korDen
parents: 126
diff changeset
222 auto ad = t.toDsymbol(sc).isThis();
010eb8f0e18d further work on dmd test suite
korDen
parents: 126
diff changeset
223 /*
010eb8f0e18d further work on dmd test suite
korDen
parents: 126
diff changeset
224 StructDeclaration sd = cast(StructDeclaration)foo;
010eb8f0e18d further work on dmd test suite
korDen
parents: 126
diff changeset
225 if (foo && !sd) {
010eb8f0e18d further work on dmd test suite
korDen
parents: 126
diff changeset
226 writeln(t.classin);
010eb8f0e18d further work on dmd test suite
korDen
parents: 126
diff changeset
227 writeln(foo.classinfo.name);
010eb8f0e18d further work on dmd test suite
korDen
parents: 126
diff changeset
228 assert(false);
010eb8f0e18d further work on dmd test suite
korDen
parents: 126
diff changeset
229 }
010eb8f0e18d further work on dmd test suite
korDen
parents: 126
diff changeset
230 */
010eb8f0e18d further work on dmd test suite
korDen
parents: 126
diff changeset
231 if (ad && ad.isnested)
010eb8f0e18d further work on dmd test suite
korDen
parents: 126
diff changeset
232 error("inner struct %s cannot be a field", ad.toChars());
010eb8f0e18d further work on dmd test suite
korDen
parents: 126
diff changeset
233 }
010eb8f0e18d further work on dmd test suite
korDen
parents: 126
diff changeset
234 }
010eb8f0e18d further work on dmd test suite
korDen
parents: 126
diff changeset
235 }
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
236 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
237
130
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
238 version(DMDV1) {
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
239 /* This doesn't work for DMDV2 because (ref S) and (S) parameter
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
240 * lists will overload the same.
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
241 */
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
242 /* The TypeInfo_Struct is expecting an opEquals and opCmp with
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
243 * a parameter that is a pointer to the struct. But if there
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
244 * isn't one, but is an opEquals or opCmp with a value, write
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
245 * another that is a shell around the value:
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
246 * int opCmp(struct *p) { return opCmp(*p); }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
247 */
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
248
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
249 TypeFunction tfeqptr;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
250 {
130
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
251 auto arguments = new Parameters;
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
252 auto arg = new Parameter(STC.STCin, handle, Id.p, null);
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
253
129
010eb8f0e18d further work on dmd test suite
korDen
parents: 126
diff changeset
254 arguments.push(arg);
010eb8f0e18d further work on dmd test suite
korDen
parents: 126
diff changeset
255 tfeqptr = new TypeFunction(arguments, Type.tint32, 0, LINK.LINKd);
010eb8f0e18d further work on dmd test suite
korDen
parents: 126
diff changeset
256 tfeqptr = cast(TypeFunction)tfeqptr.semantic(Loc(0), sc);
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
257 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
258
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
259 TypeFunction tfeq;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
260 {
130
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
261 auto arguments = new Parameters;
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
262 auto arg = new Parameter(STC.STCin, type, null, null);
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
263
129
010eb8f0e18d further work on dmd test suite
korDen
parents: 126
diff changeset
264 arguments.push(arg);
010eb8f0e18d further work on dmd test suite
korDen
parents: 126
diff changeset
265 tfeq = new TypeFunction(arguments, Type.tint32, 0, LINK.LINKd);
010eb8f0e18d further work on dmd test suite
korDen
parents: 126
diff changeset
266 tfeq = cast(TypeFunction)tfeq.semantic(Loc(0), sc);
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
267 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
268
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
269 Identifier id = Id.eq;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
270 for (int j = 0; j < 2; j++)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
271 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
272 Dsymbol s = search_function(this, id);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
273 FuncDeclaration fdx = s ? s.isFuncDeclaration() : null;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
274 if (fdx)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
275 { FuncDeclaration fd = fdx.overloadExactMatch(tfeqptr);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
276 if (!fd)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
277 { fd = fdx.overloadExactMatch(tfeq);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
278 if (fd)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
279 { // Create the thunk, fdptr
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
280 FuncDeclaration fdptr = new FuncDeclaration(loc, loc, fdx.ident, STC.STCundefined, tfeqptr);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
281 Expression e = new IdentifierExp(loc, Id.p);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
282 e = new PtrExp(loc, e);
84
be2ab491772e Expressions -> Vector!Expression
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 79
diff changeset
283 auto args = new Expressions();
be2ab491772e Expressions -> Vector!Expression
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 79
diff changeset
284 args.push(e);
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
285 e = new IdentifierExp(loc, id);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
286 e = new CallExp(loc, e, args);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
287 fdptr.fbody = new ReturnStatement(loc, e);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
288 ScopeDsymbol ss = fdx.parent.isScopeDsymbol();
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
289 assert(ss);
74
7e0d548de9e6 Switch Arrays of Dsymbols to the new templated Vector type
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 72
diff changeset
290 ss.members.push(fdptr);
13
427f8aa74d28 On the road to make Phobos compilable
korDen
parents: 0
diff changeset
291 fdptr.addMember(sc, ss, true);
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
292 fdptr.semantic(sc2);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
293 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
294 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
295 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
296
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
297 id = Id.cmp;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
298 }
130
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
299 }
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
300 version (DMDV2) {
130
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
301 /* Try to find the opEquals function. Build it if necessary.
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
302 */
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
303 TypeFunction *tfeqptr;
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
304 { // bool opEquals(const T*) const;
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
305 auto parameters = new Parameters;
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
306 version(STRUCTTHISREF) {
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
307 // bool opEquals(ref const T) const;
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
308 auto param = new Parameter(STC.STCref, type.constOf(), null, null);
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
309 } else {
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
310 // bool opEquals(const T*) const;
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
311 auto param = new Parameter(STC.STCin, type.pointerTo(), null, null);
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
312 }
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
313
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
314 parameters.push(param);
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
315 tfeqptr = new TypeFunction(parameters, Type.tbool, 0, LINK.LINKd);
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
316 tfeqptr.mod = MOD.MODconst;
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
317 tfeqptr = cast(TypeFunction)(tfeqptr.semantic(0, sc2));
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
318
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
319 Dsymbol s = search_function(this, Id.eq);
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
320 FuncDeclaration fdx = s ? s.isFuncDeclaration() : null;
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
321 if (fdx)
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
322 {
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
323 eq = fdx.overloadExactMatch(tfeqptr);
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
324 if (!eq)
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
325 fdx.error("type signature should be %s not %s", tfeqptr.toChars(), fdx.type.toChars());
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
326 }
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
327
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
328 if (!eq)
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
329 eq = buildOpEquals(sc2);
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
330 }
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
331
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
332 dtor = buildDtor(sc2);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
333 postblit = buildPostBlit(sc2);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
334 cpctor = buildCpCtor(sc2);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
335 buildOpAssign(sc2);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
336 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
337
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
338 sc2.pop();
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
339
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
340 if (sizeok == 2)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
341 { // semantic() failed because of forward references.
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
342 // Unwind what we did, and defer it for later
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
343 fields.setDim(0);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
344 structsize = 0;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
345 alignsize = 0;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
346 structalign = 0;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
347
87
b17640f0e4e8 Fixed a bug with a Scope.this(Scope enclosing) being called instead of Scope.clone() method (as a copy ctor replacement)
korDen
parents: 79
diff changeset
348 scope_ = scx ? scx : sc.clone();
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
349 scope_.setNoFree();
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
350 scope_.module_.addDeferredSemantic(this);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
351 //printf("\tdeferring %s\n", toChars());
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
352 return;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
353 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
354
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
355 // 0 sized struct's are set to 1 byte
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
356 if (structsize == 0)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
357 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
358 structsize = 1;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
359 alignsize = 1;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
360 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
361
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
362 // Round struct size up to next alignsize boundary.
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
363 // This will ensure that arrays of structs will get their internals
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
364 // aligned properly.
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
365 structsize = (structsize + alignsize - 1) & ~(alignsize - 1);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
366
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
367 sizeok = 1;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
368 Module.dprogress++;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
369
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
370 //printf("-StructDeclaration.semantic(this=%p, '%s')\n", this, toChars());
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
371
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
372 // Determine if struct is all zeros or not
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
373 zeroInit = true;
85
8e69d041a99d Previous commit didn't compile. Fixed.
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 84
diff changeset
374 foreach (VarDeclaration vd; fields)
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
375 {
79
43073c7c7769 updated to 2.035
Trass3r
parents: 77
diff changeset
376 if (vd && !vd.isDataseg())
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
377 {
79
43073c7c7769 updated to 2.035
Trass3r
parents: 77
diff changeset
378 if (vd.init)
43073c7c7769 updated to 2.035
Trass3r
parents: 77
diff changeset
379 {
43073c7c7769 updated to 2.035
Trass3r
parents: 77
diff changeset
380 // Should examine init to see if it is really all 0's
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
381 zeroInit = false;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
382 break;
79
43073c7c7769 updated to 2.035
Trass3r
parents: 77
diff changeset
383 }
43073c7c7769 updated to 2.035
Trass3r
parents: 77
diff changeset
384 else
43073c7c7769 updated to 2.035
Trass3r
parents: 77
diff changeset
385 {
43073c7c7769 updated to 2.035
Trass3r
parents: 77
diff changeset
386 if (!vd.type.isZeroInit(loc))
43073c7c7769 updated to 2.035
Trass3r
parents: 77
diff changeset
387 {
43073c7c7769 updated to 2.035
Trass3r
parents: 77
diff changeset
388 zeroInit = false;
43073c7c7769 updated to 2.035
Trass3r
parents: 77
diff changeset
389 break;
43073c7c7769 updated to 2.035
Trass3r
parents: 77
diff changeset
390 }
43073c7c7769 updated to 2.035
Trass3r
parents: 77
diff changeset
391 }
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
392 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
393 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
394
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
395 /* Look for special member functions.
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
396 */
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
397 version (DMDV2) {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
398 ctor = search(Loc(0), Id.ctor, 0);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
399 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
400 inv = cast(InvariantDeclaration)search(Loc(0), Id.classInvariant, 0);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
401 aggNew = cast(NewDeclaration)search(Loc(0), Id.classNew, 0);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
402 aggDelete = cast(DeleteDeclaration)search(Loc(0), Id.classDelete, 0);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
403
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
404 if (sc.func)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
405 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
406 semantic2(sc);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
407 semantic3(sc);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
408 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
409 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
410
72
2e2a5c3f943a reduced warnings by adding override to the methods
Trass3r
parents: 49
diff changeset
411 override void toCBuffer(OutBuffer buf, HdrGenState* hgs)
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
412 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
413 assert(false);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
414 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
415
72
2e2a5c3f943a reduced warnings by adding override to the methods
Trass3r
parents: 49
diff changeset
416 override string mangle()
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
417 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
418 //printf("StructDeclaration.mangle() '%s'\n", toChars());
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
419 return Dsymbol.mangle();
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
420 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
421
72
2e2a5c3f943a reduced warnings by adding override to the methods
Trass3r
parents: 49
diff changeset
422 override string kind()
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
423 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
424 assert(false);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
425 }
130
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
426
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
427 version(DMDV1)
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
428 {
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
429 Expression cloneMembers()
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
430 {
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
431 assert(false);
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
432 }
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
433 }
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
434
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
435 version(DMDV2)
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
436 {
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
437 /*******************************************
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
438 * We need an opAssign for the struct if
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
439 * it has a destructor or a postblit.
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
440 * We need to generate one if a user-specified one does not exist.
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
441 */
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
442 bool needOpAssign()
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
443 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
444 static if (false) {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
445 printf("StructDeclaration.needOpAssign() %s\n", toChars());
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
446 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
447 if (hasIdentityAssign)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
448 goto Ldontneed;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
449
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
450 if (dtor || postblit)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
451 goto Lneed;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
452
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
453 /* If any of the fields need an opAssign, then we
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
454 * need it too.
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
455 */
85
8e69d041a99d Previous commit didn't compile. Fixed.
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 84
diff changeset
456 foreach (VarDeclaration v; fields)
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
457 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
458 assert(v && v.storage_class & STC.STCfield);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
459 if (v.storage_class & STC.STCref)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
460 continue;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
461 Type tv = v.type.toBasetype();
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
462 while (tv.ty == TY.Tsarray)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
463 { TypeSArray ta = cast(TypeSArray)tv;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
464 tv = tv.nextOf().toBasetype();
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
465 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
466 if (tv.ty == TY.Tstruct)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
467 { TypeStruct ts = cast(TypeStruct)tv;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
468 StructDeclaration sd = ts.sym;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
469 if (sd.needOpAssign())
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
470 goto Lneed;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
471 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
472 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
473 Ldontneed:
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
474 static if (false) {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
475 printf("\tdontneed\n");
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
476 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
477 return false;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
478
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
479 Lneed:
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
480 static if (false) {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
481 printf("\tneed\n");
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
482 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
483 return true;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
484 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
485
130
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
486 /*******************************************
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
487 * We need an opEquals for the struct if
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
488 * any fields has an opEquals.
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
489 * Generate one if a user-specified one does not exist.
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
490 */
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
491 bool needOpEquals()
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
492 {
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
493 enum X = 0;
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
494 static if (X) printf("StructDeclaration::needOpEquals() %s\n", toChars());
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
495
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
496 /* If any of the fields has an opEquals, then we
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
497 * need it too.
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
498 */
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
499 foreach (s; fields)
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
500 {
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
501 VarDeclaration v = s.isVarDeclaration();
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
502 assert(v && v.storage_class & STC.STCfield);
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
503 if (v.storage_class & STC.STCref)
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
504 continue;
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
505 Type tv = v.type.toBasetype();
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
506 while (tv.ty == Tsarray)
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
507 { auto ta = cast(TypeSArray)tv;
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
508 tv = tv.nextOf().toBasetype();
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
509 }
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
510 if (tv.ty == Tstruct)
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
511 { auto ts = cast(TypeStruct)tv;
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
512 StructDeclaration sd = ts.sym;
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
513 if (sd.eq)
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
514 goto Lneed;
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
515 }
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
516 }
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
517 Ldontneed:
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
518 static if (X) printf("\tdontneed\n");
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
519 return false;
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
520
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
521 Lneed:
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
522 static if (X) printf("\tneed\n");
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
523 return true;
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
524 }
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
525
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
526 /******************************************
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
527 * Build opAssign for struct.
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
528 * S* opAssign(S s) { ... }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
529 */
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
530 FuncDeclaration buildOpAssign(Scope sc)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
531 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
532 if (!needOpAssign())
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
533 return null;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
534
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
535 //printf("StructDeclaration.buildOpAssign() %s\n", toChars());
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
536
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
537 FuncDeclaration fop = null;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
538
130
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
539 auto param = new Parameter(STC.STCnodtor, type, Id.p, null);
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
540 auto fparams = new Parameters;
126
1765f3ef917d ClassDeclarations, Arguments -> Vector
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 114
diff changeset
541 fparams.push(param);
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
542 Type ftype = new TypeFunction(fparams, handle, false, LINK.LINKd);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
543 version (STRUCTTHISREF) {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
544 (cast(TypeFunction)ftype).isref = 1;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
545 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
546
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
547 fop = new FuncDeclaration(Loc(0), Loc(0), Id.assign, STC.STCundefined, ftype);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
548
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
549 Expression e = null;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
550 if (postblit)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
551 { /* Swap:
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
552 * tmp = *this; *this = s; tmp.dtor();
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
553 */
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
554 //printf("\tswap copy\n");
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
555 Identifier idtmp = Lexer.uniqueId("__tmp");
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
556 VarDeclaration tmp;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
557 AssignExp ec = null;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
558 if (dtor)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
559 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
560 tmp = new VarDeclaration(Loc(0), type, idtmp, new VoidInitializer(Loc(0)));
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
561 tmp.noauto = true;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
562 e = new DeclarationExp(Loc(0), tmp);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
563
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
564 Expression e2;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
565 version (STRUCTTHISREF) {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
566 e2 = new ThisExp(Loc(0));
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
567 } else {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
568 e2 = new PtrExp(Loc(0), new ThisExp(Loc(0)));
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
569 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
570 ec = new AssignExp(Loc(0), new VarExp(Loc(0), tmp), e2);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
571 ec.op = TOK.TOKblit;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
572 e = Expression.combine(e, ec);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
573 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
574 Expression e2;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
575 version (STRUCTTHISREF) {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
576 e2 = new ThisExp(Loc(0));
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
577 } else {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
578 e2 = new PtrExp(Loc(0), new ThisExp(Loc(0)));
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
579 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
580
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
581 ec = new AssignExp(Loc(0), e2, new IdentifierExp(Loc(0), Id.p));
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
582 ec.op = TOK.TOKblit;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
583 e = Expression.combine(e, ec);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
584 if (dtor)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
585 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
586 /* Instead of running the destructor on s, run it
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
587 * on tmp. This avoids needing to copy tmp back in to s.
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
588 */
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
589 Expression ecc = new DotVarExp(Loc(0), new VarExp(Loc(0), tmp), dtor, 0);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
590 ecc = new CallExp(Loc(0), ecc);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
591 e = Expression.combine(e, ecc);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
592 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
593 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
594 else
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
595 { /* Do memberwise copy
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
596 */
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
597 //printf("\tmemberwise copy\n");
85
8e69d041a99d Previous commit didn't compile. Fixed.
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 84
diff changeset
598 foreach (VarDeclaration v; fields)
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
599 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
600 assert(v && v.storage_class & STC.STCfield);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
601 // this.v = s.v;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
602 AssignExp ec = new AssignExp(Loc(0), new DotVarExp(Loc(0), new ThisExp(Loc(0)), v, 0), new DotVarExp(Loc(0), new IdentifierExp(Loc(0), Id.p), v, 0));
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
603 ec.op = TOK.TOKblit;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
604 e = Expression.combine(e, ec);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
605 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
606 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
607 Statement s1 = new ExpStatement(Loc(0), e);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
608
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
609 /* Add:
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
610 * return this;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
611 */
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
612 e = new ThisExp(Loc(0));
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
613 Statement s2 = new ReturnStatement(Loc(0), e);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
614
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
615 fop.fbody = new CompoundStatement(Loc(0), s1, s2);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
616
74
7e0d548de9e6 Switch Arrays of Dsymbols to the new templated Vector type
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 72
diff changeset
617 members.push(fop);
13
427f8aa74d28 On the road to make Phobos compilable
korDen
parents: 0
diff changeset
618 fop.addMember(sc, this, true);
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
619
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
620 sc = sc.push();
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
621 sc.stc = STC.STCundefined;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
622 sc.linkage = LINK.LINKd;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
623
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
624 fop.semantic(sc);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
625
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
626 sc.pop();
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
627
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
628 //printf("-StructDeclaration.buildOpAssign() %s\n", toChars());
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
629
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
630 return fop;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
631 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
632
130
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
633 /******************************************
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
634 * Build opEquals for struct.
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
635 * const bool opEquals(const ref S s) { ... }
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
636 */
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
637 FuncDeclaration buildOpEquals(Scope sc)
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
638 {
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
639 if (!needOpEquals())
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
640 return null;
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
641 //printf("StructDeclaration::buildOpEquals() %s\n", toChars());
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
642 Loc loc = this.loc;
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
643
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
644 auto parameters = new Parameters;
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
645 version (STRUCTTHISREF) {
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
646 // bool opEquals(ref const T) const;
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
647 auto param = new Parameter(STC.STCref, type.constOf(), Id.p, NULL);
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
648 } else {
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
649 // bool opEquals(const T*) const;
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
650 auto param = new Parameter(STC.STCin, type.pointerTo(), Id.p, NULL);
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
651 }
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
652
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
653 parameters.push(param);
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
654 auto ftype = new TypeFunction(parameters, Type.tbool, 0, LINKd);
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
655 ftype.mod = MOD.MODconst;
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
656 ftype = cast(TypeFunction)ftype.semantic(loc, sc);
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
657
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
658 auto fop = new FuncDeclaration(loc, 0, Id.eq, STD.STCundefined, ftype);
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
659
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
660 Expression *e = NULL;
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
661 /* Do memberwise compare
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
662 */
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
663 //printf("\tmemberwise compare\n");
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
664 foreach (s; fields)
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
665 {
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
666 VarDeclaration v = s.isVarDeclaration();
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
667 assert(v && v.storage_class & STD.STCfield);
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
668 if (v.storage_class & STC.STCref)
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
669 assert(0); // what should we do with this?
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
670 // this.v == s.v;
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
671 auto ec = new EqualExp(TOKequal, loc,
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
672 new DotVarExp(loc, new ThisExp(loc), v, 0),
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
673 new DotVarExp(loc, new IdentifierExp(loc, Id.p), v, 0));
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
674 if (e)
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
675 e = new AndAndExp(loc, e, ec);
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
676 else
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
677 e = ec;
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
678 }
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
679 if (!e)
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
680 e = new IntegerExp(loc, 1, Type.tbool);
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
681 fop.fbody = new ReturnStatement(loc, e);
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
682
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
683 members.push(fop);
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
684 fop.addMember(sc, this, 1);
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
685
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
686 sc = sc.push();
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
687 sc.stc = 0;
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
688 sc.linkage = LINK.LINKd;
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
689
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
690 fop.semantic(sc);
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
691
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
692 sc.pop();
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
693
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
694 //printf("-StructDeclaration::buildOpEquals() %s\n", toChars());
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
695
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
696 return fop;
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
697 }
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
698
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
699 /*****************************************
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
700 * Create inclusive postblit for struct by aggregating
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
701 * all the postblits in postblits[] with the postblits for
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
702 * all the members.
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
703 * Note the close similarity with AggregateDeclaration.buildDtor(),
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
704 * and the ordering changes (runs forward instead of backwards).
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
705 */
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
706
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
707 version (DMDV2) {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
708 FuncDeclaration buildPostBlit(Scope sc)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
709 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
710 //printf("StructDeclaration.buildPostBlit() %s\n", toChars());
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
711 Expression e = null;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
712
85
8e69d041a99d Previous commit didn't compile. Fixed.
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 84
diff changeset
713 foreach (VarDeclaration v; fields)
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
714 {
79
43073c7c7769 updated to 2.035
Trass3r
parents: 77
diff changeset
715 assert(v && v.storage_class & STC.STCfield);
43073c7c7769 updated to 2.035
Trass3r
parents: 77
diff changeset
716 if (v.storage_class & STC.STCref)
43073c7c7769 updated to 2.035
Trass3r
parents: 77
diff changeset
717 continue;
43073c7c7769 updated to 2.035
Trass3r
parents: 77
diff changeset
718 Type tv = v.type.toBasetype();
43073c7c7769 updated to 2.035
Trass3r
parents: 77
diff changeset
719 size_t dim = 1;
43073c7c7769 updated to 2.035
Trass3r
parents: 77
diff changeset
720 while (tv.ty == TY.Tsarray)
43073c7c7769 updated to 2.035
Trass3r
parents: 77
diff changeset
721 { TypeSArray ta = cast(TypeSArray)tv;
43073c7c7769 updated to 2.035
Trass3r
parents: 77
diff changeset
722 dim *= (cast(TypeSArray)tv).dim.toInteger();
43073c7c7769 updated to 2.035
Trass3r
parents: 77
diff changeset
723 tv = tv.nextOf().toBasetype();
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
724 }
79
43073c7c7769 updated to 2.035
Trass3r
parents: 77
diff changeset
725 if (tv.ty == TY.Tstruct)
43073c7c7769 updated to 2.035
Trass3r
parents: 77
diff changeset
726 { TypeStruct ts = cast(TypeStruct)tv;
43073c7c7769 updated to 2.035
Trass3r
parents: 77
diff changeset
727 StructDeclaration sd = ts.sym;
43073c7c7769 updated to 2.035
Trass3r
parents: 77
diff changeset
728 if (sd.postblit)
43073c7c7769 updated to 2.035
Trass3r
parents: 77
diff changeset
729 { Expression ex;
43073c7c7769 updated to 2.035
Trass3r
parents: 77
diff changeset
730
43073c7c7769 updated to 2.035
Trass3r
parents: 77
diff changeset
731 // this.v
43073c7c7769 updated to 2.035
Trass3r
parents: 77
diff changeset
732 ex = new ThisExp(Loc(0));
43073c7c7769 updated to 2.035
Trass3r
parents: 77
diff changeset
733 ex = new DotVarExp(Loc(0), ex, v, 0);
43073c7c7769 updated to 2.035
Trass3r
parents: 77
diff changeset
734
43073c7c7769 updated to 2.035
Trass3r
parents: 77
diff changeset
735 if (dim == 1)
43073c7c7769 updated to 2.035
Trass3r
parents: 77
diff changeset
736 { // this.v.postblit()
43073c7c7769 updated to 2.035
Trass3r
parents: 77
diff changeset
737 ex = new DotVarExp(Loc(0), ex, sd.postblit, 0);
43073c7c7769 updated to 2.035
Trass3r
parents: 77
diff changeset
738 ex = new CallExp(Loc(0), ex);
43073c7c7769 updated to 2.035
Trass3r
parents: 77
diff changeset
739 }
43073c7c7769 updated to 2.035
Trass3r
parents: 77
diff changeset
740 else
43073c7c7769 updated to 2.035
Trass3r
parents: 77
diff changeset
741 {
43073c7c7769 updated to 2.035
Trass3r
parents: 77
diff changeset
742 // Typeinfo.postblit(cast(void*)&this.v);
43073c7c7769 updated to 2.035
Trass3r
parents: 77
diff changeset
743 Expression ea = new AddrExp(Loc(0), ex);
43073c7c7769 updated to 2.035
Trass3r
parents: 77
diff changeset
744 ea = new CastExp(Loc(0), ea, Type.tvoid.pointerTo());
43073c7c7769 updated to 2.035
Trass3r
parents: 77
diff changeset
745
43073c7c7769 updated to 2.035
Trass3r
parents: 77
diff changeset
746 Expression et = v.type.getTypeInfo(sc);
103
e6090d1aea7c fixed dmd.Id + some small fixups
Trass3r
parents: 95
diff changeset
747 et = new DotIdExp(Loc(0), et, Id._postblit);
79
43073c7c7769 updated to 2.035
Trass3r
parents: 77
diff changeset
748
43073c7c7769 updated to 2.035
Trass3r
parents: 77
diff changeset
749 ex = new CallExp(Loc(0), et, ea);
43073c7c7769 updated to 2.035
Trass3r
parents: 77
diff changeset
750 }
43073c7c7769 updated to 2.035
Trass3r
parents: 77
diff changeset
751 e = Expression.combine(e, ex); // combine in forward order
43073c7c7769 updated to 2.035
Trass3r
parents: 77
diff changeset
752 }
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
753 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
754 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
755
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
756 /* Build our own "postblit" which executes e
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
757 */
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
758 if (e)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
759 { //printf("Building __fieldPostBlit()\n");
74
7e0d548de9e6 Switch Arrays of Dsymbols to the new templated Vector type
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 72
diff changeset
760 auto dd = new PostBlitDeclaration(Loc(0), Loc(0), Lexer.idPool("__fieldPostBlit"));
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
761 dd.fbody = new ExpStatement(Loc(0), e);
74
7e0d548de9e6 Switch Arrays of Dsymbols to the new templated Vector type
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 72
diff changeset
762 postblits.shift(dd);
7e0d548de9e6 Switch Arrays of Dsymbols to the new templated Vector type
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 72
diff changeset
763 members.push(dd);
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
764 dd.semantic(sc);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
765 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
766
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
767 switch (postblits.dim)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
768 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
769 case 0:
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
770 return null;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
771
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
772 case 1:
77
ad4792a1cfd6 more D-ification container accessing
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 74
diff changeset
773 return cast(FuncDeclaration)postblits[0];
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
774
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
775 default:
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
776 e = null;
77
ad4792a1cfd6 more D-ification container accessing
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 74
diff changeset
777 foreach(FuncDeclaration fd; postblits)
ad4792a1cfd6 more D-ification container accessing
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 74
diff changeset
778 {
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
779 Expression ex = new ThisExp(Loc(0));
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
780 ex = new DotVarExp(Loc(0), ex, fd, 0);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
781 ex = new CallExp(Loc(0), ex);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
782 e = Expression.combine(e, ex);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
783 }
74
7e0d548de9e6 Switch Arrays of Dsymbols to the new templated Vector type
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 72
diff changeset
784 auto dd = new PostBlitDeclaration(Loc(0), Loc(0), Lexer.idPool("__aggrPostBlit"));
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
785 dd.fbody = new ExpStatement(Loc(0), e);
74
7e0d548de9e6 Switch Arrays of Dsymbols to the new templated Vector type
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 72
diff changeset
786 members.push(dd);
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
787 dd.semantic(sc);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
788 return dd;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
789 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
790 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
791 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
792
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
793 /*******************************************
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
794 * Build copy constructor for struct.
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
795 * Copy constructors are compiler generated only, and are only
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
796 * callable from the compiler. They are not user accessible.
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
797 * A copy constructor is:
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
798 * void cpctpr(ref S s)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
799 * {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
800 * *this = s;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
801 * this.postBlit();
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
802 * }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
803 * This is done so:
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
804 * - postBlit() never sees uninitialized data
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
805 * - memcpy can be much more efficient than memberwise copy
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
806 * - no fields are overlooked
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
807 */
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
808 FuncDeclaration buildCpCtor(Scope sc)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
809 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
810 //printf("StructDeclaration.buildCpCtor() %s\n", toChars());
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
811 FuncDeclaration fcp = null;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
812
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
813 /* Copy constructor is only necessary if there is a postblit function,
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
814 * otherwise the code generator will just do a bit copy.
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
815 */
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
816 if (postblit)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
817 {
129
010eb8f0e18d further work on dmd test suite
korDen
parents: 126
diff changeset
818 //printf("generating cpctor\n");
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
819
130
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
820 auto param = new Parameter(STC.STCref, type, Id.p, null);
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
821 auto fparams = new Parameters;
129
010eb8f0e18d further work on dmd test suite
korDen
parents: 126
diff changeset
822 fparams.push(param);
010eb8f0e18d further work on dmd test suite
korDen
parents: 126
diff changeset
823 Type ftype = new TypeFunction(fparams, Type.tvoid, false, LINK.LINKd);
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
824
129
010eb8f0e18d further work on dmd test suite
korDen
parents: 126
diff changeset
825 fcp = new FuncDeclaration(Loc(0), Loc(0), Id.cpctor, STC.STCundefined, ftype);
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
826
129
010eb8f0e18d further work on dmd test suite
korDen
parents: 126
diff changeset
827 // Build *this = p;
010eb8f0e18d further work on dmd test suite
korDen
parents: 126
diff changeset
828 Expression e = new ThisExp(Loc(0));
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
829 version (STRUCTTHISREF) {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
830 } else {
129
010eb8f0e18d further work on dmd test suite
korDen
parents: 126
diff changeset
831 e = new PtrExp(Loc(0), e);
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
832 }
129
010eb8f0e18d further work on dmd test suite
korDen
parents: 126
diff changeset
833 AssignExp ea = new AssignExp(Loc(0), e, new IdentifierExp(Loc(0), Id.p));
010eb8f0e18d further work on dmd test suite
korDen
parents: 126
diff changeset
834 ea.op = TOK.TOKblit;
010eb8f0e18d further work on dmd test suite
korDen
parents: 126
diff changeset
835 Statement s = new ExpStatement(Loc(0), ea);
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
836
129
010eb8f0e18d further work on dmd test suite
korDen
parents: 126
diff changeset
837 // Build postBlit();
010eb8f0e18d further work on dmd test suite
korDen
parents: 126
diff changeset
838 e = new VarExp(Loc(0), postblit, 0);
010eb8f0e18d further work on dmd test suite
korDen
parents: 126
diff changeset
839 e = new CallExp(Loc(0), e);
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
840
129
010eb8f0e18d further work on dmd test suite
korDen
parents: 126
diff changeset
841 s = new CompoundStatement(Loc(0), s, new ExpStatement(Loc(0), e));
010eb8f0e18d further work on dmd test suite
korDen
parents: 126
diff changeset
842 fcp.fbody = s;
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
843
129
010eb8f0e18d further work on dmd test suite
korDen
parents: 126
diff changeset
844 members.push(fcp);
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
845
129
010eb8f0e18d further work on dmd test suite
korDen
parents: 126
diff changeset
846 sc = sc.push();
010eb8f0e18d further work on dmd test suite
korDen
parents: 126
diff changeset
847 sc.stc = STC.STCundefined;
010eb8f0e18d further work on dmd test suite
korDen
parents: 126
diff changeset
848 sc.linkage = LINK.LINKd;
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
849
129
010eb8f0e18d further work on dmd test suite
korDen
parents: 126
diff changeset
850 fcp.semantic(sc);
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
851
129
010eb8f0e18d further work on dmd test suite
korDen
parents: 126
diff changeset
852 sc.pop();
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
853 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
854
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
855 return fcp;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
856 }
130
60bb0fe4563e dmdfe 2.037 first main iteration
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 129
diff changeset
857 }
72
2e2a5c3f943a reduced warnings by adding override to the methods
Trass3r
parents: 49
diff changeset
858 override void toDocBuffer(OutBuffer buf)
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
859 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
860 assert(false);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
861 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
862
72
2e2a5c3f943a reduced warnings by adding override to the methods
Trass3r
parents: 49
diff changeset
863 override PROT getAccess(Dsymbol smember) // determine access to smember
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
864 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
865 assert(false);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
866 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
867
72
2e2a5c3f943a reduced warnings by adding override to the methods
Trass3r
parents: 49
diff changeset
868 override void toObjFile(int multiobj) // compile to .obj file
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
869 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
870 //printf("StructDeclaration.toObjFile('%s')\n", toChars());
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
871
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
872 if (multiobj)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
873 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
874 obj_append(this);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
875 return;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
876 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
877
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
878 // Anonymous structs/unions only exist as part of others,
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
879 // do not output forward referenced structs's
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
880 if (!isAnonymous() && members)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
881 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
882 if (global.params.symdebug) {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
883 toDebug();
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
884 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
885
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
886 type.getTypeInfo(null); // generate TypeInfo
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
887
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
888 if (true)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
889 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
890 // Generate static initializer
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
891 toInitializer();
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
892
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
893 static if (false) {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
894 sinit.Sclass = SC.SCcomdat;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
895 } else {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
896 if (inTemplateInstance())
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
897 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
898 sinit.Sclass = SC.SCcomdat;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
899 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
900 else
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
901 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
902 sinit.Sclass = SC.SCglobal;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
903 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
904 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
905 sinit.Sfl = FL.FLdata;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
906
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
907 toDt(&sinit.Sdt);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
908
95
ae5b11064a9a beginning of 2.036 branch
Trass3r
parents: 89
diff changeset
909 version (OMFOBJ)
ae5b11064a9a beginning of 2.036 branch
Trass3r
parents: 89
diff changeset
910 {
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
911 /* For OMF, common blocks aren't pulled in from the library.
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
912 */
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
913 /* ELF comdef's generate multiple
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
914 * definition errors for them from the gnu linker.
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
915 * Need to figure out how to generate proper comdef's for ELF.
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
916 */
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
917 // See if we can convert a comdat to a comdef,
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
918 // which saves on exe file space.
95
ae5b11064a9a beginning of 2.036 branch
Trass3r
parents: 89
diff changeset
919 if (0 && // causes multiple def problems with COMMON in one file and COMDAT in library
ae5b11064a9a beginning of 2.036 branch
Trass3r
parents: 89
diff changeset
920 sinit.Sclass == SCcomdat &&
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
921 sinit.Sdt &&
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
922 sinit.Sdt.dt == DT.DT_azeros &&
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
923 sinit.Sdt.DTnext == null &&
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
924 !global.params.multiobj)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
925 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
926 sinit.Sclass = SC.SCglobal;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
927 sinit.Sdt.dt = DT.DT_common;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
928 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
929 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
930
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
931 version (ELFOBJ) {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
932 sinit.Sseg = Segment.CDATA;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
933 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
934 version (MACHOBJ) {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
935 sinit.Sseg = Segment.DATA;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
936 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
937 outdata(sinit);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
938 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
939
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
940 // Put out the members
77
ad4792a1cfd6 more D-ification container accessing
Eldar Insafutdinov <e.insafutdinov@gmail.com>
parents: 74
diff changeset
941 foreach(Dsymbol member; members)
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
942 member.toObjFile(0);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
943 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
944 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
945
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
946 void toDt(dt_t** pdt)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
947 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
948 uint offset;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
949 dt_t* dt;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
950
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
951 //printf("StructDeclaration.toDt(), this='%s'\n", toChars());
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
952 offset = 0;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
953
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
954 // Note equivalence of this loop to class's
79
43073c7c7769 updated to 2.035
Trass3r
parents: 77
diff changeset
955 for (uint i = 0; i < fields.dim; i++)
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
956 {
79
43073c7c7769 updated to 2.035
Trass3r
parents: 77
diff changeset
957 VarDeclaration v = cast(VarDeclaration)fields[i];
0
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
958 //printf("\tfield '%s' voffset %d, offset = %d\n", v.toChars(), v.offset, offset);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
959 dt = null;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
960 int sz;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
961
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
962 if (v.storage_class & STC.STCref)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
963 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
964 sz = PTRSIZE;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
965 if (v.offset >= offset)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
966 dtnzeros(&dt, sz);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
967 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
968 else
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
969 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
970 sz = cast(uint)v.type.size();
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
971 Initializer init = v.init;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
972 if (init)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
973 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
974 //printf("\t\thas initializer %s\n", init.toChars());
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
975 ExpInitializer ei = init.isExpInitializer();
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
976 Type tb = v.type.toBasetype();
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
977 if (ei && tb.ty == TY.Tsarray)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
978 (cast(TypeSArray)tb).toDtElem(&dt, ei.exp);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
979 else
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
980 dt = init.toDt();
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
981 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
982 else if (v.offset >= offset)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
983 v.type.toDt(&dt);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
984 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
985 if (dt)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
986 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
987 if (v.offset < offset)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
988 error("overlapping initialization for struct %s.%s", toChars(), v.toChars());
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
989 else
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
990 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
991 if (offset < v.offset)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
992 dtnzeros(pdt, v.offset - offset);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
993 dtcat(pdt, dt);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
994 offset = v.offset + sz;
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
995 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
996 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
997 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
998
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
999 if (offset < structsize)
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
1000 dtnzeros(pdt, structsize - offset);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
1001
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
1002 dt_optimize(*pdt);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
1003 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
1004
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
1005 void toDebug() // to symbolic debug info
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
1006 {
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
1007 assert(false);
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
1008 }
10317f0c89a5 Initial commit
korDen
parents:
diff changeset
1009
72
2e2a5c3f943a reduced warnings by adding override to the methods
Trass3r
parents: 49
diff changeset
1010 override StructDeclaration isStructDeclaration() { return this; }
2e2a5c3f943a reduced warnings by adding override to the methods
Trass3r
parents: 49
diff changeset
1011 }