2008-03-09 19:14:36 +00:00
|
|
|
/*-
|
|
|
|
* Copyright (c) 2005-2008 Poul-Henning Kamp
|
|
|
|
* All rights reserved.
|
|
|
|
*
|
|
|
|
* Redistribution and use in source and binary forms, with or without
|
|
|
|
* modification, are permitted provided that the following conditions
|
|
|
|
* are met:
|
|
|
|
* 1. Redistributions of source code must retain the above copyright
|
|
|
|
* notice, this list of conditions and the following disclaimer.
|
|
|
|
* 2. Redistributions in binary form must reproduce the above copyright
|
|
|
|
* notice, this list of conditions and the following disclaimer in the
|
|
|
|
* documentation and/or other materials provided with the distribution.
|
|
|
|
*
|
|
|
|
* THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
|
|
|
|
* ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
|
|
|
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
|
|
|
|
* ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
|
|
|
|
* FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
|
|
|
|
* DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
|
|
|
|
* OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
|
|
|
|
* HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
|
|
|
|
* LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
|
|
|
|
* OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
|
|
|
|
* SUCH DAMAGE.
|
|
|
|
*
|
|
|
|
* $FreeBSD$
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <assert.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <unistd.h>
|
|
|
|
#include <time.h>
|
|
|
|
#include <sys/endian.h>
|
2010-12-07 16:30:52 +00:00
|
|
|
#if 0
|
|
|
|
#include <sys/uio.h>
|
|
|
|
#endif
|
2008-03-09 19:14:36 +00:00
|
|
|
|
|
|
|
#include <zlib.h>
|
|
|
|
|
|
|
|
#include "fifolog.h"
|
|
|
|
#include "libfifolog.h"
|
|
|
|
#include "libfifolog_int.h"
|
|
|
|
#include "fifolog_write.h"
|
|
|
|
#include "miniobj.h"
|
|
|
|
|
|
|
|
#define ALLOC(ptr, size) do { \
|
|
|
|
(*(ptr)) = calloc(size, 1); \
|
|
|
|
assert(*(ptr) != NULL); \
|
|
|
|
} while (0)
|
|
|
|
|
|
|
|
|
|
|
|
const char *fifolog_write_statnames[] = {
|
|
|
|
[FIFOLOG_PT_BYTES_PRE] = "Bytes before compression",
|
|
|
|
[FIFOLOG_PT_BYTES_POST] = "Bytes after compression",
|
|
|
|
[FIFOLOG_PT_WRITES] = "Writes",
|
|
|
|
[FIFOLOG_PT_FLUSH] = "Flushes",
|
|
|
|
[FIFOLOG_PT_SYNC] = "Syncs",
|
|
|
|
[FIFOLOG_PT_RUNTIME] = "Runtime"
|
|
|
|
};
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Check that everything is all right
|
|
|
|
*/
|
|
|
|
static void
|
|
|
|
fifolog_write_assert(const struct fifolog_writer *f)
|
|
|
|
{
|
|
|
|
|
|
|
|
CHECK_OBJ_NOTNULL(f, FIFOLOG_WRITER_MAGIC);
|
|
|
|
assert(f->ff->zs->next_out + f->ff->zs->avail_out == \
|
2010-12-07 16:30:52 +00:00
|
|
|
f->obuf + f->obufsize);
|
2008-03-09 19:14:36 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
struct fifolog_writer *
|
|
|
|
fifolog_write_new(void)
|
|
|
|
{
|
|
|
|
struct fifolog_writer *f;
|
|
|
|
|
2010-12-07 16:30:52 +00:00
|
|
|
ALLOC_OBJ(f, FIFOLOG_WRITER_MAGIC);
|
|
|
|
assert(f != NULL);
|
2008-03-09 19:14:36 +00:00
|
|
|
return (f);
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
fifolog_write_destroy(struct fifolog_writer *f)
|
|
|
|
{
|
|
|
|
CHECK_OBJ_NOTNULL(f, FIFOLOG_WRITER_MAGIC);
|
|
|
|
free(f);
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
fifolog_write_close(struct fifolog_writer *f)
|
|
|
|
{
|
|
|
|
|
|
|
|
CHECK_OBJ_NOTNULL(f, FIFOLOG_WRITER_MAGIC);
|
|
|
|
fifolog_int_close(&f->ff);
|
2008-03-19 10:56:51 +00:00
|
|
|
free(f->ff);
|
2010-12-07 16:30:52 +00:00
|
|
|
if (f->obuf != NULL)
|
|
|
|
free(f->obuf);
|
2008-03-09 19:14:36 +00:00
|
|
|
free(f);
|
|
|
|
}
|
|
|
|
|
|
|
|
const char *
|
|
|
|
fifolog_write_open(struct fifolog_writer *f, const char *fn, unsigned writerate, unsigned syncrate, int compression)
|
|
|
|
{
|
|
|
|
const char *es;
|
|
|
|
int i;
|
|
|
|
time_t now;
|
|
|
|
off_t o;
|
|
|
|
|
|
|
|
CHECK_OBJ_NOTNULL(f, FIFOLOG_WRITER_MAGIC);
|
|
|
|
|
|
|
|
/* Check for legal compression value */
|
|
|
|
if (compression < Z_DEFAULT_COMPRESSION ||
|
|
|
|
compression > Z_BEST_COMPRESSION)
|
|
|
|
return ("Illegal compression value");
|
|
|
|
|
|
|
|
f->writerate = writerate;
|
|
|
|
f->syncrate = syncrate;
|
|
|
|
f->compression = compression;
|
|
|
|
|
|
|
|
/* Reset statistics */
|
|
|
|
memset(f->cnt, 0, sizeof f->cnt);
|
|
|
|
|
|
|
|
es = fifolog_int_open(&f->ff, fn, 1);
|
|
|
|
if (es != NULL)
|
|
|
|
return (es);
|
|
|
|
es = fifolog_int_findend(f->ff, &o);
|
|
|
|
if (es != NULL)
|
|
|
|
return (es);
|
2009-02-02 14:29:15 +00:00
|
|
|
i = fifolog_int_read(f->ff, o);
|
|
|
|
if (i)
|
|
|
|
return ("Read error, looking for seq");
|
|
|
|
f->seq = be32dec(f->ff->recbuf);
|
|
|
|
if (f->seq == 0) {
|
|
|
|
/* Empty fifolog */
|
|
|
|
f->seq = random();
|
2008-03-09 19:14:36 +00:00
|
|
|
} else {
|
|
|
|
f->recno = o + 1;
|
2009-02-02 14:29:15 +00:00
|
|
|
f->seq++;
|
2008-03-09 19:14:36 +00:00
|
|
|
}
|
|
|
|
|
2010-12-07 16:30:52 +00:00
|
|
|
f->obufsize = f->ff->recsize;
|
|
|
|
ALLOC(&f->obuf, f->obufsize);
|
|
|
|
|
2008-03-09 19:14:36 +00:00
|
|
|
i = deflateInit(f->ff->zs, (int)f->compression);
|
|
|
|
assert(i == Z_OK);
|
|
|
|
|
|
|
|
f->flag |= FIFOLOG_FLG_RESTART;
|
2010-12-07 16:30:52 +00:00
|
|
|
f->flag |= FIFOLOG_FLG_SYNC;
|
|
|
|
f->ff->zs->next_out = f->obuf + 9;
|
|
|
|
f->ff->zs->avail_out = f->obufsize - 9;
|
2008-03-09 19:14:36 +00:00
|
|
|
|
|
|
|
time(&now);
|
|
|
|
f->starttime = now;
|
2010-12-07 16:30:52 +00:00
|
|
|
f->lastsync = now;
|
|
|
|
f->lastwrite = now;
|
2008-03-09 19:14:36 +00:00
|
|
|
|
|
|
|
fifolog_write_assert(f);
|
|
|
|
return (NULL);
|
|
|
|
}
|
|
|
|
|
2010-12-07 16:30:52 +00:00
|
|
|
static int
|
|
|
|
fifolog_write_output(struct fifolog_writer *f, int fl, time_t now)
|
2008-03-09 19:14:36 +00:00
|
|
|
{
|
2010-12-07 16:30:52 +00:00
|
|
|
long h, l = f->ff->zs->next_out - f->obuf;
|
|
|
|
int i, w;
|
|
|
|
|
|
|
|
h = 4; /* seq */
|
|
|
|
be32enc(f->obuf, f->seq);
|
|
|
|
f->obuf[h] = f->flag;
|
|
|
|
h += 1; /* flag */
|
|
|
|
if (f->flag & FIFOLOG_FLG_SYNC) {
|
|
|
|
be32enc(f->obuf + h, now);
|
|
|
|
h += 4; /* timestamp */
|
|
|
|
}
|
2008-03-09 19:14:36 +00:00
|
|
|
|
2010-12-07 16:30:52 +00:00
|
|
|
assert(l <= (long)f->ff->recsize);
|
|
|
|
assert(l >= h);
|
|
|
|
if (l == h)
|
|
|
|
return (0);
|
|
|
|
|
|
|
|
|
|
|
|
if (h + l < (long)f->ff->recsize && fl == Z_NO_FLUSH)
|
|
|
|
return (0);
|
|
|
|
|
|
|
|
w = f->ff->recsize - l;
|
|
|
|
if (w > 255) {
|
|
|
|
be32enc(f->obuf + f->ff->recsize - 4, w);
|
|
|
|
f->obuf[4] |= FIFOLOG_FLG_4BYTE;
|
|
|
|
} else if (w > 0) {
|
|
|
|
f->obuf[f->ff->recsize - 1] = w;
|
|
|
|
f->obuf[4] |= FIFOLOG_FLG_1BYTE;
|
2008-03-09 19:14:36 +00:00
|
|
|
}
|
2010-12-07 16:30:52 +00:00
|
|
|
|
|
|
|
f->cnt[FIFOLOG_PT_BYTES_POST] += w;
|
|
|
|
|
|
|
|
#ifdef DBG
|
|
|
|
fprintf(stderr, "W: fl=%d h=%ld l=%ld w=%d recno=%jd fx %02x\n",
|
|
|
|
fl, h, l, w, f->recno, f->obuf[4]);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
i = pwrite(f->ff->fd, f->obuf, f->ff->recsize,
|
|
|
|
(f->recno + 1) * f->ff->recsize);
|
|
|
|
assert(i == (int)f->ff->recsize);
|
|
|
|
|
2008-03-09 19:14:36 +00:00
|
|
|
f->cnt[FIFOLOG_PT_WRITES]++;
|
2010-12-07 16:30:52 +00:00
|
|
|
|
|
|
|
f->lastwrite = now;
|
|
|
|
f->seq++;
|
|
|
|
f->recno++;
|
|
|
|
#ifdef DBG
|
|
|
|
if (f->flag)
|
|
|
|
fprintf(stderr, "SYNC- %d\n", __LINE__);
|
|
|
|
#endif
|
|
|
|
f->flag = 0;
|
|
|
|
|
|
|
|
memset(f->obuf, 0, f->obufsize);
|
|
|
|
f->ff->zs->next_out = f->obuf + 5;
|
|
|
|
f->ff->zs->avail_out = f->obufsize - 5;
|
|
|
|
return (1);
|
2008-03-09 19:14:36 +00:00
|
|
|
}
|
|
|
|
|
2010-12-07 16:30:52 +00:00
|
|
|
static void
|
|
|
|
fifolog_write_gzip(struct fifolog_writer *f, const void *p, int len, time_t now, int fin)
|
2008-03-09 19:14:36 +00:00
|
|
|
{
|
2010-12-07 16:30:52 +00:00
|
|
|
int i, fl;
|
2008-03-09 19:14:36 +00:00
|
|
|
|
2010-12-07 16:30:52 +00:00
|
|
|
f->cnt[FIFOLOG_PT_BYTES_PRE] += len;
|
2008-03-09 19:14:36 +00:00
|
|
|
|
2010-12-07 16:30:52 +00:00
|
|
|
if (fin == 0)
|
|
|
|
fl = Z_NO_FLUSH;
|
|
|
|
else if (f->cleanup || now >= (int)(f->lastsync + f->syncrate)) {
|
2008-03-09 19:14:36 +00:00
|
|
|
f->cleanup = 0;
|
|
|
|
fl = Z_FINISH;
|
|
|
|
f->cnt[FIFOLOG_PT_SYNC]++;
|
2010-12-07 16:30:52 +00:00
|
|
|
} else if (now >= (int)(f->lastwrite + f->writerate)) {
|
2008-03-09 19:14:36 +00:00
|
|
|
fl = Z_SYNC_FLUSH;
|
|
|
|
f->cnt[FIFOLOG_PT_FLUSH]++;
|
2010-12-07 16:30:52 +00:00
|
|
|
} else if (p == NULL)
|
|
|
|
return;
|
2008-03-09 19:14:36 +00:00
|
|
|
else
|
|
|
|
fl = Z_NO_FLUSH;
|
|
|
|
|
2010-12-07 16:30:52 +00:00
|
|
|
f->ff->zs->avail_in = len;
|
|
|
|
f->ff->zs->next_in = (void*)(uintptr_t)p;
|
|
|
|
#ifdef DBG
|
|
|
|
if (fl != Z_NO_FLUSH)
|
|
|
|
fprintf(stderr, "Z len %3d fin %d now %ld fl %d ai %u ao %u\n",
|
|
|
|
len, fin, now, fl,
|
|
|
|
f->ff->zs->avail_in,
|
|
|
|
f->ff->zs->avail_out);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
while (1) {
|
2008-03-09 19:14:36 +00:00
|
|
|
i = deflate(f->ff->zs, fl);
|
|
|
|
|
2010-12-07 16:30:52 +00:00
|
|
|
#ifdef DBG
|
|
|
|
if (i || f->ff->zs->avail_in)
|
|
|
|
fprintf(stderr, "fl = %d, i = %d ai = %u ao = %u fx=%02x\n", fl, i,
|
|
|
|
f->ff->zs->avail_in,
|
|
|
|
f->ff->zs->avail_out, f->flag);
|
|
|
|
#endif
|
2008-03-09 19:14:36 +00:00
|
|
|
|
2010-12-07 16:30:52 +00:00
|
|
|
assert(i == Z_OK || i == Z_BUF_ERROR || i == Z_STREAM_END);
|
|
|
|
assert(f->ff->zs->avail_in == 0);
|
2008-03-09 19:14:36 +00:00
|
|
|
|
2010-12-07 16:30:52 +00:00
|
|
|
if (!fifolog_write_output(f, fl, now))
|
|
|
|
break;
|
2008-03-09 19:14:36 +00:00
|
|
|
}
|
2010-12-07 16:30:52 +00:00
|
|
|
assert(f->ff->zs->avail_in == 0);
|
|
|
|
if (fl == Z_FINISH) {
|
|
|
|
f->flag |= FIFOLOG_FLG_SYNC;
|
|
|
|
f->ff->zs->next_out = f->obuf + 9;
|
|
|
|
f->ff->zs->avail_out = f->obufsize - 9;
|
|
|
|
f->lastsync = now;
|
|
|
|
#ifdef DBG
|
|
|
|
fprintf(stderr, "SYNC %d\n", __LINE__);
|
|
|
|
#endif
|
|
|
|
assert(Z_OK == deflateReset(f->ff->zs));
|
2008-03-09 19:14:36 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-12-07 16:30:52 +00:00
|
|
|
int
|
|
|
|
fifolog_write_poll(struct fifolog_writer *f, time_t now)
|
2008-03-09 19:14:36 +00:00
|
|
|
{
|
2010-12-07 16:30:52 +00:00
|
|
|
if (now == 0)
|
|
|
|
time(&now);
|
|
|
|
fifolog_write_gzip(f, NULL, 0, now, 1);
|
|
|
|
return (0);
|
2008-03-09 19:14:36 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Attempt to write an entry.
|
|
|
|
* Return zero if there is no space, one otherwise
|
|
|
|
*/
|
|
|
|
|
|
|
|
int
|
|
|
|
fifolog_write_bytes(struct fifolog_writer *f, uint32_t id, time_t now, const void *ptr, unsigned len)
|
|
|
|
{
|
|
|
|
const unsigned char *p;
|
2010-12-07 16:30:52 +00:00
|
|
|
uint8_t buf[4];
|
2008-03-09 19:14:36 +00:00
|
|
|
|
|
|
|
fifolog_write_assert(f);
|
|
|
|
assert(!(id & (FIFOLOG_TIMESTAMP|FIFOLOG_LENGTH)));
|
|
|
|
assert(ptr != NULL);
|
|
|
|
|
2008-03-19 10:56:51 +00:00
|
|
|
p = ptr;
|
2008-03-09 19:14:36 +00:00
|
|
|
if (len == 0) {
|
2008-03-19 10:56:51 +00:00
|
|
|
len = strlen(ptr) + 1;
|
2008-03-09 19:14:36 +00:00
|
|
|
} else {
|
|
|
|
assert(len <= 255);
|
|
|
|
id |= FIFOLOG_LENGTH;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Now do timestamp, if needed */
|
|
|
|
if (now == 0)
|
|
|
|
time(&now);
|
|
|
|
|
|
|
|
if (now != f->last) {
|
|
|
|
id |= FIFOLOG_TIMESTAMP;
|
|
|
|
f->last = now;
|
|
|
|
}
|
|
|
|
|
2010-12-07 16:30:52 +00:00
|
|
|
/* Emit instance+flag */
|
|
|
|
be32enc(buf, id);
|
|
|
|
fifolog_write_gzip(f, buf, 4, now, 0);
|
2008-03-09 19:14:36 +00:00
|
|
|
|
|
|
|
if (id & FIFOLOG_TIMESTAMP) {
|
2010-12-07 16:30:52 +00:00
|
|
|
be32enc(buf, (uint32_t)f->last);
|
|
|
|
fifolog_write_gzip(f, buf, 4, now, 0);
|
2008-03-09 19:14:36 +00:00
|
|
|
}
|
|
|
|
if (id & FIFOLOG_LENGTH) {
|
2010-12-07 16:30:52 +00:00
|
|
|
buf[0] = (u_char)len;
|
|
|
|
fifolog_write_gzip(f, buf, 1, now, 0);
|
2008-03-09 19:14:36 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
assert (len > 0);
|
2010-12-07 16:30:52 +00:00
|
|
|
#if 1
|
|
|
|
if (len > f->ibufsize) {
|
|
|
|
free(f->ibuf);
|
|
|
|
f->ibufsize = len;
|
|
|
|
ALLOC(&f->ibuf, f->ibufsize);
|
|
|
|
}
|
|
|
|
memcpy(f->ibuf, p, len);
|
|
|
|
fifolog_write_gzip(f, f->ibuf, len, now, 1);
|
|
|
|
#else
|
|
|
|
fifolog_write_gzip(f, p, len, now, 1);
|
|
|
|
#endif
|
2008-03-09 19:14:36 +00:00
|
|
|
fifolog_write_assert(f);
|
|
|
|
return (1);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Write an entry, polling until success.
|
|
|
|
* Long binary entries are broken into 255 byte chunks.
|
|
|
|
*/
|
|
|
|
|
|
|
|
void
|
|
|
|
fifolog_write_bytes_poll(struct fifolog_writer *f, uint32_t id, time_t now, const void *ptr, unsigned len)
|
|
|
|
{
|
|
|
|
u_int l;
|
|
|
|
const unsigned char *p;
|
|
|
|
|
|
|
|
fifolog_write_assert(f);
|
|
|
|
|
|
|
|
assert(!(id & (FIFOLOG_TIMESTAMP|FIFOLOG_LENGTH)));
|
|
|
|
assert(ptr != NULL);
|
|
|
|
|
|
|
|
if (len == 0) {
|
|
|
|
while (!fifolog_write_bytes(f, id, now, ptr, len)) {
|
|
|
|
(void)usleep(10000);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
p = ptr;
|
|
|
|
for (p = ptr; len > 0; len -= l, p += l) {
|
|
|
|
l = len;
|
|
|
|
if (l > 255)
|
|
|
|
l = 255;
|
|
|
|
while (!fifolog_write_bytes(f, id, now, p, l)) {
|
|
|
|
(void)usleep(10000);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
fifolog_write_assert(f);
|
|
|
|
}
|
|
|
|
|
|
|
|
int
|
|
|
|
fifolog_write_flush(struct fifolog_writer *f)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
|
|
|
|
fifolog_write_assert(f);
|
|
|
|
|
|
|
|
f->cleanup = 1;
|
|
|
|
for (i = 0; fifolog_write_poll(f, 0); i = 1)
|
|
|
|
continue;
|
|
|
|
fifolog_write_assert(f);
|
|
|
|
return (i);
|
|
|
|
}
|