2012-07-22 18:19:28 -04:00
|
|
|
// Copyright (C) 2011, 2012 Strahinja Val Markovic <val@markovic.io>
|
|
|
|
//
|
|
|
|
// This file is part of YouCompleteMe.
|
|
|
|
//
|
|
|
|
// YouCompleteMe is free software: you can redistribute it and/or modify
|
|
|
|
// it under the terms of the GNU General Public License as published by
|
|
|
|
// the Free Software Foundation, either version 3 of the License, or
|
|
|
|
// (at your option) any later version.
|
|
|
|
//
|
|
|
|
// YouCompleteMe is distributed in the hope that it will be useful,
|
|
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
// GNU General Public License for more details.
|
|
|
|
//
|
|
|
|
// You should have received a copy of the GNU General Public License
|
|
|
|
// along with YouCompleteMe. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
|
|
|
|
#include "ClangUtils.h"
|
2012-08-02 01:09:31 -04:00
|
|
|
#include "standard.h"
|
2012-07-22 18:19:28 -04:00
|
|
|
#include "Utils.h"
|
2012-08-02 01:09:31 -04:00
|
|
|
#include "UnsavedFile.h"
|
2012-07-22 18:19:28 -04:00
|
|
|
|
2012-08-02 01:09:31 -04:00
|
|
|
#include <vector>
|
|
|
|
#include <string>
|
2012-07-22 18:19:28 -04:00
|
|
|
|
2012-08-02 01:09:31 -04:00
|
|
|
#include <boost/unordered_map.hpp>
|
|
|
|
using boost::unordered_map;
|
2012-07-22 18:19:28 -04:00
|
|
|
|
|
|
|
namespace YouCompleteMe
|
|
|
|
{
|
|
|
|
|
2012-08-02 01:09:31 -04:00
|
|
|
std::vector< CXUnsavedFile > ToCXUnsavedFiles(
|
|
|
|
const std::vector< UnsavedFile > &unsaved_files )
|
|
|
|
{
|
|
|
|
std::vector< CXUnsavedFile > clang_unsaved_files( unsaved_files.size() );
|
|
|
|
for ( uint i = 0; i < unsaved_files.size(); ++i )
|
|
|
|
{
|
|
|
|
// TODO: assert non-null
|
|
|
|
clang_unsaved_files[ i ].Filename = unsaved_files[ i ].filename_;
|
|
|
|
clang_unsaved_files[ i ].Contents = unsaved_files[ i ].contents_;
|
|
|
|
clang_unsaved_files[ i ].Length = unsaved_files[ i ].length_;
|
|
|
|
}
|
|
|
|
|
|
|
|
return clang_unsaved_files;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// Returns true when the provided completion string is available to the user;
|
|
|
|
// unavailable completion strings refer to entities that are private/protected,
|
|
|
|
// deprecated etc.
|
|
|
|
bool CompletionStringAvailable( CXCompletionString completion_string )
|
2012-07-22 18:19:28 -04:00
|
|
|
{
|
2012-08-04 22:29:11 -04:00
|
|
|
if ( !completion_string )
|
|
|
|
return false;
|
2012-08-02 01:09:31 -04:00
|
|
|
return clang_getCompletionAvailability( completion_string ) ==
|
|
|
|
CXAvailability_Available;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
char DiagnosticSeverityToType( CXDiagnosticSeverity severity )
|
|
|
|
{
|
|
|
|
switch ( severity )
|
|
|
|
{
|
|
|
|
case CXDiagnostic_Ignored:
|
|
|
|
case CXDiagnostic_Note:
|
|
|
|
return 'I';
|
|
|
|
|
|
|
|
case CXDiagnostic_Warning:
|
|
|
|
return 'W';
|
2012-07-22 18:19:28 -04:00
|
|
|
|
2012-08-02 01:09:31 -04:00
|
|
|
case CXDiagnostic_Error:
|
|
|
|
case CXDiagnostic_Fatal:
|
|
|
|
return 'E';
|
|
|
|
|
|
|
|
default:
|
|
|
|
return 'E';
|
|
|
|
}
|
2012-07-22 18:19:28 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-08-02 01:09:31 -04:00
|
|
|
std::vector< CompletionData > ToCompletionDataVector(
|
|
|
|
CXCodeCompleteResults *results )
|
2012-07-22 18:19:28 -04:00
|
|
|
{
|
2012-08-02 01:09:31 -04:00
|
|
|
std::vector< CompletionData > completions;
|
2012-08-04 22:29:11 -04:00
|
|
|
if ( !results || !results->Results )
|
|
|
|
return completions;
|
2012-07-22 18:19:28 -04:00
|
|
|
|
2012-08-04 22:29:11 -04:00
|
|
|
completions.reserve( results->NumResults );
|
2012-08-02 01:09:31 -04:00
|
|
|
unordered_map< std::string, uint > seen_data;
|
2012-07-22 18:19:28 -04:00
|
|
|
|
2012-08-02 01:09:31 -04:00
|
|
|
for ( uint i = 0; i < results->NumResults; ++i )
|
2012-07-22 18:19:28 -04:00
|
|
|
{
|
2012-08-02 01:09:31 -04:00
|
|
|
CXCompletionResult completion_result = results->Results[ i ];
|
|
|
|
|
|
|
|
if ( !CompletionStringAvailable( completion_result.CompletionString ) )
|
|
|
|
continue;
|
|
|
|
|
2012-08-06 00:01:42 -04:00
|
|
|
CompletionData data( completion_result );
|
2012-08-02 01:09:31 -04:00
|
|
|
uint index = GetValueElseInsert( seen_data,
|
|
|
|
data.original_string_,
|
|
|
|
completions.size() );
|
|
|
|
|
|
|
|
if ( index == completions.size() )
|
2012-07-22 18:19:28 -04:00
|
|
|
{
|
2012-08-06 00:01:42 -04:00
|
|
|
completions.push_back( boost::move( data ) );
|
2012-07-22 18:19:28 -04:00
|
|
|
}
|
|
|
|
|
2012-08-02 01:09:31 -04:00
|
|
|
else
|
|
|
|
{
|
2012-08-03 00:37:21 -04:00
|
|
|
// If we have already seen this completion, then this is an overload of a
|
|
|
|
// function we have seen. We add the signature of the overload to the
|
|
|
|
// detailed information.
|
2012-08-02 01:09:31 -04:00
|
|
|
completions[ index ].detailed_info_
|
2012-08-03 00:37:21 -04:00
|
|
|
.append( data.return_type_ )
|
|
|
|
.append( " " )
|
2012-08-06 00:13:01 -04:00
|
|
|
.append( data.everything_except_return_type_ )
|
|
|
|
.append( "\n" );
|
2012-08-02 01:09:31 -04:00
|
|
|
}
|
2012-07-22 18:19:28 -04:00
|
|
|
}
|
|
|
|
|
2012-08-02 01:09:31 -04:00
|
|
|
return completions;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-08-15 22:39:03 -04:00
|
|
|
// NOTE: The passed in pointer should never be NULL!
|
|
|
|
// TODO: move all functions that are not external into an unnamed namespace
|
|
|
|
std::string FullDiagnosticText( CXDiagnostic cxdiagnostic )
|
|
|
|
{
|
|
|
|
std::string full_text = CXStringToString( clang_formatDiagnostic(
|
|
|
|
cxdiagnostic,
|
|
|
|
clang_defaultDiagnosticDisplayOptions() ) );
|
|
|
|
|
|
|
|
// Note: clang docs say that a CXDiagnosticSet retrieved with
|
|
|
|
// clang_getChildDiagnostics do NOT need to be released with
|
|
|
|
// clang_diposeDiagnosticSet
|
|
|
|
CXDiagnosticSet diag_set = clang_getChildDiagnostics( cxdiagnostic );
|
|
|
|
if ( !diag_set )
|
|
|
|
return full_text;
|
|
|
|
|
|
|
|
uint num_child_diagnostics = clang_getNumDiagnosticsInSet( diag_set );
|
|
|
|
if ( !num_child_diagnostics )
|
|
|
|
return full_text;
|
|
|
|
|
|
|
|
for ( uint i = 0; i < num_child_diagnostics; ++i )
|
|
|
|
{
|
|
|
|
CXDiagnostic diagnostic = clang_getDiagnosticInSet( diag_set, i );
|
|
|
|
if ( !diagnostic )
|
|
|
|
continue;
|
|
|
|
|
|
|
|
full_text.append( FullDiagnosticText( diagnostic ) );
|
|
|
|
}
|
|
|
|
|
|
|
|
return full_text;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-08-02 01:09:31 -04:00
|
|
|
Diagnostic CXDiagnosticToDiagnostic( CXDiagnostic cxdiagnostic )
|
|
|
|
{
|
|
|
|
Diagnostic diagnostic;
|
|
|
|
if ( !cxdiagnostic )
|
|
|
|
return diagnostic;
|
|
|
|
|
|
|
|
diagnostic.kind_ = DiagnosticSeverityToType(
|
|
|
|
clang_getDiagnosticSeverity( cxdiagnostic ) );
|
|
|
|
|
|
|
|
// If this is an "ignored" diagnostic, there's no point in continuing since we
|
|
|
|
// won't display those to the user
|
|
|
|
if ( diagnostic.kind_ == 'I' )
|
|
|
|
return diagnostic;
|
|
|
|
|
|
|
|
CXSourceLocation location = clang_getDiagnosticLocation( cxdiagnostic );
|
|
|
|
CXFile file;
|
|
|
|
uint unused_offset;
|
|
|
|
clang_getSpellingLocation( location,
|
|
|
|
&file,
|
|
|
|
&diagnostic.line_number_,
|
|
|
|
&diagnostic.column_number_,
|
|
|
|
&unused_offset );
|
2012-08-15 22:39:03 -04:00
|
|
|
|
2012-08-02 01:09:31 -04:00
|
|
|
diagnostic.filename_ = CXStringToString( clang_getFileName( file ) );
|
|
|
|
diagnostic.text_ = CXStringToString(
|
|
|
|
clang_getDiagnosticSpelling( cxdiagnostic ) );
|
2012-08-15 22:39:03 -04:00
|
|
|
diagnostic.long_formatted_text_ = FullDiagnosticText( cxdiagnostic );
|
2012-08-02 01:09:31 -04:00
|
|
|
|
|
|
|
clang_disposeDiagnostic( cxdiagnostic );
|
|
|
|
return diagnostic;
|
2012-07-22 18:19:28 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace YouCompleteMe
|