gcc/libphobos/libdruntime/core/checkedint.d
Iain Buclaw 5fee5ec362 d: Import dmd b8384668f, druntime e6caaab9, phobos 5ab9ad256 (v2.098.0-beta.1)
The D front-end is now itself written in D, in order to build GDC, you
will need a working GDC compiler (GCC version 9.1 or later).

GCC changes:

    - Add support for bootstrapping the D front-end.

These add the required components in order to have a D front-end written
in D itself.  Because the compiler front-end only depends on the core
runtime modules, only libdruntime is built for the bootstrap stages.

D front-end changes:

    - Import dmd v2.098.0-beta.1.

Druntime changes:

    - Import druntime v2.098.0-beta.1.

Phobos changes:

    - Import phobos v2.098.0-beta.1.

The jump from v2.076.1 to v2.098.0 covers nearly 4 years worth of
development on the D programming language and run-time libraries.

ChangeLog:

	* Makefile.def: Add bootstrap to libbacktrace, libphobos, zlib, and
	libatomic.
	* Makefile.in: Regenerate.
	* Makefile.tpl (POSTSTAGE1_HOST_EXPORTS): Fix command for GDC.
	(STAGE1_CONFIGURE_FLAGS): Add --with-libphobos-druntime-only if
	target-libphobos-bootstrap.
	(STAGE2_CONFIGURE_FLAGS): Likewise.
	* configure: Regenerate.
	* configure.ac: Add support for bootstrapping D front-end.

config/ChangeLog:

	* acx.m4 (ACX_PROG_GDC): New m4 function.

gcc/ChangeLog:

	* Makefile.in (GDC): New variable.
	(GDCFLAGS): New variable.
	* configure: Regenerate.
	* configure.ac: Add call to ACX_PROG_GDC.  Substitute GDCFLAGS.

