diff --git a/src/TDF/TDF_Attribute.cxx b/src/TDF/TDF_Attribute.cxx index c0eaf85b76..7ffc372bd9 100644 --- a/src/TDF/TDF_Attribute.cxx +++ b/src/TDF/TDF_Attribute.cxx @@ -44,7 +44,7 @@ IMPLEMENT_STANDARD_RTTIEXT(TDF_Attribute,Standard_Transient) -#undef TDF_DATA_COMMIT_OPTIMIZED +#define TDF_DATA_COMMIT_OPTIMIZED //======================================================================= //function : TDF_Attribute @@ -131,6 +131,10 @@ void TDF_Attribute::Forget (const Standard_Integer aTransaction) mySavedTransaction = myTransaction; myTransaction = aTransaction; myFlags = (myFlags | TDF_AttributeForgottenMsk); +#ifdef TDF_DATA_COMMIT_OPTIMIZED + if (myLabelNode) + myLabelNode->AttributesModified(Standard_True); +#endif Validate(Standard_False); } diff --git a/src/TDF/TDF_Data.cxx b/src/TDF/TDF_Data.cxx index 44e103c456..7cd8877047 100644 --- a/src/TDF/TDF_Data.cxx +++ b/src/TDF/TDF_Data.cxx @@ -45,7 +45,7 @@ typedef NCollection_Array1 TDF_Array1OfAttributeIDel IMPLEMENT_STANDARD_RTTIEXT(TDF_Data,Standard_Transient) #undef DEB_DELTA_CREATION -#undef TDF_DATA_COMMIT_OPTIMIZED +#define TDF_DATA_COMMIT_OPTIMIZED #ifdef OCCT_DEBUG_DELTA #define TDF_Data_DebugModified(ACTION) \ diff --git a/tests/bugs/caf/bug31748_1 b/tests/bugs/caf/bug31748_1 new file mode 100644 index 0000000000..6767bf1315 --- /dev/null +++ b/tests/bugs/caf/bug31748_1 @@ -0,0 +1,57 @@ +puts "===========" +puts "0031748: Application Framework - Efficient OCAF transactions in OCCT" +puts "===========" + +NewDocument D +UndoLimit D 10 + +# number of labels of objects in the document +set labs 100000 +# number of sub-labels of each object +set sublabs 10 + +NewCommand D + +# store at each object-label sub-labels with two attributes at each +set creation_time [lindex [time { + for {set i 1} {$i <= $labs} {incr i} { + set lab [Label D 0:1:${i}] + SetName D ${lab} Object$i + for {set ii 1} {$ii <= $sublabs} {incr ii} { + set sublab [Label D ${lab}:$ii] + SetInteger D ${sublab} 10 + SetReal D ${sublab} 12.3 + } + } +}] 0] + +set commit_time [lindex [time { + CommitCommand D +}] 0] + +# modify only one object one attribute +NewCommand D +set one_modif_time [lindex [time { + SetInteger D 0:1:[expr $labs/2]:[expr $sublabs/2] 20 +}] 0] + +set one_commit_time [lindex [time { + CommitCommand D +}] 0] + +puts "Tree creation time $creation_time mcs" +puts "Creation commit time $commit_time mcs" +puts "One attribute modification time $one_modif_time mcs" +puts "Commit of single modification time $one_commit_time mcs" + +if {$creation_time > $commit_time} { + set max_creation $creation_time +} else { + set max_creation $commit_time +} + +# Check that creation time (or commit) is at least 1000 times slower than commit of one modification (normally 4000). +# Before the optimization it was only 150 times slower. +if { [expr $max_creation/1000.] < $one_commit_time } { + puts "Error : one modification time commit is too big relatively to the whole document creation or commit" +} diff --git a/tests/bugs/caf/bug31748_2 b/tests/bugs/caf/bug31748_2 new file mode 100644 index 0000000000..216b42f15f --- /dev/null +++ b/tests/bugs/caf/bug31748_2 @@ -0,0 +1,212 @@ +puts "===========" +puts "0031748: Application Framework - Efficient OCAF transactions in OCCT" +puts "===========" + +# This test checks different combinations of creation/modification/forget of attributes in +# transactions: when there are one or two attributes on a label, on father and child labels. + +NewDocument D +UndoLimit D 100 + +# enumeration of actions: first come actions where no attribute needed on the label, after none, after - attribute must be on the label +set act_create 0 +set act_createforget 1 +set act_createmodify 2 +set act_none 3 +set act_modify 4 +set act_modifyforget 5 +set act_forget 6 +set act_forgetcreate 7 +# the number of possible actions that can be done on attribute +set actions 8 + +set act_afternone [expr $act_none+1] + +# retuns 1 if after this action there exists attribute on the label +proc produces_attribute1 {action} { + global act_create act_modify act_forgetcreate act_createmodify + if {$action==$act_create || $action==$act_createmodify || $action==$act_modify || $action==$act_forgetcreate} { + return 1 + } + return 0 +} + +# retuns 1 if after two actions there exists attribute on the label +proc produces_attribute2 {action1 action2} { + global act_create act_modify act_modifyforget act_forget act_createforget + if {[produces_attribute1 $action2]} { + return 1 + } + if {$action2!=$act_forget && $action2!=$act_createforget && $action2!=$act_modifyforget && [produces_attribute1 $action1]} { + return 1 + } + return 0 +} + +# retuns value of the attribute produced by two actions +proc produces_value {action1 action2} { + global act_modify act_createmodify act_none act_forgetcreate + if {$action2==$act_modify} { + if {$action1==$act_createmodify} { + return 3 + } + return 2 + } + if {$action2==$act_createmodify} { + return 2 + } + if {$action1==$act_createmodify && $action2!=$act_forgetcreate} { + return 2 + } + if {$action1==$act_none && $action1==$act_createmodify} { + return 2 + } + return 1 +} + +proc attribute_name attr_id { + if $attr_id==0 { + return Integer + } + return Real +} + +proc do_action {label attr action} { + global D + set attrName [attribute_name $attr] + switch $action { # first there are atcions that leave attribute, then - none, then - that remove it + 0 { # sets a new attribute + Set$attrName D $label 1 + } + 1 { # creates and immediately forgets a new attribute + Set$attrName D $label 1 + if $attr==0 { # forget integer + ForgetAtt D $label 2a96b606-ec8b-11d0-bee7-080009dc3333 + } else { # forget real + ForgetAtt D $label 2a96b60f-ec8b-11d0-bee7-080009dc3333 + } + } + 2 { # sets and modifies attribute + Set$attrName D $label 1 + Set$attrName D $label 2 + } + 3 { # nothing to do + } + 4 { # modifies (increments) an attribute value if it is already exists on this label + set value [Get$attrName D $label] + Set$attrName D $label [expr $value+1] + } + 5 { # modifies and immediately forgets an attribute + set value [Get$attrName D $label] + Set$attrName D $label [expr $value+1] + if $attr==0 { # forget integer + ForgetAtt D $label 2a96b606-ec8b-11d0-bee7-080009dc3333 + } else { # forget real + ForgetAtt D $label 2a96b60f-ec8b-11d0-bee7-080009dc3333 + } + } + 6 { # forgets the attribute + if $attr==0 { # forget integer + ForgetAtt D $label 2a96b606-ec8b-11d0-bee7-080009dc3333 + } else { # forget real + ForgetAtt D $label 2a96b60f-ec8b-11d0-bee7-080009dc3333 + } + } + 7 { # forgets and immediately creates an attribute + if $attr==0 { # forget integer + ForgetAtt D $label 2a96b606-ec8b-11d0-bee7-080009dc3333 + } else { # forget real + ForgetAtt D $label 2a96b60f-ec8b-11d0-bee7-080009dc3333 + } + Set$attrName D $label 1 + } + } +} + +proc check_attribute {action1 action2 lab attr} { + global D + set attrName [attribute_name $attr] + if [produces_attribute2 $action1 $action2] { + set value [Get$attrName D $lab] + set expected_value [produces_value $action1 $action2] + if $value!=$expected_value { + puts "Error : attribute $attrName value $value does not match the expected $expected_value at the label $lab" + } + } else { + set attributes [Attributes D $lab] + if {[lsearch $attributes TDataStd_$attrName]>=0} { + puts "Error : attribute $attrName exists but it should not at the label $lab" + } + } +} + + +set lab_index 1 +set num_variants 0 +# cycles by variables tl