hvsupport.pl 13.5 KB
Newer Older
1
#!/usr/bin/env perl
2 3 4 5 6 7 8 9 10 11 12 13

use strict;
use warnings;

use File::Find;

die "syntax: $0 SRCDIR\n" unless int(@ARGV) == 1;

my $srcdir = shift @ARGV;

my $symslibvirt = "$srcdir/libvirt_public.syms";
my $symsqemu = "$srcdir/libvirt_qemu.syms";
14
my $symslxc = "$srcdir/libvirt_lxc.syms";
15 16 17 18 19 20 21 22 23 24 25
my @drivertable = (
    "$srcdir/driver-hypervisor.h",
    "$srcdir/driver-interface.h",
    "$srcdir/driver-network.h",
    "$srcdir/driver-nodedev.h",
    "$srcdir/driver-nwfilter.h",
    "$srcdir/driver-secret.h",
    "$srcdir/driver-state.h",
    "$srcdir/driver-storage.h",
    "$srcdir/driver-stream.h",
    );
26 27

my %groupheaders = (
28
    "virHypervisorDriver" => "Hypervisor APIs",
29 30
    "virNetworkDriver" => "Virtual Network APIs",
    "virInterfaceDriver" => "Host Interface APIs",
31
    "virNodeDeviceDriver" => "Host Device APIs",
32 33 34 35 36 37 38 39 40
    "virStorageDriver" => "Storage Pool APIs",
    "virSecretDriver" => "Secret APIs",
    "virNWFilterDriver" => "Network Filter APIs",
    );


my @srcs;
find({
    wanted => sub {
41
        if (m!$srcdir/.*/\w+_(driver|common|tmpl|monitor|hal|udev)\.c$!) {
42 43
            push @srcs, $_ if $_ !~ /vbox_driver\.c/;
        }
44 45
    }, no_chdir => 1}, $srcdir);

46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71
# Map API functions to the header and documentation files they're in
# so that we can generate proper hyperlinks to their documentation.
#
# The function names are grep'd from the XML output of apibuild.py.
sub getAPIFilenames {
    my $filename = shift;

    my %files;
    my $line;

    open FILE, "<", $filename or die "cannot read $filename: $!";

    while (defined($line = <FILE>)) {
        if ($line =~ /function name='([^']+)' file='([^']+)'/) {
            $files{$1} = $2;
        }
    }

    close FILE;

    if (keys %files == 0) {
        die "No functions found in $filename. Has the apibuild.py output changed?";
    }
    return \%files;
}

J
Ján Tomko 已提交
72 73 74 75 76
sub parseSymsFile {
    my $apisref = shift;
    my $prefix = shift;
    my $filename = shift;
    my $xmlfilename = shift;
77

J
Ján Tomko 已提交
78 79 80
    my $line;
    my $vers;
    my $prevvers;
81

82
    my $filenames = getAPIFilenames($xmlfilename);
83

J
Ján Tomko 已提交
84 85
    open FILE, "<$filename"
        or die "cannot read $filename: $!";
86

J
Ján Tomko 已提交
87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111
    while (defined($line = <FILE>)) {
        chomp $line;
        next if $line =~ /^\s*#/;
        next if $line =~ /^\s*$/;
        next if $line =~ /^\s*(global|local):/;
        if ($line =~ /^\s*${prefix}_(\d+\.\d+\.\d+)\s*{\s*$/) {
            if (defined $vers) {
                die "malformed syms file";
            }
            $vers = $1;
        } elsif ($line =~ /\s*}\s*;\s*$/) {
            if (defined $prevvers) {
                die "malformed syms file";
            }
            $prevvers = $vers;
            $vers = undef;
        } elsif ($line =~ /\s*}\s*${prefix}_(\d+\.\d+\.\d+)\s*;\s*$/) {
            if ($1 ne $prevvers) {
                die "malformed syms file $1 != $vers";
            }
            $prevvers = $vers;
            $vers = undef;
        } elsif ($line =~ /\s*(\w+)\s*;\s*$/) {
            $$apisref{$1} = {};
            $$apisref{$1}->{vers} = $vers;
112
            $$apisref{$1}->{file} = $$filenames{$1};
J
Ján Tomko 已提交
113 114
        } else {
            die "unexpected data $line\n";
115
        }
116
    }
J
Ján Tomko 已提交
117 118

    close FILE;
119 120
}

J
Ján Tomko 已提交
121 122 123
my %apis;
# Get the list of all public APIs and their corresponding version
parseSymsFile(\%apis, "LIBVIRT", $symslibvirt, "$srcdir/../docs/libvirt-api.xml");
124

