#ifndef _DEV_PV_VIRTIOVAR_H_
#define _DEV_PV_VIRTIOVAR_H_
#include <sys/param.h>
#include <sys/queue.h>
#include <sys/device.h>
#include <sys/mutex.h>
#include <machine/bus.h>
#include <dev/pv/virtioreg.h>
#ifndef VIRTIO_DEBUG
#define VIRTIO_DEBUG 0
#endif
#define VIRTIO_CF_NO_INDIRECT 1
#define VIRTIO_CF_NO_EVENT_IDX 2
#define VIRTIO_CF_PREFER_VERSION_09 8
struct virtio_attach_args {
int va_devid;
unsigned int va_nintr;
};
struct vq_entry {
SLIST_ENTRY(vq_entry) qe_list;
uint16_t qe_index;
int16_t qe_next;
int16_t qe_indirect;
int16_t qe_vr_index;
struct vring_desc *qe_desc_base;
};
struct virtqueue {
struct virtio_softc *vq_owner;
unsigned int vq_num;
unsigned int vq_mask;
int vq_index;
struct vring_desc *vq_desc;
struct vring_avail *vq_avail;
struct vring_used *vq_used;
void *vq_indirect;
void *vq_vaddr;
int vq_availoffset;
int vq_usedoffset;
int vq_indirectoffset;
bus_dma_segment_t vq_segs[1];
unsigned int vq_bytesize;
bus_dmamap_t vq_dmamap;
int vq_maxnsegs;
struct vq_entry *vq_entries;
SLIST_HEAD(, vq_entry) vq_freelist;
uint16_t vq_avail_idx;
uint16_t vq_used_idx;
int vq_queued;
int (*vq_done)(struct virtqueue*);
uint32_t vq_notify_off;
int vq_intr_vec;
};
struct virtio_feature_name {
uint64_t bit;
const char *name;
};
struct virtio_ops {
void (*kick)(struct virtio_softc *, uint16_t);
uint8_t (*read_dev_cfg_1)(struct virtio_softc *, int);
uint16_t (*read_dev_cfg_2)(struct virtio_softc *, int);
uint32_t (*read_dev_cfg_4)(struct virtio_softc *, int);
uint64_t (*read_dev_cfg_8)(struct virtio_softc *, int);
void (*write_dev_cfg_1)(struct virtio_softc *, int, uint8_t);
void (*write_dev_cfg_2)(struct virtio_softc *, int, uint16_t);
void (*write_dev_cfg_4)(struct virtio_softc *, int, uint32_t);
void (*write_dev_cfg_8)(struct virtio_softc *, int, uint64_t);
uint16_t (*read_queue_size)(struct virtio_softc *, uint16_t);
void (*setup_queue)(struct virtio_softc *, struct virtqueue *, uint64_t);
void (*setup_intrs)(struct virtio_softc *);
int (*get_status)(struct virtio_softc *);
void (*set_status)(struct virtio_softc *, int);
int (*neg_features)(struct virtio_softc *, const struct virtio_feature_name *);
int (*attach_finish)(struct virtio_softc *, struct virtio_attach_args *);
int (*poll_intr)(void *);
void (*intr_barrier)(struct virtio_softc *);
int (*intr_establish)(struct virtio_softc *, struct virtio_attach_args *,
int, struct cpu_info *, int (*)(void *), void *);
};
#define VIRTIO_CHILD_ERROR ((void*)1)
struct virtio_softc {
struct device sc_dev;
bus_dma_tag_t sc_dmat;
const struct virtio_ops *sc_ops;
int sc_ipl;
uint64_t sc_driver_features;
uint64_t sc_active_features;
int sc_indirect;
int sc_version_1;
int sc_nvqs;
struct virtqueue *sc_vqs;
struct device *sc_child;
int (*sc_config_change)(struct virtio_softc*);
};
#define virtio_read_device_config_1(sc, o) (sc)->sc_ops->read_dev_cfg_1(sc, o)
#define virtio_read_device_config_2(sc, o) (sc)->sc_ops->read_dev_cfg_2(sc, o)
#define virtio_read_device_config_4(sc, o) (sc)->sc_ops->read_dev_cfg_4(sc, o)
#define virtio_read_device_config_8(sc, o) (sc)->sc_ops->read_dev_cfg_8(sc, o)
#define virtio_write_device_config_1(sc, o, v) (sc)->sc_ops->write_dev_cfg_1(sc, o, v)
#define virtio_write_device_config_2(sc, o, v) (sc)->sc_ops->write_dev_cfg_2(sc, o, v)
#define virtio_write_device_config_4(sc, o, v) (sc)->sc_ops->write_dev_cfg_4(sc, o, v)
#define virtio_write_device_config_8(sc, o, v) (sc)->sc_ops->write_dev_cfg_8(sc, o, v)
#define virtio_read_queue_size(sc, i) (sc)->sc_ops->read_queue_size(sc, i)
#define virtio_setup_queue(sc, i, v) (sc)->sc_ops->setup_queue(sc, i, v)
#define virtio_negotiate_features(sc, n) (sc)->sc_ops->neg_features(sc, n)
#define virtio_poll_intr(sc) (sc)->sc_ops->poll_intr(sc)
#define virtio_get_status(sc) (sc)->sc_ops->get_status(sc)
#define virtio_set_status(sc, i) (sc)->sc_ops->set_status(sc, i)
#define virtio_intr_barrier(sc) (sc)->sc_ops->intr_barrier(sc)
#define virtio_intr_establish(sc, va, v, ci, fn, a) (sc)->sc_ops->intr_establish(sc, va, v, ci, fn, a)
#define virtio_device_reset(sc) virtio_set_status((sc), 0)
static inline int
virtio_has_feature(struct virtio_softc *sc, uint64_t fbit)
{
if (sc->sc_active_features & fbit)
return 1;
return 0;
}
int virtio_alloc_vq(struct virtio_softc*, struct virtqueue*, int, int,
const char*);
int virtio_free_vq(struct virtio_softc*, struct virtqueue*);
int virtio_attach_finish(struct virtio_softc *, struct virtio_attach_args *);
void virtio_reset(struct virtio_softc *);
void virtio_reinit_start(struct virtio_softc *);
void virtio_reinit_end(struct virtio_softc *);
int virtio_enqueue_prep(struct virtqueue*, int*);
int virtio_enqueue_reserve(struct virtqueue*, int, int);
int virtio_enqueue(struct virtqueue*, int, bus_dmamap_t, int);
int virtio_enqueue_p(struct virtqueue*, int, bus_dmamap_t, bus_addr_t,
bus_size_t, int);
void virtio_enqueue_commit(struct virtio_softc*, struct virtqueue*, int, int);
#define virtio_notify(sc,vq) virtio_enqueue_commit(sc, vq, -1, 1)
int virtio_enqueue_abort(struct virtqueue*, int);
void virtio_enqueue_trim(struct virtqueue*, int, int);
int virtio_dequeue(struct virtio_softc*, struct virtqueue*, int *, int *);
int virtio_dequeue_commit(struct virtqueue*, int);
int virtio_check_vqs(struct virtio_softc *);
int virtio_check_vq(struct virtio_softc *, struct virtqueue *);
void virtio_stop_vq_intr(struct virtio_softc *, struct virtqueue *);
int virtio_start_vq_intr(struct virtio_softc *, struct virtqueue *);
const char *virtio_device_string(int);
#if VIRTIO_DEBUG
void virtio_log_features(uint64_t, uint64_t, const struct virtio_feature_name *);
void virtio_vq_dump(struct virtqueue *vq);
#endif
int virtio_nused(struct virtqueue *vq);
int virtio_postpone_intr(struct virtqueue *vq, uint16_t nslots);
int virtio_postpone_intr_smart(struct virtqueue *vq);
int virtio_postpone_intr_far(struct virtqueue *vq);
#endif