mirror of
https://github.com/tomahawk-player/tomahawk.git
synced 2025-08-11 00:24:12 +02:00
Finish the work begun in 0c68f559
to make scanning more thread-safe
This commit is contained in:
@@ -124,12 +124,12 @@ DirListerThreadController::run()
|
|||||||
QMetaObject::invokeMethod( m_dirLister.data(), "go", Qt::QueuedConnection );
|
QMetaObject::invokeMethod( m_dirLister.data(), "go", Qt::QueuedConnection );
|
||||||
|
|
||||||
exec();
|
exec();
|
||||||
if( !m_dirLister.isNull() )
|
if ( !m_dirLister.isNull() )
|
||||||
delete m_dirLister.data();
|
delete m_dirLister.data();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
MusicScanner::MusicScanner( ScanManager::ScanMode scanMode, const QStringList& paths, quint32 bs )
|
MusicScanner::MusicScanner( MusicScanner::ScanMode scanMode, const QStringList& paths, quint32 bs )
|
||||||
: QObject()
|
: QObject()
|
||||||
, m_scanMode( scanMode )
|
, m_scanMode( scanMode )
|
||||||
, m_paths( paths )
|
, m_paths( paths )
|
||||||
@@ -204,7 +204,7 @@ MusicScanner::scan()
|
|||||||
connect( this, SIGNAL( batchReady( QVariantList, QVariantList ) ),
|
connect( this, SIGNAL( batchReady( QVariantList, QVariantList ) ),
|
||||||
SLOT( commitBatch( QVariantList, QVariantList ) ), Qt::DirectConnection );
|
SLOT( commitBatch( QVariantList, QVariantList ) ), Qt::DirectConnection );
|
||||||
|
|
||||||
if ( m_scanMode == ScanManager::FileScan )
|
if ( m_scanMode == MusicScanner::FileScan )
|
||||||
{
|
{
|
||||||
scanFilePaths();
|
scanFilePaths();
|
||||||
return;
|
return;
|
||||||
@@ -236,7 +236,7 @@ MusicScanner::postOps()
|
|||||||
{
|
{
|
||||||
tDebug( LOGVERBOSE ) << Q_FUNC_INFO;
|
tDebug( LOGVERBOSE ) << Q_FUNC_INFO;
|
||||||
|
|
||||||
if ( m_scanMode == ScanManager::DirScan )
|
if ( m_scanMode == MusicScanner::DirScan )
|
||||||
{
|
{
|
||||||
// any remaining stuff that wasnt emitted as a batch:
|
// any remaining stuff that wasnt emitted as a batch:
|
||||||
foreach( const QString& key, m_filemtimes.keys() )
|
foreach( const QString& key, m_filemtimes.keys() )
|
||||||
|
@@ -23,8 +23,6 @@
|
|||||||
#include "TomahawkSettings.h"
|
#include "TomahawkSettings.h"
|
||||||
#include "database/DatabaseCommand.h"
|
#include "database/DatabaseCommand.h"
|
||||||
|
|
||||||
#include "ScanManager.h"
|
|
||||||
|
|
||||||
/* taglib */
|
/* taglib */
|
||||||
#include <taglib/fileref.h>
|
#include <taglib/fileref.h>
|
||||||
#include <taglib/tag.h>
|
#include <taglib/tag.h>
|
||||||
@@ -100,7 +98,10 @@ class MusicScanner : public QObject
|
|||||||
Q_OBJECT
|
Q_OBJECT
|
||||||
|
|
||||||
public:
|
public:
|
||||||
MusicScanner( ScanManager::ScanMode scanMode, const QStringList& paths, quint32 bs = 0 );
|
enum ScanMode { DirScan, FileScan };
|
||||||
|
enum ScanType { None, Full, Normal, File };
|
||||||
|
|
||||||
|
MusicScanner( MusicScanner::ScanMode scanMode, const QStringList& paths, quint32 bs = 0 );
|
||||||
~MusicScanner();
|
~MusicScanner();
|
||||||
|
|
||||||
signals:
|
signals:
|
||||||
@@ -125,7 +126,7 @@ private slots:
|
|||||||
private:
|
private:
|
||||||
void scanFilePaths();
|
void scanFilePaths();
|
||||||
|
|
||||||
ScanManager::ScanMode m_scanMode;
|
MusicScanner::ScanMode m_scanMode;
|
||||||
QStringList m_paths;
|
QStringList m_paths;
|
||||||
QMap<QString, QString> m_ext2mime; // eg: mp3 -> audio/mpeg
|
QMap<QString, QString> m_ext2mime; // eg: mp3 -> audio/mpeg
|
||||||
unsigned int m_scanned;
|
unsigned int m_scanned;
|
||||||
|
@@ -35,6 +35,35 @@
|
|||||||
#include <QTimer>
|
#include <QTimer>
|
||||||
#include <QSet>
|
#include <QSet>
|
||||||
|
|
||||||
|
|
||||||
|
MusicScannerThreadController::MusicScannerThreadController( QObject* parent )
|
||||||
|
: QThread( parent )
|
||||||
|
, m_bs( 0 )
|
||||||
|
{
|
||||||
|
tDebug() << Q_FUNC_INFO;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
MusicScannerThreadController::~MusicScannerThreadController()
|
||||||
|
{
|
||||||
|
tDebug() << Q_FUNC_INFO;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void
|
||||||
|
MusicScannerThreadController::run()
|
||||||
|
{
|
||||||
|
m_musicScanner = QPointer< MusicScanner >( new MusicScanner( m_mode, m_paths, m_bs ) );
|
||||||
|
connect( m_musicScanner.data(), SIGNAL( finished() ), parent(), SLOT( scannerFinished() ), Qt::QueuedConnection );
|
||||||
|
QMetaObject::invokeMethod( m_musicScanner.data(), "startScan", Qt::QueuedConnection );
|
||||||
|
|
||||||
|
exec();
|
||||||
|
|
||||||
|
if ( !m_musicScanner.isNull() )
|
||||||
|
delete m_musicScanner.data();
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
ScanManager* ScanManager::s_instance = 0;
|
ScanManager* ScanManager::s_instance = 0;
|
||||||
|
|
||||||
|
|
||||||
@@ -50,7 +79,7 @@ ScanManager::ScanManager( QObject* parent )
|
|||||||
, m_musicScannerThreadController( 0 )
|
, m_musicScannerThreadController( 0 )
|
||||||
, m_currScannerPaths()
|
, m_currScannerPaths()
|
||||||
, m_cachedScannerDirs()
|
, m_cachedScannerDirs()
|
||||||
, m_queuedScanType( None )
|
, m_queuedScanType( MusicScanner::None )
|
||||||
, m_updateGUI( true )
|
, m_updateGUI( true )
|
||||||
{
|
{
|
||||||
s_instance = this;
|
s_instance = this;
|
||||||
@@ -76,12 +105,11 @@ ScanManager::~ScanManager()
|
|||||||
{
|
{
|
||||||
qDebug() << Q_FUNC_INFO;
|
qDebug() << Q_FUNC_INFO;
|
||||||
|
|
||||||
if ( !m_scanner.isNull() )
|
if ( m_musicScannerThreadController )
|
||||||
{
|
{
|
||||||
m_musicScannerThreadController->quit();
|
m_musicScannerThreadController->quit();
|
||||||
m_musicScannerThreadController->wait( 60000 );
|
m_musicScannerThreadController->wait( 60000 );
|
||||||
|
|
||||||
delete m_scanner.data();
|
|
||||||
delete m_musicScannerThreadController;
|
delete m_musicScannerThreadController;
|
||||||
m_musicScannerThreadController = 0;
|
m_musicScannerThreadController = 0;
|
||||||
}
|
}
|
||||||
@@ -157,17 +185,17 @@ ScanManager::runNormalScan( bool manualFull )
|
|||||||
|
|
||||||
tDebug( LOGVERBOSE ) << Q_FUNC_INFO;
|
tDebug( LOGVERBOSE ) << Q_FUNC_INFO;
|
||||||
|
|
||||||
if ( m_musicScannerThreadController || !m_scanner.isNull() ) //still running if these are not zero
|
if ( m_musicScannerThreadController ) //still running if these are not zero
|
||||||
{
|
{
|
||||||
if ( m_queuedScanType != Full )
|
if ( m_queuedScanType != MusicScanner::Full )
|
||||||
m_queuedScanType = manualFull ? Full : Normal;
|
m_queuedScanType = manualFull ? MusicScanner::Full : MusicScanner::Normal;
|
||||||
tDebug( LOGVERBOSE ) << "Could not run dir scan, old scan still running";
|
tDebug( LOGVERBOSE ) << "Could not run dir scan, old scan still running";
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
m_scanTimer->stop();
|
m_scanTimer->stop();
|
||||||
m_musicScannerThreadController = new QThread( this );
|
m_musicScannerThreadController = new MusicScannerThreadController( this );
|
||||||
m_currScanMode = DirScan;
|
m_currScanMode = MusicScanner::DirScan;
|
||||||
|
|
||||||
if ( manualFull )
|
if ( manualFull )
|
||||||
{
|
{
|
||||||
@@ -204,17 +232,17 @@ ScanManager::runFileScan( const QStringList& paths, bool updateGUI )
|
|||||||
foreach( const QString& path, paths )
|
foreach( const QString& path, paths )
|
||||||
m_currScannerPaths.insert( path );
|
m_currScannerPaths.insert( path );
|
||||||
|
|
||||||
if ( m_musicScannerThreadController || !m_scanner.isNull() ) //still running if these are not zero
|
if ( m_musicScannerThreadController ) //still running if these are not zero
|
||||||
{
|
{
|
||||||
if ( m_queuedScanType == None )
|
if ( m_queuedScanType == MusicScanner::None )
|
||||||
m_queuedScanType = File;
|
m_queuedScanType = MusicScanner::File;
|
||||||
tDebug( LOGVERBOSE ) << "Could not run file scan, old scan still running";
|
tDebug( LOGVERBOSE ) << "Could not run file scan, old scan still running";
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
m_scanTimer->stop();
|
m_scanTimer->stop();
|
||||||
m_musicScannerThreadController = new QThread( this );
|
m_musicScannerThreadController = new MusicScannerThreadController( this );
|
||||||
m_currScanMode = FileScan;
|
m_currScanMode = MusicScanner::FileScan;
|
||||||
m_updateGUI = updateGUI;
|
m_updateGUI = updateGUI;
|
||||||
|
|
||||||
QMetaObject::invokeMethod( this, "runScan", Qt::QueuedConnection );
|
QMetaObject::invokeMethod( this, "runScan", Qt::QueuedConnection );
|
||||||
@@ -224,7 +252,7 @@ ScanManager::runFileScan( const QStringList& paths, bool updateGUI )
|
|||||||
void
|
void
|
||||||
ScanManager::fileMtimesCheck( const QMap< QString, QMap< unsigned int, unsigned int > >& mtimes )
|
ScanManager::fileMtimesCheck( const QMap< QString, QMap< unsigned int, unsigned int > >& mtimes )
|
||||||
{
|
{
|
||||||
if ( !mtimes.isEmpty() && m_currScanMode == DirScan && TomahawkSettings::instance()->scannerPaths().isEmpty() )
|
if ( !mtimes.isEmpty() && m_currScanMode == MusicScanner::DirScan && TomahawkSettings::instance()->scannerPaths().isEmpty() )
|
||||||
{
|
{
|
||||||
DatabaseCommand_DeleteFiles *cmd = new DatabaseCommand_DeleteFiles( SourceList::instance()->getLocal() );
|
DatabaseCommand_DeleteFiles *cmd = new DatabaseCommand_DeleteFiles( SourceList::instance()->getLocal() );
|
||||||
connect( cmd, SIGNAL( finished() ), SLOT( filesDeleted() ) );
|
connect( cmd, SIGNAL( finished() ), SLOT( filesDeleted() ) );
|
||||||
@@ -253,14 +281,9 @@ ScanManager::runScan()
|
|||||||
|
|
||||||
QStringList paths = m_currScannerPaths.empty() ? TomahawkSettings::instance()->scannerPaths() : m_currScannerPaths.toList();
|
QStringList paths = m_currScannerPaths.empty() ? TomahawkSettings::instance()->scannerPaths() : m_currScannerPaths.toList();
|
||||||
|
|
||||||
if ( m_musicScannerThreadController && m_scanner.isNull() )
|
m_musicScannerThreadController->setScanMode( m_currScanMode );
|
||||||
{
|
m_musicScannerThreadController->setPaths( paths );
|
||||||
m_scanner = QPointer< MusicScanner >( new MusicScanner( m_currScanMode, paths ) );
|
m_musicScannerThreadController->start( QThread::IdlePriority );
|
||||||
m_scanner.data()->moveToThread( m_musicScannerThreadController );
|
|
||||||
connect( m_scanner.data(), SIGNAL( finished() ), SLOT( scannerFinished() ) );
|
|
||||||
m_musicScannerThreadController->start( QThread::IdlePriority );
|
|
||||||
QMetaObject::invokeMethod( m_scanner.data(), "startScan" );
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@@ -268,14 +291,11 @@ void
|
|||||||
ScanManager::scannerFinished()
|
ScanManager::scannerFinished()
|
||||||
{
|
{
|
||||||
tLog( LOGVERBOSE ) << Q_FUNC_INFO;
|
tLog( LOGVERBOSE ) << Q_FUNC_INFO;
|
||||||
if ( !m_scanner.isNull() || m_musicScannerThreadController )
|
if ( m_musicScannerThreadController )
|
||||||
{
|
{
|
||||||
m_musicScannerThreadController->quit();
|
m_musicScannerThreadController->quit();
|
||||||
m_musicScannerThreadController->wait( 60000 );
|
m_musicScannerThreadController->wait( 60000 );
|
||||||
|
|
||||||
if ( !m_scanner.isNull() )
|
|
||||||
delete m_scanner.data();
|
|
||||||
|
|
||||||
delete m_musicScannerThreadController;
|
delete m_musicScannerThreadController;
|
||||||
m_musicScannerThreadController = 0;
|
m_musicScannerThreadController = 0;
|
||||||
}
|
}
|
||||||
@@ -284,21 +304,21 @@ ScanManager::scannerFinished()
|
|||||||
m_updateGUI = true;
|
m_updateGUI = true;
|
||||||
emit finished();
|
emit finished();
|
||||||
|
|
||||||
if ( m_queuedScanType != File )
|
if ( m_queuedScanType != MusicScanner::File )
|
||||||
m_currScannerPaths.clear();
|
m_currScannerPaths.clear();
|
||||||
switch ( m_queuedScanType )
|
switch ( m_queuedScanType )
|
||||||
{
|
{
|
||||||
case Full:
|
case MusicScanner::Full:
|
||||||
case Normal:
|
case MusicScanner::Normal:
|
||||||
QMetaObject::invokeMethod( this, "runNormalScan", Qt::QueuedConnection, Q_ARG( bool, m_queuedScanType == Full ) );
|
QMetaObject::invokeMethod( this, "runNormalScan", Qt::QueuedConnection, Q_ARG( bool, m_queuedScanType == MusicScanner::Full ) );
|
||||||
break;
|
break;
|
||||||
case File:
|
case MusicScanner::File:
|
||||||
QMetaObject::invokeMethod( this, "", Qt::QueuedConnection, Q_ARG( QStringList, QStringList() ) );
|
QMetaObject::invokeMethod( this, "", Qt::QueuedConnection, Q_ARG( QStringList, QStringList() ) );
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
m_queuedScanType = None;
|
m_queuedScanType = MusicScanner::None;
|
||||||
|
|
||||||
m_scanTimer->start();
|
m_scanTimer->start();
|
||||||
}
|
}
|
||||||
|
@@ -23,26 +23,44 @@
|
|||||||
#include "Typedefs.h"
|
#include "Typedefs.h"
|
||||||
#include "DllMacro.h"
|
#include "DllMacro.h"
|
||||||
|
|
||||||
#include <QtCore/QHash>
|
#include "MusicScanner.h"
|
||||||
#include <QtCore/QMap>
|
|
||||||
#include <QtCore/QObject>
|
#include <QHash>
|
||||||
#include <QtCore/QStringList>
|
#include <QMap>
|
||||||
#include <QtCore/QPointer>
|
#include <QObject>
|
||||||
#include <QtCore/QSet>
|
#include <QStringList>
|
||||||
|
#include <QPointer>
|
||||||
|
#include <QSet>
|
||||||
|
#include <QThread>
|
||||||
|
|
||||||
class MusicScanner;
|
|
||||||
class QThread;
|
|
||||||
class QFileSystemWatcher;
|
class QFileSystemWatcher;
|
||||||
class QTimer;
|
class QTimer;
|
||||||
|
|
||||||
|
class MusicScannerThreadController : public QThread
|
||||||
|
{
|
||||||
|
Q_OBJECT
|
||||||
|
|
||||||
|
public:
|
||||||
|
MusicScannerThreadController( QObject* parent );
|
||||||
|
virtual ~MusicScannerThreadController();
|
||||||
|
|
||||||
|
void setScanMode( MusicScanner::ScanMode mode ) { m_mode = mode; }
|
||||||
|
void setPaths( const QStringList& paths ) { m_paths = paths; }
|
||||||
|
void run();
|
||||||
|
|
||||||
|
private:
|
||||||
|
QPointer< MusicScanner > m_musicScanner;
|
||||||
|
MusicScanner::ScanMode m_mode;
|
||||||
|
QStringList m_paths;
|
||||||
|
quint32 m_bs;
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
class DLLEXPORT ScanManager : public QObject
|
class DLLEXPORT ScanManager : public QObject
|
||||||
{
|
{
|
||||||
Q_OBJECT
|
Q_OBJECT
|
||||||
|
|
||||||
public:
|
public:
|
||||||
enum ScanMode { DirScan, FileScan };
|
|
||||||
enum ScanType { None, Full, Normal, File };
|
|
||||||
|
|
||||||
static ScanManager* instance();
|
static ScanManager* instance();
|
||||||
|
|
||||||
explicit ScanManager( QObject* parent = 0 );
|
explicit ScanManager( QObject* parent = 0 );
|
||||||
@@ -71,14 +89,13 @@ private slots:
|
|||||||
private:
|
private:
|
||||||
static ScanManager* s_instance;
|
static ScanManager* s_instance;
|
||||||
|
|
||||||
ScanMode m_currScanMode;
|
MusicScanner::ScanMode m_currScanMode;
|
||||||
QPointer< MusicScanner > m_scanner;
|
MusicScannerThreadController* m_musicScannerThreadController;
|
||||||
QThread* m_musicScannerThreadController;
|
|
||||||
QSet< QString > m_currScannerPaths;
|
QSet< QString > m_currScannerPaths;
|
||||||
QStringList m_cachedScannerDirs;
|
QStringList m_cachedScannerDirs;
|
||||||
|
|
||||||
QTimer* m_scanTimer;
|
QTimer* m_scanTimer;
|
||||||
ScanType m_queuedScanType;
|
MusicScanner::ScanType m_queuedScanType;
|
||||||
|
|
||||||
bool m_updateGUI;
|
bool m_updateGUI;
|
||||||
};
|
};
|
||||||
|
Reference in New Issue
Block a user