Workshop o mikrokontrolérech na SKSP 2024.
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

2 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