1
0
mirror of https://github.com/postgres/postgres.git synced 2025-10-25 13:17:41 +03:00

Move some pg_dump function around.

Move functions used only by pg_dump and pg_restore from dumputils.c to a new
file, pg_backup_utils.c. dumputils.c is linked into psql and some programs
in bin/scripts, so it seems good to keep it slim. The parallel functionality
is moved to parallel.c, as is exit_horribly, because the interesting code in
exit_horribly is parallel-related.

This refactoring gets rid of the on_exit_msg_func function pointer. It was
problematic, because a modern gcc version with -Wmissing-format-attribute
complained if it wasn't marked with PF_PRINTF_ATTRIBUTE, but the ancient gcc
version that Tom Lane's old HP-UX box has didn't accept that attribute on a
function pointer, and gave an error. We still use a similar function pointer
trick for getLocalPQBuffer() function, to use a thread-local version of that
in parallel mode on Windows, but that dodges the problem because it doesn't
take printf-like arguments.
This commit is contained in:
Heikki Linnakangas
2013-03-27 18:10:40 +02:00
parent 1cea9bbb21
commit 7800a71291
19 changed files with 336 additions and 247 deletions

View File

@@ -19,8 +19,8 @@ include $(top_builddir)/src/Makefile.global
override CPPFLAGS := -I$(libpq_srcdir) $(CPPFLAGS) override CPPFLAGS := -I$(libpq_srcdir) $(CPPFLAGS)
OBJS= pg_backup_archiver.o pg_backup_db.o pg_backup_custom.o \ OBJS= pg_backup_archiver.o pg_backup_db.o pg_backup_custom.o \
pg_backup_null.o pg_backup_tar.o parallel.o \ pg_backup_null.o pg_backup_tar.o pg_backup_directory.o \
pg_backup_directory.o dumputils.o compress_io.o $(WIN32RES) pg_backup_utils.o parallel.o compress_io.o dumputils.o $(WIN32RES)
KEYWRDOBJS = keywords.o kwlookup.o KEYWRDOBJS = keywords.o kwlookup.o

View File

@@ -14,6 +14,7 @@
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
#include "pg_backup_archiver.h" #include "pg_backup_archiver.h"
#include "pg_backup_utils.h"
#include <ctype.h> #include <ctype.h>

View File

