dfu_mmc.c 5.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213
  1. /*
  2. * dfu.c -- DFU back-end routines
  3. *
  4. * Copyright (C) 2012 Samsung Electronics
  5. * author: Lukasz Majewski <l.majewski@samsung.com>
  6. *
  7. * This program is free software; you can redistribute it and/or modify
  8. * it under the terms of the GNU General Public License as published by
  9. * the Free Software Foundation; either version 2 of the License, or
  10. * (at your option) any later version.
  11. *
  12. * This program is distributed in the hope that it will be useful,
  13. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  14. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  15. * GNU General Public License for more details.
  16. *
  17. * You should have received a copy of the GNU General Public License
  18. * along with this program; if not, write to the Free Software
  19. * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
  20. */
  21. #include <common.h>
  22. #include <malloc.h>
  23. #include <errno.h>
  24. #include <dfu.h>
  25. enum dfu_mmc_op {
  26. DFU_OP_READ = 1,
  27. DFU_OP_WRITE,
  28. };
  29. static int mmc_block_op(enum dfu_mmc_op op, struct dfu_entity *dfu,
  30. void *buf, long *len)
  31. {
  32. char cmd_buf[DFU_CMD_BUF_SIZE];
  33. sprintf(cmd_buf, "mmc %s 0x%x %x %x",
  34. op == DFU_OP_READ ? "read" : "write",
  35. (unsigned int) buf,
  36. dfu->data.mmc.lba_start,
  37. dfu->data.mmc.lba_size);
  38. if (op == DFU_OP_READ)
  39. *len = dfu->data.mmc.lba_blk_size * dfu->data.mmc.lba_size;
  40. debug("%s: %s 0x%p\n", __func__, cmd_buf, cmd_buf);
  41. return run_command(cmd_buf, 0);
  42. }
  43. static inline int mmc_block_write(struct dfu_entity *dfu, void *buf, long *len)
  44. {
  45. return mmc_block_op(DFU_OP_WRITE, dfu, buf, len);
  46. }
  47. static inline int mmc_block_read(struct dfu_entity *dfu, void *buf, long *len)
  48. {
  49. return mmc_block_op(DFU_OP_READ, dfu, buf, len);
  50. }
  51. static int mmc_file_op(enum dfu_mmc_op op, struct dfu_entity *dfu,
  52. void *buf, long *len)
  53. {
  54. char cmd_buf[DFU_CMD_BUF_SIZE];
  55. char *str_env;
  56. int ret;
  57. switch (dfu->layout) {
  58. case DFU_FS_FAT:
  59. sprintf(cmd_buf, "fat%s mmc %d:%d 0x%x %s %lx",
  60. op == DFU_OP_READ ? "load" : "write",
  61. dfu->data.mmc.dev, dfu->data.mmc.part,
  62. (unsigned int) buf, dfu->name, *len);
  63. break;
  64. case DFU_FS_EXT4:
  65. sprintf(cmd_buf, "ext4%s mmc %d:%d /%s 0x%x %ld",
  66. op == DFU_OP_READ ? "load" : "write",
  67. dfu->data.mmc.dev, dfu->data.mmc.part,
  68. dfu->name, (unsigned int) buf, *len);
  69. break;
  70. default:
  71. printf("%s: Layout (%s) not (yet) supported!\n", __func__,
  72. dfu_get_layout(dfu->layout));
  73. }
  74. debug("%s: %s 0x%p\n", __func__, cmd_buf, cmd_buf);
  75. ret = run_command(cmd_buf, 0);
  76. if (ret) {
  77. puts("dfu: Read error!\n");
  78. return ret;
  79. }
  80. if (dfu->layout != DFU_RAW_ADDR && op == DFU_OP_READ) {
  81. str_env = getenv("filesize");
  82. if (str_env == NULL) {
  83. puts("dfu: Wrong file size!\n");
  84. return -1;
  85. }
  86. *len = simple_strtoul(str_env, NULL, 16);
  87. }
  88. return ret;
  89. }
  90. static inline int mmc_file_write(struct dfu_entity *dfu, void *buf, long *len)
  91. {
  92. return mmc_file_op(DFU_OP_WRITE, dfu, buf, len);
  93. }
  94. static inline int mmc_file_read(struct dfu_entity *dfu, void *buf, long *len)
  95. {
  96. return mmc_file_op(DFU_OP_READ, dfu, buf, len);
  97. }
  98. int dfu_write_medium_mmc(struct dfu_entity *dfu, void *buf, long *len)
  99. {
  100. int ret = -1;
  101. switch (dfu->layout) {
  102. case DFU_RAW_ADDR:
  103. ret = mmc_block_write(dfu, buf, len);
  104. break;
  105. case DFU_FS_FAT:
  106. case DFU_FS_EXT4:
  107. ret = mmc_file_write(dfu, buf, len);
  108. break;
  109. default:
  110. printf("%s: Layout (%s) not (yet) supported!\n", __func__,
  111. dfu_get_layout(dfu->layout));
  112. }
  113. return ret;
  114. }
  115. int dfu_read_medium_mmc(struct dfu_entity *dfu, void *buf, long *len)
  116. {
  117. int ret = -1;
  118. switch (dfu->layout) {
  119. case DFU_RAW_ADDR:
  120. ret = mmc_block_read(dfu, buf, len);
  121. break;
  122. case DFU_FS_FAT:
  123. case DFU_FS_EXT4:
  124. ret = mmc_file_read(dfu, buf, len);
  125. break;
  126. default:
  127. printf("%s: Layout (%s) not (yet) supported!\n", __func__,
  128. dfu_get_layout(dfu->layout));
  129. }
  130. return ret;
  131. }
  132. int dfu_fill_entity_mmc(struct dfu_entity *dfu, char *s)
  133. {
  134. int dev, part;
  135. struct mmc *mmc;
  136. block_dev_desc_t *blk_dev;
  137. disk_partition_t partinfo;
  138. char *st;
  139. dfu->dev_type = DFU_DEV_MMC;
  140. st = strsep(&s, " ");
  141. if (!strcmp(st, "mmc")) {
  142. dfu->layout = DFU_RAW_ADDR;
  143. dfu->data.mmc.lba_start = simple_strtoul(s, &s, 16);
  144. dfu->data.mmc.lba_size = simple_strtoul(++s, &s, 16);
  145. dfu->data.mmc.lba_blk_size = get_mmc_blk_size(dfu->dev_num);
  146. } else if (!strcmp(st, "fat")) {
  147. dfu->layout = DFU_FS_FAT;
  148. } else if (!strcmp(st, "ext4")) {
  149. dfu->layout = DFU_FS_EXT4;
  150. } else if (!strcmp(st, "part")) {
  151. dfu->layout = DFU_RAW_ADDR;
  152. dev = simple_strtoul(s, &s, 10);
  153. s++;
  154. part = simple_strtoul(s, &s, 10);
  155. mmc = find_mmc_device(dev);
  156. if (mmc == NULL || mmc_init(mmc)) {
  157. printf("%s: could not find mmc device #%d!\n", __func__, dev);
  158. return -ENODEV;
  159. }
  160. blk_dev = &mmc->block_dev;
  161. if (get_partition_info(blk_dev, part, &partinfo) != 0) {
  162. printf("%s: could not find partition #%d on mmc device #%d!\n",
  163. __func__, part, dev);
  164. return -ENODEV;
  165. }
  166. dfu->data.mmc.lba_start = partinfo.start;
  167. dfu->data.mmc.lba_size = partinfo.size;
  168. dfu->data.mmc.lba_blk_size = partinfo.blksz;
  169. } else {
  170. printf("%s: Memory layout (%s) not supported!\n", __func__, st);
  171. return -ENODEV;
  172. }
  173. if (dfu->layout == DFU_FS_EXT4 || dfu->layout == DFU_FS_FAT) {
  174. dfu->data.mmc.dev = simple_strtoul(s, &s, 10);
  175. dfu->data.mmc.part = simple_strtoul(++s, &s, 10);
  176. }
  177. dfu->read_medium = dfu_read_medium_mmc;
  178. dfu->write_medium = dfu_write_medium_mmc;
  179. return 0;
  180. }