1
0
mirror of https://git.dev.opencascade.org/repos/occt.git synced 2025-04-05 18:16:23 +03:00

0026029: Visualization - Poor performance of connected objects

Fix performance issue with SelectMgr_SelectableObjectSet.
This commit is contained in:
dbp 2015-05-07 18:20:27 +03:00 committed by abv
parent 9319499b57
commit d4aaad5b82
16 changed files with 456 additions and 368 deletions

View File

@ -32,6 +32,7 @@ Graphic3d_ShaderVariable.cxx
Graphic3d_ShaderVariable.lxx Graphic3d_ShaderVariable.lxx
Graphic3d_MapOfStructure.hxx Graphic3d_MapOfStructure.hxx
Graphic3d_MapIteratorOfMapOfStructure.hxx Graphic3d_MapIteratorOfMapOfStructure.hxx
Graphic3d_IndexedMapOfAddress.hxx
Graphic3d_TypeOfShaderObject.hxx Graphic3d_TypeOfShaderObject.hxx
Graphic3d_DataStructure.pxx Graphic3d_DataStructure.pxx
Graphic3d_StructureManager.pxx Graphic3d_StructureManager.pxx

View File

@ -493,6 +493,7 @@ is
imported transient class NMapOfTransient; imported transient class NMapOfTransient;
imported MapOfStructure; imported MapOfStructure;
imported IndexedMapOfAddress;
imported SequenceOfDisplayedStructures; imported SequenceOfDisplayedStructures;
--------------------------------- ---------------------------------

View File

@ -0,0 +1,23 @@
// Created on: 2015-04-2015
// Created by: Denis BOGOLEPOV
// Copyright (c) 2015 OPEN CASCADE SAS
//
// This file is part of Open CASCADE Technology software library.
//
// This library is free software; you can redistribute it and/or modify it under
// the terms of the GNU Lesser General Public License version 2.1 as published
// by the Free Software Foundation, with special exception defined in the file
// OCCT_LGPL_EXCEPTION.txt. Consult the file LICENSE_LGPL_21.txt included in OCCT
// distribution for complete text of the license and disclaimer of any warranty.
//
// Alternatively, this file may be used under the terms of Open CASCADE
// commercial license or contractual agreement.
#ifndef _Graphic3d_IndexedMapOfAddress
#define _Graphic3d_IndexedMapOfAddress
#include <NCollection_DataMap.hxx>
typedef NCollection_DataMap<const Standard_Address, Standard_Integer> Graphic3d_IndexedMapOfAddress;
#endif // _Graphic3d_IndexedMapOfAddress

View File

@ -40,42 +40,42 @@ class Structure from Graphic3d inherits TShared
uses uses
Array2OfReal from TColStd, Array2OfReal from TColStd,
SequenceOfAddress from TColStd, SequenceOfAddress from TColStd,
Color from Quantity, Color from Quantity,
GenId from Aspect, GenId from Aspect,
TypeOfHighlightMethod from Aspect, TypeOfHighlightMethod from Aspect,
DataStructureManager from Graphic3d, DataStructureManager from Graphic3d,
AspectFillArea3d from Graphic3d, AspectFillArea3d from Graphic3d,
AspectLine3d from Graphic3d, AspectLine3d from Graphic3d,
AspectMarker3d from Graphic3d, AspectMarker3d from Graphic3d,
AspectText3d from Graphic3d, AspectText3d from Graphic3d,
CStructure from Graphic3d, CStructure from Graphic3d,
CStructure from Graphic3d, GraphicDriver from Graphic3d,
GraphicDriver from Graphic3d, Group from Graphic3d,
Group from Graphic3d, SequenceOfGroup from Graphic3d,
SequenceOfGroup from Graphic3d, SequenceOfStructure from Graphic3d,
SequenceOfStructure from Graphic3d, HSequenceOfStructure from Graphic3d,
HSequenceOfStructure from Graphic3d, MapOfStructure from Graphic3d,
MapOfStructure from Graphic3d, IndexedMapOfAddress from Graphic3d,
StructureManager from Graphic3d, StructureManager from Graphic3d,
StructureManagerPtr from Graphic3d, StructureManagerPtr from Graphic3d,
TypeOfComposition from Graphic3d, TypeOfComposition from Graphic3d,
TypeOfConnection from Graphic3d, TypeOfConnection from Graphic3d,
TypeOfPrimitive from Graphic3d, TypeOfPrimitive from Graphic3d,
TypeOfStructure from Graphic3d, TypeOfStructure from Graphic3d,
Vector from Graphic3d, Vector from Graphic3d,
Vertex from Graphic3d, Vertex from Graphic3d,
TransModeFlags from Graphic3d, TransModeFlags from Graphic3d,
ZLayerId from Graphic3d, ZLayerId from Graphic3d,
Pnt from gp, Pnt from gp,
SequenceOfHClipPlane from Graphic3d, SequenceOfHClipPlane from Graphic3d,
BndBox4f from Graphic3d, BndBox4f from Graphic3d,
BndBox4d from Graphic3d, BndBox4d from Graphic3d,
Box from Bnd Box from Bnd
raises raises
@ -957,6 +957,38 @@ is
---C++: return const & ---C++: return const &
---C++: inline ---C++: inline
AppendDescendant (me : mutable;
theDescendant : Address from Standard)
returns Boolean from Standard
is protected;
---Level: Internal
---Purpose: Appends new descendant structure.
---Category: Private methods
RemoveDescendant (me : mutable;
theDescendant : Address from Standard)
returns Boolean from Standard
is protected;
---Level: Internal
---Purpose: Removes the given descendant structure.
---Category: Private methods
AppendAncestor (me : mutable;
theAncestor : Address from Standard)
returns Boolean from Standard
is protected;
---Level: Internal
---Purpose: Appends new ancestor structure.
---Category: Private methods
RemoveAncestor (me : mutable;
theAncestor : Address from Standard)
returns Boolean from Standard
is protected;
---Level: Internal
---Purpose: Removes the given ancestor structure.
---Category: Private methods
fields fields
-- --
@ -968,14 +1000,20 @@ fields
-- It is a sequence of groups of primitives. -- It is a sequence of groups of primitives.
-- --
-- the associated low-level structure -- the associated low-level structure
myCStructure : CStructure from Graphic3d; myCStructure : CStructure from Graphic3d;
-- the structures to which the structure is attached -- the structures to which the structure is attached
myAncestors : SequenceOfAddress from TColStd; myAncestors : SequenceOfAddress from TColStd;
-- the structures attached to the structure -- the structures attached to the structure
myDescendants : SequenceOfAddress from TColStd; myDescendants : SequenceOfAddress from TColStd;
-- the map of structures to which the structure is attached
myAncestorMap : IndexedMapOfAddress from Graphic3d;
-- the map of structures attached to the structure
myDescendantMap : IndexedMapOfAddress from Graphic3d;
-- the highlight method of the structure -- the highlight method of the structure
myHighlightColor : Color from Quantity; myHighlightColor : Color from Quantity;

