ViewVC Help
View File | Revision Log | Show Annotations | Download File
/cvs/IO-AIO/AIO.xs
Revision: 1.303
Committed: Sun Mar 30 13:03:20 2025 UTC (7 weeks, 4 days ago) by root
Branch: MAIN
CVS Tags: HEAD
Changes since 1.302: +49 -38 lines
Log Message:
*** empty log message ***

File Contents

# Content
1 #include "libeio/xthread.h"
2
3 #include <errno.h>
4
5 #include "EXTERN.h"
6 #include "perl.h"
7 #include "XSUB.h"
8 #include "perliol.h"
9
10 #if !defined mg_findext
11 # define mg_findext(sv,type,vtbl) mg_find (sv, type)
12 #endif
13
14 #include <stddef.h>
15 #include <stdlib.h>
16 #include <errno.h>
17 #include <sys/types.h>
18 #include <sys/socket.h>
19 #include <sys/stat.h>
20 #include <limits.h>
21 #include <fcntl.h>
22 #include <sched.h>
23
24 /* perl namespace pollution */
25 #undef VERSION
26
27 /* perl stupidly overrides readdir and maybe others */
28 /* with thread-unsafe versions, imagine that :( */
29 #undef readdir
30 #undef opendir
31 #undef closedir
32
33 #ifdef _WIN32
34
35 // perl overrides all those nice libc functions
36
37 #undef malloc
38 #undef free
39 #undef open
40 #undef read
41 #undef write
42 #undef send
43 #undef recv
44 #undef stat
45 #undef lstat
46 #undef fstat
47 #undef truncate
48 #undef ftruncate
49 #undef open
50 #undef link
51 #undef close
52 #undef unlink
53 #undef mkdir
54 #undef rmdir
55 #undef rename
56 #undef lseek
57 #undef opendir
58 #undef readdir
59 #undef closedir
60 #undef chmod
61 #undef fchmod
62 #undef dup
63 #undef dup2
64 #undef abort
65 #undef pipe
66 #undef utime
67
68 #define EIO_STRUCT_STAT struct _stati64
69 #define EIO_STRUCT_STATI64
70
71 #else
72
73 #include <sys/time.h>
74 #include <sys/select.h>
75 #include <sys/wait.h>
76 #include <unistd.h>
77 #include <utime.h>
78 #include <signal.h>
79
80 #define EIO_STRUCT_STAT Stat_t
81
82 #endif
83
84 /*****************************************************************************/
85
86 #if __GNUC__ >= 3
87 # define expect(expr,value) __builtin_expect ((expr),(value))
88 #else
89 # define expect(expr,value) (expr)
90 #endif
91
92 #define expect_false(expr) expect ((expr) != 0, 0)
93 #define expect_true(expr) expect ((expr) != 0, 1)
94
95 /*****************************************************************************/
96
97 #include "config.h"
98
99 #if HAVE_SYS_MKDEV_H
100 # include <sys/mkdev.h>
101 #elif HAVE_SYS_SYSMACROS_H
102 # include <sys/sysmacros.h>
103 #endif
104
105 #if _POSIX_MEMLOCK || _POSIX_MEMLOCK_RANGE || _POSIX_MAPPED_FILES
106 # include <sys/mman.h>
107 #endif
108
109 #if HAVE_SYS_UIO_H
110 # include <sys/uio.h>
111 #endif
112
113 /* MUST be included before linux/fs.h, as the latter includes
114 * linux/mount.h, which is incompatible to sys/mount.h
115 */
116 #if HAVE_SYS_MOUNT_H
117 # include <sys/mount.h>
118 #endif
119
120 /* the incompetent fool that created musl keeps __linux__, refuses
121 * to implement any linux standard apis, and also has no way to test
122 * for his broken implementation. don't complain to me if this fails
123 * for you.
124 */
125 #if __linux__ && (defined __GLIBC__ || defined __UCLIBC__)
126 # include <linux/fs.h> /* MUST be included after sys/mount.h */
127 # ifdef FS_IOC_FIEMAP
128 # include <linux/types.h>
129 # include <linux/fiemap.h>
130 # undef HAVE_FIEMAP
131 # define HAVE_FIEMAP 1
132 # endif
133 #endif
134
135 #if HAVE_ST_XTIMENSEC
136 # define ATIMENSEC PL_statcache.st_atimensec
137 # define MTIMENSEC PL_statcache.st_mtimensec
138 # define CTIMENSEC PL_statcache.st_ctimensec
139 #elif HAVE_ST_XTIMESPEC
140 # define ATIMENSEC PL_statcache.st_atim.tv_nsec
141 # define MTIMENSEC PL_statcache.st_mtim.tv_nsec
142 # define CTIMENSEC PL_statcache.st_ctim.tv_nsec
143 #else
144 # define ATIMENSEC 0
145 # define MTIMENSEC 0
146 # define CTIMENSEC 0
147 #endif
148
149 #if HAVE_ST_BIRTHTIMENSEC
150 # define BTIMESEC PL_statcache.st_birthtime
151 # define BTIMENSEC PL_statcache.st_birthtimensec
152 #elif HAVE_ST_BIRTHTIMESPEC
153 # define BTIMESEC PL_statcache.st_birthtim.tv_sec
154 # define BTIMENSEC PL_statcache.st_birthtim.tv_nsec
155 #else
156 # define BTIMESEC 0
157 # define BTIMENSEC 0
158 #endif
159
160 #if HAVE_ST_GEN
161 # define ST_GEN PL_statcache.st_gen
162 #else
163 # define ST_GEN 0
164 #endif
165
166 #include "schmorp.h"
167
168 #if HAVE_EVENTFD
169 # include <sys/eventfd.h>
170 #endif
171
172 #if HAVE_TIMERFD
173 # include <sys/timerfd.h>
174 #endif
175
176 #if HAVE_RLIMITS
177 #include <sys/time.h>
178 #include <sys/resource.h>
179 #endif
180
181 typedef SV SV8; /* byte-sv, used for argument-checking */
182 typedef char *octet_string;
183 typedef char *octet_string_ornull;
184 typedef int aio_rfd; /* read file desriptor */
185 typedef int aio_wfd; /* write file descriptor */
186
187 static HV *aio_stash, *aio_req_stash, *aio_grp_stash, *aio_wd_stash;
188
189 #define EIO_REQ_MEMBERS \
190 SV *callback; \
191 SV *sv1, *sv2; \
192 SV *sv3, *sv4; \
193 STRLEN stroffset; \
194 SV *self;
195
196 #define EIO_NO_WRAPPERS 1
197 #define EIO_TSTAMP_IS_DOUBLE 0
198
199 #include "libeio/eio.h"
200
201 static int req_invoke (eio_req *req);
202 #define EIO_FINISH(req) req_invoke (req)
203 static void req_destroy (eio_req *grp);
204 #define EIO_DESTROY(req) req_destroy (req)
205
206 #include "libeio/eio.c"
207
208 #if !HAVE_POSIX_FADVISE
209 # define posix_fadvise(a,b,c,d) errno = ENOSYS /* also return ENOSYS */
210 #endif
211
212 #if !HAVE_POSIX_MADVISE
213 # define posix_madvise(a,b,c) errno = ENOSYS /* also return ENOSYS */
214 #endif
215
216 #ifndef MAP_ANONYMOUS
217 # ifdef MAP_ANON
218 # define MAP_ANONYMOUS MAP_ANON
219 # else
220 # define MAP_ANONYMOUS MAP_FIXED /* and hope this fails */
221 # endif
222 #endif
223
224 #ifndef makedev
225 # define makedev(maj,min) (((maj) << 8) | (min))
226 #endif
227 #ifndef major
228 # define major(dev) ((dev) >> 8)
229 #endif
230 #ifndef minor
231 # define minor(dev) ((dev) & 0xff)
232 #endif
233
234 /* solaris has a non-posix/unix compliant PAGESIZE that breaks compilation */
235 #ifdef __sun
236 # undef PAGESIZE
237 #endif
238
239 #if PAGESIZE <= 0
240 # define PAGESIZE sysconf (_SC_PAGESIZE)
241 #endif
242
243 /* solaris perl seems to declare a wrong syscall function that clashes with system includes */
244 #ifdef __sun
245 # undef HAVE_SYSCALL
246 #endif
247
248 #if HAVE_SYSCALL
249 #include <sys/syscall.h>
250 #else
251 # define syscall(nr,...) (errno = ENOSYS, -1)
252 #endif
253
254 /*****************************************************************************/
255
256 #if !_POSIX_MAPPED_FILES
257 # define mmap(addr,length,prot,flags,fd,offs) (errno = ENOSYS, (void *)-1)
258 # define munmap(addr,length) EIO_ENOSYS ()
259 #endif
260
261 #if !_POSIX_MEMORY_PROTECTION
262 # define mprotect(addr,len,prot) EIO_ENOSYS ()
263 #endif
264
265 #if !MREMAP_MAYMOVE
266 # define mremap(old_address,old_size,new_size,flags,new_address) (errno = ENOSYS, (void *)-1)
267 #endif
268
269 #define FOREIGN_MAGIC PERL_MAGIC_ext
270
271 static int ecb_cold
272 mmap_free (pTHX_ SV *sv, MAGIC *mg)
273 {
274 int old_errno = errno;
275 munmap (mg->mg_ptr, (size_t)mg->mg_obj);
276 errno = old_errno;
277
278 mg->mg_obj = 0; /* just in case */
279
280 SvREADONLY_off (sv);
281
282 if (SvPVX (sv) != mg->mg_ptr)
283 croak ("ERROR: IO::AIO::mmap-mapped scalar changed location, detected");
284
285 SvCUR_set (sv, 0);
286 SvPVX (sv) = 0;
287 SvOK_off (sv);
288
289 return 0;
290 }
291
292 static MGVTBL mmap_vtbl = {
293 0, 0, 0, 0, mmap_free
294 };
295
296 static int ecb_cold
297 sysfree_free (pTHX_ SV *sv, MAGIC *mg)
298 {
299 free (mg->mg_ptr);
300 mg->mg_obj = 0; /* just in case */
301
302 SvREADONLY_off (sv);
303
304 if (SvPVX (sv) != mg->mg_ptr)
305 croak ("ERROR: IO::AIO mapped scalar changed location, detected");
306
307 SvCUR_set (sv, 0);
308 SvPVX (sv) = 0;
309 SvOK_off (sv);
310
311 return 0;
312 }
313
314 static MGVTBL sysfree_vtbl = {
315 0, 0, 0, 0, sysfree_free
316 };
317
318 /*****************************************************************************/
319
320 /* helper: set scalar to foreign ptr with custom free */
321 ecb_noinline
322 static void
323 sv_set_foreign (SV *sv, const MGVTBL *const vtbl, void *addr, IV length)
324 {
325 sv_force_normal (sv);
326
327 /* we store the length in mg_obj, as namlen is I32 :/ */
328 sv_magicext (sv, 0, FOREIGN_MAGIC, vtbl, (char *)addr, 0)
329 ->mg_obj = (SV *)length;
330
331 SvUPGRADE (sv, SVt_PV); /* nop... */
332
333 if (SvLEN (sv))
334 Safefree (SvPVX (sv));
335
336 SvPVX (sv) = (char *)addr;
337 SvCUR_set (sv, length);
338 SvLEN_set (sv, 0);
339 SvPOK_only (sv);
340 }
341
342 static void
343 sv_clear_foreign (SV *sv)
344 {
345 /* todo: iterate over magic and only free ours, but of course */
346 /* the perl5porters will call that (correct) behaviour buggy */
347 sv_unmagic (sv, FOREIGN_MAGIC);
348 }
349
350 /*****************************************************************************/
351
352 /* defines all sorts of constants to 0 unless they are already defined */
353 /* also provides const_iv_ and const_niv_ macros for them */
354 #include "def0.h"
355
356 /*****************************************************************************/
357
358 static void
359 fiemap (eio_req *req)
360 {
361 req->result = -1;
362
363 #if HAVE_FIEMAP
364 /* assume some c99 */
365 struct fiemap *fiemap = 0;
366 size_t end_offset;
367 int count = req->int3;
368
369 req->flags |= EIO_FLAG_PTR1_FREE;
370
371 /* heuristic: start with 512 bytes (8 extents), and if that isn't enough, */
372 /* increase in fixed steps */
373 if (count < 0)
374 count = 8;
375
376 fiemap = malloc (sizeof (*fiemap) + sizeof (struct fiemap_extent) * count);
377 errno = ENOMEM;
378 if (!fiemap)
379 return;
380
381 req->ptr1 = fiemap;
382
383 fiemap->fm_start = req->offs;
384 fiemap->fm_length = req->size;
385 fiemap->fm_flags = req->int2;
386 fiemap->fm_extent_count = count;
387
388 if (ioctl (req->int1, FS_IOC_FIEMAP, fiemap) < 0)
389 return;
390
391 if (req->int3 >= 0 /* not autosizing */
392 || !fiemap->fm_mapped_extents /* no more extents */
393 || fiemap->fm_extents [fiemap->fm_mapped_extents - 1].fe_flags & FIEMAP_EXTENT_LAST /* hit eof */)
394 goto done;
395
396 /* else we have to loop -
397 * it would be tempting (actually I tried that first) to just query the
398 * number of extents needed, but linux often feels like not returning all
399 * extents, without telling us it left any out. this complicates
400 * this quite a bit.
401 */
402
403 end_offset = fiemap->fm_length + (fiemap->fm_length == FIEMAP_MAX_OFFSET ? 0 : fiemap->fm_start);
404
405 for (;;)
406 {
407 /* we go in 54 extent steps - 3kb, in the hope that this fits nicely on the eio stack (normally 16+ kb) */
408 char scratch[3072];
409 struct fiemap *incmap = (struct fiemap *)scratch;
410
411 incmap->fm_start = fiemap->fm_extents [fiemap->fm_mapped_extents - 1].fe_logical
412 + fiemap->fm_extents [fiemap->fm_mapped_extents - 1].fe_length;
413 incmap->fm_length = fiemap->fm_length - (incmap->fm_start - fiemap->fm_start);
414 incmap->fm_flags = fiemap->fm_flags;
415 incmap->fm_extent_count = (sizeof (scratch) - sizeof (struct fiemap)) / sizeof (struct fiemap_extent);
416
417 if (ioctl (req->int1, FS_IOC_FIEMAP, incmap) < 0)
418 return;
419
420 if (!incmap->fm_mapped_extents)
421 goto done;
422
423 count = fiemap->fm_mapped_extents + incmap->fm_mapped_extents;
424 fiemap = realloc (fiemap, sizeof (*fiemap) + sizeof (struct fiemap_extent) * count);
425 errno = ENOMEM;
426 if (!fiemap)
427 return;
428
429 req->ptr1 = fiemap;
430
431 for (count = 0; count < incmap->fm_mapped_extents; ++count)
432 {
433 struct fiemap_extent *e = incmap->fm_extents + count;
434
435 fiemap->fm_extents [fiemap->fm_mapped_extents++] = *e;
436
437 if (e->fe_logical >= end_offset)
438 goto done;
439
440 if (e->fe_flags & FIEMAP_EXTENT_LAST)
441 goto done;
442
443 }
444 }
445
446 done:
447 req->result = 0;
448
449 #else
450 errno = ENOSYS;
451 #endif
452 }
453
454 /*****************************************************************************/
455
456 static int close_fd; /* dummy fd to close fds via dup2 */
457
458 #if HAVE_STATX
459 static struct statx stx;
460 #define statx_offsetof(member) offsetof (struct statx, member)
461 #define eio__statx statx
462 #else
463 #define statx_offsetof(member) 0
464 #define eio__statx(dir,path,flags,mask,stx) EIO_ENOSYS()
465 #endif
466
467 enum {
468 FLAG_SV2_RO_OFF = 0x40, /* data was set readonly */
469 };
470
471 typedef eio_req *aio_req;
472 typedef eio_req *aio_req_ornot;
473 typedef eio_wd aio_wd;
474
475 static SV *on_next_submit;
476 static int next_pri = EIO_PRI_DEFAULT;
477 static int max_outstanding;
478
479 static s_epipe respipe;
480
481 static void req_destroy (aio_req req);
482 static void req_cancel (aio_req req);
483
484 static void
485 want_poll (void)
486 {
487 /* write a dummy byte to the pipe so fh becomes ready */
488 s_epipe_signal (&respipe);
489 }
490
491 static void
492 done_poll (void)
493 {
494 /* read any signals sent by the worker threads */
495 s_epipe_drain (&respipe);
496 }
497
498 /* must be called at most once */
499 ecb_noinline
500 static SV *
501 req_sv (aio_req req, HV *stash)
502 {
503 if (!req->self)
504 {
505 req->self = (SV *)newHV ();
506 sv_magic (req->self, 0, PERL_MAGIC_ext, (char *)req, 0);
507 }
508
509 return sv_2mortal (sv_bless (newRV_inc (req->self), stash));
510 }
511
512 static SV *
513 newSVaio_wd (aio_wd wd)
514 {
515 return sv_bless (newRV_noinc (newSViv ((intptr_t)wd)), aio_wd_stash);
516 }
517
518 ecb_noinline
519 static aio_req
520 SvAIO_REQ (SV *sv)
521 {
522 MAGIC *mg;
523
524 if (!SvROK (sv)
525 /* for speed reasons, we do not verify that SvROK actually has a stash ptr */
526 || (SvSTASH (SvRV (sv)) != aio_grp_stash
527 && SvSTASH (SvRV (sv)) != aio_req_stash
528 && !sv_derived_from (sv, "IO::AIO::REQ")))
529 croak ("object of class IO::AIO::REQ expected");
530
531 mg = mg_find (SvRV (sv), PERL_MAGIC_ext);
532
533 return mg ? (aio_req)mg->mg_ptr : 0;
534 }
535
536 static aio_wd
537 SvAIO_WD (SV *sv)
538 {
539 if (!SvROK (sv)
540 || SvTYPE (SvRV (sv)) != SVt_PVMG
541 || SvSTASH (SvRV (sv)) != aio_wd_stash)
542 croak ("IO::AIO: expected a working directory object as returned by aio_wd");
543
544 return (aio_wd)(long)SvIVX (SvRV (sv));
545 }
546
547 static SV *
548 newmortalFH (int fd, int flags)
549 {
550 if (fd < 0)
551 return &PL_sv_undef;
552
553 GV *gv = (GV *)sv_newmortal ();
554 char sym[64];
555 int symlen;
556
557 symlen = snprintf (sym, sizeof (sym), "fd#%d", fd);
558 gv_init (gv, aio_stash, sym, symlen, 0);
559
560 symlen = snprintf (
561 sym,
562 sizeof (sym),
563 "%s&=%d",
564 flags == O_RDONLY ? "<" : flags == O_WRONLY ? ">" : "+<",
565 fd
566 );
567
568 return do_open (gv, sym, symlen, 0, 0, 0, 0)
569 ? (SV *)gv : &PL_sv_undef;
570 }
571
572 static void
573 aio_grp_feed (aio_req grp)
574 {
575 if (grp->sv2 && SvOK (grp->sv2))
576 {
577 dSP;
578
579 ENTER;
580 SAVETMPS;
581 PUSHMARK (SP);
582 XPUSHs (req_sv (grp, aio_grp_stash));
583 PUTBACK;
584 call_sv (grp->sv2, G_VOID | G_EVAL | G_KEEPERR);
585 SPAGAIN;
586 FREETMPS;
587 LEAVE;
588 }
589 }
590
591 ecb_noinline
592 static void
593 req_submit (eio_req *req)
594 {
595 eio_submit (req);
596
597 if (expect_false (on_next_submit))
598 {
599 dSP;
600 SV *cb = sv_2mortal (on_next_submit);
601
602 on_next_submit = 0;
603
604 PUSHMARK (SP);
605 PUTBACK;
606 call_sv (cb, G_DISCARD | G_EVAL);
607 }
608 }
609
610 static int
611 req_invoke (eio_req *req)
612 {
613 if (req->flags & FLAG_SV2_RO_OFF)
614 SvREADONLY_off (req->sv2);
615
616 if (!EIO_CANCELLED (req) && req->callback)
617 {
618 dSP;
619 static SV *sv_result_cache; /* caches the result integer SV */
620 SV *sv_result;
621
622 ENTER;
623 SAVETMPS;
624 PUSHMARK (SP);
625 EXTEND (SP, 1);
626
627 /* do not recreate the result IV from scratch each time */
628 if (expect_true (sv_result_cache))
629 {
630 sv_result = sv_result_cache; sv_result_cache = 0;
631 SvIV_set (sv_result, req->result);
632 SvIOK_only (sv_result);
633 }
634 else
635 {
636 sv_result = newSViv (req->result);
637 SvREADONLY_on (sv_result);
638 }
639
640 switch (req->type)
641 {
642 case EIO_WD_OPEN:
643 PUSHs (req->result ? &PL_sv_undef : sv_2mortal (newSVaio_wd (req->wd)));
644 break;
645
646 case EIO_READDIR:
647 {
648 SV *rv = &PL_sv_undef;
649
650 if (req->result >= 0)
651 {
652 int i;
653 char *names = (char *)req->ptr2;
654 eio_dirent *ent = (eio_dirent *)req->ptr1; /* might be 0 */
655 AV *av = newAV ();
656
657 av_extend (av, req->result - 1);
658
659 for (i = 0; i < req->result; ++i)
660 {
661 if (req->int1 & EIO_READDIR_DENTS)
662 {
663 SV *namesv = newSVpvn (names + ent->nameofs, ent->namelen);
664
665 if (req->int1 & EIO_READDIR_CUSTOM2)
666 {
667 static SV *sv_type [EIO_DT_MAX + 1]; /* type sv cache */
668 AV *avent = newAV ();
669
670 av_extend (avent, 2);
671
672 if (!sv_type [ent->type])
673 {
674 sv_type [ent->type] = newSViv (ent->type);
675 SvREADONLY_on (sv_type [ent->type]);
676 }
677
678 av_store (avent, 0, namesv);
679 av_store (avent, 1, SvREFCNT_inc (sv_type [ent->type]));
680 av_store (avent, 2, IVSIZE >= 8 ? newSVuv (ent->inode) : newSVnv (ent->inode));
681
682 av_store (av, i, newRV_noinc ((SV *)avent));
683 }
684 else
685 av_store (av, i, namesv);
686
687 ++ent;
688 }
689 else
690 {
691 SV *name = newSVpv (names, 0);
692 av_store (av, i, name);
693 names += SvCUR (name) + 1;
694 }
695 }
696
697 rv = sv_2mortal (newRV_noinc ((SV *)av));
698 }
699
700 PUSHs (rv);
701
702 if (req->int1 & EIO_READDIR_CUSTOM1)
703 XPUSHs (sv_2mortal (newSViv (req->int1 & ~(EIO_READDIR_CUSTOM1 | EIO_READDIR_CUSTOM2))));
704 }
705 break;
706
707 case EIO_OPEN:
708 PUSHs (newmortalFH (req->result, req->int1 & (O_RDONLY | O_WRONLY | O_RDWR)));
709 break;
710
711 case EIO_STATVFS:
712 case EIO_FSTATVFS:
713 {
714 SV *rv = &PL_sv_undef;
715
716 #ifndef _WIN32
717 if (req->result >= 0)
718 {
719 EIO_STRUCT_STATVFS *f = EIO_STATVFS_BUF (req);
720 HV *hv = newHV ();
721 /* POSIX requires fsid to be unsigned long, but AIX in its infinite wisdom
722 * chooses to make it a struct.
723 */
724 unsigned long fsid = 0;
725 memcpy (&fsid, &f->f_fsid, sizeof (unsigned long) < sizeof (f->f_fsid) ? sizeof (unsigned long) : sizeof (f->f_fsid));
726
727 rv = sv_2mortal (newRV_noinc ((SV *)hv));
728
729 hv_store (hv, "bsize" , sizeof ("bsize" ) - 1, newSVval64 (f->f_bsize ), 0);
730 hv_store (hv, "frsize" , sizeof ("frsize" ) - 1, newSVval64 (f->f_frsize ), 0);
731 hv_store (hv, "blocks" , sizeof ("blocks" ) - 1, newSVval64 (f->f_blocks ), 0);
732 hv_store (hv, "bfree" , sizeof ("bfree" ) - 1, newSVval64 (f->f_bfree ), 0);
733 hv_store (hv, "bavail" , sizeof ("bavail" ) - 1, newSVval64 (f->f_bavail ), 0);
734 hv_store (hv, "files" , sizeof ("files" ) - 1, newSVval64 (f->f_files ), 0);
735 hv_store (hv, "ffree" , sizeof ("ffree" ) - 1, newSVval64 (f->f_ffree ), 0);
736 hv_store (hv, "favail" , sizeof ("favail" ) - 1, newSVval64 (f->f_favail ), 0);
737 hv_store (hv, "fsid" , sizeof ("fsid" ) - 1, newSVval64 (fsid ), 0);
738 hv_store (hv, "flag" , sizeof ("flag" ) - 1, newSVval64 (f->f_flag ), 0);
739 hv_store (hv, "namemax", sizeof ("namemax") - 1, newSVval64 (f->f_namemax), 0);
740 }
741 #endif
742
743 PUSHs (rv);
744 }
745
746 break;
747
748 case EIO_GROUP:
749 req->int1 = 2; /* mark group as finished */
750
751 if (req->sv1)
752 {
753 int i;
754 AV *av = (AV *)req->sv1;
755
756 EXTEND (SP, AvFILL (av) + 1);
757 for (i = 0; i <= AvFILL (av); ++i)
758 PUSHs (*av_fetch (av, i, 0));
759 }
760 break;
761
762 case EIO_NOP:
763 case EIO_BUSY:
764 break;
765
766 case EIO_READLINK:
767 case EIO_REALPATH:
768 if (req->result > 0)
769 PUSHs (sv_2mortal (newSVpvn (req->ptr2, req->result)));
770 break;
771
772 case EIO_STAT:
773 case EIO_LSTAT:
774 case EIO_FSTAT:
775 PL_laststype = req->type == EIO_LSTAT ? OP_LSTAT : OP_STAT;
776
777 if (!(PL_laststatval = req->result))
778 /* if compilation fails here then perl's Stat_t is not struct _stati64 */
779 PL_statcache = *(EIO_STRUCT_STAT *)(req->ptr2);
780
781 PUSHs (sv_result);
782 break;
783
784 case EIO_SEEK:
785 PUSHs (req->result ? sv_result : sv_2mortal (newSVval64 (req->offs)));
786 break;
787
788 case EIO_READ:
789 {
790 SvCUR_set (req->sv2, req->stroffset + (req->result > 0 ? req->result : 0));
791 *SvEND (req->sv2) = 0;
792 SvPOK_only (req->sv2);
793 SvSETMAGIC (req->sv2);
794 PUSHs (sv_result);
795 }
796 break;
797
798 case EIO_SLURP:
799 {
800 if (req->result >= 0)
801 {
802 /* if length was originally not known, we steal the malloc'ed memory */
803 if (req->flags & EIO_FLAG_PTR2_FREE)
804 {
805 req->flags &= ~EIO_FLAG_PTR2_FREE;
806 sv_set_foreign (req->sv2, &sysfree_vtbl, req->ptr2, req->result);
807 }
808 else
809 {
810 SvCUR_set (req->sv2, req->result);
811 *SvEND (req->sv2) = 0;
812 SvPOK_only (req->sv2);
813 }
814
815 SvSETMAGIC (req->sv2);
816 }
817
818 PUSHs (sv_result);
819 }
820 break;
821
822 case EIO_CUSTOM:
823 if (req->feed == fiemap)
824 {
825 #if HAVE_FIEMAP
826 if (!req->result)
827 {
828 struct fiemap *fiemap = (struct fiemap *)req->ptr1;
829
830 if (fiemap->fm_extent_count)
831 {
832 AV *av = newAV ();
833 int i;
834
835 while (fiemap->fm_mapped_extents)
836 {
837 struct fiemap_extent *extent = &fiemap->fm_extents [--fiemap->fm_mapped_extents];
838 AV *ext_av = newAV ();
839
840 av_store (ext_av, 3, newSVuv (extent->fe_flags));
841 av_store (ext_av, 2, newSVval64 (extent->fe_length));
842 av_store (ext_av, 1, newSVval64 (extent->fe_physical));
843 av_store (ext_av, 0, newSVval64 (extent->fe_logical));
844
845 av_store (av, fiemap->fm_mapped_extents, newRV_noinc ((SV *)ext_av));
846 }
847
848 PUSHs (sv_2mortal (newRV_noinc ((SV *)av)));
849 }
850 else
851 {
852 SvIV_set (sv_result, fiemap->fm_mapped_extents);
853 PUSHs (sv_result);
854 }
855 }
856 #endif
857 }
858 else
859 PUSHs (sv_result);
860 break;
861
862 #if 0
863 case EIO_CLOSE:
864 PerlIOUnix_refcnt_dec (req->int1);
865 break;
866 #endif
867
868 case EIO_DUP2: /* EIO_DUP2 actually means aio_close(), so fudge result value */
869 if (req->result > 0)
870 SvIV_set (sv_result, 0);
871 /* FALLTHROUGH */
872
873 default:
874 PUSHs (sv_result);
875 break;
876 }
877
878 errno = req->errorno;
879
880 PUTBACK;
881 call_sv (req->callback, G_VOID | G_EVAL | G_DISCARD);
882 SPAGAIN;
883
884 if (expect_false (SvREFCNT (sv_result) != 1 || sv_result_cache))
885 SvREFCNT_dec (sv_result);
886 else
887 sv_result_cache = sv_result;
888
889 FREETMPS;
890 LEAVE;
891
892 PUTBACK;
893 }
894
895 return !!SvTRUE (ERRSV);
896 }
897
898 static void
899 req_destroy (aio_req req)
900 {
901 if (req->self)
902 {
903 sv_unmagic (req->self, PERL_MAGIC_ext);
904 SvREFCNT_dec (req->self);
905 }
906
907 SvREFCNT_dec (req->sv1);
908 SvREFCNT_dec (req->sv2);
909 SvREFCNT_dec (req->sv3);
910 SvREFCNT_dec (req->sv4);
911 SvREFCNT_dec (req->callback);
912
913 free (req);
914 }
915
916 static void
917 req_cancel_subs (aio_req grp)
918 {
919 if (grp->type != EIO_GROUP)
920 return;
921
922 SvREFCNT_dec (grp->sv2);
923 grp->sv2 = 0;
924
925 eio_grp_cancel (grp);
926 }
927
928 ecb_cold
929 static void
930 create_respipe (void)
931 {
932 if (s_epipe_renew (&respipe))
933 croak ("IO::AIO: unable to initialize result pipe");
934 }
935
936 static void
937 poll_wait (void)
938 {
939 while (eio_nreqs ())
940 {
941 int size;
942
943 X_LOCK (EIO_POOL->reslock);
944 size = EIO_POOL->res_queue.size;
945 X_UNLOCK (EIO_POOL->reslock);
946
947 if (size)
948 return;
949
950 etp_maybe_start_thread (EIO_POOL);
951
952 s_epipe_wait (&respipe);
953 }
954 }
955
956 static int
957 poll_cb (void)
958 {
959 for (;;)
960 {
961 int res = eio_poll ();
962
963 if (res > 0)
964 croak (0);
965
966 if (!max_outstanding || max_outstanding > eio_nreqs ())
967 return res;
968
969 poll_wait ();
970 }
971 }
972
973 ecb_cold
974 static void
975 reinit (void)
976 {
977 create_respipe ();
978
979 if (eio_init (want_poll, done_poll) < 0)
980 croak ("IO::AIO: unable to initialise eio library");
981 }
982
983 /*****************************************************************************/
984
985 static SV *
986 get_cb (SV *cb_sv)
987 {
988 SvGETMAGIC (cb_sv);
989 return SvOK (cb_sv) ? s_get_cv_croak (cb_sv) : 0;
990 }
991
992 ecb_noinline
993 static aio_req ecb_noinline
994 dreq (SV *callback)
995 {
996 SV *cb_cv;
997 aio_req req;
998 int req_pri = next_pri;
999 next_pri = EIO_PRI_DEFAULT;
1000
1001 cb_cv = get_cb (callback);
1002
1003 req = calloc (sizeof (*req), 1);
1004 if (!req)
1005 croak ("out of memory during eio_req allocation");
1006
1007 req->callback = SvREFCNT_inc (cb_cv);
1008 req->pri = req_pri;
1009
1010 return req;
1011 }
1012
1013 #define dREQ \
1014 aio_req req = dreq (callback); \
1015
1016 #define REQ_SEND \
1017 PUTBACK; \
1018 req_submit (req); \
1019 SPAGAIN; \
1020 \
1021 if (GIMME_V != G_VOID) \
1022 XPUSHs (req_sv (req, aio_req_stash));
1023
1024 /* *wdsv, *pathsv, *wd and *ptr must be 0-initialized */
1025 ecb_inline
1026 void
1027 req_set_path (SV *path, SV **wdsv, SV **pathsv, eio_wd *wd, void **ptr)
1028 {
1029 if (expect_false (SvROK (path)))
1030 {
1031 SV *rv = SvRV (path);
1032 SV *wdob;
1033
1034 if (SvTYPE (rv) == SVt_PVAV && AvFILLp (rv) == 1)
1035 {
1036 path = AvARRAY (rv)[1];
1037 wdob = AvARRAY (rv)[0];
1038
1039 if (SvOK (wdob))
1040 {
1041 *wd = SvAIO_WD (wdob);
1042 *wdsv = SvREFCNT_inc_NN (SvRV (wdob));
1043 }
1044 else
1045 *wd = EIO_INVALID_WD;
1046 }
1047 else if (SvTYPE (rv) == SVt_PVMG && SvSTASH (rv) == aio_wd_stash)
1048 {
1049 *wd = (aio_wd)(long)SvIVX (rv);
1050 *wdsv = SvREFCNT_inc_NN (rv);
1051 *ptr = ".";
1052 return; /* path set to "." */
1053 }
1054 else
1055 croak ("IO::AIO: pathname arguments must be specified as a string, an IO::AIO::WD object or a [IO::AIO::WD, path] pair");
1056 }
1057
1058 *pathsv = newSVsv (path);
1059 *ptr = SvPVbyte_nolen (*pathsv);
1060 }
1061
1062 ecb_noinline
1063 static void
1064 req_set_path1 (aio_req req, SV *path)
1065 {
1066 req_set_path (path, &req->sv1, &req->sv3, &req->wd, &req->ptr1);
1067 }
1068
1069 ecb_noinline
1070 static void
1071 req_set_fh_or_path (aio_req req, int type_path, int type_fh, SV *fh_or_path)
1072 {
1073 SV *rv = SvROK (fh_or_path) ? SvRV (fh_or_path) : fh_or_path;
1074
1075 switch (SvTYPE (rv))
1076 {
1077 case SVt_PVIO:
1078 case SVt_PVLV:
1079 case SVt_PVGV:
1080 req->type = type_fh;
1081 req->sv1 = newSVsv (fh_or_path);
1082 req->int1 = PerlIO_fileno (IoIFP (sv_2io (fh_or_path)));
1083 break;
1084
1085 default:
1086 req->type = type_path;
1087 req_set_path1 (req, fh_or_path);
1088 break;
1089 }
1090 }
1091
1092 /*****************************************************************************/
1093
1094 static void
1095 ts_set (struct timespec *ts, NV value)
1096 {
1097 ts->tv_sec = value;
1098 ts->tv_nsec = (value - ts->tv_sec) * 1e9;
1099 }
1100
1101 static NV
1102 ts_get (const struct timespec *ts)
1103 {
1104 return ts->tv_sec + ts->tv_nsec * 1e-9;
1105 }
1106
1107 static void
1108 sv_to_tstamp (SV *sv, eio_tstamp *ts)
1109 {
1110 if (!SvOK (sv))
1111 ts->n = EIO_TSTAMP_NSEC_NOW;
1112 else if (SvROK (sv) && SvTYPE (SvRV (sv)) == SVt_PVAV)
1113 {
1114 SV **s1 = av_fetch ((AV *)SvRV (sv), 0, 0);
1115 SV **s2 = av_fetch ((AV *)SvRV (sv), 1, 0);
1116
1117 ts->s = s1 ? SvUV (*s1) : 0;
1118 ts->n = s2 ? SvUV (*s2) : 0;
1119 }
1120 else
1121 {
1122 NV nv = SvNV (sv);
1123
1124 U32 n = nv * (1e-9 * (1. / 4294967296.));
1125 NV low = nv - ((NV)n) * 4294967296.;
1126 ts->s = low;
1127 ts->n = (low - ts->s) * 1e9;
1128 }
1129 }
1130
1131 static SV *
1132 newSVts (VAL64 sec, U32 nsec)
1133 {
1134 AV *av = newAV ();
1135
1136 av_store (av, 1, newSVuv (nsec));
1137 av_store (av, 0, newSVval64 (sec ));
1138
1139 return newRV_noinc ((SV *)av);
1140 }
1141
1142 /*****************************************************************************/
1143
1144 /* extract a ref-to-array of strings into a temporary c style string vector */
1145 static char **
1146 extract_stringvec (SV *sv, const char *croakmsg)
1147 {
1148 if (!SvROK (sv) || SvTYPE (SvRV (sv)) != SVt_PVAV)
1149 croak ("%s", croakmsg);
1150
1151 AV *av = (AV *)SvRV (sv);
1152 int i, nelem = av_len (av) + 1;
1153 char **vecp = (char **)SvPVX (sv_2mortal (newSV (sizeof (char *) * (nelem + 1))));
1154
1155 for (i = 0; i < nelem; ++i)
1156 {
1157 SV **e = av_fetch (av, i, 0);
1158
1159 if (e && *e)
1160 vecp[i] = SvPVbyte_nolen (*e);
1161 else
1162 vecp[i] = "";
1163 }
1164
1165 vecp[nelem] = 0;
1166
1167 return vecp;
1168 }
1169
1170 /*****************************************************************************/
1171
1172 XS(boot_IO__AIO) ecb_cold;
1173
1174 MODULE = IO::AIO PACKAGE = IO::AIO
1175
1176 PROTOTYPES: ENABLE
1177
1178 BOOT:
1179 {
1180 static const struct {
1181 const char *name;
1182 IV iv;
1183 } *civ, const_iv[] = {
1184 # define const_niv(name, value) { # name, (IV) value },
1185 # define const_iv(name) { # name, (IV) name },
1186 # define const_eio(name) { # name, (IV) EIO_ ## name },
1187
1188 /* you have to re-run ./gendef0 after adding/removing any constants here */
1189 /* the first block can be undef if missing */
1190 const_iv (ENOSYS)
1191 const_iv (EXDEV)
1192 const_iv (EBADR)
1193
1194 /* for lseek */
1195 const_iv (SEEK_DATA)
1196 const_iv (SEEK_HOLE)
1197
1198 const_niv (FADV_NORMAL , POSIX_FADV_NORMAL)
1199 const_niv (FADV_SEQUENTIAL, POSIX_FADV_SEQUENTIAL)
1200 const_niv (FADV_RANDOM , POSIX_FADV_RANDOM)
1201 const_niv (FADV_NOREUSE , POSIX_FADV_NOREUSE)
1202 const_niv (FADV_WILLNEED , POSIX_FADV_WILLNEED)
1203 const_niv (FADV_DONTNEED , POSIX_FADV_DONTNEED)
1204
1205 const_niv (MADV_NORMAL , POSIX_MADV_NORMAL)
1206 const_niv (MADV_SEQUENTIAL, POSIX_MADV_SEQUENTIAL)
1207 const_niv (MADV_RANDOM , POSIX_MADV_RANDOM)
1208 const_niv (MADV_WILLNEED , POSIX_MADV_WILLNEED)
1209 const_niv (MADV_DONTNEED , POSIX_MADV_DONTNEED)
1210
1211 /* linux extensions */
1212 const_iv (MADV_FREE)
1213 const_iv (MADV_REMOVE)
1214 const_iv (MADV_DONTFORK)
1215 const_iv (MADV_DOFORK)
1216 const_iv (MADV_MERGEABLE)
1217 const_iv (MADV_UNMERGEABLE)
1218 const_iv (MADV_HUGEPAGE)
1219 const_iv (MADV_NOHUGEPAGE)
1220 const_iv (MADV_DONTDUMP)
1221 const_iv (MADV_DODUMP)
1222 const_iv (MADV_WIPEONFORK)
1223 const_iv (MADV_KEEPONFORK)
1224 const_iv (MADV_COLD)
1225 const_iv (MADV_PAGEOUT)
1226 const_iv (MADV_POPULATE_READ)
1227 const_iv (MADV_POPULATE_WRITE)
1228 const_iv (MADV_DONTNEED_LOCKED)
1229 const_iv (MADV_HWPOISON)
1230 const_iv (MADV_SOFT_OFFLINE)
1231
1232 /* the second block will be 0 when missing */
1233 const_iv (O_ACCMODE)
1234
1235 const_iv (O_RDONLY)
1236 const_iv (O_WRONLY)
1237 const_iv (O_RDWR)
1238 const_iv (O_CREAT)
1239 const_iv (O_TRUNC)
1240 const_iv (O_EXCL)
1241 const_iv (O_APPEND)
1242
1243 const_iv (O_ASYNC)
1244 const_iv (O_DIRECT)
1245 const_iv (O_NOATIME)
1246
1247 const_iv (O_CLOEXEC)
1248 const_iv (O_NOCTTY)
1249 const_iv (O_NOFOLLOW)
1250 const_iv (O_NONBLOCK)
1251 const_iv (O_EXEC)
1252 const_iv (O_SEARCH)
1253 const_iv (O_DIRECTORY)
1254 const_iv (O_DSYNC)
1255 const_iv (O_RSYNC)
1256 const_iv (O_SYNC)
1257 const_iv (O_PATH)
1258 const_iv (O_TMPFILE)
1259 const_iv (O_TTY_INIT)
1260
1261 const_iv (S_IFIFO)
1262 const_iv (S_IFCHR)
1263 const_iv (S_IFBLK)
1264 const_iv (S_IFLNK)
1265 const_iv (S_IFREG)
1266 const_iv (S_IFDIR)
1267 const_iv (S_IFWHT)
1268 const_iv (S_IFSOCK)
1269 const_iv (S_IFMT)
1270
1271 const_iv (ST_RDONLY)
1272 const_iv (ST_NOSUID)
1273 const_iv (ST_NODEV)
1274 const_iv (ST_NOEXEC)
1275 const_iv (ST_SYNCHRONOUS)
1276 const_iv (ST_MANDLOCK)
1277 const_iv (ST_WRITE)
1278 const_iv (ST_APPEND)
1279 const_iv (ST_IMMUTABLE)
1280 const_iv (ST_NOATIME)
1281 const_iv (ST_NODIRATIME)
1282 const_iv (ST_RELATIME)
1283
1284 const_iv (PROT_NONE)
1285 const_iv (PROT_EXEC)
1286 const_iv (PROT_READ)
1287 const_iv (PROT_WRITE)
1288
1289 const_iv (MAP_PRIVATE)
1290 const_iv (MAP_SHARED)
1291 const_iv (MAP_FIXED)
1292 const_iv (MAP_ANONYMOUS)
1293
1294 /* linuxish */
1295 const_iv (MAP_LOCKED)
1296 const_iv (MAP_NORESERVE)
1297 const_iv (MAP_POPULATE)
1298 const_iv (MAP_NONBLOCK)
1299 const_iv (MAP_GROWSDOWN)
1300 const_iv (MAP_32BIT)
1301 const_iv (MAP_HUGETLB)
1302 const_iv (MAP_STACK)
1303 const_iv (MAP_FIXED_NOREPLACE)
1304 const_iv (MAP_SHARED_VALIDATE)
1305 const_iv (MAP_SYNC)
1306 const_iv (MAP_UNINITIALIZED)
1307 const_iv (MAP_DENYWRITE)
1308 const_iv (MAP_EXECUTABLE)
1309 const_iv (MAP_LOCKED)
1310 const_iv (MAP_DROPPABLE)
1311
1312 const_iv (MREMAP_MAYMOVE)
1313 const_iv (MREMAP_FIXED)
1314
1315 const_iv (MSG_CMSG_CLOEXEC)
1316 const_iv (SOCK_CLOEXEC)
1317
1318 const_iv (F_DUPFD_CLOEXEC)
1319
1320 const_iv (F_ADD_SEALS)
1321 const_iv (F_GET_SEALS)
1322 const_iv (F_SEAL_SEAL)
1323 const_iv (F_SEAL_SHRINK)
1324 const_iv (F_SEAL_GROW)
1325 const_iv (F_SEAL_WRITE)
1326
1327 const_iv (F_OFD_GETLK)
1328 const_iv (F_OFD_SETLK)
1329 const_iv (F_OFD_GETLKW)
1330
1331 const_iv (FIFREEZE)
1332 const_iv (FITHAW)
1333 const_iv (FITRIM)
1334 const_iv (FICLONE)
1335 const_iv (FICLONERANGE)
1336 const_iv (FIDEDUPERANGE)
1337
1338 const_iv (FS_IOC_GETFLAGS)
1339 const_iv (FS_IOC_SETFLAGS)
1340 const_iv (FS_IOC_GETVERSION)
1341 const_iv (FS_IOC_SETVERSION)
1342 #if HAVE_FIEMAP /* broken on musl for, like, foreever */
1343 const_iv (FS_IOC_FIEMAP)
1344 #endif
1345 const_iv (FS_IOC_FSGETXATTR)
1346 const_iv (FS_IOC_FSSETXATTR)
1347 const_iv (FS_IOC_SET_ENCRYPTION_POLICY)
1348 const_iv (FS_IOC_GET_ENCRYPTION_PWSALT)
1349 const_iv (FS_IOC_GET_ENCRYPTION_POLICY)
1350
1351 const_iv (FS_KEY_DESCRIPTOR_SIZE)
1352
1353 const_iv (FS_SECRM_FL)
1354 const_iv (FS_UNRM_FL)
1355 const_iv (FS_COMPR_FL)
1356 const_iv (FS_SYNC_FL)
1357 const_iv (FS_IMMUTABLE_FL)
1358 const_iv (FS_APPEND_FL)
1359 const_iv (FS_NODUMP_FL)
1360 const_iv (FS_NOATIME_FL)
1361 const_iv (FS_DIRTY_FL)
1362 const_iv (FS_COMPRBLK_FL)
1363 const_iv (FS_NOCOMP_FL)
1364 const_iv (FS_ENCRYPT_FL)
1365 const_iv (FS_BTREE_FL)
1366 const_iv (FS_INDEX_FL)
1367 const_iv (FS_JOURNAL_DATA_FL)
1368 const_iv (FS_NOTAIL_FL)
1369 const_iv (FS_DIRSYNC_FL)
1370 const_iv (FS_TOPDIR_FL)
1371 const_iv (FS_FL_USER_MODIFIABLE)
1372
1373 const_iv (FS_XFLAG_REALTIME)
1374 const_iv (FS_XFLAG_PREALLOC)
1375 const_iv (FS_XFLAG_IMMUTABLE)
1376 const_iv (FS_XFLAG_APPEND)
1377 const_iv (FS_XFLAG_SYNC)
1378 const_iv (FS_XFLAG_NOATIME)
1379 const_iv (FS_XFLAG_NODUMP)
1380 const_iv (FS_XFLAG_RTINHERIT)
1381 const_iv (FS_XFLAG_PROJINHERIT)
1382 const_iv (FS_XFLAG_NOSYMLINKS)
1383 const_iv (FS_XFLAG_EXTSIZE)
1384 const_iv (FS_XFLAG_EXTSZINHERIT)
1385 const_iv (FS_XFLAG_NODEFRAG)
1386 const_iv (FS_XFLAG_FILESTREAM)
1387 const_iv (FS_XFLAG_DAX)
1388 const_iv (FS_XFLAG_HASATTR)
1389
1390 const_iv (FIEMAP_FLAG_SYNC)
1391 const_iv (FIEMAP_FLAG_XATTR)
1392 const_iv (FIEMAP_FLAGS_COMPAT)
1393 const_iv (FIEMAP_EXTENT_LAST)
1394 const_iv (FIEMAP_EXTENT_UNKNOWN)
1395 const_iv (FIEMAP_EXTENT_DELALLOC)
1396 const_iv (FIEMAP_EXTENT_ENCODED)
1397 const_iv (FIEMAP_EXTENT_DATA_ENCRYPTED)
1398 const_iv (FIEMAP_EXTENT_NOT_ALIGNED)
1399 const_iv (FIEMAP_EXTENT_DATA_INLINE)
1400 const_iv (FIEMAP_EXTENT_DATA_TAIL)
1401 const_iv (FIEMAP_EXTENT_UNWRITTEN)
1402 const_iv (FIEMAP_EXTENT_MERGED)
1403 const_iv (FIEMAP_EXTENT_SHARED)
1404
1405 const_iv (SPLICE_F_MOVE)
1406 const_iv (SPLICE_F_NONBLOCK)
1407 const_iv (SPLICE_F_MORE)
1408 const_iv (SPLICE_F_GIFT)
1409
1410 const_iv (EFD_CLOEXEC)
1411 const_iv (EFD_NONBLOCK)
1412 const_iv (EFD_SEMAPHORE)
1413
1414 const_iv (MFD_CLOEXEC)
1415 const_iv (MFD_ALLOW_SEALING)
1416 const_iv (MFD_HUGETLB)
1417 const_iv (MFD_HUGETLB_2MB)
1418 const_iv (MFD_HUGETLB_1GB)
1419
1420 const_iv (CLOCK_REALTIME)
1421 const_iv (CLOCK_MONOTONIC)
1422 const_iv (CLOCK_BOOTTIME)
1423 const_iv (CLOCK_REALTIME_ALARM)
1424 const_iv (CLOCK_BOOTTIME_ALARM)
1425
1426 const_iv (TFD_NONBLOCK)
1427 const_iv (TFD_CLOEXEC)
1428
1429 const_iv (TFD_TIMER_ABSTIME)
1430 const_iv (TFD_TIMER_CANCEL_ON_SET)
1431
1432 const_iv (STATX_TYPE)
1433 const_iv (STATX_MODE)
1434 const_iv (STATX_NLINK)
1435 const_iv (STATX_UID)
1436 const_iv (STATX_GID)
1437 const_iv (STATX_ATIME)
1438 const_iv (STATX_MTIME)
1439 const_iv (STATX_CTIME)
1440 const_iv (STATX_INO)
1441 const_iv (STATX_SIZE)
1442 const_iv (STATX_BLOCKS)
1443 const_iv (STATX_BASIC_STATS)
1444 const_iv (STATX_ALL)
1445 const_iv (STATX_BTIME)
1446 const_iv (STATX_ATTR_COMPRESSED)
1447 const_iv (STATX_ATTR_IMMUTABLE)
1448 const_iv (STATX_ATTR_APPEND)
1449 const_iv (STATX_ATTR_NODUMP)
1450 const_iv (STATX_ATTR_ENCRYPTED)
1451 const_iv (STATX_ATTR_AUTOMOUNT)
1452
1453 const_iv (AT_FDCWD)
1454 const_iv (AT_SYMLINK_NOFOLLOW)
1455 const_iv (AT_EACCESS)
1456 const_iv (AT_REMOVEDIR)
1457 const_iv (AT_SYMLINK_FOLLOW)
1458 const_iv (AT_NO_AUTOMOUNT)
1459 const_iv (AT_EMPTY_PATH)
1460 const_iv (AT_STATX_SYNC_TYPE)
1461 const_iv (AT_STATX_AS_STAT)
1462 const_iv (AT_STATX_FORCE_SYNC)
1463 const_iv (AT_STATX_DONT_SYNC)
1464 const_iv (AT_RECURSIVE)
1465
1466 const_iv (OPEN_TREE_CLONE)
1467
1468 const_iv (FSOPEN_CLOEXEC)
1469
1470 const_iv (FSPICK_CLOEXEC)
1471 const_iv (FSPICK_SYMLINK_NOFOLLOW)
1472 const_iv (FSPICK_NO_AUTOMOUNT)
1473 const_iv (FSPICK_EMPTY_PATH)
1474
1475 const_iv (MOVE_MOUNT_F_SYMLINKS)
1476 const_iv (MOVE_MOUNT_F_AUTOMOUNTS)
1477 const_iv (MOVE_MOUNT_F_EMPTY_PATH)
1478 const_iv (MOVE_MOUNT_T_SYMLINKS)
1479 const_iv (MOVE_MOUNT_T_AUTOMOUNTS)
1480 const_iv (MOVE_MOUNT_T_EMPTY_PATH)
1481
1482 /* waitid */
1483 const_iv (P_PID)
1484 const_iv (P_PIDFD)
1485 const_iv (P_PGID)
1486 const_iv (P_ALL)
1487
1488 const_iv (FSCONFIG_SET_FLAG)
1489 const_iv (FSCONFIG_SET_STRING)
1490 const_iv (FSCONFIG_SET_BINARY)
1491 const_iv (FSCONFIG_SET_PATH)
1492 const_iv (FSCONFIG_SET_PATH_EMPTY)
1493 const_iv (FSCONFIG_SET_FD)
1494 const_iv (FSCONFIG_CMD_CREATE)
1495 const_iv (FSCONFIG_CMD_RECONFIGURE)
1496
1497 const_iv (MOUNT_ATTR_RDONLY)
1498 const_iv (MOUNT_ATTR_NOSUID)
1499 const_iv (MOUNT_ATTR_NODEV)
1500 const_iv (MOUNT_ATTR_NOEXEC)
1501 const_iv (MOUNT_ATTR__ATIME)
1502 const_iv (MOUNT_ATTR_RELATIME)
1503 const_iv (MOUNT_ATTR_NOATIME)
1504 const_iv (MOUNT_ATTR_STRICTATIME)
1505 const_iv (MOUNT_ATTR_NODIRATIME)
1506
1507 /* sys/mount.h */
1508 const_iv (MS_RDONLY)
1509 const_iv (MS_NOSUID)
1510 const_iv (MS_NODEV)
1511 const_iv (MS_NOEXEC)
1512 const_iv (MS_SYNCHRONOUS)
1513 const_iv (MS_REMOUNT)
1514 const_iv (MS_MANDLOCK)
1515 const_iv (MS_DIRSYNC)
1516 const_iv (MS_NOATIME)
1517 const_iv (MS_NODIRATIME)
1518 const_iv (MS_BIND)
1519 const_iv (MS_MOVE)
1520 const_iv (MS_REC)
1521 const_iv (MS_SILENT)
1522 const_iv (MS_POSIXACL)
1523 const_iv (MS_UNBINDABLE)
1524 const_iv (MS_PRIVATE)
1525 const_iv (MS_SLAVE)
1526 const_iv (MS_SHARED)
1527 const_iv (MS_RELATIME)
1528 const_iv (MS_KERNMOUNT)
1529 const_iv (MS_I_VERSION)
1530 const_iv (MS_STRICTATIME)
1531 const_iv (MS_LAZYTIME)
1532 const_iv (MS_ACTIVE)
1533 const_iv (MS_NOUSER)
1534 const_iv (MS_RMT_MASK)
1535 const_iv (MS_MGC_VAL)
1536 const_iv (MS_MGC_MSK)
1537
1538 const_iv (MNT_FORCE)
1539 const_iv (MNT_DETACH)
1540 const_iv (MNT_EXPIRE)
1541 const_iv (UMOUNT_NOFOLLOW)
1542
1543 const_iv (BLKROSET)
1544 const_iv (BLKROGET)
1545 const_iv (BLKRRPART)
1546 const_iv (BLKGETSIZE)
1547 const_iv (BLKFLSBUF)
1548 const_iv (BLKRASET)
1549 const_iv (BLKRAGET)
1550 const_iv (BLKFRASET)
1551 const_iv (BLKFRAGET)
1552 const_iv (BLKSECTSET)
1553 const_iv (BLKSECTGET)
1554 const_iv (BLKSSZGET)
1555 const_iv (BLKBSZGET)
1556 const_iv (BLKBSZSET)
1557 const_iv (BLKGETSIZE64)
1558
1559 /* these are libeio constants, and are independent of gendef0 */
1560 const_eio (SEEK_SET)
1561 const_eio (SEEK_CUR)
1562 const_eio (SEEK_END)
1563
1564 const_eio (MCL_FUTURE)
1565 const_eio (MCL_CURRENT)
1566 const_eio (MCL_ONFAULT)
1567
1568 const_eio (MS_ASYNC)
1569 const_eio (MS_INVALIDATE)
1570 const_eio (MS_SYNC)
1571
1572 const_eio (MT_MODIFY)
1573
1574 const_eio (SYNC_FILE_RANGE_WAIT_BEFORE)
1575 const_eio (SYNC_FILE_RANGE_WRITE)
1576 const_eio (SYNC_FILE_RANGE_WAIT_AFTER)
1577
1578 const_eio (FALLOC_FL_KEEP_SIZE)
1579 const_eio (FALLOC_FL_PUNCH_HOLE)
1580 const_eio (FALLOC_FL_COLLAPSE_RANGE)
1581 const_eio (FALLOC_FL_ZERO_RANGE)
1582 const_eio (FALLOC_FL_INSERT_RANGE)
1583 const_eio (FALLOC_FL_UNSHARE_RANGE)
1584
1585 const_eio (RENAME_NOREPLACE)
1586 const_eio (RENAME_EXCHANGE)
1587 const_eio (RENAME_WHITEOUT)
1588
1589 const_eio (READDIR_DENTS)
1590 const_eio (READDIR_DIRS_FIRST)
1591 const_eio (READDIR_STAT_ORDER)
1592 const_eio (READDIR_FOUND_UNKNOWN)
1593
1594 const_eio (DT_UNKNOWN)
1595 const_eio (DT_FIFO)
1596 const_eio (DT_CHR)
1597 const_eio (DT_DIR)
1598 const_eio (DT_BLK)
1599 const_eio (DT_REG)
1600 const_eio (DT_LNK)
1601 const_eio (DT_SOCK)
1602 const_eio (DT_WHT)
1603 };
1604
1605 aio_stash = gv_stashpv ("IO::AIO" , 1);
1606 aio_req_stash = gv_stashpv ("IO::AIO::REQ", 1);
1607 aio_grp_stash = gv_stashpv ("IO::AIO::GRP", 1);
1608 aio_wd_stash = gv_stashpv ("IO::AIO::WD" , 1);
1609
1610 for (civ = const_iv + sizeof (const_iv) / sizeof (const_iv [0]); civ > const_iv; civ--)
1611 newCONSTSUB (aio_stash, (char *)civ[-1].name, newSViv (civ[-1].iv));
1612
1613 newCONSTSUB (aio_stash, "PAGESIZE", newSViv (PAGESIZE));
1614
1615 /* allocate dummy pipe fd for aio_close */
1616 {
1617 int pipefd [2];
1618
1619 if (
1620 #ifdef _WIN32
1621 _pipe (pipefd, 1, _O_BINARY) < 0
1622 #else
1623 pipe (pipefd) < 0
1624 || fcntl (pipefd [0], F_SETFD, FD_CLOEXEC) < 0
1625 #endif
1626 || close (pipefd [1]) < 0
1627 )
1628 croak ("IO::AIO: unable to create dummy pipe for aio_close");
1629
1630 close_fd = pipefd [0];
1631 }
1632
1633 reinit ();
1634 }
1635
1636 void
1637 reinit ()
1638 PROTOTYPE:
1639
1640 void
1641 max_poll_reqs (unsigned int nreqs)
1642 PROTOTYPE: $
1643 CODE:
1644 eio_set_max_poll_reqs (nreqs);
1645
1646 void
1647 max_poll_time (double nseconds)
1648 PROTOTYPE: $
1649 CODE:
1650 eio_set_max_poll_time (nseconds);
1651
1652 void
1653 min_parallel (unsigned int nthreads)
1654 PROTOTYPE: $
1655 CODE:
1656 eio_set_min_parallel (nthreads);
1657
1658 void
1659 max_parallel (unsigned int nthreads)
1660 PROTOTYPE: $
1661 CODE:
1662 eio_set_max_parallel (nthreads);
1663
1664 void
1665 max_idle (unsigned int nthreads)
1666 PROTOTYPE: $
1667 CODE:
1668 eio_set_max_idle (nthreads);
1669
1670 void
1671 idle_timeout (unsigned int seconds)
1672 PROTOTYPE: $
1673 CODE:
1674 eio_set_idle_timeout (seconds);
1675
1676 void
1677 max_outstanding (unsigned int maxreqs)
1678 PROTOTYPE: $
1679 CODE:
1680 max_outstanding = maxreqs;
1681
1682 void
1683 aio_wd (SV8 *pathname, SV *callback = &PL_sv_undef)
1684 PPCODE:
1685 {
1686 dREQ;
1687
1688 req->type = EIO_WD_OPEN;
1689 req_set_path1 (req, pathname);
1690
1691 REQ_SEND;
1692 }
1693
1694 void
1695 aio_open (SV8 *pathname, int flags, int mode, SV *callback = &PL_sv_undef)
1696 PPCODE:
1697 {
1698 dREQ;
1699
1700 req->type = EIO_OPEN;
1701 req_set_path1 (req, pathname);
1702 req->int1 = flags;
1703 req->int2 = mode;
1704
1705 REQ_SEND;
1706 }
1707
1708 void
1709 aio_fsync (SV *fh, SV *callback = &PL_sv_undef)
1710 ALIAS:
1711 aio_fsync = EIO_FSYNC
1712 aio_fdatasync = EIO_FDATASYNC
1713 aio_syncfs = EIO_SYNCFS
1714 PPCODE:
1715 {
1716 int fd = s_fileno_croak (fh, 0);
1717 dREQ;
1718
1719 req->type = ix;
1720 req->sv1 = newSVsv (fh);
1721 req->int1 = fd;
1722
1723 REQ_SEND;
1724 }
1725
1726 void
1727 aio_sync_file_range (SV *fh, off_t offset, size_t nbytes, UV flags, SV *callback = &PL_sv_undef)
1728 PPCODE:
1729 {
1730 int fd = s_fileno_croak (fh, 0);
1731 dREQ;
1732
1733 req->type = EIO_SYNC_FILE_RANGE;
1734 req->sv1 = newSVsv (fh);
1735 req->int1 = fd;
1736 req->offs = offset;
1737 req->size = nbytes;
1738 req->int2 = flags;
1739
1740 REQ_SEND;
1741 }
1742
1743 void
1744 aio_allocate (SV *fh, int mode, off_t offset, size_t len, SV *callback = &PL_sv_undef)
1745 PPCODE:
1746 {
1747 int fd = s_fileno_croak (fh, 0);
1748 dREQ;
1749
1750 req->type = EIO_FALLOCATE;
1751 req->sv1 = newSVsv (fh);
1752 req->int1 = fd;
1753 req->int2 = mode;
1754 req->offs = offset;
1755 req->size = len;
1756
1757 REQ_SEND;
1758 }
1759
1760 void
1761 aio_close (SV *fh, SV *callback = &PL_sv_undef)
1762 PPCODE:
1763 {
1764 int fd = s_fileno_croak (fh, 0);
1765 dREQ;
1766 #if 0
1767 /* partially duplicate logic in s_fileno */
1768 SvGETMAGIC (fh);
1769
1770 if (SvROK (fh))
1771 {
1772 fh = SvRV (fh);
1773 SvGETMAGIC (fh);
1774 }
1775
1776 if (SvTYPE (fh) == SVt_PVGV)
1777 {
1778 /* perl filehandle */
1779 PerlIOUnix_refcnt_inc (fd);
1780 do_close ((GV *)fh, 1);
1781
1782 req->type = EIO_CLOSE;
1783 req->int1 = fd;
1784 /*req->sv2 = newSVsv (fh);*/ /* since we stole the fd, no need to keep the fh */
1785 }
1786 else
1787 #endif
1788 {
1789 /* fd number */
1790 req->type = EIO_DUP2;
1791 req->int1 = close_fd;
1792 req->sv2 = newSVsv (fh);
1793 req->int2 = fd;
1794 }
1795
1796 REQ_SEND;
1797 }
1798
1799 void
1800 aio_seek (SV *fh, SV *offset, int whence, SV *callback = &PL_sv_undef)
1801 PPCODE:
1802 {
1803 int fd = s_fileno_croak (fh, 0);
1804 dREQ;
1805
1806 req->type = EIO_SEEK;
1807 req->sv1 = newSVsv (fh);
1808 req->int1 = fd;
1809 req->offs = SvVAL64 (offset);
1810 req->int2 = whence;
1811
1812 REQ_SEND;
1813 }
1814
1815 void
1816 aio_read (SV *fh, SV *offset, SV *length, SV8 *data, IV dataoffset, SV *callback = &PL_sv_undef)
1817 ALIAS:
1818 aio_read = EIO_READ
1819 aio_write = EIO_WRITE
1820 PPCODE:
1821 {
1822 STRLEN svlen;
1823 int fd = s_fileno_croak (fh, ix == EIO_WRITE);
1824 char *svptr = SvPVbyte (data, svlen);
1825 UV len = SvUV (length);
1826
1827 if (dataoffset < 0)
1828 dataoffset += svlen;
1829
1830 if (dataoffset < 0 || dataoffset > svlen)
1831 croak ("dataoffset outside of data scalar");
1832
1833 if (ix == EIO_WRITE)
1834 {
1835 /* write: check length and adjust. */
1836 if (!SvOK (length) || len + dataoffset > svlen)
1837 len = svlen - dataoffset;
1838 }
1839 else
1840 {
1841 /* read: check type and grow scalar as necessary */
1842 if (!SvPOK (data) || SvLEN (data) >= SvCUR (data))
1843 svptr = sv_grow (data, len + dataoffset + 1);
1844 else if (SvCUR (data) < len + dataoffset)
1845 croak ("length + dataoffset outside of scalar, and cannot grow");
1846 }
1847
1848 {
1849 dREQ;
1850
1851 req->type = ix;
1852 req->sv1 = newSVsv (fh);
1853 req->int1 = fd;
1854 req->offs = SvOK (offset) ? SvVAL64 (offset) : -1;
1855 req->size = len;
1856 req->sv2 = SvREFCNT_inc (data);
1857 req->ptr2 = (char *)svptr + dataoffset;
1858 req->stroffset = dataoffset;
1859
1860 if (!SvREADONLY (data))
1861 {
1862 SvREADONLY_on (data);
1863 req->flags |= FLAG_SV2_RO_OFF;
1864 }
1865
1866 REQ_SEND;
1867 }
1868 }
1869
1870 void
1871 aio_ioctl (SV *fh, unsigned long request, SV8 *arg, SV *callback = &PL_sv_undef)
1872 ALIAS:
1873 aio_ioctl = EIO_IOCTL
1874 aio_fcntl = EIO_FCNTL
1875 PPCODE:
1876 {
1877 int fd = s_fileno_croak (fh, 0);
1878 char *svptr;
1879
1880 if (SvPOK (arg) || !SvNIOK (arg))
1881 {
1882 STRLEN svlen;
1883 /* perl uses IOCPARM_LEN for fcntl, so we do, too */
1884 #ifdef IOCPARM_LEN
1885 STRLEN need = IOCPARM_LEN (request);
1886 #else
1887 STRLEN need = 256;
1888 #endif
1889
1890 if (svlen < need)
1891 svptr = SvGROW (arg, need);
1892 }
1893 else
1894 svptr = (char *)SvIV (arg);
1895
1896 {
1897 dREQ;
1898
1899 req->type = ix;
1900 req->sv1 = newSVsv (fh);
1901 req->int1 = fd;
1902 req->int2 = (long)request;
1903 req->sv2 = SvREFCNT_inc (arg);
1904 req->ptr2 = svptr;
1905
1906 REQ_SEND;
1907 }
1908 }
1909
1910 void
1911 aio_readlink (SV8 *pathname, SV *callback = &PL_sv_undef)
1912 ALIAS:
1913 aio_readlink = EIO_READLINK
1914 aio_realpath = EIO_REALPATH
1915 PPCODE:
1916 {
1917 dREQ;
1918
1919 req->type = ix;
1920 req_set_path1 (req, pathname);
1921
1922 REQ_SEND;
1923 }
1924
1925 void
1926 aio_sendfile (SV *out_fh, SV *in_fh, off_t in_offset, size_t length, SV *callback = &PL_sv_undef)
1927 PPCODE:
1928 {
1929 int ifd = s_fileno_croak (in_fh , 0);
1930 int ofd = s_fileno_croak (out_fh, 1);
1931 dREQ;
1932
1933 req->type = EIO_SENDFILE;
1934 req->sv1 = newSVsv (out_fh);
1935 req->int1 = ofd;
1936 req->sv2 = newSVsv (in_fh);
1937 req->int2 = ifd;
1938 req->offs = in_offset;
1939 req->size = length;
1940
1941 REQ_SEND;
1942 }
1943
1944 void
1945 aio_readahead (SV *fh, off_t offset, size_t length, SV *callback = &PL_sv_undef)
1946 PPCODE:
1947 {
1948 int fd = s_fileno_croak (fh, 0);
1949 dREQ;
1950
1951 req->type = EIO_READAHEAD;
1952 req->sv1 = newSVsv (fh);
1953 req->int1 = fd;
1954 req->offs = offset;
1955 req->size = length;
1956
1957 REQ_SEND;
1958 }
1959
1960 void
1961 aio_stat (SV8 *fh_or_path, SV *callback = &PL_sv_undef)
1962 ALIAS:
1963 aio_stat = EIO_STAT
1964 aio_lstat = EIO_LSTAT
1965 aio_statvfs = EIO_STATVFS
1966 PPCODE:
1967 {
1968 dREQ;
1969
1970 req_set_fh_or_path (req, ix, ix == EIO_STATVFS ? EIO_FSTATVFS : EIO_FSTAT, fh_or_path);
1971
1972 REQ_SEND;
1973 }
1974
1975 void
1976 st_xtime ()
1977 ALIAS:
1978 PPCODE:
1979 EXTEND (SP, 4);
1980
1981 void
1982 st_xtimensec ()
1983 ALIAS:
1984 st_atime = 0x0001
1985 st_mtime = 0x0002
1986 st_ctime = 0x0004
1987 st_btime = 0x0008
1988 st_xtime = 0x000f
1989
1990 st_atime2 = 0x0010
1991 st_mtime2 = 0x0020
1992 st_ctime2 = 0x0040
1993 st_btime2 = 0x0080
1994 st_xtime2 = 0x00f0
1995
1996 st_atimesec = 0x0100
1997 st_mtimesec = 0x0200
1998 st_ctimesec = 0x0400
1999 st_btimesec = 0x0800
2000 st_xtimesec = 0x0f00
2001 st_atimensec = 0x1000
2002 st_mtimensec = 0x2000
2003 st_ctimensec = 0x4000
2004 st_btimensec = 0x8000
2005 st_xtimensec = 0xf000
2006
2007 st_gen = 0x10000
2008 PPCODE:
2009 {
2010 EXTEND (SP, 4); /* one call generates at most 4 values */
2011
2012 if (ix & 0x0001) PUSHs (newSVnv (PL_statcache.st_atime + 1e-9 * ATIMENSEC));
2013 if (ix & 0x0002) PUSHs (newSVnv (PL_statcache.st_mtime + 1e-9 * MTIMENSEC));
2014 if (ix & 0x0004) PUSHs (newSVnv (PL_statcache.st_ctime + 1e-9 * CTIMENSEC));
2015 if (ix & 0x0008) PUSHs (newSVnv (BTIMESEC + 1e-9 * BTIMENSEC));
2016
2017 if (ix & 0x0010) PUSHs (newSVts (PL_statcache.st_atime, ATIMENSEC));
2018 if (ix & 0x0020) PUSHs (newSVts (PL_statcache.st_mtime, MTIMENSEC));
2019 if (ix & 0x0040) PUSHs (newSVts (PL_statcache.st_ctime, CTIMENSEC));
2020 if (ix & 0x0080) PUSHs (newSVts (BTIMESEC , BTIMENSEC));
2021
2022 if (ix & 0x0100) PUSHs (newSVval64 (PL_statcache.st_atime));
2023 if (ix & 0x0200) PUSHs (newSVval64 (PL_statcache.st_mtime));
2024 if (ix & 0x0400) PUSHs (newSVval64 (PL_statcache.st_ctime));
2025 if (ix & 0x0800) PUSHs (newSVval64 (BTIMESEC ));
2026
2027 if (ix & 0x1000) PUSHs (newSViv (ATIMENSEC));
2028 if (ix & 0x2000) PUSHs (newSViv (MTIMENSEC));
2029 if (ix & 0x4000) PUSHs (newSViv (CTIMENSEC));
2030 if (ix & 0x8000) PUSHs (newSViv (BTIMENSEC));
2031
2032 if (ix & 0x10000) PUSHs (newSVuv (ST_GEN));
2033 }
2034
2035 UV
2036 major (UV dev)
2037 ALIAS:
2038 minor = 1
2039 CODE:
2040 RETVAL = ix ? minor (dev) : major (dev);
2041 OUTPUT:
2042 RETVAL
2043
2044 UV
2045 makedev (UV maj, UV min)
2046 CODE:
2047 RETVAL = makedev (maj, min);
2048 OUTPUT:
2049 RETVAL
2050
2051 void
2052 aio_utime (SV8 *fh_or_path, SV *atime, SV *mtime, SV *callback = &PL_sv_undef)
2053 PPCODE:
2054 {
2055 dREQ;
2056
2057 sv_to_tstamp (atime, &req->nv1);
2058 sv_to_tstamp (mtime, &req->nv2);
2059 req_set_fh_or_path (req, EIO_UTIME, EIO_FUTIME, fh_or_path);
2060
2061 REQ_SEND;
2062 }
2063
2064 void
2065 aio_truncate (SV8 *fh_or_path, SV *offset, SV *callback = &PL_sv_undef)
2066 PPCODE:
2067 {
2068 dREQ;
2069
2070 req->offs = SvOK (offset) ? SvVAL64 (offset) : -1;
2071 req_set_fh_or_path (req, EIO_TRUNCATE, EIO_FTRUNCATE, fh_or_path);
2072
2073 REQ_SEND;
2074 }
2075
2076 void
2077 aio_chmod (SV8 *fh_or_path, int mode, SV *callback = &PL_sv_undef)
2078 PPCODE:
2079 {
2080 dREQ;
2081
2082 req->int2 = mode;
2083 req_set_fh_or_path (req, EIO_CHMOD, EIO_FCHMOD, fh_or_path);
2084
2085 REQ_SEND;
2086 }
2087
2088 void
2089 aio_chown (SV8 *fh_or_path, SV *uid, SV *gid, SV *callback = &PL_sv_undef)
2090 PPCODE:
2091 {
2092 dREQ;
2093
2094 req->int2 = SvOK (uid) ? SvIV (uid) : -1;
2095 req->int3 = SvOK (gid) ? SvIV (gid) : -1;
2096 req_set_fh_or_path (req, EIO_CHOWN, EIO_FCHOWN, fh_or_path);
2097
2098 REQ_SEND;
2099 }
2100
2101 void
2102 aio_readdirx (SV8 *pathname, IV flags, SV *callback = &PL_sv_undef)
2103 PPCODE:
2104 {
2105 dREQ;
2106
2107 req->type = EIO_READDIR;
2108 req->int1 = flags | EIO_READDIR_DENTS | EIO_READDIR_CUSTOM1;
2109
2110 if (flags & EIO_READDIR_DENTS)
2111 req->int1 |= EIO_READDIR_CUSTOM2;
2112
2113 req_set_path1 (req, pathname);
2114
2115 REQ_SEND;
2116 }
2117
2118 void
2119 aio_mkdir (SV8 *pathname, int mode, SV *callback = &PL_sv_undef)
2120 PPCODE:
2121 {
2122 dREQ;
2123
2124 req->type = EIO_MKDIR;
2125 req->int2 = mode;
2126 req_set_path1 (req, pathname);
2127
2128 REQ_SEND;
2129 }
2130
2131 void
2132 aio_unlink (SV8 *pathname, SV *callback = &PL_sv_undef)
2133 ALIAS:
2134 aio_unlink = EIO_UNLINK
2135 aio_rmdir = EIO_RMDIR
2136 aio_readdir = EIO_READDIR
2137 PPCODE:
2138 {
2139 dREQ;
2140
2141 req->type = ix;
2142 req_set_path1 (req, pathname);
2143
2144 REQ_SEND;
2145 }
2146
2147 void
2148 aio_link (SV8 *oldpath, SV8 *newpath, SV *callback = &PL_sv_undef)
2149 ALIAS:
2150 aio_link = EIO_LINK
2151 aio_symlink = EIO_SYMLINK
2152 aio_rename = EIO_RENAME
2153 PPCODE:
2154 {
2155 eio_wd wd2 = 0;
2156 dREQ;
2157
2158 req->type = ix;
2159 req_set_path1 (req, oldpath);
2160 req_set_path (newpath, &req->sv2, &req->sv4, &wd2, &req->ptr2);
2161 req->int3 = (long)wd2;
2162
2163 REQ_SEND;
2164 }
2165
2166 void
2167 aio_rename2 (SV8 *oldpath, SV8 *newpath, int flags = 0, SV *callback = &PL_sv_undef)
2168 PPCODE:
2169 {
2170 eio_wd wd2 = 0;
2171 dREQ;
2172
2173 req->type = EIO_RENAME;
2174 req_set_path1 (req, oldpath);
2175 req_set_path (newpath, &req->sv2, &req->sv4, &wd2, &req->ptr2);
2176 req->int2 = flags;
2177 req->int3 = (long)wd2;
2178
2179 REQ_SEND;
2180 }
2181
2182 void
2183 aio_mknod (SV8 *pathname, int mode, UV dev, SV *callback = &PL_sv_undef)
2184 PPCODE:
2185 {
2186 dREQ;
2187
2188 req->type = EIO_MKNOD;
2189 req->int2 = (mode_t)mode;
2190 req->offs = dev;
2191 req_set_path1 (req, pathname);
2192
2193 REQ_SEND;
2194 }
2195
2196 void
2197 aio_mtouch (SV8 *data, IV offset = 0, SV *length = &PL_sv_undef, int flags = -1, SV *callback = &PL_sv_undef)
2198 ALIAS:
2199 aio_mtouch = EIO_MTOUCH
2200 aio_msync = EIO_MSYNC
2201 PPCODE:
2202 {
2203 STRLEN svlen;
2204 char *svptr = SvPVbyte (data, svlen);
2205 UV len = SvUV (length);
2206
2207 if (flags < 0)
2208 flags = ix == EIO_MSYNC ? EIO_MS_SYNC : 0;
2209
2210 if (offset < 0)
2211 offset += svlen;
2212
2213 if (offset < 0 || offset > svlen)
2214 croak ("offset outside of scalar");
2215
2216 if (!SvOK (length) || len + offset > svlen)
2217 len = svlen - offset;
2218
2219 {
2220 dREQ;
2221
2222 req->type = ix;
2223 req->sv2 = SvREFCNT_inc (data);
2224 req->ptr2 = (char *)svptr + offset;
2225 req->size = len;
2226 req->int1 = flags;
2227
2228 REQ_SEND;
2229 }
2230 }
2231
2232 void
2233 aio_mlock (SV8 *data, IV offset = 0, SV *length = &PL_sv_undef, SV *callback = &PL_sv_undef)
2234 PPCODE:
2235 {
2236 STRLEN svlen;
2237 char *svptr = SvPVbyte (data, svlen);
2238 UV len = SvUV (length);
2239
2240 if (offset < 0)
2241 offset += svlen;
2242
2243 if (offset < 0 || offset > svlen)
2244 croak ("offset outside of scalar");
2245
2246 if (!SvOK (length) || len + offset > svlen)
2247 len = svlen - offset;
2248
2249 {
2250 dREQ;
2251
2252 req->type = EIO_MLOCK;
2253 req->sv2 = SvREFCNT_inc (data);
2254 req->ptr2 = (char *)svptr + offset;
2255 req->size = len;
2256
2257 REQ_SEND;
2258 }
2259 }
2260
2261 void
2262 aio_mlockall (IV flags, SV *callback = &PL_sv_undef)
2263 PPCODE:
2264 {
2265 dREQ;
2266
2267 req->type = EIO_MLOCKALL;
2268 req->int1 = flags;
2269
2270 REQ_SEND;
2271 }
2272
2273 void
2274 aio_fiemap (SV *fh, off_t start, SV *length, U32 flags, SV *count, SV *callback = &PL_sv_undef)
2275 PPCODE:
2276 {
2277 int fd = s_fileno_croak (fh, 0);
2278 dREQ;
2279
2280 req->type = EIO_CUSTOM;
2281 req->sv1 = newSVsv (fh);
2282 req->int1 = fd;
2283
2284 req->feed = fiemap;
2285 #if HAVE_FIEMAP
2286 /* keep our fingers crossed that the next two types are 64 bit */
2287 req->offs = start;
2288 req->size = SvOK (length) ? SvVAL64 (length) : ~0ULL;
2289 req->int2 = flags;
2290 req->int3 = SvOK (count) ? SvIV (count) : -1;
2291 #endif
2292
2293 REQ_SEND;
2294 }
2295
2296 void
2297 aio_slurp (SV *pathname, off_t offset, UV length, SV8 *data, SV *callback = &PL_sv_undef)
2298 PPCODE:
2299 {
2300 char *svptr = 0;
2301
2302 sv_clear_foreign (data);
2303
2304 if (length) /* known length, directly read into scalar */
2305 {
2306 if (!SvPOK (data) || SvLEN (data) >= SvCUR (data))
2307 svptr = sv_grow (data, length + 1);
2308 else if (SvCUR (data) < length)
2309 croak ("length outside of scalar, and cannot grow");
2310 else
2311 svptr = SvPVbyte_nolen (data);
2312 }
2313
2314 {
2315 dREQ;
2316
2317 req->type = EIO_SLURP;
2318 req_set_path1 (req, pathname);
2319 req->offs = offset;
2320 req->size = length;
2321 req->sv2 = SvREFCNT_inc (data);
2322 req->ptr2 = svptr;
2323
2324 if (!SvREADONLY (data))
2325 {
2326 SvREADONLY_on (data);
2327 req->flags |= FLAG_SV2_RO_OFF;
2328 }
2329
2330 REQ_SEND;
2331 }
2332 }
2333
2334 void
2335 aio_busy (SV *delay, SV *callback = &PL_sv_undef)
2336 PPCODE:
2337 {
2338 dREQ;
2339
2340 req->type = EIO_BUSY;
2341 sv_to_tstamp (delay, &req->nv1);
2342
2343 REQ_SEND;
2344 }
2345
2346 void
2347 aio_group (SV *callback = &PL_sv_undef)
2348 PPCODE:
2349 {
2350 dREQ;
2351
2352 req->type = EIO_GROUP;
2353
2354 PUTBACK;
2355 req_submit (req);
2356 SPAGAIN;
2357
2358 XPUSHs (req_sv (req, aio_grp_stash));
2359 }
2360
2361 void
2362 aio_nop (SV *callback = &PL_sv_undef)
2363 ALIAS:
2364 aio_nop = EIO_NOP
2365 aio_sync = EIO_SYNC
2366 PPCODE:
2367 {
2368 dREQ;
2369
2370 req->type = ix;
2371
2372 REQ_SEND;
2373 }
2374
2375 int
2376 aioreq_pri (int pri = NO_INIT)
2377 CODE:
2378 RETVAL = next_pri;
2379 if (items > 0)
2380 {
2381 if (pri < EIO_PRI_MIN) pri = EIO_PRI_MIN;
2382 if (pri > EIO_PRI_MAX) pri = EIO_PRI_MAX;
2383 next_pri = pri;
2384 }
2385 OUTPUT:
2386 RETVAL
2387
2388 void
2389 aioreq_nice (int nice = 0)
2390 CODE:
2391 nice = next_pri - nice;
2392 if (nice < EIO_PRI_MIN) nice = EIO_PRI_MIN;
2393 if (nice > EIO_PRI_MAX) nice = EIO_PRI_MAX;
2394 next_pri = nice;
2395
2396 void
2397 flush ()
2398 CODE:
2399 while (eio_nreqs ())
2400 {
2401 poll_wait ();
2402 poll_cb ();
2403 }
2404
2405 int
2406 poll ()
2407 CODE:
2408 poll_wait ();
2409 RETVAL = poll_cb ();
2410 OUTPUT:
2411 RETVAL
2412
2413 int
2414 poll_fileno ()
2415 CODE:
2416 RETVAL = s_epipe_fd (&respipe);
2417 OUTPUT:
2418 RETVAL
2419
2420 int
2421 poll_cb (...)
2422 PROTOTYPE:
2423 CODE:
2424 RETVAL = poll_cb ();
2425 OUTPUT:
2426 RETVAL
2427
2428 void
2429 poll_wait ()
2430 CODE:
2431 poll_wait ();
2432
2433 int
2434 nreqs ()
2435 CODE:
2436 RETVAL = eio_nreqs ();
2437 OUTPUT:
2438 RETVAL
2439
2440 int
2441 nready ()
2442 CODE:
2443 RETVAL = eio_nready ();
2444 OUTPUT:
2445 RETVAL
2446
2447 int
2448 npending ()
2449 CODE:
2450 RETVAL = eio_npending ();
2451 OUTPUT:
2452 RETVAL
2453
2454 int
2455 nthreads ()
2456 CODE:
2457 RETVAL = eio_nthreads ();
2458 OUTPUT:
2459 RETVAL
2460
2461 int
2462 fadvise (aio_rfd fh, off_t offset, off_t length, IV advice)
2463 CODE:
2464 RETVAL = posix_fadvise (fh, offset, length, advice);
2465 OUTPUT:
2466 RETVAL
2467
2468 IV
2469 sendfile (aio_wfd ofh, aio_rfd ifh, off_t offset, size_t count)
2470 CODE:
2471 RETVAL = eio_sendfile_sync (ofh, ifh, offset, count);
2472 OUTPUT:
2473 RETVAL
2474
2475 void
2476 mmap (SV *scalar, STRLEN length, int prot, int flags, SV *fh = &PL_sv_undef, off_t offset = 0)
2477 PPCODE:
2478 sv_clear_foreign (scalar);
2479 {
2480 int fd = SvOK (fh) ? s_fileno_croak (fh, flags & PROT_WRITE) : -1;
2481 void *addr = (void *)mmap (0, length, prot, flags, fd, offset);
2482 if (addr == (void *)-1)
2483 XSRETURN_NO;
2484
2485 sv_set_foreign (scalar, &mmap_vtbl, addr, length);
2486
2487 if (!(prot & PROT_WRITE))
2488 SvREADONLY_on (scalar);
2489
2490 XSRETURN_YES;
2491 }
2492
2493 void
2494 munmap (SV *scalar)
2495 CODE:
2496 sv_clear_foreign (scalar);
2497
2498 SV *
2499 mremap (SV *scalar, STRLEN new_length, int flags = MREMAP_MAYMOVE, IV new_address = 0)
2500 CODE:
2501 {
2502 MAGIC *mg = mg_findext (scalar, FOREIGN_MAGIC, &mmap_vtbl);
2503 void *new;
2504
2505 if (!mg || SvPVX (scalar) != mg->mg_ptr)
2506 croak ("IO::AIO::mremap: scalar not mapped by IO::AIO::mmap or improperly modified");
2507
2508 new = mremap (mg->mg_ptr, (size_t)mg->mg_obj, new_length, flags, (void *)new_address);
2509
2510 RETVAL = &PL_sv_no;
2511
2512 if (new != (void *)-1)
2513 {
2514 RETVAL = new == (void *)mg->mg_ptr
2515 ? newSVpvn ("0 but true", 10)
2516 : &PL_sv_yes;
2517
2518 mg->mg_ptr = (char *)new;
2519 mg->mg_obj = (SV *)new_length;
2520
2521 SvPVX (scalar) = mg->mg_ptr;
2522 SvCUR_set (scalar, new_length);
2523 }
2524 }
2525 OUTPUT:
2526 RETVAL
2527
2528 int
2529 madvise (SV *scalar, IV offset = 0, SV *length = &PL_sv_undef, IV advice_or_prot)
2530 ALIAS:
2531 mprotect = 1
2532 CODE:
2533 {
2534 STRLEN svlen;
2535 void *addr = SvPVbyte (scalar, svlen);
2536 STRLEN len = SvUV (length);
2537
2538 if (offset < 0)
2539 offset += svlen;
2540
2541 if (offset < 0 || offset > svlen)
2542 croak ("offset outside of scalar");
2543
2544 if (!SvOK (length) || len + offset > svlen)
2545 len = svlen - offset;
2546
2547 addr = (void *)(((intptr_t)addr) + offset);
2548 eio_page_align (&addr, &len);
2549
2550 switch (ix)
2551 {
2552 case 0: RETVAL = posix_madvise (addr, len, advice_or_prot); break;
2553 case 1: RETVAL = mprotect (addr, len, advice_or_prot); break;
2554 }
2555 }
2556 OUTPUT:
2557 RETVAL
2558
2559 int
2560 munlock (SV *scalar, IV offset = 0, SV *length = &PL_sv_undef)
2561 CODE:
2562 {
2563 STRLEN svlen;
2564 void *addr = SvPVbyte (scalar, svlen);
2565 size_t len = SvUV (length);
2566
2567 if (offset < 0)
2568 offset += svlen;
2569
2570 if (offset < 0 || offset > svlen)
2571 croak ("offset outside of scalar");
2572
2573 if (!SvOK (length) || len + offset > svlen)
2574 len = svlen - offset;
2575
2576 addr = (void *)(((intptr_t)addr) + offset);
2577 eio_page_align (&addr, &len);
2578 #if _POSIX_MEMLOCK_RANGE
2579 RETVAL = munlock (addr, len);
2580 #else
2581 RETVAL = EIO_ENOSYS ();
2582 #endif
2583 }
2584 OUTPUT:
2585 RETVAL
2586
2587 int
2588 mlockall (int flags)
2589 PROTOTYPE: $;
2590 CODE:
2591 RETVAL = eio_mlockall_sync (flags);
2592 OUTPUT:
2593 RETVAL
2594
2595 int
2596 munlockall ()
2597 CODE:
2598 #if _POSIX_MEMLOCK
2599 munlockall ();
2600 #else
2601 RETVAL = EIO_ENOSYS ();
2602 #endif
2603 OUTPUT:
2604 RETVAL
2605
2606 int
2607 statx (SV8 *pathname, int flags, UV mask)
2608 CODE:
2609 {
2610 /* undocumented, and might go away, and anyway, should use eio_statx */
2611 SV *wdsv = 0;
2612 SV *pathsv = 0;
2613 eio_wd wd = EIO_CWD;
2614 void *ptr;
2615 int res;
2616
2617 req_set_path (pathname, &wdsv, &pathsv, &wd, &ptr);
2618 RETVAL = eio__statx (!wd || wd->fd == EIO_CWD ? AT_FDCWD : wd->fd, ptr, flags, mask & STATX_ALL, &stx);
2619
2620 SvREFCNT_dec (pathsv);
2621 SvREFCNT_dec (wdsv);
2622 }
2623 OUTPUT:
2624 RETVAL
2625
2626 U32
2627 stx_mode ()
2628 PROTOTYPE:
2629 CODE:
2630 #if HAVE_STATX
2631 RETVAL = stx.stx_mode;
2632 #else
2633 XSRETURN_UNDEF;
2634 #endif
2635 OUTPUT:
2636 RETVAL
2637
2638 #define STATX_OFFSET_mask statx_offsetof (stx_mask)
2639 #define STATX_OFFSET_blksize statx_offsetof (stx_blksize)
2640 #define STATX_OFFSET_nlink statx_offsetof (stx_nlink)
2641 #define STATX_OFFSET_uid statx_offsetof (stx_uid)
2642 #define STATX_OFFSET_gid statx_offsetof (stx_gid)
2643 #define STATX_OFFSET_rdev_major statx_offsetof (stx_rdev_major)
2644 #define STATX_OFFSET_rdev_minor statx_offsetof (stx_rdev_minor)
2645 #define STATX_OFFSET_dev_major statx_offsetof (stx_dev_major)
2646 #define STATX_OFFSET_dev_minor statx_offsetof (stx_dev_minor)
2647 #define STATX_OFFSET_attributes statx_offsetof (stx_attributes)
2648 #define STATX_OFFSET_ino statx_offsetof (stx_ino)
2649 #define STATX_OFFSET_size statx_offsetof (stx_size)
2650 #define STATX_OFFSET_blocks statx_offsetof (stx_blocks)
2651 #define STATX_OFFSET_attributes_mask statx_offsetof (stx_attributes_mask)
2652 #define STATX_OFFSET_atime statx_offsetof (stx_atime)
2653 #define STATX_OFFSET_btime statx_offsetof (stx_btime)
2654 #define STATX_OFFSET_ctime statx_offsetof (stx_ctime)
2655 #define STATX_OFFSET_mtime statx_offsetof (stx_mtime)
2656
2657 U32
2658 stx_mask ()
2659 PROTOTYPE:
2660 ALIAS:
2661 stx_mask = STATX_OFFSET_mask
2662 stx_blksize = STATX_OFFSET_blksize
2663 stx_nlink = STATX_OFFSET_nlink
2664 stx_uid = STATX_OFFSET_uid
2665 stx_gid = STATX_OFFSET_gid
2666 stx_rdev_major = STATX_OFFSET_rdev_major
2667 stx_rdev_minor = STATX_OFFSET_rdev_minor
2668 stx_dev_major = STATX_OFFSET_dev_major
2669 stx_dev_minor = STATX_OFFSET_dev_minor
2670 CODE:
2671 #if HAVE_STATX
2672 RETVAL = *(__u32 *)((char *)&stx + ix);
2673 #else
2674 XSRETURN_UNDEF;
2675 #endif
2676 OUTPUT:
2677 RETVAL
2678
2679 VAL64
2680 stx_attributes ()
2681 PROTOTYPE:
2682 ALIAS:
2683 stx_attributes = STATX_OFFSET_attributes
2684 stx_ino = STATX_OFFSET_ino
2685 stx_size = STATX_OFFSET_size
2686 stx_blocks = STATX_OFFSET_blocks
2687 stx_attributes_mask = STATX_OFFSET_attributes_mask
2688 CODE:
2689 #if HAVE_STATX
2690 RETVAL = *(__u64 *)((char *)&stx + ix);
2691 #else
2692 XSRETURN_UNDEF;
2693 #endif
2694 OUTPUT:
2695 RETVAL
2696
2697 NV
2698 stx_atime ()
2699 PROTOTYPE:
2700 ALIAS:
2701 stx_atime = STATX_OFFSET_atime
2702 stx_btime = STATX_OFFSET_btime
2703 stx_ctime = STATX_OFFSET_ctime
2704 stx_mtime = STATX_OFFSET_mtime
2705 CODE:
2706 #if HAVE_STATX
2707 struct statx_timestamp *ts = (struct statx_timestamp *)((char *)&stx + ix);
2708 RETVAL = ts->tv_sec + ts->tv_nsec * 1e-9;
2709 #else
2710 XSRETURN_UNDEF;
2711 #endif
2712 OUTPUT:
2713 RETVAL
2714
2715 VAL64
2716 stx_atimesec ()
2717 PROTOTYPE:
2718 ALIAS:
2719 stx_atimesec = STATX_OFFSET_atime
2720 stx_btimesec = STATX_OFFSET_btime
2721 stx_ctimesec = STATX_OFFSET_ctime
2722 stx_mtimesec = STATX_OFFSET_mtime
2723 CODE:
2724 #if HAVE_STATX
2725 struct statx_timestamp *ts = (struct statx_timestamp *)((char *)&stx + ix);
2726 RETVAL = ts->tv_sec;
2727 #else
2728 XSRETURN_UNDEF;
2729 #endif
2730 OUTPUT:
2731 RETVAL
2732
2733 U32
2734 stx_atimensec ()
2735 PROTOTYPE:
2736 ALIAS:
2737 stx_atimensec = STATX_OFFSET_atime
2738 stx_btimensec = STATX_OFFSET_btime
2739 stx_ctimensec = STATX_OFFSET_ctime
2740 stx_mtimensec = STATX_OFFSET_mtime
2741 CODE:
2742 #if HAVE_STATX
2743 struct statx_timestamp *ts = (struct statx_timestamp *)((char *)&stx + ix);
2744 RETVAL = ts->tv_nsec;
2745 #else
2746 RETVAL = 0;
2747 #endif
2748 OUTPUT:
2749 RETVAL
2750
2751 SV *
2752 stx_atime2 ()
2753 PROTOTYPE:
2754 ALIAS:
2755 stx_atime2 = STATX_OFFSET_atime
2756 stx_btime2 = STATX_OFFSET_btime
2757 stx_ctime2 = STATX_OFFSET_ctime
2758 stx_mtime2 = STATX_OFFSET_mtime
2759 CODE:
2760 #if HAVE_STATX
2761 struct statx_timestamp *ts = (struct statx_timestamp *)((char *)&stx + ix);
2762 RETVAL = newSVts (ts->tv_sec, ts->tv_nsec);
2763 #else
2764 XSRETURN_UNDEF;
2765 #endif
2766 OUTPUT:
2767 RETVAL
2768
2769 void
2770 accept4 (aio_rfd rfh, SV *sockaddr, int salen, int flags)
2771 PPCODE:
2772 {
2773 SV *retval;
2774 #if HAVE_ACCEPT4
2775 socklen_t salen_ = salen ? salen + 1 : 0;
2776
2777 if (salen)
2778 {
2779 sv_upgrade (sockaddr, SVt_PV);
2780 sv_grow (sockaddr, salen_);
2781 }
2782
2783 int res = accept4 (rfh, salen ? (struct sockaddr *)SvPVX (sockaddr) : 0, salen ? &salen_ : 0, flags);
2784
2785 retval = newmortalFH (res, O_RDWR);
2786
2787 if (res >= 0 && salen > 0)
2788 {
2789 if (salen_ > salen + 1)
2790 salen_ = salen + 1;
2791
2792 SvPOK_only (sockaddr);
2793 SvCUR_set (sockaddr, salen_);
2794 }
2795 #else
2796 errno = ENOSYS;
2797 retval = &PL_sv_undef;
2798 #endif
2799 XPUSHs (retval);
2800 }
2801
2802 ssize_t
2803 splice (aio_rfd rfh, SV *off_in, aio_wfd wfh, SV *off_out, size_t length, unsigned int flags)
2804 CODE:
2805 {
2806 #if HAVE_LINUX_SPLICE
2807 loff_t off_in_, off_out_;
2808 RETVAL = splice (
2809 rfh, SvOK (off_in ) ? (off_in_ = SvVAL64 (off_in )), &off_in_ : 0,
2810 wfh, SvOK (off_out) ? (off_out_ = SvVAL64 (off_out)), &off_out_ : 0,
2811 length, flags
2812 );
2813 #else
2814 RETVAL = EIO_ENOSYS ();
2815 #endif
2816 }
2817 OUTPUT:
2818 RETVAL
2819
2820 ssize_t
2821 tee (aio_rfd rfh, aio_wfd wfh, size_t length, unsigned int flags)
2822 CODE:
2823 #if HAVE_LINUX_SPLICE
2824 RETVAL = tee (rfh, wfh, length, flags);
2825 #else
2826 RETVAL = EIO_ENOSYS ();
2827 #endif
2828 OUTPUT:
2829 RETVAL
2830
2831 int
2832 pipesize (aio_rfd rfh, int new_size = -1)
2833 PROTOTYPE: $;$
2834 CODE:
2835 #if defined(F_SETPIPE_SZ) && defined(F_GETPIPE_SZ)
2836 if (new_size >= 0)
2837 RETVAL = fcntl (rfh, F_SETPIPE_SZ, new_size);
2838 else
2839 RETVAL = fcntl (rfh, F_GETPIPE_SZ);
2840 #else
2841 errno = ENOSYS;
2842 RETVAL = -1;
2843 #endif
2844 OUTPUT:
2845 RETVAL
2846
2847 void
2848 pipe2 (int flags = 0)
2849 PROTOTYPE: ;$
2850 PPCODE:
2851 {
2852 int fd[2];
2853 int res;
2854
2855 if (flags)
2856 #if HAVE_PIPE2
2857 res = pipe2 (fd, flags);
2858 #else
2859 res = (errno = ENOSYS, -1);
2860 #endif
2861 else
2862 res = pipe (fd);
2863
2864 if (!res)
2865 {
2866 EXTEND (SP, 2);
2867 PUSHs (newmortalFH (fd[0], O_RDONLY));
2868 PUSHs (newmortalFH (fd[1], O_WRONLY));
2869 }
2870 }
2871
2872 void
2873 pidfd_open (int pid, unsigned int flags = 0)
2874 PPCODE:
2875 {
2876 /*GENDEF0_SYSCALL(pidfd_open,434)*/
2877 int fd = syscall (SYS_pidfd_open, pid, flags);
2878 XPUSHs (newmortalFH (fd, O_RDWR));
2879 }
2880
2881 int
2882 pidfd_send_signal (SV *pidfh, int sig, SV *siginfo = &PL_sv_undef, unsigned int flags = 0)
2883 PPCODE:
2884 {
2885 int res;
2886 #if HAVE_SIGINFO_T
2887 siginfo_t si = { 0 };
2888
2889 if (SvOK (siginfo))
2890 {
2891 HV *hv;
2892 SV **svp;
2893
2894 if (!SvROK (siginfo) || SvTYPE (SvRV (siginfo)) != SVt_PVHV)
2895 croak ("siginfo argument must be a hashref with 'code', 'pid', 'uid' and 'value_int' or 'value_ptr' members, caught");
2896
2897 hv = (HV *)SvRV (siginfo);
2898
2899 if ((svp = hv_fetchs (hv, "code" , 0))) si.si_code = SvIV (*svp);
2900 if ((svp = hv_fetchs (hv, "pid" , 0))) si.si_pid = SvIV (*svp);
2901 if ((svp = hv_fetchs (hv, "uid" , 0))) si.si_uid = SvIV (*svp);
2902 if ((svp = hv_fetchs (hv, "value_int", 0))) si.si_value.sival_int = SvIV (*svp);
2903 if ((svp = hv_fetchs (hv, "value_ptr", 0))) si.si_value.sival_ptr = (void *)SvIV (*svp);
2904 }
2905
2906 /*GENDEF0_SYSCALL(pidfd_send_signal,424)*/
2907 res = syscall (SYS_pidfd_send_signal, s_fileno_croak (pidfh, 0), sig, SvOK (siginfo) ? &si : 0, flags);
2908 #else
2909 res = (errno = ENOSYS, -1);
2910 #endif
2911
2912 XPUSHs (sv_2mortal (newSViv (res)));
2913 }
2914
2915 void
2916 pidfd_getfd (SV *pidfh, int targetfd, unsigned int flags = 0)
2917 PPCODE:
2918 {
2919 /*GENDEF0_SYSCALL(pidfd_getfd,438)*/
2920 int fd = syscall (SYS_pidfd_getfd, s_fileno_croak (pidfh, 0), targetfd, flags);
2921 XPUSHs (newmortalFH (fd, O_RDWR));
2922 }
2923
2924 void
2925 eventfd (unsigned int initval = 0, int flags = 0)
2926 PPCODE:
2927 {
2928 int fd;
2929 #if HAVE_EVENTFD
2930 fd = eventfd (initval, flags);
2931 #else
2932 fd = (errno = ENOSYS, -1);
2933 #endif
2934
2935 XPUSHs (newmortalFH (fd, O_RDWR));
2936 }
2937
2938 void
2939 timerfd_create (int clockid, int flags = 0)
2940 PPCODE:
2941 {
2942 int fd;
2943 #if HAVE_TIMERFD
2944 fd = timerfd_create (clockid, flags);
2945 #else
2946 fd = (errno = ENOSYS, -1);
2947 #endif
2948
2949 XPUSHs (newmortalFH (fd, O_RDWR));
2950 }
2951
2952 void
2953 timerfd_settime (SV *fh, int flags, NV interval, NV value)
2954 PPCODE:
2955 {
2956 int fd = s_fileno_croak (fh, 0);
2957 #if HAVE_TIMERFD
2958 int res;
2959 struct itimerspec its, ots;
2960
2961 ts_set (&its.it_interval, interval);
2962 ts_set (&its.it_value , value);
2963 res = timerfd_settime (fd, flags, &its, &ots);
2964
2965 if (!res)
2966 {
2967 EXTEND (SP, 2);
2968 PUSHs (newSVnv (ts_get (&ots.it_interval)));
2969 PUSHs (newSVnv (ts_get (&ots.it_value)));
2970 }
2971 #else
2972 errno = ENOSYS;
2973 #endif
2974 }
2975
2976 void
2977 timerfd_gettime (SV *fh)
2978 PPCODE:
2979 {
2980 int fd = s_fileno_croak (fh, 0);
2981 #if HAVE_TIMERFD
2982 int res;
2983 struct itimerspec ots;
2984 res = timerfd_gettime (fd, &ots);
2985
2986 if (!res)
2987 {
2988 EXTEND (SP, 2);
2989 PUSHs (newSVnv (ts_get (&ots.it_interval)));
2990 PUSHs (newSVnv (ts_get (&ots.it_value)));
2991 }
2992 #else
2993 errno = ENOSYS;
2994 #endif
2995 }
2996
2997 void
2998 memfd_create (octet_string pathname, int flags = 0)
2999 PPCODE:
3000 {
3001 int fd;
3002 #if HAVE_MEMFD_CREATE
3003 fd = memfd_create (pathname, flags);
3004 #else
3005 fd = (errno = ENOSYS, -1);
3006 #endif
3007
3008 XPUSHs (newmortalFH (fd, O_RDWR));
3009 }
3010
3011 int
3012 fexecve (SV *fh, SV *args, SV *envs = &PL_sv_undef)
3013 CODE:
3014 {
3015 int fd = PerlIO_fileno (IoIFP (sv_2io (fh)));
3016 char **envp, **argv;
3017 argv = extract_stringvec (args, "IO::AIO::fexecve: args must be an array of strings");
3018 if (!SvOK (envs))
3019 {
3020 extern char **environ;
3021 envp = environ;
3022 }
3023 else
3024 envp = extract_stringvec (envs, "IO::AIO::fexecve: envs must be an array of strings");
3025 #if HAVE_FEXECVE
3026 RETVAL = fexecve (fd, argv, envp);
3027 #else
3028 RETVAL = (errno = ENOSYS, -1);
3029 #endif
3030 }
3031 OUTPUT: RETVAL
3032
3033 int
3034 mount (octet_string special, octet_string path, octet_string fstype, UV flags = 0, octet_string_ornull data = 0)
3035 CODE:
3036 #if HAVE_MOUNT
3037 RETVAL = mount (special, path, fstype, flags, data);
3038 #else
3039 RETVAL = (errno = ENOSYS, -1);
3040 #endif
3041 OUTPUT: RETVAL
3042
3043 int
3044 umount (octet_string path, int flags = 0)
3045 CODE:
3046 if (flags)
3047 #if HAVE_UMOUNT2
3048 RETVAL = umount2 (path, flags);
3049 #else
3050 RETVAL = (errno = ENOSYS, -1);
3051 #endif
3052 else
3053 #if HAVE_UMOUNT
3054 RETVAL = umount (path);
3055 #else
3056 RETVAL = (errno = ENOSYS, -1);
3057 #endif
3058 OUTPUT: RETVAL
3059
3060 UV
3061 get_fdlimit ()
3062 CODE:
3063 #if HAVE_RLIMITS
3064 struct rlimit rl;
3065 if (0 == getrlimit (RLIMIT_NOFILE, &rl))
3066 XSRETURN_UV (rl.rlim_cur == RLIM_INFINITY ? (UV)-1 : rl.rlim_cur);
3067 #endif
3068 XSRETURN_UNDEF;
3069 OUTPUT:
3070 RETVAL
3071
3072 void
3073 min_fdlimit (UV limit = 0x7fffffffU)
3074 CODE:
3075 {
3076 #if HAVE_RLIMITS
3077 struct rlimit rl;
3078 rlim_t orig_rlim_max;
3079 UV bit;
3080
3081 if (0 != getrlimit (RLIMIT_NOFILE, &rl))
3082 goto fail;
3083
3084 if (rl.rlim_cur == RLIM_INFINITY)
3085 XSRETURN_YES;
3086
3087 orig_rlim_max = rl.rlim_max == RLIM_INFINITY ? ((rlim_t)0)-1 : rl.rlim_max;
3088
3089 if (rl.rlim_cur < limit)
3090 {
3091 rl.rlim_cur = limit;
3092
3093 if (rl.rlim_max < rl.rlim_cur && rl.rlim_max != RLIM_INFINITY)
3094 rl.rlim_max = rl.rlim_cur;
3095 }
3096
3097 if (0 == setrlimit (RLIMIT_NOFILE, &rl))
3098 XSRETURN_YES;
3099
3100 if (errno == EPERM)
3101 {
3102 /* setrlimit failed with EPERM - maybe we can't raise the hardlimit, or maybe */
3103 /* our limit overflows a system-wide limit */
3104 /* try an adaptive algorithm, but do not lower the hardlimit */
3105 rl.rlim_max = 0;
3106 for (bit = 0x40000000U; bit; bit >>= 1)
3107 {
3108 rl.rlim_max |= bit;
3109 rl.rlim_cur = rl.rlim_max;
3110
3111 /* never decrease the hard limit */
3112 if (rl.rlim_max < orig_rlim_max)
3113 break;
3114
3115 if (0 != setrlimit (RLIMIT_NOFILE, &rl))
3116 rl.rlim_max &= ~bit; /* too high, remove bit again */
3117 }
3118
3119 /* now, raise the soft limit to the max permitted */
3120 if (0 == getrlimit (RLIMIT_NOFILE, &rl))
3121 {
3122 rl.rlim_cur = rl.rlim_max;
3123 if (0 == setrlimit (RLIMIT_NOFILE, &rl))
3124 errno = EPERM;
3125 }
3126 }
3127 #endif
3128 fail:
3129 XSRETURN_UNDEF;
3130 }
3131
3132 void _on_next_submit (SV *cb)
3133 CODE:
3134 SvREFCNT_dec (on_next_submit);
3135 on_next_submit = SvOK (cb) ? newSVsv (cb) : 0;
3136
3137 PROTOTYPES: DISABLE
3138
3139 MODULE = IO::AIO PACKAGE = IO::AIO::WD
3140
3141 BOOT:
3142 {
3143 newCONSTSUB (aio_stash, "CWD" , newSVaio_wd (EIO_CWD ));
3144 newCONSTSUB (aio_stash, "INVALID_WD", newSVaio_wd (EIO_INVALID_WD));
3145 }
3146
3147 void
3148 DESTROY (SV *self)
3149 CODE:
3150 {
3151 aio_wd wd = SvAIO_WD (self);
3152 #if HAVE_AT
3153 {
3154 SV *callback = &PL_sv_undef;
3155 dREQ; /* clobbers next_pri :/ */
3156 next_pri = req->pri; /* restore next_pri */
3157 req->pri = EIO_PRI_MAX; /* better use max. priority to conserve fds */
3158 req->type = EIO_WD_CLOSE;
3159 req->wd = wd;
3160 REQ_SEND;
3161 }
3162 #else
3163 eio_wd_close_sync (wd);
3164 #endif
3165 }
3166
3167 MODULE = IO::AIO PACKAGE = IO::AIO::REQ
3168
3169 void
3170 cancel (aio_req_ornot req)
3171 CODE:
3172 eio_cancel (req);
3173
3174 void
3175 cb (aio_req_ornot req, SV *callback = NO_INIT)
3176 PPCODE:
3177 {
3178 if (GIMME_V != G_VOID)
3179 XPUSHs (req->callback ? sv_2mortal (newRV_inc (req->callback)) : &PL_sv_undef);
3180
3181 if (items > 1)
3182 {
3183 SV *cb_cv = get_cb (callback);
3184
3185 SvREFCNT_dec (req->callback);
3186 req->callback = SvREFCNT_inc (cb_cv);
3187 }
3188 }
3189
3190 MODULE = IO::AIO PACKAGE = IO::AIO::GRP
3191
3192 void
3193 add (aio_req grp, ...)
3194 PPCODE:
3195 {
3196 int i;
3197
3198 if (grp->int1 == 2)
3199 croak ("cannot add requests to IO::AIO::GRP after the group finished");
3200
3201 for (i = 1; i < items; ++i )
3202 {
3203 aio_req req;
3204
3205 if (GIMME_V != G_VOID)
3206 XPUSHs (sv_2mortal (newSVsv (ST (i))));
3207
3208 req = SvAIO_REQ (ST (i));
3209
3210 if (req)
3211 eio_grp_add (grp, req);
3212 }
3213 }
3214
3215 void
3216 cancel_subs (aio_req_ornot req)
3217 CODE:
3218 req_cancel_subs (req);
3219
3220 void
3221 result (aio_req grp, ...)
3222 CODE:
3223 {
3224 int i;
3225 AV *av;
3226
3227 grp->errorno = errno;
3228
3229 av = newAV ();
3230 av_extend (av, items - 1);
3231
3232 for (i = 1; i < items; ++i )
3233 av_push (av, newSVsv (ST (i)));
3234
3235 SvREFCNT_dec (grp->sv1);
3236 grp->sv1 = (SV *)av;
3237 }
3238
3239 void
3240 errno (aio_req grp, int errorno = errno)
3241 CODE:
3242 grp->errorno = errorno;
3243
3244 void
3245 limit (aio_req grp, int limit)
3246 CODE:
3247 eio_grp_limit (grp, limit);
3248
3249 void
3250 feed (aio_req grp, SV *callback = &PL_sv_undef)
3251 CODE:
3252 {
3253 SvREFCNT_dec (grp->sv2);
3254 grp->sv2 = newSVsv (callback);
3255 grp->feed = aio_grp_feed;
3256
3257 if (grp->int2 <= 0)
3258 grp->int2 = 2;
3259
3260 eio_grp_limit (grp, grp->int2);
3261 }
3262