302 lines
5.6 KiB
Perl
Executable File
302 lines
5.6 KiB
Perl
Executable File
#!/usr/bin/perl -w
|
|
|
|
use strict;
|
|
use File::Sync;
|
|
use Time::HiRes qw( usleep ualarm gettimeofday tv_interval );
|
|
use Net::DNS::Resolver;
|
|
|
|
if ($#ARGV >= 1) {
|
|
print STDERR "usage: $0 [PATH]\n";
|
|
exit -1;
|
|
}
|
|
|
|
my $path = $ARGV[0] || '/';
|
|
|
|
sub drop_cache {
|
|
|
|
# free pagecache,dentries,inode cache
|
|
if (-f '/proc/sys/vm/drop_caches') {
|
|
system ("echo 3 > /proc/sys/vm/drop_caches");
|
|
}
|
|
}
|
|
|
|
sub test_bogomips {
|
|
my $bogomips = 0;
|
|
|
|
open (TMP, "/proc/cpuinfo");
|
|
|
|
while (my $line = <TMP>) {
|
|
if ($line =~ m/^bogomips\s*:\s*(\d+\.\d+)\s*$/) {
|
|
$bogomips += $1;
|
|
}
|
|
}
|
|
|
|
close (TMP);
|
|
|
|
printf "CPU BOGOMIPS: %.2f\n", $bogomips;
|
|
}
|
|
|
|
sub test_regex {
|
|
|
|
my $starttime = [gettimeofday];
|
|
|
|
my $count = 0;
|
|
my $elapsed = 0;
|
|
|
|
for (;; $count++) {
|
|
|
|
my $str = int(rand(1000000)) . time();
|
|
if ($str =~ m/(.+)123.?123/) {
|
|
}
|
|
$elapsed = tv_interval ($starttime);
|
|
|
|
last if $elapsed > 3;
|
|
}
|
|
|
|
printf "REGEX/SECOND: %d\n", $count;
|
|
}
|
|
|
|
sub test_fsync {
|
|
my $basedir = shift;
|
|
|
|
drop_cache ();
|
|
|
|
my $dir = "$basedir/ptest.$$";
|
|
|
|
eval {
|
|
|
|
mkdir $dir;
|
|
|
|
my $data = ('A' x 4000) . "\n";
|
|
|
|
my $starttime = [gettimeofday];
|
|
|
|
my $count;
|
|
my $elapsed = 0;
|
|
|
|
for ($count=1;;$count++) {
|
|
my $m = $count % 300;
|
|
|
|
my $filename = "$dir/tf_$m.dat";
|
|
|
|
open (TMP, ">$filename") || die "open failed";
|
|
|
|
print TMP $data;
|
|
|
|
File::Sync::fsync (\*TMP);
|
|
|
|
close (TMP);
|
|
|
|
$elapsed = tv_interval ($starttime);
|
|
|
|
last if $elapsed > 3;
|
|
}
|
|
my $sps = $count /$elapsed; # fsync per second
|
|
|
|
printf "FSYNCS/SECOND: %.2f\n", $sps;
|
|
};
|
|
|
|
my $err = $@;
|
|
|
|
system ("rm -rf $dir");
|
|
|
|
die $err if $err;
|
|
}
|
|
|
|
sub test_seektime {
|
|
my ($rootdev, $hdsize) = @_;
|
|
|
|
drop_cache ();
|
|
|
|
open (ROOTHD, "<$rootdev") || die "unable to open HD";
|
|
|
|
my $starttime = [gettimeofday];
|
|
my $count;
|
|
my $elapsed = 0;
|
|
my $readbuf;
|
|
|
|
for ($count=1;;$count++) {
|
|
|
|
my $pos = int (rand (int($hdsize/512))) * 512;
|
|
|
|
sysseek (ROOTHD, $pos, 0);
|
|
|
|
(sysread (ROOTHD, $readbuf, 512) == 512) || die "read failed";
|
|
|
|
$elapsed = tv_interval ($starttime);
|
|
|
|
last if $elapsed > 3;
|
|
}
|
|
|
|
close (ROOTHD);
|
|
|
|
my $rps = $count /$elapsed; # blocks per second
|
|
my $ast = (1000/$rps);
|
|
printf "AVERAGE SEEK TIME: %.2f ms\n", $ast;
|
|
}
|
|
|
|
sub test_read {
|
|
my $rootdev = shift;
|
|
|
|
drop_cache ();
|
|
|
|
my $starttime = [gettimeofday];
|
|
my $bytes = 0;
|
|
my $elapsed = 0;
|
|
my $readbuf;
|
|
|
|
|
|
open (ROOTHD, "<$rootdev") || die "unable to open HD";
|
|
|
|
|
|
for (;;) {
|
|
|
|
my $c = sysread (ROOTHD, $readbuf, 2 * 1024 *1024);
|
|
die "read failed" if $c < 0;
|
|
|
|
$bytes += $c;
|
|
|
|
$elapsed = tv_interval ($starttime);
|
|
|
|
last if $elapsed > 3;
|
|
}
|
|
|
|
close (ROOTHD);
|
|
|
|
my $bps = $bytes /($elapsed * 1024 * 1024); # MB per second
|
|
printf "BUFFERED READS: %.2f MB/sec\n", $bps;
|
|
}
|
|
|
|
sub get_address {
|
|
my ($resolv, $dns) = @_;
|
|
|
|
if (my $a = $resolv->send ($dns, 'A')) {
|
|
foreach my $rra ($a->answer) {
|
|
if ($rra->type eq 'A') {
|
|
return $rra->address;
|
|
}
|
|
}
|
|
}
|
|
|
|
return undef;
|
|
}
|
|
|
|
sub test_dns {
|
|
|
|
my %dnsargs = (
|
|
tcp_timeout => 10,
|
|
udp_timeout => 10,
|
|
retry => 1,
|
|
retrans => 0,
|
|
dnsrch => 0,
|
|
defnames => 0,
|
|
debug => 0,
|
|
);
|
|
|
|
#$dnsargs{nameservers} = [ qw (208.67.222.222) ];
|
|
#$dnsargs{nameservers} = [ qw (127.0.0.1) ];
|
|
|
|
my $resolv = Net::DNS::Resolver->new (%dnsargs);
|
|
|
|
my $starttime = [gettimeofday];
|
|
|
|
my $count;
|
|
my $elapsed = 0;
|
|
|
|
my $uid = time() . int(rand(1000000));
|
|
my $domain = "nonexistent$uid.com";
|
|
|
|
for ($count=1;;$count++) {
|
|
|
|
my $hid = int(rand(1000000));
|
|
my $hname = "test${hid}.$domain";
|
|
get_address ($resolv, $hname);
|
|
$elapsed = tv_interval ($starttime);
|
|
|
|
last if ($count > 100) || ($elapsed > 3);
|
|
}
|
|
|
|
printf "DNS EXT: %0.2f ms\n", ($elapsed * 1000)/$count;
|
|
|
|
my $resolv_conf = `cat /etc/resolv.conf`;
|
|
($domain) = $resolv_conf =~ m/^search\s+(\S+)\s*$/mg;
|
|
|
|
if ($domain) {
|
|
$starttime = [gettimeofday];
|
|
$elapsed = 0;
|
|
|
|
for ($count=1;;$count++) {
|
|
|
|
my $hid = int(rand(1000000));
|
|
my $hname = "test${hid}.$domain";
|
|
get_address ($resolv, $hname);
|
|
$elapsed = tv_interval ($starttime);
|
|
|
|
last if ($count > 100) || ($elapsed > 3);
|
|
}
|
|
|
|
printf "DNS INT: %0.2f ms (%s)\n",
|
|
($elapsed * 1000)/ $count, $domain;
|
|
}
|
|
}
|
|
|
|
test_bogomips ();
|
|
test_regex ();
|
|
|
|
my $hd = `df -P '$path'`;
|
|
|
|
my ($rootdev, $hdo_total, $hdo_used, $hdo_avail) = $hd =~
|
|
m/^(\S+)\s+(\d+)\s+(\d+)\s+(\d+)\s+\S+\s+.*$/mg;
|
|
|
|
if ($rootdev) {
|
|
my $hdsize = $hdo_total*1024;
|
|
printf "HD SIZE: %.2f GB ($rootdev)\n", ($hdsize / (1024*1024*1024));
|
|
|
|
if ($rootdev =~ m|^/dev/|) {
|
|
test_read ($rootdev);
|
|
|
|
test_seektime ($rootdev, $hdsize);
|
|
|
|
}
|
|
}
|
|
|
|
test_fsync ($path) if $hdo_avail;
|
|
|
|
test_dns ();
|
|
|
|
exit (0);
|
|
|
|
__END__
|
|
|
|
=head1 NAME
|
|
|
|
pveperf - the Proxmox benchmark
|
|
|
|
=head1 SYNOPSIS
|
|
|
|
pveperf [PATH]
|
|
|
|
=head1 DESCRIPTION
|
|
|
|
Tries to gather some CPU/Hardisk performance data on the hardisk
|
|
mounted at PATH (/ is used as default)
|
|
|
|
* CPU BOGOMIPS: bogomips sum of all CPUs
|
|
|
|
* REGEX/SECOND: regular expressions per second (perl performance test), should be above 300000
|
|
|
|
* HD SIZE: harddisk size
|
|
|
|
* BUFFERED READS: simple HD read test. Modern HDs should reach
|
|
at least 40 MB/sec
|
|
|
|
* AVERAGE SEEK TIME: tests average seek time. Fast SCSI HDs reach values < 8 milliseconds. Common IDE/SATA disks get values from 15 to 20 ms.
|
|
|
|
* FSYNCS/SECOND: value should be greater than 200 (you should
|
|
enable 'write back' cache mode on you RAID controller - needs a battery
|
|
backed cache (BBWC)).
|
|
|
|
* DNS EXT: average time to resolve an external DNS name
|
|
|
|
* DNS INT: average time to resolve a local DNS name
|