static int vm_allocate_dh(struct vm_state *vms, struct ast_vm_user *vmu, int count_msg) {
 
        int arraysize = (vmu->maxmsg > count_msg ? vmu->maxmsg : count_msg);
-       if (!vms->dh_arraysize) {
-               /* initial allocation */
+
+       /* remove old allocation */
+       if (vms->deleted) {
+               ast_free(vms->deleted);
+               vms->deleted = NULL;
+       }
+       if (vms->heard) {
+               ast_free(vms->heard);
+               vms->heard = NULL;
+       }
+       vms->dh_arraysize = 0;
+
+       if (arraysize > 0) {
                if (!(vms->deleted = ast_calloc(arraysize, sizeof(int)))) {
                        return -1;
                }
                if (!(vms->heard = ast_calloc(arraysize, sizeof(int)))) {
+                       ast_free(vms->deleted);
+                       vms->deleted = NULL;
                        return -1;
                }
                vms->dh_arraysize = arraysize;
-       } else if (vms->dh_arraysize < arraysize) {
-               if (!(vms->deleted = ast_realloc(vms->deleted, arraysize * sizeof(int)))) {
-                       return -1;
-               }
-               if (!(vms->heard = ast_realloc(vms->heard, arraysize * sizeof(int)))) {
-                       return -1;
-               }
-               memset(vms->deleted, 0, arraysize * sizeof(int));
-               memset(vms->heard, 0, arraysize * sizeof(int));
-               vms->dh_arraysize = arraysize;
        }
 
        return 0;
                name = "Unknown Caller";
 
        /* If deleted, show "undeleted" */
-
-       if (vms->deleted[vms->curmsg])
-               keys[1] = ADSI_KEY_SKT | (ADSI_KEY_APPS + 11);
+#ifdef IMAP_STORAGE
+       ast_mutex_lock(&vms->lock);
+#endif
+       if (vms->deleted[vms->curmsg]) {
+               keys[1] = ADSI_KEY_SKT | (ADSI_KEY_APPS + 11);
+       }
+#ifdef IMAP_STORAGE
+       ast_mutex_unlock(&vms->lock);
+#endif
 
        /* Except "Exit" */
        keys[5] = ADSI_KEY_SKT | (ADSI_KEY_APPS + 5);
        }
 
        /* If deleted, show "undeleted" */
-       if (vms->deleted[vms->curmsg]) 
+#ifdef IMAP_STORAGE
+       ast_mutex_lock(&vms->lock);
+#endif
+       if (vms->deleted[vms->curmsg]) {
                keys[1] = ADSI_KEY_SKT | (ADSI_KEY_APPS + 11);
+       }
+#ifdef IMAP_STORAGE
+       ast_mutex_unlock(&vms->lock);
+#endif
 
        /* Except "Exit" */
        keys[5] = ADSI_KEY_SKT | (ADSI_KEY_APPS + 5);
 
        if (!res) {
                make_file(vms->fn, sizeof(vms->fn), vms->curdir, vms->curmsg);
+#ifdef IMAP_STORAGE
+               ast_mutex_lock(&vms->lock);
+#endif
                vms->heard[vms->curmsg] = 1;
 #ifdef IMAP_STORAGE
+               ast_mutex_unlock(&vms->lock);
                /*IMAP storage stores any prepended message from a forward
                 * as a separate file from the rest of the message
                 */
        }
        ast_unlock_path(vms->curdir);
 #else /* defined(IMAP_STORAGE) */
+       ast_mutex_lock(&vms->lock);
        if (vms->deleted) {
                /* Since we now expunge after each delete, deleting in reverse order
                 * ensures that no reordering occurs between each step. */
 #endif
 
 done:
-       if (vms->deleted && last_msg_idx) {
+       if (vms->deleted) {
                ast_free(vms->deleted);
+               vms->deleted = NULL;
        }
-       if (vms->heard && last_msg_idx) {
+       if (vms->heard) {
                ast_free(vms->heard);
+               vms->heard = NULL;
        }
+       vms->dh_arraysize = 0;
+#ifdef IMAP_STORAGE
+       ast_mutex_unlock(&vms->lock);
+#endif
 
        return 0;
 }
                                res = ast_play_and_wait(chan, "vm-next");
                        }
                        if (!res) {
-                               if (!vms->deleted[vms->curmsg])
+                               int curmsg_deleted;
+#ifdef IMAP_STORAGE
+                               ast_mutex_lock(&vms->lock);
+#endif
+                               curmsg_deleted = vms->deleted[vms->curmsg];
+#ifdef IMAP_STORAGE
+                               ast_mutex_unlock(&vms->lock);
+#endif
+                               if (!curmsg_deleted) {
                                        res = ast_play_and_wait(chan, "vm-delete");
-                               else
+                               } else {
                                        res = ast_play_and_wait(chan, "vm-undelete");
-                               if (!res)
+                               }
+                               if (!res) {
                                        res = ast_play_and_wait(chan, "vm-toforward");
-                               if (!res)
+                               }
+                               if (!res) {
                                        res = ast_play_and_wait(chan, "vm-savemessage");
+                               }
                        }
                }
                if (!res) {
                        }
 
                        vms.starting = 1;
+                       vms.curmsg = 0;
                        break;
                case '3': /* Advanced options */
                        ast_test_suite_event_notify("ADVOPTIONS", "Message: entering advanced options menu");