sdmmc_stm32.c 9.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439
  1. /*
  2. * Copyright (c) 2020 Amarula Solutions.
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. #define DT_DRV_COMPAT st_stm32_sdmmc
  7. #include <devicetree.h>
  8. #include <drivers/disk.h>
  9. #include <drivers/clock_control.h>
  10. #include <drivers/clock_control/stm32_clock_control.h>
  11. #include <pinmux/pinmux_stm32.h>
  12. #include <drivers/gpio.h>
  13. #include <logging/log.h>
  14. #include <soc.h>
  15. #include <stm32_ll_rcc.h>
  16. LOG_MODULE_REGISTER(stm32_sdmmc, CONFIG_SDMMC_LOG_LEVEL);
  17. #ifndef MMC_TypeDef
  18. #define MMC_TypeDef SDMMC_TypeDef
  19. #endif
  20. struct stm32_sdmmc_priv {
  21. SD_HandleTypeDef hsd;
  22. int status;
  23. struct k_work work;
  24. struct gpio_callback cd_cb;
  25. struct {
  26. const char *name;
  27. const struct device *port;
  28. int pin;
  29. int flags;
  30. } cd;
  31. struct {
  32. const char *name;
  33. const struct device *port;
  34. int pin;
  35. int flags;
  36. } pe;
  37. struct stm32_pclken pclken;
  38. struct {
  39. const struct soc_gpio_pinctrl *list;
  40. size_t len;
  41. } pinctrl;
  42. };
  43. static int stm32_sdmmc_clock_enable(struct stm32_sdmmc_priv *priv)
  44. {
  45. const struct device *clock;
  46. #if CONFIG_SOC_SERIES_STM32L4X
  47. LL_RCC_PLLSAI1_Disable();
  48. /* Configure PLLSA11 to enable 48M domain */
  49. LL_RCC_PLLSAI1_ConfigDomain_48M(LL_RCC_PLLSOURCE_HSI,
  50. LL_RCC_PLLM_DIV_1,
  51. 8, LL_RCC_PLLSAI1Q_DIV_8);
  52. /* Enable PLLSA1 */
  53. LL_RCC_PLLSAI1_Enable();
  54. /* Enable PLLSAI1 output mapped on 48MHz domain clock */
  55. LL_RCC_PLLSAI1_EnableDomain_48M();
  56. /* Wait for PLLSA1 ready flag */
  57. while (LL_RCC_PLLSAI1_IsReady() != 1)
  58. ;
  59. LL_RCC_SetSDMMCClockSource(LL_RCC_SDMMC1_CLKSOURCE_PLLSAI1);
  60. #endif
  61. clock = DEVICE_DT_GET(STM32_CLOCK_CONTROL_NODE);
  62. /* Enable the APB clock for stm32_sdmmc */
  63. return clock_control_on(clock, (clock_control_subsys_t *)&priv->pclken);
  64. }
  65. static int stm32_sdmmc_clock_disable(struct stm32_sdmmc_priv *priv)
  66. {
  67. const struct device *clock;
  68. clock = DEVICE_DT_GET(STM32_CLOCK_CONTROL_NODE);
  69. return clock_control_off(clock,
  70. (clock_control_subsys_t *)&priv->pclken);
  71. }
  72. static int stm32_sdmmc_access_init(struct disk_info *disk)
  73. {
  74. const struct device *dev = disk->dev;
  75. struct stm32_sdmmc_priv *priv = dev->data;
  76. int err;
  77. if (priv->status == DISK_STATUS_OK) {
  78. return 0;
  79. }
  80. if (priv->status == DISK_STATUS_NOMEDIA) {
  81. return -ENODEV;
  82. }
  83. err = stm32_sdmmc_clock_enable(priv);
  84. if (err) {
  85. LOG_ERR("failed to init clocks");
  86. return err;
  87. }
  88. err = HAL_SD_Init(&priv->hsd);
  89. if (err != HAL_OK) {
  90. LOG_ERR("failed to init stm32_sdmmc");
  91. return -EIO;
  92. }
  93. priv->status = DISK_STATUS_OK;
  94. return 0;
  95. }
  96. static void stm32_sdmmc_access_deinit(struct stm32_sdmmc_priv *priv)
  97. {
  98. HAL_SD_DeInit(&priv->hsd);
  99. stm32_sdmmc_clock_disable(priv);
  100. }
  101. static int stm32_sdmmc_access_status(struct disk_info *disk)
  102. {
  103. const struct device *dev = disk->dev;
  104. struct stm32_sdmmc_priv *priv = dev->data;
  105. return priv->status;
  106. }
  107. static int stm32_sdmmc_access_read(struct disk_info *disk, uint8_t *data_buf,
  108. uint32_t start_sector, uint32_t num_sector)
  109. {
  110. const struct device *dev = disk->dev;
  111. struct stm32_sdmmc_priv *priv = dev->data;
  112. int err;
  113. err = HAL_SD_ReadBlocks(&priv->hsd, data_buf, start_sector,
  114. num_sector, 30000);
  115. if (err != HAL_OK) {
  116. LOG_ERR("sd read block failed %d", err);
  117. return -EIO;
  118. }
  119. while (HAL_SD_GetCardState(&priv->hsd) != HAL_SD_CARD_TRANSFER)
  120. ;
  121. return 0;
  122. }
  123. static int stm32_sdmmc_access_write(struct disk_info *disk,
  124. const uint8_t *data_buf,
  125. uint32_t start_sector, uint32_t num_sector)
  126. {
  127. const struct device *dev = disk->dev;
  128. struct stm32_sdmmc_priv *priv = dev->data;
  129. int err;
  130. err = HAL_SD_WriteBlocks(&priv->hsd, (uint8_t *)data_buf, start_sector,
  131. num_sector, 30000);
  132. if (err != HAL_OK) {
  133. LOG_ERR("sd write block failed %d", err);
  134. return -EIO;
  135. }
  136. while (HAL_SD_GetCardState(&priv->hsd) != HAL_SD_CARD_TRANSFER)
  137. ;
  138. return 0;
  139. }
  140. static int stm32_sdmmc_access_ioctl(struct disk_info *disk, uint8_t cmd,
  141. void *buff)
  142. {
  143. const struct device *dev = disk->dev;
  144. struct stm32_sdmmc_priv *priv = dev->data;
  145. HAL_SD_CardInfoTypeDef info;
  146. int err;
  147. switch (cmd) {
  148. case DISK_IOCTL_GET_SECTOR_COUNT:
  149. err = HAL_SD_GetCardInfo(&priv->hsd, &info);
  150. if (err != HAL_OK) {
  151. return -EIO;
  152. }
  153. *(uint32_t *)buff = info.LogBlockNbr;
  154. break;
  155. case DISK_IOCTL_GET_SECTOR_SIZE:
  156. err = HAL_SD_GetCardInfo(&priv->hsd, &info);
  157. if (err != HAL_OK) {
  158. return -EIO;
  159. }
  160. *(uint32_t *)buff = info.LogBlockSize;
  161. break;
  162. case DISK_IOCTL_GET_ERASE_BLOCK_SZ:
  163. *(uint32_t *)buff = 1;
  164. break;
  165. case DISK_IOCTL_CTRL_SYNC:
  166. /* we use a blocking API, so nothing to do for sync */
  167. break;
  168. default:
  169. return -EINVAL;
  170. }
  171. return 0;
  172. }
  173. static const struct disk_operations stm32_sdmmc_ops = {
  174. .init = stm32_sdmmc_access_init,
  175. .status = stm32_sdmmc_access_status,
  176. .read = stm32_sdmmc_access_read,
  177. .write = stm32_sdmmc_access_write,
  178. .ioctl = stm32_sdmmc_access_ioctl,
  179. };
  180. static struct disk_info stm32_sdmmc_info = {
  181. .name = CONFIG_SDMMC_VOLUME_NAME,
  182. .ops = &stm32_sdmmc_ops,
  183. };
  184. /*
  185. * Check if the card is present or not. If no card detect gpio is set, assume
  186. * the card is present. If reading the gpio fails for some reason, assume the
  187. * card is there.
  188. */
  189. static bool stm32_sdmmc_card_present(struct stm32_sdmmc_priv *priv)
  190. {
  191. int err;
  192. if (!priv->cd.name) {
  193. return true;
  194. }
  195. err = gpio_pin_get(priv->cd.port, priv->cd.pin);
  196. if (err < 0) {
  197. LOG_WRN("reading card detect failed %d", err);
  198. return true;
  199. }
  200. return err;
  201. }
  202. static void stm32_sdmmc_cd_handler(struct k_work *item)
  203. {
  204. struct stm32_sdmmc_priv *priv = CONTAINER_OF(item,
  205. struct stm32_sdmmc_priv,
  206. work);
  207. if (stm32_sdmmc_card_present(priv)) {
  208. LOG_DBG("card inserted");
  209. priv->status = DISK_STATUS_UNINIT;
  210. } else {
  211. LOG_DBG("card removed");
  212. stm32_sdmmc_access_deinit(priv);
  213. priv->status = DISK_STATUS_NOMEDIA;
  214. }
  215. }
  216. static void stm32_sdmmc_cd_callback(const struct device *gpiodev,
  217. struct gpio_callback *cb,
  218. uint32_t pin)
  219. {
  220. struct stm32_sdmmc_priv *priv = CONTAINER_OF(cb,
  221. struct stm32_sdmmc_priv,
  222. cd_cb);
  223. k_work_submit(&priv->work);
  224. }
  225. static int stm32_sdmmc_card_detect_init(struct stm32_sdmmc_priv *priv)
  226. {
  227. int err;
  228. if (!priv->cd.name) {
  229. return 0;
  230. }
  231. priv->cd.port = device_get_binding(priv->cd.name);
  232. if (!priv->cd.port) {
  233. return -ENODEV;
  234. }
  235. gpio_init_callback(&priv->cd_cb, stm32_sdmmc_cd_callback,
  236. 1 << priv->cd.pin);
  237. err = gpio_add_callback(priv->cd.port, &priv->cd_cb);
  238. if (err) {
  239. return err;
  240. }
  241. err = gpio_pin_configure(priv->cd.port, priv->cd.pin,
  242. priv->cd.flags | GPIO_INPUT);
  243. if (err) {
  244. goto remove_callback;
  245. }
  246. err = gpio_pin_interrupt_configure(priv->cd.port, priv->cd.pin,
  247. GPIO_INT_EDGE_BOTH);
  248. if (err) {
  249. goto unconfigure_pin;
  250. }
  251. return 0;
  252. unconfigure_pin:
  253. gpio_pin_configure(priv->cd.port, priv->cd.pin, GPIO_DISCONNECTED);
  254. remove_callback:
  255. gpio_remove_callback(priv->cd.port, &priv->cd_cb);
  256. return err;
  257. }
  258. static int stm32_sdmmc_card_detect_uninit(struct stm32_sdmmc_priv *priv)
  259. {
  260. if (!priv->cd.name) {
  261. return 0;
  262. }
  263. gpio_pin_interrupt_configure(priv->cd.port, priv->cd.pin,
  264. GPIO_INT_MODE_DISABLED);
  265. gpio_pin_configure(priv->cd.port, priv->cd.pin, GPIO_DISCONNECTED);
  266. gpio_remove_callback(priv->cd.port, &priv->cd_cb);
  267. return 0;
  268. }
  269. static int stm32_sdmmc_pwr_init(struct stm32_sdmmc_priv *priv)
  270. {
  271. int err;
  272. if (!priv->pe.name) {
  273. return 0;
  274. }
  275. priv->pe.port = device_get_binding(priv->pe.name);
  276. if (!priv->pe.port) {
  277. return -ENODEV;
  278. }
  279. err = gpio_pin_configure(priv->pe.port, priv->pe.pin,
  280. priv->pe.flags | GPIO_OUTPUT_ACTIVE);
  281. if (err) {
  282. return err;
  283. }
  284. k_sleep(K_MSEC(50));
  285. return 0;
  286. }
  287. static int stm32_sdmmc_pwr_uninit(struct stm32_sdmmc_priv *priv)
  288. {
  289. if (!priv->pe.name) {
  290. return 0;
  291. }
  292. gpio_pin_configure(priv->pe.port, priv->pe.pin, GPIO_DISCONNECTED);
  293. return 0;
  294. }
  295. static int disk_stm32_sdmmc_init(const struct device *dev)
  296. {
  297. struct stm32_sdmmc_priv *priv = dev->data;
  298. int err;
  299. k_work_init(&priv->work, stm32_sdmmc_cd_handler);
  300. /* Configure dt provided device signals when available */
  301. err = stm32_dt_pinctrl_configure(priv->pinctrl.list,
  302. priv->pinctrl.len,
  303. (uint32_t)priv->hsd.Instance);
  304. if (err < 0) {
  305. return err;
  306. }
  307. err = stm32_sdmmc_card_detect_init(priv);
  308. if (err) {
  309. return err;
  310. }
  311. err = stm32_sdmmc_pwr_init(priv);
  312. if (err) {
  313. goto err_card_detect;
  314. }
  315. if (stm32_sdmmc_card_present(priv)) {
  316. priv->status = DISK_STATUS_UNINIT;
  317. } else {
  318. priv->status = DISK_STATUS_NOMEDIA;
  319. }
  320. stm32_sdmmc_info.dev = dev;
  321. err = disk_access_register(&stm32_sdmmc_info);
  322. if (err) {
  323. goto err_pwr;
  324. }
  325. return 0;
  326. err_pwr:
  327. stm32_sdmmc_pwr_uninit(priv);
  328. err_card_detect:
  329. stm32_sdmmc_card_detect_uninit(priv);
  330. return err;
  331. }
  332. #if DT_NODE_HAS_STATUS(DT_DRV_INST(0), okay)
  333. static const struct soc_gpio_pinctrl sdmmc_pins_1[] =
  334. ST_STM32_DT_INST_PINCTRL(0, 0);
  335. static struct stm32_sdmmc_priv stm32_sdmmc_priv_1 = {
  336. .hsd = {
  337. .Instance = (MMC_TypeDef *)DT_INST_REG_ADDR(0),
  338. },
  339. #if DT_INST_NODE_HAS_PROP(0, cd_gpios)
  340. .cd = {
  341. .name = DT_INST_GPIO_LABEL(0, cd_gpios),
  342. .pin = DT_INST_GPIO_PIN(0, cd_gpios),
  343. .flags = DT_INST_GPIO_FLAGS(0, cd_gpios),
  344. },
  345. #endif
  346. #if DT_INST_NODE_HAS_PROP(0, pwr_gpios)
  347. .pe = {
  348. .name = DT_INST_GPIO_LABEL(0, pwr_gpios),
  349. .pin = DT_INST_GPIO_PIN(0, pwr_gpios),
  350. .flags = DT_INST_GPIO_FLAGS(0, pwr_gpios),
  351. },
  352. #endif
  353. .pclken = {
  354. .bus = DT_INST_CLOCKS_CELL(0, bus),
  355. .enr = DT_INST_CLOCKS_CELL(0, bits),
  356. },
  357. .pinctrl = {
  358. .list = sdmmc_pins_1,
  359. .len = ARRAY_SIZE(sdmmc_pins_1)
  360. }
  361. };
  362. DEVICE_DT_INST_DEFINE(0, disk_stm32_sdmmc_init, NULL,
  363. &stm32_sdmmc_priv_1, NULL, POST_KERNEL,
  364. CONFIG_SDMMC_INIT_PRIORITY,
  365. NULL);
  366. #endif