// SPDX-License-Identifier: GPL-2.0-only /* * Copyright (c) 2023, HiSilicon Ltd. */ #include #include #include #include #include "vfio.h" static struct dentry *vfio_debugfs_root; static int vfio_device_state_read(struct seq_file *seq, void *data) { struct device *vf_dev = seq->private; struct vfio_device *vdev = container_of(vf_dev, struct vfio_device, device); enum vfio_device_mig_state state; int ret; BUILD_BUG_ON(VFIO_DEVICE_STATE_NR != VFIO_DEVICE_STATE_PRE_COPY_P2P + 1); ret = vdev->mig_ops->migration_get_state(vdev, &state); if (ret) return -EINVAL; switch (state) { case VFIO_DEVICE_STATE_ERROR: seq_puts(seq, "ERROR\n"); break; case VFIO_DEVICE_STATE_STOP: seq_puts(seq, "STOP\n"); break; case VFIO_DEVICE_STATE_RUNNING: seq_puts(seq, "RUNNING\n"); break; case VFIO_DEVICE_STATE_STOP_COPY: seq_puts(seq, "STOP_COPY\n"); break; case VFIO_DEVICE_STATE_RESUMING: seq_puts(seq, "RESUMING\n"); break; case VFIO_DEVICE_STATE_RUNNING_P2P: seq_puts(seq, "RUNNING_P2P\n"); break; case VFIO_DEVICE_STATE_PRE_COPY: seq_puts(seq, "PRE_COPY\n"); break; case VFIO_DEVICE_STATE_PRE_COPY_P2P: seq_puts(seq, "PRE_COPY_P2P\n"); break; default: seq_puts(seq, "Invalid\n"); } return 0; } void vfio_device_debugfs_init(struct vfio_device *vdev) { struct device *dev = &vdev->device; vdev->debug_root = debugfs_create_dir(dev_name(vdev->dev), vfio_debugfs_root); if (vdev->mig_ops) { struct dentry *vfio_dev_migration = NULL; vfio_dev_migration = debugfs_create_dir("migration", vdev->debug_root); debugfs_create_devm_seqfile(dev, "state", vfio_dev_migration, vfio_device_state_read); } } void vfio_device_debugfs_exit(struct vfio_device *vdev) { debugfs_remove_recursive(vdev->debug_root); } void vfio_debugfs_create_root(void) { vfio_debugfs_root = debugfs_create_dir("vfio", NULL); } void vfio_debugfs_remove_root(void) { debugfs_remove_recursive(vfio_debugfs_root); vfio_debugfs_root = NULL; }