mirror of
https://github.com/FirebirdSQL/firebird.git
synced 2025-01-27 06:43:04 +01:00
704 lines
15 KiB
C++
704 lines
15 KiB
C++
/*
|
|
* PROGRAM: JRD Backup and Restore Program
|
|
* MODULE: canonical.cpp
|
|
* DESCRIPTION:
|
|
*
|
|
* The contents of this file are subject to the Interbase Public
|
|
* License Version 1.0 (the "License"); you may not use this file
|
|
* except in compliance with the License. You may obtain a copy
|
|
* of the License at http://www.Inprise.com/IPL.html
|
|
*
|
|
* Software distributed under the License is distributed on an
|
|
* "AS IS" basis, WITHOUT WARRANTY OF ANY KIND, either express
|
|
* or implied. See the License for the specific language governing
|
|
* rights and limitations under the License.
|
|
*
|
|
* The Original Code was created by Inprise Corporation
|
|
* and its predecessors. Portions created by Inprise Corporation are
|
|
* Copyright (C) Inprise Corporation.
|
|
*
|
|
* All Rights Reserved.
|
|
* Contributor(s): ______________________________________.
|
|
*
|
|
* 2002.10.27 Sean Leyne - Code Cleanup, removed obsolete "Ultrix/MIPS" port
|
|
*
|
|
* 2002.10.29 Sean Leyne - Removed obsolete "Netware" port
|
|
*
|
|
* 2002.10.30 Sean Leyne - Removed support for obsolete "PC_PLATFORM" define
|
|
*
|
|
*/
|
|
|
|
#include "firebird.h"
|
|
#include <stdio.h>
|
|
#include <string.h>
|
|
#ifdef HP11
|
|
#include <arpa/inet.h>
|
|
#endif
|
|
#include "../burp/burp.h"
|
|
#include "../jrd/align.h"
|
|
#include "../jrd/sdl.h"
|
|
#include "../burp/canon_proto.h"
|
|
#include "../jrd/sdl_proto.h"
|
|
#include "../remote/xdr_proto.h"
|
|
#include "../jrd/gdsassert.h"
|
|
#include "../include/fb_types.h"
|
|
|
|
// TMN: Currently we can't include remote/remote.h because we'd get
|
|
// conflicting blk_t definitions (we are gonna fix this, in due time).
|
|
|
|
|
|
static XDR_INT burp_destroy(XDR*);
|
|
static bool_t burp_getbytes(XDR*, SCHAR *, u_int);
|
|
static bool_t burp_getlong(XDR*, SLONG *);
|
|
static u_int burp_getpostn(XDR*);
|
|
static caddr_t burp_inline(XDR*, u_int);
|
|
static bool_t burp_putbytes(XDR*, const SCHAR*, u_int);
|
|
static bool_t burp_putlong(XDR*, const SLONG*);
|
|
static bool_t burp_setpostn(XDR*, u_int);
|
|
static bool_t expand_buffer(XDR*);
|
|
static bool_t xdr_datum(XDR*, DSC*, UCHAR*);
|
|
static bool_t xdr_quad(XDR*, SLONG*);
|
|
static int xdr_init(XDR*, lstring*, enum xdr_op);
|
|
static bool_t xdr_slice(XDR*, lstring*, /*USHORT,*/ const UCHAR*);
|
|
|
|
static xdr_t::xdr_ops burp_ops =
|
|
{
|
|
burp_getlong,
|
|
burp_putlong,
|
|
burp_getbytes,
|
|
burp_putbytes,
|
|
burp_getpostn,
|
|
burp_setpostn,
|
|
burp_inline,
|
|
burp_destroy
|
|
};
|
|
|
|
const int increment = 1024;
|
|
|
|
|
|
ULONG CAN_encode_decode(burp_rel* relation,
|
|
lstring* buffer,
|
|
UCHAR* data,
|
|
bool_t direction)
|
|
{
|
|
/**************************************
|
|
*
|
|
* C A N _ e n c o d e _ d e c o d e
|
|
*
|
|
**************************************
|
|
*
|
|
* Functional description
|
|
* encode and decode canonical backup.
|
|
*
|
|
**************************************/
|
|
const burp_fld* field;
|
|
SSHORT n;
|
|
|
|
XDR xdr;
|
|
XDR* xdrs = &xdr;
|
|
|
|
xdr_init(xdrs, buffer, direction ? XDR_ENCODE : XDR_DECODE);
|
|
|
|
RCRD_OFFSET offset = 0;
|
|
for (field = relation->rel_fields; field; field = field->fld_next)
|
|
{
|
|
if (field->fld_flags & FLD_computed)
|
|
continue;
|
|
UCHAR* p = data + field->fld_offset;
|
|
const bool array_fld = ((field->fld_flags & FLD_array) != 0);
|
|
const FLD_LENGTH length = array_fld ? 8 : field->fld_length;
|
|
if (field->fld_offset >= offset)
|
|
offset = field->fld_offset + length;
|
|
if (field->fld_type == blr_varying && !array_fld)
|
|
offset += sizeof(SSHORT);
|
|
SSHORT dtype;
|
|
if (field->fld_type == blr_blob || array_fld)
|
|
dtype = dtype_blob;
|
|
else
|
|
dtype = (SSHORT) gds_cvt_blr_dtype[field->fld_type];
|
|
switch (dtype)
|
|
{
|
|
case dtype_text:
|
|
if (!xdr_opaque(xdrs, reinterpret_cast<char*>(p), length))
|
|
{
|
|
return FALSE;
|
|
}
|
|
break;
|
|
|
|
case dtype_varying:
|
|
{
|
|
vary* pVary = reinterpret_cast<vary*>(p);
|
|
if (!xdr_short(xdrs, reinterpret_cast<SSHORT*>(&pVary->vary_length)))
|
|
{
|
|
return FALSE;
|
|
}
|
|
if (!xdr_opaque(xdrs, reinterpret_cast<SCHAR*>(pVary->vary_string),
|
|
MIN(pVary->vary_length, length)))
|
|
{
|
|
return FALSE;
|
|
}
|
|
}
|
|
break;
|
|
|
|
case dtype_cstring:
|
|
if (xdrs->x_op == XDR_ENCODE)
|
|
n = MIN(strlen(reinterpret_cast<const char*>(p)), length);
|
|
if (!xdr_short(xdrs, &n))
|
|
return FALSE;
|
|
if (!xdr_opaque(xdrs, reinterpret_cast<SCHAR*>(p), n))
|
|
return FALSE;
|
|
if (xdrs->x_op == XDR_DECODE)
|
|
p[n] = 0;
|
|
break;
|
|
|
|
case dtype_short:
|
|
if (!xdr_short(xdrs, (SSHORT *) p))
|
|
return FALSE;
|
|
break;
|
|
|
|
case dtype_long:
|
|
case dtype_sql_time:
|
|
case dtype_sql_date:
|
|
if (!xdr_long(xdrs, (SLONG *) p))
|
|
return FALSE;
|
|
break;
|
|
|
|
case dtype_real:
|
|
if (!xdr_float(xdrs, (float *) p))
|
|
return FALSE;
|
|
break;
|
|
|
|
case dtype_double:
|
|
if (!xdr_double(xdrs, (double *) p))
|
|
return FALSE;
|
|
break;
|
|
|
|
case dtype_timestamp:
|
|
if (!xdr_long(xdrs, &((SLONG*) p)[0]))
|
|
return FALSE;
|
|
if (!xdr_long(xdrs, &((SLONG*) p)[1]))
|
|
return FALSE;
|
|
break;
|
|
|
|
case dtype_quad:
|
|
case dtype_blob:
|
|
if (!xdr_quad(xdrs, (SLONG*) p))
|
|
return FALSE;
|
|
break;
|
|
|
|
case dtype_int64:
|
|
if (!xdr_hyper(xdrs, (SINT64*) p))
|
|
return FALSE;
|
|
break;
|
|
|
|
default:
|
|
fb_assert(FALSE);
|
|
return FALSE;
|
|
}
|
|
}
|
|
|
|
// Next, get null flags
|
|
|
|
for (field = relation->rel_fields; field; field = field->fld_next)
|
|
{
|
|
if (field->fld_flags & FLD_computed)
|
|
continue;
|
|
offset = FB_ALIGN(offset, sizeof(SSHORT));
|
|
UCHAR* p = data + offset;
|
|
if (!xdr_short(xdrs, (SSHORT*) p))
|
|
return FALSE;
|
|
offset += sizeof(SSHORT);
|
|
}
|
|
return (xdrs->x_private - xdrs->x_base);
|
|
}
|
|
|
|
|
|
ULONG CAN_slice(lstring* buffer,
|
|
lstring* slice,
|
|
bool_t direction,
|
|
//USHORT sdl_length,
|
|
UCHAR* sdl)
|
|
{
|
|
/**************************************
|
|
*
|
|
* C A N _ s l i c e
|
|
*
|
|
**************************************
|
|
*
|
|
* Functional description
|
|
* encode and decode canonical backup.
|
|
*
|
|
**************************************/
|
|
XDR xdr;
|
|
XDR* xdrs = &xdr;
|
|
|
|
xdr_init(xdrs, buffer, direction ? XDR_ENCODE : XDR_DECODE);
|
|
|
|
xdr_slice(xdrs, slice, /*sdl_length,*/ sdl);
|
|
return (xdrs->x_private - xdrs->x_base);
|
|
}
|
|
|
|
|
|
static XDR_INT burp_destroy(XDR*)
|
|
{
|
|
/**************************************
|
|
*
|
|
* b u r p _ d e s t r o y
|
|
*
|
|
**************************************
|
|
*
|
|
* Functional description
|
|
* Destroy a stream. A no-op.
|
|
*
|
|
**************************************/
|
|
return 0;
|
|
}
|
|
|
|
|
|
static bool_t burp_getbytes(XDR* xdrs, SCHAR* buff, u_int bytecount)
|
|
{
|
|
/**************************************
|
|
*
|
|
* b u r p _ g e t b y t e s
|
|
*
|
|
**************************************
|
|
*
|
|
* Functional description
|
|
* Fetch a bunch of bytes into a memory stream if it fits.
|
|
*
|
|
**************************************/
|
|
|
|
if (bytecount && xdrs->x_handy >= (int) bytecount)
|
|
{
|
|
xdrs->x_handy -= bytecount;
|
|
do {
|
|
*buff++ = *xdrs->x_private++;
|
|
} while (--bytecount);
|
|
return TRUE;
|
|
}
|
|
|
|
while (bytecount)
|
|
{
|
|
if (!xdrs->x_handy && !expand_buffer(xdrs))
|
|
return FALSE;
|
|
*buff++ = *xdrs->x_private++;
|
|
--xdrs->x_handy;
|
|
--bytecount;
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
|
|
static bool_t burp_getlong(XDR* xdrs, SLONG* lp)
|
|
{
|
|
/**************************************
|
|
*
|
|
* b u r p _ g e t l o n g
|
|
*
|
|
**************************************
|
|
*
|
|
* Functional description
|
|
* Fetch a longword into a memory stream if it fits.
|
|
*
|
|
**************************************/
|
|
|
|
SLONG l;
|
|
|
|
if (!(*xdrs->x_ops->x_getbytes) (xdrs, reinterpret_cast<char*>(&l), 4))
|
|
return FALSE;
|
|
|
|
*lp = ntohl(l);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
|
|
static u_int burp_getpostn(XDR* xdrs)
|
|
{
|
|
/**************************************
|
|
*
|
|
* b u r p _ g e t p o s t n
|
|
*
|
|
**************************************
|
|
*
|
|
* Functional description
|
|
* Get the current position (which is also current length) from stream.
|
|
*
|
|
**************************************/
|
|
|
|
return xdrs->x_private - xdrs->x_base;
|
|
}
|
|
|
|
|
|
static caddr_t burp_inline(XDR* xdrs, u_int bytecount)
|
|
{
|
|
/**************************************
|
|
*
|
|
* b u r p _ i n l i n e
|
|
*
|
|
**************************************
|
|
*
|
|
* Functional description
|
|
* Return a pointer to somewhere in the buffer.
|
|
*
|
|
**************************************/
|
|
|
|
if (bytecount > (u_int) xdrs->x_handy)
|
|
return FALSE;
|
|
|
|
return xdrs->x_base + bytecount;
|
|
}
|
|
|
|
|
|
static bool_t burp_putbytes(XDR* xdrs, const SCHAR* buff, u_int bytecount)
|
|
{
|
|
/**************************************
|
|
*
|
|
* b u r p _ p u t b y t e s
|
|
*
|
|
**************************************
|
|
*
|
|
* Functional description
|
|
* Fetch a bunch of bytes into a memory stream if it fits.
|
|
*
|
|
**************************************/
|
|
|
|
if (bytecount && xdrs->x_handy >= (int) bytecount)
|
|
{
|
|
xdrs->x_handy -= bytecount;
|
|
do {
|
|
*xdrs->x_private++ = *buff++;
|
|
} while (--bytecount);
|
|
return TRUE;
|
|
}
|
|
|
|
while (bytecount)
|
|
{
|
|
if (xdrs->x_handy <= 0 && !expand_buffer(xdrs))
|
|
{
|
|
return FALSE;
|
|
}
|
|
--xdrs->x_handy;
|
|
*xdrs->x_private++ = *buff++;
|
|
--bytecount;
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
|
|
static bool_t burp_putlong(XDR* xdrs, const SLONG* lp)
|
|
{
|
|
/**************************************
|
|
*
|
|
* b u r p _ p u t l o n g
|
|
*
|
|
**************************************
|
|
*
|
|
* Functional description
|
|
* Fetch a longword into a memory stream if it fits.
|
|
*
|
|
**************************************/
|
|
SLONG l = htonl(*lp);
|
|
return (*xdrs->x_ops->x_putbytes) (xdrs, reinterpret_cast<char*>(&l), 4);
|
|
}
|
|
|
|
|
|
static bool_t burp_setpostn(XDR* xdrs, u_int bytecount)
|
|
{
|
|
/**************************************
|
|
*
|
|
* b u r p _ s e t p o s t n
|
|
*
|
|
**************************************
|
|
*
|
|
* Functional description
|
|
* Set the current position (which is also current length) from stream.
|
|
*
|
|
**************************************/
|
|
|
|
if (bytecount > (u_int) xdrs->x_handy)
|
|
return FALSE;
|
|
|
|
xdrs->x_private = xdrs->x_base + bytecount;
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
|
|
static bool_t expand_buffer(XDR* xdrs)
|
|
{
|
|
/**************************************
|
|
*
|
|
* e x p a n d _ b u f f e r
|
|
*
|
|
**************************************
|
|
*
|
|
* Functional description
|
|
* Allocate a new, larger buffer, copy
|
|
* everything we've got, and release the
|
|
* old one.
|
|
*
|
|
**************************************/
|
|
lstring* buffer = (lstring*) xdrs->x_public;
|
|
const SSHORT length = (xdrs->x_private - xdrs->x_base) + xdrs->x_handy + increment;
|
|
buffer->lstr_allocated = buffer->lstr_length = length;
|
|
|
|
caddr_t new_buf = (caddr_t) BURP_alloc(length);
|
|
|
|
caddr_t p = new_buf;
|
|
for (caddr_t q = xdrs->x_base; q < xdrs->x_private; *p++ = *q++)
|
|
;
|
|
|
|
BURP_free(xdrs->x_base);
|
|
|
|
xdrs->x_base = new_buf;
|
|
xdrs->x_private = p;
|
|
xdrs->x_handy += increment;
|
|
|
|
buffer->lstr_address = (UCHAR *) new_buf;
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
|
|
|
|
static bool_t xdr_datum(XDR* xdrs, DSC* desc, UCHAR* buffer)
|
|
{
|
|
/**************************************
|
|
*
|
|
* x d r _ d a t u m
|
|
*
|
|
**************************************
|
|
*
|
|
* Functional description
|
|
* Handle a data item by relative descriptor and buffer.
|
|
*
|
|
**************************************/
|
|
SSHORT n;
|
|
|
|
UCHAR* p = buffer + (IPTR) desc->dsc_address;
|
|
|
|
switch (desc->dsc_dtype)
|
|
{
|
|
case dtype_text:
|
|
if (!xdr_opaque(xdrs, reinterpret_cast<SCHAR*>(p), desc->dsc_length))
|
|
{
|
|
return FALSE;
|
|
}
|
|
break;
|
|
|
|
case dtype_varying:
|
|
{
|
|
vary* pVary = reinterpret_cast<vary*>(p);
|
|
if (!xdr_short(xdrs, reinterpret_cast<short*>(&pVary->vary_length)))
|
|
{
|
|
return FALSE;
|
|
}
|
|
if (!xdr_opaque(xdrs, reinterpret_cast<SCHAR*>(pVary->vary_string),
|
|
MIN(desc->dsc_length - 2, pVary->vary_length)))
|
|
{
|
|
return FALSE;
|
|
}
|
|
}
|
|
break;
|
|
|
|
case dtype_cstring:
|
|
if (xdrs->x_op == XDR_ENCODE) {
|
|
n = MIN(strlen(reinterpret_cast<const char*>(p)), (size_t) (desc->dsc_length - 1));
|
|
}
|
|
if (!xdr_short(xdrs, &n))
|
|
return FALSE;
|
|
if (!xdr_opaque(xdrs, reinterpret_cast<char*>(p), n))
|
|
return FALSE;
|
|
if (xdrs->x_op == XDR_DECODE)
|
|
p[n] = 0;
|
|
break;
|
|
|
|
case dtype_short:
|
|
if (!xdr_short(xdrs, (SSHORT*) p))
|
|
return FALSE;
|
|
break;
|
|
|
|
case dtype_sql_date:
|
|
case dtype_sql_time:
|
|
case dtype_long:
|
|
if (!xdr_long(xdrs, (SLONG*) p))
|
|
return FALSE;
|
|
break;
|
|
|
|
case dtype_real:
|
|
if (!xdr_float(xdrs, (float *) p))
|
|
return FALSE;
|
|
break;
|
|
|
|
case dtype_double:
|
|
if (!xdr_double(xdrs, (double *) p))
|
|
return FALSE;
|
|
break;
|
|
|
|
case dtype_timestamp:
|
|
if (!xdr_long(xdrs, &((SLONG*) p)[0]))
|
|
return FALSE;
|
|
if (!xdr_long(xdrs, &((SLONG*) p)[1]))
|
|
return FALSE;
|
|
break;
|
|
|
|
case dtype_quad:
|
|
case dtype_blob:
|
|
if (!xdr_quad(xdrs, (SLONG*) p))
|
|
return FALSE;
|
|
break;
|
|
|
|
case dtype_int64:
|
|
if (!xdr_hyper(xdrs, (SINT64 *) p))
|
|
return FALSE;
|
|
break;
|
|
|
|
default:
|
|
fb_assert(FALSE);
|
|
return FALSE;
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static bool_t xdr_quad(XDR* xdrs, SLONG* ip)
|
|
{
|
|
/**************************************
|
|
*
|
|
* x d r _ q u a d
|
|
*
|
|
**************************************
|
|
*
|
|
* Functional description
|
|
* Map from external to internal representation (or vice versa).
|
|
*
|
|
**************************************/
|
|
|
|
switch (xdrs->x_op)
|
|
{
|
|
case XDR_ENCODE:
|
|
if ((*xdrs->x_ops->x_putlong) (xdrs, &ip[0]) && (*xdrs->x_ops->x_putlong) (xdrs, &ip[1]))
|
|
{
|
|
return TRUE;
|
|
}
|
|
return FALSE;
|
|
|
|
case XDR_DECODE:
|
|
if (!(*xdrs->x_ops->x_getlong) (xdrs, &ip[0]))
|
|
return FALSE;
|
|
return (*xdrs->x_ops->x_getlong) (xdrs, &ip[1]);
|
|
|
|
case XDR_FREE:
|
|
return TRUE;
|
|
|
|
default:
|
|
fb_assert(FALSE);
|
|
return FALSE;
|
|
}
|
|
}
|
|
|
|
|
|
|
|
static int xdr_init(XDR* xdrs, lstring* buffer, enum xdr_op x_op)
|
|
{
|
|
/**************************************
|
|
*
|
|
* x d r _ i n i t
|
|
*
|
|
**************************************
|
|
*
|
|
* Functional description
|
|
* Initialize an XDR stream for Apollo mailboxes.
|
|
*
|
|
**************************************/
|
|
|
|
xdrs->x_public = (caddr_t) buffer;
|
|
xdrs->x_base = xdrs->x_private = (caddr_t) buffer->lstr_address;
|
|
xdrs->x_handy = buffer->lstr_length;
|
|
xdrs->x_ops = &burp_ops;
|
|
xdrs->x_op = x_op;
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
|
|
|
|
static bool_t xdr_slice(XDR* xdrs,
|
|
lstring* slice,
|
|
//USHORT sdl_length,
|
|
const UCHAR* sdl)
|
|
{
|
|
/**************************************
|
|
*
|
|
* x d r _ s l i c e
|
|
*
|
|
**************************************
|
|
*
|
|
* Functional description
|
|
* Move a slice of an array under
|
|
*
|
|
**************************************/
|
|
if (!xdr_long(xdrs, reinterpret_cast<SLONG*>(&slice->lstr_length)))
|
|
return FALSE;
|
|
|
|
// Handle operation specific stuff, particularly memory allocation/deallocation
|
|
|
|
switch (xdrs->x_op)
|
|
{
|
|
case XDR_ENCODE:
|
|
break;
|
|
|
|
case XDR_DECODE:
|
|
if (!slice->lstr_length)
|
|
return TRUE;
|
|
if (slice->lstr_length > slice->lstr_allocated && slice->lstr_allocated)
|
|
{
|
|
BURP_free(slice->lstr_address);
|
|
slice->lstr_address = NULL;
|
|
}
|
|
if (!slice->lstr_address)
|
|
{
|
|
slice->lstr_address = BURP_alloc((SLONG) slice->lstr_length);
|
|
if (!slice->lstr_address) {
|
|
return FALSE;
|
|
}
|
|
slice->lstr_allocated = slice->lstr_length;
|
|
}
|
|
break;
|
|
|
|
case XDR_FREE:
|
|
if (slice->lstr_allocated)
|
|
BURP_free(slice->lstr_address);
|
|
slice->lstr_address = NULL;
|
|
slice->lstr_allocated = 0;
|
|
return TRUE;
|
|
|
|
default:
|
|
fb_assert(FALSE);
|
|
return FALSE;
|
|
}
|
|
|
|
// Get descriptor of array element
|
|
|
|
ISC_STATUS_ARRAY status_vector;
|
|
sdl_info info;
|
|
if (SDL_info(status_vector, sdl, &info, 0))
|
|
return FALSE;
|
|
|
|
dsc* desc = &info.sdl_info_element;
|
|
const ULONG n = slice->lstr_length / desc->dsc_length;
|
|
UCHAR* p = slice->lstr_address;
|
|
|
|
for (UCHAR* const end = p + n * desc->dsc_length; p < end; p += desc->dsc_length) {
|
|
if (!xdr_datum(xdrs, desc, p)) {
|
|
return FALSE;
|
|
}
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|