J
Ján Tomko 已提交
125 126
# And the same for the QEMU specific APIs
parseSymsFile(\%apis, "LIBVIRT_QEMU", $symsqemu, "$srcdir/../docs/libvirt-qemu-api.xml");
127 128

# And the same for the LXC specific APIs
J
Ján Tomko 已提交
129
parseSymsFile(\%apis, "LIBVIRT_LXC", $symslxc, "$srcdir/../docs/libvirt-lxc-api.xml");
130 131 132 133


# Some special things which aren't public APIs,
# but we want to report
134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154
$apis{virConnectSupportsFeature}->{vers} = "0.3.2";
$apis{virDomainMigratePrepare}->{vers} = "0.3.2";
$apis{virDomainMigratePerform}->{vers} = "0.3.2";
$apis{virDomainMigrateFinish}->{vers} = "0.3.2";
$apis{virDomainMigratePrepare2}->{vers} = "0.5.0";
$apis{virDomainMigrateFinish2}->{vers} = "0.5.0";
$apis{virDomainMigratePrepareTunnel}->{vers} = "0.7.2";

$apis{virDomainMigrateBegin3}->{vers} = "0.9.2";
$apis{virDomainMigratePrepare3}->{vers} = "0.9.2";
$apis{virDomainMigratePrepareTunnel3}->{vers} = "0.9.2";
$apis{virDomainMigratePerform3}->{vers} = "0.9.2";
$apis{virDomainMigrateFinish3}->{vers} = "0.9.2";
$apis{virDomainMigrateConfirm3}->{vers} = "0.9.2";

$apis{virDomainMigrateBegin3Params}->{vers} = "1.1.0";
$apis{virDomainMigratePrepare3Params}->{vers} = "1.1.0";
$apis{virDomainMigratePrepareTunnel3Params}->{vers} = "1.1.0";
$apis{virDomainMigratePerform3Params}->{vers} = "1.1.0";
$apis{virDomainMigrateFinish3Params}->{vers} = "1.1.0";
$apis{virDomainMigrateConfirm3Params}->{vers} = "1.1.0";
155

156 157 158 159 160 161


# Now we want to get the mapping between public APIs
# and driver struct fields. This lets us later match
# update the driver impls with the public APis.

J
Ján Tomko 已提交
162 163
my $line;

164 165 166
# Group name -> hash of APIs { fields -> api name }
my %groups;
my $ingrp;
167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186
foreach my $drivertable (@drivertable) {
    open FILE, "<$drivertable"
        or die "cannot read $drivertable: $!";

    while (defined($line = <FILE>)) {
        if ($line =~ /struct _(vir\w*Driver)/) {
            my $grp = $1;
            if ($grp ne "virStateDriver" &&
                $grp ne "virStreamDriver") {
                $ingrp = $grp;
                $groups{$ingrp} = { apis => {}, drivers => {} };
            }
        } elsif ($ingrp) {
            if ($line =~ /^\s*vir(?:Drv)(\w+)\s+(\w+);\s*$/) {
                my $field = $2;
                my $name = $1;

                my $api;
                if (exists $apis{"vir$name"}) {
                    $api = "vir$name";
187
                } elsif ($name =~ /\w+(Open|Close|URIProbe)/) {
188 189 190 191 192 193 194
                    next;
                } else {
                    die "driver $name does not have a public API";
                }
                $groups{$ingrp}->{apis}->{$field} = $api;
            } elsif ($line =~ /};/) {
                $ingrp = undef;
195 196
            }
        }
197 198
    }

199 200
    close FILE;
}
201 202 203 204 205 206 207


# Finally, we read all the primary driver files and extract
# the driver API tables from each one.

