You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 
 

234 lines
6.4 KiB

  1. /***************************************************************************
  2. * Copyright (C) 2010 by Spencer Oliver *
  3. * spen@spen-soft.co.uk *
  4. * *
  5. * This program is free software; you can redistribute it and/or modify *
  6. * it under the terms of the GNU General Public License as published by *
  7. * the Free Software Foundation; either version 2 of the License, or *
  8. * (at your option) any later version. *
  9. * *
  10. * This program is distributed in the hope that it will be useful, *
  11. * but WITHOUT ANY WARRANTY; without even the implied warranty of *
  12. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the *
  13. * GNU General Public License for more details. *
  14. * *
  15. * You should have received a copy of the GNU General Public License *
  16. * along with this program. If not, see <http://www.gnu.org/licenses/>. *
  17. ***************************************************************************/
  18. #ifdef HAVE_CONFIG_H
  19. #include "config.h"
  20. #endif
  21. #include "imp.h"
  22. static struct flash_bank *virtual_get_master_bank(struct flash_bank *bank)
  23. {
  24. struct flash_bank *master_bank;
  25. master_bank = get_flash_bank_by_name_noprobe(bank->driver_priv);
  26. if (master_bank == NULL)
  27. LOG_ERROR("master flash bank '%s' does not exist", (char *)bank->driver_priv);
  28. return master_bank;
  29. }
  30. static void virtual_update_bank_info(struct flash_bank *bank)
  31. {
  32. struct flash_bank *master_bank = virtual_get_master_bank(bank);
  33. if (master_bank == NULL)
  34. return;
  35. /* update the info we do not have */
  36. bank->size = master_bank->size;
  37. bank->chip_width = master_bank->chip_width;
  38. bank->bus_width = master_bank->bus_width;
  39. bank->default_padded_value = master_bank->default_padded_value;
  40. bank->num_sectors = master_bank->num_sectors;
  41. bank->sectors = master_bank->sectors;
  42. }
  43. FLASH_BANK_COMMAND_HANDLER(virtual_flash_bank_command)
  44. {
  45. if (CMD_ARGC < 7)
  46. return ERROR_COMMAND_SYNTAX_ERROR;
  47. /* get the master flash bank */
  48. const char *bank_name = CMD_ARGV[6];
  49. struct flash_bank *master_bank = get_flash_bank_by_name_noprobe(bank_name);
  50. if (master_bank == NULL) {
  51. LOG_ERROR("master flash bank '%s' does not exist", bank_name);
  52. return ERROR_FLASH_OPERATION_FAILED;
  53. }
  54. /* save master bank name - use this to get settings later */
  55. bank->driver_priv = strdup(bank_name);
  56. return ERROR_OK;
  57. }
  58. static int virtual_protect(struct flash_bank *bank, int set, int first, int last)
  59. {
  60. struct flash_bank *master_bank = virtual_get_master_bank(bank);
  61. int retval;
  62. if (master_bank == NULL)
  63. return ERROR_FLASH_OPERATION_FAILED;
  64. /* call master handler */
  65. retval = master_bank->driver->protect(master_bank, set, first, last);
  66. if (retval != ERROR_OK)
  67. return retval;
  68. return ERROR_OK;
  69. }
  70. static int virtual_protect_check(struct flash_bank *bank)
  71. {
  72. struct flash_bank *master_bank = virtual_get_master_bank(bank);
  73. int retval;
  74. if (master_bank == NULL)
  75. return ERROR_FLASH_OPERATION_FAILED;
  76. /* call master handler */
  77. retval = master_bank->driver->protect_check(master_bank);
  78. if (retval != ERROR_OK)
  79. return retval;
  80. return ERROR_OK;
  81. }
  82. static int virtual_erase(struct flash_bank *bank, int first, int last)
  83. {
  84. struct flash_bank *master_bank = virtual_get_master_bank(bank);
  85. int retval;
  86. if (master_bank == NULL)
  87. return ERROR_FLASH_OPERATION_FAILED;
  88. /* call master handler */
  89. retval = master_bank->driver->erase(master_bank, first, last);
  90. if (retval != ERROR_OK)
  91. return retval;
  92. return ERROR_OK;
  93. }
  94. static int virtual_write(struct flash_bank *bank, const uint8_t *buffer,
  95. uint32_t offset, uint32_t count)
  96. {
  97. struct flash_bank *master_bank = virtual_get_master_bank(bank);
  98. int retval;
  99. if (master_bank == NULL)
  100. return ERROR_FLASH_OPERATION_FAILED;
  101. /* call master handler */
  102. retval = master_bank->driver->write(master_bank, buffer, offset, count);
  103. if (retval != ERROR_OK)
  104. return retval;
  105. return ERROR_OK;
  106. }
  107. static int virtual_probe(struct flash_bank *bank)
  108. {
  109. struct flash_bank *master_bank = virtual_get_master_bank(bank);
  110. int retval;
  111. if (master_bank == NULL)
  112. return ERROR_FLASH_OPERATION_FAILED;
  113. /* call master handler */
  114. retval = master_bank->driver->probe(master_bank);
  115. if (retval != ERROR_OK)
  116. return retval;
  117. /* update the info we do not have */
  118. virtual_update_bank_info(bank);
  119. return ERROR_OK;
  120. }
  121. static int virtual_auto_probe(struct flash_bank *bank)
  122. {
  123. struct flash_bank *master_bank = virtual_get_master_bank(bank);
  124. int retval;
  125. if (master_bank == NULL)
  126. return ERROR_FLASH_OPERATION_FAILED;
  127. /* call master handler */
  128. retval = master_bank->driver->auto_probe(master_bank);
  129. if (retval != ERROR_OK)
  130. return retval;
  131. /* update the info we do not have */
  132. virtual_update_bank_info(bank);
  133. return ERROR_OK;
  134. }
  135. static int virtual_info(struct flash_bank *bank, char *buf, int buf_size)
  136. {
  137. struct flash_bank *master_bank = virtual_get_master_bank(bank);
  138. if (master_bank == NULL)
  139. return ERROR_FLASH_OPERATION_FAILED;
  140. snprintf(buf, buf_size, "%s driver for flash bank %s at 0x%8.8" PRIx32 "",
  141. bank->driver->name, master_bank->name, master_bank->base);
  142. return ERROR_OK;
  143. }
  144. static int virtual_blank_check(struct flash_bank *bank)
  145. {
  146. struct flash_bank *master_bank = virtual_get_master_bank(bank);
  147. int retval;
  148. if (master_bank == NULL)
  149. return ERROR_FLASH_OPERATION_FAILED;
  150. /* call master handler */
  151. retval = master_bank->driver->erase_check(master_bank);
  152. if (retval != ERROR_OK)
  153. return retval;
  154. return ERROR_OK;
  155. }
  156. static int virtual_flash_read(struct flash_bank *bank,
  157. uint8_t *buffer, uint32_t offset, uint32_t count)
  158. {
  159. struct flash_bank *master_bank = virtual_get_master_bank(bank);
  160. int retval;
  161. if (master_bank == NULL)
  162. return ERROR_FLASH_OPERATION_FAILED;
  163. /* call master handler */
  164. retval = master_bank->driver->read(master_bank, buffer, offset, count);
  165. if (retval != ERROR_OK)
  166. return retval;
  167. return ERROR_OK;
  168. }
  169. struct flash_driver virtual_flash = {
  170. .name = "virtual",
  171. .flash_bank_command = virtual_flash_bank_command,
  172. .erase = virtual_erase,
  173. .protect = virtual_protect,
  174. .write = virtual_write,
  175. .read = virtual_flash_read,
  176. .probe = virtual_probe,
  177. .auto_probe = virtual_auto_probe,
  178. .erase_check = virtual_blank_check,
  179. .protect_check = virtual_protect_check,
  180. .info = virtual_info,
  181. };