From 8e258e150743f9d71a7bbd9f70af78a05d354456 Mon Sep 17 00:00:00 2001 From: Boris Kolpackov Date: Tue, 6 Dec 2016 15:54:07 +0200 Subject: Work around wrong static initialization order in VC --- build2/parser.cxx | 24 +++++++++++++----------- build2/variable | 22 ++++++++++++++++++---- build2/variable.txx | 33 ++++++++++++++++++++++++--------- 3 files changed, 55 insertions(+), 24 deletions(-) diff --git a/build2/parser.cxx b/build2/parser.cxx index 87ec8b1..c47cc8b 100644 --- a/build2/parser.cxx +++ b/build2/parser.cxx @@ -1520,18 +1520,20 @@ namespace build2 static const value_type* map_type (const string& n) { + auto ptr = [] (const value_type& vt) {return &vt;}; + return - n == "bool" ? &value_traits::value_type : - n == "uint64" ? &value_traits::value_type : - n == "string" ? &value_traits::value_type : - n == "path" ? &value_traits::value_type : - n == "dir_path" ? &value_traits::value_type : - n == "abs_dir_path" ? &value_traits::value_type : - n == "name" ? &value_traits::value_type : - n == "strings" ? &value_traits::value_type : - n == "paths" ? &value_traits::value_type : - n == "dir_paths" ? &value_traits::value_type : - n == "names" ? &value_traits>::value_type : + n == "bool" ? ptr (value_traits::value_type) : + n == "uint64" ? ptr (value_traits::value_type) : + n == "string" ? ptr (value_traits::value_type) : + n == "path" ? ptr (value_traits::value_type) : + n == "dir_path" ? ptr (value_traits::value_type) : + n == "abs_dir_path" ? ptr (value_traits::value_type) : + n == "name" ? ptr (value_traits::value_type) : + n == "strings" ? ptr (value_traits::value_type) : + n == "paths" ? ptr (value_traits::value_type) : + n == "dir_paths" ? ptr (value_traits::value_type) : + n == "names" ? ptr (value_traits>::value_type) : nullptr; } diff --git a/build2/variable b/build2/variable index 8fc1fb9..0480a6c 100644 --- a/build2/variable +++ b/build2/variable @@ -694,8 +694,15 @@ namespace build2 static void prepend (value&, vector&&); static bool empty (const vector& x) {return x.empty ();} - static const string type_name; - static const build2::value_type value_type; + // Make sure these are static-initialized together. Failed that VC will + // make sure it's done in the wrong order. + // + struct value_type_ex: build2::value_type + { + string type_name; + value_type_ex (value_type&&); + }; + static const value_type_ex value_type; }; // map @@ -713,8 +720,15 @@ namespace build2 return append (v, move (x));} static bool empty (const map& x) {return x.empty ();} - static const string type_name; - static const build2::value_type value_type; + // Make sure these are static-initialized together. Failed that VC will + // make sure it's done in the wrong order. + // + struct value_type_ex: build2::value_type + { + string type_name; + value_type_ex (value_type&&); + }; + static const value_type_ex value_type; }; // variable_pool diff --git a/build2/variable.txx b/build2/variable.txx index d566a04..8db2ea9 100644 --- a/build2/variable.txx +++ b/build2/variable.txx @@ -357,13 +357,20 @@ namespace build2 } template - const string value_traits>::type_name = string ( - value_traits::type_name) + 's'; + value_traits>::value_type_ex:: + value_type_ex (value_type&& v) + : value_type (move (v)) + { + type_name = value_traits::type_name; + type_name += 's'; + name = type_name.c_str (); + } template - const value_type value_traits>::value_type = // VC14 wants =. + const typename value_traits>::value_type_ex + value_traits>::value_type = build2::value_type // VC14 wants =. { - value_traits>::type_name.c_str (), + nullptr, // Patched above. sizeof (vector), nullptr, // No base. &default_dtor>, @@ -516,14 +523,22 @@ namespace build2 } template - const string value_traits>::type_name = string ( - value_traits::type_name) + '_' + - value_traits::type_name + "_map"; + value_traits>::value_type_ex:: + value_type_ex (value_type&& v) + : value_type (move (v)) + { + type_name = value_traits::type_name; + type_name += '_'; + type_name += value_traits::type_name; + type_name += "_map"; + name = type_name.c_str (); + } template - const value_type value_traits>::value_type = // VC14 wants =. + const typename value_traits>::value_type_ex + value_traits>::value_type = build2::value_type // VC14 wants = { - value_traits>::type_name.c_str (), + nullptr, // Patched above. sizeof (map), nullptr, // No base. &default_dtor>, -- cgit v1.1