162306a36Sopenharmony_ci/* SPDX-License-Identifier: GPL-2.0-only */
262306a36Sopenharmony_ci
362306a36Sopenharmony_ci#ifndef __KVM_IODEV_H__
462306a36Sopenharmony_ci#define __KVM_IODEV_H__
562306a36Sopenharmony_ci
662306a36Sopenharmony_ci#include <linux/kvm_types.h>
762306a36Sopenharmony_ci#include <linux/errno.h>
862306a36Sopenharmony_ci
962306a36Sopenharmony_cistruct kvm_io_device;
1062306a36Sopenharmony_cistruct kvm_vcpu;
1162306a36Sopenharmony_ci
1262306a36Sopenharmony_ci/**
1362306a36Sopenharmony_ci * kvm_io_device_ops are called under kvm slots_lock.
1462306a36Sopenharmony_ci * read and write handlers return 0 if the transaction has been handled,
1562306a36Sopenharmony_ci * or non-zero to have it passed to the next device.
1662306a36Sopenharmony_ci **/
1762306a36Sopenharmony_cistruct kvm_io_device_ops {
1862306a36Sopenharmony_ci	int (*read)(struct kvm_vcpu *vcpu,
1962306a36Sopenharmony_ci		    struct kvm_io_device *this,
2062306a36Sopenharmony_ci		    gpa_t addr,
2162306a36Sopenharmony_ci		    int len,
2262306a36Sopenharmony_ci		    void *val);
2362306a36Sopenharmony_ci	int (*write)(struct kvm_vcpu *vcpu,
2462306a36Sopenharmony_ci		     struct kvm_io_device *this,
2562306a36Sopenharmony_ci		     gpa_t addr,
2662306a36Sopenharmony_ci		     int len,
2762306a36Sopenharmony_ci		     const void *val);
2862306a36Sopenharmony_ci	void (*destructor)(struct kvm_io_device *this);
2962306a36Sopenharmony_ci};
3062306a36Sopenharmony_ci
3162306a36Sopenharmony_ci
3262306a36Sopenharmony_cistruct kvm_io_device {
3362306a36Sopenharmony_ci	const struct kvm_io_device_ops *ops;
3462306a36Sopenharmony_ci};
3562306a36Sopenharmony_ci
3662306a36Sopenharmony_cistatic inline void kvm_iodevice_init(struct kvm_io_device *dev,
3762306a36Sopenharmony_ci				     const struct kvm_io_device_ops *ops)
3862306a36Sopenharmony_ci{
3962306a36Sopenharmony_ci	dev->ops = ops;
4062306a36Sopenharmony_ci}
4162306a36Sopenharmony_ci
4262306a36Sopenharmony_cistatic inline int kvm_iodevice_read(struct kvm_vcpu *vcpu,
4362306a36Sopenharmony_ci				    struct kvm_io_device *dev, gpa_t addr,
4462306a36Sopenharmony_ci				    int l, void *v)
4562306a36Sopenharmony_ci{
4662306a36Sopenharmony_ci	return dev->ops->read ? dev->ops->read(vcpu, dev, addr, l, v)
4762306a36Sopenharmony_ci				: -EOPNOTSUPP;
4862306a36Sopenharmony_ci}
4962306a36Sopenharmony_ci
5062306a36Sopenharmony_cistatic inline int kvm_iodevice_write(struct kvm_vcpu *vcpu,
5162306a36Sopenharmony_ci				     struct kvm_io_device *dev, gpa_t addr,
5262306a36Sopenharmony_ci				     int l, const void *v)
5362306a36Sopenharmony_ci{
5462306a36Sopenharmony_ci	return dev->ops->write ? dev->ops->write(vcpu, dev, addr, l, v)
5562306a36Sopenharmony_ci				 : -EOPNOTSUPP;
5662306a36Sopenharmony_ci}
5762306a36Sopenharmony_ci
5862306a36Sopenharmony_ci#endif /* __KVM_IODEV_H__ */
59