blob: 619ccf9ba96e54b910df6adedbd1976397d1a49a [file] [log] [blame]
/*
* Copyright (c) 2020 Nutanix Inc. All rights reserved.
*
* Authors: Thanos Makatos <thanos@nutanix.com>
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions are met:
* * Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* * Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution.
* * Neither the name of Nutanix nor the names of its contributors may be
* used to endorse or promote products derived from this software without
* specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
* AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
* ARE DISCLAIMED. IN NO EVENT SHALL <COPYRIGHT HOLDER> BE LIABLE FOR ANY
* DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
* (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
* SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
* CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
* LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
* OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH
* DAMAGE.
*
*/
#include <dlfcn.h>
#include <setjmp.h>
#include <stdarg.h>
#include <stddef.h>
#include <string.h>
#include <stdio.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <cmocka.h>
#include "dma.h"
#include "migration.h"
#include "mocks.h"
#include "private.h"
#include "tran_sock.h"
#include "migration_priv.h"
struct function
{
const char *name;
bool patched;
};
static int (*__real_close)(int);
static struct function funcs[] = {
/* mocked internal funcs */
{ .name = "cmd_allowed_when_stopped_and_copying" },
{ .name = "device_is_stopped_and_copying" },
{ .name = "device_is_stopped" },
{ .name = "dma_controller_add_region" },
{ .name = "dma_controller_remove_region" },
{ .name = "dma_controller_unmap_region" },
{ .name = "should_exec_command" },
{ .name = "migration_region_access_registers" },
{ .name = "handle_device_state" },
{. name = "vfio_migr_state_transition_is_valid" },
{ .name = "state_trans_notify" },
{ .name = "migr_trans_to_valid_state" },
{ .name = "migr_state_vfio_to_vfu" },
{ .name = "migr_state_transition" },
/* system libs */
{ .name = "bind" },
{ .name = "close" },
{ .name = "listen" },
};
static struct function *
find(const char *name)
{
size_t i;
for (i = 0; i < ARRAY_SIZE(funcs); i++) {
if (strcmp(name, funcs[i].name) == 0) {
return &funcs[i];
}
}
assert(false);
}
void
patch(const char *name)
{
struct function *func = find(name);
func->patched = true;
}
static bool
is_patched(const char *name)
{
return find(name)->patched;
}
void
unpatch_all(void)
{
size_t i;
for (i = 0; i < ARRAY_SIZE(funcs); i++) {
funcs[i].patched = false;
}
}
int
dma_controller_add_region(dma_controller_t *dma, void *dma_addr,
size_t size, int fd, off_t offset,
uint32_t prot)
{
if (!is_patched("dma_controller_add_region")) {
return __real_dma_controller_add_region(dma, dma_addr, size, fd, offset,
prot);
}
check_expected_ptr(dma);
check_expected(dma_addr);
check_expected(size);
check_expected(fd);
check_expected(offset);
check_expected(prot);
errno = mock();
return mock();
}
int
dma_controller_remove_region(dma_controller_t *dma,
void *dma_addr, size_t size,
vfu_dma_unregister_cb_t *dma_unregister,
void *data)
{
if (!is_patched("dma_controller_remove_region")) {
return __real_dma_controller_remove_region(dma, dma_addr, size,
dma_unregister, data);
}
check_expected(dma);
check_expected(dma_addr);
check_expected(size);
check_expected(dma_unregister);
check_expected(data);
return mock();
}
void
dma_controller_unmap_region(dma_controller_t *dma,
dma_memory_region_t *region)
{
check_expected(dma);
check_expected(region);
}
bool
device_is_stopped(struct migration *migration)
{
if (!is_patched("device_is_stopped")) {
return __real_device_is_stopped(migration);
}
check_expected(migration);
return mock();
}
bool
device_is_stopped_and_copying(struct migration *migration)
{
if (!is_patched("device_is_stopped_and_copying")) {
return __real_device_is_stopped_and_copying(migration);
}
check_expected(migration);
return mock();
}
bool
cmd_allowed_when_stopped_and_copying(uint16_t cmd)
{
if (!is_patched("cmd_allowed_when_stopped_and_copying")) {
return __real_cmd_allowed_when_stopped_and_copying(cmd);
}
check_expected(cmd);
return mock();
}
bool
should_exec_command(vfu_ctx_t *vfu_ctx, uint16_t cmd)
{
if (!is_patched("should_exec_command")) {
return __real_should_exec_command(vfu_ctx, cmd);
}
check_expected(vfu_ctx);
check_expected(cmd);
return mock();
}
ssize_t
migration_region_access_registers(vfu_ctx_t *vfu_ctx, char *buf, size_t count,
loff_t pos, bool is_write)
{
if (!is_patched("migration_region_access_registers")) {
return __real_migration_region_access_registers(vfu_ctx, buf, count,
pos, is_write);
}
check_expected(vfu_ctx);
check_expected(buf);
check_expected(count);
check_expected(pos);
check_expected(is_write);
errno = mock();
return mock();
}
ssize_t
handle_device_state(vfu_ctx_t *vfu_ctx, struct migration *migr,
uint32_t device_state, bool notify) {
if (!is_patched("handle_device_state")) {
return __real_handle_device_state(vfu_ctx, migr, device_state,
notify);
}
check_expected(vfu_ctx);
check_expected(migr);
check_expected(device_state);
check_expected(notify);
return mock();
}
void
migr_state_transition(struct migration *migr, enum migr_iter_state state)
{
if (!is_patched("migr_state_transition")) {
__real_migr_state_transition(migr, state);
return;
}
check_expected(migr);
check_expected(state);
}
bool
vfio_migr_state_transition_is_valid(uint32_t from, uint32_t to)
{
if (!is_patched("vfio_migr_state_transition_is_valid")) {
return __real_vfio_migr_state_transition_is_valid(from, to);
}
check_expected(from);
check_expected(to);
return mock();
}
int
state_trans_notify(vfu_ctx_t *vfu_ctx, int (*fn)(vfu_ctx_t*, vfu_migr_state_t),
uint32_t vfio_device_state)
{
if (!is_patched("state_trans_notify")) {
return __real_state_trans_notify(vfu_ctx, fn, vfio_device_state);
}
check_expected(vfu_ctx);
check_expected(fn);
check_expected(vfio_device_state);
errno = mock();
return mock();
}
ssize_t
migr_trans_to_valid_state(vfu_ctx_t *vfu_ctx, struct migration *migr,
uint32_t device_state, bool notify)
{
if (!is_patched("migr_trans_to_valid_state")) {
return __real_migr_trans_to_valid_state(vfu_ctx, migr, device_state,
notify);
}
check_expected(vfu_ctx);
check_expected(migr);
check_expected(device_state);
check_expected(notify);
return mock();
}
vfu_migr_state_t
migr_state_vfio_to_vfu(uint32_t vfio_device_state)
{
if (!is_patched("migr_state_vfio_to_vfu")) {
return __real_migr_state_vfio_to_vfu(vfio_device_state);
}
check_expected(vfio_device_state);
return mock();
}
/* Always mocked. */
void
mock_dma_register(vfu_ctx_t *vfu_ctx, vfu_dma_info_t *info)
{
check_expected(vfu_ctx);
check_expected(info);
}
void
mock_dma_unregister(vfu_ctx_t *vfu_ctx, vfu_dma_info_t *info)
{
check_expected(vfu_ctx);
check_expected(info);
}
int
mock_reset_cb(vfu_ctx_t *vfu_ctx, vfu_reset_type_t type)
{
check_expected(vfu_ctx);
check_expected(type);
return mock();
}
int
mock_notify_migr_state_trans_cb(vfu_ctx_t *vfu_ctx, vfu_migr_state_t vfu_state)
{
check_expected(vfu_ctx);
check_expected(vfu_state);
return mock();
}
/* System-provided funcs. */
int
bind(int sockfd UNUSED, const struct sockaddr *addr UNUSED,
socklen_t addrlen UNUSED)
{
return 0;
}
int
close(int fd)
{
if (!is_patched("close")) {
if (__real_close == NULL) {
__real_close = dlsym(RTLD_NEXT, "close");
}
return __real_close(fd);
}
check_expected(fd);
return mock();
}
int
listen(int sockfd UNUSED, int backlog UNUSED)
{
return 0;
}
/* ex: set tabstop=4 shiftwidth=4 softtabstop=4 expandtab: */