2012-04-09 11:11:00 +02:00
|
|
|
/* rio.c is a simple stream-oriented I/O abstraction that provides an interface
|
|
|
|
* to write code that can consume/produce data using different concrete input
|
2012-11-08 18:25:23 +01:00
|
|
|
* and output devices. For instance the same rdb.c code using the rio
|
|
|
|
* abstraction can be used to read and write the RDB format using in-memory
|
|
|
|
* buffers or files.
|
2012-04-09 11:11:00 +02:00
|
|
|
*
|
|
|
|
* A rio object provides the following methods:
|
|
|
|
* read: read from stream.
|
|
|
|
* write: write to stream.
|
|
|
|
* tell: get the current offset.
|
|
|
|
*
|
|
|
|
* It is also possible to set a 'checksum' method that is used by rio.c in order
|
|
|
|
* to compute a checksum of the data written or read, or to query the rio object
|
2012-11-08 18:25:23 +01:00
|
|
|
* for the current checksum.
|
|
|
|
*
|
|
|
|
* ----------------------------------------------------------------------------
|
|
|
|
*
|
|
|
|
* Copyright (c) 2009-2012, Pieter Noordhuis <pcnoordhuis at gmail dot com>
|
|
|
|
* Copyright (c) 2009-2012, Salvatore Sanfilippo <antirez at gmail dot com>
|
|
|
|
* All rights reserved.
|
|
|
|
*
|
|
|
|
* Redistribution and use in source and binary forms, with or without
|
|
|
|
* modification, are permitted provided that the following conditions are met:
|
|
|
|
*
|
|
|
|
* * Redistributions of source code must retain the above copyright notice,
|
|
|
|
* this list of conditions and the following disclaimer.
|
|
|
|
* * 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.
|
|
|
|
* * Neither the name of Redis nor the names of its contributors may be used
|
|
|
|
* to endorse or promote products derived from this software without
|
|
|
|
* specific prior written permission.
|
|
|
|
*
|
|
|
|
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS 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 COPYRIGHT OWNER 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.
|
|
|
|
*/
|
|
|
|
|
2012-04-09 11:11:00 +02:00
|
|
|
|
2011-10-23 10:57:01 +02:00
|
|
|
#include "fmacros.h"
|
2011-05-13 17:31:00 +02:00
|
|
|
#include <string.h>
|
2011-10-23 10:57:01 +02:00
|
|
|
#include <stdio.h>
|
2013-04-03 15:31:40 +02:00
|
|
|
#include <unistd.h>
|
2011-05-13 17:31:00 +02:00
|
|
|
#include "rio.h"
|
|
|
|
#include "util.h"
|
2012-11-29 14:20:08 +01:00
|
|
|
#include "crc64.h"
|
2013-04-03 15:31:40 +02:00
|
|
|
#include "config.h"
|
|
|
|
#include "redis.h"
|
2012-04-09 12:36:44 +02:00
|
|
|
|
2011-05-13 17:31:00 +02:00
|
|
|
/* Returns 1 or 0 for success/failure. */
|
|
|
|
static size_t rioBufferWrite(rio *r, const void *buf, size_t len) {
|
|
|
|
r->io.buffer.ptr = sdscatlen(r->io.buffer.ptr,(char*)buf,len);
|
|
|
|
r->io.buffer.pos += len;
|
2011-09-22 15:54:08 +02:00
|
|
|
return 1;
|
2011-05-13 17:31:00 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Returns 1 or 0 for success/failure. */
|
|
|
|
static size_t rioBufferRead(rio *r, void *buf, size_t len) {
|
|
|
|
if (sdslen(r->io.buffer.ptr)-r->io.buffer.pos < len)
|
2011-09-22 15:54:08 +02:00
|
|
|
return 0; /* not enough buffer to return len bytes. */
|
2011-05-13 17:31:00 +02:00
|
|
|
memcpy(buf,r->io.buffer.ptr+r->io.buffer.pos,len);
|
|
|
|
r->io.buffer.pos += len;
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Returns read/write position in buffer. */
|
|
|
|
static off_t rioBufferTell(rio *r) {
|
|
|
|
return r->io.buffer.pos;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Returns 1 or 0 for success/failure. */
|
|
|
|
static size_t rioFileWrite(rio *r, const void *buf, size_t len) {
|
2013-04-03 15:31:40 +02:00
|
|
|
size_t retval;
|
|
|
|
|
|
|
|
retval = fwrite(buf,len,1,r->io.file.fp);
|
|
|
|
r->io.file.buffered += len;
|
|
|
|
|
|
|
|
if (r->io.file.autosync &&
|
|
|
|
r->io.file.buffered >= r->io.file.autosync)
|
|
|
|
{
|
2014-01-22 09:54:43 +01:00
|
|
|
fflush(r->io.file.fp);
|
2013-04-03 15:31:40 +02:00
|
|
|
aof_fsync(fileno(r->io.file.fp));
|
|
|
|
r->io.file.buffered = 0;
|
|
|
|
}
|
|
|
|
return retval;
|
2011-05-13 17:31:00 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Returns 1 or 0 for success/failure. */
|
|
|
|
static size_t rioFileRead(rio *r, void *buf, size_t len) {
|
|
|
|
return fread(buf,len,1,r->io.file.fp);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Returns read/write position in file. */
|
|
|
|
static off_t rioFileTell(rio *r) {
|
|
|
|
return ftello(r->io.file.fp);
|
|
|
|
}
|
|
|
|
|
|
|
|
static const rio rioBufferIO = {
|
|
|
|
rioBufferRead,
|
|
|
|
rioBufferWrite,
|
|
|
|
rioBufferTell,
|
2012-04-09 12:33:09 +02:00
|
|
|
NULL, /* update_checksum */
|
|
|
|
0, /* current checksum */
|
2012-12-12 15:59:22 +02:00
|
|
|
0, /* bytes read or written */
|
|
|
|
0, /* read/write chunk size */
|
2011-05-13 17:31:00 +02:00
|
|
|
{ { NULL, 0 } } /* union for io-specific vars */
|
|
|
|
};
|
|
|
|
|
|
|
|
static const rio rioFileIO = {
|
|
|
|
rioFileRead,
|
|
|
|
rioFileWrite,
|
|
|
|
rioFileTell,
|
2012-04-09 12:33:09 +02:00
|
|
|
NULL, /* update_checksum */
|
|
|
|
0, /* current checksum */
|
2012-12-12 15:59:22 +02:00
|
|
|
0, /* bytes read or written */
|
|
|
|
0, /* read/write chunk size */
|
2011-05-13 17:31:00 +02:00
|
|
|
{ { NULL, 0 } } /* union for io-specific vars */
|
|
|
|
};
|
|
|
|
|
2011-09-22 16:00:40 +02:00
|
|
|
void rioInitWithFile(rio *r, FILE *fp) {
|
|
|
|
*r = rioFileIO;
|
|
|
|
r->io.file.fp = fp;
|
2013-04-03 15:31:40 +02:00
|
|
|
r->io.file.buffered = 0;
|
|
|
|
r->io.file.autosync = 0;
|
2011-05-13 17:31:00 +02:00
|
|
|
}
|
2011-09-22 16:00:40 +02:00
|
|
|
|
|
|
|
void rioInitWithBuffer(rio *r, sds s) {
|
|
|
|
*r = rioBufferIO;
|
|
|
|
r->io.buffer.ptr = s;
|
|
|
|
r->io.buffer.pos = 0;
|
2011-05-13 17:31:00 +02:00
|
|
|
}
|
|
|
|
|
2012-04-09 12:33:09 +02:00
|
|
|
/* This function can be installed both in memory and file streams when checksum
|
|
|
|
* computation is needed. */
|
|
|
|
void rioGenericUpdateChecksum(rio *r, const void *buf, size_t len) {
|
2012-04-09 12:36:44 +02:00
|
|
|
r->cksum = crc64(r->cksum,buf,len);
|
2012-04-09 12:33:09 +02:00
|
|
|
}
|
|
|
|
|
2013-04-03 15:31:40 +02:00
|
|
|
/* Set the file-based rio object to auto-fsync every 'bytes' file written.
|
|
|
|
* By default this is set to zero that means no automatic file sync is
|
|
|
|
* performed.
|
|
|
|
*
|
|
|
|
* This feature is useful in a few contexts since when we rely on OS write
|
|
|
|
* buffers sometimes the OS buffers way too much, resulting in too many
|
|
|
|
* disk I/O concentrated in very little time. When we fsync in an explicit
|
|
|
|
* way instead the I/O pressure is more distributed across time. */
|
|
|
|
void rioSetAutoSync(rio *r, off_t bytes) {
|
|
|
|
redisAssert(r->read == rioFileIO.read);
|
|
|
|
r->io.file.autosync = bytes;
|
|
|
|
}
|
|
|
|
|
2012-04-09 11:11:00 +02:00
|
|
|
/* ------------------------------ Higher level interface ---------------------------
|
|
|
|
* The following higher level functions use lower level rio.c functions to help
|
|
|
|
* generating the Redis protocol for the Append Only File. */
|
|
|
|
|
2011-05-13 17:31:00 +02:00
|
|
|
/* Write multi bulk count in the format: "*<count>\r\n". */
|
|
|
|
size_t rioWriteBulkCount(rio *r, char prefix, int count) {
|
|
|
|
char cbuf[128];
|
|
|
|
int clen;
|
|
|
|
|
|
|
|
cbuf[0] = prefix;
|
|
|
|
clen = 1+ll2string(cbuf+1,sizeof(cbuf)-1,count);
|
|
|
|
cbuf[clen++] = '\r';
|
|
|
|
cbuf[clen++] = '\n';
|
|
|
|
if (rioWrite(r,cbuf,clen) == 0) return 0;
|
|
|
|
return clen;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Write binary-safe string in the format: "$<count>\r\n<payload>\r\n". */
|
|
|
|
size_t rioWriteBulkString(rio *r, const char *buf, size_t len) {
|
|
|
|
size_t nwritten;
|
|
|
|
|
|
|
|
if ((nwritten = rioWriteBulkCount(r,'$',len)) == 0) return 0;
|
|
|
|
if (len > 0 && rioWrite(r,buf,len) == 0) return 0;
|
|
|
|
if (rioWrite(r,"\r\n",2) == 0) return 0;
|
|
|
|
return nwritten+len+2;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Write a long long value in format: "$<count>\r\n<payload>\r\n". */
|
|
|
|
size_t rioWriteBulkLongLong(rio *r, long long l) {
|
|
|
|
char lbuf[32];
|
|
|
|
unsigned int llen;
|
|
|
|
|
|
|
|
llen = ll2string(lbuf,sizeof(lbuf),l);
|
|
|
|
return rioWriteBulkString(r,lbuf,llen);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Write a double value in the format: "$<count>\r\n<payload>\r\n" */
|
|
|
|
size_t rioWriteBulkDouble(rio *r, double d) {
|
|
|
|
char dbuf[128];
|
|
|
|
unsigned int dlen;
|
|
|
|
|
|
|
|
dlen = snprintf(dbuf,sizeof(dbuf),"%.17g",d);
|
|
|
|
return rioWriteBulkString(r,dbuf,dlen);
|
|
|
|
}
|