Rashed Abdel-Tawab | 4db47f4 | 2019-09-06 10:38:22 -0700 | [diff] [blame] | 1 | package Socket; |
| 2 | |
| 3 | use strict; |
| 4 | { use 5.006001; } |
| 5 | |
| 6 | our $VERSION = '2.020_03'; # patched in perl5.git |
| 7 | |
| 8 | # Still undocumented: SCM_*, SOMAXCONN, IOV_MAX, UIO_MAXIOV |
| 9 | |
| 10 | use Carp; |
| 11 | use warnings::register; |
| 12 | |
| 13 | require Exporter; |
| 14 | require XSLoader; |
| 15 | our @ISA = qw(Exporter); |
| 16 | |
| 17 | # <@Nicholas> you can't change @EXPORT without breaking the implicit API |
| 18 | # Please put any new constants in @EXPORT_OK! |
| 19 | |
| 20 | # List re-ordered to match documentation above. Try to keep the ordering |
| 21 | # consistent so it's easier to see which ones are or aren't documented. |
| 22 | our @EXPORT = qw( |
| 23 | PF_802 PF_AAL PF_APPLETALK PF_CCITT PF_CHAOS PF_CTF PF_DATAKIT |
| 24 | PF_DECnet PF_DLI PF_ECMA PF_GOSIP PF_HYLINK PF_IMPLINK PF_INET PF_INET6 |
| 25 | PF_ISO PF_KEY PF_LAST PF_LAT PF_LINK PF_MAX PF_NBS PF_NIT PF_NS PF_OSI |
| 26 | PF_OSINET PF_PUP PF_ROUTE PF_SNA PF_UNIX PF_UNSPEC PF_USER PF_WAN |
| 27 | PF_X25 |
| 28 | |
| 29 | AF_802 AF_AAL AF_APPLETALK AF_CCITT AF_CHAOS AF_CTF AF_DATAKIT |
| 30 | AF_DECnet AF_DLI AF_ECMA AF_GOSIP AF_HYLINK AF_IMPLINK AF_INET AF_INET6 |
| 31 | AF_ISO AF_KEY AF_LAST AF_LAT AF_LINK AF_MAX AF_NBS AF_NIT AF_NS AF_OSI |
| 32 | AF_OSINET AF_PUP AF_ROUTE AF_SNA AF_UNIX AF_UNSPEC AF_USER AF_WAN |
| 33 | AF_X25 |
| 34 | |
| 35 | SOCK_DGRAM SOCK_RAW SOCK_RDM SOCK_SEQPACKET SOCK_STREAM |
| 36 | |
| 37 | SOL_SOCKET |
| 38 | |
| 39 | SO_ACCEPTCONN SO_ATTACH_FILTER SO_BACKLOG SO_BROADCAST SO_CHAMELEON |
| 40 | SO_DEBUG SO_DETACH_FILTER SO_DGRAM_ERRIND SO_DOMAIN SO_DONTLINGER |
| 41 | SO_DONTROUTE SO_ERROR SO_FAMILY SO_KEEPALIVE SO_LINGER SO_OOBINLINE |
| 42 | SO_PASSCRED SO_PASSIFNAME SO_PEERCRED SO_PROTOCOL SO_PROTOTYPE |
| 43 | SO_RCVBUF SO_RCVLOWAT SO_RCVTIMEO SO_REUSEADDR SO_REUSEPORT |
| 44 | SO_SECURITY_AUTHENTICATION SO_SECURITY_ENCRYPTION_NETWORK |
| 45 | SO_SECURITY_ENCRYPTION_TRANSPORT SO_SNDBUF SO_SNDLOWAT SO_SNDTIMEO |
| 46 | SO_STATE SO_TYPE SO_USELOOPBACK SO_XOPEN SO_XSE |
| 47 | |
| 48 | IP_OPTIONS IP_HDRINCL IP_TOS IP_TTL IP_RECVOPTS IP_RECVRETOPTS |
| 49 | IP_RETOPTS |
| 50 | |
| 51 | MSG_BCAST MSG_BTAG MSG_CTLFLAGS MSG_CTLIGNORE MSG_CTRUNC MSG_DONTROUTE |
| 52 | MSG_DONTWAIT MSG_EOF MSG_EOR MSG_ERRQUEUE MSG_ETAG MSG_FIN |
| 53 | MSG_MAXIOVLEN MSG_MCAST MSG_NOSIGNAL MSG_OOB MSG_PEEK MSG_PROXY MSG_RST |
| 54 | MSG_SYN MSG_TRUNC MSG_URG MSG_WAITALL MSG_WIRE |
| 55 | |
| 56 | SHUT_RD SHUT_RDWR SHUT_WR |
| 57 | |
| 58 | INADDR_ANY INADDR_BROADCAST INADDR_LOOPBACK INADDR_NONE |
| 59 | |
| 60 | SCM_CONNECT SCM_CREDENTIALS SCM_CREDS SCM_RIGHTS SCM_TIMESTAMP |
| 61 | |
| 62 | SOMAXCONN |
| 63 | |
| 64 | IOV_MAX |
| 65 | UIO_MAXIOV |
| 66 | |
| 67 | sockaddr_family |
| 68 | pack_sockaddr_in unpack_sockaddr_in sockaddr_in |
| 69 | pack_sockaddr_in6 unpack_sockaddr_in6 sockaddr_in6 |
| 70 | pack_sockaddr_un unpack_sockaddr_un sockaddr_un |
| 71 | |
| 72 | inet_aton inet_ntoa |
| 73 | ); |
| 74 | |
| 75 | # List re-ordered to match documentation above. Try to keep the ordering |
| 76 | # consistent so it's easier to see which ones are or aren't documented. |
| 77 | our @EXPORT_OK = qw( |
| 78 | CR LF CRLF $CR $LF $CRLF |
| 79 | |
| 80 | SOCK_NONBLOCK SOCK_CLOEXEC |
| 81 | |
| 82 | IP_ADD_MEMBERSHIP IP_ADD_SOURCE_MEMBERSHIP IP_DROP_MEMBERSHIP |
| 83 | IP_DROP_SOURCE_MEMBERSHIP IP_MULTICAST_IF IP_MULTICAST_LOOP |
| 84 | IP_MULTICAST_TTL |
| 85 | |
| 86 | IPPROTO_IP IPPROTO_IPV6 IPPROTO_RAW IPPROTO_ICMP IPPROTO_IGMP |
| 87 | IPPROTO_TCP IPPROTO_UDP IPPROTO_GRE IPPROTO_ESP IPPROTO_AH |
| 88 | IPPROTO_SCTP |
| 89 | |
| 90 | IPTOS_LOWDELAY IPTOS_THROUGHPUT IPTOS_RELIABILITY IPTOS_MINCOST |
| 91 | |
| 92 | TCP_CONGESTION TCP_CONNECTIONTIMEOUT TCP_CORK TCP_DEFER_ACCEPT TCP_INFO |
| 93 | TCP_INIT_CWND TCP_KEEPALIVE TCP_KEEPCNT TCP_KEEPIDLE TCP_KEEPINTVL |
| 94 | TCP_LINGER2 TCP_MAXRT TCP_MAXSEG TCP_MD5SIG TCP_NODELAY TCP_NOOPT |
| 95 | TCP_NOPUSH TCP_QUICKACK TCP_SACK_ENABLE TCP_STDURG TCP_SYNCNT |
| 96 | TCP_WINDOW_CLAMP |
| 97 | |
| 98 | IN6ADDR_ANY IN6ADDR_LOOPBACK |
| 99 | |
| 100 | IPV6_ADD_MEMBERSHIP IPV6_DROP_MEMBERSHIP IPV6_JOIN_GROUP |
| 101 | IPV6_LEAVE_GROUP IPV6_MTU IPV6_MTU_DISCOVER IPV6_MULTICAST_HOPS |
| 102 | IPV6_MULTICAST_IF IPV6_MULTICAST_LOOP IPV6_UNICAST_HOPS IPV6_V6ONLY |
| 103 | |
| 104 | pack_ip_mreq unpack_ip_mreq pack_ip_mreq_source unpack_ip_mreq_source |
| 105 | |
| 106 | pack_ipv6_mreq unpack_ipv6_mreq |
| 107 | |
| 108 | inet_pton inet_ntop |
| 109 | |
| 110 | getaddrinfo getnameinfo |
| 111 | |
| 112 | AI_ADDRCONFIG AI_ALL AI_CANONIDN AI_CANONNAME AI_IDN |
| 113 | AI_IDN_ALLOW_UNASSIGNED AI_IDN_USE_STD3_ASCII_RULES AI_NUMERICHOST |
| 114 | AI_NUMERICSERV AI_PASSIVE AI_V4MAPPED |
| 115 | |
| 116 | NI_DGRAM NI_IDN NI_IDN_ALLOW_UNASSIGNED NI_IDN_USE_STD3_ASCII_RULES |
| 117 | NI_NAMEREQD NI_NOFQDN NI_NUMERICHOST NI_NUMERICSERV |
| 118 | |
| 119 | NIx_NOHOST NIx_NOSERV |
| 120 | |
| 121 | EAI_ADDRFAMILY EAI_AGAIN EAI_BADFLAGS EAI_BADHINTS EAI_FAIL EAI_FAMILY |
| 122 | EAI_NODATA EAI_NONAME EAI_PROTOCOL EAI_SERVICE EAI_SOCKTYPE EAI_SYSTEM |
| 123 | ); |
| 124 | |
| 125 | our %EXPORT_TAGS = ( |
| 126 | crlf => [qw(CR LF CRLF $CR $LF $CRLF)], |
| 127 | addrinfo => [qw(getaddrinfo getnameinfo), grep m/^(?:AI|NI|NIx|EAI)_/, @EXPORT_OK], |
| 128 | all => [@EXPORT, @EXPORT_OK], |
| 129 | ); |
| 130 | |
| 131 | BEGIN { |
| 132 | sub CR () {"\015"} |
| 133 | sub LF () {"\012"} |
| 134 | sub CRLF () {"\015\012"} |
| 135 | |
| 136 | # These are not gni() constants; they're extensions for the perl API |
| 137 | # The definitions in Socket.pm and Socket.xs must match |
| 138 | sub NIx_NOHOST() {1 << 0} |
| 139 | sub NIx_NOSERV() {1 << 1} |
| 140 | } |
| 141 | |
| 142 | *CR = \CR(); |
| 143 | *LF = \LF(); |
| 144 | *CRLF = \CRLF(); |
| 145 | |
| 146 | sub sockaddr_in { |
| 147 | if (@_ == 6 && !wantarray) { # perl5.001m compat; use this && die |
| 148 | my($af, $port, @quad) = @_; |
| 149 | warnings::warn "6-ARG sockaddr_in call is deprecated" |
| 150 | if warnings::enabled(); |
| 151 | pack_sockaddr_in($port, inet_aton(join('.', @quad))); |
| 152 | } elsif (wantarray) { |
| 153 | croak "usage: (port,iaddr) = sockaddr_in(sin_sv)" unless @_ == 1; |
| 154 | unpack_sockaddr_in(@_); |
| 155 | } else { |
| 156 | croak "usage: sin_sv = sockaddr_in(port,iaddr))" unless @_ == 2; |
| 157 | pack_sockaddr_in(@_); |
| 158 | } |
| 159 | } |
| 160 | |
| 161 | sub sockaddr_in6 { |
| 162 | if (wantarray) { |
| 163 | croak "usage: (port,in6addr,scope_id,flowinfo) = sockaddr_in6(sin6_sv)" unless @_ == 1; |
| 164 | unpack_sockaddr_in6(@_); |
| 165 | } |
| 166 | else { |
| 167 | croak "usage: sin6_sv = sockaddr_in6(port,in6addr,[scope_id,[flowinfo]])" unless @_ >= 2 and @_ <= 4; |
| 168 | pack_sockaddr_in6(@_); |
| 169 | } |
| 170 | } |
| 171 | |
| 172 | sub sockaddr_un { |
| 173 | if (wantarray) { |
| 174 | croak "usage: (filename) = sockaddr_un(sun_sv)" unless @_ == 1; |
| 175 | unpack_sockaddr_un(@_); |
| 176 | } else { |
| 177 | croak "usage: sun_sv = sockaddr_un(filename)" unless @_ == 1; |
| 178 | pack_sockaddr_un(@_); |
| 179 | } |
| 180 | } |
| 181 | |
| 182 | XSLoader::load(__PACKAGE__, $VERSION); |
| 183 | |
| 184 | my %errstr; |
| 185 | |
| 186 | if( defined &getaddrinfo ) { |
| 187 | # These are not part of the API, nothing uses them, and deleting them |
| 188 | # reduces the size of %Socket:: by about 12K |
| 189 | delete $Socket::{fake_getaddrinfo}; |
| 190 | delete $Socket::{fake_getnameinfo}; |
| 191 | } else { |
| 192 | require Scalar::Util; |
| 193 | |
| 194 | *getaddrinfo = \&fake_getaddrinfo; |
| 195 | *getnameinfo = \&fake_getnameinfo; |
| 196 | |
| 197 | # These numbers borrowed from GNU libc's implementation, but since |
| 198 | # they're only used by our emulation, it doesn't matter if the real |
| 199 | # platform's values differ |
| 200 | my %constants = ( |
| 201 | AI_PASSIVE => 1, |
| 202 | AI_CANONNAME => 2, |
| 203 | AI_NUMERICHOST => 4, |
| 204 | AI_V4MAPPED => 8, |
| 205 | AI_ALL => 16, |
| 206 | AI_ADDRCONFIG => 32, |
| 207 | # RFC 2553 doesn't define this but Linux does - lets be nice and |
| 208 | # provide it since we can |
| 209 | AI_NUMERICSERV => 1024, |
| 210 | |
| 211 | EAI_BADFLAGS => -1, |
| 212 | EAI_NONAME => -2, |
| 213 | EAI_NODATA => -5, |
| 214 | EAI_FAMILY => -6, |
| 215 | EAI_SERVICE => -8, |
| 216 | |
| 217 | NI_NUMERICHOST => 1, |
| 218 | NI_NUMERICSERV => 2, |
| 219 | NI_NOFQDN => 4, |
| 220 | NI_NAMEREQD => 8, |
| 221 | NI_DGRAM => 16, |
| 222 | |
| 223 | # Constants we don't support. Export them, but croak if anyone tries to |
| 224 | # use them |
| 225 | AI_IDN => 64, |
| 226 | AI_CANONIDN => 128, |
| 227 | AI_IDN_ALLOW_UNASSIGNED => 256, |
| 228 | AI_IDN_USE_STD3_ASCII_RULES => 512, |
| 229 | NI_IDN => 32, |
| 230 | NI_IDN_ALLOW_UNASSIGNED => 64, |
| 231 | NI_IDN_USE_STD3_ASCII_RULES => 128, |
| 232 | |
| 233 | # Error constants we'll never return, so it doesn't matter what value |
| 234 | # these have, nor that we don't provide strings for them |
| 235 | EAI_SYSTEM => -11, |
| 236 | EAI_BADHINTS => -1000, |
| 237 | EAI_PROTOCOL => -1001 |
| 238 | ); |
| 239 | |
| 240 | foreach my $name ( keys %constants ) { |
| 241 | my $value = $constants{$name}; |
| 242 | |
| 243 | no strict 'refs'; |
| 244 | defined &$name or *$name = sub () { $value }; |
| 245 | } |
| 246 | |
| 247 | %errstr = ( |
| 248 | # These strings from RFC 2553 |
| 249 | EAI_BADFLAGS() => "invalid value for ai_flags", |
| 250 | EAI_NONAME() => "nodename nor servname provided, or not known", |
| 251 | EAI_NODATA() => "no address associated with nodename", |
| 252 | EAI_FAMILY() => "ai_family not supported", |
| 253 | EAI_SERVICE() => "servname not supported for ai_socktype", |
| 254 | ); |
| 255 | } |
| 256 | |
| 257 | # The following functions are used if the system does not have a |
| 258 | # getaddrinfo(3) function in libc; and are used to emulate it for the AF_INET |
| 259 | # family |
| 260 | |
| 261 | # Borrowed from Regexp::Common::net |
| 262 | my $REGEXP_IPv4_DECIMAL = qr/25[0-5]|2[0-4][0-9]|[0-1]?[0-9]{1,2}/; |
| 263 | my $REGEXP_IPv4_DOTTEDQUAD = qr/$REGEXP_IPv4_DECIMAL\.$REGEXP_IPv4_DECIMAL\.$REGEXP_IPv4_DECIMAL\.$REGEXP_IPv4_DECIMAL/; |
| 264 | |
| 265 | sub fake_makeerr |
| 266 | { |
| 267 | my ( $errno ) = @_; |
| 268 | my $errstr = $errno == 0 ? "" : ( $errstr{$errno} || $errno ); |
| 269 | return Scalar::Util::dualvar( $errno, $errstr ); |
| 270 | } |
| 271 | |
| 272 | sub fake_getaddrinfo |
| 273 | { |
| 274 | my ( $node, $service, $hints ) = @_; |
| 275 | |
| 276 | $node = "" unless defined $node; |
| 277 | |
| 278 | $service = "" unless defined $service; |
| 279 | |
| 280 | my ( $family, $socktype, $protocol, $flags ) = @$hints{qw( family socktype protocol flags )}; |
| 281 | |
| 282 | $family ||= Socket::AF_INET(); # 0 == AF_UNSPEC, which we want too |
| 283 | $family == Socket::AF_INET() or return fake_makeerr( EAI_FAMILY() ); |
| 284 | |
| 285 | $socktype ||= 0; |
| 286 | |
| 287 | $protocol ||= 0; |
| 288 | |
| 289 | $flags ||= 0; |
| 290 | |
| 291 | my $flag_passive = $flags & AI_PASSIVE(); $flags &= ~AI_PASSIVE(); |
| 292 | my $flag_canonname = $flags & AI_CANONNAME(); $flags &= ~AI_CANONNAME(); |
| 293 | my $flag_numerichost = $flags & AI_NUMERICHOST(); $flags &= ~AI_NUMERICHOST(); |
| 294 | my $flag_numericserv = $flags & AI_NUMERICSERV(); $flags &= ~AI_NUMERICSERV(); |
| 295 | |
| 296 | # These constants don't apply to AF_INET-only lookups, so we might as well |
| 297 | # just ignore them. For AI_ADDRCONFIG we just presume the host has ability |
| 298 | # to talk AF_INET. If not we'd have to return no addresses at all. :) |
| 299 | $flags &= ~(AI_V4MAPPED()|AI_ALL()|AI_ADDRCONFIG()); |
| 300 | |
| 301 | $flags & (AI_IDN()|AI_CANONIDN()|AI_IDN_ALLOW_UNASSIGNED()|AI_IDN_USE_STD3_ASCII_RULES()) and |
| 302 | croak "Socket::getaddrinfo() does not support IDN"; |
| 303 | |
| 304 | $flags == 0 or return fake_makeerr( EAI_BADFLAGS() ); |
| 305 | |
| 306 | $node eq "" and $service eq "" and return fake_makeerr( EAI_NONAME() ); |
| 307 | |
| 308 | my $canonname; |
| 309 | my @addrs; |
| 310 | if( $node ne "" ) { |
| 311 | return fake_makeerr( EAI_NONAME() ) if( $flag_numerichost and $node !~ m/^$REGEXP_IPv4_DOTTEDQUAD$/ ); |
| 312 | ( $canonname, undef, undef, undef, @addrs ) = gethostbyname( $node ); |
| 313 | defined $canonname or return fake_makeerr( EAI_NONAME() ); |
| 314 | |
| 315 | undef $canonname unless $flag_canonname; |
| 316 | } |
| 317 | else { |
| 318 | $addrs[0] = $flag_passive ? Socket::inet_aton( "0.0.0.0" ) |
| 319 | : Socket::inet_aton( "127.0.0.1" ); |
| 320 | } |
| 321 | |
| 322 | my @ports; # Actually ARRAYrefs of [ socktype, protocol, port ] |
| 323 | my $protname = ""; |
| 324 | if( $protocol ) { |
| 325 | $protname = eval { getprotobynumber( $protocol ) }; |
| 326 | } |
| 327 | |
| 328 | if( $service ne "" and $service !~ m/^\d+$/ ) { |
| 329 | return fake_makeerr( EAI_NONAME() ) if( $flag_numericserv ); |
| 330 | getservbyname( $service, $protname ) or return fake_makeerr( EAI_SERVICE() ); |
| 331 | } |
| 332 | |
| 333 | foreach my $this_socktype ( Socket::SOCK_STREAM(), Socket::SOCK_DGRAM(), Socket::SOCK_RAW() ) { |
| 334 | next if $socktype and $this_socktype != $socktype; |
| 335 | |
| 336 | my $this_protname = "raw"; |
| 337 | $this_socktype == Socket::SOCK_STREAM() and $this_protname = "tcp"; |
| 338 | $this_socktype == Socket::SOCK_DGRAM() and $this_protname = "udp"; |
| 339 | |
| 340 | next if $protname and $this_protname ne $protname; |
| 341 | |
| 342 | my $port; |
| 343 | if( $service ne "" ) { |
| 344 | if( $service =~ m/^\d+$/ ) { |
| 345 | $port = "$service"; |
| 346 | } |
| 347 | else { |
| 348 | ( undef, undef, $port, $this_protname ) = getservbyname( $service, $this_protname ); |
| 349 | next unless defined $port; |
| 350 | } |
| 351 | } |
| 352 | else { |
| 353 | $port = 0; |
| 354 | } |
| 355 | |
| 356 | push @ports, [ $this_socktype, eval { scalar getprotobyname( $this_protname ) } || 0, $port ]; |
| 357 | } |
| 358 | |
| 359 | my @ret; |
| 360 | foreach my $addr ( @addrs ) { |
| 361 | foreach my $portspec ( @ports ) { |
| 362 | my ( $socktype, $protocol, $port ) = @$portspec; |
| 363 | push @ret, { |
| 364 | family => $family, |
| 365 | socktype => $socktype, |
| 366 | protocol => $protocol, |
| 367 | addr => Socket::pack_sockaddr_in( $port, $addr ), |
| 368 | canonname => undef, |
| 369 | }; |
| 370 | } |
| 371 | } |
| 372 | |
| 373 | # Only supply canonname for the first result |
| 374 | if( defined $canonname ) { |
| 375 | $ret[0]->{canonname} = $canonname; |
| 376 | } |
| 377 | |
| 378 | return ( fake_makeerr( 0 ), @ret ); |
| 379 | } |
| 380 | |
| 381 | sub fake_getnameinfo |
| 382 | { |
| 383 | my ( $addr, $flags, $xflags ) = @_; |
| 384 | |
| 385 | my ( $port, $inetaddr ); |
| 386 | eval { ( $port, $inetaddr ) = Socket::unpack_sockaddr_in( $addr ) } |
| 387 | or return fake_makeerr( EAI_FAMILY() ); |
| 388 | |
| 389 | my $family = Socket::AF_INET(); |
| 390 | |
| 391 | $flags ||= 0; |
| 392 | |
| 393 | my $flag_numerichost = $flags & NI_NUMERICHOST(); $flags &= ~NI_NUMERICHOST(); |
| 394 | my $flag_numericserv = $flags & NI_NUMERICSERV(); $flags &= ~NI_NUMERICSERV(); |
| 395 | my $flag_nofqdn = $flags & NI_NOFQDN(); $flags &= ~NI_NOFQDN(); |
| 396 | my $flag_namereqd = $flags & NI_NAMEREQD(); $flags &= ~NI_NAMEREQD(); |
| 397 | my $flag_dgram = $flags & NI_DGRAM() ; $flags &= ~NI_DGRAM(); |
| 398 | |
| 399 | $flags & (NI_IDN()|NI_IDN_ALLOW_UNASSIGNED()|NI_IDN_USE_STD3_ASCII_RULES()) and |
| 400 | croak "Socket::getnameinfo() does not support IDN"; |
| 401 | |
| 402 | $flags == 0 or return fake_makeerr( EAI_BADFLAGS() ); |
| 403 | |
| 404 | $xflags ||= 0; |
| 405 | |
| 406 | my $node; |
| 407 | if( $xflags & NIx_NOHOST ) { |
| 408 | $node = undef; |
| 409 | } |
| 410 | elsif( $flag_numerichost ) { |
| 411 | $node = Socket::inet_ntoa( $inetaddr ); |
| 412 | } |
| 413 | else { |
| 414 | $node = gethostbyaddr( $inetaddr, $family ); |
| 415 | if( !defined $node ) { |
| 416 | return fake_makeerr( EAI_NONAME() ) if $flag_namereqd; |
| 417 | $node = Socket::inet_ntoa( $inetaddr ); |
| 418 | } |
| 419 | elsif( $flag_nofqdn ) { |
| 420 | my ( $shortname ) = split m/\./, $node; |
| 421 | my ( $fqdn ) = gethostbyname $shortname; |
| 422 | $node = $shortname if defined $fqdn and $fqdn eq $node; |
| 423 | } |
| 424 | } |
| 425 | |
| 426 | my $service; |
| 427 | if( $xflags & NIx_NOSERV ) { |
| 428 | $service = undef; |
| 429 | } |
| 430 | elsif( $flag_numericserv ) { |
| 431 | $service = "$port"; |
| 432 | } |
| 433 | else { |
| 434 | my $protname = $flag_dgram ? "udp" : ""; |
| 435 | $service = getservbyport( $port, $protname ); |
| 436 | if( !defined $service ) { |
| 437 | $service = "$port"; |
| 438 | } |
| 439 | } |
| 440 | |
| 441 | return ( fake_makeerr( 0 ), $node, $service ); |
| 442 | } |
| 443 | |
| 444 | 1; |