sandbox_mmc.c 3.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147
  1. // SPDX-License-Identifier: GPL-2.0+
  2. /*
  3. * Copyright (c) 2015 Google, Inc
  4. * Written by Simon Glass <sjg@chromium.org>
  5. */
  6. #include <common.h>
  7. #include <dm.h>
  8. #include <errno.h>
  9. #include <fdtdec.h>
  10. #include <mmc.h>
  11. #include <asm/test.h>
  12. struct sandbox_mmc_plat {
  13. struct mmc_config cfg;
  14. struct mmc mmc;
  15. };
  16. /**
  17. * sandbox_mmc_send_cmd() - Emulate SD commands
  18. *
  19. * This emulate an SD card version 2. Single-block reads result in zero data.
  20. * Multiple-block reads return a test string.
  21. */
  22. static int sandbox_mmc_send_cmd(struct udevice *dev, struct mmc_cmd *cmd,
  23. struct mmc_data *data)
  24. {
  25. switch (cmd->cmdidx) {
  26. case MMC_CMD_ALL_SEND_CID:
  27. break;
  28. case SD_CMD_SEND_RELATIVE_ADDR:
  29. cmd->response[0] = 0 << 16; /* mmc->rca */
  30. case MMC_CMD_GO_IDLE_STATE:
  31. break;
  32. case SD_CMD_SEND_IF_COND:
  33. cmd->response[0] = 0xaa;
  34. break;
  35. case MMC_CMD_SEND_STATUS:
  36. cmd->response[0] = MMC_STATUS_RDY_FOR_DATA;
  37. break;
  38. case MMC_CMD_SELECT_CARD:
  39. break;
  40. case MMC_CMD_SEND_CSD:
  41. cmd->response[0] = 0;
  42. cmd->response[1] = 10 << 16; /* 1 << block_len */
  43. break;
  44. case SD_CMD_SWITCH_FUNC: {
  45. if (!data)
  46. break;
  47. u32 *resp = (u32 *)data->dest;
  48. resp[7] = cpu_to_be32(SD_HIGHSPEED_BUSY);
  49. if ((cmd->cmdarg & 0xF) == UHS_SDR12_BUS_SPEED)
  50. resp[4] = (cmd->cmdarg & 0xF) << 24;
  51. break;
  52. }
  53. case MMC_CMD_READ_SINGLE_BLOCK:
  54. memset(data->dest, '\0', data->blocksize);
  55. break;
  56. case MMC_CMD_READ_MULTIPLE_BLOCK:
  57. strcpy(data->dest, "this is a test");
  58. break;
  59. case MMC_CMD_STOP_TRANSMISSION:
  60. break;
  61. case SD_CMD_APP_SEND_OP_COND:
  62. cmd->response[0] = OCR_BUSY | OCR_HCS;
  63. cmd->response[1] = 0;
  64. cmd->response[2] = 0;
  65. break;
  66. case MMC_CMD_APP_CMD:
  67. break;
  68. case MMC_CMD_SET_BLOCKLEN:
  69. debug("block len %d\n", cmd->cmdarg);
  70. break;
  71. case SD_CMD_APP_SEND_SCR: {
  72. u32 *scr = (u32 *)data->dest;
  73. scr[0] = cpu_to_be32(2 << 24 | 1 << 15); /* SD version 3 */
  74. break;
  75. }
  76. default:
  77. debug("%s: Unknown command %d\n", __func__, cmd->cmdidx);
  78. break;
  79. }
  80. return 0;
  81. }
  82. static int sandbox_mmc_set_ios(struct udevice *dev)
  83. {
  84. return 0;
  85. }
  86. static int sandbox_mmc_get_cd(struct udevice *dev)
  87. {
  88. return 1;
  89. }
  90. static const struct dm_mmc_ops sandbox_mmc_ops = {
  91. .send_cmd = sandbox_mmc_send_cmd,
  92. .set_ios = sandbox_mmc_set_ios,
  93. .get_cd = sandbox_mmc_get_cd,
  94. };
  95. int sandbox_mmc_probe(struct udevice *dev)
  96. {
  97. struct sandbox_mmc_plat *plat = dev_get_platdata(dev);
  98. return mmc_init(&plat->mmc);
  99. }
  100. int sandbox_mmc_bind(struct udevice *dev)
  101. {
  102. struct sandbox_mmc_plat *plat = dev_get_platdata(dev);
  103. struct mmc_config *cfg = &plat->cfg;
  104. cfg->name = dev->name;
  105. cfg->host_caps = MMC_MODE_HS_52MHz | MMC_MODE_HS | MMC_MODE_8BIT;
  106. cfg->voltages = MMC_VDD_165_195 | MMC_VDD_32_33 | MMC_VDD_33_34;
  107. cfg->f_min = 1000000;
  108. cfg->f_max = 52000000;
  109. cfg->b_max = U32_MAX;
  110. return mmc_bind(dev, &plat->mmc, cfg);
  111. }
  112. int sandbox_mmc_unbind(struct udevice *dev)
  113. {
  114. mmc_unbind(dev);
  115. return 0;
  116. }
  117. static const struct udevice_id sandbox_mmc_ids[] = {
  118. { .compatible = "sandbox,mmc" },
  119. { }
  120. };
  121. U_BOOT_DRIVER(mmc_sandbox) = {
  122. .name = "mmc_sandbox",
  123. .id = UCLASS_MMC,
  124. .of_match = sandbox_mmc_ids,
  125. .ops = &sandbox_mmc_ops,
  126. .bind = sandbox_mmc_bind,
  127. .unbind = sandbox_mmc_unbind,
  128. .probe = sandbox_mmc_probe,
  129. .platdata_auto_alloc_size = sizeof(struct sandbox_mmc_plat),
  130. };