NAME
    IO::FD - Faster accept, socket, listen with file descriptors, not
    handles

SYNOPSIS
    Create and bind a STREAM socket (server):

            use IO::FD;
            use Socket ":all";

            die "Error creating socket"
                    unless defined IO::FD::socket(my $listen_fd, AF_INET, SOCK_STREAM, 0);

            my ($err, @sockaddr)=addrinfo "0.0.0.0", 80, {
                    family=>        AF_INET,
                    socktype=>      SOCK_STREAM,
                    flags=>         AI_NUMERICHOST|AI_PASSIVE
            };

            die "Error binding"
                    unless defined FD::IO::bind($listen_fd, $sockaddr[0]{addr});    

        
            die "Error accepting" 
                    unless defined IO::FD::accept(my $client_fd, $listen_fd);
        
            #read and write here

    Create and connect a STREAM socket(client):

            use IO::FD;
            use Socket ":all";

            die "Error creating socket"
                    unless defined IO::FD::socket(my $fd, AF_INET,SOCK_STREAM,0);

            my ($err,@sockaddr)=addrinfo "127.0.0.1", 80, {
                    family=>        AF_INET,
                    socktype=>      SOCK_STREAM,
                    flags=>         AI_NUMERICHOST
            };

            die "Error connecting";
                    unless defined FD::IO::connect($fd, $sockaddr[0]{addr});

            #read and write here

    Open a file

            use IO::FD;
            use Fcntl;
            die "could not open file" 
                    unless defined IO::FD::sysopen(my $fd, "path.txt", O_RDONLY);

    Read/Write/Close an fd

            use IO::FD;

            my $fd; #From IO::FD::socket, IO::FD::accept IO::FD::sysopen, POSIX::open

            die "Error writing"
                    unless defined IO::FD::syswrite $fd, "This is some data"; #Length and optional offset

            die "Error reading"
                    unless defined IO::FD::sysread $fd, my $buffer, $length); 

            die "Error closing" 
                    unless defined IO::FD::close $fd;

    Advanced:

            fcntl
      sendfile
      accept4
      accept_multiple

            #TODO:
            ioctl...

DESCRIPTION
    IO::FD is an XS module implementing common core Perl system I/O
    functions to work with file descriptors instead of Perl file handles.
    Functions include but are not limited to "accept", "connect", "socket",
    "bind", "sysopen", "sysread", and "syswrite".

    Many non core system functions such as "sendfile", "dup" and "mkstemp",
    "pread", "pwrite", "mkfifo" which work with file descriptors are also
    implemented.

    Additional support for streamlined connection accepting is included via
    "accept_multiple".

    This module can significantly lower memory usage per file descriptor and
    decrease file/socket opening and socket accepting times. "accept"
    performance is particularly improved with much higher connection
    handling rates for a given backlog.

    Actual byte throughput (read/write) is basically unchanged compared to
    the core Perl sysread/syswrite. Please see the PERFORMANCE section later
    in this document

    The supported interfaces mostly resemble the core Perl implementations
    of similarly named functions. The largest difference being you should
    check for defined values for error checks. For example:

            #Perl:
            sysopen(my $file_handle, ...) or die $!;
            sysread($file_handle, ...) or die $!;

            #IO::FD
            defined IO::FD::sysopen(my $file_descriptor, ...) or die $!
            defined IO::FD::sysread($file_descriptor, ...) or die $!;

    This modules IS NOT intended to be a drop in replacement for core IO
    subroutines in existing code. If you want a 'drop in replacement' please
    look at IO::FD::DWIM which is part of the same distribution.

    Currently this module is focused on UNIX/Linux systems, as this is the
    natural habitat of a file descriptor.

