2014-06-25 15:24:53 +04:00
/*
* Copyright ( C ) Red Hat , Inc . 2014
*
* This library is free software ; you can redistribute it and / or
* modify it under the terms of the GNU Lesser General Public
* License as published by the Free Software Foundation ; either
* version 2.1 of the License , or ( at your option ) any later version .
*
* This library is distributed in the hope that it will be useful ,
* but WITHOUT ANY WARRANTY ; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the GNU
* Lesser General Public License for more details .
*
* You should have received a copy of the GNU Lesser General Public
* License along with this library . If not , see
* < http : //www.gnu.org/licenses/>.
*
* Authors :
* Michal Privoznik < mprivozn @ redhat . com >
*/
# include <config.h>
# include <stdlib.h>
# include "testutils.h"
# include "domain_capabilities.h"
# define VIR_FROM_THIS VIR_FROM_NONE
2014-09-17 13:33:35 +04:00
typedef int ( * virDomainCapsFill ) ( virDomainCapsPtr domCaps ,
void * opaque ) ;
2014-06-25 15:24:53 +04:00
# define SET_ALL_BITS(x) \
memset ( & ( x . values ) , 0xff , sizeof ( x . values ) )
2014-09-17 03:52:54 +04:00
static int ATTRIBUTE_SENTINEL
fillStringValues ( virDomainCapsStringValuesPtr values , . . . )
{
int ret = 0 ;
va_list list ;
const char * str ;
va_start ( list , values ) ;
while ( ( str = va_arg ( list , const char * ) ) ) {
if ( VIR_REALLOC_N ( values - > values , values - > nvalues + 1 ) < 0 | |
VIR_STRDUP ( values - > values [ values - > nvalues ] , str ) < 0 ) {
ret = - 1 ;
break ;
}
values - > nvalues + + ;
}
va_end ( list ) ;
return ret ;
}
2014-09-17 13:33:35 +04:00
static int
2014-06-25 15:24:53 +04:00
fillAll ( virDomainCapsPtr domCaps ,
void * opaque ATTRIBUTE_UNUSED )
{
2014-09-16 16:47:47 +04:00
virDomainCapsOSPtr os = & domCaps - > os ;
virDomainCapsLoaderPtr loader = & os - > loader ;
2014-06-25 15:24:53 +04:00
virDomainCapsDeviceDiskPtr disk = & domCaps - > disk ;
virDomainCapsDeviceHostdevPtr hostdev = & domCaps - > hostdev ;
domCaps - > maxvcpus = 255 ;
2014-09-16 16:47:47 +04:00
os - > device . supported = true ;
loader - > device . supported = true ;
SET_ALL_BITS ( loader - > type ) ;
SET_ALL_BITS ( loader - > readonly ) ;
2014-09-17 03:52:54 +04:00
if ( fillStringValues ( & loader - > values ,
" /foo/bar " ,
" /tmp/my_path " ,
NULL ) < 0 )
return - 1 ;
2014-09-16 16:47:47 +04:00
2014-06-25 15:24:53 +04:00
disk - > device . supported = true ;
SET_ALL_BITS ( disk - > diskDevice ) ;
SET_ALL_BITS ( disk - > bus ) ;
hostdev - > device . supported = true ;
SET_ALL_BITS ( hostdev - > mode ) ;
SET_ALL_BITS ( hostdev - > startupPolicy ) ;
SET_ALL_BITS ( hostdev - > subsysType ) ;
SET_ALL_BITS ( hostdev - > capsType ) ;
SET_ALL_BITS ( hostdev - > pciBackend ) ;
2014-09-17 13:33:35 +04:00
return 0 ;
2014-06-25 15:24:53 +04:00
}
2014-06-25 20:39:29 +04:00
# ifdef WITH_QEMU
# include "testutilsqemu.h"
2014-09-17 03:52:54 +04:00
struct fillQemuCapsData {
virQEMUCapsPtr qemuCaps ;
virQEMUDriverConfigPtr cfg ;
} ;
2014-09-17 13:33:35 +04:00
static int
2014-06-25 20:39:29 +04:00
fillQemuCaps ( virDomainCapsPtr domCaps ,
void * opaque )
{
2014-09-17 03:52:54 +04:00
struct fillQemuCapsData * data = ( struct fillQemuCapsData * ) opaque ;
virQEMUCapsPtr qemuCaps = data - > qemuCaps ;
virQEMUDriverConfigPtr cfg = data - > cfg ;
2014-06-25 20:39:29 +04:00
2014-09-18 10:30:40 +04:00
if ( virQEMUCapsFillDomainCaps ( domCaps , qemuCaps ,
cfg - > loader , cfg - > nloader ) < 0 )
2014-09-17 13:33:35 +04:00
return - 1 ;
2014-06-25 20:39:29 +04:00
/* The function above tries to query host's KVM & VFIO capabilities by
* calling qemuHostdevHostSupportsPassthroughLegacy ( ) and
* qemuHostdevHostSupportsPassthroughVFIO ( ) which , however , can ' t be
* successfully mocked as they are not exposed as internal APIs . Therefore ,
* instead of mocking set the expected values here by hand . */
VIR_DOMAIN_CAPS_ENUM_SET ( domCaps - > hostdev . pciBackend ,
VIR_DOMAIN_HOSTDEV_PCI_BACKEND_DEFAULT ,
VIR_DOMAIN_HOSTDEV_PCI_BACKEND_KVM ,
VIR_DOMAIN_HOSTDEV_PCI_BACKEND_VFIO ) ;
2014-09-17 19:17:03 +04:00
/* Moreover, as of f05b6a918e28 we are expecting to see
* OVMF_CODE . fd file which may not exists everywhere . */
if ( ! domCaps - > os . loader . values . nvalues ) {
virDomainCapsLoaderPtr loader = & domCaps - > os . loader ;
if ( fillStringValues ( & loader - > values ,
" /usr/share/OVMF/OVMF_CODE.fd " ,
NULL ) < 0 )
return - 1 ;
}
2014-09-17 13:33:35 +04:00
return 0 ;
2014-06-25 20:39:29 +04:00
}
# endif /* WITH_QEMU */
2014-06-25 15:24:53 +04:00
static virDomainCapsPtr
buildVirDomainCaps ( const char * emulatorbin ,
const char * machine ,
virArch arch ,
virDomainVirtType type ,
virDomainCapsFill fillFunc ,
void * opaque )
{
2014-09-17 03:52:54 +04:00
virDomainCapsPtr domCaps , ret = NULL ;
2014-06-25 15:24:53 +04:00
if ( ! ( domCaps = virDomainCapsNew ( emulatorbin , machine , arch , type ) ) )
goto cleanup ;
2014-09-17 13:33:35 +04:00
if ( fillFunc & & fillFunc ( domCaps , opaque ) < 0 ) {
virObjectUnref ( domCaps ) ;
domCaps = NULL ;
}
2014-06-25 15:24:53 +04:00
2014-09-17 03:52:54 +04:00
ret = domCaps ;
2014-06-25 15:24:53 +04:00
cleanup :
2014-09-17 03:52:54 +04:00
return ret ;
2014-06-25 15:24:53 +04:00
}
struct test_virDomainCapsFormatData {
const char * filename ;
const char * emulatorbin ;
const char * machine ;
virArch arch ;
virDomainVirtType type ;
virDomainCapsFill fillFunc ;
void * opaque ;
} ;
static int
test_virDomainCapsFormat ( const void * opaque )
{
struct test_virDomainCapsFormatData * data =
( struct test_virDomainCapsFormatData * ) opaque ;
virDomainCapsPtr domCaps = NULL ;
char * path = NULL ;
char * domCapsXML = NULL ;
char * domCapsFromFile = NULL ;
int ret = - 1 ;
if ( virAsprintf ( & path , " %s/domaincapsschemadata/domaincaps-%s.xml " ,
abs_srcdir , data - > filename ) < 0 )
goto cleanup ;
if ( virFileReadAll ( path , 8192 , & domCapsFromFile ) < 0 )
goto cleanup ;
if ( ! ( domCaps = buildVirDomainCaps ( data - > emulatorbin , data - > machine ,
data - > arch , data - > type ,
data - > fillFunc , data - > opaque ) ) )
goto cleanup ;
if ( ! ( domCapsXML = virDomainCapsFormat ( domCaps ) ) )
goto cleanup ;
if ( STRNEQ ( domCapsFromFile , domCapsXML ) ) {
virtTestDifference ( stderr , domCapsFromFile , domCapsXML ) ;
goto cleanup ;
}
ret = 0 ;
cleanup :
VIR_FREE ( domCapsFromFile ) ;
VIR_FREE ( domCapsXML ) ;
VIR_FREE ( path ) ;
virObjectUnref ( domCaps ) ;
return ret ;
}
static int
mymain ( void )
{
int ret = 0 ;
# define DO_TEST(Filename, Emulatorbin, Machine, Arch, Type, ...) \
do { \
struct test_virDomainCapsFormatData data = { . filename = Filename , \
. emulatorbin = Emulatorbin , . machine = Machine , . arch = Arch , \
. type = Type , __VA_ARGS__ } ; \
if ( virtTestRun ( Filename , test_virDomainCapsFormat , & data ) < 0 ) \
ret = - 1 ; \
} while ( 0 )
DO_TEST ( " basic " , " /bin/emulatorbin " , " my-machine-type " ,
VIR_ARCH_X86_64 , VIR_DOMAIN_VIRT_UML ) ;
DO_TEST ( " full " , " /bin/emulatorbin " , " my-machine-type " ,
VIR_ARCH_X86_64 , VIR_DOMAIN_VIRT_KVM , . fillFunc = fillAll ) ;
2014-06-25 20:39:29 +04:00
# ifdef WITH_QEMU
2014-09-17 03:52:54 +04:00
virQEMUDriverConfigPtr cfg = virQEMUDriverConfigNew ( false ) ;
2014-06-25 20:39:29 +04:00
# define DO_TEST_QEMU(Filename, QemuCapsFile, Emulatorbin, Machine, Arch, Type, ...) \
do { \
const char * capsPath = abs_srcdir " /qemucapabilitiesdata/ " QemuCapsFile " .caps " ; \
virQEMUCapsPtr qemuCaps = qemuTestParseCapabilities ( capsPath ) ; \
2014-09-17 03:52:54 +04:00
struct fillQemuCapsData fillData = { . qemuCaps = qemuCaps , . cfg = cfg } ; \
2014-06-25 20:39:29 +04:00
struct test_virDomainCapsFormatData data = { . filename = Filename , \
. emulatorbin = Emulatorbin , . machine = Machine , . arch = Arch , \
2014-09-17 03:52:54 +04:00
. type = Type , . fillFunc = fillQemuCaps , . opaque = & fillData } ; \
2014-06-25 20:39:29 +04:00
if ( ! qemuCaps ) { \
fprintf ( stderr , " Unable to build qemu caps from %s \n " , capsPath ) ; \
ret = - 1 ; \
} else if ( virtTestRun ( Filename , test_virDomainCapsFormat , & data ) < 0 ) \
ret = - 1 ; \
} while ( 0 )
DO_TEST_QEMU ( " qemu_1.6.50-1 " , " caps_1.6.50-1 " , " /usr/bin/qemu-system-x86_64 " ,
" pc-1.2 " , VIR_ARCH_X86_64 , VIR_DOMAIN_VIRT_KVM ) ;
2014-09-17 03:52:54 +04:00
virObjectUnref ( cfg ) ;
2014-06-25 20:39:29 +04:00
# endif /* WITH_QEMU */
2014-06-25 15:24:53 +04:00
return ret ;
}
VIRT_TEST_MAIN ( mymain )