mirror of
https://github.com/torvalds/linux.git
synced 2024-11-12 23:23:03 +00:00
245 lines
6.8 KiB
C
245 lines
6.8 KiB
C
|
// 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);
|