mirror of
				https://github.com/saitohirga/WSJT-X.git
				synced 2025-10-26 02:20:20 -04:00 
			
		
		
		
	
		
			
	
	
		
			190 lines
		
	
	
		
			5.2 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
		
		
			
		
	
	
			190 lines
		
	
	
		
			5.2 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
|  | // Distributed under the Boost Software License, Version 1.0.
 | ||
|  | // (See accompanying file LICENSE_1_0.txt
 | ||
|  | // or copy at http://www.boost.org/LICENSE_1_0.txt)
 | ||
|  | 
 | ||
|  | // Copyright (c) 2006 Johan Rade
 | ||
|  | // Copyright (c) 2011 Paul A. Bristow
 | ||
|  | 
 | ||
|  | /*!
 | ||
|  | \file | ||
|  | \brief Tests of nonfinite signaling NaN loopback. | ||
|  | 
 | ||
|  | \detail  nonfinite signaling NaN | ||
|  | test outputs using nonfinite facets | ||
|  | (output and input) and reads back in, and checks if loopback OK. | ||
|  | 
 | ||
|  | Not expected to work on all platforms (if any).  But shows that on MSVC, | ||
|  | this legacy locale can ensure a consistent quiet NaN input from representations | ||
|  | "1.#QNAN", "1.#SNAN" and "1.#IND" | ||
|  | 
 | ||
|  | */ | ||
|  | 
 | ||
|  | #ifdef _MSC_VER
 | ||
|  | #   pragma warning(disable : 4702)
 | ||
|  | #endif
 | ||
|  | 
 | ||
|  | #include <boost/math/special_functions/nonfinite_num_facets.hpp>
 | ||
|  | using boost::math::nonfinite_num_get; | ||
|  | using boost::math::nonfinite_num_put; | ||
|  | 
 | ||
|  | #include <iostream>
 | ||
|  | using std::cout; | ||
|  | using std::endl; | ||
|  | 
 | ||
|  | #include <locale>
 | ||
|  | using std::locale; | ||
|  | 
 | ||
|  | #include <string>
 | ||
|  | using std::string; | ||
|  | 
 | ||
|  | #include <sstream>
 | ||
|  |   using std::stringstream; | ||
|  |   using std::istringstream; | ||
|  | 
 | ||
|  | #include <limits>
 | ||
|  | using std::numeric_limits; | ||
|  | 
 | ||
