emailrelay/bin/CompilationDatabase.pm
Graeme Walker 6a32f90311 v2.4
2022-11-01 12:00:00 +00:00

176 lines
4.9 KiB
Perl

#!/usr/bin/perl
#
# Copyright (C) 2001-2022 Graeme Walker <graeme_walker@users.sourceforge.net>
#
# This program 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.
#
# This program 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 this program. If not, see <http://www.gnu.org/licenses/>.
# ===
#
# CompilationDatabase.pm
#
# Parses automake files throughout a source tree using AutoMakeParser
# and generates a "compilation database" file, "compile_commands.json".
#
# See also: https://clang.llvm.org/docs/JSONCompilationDatabase.html
#
# Synopsis:
#
# use CompilationDatabase ;
# my @makefiles = AutoMakeParser::readall( ... ) ;
# my $cdb = new CompilationDatabase( \@makefiles , {full_paths=>1} ) ;
# my $cdb = new CompilationDatabase( $src_dir , {WINDOWS=>0,...} , {top_srcdir=>'..'} , {full_paths=>1} ) ;
# my @files = $cdb->list() ;
# my @stanzas = $cdb->stanzas() ;
# $cdb->print() ;
#
use strict ;
use File::Basename ;
use AutoMakeParser ;
package CompilationDatabase ;
our $debug = 0 ;
sub new
{
if( ref($_[1]) )
{
# Parses a set of makefiles as given by an array of Makefile
# objects obtained from AutoMakeParser::readall().
#
my ( $classname , $makefiles_ref , $config ) = @_ ;
$config ||= {} ;
$config->{test_mode} ||= 0 ;
$config->{full_paths} ||= 0 ;
my %me = (
m_makefiles => $makefiles_ref ,
m_config => $config ,
) ;
return bless \%me , $classname ;
}
else
{
# Finds makefiles under the given base directory and parses
# them. The switches and read-only expansion variables can
# be hard-coded or extracted from a config.status file
# (see ConfigStatus).
#
my ( $classname , $base_makefile_dir , $switches , $ro_vars , $config ) = @_ ;
$AutoMakeParser::debug = 1 if $debug > 1 ;
$config ||= {} ;
$config->{test_mode} ||= 0 ;
$config->{full_paths} ||= 0 ;
my @makefiles = AutoMakeParser::readall( $base_makefile_dir , $switches , $ro_vars ) ;
my %me = (
m_makefiles => \@makefiles ,
m_config => $config ,
) ;
return bless \%me , $classname ;
}
}
sub list
{
# Returns a list of all the source files in all the makefiles found under base-dir.
my ( $this ) = @_ ;
my @list = () ;
my $verbose = $debug ;
for my $m ( @{$this->{m_makefiles}} )
{
my $sub_dir = File::Basename::dirname( $m->path() ) ;
for my $library ( $m->libraries() )
{
push @list , map { "$sub_dir/$_" } $m->sources($library) ;
}
for my $program ( $m->programs() )
{
push @list , map { "$sub_dir/$_" } $m->sources($program) ;
}
}
return @list ;
}
sub print
{
# Prints the complete compilation database json structure to stdout.
my ( $this ) = @_ ;
print "[\n" ;
print join( ",\n" , $this->stanzas() ) ;
print "]\n" ;
}
sub stanzas
{
# Returns a list of separate compilation database stanzas for all the source files
# in all the makefiles found under base-dir.
my ( $this ) = @_ ;
my @output = () ;
for my $m ( @{$this->{m_makefiles}} )
{
my $dir = File::Basename::dirname( $m->path() ) ;
my @includes = map { "-I$_" } $m->includes( $m->base() , undef , undef , $this->{m_config}->{full_paths} ) ;
my @definitions = map { "-D$_" } $m->definitions() ;
my @compile_options = $m->compile_options() ;
my @link_options = $m->link_options() ;
if( $debug )
{
print "cdb: makefile=" , $m->path() , "\n" ;
print "cdb: base=",$m->base(),"\n" ;
print "cdb: \@includes=" , join("|",@includes) , "\n" ;
print "cdb: \@definitions=" , join("|",@definitions) , "\n" ;
print "cdb: \@compile_options=" , join("|",@compile_options) , "\n" ;
print "cdb: \@link_options=" , join("|",@link_options) , "\n" ;
}
for my $library ( $m->libraries() )
{
map { push @output , $this->_stanza($dir,$_,\@includes,\@definitions,\@compile_options) } $m->sources($library) ;
}
for my $program ( $m->programs() )
{
map { push @output , $this->_stanza($dir,$_,\@includes,\@definitions,\@compile_options) } $m->sources($program) ;
}
}
return @output ;
}
sub _stanza
{
my ( $this , $dir , $source , $includes , $definitions , $options ) = @_ ;
my @all_flags = ( @$definitions , @$includes , @$options ) ;
my $directory = Cwd::realpath( $dir ) ;
$source = join("/",$directory,$source) if $this->{m_config}->{full_paths} ;
my $command = "/usr/bin/c++ @all_flags -c $source" ;
$command =~ s/"/\\"/g unless $this->{m_config}->{test_mode} ;
if( $this->{m_config}->{test_mode} )
{
return "cd $directory && $command\n" ;
}
return
"{\n" .
" \"directory\" : \"$directory\" ,\n" .
" \"command\" : \"$command\",\n" .
" \"file\" : \"$source\" ,\n" .
"}" ;
}
1 ;