2021-10-20 11:30:31 +03:00
|
|
|
|
/*
|
|
|
|
|
* Copyright (C) 2021 Canonical Ltd.
|
|
|
|
|
*
|
|
|
|
|
* 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 "internal.h"
|
|
|
|
|
#include "testutils.h"
|
|
|
|
|
#include "virpcivpd.h"
|
|
|
|
|
|
|
|
|
|
#define LIBVIRT_VIRPCIVPDPRIV_H_ALLOW
|
|
|
|
|
|
|
|
|
|
#include "virpcivpdpriv.h"
|
|
|
|
|
#include "virlog.h"
|
|
|
|
|
|
|
|
|
|
#define VIR_FROM_THIS VIR_FROM_NONE
|
|
|
|
|
|
|
|
|
|
#ifdef __linux__
|
|
|
|
|
|
|
|
|
|
VIR_LOG_INIT("tests.vpdtest");
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
typedef struct _TestPCIVPDKeywordValue {
|
|
|
|
|
const char *keyword;
|
|
|
|
|
const char *value;
|
|
|
|
|
char **actual;
|
|
|
|
|
} TestPCIVPDKeywordValue;
|
|
|
|
|
|
|
|
|
|
static int
|
|
|
|
|
testPCIVPDResourceBasic(const void *data G_GNUC_UNUSED)
|
|
|
|
|
{
|
|
|
|
|
size_t i = 0;
|
|
|
|
|
g_autoptr(virPCIVPDResourceRO) ro = virPCIVPDResourceRONew();
|
|
|
|
|
g_autoptr(virPCIVPDResourceRW) rw = virPCIVPDResourceRWNew();
|
|
|
|
|
/* Note: when the same keyword is updated multiple times the
|
|
|
|
|
* virPCIVPDResourceUpdateKeyword function is expected to free the
|
|
|
|
|
* previous value whether it is a fixed keyword or a custom one.
|
|
|
|
|
* */
|
|
|
|
|
const TestPCIVPDKeywordValue readOnlyCases[] = {
|
|
|
|
|
{.keyword = "EC", .value = "level1", .actual = &ro->change_level},
|
|
|
|
|
{.keyword = "EC", .value = "level2", .actual = &ro->change_level},
|
|
|
|
|
{.keyword = "change_level", .value = "level3", .actual = &ro->change_level},
|
|
|
|
|
{.keyword = "PN", .value = "number1", .actual = &ro->part_number},
|
|
|
|
|
{.keyword = "PN", .value = "number2", .actual = &ro->part_number},
|
|
|
|
|
{.keyword = "part_number", .value = "number3", .actual = &ro->part_number},
|
|
|
|
|
{.keyword = "MN", .value = "id1", .actual = &ro->manufacture_id},
|
|
|
|
|
{.keyword = "MN", .value = "id2", .actual = &ro->manufacture_id},
|
|
|
|
|
{.keyword = "manufacture_id", .value = "id3", &ro->manufacture_id},
|
|
|
|
|
{.keyword = "SN", .value = "serial1", .actual = &ro->serial_number},
|
|
|
|
|
{.keyword = "SN", .value = "serial2", .actual = &ro->serial_number},
|
|
|
|
|
{.keyword = "serial_number", .value = "serial3", .actual = &ro->serial_number},
|
|
|
|
|
};
|
|
|
|
|
const TestPCIVPDKeywordValue unsupportedFieldCases[] = {
|
|
|
|
|
{.keyword = "FG", .value = "42", .actual = NULL},
|
|
|
|
|
{.keyword = "LC", .value = "42", .actual = NULL},
|
|
|
|
|
{.keyword = "PG", .value = "42", .actual = NULL},
|
|
|
|
|
{.keyword = "CP", .value = "42", .actual = NULL},
|
|
|
|
|
{.keyword = "EX", .value = "42", .actual = NULL},
|
|
|
|
|
};
|
2021-11-02 17:32:43 +01:00
|
|
|
|
size_t numROCases = G_N_ELEMENTS(readOnlyCases);
|
|
|
|
|
size_t numUnsupportedCases = G_N_ELEMENTS(unsupportedFieldCases);
|
2021-10-20 11:30:31 +03:00
|
|
|
|
g_autoptr(virPCIVPDResource) res = g_new0(virPCIVPDResource, 1);
|
|
|
|
|
virPCIVPDResourceCustom *custom = NULL;
|
|
|
|
|
|
|
|
|
|
g_autofree char *val = g_strdup("testval");
|
|
|
|
|
res->name = g_steal_pointer(&val);
|
|
|
|
|
|
|
|
|
|
/* Initialize RO */
|
|
|
|
|
res->ro = g_steal_pointer(&ro);
|
|
|
|
|
|
|
|
|
|
/* Update keywords one by one and compare actual values with the expected ones. */
|
|
|
|
|
for (i = 0; i < numROCases; ++i) {
|
2024-01-24 17:15:10 +01:00
|
|
|
|
virPCIVPDResourceUpdateKeyword(res, true,
|
|
|
|
|
readOnlyCases[i].keyword,
|
|
|
|
|
readOnlyCases[i].value);
|
2021-10-20 11:30:31 +03:00
|
|
|
|
if (STRNEQ(readOnlyCases[i].value, *readOnlyCases[i].actual))
|
|
|
|
|
return -1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Do a basic vendor field check. */
|
2024-01-24 17:15:10 +01:00
|
|
|
|
virPCIVPDResourceUpdateKeyword(res, true, "V0", "vendor0");
|
2021-10-20 11:30:31 +03:00
|
|
|
|
|
|
|
|
|
if (res->ro->vendor_specific->len != 1)
|
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
|
|
custom = g_ptr_array_index(res->ro->vendor_specific, 0);
|
|
|
|
|
if (custom->idx != '0' || STRNEQ(custom->value, "vendor0"))
|
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
|
|
/* Make sure unsupported RO keyword updates are not fatal. */
|
|
|
|
|
for (i = 0; i < numUnsupportedCases; ++i) {
|
2024-01-24 17:15:10 +01:00
|
|
|
|
virPCIVPDResourceUpdateKeyword(res, true,
|
|
|
|
|
unsupportedFieldCases[i].keyword,
|
|
|
|
|
unsupportedFieldCases[i].value);
|
2021-10-20 11:30:31 +03:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Initialize RW */
|
|
|
|
|
res->rw = g_steal_pointer(&rw);
|
2024-01-24 17:15:10 +01:00
|
|
|
|
virPCIVPDResourceUpdateKeyword(res, false, "YA", "tag1");
|
|
|
|
|
if (STRNEQ(res->rw->asset_tag, "tag1"))
|
2021-10-20 11:30:31 +03:00
|
|
|
|
return -1;
|
|
|
|
|
|
2024-01-24 17:15:10 +01:00
|
|
|
|
virPCIVPDResourceUpdateKeyword(res, false, "asset_tag", "tag2");
|
|
|
|
|
if (STRNEQ(res->rw->asset_tag, "tag2"))
|
2021-10-20 11:30:31 +03:00
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
|
|
/* Do a basic system field check. */
|
2024-01-24 17:15:10 +01:00
|
|
|
|
virPCIVPDResourceUpdateKeyword(res, false, "Y0", "system0");
|
2021-10-20 11:30:31 +03:00
|
|
|
|
|
|
|
|
|
if (res->rw->system_specific->len != 1)
|
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
|
|
custom = g_ptr_array_index(res->rw->system_specific, 0);
|
|
|
|
|
if (custom->idx != '0' || STRNEQ(custom->value, "system0"))
|
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
|
|
/* Make sure unsupported RW keyword updates are not fatal. */
|
|
|
|
|
for (i = 0; i < numUnsupportedCases; ++i) {
|
2024-01-24 17:15:10 +01:00
|
|
|
|
/* This test is deliberately left in despite
|
|
|
|
|
* virPCIVPDResourceUpdateKeyword always succeeding to prevent
|
|
|
|
|
* possible regressions if the function is ever rewritten */
|
|
|
|
|
virPCIVPDResourceUpdateKeyword(res, false,
|
|
|
|
|
unsupportedFieldCases[i].keyword,
|
|
|
|
|
unsupportedFieldCases[i].value);
|
2021-10-20 11:30:31 +03:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Just make sure the name has not been changed during keyword updates. */
|
|
|
|
|
if (!STREQ_NULLABLE(res->name, "testval"))
|
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static int
|
|
|
|
|
testPCIVPDResourceCustomCompareIndex(const void *data G_GNUC_UNUSED)
|
|
|
|
|
{
|
2021-11-02 17:26:48 +01:00
|
|
|
|
g_autoptr(virPCIVPDResourceCustom) a = NULL;
|
|
|
|
|
g_autoptr(virPCIVPDResourceCustom) b = NULL;
|
2021-10-20 11:30:31 +03:00
|
|
|
|
|
|
|
|
|
/* Both are NULL */
|
|
|
|
|
if (!virPCIVPDResourceCustomCompareIndex(a, b))
|
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
|
|
/* a is not NULL */
|
|
|
|
|
a = g_new0(virPCIVPDResourceCustom, 1);
|
|
|
|
|
if (virPCIVPDResourceCustomCompareIndex(a, b))
|
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
|
|
/* Reverse */
|
|
|
|
|
if (virPCIVPDResourceCustomCompareIndex(b, a))
|
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
|
|
/* Same index, different strings */
|
|
|
|
|
b = g_new0(virPCIVPDResourceCustom, 1);
|
|
|
|
|
a->idx = 'z';
|
|
|
|
|
a->value = g_strdup("42");
|
|
|
|
|
b->idx = 'z';
|
|
|
|
|
b->value = g_strdup("24");
|
|
|
|
|
if (!virPCIVPDResourceCustomCompareIndex(b, a))
|
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
|
|
/* Different index, different strings */
|
|
|
|
|
a->idx = 'a';
|
|
|
|
|
if (virPCIVPDResourceCustomCompareIndex(b, a))
|
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
|
|
virPCIVPDResourceCustomFree(a);
|
|
|
|
|
virPCIVPDResourceCustomFree(b);
|
|
|
|
|
a = g_new0(virPCIVPDResourceCustom, 1);
|
|
|
|
|
b = g_new0(virPCIVPDResourceCustom, 1);
|
|
|
|
|
|
|
|
|
|
/* Same index, same strings */
|
|
|
|
|
a->idx = 'z';
|
|
|
|
|
a->value = g_strdup("42");
|
|
|
|
|
b->idx = 'z';
|
|
|
|
|
b->value = g_strdup("42");
|
|
|
|
|
if (!virPCIVPDResourceCustomCompareIndex(b, a))
|
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
|
|
/* Different index, same strings */
|
|
|
|
|
a->idx = 'a';
|
|
|
|
|
if (virPCIVPDResourceCustomCompareIndex(b, a))
|
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
|
|
/* Different index, same value pointers */
|
2022-01-28 18:42:45 +01:00
|
|
|
|
g_clear_pointer(&b->value, g_free);
|
2021-10-20 11:30:31 +03:00
|
|
|
|
b->value = a->value;
|
2022-01-28 20:37:42 +01:00
|
|
|
|
if (virPCIVPDResourceCustomCompareIndex(b, a)) {
|
|
|
|
|
b->value = NULL;
|
2021-10-20 11:30:31 +03:00
|
|
|
|
return -1;
|
2022-01-28 20:37:42 +01:00
|
|
|
|
}
|
2021-10-20 11:30:31 +03:00
|
|
|
|
|
|
|
|
|
b->value = NULL;
|
|
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static int
|
|
|
|
|
testPCIVPDResourceCustomUpsertValue(const void *data G_GNUC_UNUSED)
|
|
|
|
|
{
|
|
|
|
|
g_autoptr(GPtrArray) arr = g_ptr_array_new_full(0, (GDestroyNotify)virPCIVPDResourceCustomFree);
|
|
|
|
|
virPCIVPDResourceCustom *custom = NULL;
|
2024-01-24 16:11:24 +01:00
|
|
|
|
virPCIVPDResourceCustomUpsertValue(arr, 'A', "testval");
|
2021-10-20 11:30:31 +03:00
|
|
|
|
|
|
|
|
|
if (arr->len != 1)
|
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
|
|
custom = g_ptr_array_index(arr, 0);
|
|
|
|
|
if (custom == NULL || custom->idx != 'A' || STRNEQ_NULLABLE(custom->value, "testval"))
|
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
|
|
/* Idempotency */
|
2024-01-24 16:11:24 +01:00
|
|
|
|
virPCIVPDResourceCustomUpsertValue(arr, 'A', "testval");
|
2021-10-20 11:30:31 +03:00
|
|
|
|
|
|
|
|
|
if (arr->len != 1)
|
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
|
|
custom = g_ptr_array_index(arr, 0);
|
|
|
|
|
if (custom == NULL || custom->idx != 'A' || STRNEQ_NULLABLE(custom->value, "testval"))
|
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
|
|
/* Existing value updates. */
|
2024-01-24 16:11:24 +01:00
|
|
|
|
virPCIVPDResourceCustomUpsertValue(arr, 'A', "testvalnew");
|
2021-10-20 11:30:31 +03:00
|
|
|
|
|
|
|
|
|
if (arr->len != 1)
|
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
|
|
custom = g_ptr_array_index(arr, 0);
|
|
|
|
|
if (custom == NULL || custom->idx != 'A' || STRNEQ_NULLABLE(custom->value, "testvalnew"))
|
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
|
|
/* Inserting multiple values */
|
2024-01-24 16:11:24 +01:00
|
|
|
|
virPCIVPDResourceCustomUpsertValue(arr, '1', "42");
|
2021-10-20 11:30:31 +03:00
|
|
|
|
|
|
|
|
|
if (arr->len != 2)
|
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
|
|
custom = g_ptr_array_index(arr, 1);
|
|
|
|
|
if (custom == NULL || custom->idx != '1' || STRNEQ_NULLABLE(custom->value, "42"))
|
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
typedef struct _TestPCIVPDExpectedString {
|
|
|
|
|
const char *keyword;
|
|
|
|
|
bool expected;
|
|
|
|
|
} TestPCIVPDExpectedString;
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* testPCIVPDIsValidTextValue:
|
|
|
|
|
*
|
|
|
|
|
* Test expected text value validation. Static metadata about possible values is taken
|
|
|
|
|
* from the PCI(e) standards and based on some real-world hardware examples.
|
|
|
|
|
* */
|
|
|
|
|
static int
|
|
|
|
|
testPCIVPDIsValidTextValue(const void *data G_GNUC_UNUSED)
|
|
|
|
|
{
|
|
|
|
|
size_t i = 0;
|
|
|
|
|
|
|
|
|
|
const TestPCIVPDExpectedString textValueCases[] = {
|
|
|
|
|
/* Numbers */
|
|
|
|
|
{"42", true},
|
|
|
|
|
/* Alphanumeric */
|
|
|
|
|
{"DCM1001008FC52101008FC53201008FC54301008FC5", true},
|
|
|
|
|
/* Dots */
|
|
|
|
|
{"DSV1028VPDR.VER1.0", true},
|
|
|
|
|
/* Whitespace presence */
|
|
|
|
|
{"NMVIntel Corp", true},
|
|
|
|
|
/* Comma and spaces */
|
|
|
|
|
{"BlueField-2 DPU 25GbE Dual-Port SFP56, Tall Bracket", true},
|
|
|
|
|
/* Equal signs and colons. */
|
|
|
|
|
{"MLX:MN=MLNX:CSKU=V2:UUID=V3:PCI=V0:MODL=BF2H332A", true},
|
|
|
|
|
/* Dashes */
|
|
|
|
|
{"MBF2H332A-AEEOT", true},
|
|
|
|
|
{"under_score_example", true},
|
|
|
|
|
{"", true},
|
|
|
|
|
{";", true},
|
2021-10-29 21:57:17 +03:00
|
|
|
|
{"\\42", true},
|
|
|
|
|
{"N/A", true},
|
|
|
|
|
/* The first and last code points are outside ASCII (multi-byte in UTF-8). */
|
|
|
|
|
{"гbl🐧", false},
|
2021-10-20 11:30:31 +03:00
|
|
|
|
};
|
2021-11-02 17:32:43 +01:00
|
|
|
|
for (i = 0; i < G_N_ELEMENTS(textValueCases); ++i) {
|
2021-10-20 11:30:31 +03:00
|
|
|
|
if (virPCIVPDResourceIsValidTextValue(textValueCases[i].keyword) !=
|
|
|
|
|
textValueCases[i].expected)
|
|
|
|
|
return -1;
|
|
|
|
|
}
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* testPCIVPDGetFieldValueFormat:
|
|
|
|
|
*
|
|
|
|
|
* A simple test to assess the functionality of the
|
|
|
|
|
* virPCIVPDResourceGetFieldValueFormat function.
|
|
|
|
|
* */
|
|
|
|
|
static int
|
|
|
|
|
testPCIVPDGetFieldValueFormat(const void *data G_GNUC_UNUSED)
|
|
|
|
|
{
|
|
|
|
|
typedef struct _TestPCIVPDExpectedFieldValueFormat {
|
|
|
|
|
const char *keyword;
|
|
|
|
|
virPCIVPDResourceFieldValueFormat expected;
|
|
|
|
|
} TestPCIVPDExpectedFieldValueFormat;
|
|
|
|
|
|
|
|
|
|
size_t i = 0;
|
|
|
|
|
|
|
|
|
|
const TestPCIVPDExpectedFieldValueFormat valueFormatCases[] = {
|
|
|
|
|
{"SN", VIR_PCI_VPD_RESOURCE_FIELD_VALUE_FORMAT_TEXT},
|
|
|
|
|
{"EC", VIR_PCI_VPD_RESOURCE_FIELD_VALUE_FORMAT_TEXT},
|
|
|
|
|
{"MN", VIR_PCI_VPD_RESOURCE_FIELD_VALUE_FORMAT_TEXT},
|
|
|
|
|
{"PN", VIR_PCI_VPD_RESOURCE_FIELD_VALUE_FORMAT_TEXT},
|
|
|
|
|
{"RV", VIR_PCI_VPD_RESOURCE_FIELD_VALUE_FORMAT_RESVD},
|
|
|
|
|
{"RW", VIR_PCI_VPD_RESOURCE_FIELD_VALUE_FORMAT_RDWR},
|
|
|
|
|
{"VA", VIR_PCI_VPD_RESOURCE_FIELD_VALUE_FORMAT_TEXT},
|
|
|
|
|
{"YA", VIR_PCI_VPD_RESOURCE_FIELD_VALUE_FORMAT_TEXT},
|
|
|
|
|
{"YZ", VIR_PCI_VPD_RESOURCE_FIELD_VALUE_FORMAT_TEXT},
|
|
|
|
|
{"CP", VIR_PCI_VPD_RESOURCE_FIELD_VALUE_FORMAT_BINARY},
|
|
|
|
|
/* Invalid keywords. */
|
|
|
|
|
{"", VIR_PCI_VPD_RESOURCE_FIELD_VALUE_FORMAT_LAST},
|
|
|
|
|
{"sn", VIR_PCI_VPD_RESOURCE_FIELD_VALUE_FORMAT_LAST},
|
|
|
|
|
{"ec", VIR_PCI_VPD_RESOURCE_FIELD_VALUE_FORMAT_LAST},
|
|
|
|
|
{"mn", VIR_PCI_VPD_RESOURCE_FIELD_VALUE_FORMAT_LAST},
|
|
|
|
|
{"pn", VIR_PCI_VPD_RESOURCE_FIELD_VALUE_FORMAT_LAST},
|
|
|
|
|
{"4", VIR_PCI_VPD_RESOURCE_FIELD_VALUE_FORMAT_LAST},
|
|
|
|
|
{"42", VIR_PCI_VPD_RESOURCE_FIELD_VALUE_FORMAT_LAST},
|
|
|
|
|
{"Y", VIR_PCI_VPD_RESOURCE_FIELD_VALUE_FORMAT_LAST},
|
|
|
|
|
{"V", VIR_PCI_VPD_RESOURCE_FIELD_VALUE_FORMAT_LAST},
|
|
|
|
|
{"v", VIR_PCI_VPD_RESOURCE_FIELD_VALUE_FORMAT_LAST},
|
|
|
|
|
{"vA", VIR_PCI_VPD_RESOURCE_FIELD_VALUE_FORMAT_LAST},
|
|
|
|
|
{"va", VIR_PCI_VPD_RESOURCE_FIELD_VALUE_FORMAT_LAST},
|
|
|
|
|
{"ya", VIR_PCI_VPD_RESOURCE_FIELD_VALUE_FORMAT_LAST},
|
|
|
|
|
{"Ya", VIR_PCI_VPD_RESOURCE_FIELD_VALUE_FORMAT_LAST},
|
|
|
|
|
/* 2 bytes but not present in the spec. */
|
|
|
|
|
{"EX", VIR_PCI_VPD_RESOURCE_FIELD_VALUE_FORMAT_LAST},
|
|
|
|
|
/* Many numeric bytes. */
|
|
|
|
|
{"4242", VIR_PCI_VPD_RESOURCE_FIELD_VALUE_FORMAT_LAST},
|
|
|
|
|
/* Many letters. */
|
|
|
|
|
{"EXAMPLE", VIR_PCI_VPD_RESOURCE_FIELD_VALUE_FORMAT_LAST},
|
|
|
|
|
};
|
2021-11-02 17:32:43 +01:00
|
|
|
|
for (i = 0; i < G_N_ELEMENTS(valueFormatCases); ++i) {
|
2021-10-20 11:30:31 +03:00
|
|
|
|
if (virPCIVPDResourceGetFieldValueFormat(valueFormatCases[i].keyword) !=
|
|
|
|
|
valueFormatCases[i].expected)
|
|
|
|
|
return -1;
|
|
|
|
|
}
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
# define VPD_STRING_RESOURCE_EXAMPLE_HEADER \
|
|
|
|
|
PCI_VPD_LARGE_RESOURCE_FLAG | PCI_VPD_STRING_RESOURCE_FLAG, 0x08, 0x00
|
|
|
|
|
|
|
|
|
|
# define VPD_STRING_RESOURCE_EXAMPLE_DATA \
|
|
|
|
|
't', 'e', 's', 't', 'n', 'a', 'm', 'e'
|
|
|
|
|
|
|
|
|
|
# define VPD_R_FIELDS_EXAMPLE_HEADER \
|
|
|
|
|
PCI_VPD_LARGE_RESOURCE_FLAG | PCI_VPD_READ_ONLY_LARGE_RESOURCE_FLAG, 0x16, 0x00
|
|
|
|
|
|
|
|
|
|
# define VPD_R_EXAMPLE_VALID_RV_FIELD \
|
|
|
|
|
'R', 'V', 0x02, 0x31, 0x00
|
|
|
|
|
|
|
|
|
|
# define VPD_R_EXAMPLE_INVALID_RV_FIELD \
|
|
|
|
|
'R', 'V', 0x02, 0xFF, 0x00
|
|
|
|
|
|
|
|
|
|
# define VPD_R_EXAMPLE_FIELDS \
|
|
|
|
|
'P', 'N', 0x02, '4', '2', \
|
|
|
|
|
'E', 'C', 0x04, '4', '2', '4', '2', \
|
|
|
|
|
'V', 'A', 0x02, 'E', 'X'
|
|
|
|
|
|
|
|
|
|
# define VPD_R_FIELDS_EXAMPLE_DATA \
|
|
|
|
|
VPD_R_EXAMPLE_FIELDS, \
|
|
|
|
|
VPD_R_EXAMPLE_VALID_RV_FIELD
|
|
|
|
|
|
|
|
|
|
# define VPD_W_FIELDS_EXAMPLE_HEADER \
|
|
|
|
|
PCI_VPD_LARGE_RESOURCE_FLAG | PCI_VPD_READ_WRITE_LARGE_RESOURCE_FLAG, 0x19, 0x00
|
|
|
|
|
|
|
|
|
|
# define VPD_W_EXAMPLE_FIELDS \
|
|
|
|
|
'V', 'Z', 0x02, '4', '2', \
|
2024-01-23 16:40:34 +01:00
|
|
|
|
'Y', 'A', 0x04, '!', '<', '>', ':', \
|
2021-10-20 11:30:31 +03:00
|
|
|
|
'Y', 'F', 0x02, 'E', 'X', \
|
|
|
|
|
'Y', 'E', 0x00, \
|
|
|
|
|
'R', 'W', 0x02, 0x00, 0x00
|
|
|
|
|
|
|
|
|
|
static int
|
|
|
|
|
testVirPCIVPDValidateExampleReadOnlyFields(virPCIVPDResource *res)
|
|
|
|
|
{
|
|
|
|
|
const char *expectedName = "testname";
|
|
|
|
|
virPCIVPDResourceCustom *custom = NULL;
|
|
|
|
|
if (STRNEQ(res->name, expectedName)) {
|
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
|
"Unexpected string resource value: %s, expected: %s",
|
|
|
|
|
res->name, expectedName);
|
|
|
|
|
return -1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (!res->ro) {
|
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
|
"Read-only keywords are missing from the VPD resource.");
|
|
|
|
|
return -1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (STRNEQ_NULLABLE(res->ro->part_number, "42")) {
|
|
|
|
|
return -1;
|
|
|
|
|
} else if (STRNEQ_NULLABLE(res->ro->change_level, "4242")) {
|
|
|
|
|
return -1;
|
|
|
|
|
}
|
|
|
|
|
if (!res->ro->vendor_specific)
|
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
|
|
custom = g_ptr_array_index(res->ro->vendor_specific, 0);
|
|
|
|
|
if (custom->idx != 'A' || STRNEQ_NULLABLE(custom->value, "EX"))
|
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static int
|
|
|
|
|
testVirPCIVPDParseFullVPD(const void *opaque G_GNUC_UNUSED)
|
|
|
|
|
{
|
2022-02-10 11:32:00 +01:00
|
|
|
|
VIR_AUTOCLOSE fd = -1;
|
2021-10-20 11:30:31 +03:00
|
|
|
|
size_t dataLen = 0;
|
|
|
|
|
|
|
|
|
|
g_autoptr(virPCIVPDResource) res = NULL;
|
|
|
|
|
/* Note: Custom fields are supposed to be freed by the resource cleanup code. */
|
|
|
|
|
virPCIVPDResourceCustom *custom = NULL;
|
|
|
|
|
|
|
|
|
|
const uint8_t fullVPDExample[] = {
|
|
|
|
|
VPD_STRING_RESOURCE_EXAMPLE_HEADER, VPD_STRING_RESOURCE_EXAMPLE_DATA,
|
|
|
|
|
VPD_R_FIELDS_EXAMPLE_HEADER, VPD_R_FIELDS_EXAMPLE_DATA,
|
|
|
|
|
VPD_W_FIELDS_EXAMPLE_HEADER, VPD_W_EXAMPLE_FIELDS,
|
|
|
|
|
PCI_VPD_RESOURCE_END_VAL
|
|
|
|
|
};
|
|
|
|
|
|
2021-11-02 17:32:43 +01:00
|
|
|
|
dataLen = G_N_ELEMENTS(fullVPDExample);
|
2021-11-23 15:05:43 +01:00
|
|
|
|
if ((fd = virCreateAnonymousFile(fullVPDExample, dataLen)) < 0)
|
|
|
|
|
return -1;
|
|
|
|
|
|
2021-10-20 11:30:31 +03:00
|
|
|
|
res = virPCIVPDParse(fd);
|
|
|
|
|
|
|
|
|
|
if (!res) {
|
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
|
"The resource pointer is NULL after parsing which is unexpected");
|
2021-10-29 21:57:18 +03:00
|
|
|
|
return -1;
|
2021-10-20 11:30:31 +03:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (!res->ro) {
|
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
|
"Read-only keywords are missing from the VPD resource.");
|
|
|
|
|
return -1;
|
|
|
|
|
} else if (!res->rw) {
|
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
|
"Read-write keywords are missing from the VPD resource.");
|
|
|
|
|
return -1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (testVirPCIVPDValidateExampleReadOnlyFields(res))
|
|
|
|
|
return -1;
|
|
|
|
|
|
2024-01-23 16:40:34 +01:00
|
|
|
|
if (STRNEQ_NULLABLE(res->rw->asset_tag, "!<>:"))
|
2021-10-20 11:30:31 +03:00
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
|
|
if (!res->rw->vendor_specific)
|
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
|
|
custom = g_ptr_array_index(res->rw->vendor_specific, 0);
|
|
|
|
|
if (custom->idx != 'Z' || STRNEQ_NULLABLE(custom->value, "42"))
|
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
|
|
if (!res->rw->system_specific)
|
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
|
|
custom = g_ptr_array_index(res->rw->system_specific, 0);
|
|
|
|
|
if (custom->idx != 'F' || STRNEQ_NULLABLE(custom->value, "EX"))
|
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
|
|
custom = g_ptr_array_index(res->rw->system_specific, 1);
|
|
|
|
|
if (custom->idx != 'E' || STRNEQ_NULLABLE(custom->value, ""))
|
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
|
|
custom = NULL;
|
2021-10-29 21:57:18 +03:00
|
|
|
|
return 0;
|
2021-10-20 11:30:31 +03:00
|
|
|
|
}
|
|
|
|
|
|
2021-10-29 21:57:16 +03:00
|
|
|
|
static int
|
|
|
|
|
testVirPCIVPDParseZeroLengthRW(const void *opaque G_GNUC_UNUSED)
|
|
|
|
|
{
|
2022-02-10 11:32:00 +01:00
|
|
|
|
VIR_AUTOCLOSE fd = -1;
|
2021-10-29 21:57:16 +03:00
|
|
|
|
size_t dataLen = 0;
|
|
|
|
|
|
|
|
|
|
g_autoptr(virPCIVPDResource) res = NULL;
|
|
|
|
|
virPCIVPDResourceCustom *custom = NULL;
|
|
|
|
|
|
|
|
|
|
/* The RW field has a zero length which means there is no more RW space left. */
|
|
|
|
|
const uint8_t fullVPDExample[] = {
|
|
|
|
|
VPD_STRING_RESOURCE_EXAMPLE_HEADER, VPD_STRING_RESOURCE_EXAMPLE_DATA,
|
|
|
|
|
VPD_R_FIELDS_EXAMPLE_HEADER, VPD_R_FIELDS_EXAMPLE_DATA,
|
|
|
|
|
PCI_VPD_LARGE_RESOURCE_FLAG | PCI_VPD_READ_WRITE_LARGE_RESOURCE_FLAG, 0x08, 0x00,
|
|
|
|
|
'V', 'Z', 0x02, '4', '2',
|
|
|
|
|
'R', 'W', 0x00,
|
|
|
|
|
PCI_VPD_RESOURCE_END_VAL
|
|
|
|
|
};
|
|
|
|
|
|
2021-11-02 17:32:43 +01:00
|
|
|
|
dataLen = G_N_ELEMENTS(fullVPDExample);
|
2021-11-23 15:05:43 +01:00
|
|
|
|
if ((fd = virCreateAnonymousFile(fullVPDExample, dataLen)) < 0)
|
|
|
|
|
return -1;
|
|
|
|
|
|
2021-10-29 21:57:16 +03:00
|
|
|
|
res = virPCIVPDParse(fd);
|
|
|
|
|
|
|
|
|
|
if (!res) {
|
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
|
"The resource pointer is NULL after parsing which is unexpected");
|
|
|
|
|
return -1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (!res->ro) {
|
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
|
"Read-only keywords are missing from the VPD resource.");
|
|
|
|
|
return -1;
|
|
|
|
|
} else if (!res->rw) {
|
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
|
"Read-write keywords are missing from the VPD resource.");
|
|
|
|
|
return -1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (testVirPCIVPDValidateExampleReadOnlyFields(res))
|
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
|
|
custom = g_ptr_array_index(res->rw->vendor_specific, 0);
|
|
|
|
|
if (custom->idx != 'Z' || STRNEQ_NULLABLE(custom->value, "42"))
|
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
|
|
custom = NULL;
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static int
|
|
|
|
|
testVirPCIVPDParseNoRW(const void *opaque G_GNUC_UNUSED)
|
|
|
|
|
{
|
2022-02-10 11:32:00 +01:00
|
|
|
|
VIR_AUTOCLOSE fd = -1;
|
2021-10-29 21:57:16 +03:00
|
|
|
|
size_t dataLen = 0;
|
|
|
|
|
|
|
|
|
|
g_autoptr(virPCIVPDResource) res = NULL;
|
|
|
|
|
virPCIVPDResourceCustom *custom = NULL;
|
|
|
|
|
|
|
|
|
|
/* The RW field has a zero length which means there is no more RW space left. */
|
|
|
|
|
const uint8_t fullVPDExample[] = {
|
|
|
|
|
VPD_STRING_RESOURCE_EXAMPLE_HEADER, VPD_STRING_RESOURCE_EXAMPLE_DATA,
|
|
|
|
|
VPD_R_FIELDS_EXAMPLE_HEADER, VPD_R_FIELDS_EXAMPLE_DATA,
|
|
|
|
|
PCI_VPD_LARGE_RESOURCE_FLAG | PCI_VPD_READ_WRITE_LARGE_RESOURCE_FLAG, 0x05, 0x00,
|
|
|
|
|
'V', 'Z', 0x02, '4', '2',
|
|
|
|
|
PCI_VPD_RESOURCE_END_VAL
|
|
|
|
|
};
|
|
|
|
|
|
2021-11-02 17:32:43 +01:00
|
|
|
|
dataLen = G_N_ELEMENTS(fullVPDExample);
|
2021-11-23 15:05:43 +01:00
|
|
|
|
if ((fd = virCreateAnonymousFile(fullVPDExample, dataLen)) < 0)
|
|
|
|
|
return -1;
|
|
|
|
|
|
2021-10-29 21:57:16 +03:00
|
|
|
|
res = virPCIVPDParse(fd);
|
|
|
|
|
|
|
|
|
|
if (!res) {
|
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
|
"The resource pointer is NULL after parsing which is unexpected");
|
|
|
|
|
return -1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (!res->ro) {
|
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
|
"Read-only keywords are missing from the VPD resource.");
|
|
|
|
|
return -1;
|
|
|
|
|
} else if (!res->rw) {
|
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
|
"Read-write keywords are missing from the VPD resource.");
|
|
|
|
|
return -1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (testVirPCIVPDValidateExampleReadOnlyFields(res))
|
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
|
|
custom = g_ptr_array_index(res->rw->vendor_specific, 0);
|
|
|
|
|
if (custom->idx != 'Z' || STRNEQ_NULLABLE(custom->value, "42"))
|
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
|
|
custom = NULL;
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
2021-10-20 11:30:31 +03:00
|
|
|
|
static int
|
|
|
|
|
testVirPCIVPDParseFullVPDSkipInvalidKeywords(const void *opaque G_GNUC_UNUSED)
|
|
|
|
|
{
|
2022-02-10 11:32:00 +01:00
|
|
|
|
VIR_AUTOCLOSE fd = -1;
|
2021-10-20 11:30:31 +03:00
|
|
|
|
size_t dataLen = 0;
|
|
|
|
|
|
|
|
|
|
g_autoptr(virPCIVPDResource) res = NULL;
|
|
|
|
|
|
|
|
|
|
const uint8_t fullVPDExample[] = {
|
|
|
|
|
VPD_STRING_RESOURCE_EXAMPLE_HEADER,
|
|
|
|
|
VPD_STRING_RESOURCE_EXAMPLE_DATA,
|
|
|
|
|
PCI_VPD_LARGE_RESOURCE_FLAG | PCI_VPD_READ_ONLY_LARGE_RESOURCE_FLAG, 0x25, 0x00,
|
|
|
|
|
VPD_R_EXAMPLE_FIELDS,
|
|
|
|
|
/* The keywords below (except for "RV") are invalid but will be skipped by the parser */
|
|
|
|
|
0x07, 'A', 0x02, 0x00, 0x00,
|
|
|
|
|
'V', 0x07, 0x02, 0x00, 0x00,
|
|
|
|
|
'e', 'x', 0x02, 0x00, 0x00,
|
|
|
|
|
'R', 'V', 0x02, 0x9A, 0x00,
|
|
|
|
|
PCI_VPD_RESOURCE_END_VAL
|
|
|
|
|
};
|
|
|
|
|
|
2021-11-02 17:32:43 +01:00
|
|
|
|
dataLen = G_N_ELEMENTS(fullVPDExample);
|
2021-11-23 15:05:43 +01:00
|
|
|
|
if ((fd = virCreateAnonymousFile(fullVPDExample, dataLen)) < 0)
|
|
|
|
|
return -1;
|
|
|
|
|
|
2021-10-20 11:30:31 +03:00
|
|
|
|
res = virPCIVPDParse(fd);
|
|
|
|
|
|
|
|
|
|
if (!res) {
|
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
|
"The resource pointer is NULL after parsing which is unexpected.");
|
|
|
|
|
return -1;
|
|
|
|
|
}
|
|
|
|
|
if (!res->ro) {
|
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
|
"The RO portion of the VPD resource is NULL.");
|
|
|
|
|
return -1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (testVirPCIVPDValidateExampleReadOnlyFields(res))
|
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
2021-10-29 21:57:17 +03:00
|
|
|
|
static int
|
|
|
|
|
testVirPCIVPDParseFullVPDSkipInvalidValues(const void *opaque G_GNUC_UNUSED)
|
|
|
|
|
{
|
2022-02-10 11:32:00 +01:00
|
|
|
|
VIR_AUTOCLOSE fd = -1;
|
2021-10-29 21:57:17 +03:00
|
|
|
|
size_t dataLen = 0;
|
|
|
|
|
size_t i = 0;
|
|
|
|
|
virPCIVPDResourceCustom *custom = NULL;
|
|
|
|
|
|
|
|
|
|
g_autoptr(virPCIVPDResource) res = NULL;
|
|
|
|
|
|
|
|
|
|
/* This example is based on real-world hardware which was programmed by the vendor with
|
|
|
|
|
* invalid field values in both the RO section and RW section. The RO section contains
|
|
|
|
|
* fields that are not valid per the spec but accepted by Libvirt as printable ASCII
|
|
|
|
|
* characters. The RW field has a 0 length which means there is no more space in the
|
|
|
|
|
* RW section. */
|
|
|
|
|
const uint8_t fullVPDExample[] = {
|
|
|
|
|
0x82, 0x23, 0x00, 0x48, 0x50, 0x20, 0x45, 0x74, 0x68, 0x65, 0x72, 0x6e, 0x65, 0x74,
|
|
|
|
|
0x20, 0x31, 0x47, 0x62, 0x20, 0x32, 0x2d, 0x70, 0x6f, 0x72, 0x74, 0x20, 0x33, 0x36,
|
|
|
|
|
0x31, 0x69, 0x20, 0x41, 0x64, 0x61, 0x70, 0x74, 0x65, 0x72, 0x90, 0x42, 0x00, 0x50,
|
|
|
|
|
0x4e, 0x03, 0x4e, 0x2f, 0x41, 0x45, 0x43, 0x03, 0x4e, 0x2f, 0x41, 0x53, 0x4e, 0x03,
|
|
|
|
|
0x4e, 0x2f, 0x41, 0x56, 0x30, 0x29, 0x34, 0x57, 0x2f, 0x31, 0x57, 0x20, 0x50, 0x43,
|
|
|
|
|
0x49, 0x65, 0x47, 0x32, 0x78, 0x34, 0x20, 0x32, 0x70, 0x20, 0x31, 0x47, 0x62, 0x45,
|
|
|
|
|
0x20, 0x52, 0x4a, 0x34, 0x35, 0x20, 0x49, 0x6e, 0x74, 0x65, 0x6c, 0x20, 0x69, 0x33,
|
|
|
|
|
0x35, 0x30, 0x20, 0x20, 0x20, 0x52, 0x56, 0x01, 0x63, 0x91, 0x47, 0x00, 0x56, 0x31,
|
|
|
|
|
0x06, 0x35, 0x2e, 0x37, 0x2e, 0x30, 0x36, 0x56, 0x33, 0x06, 0x32, 0x2e, 0x38, 0x2e,
|
|
|
|
|
0x32, 0x30, 0x56, 0x36, 0x06, 0x31, 0x2e, 0x35, 0x2e, 0x33, 0x35, 0x59, 0x41, 0x03,
|
|
|
|
|
0x4e, 0x2f, 0x41, 0x59, 0x42, 0x10, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff,
|
|
|
|
|
0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0x59, 0x43, 0x0D, 0xff, 0xff, 0xff,
|
|
|
|
|
0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 'R', 'W', 0x00, 0x78,
|
|
|
|
|
};
|
|
|
|
|
|
2021-11-02 17:32:43 +01:00
|
|
|
|
dataLen = G_N_ELEMENTS(fullVPDExample);
|
2021-11-23 15:05:43 +01:00
|
|
|
|
if ((fd = virCreateAnonymousFile(fullVPDExample, dataLen)) < 0)
|
|
|
|
|
return -1;
|
|
|
|
|
|
2021-10-29 21:57:17 +03:00
|
|
|
|
res = virPCIVPDParse(fd);
|
|
|
|
|
|
|
|
|
|
if (!res) {
|
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
|
"The resource pointer is NULL after parsing which is unexpected.");
|
|
|
|
|
return -1;
|
|
|
|
|
}
|
|
|
|
|
/* Some values in the read-write section are invalid but parsing should succeed
|
|
|
|
|
* considering the parser is implemented to be graceful about invalid keywords and
|
|
|
|
|
* values. */
|
|
|
|
|
if (!res->ro) {
|
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
|
"The RO section consisting of only invalid fields got parsed successfully");
|
|
|
|
|
return -1;
|
|
|
|
|
}
|
|
|
|
|
if (!res->rw) {
|
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
|
"Could not successfully parse an RW section with some invalid fields");
|
|
|
|
|
return -1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (!STREQ_NULLABLE(res->ro->change_level, "N/A")) {
|
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
|
"Could not parse a change level field with acceptable contents");
|
|
|
|
|
return -1;
|
|
|
|
|
}
|
|
|
|
|
if (!STREQ_NULLABLE(res->ro->part_number, "N/A")) {
|
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
|
"Could not parse a part number field with acceptable contents");
|
|
|
|
|
return -1;
|
|
|
|
|
}
|
|
|
|
|
if (!STREQ_NULLABLE(res->ro->serial_number, "N/A")) {
|
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
|
"Could not parse a serial number with acceptable contents");
|
|
|
|
|
return -1;
|
|
|
|
|
}
|
|
|
|
|
if (!STREQ_NULLABLE(res->rw->asset_tag, "N/A")) {
|
|
|
|
|
/* The asset tag has an invalid value in this case so it should be NULL. */
|
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
|
"Could not parse an asset tag with acceptable contents");
|
|
|
|
|
return -1;
|
|
|
|
|
}
|
|
|
|
|
if (res->rw->vendor_specific->len != 3) {
|
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
|
"The number of parsed vendor fields is not equal to the expected number.");
|
|
|
|
|
return -1;
|
|
|
|
|
}
|
|
|
|
|
if (res->rw->system_specific->len > 0) {
|
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
|
"Successfully parsed some systems-specific fields while none are valid");
|
|
|
|
|
return -1;
|
|
|
|
|
}
|
|
|
|
|
for (i = 0; i < res->rw->vendor_specific->len; ++i) {
|
|
|
|
|
custom = ((virPCIVPDResourceCustom*)g_ptr_array_index(res->rw->vendor_specific, i));
|
|
|
|
|
if (custom->idx == '1') {
|
|
|
|
|
if (STRNEQ(custom->value, "5.7.06")) {
|
|
|
|
|
return -1;
|
|
|
|
|
}
|
|
|
|
|
} else if (custom->idx == '3') {
|
|
|
|
|
if (STRNEQ(custom->value, "2.8.20")) {
|
|
|
|
|
return -1;
|
|
|
|
|
}
|
|
|
|
|
} else if (custom->idx == '6') {
|
|
|
|
|
if (STRNEQ(custom->value, "1.5.35")) {
|
|
|
|
|
return -1;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2021-10-20 11:30:31 +03:00
|
|
|
|
static int
|
|
|
|
|
testVirPCIVPDParseFullVPDInvalid(const void *opaque G_GNUC_UNUSED)
|
|
|
|
|
{
|
|
|
|
|
size_t dataLen = 0;
|
|
|
|
|
|
|
|
|
|
# define VPD_INVALID_ZERO_BYTE \
|
|
|
|
|
0x00
|
|
|
|
|
|
|
|
|
|
# define VPD_INVALID_STRING_HEADER_DATA_LONG \
|
|
|
|
|
PCI_VPD_LARGE_RESOURCE_FLAG | PCI_VPD_STRING_RESOURCE_FLAG, 0x04, 0x00, \
|
|
|
|
|
VPD_STRING_RESOURCE_EXAMPLE_DATA, \
|
|
|
|
|
PCI_VPD_LARGE_RESOURCE_FLAG | PCI_VPD_READ_ONLY_LARGE_RESOURCE_FLAG, 0x05, 0x00, \
|
|
|
|
|
'R', 'V', 0x02, 0xDA, 0x00, \
|
|
|
|
|
PCI_VPD_RESOURCE_END_VAL
|
|
|
|
|
|
|
|
|
|
# define VPD_INVALID_STRING_HEADER_DATA_SHORT \
|
|
|
|
|
PCI_VPD_LARGE_RESOURCE_FLAG | PCI_VPD_STRING_RESOURCE_FLAG, 0x0A, 0x00, \
|
|
|
|
|
VPD_STRING_RESOURCE_EXAMPLE_DATA, \
|
|
|
|
|
PCI_VPD_LARGE_RESOURCE_FLAG | PCI_VPD_READ_ONLY_LARGE_RESOURCE_FLAG, 0x05, 0x00, \
|
|
|
|
|
'R', 'V', 0x02, 0xD4, 0x00, \
|
|
|
|
|
PCI_VPD_RESOURCE_END_VAL
|
|
|
|
|
|
|
|
|
|
# define VPD_NO_VPD_R \
|
|
|
|
|
VPD_STRING_RESOURCE_EXAMPLE_HEADER, \
|
|
|
|
|
VPD_STRING_RESOURCE_EXAMPLE_DATA, \
|
|
|
|
|
PCI_VPD_RESOURCE_END_VAL
|
|
|
|
|
|
|
|
|
|
# define VPD_R_NO_RV \
|
|
|
|
|
VPD_STRING_RESOURCE_EXAMPLE_HEADER, \
|
|
|
|
|
VPD_STRING_RESOURCE_EXAMPLE_DATA, \
|
|
|
|
|
VPD_R_FIELDS_EXAMPLE_HEADER, \
|
|
|
|
|
VPD_R_EXAMPLE_FIELDS, \
|
|
|
|
|
PCI_VPD_RESOURCE_END_VAL
|
|
|
|
|
|
|
|
|
|
# define VPD_R_INVALID_RV \
|
|
|
|
|
VPD_STRING_RESOURCE_EXAMPLE_HEADER, \
|
|
|
|
|
VPD_STRING_RESOURCE_EXAMPLE_DATA, \
|
|
|
|
|
VPD_R_FIELDS_EXAMPLE_HEADER, \
|
|
|
|
|
VPD_R_EXAMPLE_FIELDS, \
|
|
|
|
|
VPD_R_EXAMPLE_INVALID_RV_FIELD, \
|
|
|
|
|
PCI_VPD_RESOURCE_END_VAL
|
|
|
|
|
|
|
|
|
|
# define VPD_R_INVALID_RV_ZERO_LENGTH \
|
|
|
|
|
VPD_STRING_RESOURCE_EXAMPLE_HEADER, \
|
|
|
|
|
VPD_STRING_RESOURCE_EXAMPLE_DATA, \
|
|
|
|
|
PCI_VPD_LARGE_RESOURCE_FLAG | PCI_VPD_READ_ONLY_LARGE_RESOURCE_FLAG, 0x14, 0x00, \
|
|
|
|
|
VPD_R_EXAMPLE_FIELDS, \
|
|
|
|
|
'R', 'V', 0x00, \
|
|
|
|
|
PCI_VPD_RESOURCE_END_VAL
|
|
|
|
|
|
|
|
|
|
/* The RW key is not expected in a VPD-R record. */
|
|
|
|
|
# define VPD_R_UNEXPECTED_RW_IN_VPD_R_KEY \
|
|
|
|
|
VPD_STRING_RESOURCE_EXAMPLE_HEADER, \
|
|
|
|
|
VPD_STRING_RESOURCE_EXAMPLE_DATA, \
|
|
|
|
|
PCI_VPD_LARGE_RESOURCE_FLAG | PCI_VPD_READ_ONLY_LARGE_RESOURCE_FLAG, 0x1B, 0x00, \
|
|
|
|
|
VPD_R_EXAMPLE_FIELDS, \
|
|
|
|
|
'R', 'W', 0x02, 0x00, 0x00, \
|
|
|
|
|
'R', 'V', 0x02, 0x81, 0x00, \
|
|
|
|
|
PCI_VPD_RESOURCE_END_VAL
|
|
|
|
|
|
|
|
|
|
# define VPD_INVALID_STRING_RESOURCE_VALUE \
|
|
|
|
|
VPD_STRING_RESOURCE_EXAMPLE_HEADER, \
|
|
|
|
|
't', 0x03, 's', 't', 'n', 'a', 'm', 'e', \
|
|
|
|
|
PCI_VPD_LARGE_RESOURCE_FLAG | PCI_VPD_READ_ONLY_LARGE_RESOURCE_FLAG, 0x0A, 0x00, \
|
|
|
|
|
'S', 'N', 0x02, 0x04, 0x02, \
|
|
|
|
|
'R', 'V', 0x02, 0x8A, 0x00, \
|
|
|
|
|
PCI_VPD_RESOURCE_END_VAL
|
|
|
|
|
|
2021-10-29 21:57:16 +03:00
|
|
|
|
/* The SN field has a length field that goes past the resource boundaries. */
|
|
|
|
|
# define VPD_INVALID_SN_FIELD_LENGTH \
|
|
|
|
|
VPD_STRING_RESOURCE_EXAMPLE_HEADER, \
|
|
|
|
|
't', 'e', 's', 't', 'n', 'a', 'm', 'e', \
|
|
|
|
|
PCI_VPD_LARGE_RESOURCE_FLAG | PCI_VPD_READ_ONLY_LARGE_RESOURCE_FLAG, 0x0A, 0x00, \
|
|
|
|
|
'S', 'N', 0x42, 0x04, 0x02, \
|
|
|
|
|
'R', 'V', 0x02, 0xE8, 0x00, \
|
|
|
|
|
PCI_VPD_RESOURCE_END_VAL
|
|
|
|
|
|
|
|
|
|
/* The RV field is not the last one in VPD-R while the checksum is valid. */
|
|
|
|
|
# define VPD_INVALID_RV_NOT_LAST \
|
|
|
|
|
VPD_STRING_RESOURCE_EXAMPLE_HEADER, \
|
|
|
|
|
't', 'e', 's', 't', 'n', 'a', 'm', 'e', \
|
|
|
|
|
PCI_VPD_LARGE_RESOURCE_FLAG | PCI_VPD_READ_ONLY_LARGE_RESOURCE_FLAG, 0x0A, 0x00, \
|
|
|
|
|
'R', 'V', 0x02, 0xD1, 0x00, \
|
|
|
|
|
'S', 'N', 0x02, 0x04, 0x02, \
|
|
|
|
|
PCI_VPD_RESOURCE_END_VAL
|
|
|
|
|
|
|
|
|
|
# define VPD_INVALID_RW_NOT_LAST \
|
|
|
|
|
VPD_STRING_RESOURCE_EXAMPLE_HEADER, VPD_STRING_RESOURCE_EXAMPLE_DATA, \
|
|
|
|
|
VPD_R_FIELDS_EXAMPLE_HEADER, VPD_R_FIELDS_EXAMPLE_DATA, \
|
|
|
|
|
PCI_VPD_LARGE_RESOURCE_FLAG | PCI_VPD_READ_WRITE_LARGE_RESOURCE_FLAG, 0x08, 0x00, \
|
|
|
|
|
'R', 'W', 0x00, \
|
|
|
|
|
'V', 'Z', 0x02, '4', '2', \
|
|
|
|
|
PCI_VPD_RESOURCE_END_VAL
|
|
|
|
|
|
|
|
|
|
|
2021-10-20 11:30:31 +03:00
|
|
|
|
# define TEST_INVALID_VPD(invalidVPD) \
|
|
|
|
|
do { \
|
2022-02-10 11:32:00 +01:00
|
|
|
|
VIR_AUTOCLOSE fd = -1; \
|
2021-10-20 11:30:31 +03:00
|
|
|
|
g_autoptr(virPCIVPDResource) res = NULL; \
|
|
|
|
|
const uint8_t testCase[] = { invalidVPD }; \
|
2021-11-02 17:32:43 +01:00
|
|
|
|
dataLen = G_N_ELEMENTS(testCase); \
|
2021-11-23 15:05:43 +01:00
|
|
|
|
if ((fd = virCreateAnonymousFile(testCase, dataLen)) < 0) \
|
|
|
|
|
return -1; \
|
2021-10-20 11:30:31 +03:00
|
|
|
|
if ((res = virPCIVPDParse(fd))) { \
|
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s", \
|
|
|
|
|
"Successfully parsed an invalid VPD - this is not expected"); \
|
|
|
|
|
return -1; \
|
|
|
|
|
} \
|
|
|
|
|
} while (0);
|
|
|
|
|
|
|
|
|
|
TEST_INVALID_VPD(VPD_INVALID_ZERO_BYTE);
|
|
|
|
|
TEST_INVALID_VPD(VPD_INVALID_STRING_HEADER_DATA_SHORT);
|
|
|
|
|
TEST_INVALID_VPD(VPD_INVALID_STRING_HEADER_DATA_LONG);
|
|
|
|
|
TEST_INVALID_VPD(VPD_NO_VPD_R);
|
|
|
|
|
TEST_INVALID_VPD(VPD_R_NO_RV);
|
|
|
|
|
TEST_INVALID_VPD(VPD_R_INVALID_RV);
|
|
|
|
|
TEST_INVALID_VPD(VPD_R_INVALID_RV_ZERO_LENGTH);
|
|
|
|
|
TEST_INVALID_VPD(VPD_R_UNEXPECTED_RW_IN_VPD_R_KEY);
|
|
|
|
|
TEST_INVALID_VPD(VPD_INVALID_STRING_RESOURCE_VALUE);
|
2021-10-29 21:57:16 +03:00
|
|
|
|
TEST_INVALID_VPD(VPD_INVALID_SN_FIELD_LENGTH);
|
|
|
|
|
TEST_INVALID_VPD(VPD_INVALID_RV_NOT_LAST);
|
|
|
|
|
TEST_INVALID_VPD(VPD_INVALID_RW_NOT_LAST);
|
2021-10-20 11:30:31 +03:00
|
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static int
|
|
|
|
|
mymain(void)
|
|
|
|
|
{
|
|
|
|
|
int ret = 0;
|
|
|
|
|
|
|
|
|
|
if (virTestRun("Basic functionality of virPCIVPDResource ", testPCIVPDResourceBasic, NULL) < 0)
|
|
|
|
|
ret = -1;
|
|
|
|
|
if (virTestRun("Custom field index comparison",
|
|
|
|
|
testPCIVPDResourceCustomCompareIndex, NULL) < 0)
|
|
|
|
|
ret = -1;
|
|
|
|
|
if (virTestRun("Custom field value insertion and updates ",
|
|
|
|
|
testPCIVPDResourceCustomUpsertValue, NULL) < 0)
|
|
|
|
|
ret = -1;
|
|
|
|
|
if (virTestRun("Valid text values ", testPCIVPDIsValidTextValue, NULL) < 0)
|
|
|
|
|
ret = -1;
|
|
|
|
|
if (virTestRun("Determining a field value format by a key ",
|
|
|
|
|
testPCIVPDGetFieldValueFormat, NULL) < 0)
|
|
|
|
|
ret = -1;
|
2021-10-29 21:57:16 +03:00
|
|
|
|
if (virTestRun("Parsing a VPD resource with a zero-length RW ",
|
|
|
|
|
testVirPCIVPDParseZeroLengthRW, NULL) < 0)
|
|
|
|
|
ret = -1;
|
|
|
|
|
if (virTestRun("Parsing a VPD resource without an RW ",
|
|
|
|
|
testVirPCIVPDParseNoRW, NULL) < 0)
|
|
|
|
|
ret = -1;
|
2021-10-29 21:57:17 +03:00
|
|
|
|
if (virTestRun("Parsing a VPD resource with an invalid values ",
|
|
|
|
|
testVirPCIVPDParseFullVPDSkipInvalidValues, NULL) < 0)
|
|
|
|
|
ret = -1;
|
2021-10-20 11:30:31 +03:00
|
|
|
|
if (virTestRun("Parsing a VPD resource with an invalid keyword ",
|
|
|
|
|
testVirPCIVPDParseFullVPDSkipInvalidKeywords, NULL) < 0)
|
|
|
|
|
ret = -1;
|
|
|
|
|
if (virTestRun("Parsing VPD resources from a full VPD ", testVirPCIVPDParseFullVPD, NULL) < 0)
|
|
|
|
|
ret = -1;
|
|
|
|
|
if (virTestRun("Parsing invalid VPD records ", testVirPCIVPDParseFullVPDInvalid, NULL) < 0)
|
|
|
|
|
ret = -1;
|
|
|
|
|
|
|
|
|
|
return ret == 0 ? EXIT_SUCCESS : EXIT_FAILURE;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
VIR_TEST_MAIN(mymain)
|
|
|
|
|
|
|
|
|
|
#else /* ! __linux__ */
|
|
|
|
|
int
|
|
|
|
|
main(void)
|
|
|
|
|
{
|
|
|
|
|
return EXIT_AM_SKIP;
|
|
|
|
|
}
|
|
|
|
|
#endif /* ! __linux__ */
|