@@ -53,7 +53,7 @@
*/ */
#include "compress_io.h" #include "compress_io.h"
#include "dumputils.h" #include "pg_backup_utils.h"
#include "parallel.h" #include "parallel.h"
/*---------------------- /*----------------------

View File

@@ -25,21 +25,6 @@
extern const ScanKeyword FEScanKeywords[]; extern const ScanKeyword FEScanKeywords[];
extern const int NumFEScanKeywords; extern const int NumFEScanKeywords;
/* Globals exported by this file */
int quote_all_identifiers = 0;
const char *progname = NULL;
#define MAX_ON_EXIT_NICELY 20
static struct
{
on_exit_nicely_callback function;
void *arg;
} on_exit_nicely_list[MAX_ON_EXIT_NICELY];
static int on_exit_nicely_index;
void (*on_exit_msg_func) (const char *modulename, const char *fmt, va_list ap) = vwrite_msg;
#define supports_grant_options(version) ((version) >= 70400) #define supports_grant_options(version) ((version) >= 70400)
static bool parseAclItem(const char *item, const char *type, static bool parseAclItem(const char *item, const char *type,
@@ -49,68 +34,24 @@ static bool parseAclItem(const char *item, const char *type,
static char *copyAclUserName(PQExpBuffer output, char *input); static char *copyAclUserName(PQExpBuffer output, char *input);
static void AddAcl(PQExpBuffer aclbuf, const char *keyword, static void AddAcl(PQExpBuffer aclbuf, const char *keyword,
const char *subname); const char *subname);
static PQExpBuffer getThreadLocalPQExpBuffer(void); static PQExpBuffer defaultGetLocalPQExpBuffer(void);
#ifdef WIN32 /* Globals exported by this file */
static void shutdown_parallel_dump_utils(int code, void *unused); int quote_all_identifiers = 0;
static bool parallel_init_done = false; PQExpBuffer (*getLocalPQExpBuffer) (void) = defaultGetLocalPQExpBuffer;
static DWORD tls_index;
static DWORD mainThreadId;
static void
shutdown_parallel_dump_utils(int code, void *unused)
{
/* Call the cleanup function only from the main thread */
if (mainThreadId == GetCurrentThreadId())
WSACleanup();
}
#endif
void
init_parallel_dump_utils(void)
{
#ifdef WIN32
if (!parallel_init_done)
{
WSADATA wsaData;
int err;
tls_index = TlsAlloc();
mainThreadId = GetCurrentThreadId();
err = WSAStartup(MAKEWORD(2, 2), &wsaData);
if (err != 0)
{
fprintf(stderr, _("WSAStartup failed: %d\n"), err);
exit_nicely(1);
}
on_exit_nicely(shutdown_parallel_dump_utils, NULL);
parallel_init_done = true;
}
#endif
}
/* /*
* Non-reentrant but reduces memory leakage. (On Windows the memory leakage * Returns a temporary PQExpBuffer, valid until the next call to the function.
* will be one buffer per thread, which is at least better than one per call). * This is used by fmtId and fmtQualifiedId.
*
* Non-reentrant and non-thread-safe but reduces memory leakage. You can
* replace this with a custom version by setting the getLocalPQExpBuffer
* function pointer.
*/ */
static PQExpBuffer static PQExpBuffer
getThreadLocalPQExpBuffer(void) defaultGetLocalPQExpBuffer(void)
{ {
/* static PQExpBuffer id_return = NULL;
* The Tls code goes awry if we use a static var, so we provide for both
* static and auto, and omit any use of the static var when using Tls.
*/
static PQExpBuffer s_id_return = NULL;
PQExpBuffer id_return;
#ifdef WIN32
if (parallel_init_done)
id_return = (PQExpBuffer) TlsGetValue(tls_index); /* 0 when not set */
else
id_return = s_id_return;
#else
id_return = s_id_return;
#endif
if (id_return) /* first time through? */ if (id_return) /* first time through? */
{ {
@@ -121,15 +62,6 @@ getThreadLocalPQExpBuffer(void)
{ {
/* new buffer */ /* new buffer */
id_return = createPQExpBuffer(); id_return = createPQExpBuffer();
#ifdef WIN32
if (parallel_init_done)
TlsSetValue(tls_index, id_return);
else
s_id_return = id_return;
#else
s_id_return = id_return;
#endif
} }
return id_return; return id_return;
@@ -144,7 +76,7 @@ getThreadLocalPQExpBuffer(void)
const char * const char *
fmtId(const char *rawid) fmtId(const char *rawid)
{ {
PQExpBuffer id_return = getThreadLocalPQExpBuffer(); PQExpBuffer id_return = getLocalPQExpBuffer();
const char *cp; const char *cp;
bool need_quotes = false; bool need_quotes = false;
@@ -238,7 +170,7 @@ fmtQualifiedId(int remoteVersion, const char *schema, const char *id)
} }
appendPQExpBuffer(lcl_pqexp, "%s", fmtId(id)); appendPQExpBuffer(lcl_pqexp, "%s", fmtId(id));
id_return = getThreadLocalPQExpBuffer(); id_return = getLocalPQExpBuffer();
appendPQExpBuffer(id_return, "%s", lcl_pqexp->data); appendPQExpBuffer(id_return, "%s", lcl_pqexp->data);
destroyPQExpBuffer(lcl_pqexp); destroyPQExpBuffer(lcl_pqexp);
@@ -1278,118 +1210,6 @@ emitShSecLabels(PGconn *conn, PGresult *res, PQExpBuffer buffer,
} }
/*
* Parse a --section=foo command line argument.
*
* Set or update the bitmask in *dumpSections according to arg.
* dumpSections is initialised as DUMP_UNSECTIONED by pg_dump and
* pg_restore so they can know if this has even been called.
*/
void
set_dump_section(const char *arg, int *dumpSections)
{
/* if this is the first call, clear all the bits */
if (*dumpSections == DUMP_UNSECTIONED)
*dumpSections = 0;
if (strcmp(arg, "pre-data") == 0)
*dumpSections |= DUMP_PRE_DATA;
else if (strcmp(arg, "data") == 0)
*dumpSections |= DUMP_DATA;
else if (strcmp(arg, "post-data") == 0)
*dumpSections |= DUMP_POST_DATA;
else
{
fprintf(stderr, _("%s: unrecognized section name: \"%s\"\n"),
progname, arg);
fprintf(stderr, _("Try \"%s --help\" for more information.\n"),
progname);
exit_nicely(1);
}
}
/*
* Write a printf-style message to stderr.
*
* The program name is prepended, if "progname" has been set.
* Also, if modulename isn't NULL, that's included too.
* Note that we'll try to translate the modulename and the fmt string.
*/
void
write_msg(const char *modulename, const char *fmt,...)
{
va_list ap;
va_start(ap, fmt);
vwrite_msg(modulename, fmt, ap);
va_end(ap);
}
/*
* As write_msg, but pass a va_list not variable arguments.
*/
void
vwrite_msg(const char *modulename, const char *fmt, va_list ap)
{
if (progname)
{
if (modulename)
fprintf(stderr, "%s: [%s] ", progname, _(modulename));
else
fprintf(stderr, "%s: ", progname);
}
vfprintf(stderr, _(fmt), ap);
}
/*
* Fail and die, with a message to stderr. Parameters as for write_msg.
*/
void
exit_horribly(const char *modulename, const char *fmt,...)
{
va_list ap;
va_start(ap, fmt);
on_exit_msg_func(modulename, fmt, ap);
va_end(ap);
exit_nicely(1);
}
/* Register a callback to be run when exit_nicely is invoked. */
void
on_exit_nicely(on_exit_nicely_callback function, void *arg)
{
if (on_exit_nicely_index >= MAX_ON_EXIT_NICELY)
exit_horribly(NULL, "out of on_exit_nicely slots\n");
on_exit_nicely_list[on_exit_nicely_index].function = function;
on_exit_nicely_list[on_exit_nicely_index].arg = arg;
on_exit_nicely_index++;
}
/*
* Run accumulated on_exit_nicely callbacks in reverse order and then exit
* quietly. This needs to be thread-safe.
*/
void
exit_nicely(int code)
{
int i;
for (i = on_exit_nicely_index - 1; i >= 0; i--)
(*on_exit_nicely_list[i].function) (code,
on_exit_nicely_list[i].arg);
#ifdef WIN32
if (parallel_init_done && GetCurrentThreadId() != mainThreadId)
ExitThread(code);
#endif
exit(code);
}
void void
simple_string_list_append(SimpleStringList *list, const char *val) simple_string_list_append(SimpleStringList *list, const char *val)
{ {

View File

@@ -19,14 +19,6 @@
#include "libpq-fe.h" #include "libpq-fe.h"
#include "pqexpbuffer.h" #include "pqexpbuffer.h"
typedef enum /* bits returned by set_dump_section */
{
DUMP_PRE_DATA = 0x01,
DUMP_DATA = 0x02,
DUMP_POST_DATA = 0x04,
DUMP_UNSECTIONED = 0xff
} DumpSections;
typedef struct SimpleStringListCell typedef struct SimpleStringListCell
{ {
struct SimpleStringListCell *next; struct SimpleStringListCell *next;
@@ -40,14 +32,9 @@ typedef struct SimpleStringList
} SimpleStringList; } SimpleStringList;
typedef void (*on_exit_nicely_callback) (int code, void *arg);
extern int quote_all_identifiers; extern int quote_all_identifiers;
extern const char *progname; extern PQExpBuffer (*getLocalPQExpBuffer) (void);
extern void (*on_exit_msg_func) (const char *modulename, const char *fmt, va_list ap)
__attribute__((format(PG_PRINTF_ATTRIBUTE, 2, 0)));
extern void init_parallel_dump_utils(void);
extern const char *fmtId(const char *identifier); extern const char *fmtId(const char *identifier);
extern const char *fmtQualifiedId(int remoteVersion, extern const char *fmtQualifiedId(int remoteVersion,
const char *schema, const char *id); const char *schema, const char *id);
@@ -79,17 +66,6 @@ extern void buildShSecLabelQuery(PGconn *conn, const char *catalog_name,
extern void emitShSecLabels(PGconn *conn, PGresult *res, extern void emitShSecLabels(PGconn *conn, PGresult *res,
PQExpBuffer buffer, const char *target, const char *objname); PQExpBuffer buffer, const char *target, const char *objname);
extern void set_dump_section(const char *arg, int *dumpSections); extern void set_dump_section(const char *arg, int *dumpSections);
extern void
write_msg(const char *modulename, const char *fmt,...)
__attribute__((format(PG_PRINTF_ATTRIBUTE, 2, 3)));
extern void
vwrite_msg(const char *modulename, const char *fmt, va_list ap)
__attribute__((format(PG_PRINTF_ATTRIBUTE, 2, 0)));
extern void
exit_horribly(const char *modulename, const char *fmt,...)
__attribute__((format(PG_PRINTF_ATTRIBUTE, 2, 3), noreturn));
extern void on_exit_nicely(on_exit_nicely_callback function, void *arg);
extern void exit_nicely(int code) __attribute__((noreturn));
extern void simple_string_list_append(SimpleStringList *list, const char *val); extern void simple_string_list_append(SimpleStringList *list, const char *val);
extern bool simple_string_list_member(SimpleStringList *list, const char *val); extern bool simple_string_list_member(SimpleStringList *list, const char *val);

View File

@@ -16,9 +16,9 @@
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
#include "pg_backup_db.h" #include "postgres_fe.h"
#include "dumputils.h" #include "pg_backup_utils.h"
#include "parallel.h" #include "parallel.h"
#ifndef WIN32 #ifndef WIN32
@@ -78,10 +78,6 @@ static const char *modulename = gettext_noop("parallel archiver");
static ParallelSlot *GetMyPSlot(ParallelState *pstate); static ParallelSlot *GetMyPSlot(ParallelState *pstate);
static void static void
parallel_exit_msg_func(const char *modulename,
const char *fmt, va_list ap)
__attribute__((format(PG_PRINTF_ATTRIBUTE, 2, 0)));
static void
parallel_msg_master(ParallelSlot *slot, const char *modulename, parallel_msg_master(ParallelSlot *slot, const char *modulename,
const char *fmt, va_list ap) const char *fmt, va_list ap)
__attribute__((format(PG_PRINTF_ATTRIBUTE, 3, 0))); __attribute__((format(PG_PRINTF_ATTRIBUTE, 3, 0)));
@@ -112,6 +108,47 @@ static char *readMessageFromPipe(int fd);
#define messageEquals(msg, pattern) \ #define messageEquals(msg, pattern) \
(strcmp(msg, pattern) == 0) (strcmp(msg, pattern) == 0)
#ifdef WIN32
static void shutdown_parallel_dump_utils(int code, void *unused);
bool parallel_init_done = false;
static DWORD tls_index;
DWORD mainThreadId;
#endif
#ifdef WIN32
static void
shutdown_parallel_dump_utils(int code, void *unused)
{
/* Call the cleanup function only from the main thread */
if (mainThreadId == GetCurrentThreadId())
WSACleanup();
}
#endif
void
init_parallel_dump_utils(void)
{
#ifdef WIN32
if (!parallel_init_done)
{
WSADATA wsaData;
int err;
tls_index = TlsAlloc();
mainThreadId = GetCurrentThreadId();
err = WSAStartup(MAKEWORD(2, 2), &wsaData);
if (err != 0)
{
fprintf(stderr, _("WSAStartup failed: %d\n"), err);
exit_nicely(1);
}
on_exit_nicely(shutdown_parallel_dump_utils, NULL);
parallel_init_done = true;
}
#endif
}
static ParallelSlot * static ParallelSlot *
GetMyPSlot(ParallelState *pstate) GetMyPSlot(ParallelState *pstate)
{ {
@@ -129,21 +166,31 @@ GetMyPSlot(ParallelState *pstate)
} }
/* /*
* This is the function that will be called from exit_horribly() to print the * Fail and die, with a message to stderr. Parameters as for write_msg.
* error message. If the worker process does exit_horribly(), we forward its *
* This is defined in parallel.c, because in parallel mode, things are more
* complicated. If the worker process does exit_horribly(), we forward its
* last words to the master process. The master process then does * last words to the master process. The master process then does
* exit_horribly() with this error message itself and prints it normally. * exit_horribly() with this error message itself and prints it normally.
* After printing the message, exit_horribly() on the master will shut down * After printing the message, exit_horribly() on the master will shut down
* the remaining worker processes. * the remaining worker processes.
*/ */
static void void
parallel_exit_msg_func(const char *modulename, const char *fmt, va_list ap) exit_horribly(const char *modulename, const char *fmt,...)
{ {
va_list ap;
ParallelState *pstate = shutdown_info.pstate; ParallelState *pstate = shutdown_info.pstate;
ParallelSlot *slot; ParallelSlot *slot;
Assert(pstate); va_start(ap, fmt);
if (pstate == NULL)
{
/* Not in parallel mode, just write to stderr */
vwrite_msg(modulename, fmt, ap);
}
else
{
slot = GetMyPSlot(pstate); slot = GetMyPSlot(pstate);
if (!slot) if (!slot)
@@ -152,6 +199,11 @@ parallel_exit_msg_func(const char *modulename, const char *fmt, va_list ap)
else else
/* If we're a worker process, send the msg to the master process */ /* If we're a worker process, send the msg to the master process */
parallel_msg_master(slot, modulename, fmt, ap); parallel_msg_master(slot, modulename, fmt, ap);
}
va_end(ap);
exit_nicely(1);
} }
/* Sends the error message from the worker to the master process */ /* Sends the error message from the worker to the master process */
@@ -172,6 +224,54 @@ parallel_msg_master(ParallelSlot *slot, const char *modulename,
sendMessageToMaster(pipefd, buf); sendMessageToMaster(pipefd, buf);
} }
/*
* A thread-local version of getLocalPQExpBuffer().
*
* Non-reentrant but reduces memory leakage. (On Windows the memory leakage
* will be one buffer per thread, which is at least better than one per call).
*/
static PQExpBuffer
getThreadLocalPQExpBuffer(void)
{
/*
* The Tls code goes awry if we use a static var, so we provide for both
* static and auto, and omit any use of the static var when using Tls.
*/
static PQExpBuffer s_id_return = NULL;
PQExpBuffer id_return;
#ifdef WIN32
if (parallel_init_done)
id_return = (PQExpBuffer) TlsGetValue(tls_index); /* 0 when not set */
else
id_return = s_id_return;
#else
id_return = s_id_return;
#endif
if (id_return) /* first time through? */
{
/* same buffer, just wipe contents */
resetPQExpBuffer(id_return);
}
else
{
/* new buffer */
id_return = createPQExpBuffer();
#ifdef WIN32
if (parallel_init_done)
TlsSetValue(tls_index, id_return);
else
s_id_return = id_return;
#else
s_id_return = id_return;
#endif
}
return id_return;
}
/* /*
* pg_dump and pg_restore register the Archive pointer for the exit handler * pg_dump and pg_restore register the Archive pointer for the exit handler
* (called from exit_horribly). This function mainly exists so that we can * (called from exit_horribly). This function mainly exists so that we can
@@ -408,7 +508,7 @@ ParallelBackupStart(ArchiveHandle *AH, RestoreOptions *ropt)
* set and falls back to AHX otherwise. * set and falls back to AHX otherwise.
*/ */
shutdown_info.pstate = pstate; shutdown_info.pstate = pstate;
on_exit_msg_func = parallel_exit_msg_func; getLocalPQExpBuffer = getThreadLocalPQExpBuffer;
#ifdef WIN32 #ifdef WIN32
tMasterThreadId = GetCurrentThreadId(); tMasterThreadId = GetCurrentThreadId();

View File

@@ -16,6 +16,9 @@
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
#ifndef PG_DUMP_PARALLEL_H
#define PG_DUMP_PARALLEL_H
#include "pg_backup_db.h" #include "pg_backup_db.h"
struct _archiveHandle; struct _archiveHandle;
@@ -62,6 +65,13 @@ typedef struct ParallelState
ParallelSlot *parallelSlot; ParallelSlot *parallelSlot;
} ParallelState; } ParallelState;
#ifdef WIN32
extern bool parallel_init_done;
extern DWORD mainThreadId;
#endif
extern void init_parallel_dump_utils(void);
extern int GetIdleWorker(ParallelState *pstate); extern int GetIdleWorker(ParallelState *pstate);
extern bool IsEveryWorkerIdle(ParallelState *pstate); extern bool IsEveryWorkerIdle(ParallelState *pstate);
extern void ListenToWorkers(struct _archiveHandle * AH, ParallelState *pstate, bool do_wait); extern void ListenToWorkers(struct _archiveHandle * AH, ParallelState *pstate, bool do_wait);
@@ -77,3 +87,9 @@ extern void DispatchJobForTocEntry(struct _archiveHandle * AH,
extern void ParallelBackupEnd(struct _archiveHandle * AH, ParallelState *pstate); extern void ParallelBackupEnd(struct _archiveHandle * AH, ParallelState *pstate);
extern void checkAborting(struct _archiveHandle * AH); extern void checkAborting(struct _archiveHandle * AH);
extern void
exit_horribly(const char *modulename, const char *fmt,...)
__attribute__((format(PG_PRINTF_ATTRIBUTE, 2, 3), noreturn));
#endif /* PG_DUMP_PARALLEL_H */

View File

@@ -21,7 +21,7 @@
*/ */
#include "pg_backup_db.h" #include "pg_backup_db.h"
#include "dumputils.h" #include "pg_backup_utils.h"
#include "parallel.h" #include "parallel.h"
#include <ctype.h> #include <ctype.h>

View File

@@ -25,8 +25,8 @@
*/ */
#include "compress_io.h" #include "compress_io.h"
#include "dumputils.h"
#include "parallel.h" #include "parallel.h"
#include "pg_backup_utils.h"
/*-------- /*--------
* Routines in the format interface * Routines in the format interface

View File

@@ -11,7 +11,9 @@
*/ */
#include "pg_backup_db.h" #include "pg_backup_db.h"
#include "pg_backup_utils.h"
#include "dumputils.h" #include "dumputils.h"
#include "parallel.h"
#include <unistd.h> #include <unistd.h>
#include <ctype.h> #include <ctype.h>

View File

@@ -34,7 +34,7 @@
*/ */
#include "compress_io.h" #include "compress_io.h"
#include "dumputils.h" #include "pg_backup_utils.h"
#include "parallel.h" #include "parallel.h"
#include <dirent.h> #include <dirent.h>

View File

@@ -23,7 +23,8 @@
*/ */
#include "pg_backup_archiver.h" #include "pg_backup_archiver.h"
#include "dumputils.h" #include "pg_backup_utils.h"
#include "parallel.h"
#include <unistd.h> /* for dup */ #include <unistd.h> /* for dup */

View File

@@ -31,7 +31,8 @@
#include "pg_backup.h" #include "pg_backup.h"
#include "pg_backup_archiver.h" #include "pg_backup_archiver.h"
#include "pg_backup_tar.h" #include "pg_backup_tar.h"
#include "dumputils.h" #include "pg_backup_utils.h"
#include "parallel.h"
#include "pgtar.h" #include "pgtar.h"
#include <sys/stat.h> #include <sys/stat.h>

View File

@@ -0,0 +1,126 @@
/*-------------------------------------------------------------------------
*
* pg_backup_utils.c
* Utility routines shared by pg_dump and pg_restore
*
*
* Portions Copyright (c) 1996-2013, PostgreSQL Global Development Group
* Portions Copyright (c) 1994, Regents of the University of California
*
* src/bin/pg_dump/pg_backup_utils.c
*
*-------------------------------------------------------------------------
*/
#include "postgres_fe.h"
#include "pg_backup_utils.h"
#include "parallel.h"
/* Globals exported by this file */
const char *progname = NULL;
#define MAX_ON_EXIT_NICELY 20
static struct
{
on_exit_nicely_callback function;
void *arg;
} on_exit_nicely_list[MAX_ON_EXIT_NICELY];
static int on_exit_nicely_index;
/*
* Parse a --section=foo command line argument.
*
* Set or update the bitmask in *dumpSections according to arg.
* dumpSections is initialised as DUMP_UNSECTIONED by pg_dump and
* pg_restore so they can know if this has even been called.
*/
void
set_dump_section(const char *arg, int *dumpSections)
{
/* if this is the first call, clear all the bits */
if (*dumpSections == DUMP_UNSECTIONED)
*dumpSections = 0;
if (strcmp(arg, "pre-data") == 0)
*dumpSections |= DUMP_PRE_DATA;
else if (strcmp(arg, "data") == 0)
*dumpSections |= DUMP_DATA;
else if (strcmp(arg, "post-data") == 0)
*dumpSections |= DUMP_POST_DATA;
else
{
fprintf(stderr, _("%s: unrecognized section name: \"%s\"\n"),
progname, arg);
fprintf(stderr, _("Try \"%s --help\" for more information.\n"),
progname);
exit_nicely(1);
}
}
/*
* Write a printf-style message to stderr.
*
* The program name is prepended, if "progname" has been set.
* Also, if modulename isn't NULL, that's included too.
* Note that we'll try to translate the modulename and the fmt string.
*/
void
write_msg(const char *modulename, const char *fmt,...)
{
va_list ap;
va_start(ap, fmt);
vwrite_msg(modulename, fmt, ap);
va_end(ap);
}
/*
* As write_msg, but pass a va_list not variable arguments.
*/
void
vwrite_msg(const char *modulename, const char *fmt, va_list ap)
{
if (progname)
{
if (modulename)
fprintf(stderr, "%s: [%s] ", progname, _(modulename));
else
fprintf(stderr, "%s: ", progname);
}
vfprintf(stderr, _(fmt), ap);
}
/* Register a callback to be run when exit_nicely is invoked. */
void
on_exit_nicely(on_exit_nicely_callback function, void *arg)
{
if (on_exit_nicely_index >= MAX_ON_EXIT_NICELY)
exit_horribly(NULL, "out of on_exit_nicely slots\n");
on_exit_nicely_list[on_exit_nicely_index].function = function;
on_exit_nicely_list[on_exit_nicely_index].arg = arg;
on_exit_nicely_index++;
}
/*
* Run accumulated on_exit_nicely callbacks in reverse order and then exit
* quietly. This needs to be thread-safe.
*/
void
exit_nicely(int code)
{
int i;
for (i = on_exit_nicely_index - 1; i >= 0; i--)
(*on_exit_nicely_list[i].function) (code,
on_exit_nicely_list[i].arg);
#ifdef WIN32
if (parallel_init_done && GetCurrentThreadId() != mainThreadId)
ExitThread(code);
#endif
exit(code);
}

View File

@@ -0,0 +1,40 @@
/*-------------------------------------------------------------------------
*
* pg_backup_utils.h
* Utility routines shared by pg_dump and pg_restore.
*
*
* Portions Copyright (c) 1996-2013, PostgreSQL Global Development Group
* Portions Copyright (c) 1994, Regents of the University of California
*
* src/bin/pg_dump/pg_backup_utils.h
*
*-------------------------------------------------------------------------
*/
#ifndef PG_BACKUP_UTILS_H
#define PG_BACKUP_UTILS_H
typedef enum /* bits returned by set_dump_section */
{
DUMP_PRE_DATA = 0x01,
DUMP_DATA = 0x02,
DUMP_POST_DATA = 0x04,
DUMP_UNSECTIONED = 0xff
} DumpSections;
typedef void (*on_exit_nicely_callback) (int code, void *arg);
extern const char *progname;
extern void set_dump_section(const char *arg, int *dumpSections);
extern void
write_msg(const char *modulename, const char *fmt,...)
__attribute__((format(PG_PRINTF_ATTRIBUTE, 2, 3)));
extern void
vwrite_msg(const char *modulename, const char *fmt, va_list ap)
__attribute__((format(PG_PRINTF_ATTRIBUTE, 2, 0)));
extern void on_exit_nicely(on_exit_nicely_callback function, void *arg);
extern void exit_nicely(int code) __attribute__((noreturn));
#endif /* PG_BACKUP_UTILS_H */

View File

@@ -59,7 +59,9 @@
#include "pg_backup_archiver.h" #include "pg_backup_archiver.h"
#include "pg_backup_db.h" #include "pg_backup_db.h"
#include "pg_backup_utils.h"
#include "dumputils.h" #include "dumputils.h"
#include "parallel.h"
extern char *optarg; extern char *optarg;
extern int optind, extern int optind,

View File

@@ -14,7 +14,8 @@
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
#include "pg_backup_archiver.h" #include "pg_backup_archiver.h"
#include "dumputils.h" #include "pg_backup_utils.h"
#include "parallel.h"
/* translator: this is a module name */ /* translator: this is a module name */
static const char *modulename = gettext_noop("sorter"); static const char *modulename = gettext_noop("sorter");

View File

@@ -63,6 +63,7 @@ static PGresult *executeQuery(PGconn *conn, const char *query);
static void executeCommand(PGconn *conn, const char *query); static void executeCommand(PGconn *conn, const char *query);
static char pg_dump_bin[MAXPGPATH]; static char pg_dump_bin[MAXPGPATH];
static const char *progname;
static PQExpBuffer pgdumpopts; static PQExpBuffer pgdumpopts;
static char *connstr = ""; static char *connstr = "";
static bool skip_acls = false; static bool skip_acls = false;
@@ -82,6 +83,7 @@ static int server_version;
static FILE *OPF; static FILE *OPF;
static char *filename = NULL; static char *filename = NULL;
#define exit_nicely(code) exit(code)
int int
main(int argc, char *argv[]) main(int argc, char *argv[])

View File

@@ -40,8 +40,9 @@
*/ */
#include "pg_backup_archiver.h" #include "pg_backup_archiver.h"
#include "pg_backup_utils.h"
#include "dumputils.h" #include "dumputils.h"
#include "parallel.h"
#include <ctype.h> #include <ctype.h>