mirror of
https://github.com/qemu/qemu.git
synced 2024-11-25 20:03:37 +08:00
e8637c9013
The request completion callback of the LSI controller may start the next request that can use the same tag as the completed one. As the latter is still enqueued at that point, scsi_send_command will complain about the tag reuse and cancel the completed request. That will cause a double free later on when the completion path cleans up as well. Fix this by dequeuing the request before invoking the callback. Signed-off-by: Jan Kiszka <jan.kiszka@siemens.com> Signed-off-by: Aurelien Jarno <aurelien@aurel32.net>
116 lines
3.1 KiB
C
116 lines
3.1 KiB
C
#ifndef QEMU_HW_SCSI_H
|
|
#define QEMU_HW_SCSI_H
|
|
|
|
#include "qdev.h"
|
|
#include "block.h"
|
|
#include "block_int.h"
|
|
|
|
#define SCSI_CMD_BUF_SIZE 16
|
|
|
|
/* scsi-disk.c */
|
|
enum scsi_reason {
|
|
SCSI_REASON_DONE, /* Command complete. */
|
|
SCSI_REASON_DATA /* Transfer complete, more data required. */
|
|
};
|
|
|
|
typedef struct SCSIBus SCSIBus;
|
|
typedef struct SCSIDevice SCSIDevice;
|
|
typedef struct SCSIDeviceInfo SCSIDeviceInfo;
|
|
typedef void (*scsi_completionfn)(SCSIBus *bus, int reason, uint32_t tag,
|
|
uint32_t arg);
|
|
|
|
enum SCSIXferMode {
|
|
SCSI_XFER_NONE, /* TEST_UNIT_READY, ... */
|
|
SCSI_XFER_FROM_DEV, /* READ, INQUIRY, MODE_SENSE, ... */
|
|
SCSI_XFER_TO_DEV, /* WRITE, MODE_SELECT, ... */
|
|
};
|
|
|
|
typedef struct SCSISense {
|
|
uint8_t key;
|
|
} SCSISense;
|
|
|
|
typedef struct SCSIRequest {
|
|
SCSIBus *bus;
|
|
SCSIDevice *dev;
|
|
uint32_t tag;
|
|
uint32_t lun;
|
|
uint32_t status;
|
|
struct {
|
|
uint8_t buf[SCSI_CMD_BUF_SIZE];
|
|
int len;
|
|
size_t xfer;
|
|
uint64_t lba;
|
|
enum SCSIXferMode mode;
|
|
} cmd;
|
|
BlockDriverAIOCB *aiocb;
|
|
bool enqueued;
|
|
QTAILQ_ENTRY(SCSIRequest) next;
|
|
} SCSIRequest;
|
|
|
|
struct SCSIDevice
|
|
{
|
|
DeviceState qdev;
|
|
uint32_t id;
|
|
BlockConf conf;
|
|
SCSIDeviceInfo *info;
|
|
QTAILQ_HEAD(, SCSIRequest) requests;
|
|
int blocksize;
|
|
int type;
|
|
struct SCSISense sense;
|
|
};
|
|
|
|
/* cdrom.c */
|
|
int cdrom_read_toc(int nb_sectors, uint8_t *buf, int msf, int start_track);
|
|
int cdrom_read_toc_raw(int nb_sectors, uint8_t *buf, int msf, int session_num);
|
|
|
|
/* scsi-bus.c */
|
|
typedef int (*scsi_qdev_initfn)(SCSIDevice *dev);
|
|
struct SCSIDeviceInfo {
|
|
DeviceInfo qdev;
|
|
scsi_qdev_initfn init;
|
|
void (*destroy)(SCSIDevice *s);
|
|
int32_t (*send_command)(SCSIDevice *s, uint32_t tag, uint8_t *buf,
|
|
int lun);
|
|
void (*read_data)(SCSIDevice *s, uint32_t tag);
|
|
int (*write_data)(SCSIDevice *s, uint32_t tag);
|
|
void (*cancel_io)(SCSIDevice *s, uint32_t tag);
|
|
uint8_t *(*get_buf)(SCSIDevice *s, uint32_t tag);
|
|
};
|
|
|
|
typedef void (*SCSIAttachFn)(DeviceState *host, BlockDriverState *bdrv,
|
|
int unit);
|
|
struct SCSIBus {
|
|
BusState qbus;
|
|
int busnr;
|
|
|
|
int tcq, ndev;
|
|
scsi_completionfn complete;
|
|
|
|
SCSIDevice *devs[8];
|
|
};
|
|
|
|
void scsi_bus_new(SCSIBus *bus, DeviceState *host, int tcq, int ndev,
|
|
scsi_completionfn complete);
|
|
void scsi_qdev_register(SCSIDeviceInfo *info);
|
|
|
|
static inline SCSIBus *scsi_bus_from_device(SCSIDevice *d)
|
|
{
|
|
return DO_UPCAST(SCSIBus, qbus, d->qdev.parent_bus);
|
|
}
|
|
|
|
SCSIDevice *scsi_bus_legacy_add_drive(SCSIBus *bus, BlockDriverState *bdrv, int unit);
|
|
int scsi_bus_legacy_handle_cmdline(SCSIBus *bus);
|
|
|
|
void scsi_dev_clear_sense(SCSIDevice *dev);
|
|
void scsi_dev_set_sense(SCSIDevice *dev, uint8_t key);
|
|
|
|
SCSIRequest *scsi_req_alloc(size_t size, SCSIDevice *d, uint32_t tag, uint32_t lun);
|
|
SCSIRequest *scsi_req_find(SCSIDevice *d, uint32_t tag);
|
|
void scsi_req_free(SCSIRequest *req);
|
|
|
|
int scsi_req_parse(SCSIRequest *req, uint8_t *buf);
|
|
void scsi_req_print(SCSIRequest *req);
|
|
void scsi_req_complete(SCSIRequest *req);
|
|
|
|
#endif
|