gcc/d/ChangeLog:

	* dmd/MERGE: Merge upstream dmd b8384668f.
	* Make-lang.in (d-warn): Use strict warnings.
	(DMD_WARN_CXXFLAGS): Remove.
	(DMD_COMPILE): Remove.
	(CHECKING_DFLAGS): Define.
	(WARN_DFLAGS): Define.
	(ALL_DFLAGS): Define.
	(DCOMPILE.base): Define.
	(DCOMPILE): Define.
	(DPOSTCOMPILE): Define.
	(DLINKER): Define.
	(DLLINKER): Define.
	(D_FRONTEND_OBJS): Add new dmd front-end objects.
	(D_GENERATED_SRCS): Remove.
	(D_GENERATED_OBJS): Remove.
	(D_ALL_OBJS): Remove D_GENERATED_OBJS.
	(d21$(exeext)): Build using DLLINKER and -static-libphobos.
	(d.tags): Remove dmd/*.c and dmd/root/*.c.
	(d.mostlyclean): Remove D_GENERATED_SRCS, d/idgen$(build_exeext),
	d/impcnvgen$(build_exeext).
	(D_INCLUDES): Include $(srcdir)/d/dmd/res.
	(CFLAGS-d/id.o): Remove.
	(CFLAGS-d/impcnvtab.o): Remove.
	(d/%.o): Build using DCOMPILE and DPOSTCOMPILE.  Update dependencies
	from d/dmd/%.c to d/dmd/%.d.
	(d/idgen$(build_exeext)): Remove.
	(d/impcnvgen$(build_exeext)): Remove.
	(d/id.c): Remove.
	(d/id.h): Remove.
	(d/impcnvtab.c): Remove.
	(d/%.dmdgen.o): Remove.
	(D_SYSTEM_H): Remove.
	(d/idgen.dmdgen.o): Remove.
	(d/impcnvgen.dmdgen.o): Remove.
	* config-lang.in (boot_language): New variable.
	* d-attribs.cc: Include dmd/expression.h.
	* d-builtins.cc: Include d-frontend.h.
	(build_frontend_type): Update for new front-end interface.
	(d_eval_constant_expression): Likewise.
	(d_build_builtins_module): Likewise.
	(maybe_set_builtin_1): Likewise.
	(d_build_d_type_nodes): Likewise.
	* d-codegen.cc (d_decl_context): Likewise.
	(declaration_reference_p): Likewise.
	(declaration_type): Likewise.
	(parameter_reference_p): Likewise.
	(parameter_type): Likewise.
	(get_array_length): Likewise.
	(build_delegate_cst): Likewise.
	(build_typeof_null_value): Likewise.
	(identity_compare_p): Likewise.
	(lower_struct_comparison): Likewise.
	(build_filename_from_loc): Likewise.
	(build_assert_call): Remove LIBCALL_SWITCH_ERROR.
	(build_bounds_index_condition): Call LIBCALL_ARRAYBOUNDS_INDEXP on
	bounds error.
	(build_bounds_slice_condition): Call LIBCALL_ARRAYBOUNDS_SLICEP on
	bounds error.
	(array_bounds_check): Update for new front-end interface.
	(checkaction_trap_p): Handle CHECKACTION_context.
	(get_function_type): Update for new front-end interface.
	(d_build_call): Likewise.
	* d-compiler.cc: Remove include of dmd/scope.h.
	(Compiler::genCmain): Remove.
	(Compiler::paintAsType): Update for new front-end interface.
	(Compiler::onParseModule): Likewise.
	* d-convert.cc (convert_expr): Remove call to LIBCALL_ARRAYCAST.
	(convert_for_rvalue): Update for new front-end interface.
	(convert_for_assignment): Likewise.
	(convert_for_condition): Likewise.
	(d_array_convert): Likewise.
	* d-diagnostic.cc (error): Remove.
	(errorSupplemental): Remove.
	(warning): Remove.
	(warningSupplemental): Remove.
	(deprecation): Remove.
	(deprecationSupplemental): Remove.
	(message): Remove.
	(vtip): New.
	* d-frontend.cc (global): Remove.
	(Global::_init): Remove.
	(Global::startGagging): Remove.
	(Global::endGagging): Remove.
	(Global::increaseErrorCount): Remove.
	(Loc::Loc): Remove.
	(Loc::toChars): Remove.
	(Loc::equals): Remove.
	(isBuiltin): Update for new front-end interface.
	(eval_builtin): Likewise.
	(getTypeInfoType): Likewise.
	(inlineCopy): Remove.
	* d-incpath.cc: Include d-frontend.h.
	(add_globalpaths): Call d_gc_malloc to allocate Strings.
	(add_filepaths): Likewise.
	* d-lang.cc: Include dmd/id.h, dmd/root/file.h, d-frontend.h.  Remove
	include of dmd/mars.h, id.h.
	(entrypoint_module): Remove.
	(entrypoint_root_module): Remove.
	(deps_write_string): Update for new front-end interface.
	(deps_write): Likewise.
	(d_init_options): Call rt_init.  Remove setting global params that are
	default initialized by the front-end.
	(d_handle_option): Handle OPT_fcheckaction_, OPT_fdump_c___spec_,
	OPT_fdump_c___spec_verbose, OPT_fextern_std_, OPT_fpreview,
	OPT_revert, OPT_fsave_mixins_, and OPT_ftransition.
	(d_post_options): Propagate dip1021 and dip1000 preview flags to
	dip25, and flag_diagnostics_show_caret to printErrorContext.
	(d_add_entrypoint_module): Remove.
	(d_parse_file): Update for new front-end interface.
	(d_type_promotes_to): Likewise.
	(d_types_compatible_p): Likewise.
	* d-longdouble.cc (CTFloat::zero): Remove.
	(CTFloat::one): Remove.
	(CTFloat::minusone): Remove.
	(CTFloat::half): Remove.
	* d-system.h (POSIX): Remove.
	(realpath): Remove.
	(isalpha): Remove.
	(isalnum): Remove.
	(isdigit): Remove.
	(islower): Remove.
	(isprint): Remove.
	(isspace): Remove.
	(isupper): Remove.
	(isxdigit): Remove.
	(tolower): Remove.
	(_mkdir): Remove.
	(INT32_MAX): Remove.
	(INT32_MIN): Remove.
	(INT64_MIN): Remove.
	(UINT32_MAX): Remove.
	(UINT64_MAX): Remove.
	* d-target.cc: Include calls.h.
	(target): Remove.
	(define_float_constants): Remove initialization of snan.
	(Target::_init): Update for new front-end interface.
	(Target::isVectorTypeSupported): Likewise.
	(Target::isVectorOpSupported): Remove cases for unordered operators.
	(TargetCPP::typeMangle): Update for new front-end interface.
	(TargetCPP::parameterType): Likewise.
	(Target::systemLinkage): Likewise.
	(Target::isReturnOnStack): Likewise.
	(Target::isCalleeDestroyingArgs): Define.
	(Target::preferPassByRef): Define.
	* d-tree.h (d_add_entrypoint_module): Remove.
	* decl.cc (gcc_attribute_p): Update for new front-end interface.
	(apply_pragma_crt): Define.
	(DeclVisitor::visit(PragmaDeclaration *)): Handle pragmas
	crt_constructor and crt_destructor.
	(DeclVisitor::visit(TemplateDeclaration *)): Update for new front-end
	interface.
	(DeclVisitor::visit): Likewise.
	(DeclVisitor::finish_vtable): Likewise.
	(get_symbol_decl): Error if template has more than one nesting
	context.  Update for new front-end interface.
	(make_thunk): Update for new front-end interface.
	(get_vtable_decl): Likewise.
	* expr.cc (ExprVisitor::visit): Likewise.
	(build_return_dtor): Likewise.
	* imports.cc (ImportVisitor::visit): Likewise.
	* intrinsics.cc: Include dmd/expression.h.  Remove include of
	dmd/mangle.h.
	(maybe_set_intrinsic): Update for new front-end interface.
	* intrinsics.def (INTRINSIC_ROL): Update intrinsic signature.
	(INTRINSIC_ROR): Likewise.
	(INTRINSIC_ROR_TIARG): Likewise.
	(INTRINSIC_TOPREC): Likewise.
	(INTRINSIC_TOPRECL): Likewise.
	(INTRINSIC_TAN): Update intrinsic module and signature.
	(INTRINSIC_ISNAN): Likewise.
	(INTRINSIC_ISFINITE): Likewise.
	(INTRINSIC_COPYSIGN): Define intrinsic.
	(INTRINSIC_COPYSIGNI): Define intrinsic.
	(INTRINSIC_EXP): Update intrinsic module.
	(INTRINSIC_EXPM1): Likewise.
	(INTRINSIC_EXP2): Likewise.
	(INTRINSIC_LOG): Likewise.
	(INTRINSIC_LOG2): Likewise.
	(INTRINSIC_LOG10): Likewise.
	(INTRINSIC_POW): Likewise.
	(INTRINSIC_ROUND): Likewise.
	(INTRINSIC_FLOORF): Likewise.
	(INTRINSIC_FLOOR): Likewise.
	(INTRINSIC_FLOORL): Likewise.
	(INTRINSIC_CEILF): Likewise.
	(INTRINSIC_CEIL): Likewise.
	(INTRINSIC_CEILL): Likewise.
	(INTRINSIC_TRUNC): Likewise.
	(INTRINSIC_FMIN): Likewise.
	(INTRINSIC_FMAX): Likewise.
	(INTRINSIC_FMA): Likewise.
	(INTRINSIC_VA_ARG): Update intrinsic signature.
	(INTRINSIC_VASTART): Likewise.
	* lang.opt (fcheck=): Add alternate aliases for contract switches.
	(fcheckaction=): New option.
	(check_action): New Enum and EnumValue entries.
	(fdump-c++-spec-verbose): New option.
	(fdump-c++-spec=): New option.
	(fextern-std=): New option.
	(extern_stdcpp): New Enum and EnumValue entries
	(fpreview=): New options.
	(frevert=): New options.
	(fsave-mixins): New option.
	(ftransition=): Update options.
	* modules.cc (get_internal_fn): Replace Prot with Visibility.
	(build_internal_fn): Likewise.
	(build_dso_cdtor_fn): Likewise.
	(build_module_tree): Remove check for __entrypoint module.
	* runtime.def (P5): Define.
	(ARRAYBOUNDS_SLICEP): Define.
	(ARRAYBOUNDS_INDEXP): Define.
	(NEWTHROW): Define.
	(ADCMP2): Remove.
	(ARRAYCAST): Remove.
	(SWITCH_STRING): Remove.
	(SWITCH_USTRING): Remove.
	(SWITCH_DSTRING): Remove.
	(SWITCH_ERROR): Remove.
	* toir.cc (IRVisitor::visit): Update for new front-end interface.
	(IRVisitor::check_previous_goto): Remove checks for case and default
	statements.
	(IRVisitor::visit(SwitchStatement *)): Remove handling of string
	switch conditions.
	* typeinfo.cc: Include d-frontend.h.
	(get_typeinfo_kind): Update for new front-end interface.
	(make_frontend_typeinfo): Likewise.
	(TypeInfoVisitor::visit): Likewise.
	(builtin_typeinfo_p): Likewise.
	(get_typeinfo_decl): Likewise.
	(build_typeinfo): Likewise.
	* types.cc (valist_array_p): Likewise.
	(make_array_type): Likewise.
	(merge_aggregate_types): Likewise.
	(TypeVisitor::visit(TypeBasic *)): Likewise.
	(TypeVisitor::visit(TypeFunction *)): Likewise.
	(TypeVisitor::visit(TypeStruct *)): Update comment.
	* verstr.h: Removed.
	* d-frontend.h: New file.

gcc/po/ChangeLog:

	* EXCLUDES: Remove d/dmd sources from list.

gcc/testsuite/ChangeLog:

	* gdc.dg/Wcastresult2.d: Update test.
	* gdc.dg/asm1.d: Likewise.
	* gdc.dg/asm2.d: Likewise.
	* gdc.dg/asm3.d: Likewise.
	* gdc.dg/gdc282.d: Likewise.
	* gdc.dg/imports/gdc170.d: Likewise.
	* gdc.dg/intrinsics.d: Likewise.
	* gdc.dg/pr101672.d: Likewise.
	* gdc.dg/pr90650a.d: Likewise.
	* gdc.dg/pr90650b.d: Likewise.
	* gdc.dg/pr94777a.d: Likewise.
	* gdc.dg/pr95250.d: Likewise.
	* gdc.dg/pr96869.d: Likewise.
	* gdc.dg/pr98277.d: Likewise.
	* gdc.dg/pr98457.d: Likewise.
	* gdc.dg/simd1.d: Likewise.
	* gdc.dg/simd2a.d: Likewise.
	* gdc.dg/simd2b.d: Likewise.
	* gdc.dg/simd2c.d: Likewise.
	* gdc.dg/simd2d.d: Likewise.
	* gdc.dg/simd2e.d: Likewise.
	* gdc.dg/simd2f.d: Likewise.
	* gdc.dg/simd2g.d: Likewise.
	* gdc.dg/simd2h.d: Likewise.
	* gdc.dg/simd2i.d: Likewise.
	* gdc.dg/simd2j.d: Likewise.
	* gdc.dg/simd7951.d: Likewise.
	* gdc.dg/torture/gdc309.d: Likewise.
	* gdc.dg/torture/pr94424.d: Likewise.
	* gdc.dg/torture/pr94777b.d: Likewise.
	* lib/gdc-utils.exp (gdc-convert-args): Handle new compiler options.
	(gdc-convert-test): Handle CXXFLAGS, EXTRA_OBJC_SOURCES, and ARG_SETS
	test directives.
	(gdc-do-test): Only import modules in the test run directory.
	* gdc.dg/pr94777c.d: New test.
	* gdc.dg/pr96156b.d: New test.
	* gdc.dg/pr96157c.d: New test.
	* gdc.dg/simd_ctfe.d: New test.
	* gdc.dg/torture/simd17344.d: New test.
	* gdc.dg/torture/simd20052.d: New test.
	* gdc.dg/torture/simd6.d: New test.
	* gdc.dg/torture/simd7.d: New test.

libphobos/ChangeLog:

	* libdruntime/MERGE: Merge upstream druntime e6caaab9.
	* libdruntime/Makefile.am (D_EXTRA_FLAGS): Build libdruntime with
	-fpreview=dip1000, -fpreview=fieldwise, and -fpreview=dtorfields.
	(ALL_DRUNTIME_SOURCES): Add DRUNTIME_DSOURCES_STDCXX.
	(DRUNTIME_DSOURCES): Update list of C binding modules.
	(DRUNTIME_DSOURCES_STDCXX): Likewise.
	(DRUNTIME_DSOURCES_LINUX): Likewise.
	(DRUNTIME_DSOURCES_OPENBSD): Likewise.
	(DRUNTIME_DISOURCES): Remove __entrypoint.di.
	* libdruntime/Makefile.in: Regenerated.
	* libdruntime/__entrypoint.di: Removed.
	* libdruntime/gcc/deh.d (_d_isbaseof): Update signature.
	(_d_createTrace): Likewise.
	(__gdc_begin_catch): Remove reference to the exception.
	(_d_throw): Increment reference count of thrown object before unwind.
	(__gdc_personality): Chain exceptions with  Throwable.chainTogether.
	* libdruntime/gcc/emutls.d: Update imports.
	* libdruntime/gcc/sections/elf.d: Update imports.
	(DSO.moduleGroup): Update signature.
	* libdruntime/gcc/sections/macho.d: Update imports.
	(DSO.moduleGroup): Update signature.
	* libdruntime/gcc/sections/pecoff.d: Update imports.
	(DSO.moduleGroup): Update signature.
	* src/MERGE: Merge upstream phobos 5ab9ad256.
	* src/Makefile.am (D_EXTRA_DFLAGS): Add -fpreview=dip1000 and
	-fpreview=dtorfields flags.
	(PHOBOS_DSOURCES): Update list of std modules.
	* src/Makefile.in: Regenerate.
	* testsuite/lib/libphobos.exp (libphobos-dg-test): Handle assembly
	compile types.
	(dg-test): Override.
	(additional_prunes): Define.
	(libphobos-dg-prune): Filter any additional_prunes set by tests.
	* testsuite/libphobos.aa/test_aa.d: Update test.
	* testsuite/libphobos.druntime/druntime.exp (version_flags): Add
	-fversion=CoreUnittest.
	* testsuite/libphobos.druntime_shared/druntime_shared.exp
	(version_flags): Add -fversion=CoreUnittest -fversion=Shared.
	* testsuite/libphobos.exceptions/unknown_gc.d: Update test.
	* testsuite/libphobos.hash/test_hash.d: Update test.
	* testsuite/libphobos.phobos/phobos.exp (version_flags): Add
	-fversion=StdUnittest
	* testsuite/libphobos.phobos_shared/phobos_shared.exp (version_flags):
	Likewise.
	* testsuite/libphobos.shared/host.c: Update test.
	* testsuite/libphobos.shared/load.d: Update test.
	* testsuite/libphobos.shared/load_13414.d: Update test.
	* testsuite/libphobos.thread/fiber_guard_page.d: Update test.
	* testsuite/libphobos.thread/tlsgc_sections.d: Update test.
	* testsuite/testsuite_flags.in: Add -fpreview=dip1000 to --gdcflags.
	* testsuite/libphobos.shared/link_mod_collision.d: Removed.
	* testsuite/libphobos.shared/load_mod_collision.d: Removed.
	* testsuite/libphobos.betterc/betterc.exp: New test.
	* testsuite/libphobos.config/config.exp: New test.
	* testsuite/libphobos.gc/gc.exp: New test.
	* testsuite/libphobos.imports/imports.exp: New test.
	* testsuite/libphobos.lifetime/lifetime.exp: New test.
	* testsuite/libphobos.unittest/unittest.exp: New test.
2021-11-30 16:53:28 +01:00

890 lines
21 KiB
D

/**********************************************
* This module implements integral arithmetic primitives that check
* for out-of-range results.
*
* Integral arithmetic operators operate on fixed width types.
* Results that are not representable in those fixed widths are silently
* truncated to fit.
* This module offers integral arithmetic primitives that produce the
* same results, but set an 'overflow' flag when such truncation occurs.
* The setting is sticky, meaning that numerous operations can be cascaded
* and then the flag need only be checked at the end.
* Whether the operation is signed or unsigned is indicated by an 's' or 'u'
* suffix, respectively. While this could be achieved without such suffixes by
* using overloading on the signedness of the types, the suffix makes it clear
* which is happening without needing to examine the types.
*
* While the generic versions of these functions are computationally expensive
* relative to the cost of the operation itself, compiler implementations are free
* to recognize them and generate equivalent and faster code.
*
* The functions here are templates so they can be used with -betterC,
* as betterC does not link with this library.
*
* References: $(LINK2 http://blog.regehr.org/archives/1139, Fast Integer Overflow Checks)
* Copyright: Copyright (c) Walter Bright 2014.
* License: $(LINK2 http://www.boost.org/LICENSE_1_0.txt, Boost License 1.0)
* Authors: Walter Bright
* Source: $(DRUNTIMESRC core/_checkedint.d)
*/
module core.checkedint;
import core.internal.attributes : betterC;
nothrow:
@safe:
@nogc:
pure:
/*******************************
* Add two signed integers, checking for overflow.
*
* The overflow is sticky, meaning a sequence of operations can
* be done and overflow need only be checked at the end.
* Params:
* x = left operand
* y = right operand
* overflow = set if an overflow occurs, is not affected otherwise
* Returns:
* the sum
*/
pragma(inline, true)
int adds()(int x, int y, ref bool overflow)
{
long r = cast(long)x + cast(long)y;
if (r < int.min || r > int.max)
overflow = true;
return cast(int)r;
}
///
@betterC
unittest
{
bool overflow;
assert(adds(2, 3, overflow) == 5);
assert(!overflow);
assert(adds(1, int.max - 1, overflow) == int.max);
assert(!overflow);
assert(adds(int.min + 1, -1, overflow) == int.min);
assert(!overflow);
assert(adds(int.max, 1, overflow) == int.min);
assert(overflow);
overflow = false;
assert(adds(int.min, -1, overflow) == int.max);
assert(overflow);
assert(adds(0, 0, overflow) == 0);
assert(overflow); // sticky
}
/// ditto
pragma(inline, true)
long adds()(long x, long y, ref bool overflow)
{
long r = cast(ulong)x + cast(ulong)y;
if (x < 0 && y < 0 && r >= 0 ||
x >= 0 && y >= 0 && r < 0)
overflow = true;
return r;
}
///
@betterC
unittest
{
bool overflow;
assert(adds(2L, 3L, overflow) == 5);
assert(!overflow);
assert(adds(1L, long.max - 1, overflow) == long.max);
assert(!overflow);
assert(adds(long.min + 1, -1, overflow) == long.min);
assert(!overflow);
assert(adds(long.max, 1, overflow) == long.min);
assert(overflow);
overflow = false;
assert(adds(long.min, -1, overflow) == long.max);
assert(overflow);
assert(adds(0L, 0L, overflow) == 0);
assert(overflow); // sticky
}
static if (is(cent))
{
/// ditto
pragma(inline, true)
cent adds()(cent x, cent y, ref bool overflow)
{
cent r = cast(ucent)x + cast(ucent)y;
if (x < 0 && y < 0 && r >= 0 ||
x >= 0 && y >= 0 && r < 0)
overflow = true;
return r;
}
unittest
{
bool overflow;
assert(adds(cast(cent)2L, 3L, overflow) == 5);
assert(!overflow);
assert(adds(1L, cent.max - 1, overflow) == cent.max);
assert(!overflow);
assert(adds(cent.min + 1, -1, overflow) == cent.min);
assert(!overflow);
assert(adds(cent.max, 1, overflow) == cent.min);
assert(overflow);
overflow = false;
assert(adds(cent.min, -1, overflow) == cent.max);
assert(overflow);
assert(adds(cast(cent)0L, 0L, overflow) == 0);
assert(overflow); // sticky
}
}
/*******************************
* Add two unsigned integers, checking for overflow (aka carry).
*
* The overflow is sticky, meaning a sequence of operations can
* be done and overflow need only be checked at the end.
* Params:
* x = left operand
* y = right operand
* overflow = set if an overflow occurs, is not affected otherwise
* Returns:
* the sum
*/
pragma(inline, true)
uint addu()(uint x, uint y, ref bool overflow)
{
immutable uint r = x + y;
immutable bool o = r < x;
assert(o == (r < y));
if (o)
overflow = true;
return r;
}
///
@betterC
unittest
{
for (uint i = 0; i < 10; ++i)
{
bool overflow;
immutable uint r = addu (uint.max - i, uint.max - i, overflow);
assert (r == 2 * (uint.max - i));
assert (overflow);
}
bool overflow;
assert(addu(2, 3, overflow) == 5);
assert(!overflow);
assert(addu(1, uint.max - 1, overflow) == uint.max);
assert(!overflow);
assert(addu(uint.min, -1, overflow) == uint.max);
assert(!overflow);
assert(addu(uint.max, 1, overflow) == uint.min);
assert(overflow);
overflow = false;
assert(addu(uint.min + 1, -1, overflow) == uint.min);
assert(overflow);
assert(addu(0, 0, overflow) == 0);
assert(overflow); // sticky
}
/// ditto
pragma(inline, true)
ulong addu()(ulong x, ulong y, ref bool overflow)
{
immutable ulong r = x + y;
immutable bool o = r < x;
assert(o == (r < y));
if (o)
overflow = true;
return r;
}
///
@betterC
unittest
{
bool overflow;
assert(addu(2L, 3L, overflow) == 5);
assert(!overflow);
assert(addu(1, ulong.max - 1, overflow) == ulong.max);
assert(!overflow);
assert(addu(ulong.min, -1L, overflow) == ulong.max);
assert(!overflow);
assert(addu(ulong.max, 1, overflow) == ulong.min);
assert(overflow);
overflow = false;
assert(addu(ulong.min + 1, -1L, overflow) == ulong.min);
assert(overflow);
assert(addu(0L, 0L, overflow) == 0);
assert(overflow); // sticky
}
static if (is(ucent))
{
/// ditto
pragma(inline, true)
ucent addu()(ucent x, ucent y, ref bool overflow)
{
immutable ucent r = x + y;
immutable bool o = r < x;
assert(o == (r < y));
if (o)
overflow = true;
return r;
}
unittest
{
bool overflow;
assert(addu(cast(ucent)2L, 3L, overflow) == 5);
assert(!overflow);
assert(addu(1, ucent.max - 1, overflow) == ucent.max);
assert(!overflow);
assert(addu(ucent.min, -1L, overflow) == ucent.max);
assert(!overflow);
assert(addu(ucent.max, 1, overflow) == ucent.min);
assert(overflow);
overflow = false;
assert(addu(ucent.min + 1, -1L, overflow) == ucent.min);
assert(overflow);
assert(addu(cast(ucent)0L, 0L, overflow) == 0);
assert(overflow); // sticky
}
}
/*******************************
* Subtract two signed integers, checking for overflow.
*
* The overflow is sticky, meaning a sequence of operations can
* be done and overflow need only be checked at the end.
* Params:
* x = left operand
* y = right operand
* overflow = set if an overflow occurs, is not affected otherwise
* Returns:
* the difference
*/
pragma(inline, true)
int subs()(int x, int y, ref bool overflow)
{
immutable long r = cast(long)x - cast(long)y;
if (r < int.min || r > int.max)
overflow = true;
return cast(int)r;
}
///
@betterC
unittest
{
bool overflow;
assert(subs(2, -3, overflow) == 5);
assert(!overflow);
assert(subs(1, -int.max + 1, overflow) == int.max);
assert(!overflow);
assert(subs(int.min + 1, 1, overflow) == int.min);
assert(!overflow);
assert(subs(int.max, -1, overflow) == int.min);
assert(overflow);
overflow = false;
assert(subs(int.min, 1, overflow) == int.max);
assert(overflow);
assert(subs(0, 0, overflow) == 0);
assert(overflow); // sticky
}
/// ditto
pragma(inline, true)
long subs()(long x, long y, ref bool overflow)
{
immutable long r = cast(ulong)x - cast(ulong)y;
if (x < 0 && y >= 0 && r >= 0 ||
x >= 0 && y < 0 && (r < 0 || y == long.min))
overflow = true;
return r;
}
///
@betterC
unittest
{
bool overflow;
assert(subs(2L, -3L, overflow) == 5);
assert(!overflow);
assert(subs(1L, -long.max + 1, overflow) == long.max);
assert(!overflow);
assert(subs(long.min + 1, 1, overflow) == long.min);
assert(!overflow);
assert(subs(-1L, long.min, overflow) == long.max);
assert(!overflow);
assert(subs(long.max, -1, overflow) == long.min);
assert(overflow);
overflow = false;
assert(subs(long.min, 1, overflow) == long.max);
assert(overflow);
assert(subs(0L, 0L, overflow) == 0);
assert(overflow); // sticky
}
static if (is(cent))
{
/// ditto
pragma(inline, true)
cent subs()(cent x, cent y, ref bool overflow)
{
immutable cent r = cast(ucent)x - cast(ucent)y;
if (x < 0 && y >= 0 && r >= 0 ||
x >= 0 && y < 0 && (r < 0 || y == long.min))
overflow = true;
return r;
}
unittest
{
bool overflow;
assert(subs(cast(cent)2L, -3L, overflow) == 5);
assert(!overflow);
assert(subs(1L, -cent.max + 1, overflow) == cent.max);
assert(!overflow);
assert(subs(cent.min + 1, 1, overflow) == cent.min);
assert(!overflow);
assert(subs(-1L, cent.min, overflow) == cent.max);
assert(!overflow);
assert(subs(cent.max, -1, overflow) == cent.min);
assert(overflow);
overflow = false;
assert(subs(cent.min, 1, overflow) == cent.max);
assert(overflow);
assert(subs(cast(cent)0L, 0L, overflow) == 0);
assert(overflow); // sticky
}
}
/*******************************
* Subtract two unsigned integers, checking for overflow (aka borrow).
*
* The overflow is sticky, meaning a sequence of operations can
* be done and overflow need only be checked at the end.
* Params:
* x = left operand
* y = right operand
* overflow = set if an overflow occurs, is not affected otherwise
* Returns:
* the difference
*/
pragma(inline, true)
uint subu()(uint x, uint y, ref bool overflow)
{
if (x < y)
overflow = true;
return x - y;
}
///
@betterC
unittest
{
bool overflow;
assert(subu(3, 2, overflow) == 1);
assert(!overflow);
assert(subu(uint.max, 1, overflow) == uint.max - 1);
assert(!overflow);
assert(subu(1, 1, overflow) == uint.min);
assert(!overflow);
assert(subu(0, 1, overflow) == uint.max);
assert(overflow);
overflow = false;
assert(subu(uint.max - 1, uint.max, overflow) == uint.max);
assert(overflow);
assert(subu(0, 0, overflow) == 0);
assert(overflow); // sticky
}
/// ditto
pragma(inline, true)
ulong subu()(ulong x, ulong y, ref bool overflow)
{
if (x < y)
overflow = true;
return x - y;
}
///
@betterC
unittest
{
bool overflow;
assert(subu(3UL, 2UL, overflow) == 1);
assert(!overflow);
assert(subu(ulong.max, 1, overflow) == ulong.max - 1);
assert(!overflow);
assert(subu(1UL, 1UL, overflow) == ulong.min);
assert(!overflow);
assert(subu(0UL, 1UL, overflow) == ulong.max);
assert(overflow);
overflow = false;
assert(subu(ulong.max - 1, ulong.max, overflow) == ulong.max);
assert(overflow);
assert(subu(0UL, 0UL, overflow) == 0);
assert(overflow); // sticky
}
static if (is(ucent))
{
/// ditto
pragma(inline, true)
ucent subu()(ucent x, ucent y, ref bool overflow)
{
if (x < y)
overflow = true;
return x - y;
}
unittest
{
bool overflow;
assert(subu(cast(ucent)3UL, 2UL, overflow) == 1);
assert(!overflow);
assert(subu(ucent.max, 1, overflow) == ucent.max - 1);
assert(!overflow);
assert(subu(1UL, 1UL, overflow) == ucent.min);
assert(!overflow);
assert(subu(cast(ucent)0UL, 1UL, overflow) == ucent.max);
assert(overflow);
overflow = false;
assert(subu(ucent.max - 1, ucent.max, overflow) == ucent.max);
assert(overflow);
assert(subu(cast(ucent)0UL, 0UL, overflow) == 0);
assert(overflow); // sticky
}
}
/***********************************************
* Negate an integer.
*
* Params:
* x = operand
* overflow = set if x cannot be negated, is not affected otherwise
* Returns:
* the negation of x
*/
pragma(inline, true)
int negs()(int x, ref bool overflow)
{
if (x == int.min)
overflow = true;
return -x;
}
///
@betterC
unittest
{
bool overflow;
assert(negs(0, overflow) == -0);
assert(!overflow);
assert(negs(1234, overflow) == -1234);
assert(!overflow);
assert(negs(-5678, overflow) == 5678);
assert(!overflow);
assert(negs(int.min, overflow) == -int.min);
assert(overflow);
assert(negs(0, overflow) == -0);
assert(overflow); // sticky
}
/// ditto
pragma(inline, true)
long negs()(long x, ref bool overflow)
{
if (x == long.min)
overflow = true;
return -x;
}
///
@betterC
unittest
{
bool overflow;
assert(negs(0L, overflow) == -0);
assert(!overflow);
assert(negs(1234L, overflow) == -1234);
assert(!overflow);
assert(negs(-5678L, overflow) == 5678);
assert(!overflow);
assert(negs(long.min, overflow) == -long.min);
assert(overflow);
assert(negs(0L, overflow) == -0);
assert(overflow); // sticky
}
static if (is(cent))
{
/// ditto
pragma(inline, true)
cent negs()(cent x, ref bool overflow)
{
if (x == cent.min)
overflow = true;
return -x;
}
unittest
{
bool overflow;
assert(negs(cast(cent)0L, overflow) == -0);
assert(!overflow);
assert(negs(cast(cent)1234L, overflow) == -1234);
assert(!overflow);
assert(negs(cast(cent)-5678L, overflow) == 5678);
assert(!overflow);
assert(negs(cent.min, overflow) == -cent.min);
assert(overflow);
assert(negs(cast(cent)0L, overflow) == -0);
assert(overflow); // sticky
}
}
/*******************************
* Multiply two signed integers, checking for overflow.
*
* The overflow is sticky, meaning a sequence of operations can
* be done and overflow need only be checked at the end.
* Params:
* x = left operand
* y = right operand
* overflow = set if an overflow occurs, is not affected otherwise
* Returns:
* the product
*/
pragma(inline, true)
int muls()(int x, int y, ref bool overflow)
{
long r = cast(long)x * cast(long)y;
if (r < int.min || r > int.max)
overflow = true;
return cast(int)r;
}
///
@betterC
unittest
{
bool overflow;
assert(muls(2, 3, overflow) == 6);
assert(!overflow);
assert(muls(-200, 300, overflow) == -60_000);
assert(!overflow);
assert(muls(1, int.max, overflow) == int.max);
assert(!overflow);
assert(muls(int.min, 1, overflow) == int.min);
assert(!overflow);
assert(muls(int.max, 2, overflow) == (int.max * 2));
assert(overflow);
overflow = false;
assert(muls(int.min, -1, overflow) == int.min);
assert(overflow);
assert(muls(0, 0, overflow) == 0);
assert(overflow); // sticky
}
/// ditto
pragma(inline, true)
long muls()(long x, long y, ref bool overflow)
{
immutable long r = cast(ulong)x * cast(ulong)y;
enum not0or1 = ~1L;
if ((x & not0or1) &&
((r == y) ? r != 0
: (r == 0x8000_0000_0000_0000 && x == -1L) || ((r / x) != y)))
overflow = true;
return r;
}
///
@betterC
unittest
{
bool overflow;
assert(muls(2L, 3L, overflow) == 6);
assert(!overflow);
assert(muls(-200L, 300L, overflow) == -60_000);
assert(!overflow);
assert(muls(1, long.max, overflow) == long.max);
assert(!overflow);
assert(muls(long.min, 1L, overflow) == long.min);
assert(!overflow);
assert(muls(long.max, 2L, overflow) == (long.max * 2));
assert(overflow);
overflow = false;
assert(muls(-1L, long.min, overflow) == long.min);
assert(overflow);
overflow = false;
assert(muls(long.min, -1L, overflow) == long.min);
assert(overflow);
assert(muls(0L, 0L, overflow) == 0);
assert(overflow); // sticky
}
static if (is(cent))
{
/// ditto
pragma(inline, true)
cent muls()(cent x, cent y, ref bool overflow)
{
immutable cent r = cast(ucent)x * cast(ucent)y;
enum not0or1 = ~1L;
if ((x & not0or1) && ((r == y)? r : (r / x) != y))
overflow = true;
return r;
}
unittest
{
bool overflow;
assert(muls(cast(cent)2L, 3L, overflow) == 6);
assert(!overflow);
assert(muls(cast(cent)-200L, 300L, overflow) == -60_000);
assert(!overflow);
assert(muls(1, cent.max, overflow) == cent.max);
assert(!overflow);
assert(muls(cent.min, 1L, overflow) == cent.min);
assert(!overflow);
assert(muls(cent.max, 2L, overflow) == (cent.max * 2));
assert(overflow);
overflow = false;
assert(muls(-1L, cent.min, overflow) == cent.min);
assert(overflow);
overflow = false;
assert(muls(cent.min, -1L, overflow) == cent.min);
assert(overflow);
assert(muls(cast(cent)0L, 0L, overflow) == 0);
assert(overflow); // sticky
}
}
/*******************************
* Multiply two unsigned integers, checking for overflow (aka carry).
*
* The overflow is sticky, meaning a sequence of operations can
* be done and overflow need only be checked at the end.
* Params:
* x = left operand
* y = right operand
* overflow = set if an overflow occurs, is not affected otherwise
* Returns:
* the product
*/
pragma(inline, true)
uint mulu()(uint x, uint y, ref bool overflow)
{
immutable ulong r = ulong(x) * ulong(y);
if (r >> 32)
overflow = true;
return cast(uint) r;
}
@betterC
unittest
{
void test(uint x, uint y, uint r, bool overflow) @nogc nothrow
{
bool o;
assert(mulu(x, y, o) == r);
assert(o == overflow);
}
test(2, 3, 6, false);
test(1, uint.max, uint.max, false);
test(0, 1, 0, false);
test(0, uint.max, 0, false);
test(uint.max, 2, 2 * uint.max, true);
test(1 << 16, 1U << 16, 0, true);
bool overflow = true;
assert(mulu(0, 0, overflow) == 0);
assert(overflow); // sticky
}
/// ditto
pragma(inline, true)
ulong mulu()(ulong x, uint y, ref bool overflow)
{
ulong r = x * y;
if (x >> 32 &&
r / x != y)
overflow = true;
return r;
}
/// ditto
pragma(inline, true)
ulong mulu()(ulong x, ulong y, ref bool overflow)
{
immutable ulong r = x * y;
if ((x | y) >> 32 &&
x &&
r / x != y)
overflow = true;
return r;
}
@betterC
unittest
{
void test(T, U)(T x, U y, ulong r, bool overflow) @nogc nothrow
{
bool o;
assert(mulu(x, y, o) == r);
assert(o == overflow);
}
// One operand is zero
test(0, 3, 0, false);
test(0UL, 3, 0, false);
test(0UL, 3UL, 0, false);
test(3, 0, 0, false);
test(3UL, 0, 0, false);
test(3UL, 0UL, 0, false);
// Small numbers
test(2, 3, 6, false);
test(2UL, 3, 6, false);
test(2UL, 3UL, 6, false);
// At the 32/64 border
test(1, ulong(uint.max), uint.max, false);
test(1UL, ulong(uint.max), uint.max, false);
test(ulong(uint.max), 1, uint.max, false);
test(ulong(uint.max), 1UL, uint.max, false);
test(1, 1 + ulong(uint.max), 1 + ulong(uint.max), false);
test(1UL, 1 + ulong(uint.max), 1 + ulong(uint.max), false);
test(1 + ulong(uint.max), 1, 1 + ulong(uint.max), false);
test(1 + ulong(uint.max), 1UL, 1 + ulong(uint.max), false);
// At the limit
test(1, ulong.max, ulong.max, false);
test(1UL, ulong.max, ulong.max, false);
test(ulong.max, 1, ulong.max, false);
test(ulong.max, 1UL, ulong.max, false);
// Miscellaneous
test(0, 1, 0, false);
test(0, ulong.max, 0, false);
test(ulong.max, 2, 2 * ulong.max, true);
test(1UL << 32, 1UL << 32, 0, true);
// Must be sticky
bool overflow = true;
assert(mulu(0UL, 0UL, overflow) == 0);
assert(overflow); // sticky
}
static if (is(ucent))
{
/// ditto
pragma(inline, true)
ucent mulu()(ucent x, ucent y, ref bool overflow)
{
immutable ucent r = x * y;
if (x && (r / x) != y)
overflow = true;
return r;
}
unittest
{
void test(ucent x, ucent y, ucent r, bool overflow) @nogc nothrow
{
bool o;
assert(mulu(x, y, o) == r);
assert(o == overflow);
}
test(2, 3, 6, false);
test(1, ucent.max, ucent.max, false);
test(0, 1, 0, false);
test(0, ucent.max, 0, false);
test(ucent.max, 2, 2 * ucent.max, true);
test(cast(ucent)1UL << 64, cast(ucent)1UL << 64, 0, true);
bool overflow = true;
assert(mulu(0UL, 0UL, overflow) == 0);
assert(overflow); // sticky
}
}