2013-01-18 17:22:36 -08: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 "CompilationDatabase.h"
|
|
|
|
#include "ClangUtils.h"
|
|
|
|
#include "standard.h"
|
|
|
|
|
2013-01-22 19:40:05 -08:00
|
|
|
#include <boost/shared_ptr.hpp>
|
2013-01-23 17:23:51 -08:00
|
|
|
#include <boost/bind.hpp>
|
|
|
|
#include <boost/make_shared.hpp>
|
2013-01-22 19:40:05 -08:00
|
|
|
#include <boost/type_traits/remove_pointer.hpp>
|
|
|
|
|
2013-01-23 17:23:51 -08:00
|
|
|
using boost::bind;
|
|
|
|
using boost::make_shared;
|
|
|
|
using boost::packaged_task;
|
2013-01-22 19:40:05 -08:00
|
|
|
using boost::remove_pointer;
|
2013-01-23 17:23:51 -08:00
|
|
|
using boost::shared_ptr;
|
|
|
|
using boost::thread;
|
|
|
|
using boost::unique_future;
|
|
|
|
using boost::function;
|
2013-01-22 19:40:05 -08:00
|
|
|
|
2013-01-19 20:03:32 -08:00
|
|
|
namespace YouCompleteMe {
|
2013-01-23 17:23:51 -08:00
|
|
|
|
2013-01-22 19:42:44 -08:00
|
|
|
typedef shared_ptr <
|
|
|
|
remove_pointer< CXCompileCommands >::type > CompileCommandsWrap;
|
2013-01-18 17:22:36 -08:00
|
|
|
|
2013-01-23 17:23:51 -08:00
|
|
|
|
|
|
|
void QueryThreadMain( CompilationDatabase::InfoTaskStack &info_task_stack ) {
|
|
|
|
while ( true ) {
|
|
|
|
try {
|
|
|
|
( *info_task_stack.Pop() )();
|
|
|
|
} catch ( boost::thread_interrupted & ) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-01-18 17:22:36 -08:00
|
|
|
CompilationDatabase::CompilationDatabase(
|
2013-01-19 20:03:32 -08:00
|
|
|
const std::string &path_to_directory )
|
2013-01-23 17:23:51 -08:00
|
|
|
: threading_enabled_( false ),
|
|
|
|
is_loaded_( false ) {
|
2013-01-18 17:22:36 -08:00
|
|
|
CXCompilationDatabase_Error status;
|
|
|
|
compilation_database_ = clang_CompilationDatabase_fromDirectory(
|
2013-01-19 20:03:32 -08:00
|
|
|
path_to_directory.c_str(),
|
|
|
|
&status );
|
2013-01-18 17:22:36 -08:00
|
|
|
is_loaded_ = status == CXCompilationDatabase_NoError;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-01-19 20:03:32 -08:00
|
|
|
CompilationDatabase::~CompilationDatabase() {
|
2013-01-18 17:22:36 -08:00
|
|
|
clang_CompilationDatabase_dispose( compilation_database_ );
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-01-23 17:23:51 -08:00
|
|
|
// We need this mostly so that we can not use it in tests. Apparently the
|
|
|
|
// GoogleTest framework goes apeshit on us if we enable threads by default.
|
|
|
|
void CompilationDatabase::EnableThreading() {
|
|
|
|
threading_enabled_ = true;
|
|
|
|
InitThreads();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-01-19 20:03:32 -08:00
|
|
|
bool CompilationDatabase::DatabaseSuccessfullyLoaded() {
|
2013-01-18 17:22:36 -08:00
|
|
|
return is_loaded_;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-01-23 17:23:51 -08:00
|
|
|
CompilationInfoForFile CompilationDatabase::GetCompilationInfoForFile(
|
2013-02-02 21:49:55 -08:00
|
|
|
const std::string &path_to_file ) {
|
2013-01-23 17:23:51 -08:00
|
|
|
CompilationInfoForFile info;
|
2013-01-19 20:03:32 -08:00
|
|
|
|
2013-01-18 17:22:36 -08:00
|
|
|
if ( !is_loaded_ )
|
2013-01-23 17:23:51 -08:00
|
|
|
return info;
|
|
|
|
|
|
|
|
// TODO: mutex protect calls to getCompileCommands and getDirectory
|
2013-01-18 17:22:36 -08:00
|
|
|
|
2013-01-22 19:40:05 -08:00
|
|
|
CompileCommandsWrap commands(
|
2013-01-19 20:03:32 -08:00
|
|
|
clang_CompilationDatabase_getCompileCommands(
|
|
|
|
compilation_database_,
|
2013-01-22 19:40:05 -08:00
|
|
|
path_to_file.c_str() ), clang_CompileCommands_dispose );
|
2013-01-18 17:22:36 -08:00
|
|
|
|
2013-01-22 19:40:05 -08:00
|
|
|
uint num_commands = clang_CompileCommands_getSize( commands.get() );
|
2013-01-19 20:03:32 -08:00
|
|
|
|
2013-01-18 17:22:36 -08:00
|
|
|
if ( num_commands < 1 ) {
|
2013-01-23 17:23:51 -08:00
|
|
|
return info;
|
2013-01-18 17:22:36 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
// We always pick the first command offered
|
|
|
|
CXCompileCommand command = clang_CompileCommands_getCommand(
|
2013-01-22 19:40:05 -08:00
|
|
|
commands.get(),
|
2013-01-19 20:03:32 -08:00
|
|
|
0 );
|
2013-01-18 17:22:36 -08:00
|
|
|
|
2013-01-23 17:23:51 -08:00
|
|
|
info.compiler_working_dir_ = CXStringToString(
|
2013-02-02 21:49:55 -08:00
|
|
|
clang_CompileCommand_getDirectory( command ) );
|
2013-01-23 17:23:51 -08:00
|
|
|
|
2013-01-18 17:22:36 -08:00
|
|
|
uint num_flags = clang_CompileCommand_getNumArgs( command );
|
2013-01-23 17:23:51 -08:00
|
|
|
info.compiler_flags_.reserve( num_flags );
|
2013-01-19 20:03:32 -08:00
|
|
|
|
|
|
|
for ( uint i = 0; i < num_flags; ++i ) {
|
2013-01-23 17:23:51 -08:00
|
|
|
info.compiler_flags_.push_back(
|
2013-02-02 21:49:55 -08:00
|
|
|
CXStringToString( clang_CompileCommand_getArg( command, i ) ) );
|
2013-01-18 17:22:36 -08:00
|
|
|
}
|
|
|
|
|
2013-01-23 17:23:51 -08:00
|
|
|
return info;
|
2013-01-18 17:22:36 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-01-23 17:23:51 -08:00
|
|
|
Future< AsyncCompilationInfoForFile >
|
|
|
|
CompilationDatabase::GetCompilationInfoForFileAsync(
|
2013-02-02 21:49:55 -08:00
|
|
|
const std::string &path_to_file ) {
|
2013-01-23 17:23:51 -08:00
|
|
|
// TODO: throw exception when threading is not enabled and this is called
|
|
|
|
if ( !threading_enabled_ )
|
|
|
|
return Future< AsyncCompilationInfoForFile >();
|
2013-01-19 20:03:32 -08:00
|
|
|
|
2013-01-23 17:23:51 -08:00
|
|
|
function< CompilationInfoForFile() > functor =
|
2013-04-29 21:41:49 +04:00
|
|
|
boost::bind( &CompilationDatabase::GetCompilationInfoForFile,
|
2013-01-23 17:23:51 -08:00
|
|
|
boost::ref( *this ),
|
|
|
|
path_to_file );
|
2013-01-18 17:22:36 -08:00
|
|
|
|
2013-01-23 17:23:51 -08:00
|
|
|
InfoTask task =
|
2013-02-02 21:49:55 -08:00
|
|
|
make_shared< packaged_task< AsyncCompilationInfoForFile > >(
|
|
|
|
bind( ReturnValueAsShared< CompilationInfoForFile >,
|
|
|
|
functor ) );
|
2013-01-19 20:03:32 -08:00
|
|
|
|
2013-01-23 17:23:51 -08:00
|
|
|
unique_future< AsyncCompilationInfoForFile > future = task->get_future();
|
|
|
|
info_task_stack_.Push( task );
|
|
|
|
return Future< AsyncCompilationInfoForFile >( boost::move( future ) );
|
|
|
|
}
|
2013-01-18 17:22:36 -08:00
|
|
|
|
|
|
|
|
2013-01-23 17:23:51 -08:00
|
|
|
void CompilationDatabase::InitThreads() {
|
|
|
|
info_thread_ = boost::thread( QueryThreadMain,
|
|
|
|
boost::ref( info_task_stack_ ) );
|
2013-01-18 17:22:36 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace YouCompleteMe
|
|
|
|
|