Merge pull request #215 from petiepooo/patch-1
[fw/stlink] / flash / main.c
index 609a6f7d3b7b549afe2953077656d3535ed7dfb3..d550eeeeda9c158c34d6a06a0341d20aaf07ef14 100644 (file)
@@ -9,20 +9,23 @@
 #include <sys/types.h>
 #include "stlink-common.h"
 
-
+enum st_cmds {DO_WRITE = 0, DO_READ = 1, DO_ERASE = 2};
 struct opts
 {
-  unsigned int do_read;
+  enum st_cmds cmd;
   const char* devname;
   const char* filename;
   stm32_addr_t addr;
   size_t size;
+  int reset;
 };
 
 static void usage(void)
 {
-    puts("stlinkv1 command line: ./flash {read|write} /dev/sgX path addr <size>");
-    puts("stlinkv2 command line: ./flash {read|write} path addr <size>");
+    puts("stlinkv1 command line: ./flash [--reset] {read|write} /dev/sgX path addr <size>");
+    puts("stlinkv1 command line: ./flash /dev/sgX erase");
+    puts("stlinkv2 command line: ./flash [--reset] {read|write} path addr <size>");
+    puts("stlinkv2 command line: ./flash erase");
     puts("                       use hex format for addr and <size>");
 }
 
@@ -33,38 +36,65 @@ static int get_opts(struct opts* o, int ac, char** av)
 
   unsigned int i = 0;
 
-  if (ac < 3) return -1;
+  if (ac < 1) return -1;
+
+  if (strcmp(av[0], "--reset") == 0)
+  {
+    o->reset = 1;
+    ac--;
+    av++;
+  }
+  else
+  {
+    o->reset = 0;
+  }
+  
+  if (ac < 1) return -1;
 
   /* stlinkv2 */
   o->devname = NULL;
 
-  if (strcmp(av[0], "read") == 0)
+  if (strcmp(av[0], "erase") == 0)
   {
-    o->do_read = 1;
+    o->cmd = DO_ERASE;
 
     /* stlinkv1 mode */
-    if (ac == 5)
+    if (ac == 2)
     {
       o->devname = av[1];
       i = 1;
     }
-    if (ac > 3)
-       o->size = strtoul(av[i + 3], NULL, 16);
   }
-  else if (strcmp(av[0], "write") == 0)
-  {
-    o->do_read = 0;
-
-    /* stlinkv1 mode */
-    if (ac == 4)
-    {
-      o->devname = av[1];
-      i = 1;
-    }
-  }
-  else
-  {
-    return -1;
+  else {
+      if (ac < 3) return -1;
+      if (strcmp(av[0], "read") == 0)
+      {
+         o->cmd = DO_READ;
+         
+         /* stlinkv1 mode */
+         if (ac == 5)
+         {
+             o->devname = av[1];
+             i = 1;
+         }
+         if (ac > 3)
+             o->size = strtoul(av[i + 3], NULL, 16);
+      }
+      else if (strcmp(av[0], "write") == 0)
+      {
+         o->cmd = DO_WRITE;
+         
+         /* stlinkv1 mode */
+         if (ac == 4)
+         {
+             o->devname = av[1];
+             i = 1;
+         }
+      }
+      else
+      {
+         return -1;
+      }
   }
 
   o->filename = av[i + 1];
@@ -90,13 +120,13 @@ int main(int ac, char** av)
 
   if (o.devname != NULL) /* stlinkv1 */
   {
-    sl = stlink_v1_open(50);
+    sl = stlink_v1_open(50, 1);
     if (sl == NULL) goto on_error;
     sl->verbose = 50;
   }
   else /* stlinkv2 */
   {
-    sl = stlink_open_usb(50);
+    sl = stlink_open_usb(50, 1);
     if (sl == NULL) goto on_error;
     sl->verbose = 50;
   }
@@ -107,12 +137,43 @@ int main(int ac, char** av)
   if (stlink_current_mode(sl) != STLINK_DEV_DEBUG_MODE)
     stlink_enter_swd_mode(sl);
 
-  stlink_reset(sl);
-  stlink_load_device_params(sl);
+  if (o.reset)
+    stlink_reset(sl);
 
-  if (o.do_read == 0) /* write */
+// Disable DMA - Set All DMA CCR Registers to zero. - AKS 1/7/2013
+  if (sl->chip_id == STM32_CHIPID_F4)
+  {
+    memset(sl->q_buf,0,4);
+       for (int i=0;i<8;i++) {
+           stlink_write_mem32(sl,0x40026000+0x10+0x18*i,4);
+           stlink_write_mem32(sl,0x40026400+0x10+0x18*i,4);
+           stlink_write_mem32(sl,0x40026000+0x24+0x18*i,4);
+           stlink_write_mem32(sl,0x40026400+0x24+0x18*i,4);
+       }
+  }
+  if (o.cmd == DO_WRITE) /* write */
+  {
+    if ((o.addr >= sl->flash_base) &&
+       (o.addr < sl->flash_base + sl->flash_size)) {
+       err = stlink_fwrite_flash(sl, o.filename, o.addr);
+       if (err == -1)
+       {
+           printf("stlink_fwrite_flash() == -1\n");
+           goto on_error;
+       }
+    }
+   else if ((o.addr >= sl->sram_base) &&
+            (o.addr < sl->sram_base + sl->sram_size)) {
+       err = stlink_fwrite_sram(sl, o.filename, o.addr);
+       if (err == -1)
+       {
+           printf("stlink_sram_flash() == -1\n");
+           goto on_error;
+       }
+   }
+  } else if (o.cmd == DO_ERASE) 
   {
-    err = stlink_fwrite_flash(sl, o.filename, o.addr);
+     err = stlink_erase_flash_mass(sl);
     if (err == -1)
     {
       printf("stlink_fwrite_flash() == -1\n");
@@ -121,6 +182,12 @@ int main(int ac, char** av)
   }
   else /* read */
   {
+    if ((o.addr >= sl->flash_base) && (o.size == 0) &&
+       (o.addr < sl->flash_base + sl->flash_size))
+        o.size = sl->flash_size;
+    else if ((o.addr >= sl->sram_base) && (o.size == 0) &&
+            (o.addr < sl->sram_base + sl->sram_size))
+        o.size = sl->sram_size;
     err = stlink_fread(sl, o.filename, o.addr, o.size);
     if (err == -1)
     {
@@ -129,14 +196,16 @@ int main(int ac, char** av)
     }
   }
 
+  if (o.reset)
+    stlink_reset(sl);
+
   /* success */
   err = 0;
 
  on_error:
   if (sl != NULL)
   {
-    stlink_reset(sl);
-    stlink_run(sl);
+    stlink_exit_debug_mode(sl);
     stlink_close(sl);
   }