1 /** 2 * Put initializers and objects created from CTFE into a `dt_t` data structure 3 * so the backend puts them into the data segment. 4 * 5 * Copyright: Copyright (C) 1999-2020 by The D Language Foundation, All Rights Reserved 6 * Authors: $(LINK2 http://www.digitalmars.com, Walter Bright) 7 * License: $(LINK2 http://www.boost.org/LICENSE_1_0.txt, Boost License 1.0) 8 * Source: $(LINK2 https://github.com/dlang/dmd/blob/master/src/dmd/todt.d, _todt.d) 9 * Documentation: https://dlang.org/phobos/dmd_todt.html 10 * Coverage: https://codecov.io/gh/dlang/dmd/src/master/src/dmd/todt.d 11 */ 12 13 module dmd.todt; 14 15 import core.stdc.stdio; 16 import core.stdc.string; 17 18 import dmd.root.array; 19 import dmd.root.rmem; 20 21 import dmd.aggregate; 22 import dmd.arraytypes; 23 import dmd.backend.type; 24 import dmd.complex; 25 import dmd.ctfeexpr; 26 import dmd.declaration; 27 import dmd.dclass; 28 import dmd.denum; 29 import dmd.dstruct; 30 import dmd.dsymbol; 31 import dmd.dtemplate; 32 import dmd.errors; 33 import dmd.expression; 34 import dmd.func; 35 import dmd.globals; 36 import dmd.init; 37 import dmd.mtype; 38 import dmd.target; 39 import dmd.tokens; 40 import dmd.tocsym; 41 import dmd.toobj; 42 import dmd.typesem; 43 import dmd.typinf; 44 import dmd.visitor; 45 46 import dmd.backend.cc; 47 import dmd.backend.dt; 48 49 alias toSymbol = dmd.tocsym.toSymbol; 50 alias toSymbol = dmd.glue.toSymbol; 51 52 /* A dt_t is a simple structure representing data to be added 53 * to the data segment of the output object file. As such, 54 * it is a list of initialized bytes, 0 data, and offsets from 55 * other symbols. 56 * Each D symbol and type can be converted into a dt_t so it can 57 * be written to the data segment. 58 */ 59 60 alias Dts = Array!(dt_t*); 61 62 /* ================================================================ */ 63 64 extern (C++) void Initializer_toDt(Initializer init, ref DtBuilder dtb) 65 { 66 void visitError(ErrorInitializer) 67 { 68 assert(0); 69 } 70 71 void visitVoid(VoidInitializer vi) 72 { 73 /* Void initializers are set to 0, just because we need something 74 * to set them to in the static data segment. 75 */ 76 dtb.nzeros(cast(uint)vi.type.size()); 77 } 78 79 void visitStruct(StructInitializer si) 80 { 81 //printf("StructInitializer.toDt('%s')\n", si.toChars()); 82 assert(0); 83 } 84 85 void visitArray(ArrayInitializer ai) 86 { 87 //printf("ArrayInitializer.toDt('%s')\n", ai.toChars()); 88 Type tb = ai.type.toBasetype(); 89 if (tb.ty == Tvector) 90 tb = (cast(TypeVector)tb).basetype; 91 92 Type tn = tb.nextOf().toBasetype(); 93 94 //printf("\tdim = %d\n", ai.dim); 95 Dts dts; 96 dts.setDim(ai.dim); 97 dts.zero(); 98 99 uint size = cast(uint)tn.size(); 100 101 uint length = 0; 102 foreach (i, idx; ai.index) 103 { 104 if (idx) 105 length = cast(uint)idx.toInteger(); 106 //printf("\tindex[%d] = %p, length = %u, dim = %u\n", i, idx, length, ai.dim); 107 108 assert(length < ai.dim); 109 auto dtb = DtBuilder(0); 110 Initializer_toDt(ai.value[i], dtb); 111 if (dts[length]) 112 error(ai.loc, "duplicate initializations for index `%d`", length); 113 dts[length] = dtb.finish(); 114 length++; 115 } 116 117 Expression edefault = tb.nextOf().defaultInit(Loc.initial); 118 119 const n = tn.numberOfElems(ai.loc); 120 121 dt_t* dtdefault = null; 122 123 auto dtbarray = DtBuilder(0); 124 foreach (dt; dts) 125 { 126 if (dt) 127 dtbarray.cat(dt); 128 else 129 { 130 if (!dtdefault) 131 { 132 auto dtb = DtBuilder(0); 133 Expression_toDt(edefault, dtb); 134 dtdefault = dtb.finish(); 135 } 136 dtbarray.repeat(dtdefault, n); 137 } 138 } 139 switch (tb.ty) 140 { 141 case Tsarray: 142 { 143 TypeSArray ta = cast(TypeSArray)tb; 144 size_t tadim = cast(size_t)ta.dim.toInteger(); 145 if (ai.dim < tadim) 146 { 147 if (edefault.isBool(false)) 148 { 149 // pad out end of array 150 dtbarray.nzeros(cast(uint)(size * (tadim - ai.dim))); 151 } 152 else 153 { 154 if (!dtdefault) 155 { 156 auto dtb = DtBuilder(0); 157 Expression_toDt(edefault, dtb); 158 dtdefault = dtb.finish(); 159 } 160 161 const m = n * (tadim - ai.dim); 162 assert(m <= uint.max); 163 dtbarray.repeat(dtdefault, cast(uint)m); 164 } 165 } 166 else if (ai.dim > tadim) 167 { 168 error(ai.loc, "too many initializers, %d, for array[%d]", ai.dim, tadim); 169 } 170 dtb.cat(dtbarray); 171 break; 172 } 173 174 case Tpointer: 175 case Tarray: 176 { 177 if (tb.ty == Tarray) 178 dtb.size(ai.dim); 179 Symbol* s = dtb.dtoff(dtbarray.finish(), 0); 180 if (tn.isMutable()) 181 foreach (i; 0 .. ai.dim) 182 write_pointers(tn, s, size * cast(int)i); 183 break; 184 } 185 186 default: 187 assert(0); 188 } 189 dt_free(dtdefault); 190 } 191 192 void visitExp(ExpInitializer ei) 193 { 194 //printf("ExpInitializer.toDt() %s\n", ei.exp.toChars()); 195 ei.exp = ei.exp.optimize(WANTvalue); 196 Expression_toDt(ei.exp, dtb); 197 } 198 199 final switch (init.kind) 200 { 201 case InitKind.void_: return visitVoid (cast( VoidInitializer)init); 202 case InitKind.error: return visitError (cast( ErrorInitializer)init); 203 case InitKind.struct_: return visitStruct(cast(StructInitializer)init); 204 case InitKind.array: return visitArray (cast( ArrayInitializer)init); 205 case InitKind.exp: return visitExp (cast( ExpInitializer)init); 206 } 207 } 208 209 /* ================================================================ */ 210 211 extern (C++) void Expression_toDt(Expression e, ref DtBuilder dtb) 212 { 213 void nonConstExpError(Expression e) 214 { 215 version (none) 216 { 217 printf("Expression.toDt() %d\n", e.op); 218 } 219 e.error("non-constant expression `%s`", e.toChars()); 220 dtb.nzeros(1); 221 } 222 223 void visitCast(CastExp e) 224 { 225 version (none) 226 { 227 printf("CastExp.toDt() %d from %s to %s\n", e.op, e.e1.type.toChars(), e.type.toChars()); 228 } 229 if (e.e1.type.ty == Tclass) 230 { 231 if (auto toc = e.type.isTypeClass()) 232 { 233 if (auto toi = toc.sym.isInterfaceDeclaration()) // casting from class to interface 234 { 235 auto cre1 = e.e1.isClassReferenceExp(); 236 ClassDeclaration from = cre1.originalClass(); 237 int off = 0; 238 const isbase = toi.isBaseOf(from, &off); 239 assert(isbase); 240 ClassReferenceExp_toDt(cre1, dtb, off); 241 } 242 else //casting from class to class 243 { 244 Expression_toDt(e.e1, dtb); 245 } 246 return; 247 } 248 } 249 nonConstExpError(e); 250 } 251 252 void visitAddr(AddrExp e) 253 { 254 version (none) 255 { 256 printf("AddrExp.toDt() %d\n", e.op); 257 } 258 if (auto sl = e.e1.isStructLiteralExp()) 259 { 260 Symbol* s = toSymbol(sl); 261 dtb.xoff(s, 0); 262 if (sl.type.isMutable()) 263 write_pointers(sl.type, s, 0); 264 return; 265 } 266 nonConstExpError(e); 267 } 268 269 void visitInteger(IntegerExp e) 270 { 271 //printf("IntegerExp.toDt() %d\n", e.op); 272 const sz = cast(uint)e.type.size(); 273 if (auto value = e.getInteger()) 274 dtb.nbytes(sz, cast(char*)&value); 275 else 276 dtb.nzeros(sz); 277 } 278 279 void visitReal(RealExp e) 280 { 281 //printf("RealExp.toDt(%Lg)\n", e.value); 282 switch (e.type.toBasetype().ty) 283 { 284 case Tfloat32: 285 case Timaginary32: 286 { 287 auto fvalue = cast(float)e.value; 288 dtb.nbytes(4, cast(char*)&fvalue); 289 break; 290 } 291 292 case Tfloat64: 293 case Timaginary64: 294 { 295 auto dvalue = cast(double)e.value; 296 dtb.nbytes(8, cast(char*)&dvalue); 297 break; 298 } 299 300 case Tfloat80: 301 case Timaginary80: 302 { 303 auto evalue = e.value; 304 dtb.nbytes(target.realsize - target.realpad, cast(char*)&evalue); 305 dtb.nzeros(target.realpad); 306 break; 307 } 308 309 default: 310 printf("%s, e.type=%s\n", e.toChars(), e.type.toChars()); 311 assert(0); 312 } 313 } 314 315 void visitComplex(ComplexExp e) 316 { 317 //printf("ComplexExp.toDt() '%s'\n", e.toChars()); 318 switch (e.type.toBasetype().ty) 319 { 320 case Tcomplex32: 321 { 322 auto fvalue = cast(float)creall(e.value); 323 dtb.nbytes(4, cast(char*)&fvalue); 324 fvalue = cast(float)cimagl(e.value); 325 dtb.nbytes(4, cast(char*)&fvalue); 326 break; 327 } 328 329 case Tcomplex64: 330 { 331 auto dvalue = cast(double)creall(e.value); 332 dtb.nbytes(8, cast(char*)&dvalue); 333 dvalue = cast(double)cimagl(e.value); 334 dtb.nbytes(8, cast(char*)&dvalue); 335 break; 336 } 337 338 case Tcomplex80: 339 { 340 auto evalue = creall(e.value); 341 dtb.nbytes(target.realsize - target.realpad, cast(char*)&evalue); 342 dtb.nzeros(target.realpad); 343 evalue = cimagl(e.value); 344 dtb.nbytes(target.realsize - target.realpad, cast(char*)&evalue); 345 dtb.nzeros(target.realpad); 346 break; 347 } 348 349 default: 350 assert(0); 351 } 352 } 353 354 void visitNull(NullExp e) 355 { 356 assert(e.type); 357 dtb.nzeros(cast(uint)e.type.size()); 358 } 359 360 void visitString(StringExp e) 361 { 362 //printf("StringExp.toDt() '%s', type = %s\n", e.toChars(), e.type.toChars()); 363 Type t = e.type.toBasetype(); 364 365 // BUG: should implement some form of static string pooling 366 const n = cast(int)e.numberOfCodeUnits(); 367 const(char)* p; 368 char* q; 369 if (e.sz == 1) 370 p = e.peekString().ptr; 371 else 372 { 373 q = cast(char*)mem.xmalloc(n * e.sz); 374 e.writeTo(q, false); 375 p = q; 376 } 377 378 switch (t.ty) 379 { 380 case Tarray: 381 dtb.size(n); 382 goto case Tpointer; 383 384 case Tpointer: 385 if (e.sz == 1) 386 { 387 import dmd.e2ir : toStringSymbol; 388 import dmd.glue : totym; 389 Symbol* s = toStringSymbol(p, n, e.sz); 390 dtb.xoff(s, 0); 391 } 392 else 393 dtb.abytes(0, n * e.sz, p, cast(uint)e.sz); 394 break; 395 396 case Tsarray: 397 { 398 auto tsa = t.isTypeSArray(); 399 400 dtb.nbytes(n * e.sz, p); 401 if (tsa.dim) 402 { 403 dinteger_t dim = tsa.dim.toInteger(); 404 if (n < dim) 405 { 406 // Pad remainder with 0 407 dtb.nzeros(cast(uint)((dim - n) * tsa.next.size())); 408 } 409 } 410 break; 411 } 412 413 default: 414 printf("StringExp.toDt(type = %s)\n", e.type.toChars()); 415 assert(0); 416 } 417 mem.xfree(q); 418 } 419 420 void visitArrayLiteral(ArrayLiteralExp e) 421 { 422 //printf("ArrayLiteralExp.toDt() '%s', type = %s\n", e.toChars(), e.type.toChars()); 423 424 auto dtbarray = DtBuilder(0); 425 foreach (i; 0 .. e.elements.dim) 426 { 427 Expression_toDt(e[i], dtbarray); 428 } 429 430 Type t = e.type.toBasetype(); 431 switch (t.ty) 432 { 433 case Tsarray: 434 dtb.cat(dtbarray); 435 break; 436 437 case Tarray: 438 dtb.size(e.elements.dim); 439 goto case Tpointer; 440 441 case Tpointer: 442 { 443 if (auto d = dtbarray.finish()) 444 dtb.dtoff(d, 0); 445 else 446 dtb.size(0); 447 448 break; 449 } 450 451 default: 452 assert(0); 453 } 454 } 455 456 void visitStructLiteral(StructLiteralExp sle) 457 { 458 //printf("StructLiteralExp.toDt() %s, ctfe = %d\n", sle.toChars(), sle.ownedByCtfe); 459 assert(sle.sd.nonHiddenFields() <= sle.elements.dim); 460 membersToDt(sle.sd, dtb, sle.elements, 0, null); 461 } 462 463 void visitSymOff(SymOffExp e) 464 { 465 //printf("SymOffExp.toDt('%s')\n", e.var.toChars()); 466 assert(e.var); 467 if (!(e.var.isDataseg() || e.var.isCodeseg()) || 468 e.var.needThis() || 469 e.var.isThreadlocal()) 470 { 471 return nonConstExpError(e); 472 } 473 dtb.xoff(toSymbol(e.var), cast(uint)e.offset); 474 } 475 476 void visitVar(VarExp e) 477 { 478 //printf("VarExp.toDt() %d\n", e.op); 479 480 if (auto v = e.var.isVarDeclaration()) 481 { 482 if ((v.isConst() || v.isImmutable()) && 483 e.type.toBasetype().ty != Tsarray && v._init) 484 { 485 e.error("recursive reference `%s`", e.toChars()); 486 return; 487 } 488 v.inuse++; 489 Initializer_toDt(v._init, dtb); 490 v.inuse--; 491 return; 492 } 493 494 if (auto sd = e.var.isSymbolDeclaration()) 495 if (sd.dsym) 496 { 497 StructDeclaration_toDt(sd.dsym, dtb); 498 return; 499 } 500 501 return nonConstExpError(e); 502 } 503 504 void visitFunc(FuncExp e) 505 { 506 //printf("FuncExp.toDt() %d\n", e.op); 507 if (e.fd.tok == TOK.reserved && e.type.ty == Tpointer) 508 { 509 // change to non-nested 510 e.fd.tok = TOK.function_; 511 e.fd.vthis = null; 512 } 513 Symbol *s = toSymbol(e.fd); 514 toObjFile(e.fd, false); 515 if (e.fd.tok == TOK.delegate_) 516 dtb.size(0); 517 dtb.xoff(s, 0); 518 } 519 520 void visitVector(VectorExp e) 521 { 522 //printf("VectorExp.toDt() %s\n", e.toChars()); 523 foreach (i; 0 .. e.dim) 524 { 525 Expression elem; 526 if (auto ale = e.e1.isArrayLiteralExp()) 527 elem = ale[i]; 528 else 529 elem = e.e1; 530 Expression_toDt(elem, dtb); 531 } 532 } 533 534 void visitClassReference(ClassReferenceExp e) 535 { 536 InterfaceDeclaration to = (cast(TypeClass)e.type).sym.isInterfaceDeclaration(); 537 538 if (to) //Static typeof this literal is an interface. We must add offset to symbol 539 { 540 ClassDeclaration from = e.originalClass(); 541 int off = 0; 542 const isbase = to.isBaseOf(from, &off); 543 assert(isbase); 544 ClassReferenceExp_toDt(e, dtb, off); 545 } 546 else 547 ClassReferenceExp_toDt(e, dtb, 0); 548 } 549 550 void visitTypeid(TypeidExp e) 551 { 552 if (Type t = isType(e.obj)) 553 { 554 genTypeInfo(e.loc, t, null); 555 Symbol *s = toSymbol(t.vtinfo); 556 dtb.xoff(s, 0); 557 return; 558 } 559 assert(0); 560 } 561 562 switch (e.op) 563 { 564 default: return nonConstExpError(e); 565 case TOK.cast_: return visitCast (e.isCastExp()); 566 case TOK.address: return visitAddr (e.isAddrExp()); 567 case TOK.int64: return visitInteger (e.isIntegerExp()); 568 case TOK.float64: return visitReal (e.isRealExp()); 569 case TOK.complex80: return visitComplex (e.isComplexExp()); 570 case TOK.null_: return visitNull (e.isNullExp()); 571 case TOK.string_: return visitString (e.isStringExp()); 572 case TOK.arrayLiteral: return visitArrayLiteral (e.isArrayLiteralExp()); 573 case TOK.structLiteral: return visitStructLiteral (e.isStructLiteralExp()); 574 case TOK.symbolOffset: return visitSymOff (e.isSymOffExp()); 575 case TOK.variable: return visitVar (e.isVarExp()); 576 case TOK.function_: return visitFunc (e.isFuncExp()); 577 case TOK.vector: return visitVector (e.isVectorExp()); 578 case TOK.classReference: return visitClassReference(e.isClassReferenceExp()); 579 case TOK.typeid_: return visitTypeid (e.isTypeidExp()); 580 } 581 } 582 583 /* ================================================================= */ 584 585 // Generate the data for the static initializer. 586 587 extern (C++) void ClassDeclaration_toDt(ClassDeclaration cd, ref DtBuilder dtb) 588 { 589 //printf("ClassDeclaration.toDt(this = '%s')\n", cd.toChars()); 590 591 membersToDt(cd, dtb, null, 0, cd); 592 593 //printf("-ClassDeclaration.toDt(this = '%s')\n", cd.toChars()); 594 } 595 596 extern (C++) void StructDeclaration_toDt(StructDeclaration sd, ref DtBuilder dtb) 597 { 598 //printf("+StructDeclaration.toDt(), this='%s'\n", sd.toChars()); 599 membersToDt(sd, dtb, null, 0, null); 600 601 //printf("-StructDeclaration.toDt(), this='%s'\n", sd.toChars()); 602 } 603 604 /****************************** 605 * Generate data for instance of __cpp_type_info_ptr that refers 606 * to the C++ RTTI symbol for cd. 607 * Params: 608 * cd = C++ class 609 * dtb = data table builder 610 */ 611 extern (C++) void cpp_type_info_ptr_toDt(ClassDeclaration cd, ref DtBuilder dtb) 612 { 613 //printf("cpp_type_info_ptr_toDt(this = '%s')\n", cd.toChars()); 614 assert(cd.isCPPclass()); 615 616 // Put in first two members, the vtbl[] and the monitor 617 dtb.xoff(toVtblSymbol(ClassDeclaration.cpp_type_info_ptr), 0); 618 if (ClassDeclaration.cpp_type_info_ptr.hasMonitor()) 619 dtb.size(0); // monitor 620 621 // Create symbol for C++ type info 622 Symbol *s = toSymbolCppTypeInfo(cd); 623 624 // Put in address of cd's C++ type info 625 dtb.xoff(s, 0); 626 627 //printf("-cpp_type_info_ptr_toDt(this = '%s')\n", cd.toChars()); 628 } 629 630 /**************************************************** 631 * Put out initializers of ad.fields[]. 632 * Although this is consistent with the elements[] version, we 633 * have to use this optimized version to reduce memory footprint. 634 * Params: 635 * ad = aggregate with members 636 * pdt = tail of initializer list to start appending initialized data to 637 * elements = values to use as initializers, null means use default initializers 638 * firstFieldIndex = starting place is elements[firstFieldIndex] 639 * concreteType = structs: null, classes: most derived class 640 * ppb = pointer that moves through BaseClass[] from most derived class 641 * Returns: 642 * updated tail of dt_t list 643 */ 644 645 private void membersToDt(AggregateDeclaration ad, ref DtBuilder dtb, 646 Expressions* elements, size_t firstFieldIndex, 647 ClassDeclaration concreteType, 648 BaseClass*** ppb = null) 649 { 650 //printf("membersToDt(ad = '%s', concrete = '%s', ppb = %p)\n", ad.toChars(), concreteType ? concreteType.toChars() : "null", ppb); 651 ClassDeclaration cd = ad.isClassDeclaration(); 652 version (none) 653 { 654 printf(" interfaces.length = %d\n", cast(int)cd.interfaces.length); 655 foreach (i, b; cd.vtblInterfaces[]) 656 { 657 printf(" vbtblInterfaces[%d] b = %p, b.sym = %s\n", cast(int)i, b, b.sym.toChars()); 658 } 659 } 660 661 /* Order: 662 * { base class } or { __vptr, __monitor } 663 * interfaces 664 * fields 665 */ 666 667 uint offset; 668 if (cd) 669 { 670 if (ClassDeclaration cdb = cd.baseClass) 671 { 672 size_t index = 0; 673 for (ClassDeclaration c = cdb.baseClass; c; c = c.baseClass) 674 index += c.fields.dim; 675 membersToDt(cdb, dtb, elements, index, concreteType); 676 offset = cdb.structsize; 677 } 678 else if (InterfaceDeclaration id = cd.isInterfaceDeclaration()) 679 { 680 offset = (**ppb).offset; 681 if (id.vtblInterfaces.dim == 0) 682 { 683 BaseClass* b = **ppb; 684 //printf(" Interface %s, b = %p\n", id.toChars(), b); 685 ++(*ppb); 686 for (ClassDeclaration cd2 = concreteType; 1; cd2 = cd2.baseClass) 687 { 688 assert(cd2); 689 uint csymoffset = baseVtblOffset(cd2, b); 690 //printf(" cd2 %s csymoffset = x%x\n", cd2 ? cd2.toChars() : "null", csymoffset); 691 if (csymoffset != ~0) 692 { 693 dtb.xoff(toSymbol(cd2), csymoffset); 694 offset += target.ptrsize; 695 break; 696 } 697 } 698 } 699 } 700 else 701 { 702 dtb.xoff(toVtblSymbol(concreteType), 0); // __vptr 703 offset = target.ptrsize; 704 if (cd.hasMonitor()) 705 { 706 dtb.size(0); // __monitor 707 offset += target.ptrsize; 708 } 709 } 710 711 // Interface vptr initializations 712 toSymbol(cd); // define csym 713 714 BaseClass** pb; 715 if (!ppb) 716 { 717 pb = (*cd.vtblInterfaces)[].ptr; 718 ppb = &pb; 719 } 720 721 foreach (si; cd.interfaces[]) 722 { 723 BaseClass* b = **ppb; 724 if (offset < b.offset) 725 dtb.nzeros(b.offset - offset); 726 membersToDt(si.sym, dtb, elements, firstFieldIndex, concreteType, ppb); 727 //printf("b.offset = %d, b.sym.structsize = %d\n", (int)b.offset, (int)b.sym.structsize); 728 offset = b.offset + b.sym.structsize; 729 } 730 } 731 else 732 offset = 0; 733 734 assert(!elements || 735 firstFieldIndex <= elements.dim && 736 firstFieldIndex + ad.fields.dim <= elements.dim); 737 738 foreach (i, field; ad.fields) 739 { 740 if (elements && !(*elements)[firstFieldIndex + i]) 741 continue; 742 743 if (!elements || !(*elements)[firstFieldIndex + i]) 744 { 745 if (field._init && field._init.isVoidInitializer()) 746 continue; 747 } 748 749 VarDeclaration vd; 750 size_t k; 751 foreach (j; i .. ad.fields.length) 752 { 753 VarDeclaration v2 = ad.fields[j]; 754 if (v2.offset < offset) 755 continue; 756 757 if (elements && !(*elements)[firstFieldIndex + j]) 758 continue; 759 760 if (!elements || !(*elements)[firstFieldIndex + j]) 761 { 762 if (v2._init && v2._init.isVoidInitializer()) 763 continue; 764 } 765 766 // find the nearest field 767 if (!vd || v2.offset < vd.offset) 768 { 769 vd = v2; 770 k = j; 771 assert(vd == v2 || !vd.isOverlappedWith(v2)); 772 } 773 } 774 if (!vd) 775 continue; 776 777 assert(offset <= vd.offset); 778 if (offset < vd.offset) 779 dtb.nzeros(vd.offset - offset); 780 781 auto dtbx = DtBuilder(0); 782 if (elements) 783 { 784 Expression e = (*elements)[firstFieldIndex + k]; 785 if (auto tsa = vd.type.toBasetype().isTypeSArray()) 786 toDtElem(tsa, dtbx, e); 787 else 788 Expression_toDt(e, dtbx); // convert e to an initializer dt 789 } 790 else 791 { 792 if (Initializer init = vd._init) 793 { 794 //printf("\t\t%s has initializer %s\n", vd.toChars(), init.toChars()); 795 if (init.isVoidInitializer()) 796 continue; 797 798 assert(vd.semanticRun >= PASS.semantic2done); 799 800 auto ei = init.isExpInitializer(); 801 auto tsa = vd.type.toBasetype().isTypeSArray(); 802 if (ei && tsa) 803 toDtElem(tsa, dtbx, ei.exp); 804 else 805 Initializer_toDt(init, dtbx); 806 } 807 else if (offset <= vd.offset) 808 { 809 //printf("\t\tdefault initializer\n"); 810 Type_toDt(vd.type, dtbx); 811 } 812 if (dtbx.isZeroLength()) 813 continue; 814 } 815 816 dtb.cat(dtbx); 817 offset = cast(uint)(vd.offset + vd.type.size()); 818 } 819 820 if (offset < ad.structsize) 821 dtb.nzeros(ad.structsize - offset); 822 } 823 824 825 /* ================================================================= */ 826 827 extern (C++) void Type_toDt(Type t, ref DtBuilder dtb) 828 { 829 switch (t.ty) 830 { 831 case Tvector: 832 toDtElem(t.isTypeVector().basetype.isTypeSArray(), dtb, null); 833 break; 834 835 case Tsarray: 836 toDtElem(t.isTypeSArray(), dtb, null); 837 break; 838 839 case Tstruct: 840 StructDeclaration_toDt(t.isTypeStruct().sym, dtb); 841 break; 842 843 default: 844 Expression_toDt(t.defaultInit(Loc.initial), dtb); 845 break; 846 } 847 } 848 849 private void toDtElem(TypeSArray tsa, ref DtBuilder dtb, Expression e) 850 { 851 //printf("TypeSArray.toDtElem() tsa = %s\n", tsa.toChars()); 852 if (tsa.size(Loc.initial) == 0) 853 { 854 dtb.nzeros(0); 855 } 856 else 857 { 858 size_t len = cast(size_t)tsa.dim.toInteger(); 859 assert(len); 860 Type tnext = tsa.next; 861 Type tbn = tnext.toBasetype(); 862 Type ten = e ? e.type : null; 863 if (ten && (ten.ty == Tsarray || ten.ty == Tarray)) 864 ten = ten.nextOf(); 865 while (tbn.ty == Tsarray && (!e || !tbn.equivalent(ten))) 866 { 867 len *= tbn.isTypeSArray().dim.toInteger(); 868 tnext = tbn.nextOf(); 869 tbn = tnext.toBasetype(); 870 } 871 if (!e) // if not already supplied 872 e = tsa.defaultInit(Loc.initial); // use default initializer 873 874 if (!e.type.implicitConvTo(tnext)) // https://issues.dlang.org/show_bug.cgi?id=14996 875 { 876 // https://issues.dlang.org/show_bug.cgi?id=1914 877 // https://issues.dlang.org/show_bug.cgi?id=3198 878 if (auto se = e.isStringExp()) 879 len /= se.numberOfCodeUnits(); 880 else if (auto ae = e.isArrayLiteralExp()) 881 len /= ae.elements.dim; 882 } 883 884 auto dtb2 = DtBuilder(0); 885 Expression_toDt(e, dtb2); 886 dt_t* dt2 = dtb2.finish(); 887 assert(len <= uint.max); 888 dtb.repeat(dt2, cast(uint)len); 889 } 890 } 891 892 /*****************************************************/ 893 /* CTFE stuff */ 894 /*****************************************************/ 895 896 private void ClassReferenceExp_toDt(ClassReferenceExp e, ref DtBuilder dtb, int off) 897 { 898 //printf("ClassReferenceExp.toDt() %d\n", e.op); 899 Symbol* s = toSymbol(e); 900 dtb.xoff(s, off); 901 if (e.type.isMutable()) 902 write_instance_pointers(e.type, s, 0); 903 } 904 905 extern (C++) void ClassReferenceExp_toInstanceDt(ClassReferenceExp ce, ref DtBuilder dtb) 906 { 907 //printf("ClassReferenceExp.toInstanceDt() %d\n", ce.op); 908 ClassDeclaration cd = ce.originalClass(); 909 910 // Put in the rest 911 size_t firstFieldIndex = 0; 912 for (ClassDeclaration c = cd.baseClass; c; c = c.baseClass) 913 firstFieldIndex += c.fields.dim; 914 membersToDt(cd, dtb, ce.value.elements, firstFieldIndex, cd); 915 } 916 917 /**************************************************** 918 */ 919 private extern (C++) class TypeInfoDtVisitor : Visitor 920 { 921 DtBuilder* dtb; 922 923 /* 924 * Used in TypeInfo*.toDt to verify the runtime TypeInfo sizes 925 */ 926 static void verifyStructSize(ClassDeclaration typeclass, size_t expected) 927 { 928 if (typeclass.structsize != expected) 929 { 930 debug 931 { 932 printf("expected = x%x, %s.structsize = x%x\n", cast(uint)expected, 933 typeclass.toChars(), cast(uint)typeclass.structsize); 934 } 935 error(typeclass.loc, "`%s`: mismatch between compiler (%d bytes) and object.d or object.di (%d bytes) found. Check installation and import paths with -v compiler switch.", 936 typeclass.toChars(), cast(uint)expected, cast(uint)typeclass.structsize); 937 fatal(); 938 } 939 } 940 941 this(ref DtBuilder dtb) 942 { 943 this.dtb = &dtb; 944 } 945 946 alias visit = Visitor.visit; 947 948 override void visit(TypeInfoDeclaration d) 949 { 950 //printf("TypeInfoDeclaration.toDt() %s\n", toChars()); 951 verifyStructSize(Type.dtypeinfo, 2 * target.ptrsize); 952 953 dtb.xoff(toVtblSymbol(Type.dtypeinfo), 0); // vtbl for TypeInfo 954 if (Type.dtypeinfo.hasMonitor()) 955 dtb.size(0); // monitor 956 } 957 958 override void visit(TypeInfoConstDeclaration d) 959 { 960 //printf("TypeInfoConstDeclaration.toDt() %s\n", toChars()); 961 verifyStructSize(Type.typeinfoconst, 3 * target.ptrsize); 962 963 dtb.xoff(toVtblSymbol(Type.typeinfoconst), 0); // vtbl for TypeInfo_Const 964 if (Type.typeinfoconst.hasMonitor()) 965 dtb.size(0); // monitor 966 Type tm = d.tinfo.mutableOf(); 967 tm = tm.merge(); 968 genTypeInfo(d.loc, tm, null); 969 dtb.xoff(toSymbol(tm.vtinfo), 0); 970 } 971 972 override void visit(TypeInfoInvariantDeclaration d) 973 { 974 //printf("TypeInfoInvariantDeclaration.toDt() %s\n", toChars()); 975 verifyStructSize(Type.typeinfoinvariant, 3 * target.ptrsize); 976 977 dtb.xoff(toVtblSymbol(Type.typeinfoinvariant), 0); // vtbl for TypeInfo_Invariant 978 if (Type.typeinfoinvariant.hasMonitor()) 979 dtb.size(0); // monitor 980 Type tm = d.tinfo.mutableOf(); 981 tm = tm.merge(); 982 genTypeInfo(d.loc, tm, null); 983 dtb.xoff(toSymbol(tm.vtinfo), 0); 984 } 985 986 override void visit(TypeInfoSharedDeclaration d) 987 { 988 //printf("TypeInfoSharedDeclaration.toDt() %s\n", toChars()); 989 verifyStructSize(Type.typeinfoshared, 3 * target.ptrsize); 990 991 dtb.xoff(toVtblSymbol(Type.typeinfoshared), 0); // vtbl for TypeInfo_Shared 992 if (Type.typeinfoshared.hasMonitor()) 993 dtb.size(0); // monitor 994 Type tm = d.tinfo.unSharedOf(); 995 tm = tm.merge(); 996 genTypeInfo(d.loc, tm, null); 997 dtb.xoff(toSymbol(tm.vtinfo), 0); 998 } 999 1000 override void visit(TypeInfoWildDeclaration d) 1001 { 1002 //printf("TypeInfoWildDeclaration.toDt() %s\n", toChars()); 1003 verifyStructSize(Type.typeinfowild, 3 * target.ptrsize); 1004 1005 dtb.xoff(toVtblSymbol(Type.typeinfowild), 0); // vtbl for TypeInfo_Wild 1006 if (Type.typeinfowild.hasMonitor()) 1007 dtb.size(0); // monitor 1008 Type tm = d.tinfo.mutableOf(); 1009 tm = tm.merge(); 1010 genTypeInfo(d.loc, tm, null); 1011 dtb.xoff(toSymbol(tm.vtinfo), 0); 1012 } 1013 1014 override void visit(TypeInfoEnumDeclaration d) 1015 { 1016 //printf("TypeInfoEnumDeclaration.toDt()\n"); 1017 verifyStructSize(Type.typeinfoenum, 7 * target.ptrsize); 1018 1019 dtb.xoff(toVtblSymbol(Type.typeinfoenum), 0); // vtbl for TypeInfo_Enum 1020 if (Type.typeinfoenum.hasMonitor()) 1021 dtb.size(0); // monitor 1022 1023 assert(d.tinfo.ty == Tenum); 1024 1025 TypeEnum tc = cast(TypeEnum)d.tinfo; 1026 EnumDeclaration sd = tc.sym; 1027 1028 /* Put out: 1029 * TypeInfo base; 1030 * string name; 1031 * void[] m_init; 1032 */ 1033 1034 // TypeInfo for enum members 1035 if (sd.memtype) 1036 { 1037 genTypeInfo(d.loc, sd.memtype, null); 1038 dtb.xoff(toSymbol(sd.memtype.vtinfo), 0); 1039 } 1040 else 1041 dtb.size(0); 1042 1043 // string name; 1044 const(char)* name = sd.toPrettyChars(); 1045 size_t namelen = strlen(name); 1046 dtb.size(namelen); 1047 dtb.xoff(d.csym, Type.typeinfoenum.structsize); 1048 1049 // void[] init; 1050 if (!sd.members || d.tinfo.isZeroInit(Loc.initial)) 1051 { 1052 // 0 initializer, or the same as the base type 1053 dtb.size(0); // init.length 1054 dtb.size(0); // init.ptr 1055 } 1056 else 1057 { 1058 dtb.size(sd.type.size()); // init.length 1059 dtb.xoff(toInitializer(sd), 0); // init.ptr 1060 } 1061 1062 // Put out name[] immediately following TypeInfo_Enum 1063 dtb.nbytes(cast(uint)(namelen + 1), name); 1064 } 1065 1066 override void visit(TypeInfoPointerDeclaration d) 1067 { 1068 //printf("TypeInfoPointerDeclaration.toDt()\n"); 1069 verifyStructSize(Type.typeinfopointer, 3 * target.ptrsize); 1070 1071 dtb.xoff(toVtblSymbol(Type.typeinfopointer), 0); // vtbl for TypeInfo_Pointer 1072 if (Type.typeinfopointer.hasMonitor()) 1073 dtb.size(0); // monitor 1074 1075 auto tc = d.tinfo.isTypePointer(); 1076 1077 genTypeInfo(d.loc, tc.next, null); 1078 dtb.xoff(toSymbol(tc.next.vtinfo), 0); // TypeInfo for type being pointed to 1079 } 1080 1081 override void visit(TypeInfoArrayDeclaration d) 1082 { 1083 //printf("TypeInfoArrayDeclaration.toDt()\n"); 1084 verifyStructSize(Type.typeinfoarray, 3 * target.ptrsize); 1085 1086 dtb.xoff(toVtblSymbol(Type.typeinfoarray), 0); // vtbl for TypeInfo_Array 1087 if (Type.typeinfoarray.hasMonitor()) 1088 dtb.size(0); // monitor 1089 1090 auto tc = d.tinfo.isTypeDArray(); 1091 1092 genTypeInfo(d.loc, tc.next, null); 1093 dtb.xoff(toSymbol(tc.next.vtinfo), 0); // TypeInfo for array of type 1094 } 1095 1096 override void visit(TypeInfoStaticArrayDeclaration d) 1097 { 1098 //printf("TypeInfoStaticArrayDeclaration.toDt()\n"); 1099 verifyStructSize(Type.typeinfostaticarray, 4 * target.ptrsize); 1100 1101 dtb.xoff(toVtblSymbol(Type.typeinfostaticarray), 0); // vtbl for TypeInfo_StaticArray 1102 if (Type.typeinfostaticarray.hasMonitor()) 1103 dtb.size(0); // monitor 1104 1105 auto tc = d.tinfo.isTypeSArray(); 1106 1107 genTypeInfo(d.loc, tc.next, null); 1108 dtb.xoff(toSymbol(tc.next.vtinfo), 0); // TypeInfo for array of type 1109 1110 dtb.size(tc.dim.toInteger()); // length 1111 } 1112 1113 override void visit(TypeInfoVectorDeclaration d) 1114 { 1115 //printf("TypeInfoVectorDeclaration.toDt()\n"); 1116 verifyStructSize(Type.typeinfovector, 3 * target.ptrsize); 1117 1118 dtb.xoff(toVtblSymbol(Type.typeinfovector), 0); // vtbl for TypeInfo_Vector 1119 if (Type.typeinfovector.hasMonitor()) 1120 dtb.size(0); // monitor 1121 1122 auto tc = d.tinfo.isTypeVector(); 1123 1124 genTypeInfo(d.loc, tc.basetype, null); 1125 dtb.xoff(toSymbol(tc.basetype.vtinfo), 0); // TypeInfo for equivalent static array 1126 } 1127 1128 override void visit(TypeInfoAssociativeArrayDeclaration d) 1129 { 1130 //printf("TypeInfoAssociativeArrayDeclaration.toDt()\n"); 1131 verifyStructSize(Type.typeinfoassociativearray, 4 * target.ptrsize); 1132 1133 dtb.xoff(toVtblSymbol(Type.typeinfoassociativearray), 0); // vtbl for TypeInfo_AssociativeArray 1134 if (Type.typeinfoassociativearray.hasMonitor()) 1135 dtb.size(0); // monitor 1136 1137 auto tc = d.tinfo.isTypeAArray(); 1138 1139 genTypeInfo(d.loc, tc.next, null); 1140 dtb.xoff(toSymbol(tc.next.vtinfo), 0); // TypeInfo for array of type 1141 1142 genTypeInfo(d.loc, tc.index, null); 1143 dtb.xoff(toSymbol(tc.index.vtinfo), 0); // TypeInfo for array of type 1144 } 1145 1146 override void visit(TypeInfoFunctionDeclaration d) 1147 { 1148 //printf("TypeInfoFunctionDeclaration.toDt()\n"); 1149 verifyStructSize(Type.typeinfofunction, 5 * target.ptrsize); 1150 1151 dtb.xoff(toVtblSymbol(Type.typeinfofunction), 0); // vtbl for TypeInfo_Function 1152 if (Type.typeinfofunction.hasMonitor()) 1153 dtb.size(0); // monitor 1154 1155 auto tc = d.tinfo.isTypeFunction(); 1156 1157 genTypeInfo(d.loc, tc.next, null); 1158 dtb.xoff(toSymbol(tc.next.vtinfo), 0); // TypeInfo for function return value 1159 1160 const name = d.tinfo.deco; 1161 assert(name); 1162 const namelen = strlen(name); 1163 dtb.size(namelen); 1164 dtb.xoff(d.csym, Type.typeinfofunction.structsize); 1165 1166 // Put out name[] immediately following TypeInfo_Function 1167 dtb.nbytes(cast(uint)(namelen + 1), name); 1168 } 1169 1170 override void visit(TypeInfoDelegateDeclaration d) 1171 { 1172 //printf("TypeInfoDelegateDeclaration.toDt()\n"); 1173 verifyStructSize(Type.typeinfodelegate, 5 * target.ptrsize); 1174 1175 dtb.xoff(toVtblSymbol(Type.typeinfodelegate), 0); // vtbl for TypeInfo_Delegate 1176 if (Type.typeinfodelegate.hasMonitor()) 1177 dtb.size(0); // monitor 1178 1179 auto tc = d.tinfo.isTypeDelegate(); 1180 1181 genTypeInfo(d.loc, tc.next.nextOf(), null); 1182 dtb.xoff(toSymbol(tc.next.nextOf().vtinfo), 0); // TypeInfo for delegate return value 1183 1184 const name = d.tinfo.deco; 1185 assert(name); 1186 const namelen = strlen(name); 1187 dtb.size(namelen); 1188 dtb.xoff(d.csym, Type.typeinfodelegate.structsize); 1189 1190 // Put out name[] immediately following TypeInfo_Delegate 1191 dtb.nbytes(cast(uint)(namelen + 1), name); 1192 } 1193 1194 override void visit(TypeInfoStructDeclaration d) 1195 { 1196 //printf("TypeInfoStructDeclaration.toDt() '%s'\n", d.toChars()); 1197 if (global.params.is64bit) 1198 verifyStructSize(Type.typeinfostruct, 17 * target.ptrsize); 1199 else 1200 verifyStructSize(Type.typeinfostruct, 15 * target.ptrsize); 1201 1202 dtb.xoff(toVtblSymbol(Type.typeinfostruct), 0); // vtbl for TypeInfo_Struct 1203 if (Type.typeinfostruct.hasMonitor()) 1204 dtb.size(0); // monitor 1205 1206 auto tc = d.tinfo.isTypeStruct(); 1207 StructDeclaration sd = tc.sym; 1208 1209 if (!sd.members) 1210 return; 1211 1212 if (TemplateInstance ti = sd.isInstantiated()) 1213 { 1214 if (!ti.needsCodegen()) 1215 { 1216 assert(ti.minst || sd.requestTypeInfo); 1217 1218 /* ti.toObjFile() won't get called. So, store these 1219 * member functions into object file in here. 1220 */ 1221 if (sd.xeq && sd.xeq != StructDeclaration.xerreq) 1222 toObjFile(sd.xeq, global.params.multiobj); 1223 if (sd.xcmp && sd.xcmp != StructDeclaration.xerrcmp) 1224 toObjFile(sd.xcmp, global.params.multiobj); 1225 if (FuncDeclaration ftostr = search_toString(sd)) 1226 toObjFile(ftostr, global.params.multiobj); 1227 if (sd.xhash) 1228 toObjFile(sd.xhash, global.params.multiobj); 1229 if (sd.postblit) 1230 toObjFile(sd.postblit, global.params.multiobj); 1231 if (sd.dtor) 1232 toObjFile(sd.dtor, global.params.multiobj); 1233 } 1234 } 1235 1236 /* Put out: 1237 * char[] name; 1238 * void[] init; 1239 * hash_t function(in void*) xtoHash; 1240 * bool function(in void*, in void*) xopEquals; 1241 * int function(in void*, in void*) xopCmp; 1242 * string function(const(void)*) xtoString; 1243 * StructFlags m_flags; 1244 * //xgetMembers; 1245 * xdtor; 1246 * xpostblit; 1247 * uint m_align; 1248 * version (X86_64) 1249 * TypeInfo m_arg1; 1250 * TypeInfo m_arg2; 1251 * xgetRTInfo 1252 */ 1253 1254 const name = sd.toPrettyChars(); 1255 const namelen = strlen(name); 1256 dtb.size(namelen); 1257 dtb.xoff(d.csym, Type.typeinfostruct.structsize); 1258 1259 // void[] init; 1260 dtb.size(sd.structsize); // init.length 1261 if (sd.zeroInit) 1262 dtb.size(0); // null for 0 initialization 1263 else 1264 dtb.xoff(toInitializer(sd), 0); // init.ptr 1265 1266 if (FuncDeclaration fd = sd.xhash) 1267 { 1268 dtb.xoff(toSymbol(fd), 0); 1269 TypeFunction tf = cast(TypeFunction)fd.type; 1270 assert(tf.ty == Tfunction); 1271 /* I'm a little unsure this is the right way to do it. Perhaps a better 1272 * way would to automatically add these attributes to any struct member 1273 * function with the name "toHash". 1274 * So I'm leaving this here as an experiment for the moment. 1275 */ 1276 if (!tf.isnothrow || tf.trust == TRUST.system /*|| tf.purity == PURE.impure*/) 1277 warning(fd.loc, "toHash() must be declared as extern (D) size_t toHash() const nothrow @safe, not %s", tf.toChars()); 1278 } 1279 else 1280 dtb.size(0); 1281 1282 if (sd.xeq) 1283 dtb.xoff(toSymbol(sd.xeq), 0); 1284 else 1285 dtb.size(0); 1286 1287 if (sd.xcmp) 1288 dtb.xoff(toSymbol(sd.xcmp), 0); 1289 else 1290 dtb.size(0); 1291 1292 if (FuncDeclaration fd = search_toString(sd)) 1293 { 1294 dtb.xoff(toSymbol(fd), 0); 1295 } 1296 else 1297 dtb.size(0); 1298 1299 // StructFlags m_flags; 1300 StructFlags m_flags = StructFlags.none; 1301 if (tc.hasPointers()) m_flags |= StructFlags.hasPointers; 1302 dtb.size(m_flags); 1303 1304 version (none) 1305 { 1306 // xgetMembers 1307 if (auto sgetmembers = sd.findGetMembers()) 1308 dtb.xoff(toSymbol(sgetmembers), 0); 1309 else 1310 dtb.size(0); // xgetMembers 1311 } 1312 1313 // xdtor 1314 if (auto sdtor = sd.tidtor) 1315 dtb.xoff(toSymbol(sdtor), 0); 1316 else 1317 dtb.size(0); // xdtor 1318 1319 // xpostblit 1320 FuncDeclaration spostblit = sd.postblit; 1321 if (spostblit && !(spostblit.storage_class & STC.disable)) 1322 dtb.xoff(toSymbol(spostblit), 0); 1323 else 1324 dtb.size(0); // xpostblit 1325 1326 // uint m_align; 1327 dtb.size(tc.alignsize()); 1328 1329 if (global.params.is64bit) 1330 { 1331 Type t = sd.arg1type; 1332 foreach (i; 0 .. 2) 1333 { 1334 // m_argi 1335 if (t) 1336 { 1337 genTypeInfo(d.loc, t, null); 1338 dtb.xoff(toSymbol(t.vtinfo), 0); 1339 } 1340 else 1341 dtb.size(0); 1342 1343 t = sd.arg2type; 1344 } 1345 } 1346 1347 // xgetRTInfo 1348 if (sd.getRTInfo) 1349 { 1350 Expression_toDt(sd.getRTInfo, *dtb); 1351 } 1352 else if (m_flags & StructFlags.hasPointers) 1353 dtb.size(1); 1354 else 1355 dtb.size(0); 1356 1357 // Put out name[] immediately following TypeInfo_Struct 1358 dtb.nbytes(cast(uint)(namelen + 1), name); 1359 } 1360 1361 override void visit(TypeInfoClassDeclaration d) 1362 { 1363 //printf("TypeInfoClassDeclaration.toDt() %s\n", tinfo.toChars()); 1364 assert(0); 1365 } 1366 1367 override void visit(TypeInfoInterfaceDeclaration d) 1368 { 1369 //printf("TypeInfoInterfaceDeclaration.toDt() %s\n", tinfo.toChars()); 1370 verifyStructSize(Type.typeinfointerface, 3 * target.ptrsize); 1371 1372 dtb.xoff(toVtblSymbol(Type.typeinfointerface), 0); // vtbl for TypeInfoInterface 1373 if (Type.typeinfointerface.hasMonitor()) 1374 dtb.size(0); // monitor 1375 1376 auto tc = d.tinfo.isTypeClass(); 1377 1378 if (!tc.sym.vclassinfo) 1379 tc.sym.vclassinfo = TypeInfoClassDeclaration.create(tc); 1380 auto s = toSymbol(tc.sym.vclassinfo); 1381 dtb.xoff(s, 0); // ClassInfo for tinfo 1382 } 1383 1384 override void visit(TypeInfoTupleDeclaration d) 1385 { 1386 //printf("TypeInfoTupleDeclaration.toDt() %s\n", tinfo.toChars()); 1387 verifyStructSize(Type.typeinfotypelist, 4 * target.ptrsize); 1388 1389 dtb.xoff(toVtblSymbol(Type.typeinfotypelist), 0); // vtbl for TypeInfoInterface 1390 if (Type.typeinfotypelist.hasMonitor()) 1391 dtb.size(0); // monitor 1392 1393 auto tu = d.tinfo.isTypeTuple(); 1394 1395 const dim = tu.arguments.dim; 1396 dtb.size(dim); // elements.length 1397 1398 auto dtbargs = DtBuilder(0); 1399 foreach (arg; *tu.arguments) 1400 { 1401 genTypeInfo(d.loc, arg.type, null); 1402 Symbol* s = toSymbol(arg.type.vtinfo); 1403 dtbargs.xoff(s, 0); 1404 } 1405 1406 dtb.dtoff(dtbargs.finish(), 0); // elements.ptr 1407 } 1408 } 1409 1410 extern (C++) void TypeInfo_toDt(ref DtBuilder dtb, TypeInfoDeclaration d) 1411 { 1412 scope v = new TypeInfoDtVisitor(dtb); 1413 d.accept(v); 1414 }