1 /** 2 * Break down a D type into basic (register) types for the x86_64 System V ABI. 3 * 4 * Copyright: Copyright (C) 1999-2021 by The D Language Foundation, All Rights Reserved 5 * Authors: Martin Kinkelin 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/argtypes_sysv_x64.d, _argtypes_sysv_x64.d) 8 * Documentation: https://dlang.org/phobos/dmd_argtypes_sysv_x64.html 9 * Coverage: https://codecov.io/gh/dlang/dmd/src/master/src/dmd/argtypes_sysv_x64.d 10 */ 11 12 module dmd.argtypes_sysv_x64; 13 14 import dmd.declaration; 15 import dmd.globals; 16 import dmd.mtype; 17 import dmd.visitor; 18 19 /**************************************************** 20 * This breaks a type down into 'simpler' types that can be passed to a function 21 * in registers, and returned in registers. 22 * This is the implementation for the x86_64 System V ABI (not used for Win64), 23 * based on https://www.uclibc.org/docs/psABI-x86_64.pdf. 24 * Params: 25 * t = type to break down 26 * Returns: 27 * tuple of types, each element can be passed in a register. 28 * A tuple of zero length means the type cannot be passed/returned in registers. 29 * null indicates a `void`. 30 */ 31 extern (C++) TypeTuple toArgTypes_sysv_x64(Type t) 32 { 33 if (t == Type.terror) 34 return new TypeTuple(t); 35 36 const size = cast(size_t) t.size(); 37 if (size == 0) 38 return null; 39 if (size > 32) 40 return TypeTuple.empty; 41 42 const classification = classify(t, size); 43 const classes = classification.slice(); 44 const N = classes.length; 45 const c0 = classes[0]; 46 47 switch (c0) 48 { 49 case Class.memory: 50 return TypeTuple.empty; 51 case Class.x87: 52 return new TypeTuple(Type.tfloat80); 53 case Class.complexX87: 54 return new TypeTuple(Type.tfloat80, Type.tfloat80); 55 default: 56 break; 57 } 58 59 if (N > 2 || (N == 2 && classes[1] == Class.sseUp)) 60 { 61 assert(c0 == Class.sse); 62 foreach (c; classes[1 .. $]) 63 assert(c == Class.sseUp); 64 65 assert(size % 8 == 0); 66 return new TypeTuple(new TypeVector(Type.tfloat64.sarrayOf(N))); 67 } 68 69 assert(N >= 1 && N <= 2); 70 Type[2] argtypes; 71 foreach (i, c; classes) 72 { 73 // the last eightbyte may be filled partially only 74 auto sizeInEightbyte = (i < N - 1) ? 8 : size % 8; 75 if (sizeInEightbyte == 0) 76 sizeInEightbyte = 8; 77 78 if (c == Class.integer) 79 { 80 argtypes[i] = 81 sizeInEightbyte > 4 ? Type.tint64 : 82 sizeInEightbyte > 2 ? Type.tint32 : 83 sizeInEightbyte > 1 ? Type.tint16 : 84 Type.tint8; 85 } 86 else if (c == Class.sse) 87 { 88 argtypes[i] = 89 sizeInEightbyte > 4 ? Type.tfloat64 : 90 Type.tfloat32; 91 } 92 else 93 assert(0, "Unexpected class"); 94 } 95 96 return N == 1 97 ? new TypeTuple(argtypes[0]) 98 : new TypeTuple(argtypes[0], argtypes[1]); 99 } 100 101 102 private: 103 104 // classification per eightbyte (64-bit chunk) 105 enum Class : ubyte 106 { 107 integer, 108 sse, 109 sseUp, 110 x87, 111 x87Up, 112 complexX87, 113 noClass, 114 memory 115 } 116 117 Class merge(Class a, Class b) 118 { 119 bool any(Class value) { return a == value || b == value; } 120 121 if (a == b) 122 return a; 123 if (a == Class.noClass) 124 return b; 125 if (b == Class.noClass) 126 return a; 127 if (any(Class.memory)) 128 return Class.memory; 129 if (any(Class.integer)) 130 return Class.integer; 131 if (any(Class.x87) || any(Class.x87Up) || any(Class.complexX87)) 132 return Class.memory; 133 return Class.sse; 134 } 135 136 struct Classification 137 { 138 Class[4] classes; 139 int numEightbytes; 140 141 const(Class[]) slice() const return { return classes[0 .. numEightbytes]; } 142 } 143 144 Classification classify(Type t, size_t size) 145 { 146 scope v = new ToClassesVisitor(size); 147 t.accept(v); 148 return Classification(v.result, v.numEightbytes); 149 } 150 151 extern (C++) final class ToClassesVisitor : Visitor 152 { 153 const size_t size; 154 int numEightbytes; 155 Class[4] result = Class.noClass; 156 157 this(size_t size) 158 { 159 assert(size > 0); 160 this.size = size; 161 this.numEightbytes = cast(int) ((size + 7) / 8); 162 } 163 164 void memory() 165 { 166 result[0 .. numEightbytes] = Class.memory; 167 } 168 169 void one(Class a) 170 { 171 result[0] = a; 172 } 173 174 void two(Class a, Class b) 175 { 176 result[0] = a; 177 result[1] = b; 178 } 179 180 alias visit = Visitor.visit; 181 182 override void visit(Type) 183 { 184 assert(0, "Unexpected type"); 185 } 186 187 override void visit(TypeEnum t) 188 { 189 t.toBasetype().accept(this); 190 } 191 192 override void visit(TypeBasic t) 193 { 194 switch (t.ty) 195 { 196 case Tvoid: 197 case Tbool: 198 case Tint8: 199 case Tuns8: 200 case Tint16: 201 case Tuns16: 202 case Tint32: 203 case Tuns32: 204 case Tint64: 205 case Tuns64: 206 case Tchar: 207 case Twchar: 208 case Tdchar: 209 return one(Class.integer); 210 211 case Tint128: 212 case Tuns128: 213 return two(Class.integer, Class.integer); 214 215 case Tfloat80: 216 case Timaginary80: 217 return two(Class.x87, Class.x87Up); 218 219 case Tfloat32: 220 case Tfloat64: 221 case Timaginary32: 222 case Timaginary64: 223 case Tcomplex32: // struct { float a, b; } 224 return one(Class.sse); 225 226 case Tcomplex64: // struct { double a, b; } 227 return two(Class.sse, Class.sse); 228 229 case Tcomplex80: // struct { real a, b; } 230 result[0 .. 4] = Class.complexX87; 231 return; 232 233 default: 234 assert(0, "Unexpected basic type"); 235 } 236 } 237 238 override void visit(TypeVector t) 239 { 240 result[0] = Class.sse; 241 result[1 .. numEightbytes] = Class.sseUp; 242 } 243 244 override void visit(TypeAArray) 245 { 246 return one(Class.integer); 247 } 248 249 override void visit(TypePointer) 250 { 251 return one(Class.integer); 252 } 253 254 override void visit(TypeNull) 255 { 256 return one(Class.integer); 257 } 258 259 override void visit(TypeClass) 260 { 261 return one(Class.integer); 262 } 263 264 override void visit(TypeDArray) 265 { 266 if (!global.params.isLP64) 267 return one(Class.integer); 268 return two(Class.integer, Class.integer); 269 } 270 271 override void visit(TypeDelegate) 272 { 273 if (!global.params.isLP64) 274 return one(Class.integer); 275 return two(Class.integer, Class.integer); 276 } 277 278 override void visit(TypeSArray t) 279 { 280 // treat as struct with N fields 281 282 Type baseElemType = t.next.toBasetype(); 283 if (baseElemType.ty == Tstruct && !(cast(TypeStruct) baseElemType).sym.isPOD()) 284 return memory(); 285 286 classifyStaticArrayElements(0, t); 287 finalizeAggregate(); 288 } 289 290 override void visit(TypeStruct t) 291 { 292 if (!t.sym.isPOD()) 293 return memory(); 294 295 classifyStructFields(0, t); 296 finalizeAggregate(); 297 } 298 299 void classifyStructFields(uint baseOffset, TypeStruct t) 300 { 301 extern(D) Type getNthField(size_t n, out uint offset, out uint typeAlignment) 302 { 303 auto field = t.sym.fields[n]; 304 offset = field.offset; 305 typeAlignment = field.type.alignsize(); 306 return field.type; 307 } 308 309 classifyFields(baseOffset, t.sym.fields.dim, &getNthField); 310 } 311 312 void classifyStaticArrayElements(uint baseOffset, TypeSArray t) 313 { 314 Type elemType = t.next; 315 const elemSize = elemType.size(); 316 const elemTypeAlignment = elemType.alignsize(); 317 318 extern(D) Type getNthElement(size_t n, out uint offset, out uint typeAlignment) 319 { 320 offset = cast(uint)(n * elemSize); 321 typeAlignment = elemTypeAlignment; 322 return elemType; 323 } 324 325 classifyFields(baseOffset, cast(size_t) t.dim.toInteger(), &getNthElement); 326 } 327 328 extern(D) void classifyFields(uint baseOffset, size_t nfields, Type delegate(size_t, out uint, out uint) getFieldInfo) 329 { 330 if (nfields == 0) 331 return memory(); 332 333 // classify each field (recursively for aggregates) and merge all classes per eightbyte 334 foreach (n; 0 .. nfields) 335 { 336 uint foffset_relative; 337 uint ftypeAlignment; 338 Type ftype = getFieldInfo(n, foffset_relative, ftypeAlignment); 339 const fsize = cast(size_t) ftype.size(); 340 341 const foffset = baseOffset + foffset_relative; 342 if (foffset & (ftypeAlignment - 1)) // not aligned 343 return memory(); 344 345 if (ftype.ty == Tstruct) 346 classifyStructFields(foffset, cast(TypeStruct) ftype); 347 else if (ftype.ty == Tsarray) 348 classifyStaticArrayElements(foffset, cast(TypeSArray) ftype); 349 else 350 { 351 const fEightbyteStart = foffset / 8; 352 const fEightbyteEnd = (foffset + fsize + 7) / 8; 353 if (ftype.ty == Tcomplex32) // may lie in 2 eightbytes 354 { 355 assert(foffset % 4 == 0); 356 foreach (ref existingClass; result[fEightbyteStart .. fEightbyteEnd]) 357 existingClass = merge(existingClass, Class.sse); 358 } 359 else 360 { 361 assert(foffset % 8 == 0 || 362 fEightbyteEnd - fEightbyteStart <= 1 || 363 !global.params.isLP64, 364 "Field not aligned at eightbyte boundary but contributing to multiple eightbytes?" 365 ); 366 foreach (i, fclass; classify(ftype, fsize).slice()) 367 { 368 Class* existingClass = &result[fEightbyteStart + i]; 369 *existingClass = merge(*existingClass, fclass); 370 } 371 } 372 } 373 } 374 } 375 376 void finalizeAggregate() 377 { 378 foreach (i, ref c; result) 379 { 380 if (c == Class.memory || 381 (c == Class.x87Up && !(i > 0 && result[i - 1] == Class.x87))) 382 return memory(); 383 384 if (c == Class.sseUp && !(i > 0 && 385 (result[i - 1] == Class.sse || result[i - 1] == Class.sseUp))) 386 c = Class.sse; 387 } 388 389 if (numEightbytes > 2) 390 { 391 if (result[0] != Class.sse) 392 return memory(); 393 394 foreach (c; result[1 .. numEightbytes]) 395 if (c != Class.sseUp) 396 return memory(); 397 } 398 399 // Undocumented special case for aggregates with the 2nd eightbyte 400 // consisting of padding only (`struct S { align(16) int a; }`). 401 // clang only passes the first eightbyte in that case, so let's do the 402 // same. 403 if (numEightbytes == 2 && result[1] == Class.noClass) 404 numEightbytes = 1; 405 } 406 }