2013-04-17 00:00:36 -04:00
/* -*- mode: C; c-basic-offset: 4; indent-tabs-mode: nil -*- */
// vim: expandtab:ts=8:sw=4:softtabstop=4:
2007-08-21 23:32:17 +00:00
# ifndef WBUF_H
# define WBUF_H
2013-04-16 23:57:48 -04:00
# ident "$Id$"
2013-04-17 00:00:14 -04:00
# ident "Copyright (c) 2007-2011 Tokutek Inc. All rights reserved."
2013-04-16 23:57:48 -04:00
# ident "The technology is licensed by the Massachusetts Institute of Technology, Rutgers State University of New Jersey, and the Research Foundation of State University of New York at Stony Brook under United States of America Serial No. 11 / 760379 and to the patents and / or patent applications resulting from it."
2007-11-29 14:18:54 +00:00
2008-07-27 22:16:49 +00:00
# include "x1764.h"
2007-09-28 17:11:22 +00:00
# include "memory.h"
2008-02-08 03:17:38 +00:00
# include "toku_assert.h"
2013-04-17 00:00:17 -04:00
# include "db.h"
2008-02-08 03:17:38 +00:00
# include <errno.h>
# include <string.h>
2007-08-21 23:32:17 +00:00
2013-04-16 23:59:09 -04:00
# if defined(__cplusplus) || defined(__cilkplusplus)
extern " C " {
# endif
2007-11-14 17:58:38 +00:00
# define CRC_INCR
2007-08-21 23:32:17 +00:00
/* When serializing a value, write it into a buffer. */
2013-04-16 23:57:41 -04:00
/* This code requires that the buffer be big enough to hold whatever you put into it. */
2007-08-21 23:32:17 +00:00
/* This abstraction doesn't do a good job of hiding its internals.
* Why ? The performance of this code is important , and we want to inline stuff */
2013-04-16 23:57:41 -04:00
//Why is size here an int instead of DISKOFF like in the initializer?
2007-08-21 23:32:17 +00:00
struct wbuf {
unsigned char * buf ;
unsigned int size ;
unsigned int ndone ;
2013-04-16 23:59:05 -04:00
struct x1764 checksum ; // The checksum state
2007-08-21 23:32:17 +00:00
} ;
2013-04-16 23:58:04 -04:00
static inline void wbuf_nocrc_init ( struct wbuf * w , void * buf , DISKOFF size ) {
2013-04-16 23:59:09 -04:00
w - > buf = ( unsigned char * ) buf ;
w - > size = size ;
w - > ndone = 0 ;
2013-04-16 23:58:04 -04:00
}
static inline void wbuf_init ( struct wbuf * w , void * buf , DISKOFF size ) {
wbuf_nocrc_init ( w , buf , size ) ;
2008-07-27 22:16:49 +00:00
x1764_init ( & w - > checksum ) ;
2007-08-21 23:32:17 +00:00
}
2013-04-16 23:59:05 -04:00
static inline size_t wbuf_get_woffset ( struct wbuf * w ) {
return w - > ndone ;
}
2007-08-21 23:32:17 +00:00
/* Write a character. */
2013-04-16 23:58:04 -04:00
static inline void wbuf_nocrc_char ( struct wbuf * w , unsigned char ch ) {
2007-08-21 23:32:17 +00:00
assert ( w - > ndone < w - > size ) ;
w - > buf [ w - > ndone + + ] = ch ;
2013-04-16 23:58:04 -04:00
}
2013-04-16 23:58:04 -04:00
/* Write a character. */
static inline void wbuf_nocrc_u_int8_t ( struct wbuf * w , u_int8_t ch ) {
assert ( w - > ndone < w - > size ) ;
w - > buf [ w - > ndone + + ] = ch ;
}
2013-04-16 23:58:04 -04:00
static inline void wbuf_char ( struct wbuf * w , unsigned char ch ) {
wbuf_nocrc_char ( w , ch ) ;
2008-07-27 22:16:49 +00:00
x1764_add ( & w - > checksum , & w - > buf [ w - > ndone - 1 ] , 1 ) ;
2007-08-21 23:32:17 +00:00
}
2013-04-16 23:57:46 -04:00
//Write an int that MUST be in network order regardless of disk order
static void wbuf_network_int ( struct wbuf * w , int32_t i ) __attribute__ ( ( __unused__ ) ) ;
static void wbuf_network_int ( struct wbuf * w , int32_t i ) {
assert ( w - > ndone + 4 < = w - > size ) ;
* ( u_int32_t * ) ( & w - > buf [ w - > ndone ] ) = toku_htonl ( i ) ;
x1764_add ( & w - > checksum , & w - > buf [ w - > ndone ] , 4 ) ;
w - > ndone + = 4 ;
}
2013-04-16 23:58:04 -04:00
static inline void wbuf_nocrc_int ( struct wbuf * w , int32_t i ) {
2007-08-21 23:32:17 +00:00
#if 0
2013-04-16 23:58:04 -04:00
wbuf_nocrc_char ( w , i > > 24 ) ;
wbuf_nocrc_char ( w , i > > 16 ) ;
wbuf_nocrc_char ( w , i > > 8 ) ;
wbuf_nocrc_char ( w , i > > 0 ) ;
2007-08-21 23:32:17 +00:00
# else
assert ( w - > ndone + 4 < = w - > size ) ;
2008-05-29 03:12:59 +00:00
#if 0
2007-08-21 23:32:17 +00:00
w - > buf [ w - > ndone + 0 ] = i > > 24 ;
w - > buf [ w - > ndone + 1 ] = i > > 16 ;
w - > buf [ w - > ndone + 2 ] = i > > 8 ;
w - > buf [ w - > ndone + 3 ] = i > > 0 ;
2008-05-29 03:12:59 +00:00
# else
2013-04-16 23:57:46 -04:00
* ( u_int32_t * ) ( & w - > buf [ w - > ndone ] ) = toku_htod32 ( i ) ;
2008-05-29 03:12:59 +00:00
# endif
2007-08-21 23:32:17 +00:00
w - > ndone + = 4 ;
# endif
}
2013-04-16 23:57:46 -04:00
2013-04-16 23:58:04 -04:00
static inline void wbuf_int ( struct wbuf * w , int32_t i ) {
wbuf_nocrc_int ( w , i ) ;
x1764_add ( & w - > checksum , & w - > buf [ w - > ndone - 4 ] , 4 ) ;
}
static inline void wbuf_nocrc_uint ( struct wbuf * w , u_int32_t i ) {
wbuf_nocrc_int ( w , ( int32_t ) i ) ;
}
static inline void wbuf_uint ( struct wbuf * w , u_int32_t i ) {
2008-04-02 23:40:36 +00:00
wbuf_int ( w , ( int32_t ) i ) ;
}
2007-08-21 23:32:17 +00:00
2013-04-16 23:58:04 -04:00
static inline void wbuf_nocrc_literal_bytes ( struct wbuf * w , bytevec bytes_bv , u_int32_t nbytes ) {
2013-04-16 23:59:09 -04:00
const unsigned char * bytes = ( const unsigned char * ) bytes_bv ;
2007-08-21 23:32:17 +00:00
#if 0
2013-04-16 23:58:04 -04:00
{ int i ; for ( i = 0 ; i < nbytes ; i + + ) wbuf_nocrc_char ( w , bytes [ i ] ) ; }
2007-08-21 23:32:17 +00:00
# else
assert ( w - > ndone + nbytes < = w - > size ) ;
2008-04-02 23:40:36 +00:00
memcpy ( w - > buf + w - > ndone , bytes , ( size_t ) nbytes ) ;
2007-08-21 23:32:17 +00:00
w - > ndone + = nbytes ;
# endif
2013-04-16 23:58:04 -04:00
}
static inline void wbuf_literal_bytes ( struct wbuf * w , bytevec bytes_bv , u_int32_t nbytes ) {
wbuf_nocrc_literal_bytes ( w , bytes_bv , nbytes ) ;
x1764_add ( & w - > checksum , & w - > buf [ w - > ndone - nbytes ] , nbytes ) ;
}
2013-04-16 23:57:41 -04:00
2013-04-16 23:58:04 -04:00
static void wbuf_nocrc_bytes ( struct wbuf * w , bytevec bytes_bv , u_int32_t nbytes ) {
wbuf_nocrc_uint ( w , nbytes ) ;
wbuf_nocrc_literal_bytes ( w , bytes_bv , nbytes ) ;
2007-11-14 17:58:38 +00:00
}
2008-04-02 23:40:36 +00:00
static void wbuf_bytes ( struct wbuf * w , bytevec bytes_bv , u_int32_t nbytes ) {
wbuf_uint ( w , nbytes ) ;
2007-11-14 17:58:38 +00:00
wbuf_literal_bytes ( w , bytes_bv , nbytes ) ;
2007-08-21 23:32:17 +00:00
}
2013-04-16 23:58:04 -04:00
static void wbuf_nocrc_ulonglong ( struct wbuf * w , u_int64_t ull ) {
wbuf_nocrc_uint ( w , ( u_int32_t ) ( ull > > 32 ) ) ;
wbuf_nocrc_uint ( w , ( u_int32_t ) ( ull & 0xFFFFFFFF ) ) ;
}
2008-04-02 23:40:36 +00:00
static void wbuf_ulonglong ( struct wbuf * w , u_int64_t ull ) {
wbuf_uint ( w , ( u_int32_t ) ( ull > > 32 ) ) ;
wbuf_uint ( w , ( u_int32_t ) ( ull & 0xFFFFFFFF ) ) ;
2007-09-28 17:11:22 +00:00
}
2013-04-16 23:58:04 -04:00
static inline void wbuf_nocrc_u_int64_t ( struct wbuf * w , u_int64_t ull ) {
wbuf_nocrc_ulonglong ( w , ull ) ;
}
2013-04-16 23:57:55 -04:00
static inline void wbuf_u_int64_t ( struct wbuf * w , u_int64_t ull ) {
wbuf_ulonglong ( w , ull ) ;
}
2013-04-16 23:59:40 -04:00
static inline void wbuf_nocrc_BOOL ( struct wbuf * w , BOOL b ) {
wbuf_nocrc_u_int8_t ( w , ( u_int8_t ) ( b ? 1 : 0 ) ) ;
}
2013-04-16 23:58:04 -04:00
static inline void wbuf_nocrc_BYTESTRING ( struct wbuf * w , BYTESTRING v ) {
wbuf_nocrc_bytes ( w , v . data , v . len ) ;
}
2007-11-22 18:45:22 +00:00
static inline void wbuf_BYTESTRING ( struct wbuf * w , BYTESTRING v ) {
2007-11-22 20:30:38 +00:00
wbuf_bytes ( w , v . data , v . len ) ;
2007-11-22 18:45:22 +00:00
}
2007-11-23 17:16:26 +00:00
static inline void wbuf_u_int8_t ( struct wbuf * w , u_int8_t v ) {
wbuf_char ( w , v ) ;
}
2013-04-16 23:58:04 -04:00
static inline void wbuf_nocrc_u_int32_t ( struct wbuf * w , u_int32_t v ) {
wbuf_nocrc_uint ( w , v ) ;
}
2007-11-22 18:45:22 +00:00
static inline void wbuf_u_int32_t ( struct wbuf * w , u_int32_t v ) {
2008-04-02 23:40:36 +00:00
wbuf_uint ( w , v ) ;
2007-11-22 18:45:22 +00:00
}
static inline void wbuf_DISKOFF ( struct wbuf * w , DISKOFF off ) {
2008-04-02 23:40:36 +00:00
wbuf_ulonglong ( w , ( u_int64_t ) off ) ;
2007-08-21 23:32:17 +00:00
}
2013-04-16 23:59:05 -04:00
2013-04-16 23:57:18 -04:00
static inline void wbuf_BLOCKNUM ( struct wbuf * w , BLOCKNUM b ) {
wbuf_ulonglong ( w , b . b ) ;
}
2013-04-16 23:59:05 -04:00
static inline void wbuf_nocrc_BLOCKNUM ( struct wbuf * w , BLOCKNUM b ) {
wbuf_nocrc_ulonglong ( w , b . b ) ;
}
2013-04-16 23:57:18 -04:00
2013-04-16 23:58:04 -04:00
static inline void wbuf_nocrc_TXNID ( struct wbuf * w , TXNID tid ) {
wbuf_nocrc_ulonglong ( w , tid ) ;
}
2007-09-28 17:11:22 +00:00
2007-11-22 18:45:22 +00:00
static inline void wbuf_TXNID ( struct wbuf * w , TXNID tid ) {
2007-09-28 17:11:22 +00:00
wbuf_ulonglong ( w , tid ) ;
}
2013-04-17 00:00:17 -04:00
static inline void wbuf_nocrc_XIDP ( struct wbuf * w , XIDP xid ) {
wbuf_nocrc_u_int32_t ( w , xid - > formatID ) ;
wbuf_nocrc_u_int8_t ( w , xid - > gtrid_length ) ;
wbuf_nocrc_u_int8_t ( w , xid - > bqual_length ) ;
wbuf_nocrc_literal_bytes ( w , xid - > data , xid - > gtrid_length + xid - > bqual_length ) ;
2013-04-17 00:00:14 -04:00
}
2013-04-16 23:58:04 -04:00
static inline void wbuf_nocrc_LSN ( struct wbuf * w , LSN lsn ) {
wbuf_nocrc_ulonglong ( w , lsn . lsn ) ;
}
2007-11-22 18:45:22 +00:00
static inline void wbuf_LSN ( struct wbuf * w , LSN lsn ) {
2007-11-14 17:58:38 +00:00
wbuf_ulonglong ( w , lsn . lsn ) ;
}
2013-04-16 23:59:40 -04:00
static inline void wbuf_MSN ( struct wbuf * w , MSN msn ) {
wbuf_ulonglong ( w , msn . msn ) ;
}
2013-04-16 23:58:04 -04:00
static inline void wbuf_nocrc_FILENUM ( struct wbuf * w , FILENUM fileid ) {
wbuf_nocrc_uint ( w , fileid . fileid ) ;
}
2007-11-22 18:45:22 +00:00
static inline void wbuf_FILENUM ( struct wbuf * w , FILENUM fileid ) {
2008-04-02 23:40:36 +00:00
wbuf_uint ( w , fileid . fileid ) ;
2007-10-15 02:06:19 +00:00
}
2013-04-16 23:59:33 -04:00
// 2954
2013-04-16 23:58:58 -04:00
static inline void wbuf_nocrc_FILENUMS ( struct wbuf * w , FILENUMS v ) {
wbuf_nocrc_uint ( w , v . num ) ;
uint32_t i ;
for ( i = 0 ; i < v . num ; i + + ) {
wbuf_nocrc_FILENUM ( w , v . filenums [ i ] ) ;
}
}
2013-04-16 23:59:33 -04:00
// 2954
2013-04-16 23:58:58 -04:00
static inline void wbuf_FILENUMS ( struct wbuf * w , FILENUMS v ) {
wbuf_uint ( w , v . num ) ;
uint32_t i ;
for ( i = 0 ; i < v . num ; i + + ) {
wbuf_FILENUM ( w , v . filenums [ i ] ) ;
}
}
2013-04-16 23:59:09 -04:00
# if defined(__cplusplus) || defined(__cilkplusplus)
} ;
# endif
2013-04-16 23:58:58 -04:00
2007-08-21 23:32:17 +00:00
# endif