# $OpenBSD: Server.pm,v 1.6 2021/12/12 21:16:53 bluhm Exp $ # Copyright (c) 2010-2017 Alexander Bluhm # # Permission to use, copy, modify, and distribute this software for any # purpose with or without fee is hereby granted, provided that the above # copyright notice and this permission notice appear in all copies. # # THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES # WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF # MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR # ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES # WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN # ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF # OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. use strict; use warnings; package Server; use parent 'Proc'; use Carp; use Socket qw(IPPROTO_TCP TCP_NODELAY); use Socket6; use IO::Socket; use IO::Socket::IP -register; sub new { my $class = shift; my %args = @_; $args{ktracefile} ||= "server.ktrace"; $args{logfile} ||= "server.log"; $args{up} ||= "Accepted"; $args{down} ||= "Shutdown $class"; my $self = Proc::new($class, %args); $self->{domain} or croak "$class domain not given"; $self->{protocol} or croak "$class protocol not given"; if ($self->{ktrace}) { unlink $self->{ktracefile}; my @cmd = ("ktrace", "-f", $self->{ktracefile}, "-p", $$); do { local $> = 0; system(@cmd) } and die ref($self), " system '@cmd' failed: $?"; } my $ls = do { local $> = 0; IO::Socket->new( Type => $self->{socktype}, Proto => $self->{protocol}, ReuseAddr => 1, Domain => $self->{domain}, $self->{listenaddr} ? (LocalAddr => $self->{listenaddr}) : (), $self->{listenport} ? (LocalPort => $self->{listenport}) : (), ) } or die ref($self), " socket failed: $!"; if ($self->{oobinline}) { setsockopt($ls, SOL_SOCKET, SO_OOBINLINE, pack('i', 1)) or die ref($self), " set oobinline listen failed: $!"; } if ($self->{sndbuf}) { setsockopt($ls, SOL_SOCKET, SO_SNDBUF, pack('i', $self->{sndbuf})) or die ref($self), " set sndbuf listen failed: $!"; } if ($self->{rcvbuf}) { setsockopt($ls, SOL_SOCKET, SO_RCVBUF, pack('i', $self->{rcvbuf})) or die ref($self), " set rcvbuf listen failed: $!"; } if ($self->{protocol} eq "tcp") { setsockopt($ls, IPPROTO_TCP, TCP_NODELAY, pack('i', 1)) or die ref($self), " set nodelay listen failed: $!"; listen($ls, 1) or die ref($self), " socket failed: $!"; } my $log = $self->{log}; print $log "listen sock: ",$ls->sockhost()," ",$ls->sockport(),"\n"; $self->{listenaddr} = $ls->sockhost() unless $self->{listenaddr}; $self->{listenport} = $ls->sockport() unless $self->{listenport}; $self->{ls} = $ls; if ($self->{ktrace}) { my @cmd = ("ktrace", "-c", "-f", $self->{ktracefile}, "-p", $$); do { local $> = 0; system(@cmd) } and die ref($self), " system '@cmd' failed: $?"; } return $self; } sub child { my $self = shift; my $as = $self->{ls}; if ($self->{protocol} eq "tcp") { $as = $self->{ls}->accept() or die ref($self), " socket accept failed: $!"; print STDERR "accept sock: ",$as->sockhost()," ", $as->sockport(),"\n"; print STDERR "accept peer: ",$as->peerhost()," ", $as->peerport(),"\n"; } if ($self->{nonblocking}) { $as->blocking(0) or die ref($self), " set non-blocking accept failed: $!"; } open(STDIN, '<&', $as) or die ref($self), " dup STDIN failed: $!"; open(STDOUT, '>&', $as) or die ref($self), " dup STDOUT failed: $!"; } 1;