1 /** 2 * Checks whether member access or array casting is allowed in `@safe` code. 3 * 4 * Specification: $(LINK2 https://dlang.org/spec/function.html#function-safety, Function Safety) 5 * 6 * Copyright: Copyright (C) 1999-2021 by The D Language Foundation, All Rights Reserved 7 * Authors: $(LINK2 http://www.digitalmars.com, Walter Bright) 8 * License: $(LINK2 http://www.boost.org/LICENSE_1_0.txt, Boost License 1.0) 9 * Source: $(LINK2 https://github.com/dlang/dmd/blob/master/src/dmd/safe.d, _safe.d) 10 * Documentation: https://dlang.org/phobos/dmd_safe.html 11 * Coverage: https://codecov.io/gh/dlang/dmd/src/master/src/dmd/safe.d 12 */ 13 14 module dmd.safe; 15 16 import core.stdc.stdio; 17 18 import dmd.aggregate; 19 import dmd.dclass; 20 import dmd.declaration; 21 import dmd.dscope; 22 import dmd.expression; 23 import dmd.mtype; 24 import dmd.target; 25 import dmd.tokens; 26 27 28 /************************************************************* 29 * Check for unsafe access in @safe code: 30 * 1. read overlapped pointers 31 * 2. write misaligned pointers 32 * 3. write overlapped storage classes 33 * Print error if unsafe. 34 * Params: 35 * sc = scope 36 * e = expression to check 37 * readonly = if access is read-only 38 * printmsg = print error message if true 39 * Returns: 40 * true if error 41 */ 42 43 bool checkUnsafeAccess(Scope* sc, Expression e, bool readonly, bool printmsg) 44 { 45 //printf("checkUnsafeAccess(e: '%s', readonly: %d, printmsg: %d)\n", e.toChars(), readonly, printmsg); 46 if (e.op != TOK.dotVariable) 47 return false; 48 DotVarExp dve = cast(DotVarExp)e; 49 if (VarDeclaration v = dve.var.isVarDeclaration()) 50 { 51 if (sc.intypeof || !sc.func || !sc.func.isSafeBypassingInference()) 52 return false; 53 auto ad = v.toParent2().isAggregateDeclaration(); 54 if (!ad) 55 return false; 56 57 // needed to set v.overlapped and v.overlapUnsafe 58 if (ad.sizeok != Sizeok.done) 59 ad.determineSize(ad.loc); 60 61 const hasPointers = v.type.hasPointers(); 62 if (hasPointers) 63 { 64 65 if (v.overlapped && sc.func.setUnsafe()) 66 { 67 if (printmsg) 68 e.error("field `%s.%s` cannot access pointers in `@safe` code that overlap other fields", 69 ad.toChars(), v.toChars()); 70 return true; 71 } 72 } 73 74 if (readonly || !e.type.isMutable()) 75 return false; 76 77 if (hasPointers && v.type.toBasetype().ty != Tstruct) 78 { 79 if ((ad.type.alignment() < target.ptrsize || 80 (v.offset & (target.ptrsize - 1))) && 81 sc.func.setUnsafe()) 82 { 83 if (printmsg) 84 e.error("field `%s.%s` cannot modify misaligned pointers in `@safe` code", 85 ad.toChars(), v.toChars()); 86 return true; 87 } 88 } 89 90 if (v.overlapUnsafe && sc.func.setUnsafe()) 91 { 92 if (printmsg) 93 e.error("field `%s.%s` cannot modify fields in `@safe` code that overlap fields with other storage classes", 94 ad.toChars(), v.toChars()); 95 return true; 96 } 97 } 98 return false; 99 } 100 101 102 /********************************************** 103 * Determine if it is @safe to cast e from tfrom to tto. 104 * Params: 105 * e = expression to be cast 106 * tfrom = type of e 107 * tto = type to cast e to 108 * Returns: 109 * true if @safe 110 */ 111 bool isSafeCast(Expression e, Type tfrom, Type tto) 112 { 113 // Implicit conversions are always safe 114 if (tfrom.implicitConvTo(tto)) 115 return true; 116 117 if (!tto.hasPointers()) 118 return true; 119 120 auto tfromb = tfrom.toBasetype(); 121 auto ttob = tto.toBasetype(); 122 123 if (ttob.ty == Tclass && tfromb.ty == Tclass) 124 { 125 ClassDeclaration cdfrom = tfromb.isClassHandle(); 126 ClassDeclaration cdto = ttob.isClassHandle(); 127 128 int offset; 129 if (!cdfrom.isBaseOf(cdto, &offset) && 130 !((cdfrom.isInterfaceDeclaration() || cdto.isInterfaceDeclaration()) 131 && cdfrom.classKind == ClassKind.d && cdto.classKind == ClassKind.d)) 132 return false; 133 134 if (cdfrom.isCPPinterface() || cdto.isCPPinterface()) 135 return false; 136 137 if (!MODimplicitConv(tfromb.mod, ttob.mod)) 138 return false; 139 return true; 140 } 141 142 if (ttob.ty == Tarray && tfromb.ty == Tsarray) // https://issues.dlang.org/show_bug.cgi?id=12502 143 tfromb = tfromb.nextOf().arrayOf(); 144 145 if (ttob.ty == Tarray && tfromb.ty == Tarray || 146 ttob.ty == Tpointer && tfromb.ty == Tpointer) 147 { 148 Type ttobn = ttob.nextOf().toBasetype(); 149 Type tfromn = tfromb.nextOf().toBasetype(); 150 151 /* From void[] to anything mutable is unsafe because: 152 * int*[] api; 153 * void[] av = api; 154 * int[] ai = cast(int[]) av; 155 * ai[0] = 7; 156 * *api[0] crash! 157 */ 158 if (tfromn.ty == Tvoid && ttobn.isMutable()) 159 { 160 if (ttob.ty == Tarray && e.op == TOK.arrayLiteral) 161 return true; 162 return false; 163 } 164 165 // If the struct is opaque we don't know about the struct members then the cast becomes unsafe 166 if (ttobn.ty == Tstruct && !(cast(TypeStruct)ttobn).sym.members || 167 tfromn.ty == Tstruct && !(cast(TypeStruct)tfromn).sym.members) 168 return false; 169 170 const frompointers = tfromn.hasPointers(); 171 const topointers = ttobn.hasPointers(); 172 173 if (frompointers && !topointers && ttobn.isMutable()) 174 return false; 175 176 if (!frompointers && topointers) 177 return false; 178 179 if (!topointers && 180 ttobn.ty != Tfunction && tfromn.ty != Tfunction && 181 (ttob.ty == Tarray || ttobn.size() <= tfromn.size()) && 182 MODimplicitConv(tfromn.mod, ttobn.mod)) 183 { 184 return true; 185 } 186 } 187 return false; 188 } 189