11eba4c79SScott Long /*-
24d846d26SWarner Losh * SPDX-License-Identifier: BSD-2-Clause
3bec9534dSPedro F. Giffuni *
41eba4c79SScott Long * Copyright (c) 2007 Scott Long
51eba4c79SScott Long * All rights reserved.
61eba4c79SScott Long *
71eba4c79SScott Long * Redistribution and use in source and binary forms, with or without
81eba4c79SScott Long * modification, are permitted provided that the following conditions
91eba4c79SScott Long * are met:
101eba4c79SScott Long * 1. Redistributions of source code must retain the above copyright
111eba4c79SScott Long * notice, this list of conditions, and the following disclaimer,
121eba4c79SScott Long * without modification, immediately at the beginning of the file.
131eba4c79SScott Long * 2. The name of the author may not be used to endorse or promote products
141eba4c79SScott Long * derived from this software without specific prior written permission.
151eba4c79SScott Long *
161eba4c79SScott Long * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
171eba4c79SScott Long * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
181eba4c79SScott Long * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
191eba4c79SScott Long * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE FOR
201eba4c79SScott Long * ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
211eba4c79SScott Long * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
221eba4c79SScott Long * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
231eba4c79SScott Long * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
241eba4c79SScott Long * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
251eba4c79SScott Long * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
261eba4c79SScott Long * SUCH DAMAGE.
271eba4c79SScott Long */
281eba4c79SScott Long
291eba4c79SScott Long /*
301eba4c79SScott Long * scsi_sg peripheral driver. This driver is meant to implement the Linux
311eba4c79SScott Long * SG passthrough interface for SCSI.
321eba4c79SScott Long */
331eba4c79SScott Long
341eba4c79SScott Long #include <sys/param.h>
351eba4c79SScott Long #include <sys/systm.h>
361eba4c79SScott Long #include <sys/kernel.h>
371eba4c79SScott Long #include <sys/types.h>
381eba4c79SScott Long #include <sys/bio.h>
391eba4c79SScott Long #include <sys/malloc.h>
401eba4c79SScott Long #include <sys/fcntl.h>
411eba4c79SScott Long #include <sys/ioccom.h>
421eba4c79SScott Long #include <sys/conf.h>
431eba4c79SScott Long #include <sys/errno.h>
441eba4c79SScott Long #include <sys/devicestat.h>
451eba4c79SScott Long #include <sys/proc.h>
461eba4c79SScott Long #include <sys/uio.h>
471eba4c79SScott Long
481eba4c79SScott Long #include <cam/cam.h>
491eba4c79SScott Long #include <cam/cam_ccb.h>
501eba4c79SScott Long #include <cam/cam_periph.h>
511eba4c79SScott Long #include <cam/cam_queue.h>
521eba4c79SScott Long #include <cam/cam_xpt_periph.h>
531eba4c79SScott Long #include <cam/cam_debug.h>
541eba4c79SScott Long #include <cam/cam_sim.h>
551eba4c79SScott Long
561eba4c79SScott Long #include <cam/scsi/scsi_all.h>
571eba4c79SScott Long #include <cam/scsi/scsi_message.h>
581eba4c79SScott Long #include <cam/scsi/scsi_sg.h>
591eba4c79SScott Long
601eba4c79SScott Long typedef enum {
61c552ebe1SKenneth D. Merry SG_FLAG_LOCKED = 0x01,
62c552ebe1SKenneth D. Merry SG_FLAG_INVALID = 0x02
631eba4c79SScott Long } sg_flags;
641eba4c79SScott Long
651eba4c79SScott Long typedef enum {
661eba4c79SScott Long SG_STATE_NORMAL
671eba4c79SScott Long } sg_state;
681eba4c79SScott Long
691eba4c79SScott Long typedef enum {
70472cdbefSScott Long SG_RDWR_FREE,
711eba4c79SScott Long SG_RDWR_INPROG,
721eba4c79SScott Long SG_RDWR_DONE
731eba4c79SScott Long } sg_rdwr_state;
741eba4c79SScott Long
751eba4c79SScott Long typedef enum {
76227d67aaSAlexander Motin SG_CCB_RDWR_IO
771eba4c79SScott Long } sg_ccb_types;
781eba4c79SScott Long
791eba4c79SScott Long #define ccb_type ppriv_field0
801eba4c79SScott Long #define ccb_rdwr ppriv_ptr1
811eba4c79SScott Long
821eba4c79SScott Long struct sg_rdwr {
831eba4c79SScott Long TAILQ_ENTRY(sg_rdwr) rdwr_link;
841eba4c79SScott Long int tag;
851eba4c79SScott Long int state;
861eba4c79SScott Long int buf_len;
871eba4c79SScott Long char *buf;
881eba4c79SScott Long union ccb *ccb;
891eba4c79SScott Long union {
901eba4c79SScott Long struct sg_header hdr;
911eba4c79SScott Long struct sg_io_hdr io_hdr;
921eba4c79SScott Long } hdr;
931eba4c79SScott Long };
941eba4c79SScott Long
951eba4c79SScott Long struct sg_softc {
961eba4c79SScott Long sg_state state;
971eba4c79SScott Long sg_flags flags;
9886d45c7fSKenneth D. Merry int open_count;
99de239312SAlexander Motin u_int maxio;
1001eba4c79SScott Long struct devstat *device_stats;
1011eba4c79SScott Long TAILQ_HEAD(, sg_rdwr) rdwr_done;
1021eba4c79SScott Long struct cdev *dev;
103715ab212SScott Long int sg_timeout;
104715ab212SScott Long int sg_user_timeout;
105715ab212SScott Long uint8_t pd_type;
1061eba4c79SScott Long };
1071eba4c79SScott Long
1081eba4c79SScott Long static d_open_t sgopen;
1091eba4c79SScott Long static d_close_t sgclose;
1101eba4c79SScott Long static d_ioctl_t sgioctl;
1111eba4c79SScott Long static d_write_t sgwrite;
1121eba4c79SScott Long static d_read_t sgread;
1131eba4c79SScott Long
1141eba4c79SScott Long static periph_init_t sginit;
1151eba4c79SScott Long static periph_ctor_t sgregister;
1161eba4c79SScott Long static periph_oninv_t sgoninvalidate;
1171eba4c79SScott Long static periph_dtor_t sgcleanup;
1181eba4c79SScott Long static void sgasync(void *callback_arg, uint32_t code,
1191eba4c79SScott Long struct cam_path *path, void *arg);
1201eba4c79SScott Long static void sgdone(struct cam_periph *periph, union ccb *done_ccb);
1211eba4c79SScott Long static int sgsendccb(struct cam_periph *periph, union ccb *ccb);
1221eba4c79SScott Long static int sgsendrdwr(struct cam_periph *periph, union ccb *ccb);
1231eba4c79SScott Long static int sgerror(union ccb *ccb, uint32_t cam_flags,
1241eba4c79SScott Long uint32_t sense_flags);
1251eba4c79SScott Long static void sg_scsiio_status(struct ccb_scsiio *csio,
1261eba4c79SScott Long u_short *hoststat, u_short *drvstat);
1271eba4c79SScott Long
1281eba4c79SScott Long static int scsi_group_len(u_char cmd);
1291eba4c79SScott Long
1301eba4c79SScott Long static struct periph_driver sgdriver =
1311eba4c79SScott Long {
1321eba4c79SScott Long sginit, "sg",
1331eba4c79SScott Long TAILQ_HEAD_INITIALIZER(sgdriver.units), /* gen */ 0
1341eba4c79SScott Long };
1351eba4c79SScott Long PERIPHDRIVER_DECLARE(sg, sgdriver);
1361eba4c79SScott Long
1371eba4c79SScott Long static struct cdevsw sg_cdevsw = {
1381eba4c79SScott Long .d_version = D_VERSION,
139f0d6f577SScott Long .d_flags = D_TRACKCLOSE,
1401eba4c79SScott Long .d_open = sgopen,
1411eba4c79SScott Long .d_close = sgclose,
1421eba4c79SScott Long .d_ioctl = sgioctl,
1431eba4c79SScott Long .d_write = sgwrite,
1441eba4c79SScott Long .d_read = sgread,
1451eba4c79SScott Long .d_name = "sg",
1461eba4c79SScott Long };
1471eba4c79SScott Long
1481eba4c79SScott Long static int sg_version = 30125;
1491eba4c79SScott Long
1501eba4c79SScott Long static void
sginit(void)1511eba4c79SScott Long sginit(void)
1521eba4c79SScott Long {
1531eba4c79SScott Long cam_status status;
1541eba4c79SScott Long
1551eba4c79SScott Long /*
1561eba4c79SScott Long * Install a global async callback. This callback will receive aync
1571eba4c79SScott Long * callbacks like "new device found".
1581eba4c79SScott Long */
15985d92640SScott Long status = xpt_register_async(AC_FOUND_DEVICE, sgasync, NULL, NULL);
1601eba4c79SScott Long
1611eba4c79SScott Long if (status != CAM_REQ_CMP) {
1621eba4c79SScott Long printf("sg: Failed to attach master async callbac "
1631eba4c79SScott Long "due to status 0x%x!\n", status);
1641eba4c79SScott Long }
1651eba4c79SScott Long }
1661eba4c79SScott Long
1671eba4c79SScott Long static void
sgdevgonecb(void * arg)16886d45c7fSKenneth D. Merry sgdevgonecb(void *arg)
16986d45c7fSKenneth D. Merry {
17086d45c7fSKenneth D. Merry struct cam_periph *periph;
17186d45c7fSKenneth D. Merry struct sg_softc *softc;
172227d67aaSAlexander Motin struct mtx *mtx;
17386d45c7fSKenneth D. Merry int i;
17486d45c7fSKenneth D. Merry
17586d45c7fSKenneth D. Merry periph = (struct cam_periph *)arg;
176227d67aaSAlexander Motin mtx = cam_periph_mtx(periph);
177227d67aaSAlexander Motin mtx_lock(mtx);
17886d45c7fSKenneth D. Merry
179227d67aaSAlexander Motin softc = (struct sg_softc *)periph->softc;
18086d45c7fSKenneth D. Merry KASSERT(softc->open_count >= 0, ("Negative open count %d",
18186d45c7fSKenneth D. Merry softc->open_count));
18286d45c7fSKenneth D. Merry
18386d45c7fSKenneth D. Merry /*
18486d45c7fSKenneth D. Merry * When we get this callback, we will get no more close calls from
18586d45c7fSKenneth D. Merry * devfs. So if we have any dangling opens, we need to release the
18686d45c7fSKenneth D. Merry * reference held for that particular context.
18786d45c7fSKenneth D. Merry */
18886d45c7fSKenneth D. Merry for (i = 0; i < softc->open_count; i++)
18986d45c7fSKenneth D. Merry cam_periph_release_locked(periph);
19086d45c7fSKenneth D. Merry
19186d45c7fSKenneth D. Merry softc->open_count = 0;
19286d45c7fSKenneth D. Merry
19386d45c7fSKenneth D. Merry /*
19486d45c7fSKenneth D. Merry * Release the reference held for the device node, it is gone now.
19586d45c7fSKenneth D. Merry */
19686d45c7fSKenneth D. Merry cam_periph_release_locked(periph);
19786d45c7fSKenneth D. Merry
19886d45c7fSKenneth D. Merry /*
199227d67aaSAlexander Motin * We reference the lock directly here, instead of using
20086d45c7fSKenneth D. Merry * cam_periph_unlock(). The reason is that the final call to
20186d45c7fSKenneth D. Merry * cam_periph_release_locked() above could result in the periph
20286d45c7fSKenneth D. Merry * getting freed. If that is the case, dereferencing the periph
20386d45c7fSKenneth D. Merry * with a cam_periph_unlock() call would cause a page fault.
20486d45c7fSKenneth D. Merry */
205227d67aaSAlexander Motin mtx_unlock(mtx);
20686d45c7fSKenneth D. Merry }
20786d45c7fSKenneth D. Merry
20886d45c7fSKenneth D. Merry static void
sgoninvalidate(struct cam_periph * periph)2091eba4c79SScott Long sgoninvalidate(struct cam_periph *periph)
2101eba4c79SScott Long {
2111eba4c79SScott Long struct sg_softc *softc;
2121eba4c79SScott Long
2131eba4c79SScott Long softc = (struct sg_softc *)periph->softc;
2141eba4c79SScott Long
2151eba4c79SScott Long /*
2161eba4c79SScott Long * Deregister any async callbacks.
2171eba4c79SScott Long */
21885d92640SScott Long xpt_register_async(0, sgasync, periph, periph->path);
2191eba4c79SScott Long
2201eba4c79SScott Long softc->flags |= SG_FLAG_INVALID;
2211eba4c79SScott Long
2221eba4c79SScott Long /*
22386d45c7fSKenneth D. Merry * Tell devfs this device has gone away, and ask for a callback
22486d45c7fSKenneth D. Merry * when it has cleaned up its state.
22586d45c7fSKenneth D. Merry */
22686d45c7fSKenneth D. Merry destroy_dev_sched_cb(softc->dev, sgdevgonecb, periph);
22786d45c7fSKenneth D. Merry
22886d45c7fSKenneth D. Merry /*
2291eba4c79SScott Long * XXX Return all queued I/O with ENXIO.
2301eba4c79SScott Long * XXX Handle any transactions queued to the card
2311eba4c79SScott Long * with XPT_ABORT_CCB.
2321eba4c79SScott Long */
2331eba4c79SScott Long
2341eba4c79SScott Long }
2351eba4c79SScott Long
2361eba4c79SScott Long static void
sgcleanup(struct cam_periph * periph)2371eba4c79SScott Long sgcleanup(struct cam_periph *periph)
2381eba4c79SScott Long {
2391eba4c79SScott Long struct sg_softc *softc;
2401eba4c79SScott Long
2411eba4c79SScott Long softc = (struct sg_softc *)periph->softc;
24286d45c7fSKenneth D. Merry
2435f3fed85SEdward Tomasz Napierala devstat_remove_entry(softc->device_stats);
24486d45c7fSKenneth D. Merry
2451eba4c79SScott Long free(softc, M_DEVBUF);
2461eba4c79SScott Long }
2471eba4c79SScott Long
2481eba4c79SScott Long static void
sgasync(void * callback_arg,uint32_t code,struct cam_path * path,void * arg)2491eba4c79SScott Long sgasync(void *callback_arg, uint32_t code, struct cam_path *path, void *arg)
2501eba4c79SScott Long {
2511eba4c79SScott Long struct cam_periph *periph;
2521eba4c79SScott Long
2531eba4c79SScott Long periph = (struct cam_periph *)callback_arg;
2541eba4c79SScott Long
2551eba4c79SScott Long switch (code) {
2561eba4c79SScott Long case AC_FOUND_DEVICE:
2571eba4c79SScott Long {
2581eba4c79SScott Long struct ccb_getdev *cgd;
2591eba4c79SScott Long cam_status status;
2601eba4c79SScott Long
2611eba4c79SScott Long cgd = (struct ccb_getdev *)arg;
2621eba4c79SScott Long if (cgd == NULL)
2631eba4c79SScott Long break;
2641eba4c79SScott Long
26552c9ce25SScott Long if (cgd->protocol != PROTO_SCSI)
26652c9ce25SScott Long break;
26752c9ce25SScott Long
2681eba4c79SScott Long /*
2691eba4c79SScott Long * Allocate a peripheral instance for this device and
2701eba4c79SScott Long * start the probe process.
2711eba4c79SScott Long */
2721eba4c79SScott Long status = cam_periph_alloc(sgregister, sgoninvalidate,
273227d67aaSAlexander Motin sgcleanup, NULL, "sg",
274227d67aaSAlexander Motin CAM_PERIPH_BIO, path,
2751eba4c79SScott Long sgasync, AC_FOUND_DEVICE, cgd);
2761eba4c79SScott Long if ((status != CAM_REQ_CMP) && (status != CAM_REQ_INPROG)) {
2771eba4c79SScott Long const struct cam_status_entry *entry;
2781eba4c79SScott Long
2791eba4c79SScott Long entry = cam_fetch_status_entry(status);
2801eba4c79SScott Long printf("sgasync: Unable to attach new device "
2811eba4c79SScott Long "due to status %#x: %s\n", status, entry ?
2821eba4c79SScott Long entry->status_text : "Unknown");
2831eba4c79SScott Long }
2841eba4c79SScott Long break;
2851eba4c79SScott Long }
2861eba4c79SScott Long default:
2871eba4c79SScott Long cam_periph_async(periph, code, path, arg);
2881eba4c79SScott Long break;
2891eba4c79SScott Long }
2901eba4c79SScott Long }
2911eba4c79SScott Long
2921eba4c79SScott Long static cam_status
sgregister(struct cam_periph * periph,void * arg)2931eba4c79SScott Long sgregister(struct cam_periph *periph, void *arg)
2941eba4c79SScott Long {
2951eba4c79SScott Long struct sg_softc *softc;
2961eba4c79SScott Long struct ccb_getdev *cgd;
297b8b6b5d3SAlexander Motin struct ccb_pathinq cpi;
298ee198893SKonstantin Belousov struct make_dev_args args;
299ee198893SKonstantin Belousov int no_tags, error;
3001eba4c79SScott Long
3011eba4c79SScott Long cgd = (struct ccb_getdev *)arg;
3021eba4c79SScott Long if (cgd == NULL) {
3031eba4c79SScott Long printf("sgregister: no getdev CCB, can't register device\n");
3041eba4c79SScott Long return (CAM_REQ_CMP_ERR);
3051eba4c79SScott Long }
3061eba4c79SScott Long
3074400b36dSScott Long softc = malloc(sizeof(*softc), M_DEVBUF, M_ZERO | M_NOWAIT);
3081eba4c79SScott Long if (softc == NULL) {
3091eba4c79SScott Long printf("sgregister: Unable to allocate softc\n");
3101eba4c79SScott Long return (CAM_REQ_CMP_ERR);
3111eba4c79SScott Long }
3121eba4c79SScott Long
3131eba4c79SScott Long softc->state = SG_STATE_NORMAL;
3141eba4c79SScott Long softc->pd_type = SID_TYPE(&cgd->inq_data);
315715ab212SScott Long softc->sg_timeout = SG_DEFAULT_TIMEOUT / SG_DEFAULT_HZ * hz;
316715ab212SScott Long softc->sg_user_timeout = SG_DEFAULT_TIMEOUT;
3171eba4c79SScott Long TAILQ_INIT(&softc->rdwr_done);
3181eba4c79SScott Long periph->softc = softc;
3191eba4c79SScott Long
320762a7f4fSWarner Losh xpt_path_inq(&cpi, periph->path);
321b8b6b5d3SAlexander Motin
322de239312SAlexander Motin if (cpi.maxio == 0)
323de239312SAlexander Motin softc->maxio = DFLTPHYS; /* traditional default */
324cd853791SKonstantin Belousov else if (cpi.maxio > maxphys)
325cd853791SKonstantin Belousov softc->maxio = maxphys; /* for safety */
326de239312SAlexander Motin else
327de239312SAlexander Motin softc->maxio = cpi.maxio; /* real value */
328de239312SAlexander Motin
3291eba4c79SScott Long /*
3301eba4c79SScott Long * We pass in 0 for all blocksize, since we don't know what the
3311eba4c79SScott Long * blocksize of the device is, if it even has a blocksize.
3321eba4c79SScott Long */
33385d92640SScott Long cam_periph_unlock(periph);
3341eba4c79SScott Long no_tags = (cgd->inq_data.flags & SID_CmdQue) == 0;
3351eba4c79SScott Long softc->device_stats = devstat_new_entry("sg",
336d3ce8327SEd Schouten periph->unit_number, 0,
3371eba4c79SScott Long DEVSTAT_NO_BLOCKSIZE
3381eba4c79SScott Long | (no_tags ? DEVSTAT_NO_ORDERED_TAGS : 0),
3391eba4c79SScott Long softc->pd_type |
340b8b6b5d3SAlexander Motin XPORT_DEVSTAT_TYPE(cpi.transport) |
3411eba4c79SScott Long DEVSTAT_TYPE_PASS,
3421eba4c79SScott Long DEVSTAT_PRIORITY_PASS);
3431eba4c79SScott Long
34486d45c7fSKenneth D. Merry /*
34586d45c7fSKenneth D. Merry * Acquire a reference to the periph before we create the devfs
34686d45c7fSKenneth D. Merry * instance for it. We'll release this reference once the devfs
34786d45c7fSKenneth D. Merry * instance has been freed.
34886d45c7fSKenneth D. Merry */
34999e7a4adSScott Long if (cam_periph_acquire(periph) != 0) {
35086d45c7fSKenneth D. Merry xpt_print(periph->path, "%s: lost periph during "
35186d45c7fSKenneth D. Merry "registration!\n", __func__);
35286d45c7fSKenneth D. Merry cam_periph_lock(periph);
35386d45c7fSKenneth D. Merry return (CAM_REQ_CMP_ERR);
35486d45c7fSKenneth D. Merry }
35586d45c7fSKenneth D. Merry
3561eba4c79SScott Long /* Register the device */
357ee198893SKonstantin Belousov make_dev_args_init(&args);
358ee198893SKonstantin Belousov args.mda_devsw = &sg_cdevsw;
359ee198893SKonstantin Belousov args.mda_unit = periph->unit_number;
360ee198893SKonstantin Belousov args.mda_uid = UID_ROOT;
361ee198893SKonstantin Belousov args.mda_gid = GID_OPERATOR;
362ee198893SKonstantin Belousov args.mda_mode = 0600;
363ee198893SKonstantin Belousov args.mda_si_drv1 = periph;
364ee198893SKonstantin Belousov error = make_dev_s(&args, &softc->dev, "%s%d",
3651eba4c79SScott Long periph->periph_name, periph->unit_number);
366ee198893SKonstantin Belousov if (error != 0) {
367ee198893SKonstantin Belousov cam_periph_lock(periph);
368ee198893SKonstantin Belousov cam_periph_release_locked(periph);
369ee198893SKonstantin Belousov return (CAM_REQ_CMP_ERR);
370ee198893SKonstantin Belousov }
371cf454e30SMatt Jacob if (periph->unit_number < 26) {
372c59b4dcdSMatt Jacob (void)make_dev_alias(softc->dev, "sg%c",
373c59b4dcdSMatt Jacob periph->unit_number + 'a');
374cf454e30SMatt Jacob } else {
375cf454e30SMatt Jacob (void)make_dev_alias(softc->dev, "sg%c%c",
376c59b4dcdSMatt Jacob ((periph->unit_number / 26) - 1) + 'a',
377c59b4dcdSMatt Jacob (periph->unit_number % 26) + 'a');
378cf454e30SMatt Jacob }
3792b83592fSScott Long cam_periph_lock(periph);
3801eba4c79SScott Long
3811eba4c79SScott Long /*
3821eba4c79SScott Long * Add as async callback so that we get
3831eba4c79SScott Long * notified if this device goes away.
3841eba4c79SScott Long */
38585d92640SScott Long xpt_register_async(AC_LOST_DEVICE, sgasync, periph, periph->path);
3861eba4c79SScott Long
3871eba4c79SScott Long if (bootverbose)
3881eba4c79SScott Long xpt_announce_periph(periph, NULL);
3891eba4c79SScott Long
3901eba4c79SScott Long return (CAM_REQ_CMP);
3911eba4c79SScott Long }
3921eba4c79SScott Long
3931eba4c79SScott Long static void
sgdone(struct cam_periph * periph,union ccb * done_ccb)3941eba4c79SScott Long sgdone(struct cam_periph *periph, union ccb *done_ccb)
3951eba4c79SScott Long {
3961eba4c79SScott Long struct sg_softc *softc;
3971eba4c79SScott Long struct ccb_scsiio *csio;
3981eba4c79SScott Long
3991eba4c79SScott Long softc = (struct sg_softc *)periph->softc;
4001eba4c79SScott Long csio = &done_ccb->csio;
4011eba4c79SScott Long switch (csio->ccb_h.ccb_type) {
4021eba4c79SScott Long case SG_CCB_RDWR_IO:
4031eba4c79SScott Long {
4041eba4c79SScott Long struct sg_rdwr *rdwr;
4051eba4c79SScott Long
4061eba4c79SScott Long devstat_end_transaction(softc->device_stats,
4071eba4c79SScott Long csio->dxfer_len,
4081eba4c79SScott Long csio->tag_action & 0xf,
4091eba4c79SScott Long ((csio->ccb_h.flags & CAM_DIR_MASK) ==
4101eba4c79SScott Long CAM_DIR_NONE) ? DEVSTAT_NO_DATA :
4111eba4c79SScott Long (csio->ccb_h.flags & CAM_DIR_OUT) ?
4121eba4c79SScott Long DEVSTAT_WRITE : DEVSTAT_READ,
4131eba4c79SScott Long NULL, NULL);
4141eba4c79SScott Long
4151eba4c79SScott Long rdwr = done_ccb->ccb_h.ccb_rdwr;
4161eba4c79SScott Long rdwr->state = SG_RDWR_DONE;
4171eba4c79SScott Long wakeup(rdwr);
4181eba4c79SScott Long break;
4191eba4c79SScott Long }
4201eba4c79SScott Long default:
4211eba4c79SScott Long panic("unknown sg CCB type");
4221eba4c79SScott Long }
4231eba4c79SScott Long }
4241eba4c79SScott Long
4251eba4c79SScott Long static int
sgopen(struct cdev * dev,int flags,int fmt,struct thread * td)4261eba4c79SScott Long sgopen(struct cdev *dev, int flags, int fmt, struct thread *td)
4271eba4c79SScott Long {
4281eba4c79SScott Long struct cam_periph *periph;
4291eba4c79SScott Long struct sg_softc *softc;
4301eba4c79SScott Long int error = 0;
4311eba4c79SScott Long
4321eba4c79SScott Long periph = (struct cam_periph *)dev->si_drv1;
43399e7a4adSScott Long if (cam_periph_acquire(periph) != 0)
4348900f4b8SKenneth D. Merry return (ENXIO);
4358900f4b8SKenneth D. Merry
4361eba4c79SScott Long /*
4371eba4c79SScott Long * Don't allow access when we're running at a high securelevel.
4381eba4c79SScott Long */
4391eba4c79SScott Long error = securelevel_gt(td->td_ucred, 1);
4408900f4b8SKenneth D. Merry if (error) {
4418900f4b8SKenneth D. Merry cam_periph_release(periph);
4421eba4c79SScott Long return (error);
4438900f4b8SKenneth D. Merry }
4441eba4c79SScott Long
4452b83592fSScott Long cam_periph_lock(periph);
4462b83592fSScott Long
4472b83592fSScott Long softc = (struct sg_softc *)periph->softc;
4482b83592fSScott Long if (softc->flags & SG_FLAG_INVALID) {
449c552ebe1SKenneth D. Merry cam_periph_release_locked(periph);
4502b83592fSScott Long cam_periph_unlock(periph);
4512b83592fSScott Long return (ENXIO);
4522b83592fSScott Long }
4531eba4c79SScott Long
45486d45c7fSKenneth D. Merry softc->open_count++;
45586d45c7fSKenneth D. Merry
456835187bfSScott Long cam_periph_unlock(periph);
4571eba4c79SScott Long
4581eba4c79SScott Long return (error);
4591eba4c79SScott Long }
4601eba4c79SScott Long
4611eba4c79SScott Long static int
sgclose(struct cdev * dev,int flag,int fmt,struct thread * td)4621eba4c79SScott Long sgclose(struct cdev *dev, int flag, int fmt, struct thread *td)
4631eba4c79SScott Long {
4641eba4c79SScott Long struct cam_periph *periph;
46586d45c7fSKenneth D. Merry struct sg_softc *softc;
466227d67aaSAlexander Motin struct mtx *mtx;
4671eba4c79SScott Long
4681eba4c79SScott Long periph = (struct cam_periph *)dev->si_drv1;
469227d67aaSAlexander Motin mtx = cam_periph_mtx(periph);
470227d67aaSAlexander Motin mtx_lock(mtx);
4711eba4c79SScott Long
47286d45c7fSKenneth D. Merry softc = periph->softc;
47386d45c7fSKenneth D. Merry softc->open_count--;
47486d45c7fSKenneth D. Merry
47586d45c7fSKenneth D. Merry cam_periph_release_locked(periph);
47686d45c7fSKenneth D. Merry
47786d45c7fSKenneth D. Merry /*
478227d67aaSAlexander Motin * We reference the lock directly here, instead of using
47986d45c7fSKenneth D. Merry * cam_periph_unlock(). The reason is that the call to
48086d45c7fSKenneth D. Merry * cam_periph_release_locked() above could result in the periph
48186d45c7fSKenneth D. Merry * getting freed. If that is the case, dereferencing the periph
48286d45c7fSKenneth D. Merry * with a cam_periph_unlock() call would cause a page fault.
48386d45c7fSKenneth D. Merry *
48486d45c7fSKenneth D. Merry * cam_periph_release() avoids this problem using the same method,
48586d45c7fSKenneth D. Merry * but we're manually acquiring and dropping the lock here to
48686d45c7fSKenneth D. Merry * protect the open count and avoid another lock acquisition and
48786d45c7fSKenneth D. Merry * release.
48886d45c7fSKenneth D. Merry */
489227d67aaSAlexander Motin mtx_unlock(mtx);
4901eba4c79SScott Long
4911eba4c79SScott Long return (0);
4921eba4c79SScott Long }
4931eba4c79SScott Long
4941eba4c79SScott Long static int
sgioctl(struct cdev * dev,u_long cmd,caddr_t arg,int flag,struct thread * td)4951eba4c79SScott Long sgioctl(struct cdev *dev, u_long cmd, caddr_t arg, int flag, struct thread *td)
4961eba4c79SScott Long {
4971eba4c79SScott Long union ccb *ccb;
4981eba4c79SScott Long struct ccb_scsiio *csio;
4991eba4c79SScott Long struct cam_periph *periph;
5001eba4c79SScott Long struct sg_softc *softc;
501fcaf473cSAlexander Motin struct sg_io_hdr *req;
5021eba4c79SScott Long int dir, error;
5031eba4c79SScott Long
5041eba4c79SScott Long periph = (struct cam_periph *)dev->si_drv1;
5052b83592fSScott Long cam_periph_lock(periph);
5062b83592fSScott Long
5071eba4c79SScott Long softc = (struct sg_softc *)periph->softc;
5081eba4c79SScott Long error = 0;
5091eba4c79SScott Long
5101eba4c79SScott Long switch (cmd) {
5111eba4c79SScott Long case SG_GET_VERSION_NUM:
512fcaf473cSAlexander Motin {
513fcaf473cSAlexander Motin int *version = (int *)arg;
514715ab212SScott Long
515fcaf473cSAlexander Motin *version = sg_version;
516fcaf473cSAlexander Motin break;
517fcaf473cSAlexander Motin }
518fcaf473cSAlexander Motin case SG_SET_TIMEOUT:
519fcaf473cSAlexander Motin {
520fcaf473cSAlexander Motin u_int user_timeout = *(u_int *)arg;
521fcaf473cSAlexander Motin
522715ab212SScott Long softc->sg_user_timeout = user_timeout;
523715ab212SScott Long softc->sg_timeout = user_timeout / SG_DEFAULT_HZ * hz;
5241eba4c79SScott Long break;
525715ab212SScott Long }
5261eba4c79SScott Long case SG_GET_TIMEOUT:
5271eba4c79SScott Long /*
528715ab212SScott Long * The value is returned directly to the syscall.
5291eba4c79SScott Long */
530715ab212SScott Long td->td_retval[0] = softc->sg_user_timeout;
5311eba4c79SScott Long error = 0;
5321eba4c79SScott Long break;
5331eba4c79SScott Long case SG_IO:
534fcaf473cSAlexander Motin req = (struct sg_io_hdr *)arg;
5351eba4c79SScott Long
536fcaf473cSAlexander Motin if (req->cmd_len > IOCDBLEN) {
5371eba4c79SScott Long error = EINVAL;
5381eba4c79SScott Long break;
5391eba4c79SScott Long }
5401eba4c79SScott Long
541fcaf473cSAlexander Motin if (req->iovec_count != 0) {
5421eba4c79SScott Long error = EOPNOTSUPP;
5431eba4c79SScott Long break;
5441eba4c79SScott Long }
5451eba4c79SScott Long
5461e637ba6SAlexander Motin ccb = cam_periph_getccb(periph, CAM_PRIORITY_NORMAL);
5471eba4c79SScott Long csio = &ccb->csio;
5481eba4c79SScott Long
549fcaf473cSAlexander Motin error = copyin(req->cmdp, &csio->cdb_io.cdb_bytes,
550fcaf473cSAlexander Motin req->cmd_len);
5511eba4c79SScott Long if (error) {
5521eba4c79SScott Long xpt_release_ccb(ccb);
5531eba4c79SScott Long break;
5541eba4c79SScott Long }
5551eba4c79SScott Long
556fcaf473cSAlexander Motin switch(req->dxfer_direction) {
5571eba4c79SScott Long case SG_DXFER_TO_DEV:
5581eba4c79SScott Long dir = CAM_DIR_OUT;
5591eba4c79SScott Long break;
5601eba4c79SScott Long case SG_DXFER_FROM_DEV:
5611eba4c79SScott Long dir = CAM_DIR_IN;
5621eba4c79SScott Long break;
5631eba4c79SScott Long case SG_DXFER_TO_FROM_DEV:
56481b62a76SJohn Baldwin dir = CAM_DIR_BOTH;
5651eba4c79SScott Long break;
5661eba4c79SScott Long case SG_DXFER_NONE:
5671eba4c79SScott Long default:
5681eba4c79SScott Long dir = CAM_DIR_NONE;
5691eba4c79SScott Long break;
5701eba4c79SScott Long }
5711eba4c79SScott Long
5721eba4c79SScott Long cam_fill_csio(csio,
5731eba4c79SScott Long /*retries*/1,
574eed99e75SScott Long /*cbfcnp*/NULL,
5751eba4c79SScott Long dir|CAM_DEV_QFRZDIS,
5761eba4c79SScott Long MSG_SIMPLE_Q_TAG,
57783a27783SJohn Baldwin req->dxferp,
578fcaf473cSAlexander Motin req->dxfer_len,
579fcaf473cSAlexander Motin req->mx_sb_len,
580fcaf473cSAlexander Motin req->cmd_len,
581fcaf473cSAlexander Motin req->timeout);
5821eba4c79SScott Long
5831eba4c79SScott Long error = sgsendccb(periph, ccb);
5841eba4c79SScott Long if (error) {
585fcaf473cSAlexander Motin req->host_status = DID_ERROR;
586fcaf473cSAlexander Motin req->driver_status = DRIVER_INVALID;
5871eba4c79SScott Long xpt_release_ccb(ccb);
5881eba4c79SScott Long break;
5891eba4c79SScott Long }
5901eba4c79SScott Long
591fcaf473cSAlexander Motin req->status = csio->scsi_status;
592fcaf473cSAlexander Motin req->masked_status = (csio->scsi_status >> 1) & 0x7f;
593fcaf473cSAlexander Motin sg_scsiio_status(csio, &req->host_status, &req->driver_status);
594fcaf473cSAlexander Motin req->resid = csio->resid;
595fcaf473cSAlexander Motin req->duration = csio->ccb_h.timeout;
596fcaf473cSAlexander Motin req->info = 0;
5971eba4c79SScott Long
598fcaf473cSAlexander Motin if ((csio->ccb_h.status & CAM_AUTOSNS_VALID)
599fcaf473cSAlexander Motin && (req->sbp != NULL)) {
600fcaf473cSAlexander Motin req->sb_len_wr = req->mx_sb_len - csio->sense_resid;
601fcaf473cSAlexander Motin error = copyout(&csio->sense_data, req->sbp,
602fcaf473cSAlexander Motin req->sb_len_wr);
6031eba4c79SScott Long }
6041eba4c79SScott Long
6051eba4c79SScott Long xpt_release_ccb(ccb);
6061eba4c79SScott Long break;
6071eba4c79SScott Long
6081eba4c79SScott Long case SG_GET_RESERVED_SIZE:
609fcaf473cSAlexander Motin {
610fcaf473cSAlexander Motin int *size = (int *)arg;
611fcaf473cSAlexander Motin *size = DFLTPHYS;
6121eba4c79SScott Long break;
6131eba4c79SScott Long }
6141eba4c79SScott Long
6151eba4c79SScott Long case SG_GET_SCSI_ID:
6161eba4c79SScott Long {
617fcaf473cSAlexander Motin struct sg_scsi_id *id = (struct sg_scsi_id *)arg;
6181eba4c79SScott Long
619fcaf473cSAlexander Motin id->host_no = cam_sim_path(xpt_path_sim(periph->path));
620fcaf473cSAlexander Motin id->channel = xpt_path_path_id(periph->path);
621fcaf473cSAlexander Motin id->scsi_id = xpt_path_target_id(periph->path);
622fcaf473cSAlexander Motin id->lun = xpt_path_lun_id(periph->path);
623fcaf473cSAlexander Motin id->scsi_type = softc->pd_type;
624fcaf473cSAlexander Motin id->h_cmd_per_lun = 1;
625fcaf473cSAlexander Motin id->d_queue_depth = 1;
626fcaf473cSAlexander Motin id->unused[0] = 0;
627fcaf473cSAlexander Motin id->unused[1] = 0;
6281eba4c79SScott Long break;
6291eba4c79SScott Long }
6301eba4c79SScott Long
63194fe9f95SAlexander Motin case SG_GET_SG_TABLESIZE:
63294fe9f95SAlexander Motin {
63394fe9f95SAlexander Motin int *size = (int *)arg;
63494fe9f95SAlexander Motin *size = 0;
63594fe9f95SAlexander Motin break;
63694fe9f95SAlexander Motin }
63794fe9f95SAlexander Motin
6381eba4c79SScott Long case SG_EMULATED_HOST:
6391eba4c79SScott Long case SG_SET_TRANSFORM:
6401eba4c79SScott Long case SG_GET_TRANSFORM:
6411eba4c79SScott Long case SG_GET_NUM_WAITING:
6421eba4c79SScott Long case SG_SCSI_RESET:
6431eba4c79SScott Long case SG_GET_REQUEST_TABLE:
6441eba4c79SScott Long case SG_SET_KEEP_ORPHAN:
6451eba4c79SScott Long case SG_GET_KEEP_ORPHAN:
6461eba4c79SScott Long case SG_GET_ACCESS_COUNT:
6471eba4c79SScott Long case SG_SET_FORCE_LOW_DMA:
6481eba4c79SScott Long case SG_GET_LOW_DMA:
6491eba4c79SScott Long case SG_SET_FORCE_PACK_ID:
6501eba4c79SScott Long case SG_GET_PACK_ID:
6511eba4c79SScott Long case SG_SET_RESERVED_SIZE:
6521eba4c79SScott Long case SG_GET_COMMAND_Q:
6531eba4c79SScott Long case SG_SET_COMMAND_Q:
6541eba4c79SScott Long case SG_SET_DEBUG:
6551eba4c79SScott Long case SG_NEXT_CMD_LEN:
6561eba4c79SScott Long default:
6571eba4c79SScott Long #ifdef CAMDEBUG
6581eba4c79SScott Long printf("sgioctl: rejecting cmd 0x%lx\n", cmd);
6591eba4c79SScott Long #endif
6601eba4c79SScott Long error = ENODEV;
6611eba4c79SScott Long break;
6621eba4c79SScott Long }
6631eba4c79SScott Long
6642b83592fSScott Long cam_periph_unlock(periph);
6651eba4c79SScott Long return (error);
6661eba4c79SScott Long }
6671eba4c79SScott Long
6681eba4c79SScott Long static int
sgwrite(struct cdev * dev,struct uio * uio,int ioflag)6691eba4c79SScott Long sgwrite(struct cdev *dev, struct uio *uio, int ioflag)
6701eba4c79SScott Long {
6711eba4c79SScott Long union ccb *ccb;
6721eba4c79SScott Long struct cam_periph *periph;
6731eba4c79SScott Long struct ccb_scsiio *csio;
6741eba4c79SScott Long struct sg_softc *sc;
6751eba4c79SScott Long struct sg_header *hdr;
6761eba4c79SScott Long struct sg_rdwr *rdwr;
6771eba4c79SScott Long u_char cdb_cmd;
6781eba4c79SScott Long char *buf;
6791eba4c79SScott Long int error = 0, cdb_len, buf_len, dir;
6801eba4c79SScott Long
6811eba4c79SScott Long periph = dev->si_drv1;
6824400b36dSScott Long rdwr = malloc(sizeof(*rdwr), M_DEVBUF, M_WAITOK | M_ZERO);
6831eba4c79SScott Long hdr = &rdwr->hdr.hdr;
6841eba4c79SScott Long
6851eba4c79SScott Long /* Copy in the header block and sanity check it */
6861eba4c79SScott Long if (uio->uio_resid < sizeof(*hdr)) {
6871eba4c79SScott Long error = EINVAL;
6881eba4c79SScott Long goto out_hdr;
6891eba4c79SScott Long }
6901eba4c79SScott Long error = uiomove(hdr, sizeof(*hdr), uio);
6911eba4c79SScott Long if (error)
6921eba4c79SScott Long goto out_hdr;
6931eba4c79SScott Long
69494fe9f95SAlexander Motin /* XXX: We don't support SG 3.x read/write API. */
69594fe9f95SAlexander Motin if (hdr->reply_len < 0) {
69694fe9f95SAlexander Motin error = ENODEV;
69794fe9f95SAlexander Motin goto out_hdr;
69894fe9f95SAlexander Motin }
69994fe9f95SAlexander Motin
7008008a935SScott Long ccb = xpt_alloc_ccb();
7011eba4c79SScott Long if (ccb == NULL) {
7021eba4c79SScott Long error = ENOMEM;
7031eba4c79SScott Long goto out_hdr;
7041eba4c79SScott Long }
7051eba4c79SScott Long csio = &ccb->csio;
7061eba4c79SScott Long
7071eba4c79SScott Long /*
7081eba4c79SScott Long * Copy in the CDB block. The designers of the interface didn't
7091eba4c79SScott Long * bother to provide a size for this in the header, so we have to
7101eba4c79SScott Long * figure it out ourselves.
7111eba4c79SScott Long */
7121eba4c79SScott Long if (uio->uio_resid < 1)
7131eba4c79SScott Long goto out_ccb;
7141eba4c79SScott Long error = uiomove(&cdb_cmd, 1, uio);
7151eba4c79SScott Long if (error)
7161eba4c79SScott Long goto out_ccb;
7171eba4c79SScott Long if (hdr->twelve_byte)
7181eba4c79SScott Long cdb_len = 12;
7191eba4c79SScott Long else
7201eba4c79SScott Long cdb_len = scsi_group_len(cdb_cmd);
7211eba4c79SScott Long /*
7221eba4c79SScott Long * We've already read the first byte of the CDB and advanced the uio
7231eba4c79SScott Long * pointer. Just read the rest.
7241eba4c79SScott Long */
7251eba4c79SScott Long csio->cdb_io.cdb_bytes[0] = cdb_cmd;
7261eba4c79SScott Long error = uiomove(&csio->cdb_io.cdb_bytes[1], cdb_len - 1, uio);
7271eba4c79SScott Long if (error)
7281eba4c79SScott Long goto out_ccb;
7291eba4c79SScott Long
7301eba4c79SScott Long /*
7311eba4c79SScott Long * Now set up the data block. Again, the designers didn't bother
7321eba4c79SScott Long * to make this reliable.
7331eba4c79SScott Long */
7341eba4c79SScott Long buf_len = uio->uio_resid;
7351eba4c79SScott Long if (buf_len != 0) {
7364400b36dSScott Long buf = malloc(buf_len, M_DEVBUF, M_WAITOK | M_ZERO);
7371eba4c79SScott Long error = uiomove(buf, buf_len, uio);
7381eba4c79SScott Long if (error)
7391eba4c79SScott Long goto out_buf;
7401eba4c79SScott Long dir = CAM_DIR_OUT;
7411eba4c79SScott Long } else if (hdr->reply_len != 0) {
7424400b36dSScott Long buf = malloc(hdr->reply_len, M_DEVBUF, M_WAITOK | M_ZERO);
7431eba4c79SScott Long buf_len = hdr->reply_len;
7441eba4c79SScott Long dir = CAM_DIR_IN;
7451eba4c79SScott Long } else {
7461eba4c79SScott Long buf = NULL;
7471eba4c79SScott Long buf_len = 0;
7481eba4c79SScott Long dir = CAM_DIR_NONE;
7491eba4c79SScott Long }
7501eba4c79SScott Long
7512b83592fSScott Long cam_periph_lock(periph);
7522b83592fSScott Long sc = periph->softc;
7531e637ba6SAlexander Motin xpt_setup_ccb(&ccb->ccb_h, periph->path, CAM_PRIORITY_NORMAL);
7541eba4c79SScott Long cam_fill_csio(csio,
7551eba4c79SScott Long /*retries*/1,
7561eba4c79SScott Long sgdone,
7571eba4c79SScott Long dir|CAM_DEV_QFRZDIS,
7581eba4c79SScott Long MSG_SIMPLE_Q_TAG,
7591eba4c79SScott Long buf,
7601eba4c79SScott Long buf_len,
7611eba4c79SScott Long SG_MAX_SENSE,
7621eba4c79SScott Long cdb_len,
763715ab212SScott Long sc->sg_timeout);
7641eba4c79SScott Long
7651eba4c79SScott Long /*
7661eba4c79SScott Long * Send off the command and hope that it works. This path does not
7671eba4c79SScott Long * go through sgstart because the I/O is supposed to be asynchronous.
7681eba4c79SScott Long */
7691eba4c79SScott Long rdwr->buf = buf;
7701eba4c79SScott Long rdwr->buf_len = buf_len;
7711eba4c79SScott Long rdwr->tag = hdr->pack_id;
7721eba4c79SScott Long rdwr->ccb = ccb;
7731eba4c79SScott Long rdwr->state = SG_RDWR_INPROG;
7741eba4c79SScott Long ccb->ccb_h.ccb_rdwr = rdwr;
7751eba4c79SScott Long ccb->ccb_h.ccb_type = SG_CCB_RDWR_IO;
7761eba4c79SScott Long TAILQ_INSERT_TAIL(&sc->rdwr_done, rdwr, rdwr_link);
7772b83592fSScott Long error = sgsendrdwr(periph, ccb);
7782b83592fSScott Long cam_periph_unlock(periph);
7792b83592fSScott Long return (error);
7801eba4c79SScott Long
7811eba4c79SScott Long out_buf:
7821eba4c79SScott Long free(buf, M_DEVBUF);
7831eba4c79SScott Long out_ccb:
7841eba4c79SScott Long xpt_free_ccb(ccb);
7851eba4c79SScott Long out_hdr:
7861eba4c79SScott Long free(rdwr, M_DEVBUF);
7871eba4c79SScott Long return (error);
7881eba4c79SScott Long }
7891eba4c79SScott Long
7901eba4c79SScott Long static int
sgread(struct cdev * dev,struct uio * uio,int ioflag)7911eba4c79SScott Long sgread(struct cdev *dev, struct uio *uio, int ioflag)
7921eba4c79SScott Long {
7931eba4c79SScott Long struct ccb_scsiio *csio;
7941eba4c79SScott Long struct cam_periph *periph;
7951eba4c79SScott Long struct sg_softc *sc;
7961eba4c79SScott Long struct sg_header *hdr;
7971eba4c79SScott Long struct sg_rdwr *rdwr;
7981eba4c79SScott Long u_short hstat, dstat;
7991eba4c79SScott Long int error, pack_len, reply_len, pack_id;
8001eba4c79SScott Long
8011eba4c79SScott Long periph = dev->si_drv1;
8021eba4c79SScott Long
8031eba4c79SScott Long /* XXX The pack len field needs to be updated and written out instead
8041eba4c79SScott Long * of discarded. Not sure how to do that.
8051eba4c79SScott Long */
8061eba4c79SScott Long uio->uio_rw = UIO_WRITE;
8071eba4c79SScott Long if ((error = uiomove(&pack_len, 4, uio)) != 0)
8081eba4c79SScott Long return (error);
8091eba4c79SScott Long if ((error = uiomove(&reply_len, 4, uio)) != 0)
8101eba4c79SScott Long return (error);
8111eba4c79SScott Long if ((error = uiomove(&pack_id, 4, uio)) != 0)
8121eba4c79SScott Long return (error);
8131eba4c79SScott Long uio->uio_rw = UIO_READ;
8141eba4c79SScott Long
8152b83592fSScott Long cam_periph_lock(periph);
8162b83592fSScott Long sc = periph->softc;
8171eba4c79SScott Long search:
8181eba4c79SScott Long TAILQ_FOREACH(rdwr, &sc->rdwr_done, rdwr_link) {
8191eba4c79SScott Long if (rdwr->tag == pack_id)
8201eba4c79SScott Long break;
8211eba4c79SScott Long }
8221eba4c79SScott Long if ((rdwr == NULL) || (rdwr->state != SG_RDWR_DONE)) {
823227d67aaSAlexander Motin if (cam_periph_sleep(periph, rdwr, PCATCH, "sgread", 0) == ERESTART)
8241eba4c79SScott Long return (EAGAIN);
8251eba4c79SScott Long goto search;
8261eba4c79SScott Long }
8271eba4c79SScott Long TAILQ_REMOVE(&sc->rdwr_done, rdwr, rdwr_link);
8282b83592fSScott Long cam_periph_unlock(periph);
8291eba4c79SScott Long
8301eba4c79SScott Long hdr = &rdwr->hdr.hdr;
8311eba4c79SScott Long csio = &rdwr->ccb->csio;
8321eba4c79SScott Long sg_scsiio_status(csio, &hstat, &dstat);
8331eba4c79SScott Long hdr->host_status = hstat;
8341eba4c79SScott Long hdr->driver_status = dstat;
8351eba4c79SScott Long hdr->target_status = csio->scsi_status >> 1;
8361eba4c79SScott Long
8371eba4c79SScott Long switch (hstat) {
8381eba4c79SScott Long case DID_OK:
8391eba4c79SScott Long case DID_PASSTHROUGH:
8401eba4c79SScott Long case DID_SOFT_ERROR:
8411eba4c79SScott Long hdr->result = 0;
8421eba4c79SScott Long break;
8431eba4c79SScott Long case DID_NO_CONNECT:
8441eba4c79SScott Long case DID_BUS_BUSY:
8451eba4c79SScott Long case DID_TIME_OUT:
8461eba4c79SScott Long hdr->result = EBUSY;
8471eba4c79SScott Long break;
8481eba4c79SScott Long case DID_BAD_TARGET:
8491eba4c79SScott Long case DID_ABORT:
8501eba4c79SScott Long case DID_PARITY:
8511eba4c79SScott Long case DID_RESET:
8521eba4c79SScott Long case DID_BAD_INTR:
8531eba4c79SScott Long case DID_ERROR:
8541eba4c79SScott Long default:
8551eba4c79SScott Long hdr->result = EIO;
8561eba4c79SScott Long break;
8571eba4c79SScott Long }
8581eba4c79SScott Long
8591eba4c79SScott Long if (dstat == DRIVER_SENSE) {
8601eba4c79SScott Long bcopy(&csio->sense_data, hdr->sense_buffer,
8611eba4c79SScott Long min(csio->sense_len, SG_MAX_SENSE));
8621eba4c79SScott Long #ifdef CAMDEBUG
8631eba4c79SScott Long scsi_sense_print(csio);
8641eba4c79SScott Long #endif
8651eba4c79SScott Long }
8661eba4c79SScott Long
8671eba4c79SScott Long error = uiomove(&hdr->result, sizeof(*hdr) -
8681eba4c79SScott Long offsetof(struct sg_header, result), uio);
8691eba4c79SScott Long if ((error == 0) && (hdr->result == 0))
8701eba4c79SScott Long error = uiomove(rdwr->buf, rdwr->buf_len, uio);
8711eba4c79SScott Long
8722b83592fSScott Long cam_periph_lock(periph);
8731eba4c79SScott Long xpt_free_ccb(rdwr->ccb);
8742b83592fSScott Long cam_periph_unlock(periph);
8751eba4c79SScott Long free(rdwr->buf, M_DEVBUF);
8761eba4c79SScott Long free(rdwr, M_DEVBUF);
8771eba4c79SScott Long return (error);
8781eba4c79SScott Long }
8791eba4c79SScott Long
8801eba4c79SScott Long static int
sgsendccb(struct cam_periph * periph,union ccb * ccb)8811eba4c79SScott Long sgsendccb(struct cam_periph *periph, union ccb *ccb)
8821eba4c79SScott Long {
8831eba4c79SScott Long struct sg_softc *softc;
8841eba4c79SScott Long struct cam_periph_map_info mapinfo;
885*d068ea16SMark Johnston int error, error1;
8861eba4c79SScott Long
8871eba4c79SScott Long softc = periph->softc;
8881eba4c79SScott Long bzero(&mapinfo, sizeof(mapinfo));
8892b83592fSScott Long
8902b83592fSScott Long /*
8912b83592fSScott Long * cam_periph_mapmem calls into proc and vm functions that can
8922b83592fSScott Long * sleep as well as trigger I/O, so we can't hold the lock.
8932b83592fSScott Long * Dropping it here is reasonably safe.
89495fbded6SScott Long * The only CCB opcode that is possible here is XPT_SCSI_IO, no
89595fbded6SScott Long * need for additional checks.
8962b83592fSScott Long */
8972b83592fSScott Long cam_periph_unlock(periph);
898de239312SAlexander Motin error = cam_periph_mapmem(ccb, &mapinfo, softc->maxio);
8992b83592fSScott Long cam_periph_lock(periph);
9001eba4c79SScott Long if (error)
9011eba4c79SScott Long return (error);
9021eba4c79SScott Long
9031eba4c79SScott Long error = cam_periph_runccb(ccb,
9041eba4c79SScott Long sgerror,
9051eba4c79SScott Long CAM_RETRY_SELTO,
9061eba4c79SScott Long SF_RETRY_UA,
9071eba4c79SScott Long softc->device_stats);
9081eba4c79SScott Long
9098cb46437SAlexander Motin cam_periph_unlock(periph);
910*d068ea16SMark Johnston error1 = cam_periph_unmapmem(ccb, &mapinfo);
911*d068ea16SMark Johnston if (error == 0)
912*d068ea16SMark Johnston error = error1;
9138cb46437SAlexander Motin cam_periph_lock(periph);
9141eba4c79SScott Long
9151eba4c79SScott Long return (error);
9161eba4c79SScott Long }
9171eba4c79SScott Long
9181eba4c79SScott Long static int
sgsendrdwr(struct cam_periph * periph,union ccb * ccb)9191eba4c79SScott Long sgsendrdwr(struct cam_periph *periph, union ccb *ccb)
9201eba4c79SScott Long {
9211eba4c79SScott Long struct sg_softc *softc;
9221eba4c79SScott Long
9231eba4c79SScott Long softc = periph->softc;
9241eba4c79SScott Long devstat_start_transaction(softc->device_stats, NULL);
9251eba4c79SScott Long xpt_action(ccb);
9261eba4c79SScott Long return (0);
9271eba4c79SScott Long }
9281eba4c79SScott Long
9291eba4c79SScott Long static int
sgerror(union ccb * ccb,uint32_t cam_flags,uint32_t sense_flags)9301eba4c79SScott Long sgerror(union ccb *ccb, uint32_t cam_flags, uint32_t sense_flags)
9311eba4c79SScott Long {
9321eba4c79SScott Long
933553484aeSWarner Losh return (cam_periph_error(ccb, cam_flags, sense_flags));
9341eba4c79SScott Long }
9351eba4c79SScott Long
9361eba4c79SScott Long static void
sg_scsiio_status(struct ccb_scsiio * csio,u_short * hoststat,u_short * drvstat)9371eba4c79SScott Long sg_scsiio_status(struct ccb_scsiio *csio, u_short *hoststat, u_short *drvstat)
9381eba4c79SScott Long {
9391eba4c79SScott Long int status;
9401eba4c79SScott Long
9411eba4c79SScott Long status = csio->ccb_h.status;
9421eba4c79SScott Long
9431eba4c79SScott Long switch (status & CAM_STATUS_MASK) {
9441eba4c79SScott Long case CAM_REQ_CMP:
9451eba4c79SScott Long *hoststat = DID_OK;
9461eba4c79SScott Long *drvstat = 0;
9471eba4c79SScott Long break;
9481eba4c79SScott Long case CAM_REQ_CMP_ERR:
9491eba4c79SScott Long *hoststat = DID_ERROR;
9501eba4c79SScott Long *drvstat = 0;
9511eba4c79SScott Long break;
9521eba4c79SScott Long case CAM_REQ_ABORTED:
9531eba4c79SScott Long *hoststat = DID_ABORT;
9541eba4c79SScott Long *drvstat = 0;
9551eba4c79SScott Long break;
9561eba4c79SScott Long case CAM_REQ_INVALID:
9571eba4c79SScott Long *hoststat = DID_ERROR;
9581eba4c79SScott Long *drvstat = DRIVER_INVALID;
9591eba4c79SScott Long break;
9601eba4c79SScott Long case CAM_DEV_NOT_THERE:
9611eba4c79SScott Long *hoststat = DID_BAD_TARGET;
9621eba4c79SScott Long *drvstat = 0;
9634fee613eSEdward Tomasz Napierala break;
9641eba4c79SScott Long case CAM_SEL_TIMEOUT:
9651eba4c79SScott Long *hoststat = DID_NO_CONNECT;
9661eba4c79SScott Long *drvstat = 0;
9671eba4c79SScott Long break;
9681eba4c79SScott Long case CAM_CMD_TIMEOUT:
9691eba4c79SScott Long *hoststat = DID_TIME_OUT;
9701eba4c79SScott Long *drvstat = 0;
9711eba4c79SScott Long break;
9721eba4c79SScott Long case CAM_SCSI_STATUS_ERROR:
9731eba4c79SScott Long *hoststat = DID_ERROR;
9741eba4c79SScott Long *drvstat = 0;
9750c70e307SEdward Tomasz Napierala break;
9761eba4c79SScott Long case CAM_SCSI_BUS_RESET:
9771eba4c79SScott Long *hoststat = DID_RESET;
9781eba4c79SScott Long *drvstat = 0;
9791eba4c79SScott Long break;
9801eba4c79SScott Long case CAM_UNCOR_PARITY:
9811eba4c79SScott Long *hoststat = DID_PARITY;
9821eba4c79SScott Long *drvstat = 0;
9831eba4c79SScott Long break;
9841eba4c79SScott Long case CAM_SCSI_BUSY:
9851eba4c79SScott Long *hoststat = DID_BUS_BUSY;
9861eba4c79SScott Long *drvstat = 0;
9870c70e307SEdward Tomasz Napierala break;
9881eba4c79SScott Long default:
9891eba4c79SScott Long *hoststat = DID_ERROR;
9901eba4c79SScott Long *drvstat = DRIVER_ERROR;
9911eba4c79SScott Long }
9921eba4c79SScott Long
9931eba4c79SScott Long if (status & CAM_AUTOSNS_VALID)
9941eba4c79SScott Long *drvstat = DRIVER_SENSE;
9951eba4c79SScott Long }
9961eba4c79SScott Long
9971eba4c79SScott Long static int
scsi_group_len(u_char cmd)9981eba4c79SScott Long scsi_group_len(u_char cmd)
9991eba4c79SScott Long {
10001eba4c79SScott Long int len[] = {6, 10, 10, 12, 12, 12, 10, 10};
10011eba4c79SScott Long int group;
10021eba4c79SScott Long
10031eba4c79SScott Long group = (cmd >> 5) & 0x7;
10041eba4c79SScott Long return (len[group]);
10051eba4c79SScott Long }
1006