X-Git-Url: https://www.ginac.de/ginac.git//ginac.git?p=ginac.git;a=blobdiff_plain;f=ginac%2Fbasic.h;h=fd6a74718032a18a0c5328b3a7967cfd2aefab8e;hp=0295d015920e1807464d51c5df309e25abd82a7d;hb=74bc832973cda4204e349396c9f4f7d056f16366;hpb=852fb174c43b49a3ce1b568f959d23e8a1959ee1 diff --git a/ginac/basic.h b/ginac/basic.h index 0295d015..fd6a7471 100644 --- a/ginac/basic.h +++ b/ginac/basic.h @@ -3,7 +3,7 @@ * Interface to GiNaC's ABC. */ /* - * GiNaC Copyright (C) 1999-2015 Johannes Gutenberg University Mainz, Germany + * GiNaC Copyright (C) 1999-2020 Johannes Gutenberg University Mainz, Germany * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by @@ -28,13 +28,12 @@ #include "assertion.h" #include "registrar.h" -// CINT needs to work properly with -#include #include // for size_t #include #include #include // for typeid #include +#include namespace GiNaC { @@ -93,7 +92,8 @@ struct map_function { /** Degenerate base class for visitors. basic and derivative classes * support Robert C. Martin's Acyclic Visitor pattern (cf. - * http://objectmentor.com/publications/acv.pdf). */ + * http://condor.depaul.edu/dmumaugh/OOT/Design-Principles/acv.pdf + * or chapter 10 of Andrei Alexandrescu's "Modern C++ Design"). */ class visitor { protected: virtual ~visitor() {} @@ -128,11 +128,16 @@ public: // only const functions please (may break reference counting) /** Create a clone of this object on the heap. One can think of this as * simulating a virtual copy constructor which is needed for instance by * the refcounted construction of an ex from a basic. */ - virtual basic * duplicate() const { return new basic(*this); } + virtual basic * duplicate() const + { + basic * bp = new basic(*this); + bp->setflag(status_flags::dynallocated); + return bp; + } // evaluation - virtual ex eval(int level = 0) const; - virtual ex evalf(int level = 0) const; + virtual ex eval() const; + virtual ex evalf() const; virtual ex evalm() const; virtual ex eval_integ() const; protected: @@ -195,7 +200,7 @@ public: virtual ex series(const relational & r, int order, unsigned options = 0) const; // rational functions - virtual ex normal(exmap & repl, exmap & rev_lookup, int level = 0) const; + virtual ex normal(exmap & repl, exmap & rev_lookup) const; virtual ex to_rational(exmap & repl) const; virtual ex to_polynomial(exmap & repl) const; @@ -298,11 +303,8 @@ protected: mutable unsigned hashvalue; ///< hash value }; - // global variables -extern int max_recursion_level; - // convenience type checker template functions @@ -320,6 +322,31 @@ inline bool is_exactly_a(const basic & obj) return typeid(T) == typeid(obj); } +/** Constructs a new (class basic or derived) B object on the heap. + * + * This function picks the object's ctor based on the given argument types. + * + * This helps the constructor of ex from basic (or a derived class B) because + * then the constructor doesn't have to duplicate the object onto the heap. + * See ex::construct_from_basic(const basic &) for more information. + */ +template +inline B & dynallocate(Args &&... args) +{ + return const_cast(static_cast((new B(std::forward(args)...))->setflag(status_flags::dynallocated))); +} +/** Constructs a new (class basic or derived) B object on the heap. + * + * This function is needed for GiNaC classes which have public ctors from + * initializer lists of expressions (which are not a type and not captured + * by the variadic template version). + */ +template +inline B & dynallocate(std::initializer_list il) +{ + return const_cast(static_cast((new B(il))->setflag(status_flags::dynallocated))); +} + } // namespace GiNaC #endif // ndef GINAC_BASIC_H