IMPORTANT VERSION DIFFERENCES
  v0.3.0 and later
    New functions:

    "pread", "pwrite", "mkfifo", "mkfifoat", "open", "openat"

  v0.2.0 and later
    New functions:

    "accept4", "accept_multiple", "sendfile"

    Changes:

    All functions creating a new fd now behave more perlish and apply
    O_CLOEXEC if larger than $^F to prevent fd leakage. This may result in
    an extra system call you didn't need if your program never calls "exec".
    To disable this, increase the value of $^F as per normal.

    Functions now throw exceptions when output variables (fds) are read only
    when they need to be writable. This matches Perl behaviour in the same
    scenario for "sysopen" etc.

    When function input fd variables doesn't look like an fd (an IV), a
    warning 'IO::FD::xxxx called with something other than a file
    descriptor' is generated, return value is "undef" and the $! variable is
    set to "EBADF" (bad file descriptor>). This is analogous to Perl
    behaviour when checking for valid GLOB/refs with "sysread" and friends.

WHERE SHOULD I USE THIS MODULE?
  Networking ... Oh Yes
    Socket centric programs will benefit greatly from this module. The
    process of socket creation/opening/accepting/listening, where it is
    INET/INET6 or UNIX families is much improved.

  Slurp entire file ... Yes
    If a file can be loaded completely into memory for processing, this
    module will provide improved opening and closing times. Any decoding and
    line processing will need to be done manually

  Line Processing ... Hmmm, No
    General text file line processing is best left to Perl file handles.
    File handles do the heavy lifting of line splitting, EOL handling,
    encodings, which this modules does not implement.

    You can do it, but it is not in the scope of this module.

LIMITATIONS
    Perl does a lot of nice things when working with files and handles. When
    using file descriptors directly you will loose:

    Buffering for file small read/write performance (via print and <FH>)
    Automatic close when out of scope
    Special variables not supported (ie '_' in stat)
    <FH> 'readline' support
    IO::Handle inheritance

MOTIVATION
    Perl makes working with text files easy, thanks to file handles. Line
    splitting, UTF-8, EOL processing etc. are awesome and make your life
    easier.

    However, the benefits of file handles when working within a network or
    binary file context are not so clear cut. All the nice line ending and
    encoding support doesn't help in these scenarios.

    In addition, the OS kernel does a lot of buffering for networking
    already. Do we really need to add more?

    So if these features are not being fully utilised for binary/network
    programming, the hypothesis is that opening and accepting operations
    would be faster with file descriptors as less setup is required
    internally.

