]> www.ginac.de Git - cln.git/commitdiff
Replace CL_REQUIRE/CL_PROVIDE(cl_RA_ring) with portable code.
authorAlexei Sheplyakov <varg@theor.jinr.ru>
Thu, 21 Aug 2008 11:55:45 +0000 (15:55 +0400)
committerAlexei Sheplyakov <varg@theor.jinr.ru>
Wed, 27 Aug 2008 04:41:05 +0000 (08:41 +0400)
The order of initialization of non-local objects in different compilation units
is not specified in C++. Hence special care should be taken to avoid static
initialization order fiasco. CLN solved the problem with some evil (GCC
specific, and even GCC-version-specific) hack. Replace it with a technique
similar to one used in STL to initialize std::cout and friends.

include/cln/rational_ring.h
include/cln/univpoly_rational.h
src/rational/ring/cl_RA_ring.cc

index b6026cc929417854f0dbfaf0bad58b45ecd3160e..0a3dbfd2e193dcbe400ce1301897381e87436923 100644 (file)
@@ -11,7 +11,15 @@ namespace cln {
 typedef cl_specialized_number_ring<cl_RA> cl_rational_ring;
 extern const cl_rational_ring cl_RA_ring;      // math. Q
 extern cl_class cl_class_rational_ring;
-//CL_REQUIRE(cl_RA_ring)
+
+class cl_RA_ring_init_helper
+{
+       static int count;
+public:
+       cl_RA_ring_init_helper();
+       ~cl_RA_ring_init_helper();
+};
+static cl_RA_ring_init_helper cl_RA_ring_init_helper_instance;
 
 }  // namespace cln
 
index a1a6283f38dcaa9ff1a4f33d42667e393fa7aadc..b95824505e81e2c66ec44fc9857ce804e8237f27 100644 (file)
@@ -223,8 +223,6 @@ inline const cl_UP_RA deriv (const cl_UP_RA& x)
 
 #endif
 
-CL_REQUIRE(cl_RA_ring)
-
 
 // Returns the n-th Legendre polynomial (n >= 0).
 extern const cl_UP_RA legendre (sintL n);
index f23035f974e690ef5f7826ecbcb49eafd3b465d7..b42130fd9b45c2a9a6cc1dee751d92033202255b 100644 (file)
@@ -3,8 +3,6 @@
 // General includes.
 #include "cl_sysdep.h"
 
-CL_PROVIDE(cl_RA_ring)
-
 // Specification.
 #include "cln/rational_ring.h"
 
@@ -143,19 +141,34 @@ static void cl_rational_ring_dprint (cl_heap* pointer)
        fprint(cl_debugout, "(cl_rational_ring) cl_RA_ring");
 }
 
-cl_class cl_class_rational_ring = {
-       cl_rational_ring_destructor,
-       cl_class_flags_number_ring,
-       cl_rational_ring_dprint
-};
+cl_class cl_class_rational_ring;
+static cl_heap_rational_ring* cl_heap_rational_ring_instance;
 
 // Constructor.
 template <>
 inline cl_rational_ring::cl_specialized_number_ring ()
-       : cl_number_ring (new cl_heap_rational_ring()) {}
+       : cl_number_ring(cl_heap_rational_ring_instance) { }
 
-const cl_rational_ring cl_RA_ring;
+const cl_rational_ring cl_RA_ring = cl_RA_ring;
 
-}  // namespace cln
+int cl_RA_ring_init_helper::count = 0;
+
+cl_RA_ring_init_helper::cl_RA_ring_init_helper()
+{
+       if (count++ == 0) {
+               cl_class_rational_ring.destruct = cl_rational_ring_destructor;
+               cl_class_rational_ring.flags = cl_class_flags_number_ring;
+               cl_class_rational_ring.dprint = cl_rational_ring_dprint;
+               cl_heap_rational_ring_instance = new cl_heap_rational_ring();
+               new ((void *)&cl_RA_ring) cl_rational_ring();
+       }
+}
 
-CL_PROVIDE_END(cl_RA_ring)
+cl_RA_ring_init_helper::~cl_RA_ring_init_helper()
+{
+       if (--count == 0) {
+               delete cl_heap_rational_ring_instance;
+       }
+}
+
+}  // namespace cln