#!/usr/bin/perl # IBM_PROLOG_BEGIN_TAG # This is an automatically generated prolog. # # $Source: src/build/tools/addCopyright.pl $ # # OpenPOWER HostBoot Project # # COPYRIGHT International Business Machines Corp. 2011,2014 # # Licensed under the Apache License, Version 2.0 (the "License"); # you may not use this file except in compliance with the License. # You may obtain a copy of the License at # # http://www.apache.org/licenses/LICENSE-2.0 # # Unless required by applicable law or agreed to in writing, software # distributed under the License is distributed on an "AS IS" BASIS, # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express # or implied. See the License for the specific language governing # permissions and limitations under the License. # # IBM_PROLOG_END_TAG #################################### ABOUT #################################### # Forked from: # # Author: Mark Jerde (mjerde@us.ibm.com) # # Date: Fri Mar 19 17:40:32 2010 UTC # # # # addCopyright.pl will automatically insert appropriate copyright statements # # in source files following the IBM copyright guidelines (and templates) # # referenced below : # # FSP ClearCase Architecture # # Version 1.9 # # 10/12/2010 # # Editor: Alan Hlava # # # # Section 3.14.1 of the above doc has templates for different files # # # # NOTE: FSP uses the phrase "OCO Source materials" in their copyright # # block, which is classified as 'p1' . We will use the same # # classification here. # # NOTE: to list all files in src EXCEPT the build dir, run: # # make clean # remove autogenerated files # # find src -path 'src/build' -prune -o ! -type d -print | tr '\n' ' ' # # # # addCopyright.pl does not support piping, but you can send these # # to a file, add "addCopyright.pl update" to the beginning of the line, # # and run the file to update all # ############################################################################### use strict; use warnings; use POSIX; use Getopt::Long; use File::Basename; use lib dirname (__FILE__); #------------------------------------------------------------------------------ # Project-specific settings. #------------------------------------------------------------------------------ my $ReleaseYear = `date +%Y`; chomp( $ReleaseYear ); $ReleaseYear = $ENV{'DATE_OVERRIDE'} if defined $ENV{'DATE_OVERRIDE'}; my $copyrightSymbol = ""; # my $copyrightSymbol = "(C)"; # Uncomment if unable to use  character. my $projectName = "HostBoot"; my $copyright_IBM = 'COPYRIGHT International Business Machines Corp'; ## note that these use single ticks so that the escape chars are NOT evaluated yet. my $OLD_DELIMITER_END = 'IBM_PROLOG_END'; my $DELIMITER_END = 'IBM_PROLOG_END_TAG'; my $DELIMITER_BEGIN = 'IBM_PROLOG_BEGIN_TAG'; my $SOURCE_BEGIN_TAG = "\$Source:"; my $SOURCE_END_TAG = "\$"; # End Project-specific settings #------------------------------------------------------------------------------ # Constants #------------------------------------------------------------------------------ use constant RC_INVALID_PARAMETERS => 1; use constant RC_OLD_COPYRIGHT_BLOCK => 2; use constant RC_NO_COPYRIGHT_BLOCK => 3; use constant RC_INVALID_SOURCE_LINE => 4; use constant RC_INVALID_YEAR => 5; use constant RC_OLD_DELIMITER_END => 6; use constant RC_NOT_HOSTBOOT_BLOCK => 7; use constant RC_NO_COPYRIGHT_STRING => 8; use constant RC_INVALID_FILETYPE => 9; #------------------------------------------------------------------------------ # Global Vars #------------------------------------------------------------------------------ my $opt_help = 0; my $opt_debug = 0; my $operation = ""; my $opt_logfile = ""; my $DelimiterBegin = ""; my $CopyrightBlock = ""; my $DelimiterEnd = ""; my $CopyRightString = ""; my $TempFile = ""; my @Files = (); my $rc = 0; # NOTE: $OLD_DELIMITER_END is a subset of $DELIMITER_END so must match # $DELIMITER_END first in order to return the entire string. my $g_end_del_re = "($DELIMITER_END|$OLD_DELIMITER_END)"; my $g_prolog_re = "($DELIMITER_BEGIN)((.|\n)+?)$g_end_del_re"; #------------------------------------------------------------------------------ # Forward Declaration #------------------------------------------------------------------------------ sub validate( $ ); sub update( $$ ); sub extractCopyrightBlock( $ ); sub checkCopyrightblock( $$ ); sub createYearString( $ ); sub removeCopyrightBlock( $$ ); sub addEmptyCopyrightBlock( $$$ ); sub fillinCopyrightBlock( $$ ); ####################################################################### # Main ####################################################################### if (scalar(@ARGV) < 2) { ## needs at least one filename and an operation as a parameter usage(); } my @SaveArgV = @ARGV; #------------------------------------------------------------------------------ # Parse optional input arguments #------------------------------------------------------------------------------ GetOptions( "help|?" => \$opt_help, "validate" => sub { $operation="validate"; }, "update" => sub { $operation="update"; }, "log-failed-files=s" => \$opt_logfile, "debug" => \$opt_debug, ); ## scan through remaining args and store all files in @Files ## check for old-type parms, just in case foreach ( @ARGV ) { ## print $_; if ( m/^debug$/ ) { $opt_debug = 1; next; } if ( m/^update$/ ) { $operation = $_; next; } if ( m/^validate$/ ) { $operation = $_; next; } push @Files, $_ ; } if ( $opt_debug ) { print STDERR __LINE__, " : ---- DEBUG -----\n"; print STDERR "help = $opt_help\n"; print STDERR "debug = $opt_debug\n"; print STDERR "operation = $operation\n"; print STDERR "log-failed-files = $opt_logfile\n"; ## dump files specified print STDERR "Files:\n"; print STDERR join( ' ', @Files ), "\n"; print STDERR "ReleaseYear = $ReleaseYear\n"; print "\n"; } if ( $operation eq "" ) { print STDOUT "No operation specified\n"; usage(); exit RC_INVALID_PARAMETERS; } if ( ( $opt_logfile ne "" ) ## && ( $operation eq "validate" ) ) { my $logdate = `date +%Y-%m-%d:%H%M`; chomp $logdate; open( LOGFH, "> $opt_logfile" ) or die "ERROR $?: Failed to open $opt_logfile: $!"; print LOGFH "## logfile generated $logdate from command line:\n"; print LOGFH $0, " ", join( ' ', @SaveArgV ); print LOGFH "\nFAILING files:\n"; } ######################################################################## ## MAIN ######################################################################## # Loop through all files and process. foreach ( @Files ) { ## clear global vars $DelimiterBegin = ""; $CopyrightBlock = ""; $DelimiterEnd = ""; $CopyRightString = ""; $rc = 0; ## get filetype my $filetype = filetype($_); print STDOUT "File $_: Type $filetype\n"; ## set Temporary file name. $TempFile = "$_.gitCPYWRT"; if ( $opt_debug ) { print STDERR __LINE__, ": Temporary file name = $TempFile\n"; } ## ## Special case is this file, just return 0 and add copyright manually. ## if ( m/addCopyright\.pl/ ) { print STDOUT "---------------------------------------------------------\n"; print STDOUT "Skipping special case file: $_\n"; print STDOUT " Please add the copyright prolog manually.\n"; print STDOUT "---------------------------------------------------------\n"; next; } ## ## Gerrit submissions can include deleted files, just warn and continue if ( ! -e $_ ) { print STDOUT "---------------------------------------------------------\n"; print STDOUT "Skipping deleted file: $_\n"; print STDOUT "---------------------------------------------------------\n"; next; } ## ## Unknown files are valid, but should generate a warning. if ("Unknown" eq $filetype) { print STDOUT "!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!\n"; print STDOUT "WARNING:: File $_ :Unknown Filetype: $filetype\n"; print STDOUT " Skipping this file and continuing.\n"; print STDOUT " Please add the copyright prolog manually.\n"; print STDOUT "!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!\n"; next; } ## ## text files are valid, but should generate a warning. if (("txt" eq $filetype) || "Initfile" eq $filetype) { print STDOUT "!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!\n"; print STDOUT "WARNING:: File $_ : Filetype: $filetype\n"; print STDOUT " Skipping this file and continuing.\n"; print STDOUT " If needed, Please add the copyright \n"; print STDOUT " prolog manually.\n"; print STDOUT "!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!\n"; next; } ## extract the existing copyright block ( $DelimiterBegin, $CopyrightBlock, $DelimiterEnd ) = extractCopyrightBlock( $_ ); ## ## validate the file. ## if $logfile exists, print failing filename to $logfile and ## keep going. ## Otherwise, exit with $rc ## if ( $operation =~ m/validate/i ) { $rc = validate( $_ ); if ( $rc == RC_OLD_DELIMITER_END ) { ## special case, let it go for now print STDOUT "$_ has old prolog end '$OLD_DELIMITER_END', please fix\n"; } elsif ( $rc ) { print STDOUT "$_ FAILED copyright validation: $rc \n"; if ( $opt_logfile ne "" ) { print LOGFH "$_ # FAILED $rc \n"; } else { exit $rc; } } ## continue to next file next; } ## endif validate ## ## update ## if ($operation =~ m/update/i) { $rc = update( $_, $filetype ); if ( $rc ) { print STDOUT "$_ FAILED copyright update: $rc \n"; exit $rc; } ## continue to next file next; } ## endif update } # end foreach if ( $opt_logfile ne "" ) { close( LOGFH ); } ######################################################################### ## Subroutines ######################################################################### ####################################### ## usage: print usage and quit ####################################### sub usage { print STDOUT "Usage: addCopyright.pl { update | validate } \n"; print STDOUT " [ --log-failed-files ]\n"; print STDOUT " [ --debug ] \n"; print STDOUT " file1 file2 ...\n"; } ####################################### ## validate the file ## param[in] $filename to validate ## returns 0 success, nonzero failure ## See constants above for values of failure ####################################### sub validate( $ ) { my ( $filename ) = @_; my $rc = 0; if ( $CopyrightBlock eq "" ) { print STDOUT "WARNING: No copyright block.\n"; return RC_NO_COPYRIGHT_BLOCK; } $rc = checkCopyrightBlock( $CopyrightBlock, $filename ); # good file return $rc; } ## ## @sub update the copyright block. ## ## @param[in] filename ## @param[in] filetype ## ## @return success or failure (currently only return success) ## sub update( $$ ) { my ( $filename, $filetype ) = @_; my $olddelimiter = 0; my $localrc = 0; $localrc = validate( $filename ); if ( $localrc != 0 ) { print STDOUT "Copyright Block check returned $localrc , fixing...\n"; if ( $localrc != RC_NO_COPYRIGHT_BLOCK ) { if ( $opt_debug) { print STDERR __LINE__, ": remove old copyright block...\n"; } removeCopyrightBlock( $filename, $filetype ); } if ($opt_debug) { print STDERR __LINE__, ": Add empty copyright block...\n"; } addEmptyCopyrightBlock( $filename, $filetype, $localrc ); if ( $opt_debug ) { print STDERR __LINE__, ": fill in new copyright block...\n"; } fillinEmptyCopyrightBlock( $filename, $filetype ); } ## return OK by default. return 0; } ##################################### ## Analyze file and return a text string of the file type ##################################### sub filetype { my $filename = shift; my $fileinfo = `file $filename | sed 's/^.*: //'`; chomp $fileinfo; # Sorted by anticipated frequency of occurrence. if ( $filename =~ m/\.xml$/i ) # Added XML file to the top of the list because some comments in # an XML file cause older versions of 'file' to incorrectly return # "ASCII C++ program text" even though the file is obviously XML. # Specifically we are seeing "' ); } elsif ( "Assembly" eq $filetype ) { $data = removeProlog( $data, '\#', '' ); } elsif ( ("Autoconf" eq $filetype) or ("Automake" eq $filetype) or ("CVS" eq $filetype) or ("Makefile" eq $filetype) or ("Perl" eq $filetype) or ("PrdRuleFile" eq $filetype) or ("Python" eq $filetype) or ("Shellscript" eq $filetype) or ("Tcl" eq $filetype) ) { # Don't wipe the the '#!' line at the top. $data = removeProlog( $data, '\#', '' ); } else { print STDOUT "ERROR: Don't know how to remove old block from $filetype file.\n"; close OUTPUT; return RC_INVALID_FILETYPE; } print OUTPUT $data; ## finish up the files close( OUTPUT ) or die " $? can't close $TempFile: $!" ; rename( $filename, "$savedbgfile" ) or die " $? can't rename $filename: $!" ; rename( $TempFile, $filename ) or die " $? can't rename $TempFile: $!" ; if ( !$opt_debug ) { ## leave the files around for debug unlink( $savedbgfile ) or die " $? can't delete $savedbgfile: $!"; } } ################################### ## Add an empty copyright block to the file, for example (C/C++ files): ## ## // IBM_PROLOG_BEGIN_TAG IBM_PROLOG_END_TAG ## ## - The block will be filled-in in the next step. ## ## @param[in] - filename to modify ## @param[in] - filetype ## @param[in] - returncode from validate ## ## @return none ## ## - Makes up a debug file called ".empty" ################################## sub addEmptyCopyrightBlock( $$$ ) { my ( $filename, $filetype, $validaterc ) = @_; my $line; ## Modify file in place with temp file Perl Cookbook 7.8 my $savedbgfile = "$filename.empty"; system( "cp -p $filename $TempFile" ) ; ## preserve permissions open( INPUT, "< $filename" ) or die " $? can't open $filename: $!" ; open( OUTPUT, "> $TempFile" ) or die " $? can't open $TempFile: $!" ; select( OUTPUT ); ## new default filehandle for print if ("Assembly" eq $filetype) { print OUTPUT "# $DELIMITER_BEGIN $DELIMITER_END\n"; } elsif ( ("Makefile" eq $filetype) or ("PrdRuleFile" eq $filetype) ) { print OUTPUT "# $DELIMITER_BEGIN $DELIMITER_END\n"; } elsif (("Autoconf" eq $filetype) or ("Automake" eq $filetype) or ("CVS" eq $filetype) or ("Perl" eq $filetype) or ("Python" eq $filetype) or ("Shellscript" eq $filetype) or ("Tcl" eq $filetype)) { ## All files with a "shebang" at the beginning $line = ; # Keep the '#!' line at the top. ## The following says : if the first line is a "shebang" line ## (e.g. "#!/usr/bin/perl"), then print it _before_ the copyright ## block, otherwise (the unless line), print it _after_ the copyright ## block. if ($line =~ m/^#!/) { print OUTPUT $line; } print OUTPUT "# $DELIMITER_BEGIN $DELIMITER_END\n"; unless ($line =~ m/^#!/) { print OUTPUT $line; } } elsif ( "C" eq $filetype ) { print OUTPUT "/* $DELIMITER_BEGIN $DELIMITER_END */\n"; } elsif ( ("RPC" eq $filetype) or ("LinkerScript" eq $filetype) ) { # ld stubbornly refuses to use modern comment lines. print OUTPUT "/* $DELIMITER_BEGIN $DELIMITER_END */\n"; } elsif ("MofFile" eq $filetype) { print OUTPUT "// $DELIMITER_BEGIN $DELIMITER_END\n"; } elsif ("xml" eq $filetype ) { print OUTPUT "\n"; } else { print STDOUT "ERROR: Can\'t handle filetype: $filetype\n"; close( INPUT ) or die " $? can't close $filename: $!" ; close( OUTPUT ) or die " $? can't close $TempFile: $!" ; ## leave $Tempfile around for debug return RC_INVALID_FILETYPE; } # Copy rest of file while (defined($line = )) { print OUTPUT $line; } ## finish up the files close( INPUT ) or die " $? can't close $filename: $!" ; close( OUTPUT ) or die " $? can't close $TempFile: $!" ; rename( $filename, "$savedbgfile" ) or die " $? can't rename $filename: $!" ; rename( $TempFile, $filename ) or die " $? can't rename $TempFile: $!" ; if ( !$opt_debug ) { ## leave the files around for debug unlink( $savedbgfile ) or die " $? can't delete $savedbgfile: $!"; } } ############################################ ## Helper functions for fillinEmptyCopyrightBlock() ############################################ sub addPrologComments($$$) { my ( $data, $begin, $end ) = @_; my @lines = split( /\n/, $data ); $data = ''; for my $line ( @lines ) { # If there is an block comment end tag, fill the end of the line with # spaces. if ( $end ) { my $max_line_len = 70; my $len = length($line); if ( $len < $max_line_len ) { my $fill = ' ' x ($max_line_len - $len); $line .= $fill; } } # NOTE: CMVC prologs with inline comments will have a single trailing # space at the end of the line. This is undesirable for most # hostboot developers so it will not be added. if ( $line =~ m/$DELIMITER_BEGIN/ ) { $line = "$line $end" if ( $end ); $line = "$line\n"; } elsif ( $line =~ m/$DELIMITER_END/ ) { $line = "$begin $line"; } else { if ( not $end and not $line ) { # Compensate for blank lines with no end delimeter. $line = "$begin\n"; } else { $line = "$begin $line"; $line = "$line $end" if ( $end ); $line = "$line\n"; } } $data .= $line; } return $data; } ############################################ ## fill in the empty copyright block ## Copyright guidelines from: ## FSP ClearCase Architecture ## Version 1.9 ## 10/12/2010 ## Editor: Alan Hlava ## ## Section 3.14.1 has templates for different files ## ############################################ sub fillinEmptyCopyrightBlock( $$ ) { my ( $filename, $filetype ) = @_; my $copyrightYear = createYearString( $filename ); ## define the final copyright block template here. my $IBMCopyrightBlock = < $TempFile" ) or die " $? can't open $TempFile: $!" ; select( OUTPUT ); ## new default filehandle for print my $newline; my $lines = ""; while ( defined($newline = ) ) { $lines .= $newline; } if ("Assembly" eq $filetype) { $IBMCopyrightBlock = addPrologComments($IBMCopyrightBlock, '#', ''); } elsif ( ("Makefile" eq $filetype) or ("PrdRuleFile" eq $filetype) ) { $IBMCopyrightBlock = addPrologComments($IBMCopyrightBlock, '#', ''); } elsif (("Autoconf" eq $filetype) or ("Automake" eq $filetype) or ("CVS" eq $filetype) or ("Perl" eq $filetype) or ("Python" eq $filetype) or ("Shellscript" eq $filetype) or ("Tcl" eq $filetype)) { ## all files with a "shebang" $IBMCopyrightBlock = addPrologComments($IBMCopyrightBlock, '#', ''); } elsif ( "C" eq $filetype ) { ## lex files are classified as C, but do not recognize '//' comments $IBMCopyrightBlock = addPrologComments($IBMCopyrightBlock, '/*', '*/'); } elsif ( ("RPC" eq $filetype) or ("LinkerScript" eq $filetype) ) { $IBMCopyrightBlock = addPrologComments($IBMCopyrightBlock, '/*', '*/'); } elsif ("EmxFile" eq $filetype) { # Not yet formatted correctly for EmxFile needs, but should coexist. $IBMCopyrightBlock = "$DELIMITER_BEGIN IBM Confidential OCO Source Materials (C) Copyright IBM Corp. $copyrightYear The source code for this program is not published or otherwise divested of its trade secrets, irrespective of what has been deposited with the U.S. Copyright Office. $DELIMITER_END"; } elsif ("MofFile" eq $filetype) { $IBMCopyrightBlock = addPrologComments($IBMCopyrightBlock, '//', ''); } elsif ( "xml" eq $filetype) { $IBMCopyrightBlock = addPrologComments($IBMCopyrightBlock, ''); } else { print STDOUT "ERROR: Can\'t handle filetype: $filetype\n"; return RC_INVALID_FILETYPE; } # Replace existing block with the current content. $lines =~ s/$DELIMITER_BEGIN[^\$]*$DELIMITER_END/$IBMCopyrightBlock/s; print OUTPUT $lines; ## finish up the files close( INPUT ) or die " $? can't close $filename: $!" ; close( OUTPUT ) or die " $? can't close $TempFile: $!" ; rename( $filename, "$savedbgfile" ) or die " $? can't rename $filename: $!" ; rename( $TempFile, $filename ) or die " $? can't rename $TempFile: $!" ; if ( !$opt_debug ) { ## leave the files around for debug unlink( $savedbgfile ) or die " $? can't delete $savedbgfile: $!"; } }