1 /** 2 * CTFE for expressions involving pointers, slices, array concatenation etc. 3 * 4 * Copyright: Copyright (C) 1999-2020 by The D Language Foundation, All Rights Reserved 5 * Authors: $(LINK2 http://www.digitalmars.com, Walter Bright) 6 * License: $(LINK2 http://www.boost.org/LICENSE_1_0.txt, Boost License 1.0) 7 * Source: $(LINK2 https://github.com/dlang/dmd/blob/master/src/dmd/ctfeexpr.d, _ctfeexpr.d) 8 * Documentation: https://dlang.org/phobos/dmd_ctfeexpr.html 9 * Coverage: https://codecov.io/gh/dlang/dmd/src/master/src/dmd/ctfeexpr.d 10 */ 11 12 module dmd.ctfeexpr; 13 14 import core.stdc.stdio; 15 import core.stdc.stdlib; 16 import core.stdc.string; 17 import dmd.arraytypes; 18 import dmd.complex; 19 import dmd.constfold; 20 import dmd.compiler; 21 import dmd.dclass; 22 import dmd.declaration; 23 import dmd.dinterpret; 24 import dmd.dstruct; 25 import dmd.dtemplate; 26 import dmd.errors; 27 import dmd.expression; 28 import dmd.func; 29 import dmd.globals; 30 import dmd.mtype; 31 import dmd.root.ctfloat; 32 import dmd.root.port; 33 import dmd.root.rmem; 34 import dmd.tokens; 35 import dmd.visitor; 36 37 38 /*********************************************************** 39 * A reference to a class, or an interface. We need this when we 40 * point to a base class (we must record what the type is). 41 */ 42 extern (C++) final class ClassReferenceExp : Expression 43 { 44 StructLiteralExp value; 45 46 extern (D) this(const ref Loc loc, StructLiteralExp lit, Type type) 47 { 48 super(loc, TOK.classReference, __traits(classInstanceSize, ClassReferenceExp)); 49 assert(lit && lit.sd && lit.sd.isClassDeclaration()); 50 this.value = lit; 51 this.type = type; 52 } 53 54 ClassDeclaration originalClass() 55 { 56 return value.sd.isClassDeclaration(); 57 } 58 59 // Return index of the field, or -1 if not found 60 private int getFieldIndex(Type fieldtype, uint fieldoffset) 61 { 62 ClassDeclaration cd = originalClass(); 63 uint fieldsSoFar = 0; 64 for (size_t j = 0; j < value.elements.dim; j++) 65 { 66 while (j - fieldsSoFar >= cd.fields.dim) 67 { 68 fieldsSoFar += cd.fields.dim; 69 cd = cd.baseClass; 70 } 71 VarDeclaration v2 = cd.fields[j - fieldsSoFar]; 72 if (fieldoffset == v2.offset && fieldtype.size() == v2.type.size()) 73 { 74 return cast(int)(value.elements.dim - fieldsSoFar - cd.fields.dim + (j - fieldsSoFar)); 75 } 76 } 77 return -1; 78 } 79 80 // Return index of the field, or -1 if not found 81 // Same as getFieldIndex, but checks for a direct match with the VarDeclaration 82 int findFieldIndexByName(VarDeclaration v) 83 { 84 ClassDeclaration cd = originalClass(); 85 size_t fieldsSoFar = 0; 86 for (size_t j = 0; j < value.elements.dim; j++) 87 { 88 while (j - fieldsSoFar >= cd.fields.dim) 89 { 90 fieldsSoFar += cd.fields.dim; 91 cd = cd.baseClass; 92 } 93 VarDeclaration v2 = cd.fields[j - fieldsSoFar]; 94 if (v == v2) 95 { 96 return cast(int)(value.elements.dim - fieldsSoFar - cd.fields.dim + (j - fieldsSoFar)); 97 } 98 } 99 return -1; 100 } 101 102 override void accept(Visitor v) 103 { 104 v.visit(this); 105 } 106 } 107 108 /************************* 109 * Same as getFieldIndex, but checks for a direct match with the VarDeclaration 110 * Returns: 111 * index of the field, or -1 if not found 112 */ 113 int findFieldIndexByName(const StructDeclaration sd, const VarDeclaration v) pure 114 { 115 foreach (i, field; sd.fields) 116 { 117 if (field == v) 118 return cast(int)i; 119 } 120 return -1; 121 } 122 123 /*********************************************************** 124 * Fake class which holds the thrown exception. 125 * Used for implementing exception handling. 126 */ 127 extern (C++) final class ThrownExceptionExp : Expression 128 { 129 ClassReferenceExp thrown; // the thing being tossed 130 131 extern (D) this(const ref Loc loc, ClassReferenceExp victim) 132 { 133 super(loc, TOK.thrownException, __traits(classInstanceSize, ThrownExceptionExp)); 134 this.thrown = victim; 135 this.type = victim.type; 136 } 137 138 override const(char)* toChars() const 139 { 140 return "CTFE ThrownException"; 141 } 142 143 // Generate an error message when this exception is not caught 144 extern (D) void generateUncaughtError() 145 { 146 UnionExp ue = void; 147 Expression e = resolveSlice((*thrown.value.elements)[0], &ue); 148 StringExp se = e.toStringExp(); 149 thrown.error("uncaught CTFE exception `%s(%s)`", thrown.type.toChars(), se ? se.toChars() : e.toChars()); 150 /* Also give the line where the throw statement was. We won't have it 151 * in the case where the ThrowStatement is generated internally 152 * (eg, in ScopeStatement) 153 */ 154 if (loc.isValid() && !loc.equals(thrown.loc)) 155 .errorSupplemental(loc, "thrown from here"); 156 } 157 158 override void accept(Visitor v) 159 { 160 v.visit(this); 161 } 162 } 163 164 /*********************************************************** 165 * This type is only used by the interpreter. 166 */ 167 extern (C++) final class CTFEExp : Expression 168 { 169 extern (D) this(TOK tok) 170 { 171 super(Loc.initial, tok, __traits(classInstanceSize, CTFEExp)); 172 type = Type.tvoid; 173 } 174 175 override const(char)* toChars() const 176 { 177 switch (op) 178 { 179 case TOK.cantExpression: 180 return "<cant>"; 181 case TOK.voidExpression: 182 return "<void>"; 183 case TOK.showCtfeContext: 184 return "<error>"; 185 case TOK.break_: 186 return "<break>"; 187 case TOK.continue_: 188 return "<continue>"; 189 case TOK.goto_: 190 return "<goto>"; 191 default: 192 assert(0); 193 } 194 } 195 196 extern (D) __gshared CTFEExp cantexp; 197 extern (D) __gshared CTFEExp voidexp; 198 extern (D) __gshared CTFEExp breakexp; 199 extern (D) __gshared CTFEExp continueexp; 200 extern (D) __gshared CTFEExp gotoexp; 201 /* Used when additional information is needed regarding 202 * a ctfe error. 203 */ 204 extern (D) __gshared CTFEExp showcontext; 205 206 extern (D) static bool isCantExp(const Expression e) 207 { 208 return e && e.op == TOK.cantExpression; 209 } 210 211 extern (D) static bool isGotoExp(const Expression e) 212 { 213 return e && e.op == TOK.goto_; 214 } 215 } 216 217 // True if 'e' is CTFEExp::cantexp, or an exception 218 bool exceptionOrCantInterpret(const Expression e) 219 { 220 return e && (e.op == TOK.cantExpression || e.op == TOK.thrownException || e.op == TOK.showCtfeContext); 221 } 222 223 /************** Aggregate literals (AA/string/array/struct) ******************/ 224 // Given expr, which evaluates to an array/AA/string literal, 225 // return true if it needs to be copied 226 bool needToCopyLiteral(const Expression expr) 227 { 228 Expression e = cast()expr; 229 for (;;) 230 { 231 switch (e.op) 232 { 233 case TOK.arrayLiteral: 234 return (cast(ArrayLiteralExp)e).ownedByCtfe == OwnedBy.code; 235 case TOK.assocArrayLiteral: 236 return (cast(AssocArrayLiteralExp)e).ownedByCtfe == OwnedBy.code; 237 case TOK.structLiteral: 238 return (cast(StructLiteralExp)e).ownedByCtfe == OwnedBy.code; 239 case TOK.string_: 240 case TOK.this_: 241 case TOK.variable: 242 return false; 243 case TOK.assign: 244 return false; 245 case TOK.index: 246 case TOK.dotVariable: 247 case TOK.slice: 248 case TOK.cast_: 249 e = (cast(UnaExp)e).e1; 250 continue; 251 case TOK.concatenate: 252 return needToCopyLiteral((cast(BinExp)e).e1) || needToCopyLiteral((cast(BinExp)e).e2); 253 case TOK.concatenateAssign: 254 case TOK.concatenateElemAssign: 255 case TOK.concatenateDcharAssign: 256 e = (cast(BinExp)e).e2; 257 continue; 258 default: 259 return false; 260 } 261 } 262 } 263 264 private Expressions* copyLiteralArray(Expressions* oldelems, Expression basis = null) 265 { 266 if (!oldelems) 267 return oldelems; 268 incArrayAllocs(); 269 auto newelems = new Expressions(oldelems.dim); 270 foreach (i, el; *oldelems) 271 { 272 (*newelems)[i] = copyLiteral(el ? el : basis).copy(); 273 } 274 return newelems; 275 } 276 277 // Make a copy of the ArrayLiteral, AALiteral, String, or StructLiteral. 278 // This value will be used for in-place modification. 279 UnionExp copyLiteral(Expression e) 280 { 281 UnionExp ue = void; 282 if (auto se = e.isStringExp()) // syntaxCopy doesn't make a copy for StringExp! 283 { 284 char* s = cast(char*)mem.xcalloc(se.len + 1, se.sz); 285 const slice = se.peekData(); 286 memcpy(s, slice.ptr, slice.length); 287 emplaceExp!(StringExp)(&ue, se.loc, s[0 .. se.len * se.sz], se.len, se.sz); 288 StringExp se2 = cast(StringExp)ue.exp(); 289 se2.committed = se.committed; 290 se2.postfix = se.postfix; 291 se2.type = se.type; 292 se2.ownedByCtfe = OwnedBy.ctfe; 293 return ue; 294 } 295 if (auto ale = e.isArrayLiteralExp()) 296 { 297 auto elements = copyLiteralArray(ale.elements, ale.basis); 298 299 emplaceExp!(ArrayLiteralExp)(&ue, e.loc, e.type, elements); 300 301 ArrayLiteralExp r = cast(ArrayLiteralExp)ue.exp(); 302 r.ownedByCtfe = OwnedBy.ctfe; 303 return ue; 304 } 305 if (auto aae = e.isAssocArrayLiteralExp()) 306 { 307 emplaceExp!(AssocArrayLiteralExp)(&ue, e.loc, copyLiteralArray(aae.keys), copyLiteralArray(aae.values)); 308 AssocArrayLiteralExp r = cast(AssocArrayLiteralExp)ue.exp(); 309 r.type = e.type; 310 r.ownedByCtfe = OwnedBy.ctfe; 311 return ue; 312 } 313 if (auto sle = e.isStructLiteralExp()) 314 { 315 /* syntaxCopy doesn't work for struct literals, because of a nasty special 316 * case: block assignment is permitted inside struct literals, eg, 317 * an int[4] array can be initialized with a single int. 318 */ 319 auto oldelems = sle.elements; 320 auto newelems = new Expressions(oldelems.dim); 321 foreach (i, ref el; *newelems) 322 { 323 // We need the struct definition to detect block assignment 324 auto v = sle.sd.fields[i]; 325 auto m = (*oldelems)[i]; 326 327 // If it is a void assignment, use the default initializer 328 if (!m) 329 m = voidInitLiteral(v.type, v).copy(); 330 331 if (v.type.ty == Tarray || v.type.ty == Taarray) 332 { 333 // Don't have to copy array references 334 } 335 else 336 { 337 // Buzilla 15681: Copy the source element always. 338 m = copyLiteral(m).copy(); 339 340 // Block assignment from inside struct literals 341 if (v.type.ty != m.type.ty && v.type.ty == Tsarray) 342 { 343 auto tsa = v.type.isTypeSArray(); 344 auto len = cast(size_t)tsa.dim.toInteger(); 345 UnionExp uex = void; 346 m = createBlockDuplicatedArrayLiteral(&uex, e.loc, v.type, m, len); 347 if (m == uex.exp()) 348 m = uex.copy(); 349 } 350 } 351 el = m; 352 } 353 emplaceExp!(StructLiteralExp)(&ue, e.loc, sle.sd, newelems, sle.stype); 354 auto r = ue.exp().isStructLiteralExp(); 355 r.type = e.type; 356 r.ownedByCtfe = OwnedBy.ctfe; 357 r.origin = sle.origin; 358 return ue; 359 } 360 if (e.op == TOK.function_ || e.op == TOK.delegate_ || e.op == TOK.symbolOffset || e.op == TOK.null_ || e.op == TOK.variable || e.op == TOK.dotVariable || e.op == TOK.int64 || e.op == TOK.float64 || e.op == TOK.char_ || e.op == TOK.complex80 || e.op == TOK.void_ || e.op == TOK.vector || e.op == TOK.typeid_) 361 { 362 // Simple value types 363 // Keep e1 for DelegateExp and DotVarExp 364 emplaceExp!(UnionExp)(&ue, e); 365 Expression r = ue.exp(); 366 r.type = e.type; 367 return ue; 368 } 369 if (auto se = e.isSliceExp()) 370 { 371 if (se.type.toBasetype().ty == Tsarray) 372 { 373 // same with resolveSlice() 374 if (se.e1.op == TOK.null_) 375 { 376 emplaceExp!(NullExp)(&ue, se.loc, se.type); 377 return ue; 378 } 379 ue = Slice(se.type, se.e1, se.lwr, se.upr); 380 auto r = ue.exp().isArrayLiteralExp(); 381 r.elements = copyLiteralArray(r.elements); 382 r.ownedByCtfe = OwnedBy.ctfe; 383 return ue; 384 } 385 else 386 { 387 // Array slices only do a shallow copy 388 emplaceExp!(SliceExp)(&ue, e.loc, se.e1, se.lwr, se.upr); 389 Expression r = ue.exp(); 390 r.type = e.type; 391 return ue; 392 } 393 } 394 if (isPointer(e.type)) 395 { 396 // For pointers, we only do a shallow copy. 397 if (auto ae = e.isAddrExp()) 398 emplaceExp!(AddrExp)(&ue, e.loc, ae.e1); 399 else if (auto ie = e.isIndexExp()) 400 emplaceExp!(IndexExp)(&ue, e.loc, ie.e1, ie.e2); 401 else if (auto dve = e.isDotVarExp()) 402 { 403 emplaceExp!(DotVarExp)(&ue, e.loc, dve.e1, dve.var, dve.hasOverloads); 404 } 405 else 406 assert(0); 407 408 Expression r = ue.exp(); 409 r.type = e.type; 410 return ue; 411 } 412 if (auto cre = e.isClassReferenceExp()) 413 { 414 emplaceExp!(ClassReferenceExp)(&ue, e.loc, cre.value, e.type); 415 return ue; 416 } 417 if (e.op == TOK.error) 418 { 419 emplaceExp!(UnionExp)(&ue, e); 420 return ue; 421 } 422 e.error("CTFE internal error: literal `%s`", e.toChars()); 423 assert(0); 424 } 425 426 /* Deal with type painting. 427 * Type painting is a major nuisance: we can't just set 428 * e.type = type, because that would change the original literal. 429 * But, we can't simply copy the literal either, because that would change 430 * the values of any pointers. 431 */ 432 Expression paintTypeOntoLiteral(Type type, Expression lit) 433 { 434 if (lit.type.equals(type)) 435 return lit; 436 return paintTypeOntoLiteralCopy(type, lit).copy(); 437 } 438 439 Expression paintTypeOntoLiteral(UnionExp* pue, Type type, Expression lit) 440 { 441 if (lit.type.equals(type)) 442 return lit; 443 *pue = paintTypeOntoLiteralCopy(type, lit); 444 return pue.exp(); 445 } 446 447 private UnionExp paintTypeOntoLiteralCopy(Type type, Expression lit) 448 { 449 UnionExp ue; 450 if (lit.type.equals(type)) 451 { 452 emplaceExp!(UnionExp)(&ue, lit); 453 return ue; 454 } 455 // If it is a cast to inout, retain the original type of the referenced part. 456 if (type.hasWild() && type.hasPointers()) 457 { 458 emplaceExp!(UnionExp)(&ue, lit); 459 ue.exp().type = type; 460 return ue; 461 } 462 if (auto se = lit.isSliceExp()) 463 { 464 emplaceExp!(SliceExp)(&ue, lit.loc, se.e1, se.lwr, se.upr); 465 } 466 else if (auto ie = lit.isIndexExp()) 467 { 468 emplaceExp!(IndexExp)(&ue, lit.loc, ie.e1, ie.e2); 469 } 470 else if (lit.op == TOK.arrayLiteral) 471 { 472 emplaceExp!(SliceExp)(&ue, lit.loc, lit, ctfeEmplaceExp!IntegerExp(Loc.initial, 0, Type.tsize_t), ArrayLength(Type.tsize_t, lit).copy()); 473 } 474 else if (lit.op == TOK.string_) 475 { 476 // For strings, we need to introduce another level of indirection 477 emplaceExp!(SliceExp)(&ue, lit.loc, lit, ctfeEmplaceExp!IntegerExp(Loc.initial, 0, Type.tsize_t), ArrayLength(Type.tsize_t, lit).copy()); 478 } 479 else if (auto aae = lit.isAssocArrayLiteralExp()) 480 { 481 // TODO: we should be creating a reference to this AAExp, not 482 // just a ref to the keys and values. 483 OwnedBy wasOwned = aae.ownedByCtfe; 484 emplaceExp!(AssocArrayLiteralExp)(&ue, lit.loc, aae.keys, aae.values); 485 aae = cast(AssocArrayLiteralExp)ue.exp(); 486 aae.ownedByCtfe = wasOwned; 487 } 488 else 489 { 490 // Can't type paint from struct to struct*; this needs another 491 // level of indirection 492 if (lit.op == TOK.structLiteral && isPointer(type)) 493 lit.error("CTFE internal error: painting `%s`", type.toChars()); 494 ue = copyLiteral(lit); 495 } 496 ue.exp().type = type; 497 return ue; 498 } 499 500 /************************************* 501 * If e is a SliceExp, constant fold it. 502 * Params: 503 * e = expression to resolve 504 * pue = if not null, store resulting expression here 505 * Returns: 506 * resulting expression 507 */ 508 Expression resolveSlice(Expression e, UnionExp* pue = null) 509 { 510 SliceExp se = e.isSliceExp(); 511 if (!se) 512 return e; 513 if (se.e1.op == TOK.null_) 514 return se.e1; 515 if (pue) 516 { 517 *pue = Slice(e.type, se.e1, se.lwr, se.upr); 518 return pue.exp(); 519 } 520 else 521 return Slice(e.type, se.e1, se.lwr, se.upr).copy(); 522 } 523 524 /* Determine the array length, without interpreting it. 525 * e must be an array literal, or a slice 526 * It's very wasteful to resolve the slice when we only 527 * need the length. 528 */ 529 uinteger_t resolveArrayLength(const Expression e) 530 { 531 switch (e.op) 532 { 533 case TOK.vector: 534 return e.isVectorExp().dim; 535 536 case TOK.null_: 537 return 0; 538 539 case TOK.slice: 540 { 541 auto se = cast(SliceExp)e; 542 const ilo = se.lwr.toInteger(); 543 const iup = se.upr.toInteger(); 544 return iup - ilo; 545 } 546 547 case TOK.string_: 548 return e.isStringExp().len; 549 550 case TOK.arrayLiteral: 551 { 552 const ale = e.isArrayLiteralExp(); 553 return ale.elements ? ale.elements.dim : 0; 554 } 555 556 case TOK.assocArrayLiteral: 557 { 558 return e.isAssocArrayLiteralExp().keys.dim; 559 } 560 561 default: 562 assert(0); 563 } 564 } 565 566 /****************************** 567 * Helper for NewExp 568 * Create an array literal consisting of 'elem' duplicated 'dim' times. 569 * Params: 570 * pue = where to store result 571 * loc = source location where the interpretation occurs 572 * type = target type of the result 573 * elem = the source of array element, it will be owned by the result 574 * dim = element number of the result 575 * Returns: 576 * Constructed ArrayLiteralExp 577 */ 578 ArrayLiteralExp createBlockDuplicatedArrayLiteral(UnionExp* pue, const ref Loc loc, Type type, Expression elem, size_t dim) 579 { 580 if (type.ty == Tsarray && type.nextOf().ty == Tsarray && elem.type.ty != Tsarray) 581 { 582 // If it is a multidimensional array literal, do it recursively 583 auto tsa = type.nextOf().isTypeSArray(); 584 const len = cast(size_t)tsa.dim.toInteger(); 585 UnionExp ue = void; 586 elem = createBlockDuplicatedArrayLiteral(&ue, loc, type.nextOf(), elem, len); 587 if (elem == ue.exp()) 588 elem = ue.copy(); 589 } 590 591 // Buzilla 15681 592 const tb = elem.type.toBasetype(); 593 const mustCopy = tb.ty == Tstruct || tb.ty == Tsarray; 594 595 auto elements = new Expressions(dim); 596 foreach (i, ref el; *elements) 597 { 598 el = mustCopy && i ? copyLiteral(elem).copy() : elem; 599 } 600 emplaceExp!(ArrayLiteralExp)(pue, loc, type, elements); 601 auto ale = pue.exp().isArrayLiteralExp(); 602 ale.ownedByCtfe = OwnedBy.ctfe; 603 return ale; 604 } 605 606 /****************************** 607 * Helper for NewExp 608 * Create a string literal consisting of 'value' duplicated 'dim' times. 609 */ 610 StringExp createBlockDuplicatedStringLiteral(UnionExp* pue, const ref Loc loc, Type type, dchar value, size_t dim, ubyte sz) 611 { 612 auto s = cast(char*)mem.xcalloc(dim, sz); 613 foreach (elemi; 0 .. dim) 614 { 615 switch (sz) 616 { 617 case 1: 618 s[elemi] = cast(char)value; 619 break; 620 case 2: 621 (cast(wchar*)s)[elemi] = cast(wchar)value; 622 break; 623 case 4: 624 (cast(dchar*)s)[elemi] = value; 625 break; 626 default: 627 assert(0); 628 } 629 } 630 emplaceExp!(StringExp)(pue, loc, s[0 .. dim * sz], dim, sz); 631 auto se = pue.exp().isStringExp(); 632 se.type = type; 633 se.committed = true; 634 se.ownedByCtfe = OwnedBy.ctfe; 635 return se; 636 } 637 638 // Return true if t is an AA 639 bool isAssocArray(Type t) 640 { 641 return t.toBasetype().isTypeAArray() !is null; 642 } 643 644 // Given a template AA type, extract the corresponding built-in AA type 645 TypeAArray toBuiltinAAType(Type t) 646 { 647 return t.toBasetype().isTypeAArray(); 648 } 649 650 /************** TypeInfo operations ************************************/ 651 // Return true if type is TypeInfo_Class 652 bool isTypeInfo_Class(const Type type) 653 { 654 auto tc = cast()type.isTypeClass(); 655 return tc && (Type.dtypeinfo == tc.sym || Type.dtypeinfo.isBaseOf(tc.sym, null)); 656 } 657 658 /************** Pointer operations ************************************/ 659 // Return true if t is a pointer (not a function pointer) 660 bool isPointer(Type t) 661 { 662 Type tb = t.toBasetype(); 663 return tb.ty == Tpointer && tb.nextOf().ty != Tfunction; 664 } 665 666 // For CTFE only. Returns true if 'e' is true or a non-null pointer. 667 bool isTrueBool(Expression e) 668 { 669 return e.isBool(true) || ((e.type.ty == Tpointer || e.type.ty == Tclass) && e.op != TOK.null_); 670 } 671 672 /* Is it safe to convert from srcPointee* to destPointee* ? 673 * srcPointee is the genuine type (never void). 674 * destPointee may be void. 675 */ 676 bool isSafePointerCast(Type srcPointee, Type destPointee) 677 { 678 // It's safe to cast S** to D** if it's OK to cast S* to D* 679 while (srcPointee.ty == Tpointer && destPointee.ty == Tpointer) 680 { 681 srcPointee = srcPointee.nextOf(); 682 destPointee = destPointee.nextOf(); 683 } 684 // It's OK if both are the same (modulo const) 685 if (srcPointee.constConv(destPointee)) 686 return true; 687 // It's OK if function pointers differ only in safe/pure/nothrow 688 if (srcPointee.ty == Tfunction && destPointee.ty == Tfunction) 689 return srcPointee.covariant(destPointee) == 1; 690 // it's OK to cast to void* 691 if (destPointee.ty == Tvoid) 692 return true; 693 // It's OK to cast from V[K] to void* 694 if (srcPointee.ty == Taarray && destPointee == Type.tvoidptr) 695 return true; 696 // It's OK if they are the same size (static array of) integers, eg: 697 // int* --> uint* 698 // int[5][] --> uint[5][] 699 if (srcPointee.ty == Tsarray && destPointee.ty == Tsarray) 700 { 701 if (srcPointee.size() != destPointee.size()) 702 return false; 703 srcPointee = srcPointee.baseElemOf(); 704 destPointee = destPointee.baseElemOf(); 705 } 706 return srcPointee.isintegral() && destPointee.isintegral() && srcPointee.size() == destPointee.size(); 707 } 708 709 Expression getAggregateFromPointer(Expression e, dinteger_t* ofs) 710 { 711 *ofs = 0; 712 if (auto ae = e.isAddrExp()) 713 e = ae.e1; 714 if (auto soe = e.isSymOffExp()) 715 *ofs = soe.offset; 716 if (auto dve = e.isDotVarExp()) 717 { 718 const ex = dve.e1; 719 const v = dve.var.isVarDeclaration(); 720 assert(v); 721 StructLiteralExp se = (ex.op == TOK.classReference) 722 ? (cast(ClassReferenceExp)ex).value 723 : cast(StructLiteralExp)ex; 724 725 // We can't use getField, because it makes a copy 726 const i = (ex.op == TOK.classReference) 727 ? (cast(ClassReferenceExp)ex).getFieldIndex(e.type, v.offset) 728 : se.getFieldIndex(e.type, v.offset); 729 e = (*se.elements)[i]; 730 } 731 if (auto ie = e.isIndexExp()) 732 { 733 // Note that each AA element is part of its own memory block 734 if ((ie.e1.type.ty == Tarray || ie.e1.type.ty == Tsarray || ie.e1.op == TOK.string_ || ie.e1.op == TOK.arrayLiteral) && ie.e2.op == TOK.int64) 735 { 736 *ofs = ie.e2.toInteger(); 737 return ie.e1; 738 } 739 } 740 if (auto se = e.isSliceExp()) 741 { 742 if (se && e.type.toBasetype().ty == Tsarray && 743 (se.e1.type.ty == Tarray || se.e1.type.ty == Tsarray || se.e1.op == TOK.string_ || se.e1.op == TOK.arrayLiteral) && se.lwr.op == TOK.int64) 744 { 745 *ofs = se.lwr.toInteger(); 746 return se.e1; 747 } 748 } 749 return e; 750 } 751 752 /** Return true if agg1 and agg2 are pointers to the same memory block 753 */ 754 bool pointToSameMemoryBlock(Expression agg1, Expression agg2) 755 { 756 if (agg1 == agg2) 757 return true; 758 // For integers cast to pointers, we regard them as non-comparable 759 // unless they are identical. (This may be overly strict). 760 if (agg1.op == TOK.int64 && agg2.op == TOK.int64 && agg1.toInteger() == agg2.toInteger()) 761 { 762 return true; 763 } 764 // Note that type painting can occur with VarExp, so we 765 // must compare the variables being pointed to. 766 if (agg1.op == TOK.variable && agg2.op == TOK.variable && (cast(VarExp)agg1).var == (cast(VarExp)agg2).var) 767 { 768 return true; 769 } 770 if (agg1.op == TOK.symbolOffset && agg2.op == TOK.symbolOffset && (cast(SymOffExp)agg1).var == (cast(SymOffExp)agg2).var) 771 { 772 return true; 773 } 774 return false; 775 } 776 777 // return e1 - e2 as an integer, or error if not possible 778 UnionExp pointerDifference(const ref Loc loc, Type type, Expression e1, Expression e2) 779 { 780 UnionExp ue = void; 781 dinteger_t ofs1, ofs2; 782 Expression agg1 = getAggregateFromPointer(e1, &ofs1); 783 Expression agg2 = getAggregateFromPointer(e2, &ofs2); 784 if (agg1 == agg2) 785 { 786 Type pointee = (cast(TypePointer)agg1.type).next; 787 const sz = pointee.size(); 788 emplaceExp!(IntegerExp)(&ue, loc, (ofs1 - ofs2) * sz, type); 789 } 790 else if (agg1.op == TOK.string_ && agg2.op == TOK.string_ && 791 (cast(StringExp)agg1).peekString().ptr == (cast(StringExp)agg2).peekString().ptr) 792 { 793 Type pointee = (cast(TypePointer)agg1.type).next; 794 const sz = pointee.size(); 795 emplaceExp!(IntegerExp)(&ue, loc, (ofs1 - ofs2) * sz, type); 796 } 797 else if (agg1.op == TOK.symbolOffset && agg2.op == TOK.symbolOffset && 798 (cast(SymOffExp)agg1).var == (cast(SymOffExp)agg2).var) 799 { 800 emplaceExp!(IntegerExp)(&ue, loc, ofs1 - ofs2, type); 801 } 802 else 803 { 804 error(loc, "`%s - %s` cannot be interpreted at compile time: cannot subtract pointers to two different memory blocks", e1.toChars(), e2.toChars()); 805 emplaceExp!(CTFEExp)(&ue, TOK.cantExpression); 806 } 807 return ue; 808 } 809 810 // Return eptr op e2, where eptr is a pointer, e2 is an integer, 811 // and op is TOK.add or TOK.min 812 UnionExp pointerArithmetic(const ref Loc loc, TOK op, Type type, Expression eptr, Expression e2) 813 { 814 UnionExp ue; 815 if (eptr.type.nextOf().ty == Tvoid) 816 { 817 error(loc, "cannot perform arithmetic on `void*` pointers at compile time"); 818 Lcant: 819 emplaceExp!(CTFEExp)(&ue, TOK.cantExpression); 820 return ue; 821 } 822 if (eptr.op == TOK.address) 823 eptr = (cast(AddrExp)eptr).e1; 824 dinteger_t ofs1; 825 Expression agg1 = getAggregateFromPointer(eptr, &ofs1); 826 if (agg1.op == TOK.symbolOffset) 827 { 828 if ((cast(SymOffExp)agg1).var.type.ty != Tsarray) 829 { 830 error(loc, "cannot perform pointer arithmetic on arrays of unknown length at compile time"); 831 goto Lcant; 832 } 833 } 834 else if (agg1.op != TOK.string_ && agg1.op != TOK.arrayLiteral) 835 { 836 error(loc, "cannot perform pointer arithmetic on non-arrays at compile time"); 837 goto Lcant; 838 } 839 dinteger_t ofs2 = e2.toInteger(); 840 Type pointee = (cast(TypeNext)agg1.type.toBasetype()).next; 841 dinteger_t sz = pointee.size(); 842 sinteger_t indx; 843 dinteger_t len; 844 if (agg1.op == TOK.symbolOffset) 845 { 846 indx = ofs1 / sz; 847 len = (cast(TypeSArray)(cast(SymOffExp)agg1).var.type).dim.toInteger(); 848 } 849 else 850 { 851 Expression dollar = ArrayLength(Type.tsize_t, agg1).copy(); 852 assert(!CTFEExp.isCantExp(dollar)); 853 indx = ofs1; 854 len = dollar.toInteger(); 855 } 856 if (op == TOK.add || op == TOK.addAssign || op == TOK.plusPlus) 857 indx += ofs2 / sz; 858 else if (op == TOK.min || op == TOK.minAssign || op == TOK.minusMinus) 859 indx -= ofs2 / sz; 860 else 861 { 862 error(loc, "CTFE internal error: bad pointer operation"); 863 goto Lcant; 864 } 865 if (indx < 0 || len < indx) 866 { 867 error(loc, "cannot assign pointer to index %lld inside memory block `[0..%lld]`", indx, len); 868 goto Lcant; 869 } 870 if (agg1.op == TOK.symbolOffset) 871 { 872 emplaceExp!(SymOffExp)(&ue, loc, (cast(SymOffExp)agg1).var, indx * sz); 873 SymOffExp se = cast(SymOffExp)ue.exp(); 874 se.type = type; 875 return ue; 876 } 877 if (agg1.op != TOK.arrayLiteral && agg1.op != TOK.string_) 878 { 879 error(loc, "CTFE internal error: pointer arithmetic `%s`", agg1.toChars()); 880 goto Lcant; 881 } 882 if (eptr.type.toBasetype().ty == Tsarray) 883 { 884 dinteger_t dim = (cast(TypeSArray)eptr.type.toBasetype()).dim.toInteger(); 885 // Create a CTFE pointer &agg1[indx .. indx+dim] 886 auto se = ctfeEmplaceExp!SliceExp(loc, agg1, 887 ctfeEmplaceExp!IntegerExp(loc, indx, Type.tsize_t), 888 ctfeEmplaceExp!IntegerExp(loc, indx + dim, Type.tsize_t)); 889 se.type = type.toBasetype().nextOf(); 890 emplaceExp!(AddrExp)(&ue, loc, se); 891 ue.exp().type = type; 892 return ue; 893 } 894 // Create a CTFE pointer &agg1[indx] 895 auto ofs = ctfeEmplaceExp!IntegerExp(loc, indx, Type.tsize_t); 896 Expression ie = ctfeEmplaceExp!IndexExp(loc, agg1, ofs); 897 ie.type = type.toBasetype().nextOf(); // https://issues.dlang.org/show_bug.cgi?id=13992 898 emplaceExp!(AddrExp)(&ue, loc, ie); 899 ue.exp().type = type; 900 return ue; 901 } 902 903 // Return 1 if true, 0 if false 904 // -1 if comparison is illegal because they point to non-comparable memory blocks 905 int comparePointers(TOK op, Expression agg1, dinteger_t ofs1, Expression agg2, dinteger_t ofs2) 906 { 907 if (pointToSameMemoryBlock(agg1, agg2)) 908 { 909 int n; 910 switch (op) 911 { 912 case TOK.lessThan: 913 n = (ofs1 < ofs2); 914 break; 915 case TOK.lessOrEqual: 916 n = (ofs1 <= ofs2); 917 break; 918 case TOK.greaterThan: 919 n = (ofs1 > ofs2); 920 break; 921 case TOK.greaterOrEqual: 922 n = (ofs1 >= ofs2); 923 break; 924 case TOK.identity: 925 case TOK.equal: 926 n = (ofs1 == ofs2); 927 break; 928 case TOK.notIdentity: 929 case TOK.notEqual: 930 n = (ofs1 != ofs2); 931 break; 932 default: 933 assert(0); 934 } 935 return n; 936 } 937 const null1 = (agg1.op == TOK.null_); 938 const null2 = (agg2.op == TOK.null_); 939 int cmp; 940 if (null1 || null2) 941 { 942 switch (op) 943 { 944 case TOK.lessThan: 945 cmp = null1 && !null2; 946 break; 947 case TOK.greaterThan: 948 cmp = !null1 && null2; 949 break; 950 case TOK.lessOrEqual: 951 cmp = null1; 952 break; 953 case TOK.greaterOrEqual: 954 cmp = null2; 955 break; 956 case TOK.identity: 957 case TOK.equal: 958 case TOK.notIdentity: // 'cmp' gets inverted below 959 case TOK.notEqual: 960 cmp = (null1 == null2); 961 break; 962 default: 963 assert(0); 964 } 965 } 966 else 967 { 968 switch (op) 969 { 970 case TOK.identity: 971 case TOK.equal: 972 case TOK.notIdentity: // 'cmp' gets inverted below 973 case TOK.notEqual: 974 cmp = 0; 975 break; 976 default: 977 return -1; // memory blocks are different 978 } 979 } 980 if (op == TOK.notIdentity || op == TOK.notEqual) 981 cmp ^= 1; 982 return cmp; 983 } 984 985 // True if conversion from type 'from' to 'to' involves a reinterpret_cast 986 // floating point -> integer or integer -> floating point 987 bool isFloatIntPaint(Type to, Type from) 988 { 989 return from.size() == to.size() && (from.isintegral() && to.isfloating() || from.isfloating() && to.isintegral()); 990 } 991 992 // Reinterpret float/int value 'fromVal' as a float/integer of type 'to'. 993 Expression paintFloatInt(UnionExp* pue, Expression fromVal, Type to) 994 { 995 if (exceptionOrCantInterpret(fromVal)) 996 return fromVal; 997 assert(to.size() == 4 || to.size() == 8); 998 return Compiler.paintAsType(pue, fromVal, to); 999 } 1000 1001 /******** Constant folding, with support for CTFE ***************************/ 1002 /// Return true if non-pointer expression e can be compared 1003 /// with >,is, ==, etc, using ctfeCmp, ctfeEqual, ctfeIdentity 1004 bool isCtfeComparable(Expression e) 1005 { 1006 if (e.op == TOK.slice) 1007 e = (cast(SliceExp)e).e1; 1008 if (e.isConst() != 1) 1009 { 1010 if (e.op == TOK.null_ || e.op == TOK.string_ || e.op == TOK.function_ || e.op == TOK.delegate_ || e.op == TOK.arrayLiteral || e.op == TOK.structLiteral || e.op == TOK.assocArrayLiteral || e.op == TOK.classReference) 1011 { 1012 return true; 1013 } 1014 // https://issues.dlang.org/show_bug.cgi?id=14123 1015 // TypeInfo object is comparable in CTFE 1016 if (e.op == TOK.typeid_) 1017 return true; 1018 return false; 1019 } 1020 return true; 1021 } 1022 1023 /// Map TOK comparison ops 1024 private bool numCmp(N)(TOK op, N n1, N n2) 1025 { 1026 switch (op) 1027 { 1028 case TOK.lessThan: 1029 return n1 < n2; 1030 case TOK.lessOrEqual: 1031 return n1 <= n2; 1032 case TOK.greaterThan: 1033 return n1 > n2; 1034 case TOK.greaterOrEqual: 1035 return n1 >= n2; 1036 1037 default: 1038 assert(0); 1039 } 1040 } 1041 1042 /// Returns cmp OP 0; where OP is ==, !=, <, >=, etc. Result is 0 or 1 1043 bool specificCmp(TOK op, int rawCmp) 1044 { 1045 return numCmp!int(op, rawCmp, 0); 1046 } 1047 1048 /// Returns e1 OP e2; where OP is ==, !=, <, >=, etc. Result is 0 or 1 1049 bool intUnsignedCmp(TOK op, dinteger_t n1, dinteger_t n2) 1050 { 1051 return numCmp!dinteger_t(op, n1, n2); 1052 } 1053 1054 /// Returns e1 OP e2; where OP is ==, !=, <, >=, etc. Result is 0 or 1 1055 bool intSignedCmp(TOK op, sinteger_t n1, sinteger_t n2) 1056 { 1057 return numCmp!sinteger_t(op, n1, n2); 1058 } 1059 1060 /// Returns e1 OP e2; where OP is ==, !=, <, >=, etc. Result is 0 or 1 1061 bool realCmp(TOK op, real_t r1, real_t r2) 1062 { 1063 // Don't rely on compiler, handle NAN arguments separately 1064 if (CTFloat.isNaN(r1) || CTFloat.isNaN(r2)) // if unordered 1065 { 1066 switch (op) 1067 { 1068 case TOK.lessThan: 1069 case TOK.lessOrEqual: 1070 case TOK.greaterThan: 1071 case TOK.greaterOrEqual: 1072 return false; 1073 1074 default: 1075 assert(0); 1076 } 1077 } 1078 else 1079 { 1080 return numCmp!real_t(op, r1, r2); 1081 } 1082 } 1083 1084 /* Conceptually the same as memcmp(e1, e2). 1085 * e1 and e2 may be strings, arrayliterals, or slices. 1086 * For string types, return <0 if e1 < e2, 0 if e1==e2, >0 if e1 > e2. 1087 * For all other types, return 0 if e1 == e2, !=0 if e1 != e2. 1088 * Returns: 1089 * -1,0,1 1090 */ 1091 private int ctfeCmpArrays(const ref Loc loc, Expression e1, Expression e2, uinteger_t len) 1092 { 1093 // Resolve slices, if necessary 1094 uinteger_t lo1 = 0; 1095 uinteger_t lo2 = 0; 1096 1097 Expression x1 = e1; 1098 if (auto sle1 = x1.isSliceExp()) 1099 { 1100 lo1 = sle1.lwr.toInteger(); 1101 x1 = sle1.e1; 1102 } 1103 auto se1 = x1.isStringExp(); 1104 auto ae1 = x1.isArrayLiteralExp(); 1105 1106 Expression x2 = e2; 1107 if (auto sle2 = x2.isSliceExp()) 1108 { 1109 lo2 = sle2.lwr.toInteger(); 1110 x2 = sle2.e1; 1111 } 1112 auto se2 = x2.isStringExp(); 1113 auto ae2 = x2.isArrayLiteralExp(); 1114 1115 // Now both must be either TOK.arrayLiteral or TOK.string_ 1116 if (se1 && se2) 1117 return sliceCmpStringWithString(se1, se2, cast(size_t)lo1, cast(size_t)lo2, cast(size_t)len); 1118 if (se1 && ae2) 1119 return sliceCmpStringWithArray(se1, ae2, cast(size_t)lo1, cast(size_t)lo2, cast(size_t)len); 1120 if (se2 && ae1) 1121 return -sliceCmpStringWithArray(se2, ae1, cast(size_t)lo2, cast(size_t)lo1, cast(size_t)len); 1122 assert(ae1 && ae2); 1123 // Comparing two array literals. This case is potentially recursive. 1124 // If they aren't strings, we just need an equality check rather than 1125 // a full cmp. 1126 const bool needCmp = ae1.type.nextOf().isintegral(); 1127 foreach (size_t i; 0 .. cast(size_t)len) 1128 { 1129 Expression ee1 = (*ae1.elements)[cast(size_t)(lo1 + i)]; 1130 Expression ee2 = (*ae2.elements)[cast(size_t)(lo2 + i)]; 1131 if (needCmp) 1132 { 1133 const sinteger_t c = ee1.toInteger() - ee2.toInteger(); 1134 if (c > 0) 1135 return 1; 1136 if (c < 0) 1137 return -1; 1138 } 1139 else 1140 { 1141 if (ctfeRawCmp(loc, ee1, ee2)) 1142 return 1; 1143 } 1144 } 1145 return 0; 1146 } 1147 1148 /* Given a delegate expression e, return .funcptr. 1149 * If e is NullExp, return NULL. 1150 */ 1151 private FuncDeclaration funcptrOf(Expression e) 1152 { 1153 assert(e.type.ty == Tdelegate); 1154 if (auto de = e.isDelegateExp()) 1155 return de.func; 1156 if (auto fe = e.isFuncExp()) 1157 return fe.fd; 1158 assert(e.op == TOK.null_); 1159 return null; 1160 } 1161 1162 private bool isArray(const Expression e) 1163 { 1164 return e.op == TOK.arrayLiteral || e.op == TOK.string_ || e.op == TOK.slice || e.op == TOK.null_; 1165 } 1166 1167 /***** 1168 * Params: 1169 * loc = source file location 1170 * e1 = left operand 1171 * e2 = right operand 1172 * identity = true for `is` identity comparisons 1173 * Returns: 1174 * For strings, return <0 if e1 < e2, 0 if e1==e2, >0 if e1 > e2. 1175 * For all other types, return 0 if e1 == e2, !=0 if e1 != e2. 1176 */ 1177 private int ctfeRawCmp(const ref Loc loc, Expression e1, Expression e2, bool identity = false) 1178 { 1179 if (e1.op == TOK.classReference || e2.op == TOK.classReference) 1180 { 1181 if (e1.op == TOK.classReference && e2.op == TOK.classReference && 1182 (cast(ClassReferenceExp)e1).value == (cast(ClassReferenceExp)e2).value) 1183 return 0; 1184 return 1; 1185 } 1186 if (e1.op == TOK.typeid_ && e2.op == TOK.typeid_) 1187 { 1188 // printf("e1: %s\n", e1.toChars()); 1189 // printf("e2: %s\n", e2.toChars()); 1190 Type t1 = isType((cast(TypeidExp)e1).obj); 1191 Type t2 = isType((cast(TypeidExp)e2).obj); 1192 assert(t1); 1193 assert(t2); 1194 return t1 != t2; 1195 } 1196 // null == null, regardless of type 1197 if (e1.op == TOK.null_ && e2.op == TOK.null_) 1198 return 0; 1199 if (e1.type.ty == Tpointer && e2.type.ty == Tpointer) 1200 { 1201 // Can only be an equality test. 1202 dinteger_t ofs1, ofs2; 1203 Expression agg1 = getAggregateFromPointer(e1, &ofs1); 1204 Expression agg2 = getAggregateFromPointer(e2, &ofs2); 1205 if ((agg1 == agg2) || (agg1.op == TOK.variable && agg2.op == TOK.variable && (cast(VarExp)agg1).var == (cast(VarExp)agg2).var)) 1206 { 1207 if (ofs1 == ofs2) 1208 return 0; 1209 } 1210 return 1; 1211 } 1212 if (e1.type.ty == Tdelegate && e2.type.ty == Tdelegate) 1213 { 1214 // If .funcptr isn't the same, they are not equal 1215 if (funcptrOf(e1) != funcptrOf(e2)) 1216 return 1; 1217 // If both are delegate literals, assume they have the 1218 // same closure pointer. TODO: We don't support closures yet! 1219 if (e1.op == TOK.function_ && e2.op == TOK.function_) 1220 return 0; 1221 assert(e1.op == TOK.delegate_ && e2.op == TOK.delegate_); 1222 // Same .funcptr. Do they have the same .ptr? 1223 Expression ptr1 = (cast(DelegateExp)e1).e1; 1224 Expression ptr2 = (cast(DelegateExp)e2).e1; 1225 dinteger_t ofs1, ofs2; 1226 Expression agg1 = getAggregateFromPointer(ptr1, &ofs1); 1227 Expression agg2 = getAggregateFromPointer(ptr2, &ofs2); 1228 // If they are TOK.variable, it means they are FuncDeclarations 1229 if ((agg1 == agg2 && ofs1 == ofs2) || (agg1.op == TOK.variable && agg2.op == TOK.variable && (cast(VarExp)agg1).var == (cast(VarExp)agg2).var)) 1230 { 1231 return 0; 1232 } 1233 return 1; 1234 } 1235 if (isArray(e1) && isArray(e2)) 1236 { 1237 const uinteger_t len1 = resolveArrayLength(e1); 1238 const uinteger_t len2 = resolveArrayLength(e2); 1239 // workaround for dmc optimizer bug calculating wrong len for 1240 // uinteger_t len = (len1 < len2 ? len1 : len2); 1241 // if (len == 0) ... 1242 if (len1 > 0 && len2 > 0) 1243 { 1244 const uinteger_t len = (len1 < len2 ? len1 : len2); 1245 const int res = ctfeCmpArrays(loc, e1, e2, len); 1246 if (res != 0) 1247 return res; 1248 } 1249 return cast(int)(len1 - len2); 1250 } 1251 if (e1.type.isintegral()) 1252 { 1253 return e1.toInteger() != e2.toInteger(); 1254 } 1255 if (e1.type.isreal() || e1.type.isimaginary()) 1256 { 1257 real_t r1 = e1.type.isreal() ? e1.toReal() : e1.toImaginary(); 1258 real_t r2 = e1.type.isreal() ? e2.toReal() : e2.toImaginary(); 1259 if (identity) 1260 return !RealIdentical(r1, r2); 1261 if (CTFloat.isNaN(r1) || CTFloat.isNaN(r2)) // if unordered 1262 { 1263 return 1; // they are not equal 1264 } 1265 else 1266 { 1267 return (r1 != r2); 1268 } 1269 } 1270 else if (e1.type.iscomplex()) 1271 { 1272 auto c1 = e1.toComplex(); 1273 auto c2 = e2.toComplex(); 1274 if (identity) 1275 { 1276 return !RealIdentical(c1.re, c2.re) && !RealIdentical(c1.im, c2.im); 1277 } 1278 return c1 != c2; 1279 } 1280 if (e1.op == TOK.structLiteral && e2.op == TOK.structLiteral) 1281 { 1282 StructLiteralExp es1 = cast(StructLiteralExp)e1; 1283 StructLiteralExp es2 = cast(StructLiteralExp)e2; 1284 // For structs, we only need to return 0 or 1 (< and > aren't legal). 1285 if (es1.sd != es2.sd) 1286 return 1; 1287 else if ((!es1.elements || !es1.elements.dim) && (!es2.elements || !es2.elements.dim)) 1288 return 0; // both arrays are empty 1289 else if (!es1.elements || !es2.elements) 1290 return 1; 1291 else if (es1.elements.dim != es2.elements.dim) 1292 return 1; 1293 else 1294 { 1295 foreach (size_t i; 0 .. es1.elements.dim) 1296 { 1297 Expression ee1 = (*es1.elements)[i]; 1298 Expression ee2 = (*es2.elements)[i]; 1299 1300 // https://issues.dlang.org/show_bug.cgi?id=16284 1301 if (ee1.op == TOK.void_ && ee2.op == TOK.void_) // if both are VoidInitExp 1302 continue; 1303 1304 if (ee1 == ee2) 1305 continue; 1306 if (!ee1 || !ee2) 1307 return 1; 1308 const int cmp = ctfeRawCmp(loc, ee1, ee2, identity); 1309 if (cmp) 1310 return 1; 1311 } 1312 return 0; // All elements are equal 1313 } 1314 } 1315 if (e1.op == TOK.assocArrayLiteral && e2.op == TOK.assocArrayLiteral) 1316 { 1317 AssocArrayLiteralExp es1 = cast(AssocArrayLiteralExp)e1; 1318 AssocArrayLiteralExp es2 = cast(AssocArrayLiteralExp)e2; 1319 size_t dim = es1.keys.dim; 1320 if (es2.keys.dim != dim) 1321 return 1; 1322 bool* used = cast(bool*)mem.xmalloc(bool.sizeof * dim); 1323 memset(used, 0, bool.sizeof * dim); 1324 foreach (size_t i; 0 .. dim) 1325 { 1326 Expression k1 = (*es1.keys)[i]; 1327 Expression v1 = (*es1.values)[i]; 1328 Expression v2 = null; 1329 foreach (size_t j; 0 .. dim) 1330 { 1331 if (used[j]) 1332 continue; 1333 Expression k2 = (*es2.keys)[j]; 1334 if (ctfeRawCmp(loc, k1, k2, identity)) 1335 continue; 1336 used[j] = true; 1337 v2 = (*es2.values)[j]; 1338 break; 1339 } 1340 if (!v2 || ctfeRawCmp(loc, v1, v2, identity)) 1341 { 1342 mem.xfree(used); 1343 return 1; 1344 } 1345 } 1346 mem.xfree(used); 1347 return 0; 1348 } 1349 error(loc, "CTFE internal error: bad compare of `%s` and `%s`", e1.toChars(), e2.toChars()); 1350 assert(0); 1351 } 1352 1353 /// Evaluate ==, !=. Resolves slices before comparing. Returns 0 or 1 1354 bool ctfeEqual(const ref Loc loc, TOK op, Expression e1, Expression e2) 1355 { 1356 return !ctfeRawCmp(loc, e1, e2) ^ (op == TOK.notEqual); 1357 } 1358 1359 /// Evaluate is, !is. Resolves slices before comparing. Returns 0 or 1 1360 bool ctfeIdentity(const ref Loc loc, TOK op, Expression e1, Expression e2) 1361 { 1362 //printf("ctfeIdentity %s %s\n", e1.toChars(), e2.toChars()); 1363 //printf("ctfeIdentity op = '%s', e1 = %s %s, e2 = %s %s\n", Token::toChars(op), 1364 // Token::toChars(e1.op), e1.toChars(), Token::toChars(e2.op), e1.toChars()); 1365 bool cmp; 1366 if (e1.op == TOK.null_) 1367 { 1368 cmp = (e2.op == TOK.null_); 1369 } 1370 else if (e2.op == TOK.null_) 1371 { 1372 cmp = false; 1373 } 1374 else if (e1.op == TOK.symbolOffset && e2.op == TOK.symbolOffset) 1375 { 1376 SymOffExp es1 = cast(SymOffExp)e1; 1377 SymOffExp es2 = cast(SymOffExp)e2; 1378 cmp = (es1.var == es2.var && es1.offset == es2.offset); 1379 } 1380 else if (e1.type.isreal()) 1381 cmp = RealIdentical(e1.toReal(), e2.toReal()); 1382 else if (e1.type.isimaginary()) 1383 cmp = RealIdentical(e1.toImaginary(), e2.toImaginary()); 1384 else if (e1.type.iscomplex()) 1385 { 1386 complex_t v1 = e1.toComplex(); 1387 complex_t v2 = e2.toComplex(); 1388 cmp = RealIdentical(creall(v1), creall(v2)) && RealIdentical(cimagl(v1), cimagl(v1)); 1389 } 1390 else 1391 { 1392 cmp = !ctfeRawCmp(loc, e1, e2, true); 1393 } 1394 if (op == TOK.notIdentity || op == TOK.notEqual) 1395 cmp ^= true; 1396 return cmp; 1397 } 1398 1399 /// Evaluate >,<=, etc. Resolves slices before comparing. Returns 0 or 1 1400 bool ctfeCmp(const ref Loc loc, TOK op, Expression e1, Expression e2) 1401 { 1402 Type t1 = e1.type.toBasetype(); 1403 Type t2 = e2.type.toBasetype(); 1404 1405 if (t1.isString() && t2.isString()) 1406 return specificCmp(op, ctfeRawCmp(loc, e1, e2)); 1407 else if (t1.isreal()) 1408 return realCmp(op, e1.toReal(), e2.toReal()); 1409 else if (t1.isimaginary()) 1410 return realCmp(op, e1.toImaginary(), e2.toImaginary()); 1411 else if (t1.isunsigned() || t2.isunsigned()) 1412 return intUnsignedCmp(op, e1.toInteger(), e2.toInteger()); 1413 else 1414 return intSignedCmp(op, e1.toInteger(), e2.toInteger()); 1415 } 1416 1417 UnionExp ctfeCat(const ref Loc loc, Type type, Expression e1, Expression e2) 1418 { 1419 Type t1 = e1.type.toBasetype(); 1420 Type t2 = e2.type.toBasetype(); 1421 UnionExp ue; 1422 if (e2.op == TOK.string_ && e1.op == TOK.arrayLiteral && t1.nextOf().isintegral()) 1423 { 1424 // [chars] ~ string => string (only valid for CTFE) 1425 StringExp es1 = cast(StringExp)e2; 1426 ArrayLiteralExp es2 = cast(ArrayLiteralExp)e1; 1427 const len = es1.len + es2.elements.dim; 1428 const sz = es1.sz; 1429 void* s = mem.xmalloc((len + 1) * sz); 1430 const data1 = es1.peekData(); 1431 memcpy(cast(char*)s + sz * es2.elements.dim, data1.ptr, data1.length); 1432 foreach (size_t i; 0 .. es2.elements.dim) 1433 { 1434 Expression es2e = (*es2.elements)[i]; 1435 if (es2e.op != TOK.int64) 1436 { 1437 emplaceExp!(CTFEExp)(&ue, TOK.cantExpression); 1438 return ue; 1439 } 1440 dinteger_t v = es2e.toInteger(); 1441 Port.valcpy(cast(char*)s + i * sz, v, sz); 1442 } 1443 // Add terminating 0 1444 memset(cast(char*)s + len * sz, 0, sz); 1445 emplaceExp!(StringExp)(&ue, loc, s[0 .. len * sz], len, sz); 1446 StringExp es = cast(StringExp)ue.exp(); 1447 es.committed = 0; 1448 es.type = type; 1449 return ue; 1450 } 1451 if (e1.op == TOK.string_ && e2.op == TOK.arrayLiteral && t2.nextOf().isintegral()) 1452 { 1453 // string ~ [chars] => string (only valid for CTFE) 1454 // Concatenate the strings 1455 StringExp es1 = cast(StringExp)e1; 1456 ArrayLiteralExp es2 = cast(ArrayLiteralExp)e2; 1457 const len = es1.len + es2.elements.dim; 1458 const sz = es1.sz; 1459 void* s = mem.xmalloc((len + 1) * sz); 1460 auto slice = es1.peekData(); 1461 memcpy(s, slice.ptr, slice.length); 1462 foreach (size_t i; 0 .. es2.elements.dim) 1463 { 1464 Expression es2e = (*es2.elements)[i]; 1465 if (es2e.op != TOK.int64) 1466 { 1467 emplaceExp!(CTFEExp)(&ue, TOK.cantExpression); 1468 return ue; 1469 } 1470 const v = es2e.toInteger(); 1471 Port.valcpy(cast(char*)s + (es1.len + i) * sz, v, sz); 1472 } 1473 // Add terminating 0 1474 memset(cast(char*)s + len * sz, 0, sz); 1475 emplaceExp!(StringExp)(&ue, loc, s[0 .. len * sz], len, sz); 1476 StringExp es = cast(StringExp)ue.exp(); 1477 es.sz = sz; 1478 es.committed = 0; //es1.committed; 1479 es.type = type; 1480 return ue; 1481 } 1482 if (e1.op == TOK.arrayLiteral && e2.op == TOK.arrayLiteral && t1.nextOf().equals(t2.nextOf())) 1483 { 1484 // [ e1 ] ~ [ e2 ] ---> [ e1, e2 ] 1485 ArrayLiteralExp es1 = cast(ArrayLiteralExp)e1; 1486 ArrayLiteralExp es2 = cast(ArrayLiteralExp)e2; 1487 emplaceExp!(ArrayLiteralExp)(&ue, es1.loc, type, copyLiteralArray(es1.elements)); 1488 es1 = cast(ArrayLiteralExp)ue.exp(); 1489 es1.elements.insert(es1.elements.dim, copyLiteralArray(es2.elements)); 1490 return ue; 1491 } 1492 if (e1.op == TOK.arrayLiteral && e2.op == TOK.null_ && t1.nextOf().equals(t2.nextOf())) 1493 { 1494 // [ e1 ] ~ null ----> [ e1 ].dup 1495 ue = paintTypeOntoLiteralCopy(type, copyLiteral(e1).copy()); 1496 return ue; 1497 } 1498 if (e1.op == TOK.null_ && e2.op == TOK.arrayLiteral && t1.nextOf().equals(t2.nextOf())) 1499 { 1500 // null ~ [ e2 ] ----> [ e2 ].dup 1501 ue = paintTypeOntoLiteralCopy(type, copyLiteral(e2).copy()); 1502 return ue; 1503 } 1504 ue = Cat(type, e1, e2); 1505 return ue; 1506 } 1507 1508 /* Given an AA literal 'ae', and a key 'e2': 1509 * Return ae[e2] if present, or NULL if not found. 1510 */ 1511 Expression findKeyInAA(const ref Loc loc, AssocArrayLiteralExp ae, Expression e2) 1512 { 1513 /* Search the keys backwards, in case there are duplicate keys 1514 */ 1515 for (size_t i = ae.keys.dim; i;) 1516 { 1517 --i; 1518 Expression ekey = (*ae.keys)[i]; 1519 const int eq = ctfeEqual(loc, TOK.equal, ekey, e2); 1520 if (eq) 1521 { 1522 return (*ae.values)[i]; 1523 } 1524 } 1525 return null; 1526 } 1527 1528 /* Same as for constfold.Index, except that it only works for static arrays, 1529 * dynamic arrays, and strings. We know that e1 is an 1530 * interpreted CTFE expression, so it cannot have side-effects. 1531 */ 1532 Expression ctfeIndex(UnionExp* pue, const ref Loc loc, Type type, Expression e1, uinteger_t indx) 1533 { 1534 //printf("ctfeIndex(e1 = %s)\n", e1.toChars()); 1535 assert(e1.type); 1536 if (auto es1 = e1.isStringExp()) 1537 { 1538 if (indx >= es1.len) 1539 { 1540 error(loc, "string index %llu is out of bounds `[0 .. %zu]`", indx, es1.len); 1541 return CTFEExp.cantexp; 1542 } 1543 emplaceExp!IntegerExp(pue, loc, es1.charAt(indx), type); 1544 return pue.exp(); 1545 } 1546 1547 if (auto ale = e1.isArrayLiteralExp()) 1548 { 1549 if (indx >= ale.elements.dim) 1550 { 1551 error(loc, "array index %llu is out of bounds `%s[0 .. %zu]`", indx, e1.toChars(), ale.elements.dim); 1552 return CTFEExp.cantexp; 1553 } 1554 Expression e = (*ale.elements)[cast(size_t)indx]; 1555 return paintTypeOntoLiteral(pue, type, e); 1556 } 1557 1558 assert(0); 1559 } 1560 1561 Expression ctfeCast(UnionExp* pue, const ref Loc loc, Type type, Type to, Expression e) 1562 { 1563 Expression paint() 1564 { 1565 return paintTypeOntoLiteral(pue, to, e); 1566 } 1567 1568 if (e.op == TOK.null_) 1569 return paint(); 1570 1571 if (e.op == TOK.classReference) 1572 { 1573 // Disallow reinterpreting class casts. Do this by ensuring that 1574 // the original class can implicitly convert to the target class 1575 ClassDeclaration originalClass = (cast(ClassReferenceExp)e).originalClass(); 1576 if (originalClass.type.implicitConvTo(to.mutableOf())) 1577 return paint(); 1578 else 1579 { 1580 emplaceExp!(NullExp)(pue, loc, to); 1581 return pue.exp(); 1582 } 1583 } 1584 1585 // Allow TypeInfo type painting 1586 if (isTypeInfo_Class(e.type) && e.type.implicitConvTo(to)) 1587 return paint(); 1588 1589 // Allow casting away const for struct literals 1590 if (e.op == TOK.structLiteral && e.type.toBasetype().castMod(0) == to.toBasetype().castMod(0)) 1591 return paint(); 1592 1593 Expression r; 1594 if (e.type.equals(type) && type.equals(to)) 1595 { 1596 // necessary not to change e's address for pointer comparisons 1597 r = e; 1598 } 1599 else if (to.toBasetype().ty == Tarray && 1600 type.toBasetype().ty == Tarray && 1601 to.toBasetype().nextOf().size() == type.toBasetype().nextOf().size()) 1602 { 1603 // https://issues.dlang.org/show_bug.cgi?id=12495 1604 // Array reinterpret casts: eg. string to immutable(ubyte)[] 1605 return paint(); 1606 } 1607 else 1608 { 1609 *pue = Cast(loc, type, to, e); 1610 r = pue.exp(); 1611 } 1612 1613 if (CTFEExp.isCantExp(r)) 1614 error(loc, "cannot cast `%s` to `%s` at compile time", e.toChars(), to.toChars()); 1615 1616 if (auto ae = e.isArrayLiteralExp()) 1617 ae.ownedByCtfe = OwnedBy.ctfe; 1618 1619 if (auto se = e.isStringExp()) 1620 se.ownedByCtfe = OwnedBy.ctfe; 1621 1622 return r; 1623 } 1624 1625 /******** Assignment helper functions ***************************/ 1626 /* Set dest = src, where both dest and src are container value literals 1627 * (ie, struct literals, or static arrays (can be an array literal or a string)) 1628 * Assignment is recursively in-place. 1629 * Purpose: any reference to a member of 'dest' will remain valid after the 1630 * assignment. 1631 */ 1632 void assignInPlace(Expression dest, Expression src) 1633 { 1634 if (!(dest.op == TOK.structLiteral || dest.op == TOK.arrayLiteral || dest.op == TOK.string_)) 1635 { 1636 printf("invalid op %d %d\n", src.op, dest.op); 1637 assert(0); 1638 } 1639 Expressions* oldelems; 1640 Expressions* newelems; 1641 if (dest.op == TOK.structLiteral) 1642 { 1643 assert(dest.op == src.op); 1644 oldelems = (cast(StructLiteralExp)dest).elements; 1645 newelems = (cast(StructLiteralExp)src).elements; 1646 auto sd = (cast(StructLiteralExp)dest).sd; 1647 const nfields = sd.nonHiddenFields(); 1648 const nvthis = sd.fields.dim - nfields; 1649 if (nvthis && oldelems.dim >= nfields && oldelems.dim < newelems.dim) 1650 foreach (_; 0 .. newelems.dim - oldelems.dim) 1651 oldelems.push(null); 1652 } 1653 else if (dest.op == TOK.arrayLiteral && src.op == TOK.arrayLiteral) 1654 { 1655 oldelems = (cast(ArrayLiteralExp)dest).elements; 1656 newelems = (cast(ArrayLiteralExp)src).elements; 1657 } 1658 else if (dest.op == TOK.string_ && src.op == TOK.string_) 1659 { 1660 sliceAssignStringFromString(cast(StringExp)dest, cast(StringExp)src, 0); 1661 return; 1662 } 1663 else if (dest.op == TOK.arrayLiteral && src.op == TOK.string_) 1664 { 1665 sliceAssignArrayLiteralFromString(cast(ArrayLiteralExp)dest, cast(StringExp)src, 0); 1666 return; 1667 } 1668 else if (src.op == TOK.arrayLiteral && dest.op == TOK.string_) 1669 { 1670 sliceAssignStringFromArrayLiteral(cast(StringExp)dest, cast(ArrayLiteralExp)src, 0); 1671 return; 1672 } 1673 else 1674 { 1675 printf("invalid op %d %d\n", src.op, dest.op); 1676 assert(0); 1677 } 1678 assert(oldelems.dim == newelems.dim); 1679 foreach (size_t i; 0 .. oldelems.dim) 1680 { 1681 Expression e = (*newelems)[i]; 1682 Expression o = (*oldelems)[i]; 1683 if (e.op == TOK.structLiteral) 1684 { 1685 assert(o.op == e.op); 1686 assignInPlace(o, e); 1687 } 1688 else if (e.type.ty == Tsarray && e.op != TOK.void_ && o.type.ty == Tsarray) 1689 { 1690 assignInPlace(o, e); 1691 } 1692 else 1693 { 1694 (*oldelems)[i] = (*newelems)[i]; 1695 } 1696 } 1697 } 1698 1699 // Given an AA literal aae, set aae[index] = newval and return newval. 1700 Expression assignAssocArrayElement(const ref Loc loc, AssocArrayLiteralExp aae, Expression index, Expression newval) 1701 { 1702 /* Create new associative array literal reflecting updated key/value 1703 */ 1704 Expressions* keysx = aae.keys; 1705 Expressions* valuesx = aae.values; 1706 int updated = 0; 1707 for (size_t j = valuesx.dim; j;) 1708 { 1709 j--; 1710 Expression ekey = (*aae.keys)[j]; 1711 int eq = ctfeEqual(loc, TOK.equal, ekey, index); 1712 if (eq) 1713 { 1714 (*valuesx)[j] = newval; 1715 updated = 1; 1716 } 1717 } 1718 if (!updated) 1719 { 1720 // Append index/newval to keysx[]/valuesx[] 1721 valuesx.push(newval); 1722 keysx.push(index); 1723 } 1724 return newval; 1725 } 1726 1727 /// Given array literal oldval of type ArrayLiteralExp or StringExp, of length 1728 /// oldlen, change its length to newlen. If the newlen is longer than oldlen, 1729 /// all new elements will be set to the default initializer for the element type. 1730 UnionExp changeArrayLiteralLength(const ref Loc loc, TypeArray arrayType, Expression oldval, size_t oldlen, size_t newlen) 1731 { 1732 UnionExp ue; 1733 Type elemType = arrayType.next; 1734 assert(elemType); 1735 Expression defaultElem = elemType.defaultInitLiteral(loc); 1736 auto elements = new Expressions(newlen); 1737 // Resolve slices 1738 size_t indxlo = 0; 1739 if (oldval.op == TOK.slice) 1740 { 1741 indxlo = cast(size_t)(cast(SliceExp)oldval).lwr.toInteger(); 1742 oldval = (cast(SliceExp)oldval).e1; 1743 } 1744 size_t copylen = oldlen < newlen ? oldlen : newlen; 1745 if (oldval.op == TOK.string_) 1746 { 1747 StringExp oldse = cast(StringExp)oldval; 1748 void* s = mem.xcalloc(newlen + 1, oldse.sz); 1749 const data = oldse.peekData(); 1750 memcpy(s, data.ptr, copylen * oldse.sz); 1751 const defaultValue = cast(uint)defaultElem.toInteger(); 1752 foreach (size_t elemi; copylen .. newlen) 1753 { 1754 switch (oldse.sz) 1755 { 1756 case 1: 1757 (cast(char*)s)[cast(size_t)(indxlo + elemi)] = cast(char)defaultValue; 1758 break; 1759 case 2: 1760 (cast(wchar*)s)[cast(size_t)(indxlo + elemi)] = cast(wchar)defaultValue; 1761 break; 1762 case 4: 1763 (cast(dchar*)s)[cast(size_t)(indxlo + elemi)] = cast(dchar)defaultValue; 1764 break; 1765 default: 1766 assert(0); 1767 } 1768 } 1769 emplaceExp!(StringExp)(&ue, loc, s[0 .. newlen * oldse.sz], newlen, oldse.sz); 1770 StringExp se = cast(StringExp)ue.exp(); 1771 se.type = arrayType; 1772 se.sz = oldse.sz; 1773 se.committed = oldse.committed; 1774 se.ownedByCtfe = OwnedBy.ctfe; 1775 } 1776 else 1777 { 1778 if (oldlen != 0) 1779 { 1780 assert(oldval.op == TOK.arrayLiteral); 1781 ArrayLiteralExp ae = cast(ArrayLiteralExp)oldval; 1782 foreach (size_t i; 0 .. copylen) 1783 (*elements)[i] = (*ae.elements)[indxlo + i]; 1784 } 1785 if (elemType.ty == Tstruct || elemType.ty == Tsarray) 1786 { 1787 /* If it is an aggregate literal representing a value type, 1788 * we need to create a unique copy for each element 1789 */ 1790 foreach (size_t i; copylen .. newlen) 1791 (*elements)[i] = copyLiteral(defaultElem).copy(); 1792 } 1793 else 1794 { 1795 foreach (size_t i; copylen .. newlen) 1796 (*elements)[i] = defaultElem; 1797 } 1798 emplaceExp!(ArrayLiteralExp)(&ue, loc, arrayType, elements); 1799 ArrayLiteralExp aae = cast(ArrayLiteralExp)ue.exp(); 1800 aae.ownedByCtfe = OwnedBy.ctfe; 1801 } 1802 return ue; 1803 } 1804 1805 /*************************** CTFE Sanity Checks ***************************/ 1806 1807 bool isCtfeValueValid(Expression newval) 1808 { 1809 Type tb = newval.type.toBasetype(); 1810 switch (newval.op) 1811 { 1812 case TOK.int64: 1813 case TOK.float64: 1814 case TOK.char_: 1815 case TOK.complex80: 1816 return tb.isscalar(); 1817 1818 case TOK.null_: 1819 return tb.ty == Tnull || 1820 tb.ty == Tpointer || 1821 tb.ty == Tarray || 1822 tb.ty == Taarray || 1823 tb.ty == Tclass || 1824 tb.ty == Tdelegate; 1825 1826 case TOK.string_: 1827 return true; // CTFE would directly use the StringExp in AST. 1828 1829 case TOK.arrayLiteral: 1830 return true; //((ArrayLiteralExp *)newval)->ownedByCtfe; 1831 1832 case TOK.assocArrayLiteral: 1833 return true; //((AssocArrayLiteralExp *)newval)->ownedByCtfe; 1834 1835 case TOK.structLiteral: 1836 return true; //((StructLiteralExp *)newval)->ownedByCtfe; 1837 1838 case TOK.classReference: 1839 return true; 1840 1841 case TOK.type: 1842 return true; 1843 1844 case TOK.vector: 1845 return true; // vector literal 1846 1847 case TOK.function_: 1848 return true; // function literal or delegate literal 1849 1850 case TOK.delegate_: 1851 { 1852 // &struct.func or &clasinst.func 1853 // &nestedfunc 1854 Expression ethis = (cast(DelegateExp)newval).e1; 1855 return (ethis.op == TOK.structLiteral || ethis.op == TOK.classReference || ethis.op == TOK.variable && (cast(VarExp)ethis).var == (cast(DelegateExp)newval).func); 1856 } 1857 1858 case TOK.symbolOffset: 1859 { 1860 // function pointer, or pointer to static variable 1861 Declaration d = (cast(SymOffExp)newval).var; 1862 return d.isFuncDeclaration() || d.isDataseg(); 1863 } 1864 1865 case TOK.typeid_: 1866 { 1867 // always valid 1868 return true; 1869 } 1870 1871 case TOK.address: 1872 { 1873 // e1 should be a CTFE reference 1874 Expression e1 = (cast(AddrExp)newval).e1; 1875 return tb.ty == Tpointer && 1876 ( 1877 (e1.op == TOK.structLiteral || e1.op == TOK.arrayLiteral) && isCtfeValueValid(e1) || 1878 e1.op == TOK.variable || 1879 e1.op == TOK.dotVariable && isCtfeReferenceValid(e1) || 1880 e1.op == TOK.index && isCtfeReferenceValid(e1) || 1881 e1.op == TOK.slice && e1.type.toBasetype().ty == Tsarray 1882 ); 1883 } 1884 1885 case TOK.slice: 1886 { 1887 // e1 should be an array aggregate 1888 const SliceExp se = cast(SliceExp)newval; 1889 assert(se.lwr && se.lwr.op == TOK.int64); 1890 assert(se.upr && se.upr.op == TOK.int64); 1891 return (tb.ty == Tarray || tb.ty == Tsarray) && (se.e1.op == TOK.string_ || se.e1.op == TOK.arrayLiteral); 1892 } 1893 1894 case TOK.void_: 1895 return true; // uninitialized value 1896 1897 default: 1898 newval.error("CTFE internal error: illegal CTFE value `%s`", newval.toChars()); 1899 return false; 1900 } 1901 } 1902 1903 bool isCtfeReferenceValid(Expression newval) 1904 { 1905 switch (newval.op) 1906 { 1907 case TOK.this_: 1908 return true; 1909 1910 case TOK.variable: 1911 { 1912 const VarDeclaration v = (cast(VarExp)newval).var.isVarDeclaration(); 1913 assert(v); 1914 // Must not be a reference to a reference 1915 return true; 1916 } 1917 1918 case TOK.index: 1919 { 1920 const Expression eagg = (cast(IndexExp)newval).e1; 1921 return eagg.op == TOK.string_ || eagg.op == TOK.arrayLiteral || eagg.op == TOK.assocArrayLiteral; 1922 } 1923 1924 case TOK.dotVariable: 1925 { 1926 Expression eagg = (cast(DotVarExp)newval).e1; 1927 return (eagg.op == TOK.structLiteral || eagg.op == TOK.classReference) && isCtfeValueValid(eagg); 1928 } 1929 1930 default: 1931 // Internally a ref variable may directly point a stack memory. 1932 // e.g. ref int v = 1; 1933 return isCtfeValueValid(newval); 1934 } 1935 } 1936 1937 // Used for debugging only 1938 void showCtfeExpr(Expression e, int level = 0) 1939 { 1940 for (int i = level; i > 0; --i) 1941 printf(" "); 1942 Expressions* elements = null; 1943 // We need the struct definition to detect block assignment 1944 StructDeclaration sd = null; 1945 ClassDeclaration cd = null; 1946 if (e.op == TOK.structLiteral) 1947 { 1948 elements = (cast(StructLiteralExp)e).elements; 1949 sd = (cast(StructLiteralExp)e).sd; 1950 printf("STRUCT type = %s %p:\n", e.type.toChars(), e); 1951 } 1952 else if (e.op == TOK.classReference) 1953 { 1954 elements = (cast(ClassReferenceExp)e).value.elements; 1955 cd = (cast(ClassReferenceExp)e).originalClass(); 1956 printf("CLASS type = %s %p:\n", e.type.toChars(), (cast(ClassReferenceExp)e).value); 1957 } 1958 else if (e.op == TOK.arrayLiteral) 1959 { 1960 elements = (cast(ArrayLiteralExp)e).elements; 1961 printf("ARRAY LITERAL type=%s %p:\n", e.type.toChars(), e); 1962 } 1963 else if (e.op == TOK.assocArrayLiteral) 1964 { 1965 printf("AA LITERAL type=%s %p:\n", e.type.toChars(), e); 1966 } 1967 else if (e.op == TOK.string_) 1968 { 1969 printf("STRING %s %p\n", e.toChars(), e.isStringExp.peekString.ptr); 1970 } 1971 else if (e.op == TOK.slice) 1972 { 1973 printf("SLICE %p: %s\n", e, e.toChars()); 1974 showCtfeExpr((cast(SliceExp)e).e1, level + 1); 1975 } 1976 else if (e.op == TOK.variable) 1977 { 1978 printf("VAR %p %s\n", e, e.toChars()); 1979 VarDeclaration v = (cast(VarExp)e).var.isVarDeclaration(); 1980 if (v && getValue(v)) 1981 showCtfeExpr(getValue(v), level + 1); 1982 } 1983 else if (e.op == TOK.address) 1984 { 1985 // This is potentially recursive. We mustn't try to print the thing we're pointing to. 1986 printf("POINTER %p to %p: %s\n", e, (cast(AddrExp)e).e1, e.toChars()); 1987 } 1988 else 1989 printf("VALUE %p: %s\n", e, e.toChars()); 1990 if (elements) 1991 { 1992 size_t fieldsSoFar = 0; 1993 for (size_t i = 0; i < elements.dim; i++) 1994 { 1995 Expression z = null; 1996 VarDeclaration v = null; 1997 if (i > 15) 1998 { 1999 printf("...(total %d elements)\n", cast(int)elements.dim); 2000 return; 2001 } 2002 if (sd) 2003 { 2004 v = sd.fields[i]; 2005 z = (*elements)[i]; 2006 } 2007 else if (cd) 2008 { 2009 while (i - fieldsSoFar >= cd.fields.dim) 2010 { 2011 fieldsSoFar += cd.fields.dim; 2012 cd = cd.baseClass; 2013 for (int j = level; j > 0; --j) 2014 printf(" "); 2015 printf(" BASE CLASS: %s\n", cd.toChars()); 2016 } 2017 v = cd.fields[i - fieldsSoFar]; 2018 assert((elements.dim + i) >= (fieldsSoFar + cd.fields.dim)); 2019 size_t indx = (elements.dim - fieldsSoFar) - cd.fields.dim + i; 2020 assert(indx < elements.dim); 2021 z = (*elements)[indx]; 2022 } 2023 if (!z) 2024 { 2025 for (int j = level; j > 0; --j) 2026 printf(" "); 2027 printf(" void\n"); 2028 continue; 2029 } 2030 if (v) 2031 { 2032 // If it is a void assignment, use the default initializer 2033 if ((v.type.ty != z.type.ty) && v.type.ty == Tsarray) 2034 { 2035 for (int j = level; --j;) 2036 printf(" "); 2037 printf(" field: block initialized static array\n"); 2038 continue; 2039 } 2040 } 2041 showCtfeExpr(z, level + 1); 2042 } 2043 } 2044 } 2045 2046 /*************************** Void initialization ***************************/ 2047 UnionExp voidInitLiteral(Type t, VarDeclaration var) 2048 { 2049 UnionExp ue; 2050 if (t.ty == Tsarray) 2051 { 2052 TypeSArray tsa = cast(TypeSArray)t; 2053 Expression elem = voidInitLiteral(tsa.next, var).copy(); 2054 // For aggregate value types (structs, static arrays) we must 2055 // create an a separate copy for each element. 2056 const mustCopy = (elem.op == TOK.arrayLiteral || elem.op == TOK.structLiteral); 2057 const d = cast(size_t)tsa.dim.toInteger(); 2058 auto elements = new Expressions(d); 2059 foreach (i; 0 .. d) 2060 { 2061 if (mustCopy && i > 0) 2062 elem = copyLiteral(elem).copy(); 2063 (*elements)[i] = elem; 2064 } 2065 emplaceExp!(ArrayLiteralExp)(&ue, var.loc, tsa, elements); 2066 ArrayLiteralExp ae = cast(ArrayLiteralExp)ue.exp(); 2067 ae.ownedByCtfe = OwnedBy.ctfe; 2068 } 2069 else if (t.ty == Tstruct) 2070 { 2071 TypeStruct ts = cast(TypeStruct)t; 2072 auto exps = new Expressions(ts.sym.fields.dim); 2073 foreach (size_t i; 0 .. ts.sym.fields.dim) 2074 { 2075 (*exps)[i] = voidInitLiteral(ts.sym.fields[i].type, ts.sym.fields[i]).copy(); 2076 } 2077 emplaceExp!(StructLiteralExp)(&ue, var.loc, ts.sym, exps); 2078 StructLiteralExp se = cast(StructLiteralExp)ue.exp(); 2079 se.type = ts; 2080 se.ownedByCtfe = OwnedBy.ctfe; 2081 } 2082 else 2083 emplaceExp!(VoidInitExp)(&ue, var); 2084 return ue; 2085 }