2014-10-04 21:01:22 +02:00
|
|
|
|
package StartXServer;
|
2011-11-08 00:04:45 +01:00
|
|
|
|
# vim:ts=4:sw=4:expandtab
|
|
|
|
|
|
|
|
|
|
use strict;
|
|
|
|
|
use warnings;
|
|
|
|
|
use Exporter 'import';
|
|
|
|
|
use Time::HiRes qw(sleep);
|
|
|
|
|
use v5.10;
|
|
|
|
|
|
2014-10-02 01:26:52 +02:00
|
|
|
|
our @EXPORT = qw(start_xserver);
|
2011-11-08 00:04:45 +01:00
|
|
|
|
|
2012-08-18 16:27:00 +02:00
|
|
|
|
my @pids;
|
2011-12-24 15:34:28 +01:00
|
|
|
|
my $x_socketpath = '/tmp/.X11-unix/X';
|
|
|
|
|
|
2011-11-08 00:04:45 +01:00
|
|
|
|
# reads in a whole file
|
|
|
|
|
sub slurp {
|
|
|
|
|
open(my $fh, '<', shift) or return '';
|
|
|
|
|
local $/;
|
|
|
|
|
<$fh>;
|
|
|
|
|
}
|
|
|
|
|
|
2014-10-02 01:26:52 +02:00
|
|
|
|
# forks an X server process
|
2011-12-24 15:34:28 +01:00
|
|
|
|
sub fork_xserver {
|
2014-10-02 01:26:52 +02:00
|
|
|
|
my $keep_xserver_output = shift;
|
2011-12-24 15:34:28 +01:00
|
|
|
|
my $displaynum = shift;
|
|
|
|
|
my $pid = fork();
|
|
|
|
|
die "Could not fork: $!" unless defined($pid);
|
|
|
|
|
if ($pid == 0) {
|
2014-10-02 01:26:52 +02:00
|
|
|
|
# Child, close stdout/stderr, then start Xephyr
|
|
|
|
|
if (!$keep_xserver_output) {
|
2012-08-18 16:27:00 +02:00
|
|
|
|
close STDOUT;
|
|
|
|
|
close STDERR;
|
|
|
|
|
}
|
2011-12-24 15:34:28 +01:00
|
|
|
|
|
|
|
|
|
exec @_;
|
|
|
|
|
exit 1;
|
|
|
|
|
}
|
|
|
|
|
push(@complete_run::CLEANUP, sub {
|
|
|
|
|
kill(15, $pid);
|
|
|
|
|
# Unlink the X11 socket, Xdmx seems to leave it there.
|
|
|
|
|
unlink($x_socketpath . $displaynum);
|
|
|
|
|
});
|
|
|
|
|
|
2012-08-18 16:27:00 +02:00
|
|
|
|
push @pids, $pid;
|
|
|
|
|
|
2011-12-24 15:34:28 +01:00
|
|
|
|
return $x_socketpath . $displaynum;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
# Blocks until the socket paths specified in the given array reference actually
|
|
|
|
|
# exist.
|
|
|
|
|
sub wait_for_x {
|
|
|
|
|
my ($sockets_waiting) = @_;
|
|
|
|
|
|
|
|
|
|
# Wait until Xdmx actually runs. Pretty ugly solution, but as long as we
|
|
|
|
|
# can’t socket-activate X11…
|
|
|
|
|
while (1) {
|
|
|
|
|
@$sockets_waiting = grep { ! -S $_ } @$sockets_waiting;
|
|
|
|
|
last unless @$sockets_waiting;
|
|
|
|
|
sleep 0.1;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2014-10-02 01:26:52 +02:00
|
|
|
|
=head2 start_xserver($parallel)
|
2011-11-08 00:04:45 +01:00
|
|
|
|
|
2014-10-02 01:26:52 +02:00
|
|
|
|
Starts C<$parallel> (or number of cores * 2 if undef) Xephyr processes (see
|
|
|
|
|
http://www.freedesktop.org/wiki/Software/Xephyr/) and returns two arrayrefs: a
|
|
|
|
|
list of X11 display numbers to the Xephyr processes and a list of PIDs of the
|
|
|
|
|
processes.
|
2011-11-08 00:04:45 +01:00
|
|
|
|
|
|
|
|
|
=cut
|
2011-11-24 14:06:55 +01:00
|
|
|
|
|
2014-10-02 01:26:52 +02:00
|
|
|
|
sub start_xserver {
|
|
|
|
|
my ($parallel, $numtests, $keep_xserver_output) = @_;
|
2011-11-08 00:04:45 +01:00
|
|
|
|
|
|
|
|
|
my @displays = ();
|
|
|
|
|
my @childpids = ();
|
|
|
|
|
|
2012-08-18 16:27:00 +02:00
|
|
|
|
$SIG{CHLD} = sub {
|
|
|
|
|
my $child = waitpid -1, POSIX::WNOHANG;
|
|
|
|
|
@pids = grep { $_ != $child } @pids;
|
|
|
|
|
return unless @pids == 0;
|
2014-10-02 01:26:52 +02:00
|
|
|
|
print STDERR "All X server processes died.\n";
|
|
|
|
|
print STDERR "Use ./complete-run.pl --parallel 1 --keep-xserver-output\n";
|
2012-08-18 16:27:00 +02:00
|
|
|
|
exit 1;
|
|
|
|
|
};
|
|
|
|
|
|
2011-11-08 00:04:45 +01:00
|
|
|
|
# Yeah, I know it’s non-standard, but Perl’s POSIX module doesn’t have
|
|
|
|
|
# _SC_NPROCESSORS_CONF.
|
|
|
|
|
my $cpuinfo = slurp('/proc/cpuinfo');
|
|
|
|
|
my $num_cores = scalar grep { /model name/ } split("\n", $cpuinfo);
|
|
|
|
|
# If /proc/cpuinfo does not exist, we fall back to 2 cores.
|
|
|
|
|
$num_cores ||= 2;
|
|
|
|
|
|
2012-04-09 14:27:33 +02:00
|
|
|
|
# If unset, we use num_cores * 2.
|
|
|
|
|
$parallel ||= ($num_cores * 2);
|
2011-11-08 00:04:45 +01:00
|
|
|
|
|
2011-12-17 12:19:31 +01:00
|
|
|
|
# If we are running a small number of tests, don’t over-parallelize.
|
|
|
|
|
$parallel = $numtests if $numtests < $parallel;
|
|
|
|
|
|
2011-11-08 00:04:45 +01:00
|
|
|
|
# First get the last used display number, then increment it by one.
|
|
|
|
|
# Effectively falls back to 1 if no X server is running.
|
2011-11-24 14:06:55 +01:00
|
|
|
|
my ($displaynum) = map { /(\d+)$/ } reverse sort glob($x_socketpath . '*');
|
2011-11-08 00:04:45 +01:00
|
|
|
|
$displaynum++;
|
|
|
|
|
|
2014-10-02 01:26:52 +02:00
|
|
|
|
say "Starting $parallel Xephyr instances, starting at :$displaynum...";
|
2011-11-08 00:04:45 +01:00
|
|
|
|
|
2011-11-24 14:06:55 +01:00
|
|
|
|
my @sockets_waiting;
|
2011-12-24 15:34:28 +01:00
|
|
|
|
for (1 .. $parallel) {
|
2014-10-02 01:26:52 +02:00
|
|
|
|
my $socket = fork_xserver($keep_xserver_output, $displaynum,
|
|
|
|
|
'Xephyr', ":$displaynum", '-screen', '1280x800',
|
|
|
|
|
'-nolisten', 'tcp');
|
2011-11-08 00:04:45 +01:00
|
|
|
|
push(@displays, ":$displaynum");
|
2011-12-24 15:34:28 +01:00
|
|
|
|
push(@sockets_waiting, $socket);
|
2011-11-08 00:04:45 +01:00
|
|
|
|
$displaynum++;
|
|
|
|
|
}
|
|
|
|
|
|
2011-12-24 15:34:28 +01:00
|
|
|
|
wait_for_x(\@sockets_waiting);
|
|
|
|
|
|
2012-04-09 14:27:33 +02:00
|
|
|
|
return @displays;
|
2011-11-08 00:04:45 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
1
|