// Copyright (c) 2008 Roberto Raggi // // Permission is hereby granted, free of charge, to any person obtaining a copy // of this software and associated documentation files (the "Software"), to deal // in the Software without restriction, including without limitation the rights // to use, copy, modify, merge, publish, distribute, sublicense, and/or sell // copies of the Software, and to permit persons to whom the Software is // furnished to do so, subject to the following conditions: // // The above copyright notice and this permission notice shall be included in // all copies or substantial portions of the Software. // // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR // IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, // FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE // AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER // LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, // OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN // THE SOFTWARE. #ifndef CPLUSPLUS_TRANSLATIONUNIT_H #define CPLUSPLUS_TRANSLATIONUNIT_H #include "CPlusPlusForwardDeclarations.h" #include "ASTfwd.h" #include "Token.h" #include "DiagnosticClient.h" #include #include #if !(__cplusplus > 199711L || __GXX_EXPERIMENTAL_CXX0X__ || _MSC_VER >= 1600 || defined( _LIBCPP_VERSION )) #define USE_TR1 #endif #if defined(_MSC_VER) && _MSC_VER < 1600 # include #elif defined(USE_TR1) # include #else # include #endif namespace CPlusPlus { class CPLUSPLUS_EXPORT TranslationUnit { TranslationUnit(const TranslationUnit &other); void operator =(const TranslationUnit &other); public: TranslationUnit(Control *control, const StringLiteral *fileId); ~TranslationUnit(); Control *control() const; const StringLiteral *fileId() const; const char *fileName() const; unsigned fileNameLength() const; const char *firstSourceChar() const; const char *lastSourceChar() const; unsigned sourceLength() const; void setSource(const char *source, unsigned size); unsigned tokenCount() const { return _tokens ? unsigned(_tokens->size()) : unsigned(0); } const Token &tokenAt(unsigned index) const { return _tokens && index < tokenCount() ? (*_tokens)[index] : nullToken; } int tokenKind(unsigned index) const { return tokenAt(index).f.kind; } const char *spell(unsigned index) const; unsigned commentCount() const; const Token &commentAt(unsigned index) const; unsigned matchingBrace(unsigned index) const; const Identifier *identifier(unsigned index) const; const Literal *literal(unsigned index) const; const StringLiteral *stringLiteral(unsigned index) const; const NumericLiteral *numericLiteral(unsigned index) const; MemoryPool *memoryPool() const; AST *ast() const; bool blockErrors() const { return f._blockErrors; } bool blockErrors(bool block) { const bool previous = f._blockErrors; f._blockErrors = block; return previous; } void warning(unsigned index, const char *fmt, ...); void error(unsigned index, const char *fmt, ...); void fatal(unsigned index, const char *fmt, ...); void message(DiagnosticClient::Level level, unsigned index, const char *format, va_list ap); bool isTokenized() const; void tokenize(); bool skipFunctionBody() const; void setSkipFunctionBody(bool skipFunctionBody); bool isParsed() const; enum ParseMode { ParseTranlationUnit, ParseDeclaration, ParseExpression, ParseDeclarator, ParseStatement }; bool parse(ParseMode mode = ParseTranlationUnit); void resetAST(); void release(); void getTokenStartPosition(unsigned index, unsigned *line, unsigned *column = 0, const StringLiteral **fileName = 0) const; void getTokenEndPosition(unsigned index, unsigned *line, unsigned *column = 0, const StringLiteral **fileName = 0) const; void getPosition(unsigned offset, unsigned *line, unsigned *column = 0, const StringLiteral **fileName = 0) const; void getTokenPosition(unsigned index, unsigned *line, unsigned *column = 0, const StringLiteral **fileName = 0) const; void pushLineOffset(unsigned offset); void pushPreprocessorLine(unsigned offset, unsigned line, const StringLiteral *fileName); unsigned findPreviousLineOffset(unsigned tokenIndex) const; bool maybeSplitGreaterGreaterToken(unsigned tokenIndex); LanguageFeatures languageFeatures() const { return _languageFeatures; } void setLanguageFeatures(LanguageFeatures features) { _languageFeatures = features; } private: struct PPLine { unsigned offset; unsigned line; const StringLiteral *fileName; PPLine(unsigned offset = 0, unsigned line = 0, const StringLiteral *fileName = 0) : offset(offset), line(line), fileName(fileName) { } bool operator == (const PPLine &other) const { return offset == other.offset; } bool operator != (const PPLine &other) const { return offset != other.offset; } bool operator < (const PPLine &other) const { return offset < other.offset; } }; void releaseTokensAndComments(); unsigned findLineNumber(unsigned offset) const; unsigned findColumnNumber(unsigned offset, unsigned lineNumber) const; PPLine findPreprocessorLine(unsigned offset) const; void showErrorLine(unsigned index, unsigned column, FILE *out); static const Token nullToken; Control *_control; const StringLiteral *_fileId; const char *_firstSourceChar; const char *_lastSourceChar; std::vector *_tokens; std::vector *_comments; std::vector _lineOffsets; std::vector _ppLines; #if defined(_MSC_VER) && _MSC_VER < 1600 // MSVC2008 and earlier do not implement TR1. typedef std::map > TokenLineColumn; #elif defined(USE_TR1) typedef std::tr1::unordered_map > TokenLineColumn; #else typedef std::unordered_map > TokenLineColumn; #endif TokenLineColumn _expandedLineColumn; MemoryPool *_pool; AST *_ast; TranslationUnit *_previousTranslationUnit; struct Flags { unsigned _tokenized: 1; unsigned _parsed: 1; unsigned _blockErrors: 1; unsigned _skipFunctionBody: 1; }; union { unsigned _flags; Flags f; }; LanguageFeatures _languageFeatures; }; } // namespace CPlusPlus #endif // CPLUSPLUS_TRANSLATIONUNIT_H