APIs
    Each of the APIs mimic the Perl counterpart, if applicable, as much as
    possible. Unless explicitly mentioned, they should operate like built in
    routines. Consult perldoc -f FUNCTION for details. The general exception
    however is return values should be explicitly tested for definedness and
    not relying on a 'true' value

    As none of these functions are exported, they must be called with full
    package name.

  Socket Manipulation
   IO::FD::socket
   IO::FD::socketpair
   IO::FD::bind
   IO::FD::listen
   IO::FD::accept
   IO::FD::accept4
      my $ok=defined IO::FD::accept4 $new, $listen, $flags;
  
      Constants: IO::FD::SOCK_NONBLOCK, IO::FD::SOCK_CLOEXEC

    Implements the linux "accept4" syscall. On non linux systems this is
    emulated by calling "fcntl" to set the FD_CLOEXEC flag and O_NONBLOCK
    status. Returns "undef" on error, for the user to test $!.

    The flags argument can be the bitwise or'ed value of "SOCK_NONBLOCK" and
    "SOCK_CLOEXEC" from the Socket module on linux and bsd. Darwin (macos)
    does not have these values. so please use "IO::FD::SOCK_NONBLOCK",
    "IO::FD::SOCK_CLOEXEC" on that platform.

    NOTE:Unlike other functions returning new file descriptors, this DOES
    NOT automatically apply the CLOEXEC flag. The $flags argument must be
    set accordingly to achieve this.

    NOTE: On emulated systems, any errors reported are only from the accept
    call, not subsequent "fcntl" calls

   IO::FD::accept_multiple
            my @new_fds;
            my @peers;
            my $count=accept_multiple(@new_fds, @peers, $listen_sock);

    NOTE: DO NOT use this function on a blocking socket!!

    Accepts as many new connection sockets as available. The new sockets are
    stored in "new_fds", which is an array, not a array ref. The
    corresponding peers to the connections are stored in @peers, also an
    array not a reference.

    $listen_sock is the file descriptor from which the sockets are accepted
    from. It MUST be configured for non blocking operation, otherwise your
    program will just loop forever in this function

    Because this function will only works for non blocking listening
    sockets, the sockets/fds returned are configured for non blocking mode
    also. On BSD type systems the socket will already be non blocking. On
    linux systems the accept4 call is used to set the SOCK_NONBLOCK flag.

    Returns the number of sockets accepted until an error condition
    occurred. Returns "undef" if no sockets where accepted. Check the $! for
    normal non blocking error codes.

   IO::FD::connect
   IO::FD::getsockopt
   IO::FD::setsockopt
    Note: Implements the integer shorthand as per perldoc -f setsockopt

   IO::FD::getpeername
   IO::FD::getsockname
   IO::FD::sendfile
      sendfile $socket, $source_fd, $length, $offset

    Calls system sendfile. Returns "undef" on error or the number of bytes
    transferred otherwise. The error might be an EAGAIN for non blocking
    sockets. Please reference the manual page for sendfile on your system,
    but be mindful the position of the arguments might not match.

    Currently advanced header/trailer features of BSD sendfile are not
    supported.

  File Maniupulation
   IO::FD::sysopen
   IO::FD::sysopen4
    Same as "IO::FD::sysopen", but expects all four arguments

   IO::FD::open
    Binding to "open". Please see your system manual. If no mode is
    specified, the 'perlish' 0666, is used.

   IO::FD::openat
    Binding to "openat". Please see your system manual. If no mode is
    specified, the 'perlish' 0666, is used.

   IO::FD::mktemp
    Behaves similar to File::Temp::mktemp

    Requires at least six 'X' characters at the end of the template

    The template string used as input is modified and is the same as the
    return value on success

    NOTE: This function does not return a file descriptor. It might be
    included in future versions of this module

   IO::FD::mkstemp
    Behaves like File::Temp::mkstemp

    Requires at least six 'X' characters at the end of the template

    In list context returns "($fd,$path)", where $fd is the already open
    file descriptor, and $path is the unique path generated from the
    template.

    The template string used as input is modified and is the same as the
    $path return value on success

   IO::FD::sysseek
  Pipes
   IO::FD::pipe
   IO::FD::syspipe
    A alias of "IO::FD::pipe".

   mkfifo
    Binding to "mkfifo". Please see your system manual. If no mode is
    specified, the 'perlish' 0666, is used.

   mkfifoat
    Binding to "mkfifoat". Please see your system manual. If no mode is
    specified, the 'perlish' 0666, is used.

  Common
   IO::FD::dup
   IO::FD::dup2
   IO::FD::close
   IO::FD::recv
   IO::FD::send
   IO::FD::sysread
    NOTE: Versions prior to 0.1.4 would end up using fd = 0 (normally STDIN)
    when it was non numeric. This is fixed in 0.1.4. An fd which is not
    numeric will cause an immediate return of undefined.

   IO::FD::sysread3
    Same as "IO::FD::sysread", but expects only 3 of 4 arguments

   IO::FD::sysread4
    Same as "IO::FD::sysread", but expects all four arguments

   IO::FD::syswrite
    NOTE: Versions prior to 0.1.4 would end up using fd = 0 (normally STDIN)
    when it was non numeric. This is fixed in 0.1.4. An fd which is not
    numeric will cause an immediate return of undefined.

   IO::FD::syswrite2
    Same as "IO::FD::syswrite", but expect 2 of 4 arguments.

   IO::FD::syswrite3
    Same as "IO::FD::syswrite", but expect 3 of 4 arguments.

   IO::FD::syswrite4
    Same as "IO::FD::syswrite", but expect 4 of 4 arguments.

   IO::FD::fcntl
   IO::FD::sysfcntl
    Alias to "IO::FD::fcntl"

   IO::FD::stat
    Almost the same as CORE::stat. If your system uses a signed dev_t for
    st_dev and st_rdev, (ie macos), this module will preserve both. At the
    time of writing, Perl CORE::stat will only preserve the sign of st_dev,
    and assume st_rdev is signed in all cases.

    Attempts to replicate string expansion of values for some stat values as
    CORE::stat does. Tests of equality should be done using the "eq"
    operator (just like CORE::stat results) if your Perl is not 64bit.

   IO::FD::lstat
    As above.

   IO::FD::pread
    Binding to "pread". Please see your system manual.

   IO::FD:pwrite
    Binding to "pwrite". Please see your system manual.

  Experimental
    These functions haven't really been tested, documented or finished. They
    exist none the less. You will need to Look at the code for documentation
    at the moment. Their behaviour and interface are LIKELY TO CHANGE
    without notice.

   IO::FD::ioctl
    Not complete

   IO::FD::sysioctl
    Alias to ioctl

   IO::FD::clock_gettime_monotonic
   IO::FD::select
    Broken. Probably will be removed as core Perl has this already.

   IO::FD::poll
    Constants for use with poll are available via "IO::FD:Constants"

   IO::FD::kqueue
   IO::FD::kevent
    This is broken ok 32 bit BSD at the moment. Constants for use with
    kevent are available via "IO::FD:Constants"

   IO::FD::pack_kevent
   IO::FD::sv_to_pointer
   IO::FD::pointer_to_sv
   IO::FD::SV
            IO::FD::SV($size)

    Allocates a string SV with the given size preallocated. The current
    string length is set to 0. For short string this is not the fastest way
    to allocate. For 4k and above, it is much faster, and doesn't use extra
    memory in compilation

   IO::FD::readline
            #SLURP A FILE
            local $/=undef;
            my $slurp=IO::FD::readline;

                    #or
            #SLURP ALL RECORDS OF KNOWN LENGTH
            local $/=\1234;
            my @records=IO::FD::readline;

    A read line function is available, but is only operates in file slurp or
    record slurp mode (see perldoc -f readline). As no buffering is used, It
    does not attempt to split lines or read a line at a time like the normal
    Perl readline or <> operator

