}
sub next_unused_port {
- my ($range_start, $range_end) = @_;
+ my ($range_start, $range_end, $family) = @_;
# We use a file to register allocated ports.
# Those registrations expires after $expiretime.
next if $ports->{$p}; # reserved
my $sock = IO::Socket::IP->new(Listen => 5,
- LocalAddr => '0.0.0.0',
LocalPort => $p,
ReuseAddr => 1,
- Proto => 0);
+ Family => $family,
+ Proto => 0,
+ GetAddrInfoFlags => 0);
if ($sock) {
close($sock);
}
sub next_migrate_port {
- return next_unused_port(60000, 60050);
+ my ($family) = @_;
+ return next_unused_port(60000, 60050, $family);
}
sub next_vnc_port {
- return next_unused_port(5900, 6000);
+ my ($family) = @_;
+ return next_unused_port(5900, 6000, $family);
}
sub next_spice_port {
- return next_unused_port(61000, 61099);
+ my ($family) = @_;
+ return next_unused_port(61000, 61099, $family);
}
# NOTE: NFS syscall can't be interrupted, so alarm does
return ($family, $port, $host);
}
-sub get_host_address_family {
- my ($hostname, $socktype) = @_;
+sub getaddrinfo_all {
+ my ($hostname, @opts) = @_;
my %hints = ( flags => AI_V4MAPPED | AI_ALL,
- socktype => $socktype );
+ @opts );
my ($err, @res) = Socket::getaddrinfo($hostname, '0', \%hints);
- die "failed to resolve $hostname: $err\n" if $err;
+ die "failed to get address info for: $hostname: $err\n" if $err;
+ return @res;
+}
- return ${res[0]}->{family};
+sub get_host_address_family {
+ my ($hostname, $socktype) = @_;
+ my @res = getaddrinfo_all($hostname, socktype => $socktype);
+ return $res[0]->{family};
}
1;