364 lines
8.7 KiB
C++
364 lines
8.7 KiB
C++
///////////////////////////////////////////////////////////////////////////////
|
|
// LameXP - Audio Encoder Front-End
|
|
// Copyright (C) 2004-2018 LoRd_MuldeR <MuldeR2@GMX.de>
|
|
//
|
|
// This program is free software; you can redistribute it and/or modify
|
|
// it under the terms of the GNU General Public License as published by
|
|
// the Free Software Foundation; either version 2 of the License, or
|
|
// (at your option) any later version, but always including the *additional*
|
|
// restrictions defined in the "License.txt" file.
|
|
//
|
|
// This program 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 General Public License for more details.
|
|
//
|
|
// You should have received a copy of the GNU General Public License along
|
|
// with this program; if not, write to the Free Software Foundation, Inc.,
|
|
// 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
|
|
//
|
|
// http://www.gnu.org/licenses/gpl-2.0.txt
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
|
|
#include "Thread_FileAnalyzer.h"
|
|
|
|
//Internal
|
|
#include "Global.h"
|
|
#include "LockedFile.h"
|
|
#include "Model_AudioFile.h"
|
|
#include "Thread_FileAnalyzer_Task.h"
|
|
#include "PlaylistImporter.h"
|
|
|
|
//MUtils
|
|
#include <MUtils/Global.h>
|
|
#include <MUtils/Exception.h>
|
|
|
|
//Qt
|
|
#include <QDir>
|
|
#include <QFileInfo>
|
|
#include <QProcess>
|
|
#include <QDate>
|
|
#include <QTime>
|
|
#include <QDebug>
|
|
#include <QImage>
|
|
#include <QThreadPool>
|
|
#include <QTime>
|
|
#include <QElapsedTimer>
|
|
#include <QTimer>
|
|
#include <QQueue>
|
|
|
|
//Insert into QStringList *without* duplicates
|
|
static inline void SAFE_APPEND_STRING(QStringList &list, const QString &str)
|
|
{
|
|
if(!list.contains(str, Qt::CaseInsensitive))
|
|
{
|
|
list << str;
|
|
}
|
|
}
|
|
|
|
////////////////////////////////////////////////////////////
|
|
// Constructor
|
|
////////////////////////////////////////////////////////////
|
|
|
|
FileAnalyzer::FileAnalyzer(const QStringList &inputFiles)
|
|
:
|
|
m_tasksCounterNext(0),
|
|
m_tasksCounterDone(0),
|
|
m_inputFiles(inputFiles)
|
|
{
|
|
m_filesAccepted = 0;
|
|
m_filesRejected = 0;
|
|
m_filesDenied = 0;
|
|
m_filesDummyCDDA = 0;
|
|
m_filesCueSheet = 0;
|
|
|
|
moveToThread(this); /*makes sure queued slots are executed in the proper thread context*/
|
|
m_timer.reset(new QElapsedTimer());
|
|
}
|
|
|
|
FileAnalyzer::~FileAnalyzer(void)
|
|
{
|
|
if(!m_pool.isNull())
|
|
{
|
|
if(!m_pool->waitForDone(2500))
|
|
{
|
|
qWarning("There are still running tasks in the thread pool!");
|
|
}
|
|
}
|
|
}
|
|
|
|
////////////////////////////////////////////////////////////
|
|
// Thread Main
|
|
////////////////////////////////////////////////////////////
|
|
|
|
void FileAnalyzer::run()
|
|
{
|
|
m_bSuccess.fetchAndStoreOrdered(0);
|
|
|
|
m_tasksCounterNext = 0;
|
|
m_tasksCounterDone = 0;
|
|
m_completedCounter = 0;
|
|
|
|
m_completedFiles.clear();
|
|
m_completedTaskIds.clear();
|
|
m_runningTaskIds.clear();
|
|
|
|
m_filesAccepted = 0;
|
|
m_filesRejected = 0;
|
|
m_filesDenied = 0;
|
|
m_filesDummyCDDA = 0;
|
|
m_filesCueSheet = 0;
|
|
|
|
m_timer->invalidate();
|
|
|
|
//Sort files
|
|
MUtils::natural_string_sort(m_inputFiles, true);
|
|
|
|
//Handle playlist files first!
|
|
handlePlaylistFiles();
|
|
|
|
const unsigned int nFiles = m_inputFiles.count();
|
|
if(nFiles < 1)
|
|
{
|
|
qWarning("File list is empty, nothing to do!");
|
|
return;
|
|
}
|
|
|
|
//Update progress
|
|
emit progressMaxChanged(nFiles);
|
|
emit progressValChanged(0);
|
|
|
|
//Create the thread pool
|
|
if (m_pool.isNull())
|
|
{
|
|
m_pool.reset(new QThreadPool());
|
|
}
|
|
|
|
//Update thread count
|
|
const int idealThreadCount = QThread::idealThreadCount();
|
|
if(idealThreadCount > 0)
|
|
{
|
|
m_pool->setMaxThreadCount(qBound(2, ((idealThreadCount * 3) / 2), 12));
|
|
}
|
|
|
|
//Start first N threads
|
|
QTimer::singleShot(0, this, SLOT(initializeTasks()));
|
|
|
|
//Start event processing
|
|
this->exec();
|
|
|
|
//Wait for pending tasks to complete
|
|
m_pool->waitForDone();
|
|
|
|
//Was opertaion aborted?
|
|
if(MUTILS_BOOLIFY(m_bAborted))
|
|
{
|
|
qWarning("Operation cancelled by user!");
|
|
return;
|
|
}
|
|
|
|
//Update progress
|
|
emit progressValChanged(nFiles);
|
|
|
|
//Emit pending files (this should not be required though!)
|
|
if(!m_completedFiles.isEmpty())
|
|
{
|
|
qWarning("FileAnalyzer: Pending file information found after last thread terminated!");
|
|
QList<unsigned int> keys = m_completedFiles.keys(); qSort(keys);
|
|
while(!keys.isEmpty())
|
|
{
|
|
emit fileAnalyzed(m_completedFiles.take(keys.takeFirst()));
|
|
}
|
|
}
|
|
|
|
qDebug("All files added.\n");
|
|
m_bSuccess.fetchAndStoreOrdered(1);
|
|
QThread::msleep(333);
|
|
}
|
|
|
|
////////////////////////////////////////////////////////////
|
|
// Privtae Functions
|
|
////////////////////////////////////////////////////////////
|
|
|
|
bool FileAnalyzer::analyzeNextFile(void)
|
|
{
|
|
if(!(m_inputFiles.isEmpty() || MUTILS_BOOLIFY(m_bAborted)))
|
|
{
|
|
const unsigned int taskId = m_tasksCounterNext++;
|
|
const QString currentFile = QDir::fromNativeSeparators(m_inputFiles.takeFirst());
|
|
|
|
if((!m_timer->isValid()) || (m_timer->elapsed() >= 333))
|
|
{
|
|
emit fileSelected(QFileInfo(currentFile).fileName());
|
|
m_timer->restart();
|
|
}
|
|
|
|
AnalyzeTask *task = new AnalyzeTask(taskId, currentFile, m_bAborted);
|
|
connect(task, SIGNAL(fileAnalyzed(const unsigned int, const int, AudioFileModel)), this, SLOT(taskFileAnalyzed(unsigned int, const int, AudioFileModel)), Qt::QueuedConnection);
|
|
connect(task, SIGNAL(taskCompleted(const unsigned int)), this, SLOT(taskThreadFinish(const unsigned int)), Qt::QueuedConnection);
|
|
m_runningTaskIds.insert(taskId); m_pool->start(task);
|
|
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
void FileAnalyzer::handlePlaylistFiles(void)
|
|
{
|
|
QQueue<QVariant> queue;
|
|
QStringList importedFromPlaylist;
|
|
|
|
//Import playlist files into "hierarchical" list
|
|
while(!m_inputFiles.isEmpty())
|
|
{
|
|
const QString currentFile = m_inputFiles.takeFirst();
|
|
QStringList importedFiles;
|
|
if(PlaylistImporter::importPlaylist(importedFiles, currentFile))
|
|
{
|
|
queue.enqueue(importedFiles);
|
|
importedFromPlaylist << importedFiles;
|
|
}
|
|
else
|
|
{
|
|
queue.enqueue(currentFile);
|
|
}
|
|
}
|
|
|
|
//Reduce temporary list
|
|
importedFromPlaylist.removeDuplicates();
|
|
|
|
//Now build the complete "flat" file list (files imported from playlist take precedence!)
|
|
while(!queue.isEmpty())
|
|
{
|
|
const QVariant current = queue.dequeue();
|
|
if(current.type() == QVariant::String)
|
|
{
|
|
const QString temp = current.toString();
|
|
if(!importedFromPlaylist.contains(temp, Qt::CaseInsensitive))
|
|
{
|
|
SAFE_APPEND_STRING(m_inputFiles, temp);
|
|
}
|
|
}
|
|
else if(current.type() == QVariant::StringList)
|
|
{
|
|
const QStringList temp = current.toStringList();
|
|
for(QStringList::ConstIterator iter = temp.constBegin(); iter != temp.constEnd(); iter++)
|
|
{
|
|
SAFE_APPEND_STRING(m_inputFiles, (*iter));
|
|
}
|
|
}
|
|
else
|
|
{
|
|
qWarning("Encountered an unexpected variant type!");
|
|
}
|
|
}
|
|
}
|
|
|
|
////////////////////////////////////////////////////////////
|
|
// Slot Functions
|
|
////////////////////////////////////////////////////////////
|
|
|
|
void FileAnalyzer::initializeTasks(void)
|
|
{
|
|
for(int i = 0; i < m_pool->maxThreadCount(); i++)
|
|
{
|
|
if(!analyzeNextFile()) break;
|
|
}
|
|
}
|
|
|
|
void FileAnalyzer::taskFileAnalyzed(const unsigned int taskId, const int fileType, const AudioFileModel &file)
|
|
{
|
|
m_completedTaskIds.insert(taskId);
|
|
|
|
switch(fileType)
|
|
{
|
|
case AnalyzeTask::fileTypeNormal:
|
|
m_filesAccepted++;
|
|
if(m_tasksCounterDone == taskId)
|
|
{
|
|
emit fileAnalyzed(file);
|
|
m_tasksCounterDone++;
|
|
}
|
|
else
|
|
{
|
|
m_completedFiles.insert(taskId, file);
|
|
}
|
|
break;
|
|
case AnalyzeTask::fileTypeCDDA:
|
|
m_filesDummyCDDA++;
|
|
break;
|
|
case AnalyzeTask::fileTypeDenied:
|
|
m_filesDenied++;
|
|
break;
|
|
case AnalyzeTask::fileTypeCueSheet:
|
|
m_filesCueSheet++;
|
|
break;
|
|
case AnalyzeTask::fileTypeUnknown:
|
|
m_filesRejected++;
|
|
break;
|
|
default:
|
|
MUTILS_THROW("Unknown file type identifier!");
|
|
}
|
|
|
|
//Emit all pending files
|
|
while(m_completedTaskIds.contains(m_tasksCounterDone))
|
|
{
|
|
if(m_completedFiles.contains(m_tasksCounterDone))
|
|
{
|
|
emit fileAnalyzed(m_completedFiles.take(m_tasksCounterDone));
|
|
}
|
|
m_completedTaskIds.remove(m_tasksCounterDone);
|
|
m_tasksCounterDone++;
|
|
}
|
|
}
|
|
|
|
void FileAnalyzer::taskThreadFinish(const unsigned int taskId)
|
|
{
|
|
m_runningTaskIds.remove(taskId);
|
|
emit progressValChanged(++m_completedCounter);
|
|
|
|
if(!analyzeNextFile())
|
|
{
|
|
if(m_runningTaskIds.empty())
|
|
{
|
|
QTimer::singleShot(0, this, SLOT(quit())); //Stop event processing, if all threads have completed!
|
|
}
|
|
}
|
|
}
|
|
|
|
////////////////////////////////////////////////////////////
|
|
// Public Functions
|
|
////////////////////////////////////////////////////////////
|
|
|
|
unsigned int FileAnalyzer::filesAccepted(void)
|
|
{
|
|
return m_filesAccepted;
|
|
}
|
|
|
|
unsigned int FileAnalyzer::filesRejected(void)
|
|
{
|
|
return m_filesRejected;
|
|
}
|
|
|
|
unsigned int FileAnalyzer::filesDenied(void)
|
|
{
|
|
return m_filesDenied;
|
|
}
|
|
|
|
unsigned int FileAnalyzer::filesDummyCDDA(void)
|
|
{
|
|
return m_filesDummyCDDA;
|
|
}
|
|
|
|
unsigned int FileAnalyzer::filesCueSheet(void)
|
|
{
|
|
return m_filesCueSheet;
|
|
}
|
|
|
|
////////////////////////////////////////////////////////////
|
|
// EVENTS
|
|
////////////////////////////////////////////////////////////
|
|
|
|
/*NONE*/
|