[perl #2726] Prototype is not applied until BLOCK is defined
authorPeter Martini <PeterCMartini@GMail.com>
Tue, 6 Aug 2013 07:16:35 +0000 (03:16 -0400)
committerFather Chrysostomos <sprout@cpan.org>
Tue, 6 Aug 2013 12:53:07 +0000 (05:53 -0700)
In the case of a sub definition with a prototype, the prototype
is not attached to the sub until after the body is completely
defined.  This means that any sub which calls itself will
not honor its prototype unless the prototype was declared prior to
the sub's definition.  Whether or not this behavior is desirable is
debatable, but its far too late to do anything about it other than
document it and test to make sure it doesn't change.

pod/perlsub.pod
t/comp/proto.t

index ff5feb5..455fa23 100644 (file)
@@ -1338,6 +1338,16 @@ C<func()> now gets passed in a C<1>; that is, the number of elements
 in C<@foo>.  And the C<split> gets called in scalar context so it
 starts scribbling on your C<@_> parameter list.  Ouch!
 
+If a sub has both a PROTO and a BLOCK, the prototype is not applied
+until after the BLOCK is completely defined.  This means that a recursive
+function with a prototype has to be predeclared for the prototype to take
+effect, like so:
+
+       sub foo($$);
+       sub foo($$) {
+               foo 1, 2;
+       }
+
 This is all very powerful, of course, and should be used only in moderation
 to make the world a better place.
 
index 947a232..47ebf74 100644 (file)
@@ -18,7 +18,7 @@ BEGIN {
 # strict
 use strict;
 
-print "1..199\n";
+print "1..201\n";
 
 my $i = 1;
 
@@ -559,6 +559,13 @@ print "ok ", $i++, " star3 STDERR\n";
 print "not " unless eval 'star4 STDERR; 1';
 print "ok ", $i++, " star4 STDERR\n";
 
+# [perl #2726]
+# Test that prototype binding is late
+print "not " unless eval 'sub l564($){ l564(); } 1';
+print "ok ", $i++, " prototype checking not done within initial definition\n";
+print "not " if eval 'sub l566($); sub l566($){ l566(); } 1';
+print "ok ", $i++, " prototype checking done if sub pre-declared\n";
+
 # test scalarref prototype
 sub sreftest (\$$) {
     print "not " unless ref $_[0];