#!/usr/bin/perl # # Copyright (C) 2001-2022 Graeme Walker # # 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 . # === # # 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 ;