]> CyberLeo.Net >> Repos - FreeBSD/releng/10.0.git/blob - contrib/ofed/management/opensm/opensm/osm_check_n_fix
- Copy stable/10 (r259064) to releng/10.0 as part of the
[FreeBSD/releng/10.0.git] / contrib / ofed / management / opensm / opensm / osm_check_n_fix
1 eval '(exit $?0)' &&
2   eval 'exec perl -S $0 ${1+"$@"}' &&
3   eval 'exec perl -S $0 $argv:q'
4   if 0;
5
6 #!/usr/bin/perl -W
7 #
8 # Copyright (c) 2004, 2005 Voltaire, Inc. All rights reserved.
9 # Copyright (c) 2002-2005 Mellanox Technologies LTD. All rights reserved.
10 # Copyright (c) 1996-2003 Intel Corporation. All rights reserved.
11 #
12 # This software is available to you under a choice of one of two
13 # licenses.  You may choose to be licensed under the terms of the GNU
14 # General Public License (GPL) Version 2, available from the file
15 # COPYING in the main directory of this source tree, or the
16 # OpenIB.org BSD license below:
17 #
18 #     Redistribution and use in source and binary forms, with or
19 #     without modification, are permitted provided that the following
20 #     conditions are met:
21 #
22 #      - Redistributions of source code must retain the above
23 #        copyright notice, this list of conditions and the following
24 #        disclaimer.
25 #
26 #      - Redistributions in binary form must reproduce the above
27 #        copyright notice, this list of conditions and the following
28 #        disclaimer in the documentation and/or other materials
29 #        provided with the distribution.
30 #
31 # THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
32 # EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
33 # MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
34 # NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS
35 # BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN
36 # ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
37 # CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
38 # SOFTWARE.
39 #
40 #########################################################################
41 #
42 #  Abstract:
43 #     Perl script for simple source code error checking and fixing
44 #
45 #  Environment:
46 #     Linux User Mode
47 #
48 #  Author:
49 #     Eitan Zahavi, Mellanox Technologies LTD Yokneam Israel.
50 #
51 #  $Revision: 1.4 $
52 #
53 #
54 #
55 # DESCRIPTION:
56 #
57 # This script performs some simple conformance checks on the
58 # OpenSM source code.  It does NOT attempt to act like a full
59 # blown 'C' language parser, so it can be fooled.  Something
60 # is better than nothing.
61 #
62 # The script starts by running the 'osm_indent' script on teh given files.
63 #
64 # We use an extra file for tracking error codes used by each file.
65 # The name is osm_errors_codes.
66 #
67 # The following checks are performed:
68 # 1) Verify that the function name provided in a log statement
69 #    matches the name of the current function.
70 #
71 # 2) Verify that log statements are in the form that this script
72 #    can readily parse.  Improvements to the regular expressions
73 #    might make this unnecessary.
74 #
75 # 3) Verify that lower two digits of the error codes used in log
76 #    statements are unique within that file.
77 #
78 # 4) Verify that upper two digits of the error codes used in log
79 #    statements are not used by any other module.
80 #
81 # 5) Verify the lines do not have extra spaces.
82 #
83 # USAGE:
84 #
85 # In the OpenSM source directory, type:
86 # osm_check_n_fix -f *.c
87 #
88 #########################################################################
89
90 # Do necessary upfront initialization
91 $verbose = 0;
92 $in_c_comment = 0;
93 $fix_mode = 0;
94 $confirm_mode = 0;
95 $re_assign_err_prefix = 0;
96
97 if( !scalar(@ARGV) )
98 {
99     print "ERROR: You must specify the files on which to operate, such as '*.c'\n";
100     osm_check_usage();
101     exit;
102 }
103
104 # loop through all the command line options
105 do
106 {
107     $doing_params = 0;
108
109     # First, look for command line options.
110     if( $ARGV[0] =~ /-[v|V]/ )
111     {
112         $verbose += 1;
113         shift;
114         print "Verbose mode on, level = $verbose.\n";
115         $doing_params = 1;
116     }
117
118     if( $ARGV[0] =~ /(-f|--fix)/ )
119     {
120         $fix_mode += 1;
121         shift;
122         print "Fix mode on.\n";
123         $doing_params = 1;
124     }
125
126     if( $ARGV[0] =~ /(-c|--confirm)/ )
127     {
128         $confirm_mode += 1;
129         shift;
130         print "Confirm mode on.\n";
131         $doing_params = 1;
132     }
133
134     if( $ARGV[0] =~ /(-r|--re-assign-mod-err-prefix)/ )
135     {
136         $re_assign_err_prefix += 1;
137         shift;
138         print "Allow Re-Assignment of Module Err Prefixes.\n";
139         $doing_params = 1;
140     }
141
142     if( !scalar(@ARGV))
143     {
144         print "ERROR: You must specify the files on which to operate, such as '*.c'\n";
145         osm_check_usage();
146         exit;
147     }
148 } while( $doing_params == 1 );
149
150 # parse the osm_error_codes file and define:
151 # module_by_prefix
152 # module_err_prefixes
153 # module_last_err_used
154 if (open(ERRS, "<osm_error_codes")) {
155   @ERR_DEFS = <ERRS>;
156   close(ERRS);
157   foreach $errDef (@ERR_DEFS) {
158     # the format should be <file name> <err prefix> <last err>
159     if ($errDef =~ m/^(\S+)\s+(\S+)\s+([0-9]+)$/) {
160       ($file_name,$mod_prefix,$last_err) = ($1,$2,$3);
161       if (defined($module_by_prefix{$mod_prefix})) {
162         print "ERROR: Double module prefix:$mod_prefix on:$module_by_prefix($mod_prefix) and $file_name\n";
163         exit 3;
164       }
165       $module_by_prefix{$mod_prefix} = $file_name;
166       $module_err_prefixes{$file_name} = $mod_prefix;
167       $module_last_err_used{$file_name} = $last_err;
168     } else {
169       print "ERROR: Fail to parse sm_error_codes: $errDef\n";
170       exit 3;
171     }
172   }
173 }
174
175 # do a file by file read into memory so we can tweek it:
176 foreach $file_name (@ARGV) {
177         print "- $file_name ----------------------------------------------------\n";
178    # first step is to run indent
179          $res=`osm_indent $file_name`;
180
181     open(INFILE, "<$file_name") || die("Fail to open $file_name");
182     @LINES = <INFILE>;
183     close(INFILE);
184     $any_fix = 0;
185     $needed_fixing = 0;
186     $need_indentation = 0;
187
188   LINE: for ($line_num = 0; $line_num <scalar(@LINES); $line_num++) {
189       $line = $LINES[$line_num];
190       $_ = $line;
191
192       # Skip C single line C style comments
193       # This line must come before the multi-line C comment check!
194       if( /\/\*.*\*\// )
195       {
196           $in_c_comment = 0;
197           next LINE;
198       }
199
200       # skip multi-line C style comments
201       if( /\/\*/ )
202       {
203           $in_c_comment = 1;
204           next LINE;
205       }
206
207       # end skipping of multi-line C style comments
208       if( /\*\// )
209       {
210           $in_c_comment = 0;
211           next LINE;
212       }
213
214       # We're still in a C comment, so ignore input
215       if( $in_c_comment == 1 )
216       {
217         next LINE;
218       }
219
220
221       # Error on C++ style comment lines
222       if( /\/\// )
223       {
224         print "C++ style comment on $file_name $line_num\n";
225         $needed_fixing++;
226         if ($fix_mode) {
227           $line =~ s=\/\/(.*)$=/* \1 */=;
228           if (confirm_change($line, $LINES[$line_num])) {
229             $LINES[$line_num] = $line;
230             $any_fix++;
231           }
232           $any_fix++;
233         }
234       }
235
236       # check for lines with trailing spaces:
237       if (/[ \t]+$/) {
238         $needed_fixing++;
239         if ($fix_mode) {
240           $line =~ s/\s+$/\n/;
241           if (confirm_change($line, $LINES[$line_num])) {
242             $LINES[$line_num] = $line;
243             $any_fix++;
244           }
245           $any_fix++;
246         }
247       }
248
249       # check for bad PRIx64 usage
250       # It's a common mistake to forget the % before the PRIx64
251       if (/[^%0-9][0-9]*\"\s*PRIx64/ ) {
252         $needed_fixing++;
253         print "No % sign before PRx64!!: $file_name $line_num\n";
254         if ($fix_mode) {
255           $line =~ s/([0-9]*)\"\s*PRIx64/%$1\" PRIx64/;
256           if (confirm_change($line, $LINES[$line_num])) {
257             $LINES[$line_num] = $line;
258             $any_fix++;
259           }
260         }
261       }
262
263       # This simple script doesn't handle checking PRIx64 usage
264       # when PRIx64 starts the line.  Just give a warning.
265       if( /^\s*PRIx64/ )
266       {
267         $needed_fixing++;
268         print "Warning: PRIx64 at start of line.  $file_name $line_num\n";
269 #        if ($fix_mode) {
270 #          print "Fatal: can not auto fix\n";
271 #          exit 1;
272 #        }
273       }
274
275       # Attempt to locate function names.
276       # Function names must start on the beginning of the line.
277       if( /^(\w+)\s*\(/ )
278       {
279         $current_func = $1;
280         if( $verbose == 1 )
281           {
282             print "Processing $file_name: $current_func\n";
283           }
284       }
285
286       # Attempt to find OSM_LOG_ENTER entries.
287       # When found, verify that the function name provided matches
288       # the actual function.
289       if( /OSM_LOG_ENTER\s*\(\s*([\-\.\>\w]+)\s*,\s*(\w+)\s*\)/ ) {
290         $log_func = $2;
291         if( $current_func ne $log_func ) {
292           printf "MISMATCH!! $file_name $line_num: $current_func != $log_func\n";
293           $needed_fixing++;
294           if ($fix_mode) {
295             $line =~
296               s/OSM_LOG_ENTER\s*\(\s*([\-\.\>\w]+)\s*,\s*(\w+)\s*\)/OSM_LOG_ENTER( $1, $current_func )/;
297             if (confirm_change($line, $LINES[$line_num])) {
298               $LINES[$line_num] = $line;
299               $any_fix++;
300             }
301           }
302         }
303       }
304
305       # Check for non-conforming log statements.
306       # Log statements must not start the log string on the same line
307       # as the osm_log function itself.
308       # Watch out for the #include "osm_log.h" statement as a false positive.
309       if (/osm_log\s*\(.*OSM_.*\"/ ) {
310         if (/Format Waved/) {
311           print "Skipping log format waiver at $file_name $line_num\n";
312         } else {
313           print "NON-CONFORMING LOG STATEMENT!! $file_name $line_num\n";
314           $needed_fixing++;
315           if ($fix_mode) {
316             print "Fatal: can not auto fix\n";
317             exit 1;
318           }
319         }
320       }
321
322       # Attempt to find osm_log entries.
323       if( /^\s*\"(\w+):/ )
324       {
325           $log_func = $1;
326           if( $current_func ne $log_func )
327           {
328               print "MISMATCHED LOG FUNCTION!! $file_name $line_num: $current_func != $log_func\n";
329               $needed_fixing++;
330               if ($fix_mode) {
331                   $line =~
332                       s/^(\s*)\"(\w+):/$1\"$current_func:/;
333                   if (confirm_change($line, $LINES[$line_num])) {
334                       $LINES[$line_num] = $line;
335                       $any_fix++;
336                   }
337               }
338           }
339       }
340
341       # Error logging must look like 'ERR 1234:'
342       # The upper two digits are error range assigned to that module.
343       # The lower two digits are the error code itself.
344       # Error codes are in hexadecimal.
345       if( /ERR(\s+)([0-9a-fA-F]{2})([0-9a-fA-F]{2})(..)/ )
346       {
347           # track any error for this exp:
348           $exp_err = 0;
349
350           # the parsed prefix and err code:
351           ($found_prefix,$found_code) = ($2,$3);
352
353           # Check if we already established the error prefix for this module
354           $err_prefix = $module_err_prefixes{$file_name};
355
356           # err prefix is not available for this file
357           if ( ! $err_prefix ) {
358             # make sure no other file uses this prefix:
359             if ($module_by_prefix{$found_prefix}) {
360               # some other file uses that prefix:
361
362               # two modes: either use a new one or abort
363               if ($re_assign_err_prefix) {
364                 # scan the available module prefixes for an empty one:
365                 $found = 0;
366                 for ($new_prefix_idx = 1; $found == 0; $new_prefix_idx++) {
367                   $prefix = sprintf("%02X", $new_prefix_idx);
368                   if (!defined($module_by_prefix{$prefix})) {
369                     $module_err_prefixes{$file_name} = $prefix;
370                     $module_by_prefix{$prefix} = $file_name;
371                     $found = 1;
372                   }
373                   $exp_err = 1;
374                 }
375               } else {
376                 print "Fatal: File $module_by_prefix{$2} already uses same prefix:$2 used by: $file_name (line=$line_num)\n";
377                 exit 1;
378               }
379             } else {
380               # the prefix found is unused:
381
382               # Create a new prefix for this module.
383               $module_err_prefixes{$file_name} = $found_prefix;
384               $module_by_prefix{$found_prefix} = $file_name;
385               $err_prefix = $found_prefix;
386             }
387           } else {
388             # we already have a prefix for this file
389
390             if( $err_prefix ne $found_prefix )
391             {
392               $needed_fixing++;
393               print "BAD ERR RANGE IN LOG ENTRY!! $file_name $line_num: $current_func\n";
394               print "\tExpected $err_prefix but found $found_prefix\n";
395               $exp_err = 1;
396             }
397           }
398
399           # now check for code duplicates
400           $err_base = $module_err_bases{$found_code};
401           if( $err_base ) {
402             $needed_fixing++;
403             print "DUPLICATE ERR NUMBER IN LOG ENTRY!! $file_name $line_num: $current_func: $3\n";
404             print "\tPrevious use on line $err_base.\n";
405
406             # use the last error code for this module:
407             $module_last_err_used{$file_name}++;
408             $err_code = sprintf("%02X", $module_last_err_used{$file_name});
409             print "\tUsing new err code:0x$err_code ($module_last_err_used{$file_name})\n";
410             $module_err_bases{$err_code} = $line_num;
411             $exp_err = 1;
412           } else {
413             # Add this error code to the list used by this module
414             # The data stored in the line number on which it is used.
415             $module_err_bases{$found_code} = $line_num;
416             # track the last code used
417             $err_code_num = eval("0x$found_code");
418             if ($module_last_err_used{$file_name} < $err_code_num) {
419               $module_last_err_used{$file_name} = $err_code_num;
420             }
421             $err_code = $found_code;
422
423             if( $verbose > 1 ) {
424               print "Adding new error: $err_prefix$found_code in $file_name.\n";
425             }
426           }
427
428           if( $4 ne ": " ) {
429             $needed_fixing++;
430             print "MALFORMED LOG STATEMENT!!  NEEDS ': ' $file_name $line_num\n";
431             $exp_err = 1;
432           }
433
434           if( $1 ne " " )
435           {
436             $needed_fixing++;
437             print "USE ONLY 1 SPACE AFTER ERR!! $file_name $line_num\n";
438             $exp_err = 1;
439           }
440
441           if ($exp_err && $fix_mode) {
442               $line =~
443                   s/ERR(\s+)([0-9a-fA-F]{2})([0-9a-fA-F]{2})([^\"]*\")/ERR ${err_prefix}$err_code: \" /;
444               if (confirm_change($line, $LINES[$line_num])) {
445                   $LINES[$line_num] = $line;
446                   $any_fix++;
447               }
448           }
449       }
450
451       # verify expected use of sizeof() with pointers
452       if( /sizeof\s*\(\s*[h|p]_[^-]+\)/ )
453       {
454           print "SUSPICIOUS USE OF SIZEOF(), DO YOU NEED AN '*' $file_name $line_num\n";
455           $needed_fixing++;
456           if ($fix_mode) {
457               $line =~
458                   s/sizeof\s*\(\s*([h|p])_/sizeof \(*$1_/;
459               if (confirm_change($line, $LINES[$line_num])) {
460                   $LINES[$line_num] = $line;
461                   $any_fix++;
462               }
463           }
464       }
465    }
466
467     # reset the base error value, since each module can
468     # repeat this range.
469     %module_err_bases = ();
470
471     # if any fix write out the fixed file:
472     if ($any_fix) {
473       open(OF,">$file_name.fix");
474       print OF @LINES;
475       close(OF);
476     } elsif ($needed_fixing) {
477       print "Found $needed_fixing Errors on file: $file_name\n";
478     }
479 }
480
481 # write out the error codes.
482 # module_by_prefix
483 # module_err_prefixes
484 # module_last_err_used
485 open(ERRS,">osm_error_codes");
486 foreach $fn (sort(keys(%module_err_prefixes))) {
487   print ERRS "$fn $module_err_prefixes{$fn} $module_last_err_used{$fn}\n";
488 }
489 close(ERRS);
490
491 sub osm_check_usage
492 {
493     print "Usage:\n";
494     print "osm_check.pl [-v|V] [-f|--fix] [-c|--confirm] [-r|--re-assign-mod-err-prefix] <file list>\n";
495     print "[-v|V] - enable verbose mode.\n";
496     print "[-f|--fix] - enable auto fix mode.\n";
497     print "[-c|--confirm] - enable manual confirmation mode.\n";
498     print "[-r|--re-assign-mod-err-prefix] - enables re-assign error prefixes if the file does not have one.\n";
499 }
500
501 sub confirm_change {
502     local ($line, $orig_line) = @_;
503     if ($confirm_mode) {
504         print "In Line:".($line_num + 1)."\n";
505         print "From: ${orig_line}To:   ${line}Ok [y] ?";
506         $| = 1;
507         $ans = <STDIN>;
508         chomp $ans;
509
510         if ($ans && $ans ne "y") {
511             return 0;
512         }
513     } else {
514         print "From: ${orig_line}To:   ${line}";
515     }
516     return 1;
517 }