/*
* viracpitest.c: Test ACPI table parsing
*
* Copyright (C) 2023 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
* .
*
*/
#include
#define LIBVIRT_VIRACPIPRIV_H_ALLOW
#include "testutils.h"
#include "viracpi.h"
#include "viracpipriv.h"
#define VIR_FROM_THIS VIR_FROM_NONE
typedef struct testAarch64SMMUData testAarch64SMMUData;
struct testAarch64SMMUData {
const char *filename;
ssize_t nnodes;
const virIORTNodeType *node_types;
};
static void
printBitmap(virBitmap *types)
{
size_t i;
for (i = 0; i < VIR_IORT_NODE_TYPE_LAST; i++) {
if (virBitmapIsBitSet(types, i)) {
fprintf(stderr, "%s\n", virIORTNodeTypeTypeToString(i));
}
}
}
static int
testAarch64SMMU(const void *opaque)
{
const testAarch64SMMUData *data = opaque;
g_autofree char *path = NULL;
g_autofree virIORTNodeHeader *nodes = NULL;
ssize_t nnodes = 0;
path = g_strdup_printf("%s/viracpidata/%s",
abs_srcdir, data->filename);
nnodes = virAcpiParseIORT(&nodes, path);
if (nnodes != data->nnodes) {
fprintf(stderr,
"virAcpiParseIORT() returned wrong number of nodes: %zd, expected %zd\n",
nnodes, data->nnodes);
return -1;
}
if (nnodes > 0) {
g_autoptr(virBitmap) typesSeen = virBitmapNew(VIR_IORT_NODE_TYPE_LAST);
g_autoptr(virBitmap) typesExp = virBitmapNew(VIR_IORT_NODE_TYPE_LAST);
size_t i = 0;
for (i = 0; data->node_types[i] != VIR_IORT_NODE_TYPE_LAST; i++) {
size_t type = data->node_types[i];
ignore_value(virBitmapSetBit(typesExp, type));
}
for (i = 0; i < nnodes; i++) {
virIORTNodeHeader *h = &nodes[i];
ignore_value(virBitmapSetBit(typesSeen, h->type));
}
if (!virBitmapEqual(typesSeen, typesExp)) {
fprintf(stderr, "node types mismatch.\n\nExpected:\n");
printBitmap(typesExp);
fprintf(stderr, "\nActual:\n");
printBitmap(typesSeen);
return -1;
}
}
return 0;
}
static int
mymain(void)
{
int ret = 0;
#define DO_TEST(filename, nnodes, ...) \
do { \
const virIORTNodeType node_types[] = { __VA_ARGS__, VIR_IORT_NODE_TYPE_LAST }; \
const testAarch64SMMUData data = {filename, nnodes, node_types }; \
if (virTestRun("aarch64 SMMU " filename, testAarch64SMMU, &data) < 0) \
ret = -1; \
} while (0)
DO_TEST("IORT_empty", 0, VIR_IORT_NODE_TYPE_LAST);
DO_TEST("IORT_virt_aarch64", 2,
VIR_IORT_NODE_TYPE_ITS_GROUP,
VIR_IORT_NODE_TYPE_ROOT_COMPLEX);
DO_TEST("IORT_ampere", 36,
VIR_IORT_NODE_TYPE_ITS_GROUP,
VIR_IORT_NODE_TYPE_ROOT_COMPLEX,
VIR_IORT_NODE_TYPE_SMMUV3);
DO_TEST("IORT_gigabyte", 30,
VIR_IORT_NODE_TYPE_ITS_GROUP,
VIR_IORT_NODE_TYPE_ROOT_COMPLEX,
VIR_IORT_NODE_TYPE_SMMUV1_OR_SMMUV2);
DO_TEST("IORT_qualcomm", 69,
VIR_IORT_NODE_TYPE_ITS_GROUP,
VIR_IORT_NODE_TYPE_NAMED_COMPONENT,
VIR_IORT_NODE_TYPE_ROOT_COMPLEX,
VIR_IORT_NODE_TYPE_SMMUV3,
VIR_IORT_NODE_TYPE_PMCG);
return ret == 0 ? EXIT_SUCCESS : EXIT_FAILURE;
}
VIR_TEST_MAIN(mymain)