/**************************************************************************** ** ** Copyright (C) 2013 Digia Plc and/or its subsidiary(-ies). ** Contact: http://www.qt-project.org/legal ** ** This file is part of Qt Creator. ** ** 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 Digia. For licensing terms and ** conditions see http://qt.digia.com/licensing. For further information ** use the contact form at http://qt.digia.com/contact-us. ** ** GNU Lesser General Public License Usage ** Alternatively, this file may be used under the terms of the GNU Lesser ** General Public License version 2.1 as published by the Free Software ** Foundation and appearing in the file LICENSE.LGPL included in the ** packaging of this file. Please review the following information to ** ensure the GNU Lesser General Public License version 2.1 requirements ** will be met: http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html. ** ** In addition, as a special exception, Digia gives you certain additional ** rights. These rights are described in the Digia Qt LGPL Exception ** version 1.1, included in the file LGPL_EXCEPTION.txt in this package. ** ****************************************************************************/ #ifndef UNIT_H #define UNIT_H #include "utils.h" #include "cxraii.h" #include #include #include #include #include QT_BEGIN_NAMESPACE class QDateTime; QT_END_NAMESPACE namespace ClangCodeModel { namespace Internal { class UnitData; /* * This is a minimal wrapper around clang's translation unit functionality. * It should should contain only the very basic primitives which allow other * components such as code completion, code navigation, and others to access * data which directly depends on the translation unit. * * In other words, what's wrapped here is only the functions that receive a * CXTranslationUnit as a parameter. And this class itself is then the corresponding * abstraction of the CXTranslationUnit. * * Notes: * - This class is not thread-safe. * - It's responsibility of the client to make sure that the wrapped translation * unit is consistent with the other data such as cursor and locations being used. * - The data of the TU is shared. * * * @TODO: This is similar but not exactly the same as the current ClangWrapper class. * That class is now tuned to specific features, so it's not generic enough to be used * an underlying component and aslo don't provide the data in a fine granularity as * needed here. At some point we should split ClangWrapper into its different logical * components and use this is the underlying structure. */ class Unit { public: Unit(); explicit Unit(const QString &fileName); Unit(const Unit &unit); Unit &operator=(const Unit &unit); ~Unit(); bool isLoaded() const; const QString fileName() const; const QDateTime &timeStamp() const; QStringList compilationOptions() const; void setCompilationOptions(const QStringList &compOptions); UnsavedFiles unsavedFiles() const; void setUnsavedFiles(const UnsavedFiles &unsavedFiles); unsigned managementOptions() const; void setManagementOptions(unsigned managementOptions); // Sharing control facilities // Method isUnique is just like an "isDetached", however makeUnique is mostly some kind // of "detach" but which actually moves the data to this particular instance, invalidating // then all the other shares. bool isUnique() const; void makeUnique(); // Methods for generating the TU. Name mappings are direct, for example: // - parse corresponds to clang_parseTranslationUnit // - createFromSourceFile corresponds to clang_createTranslationUnitFromSourceFile void parse(); void reparse(); void create(); void createFromSourceFile(); int save(const QString &unitFileName); void unload(); // Simple forwarding methods, separated by clang categories for convenience. // As above, the names are directly mapped. Separated by categories as clang for convenience. // Note that only methods that take the TU as a parameter should be wrapped. // - Diagnostic reporting unsigned getNumDiagnostics() const; CXDiagnostic getDiagnostic(unsigned index) const; // - Translation unit manipulation CXString getTranslationUnitSpelling() const; // - File manipulation routines CXFile getFile() const; // - Mapping between cursors and source code CXCursor getCursor(const CXSourceLocation &location) const; // - Miscellaneous utility functions void getInclusions(CXInclusionVisitor visitor, CXClientData clientData) const; // - Cursor manipulations CXCursor getTranslationUnitCursor() const; // - Physical source locations CXSourceLocation getLocation(const CXFile &file, unsigned line, unsigned column) const; void codeCompleteAt(unsigned line, unsigned column, ScopedCXCodeCompleteResults &results); void tokenize(CXSourceRange range, CXToken **tokens, unsigned *tokenCount) const; void disposeTokens(CXToken *tokens, unsigned tokenCount) const; CXSourceRange getTokenExtent(const CXToken &token) const; void annotateTokens(CXToken *tokens, unsigned tokenCount, CXCursor *cursors) const; CXTranslationUnit clangTranslationUnit() const; CXIndex clangIndex() const; QString getTokenSpelling(const CXToken &tok) const; private: QExplicitlySharedDataPointer m_data; }; class IdentifierTokens { Q_DISABLE_COPY(IdentifierTokens) public: IdentifierTokens(const Unit &m_unit, unsigned firstLine, unsigned lastLine); ~IdentifierTokens(); unsigned count() const { return m_tokenCount; } const CXToken &token(unsigned nr) const { Q_ASSERT(nr < count()); return m_tokens[nr]; } const CXCursor &cursor(unsigned nr) const { Q_ASSERT(nr < count()); return m_cursors[nr]; } const CXSourceRange &extent(unsigned nr) const { Q_ASSERT(nr < count()); return m_extents[nr]; } private: void dispose(); private: const Unit &m_unit; unsigned m_tokenCount; CXToken *m_tokens; CXCursor *m_cursors; CXSourceRange *m_extents; }; } // Internal } // Clang Q_DECLARE_METATYPE(ClangCodeModel::Internal::Unit) #endif // UNIT_H