smartmontools SVN Rev 5680
Utility to control and monitor storage systems with "S.M.A.R.T."
scsinvme.cpp
Go to the documentation of this file.
1/*
2 * scsinvme.cpp
3 *
4 * Home page of code is: https://www.smartmontools.org
5 *
6 * Copyright (C) 2020-25 Christian Franke
7 * Copyright (C) 2018 Harry Mallon <hjmallon@gmail.com>
8 *
9 * SPDX-License-Identifier: GPL-2.0-or-later
10 */
11
12#include "config.h"
13
14#include "dev_interface.h"
15#include "dev_tunnelled.h"
16#include "nvmecmds.h"
17#include "scsicmds.h"
18#include "sg_unaligned.h"
19#include "utility.h"
20
21#include <errno.h>
22
23const char * scsinvme_cpp_svnid = "$Id: scsinvme.cpp 5677 2025-03-20 17:22:18Z chrfranke $";
24
25// SNT (SCSI NVMe Translation) namespace and prefix
26namespace snt {
27
28/////////////////////////////////////////////////////////////////////////////
29// nvme_or_sat_device: Common base class for NVMe/SATA -> USB bridges
30
32: public tunnelled_device<
33 /*implements*/ nvme_device,
34 /*by tunnelling through a*/ scsi_device
35 >
36{
37public:
38 nvme_or_sat_device(scsi_device * scsidev, unsigned nsid, bool maybe_sat);
39
40 virtual smart_device * autodetect_open() override;
41
42private:
44};
45
46nvme_or_sat_device::nvme_or_sat_device(scsi_device * scsidev, unsigned nsid, bool maybe_sat)
47: smart_device(never_called),
49 m_maybe_sat(maybe_sat)
50{
51}
52
54{
55 if (!open() || !m_maybe_sat)
56 return this;
57
58 // SAT not tried first because some USB bridges emulate ATA IDENTIFY via SAT
59 // if a NVMe device is connected.
60 // TODO: Preserve id_ctrl for next nvme_read_id_ctrl() call
62 if (nvme_read_id_ctrl(this, id_ctrl))
63 return this;
64
65 // NVMe Identify Controller failed, use the already opened SCSI device for SAT.
66 // IMPORTANT for derived classes: this->close() not called before delete.
67 // TODO: preserve requested 'type'.
68 scsi_device * scsidev = get_tunnel_dev();
69 ata_device * satdev = smi()->get_sat_device("sat", scsidev);
70 release(scsidev); // 'scsidev' is now owned by 'satdev'
71 delete this;
72 return satdev;
73}
74
75/////////////////////////////////////////////////////////////////////////////
76// sntasmedia_device
77
79: public nvme_or_sat_device
80{
81public:
83 const char * req_type, unsigned nsid, bool maybe_sat);
84
85 virtual ~sntasmedia_device();
86
87 virtual bool nvme_pass_through(const nvme_cmd_in & in, nvme_cmd_out & out) override;
88};
89
91 const char * req_type, unsigned nsid, bool maybe_sat)
92: smart_device(intf, scsidev->get_dev_name(), "sntasmedia", req_type),
93 nvme_or_sat_device(scsidev, nsid, maybe_sat)
94{
95 set_info().info_name = strprintf("%s [USB NVMe ASMedia]", scsidev->get_info_name());
96}
97
99{
100}
101
103{
104 unsigned size = in.size;
105 unsigned cdw10_hi = in.cdw10 >> 16;
106 switch (in.opcode) {
108 if (in.cdw10 == 0x0000001) // Identify controller
109 break;
110 if (in.cdw10 == 0x0000000) { // Identify namespace
111 if (in.nsid == 1)
112 break;
113 return set_err(ENOSYS, "NVMe Identify Namespace 0x%x not supported", in.nsid);
114 }
115 return set_err(ENOSYS, "NVMe Identify with CDW10=0x%08x not supported", in.cdw10);
117 if (!(in.nsid == nvme_broadcast_nsid || !in.nsid))
118 return set_err(ENOSYS, "NVMe Get Log Page with NSID=0x%x not supported", in.nsid);
119 break;
120 default:
121 return set_err(ENOSYS, "NVMe admin command 0x%02x not supported", in.opcode);
122 break;
123 }
124 if (in.cdw11 || in.cdw14 || in.cdw15)
125 return set_err(ENOSYS, "Nonzero NVMe command dwords 11, 14, or 15 not supported");
126
127 uint8_t cdb[16] = {0, };
128 cdb[0] = 0xe6;
129 cdb[1] = in.opcode;
130 //cdb[2] = 0
131 cdb[3] = (uint8_t)in.cdw10;
132 //cdb[4..5] = 0
133 cdb[6] = (uint8_t)(cdw10_hi >> 8);
134 cdb[7] = (uint8_t)cdw10_hi;
135 cdb[8] = (uint8_t)(in.cdw13 >> 24);
136 cdb[9] = (uint8_t)(in.cdw13 >> 16);
137 cdb[10] = (uint8_t)(in.cdw13 >> 8);
138 cdb[11] = (uint8_t)in.cdw13;
139 cdb[12] = (uint8_t)(in.cdw12 >> 24);
140 cdb[13] = (uint8_t)(in.cdw12 >> 16);
141 cdb[14] = (uint8_t)(in.cdw12 >> 8);
142 cdb[15] = (uint8_t)in.cdw12;
143
144 scsi_cmnd_io io_hdr = {};
145 io_hdr.cmnd = cdb;
146 io_hdr.cmnd_len = sizeof(cdb);
147 io_hdr.dxfer_dir = DXFER_FROM_DEVICE;
148 io_hdr.dxferp = (uint8_t *)in.buffer;
149 io_hdr.dxfer_len = size;
150 memset(in.buffer, 0, in.size);
151
152 scsi_device * scsidev = get_tunnel_dev();
153 if (!scsidev->scsi_pass_through_and_check(&io_hdr, "sntasmedia_device::nvme_pass_through: "))
154 return set_err(scsidev->get_err());
155
156 //out.result = ?;
157 return true;
158}
159
160/////////////////////////////////////////////////////////////////////////////
161// sntjmicron_device
162
163#define SNT_JMICRON_NVME_SIGNATURE 0x454d564eu // 'NVME' reversed (little endian)
164#define SNT_JMICRON_CDB_LEN 12
165#define SNT_JMICRON_NVM_CMD_LEN 512
166
168: public nvme_or_sat_device
169{
170public:
172 const char * req_type, unsigned nsid, bool maybe_sat);
173
174 virtual ~sntjmicron_device();
175
176 virtual bool nvme_pass_through(const nvme_cmd_in & in, nvme_cmd_out & out) override;
177
178private:
179 enum {
182 };
183};
184
186 const char * req_type, unsigned nsid, bool maybe_sat)
187: smart_device(intf, scsidev->get_dev_name(), "sntjmicron", req_type),
188 nvme_or_sat_device(scsidev, nsid, maybe_sat)
189{
190 set_info().info_name = strprintf("%s [USB NVMe JMicron]", scsidev->get_info_name());
191}
192
194{
195}
196
197// cdb[0]: ATA PASS THROUGH (12) SCSI command opcode byte (0xa1)
198// cdb[1]: [ is admin cmd: 1 ] [ protocol : 7 ]
199// cdb[2]: reserved
200// cdb[3]: parameter list length (23:16)
201// cdb[4]: parameter list length (15:08)
202// cdb[5]: parameter list length (07:00)
203// cdb[6]: reserved
204// cdb[7]: reserved
205// cdb[8]: reserved
206// cdb[9]: reserved
207// cdb[10]: reserved
208// cdb[11]: CONTROL (?)
210{
211 /* Only admin commands used */
212 constexpr bool admin = true;
213
214 // 1: "NVM Command Set Payload"
215 {
216 // for whatever reason selftest log causing controller to hang if size is set > 0x230b
217 // see GH issue #256 for the details. Patching it to include last 19 log records instead
218 unsigned cdw10 = in.cdw10;
220 unsigned int lid = in.cdw10 & 0xFF;
221 if (lid == 0x6 && in.size > 0x218) {
222 unsigned size = 0x218;
223 cdw10 = lid | ((size/4 - 1) << 16);
224 pout("Warning: self-test output truncated to 19 items to workaround controller bug\n");
225 }
226 }
227 unsigned char cdb[SNT_JMICRON_CDB_LEN] = { 0 };
229 cdb[1] = (admin ? 0x80 : 0x00) | proto_nvm_cmd;
231
232 unsigned nvm_cmd[SNT_JMICRON_NVM_CMD_LEN / sizeof(unsigned)] = { 0 };
233 nvm_cmd[0] = SNT_JMICRON_NVME_SIGNATURE;
234 // nvm_cmd[1]: reserved
235 nvm_cmd[2] = in.opcode; // More of CDW0 may go in here in future
236 nvm_cmd[3] = in.nsid;
237 // nvm_cmd[4-5]: reserved
238 // nvm_cmd[6-7]: metadata pointer
239 // nvm_cmd[8-11]: data ptr (?)
240 nvm_cmd[12] = cdw10;
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;
246 // nvm_cmd[18-127]: reserved
247
248 if (isbigendian())
249 for (unsigned i = 0; i < (SNT_JMICRON_NVM_CMD_LEN / sizeof(uint32_t)); i++)
250 swapx(&nvm_cmd[i]);
251
252 scsi_cmnd_io io_nvm = {};
253
254 io_nvm.cmnd = cdb;
256 io_nvm.dxfer_dir = DXFER_TO_DEVICE;
257 io_nvm.dxferp = (uint8_t *)nvm_cmd;
259
260 scsi_device * scsidev = get_tunnel_dev();
261 if (!scsidev->scsi_pass_through_and_check(&io_nvm,
262 "sntjmicron_device::nvme_pass_through:NVM: "))
263 return set_err(scsidev->get_err());
264 }
265
266 // 2: DMA or Non-Data
267 {
268 unsigned char cdb[SNT_JMICRON_CDB_LEN] = { 0 };
270
271 scsi_cmnd_io io_data = {};
272 io_data.cmnd = cdb;
274
275 switch (in.direction()) {
277 cdb[1] = (admin ? 0x80 : 0x00) | proto_non_data;
278 io_data.dxfer_dir = DXFER_NONE;
279 break;
281 cdb[1] = (admin ? 0x80 : 0x00) | proto_dma_out;
283 io_data.dxfer_dir = DXFER_TO_DEVICE;
284 io_data.dxferp = (uint8_t *)in.buffer;
285 io_data.dxfer_len = in.size;
286 break;
288 cdb[1] = (admin ? 0x80 : 0x00) | proto_dma_in;
291 io_data.dxferp = (uint8_t *)in.buffer;
292 io_data.dxfer_len = in.size;
293 memset(in.buffer, 0, in.size);
294 break;
296 default:
297 return set_err(EINVAL);
298 }
299
300 scsi_device * scsidev = get_tunnel_dev();
301 if (!scsidev->scsi_pass_through_and_check(&io_data,
302 "sntjmicron_device::nvme_pass_through:Data: "))
303 return set_err(scsidev->get_err());
304 }
305
306 // 3: "Return Response Information"
307 {
308 unsigned char cdb[SNT_JMICRON_CDB_LEN] = { 0 };
310 cdb[1] = (admin ? 0x80 : 0x00) | proto_response;
312
313 unsigned nvm_reply[SNT_JMICRON_NVM_CMD_LEN / sizeof(unsigned)] = { 0 };
314
315 scsi_cmnd_io io_reply = {};
316
317 io_reply.cmnd = cdb;
318 io_reply.cmnd_len = SNT_JMICRON_CDB_LEN;
319 io_reply.dxfer_dir = DXFER_FROM_DEVICE;
320 io_reply.dxferp = (uint8_t *)nvm_reply;
322
323 scsi_device * scsidev = get_tunnel_dev();
324 if (!scsidev->scsi_pass_through_and_check(&io_reply,
325 "sntjmicron_device::nvme_pass_through:Reply: "))
326 return set_err(scsidev->get_err());
327
328 if (isbigendian())
329 for (unsigned i = 0; i < (SNT_JMICRON_NVM_CMD_LEN / sizeof(uint32_t)); i++)
330 swapx(&nvm_reply[i]);
331
332 if (nvm_reply[0] != SNT_JMICRON_NVME_SIGNATURE)
333 return set_err(EIO, "Out of spec JMicron NVMe reply");
334
335 int status = nvm_reply[5] >> 17;
336
337 if (status > 0)
338 return set_nvme_err(out, status);
339
340 out.result = nvm_reply[2];
341 }
342
343 return true;
344}
345
346/////////////////////////////////////////////////////////////////////////////
347// sntrealtek_device
348
350: public nvme_or_sat_device
351{
352public:
354 const char * req_type, unsigned nsid, bool maybe_sat);
355
356 virtual ~sntrealtek_device();
357
358 virtual bool nvme_pass_through(const nvme_cmd_in & in, nvme_cmd_out & out) override;
359};
360
362 const char * req_type, unsigned nsid, bool maybe_sat)
363: smart_device(intf, scsidev->get_dev_name(), "sntrealtek", req_type),
364 nvme_or_sat_device(scsidev, nsid, maybe_sat)
365{
366 set_info().info_name = strprintf("%s [USB NVMe Realtek]", scsidev->get_info_name());
367}
368
370{
371}
372
374{
375 unsigned size = in.size;
376 switch (in.opcode) {
378 if (in.cdw10 == 0x0000001) // Identify controller
379 break;
380 if (in.cdw10 == 0x0000000) { // Identify namespace
381 if (in.nsid == 1)
382 break;
383 return set_err(ENOSYS, "NVMe Identify Namespace 0x%x not supported", in.nsid);
384 }
385 return set_err(ENOSYS, "NVMe Identify with CDW10=0x%08x not supported", in.cdw10);
387 if (!(in.nsid == nvme_broadcast_nsid || !in.nsid))
388 return set_err(ENOSYS, "NVMe Get Log Page with NSID=0x%x not supported", in.nsid);
389 if (size > 0x200) { // Reading more apparently returns old data from previous command
390 // TODO: Add ability to return short reads to caller
391 size = 0x200;
392 pout("Warning: NVMe Get Log truncated to 0x%03x bytes, 0x%03x bytes zero filled\n", size, in.size - size);
393 }
394 break;
395 default:
396 return set_err(ENOSYS, "NVMe admin command 0x%02x not supported", in.opcode);
397 break;
398 }
399 if (in.cdw11 || in.cdw12 || in.cdw13 || in.cdw14 || in.cdw15)
400 return set_err(ENOSYS, "Nonzero NVMe command dwords 11-15 not supported");
401
402 uint8_t cdb[16] = {0, };
403 cdb[0] = 0xe4;
405 cdb[3] = in.opcode;
406 cdb[4] = (uint8_t)in.cdw10;
407
408 scsi_cmnd_io io_hdr = {};
409 io_hdr.cmnd = cdb;
410 io_hdr.cmnd_len = sizeof(cdb);
411 io_hdr.dxfer_dir = DXFER_FROM_DEVICE;
412 io_hdr.dxferp = (uint8_t *)in.buffer;
413 io_hdr.dxfer_len = size;
414 memset(in.buffer, 0, in.size);
415
416 scsi_device * scsidev = get_tunnel_dev();
417 if (!scsidev->scsi_pass_through_and_check(&io_hdr, "sntrealtek_device::nvme_pass_through: "))
418 return set_err(scsidev->get_err());
419
420 //out.result = ?; // TODO
421 return true;
422}
423
424
425} // namespace snt
426
427using namespace snt;
428
430{
431 if (!scsidev)
432 throw std::logic_error("smart_interface: get_snt_device() called with scsidev=0");
433
434 // Check for "snt*/sat"
435 bool maybe_sat = false;
436 char snt_type[32];
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;
441 maybe_sat = true;
442 }
443
444 // Take temporary ownership of 'scsidev' to delete it on error
445 scsi_device_auto_ptr scsidev_holder(scsidev);
446 nvme_device * sntdev = nullptr;
447
448 if (!strcmp(snt_type, "sntasmedia")) {
449 // No namespace supported
450 sntdev = new sntasmedia_device(this, scsidev, type, nvme_broadcast_nsid, maybe_sat);
451 }
452
453 else if (str_starts_with(snt_type, "sntjmicron")) {
454 int n1 = -1, n2 = -1;
455 unsigned nsid = nvme_broadcast_nsid;
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);
459 sntdev = new sntjmicron_device(this, scsidev, type, nsid, maybe_sat);
460 }
461
462 else if (!strcmp(snt_type, "sntrealtek")) {
463 // No namespace supported
464 sntdev = new sntrealtek_device(this, scsidev, type, nvme_broadcast_nsid, maybe_sat);
465 }
466
467 else {
468 return set_err_np(EINVAL, "Unknown SNT device type '%s'", type);
469 }
470
471 // 'scsidev' is now owned by 'sntdev'
472 scsidev_holder.release();
473 return sntdev;
474}
Smart pointer class for device pointers.
device_type * release()
Return the pointer and release ownership.
ATA device access.
NVMe device access.
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.
SCSI device access.
bool scsi_pass_through_and_check(scsi_cmnd_io *iop, const char *msg="")
Base class for all devices.
Definition: dev_interface.h:33
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'.
Definition: scsinvme.cpp:429
virtual ata_device * get_sat_device(const char *type, scsi_device *scsidev)
Return ATA->SCSI filter for a SAT or USB 'type'.
Definition: scsiata.cpp:1405
nvme_or_sat_device(scsi_device *scsidev, unsigned nsid, bool maybe_sat)
Definition: scsinvme.cpp:46
virtual smart_device * autodetect_open() override
Open device with autodetection support.
Definition: scsinvme.cpp:53
sntasmedia_device(smart_interface *intf, scsi_device *scsidev, const char *req_type, unsigned nsid, bool maybe_sat)
Definition: scsinvme.cpp:90
virtual bool nvme_pass_through(const nvme_cmd_in &in, nvme_cmd_out &out) override
NVMe pass through.
Definition: scsinvme.cpp:102
virtual ~sntasmedia_device()
Definition: scsinvme.cpp:98
virtual bool nvme_pass_through(const nvme_cmd_in &in, nvme_cmd_out &out) override
NVMe pass through.
Definition: scsinvme.cpp:209
virtual ~sntjmicron_device()
Definition: scsinvme.cpp:193
sntjmicron_device(smart_interface *intf, scsi_device *scsidev, const char *req_type, unsigned nsid, bool maybe_sat)
Definition: scsinvme.cpp:185
sntrealtek_device(smart_interface *intf, scsi_device *scsidev, const char *req_type, unsigned nsid, bool maybe_sat)
Definition: scsinvme.cpp:361
virtual ~sntrealtek_device()
Definition: scsinvme.cpp:369
virtual bool nvme_pass_through(const nvme_cmd_in &in, nvme_cmd_out &out) override
NVMe pass through.
Definition: scsinvme.cpp:373
virtual bool open() override
Open device, return false on error.
Implement a device by tunneling through another device.
Definition: dev_tunnelled.h:56
virtual void release(const smart_device *dev) override
Definition: dev_tunnelled.h:76
u8 cdb[16]
Definition: megaraid.h:21
u32 size
Definition: megaraid.h:0
@ nvme_admin_identify
Definition: nvmecmds.h:210
@ nvme_admin_get_log_page
Definition: nvmecmds.h:207
Definition: scsinvme.cpp:26
uint32_t cdw10
uint32_t nsid
bool nvme_read_id_ctrl(nvme_device *device, nvme_id_ctrl &id_ctrl)
Definition: nvmecmds.cpp:132
constexpr uint32_t nvme_broadcast_nsid
Definition: nvmecmds.h:257
#define SAT_ATA_PASSTHROUGH_12
Definition: scsicmds.h:101
#define DXFER_NONE
Definition: scsicmds.h:108
#define DXFER_FROM_DEVICE
Definition: scsicmds.h:109
#define DXFER_TO_DEVICE
Definition: scsicmds.h:110
#define SNT_JMICRON_CDB_LEN
Definition: scsinvme.cpp:164
#define SNT_JMICRON_NVM_CMD_LEN
Definition: scsinvme.cpp:165
const char * scsinvme_cpp_svnid
Definition: scsinvme.cpp:23
#define SNT_JMICRON_NVME_SIGNATURE
Definition: scsinvme.cpp:163
static void sg_put_unaligned_be24(uint32_t val, void *p)
Definition: sg_unaligned.h:364
static void sg_put_unaligned_le16(uint16_t val, void *p)
Definition: sg_unaligned.h:309
void pout(const char *fmt,...)
Definition: smartd.cpp:1335
NVMe pass through input parameters.
unsigned char direction() const
Get I/O direction from opcode.
unsigned cdw10
unsigned cdw13
unsigned cdw11
unsigned char opcode
Opcode (CDW0 07:00)
unsigned size
Size of buffer.
unsigned cdw14
unsigned cdw15
Cmd specific.
unsigned nsid
Namespace ID.
unsigned cdw12
void * buffer
Pointer to data buffer.
NVMe pass through output parameters.
unsigned result
Command specific result (DW0)
uint8_t * dxferp
Definition: scsicmds.h:121
int dxfer_dir
Definition: scsicmds.h:119
size_t cmnd_len
Definition: scsicmds.h:118
size_t dxfer_len
Definition: scsicmds.h:122
uint8_t * cmnd
Definition: scsicmds.h:117
std::string info_name
Informal name.
Definition: dev_interface.h:46
std::string strprintf(const char *fmt,...)
Definition: utility.cpp:799
bool isbigendian()
Definition: utility.h:82
void swapx(unsigned short *p)
Definition: utility.h:95
bool str_starts_with(const char *str, const char *prefix)
Definition: utility.h:52