/**************************************************************************** ** ** Copyright (C) 2016 The Qt Company Ltd. ** Contact: https://www.qt.io/licensing/ ** ** This file is part of PySide2. ** ** $QT_BEGIN_LICENSE:GPL-EXCEPT$ ** Commercial License Usage ** Licensees holding valid commercial Qt licenses may use this file in ** accordance with the commercial license agreement provided with the ** Software or, alternatively, in accordance with the terms contained in ** a written agreement between you and The Qt Company. For licensing terms ** and conditions see https://www.qt.io/terms-conditions. For further ** information use the contact form at https://www.qt.io/contact-us. ** ** GNU General Public License Usage ** Alternatively, this file may be used under the terms of the GNU ** General Public License version 3 as published by the Free Software ** Foundation with exceptions as appearing in the file LICENSE.GPL3-EXCEPT ** included in the packaging of this file. Please review the following ** information to ensure the GNU General Public License requirements will ** be met: https://www.gnu.org/licenses/gpl-3.0.html. ** ** $QT_END_LICENSE$ ** ****************************************************************************/ #include "apiextractor.h" #include "abstractmetalang.h" #include #include #include #include #include "reporthandler.h" #include "typesystem.h" #include "fileout.h" #include "parser/rpp/pp.h" #include "abstractmetabuilder.h" #include "typedatabase.h" #include "typesystem.h" static bool preprocess(const QString& sourceFile, QFile& targetFile, const QStringList& includes); ApiExtractor::ApiExtractor() : m_builder(0) { static bool qrcInitialized = false; if (!qrcInitialized) Q_INIT_RESOURCE(generator); // Environment TYPESYSTEMPATH QString envTypesystemPaths = QFile::decodeName(getenv("TYPESYSTEMPATH")); if (!envTypesystemPaths.isEmpty()) TypeDatabase::instance()->addTypesystemPath(envTypesystemPaths); } ApiExtractor::~ApiExtractor() { delete m_builder; } void ApiExtractor::addTypesystemSearchPath (const QString& path) { TypeDatabase::instance()->addTypesystemPath(path); } void ApiExtractor::addTypesystemSearchPath(const QStringList& paths) { foreach (const QString &path, paths) addTypesystemSearchPath(path); } void ApiExtractor::addIncludePath(const QString& path) { m_includePaths << path; } void ApiExtractor::addIncludePath(const QStringList& paths) { m_includePaths << paths; } void ApiExtractor::setLogDirectory(const QString& logDir) { m_logDirectory = logDir; } void ApiExtractor::setCppFileName(const QString& cppFileName) { m_cppFileName = cppFileName; } void ApiExtractor::setTypeSystem(const QString& typeSystemFileName) { m_typeSystemFileName = typeSystemFileName; } void ApiExtractor::setDebugLevel(ReportHandler::DebugLevel debugLevel) { ReportHandler::setDebugLevel(debugLevel); } void ApiExtractor::setSuppressWarnings ( bool value ) { TypeDatabase::instance()->setSuppressWarnings(value); } void ApiExtractor::setSilent ( bool value ) { ReportHandler::setSilent(value); } bool ApiExtractor::setApiVersion(const QString& package, const QString &version) { return TypeDatabase::instance()->setApiVersion(package, version); } void ApiExtractor::setDropTypeEntries(QString dropEntries) { dropEntries.remove(QLatin1Char(' ')); QStringList entries = dropEntries.split(QLatin1Char(';')); TypeDatabase::instance()->setDropTypeEntries(entries); } AbstractMetaEnumList ApiExtractor::globalEnums() const { Q_ASSERT(m_builder); return m_builder->globalEnums(); } AbstractMetaFunctionList ApiExtractor::globalFunctions() const { Q_ASSERT(m_builder); return m_builder->globalFunctions(); } AbstractMetaClassList ApiExtractor::classes() const { Q_ASSERT(m_builder); return m_builder->classes(); } AbstractMetaClassList ApiExtractor::smartPointers() const { Q_ASSERT(m_builder); return m_builder->smartPointers(); } AbstractMetaClassList ApiExtractor::classesTopologicalSorted(const Dependencies &additionalDependencies) const { Q_ASSERT(m_builder); return m_builder->classesTopologicalSorted(Q_NULLPTR, additionalDependencies); } PrimitiveTypeEntryList ApiExtractor::primitiveTypes() const { return TypeDatabase::instance()->primitiveTypes(); } ContainerTypeEntryList ApiExtractor::containerTypes() const { return TypeDatabase::instance()->containerTypes(); } QSet ApiExtractor::qtMetaTypeDeclaredTypeNames() const { Q_ASSERT(m_builder); return m_builder->qtMetaTypeDeclaredTypeNames(); } static const AbstractMetaEnum* findEnumOnClasses(AbstractMetaClassList metaClasses, const EnumTypeEntry* typeEntry) { const AbstractMetaEnum* result = 0; foreach (const AbstractMetaClass* metaClass, metaClasses) { foreach (const AbstractMetaEnum* metaEnum, metaClass->enums()) { if (metaEnum->typeEntry() == typeEntry) { result = metaEnum; break; } } if (result) break; result = findEnumOnClasses(metaClass->innerClasses(), typeEntry); } return result; } const AbstractMetaEnum* ApiExtractor::findAbstractMetaEnum(const EnumTypeEntry* typeEntry) const { if (!typeEntry) return 0; foreach (AbstractMetaEnum* metaEnum, m_builder->globalEnums()) { if (metaEnum->typeEntry() == typeEntry) return metaEnum; } return findEnumOnClasses(m_builder->classes(), typeEntry); } const AbstractMetaEnum* ApiExtractor::findAbstractMetaEnum(const TypeEntry* typeEntry) const { if (!typeEntry) return 0; if (typeEntry->isFlags()) return findAbstractMetaEnum(reinterpret_cast(typeEntry)); if (typeEntry->isEnum()) return findAbstractMetaEnum(reinterpret_cast(typeEntry)); return 0; } const AbstractMetaEnum* ApiExtractor::findAbstractMetaEnum(const FlagsTypeEntry* typeEntry) const { if (!typeEntry) return 0; return findAbstractMetaEnum(typeEntry->originator()); } const AbstractMetaEnum* ApiExtractor::findAbstractMetaEnum(const AbstractMetaType* metaType) const { if (!metaType) return 0; return findAbstractMetaEnum(metaType->typeEntry()); } int ApiExtractor::classCount() const { Q_ASSERT(m_builder); return m_builder->classes().count(); } bool ApiExtractor::run() { if (m_builder) return false; if (m_typeSystemFileName.isEmpty()) { std::cerr << "You must specify a Type System file." << std::endl; return false; } else if (!TypeDatabase::instance()->parseFile(m_typeSystemFileName)) { std::cerr << "Cannot parse file: " << qPrintable(m_typeSystemFileName); return false; } QTemporaryFile ppFile; #ifndef NDEBUG ppFile.setAutoRemove(false); #endif // make sure that a tempfile can be written if (!ppFile.open()) { std::cerr << "could not create tempfile in " << qPrintable(QDir::tempPath()); return false; } // run rpp pre-processor if (!preprocess(m_cppFileName, ppFile, m_includePaths)) { std::cerr << "Preprocessor failed on file: " << qPrintable(m_cppFileName); return false; } ppFile.seek(0); m_builder = new AbstractMetaBuilder; m_builder->setLogDirectory(m_logDirectory); m_builder->setGlobalHeader(m_cppFileName); m_builder->build(&ppFile); return true; } static bool preprocess(const QString& sourceFile, QFile& targetFile, const QStringList& includes) { rpp::pp_environment env; rpp::pp preprocess(env); rpp::pp_null_output_iterator null_out; const char *ppconfig = ":/trolltech/generator/pp-qt-configuration"; const QString fileName = QLatin1String(ppconfig); QFile file(fileName); if (!file.open(QFile::ReadOnly)) { std::cerr << "Preprocessor configuration file not found " << ppconfig << std::endl; return false; } QByteArray ba = file.readAll(); file.close(); preprocess.operator()(ba.constData(), ba.constData() + ba.size(), null_out); preprocess.push_include_path("."); foreach (const QString &include, includes) preprocess.push_include_path(QDir::toNativeSeparators(include).toStdString()); preprocess.push_include_path("/usr/include"); QString currentDir = QDir::current().absolutePath(); QFileInfo sourceInfo(sourceFile); if (!sourceInfo.exists()) { std::cerr << "File not found " << qPrintable(sourceFile) << std::endl; return false; } QDir::setCurrent(sourceInfo.absolutePath()); std::string result; result.reserve(20 * 1024); // 20K result += "# 1 \"builtins\"\n"; result += "# 1 \""; result += sourceFile.toStdString(); result += "\"\n"; preprocess.file(sourceInfo.fileName().toStdString(), rpp::pp_output_iterator (result)); QDir::setCurrent(currentDir); if (!targetFile.open(QIODevice::ReadWrite | QIODevice::Text)) { std::cerr << "Failed to write preprocessed file: " << qPrintable(targetFile.fileName()) << std::endl; return false; } targetFile.write(result.c_str(), result.length()); return true; } #ifndef QT_NO_DEBUG_STREAM template static void debugFormatSequence(QDebug &d, const char *key, const Container& c) { typedef typename Container::const_iterator ConstIt; if (c.isEmpty()) return; const ConstIt begin = c.begin(); const ConstIt end = c.end(); d << "\n " << key << '[' << c.size() << "]=("; for (ConstIt it = begin; it != end; ++it) { if (it != begin) d << ", "; d << *it; } d << ')'; } QDebug operator<<(QDebug d, const ApiExtractor &ae) { QDebugStateSaver saver(d); d.noquote(); d.nospace(); d << "ApiExtractor(typeSystem=\"" << ae.typeSystem() << "\", cppFileName=\"" << ae.cppFileName() << ", "; ae.m_builder->formatDebug(d); d << ')'; return d; } #endif // QT_NO_DEBUG_STREAM