-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathnvme_util.h
309 lines (272 loc) · 8.86 KB
/
nvme_util.h
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
#include <string>
#include <stdio.h>
#include <stdlib.h>
#include <stdint.h>
#ifndef NVME_UTIL_H
#define NVME_UTIL_H
class nvme_Device
{
public:
nvme_Device();
nvme_Device(const char * dev_name, const char * req_type, unsigned nsid);
struct device_info {
device_info()
{ }
device_info(const char * d_name, const char * d_type, const char * r_type)
: dev_name(d_name), info_name(d_name),
dev_type(d_type), req_type(r_type)
{ }
std::string dev_name; ///< Device (path)name
std::string info_name; ///< Informal name
std::string dev_type; ///< Actual device type
std::string req_type; ///< Device type requested by user, empty if none
};
device_info m_info;
struct nvme_cmd_in
{
unsigned char opcode; ///< Opcode (CDW0 07:00)
unsigned nsid; ///< Namespace ID
unsigned cdw10, cdw11, cdw12, cdw13, cdw14, cdw15; ///< Cmd specific
void * buffer; ///< Pointer to data buffer
unsigned size; ///< Size of buffer
enum {
no_data = 0x0, data_out = 0x1, data_in = 0x2, data_io = 0x3
};
/// Get I/O direction from opcode
unsigned char direction() const
{ return (opcode & 0x3); }
// Prepare for DATA IN command
void set_data_in(unsigned char op, void * buf, unsigned sz)
{
opcode = op;
if (direction() != data_in)
{
// QMessageBox::information(NULL, "Error", "invalid opcode", NULL);
}
buffer = buf;
size = sz;
}
nvme_cmd_in()
: opcode(0), nsid(0),
cdw10(0), cdw11(0), cdw12(0), cdw13(0), cdw14(0), cdw15(0),
buffer(0), size(0)
{ }
};
/// NVMe pass through output parameters
struct nvme_cmd_out
{
unsigned result; ///< Command specific result (DW0)
unsigned short status; ///< Status Field (DW3 31:17)
bool status_valid; ///< true if status is valid
nvme_cmd_out()
: result(0), status(0), status_valid(false)
{ }
};
bool nvme_pass_through(const nvme_cmd_in & in);
unsigned m_nsid;
struct nvme_id_power_state {
unsigned short max_power; // centiwatts
unsigned char rsvd2;
unsigned char flags;
unsigned int entry_lat; // microseconds
unsigned int exit_lat; // microseconds
unsigned char read_tput;
unsigned char read_lat;
unsigned char write_tput;
unsigned char write_lat;
unsigned short idle_power;
unsigned char idle_scale;
unsigned char rsvd19;
unsigned short active_power;
unsigned char active_work_scale;
unsigned char rsvd23[9];
};
struct nvme_id_ctrl {
unsigned short vid;
unsigned short ssvid;
char sn[20];
char mn[40];
char fr[8];
unsigned char rab;
unsigned char ieee[3];
unsigned char cmic;
unsigned char mdts;
unsigned short cntlid;
unsigned int ver;
unsigned int rtd3r;
unsigned int rtd3e;
unsigned int oaes;
unsigned char rsvd96[160];
unsigned short oacs;
unsigned char acl;
unsigned char aerl;
unsigned char frmw;
unsigned char lpa;
unsigned char elpe;
unsigned char npss;
unsigned char avscc;
unsigned char apsta;
unsigned short wctemp;
unsigned short cctemp;
unsigned short mtfa;
unsigned int hmpre;
unsigned int hmmin;
unsigned char tnvmcap[16];
unsigned char unvmcap[16];
unsigned int rpmbs;
unsigned char rsvd316[196];
unsigned char sqes;
unsigned char cqes;
unsigned char rsvd514[2];
unsigned int nn;
unsigned short oncs;
unsigned short fuses;
unsigned char fna;
unsigned char vwc;
unsigned short awun;
unsigned short awupf;
unsigned char nvscc;
unsigned char rsvd531;
unsigned short acwu;
unsigned char rsvd534[2];
unsigned int sgls;
unsigned char rsvd540[1508];
struct nvme_id_power_state psd[32];
unsigned char vs[1024];
};
struct nvme_smart_log {
unsigned char critical_warning;
unsigned char temperature[2];
unsigned char avail_spare;
unsigned char spare_thresh;
unsigned char percent_used;
unsigned char rsvd6[26];
unsigned char data_units_read[16];
unsigned char data_units_written[16];
unsigned char host_reads[16];
unsigned char host_writes[16];
unsigned char ctrl_busy_time[16];
unsigned char power_cycles[16];
unsigned char power_on_hours[16];
unsigned char unsafe_shutdowns[16];
unsigned char media_errors[16];
unsigned char num_err_log_entries[16];
unsigned int warning_temp_time;
unsigned int critical_comp_time;
unsigned short temp_sensor[8];
unsigned char rsvd216[296];
};
struct nvme_badblock {
char badblock[1];
};
struct nvme_WAI_Info {
unsigned char wai[4096];
};
struct nvme_lbaf {
unsigned short ms;
unsigned char ds;
unsigned char rp;
};
struct nvme_id_ns {
uint64_t nsze;
uint64_t ncap;
uint64_t nuse;
unsigned char nsfeat;
unsigned char nlbaf;
unsigned char flbas;
unsigned char mc;
unsigned char dpc;
unsigned char dps;
unsigned char nmic;
unsigned char rescap;
unsigned char fpi;
unsigned char rsvd33;
unsigned short nawun;
unsigned short nawupf;
unsigned short nacwu;
unsigned short nabsn;
unsigned short nabo;
unsigned short nabspf;
unsigned char rsvd46[2];
unsigned char nvmcap[16];
unsigned char rsvd64[40];
unsigned char nguid[16];
unsigned char eui64[8];
struct nvme_lbaf lbaf[16];
unsigned char rsvd192[192];
unsigned char vs[3712];
};
struct nvme_print_options
{
bool drive_info;
bool drive_capabilities;
bool smart_check_status;
bool smart_vendor_attrib;
unsigned error_log_entries;
unsigned char log_page;
unsigned log_page_size;
nvme_print_options()
: drive_info(false),
drive_capabilities(false),
smart_check_status(false),
smart_vendor_attrib(false),
error_log_entries(0),
log_page(0),
log_page_size(0)
{ }
};
nvme_print_options nvmeopts;
int myOpen();
bool nvme_read_id_ctrl(nvme_id_ctrl & id_ctrl);
bool nvme_read_identify(unsigned nsid,
unsigned char cns, void * data, unsigned size);
bool nvme_read_badblock( void * data, unsigned size);
bool nvme_read_WAI( void * data, unsigned size);
bool nvme_read_smart_log(nvme_smart_log & smart_log);
bool nvme_read_log_page(unsigned char lid, void * data, unsigned size);
void swap2(char *location);
void swap4(char *location);
void swap8(char *location);
// Typesafe variants using overloading
inline void swapx(unsigned short * p)
{ swap2((char*)p); }
inline void swapx(unsigned int * p)
{ swap4((char*)p); }
inline void swapx(uint64_t * p)
{ swap8((char*)p); }
//void print_drive_info(const nvme_id_ctrl & id_ctrl, const nvme_id_ns & id_ns, unsigned nsid, bool show_all);
bool nvme_read_id_ns(unsigned nsid, nvme_id_ns & id_ns);
private:
int m_fd; ///< filedesc, -1 if not open.
int m_flags; ///< Flags for ::open()
int m_retry_flags; ///< Flags to retry ::open(), -1 if no retry
};
inline bool isbigendian()
{
#ifdef WORDS_BIGENDIAN
return true;
#else
return false;
#endif
}
enum nvme_admin_opcode {
//nvme_admin_delete_sq = 0x00,
//nvme_admin_create_sq = 0x01,
nvme_admin_get_log_page = 0x02,
//nvme_admin_delete_cq = 0x04,
//nvme_admin_create_cq = 0x05,
nvme_admin_identify = 0x06,
//nvme_admin_abort_cmd = 0x08,
//nvme_admin_set_features = 0x09,
//nvme_admin_get_features = 0x0a,
//nvme_admin_async_event = 0x0c,
//nvme_admin_ns_mgmt = 0x0d,
//nvme_admin_activate_fw = 0x10,
//nvme_admin_download_fw = 0x11,
//nvme_admin_ns_attach = 0x15,
//nvme_admin_format_nvm = 0x80,
//nvme_admin_security_send = 0x81,
//nvme_admin_security_recv = 0x82,
nvme_read_bad_block = 0xC2,
};
#endif // NVME_UTIL_H