Edit file File name : subs.pl Content :#! /usr/bin/perl # grog - guess options for groff command # Inspired by doctype script in Kernighan & Pike, Unix Programming # Environment, pp 306-8. # Source file position: <groff-source>/src/roff/grog/subs.pl # Installed position: <prefix>/lib/grog/subs.pl # Copyright (C) 1993-2018 Free Software Foundation, Inc. # This file was split from grog.pl and put under GPL2 by # Bernd Warken <groff-bernd.warken-72@web.de>. # The macros for identifying the devices were taken from Ralph # Corderoy's 'grog.sh' of 2006. # Last update: 10 Sep 2015 # This file is part of 'grog', which is part of 'groff'. # 'groff' 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 2 of the License, or # (at your option) any later version. # 'groff' 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 can get the license text for the GNU General Public License # version 2 in the internet at # <http://www.gnu.org/licenses/gpl-2.0.html>. ######################################################################## require v5.6; use warnings; use strict; use File::Spec; # printing of hashes: my %hash = ...; print Dumper(\%hash); use Data::Dumper; # for running shell based programs within Perl; use `` instead of # use IPC::System::Simple qw(capture capturex run runx system systemx); $\ = "\n"; # my $Sp = "[\\s\\n]"; # my $Sp = qr([\s\n]); # my $Sp = '' if $arg eq '-C'; my $Sp = ''; # from 'src/roff/groff/groff.cpp' near 'getopt_long' my $groff_opts = 'abcCd:D:eEf:F:gGhiI:jJkK:lL:m:M:n:No:pP:r:RsStT:UvVw:W:XzZ'; my @Command = (); # stores the final output my @Mparams = (); # stores the options '-m*' my @devices = (); # stores -T my $do_run = 0; # run generated 'groff' command my $pdf_with_ligatures = 0; # '-P-y -PU' for 'pdf' device my $with_warnings = 0; my $Prog = $0; { my ($v, $d, $f) = File::Spec->splitpath($Prog); $Prog = $f; } my %macros; my %Groff = ( # preprocessors 'chem' => 0, 'eqn' => 0, 'gperl' => 0, 'grap' => 0, 'grn' => 0, 'gideal' => 0, 'gpinyin' => 0, 'lilypond' => 0, 'pic' => 0, 'PS' => 0, # opening for pic 'PF' => 0, # alternative opening for pic 'PE' => 0, # closing for pic 'refer' => 0, 'refer_open' => 0, 'refer_close' => 0, 'soelim' => 0, 'tbl' => 0, # tmacs # 'man' => 0, # 'mandoc' => 0, # 'mdoc' => 0, # 'mdoc_old' => 0, # 'me' => 0, # 'mm' => 0, # 'mom' => 0, # 'ms' => 0, # requests 'AB' => 0, # ms 'AE' => 0, # ms 'AI' => 0, # ms 'AU' => 0, # ms 'NH' => 0, # ms 'TH_later' => 0, # TH not 1st command is ms 'TL' => 0, # ms 'UL' => 0, # ms 'XP' => 0, # ms 'IP' => 0, # man and ms 'LP' => 0, # man and ms 'P' => 0, # man and ms 'PP' => 0, # man and ms 'SH' => 0, # man and ms 'OP' => 0, # man 'SS' => 0, # man 'SY' => 0, # man 'TH_first' => 0, # TH as 1st command is man 'TP' => 0, # man 'UR' => 0, # man 'YS' => 0, # man # for mdoc and mdoc-old # .Oo and .Oc for modern mdoc, only .Oo for mdoc-old 'Oo' => 0, # mdoc and mdoc-old 'Oc' => 0, # mdoc 'Dd' => 0, # mdoc ); # end of %Groff # for first line check my %preprocs_tmacs = ( 'chem' => 0, 'eqn' => 0, 'gideal' => 0, 'gpinyin' => 0, 'grap' => 0, 'grn' => 0, 'pic' => 0, 'refer' => 0, 'soelim' => 0, 'tbl' => 0, 'geqn' => 0, 'gpic' => 0, 'neqn' => 0, 'man' => 0, 'mandoc' => 0, 'mdoc' => 0, 'mdoc-old' => 0, 'me' => 0, 'mm' => 0, 'mom' => 0, 'ms' => 0, ); my @filespec; my $tmac_ext = ''; ######################################################################## # err() ######################################################################## sub err { my $text = shift; print STDERR $text; } ######################################################################## # handle_args() ######################################################################## sub handle_args { my $double_minus = 0; my $was_minus = 0; my $was_T = 0; my $optarg = 0; # globals: @filespec, @Command, @devices, @Mparams foreach my $arg (@ARGV) { if ( $optarg ) { push @Command, $arg; $optarg = 0; next; } if ( $double_minus ) { if (-f $arg && -r $arg) { push @filespec, $arg; } else { print STDERR __FILE__ . ' ' . __LINE__ . ': ' . "grog: $arg is not a readable file."; } next; } if ( $was_T ) { push @devices, $arg; $was_T = 0; next; } ####### handle_args() unless ( $arg =~ /^-/ ) { # file name, no opt, no optarg unless (-f $arg && -r $arg) { print 'unknown file name: ' . $arg; } push @filespec, $arg; next; } # now $arg starts with '-' if ($arg eq '-') { unless ($was_minus) { push @filespec, $arg; $was_minus = 1; } next; } if ($arg eq '--') { $double_minus = 1; push(@filespec, $arg); next; } &version() if $arg =~ /^--?v/; # --version, with exit &help() if $arg =~ /--?h/; # --help, with exit if ( $arg =~ /^--r/ ) { # --run, no exit $do_run = 1; next; } if ( $arg =~ /^--wa/ ) { # --warnings, no exit $with_warnings = 1; next; } ####### handle_args() if ( $arg =~ /^--(wi|l)/ ) { # --ligatures, no exit # the old --with_ligatures is only kept for compatibility $pdf_with_ligatures = 1; next; } if ($arg =~ /^-m/) { push @Mparams, $arg; next; } if ($arg =~ /^-T$/) { $was_T = 1; next; } if ($arg =~ s/^-T(\w+)$/$1/) { push @devices, $1; next; } if ($arg =~ /^-(\w)(\w*)$/) { # maybe a groff option my $opt_char = $1; my $opt_char_with_arg = $opt_char . ':'; my $others = $2; if ( $groff_opts =~ /$opt_char_with_arg/ ) { # groff optarg if ( $others ) { # optarg is here push @Command, '-' . $opt_char; push @Command, '-' . $others; next; } # next arg is optarg $optarg = 1; next; ####### handle_args() } elsif ( $groff_opts =~ /$opt_char/ ) { # groff no optarg push @Command, '-' . $opt_char; if ( $others ) { # $others is now an opt collection $arg = '-' . $others; redo; } # arg finished next; } else { # not a groff opt print STDERR __FILE__ . ' ' . __LINE__ . ': ' . 'unknown argument ' . $arg; push(@Command, $arg); next; } } } @filespec = ('-') unless (@filespec); } # handle_args() ######################################################################## # handle_file_ext() ######################################################################## sub handle_file_ext { # get tmac from file name extension # output number of found single tmac # globals: @filespec, $tmac_ext; foreach my $file ( @filespec ) { # test for each file name in the arguments unless ( open(FILE, $file eq "-" ? $file : "< $file") ) { print STDERR __FILE__ . ' ' . __LINE__ . ': ' . "$Prog: can't open \'$file\': $!"; next; } next unless ( $file =~ /\./ ); # file name has no dot '.' ##### handle_file_ext() # get extension my $ext = $file; $ext =~ s/^ .* \. ([^.]*) $ /$1/x; next unless ( $ext ); ##### handle_file_ext() # these extensions are correct, but not based on a tmac next if ( $ext =~ /^( chem| eqn| g| grap| grn| groff| hdtbl| pdfroff| pic| pinyin| ref| roff| t| tbl| tr| www )$/x ); ##### handle_file_ext() # extensions for man tmac if ( $ext =~ /^( [1-9lno]| man| n| 1b )$/x ) { # 'man|n' from 'groff' source # '1b' from 'heirloom' # '[1-9lno]' from man-pages if ( $tmac_ext && $tmac_ext ne 'man' ) { # found tmac is not 'man' print STDERR __FILE__ . ' ' . __LINE__ . ': ' . '2 different file name extensions found ' . $tmac_ext . ' and ' . $ext; $tmac_ext = ''; next; } ##### handle_file_ext() $tmac_ext = 'man'; next; } if ( $ext =~ /^( mandoc| mdoc| me| mm| mmse| mom| ms| $)/x ) { if ( $tmac_ext && $tmac_ext ne $ext ) { # found tmac is not identical to former found tmac ##### handle_file_ext() print STDERR __FILE__ . ' ' . __LINE__ . ': ' . '2 different file name extensions found ' . $tmac_ext . ' and ' . $ext; $tmac_ext = ''; next; } $tmac_ext = $ext; next; } print STDERR __FILE__ . ' ' . __LINE__ . ': ' . 'Unknown file name extension '. $file . '.'; next; } # end foreach file 1; } # handle_file_ext() ######################################################################## # handle_whole_files() ######################################################################## sub handle_whole_files { # globals: @filespec foreach my $file ( @filespec ) { unless ( open(FILE, $file eq "-" ? $file : "< $file") ) { print STDERR __FILE__ . ' ' . __LINE__ . ': ' . "$Prog: can't open \'$file\': $!"; next; } my $line = <FILE>; # get single line unless ( defined($line) ) { # empty file, go to next filearg close (FILE); next; } if ( $line ) { chomp $line; unless ( &do_first_line( $line, $file ) ) { # not an option line &do_line( $line, $file ); } } else { # empty line next; } while (<FILE>) { # get lines by and by chomp; &do_line( $_, $file ); } close(FILE); } # end foreach } # handle_whole_files() ######################################################################## # do_first_line() ######################################################################## # As documented for the 'man' program, the first line can be # used as a groff option line. This is done by: # - start the line with '\" (apostrophe, backslash, double quote) # - add a space character # - a word using the following characters can be appended: 'egGjJpRst'. # Each of these characters means an option for the generated # 'groff' command line, e.g. '-t'. sub do_first_line { my ( $line, $file ) = @_; # globals: %preprocs_tmacs # For a leading groff options line use only [egGjJpRst] if ( $line =~ /^[.']\\"[\segGjJpRst]+&/ ) { # this is a groff options leading line if ( $line =~ /^\./ ) { # line is a groff options line with . instead of ' print "First line in $file must start with an apostrophe \ " . "instead of a period . for groff options line!"; } if ( $line =~ /j/ ) { $Groff{'chem'}++; } if ( $line =~ /e/ ) { $Groff{'eqn'}++; } if ( $line =~ /g/ ) { $Groff{'grn'}++; } if ( $line =~ /G/ ) { $Groff{'grap'}++; } if ( $line =~ /i/ ) { $Groff{'gideal'}++; } if ( $line =~ /p/ ) { $Groff{'pic'}++; } if ( $line =~ /R/ ) { $Groff{'refer'}++; } if ( $line =~ /s/ ) { $Groff{'soelim'}++; } ####### do_first_line() if ( $line =~ /t/ ) { $Groff{'tbl'}++; } return 1; # a leading groff options line, 1 means yes, 0 means no } # not a leading short groff options line return 0 if ( $line !~ /^[.']\\"\s*(.*)$/ ); # ignore non-comments return 0 unless ( $1 ); # for empty comment # all following array members are either preprocs or 1 tmac my @words = split '\s+', $1; my @in = (); my $word; for $word ( @words ) { if ( $word eq 'ideal' ) { $word = 'gideal'; } elsif ( $word eq 'gpic' ) { $word = 'pic'; } elsif ( $word =~ /^(gn|)eqn$/ ) { $word = 'eqn'; } if ( exists $preprocs_tmacs{$word} ) { push @in, $word; } else { # not word for preproc or tmac return 0; } } for $word ( @in ) { $Groff{$word}++; } } # do_first_line() ######################################################################## # do_line() ######################################################################## my $before_first_command = 1; # for check of .TH sub do_line { my ( $line, $file ) = @_; return if ( $line =~ /^[.']\s*\\"/ ); # comment return unless ( $line =~ /^[.']/ ); # ignore text lines $line =~ s/^['.]\s*/./; # let only a dot as leading character, # remove spaces after the leading dot $line =~ s/\s+$//; # remove final spaces return if ( $line =~ /^\.$/ ); # ignore . return if ( $line =~ /^\.\.$/ ); # ignore .. if ( $before_first_command ) { # so far without 1st command if ( $line =~ /^\.TH/ ) { # check if .TH is 1st command for man $Groff{'TH_first'} = 1 if ( $line =~ /^\.\s*TH/ ); } if ( $line =~ /^\./ ) { $before_first_command = 0; } } # split command $line =~ /^(\.\w+)\s*(.*)$/; my $command = $1; $command = '' unless ( defined $command ); my $args = $2; $args = '' unless ( defined $args ); ###################################################################### # soelim if ( $line =~ /^\.(do)?\s*(so|mso|PS\s*<|SO_START).*$/ ) { # '.so', '.mso', '.PS<...', '.SO_START' $Groff{'soelim'}++; return; } if ( $line =~ /^\.(do)?\s*(so|mso|PS\s*<|SO_START).*$/ ) { # '.do so', '.do mso', '.do PS<...', '.do SO_START' $Groff{'soelim'}++; return; } ####### do_line() ###################################################################### # macros if ( $line =~ /^\.de1?\W?/ ) { # this line is a macro definition, add it to %macros my $macro = $line; $macro =~ s/^\.de1?\s+(\w+)\W*/.$1/; return if ( exists $macros{$macro} ); $macros{$macro} = 1; return; } # if line command is a defined macro, just ignore this line return if ( exists $macros{$command} ); ###################################################################### # preprocessors if ( $command =~ /^(\.cstart)|(begin\s+chem)$/ ) { $Groff{'chem'}++; # for chem return; } if ( $command =~ /^\.EQ$/ ) { $Groff{'eqn'}++; # for eqn return; } if ( $command =~ /^\.G1$/ ) { $Groff{'grap'}++; # for grap return; } if ( $command =~ /^\.Perl/ ) { $Groff{'gperl'}++; # for gperl return; } if ( $command =~ /^\.pinyin/ ) { $Groff{'gpinyin'}++; # for gperl return; } if ( $command =~ /^\.GS$/ ) { $Groff{'grn'}++; # for grn return; } if ( $command =~ /^\.IS$/ ) { $Groff{'gideal'}++; # preproc gideal for ideal return; } if ( $command =~ /^\.lilypond$/ ) { $Groff{'lilypond'}++; # for glilypond return; } ####### do_line() # pic can be opened by .PS or .PF and closed by .PE if ( $command =~ /^\.PS$/ ) { $Groff{'pic'}++; # normal opening for pic return; } if ( $command =~ /^\.PF$/ ) { $Groff{'PF'}++; # alternate opening for pic return; } if ( $command =~ /^\.PE$/ ) { $Groff{'PE'}++; # closing for pic return; } if ( $command =~ /^\.R1$/ ) { $Groff{'refer'}++; # for refer return; } if ( $command =~ /^\.\[$/ ) { $Groff{'refer_open'}++; # for refer open return; } if ( $command =~ /^\.\]$/ ) { $Groff{'refer_close'}++; # for refer close return; } if ( $command =~ /^\.TS$/ ) { $Groff{'tbl'}++; # for tbl return; } if ( $command =~ /^\.TH$/ ) { unless ( $Groff{'TH_first'} ) { $Groff{'TH_later'}++; # for tbl } return; } ###################################################################### # macro package (tmac) ###################################################################### ########## # modern mdoc if ( $command =~ /^\.(Dd)$/ ) { $Groff{'Dd'}++; # for modern mdoc return; } ####### do_line() # In the old version of -mdoc 'Oo' is a toggle, in the new it's # closed by 'Oc'. if ( $command =~ /^\.Oc$/ ) { $Groff{'Oc'}++; # only for modern mdoc return; } ########## # old and modern mdoc if ( $command =~ /^\.Oo$/ ) { $Groff{'Oo'}++; # for mdoc and mdoc-old return; } ########## # old mdoc if ( $command =~ /^\.(Tp|Dp|De|Cx|Cl)$/ ) { $Groff{'mdoc_old'}++; # true for old mdoc return; } ########## # for ms ####### do_line() if ( $command =~ /^\.AB$/ ) { $Groff{'AB'}++; # for ms return; } if ( $command =~ /^\.AE$/ ) { $Groff{'AE'}++; # for ms return; } if ( $command =~ /^\.AI$/ ) { $Groff{'AI'}++; # for ms return; } if ( $command =~ /^\.AU$/ ) { $Groff{'AU'}++; # for ms return; } if ( $command =~ /^\.NH$/ ) { $Groff{'NH'}++; # for ms return; } if ( $command =~ /^\.TL$/ ) { $Groff{'TL'}++; # for ms return; } if ( $command =~ /^\.XP$/ ) { $Groff{'XP'}++; # for ms return; } ########## # for man and ms if ( $command =~ /^\.IP$/ ) { $Groff{'IP'}++; # for man and ms return; } if ( $command =~ /^\.LP$/ ) { $Groff{'LP'}++; # for man and ms return; } ####### do_line() if ( $command =~ /^\.P$/ ) { $Groff{'P'}++; # for man and ms return; } if ( $command =~ /^\.PP$/ ) { $Groff{'PP'}++; # for man and ms return; } if ( $command =~ /^\.SH$/ ) { $Groff{'SH'}++; # for man and ms return; } if ( $command =~ /^\.UL$/ ) { $Groff{'UL'}++; # for man and ms return; } ########## # for man only if ( $command =~ /^\.OP$/ ) { # for man $Groff{'OP'}++; return; } if ( $command =~ /^\.SS$/ ) { # for man $Groff{'SS'}++; return; } if ( $command =~ /^\.SY$/ ) { # for man $Groff{'SY'}++; return; } if ( $command =~ /^\.TP$/ ) { # for man $Groff{'TP'}++; return; } if ( $command =~ /^\.UR$/ ) { $Groff{'UR'}++; # for man return; } if ( $command =~ /^\.YS$/ ) { # for man $Groff{'YS'}++; return; } ####### do_line() ########## # me if ( $command =~ /^\.( [ilnp]p| sh )$/x ) { $Groff{'me'}++; # for me return; } ############# # mm and mmse if ( $command =~ /^\.( H| MULB| LO| LT| NCOL| P\$| PH| SA )$/x ) { $Groff{'mm'}++; # for mm and mmse if ( $command =~ /^\.LO$/ ) { if ( $args =~ /^(DNAMN|MDAT|BIL|KOMP|DBET|BET|SIDOR)/ ) { $Groff{'mmse'}++; # for mmse } } elsif ( $command =~ /^\.LT$/ ) { if ( $args =~ /^(SVV|SVH)/ ) { $Groff{'mmse'}++; # for mmse } } return; } ####### do_line() ########## # mom if ( $line =~ /^\.( ALD| DOCTYPE| FAMILY| FT| FAM| LL| LS| NEWPAGE| PAGE| PAPER| PRINTSTYLE| PT_SIZE| T_MARGIN )$/x ) { $Groff{'mom'}++; # for mom return; } } # do_line() ######################################################################## # sub make_groff_device ######################################################################## my @m = (); my @preprograms = (); my $correct_tmac = ''; sub make_groff_device { # globals: @devices # default device is 'ps' when without '-T' my $device; push @devices, 'ps' unless ( @devices ); ###### make_groff_device() for my $d ( @devices ) { if ( $d =~ /^( # suitable devices dvi| html| xhtml| lbp| lj4| ps| pdf| ascii| cp1047| latin1| utf8 )$/x ) { ###### make_groff_device() $device = $d; } else { next; } if ( $device ) { push @Command, '-T'; push @Command, $device; } } ###### make_groff_device() if ( $device eq 'pdf' ) { if ( $pdf_with_ligatures ) { # with --ligature argument push( @Command, '-P-y' ); push( @Command, '-PU' ); } else { # no --ligature argument if ( $with_warnings ) { print STDERR <<EOF; If you have trouble with ligatures like 'fi' in the 'groff' output, you can proceed as one of - add 'grog' option '--with_ligatures' or - use the 'grog' option combination '-P-y -PU' or - try to remove the font named similar to 'fonts-texgyre' from your system. EOF } # end of warning } # end of ligature } # end of pdf device } # make_groff_device() ######################################################################## # make_groff_preproc() ######################################################################## sub make_groff_preproc { # globals: %Groff, @preprograms, @Command # preprocessors without 'groff' option if ( $Groff{'lilypond'} ) { push @preprograms, 'glilypond'; } if ( $Groff{'gperl'} ) { push @preprograms, 'gperl'; } if ( $Groff{'gpinyin'} ) { push @preprograms, 'gpinyin'; } # preprocessors with 'groff' option if ( ( $Groff{'PS'} || $Groff{'PF'} ) && $Groff{'PE'} ) { $Groff{'pic'} = 1; } if ( $Groff{'gideal'} ) { $Groff{'pic'} = 1; } ###### make_groff_preproc() $Groff{'refer'} ||= $Groff{'refer_open'} && $Groff{'refer_close'}; if ( $Groff{'chem'} || $Groff{'eqn'} || $Groff{'gideal'} || $Groff{'grap'} || $Groff{'grn'} || $Groff{'pic'} || $Groff{'refer'} || $Groff{'tbl'} ) { push(@Command, '-s') if $Groff{'soelim'}; push(@Command, '-R') if $Groff{'refer'}; push(@Command, '-t') if $Groff{'tbl'}; # tbl before eqn push(@Command, '-e') if $Groff{'eqn'}; push(@Command, '-j') if $Groff{'chem'}; # chem produces pic code push(@Command, '-J') if $Groff{'gideal'}; # gideal produces pic push(@Command, '-G') if $Groff{'grap'}; push(@Command, '-g') if $Groff{'grn'}; # gremlin files for -me push(@Command, '-p') if $Groff{'pic'}; } } # make_groff_preproc() ######################################################################## # make_groff_tmac_man_ms() ######################################################################## sub make_groff_tmac_man_ms { # globals: @filespec, $tmac_ext, %Groff # 'man' requests, not from 'ms' if ( $Groff{'SS'} || $Groff{'SY'} || $Groff{'OP'} || $Groff{'TH_first'} || $Groff{'TP'} || $Groff{'UR'} ) { $Groff{'man'} = 1; push(@m, '-man'); $tmac_ext = 'man' unless ( $tmac_ext ); &err('man requests found, but file name extension ' . 'was: ' . $tmac_ext) unless ( $tmac_ext eq 'man' ); $tmac_ext = 'man'; return 1; # true } ###### make_groff_tmac_man_ms() # 'ms' requests, not from 'man' if ( $Groff{'1C'} || $Groff{'2C'} || $Groff{'AB'} || $Groff{'AE'} || $Groff{'AI'} || $Groff{'AU'} || $Groff{'BX'} || $Groff{'CD'} || $Groff{'DA'} || $Groff{'DE'} || $Groff{'DS'} || $Groff{'ID'} || $Groff{'LD'} || $Groff{'NH'} || $Groff{'TH_later'} || $Groff{'TL'} || $Groff{'UL'} || $Groff{'XP'} ) { $Groff{'ms'} = 1; push(@m, '-ms'); $tmac_ext = 'ms' unless ( $tmac_ext ); &err('ms requests found, but file name extension ' . 'was: ' . $tmac_ext) unless ( $tmac_ext eq 'ms' ); $tmac_ext = 'ms'; return 1; # true } ###### make_groff_tmac_man_ms() # both 'man' and 'ms' requests if ( $Groff{'P'} || $Groff{'IP'} || $Groff{'LP'} || $Groff{'PP'} || $Groff{'SH'} ) { if ( $tmac_ext eq 'man' ) { $Groff{'man'} = 1; push(@m, '-man'); return 1; # true } elsif ( $tmac_ext eq 'ms' ) { $Groff{'ms'} = 1; push(@m, '-ms'); return 1; # true } return 0; } } # make_groff_tmac_man_ms() ######################################################################## # make_groff_tmac_others() ######################################################################## sub make_groff_tmac_others { # globals: @filespec, $tmac_ext, %Groff # mdoc if ( ( $Groff{'Oo'} && $Groff{'Oc'} ) || $Groff{'Dd'} ) { $Groff{'Oc'} = 0; $Groff{'Oo'} = 0; push(@m, '-mdoc'); return 1; # true } if ( $Groff{'mdoc_old'} || $Groff{'Oo'} ) { push(@m, '-mdoc_old'); return 1; # true } # me if ( $Groff{'me'} ) { push(@m, '-me'); return 1; # true } ##### make_groff_tmac_others() # mm and mmse if ( $Groff{'mm'} ) { push(@m, '-mm'); return 1; # true } if ( $Groff{'mmse'} ) { # Swedish mm push(@m, '-mmse'); return 1; # true } # mom if ( $Groff{'mom'} ) { push(@m, '-mom'); return 1; # true } } # make_groff_tmac_others() ######################################################################## # make_groff_line_rest() ######################################################################## sub make_groff_line_rest { my $file_args_included; # file args now only at 1st preproc unshift @Command, 'groff'; if ( @preprograms ) { my @progs; $progs[0] = shift @preprograms; push(@progs, @filespec); for ( @preprograms ) { push @progs, '|'; push @progs, $_; } push @progs, '|'; unshift @Command, @progs; $file_args_included = 1; } else { $file_args_included = 0; } ###### make_groff_line_rest() foreach (@Command) { next unless /\s/; # when one argument has several words, use accents $_ = "'" . $_ . "'"; } ###### make_groff_line_rest() ########## # -m arguments my $nr_m_guessed = scalar @m; if ( $nr_m_guessed > 1 ) { print STDERR __FILE__ . ' ' . __LINE__ . ': ' . 'argument for -m found: ' . @m; } my $nr_m_args = scalar @Mparams; # m-arguments for grog my $last_m_arg = ''; # last provided -m option if ( $nr_m_args > 1 ) { # take the last given -m argument of grog call, # ignore other -m arguments and the found ones $last_m_arg = $Mparams[-1]; # take the last -m argument print STDERR __FILE__ . ' ' . __LINE__ . ': ' . $Prog . ": more than 1 '-m' argument: @Mparams"; print STDERR __FILE__ . ' ' . __LINE__ . ': ' . 'We take the last one: ' . $last_m_arg; } elsif ( $nr_m_args == 1 ) { $last_m_arg = $Mparams[0]; } ###### make_groff_line_rest() my $final_m = ''; if ( $last_m_arg ) { my $is_equal = 0; for ( @m ) { if ( $_ eq $last_m_arg ) { $is_equal = 1; last; } next; } # end for @m if ( $is_equal ) { $final_m = $last_m_arg; } else { print STDERR __FILE__ . ' ' . __LINE__ . ': ' . 'Provided -m argument ' . $last_m_arg . ' differs from guessed -m args: ' . @m; print STDERR __FILE__ . ' ' . __LINE__ . ': ' . 'The argument is taken.'; $final_m = $last_m_arg; } ###### make_groff_line_rest() } else { # no -m arg provided if ( $nr_m_guessed > 1 ) { print STDERR __FILE__ . ' ' . __LINE__ . ': ' . 'More than 1 -m arguments were guessed: ' . @m; print STDERR __FILE__ . ' ' . __LINE__ . ': ' . 'Guessing stopped.'; exit 1; } elsif ( $nr_m_guessed == 1 ) { $final_m = $m[0]; } else { # no -m provided or guessed } } push @Command, $final_m if ( $final_m ); push(@Command, @filespec) unless ( $file_args_included ); ######### # execute the 'groff' command here with option '--run' if ( $do_run ) { # with --run print STDERR __FILE__ . ' ' . __LINE__ . ': ' . "@Command"; my $cmd = join ' ', @Command; system($cmd); } else { print "@Command"; } exit 0; } # make_groff_line_rest() ######################################################################## # sub help ######################################################################## sub help { print <<EOF; usage: grog [option]... [--] [filespec]... "filespec" is either the name of an existing, readable file or "-" for standard input. If no 'filespec' is specified, standard input is assumed automatically. All arguments after a '--' are regarded as file names, even if they start with a '-' character. 'option' is either a 'groff' option or one of these: -h|--help print this uasge message and exit -v|--version print version information and exit -C compatibility mode --ligatures include options '-P-y -PU' for internal font, which preserves the ligatures like 'fi' --run run the checked-out groff command --warnings display more warnings to standard error All other options should be 'groff' 1-character options. These are then appended to the generated 'groff' command line. The '-m' options will be checked by 'grog'. EOF exit 0; } # help() ######################################################################## # sub version ######################################################################## sub version { our %at_at; print "Perl version of GNU $Prog " . "in groff version " . $at_at{'GROFF_VERSION'}; exit 0; } # version() 1; ######################################################################## ### Emacs settings # Local Variables: # mode: CPerl # End: Save