1f0cfa1b1SPedro F. Giffuni /*- 2*4d846d26SWarner Losh * SPDX-License-Identifier: BSD-2-Clause 3f0cfa1b1SPedro F. Giffuni * 4c38b150aSNate Lawson * SCSI Disk Emulator 5c38b150aSNate Lawson * 6c38b150aSNate Lawson * Copyright (c) 2002 Nate Lawson. 7c38b150aSNate Lawson * All rights reserved. 8c38b150aSNate Lawson * 9c38b150aSNate Lawson * Redistribution and use in source and binary forms, with or without 10c38b150aSNate Lawson * modification, are permitted provided that the following conditions 11c38b150aSNate Lawson * are met: 12c38b150aSNate Lawson * 1. Redistributions of source code must retain the above copyright 13c38b150aSNate Lawson * notice, this list of conditions, and the following disclaimer, 14c38b150aSNate Lawson * without modification, immediately at the beginning of the file. 15c38b150aSNate Lawson * 2. The name of the author may not be used to endorse or promote products 16c38b150aSNate Lawson * derived from this software without specific prior written permission. 17c38b150aSNate Lawson * 18c38b150aSNate Lawson * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND 19c38b150aSNate Lawson * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE 20c38b150aSNate Lawson * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE 21c38b150aSNate Lawson * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE FOR 22c38b150aSNate Lawson * ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL 23c38b150aSNate Lawson * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS 24c38b150aSNate Lawson * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) 25c38b150aSNate Lawson * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT 26c38b150aSNate Lawson * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY 27c38b150aSNate Lawson * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF 28c38b150aSNate Lawson * SUCH DAMAGE. 29c38b150aSNate Lawson * 30c38b150aSNate Lawson * $FreeBSD$ 31c38b150aSNate Lawson */ 32c38b150aSNate Lawson 33c38b150aSNate Lawson #include <stdio.h> 34c38b150aSNate Lawson #include <stddef.h> 35c38b150aSNate Lawson #include <stdarg.h> 36c38b150aSNate Lawson #include <stdlib.h> 37c38b150aSNate Lawson #include <string.h> 38c38b150aSNate Lawson #include <err.h> 39c38b150aSNate Lawson #include <aio.h> 40cd6dedfcSMatt Jacob #include <unistd.h> 41c38b150aSNate Lawson #include <assert.h> 42068d70baSHidetoshi Shimokawa #include <sys/param.h> 43c38b150aSNate Lawson #include <sys/types.h> 44c38b150aSNate Lawson 45c38b150aSNate Lawson #include <cam/cam.h> 46c38b150aSNate Lawson #include <cam/cam_ccb.h> 47c38b150aSNate Lawson #include <cam/scsi/scsi_all.h> 48c38b150aSNate Lawson #include <cam/scsi/scsi_targetio.h> 49c38b150aSNate Lawson #include "scsi_target.h" 50c38b150aSNate Lawson 51c38b150aSNate Lawson typedef int targ_start_func(struct ccb_accept_tio *, struct ccb_scsiio *); 52c38b150aSNate Lawson typedef void targ_done_func(struct ccb_accept_tio *, struct ccb_scsiio *, 53c38b150aSNate Lawson io_ops); 540fa27452SMatt Jacob #ifndef REPORT_LUNS 550fa27452SMatt Jacob #define REPORT_LUNS 0xa0 560fa27452SMatt Jacob #endif 57c38b150aSNate Lawson 58c38b150aSNate Lawson struct targ_cdb_handlers { 59c38b150aSNate Lawson u_int8_t cmd; 60c38b150aSNate Lawson targ_start_func *start; 61c38b150aSNate Lawson targ_done_func *done; 62c38b150aSNate Lawson #define ILLEGAL_CDB 0xFF 63c38b150aSNate Lawson }; 64c38b150aSNate Lawson 65c38b150aSNate Lawson static targ_start_func tcmd_inquiry; 66c38b150aSNate Lawson static targ_start_func tcmd_req_sense; 67c38b150aSNate Lawson static targ_start_func tcmd_rd_cap; 68068d70baSHidetoshi Shimokawa #ifdef READ_16 69068d70baSHidetoshi Shimokawa static targ_start_func tcmd_rd_cap16; 70068d70baSHidetoshi Shimokawa #endif 71c38b150aSNate Lawson static targ_start_func tcmd_rdwr; 72c38b150aSNate Lawson static targ_start_func tcmd_rdwr_decode; 73c38b150aSNate Lawson static targ_done_func tcmd_rdwr_done; 74c38b150aSNate Lawson static targ_start_func tcmd_null_ok; 75c38b150aSNate Lawson static targ_start_func tcmd_illegal_req; 76c38b150aSNate Lawson static int start_io(struct ccb_accept_tio *atio, 77c38b150aSNate Lawson struct ccb_scsiio *ctio, int dir); 78c38b150aSNate Lawson static int init_inquiry(u_int16_t req_flags, u_int16_t sim_flags); 79c38b150aSNate Lawson static struct initiator_state * 80c38b150aSNate Lawson tcmd_get_istate(u_int init_id); 81c38b150aSNate Lawson static void cdb_debug(u_int8_t *cdb, const char *msg, ...); 82c38b150aSNate Lawson 83c38b150aSNate Lawson static struct targ_cdb_handlers cdb_handlers[] = { 84c38b150aSNate Lawson { READ_10, tcmd_rdwr, tcmd_rdwr_done }, 85c38b150aSNate Lawson { WRITE_10, tcmd_rdwr, tcmd_rdwr_done }, 86c38b150aSNate Lawson { READ_6, tcmd_rdwr, tcmd_rdwr_done }, 87c38b150aSNate Lawson { WRITE_6, tcmd_rdwr, tcmd_rdwr_done }, 88c38b150aSNate Lawson { INQUIRY, tcmd_inquiry, NULL }, 89c38b150aSNate Lawson { REQUEST_SENSE, tcmd_req_sense, NULL }, 90c38b150aSNate Lawson { READ_CAPACITY, tcmd_rd_cap, NULL }, 91c38b150aSNate Lawson { TEST_UNIT_READY, tcmd_null_ok, NULL }, 92c38b150aSNate Lawson { START_STOP_UNIT, tcmd_null_ok, NULL }, 93c38b150aSNate Lawson { SYNCHRONIZE_CACHE, tcmd_null_ok, NULL }, 94c38b150aSNate Lawson { MODE_SENSE_6, tcmd_illegal_req, NULL }, 95c38b150aSNate Lawson { MODE_SELECT_6, tcmd_illegal_req, NULL }, 960fa27452SMatt Jacob { REPORT_LUNS, tcmd_illegal_req, NULL }, 97068d70baSHidetoshi Shimokawa #ifdef READ_16 98068d70baSHidetoshi Shimokawa { READ_16, tcmd_rdwr, tcmd_rdwr_done }, 99068d70baSHidetoshi Shimokawa { WRITE_16, tcmd_rdwr, tcmd_rdwr_done }, 100068d70baSHidetoshi Shimokawa { SERVICE_ACTION_IN, tcmd_rd_cap16, NULL }, 101068d70baSHidetoshi Shimokawa #endif 102c38b150aSNate Lawson { ILLEGAL_CDB, NULL, NULL } 103c38b150aSNate Lawson }; 104c38b150aSNate Lawson 105c38b150aSNate Lawson static struct scsi_inquiry_data inq_data; 106c38b150aSNate Lawson static struct initiator_state istates[MAX_INITIATORS]; 107c38b150aSNate Lawson extern int debug; 10888c707c8SAlexander Motin extern off_t volume_size; 10988c707c8SAlexander Motin extern u_int sector_size; 110c38b150aSNate Lawson extern size_t buf_size; 111c38b150aSNate Lawson 112c38b150aSNate Lawson cam_status 113c38b150aSNate Lawson tcmd_init(u_int16_t req_inq_flags, u_int16_t sim_inq_flags) 114c38b150aSNate Lawson { 115c38b150aSNate Lawson struct initiator_state *istate; 116c38b150aSNate Lawson int i, ret; 117c38b150aSNate Lawson 118c38b150aSNate Lawson /* Initialize our inquiry data */ 119c38b150aSNate Lawson ret = init_inquiry(req_inq_flags, sim_inq_flags); 120c38b150aSNate Lawson if (ret != 0) 121c38b150aSNate Lawson return (ret); 122c38b150aSNate Lawson 123c38b150aSNate Lawson /* We start out life with a UA to indicate power-on/reset. */ 124c38b150aSNate Lawson for (i = 0; i < MAX_INITIATORS; i++) { 125c38b150aSNate Lawson istate = tcmd_get_istate(i); 126c38b150aSNate Lawson bzero(istate, sizeof(*istate)); 127c38b150aSNate Lawson istate->pending_ua = UA_POWER_ON; 128c38b150aSNate Lawson } 129c38b150aSNate Lawson 130c38b150aSNate Lawson return (0); 131c38b150aSNate Lawson } 132c38b150aSNate Lawson 133c38b150aSNate Lawson /* Caller allocates CTIO, sets its init_id 134c38b150aSNate Lawson return 0 if done, 1 if more processing needed 135c38b150aSNate Lawson on 0, caller sets SEND_STATUS */ 136c38b150aSNate Lawson int 137c38b150aSNate Lawson tcmd_handle(struct ccb_accept_tio *atio, struct ccb_scsiio *ctio, io_ops event) 138c38b150aSNate Lawson { 139c38b150aSNate Lawson static struct targ_cdb_handlers *last_cmd; 140c38b150aSNate Lawson struct initiator_state *istate; 141c38b150aSNate Lawson struct atio_descr *a_descr; 142c38b150aSNate Lawson int ret; 143c38b150aSNate Lawson 144031bacf8SNate Lawson if (debug) { 145c38b150aSNate Lawson warnx("tcmd_handle atio %p ctio %p atioflags %#x", atio, ctio, 146c38b150aSNate Lawson atio->ccb_h.flags); 147031bacf8SNate Lawson } 148c38b150aSNate Lawson ret = 0; 149c38b150aSNate Lawson a_descr = (struct atio_descr *)atio->ccb_h.targ_descr; 150c38b150aSNate Lawson 151c38b150aSNate Lawson /* Do a full lookup if one-behind cache failed */ 152c38b150aSNate Lawson if (last_cmd == NULL || last_cmd->cmd != a_descr->cdb[0]) { 153c38b150aSNate Lawson struct targ_cdb_handlers *h; 154c38b150aSNate Lawson 155c38b150aSNate Lawson for (h = cdb_handlers; h->cmd != ILLEGAL_CDB; h++) { 156c38b150aSNate Lawson if (a_descr->cdb[0] == h->cmd) 157c38b150aSNate Lawson break; 158c38b150aSNate Lawson } 159c38b150aSNate Lawson last_cmd = h; 160c38b150aSNate Lawson } 161441201ceSMatt Jacob 162441201ceSMatt Jacob /* call completion and exit */ 163441201ceSMatt Jacob if (event != ATIO_WORK) { 164441201ceSMatt Jacob if (last_cmd->done != NULL) 165441201ceSMatt Jacob last_cmd->done(atio, ctio, event); 166441201ceSMatt Jacob else 167441201ceSMatt Jacob free_ccb((union ccb *)ctio); 168441201ceSMatt Jacob return (1); 169441201ceSMatt Jacob } 170441201ceSMatt Jacob 171c38b150aSNate Lawson if (last_cmd->cmd == ILLEGAL_CDB) { 172c38b150aSNate Lawson if (event != ATIO_WORK) { 173c38b150aSNate Lawson warnx("no done func for %#x???", a_descr->cdb[0]); 174c38b150aSNate Lawson abort(); 175c38b150aSNate Lawson } 176c38b150aSNate Lawson /* Not found, return illegal request */ 177c38b150aSNate Lawson warnx("cdb %#x not handled", a_descr->cdb[0]); 178c38b150aSNate Lawson tcmd_illegal_req(atio, ctio); 179c38b150aSNate Lawson send_ccb((union ccb *)ctio, /*priority*/1); 180c38b150aSNate Lawson return (0); 181c38b150aSNate Lawson } 182c38b150aSNate Lawson 183c38b150aSNate Lawson istate = tcmd_get_istate(ctio->init_id); 184c38b150aSNate Lawson if (istate == NULL) { 185c38b150aSNate Lawson tcmd_illegal_req(atio, ctio); 186c38b150aSNate Lawson send_ccb((union ccb *)ctio, /*priority*/1); 187c38b150aSNate Lawson return (0); 188c38b150aSNate Lawson } 189c38b150aSNate Lawson 190c38b150aSNate Lawson if (istate->pending_ca == 0 && istate->pending_ua != 0 && 191c38b150aSNate Lawson a_descr->cdb[0] != INQUIRY) { 192c38b150aSNate Lawson tcmd_sense(ctio->init_id, ctio, SSD_KEY_UNIT_ATTENTION, 193c38b150aSNate Lawson 0x29, istate->pending_ua == UA_POWER_ON ? 1 : 2); 194c38b150aSNate Lawson istate->pending_ca = CA_UNIT_ATTN; 195c38b150aSNate Lawson if (debug) { 196c38b150aSNate Lawson cdb_debug(a_descr->cdb, "UA active for %u: ", 197c38b150aSNate Lawson atio->init_id); 198c38b150aSNate Lawson } 199c38b150aSNate Lawson send_ccb((union ccb *)ctio, /*priority*/1); 200c38b150aSNate Lawson return (0); 201c38b150aSNate Lawson } 202c38b150aSNate Lawson 203c38b150aSNate Lawson /* Store current CA and UA for later */ 204c38b150aSNate Lawson istate->orig_ua = istate->pending_ua; 205c38b150aSNate Lawson istate->orig_ca = istate->pending_ca; 206c38b150aSNate Lawson 207c38b150aSNate Lawson /* 208c38b150aSNate Lawson * As per SAM2, any command that occurs 209c38b150aSNate Lawson * after a CA is reported, clears the CA. We must 210c38b150aSNate Lawson * also clear the UA condition, if any, that caused 211c38b150aSNate Lawson * the CA to occur assuming the UA is not for a 212c38b150aSNate Lawson * persistent condition. 213c38b150aSNate Lawson */ 214c38b150aSNate Lawson istate->pending_ca = CA_NONE; 215c38b150aSNate Lawson if (istate->orig_ca == CA_UNIT_ATTN) 216c38b150aSNate Lawson istate->pending_ua = UA_NONE; 217c38b150aSNate Lawson 218c38b150aSNate Lawson /* If we have a valid handler, call start or completion function */ 219c38b150aSNate Lawson if (last_cmd->cmd != ILLEGAL_CDB) { 220c38b150aSNate Lawson ret = last_cmd->start(atio, ctio); 221c38b150aSNate Lawson /* XXX hack */ 222c38b150aSNate Lawson if (last_cmd->start != tcmd_rdwr) { 223c38b150aSNate Lawson a_descr->init_req += ctio->dxfer_len; 224c38b150aSNate Lawson send_ccb((union ccb *)ctio, /*priority*/1); 225c38b150aSNate Lawson } 226c38b150aSNate Lawson } 227c38b150aSNate Lawson 228c38b150aSNate Lawson return (ret); 229c38b150aSNate Lawson } 230c38b150aSNate Lawson 231c38b150aSNate Lawson static struct initiator_state * 232c38b150aSNate Lawson tcmd_get_istate(u_int init_id) 233c38b150aSNate Lawson { 234c38b150aSNate Lawson if (init_id >= MAX_INITIATORS) { 235c38b150aSNate Lawson warnx("illegal init_id %d, max %d", init_id, MAX_INITIATORS - 1); 236c38b150aSNate Lawson return (NULL); 237c38b150aSNate Lawson } else { 238c38b150aSNate Lawson return (&istates[init_id]); 239c38b150aSNate Lawson } 240c38b150aSNate Lawson } 241c38b150aSNate Lawson 242c38b150aSNate Lawson void 243c38b150aSNate Lawson tcmd_sense(u_int init_id, struct ccb_scsiio *ctio, u_int8_t flags, 244c38b150aSNate Lawson u_int8_t asc, u_int8_t ascq) 245c38b150aSNate Lawson { 246c38b150aSNate Lawson struct initiator_state *istate; 2471cc052e8SKenneth D. Merry struct scsi_sense_data_fixed *sense; 248c38b150aSNate Lawson 249c38b150aSNate Lawson /* Set our initiator's istate */ 250c38b150aSNate Lawson istate = tcmd_get_istate(init_id); 251c38b150aSNate Lawson if (istate == NULL) 252c38b150aSNate Lawson return; 253c38b150aSNate Lawson istate->pending_ca |= CA_CMD_SENSE; /* XXX set instead of or? */ 2541cc052e8SKenneth D. Merry sense = (struct scsi_sense_data_fixed *)&istate->sense_data; 255c38b150aSNate Lawson bzero(sense, sizeof(*sense)); 256c38b150aSNate Lawson sense->error_code = SSD_CURRENT_ERROR; 257c38b150aSNate Lawson sense->flags = flags; 258c38b150aSNate Lawson sense->add_sense_code = asc; 259c38b150aSNate Lawson sense->add_sense_code_qual = ascq; 260c38b150aSNate Lawson sense->extra_len = 2611cc052e8SKenneth D. Merry offsetof(struct scsi_sense_data_fixed, sense_key_spec[2]) - 2621cc052e8SKenneth D. Merry offsetof(struct scsi_sense_data_fixed, extra_len); 263c38b150aSNate Lawson 264c38b150aSNate Lawson /* Fill out the supplied CTIO */ 265c38b150aSNate Lawson if (ctio != NULL) { 266c38b150aSNate Lawson bcopy(sense, &ctio->sense_data, sizeof(*sense)); 267c4b3637bSHidetoshi Shimokawa ctio->sense_len = sizeof(*sense); /* XXX */ 268c38b150aSNate Lawson ctio->ccb_h.flags &= ~CAM_DIR_MASK; 2694c1cc94eSNate Lawson ctio->ccb_h.flags |= CAM_DIR_NONE | CAM_SEND_SENSE | 270c38b150aSNate Lawson CAM_SEND_STATUS; 271c38b150aSNate Lawson ctio->dxfer_len = 0; 272c38b150aSNate Lawson ctio->scsi_status = SCSI_STATUS_CHECK_COND; 273c38b150aSNate Lawson } 274c38b150aSNate Lawson } 275c38b150aSNate Lawson 276c38b150aSNate Lawson void 277c38b150aSNate Lawson tcmd_ua(u_int init_id, ua_types new_ua) 278c38b150aSNate Lawson { 279c38b150aSNate Lawson struct initiator_state *istate; 280c38b150aSNate Lawson u_int start, end; 281c38b150aSNate Lawson 282c38b150aSNate Lawson if (init_id == CAM_TARGET_WILDCARD) { 283c38b150aSNate Lawson start = 0; 284c38b150aSNate Lawson end = MAX_INITIATORS - 1; 285c38b150aSNate Lawson } else { 286c38b150aSNate Lawson start = end = init_id; 287c38b150aSNate Lawson } 288c38b150aSNate Lawson 289c38b150aSNate Lawson for (; start <= end; start++) { 290c38b150aSNate Lawson istate = tcmd_get_istate(start); 291c38b150aSNate Lawson if (istate == NULL) 292c38b150aSNate Lawson break; 293c38b150aSNate Lawson istate->pending_ua = new_ua; 294c38b150aSNate Lawson } 295c38b150aSNate Lawson } 296c38b150aSNate Lawson 297c38b150aSNate Lawson static int 298c38b150aSNate Lawson tcmd_inquiry(struct ccb_accept_tio *atio, struct ccb_scsiio *ctio) 299c38b150aSNate Lawson { 300c38b150aSNate Lawson struct scsi_inquiry *inq; 301c38b150aSNate Lawson struct atio_descr *a_descr; 302c38b150aSNate Lawson struct initiator_state *istate; 3031cc052e8SKenneth D. Merry struct scsi_sense_data_fixed *sense; 304c38b150aSNate Lawson 305c38b150aSNate Lawson a_descr = (struct atio_descr *)atio->ccb_h.targ_descr; 306c38b150aSNate Lawson inq = (struct scsi_inquiry *)a_descr->cdb; 307c38b150aSNate Lawson 308c38b150aSNate Lawson if (debug) 309c38b150aSNate Lawson cdb_debug(a_descr->cdb, "INQUIRY from %u: ", atio->init_id); 310c38b150aSNate Lawson /* 311c38b150aSNate Lawson * Validate the command. We don't support any VPD pages, so 312c38b150aSNate Lawson * complain if EVPD or CMDDT is set. 313c38b150aSNate Lawson */ 314c38b150aSNate Lawson istate = tcmd_get_istate(ctio->init_id); 3151cc052e8SKenneth D. Merry sense = (struct scsi_sense_data_fixed *)&istate->sense_data; 316c38b150aSNate Lawson if ((inq->byte2 & SI_EVPD) != 0) { 317c38b150aSNate Lawson tcmd_illegal_req(atio, ctio); 318c38b150aSNate Lawson sense->sense_key_spec[0] = SSD_SCS_VALID | SSD_FIELDPTR_CMD | 319c38b150aSNate Lawson SSD_BITPTR_VALID | /*bit value*/1; 320c38b150aSNate Lawson sense->sense_key_spec[1] = 0; 321c38b150aSNate Lawson sense->sense_key_spec[2] = 322c38b150aSNate Lawson offsetof(struct scsi_inquiry, byte2); 323c38b150aSNate Lawson } else if (inq->page_code != 0) { 324c38b150aSNate Lawson tcmd_illegal_req(atio, ctio); 325c38b150aSNate Lawson sense->sense_key_spec[0] = SSD_SCS_VALID | SSD_FIELDPTR_CMD; 326c38b150aSNate Lawson sense->sense_key_spec[1] = 0; 327c38b150aSNate Lawson sense->sense_key_spec[2] = 328c38b150aSNate Lawson offsetof(struct scsi_inquiry, page_code); 329c38b150aSNate Lawson } else { 330c38b150aSNate Lawson bcopy(&inq_data, ctio->data_ptr, sizeof(inq_data)); 331c38b150aSNate Lawson ctio->dxfer_len = inq_data.additional_length + 4; 332c38b150aSNate Lawson ctio->dxfer_len = min(ctio->dxfer_len, 333130f4520SKenneth D. Merry scsi_2btoul(inq->length)); 334c38b150aSNate Lawson ctio->ccb_h.flags |= CAM_DIR_IN | CAM_SEND_STATUS; 335c38b150aSNate Lawson ctio->scsi_status = SCSI_STATUS_OK; 336c38b150aSNate Lawson } 337c38b150aSNate Lawson return (0); 338c38b150aSNate Lawson } 339c38b150aSNate Lawson 340c38b150aSNate Lawson /* Initialize the inquiry response structure with the requested flags */ 341c38b150aSNate Lawson static int 342c38b150aSNate Lawson init_inquiry(u_int16_t req_flags, u_int16_t sim_flags) 343c38b150aSNate Lawson { 344c38b150aSNate Lawson struct scsi_inquiry_data *inq; 345c38b150aSNate Lawson 346c38b150aSNate Lawson inq = &inq_data; 347c38b150aSNate Lawson bzero(inq, sizeof(*inq)); 348c38b150aSNate Lawson inq->device = T_DIRECT | (SID_QUAL_LU_CONNECTED << 5); 349c4b3637bSHidetoshi Shimokawa #ifdef SCSI_REV_SPC 350c38b150aSNate Lawson inq->version = SCSI_REV_SPC; /* was 2 */ 351c4b3637bSHidetoshi Shimokawa #else 352c4b3637bSHidetoshi Shimokawa inq->version = SCSI_REV_3; /* was 2 */ 353c4b3637bSHidetoshi Shimokawa #endif 354c38b150aSNate Lawson 355c38b150aSNate Lawson /* 356c38b150aSNate Lawson * XXX cpi.hba_inquiry doesn't support Addr16 so we give the 357c38b150aSNate Lawson * user what they want if they ask for it. 358c38b150aSNate Lawson */ 359c38b150aSNate Lawson if ((req_flags & SID_Addr16) != 0) { 360c38b150aSNate Lawson sim_flags |= SID_Addr16; 361c38b150aSNate Lawson warnx("Not sure SIM supports Addr16 but enabling it anyway"); 362c38b150aSNate Lawson } 363c38b150aSNate Lawson 364c38b150aSNate Lawson /* Advertise only what the SIM can actually support */ 365c38b150aSNate Lawson req_flags &= sim_flags; 366cd6dedfcSMatt Jacob scsi_ulto2b(req_flags, &inq->spc2_flags); 367c38b150aSNate Lawson 368c38b150aSNate Lawson inq->response_format = 2; /* SCSI2 Inquiry Format */ 369c38b150aSNate Lawson inq->additional_length = SHORT_INQUIRY_LENGTH - 370c38b150aSNate Lawson offsetof(struct scsi_inquiry_data, additional_length); 371c38b150aSNate Lawson bcopy("FreeBSD ", inq->vendor, SID_VENDOR_SIZE); 372c38b150aSNate Lawson bcopy("Emulated Disk ", inq->product, SID_PRODUCT_SIZE); 373c38b150aSNate Lawson bcopy("0.1 ", inq->revision, SID_REVISION_SIZE); 374c38b150aSNate Lawson return (0); 375c38b150aSNate Lawson } 376c38b150aSNate Lawson 377c38b150aSNate Lawson static int 378c38b150aSNate Lawson tcmd_req_sense(struct ccb_accept_tio *atio, struct ccb_scsiio *ctio) 379c38b150aSNate Lawson { 380c38b150aSNate Lawson struct scsi_request_sense *rsense; 3811cc052e8SKenneth D. Merry struct scsi_sense_data_fixed *sense; 382c38b150aSNate Lawson struct initiator_state *istate; 383c38b150aSNate Lawson size_t dlen; 384c38b150aSNate Lawson struct atio_descr *a_descr; 385c38b150aSNate Lawson 386c38b150aSNate Lawson a_descr = (struct atio_descr *)atio->ccb_h.targ_descr; 387c38b150aSNate Lawson rsense = (struct scsi_request_sense *)a_descr->cdb; 388c38b150aSNate Lawson 389c38b150aSNate Lawson istate = tcmd_get_istate(ctio->init_id); 3901cc052e8SKenneth D. Merry sense = (struct scsi_sense_data_fixed *)&istate->sense_data; 391c38b150aSNate Lawson 392c38b150aSNate Lawson if (debug) { 393c38b150aSNate Lawson cdb_debug(a_descr->cdb, "REQ SENSE from %u: ", atio->init_id); 394c38b150aSNate Lawson warnx("Sending sense: %#x %#x %#x", sense->flags, 395c38b150aSNate Lawson sense->add_sense_code, sense->add_sense_code_qual); 396c38b150aSNate Lawson } 397c38b150aSNate Lawson 398c38b150aSNate Lawson if (istate->orig_ca == 0) { 399c38b150aSNate Lawson tcmd_sense(ctio->init_id, NULL, SSD_KEY_NO_SENSE, 0, 0); 400c38b150aSNate Lawson warnx("REQUEST SENSE from %u but no pending CA!", 401c38b150aSNate Lawson ctio->init_id); 402c38b150aSNate Lawson } 403c38b150aSNate Lawson 404c38b150aSNate Lawson bcopy(sense, ctio->data_ptr, sizeof(struct scsi_sense_data)); 4051cc052e8SKenneth D. Merry dlen = offsetof(struct scsi_sense_data_fixed, extra_len) + 406c38b150aSNate Lawson sense->extra_len + 1; 407c38b150aSNate Lawson ctio->dxfer_len = min(dlen, SCSI_CDB6_LEN(rsense->length)); 408c38b150aSNate Lawson ctio->ccb_h.flags |= CAM_DIR_IN | CAM_SEND_STATUS; 409c38b150aSNate Lawson ctio->scsi_status = SCSI_STATUS_OK; 410c38b150aSNate Lawson return (0); 411c38b150aSNate Lawson } 412c38b150aSNate Lawson 413c38b150aSNate Lawson static int 414c38b150aSNate Lawson tcmd_rd_cap(struct ccb_accept_tio *atio, struct ccb_scsiio *ctio) 415c38b150aSNate Lawson { 416c38b150aSNate Lawson struct scsi_read_capacity_data *srp; 417c38b150aSNate Lawson struct atio_descr *a_descr; 418068d70baSHidetoshi Shimokawa uint32_t vsize; 419c38b150aSNate Lawson 420c38b150aSNate Lawson a_descr = (struct atio_descr *)atio->ccb_h.targ_descr; 421c38b150aSNate Lawson srp = (struct scsi_read_capacity_data *)ctio->data_ptr; 422c38b150aSNate Lawson 423068d70baSHidetoshi Shimokawa if (volume_size > 0xffffffff) 424068d70baSHidetoshi Shimokawa vsize = 0xffffffff; 425068d70baSHidetoshi Shimokawa else 426068d70baSHidetoshi Shimokawa vsize = (uint32_t)(volume_size - 1); 427068d70baSHidetoshi Shimokawa 428c38b150aSNate Lawson if (debug) { 429c38b150aSNate Lawson cdb_debug(a_descr->cdb, "READ CAP from %u (%u, %u): ", 430068d70baSHidetoshi Shimokawa atio->init_id, vsize, sector_size); 431c38b150aSNate Lawson } 432c38b150aSNate Lawson 433c38b150aSNate Lawson bzero(srp, sizeof(*srp)); 434068d70baSHidetoshi Shimokawa scsi_ulto4b(vsize, srp->addr); 435c38b150aSNate Lawson scsi_ulto4b(sector_size, srp->length); 436c38b150aSNate Lawson 437c38b150aSNate Lawson ctio->dxfer_len = sizeof(*srp); 438c38b150aSNate Lawson ctio->ccb_h.flags |= CAM_DIR_IN | CAM_SEND_STATUS; 439c38b150aSNate Lawson ctio->scsi_status = SCSI_STATUS_OK; 440c38b150aSNate Lawson return (0); 441c38b150aSNate Lawson } 442c38b150aSNate Lawson 443068d70baSHidetoshi Shimokawa #ifdef READ_16 444068d70baSHidetoshi Shimokawa static int 445068d70baSHidetoshi Shimokawa tcmd_rd_cap16(struct ccb_accept_tio *atio, struct ccb_scsiio *ctio) 446068d70baSHidetoshi Shimokawa { 447068d70baSHidetoshi Shimokawa struct scsi_read_capacity_16 *scsi_cmd; 448068d70baSHidetoshi Shimokawa struct scsi_read_capacity_data_long *srp; 449068d70baSHidetoshi Shimokawa struct atio_descr *a_descr; 450068d70baSHidetoshi Shimokawa 451068d70baSHidetoshi Shimokawa a_descr = (struct atio_descr *)atio->ccb_h.targ_descr; 452068d70baSHidetoshi Shimokawa scsi_cmd = (struct scsi_read_capacity_16 *)a_descr->cdb; 453068d70baSHidetoshi Shimokawa srp = (struct scsi_read_capacity_data_long *)ctio->data_ptr; 454068d70baSHidetoshi Shimokawa 455068d70baSHidetoshi Shimokawa if (scsi_cmd->service_action != SRC16_SERVICE_ACTION) { 456068d70baSHidetoshi Shimokawa tcmd_illegal_req(atio, ctio); 457068d70baSHidetoshi Shimokawa return (0); 458068d70baSHidetoshi Shimokawa } 459068d70baSHidetoshi Shimokawa 460068d70baSHidetoshi Shimokawa if (debug) { 461068d70baSHidetoshi Shimokawa cdb_debug(a_descr->cdb, "READ CAP16 from %u (%u, %u): ", 462068d70baSHidetoshi Shimokawa atio->init_id, volume_size - 1, sector_size); 463068d70baSHidetoshi Shimokawa } 464068d70baSHidetoshi Shimokawa 465068d70baSHidetoshi Shimokawa bzero(srp, sizeof(*srp)); 466068d70baSHidetoshi Shimokawa scsi_u64to8b(volume_size - 1, srp->addr); 467068d70baSHidetoshi Shimokawa scsi_ulto4b(sector_size, srp->length); 468068d70baSHidetoshi Shimokawa 469068d70baSHidetoshi Shimokawa ctio->dxfer_len = sizeof(*srp); 470068d70baSHidetoshi Shimokawa ctio->ccb_h.flags |= CAM_DIR_IN | CAM_SEND_STATUS; 471068d70baSHidetoshi Shimokawa ctio->scsi_status = SCSI_STATUS_OK; 472068d70baSHidetoshi Shimokawa return (0); 473068d70baSHidetoshi Shimokawa } 474068d70baSHidetoshi Shimokawa #endif 475068d70baSHidetoshi Shimokawa 476c38b150aSNate Lawson static int 477c38b150aSNate Lawson tcmd_rdwr(struct ccb_accept_tio *atio, struct ccb_scsiio *ctio) 478c38b150aSNate Lawson { 479c38b150aSNate Lawson struct atio_descr *a_descr; 480c38b150aSNate Lawson struct ctio_descr *c_descr; 481c38b150aSNate Lawson int ret; 482c38b150aSNate Lawson 483c38b150aSNate Lawson a_descr = (struct atio_descr *)atio->ccb_h.targ_descr; 484c38b150aSNate Lawson c_descr = (struct ctio_descr *)ctio->ccb_h.targ_descr; 485c38b150aSNate Lawson 486c38b150aSNate Lawson /* Command needs to be decoded */ 4871cc052e8SKenneth D. Merry if ((a_descr->flags & CAM_DIR_MASK) == CAM_DIR_BOTH) { 488c38b150aSNate Lawson if (debug) 489c38b150aSNate Lawson warnx("Calling rdwr_decode"); 490c38b150aSNate Lawson ret = tcmd_rdwr_decode(atio, ctio); 491c38b150aSNate Lawson if (ret == 0) { 492c38b150aSNate Lawson send_ccb((union ccb *)ctio, /*priority*/1); 493c38b150aSNate Lawson return (0); 494c38b150aSNate Lawson } 495c38b150aSNate Lawson } 496c38b150aSNate Lawson ctio->ccb_h.flags |= a_descr->flags; 497c38b150aSNate Lawson 498c38b150aSNate Lawson /* Call appropriate work function */ 499c38b150aSNate Lawson if ((a_descr->flags & CAM_DIR_IN) != 0) { 500c38b150aSNate Lawson ret = start_io(atio, ctio, CAM_DIR_IN); 501c38b150aSNate Lawson if (debug) 502cd6dedfcSMatt Jacob warnx("Starting %p DIR_IN @" OFF_FMT ":%u", 503cd6dedfcSMatt Jacob a_descr, c_descr->offset, a_descr->targ_req); 504c38b150aSNate Lawson } else { 505c38b150aSNate Lawson ret = start_io(atio, ctio, CAM_DIR_OUT); 506c38b150aSNate Lawson if (debug) 507cd6dedfcSMatt Jacob warnx("Starting %p DIR_OUT @" OFF_FMT ":%u", 508cd6dedfcSMatt Jacob a_descr, c_descr->offset, a_descr->init_req); 509c38b150aSNate Lawson } 510c38b150aSNate Lawson 511c38b150aSNate Lawson return (ret); 512c38b150aSNate Lawson } 513c38b150aSNate Lawson 514c38b150aSNate Lawson static int 515c38b150aSNate Lawson tcmd_rdwr_decode(struct ccb_accept_tio *atio, struct ccb_scsiio *ctio) 516c38b150aSNate Lawson { 517068d70baSHidetoshi Shimokawa uint64_t blkno; 518068d70baSHidetoshi Shimokawa uint32_t count; 519c38b150aSNate Lawson struct atio_descr *a_descr; 520c38b150aSNate Lawson u_int8_t *cdb; 521c38b150aSNate Lawson 522c38b150aSNate Lawson a_descr = (struct atio_descr *)atio->ccb_h.targ_descr; 523c38b150aSNate Lawson cdb = a_descr->cdb; 524c38b150aSNate Lawson if (debug) 525c38b150aSNate Lawson cdb_debug(cdb, "R/W from %u: ", atio->init_id); 526c38b150aSNate Lawson 527068d70baSHidetoshi Shimokawa switch (cdb[0]) { 528068d70baSHidetoshi Shimokawa case READ_6: 529068d70baSHidetoshi Shimokawa case WRITE_6: 530068d70baSHidetoshi Shimokawa { 531c38b150aSNate Lawson struct scsi_rw_6 *rw_6 = (struct scsi_rw_6 *)cdb; 532c38b150aSNate Lawson blkno = scsi_3btoul(rw_6->addr); 533c38b150aSNate Lawson count = rw_6->length; 534068d70baSHidetoshi Shimokawa break; 535068d70baSHidetoshi Shimokawa } 536068d70baSHidetoshi Shimokawa case READ_10: 537068d70baSHidetoshi Shimokawa case WRITE_10: 538068d70baSHidetoshi Shimokawa { 539c38b150aSNate Lawson struct scsi_rw_10 *rw_10 = (struct scsi_rw_10 *)cdb; 540c38b150aSNate Lawson blkno = scsi_4btoul(rw_10->addr); 541c38b150aSNate Lawson count = scsi_2btoul(rw_10->length); 542068d70baSHidetoshi Shimokawa break; 543068d70baSHidetoshi Shimokawa } 544068d70baSHidetoshi Shimokawa #ifdef READ_16 545068d70baSHidetoshi Shimokawa case READ_16: 546068d70baSHidetoshi Shimokawa case WRITE_16: 547068d70baSHidetoshi Shimokawa { 548068d70baSHidetoshi Shimokawa struct scsi_rw_16 *rw_16 = (struct scsi_rw_16 *)cdb; 549068d70baSHidetoshi Shimokawa blkno = scsi_8btou64(rw_16->addr); 550068d70baSHidetoshi Shimokawa count = scsi_4btoul(rw_16->length); 551068d70baSHidetoshi Shimokawa break; 552068d70baSHidetoshi Shimokawa } 553068d70baSHidetoshi Shimokawa #endif 554068d70baSHidetoshi Shimokawa default: 555068d70baSHidetoshi Shimokawa tcmd_illegal_req(atio, ctio); 556068d70baSHidetoshi Shimokawa return (0); 557c38b150aSNate Lawson } 558c38b150aSNate Lawson if (blkno + count > volume_size) { 559c38b150aSNate Lawson warnx("Attempt to access past end of volume"); 560c38b150aSNate Lawson tcmd_sense(ctio->init_id, ctio, 561c38b150aSNate Lawson SSD_KEY_ILLEGAL_REQUEST, 0x21, 0); 562c38b150aSNate Lawson return (0); 563c38b150aSNate Lawson } 564c38b150aSNate Lawson 565c38b150aSNate Lawson /* Get an (overall) data length and set direction */ 566c38b150aSNate Lawson a_descr->base_off = ((off_t)blkno) * sector_size; 567c38b150aSNate Lawson a_descr->total_len = count * sector_size; 568c38b150aSNate Lawson if (a_descr->total_len == 0) { 569c38b150aSNate Lawson if (debug) 570cd6dedfcSMatt Jacob warnx("r/w 0 blocks @ blkno " OFF_FMT, blkno); 571c38b150aSNate Lawson tcmd_null_ok(atio, ctio); 572c38b150aSNate Lawson return (0); 573c38b150aSNate Lawson } else if (cdb[0] == WRITE_6 || cdb[0] == WRITE_10) { 574c38b150aSNate Lawson a_descr->flags |= CAM_DIR_OUT; 575c38b150aSNate Lawson if (debug) 576cd6dedfcSMatt Jacob warnx("write %u blocks @ blkno " OFF_FMT, count, blkno); 577c38b150aSNate Lawson } else { 578c38b150aSNate Lawson a_descr->flags |= CAM_DIR_IN; 579c38b150aSNate Lawson if (debug) 580cd6dedfcSMatt Jacob warnx("read %u blocks @ blkno " OFF_FMT, count, blkno); 581c38b150aSNate Lawson } 582c38b150aSNate Lawson return (1); 583c38b150aSNate Lawson } 584c38b150aSNate Lawson 585c38b150aSNate Lawson static int 586c38b150aSNate Lawson start_io(struct ccb_accept_tio *atio, struct ccb_scsiio *ctio, int dir) 587c38b150aSNate Lawson { 588c38b150aSNate Lawson struct atio_descr *a_descr; 589c38b150aSNate Lawson struct ctio_descr *c_descr; 590c38b150aSNate Lawson int ret; 591c38b150aSNate Lawson 592c38b150aSNate Lawson /* Set up common structures */ 593c38b150aSNate Lawson a_descr = (struct atio_descr *)atio->ccb_h.targ_descr; 594c38b150aSNate Lawson c_descr = (struct ctio_descr *)ctio->ccb_h.targ_descr; 595c38b150aSNate Lawson 596c38b150aSNate Lawson if (dir == CAM_DIR_IN) { 597c38b150aSNate Lawson c_descr->offset = a_descr->base_off + a_descr->targ_req; 598c38b150aSNate Lawson ctio->dxfer_len = a_descr->total_len - a_descr->targ_req; 599c38b150aSNate Lawson } else { 600c38b150aSNate Lawson c_descr->offset = a_descr->base_off + a_descr->init_req; 601c38b150aSNate Lawson ctio->dxfer_len = a_descr->total_len - a_descr->init_req; 602c38b150aSNate Lawson } 603c38b150aSNate Lawson ctio->dxfer_len = min(ctio->dxfer_len, buf_size); 604c38b150aSNate Lawson assert(ctio->dxfer_len >= 0); 605c38b150aSNate Lawson 606c38b150aSNate Lawson c_descr->aiocb.aio_offset = c_descr->offset; 607c38b150aSNate Lawson c_descr->aiocb.aio_nbytes = ctio->dxfer_len; 608c38b150aSNate Lawson 609c38b150aSNate Lawson /* If DIR_IN, start read from target, otherwise begin CTIO xfer. */ 610c38b150aSNate Lawson ret = 1; 611c38b150aSNate Lawson if (dir == CAM_DIR_IN) { 612cd6dedfcSMatt Jacob if (notaio) { 613cd6dedfcSMatt Jacob if (debug) 61488c707c8SAlexander Motin warnx("read sync %lu @ block " OFF_FMT, 615cd6dedfcSMatt Jacob (unsigned long) 616cd6dedfcSMatt Jacob (ctio->dxfer_len / sector_size), 617cd6dedfcSMatt Jacob c_descr->offset / sector_size); 618cd6dedfcSMatt Jacob if (lseek(c_descr->aiocb.aio_fildes, 619cd6dedfcSMatt Jacob c_descr->aiocb.aio_offset, SEEK_SET) < 0) { 620cd6dedfcSMatt Jacob perror("lseek"); 621cd6dedfcSMatt Jacob err(1, "lseek"); 622cd6dedfcSMatt Jacob } 623cd6dedfcSMatt Jacob if (read(c_descr->aiocb.aio_fildes, 624cd6dedfcSMatt Jacob (void *)c_descr->aiocb.aio_buf, 625cd6dedfcSMatt Jacob ctio->dxfer_len) != ctio->dxfer_len) { 626cd6dedfcSMatt Jacob err(1, "read"); 627cd6dedfcSMatt Jacob } 628cd6dedfcSMatt Jacob } else { 629cd6dedfcSMatt Jacob if (debug) 63088c707c8SAlexander Motin warnx("read async %lu @ block " OFF_FMT, 631cd6dedfcSMatt Jacob (unsigned long) 632cd6dedfcSMatt Jacob (ctio->dxfer_len / sector_size), 633cd6dedfcSMatt Jacob c_descr->offset / sector_size); 634cd6dedfcSMatt Jacob if (aio_read(&c_descr->aiocb) < 0) { 635c38b150aSNate Lawson err(1, "aio_read"); /* XXX */ 636cd6dedfcSMatt Jacob } 637cd6dedfcSMatt Jacob } 638c38b150aSNate Lawson a_descr->targ_req += ctio->dxfer_len; 639cd6dedfcSMatt Jacob /* if we're done, we can mark the CCB as to send status */ 640c38b150aSNate Lawson if (a_descr->targ_req == a_descr->total_len) { 641c38b150aSNate Lawson ctio->ccb_h.flags |= CAM_SEND_STATUS; 642c38b150aSNate Lawson ctio->scsi_status = SCSI_STATUS_OK; 643c38b150aSNate Lawson ret = 0; 644c38b150aSNate Lawson } 645cd6dedfcSMatt Jacob if (notaio) 646cd6dedfcSMatt Jacob tcmd_rdwr_done(atio, ctio, AIO_DONE); 647c38b150aSNate Lawson } else { 648c38b150aSNate Lawson if (a_descr->targ_ack == a_descr->total_len) 649c38b150aSNate Lawson tcmd_null_ok(atio, ctio); 650c38b150aSNate Lawson a_descr->init_req += ctio->dxfer_len; 651c38b150aSNate Lawson if (a_descr->init_req == a_descr->total_len && 652c38b150aSNate Lawson ctio->dxfer_len > 0) { 653c38b150aSNate Lawson /* 654c38b150aSNate Lawson * If data phase done, remove atio from workq. 655c38b150aSNate Lawson * The completion handler will call work_atio to 656c38b150aSNate Lawson * send the final status. 657c38b150aSNate Lawson */ 658c38b150aSNate Lawson ret = 0; 659c38b150aSNate Lawson } 660c38b150aSNate Lawson send_ccb((union ccb *)ctio, /*priority*/1); 661c38b150aSNate Lawson } 662c38b150aSNate Lawson 663c38b150aSNate Lawson return (ret); 664c38b150aSNate Lawson } 665c38b150aSNate Lawson 666c38b150aSNate Lawson static void 667c38b150aSNate Lawson tcmd_rdwr_done(struct ccb_accept_tio *atio, struct ccb_scsiio *ctio, 668c38b150aSNate Lawson io_ops event) 669c38b150aSNate Lawson { 670c38b150aSNate Lawson struct atio_descr *a_descr; 671c38b150aSNate Lawson struct ctio_descr *c_descr; 672c38b150aSNate Lawson 673c38b150aSNate Lawson a_descr = (struct atio_descr *)atio->ccb_h.targ_descr; 674c38b150aSNate Lawson c_descr = (struct ctio_descr *)ctio->ccb_h.targ_descr; 675c38b150aSNate Lawson 676c38b150aSNate Lawson switch (event) { 677c38b150aSNate Lawson case AIO_DONE: 678cd6dedfcSMatt Jacob if (!notaio && aio_return(&c_descr->aiocb) < 0) { 679c38b150aSNate Lawson warn("aio_return error"); 680c38b150aSNate Lawson /* XXX */ 681c38b150aSNate Lawson tcmd_sense(ctio->init_id, ctio, 682c38b150aSNate Lawson SSD_KEY_MEDIUM_ERROR, 0, 0); 683c38b150aSNate Lawson send_ccb((union ccb *)ctio, /*priority*/1); 684c38b150aSNate Lawson break; 685c38b150aSNate Lawson } 686c38b150aSNate Lawson a_descr->targ_ack += ctio->dxfer_len; 687c38b150aSNate Lawson if ((a_descr->flags & CAM_DIR_IN) != 0) { 688cd6dedfcSMatt Jacob if (debug) { 689cd6dedfcSMatt Jacob if (notaio) 690c38b150aSNate Lawson warnx("sending CTIO for AIO read"); 691cd6dedfcSMatt Jacob else 692cd6dedfcSMatt Jacob warnx("sending CTIO for sync read"); 693cd6dedfcSMatt Jacob } 694c38b150aSNate Lawson a_descr->init_req += ctio->dxfer_len; 695c38b150aSNate Lawson send_ccb((union ccb *)ctio, /*priority*/1); 696c38b150aSNate Lawson } else { 697c38b150aSNate Lawson /* Use work function to send final status */ 698c38b150aSNate Lawson if (a_descr->init_req == a_descr->total_len) 699c38b150aSNate Lawson work_atio(atio); 700c38b150aSNate Lawson if (debug) 701c38b150aSNate Lawson warnx("AIO done freeing CTIO"); 702c38b150aSNate Lawson free_ccb((union ccb *)ctio); 703c38b150aSNate Lawson } 704c38b150aSNate Lawson break; 705c38b150aSNate Lawson case CTIO_DONE: 7066070eb3fSMatt Jacob switch (ctio->ccb_h.status & CAM_STATUS_MASK) { 7076070eb3fSMatt Jacob case CAM_REQ_CMP: 7086070eb3fSMatt Jacob break; 7096070eb3fSMatt Jacob case CAM_REQUEUE_REQ: 7106070eb3fSMatt Jacob warnx("requeueing request"); 7116070eb3fSMatt Jacob if ((a_descr->flags & CAM_DIR_MASK) == CAM_DIR_OUT) { 7126070eb3fSMatt Jacob if (aio_write(&c_descr->aiocb) < 0) { 7136070eb3fSMatt Jacob err(1, "aio_write"); /* XXX */ 7146070eb3fSMatt Jacob } 7156070eb3fSMatt Jacob } else { 7166070eb3fSMatt Jacob if (aio_read(&c_descr->aiocb) < 0) { 7176070eb3fSMatt Jacob err(1, "aio_read"); /* XXX */ 7186070eb3fSMatt Jacob } 7196070eb3fSMatt Jacob } 7206070eb3fSMatt Jacob return; 7216070eb3fSMatt Jacob default: 722c38b150aSNate Lawson errx(1, "CTIO failed, status %#x", ctio->ccb_h.status); 723c38b150aSNate Lawson } 724c38b150aSNate Lawson a_descr->init_ack += ctio->dxfer_len; 725c38b150aSNate Lawson if ((a_descr->flags & CAM_DIR_MASK) == CAM_DIR_OUT && 726c38b150aSNate Lawson ctio->dxfer_len > 0) { 727c38b150aSNate Lawson a_descr->targ_req += ctio->dxfer_len; 728cd6dedfcSMatt Jacob if (notaio) { 729cd6dedfcSMatt Jacob if (debug) 73088c707c8SAlexander Motin warnx("write sync %lu @ block " 731cd6dedfcSMatt Jacob OFF_FMT, (unsigned long) 732cd6dedfcSMatt Jacob (ctio->dxfer_len / sector_size), 733cd6dedfcSMatt Jacob c_descr->offset / sector_size); 734cd6dedfcSMatt Jacob if (lseek(c_descr->aiocb.aio_fildes, 735cd6dedfcSMatt Jacob c_descr->aiocb.aio_offset, SEEK_SET) < 0) { 736cd6dedfcSMatt Jacob perror("lseek"); 737cd6dedfcSMatt Jacob err(1, "lseek"); 738cd6dedfcSMatt Jacob } 739cd6dedfcSMatt Jacob if (write(c_descr->aiocb.aio_fildes, 740cd6dedfcSMatt Jacob (void *) c_descr->aiocb.aio_buf, 741cd6dedfcSMatt Jacob ctio->dxfer_len) != ctio->dxfer_len) { 742cd6dedfcSMatt Jacob err(1, "write"); 743cd6dedfcSMatt Jacob } 744cd6dedfcSMatt Jacob tcmd_rdwr_done(atio, ctio, AIO_DONE); 745cd6dedfcSMatt Jacob } else { 746cd6dedfcSMatt Jacob if (debug) 74788c707c8SAlexander Motin warnx("write async %lu @ block " 748cd6dedfcSMatt Jacob OFF_FMT, (unsigned long) 749cd6dedfcSMatt Jacob (ctio->dxfer_len / sector_size), 750cd6dedfcSMatt Jacob c_descr->offset / sector_size); 751cd6dedfcSMatt Jacob if (aio_write(&c_descr->aiocb) < 0) { 752c38b150aSNate Lawson err(1, "aio_write"); /* XXX */ 753cd6dedfcSMatt Jacob } 754cd6dedfcSMatt Jacob } 755c38b150aSNate Lawson } else { 756c38b150aSNate Lawson if (debug) 757c38b150aSNate Lawson warnx("CTIO done freeing CTIO"); 758c38b150aSNate Lawson free_ccb((union ccb *)ctio); 759c38b150aSNate Lawson } 760c38b150aSNate Lawson break; 761c38b150aSNate Lawson default: 762c38b150aSNate Lawson warnx("Unknown completion code %d", event); 763c38b150aSNate Lawson abort(); 764c38b150aSNate Lawson /* NOTREACHED */ 765c38b150aSNate Lawson } 766c38b150aSNate Lawson } 767c38b150aSNate Lawson 768c38b150aSNate Lawson /* Simple ok message used by TUR, SYNC_CACHE, etc. */ 769c38b150aSNate Lawson static int 770c38b150aSNate Lawson tcmd_null_ok(struct ccb_accept_tio *atio, struct ccb_scsiio *ctio) 771c38b150aSNate Lawson { 772c38b150aSNate Lawson if (debug) { 773c38b150aSNate Lawson struct atio_descr *a_descr; 774c38b150aSNate Lawson 775c38b150aSNate Lawson a_descr = (struct atio_descr *)atio->ccb_h.targ_descr; 776c38b150aSNate Lawson cdb_debug(a_descr->cdb, "Sending null ok to %u : ", atio->init_id); 777c38b150aSNate Lawson } 778c38b150aSNate Lawson 779c38b150aSNate Lawson ctio->dxfer_len = 0; 780c38b150aSNate Lawson ctio->ccb_h.flags &= ~CAM_DIR_MASK; 781c38b150aSNate Lawson ctio->ccb_h.flags |= CAM_DIR_NONE | CAM_SEND_STATUS; 782c38b150aSNate Lawson ctio->scsi_status = SCSI_STATUS_OK; 783c38b150aSNate Lawson return (0); 784c38b150aSNate Lawson } 785c38b150aSNate Lawson 786c38b150aSNate Lawson /* Simple illegal request message used by MODE SENSE, etc. */ 787c38b150aSNate Lawson static int 788c38b150aSNate Lawson tcmd_illegal_req(struct ccb_accept_tio *atio, struct ccb_scsiio *ctio) 789c38b150aSNate Lawson { 790c38b150aSNate Lawson if (debug) { 791c38b150aSNate Lawson struct atio_descr *a_descr; 792c38b150aSNate Lawson 793c38b150aSNate Lawson a_descr = (struct atio_descr *)atio->ccb_h.targ_descr; 794c38b150aSNate Lawson cdb_debug(a_descr->cdb, "Sending ill req to %u: ", atio->init_id); 795c38b150aSNate Lawson } 796c38b150aSNate Lawson 797c38b150aSNate Lawson tcmd_sense(atio->init_id, ctio, SSD_KEY_ILLEGAL_REQUEST, 798c38b150aSNate Lawson /*asc*/0x24, /*ascq*/0); 799c38b150aSNate Lawson return (0); 800c38b150aSNate Lawson } 801c38b150aSNate Lawson 802c38b150aSNate Lawson static void 803c38b150aSNate Lawson cdb_debug(u_int8_t *cdb, const char *msg, ...) 804c38b150aSNate Lawson { 805c38b150aSNate Lawson char msg_buf[512]; 806c38b150aSNate Lawson int len; 807c38b150aSNate Lawson va_list ap; 808c38b150aSNate Lawson 809c38b150aSNate Lawson va_start(ap, msg); 810c38b150aSNate Lawson vsnprintf(msg_buf, sizeof(msg_buf), msg, ap); 811c38b150aSNate Lawson va_end(ap); 812c38b150aSNate Lawson len = strlen(msg_buf); 813c38b150aSNate Lawson scsi_cdb_string(cdb, msg_buf + len, sizeof(msg_buf) - len); 814c38b150aSNate Lawson warnx("%s", msg_buf); 815c38b150aSNate Lawson } 816