1 // SPDX-License-Identifier: GPL-2.0-only 2 // Copyright(c) 2021 Intel Corporation. All rights reserved. 3 4 #include <linux/platform_device.h> 5 #include <linux/mod_devicetable.h> 6 #include <linux/vmalloc.h> 7 #include <linux/module.h> 8 #include <linux/delay.h> 9 #include <linux/sizes.h> 10 #include <linux/bits.h> 11 #include <cxl/mailbox.h> 12 #include <asm/unaligned.h> 13 #include <crypto/sha2.h> 14 #include <cxlmem.h> 15 16 #include "trace.h" 17 18 #define LSA_SIZE SZ_128K 19 #define FW_SIZE SZ_64M 20 #define FW_SLOTS 3 21 #define DEV_SIZE SZ_2G 22 #define EFFECT(x) (1U << x) 23 24 #define MOCK_INJECT_DEV_MAX 8 25 #define MOCK_INJECT_TEST_MAX 128 26 27 static unsigned int poison_inject_dev_max = MOCK_INJECT_DEV_MAX; 28 29 enum cxl_command_effects { 30 CONF_CHANGE_COLD_RESET = 0, 31 CONF_CHANGE_IMMEDIATE, 32 DATA_CHANGE_IMMEDIATE, 33 POLICY_CHANGE_IMMEDIATE, 34 LOG_CHANGE_IMMEDIATE, 35 SECURITY_CHANGE_IMMEDIATE, 36 BACKGROUND_OP, 37 SECONDARY_MBOX_SUPPORTED, 38 }; 39 40 #define CXL_CMD_EFFECT_NONE cpu_to_le16(0) 41 42 static struct cxl_cel_entry mock_cel[] = { 43 { 44 .opcode = cpu_to_le16(CXL_MBOX_OP_GET_SUPPORTED_LOGS), 45 .effect = CXL_CMD_EFFECT_NONE, 46 }, 47 { 48 .opcode = cpu_to_le16(CXL_MBOX_OP_IDENTIFY), 49 .effect = CXL_CMD_EFFECT_NONE, 50 }, 51 { 52 .opcode = cpu_to_le16(CXL_MBOX_OP_GET_LSA), 53 .effect = CXL_CMD_EFFECT_NONE, 54 }, 55 { 56 .opcode = cpu_to_le16(CXL_MBOX_OP_GET_PARTITION_INFO), 57 .effect = CXL_CMD_EFFECT_NONE, 58 }, 59 { 60 .opcode = cpu_to_le16(CXL_MBOX_OP_SET_LSA), 61 .effect = cpu_to_le16(EFFECT(CONF_CHANGE_IMMEDIATE) | 62 EFFECT(DATA_CHANGE_IMMEDIATE)), 63 }, 64 { 65 .opcode = cpu_to_le16(CXL_MBOX_OP_GET_HEALTH_INFO), 66 .effect = CXL_CMD_EFFECT_NONE, 67 }, 68 { 69 .opcode = cpu_to_le16(CXL_MBOX_OP_GET_POISON), 70 .effect = CXL_CMD_EFFECT_NONE, 71 }, 72 { 73 .opcode = cpu_to_le16(CXL_MBOX_OP_INJECT_POISON), 74 .effect = cpu_to_le16(EFFECT(DATA_CHANGE_IMMEDIATE)), 75 }, 76 { 77 .opcode = cpu_to_le16(CXL_MBOX_OP_CLEAR_POISON), 78 .effect = cpu_to_le16(EFFECT(DATA_CHANGE_IMMEDIATE)), 79 }, 80 { 81 .opcode = cpu_to_le16(CXL_MBOX_OP_GET_FW_INFO), 82 .effect = CXL_CMD_EFFECT_NONE, 83 }, 84 { 85 .opcode = cpu_to_le16(CXL_MBOX_OP_TRANSFER_FW), 86 .effect = cpu_to_le16(EFFECT(CONF_CHANGE_COLD_RESET) | 87 EFFECT(BACKGROUND_OP)), 88 }, 89 { 90 .opcode = cpu_to_le16(CXL_MBOX_OP_ACTIVATE_FW), 91 .effect = cpu_to_le16(EFFECT(CONF_CHANGE_COLD_RESET) | 92 EFFECT(CONF_CHANGE_IMMEDIATE)), 93 }, 94 { 95 .opcode = cpu_to_le16(CXL_MBOX_OP_SANITIZE), 96 .effect = cpu_to_le16(EFFECT(DATA_CHANGE_IMMEDIATE) | 97 EFFECT(SECURITY_CHANGE_IMMEDIATE) | 98 EFFECT(BACKGROUND_OP)), 99 }, 100 }; 101 102 /* See CXL 2.0 Table 181 Get Health Info Output Payload */ 103 struct cxl_mbox_health_info { 104 u8 health_status; 105 u8 media_status; 106 u8 ext_status; 107 u8 life_used; 108 __le16 temperature; 109 __le32 dirty_shutdowns; 110 __le32 volatile_errors; 111 __le32 pmem_errors; 112 } __packed; 113 114 static struct { 115 struct cxl_mbox_get_supported_logs gsl; 116 struct cxl_gsl_entry entry; 117 } mock_gsl_payload = { 118 .gsl = { 119 .entries = cpu_to_le16(1), 120 }, 121 .entry = { 122 .uuid = DEFINE_CXL_CEL_UUID, 123 .size = cpu_to_le32(sizeof(mock_cel)), 124 }, 125 }; 126 127 #define PASS_TRY_LIMIT 3 128 129 #define CXL_TEST_EVENT_CNT_MAX 15 130 131 /* Set a number of events to return at a time for simulation. */ 132 #define CXL_TEST_EVENT_RET_MAX 4 133 134 struct mock_event_log { 135 u16 clear_idx; 136 u16 cur_idx; 137 u16 nr_events; 138 u16 nr_overflow; 139 u16 overflow_reset; 140 struct cxl_event_record_raw *events[CXL_TEST_EVENT_CNT_MAX]; 141 }; 142 143 struct mock_event_store { 144 struct mock_event_log mock_logs[CXL_EVENT_TYPE_MAX]; 145 u32 ev_status; 146 }; 147 148 struct cxl_mockmem_data { 149 void *lsa; 150 void *fw; 151 int fw_slot; 152 int fw_staged; 153 size_t fw_size; 154 u32 security_state; 155 u8 user_pass[NVDIMM_PASSPHRASE_LEN]; 156 u8 master_pass[NVDIMM_PASSPHRASE_LEN]; 157 int user_limit; 158 int master_limit; 159 struct mock_event_store mes; 160 struct cxl_memdev_state *mds; 161 u8 event_buf[SZ_4K]; 162 u64 timestamp; 163 unsigned long sanitize_timeout; 164 }; 165 166 static struct mock_event_log *event_find_log(struct device *dev, int log_type) 167 { 168 struct cxl_mockmem_data *mdata = dev_get_drvdata(dev); 169 170 if (log_type >= CXL_EVENT_TYPE_MAX) 171 return NULL; 172 return &mdata->mes.mock_logs[log_type]; 173 } 174 175 static struct cxl_event_record_raw *event_get_current(struct mock_event_log *log) 176 { 177 return log->events[log->cur_idx]; 178 } 179 180 static void event_reset_log(struct mock_event_log *log) 181 { 182 log->cur_idx = 0; 183 log->clear_idx = 0; 184 log->nr_overflow = log->overflow_reset; 185 } 186 187 /* Handle can never be 0 use 1 based indexing for handle */ 188 static u16 event_get_clear_handle(struct mock_event_log *log) 189 { 190 return log->clear_idx + 1; 191 } 192 193 /* Handle can never be 0 use 1 based indexing for handle */ 194 static __le16 event_get_cur_event_handle(struct mock_event_log *log) 195 { 196 u16 cur_handle = log->cur_idx + 1; 197 198 return cpu_to_le16(cur_handle); 199 } 200 201 static bool event_log_empty(struct mock_event_log *log) 202 { 203 return log->cur_idx == log->nr_events; 204 } 205 206 static void mes_add_event(struct mock_event_store *mes, 207 enum cxl_event_log_type log_type, 208 struct cxl_event_record_raw *event) 209 { 210 struct mock_event_log *log; 211 212 if (WARN_ON(log_type >= CXL_EVENT_TYPE_MAX)) 213 return; 214 215 log = &mes->mock_logs[log_type]; 216 217 if ((log->nr_events + 1) > CXL_TEST_EVENT_CNT_MAX) { 218 log->nr_overflow++; 219 log->overflow_reset = log->nr_overflow; 220 return; 221 } 222 223 log->events[log->nr_events] = event; 224 log->nr_events++; 225 } 226 227 /* 228 * Vary the number of events returned to simulate events occuring while the 229 * logs are being read. 230 */ 231 static int ret_limit = 0; 232 233 static int mock_get_event(struct device *dev, struct cxl_mbox_cmd *cmd) 234 { 235 struct cxl_get_event_payload *pl; 236 struct mock_event_log *log; 237 u16 nr_overflow; 238 u8 log_type; 239 int i; 240 241 if (cmd->size_in != sizeof(log_type)) 242 return -EINVAL; 243 244 ret_limit = (ret_limit + 1) % CXL_TEST_EVENT_RET_MAX; 245 if (!ret_limit) 246 ret_limit = 1; 247 248 if (cmd->size_out < struct_size(pl, records, ret_limit)) 249 return -EINVAL; 250 251 log_type = *((u8 *)cmd->payload_in); 252 if (log_type >= CXL_EVENT_TYPE_MAX) 253 return -EINVAL; 254 255 memset(cmd->payload_out, 0, struct_size(pl, records, 0)); 256 257 log = event_find_log(dev, log_type); 258 if (!log || event_log_empty(log)) 259 return 0; 260 261 pl = cmd->payload_out; 262 263 for (i = 0; i < ret_limit && !event_log_empty(log); i++) { 264 memcpy(&pl->records[i], event_get_current(log), 265 sizeof(pl->records[i])); 266 pl->records[i].event.generic.hdr.handle = 267 event_get_cur_event_handle(log); 268 log->cur_idx++; 269 } 270 271 cmd->size_out = struct_size(pl, records, i); 272 pl->record_count = cpu_to_le16(i); 273 if (!event_log_empty(log)) 274 pl->flags |= CXL_GET_EVENT_FLAG_MORE_RECORDS; 275 276 if (log->nr_overflow) { 277 u64 ns; 278 279 pl->flags |= CXL_GET_EVENT_FLAG_OVERFLOW; 280 pl->overflow_err_count = cpu_to_le16(nr_overflow); 281 ns = ktime_get_real_ns(); 282 ns -= 5000000000; /* 5s ago */ 283 pl->first_overflow_timestamp = cpu_to_le64(ns); 284 ns = ktime_get_real_ns(); 285 ns -= 1000000000; /* 1s ago */ 286 pl->last_overflow_timestamp = cpu_to_le64(ns); 287 } 288 289 return 0; 290 } 291 292 static int mock_clear_event(struct device *dev, struct cxl_mbox_cmd *cmd) 293 { 294 struct cxl_mbox_clear_event_payload *pl = cmd->payload_in; 295 struct mock_event_log *log; 296 u8 log_type = pl->event_log; 297 u16 handle; 298 int nr; 299 300 if (log_type >= CXL_EVENT_TYPE_MAX) 301 return -EINVAL; 302 303 log = event_find_log(dev, log_type); 304 if (!log) 305 return 0; /* No mock data in this log */ 306 307 /* 308 * This check is technically not invalid per the specification AFAICS. 309 * (The host could 'guess' handles and clear them in order). 310 * However, this is not good behavior for the host so test it. 311 */ 312 if (log->clear_idx + pl->nr_recs > log->cur_idx) { 313 dev_err(dev, 314 "Attempting to clear more events than returned!\n"); 315 return -EINVAL; 316 } 317 318 /* Check handle order prior to clearing events */ 319 for (nr = 0, handle = event_get_clear_handle(log); 320 nr < pl->nr_recs; 321 nr++, handle++) { 322 if (handle != le16_to_cpu(pl->handles[nr])) { 323 dev_err(dev, "Clearing events out of order\n"); 324 return -EINVAL; 325 } 326 } 327 328 if (log->nr_overflow) 329 log->nr_overflow = 0; 330 331 /* Clear events */ 332 log->clear_idx += pl->nr_recs; 333 return 0; 334 } 335 336 static void cxl_mock_event_trigger(struct device *dev) 337 { 338 struct cxl_mockmem_data *mdata = dev_get_drvdata(dev); 339 struct mock_event_store *mes = &mdata->mes; 340 int i; 341 342 for (i = CXL_EVENT_TYPE_INFO; i < CXL_EVENT_TYPE_MAX; i++) { 343 struct mock_event_log *log; 344 345 log = event_find_log(dev, i); 346 if (log) 347 event_reset_log(log); 348 } 349 350 cxl_mem_get_event_records(mdata->mds, mes->ev_status); 351 } 352 353 struct cxl_event_record_raw maint_needed = { 354 .id = UUID_INIT(0xBA5EBA11, 0xABCD, 0xEFEB, 355 0xa5, 0x5a, 0xa5, 0x5a, 0xa5, 0xa5, 0x5a, 0xa5), 356 .event.generic = { 357 .hdr = { 358 .length = sizeof(struct cxl_event_record_raw), 359 .flags[0] = CXL_EVENT_RECORD_FLAG_MAINT_NEEDED, 360 /* .handle = Set dynamically */ 361 .related_handle = cpu_to_le16(0xa5b6), 362 }, 363 .data = { 0xDE, 0xAD, 0xBE, 0xEF }, 364 }, 365 }; 366 367 struct cxl_event_record_raw hardware_replace = { 368 .id = UUID_INIT(0xABCDEFEB, 0xBA11, 0xBA5E, 369 0xa5, 0x5a, 0xa5, 0x5a, 0xa5, 0xa5, 0x5a, 0xa5), 370 .event.generic = { 371 .hdr = { 372 .length = sizeof(struct cxl_event_record_raw), 373 .flags[0] = CXL_EVENT_RECORD_FLAG_HW_REPLACE, 374 /* .handle = Set dynamically */ 375 .related_handle = cpu_to_le16(0xb6a5), 376 }, 377 .data = { 0xDE, 0xAD, 0xBE, 0xEF }, 378 }, 379 }; 380 381 struct cxl_test_gen_media { 382 uuid_t id; 383 struct cxl_event_gen_media rec; 384 } __packed; 385 386 struct cxl_test_gen_media gen_media = { 387 .id = CXL_EVENT_GEN_MEDIA_UUID, 388 .rec = { 389 .media_hdr = { 390 .hdr = { 391 .length = sizeof(struct cxl_test_gen_media), 392 .flags[0] = CXL_EVENT_RECORD_FLAG_PERMANENT, 393 /* .handle = Set dynamically */ 394 .related_handle = cpu_to_le16(0), 395 }, 396 .phys_addr = cpu_to_le64(0x2000), 397 .descriptor = CXL_GMER_EVT_DESC_UNCORECTABLE_EVENT, 398 .type = CXL_GMER_MEM_EVT_TYPE_DATA_PATH_ERROR, 399 .transaction_type = CXL_GMER_TRANS_HOST_WRITE, 400 /* .validity_flags = <set below> */ 401 .channel = 1, 402 .rank = 30, 403 }, 404 }, 405 }; 406 407 struct cxl_test_dram { 408 uuid_t id; 409 struct cxl_event_dram rec; 410 } __packed; 411 412 struct cxl_test_dram dram = { 413 .id = CXL_EVENT_DRAM_UUID, 414 .rec = { 415 .media_hdr = { 416 .hdr = { 417 .length = sizeof(struct cxl_test_dram), 418 .flags[0] = CXL_EVENT_RECORD_FLAG_PERF_DEGRADED, 419 /* .handle = Set dynamically */ 420 .related_handle = cpu_to_le16(0), 421 }, 422 .phys_addr = cpu_to_le64(0x8000), 423 .descriptor = CXL_GMER_EVT_DESC_THRESHOLD_EVENT, 424 .type = CXL_GMER_MEM_EVT_TYPE_INV_ADDR, 425 .transaction_type = CXL_GMER_TRANS_INTERNAL_MEDIA_SCRUB, 426 /* .validity_flags = <set below> */ 427 .channel = 1, 428 }, 429 .bank_group = 5, 430 .bank = 2, 431 .column = {0xDE, 0xAD}, 432 }, 433 }; 434 435 struct cxl_test_mem_module { 436 uuid_t id; 437 struct cxl_event_mem_module rec; 438 } __packed; 439 440 struct cxl_test_mem_module mem_module = { 441 .id = CXL_EVENT_MEM_MODULE_UUID, 442 .rec = { 443 .hdr = { 444 .length = sizeof(struct cxl_test_mem_module), 445 /* .handle = Set dynamically */ 446 .related_handle = cpu_to_le16(0), 447 }, 448 .event_type = CXL_MMER_TEMP_CHANGE, 449 .info = { 450 .health_status = CXL_DHI_HS_PERFORMANCE_DEGRADED, 451 .media_status = CXL_DHI_MS_ALL_DATA_LOST, 452 .add_status = (CXL_DHI_AS_CRITICAL << 2) | 453 (CXL_DHI_AS_WARNING << 4) | 454 (CXL_DHI_AS_WARNING << 5), 455 .device_temp = { 0xDE, 0xAD}, 456 .dirty_shutdown_cnt = { 0xde, 0xad, 0xbe, 0xef }, 457 .cor_vol_err_cnt = { 0xde, 0xad, 0xbe, 0xef }, 458 .cor_per_err_cnt = { 0xde, 0xad, 0xbe, 0xef }, 459 } 460 }, 461 }; 462 463 static int mock_set_timestamp(struct cxl_dev_state *cxlds, 464 struct cxl_mbox_cmd *cmd) 465 { 466 struct cxl_mockmem_data *mdata = dev_get_drvdata(cxlds->dev); 467 struct cxl_mbox_set_timestamp_in *ts = cmd->payload_in; 468 469 if (cmd->size_in != sizeof(*ts)) 470 return -EINVAL; 471 472 if (cmd->size_out != 0) 473 return -EINVAL; 474 475 mdata->timestamp = le64_to_cpu(ts->timestamp); 476 return 0; 477 } 478 479 static void cxl_mock_add_event_logs(struct mock_event_store *mes) 480 { 481 put_unaligned_le16(CXL_GMER_VALID_CHANNEL | CXL_GMER_VALID_RANK, 482 &gen_media.rec.media_hdr.validity_flags); 483 484 put_unaligned_le16(CXL_DER_VALID_CHANNEL | CXL_DER_VALID_BANK_GROUP | 485 CXL_DER_VALID_BANK | CXL_DER_VALID_COLUMN, 486 &dram.rec.media_hdr.validity_flags); 487 488 mes_add_event(mes, CXL_EVENT_TYPE_INFO, &maint_needed); 489 mes_add_event(mes, CXL_EVENT_TYPE_INFO, 490 (struct cxl_event_record_raw *)&gen_media); 491 mes_add_event(mes, CXL_EVENT_TYPE_INFO, 492 (struct cxl_event_record_raw *)&mem_module); 493 mes->ev_status |= CXLDEV_EVENT_STATUS_INFO; 494 495 mes_add_event(mes, CXL_EVENT_TYPE_FAIL, &maint_needed); 496 mes_add_event(mes, CXL_EVENT_TYPE_FAIL, &hardware_replace); 497 mes_add_event(mes, CXL_EVENT_TYPE_FAIL, 498 (struct cxl_event_record_raw *)&dram); 499 mes_add_event(mes, CXL_EVENT_TYPE_FAIL, 500 (struct cxl_event_record_raw *)&gen_media); 501 mes_add_event(mes, CXL_EVENT_TYPE_FAIL, 502 (struct cxl_event_record_raw *)&mem_module); 503 mes_add_event(mes, CXL_EVENT_TYPE_FAIL, &hardware_replace); 504 mes_add_event(mes, CXL_EVENT_TYPE_FAIL, 505 (struct cxl_event_record_raw *)&dram); 506 /* Overflow this log */ 507 mes_add_event(mes, CXL_EVENT_TYPE_FAIL, &hardware_replace); 508 mes_add_event(mes, CXL_EVENT_TYPE_FAIL, &hardware_replace); 509 mes_add_event(mes, CXL_EVENT_TYPE_FAIL, &hardware_replace); 510 mes_add_event(mes, CXL_EVENT_TYPE_FAIL, &hardware_replace); 511 mes_add_event(mes, CXL_EVENT_TYPE_FAIL, &hardware_replace); 512 mes_add_event(mes, CXL_EVENT_TYPE_FAIL, &hardware_replace); 513 mes_add_event(mes, CXL_EVENT_TYPE_FAIL, &hardware_replace); 514 mes_add_event(mes, CXL_EVENT_TYPE_FAIL, &hardware_replace); 515 mes_add_event(mes, CXL_EVENT_TYPE_FAIL, &hardware_replace); 516 mes_add_event(mes, CXL_EVENT_TYPE_FAIL, &hardware_replace); 517 mes->ev_status |= CXLDEV_EVENT_STATUS_FAIL; 518 519 mes_add_event(mes, CXL_EVENT_TYPE_FATAL, &hardware_replace); 520 mes_add_event(mes, CXL_EVENT_TYPE_FATAL, 521 (struct cxl_event_record_raw *)&dram); 522 mes->ev_status |= CXLDEV_EVENT_STATUS_FATAL; 523 } 524 525 static int mock_gsl(struct cxl_mbox_cmd *cmd) 526 { 527 if (cmd->size_out < sizeof(mock_gsl_payload)) 528 return -EINVAL; 529 530 memcpy(cmd->payload_out, &mock_gsl_payload, sizeof(mock_gsl_payload)); 531 cmd->size_out = sizeof(mock_gsl_payload); 532 533 return 0; 534 } 535 536 static int mock_get_log(struct cxl_memdev_state *mds, struct cxl_mbox_cmd *cmd) 537 { 538 struct cxl_mailbox *cxl_mbox = &mds->cxlds.cxl_mbox; 539 struct cxl_mbox_get_log *gl = cmd->payload_in; 540 u32 offset = le32_to_cpu(gl->offset); 541 u32 length = le32_to_cpu(gl->length); 542 uuid_t uuid = DEFINE_CXL_CEL_UUID; 543 void *data = &mock_cel; 544 545 if (cmd->size_in < sizeof(*gl)) 546 return -EINVAL; 547 if (length > cxl_mbox->payload_size) 548 return -EINVAL; 549 if (offset + length > sizeof(mock_cel)) 550 return -EINVAL; 551 if (!uuid_equal(&gl->uuid, &uuid)) 552 return -EINVAL; 553 if (length > cmd->size_out) 554 return -EINVAL; 555 556 memcpy(cmd->payload_out, data + offset, length); 557 558 return 0; 559 } 560 561 static int mock_rcd_id(struct cxl_mbox_cmd *cmd) 562 { 563 struct cxl_mbox_identify id = { 564 .fw_revision = { "mock fw v1 " }, 565 .total_capacity = 566 cpu_to_le64(DEV_SIZE / CXL_CAPACITY_MULTIPLIER), 567 .volatile_capacity = 568 cpu_to_le64(DEV_SIZE / CXL_CAPACITY_MULTIPLIER), 569 }; 570 571 if (cmd->size_out < sizeof(id)) 572 return -EINVAL; 573 574 memcpy(cmd->payload_out, &id, sizeof(id)); 575 576 return 0; 577 } 578 579 static int mock_id(struct cxl_mbox_cmd *cmd) 580 { 581 struct cxl_mbox_identify id = { 582 .fw_revision = { "mock fw v1 " }, 583 .lsa_size = cpu_to_le32(LSA_SIZE), 584 .partition_align = 585 cpu_to_le64(SZ_256M / CXL_CAPACITY_MULTIPLIER), 586 .total_capacity = 587 cpu_to_le64(DEV_SIZE / CXL_CAPACITY_MULTIPLIER), 588 .inject_poison_limit = cpu_to_le16(MOCK_INJECT_TEST_MAX), 589 }; 590 591 put_unaligned_le24(CXL_POISON_LIST_MAX, id.poison_list_max_mer); 592 593 if (cmd->size_out < sizeof(id)) 594 return -EINVAL; 595 596 memcpy(cmd->payload_out, &id, sizeof(id)); 597 598 return 0; 599 } 600 601 static int mock_partition_info(struct cxl_mbox_cmd *cmd) 602 { 603 struct cxl_mbox_get_partition_info pi = { 604 .active_volatile_cap = 605 cpu_to_le64(DEV_SIZE / 2 / CXL_CAPACITY_MULTIPLIER), 606 .active_persistent_cap = 607 cpu_to_le64(DEV_SIZE / 2 / CXL_CAPACITY_MULTIPLIER), 608 }; 609 610 if (cmd->size_out < sizeof(pi)) 611 return -EINVAL; 612 613 memcpy(cmd->payload_out, &pi, sizeof(pi)); 614 615 return 0; 616 } 617 618 void cxl_mockmem_sanitize_work(struct work_struct *work) 619 { 620 struct cxl_memdev_state *mds = 621 container_of(work, typeof(*mds), security.poll_dwork.work); 622 struct cxl_mailbox *cxl_mbox = &mds->cxlds.cxl_mbox; 623 624 mutex_lock(&cxl_mbox->mbox_mutex); 625 if (mds->security.sanitize_node) 626 sysfs_notify_dirent(mds->security.sanitize_node); 627 mds->security.sanitize_active = false; 628 mutex_unlock(&cxl_mbox->mbox_mutex); 629 630 dev_dbg(mds->cxlds.dev, "sanitize complete\n"); 631 } 632 633 static int mock_sanitize(struct cxl_mockmem_data *mdata, 634 struct cxl_mbox_cmd *cmd) 635 { 636 struct cxl_memdev_state *mds = mdata->mds; 637 struct cxl_mailbox *cxl_mbox = &mds->cxlds.cxl_mbox; 638 int rc = 0; 639 640 if (cmd->size_in != 0) 641 return -EINVAL; 642 643 if (cmd->size_out != 0) 644 return -EINVAL; 645 646 if (mdata->security_state & CXL_PMEM_SEC_STATE_USER_PASS_SET) { 647 cmd->return_code = CXL_MBOX_CMD_RC_SECURITY; 648 return -ENXIO; 649 } 650 if (mdata->security_state & CXL_PMEM_SEC_STATE_LOCKED) { 651 cmd->return_code = CXL_MBOX_CMD_RC_SECURITY; 652 return -ENXIO; 653 } 654 655 mutex_lock(&cxl_mbox->mbox_mutex); 656 if (schedule_delayed_work(&mds->security.poll_dwork, 657 msecs_to_jiffies(mdata->sanitize_timeout))) { 658 mds->security.sanitize_active = true; 659 dev_dbg(mds->cxlds.dev, "sanitize issued\n"); 660 } else 661 rc = -EBUSY; 662 mutex_unlock(&cxl_mbox->mbox_mutex); 663 664 return rc; 665 } 666 667 static int mock_secure_erase(struct cxl_mockmem_data *mdata, 668 struct cxl_mbox_cmd *cmd) 669 { 670 if (cmd->size_in != 0) 671 return -EINVAL; 672 673 if (cmd->size_out != 0) 674 return -EINVAL; 675 676 if (mdata->security_state & CXL_PMEM_SEC_STATE_USER_PASS_SET) { 677 cmd->return_code = CXL_MBOX_CMD_RC_SECURITY; 678 return -ENXIO; 679 } 680 681 if (mdata->security_state & CXL_PMEM_SEC_STATE_LOCKED) { 682 cmd->return_code = CXL_MBOX_CMD_RC_SECURITY; 683 return -ENXIO; 684 } 685 686 return 0; 687 } 688 689 static int mock_get_security_state(struct cxl_mockmem_data *mdata, 690 struct cxl_mbox_cmd *cmd) 691 { 692 if (cmd->size_in) 693 return -EINVAL; 694 695 if (cmd->size_out != sizeof(u32)) 696 return -EINVAL; 697 698 memcpy(cmd->payload_out, &mdata->security_state, sizeof(u32)); 699 700 return 0; 701 } 702 703 static void master_plimit_check(struct cxl_mockmem_data *mdata) 704 { 705 if (mdata->master_limit == PASS_TRY_LIMIT) 706 return; 707 mdata->master_limit++; 708 if (mdata->master_limit == PASS_TRY_LIMIT) 709 mdata->security_state |= CXL_PMEM_SEC_STATE_MASTER_PLIMIT; 710 } 711 712 static void user_plimit_check(struct cxl_mockmem_data *mdata) 713 { 714 if (mdata->user_limit == PASS_TRY_LIMIT) 715 return; 716 mdata->user_limit++; 717 if (mdata->user_limit == PASS_TRY_LIMIT) 718 mdata->security_state |= CXL_PMEM_SEC_STATE_USER_PLIMIT; 719 } 720 721 static int mock_set_passphrase(struct cxl_mockmem_data *mdata, 722 struct cxl_mbox_cmd *cmd) 723 { 724 struct cxl_set_pass *set_pass; 725 726 if (cmd->size_in != sizeof(*set_pass)) 727 return -EINVAL; 728 729 if (cmd->size_out != 0) 730 return -EINVAL; 731 732 if (mdata->security_state & CXL_PMEM_SEC_STATE_FROZEN) { 733 cmd->return_code = CXL_MBOX_CMD_RC_SECURITY; 734 return -ENXIO; 735 } 736 737 set_pass = cmd->payload_in; 738 switch (set_pass->type) { 739 case CXL_PMEM_SEC_PASS_MASTER: 740 if (mdata->security_state & CXL_PMEM_SEC_STATE_MASTER_PLIMIT) { 741 cmd->return_code = CXL_MBOX_CMD_RC_SECURITY; 742 return -ENXIO; 743 } 744 /* 745 * CXL spec rev3.0 8.2.9.8.6.2, The master pasphrase shall only be set in 746 * the security disabled state when the user passphrase is not set. 747 */ 748 if (mdata->security_state & CXL_PMEM_SEC_STATE_USER_PASS_SET) { 749 cmd->return_code = CXL_MBOX_CMD_RC_SECURITY; 750 return -ENXIO; 751 } 752 if (memcmp(mdata->master_pass, set_pass->old_pass, NVDIMM_PASSPHRASE_LEN)) { 753 master_plimit_check(mdata); 754 cmd->return_code = CXL_MBOX_CMD_RC_PASSPHRASE; 755 return -ENXIO; 756 } 757 memcpy(mdata->master_pass, set_pass->new_pass, NVDIMM_PASSPHRASE_LEN); 758 mdata->security_state |= CXL_PMEM_SEC_STATE_MASTER_PASS_SET; 759 return 0; 760 761 case CXL_PMEM_SEC_PASS_USER: 762 if (mdata->security_state & CXL_PMEM_SEC_STATE_USER_PLIMIT) { 763 cmd->return_code = CXL_MBOX_CMD_RC_SECURITY; 764 return -ENXIO; 765 } 766 if (memcmp(mdata->user_pass, set_pass->old_pass, NVDIMM_PASSPHRASE_LEN)) { 767 user_plimit_check(mdata); 768 cmd->return_code = CXL_MBOX_CMD_RC_PASSPHRASE; 769 return -ENXIO; 770 } 771 memcpy(mdata->user_pass, set_pass->new_pass, NVDIMM_PASSPHRASE_LEN); 772 mdata->security_state |= CXL_PMEM_SEC_STATE_USER_PASS_SET; 773 return 0; 774 775 default: 776 cmd->return_code = CXL_MBOX_CMD_RC_INPUT; 777 } 778 return -EINVAL; 779 } 780 781 static int mock_disable_passphrase(struct cxl_mockmem_data *mdata, 782 struct cxl_mbox_cmd *cmd) 783 { 784 struct cxl_disable_pass *dis_pass; 785 786 if (cmd->size_in != sizeof(*dis_pass)) 787 return -EINVAL; 788 789 if (cmd->size_out != 0) 790 return -EINVAL; 791 792 if (mdata->security_state & CXL_PMEM_SEC_STATE_FROZEN) { 793 cmd->return_code = CXL_MBOX_CMD_RC_SECURITY; 794 return -ENXIO; 795 } 796 797 dis_pass = cmd->payload_in; 798 switch (dis_pass->type) { 799 case CXL_PMEM_SEC_PASS_MASTER: 800 if (mdata->security_state & CXL_PMEM_SEC_STATE_MASTER_PLIMIT) { 801 cmd->return_code = CXL_MBOX_CMD_RC_SECURITY; 802 return -ENXIO; 803 } 804 805 if (!(mdata->security_state & CXL_PMEM_SEC_STATE_MASTER_PASS_SET)) { 806 cmd->return_code = CXL_MBOX_CMD_RC_SECURITY; 807 return -ENXIO; 808 } 809 810 if (memcmp(dis_pass->pass, mdata->master_pass, NVDIMM_PASSPHRASE_LEN)) { 811 master_plimit_check(mdata); 812 cmd->return_code = CXL_MBOX_CMD_RC_PASSPHRASE; 813 return -ENXIO; 814 } 815 816 mdata->master_limit = 0; 817 memset(mdata->master_pass, 0, NVDIMM_PASSPHRASE_LEN); 818 mdata->security_state &= ~CXL_PMEM_SEC_STATE_MASTER_PASS_SET; 819 return 0; 820 821 case CXL_PMEM_SEC_PASS_USER: 822 if (mdata->security_state & CXL_PMEM_SEC_STATE_USER_PLIMIT) { 823 cmd->return_code = CXL_MBOX_CMD_RC_SECURITY; 824 return -ENXIO; 825 } 826 827 if (!(mdata->security_state & CXL_PMEM_SEC_STATE_USER_PASS_SET)) { 828 cmd->return_code = CXL_MBOX_CMD_RC_SECURITY; 829 return -ENXIO; 830 } 831 832 if (memcmp(dis_pass->pass, mdata->user_pass, NVDIMM_PASSPHRASE_LEN)) { 833 user_plimit_check(mdata); 834 cmd->return_code = CXL_MBOX_CMD_RC_PASSPHRASE; 835 return -ENXIO; 836 } 837 838 mdata->user_limit = 0; 839 memset(mdata->user_pass, 0, NVDIMM_PASSPHRASE_LEN); 840 mdata->security_state &= ~(CXL_PMEM_SEC_STATE_USER_PASS_SET | 841 CXL_PMEM_SEC_STATE_LOCKED); 842 return 0; 843 844 default: 845 cmd->return_code = CXL_MBOX_CMD_RC_INPUT; 846 return -EINVAL; 847 } 848 849 return 0; 850 } 851 852 static int mock_freeze_security(struct cxl_mockmem_data *mdata, 853 struct cxl_mbox_cmd *cmd) 854 { 855 if (cmd->size_in != 0) 856 return -EINVAL; 857 858 if (cmd->size_out != 0) 859 return -EINVAL; 860 861 if (mdata->security_state & CXL_PMEM_SEC_STATE_FROZEN) 862 return 0; 863 864 mdata->security_state |= CXL_PMEM_SEC_STATE_FROZEN; 865 return 0; 866 } 867 868 static int mock_unlock_security(struct cxl_mockmem_data *mdata, 869 struct cxl_mbox_cmd *cmd) 870 { 871 if (cmd->size_in != NVDIMM_PASSPHRASE_LEN) 872 return -EINVAL; 873 874 if (cmd->size_out != 0) 875 return -EINVAL; 876 877 if (mdata->security_state & CXL_PMEM_SEC_STATE_FROZEN) { 878 cmd->return_code = CXL_MBOX_CMD_RC_SECURITY; 879 return -ENXIO; 880 } 881 882 if (!(mdata->security_state & CXL_PMEM_SEC_STATE_USER_PASS_SET)) { 883 cmd->return_code = CXL_MBOX_CMD_RC_SECURITY; 884 return -ENXIO; 885 } 886 887 if (mdata->security_state & CXL_PMEM_SEC_STATE_USER_PLIMIT) { 888 cmd->return_code = CXL_MBOX_CMD_RC_SECURITY; 889 return -ENXIO; 890 } 891 892 if (!(mdata->security_state & CXL_PMEM_SEC_STATE_LOCKED)) { 893 cmd->return_code = CXL_MBOX_CMD_RC_SECURITY; 894 return -ENXIO; 895 } 896 897 if (memcmp(cmd->payload_in, mdata->user_pass, NVDIMM_PASSPHRASE_LEN)) { 898 if (++mdata->user_limit == PASS_TRY_LIMIT) 899 mdata->security_state |= CXL_PMEM_SEC_STATE_USER_PLIMIT; 900 cmd->return_code = CXL_MBOX_CMD_RC_PASSPHRASE; 901 return -ENXIO; 902 } 903 904 mdata->user_limit = 0; 905 mdata->security_state &= ~CXL_PMEM_SEC_STATE_LOCKED; 906 return 0; 907 } 908 909 static int mock_passphrase_secure_erase(struct cxl_mockmem_data *mdata, 910 struct cxl_mbox_cmd *cmd) 911 { 912 struct cxl_pass_erase *erase; 913 914 if (cmd->size_in != sizeof(*erase)) 915 return -EINVAL; 916 917 if (cmd->size_out != 0) 918 return -EINVAL; 919 920 erase = cmd->payload_in; 921 if (mdata->security_state & CXL_PMEM_SEC_STATE_FROZEN) { 922 cmd->return_code = CXL_MBOX_CMD_RC_SECURITY; 923 return -ENXIO; 924 } 925 926 if (mdata->security_state & CXL_PMEM_SEC_STATE_USER_PLIMIT && 927 erase->type == CXL_PMEM_SEC_PASS_USER) { 928 cmd->return_code = CXL_MBOX_CMD_RC_SECURITY; 929 return -ENXIO; 930 } 931 932 if (mdata->security_state & CXL_PMEM_SEC_STATE_MASTER_PLIMIT && 933 erase->type == CXL_PMEM_SEC_PASS_MASTER) { 934 cmd->return_code = CXL_MBOX_CMD_RC_SECURITY; 935 return -ENXIO; 936 } 937 938 switch (erase->type) { 939 case CXL_PMEM_SEC_PASS_MASTER: 940 /* 941 * The spec does not clearly define the behavior of the scenario 942 * where a master passphrase is passed in while the master 943 * passphrase is not set and user passphrase is not set. The 944 * code will take the assumption that it will behave the same 945 * as a CXL secure erase command without passphrase (0x4401). 946 */ 947 if (mdata->security_state & CXL_PMEM_SEC_STATE_MASTER_PASS_SET) { 948 if (memcmp(mdata->master_pass, erase->pass, 949 NVDIMM_PASSPHRASE_LEN)) { 950 master_plimit_check(mdata); 951 cmd->return_code = CXL_MBOX_CMD_RC_PASSPHRASE; 952 return -ENXIO; 953 } 954 mdata->master_limit = 0; 955 mdata->user_limit = 0; 956 mdata->security_state &= ~CXL_PMEM_SEC_STATE_USER_PASS_SET; 957 memset(mdata->user_pass, 0, NVDIMM_PASSPHRASE_LEN); 958 mdata->security_state &= ~CXL_PMEM_SEC_STATE_LOCKED; 959 } else { 960 /* 961 * CXL rev3 8.2.9.8.6.3 Disable Passphrase 962 * When master passphrase is disabled, the device shall 963 * return Invalid Input for the Passphrase Secure Erase 964 * command with master passphrase. 965 */ 966 return -EINVAL; 967 } 968 /* Scramble encryption keys so that data is effectively erased */ 969 break; 970 case CXL_PMEM_SEC_PASS_USER: 971 /* 972 * The spec does not clearly define the behavior of the scenario 973 * where a user passphrase is passed in while the user 974 * passphrase is not set. The code will take the assumption that 975 * it will behave the same as a CXL secure erase command without 976 * passphrase (0x4401). 977 */ 978 if (mdata->security_state & CXL_PMEM_SEC_STATE_USER_PASS_SET) { 979 if (memcmp(mdata->user_pass, erase->pass, 980 NVDIMM_PASSPHRASE_LEN)) { 981 user_plimit_check(mdata); 982 cmd->return_code = CXL_MBOX_CMD_RC_PASSPHRASE; 983 return -ENXIO; 984 } 985 mdata->user_limit = 0; 986 mdata->security_state &= ~CXL_PMEM_SEC_STATE_USER_PASS_SET; 987 memset(mdata->user_pass, 0, NVDIMM_PASSPHRASE_LEN); 988 } 989 990 /* 991 * CXL rev3 Table 8-118 992 * If user passphrase is not set or supported by device, current 993 * passphrase value is ignored. Will make the assumption that 994 * the operation will proceed as secure erase w/o passphrase 995 * since spec is not explicit. 996 */ 997 998 /* Scramble encryption keys so that data is effectively erased */ 999 break; 1000 default: 1001 return -EINVAL; 1002 } 1003 1004 return 0; 1005 } 1006 1007 static int mock_get_lsa(struct cxl_mockmem_data *mdata, 1008 struct cxl_mbox_cmd *cmd) 1009 { 1010 struct cxl_mbox_get_lsa *get_lsa = cmd->payload_in; 1011 void *lsa = mdata->lsa; 1012 u32 offset, length; 1013 1014 if (sizeof(*get_lsa) > cmd->size_in) 1015 return -EINVAL; 1016 offset = le32_to_cpu(get_lsa->offset); 1017 length = le32_to_cpu(get_lsa->length); 1018 if (offset + length > LSA_SIZE) 1019 return -EINVAL; 1020 if (length > cmd->size_out) 1021 return -EINVAL; 1022 1023 memcpy(cmd->payload_out, lsa + offset, length); 1024 return 0; 1025 } 1026 1027 static int mock_set_lsa(struct cxl_mockmem_data *mdata, 1028 struct cxl_mbox_cmd *cmd) 1029 { 1030 struct cxl_mbox_set_lsa *set_lsa = cmd->payload_in; 1031 void *lsa = mdata->lsa; 1032 u32 offset, length; 1033 1034 if (sizeof(*set_lsa) > cmd->size_in) 1035 return -EINVAL; 1036 offset = le32_to_cpu(set_lsa->offset); 1037 length = cmd->size_in - sizeof(*set_lsa); 1038 if (offset + length > LSA_SIZE) 1039 return -EINVAL; 1040 1041 memcpy(lsa + offset, &set_lsa->data[0], length); 1042 return 0; 1043 } 1044 1045 static int mock_health_info(struct cxl_mbox_cmd *cmd) 1046 { 1047 struct cxl_mbox_health_info health_info = { 1048 /* set flags for maint needed, perf degraded, hw replacement */ 1049 .health_status = 0x7, 1050 /* set media status to "All Data Lost" */ 1051 .media_status = 0x3, 1052 /* 1053 * set ext_status flags for: 1054 * ext_life_used: normal, 1055 * ext_temperature: critical, 1056 * ext_corrected_volatile: warning, 1057 * ext_corrected_persistent: normal, 1058 */ 1059 .ext_status = 0x18, 1060 .life_used = 15, 1061 .temperature = cpu_to_le16(25), 1062 .dirty_shutdowns = cpu_to_le32(10), 1063 .volatile_errors = cpu_to_le32(20), 1064 .pmem_errors = cpu_to_le32(30), 1065 }; 1066 1067 if (cmd->size_out < sizeof(health_info)) 1068 return -EINVAL; 1069 1070 memcpy(cmd->payload_out, &health_info, sizeof(health_info)); 1071 return 0; 1072 } 1073 1074 static struct mock_poison { 1075 struct cxl_dev_state *cxlds; 1076 u64 dpa; 1077 } mock_poison_list[MOCK_INJECT_TEST_MAX]; 1078 1079 static struct cxl_mbox_poison_out * 1080 cxl_get_injected_po(struct cxl_dev_state *cxlds, u64 offset, u64 length) 1081 { 1082 struct cxl_mbox_poison_out *po; 1083 int nr_records = 0; 1084 u64 dpa; 1085 1086 po = kzalloc(struct_size(po, record, poison_inject_dev_max), GFP_KERNEL); 1087 if (!po) 1088 return NULL; 1089 1090 for (int i = 0; i < MOCK_INJECT_TEST_MAX; i++) { 1091 if (mock_poison_list[i].cxlds != cxlds) 1092 continue; 1093 if (mock_poison_list[i].dpa < offset || 1094 mock_poison_list[i].dpa > offset + length - 1) 1095 continue; 1096 1097 dpa = mock_poison_list[i].dpa + CXL_POISON_SOURCE_INJECTED; 1098 po->record[nr_records].address = cpu_to_le64(dpa); 1099 po->record[nr_records].length = cpu_to_le32(1); 1100 nr_records++; 1101 if (nr_records == poison_inject_dev_max) 1102 break; 1103 } 1104 1105 /* Always return count, even when zero */ 1106 po->count = cpu_to_le16(nr_records); 1107 1108 return po; 1109 } 1110 1111 static int mock_get_poison(struct cxl_dev_state *cxlds, 1112 struct cxl_mbox_cmd *cmd) 1113 { 1114 struct cxl_mbox_poison_in *pi = cmd->payload_in; 1115 struct cxl_mbox_poison_out *po; 1116 u64 offset = le64_to_cpu(pi->offset); 1117 u64 length = le64_to_cpu(pi->length); 1118 int nr_records; 1119 1120 po = cxl_get_injected_po(cxlds, offset, length); 1121 if (!po) 1122 return -ENOMEM; 1123 nr_records = le16_to_cpu(po->count); 1124 memcpy(cmd->payload_out, po, struct_size(po, record, nr_records)); 1125 cmd->size_out = struct_size(po, record, nr_records); 1126 kfree(po); 1127 1128 return 0; 1129 } 1130 1131 static bool mock_poison_dev_max_injected(struct cxl_dev_state *cxlds) 1132 { 1133 int count = 0; 1134 1135 for (int i = 0; i < MOCK_INJECT_TEST_MAX; i++) { 1136 if (mock_poison_list[i].cxlds == cxlds) 1137 count++; 1138 } 1139 return (count >= poison_inject_dev_max); 1140 } 1141 1142 static int mock_poison_add(struct cxl_dev_state *cxlds, u64 dpa) 1143 { 1144 /* Return EBUSY to match the CXL driver handling */ 1145 if (mock_poison_dev_max_injected(cxlds)) { 1146 dev_dbg(cxlds->dev, 1147 "Device poison injection limit has been reached: %d\n", 1148 poison_inject_dev_max); 1149 return -EBUSY; 1150 } 1151 1152 for (int i = 0; i < MOCK_INJECT_TEST_MAX; i++) { 1153 if (!mock_poison_list[i].cxlds) { 1154 mock_poison_list[i].cxlds = cxlds; 1155 mock_poison_list[i].dpa = dpa; 1156 return 0; 1157 } 1158 } 1159 dev_dbg(cxlds->dev, 1160 "Mock test poison injection limit has been reached: %d\n", 1161 MOCK_INJECT_TEST_MAX); 1162 1163 return -ENXIO; 1164 } 1165 1166 static bool mock_poison_found(struct cxl_dev_state *cxlds, u64 dpa) 1167 { 1168 for (int i = 0; i < MOCK_INJECT_TEST_MAX; i++) { 1169 if (mock_poison_list[i].cxlds == cxlds && 1170 mock_poison_list[i].dpa == dpa) 1171 return true; 1172 } 1173 return false; 1174 } 1175 1176 static int mock_inject_poison(struct cxl_dev_state *cxlds, 1177 struct cxl_mbox_cmd *cmd) 1178 { 1179 struct cxl_mbox_inject_poison *pi = cmd->payload_in; 1180 u64 dpa = le64_to_cpu(pi->address); 1181 1182 if (mock_poison_found(cxlds, dpa)) { 1183 /* Not an error to inject poison if already poisoned */ 1184 dev_dbg(cxlds->dev, "DPA: 0x%llx already poisoned\n", dpa); 1185 return 0; 1186 } 1187 1188 return mock_poison_add(cxlds, dpa); 1189 } 1190 1191 static bool mock_poison_del(struct cxl_dev_state *cxlds, u64 dpa) 1192 { 1193 for (int i = 0; i < MOCK_INJECT_TEST_MAX; i++) { 1194 if (mock_poison_list[i].cxlds == cxlds && 1195 mock_poison_list[i].dpa == dpa) { 1196 mock_poison_list[i].cxlds = NULL; 1197 return true; 1198 } 1199 } 1200 return false; 1201 } 1202 1203 static int mock_clear_poison(struct cxl_dev_state *cxlds, 1204 struct cxl_mbox_cmd *cmd) 1205 { 1206 struct cxl_mbox_clear_poison *pi = cmd->payload_in; 1207 u64 dpa = le64_to_cpu(pi->address); 1208 1209 /* 1210 * A real CXL device will write pi->write_data to the address 1211 * being cleared. In this mock, just delete this address from 1212 * the mock poison list. 1213 */ 1214 if (!mock_poison_del(cxlds, dpa)) 1215 dev_dbg(cxlds->dev, "DPA: 0x%llx not in poison list\n", dpa); 1216 1217 return 0; 1218 } 1219 1220 static bool mock_poison_list_empty(void) 1221 { 1222 for (int i = 0; i < MOCK_INJECT_TEST_MAX; i++) { 1223 if (mock_poison_list[i].cxlds) 1224 return false; 1225 } 1226 return true; 1227 } 1228 1229 static ssize_t poison_inject_max_show(struct device_driver *drv, char *buf) 1230 { 1231 return sysfs_emit(buf, "%u\n", poison_inject_dev_max); 1232 } 1233 1234 static ssize_t poison_inject_max_store(struct device_driver *drv, 1235 const char *buf, size_t len) 1236 { 1237 int val; 1238 1239 if (kstrtoint(buf, 0, &val) < 0) 1240 return -EINVAL; 1241 1242 if (!mock_poison_list_empty()) 1243 return -EBUSY; 1244 1245 if (val <= MOCK_INJECT_TEST_MAX) 1246 poison_inject_dev_max = val; 1247 else 1248 return -EINVAL; 1249 1250 return len; 1251 } 1252 1253 static DRIVER_ATTR_RW(poison_inject_max); 1254 1255 static struct attribute *cxl_mock_mem_core_attrs[] = { 1256 &driver_attr_poison_inject_max.attr, 1257 NULL 1258 }; 1259 ATTRIBUTE_GROUPS(cxl_mock_mem_core); 1260 1261 static int mock_fw_info(struct cxl_mockmem_data *mdata, 1262 struct cxl_mbox_cmd *cmd) 1263 { 1264 struct cxl_mbox_get_fw_info fw_info = { 1265 .num_slots = FW_SLOTS, 1266 .slot_info = (mdata->fw_slot & 0x7) | 1267 ((mdata->fw_staged & 0x7) << 3), 1268 .activation_cap = 0, 1269 }; 1270 1271 strcpy(fw_info.slot_1_revision, "cxl_test_fw_001"); 1272 strcpy(fw_info.slot_2_revision, "cxl_test_fw_002"); 1273 strcpy(fw_info.slot_3_revision, "cxl_test_fw_003"); 1274 strcpy(fw_info.slot_4_revision, ""); 1275 1276 if (cmd->size_out < sizeof(fw_info)) 1277 return -EINVAL; 1278 1279 memcpy(cmd->payload_out, &fw_info, sizeof(fw_info)); 1280 return 0; 1281 } 1282 1283 static int mock_transfer_fw(struct cxl_mockmem_data *mdata, 1284 struct cxl_mbox_cmd *cmd) 1285 { 1286 struct cxl_mbox_transfer_fw *transfer = cmd->payload_in; 1287 void *fw = mdata->fw; 1288 size_t offset, length; 1289 1290 offset = le32_to_cpu(transfer->offset) * CXL_FW_TRANSFER_ALIGNMENT; 1291 length = cmd->size_in - sizeof(*transfer); 1292 if (offset + length > FW_SIZE) 1293 return -EINVAL; 1294 1295 switch (transfer->action) { 1296 case CXL_FW_TRANSFER_ACTION_FULL: 1297 if (offset != 0) 1298 return -EINVAL; 1299 fallthrough; 1300 case CXL_FW_TRANSFER_ACTION_END: 1301 if (transfer->slot == 0 || transfer->slot > FW_SLOTS) 1302 return -EINVAL; 1303 mdata->fw_size = offset + length; 1304 break; 1305 case CXL_FW_TRANSFER_ACTION_INITIATE: 1306 case CXL_FW_TRANSFER_ACTION_CONTINUE: 1307 break; 1308 case CXL_FW_TRANSFER_ACTION_ABORT: 1309 return 0; 1310 default: 1311 return -EINVAL; 1312 } 1313 1314 memcpy(fw + offset, transfer->data, length); 1315 usleep_range(1500, 2000); 1316 return 0; 1317 } 1318 1319 static int mock_activate_fw(struct cxl_mockmem_data *mdata, 1320 struct cxl_mbox_cmd *cmd) 1321 { 1322 struct cxl_mbox_activate_fw *activate = cmd->payload_in; 1323 1324 if (activate->slot == 0 || activate->slot > FW_SLOTS) 1325 return -EINVAL; 1326 1327 switch (activate->action) { 1328 case CXL_FW_ACTIVATE_ONLINE: 1329 mdata->fw_slot = activate->slot; 1330 mdata->fw_staged = 0; 1331 return 0; 1332 case CXL_FW_ACTIVATE_OFFLINE: 1333 mdata->fw_staged = activate->slot; 1334 return 0; 1335 } 1336 1337 return -EINVAL; 1338 } 1339 1340 static int cxl_mock_mbox_send(struct cxl_mailbox *cxl_mbox, 1341 struct cxl_mbox_cmd *cmd) 1342 { 1343 struct device *dev = cxl_mbox->host; 1344 struct cxl_mockmem_data *mdata = dev_get_drvdata(dev); 1345 struct cxl_memdev_state *mds = mdata->mds; 1346 struct cxl_dev_state *cxlds = &mds->cxlds; 1347 int rc = -EIO; 1348 1349 switch (cmd->opcode) { 1350 case CXL_MBOX_OP_SET_TIMESTAMP: 1351 rc = mock_set_timestamp(cxlds, cmd); 1352 break; 1353 case CXL_MBOX_OP_GET_SUPPORTED_LOGS: 1354 rc = mock_gsl(cmd); 1355 break; 1356 case CXL_MBOX_OP_GET_LOG: 1357 rc = mock_get_log(mds, cmd); 1358 break; 1359 case CXL_MBOX_OP_IDENTIFY: 1360 if (cxlds->rcd) 1361 rc = mock_rcd_id(cmd); 1362 else 1363 rc = mock_id(cmd); 1364 break; 1365 case CXL_MBOX_OP_GET_LSA: 1366 rc = mock_get_lsa(mdata, cmd); 1367 break; 1368 case CXL_MBOX_OP_GET_PARTITION_INFO: 1369 rc = mock_partition_info(cmd); 1370 break; 1371 case CXL_MBOX_OP_GET_EVENT_RECORD: 1372 rc = mock_get_event(dev, cmd); 1373 break; 1374 case CXL_MBOX_OP_CLEAR_EVENT_RECORD: 1375 rc = mock_clear_event(dev, cmd); 1376 break; 1377 case CXL_MBOX_OP_SET_LSA: 1378 rc = mock_set_lsa(mdata, cmd); 1379 break; 1380 case CXL_MBOX_OP_GET_HEALTH_INFO: 1381 rc = mock_health_info(cmd); 1382 break; 1383 case CXL_MBOX_OP_SANITIZE: 1384 rc = mock_sanitize(mdata, cmd); 1385 break; 1386 case CXL_MBOX_OP_SECURE_ERASE: 1387 rc = mock_secure_erase(mdata, cmd); 1388 break; 1389 case CXL_MBOX_OP_GET_SECURITY_STATE: 1390 rc = mock_get_security_state(mdata, cmd); 1391 break; 1392 case CXL_MBOX_OP_SET_PASSPHRASE: 1393 rc = mock_set_passphrase(mdata, cmd); 1394 break; 1395 case CXL_MBOX_OP_DISABLE_PASSPHRASE: 1396 rc = mock_disable_passphrase(mdata, cmd); 1397 break; 1398 case CXL_MBOX_OP_FREEZE_SECURITY: 1399 rc = mock_freeze_security(mdata, cmd); 1400 break; 1401 case CXL_MBOX_OP_UNLOCK: 1402 rc = mock_unlock_security(mdata, cmd); 1403 break; 1404 case CXL_MBOX_OP_PASSPHRASE_SECURE_ERASE: 1405 rc = mock_passphrase_secure_erase(mdata, cmd); 1406 break; 1407 case CXL_MBOX_OP_GET_POISON: 1408 rc = mock_get_poison(cxlds, cmd); 1409 break; 1410 case CXL_MBOX_OP_INJECT_POISON: 1411 rc = mock_inject_poison(cxlds, cmd); 1412 break; 1413 case CXL_MBOX_OP_CLEAR_POISON: 1414 rc = mock_clear_poison(cxlds, cmd); 1415 break; 1416 case CXL_MBOX_OP_GET_FW_INFO: 1417 rc = mock_fw_info(mdata, cmd); 1418 break; 1419 case CXL_MBOX_OP_TRANSFER_FW: 1420 rc = mock_transfer_fw(mdata, cmd); 1421 break; 1422 case CXL_MBOX_OP_ACTIVATE_FW: 1423 rc = mock_activate_fw(mdata, cmd); 1424 break; 1425 default: 1426 break; 1427 } 1428 1429 dev_dbg(dev, "opcode: %#x sz_in: %zd sz_out: %zd rc: %d\n", cmd->opcode, 1430 cmd->size_in, cmd->size_out, rc); 1431 1432 return rc; 1433 } 1434 1435 static void label_area_release(void *lsa) 1436 { 1437 vfree(lsa); 1438 } 1439 1440 static void fw_buf_release(void *buf) 1441 { 1442 vfree(buf); 1443 } 1444 1445 static bool is_rcd(struct platform_device *pdev) 1446 { 1447 const struct platform_device_id *id = platform_get_device_id(pdev); 1448 1449 return !!id->driver_data; 1450 } 1451 1452 static ssize_t event_trigger_store(struct device *dev, 1453 struct device_attribute *attr, 1454 const char *buf, size_t count) 1455 { 1456 cxl_mock_event_trigger(dev); 1457 return count; 1458 } 1459 static DEVICE_ATTR_WO(event_trigger); 1460 1461 static int cxl_mock_mailbox_create(struct cxl_dev_state *cxlds) 1462 { 1463 int rc; 1464 1465 rc = cxl_mailbox_init(&cxlds->cxl_mbox, cxlds->dev); 1466 if (rc) 1467 return rc; 1468 1469 return 0; 1470 } 1471 1472 static int cxl_mock_mem_probe(struct platform_device *pdev) 1473 { 1474 struct device *dev = &pdev->dev; 1475 struct cxl_memdev *cxlmd; 1476 struct cxl_memdev_state *mds; 1477 struct cxl_dev_state *cxlds; 1478 struct cxl_mockmem_data *mdata; 1479 struct cxl_mailbox *cxl_mbox; 1480 int rc; 1481 1482 mdata = devm_kzalloc(dev, sizeof(*mdata), GFP_KERNEL); 1483 if (!mdata) 1484 return -ENOMEM; 1485 dev_set_drvdata(dev, mdata); 1486 1487 mdata->lsa = vmalloc(LSA_SIZE); 1488 if (!mdata->lsa) 1489 return -ENOMEM; 1490 mdata->fw = vmalloc(FW_SIZE); 1491 if (!mdata->fw) 1492 return -ENOMEM; 1493 mdata->fw_slot = 2; 1494 1495 rc = devm_add_action_or_reset(dev, label_area_release, mdata->lsa); 1496 if (rc) 1497 return rc; 1498 1499 rc = devm_add_action_or_reset(dev, fw_buf_release, mdata->fw); 1500 if (rc) 1501 return rc; 1502 1503 mds = cxl_memdev_state_create(dev); 1504 if (IS_ERR(mds)) 1505 return PTR_ERR(mds); 1506 1507 cxlds = &mds->cxlds; 1508 rc = cxl_mock_mailbox_create(cxlds); 1509 if (rc) 1510 return rc; 1511 1512 cxl_mbox = &mds->cxlds.cxl_mbox; 1513 mdata->mds = mds; 1514 cxl_mbox->mbox_send = cxl_mock_mbox_send; 1515 cxl_mbox->payload_size = SZ_4K; 1516 mds->event.buf = (struct cxl_get_event_payload *) mdata->event_buf; 1517 INIT_DELAYED_WORK(&mds->security.poll_dwork, cxl_mockmem_sanitize_work); 1518 1519 cxlds->serial = pdev->id; 1520 if (is_rcd(pdev)) 1521 cxlds->rcd = true; 1522 1523 rc = cxl_enumerate_cmds(mds); 1524 if (rc) 1525 return rc; 1526 1527 rc = cxl_poison_state_init(mds); 1528 if (rc) 1529 return rc; 1530 1531 rc = cxl_set_timestamp(mds); 1532 if (rc) 1533 return rc; 1534 1535 cxlds->media_ready = true; 1536 rc = cxl_dev_state_identify(mds); 1537 if (rc) 1538 return rc; 1539 1540 rc = cxl_mem_create_range_info(mds); 1541 if (rc) 1542 return rc; 1543 1544 cxl_mock_add_event_logs(&mdata->mes); 1545 1546 cxlmd = devm_cxl_add_memdev(&pdev->dev, cxlds); 1547 if (IS_ERR(cxlmd)) 1548 return PTR_ERR(cxlmd); 1549 1550 rc = devm_cxl_setup_fw_upload(&pdev->dev, mds); 1551 if (rc) 1552 return rc; 1553 1554 rc = devm_cxl_sanitize_setup_notifier(&pdev->dev, cxlmd); 1555 if (rc) 1556 return rc; 1557 1558 cxl_mem_get_event_records(mds, CXLDEV_EVENT_STATUS_ALL); 1559 1560 return 0; 1561 } 1562 1563 static ssize_t security_lock_show(struct device *dev, 1564 struct device_attribute *attr, char *buf) 1565 { 1566 struct cxl_mockmem_data *mdata = dev_get_drvdata(dev); 1567 1568 return sysfs_emit(buf, "%u\n", 1569 !!(mdata->security_state & CXL_PMEM_SEC_STATE_LOCKED)); 1570 } 1571 1572 static ssize_t security_lock_store(struct device *dev, struct device_attribute *attr, 1573 const char *buf, size_t count) 1574 { 1575 struct cxl_mockmem_data *mdata = dev_get_drvdata(dev); 1576 u32 mask = CXL_PMEM_SEC_STATE_FROZEN | CXL_PMEM_SEC_STATE_USER_PLIMIT | 1577 CXL_PMEM_SEC_STATE_MASTER_PLIMIT; 1578 int val; 1579 1580 if (kstrtoint(buf, 0, &val) < 0) 1581 return -EINVAL; 1582 1583 if (val == 1) { 1584 if (!(mdata->security_state & CXL_PMEM_SEC_STATE_USER_PASS_SET)) 1585 return -ENXIO; 1586 mdata->security_state |= CXL_PMEM_SEC_STATE_LOCKED; 1587 mdata->security_state &= ~mask; 1588 } else { 1589 return -EINVAL; 1590 } 1591 return count; 1592 } 1593 1594 static DEVICE_ATTR_RW(security_lock); 1595 1596 static ssize_t fw_buf_checksum_show(struct device *dev, 1597 struct device_attribute *attr, char *buf) 1598 { 1599 struct cxl_mockmem_data *mdata = dev_get_drvdata(dev); 1600 u8 hash[SHA256_DIGEST_SIZE]; 1601 unsigned char *hstr, *hptr; 1602 struct sha256_state sctx; 1603 ssize_t written = 0; 1604 int i; 1605 1606 sha256_init(&sctx); 1607 sha256_update(&sctx, mdata->fw, mdata->fw_size); 1608 sha256_final(&sctx, hash); 1609 1610 hstr = kzalloc((SHA256_DIGEST_SIZE * 2) + 1, GFP_KERNEL); 1611 if (!hstr) 1612 return -ENOMEM; 1613 1614 hptr = hstr; 1615 for (i = 0; i < SHA256_DIGEST_SIZE; i++) 1616 hptr += sprintf(hptr, "%02x", hash[i]); 1617 1618 written = sysfs_emit(buf, "%s\n", hstr); 1619 1620 kfree(hstr); 1621 return written; 1622 } 1623 1624 static DEVICE_ATTR_RO(fw_buf_checksum); 1625 1626 static ssize_t sanitize_timeout_show(struct device *dev, 1627 struct device_attribute *attr, char *buf) 1628 { 1629 struct cxl_mockmem_data *mdata = dev_get_drvdata(dev); 1630 1631 return sysfs_emit(buf, "%lu\n", mdata->sanitize_timeout); 1632 } 1633 1634 static ssize_t sanitize_timeout_store(struct device *dev, 1635 struct device_attribute *attr, 1636 const char *buf, size_t count) 1637 { 1638 struct cxl_mockmem_data *mdata = dev_get_drvdata(dev); 1639 unsigned long val; 1640 int rc; 1641 1642 rc = kstrtoul(buf, 0, &val); 1643 if (rc) 1644 return rc; 1645 1646 mdata->sanitize_timeout = val; 1647 1648 return count; 1649 } 1650 1651 static DEVICE_ATTR_RW(sanitize_timeout); 1652 1653 static struct attribute *cxl_mock_mem_attrs[] = { 1654 &dev_attr_security_lock.attr, 1655 &dev_attr_event_trigger.attr, 1656 &dev_attr_fw_buf_checksum.attr, 1657 &dev_attr_sanitize_timeout.attr, 1658 NULL 1659 }; 1660 ATTRIBUTE_GROUPS(cxl_mock_mem); 1661 1662 static const struct platform_device_id cxl_mock_mem_ids[] = { 1663 { .name = "cxl_mem", 0 }, 1664 { .name = "cxl_rcd", 1 }, 1665 { }, 1666 }; 1667 MODULE_DEVICE_TABLE(platform, cxl_mock_mem_ids); 1668 1669 static struct platform_driver cxl_mock_mem_driver = { 1670 .probe = cxl_mock_mem_probe, 1671 .id_table = cxl_mock_mem_ids, 1672 .driver = { 1673 .name = KBUILD_MODNAME, 1674 .dev_groups = cxl_mock_mem_groups, 1675 .groups = cxl_mock_mem_core_groups, 1676 }, 1677 }; 1678 1679 module_platform_driver(cxl_mock_mem_driver); 1680 MODULE_LICENSE("GPL v2"); 1681 MODULE_IMPORT_NS(CXL); 1682