Skip to content
Snippets Groups Projects
Commit 0f13f47c authored by Marek Vavrusa's avatar Marek Vavrusa Committed by Jan Kadlec
Browse files

Resolved journal file size limiting and trimming.

Commit fixes #935, #934.
parent d1c7389f
No related branches found
No related tags found
No related merge requests found
#include <stdlib.h>
#include <stdio.h>
#include <string.h>
#include <unistd.h>
#include <sys/stat.h>
#include "knot/other/error.h"
#include "knot/other/debug.h"
#include "journal.h"
/*! \brief Infinite file size limit. */
#define FSLIMIT_INF (~((size_t)0))
static inline int sfread(void *dst, size_t len, FILE *fp)
{
return fread(dst, len, 1, fp) == 1;
......@@ -79,12 +84,18 @@ int journal_create(const char *fn, uint16_t max_nodes)
return KNOT_EOK;
}
journal_t* journal_open(const char *fn)
journal_t* journal_open(const char *fn, int fslimit)
{
/*! \todo Memory mapping may be faster than stdio? */
/*! \todo Lock file. */
/* Check file. */
struct stat st;
if (stat(fn, &st) < 0) {
return 0;
}
/* Open journal file for r/w. */
FILE *fp = fopen(fn, "r+");
if (!fp) {
......@@ -141,6 +152,14 @@ journal_t* journal_open(const char *fn)
return 0;
}
/* Set file size. */
j->fsize = st.st_size;
if (fslimit < 0) {
j->fslimit = FSLIMIT_INF;
} else {
j->fslimit = (size_t)fslimit;
}
/*! \todo Some file checksum, check node integrity. */
debug_journal("journal: opened journal size=%u, queue=<%u, %u>, fd=%d\n",
max_nodes, j->qhead, j->qtail, fileno(j->fp));
......@@ -202,18 +221,39 @@ int journal_write(journal_t *journal, int id, const char *src, size_t size)
/* Find next free node. */
uint16_t jnext = (journal->qtail + 1) % journal->max_nodes;
debug_journal("journal: writing id=%d, node=%u, next=%u\n",
id, journal->qtail, jnext);
debug_journal("journal: will write id=%d, node=%u, size=%zu, fsize=%zu\n",
id, journal->qtail, size, journal->fsize);
/* Calculate remaining bytes to reach file size limit. */
size_t fs_remaining = journal->fslimit - journal->fsize;
/* Increase free segment if on the end of file. */
journal_node_t *n = journal->nodes + journal->qtail;
if (journal->free.len < size) {
if (journal->free.pos + journal->free.len == journal->fsize) {
debug_journal("journal: * is last node\n");
/* Grow journal file until the size limit. */
if(journal->free.len < size && size <= fs_remaining) {
size_t diff = size - journal->free.len;
debug_journal("journal: * growing by +%zu, pos=%u, new fsize=%zu\n",
diff, journal->free.pos,
journal->fsize + diff);
journal->fsize += diff; /* Appending increases file size. */
journal->free.len += diff;
/* Increase on the end of node queue / uninitialized. */
if (journal->qtail > jnext || n->flags == JOURNAL_NULL) {
debug_journal("journal: growing by +%zu, pos=%u\n",
size - journal->free.len, journal->free.pos);
journal->free.len = size;
}
/* Rewind if resize is needed, but the limit is reached. */
if(journal->free.len < size && size > fs_remaining) {
journal_node_t *head = journal->nodes + journal->qhead;
journal->fsize = journal->free.pos;
journal->free.pos = head->pos;
journal->free.len = 0;
debug_journal("journal: * fslimit reached, rewinding to %u\n",
head->pos);
debug_journal("journal: * file size trimmed to %zu\n",
journal->fsize);
}
}
......@@ -230,7 +270,7 @@ int journal_write(journal_t *journal, int id, const char *src, size_t size)
return KNOT_ERROR;
}
debug_journal("journal: evicted node=%u, growing by +%u\n",
debug_journal("journal: * evicted node=%u, growing by +%u\n",
journal->qhead, head->len);
/* Write back query state. */
......@@ -254,7 +294,7 @@ int journal_write(journal_t *journal, int id, const char *src, size_t size)
n->len = size;
n->flags = JOURNAL_FREE;
if (!sfwrite(n, node_len, journal->fp)) {
debug_journal("journal: failed to writeback node=%d to %u\n",
debug_journal("journal: failed to writeback node=%d to %ld\n",
n->id, jn_fpos);
return KNOT_ERROR;
}
......@@ -272,23 +312,26 @@ int journal_write(journal_t *journal, int id, const char *src, size_t size)
return KNOT_ERROR;
}
debug_journal("journal: written node=%u, data=<%u, %u>\n",
journal->qtail, n->pos, n->pos + n->len);
/* Handle free segment on node rotation. */
if (journal->qtail > jnext && journal->fslimit == FSLIMIT_INF) {
/* Trim free space. */
journal->fsize -= journal->free.len;
debug_journal("journal: * trimmed filesize to %zu\n",
journal->fsize);
/* Trim free space on the last node. */
if (journal->qtail > jnext) {
debug_journal("journal: trimming free space, next=%u\n",
jnext);
journal_node_t *next = journal->nodes + jnext;
journal->free.pos = next->pos;
/* Rewind free segment. */
journal_node_t *n = journal->nodes + jnext;
journal->free.pos = n->pos;
journal->free.len = 0;
} else {
/* Mark used space. */
journal->free.pos += size;
journal->free.len -= size;
debug_journal("journal: free segment <%u, %u>\n",
journal->free.pos, journal->free.pos + journal->free.len);
}
debug_journal("journal: finished node=%u, data=<%u, %u> free=<%u, %u>\n",
journal->qtail, n->pos, n->pos + n->len,
journal->free.pos, journal->free.pos + journal->free.len);
/* Node write successful. */
journal->qtail = jnext;
......
......@@ -69,6 +69,8 @@ typedef struct journal_t
uint16_t max_nodes; /*!< Number of nodes. */
uint16_t qhead; /*!< Node queue head. */
uint16_t qtail; /*!< Node queue tail. */
size_t fsize; /*!< Journal file size. */
size_t fslimit; /*!< File size limit. */
journal_node_t free; /*!< Free segment. */
journal_node_t nodes[]; /*!< Array of nodes. */
} journal_t;
......@@ -95,11 +97,12 @@ int journal_create(const char *fn, uint16_t max_nodes);
* \brief Open journal file for read/write.
*
* \param fn Journal file name.
* \param fslimit File size limit (-1 for no limit).
*
* \retval new journal instance if successful.
* \retval NULL on error.
*/
journal_t* journal_open(const char *fn);
journal_t* journal_open(const char *fn, int fslimit);
/*!
* \brief Fetch entry node for given identifier.
......
......@@ -42,7 +42,8 @@ static int journal_tests_count(int argc, char *argv[])
static int journal_tests_run(int argc, char *argv[])
{
/* Test 1: Create tmpfile. */
const int jsize = 256;
const int fsize = 8092;
const int jsize = 512;
char jfn_buf[] = "/tmp/journal.XXXXXX";
int tmp_fd = mkstemp(jfn_buf);
ok(tmp_fd >= 0, "journal: create temporary file");
......@@ -54,7 +55,7 @@ static int journal_tests_run(int argc, char *argv[])
ok(ret == KNOT_EOK, "journal: create journal '%s'", jfilename);
/* Test 3: Open journal. */
journal_t *j = journal_open(jfilename);
journal_t *j = journal_open(jfilename, fsize);
ok(j != 0, "journal: open");
/* Test 4: Write entry to log. */
......@@ -85,7 +86,7 @@ static int journal_tests_run(int argc, char *argv[])
}
/* Store some key on the end. */
if (i == itcount - jsize/5) {
if (i == itcount - 2) {
chk_key = key;
memcpy(chk_buf, tmpbuf, sizeof(chk_buf));
}
......@@ -101,7 +102,7 @@ static int journal_tests_run(int argc, char *argv[])
/* Test 9: Reopen log and re-read value. */
memset(tmpbuf, 0, sizeof(tmpbuf));
journal_close(j);
j = journal_open(jfilename);
j = journal_open(jfilename, fsize);
journal_read(j, chk_key, tmpbuf);
ret = strncmp(chk_buf, tmpbuf, sizeof(chk_buf));
ok(ret == 0, "journal: read data integrity check after close/open");
......
0% Loading or .
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment