#!/usr/bin/perl # # Copyright (C) 2001-2021 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 $cdb = new CompilationDatabase( $src_dir , {WINDOWS=>0,...} , {top_srcdir=>'..'} , {} ) ; # my @files = $cdb->list() ; # my @stanzas = $cdb->stanzas() ; # $cdb->print() ; # use strict ; use File::Basename ; use AutoMakeParser ; package CompilationDatabase ; our $debug = 0 ; sub new { 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 %me = ( m_base_dir => $base_makefile_dir , m_switches => $switches , m_ro_vars => $ro_vars , m_config => $config , ) ; return bless \%me , $classname ; } sub list { my ( $this ) = @_ ; my @list = () ; my $verbose = $debug ; my @makefiles = AutoMakeParser::readall( $this->{m_base_dir} , $this->{m_switches} , $this->{m_ro_vars} , $verbose ) ; for my $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 { my ( $this ) = @_ ; print "[\n" ; print join( ",\n" , $this->stanzas() ) ; print "]\n" ; } sub stanzas { my ( $this ) = @_ ; my @makefiles = AutoMakeParser::readall( $this->{m_base_dir} , $this->{m_switches} , $this->{m_ro_vars} ) ; my @output = () ; for my $m ( @makefiles ) { my $dir = File::Basename::dirname( $m->path() ) ; my @includes = map { "-I$_" } $m->includes( $m->top() , 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: top=",$m->top(),"\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 ;