Files
qt-creator/src/tools/clangrefactoringbackend/source/sourcerangeextractor.cpp
Marco Bubke 542c730b33 Clang: Suppress unused parameter warnings in clang headers
Change-Id: I8b2d50be5bda75b6759b7ce73b9d37b2500f3efc
Reviewed-by: Tim Jenssen <tim.jenssen@qt.io>
2016-12-08 11:57:35 +00:00

193 lines
7.6 KiB
C++

/****************************************************************************
**
** Copyright (C) 2016 The Qt Company Ltd.
** Contact: https://www.qt.io/licensing/
**
** 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 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.
**
****************************************************************************/
#include "sourcerangeextractor.h"
#include "sourcelocationsutils.h"
#include <sourcerangescontainer.h>
#if defined(__GNUC__)
# pragma GCC diagnostic push
# pragma GCC diagnostic ignored "-Wunused-parameter"
#elif defined(_MSC_VER)
# pragma warning(push)
# pragma warning( disable : 4100 )
#endif
#include <clang/Basic/SourceManager.h>
#include <clang/Lex/Lexer.h>
#include <llvm/Support/FileSystem.h>
#include <llvm/Support/FileUtilities.h>
#include <llvm/ADT/SmallVector.h>
#if defined(__GNUC__)
# pragma GCC diagnostic pop
#elif defined(_MSC_VER)
# pragma warning(pop)
#endif
namespace ClangBackEnd {
SourceRangeExtractor::SourceRangeExtractor(const clang::SourceManager &sourceManager,
const clang::LangOptions &languageOptions,
SourceRangesContainer &sourceRangesContainer)
: sourceManager(sourceManager),
languageOptions(languageOptions),
sourceRangesContainer(sourceRangesContainer)
{
}
namespace {
template<typename Iterator>
std::reverse_iterator<Iterator> make_reverse_iterator(Iterator iterator)
{
return std::reverse_iterator<Iterator>(iterator);
}
}
const char *SourceRangeExtractor::findStartOfLineInBuffer(llvm::StringRef buffer, uint startOffset)
{
auto beginText = buffer.begin() + startOffset;
auto reverseBegin = make_reverse_iterator(beginText);
auto reverseEnd = make_reverse_iterator(buffer.begin());
auto found = std::find_if(reverseBegin,
reverseEnd,
[] (const char character) {
return character == '\n' || character == '\r';
});
if (found != reverseEnd)
return found.base();
return buffer.begin();
}
const char *SourceRangeExtractor::findEndOfLineInBuffer(llvm::StringRef buffer, uint endOffset)
{
auto beginText = buffer.begin() + endOffset;
auto found = std::find_if(beginText,
buffer.end(),
[] (const char character) {
return character == '\n' || character == '\r';
});
if (found != buffer.end())
return found;
return buffer.end();
}
Utils::SmallString SourceRangeExtractor::getExpandedText(llvm::StringRef buffer,
uint startOffset,
uint endOffset)
{
auto startBuffer = findStartOfLineInBuffer(buffer, startOffset);
auto endBuffer = findEndOfLineInBuffer(buffer, endOffset);
return Utils::SmallString(startBuffer, endBuffer);
}
const clang::SourceRange SourceRangeExtractor::extendSourceRangeToLastTokenEnd(const clang::SourceRange sourceRange)
{
auto endLocation = sourceRange.getEnd();
uint length = clang::Lexer::MeasureTokenLength(sourceManager.getSpellingLoc(endLocation),
sourceManager,
languageOptions);
endLocation = endLocation.getLocWithOffset(length);
return {sourceRange.getBegin(), endLocation};
}
void SourceRangeExtractor::insertSourceRange(uint fileHash,
Utils::SmallString &&directoryPath,
Utils::SmallString &&fileName,
const clang::FullSourceLoc &startLocation,
uint startOffset,
const clang::FullSourceLoc &endLocation,
uint endOffset,
Utils::SmallString &&lineSnippet)
{
sourceRangesContainer.insertFilePath(fileHash,
std::move(directoryPath),
std::move(fileName));
sourceRangesContainer.insertSourceRange(fileHash,
startLocation.getSpellingLineNumber(),
startLocation.getSpellingColumnNumber(),
startOffset,
endLocation.getSpellingLineNumber(),
endLocation.getSpellingColumnNumber(),
endOffset,
std::move(lineSnippet));
}
void SourceRangeExtractor::addSourceRange(const clang::SourceRange &sourceRange)
{
auto extendedSourceRange = extendSourceRangeToLastTokenEnd(sourceRange);
clang::FullSourceLoc startSourceLocation(extendedSourceRange.getBegin(), sourceManager);
clang::FullSourceLoc endSourceLocation(extendedSourceRange.getEnd(), sourceManager);
if (startSourceLocation.isFileID() && endSourceLocation.isFileID()) {
const auto startDecomposedLoction = startSourceLocation.getDecomposedLoc();
const auto endDecomposedLoction = endSourceLocation.getDecomposedLoc();
const auto fileId = startDecomposedLoction.first;
const auto startOffset = startDecomposedLoction.second;
const auto endOffset = endDecomposedLoction.second;
const auto fileEntry = sourceManager.getFileEntryForID(fileId);
auto filePath = absolutePath(fileEntry->getName());
const auto fileName = llvm::sys::path::filename(filePath);
llvm::sys::path::remove_filename(filePath);
Utils::SmallString lineSnippet = getExpandedText(startSourceLocation.getBufferData(),
startOffset,
endOffset);
insertSourceRange(fileId.getHashValue(),
fromNativePath(filePath),
{fileName.data(), fileName.size()},
startSourceLocation,
startOffset,
endSourceLocation,
endOffset,
std::move(lineSnippet));
}
}
void SourceRangeExtractor::addSourceRanges(const std::vector<clang::SourceRange> &sourceRanges)
{
sourceRangesContainer.reserve(sourceRanges.size() + sourceRangeWithTextContainers().size());
for (const clang::SourceRange &sourceRange : sourceRanges)
addSourceRange(sourceRange);
}
const std::vector<SourceRangeWithTextContainer> &SourceRangeExtractor::sourceRangeWithTextContainers() const
{
return sourceRangesContainer.sourceRangeWithTextContainers();
}
} // namespace ClangBackEnd