|  | int main() | ||
|  | { | ||
|  |     if((std::numeric_limits<double>::has_infinity == false) || (std::numeric_limits<double>::infinity() == 0)) | ||
|  |   { | ||
|  |     std::cout << "Infinity not supported on this platform." << std::endl; | ||
|  |     return 0; | ||
|  |   } | ||
|  | 
 | ||
|  |   if((std::numeric_limits<double>::has_quiet_NaN == false) || (std::numeric_limits<double>::quiet_NaN() == 0)) | ||
|  |   { | ||
|  |     std::cout << "NaN not supported on this platform." << std::endl; | ||
|  |     return 0; | ||
|  |   } | ||
|  | 
 | ||
|  |   locale default_locale; // Current global locale.
 | ||
|  |   // Try to use the default locale first.
 | ||
|  |   // On MSVC this doesn't work.
 | ||
|  | 
 | ||
|  |   { // Try Quiet NaN
 | ||
|  |     stringstream ss; // Both input and output.
 | ||
|  |     ss.imbue(default_locale); // Redundant, of course.
 | ||
|  |     string infs; | ||
|  |     if(numeric_limits<double>::has_quiet_NaN) | ||
|  |     {  // Make sure quiet NaN is specialised for type double.
 | ||
|  |       double qnan = numeric_limits<double>::quiet_NaN(); | ||
|  |       ss << qnan; // Output quiet_NaN.
 | ||
|  |       infs = ss.str();  //
 | ||
|  |     } | ||
|  |     else | ||
|  |     { // Need to provide a suitable string for quiet NaN.
 | ||
|  |      infs =  "1.#QNAN"; | ||
|  |       ss << infs; | ||
|  |     } | ||
|  |     double r; | ||
|  |     ss >> r; // Read back in.
 | ||
|  | 
 | ||
|  |     cout << "quiet_NaN output was " << infs << endl; // "1.#QNAN"
 | ||
|  |     cout << "quiet_NaN input was " << r << endl; // "1"
 | ||
|  |   } | ||
|  | 
 | ||
|  | #if (!defined __BORLANDC__ && !defined __CODEGEARC__)
 | ||
|  |   // These compilers trap when trying to create a signaling_NaN!
 | ||
|  |   { // Try Signaling NaN
 | ||
|  |     stringstream ss; // Both input and output.
 | ||
|  |     ss.imbue(default_locale); // Redundant, of course.
 | ||
|  |     string infs; | ||
|  |     if(numeric_limits<double>::has_signaling_NaN) | ||
|  |     {  // Make sure signaling NaN is specialised for type double.
 | ||
|  |       double qnan = numeric_limits<double>::signaling_NaN(); | ||
|  |       ss << qnan; // Output signaling_NaN.
 | ||
|  |       infs = ss.str();  //
 | ||
|  |     } | ||
|  |     else | ||
|  |     { // Need to provide a suitable string for signaling NaN.
 | ||
|  |      infs =  "1.#SNAN"; | ||
|  |       ss << infs; | ||
|  |     } | ||
|  |     double r; | ||
|  |     ss >> r; // Read back in.
 | ||
|  | 
 | ||
|  |     cout << "signaling_NaN output was " << infs << endl; // "1.#QNAN" (or "1.#SNAN"?)
 | ||
|  |     cout << "signaling_NaN input was " << r << endl; // "1"
 | ||
|  |   } | ||
|  | #endif // Not Borland or CodeGear.
 | ||
|  | 
 | ||
|  |   // Create legacy_locale and store the nonfinite_num_get facet (with legacy flag) in it.
 | ||
|  |   locale legacy_locale(default_locale, new nonfinite_num_get<char>(boost::math::legacy)); | ||
|  |   // Note that the legacy flag has no effect on the nonfinite_num_put output facet.
 | ||
|  | 
 | ||
|  |   cout << "Use legacy locale." << endl; | ||
|  | 
 | ||
|  |   { // Try infinity.
 | ||
|  |     stringstream ss; // Both input and output.
 | ||
|  |     ss.imbue(legacy_locale); | ||
|  |     string infs; | ||
|  |     if(numeric_limits<double>::has_infinity) | ||
|  |     {  // Make sure infinity is specialised for type double.
 | ||
|  |       double inf = numeric_limits<double>::infinity(); | ||
|  |       ss << inf; // Output infinity.
 | ||
|  |       infs = ss.str();  //
 | ||
|  |     } | ||
|  |     else | ||
|  |     { // Need to provide a suitable string for infinity.
 | ||
|  |      infs =  "1.#INF"; | ||
|  |       ss << infs; | ||
|  |     } | ||
|  |     double r; | ||
|  |     ss >> r; // Read back in.
 | ||
|  | 
 | ||
|  |     cout << "infinity output was " << infs << endl; // "1.#INF"
 | ||
|  |     cout << "infinity input was " << r << endl; // "1.#INF"
 | ||
|  |   } | ||
|  | 
 | ||
|  |   { // Try input of "1.#SNAN".
 | ||
|  |     //double inf = numeric_limits<double>::signaling_NaN(); // Assigns "1.#QNAN" on MSVC.
 | ||
|  |     // So must use explicit string "1.#SNAN" instead.
 | ||
|  |     stringstream ss; // Both input and output.
 | ||
|  |     ss.imbue(legacy_locale); | ||
|  |     string s = "1.#SNAN"; | ||
|  | 
 | ||
|  |     ss << s; // Write out.
 | ||
|  |     double r; | ||
|  | 
 | ||
|  |     ss >> r; // Read back in.
 | ||
|  | 
 | ||
|  |     cout << "SNAN output was " << s << endl; // "1.#SNAN"
 | ||
|  |     cout << "SNAN input was " << r << endl;  // "1.#QNAN"
 | ||
|  |   } | ||
|  | 
 | ||
|  |   { // Try input of "1.#IND" .
 | ||
|  |     stringstream ss; // Both input and output.
 | ||
|  |     ss.imbue(legacy_locale); | ||
|  |     string s = "1.#IND"; | ||
|  |     ss << s; // Write out.
 | ||
|  |     double r; | ||
|  |     ss >> r; // Read back in.
 | ||
|  | 
 | ||
|  |     cout << "IND output was " << s << endl; // "1.#IND"
 | ||
|  |     cout << "IND input was " << r << endl;  // "1.#QNAN"
 | ||
|  |   } | ||
|  | 
 | ||
|  | } // int main()
 | ||
|  | 
 | ||
|  | /*
 | ||
|  | 
 | ||
|  | Output: | ||
|  |   nonfinite_signaling_NaN.vcxproj -> J:\Cpp\fp_facet\fp_facet\Debug\nonfinite_signaling_NaN.exe | ||
|  | 
 | ||
|  |   quiet_NaN output was 1.#QNAN | ||
|  |   quiet_NaN input was 1 | ||
|  |   signaling_NaN output was 1.#QNAN | ||
|  |   signaling_NaN input was 1 | ||
|  |   Use legacy locale. | ||
|  |   infinity output was 1.#INF | ||
|  |   infinity input was 1.#INF | ||
|  |   SNAN output was 1.#SNAN | ||
|  |   SNAN input was 1.#QNAN | ||
|  |   IND output was 1.#IND | ||
|  |   IND input was 1.#QNAN | ||
|  | 
 | ||
|  | 
 | ||
|  | */ | ||
|  | 
 |