| /* SPDX-License-Identifier: GPL-2.0-only */ |
| |
| #ifndef __KVM_IODEV_H__ |
| #define __KVM_IODEV_H__ |
| |
| #include <linux/kvm_types.h> |
| #include <linux/errno.h> |
| |
| struct kvm_io_device; |
| struct kvm_vcpu; |
| |
| /** |
| * kvm_io_device_ops are called under kvm slots_lock. |
| * read and write handlers return 0 if the transaction has been handled, |
| * or non-zero to have it passed to the next device. |
| **/ |
| struct kvm_io_device_ops { |
| int (*read)(struct kvm_vcpu *vcpu, |
| struct kvm_io_device *this, |
| gpa_t addr, |
| int len, |
| void *val); |
| int (*write)(struct kvm_vcpu *vcpu, |
| struct kvm_io_device *this, |
| gpa_t addr, |
| int len, |
| const void *val); |
| void (*destructor)(struct kvm_io_device *this); |
| }; |
| |
| |
| struct kvm_io_device { |
| const struct kvm_io_device_ops *ops; |
| }; |
| |
| static inline void kvm_iodevice_init(struct kvm_io_device *dev, |
| const struct kvm_io_device_ops *ops) |
| { |
| dev->ops = ops; |
| } |
| |
| static inline int kvm_iodevice_read(struct kvm_vcpu *vcpu, |
| struct kvm_io_device *dev, gpa_t addr, |
| int l, void *v) |
| { |
| return dev->ops->read ? dev->ops->read(vcpu, dev, addr, l, v) |
| : -EOPNOTSUPP; |
| } |
| |
| static inline int kvm_iodevice_write(struct kvm_vcpu *vcpu, |
| struct kvm_io_device *dev, gpa_t addr, |
| int l, const void *v) |
| { |
| return dev->ops->write ? dev->ops->write(vcpu, dev, addr, l, v) |
| : -EOPNOTSUPP; |
| } |
| |
| static inline void kvm_iodevice_destructor(struct kvm_io_device *dev) |
| { |
| if (dev->ops->destructor) |
| dev->ops->destructor(dev); |
| } |
| |
| #endif /* __KVM_IODEV_H__ */ |