| /** |
| * The vararg module is intended to facilitate vararg manipulation in D. |
| * It should be interface compatible with the C module "stdarg," and the |
| * two modules may share a common implementation if possible (as is done |
| * here). |
| * Copyright: Copyright Digital Mars 2000 - 2009. |
| * License: $(LINK2 http://www.boost.org/LICENSE_1_0.txt, Boost License 1.0) |
| * Authors: Walter Bright, Hauke Duden |
| * Source: $(DRUNTIMESRC core/_vararg.d) |
| */ |
| |
| /* Copyright Digital Mars 2000 - 2009. |
| * Distributed under the Boost Software License, Version 1.0. |
| * (See accompanying file LICENSE or copy at |
| * http://www.boost.org/LICENSE_1_0.txt) |
| */ |
| module core.vararg; |
| |
| public import core.stdc.stdarg; |
| |
| |
| version (GNU) { /* TypeInfo-based va_arg overload unsupported */ } |
| else: |
| |
| version (ARM) version = ARM_Any; |
| version (AArch64) version = ARM_Any; |
| version (MIPS32) version = MIPS_Any; |
| version (MIPS64) version = MIPS_Any; |
| version (PPC) version = PPC_Any; |
| version (PPC64) version = PPC_Any; |
| |
| version (ARM_Any) |
| { |
| // Darwin uses a simpler varargs implementation |
| version (OSX) {} |
| else version (iOS) {} |
| else version (TVOS) {} |
| else version (WatchOS) {} |
| else: |
| |
| version (ARM) version = AAPCS32; |
| version (AArch64) version = AAPCS64; |
| } |
| |
| |
| /// |
| alias va_arg = core.stdc.stdarg.va_arg; |
| |
| |
| /** |
| * Retrieve and store through parmn the next value that is of TypeInfo ti. |
| * Used when the static type is not known. |
| */ |
| void va_arg()(ref va_list ap, TypeInfo ti, void* parmn) |
| { |
| version (X86) |
| { |
| // Wait until everyone updates to get TypeInfo.talign |
| //auto talign = ti.talign; |
| //auto p = cast(void*)(cast(size_t)ap + talign - 1) & ~(talign - 1); |
| auto p = ap; |
| auto tsize = ti.tsize; |
| ap = cast(va_list) (p + tsize.alignUp); |
| parmn[0..tsize] = p[0..tsize]; |
| } |
| else version (Win64) |
| { |
| version (LDC) enum isLDC = true; |
| else enum isLDC = false; |
| |
| // Wait until everyone updates to get TypeInfo.talign |
| //auto talign = ti.talign; |
| //auto p = cast(void*)(cast(size_t)ap + talign - 1) & ~(talign - 1); |
| auto p = ap; |
| auto tsize = ti.tsize; |
| void* q; |
| if (isLDC && tsize == 16 && cast(TypeInfo_Array) ti) |
| { |
| q = p; |
| ap = cast(va_list) (p + tsize); |
| } |
| else |
| { |
| q = (tsize > size_t.sizeof || (tsize & (tsize - 1)) != 0) ? *cast(void**) p : p; |
| ap = cast(va_list) (p + size_t.sizeof); |
| } |
| parmn[0..tsize] = q[0..tsize]; |
| } |
| else version (X86_64) |
| { |
| static import core.internal.vararg.sysv_x64; |
| core.internal.vararg.sysv_x64.va_arg(ap, ti, parmn); |
| } |
| else version (AAPCS32) |
| { |
| const tsize = ti.tsize; |
| if (ti.talign >= 8) |
| ap.__ap = ap.__ap.alignUp!8; |
| auto p = ap.__ap; |
| version (BigEndian) |
| p = adjustForBigEndian(p, tsize); |
| ap.__ap += tsize.alignUp; |
| parmn[0..tsize] = p[0..tsize]; |
| } |
| else version (AAPCS64) |
| { |
| static import core.internal.vararg.aarch64; |
| core.internal.vararg.aarch64.va_arg(ap, ti, parmn); |
| } |
| else version (ARM_Any) |
| { |
| const tsize = ti.tsize; |
| auto p = cast(void*) ap; |
| version (BigEndian) |
| p = adjustForBigEndian(p, tsize); |
| ap += tsize.alignUp; |
| parmn[0..tsize] = p[0..tsize]; |
| } |
| else version (PPC_Any) |
| { |
| if (ti.talign >= 8) |
| ap = ap.alignUp!8; |
| const tsize = ti.tsize; |
| auto p = cast(void*) ap; |
| version (BigEndian) |
| p = adjustForBigEndian(p, tsize); |
| ap += tsize.alignUp; |
| parmn[0..tsize] = p[0..tsize]; |
| } |
| else version (MIPS_Any) |
| { |
| const tsize = ti.tsize; |
| auto p = cast(void*) ap; |
| version (BigEndian) |
| p = adjustForBigEndian(p, tsize); |
| ap += tsize.alignUp; |
| parmn[0..tsize] = p[0..tsize]; |
| } |
| else |
| static assert(0, "Unsupported platform"); |
| } |