#!/usr/bin/perl 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"; my $symslxc = "$srcdir/libvirt_lxc.syms"; 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", ); my %groupheaders = ( "virHypervisorDriver" => "Hypervisor APIs", "virNetworkDriver" => "Virtual Network APIs", "virInterfaceDriver" => "Host Interface APIs", "virNodeDeviceDriver" => "Host Device APIs", "virStorageDriver" => "Storage Pool APIs", "virSecretDriver" => "Secret APIs", "virNWFilterDriver" => "Network Filter APIs", ); my @srcs; find({ wanted => sub { if (m!$srcdir/.*/\w+_(driver|common|tmpl|monitor|hal|udev)\.c$!) { push @srcs, $_ if $_ !~ /vbox_driver\.c/; } }, no_chdir => 1}, $srcdir); my $line; # Get the list of all public APIs and their corresponding version my %apis; open FILE, "<$symslibvirt" or die "cannot read $symslibvirt: $!"; my $vers; my $prevvers; while (defined($line = )) { chomp $line; next if $line =~ /^\s*#/; next if $line =~ /^\s*$/; next if $line =~ /^\s*(global|local):/; if ($line =~ /^\s*LIBVIRT_(\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*LIBVIRT_(\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*$/) { $apis{$1} = $vers; } else { die "unexpected data $line\n"; } } close FILE; # And the same for the QEMU specific APIs open FILE, "<$symsqemu" or die "cannot read $symsqemu: $!"; $prevvers = undef; $vers = undef; while (defined($line = )) { chomp $line; next if $line =~ /^\s*#/; next if $line =~ /^\s*$/; next if $line =~ /^\s*(global|local):/; if ($line =~ /^\s*LIBVIRT_QEMU_(\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*LIBVIRT_QEMU_(\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*$/) { $apis{$1} = $vers; } else { die "unexpected data $line\n"; } } close FILE; # And the same for the LXC specific APIs open FILE, "<$symslxc" or die "cannot read $symslxc: $!"; $prevvers = undef; $vers = undef; while (defined($line = )) { chomp $line; next if $line =~ /^\s*#/; next if $line =~ /^\s*$/; next if $line =~ /^\s*(global|local):/; if ($line =~ /^\s*LIBVIRT_LXC_(\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*LIBVIRT_LXC_(\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*$/) { $apis{$1} = $vers; } else { die "unexpected data $line\n"; } } close FILE; # Some special things which aren't public APIs, # but we want to report $apis{virConnectSupportsFeature} = "0.3.2"; $apis{virDomainMigratePrepare} = "0.3.2"; $apis{virDomainMigratePerform} = "0.3.2"; $apis{virDomainMigrateFinish} = "0.3.2"; $apis{virDomainMigratePrepare2} = "0.5.0"; $apis{virDomainMigrateFinish2} = "0.5.0"; $apis{virDomainMigratePrepareTunnel} = "0.7.2"; $apis{virDomainMigrateBegin3} = "0.9.2"; $apis{virDomainMigratePrepare3} = "0.9.2"; $apis{virDomainMigratePrepareTunnel3} = "0.9.2"; $apis{virDomainMigratePerform3} = "0.9.2"; $apis{virDomainMigrateFinish3} = "0.9.2"; $apis{virDomainMigrateConfirm3} = "0.9.2"; $apis{virDomainMigrateBegin3Params} = "1.1.0"; $apis{virDomainMigratePrepare3Params} = "1.1.0"; $apis{virDomainMigratePrepareTunnel3Params} = "1.1.0"; $apis{virDomainMigratePerform3Params} = "1.1.0"; $apis{virDomainMigrateFinish3Params} = "1.1.0"; $apis{virDomainMigrateConfirm3Params} = "1.1.0"; # 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. # Group name -> hash of APIs { fields -> api name } my %groups; my $ingrp; foreach my $drivertable (@drivertable) { open FILE, "<$drivertable" or die "cannot read $drivertable: $!"; while (defined($line = )) { 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"; } elsif ($name =~ /\w+(Open|Close)/) { next; } else { die "driver $name does not have a public API"; } $groups{$ingrp}->{apis}->{$field} = $api; } elsif ($line =~ /};/) { $ingrp = undef; } } } close FILE; } # 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 die "cannot read $src: $!"; $ingrp = undef; my $impl; while (defined($line = )) { if (!$ingrp) { foreach my $grp (keys %groups) { if ($line =~ /^\s*(?:static\s+)?$grp\s+(\w+)\s*=\s*{/ || $line =~ /^\s*(?:static\s+)?$grp\s+NAME\(\w+\)\s*=\s*{/) { $ingrp = $grp; $impl = $src; if ($impl =~ m,.*/node_device_(\w+)\.c,) { $impl = $1; } else { $impl =~ s,.*/(\w+?)_((\w+)_)?(\w+)\.c,$1,; } if ($groups{$ingrp}->{drivers}->{$impl}) { die "Group $ingrp already contains $impl"; } $groups{$ingrp}->{drivers}->{$impl} = {}; } } } else { if ($line =~ m!\s*\.(\w+)\s*=\s*(\w+)\s*,?\s*(?:/\*\s*(\d+\.\d+\.\d+)\s*\*/\s*)?$!) { my $api = $1; my $meth = $2; my $vers = $3; next if $api eq "no" || $api eq "name"; die "Method $meth in $src is missing version" unless defined $vers; die "Driver method for $api is NULL in $src" if $meth eq "NULL"; if (!exists($groups{$ingrp}->{apis}->{$api})) { next if $api =~ /\w(Open|Close)/; die "Found unexpected method $api in $ingrp\n"; } $groups{$ingrp}->{drivers}->{$impl}->{$api} = $vers; if ($api eq "domainMigratePrepare" || $api eq "domainMigratePrepare2" || $api eq "domainMigratePrepare3") { $groups{$ingrp}->{drivers}->{$impl}->{"domainMigrate"} = $vers unless $groups{$ingrp}->{drivers}->{$impl}->{"domainMigrate"}; } } elsif ($line =~ /}/) { $ingrp = undef; } } } 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 $groups{virHypervisorDriver}->{apis}->{"openAuth"} = "virConnectOpenAuth"; $groups{virHypervisorDriver}->{apis}->{"openReadOnly"} = "virConnectOpenReadOnly"; $groups{virHypervisorDriver}->{apis}->{"domainMigrate"} = "virDomainMigrate"; my $openAuthVers = (0 * 1000 * 1000) + (4 * 1000) + 0; foreach my $drv (keys %{$groups{"virHypervisorDriver"}->{drivers}}) { my $openVersStr = $groups{"virHypervisorDriver"}->{drivers}->{$drv}->{"connectOpen"}; my $openVers; if ($openVersStr =~ /(\d+)\.(\d+)\.(\d+)/) { $openVers = ($1 * 1000 * 1000) + ($2 * 1000) + $3; } # virConnectOpenReadOnly always matches virConnectOpen version $groups{"virHypervisorDriver"}->{drivers}->{$drv}->{"connectOpenReadOnly"} = $groups{"virHypervisorDriver"}->{drivers}->{$drv}->{"connectOpen"}; # 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" || $openVers >= $openAuthVers) { $groups{"virHypervisorDriver"}->{drivers}->{$drv}->{"connectOpenAuth"} = $openVersStr; } else { $groups{"virHypervisorDriver"}->{drivers}->{$drv}->{"connectOpenAuth"} = "0.4.0"; } } # Another special case for the virDomainCreateLinux which was replaced # with virDomainCreateXML $groups{virHypervisorDriver}->{apis}->{"domainCreateLinux"} = "virDomainCreateLinux"; my $createAPIVers = (0 * 1000 * 1000) + (0 * 1000) + 3; foreach my $drv (keys %{$groups{"virHypervisorDriver"}->{drivers}}) { my $createVersStr = $groups{"virHypervisorDriver"}->{drivers}->{$drv}->{"domainCreateXML"}; next unless defined $createVersStr; my $createVers; if ($createVersStr =~ /(\d+)\.(\d+)\.(\d+)/) { $createVers = ($1 * 1000 * 1000) + ($2 * 1000) + $3; } # 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" || $createVers >= $createAPIVers) { $groups{"virHypervisorDriver"}->{drivers}->{$drv}->{"domainCreateLinux"} = $createVersStr; } else { $groups{"virHypervisorDriver"}->{drivers}->{$drv}->{"domainCreateLinux"} = "0.0.3"; } } # Finally we generate the HTML file with the tables print < libvirt API support matrix

libvirt API support matrix

    This page documents which libvirt calls work on which libvirt drivers / hypervisors, and which version the API appeared in.

    EOF foreach my $grp (sort { $a cmp $b } keys %groups) { print "

    ", $groupheaders{$grp}, "

    \n"; print < API Version EOF foreach my $drv (sort { $a cmp $b } keys %{$groups{$grp}->{drivers}}) { print " $drv\n"; } print < EOF my $row = 0; foreach my $field (sort { $groups{$grp}->{apis}->{$a} cmp $groups{$grp}->{apis}->{$b} } keys %{$groups{$grp}->{apis}}) { my $api = $groups{$grp}->{apis}->{$field}; my $vers = $apis{$api}; print < $api $vers EOF foreach my $drv (sort {$a cmp $b } keys %{$groups{$grp}->{drivers}}) { if (exists $groups{$grp}->{drivers}->{$drv}->{$field}) { print "", $groups{$grp}->{drivers}->{$drv}->{$field}, "\n"; } else { print "\n"; } } print < EOF $row++; if (($row % 15) == 0) { print < API Version EOF foreach my $drv (sort { $a cmp $b } keys %{$groups{$grp}->{drivers}}) { print " $drv\n"; } print < EOF } } print < EOF } print < EOF