mirror of
https://github.com/FirebirdSQL/firebird.git
synced 2025-01-22 23:23:02 +01:00
2940 lines
72 KiB
C++
2940 lines
72 KiB
C++
/*
|
|
* PROGRAM: JRD Backup and Restore Program
|
|
* MODULE: burp.cpp
|
|
* DESCRIPTION: Command line interpreter for backup/restore
|
|
*
|
|
* 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): ______________________________________.
|
|
* 2001.07.06 Sean Leyne - Code Cleanup, removed "#ifdef READONLY_DATABASE"
|
|
* conditionals, as the engine now fully supports
|
|
* readonly databases.
|
|
*
|
|
* 2002.02.15 Sean Leyne - Code Cleanup, removed obsolete "EPSON" defines
|
|
*
|
|
* 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 "memory_routines.h"
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
#include <signal.h>
|
|
#include "ibase.h"
|
|
#include <stdarg.h>
|
|
#include "../jrd/ibsetjmp.h"
|
|
#include "../common/msg_encode.h"
|
|
#include "../jrd/ods.h" // to get MAX_PAGE_SIZE
|
|
#include "../jrd/constants.h"
|
|
#include "../burp/burp.h"
|
|
#include "../burp/std_desc.h"
|
|
#include "../burp/split/spit.h"
|
|
#include "../jrd/license.h"
|
|
|
|
#include "../common/classes/timestamp.h"
|
|
#include "../burp/burp_proto.h"
|
|
#include "../burp/backu_proto.h"
|
|
#include "../burp/mvol_proto.h"
|
|
#include "../burp/resto_proto.h"
|
|
#include "../yvalve/gds_proto.h"
|
|
#include "../common/gdsassert.h"
|
|
#include "../common/isc_f_proto.h"
|
|
#include "../common/classes/ClumpletWriter.h"
|
|
#include "../common/classes/Switches.h"
|
|
#include "../common/IntlUtil.h"
|
|
#include "../common/os/os_utils.h"
|
|
#include "../burp/burpswi.h"
|
|
#include "../common/db_alias.h"
|
|
#include "../burp/BurpTasks.h"
|
|
|
|
#ifdef HAVE_CTYPE_H
|
|
#include <ctype.h>
|
|
#endif
|
|
#include "../common/utils_proto.h"
|
|
#include "../common/status.h"
|
|
|
|
#ifdef HAVE_UNISTD_H
|
|
#include <unistd.h>
|
|
#endif
|
|
|
|
#include <fcntl.h>
|
|
|
|
#ifdef HAVE_IO_H
|
|
#include <io.h>
|
|
#endif
|
|
|
|
#ifndef O_CREAT
|
|
#include <sys/types.h>
|
|
#include <sys/file.h>
|
|
#endif
|
|
|
|
using namespace Firebird;
|
|
using MsgFormat::SafeArg;
|
|
using namespace Burp;
|
|
|
|
const char* fopen_write_type = "w";
|
|
const char* fopen_read_type = "r";
|
|
|
|
const int open_mask = 0666;
|
|
const char switch_char = '-';
|
|
|
|
|
|
const char* const output_suppress = "SUPPRESS";
|
|
const int MIN_VERBOSE_INTERVAL = 100;
|
|
|
|
enum gbak_action
|
|
{
|
|
QUIT = 0,
|
|
BACKUP = 1 ,
|
|
RESTORE = 2
|
|
//FDESC = 3 // CVC: Unused
|
|
};
|
|
|
|
static void close_out_transaction(gbak_action, Firebird::ITransaction**);
|
|
//static void enable_signals();
|
|
//static void excp_handler();
|
|
static SLONG get_number(const SCHAR*);
|
|
static ULONG get_size(const SCHAR*, burp_fil*);
|
|
static gbak_action open_files(const TEXT *, const TEXT**, USHORT,
|
|
const Firebird::ClumpletWriter&);
|
|
static int svc_api_gbak(Firebird::UtilSvc*, const Switches& switches);
|
|
static void burp_output(bool err, const SCHAR*, ...) ATTRIBUTE_FORMAT(2,3);
|
|
static void burp_usage(const Switches& switches);
|
|
static Switches::in_sw_tab_t* findSwitchOrThrow(Firebird::UtilSvc*, Switches& switches, Firebird::string& sw);
|
|
static void processFetchPass(const SCHAR*& password, int& itr, const int argc, Firebird::UtilSvc::ArgvType& argv);
|
|
|
|
|
|
// fil.fil_length is FB_UINT64
|
|
const ULONG KBYTE = 1024;
|
|
const ULONG MBYTE = KBYTE * KBYTE;
|
|
const ULONG GBYTE = MBYTE * KBYTE;
|
|
|
|
// Must be consistent with enum BurpGlobals::StatCounter
|
|
struct StatFormat
|
|
{
|
|
const char* header;
|
|
const char* format;
|
|
char width;
|
|
};
|
|
static const char* STAT_CHARS = "TDRW";
|
|
static const StatFormat STAT_FORMATS[] =
|
|
{
|
|
{"time", "%4lu.%03u ", 9},
|
|
{"delta", "%2lu.%03u ", 7},
|
|
{"reads", "%6" UQUADFORMAT" ", 7},
|
|
{"writes", "%6" UQUADFORMAT" ", 7}
|
|
};
|
|
|
|
int BURP_main(Firebird::UtilSvc* uSvc)
|
|
{
|
|
/**************************************
|
|
*
|
|
* B U R P _ m a i n
|
|
*
|
|
**************************************
|
|
*
|
|
* Functional description
|
|
* Entrypoint for GBAK via services manager.
|
|
*
|
|
**************************************/
|
|
int exit_code = FINI_OK;
|
|
|
|
try {
|
|
exit_code = gbak(uSvc);
|
|
}
|
|
catch (const Firebird::Exception& e)
|
|
{
|
|
Firebird::StaticStatusVector status;
|
|
e.stuffException(status);
|
|
UtilSvc::StatusAccessor sa = uSvc->getStatusAccessor();
|
|
sa.init();
|
|
sa.setServiceStatus(status.begin());
|
|
exit_code = FB_FAILURE;
|
|
}
|
|
|
|
return exit_code;
|
|
}
|
|
|
|
|
|
static void binOut(const void* data, unsigned len)
|
|
{
|
|
#ifdef WIN_NT
|
|
static int bin = -1;
|
|
if (bin == -1)
|
|
{
|
|
bin = fileno(stdout);
|
|
_setmode(bin, _O_BINARY);
|
|
}
|
|
#else
|
|
const int bin = 1;
|
|
#endif
|
|
|
|
FB_UNUSED(write(bin, data, len));
|
|
}
|
|
|
|
|
|
static unsigned int binIn(void* data, int len)
|
|
{
|
|
#ifdef WIN_NT
|
|
static int bin = -1;
|
|
if (bin == -1)
|
|
{
|
|
bin = fileno(stdin);
|
|
_setmode(bin, _O_BINARY);
|
|
}
|
|
#else
|
|
const int bin = 0;
|
|
#endif
|
|
|
|
int n = read(bin, data, len);
|
|
if (n < 0)
|
|
Firebird::system_call_failed::raise("read(stdin)");
|
|
|
|
return n;
|
|
}
|
|
|
|
|
|
static int svc_api_gbak(Firebird::UtilSvc* uSvc, const Switches& switches)
|
|
{
|
|
/**********************************************
|
|
*
|
|
* s v c _ a p i _ g b a k
|
|
*
|
|
**********************************************
|
|
*
|
|
* Functional description
|
|
* Run gbak using services APIs
|
|
*
|
|
**********************************************/
|
|
Firebird::string usr, pswd, service;
|
|
const SCHAR* pswd2 = NULL;
|
|
bool flag_restore = false;
|
|
bool flag_verbose = false;
|
|
#ifdef TRUSTED_AUTH
|
|
bool flag_trusted = false;
|
|
#endif
|
|
bool flag_verbint = false;
|
|
SLONG verbint_val = 0;
|
|
|
|
Firebird::UtilSvc::ArgvType& argv = uSvc->argv;
|
|
const int argc = uSvc->argv.getCount();
|
|
Firebird::string files[2];
|
|
unsigned fileIndex = 0;
|
|
|
|
for (int itr = 1; itr < argc; ++itr)
|
|
{
|
|
const Switches::in_sw_tab_t* inSw = switches.findSwitch(argv[itr]);
|
|
if (! inSw)
|
|
{
|
|
if (argv[itr][0] && fileIndex < 2)
|
|
{
|
|
files[fileIndex++] = argv[itr];
|
|
}
|
|
continue;
|
|
}
|
|
|
|
switch (inSw->in_sw)
|
|
{
|
|
case IN_SW_BURP_C: // create database
|
|
case IN_SW_BURP_R: // replace database
|
|
case IN_SW_BURP_RECREATE: // recreate database
|
|
flag_restore = true;
|
|
break;
|
|
case IN_SW_BURP_USER: // default user name
|
|
case IN_SW_BURP_PASS: // default password
|
|
case IN_SW_BURP_SE: // service name
|
|
if (itr >= argc - 1)
|
|
{
|
|
int errnum = inSw->in_sw == IN_SW_BURP_USER ? 188 : // user name parameter missing
|
|
inSw->in_sw == IN_SW_BURP_PASS ? 189 : // password parameter missing
|
|
273; // service name parameter missing
|
|
|
|
BURP_error(errnum, true);
|
|
}
|
|
else
|
|
{
|
|
argv[itr++] = 0;
|
|
switch (inSw->in_sw)
|
|
{
|
|
case IN_SW_BURP_USER: // default user name
|
|
usr = argv[itr];
|
|
break;
|
|
case IN_SW_BURP_PASS: // default password
|
|
pswd = argv[itr];
|
|
pswd2 = pswd.nullStr();
|
|
uSvc->hidePasswd(argv, itr);
|
|
break;
|
|
case IN_SW_BURP_SE: // service name
|
|
service = argv[itr];
|
|
break;
|
|
}
|
|
argv[itr] = 0;
|
|
}
|
|
break;
|
|
case IN_SW_BURP_FETCHPASS:
|
|
argv[itr] = 0;
|
|
processFetchPass(pswd2, itr, argc, argv);
|
|
pswd = pswd2;
|
|
argv[itr] = 0;
|
|
break;
|
|
case IN_SW_BURP_V: // verify actions
|
|
if (flag_verbint)
|
|
BURP_error(329, true); // verify (verbose) and verbint options are mutually exclusive
|
|
flag_verbose = true;
|
|
break;
|
|
case IN_SW_BURP_VERBINT: // verify with explicit reporting interval for records
|
|
if (flag_verbose)
|
|
BURP_error(329, true); // verify (verbose) and verbint options are mutually exclusive
|
|
if (flag_verbint)
|
|
BURP_error(333, true, SafeArg() << inSw->in_sw_name << verbint_val);
|
|
if (itr >= argc - 1)
|
|
BURP_error(326, true); // verbose interval value parameter missing
|
|
verbint_val = get_number(argv[++itr]);
|
|
if (verbint_val < MIN_VERBOSE_INTERVAL)
|
|
{
|
|
// verbose interval value cannot be smaller than @1
|
|
BURP_error(327, true, SafeArg() << MIN_VERBOSE_INTERVAL);
|
|
}
|
|
flag_verbint = true;
|
|
break;
|
|
#ifdef TRUSTED_AUTH
|
|
case IN_SW_BURP_TRUSTED_AUTH: // use trusted auth
|
|
flag_trusted = true;
|
|
argv[itr] = 0;
|
|
break;
|
|
#endif
|
|
}
|
|
}
|
|
|
|
const Firebird::string* dbName = flag_restore ? &files[1] : &files[0];
|
|
|
|
FbLocalStatus status;
|
|
Firebird::IService* svc_handle = nullptr;
|
|
|
|
try
|
|
{
|
|
Firebird::ClumpletWriter spb(Firebird::ClumpletWriter::spbList, MAX_DPB_SIZE);
|
|
|
|
// isc_spb_user_name
|
|
// isc_spb_password
|
|
// isc_spb_trusted_auth
|
|
// isc_spb_options
|
|
|
|
if (usr.hasData())
|
|
{
|
|
spb.insertString(isc_spb_user_name, usr);
|
|
}
|
|
if (pswd.hasData())
|
|
{
|
|
spb.insertString(isc_spb_password, pswd);
|
|
}
|
|
if (dbName->hasData())
|
|
{
|
|
spb.insertString(isc_spb_expected_db, *dbName);
|
|
}
|
|
#ifdef TRUSTED_AUTH
|
|
if (flag_trusted)
|
|
{
|
|
spb.insertTag(isc_spb_trusted_auth);
|
|
}
|
|
#endif
|
|
|
|
// Fill command line options
|
|
Firebird::string options;
|
|
for (int itr = 1; itr < argc; ++itr)
|
|
{
|
|
if (!argv[itr])
|
|
{
|
|
continue;
|
|
}
|
|
Firebird::UtilSvc::addStringWithSvcTrmntr(argv[itr], options);
|
|
}
|
|
options.rtrim();
|
|
|
|
spb.insertString(isc_spb_command_line, options);
|
|
|
|
svc_handle = Firebird::DispatcherPtr()->attachServiceManager(&status, service.c_str(),
|
|
spb.getBufferLength(), spb.getBuffer());
|
|
if (!status.isSuccess())
|
|
{
|
|
BURP_print_status(true, &status, 83);
|
|
// msg 83 Exiting before completion due to errors
|
|
return FINI_ERROR;
|
|
}
|
|
|
|
UCHAR thd[10];
|
|
// 'isc_action_svc_restore/isc_action_svc_backup'
|
|
// 'isc_spb_verbose'
|
|
// 'isc_spb_verbint'
|
|
|
|
UCHAR* thd_ptr = thd;
|
|
if (flag_restore)
|
|
*thd_ptr++ = isc_action_svc_restore;
|
|
else
|
|
*thd_ptr++ = isc_action_svc_backup;
|
|
|
|
if (flag_verbose)
|
|
*thd_ptr++ = isc_spb_verbose;
|
|
|
|
if (flag_verbint)
|
|
{
|
|
*thd_ptr++ = isc_spb_verbint;
|
|
//stream verbint_val into a SPB
|
|
put_vax_long(thd_ptr, verbint_val);
|
|
thd_ptr += sizeof(SLONG);
|
|
flag_verbose = true;
|
|
}
|
|
|
|
const USHORT thdlen = thd_ptr - thd;
|
|
fb_assert(thdlen <= sizeof(thd));
|
|
|
|
svc_handle->start(&status, thdlen, thd);
|
|
if (!status.isSuccess())
|
|
{
|
|
BURP_print_status(true, &status, 83);
|
|
// msg 83 Exiting before completion due to errors
|
|
svc_handle->release();
|
|
return FINI_ERROR;
|
|
}
|
|
|
|
// What are we going to receive from service manager
|
|
Firebird::ClumpletWriter receive(Firebird::ClumpletWriter::SpbReceiveItems, 16);
|
|
receive.insertTag(flag_verbose ? isc_info_svc_line : isc_info_svc_to_eof);
|
|
if (flag_restore)
|
|
receive.insertTag(isc_info_svc_stdin);
|
|
|
|
unsigned int stdinRequest = 0;
|
|
for (bool running = true; running;) {
|
|
// Were we requested to send some data
|
|
Firebird::ClumpletWriter send(Firebird::ClumpletWriter::SpbSendItems, MAX_DPB_SIZE);
|
|
UCHAR respbuf[16384];
|
|
if (stdinRequest)
|
|
{
|
|
if (stdinRequest > sizeof(respbuf))
|
|
stdinRequest = sizeof(respbuf);
|
|
|
|
stdinRequest = binIn(respbuf, stdinRequest);
|
|
send.insertBytes(isc_info_svc_line, respbuf, stdinRequest);
|
|
}
|
|
|
|
svc_handle->query(&status, send.getBufferLength(), send.getBuffer(),
|
|
receive.getBufferLength(), receive.getBuffer(),
|
|
sizeof(respbuf), respbuf);
|
|
if (!status.isSuccess())
|
|
{
|
|
BURP_print_status(true, &status, 83);
|
|
// msg 83 Exiting before completion due to errors
|
|
svc_handle->release();
|
|
return FINI_ERROR;
|
|
}
|
|
|
|
Firebird::ClumpletReader resp(Firebird::ClumpletReader::SpbResponse, respbuf, sizeof(respbuf));
|
|
stdinRequest = 0;
|
|
int len = 0;
|
|
Firebird::string line;
|
|
bool not_ready = false;
|
|
for (resp.rewind(); running && !resp.isEof(); resp.moveNext())
|
|
{
|
|
switch(resp.getClumpTag())
|
|
{
|
|
case isc_info_svc_to_eof:
|
|
len = resp.getClumpLength();
|
|
if (len)
|
|
binOut(resp.getBytes(), len);
|
|
break;
|
|
case isc_info_svc_line:
|
|
resp.getString(line);
|
|
len = line.length();
|
|
if (len)
|
|
burp_output(false, "%s\n", line.c_str());
|
|
break;
|
|
case isc_info_svc_stdin:
|
|
stdinRequest = resp.getInt();
|
|
break;
|
|
case isc_info_end:
|
|
running = false;
|
|
break;
|
|
case isc_info_data_not_ready:
|
|
case isc_info_svc_timeout:
|
|
not_ready = true;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (len || stdinRequest || not_ready)
|
|
running = true;
|
|
}
|
|
|
|
svc_handle->release();
|
|
return FINI_OK;
|
|
}
|
|
catch (const Firebird::Exception& e)
|
|
{
|
|
FbLocalStatus s;
|
|
e.stuffException(&s);
|
|
BURP_print_status(true, &s, 83);
|
|
// msg 83 Exiting before completion due to errors
|
|
if (svc_handle)
|
|
svc_handle->release();
|
|
return FINI_ERROR;
|
|
}
|
|
}
|
|
|
|
|
|
static Switches::in_sw_tab_t* findSwitchOrThrow(Firebird::UtilSvc* uSvc, Switches& switches, Firebird::string& sw)
|
|
{
|
|
/**************************************
|
|
*
|
|
* f i n d S w i t c h O r T h r o w
|
|
*
|
|
**************************************
|
|
*
|
|
* Functional description
|
|
* Returns pointer to in_sw_tab entry for current switch
|
|
* If not a switch, returns 0.
|
|
* If no match, throws.
|
|
*
|
|
**************************************/
|
|
bool invalid = false;
|
|
Switches::in_sw_tab_t* rc = switches.findSwitchMod(sw, &invalid);
|
|
if (rc)
|
|
return rc;
|
|
|
|
if (invalid)
|
|
{
|
|
if (! uSvc->isService())
|
|
{
|
|
BURP_print(true, 137, sw.c_str());
|
|
// msg 137 unknown switch %s
|
|
burp_usage(switches);
|
|
BURP_error(1, true);
|
|
// msg 1: found unknown switch
|
|
}
|
|
else
|
|
{
|
|
BURP_error(137, true, sw.c_str());
|
|
// msg 137 unknown switch %s
|
|
}
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
|
|
int gbak(Firebird::UtilSvc* uSvc)
|
|
{
|
|
/**************************************
|
|
*
|
|
* g b a k
|
|
*
|
|
**************************************
|
|
*
|
|
* Functional description
|
|
* Routine called by command line utility and services API.
|
|
*
|
|
**************************************/
|
|
gbak_action action = QUIT;
|
|
int exit_code = FINI_ERROR;
|
|
|
|
BurpGlobals sgbl(uSvc);
|
|
BurpGlobals *tdgbl = &sgbl;
|
|
BurpGlobals::putSpecific(tdgbl);
|
|
|
|
tdgbl->burp_throw = true;
|
|
tdgbl->file_desc = INVALID_HANDLE_VALUE;
|
|
|
|
Firebird::UtilSvc::ArgvType& argv = uSvc->argv;
|
|
const int argc = uSvc->argv.getCount();
|
|
|
|
try
|
|
{
|
|
|
|
// Copy the static const table to a local array for processing.
|
|
Switches switches(reference_burp_in_sw_table, FB_NELEM(reference_burp_in_sw_table), true, true);
|
|
|
|
// test for "-service" switch
|
|
if (switches.exists(IN_SW_BURP_SE, argv.begin(), 1, argc))
|
|
return svc_api_gbak(uSvc, switches);
|
|
|
|
if (argc <= 1)
|
|
{
|
|
burp_usage(switches);
|
|
BURP_exit_local(FINI_ERROR, tdgbl);
|
|
}
|
|
|
|
if (argc == 2 && strcmp(argv[1], "-?") == 0)
|
|
{
|
|
burp_usage(switches);
|
|
BURP_exit_local(FINI_OK, tdgbl);
|
|
}
|
|
|
|
USHORT sw_replace = 0;
|
|
|
|
tdgbl->gbl_sw_compress = true;
|
|
tdgbl->gbl_sw_convert_ext_tables = false;
|
|
tdgbl->gbl_sw_transportable = true;
|
|
tdgbl->gbl_sw_ignore_limbo = false;
|
|
tdgbl->gbl_sw_blk_factor = 0;
|
|
tdgbl->gbl_sw_no_reserve = false;
|
|
tdgbl->gbl_sw_old_descriptions = false;
|
|
tdgbl->gbl_sw_mode = false;
|
|
tdgbl->gbl_sw_skip_count = 0;
|
|
tdgbl->gbl_sw_par_workers = uSvc->getParallelWorkers();
|
|
tdgbl->action = NULL;
|
|
|
|
burp_fil* file = NULL;
|
|
burp_fil* file_list = NULL;
|
|
tdgbl->io_buffer_size = GBAK_IO_BUFFER_SIZE;
|
|
|
|
const char none[] = "-*NONE*";
|
|
bool verbint = false;
|
|
bool noGarbage = false, ignoreDamaged = false, noDbTrig = false;
|
|
bool transportableMentioned = false;
|
|
Firebird::string replicaMode;
|
|
|
|
for (int itr = 1; itr < argc; ++itr)
|
|
{
|
|
Firebird::string str = argv[itr];
|
|
if (str.isEmpty())
|
|
{
|
|
continue;
|
|
}
|
|
if (str[str.length() - 1] == ',')
|
|
{
|
|
str.erase(str.length() - 1, 1);
|
|
// Let's ignore single comma
|
|
if (str.isEmpty())
|
|
continue;
|
|
}
|
|
|
|
if (str[0] != switch_char)
|
|
{
|
|
if (!file || file->fil_length || !get_size(argv[itr], file))
|
|
{
|
|
// Miserable thing must be a filename
|
|
// (dummy in a length for the backup file
|
|
|
|
file = FB_NEW_POOL(tdgbl->getPool()) burp_fil(tdgbl->getPool());
|
|
file->fil_name = str.ToPathName();
|
|
file->fil_length = file_list ? 0 : MAX_LENGTH;
|
|
file->fil_next = file_list;
|
|
file_list = file;
|
|
}
|
|
continue;
|
|
}
|
|
|
|
if (str.length() == 1)
|
|
{
|
|
str = none;
|
|
}
|
|
|
|
Switches::in_sw_tab_t* const in_sw_tab = findSwitchOrThrow(uSvc, switches, str);
|
|
fb_assert(in_sw_tab);
|
|
//in_sw_tab->in_sw_state = true; It's not enough with switches that have multiple spellings
|
|
switches.activate(in_sw_tab->in_sw);
|
|
|
|
switch (in_sw_tab->in_sw)
|
|
{
|
|
case IN_SW_BURP_RECREATE:
|
|
{
|
|
int real_sw = IN_SW_BURP_C;
|
|
if ((itr < argc - 1) && (*argv[itr + 1] != switch_char))
|
|
{
|
|
// find optional BURP_SW_OVERWRITE parameter
|
|
Firebird::string next(argv[itr + 1]);
|
|
next.upper();
|
|
if (strstr(BURP_SW_OVERWRITE, next.c_str()) == BURP_SW_OVERWRITE)
|
|
{
|
|
real_sw = IN_SW_BURP_R;
|
|
itr++;
|
|
}
|
|
}
|
|
|
|
// replace IN_SW_BURP_RECREATE by IN_SW_BURP_R or IN_SW_BURP_C
|
|
in_sw_tab->in_sw_state = false;
|
|
switches.activate(real_sw);
|
|
}
|
|
break;
|
|
case IN_SW_BURP_S:
|
|
if (tdgbl->gbl_sw_skip_count)
|
|
BURP_error(333, true, SafeArg() << in_sw_tab->in_sw_name << tdgbl->gbl_sw_skip_count);
|
|
if (++itr >= argc)
|
|
{
|
|
BURP_error(200, true);
|
|
// msg 200: missing parameter for the number of bytes to be skipped
|
|
}
|
|
tdgbl->gbl_sw_skip_count = get_number(argv[itr]);
|
|
if (!tdgbl->gbl_sw_skip_count)
|
|
{
|
|
BURP_error(201, true, argv[itr]);
|
|
// msg 201: expected number of bytes to be skipped, encountered "%s"
|
|
}
|
|
break;
|
|
case IN_SW_BURP_P:
|
|
if (tdgbl->gbl_sw_page_size)
|
|
BURP_error(333, true, SafeArg() << in_sw_tab->in_sw_name << tdgbl->gbl_sw_page_size);
|
|
if (++itr >= argc)
|
|
{
|
|
BURP_error(2, true);
|
|
// msg 2 page size parameter missing
|
|
}
|
|
tdgbl->gbl_sw_page_size = (USHORT) get_number(argv[itr]);
|
|
if (!tdgbl->gbl_sw_page_size)
|
|
{
|
|
BURP_error(12, true, argv[itr]);
|
|
// msg 12 expected page size, encountered "%s"
|
|
}
|
|
break;
|
|
case IN_SW_BURP_BU:
|
|
if (tdgbl->gbl_sw_page_buffers)
|
|
BURP_error(333, true, SafeArg() << in_sw_tab->in_sw_name << tdgbl->gbl_sw_page_buffers);
|
|
if (++itr >= argc)
|
|
{
|
|
BURP_error(258, true);
|
|
// msg 258 page buffers parameter missing
|
|
}
|
|
tdgbl->gbl_sw_page_buffers = get_number(argv[itr]);
|
|
if (!tdgbl->gbl_sw_page_buffers)
|
|
{
|
|
BURP_error(259, true, argv[itr]);
|
|
// msg 259 expected page buffers, encountered "%s"
|
|
}
|
|
break;
|
|
case IN_SW_BURP_MODE:
|
|
if (tdgbl->gbl_sw_mode)
|
|
{
|
|
BURP_error(333, true, SafeArg() << in_sw_tab->in_sw_name <<
|
|
(tdgbl->gbl_sw_mode_val ? BURP_SW_MODE_RO : BURP_SW_MODE_RW));
|
|
}
|
|
if (++itr >= argc)
|
|
{
|
|
BURP_error(279, true);
|
|
// msg 279: "read_only" or "read_write" required
|
|
}
|
|
str = argv[itr];
|
|
str.upper();
|
|
if (str == BURP_SW_MODE_RO)
|
|
tdgbl->gbl_sw_mode_val = true;
|
|
else if (str == BURP_SW_MODE_RW)
|
|
tdgbl->gbl_sw_mode_val = false;
|
|
else
|
|
{
|
|
BURP_error(279, true);
|
|
// msg 279: "read_only" or "read_write" required
|
|
}
|
|
tdgbl->gbl_sw_mode = true;
|
|
break;
|
|
case IN_SW_BURP_PASS:
|
|
if (++itr >= argc)
|
|
{
|
|
BURP_error(189, true);
|
|
// password parameter missing
|
|
}
|
|
uSvc->hidePasswd(argv, itr);
|
|
if (tdgbl->gbl_sw_password)
|
|
{
|
|
BURP_error(307, true);
|
|
// too many passwords provided
|
|
}
|
|
tdgbl->gbl_sw_password = argv[itr];
|
|
break;
|
|
case IN_SW_BURP_FETCHPASS:
|
|
processFetchPass(tdgbl->gbl_sw_password, itr, argc, argv);
|
|
break;
|
|
case IN_SW_BURP_USER:
|
|
if (++itr >= argc)
|
|
{
|
|
BURP_error(188, true);
|
|
// user name parameter missing
|
|
}
|
|
tdgbl->gbl_sw_user = argv[itr];
|
|
break;
|
|
case IN_SW_BURP_SKIP_DATA:
|
|
if (++itr >= argc)
|
|
{
|
|
BURP_error(354, true);
|
|
// missing regular expression to skip tables
|
|
}
|
|
tdgbl->setupSkipData(argv[itr]);
|
|
break;
|
|
case IN_SW_BURP_INCLUDE_DATA:
|
|
if (++itr >= argc)
|
|
{
|
|
BURP_error(389, true);
|
|
// missing regular expression to include tables
|
|
}
|
|
tdgbl->setupIncludeData(argv[itr]);
|
|
break;
|
|
case IN_SW_BURP_ROLE:
|
|
if (++itr >= argc)
|
|
{
|
|
BURP_error(253, true);
|
|
// SQL role parameter missing
|
|
}
|
|
tdgbl->gbl_sw_sql_role = argv[itr];
|
|
break;
|
|
case IN_SW_BURP_KEYHOLD:
|
|
if (++itr >= argc)
|
|
{
|
|
BURP_error(381, true);
|
|
// KeyHolder parameter missing
|
|
}
|
|
if (tdgbl->gbl_sw_keyholder)
|
|
BURP_error(334, true, SafeArg() << in_sw_tab->in_sw_name);
|
|
tdgbl->gbl_sw_keyholder = argv[itr];
|
|
break;
|
|
case IN_SW_BURP_CRYPT:
|
|
if (++itr >= argc)
|
|
{
|
|
BURP_error(377, true);
|
|
// CryptPlugin parameter missing
|
|
}
|
|
if (tdgbl->gbl_sw_crypt)
|
|
BURP_error(334, true, SafeArg() << in_sw_tab->in_sw_name);
|
|
tdgbl->gbl_sw_crypt = argv[itr];
|
|
break;
|
|
case IN_SW_BURP_KEYNAME:
|
|
if (++itr >= argc)
|
|
{
|
|
BURP_error(375, true);
|
|
// Key name parameter missing
|
|
}
|
|
if (tdgbl->gbl_sw_keyname)
|
|
BURP_error(334, true, SafeArg() << in_sw_tab->in_sw_name);
|
|
tdgbl->gbl_sw_keyname = argv[itr];
|
|
break;
|
|
case IN_SW_BURP_ZIP:
|
|
if (tdgbl->gbl_sw_zip)
|
|
BURP_error(334, true, SafeArg() << in_sw_tab->in_sw_name);
|
|
tdgbl->gbl_sw_zip = true;
|
|
break;
|
|
case IN_SW_BURP_FA:
|
|
if (tdgbl->gbl_sw_blk_factor)
|
|
BURP_error(333, true, SafeArg() << in_sw_tab->in_sw_name << tdgbl->gbl_sw_blk_factor);
|
|
if (++itr >= argc)
|
|
{
|
|
BURP_error(182, true);
|
|
// msg 182 blocking factor parameter missing
|
|
}
|
|
tdgbl->gbl_sw_blk_factor = get_number(argv[itr]);
|
|
if (!tdgbl->gbl_sw_blk_factor)
|
|
{
|
|
BURP_error(183, true, argv[itr]);
|
|
// msg 183 expected blocking factor, encountered "%s"
|
|
}
|
|
break;
|
|
case IN_SW_BURP_FIX_FSS_DATA:
|
|
if (tdgbl->gbl_sw_fix_fss_data)
|
|
BURP_error(333, true, SafeArg() << in_sw_tab->in_sw_name << tdgbl->gbl_sw_fix_fss_data);
|
|
if (++itr >= argc)
|
|
{
|
|
BURP_error(304, true);
|
|
// Character set parameter missing
|
|
}
|
|
tdgbl->gbl_sw_fix_fss_data = argv[itr];
|
|
break;
|
|
case IN_SW_BURP_FIX_FSS_METADATA:
|
|
if (tdgbl->gbl_sw_fix_fss_metadata)
|
|
BURP_error(333, true, SafeArg() << in_sw_tab->in_sw_name << tdgbl->gbl_sw_fix_fss_metadata);
|
|
if (++itr >= argc)
|
|
{
|
|
BURP_error(304, true);
|
|
// Character set parameter missing
|
|
}
|
|
tdgbl->gbl_sw_fix_fss_metadata = argv[itr];
|
|
break;
|
|
case IN_SW_BURP_SE:
|
|
if (++itr >= argc)
|
|
{
|
|
BURP_error(273, true);
|
|
// msg 273: service name parameter missing
|
|
}
|
|
// skip a service specification
|
|
in_sw_tab->in_sw_state = false;
|
|
break;
|
|
case IN_SW_BURP_PARALLEL_WORKERS:
|
|
if (++itr >= argc)
|
|
{
|
|
BURP_error(407, true);
|
|
// msg 407 parallel workers parameter missing
|
|
}
|
|
tdgbl->gbl_sw_par_workers = get_number(argv[itr]);
|
|
if (tdgbl->gbl_sw_par_workers <= 0)
|
|
{
|
|
BURP_error(408, true, argv[itr]);
|
|
// msg 408 expected parallel workers, encountered "%s"
|
|
}
|
|
break;
|
|
case IN_SW_BURP_Y:
|
|
{
|
|
// want to do output redirect handling now instead of waiting
|
|
const TEXT* redirect = NULL;
|
|
if (++itr < argc)
|
|
{
|
|
redirect = argv[itr];
|
|
if (*redirect == switch_char)
|
|
{
|
|
redirect = NULL;
|
|
}
|
|
}
|
|
if (!redirect)
|
|
{
|
|
BURP_error(4, true);
|
|
// msg 4 redirect location for output is not specified
|
|
}
|
|
|
|
Firebird::string up(redirect);
|
|
up.upper();
|
|
tdgbl->sw_redirect = (up == output_suppress) ? NOOUTPUT : REDIRECT;
|
|
|
|
if (tdgbl->sw_redirect == REDIRECT) // not NOREDIRECT, and not NOOUTPUT
|
|
{
|
|
// Make sure the status file doesn't already exist
|
|
FILE* tmp_outfile = os_utils::fopen(redirect, fopen_read_type);
|
|
if (tmp_outfile)
|
|
{
|
|
fclose(tmp_outfile);
|
|
BURP_error(66, true, SafeArg() << redirect);
|
|
// msg 66 can't open status and error output file %s
|
|
}
|
|
if (! (tdgbl->output_file = os_utils::fopen(redirect, fopen_write_type)))
|
|
{
|
|
BURP_error(66, true, SafeArg() << redirect);
|
|
// msg 66 can't open status and error output file %s
|
|
}
|
|
}
|
|
}
|
|
break;
|
|
case IN_SW_BURP_VERBINT:
|
|
{
|
|
if (tdgbl->gbl_sw_verbose)
|
|
BURP_error(329, true); // verify (verbose) and verbint options are mutually exclusive
|
|
if (verbint)
|
|
BURP_error(333, true, SafeArg() << in_sw_tab->in_sw_name << tdgbl->verboseInterval);
|
|
if (++itr >= argc)
|
|
BURP_error(326, true); // verbose interval parameter missing
|
|
|
|
SLONG verbint_val = get_number(argv[itr]);
|
|
if (verbint_val < MIN_VERBOSE_INTERVAL)
|
|
{
|
|
// verbose interval value cannot be smaller than @1
|
|
BURP_error(327, true, SafeArg() << MIN_VERBOSE_INTERVAL);
|
|
}
|
|
tdgbl->verboseInterval = verbint_val;
|
|
verbint = true;
|
|
}
|
|
break;
|
|
case IN_SW_BURP_V:
|
|
if (verbint)
|
|
BURP_error(329, true); // verify (verbose) and verbint options are mutually exclusive
|
|
if (tdgbl->gbl_sw_verbose)
|
|
BURP_error(334, true, SafeArg() << in_sw_tab->in_sw_name);
|
|
tdgbl->gbl_sw_verbose = true;
|
|
break;
|
|
case IN_SW_BURP_STATS:
|
|
if (tdgbl->gbl_stat_flags)
|
|
BURP_error(334, true, SafeArg() << in_sw_tab->in_sw_name);
|
|
|
|
if (++itr >= argc)
|
|
BURP_error(366, true); // statistics parameter missing
|
|
|
|
{ // scope
|
|
const char* perf_val = argv[itr];
|
|
const char* c = perf_val;
|
|
size_t len = strlen(STAT_CHARS);
|
|
|
|
for (; *c && len; c++, len--)
|
|
{
|
|
const char* pos = strchr(STAT_CHARS, toupper(*c));
|
|
if (!pos)
|
|
BURP_error(367, true, SafeArg() << *c); // wrong char "@1" at statistics parameter
|
|
|
|
tdgbl->gbl_stat_flags |= 1 << (pos - STAT_CHARS);
|
|
}
|
|
|
|
if (*c)
|
|
BURP_error(368, true); // too many chars at statistics parameter
|
|
}
|
|
break;
|
|
|
|
case IN_SW_BURP_CO:
|
|
if (tdgbl->gbl_sw_convert_ext_tables)
|
|
BURP_error(334, true, SafeArg() << in_sw_tab->in_sw_name);
|
|
tdgbl->gbl_sw_convert_ext_tables = true;
|
|
break;
|
|
case IN_SW_BURP_DIRECT_IO:
|
|
if (tdgbl->gbl_sw_direct_io)
|
|
BURP_error(334, true, SafeArg() << in_sw_tab->in_sw_name);
|
|
tdgbl->gbl_sw_direct_io = true;
|
|
break;
|
|
case IN_SW_BURP_E:
|
|
if (!tdgbl->gbl_sw_compress)
|
|
BURP_error(334, true, SafeArg() << in_sw_tab->in_sw_name);
|
|
tdgbl->gbl_sw_compress = false;
|
|
break;
|
|
case IN_SW_BURP_G:
|
|
if (noGarbage)
|
|
BURP_error(334, true, SafeArg() << in_sw_tab->in_sw_name);
|
|
noGarbage = true;
|
|
break;
|
|
case IN_SW_BURP_I:
|
|
if (tdgbl->gbl_sw_deactivate_indexes)
|
|
BURP_error(334, true, SafeArg() << in_sw_tab->in_sw_name);
|
|
tdgbl->gbl_sw_deactivate_indexes = true;
|
|
break;
|
|
case IN_SW_BURP_IG:
|
|
if (ignoreDamaged)
|
|
BURP_error(334, true, SafeArg() << in_sw_tab->in_sw_name);
|
|
ignoreDamaged = true;
|
|
break;
|
|
|
|
case IN_SW_BURP_K:
|
|
if (tdgbl->gbl_sw_kill)
|
|
BURP_error(334, true, SafeArg() << in_sw_tab->in_sw_name);
|
|
tdgbl->gbl_sw_kill = true;
|
|
break;
|
|
|
|
case IN_SW_BURP_L:
|
|
if (tdgbl->gbl_sw_ignore_limbo)
|
|
BURP_error(334, true, SafeArg() << in_sw_tab->in_sw_name);
|
|
tdgbl->gbl_sw_ignore_limbo = true;
|
|
break;
|
|
|
|
case IN_SW_BURP_M:
|
|
if (tdgbl->gbl_sw_meta)
|
|
BURP_error(334, true, SafeArg() << in_sw_tab->in_sw_name);
|
|
tdgbl->gbl_sw_meta = true;
|
|
break;
|
|
case IN_SW_BURP_N:
|
|
if (tdgbl->gbl_sw_novalidity)
|
|
BURP_error(334, true, SafeArg() << in_sw_tab->in_sw_name);
|
|
tdgbl->gbl_sw_novalidity = true;
|
|
break;
|
|
case IN_SW_BURP_NOD:
|
|
if (noDbTrig)
|
|
BURP_error(334, true, SafeArg() << in_sw_tab->in_sw_name);
|
|
noDbTrig = true;
|
|
break;
|
|
case IN_SW_BURP_O:
|
|
if (tdgbl->gbl_sw_incremental)
|
|
BURP_error(334, true, SafeArg() << in_sw_tab->in_sw_name);
|
|
tdgbl->gbl_sw_incremental = true;
|
|
break;
|
|
case IN_SW_BURP_OL:
|
|
if (tdgbl->gbl_sw_old_descriptions)
|
|
BURP_error(334, true, SafeArg() << in_sw_tab->in_sw_name);
|
|
tdgbl->gbl_sw_old_descriptions = true;
|
|
break;
|
|
case IN_SW_BURP_US:
|
|
if (tdgbl->gbl_sw_no_reserve)
|
|
BURP_error(334, true, SafeArg() << in_sw_tab->in_sw_name);
|
|
tdgbl->gbl_sw_no_reserve = true;
|
|
break;
|
|
case IN_SW_BURP_NT: // Backup non-transportable format
|
|
if (transportableMentioned)
|
|
{
|
|
if (tdgbl->gbl_sw_transportable)
|
|
BURP_error(332, true, SafeArg() << "NT" << "T");
|
|
else
|
|
BURP_error(334, true, SafeArg() << in_sw_tab->in_sw_name);
|
|
}
|
|
tdgbl->gbl_sw_transportable = false;
|
|
transportableMentioned = true;
|
|
break;
|
|
case IN_SW_BURP_T:
|
|
if (transportableMentioned)
|
|
{
|
|
if (!tdgbl->gbl_sw_transportable)
|
|
BURP_error(332, true, SafeArg() << "NT" << "T");
|
|
else
|
|
BURP_error(334, true, SafeArg() << in_sw_tab->in_sw_name);
|
|
}
|
|
tdgbl->gbl_sw_transportable = true;
|
|
transportableMentioned = true;
|
|
break;
|
|
case IN_SW_BURP_REPLICA:
|
|
if (replicaMode.length())
|
|
BURP_error(333, true, SafeArg() << in_sw_tab->in_sw_name << replicaMode.c_str());
|
|
if (++itr >= argc)
|
|
{
|
|
BURP_error(404, true);
|
|
// msg 404: "none", "read_only" or "read_write" required
|
|
}
|
|
str = argv[itr];
|
|
str.upper();
|
|
if (str == BURP_SW_MODE_NONE)
|
|
tdgbl->gbl_sw_replica = REPLICA_NONE;
|
|
else if (str == BURP_SW_MODE_RO)
|
|
tdgbl->gbl_sw_replica = REPLICA_READ_ONLY;
|
|
else if (str == BURP_SW_MODE_RW)
|
|
tdgbl->gbl_sw_replica = REPLICA_READ_WRITE;
|
|
else
|
|
{
|
|
BURP_error(404, true);
|
|
// msg 404: "none", "read_only" or "read_write" required
|
|
}
|
|
replicaMode = str;
|
|
break;
|
|
}
|
|
} // for
|
|
|
|
// reverse the linked list of file blocks
|
|
|
|
tdgbl->gbl_sw_files = NULL;
|
|
|
|
burp_fil* next_file = NULL;
|
|
for (file = file_list; file; file = next_file)
|
|
{
|
|
next_file = file->fil_next;
|
|
file->fil_next = tdgbl->gbl_sw_files;
|
|
tdgbl->gbl_sw_files = file;
|
|
}
|
|
|
|
// pop off the obviously boring ones, plus do some checking
|
|
|
|
const TEXT* file1 = NULL;
|
|
const TEXT* file2 = NULL;
|
|
for (file = tdgbl->gbl_sw_files; file; file = file->fil_next)
|
|
{
|
|
if (!file1)
|
|
file1 = file->fil_name.c_str();
|
|
else if (!file2)
|
|
file2 = file->fil_name.c_str();
|
|
|
|
Firebird::PathName expanded;
|
|
expandDatabaseName(file->fil_name, expanded, NULL);
|
|
|
|
for (file_list = file->fil_next; file_list;
|
|
file_list = file_list->fil_next)
|
|
{
|
|
if (file->fil_name == file_list->fil_name || expanded == file_list->fil_name)
|
|
{
|
|
BURP_error(9, true);
|
|
// msg 9 mutiple sources or destinations specified
|
|
}
|
|
|
|
Firebird::PathName expanded2;
|
|
expandDatabaseName(file_list->fil_name, expanded2, NULL);
|
|
if (file->fil_name == expanded2 || expanded == expanded2)
|
|
{
|
|
BURP_error(9, true);
|
|
// msg 9 mutiple sources or destinations specified
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
// Initialize 'dpb'
|
|
Firebird::ClumpletWriter dpb(Firebird::ClumpletReader::dpbList, MAX_DPB_SIZE);
|
|
|
|
dpb.insertString(isc_dpb_gbak_attach, FB_VERSION, fb_strlen(FB_VERSION));
|
|
uSvc->fillDpb(dpb);
|
|
|
|
const UCHAR* authBlock;
|
|
unsigned int authSize = uSvc->getAuthBlock(&authBlock);
|
|
if (authBlock)
|
|
{
|
|
dpb.insertBytes(isc_dpb_auth_block, authBlock, authSize);
|
|
}
|
|
|
|
// We call getTableMod() because we are interested in the items that were activated previously,
|
|
// not in the original, unchanged table that "switches" took as parameter in the constructor.
|
|
for (const Switches::in_sw_tab_t* in_sw_tab = switches.getTableMod();
|
|
in_sw_tab->in_sw_name; in_sw_tab++)
|
|
{
|
|
if (!in_sw_tab->in_sw_state)
|
|
continue;
|
|
|
|
switch (in_sw_tab->in_sw)
|
|
{
|
|
case IN_SW_BURP_B:
|
|
if (sw_replace)
|
|
BURP_error(5, true);
|
|
// msg 5 conflicting switches for backup/restore
|
|
sw_replace = IN_SW_BURP_B;
|
|
break;
|
|
|
|
case IN_SW_BURP_C:
|
|
if (sw_replace == IN_SW_BURP_B)
|
|
BURP_error(5, true);
|
|
// msg 5 conflicting switches for backup/restore
|
|
if (sw_replace != IN_SW_BURP_R)
|
|
sw_replace = IN_SW_BURP_C;
|
|
break;
|
|
|
|
case IN_SW_BURP_G:
|
|
dpb.insertTag(isc_dpb_no_garbage_collect);
|
|
break;
|
|
|
|
case IN_SW_BURP_IG:
|
|
dpb.insertByte(isc_dpb_damaged, 1);
|
|
break;
|
|
|
|
case IN_SW_BURP_MODE:
|
|
// checked before, seems irrelevant to activate it again
|
|
tdgbl->gbl_sw_mode = true;
|
|
break;
|
|
|
|
case IN_SW_BURP_NOD:
|
|
dpb.insertByte(isc_dpb_no_db_triggers, 1);
|
|
break;
|
|
|
|
case IN_SW_BURP_PASS:
|
|
case IN_SW_BURP_FETCHPASS:
|
|
if (!authBlock)
|
|
{
|
|
dpb.insertString(tdgbl->uSvc->isService() ? isc_dpb_password_enc : isc_dpb_password,
|
|
tdgbl->gbl_sw_password, fb_strlen(tdgbl->gbl_sw_password));
|
|
}
|
|
break;
|
|
|
|
case IN_SW_BURP_R:
|
|
if (sw_replace == IN_SW_BURP_B)
|
|
BURP_error(5, true);
|
|
// msg 5 conflicting switches for backup/restore
|
|
sw_replace = IN_SW_BURP_R;
|
|
break;
|
|
|
|
/*
|
|
case IN_SW_BURP_U:
|
|
BURP_error(7, true);
|
|
// msg 7 protection isn't there yet
|
|
break;
|
|
*/
|
|
|
|
case IN_SW_BURP_ROLE:
|
|
dpb.insertString(isc_dpb_sql_role_name,
|
|
tdgbl->gbl_sw_sql_role, fb_strlen(tdgbl->gbl_sw_sql_role));
|
|
break;
|
|
|
|
case IN_SW_BURP_USER:
|
|
if (!authBlock)
|
|
{
|
|
dpb.insertString(isc_dpb_user_name, tdgbl->gbl_sw_user, fb_strlen(tdgbl->gbl_sw_user));
|
|
}
|
|
break;
|
|
|
|
#ifdef TRUSTED_AUTH
|
|
case IN_SW_BURP_TRUSTED_AUTH:
|
|
if (!dpb.find(isc_dpb_trusted_auth))
|
|
{
|
|
dpb.insertTag(isc_dpb_trusted_auth);
|
|
}
|
|
break;
|
|
#endif
|
|
|
|
case IN_SW_BURP_Z:
|
|
BURP_print(false, 91, FB_VERSION);
|
|
// msg 91 gbak version %s
|
|
tdgbl->gbl_sw_version = true;
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (!sw_replace)
|
|
sw_replace = IN_SW_BURP_B;
|
|
|
|
if (sw_replace == IN_SW_BURP_B)
|
|
{
|
|
for (burp_fil* f = tdgbl->gbl_sw_files; f; f = f->fil_next)
|
|
{
|
|
if (f->fil_name == "stdout")
|
|
{
|
|
// the very first thing to do to not corrupt backup file...
|
|
tdgbl->uSvc->setDataMode(true);
|
|
}
|
|
}
|
|
}
|
|
|
|
if (tdgbl->gbl_sw_page_size)
|
|
{
|
|
if (sw_replace == IN_SW_BURP_B)
|
|
BURP_error(8, true); // msg 8 page size is allowed only on restore or create
|
|
int temp = tdgbl->gbl_sw_page_size;
|
|
for (int curr_pg_size = MIN_PAGE_SIZE; curr_pg_size <= MAX_PAGE_SIZE; curr_pg_size <<= 1)
|
|
{
|
|
if (temp <= curr_pg_size)
|
|
{
|
|
temp = curr_pg_size;
|
|
break;
|
|
}
|
|
}
|
|
if (temp > MAX_PAGE_SIZE)
|
|
{
|
|
BURP_error(3, true, SafeArg() << tdgbl->gbl_sw_page_size);
|
|
// msg 3 Page size specified (%ld) greater than limit (MAX_PAGE_SIZE bytes)
|
|
}
|
|
if (temp != tdgbl->gbl_sw_page_size)
|
|
{
|
|
BURP_print(false, 103, SafeArg() << tdgbl->gbl_sw_page_size << temp);
|
|
// msg 103 page size specified (%ld bytes) rounded up to %ld bytes
|
|
tdgbl->gbl_sw_page_size = temp;
|
|
}
|
|
}
|
|
|
|
if (sw_replace == IN_SW_BURP_B)
|
|
{
|
|
if (tdgbl->gbl_sw_page_buffers)
|
|
BURP_error(260, true); // msg 260 page buffers is allowed only on restore or create
|
|
|
|
int errNum = IN_SW_BURP_0;
|
|
|
|
if (tdgbl->gbl_sw_fix_fss_data)
|
|
errNum = IN_SW_BURP_FIX_FSS_DATA;
|
|
else if (tdgbl->gbl_sw_fix_fss_metadata)
|
|
errNum = IN_SW_BURP_FIX_FSS_METADATA;
|
|
else if (tdgbl->gbl_sw_deactivate_indexes)
|
|
errNum = IN_SW_BURP_I;
|
|
else if (tdgbl->gbl_sw_kill)
|
|
errNum = IN_SW_BURP_K;
|
|
else if (tdgbl->gbl_sw_mode)
|
|
errNum = IN_SW_BURP_MODE;
|
|
else if (tdgbl->gbl_sw_novalidity)
|
|
errNum = IN_SW_BURP_N;
|
|
else if (tdgbl->gbl_sw_incremental)
|
|
errNum = IN_SW_BURP_O;
|
|
else if (tdgbl->gbl_sw_skip_count)
|
|
errNum = IN_SW_BURP_S;
|
|
else if (tdgbl->gbl_sw_no_reserve)
|
|
errNum = IN_SW_BURP_US;
|
|
else if (tdgbl->gbl_sw_replica.has_value())
|
|
errNum = IN_SW_BURP_REPLICA;
|
|
|
|
if (errNum != IN_SW_BURP_0)
|
|
{
|
|
const char* msg = switches.findNameByTag(errNum);
|
|
BURP_error(330, true, SafeArg() << msg);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
fb_assert(sw_replace == IN_SW_BURP_C || sw_replace == IN_SW_BURP_R);
|
|
int errNum = IN_SW_BURP_0;
|
|
|
|
if (tdgbl->gbl_sw_convert_ext_tables)
|
|
errNum = IN_SW_BURP_CO;
|
|
else if (!tdgbl->gbl_sw_compress)
|
|
errNum = IN_SW_BURP_E;
|
|
else if (tdgbl->gbl_sw_blk_factor)
|
|
errNum = IN_SW_BURP_FA;
|
|
else if (noGarbage)
|
|
errNum = IN_SW_BURP_G;
|
|
else if (ignoreDamaged)
|
|
errNum = IN_SW_BURP_IG;
|
|
else if (tdgbl->gbl_sw_ignore_limbo)
|
|
errNum = IN_SW_BURP_L;
|
|
else if (noDbTrig)
|
|
errNum = IN_SW_BURP_NOD;
|
|
else if (transportableMentioned)
|
|
{
|
|
if (tdgbl->gbl_sw_transportable)
|
|
errNum = IN_SW_BURP_T;
|
|
else
|
|
errNum = IN_SW_BURP_NT;
|
|
}
|
|
else if (tdgbl->gbl_sw_old_descriptions)
|
|
errNum = IN_SW_BURP_OL;
|
|
else if (tdgbl->gbl_sw_zip)
|
|
errNum = IN_SW_BURP_ZIP;
|
|
|
|
if (errNum != IN_SW_BURP_0)
|
|
{
|
|
const char* msg = switches.findNameByTag(errNum);
|
|
BURP_error(331, true, SafeArg() << msg);
|
|
}
|
|
}
|
|
|
|
if (!tdgbl->gbl_sw_blk_factor || sw_replace != IN_SW_BURP_B)
|
|
tdgbl->gbl_sw_blk_factor = 1;
|
|
if (verbint)
|
|
tdgbl->gbl_sw_verbose = true;
|
|
|
|
if (!file2)
|
|
BURP_error(10, true);
|
|
// msg 10 requires both input and output filenames
|
|
|
|
if (!strcmp(file1, file2))
|
|
BURP_error(11, true);
|
|
// msg 11 input and output have the same name. Disallowed.
|
|
|
|
{ // scope
|
|
// The string result produced by ctime contains exactly 26 characters and
|
|
// gbl_backup_start_time is TEXT[30], but let's make sure we don't overflow
|
|
// due to any change.
|
|
const time_t clock = time(NULL);
|
|
fb_utils::copy_terminate(tdgbl->gbl_backup_start_time, ctime(&clock),
|
|
sizeof(tdgbl->gbl_backup_start_time));
|
|
TEXT* nlp = tdgbl->gbl_backup_start_time + strlen(tdgbl->gbl_backup_start_time) - 1;
|
|
if (*nlp == '\n')
|
|
*nlp = 0;
|
|
} // scope
|
|
|
|
tdgbl->action = (burp_act*) BURP_alloc_zero(ACT_LEN);
|
|
tdgbl->action->act_total = 0;
|
|
tdgbl->action->act_file = NULL;
|
|
tdgbl->action->act_action = ACT_unknown;
|
|
|
|
action = open_files(file1, &file2, sw_replace, dpb);
|
|
MVOL_init(tdgbl->io_buffer_size);
|
|
uSvc->started();
|
|
|
|
int result;
|
|
tdgbl->gbl_dpb_data.add(dpb.getBuffer(), dpb.getBufferLength());
|
|
|
|
switch (action)
|
|
{
|
|
case RESTORE:
|
|
tdgbl->gbl_sw_overwrite = (sw_replace == IN_SW_BURP_R);
|
|
result = RESTORE_restore(file1, file2);
|
|
break;
|
|
|
|
case BACKUP:
|
|
result = BACKUP_backup(file1, file2);
|
|
break;
|
|
|
|
case QUIT:
|
|
BURP_abort();
|
|
return 0;
|
|
|
|
default:
|
|
// result undefined
|
|
fb_assert(false);
|
|
return 0;
|
|
}
|
|
if (result != FINI_OK && result != FINI_DB_NOT_ONLINE)
|
|
BURP_abort();
|
|
|
|
BURP_exit_local(result, tdgbl);
|
|
} // try
|
|
|
|
catch (const Firebird::LongJump&)
|
|
{
|
|
// All calls to exit_local(), normal and error exits, wind up here
|
|
tdgbl->burp_throw = false;
|
|
if (tdgbl->action && tdgbl->action->act_action == ACT_backup_fini)
|
|
{
|
|
tdgbl->exit_code = 0;
|
|
}
|
|
exit_code = tdgbl->exit_code;
|
|
}
|
|
|
|
catch (const Firebird::Exception& e)
|
|
{
|
|
// Non-burp exception was caught
|
|
tdgbl->burp_throw = false;
|
|
e.stuffException(&tdgbl->status_vector);
|
|
BURP_print_status(true, &tdgbl->status_vector);
|
|
if (! tdgbl->uSvc->isService())
|
|
{
|
|
BURP_print(true, 83); // msg 83 Exiting before completion due to errors
|
|
}
|
|
exit_code = FINI_ERROR;
|
|
}
|
|
|
|
// Close the gbak file handles if they still open
|
|
for (burp_fil* file = tdgbl->gbl_sw_backup_files; file; file = file->fil_next)
|
|
{
|
|
if (file->fil_fd != GBAK_STDIN_DESC() && file->fil_fd != GBAK_STDOUT_DESC())
|
|
{
|
|
if (file->fil_fd != INVALID_HANDLE_VALUE)
|
|
{
|
|
close_platf(file->fil_fd);
|
|
|
|
if (exit_code != FINI_OK &&
|
|
(tdgbl->action->act_action == ACT_backup_split || tdgbl->action->act_action == ACT_backup))
|
|
{
|
|
unlink_platf(tdgbl->toSystem(file->fil_name).c_str());
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// Detach from database to release system resources
|
|
if (tdgbl->db_handle)
|
|
{
|
|
close_out_transaction(action, &tdgbl->tr_handle);
|
|
close_out_transaction(action, &tdgbl->global_trans);
|
|
|
|
tdgbl->db_handle->detach(&tdgbl->status_vector);
|
|
|
|
if (tdgbl->status_vector->getState() & Firebird::IStatus::STATE_ERRORS)
|
|
BURP_print_status(true, &tdgbl->status_vector);
|
|
else
|
|
tdgbl->db_handle = NULL;
|
|
}
|
|
|
|
// Close the status output file
|
|
if (tdgbl->sw_redirect == REDIRECT && tdgbl->output_file != NULL)
|
|
{
|
|
fclose(tdgbl->output_file);
|
|
tdgbl->output_file = NULL;
|
|
}
|
|
|
|
BurpGlobals::restoreSpecific();
|
|
|
|
return exit_code;
|
|
}
|
|
|
|
|
|
|
|
void BURP_abort(Firebird::IStatus* status)
|
|
{
|
|
/**************************************
|
|
*
|
|
* B U R P _ a b o r t
|
|
*
|
|
**************************************
|
|
*
|
|
* Functional description
|
|
* Abandon a failed operation.
|
|
*
|
|
**************************************/
|
|
BurpMaster master;
|
|
BurpGlobals* tdgbl = master.get();
|
|
|
|
USHORT code = tdgbl->action && tdgbl->action->act_action == ACT_backup_fini ? 351 : 83;
|
|
// msg 351 Error closing database, but backup file is OK
|
|
// msg 83 Exiting before completion due to errors
|
|
|
|
// StatusAccessor is used only as RAII holder here
|
|
UtilSvc::StatusAccessor sa = tdgbl->uSvc->getStatusAccessor();
|
|
|
|
if (status)
|
|
BURP_print_status(true, status, code);
|
|
else
|
|
{
|
|
sa.setServiceStatus(burp_msg_fac, code, SafeArg());
|
|
|
|
if (!tdgbl->uSvc->isService())
|
|
BURP_print(true, code);
|
|
}
|
|
|
|
tdgbl->uSvc->started();
|
|
BURP_exit_local(FINI_ERROR, tdgbl);
|
|
}
|
|
|
|
void BURP_error(USHORT errcode, bool abort, const SafeArg& arg)
|
|
{
|
|
/**************************************
|
|
*
|
|
* B U R P _ e r r o r
|
|
*
|
|
**************************************
|
|
*
|
|
* Functional description
|
|
*
|
|
**************************************/
|
|
BurpMaster master;
|
|
BurpGlobals* tdgbl = master.get();
|
|
|
|
// StatusAccessor is used only as RAII holder here
|
|
UtilSvc::StatusAccessor sa = tdgbl->uSvc->getStatusAccessor();
|
|
|
|
sa.setServiceStatus(burp_msg_fac, errcode, arg);
|
|
|
|
if (!tdgbl->uSvc->isService())
|
|
{
|
|
BURP_msg_partial(true, 256); // msg 256: gbak: ERROR:
|
|
BURP_msg_put(true, errcode, arg);
|
|
}
|
|
|
|
if (abort)
|
|
BURP_abort();
|
|
else
|
|
tdgbl->uSvc->started();
|
|
}
|
|
|
|
|
|
void BURP_error(USHORT errcode, bool abort, const char* str)
|
|
{
|
|
/**************************************
|
|
*
|
|
* B U R P _ e r r o r
|
|
*
|
|
**************************************
|
|
*
|
|
* Functional description
|
|
* Format and print an error message, then punt.
|
|
*
|
|
**************************************/
|
|
|
|
BURP_error(errcode, abort, SafeArg() << str);
|
|
}
|
|
|
|
|
|
void BURP_error_redirect(Firebird::IStatus* status_vector, USHORT errcode, const SafeArg& arg)
|
|
{
|
|
/**************************************
|
|
*
|
|
* B U R P _ e r r o r _ r e d i r e c t
|
|
*
|
|
**************************************
|
|
*
|
|
* Functional description
|
|
* Issue error message. Output messages then abort.
|
|
*
|
|
**************************************/
|
|
BurpMaster master;
|
|
|
|
// StatusAccessor is used only as RAII holder here
|
|
UtilSvc::StatusAccessor sa = master.get()->uSvc->getStatusAccessor();
|
|
|
|
BURP_print_status(true, status_vector);
|
|
BURP_error(errcode, true, arg);
|
|
}
|
|
|
|
|
|
// **********************************
|
|
// B U R P _ e x i t _ l o c a l
|
|
// **********************************
|
|
// Raises an exception when the old SEH system would jump to another place.
|
|
// **********************************
|
|
void BURP_exit_local(int code, BurpGlobals* tdgbl)
|
|
{
|
|
tdgbl->exit_code = code;
|
|
if (tdgbl->burp_throw)
|
|
throw Firebird::LongJump();
|
|
}
|
|
|
|
|
|
void BURP_msg_partial(bool err, USHORT number, const SafeArg& arg)
|
|
{
|
|
/**************************************
|
|
*
|
|
* B U R P _ m s g _ p a r t i a l
|
|
*
|
|
**************************************
|
|
*
|
|
* Functional description
|
|
* Retrieve a message from the error file,
|
|
* format it, and print it without a newline.
|
|
*
|
|
**************************************/
|
|
TEXT buffer[256];
|
|
|
|
fb_msg_format(NULL, burp_msg_fac, number, sizeof(buffer), buffer, arg);
|
|
burp_output(err, "%s", buffer);
|
|
}
|
|
|
|
|
|
void BURP_msg_put(bool err, USHORT number, const SafeArg& arg)
|
|
{
|
|
/**************************************
|
|
*
|
|
* B U R P _ m s g _ p u t
|
|
*
|
|
**************************************
|
|
*
|
|
* Functional description
|
|
* Retrieve a message from the error file, format it, and print it.
|
|
*
|
|
**************************************/
|
|
TEXT buffer[256];
|
|
|
|
fb_msg_format(NULL, burp_msg_fac, number, sizeof(buffer), buffer, arg);
|
|
burp_output(err, "%s\n", buffer);
|
|
}
|
|
|
|
|
|
void BURP_msg_get(USHORT number, TEXT* output_msg, const SafeArg& arg)
|
|
{
|
|
/**************************************
|
|
*
|
|
* B U R P _ m s g _ g e t
|
|
*
|
|
**************************************
|
|
*
|
|
* Functional description
|
|
* Retrieve a message from the error file, format it and copy it to the buffer
|
|
*
|
|
**************************************/
|
|
TEXT buffer[BURP_MSG_GET_SIZE];
|
|
|
|
fb_msg_format(NULL, burp_msg_fac, number, sizeof(buffer), buffer, arg);
|
|
strcpy(output_msg, buffer);
|
|
}
|
|
|
|
void OutputVersion::callback(Firebird::CheckStatusWrapper* status, const char* text)
|
|
{
|
|
/**************************************
|
|
*
|
|
* O u t p u t V e r s i o n :: c a l l b a c k
|
|
*
|
|
**************************************
|
|
*
|
|
* Functional description
|
|
* Callback routine for access method
|
|
* printing (specifically show version)
|
|
* will accept.
|
|
*
|
|
**************************************/
|
|
|
|
burp_output(false, format, text);
|
|
}
|
|
|
|
void BURP_print(bool err, USHORT number, const SafeArg& arg)
|
|
{
|
|
/**************************************
|
|
*
|
|
* B U R P _ p r i n t
|
|
*
|
|
**************************************
|
|
*
|
|
* Functional description
|
|
* Display a formatted error message
|
|
* in a way that civilized systems
|
|
* will accept.
|
|
*
|
|
**************************************/
|
|
BurpMaster master;
|
|
|
|
BURP_msg_partial(err, 169); // msg 169: gbak:
|
|
BURP_msg_put(err, number, arg);
|
|
}
|
|
|
|
|
|
void BURP_print(bool err, USHORT number, const char* str)
|
|
{
|
|
/**************************************
|
|
*
|
|
* B U R P _ p r i n t
|
|
*
|
|
**************************************
|
|
*
|
|
* Functional description
|
|
* Display a formatted error message
|
|
* in a way that civilized systems
|
|
* will accept.
|
|
*
|
|
**************************************/
|
|
BurpMaster master;
|
|
|
|
static const SafeArg dummy;
|
|
BURP_msg_partial(err, 169, dummy); // msg 169: gbak:
|
|
BURP_msg_put(err, number, SafeArg() << str);
|
|
}
|
|
|
|
|
|
void BURP_print_status(bool err, Firebird::IStatus* status_vector, USHORT secondNumber)
|
|
{
|
|
/**************************************
|
|
*
|
|
* B U R P _ p r i n t _ s t a t u s
|
|
*
|
|
**************************************
|
|
*
|
|
* Functional description
|
|
* Print error message. Use fb_interpret
|
|
* to allow redirecting output.
|
|
*
|
|
**************************************/
|
|
if (status_vector)
|
|
{
|
|
BurpMaster master;
|
|
BurpGlobals* tdgbl = master.get();
|
|
|
|
const ISC_STATUS* vector = status_vector->getErrors();
|
|
if (err)
|
|
{
|
|
UtilSvc::StatusAccessor sa = tdgbl->uSvc->getStatusAccessor();
|
|
sa.setServiceStatus(vector);
|
|
if (secondNumber)
|
|
sa.setServiceStatus(burp_msg_fac, secondNumber, SafeArg());
|
|
tdgbl->uSvc->started();
|
|
|
|
if (tdgbl->uSvc->isService())
|
|
{
|
|
return;
|
|
}
|
|
}
|
|
|
|
SCHAR s[1024];
|
|
if (fb_interpret(s, sizeof(s), &vector))
|
|
{
|
|
BURP_msg_partial(err, 256); // msg 256: gbak: ERROR:
|
|
burp_output(err, "%s\n", s);
|
|
|
|
while (fb_interpret(s, sizeof(s), &vector))
|
|
{
|
|
BURP_msg_partial(err, 256); // msg 256: gbak: ERROR:
|
|
burp_output(err, " %s\n", s);
|
|
}
|
|
}
|
|
|
|
if (secondNumber)
|
|
{
|
|
BURP_msg_partial(err, 169); // msg 169: gbak:
|
|
BURP_msg_put(true, secondNumber, SafeArg());
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
void BURP_print_warning(Firebird::IStatus* status)
|
|
{
|
|
/**************************************
|
|
*
|
|
* B U R P _ p r i n t _ w a r n i n g
|
|
*
|
|
**************************************
|
|
*
|
|
* Functional description
|
|
* Print warning message. Use fb_interpret
|
|
* to allow redirecting output.
|
|
*
|
|
**************************************/
|
|
if (status && (status->getState() & Firebird::IStatus::STATE_WARNINGS))
|
|
{
|
|
BurpMaster master;
|
|
BurpGlobals* tdgbl = master.get();
|
|
|
|
// print the warning message
|
|
const ISC_STATUS* vector = status->getWarnings();
|
|
SCHAR s[1024];
|
|
|
|
if (fb_interpret(s, sizeof(s), &vector))
|
|
{
|
|
BURP_msg_partial(false, 255); // msg 255: gbak: WARNING:
|
|
burp_output(false, "%s\n", s);
|
|
|
|
while (fb_interpret(s, sizeof(s), &vector))
|
|
{
|
|
BURP_msg_partial(false, 255); // msg 255: gbak: WARNING:
|
|
burp_output(false, " %s\n", s);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
void BURP_verbose(USHORT number, const SafeArg& arg)
|
|
{
|
|
/**************************************
|
|
*
|
|
* B U R P _ v e r b o s e
|
|
*
|
|
**************************************
|
|
*
|
|
* Functional description
|
|
* Calls BURP_message for displaying a message for verbose output.
|
|
* If not verbose then calls yielding function.
|
|
*
|
|
**************************************/
|
|
BurpMaster master;
|
|
BurpGlobals* tdgbl = master.get();
|
|
|
|
if (tdgbl->gbl_sw_verbose)
|
|
BURP_message(number, arg, true);
|
|
else
|
|
burp_output(false, "%s", "");
|
|
}
|
|
|
|
|
|
void BURP_message(USHORT number, const MsgFormat::SafeArg& arg, bool totals)
|
|
{
|
|
/**************************************
|
|
*
|
|
* B U R P _ m e s s a g e
|
|
*
|
|
**************************************
|
|
*
|
|
* Functional description
|
|
* Calls BURP_msg for formatting & displaying a message.
|
|
*
|
|
**************************************/
|
|
BurpMaster master;
|
|
BurpGlobals* tdgbl = master.get();
|
|
|
|
if (totals)
|
|
tdgbl->print_stats_header();
|
|
BURP_msg_partial(false, 169); // msg 169: gbak:
|
|
if (totals)
|
|
tdgbl->print_stats(number);
|
|
BURP_msg_put(false, number, arg);
|
|
}
|
|
|
|
|
|
void BURP_verbose(USHORT number, const char* str)
|
|
{
|
|
/**************************************
|
|
*
|
|
* B U R P _ v e r b o s e
|
|
*
|
|
**************************************
|
|
*
|
|
* Functional description
|
|
* Shortcut for text argument
|
|
*
|
|
**************************************/
|
|
BURP_verbose(number, SafeArg() << str);
|
|
}
|
|
|
|
|
|
static void close_out_transaction(gbak_action action, Firebird::ITransaction** tPtr)
|
|
{
|
|
/**************************************
|
|
*
|
|
* c l o s e _ o u t _ t r a n s a c t i o n
|
|
*
|
|
**************************************
|
|
*
|
|
* Functional description
|
|
* Make a transaction go away. This is
|
|
* important as we need to detach from the
|
|
* database so all engine allocated memory is
|
|
* returned to the system.
|
|
*
|
|
**************************************/
|
|
if (*tPtr)
|
|
{
|
|
FbLocalStatus status_vector;
|
|
if (action == RESTORE)
|
|
{
|
|
// Even if the restore failed, commit the transaction so that
|
|
// a partial database is at least recovered.
|
|
(*tPtr)->commit(&status_vector);
|
|
if (!status_vector.isSuccess())
|
|
{
|
|
// If we can't commit - have to roll it back, as
|
|
// we need to close all outstanding transactions before
|
|
// we can detach from the database.
|
|
(*tPtr)->rollback(&status_vector);
|
|
if (!status_vector.isSuccess())
|
|
BURP_print_status(false, &status_vector);
|
|
else
|
|
*tPtr = nullptr;
|
|
}
|
|
else
|
|
*tPtr = nullptr;
|
|
}
|
|
else
|
|
{
|
|
// A backup shouldn't touch any data - we ensure that
|
|
// by never writing data during a backup, but let's double
|
|
// ensure it by doing a rollback
|
|
(*tPtr)->rollback(&status_vector);
|
|
if (!status_vector.isSuccess())
|
|
BURP_print_status(false, &status_vector);
|
|
else
|
|
*tPtr = nullptr;
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
static SLONG get_number( const SCHAR* string)
|
|
{
|
|
/**************************************
|
|
*
|
|
* g e t _ n u m b e r
|
|
*
|
|
**************************************
|
|
*
|
|
* Functional description
|
|
* Convert a string to binary, complaining bitterly if
|
|
* the string is bum.
|
|
* CVC: where does it complain? It does return zero, nothing else.
|
|
* Worse, this function doesn't check for overflow.
|
|
**************************************/
|
|
SCHAR c;
|
|
SLONG value = 0;
|
|
|
|
for (const SCHAR* p = string; c = *p++;)
|
|
{
|
|
if (c < '0' || c > '9')
|
|
return 0;
|
|
value *= 10;
|
|
value += c - '0';
|
|
}
|
|
|
|
return value;
|
|
}
|
|
|
|
|
|
static gbak_action open_files(const TEXT* file1,
|
|
const TEXT** file2,
|
|
USHORT sw_replace,
|
|
const Firebird::ClumpletWriter& dpb)
|
|
{
|
|
/**************************************
|
|
*
|
|
* o p e n _ f i l e s
|
|
*
|
|
**************************************
|
|
*
|
|
* Functional description
|
|
* from the input file names and
|
|
* positions, guess what the users
|
|
* intention was. Return the type
|
|
* of the first file, plus open file
|
|
* and db handle.
|
|
*
|
|
**************************************/
|
|
BurpGlobals* tdgbl = BurpGlobals::getSpecific();
|
|
FbLocalStatus status_vector;
|
|
|
|
// try to attach the database using the first file_name
|
|
|
|
if (sw_replace != IN_SW_BURP_C && sw_replace != IN_SW_BURP_R)
|
|
{
|
|
Firebird::DispatcherPtr provider;
|
|
|
|
// provide crypt key(s) for engine
|
|
|
|
if (tdgbl->gbl_sw_keyholder)
|
|
{
|
|
tdgbl->gbl_database_file_name = file1;
|
|
provider->setDbCryptCallback(&status_vector, MVOL_get_crypt(tdgbl));
|
|
|
|
if (!status_vector.isSuccess())
|
|
{
|
|
BURP_print_status(true, &status_vector);
|
|
return QUIT;
|
|
}
|
|
}
|
|
|
|
tdgbl->db_handle = provider->attachDatabase(&status_vector, file1,
|
|
dpb.getBufferLength(), dpb.getBuffer());
|
|
|
|
if (!(status_vector->getState() & Firebird::IStatus::STATE_ERRORS))
|
|
{
|
|
if (sw_replace != IN_SW_BURP_B)
|
|
{
|
|
// msg 13 REPLACE specified, but the first file %s is a database
|
|
BURP_error(13, true, file1);
|
|
tdgbl->db_handle->detach(&status_vector);
|
|
|
|
if (status_vector->getState() & Firebird::IStatus::STATE_ERRORS)
|
|
BURP_print_status(true, &status_vector);
|
|
else
|
|
tdgbl->db_handle = NULL;
|
|
|
|
return QUIT;
|
|
}
|
|
if (tdgbl->gbl_sw_version)
|
|
{
|
|
// msg 139 Version(s) for database "%s"
|
|
BURP_print(false, 139, file1);
|
|
OutputVersion outputVersion("\t%s\n");
|
|
Firebird::UtilInterfacePtr()->getFbVersion(&status_vector, tdgbl->db_handle, &outputVersion);
|
|
}
|
|
BURP_verbose(166, file1); // msg 166: readied database %s for backup
|
|
|
|
if (tdgbl->gbl_sw_keyholder)
|
|
{
|
|
unsigned char info[] = {fb_info_crypt_key, fb_info_crypt_plugin};
|
|
unsigned char buffer[(1 + 2 + MAX_SQL_IDENTIFIER_SIZE) * 2 + 2];
|
|
unsigned int len;
|
|
|
|
tdgbl->db_handle->getInfo(&status_vector, sizeof(info), info, sizeof(buffer), buffer);
|
|
|
|
UCHAR* p = buffer;
|
|
|
|
while (p)
|
|
{
|
|
switch (*p++)
|
|
{
|
|
case fb_info_crypt_key:
|
|
len = gds__vax_integer(p, 2);
|
|
if (len < sizeof(tdgbl->gbl_hdr_keybuffer))
|
|
{
|
|
memcpy(tdgbl->gbl_hdr_keybuffer, p + 2, len);
|
|
tdgbl->gbl_hdr_keybuffer[len] = 0;
|
|
if (!tdgbl->gbl_sw_keyname)
|
|
tdgbl->gbl_sw_keyname = tdgbl->gbl_hdr_keybuffer;
|
|
}
|
|
break;
|
|
|
|
case fb_info_crypt_plugin:
|
|
len = gds__vax_integer(p, 2);
|
|
if (len < sizeof(tdgbl->gbl_hdr_cryptbuffer))
|
|
{
|
|
memcpy(tdgbl->gbl_hdr_cryptbuffer, p + 2, len);
|
|
tdgbl->gbl_hdr_cryptbuffer[len] = 0;
|
|
if (!tdgbl->gbl_sw_crypt)
|
|
tdgbl->gbl_sw_crypt = tdgbl->gbl_hdr_cryptbuffer;
|
|
}
|
|
break;
|
|
|
|
default:
|
|
p = NULL;
|
|
continue;
|
|
}
|
|
|
|
p += (2 + len);
|
|
}
|
|
}
|
|
}
|
|
else if (sw_replace == IN_SW_BURP_B ||
|
|
(status_vector->getErrors()[1] != isc_io_error && status_vector->getErrors()[1] != isc_bad_db_format))
|
|
{
|
|
BURP_print_status(true, &status_vector);
|
|
return QUIT;
|
|
}
|
|
}
|
|
|
|
burp_fil* fil = NULL;
|
|
|
|
if (sw_replace == IN_SW_BURP_B)
|
|
{
|
|
// Now it is safe to skip a db file
|
|
tdgbl->gbl_sw_backup_files = tdgbl->gbl_sw_files->fil_next;
|
|
tdgbl->gbl_sw_files = tdgbl->gbl_sw_files->fil_next;
|
|
fb_assert(tdgbl->gbl_sw_files->fil_name == *file2);
|
|
|
|
gbak_action flag = BACKUP;
|
|
tdgbl->action->act_action = ACT_backup;
|
|
|
|
for (fil = tdgbl->gbl_sw_files; fil; fil = fil->fil_next)
|
|
{
|
|
// adjust the file size first
|
|
FB_UINT64 fsize = fil->fil_length;
|
|
switch (fil->fil_size_code)
|
|
{
|
|
case size_n:
|
|
break;
|
|
case size_k:
|
|
fsize *= KBYTE;
|
|
break;
|
|
case size_m:
|
|
fsize *= MBYTE;
|
|
break;
|
|
case size_g:
|
|
fsize *= GBYTE;
|
|
break;
|
|
case size_e:
|
|
BURP_error(262, true, fil->fil_name.c_str());
|
|
// msg 262 size specification either missing or incorrect for file %s
|
|
break;
|
|
default:
|
|
fb_assert(FALSE);
|
|
break;
|
|
}
|
|
|
|
fil->fil_length = fsize;
|
|
|
|
if ((fil->fil_seq = ++tdgbl->action->act_total) >= 2)
|
|
{
|
|
tdgbl->action->act_action = ACT_backup_split;
|
|
}
|
|
|
|
BURP_verbose(75, fil->fil_name.c_str()); // msg 75 creating file %s
|
|
|
|
if (fil->fil_name == "stdout")
|
|
{
|
|
if (tdgbl->action->act_total >= 2 || fil->fil_next ||
|
|
(tdgbl->gbl_sw_verbose && tdgbl->sw_redirect == NOREDIRECT && tdgbl->uSvc->isService()))
|
|
{
|
|
BURP_error(266, true);
|
|
// msg 266 standard output is not supported when using split operation or in verbose mode
|
|
flag = QUIT;
|
|
break;
|
|
}
|
|
|
|
// We ignore SIGPIPE so that we can report an IO error when we
|
|
// try to write to the broken pipe.
|
|
#ifndef WIN_NT
|
|
signal(SIGPIPE, SIG_IGN);
|
|
#endif
|
|
tdgbl->uSvc->setDataMode(true);
|
|
fil->fil_fd = GBAK_STDOUT_DESC();
|
|
tdgbl->stdIoMode = true;
|
|
tdgbl->gbl_sw_direct_io = false;
|
|
break;
|
|
}
|
|
else
|
|
{
|
|
Firebird::string nm = tdgbl->toSystem(fil->fil_name);
|
|
#ifdef WIN_NT
|
|
if ((fil->fil_fd = NT_tape_open(nm.c_str(), MODE_WRITE, CREATE_ALWAYS)) == INVALID_HANDLE_VALUE)
|
|
#else
|
|
const int wmode = MODE_WRITE | (tdgbl->gbl_sw_direct_io ? O_DIRECT : 0);
|
|
if ((fil->fil_fd = open(fil->fil_name.c_str(), wmode, open_mask)) == -1)
|
|
#endif // WIN_NT
|
|
{
|
|
|
|
BURP_error(65, false, fil->fil_name.c_str());
|
|
// msg 65 can't open backup file %s
|
|
flag = QUIT;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (fil->fil_length == 0)
|
|
{
|
|
if (fil->fil_next)
|
|
{
|
|
BURP_error(262, true, fil->fil_name.c_str());
|
|
// msg 262 size specification either missing or incorrect for file %s
|
|
flag = QUIT;
|
|
break;
|
|
}
|
|
else
|
|
{
|
|
fil->fil_length = MAX_LENGTH;
|
|
// Write as much as possible to the last file
|
|
}
|
|
}
|
|
if (fil->fil_length < MIN_SPLIT_SIZE)
|
|
{
|
|
BURP_error(271, true, SafeArg() << fil->fil_length << MIN_SPLIT_SIZE);
|
|
// msg file size given (%d) is less than minimum allowed (%d)
|
|
flag = QUIT;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (flag == BACKUP)
|
|
{
|
|
tdgbl->action->act_file = tdgbl->gbl_sw_files;
|
|
tdgbl->file_desc = tdgbl->gbl_sw_files->fil_fd;
|
|
}
|
|
else
|
|
{
|
|
tdgbl->db_handle->detach(&status_vector);
|
|
|
|
if (status_vector->getState() & Firebird::IStatus::STATE_ERRORS)
|
|
BURP_print_status(true, &status_vector);
|
|
else
|
|
tdgbl->db_handle = NULL;
|
|
}
|
|
|
|
return flag;
|
|
}
|
|
|
|
|
|
// If we got to here, then we're really not backing up a database,
|
|
// so open a backup file.
|
|
|
|
// There are four possible cases such as:
|
|
//
|
|
// 1. restore single backup file to single db file
|
|
// 2. restore single backup file to multiple db files
|
|
// 3. restore multiple backup files (join operation) to single db file
|
|
// 4. restore multiple backup files (join operation) to multiple db files
|
|
//
|
|
// Just looking at the command line, we can't say for sure whether it is a
|
|
// specification of the last file to be join or it is a specification of the
|
|
// primary db file (case 4), for example:
|
|
//
|
|
// gbak -c gbk1 gbk2 gbk3 db1 200 db2 500 db3 -v
|
|
// ^^^
|
|
// db1 could be either the last file to be join or primary db file
|
|
//
|
|
// Since 'gbk' and 'gsplit' formats are different (gsplit file has its own
|
|
// header record) hence we can use it as follows:
|
|
//
|
|
// - open first file
|
|
// - read & check a header record
|
|
//
|
|
// If a header is identified as a 'gsplit' one then we know exactly how
|
|
// many files need to be join and in which order. We keep opening a file by
|
|
// file till we reach the last one to be join. During this step we check
|
|
// that the files are accessible and are in proper order. It gives us
|
|
// possibility to let silly customer know about an error as soon as possible.
|
|
// Besides we have to find out which file is going to be a db file.
|
|
//
|
|
// If header is not identified as a 'gsplit' record then we assume that
|
|
// we got a single backup file.
|
|
|
|
fil = tdgbl->gbl_sw_files;
|
|
tdgbl->gbl_sw_backup_files = tdgbl->gbl_sw_files;
|
|
|
|
tdgbl->action->act_action = ACT_restore;
|
|
if (fil->fil_name == "stdin")
|
|
{
|
|
fil->fil_fd = GBAK_STDIN_DESC();
|
|
tdgbl->file_desc = fil->fil_fd;
|
|
tdgbl->stdIoMode = true;
|
|
tdgbl->gbl_sw_direct_io = false;
|
|
tdgbl->gbl_sw_files = fil->fil_next;
|
|
}
|
|
else
|
|
{
|
|
tdgbl->stdIoMode = false;
|
|
|
|
// open first file
|
|
Firebird::string nm = tdgbl->toSystem(fil->fil_name);
|
|
#ifdef WIN_NT
|
|
if ((fil->fil_fd = NT_tape_open(nm.c_str(), MODE_READ, OPEN_EXISTING)) == INVALID_HANDLE_VALUE)
|
|
#else
|
|
const int rmode = MODE_READ | (tdgbl->gbl_sw_direct_io ? O_DIRECT : 0);
|
|
if ((fil->fil_fd = os_utils::open(nm.c_str(), rmode)) == INVALID_HANDLE_VALUE)
|
|
#endif
|
|
{
|
|
BURP_error(65, true, fil->fil_name.c_str());
|
|
// msg 65 can't open backup file %s
|
|
return QUIT;
|
|
}
|
|
|
|
BURP_verbose(100, fil->fil_name.c_str());
|
|
// msg 100 opened file %s
|
|
|
|
// read and check a header record
|
|
tdgbl->action->act_file = fil;
|
|
int seq = 1;
|
|
if (MVOL_split_hdr_read())
|
|
{
|
|
tdgbl->action->act_action = ACT_restore_join;
|
|
// number of files to be join
|
|
const int total = tdgbl->action->act_total;
|
|
if (fil->fil_seq != seq || seq > total)
|
|
{
|
|
BURP_error(263, true, fil->fil_name.c_str());
|
|
// msg 263 file %s out of sequence
|
|
return QUIT;
|
|
}
|
|
|
|
for (++seq, fil = fil->fil_next; seq <= total; fil = fil->fil_next, seq++)
|
|
{
|
|
if (!fil)
|
|
{
|
|
BURP_error(264, true);
|
|
// msg 264 can't join -- one of the files missing
|
|
return QUIT;
|
|
}
|
|
if (fil->fil_name == "stdin")
|
|
{
|
|
BURP_error(265, true);
|
|
// msg 265 standard input is not supported when using join operation
|
|
return QUIT;
|
|
}
|
|
tdgbl->action->act_file = fil;
|
|
Firebird::string nm = tdgbl->toSystem(fil->fil_name);
|
|
#ifdef WIN_NT
|
|
if ((fil->fil_fd = NT_tape_open(nm.c_str(), MODE_READ, OPEN_EXISTING)) == INVALID_HANDLE_VALUE)
|
|
#else
|
|
if ((fil->fil_fd = os_utils::open(nm.c_str(), rmode)) == INVALID_HANDLE_VALUE)
|
|
#endif
|
|
{
|
|
BURP_error(65, false, fil->fil_name.c_str());
|
|
// msg 65 can't open backup file %s
|
|
return QUIT;
|
|
}
|
|
|
|
BURP_verbose(100, fil->fil_name.c_str());
|
|
// msg 100 opened file %s
|
|
|
|
if (MVOL_split_hdr_read())
|
|
{
|
|
if ((total != tdgbl->action->act_total) || (seq != fil->fil_seq) || (seq > total))
|
|
{
|
|
BURP_error(263, true, fil->fil_name.c_str());
|
|
// msg 263 file %s out of sequence
|
|
return QUIT;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
BURP_error(267, true, fil->fil_name.c_str());
|
|
// msg 267 backup file %s might be corrupt
|
|
return QUIT;
|
|
}
|
|
}
|
|
tdgbl->action->act_file = tdgbl->gbl_sw_files;
|
|
tdgbl->file_desc = tdgbl->action->act_file->fil_fd;
|
|
if ((tdgbl->gbl_sw_files = fil) == NULL)
|
|
{
|
|
BURP_error(268, true);
|
|
// msg 268 database file specification missing
|
|
return QUIT;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// Move pointer to the begining of the file. At this point we
|
|
// assume -- this is a single backup file because we were
|
|
// not able to read a split header record.
|
|
#ifdef WIN_NT
|
|
if (strnicmp(fil->fil_name.c_str(), "\\\\.\\tape", 8))
|
|
SetFilePointer(fil->fil_fd, 0, NULL, FILE_BEGIN);
|
|
else
|
|
SetTapePosition(fil->fil_fd, TAPE_REWIND, 0, 0, 0, FALSE);
|
|
#else
|
|
os_utils::lseek(fil->fil_fd, 0, SEEK_SET);
|
|
#endif
|
|
tdgbl->file_desc = fil->fil_fd;
|
|
tdgbl->gbl_sw_files = fil->fil_next;
|
|
}
|
|
}
|
|
|
|
|
|
// If we got here, we've opened a backup file, and we're
|
|
// thinking about creating or replacing a database.
|
|
|
|
*file2 = tdgbl->gbl_sw_files->fil_name.c_str();
|
|
if (tdgbl->gbl_sw_files->fil_size_code != size_n)
|
|
BURP_error(262, true, *file2);
|
|
// msg 262 size specification either missing or incorrect for file %s
|
|
|
|
if (sw_replace == IN_SW_BURP_C || sw_replace == IN_SW_BURP_R)
|
|
{
|
|
Firebird::DispatcherPtr provider;
|
|
|
|
// provide crypt key(s) for engine
|
|
|
|
if (tdgbl->gbl_sw_keyholder)
|
|
{
|
|
tdgbl->gbl_database_file_name = *file2;
|
|
provider->setDbCryptCallback(&status_vector, MVOL_get_crypt(tdgbl));
|
|
if (!status_vector.isSuccess())
|
|
{
|
|
BURP_print_status(true, &status_vector);
|
|
return QUIT;
|
|
}
|
|
}
|
|
|
|
tdgbl->db_handle = provider->attachDatabase(&status_vector, *file2,
|
|
dpb.getBufferLength(), dpb.getBuffer());
|
|
|
|
if (!(status_vector->getState() & Firebird::IStatus::STATE_ERRORS))
|
|
{
|
|
if (sw_replace == IN_SW_BURP_C)
|
|
{
|
|
tdgbl->db_handle->detach(&status_vector);
|
|
|
|
if (status_vector->getState() & Firebird::IStatus::STATE_ERRORS)
|
|
BURP_print_status(true, &status_vector);
|
|
else
|
|
tdgbl->db_handle = NULL;
|
|
|
|
BURP_error(14, true, *file2);
|
|
// msg 14 database %s already exists. To replace it, use the -R switch
|
|
}
|
|
else
|
|
{
|
|
tdgbl->db_handle->dropDatabase(&status_vector);
|
|
|
|
if (status_vector->getState() & Firebird::IStatus::STATE_ERRORS)
|
|
{
|
|
Firebird::FbLocalStatus status2;
|
|
tdgbl->db_handle->detach(&status2);
|
|
|
|
if (status2->getState() & Firebird::IStatus::STATE_ERRORS)
|
|
BURP_print_status(true, &status2);
|
|
else
|
|
tdgbl->db_handle = NULL;
|
|
|
|
BURP_error(233, true, *file2);
|
|
// msg 233 Cannot drop database %s, might be in use
|
|
}
|
|
else
|
|
tdgbl->db_handle = NULL;
|
|
}
|
|
}
|
|
else if (sw_replace == IN_SW_BURP_R && status_vector->getErrors()[1] == isc_adm_task_denied)
|
|
{
|
|
// if we got an error from attach database and we have replace switch set
|
|
// then look for error from attach returned due to not owner, if we are
|
|
// not owner then return the error status back up
|
|
|
|
BURP_error(274, true);
|
|
// msg # 274 : Cannot restore over current database, must be sysdba
|
|
// or owner of the existing database.
|
|
}
|
|
}
|
|
|
|
// check the file size specification
|
|
for (fil = tdgbl->gbl_sw_files; fil; fil = fil->fil_next)
|
|
{
|
|
if (fil->fil_size_code != size_n)
|
|
{
|
|
BURP_error(262, true, fil->fil_name.c_str());
|
|
// msg 262 size specification either missing or incorrect for file %s
|
|
}
|
|
}
|
|
|
|
return RESTORE;
|
|
}
|
|
|
|
|
|
static void burp_output(bool err, const SCHAR* format, ...)
|
|
{
|
|
/**************************************
|
|
*
|
|
* b u r p _ o u t p u t
|
|
*
|
|
**************************************
|
|
*
|
|
* Functional description
|
|
* Platform independent output routine.
|
|
*
|
|
**************************************/
|
|
va_list arglist;
|
|
|
|
BurpMaster master;
|
|
BurpGlobals* tdgbl = master.get();
|
|
|
|
if (tdgbl->sw_redirect != NOOUTPUT && format[0] != '\0')
|
|
{
|
|
va_start(arglist, format);
|
|
if (tdgbl->sw_redirect == REDIRECT && tdgbl->output_file != NULL)
|
|
{
|
|
vfprintf(tdgbl->output_file, format, arglist);
|
|
}
|
|
else
|
|
{
|
|
Firebird::string buf;
|
|
buf.vprintf(format, arglist);
|
|
if (err)
|
|
tdgbl->uSvc->outputError(buf.c_str());
|
|
else
|
|
tdgbl->uSvc->outputVerbose(buf.c_str());
|
|
}
|
|
va_end(arglist);
|
|
}
|
|
}
|
|
|
|
|
|
static void burp_usage(const Switches& switches)
|
|
{
|
|
/**********************************************
|
|
*
|
|
* b u r p _ u s a g e
|
|
*
|
|
**********************************************
|
|
*
|
|
* Functional description
|
|
* print usage information
|
|
*
|
|
**********************************************/
|
|
const SafeArg sa(SafeArg() << switch_char);
|
|
const SafeArg dummy;
|
|
|
|
BURP_print(true, 317); // usage
|
|
for (int i = 318; i < 323; ++i)
|
|
BURP_msg_put(true, i, dummy); // usage
|
|
|
|
BURP_print(true, 95); // msg 95 legal switches are
|
|
const Switches::in_sw_tab_t* const base = switches.getTable();
|
|
for (const Switches::in_sw_tab_t* p = base; p->in_sw; ++p)
|
|
{
|
|
if (p->in_sw_msg && p->in_sw_optype == boMain)
|
|
BURP_msg_put(true, p->in_sw_msg, sa);
|
|
}
|
|
|
|
BURP_print(true, 323); // backup options are
|
|
for (const Switches::in_sw_tab_t* p = base; p->in_sw; ++p)
|
|
{
|
|
if (p->in_sw_msg && p->in_sw_optype == boBackup)
|
|
BURP_msg_put(true, p->in_sw_msg, sa);
|
|
}
|
|
|
|
BURP_print(true, 324); // restore options are
|
|
for (const Switches::in_sw_tab_t* p = base; p->in_sw; ++p)
|
|
{
|
|
if (p->in_sw_msg && p->in_sw_optype == boRestore)
|
|
BURP_msg_put(true, p->in_sw_msg, sa);
|
|
}
|
|
|
|
BURP_print(true, 325); // general options are
|
|
for (const Switches::in_sw_tab_t* p = base; p->in_sw; ++p)
|
|
{
|
|
if (p->in_sw_msg && p->in_sw_optype == boGeneral)
|
|
BURP_msg_put(true, p->in_sw_msg, sa);
|
|
}
|
|
|
|
BURP_print(true, 132); // msg 132 switches can be abbreviated to the unparenthesized characters
|
|
}
|
|
|
|
|
|
static ULONG get_size(const SCHAR* string, burp_fil* file)
|
|
{
|
|
/**********************************************
|
|
*
|
|
* g e t _ s i z e
|
|
*
|
|
**********************************************
|
|
*
|
|
* Functional description
|
|
* Get size specification for either splitting or
|
|
* restoring to multiple files
|
|
*
|
|
**********************************************/
|
|
const FB_UINT64 overflow = MAX_UINT64 / 10 - 1;
|
|
UCHAR c;
|
|
FB_UINT64 size = 0;
|
|
bool digit = false;
|
|
|
|
file->fil_size_code = size_n;
|
|
for (const SCHAR *num = string; c = *num++;)
|
|
{
|
|
if (isdigit(c))
|
|
{
|
|
int val = c - '0';
|
|
if (size >= overflow)
|
|
{
|
|
file->fil_size_code = size_e;
|
|
size = 0;
|
|
break;
|
|
}
|
|
size = size * 10 + val;
|
|
digit = true;
|
|
}
|
|
else
|
|
{
|
|
if (isalpha(c))
|
|
{
|
|
if (!digit)
|
|
{
|
|
file->fil_size_code = size_e;
|
|
size = 0;
|
|
break;
|
|
}
|
|
|
|
switch (UPPER(c))
|
|
{
|
|
case 'K':
|
|
file->fil_size_code = size_k;
|
|
break;
|
|
case 'M':
|
|
file->fil_size_code = size_m;
|
|
break;
|
|
case 'G':
|
|
file->fil_size_code = size_g;
|
|
break;
|
|
default:
|
|
file->fil_size_code = size_e;
|
|
size = 0;
|
|
break;
|
|
}
|
|
if (*num)
|
|
{
|
|
file->fil_size_code = size_e;
|
|
size = 0;
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
file->fil_length = size;
|
|
return size;
|
|
}
|
|
|
|
|
|
#ifndef WIN_NT
|
|
void close_platf(DESC file)
|
|
{
|
|
/**********************************************
|
|
*
|
|
* c l o s e _ p l a t f
|
|
*
|
|
**********************************************
|
|
*
|
|
* Functional description
|
|
* Truncate and close file - posix version
|
|
*
|
|
**********************************************/
|
|
if (sizeof(off_t) > 4) // 64 bit or b4-bit IO in 32 bit OS
|
|
{
|
|
#ifndef O_ACCMODE
|
|
// Suppose compatibility with KR where 0, 1, 2 were documented for RO, WO and RW in open() still exists
|
|
#define O_ACCMODE 3
|
|
#endif
|
|
|
|
off_t fileSize = os_utils::lseek(file, 0, SEEK_CUR);
|
|
if (fileSize != (off_t)(-1))
|
|
{
|
|
FB_UNUSED(os_utils::ftruncate(file, fileSize));
|
|
}
|
|
}
|
|
|
|
close(file);
|
|
}
|
|
#endif // WIN_NT
|
|
|
|
|
|
void BurpGlobals::setupSkipData(const Firebird::string& regexp)
|
|
{
|
|
if (skipDataMatcher)
|
|
{
|
|
BURP_error(356, true);
|
|
// msg 356 regular expression to skip tables was already set
|
|
}
|
|
|
|
// Compile skip relation expressions
|
|
try
|
|
{
|
|
if (regexp.hasData())
|
|
{
|
|
Firebird::string filter(regexp);
|
|
if (!uSvc->utf8FileNames())
|
|
ISC_systemToUtf8(filter);
|
|
|
|
BurpGlobals* tdgbl = BurpGlobals::getSpecific();
|
|
|
|
skipDataMatcher.reset(FB_NEW_POOL(tdgbl->getPool()) Firebird::SimilarToRegex(
|
|
tdgbl->getPool(), Firebird::SimilarToFlag::CASE_INSENSITIVE,
|
|
filter.c_str(), filter.length(),
|
|
"\\", 1));
|
|
}
|
|
}
|
|
catch (const Firebird::Exception&)
|
|
{
|
|
Firebird::fatal_exception::raiseFmt(
|
|
"error while compiling regular expression \"%s\"", regexp.c_str());
|
|
}
|
|
}
|
|
|
|
void BurpGlobals::setupIncludeData(const Firebird::string& regexp)
|
|
{
|
|
if (includeDataMatcher)
|
|
{
|
|
BURP_error(390, true);
|
|
// msg 390 regular expression to include tables was already set
|
|
}
|
|
|
|
// Compile include relation expressions
|
|
try
|
|
{
|
|
if (regexp.hasData())
|
|
{
|
|
Firebird::string filter(regexp);
|
|
if (!uSvc->utf8FileNames())
|
|
ISC_systemToUtf8(filter);
|
|
|
|
BurpGlobals* tdgbl = BurpGlobals::getSpecific();
|
|
|
|
includeDataMatcher.reset(FB_NEW_POOL(tdgbl->getPool()) Firebird::SimilarToRegex(
|
|
tdgbl->getPool(), Firebird::SimilarToFlag::CASE_INSENSITIVE,
|
|
filter.c_str(), filter.length(),
|
|
"\\", 1));
|
|
}
|
|
}
|
|
catch (const Firebird::Exception&)
|
|
{
|
|
Firebird::fatal_exception::raiseFmt(
|
|
"error while compiling regular expression \"%s\"", regexp.c_str());
|
|
}
|
|
}
|
|
|
|
Firebird::string BurpGlobals::toSystem(const Firebird::PathName& from)
|
|
{
|
|
Firebird::string to = from.ToString();
|
|
if (uSvc->utf8FileNames())
|
|
ISC_utf8ToSystem(to);
|
|
return to;
|
|
}
|
|
|
|
namespace // for local symbols
|
|
{
|
|
enum Pattern { NOT_SET = 0, MATCH = 1, NOT_MATCH = 2 };
|
|
|
|
Pattern checkPattern(Firebird::AutoPtr<Firebird::SimilarToRegex>& matcher,
|
|
const char* name)
|
|
{
|
|
if (!matcher)
|
|
return NOT_SET;
|
|
|
|
return matcher->matches(name, strlen(name))? MATCH : NOT_MATCH;
|
|
}
|
|
}
|
|
|
|
bool BurpGlobals::skipRelation(const char* name)
|
|
{
|
|
if (gbl_sw_meta)
|
|
return true;
|
|
|
|
// Fine-grained table controlling cases when data must be skipped for a table
|
|
static const bool result[3][3] = {
|
|
// Include filter
|
|
// NS M NM S
|
|
{ false, false, true}, // NS k
|
|
{ true, true, true}, // M i
|
|
{ false, false, true} // NM p
|
|
};
|
|
|
|
const enum Pattern res1 = checkPattern(skipDataMatcher, name);
|
|
const enum Pattern res2 = checkPattern(includeDataMatcher, name);
|
|
|
|
return result[res1][res2];
|
|
}
|
|
|
|
void BurpGlobals::read_stats(SINT64* stats)
|
|
{
|
|
if (!db_handle)
|
|
return;
|
|
|
|
const UCHAR info[] =
|
|
{
|
|
isc_info_reads,
|
|
isc_info_writes
|
|
};
|
|
|
|
FbLocalStatus status;
|
|
UCHAR buffer[sizeof(info) * (1 + 2 + 8) + 2];
|
|
|
|
db_handle->getInfo(&status, sizeof(info), info, sizeof(buffer), buffer);
|
|
|
|
UCHAR* p = buffer, *const e = buffer + sizeof(buffer);
|
|
while (p < e)
|
|
{
|
|
int flag = -1;
|
|
switch (*p)
|
|
{
|
|
case isc_info_reads:
|
|
flag = READS;
|
|
break;
|
|
|
|
case isc_info_writes:
|
|
flag = WRITES;
|
|
break;
|
|
|
|
case isc_info_end:
|
|
default:
|
|
p = e;
|
|
}
|
|
|
|
if (flag != -1)
|
|
{
|
|
const int len = gds__vax_integer(p + 1, 2);
|
|
stats[flag] = isc_portable_integer((ISC_UCHAR*) p + 1 + 2, len);
|
|
p += len + 3;
|
|
}
|
|
}
|
|
}
|
|
|
|
void BurpGlobals::print_stats(USHORT number)
|
|
{
|
|
if (!gbl_stat_flags || gbl_stat_done)
|
|
return;
|
|
|
|
const bool total = (number == 369);
|
|
// msg 369 total statistics
|
|
|
|
burp_output(false, " ");
|
|
|
|
const int time_mask = (1 << TIME_TOTAL) | (1 << TIME_DELTA);
|
|
if (gbl_stat_flags & time_mask)
|
|
{
|
|
const SINT64 t0 = fb_utils::query_performance_counter();
|
|
const SINT64 freq_ms = fb_utils::query_performance_frequency() / 1000;
|
|
|
|
if (gbl_stat_flags & (1 << TIME_TOTAL))
|
|
{
|
|
SINT64 t1 = (t0 - gbl_stats[TIME_TOTAL]) / freq_ms;
|
|
burp_output(false, STAT_FORMATS[TIME_TOTAL].format, (int)(t1 / 1000), (int)(t1 % 1000));
|
|
}
|
|
|
|
if (gbl_stat_flags & (1 << TIME_DELTA))
|
|
{
|
|
SINT64 t2 = (t0 - gbl_stats[TIME_DELTA]) / freq_ms;
|
|
burp_output(false, STAT_FORMATS[TIME_DELTA].format, (int)(t2 / 1000), (int)(t2 % 1000));
|
|
|
|
gbl_stats[TIME_DELTA] = t0;
|
|
}
|
|
}
|
|
|
|
SINT64 cur_stats[LAST_COUNTER] = {0};
|
|
if ((gbl_stat_flags & ~time_mask) && !gbl_stat_done)
|
|
read_stats(cur_stats);
|
|
|
|
for (int i = READS; i < LAST_COUNTER; i++)
|
|
{
|
|
if (gbl_stat_flags & (1 << i))
|
|
{
|
|
SINT64 val = 0;
|
|
if (total || gbl_stat_done)
|
|
val = cur_stats[i];
|
|
else
|
|
val = cur_stats[i] - gbl_stats[i];
|
|
|
|
gbl_stats[i] = cur_stats[i];
|
|
|
|
burp_output(false, STAT_FORMATS[i].format, val);
|
|
}
|
|
}
|
|
|
|
if (total)
|
|
gbl_stat_done = true;
|
|
}
|
|
|
|
void BurpGlobals::print_stats_header()
|
|
{
|
|
if (gbl_stat_header || !gbl_stat_flags)
|
|
return;
|
|
|
|
gbl_stat_header = true;
|
|
|
|
BURP_msg_partial(false, 169); // msg 169: gbak:
|
|
burp_output(false, " ");
|
|
|
|
for (int i = 0; i < LAST_COUNTER; i++)
|
|
{
|
|
if (gbl_stat_flags & (1 << i))
|
|
burp_output(false, "%-*s", STAT_FORMATS[i].width, STAT_FORMATS[i].header);
|
|
}
|
|
|
|
burp_output(false, "\n");
|
|
}
|
|
|
|
void BURP_makeSymbol(BurpGlobals* tdgbl, Firebird::string& name) // add double quotes to string
|
|
{
|
|
if (tdgbl->gbl_dialect < SQL_DIALECT_V6)
|
|
return;
|
|
|
|
const char dq = '"';
|
|
for (unsigned p = 0; p < name.length(); ++p)
|
|
{
|
|
if (name[p] == dq)
|
|
{
|
|
name.insert(p, 1, dq);
|
|
++p;
|
|
}
|
|
}
|
|
name.insert(0u, 1, dq);
|
|
name += dq;
|
|
}
|
|
|
|
static void processFetchPass(const SCHAR*& password, int& itr, const int argc, Firebird::UtilSvc::ArgvType& argv)
|
|
{
|
|
if (++itr >= argc)
|
|
{
|
|
BURP_error(189, true);
|
|
// password parameter missing
|
|
}
|
|
if (password)
|
|
{
|
|
BURP_error(307, true);
|
|
// too many passwords provided
|
|
}
|
|
|
|
switch (fb_utils::fetchPassword(argv[itr], password))
|
|
{
|
|
case fb_utils::FETCH_PASS_OK:
|
|
break;
|
|
case fb_utils::FETCH_PASS_FILE_OPEN_ERROR:
|
|
BURP_error(308, true, MsgFormat::SafeArg() << argv[itr] << errno);
|
|
// error @2 opening password file @1
|
|
break;
|
|
case fb_utils::FETCH_PASS_FILE_READ_ERROR:
|
|
BURP_error(309, true, MsgFormat::SafeArg() << argv[itr] << errno);
|
|
// error @2 reading password file @1
|
|
break;
|
|
case fb_utils::FETCH_PASS_FILE_EMPTY:
|
|
BURP_error(310, true, MsgFormat::SafeArg() << argv[itr]);
|
|
// password file @1 is empty
|
|
break;
|
|
}
|
|
}
|