mem.c 47 KB

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