Files
ThirdParty-6/ParaView-5.0.1/Qt/Components/pqLiveInsituManager.cxx

518 lines
16 KiB
C++

/*=========================================================================
Program: ParaView
Module: pqLiveInsituManager.cxx
Copyright (c) 2005-2008 Sandia Corporation, Kitware Inc.
All rights reserved.
ParaView is a free software; you can redistribute it and/or modify it
under the terms of the ParaView license version 1.2.
See License_v1.2.txt for the full ParaView license.
A copy of this license can be obtained by contacting
Kitware Inc.
28 Corporate Drive
Clifton Park, NY 12065
USA
THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHORS OR
CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
=========================================================================*/
#include "pqLiveInsituManager.h"
#include <limits>
#include <iostream>
#include <QInputDialog>
#include <QMessageBox>
#include <QTextStream>
#include <QVariant>
#include "pqActiveObjects.h"
#include "pqApplicationCore.h"
#include "pqCoreUtilities.h"
#include "pqLiveInsituVisualizationManager.h"
#include "pqProxy.h"
#include "pqServer.h"
#include "pqServerManagerModel.h"
#include "pqTestUtility.h"
#include "pqWidgetEventPlayer.h"
#include "pqEventDispatcher.h"
#include "vtkProcessModule.h"
#include "vtkPVDataInformation.h"
#include "vtkSMLiveInsituLinkProxy.h"
#include "vtkSMPropertyHelper.h"
#include "vtkSMSessionProxyManager.h"
#include "vtkSMSourceProxy.h"
//#define pqLiveInsituManagerDebugMacro(x) std::cerr << x << endl;
#define pqLiveInsituManagerDebugMacro(x)
namespace
{
class pqLiveInsituEventPlayer : public pqWidgetEventPlayer
{
typedef pqWidgetEventPlayer Superclass;
public:
QPointer<pqLiveInsituManager> Manager;
pqLiveInsituEventPlayer(QObject* p) :
Superclass(p)
{
}
virtual bool playEvent(
QObject*,
const QString& command, const QString& arguments, bool& error)
{
if (command == "pqLiveInsituManager" && this->Manager)
{
if (arguments.startsWith("wait_timestep"))
{
QString arg = arguments;
QString subCommand;
vtkIdType timeStep;
QTextStream istr(&arg);
istr >> subCommand >> timeStep;
this->Manager->waitTimestep(timeStep);
}
else if (arguments == "wait_breakpoint_hit")
{
this->Manager->waitBreakpointHit();
}
else
{
error = true;
}
return true;
}
else
{
return false;
}
}
};
};
double pqLiveInsituManager::INVALID_TIME =
- std::numeric_limits<double>().max();
vtkIdType pqLiveInsituManager::INVALID_TIME_STEP =
std::numeric_limits<vtkIdType>().min();
//-----------------------------------------------------------------------------
pqLiveInsituManager* pqLiveInsituManager::instance()
{
static pqLiveInsituManager* s_liveInsituManager = NULL;
if (! s_liveInsituManager)
{
s_liveInsituManager = new pqLiveInsituManager();
}
return s_liveInsituManager;
}
//-----------------------------------------------------------------------------
pqLiveInsituManager::pqLiveInsituManager() :
BreakpointTime(INVALID_TIME),
Time(INVALID_TIME),
BreakpointTimeStep(INVALID_TIME_STEP),
TimeStep(INVALID_TIME_STEP)
{
pqLiveInsituEventPlayer* player = new pqLiveInsituEventPlayer(NULL);
player->Manager = this;
// the testUtility takes ownership of the player.
pqApplicationCore::instance()->testUtility()->eventPlayer()->
addWidgetEventPlayer(player);
pqServerManagerModel* model =
pqApplicationCore::instance()->getServerManagerModel();
QObject::connect(model,
SIGNAL(sourceAdded(pqPipelineSource*)),
this, SLOT(onSourceAdded(pqPipelineSource*)));
QObject::connect(this, SIGNAL(breakpointHit(pqServer*)),
this, SLOT(onBreakpointHit(pqServer*)),
Qt::QueuedConnection);
}
//-----------------------------------------------------------------------------
bool pqLiveInsituManager::isInsituServer(pqServer* server)
{
return server ? (server->getResource().scheme() == "catalyst") : false;
}
//-----------------------------------------------------------------------------
bool pqLiveInsituManager::isWriterParametersProxy(vtkSMProxy* proxy)
{
return proxy && strcmp(proxy->GetXMLGroup(), "insitu_writer_parameters") == 0;
}
//-----------------------------------------------------------------------------
bool pqLiveInsituManager::isInsitu(pqPipelineSource* pipelineSource)
{
pqServer* insituSession =
pqLiveInsituManager::instance()->selectedInsituServer();
if (! insituSession)
{
return false;
}
return insituSession->session() == pipelineSource->getProxy()->GetSession();
}
//-----------------------------------------------------------------------------
bool pqLiveInsituManager::isDisplayServer(pqServer* server)
{
return this->Managers.contains(server);
}
//-----------------------------------------------------------------------------
pqServer* pqLiveInsituManager::selectedInsituServer()
{
pqActiveObjects& ao = pqActiveObjects::instance();
pqServer* as = ao.activeServer();
if (pqLiveInsituManager::isInsituServer(as))
{
return as;
}
else if (this->isDisplayServer(as))
{
return managerFromDisplay(as)->insituSession();
}
else
{
pqServerManagerModel *smModel =
pqApplicationCore::instance()->getServerManagerModel();
QList<pqServer*> servers = smModel->findItems<pqServer*>();
for(QList<pqServer*>::Iterator it = servers.begin();
it != servers.end(); ++it)
{
pqServer* server = *it;
if (pqLiveInsituManager::isInsituServer(server))
{
return server;
}
}
return NULL;
}
}
//-----------------------------------------------------------------------------
vtkSMLiveInsituLinkProxy* pqLiveInsituManager::linkProxy(
pqServer* insituSession)
{
if (insituSession)
{
pqLiveInsituVisualizationManager* mgr =
pqLiveInsituManager::managerFromInsitu(insituSession);
if (mgr)
{
vtkSMLiveInsituLinkProxy* proxy = mgr->getProxy();
return proxy;
}
}
return NULL;
}
//-----------------------------------------------------------------------------
pqLiveInsituVisualizationManager* pqLiveInsituManager::connect(pqServer* server)
{
if (! server)
{
return NULL;
}
ManagersType::iterator it = this->Managers.find(server);
if (it == this->Managers.end())
{
// Make sure we are in multi-server mode
vtkProcessModule::GetProcessModule()->MultipleSessionsSupportOn();
bool user_ok = false;
int portNumber = QInputDialog::getInt(
pqCoreUtilities::mainWidget(),
"Catalyst Server Port",
"Enter the port number to accept connections \nfrom Catalyst on:",
22222, 1024, 0x0fffffff, 1, &user_ok);
if (!user_ok)
{
// user cancelled.
return NULL;
}
pqLiveInsituVisualizationManager* mgr =
new pqLiveInsituVisualizationManager(portNumber, server);
QObject::connect(mgr, SIGNAL(insituDisconnected()),
this, SLOT(onCatalystDisconnected()));
this->Managers[server] = mgr;
QMessageBox::information(pqCoreUtilities::mainWidget(),
"Ready for Catalyst connections",
QString("Accepting connections from Catalyst Co-Processor \n"
"for live-coprocessing on port %1").arg(portNumber));
emit connectionInitiated(server);
return mgr;
}
else
{
qWarning("A Catalyst connection has already been established.");
return *it;
}
}
//-----------------------------------------------------------------------------
pqLiveInsituVisualizationManager* pqLiveInsituManager::managerFromDisplay(
pqServer* displaySession)
{
return this->Managers[displaySession];
}
//-----------------------------------------------------------------------------
pqLiveInsituVisualizationManager* pqLiveInsituManager::managerFromInsitu(
pqServer* insituSession)
{
return qobject_cast<pqLiveInsituVisualizationManager*>(
insituSession->property("LiveInsituVisualizationManager").value<QObject*>());
}
//-----------------------------------------------------------------------------
void pqLiveInsituManager::onCatalystDisconnected()
{
pqLiveInsituVisualizationManager* mgr =
qobject_cast<pqLiveInsituVisualizationManager*>(this->sender());
if (!mgr)
{
return;
}
QMessageBox::information(pqCoreUtilities::mainWidget(),
"Catalyst Disconnected",
"Connection to Catalyst Co-Processor has been terminated involuntarily. "
"This implies either a communication error, or that the "
"Catalyst co-processor has terminated. "
"The Catalyst session will now be cleaned up. "
"You can start a new one if you want to monitor for additional Catalyst "
"connection requests.");
mgr->deleteLater();
// Remove the mgr from the map, so that we can allow the user to connect to
// another Catalyst session, if he wants.
for (ManagersType::iterator iter = this->Managers.begin();
iter != this->Managers.end(); ++iter)
{
if (iter.value() == mgr)
{
this->Managers.erase(iter);
break;
}
}
}
//-----------------------------------------------------------------------------
pqPipelineSource* pqLiveInsituManager::pipelineSource(pqServer* insituSession)
{
if (insituSession)
{
vtkSMSessionProxyManager* manager = insituSession->proxyManager();
vtkSMProxy* proxy = manager->GetProxy("sources", "PVTrivialProducer");
return pqApplicationCore::instance()->getServerManagerModel()->
findItem<pqPipelineSource*>(proxy);
}
return NULL;
}
//-----------------------------------------------------------------------------
void pqLiveInsituManager::time(pqPipelineSource* pipelineSource, double* time,
vtkIdType* timeStep)
{
if (pipelineSource)
{
pqOutputPort* port = pipelineSource->getOutputPort(0);
vtkPVDataInformation* dataInfo = port->getDataInformation();
if (dataInfo->GetHasTime())
{
*time = dataInfo->GetTime();
pqServerManagerModel* model =
pqApplicationCore::instance()->getServerManagerModel();
vtkSMSession* session = pipelineSource->getSourceProxy()->GetSession();
pqServer* insituSession = model->findServer(session);
vtkSMLiveInsituLinkProxy* linkProxy = pqLiveInsituManager::linkProxy(insituSession);
Q_ASSERT (linkProxy);
*timeStep = linkProxy->GetTimeStep();
}
}
else
{
*time = INVALID_TIME;
*timeStep = 0;
}
}
//-----------------------------------------------------------------------------
void pqLiveInsituManager::onSourceAdded(pqPipelineSource* source)
{
vtkSMSourceProxy* sourceProxy = source->getSourceProxy();
pqServerManagerModel* model =
pqApplicationCore::instance()->getServerManagerModel();
vtkSMSession* session = sourceProxy->GetSession();
pqServer* server = model->findServer(session);
if (QString(sourceProxy->GetXMLGroup()) == "sources" &&
QString(sourceProxy->GetXMLName()) == "PVTrivialProducer" &&
pqLiveInsituManager::isInsituServer (server))
{
QObject::connect(source, SIGNAL(dataUpdated(pqPipelineSource*)),
this, SLOT(onDataUpdated(pqPipelineSource*)));
}
}
//-----------------------------------------------------------------------------
bool pqLiveInsituManager::isTimeBreakpointHit() const
{
return this->Time != INVALID_TIME &&
this->BreakpointTime != INVALID_TIME &&
this->BreakpointTime <= this->Time;
}
//-----------------------------------------------------------------------------
bool pqLiveInsituManager::isTimeStepBreakpointHit() const
{
return this->TimeStep != INVALID_TIME_STEP &&
this->BreakpointTimeStep != INVALID_TIME_STEP &&
this->BreakpointTimeStep <= this->TimeStep;
}
//-----------------------------------------------------------------------------
void pqLiveInsituManager::onDataUpdated(pqPipelineSource* source)
{
pqLiveInsituManager::time(source, &this->Time, &this->TimeStep);
emit timeUpdated();
if (isTimeBreakpointHit() || isTimeStepBreakpointHit())
{
pqServerManagerModel* model =
pqApplicationCore::instance()->getServerManagerModel();
vtkSMSession* session = source->getSourceProxy()->GetSession();
pqServer* insituSession = model->findServer(session);
// The server talks to the client using vtkPVSessionBase::NotifyAllClients
// While vtkSMLiveInsituLinkProxy::LoadState notifies us that the time
// changed we try to send to the server an updated property. This does not
// work unless we send this update through the message queue (we wait
// for LoadState to finish).
emit breakpointHit(insituSession);
}
}
//-----------------------------------------------------------------------------
void pqLiveInsituManager::onBreakpointHit(pqServer* insituSession)
{
vtkSMLiveInsituLinkProxy* proxy =
pqLiveInsituManager::linkProxy(insituSession);
if (proxy)
{
vtkSMPropertyHelper(proxy, "SimulationPaused").Set(true);
proxy->UpdateVTKObjects();
this->removeBreakpoint();
}
}
//-----------------------------------------------------------------------------
void pqLiveInsituManager::setBreakpoint(double t)
{
if (this->BreakpointTime != t)
{
pqServer* insituSession = this->selectedInsituServer();
if (insituSession)
{
this->BreakpointTime = t;
this->BreakpointTimeStep = INVALID_TIME_STEP;
emit breakpointAdded(insituSession);
}
}
}
//-----------------------------------------------------------------------------
void pqLiveInsituManager::setBreakpoint(vtkIdType _timeStep)
{
if (this->BreakpointTimeStep != _timeStep)
{
pqServer* insituSession = this->selectedInsituServer();
if (insituSession)
{
this->BreakpointTime = INVALID_TIME;
this->BreakpointTimeStep = _timeStep;
emit breakpointAdded(insituSession);
}
}
}
//-----------------------------------------------------------------------------
void pqLiveInsituManager::removeBreakpoint()
{
if (this->hasBreakpoint())
{
pqServer* insituSession = this->selectedInsituServer();
if (insituSession)
{
this->BreakpointTime = INVALID_TIME;
this->BreakpointTimeStep = INVALID_TIME_STEP;
emit breakpointRemoved(insituSession);
}
}
}
//-----------------------------------------------------------------------------
void pqLiveInsituManager::waitTimestep(vtkIdType ts)
{
pqEventDispatcher::deferEventsIfBlocked(true);
pqLiveInsituVisualizationManager* visManager =
this->managerFromInsitu(this->selectedInsituServer());
pqLiveInsituManagerDebugMacro("===== start waitTimestep("
<< ts << ")" << " ===== "
<< this->timeStep());
while (ts > this->timeStep())
{
QEventLoop loop;
QObject::connect(visManager, SIGNAL(nextTimestepAvailable()),
&loop, SLOT(quit()));
loop.exec();
pqLiveInsituManagerDebugMacro("===== waitTimestep(" << ts << ")" <<
" ===== " << this->timeStep());
}
pqLiveInsituManagerDebugMacro("===== end waitTimestep(" << ts
<< ")" << " ===== " << this->timeStep());
pqEventDispatcher::deferEventsIfBlocked(false);
}
//-----------------------------------------------------------------------------
void pqLiveInsituManager::waitBreakpointHit()
{
pqEventDispatcher::deferEventsIfBlocked(true);
pqLiveInsituManagerDebugMacro("=== begin waitConnected ===");
QEventLoop loop;
QObject::connect(this, SIGNAL(breakpointHit(pqServer*)),
&loop, SLOT(quit()));
loop.exec();
pqLiveInsituManagerDebugMacro("=== end waitConnected ===");
pqEventDispatcher::deferEventsIfBlocked(false);
}