foreach my $src (@srcs) {
    open FILE, "<$src" or
208
        die "cannot read $src: $!";
209

210
    my $groups_regex = join("|", keys %groups);
211 212 213
    $ingrp = undef;
    my $impl;
    while (defined($line = <FILE>)) {
214
        if (!$ingrp) {
215 216 217
            # skip non-matching lines early to save time
            next if not $line =~ /$groups_regex/;

218 219 220 221 222 223 224 225 226
            if ($line =~ /^\s*(?:static\s+)?($groups_regex)\s+(\w+)\s*=\s*{/ ||
                $line =~ /^\s*(?:static\s+)?($groups_regex)\s+NAME\(\w+\)\s*=\s*{/) {
                $ingrp = $1;
                $impl = $src;

                if ($impl =~ m,.*/node_device_(\w+)\.c,) {
                    $impl = $1;
                } else {
                    $impl =~ s,.*/(\w+?)_((\w+)_)?(\w+)\.c,$1,;
227
                }
228 229 230 231 232 233

                if ($groups{$ingrp}->{drivers}->{$impl}) {
                    die "Group $ingrp already contains $impl";
                }

                $groups{$ingrp}->{drivers}->{$impl} = {};
234 235 236
            }

        } else {
237
            if ($line =~ m!\s*\.(\w+)\s*=\s*(\w+)\s*,?\s*(?:/\*\s*(\d+\.\d+\.\d+)\s*(?:\(deprecated:\s*(\d+\.\d+\.\d+)\))?\s*\*/\s*)?$!) {
238 239 240
                my $api = $1;
                my $meth = $2;
                my $vers = $3;
241
                my $depre = $4;
242 243 244

                next if $api eq "no" || $api eq "name";

245
                die "Method $meth in $src is missing version" unless defined $vers || $api eq "connectURIProbe";
246 247 248 249

                die "Driver method for $api is NULL in $src" if $meth eq "NULL";

                if (!exists($groups{$ingrp}->{apis}->{$api})) {
250
                    next if $api =~ /\w(Open|Close|URIProbe)/;
251

252 253 254
                    die "Found unexpected method $api in $ingrp\n";
                }

255 256 257
                $groups{$ingrp}->{drivers}->{$impl}->{$api} = {};
                $groups{$ingrp}->{drivers}->{$impl}->{$api}->{vers} = $vers;
                $groups{$ingrp}->{drivers}->{$impl}->{$api}->{depre}  = $depre;
258 259 260
                if ($api eq "domainMigratePrepare" ||
                    $api eq "domainMigratePrepare2" ||
                    $api eq "domainMigratePrepare3") {
261 262 263 264
                    if (!$groups{$ingrp}->{drivers}->{$impl}->{"domainMigrate"}) {
                        $groups{$ingrp}->{drivers}->{$impl}->{"domainMigrate"} = {};
                        $groups{$ingrp}->{drivers}->{$impl}->{"domainMigrate"}->{vers} = $vers;
                    }
265 266 267 268 269 270
                }

            } elsif ($line =~ /}/) {
                $ingrp = undef;
            }
        }
271 272 273 274 275 276 277 278 279 280
    }

    close FILE;
}


# The '.open' driver method is used for 3 public APIs, so we
# have a bit of manual fixup todo with the per-driver versioning
# and support matrix

281 282 283
$groups{virHypervisorDriver}->{apis}->{"openAuth"} = "virConnectOpenAuth";
$groups{virHypervisorDriver}->{apis}->{"openReadOnly"} = "virConnectOpenReadOnly";
$groups{virHypervisorDriver}->{apis}->{"domainMigrate"} = "virDomainMigrate";
284 285 286

my $openAuthVers = (0 * 1000 * 1000) + (4 * 1000) + 0;

287
foreach my $drv (keys %{$groups{"virHypervisorDriver"}->{drivers}}) {
288
    my $openVersStr = $groups{"virHypervisorDriver"}->{drivers}->{$drv}->{"connectOpen"}->{vers};
289 290
    my $openVers;
    if ($openVersStr =~ /(\d+)\.(\d+)\.(\d+)/) {
291
        $openVers = ($1 * 1000 * 1000) + ($2 * 1000) + $3;
292 293 294
    }

    # virConnectOpenReadOnly always matches virConnectOpen version
295 296
    $groups{"virHypervisorDriver"}->{drivers}->{$drv}->{"connectOpenReadOnly"} =
        $groups{"virHypervisorDriver"}->{drivers}->{$drv}->{"connectOpen"};
297

298 299
    $groups{"virHypervisorDriver"}->{drivers}->{$drv}->{"connectOpenAuth"} = {};

300 301 302 303
    # virConnectOpenAuth is always 0.4.0 if the driver existed
    # before this time, otherwise it matches the version of
    # the driver's virConnectOpen entry
    if ($openVersStr eq "Y" ||
304
        $openVers >= $openAuthVers) {
305
        $groups{"virHypervisorDriver"}->{drivers}->{$drv}->{"connectOpenAuth"}->{vers} = $openVersStr;
306
    } else {
307
        $groups{"virHypervisorDriver"}->{drivers}->{$drv}->{"connectOpenAuth"}->{vers} = "0.4.0";
308 309 310 311 312 313
    }
}


# Another special case for the virDomainCreateLinux which was replaced
# with virDomainCreateXML
314
$groups{virHypervisorDriver}->{apis}->{"domainCreateLinux"} = "virDomainCreateLinux";
315 316 317

my $createAPIVers = (0 * 1000 * 1000) + (0 * 1000) + 3;

318
foreach my $drv (keys %{$groups{"virHypervisorDriver"}->{drivers}}) {
319
    my $createVersStr = $groups{"virHypervisorDriver"}->{drivers}->{$drv}->{"domainCreateXML"}->{vers};
320 321 322
    next unless defined $createVersStr;
    my $createVers;
    if ($createVersStr =~ /(\d+)\.(\d+)\.(\d+)/) {
323
        $createVers = ($1 * 1000 * 1000) + ($2 * 1000) + $3;
324 325
    }

326 327
    $groups{"virHypervisorDriver"}->{drivers}->{$drv}->{"domainCreateLinux"} = {};

328 329 330 331
    # virCreateLinux is always 0.0.3 if the driver existed
    # before this time, otherwise it matches the version of
    # the driver's virCreateXML entry
    if ($createVersStr eq "Y" ||
332
        $createVers >= $createAPIVers) {
333
        $groups{"virHypervisorDriver"}->{drivers}->{$drv}->{"domainCreateLinux"}->{vers} = $createVersStr;
334
    } else {
335
        $groups{"virHypervisorDriver"}->{drivers}->{$drv}->{"domainCreateLinux"}->{vers} = "0.0.3";
336 337 338 339 340 341 342
    }
}


# Finally we generate the HTML file with the tables

print <<EOF;
343
<?xml version="1.0" encoding="UTF-8"?>
344
<!DOCTYPE html>
345
<html xmlns="http://www.w3.org/1999/xhtml">
346
<body class="hvsupport">
347 348 349 350 351 352 353
<h1>libvirt API support matrix</h1>

<ul id="toc"></ul>

<p>
This page documents which <a href="html/">libvirt calls</a> work on
which libvirt drivers / hypervisors, and which version the API appeared
354 355 356
in. If a hypervisor driver deprecated the API, the version when it
was removed is also mentioned (highlighted in
<span class="deprecatedhv">dark red</span>).
357 358 359 360
</p>

EOF

361
    foreach my $grp (sort { $a cmp $b } keys %groups) {
362 363 364 365 366 367 368 369 370 371
    print "<h2><a name=\"$grp\">", $groupheaders{$grp}, "</a></h2>\n";
    print <<EOF;
<table class="top_table">
<thead>
<tr>
<th>API</th>
<th>Version</th>
EOF

    foreach my $drv (sort { $a cmp $b } keys %{$groups{$grp}->{drivers}}) {
372
        print "  <th>$drv</th>\n";
373 374 375 376 377 378 379 380 381 382
    }

    print <<EOF;
</tr>
</thead>
<tbody>
EOF

    my $row = 0;
    foreach my $field (sort {
383 384 385 386 387
        $groups{$grp}->{apis}->{$a}
        cmp
        $groups{$grp}->{apis}->{$b}
        } keys %{$groups{$grp}->{apis}}) {
        my $api = $groups{$grp}->{apis}->{$field};
388 389
        my $vers = $apis{$api}->{vers};
        my $htmlgrp = $apis{$api}->{file};
390
        print <<EOF;
391
<tr>
392 393 394 395 396 397 398 399 400 401 402 403 404
<td>
EOF

        if (defined $htmlgrp) {
            print <<EOF;
<a href=\"html/libvirt-$htmlgrp.html#$api\">$api</a>
EOF

        } else {
            print $api;
        }
        print <<EOF;
</td>
405 406 407 408
<td>$vers</td>
EOF

        foreach my $drv (sort {$a cmp $b } keys %{$groups{$grp}->{drivers}}) {
409
            print "<td>";
410
            if (exists $groups{$grp}->{drivers}->{$drv}->{$field}) {
411 412 413 414 415 416
                if ($groups{$grp}->{drivers}->{$drv}->{$field}->{vers}) {
                    print $groups{$grp}->{drivers}->{$drv}->{$field}->{vers};
                }
                if ($groups{$grp}->{drivers}->{$drv}->{$field}->{depre}) {
                    print " - <span class=\"deprecatedhv\">", $groups{$grp}->{drivers}->{$drv}->{$field}->{depre}, "</span>";
                }
417
            }
418
            print "</td>\n";
419 420
        }

421
        print <<EOF;
422 423 424 425 426 427 428 429 430 431 432 433
</tr>
EOF

        $row++;
        if (($row % 15) == 0) {
            print <<EOF;
<tr>
<th>API</th>
<th>Version</th>
EOF

            foreach my $drv (sort { $a cmp $b } keys %{$groups{$grp}->{drivers}}) {
434
                print "  <th>$drv</th>\n";
435 436 437 438 439
            }

        print <<EOF;
</tr>
EOF
440
        }
441 442 443 444 445 446 447 448 449 450 451 452 453

    }

    print <<EOF;
</tbody>
</table>
EOF
}

print <<EOF;
</body>
</html>
EOF