This is a live mirror of the Perl 5 development currently hosted at https://github.com/perl/perl5
66691fd08b9165aec10166f326b2343295e4713e
[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 ret;
371     struct timespec ts[2];
372     ret = futimens(0, ts);
373     ret == 0 ? exit(0) : exit(errno ? errno : -1);
374 }
375 EOM
376 }
377
378 sub has_utimensat{
379     return 1 if
380     try_compile_and_link(<<EOM);
381 #include "EXTERN.h"
382 #include "perl.h"
383 #include "XSUB.h"
384 #include <sys/stat.h>
385 #include <fcntl.h>
386 int main(int argc, char** argv)
387 {
388     int ret;
389     struct timespec ts[2];
390     ret = utimensat(AT_FDCWD, 0, ts, 0);
391     ret == 0 ? exit(0) : exit(errno ? errno : -1);
392 }
393 EOM
394 }
395
396 sub has_clockid_t{
397     return 1 if
398     try_compile_and_link(<<EOM);
399 #include "EXTERN.h"
400 #include "perl.h"
401 #include "XSUB.h"
402 #include <time.h>
403 int main(int argc, char** argv)
404 {
405     clockid_t id = CLOCK_REALTIME;
406     exit(id == CLOCK_REALTIME ? 1 : 0);
407 }
408 EOM
409 }
410
411 sub DEFINE {
412     my ($def, $val) = @_;
413     my $define = defined $val ? "$def=$val" : $def ;
414     unless ($DEFINE =~ /(?:^| )-D\Q$define\E(?: |$)/) {
415         $DEFINE .= " -D$define";
416     }
417 }
418
419 sub init {
420     my $hints = File::Spec->catfile("hints", "$^O.pl");
421     if (-f $hints) {
422         print "Using hints $hints...\n";
423         local $self;
424         do "./$hints";
425         if (exists $self->{LIBS}) {
426             $LIBS = $self->{LIBS};
427             print "Extra libraries: @$LIBS...\n";
428         }
429     }
430
431     $DEFINE = '';
432
433     if ($Config{d_syscall}) {
434         print "Have syscall()... looking for syscall.h... ";
435         if (has_include('syscall.h')) {
436             $SYSCALL_H = 'syscall.h';
437         } elsif (has_include('sys/syscall.h')) {
438             $SYSCALL_H = 'sys/syscall.h';
439         }
440     } else {
441         print "No syscall()...\n";
442     }
443
444     if ($Config{d_syscall}) {
445         if (defined $SYSCALL_H) {
446             print "found <$SYSCALL_H>.\n";
447         } else {
448             print "NOT found.\n";
449         }
450     }
451
452     print "Looking for gettimeofday()... ";
453     my $has_gettimeofday;
454     if (exists $Config{d_gettimeod}) {
455         $has_gettimeofday++ if $Config{d_gettimeod};
456     } elsif (has_gettimeofday()) {
457         $DEFINE .= ' -DHAS_GETTIMEOFDAY';
458         $has_gettimeofday++;
459     }
460
461     if ($has_gettimeofday) {
462         print "found.\n";
463     } else {
464         die <<EOD
465 Your operating system does not seem to have the gettimeofday() function.
466 (or, at least, I cannot find it)
467
468 There is no way Time::HiRes is going to work.
469
470 I am awfully sorry but I cannot go further.
471
472 Aborting configuration.
473
474 EOD
475     }
476
477     print "Looking for setitimer()... ";
478     my $has_setitimer;
479     if (exists $Config{d_setitimer}) {
480         $has_setitimer++ if $Config{d_setitimer};
481     } elsif (has_x("setitimer(ITIMER_REAL, 0, 0)")) {
482         $has_setitimer++;
483         $DEFINE .= ' -DHAS_SETITIMER';
484     }
485
486     if ($has_setitimer) {
487         print "found.\n";
488     } else {
489         print "NOT found.\n";
490     }
491
492     print "Looking for getitimer()... ";
493     my $has_getitimer;
494     if (exists $Config{'d_getitimer'}) {
495         $has_getitimer++ if $Config{'d_getitimer'};
496     } elsif (has_x("getitimer(ITIMER_REAL, 0)")) {
497         $has_getitimer++;
498         $DEFINE .= ' -DHAS_GETITIMER';
499     }
500
501     if ($has_getitimer) {
502         print "found.\n";
503     } else {
504         print "NOT found.\n";
505     }
506
507     if ($has_setitimer && $has_getitimer) {
508         print "You have interval timers (both setitimer and getitimer).\n";
509     } else {
510         print "You do not have interval timers.\n";
511     }
512
513     print "Looking for ualarm()... ";
514     my $has_ualarm;
515     if (exists $Config{d_ualarm}) {
516         $has_ualarm++ if $Config{d_ualarm};
517     } elsif (has_x ("ualarm (0, 0)")) {
518         $has_ualarm++;
519         $DEFINE .= ' -DHAS_UALARM';
520     }
521
522     if ($has_ualarm) {
523         print "found.\n";
524     } else {
525         print "NOT found.\n";
526         if ($has_setitimer) {
527             print "But you have setitimer().\n";
528             print "We can make a Time::HiRes::ualarm().\n";
529         }
530     }
531
532     print "Looking for usleep()... ";
533     my $has_usleep;
534     if (exists $Config{d_usleep}) {
535         $has_usleep++ if $Config{d_usleep};
536     } elsif (has_x ("usleep (0)")) {
537         $has_usleep++;
538         $DEFINE .= ' -DHAS_USLEEP';
539     }
540
541     if ($has_usleep) {
542         print "found.\n";
543     } else {
544         print "NOT found.\n";
545         print "Let's see if you have select()... ";
546         if ($Config{'d_select'}) {
547             print "found.\n";
548             print "We can make a Time::HiRes::usleep().\n";
549         } else {
550             print "NOT found.\n";
551             print "You won't have a Time::HiRes::usleep().\n";
552         }
553     }
554
555     print "Looking for nanosleep()... ";
556     my $has_nanosleep;
557     if ($ENV{FORCE_NANOSLEEP_SCAN}) {
558         print "forced scan... ";
559         if (has_nanosleep()) {
560             $has_nanosleep++;
561             $DEFINE .= ' -DTIME_HIRES_NANOSLEEP';
562         }
563     }
564     elsif (exists $Config{d_nanosleep}) {
565         print "believing \$Config{d_nanosleep}... ";
566         if ($Config{d_nanosleep}) {
567             $has_nanosleep++;
568             $DEFINE .= ' -DTIME_HIRES_NANOSLEEP';
569         }
570     } elsif ($^O =~ /^(mpeix)$/) {
571         # MPE/iX falsely finds nanosleep from its libc equivalent.
572         print "skipping because in $^O... ";
573     } else {
574         if (has_nanosleep()) {
575             $has_nanosleep++;
576             $DEFINE .= ' -DTIME_HIRES_NANOSLEEP';
577         }
578     }
579
580     if ($has_nanosleep) {
581         print "found.\n";
582         print "You can mix subsecond sleeps with signals, if you want to.\n";
583         print "(It's still not portable, though.)\n";
584     } else {
585         print "NOT found.\n";
586         my $nt = ($^O eq 'os2' ? '' : 'not');
587         print "You can$nt mix subsecond sleeps with signals.\n";
588         print "(It would not be portable anyway.)\n";
589     }
590
591     print "Looking for clockid_t... ";
592     my $has_clockid_t;
593     if (has_clockid_t()) {
594         print "found.\n";
595         $has_clockid_t++;
596         $DEFINE .= ' -DTIME_HIRES_CLOCKID_T';
597     } else {
598         print "NOT found, will use int.\n";
599     }
600
601     print "Looking for clock_gettime()... ";
602     my $has_clock_gettime;
603     my $has_clock_gettime_emulation;
604     if (exists $Config{d_clock_gettime}) {
605         $has_clock_gettime++ if $Config{d_clock_gettime}; # Unlikely...
606     } elsif (has_clock_xxx('gettime')) {
607         $has_clock_gettime++;
608         $DEFINE .= ' -DTIME_HIRES_CLOCK_GETTIME';
609     } elsif (defined $SYSCALL_H && has_clock_xxx_syscall('gettime')) {
610         $has_clock_gettime++;
611         $DEFINE .= ' -DTIME_HIRES_CLOCK_GETTIME -DTIME_HIRES_CLOCK_GETTIME_SYSCALL';
612     } elsif ($^O eq 'darwin') {
613        $has_clock_gettime_emulation++;
614        $has_clock_gettime++;
615        $DEFINE .= ' -DTIME_HIRES_CLOCK_GETTIME -DTIME_HIRES_CLOCK_GETTIME_EMULATION';
616     }
617
618     if ($has_clock_gettime) {
619         if ($DEFINE =~ /-DTIME_HIRES_CLOCK_GETTIME_SYSCALL/) {
620             print "found (via syscall).\n";
621         } elsif ($has_clock_gettime_emulation) {
622             print "found (via emulation).\n";
623         } else {
624             print "found.\n";
625         }
626     } else {
627         print "NOT found.\n";
628     }
629
630     print "Looking for clock_getres()... ";
631     my $has_clock_getres;
632     my $has_clock_getres_emulation;
633     if (exists $Config{d_clock_getres}) {
634         $has_clock_getres++ if $Config{d_clock_getres}; # Unlikely...
635     } elsif (has_clock_xxx('getres')) {
636         $has_clock_getres++;
637         $DEFINE .= ' -DTIME_HIRES_CLOCK_GETRES';
638     } elsif (defined $SYSCALL_H && has_clock_xxx_syscall('getres')) {
639         $has_clock_getres++;
640         $DEFINE .= ' -DTIME_HIRES_CLOCK_GETRES -DTIME_HIRES_CLOCK_GETRES_SYSCALL';
641     } elsif ($^O eq 'darwin') {
642        $has_clock_getres_emulation++;
643        $has_clock_getres++;
644        $DEFINE .= ' -DTIME_HIRES_CLOCK_GETRES -DTIME_HIRES_CLOCK_GETRES_EMULATION';
645     }
646
647     if ($has_clock_getres) {
648         if ($DEFINE =~ /-DTIME_HIRES_CLOCK_GETRES_SYSCALL/) {
649             print "found (via syscall).\n";
650         } elsif ($has_clock_getres_emulation) {
651             print "found (via emulation).\n";
652         } else {
653             print "found.\n";
654         }
655     } else {
656         print "NOT found.\n";
657     }
658
659     print "Looking for clock_nanosleep()... ";
660     my $has_clock_nanosleep;
661     my $has_clock_nanosleep_emulation;
662     if (exists $Config{d_clock_nanosleep}) {
663         $has_clock_nanosleep++ if $Config{d_clock_nanosleep}; # Unlikely...
664     } elsif (has_clock_nanosleep()) {
665         $has_clock_nanosleep++;
666         $DEFINE .= ' -DTIME_HIRES_CLOCK_NANOSLEEP';
667     } elsif ($^O eq 'darwin') {
668         $has_clock_nanosleep++;
669         $has_clock_nanosleep_emulation++;
670         $DEFINE .= ' -DTIME_HIRES_CLOCK_NANOSLEEP -DTIME_HIRES_CLOCK_NANOSLEEP_EMULATION';
671     }
672
673     if ($has_clock_nanosleep) {
674         if ($has_clock_nanosleep_emulation) {
675             print "found (via emulation).\n";
676         } else {
677             print "found.\n";
678         }
679     } else {
680         print "NOT found.\n";
681     }
682
683     print "Looking for clock()... ";
684     my $has_clock;
685     if (exists $Config{d_clock}) {
686         $has_clock++ if $Config{d_clock}; # Unlikely...
687     } elsif (has_clock()) {
688         $has_clock++;
689         $DEFINE .= ' -DTIME_HIRES_CLOCK';
690     }
691
692     if ($has_clock) {
693         print "found.\n";
694     } else {
695         print "NOT found.\n";
696     }
697
698     print "Looking for futimens()... ";
699     my $has_futimens;
700     if (has_futimens()) {
701         $has_futimens++;
702         $DEFINE .= ' -DHAS_FUTIMENS';
703     }
704
705     if ($has_futimens) {
706         print "found.\n";
707     } else {
708         print "NOT found.\n";
709     }
710
711     print "Looking for utimensat()... ";
712     my $has_utimensat;
713     if (has_utimensat()) {
714         $has_utimensat++;
715         $DEFINE .= ' -DHAS_UTIMENSAT';
716     }
717
718     if ($has_utimensat) {
719         print "found.\n";
720     } else {
721         print "NOT found.\n";
722     }
723
724     if ($has_futimens or $has_utimensat) {
725         $DEFINE .= ' -DTIME_HIRES_UTIME';
726     }
727
728     print "Looking for stat() subsecond timestamps...\n";
729
730     print "Trying struct stat st_atimespec.tv_nsec...";
731     my $has_stat_st_xtimespec;
732     if (try_compile_and_link(<<EOM)) {
733 $TIME_HEADERS
734 #include <sys/stat.h>
735 int main(int argc, char** argv) {
736     struct stat st;
737     st.st_atimespec.tv_nsec = 0;
738 }
739 EOM
740       $has_stat_st_xtimespec++;
741       DEFINE('TIME_HIRES_STAT_ST_XTIMESPEC');  # 1
742     }
743
744     if ($has_stat_st_xtimespec) {
745         print "found.\n";
746     } else {
747         print "NOT found.\n";
748     }
749
750     print "Trying struct stat st_atimensec...";
751     my $has_stat_st_xtimensec;
752     if (try_compile_and_link(<<EOM)) {
753 $TIME_HEADERS
754 #include <sys/stat.h>
755 int main(int argc, char** argv) {
756     struct stat st;
757     st.st_atimensec = 0;
758 }
759 EOM
760       $has_stat_st_xtimensec++;
761       DEFINE('TIME_HIRES_STAT_ST_XTIMENSEC');  # 2
762     }
763
764     if ($has_stat_st_xtimensec) {
765         print "found.\n";
766     } else {
767         print "NOT found.\n";
768     }
769
770     print "Trying struct stat st_atime_n...";
771     my $has_stat_st_xtime_n;
772     if (try_compile_and_link(<<EOM)) {
773 $TIME_HEADERS
774 #include <sys/stat.h>
775 int main(int argc, char** argv) {
776     struct stat st;
777     st.st_atime_n = 0;
778 }
779 EOM
780       $has_stat_st_xtime_n++;
781       DEFINE('TIME_HIRES_STAT_ST_XTIME_N');  # 3
782     }
783
784     if ($has_stat_st_xtime_n) {
785         print "found.\n";
786     } else {
787         print "NOT found.\n";
788     }
789
790     print "Trying struct stat st_atim.tv_nsec...";
791     my $has_stat_st_xtim;
792     if (try_compile_and_link(<<EOM)) {
793 $TIME_HEADERS
794 #include <sys/stat.h>
795 int main(int argc, char** argv) {
796     struct stat st;
797     st.st_atim.tv_nsec = 0;
798 }
799 EOM
800       $has_stat_st_xtim++;
801       DEFINE('TIME_HIRES_STAT_XTIM');  # 4
802     }
803
804     if ($has_stat_st_xtim) {
805         print "found.\n";
806     } else {
807         print "NOT found.\n";
808     }
809
810     print "Trying struct stat st_uatime...";
811     my $has_stat_st_uxtime;
812     if (try_compile_and_link(<<EOM)) {
813 $TIME_HEADERS
814 #include <sys/stat.h>
815 int main(int argc, char** argv) {
816     struct stat st;
817     st.st_uatime = 0;
818 }
819 EOM
820       $has_stat_st_uxtime++;
821       DEFINE('TIME_HIRES_STAT_ST_UXTIME');  # 5
822     }
823
824     if ($has_stat_st_uxtime) {
825         print "found.\n";
826     } else {
827         print "NOT found.\n";
828     }
829
830     # See HiRes.xs hrstatns()
831     if ($has_stat_st_xtimespec) {
832         DEFINE('TIME_HIRES_STAT', 1);
833     } elsif ($has_stat_st_xtimensec) {
834         DEFINE('TIME_HIRES_STAT', 2);
835     } elsif ($has_stat_st_xtime_n) {
836         DEFINE('TIME_HIRES_STAT', 3);
837     } elsif ($has_stat_st_xtim) {
838         DEFINE('TIME_HIRES_STAT', 4);
839     } elsif ($has_stat_st_uxtime) {
840         DEFINE('TIME_HIRES_STAT', 5);
841     }    
842
843    if ($DEFINE =~ /-DTIME_HIRES_STAT=\d+/) {
844     print "You seem to have stat() subsecond timestamps.\n";
845     print "(Your struct stat has them, but the filesystems must help.)\n";
846    } else {
847     print "You do not seem to have stat subsecond timestamps.\n";
848    }
849
850     my $has_w32api_windows_h;
851
852     if ($^O eq 'cygwin') {
853         print "Looking for <w32api/windows.h>... ";
854         if (has_include('w32api/windows.h')) {
855             $has_w32api_windows_h++;
856             DEFINE('HAS_W32API_WINDOWS_H');
857         }
858         if ($has_w32api_windows_h) {
859             print "found.\n";
860         } else {
861             print "NOT found.\n";
862         }
863     }
864
865     if ($DEFINE) {
866         $DEFINE =~ s/^\s+//;
867         if (open(XDEFINE, '>', 'xdefine')) {
868             print XDEFINE $DEFINE, "\n";
869             close(XDEFINE);
870         }
871     }
872 }
873
874 sub doMakefile {
875     my @makefileopts = ();
876
877     if ($] >= 5.005) {
878         push (@makefileopts,
879             'AUTHOR'    => 'Jarkko Hietaniemi <jhi@iki.fi>',
880             'ABSTRACT_FROM' => 'HiRes.pm',
881         );
882         DEFINE('ATLEASTFIVEOHOHFIVE');
883     }
884
885     push (@makefileopts,
886         'NAME'  => 'Time::HiRes',
887         'VERSION_FROM' => 'HiRes.pm', # finds $VERSION
888         'LIBS'  => $LIBS,   # e.g., '-lm'
889         'DEFINE'        => $DEFINE,     # e.g., '-DHAS_SOMETHING'
890         'XSOPT' => $XSOPT,
891           # Do not even think about 'INC' => '-I/usr/ucbinclude',
892           # Solaris will avenge.
893         'INC'   => '',     # e.g., '-I/usr/include/other'
894         'INSTALLDIRS' => ($] >= 5.008 && $] < 5.011 ? 'perl' : 'site'),
895         'PREREQ_PM' => {
896             'Carp' => 0,
897             'Config' => 0,
898             'DynaLoader' => 0,
899             'Exporter' => 0,
900             'ExtUtils::MakeMaker' => 0,
901             'Test::More' => 0,
902             'strict' => 0,
903         },
904         'dist'      => {
905             'CI'       => 'ci -l',
906             'COMPRESS' => 'gzip -9f',
907             'SUFFIX'   => 'gz',
908         },
909         clean => { FILES => "xdefine" },
910         realclean => { FILES=> 'const-c.inc const-xs.inc' },
911     );
912
913     if ($^O eq "MSWin32" && !(grep { /\ALD[A-Z]*=/ } @ARGV)) {
914         my $libperl = $Config{libperl} || "";
915         my $gccversion = $Config{gccversion} || "";
916         if ($gccversion =~ /\A3\.4\.[0-9]+/ and $libperl =~ /\.lib\z/) {
917             # Avoid broken linkage with ActivePerl, by linking directly
918             # against the Perl DLL rather than the import library.
919             (my $llibperl = "-l$libperl") =~ s/\.lib\z//;
920             my $lddlflags = $Config{lddlflags} || "";
921             my $ldflags = $Config{ldflags} || "";
922             s/-L(?:".*?"|\S+)//g foreach $lddlflags, $ldflags;
923             my $libdirs = join ' ',
924                 map { s/(?<!\\)((?:\\\\)*")/\\$1/g; qq[-L"$_"] }
925                 @Config{qw/bin sitebin/};
926             push @makefileopts, macro => {
927                 LDDLFLAGS => "$lddlflags $libdirs $llibperl",
928                 LDFLAGS => "$ldflags $libdirs $llibperl",
929                 PERL_ARCHIVE => "",
930             };
931         }
932     }
933
934     if ($ENV{PERL_CORE}) {
935         push @makefileopts, MAN3PODS => {};
936     }
937
938     if ($ExtUtils::MakeMaker::VERSION >= 6.48) {
939         push @makefileopts, (MIN_PERL_VERSION => '5.008',);
940     }
941
942     if ($ExtUtils::MakeMaker::VERSION >= 6.31) {
943         push @makefileopts, (LICENSE => 'perl_5');
944     }
945
946     WriteMakefile(@makefileopts);
947 }
948
949 sub doConstants {
950     if (eval {require ExtUtils::Constant; 1}) {
951         my @names = qw(
952                        CLOCKS_PER_SEC
953                        CLOCK_BOOTTIME
954                        CLOCK_HIGHRES
955                        CLOCK_MONOTONIC
956                        CLOCK_MONOTONIC_COARSE
957                        CLOCK_MONOTONIC_PRECISE
958                        CLOCK_MONOTONIC_RAW
959                        CLOCK_PROCESS_CPUTIME_ID
960                        CLOCK_REALTIME
961                        CLOCK_REALTIME_COARSE
962                        CLOCK_REALTIME_FAST
963                        CLOCK_REALTIME_PRECISE
964                        CLOCK_SECOND
965                        CLOCK_SOFTTIME
966                        CLOCK_THREAD_CPUTIME_ID
967                        CLOCK_TIMEOFDAY
968                        CLOCK_UPTIME
969                        CLOCK_UPTIME_FAST
970                        CLOCK_UPTIME_PRECISE
971                        ITIMER_PROF
972                        ITIMER_REAL
973                        ITIMER_REALPROF
974                        ITIMER_VIRTUAL
975                        TIMER_ABSTIME
976                       );
977         foreach (qw (d_usleep d_ualarm d_gettimeofday d_getitimer d_setitimer
978                      d_nanosleep d_clock_gettime d_clock_getres
979                      d_clock d_clock_nanosleep d_hires_stat
980                      d_futimens d_utimensat d_hires_utime)) {
981             my $macro = $_;
982             if ($macro =~ /^(d_nanosleep|d_clock)$/) {
983                 $macro =~ s/^d_(.+)/TIME_HIRES_\U$1/;
984             } elsif ($macro =~ /^(d_hires_stat)$/) {
985                 my $d_hires_stat = 0;
986                 $d_hires_stat = $1 if ($DEFINE =~ /-DTIME_HIRES_STAT=(\d+)/);
987                 push @names, {name => $_, macro => "TIME_HIRES_STAT", value => $d_hires_stat,
988                               default => ["IV", "0"]};
989                 next;
990             } elsif ($macro =~ /^(d_hires_utime)$/) {
991                 my $d_hires_utime =
992                     ($DEFINE =~ /-DHAS_FUTIMENS/ ||
993                      $DEFINE =~ /-DHAS_UTIMENSAT/) ? 1 : 0;
994                 push @names, {name => $_, macro => "TIME_HIRES_UTIME", value => $d_hires_utime,
995                               default => ["IV", "0"]};
996                 next;
997             } elsif ($macro =~ /^(d_clock_gettime|d_clock_getres|d_clock_nanosleep)$/) {
998                 $macro =~ s/^d_(.+)/TIME_HIRES_\U$1/;
999                 my $val = ($DEFINE =~ /-D$macro\b/) ? 1 : 0;
1000                 push @names, {name => $_, macro => $macro, value => $val,
1001                               default => ["IV", "0"]};
1002                 next;
1003             } else {
1004                 $macro =~ s/^d_(.+)/HAS_\U$1/;
1005             }
1006             push @names, {name => $_, macro => $macro, value => 1,
1007                           default => ["IV", "0"]};
1008         }
1009         ExtUtils::Constant::WriteConstants(
1010                                            NAME => 'Time::HiRes',
1011                                            NAMES => \@names,
1012                                           );
1013     } else {
1014         my $file;
1015         foreach $file ('const-c.inc', 'const-xs.inc') {
1016             my $fallback = File::Spec->catfile('fallback', $file);
1017             local $/;
1018             open IN, '<', $fallback or die "Can't open $fallback: $!";
1019             open OUT, '>', $file or die "Can't open $file: $!";
1020             print OUT <IN> or die $!;
1021             close OUT or die "Can't close $file: $!";
1022             close IN or die "Can't close $fallback: $!";
1023         }
1024     }
1025 }
1026
1027 sub main {
1028     if (-f "xdefine" && !(@ARGV  && $ARGV[0] eq '--configure')) {
1029         print qq[$0: The "xdefine" exists, skipping the configure step.\n];
1030         print qq[("$^X $0 --configure" to force the configure step)\n];
1031     } else {
1032         print "Configuring Time::HiRes...\n";
1033         1 while unlink("define");
1034         if ($^O =~ /Win32/i) {
1035             DEFINE('SELECT_IS_BROKEN');
1036             $LIBS = [];
1037             print "System is $^O, skipping full configure...\n";
1038             open(XDEFINE, '>', 'xdefine') or die "$0: Cannot create xdefine: $!\n";
1039             close(XDEFINE);
1040         } else {
1041             init();
1042         }
1043         doMakefile;
1044         doConstants;
1045     }
1046     my $make = $Config{'make'} || "make";
1047     unless (exists $ENV{PERL_CORE} && $ENV{PERL_CORE}) {
1048         print  <<EOM;
1049 Now you may issue '$make'.  Do not forget also '$make test'.
1050 EOM
1051        if ($] == 5.008 &&
1052            ((exists $ENV{LC_ALL}   && $ENV{LC_ALL}   =~ /utf-?8/i) ||
1053             (exists $ENV{LC_CTYPE} && $ENV{LC_CTYPE} =~ /utf-?8/i) ||
1054             (exists $ENV{LANG}     && $ENV{LANG}     =~ /utf-?8/i))) {
1055             print <<EOM;
1056
1057 NOTE: if you get an error like this (the Makefile line number may vary):
1058 Makefile:91: *** missing separator
1059 then set the environment variable LC_ALL to "C" and retry
1060 from scratch (re-run perl "Makefile.PL").
1061 (And consider upgrading your Perl to, say, at least Perl 5.8.8.)
1062 (You got this message because you seem to have
1063  an UTF-8 locale active in your shell environment, this used
1064  to cause broken Makefiles to be created from Makefile.PLs)
1065 EOM
1066         }
1067     }
1068 }
1069
1070 &main;
1071
1072 # EOF