-
Notifications
You must be signed in to change notification settings - Fork 138
/
c_operator.t
115 lines (88 loc) · 2.89 KB
/
c_operator.t
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
#! perl
# Copyright (C) 2006-2009, Parrot Foundation.
use strict;
use warnings;
use lib qw( . lib ../lib ../../lib );
use Test::More tests => 1;
use Parrot::Distribution;
use Pod::Simple;
=head1 NAME
t/codingstd/c_operator.t - checks spacing around operators in C source
=head1 SYNOPSIS
# test all files
% prove t/codingstd/c_operator.t
# test specific files
% perl t/codingstd/c_operator.t src/foo.c include/parrot/bar.h
=head1 DESCRIPTION
Checks that all C language source files have the proper use of spacing
around operators.
=head1 AUTHOR
Paul Cochrane <paultcochrane at gmail dot com>
=head1 SEE ALSO
L<docs/pdds/pdd07_codingstd.pod>
=cut
my $DIST = Parrot::Distribution->new;
my @files = @ARGV ? <@ARGV> : $DIST->get_c_language_files();
check_operators(@files);
exit;
sub strip_pod {
my $buf = shift;
my $parser = Pod::Simple->new();
my $non_pod_buf;
$parser->output_string( \$non_pod_buf );
# set up a code handler to get at the non-pod
# thanks to Thomas Klausner's Pod::Strip for the inspiration
$parser->code_handler(
sub {
print {$_[2]{output_fh}} $_[0], "\n";
});
$parser->parse_string_document( $buf );
return $non_pod_buf;
}
sub check_operators {
my %comma_space;
foreach my $file (@_) {
my $path = @ARGV ? $file : $file->path();
# skip lex files
next if $path =~ m/\.l$/;
my $buf = $DIST->slurp($path);
# only strip pod from .ops files
if ( $path =~ m/\.ops$/ ) {
$buf = strip_pod($buf);
}
# strip ', ", and C comments #'
$buf =~ s{ (?:
(?: (') (?: \\\\ | \\' | [^'] )* (') ) # rm ' string #'
| (?: (") (?: \\\\ | \\" | [^"] )* (") ) # rm " string #"
| /(\*) .*? (\*)/ # rm C comment
)
}{defined $1 ? "$1$2" : defined $3 ? "$3$4" : "$5$6"}egsx;
my @lines = split( /\n/, $buf );
$comma_space{$path} = [];
for (my $i=0; $i <= $#lines; $i++) {
# after a comma there should be one space or a newline
if ( $lines[$i] =~ m{ ( (?:,) (?! \s ) (?= .+) ) }gx ) {
push @{ $comma_space{$path} }, $lines[$i];
}
}
}
## L<PDD07/Code Formatting"there should be one space or a newline after a comma">/
my @comma_space_files;
for my $path ( sort keys %comma_space ) {
if (my $count = scalar @{ $comma_space{$path} }) {
push @comma_space_files, <<"END_ERROR";
$path [$count line@{[ ($count >1) ? 's': '' ]}] at :
@{[ join("\n--\n", @{$comma_space{$path}}) ]}
END_ERROR
}
}
is(join("\n",@comma_space_files),
"",
"there should be one space or a newline after a comma");
}
# Local Variables:
# mode: cperl
# cperl-indent-level: 4
# fill-column: 100
# End:
# vim: expandtab shiftwidth=4: