/* Copyright (c) 2003-2018 Dovecot authors, see the included COPYING file */
#include "lib.h"
#include "array.h"
#include "mail-index-view-private.h"
#include "mail-index-sync-private.h"
#include "mail-index-transaction-private.h"
#include "mail-transaction-log-private.h"
#include "mail-cache-private.h"
#include <stdio.h>
struct mail_index_sync_ctx {
char *reason;
const void *data;
};
{
for (i = 0; i < size; i++) {
}
}
{
for (i = 0; i < size; i++) {
e[i].guid_128);
}
}
{
for (i = 0; i < size; i++) {
if (u[i].add_flags != 0) {
u[i].add_flags);
}
if (u[i].remove_flags != 0) {
u[i].remove_flags);
}
}
}
{
uidset_offset = sizeof(*u) + u->name_size;
if ((uidset_offset % 4) != 0)
for (i = 0; i < size; i += 2) {
/* FIXME: mail_index_update_keywords_range() */
u->modify_type, keywords);
}
}
}
{
for (i = 0; i < size; i++) {
}
}
}
{
case MAIL_TRANSACTION_EXPUNGE:
break;
break;
break;
break;
break;
default:
return FALSE;
}
return TRUE;
}
{
continue;
}
}
static int
{
unsigned int i, keyword_count;
int ret;
/* show dirty flags as flag updates */
}
/* read all transactions from log into a transaction in memory.
skip the external ones, they're already synced to mailbox and
included in our view */
continue;
T_BEGIN {
if (mail_index_sync_add_transaction(ctx)) {
/* update tail_offset if needed */
} else {
/* this is an internal change. we don't
necessarily need to update tail_offset, so
avoid the extra write caused by it. */
}
} T_END;
}
/* create an array containing all expunge, flag and keyword update
arrays so we can easily go through all of the changes. */
}
}
for (i = 0; i < keyword_count; i++) {
(const void *)&keyword_updates[i].add_seq;
synclist->keyword_idx = i;
}
(const void *)&keyword_updates[i].remove_seq;
synclist->keyword_idx = i;
}
}
return ret;
}
static bool
{
if ((flags & MAIL_INDEX_SYNC_FLAG_REQUIRE_CHANGES) == 0)
return TRUE;
/* sync only if there's something to do */
(flags & MAIL_INDEX_SYNC_FLAG_DROP_RECENT) != 0)
return TRUE;
(flags & MAIL_INDEX_SYNC_FLAG_FLUSH_DIRTY) != 0 &&
return TRUE;
/* we want to sync up to transaction log's head */
}
return TRUE;
if (index->need_recreate)
return TRUE;
/* already synced */
}
static int
{
const char *reason;
bool reset;
int ret;
if (ret < 0)
return -1;
if (ret == 0) {
/* either corrupted or the file was deleted for
some reason. either way, we can't go forward */
"Unexpected transaction log desync with index %s: %s",
return 0;
}
return 1;
}
struct mail_index_sync_ctx **ctx_r,
struct mail_index_view **view_r,
struct mail_index_transaction **trans_r,
enum mail_index_sync_flags flags)
{
int ret;
(flags & MAIL_INDEX_SYNC_FLAG_REQUIRE_CHANGES) != 0);
return ret;
}
static int
enum mail_index_sync_flags flags,
{
int ret;
/* if we require changes, don't lock transaction log yet. first check
if there's anything to sync. */
if ((flags & MAIL_INDEX_SYNC_FLAG_REQUIRE_CHANGES) == 0) {
return -1;
}
/* The view must contain what we expect the mailbox to look like
currently. That allows the backend to update external flag
changes (etc.) if the view doesn't match the mailbox.
We'll update the view to contain everything that exist in the
transaction log except for expunges. They're synced in
mail_index_sync_commit(). */
if (ret == 0) {
if (locked)
return -1;
}
/* let's try again */
if (locked)
return -1;
}
}
if (locked)
return 0;
}
if (!locked) {
/* it looks like we have something to sync. lock the file and
check again. */
}
if (index->index_deleted &&
(flags & MAIL_INDEX_SYNC_FLAG_DELETING_INDEX) == 0) {
/* index is already deleted. we can't sync. */
if (locked)
return -1;
}
/* broken sync positions. fix them. */
"broken sync positions in index file %s",
}
return 1;
}
static int
struct mail_index_sync_ctx **ctx_r,
struct mail_index_view **view_r,
struct mail_index_transaction **trans_r,
{
int ret;
/* index is corrupted and need to be reopened */
return -1;
}
if (ret <= 0)
return ret;
/* set before any rollbacks are called */
/* we wish to see all the changes from last mailbox sync position to
the end of the transaction log */
if (ret < 0) {
return -1;
}
if (ret == 0) {
/* if a log file is missing, there's nothing we can do except
to skip over it. fix the problem with fsck and try again. */
return 0;
}
/* we need to have all the transactions sorted to optimize
caller's mailbox access patterns */
if (mail_index_sync_read_and_sort(ctx) < 0) {
return -1;
}
/* create the transaction after the view has been updated with
external transactions and marked as sync view */
return 1;
}
struct mail_index_sync_ctx **ctx_r,
struct mail_index_view **view_r,
struct mail_index_transaction **trans_r,
enum mail_index_sync_flags flags)
{
bool retry;
int ret;
if (retry) {
}
return ret;
}
{
}
enum mail_index_sync_flags flags,
bool expunges_only)
{
const void *data;
const char *reason;
bool reset;
int ret;
(flags & MAIL_INDEX_SYNC_FLAG_DROP_RECENT) != 0)
return TRUE;
(flags & MAIL_INDEX_SYNC_FLAG_FLUSH_DIRTY) != 0 &&
return TRUE;
/* let the actual syncing handle the error */
return TRUE;
}
continue;
case MAIL_TRANSACTION_EXPUNGE:
return TRUE;
/* extension record updates aren't exactly needed
to be synced, but cache syncing relies on tail
offsets being updated. */
if (!expunges_only)
return TRUE;
break;
default:
break;
}
}
return ret < 0;
}
enum mail_index_sync_flags flags)
{
bool ret;
return ret;
}
{
bool ret;
return ret;
}
{
}
static void
const struct mail_transaction_expunge_guid *exp)
{
}
static void
const struct mail_index_flag_update *update)
{
}
static void
struct mail_index_sync_list *sync_list)
{
}
struct mail_index_sync_rec *sync_rec)
{
/* FIXME: replace with a priority queue so we don't have to go
through the whole list constantly. and remember to make sure that
keyword resets are sent before adds! */
/* FIXME: pretty ugly to do this for expunges, which isn't even a
seq_range. */
for (i = 0; i < count; i++) {
continue;
/* use this one. */
break;
}
next_i = i;
}
}
if (i == count) {
/* nothing left in sync_list */
return FALSE;
}
i = next_i;
}
(const struct mail_transaction_expunge_guid *)uid_range);
(const struct mail_index_flag_update *)uid_range);
} else {
&sync_list[i]);
}
return TRUE;
}
{
return TRUE;
}
return FALSE;
}
{
}
{
}
{
}
const char *reason)
{
}
{
const char *lock_reason;
if (ctx->no_warning)
lock_reason = NULL;
else
lock_reason = "Mailbox was synchronized";
}
static void
{
if (!ctx->fully_synced) {
/* Everything wasn't synced. This usually means that syncing
was used for locking and nothing was synced. Don't update
tail offset. */
return;
}
/* synced everything, but we might also have committed new
transactions. include them also here. */
/* If tail offset has changed, make sure it gets written to
transaction log. do this only if we're required to make changes.
avoid writing a new tail offset if all the transactions were
external, because that wouldn't change effective the tail offset.
except e.g. mdbox map requires this to happen, so do it
optionally. */
}
}
{
if (index->last_read_log_file_seq != 0 &&
/* dovecot.index points to an old .log file. we were supposed
to rewrite the dovecot.index when rotating the log, so
we shouldn't usually get here. */
return TRUE;
}
(index->index_min_write &&
return TRUE;
if (index->need_recreate)
return TRUE;
return FALSE;
}
{
if (delete_index) {
/* finish this sync by marking the index deleted */
/* another process just marked the index deleted.
finish the sync, but return error. */
ret = -1;
}
/* if cache compression fails, we don't really care.
the cache offsets are updated only if the compression was
successful. */
&cache_lock);
}
offsetof(struct mail_index_header,
}
}
if (index->pending_log2_rotate_time != 0) {
}
if (cache_lock != NULL)
if (ret2 < 0) {
return -1;
}
if (delete_index)
else if (index_undeleted) {
}
/* refresh the mapping with newly committed external transactions
and the synced expunges. sync using file handler here so that the
expunge handlers get called. */
ret = -1;
if (ret == 0 &&
}
return ret;
}
{
}
{
}
{
const unsigned int *keyword_indexes;
unsigned int i, count;
for (i = 0; i < count; i++) {
if (keyword_indexes[i] == idx)
return FALSE;
}
return TRUE;
for (i = 0; i < count; i++) {
if (keyword_indexes[i] == idx) {
return TRUE;
}
}
return FALSE;
default:
i_unreached();
return FALSE;
}
}
const char *fmt, ...)
{
/* make sure we don't get to this same error again by updating the
dovecot.index */
/* be silent */
return;
}
T_BEGIN {
"Log synchronization error at "
} T_END;
}