@ -864,6 +864,21 @@ load_box( sync_vars_t *svars, int t, uint minwuid, uint_array_t mexcs )
svars - > drv [ t ] - > load_box ( svars - > ctx [ t ] , minwuid , maxwuid , svars - > finduid [ t ] , pairuid , svars - > maxuid [ t ] , mexcs , box_loaded , AUX ) ;
svars - > drv [ t ] - > load_box ( svars - > ctx [ t ] , minwuid , maxwuid , svars - > finduid [ t ] , pairuid , svars - > maxuid [ t ] , mexcs , box_loaded , AUX ) ;
}
}
typedef struct {
sync_rec_t * srec ;
uchar flags ;
} alive_srec_t ;
static int
cmp_srec_far ( const void * a , const void * b )
{
uint au = ( * ( const alive_srec_t * ) a ) . srec - > uid [ F ] ;
uint bu = ( * ( const alive_srec_t * ) b ) . srec - > uid [ F ] ;
assert ( au & & bu ) ;
assert ( au ! = bu ) ;
return au > bu ? 1 : - 1 ; // Can't subtract, the result might not fit into signed int.
}
typedef struct {
typedef struct {
void * aux ;
void * aux ;
sync_rec_t * srec ;
sync_rec_t * srec ;
@ -883,7 +898,7 @@ box_loaded( int sts, message_t *msgs, int total_msgs, int recent_msgs, void *aux
sync_rec_t * srec , * * srecmap ;
sync_rec_t * srec , * * srecmap ;
message_t * tmsg ;
message_t * tmsg ;
flag_vars_t * fv ;
flag_vars_t * fv ;
int no [ 2 ] , del [ 2 ] , alive , todel ;
int no [ 2 ] , del [ 2 ] ;
uchar sflags , nflags , aflags , dflags ;
uchar sflags , nflags , aflags , dflags ;
uint hashsz , idx ;
uint hashsz , idx ;
@ -1238,51 +1253,44 @@ box_loaded( int sts, message_t *msgs, int total_msgs, int recent_msgs, void *aux
/* Expire excess messages. Important (flagged, unread, or unpropagated) messages
/* Expire excess messages. Important (flagged, unread, or unpropagated) messages
* older than the first not expired message are not counted towards the total . */
* older than the first not expired message are not counted towards the total . */
debug ( " preparing message expiration \n " ) ;
debug ( " preparing message expiration \n " ) ;
// Due to looping only over the far side, we completely ignore unpaired
alive_srec_t * arecs = nfmalloc ( sizeof ( * arecs ) * svars - > nsrecs ) ;
// near-side messages. This is correct, as we cannot expire them without
int alive = 0 ;
// data loss anyway; consequently, we also don't count them.
for ( srec = svars - > srecs ; srec ; srec = srec - > next ) {
if ( srec - > status & S_DEAD )
continue ;
// We completely ignore unpaired near-side messages, as we cannot expire
// them without data loss; consequently, we also don't count them.
// Note that we also ignore near-side messages we're currently propagating,
// Note that we also ignore near-side messages we're currently propagating,
// which delays expiration of some messages by one cycle. Otherwise, we'd have
// which delays expiration of some messages by one cycle. Otherwise, we'd
// to sequence flag propagation after message propagation to avoid a race
// have to sequence flag updating after message propagation to avoid a race
// with 3rd-party expunging, and that seems unreasonably expensive.
// with external expunging, and that seems unreasonably expensive.
alive = 0 ;
if ( ! srec - > uid [ F ] )
for ( tmsg = svars - > msgs [ F ] ; tmsg ; tmsg = tmsg - > next ) {
if ( tmsg - > status & M_DEAD )
continue ;
continue ;
if ( ! ( srec - > status & S_PENDING ) ) {
// We ignore unpaired far-side messages, as there is obviously nothing
// We ignore unpaired far-side messages, as there is obviously nothing
// to expire in the first place.
// to expire in the first place.
if ( ! ( srec = tmsg - > srec ) )
continue ;
if ( ! ( srec - > status & S_PENDING ) ) {
if ( ! srec - > msg [ N ] )
if ( ! srec - > msg [ N ] )
continue ; // Already expired or skipped.
continue ;
nflags = ( srec - > msg [ N ] - > flags | srec - > aflags [ N ] ) & ~ srec - > dflags [ N ] ;
nflags = ( srec - > msg [ N ] - > flags | srec - > aflags [ N ] ) & ~ srec - > dflags [ N ] ;
} else {
} else {
nflags = t msg- > flags ;
nflags = srec - > msg [ F ] - > flags ;
}
}
if ( ! ( nflags & F_DELETED ) | | ( srec - > status & ( S_EXPIRE | S_EXPIRED ) ) ) {
if ( ! ( nflags & F_DELETED ) | | ( srec - > status & ( S_EXPIRE | S_EXPIRED ) ) ) {
// The message is not deleted, or it is, but only due to being expired.
// The message is not deleted, or it is, but only due to being expired.
alive + + ;
arecs [ a live + + ] = ( alive_srec_t ) { srec , nflags } ;
}
}
}
}
todel = alive - svars - > chan - > max_messages ;
// Sort such that the messages which have been in the
// complete store longest expire first.
qsort ( arecs , alive , sizeof ( * arecs ) , cmp_srec_far ) ;
int todel = alive - svars - > chan - > max_messages ;
debug ( " %d alive messages, %d excess - expiring \n " , alive , todel ) ;
debug ( " %d alive messages, %d excess - expiring \n " , alive , todel ) ;
alive = 0 ;
int unseen = 0 ;
for ( tmsg = svars - > msgs [ F ] ; tmsg ; tmsg = tmsg - > next ) {
for ( int sri = 0 ; sri < alive ; sri + + ) {
if ( tmsg - > status & M_DEAD )
srec = arecs [ sri ] . srec ;
continue ;
nflags = arecs [ sri ] . flags ;
if ( ! ( srec = tmsg - > srec ) )
continue ;
if ( ! ( srec - > status & S_PENDING ) ) {
if ( ! srec - > msg [ N ] )
continue ;
nflags = ( srec - > msg [ N ] - > flags | srec - > aflags [ N ] ) & ~ srec - > dflags [ N ] ;
} else {
nflags = tmsg - > flags ;
}
if ( ! ( nflags & F_DELETED ) | | ( srec - > status & ( S_EXPIRE | S_EXPIRED ) ) ) {
if ( ( nflags & F_FLAGGED ) | |
if ( ( nflags & F_FLAGGED ) | |
! ( ( nflags & F_SEEN ) | | ( ( void ) ( todel > 0 & & alive + + ) , svars - > chan - > expire_unread > 0 ) ) ) {
! ( ( nflags & F_SEEN ) | | ( ( void ) ( todel > 0 & & unseen + + ) , svars - > chan - > expire_unread > 0 ) ) ) {
// Important messages are always fetched/kept.
// Important messages are always fetched/kept.
debug ( " pair(%u,%u) is important \n " , srec - > uid [ F ] , srec - > uid [ N ] ) ;
debug ( " pair(%u,%u) is important \n " , srec - > uid [ F ] , srec - > uid [ N ] ) ;
todel - - ;
todel - - ;
@ -1295,22 +1303,18 @@ box_loaded( int sts, message_t *msgs, int total_msgs, int recent_msgs, void *aux
todel - - ;
todel - - ;
}
}
}
}
}
debug ( " %d excess messages remain \n " , todel ) ;
debug ( " %d excess messages remain \n " , todel ) ;
if ( svars - > chan - > expire_unread < 0 & & alive * 2 > svars - > chan - > max_messages ) {
if ( svars - > chan - > expire_unread < 0 & & unseen * 2 > svars - > chan - > max_messages ) {
error ( " %s: %d unread messages in excess of MaxMessages (%d). \n "
error ( " %s: %d unread messages in excess of MaxMessages (%d). \n "
" Please set ExpireUnread to decide outcome. Skipping mailbox. \n " ,
" Please set ExpireUnread to decide outcome. Skipping mailbox. \n " ,
svars - > orig_name [ N ] , alive , svars - > chan - > max_messages ) ;
svars - > orig_name [ N ] , unseen , svars - > chan - > max_messages ) ;
svars - > ret | = SYNC_FAIL ;
svars - > ret | = SYNC_FAIL ;
cancel_sync ( svars ) ;
cancel_sync ( svars ) ;
return ;
return ;
}
}
for ( srec = svars - > srecs ; srec ; srec = srec - > next ) {
for ( int sri = 0 ; sri < alive ; sri + + ) {
if ( srec - > status & S_DEAD )
srec = arecs [ sri ] . srec ;
continue ;
if ( ! ( srec - > status & S_PENDING ) ) {
if ( ! ( srec - > status & S_PENDING ) ) {
if ( ! srec - > msg [ N ] )
continue ;
uchar nex = ( srec - > status / S_NEXPIRE ) & 1 ;
uchar nex = ( srec - > status / S_NEXPIRE ) & 1 ;
if ( nex ! = ( ( srec - > status / S_EXPIRED ) & 1 ) ) {
if ( nex ! = ( ( srec - > status / S_EXPIRED ) & 1 ) ) {
/* The record needs a state change ... */
/* The record needs a state change ... */
@ -1340,6 +1344,7 @@ box_loaded( int sts, message_t *msgs, int total_msgs, int recent_msgs, void *aux
}
}
}
}
}
}
free ( arecs ) ;
}
}
sync_ref ( svars ) ;
sync_ref ( svars ) ;