This is a live mirror of the Perl 5 development currently hosted at https://github.com/perl/perl5
refer Time::HiRes RT queue to perl core
[perl5.git] / dist / Time-HiRes / Makefile.PL
1 #!/usr/bin/perl
2 #
3 # In general we trust %Config, but for nanosleep() this trust
4 # may be misplaced (it may be linkable but not really functional).
5 # Use $ENV{FORCE_NANOSLEEP_SCAN} to force rescanning whether there
6 # really is hope.
7
8 { use 5.006; }
9
10 use Config;
11 use ExtUtils::MakeMaker;
12 use strict;
13
14 my $VERBOSE = $ENV{VERBOSE};
15 my $DEFINE;
16 my $LIBS = [];
17 my $XSOPT = '';
18 my $SYSCALL_H;
19
20 our $self; # Used in 'sourcing' the hints.
21
22 # TBD: Can we just use $Config(exe_ext) here instead of this complex
23 #      expression?
24 my $ld_exeext = ($^O eq 'cygwin' ||
25                  $^O eq 'os2' && $Config{ldflags} =~ /-Zexe\b/) ? '.exe' :
26                 (($^O eq 'vos') ? $Config{exe_ext} : '');
27
28 unless($ENV{PERL_CORE}) {
29     $ENV{PERL_CORE} = 1 if grep { $_ eq 'PERL_CORE=1' } @ARGV;
30 }
31
32 # Perls 5.002 and 5.003 did not have File::Spec, fake what we need.
33
34 sub my_dirsep {
35     $^O eq 'VMS' ? '.' :
36         $^O =~ /mswin32|netware|djgpp/i ? '\\' :
37             $^O eq 'MacOS' ? ':'
38                 : '/';
39 }
40
41 sub my_catdir {
42     shift;
43     my $catdir = join(my_dirsep, @_);
44     $^O eq 'VMS' ? "[$catdir]" : $catdir;
45 }
46
47 sub my_catfile {
48     shift;
49     return join(my_dirsep, @_) unless $^O eq 'VMS';
50     my $file = pop;
51     return my_catdir (undef, @_) . $file;
52 }
53
54 sub my_updir {
55     shift;
56     $^O eq 'VMS' ? "-" : "..";
57 }
58
59 BEGIN {
60     eval { require File::Spec };
61     if ($@) {
62         *File::Spec::catdir  = \&my_catdir;
63         *File::Spec::updir   = \&my_updir;
64         *File::Spec::catfile = \&my_catfile;
65     }
66 }
67
68 # Avoid 'used only once' warnings.
69 my $nop1 = *File::Spec::catdir;
70 my $nop2 = *File::Spec::updir;
71 my $nop3 = *File::Spec::catfile;
72
73 # if you have 5.004_03 (and some slightly older versions?), xsubpp
74 # tries to generate line numbers in the C code generated from the .xs.
75 # unfortunately, it is a little buggy around #ifdef'd code.
76 # my choice is leave it in and have people with old perls complain
77 # about the "Usage" bug, or leave it out and be unable to compile myself
78 # without changing it, and then I'd always forget to change it before a
79 # release. Sorry, Edward :)
80
81 sub try_compile_and_link {
82     my ($c, %args) = @_;
83
84     my ($ok) = 0;
85     my ($tmp) = "tmp$$";
86     local(*TMPC);
87
88     my $obj_ext = $Config{obj_ext} || ".o";
89     unlink("$tmp.c", "$tmp$obj_ext");
90
91     if (open(TMPC, '>', "$tmp.c")) {
92         print TMPC $c;
93         close(TMPC);
94
95         my $cccmd = $args{cccmd};
96
97         my $errornull;
98
99         my $COREincdir;
100
101         if ($ENV{PERL_CORE}) {
102             my $updir = File::Spec->updir;
103             $COREincdir = File::Spec->catdir(($updir) x 2);
104         } else {
105             $COREincdir = File::Spec->catdir($Config{'archlibexp'}, 'CORE');
106         }
107
108         if ($ENV{PERL_CORE}) {
109             unless (-f File::Spec->catfile($COREincdir, "EXTERN.h")) {
110                 die <<__EOD__;
111 Your environment variable PERL_CORE is '$ENV{PERL_CORE}' but there
112 is no EXTERN.h in $COREincdir.
113 Cannot continue, aborting.
114 __EOD__
115             }
116         }
117
118         my $ccflags = $Config{'ccflags'} . ' ' . "-I$COREincdir"
119          . ' -DPERL_NO_INLINE_FUNCTIONS';
120
121         if ($^O eq 'VMS') {
122             $cccmd = "$Config{'cc'} /include=($COREincdir) $tmp.c";
123         }
124
125         if ($args{silent} || !$VERBOSE) {
126             $errornull = "2>/dev/null" unless defined $errornull;
127         } else {
128             $errornull = '';
129         }
130
131         $cccmd = "$Config{'cc'} -o $tmp $ccflags $tmp.c @$LIBS $errornull"
132             unless defined $cccmd;
133
134        if ($^O eq 'VMS') {
135             open( CMDFILE, '>', "$tmp.com" );
136             print CMDFILE "\$ SET MESSAGE/NOFACILITY/NOSEVERITY/NOIDENT/NOTEXT\n";
137             print CMDFILE "\$ $cccmd\n";
138             print CMDFILE "\$ IF \$SEVERITY .NE. 1 THEN EXIT 44\n"; # escalate
139             close CMDFILE;
140             system("\@ $tmp.com");
141             $ok = $?==0;
142             for ("$tmp.c", "$tmp$obj_ext", "$tmp.com", "$tmp$Config{exe_ext}") {
143                 1 while unlink $_;
144             }
145         }
146         else
147         {
148             my $tmp_exe = "$tmp$ld_exeext";
149             printf "cccmd = $cccmd\n" if $VERBOSE;
150             my $res = system($cccmd);
151             $ok = defined($res) && $res == 0 && -s $tmp_exe && -x _;
152
153             if ( $ok && exists $args{run} && $args{run} && !$ENV{TIME_HIRES_DONT_RUN_PROBES} ) {
154                 my $tmp_exe =
155                     File::Spec->catfile(File::Spec->curdir, $tmp_exe);
156                 my @run = $tmp_exe;
157                 unshift @run, $Config{run} if $Config{run} && -e $Config{run};
158                 printf "Running $tmp_exe..." if $VERBOSE;
159                 if (system(@run) == 0) {
160                     $ok = 1;
161                 } else {
162                     $ok = 0;
163                     my $errno = $? >> 8;
164                     local $! = $errno;
165                     printf <<EOF;
166
167 *** The test run of '$tmp_exe' failed: status $?
168 *** (the status means: errno = $errno or '$!')
169 *** DO NOT PANIC: this just means that *some* functionality will be missing.
170 EOF
171                 }
172             }
173             unlink("$tmp.c", $tmp_exe);
174         }
175     }
176
177     return $ok;
178 }
179
180 my $TIME_HEADERS = <<EOH;
181 #include "EXTERN.h"
182 #include "perl.h"
183 #include "XSUB.h"
184 #ifdef I_SYS_TYPES
185 #   include <sys/types.h>
186 #endif
187 #ifdef I_SYS_TIME
188 #   include <sys/time.h>
189 #endif
190 #ifdef I_SYS_SELECT
191 #   include <sys/select.h>      /* struct timeval might be hidden in here */
192 #endif
193 EOH
194
195 sub has_gettimeofday {
196     # confusing but true (if condition true ==> -DHAS_GETTIMEOFDAY already)
197     return 0 if $Config{d_gettimeod};
198     return 1 if try_compile_and_link(<<EOM);
199 $TIME_HEADERS
200 static int foo()
201 {
202     struct timeval tv;
203     gettimeofday(&tv, 0);
204 }
205 int main(int argc, char** argv)
206 {
207     foo();
208 }
209 EOM
210     return 0;
211 }
212
213 sub has_x {
214     my ($x, %args) = @_;
215
216     return 1 if
217     try_compile_and_link(<<EOM, %args);
218 #include "EXTERN.h"
219 #include "perl.h"
220 #include "XSUB.h"
221
222 #ifdef I_UNISTD
223 #   include <unistd.h>
224 #endif
225
226 #ifdef I_SYS_TYPES
227 #   include <sys/types.h>
228 #endif
229
230 #ifdef I_SYS_TIME
231 #   include <sys/time.h>
232 #endif
233
234 int main(int argc, char** argv)
235 {
236         $x;
237 }
238 EOM
239     return 0;
240 }
241
242 sub has_nanosleep {
243     print "testing... ";
244     return 1 if
245     try_compile_and_link(<<EOM, run => 1);
246 #include <time.h>
247 #include <sys/time.h>
248 #include <stdio.h>
249 #include <stdlib.h>
250 #include <errno.h>
251
252 /* int nanosleep(const struct timespec *rqtp, struct timespec *rmtp); */
253
254 int main(int argc, char** argv) {
255     struct timespec ts1, ts2;
256     int ret;
257     ts1.tv_sec  = 0;
258     ts1.tv_nsec = 750000000;
259     ts2.tv_sec  = 0;
260     ts2.tv_nsec = 0;
261     errno = 0;
262     ret = nanosleep(&ts1, &ts2); /* E.g. in AIX nanosleep() fails and sets errno to ENOSYS. */
263     ret == 0 ? exit(0) : exit(errno ? errno : -1);
264 }
265 EOM
266 }
267
268 sub has_include {
269     my ($inc) = @_;
270     return 1 if
271     try_compile_and_link(<<EOM);
272 #include "EXTERN.h"
273 #include "perl.h"
274 #include "XSUB.h"
275
276 #include <$inc>
277 int main(int argc, char** argv)
278 {
279         return 0;
280 }
281 EOM
282     return 0;
283 }
284
285 sub has_clock_xxx_syscall {
286     my $x = shift;
287     return 0 unless defined $SYSCALL_H;
288     return 1 if
289     try_compile_and_link(<<EOM, run => 1);
290 #include "EXTERN.h"
291 #include "perl.h"
292 #include "XSUB.h"
293 #include <time.h>
294 #include <$SYSCALL_H>
295 int main(int argc, char** argv)
296 {
297     struct timespec ts;
298     /* Many Linuxes get ENOSYS even though the syscall exists. */
299     /* All implementations are supposed to support CLOCK_REALTIME. */
300     int ret = syscall(SYS_clock_$x, CLOCK_REALTIME, &ts);
301     ret == 0 ? exit(0) : exit(errno ? errno : -1);
302 }
303 EOM
304 }
305
306 sub has_clock_xxx {
307     my $xxx = shift;
308     return 1 if
309     try_compile_and_link(<<EOM, run => 1);
310 #include "EXTERN.h"
311 #include "perl.h"
312 #include "XSUB.h"
313 #include <time.h>
314 int main(int argc, char** argv)
315 {
316     struct timespec ts;
317     int ret = clock_$xxx(CLOCK_REALTIME, &ts); /* Many Linuxes get ENOSYS. */
318     /* All implementations are supposed to support CLOCK_REALTIME. */
319     ret == 0 ? exit(0) : exit(errno ? errno : -1);
320 }
321 EOM
322 }
323
324 sub has_clock {
325     return 1 if
326     try_compile_and_link(<<EOM, run => 1);
327 #include "EXTERN.h"
328 #include "perl.h"
329 #include "XSUB.h"
330 #include <time.h>
331 int main(int argc, char** argv)
332 {
333     clock_t tictoc;
334     clock_t ret = clock();
335     ret == (clock_t)-1 ? exit(errno ? errno : -1) : exit(0);
336 }
337 EOM
338 }
339
340 sub has_clock_nanosleep {
341     return 1 if
342     try_compile_and_link(<<EOM, run => 1);
343 #include "EXTERN.h"
344 #include "perl.h"
345 #include "XSUB.h"
346 #include <time.h>
347 int main(int argc, char** argv)
348 {
349     int ret;
350     struct timespec ts1;
351     struct timespec ts2;
352     ts1.tv_sec  = 0;
353     ts1.tv_nsec = 750000000;;
354     /* All implementations are supposed to support CLOCK_REALTIME. */
355     ret = clock_nanosleep(CLOCK_REALTIME, 0, &ts1, &ts2);
356     ret == 0 ? exit(0) : exit(errno ? errno : -1);
357 }
358 EOM
359 }
360
361 sub has_futimens {
362     return 1 if
363     try_compile_and_link(<<EOM);
364 #include "EXTERN.h"
365 #include "perl.h"
366 #include "XSUB.h"
367 #include <sys/stat.h>
368 int main(int argc, char** argv)
369 {
370     int ret1, ret2;
371     struct timespec ts1[2], ts2[2];
372     ret1 = futimens(0, ts1);
373     char buf[1];
374     read(0, buf, 0); /* Assuming reading nothing updates atime (the [0]) */
375     ret2 = futimens(0, ts2);
376     ret1 == 0 && ret2 == 0 && (ts1[0].tv_nsec != 0 || ts2[0].tv_nsec != 0) ?
377         exit(0) : exit(errno ? errno : -1);
378 }
379 EOM
380 }
381
382 sub has_utimensat{
383     return 1 if
384     try_compile_and_link(<<EOM);
385 #include "EXTERN.h"
386 #include "perl.h"
387 #include "XSUB.h"
388 #include <sys/stat.h>
389 #include <fcntl.h>
390 int main(int argc, char** argv)
391 {
392     int ret1, ret2;
393     struct timespec ts1[2], ts2[2];
394     /* We make the brave but probably foolish assumption that systems
395      * modern enough to have utimensat also have the /dev/stdin. */
396     ret1 = utimensat(AT_FDCWD, "/dev/stdin", ts1, 0);
397     char buf[1];
398     read(0, buf, 0); /* Assuming reading nothing updates atime (the [0]) */
399     ret2 = utimensat(AT_FDCWD, "/dev/stdin", ts2, 0);
400     ret1 == 0 && ret2 == 0 && (ts1[0].tv_nsec != 0 || ts2[0].tv_nsec != 0) ?
401         exit(0) : exit(errno ? errno : -1);
402 }
403 EOM
404 }
405
406 sub has_clockid_t{
407     return 1 if
408     try_compile_and_link(<<EOM);
409 #include "EXTERN.h"
410 #include "perl.h"
411 #include "XSUB.h"
412 #include <time.h>
413 int main(int argc, char** argv)
414 {
415     clockid_t id = CLOCK_REALTIME;
416     exit(id == CLOCK_REALTIME ? 1 : 0);
417 }
418 EOM
419 }
420
421 sub DEFINE {
422     my ($def, $val) = @_;
423     my $define = defined $val ? "$def=$val" : $def ;
424     unless ($DEFINE =~ /(?:^| )-D\Q$define\E(?: |$)/) {
425         $DEFINE .= " -D$define";
426     }
427 }
428
429 sub init {
430     my $hints = File::Spec->catfile("hints", "$^O.pl");
431     if (-f $hints) {
432         print "Using hints $hints...\n";
433         local $self;
434         do "./$hints";
435         if (exists $self->{LIBS}) {
436             $LIBS = $self->{LIBS};
437             print "Extra libraries: @$LIBS...\n";
438         }
439     }
440
441     $DEFINE = '';
442
443     if ($Config{d_syscall}) {
444         print "Have syscall()... looking for syscall.h... ";
445         if (has_include('syscall.h')) {
446             $SYSCALL_H = 'syscall.h';
447         } elsif (has_include('sys/syscall.h')) {
448             $SYSCALL_H = 'sys/syscall.h';
449         }
450     } else {
451         print "No syscall()...\n";
452     }
453
454     if ($Config{d_syscall}) {
455         if (defined $SYSCALL_H) {
456             print "found <$SYSCALL_H>.\n";
457         } else {
458             print "NOT found.\n";
459         }
460     }
461
462     print "Looking for gettimeofday()... ";
463     my $has_gettimeofday;
464     if (exists $Config{d_gettimeod}) {
465         $has_gettimeofday++ if $Config{d_gettimeod};
466     } elsif (has_gettimeofday()) {
467         $DEFINE .= ' -DHAS_GETTIMEOFDAY';
468         $has_gettimeofday++;
469     }
470
471     if ($has_gettimeofday) {
472         print "found.\n";
473     } else {
474         die <<EOD
475 Your operating system does not seem to have the gettimeofday() function.
476 (or, at least, I cannot find it)
477
478 There is no way Time::HiRes is going to work.
479
480 I am awfully sorry but I cannot go further.
481
482 Aborting configuration.
483
484 EOD
485     }
486
487     print "Looking for setitimer()... ";
488     my $has_setitimer;
489     if (exists $Config{d_setitimer}) {
490         $has_setitimer++ if $Config{d_setitimer};
491     } elsif (has_x("setitimer(ITIMER_REAL, 0, 0)")) {
492         $has_setitimer++;
493         $DEFINE .= ' -DHAS_SETITIMER';
494     }
495
496     if ($has_setitimer) {
497         print "found.\n";
498     } else {
499         print "NOT found.\n";
500     }
501
502     print "Looking for getitimer()... ";
503     my $has_getitimer;
504     if (exists $Config{'d_getitimer'}) {
505         $has_getitimer++ if $Config{'d_getitimer'};
506     } elsif (has_x("getitimer(ITIMER_REAL, 0)")) {
507         $has_getitimer++;
508         $DEFINE .= ' -DHAS_GETITIMER';
509     }
510
511     if ($has_getitimer) {
512         print "found.\n";
513     } else {
514         print "NOT found.\n";
515     }
516
517     if ($has_setitimer && $has_getitimer) {
518         print "You have interval timers (both setitimer and getitimer).\n";
519     } else {
520         print "You do NOT have interval timers.\n";
521     }
522
523     print "Looking for ualarm()... ";
524     my $has_ualarm;
525     if (exists $Config{d_ualarm}) {
526         $has_ualarm++ if $Config{d_ualarm};
527     } elsif (has_x ("ualarm (0, 0)")) {
528         $has_ualarm++;
529         $DEFINE .= ' -DHAS_UALARM';
530     }
531
532     if ($has_ualarm) {
533         print "found.\n";
534     } else {
535         print "NOT found.\n";
536         if ($has_setitimer) {
537             print "But you have setitimer().\n";
538             print "We can make a Time::HiRes::ualarm().\n";
539         }
540     }
541
542     print "Looking for usleep()... ";
543     my $has_usleep;
544     if (exists $Config{d_usleep}) {
545         $has_usleep++ if $Config{d_usleep};
546     } elsif (has_x ("usleep (0)")) {
547         $has_usleep++;
548         $DEFINE .= ' -DHAS_USLEEP';
549     }
550
551     if ($has_usleep) {
552         print "found.\n";
553     } else {
554         print "NOT found.\n";
555         print "Let's see if you have select()... ";
556         if ($Config{'d_select'}) {
557             print "found.\n";
558             print "We can make a Time::HiRes::usleep().\n";
559         } else {
560             print "NOT found.\n";
561             print "You won't have a Time::HiRes::usleep().\n";
562         }
563     }
564
565     print "Looking for nanosleep()... ";
566     my $has_nanosleep;
567     if ($ENV{FORCE_NANOSLEEP_SCAN}) {
568         print "forced scan... ";
569         if (has_nanosleep()) {
570             $has_nanosleep++;
571             $DEFINE .= ' -DTIME_HIRES_NANOSLEEP';
572         }
573     }
574     elsif (exists $Config{d_nanosleep}) {
575         print "believing \$Config{d_nanosleep}... ";
576         if ($Config{d_nanosleep}) {
577             $has_nanosleep++;
578             $DEFINE .= ' -DTIME_HIRES_NANOSLEEP';
579         }
580     } else {
581         if (has_nanosleep()) {
582             $has_nanosleep++;
583             $DEFINE .= ' -DTIME_HIRES_NANOSLEEP';
584         }
585     }
586
587     if ($has_nanosleep) {
588         print "found.\n";
589         print "You can mix subsecond sleeps with signals, if you want to.\n";
590         print "(It's still not portable, though.)\n";
591     } else {
592         print "NOT found.\n";
593         my $nt = ($^O eq 'os2' ? '' : 'not');
594         print "You can$nt mix subsecond sleeps with signals.\n";
595         print "(It would not be portable anyway.)\n";
596     }
597
598     print "Looking for clockid_t... ";
599     my $has_clockid_t;
600     if (has_clockid_t()) {
601         print "found.\n";
602         $has_clockid_t++;
603         $DEFINE .= ' -DTIME_HIRES_CLOCKID_T';
604     } else {
605         print "NOT found, will use int.\n";
606     }
607
608     print "Looking for clock_gettime()... ";
609     my $has_clock_gettime;
610     my $has_clock_gettime_emulation;
611     if (exists $Config{d_clock_gettime}) {
612         $has_clock_gettime++ if $Config{d_clock_gettime}; # Unlikely...
613     } elsif (has_clock_xxx('gettime')) {
614         $has_clock_gettime++;
615         $DEFINE .= ' -DTIME_HIRES_CLOCK_GETTIME';
616     } elsif (defined $SYSCALL_H && has_clock_xxx_syscall('gettime')) {
617         $has_clock_gettime++;
618         $DEFINE .= ' -DTIME_HIRES_CLOCK_GETTIME -DTIME_HIRES_CLOCK_GETTIME_SYSCALL';
619     } elsif ($^O eq 'darwin') {
620        $has_clock_gettime_emulation++;
621        $has_clock_gettime++;
622        $DEFINE .= ' -DTIME_HIRES_CLOCK_GETTIME -DTIME_HIRES_CLOCK_GETTIME_EMULATION';
623     }
624
625     if ($has_clock_gettime) {
626         if ($DEFINE =~ /-DTIME_HIRES_CLOCK_GETTIME_SYSCALL/) {
627             print "found (via syscall).\n";
628         } elsif ($has_clock_gettime_emulation) {
629             print "found (via emulation).\n";
630         } else {
631             print "found.\n";
632         }
633     } else {
634         print "NOT found.\n";
635     }
636
637     print "Looking for clock_getres()... ";
638     my $has_clock_getres;
639     my $has_clock_getres_emulation;
640     if (exists $Config{d_clock_getres}) {
641         $has_clock_getres++ if $Config{d_clock_getres}; # Unlikely...
642     } elsif (has_clock_xxx('getres')) {
643         $has_clock_getres++;
644         $DEFINE .= ' -DTIME_HIRES_CLOCK_GETRES';
645     } elsif (defined $SYSCALL_H && has_clock_xxx_syscall('getres')) {
646         $has_clock_getres++;
647         $DEFINE .= ' -DTIME_HIRES_CLOCK_GETRES -DTIME_HIRES_CLOCK_GETRES_SYSCALL';
648     } elsif ($^O eq 'darwin') {
649        $has_clock_getres_emulation++;
650        $has_clock_getres++;
651        $DEFINE .= ' -DTIME_HIRES_CLOCK_GETRES -DTIME_HIRES_CLOCK_GETRES_EMULATION';
652     }
653
654     if ($has_clock_getres) {
655         if ($DEFINE =~ /-DTIME_HIRES_CLOCK_GETRES_SYSCALL/) {
656             print "found (via syscall).\n";
657         } elsif ($has_clock_getres_emulation) {
658             print "found (via emulation).\n";
659         } else {
660             print "found.\n";
661         }
662     } else {
663         print "NOT found.\n";
664     }
665
666     print "Looking for clock_nanosleep()... ";
667     my $has_clock_nanosleep;
668     my $has_clock_nanosleep_emulation;
669     if (exists $Config{d_clock_nanosleep}) {
670         $has_clock_nanosleep++ if $Config{d_clock_nanosleep}; # Unlikely...
671     } elsif (has_clock_nanosleep()) {
672         $has_clock_nanosleep++;
673         $DEFINE .= ' -DTIME_HIRES_CLOCK_NANOSLEEP';
674     } elsif ($^O eq 'darwin') {
675         $has_clock_nanosleep++;
676         $has_clock_nanosleep_emulation++;
677         $DEFINE .= ' -DTIME_HIRES_CLOCK_NANOSLEEP -DTIME_HIRES_CLOCK_NANOSLEEP_EMULATION';
678     }
679
680     if ($has_clock_nanosleep) {
681         if ($has_clock_nanosleep_emulation) {
682             print "found (via emulation).\n";
683         } else {
684             print "found.\n";
685         }
686     } else {
687         print "NOT found.\n";
688     }
689
690     print "Looking for clock()... ";
691     my $has_clock;
692     if (exists $Config{d_clock}) {
693         $has_clock++ if $Config{d_clock}; # Unlikely...
694     } elsif (has_clock()) {
695         $has_clock++;
696         $DEFINE .= ' -DTIME_HIRES_CLOCK';
697     }
698
699     if ($has_clock) {
700         print "found.\n";
701     } else {
702         print "NOT found.\n";
703     }
704
705     print "Looking for working futimens()... ";
706     my $has_futimens;
707     if (has_futimens()) {
708         $has_futimens++;
709         $DEFINE .= ' -DHAS_FUTIMENS';
710     }
711
712     if ($has_futimens) {
713         print "found.\n";
714     } else {
715         print "NOT found.\n";
716     }
717
718     print "Looking for working utimensat()... ";
719     my $has_utimensat;
720     if (has_utimensat()) {
721         $has_utimensat++;
722         $DEFINE .= ' -DHAS_UTIMENSAT';
723     }
724
725     if ($has_utimensat) {
726         print "found.\n";
727     } else {
728         print "NOT found.\n";
729     }
730
731     my $has_hires_utime = ($has_futimens && $has_utimensat);
732     if ($has_hires_utime) {
733         $DEFINE .= ' -DTIME_HIRES_UTIME';
734         print "You seem to have subsecond timestamp setting.\n";
735     } else {
736         print "You do NOT seem to have subsecond timestamp setting.\n";
737     }
738
739     print "Looking for stat() subsecond timestamps...\n";
740
741     print "Trying struct stat st_atimespec.tv_nsec...";
742     my $has_stat_st_xtimespec;
743     if (try_compile_and_link(<<EOM)) {
744 $TIME_HEADERS
745 #include <sys/stat.h>
746 int main(int argc, char** argv) {
747     struct stat st;
748     st.st_atimespec.tv_nsec = 0;
749 }
750 EOM
751       $has_stat_st_xtimespec++;
752       DEFINE('TIME_HIRES_STAT_ST_XTIMESPEC');  # 1
753     }
754
755     if ($has_stat_st_xtimespec) {
756         print "found.\n";
757     } else {
758         print "NOT found.\n";
759     }
760
761     print "Trying struct stat st_atimensec...";
762     my $has_stat_st_xtimensec;
763     if (try_compile_and_link(<<EOM)) {
764 $TIME_HEADERS
765 #include <sys/stat.h>
766 int main(int argc, char** argv) {
767     struct stat st;
768     st.st_atimensec = 0;
769 }
770 EOM
771       $has_stat_st_xtimensec++;
772       DEFINE('TIME_HIRES_STAT_ST_XTIMENSEC');  # 2
773     }
774
775     if ($has_stat_st_xtimensec) {
776         print "found.\n";
777     } else {
778         print "NOT found.\n";
779     }
780
781     print "Trying struct stat st_atime_n...";
782     my $has_stat_st_xtime_n;
783     if (try_compile_and_link(<<EOM)) {
784 $TIME_HEADERS
785 #include <sys/stat.h>
786 int main(int argc, char** argv) {
787     struct stat st;
788     st.st_atime_n = 0;
789 }
790 EOM
791       $has_stat_st_xtime_n++;
792       DEFINE('TIME_HIRES_STAT_ST_XTIME_N');  # 3
793     }
794
795     if ($has_stat_st_xtime_n) {
796         print "found.\n";
797     } else {
798         print "NOT found.\n";
799     }
800
801     print "Trying struct stat st_atim.tv_nsec...";
802     my $has_stat_st_xtim;
803     if (try_compile_and_link(<<EOM)) {
804 $TIME_HEADERS
805 #include <sys/stat.h>
806 int main(int argc, char** argv) {
807     struct stat st;
808     st.st_atim.tv_nsec = 0;
809 }
810 EOM
811       $has_stat_st_xtim++;
812       DEFINE('TIME_HIRES_STAT_XTIM');  # 4
813     }
814
815     if ($has_stat_st_xtim) {
816         print "found.\n";
817     } else {
818         print "NOT found.\n";
819     }
820
821     print "Trying struct stat st_uatime...";
822     my $has_stat_st_uxtime;
823     if (try_compile_and_link(<<EOM)) {
824 $TIME_HEADERS
825 #include <sys/stat.h>
826 int main(int argc, char** argv) {
827     struct stat st;
828     st.st_uatime = 0;
829 }
830 EOM
831       $has_stat_st_uxtime++;
832       DEFINE('TIME_HIRES_STAT_ST_UXTIME');  # 5
833     }
834
835     if ($has_stat_st_uxtime) {
836         print "found.\n";
837     } else {
838         print "NOT found.\n";
839     }
840
841     # See HiRes.xs hrstatns()
842     if ($has_stat_st_xtimespec) {
843         DEFINE('TIME_HIRES_STAT', 1);
844     } elsif ($has_stat_st_xtimensec) {
845         DEFINE('TIME_HIRES_STAT', 2);
846     } elsif ($has_stat_st_xtime_n) {
847         DEFINE('TIME_HIRES_STAT', 3);
848     } elsif ($has_stat_st_xtim) {
849         DEFINE('TIME_HIRES_STAT', 4);
850     } elsif ($has_stat_st_uxtime) {
851         DEFINE('TIME_HIRES_STAT', 5);
852     }
853
854     my $has_hires_stat = ($DEFINE =~ /-DTIME_HIRES_STAT=(\d+)/) ? $1 : 0;
855     if ($has_hires_stat) {
856         print "You seem to have subsecond timestamp reading.\n";
857         print "(Your struct stat has them, but the filesystems must help.)\n";
858         unless ($has_hires_utime) {
859             print "However, you do NOT seem to have subsecond timestamp setting.\n";
860         }
861     } else {
862         print "You do NOT seem to have subsecond timestamp reading.\n";
863     }
864
865     my $has_w32api_windows_h;
866
867     if ($^O eq 'cygwin') {
868         print "Looking for <w32api/windows.h>... ";
869         if (has_include('w32api/windows.h')) {
870             $has_w32api_windows_h++;
871             DEFINE('HAS_W32API_WINDOWS_H');
872         }
873         if ($has_w32api_windows_h) {
874             print "found.\n";
875         } else {
876             print "NOT found.\n";
877         }
878     }
879
880     if ($DEFINE) {
881         $DEFINE =~ s/^\s+//;
882         if (open(XDEFINE, '>', 'xdefine')) {
883             print XDEFINE $DEFINE, "\n";
884             close(XDEFINE);
885         }
886     }
887 }
888
889 sub doMakefile {
890     my @makefileopts = ();
891
892     if ($] >= 5.005) {
893         push (@makefileopts,
894             'AUTHOR'    => 'Jarkko Hietaniemi <jhi@iki.fi>',
895             'ABSTRACT_FROM' => 'HiRes.pm',
896         );
897         DEFINE('ATLEASTFIVEOHOHFIVE');
898     }
899     DEFINE('USE_PPPORT_H') unless $ENV{PERL_CORE};
900
901     push (@makefileopts,
902         'NAME'  => 'Time::HiRes',
903         'VERSION_FROM' => 'HiRes.pm', # finds $VERSION
904         'LIBS'  => $LIBS,   # e.g., '-lm'
905         'DEFINE'        => $DEFINE,     # e.g., '-DHAS_SOMETHING'
906         'XSOPT' => $XSOPT,
907           # Do not even think about 'INC' => '-I/usr/ucbinclude',
908           # Solaris will avenge.
909         'INC'   => '',     # e.g., '-I/usr/include/other'
910         'INSTALLDIRS' => ($] >= 5.008 && $] < 5.011 ? 'perl' : 'site'),
911         'PREREQ_PM' => {
912             'Carp' => 0,
913             'Config' => 0,
914             'Exporter' => 0,
915             'ExtUtils::MakeMaker' => 0,
916             'Test::More' => 0,
917             'XSLoader' => 0,
918             'strict' => 0,
919         },
920         'dist'      => {
921             'CI'       => 'ci -l',
922             'COMPRESS' => 'gzip -9f',
923             'SUFFIX'   => 'gz',
924         },
925         clean => { FILES => "xdefine" },
926         realclean => { FILES=> 'const-c.inc const-xs.inc' },
927     );
928
929     if ($^O eq "MSWin32" && !(grep { /\ALD[A-Z]*=/ } @ARGV)) {
930         my $libperl = $Config{libperl} || "";
931         my $gccversion = $Config{gccversion} || "";
932         if ($gccversion =~ /\A3\.4\.[0-9]+/ and $libperl =~ /\.lib\z/) {
933             # Avoid broken linkage with ActivePerl, by linking directly
934             # against the Perl DLL rather than the import library.
935             (my $llibperl = "-l$libperl") =~ s/\.lib\z//;
936             my $lddlflags = $Config{lddlflags} || "";
937             my $ldflags = $Config{ldflags} || "";
938             s/-L(?:".*?"|\S+)//g foreach $lddlflags, $ldflags;
939             my $libdirs = join ' ',
940                 map { s/(?<!\\)((?:\\\\)*")/\\$1/g; qq[-L"$_"] }
941                 @Config{qw/bin sitebin/};
942             push @makefileopts, macro => {
943                 LDDLFLAGS => "$lddlflags $libdirs $llibperl",
944                 LDFLAGS => "$ldflags $libdirs $llibperl",
945                 PERL_ARCHIVE => "",
946             };
947         }
948     }
949
950     if ($ENV{PERL_CORE}) {
951         push @makefileopts, MAN3PODS => {};
952     }
953
954     if ($ExtUtils::MakeMaker::VERSION >= 6.48) {
955         push @makefileopts, (MIN_PERL_VERSION => '5.006',);
956     }
957
958     if ($ExtUtils::MakeMaker::VERSION >= 6.31) {
959         push @makefileopts, (LICENSE => 'perl_5');
960     }
961
962     if ($ExtUtils::MakeMaker::VERSION >= 6.46) {
963         push @makefileopts, (
964             META_MERGE => {
965                 resources => {
966                     repository => 'git://perl5.git.perl.org/perl.git',
967                     bugtracker => 'https://rt.perl.org/rt3/',
968                     homepage   => "http://dev.perl.org/",
969                 },
970             },
971         )
972     }
973
974     WriteMakefile(@makefileopts);
975 }
976
977 sub doConstants {
978     if (eval {require ExtUtils::Constant; 1}) {
979         # More or less this same list is in HiRes.pm.  Should unify.
980         my @names = qw(
981                        CLOCKS_PER_SEC
982                        CLOCK_BOOTTIME
983                        CLOCK_HIGHRES
984                        CLOCK_MONOTONIC
985                        CLOCK_MONOTONIC_COARSE
986                        CLOCK_MONOTONIC_FAST
987                        CLOCK_MONOTONIC_PRECISE
988                        CLOCK_MONOTONIC_RAW
989                        CLOCK_PROF
990                        CLOCK_PROCESS_CPUTIME_ID
991                        CLOCK_REALTIME
992                        CLOCK_REALTIME_COARSE
993                        CLOCK_REALTIME_FAST
994                        CLOCK_REALTIME_PRECISE
995                        CLOCK_REALTIME_RAW
996                        CLOCK_SECOND
997                        CLOCK_SOFTTIME
998                        CLOCK_THREAD_CPUTIME_ID
999                        CLOCK_TIMEOFDAY
1000                        CLOCK_UPTIME
1001                        CLOCK_UPTIME_COARSE
1002                        CLOCK_UPTIME_FAST
1003                        CLOCK_UPTIME_PRECISE
1004                        CLOCK_UPTIME_RAW
1005                        CLOCK_VIRTUAL
1006                        ITIMER_PROF
1007                        ITIMER_REAL
1008                        ITIMER_REALPROF
1009                        ITIMER_VIRTUAL
1010                        TIMER_ABSTIME
1011                       );
1012         foreach (qw (d_usleep d_ualarm d_gettimeofday d_getitimer d_setitimer
1013                      d_nanosleep d_clock_gettime d_clock_getres
1014                      d_clock d_clock_nanosleep d_hires_stat
1015                      d_futimens d_utimensat d_hires_utime)) {
1016             my $macro = $_;
1017             if ($macro =~ /^(d_nanosleep|d_clock)$/) {
1018                 $macro =~ s/^d_(.+)/TIME_HIRES_\U$1/;
1019             } elsif ($macro =~ /^(d_hires_stat)$/) {
1020                 my $d_hires_stat = $1 if ($DEFINE =~ /-DTIME_HIRES_STAT=(\d+)/);
1021                 if (defined $d_hires_stat) {
1022                     push @names, {name => $_, macro => "TIME_HIRES_STAT", value => $d_hires_stat,
1023                                   default => ["IV", "0"]};
1024                     next;
1025                 }
1026             } elsif ($macro =~ /^(d_hires_utime)$/) {
1027                 my $d_hires_utime =
1028                     ($DEFINE =~ /-DHAS_FUTIMENS/ ||
1029                      $DEFINE =~ /-DHAS_UTIMENSAT/);
1030                 push @names, {name => $_, macro => "TIME_HIRES_UTIME", value => $d_hires_utime,
1031                               default => ["IV", "0"]};
1032                 next;
1033             } elsif ($macro =~ /^(d_clock_gettime|d_clock_getres|d_clock_nanosleep)$/) {
1034                 $macro =~ s/^d_(.+)/TIME_HIRES_\U$1/;
1035                 my $val = ($DEFINE =~ /-D$macro\b/) ? 1 : 0;
1036                 push @names, {name => $_, macro => $macro, value => $val,
1037                               default => ["IV", "0"]};
1038                 next;
1039             } else {
1040                 $macro =~ s/^d_(.+)/HAS_\U$1/;
1041             }
1042             push @names, {name => $_, macro => $macro, value => 1,
1043                           default => ["IV", "0"]};
1044         }
1045         ExtUtils::Constant::WriteConstants(
1046                                            NAME => 'Time::HiRes',
1047                                            NAMES => \@names,
1048                                           );
1049     } else {
1050         my $file;
1051         foreach $file ('const-c.inc', 'const-xs.inc') {
1052             my $fallback = File::Spec->catfile('fallback', $file);
1053             local $/;
1054             open IN, '<', $fallback or die "Can't open $fallback: $!";
1055             open OUT, '>', $file or die "Can't open $file: $!";
1056             print OUT <IN> or die $!;
1057             close OUT or die "Can't close $file: $!";
1058             close IN or die "Can't close $fallback: $!";
1059         }
1060     }
1061 }
1062
1063 sub main {
1064     if (-f "xdefine" && !(@ARGV  && $ARGV[0] =~ /^--(?:configure|force)$/)) {
1065         print qq[$0: The "xdefine" exists, skipping the configure step.\n];
1066         print qq[Use "$^X $0 --configure"\n];
1067         print qq[or: "$^X $0 --force\n];
1068         print qq[to force the configure step.\n];
1069     } else {
1070         print "Configuring Time::HiRes...\n";
1071         1 while unlink("define");
1072         if ($^O =~ /Win32/i) {
1073             DEFINE('SELECT_IS_BROKEN');
1074             $LIBS = [];
1075             print "System is $^O, skipping full configure...\n";
1076             open(XDEFINE, '>', 'xdefine') or die "$0: Cannot create xdefine: $!\n";
1077             close(XDEFINE);
1078         } else {
1079             init();
1080         }
1081         doMakefile;
1082         doConstants;
1083     }
1084     my $make = $Config{'make'} || "make";
1085     unless (exists $ENV{PERL_CORE} && $ENV{PERL_CORE}) {
1086         print  <<EOM;
1087 Now you may issue '$make'.  Do not forget also '$make test'.
1088 EOM
1089        if ($] == 5.008 &&
1090            ((exists $ENV{LC_ALL}   && $ENV{LC_ALL}   =~ /utf-?8/i) ||
1091             (exists $ENV{LC_CTYPE} && $ENV{LC_CTYPE} =~ /utf-?8/i) ||
1092             (exists $ENV{LANG}     && $ENV{LANG}     =~ /utf-?8/i))) {
1093             print <<EOM;
1094
1095 NOTE: if you get an error like this (the Makefile line number may vary):
1096 Makefile:91: *** missing separator
1097 then set the environment variable LC_ALL to "C" and retry
1098 from scratch (re-run perl "Makefile.PL").
1099 (And consider upgrading your Perl to, say, at least Perl 5.8.8.)
1100 (You got this message because you seem to have
1101  an UTF-8 locale active in your shell environment, this used
1102  to cause broken Makefiles to be created from Makefile.PLs)
1103 EOM
1104         }
1105     }
1106 }
1107
1108 &main;
1109
1110 # EOF