tests: Implement multi-monitor tests using Xdmx
This commit is contained in:
parent
8a67f15ef4
commit
dd9743b272
|
@ -69,8 +69,17 @@ my @testfiles = @ARGV;
|
||||||
|
|
||||||
my $numtests = scalar @testfiles;
|
my $numtests = scalar @testfiles;
|
||||||
|
|
||||||
|
# When the user specifies displays, we don’t run multi-monitor tests at all
|
||||||
|
# (because we don’t know which displaynumber is the X-Server with multiple
|
||||||
|
# monitors).
|
||||||
|
my $multidpy = undef;
|
||||||
|
|
||||||
# No displays specified, let’s start some Xdummy instances.
|
# No displays specified, let’s start some Xdummy instances.
|
||||||
@displays = start_xdummy($parallel, $numtests) if @displays == 0;
|
if (@displays == 0) {
|
||||||
|
my $dpyref;
|
||||||
|
($dpyref, $multidpy) = start_xdummy($parallel, $numtests);
|
||||||
|
@displays = @$dpyref;
|
||||||
|
}
|
||||||
|
|
||||||
# 1: create an output directory for this test-run
|
# 1: create an output directory for this test-run
|
||||||
my $outdir = "testsuite-";
|
my $outdir = "testsuite-";
|
||||||
|
@ -87,7 +96,7 @@ symlink("$outdir", "latest") or die "Could not symlink latest to $outdir";
|
||||||
# 2: keep the connection open so that i3 is not the only client. this prevents
|
# 2: keep the connection open so that i3 is not the only client. this prevents
|
||||||
# the X server from exiting (Xdummy will restart it, but not quick enough
|
# the X server from exiting (Xdummy will restart it, but not quick enough
|
||||||
# sometimes)
|
# sometimes)
|
||||||
my @worker;
|
my @single_worker;
|
||||||
for my $display (@displays) {
|
for my $display (@displays) {
|
||||||
my $screen;
|
my $screen;
|
||||||
my $x = X11::XCB::Connection->new(display => $display);
|
my $x = X11::XCB::Connection->new(display => $display);
|
||||||
|
@ -95,7 +104,17 @@ for my $display (@displays) {
|
||||||
die "Could not connect to display $display\n";
|
die "Could not connect to display $display\n";
|
||||||
} else {
|
} else {
|
||||||
# start a TestWorker for each display
|
# start a TestWorker for each display
|
||||||
push @worker, worker($display, $x, $outdir);
|
push @single_worker, worker($display, $x, $outdir);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
my @multi_worker;
|
||||||
|
if (defined($multidpy)) {
|
||||||
|
my $x = X11::XCB::Connection->new(display => $multidpy);
|
||||||
|
if ($x->has_error) {
|
||||||
|
die "Could not connect to multi-monitor display $multidpy\n";
|
||||||
|
} else {
|
||||||
|
push @multi_worker, worker($multidpy, $x, $outdir);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -127,18 +146,30 @@ my @done;
|
||||||
my $num = @testfiles;
|
my $num = @testfiles;
|
||||||
my $harness = TAP::Harness->new({ });
|
my $harness = TAP::Harness->new({ });
|
||||||
|
|
||||||
|
my @single_monitor_tests = grep { m,^t/([0-9]+)-, && $1 < 500 } @testfiles;
|
||||||
|
my @multi_monitor_tests = grep { m,^t/([0-9]+)-, && $1 >= 500 } @testfiles;
|
||||||
|
|
||||||
my $aggregator = TAP::Parser::Aggregator->new();
|
my $aggregator = TAP::Parser::Aggregator->new();
|
||||||
$aggregator->start();
|
$aggregator->start();
|
||||||
|
|
||||||
status_init(displays => \@displays, tests => $num);
|
status_init(displays => [ @displays, $multidpy ], tests => $num);
|
||||||
|
|
||||||
my $cv = AE::cv;
|
my $single_cv = AE::cv;
|
||||||
|
my $multi_cv = AE::cv;
|
||||||
|
|
||||||
# We start tests concurrently: For each display, one test gets started. Every
|
# We start tests concurrently: For each display, one test gets started. Every
|
||||||
# test starts another test after completing.
|
# test starts another test after completing.
|
||||||
for (@worker) { $cv->begin; take_job($_) }
|
for (@single_worker) {
|
||||||
|
$single_cv->begin;
|
||||||
|
take_job($_, $single_cv, \@single_monitor_tests);
|
||||||
|
}
|
||||||
|
for (@multi_worker) {
|
||||||
|
$multi_cv->begin;
|
||||||
|
take_job($_, $multi_cv, \@multi_monitor_tests);
|
||||||
|
}
|
||||||
|
|
||||||
$cv->recv;
|
$single_cv->recv;
|
||||||
|
$multi_cv->recv;
|
||||||
|
|
||||||
$aggregator->stop();
|
$aggregator->stop();
|
||||||
|
|
||||||
|
@ -198,9 +229,9 @@ exit 0;
|
||||||
# triggered to finish testing.
|
# triggered to finish testing.
|
||||||
#
|
#
|
||||||
sub take_job {
|
sub take_job {
|
||||||
my ($worker) = @_;
|
my ($worker, $cv, $tests) = @_;
|
||||||
|
|
||||||
my $test = shift @testfiles
|
my $test = shift @$tests
|
||||||
or return $cv->end;
|
or return $cv->end;
|
||||||
|
|
||||||
my $display = $worker->{display};
|
my $display = $worker->{display};
|
||||||
|
@ -269,7 +300,7 @@ sub take_job {
|
||||||
push @done, [ $test, $output ];
|
push @done, [ $test, $output ];
|
||||||
|
|
||||||
undef $w;
|
undef $w;
|
||||||
take_job($worker);
|
take_job($worker, $cv, $tests);
|
||||||
}
|
}
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
|
@ -85,8 +85,10 @@ sub activate_i3 {
|
||||||
|
|
||||||
# Construct the command to launch i3. Use maximum debug level, disable
|
# Construct the command to launch i3. Use maximum debug level, disable
|
||||||
# the interactive signalhandler to make it crash immediately instead.
|
# the interactive signalhandler to make it crash immediately instead.
|
||||||
# Also disable logging to SHM since we want to redirect the logs anyways.
|
# Also disable logging to SHM since we redirect the logs anyways.
|
||||||
my $i3cmd = abs_path("../i3") . " -V -d all --disable-signalhandler --shmlog-size=0";
|
# Force Xinerama because we use Xdmx for multi-monitor tests.
|
||||||
|
my $i3cmd = abs_path("../i3") . q| -V -d all --disable-signalhandler| .
|
||||||
|
q| --shmlog-size=0 --force-xinerama|;
|
||||||
|
|
||||||
# For convenience:
|
# For convenience:
|
||||||
my $outdir = $args{outdir};
|
my $outdir = $args{outdir};
|
||||||
|
|
|
@ -9,6 +9,8 @@ use v5.10;
|
||||||
|
|
||||||
our @EXPORT = qw(start_xdummy);
|
our @EXPORT = qw(start_xdummy);
|
||||||
|
|
||||||
|
my $x_socketpath = '/tmp/.X11-unix/X';
|
||||||
|
|
||||||
# reads in a whole file
|
# reads in a whole file
|
||||||
sub slurp {
|
sub slurp {
|
||||||
open(my $fh, '<', shift) or return '';
|
open(my $fh, '<', shift) or return '';
|
||||||
|
@ -16,6 +18,42 @@ sub slurp {
|
||||||
<$fh>;
|
<$fh>;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
# forks an Xdummy or Xdmx process
|
||||||
|
sub fork_xserver {
|
||||||
|
my $displaynum = shift;
|
||||||
|
my $pid = fork();
|
||||||
|
die "Could not fork: $!" unless defined($pid);
|
||||||
|
if ($pid == 0) {
|
||||||
|
# Child, close stdout/stderr, then start Xdummy.
|
||||||
|
close STDOUT;
|
||||||
|
close STDERR;
|
||||||
|
|
||||||
|
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);
|
||||||
|
});
|
||||||
|
|
||||||
|
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;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
=head2 start_xdummy($parallel)
|
=head2 start_xdummy($parallel)
|
||||||
|
|
||||||
Starts C<$parallel> (or number of cores * 2 if undef) Xdummy processes (see
|
Starts C<$parallel> (or number of cores * 2 if undef) Xdummy processes (see
|
||||||
|
@ -24,8 +62,6 @@ the Xdummy processes and a list of PIDs of the processes.
|
||||||
|
|
||||||
=cut
|
=cut
|
||||||
|
|
||||||
my $x_socketpath = '/tmp/.X11-unix/X';
|
|
||||||
|
|
||||||
sub start_xdummy {
|
sub start_xdummy {
|
||||||
my ($parallel, $numtests) = @_;
|
my ($parallel, $numtests) = @_;
|
||||||
|
|
||||||
|
@ -39,11 +75,17 @@ sub start_xdummy {
|
||||||
# If /proc/cpuinfo does not exist, we fall back to 2 cores.
|
# If /proc/cpuinfo does not exist, we fall back to 2 cores.
|
||||||
$num_cores ||= 2;
|
$num_cores ||= 2;
|
||||||
|
|
||||||
$parallel ||= $num_cores * 2;
|
# If unset, we use num_cores * 2, plus two extra xdummys to combine to a
|
||||||
|
# multi-monitor setup using Xdmx.
|
||||||
|
$parallel ||= ($num_cores * 2) + 2;
|
||||||
|
|
||||||
# If we are running a small number of tests, don’t over-parallelize.
|
# If we are running a small number of tests, don’t over-parallelize.
|
||||||
$parallel = $numtests if $numtests < $parallel;
|
$parallel = $numtests if $numtests < $parallel;
|
||||||
|
|
||||||
|
# Ensure we have at least 1 X-Server plus two X-Servers for multi-monitor
|
||||||
|
# tests.
|
||||||
|
$parallel = 3 if $parallel < 3;
|
||||||
|
|
||||||
# First get the last used display number, then increment it by one.
|
# First get the last used display number, then increment it by one.
|
||||||
# Effectively falls back to 1 if no X server is running.
|
# Effectively falls back to 1 if no X server is running.
|
||||||
my ($displaynum) = map { /(\d+)$/ } reverse sort glob($x_socketpath . '*');
|
my ($displaynum) = map { /(\d+)$/ } reverse sort glob($x_socketpath . '*');
|
||||||
|
@ -52,37 +94,33 @@ sub start_xdummy {
|
||||||
say "Starting $parallel Xdummy instances, starting at :$displaynum...";
|
say "Starting $parallel Xdummy instances, starting at :$displaynum...";
|
||||||
|
|
||||||
my @sockets_waiting;
|
my @sockets_waiting;
|
||||||
for my $idx (0 .. ($parallel-1)) {
|
for (1 .. $parallel) {
|
||||||
my $pid = fork();
|
|
||||||
die "Could not fork: $!" unless defined($pid);
|
|
||||||
if ($pid == 0) {
|
|
||||||
# Child, close stdout/stderr, then start Xdummy.
|
|
||||||
close STDOUT;
|
|
||||||
close STDERR;
|
|
||||||
# make sure this display isn’t in use yet
|
|
||||||
$displaynum++ while -e ($x_socketpath . $displaynum);
|
|
||||||
|
|
||||||
# We use -config /dev/null to prevent Xdummy from using the system
|
# We use -config /dev/null to prevent Xdummy from using the system
|
||||||
# Xorg configuration. The tests should be independant from the
|
# Xorg configuration. The tests should be independant from the
|
||||||
# actual system X configuration.
|
# actual system X configuration.
|
||||||
exec './Xdummy', ":$displaynum", '-config', '/dev/null';
|
my $socket = fork_xserver($displaynum, './Xdummy', ":$displaynum",
|
||||||
exit 1;
|
'-config', '/dev/null');
|
||||||
}
|
|
||||||
push(@complete_run::CLEANUP, sub { kill(15, $pid) });
|
|
||||||
push(@displays, ":$displaynum");
|
push(@displays, ":$displaynum");
|
||||||
push(@sockets_waiting, $x_socketpath . $displaynum);
|
push(@sockets_waiting, $socket);
|
||||||
$displaynum++;
|
$displaynum++;
|
||||||
}
|
}
|
||||||
|
|
||||||
# Wait until the X11 sockets actually appear. Pretty ugly solution, but as
|
wait_for_x(\@sockets_waiting);
|
||||||
# long as we can’t socket-activate X11…
|
|
||||||
while (1) {
|
|
||||||
@sockets_waiting = grep { ! -S $_ } @sockets_waiting;
|
|
||||||
last unless @sockets_waiting;
|
|
||||||
sleep 0.1;
|
|
||||||
}
|
|
||||||
|
|
||||||
return @displays;
|
# Now combine the last two displays to a multi-monitor display using Xdmx
|
||||||
|
my $first = pop @displays;
|
||||||
|
my $second = pop @displays;
|
||||||
|
|
||||||
|
# make sure this display isn’t in use yet
|
||||||
|
$displaynum++ while -e ($x_socketpath . $displaynum);
|
||||||
|
say 'starting xdmx on display :' . $displaynum;
|
||||||
|
|
||||||
|
my $multidpy = ":$displaynum";
|
||||||
|
my $socket = fork_xserver($displaynum, 'Xdmx', '+xinerama', '-xinput',
|
||||||
|
'local', '-display', $first, '-display', $second, '-ac', $multidpy);
|
||||||
|
wait_for_x([ $socket ]);
|
||||||
|
|
||||||
|
return \@displays, $multidpy;
|
||||||
}
|
}
|
||||||
|
|
||||||
1
|
1
|
||||||
|
|
|
@ -0,0 +1,21 @@
|
||||||
|
#!perl
|
||||||
|
# vim:ts=4:sw=4:expandtab
|
||||||
|
#
|
||||||
|
# Tests that the provided X-Server to the t/5??-*.t tests is actually providing
|
||||||
|
# multiple monitors.
|
||||||
|
#
|
||||||
|
use i3test;
|
||||||
|
|
||||||
|
my $i3 = i3(get_socket_path());
|
||||||
|
|
||||||
|
####################
|
||||||
|
# Request tree
|
||||||
|
####################
|
||||||
|
|
||||||
|
my $tree = $i3->get_tree->recv;
|
||||||
|
|
||||||
|
my @outputs = map { $_->{name} } @{$tree->{nodes}};
|
||||||
|
is_deeply(\@outputs, [ '__i3', 'xinerama-0', 'xinerama-1' ],
|
||||||
|
'multi-monitor outputs ok');
|
||||||
|
|
||||||
|
done_testing;
|
Loading…
Reference in New Issue