mirror of
https://bitbucket.org/smil3y/kdelibs.git
synced 2025-02-24 10:52:49 +00:00
240 lines
5.3 KiB
C++
240 lines
5.3 KiB
C++
/* -*- C++ -*-
|
|
|
|
This file implements the Job class.
|
|
|
|
$ Author: Mirko Boehm $
|
|
$ Copyright: (C) 2004-2013 Mirko Boehm $
|
|
$ Contact: mirko@kde.org
|
|
http://www.kde.org
|
|
http://creative-destruction.me $
|
|
|
|
This library is free software; you can redistribute it and/or
|
|
modify it under the terms of the GNU Library General Public
|
|
License as published by the Free Software Foundation; either
|
|
version 2 of the License, or (at your option) any later version.
|
|
|
|
This library is distributed in the hope that it will be useful,
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
Library General Public License for more details.
|
|
|
|
You should have received a copy of the GNU Library General Public License
|
|
along with this library; see the file COPYING.LIB. If not, write to
|
|
the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
|
|
Boston, MA 02110-1301, USA.
|
|
|
|
$Id: Job.cpp 20 2005-08-08 21:02:51Z mirko $
|
|
*/
|
|
|
|
#include <QtCore/QSet>
|
|
#include <QtCore/QList>
|
|
#include <QtCore/QMutex>
|
|
#include <QtCore/QObject>
|
|
#include <QtCore/QMap>
|
|
#include <QtCore/qobjectdefs.h>
|
|
#include <QtCore/QWaitCondition>
|
|
#include <kdebug.h>
|
|
|
|
#include "Job.h"
|
|
#include "Job_p.h"
|
|
#include "Thread.h"
|
|
#include "QueuePolicy.h"
|
|
#include "DependencyPolicy.h"
|
|
|
|
using namespace ThreadWeaver;
|
|
|
|
class ThreadWeaver::QueuePolicyList : public QList<QueuePolicy*> {};
|
|
|
|
class Job::Private
|
|
{
|
|
public:
|
|
Private ()
|
|
: thread (0)
|
|
, queuePolicies ( new QueuePolicyList )
|
|
, mutex (new QMutex (QMutex::NonRecursive) )
|
|
, finished (false)
|
|
{}
|
|
|
|
~Private()
|
|
{
|
|
delete queuePolicies;
|
|
delete mutex;
|
|
}
|
|
|
|
/* The thread that executes this job. Zero when the job is not executed. */
|
|
Thread * thread;
|
|
|
|
/* The list of QueuePolicies assigned to this Job. */
|
|
QueuePolicyList* queuePolicies;
|
|
|
|
QMutex *mutex;
|
|
/* d->finished is set to true when the Job has been executed. */
|
|
bool finished;
|
|
};
|
|
|
|
Job::Job ( QObject *parent )
|
|
: QObject (parent)
|
|
, d(new Private())
|
|
{
|
|
}
|
|
|
|
Job::~Job()
|
|
{
|
|
for ( int index = 0; index < d->queuePolicies->size(); ++index )
|
|
{
|
|
d->queuePolicies->at( index )->destructed( this );
|
|
}
|
|
|
|
delete d;
|
|
}
|
|
|
|
ThreadWeaver::JobRunHelper::JobRunHelper()
|
|
: QObject ( 0 )
|
|
{
|
|
}
|
|
|
|
void ThreadWeaver::JobRunHelper::runTheJob ( Thread* th, Job* job )
|
|
{
|
|
Q_ASSERT ( th == thread() );
|
|
job->d->mutex->lock();
|
|
job->d->thread = th;
|
|
job->d->mutex->unlock();
|
|
|
|
emit ( started ( job ) );
|
|
|
|
job->run();
|
|
|
|
job->d->mutex->lock();
|
|
job->d->thread = 0;
|
|
job->setFinished (true);
|
|
job->d->mutex->unlock();
|
|
job->freeQueuePolicyResources();
|
|
|
|
if ( ! job->success() )
|
|
{
|
|
emit ( failed( job ) );
|
|
}
|
|
|
|
emit ( done( job ) );
|
|
}
|
|
|
|
void Job::execute(Thread *th)
|
|
{
|
|
// Q_ASSERT (sm_dep()->values(this).isEmpty());
|
|
JobRunHelper helper;
|
|
connect ( &helper, SIGNAL (started(ThreadWeaver::Job*)),
|
|
SIGNAL (started(ThreadWeaver::Job*)) );
|
|
connect ( &helper, SIGNAL (done(ThreadWeaver::Job*)),
|
|
SIGNAL (done(ThreadWeaver::Job*)) );
|
|
connect ( &helper, SIGNAL(failed(ThreadWeaver::Job*)),
|
|
SIGNAL(failed(ThreadWeaver::Job*)) );
|
|
|
|
kDebug() << "executing job of type"
|
|
<< metaObject()->className() << objectName()
|
|
<< "in thread" << th->id();
|
|
helper.runTheJob( th, this );
|
|
kDebug() << "finished execution of job in thread" << th->id();
|
|
}
|
|
|
|
int Job::priority () const
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
bool Job::success () const
|
|
{
|
|
return true;
|
|
}
|
|
|
|
void Job::freeQueuePolicyResources()
|
|
{
|
|
for ( int index = 0; index < d->queuePolicies->size(); ++index )
|
|
{
|
|
d->queuePolicies->at( index )->free( this );
|
|
}
|
|
}
|
|
|
|
void Job::aboutToBeQueued ( WeaverInterface* )
|
|
{
|
|
}
|
|
|
|
void Job::aboutToBeDequeued ( WeaverInterface* )
|
|
{
|
|
}
|
|
|
|
bool Job::canBeExecuted()
|
|
{
|
|
QueuePolicyList acquired;
|
|
|
|
bool success = true;
|
|
|
|
if ( d->queuePolicies->size() > 0 )
|
|
{
|
|
kDebug() << "acquiring permission from" << d->queuePolicies->size()
|
|
<< "queue policies";
|
|
for ( int index = 0; index < d->queuePolicies->size(); ++index )
|
|
{
|
|
if ( d->queuePolicies->at( index )->canRun( this ) )
|
|
{
|
|
acquired.append( d->queuePolicies->at( index ) );
|
|
} else {
|
|
success = false;
|
|
break;
|
|
}
|
|
}
|
|
|
|
kDebug() << "queue policies returned" << success;
|
|
if ( ! success )
|
|
{
|
|
|
|
for ( int index = 0; index < acquired.size(); ++index )
|
|
{
|
|
acquired.at( index )->release( this );
|
|
}
|
|
}
|
|
} else {
|
|
kDebug( "no queue policies, this job can be executed." );
|
|
}
|
|
|
|
return success;
|
|
}
|
|
|
|
void Job::assignQueuePolicy( QueuePolicy* policy )
|
|
{
|
|
if ( ! d->queuePolicies->contains( policy ) )
|
|
{
|
|
d->queuePolicies->append( policy );
|
|
}
|
|
}
|
|
|
|
void Job::removeQueuePolicy( QueuePolicy* policy )
|
|
{
|
|
int index = d->queuePolicies->indexOf( policy );
|
|
if ( index != -1 )
|
|
{
|
|
d->queuePolicies->removeAt( index );
|
|
}
|
|
}
|
|
|
|
bool Job::isFinished() const
|
|
{
|
|
return d->finished;
|
|
}
|
|
|
|
Thread* Job::thread()
|
|
{
|
|
return d->thread;
|
|
}
|
|
|
|
void Job::setFinished ( bool status )
|
|
{
|
|
d->finished = status;
|
|
}
|
|
|
|
// QMutex& Job::mutex()
|
|
// {
|
|
// return * d->mutex;
|
|
// }
|
|
|
|
#include "moc_Job.cpp"
|
|
#include "moc_Job_p.cpp"
|