virkmodtest.c 4.3 KB
Newer Older
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25
/*
 * Copyright (C) 2014 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"

#ifdef __linux__

# include <stdlib.h>
26 27
# define __VIR_COMMAND_PRIV_H_ALLOW__
# include "vircommandpriv.h"
28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54
# include "virkmod.h"
# include "virstring.h"

struct testInfo {
    const char *module;
    const char *exp_cmd;
    bool useBlacklist;
};

# define VIR_FROM_THIS VIR_FROM_NONE

static int
testKModConfig(const void *args ATTRIBUTE_UNUSED)
{
    int ret = -1;
    char *outbuf = NULL;

    /* This will return the contents of a 'modprobe -c' which can differ
     * from machine to machine - be happy that we get something.
     */
    outbuf = virKModConfig();
    if (!outbuf) {
        fprintf(stderr, "Failed to get config\n");
        goto cleanup;
    }
    ret = 0;

55
 cleanup:
56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76
    VIR_FREE(outbuf);
    return ret;
}


static int
checkOutput(virBufferPtr buf, const char *exp_cmd)
{
    int ret = -1;
    char *actual_cmd = NULL;

    if (!(actual_cmd = virBufferContentAndReset(buf))) {
        int err = virBufferError(buf);
        if (err)
            fprintf(stderr, "buffer's in error state: %d", err);
        else
            fprintf(stderr, "cannot compare buffer to exp: %s", exp_cmd);
        goto cleanup;
    }

    if (STRNEQ(exp_cmd, actual_cmd)) {
77
        virTestDifference(stderr, exp_cmd, actual_cmd);
78 79 80 81 82
        goto cleanup;
    }

    ret = 0;

83
 cleanup:
84 85 86 87 88 89 90 91 92 93 94 95 96 97 98
    VIR_FREE(actual_cmd);
    return ret;
}


static int
testKModLoad(const void *args)
{
    int ret = -1;
    char *errbuf = NULL;
    const struct testInfo *info = args;
    const char *module = info->module;
    bool useBlacklist = info->useBlacklist;
    virBuffer buf = VIR_BUFFER_INITIALIZER;

99
    virCommandSetDryRun(&buf, NULL, NULL);
100 101 102 103 104 105 106 107 108 109 110 111

    errbuf = virKModLoad(module, useBlacklist);
    if (errbuf) {
        fprintf(stderr, "Failed to load, error: %s\n", errbuf);
        goto cleanup;
    }

    if (checkOutput(&buf, info->exp_cmd) < 0)
        goto cleanup;

    ret = 0;

112
 cleanup:
113
    virCommandSetDryRun(NULL, NULL, NULL);
114 115 116 117 118 119 120 121 122 123 124 125 126 127
    VIR_FREE(errbuf);
    return ret;
}


static int
testKModUnload(const void *args)
{
    int ret = -1;
    char *errbuf = NULL;
    const struct testInfo *info = args;
    const char *module = info->module;
    virBuffer buf = VIR_BUFFER_INITIALIZER;

128
    virCommandSetDryRun(&buf, NULL, NULL);
129 130 131 132 133 134 135 136 137 138 139 140

    errbuf = virKModUnload(module);
    if (errbuf) {
        fprintf(stderr, "Failed to unload, error: %s\n", errbuf);
        goto cleanup;
    }

    if (checkOutput(&buf, info->exp_cmd) < 0)
        goto cleanup;

    ret = 0;

141
 cleanup:
142
    virCommandSetDryRun(NULL, NULL, NULL);
143 144 145 146 147 148 149 150 151 152
    VIR_FREE(errbuf);
    return ret;
}


static int
mymain(void)
{
    int ret = 0;

153
    if (virTestRun("config", testKModConfig, NULL) < 0)
154 155 156 157 158 159 160 161 162 163 164
        ret = -1;

    /* Although we cannot run the command on the host, we can compare
     * the output of the created command against what we'd expect to be
     * created. So let's at least do that.
     */
# define DO_TEST(_name, _cb, _blkflag, _exp_cmd)              \
    do {                                                      \
        struct testInfo data = {.module = "vfio-pci",         \
                                .exp_cmd = _exp_cmd,          \
                                .useBlacklist = _blkflag};    \
165
        if (virTestRun(_name, _cb,  &data) < 0)               \
166 167 168 169 170 171 172
            ret = -1;                                         \
    } while (0)

    DO_TEST("load", testKModLoad, false, MODPROBE " vfio-pci\n");
    DO_TEST("unload", testKModUnload, false, RMMOD " vfio-pci\n");
    DO_TEST("blklist", testKModLoad, true, MODPROBE " -b vfio-pci\n");

173
    return ret == 0 ? EXIT_SUCCESS : EXIT_FAILURE;
174 175 176 177 178 179 180 181 182 183 184

}

VIRT_TEST_MAIN(mymain);
#else
int
main(void)
{
    return EXIT_AM_SKIP;
}
#endif