| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244 |
- // SPDX-License-Identifier: GPL-2.0-only
- /*
- * Copyright (c) 2023, NVIDIA CORPORATION & AFFILIATES. All rights reserved
- */
- #include <linux/virtio_pci_admin.h>
- #include "virtio_pci_common.h"
- /*
- * virtio_pci_admin_has_legacy_io - Checks whether the legacy IO
- * commands are supported
- * @dev: VF pci_dev
- *
- * Returns true on success.
- */
- bool virtio_pci_admin_has_legacy_io(struct pci_dev *pdev)
- {
- struct virtio_device *virtio_dev = virtio_pci_vf_get_pf_dev(pdev);
- struct virtio_pci_device *vp_dev;
- if (!virtio_dev)
- return false;
- if (!virtio_has_feature(virtio_dev, VIRTIO_F_ADMIN_VQ))
- return false;
- vp_dev = to_vp_device(virtio_dev);
- if ((vp_dev->admin_vq.supported_cmds & VIRTIO_LEGACY_ADMIN_CMD_BITMAP) ==
- VIRTIO_LEGACY_ADMIN_CMD_BITMAP)
- return true;
- return false;
- }
- EXPORT_SYMBOL_GPL(virtio_pci_admin_has_legacy_io);
- static int virtio_pci_admin_legacy_io_write(struct pci_dev *pdev, u16 opcode,
- u8 offset, u8 size, u8 *buf)
- {
- struct virtio_device *virtio_dev = virtio_pci_vf_get_pf_dev(pdev);
- struct virtio_admin_cmd_legacy_wr_data *data;
- struct virtio_admin_cmd cmd = {};
- struct scatterlist data_sg;
- int vf_id;
- int ret;
- if (!virtio_dev)
- return -ENODEV;
- vf_id = pci_iov_vf_id(pdev);
- if (vf_id < 0)
- return vf_id;
- data = kzalloc(sizeof(*data) + size, GFP_KERNEL);
- if (!data)
- return -ENOMEM;
- data->offset = offset;
- memcpy(data->registers, buf, size);
- sg_init_one(&data_sg, data, sizeof(*data) + size);
- cmd.opcode = cpu_to_le16(opcode);
- cmd.group_type = cpu_to_le16(VIRTIO_ADMIN_GROUP_TYPE_SRIOV);
- cmd.group_member_id = cpu_to_le64(vf_id + 1);
- cmd.data_sg = &data_sg;
- ret = vp_modern_admin_cmd_exec(virtio_dev, &cmd);
- kfree(data);
- return ret;
- }
- /*
- * virtio_pci_admin_legacy_io_write_common - Write legacy common configuration
- * of a member device
- * @dev: VF pci_dev
- * @offset: starting byte offset within the common configuration area to write to
- * @size: size of the data to write
- * @buf: buffer which holds the data
- *
- * Note: caller must serialize access for the given device.
- * Returns 0 on success, or negative on failure.
- */
- int virtio_pci_admin_legacy_common_io_write(struct pci_dev *pdev, u8 offset,
- u8 size, u8 *buf)
- {
- return virtio_pci_admin_legacy_io_write(pdev,
- VIRTIO_ADMIN_CMD_LEGACY_COMMON_CFG_WRITE,
- offset, size, buf);
- }
- EXPORT_SYMBOL_GPL(virtio_pci_admin_legacy_common_io_write);
- /*
- * virtio_pci_admin_legacy_io_write_device - Write legacy device configuration
- * of a member device
- * @dev: VF pci_dev
- * @offset: starting byte offset within the device configuration area to write to
- * @size: size of the data to write
- * @buf: buffer which holds the data
- *
- * Note: caller must serialize access for the given device.
- * Returns 0 on success, or negative on failure.
- */
- int virtio_pci_admin_legacy_device_io_write(struct pci_dev *pdev, u8 offset,
- u8 size, u8 *buf)
- {
- return virtio_pci_admin_legacy_io_write(pdev,
- VIRTIO_ADMIN_CMD_LEGACY_DEV_CFG_WRITE,
- offset, size, buf);
- }
- EXPORT_SYMBOL_GPL(virtio_pci_admin_legacy_device_io_write);
- static int virtio_pci_admin_legacy_io_read(struct pci_dev *pdev, u16 opcode,
- u8 offset, u8 size, u8 *buf)
- {
- struct virtio_device *virtio_dev = virtio_pci_vf_get_pf_dev(pdev);
- struct virtio_admin_cmd_legacy_rd_data *data;
- struct scatterlist data_sg, result_sg;
- struct virtio_admin_cmd cmd = {};
- int vf_id;
- int ret;
- if (!virtio_dev)
- return -ENODEV;
- vf_id = pci_iov_vf_id(pdev);
- if (vf_id < 0)
- return vf_id;
- data = kzalloc(sizeof(*data), GFP_KERNEL);
- if (!data)
- return -ENOMEM;
- data->offset = offset;
- sg_init_one(&data_sg, data, sizeof(*data));
- sg_init_one(&result_sg, buf, size);
- cmd.opcode = cpu_to_le16(opcode);
- cmd.group_type = cpu_to_le16(VIRTIO_ADMIN_GROUP_TYPE_SRIOV);
- cmd.group_member_id = cpu_to_le64(vf_id + 1);
- cmd.data_sg = &data_sg;
- cmd.result_sg = &result_sg;
- ret = vp_modern_admin_cmd_exec(virtio_dev, &cmd);
- kfree(data);
- return ret;
- }
- /*
- * virtio_pci_admin_legacy_device_io_read - Read legacy device configuration of
- * a member device
- * @dev: VF pci_dev
- * @offset: starting byte offset within the device configuration area to read from
- * @size: size of the data to be read
- * @buf: buffer to hold the returned data
- *
- * Note: caller must serialize access for the given device.
- * Returns 0 on success, or negative on failure.
- */
- int virtio_pci_admin_legacy_device_io_read(struct pci_dev *pdev, u8 offset,
- u8 size, u8 *buf)
- {
- return virtio_pci_admin_legacy_io_read(pdev,
- VIRTIO_ADMIN_CMD_LEGACY_DEV_CFG_READ,
- offset, size, buf);
- }
- EXPORT_SYMBOL_GPL(virtio_pci_admin_legacy_device_io_read);
- /*
- * virtio_pci_admin_legacy_common_io_read - Read legacy common configuration of
- * a member device
- * @dev: VF pci_dev
- * @offset: starting byte offset within the common configuration area to read from
- * @size: size of the data to be read
- * @buf: buffer to hold the returned data
- *
- * Note: caller must serialize access for the given device.
- * Returns 0 on success, or negative on failure.
- */
- int virtio_pci_admin_legacy_common_io_read(struct pci_dev *pdev, u8 offset,
- u8 size, u8 *buf)
- {
- return virtio_pci_admin_legacy_io_read(pdev,
- VIRTIO_ADMIN_CMD_LEGACY_COMMON_CFG_READ,
- offset, size, buf);
- }
- EXPORT_SYMBOL_GPL(virtio_pci_admin_legacy_common_io_read);
- /*
- * virtio_pci_admin_legacy_io_notify_info - Read the queue notification
- * information for legacy interface
- * @dev: VF pci_dev
- * @req_bar_flags: requested bar flags
- * @bar: on output the BAR number of the owner or member device
- * @bar_offset: on output the offset within bar
- *
- * Returns 0 on success, or negative on failure.
- */
- int virtio_pci_admin_legacy_io_notify_info(struct pci_dev *pdev,
- u8 req_bar_flags, u8 *bar,
- u64 *bar_offset)
- {
- struct virtio_device *virtio_dev = virtio_pci_vf_get_pf_dev(pdev);
- struct virtio_admin_cmd_notify_info_result *result;
- struct virtio_admin_cmd cmd = {};
- struct scatterlist result_sg;
- int vf_id;
- int ret;
- if (!virtio_dev)
- return -ENODEV;
- vf_id = pci_iov_vf_id(pdev);
- if (vf_id < 0)
- return vf_id;
- result = kzalloc(sizeof(*result), GFP_KERNEL);
- if (!result)
- return -ENOMEM;
- sg_init_one(&result_sg, result, sizeof(*result));
- cmd.opcode = cpu_to_le16(VIRTIO_ADMIN_CMD_LEGACY_NOTIFY_INFO);
- cmd.group_type = cpu_to_le16(VIRTIO_ADMIN_GROUP_TYPE_SRIOV);
- cmd.group_member_id = cpu_to_le64(vf_id + 1);
- cmd.result_sg = &result_sg;
- ret = vp_modern_admin_cmd_exec(virtio_dev, &cmd);
- if (!ret) {
- struct virtio_admin_cmd_notify_info_data *entry;
- int i;
- ret = -ENOENT;
- for (i = 0; i < VIRTIO_ADMIN_CMD_MAX_NOTIFY_INFO; i++) {
- entry = &result->entries[i];
- if (entry->flags == VIRTIO_ADMIN_CMD_NOTIFY_INFO_FLAGS_END)
- break;
- if (entry->flags != req_bar_flags)
- continue;
- *bar = entry->bar;
- *bar_offset = le64_to_cpu(entry->offset);
- ret = 0;
- break;
- }
- }
- kfree(result);
- return ret;
- }
- EXPORT_SYMBOL_GPL(virtio_pci_admin_legacy_io_notify_info);
|