1 ///////////////////////////////////////////////////////////////////////////////
2 // Simple x264 Launcher
3 // Copyright (C) 2004-2014 LoRd_MuldeR <MuldeR2@GMX.de>
5 // This program is free software; you can redistribute it and/or modify
6 // it under the terms of the GNU General Public License as published by
7 // the Free Software Foundation; either version 2 of the License, or
8 // (at your option) any later version.
10 // This program is distributed in the hope that it will be useful,
11 // but WITHOUT ANY WARRANTY; without even the implied warranty of
12 // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
13 // GNU General Public License for more details.
15 // You should have received a copy of the GNU General Public License along
16 // with this program; if not, write to the Free Software Foundation, Inc.,
17 // 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
19 // http://www.gnu.org/licenses/gpl-2.0.txt
20 ///////////////////////////////////////////////////////////////////////////////
22 #include "encoder_abstract.h"
25 #include "model_options.h"
26 #include "model_preferences.h"
27 #include "model_sysinfo.h"
28 #include "model_status.h"
29 #include "source_abstract.h"
41 // ------------------------------------------------------------
43 // ------------------------------------------------------------
45 #define APPEND_AND_CLEAR(LIST, STR) do \
47 if(!((STR).isEmpty())) \
55 // ------------------------------------------------------------
56 // Constructor & Destructor
57 // ------------------------------------------------------------
59 AbstractEncoder::AbstractEncoder(JobObject *jobObject, const OptionsModel *options, const SysinfoModel *const sysinfo, const PreferencesModel *const preferences, JobStatus &jobStatus, volatile bool *abort, volatile bool *pause, QSemaphore *semaphorePause, const QString &sourceFile, const QString &outputFile)
61 AbstractTool(jobObject, options, sysinfo, preferences, jobStatus, abort, pause, semaphorePause),
62 m_sourceFile(sourceFile),
63 m_outputFile(outputFile),
64 m_indexFile(QString("%1/~%2.ffindex").arg(QDir::tempPath(), stringToHash(m_sourceFile)))
66 /*Nothing to do here*/
69 AbstractEncoder::~AbstractEncoder(void)
71 /*Nothing to do here*/
74 // ------------------------------------------------------------
76 // ------------------------------------------------------------
78 bool AbstractEncoder::runEncodingPass(AbstractSource* pipedSource, const QString outputFile, const unsigned int &frames, const int &pass, const QString &passLogFile)
80 QProcess processEncode, processInput;
84 pipedSource->createProcess(processEncode, processInput);
87 QStringList cmdLine_Encode;
88 buildCommandLine(cmdLine_Encode, (pipedSource != NULL), frames, m_indexFile, pass, passLogFile);
90 log("Creating encoder process:");
91 if(!startProcess(processEncode, ENC_BINARY(m_sysinfo, m_options), cmdLine_Encode))
96 QList<QRegExp*> patterns;
97 runEncodingPass_init(patterns);
99 bool bTimeout = false;
100 bool bAborted = false;
102 unsigned int last_progress = UINT_MAX;
103 unsigned int last_indexing = UINT_MAX;
104 qint64 size_estimate = 0I64;
106 //Main processing loop
107 while(processEncode.state() != QProcess::NotRunning)
109 unsigned int waitCounter = 0;
111 //Wait until new output is available
116 processEncode.kill();
121 if((*m_pause) && (processEncode.state() == QProcess::Running))
123 JobStatus previousStatus = m_jobStatus;
124 setStatus(JobStatus_Paused);
125 log(tr("Job paused by user at %1, %2.").arg(QDate::currentDate().toString(Qt::ISODate), QTime::currentTime().toString( Qt::ISODate)));
126 bool ok[2] = {false, false};
127 QProcess *proc[2] = { &processEncode, &processInput };
128 ok[0] = x264_suspendProcess(proc[0], true);
129 ok[1] = x264_suspendProcess(proc[1], true);
130 while(*m_pause) m_semaphorePause->tryAcquire(1, 5000);
131 while(m_semaphorePause->tryAcquire(1, 0));
132 ok[0] = x264_suspendProcess(proc[0], false);
133 ok[1] = x264_suspendProcess(proc[1], false);
134 if(!(*m_abort)) setStatus(previousStatus);
135 log(tr("Job resumed by user at %1, %2.").arg(QDate::currentDate().toString(Qt::ISODate), QTime::currentTime().toString( Qt::ISODate)));
139 if(!processEncode.waitForReadyRead(m_processTimeoutInterval))
141 if(processEncode.state() == QProcess::Running)
143 if(++waitCounter > m_processTimeoutMaxCounter)
145 if(m_preferences->getAbortOnTimeout())
147 processEncode.kill();
148 qWarning("encoder process timed out <-- killing!");
149 log("\nPROCESS TIMEOUT !!!");
154 else if(waitCounter == m_processTimeoutWarning)
156 unsigned int timeOut = (waitCounter * m_processTimeoutInterval) / 1000U;
157 log(tr("Warning: encoder did not respond for %1 seconds, potential deadlock...").arg(QString::number(timeOut)));
162 if((*m_abort) || ((*m_pause) && (processEncode.state() == QProcess::Running)))
169 //Exit main processing loop now?
170 if(bAborted || bTimeout)
176 PROCESS_PENDING_LINES(processEncode, runEncodingPass_parseLine, patterns, pass);
179 if(!(bTimeout || bAborted))
181 PROCESS_PENDING_LINES(processEncode, runEncodingPass_parseLine, patterns, pass);
184 processEncode.waitForFinished(5000);
185 if(processEncode.state() != QProcess::NotRunning)
187 qWarning("Encoder process still running, going to kill it!");
188 processEncode.kill();
189 processEncode.waitForFinished(-1);
194 processInput.waitForFinished(5000);
195 if(processInput.state() != QProcess::NotRunning)
197 qWarning("Input process still running, going to kill it!");
199 processInput.waitForFinished(-1);
201 if(!(bTimeout || bAborted))
203 pipedSource->flushProcess(processInput);
207 while(!patterns.isEmpty())
209 QRegExp *pattern = patterns.takeFirst();
210 X264_DELETE(pattern);
213 if(bTimeout || bAborted || processEncode.exitCode() != EXIT_SUCCESS)
215 if(!(bTimeout || bAborted))
217 const int exitCode = processEncode.exitCode();
218 if((exitCode < 0) || (exitCode >= 32))
220 log(tr("\nFATAL ERROR: The encoder process has *crashed* -> your encode probably is *incomplete* !!!"));
221 log(tr("Note that this indicates a bug in the current encoder, *not* in Simple x264/x265 Launcher."));
223 log(tr("\nPROCESS EXITED WITH ERROR CODE: %1").arg(QString::number(exitCode)));
225 processEncode.close();
226 processInput.close();
230 QThread::yieldCurrentThread();
232 QFileInfo completedFileInfo(m_outputFile);
233 const qint64 finalSize = (completedFileInfo.exists() && completedFileInfo.isFile()) ? completedFileInfo.size() : 0;
234 QLocale locale(QLocale::English);
235 log(tr("Final file size is %1 bytes.").arg(sizeToString(finalSize)));
240 setStatus(JobStatus_Running_Pass1);
241 setDetails(tr("First pass completed. Preparing for second pass..."));
244 setStatus(JobStatus_Running_Pass2);
245 setDetails(tr("Second pass completed successfully. Final size is %1.").arg(sizeToString(finalSize)));
248 setStatus(JobStatus_Running);
249 setDetails(tr("Encode completed successfully. Final size is %1.").arg(sizeToString(finalSize)));
254 processEncode.close();
255 processInput.close();
259 // ------------------------------------------------------------
261 // ------------------------------------------------------------
263 QStringList AbstractEncoder::splitParams(const QString ¶ms, const QString &sourceFile, const QString &outputFile)
266 bool ignoreWhitespaces = false;
269 for(int i = 0; i < params.length(); i++)
271 const QChar c = params.at(i);
273 if(c == QChar::fromLatin1('"'))
275 ignoreWhitespaces = (!ignoreWhitespaces);
278 else if((!ignoreWhitespaces) && (c == QChar::fromLatin1(' ')))
280 APPEND_AND_CLEAR(list, temp);
287 APPEND_AND_CLEAR(list, temp);
289 list.replaceInStrings("$(INPUT)", QDir::toNativeSeparators(sourceFile), Qt::CaseInsensitive);
290 list.replaceInStrings("$(OUTPUT)", QDir::toNativeSeparators(outputFile), Qt::CaseInsensitive);
295 qint64 AbstractEncoder::estimateSize(const QString &fileName, const int &progress)
297 QFileInfo fileInfo(fileName);
298 if((progress >= 3) && fileInfo.exists() && fileInfo.isFile())
300 qint64 currentSize = QFileInfo(fileName).size();
301 qint64 estimatedSize = (currentSize * 100I64) / static_cast<qint64>(progress);
302 return estimatedSize;
307 QString AbstractEncoder::sizeToString(qint64 size)
309 static char *prefix[5] = {"Byte", "KB", "MB", "GB", "TB"};
313 qint64 estimatedSize = size;
314 qint64 remainderSize = 0I64;
317 while((estimatedSize > 1024I64) && (prefixIdx < 4))
319 remainderSize = estimatedSize % 1024I64;
320 estimatedSize = estimatedSize / 1024I64;
324 double value = static_cast<double>(estimatedSize) + (static_cast<double>(remainderSize) / 1024.0);
325 return QString().sprintf((value < 10.0) ? "%.2f %s" : "%.1f %s", value, prefix[prefixIdx]);
331 // ------------------------------------------------------------
333 // ------------------------------------------------------------
335 const AbstractEncoderInfo& AbstractEncoder::getEncoderInfo(void)
337 THROW("[getEncoderInfo] This function must be overwritten in sub-classes!");