Hi, On Tue, Aug 27, 2019 at 10:22:26AM +0100, Frediano Ziglio wrote: > From: Yuri Benditovich <yuri.benditovich@xxxxxxxxxx> > > Files added without including them in compilation. > They contain implementation of SCSI commands for logical > units of mass-storage device class and USB bulk-only > mass-storage device protocol. > > Signed-off-by: Alexander Nezhinsky<anezhins@xxxxxxxxxx> > Signed-off-by: Yuri Benditovich <yuri.benditovich@xxxxxxxxxx> > --- > src/cd-scsi-dev-params.h | 49 + > src/cd-scsi.c | 2740 ++++++++++++++++++++++++++++++++++++++ > src/cd-scsi.h | 120 ++ > src/cd-usb-bulk-msd.c | 544 ++++++++ > src/cd-usb-bulk-msd.h | 134 ++ > src/scsi-constants.h | 324 +++++ > 6 files changed, 3911 insertions(+) > create mode 100644 src/cd-scsi-dev-params.h > create mode 100644 src/cd-scsi.c > create mode 100644 src/cd-scsi.h > create mode 100644 src/cd-usb-bulk-msd.c > create mode 100644 src/cd-usb-bulk-msd.h > create mode 100644 src/scsi-constants.h A bit upsetting that near 4k insertions comes with commit log of two sentences. Not familiar at all with scsi but the idea of how that will be used, interfaces, limitations, etc. could come handy in the commit log. Same goes for the user of this, on patch "usb-redir: add implementation of emulated CD device" which only says "This module contains implementation of emulated device interface for shared CD." > diff --git a/src/cd-scsi-dev-params.h b/src/cd-scsi-dev-params.h > new file mode 100644 > index 00000000..b480bcdc > --- /dev/null > +++ b/src/cd-scsi-dev-params.h > @@ -0,0 +1,49 @@ > +/* -*- Mode: C; c-basic-offset: 4; indent-tabs-mode: nil -*- */ > +/* > + CD SCSI device parameters > + > + Copyright (C) 2018 Red Hat, Inc. > + > + Red Hat Authors: > + Alexander Nezhinsky<anezhins@xxxxxxxxxx> > + > + This library is free software; you can redistribute it and/or > + modify it under the terms of the GNU Lesser General Public > + License as published by the Free Software Foundation; either > + version 2.1 of the License, or (at your option) any later version. > + > + This library 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 > + Lesser General Public License for more details. > + > + You should have received a copy of the GNU Lesser General Public > + License along with this library; if not, see <http://www.gnu.org/licenses/>. > +*/ > + > +#ifndef SPICE_GTK_CD_SCSI_DEV_PARAMS_H_ > +#define SPICE_GTK_CD_SCSI_DEV_PARAMS_H_ > + > +#include <gio/gio.h> > + > +typedef struct CdScsiDeviceParameters { > + const char *vendor; > + const char *product; > + const char *version; > + const char *serial; > +} CdScsiDeviceParameters; > + > +typedef struct CdScsiDeviceInfo { > + CdScsiDeviceParameters parameters; > + uint32_t started : 1; > + uint32_t locked : 1; > + uint32_t loaded : 1; > +} CdScsiDeviceInfo; > + > +typedef struct CdScsiMediaParameters { > + GFileInputStream *stream; > + uint64_t size; > + uint32_t block_size; > +} CdScsiMediaParameters; > + > +#endif /* SPICE_GTK_CD_SCSI_DEV_PARAMS_H_ */ > diff --git a/src/cd-scsi.c b/src/cd-scsi.c > new file mode 100644 > index 00000000..25842b3b > --- /dev/null > +++ b/src/cd-scsi.c > @@ -0,0 +1,2740 @@ > +/* -*- Mode: C; c-basic-offset: 4; indent-tabs-mode: nil -*- */ > +/* > + CD device emulation - SCSI engine > + > + Copyright (C) 2018 Red Hat, Inc. > + > + Red Hat Authors: > + Alexander Nezhinsky<anezhins@xxxxxxxxxx> > + > + This library is free software; you can redistribute it and/or > + modify it under the terms of the GNU Lesser General Public > + License as published by the Free Software Foundation; either > + version 2.1 of the License, or (at your option) any later version. > + > + This library 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 > + Lesser General Public License for more details. > + > + You should have received a copy of the GNU Lesser General Public > + License along with this library; if not, see <http://www.gnu.org/licenses/>. > +*/ > + > +#include "config.h" > +#include "spice/types.h" > +#include "spice-common.h" > +#include "spice-util.h" > +#include "cd-scsi.h" > + > +#ifdef USE_USBREDIR > + > +#define SPICE_ERROR(fmt, ...) \ > + do { SPICE_DEBUG("dev-scsi error: " fmt , ## __VA_ARGS__); } while (0) > + > +#define FIXED_SENSE_CURRENT 0x70 > +#define FIXED_SENSE_LEN 18 > + > +#define MAX_LUNS 32 > + > +typedef enum CdScsiPowerCondition { > + CD_SCSI_POWER_STOPPED, > + CD_SCSI_POWER_ACTIVE, > + CD_SCSI_POWER_IDLE, > + CD_SCSI_POWER_STANDBY > +} CdScsiPowerCondition; > + > +typedef struct ScsiShortSense { > + uint8_t key; > + uint8_t asc; > + uint8_t ascq; > + const char *descr; > +} ScsiShortSense; > + > +#define CD_MEDIA_EVENT_NO_CHANGE 0x0 > +#define CD_MEDIA_EVENT_EJECT_REQ 0x1 /* user request (mechanical switch) to > + * eject the media */ > +#define CD_MEDIA_EVENT_NEW_MEDIA 0x2 /* new media received */ > +#define CD_MEDIA_EVENT_MEDIA_REMOVAL 0x3 /* media removed */ > +#define CD_MEDIA_EVENT_MEDIA_CHANGED 0x4 /* user request to load new media */ > +#define CD_MEDIA_EVENT_BG_FORMAT_COMPLETE 0x5 > +#define CD_MEDIA_EVENT_BG_FORMAT_RESTART 0x6 > + > +#define CD_POWER_EVENT_NO_CHANGE 0x0 > +#define CD_POWER_EVENT_CHANGE_SUCCESS 0x1 > +#define CD_POWER_EVENT_CHANGE_FALED 0x2 > + > +typedef struct CdScsiLU { > + CdScsiTarget *tgt; > + uint32_t lun; > + > + gboolean realized; > + gboolean removable; > + gboolean loaded; > + gboolean prevent_media_removal; > + gboolean cd_rom; > + > + CdScsiPowerCondition power_cond; > + uint32_t power_event; > + uint32_t media_event; > + > + uint32_t claim_version; > + > + uint64_t size; > + uint32_t block_size; > + uint32_t num_blocks; > + > + char *vendor; > + char *product; > + char *version; > + char *serial; > + > + GFileInputStream *stream; > + > + ScsiShortSense short_sense; /* currently held sense of the scsi device */ > + uint8_t fixed_sense[FIXED_SENSE_LEN]; > +} CdScsiLU; > + > +typedef enum CdScsiTargetState { > + CD_SCSI_TGT_STATE_RUNNING, > + CD_SCSI_TGT_STATE_RESET, > +} CdScsiTargetState; > + > +struct CdScsiTarget { > + void *user_data; > + > + CdScsiTargetState state; > + CdScsiRequest *cur_req; > + GCancellable *cancellable; > + > + uint32_t max_luns; > + CdScsiLU units[MAX_LUNS]; > +}; > + > +static const char* scsi_cmd_name[256]; > + > +/* Predefined sense codes */ > + > +const ScsiShortSense sense_code_NO_SENSE = { > + .key = NO_SENSE , .asc = 0x00 , .ascq = 0x00, > + .descr = "" > +}; > + > +const ScsiShortSense sense_code_NOT_READY_CAUSE_NOT_REPORTABLE = { > + .key = NOT_READY, .asc = 0x04, .ascq = 0x00, > + .descr = "CAUSE NOT REPORTABLE" > +}; > + > +const ScsiShortSense sense_code_BECOMING_READY = { > + .key = NOT_READY, .asc = 0x04, .ascq = 0x01, > + .descr = "IN PROCESS OF BECOMING READY" > +}; > + > +const ScsiShortSense sense_code_INIT_CMD_REQUIRED = { > + .key = NOT_READY, .asc = 0x04, .ascq = 0x02, > + .descr = "INITIALIZING COMMAND REQUIRED" > +}; > + > +const ScsiShortSense sense_code_INTERVENTION_REQUIRED = { > + .key = NOT_READY, .asc = 0x04, .ascq = 0x03, > + .descr = "MANUAL INTERVENTION REQUIRED" > +}; > + > +const ScsiShortSense sense_code_NOT_READY_NO_MEDIUM = { > + .key = NOT_READY, .asc = 0x3a, .ascq = 0x00, > + .descr = "MEDIUM NOT PRESENT" > +}; > + > +const ScsiShortSense sense_code_NOT_READY_NO_MEDIUM_TRAY_CLOSED = { > + .key = NOT_READY, .asc = 0x3a, .ascq = 0x01, > + .descr = "MEDIUM NOT PRESENT - TRAY CLOSED" > +}; > + > +const ScsiShortSense sense_code_NOT_READY_NO_MEDIUM_TRAY_OPEN = { > + .key = NOT_READY, .asc = 0x3a, .ascq = 0x02, > + .descr = "MEDIUM NOT PRESENT - TRAY OPEN" > +}; > + > +const ScsiShortSense sense_code_TARGET_FAILURE = { > + .key = HARDWARE_ERROR, .asc = 0x44, .ascq = 0x00, > + .descr = "INTERNAL TARGET FAILURE" > +}; > + > +const ScsiShortSense sense_code_INVALID_OPCODE = { > + .key = ILLEGAL_REQUEST, .asc = 0x20, .ascq = 0x00, > + .descr = "INVALID COMMAND OPERATION CODE" > +}; > + > +const ScsiShortSense sense_code_LBA_OUT_OF_RANGE = { > + .key = ILLEGAL_REQUEST, .asc = 0x21, .ascq = 0x00, > + .descr = "LOGICAL BLOCK ADDRESS OUT OF RANGE" > +}; > + > +const ScsiShortSense sense_code_INVALID_CDB_FIELD = { > + .key = ILLEGAL_REQUEST, .asc = 0x24, .ascq = 0x00, > + .descr = "INVALID FIELD IN CDB" > +}; > + > +const ScsiShortSense sense_code_INVALID_PARAM_FIELD = { > + .key = ILLEGAL_REQUEST, .asc = 0x26, .ascq = 0x00, > + .descr = "INVALID FIELD IN PARAMETER LIST" > +}; > + > +const ScsiShortSense sense_code_INVALID_PARAM_LEN = { > + .key = ILLEGAL_REQUEST, .asc = 0x1a, .ascq = 0x00, > + .descr = "PARAMETER LIST LENGTH ERROR" > +}; > + > +const ScsiShortSense sense_code_LUN_NOT_SUPPORTED = { > + .key = ILLEGAL_REQUEST, .asc = 0x25, .ascq = 0x00, > + .descr = "LOGICAL UNIT NOT SUPPORTED" > +}; > + > +const ScsiShortSense sense_code_SAVING_PARAMS_NOT_SUPPORTED = { > + .key = ILLEGAL_REQUEST, .asc = 0x39, .ascq = 0x00, > + .descr = "SAVING PARAMETERS NOT SUPPORTED" > +}; > + > +const ScsiShortSense sense_code_INCOMPATIBLE_MEDIUM = { > + .key = ILLEGAL_REQUEST, .asc = 0x30, .ascq = 0x00, > + .descr = "INCOMPATIBLE MEDIUM INSTALLED" > +}; > + > +const ScsiShortSense sense_code_MEDIUM_REMOVAL_PREVENTED = { > + .key = ILLEGAL_REQUEST, .asc = 0x53, .ascq = 0x02, > + .descr = "MEDIUM REMOVAL PREVENTED" > +}; > + > +const ScsiShortSense sense_code_PARAMETERS_CHANGED = { > + .key = UNIT_ATTENTION, .asc = 0x2a, .ascq = 0x00, > + .descr = "PARAMETERS CHANGED" > +}; > + > +const ScsiShortSense sense_code_POWER_ON_RESET = { > + .key = UNIT_ATTENTION, .asc = 0x29, .ascq = 0x00, > + .descr = "POWER ON, RESET, OR BUS DEVICE RESET" > +}; > + > +const ScsiShortSense sense_code_SCSI_BUS_RESET = { > + .key = UNIT_ATTENTION, .asc = 0x29, .ascq = 0x02, > + .descr = "SCSI BUS RESET" > +}; > + > +const ScsiShortSense sense_code_UA_NO_MEDIUM = { > + .key = UNIT_ATTENTION, .asc = 0x3a, .ascq = 0x00, > + .descr = "MEDIUM NOT PRESENT" > +}; > + > +const ScsiShortSense sense_code_MEDIUM_CHANGED = { > + .key = UNIT_ATTENTION, .asc = 0x28, .ascq = 0x00, > + .descr = "MEDIUM CHANGED" > +}; > + > +const ScsiShortSense sense_code_REPORTED_LUNS_CHANGED = { > + .key = UNIT_ATTENTION, .asc = 0x3f, .ascq = 0x0e, > + .descr = "REPORTED LUNS CHANGED" > +}; > + > +const ScsiShortSense sense_code_DEVICE_INTERNAL_RESET = { > + .key = UNIT_ATTENTION, .asc = 0x29, .ascq = 0x04, > + .descr = "DEVICE INTERNAL RESET" > +}; > + > +const ScsiShortSense sense_code_UNIT_ATTENTION_MEDIUM_REMOVAL_REQUEST = { > + .key = UNIT_ATTENTION, .asc = 0x5a, .ascq = 0x01, > + .descr = "OPERATOR MEDIUM REMOVAL REQUEST" > +}; > + > +static inline gboolean cd_scsi_opcode_ua_supress(uint32_t opcode) > +{ > + switch (opcode) { > + case INQUIRY: > + case REPORT_LUNS: > + case GET_CONFIGURATION: > + case GET_EVENT_STATUS_NOTIFICATION: > + case REQUEST_SENSE: > + return TRUE; > + default: > + return FALSE; > + } > +} > + > +static inline const char *CdScsiReqState_str(CdScsiReqState state) > +{ > + switch(state) { > + case SCSI_REQ_IDLE: > + return "IDLE"; > + case SCSI_REQ_RUNNING: > + return "RUNNING"; > + case SCSI_REQ_COMPLETE: > + return "COMPLETE"; > + case SCSI_REQ_CANCELED: > + return "CANCELED"; > + default: > + return "ILLEGAL"; > + } > +} > + > +static const char *cd_scsi_sense_key_descr(uint8_t sense_key) > +{ > + switch(sense_key) { > + case NO_SENSE: > + return "NO SENSE"; > + case RECOVERED_ERROR: > + return "RECOVERED ERROR"; > + case NOT_READY: > + return "LUN NOT READY"; > + case MEDIUM_ERROR: > + return "MEDIUM ERROR"; > + case HARDWARE_ERROR: > + return "HARDWARE ERROR"; > + case ILLEGAL_REQUEST: > + return "ILLEGAL REQUEST"; > + case UNIT_ATTENTION: > + return "UNIT ATTENTION"; > + case BLANK_CHECK: > + return "BLANK CHECK"; > + case ABORTED_COMMAND: > + return "ABORTED COMMAND"; > + default: > + return "???"; > + } > +} > +static uint32_t cd_scsi_build_fixed_sense(uint8_t *buf, const ScsiShortSense *short_sense) > +{ > + memset(buf, 0, FIXED_SENSE_LEN); > + > + buf[0] = FIXED_SENSE_CURRENT; > + buf[2] = short_sense->key; > + buf[7] = 10; > + buf[12] = short_sense->asc; > + buf[13] = short_sense->ascq; > + > + return FIXED_SENSE_LEN; > +} > + > +static inline void cd_scsi_req_init(CdScsiRequest *req) > +{ > + req->req_state = SCSI_REQ_IDLE; > + req->xfer_dir = SCSI_XFER_NONE; > + req->priv_data = NULL; > + req->in_len = 0; > + req->status = GOOD; > +} > + > +static inline void cd_scsi_dev_sense_reset(CdScsiLU *dev) > +{ > + memset(&dev->short_sense, 0, sizeof(dev->short_sense)); > + cd_scsi_build_fixed_sense(dev->fixed_sense, &dev->short_sense); > +} > + > +static inline void cd_scsi_dev_sense_set(CdScsiLU *dev, const ScsiShortSense *short_sense) > +{ > + if (short_sense != NULL) { > + /* copy short sense and generate full sense in fixed format */ > + dev->short_sense = *short_sense; > + cd_scsi_build_fixed_sense(dev->fixed_sense, short_sense); > + } > +} > + > +static inline void cd_scsi_dev_sense_set_power_on(CdScsiLU *dev) > +{ > + cd_scsi_dev_sense_set(dev, &sense_code_POWER_ON_RESET); > +} > + > +static void cd_scsi_cmd_complete_check_cond(CdScsiLU *dev, CdScsiRequest *req, > + const ScsiShortSense *short_sense) > +{ > + req->req_state = SCSI_REQ_COMPLETE; > + req->status = CHECK_CONDITION; > + req->in_len = 0; > + > + cd_scsi_dev_sense_set(dev, short_sense); > + > + SPICE_DEBUG("CHECK_COND, request lun:%u" > + " op: 0x%02x, pending sense: 0x%02x %02x %02x - %s, %s", > + dev->lun, (uint32_t)req->cdb[0], > + (uint32_t)dev->short_sense.key, > + (uint32_t)dev->short_sense.asc, > + (uint32_t)dev->short_sense.ascq, > + cd_scsi_sense_key_descr(dev->short_sense.key), > + dev->short_sense.descr); > +} > + > +static void cd_scsi_cmd_complete_good(CdScsiLU *dev, CdScsiRequest *req) > +{ > + req->req_state = SCSI_REQ_COMPLETE; > + req->status = GOOD; > +} > + > +/* SCSI Target */ > + > +SPICE_CONSTRUCTOR_FUNC(cd_scsi_cmd_names_init) > +{ > + uint32_t opcode; > + > + for (opcode = 0; opcode < 256; opcode++) { > + scsi_cmd_name[opcode] = "UNSUPPORTED"; > + } > + > + scsi_cmd_name[REPORT_LUNS] = "REPORT LUNS"; > + scsi_cmd_name[TEST_UNIT_READY] = "TEST UNIT READY"; > + scsi_cmd_name[INQUIRY] = "INQUIRY"; > + scsi_cmd_name[REQUEST_SENSE] = "REQUEST SENSE"; > + scsi_cmd_name[READ_6] = "READ(6)"; > + scsi_cmd_name[READ_10] = "READ(10)"; > + scsi_cmd_name[READ_12] = "READ(12)"; > + scsi_cmd_name[READ_16] = "READ(16)"; > + scsi_cmd_name[READ_CAPACITY_10] = "READ CAPACITY(10)"; > + scsi_cmd_name[READ_TOC] = "READ TOC"; > + scsi_cmd_name[GET_EVENT_STATUS_NOTIFICATION] = "GET EVENT/STATUS NOTIFICATION"; > + scsi_cmd_name[READ_DISC_INFORMATION] = "READ DISC INFO"; > + scsi_cmd_name[READ_TRACK_INFORMATION] = "READ TRACK INFO"; > + scsi_cmd_name[MODE_SENSE_10] = "MODE SENSE(10)"; > + scsi_cmd_name[MODE_SELECT] = "MODE SELECT(6)"; > + scsi_cmd_name[MODE_SELECT_10] = "MODE SELECT(10)"; > + scsi_cmd_name[GET_CONFIGURATION] = "GET CONFIGURATION"; > + scsi_cmd_name[ALLOW_MEDIUM_REMOVAL] = "PREVENT ALLOW MEDIUM REMOVAL"; > + scsi_cmd_name[MMC_SEND_EVENT] = "SEND EVENT"; > + scsi_cmd_name[MMC_REPORT_KEY] = "REPORT KEY"; > + scsi_cmd_name[MMC_SEND_KEY] = "SEND_KEY"; > + scsi_cmd_name[START_STOP] = "START STOP UNIT"; > + scsi_cmd_name[MMC_GET_PERFORMANCE] = "GET PERFORMANCE"; > + scsi_cmd_name[MMC_MECHANISM_STATUS] = "MECHANISM STATUS"; > +} > + > +CdScsiTarget *cd_scsi_target_alloc(void *target_user_data, uint32_t max_luns) > +{ > + CdScsiTarget *st; > + > + if (max_luns == 0 || max_luns > MAX_LUNS) { > + SPICE_ERROR("Alloc, illegal max_luns:%u", max_luns); > + return NULL; > + } > + > + st = g_malloc0(sizeof(*st)); > + > + st->user_data = target_user_data; > + st->state = CD_SCSI_TGT_STATE_RUNNING; > + st->cur_req = NULL; > + st->cancellable = g_cancellable_new(); > + st->max_luns = max_luns; > + > + return st; > +} > + > +void cd_scsi_target_free(CdScsiTarget *st) > +{ > + uint32_t lun; > + > + cd_scsi_target_reset(st); > + for (lun = 0; lun < st->max_luns; lun++) { > + CdScsiLU *unit = &st->units[lun]; > + if (unit->realized) { > + cd_scsi_dev_unrealize(st, lun); > + } > + g_clear_object(&unit->stream); > + } > + g_clear_object(&st->cancellable); > + g_free(st); > +} > + > +/* SCSI Device */ > + > +static inline gboolean cd_scsi_target_lun_legal(const CdScsiTarget *st, uint32_t lun) > +{ > + return (lun < st->max_luns) ? TRUE : FALSE; > +} > + > +static inline gboolean cd_scsi_target_lun_realized(const CdScsiTarget *st, uint32_t lun) > +{ > + return st->units[lun].realized; > +} > + > +int cd_scsi_dev_realize(CdScsiTarget *st, uint32_t lun, > + const CdScsiDeviceParameters *dev_params) > +{ > + CdScsiLU *dev; > + > + if (!cd_scsi_target_lun_legal(st, lun)) { > + SPICE_ERROR("Realize, illegal lun:%u", lun); > + return -1; > + } > + if (cd_scsi_target_lun_realized(st, lun)) { > + SPICE_ERROR("Realize, already realized lun:%u", lun); > + return -1; > + } > + dev = &st->units[lun]; > + > + memset(dev, 0, sizeof(*dev)); > + dev->tgt = st; > + dev->lun = lun; > + > + dev->realized = TRUE; > + dev->removable = TRUE; > + dev->loaded = FALSE; > + dev->prevent_media_removal = FALSE; > + dev->cd_rom = FALSE; > + > + dev->power_cond = CD_SCSI_POWER_ACTIVE; > + dev->power_event = CD_POWER_EVENT_NO_CHANGE; > + dev->media_event = CD_MEDIA_EVENT_NO_CHANGE; > + > + dev->claim_version = 3; /* 0 : none; 2,3,5 : SPC/MMC-x */ > + > + dev->vendor = g_strdup(dev_params->vendor); > + dev->product = g_strdup(dev_params->product); > + dev->version = g_strdup(dev_params->version); > + dev->serial = g_strdup(dev_params->serial); > + > + cd_scsi_dev_sense_set_power_on(dev); > + > + SPICE_DEBUG("Realize lun:%u bs:%u VR:[%s] PT:[%s] ver:[%s] SN[%s]", > + lun, dev->block_size, dev->vendor, > + dev->product, dev->version, dev->serial); > + return 0; > +} > + > +static void cd_scsi_lu_media_reset(CdScsiLU *dev) > +{ > + /* media_event is not set here, as it depends on the context */ > + g_clear_object(&dev->stream); > + dev->size = 0; > + dev->block_size = 0; > + dev->num_blocks = 0; > +} > + > +int cd_scsi_dev_lock(CdScsiTarget *st, uint32_t lun, gboolean lock) > +{ > + CdScsiLU *dev; > + > + if (!cd_scsi_target_lun_legal(st, lun)) { > + SPICE_ERROR("Lock, illegal lun:%u", lun); > + return -1; > + } > + if (!cd_scsi_target_lun_realized(st, lun)) { > + SPICE_ERROR("Lock, unrealized lun:%u", lun); > + return -1; > + } > + dev = &st->units[lun]; > + dev->prevent_media_removal = lock; > + SPICE_DEBUG("lun:%u %slock", lun, lock ? "un" :""); > + return 0; > +} > + > +static void cd_scsi_lu_load(CdScsiLU *dev, > + const CdScsiMediaParameters *media_params) > +{ > + if (media_params != NULL) { > + dev->media_event = CD_MEDIA_EVENT_NEW_MEDIA; > + dev->stream = g_object_ref(media_params->stream); > + dev->size = media_params->size; > + dev->block_size = media_params->block_size; > + dev->num_blocks = media_params->size / media_params->block_size; > + dev->loaded = TRUE; > + } else { > + dev->media_event = CD_MEDIA_EVENT_MEDIA_REMOVAL; > + cd_scsi_lu_media_reset(dev); > + dev->loaded = FALSE; > + } > +} > + > +static void cd_scsi_lu_unload(CdScsiLU *dev) > +{ > + dev->media_event = CD_MEDIA_EVENT_MEDIA_REMOVAL; > + cd_scsi_lu_media_reset(dev); > + dev->loaded = FALSE; > +} > + > +int cd_scsi_dev_load(CdScsiTarget *st, uint32_t lun, > + const CdScsiMediaParameters *media_params) > +{ > + CdScsiLU *dev; > + > + if (!cd_scsi_target_lun_legal(st, lun)) { > + SPICE_ERROR("Load, illegal lun:%u", lun); > + return -1; > + } > + if (!cd_scsi_target_lun_realized(st, lun)) { > + SPICE_ERROR("Load, unrealized lun:%u", lun); > + return -1; > + } > + dev = &st->units[lun]; > + > + cd_scsi_lu_load(dev, media_params); > + dev->power_cond = CD_SCSI_POWER_ACTIVE; > + dev->power_event = CD_POWER_EVENT_CHANGE_SUCCESS; > + > + cd_scsi_dev_sense_set(dev, &sense_code_MEDIUM_CHANGED); > + > + SPICE_DEBUG("Load lun:%u size:%" G_GUINT64_FORMAT > + " blk_sz:%u num_blocks:%u", > + lun, dev->size, dev->block_size, dev->num_blocks); > + return 0; > +} > + > +int cd_scsi_dev_get_info(CdScsiTarget *st, uint32_t lun, CdScsiDeviceInfo *lun_info) > +{ > + CdScsiLU *dev; > + > + if (!cd_scsi_target_lun_legal(st, lun)) { > + SPICE_ERROR("Load, illegal lun:%u", lun); > + return -1; > + } > + if (!cd_scsi_target_lun_realized(st, lun)) { > + SPICE_ERROR("Load, unrealized lun:%u", lun); > + return -1; > + } > + dev = &st->units[lun]; > + > + lun_info->started = dev->power_cond == CD_SCSI_POWER_ACTIVE; > + lun_info->locked = dev->prevent_media_removal; > + lun_info->loaded = dev->loaded; > + > + lun_info->parameters.vendor = dev->vendor; > + lun_info->parameters.product = dev->product; > + lun_info->parameters.version = dev->version; > + lun_info->parameters.serial = dev->serial; > + > + return 0; > +} > + > +int cd_scsi_dev_unload(CdScsiTarget *st, uint32_t lun) > +{ > + CdScsiLU *dev; > + > + if (!cd_scsi_target_lun_legal(st, lun)) { > + SPICE_ERROR("Unload, illegal lun:%u", lun); > + return -1; > + } > + if (!cd_scsi_target_lun_realized(st, lun)) { > + SPICE_ERROR("Unload, unrealized lun:%u", lun); > + return -1; > + } > + dev = &st->units[lun]; > + if (!dev->loaded) { > + SPICE_ERROR("Unload, lun:%u not loaded yet", lun); > + return 0; > + } > + if (dev->prevent_media_removal) { > + SPICE_ERROR("Unload, lun:%u prevent_media_removal set", lun); > + return -1; > + } > + > + cd_scsi_lu_unload(dev); > + dev->power_cond = CD_SCSI_POWER_STOPPED; > + dev->power_event = CD_POWER_EVENT_CHANGE_SUCCESS; > + > + cd_scsi_dev_sense_set(dev, &sense_code_UA_NO_MEDIUM); > + > + SPICE_DEBUG("Unload lun:%u", lun); > + return 0; > +} > + > +int cd_scsi_dev_unrealize(CdScsiTarget *st, uint32_t lun) > +{ > + CdScsiLU *dev; > + > + if (!cd_scsi_target_lun_legal(st, lun)) { > + SPICE_ERROR("Unrealize, illegal lun:%u", lun); > + return -1; > + } > + if (!cd_scsi_target_lun_realized(st, lun)) { > + SPICE_ERROR("Unrealize, absent lun:%u", lun); > + return -1; > + } > + dev = &st->units[lun]; > + > + g_clear_pointer(&dev->vendor, g_free); > + g_clear_pointer(&dev->product, g_free); > + g_clear_pointer(&dev->version, g_free); > + g_clear_pointer(&dev->serial, g_free); > + > + g_clear_object(&dev->stream); > + > + dev->loaded = FALSE; > + dev->realized = FALSE; > + dev->power_cond = CD_SCSI_POWER_STOPPED; > + > + SPICE_DEBUG("Unrealize lun:%u", lun); > + return 0; > +} > + > +int cd_scsi_dev_reset(CdScsiTarget *st, uint32_t lun) > +{ > + CdScsiLU *dev; > + > + if (!cd_scsi_target_lun_legal(st, lun)) { > + SPICE_ERROR("Device reset, illegal lun:%u", lun); > + return -1; > + } > + if (!cd_scsi_target_lun_realized(st, lun)) { > + SPICE_ERROR("Device reset, absent lun:%u", lun); > + return -1; > + } > + dev = &st->units[lun]; > + > + /* if we reset the 'prevent' flag we can't create > + * the unit that is locked from the beginning, so > + * we keep this flag as persistent over resets > + */ > + /* dev->prevent_media_removal = FALSE; */ > + dev->power_cond = CD_SCSI_POWER_ACTIVE; > + dev->power_event = CD_POWER_EVENT_CHANGE_SUCCESS; > + cd_scsi_dev_sense_set_power_on(dev); > + > + SPICE_DEBUG("Device reset lun:%u", lun); > + return 0; > +} > + > +static void cd_scsi_target_do_reset(CdScsiTarget *st) > +{ > + uint32_t lun; > + > + for (lun = 0; lun < st->max_luns; lun++) { > + if (st->units[lun].realized) { > + cd_scsi_dev_reset(st, lun); > + } > + } > + > + SPICE_DEBUG("Target reset complete"); > + st->state = CD_SCSI_TGT_STATE_RUNNING; > + cd_scsi_target_reset_complete(st->user_data); > +} > + > +int cd_scsi_target_reset(CdScsiTarget *st) > +{ > + if (st->state == CD_SCSI_TGT_STATE_RESET) { > + SPICE_DEBUG("Target already in reset"); > + return -1; > + } > + > + st->state = CD_SCSI_TGT_STATE_RESET; > + > + if (st->cur_req != NULL) { > + cd_scsi_dev_request_cancel(st, st->cur_req); > + if (st->cur_req != NULL) { > + SPICE_DEBUG("Target reset in progress..."); > + return 0; > + } > + } > + > + cd_scsi_target_do_reset(st); > + return 0; > +} > + > +CdScsiReqState cd_scsi_get_req_state(CdScsiRequest *req) > +{ > + return req->req_state; > +} > + > +static void strpadcpy(char *buf, int buf_size, const char *str, char pad) > +{ > + int len = strnlen(str, buf_size); > + memcpy(buf, str, len); > + memset(buf + len, pad, buf_size - len); > +} > + > +/* SCSI CDB */ > + > +static unsigned int scsi_cdb_length(const uint8_t *cdb) > +{ > + unsigned int cdb_len; > + > + switch (cdb[0] >> 5) { > + case 0: > + cdb_len = 6; > + break; > + case 1: > + case 2: > + cdb_len = 10; > + break; > + case 4: > + cdb_len = 16; > + break; > + case 5: > + cdb_len = 12; > + break; > + default: > + cdb_len = 0; > + } > + return cdb_len; > +} > + > +static uint64_t scsi_cdb_lba(const uint8_t *cdb, int cdb_len) > +{ > + uint64_t lba; > + > + switch (cdb_len) { > + case 6: > + lba = (((uint64_t)(cdb[1] & 0x1f)) << 16) | > + (((uint64_t)cdb[2]) << 8) | > + ((uint64_t)cdb[3]); > + break; > + case 10: > + case 12: > + lba = (((uint64_t)cdb[2]) << 24) | > + (((uint64_t)cdb[3]) << 16) | > + (((uint64_t)cdb[4]) << 8) | > + ((uint64_t)cdb[5]); > + break; > + case 16: > + lba = (((uint64_t)cdb[2]) << 56) | > + (((uint64_t)cdb[3]) << 48) | > + (((uint64_t)cdb[4]) << 40) | > + (((uint64_t)cdb[5]) << 32) | > + (((uint64_t)cdb[6]) << 24) | > + (((uint64_t)cdb[7]) << 16) | > + (((uint64_t)cdb[8]) << 8) | > + ((uint64_t)cdb[9]); > + break; > + default: > + lba = 0; > + } > + return lba; > +} > + > +static uint32_t scsi_cdb_xfer_length(const uint8_t *cdb, int cdb_len) > +{ > + uint32_t len; > + > + switch (cdb_len) { > + case 6: > + len = (uint32_t)cdb[4]; > + if (len == 0) > + len = 256; > + break; > + case 10: > + len = (((uint32_t)cdb[7]) << 8) | > + ((uint32_t)cdb[8]); > + break; > + case 12: > + len = (((uint32_t)cdb[6]) << 24) | > + (((uint32_t)cdb[7]) << 16) | > + (((uint32_t)cdb[8]) << 8) | > + ((uint32_t)cdb[9]); > + break; > + case 16: > + len = (((uint32_t)cdb[10]) << 24) | > + (((uint32_t)cdb[11]) << 16) | > + (((uint32_t)cdb[12]) << 8) | > + ((uint32_t)cdb[13]); > + break; > + default: > + len = 0; > + break; > + } > + return len; > +} > + > +/* SCSI commands */ > + > +static void cd_scsi_cmd_test_unit_ready(CdScsiLU *dev, CdScsiRequest *req) > +{ > + req->xfer_dir = SCSI_XFER_NONE; > + req->in_len = 0; > + > + if (dev->loaded) { > + if (dev->power_cond != CD_SCSI_POWER_STOPPED) { > + cd_scsi_cmd_complete_good(dev, req); > + } else { > + cd_scsi_cmd_complete_check_cond(dev, req, &sense_code_INIT_CMD_REQUIRED); > + } > + } else { > + cd_scsi_cmd_complete_check_cond(dev, req, &sense_code_NOT_READY_NO_MEDIUM); > + } > +} > + > +static void cd_scsi_cmd_request_sense(CdScsiLU *dev, CdScsiRequest *req) > +{ > + req->xfer_dir = SCSI_XFER_FROM_DEV; > + > + req->req_len = req->cdb[4]; > + req->in_len = MIN(req->req_len, sizeof(dev->fixed_sense)); > + > + if (dev->short_sense.key != NO_SENSE) { > + SPICE_DEBUG("%s, lun:%u reported sense: 0x%02x %02x %02x - %s, %s", > + __FUNCTION__, req->lun, > + dev->short_sense.key, dev->short_sense.asc, dev->short_sense.ascq, > + cd_scsi_sense_key_descr(dev->short_sense.key), > + dev->short_sense.descr); > + } > + memcpy(req->buf, dev->fixed_sense, sizeof(dev->fixed_sense)); > + cd_scsi_dev_sense_reset(dev); /* clear reported sense */ > + > + cd_scsi_cmd_complete_good(dev, req); > +} > + > +static void cd_scsi_cmd_report_luns(CdScsiTarget *st, CdScsiLU *dev, > + CdScsiRequest *req) > +{ > + uint8_t *out_buf = req->buf; > + uint32_t max_luns = st->max_luns; > + uint32_t buflen = 8; /* header length */ > + uint32_t lun; > + > + req->req_len = scsi_cdb_xfer_length(req->cdb, 12); > + req->xfer_dir = SCSI_XFER_FROM_DEV; > + > + /* check SELECT REPORT field */ > + if (req->cdb[2] == 0x01) { > + /* only well known logical units */ > + max_luns = 0; > + } > + > + memset(out_buf, 0, 8); > + > + for (lun = 0; lun < max_luns; lun++) { > + if (st->units[lun].realized) { > + out_buf[buflen++] = 0; > + out_buf[buflen++] = (uint8_t)(lun); > + memset(&out_buf[buflen], 0, 6); > + buflen += 6; > + } > + } > + > + /* fill LUN LIST LENGTH */ > + out_buf[0] = (uint8_t)((buflen-8) >> 24); > + out_buf[1] = (uint8_t)((buflen-8) >> 16); > + out_buf[2] = (uint8_t)((buflen-8) >> 8); > + out_buf[3] = (uint8_t)((buflen-8)); > + > + req->in_len = buflen; > + cd_scsi_cmd_complete_good(dev, req); > +} > + > +#define SCSI_MAX_INQUIRY_LEN 256 > +#define SCSI_MAX_MODE_LEN 256 > + > +static void cd_scsi_cmd_inquiry_vpd_no_lun(CdScsiLU *dev, CdScsiRequest *req, > + uint32_t perif_qual) > +{ > + uint8_t *outbuf = req->buf; > + uint8_t page_code = req->cdb[2]; > + uint32_t resp_len = 4; > + > + outbuf[0] = (perif_qual << 5) | TYPE_ROM; > + outbuf[1] = page_code ; /* this page */ > + outbuf[2] = 0x00; /* page length MSB */ > + outbuf[3] = 0x00; /* page length LSB - no more data */ > + > + req->in_len = MIN(req->req_len, resp_len); > + > + SPICE_DEBUG("inquiry_vpd, unsupported lun:%u" > + " perif_qual:0x%x resp_len: %" G_GUINT64_FORMAT, > + req->lun, perif_qual, req->in_len); > + > + cd_scsi_cmd_complete_good(dev, req); > +} > + > +static void cd_scsi_cmd_inquiry_vpd(CdScsiLU *dev, CdScsiRequest *req) > +{ > + uint8_t *outbuf = req->buf; > + uint8_t page_code = req->cdb[2]; > + int buflen = 4; > + int start = 4; > + > + outbuf[0] = TYPE_ROM; > + outbuf[1] = page_code ; /* this page */ > + outbuf[2] = 0x00; /* page length MSB */ > + outbuf[3] = 0x00; /* page length LSB, to write later */ > + > + switch (page_code) { > + case 0x00: /* Supported page codes, mandatory */ > + { > + outbuf[buflen++] = 0x00; // list of supported pages (this page) > + if (dev->serial) { > + outbuf[buflen++] = 0x80; // unit serial number > + } > + outbuf[buflen++] = 0x83; // device identification > + > + SPICE_DEBUG("Inquiry EVPD[Supported pages] lun:%u" > + " req_len: %" G_GUINT64_FORMAT " resp_len: %d", > + req->lun, req->req_len, buflen); > + break; > + } > + case 0x80: /* Device serial number, optional */ > + { > + int serial_len; > + > + serial_len = strlen(dev->serial); > + if (serial_len > 36) { > + serial_len = 36; > + } > + memcpy(outbuf+buflen, dev->serial, serial_len); > + buflen += serial_len; > + > + SPICE_DEBUG("Inquiry EVPD[Serial num] lun:%u" > + " req_len: %" G_GUINT64_FORMAT " resp_len: %d", > + req->lun, req->req_len, buflen); > + break; > + } > + case 0x83: /* Device identification page, mandatory */ > + { > + int serial_len = strlen(dev->serial); > + int max_len = 20; > + > + if (serial_len > max_len) { > + serial_len = max_len; > + } > + > + outbuf[buflen++] = 0x2; // ASCII > + outbuf[buflen++] = 0; // not officially assigned > + outbuf[buflen++] = 0; // reserved > + outbuf[buflen++] = serial_len; // length of data following > + > + memcpy(outbuf+buflen, dev->serial, serial_len); > + buflen += serial_len; > + > + SPICE_DEBUG("Inquiry EVPD[Device id] lun:%u" > + " req_len: %" G_GUINT64_FORMAT " resp_len: %d", > + req->lun, req->req_len, buflen); > + break; > + } > + > + default: > + cd_scsi_cmd_complete_check_cond(dev, req, &sense_code_INVALID_CDB_FIELD); > + SPICE_DEBUG("inquiry_standard, lun:%u invalid page_code: %02x", > + req->lun, (int)page_code); > + return; > + } > + > + /* done with EVPD */ > + g_assert(buflen - start <= 255); > + outbuf[3] = buflen - start; /* page length LSB */ > + > + req->in_len = buflen; > + cd_scsi_cmd_complete_good(dev, req); > +} > + > +#define INQUIRY_STANDARD_LEN_MIN 36 > +#define INQUIRY_STANDARD_LEN 96 > +#define INQUIRY_STANDARD_LEN_NO_VER 57 > + > +#define PERIF_QUALIFIER_CONNECTED 0x00 > +#define PERIF_QUALIFIER_NOT_CONNECTED 0x01 > +#define PERIF_QUALIFIER_UNSUPPORTED 0x03 > + > +#define INQUIRY_REMOVABLE_MEDIUM 0x80 > + > +#define INQUIRY_VERSION_NONE 0x00 > +#define INQUIRY_VERSION_SPC3 0x05 > + > +/* byte 3 */ > +#define INQUIRY_RESP_HISUP (0x01 << 4) > +#define INQUIRY_RESP_NORM_ACA (0x01 << 5) > +#define INQUIRY_RESP_DATA_FORMAT_SPC3 0x02 > + > +#define INQUIRY_VERSION_DESC_SAM2 0x040 > +#define INQUIRY_VERSION_DESC_SPC3 0x300 > +#define INQUIRY_VERSION_DESC_MMC3 0x2A0 > +#define INQUIRY_VERSION_DESC_SBC2 0x320 > + > +static void cd_scsi_cmd_inquiry_standard_no_lun(CdScsiLU *dev, CdScsiRequest *req, > + uint32_t perif_qual) > +{ > + uint8_t *outbuf = req->buf; > + uint32_t resp_len = INQUIRY_STANDARD_LEN_MIN; > + > + memset(req->buf, 0, INQUIRY_STANDARD_LEN_MIN); > + > + outbuf[0] = (perif_qual << 5) | TYPE_ROM; > + outbuf[2] = INQUIRY_VERSION_NONE; > + outbuf[3] = INQUIRY_RESP_DATA_FORMAT_SPC3; > + > + outbuf[4] = resp_len - 4; /* additional length, after header */ > + > + req->in_len = MIN(req->req_len, resp_len); > + > + SPICE_DEBUG("inquiry_standard, unsupported lun:%u perif_qual:0x%x " > + "inquiry_len: %u resp_len: %" G_GUINT64_FORMAT, > + req->lun, perif_qual, resp_len, req->in_len); > + > + cd_scsi_cmd_complete_good(dev, req); > +} > + > +static void cd_scsi_cmd_inquiry_standard(CdScsiLU *dev, CdScsiRequest *req) > +{ > + uint8_t *outbuf = req->buf; > + uint32_t resp_len = > + (dev->claim_version == 0) ? INQUIRY_STANDARD_LEN_NO_VER : INQUIRY_STANDARD_LEN; > + > + outbuf[0] = (PERIF_QUALIFIER_CONNECTED << 5) | TYPE_ROM; > + outbuf[1] = (dev->removable) ? INQUIRY_REMOVABLE_MEDIUM : 0; > + outbuf[2] = (dev->claim_version == 0) ? INQUIRY_VERSION_NONE : INQUIRY_VERSION_SPC3; > + outbuf[3] = INQUIRY_RESP_NORM_ACA | INQUIRY_RESP_HISUP | INQUIRY_RESP_DATA_FORMAT_SPC3; > + > + outbuf[4] = resp_len - 4; /* additional length, after header */ > + > + /* (outbuf[6,7] = 0) means also {BQue=0,CmdQue=0} - no queueing at all */ > + > + strpadcpy((char *) &outbuf[8], 8, dev->vendor, ' '); > + strpadcpy((char *) &outbuf[16], 16, dev->product, ' '); > + memcpy(&outbuf[32], dev->version, MIN(4, strlen(dev->version))); > + > + if (dev->claim_version > 0) { > + /* now supporting only 3 */ > + outbuf[58] = (INQUIRY_VERSION_DESC_SAM2 >> 8) & 0xff; > + outbuf[59] = INQUIRY_VERSION_DESC_SAM2 & 0xff; > + > + outbuf[60] = (INQUIRY_VERSION_DESC_SPC3 >> 8) & 0xff; > + outbuf[61] = INQUIRY_VERSION_DESC_SPC3 & 0xff; > + > + outbuf[62] = (INQUIRY_VERSION_DESC_MMC3 >> 8) & 0xff; > + outbuf[63] = INQUIRY_VERSION_DESC_MMC3 & 0xff; > + > + outbuf[64] = (INQUIRY_VERSION_DESC_SBC2 >> 8) & 0xff; > + outbuf[65] = INQUIRY_VERSION_DESC_SBC2 & 0xff; > + } > + > + req->in_len = MIN(req->req_len, resp_len); > + > + SPICE_DEBUG("inquiry_standard, lun:%u" > + " inquiry_len: %u resp_len: %" G_GUINT64_FORMAT, > + req->lun, resp_len, req->in_len); > + > + cd_scsi_cmd_complete_good(dev, req); > +} > + > +#define CD_INQUIRY_FLAG_EVPD 0x01 > +#define CD_INQUIRY_FLAG_CMD_DT 0x02 > + > +static void cd_scsi_cmd_inquiry(CdScsiLU *dev, CdScsiRequest *req) > +{ > + gboolean evpd, cmd_data; > + > + req->xfer_dir = SCSI_XFER_FROM_DEV; > + > + evpd = (req->cdb[1] & CD_INQUIRY_FLAG_EVPD) ? TRUE : FALSE; > + cmd_data = (req->cdb[1] & CD_INQUIRY_FLAG_CMD_DT) ? TRUE : FALSE; > + > + if (cmd_data) { > + SPICE_DEBUG("inquiry, lun:%u CmdDT bit set - unsupported, " > + "cdb[1]:0x%02x cdb[1]:0x%02x", > + req->lun, (int)req->cdb[1], (int)req->cdb[2]); > + cd_scsi_cmd_complete_check_cond(dev, req, &sense_code_INVALID_CDB_FIELD); > + return; > + } > + > + req->req_len = req->cdb[4] | (req->cdb[3] << 8); > + memset(req->buf, 0, req->req_len); > + > + if (evpd) { /* enable vital product data */ > + cd_scsi_cmd_inquiry_vpd(dev, req); > + } else { /* standard inquiry data */ > + if (req->cdb[2] != 0) { > + SPICE_DEBUG("inquiry_standard, lun:%u non-zero page code: %02x", > + req->lun, (int)req->cdb[2]); > + cd_scsi_cmd_complete_check_cond(dev, req, &sense_code_INVALID_CDB_FIELD); > + return; > + } > + cd_scsi_cmd_inquiry_standard(dev, req); > + } > +} > + > +static void cd_scsi_cmd_read_capacity(CdScsiLU *dev, CdScsiRequest *req) > +{ > + uint32_t last_blk = dev->num_blocks - 1; > + uint32_t blk_size = dev->block_size; > + uint32_t *last_blk_out = (uint32_t *)req->buf; > + uint32_t *blk_size_out = (uint32_t *)(req->buf + 4); > + > + req->xfer_dir = SCSI_XFER_FROM_DEV; > + req->req_len = 8; > + > + *last_blk_out = htobe32(last_blk); > + *blk_size_out = htobe32(blk_size); > + > + SPICE_DEBUG("Read capacity, lun:%u last_blk: %u blk_sz: %u", > + req->lun, last_blk, blk_size); > + > + req->in_len = 8; > + cd_scsi_cmd_complete_good(dev, req); > +} > + > +#define RDI_TYPE_STANDARD 0 /* Standard Disc Information */ > +#define RDI_TYPE_TRACK_RESOURCES 1 /* Track Resources Information */ > +#define RDI_TYPE_POW_RESOURCES 2 /* POW Resources Information */ > + > +#define RDI_STANDARD_LEN 34 > + > +#define RDI_ERAZABLE (0x01 << 4) > +#define RDI_NON_ERAZABLE (0x00 << 4) > + > +#define RDI_SESSION_EMPTY (0x00 << 2) > +#define RDI_SESSION_INCOMPLETE (0x01 << 2) > +#define RDI_SESSION_DAMAGED (0x02 << 2) > +#define RDI_SESSION_COMPLETE (0x03 << 2) > + > +#define RDI_DISC_EMPTY 0x00 > +#define RDI_DISC_INCOMPLETE 0x01 > +#define RDI_DISC_COMPLETE 0x02 > +#define RDI_DISC_RANDOM_WR 0x03 > + > +#define RDI_DISC_PMA_TYPE_CD_ROM 0x00 > +#define RDI_DISC_PMA_TYPE_CDI 0x10 > +#define RDI_DISC_PMA_TYPE_DDCD 0x20 > +#define RDI_DISC_PMA_TYPE_UNDEFINED 0xFF > + > +static void cd_scsi_cmd_get_read_disc_information(CdScsiLU *dev, CdScsiRequest *req) > +{ > + uint8_t *outbuf = req->buf; > + uint32_t data_type; > + uint32_t first_track = 1; > + uint32_t last_track = 1; > + uint32_t num_sessions = 1; > + > + req->xfer_dir = SCSI_XFER_FROM_DEV; > + > + data_type = req->cdb[1] & 0x7; > + if (data_type != RDI_TYPE_STANDARD) { > + SPICE_DEBUG("read_disc_information, lun:%u" > + " unsupported data type: %02x", > + req->lun, data_type); > + cd_scsi_cmd_complete_check_cond(dev, req, &sense_code_INVALID_CDB_FIELD); > + return; > + } > + > + req->req_len = (req->cdb[7] << 8) | req->cdb[8]; > + req->in_len = MIN(req->req_len, RDI_STANDARD_LEN); > + > + memset(outbuf, 0, RDI_STANDARD_LEN); > + outbuf[1] = RDI_STANDARD_LEN - 2; /* length excluding the counter itself */ > + outbuf[2] = RDI_NON_ERAZABLE | RDI_SESSION_COMPLETE | RDI_DISC_COMPLETE; > + outbuf[3] = first_track; /* on disk */ > + outbuf[4] = num_sessions & 0xff; /* lsb */ > + outbuf[5] = first_track & 0xff; /* in last sesson, lsb */ > + outbuf[6] = last_track & 0xff; /* in last sesson, lsb */ > + outbuf[8] = RDI_DISC_PMA_TYPE_CD_ROM; > + outbuf[9] = (num_sessions >> 8) & 0xff; /* msb */ > + outbuf[10] = (first_track >> 8) & 0xff; /* in last sesson, lsb */ > + outbuf[11] = (last_track >> 8) & 0xff; /* in last sesson, lsb */ > + > + SPICE_DEBUG("read_disc_information, lun:%u len: %" G_GUINT64_FORMAT, > + req->lun, req->in_len); > + > + cd_scsi_cmd_complete_good(dev, req); > +} > + > +#define RTI_ADDR_TYPE_LBA 0x00 > +#define RTI_ADDR_TYPE_TRACK_NUM 0x01 > +#define RTI_ADDR_TYPE_SESSION_NUM 0x02 > + > +#define RTI_TRACK_NUM_LEAD_IN 0x00 > +#define RTI_TRACK_NUM_INVISIBLE 0xff > + > +#define TIB_LEN 0x36 > + > +#define TIB_TRACK_MODE_CD 0x04 > +#define TIB_DATA_MODE_ISO_10149 0x01 > + > +#define TIB_LRA_VALID (0x01 << 1) > + > +static void cd_scsi_cmd_get_read_track_information(CdScsiLU *dev, CdScsiRequest *req) > +{ > + uint8_t *outbuf = req->buf; > + uint32_t track_size = dev->num_blocks; > + uint32_t last_addr = track_size - 1; > + uint32_t track_num = 1; > + uint32_t session_num = 1; > + uint32_t addr_type; > + uint32_t addr_num; > + > + req->xfer_dir = SCSI_XFER_FROM_DEV; > + > + addr_type = req->cdb[1] & 0x3; > + addr_num = (req->cdb[2] << 24) | (req->cdb[3] << 16) | > + (req->cdb[4] << 8) | req->cdb[5]; > + > + switch (addr_type) { > + case RTI_ADDR_TYPE_LBA: > + if (addr_num > last_addr) { > + SPICE_DEBUG("read_track_information, lun:%u" > + " addr_type LBA: %u" > + " invalid LBA: %u", > + req->lun, addr_type, addr_num); > + cd_scsi_cmd_complete_check_cond(dev, req, &sense_code_INVALID_CDB_FIELD); > + return; > + } > + break; > + case RTI_ADDR_TYPE_TRACK_NUM: > + if (addr_num != track_num) { > + SPICE_DEBUG("read_track_information, lun:%u" > + " addr_type track: %u" > + " invalid track: %u", > + req->lun, addr_type, addr_num); > + cd_scsi_cmd_complete_check_cond(dev, req, &sense_code_INVALID_CDB_FIELD); > + return; > + } > + break; > + case RTI_ADDR_TYPE_SESSION_NUM: > + if (addr_num != session_num) { > + SPICE_DEBUG("read_track_information, lun:%u" > + " addr_type session: %u" > + " invalid session: %u", > + req->lun, addr_type, addr_num); > + cd_scsi_cmd_complete_check_cond(dev, req, &sense_code_INVALID_CDB_FIELD); > + return; > + } > + break; > + default: > + SPICE_DEBUG("read_track_information, lun:%u" > + " invalid addr_type: %u" > + " addr_num: %u", > + req->lun, addr_type, addr_num); > + cd_scsi_cmd_complete_check_cond(dev, req, &sense_code_INVALID_CDB_FIELD); > + return; > + } > + > + req->req_len = (req->cdb[7] << 8) | req->cdb[8]; > + req->in_len = MIN(req->req_len, TIB_LEN); > + > + memset(outbuf, 0, TIB_LEN); > + outbuf[1] = TIB_LEN - 2; > + outbuf[2] = session_num; > + outbuf[3] = track_num; > + outbuf[5] = TIB_TRACK_MODE_CD & 0x0f; > + outbuf[6] = TIB_DATA_MODE_ISO_10149 & 0x0f; > + outbuf[7] = TIB_LRA_VALID; > + > + /* Track size */ > + outbuf[24] = (track_size >> 24) & 0xff; > + outbuf[25] = (track_size >> 16) & 0xff; > + outbuf[26] = (track_size >> 8) & 0xff; > + outbuf[27] = (track_size) & 0xff; > + > + /* Last recorded address */ > + outbuf[28] = (last_addr >> 24) & 0xff; > + outbuf[29] = (last_addr >> 16) & 0xff; > + outbuf[30] = (last_addr >> 8) & 0xff; > + outbuf[31] = (last_addr) & 0xff; > + > + SPICE_DEBUG("read_track_information, lun:%u" > + "addr_type: %u addr_num: %u", > + req->lun, addr_type, addr_num); > + > + cd_scsi_cmd_complete_good(dev, req); > +} > + > +#define READ_TOC_TRACK_DESC_LEN 8 > +#define READ_TOC_RESP_LEN (4 + 2*READ_TOC_TRACK_DESC_LEN) > + > +static void cd_scsi_cmd_read_toc(CdScsiLU *dev, CdScsiRequest *req) > +{ > + uint8_t *outbuf = req->buf; > + uint32_t msf, format, track_num; > + uint32_t last_blk = dev->num_blocks - 1; > + > + req->xfer_dir = SCSI_XFER_FROM_DEV; > + > + msf = (req->cdb[1] >> 1) & 0x1; > + format = req->cdb[2] & 0xf; > + track_num = req->cdb[6]; > + > + req->req_len = (req->cdb[7] << 8) | req->cdb[8]; > + req->in_len = MIN(req->req_len, READ_TOC_RESP_LEN); > + > + memset(outbuf, 0, READ_TOC_RESP_LEN); > + outbuf[1] = READ_TOC_RESP_LEN - 2; /* length excluding the counter itself */ > + outbuf[2] = 1; /* first track/session */ > + outbuf[3] = 1; /* last track/session */ > + > + outbuf[5] = 0x04; /* Data CD, no Q-subchannel */ > + outbuf[6] = 0x01; /* Track number */ > + outbuf[10] = msf ? 0x02 : 0x0; > + > + outbuf[13] = 0x04; /* Data CD, no Q-subchannel */ > + outbuf[14] = 0xaa; /* Track number */ > + if (msf) { > + last_blk = 0xff300000; > + } > + outbuf[16] = last_blk >> 24; > + outbuf[17] = last_blk >> 16; > + outbuf[18] = last_blk >> 8; > + outbuf[19] = last_blk; > + > + SPICE_DEBUG("read_toc, lun:%u len: %" G_GUINT64_FORMAT > + " msf: %x format: 0x%02x track/session: 0x%02x", > + req->lun, req->in_len, msf, format, track_num); > + > + cd_scsi_cmd_complete_good(dev, req); > +} > + > +#define CD_MODE_PARAM_6_LEN_HEADER 4 > +#define CD_MODE_PARAM_10_LEN_HEADER 8 > + > +#define CD_MODE_PAGE_LEN_RW_ERROR 12 > + > +static uint32_t cd_scsi_add_mode_page_rw_error_recovery(CdScsiLU *dev, uint8_t *outbuf) > +{ > + uint32_t page_len = CD_MODE_PAGE_LEN_RW_ERROR; > + > + outbuf[0] = MODE_PAGE_R_W_ERROR; > + outbuf[1] = CD_MODE_PAGE_LEN_RW_ERROR - 2; > + outbuf[3] = 1; /* read retry count */ > + > + return page_len; > +} > + > +#define CD_MODE_PAGE_LEN_POWER 12 > + > +static uint32_t cd_scsi_add_mode_page_power_condition(CdScsiLU *dev, uint8_t *outbuf) > +{ > + uint32_t page_len = CD_MODE_PAGE_LEN_POWER; > + > + outbuf[0] = MODE_PAGE_POWER; > + outbuf[1] = CD_MODE_PAGE_LEN_POWER - 2; > + > + return page_len; > +} > + > +#define CD_MODE_PAGE_LEN_FAULT_FAIL 12 > +#define CD_MODE_PAGE_FAULT_FAIL_FLAG_PERF 0x80 > + > +static uint32_t cd_scsi_add_mode_page_fault_reporting(CdScsiLU *dev, uint8_t *outbuf) > +{ > + uint32_t page_len = CD_MODE_PAGE_LEN_FAULT_FAIL; > + > + outbuf[0] = MODE_PAGE_FAULT_FAIL; > + outbuf[1] = CD_MODE_PAGE_LEN_FAULT_FAIL - 2; > + outbuf[2] |= CD_MODE_PAGE_FAULT_FAIL_FLAG_PERF; > + > + return page_len; > +} > + > +#define CD_MODE_PAGE_LEN_CAPS_MECH_STATUS_RO 26 > +/* byte 2 */ > +#define CD_MODE_PAGE_CAPS_CD_R_READ 0x01 > +#define CD_MODE_PAGE_CAPS_CD_RW_READ (0x01 << 1) > +#define CD_MODE_PAGE_CAPS_DVD_ROM_READ (0x01 << 3) > +#define CD_MODE_PAGE_CAPS_DVD_R_READ (0x01 << 4) > +#define CD_MODE_PAGE_CAPS_DVD_RAM_READ (0x01 << 5) > +/* byte 6 */ > +#define CD_MODE_PAGE_CAPS_LOCK_SUPPORT (0x01) > +#define CD_MODE_PAGE_CAPS_LOCK_STATE (0x01 << 1) > +#define CD_MODE_PAGE_CAPS_PREVENT_JUMPER (0x01 << 2) > +#define CD_MODE_PAGE_CAPS_EJECT (0x01 << 3) > +#define CD_MODE_PAGE_CAPS_LOADING_TRAY (0x01 << 5) > + > +static uint32_t cd_scsi_add_mode_page_caps_mech_status(CdScsiLU *dev, uint8_t *outbuf) > +{ > + uint32_t page_len = CD_MODE_PAGE_LEN_CAPS_MECH_STATUS_RO; /* no write */ > + > + outbuf[0] = MODE_PAGE_CAPS_MECH_STATUS; > + outbuf[1] = page_len; > + outbuf[2] = CD_MODE_PAGE_CAPS_CD_R_READ | CD_MODE_PAGE_CAPS_CD_RW_READ | > + CD_MODE_PAGE_CAPS_DVD_ROM_READ | CD_MODE_PAGE_CAPS_DVD_R_READ | > + CD_MODE_PAGE_CAPS_DVD_RAM_READ; > + outbuf[6] = CD_MODE_PAGE_CAPS_LOADING_TRAY | CD_MODE_PAGE_CAPS_EJECT | > + CD_MODE_PAGE_CAPS_LOCK_SUPPORT; > + if (dev->prevent_media_removal) { > + outbuf[6] |= CD_MODE_PAGE_CAPS_LOCK_STATE; > + } > + > + return page_len; > +} > + > +static void cd_scsi_cmd_mode_sense_10(CdScsiLU *dev, CdScsiRequest *req) > +{ > + uint8_t *outbuf = req->buf; > + int long_lba, dbd, page, sub_page, pc; > + uint32_t resp_len = CD_MODE_PARAM_10_LEN_HEADER; > + > + req->xfer_dir = SCSI_XFER_FROM_DEV; > + > + long_lba = (req->cdb[1] >> 4) & 0x1; > + dbd = (req->cdb[1] >> 3) & 0x1; > + page = req->cdb[2] & 0x3f; > + pc = req->cdb[2] >> 6; > + sub_page = req->cdb[3]; > + > + req->req_len = (req->cdb[7] << 8) | req->cdb[8]; > + > + memset(outbuf, 0, req->req_len); > + outbuf[2] = 0; /* medium type */ > + > + switch (page) { > + case MODE_PAGE_R_W_ERROR: > + /* Read/Write Error Recovery */ > + resp_len += cd_scsi_add_mode_page_rw_error_recovery(dev, outbuf + resp_len); > + break; > + case MODE_PAGE_POWER: > + /* Power Condistions */ > + resp_len += cd_scsi_add_mode_page_power_condition(dev, outbuf + resp_len); > + break; > + case MODE_PAGE_FAULT_FAIL: > + /* Fault / Failure Reporting Control */ > + resp_len += cd_scsi_add_mode_page_fault_reporting(dev, outbuf + resp_len); > + break; > + case MODE_PAGE_CAPS_MECH_STATUS: > + resp_len += cd_scsi_add_mode_page_caps_mech_status(dev, outbuf + resp_len); > + break; > + > + /* not implemented */ > + case MODE_PAGE_WRITE_PARAMETER: /* Writer Parameters */ > + case MODE_PAGE_MRW: > + case MODE_PAGE_MRW_VENDOR: /* MRW (Mount Rainier Re-writable Disks */ > + case MODE_PAGE_CD_DEVICE: /* CD Device parameters */ > + case MODE_PAGE_TO_PROTECT: /* Time-out and Protect */ > + default: > + SPICE_DEBUG("mode_sense_10, lun:%u" > + " page 0x%x not implemented", > + req->lun, (unsigned)page); > + cd_scsi_cmd_complete_check_cond(dev, req, &sense_code_INVALID_CDB_FIELD); > + return; > + } > + > + outbuf[0] = ((resp_len - 2) >> 8) & 0xff; > + outbuf[1] = (resp_len - 2) & 0xff; > + > + req->in_len = MIN(req->req_len, resp_len); > + > + SPICE_DEBUG("mode_sense_10, lun:%u" > + " long_lba %d, dbd %d, page %d, sub_page %d, pc %d; " > + "resp_len %u", > + req->lun, long_lba, dbd, page, sub_page, pc, resp_len); > + > + cd_scsi_cmd_complete_good(dev, req); > +} > + > +static void cd_scsi_cmd_mode_select_6(CdScsiLU *dev, CdScsiRequest *req) > +{ > + uint8_t *block_desc_data, *mode_data; > + uint32_t page_format, save_pages, list_len; /* cdb */ > + uint32_t num_blocks = 0, block_len = 0; /* block descriptor */ > + uint32_t mode_len, medium_type, dev_param, block_desc_len; /* mode param header */ > + uint32_t page_num = 0, page_len = 0; /* mode page */ > + > + page_format = (req->cdb[1] >> 4) & 0x1; > + save_pages = req->cdb[1] & 0x1; > + list_len = req->cdb[4]; > + > + if (list_len > req->buf_len) { > + SPICE_DEBUG("mode_select_6, lun:%u" > + " pf:%u sp:%u" > + " list_len:%u exceeds data_len:%u", > + req->lun, page_format, save_pages, list_len, req->buf_len); > + cd_scsi_cmd_complete_check_cond(dev, req, &sense_code_INVALID_PARAM_LEN); > + return; > + } > + > + mode_len = req->buf[0]; > + medium_type = req->buf[1]; > + dev_param = req->buf[2]; > + block_desc_len = req->buf[3]; > + > + if (block_desc_len) { > + block_desc_data = &req->buf[CD_MODE_PARAM_6_LEN_HEADER]; > + num_blocks = (block_desc_data[3] << 16) | (block_desc_data[2] << 8) | block_desc_data[3]; > + block_len = (block_desc_data[5] << 16) | (block_desc_data[6] << 8) | block_desc_data[7]; > + } > + > + if (mode_len) { > + mode_data = &req->buf[CD_MODE_PARAM_6_LEN_HEADER]; > + if (block_desc_len) { > + mode_data += block_desc_len; > + } > + page_num = mode_data[0] & 0x3f; > + page_len = mode_data[1]; > + } > + > + SPICE_DEBUG("mode_select_6, lun:%u" > + " pf:%u sp:%u list_len:%u data_len:%u" > + " mode_len:%u medium:%u dev_param:%u blk_desc_len:%u" > + " num_blocks:%u block_len:%u" > + " page_num:%u page_len:%u", > + req->lun, page_format, save_pages, list_len, req->buf_len, > + mode_len, medium_type, dev_param, block_desc_len, > + num_blocks, block_len, > + page_num, page_len); > + > + cd_scsi_cmd_complete_good(dev, req); > +} > + > +static void cd_scsi_cmd_mode_select_10(CdScsiLU *dev, CdScsiRequest *req) > +{ > + uint32_t page_format, save_pages, list_len; > + > + page_format = (req->cdb[1] >> 4) & 0x1; > + save_pages = req->cdb[1] & 0x1; > + list_len = (req->cdb[7] << 8) | req->cdb[8]; > + > + if (list_len > req->buf_len) { > + SPICE_DEBUG("mode_select_10, lun:%u pf:%u sp:%u" > + " list_len:%u exceeds data_len:%u", > + req->lun, page_format, save_pages, list_len, req->buf_len); > + cd_scsi_cmd_complete_check_cond(dev, req, &sense_code_INVALID_PARAM_LEN); > + return; > + } > + > + SPICE_DEBUG("mode_select_10, lun:%u pf:%u sp:%u" > + " list_len:%u data_len:%u", > + req->lun, page_format, save_pages, list_len, req->buf_len); > + > + cd_scsi_cmd_complete_good(dev, req); > +} > + > +#define CD_FEATURE_HEADER_LEN 8 > +#define CD_FEATURE_DESC_LEN 4 > + > +#define CD_PROFILE_DESC_LEN 4 > +#define CD_PROFILE_CURRENT 0x01 > + > +/* Profiles List */ > +#define CD_FEATURE_NUM_PROFILES_LIST 0x00 > +/* Core - Basic Functionality */ > +#define CD_FEATURE_NUM_CORE 0x01 > +/* Morphing - The device changes its behavior due to external events */ > +#define CD_FEATURE_NUM_MORPH 0x02 > +/* Removable Medium - The medium may be removed from the device */ > +#define CD_FEATURE_NUM_REMOVABLE 0x03 > +/* Random Readable - PP=1 Read ability for storage devices with random addressing */ > +#define CD_FEATURE_NUM_RANDOM_READ 0x10 > +/* CD Read - The ability to read CD specific structures */ > +#define CD_FEATURE_NUM_CD_READ 0x1E > +/* DVD Read - The ability to read DVD specific structures */ > +#define CD_FEATURE_NUM_DVD_READ 0x1F > +/* Power Management - Initiator and device directed power management */ > +#define CD_FEATURE_NUM_POWER_MNGT 0x100 > +/* Timeout */ > +#define CD_FEATURE_NUM_TIMEOUT 0x105 > + > +#define CD_FEATURE_REQ_ALL 0 > +#define CD_FEATURE_REQ_CURRENT 1 > +#define CD_FEATURE_REQ_SINGLE 2 > + > +#define CD_FEATURE_CURRENT 0x01 > +#define CD_FEATURE_PERSISTENT 0x02 > + > +#define CD_FEATURE_VERSION_1 (0x01 << 2) > + > +#define CD_FEATURE_PHYS_IF_SCSI 0x01 > + > +#define CD_FEATURE_REMOVABLE_LOADING_TRAY (0x01 << 5) > +#define CD_FEATURE_REMOVABLE_EJECT (0x01 << 3) > +#define CD_FEATURE_REMOVABLE_NO_PRVNT_JMPR (0x01 << 2) > +#define CD_FEATURE_REMOVABLE_LOCK (0x01) > + > +static gboolean cd_scsi_feature_reportable(uint32_t feature, uint32_t start_feature, > + uint32_t req_type) > +{ > + return (req_type == CD_FEATURE_REQ_SINGLE && start_feature == feature) || > + (feature >= start_feature); > +} > + > +static uint32_t cd_scsi_add_feature_profiles_list(CdScsiLU *dev, uint8_t *outbuf, > + uint32_t start_feature, uint32_t req_type) > +{ > + uint8_t *profile = outbuf + CD_FEATURE_DESC_LEN; > + uint32_t feature_len = CD_FEATURE_DESC_LEN; > + uint32_t add_len, profile_num; > + > + if (!cd_scsi_feature_reportable(CD_FEATURE_NUM_PROFILES_LIST, start_feature, req_type)) { > + return 0; > + } > + /* feature descriptor header */ > + outbuf[0] = (CD_FEATURE_NUM_PROFILES_LIST >> 8) & 0xff; > + outbuf[1] = CD_FEATURE_NUM_PROFILES_LIST & 0xff; > + outbuf[2] = CD_FEATURE_PERSISTENT | CD_FEATURE_CURRENT; > + > + /* DVD-ROM profile descriptor */ > + add_len = CD_PROFILE_DESC_LEN; /* start with single profile, add later */ > + profile_num = MMC_PROFILE_DVD_ROM; > + > + profile[0] = (profile_num >> 8) & 0xff; /* feature code */ > + profile[1] = profile_num & 0xff; > + profile[2] = (!dev->cd_rom) ? CD_PROFILE_CURRENT : 0; > + > + /* next profile */ > + add_len += CD_PROFILE_DESC_LEN; > + profile += CD_PROFILE_DESC_LEN; > + > + /* CD-ROM profile descriptor */ > + profile_num = MMC_PROFILE_CD_ROM; > + profile[0] = (profile_num >> 8) & 0xff; > + profile[1] = profile_num & 0xff; > + profile[2] = dev->cd_rom ? CD_PROFILE_CURRENT : 0; > + > + outbuf[3] = add_len; > + feature_len += add_len; > + > + return feature_len; > +} > + > +#define CD_FEATURE_CORE_PHYS_PROFILE_LEN 4 > + > +static uint32_t cd_scsi_add_feature_core(CdScsiLU *dev, uint8_t *outbuf, > + uint32_t start_feature, uint32_t req_type) > +{ > + uint8_t *profile = outbuf + CD_FEATURE_DESC_LEN; > + uint32_t feature_len = CD_FEATURE_DESC_LEN + CD_FEATURE_CORE_PHYS_PROFILE_LEN; > + > + if (!cd_scsi_feature_reportable(CD_FEATURE_NUM_CORE, start_feature, req_type)) { > + return 0; > + } > + outbuf[0] = (CD_FEATURE_NUM_CORE >> 8) & 0xff; > + outbuf[1] = CD_FEATURE_NUM_CORE & 0xff; > + outbuf[2] = CD_FEATURE_PERSISTENT | CD_FEATURE_CURRENT; > + outbuf[3] = CD_FEATURE_CORE_PHYS_PROFILE_LEN; > + > + profile[3] = CD_FEATURE_PHYS_IF_SCSI; > + > + return feature_len; > +} > + > +#define CD_FEATURE_MORPH_PROGILE_LEN 4 > +#define CD_FEATURE_MORPH_ASYNC_EVENTS 0x01 > + > +static uint32_t cd_scsi_add_feature_morph(CdScsiLU *dev, uint8_t *outbuf, > + uint32_t start_feature, uint32_t req_type) > +{ > + uint8_t *profile = outbuf + CD_FEATURE_DESC_LEN; > + uint32_t feature_len = CD_FEATURE_DESC_LEN + CD_FEATURE_MORPH_PROGILE_LEN; > + > + if (!cd_scsi_feature_reportable(CD_FEATURE_NUM_MORPH, start_feature, req_type)) { > + return 0; > + } > + outbuf[1] = CD_FEATURE_NUM_MORPH; > + outbuf[2] = CD_FEATURE_PERSISTENT | CD_FEATURE_CURRENT; > + outbuf[3] = CD_FEATURE_MORPH_PROGILE_LEN; > + > + profile[0] = CD_FEATURE_MORPH_ASYNC_EVENTS; > + > + return feature_len; > +} > + > +#define CD_FEATURE_REMOVABLE_PROFILE_LEN 4 > + > +static uint32_t cd_scsi_add_feature_removable(CdScsiLU *dev, uint8_t *outbuf, > + uint32_t start_feature, uint32_t req_type) > +{ > + uint8_t *profile = outbuf + CD_FEATURE_DESC_LEN; > + uint32_t feature_len = CD_FEATURE_DESC_LEN + CD_FEATURE_REMOVABLE_PROFILE_LEN; > + > + if (!cd_scsi_feature_reportable(CD_FEATURE_NUM_REMOVABLE, start_feature, req_type)) { > + return 0; > + } > + outbuf[1] = CD_FEATURE_NUM_REMOVABLE; > + outbuf[2] = CD_FEATURE_PERSISTENT | CD_FEATURE_CURRENT; > + outbuf[3] = CD_FEATURE_REMOVABLE_PROFILE_LEN; > + > + profile[0] = CD_FEATURE_REMOVABLE_NO_PRVNT_JMPR; > + if (dev->removable) { > + profile[0] |= (CD_FEATURE_REMOVABLE_LOADING_TRAY | CD_FEATURE_REMOVABLE_EJECT); > + } > + > + return feature_len; > +} > + > +#define CD_FEATURE_RANDOM_READ_PROFILE_LEN 8 > + > +static uint32_t cd_scsi_add_feature_random_read(CdScsiLU *dev, uint8_t *outbuf, > + uint32_t start_feature, uint32_t req_type) > +{ > + uint8_t *profile = outbuf + CD_FEATURE_DESC_LEN; > + uint32_t feature_len = CD_FEATURE_DESC_LEN + CD_FEATURE_RANDOM_READ_PROFILE_LEN; > + > + if (!cd_scsi_feature_reportable(CD_FEATURE_NUM_RANDOM_READ, start_feature, req_type)) { > + return 0; > + } > + outbuf[0] = (CD_FEATURE_NUM_RANDOM_READ >> 8) & 0xff; > + outbuf[1] = CD_FEATURE_NUM_RANDOM_READ & 0xff; > + outbuf[2] = CD_FEATURE_PERSISTENT | CD_FEATURE_CURRENT; > + outbuf[3] = CD_FEATURE_RANDOM_READ_PROFILE_LEN; > + > + profile[0] = (dev->block_size >> 24) & 0xff; > + profile[1] = (dev->block_size >> 16) & 0xff; > + profile[2] = (dev->block_size >> 8) & 0xff; > + profile[3] = (dev->block_size) & 0xff; > + profile[5] = (dev->cd_rom) ? 0x01 : 0x10; /* logical blocks per readable unit */ > + > + return feature_len; > +} > + > +#define CD_FEATURE_CD_READ_PROFILE_LEN 4 > + > +static uint32_t cd_scsi_add_feature_cd_read(CdScsiLU *dev, uint8_t *outbuf, > + uint32_t start_feature, uint32_t req_type) > +{ > + uint8_t *profile = outbuf + CD_FEATURE_DESC_LEN; > + uint32_t feature_len = CD_FEATURE_DESC_LEN + CD_FEATURE_CD_READ_PROFILE_LEN; > + > + if (!cd_scsi_feature_reportable(CD_FEATURE_NUM_CD_READ, start_feature, req_type)) { > + return 0; > + } > + outbuf[0] = (CD_FEATURE_NUM_CD_READ >> 8) & 0xff; > + outbuf[1] = (CD_FEATURE_NUM_CD_READ) & 0xff; > + outbuf[2] = CD_FEATURE_VERSION_1 | CD_FEATURE_PERSISTENT | CD_FEATURE_CURRENT; > + outbuf[3] = CD_FEATURE_CD_READ_PROFILE_LEN; > + > + profile[0] = 0; /* C2 Errors, CD-Text not supporte */ > + > + return feature_len; > +} > + > +#define CD_FEATURE_DVD_READ_PROFILE_LEN 0 > + > +static uint32_t cd_scsi_add_feature_dvd_read(CdScsiLU *dev, uint8_t *outbuf, > + uint32_t start_feature, uint32_t req_type) > +{ > + uint32_t feature_len = CD_FEATURE_DESC_LEN + CD_FEATURE_DVD_READ_PROFILE_LEN; > + > + if (!cd_scsi_feature_reportable(CD_FEATURE_NUM_CD_READ, start_feature, req_type)) { > + return 0; > + } > + outbuf[0] = (CD_FEATURE_NUM_DVD_READ >> 8) & 0xff; > + outbuf[1] = (CD_FEATURE_NUM_DVD_READ) & 0xff; > + outbuf[2] = CD_FEATURE_VERSION_1 | CD_FEATURE_PERSISTENT | CD_FEATURE_CURRENT; > + outbuf[3] = CD_FEATURE_DVD_READ_PROFILE_LEN; > + > + return feature_len; > +} > + > +#define CD_FEATURE_POWER_MNGT_PROFILE_LEN 0 > + > +static uint32_t cd_scsi_add_feature_power_mgmt(CdScsiLU *dev, uint8_t *outbuf, > + uint32_t start_feature, uint32_t req_type) > +{ > + uint32_t feature_len = CD_FEATURE_DESC_LEN + CD_FEATURE_POWER_MNGT_PROFILE_LEN; > + > + if (!cd_scsi_feature_reportable(CD_FEATURE_NUM_POWER_MNGT, start_feature, req_type)) { > + return 0; > + } > + outbuf[0] = (CD_FEATURE_NUM_POWER_MNGT >> 8) & 0xff; > + outbuf[1] = (CD_FEATURE_NUM_POWER_MNGT) & 0xff; > + outbuf[2] = CD_FEATURE_PERSISTENT | CD_FEATURE_CURRENT; > + outbuf[3] = CD_FEATURE_POWER_MNGT_PROFILE_LEN; > + > + return feature_len; > +} > + > +#define CD_FEATURE_TIMEOUT_PROFILE_LEN 0 > + > +static uint32_t cd_scsi_add_feature_timeout(CdScsiLU *dev, uint8_t *outbuf, > + uint32_t start_feature, uint32_t req_type) > +{ > + uint32_t feature_len = CD_FEATURE_DESC_LEN + CD_FEATURE_TIMEOUT_PROFILE_LEN; > + > + if (!cd_scsi_feature_reportable(CD_FEATURE_NUM_TIMEOUT, start_feature, req_type)) { > + return 0; > + } > + outbuf[0] = (CD_FEATURE_NUM_TIMEOUT >> 8) & 0xff; > + outbuf[1] = CD_FEATURE_NUM_TIMEOUT & 0xff; > + outbuf[2] = CD_FEATURE_PERSISTENT | CD_FEATURE_CURRENT; > + outbuf[3] = CD_FEATURE_TIMEOUT_PROFILE_LEN; > + > + return feature_len; > +} > + > +static void cd_scsi_cmd_get_configuration(CdScsiLU *dev, CdScsiRequest *req) > +{ > + uint8_t *outbuf = req->buf; > + uint32_t profile_num = (!dev->cd_rom) ? MMC_PROFILE_DVD_ROM : MMC_PROFILE_CD_ROM; > + uint32_t req_type, start_feature, resp_len; > + > + req->xfer_dir = SCSI_XFER_FROM_DEV; > + > + req_type = req->cdb[1] & 0x3; > + start_feature = (req->cdb[2] << 8) | req->cdb[3]; > + req->req_len = (req->cdb[7] << 8) | req->cdb[8]; > + > + memset(outbuf, 0, req->req_len); > + > + /* at least Feature Header should be present, to be filled later */ > + resp_len = CD_FEATURE_HEADER_LEN; > + > + switch (req_type) { > + case CD_FEATURE_REQ_ALL: > + case CD_FEATURE_REQ_CURRENT: > + resp_len += cd_scsi_add_feature_profiles_list(dev, outbuf + resp_len, start_feature, > + req_type); > + resp_len += cd_scsi_add_feature_core(dev, outbuf + resp_len, start_feature, req_type); > + resp_len += cd_scsi_add_feature_morph(dev, outbuf + resp_len, start_feature, req_type); > + resp_len += cd_scsi_add_feature_removable(dev, outbuf + resp_len, start_feature, req_type); > + resp_len += cd_scsi_add_feature_random_read(dev, outbuf + resp_len, start_feature, > + req_type); > + resp_len += cd_scsi_add_feature_cd_read(dev, outbuf + resp_len, start_feature, req_type); > + resp_len += cd_scsi_add_feature_dvd_read(dev, outbuf + resp_len, start_feature, req_type); > + resp_len += cd_scsi_add_feature_power_mgmt(dev, outbuf + resp_len, start_feature, req_type); > + resp_len += cd_scsi_add_feature_timeout(dev, outbuf + resp_len, start_feature, req_type); > + break; > + case CD_FEATURE_REQ_SINGLE: > + switch (start_feature) { > + case CD_FEATURE_NUM_CORE: > + resp_len += cd_scsi_add_feature_core(dev, outbuf + resp_len, start_feature, req_type); > + break; > + case CD_FEATURE_NUM_MORPH: > + resp_len += cd_scsi_add_feature_morph(dev, outbuf + resp_len, start_feature, req_type); > + break; > + case CD_FEATURE_NUM_REMOVABLE: > + resp_len += cd_scsi_add_feature_removable(dev, outbuf + resp_len, start_feature, > + req_type); > + break; > + case CD_FEATURE_NUM_RANDOM_READ: > + resp_len += cd_scsi_add_feature_random_read(dev, outbuf + resp_len, start_feature, > + req_type); > + break; > + case CD_FEATURE_NUM_CD_READ: > + resp_len += cd_scsi_add_feature_cd_read(dev, outbuf + resp_len, start_feature, > + req_type); > + break; > + case CD_FEATURE_NUM_DVD_READ: > + resp_len += cd_scsi_add_feature_dvd_read(dev, outbuf + resp_len, start_feature, > + req_type); > + break; > + case CD_FEATURE_NUM_POWER_MNGT: > + resp_len += cd_scsi_add_feature_power_mgmt(dev, outbuf + resp_len, start_feature, > + req_type); > + break; > + case CD_FEATURE_NUM_TIMEOUT: > + resp_len += cd_scsi_add_feature_timeout(dev, outbuf + resp_len, start_feature, > + req_type); > + break; > + default: > + break; > + } > + break; > + > + default: > + SPICE_DEBUG("get_configuration, lun:%u invalid rt:%u start_f:%u", > + req->lun, req_type, start_feature); > + cd_scsi_cmd_complete_check_cond(dev, req, &sense_code_INVALID_CDB_FIELD); > + return; > + } > + > + /* set total data len */ > + outbuf[0] = (resp_len >> 24) & 0xff; > + outbuf[1] = (resp_len >> 16) & 0xff; > + outbuf[2] = (resp_len >> 8) & 0xff; > + outbuf[3] = resp_len & 0xff; > + > + /* report current profile num */ > + outbuf[6] = (profile_num >> 8) & 0xff; > + outbuf[7] = profile_num & 0xff; > + > + req->in_len = MIN(req->req_len, resp_len); > + > + SPICE_DEBUG("get_configuration, lun:%u rt:%u start_f:%u resp_len:%u", > + req->lun, req_type, start_feature, resp_len); > + > + cd_scsi_cmd_complete_good(dev, req); > +} > + > +#define CD_GET_EVENT_STATUS_IMMED 0x01 > + > +#define CD_GET_EVENT_HEADER_NO_EVENT_AVAIL (0x01 << 7) > +#define CD_GET_EVENT_HEADER_LEN 4 > + > +#define CD_GET_EVENT_CLASS_NONE (0x00) > +#define CD_GET_EVENT_CLASS_OPER_CHANGE (0x01) > +#define CD_GET_EVENT_CLASS_POWER_MGMT (0x02) > +#define CD_GET_EVENT_CLASS_EXTERNAL_REQ (0x03) > +#define CD_GET_EVENT_CLASS_MEDIA (0x04) > +#define CD_GET_EVENT_CLASS_MULTI_INITIATOR (0x05) > +#define CD_GET_EVENT_CLASS_DEV_BUSY (0x06) > + > +#define CD_GET_EVENT_LEN_MEDIA 4 > + > +#define CD_MEDIA_STATUS_MEDIA_PRESENT 0x1 > +#define CD_MEDIA_STATUS_TRAY_OPEN 0x2 > + > +static uint32_t cd_scsi_cmd_get_event_resp_add_media(CdScsiLU *dev, uint8_t *outbuf) > +{ > + outbuf[0] = (uint8_t)dev->media_event & 0x0f; > + outbuf[1] = (uint8_t)((dev->loaded ? 0 : CD_MEDIA_STATUS_TRAY_OPEN) | > + (dev->stream != NULL ? CD_MEDIA_STATUS_MEDIA_PRESENT : 0)); > + > + dev->media_event = CD_MEDIA_EVENT_NO_CHANGE; /* reset the event */ > + return CD_GET_EVENT_LEN_MEDIA; > +} > + > +#define CD_GET_EVENT_LEN_POWER 4 > + > +#define CD_POWER_STATUS_ACTIVE 0x1 > +#define CD_POWER_STATUS_IDLE 0x2 > + > +static uint32_t cd_scsi_cmd_get_event_resp_add_power(CdScsiLU *dev, uint8_t *outbuf) > +{ > + outbuf[0] = (uint8_t)dev->power_event & 0x0f; > + outbuf[1] = (uint8_t)((dev->power_cond == CD_SCSI_POWER_ACTIVE) ? > + CD_POWER_STATUS_ACTIVE : CD_POWER_STATUS_IDLE); > + > + dev->power_event = CD_POWER_EVENT_NO_CHANGE; /* reset the event */ > + return CD_GET_EVENT_LEN_POWER; > +} > + > +static void cd_scsi_cmd_get_event_status_notification(CdScsiLU *dev, CdScsiRequest *req) > +{ > + uint8_t *outbuf = req->buf; > + uint32_t resp_len = CD_GET_EVENT_HEADER_LEN; > + const uint32_t power_class_mask = (0x01 << CD_GET_EVENT_CLASS_POWER_MGMT); > + const uint32_t media_class_mask = (0x01 << CD_GET_EVENT_CLASS_MEDIA); > + uint32_t classes_supported = power_class_mask | media_class_mask; > + uint32_t immed, classes_requested; > + > + req->xfer_dir = SCSI_XFER_FROM_DEV; > + > + immed = req->cdb[1] & CD_GET_EVENT_STATUS_IMMED; > + classes_requested = req->cdb[4]; > + req->req_len = (req->cdb[7] << 8) | req->cdb[8]; > + > + if (!immed) { > + SPICE_DEBUG("get_event_status_notification, lun:%u" > + " imm:0 class_req:%02x, Non-immediate (async) mode unsupported", > + req->lun, classes_requested); > + cd_scsi_cmd_complete_check_cond(dev, req, &sense_code_INVALID_CDB_FIELD); > + return; > + } > + > + memset(outbuf, 0, req->req_len); > + if ((classes_supported & classes_requested) != 0) { > + if (classes_requested & power_class_mask) { > + outbuf[2] = CD_GET_EVENT_CLASS_POWER_MGMT; > + outbuf[3] = (uint8_t)power_class_mask; > + > + SPICE_DEBUG("get_event_status_notification, lun:%u" > + " imm:%u class_req:0x%02x class_sup:0x%02x" > + " power_event:0x%02x power_cond:0x%02x", > + req->lun, immed, classes_requested, classes_supported, > + dev->power_event, dev->power_cond); > + > + resp_len += cd_scsi_cmd_get_event_resp_add_power(dev, outbuf + resp_len); > + } else if (classes_requested & media_class_mask) { > + outbuf[2] = CD_GET_EVENT_CLASS_MEDIA; > + outbuf[3] = (uint8_t)media_class_mask; > + > + SPICE_DEBUG("get_event_status_notification, lun:%u" > + " imm:%u class_req:0x%02x class_sup:0x%02x" > + " media_event:0x%02x loaded: %d", > + req->lun, immed, classes_requested, classes_supported, > + dev->media_event, dev->loaded); > + > + resp_len += cd_scsi_cmd_get_event_resp_add_media(dev, outbuf + resp_len); > + } > + } else { > + outbuf[2] = CD_GET_EVENT_HEADER_NO_EVENT_AVAIL | CD_GET_EVENT_CLASS_NONE; > + > + SPICE_DEBUG("get_event_status_notification, lun:%u" > + " imm:%u class_req:0x%02x class_sup:0x%02x" > + " none of requested events supported", > + req->lun, immed, classes_requested, classes_supported); > + } > + outbuf[1] = (uint8_t)(resp_len - 2); /* Event Data Length LSB, length excluding the > + * field itself */ > + outbuf[3] = (uint8_t)classes_supported; > + > + req->in_len = MIN(req->req_len, resp_len); > + cd_scsi_cmd_complete_good(dev, req); > +} > + > +#define CD_EXT_REQ_EVENT_FORMAT_NO_CHG 0x00 > +#define CD_EXT_REQ_EVENT_FORMAT_LU_KEY_DOWN 0x01 > +#define CD_EXT_REQ_EVENT_FORMAT_LU_KEY_UP 0x02 > +#define CD_EXT_REQ_EVENT_FORMAT_REQ_NOTIFY 0x03 > + > +#define CD_EXT_REQ_STATUS_READY 0x00 > +#define CD_EXT_REQ_STATUS_OTHER_PREVENT 0x01 > + > +#define CD_EXT_REQ_CODE_NO_REQUEST 0x00 > +#define CD_EXT_REQ_CODE_OVERRUN 0x01 > +#define CD_EXT_REQ_CODE_PLAY 0x101 > +#define CD_EXT_REQ_CODE_REWIND 0x102 > +#define CD_EXT_REQ_CODE_FAST_FW 0x103 > +#define CD_EXT_REQ_CODE_PAUSE 0x104 > +#define CD_EXT_REQ_CODE_STOP 0x106 > +#define CD_EXT_REQ_CODE_ASCII_BASE 0x200 /* SCSII value is LSB */ > + > +static void cd_scsi_cmd_send_event(CdScsiLU *dev, CdScsiRequest *req) > +{ > + uint8_t *param, *event; > + uint32_t immed, param_list_len; > + uint32_t event_param_len, notification_class; > + uint32_t ext_req_event, ext_req_status, pers_prevent, ext_req_code; > + > + req->xfer_dir = SCSI_XFER_TO_DEV; > + > + immed = req->cdb[1] & 0x01; > + param_list_len = (req->cdb[8] << 8) | req->cdb[9]; > + > + if (req->buf_len < param_list_len) { > + SPICE_DEBUG("send_event, lun:%u invalid param list len:0x%x, buf_len:0x%x", > + req->lun, param_list_len, req->buf_len); > + cd_scsi_cmd_complete_check_cond(dev, req, &sense_code_INVALID_PARAM_LEN); > + return; > + } > + param = req->buf; > + event_param_len = (param[0] << 8) | param[1]; > + > + notification_class = param[2] & 0x07; > + if (notification_class != CD_GET_EVENT_CLASS_EXTERNAL_REQ) { > + SPICE_DEBUG("send_event, lun:%u invalid notification class:0x%x", > + req->lun, notification_class); > + cd_scsi_cmd_complete_check_cond(dev, req, &sense_code_INVALID_CDB_FIELD); > + return; > + } > + > + event = param + CD_GET_EVENT_HEADER_LEN; > + ext_req_event = event[0] & 0xff; > + ext_req_status = event[1] & 0x0f; > + pers_prevent = event[1] & 0x80; > + ext_req_code = (event[2] << 8) | event[3]; > + > + SPICE_DEBUG("send_event, lun:0x%u immed:%u param_len:%u" > + " ext_req_event:0x%x ext_req_status:0x%x" > + " pers_prevent:0x%x ext_req_code:0x%x", > + req->lun, immed, event_param_len, ext_req_event, > + ext_req_status, pers_prevent, ext_req_code); > + > + /* ToDo: process the event */ > + > + cd_scsi_cmd_complete_good(dev, req); > +} > + > +#define CD_MEDIUM_REMOVAL_REQ_ALLOW 0x00 > +#define CD_MEDIUM_REMOVAL_REQ_PREVENT 0x01 > +#define CD_MEDIUM_REMOVAL_REQ_ALLOW_CHANGER 0x02 > +#define CD_MEDIUM_REMOVAL_REQ_PREVENT_CHANGER 0x03 > + > +static void cd_scsi_cmd_allow_medium_removal(CdScsiLU *dev, CdScsiRequest *req) > +{ > + uint32_t prevent; > + > + req->xfer_dir = SCSI_XFER_FROM_DEV; > + > + prevent = req->cdb[4] & 0x03; > + dev->prevent_media_removal = (prevent == CD_MEDIUM_REMOVAL_REQ_PREVENT || > + prevent == CD_MEDIUM_REMOVAL_REQ_PREVENT_CHANGER); > + req->in_len = 0; > + > + SPICE_DEBUG("allow_medium_removal, lun:%u prevent field::0x%02x flag:%d", > + req->lun, prevent, dev->prevent_media_removal); > + > + cd_scsi_cmd_complete_good(dev, req); > +} > + > +static void cd_scsi_cmd_report_key(CdScsiLU *dev, CdScsiRequest *req) > +{ > + SPICE_DEBUG("report_key - content protection unsupported, lun:%u", req->lun); > + req->xfer_dir = SCSI_XFER_NONE; > + cd_scsi_cmd_complete_check_cond(dev, req, &sense_code_INVALID_OPCODE); > +} > + > +static void cd_scsi_cmd_send_key(CdScsiLU *dev, CdScsiRequest *req) > +{ > + SPICE_DEBUG("send_key - content protection unsupported, lun:%u", req->lun); > + req->xfer_dir = SCSI_XFER_NONE; > + cd_scsi_cmd_complete_check_cond(dev, req, &sense_code_INVALID_OPCODE); > +} > + > +/* byte 1 */ > +#define CD_START_STOP_FLAG_IMMED 0x01 > + > +/* byte 4 */ > +#define CD_START_STOP_FLAG_START 0x01 > +#define CD_START_STOP_FLAG_LOEJ 0x02 > + > +/* POWER CONDITION field values */ > +#define CD_START_STOP_POWER_COND_START_VALID 0x00 > +#define CD_START_STOP_POWER_COND_ACTIVE 0x01 > +#define CD_START_STOP_POWER_COND_IDLE 0x02 > +#define CD_START_STOP_POWER_COND_STANDBY 0x03 > +#define CD_START_STOP_POWER_COND_LU_CONTROL 0x07 > +#define CD_START_STOP_POWER_COND_FORCE_IDLE_0 0x0a > +#define CD_START_STOP_POWER_COND_FORCE_STANDBY_0 0x0b > + > +static inline const char *cd_scsi_start_stop_power_cond_name(uint32_t power_cond) > +{ > + switch (power_cond) { > + case CD_START_STOP_POWER_COND_START_VALID: > + return "START_VALID"; > + case CD_START_STOP_POWER_COND_ACTIVE: > + return "ACTIVE"; > + case CD_START_STOP_POWER_COND_IDLE: > + return "IDLE"; > + case CD_START_STOP_POWER_COND_STANDBY: > + return "STANDBY"; > + case CD_START_STOP_POWER_COND_LU_CONTROL: > + return "LU_CONTROL"; > + case CD_START_STOP_POWER_COND_FORCE_IDLE_0: > + return "FORCE_IDLE_0"; > + case CD_START_STOP_POWER_COND_FORCE_STANDBY_0: > + return "FORCE_STANDBY_0"; > + default: > + return "RESERVED"; > + } > +} > + > +static void cd_scsi_cmd_start_stop_unit(CdScsiLU *dev, CdScsiRequest *req) > +{ > + gboolean immed, start, load_eject; > + uint32_t power_cond; > + > + req->xfer_dir = SCSI_XFER_NONE; > + req->in_len = 0; > + > + immed = (req->cdb[1] & CD_START_STOP_FLAG_IMMED) ? TRUE : FALSE; > + start = (req->cdb[4] & CD_START_STOP_FLAG_START) ? TRUE : FALSE; > + load_eject = (req->cdb[4] & CD_START_STOP_FLAG_LOEJ) ? TRUE : FALSE; > + power_cond = req->cdb[4] >> 4; > + > + SPICE_DEBUG("start_stop_unit, lun:0x%u" > + " immed:%d start:%d load_eject:%d power_cond:0x%x(%s)", > + req->lun, immed, start, load_eject, power_cond, > + cd_scsi_start_stop_power_cond_name(power_cond)); > + > + switch (power_cond) { > + case CD_START_STOP_POWER_COND_START_VALID: > + if (!start) { /* stop the unit */ > + if (load_eject) { /* eject medium */ > + if (dev->prevent_media_removal) { > + SPICE_DEBUG("start_stop_unit, lun:0x%u" > + " prevent_media_removal set, eject failed", req->lun); > + cd_scsi_cmd_complete_check_cond(dev, req, &sense_code_MEDIUM_REMOVAL_PREVENTED); > + return; > + } > + SPICE_DEBUG("start_stop_unit, lun:0x%u eject", req->lun); > + cd_scsi_lu_unload(dev); > + cd_scsi_dev_changed(dev->tgt->user_data, req->lun); > + } > + dev->power_cond = CD_SCSI_POWER_STOPPED; > + SPICE_DEBUG("start_stop_unit, lun:0x%u stopped", req->lun); > + } else { /* start the unit */ > + dev->power_cond = CD_SCSI_POWER_ACTIVE; > + SPICE_DEBUG("start_stop_unit, lun:0x%u started", req->lun); > + > + if (load_eject) { /* load medium */ > + SPICE_DEBUG("start_stop_unit, lun:0x%u load with no media", > + req->lun); > + cd_scsi_lu_load(dev, NULL); > + cd_scsi_dev_changed(dev->tgt->user_data, req->lun); > + } > + } > + break; > + case CD_START_STOP_POWER_COND_ACTIVE: > + /* not error to specify transition to the current power condition */ > + dev->power_cond = CD_SCSI_POWER_ACTIVE; > + SPICE_DEBUG("start_stop_unit, lun:0x%u active", req->lun); > + break; > + case CD_START_STOP_POWER_COND_IDLE: > + case CD_START_STOP_POWER_COND_FORCE_IDLE_0: > + dev->power_cond = CD_SCSI_POWER_IDLE; > + SPICE_DEBUG("start_stop_unit, lun:0x%u idle", req->lun); > + break; > + case CD_START_STOP_POWER_COND_STANDBY: > + case CD_START_STOP_POWER_COND_FORCE_STANDBY_0: > + dev->power_cond = CD_SCSI_POWER_STANDBY; > + SPICE_DEBUG("start_stop_unit, lun:0x%u standby", req->lun); > + break; > + case CD_START_STOP_POWER_COND_LU_CONTROL: > + break; > + default: > + cd_scsi_cmd_complete_check_cond(dev, req, &sense_code_INVALID_CDB_FIELD); > + return; > + } > + cd_scsi_cmd_complete_good(dev, req); > +} > + > +#define CD_PERF_TYPE_PERFORMANCE 0x00 > +#define CD_PERF_TYPE_UNUSABLE_AREA 0x01 > +#define CD_PERF_TYPE_DEFECT_STATUS 0x02 > +#define CD_PERF_TYPE_WRITE_SPEED 0x03 > + > +#define CD_PERF_HEADER_LEN 8 > + > +#define CD_PERF_TYPE_PERFORMANCE_DESCR_LEN 16 > + > +#define CD_PERF_TYPE_PERFORMANCE_REPORT_NOMINAL 0x00 > +#define CD_PERF_TYPE_PERFORMANCE_REPORT_ALL 0x01 > +#define CD_PERF_TYPE_PERFORMANCE_REPORT_EXCEPT 0x10 > + > + > +static void cd_scsi_get_performance_resp_empty(CdScsiLU *dev, CdScsiRequest *req, > + uint32_t type, uint32_t data_type, > + uint32_t max_num_descr) > +{ > + uint8_t *outbuf = req->buf; > + uint32_t write = (data_type >> 2) & 0x01; > + > + memset(outbuf, 0, CD_PERF_HEADER_LEN); > + if (write) { > + outbuf[4] = 0x02; > + } > + req->in_len = CD_PERF_HEADER_LEN; > + > + SPICE_DEBUG("get_performance, lun:%u" > + " type:0x%x data_type:0x%x - sending empty response", > + req->lun, type, data_type); > + > + cd_scsi_cmd_complete_good(dev, req); > +} > + > +static void cd_scsi_get_performance_resp_performance(CdScsiLU *dev, CdScsiRequest *req, > + uint32_t start_lba, > + uint32_t data_type, > + uint32_t max_num_descr) > +{ > + uint8_t *outbuf = req->buf, *perf_desc; > + uint32_t resp_len = CD_PERF_HEADER_LEN + > + CD_PERF_TYPE_PERFORMANCE_DESCR_LEN; > + uint32_t perf_data_len = resp_len - 4; /* not incl. Perf Data Length */ > + uint32_t perf_kb = 10000; > + uint32_t end_lba = dev->num_blocks - 1; > + uint32_t except, write, tolerance; > + > + except = data_type & 0x03; > + if (except != CD_PERF_TYPE_PERFORMANCE_REPORT_ALL) { > + start_lba = 0; > + } > + write = (data_type >> 2) & 0x01; > + tolerance = (data_type >> 3) & 0x03; > + > + SPICE_DEBUG("get_performance, lun:%u" > + " performance type:0x00 data_type:0x%x" > + " except:0x%x write:0x%x tolerance:0x%x" > + " max_num:%u", > + req->lun, data_type, except, write, > + tolerance, max_num_descr); > + > + if (write) { > + SPICE_DEBUG("get_performance, lun:%u" > + " performance type:0x00 data_type:0x%x - write unsupported", > + req->lun, data_type); > + cd_scsi_get_performance_resp_empty(dev, req, CD_PERF_TYPE_PERFORMANCE, > + data_type, max_num_descr); > + return; > + } > + > + memset(outbuf, 0, resp_len); > + > + outbuf[0] = (perf_data_len >> 24) & 0xff; > + outbuf[1] = (perf_data_len >> 16) & 0xff; > + outbuf[2] = (perf_data_len >> 8) & 0xff; > + outbuf[3] = perf_data_len & 0xff; > + > + perf_desc = outbuf + CD_PERF_HEADER_LEN; > + > + perf_desc[0] = (start_lba >> 24) & 0xff; > + perf_desc[1] = (start_lba >> 16) & 0xff; > + perf_desc[2] = (start_lba >> 8) & 0xff; > + perf_desc[3] = start_lba & 0xff; > + > + perf_desc[4] = (perf_kb >> 24) & 0xff; > + perf_desc[5] = (perf_kb >> 16) & 0xff; > + perf_desc[6] = (perf_kb >> 8) & 0xff; > + perf_desc[7] = perf_kb & 0xff; > + > + perf_desc[8] = (end_lba >> 24) & 0xff; > + perf_desc[9] = (end_lba >> 16) & 0xff; > + perf_desc[10] = (end_lba >> 8) & 0xff; > + perf_desc[11] = end_lba & 0xff; > + > + perf_desc[12] = (perf_kb >> 24) & 0xff; > + perf_desc[13] = (perf_kb >> 16) & 0xff; > + perf_desc[14] = (perf_kb >> 8) & 0xff; > + perf_desc[15] = perf_kb & 0xff; > + > + req->req_len = CD_PERF_HEADER_LEN + > + (max_num_descr * CD_PERF_TYPE_PERFORMANCE_DESCR_LEN); > + > + req->in_len = MIN(req->req_len, resp_len); > + > + cd_scsi_cmd_complete_good(dev, req); > +} > + > +static void cd_scsi_cmd_get_performance(CdScsiLU *dev, CdScsiRequest *req) > +{ > + uint32_t data_type, max_num_descr, start_lba, type; > + > + req->xfer_dir = SCSI_XFER_FROM_DEV; > + > + data_type = req->cdb[1] & 0x0f; > + start_lba = (req->cdb[2] << 24) | > + (req->cdb[3] << 16) | > + (req->cdb[4] << 8) | > + req->cdb[5]; > + max_num_descr = (req->cdb[8] << 8) | req->cdb[9]; > + type = req->cdb[10]; > + > + switch (type) { > + case CD_PERF_TYPE_PERFORMANCE: > + cd_scsi_get_performance_resp_performance(dev, req, start_lba, > + data_type, max_num_descr); > + break; > + case CD_PERF_TYPE_UNUSABLE_AREA: /* not writable */ > + case CD_PERF_TYPE_DEFECT_STATUS: /* not restricted overwrite media */ > + case CD_PERF_TYPE_WRITE_SPEED: /* unsupported, irrelevant */ > + default: > + SPICE_DEBUG("get_performance, lun:%u" > + " unsupported type:0x%x" > + " data_type:0x%x max_num:%u", > + req->lun, type, data_type, max_num_descr); > + cd_scsi_cmd_complete_check_cond(dev, req, &sense_code_INVALID_CDB_FIELD); > + return; > + } > +} > + > +#define CD_MECHANISM_STATUS_HDR_LEN 8 > + > +/* byte 0 */ > +#define CD_CHANGER_FAULT_FLAG 0x80 > + > +#define CD_CHANGER_READY 0x00 > +#define CD_CHANGER_LOADING 0x01 > +#define CD_CHANGER_UNLOADING 0x02 > +#define CD_CHANGER_INITIALIZING 0x03 > + > +/* byte 1 */ > +#define CD_CHANGER_DOOR_OPEN_FLAG 0x10 > + > +#define CD_MECHANISM_STATE_IDLE 0x00 > +#define CD_MECHANISM_STATE_PLAYING 0x01 > +#define CD_MECHANISM_STATE_SCANNING 0x02 > +/* ACTIVE: with Initiator, Composite or Other ports > + (i.e. READ, PLAY CD, SCAN during PLAY CD) */ > +#define CD_MECHANISM_STATE_ACTIVE 0x03 > +#define CD_MECHANISM_STATE_NO_INFO 0x07 > + > +/* slots */ > +#define CD_MECHANISM_STATUS_SLOT_LEN 4 > + > +#define CD_MECHANISM_SLOT_DISK_CHANGED 0x01 > +#define CD_MECHANISM_SLOT_DISK_PRESENT 0x80 > +#define CD_MECHANISM_SLOT_DISK_CWP_V 0x02 > +#define CD_MECHANISM_SLOT_DISK_CWP 0x01 > + > +static void cd_scsi_cmd_mechanism_status(CdScsiLU *dev, CdScsiRequest *req) > +{ > + uint8_t *outbuf = req->buf; > + uint32_t resp_len = CD_MECHANISM_STATUS_HDR_LEN; > + > + req->xfer_dir = SCSI_XFER_FROM_DEV; > + > + req->req_len = (req->cdb[8] << 8) | req->cdb[9]; > + memset(outbuf, 0, req->req_len); > + > + /* For non-changer devices the curent slot number field is reserved, set only status */ > + outbuf[0] |= (CD_CHANGER_READY << 5); > + > + outbuf[1] |= (!dev->loaded) ? CD_CHANGER_DOOR_OPEN_FLAG : 0; > + outbuf[1] |= (dev->power_cond == CD_POWER_STATUS_ACTIVE) ? > + (CD_MECHANISM_STATE_ACTIVE << 5) : (CD_MECHANISM_STATE_IDLE << 5); > + > + /* For non-changer devices the number of slot tables returned shall be zero, so we leave > + both 'Number of Slots Available' and 'Length of Slot Table' fields as zeros */ > + > + req->in_len = MIN(req->req_len, resp_len); > + > + SPICE_DEBUG("mechanism_status, lun:%u", req->lun); > + > + cd_scsi_cmd_complete_good(dev, req); > +} > + > +static void cd_scsi_read_async_complete(GObject *src_object, > + GAsyncResult *result, > + gpointer user_data) > +{ > + GFileInputStream *stream = G_FILE_INPUT_STREAM(src_object); > + CdScsiRequest *req = (CdScsiRequest *)user_data; > + CdScsiTarget *st = (CdScsiTarget *)req->priv_data; > + CdScsiLU *dev = &st->units[req->lun]; > + GError *error = NULL; > + gsize bytes_read; > + gboolean finished; > + > + req->req_state = SCSI_REQ_COMPLETE; > + req->cancel_id = 0; > + > +// g_assert(stream == dev->stream); > + if (stream != dev->stream) { > + uint32_t opcode = (uint32_t)req->cdb[0]; > + SPICE_DEBUG("read_async_complete BAD STREAM, lun: %u" > + " req: %" G_GUINT64_FORMAT " op: 0x%02x", > + req->lun, req->req_len, opcode); > + cd_scsi_cmd_complete_check_cond(dev, req, &sense_code_TARGET_FAILURE); > + cd_scsi_dev_request_complete(st->user_data, req); > + return; > + } > + > + bytes_read = g_input_stream_read_finish(G_INPUT_STREAM(stream), result, &error); > + finished = bytes_read > 0; > + if (finished) { > + SPICE_DEBUG("read_async_complete, lun: %u" > + " finished: %d bytes_read: %" G_GUINT64_FORMAT > + " req: %" G_GUINT64_FORMAT, > + req->lun, finished, (uint64_t)bytes_read, req->req_len); > + > + req->in_len = MIN(bytes_read, req->req_len); > + req->status = GOOD; > + } else { > + if (error != NULL) { > + SPICE_ERROR("g_input_stream_read_finish failed: %s", error->message); > + g_clear_error (&error); > + } else { > + SPICE_ERROR("g_input_stream_read_finish failed (no err provided)"); > + } > + req->in_len = 0; > + req->status = GOOD; > + } > + cd_scsi_dev_request_complete(st->user_data, req); > +} > + > +static void cd_scsi_read_async_canceled(GCancellable *cancellable, gpointer user_data) > +{ > + CdScsiRequest *req = (CdScsiRequest *)user_data; > + CdScsiTarget *st = (CdScsiTarget *)req->priv_data; > + > + g_assert(cancellable == st->cancellable); > + g_cancellable_disconnect(cancellable, req->cancel_id); > + req->cancel_id = 0; > + > + req->req_state = > + (st->state == CD_SCSI_TGT_STATE_RUNNING) ? SCSI_REQ_CANCELED : SCSI_REQ_DISPOSED; > + req->in_len = 0; > + req->status = GOOD; > + > + cd_scsi_dev_request_complete(st->user_data, req); > +} > + > +static int cd_scsi_read_async_start(CdScsiLU *dev, CdScsiRequest *req) > +{ > + CdScsiTarget *st = dev->tgt; > + GFileInputStream *stream = dev->stream; > + > + SPICE_DEBUG("read_async_start, lun:%u" > + " lba: %" G_GUINT64_FORMAT " offset: %" G_GUINT64_FORMAT > + " cnt: %" G_GUINT64_FORMAT " len: %" G_GUINT64_FORMAT, > + req->lun, req->lba, req->offset, req->count, req->req_len); > + > + req->cancel_id = g_cancellable_connect(st->cancellable, > + G_CALLBACK(cd_scsi_read_async_canceled), > + req, /* data */ > + NULL); /* data destroy cb */ > + if (req->cancel_id == 0) { > + /* already canceled */ > + return -1; > + } > + > + g_seekable_seek(G_SEEKABLE(stream), > + req->offset, > + G_SEEK_SET, > + NULL, /* cancellable */ > + NULL); /* error */ > + > + g_input_stream_read_async(G_INPUT_STREAM(stream), > + req->buf, /* buffer to fill */ > + req->req_len, > + G_PRIORITY_DEFAULT, > + st->cancellable, > + cd_scsi_read_async_complete, > + (gpointer)req); /* callback argument */ > + return 0; > +} > + > +static void cd_scsi_cmd_read(CdScsiLU *dev, CdScsiRequest *req) > +{ > + if (dev->power_cond == CD_SCSI_POWER_STOPPED) { > + SPICE_DEBUG("read, lun: %u is stopped", req->lun); > + cd_scsi_cmd_complete_check_cond(dev, req, &sense_code_INIT_CMD_REQUIRED); > + return; > + } else if (!dev->loaded || dev->stream == NULL) { > + SPICE_DEBUG("read, lun: %u is not loaded", req->lun); > + cd_scsi_cmd_complete_check_cond(dev, req, &sense_code_NOT_READY_NO_MEDIUM); > + return; > + } > + > + req->cdb_len = scsi_cdb_length(req->cdb); > + > + req->lba = scsi_cdb_lba(req->cdb, req->cdb_len); > + req->offset = req->lba * dev->block_size; > + > + req->count = scsi_cdb_xfer_length(req->cdb, req->cdb_len); /* xfer in blocks */ > + req->req_len = (uint64_t) req->count * dev->block_size; > + > + cd_scsi_read_async_start(dev, req); > +} > + > +void cd_scsi_dev_request_submit(CdScsiTarget *st, CdScsiRequest *req) > +{ > + uint32_t lun = req->lun; > + uint32_t opcode = (uint32_t)req->cdb[0]; > + const char *cmd_name = scsi_cmd_name[opcode]; > + CdScsiLU *dev = &st->units[lun]; > + > + SPICE_DEBUG("request_submit, lun: %u op: 0x%02x %s", lun, opcode, cmd_name); > + > + if (st->cur_req != NULL) { > + SPICE_ERROR("request_submit, request not idle"); > + cd_scsi_cmd_complete_check_cond(dev, req, &sense_code_TARGET_FAILURE); > + goto done; > + } > + if (req->req_state != SCSI_REQ_IDLE) { > + SPICE_ERROR("request_submit, prev request outstanding"); > + cd_scsi_cmd_complete_check_cond(dev, req, &sense_code_TARGET_FAILURE); > + goto done; > + } > + req->req_state = SCSI_REQ_RUNNING; > + st->cur_req = req; > + > + /* INQUIRY should send response even for non-existing LUNs */ > + if (!cd_scsi_target_lun_legal(st, lun)) { > + SPICE_ERROR("request_submit, illegal lun:%u", lun); > + if (opcode == INQUIRY) { > + if (req->cdb[1] & 0x1) { > + cd_scsi_cmd_inquiry_vpd_no_lun(dev, req, PERIF_QUALIFIER_UNSUPPORTED); > + } else { > + cd_scsi_cmd_inquiry_standard_no_lun(dev, req, PERIF_QUALIFIER_UNSUPPORTED); > + } > + } else { > + cd_scsi_cmd_complete_check_cond(dev, req, &sense_code_LUN_NOT_SUPPORTED); > + } > + goto done; > + } > + if (!cd_scsi_target_lun_realized(st, lun)) { > + SPICE_ERROR("request_submit, absent lun:%u", lun); > + if (opcode == INQUIRY) { > + if (req->cdb[1] & 0x1) { > + cd_scsi_cmd_inquiry_vpd_no_lun(dev, req, PERIF_QUALIFIER_NOT_CONNECTED); > + } else { > + cd_scsi_cmd_inquiry_standard_no_lun(dev, req, PERIF_QUALIFIER_NOT_CONNECTED); > + } > + } else { > + cd_scsi_cmd_complete_check_cond(dev, req, &sense_code_LUN_NOT_SUPPORTED); > + } > + goto done; > + } > + > + if (dev->short_sense.key != NO_SENSE) { > + gboolean pending_sense = TRUE; > + if (dev->short_sense.key == UNIT_ATTENTION) { > + if (cd_scsi_opcode_ua_supress(opcode)) { > + pending_sense = FALSE; /* UA supressed */ > + } > + } else if (opcode == REQUEST_SENSE) { > + pending_sense = FALSE; /* sense returned as data */ > + } > + if (pending_sense) { > + cd_scsi_cmd_complete_check_cond(dev, req, NULL); /* sense already set */ > + goto done; > + } > + } > + > + /* save the target to be used in callbacks where only req is passed */ > + req->priv_data = (void *)st; > + > + req->req_len = 0; > + > + switch (opcode) { > + case REPORT_LUNS: > + cd_scsi_cmd_report_luns(st, dev, req); > + break; > + case TEST_UNIT_READY: > + cd_scsi_cmd_test_unit_ready(dev, req); > + break; > + case INQUIRY: > + cd_scsi_cmd_inquiry(dev, req); > + break; > + case REQUEST_SENSE: > + cd_scsi_cmd_request_sense(dev, req); > + break; > + case READ_6: > + case READ_10: > + case READ_12: > + case READ_16: > + cd_scsi_cmd_read(dev, req); > + break; > + case READ_CAPACITY_10: > + cd_scsi_cmd_read_capacity(dev, req); > + break; > + case READ_TOC: > + cd_scsi_cmd_read_toc(dev, req); > + break; > + case GET_EVENT_STATUS_NOTIFICATION: > + cd_scsi_cmd_get_event_status_notification(dev, req); > + break; > + case READ_DISC_INFORMATION: > + cd_scsi_cmd_get_read_disc_information(dev, req); > + break; > + case READ_TRACK_INFORMATION: > + cd_scsi_cmd_get_read_track_information(dev, req); > + break; > + case MODE_SENSE_10: > + cd_scsi_cmd_mode_sense_10(dev, req); > + break; > + case MODE_SELECT: > + cd_scsi_cmd_mode_select_6(dev, req); > + break; > + case MODE_SELECT_10: > + cd_scsi_cmd_mode_select_10(dev, req); > + break; > + case GET_CONFIGURATION: > + cd_scsi_cmd_get_configuration(dev, req); > + break; > + case ALLOW_MEDIUM_REMOVAL: > + cd_scsi_cmd_allow_medium_removal(dev, req); > + break; > + case MMC_SEND_EVENT: > + cd_scsi_cmd_send_event(dev, req); > + break; > + case MMC_REPORT_KEY: > + cd_scsi_cmd_report_key(dev, req); > + break; > + case MMC_SEND_KEY: > + cd_scsi_cmd_send_key(dev, req); > + break; > + case START_STOP: > + cd_scsi_cmd_start_stop_unit(dev, req); > + break; > + case MMC_GET_PERFORMANCE: > + cd_scsi_cmd_get_performance(dev, req); > + break; > + case MMC_MECHANISM_STATUS: > + cd_scsi_cmd_mechanism_status(dev, req); > + break; > + default: > + cd_scsi_cmd_complete_check_cond(dev, req, &sense_code_INVALID_OPCODE); > + break; > + } > + > + if (req->req_len > INT32_MAX) { > + cd_scsi_cmd_complete_check_cond(dev, req, &sense_code_INVALID_CDB_FIELD); > + goto done; > + } > + > +done: > + SPICE_DEBUG("request_submit done, lun: %u" > + " op: 0x%02x %s, state: %s status: %u len: %" G_GUINT64_FORMAT, > + lun, opcode, cmd_name, CdScsiReqState_str(req->req_state), req->status, > + req->in_len); > + > + if (req->req_state == SCSI_REQ_COMPLETE) { > + cd_scsi_dev_request_complete(st->user_data, req); > + } > +} > + > +void cd_scsi_dev_request_cancel(CdScsiTarget *st, CdScsiRequest *req) > +{ > + if (st->cur_req == req) { > + if (req->req_state == SCSI_REQ_RUNNING) { > + SPICE_DEBUG("request_cancel: lun: %u" > + " op: 0x%02x len: %" G_GUINT64_FORMAT, > + req->lun, (unsigned int)req->cdb[0], req->req_len); > + g_cancellable_cancel(st->cancellable); > + } else { > + SPICE_DEBUG("request_cancel: request is not running"); > + } > + } else { > + SPICE_DEBUG("request_cancel: other request is outstanding"); > + } > +} > + > +void cd_scsi_dev_request_release(CdScsiTarget *st, CdScsiRequest *req) > +{ > + st->cur_req = NULL; > + cd_scsi_req_init(req); > + > + if (st->state == CD_SCSI_TGT_STATE_RESET) { > + cd_scsi_target_do_reset(st); > + } > +} > + > +#endif /* USE_USBREDIR */ > diff --git a/src/cd-scsi.h b/src/cd-scsi.h > new file mode 100644 > index 00000000..c058a66e > --- /dev/null > +++ b/src/cd-scsi.h > @@ -0,0 +1,120 @@ > +/* -*- Mode: C; c-basic-offset: 4; indent-tabs-mode: nil -*- */ > +/* > + CD device emulation - SCSI engine > + > + Copyright (C) 2018 Red Hat, Inc. > + > + Red Hat Authors: > + Alexander Nezhinsky<anezhins@xxxxxxxxxx> > + > + This library is free software; you can redistribute it and/or > + modify it under the terms of the GNU Lesser General Public > + License as published by the Free Software Foundation; either > + version 2.1 of the License, or (at your option) any later version. > + > + This library 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 > + Lesser General Public License for more details. > + > + You should have received a copy of the GNU Lesser General Public > + License along with this library; if not, see <http://www.gnu.org/licenses/>. > +*/ > + > +#ifndef SPICE_GTK_CD_SCSI_H__ > +#define SPICE_GTK_CD_SCSI_H__ > + > +#include "cd-scsi-dev-params.h" > +#include "cd-usb-bulk-msd.h" > +#include "scsi-constants.h" > + > +#if defined(G_OS_WIN32) > +#include <winsock2.h> > +#include <windows.h> > +/* Windows is always LE at the moment */ > +#define le32toh(x) (x) > +#define htole32(x) (x) > +#define htobe32(x) htonl(x) > +#endif > + > +typedef enum ScsiXferDir { > + SCSI_XFER_NONE = 0, /* TEST_UNIT_READY, ... */ > + SCSI_XFER_FROM_DEV, /* READ, INQUIRY, MODE_SENSE, ... */ > + SCSI_XFER_TO_DEV, /* WRITE, MODE_SELECT, ... */ > +} ScsiXferDir; > + > +#define SCSI_CDB_BUF_SIZE 16 > + > +typedef enum CdScsiReqState { > + SCSI_REQ_IDLE = 0, > + SCSI_REQ_RUNNING, > + SCSI_REQ_COMPLETE, > + SCSI_REQ_CANCELED, > + SCSI_REQ_DISPOSED, > +} CdScsiReqState; > + > +typedef struct CdScsiRequest { > + /* request */ > + uint8_t cdb[SCSI_CDB_BUF_SIZE]; > + uint32_t cdb_len; > + > + uint32_t lun; > + > + uint8_t *buf; > + uint32_t buf_len; > + > + /* internal */ > + CdScsiReqState req_state; > + ScsiXferDir xfer_dir; > + uint64_t cancel_id; > + void *priv_data; > + > + uint64_t lba; /* offset in logical blocks if relevant */ > + uint64_t count; /* count in logical blocks */ > + > + uint64_t offset; /* scsi cdb offset, normalized to bytes */ > + uint64_t req_len; /* scsi cdb request length, normalized to bytes */ > + > + /* result */ > + uint64_t in_len; /* length of data actually available after read */ > + uint32_t status; /* SCSI status code */ > + > +} CdScsiRequest; > + > +CdScsiReqState cd_scsi_get_req_state(CdScsiRequest *req); > + > +/* SCSI target/device API */ > +typedef struct CdScsiTarget CdScsiTarget; > + > +/* to be used in callbacks */ > +CdScsiTarget *cd_scsi_target_alloc(void *target_user_data, uint32_t max_luns); > +void cd_scsi_target_free(CdScsiTarget *scsi_target); > + > +int cd_scsi_dev_realize(CdScsiTarget *scsi_target, uint32_t lun, > + const CdScsiDeviceParameters *dev_params); > +int cd_scsi_dev_unrealize(CdScsiTarget *scsi_target, uint32_t lun); > + > +int cd_scsi_dev_lock(CdScsiTarget *scsi_target, uint32_t lun, gboolean lock); > +int cd_scsi_dev_load(CdScsiTarget *scsi_target, uint32_t lun, > + const CdScsiMediaParameters *media_params); > +int cd_scsi_dev_get_info(CdScsiTarget *scsi_target, uint32_t lun, CdScsiDeviceInfo *lun_info); > +int cd_scsi_dev_unload(CdScsiTarget *scsi_target, uint32_t lun); > + > +void cd_scsi_dev_request_submit(CdScsiTarget *scsi_target, CdScsiRequest *request); > +void cd_scsi_dev_request_cancel(CdScsiTarget *scsi_target, CdScsiRequest *request); > +void cd_scsi_dev_request_release(CdScsiTarget *scsi_target, CdScsiRequest *request); > + > +int cd_scsi_dev_reset(CdScsiTarget *scsi_target, uint32_t lun); > + > +int cd_scsi_target_reset(CdScsiTarget *scsi_target); > + > +/* Callbacks > + * These callbacks are used by upper layer to implement specific SCSI > + * target devices. > + */ > +void cd_scsi_dev_request_complete(void *target_user_data, CdScsiRequest *request); > +void cd_scsi_dev_changed(void *target_user_data, uint32_t lun); > +void cd_scsi_dev_reset_complete(void *target_user_data, uint32_t lun); > +void cd_scsi_target_reset_complete(void *target_user_data); > + > +#endif /* SPICE_GTK_CD_SCSI_H__ */ > diff --git a/src/cd-usb-bulk-msd.c b/src/cd-usb-bulk-msd.c > new file mode 100644 > index 00000000..49e01eb6 > --- /dev/null > +++ b/src/cd-usb-bulk-msd.c > @@ -0,0 +1,544 @@ > +/* -*- Mode: C; c-basic-offset: 4; indent-tabs-mode: nil -*- */ > +/* > + USB CD Device emulation - Data Bulk transfers - Mass Storage Device > + > + Copyright (C) 2018 Red Hat, Inc. > + > + Red Hat Authors: > + Alexander Nezhinsky<anezhins@xxxxxxxxxx> > + > + This library is free software; you can redistribute it and/or > + modify it under the terms of the GNU Lesser General Public > + License as published by the Free Software Foundation; either > + version 2.1 of the License, or (at your option) any later version. > + > + This library 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 > + Lesser General Public License for more details. > + > + You should have received a copy of the GNU Lesser General Public > + License along with this library; if not, see <http://www.gnu.org/licenses/>. > +*/ > + > +#include "config.h" > +#include "spice/types.h" > +#include "spice-common.h" > +#include "spice-util.h" > +#include "cd-usb-bulk-msd.h" > +#include "cd-scsi.h" > + > +#ifdef USE_USBREDIR > + > +#define SPICE_ERROR(fmt, ...) \ > + do { SPICE_DEBUG("usb-msd error: " fmt , ## __VA_ARGS__); } while (0) > + > +typedef enum UsbCdState { > + USB_CD_STATE_INIT, /* Not ready */ > + USB_CD_STATE_CBW, /* Waiting for Command Block */ > + USB_CD_STATE_DATAOUT, /* Transfer data to device */ > + USB_CD_STATE_DATAIN, /* Transfer data from device */ > + USB_CD_STATE_ZERO_DATAIN, /* Need to send zero bulk-in before status */ > + USB_CD_STATE_CSW, /* Send Command Status */ > + USB_CD_STATE_DEVICE_RESET, /* reset of a single device */ > + USB_CD_STATE_TARGET_RESET /* reset of entire target */ > +} UsbCdState; > + > +/* USB MSD Command Block Wrapper */ > +struct __attribute__((__packed__)) UsbCdCBW { > + uint32_t sig; > + uint32_t tag; > + uint32_t exp_data_len; /* expected data xfer length for the request */ > + uint8_t flags; > + uint8_t lun; > + uint8_t cmd_len; /* actual length of the scsi command that follows */ > + uint8_t cmd[16]; /* scsi command to perform */ > +}; > + > +/* USB MSD Command Status Wrapper */ > +struct __attribute__((__packed__)) UsbCdCSW { > + uint32_t sig; > + uint32_t tag; > + uint32_t residue; > + uint8_t status; > +}; > + > +/* UsbCdCSW::status */ > +typedef enum UsbMsdStatus { > + USB_MSD_STATUS_GOOD = 0, > + USB_MSD_STATUS_FAILED = 1, > + USB_MSD_STATUS_PHASE_ERR = 2, > +} UsbMsdStatus; > + > +typedef struct UsbCdBulkMsdRequest { > + CdScsiRequest scsi_req; > + > + uint32_t lun; > + uint32_t usb_req_len; /* length of data requested by usb */ > + uint32_t scsi_in_len; /* length of data returned by scsi limited by usb request */ > + > + uint32_t xfer_len; /* length of data transfered until now */ > + uint32_t bulk_in_len; /* length of the last postponed bulk-in request */ > + > + struct UsbCdCSW csw; /* usb status header */ > +} UsbCdBulkMsdRequest; > + > +typedef struct UsbCdBulkMsdDevice { > + UsbCdState state; > + CdScsiTarget *scsi_target; /* scsi handle */ > + void *usb_user_data; /* used in callbacks to usb */ > + UsbCdBulkMsdRequest usb_req; /* now supporting a single cmd */ > + uint8_t *data_buf; > + uint32_t data_buf_len; > +} UsbCdBulkMsdDevice; > + > +static inline const char *usb_cd_state_str(UsbCdState state) > +{ > + switch (state) { > + case USB_CD_STATE_INIT: > + return "INIT"; > + case USB_CD_STATE_CBW: > + return "CBW"; > + case USB_CD_STATE_DATAOUT: > + return "DATAOUT"; > + case USB_CD_STATE_DATAIN: > + return "DATAIN"; > + case USB_CD_STATE_ZERO_DATAIN: > + return "ZERO_DATAIN"; > + case USB_CD_STATE_CSW: > + return "CSW"; > + case USB_CD_STATE_DEVICE_RESET: > + return "DEV_RESET"; > + case USB_CD_STATE_TARGET_RESET: > + return "TGT_RESET"; > + default: > + return "ILLEGAL"; > + } > +} > + > +static void cd_usb_bulk_msd_set_state(UsbCdBulkMsdDevice *cd, UsbCdState state) > +{ > + SPICE_DEBUG("State %s -> %s", usb_cd_state_str(cd->state), usb_cd_state_str(state)); > + cd->state = state; > +} > + > +UsbCdBulkMsdDevice *cd_usb_bulk_msd_alloc(void *usb_user_data, uint32_t max_luns) > +{ > + UsbCdBulkMsdDevice *cd = g_new0(UsbCdBulkMsdDevice, 1); > + > + cd->data_buf_len = 256 * 1024; > + cd->data_buf = g_malloc(cd->data_buf_len); > + > + cd->scsi_target = cd_scsi_target_alloc(cd, max_luns); > + if (cd->scsi_target == NULL) { > + g_free(cd->data_buf); > + g_free(cd); > + return NULL; > + } > + cd_usb_bulk_msd_set_state(cd, USB_CD_STATE_INIT); > + cd->usb_user_data = usb_user_data; > + > + SPICE_DEBUG("Alloc, max_luns:%" G_GUINT32_FORMAT, max_luns); > + return cd; > +} > + > +int cd_usb_bulk_msd_realize(UsbCdBulkMsdDevice *cd, uint32_t lun, > + const CdScsiDeviceParameters *dev_params) > +{ > + CdScsiDeviceParameters scsi_dev_params; > + int rc; > + > + scsi_dev_params.vendor = dev_params->vendor ? : "SPICE"; > + scsi_dev_params.product = dev_params->product ? : "USB-CD"; > + scsi_dev_params.version = dev_params->version ? : "0.1"; > + scsi_dev_params.serial = dev_params->serial ? : "123456"; > + > + rc = cd_scsi_dev_realize(cd->scsi_target, lun, &scsi_dev_params); > + if (rc != 0) { > + SPICE_ERROR("Failed to realize lun:%" G_GUINT32_FORMAT, lun); > + return rc; > + } > + > + if (cd->state == USB_CD_STATE_INIT) { > + /* wait next request */ > + cd_usb_bulk_msd_set_state(cd, USB_CD_STATE_CBW); > + cd_scsi_dev_request_release(cd->scsi_target, &cd->usb_req.scsi_req); > + } > + > + SPICE_DEBUG("Realize OK lun:%" G_GUINT32_FORMAT, lun); > + return 0; > +} > + > +int cd_usb_bulk_msd_lock(UsbCdBulkMsdDevice *cd, uint32_t lun, gboolean lock) > +{ > + int rc; > + > + rc = cd_scsi_dev_lock(cd->scsi_target, lun, lock); > + if (rc != 0) { > + SPICE_ERROR("Failed to lock lun:%" G_GUINT32_FORMAT, lun); > + return rc; > + } > + > + SPICE_DEBUG("Lock OK lun:%" G_GUINT32_FORMAT, lun); > + return 0; > +} > + > +int cd_usb_bulk_msd_load(UsbCdBulkMsdDevice *cd, uint32_t lun, > + const CdScsiMediaParameters *media_params) > +{ > + int rc; > + > + rc = cd_scsi_dev_load(cd->scsi_target, lun, media_params); > + if (rc != 0) { > + SPICE_ERROR("Failed to load lun:%" G_GUINT32_FORMAT, lun); > + return rc; > + } > + > + SPICE_DEBUG("Load OK lun:%" G_GUINT32_FORMAT, lun); > + return 0; > +} > + > +int cd_usb_bulk_msd_get_info(UsbCdBulkMsdDevice *cd, uint32_t lun, CdScsiDeviceInfo *lun_info) > +{ > + int rc; > + > + rc = cd_scsi_dev_get_info(cd->scsi_target, lun, lun_info); > + if (rc != 0) { > + SPICE_ERROR("Failed to get info lun:%" G_GUINT32_FORMAT, lun); > + return rc; > + } > + > + return 0; > +} > + > +int cd_usb_bulk_msd_unload(UsbCdBulkMsdDevice *cd, uint32_t lun) > +{ > + int rc; > + > + rc = cd_scsi_dev_unload(cd->scsi_target, lun); > + if (rc != 0) { > + SPICE_ERROR("Failed to unload lun:%" G_GUINT32_FORMAT, lun); > + return rc; > + } > + > + SPICE_DEBUG("Unload OK lun:%" G_GUINT32_FORMAT, lun); > + return 0; > +} > + > +int cd_usb_bulk_msd_unrealize(UsbCdBulkMsdDevice *cd, uint32_t lun) > +{ > + int rc; > + > + rc = cd_scsi_dev_unrealize(cd->scsi_target, lun); > + if (rc != 0) { > + SPICE_ERROR("Unrealize lun:%" G_GUINT32_FORMAT, lun); > + return rc; > + } > + > + SPICE_DEBUG("Unrealize lun:%" G_GUINT32_FORMAT, lun); > + return 0; > +} > + > +void cd_usb_bulk_msd_free(UsbCdBulkMsdDevice *cd) > +{ > + cd_scsi_target_free(cd->scsi_target); > + g_free(cd->data_buf); > + g_free(cd); > + > + SPICE_DEBUG("Free"); > +} > + > +int cd_usb_bulk_msd_reset(UsbCdBulkMsdDevice *cd) > +{ > + cd_scsi_target_reset(cd->scsi_target); > + cd_usb_bulk_msd_set_state(cd, USB_CD_STATE_CBW); > + > + SPICE_DEBUG("Reset"); > + return 0; > +} > + > +static int parse_usb_msd_cmd(UsbCdBulkMsdDevice *cd, uint8_t *buf, uint32_t cbw_len) > +{ > + struct UsbCdCBW *cbw = (struct UsbCdCBW *)buf; > + UsbCdBulkMsdRequest *usb_req = &cd->usb_req; > + CdScsiRequest *scsi_req = &usb_req->scsi_req; > + > + if (cbw_len != sizeof(*cbw)) { > + SPICE_ERROR("CMD: Bad CBW size:%" G_GUINT32_FORMAT, cbw_len); > + return -1; > + } > + if (le32toh(cbw->sig) != 0x43425355) { /* MSD command signature */ > + SPICE_ERROR("CMD: Bad CBW signature:%08x", le32toh(cbw->sig)); > + return -1; > + } > + const uint8_t cmd_len = cbw->cmd_len & 0x1F; > + if (cmd_len < 1 || cmd_len > 16) { > + SPICE_ERROR("CMD: Bad CBW command len:%08x", cmd_len); > + return -1; > + } > + > + usb_req->lun = cbw->lun; > + usb_req->usb_req_len = le32toh(cbw->exp_data_len); > + > + usb_req->scsi_in_len = 0; /* no data from scsi yet */ > + usb_req->xfer_len = 0; /* no bulks transfered yet */ > + usb_req->bulk_in_len = 0; /* no bulk-in requests yet */ > + > + if (usb_req->usb_req_len == 0) { > + cd_usb_bulk_msd_set_state(cd, USB_CD_STATE_CSW); /* no data - return status */ > + scsi_req->buf = NULL; > + scsi_req->buf_len = 0; > + } else if (cbw->flags & 0x80) { > + cd_usb_bulk_msd_set_state(cd, USB_CD_STATE_DATAIN); /* read command */ > + scsi_req->buf = cd->data_buf; > + scsi_req->buf_len = cd->data_buf_len; > + } else { > + cd_usb_bulk_msd_set_state(cd, USB_CD_STATE_DATAOUT); /* write command */ > + scsi_req->buf = NULL; > + scsi_req->buf_len = 0; > + } > + > + scsi_req->cdb_len = cmd_len; > + g_assert(scsi_req->cdb_len <= sizeof(scsi_req->cdb)); > + memcpy(scsi_req->cdb, cbw->cmd, scsi_req->cdb_len); > + > + scsi_req->lun = usb_req->lun; > + > + SPICE_DEBUG("CMD lun:%" G_GUINT32_FORMAT " tag:%#x flags:%08x " > + "cdb_len:%" G_GUINT32_FORMAT " req_len:%" G_GUINT32_FORMAT, > + usb_req->lun, le32toh(cbw->tag), cbw->flags, > + scsi_req->cdb_len, usb_req->usb_req_len); > + > + /* prepare status - CSW */ > + usb_req->csw.sig = htole32(0x53425355); > + usb_req->csw.tag = cbw->tag; > + usb_req->csw.residue = 0; > + usb_req->csw.status = (uint8_t)USB_MSD_STATUS_GOOD; > + > + return 0; > +} > + > +static void usb_cd_cmd_done(UsbCdBulkMsdDevice *cd) > +{ > + UsbCdBulkMsdRequest *usb_req = &cd->usb_req; > + CdScsiRequest *scsi_req = &usb_req->scsi_req; > + > + cd_usb_bulk_msd_set_state(cd, USB_CD_STATE_CBW); /* Command next */ > + cd_scsi_dev_request_release(cd->scsi_target, scsi_req); > +} > + > +static void usb_cd_send_status(UsbCdBulkMsdDevice *cd) > +{ > + UsbCdBulkMsdRequest *usb_req = &cd->usb_req; > + > + SPICE_DEBUG("Command CSW tag:0x%x msd_status:%d len:%" G_GUINT64_FORMAT, > + le32toh(usb_req->csw.tag), (int)usb_req->csw.status, sizeof(usb_req->csw)); > + > + usb_cd_cmd_done(cd); > + > + g_assert(usb_req->csw.sig == htole32(0x53425355)); > + cd_usb_bulk_msd_read_complete(cd->usb_user_data, > + (uint8_t *)&usb_req->csw, sizeof(usb_req->csw), > + BULK_STATUS_GOOD); > +} > + > +static void usb_cd_send_canceled(UsbCdBulkMsdDevice *cd) > +{ > + UsbCdBulkMsdRequest *usb_req = &cd->usb_req; > + > + SPICE_DEBUG("Canceled cmd tag:0x%x, len:%" G_GUINT64_FORMAT, > + le32toh(usb_req->csw.tag), sizeof(usb_req->csw)); > + > + usb_cd_cmd_done(cd); > + > + cd_usb_bulk_msd_read_complete(cd->usb_user_data, > + NULL, 0, > + BULK_STATUS_CANCELED); > +} > + > +static void usb_cd_send_data_in(UsbCdBulkMsdDevice *cd, uint32_t max_len) > +{ > + UsbCdBulkMsdRequest *usb_req = &cd->usb_req; > + CdScsiRequest *scsi_req = &usb_req->scsi_req; > + uint8_t *buf = ((uint8_t *)scsi_req->buf) + usb_req->xfer_len; > + uint32_t avail_len = usb_req->scsi_in_len - usb_req->xfer_len; > + uint32_t send_len = MIN(avail_len, max_len); > + > + SPICE_DEBUG("Data-in cmd tag 0x%x, remains %" G_GUINT32_FORMAT > + ", requested %" G_GUINT32_FORMAT ", send %" G_GUINT32_FORMAT, > + usb_req->csw.tag, avail_len, max_len, send_len); > + > + g_assert(max_len <= usb_req->usb_req_len); > + > + cd_usb_bulk_msd_read_complete(cd->usb_user_data, > + buf, send_len, > + BULK_STATUS_GOOD); > + > + if (scsi_req->status == GOOD) { > + usb_req->xfer_len += send_len; > + if (usb_req->xfer_len == usb_req->scsi_in_len) { > + /* all data for this bulk has been transferred */ > + if (usb_req->scsi_in_len == usb_req->usb_req_len || /* req fully satisfied */ > + send_len < max_len) { /* partial bulk - no more data */ > + cd_usb_bulk_msd_set_state(cd, USB_CD_STATE_CSW); > + } else { > + /* partial cmd data fullfilled entire vulk-in request */ > + cd_usb_bulk_msd_set_state(cd, USB_CD_STATE_ZERO_DATAIN); > + } > + } > + } else { /* cmd failed - no more data */ > + cd_usb_bulk_msd_set_state(cd, USB_CD_STATE_CSW); > + } > +} > + > +int cd_usb_bulk_msd_read(UsbCdBulkMsdDevice *cd, uint32_t max_len) > +{ > + UsbCdBulkMsdRequest *usb_req = &cd->usb_req; > + CdScsiRequest *scsi_req = &usb_req->scsi_req; > + > + SPICE_DEBUG("msd_read, state: %s, len %" G_GUINT32_FORMAT, > + usb_cd_state_str(cd->state), max_len); > + > + switch (cd->state) { > + case USB_CD_STATE_CSW: /* Command Status */ > + if (max_len < 13) { > + goto fail; > + } > + if (cd_scsi_get_req_state(scsi_req) == SCSI_REQ_COMPLETE) { > + usb_cd_send_status(cd); > + } else { > + usb_req->bulk_in_len += max_len; > + SPICE_DEBUG("msd_read CSW, req incomplete, added len %" G_GUINT32_FORMAT > + " saved len %" G_GUINT32_FORMAT, > + max_len, usb_req->bulk_in_len); > + } > + break; > + > + case USB_CD_STATE_DATAIN: /* Bulk Data-IN */ > + if (cd_scsi_get_req_state(scsi_req) == SCSI_REQ_COMPLETE) { > + usb_cd_send_data_in(cd, max_len); > + } else { > + usb_req->bulk_in_len += max_len; > + SPICE_DEBUG("msd_read DATAIN, req incomplete, added len %" G_GUINT32_FORMAT > + " saved len %" G_GUINT32_FORMAT, > + max_len, usb_req->bulk_in_len); > + } > + break; > + > + case USB_CD_STATE_ZERO_DATAIN: > + cd_usb_bulk_msd_read_complete(cd->usb_user_data, > + NULL, 0, > + BULK_STATUS_GOOD); > + cd_usb_bulk_msd_set_state(cd, USB_CD_STATE_CSW); /* Status next */ > + break; > + > + default: > + SPICE_ERROR("Unexpected read state: %s, len %" G_GUINT32_FORMAT, > + usb_cd_state_str(cd->state), max_len); > + goto fail; > + } > + return 0; > + > +fail: > + return -1; > +} > + > +void cd_scsi_dev_request_complete(void *target_user_data, CdScsiRequest *scsi_req) > +{ > + UsbCdBulkMsdDevice *cd = (UsbCdBulkMsdDevice *)target_user_data; > + UsbCdBulkMsdRequest *usb_req = &cd->usb_req; > + > + g_assert(scsi_req == &usb_req->scsi_req); > + > + if (scsi_req->req_state == SCSI_REQ_COMPLETE) { > + > + usb_req->scsi_in_len = (scsi_req->in_len <= usb_req->usb_req_len) ? > + scsi_req->in_len : usb_req->usb_req_len; > + > + /* prepare CSW */ > + if (usb_req->usb_req_len > usb_req->scsi_in_len) { > + usb_req->csw.residue = htole32(usb_req->usb_req_len - usb_req->scsi_in_len); > + } > + if (scsi_req->status != GOOD) { > + usb_req->csw.status = (uint8_t)USB_MSD_STATUS_FAILED; > + } > + > + if (usb_req->bulk_in_len) { > + /* bulk-in request arrived while scsi was still running */ > + if (cd->state == USB_CD_STATE_DATAIN) { > + usb_cd_send_data_in(cd, usb_req->bulk_in_len); > + } else if (cd->state == USB_CD_STATE_CSW) { > + usb_cd_send_status(cd); > + } > + usb_req->bulk_in_len = 0; > + } > + } else if (scsi_req->req_state == SCSI_REQ_CANCELED) { > + usb_cd_send_canceled(cd); > + } else { > + g_assert(scsi_req->req_state == SCSI_REQ_DISPOSED); > + SPICE_DEBUG("Disposed cmd tag:0x%x, len:%" G_GUINT64_FORMAT, > + le32toh(usb_req->csw.tag), sizeof(usb_req->csw)); > + usb_cd_cmd_done(cd); > + } > +} > + > +int cd_usb_bulk_msd_cancel_read(UsbCdBulkMsdDevice *cd) > +{ > + UsbCdBulkMsdRequest *usb_req = &cd->usb_req; > + CdScsiRequest *scsi_req = &usb_req->scsi_req; > + > + cd_scsi_dev_request_cancel(cd->scsi_target, scsi_req); > + return 0; > +} > + > +int cd_usb_bulk_msd_write(UsbCdBulkMsdDevice *cd, uint8_t *buf_out, uint32_t buf_out_len) > +{ > + switch (cd->state) { > + case USB_CD_STATE_CBW: /* Command Block */ > + parse_usb_msd_cmd(cd, buf_out, buf_out_len); > + if (cd->state == USB_CD_STATE_DATAIN || cd->state == USB_CD_STATE_CSW) { > + cd_scsi_dev_request_submit(cd->scsi_target, &cd->usb_req.scsi_req); > + } > + break; > + case USB_CD_STATE_DATAOUT: /* Data-Out for a Write cmd */ > + cd->usb_req.scsi_req.buf = buf_out; > + cd->usb_req.scsi_req.buf_len = buf_out_len; > + cd_scsi_dev_request_submit(cd->scsi_target, &cd->usb_req.scsi_req); > + cd_usb_bulk_msd_set_state(cd, USB_CD_STATE_CSW); /* Status next */ > + break; > + default: > + SPICE_DEBUG("Unexpected write state: %s, len %" G_GUINT32_FORMAT, > + usb_cd_state_str(cd->state), buf_out_len); > + goto fail; > + } > + return 0; > + > +fail: > + return -1; > +} > + > +void cd_scsi_dev_reset_complete(void *target_user_data, uint32_t lun) > +{ > + UsbCdBulkMsdDevice *cd = (UsbCdBulkMsdDevice *)target_user_data; > + > + if (cd->state == USB_CD_STATE_DEVICE_RESET) { > + cd_usb_bulk_msd_set_state(cd, USB_CD_STATE_CBW); > + cd_usb_bulk_msd_reset_complete(cd->usb_user_data, 0); > + } > +} > + > +void cd_scsi_target_reset_complete(void *target_user_data) > +{ > + UsbCdBulkMsdDevice *cd = (UsbCdBulkMsdDevice *)target_user_data; > + cd_usb_bulk_msd_set_state(cd, USB_CD_STATE_INIT); > +} > + > +void cd_scsi_dev_changed(void *target_user_data, uint32_t lun) > +{ > + UsbCdBulkMsdDevice *cd = (UsbCdBulkMsdDevice *)target_user_data; > + SPICE_DEBUG("Device changed, state: %s lun: %" G_GUINT32_FORMAT, > + usb_cd_state_str(cd->state), lun); > + cd_usb_bulk_msd_lun_changed(cd->usb_user_data, lun); > +} > + > +#endif /* USE_USBREDIR */ > diff --git a/src/cd-usb-bulk-msd.h b/src/cd-usb-bulk-msd.h > new file mode 100644 > index 00000000..492a49b8 > --- /dev/null > +++ b/src/cd-usb-bulk-msd.h > @@ -0,0 +1,134 @@ > +/* -*- Mode: C; c-basic-offset: 4; indent-tabs-mode: nil -*- */ > +/* > + Copyright (C) 2018 Red Hat, Inc. > + > + Red Hat Authors: > + Alexander Nezhinsky<anezhins@xxxxxxxxxx> > + > + This library is free software; you can redistribute it and/or > + modify it under the terms of the GNU Lesser General Public > + License as published by the Free Software Foundation; either > + version 2.1 of the License, or (at your option) any later version. > + > + This library 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 > + Lesser General Public License for more details. > + > + You should have received a copy of the GNU Lesser General Public > + License along with this library; if not, see <http://www.gnu.org/licenses/>. > +*/ > + > +#ifndef SPICE_GTK_CD_USB_BULK_MSD_H__ > +#define SPICE_GTK_CD_USB_BULK_MSD_H__ > + > +G_BEGIN_DECLS > + > +#include <gio/gio.h> > + > +#include "cd-scsi-dev-params.h" > + > +typedef enum CdUsbBulkStatus { > + BULK_STATUS_GOOD = 0, > + BULK_STATUS_ERROR, > + BULK_STATUS_CANCELED, > + BULK_STATUS_STALL, > +} CdUsbBulkStatus; > + > +typedef struct UsbCdBulkMsdDevice UsbCdBulkMsdDevice; > + > +/* USB backend callbacks */ > + > +/* called on completed read data bulk transfer > + * user_data - user_data in unit parameters structure > + * status - bulk status code > + */ > +void cd_usb_bulk_msd_read_complete(void *user_data, > + uint8_t *data, uint32_t length, > + CdUsbBulkStatus status); > + > +/* called when state of device's unit changed to signal GUI component > + * user_data - user_data in unit parameters structure > + */ > +void cd_usb_bulk_msd_lun_changed(void *user_data, uint32_t lun); > + > +/* called on completed device reset > + * user_data - user_data in unit parameters structure > + * status - error code > + */ > +void cd_usb_bulk_msd_reset_complete(void *user_data, > + int status); > + > +/* MSD backend api */ > + > +/* allocate new device descriptor */ > +UsbCdBulkMsdDevice *cd_usb_bulk_msd_alloc(void *user_data, uint32_t max_lun); > + > +/* free device descriptor */ > +void cd_usb_bulk_msd_free(UsbCdBulkMsdDevice *device); > + > +/* configure a new Logical Unit to be represen ted by the device > + * returns: error code > + */ > +int cd_usb_bulk_msd_realize(UsbCdBulkMsdDevice *device, uint32_t lun, > + const CdScsiDeviceParameters *dev_params); > + > +/* lock the device, prevent unloading > + * returns: error code > + */ > +int cd_usb_bulk_msd_lock(UsbCdBulkMsdDevice *device, uint32_t lun, gboolean lock); > + > +/* load new media, if already loaded, simulate media change > + * returns: error code > + */ > +int cd_usb_bulk_msd_load(UsbCdBulkMsdDevice *device, uint32_t lun, > + const CdScsiMediaParameters *media_params); > + > +/* query unit parameters and status > + * returns: error code > + */ > +int cd_usb_bulk_msd_get_info(UsbCdBulkMsdDevice *device, uint32_t lun, > + CdScsiDeviceInfo *lun_info); > + > +/* unload the media > + * returns: error code > + */ > +int cd_usb_bulk_msd_unload(UsbCdBulkMsdDevice *device, uint32_t lun); > + > +/* detach a Logical Unit > + * returns: error code > + */ > +int cd_usb_bulk_msd_unrealize(UsbCdBulkMsdDevice *device, uint32_t lun); > + > +/* reset the device instance; cancel all IO ops, reset state > + * returns: error code > + */ > +int cd_usb_bulk_msd_reset(UsbCdBulkMsdDevice *device); > + > + > +/* perform a write data bulk transfer > + * data_len - length of available data to write > + * returns: error code > + */ > +int cd_usb_bulk_msd_write(UsbCdBulkMsdDevice*device, uint8_t *buf, uint32_t data_len); > + > + > +/* perform a read data bulk transfer > + * max_len - length of available buffer to fill > + * If data available immediately, should call cd_usb_bulk_msd_read_complete() > + * and return success > + * If fatal error detected immediately, should call cd_usb_bulk_msd_read_complete() > + * with error code and return success > + * > + * returns: 0 - success, -1 - error > + */ > +int cd_usb_bulk_msd_read(UsbCdBulkMsdDevice *device, uint32_t max_len); > + > +/* cancels pending read data bulk transfer > + * returns: error code > + */ > +int cd_usb_bulk_msd_cancel_read(UsbCdBulkMsdDevice *device); > + > +G_END_DECLS > + > +#endif /* SPICE_GTK_CD_USB_BULK_MSD_H__ */ > diff --git a/src/scsi-constants.h b/src/scsi-constants.h > new file mode 100644 > index 00000000..cf8a8825 > --- /dev/null > +++ b/src/scsi-constants.h > @@ -0,0 +1,324 @@ > +/* -*- Mode: C; c-basic-offset: 4; indent-tabs-mode: nil -*- */ > +/* > + Copyright (C) 2018 Red Hat, Inc. > + Based on the GLib header > + > + This library is free software; you can redistribute it and/or > + modify it under the terms of the GNU Lesser General Public > + License as published by the Free Software Foundation; either > + version 2.1 of the License, or (at your option) any later version. > + > + This library 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 > + Lesser General Public License for more details. > + > + You should have received a copy of the GNU Lesser General Public > + License along with this library; if not, see <http://www.gnu.org/licenses/>. > +*/ > + > +#ifndef SPICE_GTK_SCSI_CONSTANTS_H__ > +#define SPICE_GTK_SCSI_CONSTANTS_H__ > + > +/* > + * SCSI opcodes > + */ > + > +#define TEST_UNIT_READY 0x00 > +#define REWIND 0x01 > +#define REQUEST_SENSE 0x03 > +#define FORMAT_UNIT 0x04 > +#define READ_BLOCK_LIMITS 0x05 > +#define INITIALIZE_ELEMENT_STATUS 0x07 > +#define REASSIGN_BLOCKS_ 0x07 > +#define READ_6 0x08 > +#define WRITE_6 0x0a > +#define SET_CAPACITY 0x0b > +#define READ_REVERSE 0x0f > +#define WRITE_FILEMARKS 0x10 > +#define SPACE 0x11 > +#define INQUIRY 0x12 > +#define RECOVER_BUFFERED_DATA 0x14 > +#define MODE_SELECT 0x15 > +#define RESERVE 0x16 > +#define RELEASE 0x17 > +#define COPY 0x18 > +#define ERASE 0x19 > +#define MODE_SENSE 0x1a > +#define LOAD_UNLOAD 0x1b > +#define SCAN 0x1b > +#define START_STOP 0x1b > +#define RECEIVE_DIAGNOSTIC 0x1c > +#define SEND_DIAGNOSTIC 0x1d > +#define ALLOW_MEDIUM_REMOVAL 0x1e > +#define SET_WINDOW 0x24 > +#define READ_CAPACITY_10 0x25 > +#define GET_WINDOW 0x25 > +#define READ_10 0x28 > +#define WRITE_10 0x2a > +#define SEND 0x2a > +#define SEEK_10 0x2b > +#define LOCATE_10 0x2b > +#define POSITION_TO_ELEMENT 0x2b > +#define WRITE_VERIFY_10 0x2e > +#define VERIFY_10 0x2f > +#define SEARCH_HIGH 0x30 > +#define SEARCH_EQUAL 0x31 > +#define OBJECT_POSITION 0x31 > +#define SEARCH_LOW 0x32 > +#define SET_LIMITS 0x33 > +#define PRE_FETCH 0x34 > +#define READ_POSITION 0x34 > +#define GET_DATA_BUFFER_STATUS 0x34 > +#define SYNCHRONIZE_CACHE 0x35 > +#define LOCK_UNLOCK_CACHE 0x36 > +#define INITIALIZE_ELEMENT_STATUS_WITH_RANGE 0x37 > +#define READ_DEFECT_DATA 0x37 > +#define MEDIUM_SCAN 0x38 > +#define COMPARE 0x39 > +#define COPY_VERIFY 0x3a > +#define WRITE_BUFFER 0x3b > +#define READ_BUFFER 0x3c > +#define UPDATE_BLOCK 0x3d > +#define READ_LONG_10 0x3e > +#define WRITE_LONG_10 0x3f > +#define CHANGE_DEFINITION 0x40 > +#define WRITE_SAME_10 0x41 > +#define UNMAP 0x42 > +#define READ_TOC 0x43 > +#define REPORT_DENSITY_SUPPORT 0x44 > +#define GET_CONFIGURATION 0x46 > +#define SANITIZE 0x48 > +#define GET_EVENT_STATUS_NOTIFICATION 0x4a > +#define LOG_SELECT 0x4c > +#define LOG_SENSE 0x4d > +#define READ_DISC_INFORMATION 0x51 > +#define READ_TRACK_INFORMATION 0x52 > +#define RESERVE_TRACK 0x53 > +#define MODE_SELECT_10 0x55 > +#define RESERVE_10 0x56 > +#define RELEASE_10 0x57 > +#define MODE_SENSE_10 0x5a > +#define SEND_CUE_SHEET 0x5d > +#define PERSISTENT_RESERVE_IN 0x5e > +#define PERSISTENT_RESERVE_OUT 0x5f > +#define VARLENGTH_CDB 0x7f > +#define WRITE_FILEMARKS_16 0x80 > +#define READ_REVERSE_16 0x81 > +#define ALLOW_OVERWRITE 0x82 > +#define EXTENDED_COPY 0x83 > +#define ATA_PASSTHROUGH_16 0x85 > +#define ACCESS_CONTROL_IN 0x86 > +#define ACCESS_CONTROL_OUT 0x87 > +#define READ_16 0x88 > +#define COMPARE_AND_WRITE 0x89 > +#define WRITE_16 0x8a > +#define WRITE_VERIFY_16 0x8e > +#define VERIFY_16 0x8f > +#define PRE_FETCH_16 0x90 > +#define SPACE_16 0x91 > +#define SYNCHRONIZE_CACHE_16 0x91 > +#define LOCATE_16 0x92 > +#define WRITE_SAME_16 0x93 > +#define ERASE_16 0x93 > +#define SERVICE_ACTION_IN_16 0x9e > +#define WRITE_LONG_16 0x9f > +#define REPORT_LUNS 0xa0 > +#define ATA_PASSTHROUGH_12 0xa1 > +#define MAINTENANCE_IN 0xa3 > +#define MAINTENANCE_OUT 0xa4 > +#define MOVE_MEDIUM 0xa5 > +#define EXCHANGE_MEDIUM 0xa6 > +#define SET_READ_AHEAD 0xa7 > +#define READ_12 0xa8 > +#define WRITE_12 0xaa > +#define SERVICE_ACTION_IN_12 0xab > +#define ERASE_12 0xac > +#define WRITE_VERIFY_12 0xae > +#define VERIFY_12 0xaf > +#define SEARCH_HIGH_12 0xb0 > +#define SEARCH_EQUAL_12 0xb1 > +#define SEARCH_LOW_12 0xb2 > +#define READ_ELEMENT_STATUS 0xb8 > +#define SEND_VOLUME_TAG 0xb6 > + > +/* MMC-specific opcode assignment */ > +#define MMC_SEND_EVENT 0xa2 > +#define MMC_SEND_KEY 0xa3 > +#define MMC_REPORT_KEY 0xa4 > +#define MMC_GET_PERFORMANCE 0xac > +#define MMC_READ_DVD_STRUCTURE 0xad > +#define MMC_READ_DEFECT_DATA_12 0xb7 > +#define MMC_SET_CD_SPEED 0xbb > +#define MMC_MECHANISM_STATUS 0xbd > +#define MMC_READ_CD 0xbe > +#define MMC_SEND_DVD_STRUCTURE 0xbf > + > +/* > + * SERVICE ACTION IN subcodes > + */ > +#define SAI_READ_CAPACITY_16 0x10 > + > +/* > + * READ POSITION service action codes > + */ > +#define SHORT_FORM_BLOCK_ID 0x00 > +#define SHORT_FORM_VENDOR_SPECIFIC 0x01 > +#define LONG_FORM 0x06 > +#define EXTENDED_FORM 0x08 > + > +/* > + * SAM Status codes > + */ > + > +#define GOOD 0x00 > +#define CHECK_CONDITION 0x02 > +#define CONDITION_GOOD 0x04 > +#define BUSY 0x08 > +#define INTERMEDIATE_GOOD 0x10 > +#define INTERMEDIATE_C_GOOD 0x14 > +#define RESERVATION_CONFLICT 0x18 > +#define COMMAND_TERMINATED 0x22 > +#define TASK_SET_FULL 0x28 > +#define ACA_ACTIVE 0x30 > +#define TASK_ABORTED 0x40 > + > +#define STATUS_MASK 0x3e > + > +/* > + * SENSE KEYS > + */ > + > +#define NO_SENSE 0x00 > +#define RECOVERED_ERROR 0x01 > +#define NOT_READY 0x02 > +#define MEDIUM_ERROR 0x03 > +#define HARDWARE_ERROR 0x04 > +#define ILLEGAL_REQUEST 0x05 > +#define UNIT_ATTENTION 0x06 > +#define DATA_PROTECT 0x07 > +#define BLANK_CHECK 0x08 > +#define COPY_ABORTED 0x0a > +#define ABORTED_COMMAND 0x0b > +#define VOLUME_OVERFLOW 0x0d > +#define MISCOMPARE 0x0e > + > + > +/* > + * DEVICE TYPES > + */ > + > +#define TYPE_DISK 0x00 > +#define TYPE_TAPE 0x01 > +#define TYPE_PRINTER 0x02 > +#define TYPE_PROCESSOR 0x03 /* HP scanners use this */ > +#define TYPE_WORM 0x04 /* Treated as ROM by our system */ > +#define TYPE_ROM 0x05 > +#define TYPE_SCANNER 0x06 > +#define TYPE_MOD 0x07 /* Magneto-optical disk - > + * - treated as TYPE_DISK */ > +#define TYPE_MEDIUM_CHANGER 0x08 > +#define TYPE_STORAGE_ARRAY 0x0c /* Storage array device */ > +#define TYPE_ENCLOSURE 0x0d /* Enclosure Services Device */ > +#define TYPE_RBC 0x0e /* Simplified Direct-Access Device */ > +#define TYPE_OSD 0x11 /* Object-storage Device */ > +#define TYPE_WLUN 0x1e /* Well known LUN */ > +#define TYPE_NOT_PRESENT 0x1f > +#define TYPE_INACTIVE 0x20 > +#define TYPE_NO_LUN 0x7f > + > +/* Mode page codes for mode sense/set */ > +#define MODE_PAGE_R_W_ERROR 0x01 > +#define MODE_PAGE_MRW 0x03 > +#define MODE_PAGE_HD_GEOMETRY 0x04 > +#define MODE_PAGE_FLEXIBLE_DISK_GEOMETRY 0x05 /* SBC */ > +#define MODE_PAGE_WRITE_PARAMETER 0x05 /* MMC */ > +#define MODE_PAGE_CACHING 0x08 > +#define MODE_PAGE_CD_DEVICE 0x0d > +#define MODE_PAGE_AUDIO_CTL 0x0e > +#define MODE_PAGE_POWER 0x1a > +#define MODE_PAGE_FAULT_FAIL 0x1c > +#define MODE_PAGE_TO_PROTECT 0x1d > +#define MODE_PAGE_CAPS_MECH_STATUS 0x2a > +#define MODE_PAGE_MRW_VENDOR 0x2C > +#define MODE_PAGE_ALLS 0x3f > +/* Not in Mt. Fuji, but in ATAPI 2.6 -- deprecated now in favor > + * of MODE_PAGE_SENSE_POWER */ > +#define MODE_PAGE_CDROM 0x0d > + > +#define MODE_PAGE_CDROM 0x0d > + > +/* Event notification classes for GET EVENT STATUS NOTIFICATION */ > +#define GESN_NO_EVENTS 0 > +#define GESN_OPERATIONAL_CHANGE 1 > +#define GESN_POWER_MANAGEMENT 2 > +#define GESN_EXTERNAL_REQUEST 3 > +#define GESN_MEDIA 4 > +#define GESN_MULTIPLE_HOSTS 5 > +#define GESN_DEVICE_BUSY 6 > + > +/* Event codes for MEDIA event status notification */ > +#define MEC_NO_CHANGE 0 > +#define MEC_EJECT_REQUESTED 1 > +#define MEC_NEW_MEDIA 2 > +#define MEC_MEDIA_REMOVAL 3 /* only for media changers */ > +#define MEC_MEDIA_CHANGED 4 /* only for media changers */ > +#define MEC_BG_FORMAT_COMPLETED 5 /* MRW or DVD+RW b/g format completed */ > +#define MEC_BG_FORMAT_RESTARTED 6 /* MRW or DVD+RW b/g format restarted */ > + > +#define MS_TRAY_OPEN 1 > +#define MS_MEDIA_PRESENT 2 > + > +/* > + * Based on values from <linux/cdrom.h> but extending CD_MINS > + * to the maximum common size allowed by the Orange's Book ATIP > + * > + * 90 and 99 min CDs are also available but using them as the > + * upper limit reduces the effectiveness of the heuristic to > + * detect DVDs burned to less than 25% of their maximum capacity > + */ > + > +/* Some generally useful CD-ROM information */ > +#define CD_MINS 80 /* max. minutes per CD */ > +#define CD_SECS 60 /* seconds per minute */ > +#define CD_FRAMES 75 /* frames per second */ > +#define CD_FRAMESIZE 2048 /* bytes per frame, "cooked" mode */ > +#define CD_MAX_BYTES (CD_MINS * CD_SECS * CD_FRAMES * CD_FRAMESIZE) > +#define CD_MAX_SECTORS (CD_MAX_BYTES / 512) > + > +/* > + * The MMC values are not IDE specific and might need to be moved > + * to a common header if they are also needed for the SCSI emulation > + */ > + > +/* Profile list from MMC-6 revision 1 table 91 */ > +#define MMC_PROFILE_NONE 0x0000 > +#define MMC_PROFILE_CD_ROM 0x0008 > +#define MMC_PROFILE_CD_R 0x0009 > +#define MMC_PROFILE_CD_RW 0x000A > +#define MMC_PROFILE_DVD_ROM 0x0010 > +#define MMC_PROFILE_DVD_R_SR 0x0011 > +#define MMC_PROFILE_DVD_RAM 0x0012 > +#define MMC_PROFILE_DVD_RW_RO 0x0013 > +#define MMC_PROFILE_DVD_RW_SR 0x0014 > +#define MMC_PROFILE_DVD_R_DL_SR 0x0015 > +#define MMC_PROFILE_DVD_R_DL_JR 0x0016 > +#define MMC_PROFILE_DVD_RW_DL 0x0017 > +#define MMC_PROFILE_DVD_DDR 0x0018 > +#define MMC_PROFILE_DVD_PLUS_RW 0x001A > +#define MMC_PROFILE_DVD_PLUS_R 0x001B > +#define MMC_PROFILE_DVD_PLUS_RW_DL 0x002A > +#define MMC_PROFILE_DVD_PLUS_R_DL 0x002B > +#define MMC_PROFILE_BD_ROM 0x0040 > +#define MMC_PROFILE_BD_R_SRM 0x0041 > +#define MMC_PROFILE_BD_R_RRM 0x0042 > +#define MMC_PROFILE_BD_RE 0x0043 > +#define MMC_PROFILE_HDDVD_ROM 0x0050 > +#define MMC_PROFILE_HDDVD_R 0x0051 > +#define MMC_PROFILE_HDDVD_RAM 0x0052 > +#define MMC_PROFILE_HDDVD_RW 0x0053 > +#define MMC_PROFILE_HDDVD_R_DL 0x0058 > +#define MMC_PROFILE_HDDVD_RW_DL 0x005A > +#define MMC_PROFILE_INVALID 0xFFFF > + > +#endif > -- > 2.20.1 > > _______________________________________________ > Spice-devel mailing list > Spice-devel@xxxxxxxxxxxxxxxxxxxxx > https://lists.freedesktop.org/mailman/listinfo/spice-devel
Attachment:
signature.asc
Description: PGP signature
_______________________________________________ Spice-devel mailing list Spice-devel@xxxxxxxxxxxxxxxxxxxxx https://lists.freedesktop.org/mailman/listinfo/spice-devel