View File

@ -1312,6 +1312,90 @@ void Graphic3d_Structure::Descendants (Graphic3d_MapOfStructure& theSet) const
} }
} }
//=============================================================================
//function : AppendDescendant
//purpose :
//=============================================================================
Standard_Boolean Graphic3d_Structure::AppendDescendant (const Standard_Address theDescendant)
{
if (myDescendantMap.IsBound (theDescendant))
{
return Standard_False; // already connected
}
myDescendantMap.Bind (theDescendant, myDescendants.Length() + 1);
myDescendants.Append (theDescendant);
return Standard_True;
}
//=============================================================================
//function : RemoveDescendant
//purpose :
//=============================================================================
Standard_Boolean Graphic3d_Structure::RemoveDescendant (const Standard_Address theDescendant)
{
Standard_Integer aStructIdx;
if (!myDescendantMap.Find (theDescendant, aStructIdx))
{
return Standard_False;
}
myDescendantMap.UnBind (theDescendant);
if (aStructIdx != myDescendants.Length())
{
myDescendants.Exchange (aStructIdx, myDescendants.Length());
myDescendantMap.Bind (myDescendants (aStructIdx), aStructIdx);
}
myDescendants.Remove (myDescendants.Length());
return Standard_True;
}
//=============================================================================
//function : AppendAncestor
//purpose :
//=============================================================================
Standard_Boolean Graphic3d_Structure::AppendAncestor (const Standard_Address theAncestor)
{
if (myAncestorMap.IsBound (theAncestor))
{
return Standard_False; // already connected
}
myAncestorMap.Bind (theAncestor, myAncestors.Length() + 1);
myAncestors.Append (theAncestor);
return Standard_True;
}
//=============================================================================
//function : RemoveAncestor
//purpose :
//=============================================================================
Standard_Boolean Graphic3d_Structure::RemoveAncestor (const Standard_Address theAncestor)
{
Standard_Integer aStructIdx;
if (!myAncestorMap.Find (theAncestor, aStructIdx))
{
return Standard_False;
}
myAncestorMap.UnBind (theAncestor);
if (aStructIdx != myAncestors.Length())
{
myAncestors.Exchange (aStructIdx, myAncestors.Length());
myAncestorMap.Bind (myAncestors (aStructIdx), aStructIdx);
}
myAncestors.Remove (myAncestors.Length());
return Standard_True;
}
//============================================================================= //=============================================================================
//function : Connect //function : Connect
//purpose : //purpose :
@ -1320,7 +1404,10 @@ void Graphic3d_Structure::Connect (const Handle(Graphic3d_Structure)& theStructu
const Graphic3d_TypeOfConnection theType, const Graphic3d_TypeOfConnection theType,
const Standard_Boolean theWithCheck) const Standard_Boolean theWithCheck)
{ {
if (IsDeleted()) return; if (IsDeleted())
{
return;
}
// cycle detection // cycle detection
if (theWithCheck if (theWithCheck
@ -1329,47 +1416,34 @@ void Graphic3d_Structure::Connect (const Handle(Graphic3d_Structure)& theStructu
return; return;
} }
switch (theType) const Standard_Address aStructure = theStructure.operator->();
if (theType == Graphic3d_TOC_DESCENDANT)
{ {
case Graphic3d_TOC_DESCENDANT: if (!AppendDescendant (aStructure))
{ {
const Standard_Integer aNbDesc = myDescendants.Length();
for (Standard_Integer anIter = 1; anIter <= aNbDesc; ++anIter)
{
if (myDescendants.Value (anIter) == theStructure.operator->())
{
return;
}
}
myDescendants.Append (theStructure.operator->());
CalculateBoundBox();
theStructure->Connect (this, Graphic3d_TOC_ANCESTOR);
GraphicConnect (theStructure);
myStructureManager->Connect (this, theStructure);
Update();
return; return;
} }
case Graphic3d_TOC_ANCESTOR:
CalculateBoundBox();
theStructure->Connect (this, Graphic3d_TOC_ANCESTOR);
GraphicConnect (theStructure);
myStructureManager->Connect (this, theStructure);
Update();
}
else // Graphic3d_TOC_ANCESTOR
{
if (!AppendAncestor (aStructure))
{ {
const Standard_Integer aNbAnces = myAncestors.Length();
for (Standard_Integer anIter = 1; anIter <= aNbAnces; ++anIter)
{
if (myAncestors.Value (anIter) == theStructure.operator->())
{
return;
}
}
myAncestors.Append (theStructure.operator->());
CalculateBoundBox();
theStructure->Connect (this, Graphic3d_TOC_DESCENDANT);
// myGraphicDriver->Connect is called in case if connection between parent and child
return; return;
} }
CalculateBoundBox();
theStructure->Connect (this, Graphic3d_TOC_DESCENDANT);
// myStructureManager->Connect is called in case if connection between parent and child
} }
} }
@ -1379,37 +1453,29 @@ void Graphic3d_Structure::Connect (const Handle(Graphic3d_Structure)& theStructu
//============================================================================= //=============================================================================
void Graphic3d_Structure::Disconnect (const Handle(Graphic3d_Structure)& theStructure) void Graphic3d_Structure::Disconnect (const Handle(Graphic3d_Structure)& theStructure)
{ {
if (IsDeleted()) return; if (IsDeleted())
const Standard_Integer aNbDesc = myDescendants.Length();
for (Standard_Integer anIter = 1; anIter <= aNbDesc; ++anIter)
{ {
if (myDescendants.Value (anIter) == theStructure.operator->()) return;
{
myDescendants.Remove (anIter);
theStructure->Disconnect (this);
GraphicDisconnect (theStructure);
myStructureManager->Disconnect (this, theStructure);
CalculateBoundBox();
Update();
return;
}
} }
const Standard_Integer aNbAnces = myAncestors.Length(); const Standard_Address aStructure = theStructure.operator->();
for (Standard_Integer anIter = 1; anIter <= aNbAnces; ++anIter)
if (RemoveDescendant (aStructure))
{ {
if (myAncestors.Value (anIter) == theStructure.operator->()) theStructure->Disconnect (this);
{
myAncestors.Remove (anIter); GraphicDisconnect (theStructure);
theStructure->Disconnect (this); myStructureManager->Disconnect (this, theStructure);
CalculateBoundBox();
// no call of myGraphicDriver->Disconnect in case of an ancestor CalculateBoundBox();
return; Update();
} }
else if (RemoveAncestor (aStructure))
{
theStructure->Disconnect (this);
CalculateBoundBox();
// no call of myStructureManager->Disconnect in case of an ancestor
} }
} }
@ -1680,34 +1746,13 @@ gp_Pnt Graphic3d_Structure::TransformPersistencePoint() const
void Graphic3d_Structure::Remove (const Standard_Address thePtr, void Graphic3d_Structure::Remove (const Standard_Address thePtr,
const Graphic3d_TypeOfConnection theType) const Graphic3d_TypeOfConnection theType)
{ {
switch (theType) if (theType == Graphic3d_TOC_DESCENDANT)
{ {
case Graphic3d_TOC_DESCENDANT: RemoveDescendant (thePtr);
{ }
const Standard_Integer aNbDesc = myDescendants.Length(); else
for (Standard_Integer anIter = 1; anIter <= aNbDesc; ++anIter) {
{ RemoveAncestor (thePtr);
if (myDescendants.Value (anIter) == thePtr)
{
myDescendants.Remove (anIter);
return;
}
}
break;
}
case Graphic3d_TOC_ANCESTOR:
{
const Standard_Integer aNbAncestors = myAncestors.Length();
for (Standard_Integer anIter = 1; anIter <= aNbAncestors; ++anIter)
{
if (myAncestors.Value (anIter) == thePtr)
{
myAncestors.Remove (anIter);
return;
}
}
break;
}
} }
} }

View File

@ -42,8 +42,7 @@ OpenGl_Group::OpenGl_Group (const Handle(Graphic3d_Structure)& theStruct)
myAspectText(NULL), myAspectText(NULL),
myFirst(NULL), myFirst(NULL),
myLast(NULL), myLast(NULL),
myIsRaytracable (Standard_False), myIsRaytracable (Standard_False)
myModificationState (0)
{ {
Handle(OpenGl_Structure) aStruct = Handle(OpenGl_Structure)::DownCast (myStructure->CStructure()); Handle(OpenGl_Structure) aStruct = Handle(OpenGl_Structure)::DownCast (myStructure->CStructure());
if (aStruct == NULL) if (aStruct == NULL)
@ -116,11 +115,10 @@ void OpenGl_Group::UpdateAspectFace (const Standard_Boolean theIsGlobal)
if (myIsRaytracable) if (myIsRaytracable)
{ {
++myModificationState;
OpenGl_Structure* aStruct = GlStruct(); OpenGl_Structure* aStruct = GlStruct();
if (aStruct != NULL) if (aStruct != NULL)
{ {
aStruct->UpdateStateWithAncestorStructures(); aStruct->UpdateStateIfRaytracable (Standard_False);
} }
} }
} }
@ -302,14 +300,12 @@ void OpenGl_Group::AddElement (OpenGl_Element* theElem)
if (OpenGl_Raytrace::IsRaytracedElement (aNode)) if (OpenGl_Raytrace::IsRaytracedElement (aNode))
{ {
myModificationState++;
myIsRaytracable = Standard_True; myIsRaytracable = Standard_True;
OpenGl_Structure* aStruct = GlStruct(); OpenGl_Structure* aStruct = GlStruct();
if (aStruct != NULL) if (aStruct != NULL)
{ {
aStruct->UpdateStateWithAncestorStructures(); aStruct->UpdateStateIfRaytracable (Standard_False);
aStruct->SetRaytracableWithAncestorStructures();
} }
} }
} }
@ -366,6 +362,8 @@ void OpenGl_Group::Clear (const Standard_Boolean theToUpdateStructureMgr)
Release (aCtx); Release (aCtx);
Graphic3d_Group::Clear (theToUpdateStructureMgr); Graphic3d_Group::Clear (theToUpdateStructureMgr);
myIsRaytracable = Standard_False;
} }
// ======================================================================= // =======================================================================

