49 m_maybe_sat(maybe_sat)
83 const char * req_type,
unsigned nsid,
bool maybe_sat);
91 const char * req_type,
unsigned nsid,
bool maybe_sat)
92:
smart_device(intf, scsidev->get_dev_name(),
"sntasmedia", req_type),
105 unsigned cdw10_hi = in.
cdw10 >> 16;
108 if (in.
cdw10 == 0x0000001)
110 if (in.
cdw10 == 0x0000000) {
113 return set_err(ENOSYS,
"NVMe Identify Namespace 0x%x not supported", in.
nsid);
115 return set_err(ENOSYS,
"NVMe Identify with CDW10=0x%08x not supported", in.
cdw10);
118 return set_err(ENOSYS,
"NVMe Get Log Page with NSID=0x%x not supported", in.
nsid);
121 return set_err(ENOSYS,
"NVMe admin command 0x%02x not supported", in.
opcode);
125 return set_err(ENOSYS,
"Nonzero NVMe command dwords 11, 14, or 15 not supported");
127 uint8_t
cdb[16] = {0, };
133 cdb[6] = (uint8_t)(cdw10_hi >> 8);
134 cdb[7] = (uint8_t)cdw10_hi;
139 cdb[12] = (uint8_t)(in.
cdw12 >> 24);
140 cdb[13] = (uint8_t)(in.
cdw12 >> 16);
146 io_hdr.cmnd_len =
sizeof(
cdb);
148 io_hdr.dxferp = (uint8_t *)in.buffer;
149 io_hdr.dxfer_len =
size;
150 memset(in.buffer, 0, in.size);
154 return set_err(scsidev->
get_err());
163#define SNT_JMICRON_NVME_SIGNATURE 0x454d564eu
164#define SNT_JMICRON_CDB_LEN 12
165#define SNT_JMICRON_NVM_CMD_LEN 512
172 const char * req_type,
unsigned nsid,
bool maybe_sat);
186 const char * req_type,
unsigned nsid,
bool maybe_sat)
187:
smart_device(intf, scsidev->get_dev_name(),
"sntjmicron", req_type),
212 constexpr bool admin =
true;
220 unsigned int lid = in.
cdw10 & 0xFF;
221 if (lid == 0x6 && in.
size > 0x218) {
222 unsigned size = 0x218;
224 pout(
"Warning: self-test output truncated to 19 items to workaround controller bug\n");
236 nvm_cmd[3] = in.
nsid;
241 nvm_cmd[13] = in.
cdw11;
242 nvm_cmd[14] = in.
cdw12;
243 nvm_cmd[15] = in.
cdw13;
244 nvm_cmd[16] = in.
cdw14;
245 nvm_cmd[17] = in.
cdw15;
257 io_nvm.
dxferp = (uint8_t *)nvm_cmd;
262 "sntjmicron_device::nvme_pass_through:NVM: "))
302 "sntjmicron_device::nvme_pass_through:Data: "))
320 io_reply.
dxferp = (uint8_t *)nvm_reply;
325 "sntjmicron_device::nvme_pass_through:Reply: "))
330 swapx(&nvm_reply[i]);
333 return set_err(EIO,
"Out of spec JMicron NVMe reply");
335 int status = nvm_reply[5] >> 17;
340 out.
result = nvm_reply[2];
354 const char * req_type,
unsigned nsid,
bool maybe_sat);
362 const char * req_type,
unsigned nsid,
bool maybe_sat)
363:
smart_device(intf, scsidev->get_dev_name(),
"sntrealtek", req_type),
378 if (in.
cdw10 == 0x0000001)
380 if (in.
cdw10 == 0x0000000) {
383 return set_err(ENOSYS,
"NVMe Identify Namespace 0x%x not supported", in.
nsid);
385 return set_err(ENOSYS,
"NVMe Identify with CDW10=0x%08x not supported", in.
cdw10);
388 return set_err(ENOSYS,
"NVMe Get Log Page with NSID=0x%x not supported", in.
nsid);
392 pout(
"Warning: NVMe Get Log truncated to 0x%03x bytes, 0x%03x bytes zero filled\n",
size, in.
size -
size);
396 return set_err(ENOSYS,
"NVMe admin command 0x%02x not supported", in.
opcode);
400 return set_err(ENOSYS,
"Nonzero NVMe command dwords 11-15 not supported");
402 uint8_t
cdb[16] = {0, };
410 io_hdr.cmnd_len =
sizeof(
cdb);
412 io_hdr.dxferp = (uint8_t *)in.buffer;
413 io_hdr.dxfer_len =
size;
414 memset(in.buffer, 0, in.size);
418 return set_err(scsidev->
get_err());
432 throw std::logic_error(
"smart_interface: get_snt_device() called with scsidev=0");
435 bool maybe_sat =
false;
437 snprintf(snt_type,
sizeof(snt_type),
"%s", type);
438 int len = strlen(snt_type);
439 if (len > 4 && !strcmp(snt_type + len - 4,
"/sat")) {
440 snt_type[len -= 4] = 0;
448 if (!strcmp(snt_type,
"sntasmedia")) {
454 int n1 = -1, n2 = -1;
456 sscanf(snt_type,
"sntjmicron%n,0x%x%n", &n1, &
nsid, &n2);
457 if (!(n1 == len || n2 == len))
458 return set_err_np(EINVAL,
"Invalid NVMe namespace id in '%s'", snt_type);
462 else if (!strcmp(snt_type,
"sntrealtek")) {
468 return set_err_np(EINVAL,
"Unknown SNT device type '%s'", type);
Smart pointer class for device pointers.
device_type * release()
Return the pointer and release ownership.
bool set_nvme_err(nvme_cmd_out &out, unsigned status, const char *msg=0)
Set last error number and message if pass-through returns NVMe error status.
bool scsi_pass_through_and_check(scsi_cmnd_io *iop, const char *msg="")
Base class for all devices.
const error_info & get_err() const
Get last error info struct.
smart_interface * smi()
Get interface which produced this object.
bool set_err(int no, const char *msg,...) __attribute_format_printf(3
Set last error number and message.
const char * get_info_name() const
Get informal name.
device_info & set_info()
R/W access to device info struct.
The platform interface abstraction.
virtual nvme_device * get_snt_device(const char *type, scsi_device *scsidev)
Return NVMe->SCSI filter for a SNT or USB 'type'.
virtual ata_device * get_sat_device(const char *type, scsi_device *scsidev)
Return ATA->SCSI filter for a SAT or USB 'type'.
nvme_or_sat_device(scsi_device *scsidev, unsigned nsid, bool maybe_sat)
virtual smart_device * autodetect_open() override
Open device with autodetection support.
virtual bool nvme_pass_through(const nvme_cmd_in &in, nvme_cmd_out &out) override
NVMe pass through.
virtual ~sntjmicron_device()
sntjmicron_device(smart_interface *intf, scsi_device *scsidev, const char *req_type, unsigned nsid, bool maybe_sat)
sntrealtek_device(smart_interface *intf, scsi_device *scsidev, const char *req_type, unsigned nsid, bool maybe_sat)
virtual ~sntrealtek_device()
virtual bool nvme_pass_through(const nvme_cmd_in &in, nvme_cmd_out &out) override
NVMe pass through.
virtual bool open() override
Open device, return false on error.
Implement a device by tunneling through another device.
virtual void release(const smart_device *dev) override
tunnel_device_type * get_tunnel_dev()
bool nvme_read_id_ctrl(nvme_device *device, nvme_id_ctrl &id_ctrl)
constexpr uint32_t nvme_broadcast_nsid
#define SAT_ATA_PASSTHROUGH_12
#define DXFER_FROM_DEVICE
#define SNT_JMICRON_CDB_LEN
#define SNT_JMICRON_NVM_CMD_LEN
const char * scsinvme_cpp_svnid
#define SNT_JMICRON_NVME_SIGNATURE
static void sg_put_unaligned_be24(uint32_t val, void *p)
static void sg_put_unaligned_le16(uint16_t val, void *p)
void pout(const char *fmt,...)
NVMe pass through input parameters.
unsigned char direction() const
Get I/O direction from opcode.
unsigned char opcode
Opcode (CDW0 07:00)
unsigned size
Size of buffer.
unsigned cdw15
Cmd specific.
unsigned nsid
Namespace ID.
void * buffer
Pointer to data buffer.
NVMe pass through output parameters.
unsigned result
Command specific result (DW0)
std::string info_name
Informal name.
std::string strprintf(const char *fmt,...)
void swapx(unsigned short *p)
bool str_starts_with(const char *str, const char *prefix)