2013-09-19 13:44:41 +02:00
/*
* Copyright ( C ) 2011 - 2013 Red Hat , Inc .
*
* 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/>.
*
*/
# include <config.h>
# include "testutils.h"
# include "testutilsqemu.h"
# include "qemumonitortestutils.h"
2018-12-13 14:53:50 +00:00
# define LIBVIRT_QEMU_CAPSPRIV_H_ALLOW
2016-04-28 17:02:38 +02:00
# include "qemu/qemu_capspriv.h"
2018-12-13 14:53:50 +00:00
# define LIBVIRT_QEMU_MONITOR_PRIV_H_ALLOW
2018-06-04 11:21:51 +02:00
# include "qemu/qemu_monitor_priv.h"
2019-02-14 11:25:50 +01:00
# define LIBVIRT_QEMU_PROCESSPRIV_H_ALLOW
# include "qemu/qemu_processpriv.h"
2013-09-19 13:44:41 +02:00
# define VIR_FROM_THIS VIR_FROM_NONE
typedef struct _testQemuData testQemuData ;
2013-10-29 13:57:19 -06:00
typedef testQemuData * testQemuDataPtr ;
2013-09-19 13:44:41 +02:00
struct _testQemuData {
2018-06-01 10:03:01 +02:00
virQEMUDriver driver ;
2019-03-07 14:33:20 +01:00
const char * dataDir ;
2016-04-25 14:19:49 +02:00
const char * archName ;
2013-09-19 13:44:41 +02:00
const char * base ;
2019-03-07 14:24:14 +01:00
int ret ;
2013-09-19 13:44:41 +02:00
} ;
2019-03-07 14:22:37 +01:00
static int
testQemuDataInit ( testQemuDataPtr data )
{
if ( qemuTestDriverInit ( & data - > driver ) < 0 )
return - 1 ;
2019-03-07 14:33:20 +01:00
data - > dataDir = abs_srcdir " /qemucapabilitiesdata " ;
2019-03-07 14:24:14 +01:00
data - > ret = 0 ;
2019-03-07 14:22:37 +01:00
return 0 ;
}
static void
testQemuDataReset ( testQemuDataPtr data )
{
qemuTestDriverFree ( & data - > driver ) ;
}
2013-09-19 13:44:41 +02:00
static int
testQemuCaps ( const void * opaque )
{
int ret = - 1 ;
2018-06-01 10:03:01 +02:00
testQemuData * data = ( void * ) opaque ;
2016-04-28 17:02:38 +02:00
char * repliesFile = NULL ;
char * capsFile = NULL ;
2013-09-19 13:44:41 +02:00
qemuMonitorTestPtr mon = NULL ;
2016-04-28 15:54:31 +02:00
virQEMUCapsPtr capsActual = NULL ;
2016-04-28 17:02:38 +02:00
char * actual = NULL ;
2019-01-28 10:22:00 +01:00
unsigned int fakeMicrocodeVersion = 0 ;
const char * p ;
2013-09-19 13:44:41 +02:00
2019-03-07 14:33:20 +01:00
if ( virAsprintf ( & repliesFile , " %s/%s.%s.replies " ,
data - > dataDir , data - > base , data - > archName ) < 0 | |
virAsprintf ( & capsFile , " %s/%s.%s.xml " ,
data - > dataDir , data - > base , data - > archName ) < 0 )
2013-09-19 13:44:41 +02:00
goto cleanup ;
2018-06-01 10:03:01 +02:00
if ( ! ( mon = qemuMonitorTestNewFromFileFull ( repliesFile , & data - > driver , NULL ) ) )
2013-09-19 13:44:41 +02:00
goto cleanup ;
2019-02-14 11:25:50 +01:00
if ( qemuProcessQMPInitMonitor ( qemuMonitorTestGetMonitor ( mon ) ) < 0 )
goto cleanup ;
2016-04-28 17:02:38 +02:00
if ( ! ( capsActual = virQEMUCapsNew ( ) ) | |
virQEMUCapsInitQMPMonitor ( capsActual ,
qemuMonitorTestGetMonitor ( mon ) ) < 0 )
2013-09-19 13:44:41 +02:00
goto cleanup ;
2017-12-12 16:23:41 +01:00
if ( virQEMUCapsGet ( capsActual , QEMU_CAPS_KVM ) ) {
2018-06-04 11:21:51 +02:00
qemuMonitorResetCommandID ( qemuMonitorTestGetMonitor ( mon ) ) ;
2019-02-14 11:25:50 +01:00
if ( qemuProcessQMPInitMonitor ( qemuMonitorTestGetMonitor ( mon ) ) < 0 )
goto cleanup ;
2017-12-12 16:23:41 +01:00
if ( virQEMUCapsInitQMPMonitorTCG ( capsActual ,
qemuMonitorTestGetMonitor ( mon ) ) < 0 )
goto cleanup ;
2019-01-28 10:22:00 +01:00
/* calculate fake microcode version based on filename for a reproducible
* number for testing which does not change with the contents */
for ( p = data - > archName ; * p ; p + + )
fakeMicrocodeVersion + = * p ;
fakeMicrocodeVersion * = 100000 ;
for ( p = data - > base ; * p ; p + + )
fakeMicrocodeVersion + = * p ;
virQEMUCapsSetMicrocodeVersion ( capsActual , fakeMicrocodeVersion ) ;
2017-12-12 16:23:41 +01:00
}
2016-06-10 17:16:21 +02:00
2017-06-13 17:55:45 +02:00
if ( ! ( actual = virQEMUCapsFormatCache ( capsActual ) ) )
2013-09-19 13:44:41 +02:00
goto cleanup ;
2016-05-26 17:01:53 +02:00
if ( virTestCompareToFile ( actual , capsFile ) < 0 )
2013-09-19 13:44:41 +02:00
goto cleanup ;
ret = 0 ;
2014-03-25 07:53:44 +01:00
cleanup :
2013-09-19 13:44:41 +02:00
VIR_FREE ( repliesFile ) ;
VIR_FREE ( capsFile ) ;
2016-04-28 17:02:38 +02:00
VIR_FREE ( actual ) ;
2013-09-19 13:44:41 +02:00
qemuMonitorTestFree ( mon ) ;
2016-04-28 15:54:31 +02:00
virObjectUnref ( capsActual ) ;
2013-09-19 13:44:41 +02:00
return ret ;
}
2016-08-04 22:43:22 +02:00
static int
testQemuCapsCopy ( const void * opaque )
{
int ret = - 1 ;
const testQemuData * data = opaque ;
char * capsFile = NULL ;
virCapsPtr caps = NULL ;
virQEMUCapsPtr orig = NULL ;
virQEMUCapsPtr copy = NULL ;
char * actual = NULL ;
2019-03-07 14:33:20 +01:00
if ( virAsprintf ( & capsFile , " %s/%s.%s.xml " ,
data - > dataDir , data - > base , data - > archName ) < 0 )
2016-08-04 22:43:22 +02:00
goto cleanup ;
if ( ! ( caps = virCapabilitiesNew ( virArchFromString ( data - > archName ) ,
false , false ) ) )
goto cleanup ;
2016-06-15 14:35:18 +02:00
if ( ! ( orig = qemuTestParseCapabilities ( caps , capsFile ) ) )
2016-08-04 22:43:22 +02:00
goto cleanup ;
if ( ! ( copy = virQEMUCapsNewCopy ( orig ) ) )
goto cleanup ;
2017-06-13 17:55:45 +02:00
if ( ! ( actual = virQEMUCapsFormatCache ( copy ) ) )
2016-08-04 22:43:22 +02:00
goto cleanup ;
if ( virTestCompareToFile ( actual , capsFile ) < 0 )
goto cleanup ;
ret = 0 ;
cleanup :
VIR_FREE ( capsFile ) ;
virObjectUnref ( caps ) ;
virObjectUnref ( orig ) ;
virObjectUnref ( copy ) ;
VIR_FREE ( actual ) ;
return ret ;
}
2019-03-07 14:53:18 +01:00
static int
doCapsTest ( const char * base ,
const char * archName ,
2019-03-07 16:02:37 +01:00
void * opaque )
2019-03-07 14:53:18 +01:00
{
2019-03-07 16:02:37 +01:00
testQemuDataPtr data = ( testQemuDataPtr ) opaque ;
2019-03-07 14:53:18 +01:00
VIR_AUTOFREE ( char * ) title = NULL ;
VIR_AUTOFREE ( char * ) copyTitle = NULL ;
if ( virAsprintf ( & title , " %s (%s) " , base , archName ) < 0 | |
virAsprintf ( & copyTitle , " copy %s (%s) " , base , archName ) < 0 ) {
return - 1 ;
}
data - > base = base ;
data - > archName = archName ;
if ( virTestRun ( title , testQemuCaps , data ) < 0 )
data - > ret = - 1 ;
if ( virTestRun ( copyTitle , testQemuCapsCopy , data ) < 0 )
data - > ret = - 1 ;
return 0 ;
}
2013-09-19 13:44:41 +02:00
static int
mymain ( void )
{
testQemuData data ;
2018-08-13 13:40:18 +02:00
# if !WITH_YAJL
2018-05-09 16:42:43 +02:00
fputs ( " libvirt not compiled with JSON support, skipping this test \n " , stderr ) ;
2013-09-19 13:44:41 +02:00
return EXIT_AM_SKIP ;
# endif
2019-03-07 14:22:37 +01:00
if ( virThreadInitialize ( ) < 0 )
2013-09-19 13:44:41 +02:00
return EXIT_FAILURE ;
virEventRegisterDefaultImpl ( ) ;
2019-03-07 14:22:37 +01:00
if ( testQemuDataInit ( & data ) < 0 )
return EXIT_FAILURE ;
2019-03-07 16:02:37 +01:00
if ( testQemuCapsIterate ( data . dataDir , " .replies " , doCapsTest , & data ) < 0 )
return EXIT_FAILURE ;
2013-09-19 13:44:41 +02:00
2016-04-26 15:04:55 +02:00
/*
* Run " tests/qemucapsprobe /path/to/qemu/binary >foo.replies "
* to generate updated or new * . replies data files .
2018-03-01 15:25:05 +01:00
*
* If you manually edit replies files you can run
* " tests/qemucapsfixreplies foo.replies " to fix the replies ids .
2019-03-13 14:28:07 +01:00
*
* Once a replies file has been generated and tweaked if necessary ,
* you can drop it into tests / qemucapabilitiesdata / ( with a sensible
* name - look at what ' s already there for inspiration ) and test
* programs will automatically pick it up .
*
* To generate the corresponding output files after a new replies
* file has been added , run " VIR_TEST_REGENERATE_OUTPUT=1 make check " .
2016-04-26 15:04:55 +02:00
*/
2019-03-07 14:22:37 +01:00
testQemuDataReset ( & data ) ;
tests: Avoid use of virQEMUDriverCreateXMLConf(NULL)
We use the function to create a virDomainXMLOption object that is
required for some functions. However, we don't pass the driver
pointer to the object anywhere - rather than pass NULL. This
causes trouble later when parsing a domain XML and calling post
parse callbacks:
Program received signal SIGSEGV, Segmentation fault.
0x000000000043fa3e in qemuDomainDefPostParse (def=0x7d36c0, caps=0x7caf10, opaque=0x0) at qemu/qemu_domain.c:1043
1043 qemuCaps = virQEMUCapsCacheLookup(driver->qemuCapsCache, def->emulator);
(gdb) bt
#0 0x000000000043fa3e in qemuDomainDefPostParse (def=0x7d36c0, caps=0x7caf10, opaque=0x0) at qemu/qemu_domain.c:1043
#1 0x00007ffff2928bf9 in virDomainDefPostParse (def=0x7d36c0, caps=0x7caf10, xmlopt=0x7c82c0) at conf/domain_conf.c:4269
#2 0x00007ffff294de04 in virDomainDefParseXML (xml=0x7da8c0, root=0x7dab80, ctxt=0x7da980, caps=0x7caf10, xmlopt=0x7c82c0, flags=0) at conf/domain_conf.c:16400
#3 0x00007ffff294e5b5 in virDomainDefParseNode (xml=0x7da8c0, root=0x7dab80, caps=0x7caf10, xmlopt=0x7c82c0, flags=0) at conf/domain_conf.c:16582
#4 0x00007ffff294e424 in virDomainDefParse (xmlStr=0x0, filename=0x7c7ef0 "/home/zippy/work/libvirt/libvirt.git/tests/securityselinuxlabeldata/disks.xml", caps=0x7caf10, xmlopt=0x7c82c0, flags=0) at conf/domain_conf.c:16529
#5 0x00007ffff294e4b2 in virDomainDefParseFile (filename=0x7c7ef0 "/home/zippy/work/libvirt/libvirt.git/tests/securityselinuxlabeldata/disks.xml", caps=0x7caf10, xmlopt=0x7c82c0, flags=0) at conf/domain_conf.c:16553
#6 0x00000000004303ca in testSELinuxLoadDef (testname=0x53c929 "disks") at securityselinuxlabeltest.c:192
#7 0x00000000004309e8 in testSELinuxLabeling (opaque=0x53c929) at securityselinuxlabeltest.c:313
#8 0x0000000000431207 in virtTestRun (title=0x53c92f "Labelling \"disks\"", body=0x430964 <testSELinuxLabeling>, data=0x53c929) at testutils.c:211
#9 0x0000000000430c5d in mymain () at securityselinuxlabeltest.c:373
#10 0x00000000004325c2 in virtTestMain (argc=1, argv=0x7fffffffd7e8, func=0x430b4a <mymain>) at testutils.c:863
#11 0x0000000000430deb in main (argc=1, argv=0x7fffffffd7e8) at securityselinuxlabeltest.c:381
Signed-off-by: Michal Privoznik <mprivozn@redhat.com>
2015-09-22 16:27:57 +02:00
2019-03-07 14:24:14 +01:00
return ( data . ret = = 0 ) ? EXIT_SUCCESS : EXIT_FAILURE ;
2013-09-19 13:44:41 +02:00
}
2017-03-29 16:45:42 +02:00
VIR_TEST_MAIN ( mymain )