~ [ source navigation ] ~ [ diff markup ] ~ [ identifier search ] ~

TOMOYO Linux Cross Reference
Linux/tools/testing/cxl/test/mem.c

Version: ~ [ linux-6.11-rc3 ] ~ [ linux-6.10.4 ] ~ [ linux-6.9.12 ] ~ [ linux-6.8.12 ] ~ [ linux-6.7.12 ] ~ [ linux-6.6.45 ] ~ [ linux-6.5.13 ] ~ [ linux-6.4.16 ] ~ [ linux-6.3.13 ] ~ [ linux-6.2.16 ] ~ [ linux-6.1.104 ] ~ [ linux-6.0.19 ] ~ [ linux-5.19.17 ] ~ [ linux-5.18.19 ] ~ [ linux-5.17.15 ] ~ [ linux-5.16.20 ] ~ [ linux-5.15.164 ] ~ [ linux-5.14.21 ] ~ [ linux-5.13.19 ] ~ [ linux-5.12.19 ] ~ [ linux-5.11.22 ] ~ [ linux-5.10.223 ] ~ [ linux-5.9.16 ] ~ [ linux-5.8.18 ] ~ [ linux-5.7.19 ] ~ [ linux-5.6.19 ] ~ [ linux-5.5.19 ] ~ [ linux-5.4.281 ] ~ [ linux-5.3.18 ] ~ [ linux-5.2.21 ] ~ [ linux-5.1.21 ] ~ [ linux-5.0.21 ] ~ [ linux-4.20.17 ] ~ [ linux-4.19.319 ] ~ [ linux-4.18.20 ] ~ [ linux-4.17.19 ] ~ [ linux-4.16.18 ] ~ [ linux-4.15.18 ] ~ [ linux-4.14.336 ] ~ [ linux-4.13.16 ] ~ [ linux-4.12.14 ] ~ [ linux-4.11.12 ] ~ [ linux-4.10.17 ] ~ [ linux-4.9.337 ] ~ [ linux-4.4.302 ] ~ [ linux-3.10.108 ] ~ [ linux-2.6.32.71 ] ~ [ linux-2.6.0 ] ~ [ linux-2.4.37.11 ] ~ [ unix-v6-master ] ~ [ ccs-tools-1.8.9 ] ~ [ policy-sample ] ~
Architecture: ~ [ i386 ] ~ [ alpha ] ~ [ m68k ] ~ [ mips ] ~ [ ppc ] ~ [ sparc ] ~ [ sparc64 ] ~

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

~ [ source navigation ] ~ [ diff markup ] ~ [ identifier search ] ~

kernel.org | git.kernel.org | LWN.net | Project Home | SVN repository | Mail admin

Linux® is a registered trademark of Linus Torvalds in the United States and other countries.
TOMOYO® is a registered trademark of NTT DATA CORPORATION.

sflogo.php