xlnx-versal-cframe-reg.c 29 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856
  1. /*
  2. * QEMU model of the Configuration Frame Control module
  3. *
  4. * Copyright (C) 2023, Advanced Micro Devices, Inc.
  5. *
  6. * Written by Francisco Iglesias <francisco.iglesias@amd.com>
  7. *
  8. * SPDX-License-Identifier: GPL-2.0-or-later
  9. */
  10. #include "qemu/osdep.h"
  11. #include "hw/sysbus.h"
  12. #include "hw/register.h"
  13. #include "hw/registerfields.h"
  14. #include "qemu/bitops.h"
  15. #include "qemu/log.h"
  16. #include "qemu/units.h"
  17. #include "qapi/error.h"
  18. #include "hw/qdev-properties.h"
  19. #include "migration/vmstate.h"
  20. #include "hw/irq.h"
  21. #include "hw/misc/xlnx-versal-cframe-reg.h"
  22. #ifndef XLNX_VERSAL_CFRAME_REG_ERR_DEBUG
  23. #define XLNX_VERSAL_CFRAME_REG_ERR_DEBUG 0
  24. #endif
  25. #define KEYHOLE_STREAM_4K (4 * KiB)
  26. #define N_WORDS_128BIT 4
  27. #define MAX_BLOCKTYPE 6
  28. #define MAX_BLOCKTYPE_FRAMES 0xFFFFF
  29. enum {
  30. CFRAME_CMD_WCFG = 1,
  31. CFRAME_CMD_ROWON = 2,
  32. CFRAME_CMD_ROWOFF = 3,
  33. CFRAME_CMD_RCFG = 4,
  34. CFRAME_CMD_DLPARK = 5,
  35. };
  36. static gint int_cmp(gconstpointer a, gconstpointer b, gpointer user_data)
  37. {
  38. guint ua = GPOINTER_TO_UINT(a);
  39. guint ub = GPOINTER_TO_UINT(b);
  40. return (ua > ub) - (ua < ub);
  41. }
  42. static void cfrm_imr_update_irq(XlnxVersalCFrameReg *s)
  43. {
  44. bool pending = s->regs[R_CFRM_ISR0] & ~s->regs[R_CFRM_IMR0];
  45. qemu_set_irq(s->irq_cfrm_imr, pending);
  46. }
  47. static void cfrm_isr_postw(RegisterInfo *reg, uint64_t val64)
  48. {
  49. XlnxVersalCFrameReg *s = XLNX_VERSAL_CFRAME_REG(reg->opaque);
  50. cfrm_imr_update_irq(s);
  51. }
  52. static uint64_t cfrm_ier_prew(RegisterInfo *reg, uint64_t val64)
  53. {
  54. XlnxVersalCFrameReg *s = XLNX_VERSAL_CFRAME_REG(reg->opaque);
  55. s->regs[R_CFRM_IMR0] &= ~s->regs[R_CFRM_IER0];
  56. s->regs[R_CFRM_IER0] = 0;
  57. cfrm_imr_update_irq(s);
  58. return 0;
  59. }
  60. static uint64_t cfrm_idr_prew(RegisterInfo *reg, uint64_t val64)
  61. {
  62. XlnxVersalCFrameReg *s = XLNX_VERSAL_CFRAME_REG(reg->opaque);
  63. s->regs[R_CFRM_IMR0] |= s->regs[R_CFRM_IDR0];
  64. s->regs[R_CFRM_IDR0] = 0;
  65. cfrm_imr_update_irq(s);
  66. return 0;
  67. }
  68. static uint64_t cfrm_itr_prew(RegisterInfo *reg, uint64_t val64)
  69. {
  70. XlnxVersalCFrameReg *s = XLNX_VERSAL_CFRAME_REG(reg->opaque);
  71. s->regs[R_CFRM_ISR0] |= s->regs[R_CFRM_ITR0];
  72. s->regs[R_CFRM_ITR0] = 0;
  73. cfrm_imr_update_irq(s);
  74. return 0;
  75. }
  76. static void cframe_incr_far(XlnxVersalCFrameReg *s)
  77. {
  78. uint32_t faddr = ARRAY_FIELD_EX32(s->regs, FAR0, FRAME_ADDR);
  79. uint32_t blktype = ARRAY_FIELD_EX32(s->regs, FAR0, BLOCKTYPE);
  80. assert(blktype <= MAX_BLOCKTYPE);
  81. faddr++;
  82. if (faddr > s->cfg.blktype_num_frames[blktype]) {
  83. /* Restart from 0 and increment block type */
  84. faddr = 0;
  85. blktype++;
  86. assert(blktype <= MAX_BLOCKTYPE);
  87. ARRAY_FIELD_DP32(s->regs, FAR0, BLOCKTYPE, blktype);
  88. }
  89. ARRAY_FIELD_DP32(s->regs, FAR0, FRAME_ADDR, faddr);
  90. }
  91. static void cfrm_fdri_post_write(RegisterInfo *reg, uint64_t val)
  92. {
  93. XlnxVersalCFrameReg *s = XLNX_VERSAL_CFRAME_REG(reg->opaque);
  94. if (s->row_configured && s->rowon && s->wcfg) {
  95. if (fifo32_num_free(&s->new_f_data) >= N_WORDS_128BIT) {
  96. fifo32_push(&s->new_f_data, s->regs[R_FDRI0]);
  97. fifo32_push(&s->new_f_data, s->regs[R_FDRI1]);
  98. fifo32_push(&s->new_f_data, s->regs[R_FDRI2]);
  99. fifo32_push(&s->new_f_data, s->regs[R_FDRI3]);
  100. }
  101. if (fifo32_is_full(&s->new_f_data)) {
  102. uint32_t addr = extract32(s->regs[R_FAR0], 0, 23);
  103. XlnxCFrame *f = g_new(XlnxCFrame, 1);
  104. for (int i = 0; i < FRAME_NUM_WORDS; i++) {
  105. f->data[i] = fifo32_pop(&s->new_f_data);
  106. }
  107. g_tree_replace(s->cframes, GUINT_TO_POINTER(addr), f);
  108. cframe_incr_far(s);
  109. fifo32_reset(&s->new_f_data);
  110. }
  111. }
  112. }
  113. static void cfrm_readout_frames(XlnxVersalCFrameReg *s, uint32_t start_addr,
  114. uint32_t end_addr)
  115. {
  116. /*
  117. * NB: when our minimum glib version is at least 2.68 we can improve the
  118. * performance of the cframe traversal by using g_tree_lookup_node and
  119. * g_tree_node_next (instead of calling g_tree_lookup for finding each
  120. * cframe).
  121. */
  122. for (uint32_t addr = start_addr; addr < end_addr; addr++) {
  123. XlnxCFrame *f = g_tree_lookup(s->cframes, GUINT_TO_POINTER(addr));
  124. /* Transmit the data if a frame was found */
  125. if (f) {
  126. for (int i = 0; i < FRAME_NUM_WORDS; i += 4) {
  127. XlnxCfiPacket pkt = {};
  128. pkt.data[0] = f->data[i];
  129. pkt.data[1] = f->data[i + 1];
  130. pkt.data[2] = f->data[i + 2];
  131. pkt.data[3] = f->data[i + 3];
  132. if (s->cfg.cfu_fdro) {
  133. xlnx_cfi_transfer_packet(s->cfg.cfu_fdro, &pkt);
  134. }
  135. }
  136. }
  137. }
  138. }
  139. static void cfrm_frcnt_post_write(RegisterInfo *reg, uint64_t val)
  140. {
  141. XlnxVersalCFrameReg *s = XLNX_VERSAL_CFRAME_REG(reg->opaque);
  142. if (s->row_configured && s->rowon && s->rcfg) {
  143. uint32_t start_addr = extract32(s->regs[R_FAR0], 0, 23);
  144. uint32_t end_addr = start_addr + s->regs[R_FRCNT0] / FRAME_NUM_QWORDS;
  145. cfrm_readout_frames(s, start_addr, end_addr);
  146. }
  147. }
  148. static void cfrm_cmd_post_write(RegisterInfo *reg, uint64_t val)
  149. {
  150. XlnxVersalCFrameReg *s = XLNX_VERSAL_CFRAME_REG(reg->opaque);
  151. if (s->row_configured) {
  152. uint8_t cmd = ARRAY_FIELD_EX32(s->regs, CMD0, CMD);
  153. switch (cmd) {
  154. case CFRAME_CMD_WCFG:
  155. s->wcfg = true;
  156. break;
  157. case CFRAME_CMD_ROWON:
  158. s->rowon = true;
  159. break;
  160. case CFRAME_CMD_ROWOFF:
  161. s->rowon = false;
  162. break;
  163. case CFRAME_CMD_RCFG:
  164. s->rcfg = true;
  165. break;
  166. case CFRAME_CMD_DLPARK:
  167. s->wcfg = false;
  168. s->rcfg = false;
  169. break;
  170. default:
  171. break;
  172. };
  173. }
  174. }
  175. static uint64_t cfrm_last_frame_bot_post_read(RegisterInfo *reg,
  176. uint64_t val64)
  177. {
  178. XlnxVersalCFrameReg *s = XLNX_VERSAL_CFRAME_REG(reg->opaque);
  179. uint64_t val = 0;
  180. switch (reg->access->addr) {
  181. case A_LAST_FRAME_BOT0:
  182. val = FIELD_DP32(val, LAST_FRAME_BOT0, BLOCKTYPE1_LAST_FRAME_LSB,
  183. s->cfg.blktype_num_frames[1]);
  184. val = FIELD_DP32(val, LAST_FRAME_BOT0, BLOCKTYPE0_LAST_FRAME,
  185. s->cfg.blktype_num_frames[0]);
  186. break;
  187. case A_LAST_FRAME_BOT1:
  188. val = FIELD_DP32(val, LAST_FRAME_BOT1, BLOCKTYPE3_LAST_FRAME_LSB,
  189. s->cfg.blktype_num_frames[3]);
  190. val = FIELD_DP32(val, LAST_FRAME_BOT1, BLOCKTYPE2_LAST_FRAME,
  191. s->cfg.blktype_num_frames[2]);
  192. val = FIELD_DP32(val, LAST_FRAME_BOT1, BLOCKTYPE1_LAST_FRAME_MSB,
  193. (s->cfg.blktype_num_frames[1] >> 12));
  194. break;
  195. case A_LAST_FRAME_BOT2:
  196. val = FIELD_DP32(val, LAST_FRAME_BOT2, BLOCKTYPE3_LAST_FRAME_MSB,
  197. (s->cfg.blktype_num_frames[3] >> 4));
  198. break;
  199. case A_LAST_FRAME_BOT3:
  200. default:
  201. break;
  202. }
  203. return val;
  204. }
  205. static uint64_t cfrm_last_frame_top_post_read(RegisterInfo *reg,
  206. uint64_t val64)
  207. {
  208. XlnxVersalCFrameReg *s = XLNX_VERSAL_CFRAME_REG(reg->opaque);
  209. uint64_t val = 0;
  210. switch (reg->access->addr) {
  211. case A_LAST_FRAME_TOP0:
  212. val = FIELD_DP32(val, LAST_FRAME_TOP0, BLOCKTYPE5_LAST_FRAME_LSB,
  213. s->cfg.blktype_num_frames[5]);
  214. val = FIELD_DP32(val, LAST_FRAME_TOP0, BLOCKTYPE4_LAST_FRAME,
  215. s->cfg.blktype_num_frames[4]);
  216. break;
  217. case A_LAST_FRAME_TOP1:
  218. val = FIELD_DP32(val, LAST_FRAME_TOP1, BLOCKTYPE6_LAST_FRAME,
  219. s->cfg.blktype_num_frames[6]);
  220. val = FIELD_DP32(val, LAST_FRAME_TOP1, BLOCKTYPE5_LAST_FRAME_MSB,
  221. (s->cfg.blktype_num_frames[5] >> 12));
  222. break;
  223. case A_LAST_FRAME_TOP2:
  224. case A_LAST_FRAME_BOT3:
  225. default:
  226. break;
  227. }
  228. return val;
  229. }
  230. static void cfrm_far_sfr_post_write(RegisterInfo *reg, uint64_t val)
  231. {
  232. XlnxVersalCFrameReg *s = XLNX_VERSAL_CFRAME_REG(reg->opaque);
  233. if (s->row_configured && s->rowon && s->rcfg) {
  234. uint32_t start_addr = extract32(s->regs[R_FAR_SFR0], 0, 23);
  235. /* Readback 1 frame */
  236. cfrm_readout_frames(s, start_addr, start_addr + 1);
  237. }
  238. }
  239. static const RegisterAccessInfo cframe_reg_regs_info[] = {
  240. { .name = "CRC0", .addr = A_CRC0,
  241. .rsvd = 0x00000000,
  242. },{ .name = "CRC1", .addr = A_CRC0,
  243. .rsvd = 0xffffffff,
  244. },{ .name = "CRC2", .addr = A_CRC0,
  245. .rsvd = 0xffffffff,
  246. },{ .name = "CRC3", .addr = A_CRC0,
  247. .rsvd = 0xffffffff,
  248. },{ .name = "FAR0", .addr = A_FAR0,
  249. .rsvd = 0xfe000000,
  250. },{ .name = "FAR1", .addr = A_FAR1,
  251. .rsvd = 0xffffffff,
  252. },{ .name = "FAR2", .addr = A_FAR2,
  253. .rsvd = 0xffffffff,
  254. },{ .name = "FAR3", .addr = A_FAR3,
  255. .rsvd = 0xffffffff,
  256. },{ .name = "FAR_SFR0", .addr = A_FAR_SFR0,
  257. .rsvd = 0xff800000,
  258. },{ .name = "FAR_SFR1", .addr = A_FAR_SFR1,
  259. .rsvd = 0xffffffff,
  260. },{ .name = "FAR_SFR2", .addr = A_FAR_SFR2,
  261. .rsvd = 0xffffffff,
  262. },{ .name = "FAR_SFR3", .addr = A_FAR_SFR3,
  263. .rsvd = 0xffffffff,
  264. .post_write = cfrm_far_sfr_post_write,
  265. },{ .name = "FDRI0", .addr = A_FDRI0,
  266. },{ .name = "FDRI1", .addr = A_FDRI1,
  267. },{ .name = "FDRI2", .addr = A_FDRI2,
  268. },{ .name = "FDRI3", .addr = A_FDRI3,
  269. .post_write = cfrm_fdri_post_write,
  270. },{ .name = "FRCNT0", .addr = A_FRCNT0,
  271. .rsvd = 0x00000000,
  272. },{ .name = "FRCNT1", .addr = A_FRCNT1,
  273. .rsvd = 0xffffffff,
  274. },{ .name = "FRCNT2", .addr = A_FRCNT2,
  275. .rsvd = 0xffffffff,
  276. },{ .name = "FRCNT3", .addr = A_FRCNT3,
  277. .rsvd = 0xffffffff,
  278. .post_write = cfrm_frcnt_post_write
  279. },{ .name = "CMD0", .addr = A_CMD0,
  280. .rsvd = 0xffffffe0,
  281. },{ .name = "CMD1", .addr = A_CMD1,
  282. .rsvd = 0xffffffff,
  283. },{ .name = "CMD2", .addr = A_CMD2,
  284. .rsvd = 0xffffffff,
  285. },{ .name = "CMD3", .addr = A_CMD3,
  286. .rsvd = 0xffffffff,
  287. .post_write = cfrm_cmd_post_write
  288. },{ .name = "CR_MASK0", .addr = A_CR_MASK0,
  289. .rsvd = 0x00000000,
  290. },{ .name = "CR_MASK1", .addr = A_CR_MASK1,
  291. .rsvd = 0x00000000,
  292. },{ .name = "CR_MASK2", .addr = A_CR_MASK2,
  293. .rsvd = 0x00000000,
  294. },{ .name = "CR_MASK3", .addr = A_CR_MASK3,
  295. .rsvd = 0xffffffff,
  296. },{ .name = "CTL0", .addr = A_CTL0,
  297. .rsvd = 0xfffffff8,
  298. },{ .name = "CTL1", .addr = A_CTL1,
  299. .rsvd = 0xffffffff,
  300. },{ .name = "CTL2", .addr = A_CTL2,
  301. .rsvd = 0xffffffff,
  302. },{ .name = "CTL3", .addr = A_CTL3,
  303. .rsvd = 0xffffffff,
  304. },{ .name = "CFRM_ISR0", .addr = A_CFRM_ISR0,
  305. .rsvd = 0xffc04000,
  306. .w1c = 0x3bfff,
  307. },{ .name = "CFRM_ISR1", .addr = A_CFRM_ISR1,
  308. .rsvd = 0xffffffff,
  309. },{ .name = "CFRM_ISR2", .addr = A_CFRM_ISR2,
  310. .rsvd = 0xffffffff,
  311. },{ .name = "CFRM_ISR3", .addr = A_CFRM_ISR3,
  312. .rsvd = 0xffffffff,
  313. .post_write = cfrm_isr_postw,
  314. },{ .name = "CFRM_IMR0", .addr = A_CFRM_IMR0,
  315. .rsvd = 0xffc04000,
  316. .ro = 0xfffff,
  317. .reset = 0x3bfff,
  318. },{ .name = "CFRM_IMR1", .addr = A_CFRM_IMR1,
  319. .rsvd = 0xffffffff,
  320. },{ .name = "CFRM_IMR2", .addr = A_CFRM_IMR2,
  321. .rsvd = 0xffffffff,
  322. },{ .name = "CFRM_IMR3", .addr = A_CFRM_IMR3,
  323. .rsvd = 0xffffffff,
  324. },{ .name = "CFRM_IER0", .addr = A_CFRM_IER0,
  325. .rsvd = 0xffc04000,
  326. },{ .name = "CFRM_IER1", .addr = A_CFRM_IER1,
  327. .rsvd = 0xffffffff,
  328. },{ .name = "CFRM_IER2", .addr = A_CFRM_IER2,
  329. .rsvd = 0xffffffff,
  330. },{ .name = "CFRM_IER3", .addr = A_CFRM_IER3,
  331. .rsvd = 0xffffffff,
  332. .pre_write = cfrm_ier_prew,
  333. },{ .name = "CFRM_IDR0", .addr = A_CFRM_IDR0,
  334. .rsvd = 0xffc04000,
  335. },{ .name = "CFRM_IDR1", .addr = A_CFRM_IDR1,
  336. .rsvd = 0xffffffff,
  337. },{ .name = "CFRM_IDR2", .addr = A_CFRM_IDR2,
  338. .rsvd = 0xffffffff,
  339. },{ .name = "CFRM_IDR3", .addr = A_CFRM_IDR3,
  340. .rsvd = 0xffffffff,
  341. .pre_write = cfrm_idr_prew,
  342. },{ .name = "CFRM_ITR0", .addr = A_CFRM_ITR0,
  343. .rsvd = 0xffc04000,
  344. },{ .name = "CFRM_ITR1", .addr = A_CFRM_ITR1,
  345. .rsvd = 0xffffffff,
  346. },{ .name = "CFRM_ITR2", .addr = A_CFRM_ITR2,
  347. .rsvd = 0xffffffff,
  348. },{ .name = "CFRM_ITR3", .addr = A_CFRM_ITR3,
  349. .rsvd = 0xffffffff,
  350. .pre_write = cfrm_itr_prew,
  351. },{ .name = "SEU_SYNDRM00", .addr = A_SEU_SYNDRM00,
  352. },{ .name = "SEU_SYNDRM01", .addr = A_SEU_SYNDRM01,
  353. },{ .name = "SEU_SYNDRM02", .addr = A_SEU_SYNDRM02,
  354. },{ .name = "SEU_SYNDRM03", .addr = A_SEU_SYNDRM03,
  355. },{ .name = "SEU_SYNDRM10", .addr = A_SEU_SYNDRM10,
  356. },{ .name = "SEU_SYNDRM11", .addr = A_SEU_SYNDRM11,
  357. },{ .name = "SEU_SYNDRM12", .addr = A_SEU_SYNDRM12,
  358. },{ .name = "SEU_SYNDRM13", .addr = A_SEU_SYNDRM13,
  359. },{ .name = "SEU_SYNDRM20", .addr = A_SEU_SYNDRM20,
  360. },{ .name = "SEU_SYNDRM21", .addr = A_SEU_SYNDRM21,
  361. },{ .name = "SEU_SYNDRM22", .addr = A_SEU_SYNDRM22,
  362. },{ .name = "SEU_SYNDRM23", .addr = A_SEU_SYNDRM23,
  363. },{ .name = "SEU_SYNDRM30", .addr = A_SEU_SYNDRM30,
  364. },{ .name = "SEU_SYNDRM31", .addr = A_SEU_SYNDRM31,
  365. },{ .name = "SEU_SYNDRM32", .addr = A_SEU_SYNDRM32,
  366. },{ .name = "SEU_SYNDRM33", .addr = A_SEU_SYNDRM33,
  367. },{ .name = "SEU_VIRTUAL_SYNDRM0", .addr = A_SEU_VIRTUAL_SYNDRM0,
  368. },{ .name = "SEU_VIRTUAL_SYNDRM1", .addr = A_SEU_VIRTUAL_SYNDRM1,
  369. },{ .name = "SEU_VIRTUAL_SYNDRM2", .addr = A_SEU_VIRTUAL_SYNDRM2,
  370. },{ .name = "SEU_VIRTUAL_SYNDRM3", .addr = A_SEU_VIRTUAL_SYNDRM3,
  371. },{ .name = "SEU_CRC0", .addr = A_SEU_CRC0,
  372. },{ .name = "SEU_CRC1", .addr = A_SEU_CRC1,
  373. },{ .name = "SEU_CRC2", .addr = A_SEU_CRC2,
  374. },{ .name = "SEU_CRC3", .addr = A_SEU_CRC3,
  375. },{ .name = "CFRAME_FAR_BOT0", .addr = A_CFRAME_FAR_BOT0,
  376. },{ .name = "CFRAME_FAR_BOT1", .addr = A_CFRAME_FAR_BOT1,
  377. },{ .name = "CFRAME_FAR_BOT2", .addr = A_CFRAME_FAR_BOT2,
  378. },{ .name = "CFRAME_FAR_BOT3", .addr = A_CFRAME_FAR_BOT3,
  379. },{ .name = "CFRAME_FAR_TOP0", .addr = A_CFRAME_FAR_TOP0,
  380. },{ .name = "CFRAME_FAR_TOP1", .addr = A_CFRAME_FAR_TOP1,
  381. },{ .name = "CFRAME_FAR_TOP2", .addr = A_CFRAME_FAR_TOP2,
  382. },{ .name = "CFRAME_FAR_TOP3", .addr = A_CFRAME_FAR_TOP3,
  383. },{ .name = "LAST_FRAME_BOT0", .addr = A_LAST_FRAME_BOT0,
  384. .ro = 0xffffffff,
  385. .post_read = cfrm_last_frame_bot_post_read,
  386. },{ .name = "LAST_FRAME_BOT1", .addr = A_LAST_FRAME_BOT1,
  387. .ro = 0xffffffff,
  388. .post_read = cfrm_last_frame_bot_post_read,
  389. },{ .name = "LAST_FRAME_BOT2", .addr = A_LAST_FRAME_BOT2,
  390. .ro = 0xffffffff,
  391. .post_read = cfrm_last_frame_bot_post_read,
  392. },{ .name = "LAST_FRAME_BOT3", .addr = A_LAST_FRAME_BOT3,
  393. .ro = 0xffffffff,
  394. .post_read = cfrm_last_frame_bot_post_read,
  395. },{ .name = "LAST_FRAME_TOP0", .addr = A_LAST_FRAME_TOP0,
  396. .ro = 0xffffffff,
  397. .post_read = cfrm_last_frame_top_post_read,
  398. },{ .name = "LAST_FRAME_TOP1", .addr = A_LAST_FRAME_TOP1,
  399. .ro = 0xffffffff,
  400. .post_read = cfrm_last_frame_top_post_read,
  401. },{ .name = "LAST_FRAME_TOP2", .addr = A_LAST_FRAME_TOP2,
  402. .ro = 0xffffffff,
  403. .post_read = cfrm_last_frame_top_post_read,
  404. },{ .name = "LAST_FRAME_TOP3", .addr = A_LAST_FRAME_TOP3,
  405. .ro = 0xffffffff,
  406. .post_read = cfrm_last_frame_top_post_read,
  407. }
  408. };
  409. static void cframe_reg_cfi_transfer_packet(XlnxCfiIf *cfi_if,
  410. XlnxCfiPacket *pkt)
  411. {
  412. XlnxVersalCFrameReg *s = XLNX_VERSAL_CFRAME_REG(cfi_if);
  413. uint64_t we = MAKE_64BIT_MASK(0, 4 * 8);
  414. if (!s->row_configured) {
  415. return;
  416. }
  417. switch (pkt->reg_addr) {
  418. case CFRAME_FAR:
  419. s->regs[R_FAR0] = pkt->data[0];
  420. break;
  421. case CFRAME_SFR:
  422. s->regs[R_FAR_SFR0] = pkt->data[0];
  423. register_write(&s->regs_info[R_FAR_SFR3], 0,
  424. we, object_get_typename(OBJECT(s)),
  425. XLNX_VERSAL_CFRAME_REG_ERR_DEBUG);
  426. break;
  427. case CFRAME_FDRI:
  428. s->regs[R_FDRI0] = pkt->data[0];
  429. s->regs[R_FDRI1] = pkt->data[1];
  430. s->regs[R_FDRI2] = pkt->data[2];
  431. register_write(&s->regs_info[R_FDRI3], pkt->data[3],
  432. we, object_get_typename(OBJECT(s)),
  433. XLNX_VERSAL_CFRAME_REG_ERR_DEBUG);
  434. break;
  435. case CFRAME_CMD:
  436. ARRAY_FIELD_DP32(s->regs, CMD0, CMD, pkt->data[0]);
  437. register_write(&s->regs_info[R_CMD3], 0,
  438. we, object_get_typename(OBJECT(s)),
  439. XLNX_VERSAL_CFRAME_REG_ERR_DEBUG);
  440. break;
  441. default:
  442. break;
  443. }
  444. }
  445. static uint64_t cframe_reg_fdri_read(void *opaque, hwaddr addr, unsigned size)
  446. {
  447. qemu_log_mask(LOG_GUEST_ERROR, "%s: Unsupported read from addr=%"
  448. HWADDR_PRIx "\n", __func__, addr);
  449. return 0;
  450. }
  451. static void cframe_reg_fdri_write(void *opaque, hwaddr addr, uint64_t value,
  452. unsigned size)
  453. {
  454. XlnxVersalCFrameReg *s = XLNX_VERSAL_CFRAME_REG(opaque);
  455. uint32_t wfifo[WFIFO_SZ];
  456. if (update_wfifo(addr, value, s->wfifo, wfifo)) {
  457. uint64_t we = MAKE_64BIT_MASK(0, 4 * 8);
  458. s->regs[R_FDRI0] = wfifo[0];
  459. s->regs[R_FDRI1] = wfifo[1];
  460. s->regs[R_FDRI2] = wfifo[2];
  461. register_write(&s->regs_info[R_FDRI3], wfifo[3],
  462. we, object_get_typename(OBJECT(s)),
  463. XLNX_VERSAL_CFRAME_REG_ERR_DEBUG);
  464. }
  465. }
  466. static void cframe_reg_reset_enter(Object *obj, ResetType type)
  467. {
  468. XlnxVersalCFrameReg *s = XLNX_VERSAL_CFRAME_REG(obj);
  469. unsigned int i;
  470. for (i = 0; i < ARRAY_SIZE(s->regs_info); ++i) {
  471. register_reset(&s->regs_info[i]);
  472. }
  473. memset(s->wfifo, 0, WFIFO_SZ * sizeof(uint32_t));
  474. fifo32_reset(&s->new_f_data);
  475. if (g_tree_nnodes(s->cframes)) {
  476. /*
  477. * Take a reference so when g_tree_destroy() unrefs it we keep the
  478. * GTree and only destroy its contents. NB: when our minimum
  479. * glib version is at least 2.70 we could use g_tree_remove_all().
  480. */
  481. g_tree_ref(s->cframes);
  482. g_tree_destroy(s->cframes);
  483. }
  484. }
  485. static void cframe_reg_reset_hold(Object *obj, ResetType type)
  486. {
  487. XlnxVersalCFrameReg *s = XLNX_VERSAL_CFRAME_REG(obj);
  488. cfrm_imr_update_irq(s);
  489. }
  490. static const MemoryRegionOps cframe_reg_ops = {
  491. .read = register_read_memory,
  492. .write = register_write_memory,
  493. .endianness = DEVICE_LITTLE_ENDIAN,
  494. .valid = {
  495. .min_access_size = 4,
  496. .max_access_size = 4,
  497. },
  498. };
  499. static const MemoryRegionOps cframe_reg_fdri_ops = {
  500. .read = cframe_reg_fdri_read,
  501. .write = cframe_reg_fdri_write,
  502. .endianness = DEVICE_LITTLE_ENDIAN,
  503. .valid = {
  504. .min_access_size = 4,
  505. .max_access_size = 4,
  506. },
  507. };
  508. static uint64_t cframes_bcast_reg_read(void *opaque, hwaddr addr, unsigned size)
  509. {
  510. qemu_log_mask(LOG_GUEST_ERROR, "%s: Unsupported read from addr=%"
  511. HWADDR_PRIx "\n", __func__, addr);
  512. return 0;
  513. }
  514. static void cframes_bcast_write(XlnxVersalCFrameBcastReg *s, uint8_t reg_addr,
  515. uint32_t *wfifo)
  516. {
  517. XlnxCfiPacket pkt = {
  518. .reg_addr = reg_addr,
  519. .data[0] = wfifo[0],
  520. .data[1] = wfifo[1],
  521. .data[2] = wfifo[2],
  522. .data[3] = wfifo[3]
  523. };
  524. for (int i = 0; i < ARRAY_SIZE(s->cfg.cframe); i++) {
  525. if (s->cfg.cframe[i]) {
  526. xlnx_cfi_transfer_packet(s->cfg.cframe[i], &pkt);
  527. }
  528. }
  529. }
  530. static void cframes_bcast_reg_write(void *opaque, hwaddr addr, uint64_t value,
  531. unsigned size)
  532. {
  533. XlnxVersalCFrameBcastReg *s = XLNX_VERSAL_CFRAME_BCAST_REG(opaque);
  534. uint32_t wfifo[WFIFO_SZ];
  535. if (update_wfifo(addr, value, s->wfifo, wfifo)) {
  536. uint8_t reg_addr = extract32(addr, 4, 6);
  537. cframes_bcast_write(s, reg_addr, wfifo);
  538. }
  539. }
  540. static uint64_t cframes_bcast_fdri_read(void *opaque, hwaddr addr,
  541. unsigned size)
  542. {
  543. qemu_log_mask(LOG_GUEST_ERROR, "%s: Unsupported read from addr=%"
  544. HWADDR_PRIx "\n", __func__, addr);
  545. return 0;
  546. }
  547. static void cframes_bcast_fdri_write(void *opaque, hwaddr addr, uint64_t value,
  548. unsigned size)
  549. {
  550. XlnxVersalCFrameBcastReg *s = XLNX_VERSAL_CFRAME_BCAST_REG(opaque);
  551. uint32_t wfifo[WFIFO_SZ];
  552. if (update_wfifo(addr, value, s->wfifo, wfifo)) {
  553. cframes_bcast_write(s, CFRAME_FDRI, wfifo);
  554. }
  555. }
  556. static const MemoryRegionOps cframes_bcast_reg_reg_ops = {
  557. .read = cframes_bcast_reg_read,
  558. .write = cframes_bcast_reg_write,
  559. .endianness = DEVICE_LITTLE_ENDIAN,
  560. .valid = {
  561. .min_access_size = 4,
  562. .max_access_size = 4,
  563. },
  564. };
  565. static const MemoryRegionOps cframes_bcast_reg_fdri_ops = {
  566. .read = cframes_bcast_fdri_read,
  567. .write = cframes_bcast_fdri_write,
  568. .endianness = DEVICE_LITTLE_ENDIAN,
  569. .valid = {
  570. .min_access_size = 4,
  571. .max_access_size = 4,
  572. },
  573. };
  574. static void cframe_reg_realize(DeviceState *dev, Error **errp)
  575. {
  576. XlnxVersalCFrameReg *s = XLNX_VERSAL_CFRAME_REG(dev);
  577. for (int i = 0; i < ARRAY_SIZE(s->cfg.blktype_num_frames); i++) {
  578. if (s->cfg.blktype_num_frames[i] > MAX_BLOCKTYPE_FRAMES) {
  579. error_setg(errp,
  580. "blktype-frames%d > 0xFFFFF (max frame per block)",
  581. i);
  582. return;
  583. }
  584. if (s->cfg.blktype_num_frames[i]) {
  585. s->row_configured = true;
  586. }
  587. }
  588. }
  589. static void cframe_reg_init(Object *obj)
  590. {
  591. XlnxVersalCFrameReg *s = XLNX_VERSAL_CFRAME_REG(obj);
  592. SysBusDevice *sbd = SYS_BUS_DEVICE(obj);
  593. RegisterInfoArray *reg_array;
  594. memory_region_init(&s->iomem, obj, TYPE_XLNX_VERSAL_CFRAME_REG,
  595. CFRAME_REG_R_MAX * 4);
  596. reg_array =
  597. register_init_block32(DEVICE(obj), cframe_reg_regs_info,
  598. ARRAY_SIZE(cframe_reg_regs_info),
  599. s->regs_info, s->regs,
  600. &cframe_reg_ops,
  601. XLNX_VERSAL_CFRAME_REG_ERR_DEBUG,
  602. CFRAME_REG_R_MAX * 4);
  603. memory_region_add_subregion(&s->iomem,
  604. 0x0,
  605. &reg_array->mem);
  606. sysbus_init_mmio(sbd, &s->iomem);
  607. memory_region_init_io(&s->iomem_fdri, obj, &cframe_reg_fdri_ops, s,
  608. TYPE_XLNX_VERSAL_CFRAME_REG "-fdri",
  609. KEYHOLE_STREAM_4K);
  610. sysbus_init_mmio(sbd, &s->iomem_fdri);
  611. sysbus_init_irq(sbd, &s->irq_cfrm_imr);
  612. s->cframes = g_tree_new_full((GCompareDataFunc)int_cmp, NULL,
  613. NULL, (GDestroyNotify)g_free);
  614. fifo32_create(&s->new_f_data, FRAME_NUM_WORDS);
  615. }
  616. static const VMStateDescription vmstate_cframe = {
  617. .name = "cframe",
  618. .version_id = 1,
  619. .minimum_version_id = 1,
  620. .fields = (const VMStateField[]) {
  621. VMSTATE_UINT32_ARRAY(data, XlnxCFrame, FRAME_NUM_WORDS),
  622. VMSTATE_END_OF_LIST()
  623. }
  624. };
  625. static const VMStateDescription vmstate_cframe_reg = {
  626. .name = TYPE_XLNX_VERSAL_CFRAME_REG,
  627. .version_id = 1,
  628. .minimum_version_id = 1,
  629. .fields = (const VMStateField[]) {
  630. VMSTATE_UINT32_ARRAY(wfifo, XlnxVersalCFrameReg, 4),
  631. VMSTATE_UINT32_ARRAY(regs, XlnxVersalCFrameReg, CFRAME_REG_R_MAX),
  632. VMSTATE_BOOL(rowon, XlnxVersalCFrameReg),
  633. VMSTATE_BOOL(wcfg, XlnxVersalCFrameReg),
  634. VMSTATE_BOOL(rcfg, XlnxVersalCFrameReg),
  635. VMSTATE_GTREE_DIRECT_KEY_V(cframes, XlnxVersalCFrameReg, 1,
  636. &vmstate_cframe, XlnxCFrame),
  637. VMSTATE_FIFO32(new_f_data, XlnxVersalCFrameReg),
  638. VMSTATE_END_OF_LIST(),
  639. }
  640. };
  641. static const Property cframe_regs_props[] = {
  642. DEFINE_PROP_LINK("cfu-fdro", XlnxVersalCFrameReg, cfg.cfu_fdro,
  643. TYPE_XLNX_CFI_IF, XlnxCfiIf *),
  644. DEFINE_PROP_UINT32("blktype0-frames", XlnxVersalCFrameReg,
  645. cfg.blktype_num_frames[0], 0),
  646. DEFINE_PROP_UINT32("blktype1-frames", XlnxVersalCFrameReg,
  647. cfg.blktype_num_frames[1], 0),
  648. DEFINE_PROP_UINT32("blktype2-frames", XlnxVersalCFrameReg,
  649. cfg.blktype_num_frames[2], 0),
  650. DEFINE_PROP_UINT32("blktype3-frames", XlnxVersalCFrameReg,
  651. cfg.blktype_num_frames[3], 0),
  652. DEFINE_PROP_UINT32("blktype4-frames", XlnxVersalCFrameReg,
  653. cfg.blktype_num_frames[4], 0),
  654. DEFINE_PROP_UINT32("blktype5-frames", XlnxVersalCFrameReg,
  655. cfg.blktype_num_frames[5], 0),
  656. DEFINE_PROP_UINT32("blktype6-frames", XlnxVersalCFrameReg,
  657. cfg.blktype_num_frames[6], 0),
  658. };
  659. static void cframe_bcast_reg_init(Object *obj)
  660. {
  661. XlnxVersalCFrameBcastReg *s = XLNX_VERSAL_CFRAME_BCAST_REG(obj);
  662. SysBusDevice *sbd = SYS_BUS_DEVICE(obj);
  663. memory_region_init_io(&s->iomem_reg, obj, &cframes_bcast_reg_reg_ops, s,
  664. TYPE_XLNX_VERSAL_CFRAME_BCAST_REG, KEYHOLE_STREAM_4K);
  665. memory_region_init_io(&s->iomem_fdri, obj, &cframes_bcast_reg_fdri_ops, s,
  666. TYPE_XLNX_VERSAL_CFRAME_BCAST_REG "-fdri",
  667. KEYHOLE_STREAM_4K);
  668. sysbus_init_mmio(sbd, &s->iomem_reg);
  669. sysbus_init_mmio(sbd, &s->iomem_fdri);
  670. }
  671. static void cframe_bcast_reg_reset_enter(Object *obj, ResetType type)
  672. {
  673. XlnxVersalCFrameBcastReg *s = XLNX_VERSAL_CFRAME_BCAST_REG(obj);
  674. memset(s->wfifo, 0, WFIFO_SZ * sizeof(uint32_t));
  675. }
  676. static const VMStateDescription vmstate_cframe_bcast_reg = {
  677. .name = TYPE_XLNX_VERSAL_CFRAME_BCAST_REG,
  678. .version_id = 1,
  679. .minimum_version_id = 1,
  680. .fields = (const VMStateField[]) {
  681. VMSTATE_UINT32_ARRAY(wfifo, XlnxVersalCFrameBcastReg, 4),
  682. VMSTATE_END_OF_LIST(),
  683. }
  684. };
  685. static const Property cframe_bcast_regs_props[] = {
  686. DEFINE_PROP_LINK("cframe0", XlnxVersalCFrameBcastReg, cfg.cframe[0],
  687. TYPE_XLNX_CFI_IF, XlnxCfiIf *),
  688. DEFINE_PROP_LINK("cframe1", XlnxVersalCFrameBcastReg, cfg.cframe[1],
  689. TYPE_XLNX_CFI_IF, XlnxCfiIf *),
  690. DEFINE_PROP_LINK("cframe2", XlnxVersalCFrameBcastReg, cfg.cframe[2],
  691. TYPE_XLNX_CFI_IF, XlnxCfiIf *),
  692. DEFINE_PROP_LINK("cframe3", XlnxVersalCFrameBcastReg, cfg.cframe[3],
  693. TYPE_XLNX_CFI_IF, XlnxCfiIf *),
  694. DEFINE_PROP_LINK("cframe4", XlnxVersalCFrameBcastReg, cfg.cframe[4],
  695. TYPE_XLNX_CFI_IF, XlnxCfiIf *),
  696. DEFINE_PROP_LINK("cframe5", XlnxVersalCFrameBcastReg, cfg.cframe[5],
  697. TYPE_XLNX_CFI_IF, XlnxCfiIf *),
  698. DEFINE_PROP_LINK("cframe6", XlnxVersalCFrameBcastReg, cfg.cframe[6],
  699. TYPE_XLNX_CFI_IF, XlnxCfiIf *),
  700. DEFINE_PROP_LINK("cframe7", XlnxVersalCFrameBcastReg, cfg.cframe[7],
  701. TYPE_XLNX_CFI_IF, XlnxCfiIf *),
  702. DEFINE_PROP_LINK("cframe8", XlnxVersalCFrameBcastReg, cfg.cframe[8],
  703. TYPE_XLNX_CFI_IF, XlnxCfiIf *),
  704. DEFINE_PROP_LINK("cframe9", XlnxVersalCFrameBcastReg, cfg.cframe[9],
  705. TYPE_XLNX_CFI_IF, XlnxCfiIf *),
  706. DEFINE_PROP_LINK("cframe10", XlnxVersalCFrameBcastReg, cfg.cframe[10],
  707. TYPE_XLNX_CFI_IF, XlnxCfiIf *),
  708. DEFINE_PROP_LINK("cframe11", XlnxVersalCFrameBcastReg, cfg.cframe[11],
  709. TYPE_XLNX_CFI_IF, XlnxCfiIf *),
  710. DEFINE_PROP_LINK("cframe12", XlnxVersalCFrameBcastReg, cfg.cframe[12],
  711. TYPE_XLNX_CFI_IF, XlnxCfiIf *),
  712. DEFINE_PROP_LINK("cframe13", XlnxVersalCFrameBcastReg, cfg.cframe[13],
  713. TYPE_XLNX_CFI_IF, XlnxCfiIf *),
  714. DEFINE_PROP_LINK("cframe14", XlnxVersalCFrameBcastReg, cfg.cframe[14],
  715. TYPE_XLNX_CFI_IF, XlnxCfiIf *),
  716. };
  717. static void cframe_reg_class_init(ObjectClass *klass, void *data)
  718. {
  719. ResettableClass *rc = RESETTABLE_CLASS(klass);
  720. DeviceClass *dc = DEVICE_CLASS(klass);
  721. XlnxCfiIfClass *xcic = XLNX_CFI_IF_CLASS(klass);
  722. dc->vmsd = &vmstate_cframe_reg;
  723. dc->realize = cframe_reg_realize;
  724. rc->phases.enter = cframe_reg_reset_enter;
  725. rc->phases.hold = cframe_reg_reset_hold;
  726. device_class_set_props(dc, cframe_regs_props);
  727. xcic->cfi_transfer_packet = cframe_reg_cfi_transfer_packet;
  728. }
  729. static void cframe_bcast_reg_class_init(ObjectClass *klass, void *data)
  730. {
  731. DeviceClass *dc = DEVICE_CLASS(klass);
  732. ResettableClass *rc = RESETTABLE_CLASS(klass);
  733. dc->vmsd = &vmstate_cframe_bcast_reg;
  734. device_class_set_props(dc, cframe_bcast_regs_props);
  735. rc->phases.enter = cframe_bcast_reg_reset_enter;
  736. }
  737. static const TypeInfo cframe_reg_info = {
  738. .name = TYPE_XLNX_VERSAL_CFRAME_REG,
  739. .parent = TYPE_SYS_BUS_DEVICE,
  740. .instance_size = sizeof(XlnxVersalCFrameReg),
  741. .class_init = cframe_reg_class_init,
  742. .instance_init = cframe_reg_init,
  743. .interfaces = (InterfaceInfo[]) {
  744. { TYPE_XLNX_CFI_IF },
  745. { }
  746. }
  747. };
  748. static const TypeInfo cframe_bcast_reg_info = {
  749. .name = TYPE_XLNX_VERSAL_CFRAME_BCAST_REG,
  750. .parent = TYPE_SYS_BUS_DEVICE,
  751. .instance_size = sizeof(XlnxVersalCFrameBcastReg),
  752. .class_init = cframe_bcast_reg_class_init,
  753. .instance_init = cframe_bcast_reg_init,
  754. };
  755. static void cframe_reg_register_types(void)
  756. {
  757. type_register_static(&cframe_reg_info);
  758. type_register_static(&cframe_bcast_reg_info);
  759. }
  760. type_init(cframe_reg_register_types)