247 lines
7.7 KiB
C++
247 lines
7.7 KiB
C++
///////////////////////////////////////////////////////////////////////////////
|
|
// Simple x264 Launcher
|
|
// 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.
|
|
//
|
|
// 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
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
|
|
#pragma once
|
|
|
|
#include "source_vapoursynth.h"
|
|
|
|
#include "global.h"
|
|
|
|
#include <QDir>
|
|
#include <QProcess>
|
|
#include <QPair>
|
|
|
|
static const unsigned int VER_X264_VSPIPE_API = 3;
|
|
static const unsigned int VER_X264_VSPIPE_VER = 24;
|
|
|
|
// ------------------------------------------------------------
|
|
// Encoder Info
|
|
// ------------------------------------------------------------
|
|
|
|
class VapoursyntSourceInfo : public AbstractSourceInfo
|
|
{
|
|
public:
|
|
virtual QString getBinaryPath(const SysinfoModel *const sysinfo, const bool& x64) const
|
|
{
|
|
return QString("%1/core%2/vspipe.exe").arg(sysinfo->getVPSPath(), (x64 ? "64" : "32"));
|
|
}
|
|
};
|
|
|
|
static const VapoursyntSourceInfo s_vapoursynthEncoderInfo;
|
|
|
|
const AbstractSourceInfo &VapoursynthSource::getSourceInfo(void)
|
|
{
|
|
return s_vapoursynthEncoderInfo;
|
|
}
|
|
|
|
// ------------------------------------------------------------
|
|
// Constructor & Destructor
|
|
// ------------------------------------------------------------
|
|
|
|
VapoursynthSource::VapoursynthSource(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)
|
|
:
|
|
AbstractSource(jobObject, options, sysinfo, preferences, jobStatus, abort, pause, semaphorePause, sourceFile)
|
|
{
|
|
/*Nothing to do here*/
|
|
}
|
|
|
|
VapoursynthSource::~VapoursynthSource(void)
|
|
{
|
|
/*Nothing to do here*/
|
|
}
|
|
|
|
QString VapoursynthSource::getName(void) const
|
|
{
|
|
return tr("VapourSynth (vpy)");
|
|
}
|
|
|
|
// ------------------------------------------------------------
|
|
// Check Version
|
|
// ------------------------------------------------------------
|
|
|
|
bool VapoursynthSource::isSourceAvailable()
|
|
{
|
|
if(!(m_sysinfo->hasVapourSynth() && (!m_sysinfo->getVPSPath().isEmpty()) && QFileInfo(getBinaryPath()).isFile()))
|
|
{
|
|
log(tr("\nVPY INPUT REQUIRES VAPOURSYNTH, BUT IT IS *NOT* AVAILABLE !!!"));
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
void VapoursynthSource::checkVersion_init(QList<QRegExp*> &patterns, QStringList &cmdLine)
|
|
{
|
|
cmdLine << "--version";
|
|
patterns << new QRegExp("\\bVapourSynth\\b", Qt::CaseInsensitive);
|
|
patterns << new QRegExp("\\bCore\\s+r(\\d+)\\b", Qt::CaseInsensitive);
|
|
patterns << new QRegExp("\\bAPI\\s+r(\\d+)\\b", Qt::CaseInsensitive);
|
|
}
|
|
|
|
void VapoursynthSource::checkVersion_parseLine(const QString &line, const QList<QRegExp*> &patterns, unsigned int &core, unsigned int &build, bool &modified)
|
|
{
|
|
int offset = -1;
|
|
|
|
if((offset = patterns[1]->lastIndexIn(line)) >= 0)
|
|
{
|
|
bool ok = false;
|
|
unsigned int temp = patterns[1]->cap(1).toUInt(&ok);
|
|
if(ok) build = temp;
|
|
}
|
|
else if((offset = patterns[2]->lastIndexIn(line)) >= 0)
|
|
{
|
|
bool ok = false;
|
|
unsigned int temp = patterns[2]->cap(1).toUInt(&ok);
|
|
if(ok) core = temp;
|
|
}
|
|
|
|
if(!line.isEmpty())
|
|
{
|
|
log(line);
|
|
}
|
|
}
|
|
|
|
QString VapoursynthSource::printVersion(const unsigned int &revision, const bool &modified)
|
|
{
|
|
unsigned int core, build;
|
|
splitRevision(revision, core, build);
|
|
|
|
return tr("\nVapourSynth version: r%1 (API r%2)").arg(QString::number(build), QString::number(core));
|
|
}
|
|
|
|
bool VapoursynthSource::isVersionSupported(const unsigned int &revision, const bool &modified)
|
|
{
|
|
unsigned int core, build;
|
|
splitRevision(revision, core, build);
|
|
|
|
if((build < VER_X264_VSPIPE_VER) || (core < VER_X264_VSPIPE_API))
|
|
{
|
|
|
|
if(core < VER_X264_VSPIPE_API)
|
|
{
|
|
log(tr("\nERROR: Your version of VapourSynth is unsupported! (requires API r%1 or newer)").arg(QString::number(VER_X264_VSPIPE_API)));
|
|
}
|
|
if(build < VER_X264_VSPIPE_VER)
|
|
{
|
|
log(tr("\nERROR: Your version of VapourSynth is unsupported! (requires version r%1 or newer)").arg(QString::number(VER_X264_VSPIPE_VER)));
|
|
}
|
|
log(tr("You can find the latest VapourSynth version at: http://www.vapoursynth.com/"));
|
|
return false;
|
|
}
|
|
|
|
if(core != VER_X264_VSPIPE_API)
|
|
{
|
|
log(tr("\nWARNING: Running with an unknown VapourSynth API version, problem may appear! (this application works best with API r%1)").arg(QString::number(VER_X264_VSPIPE_API)));
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
// ------------------------------------------------------------
|
|
// Check Source Properties
|
|
// ------------------------------------------------------------
|
|
|
|
void VapoursynthSource::checkSourceProperties_init(QList<QRegExp*> &patterns, QStringList &cmdLine)
|
|
{
|
|
cmdLine << "--info";
|
|
cmdLine << QDir::toNativeSeparators(x264_path2ansi(m_sourceFile, true));
|
|
cmdLine << "-";
|
|
|
|
patterns << new QRegExp("\\bFrames:\\s+(\\d+)\\b");
|
|
patterns << new QRegExp("\\bWidth:\\s+(\\d+)\\b");
|
|
patterns << new QRegExp("\\bHeight:\\s+(\\d+)\\b");
|
|
patterns << new QRegExp("\\bFPS:\\s+(\\d+)\\b");
|
|
patterns << new QRegExp("\\bFPS:\\s+(\\d+)/(\\d+)\\b");
|
|
}
|
|
|
|
void VapoursynthSource::checkSourceProperties_parseLine(const QString &line, const QList<QRegExp*> &patterns, ClipInfo &clipInfo)
|
|
{
|
|
int offset = -1;
|
|
|
|
if((offset = patterns[0]->lastIndexIn(line)) >= 0)
|
|
{
|
|
bool ok = false;
|
|
unsigned int temp = patterns[0]->cap(1).toUInt(&ok);
|
|
if(ok) clipInfo.setFrameCount(temp);
|
|
}
|
|
if((offset = patterns[1]->lastIndexIn(line)) >= 0)
|
|
{
|
|
bool ok = false;
|
|
unsigned int temp =patterns[1]->cap(1).toUInt(&ok);
|
|
if(ok) clipInfo.setFrameSize(temp, clipInfo.getFrameSize().second);
|
|
}
|
|
if((offset = patterns[2]->lastIndexIn(line)) >= 0)
|
|
{
|
|
bool ok = false;
|
|
unsigned int temp = patterns[2]->cap(1).toUInt(&ok);
|
|
if(ok) clipInfo.setFrameSize(clipInfo.getFrameSize().first, temp);
|
|
}
|
|
if((offset = patterns[3]->lastIndexIn(line)) >= 0)
|
|
{
|
|
bool ok = false;
|
|
unsigned int temp = patterns[3]->cap(1).toUInt(&ok);
|
|
if(ok) clipInfo.setFrameRate(temp, 0);
|
|
}
|
|
if((offset = patterns[4]->lastIndexIn(line)) >= 0)
|
|
{
|
|
bool ok1 = false, ok2 = false;
|
|
unsigned int temp1 = patterns[4]->cap(1).toUInt(&ok1);
|
|
unsigned int temp2 = patterns[4]->cap(2).toUInt(&ok2);
|
|
if(ok1 && ok2)
|
|
{
|
|
clipInfo.setFrameRate(temp1, temp2);
|
|
}
|
|
}
|
|
|
|
if(!line.isEmpty())
|
|
{
|
|
log(line);
|
|
}
|
|
}
|
|
|
|
// ------------------------------------------------------------
|
|
// Check Source Properties
|
|
// ------------------------------------------------------------
|
|
|
|
void VapoursynthSource::buildCommandLine(QStringList &cmdLine)
|
|
{
|
|
cmdLine << "--y4m";
|
|
cmdLine << QDir::toNativeSeparators(x264_path2ansi(m_sourceFile, true));
|
|
cmdLine << "-";
|
|
}
|
|
|
|
void VapoursynthSource::flushProcess(QProcess &processInput)
|
|
{
|
|
while(processInput.bytesAvailable() > 0)
|
|
{
|
|
log(tr("vpyp [info]: %1").arg(QString::fromUtf8(processInput.readLine()).simplified()));
|
|
}
|
|
|
|
if(processInput.exitCode() != EXIT_SUCCESS)
|
|
{
|
|
const int exitCode = processInput.exitCode();
|
|
log(tr("\nWARNING: Input process exited with error (code: %1), your encode might be *incomplete* !!!").arg(QString::number(exitCode)));
|
|
if((exitCode < 0) || (exitCode >= 32))
|
|
{
|
|
log(tr("\nIMPORTANT: The Vapoursynth process terminated abnormally. This means Vapoursynth or one of your Vapoursynth-Plugin's just crashed."));
|
|
}
|
|
}
|
|
}
|