You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
173 lines
4.1 KiB
173 lines
4.1 KiB
3 months ago
|
/*
|
||
|
* UCW Library -- Atomic Buffered Write to Files
|
||
|
*
|
||
|
* (c) 2006 Martin Mares <mj@ucw.cz>
|
||
|
*
|
||
|
* This software may be freely distributed and used according to the terms
|
||
|
* of the GNU Lesser General Public License.
|
||
|
*/
|
||
|
|
||
|
#include <ucw/lib.h>
|
||
|
#include <ucw/fastbuf.h>
|
||
|
#include <ucw/io.h>
|
||
|
#include <ucw/conf.h>
|
||
|
#include <ucw/trans.h>
|
||
|
|
||
|
#include <string.h>
|
||
|
#include <fcntl.h>
|
||
|
#include <unistd.h>
|
||
|
|
||
|
static uint trace;
|
||
|
|
||
|
#ifndef TEST
|
||
|
|
||
|
static struct cf_section fbatomic_config = {
|
||
|
CF_ITEMS {
|
||
|
CF_UINT("Trace", &trace),
|
||
|
CF_END
|
||
|
}
|
||
|
};
|
||
|
|
||
|
static void CONSTRUCTOR fbatomic_init_config(void)
|
||
|
{
|
||
|
cf_declare_section("FBAtomic", &fbatomic_config, 1);
|
||
|
}
|
||
|
|
||
|
#endif
|
||
|
|
||
|
#define FB_ATOMIC(f) ((struct fb_atomic *)(f))
|
||
|
#define TRACE(m...) do { if(trace) msg(L_DEBUG, "FB_ATOMIC: " m); } while(0)
|
||
|
|
||
|
struct fb_atomic_file {
|
||
|
int fd;
|
||
|
int use_count;
|
||
|
int record_len;
|
||
|
uint locked;
|
||
|
byte name[1];
|
||
|
};
|
||
|
|
||
|
void
|
||
|
fbatomic_internal_write(struct fastbuf *f)
|
||
|
{
|
||
|
struct fb_atomic_file *af = FB_ATOMIC(f)->af;
|
||
|
int size = f->bptr - f->buffer;
|
||
|
if (size)
|
||
|
{
|
||
|
ASSERT(af->record_len < 0 || !(size % af->record_len));
|
||
|
int res = write(af->fd, f->buffer, size);
|
||
|
if (res < 0)
|
||
|
bthrow(f, "write", "Error writing %s: %m", f->name);
|
||
|
if (res != size)
|
||
|
bthrow(f, "write", "Unexpected partial write to %s: written only %d bytes of %d", f->name, res, size);
|
||
|
f->bptr = f->buffer;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
static void
|
||
|
fbatomic_spout(struct fastbuf *f)
|
||
|
{
|
||
|
if (f->bptr < f->bufend) /* Explicit flushes should be ignored */
|
||
|
return;
|
||
|
|
||
|
struct fb_atomic *F = FB_ATOMIC(f);
|
||
|
if (F->af->locked)
|
||
|
{
|
||
|
uint written = f->bptr - f->buffer;
|
||
|
uint size = f->bufend - f->buffer + F->slack_size;
|
||
|
F->slack_size *= 2;
|
||
|
TRACE("Reallocating buffer for atomic file %s with slack %d", f->name, F->slack_size);
|
||
|
f->buffer = xrealloc(f->buffer, size);
|
||
|
f->bufend = f->buffer + size;
|
||
|
f->bptr = f->buffer + written;
|
||
|
f->bstop = f->buffer;
|
||
|
F->expected_max_bptr = f->bufend - F->slack_size;
|
||
|
}
|
||
|
else
|
||
|
fbatomic_internal_write(f);
|
||
|
}
|
||
|
|
||
|
static void
|
||
|
fbatomic_close(struct fastbuf *f)
|
||
|
{
|
||
|
struct fb_atomic_file *af = FB_ATOMIC(f)->af;
|
||
|
if (!(f->flags & FB_DEAD))
|
||
|
fbatomic_internal_write(f); /* Need to flush explicitly, because the file can be locked */
|
||
|
if (!--af->use_count)
|
||
|
{
|
||
|
close(af->fd);
|
||
|
xfree(af);
|
||
|
}
|
||
|
xfree(f);
|
||
|
}
|
||
|
|
||
|
struct fastbuf *
|
||
|
fbatomic_open(const char *name, struct fastbuf *master, uint bufsize, int record_len)
|
||
|
{
|
||
|
struct fb_atomic *F = xmalloc_zero(sizeof(*F));
|
||
|
struct fastbuf *f = &F->fb;
|
||
|
struct fb_atomic_file *af;
|
||
|
if (master)
|
||
|
{
|
||
|
af = FB_ATOMIC(master)->af;
|
||
|
af->use_count++;
|
||
|
ASSERT(af->record_len == record_len);
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
int fd = ucw_open(name, O_WRONLY | O_CREAT | O_TRUNC | O_APPEND, 0666);
|
||
|
if (fd < 0)
|
||
|
trans_throw("ucw.fb.open", NULL, "Cannot create %s: %m", name);
|
||
|
af = xmalloc_zero(sizeof(*af) + strlen(name));
|
||
|
af->fd = fd;
|
||
|
af->use_count = 1;
|
||
|
af->record_len = record_len;
|
||
|
af->locked = (record_len < 0);
|
||
|
strcpy(af->name, name);
|
||
|
}
|
||
|
F->af = af;
|
||
|
if (record_len > 0 && bufsize % record_len)
|
||
|
bufsize += record_len - (bufsize % record_len);
|
||
|
f->buffer = xmalloc(bufsize);
|
||
|
f->bufend = f->buffer + bufsize;
|
||
|
F->slack_size = (record_len < 0) ? -record_len : 0;
|
||
|
ASSERT(bufsize > F->slack_size);
|
||
|
F->expected_max_bptr = f->bufend - F->slack_size;
|
||
|
f->bptr = f->bstop = f->buffer;
|
||
|
f->name = af->name;
|
||
|
f->spout = fbatomic_spout;
|
||
|
f->close = fbatomic_close;
|
||
|
return f;
|
||
|
}
|
||
|
|
||
|
#ifdef TEST
|
||
|
|
||
|
int main(int argc UNUSED, char **argv UNUSED)
|
||
|
{
|
||
|
struct fastbuf *f, *g;
|
||
|
|
||
|
// Always trace in the test
|
||
|
trace = 1;
|
||
|
|
||
|
msg(L_INFO, "Testing block writes");
|
||
|
f = fbatomic_open("test", NULL, 16, 4);
|
||
|
for (u32 i=0; i<17; i++)
|
||
|
bwrite(f, &i, 4);
|
||
|
bclose(f);
|
||
|
|
||
|
msg(L_INFO, "Testing interleaved var-size writes");
|
||
|
f = fbatomic_open("test2", NULL, 23, -5);
|
||
|
g = fbatomic_open("test2", f, 23, -5);
|
||
|
for (int i=0; i<100; i++)
|
||
|
{
|
||
|
struct fastbuf *x = (i%2) ? g : f;
|
||
|
bprintf(x, "%c<%d>\n", "fg"[i%2], ((259309*i) % 1000000) >> (i % 8));
|
||
|
fbatomic_commit(x);
|
||
|
}
|
||
|
bclose(f);
|
||
|
bclose(g);
|
||
|
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
#endif
|