Mercurial > hg > xemacs-beta
view modules/postgresql/postgresql.c @ 5127:a9c41067dd88 ben-lisp-object
more cleanups, terminology clarification, lots of doc work
-------------------- ChangeLog entries follow: --------------------
man/ChangeLog addition:
2010-03-05 Ben Wing <ben@xemacs.org>
* internals/internals.texi (Introduction to Allocation):
* internals/internals.texi (Integers and Characters):
* internals/internals.texi (Allocation from Frob Blocks):
* internals/internals.texi (lrecords):
* internals/internals.texi (Low-level allocation):
Rewrite section on allocation of Lisp objects to reflect the new
reality. Remove references to nonexistent XSETINT and XSETCHAR.
modules/ChangeLog addition:
2010-03-05 Ben Wing <ben@xemacs.org>
* postgresql/postgresql.c (allocate_pgconn):
* postgresql/postgresql.c (allocate_pgresult):
* postgresql/postgresql.h (struct Lisp_PGconn):
* postgresql/postgresql.h (struct Lisp_PGresult):
* ldap/eldap.c (allocate_ldap):
* ldap/eldap.h (struct Lisp_LDAP):
Same changes as in src/ dir. See large log there in ChangeLog,
but basically:
ALLOC_LISP_OBJECT -> ALLOC_NORMAL_LISP_OBJECT
LISP_OBJECT_HEADER -> NORMAL_LISP_OBJECT_HEADER
../hlo/src/ChangeLog addition:
2010-03-05 Ben Wing <ben@xemacs.org>
* alloc.c:
* alloc.c (old_alloc_sized_lcrecord):
* alloc.c (very_old_free_lcrecord):
* alloc.c (copy_lisp_object):
* alloc.c (zero_sized_lisp_object):
* alloc.c (zero_nonsized_lisp_object):
* alloc.c (lisp_object_storage_size):
* alloc.c (free_normal_lisp_object):
* alloc.c (FREE_FIXED_TYPE_WHEN_NOT_IN_GC):
* alloc.c (ALLOC_FROB_BLOCK_LISP_OBJECT):
* alloc.c (Fcons):
* alloc.c (noseeum_cons):
* alloc.c (make_float):
* alloc.c (make_bignum):
* alloc.c (make_bignum_bg):
* alloc.c (make_ratio):
* alloc.c (make_ratio_bg):
* alloc.c (make_ratio_rt):
* alloc.c (make_bigfloat):
* alloc.c (make_bigfloat_bf):
* alloc.c (size_vector):
* alloc.c (make_compiled_function):
* alloc.c (Fmake_symbol):
* alloc.c (allocate_extent):
* alloc.c (allocate_event):
* alloc.c (make_key_data):
* alloc.c (make_button_data):
* alloc.c (make_motion_data):
* alloc.c (make_process_data):
* alloc.c (make_timeout_data):
* alloc.c (make_magic_data):
* alloc.c (make_magic_eval_data):
* alloc.c (make_eval_data):
* alloc.c (make_misc_user_data):
* alloc.c (Fmake_marker):
* alloc.c (noseeum_make_marker):
* alloc.c (size_string_direct_data):
* alloc.c (make_uninit_string):
* alloc.c (make_string_nocopy):
* alloc.c (mark_lcrecord_list):
* alloc.c (alloc_managed_lcrecord):
* alloc.c (free_managed_lcrecord):
* alloc.c (sweep_lcrecords_1):
* alloc.c (malloced_storage_size):
* buffer.c (allocate_buffer):
* buffer.c (compute_buffer_usage):
* buffer.c (DEFVAR_BUFFER_LOCAL_1):
* buffer.c (nuke_all_buffer_slots):
* buffer.c (common_init_complex_vars_of_buffer):
* buffer.h (struct buffer_text):
* buffer.h (struct buffer):
* bytecode.c:
* bytecode.c (make_compiled_function_args):
* bytecode.c (size_compiled_function_args):
* bytecode.h (struct compiled_function_args):
* casetab.c (allocate_case_table):
* casetab.h (struct Lisp_Case_Table):
* charset.h (struct Lisp_Charset):
* chartab.c (fill_char_table):
* chartab.c (Fmake_char_table):
* chartab.c (make_char_table_entry):
* chartab.c (copy_char_table_entry):
* chartab.c (Fcopy_char_table):
* chartab.c (put_char_table):
* chartab.h (struct Lisp_Char_Table_Entry):
* chartab.h (struct Lisp_Char_Table):
* console-gtk-impl.h (struct gtk_device):
* console-gtk-impl.h (struct gtk_frame):
* console-impl.h (struct console):
* console-msw-impl.h (struct Lisp_Devmode):
* console-msw-impl.h (struct mswindows_device):
* console-msw-impl.h (struct msprinter_device):
* console-msw-impl.h (struct mswindows_frame):
* console-msw-impl.h (struct mswindows_dialog_id):
* console-stream-impl.h (struct stream_console):
* console-stream.c (stream_init_console):
* console-tty-impl.h (struct tty_console):
* console-tty-impl.h (struct tty_device):
* console-tty.c (allocate_tty_console_struct):
* console-x-impl.h (struct x_device):
* console-x-impl.h (struct x_frame):
* console.c (allocate_console):
* console.c (nuke_all_console_slots):
* console.c (DEFVAR_CONSOLE_LOCAL_1):
* console.c (common_init_complex_vars_of_console):
* data.c (make_weak_list):
* data.c (make_weak_box):
* data.c (make_ephemeron):
* database.c:
* database.c (struct Lisp_Database):
* database.c (allocate_database):
* database.c (finalize_database):
* device-gtk.c (allocate_gtk_device_struct):
* device-impl.h (struct device):
* device-msw.c:
* device-msw.c (mswindows_init_device):
* device-msw.c (msprinter_init_device):
* device-msw.c (finalize_devmode):
* device-msw.c (allocate_devmode):
* device-tty.c (allocate_tty_device_struct):
* device-x.c (allocate_x_device_struct):
* device.c:
* device.c (nuke_all_device_slots):
* device.c (allocate_device):
* dialog-msw.c (handle_question_dialog_box):
* elhash.c:
* elhash.c (struct Lisp_Hash_Table):
* elhash.c (finalize_hash_table):
* elhash.c (make_general_lisp_hash_table):
* elhash.c (Fcopy_hash_table):
* elhash.h (htentry):
* emacs.c (main_1):
* eval.c:
* eval.c (size_multiple_value):
* event-stream.c (finalize_command_builder):
* event-stream.c (allocate_command_builder):
* event-stream.c (free_command_builder):
* event-stream.c (event_stream_generate_wakeup):
* event-stream.c (event_stream_resignal_wakeup):
* event-stream.c (event_stream_disable_wakeup):
* event-stream.c (event_stream_wakeup_pending_p):
* events.h (struct Lisp_Timeout):
* events.h (struct command_builder):
* extents-impl.h:
* extents-impl.h (struct extent_auxiliary):
* extents-impl.h (struct extent_info):
* extents-impl.h (set_extent_no_chase_aux_field):
* extents-impl.h (set_extent_no_chase_normal_field):
* extents.c:
* extents.c (gap_array_marker):
* extents.c (gap_array):
* extents.c (extent_list_marker):
* extents.c (extent_list):
* extents.c (stack_of_extents):
* extents.c (gap_array_make_marker):
* extents.c (extent_list_make_marker):
* extents.c (allocate_extent_list):
* extents.c (SLOT):
* extents.c (mark_extent_auxiliary):
* extents.c (allocate_extent_auxiliary):
* extents.c (attach_extent_auxiliary):
* extents.c (size_gap_array):
* extents.c (finalize_extent_info):
* extents.c (allocate_extent_info):
* extents.c (uninit_buffer_extents):
* extents.c (allocate_soe):
* extents.c (copy_extent):
* extents.c (vars_of_extents):
* extents.h:
* faces.c (allocate_face):
* faces.h (struct Lisp_Face):
* faces.h (struct face_cachel):
* file-coding.c:
* file-coding.c (finalize_coding_system):
* file-coding.c (sizeof_coding_system):
* file-coding.c (Fcopy_coding_system):
* file-coding.h (struct Lisp_Coding_System):
* file-coding.h (MARKED_SLOT):
* fns.c (size_bit_vector):
* font-mgr.c:
* font-mgr.c (finalize_fc_pattern):
* font-mgr.c (print_fc_pattern):
* font-mgr.c (Ffc_pattern_p):
* font-mgr.c (Ffc_pattern_create):
* font-mgr.c (Ffc_name_parse):
* font-mgr.c (Ffc_name_unparse):
* font-mgr.c (Ffc_pattern_duplicate):
* font-mgr.c (Ffc_pattern_add):
* font-mgr.c (Ffc_pattern_del):
* font-mgr.c (Ffc_pattern_get):
* font-mgr.c (fc_config_create_using):
* font-mgr.c (fc_strlist_to_lisp_using):
* font-mgr.c (fontset_to_list):
* font-mgr.c (Ffc_config_p):
* font-mgr.c (Ffc_config_up_to_date):
* font-mgr.c (Ffc_config_build_fonts):
* font-mgr.c (Ffc_config_get_cache):
* font-mgr.c (Ffc_config_get_fonts):
* font-mgr.c (Ffc_config_set_current):
* font-mgr.c (Ffc_config_get_blanks):
* font-mgr.c (Ffc_config_get_rescan_interval):
* font-mgr.c (Ffc_config_set_rescan_interval):
* font-mgr.c (Ffc_config_app_font_add_file):
* font-mgr.c (Ffc_config_app_font_add_dir):
* font-mgr.c (Ffc_config_app_font_clear):
* font-mgr.c (size):
* font-mgr.c (Ffc_config_substitute):
* font-mgr.c (Ffc_font_render_prepare):
* font-mgr.c (Ffc_font_match):
* font-mgr.c (Ffc_font_sort):
* font-mgr.c (finalize_fc_config):
* font-mgr.c (print_fc_config):
* font-mgr.h:
* font-mgr.h (struct fc_pattern):
* font-mgr.h (XFC_PATTERN):
* font-mgr.h (struct fc_config):
* font-mgr.h (XFC_CONFIG):
* frame-gtk.c (allocate_gtk_frame_struct):
* frame-impl.h (struct frame):
* frame-msw.c (mswindows_init_frame_1):
* frame-x.c (allocate_x_frame_struct):
* frame.c (nuke_all_frame_slots):
* frame.c (allocate_frame_core):
* gc.c:
* gc.c (GC_CHECK_NOT_FREE):
* glyphs.c (finalize_image_instance):
* glyphs.c (allocate_image_instance):
* glyphs.c (Fcolorize_image_instance):
* glyphs.c (allocate_glyph):
* glyphs.c (unmap_subwindow_instance_cache_mapper):
* glyphs.c (register_ignored_expose):
* glyphs.h (struct Lisp_Image_Instance):
* glyphs.h (struct Lisp_Glyph):
* glyphs.h (struct glyph_cachel):
* glyphs.h (struct expose_ignore):
* gui.c (allocate_gui_item):
* gui.h (struct Lisp_Gui_Item):
* keymap.c (struct Lisp_Keymap):
* keymap.c (make_keymap):
* lisp.h:
* lisp.h (struct Lisp_String_Direct_Data):
* lisp.h (struct Lisp_String_Indirect_Data):
* lisp.h (struct Lisp_Vector):
* lisp.h (struct Lisp_Bit_Vector):
* lisp.h (DECLARE_INLINE_LISP_BIT_VECTOR):
* lisp.h (struct weak_box):
* lisp.h (struct ephemeron):
* lisp.h (struct weak_list):
* lrecord.h:
* lrecord.h (struct lrecord_implementation):
* lrecord.h (MC_ALLOC_CALL_FINALIZER):
* lrecord.h (struct lcrecord_list):
* lstream.c (finalize_lstream):
* lstream.c (sizeof_lstream):
* lstream.c (Lstream_new):
* lstream.c (Lstream_delete):
* lstream.h (struct lstream):
* marker.c:
* marker.c (finalize_marker):
* marker.c (compute_buffer_marker_usage):
* mule-charset.c:
* mule-charset.c (make_charset):
* mule-charset.c (compute_charset_usage):
* objects-impl.h (struct Lisp_Color_Instance):
* objects-impl.h (struct Lisp_Font_Instance):
* objects-tty-impl.h (struct tty_color_instance_data):
* objects-tty-impl.h (struct tty_font_instance_data):
* objects-tty.c (tty_initialize_color_instance):
* objects-tty.c (tty_initialize_font_instance):
* objects.c (finalize_color_instance):
* objects.c (Fmake_color_instance):
* objects.c (finalize_font_instance):
* objects.c (Fmake_font_instance):
* objects.c (reinit_vars_of_objects):
* opaque.c:
* opaque.c (sizeof_opaque):
* opaque.c (make_opaque_ptr):
* opaque.c (free_opaque_ptr):
* opaque.h:
* opaque.h (Lisp_Opaque):
* opaque.h (Lisp_Opaque_Ptr):
* print.c (printing_unreadable_lcrecord):
* print.c (external_object_printer):
* print.c (debug_p4):
* process.c (finalize_process):
* process.c (make_process_internal):
* procimpl.h (struct Lisp_Process):
* rangetab.c (Fmake_range_table):
* rangetab.c (Fcopy_range_table):
* rangetab.h (struct Lisp_Range_Table):
* scrollbar.c:
* scrollbar.c (create_scrollbar_instance):
* scrollbar.c (compute_scrollbar_instance_usage):
* scrollbar.h (struct scrollbar_instance):
* specifier.c (finalize_specifier):
* specifier.c (sizeof_specifier):
* specifier.c (set_specifier_caching):
* specifier.h (struct Lisp_Specifier):
* specifier.h (struct specifier_caching):
* symeval.h:
* symeval.h (SYMBOL_VALUE_MAGIC_P):
* symeval.h (DEFVAR_SYMVAL_FWD):
* symsinit.h:
* syntax.c (init_buffer_syntax_cache):
* syntax.h (struct syntax_cache):
* toolbar.c:
* toolbar.c (allocate_toolbar_button):
* toolbar.c (update_toolbar_button):
* toolbar.h (struct toolbar_button):
* tooltalk.c (struct Lisp_Tooltalk_Message):
* tooltalk.c (make_tooltalk_message):
* tooltalk.c (struct Lisp_Tooltalk_Pattern):
* tooltalk.c (make_tooltalk_pattern):
* ui-gtk.c:
* ui-gtk.c (allocate_ffi_data):
* ui-gtk.c (emacs_gtk_object_finalizer):
* ui-gtk.c (allocate_emacs_gtk_object_data):
* ui-gtk.c (allocate_emacs_gtk_boxed_data):
* ui-gtk.h:
* window-impl.h (struct window):
* window-impl.h (struct window_mirror):
* window.c (finalize_window):
* window.c (allocate_window):
* window.c (new_window_mirror):
* window.c (mark_window_as_deleted):
* window.c (make_dummy_parent):
* window.c (compute_window_mirror_usage):
* window.c (compute_window_usage):
Overall point of this change and previous ones in this repository:
(1) Introduce new, clearer terminology: everything other than int
or char is a "record" object, which comes in two types: "normal
objects" and "frob-block objects". Fix up all places that
referred to frob-block objects as "simple", "basic", etc.
(2) Provide an advertised interface for doing operations on Lisp
objects, including creating new types, that is clean and
consistent in its naming, uses the above-referenced terms and
avoids referencing "lrecords", "old lcrecords", etc., which should
hide under the surface.
(3) Make the size_in_bytes and finalizer methods take a
Lisp_Object rather than a void * for consistency with other methods.
(4) Separate finalizer method into finalizer and disksaver, so
that normal finalize methods don't have to worry about disksaving.
Other specifics:
(1) Renaming:
LISP_OBJECT_HEADER -> NORMAL_LISP_OBJECT_HEADER
ALLOC_LISP_OBJECT -> ALLOC_NORMAL_LISP_OBJECT
implementation->basic_p -> implementation->frob_block_p
ALLOCATE_FIXED_TYPE_AND_SET_IMPL -> ALLOC_FROB_BLOCK_LISP_OBJECT
*FCCONFIG*, wrap_fcconfig -> *FC_CONFIG*, wrap_fc_config
*FCPATTERN*, wrap_fcpattern -> *FC_PATTERN*, wrap_fc_pattern
(the last two changes make the naming of these macros consistent
with the naming of all other macros, since the objects are named
fc-config and fc-pattern with a hyphen)
(2) Lots of documentation fixes in lrecord.h.
(3) Eliminate macros for copying, freeing, zeroing objects, getting
their storage size. Instead, new functions:
zero_sized_lisp_object()
zero_nonsized_lisp_object()
lisp_object_storage_size()
free_normal_lisp_object()
(copy_lisp_object() already exists)
LISP_OBJECT_FROB_BLOCK_P() (actually a macro)
Eliminated:
free_lrecord()
zero_lrecord()
copy_lrecord()
copy_sized_lrecord()
old_copy_lcrecord()
old_copy_sized_lcrecord()
old_zero_lcrecord()
old_zero_sized_lcrecord()
LISP_OBJECT_STORAGE_SIZE()
COPY_SIZED_LISP_OBJECT()
COPY_SIZED_LCRECORD()
COPY_LISP_OBJECT()
ZERO_LISP_OBJECT()
FREE_LISP_OBJECT()
(4) Catch the remaining places where lrecord stuff was used directly
and use the advertised interface, e.g. alloc_sized_lrecord() ->
ALLOC_SIZED_LISP_OBJECT().
(5) Make certain statically-declared pseudo-objects
(buffer_local_flags, console_local_flags) have their lheader
initialized correctly, so things like copy_lisp_object() can work
on them. Make extent_auxiliary_defaults a proper heap object
Vextent_auxiliary_defaults, and make extent auxiliaries dumpable
so that this object can be dumped. allocate_extent_auxiliary()
now just creates the object, and attach_extent_auxiliary()
creates an extent auxiliary and attaches to an extent, like the
old allocate_extent_auxiliary().
(6) Create EXTENT_AUXILIARY_SLOTS macro, similar to the foo-slots.h
files but in a macro instead of a file. The purpose is to avoid
duplication when iterating over all the slots in an extent auxiliary.
Use it.
(7) In lstream.c, don't zero out object after allocation because
allocation routines take care of this.
(8) In marker.c, fix a mistake in computing marker overhead.
(9) In print.c, clean up printing_unreadable_lcrecord(),
external_object_printer() to avoid lots of ifdef NEW_GC's.
(10) Separate toolbar-button allocation into a separate
allocate_toolbar_button() function for use in the example code
in lrecord.h.
author | Ben Wing <ben@xemacs.org> |
---|---|
date | Fri, 05 Mar 2010 04:08:17 -0600 |
parents | b5df3737028a |
children | 0dcd22290039 |
line wrap: on
line source
/* postgresql.c -- Emacs Lisp binding to libpq.so Copyright (C) 2000 Electrotechnical Laboratory, JAPAN. Licensed to the Free Software Foundation. Author: SL Baur <steve@xemacs.org> Maintainer: SL Baur <steve@xemacs.org> Please send patches to this file to me first before submitting them to xemacs-patches. KNOWN PROBLEMS (Last update 15-March-2000) + None. Implementation notes: 0. Supported PostgreSQL versions This code was developed against libpq-6.5.3 and libpq-7.0-beta1. Earlier versions may work. V7 support is more complete than V6.5 support. 1. Mule Non-ASCII databases have been tested on both 6.5 and 7.0. 2. Asynchronous Operation Starting with libpq-7.0, an asynchronous interface is offered. This binding supports the asynchronous calls to a limited extent. Since the XEmacs 21.2 core does not support a sensible interface to add managed but unreadable (by XEmacs) file descriptors to the main select code, polling is required to drive the asynchronous calls. XtAppAddInput would work fine, but we want to be able to use the database when running strictly in tty mode. 3. Completeness Various calls have been deliberately not exported to Lisp. The unexported calls are either left-over backwards compatibility code that aren't needed, calls that cannot be implemented sensibly, or calls that cannot be implemented safely. A list of all global functions in libpq but not exported to Lisp is below. 4. Policy This interface tries very hard to not set any policy towards how database code in Emacs Lisp will be written. 5. Documentation For full lisp programming documentation, see the XEmacs Lisp Reference Manual. For PostgreSQL documentation, see the PostgreSQL distribution. TODO (in rough order of priority): 1. Asynchronous notifies need to be implemented to the extent they can be. 2. The large object interface needs work with Emacs buffers in addition to files. Need two functions buffer->large_object, and large_object-> buffer. */ /* Unimplemented functions: [TODO] PQsetNoticeProcessor Implemented, but undocumented functions: [TODO] PQgetline (copy in/out) PQputline (copy in/out) PQgetlineAsync (copy in/out Asynch.) PQputnbytes (copy in/out Asynch.) PQendcopy (copy in/out) Unsupported functions: PQsetdbLogin -- This function is deprecated, has a subset of the functionality of PQconnectdb, and is better done in Lisp. PQsetdb -- Same as for PQsetdbLogin PQsocket -- Abstraction error, file descriptors should not be leaked into Lisp code PQprint -- print to a file descriptor, deprecated, better done in Lisp PQdisplayTuples -- deprecated PQprintTuples -- really, really deprecated PQmblen -- Returns the length in bytes of multibyte character encoded string. PQtrace -- controls debug print tracing to a tty. PQuntrace -- Ditto. I don't see any way to do this sensibly. PQoidStatus -- deprecated and nearly identical to PQoidValue PQfn -- "Fast path" interface lo_open (large object) [*] lo_close (large object) [*] lo_read (large object) [*] lo_write (large object) [*] lo_lseek (large object) [*] lo_creat (large object) [*] lo_tell (large object) [*] lo_unlink (large object) [*] */ #include <config.h> /* This must be portable with XEmacs 21.1 so long as it is the official released version of XEmacs and provides the basis of InfoDock. The interface to lcrecord handling has changed with 21.2, so unfortunately we will need a few snippets of backwards compatibility code. */ #if (EMACS_MAJOR_VERSION == 21) && (EMACS_MINOR_VERSION <= 1) #define RUNNING_XEMACS_21_1 1 #elif (EMACS_MAJOR_VERSION == 21) && (EMACS_MINOR_VERSION <= 4) #define RUNNING_XEMACS_21_4 1 #endif /* #define POSTGRES_LO_IMPORT_IS_VOID 1 */ #include "lisp.h" #include "buffer.h" #include "postgresql.h" #include "process.h" #ifdef HAVE_SHLIB # include "emodules.h" #endif #include "sysdep.h" #include "sysfile.h" #ifdef RUNNING_XEMACS_21_1 /* handle interface changes */ #define PG_OS_CODING FORMAT_FILENAME #define TO_EXTERNAL_FORMAT(a,from,b,to,c) GET_C_STRING_EXT_DATA_ALLOCA(from,FORMAT_FILENAME,to) #else #ifdef MULE #define PG_OS_CODING get_coding_system_for_text_file (Vpg_coding_system, 1) #else #define PG_OS_CODING Qnative #endif Lisp_Object Vpg_coding_system; #endif #define CHECK_LIVE_CONNECTION(P) \ do \ { \ if (!P || (PQstatus (P) != CONNECTION_OK)) \ { \ Lisp_Object err; \ \ if (P) \ err = build_extstring (PQerrorMessage (P), PG_OS_CODING); \ else \ err = build_msg_string ("Bad value"); \ signal_error (Qprocess_error, "Dead connection", err); \ } \ } \ while (0) #define PUKE_IF_NULL(p) \ do \ { \ if (!p) signal_error (Qinvalid_argument, "Bad value", Qunbound); \ } \ while (0) #define SIGNAL_ERROR(p, reason) \ do \ { \ signal_error (Qprocess_error, reason, \ build_extstring (PQerrorMessage (p), PG_OS_CODING)); \ } \ while (0) static Lisp_Object VXPGHOST; static Lisp_Object VXPGUSER; static Lisp_Object VXPGOPTIONS; static Lisp_Object VXPGPORT; static Lisp_Object VXPGTTY; /* This needs to be blanked! */ static Lisp_Object VXPGDATABASE; static Lisp_Object VXPGREALM; #ifdef MULE static Lisp_Object VXPGCLIENTENCODING; #endif /* MULE */ /* Other variables: PGAUTHTYPE -- not used after PostgreSQL 6.5 PGGEQO PGCOSTINDEX PGCOSTHEAP PGTZ PGDATESTYLE */ #ifndef HAVE_POSTGRESQLV7 static Lisp_Object VXPGAUTHTYPE; #endif static Lisp_Object VXPGGEQO, VXPGCOSTINDEX, VXPGCOSTHEAP, VXPGTZ, VXPGDATESTYLE; static Lisp_Object Qpostgresql; static Lisp_Object Qpg_connection_ok, Qpg_connection_bad; static Lisp_Object Qpg_connection_started, Qpg_connection_made; static Lisp_Object Qpg_connection_awaiting_response, Qpg_connection_auth_ok; static Lisp_Object Qpg_connection_setenv; static Lisp_Object Qpqdb, Qpquser, Qpqpass, Qpqhost, Qpqport, Qpqtty; static Lisp_Object Qpqoptions, Qpqstatus, Qpqerrormessage, Qpqbackendpid; static Lisp_Object Qpgres_empty_query, Qpgres_command_ok, Qpgres_tuples_ok; static Lisp_Object Qpgres_copy_out, Qpgres_copy_in, Qpgres_bad_response; static Lisp_Object Qpgres_nonfatal_error, Qpgres_fatal_error; static Lisp_Object Qpgres_polling_failed, Qpgres_polling_reading; static Lisp_Object Qpgres_polling_writing, Qpgres_polling_ok; static Lisp_Object Qpgres_polling_active; /****/ /* PGconn is an opaque object and we need to be able to store them in Lisp code because libpq supports multiple connections. */ Lisp_Object Qpgconnp; static Lisp_Object make_pgconn (Lisp_PGconn *pgconn) { return wrap_pgconn (pgconn); } static const struct memory_description pgconn_description [] = { { XD_END } }; static Lisp_Object #ifdef RUNNING_XEMACS_21_1 mark_pgconn (Lisp_Object UNUSED (obj), void (*UNUSED_ARG (markobj)) (Lisp_Object) ATTRIBUTE_UNUSED) #else mark_pgconn (Lisp_Object UNUSED (obj)) #endif { return Qnil; } static void print_pgconn (Lisp_Object obj, Lisp_Object printcharfun, int UNUSED (escapeflag)) { char buf[256]; PGconn *P; ConnStatusType cst; const char *host="", *db="", *user="", *port=""; P = (XPGCONN (obj))->pgconn; if (!P) /* this may happen since we allow PQfinish() to be called */ strcpy (buf, "#<PGconn DEAD>"); /* evil! */ else if ((cst = PQstatus (P)) == CONNECTION_OK) { if (!(host = PQhost (P))) host = ""; port = PQport (P); db = PQdb (P); if (!(user = PQuser (P))) user = ""; sprintf (buf, "#<PGconn %s:%s %s/%s>", /* evil! */ !strlen (host) ? "localhost" : host, port, user, db); } else if (cst == CONNECTION_BAD) strcpy (buf, "#<PGconn BAD>"); /* evil! */ else strcpy (buf, "#<PGconn connecting>"); /* evil! */ if (print_readably) printing_unreadable_object ("%s", buf); else write_cistring (printcharfun, buf); } static Lisp_PGconn * allocate_pgconn (void) { #ifdef RUNNING_XEMACS_21_1 Lisp_PGconn *pgconn = ALLOC_LCRECORD_TYPE (Lisp_PGconn, lrecord_pgconn); #elif defined (RUNNING_XEMACS_21_4) Lisp_PGconn *pgconn = ALLOC_LCRECORD_TYPE (Lisp_PGconn, &lrecord_pgconn); #else Lisp_PGconn *pgconn = XPGCONN (ALLOC_NORMAL_LISP_OBJECT (pgconn)); #endif pgconn->pgconn = (PGconn *)NULL; return pgconn; } #ifdef RUNNING_XEMACS_21_4 static void finalize_pgconn (void *header, int for_disksave) { Lisp_PGconn *pgconn = (Lisp_PGconn *)header; if (for_disksave) invalid_operation ("Can't dump an emacs containing PGconn objects", make_pgconn (pgconn)); if (pgconn->pgconn) { PQfinish (pgconn->pgconn); pgconn->pgconn = (PGconn *)NULL; } } #else /* not RUNNING_XEMACS_21_4 */ static void finalize_pgconn (void *header) { Lisp_PGconn *pgconn = (Lisp_PGconn *)header; if (pgconn->pgconn) { PQfinish (pgconn->pgconn); pgconn->pgconn = (PGconn *)NULL; } } #endif /* (not) RUNNING_XEMACS_21_4 */ #ifdef RUNNING_XEMACS_21_1 DEFINE_LRECORD_IMPLEMENTATION ("pgconn", pgconn, mark_pgconn, print_pgconn, finalize_pgconn, NULL, NULL, Lisp_PGconn); #elif defined (RUNNING_XEMACS_21_4) DEFINE_LRECORD_IMPLEMENTATION ("pgconn", pgconn, 0, /*dumpable-flag*/ mark_pgconn, print_pgconn, finalize_pgconn, NULL, NULL, pgconn_description, Lisp_PGconn); #else DEFINE_NODUMP_LISP_OBJECT ("pgconn", pgconn, mark_pgconn, print_pgconn, finalize_pgconn, NULL, NULL, pgconn_description, Lisp_PGconn); #endif /****/ /* PGresult is an opaque object and we need to be able to store them in Lisp code. */ Lisp_Object Qpgresultp; static Lisp_Object make_pgresult (Lisp_PGresult *pgresult) { return wrap_pgresult (pgresult); } static const struct memory_description pgresult_description [] = { { XD_END } }; static Lisp_Object #ifdef RUNNING_XEMACS_21_1 mark_pgresult (Lisp_Object UNUSED (obj), void (*UNUSED_ARG (markobj)) (Lisp_Object) ATTRIBUTE_UNUSED) #else mark_pgresult (Lisp_Object UNUSED (obj)) #endif { return Qnil; } #define RESULT_TUPLES_FMT "#<PGresult %s[%d] - %s>" #define RESULT_CMD_TUPLES_FMT "#<PGresult %s[%s] - %s>" #define RESULT_DEFAULT_FMT "#<PGresult %s - %s>" static void print_pgresult (Lisp_Object obj, Lisp_Object printcharfun, int UNUSED (escapeflag)) { char buf[1024]; PGresult *res; res = (XPGRESULT (obj))->pgresult; if (res) { switch (PQresultStatus (res)) { case PGRES_TUPLES_OK: /* Add number of tuples of result to output */ sprintf (buf, RESULT_TUPLES_FMT, /* evil! */ PQresStatus (PQresultStatus (res)), PQntuples (res), PQcmdStatus (res)); break; case PGRES_COMMAND_OK: /* Add number of tuples affected by output-less command */ if (!strlen (PQcmdTuples (res))) goto notuples; sprintf (buf, RESULT_CMD_TUPLES_FMT, /* evil! */ PQresStatus (PQresultStatus (res)), PQcmdTuples (res), PQcmdStatus (res)); break; default: notuples: /* No counts to print */ sprintf (buf, RESULT_DEFAULT_FMT, /* evil! */ PQresStatus (PQresultStatus (res)), PQcmdStatus (res)); break; } } else strcpy (buf, "#<PGresult DEAD>"); /* evil! */ if (print_readably) printing_unreadable_object ("%s", buf); else write_cistring (printcharfun, buf); } #undef RESULT_TUPLES_FMT #undef RESULT_CMD_TUPLES_FMT #undef RESULT_DEFAULT_FMT static Lisp_PGresult * allocate_pgresult (void) { #ifdef RUNNING_XEMACS_21_1 Lisp_PGresult *pgresult = ALLOC_LCRECORD_TYPE (Lisp_PGresult, lrecord_pgresult); #elif defined (RUNNING_XEMACS_21_4) Lisp_PGresult *pgresult = ALLOC_LCRECORD_TYPE (Lisp_PGresult, &lrecord_pgresult); #else Lisp_PGresult *pgresult = XPGRESULT (ALLOC_NORMAL_LISP_OBJECT (pgresult)); #endif pgresult->pgresult = (PGresult *)NULL; return pgresult; } #ifdef RUNNING_XEMACS_21_4 static void finalize_pgresult (void *header, int for_disksave) { Lisp_PGresult *pgresult = (Lisp_PGresult *)header; if (for_disksave) invalid_operation ("Can't dump an emacs containing PGresult objects", make_pgresult (pgresult)); if (pgresult->pgresult) { PQclear (pgresult->pgresult); pgresult->pgresult = (PGresult *)NULL; } } #else /* not RUNNING_XEMACS_21_4 */ static void finalize_pgresult (void *header) { Lisp_PGresult *pgresult = (Lisp_PGresult *)header; if (pgresult->pgresult) { PQclear (pgresult->pgresult); pgresult->pgresult = (PGresult *)NULL; } } #endif /* (not) RUNNING_XEMACS_21_4 */ #ifdef RUNNING_XEMACS_21_1 DEFINE_LRECORD_IMPLEMENTATION ("pgresult", pgresult, mark_pgresult, print_pgresult, finalize_pgresult, NULL, NULL, Lisp_PGresult); #elif defined (RUNNING_XEMACS_21_4) DEFINE_LRECORD_IMPLEMENTATION ("pgresult", pgresult, 0, /*dumpable-flag*/ mark_pgresult, print_pgresult, finalize_pgresult, NULL, NULL, pgresult_description, Lisp_PGresult); #else DEFINE_NODUMP_LISP_OBJECT ("pgresult", pgresult, mark_pgresult, print_pgresult, finalize_pgresult, NULL, NULL, pgresult_description, Lisp_PGresult); #endif /***********************/ /* notices */ static void xemacs_notice_processor (void *UNUSED (arg), const char *msg) { warn_when_safe (Qpostgresql, Qnotice, "%s", msg); } /* There are four ways (as of PostgreSQL v7) to connect to a database. Two of them, PQsetdb and PQsetdbLogin, are deprecated. Both of those routines take a number of positional parameters and are better done in Lisp. Note that PQconnectStart does not exist prior to v7. */ /* ###autoload */ DEFUN ("pq-conn-defaults", Fpq_conn_defaults, 0, 0, 0, /* Return a connection default structure. */ ()) { /* This function can GC */ PQconninfoOption *pcio; Lisp_Object temp, temp1; int i; pcio = PQconndefaults(); if (!pcio) return Qnil; /* can never happen in libpq-7.0 */ temp = list1 (nconc2 (list4 (build_extstring (pcio[0].keyword, PG_OS_CODING), build_extstring (pcio[0].envvar, PG_OS_CODING), build_extstring (pcio[0].compiled, PG_OS_CODING), build_extstring (pcio[0].val, PG_OS_CODING)), list3 (build_extstring (pcio[0].label, PG_OS_CODING), build_extstring (pcio[0].dispchar, PG_OS_CODING), make_int (pcio[0].dispsize)))); for (i = 1; pcio[i].keyword; i++) { temp1 = list1 (nconc2 (list4 (build_extstring (pcio[i].keyword, PG_OS_CODING), build_extstring (pcio[i].envvar, PG_OS_CODING), build_extstring (pcio[i].compiled, PG_OS_CODING), build_extstring (pcio[i].val, PG_OS_CODING)), list3 (build_extstring (pcio[i].label, PG_OS_CODING), build_extstring (pcio[i].dispchar, PG_OS_CODING), make_int (pcio[i].dispsize)))); { Lisp_Object args[2]; args[0] = temp; args[1] = temp1; /* Fappend GCPROs its arguments */ temp = Fappend (2, args); } } return temp; } /* PQconnectdb Makes a new connection to a backend. PGconn *PQconnectdb(const char *conninfo) */ #ifdef HAVE_POSTGRESQLV7 #define USED_IF_V7(x) x #else #define USED_IF_V7(x) UNUSED (x) #endif static Lisp_Object postgresql_connect (Lisp_Object conninfo, int USED_IF_V7 (async)) { PGconn *P; Lisp_PGconn *lisp_pgconn; CHECK_STRING (conninfo); P = ( #ifdef HAVE_POSTGRESQLV7 async ? PQconnectStart : #endif PQconnectdb) (LISP_STRING_TO_EXTERNAL (conninfo, PG_OS_CODING)); if (P && (PQstatus (P) == CONNECTION_OK)) { (void) PQsetNoticeProcessor (P, xemacs_notice_processor, NULL); lisp_pgconn = allocate_pgconn (); lisp_pgconn->pgconn = P; return make_pgconn (lisp_pgconn); } else { /* Connection failed. Destroy the connection and signal an error. */ Lisp_Object errmsg; if (P) { errmsg = build_extstring (PQerrorMessage (P), PG_OS_CODING); PQfinish (P); } else errmsg = build_msg_string ("Out of Memory?"); signal_error (Qprocess_error, "Connecting to PostGreSQL backend", errmsg); } } /* ###autoload */ DEFUN ("pq-connectdb", Fpq_connectdb, 1, 1, 0, /* Make a new connection to a PostgreSQL backend. */ (conninfo)) { return postgresql_connect (conninfo, 0); } /* PQconnectStart Makes a new asynchronous connection to a backend. PGconn *PQconnectStart(const char *conninfo) */ #ifdef HAVE_POSTGRESQLV7 /* ###autoload */ DEFUN ("pq-connect-start", Fpq_connect_start, 1, 1, 0, /* Make a new asynchronous connection to a PostgreSQL backend. */ (conninfo)) { return postgresql_connect (conninfo, 1); } DEFUN ("pq-connect-poll", Fpq_connect_poll, 1, 1, 0, /* Poll an asynchronous connection for completion */ (conn)) { PGconn *P; PostgresPollingStatusType polling_status; CHECK_PGCONN (conn); P = (XPGCONN (conn))->pgconn; CHECK_LIVE_CONNECTION (P); polling_status = PQconnectPoll (P); switch (polling_status) { case PGRES_POLLING_FAILED: /* Something Bad has happened */ SIGNAL_ERROR (P, "Polling asynchronous connection"); case PGRES_POLLING_OK: return Qpgres_polling_ok; case PGRES_POLLING_READING: return Qpgres_polling_reading; case PGRES_POLLING_WRITING: return Qpgres_polling_writing; case PGRES_POLLING_ACTIVE: return Qpgres_polling_active; default: /* they've added a new field we don't know about */ signal_ferror (Qprocess_error, "Help! Unknown status code %08x from backend!", polling_status); } } #ifdef MULE DEFUN ("pq-client-encoding", Fpq_client_encoding, 1, 1, 0, /* Return client coding system. */ (conn)) { PGconn *P; CHECK_PGCONN (conn); P = (XPGCONN (conn))->pgconn; CHECK_LIVE_CONNECTION (P); return make_int (PQclientEncoding (P)); } DEFUN ("pq-set-client-encoding", Fpq_set_client_encoding, 2, 2, 0, /* Set client coding system. */ (conn, encoding)) { PGconn *P; int rc; char *c_encoding; CHECK_PGCONN (conn); CHECK_STRING (encoding); P = (XPGCONN (conn))->pgconn; CHECK_LIVE_CONNECTION (P); TO_EXTERNAL_FORMAT (LISP_STRING, encoding, C_STRING_ALLOCA, c_encoding, Qnative); if ((rc = PQsetClientEncoding (P, c_encoding)) < 0) signal_error (Qinvalid_argument, "bad encoding", Qunbound); else return make_int (rc); } #endif #endif /* HAVE_POSTGRESQLV7 */ /* PQfinish Close the connection to the backend. Also frees memory used by the PGconn object. void PQfinish(PGconn *conn) */ DEFUN ("pq-finish", Fpq_finish, 1, 1, 0, /* Close the connection to the backend. */ (conn)) { PGconn *P; CHECK_PGCONN (conn); P = (XPGCONN (conn))->pgconn; PUKE_IF_NULL (P); PQfinish (P); /* #### PQfinish deallocates the PGconn structure, so we now have a dangling pointer. */ /* Genocided all @'s ... */ (XPGCONN (conn))->pgconn = (PGconn *)NULL; /* You feel DEAD inside */ return Qnil; } DEFUN ("pq-clear", Fpq_clear, 1, 1, 0, /* Forcibly erase a PGresult object. */ (res)) { PGresult *R; CHECK_PGRESULT (res); R = (XPGRESULT (res))->pgresult; PUKE_IF_NULL (R); PQclear (R); /* Genocided all @'s ... */ (XPGRESULT (res))->pgresult = (PGresult *)NULL; /* You feel DEAD inside */ return Qnil; } DEFUN ("pq-is-busy", Fpq_is_busy, 1, 1, 0, /* Return t if PQgetResult would block waiting for input. */ (conn)) { PGconn *P; CHECK_PGCONN (conn); P = (XPGCONN (conn))->pgconn; CHECK_LIVE_CONNECTION (P); return PQisBusy (P) ? Qt : Qnil; } DEFUN ("pq-consume-input", Fpq_consume_input, 1, 1, 0, /* Consume any available input from the backend. Returns nil if something bad happened. */ (conn)) { PGconn *P; CHECK_PGCONN (conn); P = (XPGCONN (conn))->pgconn; CHECK_LIVE_CONNECTION (P); return PQconsumeInput (P) ? Qt : Qnil; } /* PQreset Reset the communication port with the backend. void PQreset(PGconn *conn) */ DEFUN ("pq-reset", Fpq_reset, 1, 1, 0, /* Reset the connection to the backend. This function will close the connection to the backend and attempt to reestablish a new connection to the same postmaster, using all the same parameters previously used. This may be useful for error recovery if a working connection is lost. */ (conn)) { PGconn *P; CHECK_PGCONN (conn); P = (XPGCONN (conn))->pgconn; PUKE_IF_NULL (P);/* we can resurrect a BAD connection, but not a dead one. */ PQreset (P); return Qnil; } #ifdef HAVE_POSTGRESQLV7 DEFUN ("pq-reset-start", Fpq_reset_start, 1, 1, 0, /* Reset connection to the backend asynchronously. */ (conn)) { PGconn *P; CHECK_PGCONN (conn); P = (XPGCONN (conn))->pgconn; CHECK_LIVE_CONNECTION (P); if (PQresetStart (P)) return Qt; SIGNAL_ERROR (P, "Resetting connection"); } DEFUN ("pq-reset-poll", Fpq_reset_poll, 1, 1, 0, /* Poll an asynchronous reset for completion. */ (conn)) { PGconn *P; PostgresPollingStatusType polling_status; CHECK_PGCONN (conn); P = (XPGCONN (conn))->pgconn; CHECK_LIVE_CONNECTION (P); polling_status = PQresetPoll (P); switch (polling_status) { case PGRES_POLLING_FAILED: SIGNAL_ERROR (P, "Polling asynchronous reset"); case PGRES_POLLING_OK: return Qpgres_polling_ok; case PGRES_POLLING_READING: return Qpgres_polling_reading; case PGRES_POLLING_WRITING: return Qpgres_polling_writing; case PGRES_POLLING_ACTIVE: return Qpgres_polling_active; default: /* they've added a new field we don't know about */ signal_ferror (Qprocess_error, "Help! Unknown status code %08x from backend!", polling_status); } } #endif DEFUN ("pq-request-cancel", Fpq_request_cancel, 1, 1, 0, /* Attempt to request cancellation of the current operation. The return value is t if the cancel request was successfully dispatched, nil if not (in which case conn->errorMessage is set). Note: successful dispatch is no guarantee that there will be any effect at the backend. The application must read the operation result as usual. */ (conn)) { PGconn *P; CHECK_PGCONN (conn); P = (XPGCONN (conn))->pgconn; CHECK_LIVE_CONNECTION (P); return PQrequestCancel (P) ? Qt : Qnil; } /* accessor function for the PGconn object */ DEFUN ("pq-pgconn", Fpq_pgconn, 2, 2, 0, /* Accessor function for the PGconn object. Currently recognized symbols for the field: pq::db Database name pq::user Database user name pq::pass Database user's password pq::host Hostname of PostgreSQL backend connected to pq::port TCP port number of connection pq::tty Debugging TTY (not used in Emacs) pq::options Additional backend options pq::status Connection status (either OK or BAD) pq::error-message Last error message from the backend pq::backend-pid Process ID of backend process */ (conn, field)) { PGconn *P; CHECK_PGCONN (conn); P = (XPGCONN (conn))->pgconn; PUKE_IF_NULL (P); /* BAD connections still have state to query */ if (EQ(field, Qpqdb)) /* PQdb Returns the database name of the connection. char *PQdb(PGconn *conn) */ return build_extstring (PQdb(P), PG_OS_CODING); else if (EQ (field, Qpquser)) /* PQuser Returns the user name of the connection. char *PQuser(PGconn *conn) */ return build_extstring (PQuser(P), PG_OS_CODING); else if (EQ (field, Qpqpass)) /* PQpass Returns the password of the connection. char *PQpass(PGconn *conn) */ return build_extstring (PQpass(P), PG_OS_CODING); else if (EQ (field, Qpqhost)) /* PQhost Returns the server host name of the connection. char *PQhost(PGconn *conn) */ return build_extstring (PQhost(P), PG_OS_CODING); else if (EQ (field, Qpqport)) { char *p; /* PQport Returns the port of the connection. char *PQport(PGconn *conn) */ if ((p = PQport(P))) return make_int(atoi(p)); else return make_int(-1); } else if (EQ (field, Qpqtty)) /* PQtty Returns the debug tty of the connection. char *PQtty(PGconn *conn) */ return build_extstring (PQtty(P), PG_OS_CODING); else if (EQ (field, Qpqoptions)) /* PQoptions Returns the backend options used in the connection. char *PQoptions(PGconn *conn) */ return build_extstring (PQoptions(P), PG_OS_CODING); else if (EQ (field, Qpqstatus)) { ConnStatusType cst; /* PQstatus Returns the status of the connection. The status can be CONNECTION_OK or CONNECTION_BAD. ConnStatusType PQstatus(PGconn *conn) */ switch ((cst = PQstatus (P))) { case CONNECTION_OK: return Qpg_connection_ok; case CONNECTION_BAD: return Qpg_connection_bad; #ifdef HAVE_POSTGRESQLV7 case CONNECTION_STARTED: return Qpg_connection_started; case CONNECTION_MADE: return Qpg_connection_made; case CONNECTION_AWAITING_RESPONSE: return Qpg_connection_awaiting_response; case CONNECTION_AUTH_OK: return Qpg_connection_auth_ok; case CONNECTION_SETENV: return Qpg_connection_setenv; #endif /* HAVE_POSTGRESQLV7 */ default: /* they've added a new field we don't know about */ signal_ferror (Qprocess_error, "Help! Unknown connection status code %08x from backend!", cst); } } else if (EQ (field, Qpqerrormessage)) /* PQerrorMessage Returns the error message most recently generated by an operation on the connection. char *PQerrorMessage(PGconn* conn); */ return build_extstring (PQerrorMessage(P), PG_OS_CODING); else if (EQ (field, Qpqbackendpid)) /* PQbackendPID Returns the process ID of the backend server handling this connection. int PQbackendPID(PGconn *conn); */ return make_int (PQbackendPID(P)); else signal_error (Qinvalid_argument, "bad PGconn accessor", Qunbound); } /* Query functions */ DEFUN ("pq-exec", Fpq_exec, 2, 2, 0, /* Submit a query to Postgres and wait for the result. */ (conn, query)) { PGconn *P; Lisp_PGresult *lisp_pgresult; PGresult *R; char *c_query; CHECK_PGCONN (conn); CHECK_STRING (query); P = (XPGCONN (conn))->pgconn; CHECK_LIVE_CONNECTION (P); TO_EXTERNAL_FORMAT (LISP_STRING, query, C_STRING_ALLOCA, c_query, Qnative); R = PQexec (P, c_query); { const Ascbyte *tag; char buf[BLCKSZ]; if (!R) out_of_memory ("query: out of memory", Qunbound); else switch (PQresultStatus (R)) { case PGRES_BAD_RESPONSE: tag = "bad response [%s]"; goto err; case PGRES_NONFATAL_ERROR: tag = "non-fatal error [%s]"; goto err; case PGRES_FATAL_ERROR: tag = "fatal error [%s]"; err: strncpy (buf, PQresultErrorMessage (R), sizeof (buf)); buf [sizeof (buf) - 1] = '\0'; PQclear (R); signal_ferror (Qprocess_error, tag, buf); /*NOTREACHED*/ default: break; } } lisp_pgresult = allocate_pgresult (); lisp_pgresult->pgresult = R; return make_pgresult (lisp_pgresult); } DEFUN ("pq-send-query", Fpq_send_query, 2, 2, 0, /* Submit a query to Postgres and don't wait for the result. Returns: t if successfully submitted nil if error (conn->errorMessage is set) */ (conn, query)) { PGconn *P; char *c_query; CHECK_PGCONN (conn); CHECK_STRING (query); P = (XPGCONN (conn))->pgconn; CHECK_LIVE_CONNECTION (P); TO_EXTERNAL_FORMAT (LISP_STRING, query, C_STRING_ALLOCA, c_query, Qnative); if (PQsendQuery (P, c_query)) return Qt; else SIGNAL_ERROR (P, "Sending asynchronous query"); } DEFUN ("pq-get-result", Fpq_get_result, 1, 1, 0, /* Retrieve an asynchronous result from a query. NIL is returned when no more query work remains. */ (conn)) { PGconn *P; Lisp_PGresult *lisp_pgresult; PGresult *R; CHECK_PGCONN (conn); P = (XPGCONN (conn))->pgconn; CHECK_LIVE_CONNECTION (P); R = PQgetResult (P); if (!R) return Qnil; /* not an error, there's no more data to get */ { const Ascbyte *tag; char buf[BLCKSZ]; switch (PQresultStatus (R)) { case PGRES_BAD_RESPONSE: tag = "bad response [%s]"; goto err; case PGRES_NONFATAL_ERROR: tag = "non-fatal error [%s]"; goto err; case PGRES_FATAL_ERROR: tag = "fatal error [%s]"; err: strncpy (buf, PQresultErrorMessage (R), sizeof (buf)); buf[sizeof (buf) - 1] = '\0'; PQclear (R); signal_ferror (Qprocess_error, tag, buf); /*NOTREACHED*/ default: break; } } lisp_pgresult = allocate_pgresult(); lisp_pgresult->pgresult = R; return make_pgresult (lisp_pgresult); } DEFUN ("pq-result-status", Fpq_result_status, 1, 1, 0, /* Return result status of the query. */ (result)) { PGresult *R; ExecStatusType est; CHECK_PGRESULT (result); R = (XPGRESULT (result))->pgresult; PUKE_IF_NULL (R); switch ((est = PQresultStatus (R))) { case PGRES_EMPTY_QUERY: return Qpgres_empty_query; case PGRES_COMMAND_OK: return Qpgres_command_ok; case PGRES_TUPLES_OK: return Qpgres_tuples_ok; case PGRES_COPY_OUT: return Qpgres_copy_out; case PGRES_COPY_IN: return Qpgres_copy_in; case PGRES_BAD_RESPONSE: return Qpgres_bad_response; case PGRES_NONFATAL_ERROR: return Qpgres_nonfatal_error; case PGRES_FATAL_ERROR: return Qpgres_fatal_error; default: /* they've added a new field we don't know about */ signal_ferror (Qprocess_error, "Help! Unknown exec status code %08x from backend!", est); } } DEFUN ("pq-res-status", Fpq_res_status, 1, 1, 0, /* Return stringified result status of the query. */ (result)) { PGresult *R; CHECK_PGRESULT (result); R = (XPGRESULT (result))->pgresult; PUKE_IF_NULL (R); return build_extstring (PQresStatus (PQresultStatus (R)), PG_OS_CODING); } /* Sundry PGresult accessor functions */ DEFUN ("pq-result-error-message", Fpq_result_error_message, 1, 1, 0, /* Return last message associated with the query. */ (result)) { PGresult *R; CHECK_PGRESULT (result); R = (XPGRESULT (result))->pgresult; PUKE_IF_NULL (R); return build_extstring (PQresultErrorMessage (R), PG_OS_CODING); } DEFUN ("pq-ntuples", Fpq_ntuples, 1, 1, 0, /* Return the number of tuples (instances) in the query result. */ (result)) { PGresult *R; CHECK_PGRESULT (result); R = (XPGRESULT (result))->pgresult; PUKE_IF_NULL (R); return make_int (PQntuples (R)); } DEFUN ("pq-nfields", Fpq_nfields, 1, 1, 0, /* Return the number of fields (attributes) in each tuple of the query result. */ (result)) { PGresult *R; CHECK_PGRESULT (result); R = (XPGRESULT (result))->pgresult; PUKE_IF_NULL (R); return make_int (PQnfields (R)); } DEFUN ("pq-binary-tuples", Fpq_binary_tuples, 1, 1, 0, /* Return t if the query result contains binary data, nil otherwise. */ (result)) { PGresult *R; CHECK_PGRESULT (result); R = (XPGRESULT (result))->pgresult; PUKE_IF_NULL (R); return (PQbinaryTuples (R)) ? Qt : Qnil; } DEFUN ("pq-fname", Fpq_fname, 2, 2, 0, /* Return the field (attribute) name associated with the given field index. Field indices start at 0. */ (result, field_index)) { PGresult *R; CHECK_PGRESULT (result); CHECK_INT (field_index); R = (XPGRESULT (result))->pgresult; PUKE_IF_NULL (R); return build_extstring (PQfname (R, XINT (field_index)), PG_OS_CODING); } DEFUN ("pq-fnumber", Fpq_fnumber, 2, 2, 0, /* Return the number of fields (attributes) in each tuple of the query result. */ (result, field_name)) { PGresult *R; char *c_field_name; CHECK_PGRESULT (result); CHECK_STRING (field_name); R = (XPGRESULT (result))->pgresult; PUKE_IF_NULL (R); TO_EXTERNAL_FORMAT (LISP_STRING, field_name, C_STRING_ALLOCA, c_field_name, Qnative); return make_int (PQfnumber (R, c_field_name)); } DEFUN ("pq-ftype", Fpq_ftype, 2, 2, 0, /* Return the field type associated with the given field index. The integer returned is the internal coding of the type. Field indices start at 0. */ (result, field_num)) { PGresult *R; CHECK_PGRESULT (result); CHECK_INT (field_num); R = (XPGRESULT (result))->pgresult; PUKE_IF_NULL (R); return make_int (PQftype (R, XINT (field_num))); } DEFUN ("pq-fsize", Fpq_fsize, 2, 2, 0, /* Return the field size in bytes associated with the given field index. Field indices start at 0. */ (result, field_index)) { PGresult *R; CHECK_PGRESULT (result); CHECK_INT (field_index); R = (XPGRESULT (result))->pgresult; PUKE_IF_NULL (R); return make_int (PQftype (R, XINT (field_index))); } DEFUN ("pq-fmod", Fpq_fmod, 2, 2, 0, /* Return the type modifier associated with a field. Field indices start at 0. */ (result, field_index)) { PGresult *R; CHECK_PGRESULT (result); CHECK_INT (field_index); R = (XPGRESULT (result))->pgresult; PUKE_IF_NULL (R); return make_int (PQfmod (R, XINT (field_index))); } DEFUN ("pq-get-value", Fpq_get_value, 3, 3, 0, /* Return a single field (attribute) value of one tuple of a PGresult. Tuple and field indices start at 0. */ (result, tup_num, field_num)) { PGresult *R; CHECK_PGRESULT (result); CHECK_INT (tup_num); CHECK_INT (field_num); R = (XPGRESULT (result))->pgresult; PUKE_IF_NULL (R); return build_extstring (PQgetvalue (R, XINT (tup_num), XINT (field_num)), PG_OS_CODING); } DEFUN ("pq-get-length", Fpq_get_length, 3, 3, 0, /* Returns the length of a field value in bytes. If result is binary, i.e. a result of a binary portal, then the length returned does NOT include the size field of the varlena. (The data returned by PQgetvalue doesn't either.) */ (result, tup_num, field_num)) { PGresult *R; CHECK_PGRESULT (result); CHECK_INT (tup_num); CHECK_INT (field_num); R = (XPGRESULT (result))->pgresult; PUKE_IF_NULL (R); return make_int (PQgetlength (R, XINT (tup_num), XINT (field_num))); } DEFUN ("pq-get-is-null", Fpq_get_is_null, 3, 3, 0, /* Returns the null status of a field value. */ (result, tup_num, field_num)) { PGresult *R; CHECK_PGRESULT (result); CHECK_INT (tup_num); CHECK_INT (field_num); R = (XPGRESULT (result))->pgresult; PUKE_IF_NULL (R); return PQgetisnull (R, XINT (tup_num), XINT (field_num)) ? Qt : Qnil; } DEFUN ("pq-cmd-status", Fpq_cmd_status, 1, 1, 0, /* Returns the command status string from the SQL command that generated the result. */ (result)) { PGresult *R; CHECK_PGRESULT (result); R = (XPGRESULT (result))->pgresult; PUKE_IF_NULL (R); return build_extstring (PQcmdStatus (R), PG_OS_CODING); } DEFUN ("pq-cmd-tuples", Fpq_cmd_tuples, 1, 1, 0, /* Returns the number of rows affected by the SQL command. */ (result)) { PGresult *R; CHECK_PGRESULT (result); R = (XPGRESULT (result))->pgresult; PUKE_IF_NULL (R); return build_extstring (PQcmdTuples (R), PG_OS_CODING); } DEFUN ("pq-oid-value", Fpq_oid_value, 1, 1, 0, /* Returns the object id of the tuple inserted. */ (result)) { PGresult *R; CHECK_PGRESULT (result); R = (XPGRESULT (result))->pgresult; PUKE_IF_NULL (R); #ifdef HAVE_POSTGRESQLV7 return make_int (PQoidValue (R)); #else /* Use the old interface */ return make_int (atoi (PQoidStatus (R))); #endif } #ifdef HAVE_POSTGRESQLV7 DEFUN ("pq-set-nonblocking", Fpq_set_nonblocking, 2, 2, 0, /* Sets the PGconn's database connection non-blocking if the arg is TRUE or makes it non-blocking if the arg is FALSE, this will not protect you from PQexec(), you'll only be safe when using the non-blocking API. Needs to be called only on a connected database connection. */ (conn, arg)) { PGconn *P; CHECK_PGCONN (conn); P = (XPGCONN (conn))->pgconn; CHECK_LIVE_CONNECTION (P); return make_int (PQsetnonblocking (P, !NILP (arg))); } DEFUN ("pq-is-nonblocking", Fpq_is_nonblocking, 1, 1, 0, /* Return the blocking status of the database connection. */ (conn)) { PGconn *P; CHECK_PGCONN (conn); P = (XPGCONN (conn))->pgconn; CHECK_LIVE_CONNECTION (P); return PQisnonblocking (P) ? Qt : Qnil; } DEFUN ("pq-flush", Fpq_flush, 1, 1, 0, /* Force the write buffer to be written (or at least try). */ (conn)) { PGconn *P; CHECK_PGCONN (conn); P = (XPGCONN (conn))->pgconn; CHECK_LIVE_CONNECTION (P); return make_int (PQflush (P)); } #endif DEFUN ("pq-notifies", Fpq_notifies, 1, 1, 0, /* Return the latest async notification that has not yet been handled. If there has been a notification, then a list of two elements will be returned. The first element contains the relation name being notified, the second element contains the backend process ID number. nil is returned if there aren't any notifications to process. */ (conn)) { /* This function cannot GC */ PGconn *P; PGnotify *PGN; CHECK_PGCONN (conn); P = (XPGCONN (conn))->pgconn; CHECK_LIVE_CONNECTION (P); PGN = PQnotifies (P); if (!PGN) return Qnil; else { Lisp_Object temp; temp = list2 (build_extstring (PGN->relname, PG_OS_CODING), make_int (PGN->be_pid)); free ((void *)PGN); return temp; } } #if defined (HAVE_POSTGRESQLV7) && defined(MULE) /* ###autoload */ DEFUN ("pq-env-2-encoding", Fpq_env_2_encoding, 0, 0, 0, /* Get encoding id from environment variable PGCLIENTENCODING. */ ()) { return make_int (PQenv2encoding ()); } #endif /* MULE */ DEFUN ("pq-lo-import", Fpq_lo_import, 2, 2, 0, /* */ (conn, filename)) { PGconn *P; char *c_filename; CHECK_PGCONN (conn); CHECK_STRING (filename); P = (XPGCONN (conn))->pgconn; CHECK_LIVE_CONNECTION (P); LISP_PATHNAME_CONVERT_OUT (filename, c_filename); return make_int ((int)lo_import (P, c_filename)); } DEFUN ("pq-lo-export", Fpq_lo_export, 3, 3, 0, /* */ (conn, oid, filename)) { PGconn *P; char *c_filename; CHECK_PGCONN (conn); CHECK_INT (oid); CHECK_STRING (filename); P = (XPGCONN (conn))->pgconn; CHECK_LIVE_CONNECTION (P); LISP_PATHNAME_CONVERT_OUT (filename, c_filename); return make_int ((int)lo_export (P, XINT (oid), c_filename)); } DEFUN ("pq-make-empty-pgresult", Fpq_make_empty_pgresult, 2, 2, 0, /* Make an empty PGresult object with the given status. */ (conn, status)) { PGconn *P; Lisp_PGresult *lpgr; PGresult *R; ExecStatusType est; CHECK_PGCONN (conn); P = (XPGCONN (conn))->pgconn; CHECK_LIVE_CONNECTION (P); /* needed here? */ if (EQ (status, Qpgres_empty_query)) est = PGRES_EMPTY_QUERY; else if (EQ (status, Qpgres_command_ok)) est = PGRES_COMMAND_OK; else if (EQ (status, Qpgres_tuples_ok)) est = PGRES_TUPLES_OK; else if (EQ (status, Qpgres_copy_out)) est = PGRES_COPY_OUT; else if (EQ (status, Qpgres_copy_in)) est = PGRES_COPY_IN; else if (EQ (status, Qpgres_bad_response)) est = PGRES_BAD_RESPONSE; else if (EQ (status, Qpgres_nonfatal_error)) est = PGRES_NONFATAL_ERROR; else if (EQ (status, Qpgres_fatal_error)) est = PGRES_FATAL_ERROR; else invalid_constant ("bad status symbol", status); R = PQmakeEmptyPGresult (P, est); if (!R) out_of_memory (0, Qunbound); lpgr = allocate_pgresult (); lpgr->pgresult = R; return make_pgresult (lpgr); } DEFUN ("pq-get-line", Fpq_get_line, 1, 1, 0, /* Retrieve a line from server in copy in operation. The return value is a dotted pair where the cons cell is an integer code: -1: Copying is complete 0: A record is complete 1: A record is incomplete, it will be continued in the next `pq-get-line' operation. and the cdr cell is returned string data. The copy operation is complete when the value `\.' (backslash dot) is returned. */ (conn)) { char buffer[BLCKSZ]; /* size of a Postgres disk block */ PGconn *P; int ret; CHECK_PGCONN (conn); P = (XPGCONN (conn))->pgconn; CHECK_LIVE_CONNECTION (P); ret = PQgetline (P, buffer, sizeof (buffer)); return Fcons (make_int (ret), build_extstring (buffer, PG_OS_CODING)); } DEFUN ("pq-put-line", Fpq_put_line, 2, 2, 0, /* Send a line to the server in copy out operation. Returns t if the operation succeeded, nil otherwise. */ (conn, string)) { PGconn *P; char *c_string; CHECK_PGCONN (conn); CHECK_STRING (string); P = (XPGCONN (conn))->pgconn; CHECK_LIVE_CONNECTION (P); TO_EXTERNAL_FORMAT (LISP_STRING, string, C_STRING_ALLOCA, c_string, Qnative); return !PQputline (P, c_string) ? Qt : Qnil; } DEFUN ("pq-get-line-async", Fpq_get_line_async, 1, 1, 0, /* Get a line from the server in copy in operation asynchronously. This routine is for applications that want to do "COPY <rel> to stdout" asynchronously, that is without blocking. Having issued the COPY command and gotten a PGRES_COPY_OUT response, the app should call PQconsumeInput and this routine until the end-of-data signal is detected. Unlike PQgetline, this routine takes responsibility for detecting end-of-data. On each call, PQgetlineAsync will return data if a complete newline- terminated data line is available in libpq's input buffer, or if the incoming data line is too long to fit in the buffer offered by the caller. Otherwise, no data is returned until the rest of the line arrives. If -1 is returned, the end-of-data signal has been recognized (and removed from libpq's input buffer). The caller *must* next call PQendcopy and then return to normal processing. RETURNS: -1 if the end-of-copy-data marker has been recognized 0 if no data is available >0 the number of bytes returned. The data returned will not extend beyond a newline character. If possible a whole line will be returned at one time. But if the buffer offered by the caller is too small to hold a line sent by the backend, then a partial data line will be returned. This can be detected by testing whether the last returned byte is '\n' or not. The returned string is *not* null-terminated. */ (conn)) { PGconn *P; char buffer[BLCKSZ]; int ret; CHECK_PGCONN (conn); P = (XPGCONN (conn))->pgconn; CHECK_LIVE_CONNECTION (P); ret = PQgetlineAsync (P, buffer, sizeof (buffer)); if (ret == -1) return Qt; /* done! */ else if (!ret) return Qnil; /* no data yet */ else return Fcons (make_int (ret), make_extstring ((Extbyte *) buffer, ret, PG_OS_CODING)); } DEFUN ("pq-put-nbytes", Fpq_put_nbytes, 2, 2, 0, /* Asynchronous copy out. */ (conn, data)) { /* NULs are not allowed. I don't think this matters at this time. */ PGconn *P; char *c_data; CHECK_PGCONN (conn); CHECK_STRING (data); P = (XPGCONN (conn))->pgconn; CHECK_LIVE_CONNECTION (P); TO_EXTERNAL_FORMAT (LISP_STRING, data, C_STRING_ALLOCA, c_data, Qnative); return !PQputnbytes (P, c_data, strlen (c_data)) ? Qt : Qnil; } DEFUN ("pq-end-copy", Fpq_end_copy, 1, 1, 0, /* End a copying operation. */ (conn)) { PGconn *P; CHECK_PGCONN (conn); P = (XPGCONN (conn))->pgconn; CHECK_LIVE_CONNECTION (P); return PQendcopy (P) ? Qt : Qnil; } void syms_of_postgresql(void) { #ifndef RUNNING_XEMACS_21_1 INIT_LISP_OBJECT (pgconn); INIT_LISP_OBJECT (pgresult); #endif DEFSYMBOL (Qpostgresql); /* opaque exported types */ DEFSYMBOL (Qpgconnp); DEFSYMBOL (Qpgresultp); /* connection status types */ defsymbol (&Qpg_connection_ok, "pg::connection-ok"); defsymbol (&Qpg_connection_bad, "pg::connection-bad"); defsymbol (&Qpg_connection_started, "pg::connection-started"); defsymbol (&Qpg_connection_made, "pg::connection-made"); defsymbol (&Qpg_connection_awaiting_response, "pg::connection-awaiting-response"); defsymbol (&Qpg_connection_auth_ok, "pg::connection-auth-ok"); defsymbol (&Qpg_connection_setenv, "pg::connection-setenv"); /* Fields of PGconn */ defsymbol (&Qpqdb, "pq::db"); defsymbol (&Qpquser, "pq::user"); defsymbol (&Qpqpass, "pq::pass"); defsymbol (&Qpqhost, "pq::host"); defsymbol (&Qpqport, "pq::port"); defsymbol (&Qpqtty, "pq::tty"); defsymbol (&Qpqoptions, "pq::options"); defsymbol (&Qpqstatus, "pq::status"); defsymbol (&Qpqerrormessage, "pq::error-message"); defsymbol (&Qpqbackendpid, "pq::backend-pid"); /* Query status results */ defsymbol (&Qpgres_empty_query, "pgres::empty-query"); defsymbol (&Qpgres_command_ok, "pgres::command-ok"); defsymbol (&Qpgres_tuples_ok, "pgres::tuples-ok"); defsymbol (&Qpgres_copy_out, "pgres::copy-out"); defsymbol (&Qpgres_copy_in, "pgres::copy-in"); defsymbol (&Qpgres_bad_response, "pgres::bad-response"); defsymbol (&Qpgres_nonfatal_error, "pgres::nonfatal-error"); defsymbol (&Qpgres_fatal_error, "pgres::fatal-error"); /* Poll status results */ defsymbol (&Qpgres_polling_failed, "pgres::polling-failed"); defsymbol (&Qpgres_polling_reading, "pgres::polling-reading"); defsymbol (&Qpgres_polling_writing, "pgres::polling-writing"); defsymbol (&Qpgres_polling_ok, "pgres::polling-ok"); defsymbol (&Qpgres_polling_active, "pgres::polling-active"); #ifdef HAVE_POSTGRESQLV7 DEFSUBR (Fpq_connect_start); DEFSUBR (Fpq_connect_poll); #ifdef MULE DEFSUBR (Fpq_client_encoding); DEFSUBR (Fpq_set_client_encoding); #endif /* MULE */ #endif /* HAVE_POSTGRESQLV7 */ DEFSUBR (Fpq_conn_defaults); DEFSUBR (Fpq_connectdb); DEFSUBR (Fpq_finish); DEFSUBR (Fpq_clear); DEFSUBR (Fpq_is_busy); DEFSUBR (Fpq_consume_input); DEFSUBR (Fpq_reset); #ifdef HAVE_POSTGRESQLV7 DEFSUBR (Fpq_reset_start); DEFSUBR (Fpq_reset_poll); #endif DEFSUBR (Fpq_request_cancel); DEFSUBR (Fpq_pgconn); DEFSUBR (Fpq_exec); DEFSUBR (Fpq_send_query); DEFSUBR (Fpq_get_result); DEFSUBR (Fpq_result_status); DEFSUBR (Fpq_res_status); DEFSUBR (Fpq_result_error_message); DEFSUBR (Fpq_ntuples); DEFSUBR (Fpq_nfields); DEFSUBR (Fpq_binary_tuples); DEFSUBR (Fpq_fname); DEFSUBR (Fpq_fnumber); DEFSUBR (Fpq_ftype); DEFSUBR (Fpq_fsize); DEFSUBR (Fpq_fmod); /***/ DEFSUBR (Fpq_get_value); DEFSUBR (Fpq_get_length); DEFSUBR (Fpq_get_is_null); DEFSUBR (Fpq_cmd_status); DEFSUBR (Fpq_cmd_tuples); DEFSUBR (Fpq_oid_value); #ifdef HAVE_POSTGRESQLV7 DEFSUBR (Fpq_set_nonblocking); DEFSUBR (Fpq_is_nonblocking); DEFSUBR (Fpq_flush); #endif DEFSUBR (Fpq_notifies); #if defined (HAVE_POSTGRESQLV7) && defined(MULE) DEFSUBR (Fpq_env_2_encoding); #endif DEFSUBR (Fpq_lo_import); DEFSUBR (Fpq_lo_export); DEFSUBR (Fpq_make_empty_pgresult); /* copy in/out functions */ DEFSUBR (Fpq_get_line); DEFSUBR (Fpq_put_line); DEFSUBR (Fpq_get_line_async); DEFSUBR (Fpq_put_nbytes); DEFSUBR (Fpq_end_copy); } void vars_of_postgresql(void) { Fprovide (Qpostgresql); #ifdef HAVE_POSTGRESQLV7 Fprovide (intern ("postgresqlv7")); #endif #ifndef RUNNING_XEMACS_21_1 Vpg_coding_system = Qnative; DEFVAR_LISP ("pg-coding-system", &Vpg_coding_system /* Default Postgres client coding system. */ ); #endif DEFVAR_LISP ("pg:host", &VXPGHOST /* Default PostgreSQL server name. If not set, the server running on the local host is used. The initial value is set from the PGHOST environment variable. */ ); DEFVAR_LISP ("pg:user", &VXPGUSER /* Default PostgreSQL user name. This value is used when connecting to a database for authentication. The initial value is set from the PGUSER environment variable. */ ); DEFVAR_LISP ("pg:options", &VXPGOPTIONS /* Default PostgreSQL user name. This value is used when connecting to a database for authentication. The initial value is set from the PGUSER environment variable. */ ); DEFVAR_LISP ("pg:port", &VXPGPORT /* Default port to connect to PostgreSQL backend. This value is used when connecting to a database. The initial value is set from the PGPORT environment variable. */ ); DEFVAR_LISP ("pg:tty", &VXPGTTY /* Default debugging TTY. There is no useful setting of this variable in the XEmacs Lisp API. The initial value is set from the PGTTY environment variable. */ ); DEFVAR_LISP ("pg:database", &VXPGDATABASE /* Default database to connect to. The initial value is set from the PGDATABASE environment variable. */ ); DEFVAR_LISP ("pg:realm", &VXPGREALM /* Default kerberos realm to use for authentication. The initial value is set from the PGREALM environment variable. */ ); #ifdef MULE /* It's not clear whether this is any use. My intent is to autodetect the coding system from the database. */ DEFVAR_LISP ("pg:client-encoding", &VXPGCLIENTENCODING /* Default client encoding to use. The initial value is set from the PGCLIENTENCODING environment variable. */ ); #endif #if !defined(HAVE_POSTGRESQLV7) DEFVAR_LISP ("pg:authtype", &VXPGAUTHTYPE /* Default authentication to use. The initial value is set from the PGAUTHTYPE environment variable. WARNING: This variable has gone away in versions of PostgreSQL newer than 6.5. */ ); #endif DEFVAR_LISP ("pg:geqo", &VXPGGEQO /* Genetic Query Optimizer options. The initial value is set from the PGGEQO environment variable. */ ); DEFVAR_LISP ("pg:cost-index", &VXPGCOSTINDEX /* Default cost index options. The initial value is set from the PGCOSTINDEX environment variable. */ ); DEFVAR_LISP ("pg:cost-heap", &VXPGCOSTHEAP /* Default cost heap options. The initial value is set from the PGCOSTHEAP environment variable. */ ); DEFVAR_LISP ("pg:tz", &VXPGTZ /* Default timezone to use. The initial value is set from the PGTZ environment variable. */ ); DEFVAR_LISP ("pg:date-style", &VXPGDATESTYLE /* Default date style to use. The initial value is set from the PGDATESTYLE environment variable. */ ); #ifdef HAVE_SHLIB /* If we are building this as a module, we need the initializing function to run at module load time. */ init_postgresql_from_environment (); #endif } /* These initializations should not be done at dump-time. */ void init_postgresql_from_environment (void) { Ibyte *p; #define FROB(envvar, var) \ if ((p = egetenv (envvar))) \ var = build_istring (p); \ else \ var = Qnil if (initialized) { FROB ("PGHOST", VXPGHOST); FROB ("PGUSER", VXPGUSER); FROB ("PGOPTIONS", VXPGOPTIONS); if ((p = egetenv ("PGPORT"))) VXPGPORT = make_int (atoi ((char *) p)); else VXPGPORT = Qnil; FROB ("PGTTY", VXPGTTY); FROB ("PGDATABASE", VXPGDATABASE); FROB ("PGREALM", VXPGREALM); #ifdef MULE /* It's not clear whether this is any use. My intent is to autodetect the coding system from the database. */ FROB ("PGCLIENTENCODING", VXPGCLIENTENCODING); #endif #if !defined(HAVE_POSTGRESQLV7) FROB ("PGAUTHTYPE", VXPGAUTHTYPE); #endif FROB ("PGGEQO", VXPGGEQO); FROB ("PGCOSTINDEX", VXPGCOSTINDEX); FROB ("PGCOSTHEAP", VXPGCOSTHEAP); FROB ("PGTZ", VXPGTZ); FROB ("PGDATESTYLE", VXPGDATESTYLE); #undef FROB } } #ifdef HAVE_SHLIB EXTERN_C void unload_postgresql (void); void unload_postgresql (void) { #ifndef RUNNING_XEMACS_21_1 /* Remove defined types */ UNDEF_LISP_OBJECT (pgconn); UNDEF_LISP_OBJECT (pgresult); #endif /* Remove staticpro'ing of symbols */ unstaticpro_nodump (&Qpostgresql); unstaticpro_nodump (&Qpgconnp); unstaticpro_nodump (&Qpgresultp); unstaticpro_nodump (&Qpg_connection_ok); unstaticpro_nodump (&Qpg_connection_bad); unstaticpro_nodump (&Qpg_connection_started); unstaticpro_nodump (&Qpg_connection_made); unstaticpro_nodump (&Qpg_connection_awaiting_response); unstaticpro_nodump (&Qpg_connection_auth_ok); unstaticpro_nodump (&Qpg_connection_setenv); unstaticpro_nodump (&Qpqdb); unstaticpro_nodump (&Qpquser); unstaticpro_nodump (&Qpqpass); unstaticpro_nodump (&Qpqhost); unstaticpro_nodump (&Qpqport); unstaticpro_nodump (&Qpqtty); unstaticpro_nodump (&Qpqoptions); unstaticpro_nodump (&Qpqstatus); unstaticpro_nodump (&Qpqerrormessage); unstaticpro_nodump (&Qpqbackendpid); unstaticpro_nodump (&Qpgres_empty_query); unstaticpro_nodump (&Qpgres_command_ok); unstaticpro_nodump (&Qpgres_tuples_ok); unstaticpro_nodump (&Qpgres_copy_out); unstaticpro_nodump (&Qpgres_copy_in); unstaticpro_nodump (&Qpgres_bad_response); unstaticpro_nodump (&Qpgres_nonfatal_error); unstaticpro_nodump (&Qpgres_fatal_error); unstaticpro_nodump (&Qpgres_polling_failed); unstaticpro_nodump (&Qpgres_polling_reading); unstaticpro_nodump (&Qpgres_polling_writing); unstaticpro_nodump (&Qpgres_polling_ok); unstaticpro_nodump (&Qpgres_polling_active); } #endif /* HAVE_SHLIB */