Book a Demo!
CoCalc Logo Icon
StoreFeaturesDocsShareSupportNewsAboutPoliciesSign UpSign In
torvalds
GitHub Repository: torvalds/linux
Path: blob/master/net/ethtool/cmis_cdb.c
26285 views
1
// SPDX-License-Identifier: GPL-2.0-only
2
3
#include <linux/ethtool.h>
4
#include <linux/jiffies.h>
5
6
#include "common.h"
7
#include "module_fw.h"
8
#include "cmis.h"
9
10
/* For accessing the LPL field on page 9Fh, the allowable length extension is
11
* min(i, 15) byte octets where i specifies the allowable additional number of
12
* byte octets in a READ or a WRITE.
13
*/
14
u32 ethtool_cmis_get_max_lpl_size(u8 num_of_byte_octs)
15
{
16
return 8 * (1 + min_t(u8, num_of_byte_octs, 15));
17
}
18
19
void ethtool_cmis_cdb_compose_args(struct ethtool_cmis_cdb_cmd_args *args,
20
enum ethtool_cmis_cdb_cmd_id cmd, u8 *lpl,
21
u8 lpl_len, u8 *epl, u16 epl_len,
22
u16 max_duration, u8 read_write_len_ext,
23
u16 msleep_pre_rpl, u8 rpl_exp_len, u8 flags)
24
{
25
args->req.id = cpu_to_be16(cmd);
26
args->req.lpl_len = lpl_len;
27
if (lpl)
28
memcpy(args->req.payload, lpl, args->req.lpl_len);
29
if (epl) {
30
args->req.epl_len = cpu_to_be16(epl_len);
31
args->req.epl = epl;
32
}
33
34
args->max_duration = max_duration;
35
args->read_write_len_ext =
36
ethtool_cmis_get_max_lpl_size(read_write_len_ext);
37
args->msleep_pre_rpl = msleep_pre_rpl;
38
args->rpl_exp_len = rpl_exp_len;
39
args->flags = flags;
40
args->err_msg = NULL;
41
}
42
43
void ethtool_cmis_page_init(struct ethtool_module_eeprom *page_data,
44
u8 page, u32 offset, u32 length)
45
{
46
page_data->page = page;
47
page_data->offset = offset;
48
page_data->length = length;
49
page_data->i2c_address = ETHTOOL_CMIS_CDB_PAGE_I2C_ADDR;
50
}
51
52
#define CMIS_REVISION_PAGE 0x00
53
#define CMIS_REVISION_OFFSET 0x01
54
55
struct cmis_rev_rpl {
56
u8 rev;
57
};
58
59
static u8 cmis_rev_rpl_major(struct cmis_rev_rpl *rpl)
60
{
61
return rpl->rev >> 4;
62
}
63
64
static int cmis_rev_major_get(struct net_device *dev, u8 *rev_major)
65
{
66
const struct ethtool_ops *ops = dev->ethtool_ops;
67
struct ethtool_module_eeprom page_data = {0};
68
struct netlink_ext_ack extack = {};
69
struct cmis_rev_rpl rpl = {};
70
int err;
71
72
ethtool_cmis_page_init(&page_data, CMIS_REVISION_PAGE,
73
CMIS_REVISION_OFFSET, sizeof(rpl));
74
page_data.data = (u8 *)&rpl;
75
76
err = ops->get_module_eeprom_by_page(dev, &page_data, &extack);
77
if (err < 0) {
78
if (extack._msg)
79
netdev_err(dev, "%s\n", extack._msg);
80
return err;
81
}
82
83
*rev_major = cmis_rev_rpl_major(&rpl);
84
85
return 0;
86
}
87
88
#define CMIS_CDB_ADVERTISEMENT_PAGE 0x01
89
#define CMIS_CDB_ADVERTISEMENT_OFFSET 0xA3
90
91
/* Based on section 8.4.11 "CDB Messaging Support Advertisement" in CMIS
92
* standard revision 5.2.
93
*/
94
struct cmis_cdb_advert_rpl {
95
u8 inst_supported;
96
u8 read_write_len_ext;
97
u8 resv1;
98
u8 resv2;
99
};
100
101
static u8 cmis_cdb_advert_rpl_inst_supported(struct cmis_cdb_advert_rpl *rpl)
102
{
103
return rpl->inst_supported >> 6;
104
}
105
106
static int cmis_cdb_advertisement_get(struct ethtool_cmis_cdb *cdb,
107
struct net_device *dev,
108
struct ethnl_module_fw_flash_ntf_params *ntf_params)
109
{
110
const struct ethtool_ops *ops = dev->ethtool_ops;
111
struct ethtool_module_eeprom page_data = {};
112
struct cmis_cdb_advert_rpl rpl = {};
113
struct netlink_ext_ack extack = {};
114
int err;
115
116
ethtool_cmis_page_init(&page_data, CMIS_CDB_ADVERTISEMENT_PAGE,
117
CMIS_CDB_ADVERTISEMENT_OFFSET, sizeof(rpl));
118
page_data.data = (u8 *)&rpl;
119
120
err = ops->get_module_eeprom_by_page(dev, &page_data, &extack);
121
if (err < 0) {
122
if (extack._msg)
123
netdev_err(dev, "%s\n", extack._msg);
124
return err;
125
}
126
127
if (!cmis_cdb_advert_rpl_inst_supported(&rpl)) {
128
ethnl_module_fw_flash_ntf_err(dev, ntf_params,
129
"CDB functionality is not supported",
130
NULL);
131
return -EOPNOTSUPP;
132
}
133
134
cdb->read_write_len_ext = rpl.read_write_len_ext;
135
136
return 0;
137
}
138
139
#define CMIS_PASSWORD_ENTRY_PAGE 0x00
140
#define CMIS_PASSWORD_ENTRY_OFFSET 0x7A
141
142
struct cmis_password_entry_pl {
143
__be32 password;
144
};
145
146
/* See section 9.3.1 "CMD 0000h: Query Status" in CMIS standard revision 5.2.
147
* struct cmis_cdb_query_status_pl and struct cmis_cdb_query_status_rpl are
148
* structured layouts of the flat arrays,
149
* struct ethtool_cmis_cdb_request::payload and
150
* struct ethtool_cmis_cdb_rpl::payload respectively.
151
*/
152
struct cmis_cdb_query_status_pl {
153
u16 response_delay;
154
};
155
156
struct cmis_cdb_query_status_rpl {
157
u8 length;
158
u8 status;
159
};
160
161
static int
162
cmis_cdb_validate_password(struct ethtool_cmis_cdb *cdb,
163
struct net_device *dev,
164
const struct ethtool_module_fw_flash_params *params,
165
struct ethnl_module_fw_flash_ntf_params *ntf_params)
166
{
167
const struct ethtool_ops *ops = dev->ethtool_ops;
168
struct cmis_cdb_query_status_pl qs_pl = {0};
169
struct ethtool_module_eeprom page_data = {};
170
struct ethtool_cmis_cdb_cmd_args args = {};
171
struct cmis_password_entry_pl pe_pl = {};
172
struct cmis_cdb_query_status_rpl *rpl;
173
struct netlink_ext_ack extack = {};
174
int err;
175
176
ethtool_cmis_page_init(&page_data, CMIS_PASSWORD_ENTRY_PAGE,
177
CMIS_PASSWORD_ENTRY_OFFSET, sizeof(pe_pl));
178
page_data.data = (u8 *)&pe_pl;
179
180
pe_pl = *((struct cmis_password_entry_pl *)page_data.data);
181
pe_pl.password = params->password;
182
err = ops->set_module_eeprom_by_page(dev, &page_data, &extack);
183
if (err < 0) {
184
if (extack._msg)
185
netdev_err(dev, "%s\n", extack._msg);
186
return err;
187
}
188
189
ethtool_cmis_cdb_compose_args(&args, ETHTOOL_CMIS_CDB_CMD_QUERY_STATUS,
190
(u8 *)&qs_pl, sizeof(qs_pl), NULL, 0, 0,
191
cdb->read_write_len_ext, 1000,
192
sizeof(*rpl),
193
CDB_F_COMPLETION_VALID | CDB_F_STATUS_VALID);
194
195
err = ethtool_cmis_cdb_execute_cmd(dev, &args);
196
if (err < 0) {
197
ethnl_module_fw_flash_ntf_err(dev, ntf_params,
198
"Query Status command failed",
199
args.err_msg);
200
return err;
201
}
202
203
rpl = (struct cmis_cdb_query_status_rpl *)args.req.payload;
204
if (!rpl->length || !rpl->status) {
205
ethnl_module_fw_flash_ntf_err(dev, ntf_params,
206
"Password was not accepted",
207
NULL);
208
return -EINVAL;
209
}
210
211
return 0;
212
}
213
214
/* Some CDB commands asserts the CDB completion flag only from CMIS
215
* revision 5. Therefore, check the relevant validity flag only when
216
* the revision supports it.
217
*/
218
void ethtool_cmis_cdb_check_completion_flag(u8 cmis_rev, u8 *flags)
219
{
220
*flags |= cmis_rev >= 5 ? CDB_F_COMPLETION_VALID : 0;
221
}
222
223
#define CMIS_CDB_MODULE_FEATURES_RESV_DATA 34
224
225
/* See section 9.4.1 "CMD 0040h: Module Features" in CMIS standard revision 5.2.
226
* struct cmis_cdb_module_features_rpl is structured layout of the flat
227
* array, ethtool_cmis_cdb_rpl::payload.
228
*/
229
struct cmis_cdb_module_features_rpl {
230
u8 resv1[CMIS_CDB_MODULE_FEATURES_RESV_DATA];
231
__be16 max_completion_time;
232
};
233
234
static u16
235
cmis_cdb_module_features_completion_time(struct cmis_cdb_module_features_rpl *rpl)
236
{
237
return be16_to_cpu(rpl->max_completion_time);
238
}
239
240
static int cmis_cdb_module_features_get(struct ethtool_cmis_cdb *cdb,
241
struct net_device *dev,
242
struct ethnl_module_fw_flash_ntf_params *ntf_params)
243
{
244
struct ethtool_cmis_cdb_cmd_args args = {};
245
struct cmis_cdb_module_features_rpl *rpl;
246
u8 flags = CDB_F_STATUS_VALID;
247
int err;
248
249
ethtool_cmis_cdb_check_completion_flag(cdb->cmis_rev, &flags);
250
ethtool_cmis_cdb_compose_args(&args,
251
ETHTOOL_CMIS_CDB_CMD_MODULE_FEATURES,
252
NULL, 0, NULL, 0, 0,
253
cdb->read_write_len_ext, 1000,
254
sizeof(*rpl), flags);
255
256
err = ethtool_cmis_cdb_execute_cmd(dev, &args);
257
if (err < 0) {
258
ethnl_module_fw_flash_ntf_err(dev, ntf_params,
259
"Module Features command failed",
260
args.err_msg);
261
return err;
262
}
263
264
rpl = (struct cmis_cdb_module_features_rpl *)args.req.payload;
265
cdb->max_completion_time =
266
cmis_cdb_module_features_completion_time(rpl);
267
268
return 0;
269
}
270
271
struct ethtool_cmis_cdb *
272
ethtool_cmis_cdb_init(struct net_device *dev,
273
const struct ethtool_module_fw_flash_params *params,
274
struct ethnl_module_fw_flash_ntf_params *ntf_params)
275
{
276
struct ethtool_cmis_cdb *cdb;
277
int err;
278
279
cdb = kzalloc(sizeof(*cdb), GFP_KERNEL);
280
if (!cdb)
281
return ERR_PTR(-ENOMEM);
282
283
err = cmis_rev_major_get(dev, &cdb->cmis_rev);
284
if (err < 0)
285
goto err;
286
287
if (cdb->cmis_rev < 4) {
288
ethnl_module_fw_flash_ntf_err(dev, ntf_params,
289
"CMIS revision doesn't support module firmware flashing",
290
NULL);
291
err = -EOPNOTSUPP;
292
goto err;
293
}
294
295
err = cmis_cdb_advertisement_get(cdb, dev, ntf_params);
296
if (err < 0)
297
goto err;
298
299
if (params->password_valid) {
300
err = cmis_cdb_validate_password(cdb, dev, params, ntf_params);
301
if (err < 0)
302
goto err;
303
}
304
305
err = cmis_cdb_module_features_get(cdb, dev, ntf_params);
306
if (err < 0)
307
goto err;
308
309
return cdb;
310
311
err:
312
ethtool_cmis_cdb_fini(cdb);
313
return ERR_PTR(err);
314
}
315
316
void ethtool_cmis_cdb_fini(struct ethtool_cmis_cdb *cdb)
317
{
318
kfree(cdb);
319
}
320
321
static bool is_completed(u8 data)
322
{
323
return !!(data & 0x40);
324
}
325
326
#define CMIS_CDB_STATUS_SUCCESS 0x01
327
328
static bool status_success(u8 data)
329
{
330
return data == CMIS_CDB_STATUS_SUCCESS;
331
}
332
333
#define CMIS_CDB_STATUS_FAIL 0x40
334
335
static bool status_fail(u8 data)
336
{
337
return data & CMIS_CDB_STATUS_FAIL;
338
}
339
340
struct cmis_wait_for_cond_rpl {
341
u8 state;
342
};
343
344
static int
345
ethtool_cmis_module_poll(struct net_device *dev,
346
struct cmis_wait_for_cond_rpl *rpl, u32 offset,
347
bool (*cond_success)(u8), bool (*cond_fail)(u8))
348
{
349
const struct ethtool_ops *ops = dev->ethtool_ops;
350
struct ethtool_module_eeprom page_data = {0};
351
struct netlink_ext_ack extack = {};
352
int err;
353
354
ethtool_cmis_page_init(&page_data, 0, offset, sizeof(*rpl));
355
page_data.data = (u8 *)rpl;
356
357
err = ops->get_module_eeprom_by_page(dev, &page_data, &extack);
358
if (err < 0) {
359
if (extack._msg)
360
netdev_err_once(dev, "%s\n", extack._msg);
361
return -EBUSY;
362
}
363
364
if ((*cond_success)(rpl->state))
365
return 0;
366
367
if (*cond_fail && (*cond_fail)(rpl->state))
368
return -EIO;
369
370
return -EBUSY;
371
}
372
373
int ethtool_cmis_wait_for_cond(struct net_device *dev, u8 flags, u8 flag,
374
u16 max_duration, u32 offset,
375
bool (*cond_success)(u8), bool (*cond_fail)(u8),
376
u8 *state)
377
{
378
struct cmis_wait_for_cond_rpl rpl = {};
379
unsigned long end;
380
int err;
381
382
if (!(flags & flag))
383
return 0;
384
385
if (max_duration == 0)
386
max_duration = U16_MAX;
387
388
end = jiffies + msecs_to_jiffies(max_duration);
389
do {
390
err = ethtool_cmis_module_poll(dev, &rpl, offset, cond_success,
391
cond_fail);
392
if (err != -EBUSY)
393
goto out;
394
395
msleep(20);
396
} while (time_before(jiffies, end));
397
398
err = ethtool_cmis_module_poll(dev, &rpl, offset, cond_success,
399
cond_fail);
400
if (err == -EBUSY)
401
err = -ETIMEDOUT;
402
403
out:
404
*state = rpl.state;
405
return err;
406
}
407
408
#define CMIS_CDB_COMPLETION_FLAG_OFFSET 0x08
409
410
static int cmis_cdb_wait_for_completion(struct net_device *dev,
411
struct ethtool_cmis_cdb_cmd_args *args)
412
{
413
u8 flag;
414
int err;
415
416
/* Some vendors demand waiting time before checking completion flag
417
* in some CDB commands.
418
*/
419
msleep(args->msleep_pre_rpl);
420
421
err = ethtool_cmis_wait_for_cond(dev, args->flags,
422
CDB_F_COMPLETION_VALID,
423
args->max_duration,
424
CMIS_CDB_COMPLETION_FLAG_OFFSET,
425
is_completed, NULL, &flag);
426
if (err < 0)
427
args->err_msg = "Completion Flag did not set on time";
428
429
return err;
430
}
431
432
#define CMIS_CDB_STATUS_OFFSET 0x25
433
434
static void cmis_cdb_status_fail_msg_get(u8 status, char **err_msg)
435
{
436
switch (status) {
437
case 0b10000001:
438
*err_msg = "CDB Status is in progress: Busy capturing command";
439
break;
440
case 0b10000010:
441
*err_msg =
442
"CDB Status is in progress: Busy checking/validating command";
443
break;
444
case 0b10000011:
445
*err_msg = "CDB Status is in progress: Busy executing";
446
break;
447
case 0b01000000:
448
*err_msg = "CDB status failed: no specific failure";
449
break;
450
case 0b01000010:
451
*err_msg =
452
"CDB status failed: Parameter range error or parameter not supported";
453
break;
454
case 0b01000101:
455
*err_msg = "CDB status failed: CdbChkCode error";
456
break;
457
case 0b01000110:
458
*err_msg = "CDB status failed: Password error";
459
break;
460
default:
461
*err_msg = "Unknown failure reason";
462
}
463
};
464
465
static int cmis_cdb_wait_for_status(struct net_device *dev,
466
struct ethtool_cmis_cdb_cmd_args *args)
467
{
468
u8 status;
469
int err;
470
471
/* Some vendors demand waiting time before checking status in some
472
* CDB commands.
473
*/
474
msleep(args->msleep_pre_rpl);
475
476
err = ethtool_cmis_wait_for_cond(dev, args->flags, CDB_F_STATUS_VALID,
477
args->max_duration,
478
CMIS_CDB_STATUS_OFFSET,
479
status_success, status_fail, &status);
480
if (err < 0 && !args->err_msg)
481
cmis_cdb_status_fail_msg_get(status, &args->err_msg);
482
483
return err;
484
}
485
486
#define CMIS_CDB_REPLY_OFFSET 0x86
487
488
static int cmis_cdb_process_reply(struct net_device *dev,
489
struct ethtool_module_eeprom *page_data,
490
struct ethtool_cmis_cdb_cmd_args *args)
491
{
492
u8 rpl_hdr_len = sizeof(struct ethtool_cmis_cdb_rpl_hdr);
493
u8 rpl_exp_len = args->rpl_exp_len + rpl_hdr_len;
494
const struct ethtool_ops *ops = dev->ethtool_ops;
495
struct netlink_ext_ack extack = {};
496
struct ethtool_cmis_cdb_rpl *rpl;
497
int err;
498
499
if (!args->rpl_exp_len)
500
return 0;
501
502
ethtool_cmis_page_init(page_data, ETHTOOL_CMIS_CDB_CMD_PAGE,
503
CMIS_CDB_REPLY_OFFSET, rpl_exp_len);
504
page_data->data = kmalloc(page_data->length, GFP_KERNEL);
505
if (!page_data->data)
506
return -ENOMEM;
507
508
err = ops->get_module_eeprom_by_page(dev, page_data, &extack);
509
if (err < 0) {
510
if (extack._msg)
511
netdev_err(dev, "%s\n", extack._msg);
512
goto out;
513
}
514
515
rpl = (struct ethtool_cmis_cdb_rpl *)page_data->data;
516
if ((args->rpl_exp_len > rpl->hdr.rpl_len + rpl_hdr_len) ||
517
!rpl->hdr.rpl_chk_code) {
518
err = -EIO;
519
goto out;
520
}
521
522
args->req.lpl_len = rpl->hdr.rpl_len;
523
memcpy(args->req.payload, rpl->payload, args->req.lpl_len);
524
525
out:
526
kfree(page_data->data);
527
return err;
528
}
529
530
static int
531
__ethtool_cmis_cdb_execute_cmd(struct net_device *dev,
532
struct ethtool_module_eeprom *page_data,
533
u8 page, u32 offset, u32 length, void *data)
534
{
535
const struct ethtool_ops *ops = dev->ethtool_ops;
536
struct netlink_ext_ack extack = {};
537
int err;
538
539
ethtool_cmis_page_init(page_data, page, offset, length);
540
page_data->data = kmemdup(data, page_data->length, GFP_KERNEL);
541
if (!page_data->data)
542
return -ENOMEM;
543
544
err = ops->set_module_eeprom_by_page(dev, page_data, &extack);
545
if (err < 0) {
546
if (extack._msg)
547
netdev_err(dev, "%s\n", extack._msg);
548
}
549
550
kfree(page_data->data);
551
return err;
552
}
553
554
#define CMIS_CDB_EPL_PAGE_START 0xA0
555
#define CMIS_CDB_EPL_PAGE_END 0xAF
556
#define CMIS_CDB_EPL_FW_BLOCK_OFFSET_START 128
557
#define CMIS_CDB_EPL_FW_BLOCK_OFFSET_END 255
558
559
static int
560
ethtool_cmis_cdb_execute_epl_cmd(struct net_device *dev,
561
struct ethtool_cmis_cdb_cmd_args *args,
562
struct ethtool_module_eeprom *page_data)
563
{
564
u16 epl_len = be16_to_cpu(args->req.epl_len);
565
u32 bytes_written = 0;
566
u8 page;
567
int err;
568
569
for (page = CMIS_CDB_EPL_PAGE_START;
570
page <= CMIS_CDB_EPL_PAGE_END && bytes_written < epl_len; page++) {
571
u16 offset = CMIS_CDB_EPL_FW_BLOCK_OFFSET_START;
572
573
while (offset <= CMIS_CDB_EPL_FW_BLOCK_OFFSET_END &&
574
bytes_written < epl_len) {
575
u32 bytes_left = epl_len - bytes_written;
576
u16 space_left, bytes_to_write;
577
578
space_left = CMIS_CDB_EPL_FW_BLOCK_OFFSET_END - offset + 1;
579
bytes_to_write = min_t(u16, bytes_left,
580
min_t(u16, space_left,
581
args->read_write_len_ext));
582
583
err = __ethtool_cmis_cdb_execute_cmd(dev, page_data,
584
page, offset,
585
bytes_to_write,
586
args->req.epl + bytes_written);
587
if (err < 0)
588
return err;
589
590
offset += bytes_to_write;
591
bytes_written += bytes_to_write;
592
}
593
}
594
return 0;
595
}
596
597
static u8 cmis_cdb_calc_checksum(const void *data, size_t size)
598
{
599
const u8 *bytes = (const u8 *)data;
600
u8 checksum = 0;
601
602
for (size_t i = 0; i < size; i++)
603
checksum += bytes[i];
604
605
return ~checksum;
606
}
607
608
#define CMIS_CDB_CMD_ID_OFFSET 0x80
609
610
int ethtool_cmis_cdb_execute_cmd(struct net_device *dev,
611
struct ethtool_cmis_cdb_cmd_args *args)
612
{
613
struct ethtool_module_eeprom page_data = {};
614
u32 offset;
615
int err;
616
617
args->req.chk_code =
618
cmis_cdb_calc_checksum(&args->req,
619
offsetof(struct ethtool_cmis_cdb_request,
620
epl));
621
622
if (args->req.lpl_len > args->read_write_len_ext) {
623
args->err_msg = "LPL length is longer than CDB read write length extension allows";
624
return -EINVAL;
625
}
626
627
/* According to the CMIS standard, there are two options to trigger the
628
* CDB commands. The default option is triggering the command by writing
629
* the CMDID bytes. Therefore, the command will be split to 2 calls:
630
* First, with everything except the CMDID field and then the CMDID
631
* field.
632
*/
633
offset = CMIS_CDB_CMD_ID_OFFSET +
634
offsetof(struct ethtool_cmis_cdb_request, body);
635
err = __ethtool_cmis_cdb_execute_cmd(dev, &page_data,
636
ETHTOOL_CMIS_CDB_CMD_PAGE, offset,
637
sizeof(args->req.body),
638
&args->req.body);
639
if (err < 0)
640
return err;
641
642
if (args->req.epl_len) {
643
err = ethtool_cmis_cdb_execute_epl_cmd(dev, args, &page_data);
644
if (err < 0)
645
return err;
646
}
647
648
offset = CMIS_CDB_CMD_ID_OFFSET +
649
offsetof(struct ethtool_cmis_cdb_request, id);
650
err = __ethtool_cmis_cdb_execute_cmd(dev, &page_data,
651
ETHTOOL_CMIS_CDB_CMD_PAGE, offset,
652
sizeof(args->req.id),
653
&args->req.id);
654
if (err < 0)
655
return err;
656
657
err = cmis_cdb_wait_for_completion(dev, args);
658
if (err < 0)
659
return err;
660
661
err = cmis_cdb_wait_for_status(dev, args);
662
if (err < 0)
663
return err;
664
665
return cmis_cdb_process_reply(dev, &page_data, args);
666
}
667
668