PERFORMANCE
    Part of this distribution are benchmarking scripts. The following are
    typical outputs from my Intel 2020 Macbook Pro.

  Listen Backlog
    Results from benchmark/server-perl.pl benchmark/server.pl and
    benchmark/client.pl

            Listen Backlog: 10
                    Perl server:
                    Connections before client refused: 18

                    IO::FD server
                    Connections before client refuse: 9285

            Listen Backlog: 100
                    Perl server:
                    Connections before client refused: 190

                    IO::FD server
                    Connections before client refuse:  (none refused)

            Listen Backlog: 1000

                    Perl server:
                    Connections before client refused: 245

                    IO::FD server
                    Connections before client refuse:  (none refused)

  Accept
    Results from benchmark/server-perl.pl benchmark/server.pl and
    benchmark/client.pl

            Listen Backlog: 100
        
            Perl accept rate:    73568.4857256754/s
            IO::FD Accept rate: 150984.798776367/s

  Memory Usage
    Results from benchmark/file-memory.pl

            Creating 2000 file handles/descriptors
            Start maxrss (kB): 4500

            Perl file handles
            Bytes: 905216, per handle: 452.608

            IO::FD
            Bytes: 4096, per fd: 2.048

            End maxrss (kB): 5692

  Socket creation
    Results from benchmark/socket-create.pl

                                 Rate perl_socket_INET iofd_socket_INET
            perl_socket_INET  81919/s               --             -56%
            iofd_socket_INET 185679/s             127%               --
                                  Rate perl_socket_INET6 iofd_socket_INET6
            perl_socket_INET6  81498/s                --              -57%
            iofd_socket_INET6 189253/s              132%                --
                                 Rate perl_socket_UNIX iofd_socket_UNIX
            perl_socket_UNIX 113778/s               --             -78%
            iofd_socket_UNIX 508970/s             347%               --

  File open and close
    Results from benchmark/file-open-close.pl

                                Rate     file_handle file_desc_posix           io_fd
            file_handle      91897/s              --            -35%            -37%
            file_desc_posix 140549/s             53%              --             -4%
            io_fd           146161/s             59%              4%              --

  Read Performance
    Result from benchmark/file-read-write.pl

            Read performance:
            Read (bytes): 1024 x 2^0
                                 Rate file_desc_posix     file_handle           io_fd
            file_desc_posix 1803743/s              --             -5%             -5%
            file_handle     1889325/s              5%              --             -0%
            io_fd           1890461/s              5%              0%              --
            Read (bytes): 1024 x 2^1
                                 Rate file_desc_posix           io_fd     file_handle
            file_desc_posix 1799026/s              --             -1%             -2%
            io_fd           1823610/s              1%              --             -1%
            file_handle     1837458/s              2%              1%              --
            Read (bytes): 1024 x 2^2
                                 Rate file_desc_posix           io_fd     file_handle
            file_desc_posix 1731140/s              --             -1%             -1%
            io_fd           1747626/s              1%              --             -0%
            file_handle     1747627/s              1%              0%              --
            Read (bytes): 1024 x 2^3
                                 Rate           io_fd file_desc_posix     file_handle
            io_fd           1458670/s              --             -1%             -3%
            file_desc_posix 1470359/s              1%              --             -2%
            file_handle     1499189/s              3%              2%              --
            Read (bytes): 1024 x 2^4
                                 Rate file_desc_posix     file_handle           io_fd
            file_desc_posix 1146879/s              --             -3%             -6%
            file_handle     1180322/s              3%              --             -3%
            io_fd           1214700/s              6%              3%              --

  Write Performance
    Result from benchmark/file-read-write.pl

            Write performance:
            Write (bytes): 1024 x 2^0
                                 Rate file_desc_posix           io_fd     file_handle
            file_desc_posix 1978800/s              --             -7%            -12%
            io_fd           2117316/s              7%              --             -6%
            file_handle     2244774/s             13%              6%              --
            Write (bytes): 1024 x 2^1
                                 Rate file_desc_posix           io_fd     file_handle
            file_desc_posix 2007408/s              --             -6%             -9%
            io_fd           2143700/s              7%              --             -3%
            file_handle     2205537/s             10%              3%              --
            Write (bytes): 1024 x 2^2
                                 Rate file_desc_posix           io_fd     file_handle
            file_desc_posix 1978800/s              --             -7%            -12%
            io_fd           2123851/s              7%              --             -5%
            file_handle     2244774/s             13%              6%              --
            Write (bytes): 1024 x 2^3
                                 Rate file_desc_posix           io_fd     file_handle
            file_desc_posix 1960478/s              --             -7%             -9%
            io_fd           2117316/s              8%              --             -2%
            file_handle     2163924/s             10%              2%              --
            Write (bytes): 1024 x 2^4
                                 Rate file_desc_posix           io_fd     file_handle
            file_desc_posix 1997468/s              --             -5%             -8%
            io_fd           2104367/s              5%              --             -3%
            file_handle     2163924/s              8%              3%              --

