line |
stmt |
bran |
cond |
sub |
pod |
time |
code |
1
|
|
|
|
|
|
|
package Module::CAPIMaker; |
2
|
|
|
|
|
|
|
|
3
|
|
|
|
|
|
|
our $VERSION = '0.01'; |
4
|
|
|
|
|
|
|
|
5
|
1
|
|
|
1
|
|
33455
|
use strict; |
|
1
|
|
|
|
|
2
|
|
|
1
|
|
|
|
|
32
|
|
6
|
1
|
|
|
1
|
|
4
|
use warnings; |
|
1
|
|
|
|
|
2
|
|
|
1
|
|
|
|
|
25
|
|
7
|
|
|
|
|
|
|
|
8
|
1
|
|
|
1
|
|
1008
|
use Text::Template; |
|
1
|
|
|
|
|
3857
|
|
|
1
|
|
|
|
|
57
|
|
9
|
1
|
|
|
1
|
|
10
|
use File::Spec; |
|
1
|
|
|
|
|
1
|
|
|
1
|
|
|
|
|
28
|
|
10
|
1
|
|
|
1
|
|
862
|
use POSIX qw(strftime); |
|
1
|
|
|
|
|
6973
|
|
|
1
|
|
|
|
|
8
|
|
11
|
|
|
|
|
|
|
|
12
|
1
|
|
|
1
|
|
1060
|
use Exporter qw(import); |
|
1
|
|
|
|
|
2
|
|
|
1
|
|
|
|
|
59
|
|
13
|
|
|
|
|
|
|
our @EXPORT = qw(make_c_api); |
14
|
|
|
|
|
|
|
|
15
|
1
|
|
|
1
|
|
742
|
use Module::CAPIMaker::Template::Module_H; |
|
1
|
|
|
|
|
4
|
|
|
1
|
|
|
|
|
32
|
|
16
|
1
|
|
|
1
|
|
473
|
use Module::CAPIMaker::Template::Module_C; |
|
1
|
|
|
|
|
3
|
|
|
1
|
|
|
|
|
25
|
|
17
|
1
|
|
|
1
|
|
514
|
use Module::CAPIMaker::Template::Sample_XS; |
|
1
|
|
|
|
|
3
|
|
|
1
|
|
|
|
|
30
|
|
18
|
1
|
|
|
1
|
|
572
|
use Module::CAPIMaker::Template::C_API_H; |
|
1
|
|
|
|
|
5
|
|
|
1
|
|
|
|
|
1977
|
|
19
|
|
|
|
|
|
|
|
20
|
|
|
|
|
|
|
sub new { |
21
|
0
|
|
|
0
|
0
|
|
my $class = shift; |
22
|
0
|
|
|
|
|
|
my %config = @_; |
23
|
0
|
|
|
|
|
|
my $self = { config => \%config, |
24
|
|
|
|
|
|
|
function => {}, |
25
|
|
|
|
|
|
|
data => {} |
26
|
|
|
|
|
|
|
}; |
27
|
|
|
|
|
|
|
|
28
|
0
|
|
0
|
|
|
|
$config{c_api_decl_filename} //= 'c_api.decl'; |
29
|
|
|
|
|
|
|
|
30
|
0
|
|
|
|
|
|
bless $self, $class; |
31
|
|
|
|
|
|
|
} |
32
|
|
|
|
|
|
|
|
33
|
|
|
|
|
|
|
sub load_decl { |
34
|
0
|
|
|
0
|
0
|
|
my $self = shift; |
35
|
0
|
|
|
|
|
|
my $config = $self->{config}; |
36
|
0
|
|
|
|
|
|
my $fn = $config->{c_api_decl_filename}; |
37
|
0
|
0
|
|
|
|
|
open my $fh, '<', $fn or die "Unable to open $fn: $!\n"; |
38
|
0
|
|
|
|
|
|
while (<$fh>) { |
39
|
0
|
|
|
|
|
|
chomp; |
40
|
0
|
|
|
|
|
|
s/^\s+//; s/\s+$//; |
|
0
|
|
|
|
|
|
|
41
|
0
|
0
|
|
|
|
|
next if /^(?:#.*)?$/; |
42
|
0
|
|
|
|
|
|
while (s/\s*\\$/ /) { |
43
|
0
|
|
|
|
|
|
my $next = <$fh>; |
44
|
0
|
|
|
|
|
|
chomp $next; |
45
|
0
|
|
|
|
|
|
$next =~ s/^\s+//; $next =~ s/\s+$//; |
|
0
|
|
|
|
|
|
|
46
|
0
|
|
|
|
|
|
$_ .= $next; |
47
|
|
|
|
|
|
|
} |
48
|
0
|
0
|
|
|
|
|
if (my ($k, $v) = /^(\w+)\s*=\s*(.*)/) { |
|
|
0
|
|
|
|
|
|
49
|
0
|
0
|
|
|
|
|
if (my ($mark) = $v =~ /^<<\s*(\w+)$/) { |
50
|
0
|
|
|
|
|
|
$v = ''; |
51
|
0
|
|
|
|
|
|
while (1) { |
52
|
0
|
|
|
|
|
|
my $line = <$fh>; |
53
|
0
|
0
|
|
|
|
|
defined $line or die "Ending token '$mark' missing at $fn line $.\n"; |
54
|
0
|
0
|
|
|
|
|
last if $line =~ /^$mark$/; |
55
|
0
|
|
|
|
|
|
$v .= $line; |
56
|
|
|
|
|
|
|
} |
57
|
|
|
|
|
|
|
} |
58
|
0
|
|
|
|
|
|
$self->{config}{$k} = $v; |
59
|
|
|
|
|
|
|
} |
60
|
|
|
|
|
|
|
elsif (/^((?:\w+\b\s*(?:\*+\s*)?)*)(\w+)\s*\(\s*(.*?)\s*\)$/) { |
61
|
0
|
|
|
|
|
|
my $args = $3; |
62
|
0
|
|
|
|
|
|
my %f = ( decl => $_, |
63
|
|
|
|
|
|
|
type => $1, |
64
|
|
|
|
|
|
|
name => $2, |
65
|
|
|
|
|
|
|
args => $args ); |
66
|
0
|
|
|
|
|
|
$self->{function}{$2} = \%f; |
67
|
|
|
|
|
|
|
|
68
|
0
|
0
|
|
|
|
|
if ($f{pTHX} = $args =~ s/^pTHX(?:_\s+|$)//) { |
69
|
0
|
|
|
|
|
|
$args =~ s/^void$//; |
70
|
0
|
|
|
|
|
|
my @args = split /\s*,\s*/, $args; |
71
|
|
|
|
|
|
|
# warn "args |$args| => |". join('-', @args) . "|"; |
72
|
0
|
|
|
|
|
|
$f{macro_args} = join(', ', ('a'..'z')[0..$#args]); |
73
|
0
|
0
|
|
|
|
|
$f{call_args} = (@args ? 'aTHX_ (' . join('), (', ('a'..'z')[0..$#args]) .')' : 'aTHX'); |
74
|
|
|
|
|
|
|
} |
75
|
|
|
|
|
|
|
|
76
|
|
|
|
|
|
|
} |
77
|
|
|
|
|
|
|
else { |
78
|
0
|
|
|
|
|
|
die "Invalid declaration at $fn line $.\n"; |
79
|
|
|
|
|
|
|
} |
80
|
|
|
|
|
|
|
} |
81
|
|
|
|
|
|
|
} |
82
|
|
|
|
|
|
|
|
83
|
|
|
|
|
|
|
sub check_config { |
84
|
0
|
|
|
0
|
0
|
|
my $self = shift; |
85
|
0
|
|
|
|
|
|
my $config = $self->{config}; |
86
|
|
|
|
|
|
|
|
87
|
0
|
|
|
|
|
|
my $module_name = $config->{module_name}; |
88
|
0
|
0
|
|
|
|
|
die "module_name declaration missing from $config->{decl_filename}\n" |
89
|
|
|
|
|
|
|
unless defined $module_name; |
90
|
|
|
|
|
|
|
|
91
|
0
|
0
|
|
|
|
|
die "Invalid value for module_name ($module_name)\n" |
92
|
|
|
|
|
|
|
unless $module_name =~ /^\w+(?:::\w+)*$/; |
93
|
|
|
|
|
|
|
|
94
|
0
|
|
0
|
|
|
|
my $c_module_name = $config->{c_module_name} //= do { my $cmn = lc $module_name; |
|
0
|
|
|
|
|
|
|
95
|
0
|
|
|
|
|
|
$cmn =~ s/\W+/_/g; |
96
|
0
|
|
|
|
|
|
$cmn }; |
97
|
0
|
0
|
|
|
|
|
die "Invalid value for c_module_name ($c_module_name)\n" |
98
|
|
|
|
|
|
|
unless $c_module_name =~ /^\w+$/; |
99
|
|
|
|
|
|
|
|
100
|
0
|
|
0
|
|
|
|
$config->{author} //= 'Unknown'; |
101
|
0
|
|
0
|
|
|
|
$config->{min_version} //= 1; |
102
|
0
|
|
0
|
|
|
|
$config->{max_version} //= 1; |
103
|
|
|
|
|
|
|
|
104
|
0
|
0
|
|
|
|
|
die "Invalid version declaration, min_version ($config->{min_version}) > max_version ($config->{max_version})\n" |
105
|
|
|
|
|
|
|
if $config->{max_version} < $config->{min_version}; |
106
|
|
|
|
|
|
|
|
107
|
0
|
|
0
|
|
|
|
$config->{required_version} //= $config->{max_version}; |
108
|
0
|
|
0
|
|
|
|
$config->{module_version} //= '0'; |
109
|
0
|
|
|
|
|
|
$config->{capimaker_version} = $VERSION; |
110
|
|
|
|
|
|
|
|
111
|
0
|
|
|
|
|
|
$config->{now} = strftime("%F %T", localtime); |
112
|
|
|
|
|
|
|
|
113
|
0
|
|
0
|
|
|
|
$config->{client_dir} //= 'c_api_client'; |
114
|
|
|
|
|
|
|
|
115
|
0
|
|
0
|
|
|
|
$config->{module_c_filename} //= "perl_$c_module_name.c"; |
116
|
0
|
|
0
|
|
|
|
$config->{module_h_filename} //= "perl_$c_module_name.h"; |
117
|
0
|
|
0
|
|
|
|
$config->{sample_xs_filename} //= "sample.xs"; |
118
|
0
|
|
0
|
|
|
|
$config->{c_api_h_filename} //= "c_api.h"; |
119
|
|
|
|
|
|
|
|
120
|
0
|
|
0
|
|
|
|
$config->{module_h_barrier} //= do { my $ib = "$config->{module_h_filename}_INCLUDED"; |
|
0
|
|
|
|
|
|
|
121
|
0
|
|
|
|
|
|
$ib =~ s/\W+/_/g; |
122
|
0
|
|
|
|
|
|
uc $ib }; |
123
|
0
|
0
|
|
|
|
|
die "Invalid value for module_h_barrier ($config->{module_h_barrier})\n" |
124
|
|
|
|
|
|
|
unless $config->{module_h_barrier} =~ /^\w+$/; |
125
|
|
|
|
|
|
|
|
126
|
0
|
|
0
|
|
|
|
$config->{c_api_h_barrier} //= do { my $ib = "$config->{c_api_h_filename}_INCLUDED"; |
|
0
|
|
|
|
|
|
|
127
|
0
|
|
|
|
|
|
$ib =~ s/\W+/_/g; |
128
|
0
|
|
|
|
|
|
uc $ib }; |
129
|
0
|
0
|
|
|
|
|
die "Invalid value for c_api_h_barrier ($config->{c_api_h_barrier})\n" |
130
|
|
|
|
|
|
|
unless $config->{c_api_h_barrier} =~ /^\w+$/; |
131
|
|
|
|
|
|
|
|
132
|
|
|
|
|
|
|
|
133
|
0
|
|
0
|
|
|
|
$config->{$_} //= '' for qw(export_prefix |
134
|
|
|
|
|
|
|
module_c_beginning |
135
|
|
|
|
|
|
|
module_c_end |
136
|
|
|
|
|
|
|
module_h_beginning |
137
|
|
|
|
|
|
|
module_h_end); |
138
|
|
|
|
|
|
|
} |
139
|
|
|
|
|
|
|
|
140
|
|
|
|
|
|
|
sub gen_file { |
141
|
0
|
|
|
0
|
0
|
|
my ($self, $template, $dir, $save_as) = @_; |
142
|
0
|
|
|
|
|
|
my $config = $self->{config}; |
143
|
0
|
0
|
|
|
|
|
system mkdir => -p => $dir unless -d $dir; # FIX ME! |
144
|
0
|
|
|
|
|
|
$save_as = File::Spec->rel2abs(File::Spec->join($dir, $save_as)); |
145
|
0
|
0
|
|
|
|
|
open my $fh, '>', $save_as or die "Unable to create $save_as: $!\n"; |
146
|
0
|
|
|
|
|
|
local $Text::Template::ERROR; |
147
|
0
|
0
|
|
|
|
|
my $tt = Text::Template->new(TYPE => (ref $template ? 'ARRAY' : 'FILE'), |
148
|
|
|
|
|
|
|
SOURCE => $template, |
149
|
|
|
|
|
|
|
DELIMITERS => ['<%', '%>'] ); |
150
|
0
|
|
|
|
|
|
$tt->fill_in(HASH => { %$config, function => $self->{function} }, |
151
|
|
|
|
|
|
|
OUTPUT => $fh); |
152
|
0
|
0
|
|
|
|
|
warn "Some error happened while generating $save_as: $Text::Template::ERROR\n" |
153
|
|
|
|
|
|
|
if $Text::Template::ERROR; |
154
|
|
|
|
|
|
|
} |
155
|
|
|
|
|
|
|
|
156
|
|
|
|
|
|
|
sub gen_all { |
157
|
0
|
|
|
0
|
0
|
|
my $self = shift; |
158
|
0
|
|
|
|
|
|
my $config = $self->{config}; |
159
|
0
|
|
0
|
|
|
|
$self->gen_file($config->{module_c_template_filename} // \@Module::CAPIMaker::Template::Module_C::template, |
160
|
|
|
|
|
|
|
$config->{client_dir}, |
161
|
|
|
|
|
|
|
$config->{module_c_filename}); |
162
|
0
|
|
0
|
|
|
|
$self->gen_file($config->{module_h_template_filename} // \@Module::CAPIMaker::Template::Module_H::template, |
163
|
|
|
|
|
|
|
$config->{client_dir}, |
164
|
|
|
|
|
|
|
$config->{module_h_filename}); |
165
|
0
|
|
0
|
|
|
|
$self->gen_file($config->{sample_xs_template_filename} // \@Module::CAPIMaker::Template::Sample_XS::template, |
166
|
|
|
|
|
|
|
$config->{client_dir}, |
167
|
|
|
|
|
|
|
$config->{sample_xs_filename}); |
168
|
0
|
|
0
|
|
|
|
$self->gen_file($config->{c_api_h_template_filename} // \@Module::CAPIMaker::Template::C_API_H::template, |
169
|
|
|
|
|
|
|
'.', |
170
|
|
|
|
|
|
|
$config->{c_api_h_filename}); |
171
|
|
|
|
|
|
|
} |
172
|
|
|
|
|
|
|
|
173
|
|
|
|
|
|
|
sub make_c_api { |
174
|
0
|
|
|
0
|
0
|
|
my %args; |
175
|
0
|
|
|
|
|
|
for (@ARGV) { |
176
|
0
|
0
|
|
|
|
|
/^\s*(\w+)\s*=\s*(.*?)\s*$/ |
177
|
|
|
|
|
|
|
or die "Bad argument '$_'\n"; |
178
|
0
|
|
|
|
|
|
$args{$1} = $2; |
179
|
|
|
|
|
|
|
} |
180
|
0
|
|
|
|
|
|
my $mcm = Module::CAPIMaker->new(%args); |
181
|
0
|
|
|
|
|
|
$mcm->load_decl; |
182
|
0
|
|
|
|
|
|
$mcm->check_config; |
183
|
0
|
|
|
|
|
|
$mcm->gen_all; |
184
|
|
|
|
|
|
|
} |
185
|
|
|
|
|
|
|
|
186
|
|
|
|
|
|
|
1; |
187
|
|
|
|
|
|
|
__END__ |