View File

@ -104,9 +104,6 @@ public:
//! Returns OpenGL face aspect. //! Returns OpenGL face aspect.
const OpenGl_AspectFace* AspectFace() const { return myAspectFace; } const OpenGl_AspectFace* AspectFace() const { return myAspectFace; }
//! Returns modification state for ray-tracing.
Standard_Size ModificationState() const { return myModificationState; }
//! Is the group ray-tracable (contains ray-tracable elements)? //! Is the group ray-tracable (contains ray-tracable elements)?
Standard_Boolean IsRaytracable() const { return myIsRaytracable; } Standard_Boolean IsRaytracable() const { return myIsRaytracable; }
@ -128,7 +125,6 @@ protected:
OpenGl_ElementNode* myLast; OpenGl_ElementNode* myLast;
Standard_Boolean myIsRaytracable; Standard_Boolean myIsRaytracable;
Standard_Size myModificationState;
public: public:

View File

@ -516,10 +516,9 @@ namespace OpenGl_Raytrace
// function : IsRaytracedGroup // function : IsRaytracedGroup
// purpose : Checks to see if the group contains ray-trace geometry // purpose : Checks to see if the group contains ray-trace geometry
// ======================================================================= // =======================================================================
Standard_Boolean IsRaytracedGroup (const OpenGl_Group *theGroup) Standard_Boolean IsRaytracedGroup (const OpenGl_Group* theGroup)
{ {
const OpenGl_ElementNode* aNode; for (const OpenGl_ElementNode* aNode = theGroup->FirstNode(); aNode != NULL; aNode = aNode->next)
for (aNode = theGroup->FirstNode(); aNode != NULL; aNode = aNode->next)
{ {
if (IsRaytracedElement (aNode)) if (IsRaytracedElement (aNode))
{ {
@ -535,17 +534,12 @@ namespace OpenGl_Raytrace
// ======================================================================= // =======================================================================
Standard_Boolean IsRaytracedStructure (const OpenGl_Structure* theStructure) Standard_Boolean IsRaytracedStructure (const OpenGl_Structure* theStructure)
{ {
for (OpenGl_Structure::GroupIterator aGroupIter (theStructure->DrawGroups()); for (OpenGl_Structure::GroupIterator anIter (theStructure->DrawGroups()); anIter.More(); anIter.Next())
aGroupIter.More(); aGroupIter.Next())
{ {
if (aGroupIter.Value()->IsRaytracable()) if (anIter.Value()->IsRaytracable())
return Standard_True; {
}
for (OpenGl_ListOfStructure::Iterator anIts (theStructure->ConnectedStructures());
anIts.More(); anIts.Next())
{
if (IsRaytracedStructure (anIts.Value()))
return Standard_True; return Standard_True;
}
} }
return Standard_False; return Standard_False;
} }

View File

@ -119,17 +119,18 @@ public:
// ======================================================================= // =======================================================================
OpenGl_Structure::OpenGl_Structure (const Handle(Graphic3d_StructureManager)& theManager) OpenGl_Structure::OpenGl_Structure (const Handle(Graphic3d_StructureManager)& theManager)
: Graphic3d_CStructure (theManager), : Graphic3d_CStructure (theManager),
myTransformation(NULL), myTransformation (NULL),
myTransPers(NULL), myTransPers (NULL),
myAspectLine(NULL), myAspectLine (NULL),
myAspectFace(NULL), myAspectFace (NULL),
myAspectMarker(NULL), myAspectMarker (NULL),
myAspectText(NULL), myAspectText (NULL),
myHighlightColor(NULL), myHighlightColor (NULL),
myIsRaytracable (Standard_False), myInstancedStructure (NULL),
myModificationState (0), myIsRaytracable (Standard_False),
myIsCulled (Standard_True), myModificationState (0),
myIsMirrored (Standard_False) myIsCulled (Standard_True),
myIsMirrored (Standard_False)
{ {
// //
} }
@ -189,9 +190,9 @@ void OpenGl_Structure::UpdateTransformation()
matcpy (myTransformation->mat, &Graphic3d_CStructure::Transformation[0][0]); matcpy (myTransformation->mat, &Graphic3d_CStructure::Transformation[0][0]);
if (myIsRaytracable) if (IsRaytracable())
{ {
UpdateStateWithAncestorStructures(); ++myModificationState;
} }
} }
@ -236,9 +237,9 @@ void OpenGl_Structure::SetAspectFace (const CALL_DEF_CONTEXTFILLAREA& theAspect)
} }
myAspectFace->SetAspect (theAspect); myAspectFace->SetAspect (theAspect);
if (myIsRaytracable) if (IsRaytracable())
{ {
UpdateStateWithAncestorStructures(); ++myModificationState;
} }
} }
@ -365,111 +366,41 @@ void OpenGl_Structure::clearHighlightColor (const Handle(OpenGl_Context)& theGlC
// ======================================================================= // =======================================================================
void OpenGl_Structure::OnVisibilityChanged() void OpenGl_Structure::OnVisibilityChanged()
{ {
if (myIsRaytracable) if (IsRaytracable())
{ {
UpdateStateWithAncestorStructures(); ++myModificationState;
} }
} }
// ======================================================================= // =======================================================================
// function : RegisterAncestorStructure // function : IsRaytracable
// purpose : // purpose :
// ======================================================================= // =======================================================================
void OpenGl_Structure::RegisterAncestorStructure (const OpenGl_Structure* theStructure) const Standard_Boolean OpenGl_Structure::IsRaytracable() const
{ {
for (OpenGl_ListOfStructure::Iterator anIt (myAncestorStructures); anIt.More(); anIt.Next()) if (!myGroups.IsEmpty())
{ {
if (anIt.Value() == theStructure) return myIsRaytracable; // geometry structure
{ }
return; else if (myInstancedStructure != NULL)
} {
return myInstancedStructure->IsRaytracable(); // instance structure
} }
myAncestorStructures.Append (theStructure); return Standard_False; // has no any groups or structures
} }
// ======================================================================= // =======================================================================
// function : UnregisterAncestorStructure // function : UpdateRaytracableState
// purpose : // purpose :
// ======================================================================= // =======================================================================
void OpenGl_Structure::UnregisterAncestorStructure (const OpenGl_Structure* theStructure) const void OpenGl_Structure::UpdateStateIfRaytracable (const Standard_Boolean toCheck) const
{ {
for (OpenGl_ListOfStructure::Iterator anIt (myAncestorStructures); anIt.More(); anIt.Next()) myIsRaytracable = !toCheck || OpenGl_Raytrace::IsRaytracedStructure (this);
if (IsRaytracable())
{ {
if (anIt.Value() == theStructure) ++myModificationState;
{
myAncestorStructures.Remove (anIt);
return;
}
}
}
// =======================================================================
// function : UnregisterFromAncestorStructure
// purpose :
// =======================================================================
void OpenGl_Structure::UnregisterFromAncestorStructure() const
{
for (OpenGl_ListOfStructure::Iterator anIta (myAncestorStructures); anIta.More(); anIta.Next())
{
OpenGl_Structure* anAncestor = const_cast<OpenGl_Structure*> (anIta.ChangeValue());
for (OpenGl_ListOfStructure::Iterator anIts (anAncestor->myConnected); anIts.More(); anIts.Next())
{
if (anIts.Value() == this)
{
anAncestor->myConnected.Remove (anIts);
return;
}
}
}
}
// =======================================================================
// function : UpdateStateWithAncestorStructures
// purpose :
// =======================================================================
void OpenGl_Structure::UpdateStateWithAncestorStructures() const
{
myModificationState++;
for (OpenGl_ListOfStructure::Iterator anIt (myAncestorStructures); anIt.More(); anIt.Next())
{
anIt.Value()->UpdateStateWithAncestorStructures();
}
}
// =======================================================================
// function : UpdateRaytracableWithAncestorStructures
// purpose :
// =======================================================================
void OpenGl_Structure::UpdateRaytracableWithAncestorStructures() const
{
myIsRaytracable = OpenGl_Raytrace::IsRaytracedStructure (this);
if (!myIsRaytracable)
{
for (OpenGl_ListOfStructure::Iterator anIt (myAncestorStructures); anIt.More(); anIt.Next())
{
anIt.Value()->UpdateRaytracableWithAncestorStructures();
}
}
}
// =======================================================================
// function : SetRaytracableWithAncestorStructures
// purpose :
// =======================================================================
void OpenGl_Structure::SetRaytracableWithAncestorStructures() const
{
myIsRaytracable = Standard_True;
for (OpenGl_ListOfStructure::Iterator anIt (myAncestorStructures); anIt.More(); anIt.Next())
{
if (!anIt.Value()->IsRaytracable())
{
anIt.Value()->SetRaytracableWithAncestorStructures();
}
} }
} }
@ -479,17 +410,17 @@ void OpenGl_Structure::SetRaytracableWithAncestorStructures() const
// ======================================================================= // =======================================================================
void OpenGl_Structure::Connect (Graphic3d_CStructure& theStructure) void OpenGl_Structure::Connect (Graphic3d_CStructure& theStructure)
{ {
OpenGl_Structure* aStruct = (OpenGl_Structure* )&theStructure; OpenGl_Structure* aStruct = static_cast<OpenGl_Structure*> (&theStructure);
Disconnect (theStructure);
myConnected.Append (aStruct); Standard_ASSERT_RAISE (myInstancedStructure == NULL || myInstancedStructure == aStruct,
"Error! Instanced structure is already defined");
myInstancedStructure = aStruct;
if (aStruct->IsRaytracable()) if (aStruct->IsRaytracable())
{ {
UpdateStateWithAncestorStructures(); UpdateStateIfRaytracable (Standard_False);
SetRaytracableWithAncestorStructures();
} }
aStruct->RegisterAncestorStructure (this);
} }
// ======================================================================= // =======================================================================
@ -498,22 +429,15 @@ void OpenGl_Structure::Connect (Graphic3d_CStructure& theStructure)
// ======================================================================= // =======================================================================
void OpenGl_Structure::Disconnect (Graphic3d_CStructure& theStructure) void OpenGl_Structure::Disconnect (Graphic3d_CStructure& theStructure)
{ {
OpenGl_Structure* aStruct = (OpenGl_Structure* )&theStructure; OpenGl_Structure* aStruct = static_cast<OpenGl_Structure*> (&theStructure);
for (OpenGl_ListOfStructure::Iterator anIter (myConnected); anIter.More(); anIter.Next())
if (myInstancedStructure == aStruct)
{ {
// Check for the given structure myInstancedStructure = NULL;
if (anIter.Value() == aStruct)
if (aStruct->IsRaytracable())
{ {
myConnected.Remove (anIter); UpdateStateIfRaytracable();
if (aStruct->IsRaytracable())
{
UpdateStateWithAncestorStructures();
UpdateRaytracableWithAncestorStructures();
}
aStruct->UnregisterAncestorStructure (this);
return;
} }
} }
} }
@ -545,12 +469,14 @@ void OpenGl_Structure::RemoveGroup (const Handle(Graphic3d_Group)& theGroup)
// Check for the given group // Check for the given group
if (aGroupIter.Value() == theGroup) if (aGroupIter.Value() == theGroup)
{ {
const Standard_Boolean wasRaytracable =
static_cast<const OpenGl_Group&> (*theGroup).IsRaytracable();
theGroup->Clear (Standard_False); theGroup->Clear (Standard_False);
if (((OpenGl_Group* )theGroup.operator->())->IsRaytracable()) if (wasRaytracable)
{ {
UpdateStateWithAncestorStructures(); UpdateStateIfRaytracable();
UpdateRaytracableWithAncestorStructures();
} }
myGroups.Remove (aGroupIter); myGroups.Remove (aGroupIter);
@ -588,8 +514,7 @@ void OpenGl_Structure::Clear (const Handle(OpenGl_Context)& theGlCtx)
if (aRaytracableGroupDeleted) if (aRaytracableGroupDeleted)
{ {
UpdateStateWithAncestorStructures(); myIsRaytracable = Standard_False;
UpdateRaytracableWithAncestorStructures();
} }
Is2dText = Standard_False; Is2dText = Standard_False;
@ -695,12 +620,10 @@ void OpenGl_Structure::Render (const Handle(OpenGl_Workspace) &theWorkspace) con
if (myHighlightColor) if (myHighlightColor)
theWorkspace->HighlightColor = myHighlightColor; theWorkspace->HighlightColor = myHighlightColor;
// Render connected structures // Render instanced structure (if exists)
OpenGl_ListOfStructure::Iterator anIter (myConnected); if (myInstancedStructure != NULL)
while (anIter.More())
{ {
anIter.Value()->RenderGeometry (theWorkspace); myInstancedStructure->RenderGeometry (theWorkspace);
anIter.Next();
} }
// Set up plane equations for non-structure transformed global model-view matrix // Set up plane equations for non-structure transformed global model-view matrix
@ -816,9 +739,6 @@ void OpenGl_Structure::Release (const Handle(OpenGl_Context)& theGlCtx)
OpenGl_Element::Destroy (theGlCtx.operator->(), myAspectMarker); OpenGl_Element::Destroy (theGlCtx.operator->(), myAspectMarker);
OpenGl_Element::Destroy (theGlCtx.operator->(), myAspectText); OpenGl_Element::Destroy (theGlCtx.operator->(), myAspectText);
clearHighlightColor (theGlCtx); clearHighlightColor (theGlCtx);
// Remove from connected list of ancestor
UnregisterFromAncestorStructure();
} }
// ======================================================================= // =======================================================================

View File

@ -175,8 +175,8 @@ public:
//! and will lead to broken visualization due to loosed data. //! and will lead to broken visualization due to loosed data.
Standard_EXPORT void ReleaseGlResources (const Handle(OpenGl_Context)& theGlCtx); Standard_EXPORT void ReleaseGlResources (const Handle(OpenGl_Context)& theGlCtx);
//! Returns list of connected OpenGL structures. //! Returns instanced OpenGL structure.
const OpenGl_ListOfStructure& ConnectedStructures() const { return myConnected; } const OpenGl_Structure* InstancedStructure() const { return myInstancedStructure; }
//! Returns OpenGL face aspect. //! Returns OpenGL face aspect.
const OpenGl_AspectFace* AspectFace() const { return myAspectFace; } const OpenGl_AspectFace* AspectFace() const { return myAspectFace; }
@ -194,29 +194,14 @@ public:
void ResetModificationState() const { myModificationState = 0; } void ResetModificationState() const { myModificationState = 0; }
//! Is the structure ray-tracable (contains ray-tracable elements)? //! Is the structure ray-tracable (contains ray-tracable elements)?
Standard_Boolean IsRaytracable() const { return myIsRaytracable; } Standard_Boolean IsRaytracable() const;
protected: protected:
Standard_EXPORT virtual ~OpenGl_Structure(); Standard_EXPORT virtual ~OpenGl_Structure();
//! Registers ancestor connected structure (for updating ray-tracing state).
void RegisterAncestorStructure (const OpenGl_Structure* theStructure) const;
//! Unregisters ancestor connected structure (for updating ray-tracing state).
void UnregisterAncestorStructure (const OpenGl_Structure* theStructure) const;
//! Unregisters structure from ancestor structure (for updating ray-tracing state).
void UnregisterFromAncestorStructure() const;
//! Updates modification state for structure and its parents.
void UpdateStateWithAncestorStructures() const;
//! Updates ray-tracable status for structure and its parents. //! Updates ray-tracable status for structure and its parents.
void UpdateRaytracableWithAncestorStructures() const; void UpdateStateIfRaytracable (const Standard_Boolean toCheck = Standard_True) const;
//! Sets ray-tracable status for structure and its parents.
void SetRaytracableWithAncestorStructures() const;
protected: protected:
@ -230,15 +215,14 @@ protected:
Handle(OpenGl_Group) myHighlightBox; Handle(OpenGl_Group) myHighlightBox;
TEL_COLOUR* myHighlightColor; TEL_COLOUR* myHighlightColor;
OpenGl_ListOfStructure myConnected; OpenGl_Structure* myInstancedStructure;
mutable OpenGl_ListOfStructure myAncestorStructures; mutable Standard_Boolean myIsRaytracable;
mutable Standard_Boolean myIsRaytracable; mutable Standard_Size myModificationState;
mutable Standard_Size myModificationState;
mutable Standard_Boolean myIsCulled; //!< A status specifying is structure needs to be rendered after BVH tree traverse. mutable Standard_Boolean myIsCulled; //!< A status specifying is structure needs to be rendered after BVH tree traverse.
Standard_Boolean myIsMirrored; //!< Used to tell OpenGl to interpret polygons in clockwise order. Standard_Boolean myIsMirrored; //!< Used to tell OpenGl to interpret polygons in clockwise order.
public: public:

View File

@ -448,6 +448,31 @@ protected: //! @name data types related to ray-tracing
} }
}; };
//! Describes state of OpenGL structure.
struct StructState
{
Standard_Size StructureState;
Standard_Size InstancedState;
//! Creates new structure state.
StructState (const Standard_Size theStructureState = 0,
const Standard_Size theInstancedState = 0)
: StructureState (theStructureState),
InstancedState (theInstancedState)
{
//
}
//! Creates new structure state.
StructState (const OpenGl_Structure* theStructure)
{
StructureState = theStructure->ModificationState();
InstancedState = theStructure->InstancedStructure() != NULL ?
theStructure->InstancedStructure()->ModificationState() : 0;
}
};
protected: //! @name methods related to ray-tracing protected: //! @name methods related to ray-tracing
//! Updates 3D scene geometry for ray-tracing. //! Updates 3D scene geometry for ray-tracing.
@ -671,7 +696,7 @@ protected: //! @name fields related to ray-tracing
Standard_Integer myUniformLocations[2][OpenGl_RT_NbVariables]; Standard_Integer myUniformLocations[2][OpenGl_RT_NbVariables];
//! State of OpenGL structures reflected to ray-tracing. //! State of OpenGL structures reflected to ray-tracing.
std::map<const OpenGl_Structure*, Standard_Size> myStructureStates; std::map<const OpenGl_Structure*, StructState> myStructureStates;
//! PrimitiveArray to TriangleSet map for scene partial update. //! PrimitiveArray to TriangleSet map for scene partial update.
std::map<Standard_Size, OpenGl_TriangleSet*> myArrayToTrianglesMap; std::map<Standard_Size, OpenGl_TriangleSet*> myArrayToTrianglesMap;

