b24e7fce |
1 | #!/usr/bin/perl |
2 | |
3 | # Transform K&R C function definitions into ANSI equivalent. |
4 | # |
5 | # Author: Paul Marquess |
6 | # Version: 1.0 |
7 | # Date: 3 October 2006 |
8 | |
9 | # TODO |
10 | # |
11 | # Asumes no function pointer parameters. unless they are typedefed. |
12 | # Assumes no literal strings that look like function definitions |
13 | # Assumes functions start at the beginning of a line |
14 | |
15 | use strict; |
16 | use warnings; |
17 | |
18 | local $/; |
19 | $_ = <>; |
20 | |
21 | my $sp = qr{ \s* (?: /\* .*? \*/ )? \s* }x; # assume no nested comments |
22 | |
23 | my $d1 = qr{ $sp (?: [\w\*\s]+ $sp)* $sp \w+ $sp [\[\]\s]* $sp }x ; |
24 | my $decl = qr{ $sp (?: \w+ $sp )+ $d1 }xo ; |
25 | my $dList = qr{ $sp $decl (?: $sp , $d1 )* $sp ; $sp }xo ; |
26 | |
27 | |
28 | while (s/^ |
29 | ( # Start $1 |
30 | ( # Start $2 |
31 | .*? # Minimal eat content |
32 | ( ^ \w [\w\s\*]+ ) # $3 -- function name |
33 | \s* # optional whitespace |
34 | ) # $2 - Matched up to before parameter list |
35 | |
36 | \( \s* # Literal "(" + optional whitespace |
37 | ( [^\)]+ ) # $4 - one or more anythings except ")" |
38 | \s* \) # optional whitespace surrounding a Literal ")" |
39 | |
40 | ( (?: $dList )+ ) # $5 |
41 | |
42 | $sp ^ { # literal "{" at start of line |
43 | ) # Remember to $1 |
44 | //xsom |
45 | ) |
46 | { |
47 | my $all = $1 ; |
48 | my $prefix = $2; |
49 | my $param_list = $4 ; |
50 | my $params = $5; |
51 | |
52 | StripComments($params); |
53 | StripComments($param_list); |
54 | $param_list =~ s/^\s+//; |
55 | $param_list =~ s/\s+$//; |
56 | |
57 | my $i = 0 ; |
58 | my %pList = map { $_ => $i++ } |
59 | split /\s*,\s*/, $param_list; |
60 | my $pMatch = '(\b' . join('|', keys %pList) . '\b)\W*$' ; |
61 | |
62 | my @params = split /\s*;\s*/, $params; |
63 | my @outParams = (); |
64 | foreach my $p (@params) |
65 | { |
66 | if ($p =~ /,/) |
67 | { |
68 | my @bits = split /\s*,\s*/, $p; |
69 | my $first = shift @bits; |
70 | $first =~ s/^\s*//; |
71 | push @outParams, $first; |
72 | $first =~ /^(\w+\s*)/; |
73 | my $type = $1 ; |
74 | push @outParams, map { $type . $_ } @bits; |
75 | } |
76 | else |
77 | { |
78 | $p =~ s/^\s+//; |
79 | push @outParams, $p; |
80 | } |
81 | } |
82 | |
83 | |
84 | my %tmp = map { /$pMatch/; $_ => $pList{$1} } |
85 | @outParams ; |
86 | |
87 | @outParams = map { " $_" } |
88 | sort { $tmp{$a} <=> $tmp{$b} } |
89 | @outParams ; |
90 | |
91 | print $prefix ; |
92 | print "(\n" . join(",\n", @outParams) . ")\n"; |
93 | print "{" ; |
94 | |
95 | } |
96 | |
97 | # Output any trailing code. |
98 | print ; |
99 | exit 0; |
100 | |
101 | |
102 | sub StripComments |
103 | { |
104 | |
105 | no warnings; |
106 | |
107 | # Strip C & C++ coments |
108 | # From the perlfaq |
109 | $_[0] =~ |
110 | |
111 | s{ |
112 | /\* ## Start of /* ... */ comment |
113 | [^*]*\*+ ## Non-* followed by 1-or-more *'s |
114 | ( |
115 | [^/*][^*]*\*+ |
116 | )* ## 0-or-more things which don't start with / |
117 | ## but do end with '*' |
118 | / ## End of /* ... */ comment |
119 | |
120 | | ## OR C++ Comment |
121 | // ## Start of C++ comment // |
122 | [^\n]* ## followed by 0-or-more non end of line characters |
123 | |
124 | | ## OR various things which aren't comments: |
125 | |
126 | ( |
127 | " ## Start of " ... " string |
128 | ( |
129 | \\. ## Escaped char |
130 | | ## OR |
131 | [^"\\] ## Non "\ |
132 | )* |
133 | " ## End of " ... " string |
134 | |
135 | | ## OR |
136 | |
137 | ' ## Start of ' ... ' string |
138 | ( |
139 | \\. ## Escaped char |
140 | | ## OR |
141 | [^'\\] ## Non '\ |
142 | )* |
143 | ' ## End of ' ... ' string |
144 | |
145 | | ## OR |
146 | |
147 | . ## Anything other char |
148 | [^/"'\\]* ## Chars which doesn't start a comment, string or escape |
149 | ) |
150 | }{$2}gxs; |
151 | |
152 | } |