|  | /* | 
|  | * Intel MIC Platform Software Stack (MPSS) | 
|  | * | 
|  | * Copyright(c) 2013 Intel Corporation. | 
|  | * | 
|  | * This program is free software; you can redistribute it and/or modify | 
|  | * it under the terms of the GNU General Public License, version 2, as | 
|  | * published by the Free Software Foundation. | 
|  | * | 
|  | * This program is distributed in the hope that it will be useful, but | 
|  | * WITHOUT ANY WARRANTY; without even the implied warranty of | 
|  | * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU | 
|  | * General Public License for more details. | 
|  | * | 
|  | * The full GNU General Public License is included in this distribution in | 
|  | * the file called "COPYING". | 
|  | * | 
|  | * Intel MIC Host driver. | 
|  | * | 
|  | */ | 
|  | #ifndef MIC_VIRTIO_H | 
|  | #define MIC_VIRTIO_H | 
|  |  | 
|  | #include <linux/virtio_config.h> | 
|  | #include <linux/mic_ioctl.h> | 
|  |  | 
|  | /* | 
|  | * Note on endianness. | 
|  | * 1. Host can be both BE or LE | 
|  | * 2. Guest/card is LE. Host uses le_to_cpu to access desc/avail | 
|  | *    rings and ioreadXX/iowriteXX to access used ring. | 
|  | * 3. Device page exposed by host to guest contains LE values. Guest | 
|  | *    accesses these using ioreadXX/iowriteXX etc. This way in general we | 
|  | *    obey the virtio spec according to which guest works with native | 
|  | *    endianness and host is aware of guest endianness and does all | 
|  | *    required endianness conversion. | 
|  | * 4. Data provided from user space to guest (in ADD_DEVICE and | 
|  | *    CONFIG_CHANGE ioctl's) is not interpreted by the driver and should be | 
|  | *    in guest endianness. | 
|  | */ | 
|  |  | 
|  | /** | 
|  | * struct mic_vringh - Virtio ring host information. | 
|  | * | 
|  | * @vring: The MIC vring used for setting up user space mappings. | 
|  | * @vrh: The host VRINGH used for accessing the card vrings. | 
|  | * @riov: The VRINGH read kernel IOV. | 
|  | * @wiov: The VRINGH write kernel IOV. | 
|  | * @vr_mutex: Mutex for synchronizing access to the VRING. | 
|  | * @buf: Temporary kernel buffer used to copy in/out data | 
|  | * from/to the card via DMA. | 
|  | * @buf_da: dma address of buf. | 
|  | * @mvdev: Back pointer to MIC virtio device for vringh_notify(..). | 
|  | * @head: The VRINGH head index address passed to vringh_getdesc_kern(..). | 
|  | */ | 
|  | struct mic_vringh { | 
|  | struct mic_vring vring; | 
|  | struct vringh vrh; | 
|  | struct vringh_kiov riov; | 
|  | struct vringh_kiov wiov; | 
|  | struct mutex vr_mutex; | 
|  | void *buf; | 
|  | dma_addr_t buf_da; | 
|  | struct mic_vdev *mvdev; | 
|  | u16 head; | 
|  | }; | 
|  |  | 
|  | /** | 
|  | * struct mic_vdev - Host information for a card Virtio device. | 
|  | * | 
|  | * @virtio_id - Virtio device id. | 
|  | * @waitq - Waitqueue to allow ring3 apps to poll. | 
|  | * @mdev - Back pointer to host MIC device. | 
|  | * @poll_wake - Used for waking up threads blocked in poll. | 
|  | * @out_bytes - Debug stats for number of bytes copied from host to card. | 
|  | * @in_bytes - Debug stats for number of bytes copied from card to host. | 
|  | * @out_bytes_dma - Debug stats for number of bytes copied from host to card | 
|  | * using DMA. | 
|  | * @in_bytes_dma - Debug stats for number of bytes copied from card to host | 
|  | * using DMA. | 
|  | * @tx_len_unaligned - Debug stats for number of bytes copied to the card where | 
|  | * the transfer length did not have the required DMA alignment. | 
|  | * @tx_dst_unaligned - Debug stats for number of bytes copied where the | 
|  | * destination address on the card did not have the required DMA alignment. | 
|  | * @mvr - Store per VRING data structures. | 
|  | * @virtio_bh_work - Work struct used to schedule virtio bottom half handling. | 
|  | * @dd - Virtio device descriptor. | 
|  | * @dc - Virtio device control fields. | 
|  | * @list - List of Virtio devices. | 
|  | * @virtio_db - The doorbell used by the card to interrupt the host. | 
|  | * @virtio_cookie - The cookie returned while requesting interrupts. | 
|  | */ | 
|  | struct mic_vdev { | 
|  | int virtio_id; | 
|  | wait_queue_head_t waitq; | 
|  | struct mic_device *mdev; | 
|  | int poll_wake; | 
|  | unsigned long out_bytes; | 
|  | unsigned long in_bytes; | 
|  | unsigned long out_bytes_dma; | 
|  | unsigned long in_bytes_dma; | 
|  | unsigned long tx_len_unaligned; | 
|  | unsigned long tx_dst_unaligned; | 
|  | struct mic_vringh mvr[MIC_MAX_VRINGS]; | 
|  | struct work_struct virtio_bh_work; | 
|  | struct mic_device_desc *dd; | 
|  | struct mic_device_ctrl *dc; | 
|  | struct list_head list; | 
|  | int virtio_db; | 
|  | struct mic_irq *virtio_cookie; | 
|  | }; | 
|  |  | 
|  | void mic_virtio_uninit(struct mic_device *mdev); | 
|  | int mic_virtio_add_device(struct mic_vdev *mvdev, | 
|  | void __user *argp); | 
|  | void mic_virtio_del_device(struct mic_vdev *mvdev); | 
|  | int mic_virtio_config_change(struct mic_vdev *mvdev, | 
|  | void __user *argp); | 
|  | int mic_virtio_copy_desc(struct mic_vdev *mvdev, | 
|  | struct mic_copy_desc *request); | 
|  | void mic_virtio_reset_devices(struct mic_device *mdev); | 
|  | void mic_bh_handler(struct work_struct *work); | 
|  |  | 
|  | /* Helper API to obtain the MIC PCIe device */ | 
|  | static inline struct device *mic_dev(struct mic_vdev *mvdev) | 
|  | { | 
|  | return mvdev->mdev->sdev->parent; | 
|  | } | 
|  |  | 
|  | /* Helper API to check if a virtio device is initialized */ | 
|  | static inline int mic_vdev_inited(struct mic_vdev *mvdev) | 
|  | { | 
|  | /* Device has not been created yet */ | 
|  | if (!mvdev->dd || !mvdev->dd->type) { | 
|  | dev_err(mic_dev(mvdev), "%s %d err %d\n", | 
|  | __func__, __LINE__, -EINVAL); | 
|  | return -EINVAL; | 
|  | } | 
|  |  | 
|  | /* Device has been removed/deleted */ | 
|  | if (mvdev->dd->type == -1) { | 
|  | dev_err(mic_dev(mvdev), "%s %d err %d\n", | 
|  | __func__, __LINE__, -ENODEV); | 
|  | return -ENODEV; | 
|  | } | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | /* Helper API to check if a virtio device is running */ | 
|  | static inline bool mic_vdevup(struct mic_vdev *mvdev) | 
|  | { | 
|  | return !!mvdev->dd->status; | 
|  | } | 
|  | #endif |