SEE ALSO
    The POSIX module provides an "open", "close", "read" and "write"
    routines which return/work with file descriptors. If you are only
    concerned with working with files, this is a better option as it is a
    core module, and will give you the purported benefits of this module.
    However it does not provide any networking/socket support.

FUTURE WORK (IDEAS/TODO)
      Further emulate linux/bsd SOCK_NONBLOCK and SOCK_CLOEXEC on darwin
            Add more tests for stat and DWIM module
            Wider compatability for older Perls
            Add More system functions which work with fds
            Work with win32 sockets
            Maybe make an IO::Handle sub class

AUTHOR
    Ruben Westerberg, <drclaw@mac.com>

REPOSITORTY and BUGS
    Please report any bugs via git hub:
    <http://github.com/drclaw1394/perl-io-fd>

COPYRIGHT AND LICENSE
    Copyright (C) 2023 by Ruben Westerberg

    This library is free software; you can redistribute it and/or modify it
    under the same terms as Perl or the MIT license.

DISCLAIMER OF WARRANTIES
    THIS PACKAGE IS PROVIDED "AS IS" AND WITHOUT ANY EXPRESS OR IMPLIED
    WARRANTIES, INCLUDING, WITHOUT LIMITATION, THE IMPLIED WARRANTIES OF
    MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE.