uint64_t next_inum; /* Next available inode for allocation */
BlueSkyStore *store;
+
+ /* Accounting for memory used for caches. Space is measured in blocks, not
+ * bytes. We track both total data in the caches and dirty data (total
+ * data includes dirty data). Updates to these variables must be made
+ * atomically. */
+ gint cache_total, cache_dirty;
} BlueSkyFS;
/* Inode number of the root directory. */
void bluesky_debug_dump(BlueSkyFS *fs)
{
g_print("*** DEBUG DUMP FOR FILESYSTEM %s ***\n", fs->name);
+ g_print("Cached blocks: %d\tDirty blocks: %d\n",
+ g_atomic_int_get(&fs->cache_total),
+ g_atomic_int_get(&fs->cache_dirty));
g_print("Cached inodes: %u\tNext inode: %"PRIu64"\n",
g_hash_table_size(fs->inodes), fs->next_inum);
break;
}
+ if (block->type != BLUESKY_BLOCK_CACHED
+ && block->type != BLUESKY_BLOCK_DIRTY)
+ g_atomic_int_add(&inode->fs->cache_total, 1);
+ if (block->type != BLUESKY_BLOCK_DIRTY)
+ g_atomic_int_add(&inode->fs->cache_dirty, 1);
+
block->type = BLUESKY_BLOCK_DIRTY;
}
for (guint i = inode->blocks->len; i < blocks; i++) {
BlueSkyBlock *b = &g_array_index(inode->blocks, BlueSkyBlock, i);
g_free(b->ref);
+ if (b->type == BLUESKY_BLOCK_CACHED
+ || b->type == BLUESKY_BLOCK_DIRTY)
+ g_atomic_int_add(&inode->fs->cache_total, -1);
+ if (b->type == BLUESKY_BLOCK_DIRTY)
+ g_atomic_int_add(&inode->fs->cache_dirty, -1);
bluesky_string_unref(b->data);
}
g_array_set_size(inode->blocks, blocks);
bluesky_store_async_wait(async);
bluesky_store_async_unref(async);
+ g_atomic_int_add(&fs->cache_total, 1);
}
/* Write the given block to cloud-backed storage and mark it clean. */
block->ref = name;
block->type = BLUESKY_BLOCK_CACHED;
+ g_atomic_int_add(&fs->cache_dirty, -1);
g_checksum_free(csum);
}
"Dropping block %d of inode %"PRIu64" from cache",
i, inode->inum);
+ g_log("bluesky/cache", G_LOG_LEVEL_DEBUG,
+ " (reference count was %d)", b->data->refcount);
bluesky_string_unref(b->data);
b->data = NULL;
b->type = BLUESKY_BLOCK_REF;
+ g_atomic_int_add(&inode->fs->cache_total, -1);
}
}
}