/* d-target.cc -- Target interface for the D front end.
Copyright (C) 2013-2022 Free Software Foundation, Inc.
GCC is free software; you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation; either version 3, or (at your option)
any later version.
GCC is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with GCC; see the file COPYING3. If not see
. */
#include "config.h"
#include "system.h"
#include "coretypes.h"
#include "dmd/aggregate.h"
#include "dmd/declaration.h"
#include "dmd/expression.h"
#include "dmd/mangle.h"
#include "dmd/mtype.h"
#include "dmd/tokens.h"
#include "dmd/target.h"
#include "tree.h"
#include "memmodel.h"
#include "fold-const.h"
#include "diagnostic.h"
#include "stor-layout.h"
#include "tm.h"
#include "tm_p.h"
#include "target.h"
#include "calls.h"
#include "d-tree.h"
#include "d-target.h"
/* Implements the Target interface defined by the front end.
Used for retrieving target-specific information. */
/* Internal key handlers for `__traits(getTargetInfo)'. */
static tree d_handle_target_cpp_std (void);
static tree d_handle_target_cpp_runtime_library (void);
static tree d_handle_target_object_format (void);
/* In [traits/getTargetInfo], a reliable subset of getTargetInfo keys exists
which are always available. */
static const struct d_target_info_spec d_language_target_info[] =
{
/* { name, handler } */
{ "cppStd", d_handle_target_cpp_std },
{ "cppRuntimeLibrary", d_handle_target_cpp_runtime_library },
{ "floatAbi", NULL },
{ "objectFormat", d_handle_target_object_format },
{ NULL, NULL },
};
/* Table `__traits(getTargetInfo)' keys. */
static vec d_target_info_table;
/* Initialize the floating-point constants for TYPE. */
template
static void
define_float_constants (T &f, tree type)
{
const double log10_2 = 0.30102999566398119521;
char buf[128];
/* Get back-end real mode format. */
const machine_mode mode = TYPE_MODE (type);
const real_format *fmt = REAL_MODE_FORMAT (mode);
/* The largest representable value that's not infinity. */
get_max_float (fmt, buf, sizeof (buf), false);
real_from_string (&f.max.rv (), buf);
/* The smallest representable normalized value that's not 0. */
snprintf (buf, sizeof (buf), "0x1p%d", fmt->emin - 1);
real_from_string (&f.min_normal.rv (), buf);
/* Floating-point NaN. */
real_nan (&f.nan.rv (), "", 1, mode);
/* Floating-point +Infinity if the target supports infinities. */
real_inf (&f.infinity.rv ());
/* The smallest increment to the value 1. */
if (fmt->pnan < fmt->p)
snprintf (buf, sizeof (buf), "0x1p%d", fmt->emin - fmt->p);
else
snprintf (buf, sizeof (buf), "0x1p%d", 1 - fmt->p);
real_from_string (&f.epsilon.rv (), buf);
/* The number of decimal digits of precision. */
f.dig = (fmt->p - 1) * log10_2;
/* The number of bits in mantissa. */
f.mant_dig = fmt->p;
/* The maximum int value such that 2** (value-1) is representable. */
f.max_exp = fmt->emax;
/* The minimum int value such that 2** (value-1) is representable as a
normalized value. */
f.min_exp = fmt->emin;
/* The maximum int value such that 10**value is representable. */
f.max_10_exp = fmt->emax * log10_2;
/* The minimum int value such that 10**value is representable as a
normalized value. */
f.min_10_exp = (fmt->emin - 1) * log10_2;
}
/* Initialize all variables of the Target structure. */
void
Target::_init (const Param &)
{
/* Map D frontend type and sizes to GCC back-end types. */
this->ptrsize = (POINTER_SIZE / BITS_PER_UNIT);
this->realsize = int_size_in_bytes (long_double_type_node);
this->realpad = (this->realsize -
(TYPE_PRECISION (long_double_type_node) / BITS_PER_UNIT));
this->realalignsize = TYPE_ALIGN_UNIT (long_double_type_node);
/* Much of the dmd front-end uses ints for sizes and offsets, and cannot
handle any larger data type without some pervasive rework. */
this->maxStaticDataSize = tree_to_shwi (TYPE_MAX_VALUE (integer_type_node));
/* Define what type to use for size_t, ptrdiff_t. */
if (this->ptrsize == 8)
{
this->isLP64 = true;
Type::tsize_t = Type::basic[(int)TY::Tuns64];
Type::tptrdiff_t = Type::basic[(int)TY::Tint64];
}
else if (this->ptrsize == 4)
{
Type::tsize_t = Type::basic[(int)TY::Tuns32];
Type::tptrdiff_t = Type::basic[(int)TY::Tint32];
}
else if (this->ptrsize == 2)
{
Type::tsize_t = Type::basic[(int)TY::Tuns16];
Type::tptrdiff_t = Type::basic[(int)TY::Tint16];
}
else
sorry ("D does not support pointers on this target.");
Type::thash_t = Type::tsize_t;
/* Set-up target C ABI. */
this->c.boolsize = (BOOL_TYPE_SIZE / BITS_PER_UNIT);
this->c.shortsize = (SHORT_TYPE_SIZE / BITS_PER_UNIT);
this->c.intsize = (INT_TYPE_SIZE / BITS_PER_UNIT);
this->c.longsize = (LONG_TYPE_SIZE / BITS_PER_UNIT);
this->c.long_longsize = (LONG_LONG_TYPE_SIZE / BITS_PER_UNIT);
this->c.long_doublesize = (LONG_DOUBLE_TYPE_SIZE / BITS_PER_UNIT);
this->c.wchar_tsize = (WCHAR_TYPE_SIZE / BITS_PER_UNIT);
this->c.bitFieldStyle = targetm.ms_bitfield_layout_p (unknown_type_node)
? TargetC::BitFieldStyle::MS : TargetC::BitFieldStyle::Gcc_Clang;
/* Set-up target C++ ABI. */
this->cpp.reverseOverloads = false;
this->cpp.exceptions = true;
this->cpp.twoDtorInVtable = true;
/* Set-up target Objective-C ABI. */
this->objc.supported = false;
/* Set-up environmental settings. */
this->obj_ext = "o";
this->lib_ext = "a";
this->dll_ext = "so";
this->run_noext = true;
/* Initialize all compile-time properties for floating-point types.
Should ensure that our real_t type is able to represent real_value. */
gcc_assert (sizeof (real_t) >= sizeof (real_value));
define_float_constants (this->FloatProperties, float_type_node);
define_float_constants (this->DoubleProperties, double_type_node);
define_float_constants (this->RealProperties, long_double_type_node);
/* Commonly used floating-point constants. */
const machine_mode mode = TYPE_MODE (long_double_type_node);
real_convert (&CTFloat::zero.rv (), mode, &dconst0);
real_convert (&CTFloat::one.rv (), mode, &dconst1);
real_convert (&CTFloat::minusone.rv (), mode, &dconstm1);
real_convert (&CTFloat::half.rv (), mode, &dconsthalf);
/* Initialize target info tables, the keys required by the language are added
last, so that the OS and CPU handlers can override. */
targetdm.d_register_cpu_target_info ();
targetdm.d_register_os_target_info ();
d_add_target_info_handlers (d_language_target_info);
}
/* Return GCC memory alignment size for type TYPE. */
unsigned
Target::alignsize (Type *type)
{
gcc_assert (type->isTypeBasic ());
return min_align_of_type (build_ctype (type));
}
/* Return GCC field alignment size for type TYPE. */
unsigned
Target::fieldalign (Type *type)
{
/* Work out the correct alignment for the field decl. */
unsigned int align = type->alignsize () * BITS_PER_UNIT;
#ifdef BIGGEST_FIELD_ALIGNMENT
align = MIN (align, (unsigned) BIGGEST_FIELD_ALIGNMENT);
#endif
#ifdef ADJUST_FIELD_ALIGN
if (type->isTypeBasic ())
align = ADJUST_FIELD_ALIGN (NULL_TREE, build_ctype (type), align);
#endif
/* Also controlled by -fpack-struct= */
if (maximum_field_alignment)
align = MIN (align, maximum_field_alignment);
return align / BITS_PER_UNIT;
}
/* Returns a Type for the va_list type of the target. */
Type *
Target::va_listType (const Loc &, Scope *)
{
if (this->tvalist)
return this->tvalist;
/* Build the "standard" abi va_list. */
this->tvalist = build_frontend_type (va_list_type_node);
if (!this->tvalist)
sorry ("cannot represent built-in % type in D");
/* Map the va_list type to the D frontend Type. This is to prevent both
errors in gimplification or an ICE in targetm.canonical_va_list_type. */
this->tvalist->ctype = va_list_type_node;
TYPE_LANG_SPECIFIC (va_list_type_node) = build_lang_type (this->tvalist);
return this->tvalist;
}
/* Checks whether the target supports a vector type with total size SZ
(in bytes) and element type TYPE. */
int
Target::isVectorTypeSupported (int sz, Type *type)
{
/* Size must be greater than zero, and a power of two. */
if (sz <= 0 || sz & (sz - 1))
return 3;
/* __vector(void[]) is treated same as __vector(ubyte[]) */
if (type == Type::tvoid)
type = Type::tuns8;
/* No support for non-trivial types, complex types, or booleans. */
if (!type->isTypeBasic () || type->iscomplex () || type->ty == TY::Tbool)
return 2;
/* In [simd/vector extensions], which vector types are supported depends on
the target. The implementation is expected to only support the vector
types that are implemented in the target's hardware. */
unsigned HOST_WIDE_INT nunits = sz / type->size ();
tree ctype = build_vector_type (build_ctype (type), nunits);
if (!targetm.vector_mode_supported_p (TYPE_MODE (ctype)))
return 2;
return 0;
}
/* Checks whether the target supports operation OP for vectors of type TYPE.
For binary ops T2 is the type of the right-hand operand.
Returns true if the operation is supported or type is not a vector. */
bool
Target::isVectorOpSupported (Type *type, EXP op, Type *)
{
if (type->ty != TY::Tvector)
return true;
/* Don't support if type is non-scalar, such as __vector(void[]). */
if (!type->isscalar ())
return false;
/* Don't support if expression cannot be represented. */
switch (op)
{
case EXP::pow:
case EXP::powAssign:
/* pow() is lowered as a function call. */
return false;
case EXP::mod:
case EXP::modAssign:
/* fmod() is lowered as a function call. */
if (type->isfloating ())
return false;
break;
case EXP::andAnd:
case EXP::orOr:
/* Logical operators must have a result type of bool. */
return false;
case EXP::lessOrEqual:
case EXP::lessThan:
case EXP::greaterOrEqual:
case EXP::greaterThan:
case EXP::equal:
case EXP::notEqual:
case EXP::identity:
case EXP::notIdentity:
/* Comparison operators must have a result type of bool. */
return false;
default:
break;
}
return true;
}
/* Return the symbol mangling of S for C++ linkage. */
const char *
TargetCPP::toMangle (Dsymbol *s)
{
return toCppMangleItanium (s);
}
/* Return the symbol mangling of CD for C++ linkage. */
const char *
TargetCPP::typeInfoMangle (ClassDeclaration *cd)
{
return cppTypeInfoMangleItanium (cd);
}
/* Get mangle name of a this-adjusting thunk to the function declaration FD
at call offset OFFSET for C++ linkage. */
const char *
TargetCPP::thunkMangle (FuncDeclaration *fd, int offset)
{
return cppThunkMangleItanium (fd, offset);
}
/* For a vendor-specific type, return a string containing the C++ mangling.
In all other cases, return NULL. */
const char *
TargetCPP::typeMangle (Type *type)
{
if (type->isTypeBasic () || type->ty == TY::Tvector
|| type->ty == TY::Tstruct)
{
tree ctype = build_ctype (type);
return targetm.mangle_type (ctype);
}
return NULL;
}
/* Return the type that will really be used for passing the given parameter
ARG to an extern(C++) function. */
Type *
TargetCPP::parameterType (Type *type)
{
/* Could be a va_list, which we mangle as a pointer. */
Type *tvalist = target.va_listType (Loc (), NULL);
if (type->ty == TY::Tsarray && tvalist->ty == TY::Tsarray)
{
Type *tb = type->toBasetype ()->mutableOf ();
if (tb == tvalist)
{
tb = type->nextOf ()->pointerTo ();
type = tb->castMod (type->mod);
}
}
return type;
}
/* Checks whether TYPE is a vendor-specific fundamental type. Stores the result
in IS_FUNDAMENTAL and returns true if the parameter was set. */
bool
TargetCPP::fundamentalType (const Type *, bool &)
{
return false;
}
/* Get the starting offset position for fields of an `extern(C++)` class
that is derived from the given BASE_CLASS. */
unsigned
TargetCPP::derivedClassOffset(ClassDeclaration *base_class)
{
return base_class->structsize;
}
/* Return the default `extern (System)' linkage for the target. */
LINK
Target::systemLinkage (void)
{
unsigned link_system, link_windows;
if (targetdm.d_has_stdcall_convention (&link_system, &link_windows))
{
/* In [attribute/linkage], `System' is the same as `Windows' on Windows
platforms, and `C' on other platforms. */
if (link_system)
return LINK::windows;
}
return LINK::c;
}
/* Generate a TypeTuple of the equivalent types used to determine if a
function argument of the given type can be passed in registers.
The results of this are highly platform dependent, and intended
primarly for use in implementing va_arg() with RTTI. */
TypeTuple *
Target::toArgTypes (Type *)
{
/* Not implemented, however this is not currently used anywhere. */
return NULL;
}
/* Determine return style of function, whether in registers or through a
hidden pointer to the caller's stack. */
bool
Target::isReturnOnStack (TypeFunction *tf, bool)
{
/* Need the back-end type to determine this, but this is called from the
frontend before semantic processing is finished. An accurate value
is not currently needed anyway. */
if (tf->isref ())
return false;
Type *tn = tf->next->toBasetype ();
if (tn->size () == SIZE_INVALID)
return false;
return (tn->ty == TY::Tstruct || tn->ty == TY::Tsarray);
}
/* Add all target info in HANDLERS to D_TARGET_INFO_TABLE for use by
Target::getTargetInfo(). */
void
d_add_target_info_handlers (const d_target_info_spec *handlers)
{
gcc_assert (handlers != NULL);
if (d_target_info_table.is_empty ())
d_target_info_table.create (8);
for (size_t i = 0; handlers[i].name != NULL; i++)
d_target_info_table.safe_push (handlers[i]);
}
/* Handle a call to `__traits(getTargetInfo, "cppStd")'. */
tree
d_handle_target_cpp_std (void)
{
return build_integer_cst (global.params.cplusplus);
}
/* Handle a call to `__traits(getTargetInfo, "cppRuntimeLibrary")'. */
tree
d_handle_target_cpp_runtime_library (void)
{
/* The driver only ever optionally links to libstdc++. */
const char *libstdcxx = "libstdc++";
return build_string_literal (strlen (libstdcxx) + 1, libstdcxx);
}
/* Handle a call to `__traits(getTargetInfo, "objectFormat")'. */
tree
d_handle_target_object_format (void)
{
const char *objfmt;
#ifdef OBJECT_FORMAT_ELF
objfmt = "elf";
#else
if (TARGET_COFF || TARGET_PECOFF)
objfmt = "coff";
else
objfmt = "";
#endif
return build_string_literal (strlen (objfmt) + 1, objfmt);
}
/* Look up the target info KEY in the available getTargetInfo tables, and return
the result as an Expression, or NULL if KEY is not found. When the key must
always exist, but is not supported, an empty string expression is returned.
LOC is the location to use for the returned expression. */
Expression *
Target::getTargetInfo (const char *key, const Loc &loc)
{
unsigned ix;
d_target_info_spec *spec;
FOR_EACH_VEC_ELT (d_target_info_table, ix, spec)
{
tree result;
if (strcmp (key, spec->name) != 0)
continue;
/* Get the requested information, or empty string if unhandled. */
if (spec->handler)
{
result = (spec->handler) ();
/* Handler didn't return a result, meaning it really does not support
the key in the current target configuration. Check whether there
are any other handlers which may recognize the key. */
if (result == NULL_TREE)
continue;
}
else
result = build_string_literal (1, "");
gcc_assert (result);
return d_eval_constant_expression (loc, result);
}
return NULL;
}
/* Returns true if the callee invokes destructors for arguments. */
bool
Target::isCalleeDestroyingArgs (TypeFunction *tf)
{
return tf->linkage == LINK::d;
}
/* Returns true if the implementation for object monitors is always defined
in the D runtime library (rt/monitor_.d). */
bool
Target::libraryObjectMonitors (FuncDeclaration *, Statement *)
{
return true;
}
/* Returns true if the target supports `pragma(linkerDirective)'. */
bool
Target::supportsLinkerDirective (void) const
{
return false;
}
/* Decides whether an `in' parameter of the specified POD type PARAM_TYPE is to
be passed by reference or by valie. This is used only when compiling with
`-fpreview=in' enabled. */
bool
Target::preferPassByRef (Type *param_type)
{
if (param_type->size () == SIZE_INVALID)
return false;
tree type = build_ctype (param_type);
/* Prefer a `ref' if the type is an aggregate, and its size is greater than
its alignment. */
if (AGGREGATE_TYPE_P (type)
&& (!valid_constant_size_p (TYPE_SIZE_UNIT (type))
|| compare_tree_int (TYPE_SIZE_UNIT (type), TYPE_ALIGN (type)) > 0))
return true;
/* If the back-end is always going to pass this by invisible reference. */
if (pass_by_reference (NULL, function_arg_info (type, true)))
return true;
/* If returning the parameter means the caller will do RVO. */
if (targetm.calls.return_in_memory (type, NULL_TREE))
return true;
return false;
}