12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337 |
- #include "CodecsComponent.h"
- #include <QString>
- #include <Qt>
- #include <QDir>
- #include <QDomAttr>
- #include <QDomDocument>
- #include <QDomNode>
- #include <QCoreApplication>
- #include <QProcess>
- #include <QUuid>
- #include <QUrl>
- #include <QUrlQuery>
- #include <QResource>
- #include <QSaveFile>
- #include <QStandardPaths>
- #include <QSysInfo>
- #include <QCryptographicHash>
- #include <QTemporaryDir>
- #ifdef HAVE_MINIZIP
- #include <minizip/unzip.h>
- #include <minizip/ioapi.h>
- #endif
- #include "system/SystemComponent.h"
- #include "settings/SettingsComponent.h"
- #include "utils/Utils.h"
- #include "shared/Paths.h"
- #include "PlayerComponent.h"
- #include "QsLog.h"
- #define countof(x) (sizeof(x) / sizeof((x)[0]))
- // For QVariant. Mysteriously makes Qt happy.
- Q_DECLARE_METATYPE(CodecDriver);
- #ifdef HAVE_CODEC_MANIFEST
- #include "CodecManifest.h"
- #else
- #define CODEC_VERSION "dummy"
- #define SHLIB_PREFIX ""
- #define SHLIB_EXTENSION "dummy"
- // Codec.name is the name of the codec implementation, Codec.codecName the name of the codec
- struct Codec {const char* name; const char* codecName; const char* profiles; int external;};
- static const Codec Decoders[] = {
- {"dummy", "dummy", nullptr, 1},
- };
- static const Codec Encoders[] = {
- {"dummy", "dummy", nullptr, 1},
- };
- #endif
- #define STRINGIFY_(x) #x
- #define STRINGIFY(x) STRINGIFY_(x)
- #ifdef EAE_VERSION
- #define HAVE_EAE 1
- #else
- #define EAE_VERSION unavailable
- #define HAVE_EAE 0
- #endif
- // We might want to use Codec.quality to decide this one day.
- // But for now, it's better if we can quickly change these.
- static QSet<QString> g_systemVideoDecoderWhitelist = {
- // RPI
- "h264_mmal",
- "mpeg2_mmal",
- "mpeg4_mmal",
- "vc1_mmal",
- // Windows
- "h264_mf",
- "hevc_mf",
- "vc1_mf",
- "wmv1_mf",
- "wmv2_mf",
- "wmv3_mf",
- "mpeg2_mf",
- "mpeg4_mf",
- "msmpeg4v1_mf",
- "msmpeg4v2_mf",
- "msmpeg4v3_mf",
- };
- static QSet<QString> g_systemAudioDecoderWhitelist = {
- // OSX
- "aac_at",
- "ac3_at",
- "eac3_at",
- "mp1_at",
- "mp2_at",
- "mp3_at",
- // Windows
- "ac3_mf",
- "eac3_mf",
- "aac_mf",
- "mp1_mf",
- "mp2_mf",
- "mp3_mf",
- };
- static QSet<QString> g_systemAudioEncoderWhitelist = {
- };
- static QSize g_mediaFoundationH264MaxResolution;
- static QString g_codecVersion;
- static QList<CodecDriver> g_cachedCodecList;
- static QString g_deviceID;
- static QString g_eaeWatchFolder;
- static QProcess* g_eaeProcess;
- ///////////////////////////////////////////////////////////////////////////////////////////////////
- static QString getBuildType()
- {
- #ifdef Q_OS_MAC
- return "darwin-x86_64";
- #elif defined(TARGET_RPI)
- return "openelec-armv7";
- #else
- return SystemComponent::Get().getPlatformTypeString() + "-" +
- SystemComponent::Get().getPlatformArchString();
- #endif
- }
- ///////////////////////////////////////////////////////////////////////////////////////////////////
- static QString getEAEBuildType()
- {
- #if defined(Q_OS_MAC)
- return "darwin-x86_64";
- #elif defined(Q_OS_WIN)
- return sizeof(void *) > 4 ? "windows-x86_64" : "windows-i386";
- #elif defined(TARGET_RPI)
- return "linux-raspi2-arm7";
- #elif defined(Q_OS_LINUX)
- return sizeof(void *) > 4 ? "linux-ubuntu-x86_64" : "linux-ubuntu-i686";
- #else
- return "unknown";
- #endif
- }
- ///////////////////////////////////////////////////////////////////////////////////////////////////
- QString Codecs::plexNameToFF(QString plex)
- {
- if (plex == "dca")
- return "dts";
- return plex;
- }
- ///////////////////////////////////////////////////////////////////////////////////////////////////
- QString Codecs::plexNameFromFF(QString ffname)
- {
- if (ffname == "dts")
- return "dca";
- return ffname;
- }
- ///////////////////////////////////////////////////////////////////////////////////////////////////
- static QString codecsRootPath()
- {
- return Paths::dataDir("Codecs") + QDir::separator();
- }
- ///////////////////////////////////////////////////////////////////////////////////////////////////
- static QString codecsPath()
- {
- return codecsRootPath() + g_codecVersion + "-" + getBuildType() + QDir::separator();
- }
- ///////////////////////////////////////////////////////////////////////////////////////////////////
- static QString eaePrefixPath()
- {
- return codecsRootPath() + "EasyAudioEncoder-" + STRINGIFY(EAE_VERSION) + "-" + getBuildType();
- }
- static QString eaeBinaryPath()
- {
- QString exeSuffix = "";
- #ifdef Q_OS_WIN
- exeSuffix = ".exe";
- #endif
- return eaePrefixPath() + "/EasyAudioEncoder/EasyAudioEncoder" + exeSuffix;
- }
- ///////////////////////////////////////////////////////////////////////////////////////////////////
- static int indexOfCodecInList(const QList<CodecDriver>& list, const CodecDriver& codec)
- {
- for (int n = 0; n < list.size(); n++)
- {
- if (Codecs::sameCodec(list[n], codec))
- return n;
- }
- return -1;
- }
- ///////////////////////////////////////////////////////////////////////////////////////////////////
- void Codecs::updateCachedCodecList()
- {
- g_cachedCodecList.clear();
- for (CodecType type : {CodecType::Decoder, CodecType::Encoder})
- {
- const Codec* list = (type == CodecType::Decoder) ? Decoders : Encoders;
- size_t count = (type == CodecType::Decoder) ? countof(Decoders) : countof(Encoders);
- for (size_t i = 0; i < count; i++)
- {
- CodecDriver codec = {};
- codec.type = type;
- codec.format = Codecs::plexNameToFF(list[i].codecName);
- codec.driver = list[i].name;
- codec.external = list[i].external;
- if (!codec.isSystemCodec())
- g_cachedCodecList.append(codec);
- }
- }
- // Set present flag for the installed codecs. Also, there could be codecs not
- // on the CodecManifest.h list (system codecs, or when compiled without
- // codec loading).
- QList<CodecDriver> installed = PlayerComponent::Get().installedCodecDrivers();
- // Surely O(n^2) won't be causing trouble, right?
- for (const CodecDriver& installedCodec : installed)
- {
- int index = indexOfCodecInList(g_cachedCodecList, installedCodec);
- if (index >= 0)
- g_cachedCodecList[index].present = true;
- else
- g_cachedCodecList.append(installedCodec);
- }
- }
- ///////////////////////////////////////////////////////////////////////////////////////////////////
- const QList<CodecDriver>& Codecs::getCachedCodecList()
- {
- return g_cachedCodecList;
- }
- ///////////////////////////////////////////////////////////////////////////////////////////////////
- QList<CodecDriver> Codecs::findCodecsByFormat(const QList<CodecDriver>& list, CodecType type, const QString& format)
- {
- QList<CodecDriver> result;
- for (const CodecDriver& codec : list)
- {
- if (codec.type == type && codec.format == format)
- result.append(codec);
- }
- return result;
- }
- ///////////////////////////////////////////////////////////////////////////////////////////////////
- QString CodecDriver::getMangledName() const
- {
- return driver + (type == CodecType::Decoder ? "_decoder" : "_encoder");
- }
- ///////////////////////////////////////////////////////////////////////////////////////////////////
- QString CodecDriver::getFileName() const
- {
- return SHLIB_PREFIX + getMangledName() + "." + SHLIB_EXTENSION;
- }
- ///////////////////////////////////////////////////////////////////////////////////////////////////
- QString CodecDriver::getPath() const
- {
- return QDir(codecsPath()).absoluteFilePath(getFileName());
- }
- ///////////////////////////////////////////////////////////////////////////////////////////////////
- bool CodecDriver::isSystemCodec() const
- {
- // MS Windows
- if (driver.endsWith("_mf"))
- return true;
- // OSX
- if (driver.endsWith("_at"))
- return true;
- // Linux on RPI
- if (driver.endsWith("_mmal"))
- return true;
- // Not really a system codec, but treated as such for convenience
- if (driver.endsWith("_eae"))
- return true;
- return false;
- }
- ///////////////////////////////////////////////////////////////////////////////////////////////////
- QString CodecDriver::getSystemCodecType() const
- {
- if (!isSystemCodec())
- return "";
- int splitAt = driver.indexOf("_");
- if (splitAt < 0)
- return "";
- return driver.mid(splitAt + 1);
- }
- ///////////////////////////////////////////////////////////////////////////////////////////////////
- bool CodecDriver::isWhitelistedSystemAudioCodec() const
- {
- if (type == CodecType::Decoder)
- return g_systemAudioDecoderWhitelist.contains(driver);
- else
- return g_systemAudioEncoderWhitelist.contains(driver);
- }
- ///////////////////////////////////////////////////////////////////////////////////////////////////
- bool CodecDriver::isWhitelistedSystemVideoCodec() const
- {
- if (type == CodecType::Decoder)
- return g_systemVideoDecoderWhitelist.contains(driver);
- return false;
- }
- ///////////////////////////////////////////////////////////////////////////////////////////////////
- static bool useSystemAudioDecoders()
- {
- return true;
- }
- ///////////////////////////////////////////////////////////////////////////////////////////////////
- static bool useSystemVideoDecoders()
- {
- return SettingsComponent::Get().value(SETTINGS_SECTION_MAIN, "useSystemVideoCodecs").toBool();
- }
- /////////////////////////////////////////////////////////////////////////////////////////
- // Load the device ID, do some minimal verification, return "" on error.
- static QString loadDeviceID(QString filename)
- {
- QFile path(filename);
- if (!path.open(QFile::ReadOnly))
- return "";
- auto res = QString::fromLatin1(path.readAll());
- if (res.size() < 32 || res.size() > 512)
- res = ""; // mark as invalid
- return res;
- }
- /////////////////////////////////////////////////////////////////////////////////////////
- static QString findOldDeviceID()
- {
- // First we try to reuse the ID from other Plex products (i.e. PMS) or older paths.
- QStringList candidates = {
- #ifdef Q_OS_MAC
- QDir::home().path() + "/Library/Application Support/Plex/Codecs/.device-id",
- QDir::home().path() + "/Library/Application Support/Plex Media Server/Codecs/.device-id",
- #endif
- QStandardPaths::writableLocation(QStandardPaths::GenericDataLocation) + "/Plex/Codecs/.device-id",
- QStandardPaths::writableLocation(QStandardPaths::GenericDataLocation) + "/Plex/codecs/.device-id",
- QStandardPaths::writableLocation(QStandardPaths::GenericDataLocation) + "/Plex Media Server/Codecs/.device-id",
- Paths::dataDir() + "/codecs/.device-id",
- };
- for (auto candidate : candidates)
- {
- auto id = loadDeviceID(candidate);
- if (!id.isEmpty())
- return id;
- }
- return "";
- }
- ///////////////////////////////////////////////////////////////////////////////////////////////////
- // Returns "" on error.
- static QString loadDeviceID()
- {
- QString deviceIDFilename = QDir(codecsRootPath()).absoluteFilePath(".device-id");
- QString id = loadDeviceID(deviceIDFilename);
- if (id.isEmpty())
- {
- id = findOldDeviceID();
- if (id.isEmpty())
- {
- id = QUuid::createUuid().toString();
- // The UUID should be e.g. "8f6ad954-0cb9-4dbb-a5e5-e0b085f07cf8"
- if (id.startsWith("{"))
- id = id.mid(1);
- if (id.endsWith("}"))
- id = id.mid(0, id.size() - 1);
- }
- Utils::safelyWriteFile(deviceIDFilename, id.toLatin1());
- // We load it again to make sure writing it succeeded. If it doesn't, we'll
- // error out at a later point.
- id = loadDeviceID(deviceIDFilename);
- }
- return id;
- }
- ///////////////////////////////////////////////////////////////////////////////////////////////////
- static QString getFFmpegVersion()
- {
- auto mpv = mpv::qt::Handle::FromRawHandle(mpv_create());
- if (!mpv || mpv_initialize(mpv) < 0)
- return "";
- return mpv::qt::get_property(mpv, "ffmpeg-version").toString();
- }
- ///////////////////////////////////////////////////////////////////////////////////////////////////
- static void setEnv(QString var, QString val)
- {
- #ifdef Q_OS_WIN
- SetEnvironmentVariableW(var.toStdWString().c_str(), val.toStdWString().c_str());
- #else
- qputenv(var.toUtf8().data(), val.toUtf8().data());
- #endif
- }
- ///////////////////////////////////////////////////////////////////////////////////////////////////
- void Codecs::preinitCodecs()
- {
- // Extract the CI codecs version we set with --extra-version when compiling FFmpeg.
- QString ffmpegVersion = getFFmpegVersion();
- int sep = ffmpegVersion.indexOf(',');
- if (sep >= 0)
- g_codecVersion = ffmpegVersion.mid(sep + 1);
- else
- g_codecVersion = CODEC_VERSION;
- QString path = codecsPath();
- QDir("").mkpath(path);
- // Follows the convention used by av_get_token().
- QString escapedPath = path.replace("\\", "\\\\").replace(":", "\\:").replace("'", "\\'");
- // This must be run before any threads are started etc. (for safety).
- setEnv("FFMPEG_EXTERNAL_LIBS", escapedPath);
- QTemporaryDir d(QDir::tempPath() + "/pmp-eae-XXXXXX");
- d.setAutoRemove(false);
- g_eaeWatchFolder = d.path();
- setEnv("EAE_ROOT", g_eaeWatchFolder);
- g_deviceID = loadDeviceID();
- }
- #if 0
- ///////////////////////////////////////////////////////////////////////////////////////////////////
- static bool probeDecoder(QString decoder, QString resourceName)
- {
- QResource resource(resourceName);
- QLOG_DEBUG() << "Testing decoding of" << resource.fileName();
- if (!resource.isValid())
- return false;
- auto mpv = mpv::qt::Handle::FromRawHandle(mpv_create());
- if (!mpv || mpv_initialize(mpv) < 0)
- return false;
- // Disable any output.
- mpv::qt::set_property(mpv, "vo", "null");
- // Force the decoder. The ",-" means that if the first entry fails, the next codec in the global
- // codec list will not be tried, and decoding fails.
- mpv::qt::set_property(mpv, "vd", "lavc:" + decoder + ",-");
- // Attempt decoding, and return success.
- auto data = QByteArray::fromRawData((const char *)resource.data(), resource.size());
- if (resource.isCompressed())
- data = qUncompress(data);
- auto hex = data.toHex();
- mpv::qt::command(mpv, QVariantList{"loadfile", "hex://" + QString::fromLatin1(hex)});
- bool result = false;
- while (1) {
- mpv_event *event = mpv_wait_event(mpv, 0);
- if (event->event_id == MPV_EVENT_SHUTDOWN)
- break;
- if (event->event_id == MPV_EVENT_END_FILE)
- {
- mpv_event_end_file *endFile = (mpv_event_end_file *)event->data;
- result = endFile->reason == MPV_END_FILE_REASON_EOF;
- break;
- }
- }
- QLOG_DEBUG() << "Result:" << result;
- return result;
- }
- #endif
- ///////////////////////////////////////////////////////////////////////////////////////////////////
- static void probeCodecs()
- {
- #if 0
- if (useSystemVideoDecoders())
- {
- if (probeDecoder("h264_mf", ":/testmedia/high_4096x2304.h264"))
- g_mediaFoundationH264MaxResolution = QSize(4096, 2304);
- else if (probeDecoder("h264_mf", ":/testmedia/high_4096x2160.h264"))
- g_mediaFoundationH264MaxResolution = QSize(4096, 2160);
- else if (probeDecoder("h264_mf", ":/testmedia/high_4096x1080.h264"))
- g_mediaFoundationH264MaxResolution = QSize(4096, 1080);
- else
- g_systemVideoDecoderWhitelist.remove("h264_mf");
- QLOG_DEBUG() << "h264_mf max. resolution:" << g_mediaFoundationH264MaxResolution;
- }
- #endif
- #ifdef Q_OS_MAC
- // Unsupported, but avoid picking up broken Perian decoders.
- if (QSysInfo::MacintoshVersion <= QSysInfo::MV_10_10)
- g_systemAudioDecoderWhitelist.remove("ac3_at");
- #endif
- }
- ///////////////////////////////////////////////////////////////////////////////////////////////////
- static void updateCodecs()
- {
- QStringList candidates = {
- codecsRootPath(),
- #ifdef Q_OS_MAC
- QDir::home().path() + "/Library/Application Support/Plex/Codecs/",
- QDir::home().path() + "/Library/Application Support/Plex Media Server/Codecs/",
- #endif
- QStandardPaths::writableLocation(QStandardPaths::GenericDataLocation) + "/Plex/Codecs/",
- QStandardPaths::writableLocation(QStandardPaths::GenericDataLocation) + "/Plex/codecs/",
- QStandardPaths::writableLocation(QStandardPaths::GenericDataLocation) + "/Plex Media Server/Codecs/",
- Paths::dataDir() + "/codecs/",
- };
- QSet<QString> codecFiles;
- for (auto dir : candidates)
- {
- QDir qdir(dir);
- if (!qdir.exists())
- continue;
- for (auto entry : qdir.entryList(QDir::Dirs | QDir::NoDotAndDotDot))
- {
- QDir entryDir = qdir;
- if (!entryDir.cd(entry))
- continue;
- for (auto codecdirEntry : entryDir.entryList(QDir::Files))
- codecFiles.insert(codecdirEntry);
- }
- }
- QList<CodecDriver> install;
- for (CodecDriver& codec : g_cachedCodecList)
- {
- if (codecFiles.contains(codec.getFileName()))
- {
- if (codec.external && !codec.present)
- install.append(codec);
- }
- }
- if (!install.empty())
- {
- QLOG_INFO() << "Updating some codecs...";
- auto fetcher = new CodecsFetcher();
- QObject::connect(fetcher, &CodecsFetcher::done, [](CodecsFetcher* sender)
- {
- QLOG_INFO() << "Codec update finished.";
- sender->deleteLater();
- });
- fetcher->installCodecs(install);
- }
- }
- ///////////////////////////////////////////////////////////////////////////////////////////////////
- static void deleteOldCodecs()
- {
- QStringList neededPaths = {
- codecsPath(),
- };
- for (auto entry : QDir(codecsRootPath()).entryList(QDir::Dirs | QDir::NoDotAndDotDot))
- {
- QDir entryPath = codecsRootPath();
- if (!entryPath.cd(entry))
- continue;
- bool needed = false;
- for (auto neededPath : neededPaths)
- {
- if (entryPath.absolutePath() == QDir(neededPath).absolutePath())
- {
- needed = true;
- break;
- }
- }
- if (needed)
- continue;
- // Same version, but different platform -> just keep it.
- if (entry.startsWith(g_codecVersion + "-"))
- continue;
- QLOG_DEBUG() << "Deleting old directory: " << entryPath.absolutePath();
- entryPath.removeRecursively();
- }
- }
- ///////////////////////////////////////////////////////////////////////////////////////////////////
- void Codecs::initCodecs()
- {
- if (g_deviceID.isEmpty())
- throw FatalException("Could not read device-id.");
- if (g_eaeWatchFolder.isEmpty())
- throw FatalException("Could not create EAE working directory.");
- Codecs::updateCachedCodecList();
- updateCodecs();
- deleteOldCodecs();
- probeCodecs();
- }
- ///////////////////////////////////////////////////////////////////////////////////////////////////
- bool CodecsFetcher::codecNeedsDownload(const CodecDriver& codec)
- {
- if (codec.present)
- return false;
- if (!codec.external)
- {
- QLOG_ERROR() << "Codec" << codec.driver << "does not exist and is not downloadable.";
- return false;
- }
- for (int n = 0; n < m_Codecs.size(); n++)
- {
- if (Codecs::sameCodec(codec, m_Codecs[n]))
- return false;
- }
- QFile codecFile(codec.getPath());
- if (codecFile.exists())
- {
- QLOG_ERROR() << "Codec" << codec.driver << "exists on disk as" << codec.getPath()
- << "but is not known as installed - broken codec?";
- if (!codecFile.remove())
- return false;
- QLOG_ERROR() << "Retrying download.";
- }
- return true;
- }
- ///////////////////////////////////////////////////////////////////////////////////////////////////
- void CodecsFetcher::installCodecs(const QList<CodecDriver>& codecs)
- {
- foreach (CodecDriver codec, codecs)
- {
- if (codecNeedsDownload(codec))
- m_Codecs.enqueue(codec);
- if (codec.getSystemCodecType() == "eae")
- {
- m_eaeNeeded = true;
- if (!QFile(eaeBinaryPath()).exists())
- m_fetchEAE = true;
- }
- }
- startNext();
- }
- ///////////////////////////////////////////////////////////////////////////////////////////////////
- static Downloader::HeaderList getPlexHeaders()
- {
- Downloader::HeaderList headers;
- QString auth = SystemComponent::Get().authenticationToken();
- if (auth.size())
- headers.append({"X-Plex-Token", auth});
- headers.append({"X-Plex-Product", "Plex Media Player"});
- headers.append({"X-Plex-Platform", "Konvergo"});
- return headers;
- }
- static QUrl buildCodecQuery(QString version, QString name, QString build)
- {
- QString host = "https://plex.tv";
- QUrl url = QUrl(host + "/api/codecs/" + name);
- QUrlQuery query;
- query.addQueryItem("deviceId", g_deviceID);
- query.addQueryItem("version", version);
- query.addQueryItem("build", build);
- query.addQueryItem("oldestPreviousVersion", SettingsComponent::Get().oldestPreviousVersion());
- url.setQuery(query);
- return url;
- }
- ///////////////////////////////////////////////////////////////////////////////////////////////////
- void CodecsFetcher::startNext()
- {
- if (m_fetchEAE)
- {
- m_fetchEAE = false;
- QUrl url = buildCodecQuery(STRINGIFY(EAE_VERSION), "easyaudioencoder", getEAEBuildType());
- Downloader *downloader = new Downloader(QVariant("eae"), url, getPlexHeaders(), this);
- connect(downloader, &Downloader::done, this, &CodecsFetcher::codecInfoDownloadDone);
- return;
- }
- if (m_Codecs.isEmpty())
- {
- // Do final initializations.
- if (m_eaeNeeded)
- startEAE();
- emit done(this);
- return;
- }
- CodecDriver codec = m_Codecs.dequeue();
- QUrl url = buildCodecQuery(g_codecVersion, codec.getMangledName(), getBuildType());
- Downloader *downloader = new Downloader(QVariant::fromValue(codec), url, getPlexHeaders(), this);
- connect(downloader, &Downloader::done, this, &CodecsFetcher::codecInfoDownloadDone);
- }
- ///////////////////////////////////////////////////////////////////////////////////////////////////
- bool CodecsFetcher::processCodecInfoReply(const QVariant& context, const QByteArray& data)
- {
- QLOG_INFO() << "Got reply:" << QString::fromUtf8(data);
- QDomDocument dom;
- if (!dom.setContent(data))
- {
- QLOG_ERROR() << "XML parsing error.";
- return false;
- }
- QDomNodeList list = dom.elementsByTagName("MediaContainer");
- if (list.count() != 1)
- {
- QLOG_ERROR() << "MediaContainer XML element not found.";
- return false;
- }
- list = dom.elementsByTagName("Codec");
- if (list.count() != 1)
- {
- QLOG_ERROR() << "Codec XML element not found.";
- return false;
- }
- QDomNamedNodeMap attrs = list.at(0).attributes();
- QString url = attrs.namedItem("url").toAttr().value();
- if (!url.size())
- {
- QLOG_ERROR() << "No URL found.";
- return false;
- }
- QString hash = attrs.namedItem("fileSha").toAttr().value();
- m_currentHash = QByteArray::fromHex(hash.toUtf8());
- // it's hardcoded to SHA-1
- if (!m_currentHash.size()) {
- QLOG_ERROR() << "Hash value in unexpected format or missing:" << hash;
- return false;
- }
- Downloader *downloader = new Downloader(context, url, getPlexHeaders(), this);
- connect(downloader, &Downloader::done, this, &CodecsFetcher::codecDownloadDone);
- return true;
- }
- ///////////////////////////////////////////////////////////////////////////////////////////////////
- void CodecsFetcher::codecInfoDownloadDone(QVariant userData, bool success, const QByteArray& data)
- {
- if (!success || !processCodecInfoReply(userData, data))
- {
- QLOG_ERROR() << "Codec download failed.";
- startNext();
- }
- }
- #ifdef HAVE_MINIZIP
- static voidpf unz_open_file(voidpf opaque, const char* filename, int mode)
- {
- #ifdef Q_OS_WIN32
- return _wfopen(QString::fromUtf8(filename).toStdWString().c_str(), L"rb");
- #else
- return fopen(filename, "rb");
- #endif
- }
- static uLong unz_read_file(voidpf opaque, voidpf stream, void* buf, uLong size)
- {
- return fread(buf, 1, size, (FILE *)stream);
- }
- static uLong unz_write_file(voidpf opaque, voidpf stream, const void* buf, uLong size)
- {
- return 0;
- }
- static int unz_close_file(voidpf opaque, voidpf stream)
- {
- return fclose((FILE *)stream);
- }
- static int unz_error_file(voidpf opaque, voidpf stream)
- {
- return ferror((FILE *)stream);
- }
- static long unz_tell_file(voidpf opaque, voidpf stream)
- {
- return ftell((FILE *)stream);
- }
- long unz_seek_file(voidpf opaque, voidpf stream, uLong offset, int origin)
- {
- int whence = -1;
- switch (origin)
- {
- case ZLIB_FILEFUNC_SEEK_CUR:
- whence = SEEK_CUR;
- break;
- case ZLIB_FILEFUNC_SEEK_END:
- whence = SEEK_END;
- break;
- case ZLIB_FILEFUNC_SEEK_SET:
- whence = SEEK_SET;
- break;
- }
- return fseek((FILE *)stream, offset, whence);
- }
- ///////////////////////////////////////////////////////////////////////////////////////////////////
- static bool extractZip(QString zip, QString dest)
- {
- bool success = false;
- zlib_filefunc_def unzfilefuncs = {};
- unzfilefuncs.zopen_file = unz_open_file;
- unzfilefuncs.zread_file = unz_read_file;
- unzfilefuncs.zwrite_file = unz_write_file;
- unzfilefuncs.ztell_file = unz_tell_file;
- unzfilefuncs.zseek_file = unz_seek_file;
- unzfilefuncs.zclose_file = unz_close_file;
- unzfilefuncs.zerror_file = unz_error_file;
- unzFile file = unzOpen2(zip.toUtf8().data(), &unzfilefuncs);
- if (!file)
- {
- QLOG_ERROR() << "could not open .zip file.";
- goto fail;
- }
- unz_global_info info;
- int unzerr;
- if ((unzerr = unzGetGlobalInfo(file, &info)) != UNZ_OK)
- {
- QLOG_ERROR() << "unzGlobalInfo() failed with" << unzerr;
- goto fail;
- }
- if ((unzerr = unzGoToFirstFile(file)) != UNZ_OK)
- {
- QLOG_ERROR() << "unzGoToFirstFile() failed with" << unzerr;
- goto fail;
- }
- for (ZPOS64_T n = 0; n < info.number_entry; n++)
- {
- if (n > 0 && (unzerr = unzGoToNextFile(file)) != UNZ_OK)
- {
- QLOG_ERROR() << "unzGoToNextFile() failed with" << unzerr;
- goto fail;
- }
- char filename[256];
- unz_file_info finfo;
- if ((unzerr = unzGetCurrentFileInfo(file, &finfo, filename, sizeof(filename), 0, 0, 0, 0)) != UNZ_OK)
- {
- QLOG_ERROR() << "unzGetCurrentFileInfo() failed with" << unzerr;
- goto fail;
- }
- if ((unzerr = unzOpenCurrentFile(file)) != UNZ_OK)
- {
- QLOG_ERROR() << "unzOpenCurrentFile() failed with" << unzerr;
- goto fail;
- }
- char *pathpart = strrchr(filename, '/');
- if (pathpart)
- {
- // This part sucks especially: temporarily cut off the string.
- *pathpart = '\0';
- QDir dir(dest + "/" + filename);
- if (!dir.mkpath("."))
- {
- QLOG_ERROR() << "could not create zip sub directory";
- goto fail;
- }
- *pathpart = '/';
- }
- // Directory (probably)
- if (QString(filename).endsWith("/"))
- continue;
- QString writepath = dest + "/" + filename;
- QSaveFile out(writepath);
- if (!out.open(QIODevice::WriteOnly))
- {
- QLOG_ERROR() << "could not open output file" << filename;
- goto fail;
- }
- while (true)
- {
- char buf[4096];
- int read = unzReadCurrentFile(file, buf, sizeof(buf));
- if (read == 0)
- break;
- if (read < 0)
- {
- QLOG_ERROR() << "error decompressing zip entry" << filename;
- goto fail;
- }
- if (out.write(buf, read) != read)
- {
- QLOG_ERROR() << "error writing output file" << filename;
- goto fail;
- }
- }
- if (!out.commit())
- {
- QLOG_ERROR() << "error closing output file" << filename;
- goto fail;
- }
- #ifndef _WIN32
- // Set the executable bit.
- // We could try setting the full permissions as stored in the file, but why bother.
- if (finfo.external_fa & 0x400000)
- {
- if (!QFile::setPermissions(writepath, QFileDevice::Permissions(0x5145)))
- {
- QLOG_ERROR() << "could not set output executable bit on extracted file";
- goto fail;
- }
- }
- #endif
- }
- success = true;
- fail:
- unzClose(file);
- return success;
- }
- #else /* ifdef HAVE_MINIZIP */
- ///////////////////////////////////////////////////////////////////////////////////////////////////
- static bool extractZip(QString zip, QString dest)
- {
- return false;
- }
- #endif
- ///////////////////////////////////////////////////////////////////////////////////////////////////
- void CodecsFetcher::processCodecDownloadDone(const QVariant& context, const QByteArray& data)
- {
- QByteArray hash = QCryptographicHash::hash(data, QCryptographicHash::Sha1);
- if (hash != m_currentHash)
- {
- QLOG_ERROR() << "Checksum mismatch: got" << hash.toHex() << "expected" << m_currentHash.toHex();
- return;
- }
- if (context == QVariant("eae"))
- {
- QString dest = eaePrefixPath() + ".zip";
- QLOG_INFO() << "Storing EAE as" << dest;
- if (!Utils::safelyWriteFile(dest, data))
- {
- QLOG_ERROR() << "Writing codec file failed.";
- return;
- }
- QDir dir(dest);
- dir.removeRecursively();
- if (!extractZip(dest, eaePrefixPath()))
- {
- QLOG_ERROR() << "Could not extract zip.";
- dir.removeRecursively();
- return;
- }
- QFile::remove(dest);
- }
- else
- {
- CodecDriver codec = context.value<CodecDriver>();
- QLOG_INFO() << "Storing codec as" << codec.getPath();
- if (!Utils::safelyWriteFile(codec.getPath(), data))
- {
- QLOG_ERROR() << "Writing codec file failed.";
- return;
- }
- // This causes libmpv and eventually libavcodec to rescan and load new codecs.
- Codecs::updateCachedCodecList();
- for (const CodecDriver& item : Codecs::getCachedCodecList())
- {
- if (Codecs::sameCodec(item, codec) && !item.present)
- {
- QLOG_ERROR() << "Codec could not be loaded after installing it.";
- return;
- }
- }
- }
- QLOG_INFO() << "Codec download and installation succeeded.";
- }
- ///////////////////////////////////////////////////////////////////////////////////////////////////
- void CodecsFetcher::codecDownloadDone(QVariant userData, bool success, const QByteArray& data)
- {
- QLOG_INFO() << "Codec request finished.";
- if (success)
- {
- processCodecDownloadDone(userData, data);
- }
- else
- {
- QLOG_ERROR() << "Codec download HTTP request failed.";
- }
- startNext();
- }
- ///////////////////////////////////////////////////////////////////////////////////////////////////
- void CodecsFetcher::startEAE()
- {
- if (!g_eaeProcess)
- {
- g_eaeProcess = new QProcess();
- g_eaeProcess->setProcessChannelMode(QProcess::ForwardedChannels);
- connect(g_eaeProcess, &QProcess::stateChanged,
- [](QProcess::ProcessState s)
- {
- QLOG_INFO() << "EAE process state:" << s;
- }
- );
- connect(g_eaeProcess, &QProcess::errorOccurred,
- [](QProcess::ProcessError e)
- {
- QLOG_INFO() << "EAE process error:" << e;
- }
- );
- connect(g_eaeProcess, static_cast<void(QProcess::*)(int, QProcess::ExitStatus)>(&QProcess::finished),
- [](int exitCode, QProcess::ExitStatus exitStatus)
- {
- QLOG_INFO() << "EAE process finished:" << exitCode << exitStatus;
- }
- );
- }
- if (g_eaeProcess->state() == QProcess::NotRunning)
- {
- if (g_eaeProcess->program().size())
- {
- int exitCode = g_eaeProcess->exitStatus() == QProcess::NormalExit ? g_eaeProcess->exitCode() : -1;
- QLOG_ERROR() << "EAE died with exit code" << exitCode;
- }
- QLOG_INFO() << "Starting EAE.";
- g_eaeProcess->setProgram(eaeBinaryPath());
- g_eaeProcess->setWorkingDirectory(g_eaeWatchFolder);
- QDir dir(g_eaeWatchFolder);
- dir.removeRecursively();
- dir.mkpath(".");
- static const QStringList watchfolder_names =
- {
- "Convert to WAV (to 2ch or less)",
- "Convert to WAV (to 8ch or less)",
- "Convert to Dolby Digital (Low Quality - 384 kbps)",
- "Convert to Dolby Digital (High Quality - 640 kbps)",
- "Convert to Dolby Digital Plus (High Quality - 384 kbps)",
- "Convert to Dolby Digital Plus (Max Quality - 1024 kbps)",
- };
- for (auto folder : watchfolder_names)
- {
- if (!dir.mkpath(folder))
- {
- QLOG_ERROR() << "Could not create watch folder";
- }
- }
- g_eaeProcess->start();
- }
- }
- ///////////////////////////////////////////////////////////////////////////////////////////////////
- Downloader::Downloader(QVariant userData, const QUrl& url, const HeaderList& headers, QObject* parent)
- : QObject(parent), m_userData(userData), m_lastProgress(-1)
- {
- QLOG_INFO() << "HTTP request:" << url.toDisplayString();
- m_currentStartTime.start();
- connect(&m_WebCtrl, &QNetworkAccessManager::finished, this, &Downloader::networkFinished);
- QNetworkRequest request(url);
- for (int n = 0; n < headers.size(); n++)
- request.setRawHeader(headers[n].first.toUtf8(), headers[n].second.toUtf8());
- QNetworkReply *reply = m_WebCtrl.get(request);
- if (reply)
- {
- connect(reply, &QNetworkReply::downloadProgress, this, &Downloader::downloadProgress);
- }
- }
- ///////////////////////////////////////////////////////////////////////////////////////////////////
- void Downloader::downloadProgress(qint64 bytesReceived, qint64 bytesTotal)
- {
- if (bytesTotal > 0)
- {
- int progress = bytesReceived * 100 / bytesTotal;
- if (m_lastProgress < 0 || progress > m_lastProgress + 10)
- {
- m_lastProgress = progress;
- QLOG_INFO() << "HTTP request at" << progress << "% (" << bytesReceived << "/" << bytesTotal << ")";
- }
- }
- }
- ///////////////////////////////////////////////////////////////////////////////////////////////////
- void Downloader::networkFinished(QNetworkReply* pReply)
- {
- QLOG_INFO() << "HTTP finished after" << (m_currentStartTime.elapsed() + 500) / 1000
- << "seconds for a request of" << pReply->size() << "bytes.";
- if (pReply->error() == QNetworkReply::NoError)
- {
- emit done(m_userData, true, pReply->readAll());
- }
- else
- {
- QLOG_ERROR() << "HTTP download error:" << pReply->errorString();
- emit done(m_userData, false, QByteArray());
- }
- pReply->deleteLater();
- m_WebCtrl.clearAccessCache(); // make sure the TCP connection is closed
- }
- ///////////////////////////////////////////////////////////////////////////////////////////////////
- static CodecDriver selectBestDecoder(const StreamInfo& stream)
- {
- QList<CodecDriver> codecs = Codecs::findCodecsByFormat(Codecs::getCachedCodecList(), CodecType::Decoder, stream.codec);
- CodecDriver best = {};
- int bestScore = -1;
- for (auto codec : codecs)
- {
- int score = -1;
- if (codec.isSystemCodec())
- {
- // we always want to avoid using non-whitelisted system codecs
- // on the other hand, always prefer whitelisted system codecs
- if ((codec.isWhitelistedSystemAudioCodec() && useSystemAudioDecoders()) ||
- (codec.isWhitelistedSystemVideoCodec() && useSystemVideoDecoders()))
- score = 10;
- if (codec.format == "h264")
- {
- // Avoid using system video decoders for h264 profiles usually not supported.
- if (stream.profile != "" && stream.profile != "main" && stream.profile != "baseline" && stream.profile != "high")
- score = 1;
- }
- if (codec.driver == "h264_mf")
- {
- if (!stream.videoResolution.isEmpty())
- {
- QSize res = stream.videoResolution;
- if (res.width() > g_mediaFoundationH264MaxResolution.width() ||
- res.height() > g_mediaFoundationH264MaxResolution.height())
- score = 1;
- }
- }
- if (codec.driver == "aac_mf")
- {
- // Arbitrary but documented and enforced 6 channel limit by MS.
- if (stream.audioChannels > 6)
- score = 1;
- // Another arbitrary limit.
- if (stream.audioSampleRate > 0 && (stream.audioSampleRate < 8000 || stream.audioSampleRate > 48000))
- score = 1;
- }
- if (codec.getSystemCodecType() == "eae")
- score = HAVE_EAE ? 2 : -1;
- }
- else
- {
- // prefer codecs which do not have to be downloaded over others
- if (codec.present)
- score = 15;
- else
- score = 5;
- }
- if (score > bestScore && score >= 0)
- {
- best = codec;
- bestScore = score;
- }
- }
- return best;
- }
- ///////////////////////////////////////////////////////////////////////////////////////////////////
- QList<CodecDriver> Codecs::determineRequiredCodecs(const PlaybackInfo& info)
- {
- QList<CodecDriver> result;
- bool needAC3Encoder = false;
- QLOG_INFO() << "Using system audio decoders:" << useSystemAudioDecoders();
- QLOG_INFO() << "Using system video decoders:" << useSystemVideoDecoders();
- #if !defined(HAVE_CODEC_MANIFEST)
- QLOG_INFO() << "Not using on-demand codecs.";
- #endif
- for (auto stream : info.streams)
- {
- if (!stream.isVideo && !stream.isAudio)
- continue;
- if (!stream.codec.size())
- {
- QLOG_ERROR() << "unidentified codec";
- continue;
- }
- // We could do this if we'd find a nice way to enable passthrough by default:
- #if 0
- // Can passthrough be used? If so, don't request a codec.
- if (info.audioPassthroughCodecs.contains(stream.codec))
- continue;
- #endif
- // (Would be nice to check audioChannels here to not request the encoder
- // when playing stereo - but unfortunately, the ac3 encoder is loaded first,
- // and only removed when detecting stereo input)
- if (info.enableAC3Transcoding)
- needAC3Encoder = true;
- CodecDriver best = selectBestDecoder(stream);
- if (best.valid())
- {
- result.append(best);
- }
- else
- {
- QLOG_ERROR() << "no decoder for" << stream.codec;
- }
- }
- if (needAC3Encoder)
- {
- QList<CodecDriver> codecs = Codecs::findCodecsByFormat(Codecs::getCachedCodecList(), CodecType::Encoder, "ac3");
- CodecDriver encoder = {};
- for (auto codec : codecs)
- {
- if (codec.present && (!codec.isSystemCodec() || codec.isWhitelistedSystemAudioCodec()))
- {
- encoder = codec;
- break;
- }
- if (codec.external)
- encoder = codec; // fallback
- }
- if (encoder.valid())
- {
- result.append(encoder);
- }
- else
- {
- QLOG_ERROR() << "no AC3 encoder available";
- }
- }
- return result;
- }
- void Codecs::Uninit()
- {
- if (g_eaeProcess)
- {
- delete g_eaeProcess;
- g_eaeProcess = nullptr;
- }
- if (!g_eaeWatchFolder.isEmpty())
- {
- QDir dir(g_eaeWatchFolder);
- dir.removeRecursively();
- }
- }
|