mirror of
https://git.dev.opencascade.org/repos/occt.git
synced 2025-04-21 10:13:43 +03:00
0029452: Application Framework - Failed to read an Ocaf XML document with 1.#QNAN value.
Method XmlObjMgt::GetReal() is improved to (a) recognize NAN and infinity written by old MSVC runtime (like 1.#QNAN and 1.#INF) and (b) detect situation when there are some trailing non-space symbols after the real value, returning False in such case. Reading of real-valued attributes (single real, array, list) from OCAF XML format is improved to create valid attribute even if parsing of (some) members fails; warning is generated instead of error in such case. Added test bugs caf bug29452
This commit is contained in:
parent
cbc4faa980
commit
e13b9464ef
@ -62,10 +62,21 @@ Standard_Boolean XmlMDataStd_RealArrayDriver::Paste
|
|||||||
const Handle(TDF_Attribute)& theTarget,
|
const Handle(TDF_Attribute)& theTarget,
|
||||||
XmlObjMgt_RRelocationTable& ) const
|
XmlObjMgt_RRelocationTable& ) const
|
||||||
{
|
{
|
||||||
Standard_Integer aFirstInd, aLastInd, ind;
|
|
||||||
Standard_Real aValue;
|
Handle(TDataStd_RealArray) aRealArray = Handle(TDataStd_RealArray)::DownCast(theTarget);
|
||||||
const XmlObjMgt_Element& anElement = theSource;
|
const XmlObjMgt_Element& anElement = theSource;
|
||||||
|
|
||||||
|
// attribute id
|
||||||
|
Standard_GUID aGUID;
|
||||||
|
XmlObjMgt_DOMString aGUIDStr = anElement.getAttribute(::AttributeIDString());
|
||||||
|
if (aGUIDStr.Type() == XmlObjMgt_DOMString::LDOM_NULL)
|
||||||
|
aGUID = TDataStd_RealArray::GetID(); //default case
|
||||||
|
else
|
||||||
|
aGUID = Standard_GUID(Standard_CString(aGUIDStr.GetString())); // user defined case
|
||||||
|
aRealArray->SetID(aGUID);
|
||||||
|
|
||||||
|
Standard_Integer aFirstInd, aLastInd, ind;
|
||||||
|
|
||||||
// Read the FirstIndex; if the attribute is absent initialize to 1
|
// Read the FirstIndex; if the attribute is absent initialize to 1
|
||||||
XmlObjMgt_DOMString aFirstIndex= anElement.getAttribute(::FirstIndexString());
|
XmlObjMgt_DOMString aFirstIndex= anElement.getAttribute(::FirstIndexString());
|
||||||
if (aFirstIndex == NULL)
|
if (aFirstIndex == NULL)
|
||||||
@ -89,8 +100,6 @@ Standard_Boolean XmlMDataStd_RealArrayDriver::Paste
|
|||||||
return Standard_False;
|
return Standard_False;
|
||||||
}
|
}
|
||||||
|
|
||||||
Handle(TDataStd_RealArray) aRealArray =
|
|
||||||
Handle(TDataStd_RealArray)::DownCast(theTarget);
|
|
||||||
aRealArray->Init(aFirstInd, aLastInd);
|
aRealArray->Init(aFirstInd, aLastInd);
|
||||||
|
|
||||||
// Check the type of LDOMString
|
// Check the type of LDOMString
|
||||||
@ -109,6 +118,7 @@ Standard_Boolean XmlMDataStd_RealArrayDriver::Paste
|
|||||||
return Standard_False;
|
return Standard_False;
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
|
Standard_Real aValue;
|
||||||
Standard_CString aValueStr = Standard_CString(aString.GetString());
|
Standard_CString aValueStr = Standard_CString(aString.GetString());
|
||||||
for (ind = aFirstInd; ind <= aLastInd; ind++)
|
for (ind = aFirstInd; ind <= aLastInd; ind++)
|
||||||
{
|
{
|
||||||
@ -117,8 +127,9 @@ Standard_Boolean XmlMDataStd_RealArrayDriver::Paste
|
|||||||
TCollection_ExtendedString("Cannot retrieve real member"
|
TCollection_ExtendedString("Cannot retrieve real member"
|
||||||
" for RealArray attribute as \"")
|
" for RealArray attribute as \"")
|
||||||
+ aValueStr + "\"";
|
+ aValueStr + "\"";
|
||||||
myMessageDriver->Send (aMessageString, Message_Fail);
|
myMessageDriver->Send (aMessageString, Message_Warning);
|
||||||
return Standard_False;
|
// skip to the next space separator
|
||||||
|
while (*aValueStr != 0 && ! IsSpace (*aValueStr)) ++aValueStr;
|
||||||
}
|
}
|
||||||
aRealArray->SetValue(ind, aValue);
|
aRealArray->SetValue(ind, aValue);
|
||||||
}
|
}
|
||||||
@ -145,15 +156,6 @@ Standard_Boolean XmlMDataStd_RealArrayDriver::Paste
|
|||||||
#endif
|
#endif
|
||||||
aRealArray->SetDelta(aDelta);
|
aRealArray->SetDelta(aDelta);
|
||||||
|
|
||||||
// attribute id
|
|
||||||
Standard_GUID aGUID;
|
|
||||||
XmlObjMgt_DOMString aGUIDStr = anElement.getAttribute(::AttributeIDString());
|
|
||||||
if (aGUIDStr.Type() == XmlObjMgt_DOMString::LDOM_NULL)
|
|
||||||
aGUID = TDataStd_RealArray::GetID(); //default case
|
|
||||||
else
|
|
||||||
aGUID = Standard_GUID(Standard_CString(aGUIDStr.GetString())); // user defined case
|
|
||||||
|
|
||||||
aRealArray->SetID(aGUID);
|
|
||||||
return Standard_True;
|
return Standard_True;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -53,20 +53,6 @@ Standard_Boolean XmlMDataStd_RealDriver::Paste
|
|||||||
const Handle(TDF_Attribute)& theTarget,
|
const Handle(TDF_Attribute)& theTarget,
|
||||||
XmlObjMgt_RRelocationTable& ) const
|
XmlObjMgt_RRelocationTable& ) const
|
||||||
{
|
{
|
||||||
Standard_Real aValue;
|
|
||||||
XmlObjMgt_DOMString aRealStr= XmlObjMgt::GetStringValue (theSource);
|
|
||||||
|
|
||||||
if (XmlObjMgt::GetReal(aRealStr, aValue) == Standard_False) {
|
|
||||||
TCollection_ExtendedString aMessageString =
|
|
||||||
TCollection_ExtendedString("Cannot retrieve Real attribute from \"")
|
|
||||||
+ aRealStr + "\"";
|
|
||||||
myMessageDriver->Send (aMessageString, Message_Fail);
|
|
||||||
return Standard_False;
|
|
||||||
}
|
|
||||||
|
|
||||||
Handle(TDataStd_Real) anAtt = Handle(TDataStd_Real)::DownCast(theTarget);
|
|
||||||
anAtt->Set(aValue);
|
|
||||||
|
|
||||||
// attribute id
|
// attribute id
|
||||||
Standard_GUID aGUID;
|
Standard_GUID aGUID;
|
||||||
const XmlObjMgt_Element& anElement = theSource;
|
const XmlObjMgt_Element& anElement = theSource;
|
||||||
@ -78,6 +64,18 @@ Standard_Boolean XmlMDataStd_RealDriver::Paste
|
|||||||
|
|
||||||
Handle(TDataStd_Real)::DownCast(theTarget)->SetID(aGUID);
|
Handle(TDataStd_Real)::DownCast(theTarget)->SetID(aGUID);
|
||||||
|
|
||||||
|
Standard_Real aValue(0.);
|
||||||
|
const XmlObjMgt_DOMString& aRealStr= XmlObjMgt::GetStringValue (theSource);
|
||||||
|
Standard_CString aValueStr = Standard_CString (aRealStr.GetString());
|
||||||
|
if(XmlObjMgt::GetReal(aRealStr, aValue) == Standard_False) {
|
||||||
|
TCollection_ExtendedString aMessageString =
|
||||||
|
TCollection_ExtendedString("Cannot retrieve Real attribute from \"")
|
||||||
|
+ aValueStr + "\"";
|
||||||
|
myMessageDriver->Send (aMessageString, Message_Warning);
|
||||||
|
}
|
||||||
|
Handle(TDataStd_Real) anAtt = Handle(TDataStd_Real)::DownCast(theTarget);
|
||||||
|
anAtt->Set(aValue);
|
||||||
|
|
||||||
return Standard_True;
|
return Standard_True;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -55,11 +55,20 @@ Standard_Boolean XmlMDataStd_RealListDriver::Paste(const XmlObjMgt_Persistent&
|
|||||||
const Handle(TDF_Attribute)& theTarget,
|
const Handle(TDF_Attribute)& theTarget,
|
||||||
XmlObjMgt_RRelocationTable& ) const
|
XmlObjMgt_RRelocationTable& ) const
|
||||||
{
|
{
|
||||||
Standard_Real aValue;
|
const Handle(TDataStd_RealList) aRealList = Handle(TDataStd_RealList)::DownCast(theTarget);
|
||||||
Standard_Integer aFirstInd, aLastInd, ind;
|
|
||||||
const XmlObjMgt_Element& anElement = theSource;
|
const XmlObjMgt_Element& anElement = theSource;
|
||||||
|
|
||||||
|
// attribute id
|
||||||
|
Standard_GUID aGUID;
|
||||||
|
XmlObjMgt_DOMString aGUIDStr = anElement.getAttribute(::AttributeIDString());
|
||||||
|
if (aGUIDStr.Type() == XmlObjMgt_DOMString::LDOM_NULL)
|
||||||
|
aGUID = TDataStd_RealList::GetID(); //default case
|
||||||
|
else
|
||||||
|
aGUID = Standard_GUID(Standard_CString(aGUIDStr.GetString())); // user defined case
|
||||||
|
aRealList->SetID(aGUID);
|
||||||
|
|
||||||
// Read the FirstIndex; if the attribute is absent initialize to 1
|
// Read the FirstIndex; if the attribute is absent initialize to 1
|
||||||
|
Standard_Integer aFirstInd, aLastInd, ind;
|
||||||
XmlObjMgt_DOMString aFirstIndex= anElement.getAttribute(::FirstIndexString());
|
XmlObjMgt_DOMString aFirstIndex= anElement.getAttribute(::FirstIndexString());
|
||||||
if (aFirstIndex == NULL)
|
if (aFirstIndex == NULL)
|
||||||
aFirstInd = 1;
|
aFirstInd = 1;
|
||||||
@ -84,7 +93,6 @@ Standard_Boolean XmlMDataStd_RealListDriver::Paste(const XmlObjMgt_Persistent&
|
|||||||
return Standard_False;
|
return Standard_False;
|
||||||
}
|
}
|
||||||
|
|
||||||
const Handle(TDataStd_RealList) aRealList = Handle(TDataStd_RealList)::DownCast(theTarget);
|
|
||||||
// Check the type of LDOMString
|
// Check the type of LDOMString
|
||||||
const XmlObjMgt_DOMString& aString = XmlObjMgt::GetStringValue(anElement);
|
const XmlObjMgt_DOMString& aString = XmlObjMgt::GetStringValue(anElement);
|
||||||
if(aLastInd == 0) aFirstInd = 0;
|
if(aLastInd == 0) aFirstInd = 0;
|
||||||
@ -111,28 +119,20 @@ Standard_Boolean XmlMDataStd_RealListDriver::Paste(const XmlObjMgt_Persistent&
|
|||||||
Standard_CString aValueStr = Standard_CString(aString.GetString());
|
Standard_CString aValueStr = Standard_CString(aString.GetString());
|
||||||
for (ind = aFirstInd; ind <= aLastInd; ind++)
|
for (ind = aFirstInd; ind <= aLastInd; ind++)
|
||||||
{
|
{
|
||||||
|
Standard_Real aValue;
|
||||||
if (!XmlObjMgt::GetReal(aValueStr, aValue)) {
|
if (!XmlObjMgt::GetReal(aValueStr, aValue)) {
|
||||||
TCollection_ExtendedString aMessageString =
|
TCollection_ExtendedString aMessageString =
|
||||||
TCollection_ExtendedString("Cannot retrieve real member"
|
TCollection_ExtendedString("Cannot retrieve real member"
|
||||||
" for RealList attribute as \"")
|
" for RealList attribute as \"")
|
||||||
+ aValueStr + "\"";
|
+ aValueStr + "\"";
|
||||||
myMessageDriver->Send (aMessageString, Message_Fail);
|
myMessageDriver->Send(aMessageString, Message_Warning);
|
||||||
return Standard_False;
|
// skip to the next space separator
|
||||||
|
while (*aValueStr != 0 && ! IsSpace (*aValueStr)) ++aValueStr;
|
||||||
}
|
}
|
||||||
aRealList->Append(aValue);
|
aRealList->Append(aValue);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// attribute id
|
|
||||||
Standard_GUID aGUID;
|
|
||||||
XmlObjMgt_DOMString aGUIDStr = anElement.getAttribute(::AttributeIDString());
|
|
||||||
if (aGUIDStr.Type() == XmlObjMgt_DOMString::LDOM_NULL)
|
|
||||||
aGUID = TDataStd_RealList::GetID(); //default case
|
|
||||||
else
|
|
||||||
aGUID = Standard_GUID(Standard_CString(aGUIDStr.GetString())); // user defined case
|
|
||||||
|
|
||||||
aRealList->SetID(aGUID);
|
|
||||||
|
|
||||||
return Standard_True;
|
return Standard_True;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -23,6 +23,8 @@
|
|||||||
|
|
||||||
#include <errno.h>
|
#include <errno.h>
|
||||||
#include <stdio.h>
|
#include <stdio.h>
|
||||||
|
#include <limits>
|
||||||
|
|
||||||
static const char aRefPrefix [] = "/document/label";
|
static const char aRefPrefix [] = "/document/label";
|
||||||
static const char aRefElem1 [] = "/label[@tag=";
|
static const char aRefElem1 [] = "/label[@tag=";
|
||||||
static const char aRefElem2 [] = "]";
|
static const char aRefElem2 [] = "]";
|
||||||
@ -344,11 +346,37 @@ Standard_Boolean XmlObjMgt::GetReal (Standard_CString& theString,
|
|||||||
{
|
{
|
||||||
char * ptr;
|
char * ptr;
|
||||||
errno = 0;
|
errno = 0;
|
||||||
double aValue = Strtod (theString, &ptr);
|
theValue = Strtod (theString, &ptr);
|
||||||
if (ptr == theString || errno == ERANGE || errno == EINVAL)
|
if (ptr == theString || errno == ERANGE || errno == EINVAL)
|
||||||
return Standard_False;
|
return Standard_False;
|
||||||
theValue = Standard_Real (aValue);
|
|
||||||
theString = ptr;
|
theString = ptr;
|
||||||
|
|
||||||
|
// detect NAN or infinite values written by old MSVC run-time as -1. with
|
||||||
|
// suffix "#QNAN" or "#SNAN" or "#INF"
|
||||||
|
if (*ptr == '#')
|
||||||
|
{
|
||||||
|
if (! strncmp (ptr, "#QNAN", 5) || ! strncmp (ptr, "#SNAN", 5))
|
||||||
|
{
|
||||||
|
theString = ptr + 5;
|
||||||
|
theValue = std::numeric_limits<double>::quiet_NaN();
|
||||||
|
return Standard_True;
|
||||||
|
}
|
||||||
|
else if (! strncmp (ptr, "#INF", 4))
|
||||||
|
{
|
||||||
|
theString = ptr + 4;
|
||||||
|
theValue = (theValue < 0 ? -std::numeric_limits<double>::infinity() : std::numeric_limits<double>::infinity());
|
||||||
|
return Standard_True;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
return Standard_False;
|
||||||
|
}
|
||||||
|
else if (*ptr && ! IsSpace (*ptr))
|
||||||
|
{
|
||||||
|
// report failure if reading stopped not at the end of the string or space
|
||||||
|
return Standard_False;
|
||||||
|
}
|
||||||
|
|
||||||
return Standard_True;
|
return Standard_True;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -371,13 +399,8 @@ Standard_Boolean XmlObjMgt::GetReal (const XmlObjMgt_DOMString& theString,
|
|||||||
}
|
}
|
||||||
default: // LDOM_Ascii*
|
default: // LDOM_Ascii*
|
||||||
{
|
{
|
||||||
char * ptr;
|
|
||||||
const char * aString = theString.GetString();
|
const char * aString = theString.GetString();
|
||||||
errno = 0;
|
return GetReal (aString, theValue);
|
||||||
double aValue = Strtod (aString, &ptr);
|
|
||||||
if (ptr == aString || errno == ERANGE || errno == EINVAL)
|
|
||||||
return Standard_False;
|
|
||||||
theValue = Standard_Real (aValue);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return Standard_True;
|
return Standard_True;
|
||||||
|
@ -1,4 +1,4 @@
|
|||||||
puts "REQUIRED All: XmlDriver warning: failure reading attribute TDataStd_RealArray"
|
puts "REQUIRED All: Cannot retrieve real member for RealArray attribute"
|
||||||
|
|
||||||
puts "================"
|
puts "================"
|
||||||
puts "OCC2269"
|
puts "OCC2269"
|
||||||
|
40
tests/bugs/caf/bug29452
Normal file
40
tests/bugs/caf/bug29452
Normal file
@ -0,0 +1,40 @@
|
|||||||
|
puts "=========="
|
||||||
|
puts "OCC29452"
|
||||||
|
puts "=========="
|
||||||
|
puts ""
|
||||||
|
###################################################
|
||||||
|
# XmlMDataStd_RealDriver: Failed to read an Ocaf XML document with 1.#QNAN value
|
||||||
|
###################################################
|
||||||
|
|
||||||
|
# Check for presence of warning message
|
||||||
|
puts "REQUIRED ALL: Cannot retrieve real member for RealArray attribute"
|
||||||
|
|
||||||
|
set BugNumber OCC29452
|
||||||
|
set file [locate_data_file bug29452.xml]
|
||||||
|
|
||||||
|
catch {Close D}
|
||||||
|
set res [Open $file D]
|
||||||
|
|
||||||
|
set real [GetReal D 0:1]
|
||||||
|
set rlst [GetRealList D 0:1]
|
||||||
|
set rarr [GetRealArray D 0:1]
|
||||||
|
|
||||||
|
# Check for read values
|
||||||
|
checkreal "Real value" [lindex $rlst 0] 1.1 0 1e-15
|
||||||
|
checkreal "Real value" [lindex $rlst 2] 3.3 0 1e-15
|
||||||
|
checkreal "Real value" [lindex $rarr 0] 0.111 0 1e-15
|
||||||
|
checkreal "Real value" [lindex $rarr 3] 123. 0 1e-15
|
||||||
|
checkreal "Real value" [lindex $rarr 4] 3.14e12 0 1e-15
|
||||||
|
foreach inf [list $real [lindex $rlst 3]] {
|
||||||
|
if { [string compare "$inf" "inf"] &&
|
||||||
|
[string compare "$inf" "infinity"] &&
|
||||||
|
[string compare "$inf" "1.#INF"] } {
|
||||||
|
puts "Error: Real value is read as \"$inf\", expected infinity"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
foreach nan [list [lindex $rlst 1] [lindex $rarr 1] [lindex $rarr 2]] {
|
||||||
|
if { [string compare "$nan" "nan"] &&
|
||||||
|
[string compare "$nan" "1.#QNAN"] } {
|
||||||
|
puts "Error: Real value is read as \"$nan\", expected NAN"
|
||||||
|
}
|
||||||
|
}
|
Loading…
x
Reference in New Issue
Block a user