1 /*
   2  * CDDL HEADER START
   3  *
   4  * The contents of this file are subject to the terms of the
   5  * Common Development and Distribution License (the "License").
   6  * You may not use this file except in compliance with the License.
   7  *
   8  * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
   9  * or http://www.opensolaris.org/os/licensing.
  10  * See the License for the specific language governing permissions
  11  * and limitations under the License.
  12  *
  13  * When distributing Covered Code, include this CDDL HEADER in each
  14  * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
  15  * If applicable, add the following below this CDDL HEADER, with the
  16  * fields enclosed by brackets "[]" replaced with your own identifying
  17  * information: Portions Copyright [yyyy] [name of copyright owner]
  18  *
  19  * CDDL HEADER END
  20  */
  21 
  22 /*
  23  * Copyright (c) 2008, 2010, Oracle and/or its affiliates. All rights reserved.
  24  * Copyright 2015 Nexenta Systems, Inc.  All rights reserved.
  25  */
  26 
  27 #ifndef _LIBSCSI_H
  28 #define _LIBSCSI_H
  29 
  30 #ifdef  __cplusplus
  31 extern "C" {
  32 #endif
  33 
  34 #include <sys/types.h>
  35 #include <sys/sysmacros.h>
  36 #include <sys/scsi/impl/spc3_types.h>
  37 #include <stdarg.h>
  38 
  39 #define LIBSCSI_VERSION         1
  40 #define LIBSCSI_STATUS_INVALID  ((sam4_status_t)-1)
  41 #define LIBSCSI_DEFAULT_ENGINE_PATH     "/usr/lib/scsi/plugins/scsi/engines"
  42 #define LIBSCSI_DEFAULT_ENGINE  "uscsi"
  43 
  44 /*
  45  * Flags for action creation.  Selected to avoid overlap with the uscsi
  46  * flags with similar or identical meaning.
  47  */
  48 #define LIBSCSI_AF_READ         0x80000000
  49 #define LIBSCSI_AF_WRITE        0x40000000
  50 #define LIBSCSI_AF_SILENT       0x20000000
  51 #define LIBSCSI_AF_DIAGNOSE     0x10000000
  52 #define LIBSCSI_AF_ISOLATE      0x08000000
  53 #define LIBSCSI_AF_RQSENSE      0x04000000
  54 
  55 typedef enum libscsi_errno {
  56         ESCSI_NONE,             /* no error */
  57         ESCSI_NOMEM,            /* no memory */
  58         ESCSI_ZERO_LENGTH,      /* zero-length allocation requested */
  59         ESCSI_VERSION,          /* library version mismatch */
  60         ESCSI_BADTARGET,        /* invalid target specification */
  61         ESCSI_BADCMD,           /* invalid SCSI command */
  62         ESCSI_BADENGINE,        /* engine library corrupt */
  63         ESCSI_NOENGINE,         /* engine library not found */
  64         ESCSI_ENGINE_INIT,      /* engine initialization failed */
  65         ESCSI_ENGINE_VER,       /* engine version mismatch */
  66         ESCSI_ENGINE_BADPATH,   /* engine path contains no usable components */
  67         ESCSI_BADFLAGS,         /* incorrect action flags */
  68         ESCSI_BOGUSFLAGS,       /* unknown flag value */
  69         ESCSI_BADLENGTH,        /* buffer length overflow */
  70         ESCSI_NEEDBUF,          /* missing required buffer */
  71         ESCSI_IO,               /* I/O operation failed */
  72         ESCSI_SYS,              /* system call failed */
  73         ESCSI_PERM,             /* insufficient permissions */
  74         ESCSI_RANGE,            /* parameter outside valid range */
  75         ESCSI_NOTSUP,           /* operation not supported */
  76         ESCSI_UNKNOWN,          /* error of unknown type */
  77         ESCSI_INQUIRY_FAILED,   /* initial inquiry command failed */
  78         ESCSI_MAX               /* maximum libscsi errno value */
  79 } libscsi_errno_t;
  80 
  81 struct libscsi_hdl;
  82 typedef struct libscsi_hdl libscsi_hdl_t;
  83 
  84 struct libscsi_target;
  85 typedef struct libscsi_target libscsi_target_t;
  86 
  87 typedef struct libscsi_status {
  88         uint64_t lss_status;            /* SCSI status of this command */
  89         size_t lss_sense_len;           /* Length in bytes of sense data */
  90         uint8_t *lss_sense_data;        /* Pointer to sense data */
  91 } libscsi_status_t;
  92 
  93 struct libscsi_action;
  94 typedef struct libscsi_action libscsi_action_t;
  95 
  96 typedef struct libscsi_engine_ops {
  97         void *(*lseo_open)(libscsi_hdl_t *, const void *);
  98         void (*lseo_close)(libscsi_hdl_t *, void *);
  99         int (*lseo_exec)(libscsi_hdl_t *, void *, libscsi_action_t *);
 100         void (*lseo_target_name)(libscsi_hdl_t *, void *, char *, size_t);
 101 } libscsi_engine_ops_t;
 102 
 103 typedef struct libscsi_engine {
 104         const char *lse_name;
 105         uint_t lse_libversion;
 106         const libscsi_engine_ops_t *lse_ops;
 107 } libscsi_engine_t;
 108 
 109 extern libscsi_hdl_t *libscsi_init(uint_t, libscsi_errno_t *);
 110 extern void libscsi_fini(libscsi_hdl_t *);
 111 
 112 extern libscsi_target_t *libscsi_open(libscsi_hdl_t *, const char *,
 113     const void *);
 114 extern void libscsi_close(libscsi_hdl_t *, libscsi_target_t *);
 115 extern libscsi_hdl_t *libscsi_get_handle(libscsi_target_t *);
 116 
 117 extern const char *libscsi_vendor(libscsi_target_t *);
 118 extern const char *libscsi_product(libscsi_target_t *);
 119 extern const char *libscsi_revision(libscsi_target_t *);
 120 extern const char *libscsi_lid(libscsi_target_t *);
 121 extern const char *libscsi_usn(libscsi_target_t *);
 122 
 123 extern libscsi_errno_t libscsi_errno(libscsi_hdl_t *);
 124 extern const char *libscsi_errmsg(libscsi_hdl_t *);
 125 extern const char *libscsi_strerror(libscsi_errno_t);
 126 extern const char *libscsi_errname(libscsi_errno_t);
 127 extern libscsi_errno_t libscsi_errcode(const char *);
 128 
 129 extern libscsi_action_t *libscsi_action_alloc(libscsi_hdl_t *, spc3_cmd_t,
 130     uint_t, void *, size_t);
 131 extern sam4_status_t libscsi_action_get_status(const libscsi_action_t *);
 132 extern void libscsi_action_set_timeout(libscsi_action_t *, uint32_t);
 133 extern uint32_t libscsi_action_get_timeout(const libscsi_action_t *);
 134 extern uint_t libscsi_action_get_flags(const libscsi_action_t *);
 135 extern uint8_t *libscsi_action_get_cdb(const libscsi_action_t *);
 136 extern int libscsi_action_get_buffer(const libscsi_action_t *,
 137     uint8_t **, size_t *, size_t *);
 138 extern int libscsi_action_get_sense(const libscsi_action_t *,
 139     uint8_t **, size_t *, size_t *);
 140 extern int libscsi_action_parse_sense(const libscsi_action_t *, uint64_t *,
 141     uint64_t *, uint64_t *, diskaddr_t *);
 142 extern void libscsi_action_set_status(libscsi_action_t *, sam4_status_t);
 143 extern int libscsi_action_set_datalen(libscsi_action_t *, size_t);
 144 extern int libscsi_action_set_senselen(libscsi_action_t *, size_t);
 145 extern int libscsi_exec(libscsi_action_t *, libscsi_target_t *);
 146 extern void libscsi_action_free(libscsi_action_t *);
 147 
 148 extern const char *libscsi_sense_key_name(uint64_t);
 149 extern const char *libscsi_sense_code_name(uint64_t, uint64_t);
 150 
 151 /*
 152  * Interfaces for engine providers
 153  */
 154 extern void *libscsi_alloc(libscsi_hdl_t *, size_t);
 155 extern void *libscsi_zalloc(libscsi_hdl_t *, size_t);
 156 extern char *libscsi_strdup(libscsi_hdl_t *, const char *);
 157 extern void libscsi_free(libscsi_hdl_t *, void *);
 158 extern libscsi_status_t *libscsi_status_alloc(libscsi_hdl_t *, size_t);
 159 extern int libscsi_status_fill(libscsi_hdl_t *, libscsi_status_t *,
 160     uint16_t, size_t);
 161 extern void libscsi_status_free(libscsi_hdl_t *, libscsi_status_t *);
 162 
 163 extern int libscsi_set_errno(libscsi_hdl_t *, libscsi_errno_t);
 164 extern int libscsi_verror(libscsi_hdl_t *, libscsi_errno_t, const char *,
 165     va_list);
 166 extern int libscsi_error(libscsi_hdl_t *, libscsi_errno_t, const char *, ...);
 167 
 168 typedef const libscsi_engine_t *(*libscsi_engine_init_f)(libscsi_hdl_t *);
 169 
 170 /*
 171  * Generic SCSI utility functions.
 172  */
 173 extern size_t libscsi_cmd_cdblen(libscsi_hdl_t *, uint8_t);
 174 
 175 #ifdef  __cplusplus
 176 }
 177 #endif
 178 
 179 #endif  /* _LIBSCSI_H */