• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1#!/usr/bin/env perl
2#***************************************************************************
3#                                  _   _ ____  _
4#  Project                     ___| | | |  _ \| |
5#                             / __| | | | |_) | |
6#                            | (__| |_| |  _ <| |___
7#                             \___|\___/|_| \_\_____|
8#
9# Copyright (C) Daniel Stenberg, <daniel@haxx.se>, et al.
10#
11# This software is licensed as described in the file COPYING, which
12# you should have received as part of this distribution. The terms
13# are also available at https://curl.se/docs/copyright.html.
14#
15# You may opt to use, copy, modify, merge, publish, distribute and/or sell
16# copies of the Software, and permit persons to whom the Software is
17# furnished to do so, under the terms of the COPYING file.
18#
19# This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
20# KIND, either express or implied.
21#
22# SPDX-License-Identifier: curl
23#
24#
25###########################################################################
26#
27# Check that the deprecated statuses of functions and enum values in header
28# files, man pages and symbols-in-versions are in sync.
29
30use strict;
31use warnings;
32
33use File::Basename;
34
35my $root=$ARGV[0] || ".";
36my $incdir = "$root/include/curl";
37my $docdir = "$root/docs";
38my $libdocdir = "$docdir/libcurl";
39my $errcode = 0;
40
41# Symbol-indexed hashes.
42# Values are:
43#     X       Not deprecated
44#     ?       Deprecated in unknown version
45#     x.yy.z  Deprecated in version x.yy.z
46my %syminver;       # Symbols-in-versions deprecations.
47my %hdr;            # Public header files deprecations.
48my %funcman;        # Function man pages deprecations.
49my %optman;         # Option man pages deprecations.
50
51
52# Scan header file for public function and enum values. Flag them with
53# the version they are deprecated in, if some.
54sub scan_header {
55    my ($f)=@_;
56    my $line = "";
57    my $incomment = 0;
58    my $inenum = 0;
59
60    open H, "<$f";
61    while(<H>) {
62      s/^\s*(.*?)\s*$/$1/;      # Trim.
63      # Remove multi-line comment trail.
64      if($incomment) {
65        if($_ !~ /.*?\*\/\s*(.*)$/) {
66          next;
67        }
68        $_ = $1;
69        $incomment = 0;
70      }
71      if($line ne "") {
72        # Unfold line.
73        $_ = "$line $1";
74        $line = "";
75      }
76      # Remove comments.
77      while($_ =~ /^(.*?)\/\*.*?\*\/(.*)$/) {
78        $_ = "$1 $2";
79      }
80      if($_ =~ /^(.*)\/\*/) {
81        $_ = "$1 ";
82        $incomment = 1;
83      }
84      s/^\s*(.*?)\s*$/$1/;      # Trim again.
85      # Ignore preprocessor directives and blank lines.
86      if($_ =~ /^(?:#|$)/) {
87        next;
88      }
89      # Handle lines that may be continued as if they were folded.
90      if($_ !~ /[;,{}]$/) {
91        # Folded line.
92        $line = $_;
93        next;
94      }
95      if($_ =~ /CURLOPTDEPRECATED\(/) {
96        # Handle deprecated CURLOPT_* option.
97        if($_ !~ /CURLOPTDEPRECATED\(\s*(\S+)\s*,(?:.*?,){2}\s*(.*?)\s*,.*"\)/) {
98          # Folded line.
99          $line = $_;
100          next;
101        }
102        $hdr{$1} = $2;
103      }
104      elsif($_ =~ /CURLOPT\(/) {
105        # Handle non-deprecated CURLOPT_* option.
106        if($_ !~ /CURLOPT\(\s*(\S+)\s*(?:,.*?){2}\)/) {
107          # Folded line.
108          $line = $_;
109          next;
110        }
111        $hdr{$1} = "X";
112      }
113      else {
114        my $version = "X";
115
116        # Get other kind of deprecation from this line.
117        if($_ =~ /CURL_DEPRECATED\(/) {
118          if($_ !~ /^(.*)CURL_DEPRECATED\(\s*(\S+?)\s*,.*?"\)(.*)$/) {
119            # Folded line.
120            $line = $_;
121            next;
122          }
123         $version = $2;
124         $_ = "$1 $3";
125        }
126        if($_ =~ /^CURL_EXTERN\s+.*\s+(\S+?)\s*\(/) {
127          # Flag public function.
128          $hdr{$1} = $version;
129        }
130        elsif($inenum && $_ =~ /(\w+)\s*[,=}]/) {
131          # Flag enum value.
132          $hdr{$1} = $version;
133        }
134      }
135      # Remember if we are in an enum definition.
136      $inenum |= ($_ =~ /\benum\b/);
137      if($_ =~ /}/) {
138        $inenum = 0;
139      }
140    }
141    close H;
142}
143
144# Scan function man page for options.
145# Each option has to be declared as ".IP <option>" where <option> starts with
146# the prefix. Flag each option with its deprecation version, if some.
147sub scan_man_for_opts {
148    my ($f, $prefix)=@_;
149    my $opt = "";
150    my $line = "";
151
152    open M, "<$f";
153    while(<M>) {
154      if($_ =~ /^\./) {
155        # roff directive found: end current option paragraph.
156        my $o = $opt;
157        $opt = "";
158        if($_ =~ /^\.IP\s+((?:$prefix)_\w+)/) {
159          # A new option has been found.
160          $opt = $1;
161        }
162        $_ = $line;     # Get full paragraph.
163        $line = "";
164        s/\\f.//g;      # Remove font formatting.
165        s/\s+/ /g;      # One line with single space only.
166        if($o) {
167          $funcman{$o} = "X";
168          # Check if paragraph is mentioning deprecation.
169          while($_ =~ /(?:deprecated|obsoleted?)\b\s*(?:in\b|since\b)?\s*(?:version\b|curl\b|libcurl\b)?\s*(\d[0-9.]*\d)?\b\s*(.*)$/i) {
170            $funcman{$o} = $1 || "?";
171            $_ = $2;
172          }
173        }
174      }
175      else {
176        # Text line: accumulate.
177        $line .= $_;
178      }
179    }
180    close M;
181}
182
183# Scan man page for deprecation in DESCRIPTION and/or AVAILABILITY sections.
184sub scan_man_page {
185    my ($path, $sym, $table)=@_;
186    my $version = "X";
187    my $fh;
188
189    if(open $fh, "<$path") {
190      my $section = "";
191      my $line = "";
192
193      while(<$fh>) {
194        if($_ =~ /\.so\s+man3\/(.*\.3\b)/) {
195          # Handle man page inclusion.
196          scan_man_page(dirname($path) . "/$1", $sym, $table);
197          $version = exists($$table{$sym})? $$table{$sym}: $version;
198        }
199        elsif($_ =~ /^\./) {
200          # Line is a roff directive.
201          if($_ =~ /^\.SH\b\s*(\w*)/) {
202            # Section starts. End previous one.
203            my $sh = $section;
204
205            $section = $1;
206            $_ = $line;     # Previous section text.
207            $line = "";
208            s/\\f.//g;
209            s/\s+/ /g;
210            s/\\f.//g;      # Remove font formatting.
211            s/\s+/ /g;      # One line with single space only.
212            if($sh =~ /DESCRIPTION|AVAILABILITY/) {
213              while($_ =~ /(?:deprecated|obsoleted?)\b\s*(?:in\b|since\b)?\s*(?:version\b|curl\b|libcurl\b)?\s*(\d[0-9.]*\d)?\b\s*(.*)$/i) {
214                # Flag deprecation status.
215                if($version ne "X" && $version ne "?") {
216                  if($1 && $1 ne $version) {
217                    print "error: $sym man page lists unmatching deprecation versions $version and $1\n";
218                    $errcode++;
219                  }
220                }
221                else {
222                  $version = $1 || "?";
223                }
224                $_ = $2;
225              }
226            }
227          }
228        }
229        else {
230          # Text line: accumulate.
231          $line .= $_;
232        }
233      }
234      close $fh;
235      $$table{$sym} = $version;
236    }
237}
238
239
240# Read symbols-in-versions.
241open(F, "<$libdocdir/symbols-in-versions") ||
242  die "$libdocdir/symbols-in-versions";
243while(<F>) {
244  if($_ =~ /^((?:CURL|LIBCURL)\S+)\s+\S+\s*(\S*)\s*(\S*)$/) {
245    if($3 eq "") {
246      $syminver{$1} = "X";
247      if($2 ne "" && $2 ne ".") {
248        $syminver{$1} = $2;
249      }
250    }
251  }
252}
253close(F);
254
255# Get header file names,
256opendir(my $dh, $incdir) || die "Can't opendir $incdir";
257my @hfiles = grep { /\.h$/ } readdir($dh);
258closedir $dh;
259
260# Get functions and enum symbols from header files.
261for(@hfiles) {
262  scan_header("$incdir/$_");
263}
264
265# Get function statuses from man pages.
266foreach my $sym (keys %hdr) {
267  if($sym =~/^(?:curl|curlx)_\w/) {
268    scan_man_page("$libdocdir/$sym.3", $sym, \%funcman);
269  }
270}
271
272# Get options from function man pages.
273scan_man_for_opts("$libdocdir/curl_easy_setopt.3", "CURLOPT");
274scan_man_for_opts("$libdocdir/curl_easy_getinfo.3", "CURLINFO");
275
276# Get deprecation status from option man pages.
277foreach my $sym (keys %syminver) {
278  if($sym =~ /^(?:CURLOPT|CURLINFO)_\w+$/) {
279    scan_man_page("$libdocdir/opts/$sym.3", $sym, \%optman);
280  }
281}
282
283# Print results.
284my %keys = (%syminver, %funcman, %optman, %hdr);
285my $leader = <<HEADER
286Legend:
287<empty> Not listed
288X       Not deprecated
289?       Deprecated in unknown version
290x.yy.z  Deprecated in version x.yy.z
291
292Symbol                                 symbols-in  func man  opt man   .h
293                                       -versions
294HEADER
295        ;
296foreach my $sym (sort {$a cmp $b} keys %keys) {
297  if($sym =~ /^(?:CURLOPT|CURLINFO|curl|curlx)_\w/) {
298    my $s = exists($syminver{$sym})? $syminver{$sym}: " ";
299    my $f = exists($funcman{$sym})? $funcman{$sym}: " ";
300    my $o = exists($optman{$sym})? $optman{$sym}: " ";
301    my $h = exists($hdr{$sym})? $hdr{$sym}: " ";
302    my $r = " ";
303
304    # There are deprecated symbols in symbols-in-versions that are aliases
305    # and thus not listed anywhere else. Ignore them.
306    "$f$o$h" =~ /[X ]{3}/ && next;
307
308    # Check for inconsistencies between deprecations from the different sources.
309    foreach my $k ($s, $f, $o, $h) {
310      $r = $r eq " "? $k: $r;
311      if($k ne " " && $r ne $k) {
312        if($r eq "?") {
313          $r = $k ne "X"? $k: "!";
314        }
315        elsif($r eq "X" || $k ne "?") {
316          $r = "!";
317        }
318      }
319    }
320
321    if($r eq "!") {
322      print $leader;
323      $leader = "";
324      printf("%-38s %-11s %-9s %-9s %s\n", $sym, $s, $f, $o, $h);
325      $errcode++;
326    }
327  }
328}
329
330exit $errcode;
331