View File

@ -157,7 +157,7 @@ Standard_Boolean OpenGl_View::updateRaytraceGeometry (const RaytraceUpdateMode
else if (theMode == OpenGl_GUM_REBUILD) else if (theMode == OpenGl_GUM_REBUILD)
{ {
// Actualize the hash map of structures - remove out-of-date records // Actualize the hash map of structures - remove out-of-date records
std::map<const OpenGl_Structure*, Standard_Size>::iterator anIter = myStructureStates.begin(); std::map<const OpenGl_Structure*, StructState>::iterator anIter = myStructureStates.begin();
while (anIter != myStructureStates.end()) while (anIter != myStructureStates.end())
{ {
@ -209,14 +209,18 @@ Standard_Boolean OpenGl_View::toUpdateStructure (const OpenGl_Structure* theStru
return Standard_False; // did not contain ray-trace elements return Standard_False; // did not contain ray-trace elements
} }
std::map<const OpenGl_Structure*, Standard_Size>::iterator aStructState = myStructureStates.find (theStructure); std::map<const OpenGl_Structure*, StructState>::iterator aStructState = myStructureStates.find (theStructure);
if (aStructState != myStructureStates.end()) if (aStructState == myStructureStates.end() || aStructState->second.StructureState != theStructure->ModificationState())
{ {
return aStructState->second != theStructure->ModificationState(); return Standard_True;
}
else if (theStructure->InstancedStructure() != NULL)
{
return aStructState->second.InstancedState != theStructure->InstancedStructure()->ModificationState();
} }
return Standard_True; return Standard_False;
} }
// ======================================================================= // =======================================================================
@ -354,7 +358,7 @@ Standard_Boolean OpenGl_View::addRaytraceStructure (const OpenGl_Structure*
{ {
if (!theStructure->IsVisible()) if (!theStructure->IsVisible())
{ {
myStructureStates[theStructure] = theStructure->ModificationState(); myStructureStates[theStructure] = StructState (theStructure);
return Standard_True; return Standard_True;
} }
@ -388,16 +392,15 @@ Standard_Boolean OpenGl_View::addRaytraceStructure (const OpenGl_Structure*
theStructure->Transformation()->mat ? aStructTransform : NULL, theGlContext); theStructure->Transformation()->mat ? aStructTransform : NULL, theGlContext);
// Process all connected OpenGL structures // Process all connected OpenGL structures
for (OpenGl_ListOfStructure::Iterator anIts (theStructure->ConnectedStructures()); anIts.More(); anIts.Next()) const OpenGl_Structure* anInstanced = theStructure->InstancedStructure();
if (anInstanced != NULL && anInstanced->IsRaytracable())
{ {
if (anIts.Value()->IsRaytracable()) aResult &= addRaytraceGroups (anInstanced, aStructMatID,
{ theStructure->Transformation()->mat ? aStructTransform : NULL, theGlContext);
aResult &= addRaytraceGroups (anIts.Value(), aStructMatID,
theStructure->Transformation()->mat ? aStructTransform : NULL, theGlContext);
}
} }
myStructureStates[theStructure] = theStructure->ModificationState(); myStructureStates[theStructure] = StructState (theStructure);
return aResult; return aResult;
} }

View File

@ -34,10 +34,10 @@ SelectMgr_SelectableObjectSet::SelectMgr_SelectableObjectSet()
//======================================================================= //=======================================================================
void SelectMgr_SelectableObjectSet::Append (const Handle(SelectMgr_SelectableObject)& theObject) void SelectMgr_SelectableObjectSet::Append (const Handle(SelectMgr_SelectableObject)& theObject)
{ {
myObjects.Append (theObject); if (Size() < myObjects.Add (theObject))
myObjectIdxs.Append (myObjects.Size()); {
MarkDirty();
MarkDirty(); }
} }
//======================================================================= //=======================================================================
@ -47,19 +47,15 @@ void SelectMgr_SelectableObjectSet::Append (const Handle(SelectMgr_SelectableObj
//======================================================================= //=======================================================================
void SelectMgr_SelectableObjectSet::Remove (const Handle(SelectMgr_SelectableObject)& theObject) void SelectMgr_SelectableObjectSet::Remove (const Handle(SelectMgr_SelectableObject)& theObject)
{ {
for (Standard_Integer anObjectIdx = 1; anObjectIdx <= myObjects.Size(); ++anObjectIdx) const Standard_Integer anIndex = myObjects.FindIndex (theObject);
if (anIndex != 0)
{ {
if (myObjects.Value (anObjectIdx) == theObject) Swap (anIndex - 1, Size() - 1);
{
myObjects.Remove (anObjectIdx); myObjects.RemoveLast();
myObjectIdxs.Clear();
for (Standard_Integer anObjIdxsIter = 1; anObjIdxsIter <= myObjects.Size(); ++anObjIdxsIter) MarkDirty();
{
myObjectIdxs.Append (anObjIdxsIter);
}
MarkDirty();
break;
}
} }
} }
@ -88,11 +84,9 @@ Standard_Real SelectMgr_SelectableObjectSet::Center (const Standard_Integer theI
const Standard_Integer theAxis) const const Standard_Integer theAxis) const
{ {
Select3D_BndBox3d aBndBox = Box (theIndex); Select3D_BndBox3d aBndBox = Box (theIndex);
Standard_Real aCenter = theAxis == 0 ? (aBndBox.CornerMin().x() + aBndBox.CornerMax().x()) * 0.5 :
(theAxis == 1 ? (aBndBox.CornerMin().y() + aBndBox.CornerMax().y()) * 0.5 :
(aBndBox.CornerMin().z() + aBndBox.CornerMax().z()) * 0.5);
return aCenter; return (aBndBox.CornerMin()[theAxis] +
aBndBox.CornerMax()[theAxis]) * 0.5;
} }
//======================================================================= //=======================================================================
@ -102,10 +96,15 @@ Standard_Real SelectMgr_SelectableObjectSet::Center (const Standard_Integer theI
void SelectMgr_SelectableObjectSet::Swap (const Standard_Integer theIndex1, void SelectMgr_SelectableObjectSet::Swap (const Standard_Integer theIndex1,
const Standard_Integer theIndex2) const Standard_Integer theIndex2)
{ {
Standard_Integer anObjectIdx1 = myObjectIdxs.Value (theIndex1 + 1); const Standard_Integer aIndex1 = theIndex1 + 1;
Standard_Integer anObjectIdx2 = myObjectIdxs.Value (theIndex2 + 1); const Standard_Integer aIndex2 = theIndex2 + 1;
myObjectIdxs.ChangeValue (theIndex1 + 1) = anObjectIdx2;
myObjectIdxs.ChangeValue (theIndex2 + 1) = anObjectIdx1; Handle(SelectMgr_SelectableObject) anObject1 = myObjects.FindKey (aIndex1);
Handle(SelectMgr_SelectableObject) anObject2 = myObjects.FindKey (aIndex2);
myObjects.Substitute (aIndex1, EMPTY_OBJ);
myObjects.Substitute (aIndex2, anObject1);
myObjects.Substitute (aIndex1, anObject2);
} }
//======================================================================= //=======================================================================
@ -114,18 +113,16 @@ void SelectMgr_SelectableObjectSet::Swap (const Standard_Integer theIndex1,
//======================================================================= //=======================================================================
Standard_Integer SelectMgr_SelectableObjectSet::Size() const Standard_Integer SelectMgr_SelectableObjectSet::Size() const
{ {
return myObjectIdxs.Size(); return myObjects.Size();
} }
//======================================================================= //=======================================================================
// function : GetObjectById // function : GetObjectById
// purpose : Returns object from set by theIndex given // purpose : Returns object from set by theIndex given
//======================================================================= //=======================================================================
const Handle(SelectMgr_SelectableObject)& SelectMgr_SelectableObjectSet::GetObjectById const Handle(SelectMgr_SelectableObject)& SelectMgr_SelectableObjectSet::GetObjectById (const Standard_Integer theIndex) const
(const Standard_Integer theIndex) const
{ {
Standard_Integer anIdx = myObjectIdxs.Value (theIndex + 1); return myObjects.FindKey (theIndex + 1);
return myObjects.Value (anIdx);
} }
//======================================================================= //=======================================================================
@ -134,11 +131,5 @@ const Handle(SelectMgr_SelectableObject)& SelectMgr_SelectableObjectSet::GetObje
//======================================================================= //=======================================================================
Standard_Boolean SelectMgr_SelectableObjectSet::Contains (const Handle(SelectMgr_SelectableObject)& theObject) const Standard_Boolean SelectMgr_SelectableObjectSet::Contains (const Handle(SelectMgr_SelectableObject)& theObject) const
{ {
for (Standard_Integer anObjectIdx = 1; anObjectIdx <= myObjects.Size(); ++anObjectIdx) return myObjects.Contains (theObject);
{
if (myObjects.Value (anObjectIdx) == theObject)
return Standard_True;
}
return Standard_False;
} }

View File

@ -23,18 +23,23 @@
#include <SelectMgr_SelectableObject.hxx> #include <SelectMgr_SelectableObject.hxx>
#include <SelectMgr_VectorTypes.hxx> #include <SelectMgr_VectorTypes.hxx>
#include <NCollection_IndexedMap.hxx>
//! The purpose of this class is to organize all selectable objects into //! The purpose of this class is to organize all selectable objects into
//! data structure, allowing to build BVH tree. For selectable objects //! data structure, allowing to build BVH tree. For selectable objects
//! binned BVH builder is used with 32 bins and 1 element per leaf. //! binned BVH builder is used with 32 bins and 1 element per leaf.
class SelectMgr_SelectableObjectSet : public BVH_PrimitiveSet<Standard_Real, 3> class SelectMgr_SelectableObjectSet : public BVH_PrimitiveSet<Standard_Real, 3>
{ {
Handle(SelectMgr_SelectableObject) EMPTY_OBJ;
public: public:
//! Creates new empty objects set and initializes BVH tree //! Creates new empty objects set and initializes BVH tree
//! builder to Binned builder with 1 element per list //! builder to Binned builder with 1 element per list
SelectMgr_SelectableObjectSet(); SelectMgr_SelectableObjectSet();
virtual ~SelectMgr_SelectableObjectSet() {}; //! Releases resources of selectable object set.
virtual ~SelectMgr_SelectableObjectSet() { }
//! Adds new object to the set and marks BVH tree for rebuild //! Adds new object to the set and marks BVH tree for rebuild
void Append (const Handle(SelectMgr_SelectableObject)& theObject); void Append (const Handle(SelectMgr_SelectableObject)& theObject);
@ -65,8 +70,7 @@ public:
private: private:
NCollection_Sequence<Handle(SelectMgr_SelectableObject)> myObjects; NCollection_IndexedMap<Handle(SelectMgr_SelectableObject)> myObjects;
NCollection_Sequence<Standard_Integer> myObjectIdxs;
}; };
#endif // _SelectMgr_SelectableObjectSet_HeaderFile #endif // _SelectMgr_SelectableObjectSet_HeaderFile

View File

@ -3834,7 +3834,7 @@ static Standard_Integer VConnect (Draw_Interpretor& /*di*/,
//=============================================================================================== //===============================================================================================
//function : VConnectTo //function : VConnectTo
//purpose : Creates and displays AIS_ConnectedInteractive object from input object and location //purpose : Creates and displays AIS_ConnectedInteractive object from input object and location
//Draw arg : vconnectto name Xo Yo Zo object //Draw arg : vconnectto name Xo Yo Zo object [-nodisplay|-noupdate|-update]
//=============================================================================================== //===============================================================================================
static Standard_Integer VConnectTo (Draw_Interpretor& /*di*/, static Standard_Integer VConnectTo (Draw_Interpretor& /*di*/,
@ -3843,6 +3843,7 @@ static Standard_Integer VConnectTo (Draw_Interpretor& /*di*/,
{ {
// Check the viewer // Check the viewer
Handle(AIS_InteractiveContext) aContext = ViewerTest::GetAISContext(); Handle(AIS_InteractiveContext) aContext = ViewerTest::GetAISContext();
ViewerTest_AutoUpdater anUpdateTool (aContext, ViewerTest::CurrentView());
if (aContext.IsNull()) if (aContext.IsNull())
{ {
std::cout << "vconnect error : call vinit before\n"; std::cout << "vconnect error : call vinit before\n";
@ -3925,10 +3926,15 @@ static Standard_Integer VConnectTo (Draw_Interpretor& /*di*/,
anArg.LowerCase(); anArg.LowerCase();
if (anArg == "-nodisplay") if (anArg == "-nodisplay")
return 0; return 0;
if (!anUpdateTool.parseRedrawMode (anArg))
{
std::cout << "Warning! Unknown argument '" << anArg << "' passed, -nodisplay|-noupdate|-update expected at this point.\n";
}
} }
// Display connected object // Display connected object
TheAISContext()->Display (aConnected); TheAISContext()->Display (aConnected, Standard_False);
return 0; return 0;
} }
@ -6026,7 +6032,7 @@ void ViewerTest::ObjectCommands(Draw_Interpretor& theCommands)
__FILE__, VConnect, group); __FILE__, VConnect, group);
theCommands.Add("vconnectto", theCommands.Add("vconnectto",
"vconnectto : instance_name Xo Yo Zo object [-nodisplay]" "vconnectto : instance_name Xo Yo Zo object [-nodisplay|-noupdate|-update]"
" Makes an instance 'instance_name' of 'object' with position (Xo Yo Zo)." " Makes an instance 'instance_name' of 'object' with position (Xo Yo Zo)."
"\n\t\t: -nodisplay - only creates interactive object, but not displays it", "\n\t\t: -nodisplay - only creates interactive object, but not displays it",
__FILE__, VConnectTo,group); __FILE__, VConnectTo,group);

59
tests/bugs/vis/bug26029 Normal file
View File

@ -0,0 +1,59 @@
puts "============"
puts "0026029: Visualization - Poor performance of connected objects"
puts "============"
puts ""
###########################################################
# Time spent on computation of large number of connected objects
# should grow linearly with the number of objects
###########################################################
pload MODELING VISUALIZATION
psphere s 0.5
tclean s
incmesh s 0.1
trinfo s
vinit View1
vclear
vaxo
#vcaps -vbo 0
vsetdispmode 1
vdefaults -defl 1.0 -autoTriang off
vdisplay s
# disable output of commands
decho off
set aNb1 100
# display 100x100 connected instances of single presentation
puts "Creating [expr $aNb1*$aNb1] instances..."
set t [time {for {set i 0} {$i < $aNb1} {incr i} {for {set j 0} {$j < $aNb1} {incr j} {vconnectto s_${i}_${j} ${i} ${j} 0 s -noupdate}}}]
set d1 [lindex $t 0]
puts "Done in $d1 microseconds!\n"
vclear
set aNb2 200
# display 200x200 connected instances of single presentation
puts "Creating [expr $aNb2*$aNb2] instances..."
set t [time {for {set i 0} {$i < $aNb2} {incr i} {for {set j 0} {$j < $aNb2} {incr j} {vconnectto s_${i}_${j} ${i} ${j} 0 s -noupdate}}}]
set d2 [lindex $t 0]
puts "Done in $d2 microseconds!\n"
vclear
# compare two CPU times: the ratio should be quasi-linear
set expected_ratio [expr double($aNb2 * $aNb2) / double($aNb1 * $aNb1)]
set actual_ratio [expr double($d2) / double($d1)]
puts "Comparing CPU time for the two test cases..."
puts "============================================="
puts "Expected ratio: $expected_ratio"
puts "Actual ratio: $actual_ratio"
puts "============================================="
# Allow 50% deviation from linear growth
if {[expr $actual_ratio / $expected_ratio] > 1.5} {
puts "Error: non-linear time growth detected!"
} else {
puts "Test passed!"
}