changeset 1204:e22b0213b713

[xemacs-hg @ 2003-01-12 11:07:58 by michaels] modules/ChangeLog: 2002-12-16 Ben Wing <ben@xemacs.org> * postgresql/postgresql.c: remove ifdef USE_KKCC. src/ChangeLog: 2003-01-08 Mike Sperber <mike@xemacs.org> * console.h (CDFW_CONSOLE): Don't lead to a crash if we're dealing with a dead window/frame/device/console. 2002-12-20 Mike Sperber <mike@xemacs.org> * ui-gtk.c: Fix typo from Ben's patch: emacs_ffi_data is a typedef, not a struct. emacs_gtk_object_data is a typedef, not a struct. * gtk-glue.c (gdk_event_to_emacs_event): Fix typos from Ben's patch: le -> emacs_event + rearrange the code. * event-gtk.c (gtk_event_to_emacs_event): Fix typos from Ben's patch: ..._UNDERLYING_GDK_EVENT -> ..._GDK_EVENT, ev -> key_event. * device-gtk.c: Fix typo from Ben's patch: x_keysym_map_hash_table -> x_keysym_map_hashtable. 2002-12-19 Mike Sperber <mike@xemacs.org> * menubar-x.c (set_frame_menubar): Initialize protect_me field of popup_data. 2002-12-16 Ben Wing <ben@xemacs.org> Major cleanup of KKCC, etc. KKCC, pdump-related: -- descriptions are written for all objects. this required some changes in the format of some objects, e.g. extents, popup-data, coding system, lstream, lcrecord-list. -- KKCC now handles weakness in markers, hash tables, elsewhere correctly (formerly, you'd eventually get a stack overflow due to endlessly expanding markers). -- textual changes: lrecord_description -> memory_description, struct_description -> sized_memory_description. -- extensive comment describing descriptions and pdump. -- redo XD_UNION so it works inline and change its format to provide sufficient info for pdump. implement XD_UNION in pdump. also add XD_UNION_DYNAMIC_SIZE, which works like XD_UNION except for when auto-computing structure sizes. -- add support for XD_INDIRECT in description offsets (used by extents). -- add support for "description maps", allowing for indirect descriptions that are retrieved from an object at run-time. this generalizes XD_CODING_SYSTEM_END, XD_SPECIFIER_END, etc., which have now been eliminated. -- add a fifth field "flags" to memory_description, to support flags that can be specified for this particular line. Currently defined flags are XD_FLAG_NO_KKCC (KKCC should ignore this entry; useful for the weakness above in markers, etc.), XD_FLAG_NO_PDUMP (pdump should ignore this entry), XD_FLAG_UNION_DEFAULT_ENTRY (in union maps, this specifies a "default" entry for all remaining values), and XD_FLAG_FREE_LISP_OBJECT (for use with lcrecord-lists). -- clean up the kkcc-itis in events, so that the differences between event data as separate objects and as a union are now minimized to a small number of places. with the new XD_UNION, we no longer need event data as separate objects, so this code is no longer ifdef USE_KKCC, but instead ifdef EVENT_DATA_AS_OBJECTS, not used by default. make sure that we explicitly free the separate event data objects when no longer in use, to maintain the invariant the event processing causes no consing. -- also remove other USE_KKCC ifdefs when not necessary. -- allow for KKCC compilation under MS Windows. -- fix README.kkcc. -- dump_add_root_object -> dump_add_root_lisp_object. -- implement dump_add_root_block and use this to handle dump_add_opaque. -- factor out some code duplicated in kkcc and pdump. Other allocation/object-related: -- change various *slots.h so MARKED_SLOT() call no longer includes semicolon. -- free_marker() takes a Lisp_Object not a direct pointer. -- make bit vectors lcrecords, like vectors, and eliminate code that essentially duplicated the lcrecord handling. -- additional asserts in FREE_FIXED_TYPE, formerly duplicated in the various callers of this. -- all lcrecord allocation functions now zero out the returned lcrecords. unnecessary calls to zero_lcrecord removed. add long comment describing these functions. -- extract out process and coding system slots, like for buffers, frames, etc. -- lcrecords now set the type of items sitting on the free list to lcrecord_type_free. -- changes to the way that gap arrays are allocated, for kkcc's benefit -- now, one single memory block with a stretchy array on the end, instead of a separate block holding the array. Error-checking-related: -- now can compile with C++ under MS Windows. clean up compile errors discovered that way. (a few were real problems) -- add C++ error-checking code to verify problems with mismatched GCPRO/UNGCPRO. (there were a few in the kkcc code.) add long comment about how to catch insufficient GCPRO (yes, it's possible using C++). -- add debug_p4(), a simple object printer, when debug_print() doesn't work. -- add dp() and db() as short synonyms of debug_print(), debug_backtrace(). -- `print' tries EXTREMELY hard to avoid core dumping when printing when crashing or from debug_print(), and tries as hard as it reasonably can in other situations. -- Correct the message output upon crashing to be more up-to-date. Event-related: -- document event-matches-key-specifier-p better. -- generalize the dispatch queues formerly duplicated in the various event implementations. add event methods to drain pending events. generalize and clean up QUIT handling, removing event-specific quit processing. allow arbitrary keystrokes, not just ASCII, to be the QUIT char. among other things, this should fix some longstanding bugs in X quit handling. long comment describing the various event queues. -- implement delaying of XFlush() if there are pending expose events. SOMEONE PLEASE TRY THIS OUT. -- Fix `xemacs -batch -l dunnet' under Cygwin. Try to fix under MS Windows but not quite there yet. Other: -- class -> class_ and no more C++ games with this item. new -> new_ in the lwlib code, so far not elsewhere. -- use `struct htentry' not `struct hentry' in elhash.c to avoid debugger confusion with hash.c. -- new macros ALIST_LOOP_3, ALIST_LOOP_4. * README.kkcc: * alloc.c: * alloc.c (deadbeef_memory): * alloc.c (allocate_lisp_storage): * alloc.c (copy_lisp_object): * alloc.c (ALLOCATE_FIXED_TYPE_1): * alloc.c (FREE_FIXED_TYPE): * alloc.c (make_vector_internal): * alloc.c (make_bit_vector_internal): * 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 (struct string_chars_block): * alloc.c (mark_lcrecord_list): * alloc.c (make_lcrecord_list): * alloc.c (alloc_managed_lcrecord): * alloc.c (free_managed_lcrecord): * alloc.c (alloc_automanaged_lcrecord): * alloc.c (staticpro_1): * alloc.c (staticpro): * alloc.c (lispdesc_indirect_count_1): * alloc.c (lispdesc_indirect_description_1): * alloc.c (lispdesc_one_description_line_size): * alloc.c (lispdesc_structure_size): * alloc.c (mark_object_maybe_checking_free): * alloc.c (mark_with_description): * alloc.c (mark_struct_contents): * alloc.c (mark_object): * alloc.c (tick_lcrecord_stats): * alloc.c (free_cons): * alloc.c (free_key_data): * alloc.c (free_button_data): * alloc.c (free_motion_data): * alloc.c (free_process_data): * alloc.c (free_timeout_data): * alloc.c (free_magic_data): * alloc.c (free_magic_eval_data): * alloc.c (free_eval_data): * alloc.c (free_misc_user_data): * alloc.c (free_marker): * alloc.c (compact_string_chars): * alloc.c (gc_sweep): * alloc.c (garbage_collect_1): * alloc.c (Fgarbage_collect): * alloc.c (common_init_alloc_early): * alloc.c (init_alloc_early): * alloc.c (init_alloc_once_early): * buffer.c: * buffer.c (mark_buffer): * buffer.c (MARKED_SLOT): * buffer.c (cleanup_buffer_undo_lists): * buffer.c (Fget_file_buffer): * buffer.h (MARKED_SLOT): * bufslots.h: * bytecode.c: * callint.c: * casetab.c: * chartab.c: * chartab.c (symbol_to_char_table_type): * cmdloop.c: * cmdloop.c (Fcommand_loop_1): * config.h.in (new): * conslots.h: * console-gtk-impl.h (struct gtk_frame): * console-impl.h: * console-impl.h (struct console): * console-impl.h (MARKED_SLOT): * console-impl.h (CONSOLE_QUIT_EVENT): * console-msw-impl.h (XM_BUMPQUEUE): * console-msw.c (write_string_to_mswindows_debugging_output): * console-msw.h: * console-stream-impl.h: * console-stream-impl.h (struct stream_console): * console-stream.c: * console-stream.c (stream_init_console): * console-stream.h: * console-tty.c: * console-tty.h: * console-x.h: * console.c: * console.c (mark_console): * console.c (MARKED_SLOT): * console.c (allocate_console): * console.c (get_console_variant): * console.c (create_console): * console.c (delete_console_internal): * console.c (Fset_input_mode): * console.c (Fcurrent_input_mode): * console.c (common_init_complex_vars_of_console): * console.h: * console.h (console_variant): * console.h (device_metrics): * data.c: * data.c (Faref): * data.c (Faset): * data.c (decode_weak_list_type): * database.c: * debug.c (xemacs_debug_loop): * debug.c (FROB): * debug.c (Fadd_debug_class_to_check): * debug.c (Fdelete_debug_class_to_check): * debug.c (Fset_debug_classes_to_check): * debug.c (Fset_debug_class_types_to_check): * debug.c (Fdebug_types_being_checked): * debug.h (DASSERT): * device-gtk.c: * device-impl.h (struct device): * device-impl.h (MARKED_SLOT): * device-msw.c: * device-x.c: * device-x.c (x_init_device_class): * device-x.c (x_comp_visual_info): * device-x.c (x_try_best_visual_class): * device-x.c (x_init_device): * device-x.c (construct_name_list): * device-x.c (x_get_resource_prefix): * device-x.c (Fx_get_resource): * device-x.c (Fx_display_visual_class): * device.c: * device.c (MARKED_SLOT): * device.c (allocate_device): * device.c (Fmake_device): * device.c (delete_device_internal): * device.c (Fset_device_class): * device.h: * devslots.h: * devslots.h (MARKED_SLOT): * dialog-msw.c: * dired-msw.c (mswindows_ls_sort_fcn): * dired-msw.c (mswindows_get_files): * dired-msw.c (mswindows_format_file): * doprnt.c (parse_doprnt_spec): * dumper.c: * dumper.c (struct): * dumper.c (dump_add_root_block): * dumper.c (dump_add_root_struct_ptr): * dumper.c (dump_add_root_lisp_object): * dumper.c (pdump_struct_list_elt): * dumper.c (pdump_get_entry_list): * dumper.c (pdump_backtrace): * dumper.c (pdump_bump_depth): * dumper.c (pdump_register_sub): * dumper.c (pdump_register_object): * dumper.c (pdump_register_struct_contents): * dumper.c (pdump_register_struct): * dumper.c (pdump_store_new_pointer_offsets): * dumper.c (pdump_dump_data): * dumper.c (pdump_reloc_one): * dumper.c (pdump_allocate_offset): * dumper.c (pdump_scan_by_alignment): * dumper.c (pdump_dump_root_blocks): * dumper.c (pdump_dump_rtables): * dumper.c (pdump_dump_root_lisp_objects): * dumper.c (pdump): * dumper.c (pdump_load_finish): * dumper.c (pdump_file_get): * dumper.c (pdump_resource_get): * dumper.c (pdump_load): * editfns.c (save_excursion_restore): * editfns.c (user_login_name): * editfns.c (save_restriction_restore): * elhash.c: * elhash.c (htentry): * elhash.c (struct Lisp_Hash_Table): * elhash.c (HTENTRY_CLEAR_P): * elhash.c (LINEAR_PROBING_LOOP): * elhash.c (check_hash_table_invariants): * elhash.c (mark_hash_table): * elhash.c (hash_table_equal): * elhash.c (print_hash_table_data): * elhash.c (free_hentries): * elhash.c (make_general_lisp_hash_table): * elhash.c (decode_hash_table_weakness): * elhash.c (decode_hash_table_test): * elhash.c (Fcopy_hash_table): * elhash.c (resize_hash_table): * elhash.c (pdump_reorganize_hash_table): * elhash.c (find_htentry): * elhash.c (Fgethash): * elhash.c (Fputhash): * elhash.c (remhash_1): * elhash.c (Fremhash): * elhash.c (Fclrhash): * elhash.c (copy_compress_hentries): * elhash.c (elisp_maphash_unsafe): * elhash.c (finish_marking_weak_hash_tables): * elhash.c (prune_weak_hash_tables): * elhash.h: * emacs.c: * emacs.c (main_1): * emacs.c (main): * emacs.c (shut_down_emacs): * emodules.h (dump_add_root_lisp_object): * eval.c: * eval.c (unwind_to_catch): * eval.c (maybe_signal_error_1): * eval.c (maybe_signal_continuable_error_1): * eval.c (maybe_signal_error): * eval.c (maybe_signal_continuable_error): * eval.c (maybe_signal_error_2): * eval.c (maybe_signal_continuable_error_2): * eval.c (maybe_signal_ferror): * eval.c (maybe_signal_continuable_ferror): * eval.c (maybe_signal_ferror_with_frob): * eval.c (maybe_signal_continuable_ferror_with_frob): * eval.c (maybe_syntax_error): * eval.c (maybe_sferror): * eval.c (maybe_invalid_argument): * eval.c (maybe_invalid_constant): * eval.c (maybe_invalid_operation): * eval.c (maybe_invalid_change): * eval.c (maybe_invalid_state): * eval.c (Feval): * eval.c (call_trapping_problems): * eval.c (call_with_suspended_errors): * eval.c (warn_when_safe_lispobj): * eval.c (warn_when_safe): * eval.c (vars_of_eval): * event-Xt.c: * event-Xt.c (maybe_define_x_key_as_self_inserting_character): * event-Xt.c (x_to_emacs_keysym): * event-Xt.c (x_event_to_emacs_event): * event-Xt.c (emacs_Xt_enqueue_focus_event): * event-Xt.c (emacs_Xt_format_magic_event): * event-Xt.c (emacs_Xt_compare_magic_event): * event-Xt.c (emacs_Xt_hash_magic_event): * event-Xt.c (emacs_Xt_handle_magic_event): * event-Xt.c (Xt_timeout_to_emacs_event): * event-Xt.c (Xt_process_to_emacs_event): * event-Xt.c (signal_special_Xt_user_event): * event-Xt.c (emacs_Xt_next_event): * event-Xt.c (emacs_Xt_event_handler): * event-Xt.c (emacs_Xt_drain_queue): * event-Xt.c (emacs_Xt_event_pending_p): * event-Xt.c (check_if_pending_expose_event): * event-Xt.c (reinit_vars_of_event_Xt): * event-Xt.c (vars_of_event_Xt): * event-gtk.c: * event-gtk.c (IS_MODIFIER_KEY): * event-gtk.c (emacs_gtk_format_magic_event): * event-gtk.c (emacs_gtk_compare_magic_event): * event-gtk.c (emacs_gtk_hash_magic_event): * event-gtk.c (emacs_gtk_handle_magic_event): * event-gtk.c (gtk_to_emacs_keysym): * event-gtk.c (gtk_timeout_to_emacs_event): * event-gtk.c (gtk_process_to_emacs_event): * event-gtk.c (dragndrop_data_received): * event-gtk.c (signal_special_gtk_user_event): * event-gtk.c (emacs_gtk_next_event): * event-gtk.c (gtk_event_to_emacs_event): * event-gtk.c (generic_event_handler): * event-gtk.c (emacs_shell_event_handler): * event-gtk.c (emacs_gtk_drain_queue): * event-gtk.c (emacs_gtk_event_pending_p): * event-gtk.c (reinit_vars_of_event_gtk): * event-gtk.c (vars_of_event_gtk): * event-msw.c: * event-msw.c (struct winsock_stream): * event-msw.c (winsock_reader): * event-msw.c (winsock_writer): * event-msw.c (mswindows_enqueue_dispatch_event): * event-msw.c (mswindows_enqueue_misc_user_event): * event-msw.c (mswindows_enqueue_magic_event): * event-msw.c (mswindows_enqueue_process_event): * event-msw.c (mswindows_enqueue_mouse_button_event): * event-msw.c (mswindows_enqueue_keypress_event): * event-msw.c (mswindows_dequeue_dispatch_event): * event-msw.c (emacs_mswindows_drain_queue): * event-msw.c (mswindows_need_event_in_modal_loop): * event-msw.c (mswindows_need_event): * event-msw.c (mswindows_wm_timer_callback): * event-msw.c (dde_eval_string): * event-msw.c (Fdde_alloc_advise_item): * event-msw.c (mswindows_dde_callback): * event-msw.c (mswindows_wnd_proc): * event-msw.c (remove_timeout_mapper): * event-msw.c (emacs_mswindows_remove_timeout): * event-msw.c (emacs_mswindows_event_pending_p): * event-msw.c (emacs_mswindows_format_magic_event): * event-msw.c (emacs_mswindows_compare_magic_event): * event-msw.c (emacs_mswindows_hash_magic_event): * event-msw.c (emacs_mswindows_handle_magic_event): * event-msw.c (emacs_mswindows_select_console): * event-msw.c (emacs_mswindows_unselect_console): * event-msw.c (reinit_vars_of_event_mswindows): * event-msw.c (vars_of_event_mswindows): * event-stream.c: * event-stream.c (mark_command_builder): * event-stream.c (reset_command_builder_event_chain): * event-stream.c (allocate_command_builder): * event-stream.c (copy_command_builder): * event-stream.c (command_builder_append_event): * event-stream.c (event_stream_event_pending_p): * event-stream.c (event_stream_force_event_pending): * event-stream.c (maybe_read_quit_event): * event-stream.c (event_stream_drain_queue): * event-stream.c (remove_quit_p_event): * event-stream.c (event_stream_quit_p): * event-stream.c (echo_key_event): * event-stream.c (maybe_kbd_translate): * event-stream.c (execute_help_form): * event-stream.c (event_stream_generate_wakeup): * event-stream.c (enqueue_dispatch_event): * event-stream.c (enqueue_magic_eval_event): * event-stream.c (Fenqueue_eval_event): * event-stream.c (enqueue_misc_user_event): * event-stream.c (enqueue_misc_user_event_pos): * event-stream.c (next_event_internal): * event-stream.c (Fnext_event): * event-stream.c (Faccept_process_output): * event-stream.c (execute_internal_event): * event-stream.c (munge_keymap_translate): * event-stream.c (command_builder_find_leaf_no_mule_processing): * event-stream.c (command_builder_find_leaf): * event-stream.c (lookup_command_event): * event-stream.c (is_scrollbar_event): * event-stream.c (execute_command_event): * event-stream.c (Fdispatch_event): * event-stream.c (Fread_key_sequence): * event-stream.c (dribble_out_event): * event-stream.c (vars_of_event_stream): * event-tty.c (tty_timeout_to_emacs_event): * event-tty.c (emacs_tty_next_event): * event-tty.c (emacs_tty_drain_queue): * event-tty.c (reinit_vars_of_event_tty): * event-unixoid.c: * event-unixoid.c (find_tty_or_stream_console_from_fd): * event-unixoid.c (read_event_from_tty_or_stream_desc): * event-unixoid.c (drain_tty_devices): * event-unixoid.c (poll_fds_for_input): * events.c: * events.c (deinitialize_event): * events.c (zero_event): * events.c (mark_event): * events.c (print_event_1): * events.c (print_event): * events.c (event_equal): * events.c (event_hash): * events.c (Fmake_event): * events.c (Fdeallocate_event): * events.c (Fcopy_event): * events.c (map_event_chain_remove): * events.c (character_to_event): * events.c (event_to_character): * events.c (Fevent_to_character): * events.c (format_event_object): * events.c (upshift_event): * events.c (downshift_event): * events.c (event_upshifted_p): * events.c (Fevent_live_p): * events.c (Fevent_type): * events.c (Fevent_timestamp): * events.c (CHECK_EVENT_TYPE): * events.c (CHECK_EVENT_TYPE2): * events.c (CHECK_EVENT_TYPE3): * events.c (Fevent_key): * events.c (Fevent_button): * events.c (Fevent_modifier_bits): * events.c (event_x_y_pixel_internal): * events.c (event_pixel_translation): * events.c (Fevent_process): * events.c (Fevent_function): * events.c (Fevent_object): * events.c (Fevent_properties): * events.c (syms_of_events): * events.c (vars_of_events): * events.h: * events.h (struct event_stream): * events.h (struct Lisp_Key_Data): * events.h (KEY_DATA_KEYSYM): * events.h (EVENT_KEY_KEYSYM): * events.h (struct Lisp_Button_Data): * events.h (EVENT_BUTTON_BUTTON): * events.h (struct Lisp_Motion_Data): * events.h (EVENT_MOTION_X): * events.h (struct Lisp_Process_Data): * events.h (EVENT_PROCESS_PROCESS): * events.h (struct Lisp_Timeout_Data): * events.h (EVENT_TIMEOUT_INTERVAL_ID): * events.h (struct Lisp_Eval_Data): * events.h (EVENT_EVAL_FUNCTION): * events.h (struct Lisp_Misc_User_Data): * events.h (EVENT_MISC_USER_FUNCTION): * events.h (struct Lisp_Magic_Eval_Data): * events.h (EVENT_MAGIC_EVAL_INTERNAL_FUNCTION): * events.h (struct Lisp_Magic_Data): * events.h (EVENT_MAGIC_UNDERLYING): * events.h (EVENT_MAGIC_GDK_EVENT): * events.h (struct Lisp_Event): * events.h (XEVENT_CHANNEL): * events.h (SET_EVENT_TIMESTAMP_ZERO): * events.h (SET_EVENT_CHANNEL): * events.h (SET_EVENT_NEXT): * events.h (XSET_EVENT_TYPE): * events.h (struct command_builder): * extents.c: * extents.c (gap_array_adjust_markers): * extents.c (gap_array_recompute_derived_values): * extents.c (gap_array_move_gap): * extents.c (gap_array_make_gap): * extents.c (gap_array_insert_els): * extents.c (gap_array_delete_els): * extents.c (gap_array_make_marker): * extents.c (gap_array_delete_marker): * extents.c (gap_array_move_marker): * extents.c (make_gap_array): * extents.c (free_gap_array): * extents.c (extent_list_num_els): * extents.c (extent_list_insert): * extents.c (mark_extent_auxiliary): * extents.c (allocate_extent_auxiliary): * extents.c (decode_extent_at_flag): * extents.c (verify_extent_mapper): * extents.c (symbol_to_glyph_layout): * extents.c (syms_of_extents): * faces.c: * file-coding.c: * file-coding.c (struct_detector_category_description =): * file-coding.c (detector_category_dynarr_description_1): * file-coding.c (struct_detector_description =): * file-coding.c (detector_dynarr_description_1): * file-coding.c (MARKED_SLOT): * file-coding.c (mark_coding_system): * file-coding.c (coding_system_extra_description_map): * file-coding.c (coding_system_description): * file-coding.c (allocate_coding_system): * file-coding.c (symbol_to_eol_type): * file-coding.c (Fcoding_system_aliasee): * file-coding.c (set_coding_stream_coding_system): * file-coding.c (struct convert_eol_coding_system): * file-coding.c (struct undecided_coding_system): * file-coding.c (undecided_mark_coding_stream): * file-coding.c (coding_category_symbol_to_id): * file-coding.c (struct gzip_coding_system): * file-coding.c (coding_system_type_create): * file-coding.h: * file-coding.h (struct Lisp_Coding_System): * file-coding.h (CODING_SYSTEM_SLOT_DECLARATION): * file-coding.h (coding_system_variant): * file-coding.h (struct coding_system_methods): * file-coding.h (DEFINE_CODING_SYSTEM_TYPE_WITH_DATA): * file-coding.h (INITIALIZE_CODING_SYSTEM_TYPE_WITH_DATA): * file-coding.h (struct coding_stream): * fileio.c (Fsubstitute_in_file_name): * floatfns.c: * fns.c: * fns.c (base64_encode_1): * frame-gtk.c: * frame-gtk.c (Fgtk_start_drag_internal): * frame-impl.h (struct frame): * frame-impl.h (MARKED_SLOT): * frame-msw.c: * frame-x.c: * frame-x.c (Fcde_start_drag_internal): * frame-x.c (Foffix_start_drag_internal): * frame.c: * frame.c (MARKED_SLOT): * frame.c (allocate_frame_core): * frame.c (delete_frame_internal): * frame.c (Fmouse_position_as_motion_event): * frameslots.h: * frameslots.h (MARKED_SLOT_ARRAY): * free-hook.c: * glyphs-msw.c (mswindows_widget_instantiate): * glyphs-x.c: * glyphs-x.c (convert_EImage_to_XImage): * glyphs.c: * glyphs.c (process_image_string_instantiator): * glyphs.c (mark_image_instance): * glyphs.c (allocate_image_instance): * glyphs.c (unmap_subwindow): * glyphs.c (map_subwindow): * glyphs.c (syms_of_glyphs): * glyphs.c (specifier_type_create_image): * glyphs.h: * glyphs.h (struct text_image_instance): * glyphs.h (struct Lisp_Image_Instance): * gmalloc.c: * gmalloc.c ("C"): * gpmevent.c (Freceive_gpm_event): * gpmevent.c (gpm_next_event_cb): * gpmevent.c (vars_of_gpmevent): * gtk-glue.c (gdk_event_to_emacs_event): * gtk-xemacs.c (gtk_xemacs_class_init): * gui-msw.c: * gui-msw.c (mswindows_handle_gui_wm_command): * gui-msw.c (mswindows_translate_menu_or_dialog_item): * gui-x.c: * gui-x.c (mark_popup_data): * gui-x.c (snarf_widget_value_mapper): * gui-x.c (gcpro_popup_callbacks): * gui-x.c (ungcpro_popup_callbacks): * gui-x.c (free_popup_widget_value_tree): * gui-x.c (popup_selection_callback): * gui-x.h: * gui-x.h (struct popup_data): * gui.c: * gui.c (allocate_gui_item): * gutter.c (decode_gutter_position): * hash.c (NULL_ENTRY): * indent.c (vmotion_1): * indent.c (vmotion_pixels): * input-method-motif.c (res): * input-method-xlib.c (IMInstantiateCallback): * input-method-xlib.c (XIM_init_device): * input-method-xlib.c (res): * intl-encap-win32.c: * intl-encap-win32.c (qxeSHGetDataFromIDList): * intl-win32.c: * intl-win32.c (mswindows_multibyte_cp_type): * intl-win32.c (struct mswindows_multibyte_coding_system): * keymap.c: * keymap.c (make_key_description): * keymap.c (keymap_store): * keymap.c (get_keyelt): * keymap.c (keymap_lookup_1): * keymap.c (define_key_parser): * keymap.c (key_desc_list_to_event): * keymap.c (event_matches_key_specifier_p): * keymap.c (meta_prefix_char_p): * keymap.c (ensure_meta_prefix_char_keymapp): * keymap.c (Fdefine_key): * keymap.c (struct raw_lookup_key_mapper_closure): * keymap.c (raw_lookup_key): * keymap.c (raw_lookup_key_mapper): * keymap.c (lookup_keys): * keymap.c (lookup_events): * keymap.c (Flookup_key): * keymap.c (struct map_keymap_unsorted_closure): * keymap.c (map_keymap_unsorted_mapper): * keymap.c (map_keymap_sorted): * keymap.c (map_keymap_mapper): * keymap.c (map_keymap): * keymap.c (accessible_keymaps_mapper_1): * keymap.c (Faccessible_keymaps): * keymap.c (Fsingle_key_description): * keymap.c (raw_keys_to_keys): * keymap.c (format_raw_keys): * keymap.c (where_is_recursive_mapper): * keymap.c (where_is_internal): * keymap.c (describe_map_mapper_shadow_search): * keymap.c (keymap_lookup_inherited_mapper): * keymap.c (describe_map_mapper): * keymap.h (event_matches_key_specifier_p): * lisp.h: * lisp.h (this): * lisp.h (RETURN_NOT_REACHED): * lisp.h (struct Lisp_Vector): * lisp.h (struct Lisp_Bit_Vector): * lisp.h (UNGCPRO_1): * lisp.h (NUNGCPRO): * lisp.h (NNUNGCPRO): * lisp.h (DECLARE_INLINE_HEADER): * lrecord.h: * lrecord.h (struct lrecord_header): * lrecord.h (struct lcrecord_header): * lrecord.h (lrecord_type): * lrecord.h (struct lrecord_implementation): * lrecord.h (RECORD_DUMPABLE): * lrecord.h (memory_description_type): * lrecord.h (data_description_entry_flags): * lrecord.h (struct memory_description): * lrecord.h (struct sized_memory_description): * lrecord.h (XD_INDIRECT): * lrecord.h (XD_IS_INDIRECT): * lrecord.h (XD_DYNARR_DESC): * lrecord.h (DEFINE_BASIC_LRECORD_IMPLEMENTATION): * lrecord.h (MAKE_LRECORD_IMPLEMENTATION): * lrecord.h (MAKE_EXTERNAL_LRECORD_IMPLEMENTATION): * lrecord.h (alloc_lcrecord_type): * lstream.c: * lstream.c (Lstream_new): * lstream.c (lisp_buffer_marker): * lstream.h: * lstream.h (lstream_implementation): * lstream.h (DEFINE_LSTREAM_IMPLEMENTATION): * lstream.h (DEFINE_LSTREAM_IMPLEMENTATION_WITH_DATA): * marker.c: * marker.c (copy_marker_1): * mem-limits.h: * menubar-gtk.c: * menubar-gtk.c (gtk_popup_menu): * menubar-msw.c: * menubar-msw.c (mswindows_popup_menu): * menubar-x.c (make_dummy_xbutton_event): * menubar-x.c (command_builder_operate_menu_accelerator): * menubar-x.c (menu_accelerator_safe_compare): * menubar-x.c (menu_accelerator_safe_mod_compare): * mule-charset.c: * mule-charset.c (make_charset): * mule-charset.c (Fcharset_property): * mule-coding.c: * mule-coding.c (ccs_description_1): * mule-coding.c (ccs_description =): * mule-coding.c (ccsd_description_1): * mule-coding.c (ccsd_description =): * nt.c (getpwnam): * nt.c (init_mswindows_environment): * nt.c (get_cached_volume_information): * nt.c (mswindows_is_executable): * nt.c (read_unc_volume): * nt.c (mswindows_access): * nt.c (mswindows_link): * nt.c (mswindows_fstat): * nt.c (mswindows_stat): * nt.c (mswindows_executable_type): * nt.c (Fmswindows_short_file_name): * nt.c (Fmswindows_long_file_name): * objects-impl.h (struct Lisp_Color_Instance): * objects-impl.h (struct Lisp_Font_Instance): * objects-tty.c: * objects-x.c (allocate_nearest_color): * objects.c: * objects.c (Fmake_color_instance): * objects.c (Fmake_font_instance): * objects.c (font_instantiate): * opaque.c: * opaque.c (make_opaque): * opaque.c (make_opaque_ptr): * opaque.c (reinit_opaque_early): * opaque.c (init_opaque_once_early): * print.c: * print.c (printing_badness): * print.c (printing_major_badness): * print.c (print_internal): * print.c (debug_p4): * print.c (dp): * print.c (debug_backtrace): * process-nt.c (nt_create_process): * process-nt.c (get_internet_address): * process-unix.c: * process-unix.c (struct unix_process_data): * process-unix.c (get_internet_address): * process-unix.c (unix_alloc_process_data): * process-unix.c (unix_create_process): * process-unix.c (try_to_initialize_subtty): * process-unix.c (unix_kill_child_process): * process-unix.c (process_type_create_unix): * process.c: * process.c (mark_process): * process.c (MARKED_SLOT): * process.c (make_process_internal): * process.c (Fprocess_tty_name): * process.c (decode_signal): * process.h: * procimpl.h: * procimpl.h (struct process_methods): * procimpl.h (struct Lisp_Process): * rangetab.c: * realpath.c (readlink_and_correct_case): * redisplay-x.c (x_window_output_end): * redisplay-x.c (x_redraw_exposed_area): * redisplay-x.c (x_clear_frame): * redisplay.c: * redisplay.h: * redisplay.h (struct rune_dglyph): * redisplay.h (struct rune): * scrollbar.c: * scrollbar.c (create_scrollbar_instance): * specifier.c: * specifier.c (specifier_empty_extra_description_1): * specifier.c (make_specifier_internal): * specifier.c (decode_locale_type): * specifier.c (decode_how_to_add_specification): * specifier.h: * specifier.h (struct specifier_methods): * specifier.h (DEFINE_SPECIFIER_TYPE_WITH_DATA): * specifier.h (INITIALIZE_SPECIFIER_TYPE_WITH_DATA): * symbols.c: * symbols.c (Fsetplist): * symbols.c (default_value): * symbols.c (decode_magic_handler_type): * symbols.c (handler_type_from_function_symbol): * symbols.c (Fdefvaralias): * symbols.c (init_symbols_once_early): * symbols.c (reinit_symbols_early): * symsinit.h: * sysdep.c (sys_subshell): * sysdep.c (tty_init_sys_modes_on_device): * syswindows.h: * text.c (dfc_convert_to_external_format): * text.c (dfc_convert_to_internal_format): * text.c (reinit_eistring_early): * text.c (init_eistring_once_early): * text.c (reinit_vars_of_text): * text.h: * text.h (INC_IBYTEPTR_FMT): * text.h (DEC_IBYTEPTR_FMT): * toolbar.c: * toolbar.c (decode_toolbar_position): * tooltalk.c: * ui-gtk.c: * unexnt.c: * unexnt.c (_start): * unexnt.c (unexec): * unexnt.c (get_section_info): * unicode.c: * unicode.c (vars_of_unicode): * window.c: * window.c (allocate_window): * window.c (new_window_mirror): * window.c (update_mirror_internal): * winslots.h:
author michaels
date Sun, 12 Jan 2003 11:08:22 +0000
parents 5f2f8dcbfb3e
children d608f0654d02
files modules/ChangeLog modules/postgresql/postgresql.c src/ChangeLog src/README.kkcc src/alloc.c src/buffer.c src/buffer.h src/bufslots.h src/bytecode.c src/callint.c src/casetab.c src/chartab.c src/cmdloop.c src/coding-system-slots.h src/config.h.in src/conslots.h src/console-gtk-impl.h src/console-impl.h src/console-msw-impl.h src/console-msw.c src/console-msw.h src/console-stream-impl.h src/console-stream.c src/console-stream.h src/console-tty.c src/console-tty.h src/console-x.h src/console.c src/console.h src/data.c src/database.c src/debug.c src/debug.h src/device-gtk.c src/device-impl.h src/device-msw.c src/device-x.c src/device.c src/device.h src/devslots.h src/dialog-msw.c src/dired-msw.c src/doprnt.c src/dumper.c src/editfns.c src/elhash.c src/elhash.h src/emacs.c src/emodules.h src/eval.c src/event-Xt.c src/event-gtk.c src/event-msw.c src/event-stream.c src/event-tty.c src/event-unixoid.c src/events.c src/events.h src/extents.c src/faces.c src/file-coding.c src/file-coding.h src/fileio.c src/floatfns.c src/fns.c src/frame-gtk.c src/frame-impl.h src/frame-msw.c src/frame-x.c src/frame.c src/frameslots.h src/free-hook.c src/glyphs-msw.c src/glyphs-x.c src/glyphs.c src/glyphs.h src/gmalloc.c src/gpmevent.c src/gtk-glue.c src/gtk-xemacs.c src/gui-msw.c src/gui-x.c src/gui-x.h src/gui.c src/gutter.c src/hash.c src/indent.c src/input-method-motif.c src/input-method-xlib.c src/intl-encap-win32.c src/intl-win32.c src/keymap.c src/keymap.h src/lisp.h src/lrecord.h src/lstream.c src/lstream.h src/marker.c src/mem-limits.h src/menubar-gtk.c src/menubar-msw.c src/menubar-x.c src/mule-charset.c src/mule-coding.c src/nt.c src/objects-impl.h src/objects-tty.c src/objects-x.c src/objects.c src/opaque.c src/print.c src/process-nt.c src/process-slots.h src/process-unix.c src/process.c src/process.h src/procimpl.h src/rangetab.c src/realpath.c src/redisplay-x.c src/redisplay.c src/redisplay.h src/scrollbar.c src/specifier.c src/specifier.h src/symbols.c src/symsinit.h src/sysdep.c src/syswindows.h src/text.c src/text.h src/toolbar.c src/tooltalk.c src/ui-gtk.c src/unexnt.c src/unicode.c src/window.c src/winslots.h
diffstat 138 files changed, 7192 insertions(+), 7525 deletions(-) [+]
line wrap: on
line diff
--- a/modules/ChangeLog	Sun Jan 12 11:04:30 2003 +0000
+++ b/modules/ChangeLog	Sun Jan 12 11:08:22 2003 +0000
@@ -1,3 +1,8 @@
+2002-12-16  Ben Wing  <ben@xemacs.org>
+
+	* postgresql/postgresql.c:
+	remove ifdef USE_KKCC.
+
 2003-01-04  Steve Youngs  <youngs@xemacs.org>
 
 	* XEmacs 21.5.10 "burdock" is released.
--- a/modules/postgresql/postgresql.c	Sun Jan 12 11:04:30 2003 +0000
+++ b/modules/postgresql/postgresql.c	Sun Jan 12 11:08:22 2003 +0000
@@ -178,11 +178,9 @@
   return wrap_pgconn (pgconn);
 }
 
-#ifdef USE_KKCC
-static const struct lrecord_description pgconn_description [] = {
+static const struct memory_description pgconn_description [] = {
   { XD_END }
 };
-#endif /* USE_KKCC */
 
 static Lisp_Object
 #ifdef RUNNING_XEMACS_21_1
@@ -267,20 +265,12 @@
 			       NULL, NULL,
 			       Lisp_PGconn);
 #else
-#ifdef USE_KKCC
 DEFINE_LRECORD_IMPLEMENTATION ("pgconn", pgconn,
 			       0, /*dumpable-flag*/
 			       mark_pgconn, print_pgconn, finalize_pgconn,
 			       NULL, NULL,
 			       pgconn_description,
 			       Lisp_PGconn);
-#else /* not USE_KKCC */
-DEFINE_LRECORD_IMPLEMENTATION ("pgconn", pgconn,
-			       mark_pgconn, print_pgconn, finalize_pgconn,
-			       NULL, NULL,
-			       0,
-			       Lisp_PGconn);
-#endif /* not USE_KKCC */
 #endif
 /****/
 
@@ -295,11 +285,9 @@
   return wrap_pgresult (pgresult);
 }
 
-#ifdef USE_KKCC
-static const struct lrecord_description pgresult_description [] = {
+static const struct memory_description pgresult_description [] = {
   { XD_END }
 };
-#endif /* USE_KKCC */
 
 
 static Lisp_Object
@@ -400,20 +388,12 @@
 			       NULL, NULL,
 			       Lisp_PGresult);
 #else
-#ifdef USE_KKCC
 DEFINE_LRECORD_IMPLEMENTATION ("pgresult", pgresult,
 			       0, /*dumpable-flag*/
 			       mark_pgresult, print_pgresult, finalize_pgresult,
 			       NULL, NULL,
 			       pgresult_description,
 			       Lisp_PGresult);
-#else /* not USE_KKCC */
-DEFINE_LRECORD_IMPLEMENTATION ("pgresult", pgresult,
-			       mark_pgresult, print_pgresult, finalize_pgresult,
-			       NULL, NULL,
-			       0,
-			       Lisp_PGresult);
-#endif /* not USE_KKCC */
 #endif
 
 /***********************/
--- a/src/ChangeLog	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/ChangeLog	Sun Jan 12 11:08:22 2003 +0000
@@ -1,3 +1,846 @@
+2003-01-08  Mike Sperber <mike@xemacs.org>
+
+	* console.h (CDFW_CONSOLE): Don't lead to a crash if we're dealing
+	with a dead window/frame/device/console.
+
+2002-12-20  Mike Sperber <mike@xemacs.org>
+
+	* ui-gtk.c: Fix typo from Ben's patch: emacs_ffi_data is a
+	typedef, not a struct.  emacs_gtk_object_data is a typedef, not a
+	struct.
+	
+
+	* gtk-glue.c (gdk_event_to_emacs_event): Fix typos from Ben's
+	patch: le -> emacs_event + rearrange the code.
+
+	* event-gtk.c (gtk_event_to_emacs_event): Fix typos from Ben's
+	patch: ..._UNDERLYING_GDK_EVENT -> ..._GDK_EVENT, ev -> key_event.
+
+	* device-gtk.c: Fix typo from Ben's patch: x_keysym_map_hash_table
+	-> x_keysym_map_hashtable.
+
+2002-12-19  Mike Sperber <mike@xemacs.org>
+
+	* menubar-x.c (set_frame_menubar): Initialize protect_me field of
+	popup_data.
+
+2002-12-16  Ben Wing  <ben@xemacs.org>
+	Major cleanup of KKCC, etc.
+
+	KKCC, pdump-related:
+	
+	-- descriptions are written for all objects.  this required some
+	   changes in the format of some objects, e.g. extents, popup-data,
+	   coding system, lstream, lcrecord-list.
+	-- KKCC now handles weakness in markers, hash tables, elsewhere
+	   correctly (formerly, you'd eventually get a stack overflow due
+	   to endlessly expanding markers).
+	-- textual changes: lrecord_description -> memory_description,
+	   struct_description -> sized_memory_description.
+	-- extensive comment describing descriptions and pdump.
+	-- redo XD_UNION so it works inline and change its format to provide
+	   sufficient info for pdump.  implement XD_UNION in pdump.  also add
+	   XD_UNION_DYNAMIC_SIZE, which works like XD_UNION except for when
+	   auto-computing structure sizes.
+	-- add support for XD_INDIRECT in description offsets (used by
+	   extents).
+	-- add support for "description maps", allowing for indirect
+	   descriptions that are retrieved from an object at run-time.  this
+	   generalizes XD_CODING_SYSTEM_END, XD_SPECIFIER_END, etc., which
+	   have now been eliminated.
+	-- add a fifth field "flags" to memory_description, to support flags
+	   that can be specified for this particular line.  Currently defined
+	   flags are XD_FLAG_NO_KKCC (KKCC should ignore this entry; useful
+	   for the weakness above in markers, etc.), XD_FLAG_NO_PDUMP (pdump
+	   should ignore this entry), XD_FLAG_UNION_DEFAULT_ENTRY (in
+	   union maps, this specifies a "default" entry for all remaining
+	   values), and XD_FLAG_FREE_LISP_OBJECT (for use with lcrecord-lists).
+	-- clean up the kkcc-itis in events, so that the differences
+	   between event data as separate objects and as a union are now
+	   minimized to a small number of places.  with the new XD_UNION, we
+	   no longer need event data as separate objects, so this code is no
+	   longer ifdef USE_KKCC, but instead ifdef EVENT_DATA_AS_OBJECTS,
+	   not used by default.  make sure that we explicitly free the
+	   separate event data objects when no longer in use, to maintain the
+	   invariant the event processing causes no consing.
+	-- also remove other USE_KKCC ifdefs when not necessary.
+	-- allow for KKCC compilation under MS Windows.
+	-- fix README.kkcc.
+	-- dump_add_root_object -> dump_add_root_lisp_object.
+	-- implement dump_add_root_block and use this to handle
+	   dump_add_opaque.
+	-- factor out some code duplicated in kkcc and pdump.
+
+	Other allocation/object-related:
+	
+	-- change various *slots.h so MARKED_SLOT() call no longer
+	   includes semicolon.
+	-- free_marker() takes a Lisp_Object not a direct pointer.
+	-- make bit vectors lcrecords, like vectors, and eliminate code
+	   that essentially duplicated the lcrecord handling.
+	-- additional asserts in FREE_FIXED_TYPE, formerly duplicated in
+	   the various callers of this.
+	-- all lcrecord allocation functions now zero out the returned
+	   lcrecords.  unnecessary calls to zero_lcrecord removed.  add long
+	   comment describing these functions.
+	-- extract out process and coding system slots, like for buffers,
+	   frames, etc.
+	-- lcrecords now set the type of items sitting on the free list to
+	   lcrecord_type_free.
+	-- changes to the way that gap arrays are allocated, for kkcc's
+	   benefit -- now, one single memory block with a stretchy array on
+	   the end, instead of a separate block holding the array.
+
+	Error-checking-related:
+	
+	-- now can compile with C++ under MS Windows.  clean up compile errors
+	   discovered that way. (a few were real problems)
+	-- add C++ error-checking code to verify problems with mismatched
+	   GCPRO/UNGCPRO. (there were a few in the kkcc code.) add long
+	   comment about how to catch insufficient GCPRO (yes, it's possible
+	   using C++).
+	-- add debug_p4(), a simple object printer, when debug_print()
+	   doesn't work.
+	-- add dp() and db() as short synonyms of debug_print(),
+	   debug_backtrace().
+	-- `print' tries EXTREMELY hard to avoid core dumping when printing
+	   when crashing or from debug_print(), and tries as hard as it
+	   reasonably can in other situations.
+	-- Correct the message output upon crashing to be more up-to-date.
+	
+	Event-related:
+	
+	-- document event-matches-key-specifier-p better.
+	-- generalize the dispatch queues formerly duplicated in the
+	   various event implementations.  add event methods to drain pending
+	   events.  generalize and clean up QUIT handling, removing
+	   event-specific quit processing.  allow arbitrary keystrokes, not
+	   just ASCII, to be the QUIT char.  among other things, this should
+	   fix some longstanding bugs in X quit handling.  long comment
+	   describing the various event queues.
+	-- implement delaying of XFlush() if there are pending expose events.
+	   SOMEONE PLEASE TRY THIS OUT.
+	-- Fix `xemacs -batch -l dunnet' under Cygwin.  Try to fix under
+	   MS Windows but not quite there yet.
+
+	Other:
+	
+	-- class -> class_ and no more C++ games with this item.
+	   new -> new_ in the lwlib code, so far not elsewhere.
+	-- use `struct htentry' not `struct hentry' in elhash.c to avoid
+	   debugger confusion with hash.c.
+	-- new macros ALIST_LOOP_3, ALIST_LOOP_4.
+
+	* README.kkcc:
+	* alloc.c:
+	* alloc.c (deadbeef_memory):
+	* alloc.c (allocate_lisp_storage):
+	* alloc.c (copy_lisp_object):
+	* alloc.c (ALLOCATE_FIXED_TYPE_1):
+	* alloc.c (FREE_FIXED_TYPE):
+	* alloc.c (make_vector_internal):
+	* alloc.c (make_bit_vector_internal):
+	* 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 (struct string_chars_block):
+	* alloc.c (mark_lcrecord_list):
+	* alloc.c (make_lcrecord_list):
+	* alloc.c (alloc_managed_lcrecord):
+	* alloc.c (free_managed_lcrecord):
+	* alloc.c (alloc_automanaged_lcrecord):
+	* alloc.c (staticpro_1):
+	* alloc.c (staticpro):
+	* alloc.c (lispdesc_indirect_count_1):
+	* alloc.c (lispdesc_indirect_description_1):
+	* alloc.c (lispdesc_one_description_line_size):
+	* alloc.c (lispdesc_structure_size):
+	* alloc.c (mark_object_maybe_checking_free):
+	* alloc.c (mark_with_description):
+	* alloc.c (mark_struct_contents):
+	* alloc.c (mark_object):
+	* alloc.c (tick_lcrecord_stats):
+	* alloc.c (free_cons):
+	* alloc.c (free_key_data):
+	* alloc.c (free_button_data):
+	* alloc.c (free_motion_data):
+	* alloc.c (free_process_data):
+	* alloc.c (free_timeout_data):
+	* alloc.c (free_magic_data):
+	* alloc.c (free_magic_eval_data):
+	* alloc.c (free_eval_data):
+	* alloc.c (free_misc_user_data):
+	* alloc.c (free_marker):
+	* alloc.c (compact_string_chars):
+	* alloc.c (gc_sweep):
+	* alloc.c (garbage_collect_1):
+	* alloc.c (Fgarbage_collect):
+	* alloc.c (common_init_alloc_early):
+	* alloc.c (init_alloc_early):
+	* alloc.c (init_alloc_once_early):
+	* buffer.c:
+	* buffer.c (mark_buffer):
+	* buffer.c (MARKED_SLOT):
+	* buffer.c (cleanup_buffer_undo_lists):
+	* buffer.c (Fget_file_buffer):
+	* buffer.h (MARKED_SLOT):
+	* bufslots.h:
+	* bytecode.c:
+	* callint.c:
+	* casetab.c:
+	* chartab.c:
+	* chartab.c (symbol_to_char_table_type):
+	* cmdloop.c:
+	* cmdloop.c (Fcommand_loop_1):
+	* config.h.in (new):
+	* conslots.h:
+	* console-gtk-impl.h (struct gtk_frame):
+	* console-impl.h:
+	* console-impl.h (struct console):
+	* console-impl.h (MARKED_SLOT):
+	* console-impl.h (CONSOLE_QUIT_EVENT):
+	* console-msw-impl.h (XM_BUMPQUEUE):
+	* console-msw.c (write_string_to_mswindows_debugging_output):
+	* console-msw.h:
+	* console-stream-impl.h:
+	* console-stream-impl.h (struct stream_console):
+	* console-stream.c:
+	* console-stream.c (stream_init_console):
+	* console-stream.h:
+	* console-tty.c:
+	* console-tty.h:
+	* console-x.h:
+	* console.c:
+	* console.c (mark_console):
+	* console.c (MARKED_SLOT):
+	* console.c (allocate_console):
+	* console.c (get_console_variant):
+	* console.c (create_console):
+	* console.c (delete_console_internal):
+	* console.c (Fset_input_mode):
+	* console.c (Fcurrent_input_mode):
+	* console.c (common_init_complex_vars_of_console):
+	* console.h:
+	* console.h (console_variant):
+	* console.h (device_metrics):
+	* data.c:
+	* data.c (Faref):
+	* data.c (Faset):
+	* data.c (decode_weak_list_type):
+	* database.c:
+	* debug.c (xemacs_debug_loop):
+	* debug.c (FROB):
+	* debug.c (Fadd_debug_class_to_check):
+	* debug.c (Fdelete_debug_class_to_check):
+	* debug.c (Fset_debug_classes_to_check):
+	* debug.c (Fset_debug_class_types_to_check):
+	* debug.c (Fdebug_types_being_checked):
+	* debug.h (DASSERT):
+	* device-gtk.c:
+	* device-impl.h (struct device):
+	* device-impl.h (MARKED_SLOT):
+	* device-msw.c:
+	* device-x.c:
+	* device-x.c (x_init_device_class):
+	* device-x.c (x_comp_visual_info):
+	* device-x.c (x_try_best_visual_class):
+	* device-x.c (x_init_device):
+	* device-x.c (construct_name_list):
+	* device-x.c (x_get_resource_prefix):
+	* device-x.c (Fx_get_resource):
+	* device-x.c (Fx_display_visual_class):
+	* device.c:
+	* device.c (MARKED_SLOT):
+	* device.c (allocate_device):
+	* device.c (Fmake_device):
+	* device.c (delete_device_internal):
+	* device.c (Fset_device_class):
+	* device.h:
+	* devslots.h:
+	* devslots.h (MARKED_SLOT):
+	* dialog-msw.c:
+	* dired-msw.c (mswindows_ls_sort_fcn):
+	* dired-msw.c (mswindows_get_files):
+	* dired-msw.c (mswindows_format_file):
+	* doprnt.c (parse_doprnt_spec):
+	* dumper.c:
+	* dumper.c (struct):
+	* dumper.c (dump_add_root_block):
+	* dumper.c (dump_add_root_struct_ptr):
+	* dumper.c (dump_add_root_lisp_object):
+	* dumper.c (pdump_struct_list_elt):
+	* dumper.c (pdump_get_entry_list):
+	* dumper.c (pdump_backtrace):
+	* dumper.c (pdump_bump_depth):
+	* dumper.c (pdump_register_sub):
+	* dumper.c (pdump_register_object):
+	* dumper.c (pdump_register_struct_contents):
+	* dumper.c (pdump_register_struct):
+	* dumper.c (pdump_store_new_pointer_offsets):
+	* dumper.c (pdump_dump_data):
+	* dumper.c (pdump_reloc_one):
+	* dumper.c (pdump_allocate_offset):
+	* dumper.c (pdump_scan_by_alignment):
+	* dumper.c (pdump_dump_root_blocks):
+	* dumper.c (pdump_dump_rtables):
+	* dumper.c (pdump_dump_root_lisp_objects):
+	* dumper.c (pdump):
+	* dumper.c (pdump_load_finish):
+	* dumper.c (pdump_file_get):
+	* dumper.c (pdump_resource_get):
+	* dumper.c (pdump_load):
+	* editfns.c (save_excursion_restore):
+	* editfns.c (user_login_name):
+	* editfns.c (save_restriction_restore):
+	* elhash.c:
+	* elhash.c (htentry):
+	* elhash.c (struct Lisp_Hash_Table):
+	* elhash.c (HTENTRY_CLEAR_P):
+	* elhash.c (LINEAR_PROBING_LOOP):
+	* elhash.c (check_hash_table_invariants):
+	* elhash.c (mark_hash_table):
+	* elhash.c (hash_table_equal):
+	* elhash.c (print_hash_table_data):
+	* elhash.c (free_hentries):
+	* elhash.c (make_general_lisp_hash_table):
+	* elhash.c (decode_hash_table_weakness):
+	* elhash.c (decode_hash_table_test):
+	* elhash.c (Fcopy_hash_table):
+	* elhash.c (resize_hash_table):
+	* elhash.c (pdump_reorganize_hash_table):
+	* elhash.c (find_htentry):
+	* elhash.c (Fgethash):
+	* elhash.c (Fputhash):
+	* elhash.c (remhash_1):
+	* elhash.c (Fremhash):
+	* elhash.c (Fclrhash):
+	* elhash.c (copy_compress_hentries):
+	* elhash.c (elisp_maphash_unsafe):
+	* elhash.c (finish_marking_weak_hash_tables):
+	* elhash.c (prune_weak_hash_tables):
+	* elhash.h:
+	* emacs.c:
+	* emacs.c (main_1):
+	* emacs.c (main):
+	* emacs.c (shut_down_emacs):
+	* emodules.h (dump_add_root_lisp_object):
+	* eval.c:
+	* eval.c (unwind_to_catch):
+	* eval.c (maybe_signal_error_1):
+	* eval.c (maybe_signal_continuable_error_1):
+	* eval.c (maybe_signal_error):
+	* eval.c (maybe_signal_continuable_error):
+	* eval.c (maybe_signal_error_2):
+	* eval.c (maybe_signal_continuable_error_2):
+	* eval.c (maybe_signal_ferror):
+	* eval.c (maybe_signal_continuable_ferror):
+	* eval.c (maybe_signal_ferror_with_frob):
+	* eval.c (maybe_signal_continuable_ferror_with_frob):
+	* eval.c (maybe_syntax_error):
+	* eval.c (maybe_sferror):
+	* eval.c (maybe_invalid_argument):
+	* eval.c (maybe_invalid_constant):
+	* eval.c (maybe_invalid_operation):
+	* eval.c (maybe_invalid_change):
+	* eval.c (maybe_invalid_state):
+	* eval.c (Feval):
+	* eval.c (call_trapping_problems):
+	* eval.c (call_with_suspended_errors):
+	* eval.c (warn_when_safe_lispobj):
+	* eval.c (warn_when_safe):
+	* eval.c (vars_of_eval):
+	* event-Xt.c:
+	* event-Xt.c (maybe_define_x_key_as_self_inserting_character):
+	* event-Xt.c (x_to_emacs_keysym):
+	* event-Xt.c (x_event_to_emacs_event):
+	* event-Xt.c (emacs_Xt_enqueue_focus_event):
+	* event-Xt.c (emacs_Xt_format_magic_event):
+	* event-Xt.c (emacs_Xt_compare_magic_event):
+	* event-Xt.c (emacs_Xt_hash_magic_event):
+	* event-Xt.c (emacs_Xt_handle_magic_event):
+	* event-Xt.c (Xt_timeout_to_emacs_event):
+	* event-Xt.c (Xt_process_to_emacs_event):
+	* event-Xt.c (signal_special_Xt_user_event):
+	* event-Xt.c (emacs_Xt_next_event):
+	* event-Xt.c (emacs_Xt_event_handler):
+	* event-Xt.c (emacs_Xt_drain_queue):
+	* event-Xt.c (emacs_Xt_event_pending_p):
+	* event-Xt.c (check_if_pending_expose_event):
+	* event-Xt.c (reinit_vars_of_event_Xt):
+	* event-Xt.c (vars_of_event_Xt):
+	* event-gtk.c:
+	* event-gtk.c (IS_MODIFIER_KEY):
+	* event-gtk.c (emacs_gtk_format_magic_event):
+	* event-gtk.c (emacs_gtk_compare_magic_event):
+	* event-gtk.c (emacs_gtk_hash_magic_event):
+	* event-gtk.c (emacs_gtk_handle_magic_event):
+	* event-gtk.c (gtk_to_emacs_keysym):
+	* event-gtk.c (gtk_timeout_to_emacs_event):
+	* event-gtk.c (gtk_process_to_emacs_event):
+	* event-gtk.c (dragndrop_data_received):
+	* event-gtk.c (signal_special_gtk_user_event):
+	* event-gtk.c (emacs_gtk_next_event):
+	* event-gtk.c (gtk_event_to_emacs_event):
+	* event-gtk.c (generic_event_handler):
+	* event-gtk.c (emacs_shell_event_handler):
+	* event-gtk.c (emacs_gtk_drain_queue):
+	* event-gtk.c (emacs_gtk_event_pending_p):
+	* event-gtk.c (reinit_vars_of_event_gtk):
+	* event-gtk.c (vars_of_event_gtk):
+	* event-msw.c:
+	* event-msw.c (struct winsock_stream):
+	* event-msw.c (winsock_reader):
+	* event-msw.c (winsock_writer):
+	* event-msw.c (mswindows_enqueue_dispatch_event):
+	* event-msw.c (mswindows_enqueue_misc_user_event):
+	* event-msw.c (mswindows_enqueue_magic_event):
+	* event-msw.c (mswindows_enqueue_process_event):
+	* event-msw.c (mswindows_enqueue_mouse_button_event):
+	* event-msw.c (mswindows_enqueue_keypress_event):
+	* event-msw.c (mswindows_dequeue_dispatch_event):
+	* event-msw.c (emacs_mswindows_drain_queue):
+	* event-msw.c (mswindows_need_event_in_modal_loop):
+	* event-msw.c (mswindows_need_event):
+	* event-msw.c (mswindows_wm_timer_callback):
+	* event-msw.c (dde_eval_string):
+	* event-msw.c (Fdde_alloc_advise_item):
+	* event-msw.c (mswindows_dde_callback):
+	* event-msw.c (mswindows_wnd_proc):
+	* event-msw.c (remove_timeout_mapper):
+	* event-msw.c (emacs_mswindows_remove_timeout):
+	* event-msw.c (emacs_mswindows_event_pending_p):
+	* event-msw.c (emacs_mswindows_format_magic_event):
+	* event-msw.c (emacs_mswindows_compare_magic_event):
+	* event-msw.c (emacs_mswindows_hash_magic_event):
+	* event-msw.c (emacs_mswindows_handle_magic_event):
+	* event-msw.c (emacs_mswindows_select_console):
+	* event-msw.c (emacs_mswindows_unselect_console):
+	* event-msw.c (reinit_vars_of_event_mswindows):
+	* event-msw.c (vars_of_event_mswindows):
+	* event-stream.c:
+	* event-stream.c (mark_command_builder):
+	* event-stream.c (reset_command_builder_event_chain):
+	* event-stream.c (allocate_command_builder):
+	* event-stream.c (copy_command_builder):
+	* event-stream.c (command_builder_append_event):
+	* event-stream.c (event_stream_event_pending_p):
+	* event-stream.c (event_stream_force_event_pending):
+	* event-stream.c (maybe_read_quit_event):
+	* event-stream.c (event_stream_drain_queue):
+	* event-stream.c (remove_quit_p_event):
+	* event-stream.c (event_stream_quit_p):
+	* event-stream.c (echo_key_event):
+	* event-stream.c (maybe_kbd_translate):
+	* event-stream.c (execute_help_form):
+	* event-stream.c (event_stream_generate_wakeup):
+	* event-stream.c (enqueue_dispatch_event):
+	* event-stream.c (enqueue_magic_eval_event):
+	* event-stream.c (Fenqueue_eval_event):
+	* event-stream.c (enqueue_misc_user_event):
+	* event-stream.c (enqueue_misc_user_event_pos):
+	* event-stream.c (next_event_internal):
+	* event-stream.c (Fnext_event):
+	* event-stream.c (Faccept_process_output):
+	* event-stream.c (execute_internal_event):
+	* event-stream.c (munge_keymap_translate):
+	* event-stream.c (command_builder_find_leaf_no_mule_processing):
+	* event-stream.c (command_builder_find_leaf):
+	* event-stream.c (lookup_command_event):
+	* event-stream.c (is_scrollbar_event):
+	* event-stream.c (execute_command_event):
+	* event-stream.c (Fdispatch_event):
+	* event-stream.c (Fread_key_sequence):
+	* event-stream.c (dribble_out_event):
+	* event-stream.c (vars_of_event_stream):
+	* event-tty.c (tty_timeout_to_emacs_event):
+	* event-tty.c (emacs_tty_next_event):
+	* event-tty.c (emacs_tty_drain_queue):
+	* event-tty.c (reinit_vars_of_event_tty):
+	* event-unixoid.c:
+	* event-unixoid.c (find_tty_or_stream_console_from_fd):
+	* event-unixoid.c (read_event_from_tty_or_stream_desc):
+	* event-unixoid.c (drain_tty_devices):
+	* event-unixoid.c (poll_fds_for_input):
+	* events.c:
+	* events.c (deinitialize_event):
+	* events.c (zero_event):
+	* events.c (mark_event):
+	* events.c (print_event_1):
+	* events.c (print_event):
+	* events.c (event_equal):
+	* events.c (event_hash):
+	* events.c (Fmake_event):
+	* events.c (Fdeallocate_event):
+	* events.c (Fcopy_event):
+	* events.c (map_event_chain_remove):
+	* events.c (character_to_event):
+	* events.c (event_to_character):
+	* events.c (Fevent_to_character):
+	* events.c (format_event_object):
+	* events.c (upshift_event):
+	* events.c (downshift_event):
+	* events.c (event_upshifted_p):
+	* events.c (Fevent_live_p):
+	* events.c (Fevent_type):
+	* events.c (Fevent_timestamp):
+	* events.c (CHECK_EVENT_TYPE):
+	* events.c (CHECK_EVENT_TYPE2):
+	* events.c (CHECK_EVENT_TYPE3):
+	* events.c (Fevent_key):
+	* events.c (Fevent_button):
+	* events.c (Fevent_modifier_bits):
+	* events.c (event_x_y_pixel_internal):
+	* events.c (event_pixel_translation):
+	* events.c (Fevent_process):
+	* events.c (Fevent_function):
+	* events.c (Fevent_object):
+	* events.c (Fevent_properties):
+	* events.c (syms_of_events):
+	* events.c (vars_of_events):
+	* events.h:
+	* events.h (struct event_stream):
+	* events.h (struct Lisp_Key_Data):
+	* events.h (KEY_DATA_KEYSYM):
+	* events.h (EVENT_KEY_KEYSYM):
+	* events.h (struct Lisp_Button_Data):
+	* events.h (EVENT_BUTTON_BUTTON):
+	* events.h (struct Lisp_Motion_Data):
+	* events.h (EVENT_MOTION_X):
+	* events.h (struct Lisp_Process_Data):
+	* events.h (EVENT_PROCESS_PROCESS):
+	* events.h (struct Lisp_Timeout_Data):
+	* events.h (EVENT_TIMEOUT_INTERVAL_ID):
+	* events.h (struct Lisp_Eval_Data):
+	* events.h (EVENT_EVAL_FUNCTION):
+	* events.h (struct Lisp_Misc_User_Data):
+	* events.h (EVENT_MISC_USER_FUNCTION):
+	* events.h (struct Lisp_Magic_Eval_Data):
+	* events.h (EVENT_MAGIC_EVAL_INTERNAL_FUNCTION):
+	* events.h (struct Lisp_Magic_Data):
+	* events.h (EVENT_MAGIC_UNDERLYING):
+	* events.h (EVENT_MAGIC_GDK_EVENT):
+	* events.h (struct Lisp_Event):
+	* events.h (XEVENT_CHANNEL):
+	* events.h (SET_EVENT_TIMESTAMP_ZERO):
+	* events.h (SET_EVENT_CHANNEL):
+	* events.h (SET_EVENT_NEXT):
+	* events.h (XSET_EVENT_TYPE):
+	* events.h (struct command_builder):
+	* extents.c:
+	* extents.c (gap_array_adjust_markers):
+	* extents.c (gap_array_recompute_derived_values):
+	* extents.c (gap_array_move_gap):
+	* extents.c (gap_array_make_gap):
+	* extents.c (gap_array_insert_els):
+	* extents.c (gap_array_delete_els):
+	* extents.c (gap_array_make_marker):
+	* extents.c (gap_array_delete_marker):
+	* extents.c (gap_array_move_marker):
+	* extents.c (make_gap_array):
+	* extents.c (free_gap_array):
+	* extents.c (extent_list_num_els):
+	* extents.c (extent_list_insert):
+	* extents.c (mark_extent_auxiliary):
+	* extents.c (allocate_extent_auxiliary):
+	* extents.c (decode_extent_at_flag):
+	* extents.c (verify_extent_mapper):
+	* extents.c (symbol_to_glyph_layout):
+	* extents.c (syms_of_extents):
+	* faces.c:
+	* file-coding.c:
+	* file-coding.c (struct_detector_category_description =):
+	* file-coding.c (detector_category_dynarr_description_1):
+	* file-coding.c (struct_detector_description =):
+	* file-coding.c (detector_dynarr_description_1):
+	* file-coding.c (MARKED_SLOT):
+	* file-coding.c (mark_coding_system):
+	* file-coding.c (coding_system_extra_description_map):
+	* file-coding.c (coding_system_description):
+	* file-coding.c (allocate_coding_system):
+	* file-coding.c (symbol_to_eol_type):
+	* file-coding.c (Fcoding_system_aliasee):
+	* file-coding.c (set_coding_stream_coding_system):
+	* file-coding.c (struct convert_eol_coding_system):
+	* file-coding.c (struct undecided_coding_system):
+	* file-coding.c (undecided_mark_coding_stream):
+	* file-coding.c (coding_category_symbol_to_id):
+	* file-coding.c (struct gzip_coding_system):
+	* file-coding.c (coding_system_type_create):
+	* file-coding.h:
+	* file-coding.h (struct Lisp_Coding_System):
+	* file-coding.h (CODING_SYSTEM_SLOT_DECLARATION):
+	* file-coding.h (coding_system_variant):
+	* file-coding.h (struct coding_system_methods):
+	* file-coding.h (DEFINE_CODING_SYSTEM_TYPE_WITH_DATA):
+	* file-coding.h (INITIALIZE_CODING_SYSTEM_TYPE_WITH_DATA):
+	* file-coding.h (struct coding_stream):
+	* fileio.c (Fsubstitute_in_file_name):
+	* floatfns.c:
+	* fns.c:
+	* fns.c (base64_encode_1):
+	* frame-gtk.c:
+	* frame-gtk.c (Fgtk_start_drag_internal):
+	* frame-impl.h (struct frame):
+	* frame-impl.h (MARKED_SLOT):
+	* frame-msw.c:
+	* frame-x.c:
+	* frame-x.c (Fcde_start_drag_internal):
+	* frame-x.c (Foffix_start_drag_internal):
+	* frame.c:
+	* frame.c (MARKED_SLOT):
+	* frame.c (allocate_frame_core):
+	* frame.c (delete_frame_internal):
+	* frame.c (Fmouse_position_as_motion_event):
+	* frameslots.h:
+	* frameslots.h (MARKED_SLOT_ARRAY):
+	* free-hook.c:
+	* glyphs-msw.c (mswindows_widget_instantiate):
+	* glyphs-x.c:
+	* glyphs-x.c (convert_EImage_to_XImage):
+	* glyphs.c:
+	* glyphs.c (process_image_string_instantiator):
+	* glyphs.c (mark_image_instance):
+	* glyphs.c (allocate_image_instance):
+	* glyphs.c (unmap_subwindow):
+	* glyphs.c (map_subwindow):
+	* glyphs.c (syms_of_glyphs):
+	* glyphs.c (specifier_type_create_image):
+	* glyphs.h:
+	* glyphs.h (struct text_image_instance):
+	* glyphs.h (struct Lisp_Image_Instance):
+	* gmalloc.c:
+	* gmalloc.c ("C"):
+	* gpmevent.c (Freceive_gpm_event):
+	* gpmevent.c (gpm_next_event_cb):
+	* gpmevent.c (vars_of_gpmevent):
+	* gtk-glue.c (gdk_event_to_emacs_event):
+	* gtk-xemacs.c (gtk_xemacs_class_init):
+	* gui-msw.c:
+	* gui-msw.c (mswindows_handle_gui_wm_command):
+	* gui-msw.c (mswindows_translate_menu_or_dialog_item):
+	* gui-x.c:
+	* gui-x.c (mark_popup_data):
+	* gui-x.c (snarf_widget_value_mapper):
+	* gui-x.c (gcpro_popup_callbacks):
+	* gui-x.c (ungcpro_popup_callbacks):
+	* gui-x.c (free_popup_widget_value_tree):
+	* gui-x.c (popup_selection_callback):
+	* gui-x.h:
+	* gui-x.h (struct popup_data):
+	* gui.c:
+	* gui.c (allocate_gui_item):
+	* gutter.c (decode_gutter_position):
+	* hash.c (NULL_ENTRY):
+	* indent.c (vmotion_1):
+	* indent.c (vmotion_pixels):
+	* input-method-motif.c (res):
+	* input-method-xlib.c (IMInstantiateCallback):
+	* input-method-xlib.c (XIM_init_device):
+	* input-method-xlib.c (res):
+	* intl-encap-win32.c:
+	* intl-encap-win32.c (qxeSHGetDataFromIDList):
+	* intl-win32.c:
+	* intl-win32.c (mswindows_multibyte_cp_type):
+	* intl-win32.c (struct mswindows_multibyte_coding_system):
+	* keymap.c:
+	* keymap.c (make_key_description):
+	* keymap.c (keymap_store):
+	* keymap.c (get_keyelt):
+	* keymap.c (keymap_lookup_1):
+	* keymap.c (define_key_parser):
+	* keymap.c (key_desc_list_to_event):
+	* keymap.c (event_matches_key_specifier_p):
+	* keymap.c (meta_prefix_char_p):
+	* keymap.c (ensure_meta_prefix_char_keymapp):
+	* keymap.c (Fdefine_key):
+	* keymap.c (struct raw_lookup_key_mapper_closure):
+	* keymap.c (raw_lookup_key):
+	* keymap.c (raw_lookup_key_mapper):
+	* keymap.c (lookup_keys):
+	* keymap.c (lookup_events):
+	* keymap.c (Flookup_key):
+	* keymap.c (struct map_keymap_unsorted_closure):
+	* keymap.c (map_keymap_unsorted_mapper):
+	* keymap.c (map_keymap_sorted):
+	* keymap.c (map_keymap_mapper):
+	* keymap.c (map_keymap):
+	* keymap.c (accessible_keymaps_mapper_1):
+	* keymap.c (Faccessible_keymaps):
+	* keymap.c (Fsingle_key_description):
+	* keymap.c (raw_keys_to_keys):
+	* keymap.c (format_raw_keys):
+	* keymap.c (where_is_recursive_mapper):
+	* keymap.c (where_is_internal):
+	* keymap.c (describe_map_mapper_shadow_search):
+	* keymap.c (keymap_lookup_inherited_mapper):
+	* keymap.c (describe_map_mapper):
+	* keymap.h (event_matches_key_specifier_p):
+	* lisp.h:
+	* lisp.h (this):
+	* lisp.h (RETURN_NOT_REACHED):
+	* lisp.h (struct Lisp_Vector):
+	* lisp.h (struct Lisp_Bit_Vector):
+	* lisp.h (UNGCPRO_1):
+	* lisp.h (NUNGCPRO):
+	* lisp.h (NNUNGCPRO):
+	* lisp.h (DECLARE_INLINE_HEADER):
+	* lrecord.h:
+	* lrecord.h (struct lrecord_header):
+	* lrecord.h (struct lcrecord_header):
+	* lrecord.h (lrecord_type):
+	* lrecord.h (struct lrecord_implementation):
+	* lrecord.h (RECORD_DUMPABLE):
+	* lrecord.h (memory_description_type):
+	* lrecord.h (data_description_entry_flags):
+	* lrecord.h (struct memory_description):
+	* lrecord.h (struct sized_memory_description):
+	* lrecord.h (XD_INDIRECT):
+	* lrecord.h (XD_IS_INDIRECT):
+	* lrecord.h (XD_DYNARR_DESC):
+	* lrecord.h (DEFINE_BASIC_LRECORD_IMPLEMENTATION):
+	* lrecord.h (MAKE_LRECORD_IMPLEMENTATION):
+	* lrecord.h (MAKE_EXTERNAL_LRECORD_IMPLEMENTATION):
+	* lrecord.h (alloc_lcrecord_type):
+	* lstream.c:
+	* lstream.c (Lstream_new):
+	* lstream.c (lisp_buffer_marker):
+	* lstream.h:
+	* lstream.h (lstream_implementation):
+	* lstream.h (DEFINE_LSTREAM_IMPLEMENTATION):
+	* lstream.h (DEFINE_LSTREAM_IMPLEMENTATION_WITH_DATA):
+	* marker.c:
+	* marker.c (copy_marker_1):
+	* mem-limits.h:
+	* menubar-gtk.c:
+	* menubar-gtk.c (gtk_popup_menu):
+	* menubar-msw.c:
+	* menubar-msw.c (mswindows_popup_menu):
+	* menubar-x.c (make_dummy_xbutton_event):
+	* menubar-x.c (command_builder_operate_menu_accelerator):
+	* menubar-x.c (menu_accelerator_safe_compare):
+	* menubar-x.c (menu_accelerator_safe_mod_compare):
+	* mule-charset.c:
+	* mule-charset.c (make_charset):
+	* mule-charset.c (Fcharset_property):
+	* mule-coding.c:
+	* mule-coding.c (ccs_description_1):
+	* mule-coding.c (ccs_description =):
+	* mule-coding.c (ccsd_description_1):
+	* mule-coding.c (ccsd_description =):
+	* nt.c (getpwnam):
+	* nt.c (init_mswindows_environment):
+	* nt.c (get_cached_volume_information):
+	* nt.c (mswindows_is_executable):
+	* nt.c (read_unc_volume):
+	* nt.c (mswindows_access):
+	* nt.c (mswindows_link):
+	* nt.c (mswindows_fstat):
+	* nt.c (mswindows_stat):
+	* nt.c (mswindows_executable_type):
+	* nt.c (Fmswindows_short_file_name):
+	* nt.c (Fmswindows_long_file_name):
+	* objects-impl.h (struct Lisp_Color_Instance):
+	* objects-impl.h (struct Lisp_Font_Instance):
+	* objects-tty.c:
+	* objects-x.c (allocate_nearest_color):
+	* objects.c:
+	* objects.c (Fmake_color_instance):
+	* objects.c (Fmake_font_instance):
+	* objects.c (font_instantiate):
+	* opaque.c:
+	* opaque.c (make_opaque):
+	* opaque.c (make_opaque_ptr):
+	* opaque.c (reinit_opaque_early):
+	* opaque.c (init_opaque_once_early):
+	* print.c:
+	* print.c (printing_badness):
+	* print.c (printing_major_badness):
+	* print.c (print_internal):
+	* print.c (debug_p4):
+	* print.c (dp):
+	* print.c (debug_backtrace):
+	* process-nt.c (nt_create_process):
+	* process-nt.c (get_internet_address):
+	* process-unix.c:
+	* process-unix.c (struct unix_process_data):
+	* process-unix.c (get_internet_address):
+	* process-unix.c (unix_alloc_process_data):
+	* process-unix.c (unix_create_process):
+	* process-unix.c (try_to_initialize_subtty):
+	* process-unix.c (unix_kill_child_process):
+	* process-unix.c (process_type_create_unix):
+	* process.c:
+	* process.c (mark_process):
+	* process.c (MARKED_SLOT):
+	* process.c (make_process_internal):
+	* process.c (Fprocess_tty_name):
+	* process.c (decode_signal):
+	* process.h:
+	* procimpl.h:
+	* procimpl.h (struct process_methods):
+	* procimpl.h (struct Lisp_Process):
+	* rangetab.c:
+	* realpath.c (readlink_and_correct_case):
+	* redisplay-x.c (x_window_output_end):
+	* redisplay-x.c (x_redraw_exposed_area):
+	* redisplay-x.c (x_clear_frame):
+	* redisplay.c:
+	* redisplay.h:
+	* redisplay.h (struct rune_dglyph):
+	* redisplay.h (struct rune):
+	* scrollbar.c:
+	* scrollbar.c (create_scrollbar_instance):
+	* specifier.c:
+	* specifier.c (specifier_empty_extra_description_1):
+	* specifier.c (make_specifier_internal):
+	* specifier.c (decode_locale_type):
+	* specifier.c (decode_how_to_add_specification):
+	* specifier.h:
+	* specifier.h (struct specifier_methods):
+	* specifier.h (DEFINE_SPECIFIER_TYPE_WITH_DATA):
+	* specifier.h (INITIALIZE_SPECIFIER_TYPE_WITH_DATA):
+	* symbols.c:
+	* symbols.c (Fsetplist):
+	* symbols.c (default_value):
+	* symbols.c (decode_magic_handler_type):
+	* symbols.c (handler_type_from_function_symbol):
+	* symbols.c (Fdefvaralias):
+	* symbols.c (init_symbols_once_early):
+	* symbols.c (reinit_symbols_early):
+	* symsinit.h:
+	* sysdep.c (sys_subshell):
+	* sysdep.c (tty_init_sys_modes_on_device):
+	* syswindows.h:
+	* text.c (dfc_convert_to_external_format):
+	* text.c (dfc_convert_to_internal_format):
+	* text.c (reinit_eistring_early):
+	* text.c (init_eistring_once_early):
+	* text.c (reinit_vars_of_text):
+	* text.h:
+	* text.h (INC_IBYTEPTR_FMT):
+	* text.h (DEC_IBYTEPTR_FMT):
+	* toolbar.c:
+	* toolbar.c (decode_toolbar_position):
+	* tooltalk.c:
+	* ui-gtk.c:
+	* unexnt.c:
+	* unexnt.c (_start):
+	* unexnt.c (unexec):
+	* unexnt.c (get_section_info):
+	* unicode.c:
+	* unicode.c (vars_of_unicode):
+	* window.c:
+	* window.c (allocate_window):
+	* window.c (new_window_mirror):
+	* window.c (update_mirror_internal):
+	* winslots.h:
+
 2003-01-08  Andy Piper  <andy@xemacs.org>
 
 	* unexcw.c (copy_executable_and_dump_data_section): Set the stack
--- a/src/README.kkcc	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/README.kkcc	Sun Jan 12 11:08:22 2003 +0000
@@ -28,137 +28,41 @@
 	the Object, if it is set to 0, the dumper does not care about it.
 		
 
-	XD_UNION
-	--------
-	We implemented XD_UNION support in (mark_with_description), so
-	we can describe exspecially console/device specific data with XD_UNION.
-	To describe with XD_UNION, we added a field to these objects, which 
-	holds the variant type of the object. This field is initialized in 
-	the appendant constructor. The variant is an integer, it has also to 
-	be described in an description, if XD_UNION is used.
-
-	Here is a pattern of a XD_UNION usage:
-
-	First, the existing variants are listed.
-	enum example_variant
-	{
-	  first_choice,
-	  second_choice
-	};
-
-	Then a field which holds the variant is added to the Lisp_Object.
-	This field determines, where pointer points to.
-	struct Lisp_...
-	{
-	  enum example_variant which_variant;
-	  ...
-	  void *pointer;
-	}
-	The variant field must be initialized in the constructor(s).
-
-	In the description, the first entry should be the which_variant field,
-	on which XD_UNION refers.
-	static const struct lrecord_description ..._description [] = {
-	  { XD_INT, offsetof (struct Lisp_..., which_variant) },
-	  ...
-	  { XD_UNION, offsetof (struct Lisp_..., which_variant), 
-	    XD_INDIRECT (0, 0), variant_description },
-	  ...
-	};
-
-	The variant_description looks like this:
-	static const struct struct_description variant_description []= {
-	  { first_choice, first_choice_description},
-	  { second_choice, second_choice__description},
-	  { XD_END }
-	};
-
-	first- and second_choice_description are common lrecord_descriptions:
-	static const struct lrecord_description first_choice_description [] = {
-	  ...
-	  { XD_END }
-	}
-	
-
-
 	TODO
 	----
-
-	The following objects have currently no description:
-	* alloc.c: lcrecord_list
-	mark_object is never called, marking is done per mane.
-
-	* buffer.c: mark_buffer
-	mark_conses_in_list implements weakness???
-
-	* extents.c: extent_info
-	loop to mark elements in list.
-
-	* frame.c: frame
-	calls mark_gutters, that calls mark_redisplay_structs	
-
-	* glyphs.c: image_instance
-	XD_UNION or convert the union members to Lisp_Objects (see Lisp_Event)
-
-	* gui-x.c: popup_data
-	calls lw_map_widget_values
-
-	* window.c: window
-	calls mark_face_cachels and mark_glyph_cachels
-
-		    window_configuration
-	loop to mark saved_windows
-
-	            window_mirror
-	calls mark_redisplay_structs
-
-	* lstream.c: lstream
-	
-
-	After all Lisp_Objects have pdump descriptions, 
+	After all Lisp_Objects have pdump descriptions (THEY DO NOW!!), 
 	(mark_with_description) can get rid of the mark_object calls.
 
 	
-	There are a few Lisp_Objects, where there occured differences and 
-	inexactness between the mark-method and the pdump description.
-	All these Lisp_Objects get dumped, so their descriptions have been
-	written, before we started our work:
+	There are a few Lisp_Objects, where there occured differences and
+	inexactness between the mark-method and the pdump description.  All
+	these Lisp_Objects get dumped (except image instances), so their
+	descriptions have been written, before we started our work:
+
+	* alloc.c: lcrecord_list
+	description:
+	mark: performs extra gc_checking_assert() for various checks.
 
 	* alloc.c: cons
 	description: car and cdr
 	mark: cdr is marked, only if its != Qnil
 
 	* alloc.c: string
-	description:
-	mark:
+	description: ???
+	mark: ???
 
-	* elhash.c: hash_table
-	description: the weakness receives no consideration
-	mark: weakness == HASH_TABLE_NON_WEAK
+	* buffer.c: buffer
+	description: XD_LISP_OBJECT indirect_children
+	mark: indirect_children not marked if Qnull_pointer
 
 	* eval.c: subr
 	description: XD_DOC_STRING doc
 	mark: empty, nothing is marked
 
 	* file-coding.c: coding_system
-	description: 
-	mark:
-
-	* frame.c: bit_vektor
-	description: XD_LISP_OBJECT next
-	mark: empty, nothing is marked
-
-	* marker.c: marker
-	description: prev & next (marker's chain)
-	mark: do not mark through marker's chain!
+	description: ???
+	mark: ???
 
-	* specifier.c: specifier
-	description: XD_STRUCT_PTR caching
-	mark: caching is not marked
-
-	* symbols.c: symbol-value-lisp-magic
-	description: only handler[] is described
-	mark: even harg[] and shadowed are marked
-
-	* data.c: ephemeron
-	description: everything is marked, there is no weakness!
+	* glyphs.c: image_instance
+	description:
+	mark: mark method sets device to nil if dead
--- a/src/alloc.c	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/alloc.c	Sun Jan 12 11:08:22 2003 +0000
@@ -50,9 +50,11 @@
 #include "elhash.h"
 #include "events.h"
 #include "extents-impl.h"
+#include "file-coding.h"
 #include "frame-impl.h"
 #include "glyphs.h"
 #include "opaque.h"
+#include "lstream.h"
 #include "process.h"
 #include "redisplay.h"
 #include "specifier.h"
@@ -61,18 +63,10 @@
 #include "window.h"
 #include "console-stream.h"
 
-#ifdef USE_KKCC
-#include "file-coding.h"
-#endif /* USE_KKCC */
-
 #ifdef DOUG_LEA_MALLOC
 #include <malloc.h>
 #endif
 
-#ifdef PDUMP
-#include "dumper.h"
-#endif
-
 EXFUN (Fgarbage_collect, 0);
 
 static void recompute_need_to_garbage_collect (void);
@@ -199,6 +193,11 @@
 /* Non-zero means we're in the process of doing the dump */
 int purify_flag;
 
+/* Non-zero means we're pdumping out or in */
+#ifdef PDUMP
+int in_pdump;
+#endif
+
 #ifdef ERROR_CHECK_TYPES
 
 Error_Behavior ERROR_ME, ERROR_ME_NOT, ERROR_ME_WARN, ERROR_ME_DEBUG_WARN;
@@ -368,7 +367,7 @@
 
   /* In practice, size will always be a multiple of four.  */
   while (beefs--)
-    (*ptr4++) = 0xDEADBEEF;
+    (*ptr4++) = 0xDEADBEEF; /* -559038737 base 10 */
 }
 
 #else /* !ERROR_CHECK_GC */
@@ -412,9 +411,9 @@
 
   /* But we do now (as of 3-27-02) go and zero out the memory.  This is a
      good thing, as it will guarantee we won't get any intermittent bugs
-     coming from an uninitiated field.  The speed loss if unnoticeable,
-     esp. as the object are not large -- large stuff like buffer text and
-     redisplay structures and allocated separately. */
+     coming from an uninitiated field.  The speed loss is unnoticeable,
+     esp. as the objects are not large -- large stuff like buffer text and
+     redisplay structures are allocated separately. */
   memset (val, 0, size);
 
   if (need_to_check_c_alloca)
@@ -429,9 +428,13 @@
    and free any lcrecord which hasn't been marked. */
 static struct lcrecord_header *all_lcrecords;
 
+/* The most basic of the lcrecord allocation functions.  Not usually called
+   directly.  Allocates an lrecord not managed by any lcrecord-list, of a
+   specified size.  See lrecord.h. */
+
 void *
-alloc_lcrecord (Bytecount size,
-		const struct lrecord_implementation *implementation)
+basic_alloc_lcrecord (Bytecount size,
+		      const struct lrecord_implementation *implementation)
 {
   struct lcrecord_header *lcheader;
 
@@ -510,6 +513,28 @@
     }
 }
 
+/* Bitwise copy all parts of a Lisp object other than the header */
+
+void
+copy_lisp_object (Lisp_Object dst, Lisp_Object src)
+{
+  const struct lrecord_implementation *imp =
+    XRECORD_LHEADER_IMPLEMENTATION (src);
+  Bytecount size = lisp_object_size (src);
+
+  assert (imp == XRECORD_LHEADER_IMPLEMENTATION (dst));
+  assert (size == lisp_object_size (dst));
+
+  if (imp->basic_p)
+    memcpy ((char *) XRECORD_LHEADER (dst) + sizeof (struct lrecord_header),
+	    (char *) XRECORD_LHEADER (src) + sizeof (struct lrecord_header),
+	    size - sizeof (struct lrecord_header));
+  else
+    memcpy ((char *) XRECORD_LHEADER (dst) + sizeof (struct lcrecord_header),
+	    (char *) XRECORD_LHEADER (src) + sizeof (struct lcrecord_header),
+	    size - sizeof (struct lcrecord_header));
+}
+
 
 /************************************************************************/
 /*			  Debugger support				*/
@@ -597,7 +622,7 @@
    string_chars_blocks.  Furthermore, no one string stretches across
    two string_chars_blocks.
 
-   Vectors are each malloc()ed separately, similar to lcrecords.
+   Vectors are each malloc()ed separately as lcrecords.
 
    In the following discussion, we use conses, but it applies equally
    well to the other fixed-size types.
@@ -786,8 +811,9 @@
       MINIMUM_ALLOWED_FIXED_TYPE_CELLS_##type)			\
     {								\
       result = (structtype *) type##_free_list;			\
-      /* Before actually using the chain pointer,		\
-	 we complement all its bits; see FREE_FIXED_TYPE(). */	\
+      assert (LRECORD_FREE_P (result));				\
+      /* Before actually using the chain pointer, we complement	\
+	 all its bits; see PUT_FIXED_TYPE_ON_FREE_LIST(). */	\
       type##_free_list = (Lisp_Free *)				\
 	(~ (EMACS_UINT) (type##_free_list->chain));		\
       gc_count_num_##type##_freelist--;				\
@@ -882,6 +908,7 @@
 
 #define FREE_FIXED_TYPE(type, structtype, ptr) do {		\
   structtype *FFT_ptr = (ptr);					\
+  gc_checking_assert (!LRECORD_FREE_P (FFT_ptr));		\
   ADDITIONAL_FREE_##type (FFT_ptr);				\
   deadbeef_memory (FFT_ptr, sizeof (structtype));		\
   PUT_FIXED_TYPE_ON_FREE_LIST (type, structtype, FFT_ptr);	\
@@ -896,13 +923,20 @@
    completely necessary but helps keep things saner: e.g. this way,
    repeatedly allocating and freeing a cons will not result in
    the consing-since-gc counter advancing, which would cause a GC
-   and somewhat defeat the purpose of explicitly freeing. */
-
+   and somewhat defeat the purpose of explicitly freeing.
+
+   We also disable this mechanism entirely when ALLOC_NO_POOLS is
+   set, which is used for Purify and the like. */
+
+#ifndef ALLOC_NO_POOLS
 #define FREE_FIXED_TYPE_WHEN_NOT_IN_GC(type, structtype, ptr)	\
 do { FREE_FIXED_TYPE (type, structtype, ptr);			\
      DECREMENT_CONS_COUNTER (sizeof (structtype));		\
      gc_count_num_##type##_freelist++;				\
    } while (0)
+#else
+#define FREE_FIXED_TYPE_WHEN_NOT_IN_GC(type, structtype, ptr)
+#endif
 
 
 
@@ -939,13 +973,12 @@
   return 0;
 }
 
-static const struct lrecord_description cons_description[] = {
+static const struct memory_description cons_description[] = {
   { XD_LISP_OBJECT, offsetof (Lisp_Cons, car_) },
   { XD_LISP_OBJECT, offsetof (Lisp_Cons, cdr_) },
   { XD_END }
 };
 
-#ifdef USE_KKCC
 DEFINE_BASIC_LRECORD_IMPLEMENTATION ("cons", cons,
 				     1, /*dumpable-flag*/
 				     mark_cons, print_cons, 0,
@@ -958,19 +991,6 @@
 				     0,
 				     cons_description,
 				     Lisp_Cons);
-#else /* not USE_KKCC */
-DEFINE_BASIC_LRECORD_IMPLEMENTATION ("cons", cons,
-				     mark_cons, print_cons, 0,
-				     cons_equal,
-				     /*
-				      * No `hash' method needed.
-				      * internal_hash knows how to
-				      * handle conses.
-				      */
-				     0,
-				     cons_description,
-				     Lisp_Cons);
-#endif /* not USE_KKCC */
 
 DEFUN ("cons", Fcons, 2, 2, 0, /*
 Create a new cons, give it CAR and CDR as components, and return it.
@@ -1168,36 +1188,28 @@
 				     depth + 1));
 }
 
-static const struct lrecord_description vector_description[] = {
+static const struct memory_description vector_description[] = {
   { XD_LONG,              offsetof (Lisp_Vector, size) },
   { XD_LISP_OBJECT_ARRAY, offsetof (Lisp_Vector, contents), XD_INDIRECT(0, 0) },
   { XD_END }
 };
 
-#ifdef USE_KKCC
-DEFINE_LRECORD_SEQUENCE_IMPLEMENTATION("vector", vector,
-				       1, /*dumpable-flag*/
-				       mark_vector, print_vector, 0,
-				       vector_equal,
-				       vector_hash,
-				       vector_description,
-				       size_vector, Lisp_Vector);
-#else /* not USE_KKCC */
-DEFINE_LRECORD_SEQUENCE_IMPLEMENTATION("vector", vector,
-				       mark_vector, print_vector, 0,
-				       vector_equal,
-				       vector_hash,
-				       vector_description,
-				       size_vector, Lisp_Vector);
-#endif /* not USE_KKCC */
+DEFINE_LRECORD_SEQUENCE_IMPLEMENTATION ("vector", vector,
+					1, /*dumpable-flag*/
+					mark_vector, print_vector, 0,
+					vector_equal,
+					vector_hash,
+					vector_description,
+					size_vector, Lisp_Vector);
 /* #### should allocate `small' vectors from a frob-block */
 static Lisp_Vector *
 make_vector_internal (Elemcount sizei)
 {
-  /* no vector_next */
+  /* no `next' field; we use lcrecords */
   Bytecount sizem = FLEXIBLE_ARRAY_STRUCT_SIZEOF (Lisp_Vector, Lisp_Object,
-					       contents, sizei);
-  Lisp_Vector *p = (Lisp_Vector *) alloc_lcrecord (sizem, &lrecord_vector);
+						  contents, sizei);
+  Lisp_Vector *p =
+    (Lisp_Vector *) basic_alloc_lcrecord (sizem, &lrecord_vector);
 
   p->size = sizei;
   return p;
@@ -1344,27 +1356,19 @@
 /*			 Bit Vector allocation				*/
 /************************************************************************/
 
-static Lisp_Object all_bit_vectors;
-
 /* #### should allocate `small' bit vectors from a frob-block */
 static Lisp_Bit_Vector *
 make_bit_vector_internal (Elemcount sizei)
 {
+  /* no `next' field; we use lcrecords */
   Elemcount num_longs = BIT_VECTOR_LONG_STORAGE (sizei);
   Bytecount sizem = FLEXIBLE_ARRAY_STRUCT_SIZEOF (Lisp_Bit_Vector,
-						     unsigned long,
-						     bits, num_longs);
-  Lisp_Bit_Vector *p = (Lisp_Bit_Vector *) allocate_lisp_storage (sizem);
-  set_lheader_implementation (&p->lheader, &lrecord_bit_vector);
-
-  INCREMENT_CONS_COUNTER (sizem, "bit-vector");
+						  unsigned long,
+						  bits, num_longs);
+  Lisp_Bit_Vector *p = (Lisp_Bit_Vector *)
+    basic_alloc_lcrecord (sizem, &lrecord_bit_vector);
 
   bit_vector_length (p) = sizei;
-  bit_vector_next   (p) = all_bit_vectors;
-  /* make sure the extra bits in the last long are 0; the calling
-     functions might not set them. */
-  p->bits[num_longs - 1] = 0;
-  all_bit_vectors = wrap_bit_vector (p);
   return p;
 }
 
@@ -1690,133 +1694,152 @@
   return wrap_event (e);
 }
 
-#ifdef USE_KKCC
+#ifdef EVENT_DATA_AS_OBJECTS
 DECLARE_FIXED_TYPE_ALLOC (key_data, Lisp_Key_Data);
 #define MINIMUM_ALLOWED_FIXED_TYPE_CELLS_key_data 1000
 
 Lisp_Object
-allocate_key_data (void)
+make_key_data (void)
 {
   Lisp_Key_Data *d;
 
   ALLOCATE_FIXED_TYPE (key_data, Lisp_Key_Data, d);
+  xzero (*d);
   set_lheader_implementation (&d->lheader, &lrecord_key_data);
-
-  return wrap_key_data(d);
+  d->keysym = Qnil;
+
+  return wrap_key_data (d);
 }
 
 DECLARE_FIXED_TYPE_ALLOC (button_data, Lisp_Button_Data);
 #define MINIMUM_ALLOWED_FIXED_TYPE_CELLS_button_data 1000
 
 Lisp_Object
-allocate_button_data (void)
+make_button_data (void)
 {
   Lisp_Button_Data *d;
 
   ALLOCATE_FIXED_TYPE (button_data, Lisp_Button_Data, d);
+  xzero (*d);
   set_lheader_implementation (&d->lheader, &lrecord_button_data);
 
-  return wrap_button_data(d);
+  return wrap_button_data (d);
 }
 
 DECLARE_FIXED_TYPE_ALLOC (motion_data, Lisp_Motion_Data);
 #define MINIMUM_ALLOWED_FIXED_TYPE_CELLS_motion_data 1000
 
 Lisp_Object
-allocate_motion_data (void)
+make_motion_data (void)
 {
   Lisp_Motion_Data *d;
 
   ALLOCATE_FIXED_TYPE (motion_data, Lisp_Motion_Data, d);
+  xzero (*d);
   set_lheader_implementation (&d->lheader, &lrecord_motion_data);
 
-  return wrap_motion_data(d);
+  return wrap_motion_data (d);
 }
 
 DECLARE_FIXED_TYPE_ALLOC (process_data, Lisp_Process_Data);
 #define MINIMUM_ALLOWED_FIXED_TYPE_CELLS_process_data 1000
 
 Lisp_Object
-allocate_process_data (void)
+make_process_data (void)
 {
   Lisp_Process_Data *d;
 
   ALLOCATE_FIXED_TYPE (process_data, Lisp_Process_Data, d);
+  xzero (*d);
   set_lheader_implementation (&d->lheader, &lrecord_process_data);
-
-  return wrap_process_data(d);
+  d->process = Qnil;
+
+  return wrap_process_data (d);
 }
 
 DECLARE_FIXED_TYPE_ALLOC (timeout_data, Lisp_Timeout_Data);
 #define MINIMUM_ALLOWED_FIXED_TYPE_CELLS_timeout_data 1000
 
 Lisp_Object
-allocate_timeout_data (void)
+make_timeout_data (void)
 {
   Lisp_Timeout_Data *d;
 
   ALLOCATE_FIXED_TYPE (timeout_data, Lisp_Timeout_Data, d);
+  xzero (*d);
   set_lheader_implementation (&d->lheader, &lrecord_timeout_data);
-
-  return wrap_timeout_data(d);
+  d->function = Qnil;
+  d->object = Qnil;
+
+  return wrap_timeout_data (d);
 }
 
 DECLARE_FIXED_TYPE_ALLOC (magic_data, Lisp_Magic_Data);
 #define MINIMUM_ALLOWED_FIXED_TYPE_CELLS_magic_data 1000
 
 Lisp_Object
-allocate_magic_data (void)
+make_magic_data (void)
 {
   Lisp_Magic_Data *d;
 
   ALLOCATE_FIXED_TYPE (magic_data, Lisp_Magic_Data, d);
+  xzero (*d);
   set_lheader_implementation (&d->lheader, &lrecord_magic_data);
 
-  return wrap_magic_data(d);
+  return wrap_magic_data (d);
 }
 
 DECLARE_FIXED_TYPE_ALLOC (magic_eval_data, Lisp_Magic_Eval_Data);
 #define MINIMUM_ALLOWED_FIXED_TYPE_CELLS_magic_eval_data 1000
 
 Lisp_Object
-allocate_magic_eval_data (void)
+make_magic_eval_data (void)
 {
   Lisp_Magic_Eval_Data *d;
 
   ALLOCATE_FIXED_TYPE (magic_eval_data, Lisp_Magic_Eval_Data, d);
+  xzero (*d);
   set_lheader_implementation (&d->lheader, &lrecord_magic_eval_data);
-
-  return wrap_magic_eval_data(d);
+  d->object = Qnil;
+
+  return wrap_magic_eval_data (d);
 }
 
 DECLARE_FIXED_TYPE_ALLOC (eval_data, Lisp_Eval_Data);
 #define MINIMUM_ALLOWED_FIXED_TYPE_CELLS_eval_data 1000
 
 Lisp_Object
-allocate_eval_data (void)
+make_eval_data (void)
 {
   Lisp_Eval_Data *d;
 
   ALLOCATE_FIXED_TYPE (eval_data, Lisp_Eval_Data, d);
+  xzero (*d);
   set_lheader_implementation (&d->lheader, &lrecord_eval_data);
-
-  return wrap_eval_data(d);
+  d->function = Qnil;
+  d->object = Qnil;
+
+  return wrap_eval_data (d);
 }
 
 DECLARE_FIXED_TYPE_ALLOC (misc_user_data, Lisp_Misc_User_Data);
 #define MINIMUM_ALLOWED_FIXED_TYPE_CELLS_misc_user_data 1000
 
 Lisp_Object
-allocate_misc_user_data (void)
+make_misc_user_data (void)
 {
   Lisp_Misc_User_Data *d;
 
   ALLOCATE_FIXED_TYPE (misc_user_data, Lisp_Misc_User_Data, d);
+  xzero (*d);
   set_lheader_implementation (&d->lheader, &lrecord_misc_user_data);
-
-  return wrap_misc_user_data(d);
+  d->function = Qnil;
+  d->object = Qnil;
+
+  return wrap_misc_user_data (d);
 }
-#endif /* USE_KKCC */
+
+#endif /* EVENT_DATA_AS_OBJECTS */
 
 /************************************************************************/
 /*			 Marker allocation				*/
@@ -1864,9 +1887,9 @@
 
 /* The data for "short" strings generally resides inside of structs of type
    string_chars_block. The Lisp_String structure is allocated just like any
-   other Lisp object (except for vectors), and these are freelisted when
-   they get garbage collected. The data for short strings get compacted,
-   but the data for large strings do not.
+   other basic lrecord, and these are freelisted when they get garbage
+   collected. The data for short strings get compacted, but the data for
+   large strings do not.
 
    Previously Lisp_String structures were relocated, but this caused a lot
    of bus-errors because the C code didn't include enough GCPRO's for
@@ -1896,7 +1919,7 @@
 	  !memcmp (XSTRING_DATA (obj1), XSTRING_DATA (obj2), len));
 }
 
-static const struct lrecord_description string_description[] = {
+static const struct memory_description string_description[] = {
   { XD_BYTECOUNT,       offsetof (Lisp_String, size_) },
   { XD_OPAQUE_DATA_PTR, offsetof (Lisp_String, data_), XD_INDIRECT(0, 1) },
   { XD_LISP_OBJECT,     offsetof (Lisp_String, plist) },
@@ -1955,7 +1978,6 @@
    is done with the ADDITIONAL_FREE_string macro, which is the
    standard way to do finalization when using
    SWEEP_FIXED_TYPE_BLOCK(). */
-#ifdef USE_KKCC
 DEFINE_BASIC_LRECORD_IMPLEMENTATION_WITH_PROPS ("string", string,
 						1, /*dumpable-flag*/
 						mark_string, print_string,
@@ -1966,17 +1988,6 @@
 						string_remprop,
 						string_plist,
 						Lisp_String);
-#else /* not USE_KKCC */
-DEFINE_BASIC_LRECORD_IMPLEMENTATION_WITH_PROPS ("string", string,
-						mark_string, print_string,
-						0, string_equal, 0,
-						string_description,
-						string_getprop,
-						string_putprop,
-						string_remprop,
-						string_plist,
-						Lisp_String);
-#endif /* not USE_KKCC */
 /* String blocks contain this many useful bytes. */
 #define STRING_CHARS_BLOCK_SIZE					\
   ((Bytecount) (8192 - MALLOC_OVERHEAD -			\
@@ -2484,32 +2495,30 @@
 /************************************************************************/
 
 /* Lcrecord lists are used to manage the allocation of particular
-   sorts of lcrecords, to avoid calling alloc_lcrecord() (and thus
+   sorts of lcrecords, to avoid calling basic_alloc_lcrecord() (and thus
    malloc() and garbage-collection junk) as much as possible.
    It is similar to the Blocktype class.
 
-   It works like this:
-
-   1) Create an lcrecord-list object using make_lcrecord_list().
-      This is often done at initialization.  Remember to staticpro_nodump
-      this object!  The arguments to make_lcrecord_list() are the
-      same as would be passed to alloc_lcrecord().
-   2) Instead of calling alloc_lcrecord(), call allocate_managed_lcrecord()
-      and pass the lcrecord-list earlier created.
-   3) When done with the lcrecord, call free_managed_lcrecord().
-      The standard freeing caveats apply: ** make sure there are no
-      pointers to the object anywhere! **
-   4) Calling free_managed_lcrecord() is just like kissing the
-      lcrecord goodbye as if it were garbage-collected.  This means:
-      -- the contents of the freed lcrecord are undefined, and the
-         contents of something produced by allocate_managed_lcrecord()
-	 are undefined, just like for alloc_lcrecord().
-      -- the mark method for the lcrecord's type will *NEVER* be called
-         on freed lcrecords.
-      -- the finalize method for the lcrecord's type will be called
-         at the time that free_managed_lcrecord() is called.
-
-   */
+   See detailed comment in lcrecord.h.
+*/
+
+const struct memory_description free_description[] = {
+  { XD_LISP_OBJECT, offsetof (struct free_lcrecord_header, chain), 0, 0,
+    XD_FLAG_FREE_LISP_OBJECT },
+  { XD_END }
+};
+
+DEFINE_LRECORD_IMPLEMENTATION ("free", free,
+			       0, /*dumpable-flag*/
+			       0, internal_object_printer,
+			       0, 0, 0, free_description,
+			       struct free_lcrecord_header);
+
+const struct memory_description lcrecord_list_description[] = {
+  { XD_LISP_OBJECT, offsetof (struct lcrecord_list, free), 0, 0,
+    XD_FLAG_FREE_LISP_OBJECT },
+  { XD_END }
+};
 
 static Lisp_Object
 mark_lcrecord_list (Lisp_Object obj)
@@ -2528,17 +2537,17 @@
 	 ! MARKED_RECORD_HEADER_P (lheader)
 	 &&
 	 /* Only lcrecords should be here. */
-	 ! LHEADER_IMPLEMENTATION (lheader)->basic_p
+	 ! list->implementation->basic_p
 	 &&
 	 /* Only free lcrecords should be here. */
 	 free_header->lcheader.free
 	 &&
 	 /* The type of the lcrecord must be right. */
-	 LHEADER_IMPLEMENTATION (lheader) == list->implementation
+	 lheader->type == lrecord_type_free
 	 &&
 	 /* So must the size. */
-	 (LHEADER_IMPLEMENTATION (lheader)->static_size == 0 ||
-	  LHEADER_IMPLEMENTATION (lheader)->static_size == list->size)
+	 (list->implementation->static_size == 0 ||
+	  list->implementation->static_size == list->size)
 	 );
 
       MARK_RECORD_HEADER (lheader);
@@ -2548,25 +2557,21 @@
   return Qnil;
 }
 
-#ifdef USE_KKCC
 DEFINE_LRECORD_IMPLEMENTATION ("lcrecord-list", lcrecord_list,
 			       0, /*dumpable-flag*/
 			       mark_lcrecord_list, internal_object_printer,
-			       0, 0, 0, 0, struct lcrecord_list);
-#else /* not USE_KKCC */
-DEFINE_LRECORD_IMPLEMENTATION ("lcrecord-list", lcrecord_list,
-			       mark_lcrecord_list, internal_object_printer,
-			       0, 0, 0, 0, struct lcrecord_list);
-#endif /* not USE_KKCC */
+			       0, 0, 0, lcrecord_list_description,
+			       struct lcrecord_list);
 
 Lisp_Object
 make_lcrecord_list (Elemcount size,
 		    const struct lrecord_implementation *implementation)
 {
-  struct lcrecord_list *p =
-    /* Avoid infinite recursion allocating this */
-    alloc_unmanaged_lcrecord_type (struct lcrecord_list,
-				   &lrecord_lcrecord_list);
+  /* Don't use alloc_lcrecord_type() avoid infinite recursion
+     allocating this, */
+  struct lcrecord_list *p = (struct lcrecord_list *)
+    basic_alloc_lcrecord (sizeof (struct lcrecord_list),
+			  &lrecord_lcrecord_list);
 
   p->implementation = implementation;
   p->size = size;
@@ -2575,7 +2580,7 @@
 }
 
 Lisp_Object
-allocate_managed_lcrecord (Lisp_Object lcrecord_list)
+alloc_managed_lcrecord (Lisp_Object lcrecord_list)
 {
   struct lcrecord_list *list = XLCRECORD_LIST (lcrecord_list);
   if (!NILP (list->free))
@@ -2583,33 +2588,41 @@
       Lisp_Object val = list->free;
       struct free_lcrecord_header *free_header =
 	(struct free_lcrecord_header *) XPNTR (val);
+      struct lrecord_header *lheader = &free_header->lcheader.lheader;
 
 #ifdef ERROR_CHECK_GC
-      struct lrecord_header *lheader = &free_header->lcheader.lheader;
-
+      /* Major overkill here. */
       /* There should be no other pointers to the free list. */
       assert (! MARKED_RECORD_HEADER_P (lheader));
-      /* Only lcrecords should be here. */
-      assert (! LHEADER_IMPLEMENTATION (lheader)->basic_p);
       /* Only free lcrecords should be here. */
       assert (free_header->lcheader.free);
+      assert (lheader->type == lrecord_type_free);
+      /* Only lcrecords should be here. */
+      assert (! (list->implementation->basic_p));
+#if 0 /* Not used anymore, now that we set the type of the header to
+	 lrecord_type_free. */
       /* The type of the lcrecord must be right. */
       assert (LHEADER_IMPLEMENTATION (lheader) == list->implementation);
+#endif /* 0 */
       /* So must the size. */
-      assert (LHEADER_IMPLEMENTATION (lheader)->static_size == 0 ||
-	      LHEADER_IMPLEMENTATION (lheader)->static_size == list->size);
+      assert (list->implementation->static_size == 0 ||
+	      list->implementation->static_size == list->size);
 #endif /* ERROR_CHECK_GC */
 
       list->free = free_header->chain;
       free_header->lcheader.free = 0;
+      /* Put back the correct type, as we set it to lrecord_type_free. */
+      lheader->type = list->implementation->lrecord_type_index;
+      zero_sized_lcrecord (free_header, list->size);
       return val;
     }
   else
-    return wrap_pointer_1 (alloc_lcrecord (list->size, list->implementation));
+    return wrap_pointer_1 (basic_alloc_lcrecord (list->size,
+						 list->implementation));
 }
 
 /* "Free" a Lisp object LCRECORD by placing it on its associated free list
-   LCRECORD_LIST; next time allocate_managed_lcrecord() is called with the
+   LCRECORD_LIST; next time alloc_managed_lcrecord() is called with the
    same LCRECORD_LIST as its parameter, it will return an object from the
    free list, which may be this one.  Be VERY VERY SURE there are no
    pointers to this object hanging around anywhere where they might be
@@ -2641,15 +2654,18 @@
   
   /* Make sure the size is correct.  This will catch, for example,
      putting a window configuration on the wrong free list. */
-  gc_checking_assert ((implementation->size_in_bytes_method ?
-		       implementation->size_in_bytes_method (lheader) :
-		       implementation->static_size)
-		      == list->size);
+  gc_checking_assert (detagged_lisp_object_size (lheader) == list->size);
   /* Make sure the object isn't already freed. */
   gc_checking_assert (!free_header->lcheader.free);
   
   if (implementation->finalizer)
     implementation->finalizer (lheader, 0);
+  /* Yes, there are two ways to indicate freeness -- the type is
+     lrecord_type_free or the ->free flag is set.  We used to do only the
+     latter; now we do the former as well for KKCC purposes.  Probably
+     safer in any case, as we will lose quicker this way than keeping
+     around an lrecord of apparently correct type but bogus junk in it. */
+  MARK_LRECORD_AS_FREE (lheader);
   free_header->chain = list->free;
   free_header->lcheader.free = 1;
   list->free = lcrecord;
@@ -2665,7 +2681,7 @@
     all_lcrecord_lists[imp->lrecord_type_index] =
       make_lcrecord_list (size, imp);
 
-  return XPNTR (allocate_managed_lcrecord
+  return XPNTR (alloc_managed_lcrecord
 		(all_lcrecord_lists[imp->lrecord_type_index]));
 }
 
@@ -2716,43 +2732,46 @@
    objects, which are global variables like Qfoo or Vbar, themselves are
    pointers to heap objects.  Each needs to be described to pdump as a
    "root pointer"; this happens in the call to staticpro(). */
-static const struct lrecord_description staticpro_description_1[] = {
+static const struct memory_description staticpro_description_1[] = {
   { XD_END }
 };
 
-static const struct struct_description staticpro_description = {
+static const struct sized_memory_description staticpro_description = {
   sizeof (Lisp_Object *),
   staticpro_description_1
 };
 
-static const struct lrecord_description staticpros_description_1[] = {
+static const struct memory_description staticpros_description_1[] = {
   XD_DYNARR_DESC (Lisp_Object_ptr_dynarr, &staticpro_description),
   { XD_END }
 };
 
-static const struct struct_description staticpros_description = {
+static const struct sized_memory_description staticpros_description = {
   sizeof (Lisp_Object_ptr_dynarr),
   staticpros_description_1
 };
 
 #ifdef DEBUG_XEMACS
 
-static const struct lrecord_description staticpro_one_name_description_1[] = {
+static const struct memory_description staticpro_one_name_description_1[] = {
   { XD_C_STRING, 0 },
   { XD_END }
 };
 
-static const struct struct_description staticpro_one_name_description = {
+static const struct sized_memory_description staticpro_one_name_description = {
   sizeof (char *),
   staticpro_one_name_description_1
 };
 
-static const struct lrecord_description staticpro_names_description_1[] = {
+static const struct memory_description staticpro_names_description_1[] = {
   XD_DYNARR_DESC (char_ptr_dynarr, &staticpro_one_name_description),
   { XD_END }
 };
 
-static const struct struct_description staticpro_names_description = {
+
+extern const struct sized_memory_description staticpro_names_description;
+
+const struct sized_memory_description staticpro_names_description = {
   sizeof (char_ptr_dynarr),
   staticpro_names_description_1
 };
@@ -2769,7 +2788,7 @@
 {
   Dynarr_add (staticpros, varaddress);
   Dynarr_add (staticpro_names, varname);
-  dump_add_root_object (varaddress);
+  dump_add_root_lisp_object (varaddress);
 }
 
 
@@ -2806,7 +2825,7 @@
 staticpro (Lisp_Object *varaddress)
 {
   Dynarr_add (staticpros, varaddress);
-  dump_add_root_object (varaddress);
+  dump_add_root_lisp_object (varaddress);
 }
 
 
@@ -2861,23 +2880,24 @@
 #endif
 
 
-
-#ifdef USE_KKCC
-/* The following functions implement the new mark algorithm. 
-   They mark objects according to their descriptions. They 
-   are modeled on the corresponding pdumper procedures. */
-
-static void mark_struct_contents (const void *data,
-						   const struct struct_description *
-						   sdesc,
-						   int count);
+static const struct memory_description lisp_object_description_1[] = {
+  { XD_LISP_OBJECT, 0 },
+  { XD_END }
+};
+
+const struct sized_memory_description lisp_object_description = {
+  sizeof (Lisp_Object),
+  lisp_object_description_1
+};
+
+#if defined (USE_KKCC) || defined (PDUMP)
 
 /* This function extracts the value of a count variable described somewhere 
    else in the description. It is converted corresponding to the type */ 
-static EMACS_INT
-get_indirect_count (EMACS_INT code,
-			  const struct lrecord_description *idesc,
-			  const void *idata)
+EMACS_INT
+lispdesc_indirect_count_1 (EMACS_INT code,
+			   const struct memory_description *idesc,
+			   const void *idata)
 {
   EMACS_INT count;
   const void *irdata;
@@ -2885,27 +2905,32 @@
   int line = XD_INDIRECT_VAL (code);
   int delta = XD_INDIRECT_DELTA (code);
 
-  irdata = ((char *)idata) + idesc[line].offset;
+  irdata = ((char *) idata) +
+    lispdesc_indirect_count (idesc[line].offset, idesc, idata);
   switch (idesc[line].type)
     {
     case XD_BYTECOUNT:
-      count = *(Bytecount *)irdata;
+      count = * (Bytecount *) irdata;
       break;
     case XD_ELEMCOUNT:
-      count = *(Elemcount *)irdata;
+      count = * (Elemcount *) irdata;
       break;
     case XD_HASHCODE:
-      count = *(Hashcode *)irdata;
+      count = * (Hashcode *) irdata;
       break;
     case XD_INT:
-      count = *(int *)irdata;
+      count = * (int *) irdata;
       break;
     case XD_LONG:
-      count = *(long *)irdata;
+      count = * (long *) irdata;
       break;
     default:
       stderr_out ("Unsupported count type : %d (line = %d, code = %ld)\n",
-		  idesc[line].type, line, (long)code);
+		  idesc[line].type, line, (long) code);
+#ifdef PDUMP
+      if (in_pdump)
+	pdump_backtrace ();
+#endif
       count = 0; /* warning suppression */
       abort ();
     }
@@ -2913,159 +2938,155 @@
   return count;
 }
 
-/* This function is called to mark the elements of an object. It processes
-   the description of the object and calls mark object with every described
-   object. */
-static void
-mark_with_description (const void *lheader, const struct lrecord_description *desc)
+/* SDESC is a "description map" (basically, a list of offsets used for
+   successive indirections) and OBJ is the first object to indirect off of.
+   Return the description ultimately found. */
+
+const struct sized_memory_description *
+lispdesc_indirect_description_1 (const void *obj,
+				 const struct sized_memory_description *sdesc)
 {
   int pos;
 
-  static const Lisp_Object *last_occured_object = (Lisp_Object *) 0;
-  static int mark_last_occured_object = 0;
-
- reprocess_desc:
-  for (pos=0; desc[pos].type != XD_END; pos++)
+  for (pos = 0; sdesc[pos].size >= 0; pos++)
+    obj = * (const void **) ((const char *) obj + sdesc[pos].size);
+
+  return (const struct sized_memory_description *) obj;
+}
+
+/* Compute the size of the data at RDATA, described by a single entry
+   DESC1 in a description array.  OBJ and DESC are used for
+   XD_INDIRECT references. */
+
+static Bytecount
+lispdesc_one_description_line_size (void *rdata,
+				    const struct memory_description *desc1,
+				    const void *obj,
+				    const struct memory_description *desc)
+{
+ union_switcheroo:
+  switch (desc1->type)
     {
-      const void *rdata = (const char *)lheader + desc[pos].offset;
-      switch (desc[pos].type) {
-      case XD_LISP_OBJECT: 
-	{
-	  const Lisp_Object *stored_obj = (const Lisp_Object *)rdata;
-
-	  if (EQ (*stored_obj, Qnull_pointer))
+    case XD_LISP_OBJECT_ARRAY:
+      {
+	EMACS_INT val = lispdesc_indirect_count (desc1->data1, desc, obj);
+	return (val * sizeof (Lisp_Object));
+      }
+    case XD_LISP_OBJECT:
+    case XD_LO_LINK:
+      return sizeof (Lisp_Object);
+    case XD_OPAQUE_PTR:
+      return sizeof (void *);
+    case XD_STRUCT_PTR:
+      {
+	EMACS_INT val = lispdesc_indirect_count (desc1->data1, desc, obj);
+	return val * sizeof (void *);
+      }
+    case XD_STRUCT_ARRAY:
+      {
+	EMACS_INT val = lispdesc_indirect_count (desc1->data1, desc, obj);
+	    
+	return (val *
+		lispdesc_structure_size
+		(rdata, lispdesc_indirect_description (obj, desc1->data2)));
+      }
+    case XD_OPAQUE_DATA_PTR:
+      return sizeof (void *);
+    case XD_UNION_DYNAMIC_SIZE:
+      {
+	/* If an explicit size was given in the first-level structure
+	   description, use it; else compute size based on current union
+	   constant. */
+	const struct sized_memory_description *sdesc =
+	  lispdesc_indirect_description (obj, desc1->data2);
+	if (sdesc->size)
+	  return sdesc->size;
+	else
+	  {
+	    desc1 = lispdesc_process_xd_union (desc1, desc, obj);
+	    if (desc1)
+	      goto union_switcheroo;
 	    break;
-
-	  if (desc[pos+1].type == XD_END)
-	    {
-	      mark_last_occured_object = 1;
-	      last_occured_object = stored_obj;
-	      break;
-	    }
-	  else
-	    {
-	      mark_object (*stored_obj);
-	    }
-	  
-	  
-	  break;
-	}
-      case XD_LISP_OBJECT_ARRAY:
-	{
-	  int i;
-	  EMACS_INT count = desc[pos].data1;
-	  if (XD_IS_INDIRECT (count))
-	    count = get_indirect_count (count, desc, lheader);
-	
-	  for (i = 0; i < count; i++)
-	    {
-	      const Lisp_Object *stored_obj = ((const Lisp_Object *)rdata) + i;
-
-	      if (EQ (*stored_obj, Qnull_pointer))
-		break;
-
-	      mark_object (*stored_obj);
-	    }
-	  break;
-	}
-      case XD_SPECIFIER_END:
-	desc = ((const Lisp_Specifier *)lheader)->methods->extra_description;
-	goto reprocess_desc;
-	break;
-      case XD_CODING_SYSTEM_END:
-	desc = ((const Lisp_Coding_System *)lheader)->methods->extra_description;
-	goto reprocess_desc;
-	break;
-      case XD_BYTECOUNT:
-	break;
-      case XD_ELEMCOUNT:
-	break;
-      case XD_HASHCODE:
-	break;
-      case XD_INT:
-	break;
-      case XD_LONG:
-	break;
-      case XD_INT_RESET:
-	break;
-      case XD_LO_LINK:
-	break;
-      case XD_OPAQUE_PTR:
-	break;
-      case XD_OPAQUE_DATA_PTR:
-	break;
-      case XD_C_STRING:
-	break;
-      case XD_DOC_STRING:
-	break;
-      case XD_STRUCT_PTR:
-	{
-	  EMACS_INT count = desc[pos].data1;
-	  const struct struct_description *sdesc = desc[pos].data2;
-	  const char *dobj = *(const char **)rdata;
-	  if (dobj)
-	    {
-	      if (XD_IS_INDIRECT (count))
-		count = get_indirect_count (count, desc, lheader);
-	      mark_struct_contents (dobj, sdesc, count);
-	    }
-	  break;
-	}
-      case XD_STRUCT_ARRAY:
-	{
-	  EMACS_INT count = desc[pos].data1;
-	  const struct struct_description *sdesc = desc[pos].data2;
-		      
-	  if (XD_IS_INDIRECT (count))
-	    count = get_indirect_count (count, desc, lheader);
-		      
-	  mark_struct_contents (rdata, sdesc, count);
-	  break;
-	}
-      case XD_UNION:
-	{
-	  int count = 0;
-	  int variant = desc[pos].data1;
-	  const struct struct_description *sdesc = desc[pos].data2;
-	  const char *dobj = *(const char **)rdata;
-	  if (XD_IS_INDIRECT (variant))
-	    variant = get_indirect_count (variant, desc, lheader);
-	  
-	  for (count=0; sdesc[count].size != XD_END; count++)
-	    {
-	      if (sdesc[count].size == variant)
-		{
-		  mark_with_description(dobj, sdesc[count].description);
-		  break;
-		}
-	    }
-	  break;
-	}
-		    
-      default:
-	stderr_out ("Unsupported description type : %d\n", desc[pos].type);
-	abort ();
+	  }
+      }
+    case XD_UNION:
+      {
+	/* If an explicit size was given in the first-level structure
+	   description, use it; else compute size based on maximum of all
+	   possible structures. */
+	const struct sized_memory_description *sdesc =
+	  lispdesc_indirect_description (obj, desc1->data2);
+	if (sdesc->size)
+	  return sdesc->size;
+	else
+	  {
+	    int count;
+	    Bytecount max_size = -1, size;
+
+	    desc1 = sdesc->description;
+
+	    for (count = 0; desc1[count].type != XD_END; count++)
+	      {
+		size = lispdesc_one_description_line_size (rdata,
+							   &desc1[count],
+							   obj, desc);
+		if (size > max_size)
+		  max_size = size;
+	      }
+	    return max_size;
+	  }
       }
+    case XD_C_STRING:
+      return sizeof (void *);
+    case XD_DOC_STRING:
+      return sizeof (void *);
+    case XD_INT_RESET:
+      return sizeof (int);
+    case XD_BYTECOUNT:
+      return sizeof (Bytecount);
+    case XD_ELEMCOUNT:
+      return sizeof (Elemcount);
+    case XD_HASHCODE:
+      return sizeof (Hashcode);
+    case XD_INT:
+      return sizeof (int);
+    case XD_LONG:
+      return sizeof (long);
+    default:
+      stderr_out ("Unsupported dump type : %d\n", desc1->type);
+      abort ();
     }
 
-  if (mark_last_occured_object)
-    {
-      mark_object(*last_occured_object);
-      mark_last_occured_object = 0;
-    }
+  return 0;
 }
 
 
-/* This function calculates the size of a described struct. */
-static Bytecount
-structure_size (const void *obj, const struct struct_description *sdesc)
+/* Return the size of the memory block (NOT necessarily a structure!) 
+   described by SDESC and pointed to by OBJ.  If SDESC records an
+   explicit size (i.e. non-zero), it is simply returned; otherwise,
+   the size is calculated by the maximum offset and the size of the
+   object at that offset, rounded up to the maximum alignment.  In
+   this case, we may need the object, for example when retrieving an
+   "indirect count" of an inlined array (the count is not constant,
+   but is specified by one of the elements of the memory block). (It
+   is generally not a problem if we return an overly large size -- we
+   will simply end up reserving more space than necessary; but if the
+   size is too small we could be in serious trouble, in particular
+   with nested inlined structures, where there may be alignment
+   padding in the middle of a block. #### In fact there is an (at
+   least theoretical) problem with an overly large size -- we may
+   trigger a protection fault when reading from invalid memory.  We
+   need to handle this -- perhaps in a stupid but dependable way,
+   i.e. by trapping SIGSEGV and SIGBUS.) */
+
+Bytecount
+lispdesc_structure_size (const void *obj,
+			 const struct sized_memory_description *sdesc)
 {
-  int max_offset = -1;
+  EMACS_INT max_offset = -1;
   int max_offset_pos = -1;
-  int size_at_max = 0;
   int pos;
-  const struct lrecord_description *desc;
-  void *rdata;
+  const struct memory_description *desc;
 
   if (sdesc->size)
     return sdesc->size;
@@ -3074,14 +3095,15 @@
 
   for (pos = 0; desc[pos].type != XD_END; pos++)
     {
-      if (desc[pos].offset == max_offset)
+      EMACS_INT offset = lispdesc_indirect_count (desc[pos].offset, desc, obj);
+      if (offset == max_offset)
 	{
 	  stderr_out ("Two relocatable elements at same offset?\n");
 	  abort ();
 	}
-      else if (desc[pos].offset > max_offset)
+      else if (offset > max_offset)
 	{
-	  max_offset = desc[pos].offset;
+	  max_offset = offset;
 	  max_offset_pos = pos;
 	}
     }
@@ -3089,102 +3111,188 @@
   if (max_offset_pos < 0)
     return 0;
 
-  pos = max_offset_pos;
-  rdata = (char *) obj + desc[pos].offset;
-
-  switch (desc[pos].type)
+  {
+    Bytecount size_at_max;
+    size_at_max =
+      lispdesc_one_description_line_size ((char *) obj + max_offset,
+					  &desc[max_offset_pos], obj, desc);
+
+    /* We have no way of knowing the required alignment for this structure,
+       so just make it maximally aligned. */
+    return MAX_ALIGN_SIZE (max_offset + size_at_max);
+  }
+}
+
+#endif /* defined (USE_KKCC) || defined (PDUMP) */
+
+#ifdef USE_KKCC
+/* The following functions implement the new mark algorithm. 
+   They mark objects according to their descriptions.  They 
+   are modeled on the corresponding pdumper procedures. */
+
+static void mark_struct_contents (const void *data,
+				  const struct sized_memory_description *sdesc,
+				  int count);
+
+
+#ifdef ERROR_CHECK_GC
+void
+mark_object_maybe_checking_free (Lisp_Object obj, int allow_free)
+{
+
+  if (!allow_free && XTYPE (obj) == Lisp_Type_Record)
+    {
+      struct lrecord_header *lheader = XRECORD_LHEADER (obj);
+      gc_checking_assert (LHEADER_IMPLEMENTATION (lheader)->basic_p ||
+			  ! ((struct lcrecord_header *) lheader)->free);
+    }
+  mark_object (obj);
+}
+#else
+#define mark_object_maybe_checking_free (obj, allow_free) mark_object (obj)
+#endif /* ERROR_CHECK_GC */
+
+/* This function is called to mark the elements of an object. It processes
+   the description of the object and calls mark object with every described
+   object. */
+static void
+mark_with_description (const void *data,
+		       const struct memory_description *desc)
+{
+  int pos;
+  static const Lisp_Object *last_occurred_object = (Lisp_Object *) 0;
+  static int mark_last_occurred_object = 0;
+#ifdef ERROR_CHECK_GC
+  static int last_occurred_flags;
+#endif
+
+  for (pos = 0; desc[pos].type != XD_END; pos++)
     {
-    case XD_LISP_OBJECT_ARRAY:
-      {
-	EMACS_INT val = desc[pos].data1;
-	if (XD_IS_INDIRECT (val))
-	  val = get_indirect_count (val, desc, obj);
-	size_at_max = val * sizeof (Lisp_Object);
-	break;
-      }
-    case XD_LISP_OBJECT:
-    case XD_LO_LINK:
-      size_at_max = sizeof (Lisp_Object);
-      break;
-    case XD_OPAQUE_PTR:
-      size_at_max = sizeof (void *);
-      break;
-    case XD_STRUCT_PTR:
-      {
-	EMACS_INT val = desc[pos].data1;
-	if (XD_IS_INDIRECT (val))
-	  val = get_indirect_count (val, desc, obj);
-	size_at_max = val * sizeof (void *);
-	break;
-      }
-      break;
-    case XD_STRUCT_ARRAY:
-      {
-	EMACS_INT val = desc[pos].data1;
-
-	if (XD_IS_INDIRECT (val))
-	  val = get_indirect_count (val, desc, obj);
-	    
-	size_at_max = val * structure_size (rdata, desc[pos].data2);
-	break;
-      }
-      break;
-    case XD_OPAQUE_DATA_PTR:
-      size_at_max = sizeof (void *);
-      break;
-    case XD_UNION:
-      abort ();
-      break;
-    case XD_C_STRING:
-      size_at_max = sizeof (void *);
-      break;
-    case XD_DOC_STRING:
-      size_at_max = sizeof (void *);
-      break;
-    case XD_INT_RESET:
-      size_at_max = sizeof (int);
-      break;
-    case XD_BYTECOUNT:
-      size_at_max = sizeof (Bytecount);
-      break;
-    case XD_ELEMCOUNT:
-      size_at_max = sizeof (Elemcount);
-      break;
-    case XD_HASHCODE:
-      size_at_max = sizeof (Hashcode);
-      break;
-    case XD_INT:
-      size_at_max = sizeof (int);
-      break;
-    case XD_LONG:
-      size_at_max = sizeof (long);
-      break;
-    case XD_SPECIFIER_END:
-    case XD_CODING_SYSTEM_END:
-      stderr_out
-	("Should not be seeing XD_SPECIFIER_END or\n"
-	 "XD_CODING_SYSTEM_END outside of struct Lisp_Specifier\n"
-	 "and struct Lisp_Coding_System.\n");
-      abort ();
-    default:
-      stderr_out ("Unsupported dump type : %d\n", desc[pos].type);
-      abort ();
+      const struct memory_description *desc1 = &desc[pos];
+      const void *rdata =
+	(const char *) data + lispdesc_indirect_count (desc1->offset,
+						       desc, data);
+    union_switcheroo:
+
+      /* If the flag says don't mark, then don't mark. */
+      if ((desc1->flags) & XD_FLAG_NO_KKCC)
+	continue;
+
+      switch (desc1->type)
+	{
+	case XD_BYTECOUNT:
+	case XD_ELEMCOUNT:
+	case XD_HASHCODE:
+	case XD_INT:
+	case XD_LONG:
+	case XD_INT_RESET:
+	case XD_LO_LINK:
+	case XD_OPAQUE_PTR:
+	case XD_OPAQUE_DATA_PTR:
+	case XD_C_STRING:
+	case XD_DOC_STRING:
+	  break;
+	case XD_LISP_OBJECT: 
+	  {
+	    const Lisp_Object *stored_obj = (const Lisp_Object *) rdata;
+
+	    /* Because of the way that tagged objects work (pointers and
+	       Lisp_Objects have the same representation), XD_LISP_OBJECT
+	       can be used for untagged pointers.  They might be NULL,
+	       though. */
+	    if (EQ (*stored_obj, Qnull_pointer))
+	      break;
+
+	    if (desc[pos+1].type == XD_END)
+	      {
+		mark_last_occurred_object = 1;
+		last_occurred_object = stored_obj;
+#ifdef ERROR_CHECK_GC
+		last_occurred_flags = desc1->flags;
+#endif
+		break;
+	      }
+	    else
+	      mark_object_maybe_checking_free
+		(*stored_obj, (desc1->flags) & XD_FLAG_FREE_LISP_OBJECT);
+
+	    break;
+	  }
+	case XD_LISP_OBJECT_ARRAY:
+	  {
+	    int i;
+	    EMACS_INT count =
+	      lispdesc_indirect_count (desc1->data1, desc, data);
+	
+	    for (i = 0; i < count; i++)
+	      {
+		const Lisp_Object *stored_obj =
+		  (const Lisp_Object *) rdata + i;
+
+		if (EQ (*stored_obj, Qnull_pointer))
+		  break;
+
+		mark_object_maybe_checking_free
+		  (*stored_obj, (desc1->flags) & XD_FLAG_FREE_LISP_OBJECT);
+	      }
+	    break;
+	  }
+	case XD_STRUCT_PTR:
+	  {
+	    EMACS_INT count = lispdesc_indirect_count (desc1->data1, desc,
+						       data);
+	    const struct sized_memory_description *sdesc =
+	      lispdesc_indirect_description (data, desc1->data2);
+	    const char *dobj = * (const char **) rdata;
+	    if (dobj)
+	      mark_struct_contents (dobj, sdesc, count);
+	    break;
+	  }
+	case XD_STRUCT_ARRAY:
+	  {
+	    EMACS_INT count = lispdesc_indirect_count (desc1->data1, desc,
+						       data);
+	    const struct sized_memory_description *sdesc =
+	      lispdesc_indirect_description (data, desc1->data2);
+		      
+	    mark_struct_contents (rdata, sdesc, count);
+	    break;
+	  }
+	case XD_UNION:
+	case XD_UNION_DYNAMIC_SIZE:
+	  desc1 = lispdesc_process_xd_union (desc1, desc, data);
+	  if (desc1)
+	    goto union_switcheroo;
+	  break;
+		    
+	default:
+	  stderr_out ("Unsupported description type : %d\n", desc1->type);
+	  abort ();
+	}
     }
 
-  return ALIGN_SIZE (max_offset + size_at_max, ALIGNOF (max_align_t));
+  if (mark_last_occurred_object)
+    {
+      /* NOTE: The second parameter isn't even evaluated
+	 non-ERROR_CHECK_GC, so it's OK for the variable not to exist.
+       */
+      mark_object_maybe_checking_free (*last_occurred_object,
+				       last_occurred_flags &
+				       XD_FLAG_FREE_LISP_OBJECT);
+      mark_last_occurred_object = 0;
+    }
 }
 
-
 /* This function loops all elements of a struct pointer and calls 
    mark_with_description with each element. */
 static void
 mark_struct_contents (const void *data,
-				const struct struct_description *sdesc,
-				int count)
+		      const struct sized_memory_description *sdesc,
+		      int count)
 {
   int i;
   Bytecount elsize;
-  elsize = structure_size (data, sdesc);
+  elsize = lispdesc_structure_size (data, sdesc);
 
   for (i = 0; i < count; i++)
     {
@@ -3211,15 +3319,15 @@
   if (XTYPE (obj) == Lisp_Type_Record)
     {
       struct lrecord_header *lheader = XRECORD_LHEADER (obj);
-#ifdef USE_KKCC
-      const struct lrecord_implementation *imp;
-      const struct lrecord_description *desc;
-#endif /* USE_KKCC */      
 
       GC_CHECK_LHEADER_INVARIANTS (lheader);
 
+#ifndef USE_KKCC
+      /* We handle this separately, above, so we can mark free objects */
       gc_checking_assert (LHEADER_IMPLEMENTATION (lheader)->basic_p ||
 			  ! ((struct lcrecord_header *) lheader)->free);
+#endif /* not USE_KKCC */
+
 
       /* All c_readonly objects have their mark bit set,
 	 so that we only need to check the mark bit here. */
@@ -3227,60 +3335,35 @@
 	{
 	  MARK_RECORD_HEADER (lheader);
 
+	  {
 #ifdef USE_KKCC
-	  imp = LHEADER_IMPLEMENTATION (lheader);
-	  desc = imp->description;
-	  
-	  if (desc) /* && !CONSP(obj))*/  /* KKCC cons special case */
-	    {
-	      mark_with_description (lheader, desc);
-	    }
+	    const struct lrecord_implementation *imp;
+	    const struct memory_description *desc;
+
+	    imp = LHEADER_IMPLEMENTATION (lheader);
+	    desc = imp->description;
 	  
-	  else 
-	    {
-
+	    if (desc) /* && !CONSP(obj))*/  /* KKCC cons special case */
+	      {
+		mark_with_description (lheader, desc);
+	      }
+	    else 
 #endif /* USE_KKCC */
-
-
-	      if (RECORD_MARKER (lheader))
-		{
-		  obj = RECORD_MARKER (lheader) (obj);
-		  if (!NILP (obj)) goto tail_recurse;
-		}
-
-#ifdef USE_KKCC
-	    }
-#endif /* USE_KKCC */
+	      {
+		if (RECORD_MARKER (lheader))
+		  {
+		    obj = RECORD_MARKER (lheader) (obj);
+		    if (!NILP (obj)) goto tail_recurse;
+		  }
+	      }
+	  }
 	}
     }
 }
 
-/* mark all of the conses in a list and mark the final cdr; but
-   DO NOT mark the cars.
-
-   Use only for internal lists!  There should never be other pointers
-   to the cons cells, because if so, the cars will remain unmarked
-   even when they maybe should be marked. */
-void
-mark_conses_in_list (Lisp_Object obj)
-{
-  Lisp_Object rest;
-
-  for (rest = obj; CONSP (rest); rest = XCDR (rest))
-    {
-      if (CONS_MARKED_P (XCONS (rest)))
-	return;
-      MARK_CONS (XCONS (rest));
-    }
-
-  mark_object (rest);
-}
-
 
 /* Find all structures not marked, and free them. */
 
-static int gc_count_num_bit_vector_used, gc_count_bit_vector_total_size;
-static int gc_count_bit_vector_storage;
 static int gc_count_num_short_string_in_use;
 static Bytecount gc_count_string_total_size;
 static Bytecount gc_count_short_string_total_size;
@@ -3312,12 +3395,8 @@
     }
   else
     {
-      const struct lrecord_implementation *implementation =
-	LHEADER_IMPLEMENTATION (h);
-
-      Bytecount sz = (implementation->size_in_bytes_method ?
-		   implementation->size_in_bytes_method (h) :
-		   implementation->static_size);
+      Bytecount sz = detagged_lisp_object_size (h);
+
       if (free_p)
 	{
 	  lcrecord_stats[type_index].instances_freed++;
@@ -3393,49 +3472,6 @@
   /* *total = total_size; */
 }
 
-
-static void
-sweep_bit_vectors_1 (Lisp_Object *prev,
-		     int *used, int *total, int *storage)
-{
-  Lisp_Object bit_vector;
-  int num_used = 0;
-  int total_size = 0;
-  int total_storage = 0;
-
-  /* BIT_VECTORP fails because the objects are marked, which changes
-     their implementation */
-  for (bit_vector = *prev; !EQ (bit_vector, Qzero); )
-    {
-      Lisp_Bit_Vector *v = XBIT_VECTOR (bit_vector);
-      int len = v->size;
-      if (MARKED_RECORD_P (bit_vector))
-	{
-	  if (! C_READONLY_RECORD_HEADER_P(&(v->lheader)))
-	    UNMARK_RECORD_HEADER (&(v->lheader));
-	  total_size += len;
-          total_storage +=
-	    MALLOC_OVERHEAD +
-	    FLEXIBLE_ARRAY_STRUCT_SIZEOF (Lisp_Bit_Vector, unsigned long,
-					  bits, BIT_VECTOR_LONG_STORAGE (len));
-	  num_used++;
-	  /* #### May modify next on a C_READONLY bitvector */
-	  prev = &(bit_vector_next (v));
-	  bit_vector = *prev;
-	}
-      else
-	{
-          Lisp_Object next = bit_vector_next (v);
-          *prev = next;
-	  xfree (v);
-	  bit_vector = next;
-	}
-    }
-  *used = num_used;
-  *total = total_size;
-  *storage = total_storage;
-}
-
 /* And the Lord said: Thou shalt use the `c-backslash-region' command
    to make macros prettier. */
 
@@ -3594,14 +3630,16 @@
      placed on the free list, however, its car will probably contain
      a chain pointer to the next cons on the list, which has cleverly
      had all its 0's and 1's inverted.  This allows for a quick
-     check to make sure we're not freeing something already freed. */
+     check to make sure we're not freeing something already freed.
+
+     NOTE: This check may not be necessary.  Freeing an object sets its
+     type to lrecord_type_free, which will trip up the XCONS() above -- as
+     well as a check in FREE_FIXED_TYPE(). */
   if (POINTER_TYPE_P (XTYPE (cons_car (ptr))))
     ASSERT_VALID_POINTER (XPNTR (cons_car (ptr)));
 #endif /* ERROR_CHECK_GC */
 
-#ifndef ALLOC_NO_POOLS
   FREE_FIXED_TYPE_WHEN_NOT_IN_GC (cons, Lisp_Cons, ptr);
-#endif /* ALLOC_NO_POOLS */
 }
 
 /* explicitly free a list.  You **must make sure** that you have
@@ -3685,7 +3723,7 @@
   SWEEP_FIXED_TYPE_BLOCK (event, Lisp_Event);
 }
 
-#ifdef USE_KKCC
+#ifdef EVENT_DATA_AS_OBJECTS
 
 static void
 sweep_key_data (void)
@@ -3696,6 +3734,12 @@
   SWEEP_FIXED_TYPE_BLOCK (key_data, Lisp_Key_Data);
 }
 
+void
+free_key_data (Lisp_Object ptr)
+{
+  FREE_FIXED_TYPE_WHEN_NOT_IN_GC (key_data, Lisp_Key_Data, XKEY_DATA (ptr));
+}
+
 static void
 sweep_button_data (void)
 {
@@ -3705,6 +3749,12 @@
   SWEEP_FIXED_TYPE_BLOCK (button_data, Lisp_Button_Data);
 }
 
+void
+free_button_data (Lisp_Object ptr)
+{
+  FREE_FIXED_TYPE_WHEN_NOT_IN_GC (button_data, Lisp_Button_Data, XBUTTON_DATA (ptr));
+}
+
 static void
 sweep_motion_data (void)
 {
@@ -3714,6 +3764,12 @@
   SWEEP_FIXED_TYPE_BLOCK (motion_data, Lisp_Motion_Data);
 }
 
+void
+free_motion_data (Lisp_Object ptr)
+{
+  FREE_FIXED_TYPE_WHEN_NOT_IN_GC (motion_data, Lisp_Motion_Data, XMOTION_DATA (ptr));
+}
+
 static void
 sweep_process_data (void)
 {
@@ -3723,6 +3779,12 @@
   SWEEP_FIXED_TYPE_BLOCK (process_data, Lisp_Process_Data);
 }
 
+void
+free_process_data (Lisp_Object ptr)
+{
+  FREE_FIXED_TYPE_WHEN_NOT_IN_GC (process_data, Lisp_Process_Data, XPROCESS_DATA (ptr));
+}
+
 static void
 sweep_timeout_data (void)
 {
@@ -3732,6 +3794,12 @@
   SWEEP_FIXED_TYPE_BLOCK (timeout_data, Lisp_Timeout_Data);
 }
 
+void
+free_timeout_data (Lisp_Object ptr)
+{
+  FREE_FIXED_TYPE_WHEN_NOT_IN_GC (timeout_data, Lisp_Timeout_Data, XTIMEOUT_DATA (ptr));
+}
+
 static void
 sweep_magic_data (void)
 {
@@ -3741,6 +3809,12 @@
   SWEEP_FIXED_TYPE_BLOCK (magic_data, Lisp_Magic_Data);
 }
 
+void
+free_magic_data (Lisp_Object ptr)
+{
+  FREE_FIXED_TYPE_WHEN_NOT_IN_GC (magic_data, Lisp_Magic_Data, XMAGIC_DATA (ptr));
+}
+
 static void
 sweep_magic_eval_data (void)
 {
@@ -3750,6 +3824,12 @@
   SWEEP_FIXED_TYPE_BLOCK (magic_eval_data, Lisp_Magic_Eval_Data);
 }
 
+void
+free_magic_eval_data (Lisp_Object ptr)
+{
+  FREE_FIXED_TYPE_WHEN_NOT_IN_GC (magic_eval_data, Lisp_Magic_Eval_Data, XMAGIC_EVAL_DATA (ptr));
+}
+
 static void
 sweep_eval_data (void)
 {
@@ -3759,6 +3839,12 @@
   SWEEP_FIXED_TYPE_BLOCK (eval_data, Lisp_Eval_Data);
 }
 
+void
+free_eval_data (Lisp_Object ptr)
+{
+  FREE_FIXED_TYPE_WHEN_NOT_IN_GC (eval_data, Lisp_Eval_Data, XEVAL_DATA (ptr));
+}
+
 static void
 sweep_misc_user_data (void)
 {
@@ -3768,7 +3854,13 @@
   SWEEP_FIXED_TYPE_BLOCK (misc_user_data, Lisp_Misc_User_Data);
 }
 
-#endif /* USE_KKCC */
+void
+free_misc_user_data (Lisp_Object ptr)
+{
+  FREE_FIXED_TYPE_WHEN_NOT_IN_GC (misc_user_data, Lisp_Misc_User_Data, XMISC_USER_DATA (ptr));
+}
+
+#endif /* EVENT_DATA_AS_OBJECTS */
 
 static void
 sweep_markers (void)
@@ -3785,14 +3877,9 @@
 
 /* Explicitly free a marker.  */
 void
-free_marker (Lisp_Marker *ptr)
+free_marker (Lisp_Object ptr)
 {
-  /* Perhaps this will catch freeing an already-freed marker. */
-  gc_checking_assert (ptr->lheader.type == lrecord_type_marker);
-
-#ifndef ALLOC_NO_POOLS
-  FREE_FIXED_TYPE_WHEN_NOT_IN_GC (marker, Lisp_Marker, ptr);
-#endif /* ALLOC_NO_POOLS */
+  FREE_FIXED_TYPE_WHEN_NOT_IN_GC (marker, Lisp_Marker, XMARKER (ptr));
 }
 
 
@@ -3842,7 +3929,7 @@
     }
 }
 
-#endif /* MULE && ERROR_CHECK_GC */
+#endif /* defined (MULE) && defined (VERIFY_STRING_CHARS_INTEGRITY) */
 
 /* Compactify string chars, relocating the reference to each --
    free any empty string_chars_block we see. */
@@ -3884,7 +3971,7 @@
             }
 
           string = from_s_chars->string;
-	  assert (!(LRECORD_FREE_P (string)));
+	  gc_checking_assert (!(LRECORD_FREE_P (string)));
 
           size = string->size_;
           fullsize = STRING_FULLSIZE (size);
@@ -4054,12 +4141,6 @@
   /* Put all unmarked conses on free list */
   sweep_conses ();
 
-  /* Free all unmarked bit vectors */
-  sweep_bit_vectors_1 (&all_bit_vectors,
-		       &gc_count_num_bit_vector_used,
-		       &gc_count_bit_vector_total_size,
-		       &gc_count_bit_vector_storage);
-
   /* Free all unmarked compiled-function objects */
   sweep_compiled_functions ();
 
@@ -4078,7 +4159,7 @@
 
   sweep_events ();
 
-#ifdef USE_KKCC
+#ifdef EVENT_DATA_AS_OBJECTS
   sweep_key_data ();
   sweep_button_data ();
   sweep_motion_data ();
@@ -4088,7 +4169,7 @@
   sweep_magic_eval_data ();
   sweep_eval_data ();
   sweep_misc_user_data ();
-#endif /* USE_KKCC */
+#endif /* EVENT_DATA_AS_OBJECTS */
 
 #ifdef PDUMP
   pdump_objects_unmark ();
@@ -4356,6 +4437,7 @@
   /* #### generalize this? */
   clear_event_resource ();
   cleanup_specifiers ();
+  cleanup_buffer_undo_lists ();
 
   /* Mark all the special slots that serve as the roots of accessibility. */
 
@@ -4607,11 +4689,6 @@
   pl = gc_plist_hack ("compiled-functions-used",
 		      gc_count_num_compiled_function_in_use, pl);
 
-  pl = gc_plist_hack ("bit-vector-storage", gc_count_bit_vector_storage, pl);
-  pl = gc_plist_hack ("bit-vectors-total-length",
-                      gc_count_bit_vector_total_size, pl);
-  pl = gc_plist_hack ("bit-vectors-used", gc_count_num_bit_vector_used, pl);
-
   HACK_O_MATIC (symbol, "symbol-storage", pl);
   pl = gc_plist_hack ("symbols-free", gc_count_num_symbol_freelist, pl);
   pl = gc_plist_hack ("symbols-used", gc_count_num_symbol_in_use, pl);
@@ -4845,7 +4922,7 @@
 
 /* Initialization */
 static void
-common_init_alloc_once_early (void)
+common_init_alloc_early (void)
 {
 #ifndef Qzero
   Qzero = make_int (0);	/* Only used if Lisp_Object is a union type */
@@ -4859,7 +4936,6 @@
 
   gc_generation_number[0] = 0;
   breathing_space = 0;
-  all_bit_vectors = Qzero;
   Vgc_message = Qzero;
   all_lcrecords = 0;
   ignore_malloc_warnings = 1;
@@ -4879,7 +4955,7 @@
   init_marker_alloc ();
   init_extent_alloc ();
   init_event_alloc ();
-#ifdef USE_KKCC
+#ifdef EVENT_DATA_AS_OBJECTS
   init_key_data_alloc ();
   init_button_data_alloc ();
   init_motion_data_alloc ();
@@ -4889,7 +4965,7 @@
   init_magic_eval_data_alloc ();
   init_eval_data_alloc ();
   init_misc_user_data_alloc ();
-#endif /* USE_KKCC */
+#endif /* EVENT_DATA_AS_OBJECTS */
 
   ignore_malloc_warnings = 0;
 
@@ -4920,7 +4996,6 @@
 			     systems */
   lrecord_uid_counter = 259;
   debug_string_purity = 0;
-  gcprolist = 0;
 
   gc_currently_forbidden = 0;
   gc_hooks_inhibited = 0;
@@ -4952,16 +5027,31 @@
 }
 
 void
-reinit_alloc_once_early (void)
+init_alloc_early (void)
 {
-  common_init_alloc_once_early ();
+#if defined (__cplusplus) && defined (ERROR_CHECK_GC)
+  static struct gcpro initial_gcpro;
+
+  initial_gcpro.next = 0;
+  initial_gcpro.var = &Qnil;
+  initial_gcpro.nvars = 1;
+  gcprolist = &initial_gcpro;
+#else
+  gcprolist = 0;
+#endif /* defined (__cplusplus) && defined (ERROR_CHECK_GC) */
+}
+
+void
+reinit_alloc_early (void)
+{
+  common_init_alloc_early ();
   init_lcrecord_lists ();
 }
 
 void
 init_alloc_once_early (void)
 {
-  common_init_alloc_once_early ();
+  common_init_alloc_early ();
 
   {
     int i;
@@ -4973,6 +5063,7 @@
   INIT_LRECORD_IMPLEMENTATION (vector);
   INIT_LRECORD_IMPLEMENTATION (string);
   INIT_LRECORD_IMPLEMENTATION (lcrecord_list);
+  INIT_LRECORD_IMPLEMENTATION (free);
 
   staticpros = Dynarr_new2 (Lisp_Object_ptr_dynarr, Lisp_Object *);
   Dynarr_resize (staticpros, 1410); /* merely a small optimization */
@@ -4987,12 +5078,6 @@
 }
 
 void
-init_alloc_early (void)
-{
-  gcprolist = 0;
-}
-
-void
 syms_of_alloc (void)
 {
   DEFSYMBOL (Qpre_gc_hook);
--- a/src/buffer.c	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/buffer.c	Sun Jan 12 11:08:22 2003 +0000
@@ -228,17 +228,52 @@
 static void reset_buffer_local_variables (struct buffer *, int first_time);
 static void nuke_all_buffer_slots (struct buffer *b, Lisp_Object zap);
 
+static const struct memory_description buffer_text_description_1 [] = {
+  { XD_LISP_OBJECT, offsetof (struct buffer_text, line_number_cache) },
+  { XD_END }
+};
+
+static const struct sized_memory_description buffer_text_description = {
+  sizeof (struct buffer_text),
+  buffer_text_description_1
+};
+
+static const struct memory_description syntax_cache_description_1 [] = {
+  { XD_LISP_OBJECT, offsetof (struct syntax_cache, object) },
+  { XD_LISP_OBJECT, offsetof (struct syntax_cache, buffer) },
+  { XD_LISP_OBJECT, offsetof (struct syntax_cache, current_syntax_table) },
+  { XD_LISP_OBJECT, offsetof (struct syntax_cache, start) },
+  { XD_LISP_OBJECT, offsetof (struct syntax_cache, end) },
+  { XD_END }
+};
+
+static const struct sized_memory_description syntax_cache_description = {
+  sizeof (struct syntax_cache),
+  syntax_cache_description_1
+};
+
+static const struct memory_description buffer_description [] = {
+#define MARKED_SLOT(x) { XD_LISP_OBJECT, offsetof (struct buffer, x) },
+#include "bufslots.h"
+
+  { XD_LISP_OBJECT, offsetof (struct buffer, extent_info) },
+
+  { XD_STRUCT_PTR, offsetof (struct buffer, text),
+    1, &buffer_text_description },
+  { XD_STRUCT_PTR, offsetof (struct buffer, syntax_cache),
+    1, &syntax_cache_description },
+
+  { XD_LISP_OBJECT, offsetof (struct buffer, indirect_children) },
+  { XD_LISP_OBJECT, offsetof (struct buffer, base_buffer) },
+  { XD_END }
+};
+
 static Lisp_Object
 mark_buffer (Lisp_Object obj)
 {
   struct buffer *buf = XBUFFER (obj);
 
-  /* Truncate undo information. */
-  buf->undo_list = truncate_undo_list (buf->undo_list,
-                                       undo_threshold,
-                                       undo_high_threshold);
-
-#define MARKED_SLOT(x) mark_object (buf->x)
+#define MARKED_SLOT(x) mark_object (buf->x);
 #include "bufslots.h"
 
   mark_object (buf->extent_info);
@@ -246,10 +281,17 @@
     mark_object (buf->text->line_number_cache);
   mark_buffer_syntax_cache (buf);
 
-  /* Don't mark normally through the children slot.
-     (Actually, in this case, it doesn't matter.)  */
+  /* [[ Don't mark normally through the children slot.  Actually, in this
+     case, it doesn't matter. ]]
+
+     Indirect buffers, like all buffers, are permanent objects and stay
+     around by themselves, so it doesn't matter whether we mark their
+     children.  This used to contain a call to mark_conses_in_list(), to
+     mark only the conses.  I deleted that function, since it's not used
+     any more and causes problems with KKCC.  If we really needed such a
+     weak list, just use a weak list object, like extents do. --ben */
   if (! EQ (buf->indirect_children, Qnull_pointer))
-    mark_conses_in_list (buf->indirect_children);
+    mark_object (buf->indirect_children);
 
   return buf->base_buffer ? wrap_buffer (buf->base_buffer) : Qnil;
 }
@@ -274,19 +316,28 @@
     print_internal (b->name, printcharfun, 0);
 }
 
+void
+cleanup_buffer_undo_lists (void)
+{
+  /* Truncate undo information at GC time.  Used to be in mark_object() but
+     moved here for KKCC purposes. */
+
+  ALIST_LOOP_3 (name, buf, Vbuffer_alist)
+    {
+      XBUFFER (buf)->undo_list = truncate_undo_list (XBUFFER (buf)->undo_list,
+						     undo_threshold,
+						     undo_high_threshold);
+    }
+}
+
 /* We do not need a finalize method to handle a buffer's children list
    because all buffers have `kill-buffer' applied to them before
    they disappear, and the children removal happens then. */
-#ifdef USE_KKCC
 DEFINE_LRECORD_IMPLEMENTATION ("buffer", buffer,
 			       0, /*dumpable-flag*/
-                               mark_buffer, print_buffer, 0, 0, 0, 0,
+                               mark_buffer, print_buffer, 0, 0, 0,
+			       buffer_description,
 			       struct buffer);
-#else /* not USE_KKCC */
-DEFINE_LRECORD_IMPLEMENTATION ("buffer", buffer,
-                               mark_buffer, print_buffer, 0, 0, 0, 0,
-			       struct buffer);
-#endif /* not USE_KKCC */
 
 DEFUN ("bufferp", Fbufferp, 1, 1, 0, /*
 Return t if OBJECT is an editor buffer.
@@ -441,7 +492,6 @@
        (filename))
 {
   /* This function can GC.  GC checked and fixed 7-11-2000 ben. */
-  REGISTER Lisp_Object buf;
   struct gcpro gcpro1;
 
 #ifdef I18N3
@@ -486,10 +536,8 @@
     }
 
   {
-    LIST_LOOP_2 (elt, Vbuffer_alist)
+    ALIST_LOOP_3 (name, buf, Vbuffer_alist)
       {
-	buf = Fcdr (elt);
-	if (!BUFFERP (buf)) continue;
 	if (!STRINGP (XBUFFER (buf)->filename)) continue;
 	if (!NILP (Fstring_equal (filename,
 				  (find_file_compare_truenames
@@ -2106,7 +2154,7 @@
   b->indirect_children = Qnil;
   b->own_text.line_number_cache = Qnil;
 
-#define MARKED_SLOT(x)	b->x = zap
+#define MARKED_SLOT(x)	b->x = zap;
 #include "bufslots.h"
 }
 
@@ -2252,12 +2300,12 @@
 }
 
 
-static const struct lrecord_description buffer_slots_description_1[] = {
+static const struct memory_description buffer_slots_description_1[] = {
   { XD_LISP_OBJECT_ARRAY, 0, BUFFER_SLOTS_COUNT },
   { XD_END }
 };
 
-static const struct struct_description buffer_slots_description = {
+static const struct sized_memory_description buffer_slots_description = {
   BUFFER_SLOTS_SIZE,
   buffer_slots_description_1
 };
@@ -2754,7 +2802,7 @@
   if ((XINT (buffer_local_flags.slot) != -2 &&			\
        XINT (buffer_local_flags.slot) != -3)			\
       != !(NILP (XBUFFER (Vbuffer_local_symbols)->slot)))	\
-  abort ()
+  abort ();
 #include "bufslots.h"
 
   {
--- a/src/buffer.h	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/buffer.h	Sun Jan 12 11:08:22 2003 +0000
@@ -235,7 +235,7 @@
 
   /* Everything from here down must be a Lisp_Object */
 
-#define MARKED_SLOT(x) Lisp_Object x
+#define MARKED_SLOT(x) Lisp_Object x;
 #include "bufslots.h"
 #undef MARKED_SLOT
 };
--- a/src/bufslots.h	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/bufslots.h	Sun Jan 12 11:08:22 2003 +0000
@@ -1,6 +1,6 @@
 /* Definitions of marked slots in buffers
    Copyright (C) 1990, 1992, 1993 Free Software Foundation, Inc.
-   Copyright (C) 2001 Ben Wing.
+   Copyright (C) 2001, 2002 Ben Wing.
 
 This file is part of XEmacs.
 
@@ -41,36 +41,36 @@
 #endif
 
     /* The name of this buffer.  */
-    MARKED_SLOT (name);
+    MARKED_SLOT (name)
 
     /* The name of the file visited in this buffer, or nil.  */
-    MARKED_SLOT (filename);
+    MARKED_SLOT (filename)
 
     /* Dir for expanding relative file names.  */
-    MARKED_SLOT (directory);
+    MARKED_SLOT (directory)
 
     /* True iff this buffer has been backed up (if you write to the
        visited file and it hasn't been backed up, then a backup will
        be made).  */
     /* #### This isn't really used by the C code, so could be deleted.  */
-    MARKED_SLOT (backed_up);
+    MARKED_SLOT (backed_up)
 
     /* Length of file when last read or saved.
        This is not in the  struct buffer_text
        because it's not used in indirect buffers at all.  */
-    MARKED_SLOT (saved_size);
+    MARKED_SLOT (saved_size)
 
     /* File name used for auto-saving this buffer.
        This is not in the  struct buffer_text
        because it's not used in indirect buffers at all.  */
-    MARKED_SLOT (auto_save_file_name);
+    MARKED_SLOT (auto_save_file_name)
 
     /* Non-nil if buffer read-only.  */
-    MARKED_SLOT (read_only);
+    MARKED_SLOT (read_only)
 
     /* "The mark".  This is a marker which may
        point into this buffer or may point nowhere.  */
-    MARKED_SLOT (mark);
+    MARKED_SLOT (mark)
 
     /* Alist of elements (SYMBOL . VALUE-IN-THIS-BUFFER)
        for all per-buffer variables of this buffer.
@@ -84,48 +84,48 @@
 
        Variables declared in C with DEFVAR_BUFFER_LOCAL() (i.e.
        those stored in the struct buffer) are not listed here. */
-    MARKED_SLOT (local_var_alist);
+    MARKED_SLOT (local_var_alist)
 
     /* Symbol naming major mode (eg, lisp-mode).  */
-    MARKED_SLOT (major_mode);
+    MARKED_SLOT (major_mode)
 
     /* Pretty name of major mode (eg, "Lisp"). */
-    MARKED_SLOT (mode_name);
+    MARKED_SLOT (mode_name)
 
     /* Modeline element that controls format of modeline.  */
-    MARKED_SLOT (modeline_format);
+    MARKED_SLOT (modeline_format)
 
     /* Keys that are bound local to this buffer.  */
-    MARKED_SLOT (keymap);
+    MARKED_SLOT (keymap)
 
     /* This buffer's local abbrev table.  */
-    MARKED_SLOT (abbrev_table);
+    MARKED_SLOT (abbrev_table)
     /* This buffer's syntax table.  */
-    MARKED_SLOT (syntax_table);
+    MARKED_SLOT (syntax_table)
     /* Massaged values from the syntax table, for faster lookup. */
-    MARKED_SLOT (mirror_syntax_table);
+    MARKED_SLOT (mirror_syntax_table)
 
 #ifdef MULE
     /* This buffer's category table. */
-    MARKED_SLOT (category_table);
+    MARKED_SLOT (category_table)
 #endif /* MULE */
     /* This buffer's coding system. */
-    MARKED_SLOT (buffer_file_coding_system);
+    MARKED_SLOT (buffer_file_coding_system)
     /* Values of several buffer-local variables.
 
        tab-width is buffer-local so that redisplay can find it
        in buffers that are not current */
-    MARKED_SLOT (case_fold_search);
-    MARKED_SLOT (tab_width);
-    MARKED_SLOT (fill_column);
-    MARKED_SLOT (left_margin);
+    MARKED_SLOT (case_fold_search)
+    MARKED_SLOT (tab_width)
+    MARKED_SLOT (fill_column)
+    MARKED_SLOT (left_margin)
 
     /* Function to call when insert space past fill column.  */
-    MARKED_SLOT (auto_fill_function);
+    MARKED_SLOT (auto_fill_function)
 
     /* Case table for case-conversion in this buffer. */
-    MARKED_SLOT (case_table);
-    /* It contais following char-tables: */
+    MARKED_SLOT (case_table)
+    /* It contanis following char-tables: */
     /* Char-table maps each char into its lower-case version.  */
     /* Char-table mapping each char to its upper-case version.  */
     /* Char-table for conversion for case-folding search.  */
@@ -133,31 +133,31 @@
 
     /* #### This ought to be a specifier: */
     /* Non-nil means do not display continuation lines.  */
-    MARKED_SLOT (truncate_lines);
+    MARKED_SLOT (truncate_lines)
     /* #### This ought to be a specifier: */
     /* #### Better yet, it ought to be junked.  It really sucks. */
     /* Non-nil means display ctl chars with uparrow.  */
-    MARKED_SLOT (ctl_arrow);
+    MARKED_SLOT (ctl_arrow)
     /* #### This ought to be a specifier: */
     /* #### Better yet, it ought to be junked.  It really sucks. */
     /* Non-nil means do selective display;
        see doc string in syms_of_buffer (buffer.c) for details.  */
-    MARKED_SLOT (selective_display);
+    MARKED_SLOT (selective_display)
     /* #### This ought to be a specifier: */
     /* #### Better yet, it ought to be junked.  It really sucks. */
     /* Non-nil means show ... at end of line followed by invisible lines.  */
-    MARKED_SLOT (selective_display_ellipses);
+    MARKED_SLOT (selective_display_ellipses)
     /* Alist of (FUNCTION . STRING) for each minor mode enabled in buffer.  */
-    /* Unused: MARKED_SLOT (minor_modes); */
+    /* Unused: MARKED_SLOT (minor_modes) */
     /* t if "self-insertion" should overwrite */
-    MARKED_SLOT (overwrite_mode);
+    MARKED_SLOT (overwrite_mode)
     /* non-nil means abbrev mode is on.  Expand abbrevs automatically.  */
-    MARKED_SLOT (abbrev_mode);
+    MARKED_SLOT (abbrev_mode)
 
     /* No display table here.  It's a specifier. */
 #if 0 /* FSFmacs */
     /* t means the mark and region are currently active.  */
-    MARKED_SLOT (mark_active);
+    MARKED_SLOT (mark_active)
 #endif
 
     /* Changes in the buffer are recorded here for undo.
@@ -166,7 +166,7 @@
        But we can't store it in the  struct buffer_text
        because local variables have to be right in the  struct buffer.
        So we copy it around in set_buffer_internal.  */
-    MARKED_SLOT (undo_list);
+    MARKED_SLOT (undo_list)
 
     /* FSFmacs has overlay stuff here.  We have extent info elsewhere in the
        struct buffer.  */
@@ -174,12 +174,12 @@
     /* dedicated_frame in lisp */
 
     /* Lisp of symbols naming the file format used for visited file. */
-    MARKED_SLOT (file_format);
+    MARKED_SLOT (file_format)
 
 #ifdef REGION_CACHE_NEEDS_WORK
     /* True if the newline position cache and width run cache are
        enabled.  See search.c and indent.c.  */
-    MARKED_SLOT (cache_long_line_scans);
+    MARKED_SLOT (cache_long_line_scans)
 
     /* If the width run cache is enabled, this table contains the
        character widths width_run_cache (see above) assumes.  When we
@@ -187,13 +187,13 @@
        current display table to see whether the display table has
        affected the widths of any characters.  If it has, we
        invalidate the width run cache, and re-initialize width_table.  */
-    MARKED_SLOT (width_table);
+    MARKED_SLOT (width_table)
 #endif /* REGION_CACHE_NEEDS_WORK */
 
     /* A redundant copy of text.pt, in the form of a marker.  Every time one
        is updated, so is the other.
      */
-    MARKED_SLOT (point_marker);
+    MARKED_SLOT (point_marker)
 
     /* FSFmacs has pt_marker, begv_marker, zv_marker here, used for
        indirect buffers.  We don't need them because we handle these
@@ -201,24 +201,24 @@
 
     /* This holds the point value before the last scroll operation.
        Explicitly setting point sets this to nil.  */
-    MARKED_SLOT (point_before_scroll);
+    MARKED_SLOT (point_before_scroll)
 
     /* Truename of the visited file (via qxe_realpath()),
        or nil.  */
-    MARKED_SLOT (file_truename);
+    MARKED_SLOT (file_truename)
 
     /* Invisibility spec of this buffer.
        t => any non-nil `invisible' property means invisible.
        A list => `invisible' property means invisible
                  if it is memq in that list.  */
-    MARKED_SLOT (invisibility_spec);
+    MARKED_SLOT (invisibility_spec)
 
     /* The string generated by formatting the modeline in this buffer. */
-    MARKED_SLOT (generated_modeline_string);
+    MARKED_SLOT (generated_modeline_string)
 
     /* A hash table that maps from a "generic extent" (an extent in
        `modeline-format') into a buffer-specific extent. */
-    MARKED_SLOT (modeline_extent_table);
+    MARKED_SLOT (modeline_extent_table)
 
 #ifndef BUFFER_SLOTS_LAST_NAME
 #define BUFFER_SLOTS_LAST_NAME modeline_extent_table
--- a/src/bytecode.c	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/bytecode.c	Sun Jan 12 11:08:22 2003 +0000
@@ -1869,20 +1869,10 @@
 		internal_hash (f->constants,    depth + 1));
 }
 
-static const struct lrecord_description lo_description_1[] = {
-  { XD_LISP_OBJECT, 0 },
-  { XD_END }
-};
-
-static const struct struct_description lo_description = {
-  sizeof (Lisp_Object),
-  lo_description_1
-};
-
-static const struct lrecord_description compiled_function_description[] = {
+static const struct memory_description compiled_function_description[] = {
   { XD_INT,         offsetof (Lisp_Compiled_Function, args_in_array) },
   { XD_STRUCT_PTR,  offsetof (Lisp_Compiled_Function, args),
-      XD_INDIRECT (0, 0), &lo_description },
+      XD_INDIRECT (0, 0), &lisp_object_description },
   { XD_LISP_OBJECT, offsetof (Lisp_Compiled_Function, instructions) },
   { XD_LISP_OBJECT, offsetof (Lisp_Compiled_Function, constants) },
   { XD_LISP_OBJECT, offsetof (Lisp_Compiled_Function, arglist) },
@@ -1893,7 +1883,6 @@
   { XD_END }
 };
 
-#ifdef USE_KKCC
 DEFINE_BASIC_LRECORD_IMPLEMENTATION ("compiled-function", compiled_function,
 				     1, /*dumpable_flag*/
 				     mark_compiled_function,
@@ -1902,15 +1891,6 @@
 				     compiled_function_hash,
 				     compiled_function_description,
 				     Lisp_Compiled_Function);
-#else /* not USE_KKCC */
-DEFINE_BASIC_LRECORD_IMPLEMENTATION ("compiled-function", compiled_function,
-				     mark_compiled_function,
-				     print_compiled_function, 0,
-				     compiled_function_equal,
-				     compiled_function_hash,
-				     compiled_function_description,
-				     Lisp_Compiled_Function);
-#endif /* not USE_KKCC */
 
 DEFUN ("compiled-function-p", Fcompiled_function_p, 1, 1, 0, /*
 Return t if OBJECT is a byte-compiled function object.
--- a/src/callint.c	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/callint.c	Sun Jan 12 11:08:22 2003 +0000
@@ -37,7 +37,7 @@
 #include "insdel.h"
 #include "window-impl.h" /* WINDOW_MINI_P */
 
-extern int num_input_chars;
+extern Charcount num_input_chars;
 
 Lisp_Object Vcurrent_prefix_arg;
 Lisp_Object Qcall_interactively;
--- a/src/casetab.c	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/casetab.c	Sun Jan 12 11:08:22 2003 +0000
@@ -97,7 +97,7 @@
   write_fmt_string (printcharfun, "0x%x>", ct->header.uid);
 }
 
-static const struct lrecord_description case_table_description [] = {
+static const struct memory_description case_table_description [] = {
   { XD_LISP_OBJECT, offsetof (Lisp_Case_Table, downcase_table) },
   { XD_LISP_OBJECT, offsetof (Lisp_Case_Table, upcase_table) },
   { XD_LISP_OBJECT, offsetof (Lisp_Case_Table, case_canon_table) },
@@ -106,16 +106,10 @@
 };
 
 
-#ifdef USE_KKCC
 DEFINE_LRECORD_IMPLEMENTATION("case-table", case_table,
 			      1, /*dumpable-flag*/
 			      mark_case_table, print_case_table, 0,
 			      0, 0, case_table_description, Lisp_Case_Table);
-#else /* not USE_KKCC */
-DEFINE_LRECORD_IMPLEMENTATION ("case-table", case_table,
-			      mark_case_table, print_case_table, 0,
-			      0, 0, case_table_description, Lisp_Case_Table);
-#endif /* not USE_KKCC */
 
 static Lisp_Object
 allocate_case_table (int init_tables)
--- a/src/chartab.c	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/chartab.c	Sun Jan 12 11:08:22 2003 +0000
@@ -133,12 +133,11 @@
   return internal_array_hash (cte->level2, 96, depth + 1);
 }
 
-static const struct lrecord_description char_table_entry_description[] = {
+static const struct memory_description char_table_entry_description[] = {
   { XD_LISP_OBJECT_ARRAY, offsetof (Lisp_Char_Table_Entry, level2), 96 },
   { XD_END }
 };
 
-#ifdef USE_KKCC
 DEFINE_LRECORD_IMPLEMENTATION ("char-table-entry", char_table_entry,
 			       1, /* dumpable flag */
                                mark_char_table_entry, internal_object_printer,
@@ -146,14 +145,6 @@
 			       char_table_entry_hash,
 			       char_table_entry_description,
 			       Lisp_Char_Table_Entry);
-#else /* not USE_KKCC */
-DEFINE_LRECORD_IMPLEMENTATION ("char-table-entry", char_table_entry,
-                               mark_char_table_entry, internal_object_printer,
-			       0, char_table_entry_equal,
-			       char_table_entry_hash,
-			       char_table_entry_description,
-			       Lisp_Char_Table_Entry);
-#endif /* not USE_KKCC */
 
 #endif /* MULE */
 
@@ -229,7 +220,7 @@
 #endif
 
   invalid_constant ("Unrecognized char table type", symbol);
-  RETURN_NOT_REACHED (CHAR_TABLE_TYPE_GENERIC)
+  RETURN_NOT_REACHED (CHAR_TABLE_TYPE_GENERIC);
 }
 
 static void
@@ -388,7 +379,7 @@
   return HASH2 (hashval, internal_hash (ct->default_, depth + 1));
 }
 
-static const struct lrecord_description char_table_description[] = {
+static const struct memory_description char_table_description[] = {
   { XD_LISP_OBJECT_ARRAY, offsetof (Lisp_Char_Table, ascii), NUM_ASCII_CHARS },
 #ifdef MULE
   { XD_LISP_OBJECT_ARRAY, offsetof (Lisp_Char_Table, level1), NUM_LEADING_BYTES },
@@ -400,20 +391,12 @@
   { XD_END }
 };
 
-#ifdef USE_KKCC
 DEFINE_LRECORD_IMPLEMENTATION ("char-table", char_table,
 			       1, /*dumpable-flag*/
                                mark_char_table, print_char_table, 0,
 			       char_table_equal, char_table_hash,
 			       char_table_description,
 			       Lisp_Char_Table);
-#else /* not USE_KKCC */
-DEFINE_LRECORD_IMPLEMENTATION ("char-table", char_table,
-                               mark_char_table, print_char_table, 0,
-			       char_table_equal, char_table_hash,
-			       char_table_description,
-			       Lisp_Char_Table);
-#endif /* not USE_KKCC */
 
 DEFUN ("char-table-p", Fchar_table_p, 1, 1, 0, /*
 Return non-nil if OBJECT is a char table.
--- a/src/cmdloop.c	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/cmdloop.c	Sun Jan 12 11:08:22 2003 +0000
@@ -583,17 +583,14 @@
 
       if (!was_locked)
 	any_console_state ();
-#if (defined (_MSC_VER) 			\
-     || defined (__SUNPRO_C)			\
-     || defined (__SUNPRO_CC)			\
-     || (defined (DEC_ALPHA)			\
-	 && defined (OSF1)))
-      if (0) return Qnil; /* Shut up compiler */
-#endif
+
+      DO_NOTHING_DISABLING_NO_RETURN_WARNINGS;
     }
 #ifdef LISP_COMMAND_LOOP
   UNGCPRO;
   return Qnil;
+#else
+  RETURN_NOT_REACHED (Qnil);
 #endif
 }
 
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/src/coding-system-slots.h	Sun Jan 12 11:08:22 2003 +0000
@@ -0,0 +1,110 @@
+/* Definitions of marked slots in coding systems
+   Copyright (C) 1991, 1995 Free Software Foundation, Inc.
+   Copyright (C) 1995 Sun Microsystems, Inc.
+   Copyright (C) 2000, 2001, 2002 Ben Wing.
+
+This file is part of XEmacs.
+
+XEmacs is free software; you can redistribute it and/or modify it
+under the terms of the GNU General Public License as published by the
+Free Software Foundation; either version 2, or (at your option) any
+later version.
+
+XEmacs is distributed in the hope that it will be useful, but WITHOUT
+ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
+FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
+for more details.
+
+You should have received a copy of the GNU General Public License
+along with XEmacs; see the file COPYING.  If not, write to
+the Free Software Foundation, Inc., 59 Temple Place - Suite 330,
+Boston, MA 02111-1307, USA.  */
+
+/* Synched up with: ????.  Split out of file-coding.h. */
+
+/* We define the Lisp_Objects in the coding system structure in a separate
+   file because there are numerous places we want to iterate over them,
+   such as when defining them in the structure, initializing them, or
+   marking them.
+
+   To use, define MARKED_SLOT before including this file.  In the structure
+   definition, you also need to define CODING_SYSTEM_SLOT_DECLARATION.  No
+   need to undefine either value; that happens automatically.  */
+
+#ifndef MARKED_SLOT_ARRAY
+#ifdef CODING_SYSTEM_SLOT_DECLARATION
+#define MARKED_SLOT_ARRAY(slot, size) MARKED_SLOT(slot[size])
+#else
+#define MARKED_SLOT_ARRAY(slot, size) do {		\
+    int mslotidx;					\
+    for (mslotidx = 0; mslotidx < size; mslotidx++)	\
+      {							\
+	MARKED_SLOT (slot[mslotidx])			\
+      }							\
+  } while (0);
+#endif
+#endif /* not MARKED_SLOT_ARRAY */
+
+  /* Name and description of this coding system.  The description
+     should be suitable for a menu entry. */
+  MARKED_SLOT (name)
+  MARKED_SLOT (description)
+
+  /* Mnemonic string displayed in the modeline when this coding
+     system is active for a particular buffer. */
+  MARKED_SLOT (mnemonic)
+
+  /* Long documentation on the coding system. */
+  MARKED_SLOT (documentation)
+  /* Functions to handle additional conversion after reading or before
+     writing. #### This mechanism should be replaced by the ability to
+     simply create new coding system types. */
+  MARKED_SLOT (post_read_conversion)
+  MARKED_SLOT (pre_write_conversion)
+
+  /* If this coding system is not of the correct type for text file
+     conversion (i.e. decodes byte->char), we wrap it with appropriate
+     char<->byte converters.  This is created dynamically, when it's
+     needed, and cached here. */
+  MARKED_SLOT (text_file_wrapper)
+
+  /* ------------------------ junk to handle EOL -------------------------
+     I had hoped that we could handle this without lots of special-case
+     code, but it appears not to be the case if we want to maintain
+     compatibility with the existing way.  However, at least with the way
+     we do things now, we avoid EOL junk in most of the coding system
+     methods themselves, or in the decode/encode functions.  The EOL
+     special-case code is limited to coding-system creation and to the
+     convert-eol and undecided coding system types. */
+
+  /* If this coding system wants autodetection of the EOL type, then at the
+     appropriate time we wrap this coding system with
+     convert-eol-autodetect. (We do NOT do this at creation time because
+     then we end up with multiple convert-eols wrapped into the final
+     result -- esp. with autodetection using `undecided' -- leading to a
+     big mess.) We cache the wrapped coding system here. */
+  MARKED_SLOT (auto_eol_wrapper)
+  
+  /* Subsidiary coding systems that specify a particular type of EOL
+     marking, rather than autodetecting it.  These will only be non-nil
+     if (eol_type == EOL_AUTODETECT).  These are chains. */
+  MARKED_SLOT_ARRAY (eol, 3)
+  /* If this coding system is a subsidiary, this element points back to its
+     parent. */
+  MARKED_SLOT (subsidiary_parent)
+
+  /* At decoding or encoding time, we use the following coding system, if
+     it exists, in place of the coding system object.  This is how we
+     handle coding systems with EOL types of CRLF or CR.  Formerly, we did
+     the canonicalization at creation time, returning a chain in place of
+     the original coding system; but that interferes with
+     `coding-system-property' and causes other complications.  CANONICAL is
+     used when determining the end types of a coding system.
+     canonicalize-after-coding also consults CANONICAL (it has to, because
+     the data in the lstream is based on CANONICAL, not on the original
+     coding system). */
+  MARKED_SLOT (canonical)
+
+#undef MARKED_SLOT
+#undef MARKED_SLOT_ARRAY
+#undef CODING_SYSTEM_SLOT_DECLARATION
--- a/src/config.h.in	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/config.h.in	Sun Jan 12 11:08:22 2003 +0000
@@ -377,6 +377,8 @@
 #undef HAVE_SNPRINTF
 #undef HAVE_STPCPY
 #undef HAVE_STRERROR
+#undef HAVE_STRLWR
+#undef HAVE_STRUPR
 #undef HAVE_SYMLINK
 #undef HAVE_TZSET
 #undef HAVE_ULIMIT
@@ -920,7 +922,6 @@
 #ifndef NOT_C_CODE /* Actually means C or C++ */
 # if defined (__cplusplus)
 /* Avoid C++ keywords used as ordinary C identifiers */
-#  define class c_class
 #  define new   c_new
 #  define this  c_this
 #  define catch c_catch
--- a/src/conslots.h	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/conslots.h	Sun Jan 12 11:08:22 2003 +0000
@@ -1,5 +1,6 @@
 /* Definitions of marked slots in consoles
    Copyright (C) 1990, 1992, 1993 Free Software Foundation, Inc.
+   Copyright (C) 2002 Ben Wing.
 
 This file is part of XEmacs.
 
@@ -35,25 +36,25 @@
     /* Name of this console, for resourcing and printing purposes.
        If not explicitly given, it's initialized in a console-specific
        manner. */
-    MARKED_SLOT (name);
+    MARKED_SLOT (name)
 
     /* What this console is connected to */
-    MARKED_SLOT (connection);
+    MARKED_SLOT (connection)
 
     /* A canonical name for the connection that is used to determine
        whether create_console() is being called on an existing console. */
-    MARKED_SLOT (canon_connection);
+    MARKED_SLOT (canon_connection)
 
     /* List of devices on this console.  */
-    MARKED_SLOT (device_list);
+    MARKED_SLOT (device_list)
 
     /* Currently selected device.  */
-    MARKED_SLOT (selected_device);
+    MARKED_SLOT (selected_device)
 
     /* Most-recently-selected non-minibuffer-only frame.  Always
        the same as the selected frame, unless that's a minibuffer-only
        frame. */
-    MARKED_SLOT (last_nonminibuf_frame);
+    MARKED_SLOT (last_nonminibuf_frame)
 
     /* If non-nil, a keymap that overrides all others but applies only to
        this console.  Lisp code that uses this instead of calling next-event
@@ -64,41 +65,50 @@
        #### This comes from FSF Emacs; but there's probably a better
        solution that involves making next-event itself work over all
        consoles. */
-    MARKED_SLOT (overriding_terminal_local_map);
+    MARKED_SLOT (overriding_terminal_local_map)
 
     /* Last command executed by the editor command loop, not counting
        commands that set the prefix argument.  */
-    MARKED_SLOT (last_command);
+    MARKED_SLOT (last_command)
 
     /* The prefix argument for the next command, in raw form.  */
-    MARKED_SLOT (prefix_arg);
+    MARKED_SLOT (prefix_arg)
 
     /* Where information about a partially completed key sequence
        is kept.  */
-    MARKED_SLOT (command_builder);
+    MARKED_SLOT (command_builder)
 
     /* Non-nil while a kbd macro is being defined.  */
-    MARKED_SLOT (defining_kbd_macro);
+    MARKED_SLOT (defining_kbd_macro)
 
     /* This is a lisp vector, which contains the events of the keyboard macro
        currently being read.  It is reallocated when the macro gets too large.
        */
-    MARKED_SLOT (kbd_macro_builder);
+    MARKED_SLOT (kbd_macro_builder)
 
     /* Last anonymous kbd macro defined.  */
-    MARKED_SLOT (last_kbd_macro);
+    MARKED_SLOT (last_kbd_macro)
 
 #ifdef HAVE_TTY
     /* ERASE character from stty settings.  */
-    MARKED_SLOT (tty_erase_char);
+    MARKED_SLOT (tty_erase_char)
 #endif
 
+    /* Key that invokes QUIT.  */
+    MARKED_SLOT (quit_char)
+
+    /* Event version of quit-char.  */
+    MARKED_SLOT (quit_event)
+
+    /* Event version of critical QUIT (upshifted quit-char).  */
+    MARKED_SLOT (critical_quit_event)
+
     /* Minibufferless frames on this console use this frame's minibuffer.  */
-    MARKED_SLOT (default_minibuffer_frame);
+    MARKED_SLOT (default_minibuffer_frame)
 
     /* Keymap mapping ASCII function key sequences onto their preferred forms.
        Initialized by the terminal-specific lisp files.  */
-    MARKED_SLOT (function_key_map);
+    MARKED_SLOT (function_key_map)
 
 #ifndef CONSOLE_SLOTS_LAST_NAME
 #define CONSOLE_SLOTS_LAST_NAME function_key_map
--- a/src/console-gtk-impl.h	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/console-gtk-impl.h	Sun Jan 12 11:08:22 2003 +0000
@@ -144,7 +144,7 @@
   int num_top_widgets;
 
   /* Our container widget as a Lisp_Object */
-  Lisp_Object lisp_visible_widgets[10];
+  Lisp_Object lisp_visible_widgets[3];
 
   /*************************** Miscellaneous **************************/
 
--- a/src/console-impl.h	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/console-impl.h	Sun Jan 12 11:08:22 2003 +0000
@@ -27,8 +27,8 @@
 
 #include "console.h"
 
-extern const struct struct_description cted_description;
-extern const struct struct_description console_methods_description;
+extern const struct sized_memory_description cted_description;
+extern const struct sized_memory_description console_methods_description;
 
 
 /*
@@ -407,26 +407,24 @@
   /* Description of this console's methods.  */
   struct console_methods *conmeths;
 
-#ifdef USE_KKCC
+  /* Enumerated constant listing which type of console this is (TTY, X,
+     MS-Windows, etc.).  This duplicates the symbol in conmeths->symbol,
+     which formerly was the only way to determine the console type.
+     We need this constant now for KKCC, so that it can be used in
+     an XD_UNION clause to determine the Lisp objects in console_data. */
   enum console_variant contype;
-#endif /* USE_KKCC */  
 
   /* A structure of auxiliary data specific to the console type.
      struct x_console is used for X window frames; defined in console-x.h
      struct tty_console is used to TTY's; defined in console-tty.h */
   void *console_data;
 
-  /* Character that causes a quit.  Normally C-g.
-     #### Should be possible for this not to be ASCII. (Currently works
-     under Windows.) */
-  Ichar quit_char;
-
   /* ----- begin partially-completed console localization of
            event loop ---- */
 
   int local_var_flags;
 
-#define MARKED_SLOT(x) Lisp_Object x
+#define MARKED_SLOT(x) Lisp_Object x;
 #include "conslots.h"
 
   /* Where to store the next keystroke of the macro.
@@ -561,6 +559,8 @@
   DEVICE_SELECTED_FRAME (XDEVICE ((con)->selected_device))
 #define CONSOLE_LAST_NONMINIBUF_FRAME(con) NON_LVALUE ((con)->last_nonminibuf_frame)
 #define CONSOLE_QUIT_CHAR(con) ((con)->quit_char)
+#define CONSOLE_QUIT_EVENT(con) ((con)->quit_event)
+#define CONSOLE_CRITICAL_QUIT_EVENT(con) ((con)->critical_quit_event)
 
 DECLARE_CONSOLE_TYPE (dead);
 
--- a/src/console-msw-impl.h	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/console-msw-impl.h	Sun Jan 12 11:08:22 2003 +0000
@@ -269,8 +269,6 @@
  */
 
 /* win32 messages / magic event types */
-#define EVENT_MSWINDOWS_MAGIC_TYPE(e)	\
-	((e)->event.magic.underlying_mswindows_event)
 #define XM_BUMPQUEUE	(WM_USER + 101)
 #define XM_MAPFRAME	(WM_USER + 102)
 #define XM_UNMAPFRAME	(WM_USER + 103)
--- a/src/console-msw.c	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/console-msw.c	Sun Jan 12 11:08:22 2003 +0000
@@ -293,7 +293,13 @@
       qxeOutputDebugString (extptr);
     }
   else
-    OutputDebugStringA ((char *) str);
+    {
+      /* STR may not be null-terminated so make it that way. */
+      Extbyte *ext = alloca_extbytes (len + 1);
+      memcpy (ext, str, len);
+      ext[len] = '\0';
+      OutputDebugStringA (ext);
+    }
 }
 
 #ifdef DEBUG_XEMACS
--- a/src/console-msw.h	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/console-msw.h	Sun Jan 12 11:08:22 2003 +0000
@@ -148,8 +148,6 @@
 
 #ifdef HAVE_MENUBARS
 int mswindows_char_is_accelerator (struct frame *f, Ichar ch);
-Lisp_Object mswindows_translate_menu_or_dialog_item (Lisp_Object item,
-						     Ichar *accel);
 #endif
 
 #ifdef HAVE_TOOLBARS
@@ -160,7 +158,8 @@
 #endif
 Lisp_Object mswindows_handle_gui_wm_command (struct frame *f,
 					     HWND ctrl, LPARAM id);
-
+Lisp_Object mswindows_translate_menu_or_dialog_item (Lisp_Object item,
+						     Ichar *accel);
 void mswindows_handle_destroyclipboard (void);
 
 Lisp_Object mswindows_handle_print_dialog_box (struct frame *f,
--- a/src/console-stream-impl.h	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/console-stream-impl.h	Sun Jan 12 11:08:22 2003 +0000
@@ -1,6 +1,7 @@
 /* Define stream specific console, device, and frame object for XEmacs.
    Copyright (C) 1995 Board of Trustees, University of Illinois.
    Copyright (C) 1995 Free Software Foundation, Inc.
+   Copyright (C) 2002 Ben Wing.
 
 This file is part of XEmacs.
 
@@ -37,6 +38,7 @@
   FILE *out;
   FILE *err;
   int needs_newline;
+  Lisp_Object instream;
 };
 
 #define CONSOLE_STREAM_DATA(con) CONSOLE_TYPE_DATA (con, stream)
--- a/src/console-stream.c	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/console-stream.c	Sun Jan 12 11:08:22 2003 +0000
@@ -1,6 +1,6 @@
 /* Stream device functions.
    Copyright (C) 1995 Free Software Foundation, Inc.
-   Copyright (C) 1996, 2001 Ben Wing.
+   Copyright (C) 1996, 2001, 2002 Ben Wing.
 
 This file is part of XEmacs.
 
@@ -48,6 +48,15 @@
 
 Lisp_Object Vstdio_str;
 
+static const struct memory_description stream_console_data_description_1 [] = {
+  { XD_LISP_OBJECT, offsetof (struct stream_console, instream) },
+  { XD_END }
+};
+
+const struct sized_memory_description stream_console_data_description = {
+  sizeof (struct stream_console), stream_console_data_description_1
+};
+
 static void
 stream_init_console (struct console *con, Lisp_Object params)
 {
@@ -55,11 +64,11 @@
   struct stream_console *stream_con;
 
   if (CONSOLE_STREAM_DATA (con) == NULL)
-    CONSOLE_STREAM_DATA (con) = xnew (struct stream_console);
+    CONSOLE_STREAM_DATA (con) = xnew_and_zero (struct stream_console);
 
   stream_con = CONSOLE_STREAM_DATA (con);
 
-  stream_con->needs_newline = 0;
+  stream_con->instream  = Qnil;
 
   /* Open the specified console */
   if (NILP (tty) || internal_equal (tty, Vstdio_str, 0))
--- a/src/console-stream.h	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/console-stream.h	Sun Jan 12 11:08:22 2003 +0000
@@ -1,6 +1,7 @@
 /* Define stream specific console, device, and frame object for XEmacs.
    Copyright (C) 1995 Board of Trustees, University of Illinois.
    Copyright (C) 1995 Free Software Foundation, Inc.
+   Copyright (C) 2002 Ben Wing.
 
 This file is part of XEmacs.
 
@@ -28,6 +29,8 @@
 
 #include "console.h"
 
+extern const struct sized_memory_description stream_console_data_description;
+
 extern Lisp_Object Vterminal_console, Vterminal_frame, Vterminal_device;
 
 Lisp_Object stream_semi_canonicalize_console_connection (Lisp_Object,
--- a/src/console-tty.c	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/console-tty.c	Sun Jan 12 11:08:22 2003 +0000
@@ -49,6 +49,17 @@
 Lisp_Object Qterminal_type;
 Lisp_Object Qcontrolling_process;
 
+static const struct memory_description tty_console_data_description_1 [] = {
+  { XD_LISP_OBJECT, offsetof (struct tty_console, terminal_type) },
+  { XD_LISP_OBJECT, offsetof (struct tty_console, instream) },
+  { XD_LISP_OBJECT, offsetof (struct tty_console, outstream) },
+  { XD_END }
+};
+
+const struct sized_memory_description tty_console_data_description = {
+  sizeof (struct tty_console), tty_console_data_description_1
+};
+
 
 static void
 allocate_tty_console_struct (struct console *con)
--- a/src/console-tty.h	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/console-tty.h	Sun Jan 12 11:08:22 2003 +0000
@@ -1,6 +1,6 @@
 /* Define TTY specific console, device, and frame object for XEmacs.
    Copyright (C) 1995 Board of Trustees, University of Illinois.
-   Copyright (C) 1996 Ben Wing.
+   Copyright (C) 1996, 2002 Ben Wing.
 
 This file is part of XEmacs.
 
@@ -50,6 +50,16 @@
 void send_string_to_tty_console (struct console *c, unsigned char *str,
 				 int len);
 
+/***************     Prototypes from console-tty.c     ****************/
+
+extern const struct sized_memory_description tty_console_data_description;
+
+
+/***************     Prototypes from objects-tty.c     ****************/
+
+extern const struct sized_memory_description tty_color_instance_data_description;
+extern const struct sized_memory_description tty_font_instance_data_description;
+
 
 /***************     Prototypes from redisplay-tty.c     ****************/
 
@@ -79,6 +89,5 @@
 						     Error_Behavior errb);
 Lisp_Object tty_canonicalize_device_connection (Lisp_Object connection,
 						Error_Behavior errb);
-struct console * tty_find_console_from_fd (int fd);
 
 #endif /* INCLUDED_console_tty_h_ */
--- a/src/console-x.h	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/console-x.h	Sun Jan 12 11:08:22 2003 +0000
@@ -65,6 +65,12 @@
   XtGetValues (widget, &al, 1);				\
 } while (0)
 
+#ifdef __cplusplus
+#define X_CLASSFIELD c_class
+#else
+#define X_CLASSFIELD class
+#endif
+
 /* Variables associated with the X display frame this emacs is using. */
 extern XtAppContext Xt_app_con;
 
@@ -98,6 +104,7 @@
 void x_wm_set_variable_size (Widget wmshell, int width, int height);
 
 const char *x_event_name (int event_type);
+int check_if_pending_expose_event (struct device *d);
 int x_error_handler (Display *disp, XErrorEvent *event);
 void expect_x_error (Display *dpy);
 int x_error_occurred_p (Display *dpy);
@@ -131,7 +138,6 @@
 
 int x_frame_window_state (struct frame *f);
 
-
 #define X_ERROR_OCCURRED(dpy, body)	\
      (expect_x_error (dpy), body, x_error_occurred_p (dpy))
 
--- a/src/console.c	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/console.c	Sun Jan 12 11:08:22 2003 +0000
@@ -38,6 +38,7 @@
 #include "sysdep.h"
 #include "window.h"
 
+#include "console-stream-impl.h"
 #ifdef HAVE_TTY
 #include "console-tty-impl.h"
 #endif
@@ -106,75 +107,35 @@
 
 
 
-#ifdef USE_KKCC
-
-static const struct lrecord_description empty_condata_description [] = {
-  { XD_END }
-};
-
-static const struct lrecord_description tty_condata_description [] = {
-  { XD_LISP_OBJECT, offsetof (struct tty_console, terminal_type) },
-  { XD_LISP_OBJECT, offsetof (struct tty_console, instream) },
-  { XD_LISP_OBJECT, offsetof (struct tty_console, outstream) },
-  { XD_END }
-};
-
-static const struct struct_description condata_description []= {
-  { dead_console, empty_condata_description },
-  { tty_console, tty_condata_description },
-  { gtk_console, empty_condata_description },
-  { x_console, empty_condata_description },
-  { mswindows_console, empty_condata_description },
-  { stream_console, empty_condata_description },
-  { XD_END }
-};
-
-static const struct lrecord_description conmeths_description_1 [] = {
-  { XD_LISP_OBJECT, offsetof (struct console_methods, symbol) },
-  /*{ XD_LISP_OBJECT, offsetof (struct console_methods, predicate_symbol) },
-    { XD_LISP_OBJECT, offsetof (struct console_methods, image_conversion_list) },*/
+static const struct memory_description console_data_description_1 []= {
+#ifdef HAVE_TTY
+  { XD_STRUCT_PTR, tty_console, 1, &tty_console_data_description},
+#endif
+  { XD_STRUCT_PTR, stream_console, 1, &stream_console_data_description},
   { XD_END }
 };
 
-static const struct struct_description conmeths_description = {
-  sizeof (struct console_methods),
-  conmeths_description_1
+static const struct sized_memory_description console_data_description = {
+  sizeof (void *), console_data_description_1
 };
 
-static const struct lrecord_description console_description [] = {
+static const struct memory_description console_description [] = {
   { XD_INT, offsetof (struct console, contype) },
-  { XD_LISP_OBJECT, offsetof (struct console, name) },
-  { XD_LISP_OBJECT, offsetof (struct console, connection) },
-  { XD_LISP_OBJECT, offsetof (struct console, canon_connection) },
-  { XD_LISP_OBJECT, offsetof (struct console, device_list) },
-  { XD_LISP_OBJECT, offsetof (struct console, selected_device) },
-  { XD_LISP_OBJECT, offsetof (struct console, last_nonminibuf_frame) },
-  { XD_LISP_OBJECT, offsetof (struct console, overriding_terminal_local_map) },
-  { XD_LISP_OBJECT, offsetof (struct console, last_command) },
-  { XD_LISP_OBJECT, offsetof (struct console, prefix_arg) },
-  { XD_LISP_OBJECT, offsetof (struct console, command_builder) },
-  { XD_LISP_OBJECT, offsetof (struct console, defining_kbd_macro) },
-  { XD_LISP_OBJECT, offsetof (struct console, kbd_macro_builder) },
-  { XD_LISP_OBJECT, offsetof (struct console, last_kbd_macro) },
-#ifdef HAVE_TTY
-  { XD_LISP_OBJECT, offsetof (struct console, tty_erase_char) },
-#endif
-  { XD_LISP_OBJECT, offsetof (struct console, default_minibuffer_frame) },
-  { XD_LISP_OBJECT, offsetof (struct console, function_key_map) },
-  { XD_STRUCT_PTR, offsetof (struct console, conmeths), 1, &conmeths_description },
+#define MARKED_SLOT(x) { XD_LISP_OBJECT, offsetof (struct console, x) },
+#include "conslots.h"
+  { XD_STRUCT_PTR, offsetof (struct console, conmeths), 1,
+    &console_methods_description },
   { XD_UNION, offsetof (struct console, console_data), 
-    XD_INDIRECT (0, 0), condata_description },
+    XD_INDIRECT (0, 0), &console_data_description },
   { XD_END }
 };
 
-#endif /* USE_KKCC */
-
 static Lisp_Object
 mark_console (Lisp_Object obj)
 {
   struct console *con = XCONSOLE (obj);
 
-#define MARKED_SLOT(x) mark_object (con->x)
+#define MARKED_SLOT(x) mark_object (con->x);
 #include "conslots.h"
 
   /* Can be zero for Vconsole_defaults, Vconsole_local_symbols */
@@ -204,20 +165,26 @@
   write_fmt_string (printcharfun, " 0x%x>", con->header.uid);
 }
 
-#ifdef USE_KKCC
 DEFINE_LRECORD_IMPLEMENTATION ("console", console,
 			       0, /*dumpable-flag*/
 			       mark_console, print_console, 0, 0, 0, 
 			       console_description,
 			       struct console);
-#else /* not USE_KKCC */
-DEFINE_LRECORD_IMPLEMENTATION ("console", console,
-			       mark_console, print_console, 0, 0, 0, 0,
-			       struct console);
-#endif /* not USE_KKCC */
 
+
+static void
+set_quit_events (struct console *con, Lisp_Object key)
+{
+  /* Make sure to run Fcharacter_to_event() *BEFORE* setting QUIT_CHAR,
+     so that nothing is changed when invalid values trigger an error! */
+  con->quit_event = Fcharacter_to_event (key, Qnil, wrap_console (con), Qnil);
+  con->quit_char = key;
+  con->critical_quit_event = Fcopy_event (con->quit_event, Qnil);
+  upshift_event (con->critical_quit_event);
+}
+
 static struct console *
-allocate_console (void)
+allocate_console (Lisp_Object type)
 {
   Lisp_Object console;
   struct console *con = alloc_lcrecord_type (struct console, &lrecord_console);
@@ -228,9 +195,11 @@
   console = wrap_console (con);
   GCPRO1 (console);
 
-  con->quit_char = 7; /* C-g */
+  con->conmeths = decode_console_type (type, ERROR_ME);
+  con->contype = get_console_variant (type);
   con->command_builder = allocate_command_builder (console, 1);
   con->function_key_map = Fmake_sparse_keymap (Qnil);
+  set_quit_events (con, make_char (7)); /* C-g */
 
   UNGCPRO;
   return con;
@@ -263,39 +232,27 @@
   return 0;
 }
 
-#ifdef USE_KKCC
 enum console_variant
 get_console_variant (Lisp_Object type)
 {
   if (EQ (type, Qtty)) 
-    {
-      return tty_console;
-    }
+    return tty_console;
 
   if (EQ (type, Qgtk)) 
-    {
-      return gtk_console;
-    }
+    return gtk_console;
 
   if (EQ (type, Qx)) 
-    {
-      return x_console;
-    }
+    return x_console;
 
   if (EQ (type, Qmswindows)) 
-    {
-      return mswindows_console;
-    }
+    return mswindows_console;
 
   if (EQ (type, Qstream)) 
-    {
-      return stream_console;
-    }
+    return stream_console;
 
   abort (); /* should never happen */
   return dead_console; 
 }
-#endif /* USE_KKCC */
 
 int
 valid_console_type_p (Lisp_Object type)
@@ -593,16 +550,11 @@
   if (!NILP (console))
     return console;
 
-  con = allocate_console ();
+  con = allocate_console (type);
   console = wrap_console (con);
 
   GCPRO1 (console);
 
-  con->conmeths = decode_console_type (type, ERROR_ME);
-#ifdef USE_KKCC
-  con->contype = get_console_variant (type);
-#endif /* USE_KKCC */
-
   CONSOLE_NAME (con) = name;
   CONSOLE_CONNECTION (con) =
     semi_canonicalize_console_connection (con->conmeths, connection,
@@ -701,7 +653,7 @@
 {
   zero_lcrecord (con);
 
-#define MARKED_SLOT(x)	con->x = zap
+#define MARKED_SLOT(x)	con->x = zap;
 #include "conslots.h"
 }
 
@@ -858,6 +810,7 @@
      them. */
   nuke_all_console_slots (con, Qnil);
   con->conmeths = dead_console_methods;
+  con->contype = dead_console;
   note_object_deleted (console);
 
   UNGCPRO;
@@ -1133,7 +1086,7 @@
 
 DEFUN ("set-input-mode", Fset_input_mode, 3, 5, 0, /*
 Set mode of reading keyboard input.
-First arg is ignored, for backward compatibility.
+First arg (formerly INTERRUPT-INPUT) is ignored, for backward compatibility.
 Second arg FLOW non-nil means use ^S/^Q flow control for output to terminal
  (no effect except in CBREAK mode).
 Third arg META t means accept 8-bit input (for a Meta key).
@@ -1155,9 +1108,10 @@
 
   if (!NILP (quit))
     {
-      CHECK_CHAR_COERCE_INT (quit);
-      CONSOLE_QUIT_CHAR (con) =
-	((unsigned int) XCHAR (quit)) & (meta_key ? 0377 : 0177);
+      if (CHAR_OR_CHAR_INTP (quit) && !meta_key)
+	set_quit_events (con, make_char (XCHAR_OR_CHAR_INT (quit) & 0177));
+      else
+	set_quit_events (con, quit);
     }
 
 #ifdef HAVE_TTY
@@ -1193,7 +1147,7 @@
        (console))
 {
   struct console *con = decode_console (console);
-  Lisp_Object flow, meta, quit;
+  Lisp_Object flow, meta;
 
 #ifdef HAVE_TTY
   flow = CONSOLE_TTY_P (con) && TTY_FLAGS (con).flow_control ? Qt : Qnil;
@@ -1205,9 +1159,8 @@
   flow = Qnil;
   meta = Qt;
 #endif
-  quit = make_char (CONSOLE_QUIT_CHAR (con));
 
-  return list4 (Qnil, flow, meta, quit);
+  return list4 (Qnil, flow, meta, CONSOLE_QUIT_CHAR (con));
 }
 
 
@@ -1255,35 +1208,35 @@
   DEFSYMBOL (Qsuspend_resume_hook);
 }
 
-static const struct lrecord_description cte_description_1[] = {
+static const struct memory_description cte_description_1[] = {
   { XD_LISP_OBJECT, offsetof (console_type_entry, symbol) },
   { XD_STRUCT_PTR,  offsetof (console_type_entry, meths), 1, &console_methods_description },
   { XD_END }
 };
 
-static const struct struct_description cte_description = {
+static const struct sized_memory_description cte_description = {
   sizeof (console_type_entry),
   cte_description_1
 };
 
-static const struct lrecord_description cted_description_1[] = {
+static const struct memory_description cted_description_1[] = {
   XD_DYNARR_DESC (console_type_entry_dynarr, &cte_description),
   { XD_END }
 };
 
-const struct struct_description cted_description = {
+const struct sized_memory_description cted_description = {
   sizeof (console_type_entry_dynarr),
   cted_description_1
 };
 
-static const struct lrecord_description console_methods_description_1[] = {
+static const struct memory_description console_methods_description_1[] = {
   { XD_LISP_OBJECT, offsetof (struct console_methods, symbol) },
   { XD_LISP_OBJECT, offsetof (struct console_methods, predicate_symbol) },
   { XD_LISP_OBJECT, offsetof (struct console_methods, image_conversion_list) },
   { XD_END }
 };
 
-const struct struct_description console_methods_description = {
+const struct sized_memory_description console_methods_description = {
   sizeof (struct console_methods),
   console_methods_description_1
 };
@@ -1412,7 +1365,9 @@
   /* Set up the non-nil default values of various console slots.
      Must do these before making the first console.
      */
-  /* #### Anything needed here? */
+
+  /* ... Nothing here for the moment.
+   #### Console-local variables should probably be eliminated.*/
 
   {
     /*  0 means var is always local.  Default used only at creation.
@@ -1481,12 +1436,12 @@
 }
 
 
-static const struct lrecord_description console_slots_description_1[] = {
+static const struct memory_description console_slots_description_1[] = {
   { XD_LISP_OBJECT_ARRAY, 0, CONSOLE_SLOTS_COUNT },
   { XD_END }
 };
 
-static const struct struct_description console_slots_description = {
+static const struct sized_memory_description console_slots_description = {
   CONSOLE_SLOTS_SIZE,
   console_slots_description_1
 };
@@ -1594,6 +1549,6 @@
   if ((XINT (console_local_flags.slot) != -2 &&			\
          XINT (console_local_flags.slot) != -3)			\
       != !(NILP (XCONSOLE (Vconsole_local_symbols)->slot)))	\
-  abort ()
+  abort ();
 #include "conslots.h"
 }
--- a/src/console.h	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/console.h	Sun Jan 12 11:08:22 2003 +0000
@@ -46,7 +46,6 @@
 /* GCC does not like forward enum declaration. This needs to be
    defined here. What a disgust! */
 
-#ifdef USE_KKCC
 enum console_variant
 {
   dead_console,
@@ -57,8 +56,6 @@
   stream_console
 };
 
-#endif /* USE_KKCC */
-
 enum device_metrics
 {
   DM_color_default, DM_color_select, DM_color_balloon, DM_color_3d_face,
@@ -105,10 +102,10 @@
 } while (0)
 
 #define CDFW_CONSOLE(obj)				\
-   (WINDOWP  (obj) ? WINDOW_CONSOLE (XWINDOW (obj))	\
- : (FRAMEP   (obj) ?  FRAME_CONSOLE (XFRAME  (obj))	\
- : (DEVICEP  (obj) ? DEVICE_CONSOLE (XDEVICE (obj))    	\
- : (CONSOLEP (obj) ? obj				\
+   ((WINDOWP  (obj) && WINDOW_LIVE_P (XWINDOW(obj))) ? WINDOW_CONSOLE (XWINDOW (obj))	\
+ : ((FRAMEP   (obj) && FRAME_LIVE_P (XFRAME (obj)))  ?  FRAME_CONSOLE (XFRAME  (obj))	\
+ : ((DEVICEP  (obj) && DEVICE_LIVE_P (XDEVICE (obj))) ? DEVICE_CONSOLE (XDEVICE (obj))	\
+ : ((CONSOLEP (obj) && CONSOLE_LIVE_P (XCONSOLE (obj))) ? obj				\
  : Qnil))))
 
 #define CONSOLE_LOOP(concons) LIST_LOOP (concons, Vconsole_list)
@@ -134,9 +131,7 @@
 struct console_methods *decode_console_type (Lisp_Object type,
 					     Error_Behavior errb);
 
-#ifdef USE_KKCC
 enum console_variant get_console_variant (Lisp_Object type);
-#endif /* USE_KKCC */
 
 void delete_console_internal (struct console *con, int force,
 			      int from_kill_emacs, int from_io_error);
--- a/src/data.c	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/data.c	Sun Jan 12 11:08:22 2003 +0000
@@ -739,7 +739,7 @@
 
  range_error:
   args_out_of_range (array, index_);
-  RETURN_NOT_REACHED (Qnil)
+  RETURN_NOT_REACHED (Qnil);
 }
 
 DEFUN ("aset", Faset, 3, 3, 0, /*
@@ -792,7 +792,7 @@
 
  range_error:
   args_out_of_range (array, index_);
-  RETURN_NOT_REACHED (Qnil)
+  RETURN_NOT_REACHED (Qnil);
 }
 
 
@@ -1612,26 +1612,18 @@
   return result;
 }
 
-static const struct lrecord_description weak_list_description[] = {
+static const struct memory_description weak_list_description[] = {
   { XD_LISP_OBJECT, offsetof (struct weak_list, list) },
   { XD_LO_LINK,     offsetof (struct weak_list, next_weak) },
   { XD_END }
 };
 
-#ifdef USE_KKCC
 DEFINE_LRECORD_IMPLEMENTATION ("weak-list", weak_list,
 			       1, /*dumpable-flag*/
 			       mark_weak_list, print_weak_list,
 			       0, weak_list_equal, weak_list_hash,
 			       weak_list_description,
 			       struct weak_list);
-#else /* not USE_KKCC */
-DEFINE_LRECORD_IMPLEMENTATION ("weak-list", weak_list,
-			       mark_weak_list, print_weak_list,
-			       0, weak_list_equal, weak_list_hash,
-			       weak_list_description,
-			       struct weak_list);
-#endif /* not USE_KKCC */
 /*
    -- we do not mark the list elements (either the elements themselves
       or the cons cells that hold them) in the normal marking phase.
@@ -1906,7 +1898,7 @@
   if (EQ (symbol, Qfull_assoc))  return WEAK_LIST_FULL_ASSOC;
 
   invalid_constant ("Invalid weak list type", symbol);
-  RETURN_NOT_REACHED (WEAK_LIST_SIMPLE)
+  RETURN_NOT_REACHED (WEAK_LIST_SIMPLE);
 }
 
 static Lisp_Object
@@ -2083,25 +2075,17 @@
   return result;
 }
 
-static const struct lrecord_description weak_box_description[] = {
+static const struct memory_description weak_box_description[] = {
   { XD_LO_LINK, offsetof (struct weak_box, value) },
   { XD_END}
 };
 
-#ifdef USE_KKCC
 DEFINE_LRECORD_IMPLEMENTATION ("weak_box", weak_box,
 			       0, /*dumpable-flag*/
 			       mark_weak_box, print_weak_box,
 			       0, weak_box_equal, weak_box_hash,
 			       weak_box_description,
 			       struct weak_box);
-#else /* not USE_KKCC */
-DEFINE_LRECORD_IMPLEMENTATION ("weak_box", weak_box,
-			       mark_weak_box, print_weak_box,
-			       0, weak_box_equal, weak_box_hash,
-			       weak_box_description,
-			       struct weak_box);
-#endif /* not USE_KKCC */
 
 DEFUN ("make-weak-box", Fmake_weak_box, 1, 1, 0, /*
 Return a new weak box from value CONTENTS.
@@ -2280,27 +2264,22 @@
   return result;
 }
 
-static const struct lrecord_description ephemeron_description[] = {
-  { XD_LISP_OBJECT, offsetof(struct ephemeron, key)},
-  { XD_LISP_OBJECT, offsetof(struct ephemeron, cons_chain)},
-  { XD_LISP_OBJECT, offsetof(struct ephemeron, value)},
+static const struct memory_description ephemeron_description[] = {
+  { XD_LISP_OBJECT, offsetof(struct ephemeron, key),
+    0, 0, XD_FLAG_NO_KKCC },
+  { XD_LISP_OBJECT, offsetof(struct ephemeron, cons_chain),
+    0, 0, XD_FLAG_NO_KKCC },
+  { XD_LISP_OBJECT, offsetof(struct ephemeron, value),
+    0, 0, XD_FLAG_NO_KKCC },
   { XD_END }
 };
 
-#ifdef USE_KKCC
 DEFINE_LRECORD_IMPLEMENTATION ("ephemeron", ephemeron,
 			       0, /*dumpable-flag*/
 			       mark_ephemeron, print_ephemeron,
 			       0, ephemeron_equal, ephemeron_hash,
 			       ephemeron_description,
 			       struct ephemeron);
-#else /* not USE_KKCC */
-DEFINE_LRECORD_IMPLEMENTATION ("ephemeron", ephemeron,
-			       mark_ephemeron, print_ephemeron,
-			       0, ephemeron_equal, ephemeron_hash,
-			       ephemeron_description,
-			       struct ephemeron);
-#endif /* not USE_KKCC */
 
 DEFUN ("make-ephemeron", Fmake_ephemeron, 2, 3, 0, /*
 Return a new ephemeron with key KEY, value CONTENTS, and finalizer FINALIZER.
--- a/src/database.c	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/database.c	Sun Jan 12 11:08:22 2003 +0000
@@ -142,12 +142,10 @@
   return db;
 }
 
-#ifdef USE_KKCC
-static const struct lrecord_description database_description[] = {
+static const struct memory_description database_description[] = {
   { XD_LISP_OBJECT, offsetof (struct Lisp_Database, fname) },
   { XD_END}
 };
-#endif /* USE_KKCC */
 
 static Lisp_Object
 mark_database (Lisp_Object object)
@@ -189,19 +187,12 @@
   db->funcs->close (db);
 }
 
-#ifdef USE_KKCC
 DEFINE_LRECORD_IMPLEMENTATION ("database", database,
 			       0, /*dumpable-flag*/
                                mark_database, print_database,
 			       finalize_database, 0, 0, 
 			       database_description,
 			       Lisp_Database);
-#else /* not USE_KKCC */
-DEFINE_LRECORD_IMPLEMENTATION ("database", database,
-                               mark_database, print_database,
-			       finalize_database, 0, 0, 0,
-			       Lisp_Database);
-#endif /* not USE_KKCC */
 
 DEFUN ("close-database", Fclose_database, 1, 1, 0, /*
 Close database DATABASE.
--- a/src/debug.c	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/debug.c	Sun Jan 12 11:08:22 2003 +0000
@@ -56,13 +56,13 @@
 };
 
 static Lisp_Object
-xemacs_debug_loop (enum debug_loop op, Lisp_Object class, Lisp_Object type)
+xemacs_debug_loop (enum debug_loop op, Lisp_Object class_, Lisp_Object type)
 {
   int flag = (op == X_ADD) ? 1 : 0;
   Lisp_Object retval = Qnil;
 
 #define FROB(item)							\
-  if (op == X_LIST || op == X_ACTIVE || op == X_INIT || EQ (class, Q##item))	\
+  if (op == X_LIST || op == X_ACTIVE || op == X_INIT || EQ (class_, Q##item))	\
     {									\
       if (op == X_ADD || op == X_DELETE || op == X_INIT)			\
 	active_debug_classes.item = flag;				\
@@ -94,12 +94,12 @@
 DEFUN ("add-debug-class-to-check", Fadd_debug_class_to_check, 1, 1, 0, /*
 Add a debug class to the list of active classes.
 */
-       (class))
+       (class_))
 {
-  if (NILP (xemacs_debug_loop (X_VALIDATE, class, Qnil)))
+  if (NILP (xemacs_debug_loop (X_VALIDATE, class_, Qnil)))
     invalid_argument ("No such debug class exists", Qunbound);
   else
-    xemacs_debug_loop (X_ADD, class, Qnil);
+    xemacs_debug_loop (X_ADD, class_, Qnil);
 
   return (xemacs_debug_loop (X_ACTIVE, Qnil, Qnil));
 }
@@ -107,12 +107,12 @@
 DEFUN ("delete-debug-class-to-check", Fdelete_debug_class_to_check, 1, 1, 0, /*
 Delete a debug class from the list of active classes.
 */
-       (class))
+       (class_))
 {
-  if (NILP (xemacs_debug_loop (X_VALIDATE, class, Qnil)))
+  if (NILP (xemacs_debug_loop (X_VALIDATE, class_, Qnil)))
     invalid_argument ("No such debug class exists", Qunbound);
   else
-    xemacs_debug_loop (X_DELETE, class, Qnil);
+    xemacs_debug_loop (X_DELETE, class_, Qnil);
 
   return (xemacs_debug_loop (X_ACTIVE, Qnil, Qnil));
 }
@@ -145,7 +145,7 @@
 
   /* Make sure all objects in the list are valid.  If anyone is not
      valid, reject the entire list without doing anything. */
-  LIST_LOOP (rest, classes )
+  LIST_LOOP (rest, classes)
     {
       if (NILP (xemacs_debug_loop (X_VALIDATE, XCAR (rest), Qnil)))
 	sferror ("Invalid object in class list", Qunbound);
@@ -162,26 +162,26 @@
 TYPES should be an integer representing the or'd value of all desired types.
 Lists of defined types and their values are located in the source code.
 */
-       (class, type))
+       (class_, type))
 {
   CHECK_INT (type);
-  if (NILP (xemacs_debug_loop (X_VALIDATE, class, Qnil)))
+  if (NILP (xemacs_debug_loop (X_VALIDATE, class_, Qnil)))
     invalid_argument ("Invalid debug class", Qunbound);
 
-  xemacs_debug_loop (X_SETTYPE, class, type);
+  xemacs_debug_loop (X_SETTYPE, class_, type);
 
-  return (xemacs_debug_loop (X_TYPE, class, Qnil));
+  return (xemacs_debug_loop (X_TYPE, class_, Qnil));
 }
 
 DEFUN ("debug-types-being-checked", Fdebug_types_being_checked, 1, 1, 0, /*
 For the given CLASS, return the associated type value.
 */
-       (class))
+       (class_))
 {
-  if (NILP (xemacs_debug_loop (X_VALIDATE, class, Qnil)))
+  if (NILP (xemacs_debug_loop (X_VALIDATE, class_, Qnil)))
     invalid_argument ("Invalid debug class", Qunbound);
 
-  return (xemacs_debug_loop (X_TYPE, class, Qnil));
+  return (xemacs_debug_loop (X_TYPE, class_, Qnil));
 }
 
 void
--- a/src/debug.h	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/debug.h	Sun Jan 12 11:08:22 2003 +0000
@@ -53,10 +53,10 @@
 
 extern struct debug_classes active_debug_classes;
 
-#define DASSERT(class, desired_type, action, assertion) do		\
+#define DASSERT(class_, desired_type, action, assertion) do		\
 {									\
-  if (active_debug_classes.##class					\
-      && (active_debug_classes.types_of_##class & desired_type))	\
+  if (active_debug_classes.##class_					\
+      && (active_debug_classes.types_of_##class_ & desired_type))	\
     {									\
       if (! (assertion))						\
 	{								\
@@ -70,7 +70,7 @@
 } while (0)
 #else /* !DEBUG_XEMACS */
 
-#define DASSERT(class, desired_type, action, assertion)	((void) 0)
+#define DASSERT(class_, desired_type, action, assertion)	((void) 0)
 
 #endif /* !DEBUG_XEMACS */
 
--- a/src/device-gtk.c	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/device-gtk.c	Sun Jan 12 11:08:22 2003 +0000
@@ -67,6 +67,18 @@
 
 static void gtk_device_init_x_specific_cruft (struct device *d);
 
+static const struct memory_description gtk_device_data_description_1 [] = {
+  { XD_LISP_OBJECT, offsetof (struct gtk_device, x_keysym_map_hashtable) },
+  { XD_LISP_OBJECT, offsetof (struct gtk_device, WM_COMMAND_frame) },
+  { XD_END }
+};
+
+extern const struct sized_memory_description gtk_device_data_description;
+
+const struct sized_memory_description gtk_device_data_description = {
+  sizeof (struct gtk_device), gtk_device_data_description_1
+};
+
 
 /************************************************************************/
 /*                          helper functions                            */
--- a/src/device-impl.h	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/device-impl.h	Sun Jan 12 11:08:22 2003 +0000
@@ -77,9 +77,8 @@
      through device->console, but it's faster this way. */
   struct console_methods *devmeths;
 
-#ifdef USE_KKCC
+  /* Duplicates devmeths->symbol.  See comment in struct console. */
   enum console_variant devtype;
-#endif /* USE_KKCC */  
 
   /* A structure of auxiliary data specific to the device type.
      struct x_device is used for X window frames; defined in console-x.h
@@ -123,7 +122,7 @@
   unsigned int on_console_p :1;
   unsigned int connected_to_nas_p :1;
 
-#define MARKED_SLOT(x) Lisp_Object x
+#define MARKED_SLOT(x) Lisp_Object x;
 #include "devslots.h"
 
   /* File descriptors for input and output.  Much of the time
--- a/src/device-msw.c	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/device-msw.c	Sun Jan 12 11:08:22 2003 +0000
@@ -68,6 +68,17 @@
 static Lisp_Object Q_selected_page_button;
 static Lisp_Object Qselected_page_button;
 
+static const struct memory_description mswindows_device_data_description_1 [] = {
+  { XD_LISP_OBJECT, offsetof (struct mswindows_device, fontlist) },
+  { XD_END }
+};
+
+extern const struct sized_memory_description mswindows_device_data_description;
+
+const struct sized_memory_description mswindows_device_data_description = {
+  sizeof (struct mswindows_device), mswindows_device_data_description_1
+};
+
 static Lisp_Object allocate_devmode (DEVMODEW *src_devmode, int do_copy,
 				     Lisp_Object src_name, struct device *d);
 
@@ -1074,13 +1085,11 @@
 /*                                devmode                               */
 /************************************************************************/
 
-#ifdef USE_KKCC
-static const struct lrecord_description devmode_description[] = {
+static const struct memory_description devmode_description[] = {
   { XD_LISP_OBJECT, offsetof (struct Lisp_Devmode, printer_name) },
   { XD_LISP_OBJECT, offsetof (struct Lisp_Devmode, device) },
   { XD_END }
 };
-#endif /* USE_KKCC */
 
 static Lisp_Object
 mark_devmode (Lisp_Object obj)
@@ -1151,19 +1160,12 @@
 		internal_hash (dm->printer_name, depth + 1));
 }
 
-#ifdef USE_KKCC
 DEFINE_LRECORD_IMPLEMENTATION ("msprinter-settings", devmode,
 			       0, /*dumpable-flag*/
 			       mark_devmode, print_devmode, finalize_devmode,
 			       equal_devmode, hash_devmode, 
 			       devmode_description,
 			       Lisp_Devmode);
-#else /* not USE_KKCC */
-DEFINE_LRECORD_IMPLEMENTATION ("msprinter-settings", devmode,
-			       mark_devmode, print_devmode, finalize_devmode,
-			       equal_devmode, hash_devmode, 0/*description*/,
-			       Lisp_Devmode);
-#endif /* not USE_KKCC */
 
 static Lisp_Object
 allocate_devmode (DEVMODEW* src_devmode, int do_copy,
--- a/src/device-x.c	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/device-x.c	Sun Jan 12 11:08:22 2003 +0000
@@ -102,6 +102,18 @@
   {"-fontset",  "*FontSet",                     XrmoptionSepArg, NULL},
 };
 
+static const struct memory_description x_device_data_description_1 [] = {
+  { XD_LISP_OBJECT, offsetof (struct x_device, x_keysym_map_hash_table) },
+  { XD_LISP_OBJECT, offsetof (struct x_device, WM_COMMAND_frame) },
+  { XD_END }
+};
+
+extern const struct sized_memory_description x_device_data_description;
+
+const struct sized_memory_description x_device_data_description = {
+  sizeof (struct x_device), x_device_data_description_1
+};
+
 /* Functions to synchronize mirroring resources and specifiers */
 int in_resource_setting;
 
@@ -244,7 +256,7 @@
 {
   if (DEVICE_X_DEPTH(d) > 2)
     {
-      switch (DEVICE_X_VISUAL(d)->class)
+      switch (DEVICE_X_VISUAL(d)->X_CLASSFIELD)
 	{
 	case StaticGray:
 	case GrayScale:
@@ -372,9 +384,9 @@
     {
       if ( left->colormap_size > right->colormap_size )
 	return 1;
-      if ( left->class > right->class )
+      if ( left->X_CLASSFIELD > right->X_CLASSFIELD )
 	return 1;
-      else if ( left->class < right->class )
+      else if ( left->X_CLASSFIELD < right->X_CLASSFIELD )
 	return -1;
       else
 	return 0;
@@ -393,7 +405,7 @@
   XVisualInfo *vi_out = NULL;
   int out_count;
 
-  vi_in.class = visual_class;
+  vi_in.X_CLASSFIELD = visual_class;
   vi_in.screen = scrnum;
   vi_out = XGetVisualInfo (dpy, (VisualClassMask | VisualScreenMask),
 			   &vi_in, &out_count);
@@ -744,7 +756,7 @@
       {
 	sprintf (buf1, "%s.privateColormap", app_name);
 	sprintf (buf2, "%s.PrivateColormap", app_class);
-	if ((visual->class == PseudoColor) &&
+	if ((visual->X_CLASSFIELD == PseudoColor) &&
 	    (XrmGetResource (XtDatabase (dpy), buf1, buf2, &type, &value)
 	     == True))
 	  cmap = XCopyColormapAndFree (dpy, DefaultColormap (dpy, screen));
@@ -1223,7 +1235,7 @@
 
 static void
 construct_name_list (Display *display, Widget widget, char *fake_name,
-		     char *fake_class, char *name, char *class)
+		     char *fake_class, char *name, char *class_)
 {
   char *stack [100][2];
   Widget this;
@@ -1260,10 +1272,10 @@
 				&stack [count][1]);
 
   name [0] = 0;
-  class [0] = 0;
+  class_ [0] = 0;
 
   name_tail  = name;
-  class_tail = class;
+  class_tail = class_;
   for (; count >= 0; count--)
     {
       strcat (name_tail,  stack [count][0]);
@@ -1292,7 +1304,7 @@
 static void
 x_get_resource_prefix (Lisp_Object locale, Lisp_Object device,
 		       Display **display_out, Extbyte_dynarr *name,
-		       Extbyte_dynarr *class)
+		       Extbyte_dynarr *class_)
 {
   if (NILP (locale))
     locale = Qglobal;
@@ -1333,9 +1345,9 @@
     name_len  = strlen (appname);
     class_len = strlen (appclass);
     Dynarr_add_many (name,  appname,  name_len);
-    Dynarr_add_many (class, appclass, class_len);
+    Dynarr_add_many (class_, appclass, class_len);
     validify_resource_component (Dynarr_atp (name,  0), name_len);
-    validify_resource_component (Dynarr_atp (class, 0), class_len);
+    validify_resource_component (Dynarr_atp (class_, 0), class_len);
   }
 
   if (EQ (locale, Qglobal))
@@ -1345,14 +1357,14 @@
       Dynarr_add_literal_string (name, ".buffer.");
       /* we know buffer is live; otherwise we got an error above. */
       Dynarr_add_validified_lisp_string (name, Fbuffer_name (locale));
-      Dynarr_add_literal_string (class, ".EmacsLocaleType.EmacsBuffer");
+      Dynarr_add_literal_string (class_, ".EmacsLocaleType.EmacsBuffer");
     }
   else if (FRAMEP (locale))
     {
       Dynarr_add_literal_string (name, ".frame.");
       /* we know frame is live; otherwise we got an error above. */
       Dynarr_add_validified_lisp_string (name, Fframe_name (locale));
-      Dynarr_add_literal_string (class, ".EmacsLocaleType.EmacsFrame");
+      Dynarr_add_literal_string (class_, ".EmacsLocaleType.EmacsFrame");
     }
   else
     {
@@ -1360,7 +1372,7 @@
       Dynarr_add_literal_string (name, ".device.");
       /* we know device is live; otherwise we got an error above. */
       Dynarr_add_validified_lisp_string (name, Fdevice_name (locale));
-      Dynarr_add_literal_string (class, ".EmacsLocaleType.EmacsDevice");
+      Dynarr_add_literal_string (class_, ".EmacsLocaleType.EmacsDevice");
     }
   return;
 }
@@ -1447,7 +1459,7 @@
 returned value is the list (t) for true, (nil) for false, and is nil to
 mean ``unspecified''.
 */
-       (name, class, type, locale, device, noerror))
+       (name, class_, type, locale, device, noerror))
 {
   Extbyte *name_string, *class_string;
   Extbyte *raw_result;
@@ -1457,7 +1469,7 @@
   Lisp_Object codesys;
 
   CHECK_STRING (name);
-  CHECK_STRING (class);
+  CHECK_STRING (class_);
   CHECK_SYMBOL (type);
 
   Dynarr_reset (name_Extbyte_dynarr);
@@ -1473,7 +1485,7 @@
   Dynarr_add (name_Extbyte_dynarr, '.');
   Dynarr_add_lisp_string (name_Extbyte_dynarr, name, Qbinary);
   Dynarr_add (class_Extbyte_dynarr, '.');
-  Dynarr_add_lisp_string (class_Extbyte_dynarr, class, Qbinary);
+  Dynarr_add_lisp_string (class_Extbyte_dynarr, class_, Qbinary);
   Dynarr_add (name_Extbyte_dynarr,  '\0');
   Dynarr_add (class_Extbyte_dynarr, '\0');
 
@@ -1500,7 +1512,7 @@
       {
 	maybe_signal_error_2
 	  (Qstructure_formation_error,
-	   "class list and name list must be the same length", name, class,
+	   "class list and name list must be the same length", name, class_,
 	   Qresource, errb);
 	return Qnil;
       }
@@ -1641,7 +1653,7 @@
        (device))
 {
   Visual *vis = DEVICE_X_VISUAL (decode_x_device (device));
-  switch (vis->class)
+  switch (vis->X_CLASSFIELD)
     {
     case StaticGray:  return intern ("static-gray");
     case GrayScale:   return intern ("gray-scale");
--- a/src/device.c	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/device.c	Sun Jan 12 11:08:22 2003 +0000
@@ -45,19 +45,6 @@
 #include "toolbar.h"
 #include "window.h"
 
-#ifdef USE_KKCC
-#include "console-tty-impl.h"
-#ifdef HAVE_MS_WINDOWS
-#include "console-msw-impl.h"
-#endif
-#ifdef HAVE_X_WINDOWS
-#include "console-x-impl.h"
-#endif
-#ifdef HAVE_GTK
-#include "console-gtk-impl.h"
-#endif
-#endif /* USE_KKCC */
-
 #ifdef HAVE_SCROLLBARS
 #include "scrollbar.h"
 #endif
@@ -98,89 +85,45 @@
 
 
 
-#ifdef USE_KKCC
-
-static const struct lrecord_description empty_devdata_description [] = {
-  { XD_END }
-};
-
-static const struct lrecord_description mswindows_devdata_description [] = {
-#ifdef HAVE_MS_WINDOWS
-  { XD_LISP_OBJECT, offsetof (struct mswindows_device, fontlist) },
-#endif
-  { XD_END }
-};
+extern const struct sized_memory_description gtk_device_data_description;
+extern const struct sized_memory_description mswindows_device_data_description;
+extern const struct sized_memory_description x_device_data_description;
 
-static const struct lrecord_description gtk_devdata_description [] = {
+static const struct memory_description device_data_description_1 []= {
 #ifdef HAVE_GTK
-  { XD_LISP_OBJECT, offsetof (struct gtk_device, x_keysym_map_hash_table) },
-  { XD_LISP_OBJECT, offsetof (struct gtk_device, WM_COMMAND_frame) },
+  { XD_STRUCT_PTR, gtk_console, 1, &gtk_device_data_description},
 #endif
-  { XD_END }
-};
-
-static const struct lrecord_description x_devdata_description [] = {
+#ifdef HAVE_MS_WINDOWS
+  { XD_STRUCT_PTR, mswindows_console, 1, &mswindows_device_data_description},
+#endif
 #ifdef HAVE_X_WINDOWS
-  { XD_LISP_OBJECT, offsetof (struct x_device, x_keysym_map_hash_table) },
-  { XD_LISP_OBJECT, offsetof (struct x_device, WM_COMMAND_frame) },
+  { XD_STRUCT_PTR, x_console, 1, &x_device_data_description},
 #endif
   { XD_END }
 };
 
-static const struct struct_description devdata_description []= {
-  { dead_console, empty_devdata_description},
-  { tty_console, empty_devdata_description},
-  { gtk_console, gtk_devdata_description},
-  { x_console, x_devdata_description},
-  { mswindows_console, mswindows_devdata_description},
-  { stream_console, empty_devdata_description},
-  { XD_END }
-};
-
-static const struct lrecord_description conmeths_description_1 [] = {
-  { XD_LISP_OBJECT, offsetof (struct console_methods, symbol) },
-  { XD_END }
-};
-
-static const struct struct_description conmeths_description = {
-  sizeof (struct console_methods),
-  conmeths_description_1
+static const struct sized_memory_description device_data_description = {
+  sizeof (void *), device_data_description_1
 };
 
-static const struct lrecord_description device_description [] = {
+static const struct memory_description device_description [] = {
   { XD_INT, offsetof (struct device, devtype) },
-  { XD_LISP_OBJECT, offsetof (struct device, name) },
-  { XD_LISP_OBJECT, offsetof (struct device, connection) },
-  { XD_LISP_OBJECT, offsetof (struct device, canon_connection) },
-  { XD_LISP_OBJECT, offsetof (struct device, frame_list) },
-  { XD_LISP_OBJECT, offsetof (struct device, console) },
-  { XD_LISP_OBJECT, offsetof (struct device, selected_frame) },
-  { XD_LISP_OBJECT, offsetof (struct device, frame_with_focus_real) },
-  { XD_LISP_OBJECT, offsetof (struct device, frame_with_focus_for_hooks) },
-  { XD_LISP_OBJECT, offsetof (struct device, frame_that_ought_to_have_focus) },
-  { XD_LISP_OBJECT, offsetof (struct device, device_class) },
-  { XD_LISP_OBJECT, offsetof (struct device, user_defined_tags) },
-  { XD_LISP_OBJECT, offsetof (struct device, color_instance_cache) },
-  { XD_LISP_OBJECT, offsetof (struct device, font_instance_cache) },
-#ifdef MULE
-  { XD_LISP_OBJECT, offsetof (struct device, charset_font_cache_stage_1) },
-  { XD_LISP_OBJECT, offsetof (struct device, charset_font_cache_stage_2) },
-#endif
-  { XD_LISP_OBJECT, offsetof (struct device, image_instance_cache) },
-  { XD_LISP_OBJECT, offsetof (struct device, frame_list) },
-  { XD_STRUCT_PTR, offsetof (struct device, devmeths), 1, &conmeths_description },
+#define MARKED_SLOT(x) { XD_LISP_OBJECT, offsetof (struct device, x) },
+#include "devslots.h"
+
+  { XD_STRUCT_PTR, offsetof (struct device, devmeths), 1,
+    &console_methods_description },
   { XD_UNION, offsetof (struct device, device_data), 
-    XD_INDIRECT (0, 0), devdata_description },
+    XD_INDIRECT (0, 0), &device_data_description },
   { XD_END }
 };
-#endif /* USE_KKCC */
 
 static Lisp_Object
 mark_device (Lisp_Object obj)
 {
   struct device *d = XDEVICE (obj);
 
-#define MARKED_SLOT(x) mark_object (d->x)
+#define MARKED_SLOT(x) mark_object (d->x);
 #include "devslots.h"
 
   if (d->devmeths)
@@ -208,22 +151,16 @@
   write_fmt_string (printcharfun, " 0x%x>", d->header.uid);
 }
 
-#ifdef USE_KKCC
 DEFINE_LRECORD_IMPLEMENTATION ("device", device,
 			       0, /*dumpable-flag*/
 			       mark_device, print_device, 0, 0, 0, 
 			       device_description,
 			       struct device);
-#else /* not USE_KKCC */
-DEFINE_LRECORD_IMPLEMENTATION ("device", device,
-			       mark_device, print_device, 0, 0, 0, 0,
-			       struct device);
-#endif /* not USE_KKCC */
 
 int
-valid_device_class_p (Lisp_Object class)
+valid_device_class_p (Lisp_Object class_)
 {
-  return !NILP (memq_no_quit (class, Vdevice_class_list));
+  return !NILP (memq_no_quit (class_, Vdevice_class_list));
 }
 
 DEFUN ("valid-device-class-p", Fvalid_device_class_p, 1, 1, 0, /*
@@ -248,7 +185,7 @@
 {
   zero_lcrecord (d);
 
-#define MARKED_SLOT(x)	d->x = zap
+#define MARKED_SLOT(x)	d->x = zap;
 #include "devslots.h"
 }
 
@@ -259,8 +196,6 @@
   struct device *d = alloc_lcrecord_type (struct device, &lrecord_device);
   struct gcpro gcpro1;
 
-  zero_lcrecord (d);
-
   device = wrap_device (d);
   GCPRO1 (device);
 
@@ -660,7 +595,7 @@
   Lisp_Object console = Qnil;
   Lisp_Object name = Qnil;
   struct console_methods *conmeths;
-  int speccount = specpdl_depth();
+  int speccount = specpdl_depth ();
 
   struct gcpro gcpro1, gcpro2, gcpro3;
   /* If this is the first device we are creating of a particular type
@@ -702,9 +637,7 @@
   device = wrap_device (d);
 
   d->devmeths = con->conmeths;
-#ifdef USE_KKCC
   d->devtype = get_console_variant (type);
-#endif /* USE_KKCC */
 
   DEVICE_NAME (d) = name;
   DEVICE_CONNECTION (d) =
@@ -902,7 +835,7 @@
 
     /* #### This should probably be a device method but it is time for
        19.14 to go out the door. */
-    /* #### BILL!!! Should this deal with HAVE_MSWINDOWS as well? */
+    /* #### BILL!!! Should this deal with HAVE_MS_WINDOWS as well? */
 #if defined (HAVE_X_WINDOWS) || defined (HAVE_GTK)
     /* Next delete all frames which have the popup property to avoid
        deleting a child after its parent. */
@@ -993,6 +926,7 @@
      them. */
   nuke_all_device_slots (d, Qnil);
   d->devmeths = dead_console_methods;
+  d->devtype = dead_console;
   note_object_deleted (device);
 
   UNGCPRO;
@@ -1048,18 +982,18 @@
 This is only allowed on device such as TTY devices, where the color
 behavior cannot necessarily be determined automatically.
 */
-       (device, class))
+       (device, class_))
 {
   struct device *d = decode_device (device);
   device = wrap_device (d);
   if (!DEVICE_TTY_P (d))
     gui_error ("Cannot change the class of this device", device);
-  if (!EQ (class, Qcolor) && !EQ (class, Qmono) && !EQ (class, Qgrayscale))
-    invalid_constant ("Must be color, mono, or grayscale", class);
-  if (! EQ (DEVICE_CLASS (d), class))
+  if (!EQ (class_, Qcolor) && !EQ (class_, Qmono) && !EQ (class_, Qgrayscale))
+    invalid_constant ("Must be color, mono, or grayscale", class_);
+  if (! EQ (DEVICE_CLASS (d), class_))
     {
       Lisp_Object frmcons;
-      DEVICE_CLASS (d) = class;
+      DEVICE_CLASS (d) = class_;
       DEVICE_FRAME_LOOP (frmcons, d)
 	{
 	  struct frame *f = XFRAME (XCAR (frmcons));
--- a/src/device.h	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/device.h	Sun Jan 12 11:08:22 2003 +0000
@@ -98,7 +98,7 @@
 extern Lisp_Object Qinit_post_tty_win, Qmono;
 extern Lisp_Object Vdevice_class_list;
 
-int valid_device_class_p (Lisp_Object class);
+int valid_device_class_p (Lisp_Object class_);
 
 void select_device_1 (Lisp_Object);
 struct device *decode_device (Lisp_Object);
--- a/src/devslots.h	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/devslots.h	Sun Jan 12 11:08:22 2003 +0000
@@ -26,31 +26,30 @@
    as when defining them in the structure, initializing them, or marking
    them.
 
-   To use, define MARKED_SLOT before including this file.  In the structure
-   definition, you also need to define FRAME_SLOT_DECLARATION.  No need to
-   undefine either value; that happens automatically.  */
+   To use, define MARKED_SLOT before including this file.  No need to
+   undefine; that happens automatically. */
 
   /* Name of this device, for resourcing and printing purposes.
      If not explicitly given, it's initialized in a device-specific
      manner. */
-  MARKED_SLOT (name);
+  MARKED_SLOT (name)
 
   /* What this device is connected to */
-  MARKED_SLOT (connection);
+  MARKED_SLOT (connection)
 
   /* A canonical name for the connection that is used to determine
      whether `make-device' is being called on an existing device. */
-  MARKED_SLOT (canon_connection);
+  MARKED_SLOT (canon_connection)
 
   /* List of frames on this device. */
-  MARKED_SLOT (frame_list);
+  MARKED_SLOT (frame_list)
 
   /* The console this device is on. */
-  MARKED_SLOT (console);
+  MARKED_SLOT (console)
 
   /* Frame which is "currently selected".  This is what `selected-frame'
      returns and is the default frame for many operations.  This may
-     not be the same as frame_with_focus; `select-frame' changes the
+     not be the same as frame_with_focus `select-frame' changes the
      selected_frame but not the frame_with_focus.  However, eventually
      either the two values will be the same, or frame_with_focus will
      be nil: right before waiting for an event, the focus is changed
@@ -59,18 +58,18 @@
      frames on this device have the window-system focus), but
      selected_frame will never be nil if there are any frames on
      the device. */
-  MARKED_SLOT (selected_frame);
+  MARKED_SLOT (selected_frame)
   /* Frame that currently contains the window-manager focus, or none.
      Note that we've split frame_with_focus into two variables.
      frame_with_focus_real is the value we use most of the time,
      but frame_with_focus_for_hooks is used for running the select-frame-hook
      and deselect-frame-hook.  We do this because we split the focus handling
      into two parts: one part (deals with drawing the solid/box cursor)
-     runs as soon as a focus event is received; the other (running the
+     runs as soon as a focus event is received the other (running the
      hooks) runs after any pending sit-for/sleep-for/accept-process-output
      calls are done. */
-  MARKED_SLOT (frame_with_focus_real);
-  MARKED_SLOT (frame_with_focus_for_hooks);
+  MARKED_SLOT (frame_with_focus_real)
+  MARKED_SLOT (frame_with_focus_for_hooks)
   /* If we have recently issued a request to change the focus as a
      result of select-frame having been called, the following variable
      records the frame we are trying to focus on.  The reason for this
@@ -78,13 +77,13 @@
      the focus (so we can't just change frame_with_focus), and we don't
      want to keep sending requests again and again to the window manager.
      This variable is reset whenever a focus-change event is seen. */
-  MARKED_SLOT (frame_that_ought_to_have_focus);
+  MARKED_SLOT (frame_that_ought_to_have_focus)
 
   /* Color class of this device. */
-  MARKED_SLOT (device_class);
+  MARKED_SLOT (device_class)
 
   /* Alist of values for user-defined tags in this device. */
-  MARKED_SLOT (user_defined_tags);
+  MARKED_SLOT (user_defined_tags)
 
   /* Hash tables for device-specific objects (fonts, colors, etc).
      These are key-weak hash tables (or hash tables containing key-weak
@@ -92,11 +91,11 @@
 
   /* This is a simple key-weak hash table hashing color names to
      instances. */
-  MARKED_SLOT (color_instance_cache);
+  MARKED_SLOT (color_instance_cache)
 
   /* This is a simple key-weak hash table hashing font names to
      instances. */
-  MARKED_SLOT (font_instance_cache);
+  MARKED_SLOT (font_instance_cache)
 
 #ifdef MULE
   /* This is a bi-level cache, where the hash table in this slot here
@@ -105,16 +104,16 @@
      given charset's registry.  This speeds up the horrendously
      slow XListFonts() operation that needs to be done in order
      to determine an appropriate font. */
-  MARKED_SLOT (charset_font_cache_stage_1);
+  MARKED_SLOT (charset_font_cache_stage_1)
 
   /* Similar cache for stage 2, if it exists.  See objects.c. */
-  MARKED_SLOT (charset_font_cache_stage_2);
+  MARKED_SLOT (charset_font_cache_stage_2)
 #endif
 
   /* This is a bi-level cache, where the hash table in this slot here
      indexes image-instance-type masks (there are currently 6
      image-instance types and thus 64 possible masks) to key-weak hash
      tables like the one for colors. */
-  MARKED_SLOT (image_instance_cache);
+  MARKED_SLOT (image_instance_cache)
 
 #undef MARKED_SLOT
--- a/src/dialog-msw.c	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/dialog-msw.c	Sun Jan 12 11:08:22 2003 +0000
@@ -169,13 +169,11 @@
   return 0;
 }
 
-#ifdef USE_KKCC
-static const struct lrecord_description mswindows_dialog_id_description [] = {
+static const struct memory_description mswindows_dialog_id_description [] = {
   { XD_LISP_OBJECT, offsetof (struct mswindows_dialog_id, frame) },
   { XD_LISP_OBJECT, offsetof (struct mswindows_dialog_id, callbacks) },
   { XD_END }
 };
-#endif /* USE_KKCC */
 
 static Lisp_Object
 mark_mswindows_dialog_id (Lisp_Object obj)
@@ -185,19 +183,12 @@
   return data->callbacks;
 }
 
-#ifdef USE_KKCC
 DEFINE_LRECORD_IMPLEMENTATION ("mswindows-dialog-id", mswindows_dialog_id,
 			       0, /* dump-able flag */
 			       mark_mswindows_dialog_id,
 			       internal_object_printer, 0, 0, 0, 
 			       mswindows_dialog_id_description,
 			       struct mswindows_dialog_id);
-#else /* not USE_KKCC */
-DEFINE_LRECORD_IMPLEMENTATION ("mswindows-dialog-id", mswindows_dialog_id,
-			       mark_mswindows_dialog_id,
-			       internal_object_printer, 0, 0, 0, 0,
-			       struct mswindows_dialog_id);
-#endif /* not USE_KKCC */
 
 /* Dialog procedure */
 static BOOL CALLBACK 
--- a/src/dired-msw.c	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/dired-msw.c	Sun Jan 12 11:08:22 2003 +0000
@@ -147,7 +147,7 @@
   switch (mswindows_sort_method)
     {
     case MSWINDOWS_SORT_BY_NAME:
-      status = strcmp (e1->cFileName, e2->cFileName);
+      status = qxestrcmp (e1->cFileName, e2->cFileName);
       break;
     case MSWINDOWS_SORT_BY_NAME_NOCASE:
       status = qxestrcasecmp (e1->cFileName, e2->cFileName);
@@ -274,7 +274,7 @@
 	    eilwr (filename);
 	  len = eilen (filename);
 	  result = (NILP (pattern)
-		    || (0 <= re_search (bufp, eidata (filename), 
+		    || (0 <= re_search (bufp, (char *) eidata (filename), 
 					len, 0, len, 0, Qnil, 0, scache)));
 	  if (result)
 	    {
@@ -341,11 +341,11 @@
 	DEC_IBYTEPTR (end);
 	DEC_IBYTEPTR (end);
 	DEC_IBYTEPTR (end);
-	if (qxestrcasecmp (end, ".exe") == 0
-	    || qxestrcasecmp (end, ".com") == 0
-	    || qxestrcasecmp (end, ".bat") == 0
+	if (qxestrcasecmp_c (end, ".exe") == 0
+	    || qxestrcasecmp_c (end, ".com") == 0
+	    || qxestrcasecmp_c (end, ".bat") == 0
 #if 0
-	    || qxestrcasecmp (end, ".pif") == 0
+	    || qxestrcasecmp_c (end, ".pif") == 0
 #endif
 	    )
 	  is_executable = 1;
--- a/src/doprnt.c	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/doprnt.c	Sun Jan 12 11:08:22 2003 +0000
@@ -306,7 +306,7 @@
       Dynarr_add (specs, spec);
     }
 
-  RETURN_NOT_REACHED(specs) /* suppress compiler warning */
+  RETURN_NOT_REACHED(specs); /* suppress compiler warning */
 }
 
 static int
--- a/src/dumper.c	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/dumper.c	Sun Jan 12 11:08:22 2003 +0000
@@ -30,9 +30,9 @@
 #include "specifier.h"
 #include "file-coding.h"
 #include "elhash.h"
+#include "lstream.h"
 #include "sysfile.h"
 #include "console-stream.h"
-#include "dumper.h"
 
 #ifdef WIN32_NATIVE
 #include "syswindows.h"
@@ -46,17 +46,18 @@
 {
   const void *varaddress;
   Bytecount size;
-} pdump_opaque;
+  const struct memory_description *desc;
+} pdump_root_block;
 
 typedef struct
 {
-  Dynarr_declare (pdump_opaque);
-} pdump_opaque_dynarr;
+  Dynarr_declare (pdump_root_block);
+} pdump_root_block_dynarr;
 
 typedef struct
 {
   void **ptraddress;
-  const struct struct_description *desc;
+  const struct sized_memory_description *desc;
 } pdump_root_struct_ptr;
 
 typedef struct
@@ -76,22 +77,24 @@
   char * value;
 } pdump_static_pointer;
 
-static pdump_opaque_dynarr *pdump_opaques;
+static pdump_root_block_dynarr *pdump_root_blocks;
 static pdump_root_struct_ptr_dynarr *pdump_root_struct_ptrs;
-static Lisp_Object_ptr_dynarr *pdump_root_objects;
+static Lisp_Object_ptr_dynarr *pdump_root_lisp_objects;
 static Lisp_Object_ptr_dynarr *pdump_weak_object_chains;
 
-/* Mark SIZE bytes at non-heap address VARADDRESS for dumping as is,
-   without any bit-twiddling. */
+/* Mark SIZE bytes at non-heap address VARADDRESS for dumping, described
+   by DESC. */
 void
-dump_add_opaque (const void *varaddress, Bytecount size)
+dump_add_root_block (const void *varaddress, Bytecount size,
+		     const struct memory_description *desc)
 {
-  pdump_opaque info;
+  pdump_root_block info;
   info.varaddress = varaddress;
   info.size = size;
-  if (pdump_opaques == NULL)
-    pdump_opaques = Dynarr_new (pdump_opaque);
-  Dynarr_add (pdump_opaques, info);
+  info.desc = desc;
+  if (pdump_root_blocks == NULL)
+    pdump_root_blocks = Dynarr_new (pdump_root_block);
+  Dynarr_add (pdump_root_blocks, info);
 }
 
 /* Mark the struct described by DESC and pointed to by the pointer at
@@ -99,7 +102,7 @@
    All the objects reachable from this pointer will also be dumped. */
 void
 dump_add_root_struct_ptr (void *ptraddress,
-			  const struct struct_description *desc)
+			  const struct sized_memory_description *desc)
 {
   pdump_root_struct_ptr info;
   info.ptraddress = (void **) ptraddress;
@@ -112,11 +115,11 @@
 /* Mark the Lisp_Object at non-heap address VARADDRESS for dumping.
    All the objects reachable from this var will also be dumped. */
 void
-dump_add_root_object (Lisp_Object *varaddress)
+dump_add_root_lisp_object (Lisp_Object *varaddress)
 {
-  if (pdump_root_objects == NULL)
-    pdump_root_objects = Dynarr_new2 (Lisp_Object_ptr_dynarr, Lisp_Object *);
-  Dynarr_add (pdump_root_objects, varaddress);
+  if (pdump_root_lisp_objects == NULL)
+    pdump_root_lisp_objects = Dynarr_new2 (Lisp_Object_ptr_dynarr, Lisp_Object *);
+  Dynarr_add (pdump_root_lisp_objects, varaddress);
 }
 
 /* Mark the list pointed to by the Lisp_Object at VARADDRESS for dumping. */
@@ -158,7 +161,7 @@
 
 typedef struct
 {
-  const struct lrecord_description *desc;
+  const struct memory_description *desc;
   int count;
 } pdump_reloc_table;
 
@@ -189,12 +192,13 @@
 }
 
 
-/* The structure of the file
+/* The structure of the dump file looks like this:
  0		- header
 		- dumped objects
- stab_offset	- nb_root_struct_ptrs*pair(void *, adr)
-		  for pointers to structures
-		- nb_opaques*pair(void *, size) for raw bits to restore
+ stab_offset	- nb_root_struct_ptrs*struct(void *, adr)
+		  for global pointers to structures
+		- nb_root_blocks*struct(void *, size, info) for global
+		  objects to restore
 		- relocation table
 		- root lisp object address/value couples with the count
 		  preceding the list
@@ -211,7 +215,7 @@
   EMACS_UINT stab_offset;
   EMACS_UINT reloc_address;
   int nb_root_struct_ptrs;
-  int nb_opaques;
+  int nb_root_blocks;
 } pdump_header;
 
 char *pdump_start;
@@ -260,7 +264,7 @@
 typedef struct pdump_struct_list_elt
 {
   pdump_entry_list list;
-  const struct struct_description *sdesc;
+  const struct memory_description *desc;
 } pdump_struct_list_elt;
 
 typedef struct
@@ -350,11 +354,11 @@
 }
 
 static pdump_entry_list *
-pdump_get_entry_list (const struct struct_description *sdesc)
+pdump_get_entry_list (const struct memory_description *desc)
 {
   int i;
   for (i=0; i<pdump_struct_table.count; i++)
-    if (pdump_struct_table.list[i].sdesc == sdesc)
+    if (pdump_struct_table.list[i].desc == desc)
       return &pdump_struct_table.list[i].list;
 
   if (pdump_struct_table.size <= pdump_struct_table.count)
@@ -370,7 +374,7 @@
   pdump_struct_table.list[pdump_struct_table.count].list.first = 0;
   pdump_struct_table.list[pdump_struct_table.count].list.align = ALIGNOF (max_align_t);
   pdump_struct_table.list[pdump_struct_table.count].list.count = 0;
-  pdump_struct_table.list[pdump_struct_table.count].sdesc = sdesc;
+  pdump_struct_table.list[pdump_struct_table.count].desc = desc;
 
   return &pdump_struct_table.list[pdump_struct_table.count++].list;
 }
@@ -382,14 +386,14 @@
   int offset;
 } backtrace[65536];
 
-static int depth;
+static int pdump_depth;
 
-static void
+void
 pdump_backtrace (void)
 {
   int i;
   stderr_out ("pdump backtrace :\n");
-  for (i = 0; i < depth; i++)
+  for (i = 0; i < pdump_depth; i++)
     {
       if (!backtrace[i].obj)
 	stderr_out ("  - ind. (%d, %d)\n",
@@ -398,86 +402,60 @@
       else
 	{
 	  stderr_out ("  - %s (%d, %d)\n",
-		   LHEADER_IMPLEMENTATION (backtrace[i].obj)->name,
-		   backtrace[i].position,
-		   backtrace[i].offset);
+		      LHEADER_IMPLEMENTATION (backtrace[i].obj)->name,
+		      backtrace[i].position,
+		      backtrace[i].offset);
 	}
     }
 }
 
+static void
+pdump_bump_depth (void)
+{
+  int me = pdump_depth++;
+  if (me > 65536)
+    {
+      stderr_out ("Backtrace overflow, loop ?\n");
+      abort ();
+    }
+  backtrace[me].obj = 0;
+  backtrace[me].position = 0;
+  backtrace[me].offset = 0;
+}
+
 static void pdump_register_object (Lisp_Object obj);
 static void pdump_register_struct_contents (const void *data,
-					    const struct struct_description *
+					    const struct sized_memory_description *
 					    sdesc,
 					    int count);
 static void pdump_register_struct (const void *data,
-				   const struct struct_description *sdesc,
+				   const struct sized_memory_description *sdesc,
 				   int count);
 
-static EMACS_INT
-pdump_get_indirect_count (EMACS_INT code,
-			  const struct lrecord_description *idesc,
-			  const void *idata)
-{
-  EMACS_INT count;
-  const void *irdata;
-
-  int line = XD_INDIRECT_VAL (code);
-  int delta = XD_INDIRECT_DELTA (code);
-
-  irdata = ((char *)idata) + idesc[line].offset;
-  switch (idesc[line].type)
-    {
-    case XD_BYTECOUNT:
-      count = *(Bytecount *)irdata;
-      break;
-    case XD_ELEMCOUNT:
-      count = *(Elemcount *)irdata;
-      break;
-    case XD_HASHCODE:
-      count = *(Hashcode *)irdata;
-      break;
-    case XD_INT:
-      count = *(int *)irdata;
-      break;
-    case XD_LONG:
-      count = *(long *)irdata;
-      break;
-    default:
-      stderr_out ("Unsupported count type : %d (line = %d, code=%ld)\n",
-		  idesc[line].type, line, (long)code);
-      pdump_backtrace ();
-      count = 0; /* warning suppression */
-      abort ();
-    }
-  count += delta;
-  return count;
-}
-
 static void
-pdump_register_sub (const void *data, const struct lrecord_description *desc, int me)
+pdump_register_sub (const void *data, const struct memory_description *desc)
 {
   int pos;
+  int me = pdump_depth - 1;
 
- restart:
   for (pos = 0; desc[pos].type != XD_END; pos++)
     {
-      const void *rdata = (const char *)data + desc[pos].offset;
+      const struct memory_description *desc1 = &desc[pos];
+      EMACS_INT offset = lispdesc_indirect_count (desc1->offset, desc,
+						  data);
+      const void *rdata = (const char *) data + offset;
 
       backtrace[me].position = pos;
-      backtrace[me].offset = desc[pos].offset;
+      backtrace[me].offset = offset;
+
+    union_switcheroo:
 
-      switch (desc[pos].type)
+      /* If the flag says don't dump, then don't dump. */
+      if ((desc1->flags) & XD_FLAG_NO_PDUMP)
+	continue;
+
+      switch (desc1->type)
 	{
-	case XD_SPECIFIER_END:
-	  pos = 0;
-	  desc = ((const Lisp_Specifier *)data)->methods->extra_description;
-	  goto restart;
-	case XD_CODING_SYSTEM_END:
-	  pos = 0;
-	  desc =
-	    ((const Lisp_Coding_System *)data)->methods->extra_description;
-	  goto restart;
 	case XD_BYTECOUNT:
 	case XD_ELEMCOUNT:
 	case XD_HASHCODE:
@@ -488,9 +466,8 @@
 	  break;
 	case XD_OPAQUE_DATA_PTR:
 	  {
-	    EMACS_INT count = desc[pos].data1;
-	    if (XD_IS_INDIRECT (count))
-	      count = pdump_get_indirect_count (count, desc, data);
+	    EMACS_INT count = lispdesc_indirect_count (desc1->data1, desc,
+						       data);
 
 	    pdump_add_entry (&pdump_opaque_data_list,
 			     *(void **)rdata, count, 1);
@@ -498,78 +475,80 @@
 	  }
 	case XD_C_STRING:
 	  {
-	    const char *str = *(const char **)rdata;
+	    const char *str = * (const char **) rdata;
 	    if (str)
-	      pdump_add_entry (&pdump_opaque_data_list, str, strlen (str)+1, 1);
+	      pdump_add_entry (&pdump_opaque_data_list, str, strlen (str) + 1,
+			       1);
 	    break;
 	  }
 	case XD_DOC_STRING:
 	  {
-	    const char *str = *(const char **)rdata;
-	    if ((EMACS_INT)str > 0)
-	      pdump_add_entry (&pdump_opaque_data_list, str, strlen (str)+1, 1);
+	    const char *str = * (const char **) rdata;
+	    if ((EMACS_INT) str > 0)
+	      pdump_add_entry (&pdump_opaque_data_list, str, strlen (str) + 1,
+			       1);
 	    break;
 	  }
 	case XD_LISP_OBJECT:
 	  {
-	    const Lisp_Object *pobj = (const Lisp_Object *)rdata;
+	    const Lisp_Object *pobj = (const Lisp_Object *) rdata;
 
-	    assert (desc[pos].data1 == 0);
+	    assert (desc1->data1 == 0);
 
-	    backtrace[me].offset = (const char *)pobj - (const char *)data;
+	    backtrace[me].offset = (const char *) pobj - (const char *) data;
 	    pdump_register_object (*pobj);
 	    break;
 	  }
 	case XD_LISP_OBJECT_ARRAY:
 	  {
 	    int i;
-	    EMACS_INT count = desc[pos].data1;
-	    if (XD_IS_INDIRECT (count))
-	      count = pdump_get_indirect_count (count, desc, data);
+	    EMACS_INT count = lispdesc_indirect_count (desc1->data1, desc,
+						       data);
 
 	    for (i = 0; i < count; i++)
 	      {
-		const Lisp_Object *pobj = ((const Lisp_Object *)rdata) + i;
+		const Lisp_Object *pobj = ((const Lisp_Object *) rdata) + i;
 		Lisp_Object dobj = *pobj;
 
-		backtrace[me].offset = (const char *)pobj - (const char *)data;
+		backtrace[me].offset =
+		  (const char *) pobj - (const char *) data;
 		pdump_register_object (dobj);
 	      }
 	    break;
 	  }
 	case XD_STRUCT_PTR:
 	  {
-	    EMACS_INT count = desc[pos].data1;
-	    const struct struct_description *sdesc = desc[pos].data2;
+	    EMACS_INT count = lispdesc_indirect_count (desc1->data1, desc,
+						       data);
+	    const struct sized_memory_description *sdesc =
+	      lispdesc_indirect_description (data, desc1->data2);
 	    const char *dobj = *(const char **)rdata;
 	    if (dobj)
-	      {
-		if (XD_IS_INDIRECT (count))
-		  count = pdump_get_indirect_count (count, desc, data);
-
-		pdump_register_struct (dobj, sdesc, count);
-	      }
+	      pdump_register_struct (dobj, sdesc, count);
 	    break;
 	  }
 	case XD_STRUCT_ARRAY:
 	  {
-	    EMACS_INT count = desc[pos].data1;
-	    const struct struct_description *sdesc = desc[pos].data2;
-
-	    if (XD_IS_INDIRECT (count))
-	      count = pdump_get_indirect_count (count, desc, data);
+	    EMACS_INT count = lispdesc_indirect_count (desc1->data1, desc,
+						       data);
+	    const struct sized_memory_description *sdesc =
+	      lispdesc_indirect_description (data, desc1->data2);
 
 	    pdump_register_struct_contents (rdata, sdesc, count);
 	    break;
 	  }
 	case XD_UNION:
-	  abort (); /* #### IMPLEMENT ME!  NEEDED FOR UNICODE SUPPORT */
+	case XD_UNION_DYNAMIC_SIZE:
+	  desc1 = lispdesc_process_xd_union (desc1, desc, data);
+	  if (desc1)
+	    goto union_switcheroo;
+	  break;
 
 	default:
-	  stderr_out ("Unsupported dump type : %d\n", desc[pos].type);
+	  stderr_out ("Unsupported dump type : %d\n", desc1->type);
 	  pdump_backtrace ();
 	  abort ();
-	};
+	}
     }
 }
 
@@ -591,31 +570,15 @@
 
   imp = LHEADER_IMPLEMENTATION (objh);
 
-#ifdef USE_KKCC
   if (imp->description
-      && RECORD_DUMPABLE(objh))
-#else /* not USE_KKCC */
-  if (imp->description)
-#endif /* not USE_KKCC */    
+      && RECORD_DUMPABLE (objh))
     {
-      int me = depth++;
-      if (me > 65536)
-	{
-	  stderr_out ("Backtrace overflow, loop ?\n");
-	  abort ();
-	}
-      backtrace[me].obj = objh;
-      backtrace[me].position = 0;
-      backtrace[me].offset = 0;
-
+      pdump_bump_depth ();
+      backtrace[pdump_depth - 1].obj = objh;
       pdump_add_entry (pdump_object_table + objh->type,
-		       objh,
-		       imp->static_size ?
-		       imp->static_size :
-		       imp->size_in_bytes_method (objh),
-		       1);
-      pdump_register_sub (objh, imp->description, me);
-      --depth;
+		       objh, detagged_lisp_object_size (objh), 1);
+      pdump_register_sub (objh, imp->description);
+      --pdump_depth;
     }
   else
     {
@@ -625,144 +588,6 @@
     }
 }
 
-/* Return the size of the memory block (NOT necessarily a structure!) 
-   described by SDESC and pointed to by OBJ.  If SDESC records an
-   explicit size (i.e. non-zero), it is simply returned; otherwise,
-   the size is calculated by the maximum offset and the size of the
-   object at that offset, rounded up to the maximum alignment.  In
-   this case, we may need the object, for example when retrieving an
-   "indirect count" of an inlined array (the count is not constant,
-   but is specified by one of the elements of the memory block). (It
-   is generally not a problem if we return an overly large size -- we
-   will simply end up reserving more space than necessary; but if the
-   size is too small we could be in serious trouble, in particular
-   with nested inlined structures, where there may be alignment
-   padding in the middle of a block. #### In fact there is an (at
-   least theoretical) problem with an overly large size -- we may
-   trigger a protection fault when reading from invalid memory.  We
-   need to handle this -- perhaps in a stupid but dependable way,
-   i.e. by trapping SIGSEGV and SIGBUS.) */
-
-static Bytecount
-pdump_structure_size (const void *obj, const struct struct_description *sdesc)
-{
-  int max_offset = -1;
-  int max_offset_pos = -1;
-  int size_at_max = 0;
-  int pos;
-  const struct lrecord_description *desc;
-  void *rdata;
-
-  if (sdesc->size)
-    return sdesc->size;
-
-  desc = sdesc->description;
-
-  for (pos = 0; desc[pos].type != XD_END; pos++)
-    {
-      if (desc[pos].offset == max_offset)
-	{
-	  stderr_out ("Two relocatable elements at same offset?\n");
-	  abort ();
-	}
-      else if (desc[pos].offset > max_offset)
-	{
-	  max_offset = desc[pos].offset;
-	  max_offset_pos = pos;
-	}
-    }
-
-  if (max_offset_pos < 0)
-    return 0;
-
-  pos = max_offset_pos;
-  rdata = (char *) obj + desc[pos].offset;
-
-  switch (desc[pos].type)
-    {
-    case XD_LISP_OBJECT_ARRAY:
-      {
-	EMACS_INT val = desc[pos].data1;
-	if (XD_IS_INDIRECT (val))
-	  val = pdump_get_indirect_count (val, desc, obj);
-	size_at_max = val * sizeof (Lisp_Object);
-	break;
-      }
-    case XD_LISP_OBJECT:
-    case XD_LO_LINK:
-      size_at_max = sizeof (Lisp_Object);
-      break;
-    case XD_OPAQUE_PTR:
-      size_at_max = sizeof (void *);
-      break;
-    case XD_STRUCT_PTR:
-      {
-	EMACS_INT val = desc[pos].data1;
-	if (XD_IS_INDIRECT (val))
-	  val = pdump_get_indirect_count (val, desc, obj);
-	size_at_max = val * sizeof (void *);
-	break;
-      }
-      break;
-    case XD_STRUCT_ARRAY:
-      {
-	EMACS_INT val = desc[pos].data1;
-
-	if (XD_IS_INDIRECT (val))
-	  val = pdump_get_indirect_count (val, desc, obj);
-	    
-	size_at_max = val * pdump_structure_size (rdata, desc[pos].data2);
-	break;
-      }
-      break;
-    case XD_OPAQUE_DATA_PTR:
-      size_at_max = sizeof (void *);
-      break;
-    case XD_UNION:
-      abort ();			/* #### IMPLEMENT ME!  NEEDED FOR UNICODE
-				   SUPPORT */
-      break;
-    case XD_C_STRING:
-      size_at_max = sizeof (void *);
-      break;
-    case XD_DOC_STRING:
-      size_at_max = sizeof (void *);
-      break;
-    case XD_INT_RESET:
-      size_at_max = sizeof (int);
-      break;
-    case XD_BYTECOUNT:
-      size_at_max = sizeof (Bytecount);
-      break;
-    case XD_ELEMCOUNT:
-      size_at_max = sizeof (Elemcount);
-      break;
-    case XD_HASHCODE:
-      size_at_max = sizeof (Hashcode);
-      break;
-    case XD_INT:
-      size_at_max = sizeof (int);
-      break;
-    case XD_LONG:
-      size_at_max = sizeof (long);
-      break;
-    case XD_SPECIFIER_END:
-    case XD_CODING_SYSTEM_END:
-      stderr_out
-	("Should not be seeing XD_SPECIFIER_END or\n"
-	 "XD_CODING_SYSTEM_END outside of struct Lisp_Specifier\n"
-	 "and struct Lisp_Coding_System.\n");
-      abort ();
-    default:
-      stderr_out ("Unsupported dump type : %d\n", desc[pos].type);
-      abort ();
-    }
-
-  /* We have no way of knowing the required alignment for this structure,
-     so just max it maximally aligned. */
-  return MAX_ALIGN_SIZE (max_offset + size_at_max);
-}
-
 /* Register the referenced objects in the array of COUNT objects of
    located at DATA; each object is described by SDESC.  "Object" here
    simply means any block of memory; it need not actually be a C
@@ -776,32 +601,20 @@
 
 static void
 pdump_register_struct_contents (const void *data,
-				const struct struct_description *sdesc,
+				const struct sized_memory_description *sdesc,
 				int count)
 
 {
-  int me = depth++;
   int i;
   Bytecount elsize;
 
-  if (me>65536)
-    {
-      stderr_out ("Backtrace overflow, loop ?\n");
-      abort ();
-    }
-  backtrace[me].obj = 0;
-  backtrace[me].position = 0;
-  backtrace[me].offset = 0;
-
-  elsize = pdump_structure_size (data, sdesc);
-
+  pdump_bump_depth ();
+  elsize = lispdesc_structure_size (data, sdesc);
   for (i = 0; i < count; i++)
     {
-      pdump_register_sub (((char *) data) + elsize * i,
-			  sdesc->description,
-			  me);
+      pdump_register_sub (((char *) data) + elsize * i, sdesc->description);
     }
-  --depth;
+  --pdump_depth;
 }
 
 /* Register the array of COUNT objects of located at DATA; each object is
@@ -814,47 +627,47 @@
 
 static void
 pdump_register_struct (const void *data,
-		       const struct struct_description *sdesc,
+		       const struct sized_memory_description *sdesc,
 		       int count)
 {
   if (data && !pdump_get_entry (data))
     {
-      pdump_add_entry (pdump_get_entry_list (sdesc), data,
-		       pdump_structure_size (data, sdesc), count);
+      pdump_add_entry (pdump_get_entry_list (sdesc->description), data,
+		       lispdesc_structure_size (data, sdesc), count);
 
       pdump_register_struct_contents (data, sdesc, count);
     }
 }
 
-/* Store the already-calculated new pointer offsets for all pointers
-   in the COUNT contiguous blocks of memory, each described by DESC
-   and of size SIZE, whose original is located at ORIG_DATA and the
-   modifiable copy at DATA.
+/* Store the already-calculated new pointer offsets for all pointers in the
+   COUNT contiguous blocks of memory, each described by DESC and of size
+   SIZE, whose original is located at ORIG_DATA and the modifiable copy at
+   DATA.  We examine the description to figure out where the pointers are,
+   and then look up the replacement values using pdump_get_entry().
 
-   This is done just before writing the modified block of memory to
-   the dump file.  The new pointer offsets have been carefully
-   calculated so that the data being pointed gets written at that
-   offset in the dump file.  That way, the dump file is a correct
-   memory image except perhaps for a constant that needs to be added
-   to all pointers. (#### In fact, we SHOULD be starting up a dumped
-   XEmacs, seeing where the dumped file gets loaded into memory, and
-   then rewriting the dumped file after relocating all the pointers
-   relative to this memory location.  That way, if the file gets
-   loaded again at the same location, which will be common, we don't
-   have to do any relocating, which is both faster at startup and
+   This is done just before writing the modified block of memory to the
+   dump file.  The new pointer offsets have been carefully calculated so
+   that the data being pointed gets written at that offset in the dump
+   file.  That way, the dump file is a correct memory image except perhaps
+   for a constant that needs to be added to all pointers. (#### In fact, we
+   SHOULD be starting up a dumped XEmacs, seeing where the dumped file gets
+   loaded into memory, and then rewriting the dumped file after relocating
+   all the pointers relative to this memory location.  That way, if the
+   file gets loaded again at the same location, which will be common, we
+   don't have to do any relocating, which is both faster at startup and
    allows the read-only part of the dumped data to be shared read-only
    between different invocations of XEmacs.)
 
    #### Do we distinguish between read-only and writable dumped data?
    Should we?  It's tricky because the dumped data, once loaded again,
-   cannot really be free()d or garbage collected since it's all stored
-   in one contiguous block of data with no malloc() headers, and we
-   don't keep track of the pointers used internally in malloc() and
-   the Lisp allocator to track allocated blocks of memory. */
+   cannot really be free()d or garbage collected since it's all stored in
+   one contiguous block of data with no malloc() headers, and we don't keep
+   track of the pointers used internally in malloc() and the Lisp allocator
+   to track allocated blocks of memory. */
 
 static void
 pdump_store_new_pointer_offsets (int count, void *data, const void *orig_data,
-				 const struct lrecord_description *desc,
+				 const struct memory_description *desc,
 				 int size)
 {
   int pos, i;
@@ -863,22 +676,22 @@
     {
       /* CUR points to the beginning of each block in the new data. */
       char *cur = ((char *)data) + i*size;
-    restart:
       /* Scan each line of the description for relocatable pointers */
       for (pos = 0; desc[pos].type != XD_END; pos++)
 	{
 	  /* RDATA points to the beginning of each element in the new data. */
-	  void *rdata = cur + desc[pos].offset;
-	  switch (desc[pos].type)
+	  const struct memory_description *desc1 = &desc[pos];
+	  /* #### Change ORIG_DATA to DATA.  See below. */
+	  void *rdata = cur + lispdesc_indirect_count (desc1->offset, desc,
+						       orig_data);
+	union_switcheroo:
+
+	  /* If the flag says don't dump, then don't dump. */
+	  if ((desc1->flags) & XD_FLAG_NO_PDUMP)
+	    continue;
+
+	  switch (desc1->type)
 	    {
-	    case XD_SPECIFIER_END:
-	      desc = ((const Lisp_Specifier *)(orig_data))->
-		methods->extra_description;
-	      goto restart;
-	    case XD_CODING_SYSTEM_END:
-	      desc = ((const Lisp_Coding_System *)(orig_data))->
-		methods->extra_description;
-	      goto restart;
 	    case XD_BYTECOUNT:
 	    case XD_ELEMCOUNT:
 	    case XD_HASHCODE:
@@ -887,9 +700,8 @@
 	      break;
 	    case XD_INT_RESET:
 	      {
-		EMACS_INT val = desc[pos].data1;
-		if (XD_IS_INDIRECT (val))
-		  val = pdump_get_indirect_count (val, desc, orig_data);
+		EMACS_INT val = lispdesc_indirect_count (desc1->data1, desc,
+							 orig_data);
 		* (int *) rdata = val;
 		break;
 	      }
@@ -914,12 +726,23 @@
                    chain will always be a referenced object.) */
 		Lisp_Object obj = * (Lisp_Object *) rdata;
 		pdump_entry_list_elt *elt1;
+		/* #### Figure out how to handle indirect offsets here.
+		   #### In general, when computing indirect counts, do we
+		   really need to use the orig_data pointer?  Why not just
+		   use the new stuff?
+
+		   No, we don't usually need orig_data.  We only need it
+		   when fetching pointers out of the data, not integers.
+		   This currently occurs only with description maps.  We
+		   should change the other places to DATA to emphasize
+		   this. */
+		assert (!XD_IS_INDIRECT (desc1->offset));
 		for (;;)
 		  {
 		    elt1 = pdump_get_entry (XRECORD_LHEADER (obj));
 		    if (elt1)
 		      break;
-		    obj = * (Lisp_Object *) (desc[pos].offset +
+		    obj = * (Lisp_Object *) (desc1->offset +
 					     (char *)(XRECORD_LHEADER (obj)));
 		  }
 		* (EMACS_INT *) rdata = elt1->save_offset;
@@ -929,7 +752,7 @@
 	      {
 		Lisp_Object *pobj = (Lisp_Object *) rdata;
 
-		assert (desc[pos].data1 == 0);
+		assert (desc1->data1 == 0);
 
 		if (POINTER_TYPE_P (XTYPE (*pobj)) && XRECORD_LHEADER (*pobj))
 		  * (EMACS_INT *) pobj =
@@ -938,10 +761,9 @@
 	      }
 	    case XD_LISP_OBJECT_ARRAY:
 	      {
-		EMACS_INT num = desc[pos].data1;
+		EMACS_INT num = lispdesc_indirect_count (desc1->data1, desc,
+							 orig_data);
 		int j;
-		if (XD_IS_INDIRECT (num))
-		  num = pdump_get_indirect_count (num, desc, orig_data);
 
 		for (j = 0; j < num; j++)
 		  {
@@ -963,25 +785,29 @@
 	      }
 	    case XD_STRUCT_ARRAY:
 	      {
-		EMACS_INT num = desc[pos].data1;
-		if (XD_IS_INDIRECT (num))
-		  num = pdump_get_indirect_count (num, desc, orig_data);
+		EMACS_INT num = lispdesc_indirect_count (desc1->data1, desc,
+							 orig_data);
+		const struct sized_memory_description *sdesc =
+		  lispdesc_indirect_description (orig_data, desc1->data2);
 
 		pdump_store_new_pointer_offsets
 		  (num, rdata,
 		   ((char *) rdata - (char *) data) + (char *) orig_data,
-		   desc[pos].data2->description,
-		   pdump_structure_size
+		   sdesc->description,
+		   lispdesc_structure_size
 		   (((char *) rdata - (char *) data) + (char *) orig_data,
-		    desc[pos].data2));
+		    sdesc));
 		break;
 	      }
 	    case XD_UNION:
-	      abort ();		/* #### IMPLEMENT ME!  NEEDED FOR UNICODE
-				   SUPPORT */
+	    case XD_UNION_DYNAMIC_SIZE:
+	      desc1 = lispdesc_process_xd_union (desc1, desc, orig_data);
+	      if (desc1)
+		goto union_switcheroo;
+	      break;
 
 	    default:
-	      stderr_out ("Unsupported dump type : %d\n", desc[pos].type);
+	      stderr_out ("Unsupported dump type : %d\n", desc1->type);
 	      abort ();
 	    }
 	}
@@ -997,7 +823,7 @@
 
 static void
 pdump_dump_data (pdump_entry_list_elt *elt,
-		 const struct lrecord_description *desc)
+		 const struct memory_description *desc)
 {
   Bytecount size = elt->size;
   int count = elt->count;
@@ -1013,32 +839,33 @@
 }
 
 /* Relocate a single memory block at DATA, described by DESC, from its
-   assumed load location to its actual one by adding DELTA to all
-   pointers in the block.  Does not recursively relocate any other
-   memory blocks pointed to. (We already have a list of all memory
-   blocks in the dump file.) */
+   assumed load location to its actual one by adding DELTA to all pointers
+   in the block.  Does not recursively relocate any other memory blocks
+   pointed to. (We already have a list of all memory blocks in the dump
+   file.)  This is used once the dump data has been loaded back in, both
+   for blocks sitting in the dumped data and in global data objects whose
+   contents have been restored from the dumped data. */
 
 static void
 pdump_reloc_one (void *data, EMACS_INT delta,
-		 const struct lrecord_description *desc)
+		 const struct memory_description *desc)
 {
   int pos;
 
- restart:
   for (pos = 0; desc[pos].type != XD_END; pos++)
     {
-      void *rdata = (char *)data + desc[pos].offset;
-      switch (desc[pos].type)
+      const struct memory_description *desc1 = &desc[pos];
+      void *rdata = (char *) data + lispdesc_indirect_count (desc1->offset,
+							     desc, data);
+
+    union_switcheroo:
+
+      /* If the flag says don't dump, then don't dump. */
+      if ((desc1->flags) & XD_FLAG_NO_PDUMP)
+	continue;
+
+      switch (desc1->type)
 	{
-	case XD_SPECIFIER_END:
-	  pos = 0;
-	  desc = ((const Lisp_Specifier *)data)->methods->extra_description;
-	  goto restart;
-	case XD_CODING_SYSTEM_END:
-	  pos = 0;
-	  desc =
-	    ((const Lisp_Coding_System *)data)->methods->extra_description;
-	  goto restart;
 	case XD_BYTECOUNT:
 	case XD_ELEMCOUNT:
 	case XD_HASHCODE:
@@ -1060,7 +887,7 @@
 	  {
 	    Lisp_Object *pobj = (Lisp_Object *) rdata;
 
-	    assert (desc[pos].data1 == 0);
+	    assert (desc1->data1 == 0);
 
 	    if (POINTER_TYPE_P (XTYPE (*pobj))
 		&& ! EQ (*pobj, Qnull_pointer))
@@ -1070,10 +897,9 @@
 	  }
 	case XD_LISP_OBJECT_ARRAY:
 	  {
-	    EMACS_INT num = desc[pos].data1;
+	    EMACS_INT num = lispdesc_indirect_count (desc1->data1, desc,
+						     data);
 	    int j;
-	    if (XD_IS_INDIRECT (num))
-	      num = pdump_get_indirect_count (num, desc, data);
 
 	    for (j=0; j<num; j++)
 	      {
@@ -1094,13 +920,13 @@
 	  }
 	case XD_STRUCT_ARRAY:
 	  {
-	    EMACS_INT num = desc[pos].data1;
+	    EMACS_INT num = lispdesc_indirect_count (desc1->data1, desc,
+						     data);
 	    int j;
-	    const struct struct_description *sdesc = desc[pos].data2;
-	    Bytecount size = pdump_structure_size (rdata, sdesc);
+	    const struct sized_memory_description *sdesc =
+	      lispdesc_indirect_description (data, desc1->data2);
+	    Bytecount size = lispdesc_structure_size (rdata, sdesc);
 
-	    if (XD_IS_INDIRECT (num))
-	      num = pdump_get_indirect_count (num, desc, data);
 	    /* Note: We are recursing over data in the block itself */
 	    for (j = 0; j < num; j++)
 	      pdump_reloc_one ((char *) rdata + j * size, delta,
@@ -1108,19 +934,23 @@
 
 	    break;
 	  }
+	case XD_UNION:
+	case XD_UNION_DYNAMIC_SIZE:
+	  desc1 = lispdesc_process_xd_union (desc1, desc, data);
+	  if (desc1)
+	    goto union_switcheroo;
+	  break;
 
-	case XD_UNION:
-	  abort (); /* #### IMPLEMENT ME!  NEEDED FOR UNICODE SUPPORT */
 	default:
-	  stderr_out ("Unsupported dump type : %d\n", desc[pos].type);
+	  stderr_out ("Unsupported dump type : %d\n", desc1->type);
 	  abort ();
-	};
+	}
     }
 }
 
 static void
 pdump_allocate_offset (pdump_entry_list_elt *elt,
-		       const struct lrecord_description *desc)
+		       const struct memory_description *desc)
 {
   Bytecount size = elt->count * elt->size;
   elt->save_offset = cur_offset;
@@ -1131,7 +961,7 @@
 
 static void
 pdump_scan_by_alignment (void (*f)(pdump_entry_list_elt *,
-				   const struct lrecord_description *))
+				   const struct memory_description *))
 {
   int align;
 
@@ -1150,7 +980,7 @@
 	  pdump_struct_list_elt list = pdump_struct_table.list[i];
 	  if (list.list.align == align)
 	    for (elt = list.list.first; elt; elt = elt->next)
-	      f (elt, list.sdesc->description);
+	      f (elt, list.desc);
 	}
 
       for (elt = pdump_opaque_data_list.first; elt; elt = elt->next)
@@ -1175,13 +1005,13 @@
 }
 
 static void
-pdump_dump_opaques (void)
+pdump_dump_root_blocks (void)
 {
   int i;
-  for (i = 0; i < Dynarr_length (pdump_opaques); i++)
+  for (i = 0; i < Dynarr_length (pdump_root_blocks); i++)
     {
-      pdump_opaque *info = Dynarr_atp (pdump_opaques, i);
-      PDUMP_WRITE_ALIGNED (pdump_opaque, *info);
+      pdump_root_block *info = Dynarr_atp (pdump_root_blocks, i);
+      PDUMP_WRITE_ALIGNED (pdump_root_block, *info);
       retry_fwrite (info->varaddress, info->size, 1, pdump_out);
     }
 }
@@ -1216,7 +1046,7 @@
   for (i=0; i<pdump_struct_table.count; i++)
     {
       elt = pdump_struct_table.list[i].list.first;
-      rt.desc = pdump_struct_table.list[i].sdesc->description;
+      rt.desc = pdump_struct_table.list[i].desc;
       rt.count = pdump_struct_table.list[i].list.count;
       PDUMP_WRITE_ALIGNED (pdump_reloc_table, rt);
       while (elt)
@@ -1237,19 +1067,19 @@
 }
 
 static void
-pdump_dump_root_objects (void)
+pdump_dump_root_lisp_objects (void)
 {
-  Elemcount count = (Dynarr_length (pdump_root_objects) +
+  Elemcount count = (Dynarr_length (pdump_root_lisp_objects) +
 			 Dynarr_length (pdump_weak_object_chains));
   Elemcount i;
 
   PDUMP_WRITE_ALIGNED (Elemcount, count);
   PDUMP_ALIGN_OUTPUT (pdump_static_Lisp_Object);
 
-  for (i = 0; i < Dynarr_length (pdump_root_objects); i++)
+  for (i = 0; i < Dynarr_length (pdump_root_lisp_objects); i++)
     {
       pdump_static_Lisp_Object obj;
-      obj.address = Dynarr_at (pdump_root_objects, i);
+      obj.address = Dynarr_at (pdump_root_lisp_objects, i);
       obj.value   = * obj.address;
 
       if (POINTER_TYPE_P (XTYPE (obj.value)))
@@ -1270,7 +1100,7 @@
 
       for (;;)
 	{
-	  const struct lrecord_description *desc;
+	  const struct memory_description *desc;
 	  int pos;
 	  elt = pdump_get_entry (XRECORD_LHEADER (obj.value));
 	  if (elt)
@@ -1279,7 +1109,11 @@
 	  for (pos = 0; desc[pos].type != XD_LO_LINK; pos++)
 	    assert (desc[pos].type != XD_END);
 
-	  obj.value = *(Lisp_Object *)(desc[pos].offset + (char *)(XRECORD_LHEADER (obj.value)));
+	  /* #### Figure out how to handle indirect offsets here. */
+	  assert (!XD_IS_INDIRECT (desc[pos].offset));
+	  obj.value =
+	    * (Lisp_Object *) (desc[pos].offset +
+			       (char *) (XRECORD_LHEADER (obj.value)));
 	}
       obj.value = wrap_pointer_1 ((void *) elt->save_offset);
 
@@ -1295,6 +1129,8 @@
   int none;
   pdump_header header;
 
+  in_pdump = 1;
+
   pdump_object_table = xnew_array (pdump_entry_list, lrecord_type_count);
   pdump_alert_undump_object = xnew_array (int, lrecord_type_count);
 
@@ -1312,7 +1148,8 @@
   t_device  = Vterminal_device;  Vterminal_device  = Qnil;
 
   dump_add_opaque (&lrecord_implementations_table,
-		   lrecord_type_count * sizeof (lrecord_implementations_table[0]));
+		   lrecord_type_count *
+		   sizeof (lrecord_implementations_table[0]));
   dump_add_opaque (&lrecord_markers,
 		   lrecord_type_count * sizeof (lrecord_markers[0]));
 
@@ -1331,10 +1168,10 @@
   pdump_opaque_data_list.first = 0;
   pdump_opaque_data_list.align = ALIGNOF (max_align_t);
   pdump_opaque_data_list.count = 0;
-  depth = 0;
+  pdump_depth = 0;
 
-  for (i=0; i<Dynarr_length (pdump_root_objects); i++)
-    pdump_register_object (* Dynarr_at (pdump_root_objects, i));
+  for (i = 0; i < Dynarr_length (pdump_root_lisp_objects); i++)
+    pdump_register_object (* Dynarr_at (pdump_root_lisp_objects, i));
 
   none = 1;
   for (i=0; i<lrecord_type_count; i++)
@@ -1343,10 +1180,14 @@
 	if (none)
 	  printf ("Undumpable types list :\n");
 	none = 0;
-	printf ("  - %s (%d)\n", lrecord_implementations_table[i]->name, pdump_alert_undump_object[i]);
+	printf ("  - %s (%d)\n", lrecord_implementations_table[i]->name,
+		pdump_alert_undump_object[i]);
       }
   if (!none)
-    return;
+    {
+      in_pdump = 0;
+      return;
+    }
 
   for (i=0; i<Dynarr_length (pdump_root_struct_ptrs); i++)
     {
@@ -1358,7 +1199,7 @@
   header.id = dump_id;
   header.reloc_address = 0;
   header.nb_root_struct_ptrs = Dynarr_length (pdump_root_struct_ptrs);
-  header.nb_opaques = Dynarr_length (pdump_opaques);
+  header.nb_root_blocks = Dynarr_length (pdump_root_blocks);
 
   cur_offset = MAX_ALIGN_SIZE (sizeof (header));
   max_size = 0;
@@ -1386,9 +1227,9 @@
   fseek (pdump_out, header.stab_offset, SEEK_SET);
 
   pdump_dump_root_struct_ptrs ();
-  pdump_dump_opaques ();
+  pdump_dump_root_blocks ();
   pdump_dump_rtables ();
-  pdump_dump_root_objects ();
+  pdump_dump_root_lisp_objects ();
 
   retry_fclose (pdump_out);
   retry_close (pdump_fd);
@@ -1400,6 +1241,7 @@
   Vterminal_console = t_console;
   Vterminal_frame   = t_frame;
   Vterminal_device  = t_device;
+  in_pdump = 0;
 }
 
 static int
@@ -1420,30 +1262,32 @@
   char *p;
   EMACS_INT delta;
   EMACS_INT count;
-  pdump_header *header = (pdump_header *)pdump_start;
+  pdump_header *header = (pdump_header *) pdump_start;
 
   pdump_end = pdump_start + pdump_length;
 
-  delta = ((EMACS_INT)pdump_start) - header->reloc_address;
+  delta = ((EMACS_INT) pdump_start) - header->reloc_address;
   p = pdump_start + header->stab_offset;
 
   /* Put back the pdump_root_struct_ptrs */
   p = (char *) ALIGN_PTR (p, pdump_static_pointer);
-  for (i=0; i<header->nb_root_struct_ptrs; i++)
+  for (i = 0; i < header->nb_root_struct_ptrs; i++)
     {
       pdump_static_pointer ptr = PDUMP_READ (p, pdump_static_pointer);
       (* ptr.address) = ptr.value + delta;
     }
 
-  /* Put back the pdump_opaques */
-  for (i=0; i<header->nb_opaques; i++)
+  /* Put back the pdump_root_blocks and relocate */
+  for (i = 0; i < header->nb_root_blocks; i++)
     {
-      pdump_opaque info = PDUMP_READ_ALIGNED (p, pdump_opaque);
-      memcpy ((void*)info.varaddress, p, info.size);
+      pdump_root_block info = PDUMP_READ_ALIGNED (p, pdump_root_block);
+      memcpy ((void *) info.varaddress, p, info.size);
+      if (info.desc)
+	pdump_reloc_one ((void *) info.varaddress, delta, info.desc);
       p += info.size;
     }
 
-  /* Do the relocations */
+  /* Relocate the heap objects */
   pdump_rt_list = p;
   count = 2;
   for (;;)
@@ -1452,19 +1296,19 @@
       p = (char *) ALIGN_PTR (p, char *);
       if (rt.desc)
 	{
-	  char **reloc = (char **)p;
-	  for (i=0; i < rt.count; i++)
+	  char **reloc = (char **) p;
+	  for (i = 0; i < rt.count; i++)
 	    {
 	      reloc[i] += delta;
 	      pdump_reloc_one (reloc[i], delta, rt.desc);
 	    }
 	  p += rt.count * sizeof (char *);
-	} else
-	  if (!(--count))
-	    break;
+	}
+      else if (!(--count))
+	  break;
     }
 
-  /* Put the pdump_root_objects variables in place */
+  /* Put the pdump_root_lisp_objects variables in place */
   i = PDUMP_READ_ALIGNED (p, Elemcount);
   p = (char *) ALIGN_PTR (p, pdump_static_Lisp_Object);
   while (i--)
@@ -1488,11 +1332,12 @@
 	break;
       if (rt.desc == hash_table_description)
 	{
-	  for (i=0; i < rt.count; i++)
+	  for (i = 0; i < rt.count; i++)
 	    pdump_reorganize_hash_table (PDUMP_READ (p, Lisp_Object));
 	  break;
-	} else
-	  p += sizeof (Lisp_Object) * rt.count;
+	}
+      else
+	p += sizeof (Lisp_Object) * rt.count;
     }
 
   return 1;
@@ -1532,11 +1377,11 @@
   if (pdump_hMap == INVALID_HANDLE_VALUE)
     return 0;
 
-  pdump_start = MapViewOfFile (pdump_hMap,
-			       FILE_MAP_COPY, /* Copy on write */
-			       0,	      /* Start at zero */
-			       0,
-			       0);	      /* Map all of it */
+  pdump_start = (char *) MapViewOfFile (pdump_hMap,
+					FILE_MAP_COPY, /* Copy on write */
+					0,	      /* Start at zero */
+					0,
+					0);	      /* Map all of it */
   pdump_free = pdump_file_unmap;
   return 1;
 }
@@ -1571,11 +1416,11 @@
     return 0;
 
   /* Found it, use the data in the resource */
-  hResLoad = LoadResource (NULL, hRes);
+  hResLoad = (HRSRC) LoadResource (NULL, hRes);
   if (hResLoad == NULL)
     return 0;
 
-  pdump_start = LockResource (hResLoad);
+  pdump_start = (char *) LockResource (hResLoad);
   if (pdump_start == NULL)
     return 0;
 
@@ -1696,6 +1541,7 @@
   Extbyte *w;
   const Extbyte *dir, *p;
 
+  in_pdump = 1;
   dir = argv0;
   if (dir[0] == '-')
     {
@@ -1753,6 +1599,7 @@
   if (pdump_file_try (exe_path))
     {
       pdump_load_finish ();
+      in_pdump = 0;
       return 1;
     }
 
@@ -1762,11 +1609,13 @@
       if (pdump_load_check ())
 	{
 	  pdump_load_finish ();
+	  in_pdump = 0;
 	  return 1;
 	}
       pdump_free ();
     }
 #endif
 
+  in_pdump = 0;
   return 0;
 }
--- a/src/editfns.c	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/editfns.c	Sun Jan 12 11:08:22 2003 +0000
@@ -361,8 +361,8 @@
 
   /* Free all the junk we allocated, so that a `save-excursion' comes
      for free in terms of GC junk. */
-  free_marker (XMARKER (XCAR (info)));
-  free_marker (XMARKER (XCDR (info)));
+  free_marker (XCAR (info));
+  free_marker (XCDR (info));
   free_cons (info);
   return Qnil;
 }
@@ -726,7 +726,7 @@
 	  return (Ibyte *) (pw ? pw->pw_name : "unknown");
 #else
 	  /* For all but Cygwin return NULL (nil) */
-	  return pw ? pw->pw_name : NULL;
+          return pw ? (Ibyte *) pw->pw_name : NULL;
 #endif
 	}
     }
@@ -2104,8 +2104,8 @@
 
   /* Free all the junk we allocated, so that a `save-restriction' comes
      for free in terms of GC junk. */
-  free_marker (XMARKER (XCAR (markers)));
-  free_marker (XMARKER (XCDR (markers)));
+  free_marker (XCAR (markers));
+  free_marker (XCDR (markers));
   free_cons (markers);
   free_cons (data);
 
--- a/src/elhash.c	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/elhash.c	Sun Jan 12 11:08:22 2003 +0000
@@ -73,11 +73,11 @@
 static Lisp_Object Qweak, Qkey_weak, Qvalue_weak, Qkey_or_value_weak;
 static Lisp_Object Qnon_weak, Q_type;
 
-typedef struct hentry
+typedef struct htentry
 {
   Lisp_Object key;
   Lisp_Object value;
-} hentry;
+} htentry;
 
 struct Lisp_Hash_Table
 {
@@ -90,16 +90,16 @@
   Elemcount golden_ratio;
   hash_table_hash_function_t hash_function;
   hash_table_test_function_t test_function;
-  hentry *hentries;
+  htentry *hentries;
   enum hash_table_weakness weakness;
   Lisp_Object next_weak;     /* Used to chain together all of the weak
 			        hash tables.  Don't mark through this. */
 };
 
-#define HENTRY_CLEAR_P(hentry) ((*(EMACS_UINT*)(&((hentry)->key))) == 0)
-#define CLEAR_HENTRY(hentry)   \
-  ((*(EMACS_UINT*)(&((hentry)->key)))   = 0, \
-   (*(EMACS_UINT*)(&((hentry)->value))) = 0)
+#define HTENTRY_CLEAR_P(htentry) ((*(EMACS_UINT*)(&((htentry)->key))) == 0)
+#define CLEAR_HTENTRY(htentry)   \
+  ((*(EMACS_UINT*)(&((htentry)->key)))   = 0, \
+   (*(EMACS_UINT*)(&((htentry)->value))) = 0)
 
 #define HASH_TABLE_DEFAULT_SIZE 16
 #define HASH_TABLE_DEFAULT_REHASH_SIZE 1.3
@@ -115,9 +115,9 @@
 
 #define LINEAR_PROBING_LOOP(probe, entries, size)		\
   for (;							\
-       !HENTRY_CLEAR_P (probe) ||				\
+       !HTENTRY_CLEAR_P (probe) ||				\
 	 (probe == entries + size ?				\
-	  (probe = entries, !HENTRY_CLEAR_P (probe)) : 0);	\
+	  (probe = entries, !HTENTRY_CLEAR_P (probe)) : 0);	\
        probe++)
 
 #ifdef ERROR_CHECK_STRUCTURES
@@ -128,7 +128,7 @@
   assert (ht->count <= ht->rehash_count);
   assert (ht->rehash_count < ht->size);
   assert ((double) ht->count * ht->rehash_threshold - 1 <= (double) ht->rehash_count);
-  assert (HENTRY_CLEAR_P (ht->hentries + ht->size));
+  assert (HTENTRY_CLEAR_P (ht->hentries + ht->size));
 }
 #else
 #define check_hash_table_invariants(ht)
@@ -223,10 +223,10 @@
      and mark or remove them as necessary).  */
   if (ht->weakness == HASH_TABLE_NON_WEAK)
     {
-      hentry *e, *sentinel;
+      htentry *e, *sentinel;
 
       for (e = ht->hentries, sentinel = e + ht->size; e < sentinel; e++)
-	if (!HENTRY_CLEAR_P (e))
+	if (!HTENTRY_CLEAR_P (e))
 	  {
 	    mark_object (e->key);
 	    mark_object (e->value);
@@ -250,7 +250,7 @@
 {
   Lisp_Hash_Table *ht1 = XHASH_TABLE (hash_table1);
   Lisp_Hash_Table *ht2 = XHASH_TABLE (hash_table2);
-  hentry *e, *sentinel;
+  htentry *e, *sentinel;
 
   if ((ht1->test_function != ht2->test_function) ||
       (ht1->weakness      != ht2->weakness)      ||
@@ -260,7 +260,7 @@
   depth++;
 
   for (e = ht1->hentries, sentinel = e + ht1->size; e < sentinel; e++)
-    if (!HENTRY_CLEAR_P (e))
+    if (!HTENTRY_CLEAR_P (e))
       /* Look up the key in the other hash table, and compare the values. */
       {
 	Lisp_Object value_in_other = Fgethash (e->key, hash_table2, Qunbound);
@@ -312,12 +312,12 @@
 print_hash_table_data (Lisp_Hash_Table *ht, Lisp_Object printcharfun)
 {
   int count = 0;
-  hentry *e, *sentinel;
+  htentry *e, *sentinel;
 
   write_c_string (printcharfun, " data (");
 
   for (e = ht->hentries, sentinel = e + ht->size; e < sentinel; e++)
-    if (!HENTRY_CLEAR_P (e))
+    if (!HTENTRY_CLEAR_P (e))
       {
 	if (count > 0)
 	  write_c_string (printcharfun, " ");
@@ -386,14 +386,14 @@
 }
 
 static void
-free_hentries (hentry *hentries, size_t size)
+free_hentries (htentry *hentries, size_t size)
 {
 #ifdef ERROR_CHECK_STRUCTURES
   /* Ensure a crash if other code uses the discarded entries afterwards. */
-  hentry *e, *sentinel;
+  htentry *e, *sentinel;
 
   for (e = hentries, sentinel = e + size; e < sentinel; e++)
-    * (unsigned long *) e = 0xdeadbeef;
+    * (unsigned long *) e = 0xdeadbeef; /* -559038737 base 10 */
 #endif
 
   if (!DUMPEDP (hentries))
@@ -411,25 +411,41 @@
     }
 }
 
-static const struct lrecord_description hentry_description_1[] = {
-  { XD_LISP_OBJECT, offsetof (hentry, key) },
-  { XD_LISP_OBJECT, offsetof (hentry, value) },
+static const struct memory_description htentry_description_1[] = {
+  { XD_LISP_OBJECT, offsetof (htentry, key) },
+  { XD_LISP_OBJECT, offsetof (htentry, value) },
   { XD_END }
 };
 
-static const struct struct_description hentry_description = {
-  sizeof (hentry),
-  hentry_description_1
+static const struct sized_memory_description htentry_description = {
+  sizeof (htentry),
+  htentry_description_1
 };
 
-const struct lrecord_description hash_table_description[] = {
-  { XD_ELEMCOUNT,     offsetof (Lisp_Hash_Table, size) },
-  { XD_STRUCT_PTR, offsetof (Lisp_Hash_Table, hentries), XD_INDIRECT(0, 1), &hentry_description },
+static const struct memory_description htentry_union_description_1[] = {
+  /* Note: XD_INDIRECT in this table refers to the surrounding table,
+     and so this will work. */
+  { XD_STRUCT_PTR, HASH_TABLE_NON_WEAK, XD_INDIRECT (0, 1),
+    &htentry_description },
+  { XD_STRUCT_PTR, 0, XD_INDIRECT (0, 1), &htentry_description,
+    XD_FLAG_UNION_DEFAULT_ENTRY | XD_FLAG_NO_KKCC },
+  { XD_END }
+};
+
+static const struct sized_memory_description htentry_union_description = {
+  sizeof (htentry *),
+  htentry_union_description_1
+};
+
+const struct memory_description hash_table_description[] = {
+  { XD_ELEMCOUNT,  offsetof (Lisp_Hash_Table, size) },
+  { XD_INT,	   offsetof (Lisp_Hash_Table, weakness) },
+  { XD_UNION,	   offsetof (Lisp_Hash_Table, hentries), XD_INDIRECT (1, 0),
+    &htentry_union_description },
   { XD_LO_LINK,    offsetof (Lisp_Hash_Table, next_weak) },
   { XD_END }
 };
 
-#ifdef USE_KKCC
 DEFINE_LRECORD_IMPLEMENTATION ("hash-table", hash_table,
 			       1, /*dumpable-flag*/
                                mark_hash_table, print_hash_table,
@@ -437,14 +453,6 @@
 			       hash_table_equal, hash_table_hash,
 			       hash_table_description,
 			       Lisp_Hash_Table);
-#else /* not USE_KKCC */
-DEFINE_LRECORD_IMPLEMENTATION ("hash-table", hash_table,
-                               mark_hash_table, print_hash_table,
-			       finalize_hash_table,
-			       hash_table_equal, hash_table_hash,
-			       hash_table_description,
-			       Lisp_Hash_Table);
-#endif /* not USE_KKCC */
 
 static Lisp_Hash_Table *
 xhash_table (Lisp_Object hash_table)
@@ -537,8 +545,8 @@
 
   compute_hash_table_derived_values (ht);
 
-  /* We leave room for one never-occupied sentinel hentry at the end.  */
-  ht->hentries = xnew_array_and_zero (hentry, ht->size + 1);
+  /* We leave room for one never-occupied sentinel htentry at the end.  */
+  ht->hentries = xnew_array_and_zero (htentry, ht->size + 1);
 
   hash_table = wrap_hash_table (ht);
 
@@ -628,7 +636,7 @@
   if (EQ (obj, Qvalue_weak))		return HASH_TABLE_VALUE_WEAK;
 
   invalid_constant ("Invalid hash table weakness", obj);
-  RETURN_NOT_REACHED (HASH_TABLE_NON_WEAK)
+  RETURN_NOT_REACHED (HASH_TABLE_NON_WEAK);
 }
 
 static int
@@ -654,7 +662,7 @@
   if (EQ (obj, Qeql))	return HASH_TABLE_EQL;
 
   invalid_constant ("Invalid hash table test", obj);
-  RETURN_NOT_REACHED (HASH_TABLE_EQ)
+  RETURN_NOT_REACHED (HASH_TABLE_EQ);
 }
 
 static int
@@ -941,8 +949,8 @@
 
   copy_lcrecord (ht, ht_old);
 
-  ht->hentries = xnew_array (hentry, ht_old->size + 1);
-  memcpy (ht->hentries, ht_old->hentries, (ht_old->size + 1) * sizeof (hentry));
+  ht->hentries = xnew_array (htentry, ht_old->size + 1);
+  memcpy (ht->hentries, ht_old->hentries, (ht_old->size + 1) * sizeof (htentry));
 
   hash_table = wrap_hash_table (ht);
 
@@ -958,7 +966,7 @@
 static void
 resize_hash_table (Lisp_Hash_Table *ht, Elemcount new_size)
 {
-  hentry *old_entries, *new_entries, *sentinel, *e;
+  htentry *old_entries, *new_entries, *sentinel, *e;
   Elemcount old_size;
 
   old_size = ht->size;
@@ -966,15 +974,15 @@
 
   old_entries = ht->hentries;
 
-  ht->hentries = xnew_array_and_zero (hentry, new_size + 1);
+  ht->hentries = xnew_array_and_zero (htentry, new_size + 1);
   new_entries = ht->hentries;
 
   compute_hash_table_derived_values (ht);
 
   for (e = old_entries, sentinel = e + old_size; e < sentinel; e++)
-    if (!HENTRY_CLEAR_P (e))
+    if (!HTENTRY_CLEAR_P (e))
       {
-	hentry *probe = new_entries + HASHCODE (e->key, ht);
+	htentry *probe = new_entries + HASHCODE (e->key, ht);
 	LINEAR_PROBING_LOOP (probe, new_entries, new_size)
 	  ;
 	*probe = *e;
@@ -990,19 +998,19 @@
 pdump_reorganize_hash_table (Lisp_Object hash_table)
 {
   const Lisp_Hash_Table *ht = xhash_table (hash_table);
-  hentry *new_entries = xnew_array_and_zero (hentry, ht->size + 1);
-  hentry *e, *sentinel;
+  htentry *new_entries = xnew_array_and_zero (htentry, ht->size + 1);
+  htentry *e, *sentinel;
 
   for (e = ht->hentries, sentinel = e + ht->size; e < sentinel; e++)
-    if (!HENTRY_CLEAR_P (e))
+    if (!HTENTRY_CLEAR_P (e))
       {
-	hentry *probe = new_entries + HASHCODE (e->key, ht);
+	htentry *probe = new_entries + HASHCODE (e->key, ht);
 	LINEAR_PROBING_LOOP (probe, new_entries, ht->size)
 	  ;
 	*probe = *e;
       }
 
-  memcpy (ht->hentries, new_entries, ht->size * sizeof (hentry));
+  memcpy (ht->hentries, new_entries, ht->size * sizeof (htentry));
 
   xfree (new_entries);
 }
@@ -1015,12 +1023,12 @@
   resize_hash_table (ht, new_size);
 }
 
-static hentry *
-find_hentry (Lisp_Object key, const Lisp_Hash_Table *ht)
+static htentry *
+find_htentry (Lisp_Object key, const Lisp_Hash_Table *ht)
 {
   hash_table_test_function_t test_function = ht->test_function;
-  hentry *entries = ht->hentries;
-  hentry *probe = entries + HASHCODE (key, ht);
+  htentry *entries = ht->hentries;
+  htentry *probe = entries + HASHCODE (key, ht);
 
   LINEAR_PROBING_LOOP (probe, entries, ht->size)
     if (KEYS_EQUAL_P (probe->key, key, test_function))
@@ -1036,9 +1044,9 @@
        (key, hash_table, default_))
 {
   const Lisp_Hash_Table *ht = xhash_table (hash_table);
-  hentry *e = find_hentry (key, ht);
+  htentry *e = find_htentry (key, ht);
 
-  return HENTRY_CLEAR_P (e) ? default_ : e->value;
+  return HTENTRY_CLEAR_P (e) ? default_ : e->value;
 }
 
 DEFUN ("puthash", Fputhash, 3, 3, 0, /*
@@ -1047,9 +1055,9 @@
        (key, value, hash_table))
 {
   Lisp_Hash_Table *ht = xhash_table (hash_table);
-  hentry *e = find_hentry (key, ht);
+  htentry *e = find_htentry (key, ht);
 
-  if (!HENTRY_CLEAR_P (e))
+  if (!HTENTRY_CLEAR_P (e))
     return e->value = value;
 
   e->key   = key;
@@ -1061,28 +1069,28 @@
   return value;
 }
 
-/* Remove hentry pointed at by PROBE.
+/* Remove htentry pointed at by PROBE.
    Subsequent entries are removed and reinserted.
    We don't use tombstones - too wasteful.  */
 static void
-remhash_1 (Lisp_Hash_Table *ht, hentry *entries, hentry *probe)
+remhash_1 (Lisp_Hash_Table *ht, htentry *entries, htentry *probe)
 {
   Elemcount size = ht->size;
-  CLEAR_HENTRY (probe);
+  CLEAR_HTENTRY (probe);
   probe++;
   ht->count--;
 
   LINEAR_PROBING_LOOP (probe, entries, size)
     {
       Lisp_Object key = probe->key;
-      hentry *probe2 = entries + HASHCODE (key, ht);
+      htentry *probe2 = entries + HASHCODE (key, ht);
       LINEAR_PROBING_LOOP (probe2, entries, size)
 	if (EQ (probe2->key, key))
-	  /* hentry at probe doesn't need to move. */
+	  /* htentry at probe doesn't need to move. */
 	  goto continue_outer_loop;
-      /* Move hentry from probe to new home at probe2. */
+      /* Move htentry from probe to new home at probe2. */
       *probe2 = *probe;
-      CLEAR_HENTRY (probe);
+      CLEAR_HTENTRY (probe);
     continue_outer_loop: continue;
     }
 }
@@ -1095,9 +1103,9 @@
        (key, hash_table))
 {
   Lisp_Hash_Table *ht = xhash_table (hash_table);
-  hentry *e = find_hentry (key, ht);
+  htentry *e = find_htentry (key, ht);
 
-  if (HENTRY_CLEAR_P (e))
+  if (HTENTRY_CLEAR_P (e))
     return Qnil;
 
   remhash_1 (ht, ht->hentries, e);
@@ -1110,10 +1118,10 @@
        (hash_table))
 {
   Lisp_Hash_Table *ht = xhash_table (hash_table);
-  hentry *e, *sentinel;
+  htentry *e, *sentinel;
 
   for (e = ht->hentries, sentinel = e + ht->size; e < sentinel; e++)
-    CLEAR_HENTRY (e);
+    CLEAR_HTENTRY (e);
   ht->count = 0;
 
   return hash_table;
@@ -1280,11 +1288,11 @@
   Lisp_Object * const objs =
     /* If the hash table is empty, ht->count could be 0. */
     xnew_array (Lisp_Object, 2 * (ht->count > 0 ? ht->count : 1));
-  const hentry *e, *sentinel;
+  const htentry *e, *sentinel;
   Lisp_Object *pobj;
 
   for (e = ht->hentries, sentinel = e + ht->size, pobj = objs; e < sentinel; e++)
-    if (!HENTRY_CLEAR_P (e))
+    if (!HTENTRY_CLEAR_P (e))
       {
 	*(pobj++) = e->key;
 	*(pobj++) = e->value;
@@ -1339,10 +1347,10 @@
 	       Lisp_Object hash_table, void *extra_arg)
 {
   const Lisp_Hash_Table *ht = XHASH_TABLE (hash_table);
-  const hentry *e, *sentinel;
+  const htentry *e, *sentinel;
 
   for (e = ht->hentries, sentinel = e + ht->size; e < sentinel; e++)
-    if (!HENTRY_CLEAR_P (e))
+    if (!HTENTRY_CLEAR_P (e))
       if (function (e->key, e->value, extra_arg))
 	return;
 }
@@ -1422,8 +1430,8 @@
        hash_table = XHASH_TABLE (hash_table)->next_weak)
     {
       const Lisp_Hash_Table *ht = XHASH_TABLE (hash_table);
-      const hentry *e = ht->hentries;
-      const hentry *sentinel = e + ht->size;
+      const htentry *e = ht->hentries;
+      const htentry *sentinel = e + ht->size;
 
       if (! marked_p (hash_table))
 	/* The hash table is probably garbage.  Ignore it. */
@@ -1436,21 +1444,21 @@
 	{
 	case HASH_TABLE_KEY_WEAK:
 	  for (; e < sentinel; e++)
-	    if (!HENTRY_CLEAR_P (e))
+	    if (!HTENTRY_CLEAR_P (e))
 	      if (marked_p (e->key))
 		MARK_OBJ (e->value);
 	  break;
 
 	case HASH_TABLE_VALUE_WEAK:
 	  for (; e < sentinel; e++)
-	    if (!HENTRY_CLEAR_P (e))
+	    if (!HTENTRY_CLEAR_P (e))
 	      if (marked_p (e->value))
 		MARK_OBJ (e->key);
 	  break;
 
 	case HASH_TABLE_KEY_VALUE_WEAK:
 	  for (; e < sentinel; e++)
-	    if (!HENTRY_CLEAR_P (e))
+	    if (!HTENTRY_CLEAR_P (e))
 	      {
 		if (marked_p (e->value))
 		  MARK_OBJ (e->key);
@@ -1461,7 +1469,7 @@
 
 	case HASH_TABLE_KEY_CAR_WEAK:
 	  for (; e < sentinel; e++)
-	    if (!HENTRY_CLEAR_P (e))
+	    if (!HTENTRY_CLEAR_P (e))
 	      if (!CONSP (e->key) || marked_p (XCAR (e->key)))
 		{
 		  MARK_OBJ (e->key);
@@ -1475,7 +1483,7 @@
 	     glyph code. */
 	case HASH_TABLE_KEY_CAR_VALUE_WEAK:
 	  for (; e < sentinel; e++)
-	    if (!HENTRY_CLEAR_P (e))
+	    if (!HTENTRY_CLEAR_P (e))
 	      {
 		if (!CONSP (e->key) || marked_p (XCAR (e->key)))
 		  {
@@ -1489,7 +1497,7 @@
 
 	case HASH_TABLE_VALUE_CAR_WEAK:
 	  for (; e < sentinel; e++)
-	    if (!HENTRY_CLEAR_P (e))
+	    if (!HTENTRY_CLEAR_P (e))
 	      if (!CONSP (e->value) || marked_p (XCAR (e->value)))
 		{
 		  MARK_OBJ (e->key);
@@ -1527,18 +1535,18 @@
 	     in which the key or value, or both, is unmarked
 	     (depending on the weakness of the hash table). */
 	  Lisp_Hash_Table *ht = XHASH_TABLE (hash_table);
-	  hentry *entries = ht->hentries;
-	  hentry *sentinel = entries + ht->size;
-	  hentry *e;
+	  htentry *entries = ht->hentries;
+	  htentry *sentinel = entries + ht->size;
+	  htentry *e;
 
 	  for (e = entries; e < sentinel; e++)
-	    if (!HENTRY_CLEAR_P (e))
+	    if (!HTENTRY_CLEAR_P (e))
 	      {
 	      again:
 		if (!marked_p (e->key) || !marked_p (e->value))
 		  {
 		    remhash_1 (ht, entries, e);
-		    if (!HENTRY_CLEAR_P (e))
+		    if (!HTENTRY_CLEAR_P (e))
 		      goto again;
 		  }
 	      }
--- a/src/elhash.h	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/elhash.h	Sun Jan 12 11:08:22 2003 +0000
@@ -52,7 +52,7 @@
   HASH_TABLE_EQUAL
 };
 
-extern const struct lrecord_description hash_table_description[];
+extern const struct memory_description hash_table_description[];
 
 EXFUN (Fcopy_hash_table, 1);
 EXFUN (Fhash_table_count, 1);
--- a/src/emacs.c	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/emacs.c	Sun Jan 12 11:08:22 2003 +0000
@@ -256,10 +256,6 @@
 #include "systime.h"
 #include "sysproc.h" /* for qxe_getpid() */
 
-#ifdef PDUMP
-#include "dumper.h"
-#endif
-
 #ifdef QUANTIFY
 #include <quantify.h>
 #endif
@@ -699,15 +695,6 @@
    invoking a bare (without dumped data) XEmacs (i.e. `temacs' with
    the conventional dumper or `xemacs -nd' with the pdumper).  See
    Frun_emacs_from_temacs().
-
-   restart interacts with initialized as follows (per Olivier Galibert):
-
-     It's perverted.
-
-     initialized==0 => temacs
-     initialized!=0 && restart!=0 => run-temacs
-     initialized!=0 && restart==0 => either xemacs after conventional dump,
-                                     or xemacs post pdump_load()
 */
 DECLARE_DOESNT_RETURN (main_1 (int, Extbyte **, Extbyte **, int));
 DOESNT_RETURN
@@ -1042,6 +1029,35 @@
 
   /****** Now initialize everything *******/
 
+  /* NOTE NOTE NOTE:
+
+     In the code below, there are three different states we are concerned
+     about:
+
+     "raw-temacs" == No dumped Lisp data present.  `temacs', or (with pdump)
+                     `xemacs -nd'.
+
+     "run-temacs" == We are restarting.  run-emacs-from-temacs is called,
+                     aka `run-temacs' on the command line.
+
+     "post-dump"  == We are running an unexec()ed XEmacs, or we have loaded
+                     dump data using pdump_load().
+
+     initialized==0 => raw-temacs
+     initialized!=0 && restart!=0 => run-temacs
+     initialized!=0 && restart==0 => post-dump
+
+     When post-pdump_load(), we need to reinitialize various structures.
+     This case is noted in the code below by
+
+       initialized +
+       !restart +
+       ifdef PDUMP.
+
+     In the comments below, "dump time" or "dumping" == raw-temacs.
+     "run time" == run-temacs or post-dump.
+*/
+
   /* First, do really basic environment initialization -- catching signals
      and the like.  These functions have no dependence on any part of
      the Lisp engine and need to be done both at dump time and at run time. */
@@ -1075,6 +1091,7 @@
 
   purify_flag = 0;
 #ifdef PDUMP
+  in_pdump = 0;
   if (restart)
     initialized = 1;
   else if (nodumpfile)
@@ -1129,6 +1146,8 @@
     purify_flag = 1;
 #endif
 
+  init_alloc_early ();
+
   if (!initialized)
     {
       /* Initialize things so that new Lisp objects
@@ -1155,16 +1174,7 @@
 
       /* Make sure that eistrings can be created. */
       init_eistring_once_early ();
-    }
-
-  /* The following will get called in raw-temacs, post-dump/pdump-load XEmacs,
-     and run-temacs. */
-
-  /* Initialize some vars that will also be reset post-dumping */
-  init_alloc_early ();
-
-  if (!initialized)
-    {
+
       /* Now declare all the symbols and define all the Lisp primitives.
 
 	 The *only* thing that the syms_of_*() functions are allowed to do
@@ -1591,7 +1601,7 @@
 #endif /* HAVE_X_WINDOWS */
 #ifdef HAVE_MS_WINDOWS
       image_instantiator_format_create_glyphs_mswindows ();
-#endif /* HAVE_MSWINDOWS_WINDOWS */
+#endif /* HAVE_MS_WINDOWS */
 #ifdef HAVE_GTK
       image_instantiator_format_create_glyphs_gtk ();
 #endif
@@ -1654,7 +1664,7 @@
 	    - make_int()
 	    - make_char()
 	    - make_extent()
-	    - alloc_lcrecord()
+	    - basic_alloc_lcrecord()
 	    - Fcons()
 	    - listN()
             - make_lcrecord_list()
@@ -2041,15 +2051,16 @@
 	  garbage_collect_1 ();
       }
 #endif
+    }
+
 #ifdef PDUMP
-    }
-  else if (!restart)		      /* after successful pdump_load()
+  if (initialized && !restart)	      /* after successful pdump_load()
 					 (note, we are inside ifdef PDUMP) */
     {
-      reinit_alloc_once_early ();
-      reinit_symbols_once_early ();
-      reinit_opaque_once_early ();
-      reinit_eistring_once_early ();
+      reinit_alloc_early ();
+      reinit_symbols_early ();
+      reinit_opaque_early ();
+      reinit_eistring_early ();
 
       reinit_console_type_create_stream ();
 #ifdef HAVE_TTY
@@ -2173,8 +2184,8 @@
       reinit_complex_vars_of_buffer_runtime_only ();
       reinit_complex_vars_of_console_runtime_only ();
       reinit_complex_vars_of_minibuf ();
+    }
 #endif /* PDUMP */
-    }
 
   /* CONGRATULATIONS!!!  We have successfully initialized the Lisp
      engine. */
@@ -2644,7 +2655,7 @@
   report_sheap_usage (0);
 #endif
   LONGJMP (run_temacs_catch, 1);
-  RETURN_NOT_REACHED (Qnil)
+  RETURN_NOT_REACHED (Qnil);
 }
 
 /* ARGSUSED */
@@ -2787,7 +2798,7 @@
   __except (mswindows_handle_hardware_exceptions (GetExceptionCode ())) {}
 #endif
 
-  RETURN_NOT_REACHED (0)
+  RETURN_NOT_REACHED (0);
 }
 
 
@@ -2797,8 +2808,8 @@
 
 #ifndef CANNOT_DUMP
 
-#if !defined(PDUMP) || !defined(SYSTEM_MALLOC)
-extern char my_edata[];
+#if !defined (PDUMP) || !defined (SYSTEM_MALLOC)
+extern Char_Binary my_edata[];
 #endif
 
 extern void disable_free_hook (void);
@@ -3161,30 +3172,38 @@
 	stderr_out ("\nFatal error (%d).\n\n", sig);
       stderr_out
 	("Your files have been auto-saved.\n"
-	 "Use `M-x recover-session' to recover them.\n"
-	 "\n"
-         "Your version of XEmacs was distributed with a PROBLEMS file that  may describe\n"
-	 "your crash, and with luck a workaround.  Please check it first, but do report\n"
-	 "the crash anyway.  "
+"Use `M-x recover-session' to recover them.\n"
+"\n"
+"Your version of XEmacs was distributed with a PROBLEMS file that may describe\n"
+"your crash, and with luck a workaround.  Please check it first, but do report\n"
+"the crash anyway.\n\n"
 #ifdef INFODOCK
-	 "\n\nPlease report this bug by selecting `Report-Bug' in the InfoDock menu.\n"
-	 "*BE SURE* to include the XEmacs configuration from M-x describe-installation,\n"
-	 "or the file Installation in the top directory of the build tree.\n"
+"Please report this bug by selecting `Report-Bug' in the InfoDock menu, or\n"
+"(last resort) by emailing `crashes@xemacs.org' -- note that this is for XEmacs\n"
+"in general, not just Infodock."
 #else
-	 "Please report this bug by invoking M-x report-emacs-bug,\n"
-	 "or by selecting `Send Bug Report' from the Help menu.  If necessary, send\n"
-	 "ordinary email to `crashes@xemacs.org'.  *MAKE SURE* to include the XEmacs\n"
-	 "configuration from M-x describe-installation, or equivalently the file\n"
-	 "Installation in the top of the build tree.\n"
-#endif
-#ifndef _MSC_VER
-	 "\n"
-	 "*Please* try *hard* to obtain a C stack backtrace; without it, we are unlikely\n"
-	 "to be able to analyze the problem.  Locate the core file produced as a result\n"
-	 "of this crash (often called `core' or `core.<process-id>', and located in\n"
-	 "the directory in which you started XEmacs or your home directory), and type\n"
-	 "\n"
-	 "  gdb "
+"Please report this bug by invoking M-x report-emacs-bug, or by selecting\n"
+"`Send Bug Report' from the Help menu.  If that won't work, send ordinary\n"
+"email to `crashes@xemacs.org'."
+#endif
+"  *MAKE SURE* to include this entire output\n"
+"from this crash, especially including the Lisp backtrace, as well as the\n"
+"XEmacs configuration from M-x describe-installation (or equivalently, the\n"
+"file `Installation' in the top of the build tree).\n"
+#ifdef _MSC_VER
+"\n"
+"If you are fortunate enough to have some sort of debugging aid installed\n"
+"on your system, for example Visual C++, and you can get a C stack backtrace,\n"
+"*please* include it, as it will make our life far easier.\n"
+"\n"
+#else
+"\n"
+"*Please* try *hard* to obtain a C stack backtrace; without it, we are unlikely\n"
+"to be able to analyze the problem.  Locate the core file produced as a result\n"
+"of this crash (often called `core' or `core.<process-id>', and located in\n"
+"the directory in which you started XEmacs or your home directory), and type\n"
+"\n"
+"  gdb "
 #endif
 	 );
 #ifndef _MSC_VER
@@ -3211,10 +3230,10 @@
       stderr_out
 	(" core\n"
 	 "\n"
-	 "then type `where' at the debugger prompt.  No GDB on your system?  You may\n"
-	 "have DBX, or XDB, or SDB.  (Ask your system administrator if you need help.)\n"
-	 "If no core file was produced, enable them (often with `ulimit -c unlimited'\n"
-	 "in case of future recurrance of the crash.\n");
+"then type `where' at the debugger prompt.  No GDB on your system?  You may\n"
+"have DBX, or XDB, or SDB.  (Ask your system administrator if you need help.)\n"
+"If no core file was produced, enable them (often with `ulimit -c unlimited')\n"
+"in case of future recurrance of the crash.\n");
 #endif /* _MSC_VER */
     }
 
--- a/src/emodules.h	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/emodules.h	Sun Jan 12 11:08:22 2003 +0000
@@ -89,7 +89,7 @@
 #define dump_add_opaque_int(int_varaddr) DO_NOTHING
 #undef dump_add_opaque_fixnum
 #define dump_add_opaque_fixnum(fixnum_varaddr) DO_NOTHING
-#define dump_add_root_object(varaddr) DO_NOTHING
+#define dump_add_root_lisp_object(varaddr) DO_NOTHING
 #define dump_add_weak_object_chain(varaddr) DO_NOTHING
 #undef staticpro
 #define staticpro(DSF_location) staticpro_nodump(DSF_location)
--- a/src/eval.c	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/eval.c	Sun Jan 12 11:08:22 2003 +0000
@@ -438,23 +438,16 @@
   write_c_string (printcharfun, trailer);
 }
 
-static const struct lrecord_description subr_description[] = {
+static const struct memory_description subr_description[] = {
   { XD_DOC_STRING, offsetof (Lisp_Subr, doc) },
   { XD_END }
 };
 
-#ifdef USE_KKCC
 DEFINE_BASIC_LRECORD_IMPLEMENTATION ("subr", subr,
 				     1, /*dumpable-flag*/
 				     0, print_subr, 0, 0, 0,
 				     subr_description,
 				     Lisp_Subr);
-#else /* not USE_KKCC */
-DEFINE_BASIC_LRECORD_IMPLEMENTATION ("subr", subr,
-				     0, print_subr, 0, 0, 0,
-				     subr_description,
-				     Lisp_Subr);
-#endif /* not USE_KKCC */
 
 /************************************************************************/
 /*			 Entering the debugger				*/
@@ -1601,7 +1594,7 @@
   check_catchlist_sanity ();
 #endif /* Former code */
 
-  gcprolist = c->gcpro;
+  UNWIND_GCPRO_TO (c->gcpro);
   backtrace_list = c->backlist;
   lisp_eval_depth = c->lisp_eval_depth;
 
@@ -2163,6 +2156,8 @@
 {
 }
 
+#ifdef ERROR_CHECK_TRAPPING_PROBLEMS
+
 static void
 check_proper_critical_section_gc_protection (void)
 {
@@ -2171,6 +2166,8 @@
      "Potential GC from within redisplay without being properly wrapped");
 }
 
+#endif /* ERROR_CHECK_TRAPPING_PROBLEMS */
+
 static void
 check_proper_critical_section_nonlocal_exit_protection (void)
 {
@@ -2446,15 +2443,15 @@
    Qresource, etc.). */
 
 void
-maybe_signal_error_1 (Lisp_Object sig, Lisp_Object data, Lisp_Object class,
+maybe_signal_error_1 (Lisp_Object sig, Lisp_Object data, Lisp_Object class_,
 		      Error_Behavior errb)
 {
   if (ERRB_EQ (errb, ERROR_ME_NOT))
     return;
   else if (ERRB_EQ (errb, ERROR_ME_DEBUG_WARN))
-    warn_when_safe_lispobj (class, Qdebug, Fcons (sig, data));
+    warn_when_safe_lispobj (class_, Qdebug, Fcons (sig, data));
   else if (ERRB_EQ (errb, ERROR_ME_WARN))
-    warn_when_safe_lispobj (class, Qwarning, Fcons (sig, data));
+    warn_when_safe_lispobj (class_, Qwarning, Fcons (sig, data));
   else
     for (;;)
       Fsignal (sig, data);
@@ -2465,18 +2462,18 @@
 
 Lisp_Object
 maybe_signal_continuable_error_1 (Lisp_Object sig, Lisp_Object data,
-				  Lisp_Object class, Error_Behavior errb)
+				  Lisp_Object class_, Error_Behavior errb)
 {
   if (ERRB_EQ (errb, ERROR_ME_NOT))
     return Qnil;
   else if (ERRB_EQ (errb, ERROR_ME_DEBUG_WARN))
     {
-      warn_when_safe_lispobj (class, Qdebug, Fcons (sig, data));
+      warn_when_safe_lispobj (class_, Qdebug, Fcons (sig, data));
       return Qnil;
     }
   else if (ERRB_EQ (errb, ERROR_ME_WARN))
     {
-      warn_when_safe_lispobj (class, Qwarning, Fcons (sig, data));
+      warn_when_safe_lispobj (class_, Qwarning, Fcons (sig, data));
       return Qnil;
     }
   else
@@ -2522,13 +2519,13 @@
 
 void
 maybe_signal_error (Lisp_Object type, const CIbyte *reason,
-		    Lisp_Object frob, Lisp_Object class,
+		    Lisp_Object frob, Lisp_Object class_,
 		    Error_Behavior errb)
 {
   /* Optimization: */
   if (ERRB_EQ (errb, ERROR_ME_NOT))
     return;
-  maybe_signal_error_1 (type, build_error_data (reason, frob), class, errb);
+  maybe_signal_error_1 (type, build_error_data (reason, frob), class_, errb);
 }
 
 Lisp_Object
@@ -2540,7 +2537,7 @@
 
 Lisp_Object
 maybe_signal_continuable_error (Lisp_Object type, const CIbyte *reason,
-				Lisp_Object frob, Lisp_Object class,
+				Lisp_Object frob, Lisp_Object class_,
 				Error_Behavior errb)
 {
   /* Optimization: */
@@ -2548,7 +2545,7 @@
     return Qnil;
   return maybe_signal_continuable_error_1 (type,
 					   build_error_data (reason, frob),
-					   class, errb);
+					   class_, errb);
 }
 
 
@@ -2571,13 +2568,13 @@
 void
 maybe_signal_error_2 (Lisp_Object type, const CIbyte *reason,
 		      Lisp_Object frob0, Lisp_Object frob1,
-		      Lisp_Object class, Error_Behavior errb)
+		      Lisp_Object class_, Error_Behavior errb)
 {
   /* Optimization: */
   if (ERRB_EQ (errb, ERROR_ME_NOT))
     return;
   maybe_signal_error_1 (type, list3 (build_msg_string (reason), frob0,
-				     frob1), class, errb);
+				     frob1), class_, errb);
 }
 
 Lisp_Object
@@ -2591,14 +2588,14 @@
 Lisp_Object
 maybe_signal_continuable_error_2 (Lisp_Object type, const CIbyte *reason,
 				  Lisp_Object frob0, Lisp_Object frob1,
-				  Lisp_Object class, Error_Behavior errb)
+				  Lisp_Object class_, Error_Behavior errb)
 {
   /* Optimization: */
   if (ERRB_EQ (errb, ERROR_ME_NOT))
     return Qnil;
   return maybe_signal_continuable_error_1
     (type, list3 (build_msg_string (reason), frob0, frob1),
-     class, errb);
+     class_, errb);
 }
 
 
@@ -2623,7 +2620,7 @@
 }
 
 void
-maybe_signal_ferror (Lisp_Object type, Lisp_Object class, Error_Behavior errb,
+maybe_signal_ferror (Lisp_Object type, Lisp_Object class_, Error_Behavior errb,
 		     const CIbyte *fmt, ...)
 {
   Lisp_Object obj;
@@ -2638,7 +2635,7 @@
   va_end (args);
 
   /* Fsignal GC-protects its args */
-  maybe_signal_error (type, 0, obj, class, errb);
+  maybe_signal_error (type, 0, obj, class_, errb);
 }
 
 Lisp_Object
@@ -2656,7 +2653,7 @@
 }
 
 Lisp_Object
-maybe_signal_continuable_ferror (Lisp_Object type, Lisp_Object class,
+maybe_signal_continuable_ferror (Lisp_Object type, Lisp_Object class_,
 				 Error_Behavior errb, const CIbyte *fmt, ...)
 {
   Lisp_Object obj;
@@ -2671,7 +2668,7 @@
   va_end (args);
 
   /* Fsignal GC-protects its args */
-  return maybe_signal_continuable_error (type, 0, obj, class, errb);
+  return maybe_signal_continuable_error (type, 0, obj, class_, errb);
 }
 
 
@@ -2706,7 +2703,7 @@
 
 void
 maybe_signal_ferror_with_frob (Lisp_Object type, Lisp_Object frob,
-			       Lisp_Object class, Error_Behavior errb,
+			       Lisp_Object class_, Error_Behavior errb,
 			       const CIbyte *fmt, ...)
 {
   Lisp_Object obj;
@@ -2721,7 +2718,7 @@
   va_end (args);
 
   /* Fsignal GC-protects its args */
-  maybe_signal_error_1 (type, Fcons (obj, build_error_data (0, frob)), class,
+  maybe_signal_error_1 (type, Fcons (obj, build_error_data (0, frob)), class_,
 			errb);
 }
 
@@ -2742,7 +2739,7 @@
 
 Lisp_Object
 maybe_signal_continuable_ferror_with_frob (Lisp_Object type, Lisp_Object frob,
-					   Lisp_Object class,
+					   Lisp_Object class_,
 					   Error_Behavior errb,
 					   const CIbyte *fmt, ...)
 {
@@ -2761,7 +2758,7 @@
   return maybe_signal_continuable_error_1 (type,
 					   Fcons (obj,
 						  build_error_data (0, frob)),
-					   class, errb);
+					   class_, errb);
 }
 
 
@@ -2845,9 +2842,9 @@
 
 void
 maybe_syntax_error (const CIbyte *reason, Lisp_Object frob,
-		    Lisp_Object class, Error_Behavior errb)
-{
-  maybe_signal_error (Qsyntax_error, reason, frob, class, errb);
+		    Lisp_Object class_, Error_Behavior errb)
+{
+  maybe_signal_error (Qsyntax_error, reason, frob, class_, errb);
 }
 
 DOESNT_RETURN
@@ -2864,9 +2861,9 @@
 
 void
 maybe_sferror (const CIbyte *reason, Lisp_Object frob,
-	       Lisp_Object class, Error_Behavior errb)
-{
-  maybe_signal_error (Qstructure_formation_error, reason, frob, class, errb);
+	       Lisp_Object class_, Error_Behavior errb)
+{
+  maybe_signal_error (Qstructure_formation_error, reason, frob, class_, errb);
 }
 
 DOESNT_RETURN
@@ -2884,9 +2881,9 @@
 
 void
 maybe_invalid_argument (const CIbyte *reason, Lisp_Object frob,
-			Lisp_Object class, Error_Behavior errb)
-{
-  maybe_signal_error (Qinvalid_argument, reason, frob, class, errb);
+			Lisp_Object class_, Error_Behavior errb)
+{
+  maybe_signal_error (Qinvalid_argument, reason, frob, class_, errb);
 }
 
 DOESNT_RETURN
@@ -2904,9 +2901,9 @@
 
 void
 maybe_invalid_constant (const CIbyte *reason, Lisp_Object frob,
-			Lisp_Object class, Error_Behavior errb)
-{
-  maybe_signal_error (Qinvalid_constant, reason, frob, class, errb);
+			Lisp_Object class_, Error_Behavior errb)
+{
+  maybe_signal_error (Qinvalid_constant, reason, frob, class_, errb);
 }
 
 DOESNT_RETURN
@@ -2924,9 +2921,9 @@
 
 void
 maybe_invalid_operation (const CIbyte *reason, Lisp_Object frob,
-			 Lisp_Object class, Error_Behavior errb)
-{
-  maybe_signal_error (Qinvalid_operation, reason, frob, class, errb);
+			 Lisp_Object class_, Error_Behavior errb)
+{
+  maybe_signal_error (Qinvalid_operation, reason, frob, class_, errb);
 }
 
 DOESNT_RETURN
@@ -2943,9 +2940,9 @@
 
 void
 maybe_invalid_change (const CIbyte *reason, Lisp_Object frob,
-		      Lisp_Object class, Error_Behavior errb)
-{
-  maybe_signal_error (Qinvalid_change, reason, frob, class, errb);
+		      Lisp_Object class_, Error_Behavior errb)
+{
+  maybe_signal_error (Qinvalid_change, reason, frob, class_, errb);
 }
 
 DOESNT_RETURN
@@ -2962,9 +2959,9 @@
 
 void
 maybe_invalid_state (const CIbyte *reason, Lisp_Object frob,
-		     Lisp_Object class, Error_Behavior errb)
-{
-  maybe_signal_error (Qinvalid_state, reason, frob, class, errb);
+		     Lisp_Object class_, Error_Behavior errb)
+{
+  maybe_signal_error (Qinvalid_state, reason, frob, class_, errb);
 }
 
 DOESNT_RETURN
@@ -3471,7 +3468,7 @@
 	 && !(inhibit_flags & INHIBIT_ANY_CHANGE_AFFECTING_REDISPLAY))
     {
       struct gcpro gcpro1, gcpro2, gcpro3, gcpro4;
-      Lisp_Object this_warning_cons, this_warning, class, level, messij;
+      Lisp_Object this_warning_cons, this_warning, class_, level, messij;
       int speccount = internal_bind_int (&in_warnings, 1);
 
       this_warning_cons = Vpending_warnings;
@@ -3480,7 +3477,7 @@
 	 it won't happen infinitely */
       Vpending_warnings = XCDR (Vpending_warnings);
       free_cons (this_warning_cons);
-      class = XCAR (this_warning);
+      class_ = XCAR (this_warning);
       level = XCAR (XCDR (this_warning));
       messij = XCAR (XCDR (XCDR (this_warning)));
       free_list (this_warning);
@@ -3489,10 +3486,10 @@
 	Vpending_warnings_tail = Qnil; /* perhaps not strictly necessary,
 					  but safer */
 
-      GCPRO4 (form, class, level, messij);
+      GCPRO4 (form, class_, level, messij);
       if (!STRINGP (messij))
 	messij = Fprin1_to_string (messij, Qnil);
-      call3 (Qdisplay_warning, class, messij, level);
+      call3 (Qdisplay_warning, class_, messij, level);
       UNGCPRO;
       unbind_to (speccount);
     }
@@ -5071,7 +5068,7 @@
     tem = (fun) (arg);
 
   if (thrown && !EQ (thrown_tag, package.catchtag)
-      && (!flags & INHIBIT_WARNING_ISSUE)
+      && !(flags & INHIBIT_WARNING_ISSUE)
       && !warning_will_be_discarded (current_warning_level ()))
     {
       Lisp_Object errstr;
@@ -5195,7 +5192,7 @@
 
 Lisp_Object
 call_with_suspended_errors (lisp_fn_t fun, Lisp_Object retval,
-			    Lisp_Object class, Error_Behavior errb,
+			    Lisp_Object class_, Error_Behavior errb,
 			    int nargs, ...)
 {
   va_list vargs;
@@ -5205,8 +5202,8 @@
   int flags;
   struct gcpro gcpro1;
 
-  assert (SYMBOLP (class)); /* sanity-check */
-  assert (!NILP (class));
+  assert (SYMBOLP (class_)); /* sanity-check */
+  assert (!NILP (class_));
   assert (nargs >= 0 && nargs < 20);
 
   va_start (vargs, nargs);
@@ -5243,7 +5240,7 @@
   {
     Lisp_Object its_way_too_goddamn_late =
       call_trapping_problems
-       (class, 0, flags, 0, va_call_trapping_problems_1,
+       (class_, 0, flags, 0, va_call_trapping_problems_1,
 	&fazer_invocacao_atrapalhando_problemas);
     UNGCPRO;
     if (UNBOUNDP (its_way_too_goddamn_late))
@@ -6246,13 +6243,13 @@
 }
 
 void
-warn_when_safe_lispobj (Lisp_Object class, Lisp_Object level,
+warn_when_safe_lispobj (Lisp_Object class_, Lisp_Object level,
 			Lisp_Object obj)
 {
   if (warning_will_be_discarded (level))
     return;
 
-  obj = list1 (list3 (class, level, obj));
+  obj = list1 (list3 (class_, level, obj));
   if (NILP (Vpending_warnings))
     Vpending_warnings = Vpending_warnings_tail = obj;
   else
@@ -6270,7 +6267,7 @@
    automatically be called when it is safe to do so. */
 
 void
-warn_when_safe (Lisp_Object class, Lisp_Object level, const CIbyte *fmt, ...)
+warn_when_safe (Lisp_Object class_, Lisp_Object level, const CIbyte *fmt, ...)
 {
   Lisp_Object obj;
   va_list args;
@@ -6282,7 +6279,7 @@
   obj = emacs_vsprintf_string (CGETTEXT (fmt), args);
   va_end (args);
 
-  warn_when_safe_lispobj (class, level, obj);
+  warn_when_safe_lispobj (class_, level, obj);
 }
 
 
@@ -6535,7 +6532,7 @@
 
   staticpro (&Vpending_warnings);
   Vpending_warnings = Qnil;
-  dump_add_root_object (&Vpending_warnings_tail);
+  dump_add_root_lisp_object (&Vpending_warnings_tail);
   Vpending_warnings_tail = Qnil;
 
   DEFVAR_LISP ("log-warning-minimum-level", &Vlog_warning_minimum_level);
--- a/src/event-Xt.c	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/event-Xt.c	Sun Jan 12 11:08:22 2003 +0000
@@ -295,14 +295,7 @@
 #endif
 }
 
-/* #### The way that keysym correspondence to characters should work:
-   - a Lisp_Event should contain a keysym AND a character slot.
-   - keybindings are tried with the keysym.  If no binding can be found,
-   and there is a corresponding character, call self-insert-command.
-
-   #### Nuke x-iso8859-1.el.
-   #### Nuke the Qascii_character property.
-   #### Nuke Vcharacter_set_property.
+/* See comment near character_to_event().
 */
 static void
 maybe_define_x_key_as_self_inserting_character (KeySym keysym, Lisp_Object symbol)
@@ -1075,16 +1068,10 @@
 	    Lisp_Event *ev          = XEVENT (emacs_event);
             ev->channel	            = DEVICE_CONSOLE (d);
             ev->timestamp	    = event->time;
-            ev->event_type	    = key_press_event;
-#ifdef USE_KKCC
 	    XSET_EVENT_TYPE (emacs_event, key_press_event);
-	    XSET_KEY_DATA_MODIFIERS (XEVENT_DATA (emacs_event), 0);
-	    XSET_KEY_DATA_KEYSYM (XEVENT_DATA (emacs_event), make_char (ch));
-#else
-            ev->event.key.modifiers = 0;
-            ev->event.key.keysym    = make_char (ch);
-#endif
-            enqueue_Xt_dispatch_event (emacs_event);
+	    XSET_EVENT_KEY_MODIFIERS (emacs_event, 0);
+	    XSET_EVENT_KEY_KEYSYM (emacs_event, make_char (ch));
+            enqueue_dispatch_event (emacs_event);
           }
 	Lstream_close (istr);
 	UNGCPRO;
@@ -1219,11 +1206,7 @@
 	    XKeyEvent *ev = &x_event->xkey;
 	    /* This used to compute the frame from the given X window and
 	       store it here, but we really don't care about the frame. */
-#ifdef USE_KKCC
 	    SET_EVENT_CHANNEL (emacs_event, DEVICE_CONSOLE (d));
-#else /* not USE_KKCC */
-	    emacs_event->channel = DEVICE_CONSOLE (d);
-#endif /* not USE_KKCC */
 	    keysym = x_to_emacs_keysym (&x_event->xkey, 0);
 
 	    /* If the emacs keysym is nil, then that means that the X
@@ -1267,17 +1250,10 @@
 		if (top && bot && top != bot)
 		  modifiers &= ~XEMACS_MOD_SHIFT;
 	      }
-#ifdef USE_KKCC
 	    set_event_type (emacs_event, key_press_event);
 	    SET_EVENT_TIMESTAMP (emacs_event, ev->time);
-	    XSET_KEY_DATA_MODIFIERS (EVENT_DATA (emacs_event), modifiers);
-	    XSET_KEY_DATA_KEYSYM (EVENT_DATA (emacs_event), keysym);
-#else /* not USE_KKCC */
-	    emacs_event->event_type	     = key_press_event;
-	    emacs_event->timestamp	     = ev->time;
-	    emacs_event->event.key.modifiers = modifiers;
-	    emacs_event->event.key.keysym    = keysym;
-#endif /* not USE_KKCC */
+	    SET_EVENT_KEY_MODIFIERS (emacs_event, modifiers);
+	    SET_EVENT_KEY_KEYSYM (emacs_event, keysym);
 	  }
 	else                    /* Mouse press/release event */
 	  {
@@ -1286,27 +1262,15 @@
 
 	    if (! frame)
 	      return 0;	/* not for us */
-#ifdef USE_KKCC
 	    set_event_type (emacs_event, (x_event->type == ButtonPress) ?
                         button_press_event : button_release_event);
-            SET_EVENT_CHANNEL (emacs_event, wrap_frame(frame));
-
-	    XSET_BUTTON_DATA_MODIFIERS (EVENT_DATA (emacs_event), modifiers);
+            SET_EVENT_CHANNEL (emacs_event, wrap_frame (frame));
+
+	    SET_EVENT_BUTTON_MODIFIERS (emacs_event, modifiers);
 	    SET_EVENT_TIMESTAMP (emacs_event, ev->time);
-	    XSET_BUTTON_DATA_BUTTON (EVENT_DATA (emacs_event), ev->button);
-	    XSET_BUTTON_DATA_X (EVENT_DATA (emacs_event), ev->x);
-	    XSET_BUTTON_DATA_Y (EVENT_DATA (emacs_event), ev->y);
-#else /* not USE_KKCC */
-	    emacs_event->channel = wrap_frame (frame);
-	    emacs_event->event_type = (x_event->type == ButtonPress) ?
-	      button_press_event : button_release_event;
-
-	    emacs_event->event.button.modifiers = modifiers;
-	    emacs_event->timestamp		= ev->time;
-	    emacs_event->event.button.button	= ev->button;
-	    emacs_event->event.button.x		= ev->x;
-	    emacs_event->event.button.y		= ev->y;
-#endif /* not USE_KKCC */
+	    SET_EVENT_BUTTON_BUTTON (emacs_event, ev->button);
+	    SET_EVENT_BUTTON_X (emacs_event, ev->x);
+	    SET_EVENT_BUTTON_Y (emacs_event, ev->y);
 	    /* because we don't seem to get a FocusIn event for button clicks
 	       when a widget-glyph is selected we will assume that we want the
 	       focus if a button gets pressed. */
@@ -1343,19 +1307,11 @@
           ev = &event2; /* only one structure copy */
 
         DEVICE_X_MOUSE_TIMESTAMP (d) = ev->time;
-#ifdef USE_KKCC
-        SET_EVENT_CHANNEL (emacs_event, wrap_frame(frame));
+        SET_EVENT_CHANNEL (emacs_event, wrap_frame (frame));
         set_event_type (emacs_event, pointer_motion_event);
         SET_EVENT_TIMESTAMP (emacs_event, ev->time);
-        XSET_MOTION_DATA_X (EVENT_DATA (emacs_event), ev->x);
-        XSET_MOTION_DATA_Y (EVENT_DATA (emacs_event), ev->y);
-#else /* not USE_KKCC */
-        emacs_event->channel = wrap_frame (frame);
-        emacs_event->event_type	    = pointer_motion_event;
-        emacs_event->timestamp      = ev->time;
-        emacs_event->event.motion.x = ev->x;
-        emacs_event->event.motion.y = ev->y;
-#endif /* not USE_KKCC */
+        SET_EVENT_MOTION_X (emacs_event, ev->x);
+        SET_EVENT_MOTION_Y (emacs_event, ev->y);
         if (ev->state & ShiftMask)	modifiers |= XEMACS_MOD_SHIFT;
         if (ev->state & ControlMask)	modifiers |= XEMACS_MOD_CONTROL;
         if (ev->state & xd->MetaMask)	modifiers |= XEMACS_MOD_META;
@@ -1369,11 +1325,7 @@
         if (ev->state & Button5Mask)	modifiers |= XEMACS_MOD_BUTTON5;
         /* Currently ignores Shift_Lock but probably shouldn't
            (but it definitely should ignore Caps_Lock). */
-#ifdef USE_KKCC
-        XSET_MOTION_DATA_MODIFIERS (EVENT_DATA (emacs_event), modifiers);
-#else /* not USE_KKCC */
-        emacs_event->event.motion.modifiers = modifiers;
-#endif /* not USE_KKCC */
+        SET_EVENT_MOTION_MODIFIERS (emacs_event, modifiers);
       }
     break;
 
@@ -1400,16 +1352,10 @@
 	      return 0;	/* not for us */
 
 	    GCPRO4 (l_type, l_data, l_dndlist, l_item);
-#ifdef USE_KKCC
 	    set_event_type (emacs_event, misc_user_event);
-	    SET_EVENT_CHANNEL (emacs_event, wrap_frame(frame));
+	    SET_EVENT_CHANNEL (emacs_event, wrap_frame (frame));
 	    SET_EVENT_TIMESTAMP (emacs_event, DEVICE_X_LAST_SERVER_TIMESTAMP (d));
-#else /* not USE_KKCC */
-	    emacs_event->channel = wrap_frame (frame);
-	    emacs_event->event_type = misc_user_event;
-	    emacs_event->timestamp  = DEVICE_X_LAST_SERVER_TIMESTAMP (d);
-#endif /* not USE_KKCC */
-	    state=DndDragButtons(x_event);
+	    state=DndDragButtons (x_event);
 
 	    if (state & ShiftMask)	modifiers |= XEMACS_MOD_SHIFT;
 	    if (state & ControlMask)	modifiers |= XEMACS_MOD_CONTROL;
@@ -1429,24 +1375,15 @@
 	    if (state & Button2Mask)    button = Button2;
 	    if (state & Button1Mask)    button = Button1;
 
-#ifdef USE_KKCC
-	    XSET_MISC_USER_DATA_MODIFIERS (EVENT_DATA (emacs_event), modifiers);
-	    XSET_MISC_USER_DATA_BUTTON (EVENT_DATA (emacs_event), button);
-
-	    DndDropCoordinates(FRAME_X_TEXT_WIDGET(frame), x_event,
-			       &(XMISC_USER_DATA_X (EVENT_DATA (emacs_event))),
-			       &(XMISC_USER_DATA_Y (EVENT_DATA (emacs_event))) );
-#else /* not USE_KKCC */
-	    emacs_event->event.misc.modifiers = modifiers;
-	    emacs_event->event.misc.button    = button;
-
-	    DndDropCoordinates(FRAME_X_TEXT_WIDGET(frame), x_event,
-			       &(emacs_event->event.misc.x),
-			       &(emacs_event->event.misc.y) );
-#endif /* not USE_KKCC */
-	    DndGetData(x_event,&data,&size);
-
-	    dtype=DndDataType(x_event);
+	    SET_EVENT_MISC_USER_MODIFIERS (emacs_event, modifiers);
+	    SET_EVENT_MISC_USER_BUTTON (emacs_event, button);
+
+	    DndDropCoordinates (FRAME_X_TEXT_WIDGET (frame), x_event,
+				&(EVENT_MISC_USER_X (emacs_event)),
+				&(EVENT_MISC_USER_Y (emacs_event)));
+	    DndGetData (x_event,&data,&size);
+
+	    dtype=DndDataType (x_event);
 	    switch (dtype)
 	      {
 	      case DndFiles: /* null terminated strings, end null */
@@ -1519,13 +1456,8 @@
 		break;
 	      }
 
-#ifdef USE_KKCC
-	    XSET_MISC_USER_DATA_FUNCTION (EVENT_DATA (emacs_event), Qdragdrop_drop_dispatch);
-	    XSET_MISC_USER_DATA_OBJECT (EVENT_DATA (emacs_event), Fcons (l_type, l_dndlist));
-#else /* not USE_KKCC */
-	    emacs_event->event.misc.function = Qdragdrop_drop_dispatch;
-	    emacs_event->event.misc.object = Fcons (l_type, l_dndlist);
-#endif /* not USE_KKCC */
+	    SET_EVENT_MISC_USER_FUNCTION (emacs_event, Qdragdrop_drop_dispatch);
+	    SET_EVENT_MISC_USER_OBJECT (emacs_event, Fcons (l_type, l_dndlist));
 
 	    UNGCPRO;
 
@@ -1545,13 +1477,9 @@
       {
         struct frame *frame;
         Window w;
-#ifdef USE_KKCC
 	XEvent *x_event_copy;
 	SET_EVENT_TYPE (emacs_event, magic_event);
-        x_event_copy = &XMAGIC_DATA_X_EVENT (EVENT_DATA (emacs_event));
-#else /* not USE_KKCC */
-	XEvent *x_event_copy = &emacs_event->event.magic.underlying_x_event;
-#endif /* not USE_KKCC */
+        x_event_copy = &EVENT_MAGIC_X_EVENT (emacs_event);
 
 #define FROB(event_member, window_member) \
 	x_event_copy->event_member = x_event->event_member; \
@@ -1586,14 +1514,8 @@
         if (!frame)
           return 0;
 
-#ifdef USE_KKCC
-	SET_EVENT_TYPE (emacs_event, magic_event);
-	SET_EVENT_CHANNEL (emacs_event, wrap_frame(frame));
-	XSET_MAGIC_DATA_X_EVENT (EVENT_DATA(emacs_event), *x_event_copy);
-#else /* not USE_KKCC */
-        emacs_event->event_type = magic_event;
-        emacs_event->channel = wrap_frame (frame);
-#endif /* not USE_KKCC */
+	SET_EVENT_CHANNEL (emacs_event, wrap_frame (frame));
+	SET_EVENT_MAGIC_X_EVENT (emacs_event, *x_event_copy);
         break;
       }
     }
@@ -1693,25 +1615,15 @@
   Lisp_Event *ev          = XEVENT (emacs_event);
   XEvent *x_event;
 
-  ev->event_type = magic_event;
-
-#ifdef USE_KKCC
   XSET_EVENT_TYPE (emacs_event, magic_event);
-  x_event = &XMAGIC_DATA_X_EVENT (EVENT_DATA (ev));
-#else /* not USE_KKCC */
-  x_event = &ev->event.magic.underlying_x_event;
-#endif /* not USE_KKCC */
+  x_event = &EVENT_MAGIC_X_EVENT (ev);
 
   x_event->type = in_p ? FocusIn : FocusOut;
   x_event->xfocus.window = XtWindow (wants_it);
 
-#ifdef USE_KKCC
   SET_EVENT_CHANNEL (ev, frame);
-#else /* not USE_KKCC */
-  ev->channel = frame;
-#endif /* not USE_KKCC */
-
-  enqueue_Xt_dispatch_event (emacs_event);
+
+  enqueue_dispatch_event (emacs_event);
 }
 
 /* The idea here is that when a widget glyph gets unmapped we don't
@@ -1948,9 +1860,9 @@
      cannot rely on it to do the right thing at the right time for
      widget display.
 
-     drain_X_queue () - this happens when SIGIO gets tripped,
-     processing the event queue allows C-g to be checked for. It gets
-     called from emacs_Xt_event_pending_p ().
+     emacs_Xt_drain_queue () - this happens when SIGIO gets tripped,
+     processing the event queue allows C-g to be checked for.  It gets
+     called from emacs_Xt_event_pending_p (). #### Update this comment.
 
    In order to solve this I have tried introducing a list primitive -
    dispatch-non-command-events - which forces processing of X events
@@ -1990,11 +1902,7 @@
   Lisp_Object console = CDFW_CONSOLE (EVENT_CHANNEL (event));
   if (CONSOLE_X_P (XCONSOLE (console)))
     write_c_string
-#ifdef USE_KKCC
-      (pstream, x_event_name ((XMAGIC_DATA_X_EVENT (EVENT_DATA(event))).type));
-#else /* not USE_KKCC */
-      (pstream, x_event_name (event->event.magic.underlying_x_event.type));
-#endif /* not USE_KKCC */
+      (pstream, x_event_name ((EVENT_MAGIC_X_EVENT (event)).type));
 }
 
 static int
@@ -2002,13 +1910,8 @@
 {
   if (CONSOLE_X_P (XCONSOLE (CDFW_CONSOLE (EVENT_CHANNEL (e1)))) &&
       CONSOLE_X_P (XCONSOLE (CDFW_CONSOLE (EVENT_CHANNEL (e2)))))
-#ifdef USE_KKCC
-    return ((XMAGIC_DATA_X_EVENT (EVENT_DATA(e1))).xany.serial ==
-	    (XMAGIC_DATA_X_EVENT (EVENT_DATA(e2))).xany.serial);
-#else /* not USE_KKCC */
-    return (e1->event.magic.underlying_x_event.xany.serial ==
-	    e2->event.magic.underlying_x_event.xany.serial);
-#endif /* not USE_KKCC */
+    return ((EVENT_MAGIC_X_EVENT (e1)).xany.serial ==
+	    (EVENT_MAGIC_X_EVENT (e2)).xany.serial);
   if (CONSOLE_X_P (XCONSOLE (CDFW_CONSOLE (EVENT_CHANNEL (e1)))) ||
       CONSOLE_X_P (XCONSOLE (CDFW_CONSOLE (EVENT_CHANNEL (e2)))))
     return 0;
@@ -2020,11 +1923,7 @@
 {
   Lisp_Object console = CDFW_CONSOLE (EVENT_CHANNEL (e));
   if (CONSOLE_X_P (XCONSOLE (console)))
-#ifdef USE_KKCC
-    return (XMAGIC_DATA_X_EVENT (EVENT_DATA(e))).xany.serial;
-#else /* not USE_KKCC */
-    return e->event.magic.underlying_x_event.xany.serial;
-#endif /* not USE_KKCC */
+    return (EVENT_MAGIC_X_EVENT (e)).xany.serial;
   return 0;
 }
 
@@ -2032,11 +1931,7 @@
 emacs_Xt_handle_magic_event (Lisp_Event *emacs_event)
 {
   /* This function can GC */
-#ifdef USE_KKCC
-  XEvent *event = &XMAGIC_DATA_X_EVENT (EVENT_DATA(emacs_event));
-#else /* not USE_KKCC */
-  XEvent *event = &emacs_event->event.magic.underlying_x_event;
-#endif /* not USE_KKCC */
+  XEvent *event = &EVENT_MAGIC_X_EVENT (emacs_event);
   struct frame *f = XFRAME (EVENT_CHANNEL (emacs_event));
 
   if (!FRAME_LIVE_P (f) || DEVICE_X_BEING_DELETED (XDEVICE (FRAME_DEVICE (f))))
@@ -2286,19 +2181,11 @@
   assert (timeout);
   completed_timeouts = completed_timeouts->next;
   /* timeout events have nil as channel */
-#ifdef USE_KKCC
-  set_event_type(emacs_event, timeout_event);
+  set_event_type (emacs_event, timeout_event);
   SET_EVENT_TIMESTAMP_ZERO (emacs_event); /* #### wrong!! */
-  XSET_TIMEOUT_DATA_INTERVAL_ID (EVENT_DATA (emacs_event), timeout->id);
-  XSET_TIMEOUT_DATA_FUNCTION (EVENT_DATA (emacs_event), Qnil);
-  XSET_TIMEOUT_DATA_OBJECT (EVENT_DATA (emacs_event), Qnil);
-#else /* not USE_KKCC */
-  emacs_event->event_type = timeout_event;
-  emacs_event->timestamp  = 0; /* #### wrong!! */
-  emacs_event->event.timeout.interval_id = timeout->id;
-  emacs_event->event.timeout.function = Qnil;
-  emacs_event->event.timeout.object = Qnil;
-#endif /* not USE_KKCC */
+  SET_EVENT_TIMEOUT_INTERVAL_ID (emacs_event, timeout->id);
+  SET_EVENT_TIMEOUT_FUNCTION (emacs_event, Qnil);
+  SET_EVENT_TIMEOUT_OBJECT (emacs_event, Qnil);
   Blocktype_free (the_Xt_timeout_blocktype, timeout);
 }
 
@@ -2552,15 +2439,9 @@
 	  filedesc_with_input[i] = Qnil;
 	  process_events_occurred--;
 	  /* process events have nil as channel */
-#ifdef USE_KKCC
 	  set_event_type (emacs_event, process_event);
 	  SET_EVENT_TIMESTAMP_ZERO (emacs_event); /* #### */
-	  XSET_PROCESS_DATA_PROCESS (EVENT_DATA (emacs_event), process);
-#else /* not USE_KKCC */
-	  emacs_event->event_type = process_event;
-	  emacs_event->timestamp  = 0; /* #### */
-	  emacs_event->event.process.process = process;
-#endif /* not USE_KKCC */
+	  SET_EVENT_PROCESS_PROCESS (emacs_event, process);
 	  return;
 	}
     }
@@ -2857,20 +2738,6 @@
 /*			get the next event from Xt			*/
 /************************************************************************/
 
-static Lisp_Object dispatch_event_queue, dispatch_event_queue_tail;
-
-void
-enqueue_Xt_dispatch_event (Lisp_Object event)
-{
-  enqueue_event (event, &dispatch_event_queue, &dispatch_event_queue_tail);
-}
-
-static Lisp_Object
-dequeue_Xt_dispatch_event (void)
-{
-  return dequeue_event (&dispatch_event_queue, &dispatch_event_queue_tail);
-}
-
 /* This business exists because menu events "happen" when
    menubar_selection_callback() is called from somewhere deep
    within XtAppProcessEvent in emacs_Xt_next_event().  The
@@ -2886,18 +2753,11 @@
 {
   Lisp_Object event = Fmake_event (Qnil, Qnil);
 
-#ifdef USE_KKCC
   XSET_EVENT_TYPE (event, misc_user_event);
   XSET_EVENT_CHANNEL (event, channel);
-  XSET_MISC_USER_DATA_FUNCTION (XEVENT_DATA (event), function);
-  XSET_MISC_USER_DATA_OBJECT (XEVENT_DATA (event), object);
-#else /* not USE_KKCC */
-  XEVENT (event)->event_type = misc_user_event;
-  XEVENT (event)->channel = channel;
-  XEVENT (event)->event.eval.function = function;
-  XEVENT (event)->event.eval.object = object;
-#endif /* not USE_KKCC */
-  enqueue_Xt_dispatch_event (event);
+  XSET_EVENT_MISC_USER_FUNCTION (event, function);
+  XSET_EVENT_MISC_USER_OBJECT (event, object);
+  enqueue_dispatch_event (event);
 }
 
 static void
@@ -2951,7 +2811,7 @@
     {
       Lisp_Object event, event2;
       event2 = wrap_event (emacs_event);
-      event = dequeue_Xt_dispatch_event ();
+      event = dequeue_dispatch_event ();
       Fcopy_event (event, event2);
       Fdeallocate_event (event);
     }
@@ -2967,15 +2827,9 @@
       /* A dummy event, so that a cycle of the command loop will occur. */
       fake_event_occurred = 0;
       /* eval events have nil as channel */
-#ifdef USE_KKCC
       set_event_type (emacs_event, eval_event);
-      XSET_EVAL_DATA_FUNCTION (EVENT_DATA (emacs_event), Qidentity);
-      XSET_EVAL_DATA_OBJECT (EVENT_DATA (emacs_event), Qnil);
-#else /* not USE_KKCC */
-      emacs_event->event_type = eval_event;
-      emacs_event->event.eval.function = Qidentity;
-      emacs_event->event.eval.object = Qnil;
-#endif /* not USE_KKCC */
+      SET_EVENT_EVAL_FUNCTION (emacs_event, Qidentity);
+      SET_EVENT_EVAL_OBJECT (emacs_event, Qnil);
     }
   else /* if (process_events_occurred) */
     Xt_process_to_emacs_event (emacs_event);
@@ -2996,135 +2850,18 @@
     describe_event (event, Qexternal_debugging_output);
 #endif /* DEBUG_XEMACS */
   if (x_event_to_emacs_event (event, XEVENT (emacs_event)))
-    enqueue_Xt_dispatch_event (emacs_event);
+    enqueue_dispatch_event (emacs_event);
   else
     Fdeallocate_event (emacs_event);
 }
 
 
 /************************************************************************/
-/*                      input pending / C-g checking                    */
+/*                            input pending                             */
 /************************************************************************/
 
-static Bool
-quit_char_predicate (Display *display, XEvent *event, XPointer data)
-{
-  struct device *d = get_device_from_display (display);
-  struct x_device *xd = DEVICE_X_DATA (d);
-  char c, quit_char;
-  Bool *critical = (Bool *) data;
-  Lisp_Object keysym;
-
-  if (critical)
-    *critical = False;
-  if ((event->type != KeyPress) ||
-      (! x_any_window_to_frame (d, event->xany.window)) ||
-      (event->xkey.state
-       & (xd->MetaMask | xd->HyperMask | xd->SuperMask | xd->AltMask)))
-    return 0;
-
-  /* This duplicates some code that exists elsewhere, but it's relatively
-     fast and doesn't cons. */
-  keysym = x_to_emacs_keysym (&event->xkey, 1);
-  if (NILP (keysym)) return 0;
-  if (CHAR_OR_CHAR_INTP (keysym))
-    c = XCHAR_OR_CHAR_INT (keysym);
-  /* Highly doubtful that these are the quit character, but... */
-  else if (EQ (keysym, QKbackspace))	c = '\b';
-  else if (EQ (keysym, QKtab))		c = '\t';
-  else if (EQ (keysym, QKlinefeed))	c = '\n';
-  else if (EQ (keysym, QKreturn))	c = '\r';
-  else if (EQ (keysym, QKescape))	c = 27;
-  else if (EQ (keysym, QKspace))	c = ' ';
-  else if (EQ (keysym, QKdelete))	c = 127;
-  else return 0;
-
-  if (event->xkey.state & xd->MetaMask)     c |= 0x80;
-  if ((event->xkey.state & ControlMask) && !(c >= 'A' && c <= 'Z'))
-    c &= 0x1F;			/* unshifted control characters */
-  quit_char = CONSOLE_QUIT_CHAR (XCONSOLE (DEVICE_CONSOLE (d)));
-  if (c == quit_char)
-    return True;
-  /* If we've got Control-Shift-G instead of Control-G, that means
-     we have a critical_quit.  Caps_Lock is its own modifier, so it
-     won't cause ^G to act differently than before. */
-  if (event->xkey.state & ControlMask)  c &= 0x1F;
-  if (c == quit_char)
-    {
-      if (critical) *critical = True;
-      return True;
-    }
-  return False;
-}
-
-/* This scans the X input queue for a KeyPress event that matches the
-   quit character, and sets Vquit_flag.  This is called from the
-   QUIT macro to determine whether we should quit.
-
-   In a SIGIO world, this won't be called unless a SIGIO has happened
-   since the last time we checked.
-
-   In a non-SIGIO world, this is called from emacs_Xt_event_pending_p
-   (which is called from input_pending_p).
- */
-static void
-x_check_for_quit_char (Display *display)
-{
-  XEvent event;
-  int queued;
-  Bool critical_quit = False;
-  XEventsQueued (display, QueuedAfterReading);
-  queued = XCheckIfEvent (display, &event,
-			  quit_char_predicate,
-			  (char *) &critical_quit);
-  if (queued)
-    {
-      Vquit_flag = (critical_quit ? Qcritical : Qt);
-      /* don't put the event back onto the queue.  Those functions that
-	 wanted to read a ^G directly have arranged to do this. */
-    }
-}
-
 static void
-check_for_tty_quit_char (struct device *d)
-{
-  SELECT_TYPE temp_mask;
-  int infd = DEVICE_INFD (d);
-  struct console *con = XCONSOLE (DEVICE_CONSOLE (d));
-  Ichar quit_char = CONSOLE_QUIT_CHAR (con);
-
-  FD_ZERO (&temp_mask);
-  FD_SET (infd, &temp_mask);
-
-  while (1)
-    {
-      Lisp_Object event;
-      Ichar the_char;
-
-      if (!poll_fds_for_input (temp_mask))
-	return;
-
-      event = Fmake_event (Qnil, Qnil);
-      if (!read_event_from_tty_or_stream_desc (XEVENT (event), con))
-	/* EOF, or something ... */
-	return;
-      /* #### bogus.  quit-char should be allowed to be any sort
-	 of event. */
-      the_char = event_to_character (XEVENT (event), 1, 0, 0);
-      if (the_char >= 0 && the_char == quit_char)
-	{
-	  Vquit_flag = Qt;
-	  /* do not queue the C-g.  See above. */
-	  return;
-	}
-
-      /* queue the read event to be read for real later. */
-      enqueue_Xt_dispatch_event (event);
-    }
-}
-
-static void
-emacs_Xt_quit_p (void)
+emacs_Xt_drain_queue (void)
 {
   Lisp_Object devcons, concons;
   CONSOLE_LOOP (concons)
@@ -3136,43 +2873,22 @@
       CONSOLE_DEVICE_LOOP (devcons, con)
 	{
 	  struct device *d;
+	  Display *display;
 	  d = XDEVICE (XCAR (devcons));
-
 	  if (DEVICE_X_P (d) && DEVICE_X_DISPLAY (d))
-	    /* emacs may be exiting */
-	    x_check_for_quit_char (DEVICE_X_DISPLAY (d));
-	  else if (DEVICE_TTY_P (d))
-	    check_for_tty_quit_char (d);
+	    {
+	      display = DEVICE_X_DISPLAY (d);
+	      while (XEventsQueued (display, QueuedAfterReading))
+		XtAppProcessEvent (Xt_app_con, XtIMXEvent);
+	    }
 	}
     }
-}
-
-static void
-drain_X_queue (void)
-{
-  Lisp_Object devcons, concons;
-  CONSOLE_LOOP (concons)
-  {
-    struct console *con = XCONSOLE (XCAR (concons));
-    if (!con->input_enabled)
-      continue;
-
-    CONSOLE_DEVICE_LOOP (devcons, con)
-    {
-      struct device* d;
-      Display* display;
-      d = XDEVICE (XCAR (devcons));
-      if (DEVICE_X_P (d) && DEVICE_X_DISPLAY (d)) {
-        display = DEVICE_X_DISPLAY (d);
-        while (XEventsQueued (display, QueuedAfterReading))
-          XtAppProcessEvent (Xt_app_con, XtIMXEvent);
-      }
-    }
-  }
   /*
-  while (XtAppPending (Xt_app_con) & XtIMXEvent)
+    while (XtAppPending (Xt_app_con) & XtIMXEvent)
     XtAppProcessEvent (Xt_app_con, XtIMXEvent);
   */
+
+  drain_tty_devices ();
 }
 
 static int
@@ -3203,7 +2919,7 @@
      even without this. */
 #ifndef SIGIO
   /* First check for C-g if necessary */
-  emacs_Xt_quit_p ();
+  event_stream_quit_p ();
 #endif
 #endif
 
@@ -3288,7 +3004,7 @@
          drain part of it, we may later on end up with events
          actually pending but detect_input_pending() returning
          false because there wasn't another SIGIO. */
-      drain_X_queue ();
+      emacs_Xt_drain_queue ();
 
       EVENT_CHAIN_LOOP (event, dispatch_event_queue)
         if (!user_p || command_event_p (event))
@@ -3298,6 +3014,26 @@
   return 0;
 }
 
+int
+check_if_pending_expose_event (struct device *dev)
+{
+  Display *d = DEVICE_X_DISPLAY (dev);
+  Lisp_Object event;
+
+  emacs_Xt_drain_queue ();
+
+  EVENT_CHAIN_LOOP (event, dispatch_event_queue)
+    if (XEVENT_TYPE (event) == magic_event)
+      {
+	XEvent *xev = &XEVENT_MAGIC_X_EVENT (event);
+	if (xev->type == Expose &&
+	    xev->xexpose.display == d)
+	  return 1;
+      }
+
+  return 0;
+}
+
 static int
 emacs_Xt_current_event_timestamp (struct console *c)
 {
@@ -3524,9 +3260,9 @@
 void
 reinit_vars_of_event_Xt (void)
 {
-  Xt_event_stream = xnew (struct event_stream);
+  Xt_event_stream = xnew_and_zero (struct event_stream);
   Xt_event_stream->event_pending_p 	 = emacs_Xt_event_pending_p;
-  Xt_event_stream->force_event_pending 	 = emacs_Xt_force_event_pending;
+  Xt_event_stream->force_event_pending_cb= emacs_Xt_force_event_pending;
   Xt_event_stream->next_event_cb	 = emacs_Xt_next_event;
   Xt_event_stream->handle_magic_event_cb = emacs_Xt_handle_magic_event;
   Xt_event_stream->format_magic_event_cb = emacs_Xt_format_magic_event;
@@ -3538,7 +3274,7 @@
   Xt_event_stream->unselect_console_cb 	 = emacs_Xt_unselect_console;
   Xt_event_stream->select_process_cb 	 = emacs_Xt_select_process;
   Xt_event_stream->unselect_process_cb 	 = emacs_Xt_unselect_process;
-  Xt_event_stream->quit_p_cb		 = emacs_Xt_quit_p;
+  Xt_event_stream->drain_queue_cb	 = emacs_Xt_drain_queue;
   Xt_event_stream->create_io_streams_cb  = emacs_Xt_create_io_streams;
   Xt_event_stream->delete_io_streams_cb  = emacs_Xt_delete_io_streams;
   Xt_event_stream->current_event_timestamp_cb =
@@ -3557,11 +3293,6 @@
 {
   reinit_vars_of_event_Xt ();
 
-  dispatch_event_queue = Qnil;
-  staticpro (&dispatch_event_queue);
-  dispatch_event_queue_tail = Qnil;
-  dump_add_root_object (&dispatch_event_queue_tail);
-
   DEFVAR_BOOL ("x-allow-sendevents", &x_allow_sendevents /*
 *Non-nil means to allow synthetic events.  Nil means they are ignored.
 Beware: allowing emacs to process SendEvents opens a big security hole.
--- a/src/event-gtk.c	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/event-gtk.c	Sun Jan 12 11:08:22 2003 +0000
@@ -88,8 +88,6 @@
 Lisp_Object Qkey_mapping;
 Lisp_Object Qsans_modifiers;
 
-static void enqueue_gtk_dispatch_event (Lisp_Object event);
-
 #define IS_MODIFIER_KEY(keysym)  \
   ((((keysym) >= GDK_Shift_L) && ((keysym) <= GDK_Hyper_R)) \
    || ((keysym) == GDK_Mode_switch) \
@@ -213,7 +211,7 @@
   if (CONSOLE_GTK_P (XCONSOLE (console)))
     write_c_string
       (pstream,
-       gtk_event_name (emacs_event->event.magic.underlying_gdk_event.type));
+       gtk_event_name (EVENT_MAGIC_GDK_EVENT (emacs_event).type));
 }
 
 static int
@@ -221,8 +219,8 @@
 {
   if (CONSOLE_GTK_P (XCONSOLE (CDFW_CONSOLE (EVENT_CHANNEL (e1)))) &&
       CONSOLE_GTK_P (XCONSOLE (CDFW_CONSOLE (EVENT_CHANNEL (e2)))))
-    return (!memcmp (&e1->event.magic.underlying_gdk_event,
-		     &e2->event.magic.underlying_gdk_event,
+    return (!memcmp (&EVENT_MAGIC_GDK_EVENT (e1),
+		     &EVENT_MAGIC_GDK_EVENT (e2),
 		     sizeof (GdkEvent)));
   if (CONSOLE_GTK_P (XCONSOLE (CDFW_CONSOLE (EVENT_CHANNEL (e1)))) ||
       CONSOLE_GTK_P (XCONSOLE (CDFW_CONSOLE (EVENT_CHANNEL (e2)))))
@@ -235,7 +233,7 @@
 {
   Lisp_Object console = CDFW_CONSOLE (EVENT_CHANNEL (e));
   if (CONSOLE_GTK_P (XCONSOLE (console)))
-    return memory_hash (&e->event.magic.underlying_gdk_event,
+    return memory_hash (&EVENT_MAGIC_GDK_EVENT (e),
 			sizeof (GdkEvent));
   return 0;
 }
@@ -244,7 +242,7 @@
 emacs_gtk_handle_magic_event (struct Lisp_Event *emacs_event)
 {
   /* This function can GC */
-  GdkEvent *event = &emacs_event->event.magic.underlying_gdk_event;
+  GdkEvent *event = &EVENT_MAGIC_GDK_EVENT (emacs_event);
   struct frame *f = XFRAME (EVENT_CHANNEL (emacs_event));
 
   if (!FRAME_LIVE_P (f))
@@ -467,13 +465,13 @@
       while ((ch = Lstream_get_ichar (istr)) != EOF)
 	{
 	  Lisp_Object emacs_event = Fmake_event (Qnil, Qnil);
-	  struct Lisp_Event *ev = XEVENT (emacs_event);
-	  ev->channel       = DEVICE_CONSOLE (d);
-	  ev->event_type    = key_press_event;
+	  Lisp_Event *ev          = XEVENT (emacs_event);
+	  ev->channel	            = DEVICE_CONSOLE (d);
 	  ev->timestamp	    = event->time;
-	  ev->event.key.modifiers = 0;
-	  ev->event.key.keysym    = make_char (ch);
-	  enqueue_gtk_dispatch_event (emacs_event);
+	  XSET_EVENT_TYPE (emacs_event, key_press_event);
+	  XSET_EVENT_KEY_MODIFIERS (emacs_event, 0);
+	  XSET_EVENT_KEY_KEYSYM (emacs_event, make_char (ch));
+	  enqueue_dispatch_event (emacs_event);
 	}
       Lstream_close (istr);
       UNGCPRO;
@@ -619,10 +617,12 @@
   struct GTK_timeout *timeout = completed_timeouts;
   assert (timeout);
   completed_timeouts = completed_timeouts->next;
-  emacs_event->event_type = timeout_event;
   /* timeout events have nil as channel */
-  emacs_event->timestamp  = 0; /* #### wrong!! */
-  emacs_event->event.timeout.interval_id = timeout->id;
+  set_event_type (emacs_event, timeout_event);
+  SET_EVENT_TIMESTAMP_ZERO (emacs_event); /* #### wrong!! */
+  SET_EVENT_TIMEOUT_INTERVAL_ID (emacs_event, timeout->id);
+  SET_EVENT_TIMEOUT_FUNCTION (emacs_event, Qnil);
+  SET_EVENT_TIMEOUT_OBJECT (emacs_event, Qnil);
   Blocktype_free (the_GTK_timeout_blocktype, timeout);
 }
 
@@ -862,22 +862,24 @@
 gtk_process_to_emacs_event (struct Lisp_Event *emacs_event)
 {
   int i;
-  Lisp_Object process;
 
   assert (process_events_occurred > 0);
+
   for (i = 0; i < MAXDESC; i++)
     {
-      process = filedesc_with_input[i];
+      Lisp_Object process = filedesc_with_input[i];
       if (PROCESSP (process))
-	break;
+	{
+	  filedesc_with_input[i] = Qnil;
+	  process_events_occurred--;
+	  /* process events have nil as channel */
+	  set_event_type (emacs_event, process_event);
+	  SET_EVENT_TIMESTAMP_ZERO (emacs_event); /* #### */
+	  SET_EVENT_PROCESS_PROCESS (emacs_event, process);
+	  return;
+	}
     }
-  assert (i < MAXDESC);
-  filedesc_with_input[i] = Qnil;
-  process_events_occurred--;
-  /* process events have nil as channel */
-  emacs_event->event_type = process_event;
-  emacs_event->timestamp  = 0; /* #### */
-  emacs_event->event.process.process = process;
+  abort ();
 }
 
 static void
@@ -968,13 +970,11 @@
 
   GCPRO4 (l_type, l_data, l_dndlist, l_item);
 
-  ev->event_type = misc_user_event;
-  ev->timestamp = time;
-
-  ev->channel = wrap_frame (f);
-
-  ev->event.misc.x = x;
-  ev->event.misc.y = y;
+  set_event_type (ev, misc_user_event);
+  SET_EVENT_CHANNEL (ev, wrap_frame (f));
+  SET_EVENT_TIMESTAMP (ev, time);
+  SET_EVENT_MISC_USER_X (ev, x);
+  SET_EVENT_MISC_USER_Y (ev, y);
 
   if (data->type == preferred_targets[TARGET_URI_LIST])
     {
@@ -1034,13 +1034,14 @@
 						 data->length, Qbinary)));
     }
 
-  ev->event.misc.function = Qdragdrop_drop_dispatch;
-  ev->event.misc.object = Fcons (l_type, l_dndlist);
+
+  SET_EVENT_MISC_USER_FUNCTION (ev, Qdragdrop_drop_dispatch);
+  SET_EVENT_MISC_USER_OBJECT (ev, Fcons (l_type, l_dndlist));
 
   UNGCPRO;
 
   gtk_drag_finish (context, TRUE, FALSE, time);
-  enqueue_gtk_dispatch_event (event);
+  enqueue_dispatch_event (event);
 }
 
 gboolean
@@ -1122,20 +1123,6 @@
 /*			get the next event from gtk			*/
 /************************************************************************/
 
-static Lisp_Object dispatch_event_queue, dispatch_event_queue_tail;
-
-static void
-enqueue_gtk_dispatch_event (Lisp_Object event)
-{
-  enqueue_event (event, &dispatch_event_queue, &dispatch_event_queue_tail);
-}
-
-static Lisp_Object
-dequeue_gtk_dispatch_event (void)
-{
-  return dequeue_event (&dispatch_event_queue, &dispatch_event_queue_tail);
-}
-
 /* This business exists because menu events "happen" when
    menubar_selection_callback() is called from somewhere deep
    within XtAppProcessEvent in emacs_Xt_next_event().  The
@@ -1151,12 +1138,11 @@
 {
   Lisp_Object event = Fmake_event (Qnil, Qnil);
 
-  XEVENT (event)->event_type = misc_user_event;
-  XEVENT (event)->channel = channel;
-  XEVENT (event)->event.eval.function = function;
-  XEVENT (event)->event.eval.object = object;
-
-  enqueue_gtk_dispatch_event (event);
+  XSET_EVENT_TYPE (event, misc_user_event);
+  XSET_EVENT_CHANNEL (event, channel);
+  XSET_EVENT_MISC_USER_FUNCTION (event, function);
+  XSET_EVENT_MISC_USER_OBJECT (event, object);
+  enqueue_dispatch_event (event);
 }
 
 static void
@@ -1177,7 +1163,7 @@
     {
       Lisp_Object event, event2;
       event2 = wrap_event (emacs_event);
-      event = dequeue_gtk_dispatch_event ();
+      event = dequeue_dispatch_event ();
       Fcopy_event (event, event2);
       Fdeallocate_event (event);
     }
@@ -1193,9 +1179,9 @@
       /* A dummy event, so that a cycle of the command loop will occur. */
       fake_event_occurred = 0;
       /* eval events have nil as channel */
-      emacs_event->event_type = eval_event;
-      emacs_event->event.eval.function = Qidentity;
-      emacs_event->event.eval.object = Qnil;
+      set_event_type (emacs_event, eval_event);
+      SET_EVENT_EVAL_FUNCTION (emacs_event, Qidentity);
+      SET_EVENT_EVAL_OBJECT (emacs_event, Qnil);
     }
   else /* if (process_events_occurred) */
     gtk_process_to_emacs_event (emacs_event);
@@ -1376,24 +1362,25 @@
 		  }
 	      }
 		
-	    emacs_event->event_type	     = key_press_event;
-	    emacs_event->timestamp	     = key_event->time;
-	    emacs_event->event.key.modifiers = modifiers;
-	    emacs_event->event.key.keysym    = keysym;
+	    set_event_type (emacs_event, key_press_event);
+	    SET_EVENT_TIMESTAMP (emacs_event, key_event->time);
+	    SET_EVENT_KEY_MODIFIERS (emacs_event, modifiers);
+	    SET_EVENT_KEY_KEYSYM (emacs_event, keysym);
 	  }
 	else                    /* Mouse press/release event */
 	  {
 	    GdkEventButton *button_event = &gdk_event->button;
-	    emacs_event->channel = wrap_frame (frame);
 
-	    emacs_event->event_type = (button_event->type == GDK_BUTTON_RELEASE) ?
-	      button_release_event : button_press_event;
+	    set_event_type (emacs_event,
+			    button_event->type == GDK_BUTTON_RELEASE ?
+			    button_release_event : button_press_event);
+            SET_EVENT_CHANNEL (emacs_event, wrap_frame (frame));
 
-	    emacs_event->event.button.modifiers = modifiers;
-	    emacs_event->timestamp		= button_event->time;
-	    emacs_event->event.button.button	= button_event->button;
-	    emacs_event->event.button.x		= button_event->x;
-	    emacs_event->event.button.y		= button_event->y;
+	    SET_EVENT_BUTTON_MODIFIERS (emacs_event, modifiers);
+	    SET_EVENT_TIMESTAMP (emacs_event, button_event->time);
+	    SET_EVENT_BUTTON_BUTTON (emacs_event, button_event->button);
+	    SET_EVENT_BUTTON_X (emacs_event, button_event->x);
+	    SET_EVENT_BUTTON_Y (emacs_event, button_event->y);
 	  }
       }
       break;
@@ -1419,11 +1406,12 @@
 
         DEVICE_GTK_MOUSE_TIMESTAMP (d) = ev->time;
 
-        emacs_event->channel = wrap_frame (frame);
-        emacs_event->event_type	    = pointer_motion_event;
-        emacs_event->timestamp      = ev->time;
-        emacs_event->event.motion.x = x;
-        emacs_event->event.motion.y = y;
+        SET_EVENT_CHANNEL (emacs_event, wrap_frame (frame));
+        set_event_type (emacs_event, pointer_motion_event);
+        SET_EVENT_TIMESTAMP (emacs_event, ev->time);
+        SET_EVENT_MOTION_X (emacs_event, x);
+        SET_EVENT_MOTION_Y (emacs_event, y);
+
         if (mask & GDK_SHIFT_MASK)	modifiers |= XEMACS_MOD_SHIFT;
         if (mask & GDK_CONTROL_MASK)	modifiers |= XEMACS_MOD_CONTROL;
         if (mask & gd->MetaMask)	modifiers |= XEMACS_MOD_META;
@@ -1438,7 +1426,7 @@
 
         /* Currently ignores Shift_Lock but probably shouldn't
            (but it definitely should ignore Caps_Lock). */
-        emacs_event->event.motion.modifiers = modifiers;
+        SET_EVENT_MOTION_MODIFIERS (emacs_event, modifiers);
       }
     break;
 
@@ -1465,7 +1453,7 @@
 
     if (gtk_event_to_emacs_event (GTK_XEMACS_FRAME (widget), event, XEVENT (emacs_event)))
     {
-	enqueue_gtk_dispatch_event (emacs_event);
+	enqueue_dispatch_event (emacs_event);
 	return (TRUE);
     }
     else
@@ -1498,7 +1486,7 @@
     struct frame *frame = (struct frame *) closure;
     Lisp_Object lisp_event = Fmake_event (Qnil, Qnil);
     struct Lisp_Event *emacs_event = XEVENT (lisp_event);
-    GdkEvent *gdk_event_copy = &emacs_event->event.magic.underlying_gdk_event;
+    GdkEvent *gdk_event_copy = &EVENT_MAGIC_GDK_EVENT (emacs_event);
     struct device *d = XDEVICE (FRAME_DEVICE (frame));
     gboolean ignore_p = FALSE;
 
@@ -1539,7 +1527,7 @@
       }
     else
       {
-	enqueue_gtk_dispatch_event (lisp_event);
+	enqueue_dispatch_event (lisp_event);
 	return (TRUE);
       }
 }
@@ -1548,83 +1536,18 @@
 /************************************************************************/
 /*                      input pending / C-g checking                    */
 /************************************************************************/
-static void
-gtk_check_for_quit_char (struct device *d);
-
-static void
-check_for_tty_quit_char (struct device *d)
-{
-  SELECT_TYPE temp_mask;
-  int infd = DEVICE_INFD (d);
-  struct console *con = XCONSOLE (DEVICE_CONSOLE (d));
-  Ichar quit_char = CONSOLE_QUIT_CHAR (con);
-
-  FD_ZERO (&temp_mask);
-  FD_SET (infd, &temp_mask);
-
-  while (1)
-    {
-      Lisp_Object event;
-      Ichar the_char;
-
-      if (!poll_fds_for_input (temp_mask))
-	return;
-
-      event = Fmake_event (Qnil, Qnil);
-      if (!read_event_from_tty_or_stream_desc (XEVENT (event), con))
-	/* EOF, or something ... */
-	return;
-      /* #### bogus.  quit-char should be allowed to be any sort
-	 of event. */
-      the_char = event_to_character (XEVENT (event), 1, 0, 0);
-      if (the_char >= 0 && the_char == quit_char)
-	{
-	  Vquit_flag = Qt;
-	  /* do not queue the C-g.  See above. */
-	  return;
-	}
-
-      /* queue the read event to be read for real later. */
-      enqueue_gtk_dispatch_event (event);
-    }
-}
-
-static void
-emacs_gtk_quit_p (void)
-{
-  Lisp_Object devcons, concons;
-
-  CONSOLE_LOOP (concons)
-    {
-      struct console *con = XCONSOLE (XCAR (concons));
-      if (!con->input_enabled)
-	continue;
-
-      CONSOLE_DEVICE_LOOP (devcons, con)
-	{
-	  struct device *d;
-	  d = XDEVICE (XCAR (devcons));
-
-	  if (DEVICE_GTK_P (d))
-	    /* emacs may be exiting */
-	    gtk_check_for_quit_char (d);
-	  else if (DEVICE_TTY_P (d))
-	    check_for_tty_quit_char (d);
-	}
-    }
-}
 
 #include <gdk/gdkx.h>
 
 static void
-drain_gtk_queue (void)
+emacs_gtk_drain_queue (void)
 
 {
   /* We can't just spin through here and wait for GTKs idea of the
      event queue to get empty, or the queue never gets drained.  The
      situation is as follows.  A process event gets signalled, we put
      it on the queue, then we go into Fnext_event(), which calls
-     drain_gtk_queue().  But gtk_events_pending() will always return
+     emacs_gtk_drain_queue().  But gtk_events_pending() will always return
      TRUE if there are file-descriptor (aka our process) events
      pending.  Using GDK_events_pending() only shows us windowing
      system events.
@@ -1632,6 +1555,8 @@
   if (GDK_DISPLAY ())
     while (gdk_events_pending ())
       gtk_main_iteration ();
+
+  drain_tty_devices ();
 }
 
 static int
@@ -1737,7 +1662,7 @@
          actually pending but detect_input_pending() returning
          false because there wasn't another SIGIO. */
 
-      drain_gtk_queue ();
+      emacs_gtk_drain_queue ();
 
       EVENT_CHAIN_LOOP (event, dispatch_event_queue)
         if (!user_p || command_event_p (event))
@@ -1769,7 +1694,7 @@
 
 void reinit_vars_of_event_gtk (void)
 {
-  gtk_event_stream = xnew (struct event_stream);
+  gtk_event_stream = xnew_and_zero (struct event_stream);
   gtk_event_stream->event_pending_p 	= emacs_gtk_event_pending_p;
   gtk_event_stream->next_event_cb	= emacs_gtk_next_event;
   gtk_event_stream->handle_magic_event_cb= emacs_gtk_handle_magic_event;
@@ -1782,10 +1707,10 @@
   gtk_event_stream->unselect_console_cb = emacs_gtk_unselect_console;
   gtk_event_stream->select_process_cb 	= emacs_gtk_select_process;
   gtk_event_stream->unselect_process_cb = emacs_gtk_unselect_process;
-  gtk_event_stream->quit_p_cb		= emacs_gtk_quit_p;
+  gtk_event_stream->drain_queue_cb	= emacs_gtk_drain_queue;
   gtk_event_stream->create_io_streams_cb= emacs_gtk_create_io_streams;
   gtk_event_stream->delete_io_streams_cb= emacs_gtk_delete_io_streams;
-  gtk_event_stream->force_event_pending	 = emacs_gtk_force_event_pending;
+  gtk_event_stream->force_event_pending_cb= emacs_gtk_force_event_pending;
 
   the_GTK_timeout_blocktype = Blocktype_new (struct GTK_timeout_blocktype);
 
@@ -1798,11 +1723,6 @@
 {
   reinit_vars_of_event_gtk ();
 
-  dispatch_event_queue = Qnil;
-  staticpro (&dispatch_event_queue);
-  dispatch_event_queue_tail = Qnil;
-  staticpro (&dispatch_event_queue_tail);
-
   DEFVAR_BOOL ("gtk-allow-sendevents", &gtk_allow_sendevents /*
 *Non-nil means to allow synthetic events.  Nil means they are ignored.
 Beware: allowing emacs to process SendEvents opens a big security hole.
@@ -2138,104 +2058,3 @@
   return 0;
 }
 #endif
-
-struct _quit_predicate_closure {
-  struct device *device;
-  Bool *critical;
-};
-
-static Bool
-quit_char_predicate (Display *display, XEvent *event, XPointer data)
-{
-  struct _quit_predicate_closure *cl = (struct _quit_predicate_closure *) data;
-  struct device *d = cl->device;
-  struct frame *f = NULL;
-  struct gtk_device *gd = DEVICE_GTK_DATA (d);
-  char c, quit_char;
-  Bool *critical = cl->critical;
-  Lisp_Object keysym;
-  GdkWindow *window = gdk_window_lookup (event->xany.window);
-  guint32 keycode = 0;
-  GdkEventKey gdk_event;
-
-  if (window)
-    f = gtk_any_window_to_frame (d, window);
-
-  if (critical)
-    *critical = False;
-
-  if ((event->type != KeyPress) ||
-      (! window) ||
-      (! f) ||
-      (event->xkey.state
-       & (gd->MetaMask | gd->HyperMask | gd->SuperMask | gd->AltMask)))
-    {
-      return 0;
-    }
-
-  {
-    char dummy[256];
-    XLookupString (&(event->xkey), dummy, 200, (KeySym *)&keycode, 0);
-  }
-
-  memset (&gdk_event, 0, sizeof (gdk_event));
-  gdk_event.type = GDK_KEY_PRESS;
-  gdk_event.window = window;
-  gdk_event.keyval = keycode;
-  gdk_event.state = event->xkey.state;
-
-  /* This duplicates some code that exists elsewhere, but it's relatively
-     fast and doesn't cons. */
-  keysym = gtk_to_emacs_keysym (d, &gdk_event, 1);
-  if (NILP (keysym)) return 0;
-  if (CHAR_OR_CHAR_INTP (keysym))
-    c = XCHAR_OR_CHAR_INT (keysym);
-  /* Highly doubtful that these are the quit character, but... */
-  else if (EQ (keysym, QKbackspace))	c = '\b';
-  else if (EQ (keysym, QKtab))		c = '\t';
-  else if (EQ (keysym, QKlinefeed))	c = '\n';
-  else if (EQ (keysym, QKreturn))	c = '\r';
-  else if (EQ (keysym, QKescape))	c = 27;
-  else if (EQ (keysym, QKspace))	c = ' ';
-  else if (EQ (keysym, QKdelete))	c = 127;
-  else return 0;
-
-  if (event->xkey.state & gd->MetaMask)     c |= 0x80;
-  if ((event->xkey.state & ControlMask) && !(c >= 'A' && c <= 'Z'))
-    c &= 0x1F;			/* unshifted control characters */
-  quit_char = CONSOLE_QUIT_CHAR (XCONSOLE (DEVICE_CONSOLE (d)));
-
-  if (c == quit_char)
-    return True;
-  /* If we've got Control-Shift-G instead of Control-G, that means
-     we have a critical_quit.  Caps_Lock is its own modifier, so it
-     won't cause ^G to act differently than before. */
-  if (event->xkey.state & ControlMask)  c &= 0x1F;
-  if (c == quit_char)
-    {
-      if (critical) *critical = True;
-      return True;
-    }
-  return False;
-}
-
-static void
-gtk_check_for_quit_char (struct device *d)
-{
-  XEvent event;
-  int queued;
-  Bool critical_quit = False;
-  struct _quit_predicate_closure closure;
-
-  XEventsQueued (GDK_DISPLAY (), QueuedAfterReading);
-
-  closure.device = d;
-  closure.critical = &critical_quit;
-
-  queued = XCheckIfEvent (GDK_DISPLAY (), &event, quit_char_predicate, (char *) &closure);
-
-  if (queued)
-    {
-      Vquit_flag = (critical_quit ? Qcritical : Qt);
-    }
-}
--- a/src/event-msw.c	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/event-msw.c	Sun Jan 12 11:08:22 2003 +0000
@@ -73,6 +73,7 @@
 #include "sysdep.h"
 #include "window.h"
 
+#include "console-stream-impl.h"
 #include "console-msw-impl.h"
 #include "objects-msw-impl.h"
 
@@ -94,21 +95,12 @@
 #include "systime.h"
 #include "syswait.h"
 
-#ifdef CYGWIN
-#include "console-tty.h"
-#endif
-
 #ifdef HAVE_MENUBARS
 #define ADJR_MENUFLAG TRUE
 #else
 #define ADJR_MENUFLAG FALSE
 #endif
 
-/* Fake key modifier which is attached to a quit char event.
-   Removed upon dequeueing an event */
-#define FAKE_MOD_QUIT (1 << 20)
-#define FAKE_MOD_QUIT_CRITICAL (1 << 21)
-
 /* Timer ID used for button2 emulation */
 #define BUTTON_2_TIMER_ID 1
 
@@ -147,23 +139,18 @@
 #endif
 
 /*
- * Two separate queues, for efficiency, one (_u_) for user events, and
- * another (_s_) for non-user ones. We always return events out of the
- * first one until it is empty and only then proceed with the second
- * one.
+ * We use an additional queue, as well as the normal dispatch queue, for
+ * efficiency, the normal one for user events, and another (_s_) for non-user
+ * ones. We always return events out of the first one until it is empty and
+ * only then proceed with the second one.
  */
-static Lisp_Object mswindows_u_dispatch_event_queue, mswindows_u_dispatch_event_queue_tail;
-static Lisp_Object mswindows_s_dispatch_event_queue, mswindows_s_dispatch_event_queue_tail;
+static Lisp_Object mswindows_s_dispatch_event_queue;
+static Lisp_Object mswindows_s_dispatch_event_queue_tail;
 
 /* Brush for painting widgets */
 static HBRUSH widget_brush = 0;
 static LONG	last_widget_brushed = 0;
 
-/* Count of quit chars currently in the queue */
-/* Incremented in WM_[SYS]KEYDOWN handler in the mswindows_wnd_proc()
-   Decremented in mswindows_dequeue_dispatch_event() */
-int mswindows_quit_chars_count = 0;
-
 /* These are Lisp integers; see DEFVARS in this file for description. */
 int mswindows_dynamic_frame_resize;
 int mswindows_alt_by_itself_activates_menu;
@@ -690,7 +677,7 @@
   OVERLAPPED ov;		/* Overlapped I/O structure		     */
   void *buffer;			/* Buffer.                                   */
   DWORD bufsize;		/* Number of bytes last read		     */
-  DWORD charbpos;			/* Position in buffer for next fetch	     */
+  DWORD charbpos;		/* Position in buffer for next fetch	     */
   unsigned int error_p :1;	/* I/O Error seen			     */
   unsigned int eof_p :1;	/* EOF Error seen			     */
   unsigned int pending_p :1;	/* There is a pending I/O operation	     */
@@ -737,7 +724,8 @@
 	}
       else
 	{
-	  if (!GetOverlappedResult ((HANDLE)str->s, &str->ov, &str->bufsize, TRUE))
+	  if (!GetOverlappedResult ((HANDLE)str->s, &str->ov, &str->bufsize,
+				    TRUE))
 	    {
 	      if (GetLastError() == ERROR_HANDLE_EOF)
 		str->bufsize = 0;
@@ -783,7 +771,8 @@
       else
 	{
 	  DWORD dw_unused;
-	  if (!GetOverlappedResult ((HANDLE)str->s, &str->ov, &dw_unused, TRUE))
+	  if (!GetOverlappedResult ((HANDLE)str->s, &str->ov, &dw_unused,
+				    TRUE))
 	    str->error_p = 1;
 	  str->pending_p = 0;
 	}
@@ -927,12 +916,12 @@
 void
 mswindows_enqueue_dispatch_event (Lisp_Object event)
 {
-  int user_p = mswindows_user_event_p (XEVENT(event));
-  enqueue_event (event,
-		 user_p ? &mswindows_u_dispatch_event_queue :
-		 &mswindows_s_dispatch_event_queue,
-		 user_p ? &mswindows_u_dispatch_event_queue_tail :
-		 &mswindows_s_dispatch_event_queue_tail);
+  int user_p = mswindows_user_event_p (XEVENT (event));
+  if (user_p)
+    enqueue_dispatch_event (event);
+  else
+    enqueue_event (event, &mswindows_s_dispatch_event_queue,
+		   &mswindows_s_dispatch_event_queue_tail);
 
   /* Avoid blocking on WaitMessage */
   qxePostMessage (NULL, XM_BUMPQUEUE, 0, 0);
@@ -940,9 +929,6 @@
 
 /*
  * Add a misc-user event to the dispatch queue.
- *
- * Stuff it into our own dispatch queue, so we have something
- * to return from next_event callback.
  */
 void
 mswindows_enqueue_misc_user_event (Lisp_Object channel, Lisp_Object function,
@@ -950,22 +936,11 @@
 {
   Lisp_Object event = Fmake_event (Qnil, Qnil);
 
-#ifdef USE_KKCC
-
   XSET_EVENT_TYPE (event, misc_user_event);
   XSET_EVENT_CHANNEL (event, channel);
   XSET_EVENT_TIMESTAMP (event, GetTickCount());
-  XSET_MISC_USER_DATA_FUNCTION (XEVENT_DATA (event), function);
-  XSET_MISC_USER_DATA_OBJECT (XEVENT_DATA (event), object);
-#else /* not USE_KKCC */
-  Lisp_Event *e = XEVENT (event);
-
-  e->event_type = misc_user_event;
-  e->channel = channel;
-  e->timestamp = GetTickCount ();
-  e->event.misc.function = function;
-  e->event.misc.object = object;
-#endif /* not USE_KKCC */
+  XSET_EVENT_MISC_USER_FUNCTION (event, function);
+  XSET_EVENT_MISC_USER_OBJECT (event, object);
 
   mswindows_enqueue_dispatch_event (event);
 }
@@ -975,19 +950,10 @@
 {
   Lisp_Object emacs_event = Fmake_event (Qnil, Qnil);
 
-#ifdef USE_KKCC
-  XSET_EVENT_CHANNEL (emacs_event, (hwnd ? mswindows_find_frame (hwnd) : Qnil));
+  XSET_EVENT_CHANNEL (emacs_event, hwnd ? mswindows_find_frame (hwnd) : Qnil);
   XSET_EVENT_TIMESTAMP (emacs_event, GetMessageTime ());
   XSET_EVENT_TYPE (emacs_event, magic_event);
-  XSET_MAGIC_DATA_MSWINDOWS_EVENT (XEVENT_DATA(emacs_event), msg);
-#else /* not USE_KKCC */
-  Lisp_Event *event = XEVENT (emacs_event);
-
-  event->channel = hwnd ? mswindows_find_frame (hwnd) : Qnil;
-  event->timestamp = GetMessageTime();
-  event->event_type = magic_event;
-  EVENT_MSWINDOWS_MAGIC_TYPE (event) = msg;
-#endif /* not USE_KKCC */
+  XSET_EVENT_MAGIC_MSWINDOWS_EVENT (emacs_event, msg);
 
   mswindows_enqueue_dispatch_event (emacs_event);
 }
@@ -996,19 +962,11 @@
 mswindows_enqueue_process_event (Lisp_Process *p)
 {
   Lisp_Object emacs_event = Fmake_event (Qnil, Qnil);
-  Lisp_Event *event = XEVENT (emacs_event);
   Lisp_Object process = wrap_process (p);
 
-
-#ifdef USE_KKCC
   XSET_EVENT_TYPE (emacs_event, process_event);
-  XSET_EVENT_TIMESTAMP (emacs_event, GetTickCount());
-  XSET_PROCESS_DATA_PROCESS (XEVENT_DATA (emacs_event), process);
-#else /* not USE_KKCC */
-  event->event_type = process_event;
-  event->timestamp  = GetTickCount ();
-  event->event.process.process = process;
-#endif /* not USE_KKCC */
+  XSET_EVENT_TIMESTAMP (emacs_event, GetTickCount ());
+  XSET_EVENT_PROCESS_PROCESS (emacs_event, process);
 
   mswindows_enqueue_dispatch_event (emacs_event);
 }
@@ -1025,47 +983,27 @@
      recognition will fail */
 
   Lisp_Object emacs_event = Fmake_event (Qnil, Qnil);
-  Lisp_Event *event = XEVENT (emacs_event);
 
   mswindows_handle_sticky_modifiers (0, 0, downp, 0);
 
   if (downp)
     {
-#ifdef USE_KKCC
       XSET_EVENT_TYPE (emacs_event, button_press_event);
-#else /* not USE_KKCC */
-      event->event_type = button_press_event;
-#endif /* not USE_KKCC */
     }
   else
     {
-#ifdef USE_KKCC
       XSET_EVENT_TYPE (emacs_event, button_release_event);
-#else /* not USE_KKCC */
-      event->event_type = button_release_event;
-#endif /* not USE_KKCC */
     }
 
-#ifdef USE_KKCC
   XSET_EVENT_CHANNEL (emacs_event, mswindows_find_frame (hwnd));
   XSET_EVENT_TIMESTAMP (emacs_event, when);
-  XSET_BUTTON_DATA_BUTTON (XEVENT_DATA (emacs_event), 
+  XSET_EVENT_BUTTON_BUTTON (emacs_event, 
 	 (msg==WM_LBUTTONDOWN || msg==WM_LBUTTONUP) ? 1 :
 	 ((msg==WM_RBUTTONDOWN || msg==WM_RBUTTONUP) ? 3 : 2));
-  XSET_BUTTON_DATA_X (XEVENT_DATA (emacs_event), where.x);
-  XSET_BUTTON_DATA_Y (XEVENT_DATA (emacs_event), where.y);
-  XSET_BUTTON_DATA_MODIFIERS (XEVENT_DATA (emacs_event),
-			      mswindows_modifier_state (NULL, mods, 0));
-#else /* not USE_KKCC */
-  event->channel = mswindows_find_frame (hwnd);
-  event->timestamp = when;
-  event->event.button.button =
-    (msg==WM_LBUTTONDOWN || msg==WM_LBUTTONUP) ? 1 :
-    ((msg==WM_RBUTTONDOWN || msg==WM_RBUTTONUP) ? 3 : 2);
-  event->event.button.x = where.x;
-  event->event.button.y = where.y;
-  event->event.button.modifiers = mswindows_modifier_state (NULL, mods, 0);
-#endif /* not USE_KKCC */
+  XSET_EVENT_BUTTON_X (emacs_event, where.x);
+  XSET_EVENT_BUTTON_Y (emacs_event, where.y);
+  XSET_EVENT_BUTTON_MODIFIERS (emacs_event,
+			       mswindows_modifier_state (NULL, mods, 0));
 
   if (downp)
     {
@@ -1091,21 +1029,11 @@
 {
   Lisp_Object emacs_event = Fmake_event (Qnil, Qnil);
 
-#ifdef USE_KKCC
   XSET_EVENT_CHANNEL (emacs_event,  mswindows_find_console(hwnd));
   XSET_EVENT_TIMESTAMP (emacs_event, GetMessageTime());
   XSET_EVENT_TYPE (emacs_event, key_press_event);
-  XSET_KEY_DATA_KEYSYM (XEVENT_DATA (emacs_event), keysym);
-  XSET_KEY_DATA_MODIFIERS (XEVENT_DATA (emacs_event), mods);
-#else /* not USE_KKCC */
-  Lisp_Event *event = XEVENT(emacs_event);
-
-  event->channel = mswindows_find_console(hwnd);
-  event->timestamp = GetMessageTime();
-  event->event_type = key_press_event;
-  event->event.key.keysym = keysym;
-  event->event.key.modifiers = mods;
-#endif /* not USE_KKCC */
+  XSET_EVENT_KEY_KEYSYM (emacs_event, keysym);
+  XSET_EVENT_KEY_MODIFIERS (emacs_event, mods);
   mswindows_enqueue_dispatch_event (emacs_event);
   return emacs_event;
 }
@@ -1117,92 +1045,14 @@
 static Lisp_Object
 mswindows_dequeue_dispatch_event (void)
 {
-  Lisp_Object event;
-  Lisp_Event *sevt;
-
-  assert (!NILP(mswindows_u_dispatch_event_queue) ||
-	  !NILP(mswindows_s_dispatch_event_queue));
-
-  event = dequeue_event (
-			 NILP(mswindows_u_dispatch_event_queue) ?
-			 &mswindows_s_dispatch_event_queue :
-			 &mswindows_u_dispatch_event_queue,
-			 NILP(mswindows_u_dispatch_event_queue) ?
-			 &mswindows_s_dispatch_event_queue_tail :
-			 &mswindows_u_dispatch_event_queue_tail);
-
-#ifdef USE_KKCC
-  if (XEVENT_TYPE (event) == key_press_event
-      && (XKEY_DATA_MODIFIERS (XEVENT_DATA(event)) & FAKE_MOD_QUIT))
-    XSET_KEY_DATA_MODIFIERS (XEVENT_DATA (event), 
-			    XKEY_DATA_MODIFIERS (XEVENT_DATA (event)) &
-						 ~(FAKE_MOD_QUIT | FAKE_MOD_QUIT_CRITICAL));
-#else /* not USE_KKCC */
-  sevt = XEVENT (event);
-  if (sevt->event_type == key_press_event
-      && (sevt->event.key.modifiers & FAKE_MOD_QUIT))
-    sevt->event.key.modifiers &=
-      ~(FAKE_MOD_QUIT | FAKE_MOD_QUIT_CRITICAL);
-#endif /* not USE_KKCC */
-
-  return event;
-}
-
-/*
- * Remove and return the first emacs event on the dispatch queue that matches
- * the supplied event.
- * Timeout event matches if interval_id is equal to that of the given event.
- * Keypress event matches if logical AND between modifiers bitmask of the
- * event in the queue and that of the given event is non-zero.
- * For all other event types, this function aborts.
- */
-
-Lisp_Object
-mswindows_cancel_dispatch_event (Lisp_Event *match)
-{
-  Lisp_Object event;
-  Lisp_Object previous_event = Qnil;
-  int user_p = mswindows_user_event_p (match);
-  Lisp_Object *head = user_p ? &mswindows_u_dispatch_event_queue :
-    &mswindows_s_dispatch_event_queue;
-  Lisp_Object *tail = user_p ? &mswindows_u_dispatch_event_queue_tail :
-    &mswindows_s_dispatch_event_queue_tail;
-
-  assert (match->event_type == timeout_event
-	  || match->event_type == key_press_event);
-
-  EVENT_CHAIN_LOOP (event, *head)
-    {
-      Lisp_Event *e = XEVENT (event);
-#ifdef USE_KKCC
-      if ((EVENT_TYPE (e) == EVENT_TYPE (match)) &&
-	  ((EVENT_TYPE (e) == timeout_event) ?
-	   (XTIMEOUT_DATA_INTERVAL_ID (EVENT_DATA (e)) ==
-	    XTIMEOUT_DATA_INTERVAL_ID (EVENT_DATA (match))) :
-	   ((XKEY_DATA_MODIFIERS (EVENT_DATA (e)) &
-	     XKEY_DATA_MODIFIERS (EVENT_DATA (match))) != 0)))
-#else /* not USE_KKCC */
-      if ((e->event_type == match->event_type) &&
-	  ((e->event_type == timeout_event) ?
-	   (e->event.timeout.interval_id == match->event.timeout.interval_id) :
-	   /* Must be key_press_event */
-	   ((e->event.key.modifiers & match->event.key.modifiers) != 0)))
-#endif /* not USE_KKCC */
-	{
-	  if (NILP (previous_event))
-	    dequeue_event (head, tail);
-	  else
-	    {
-	      XSET_EVENT_NEXT (previous_event, XEVENT_NEXT (event));
-	      if (EQ (*tail, event))
-		*tail = previous_event;
-	    }
-
-	  return event;
-	}
-      previous_event = event;
-    }
-  return Qnil;
+  assert (!NILP (dispatch_event_queue) ||
+	  !NILP (mswindows_s_dispatch_event_queue));
+
+  if (!NILP (dispatch_event_queue))
+    return dequeue_dispatch_event ();
+  else
+    return dequeue_event (&mswindows_s_dispatch_event_queue,
+			  &mswindows_s_dispatch_event_queue_tail);
 }
 
 #ifndef CYGWIN
@@ -1458,6 +1308,23 @@
     }
 }
 
+static void
+emacs_mswindows_drain_queue (void)
+{
+  mswindows_drain_windows_queue ();
+#ifdef HAVE_TTY
+  drain_tty_devices ();
+#endif
+}
+
+static int
+emacs_mswindows_quit_check_disallowed_p (void)
+{
+  /* Quit cannot happen in modal loop: all program
+     input is dedicated to Windows. */
+  return mswindows_in_modal_loop;
+}
+
 /*
  * This is a special flavor of the mswindows_need_event function,
  * used while in event pump. Actually, there is only kind of events
@@ -1481,7 +1348,7 @@
   MSG msg;
 
   /* Check if already have one */
-  if (!NILP (mswindows_u_dispatch_event_queue)
+  if (!NILP (dispatch_event_queue)
       || !NILP (mswindows_s_dispatch_event_queue))
     return;
 
@@ -1489,12 +1356,14 @@
   if (!badly_p)
     return;
 
-  /* We do not check the _u_ queue, because timers go to _s_ */
+  /* We do not check the user queue, because timers go to _s_ */
   while (NILP (mswindows_s_dispatch_event_queue))
     {
       /* We'll deadlock if go waiting */
       if (mswindows_pending_timers_count == 0)
- invalid_operation ("Deadlock due to an attempt to call next-event in a wrong context", Qunbound);
+	invalid_operation
+	  ("Deadlock due to an attempt to call next-event in a wrong context",
+	   Qunbound);
 
       /* Fetch and dispatch any pending timers */
       if (qxeGetMessage (&msg, NULL, WM_TIMER, WM_TIMER) > 0)
@@ -1512,7 +1381,7 @@
 static void
 mswindows_need_event (int badly_p)
 {
-  while (NILP (mswindows_u_dispatch_event_queue)
+  while (NILP (dispatch_event_queue)
 	 && NILP (mswindows_s_dispatch_event_queue))
     {
 #ifdef CYGWIN
@@ -1555,14 +1424,15 @@
 	    {
 #ifdef HAVE_TTY
 	      /* Look for a TTY event */
-	      for (i = 0; i < MAXDESC-1; i++)
+	      for (i = 0; i < MAXDESC; i++)
 		{
 		  /* To avoid race conditions (among other things, an infinite
 		     loop when called from Fdiscard_input()), we must return
 		     user events ahead of process events. */
 		  if (FD_ISSET (i, &temp_mask) && FD_ISSET (i, &tty_only_mask))
 		    {
-		      struct console *c = tty_find_console_from_fd (i);
+		      struct console *c =
+			find_tty_or_stream_console_from_fd (i);
 		      Lisp_Object emacs_event = Fmake_event (Qnil, Qnil);
 		      Lisp_Event *event = XEVENT (emacs_event);
 		      
@@ -1576,14 +1446,14 @@
 		}
 #endif
 	      /* Look for a process event */
-	      for (i = 0; i < MAXDESC-1; i++)
+	      for (i = 0; i < MAXDESC; i++)
 		{
 		  if (FD_ISSET (i, &temp_mask))
 		    {
 		      if (FD_ISSET (i, &process_only_mask))
 			{
 			  Lisp_Process *p =
-			    get_process_from_usid (FD_TO_USID(i));
+			    get_process_from_usid (FD_TO_USID (i));
 
 			  mswindows_enqueue_process_event (p);
 			}
@@ -1605,12 +1475,12 @@
 	  if (errno != EINTR)
 	    {
 	      /* something bad happened */
-	      assert(0);
+	      assert (0);
 	    }
 	}
       else
 	{
-	  assert(0);
+	  assert (0);
 	}
 #else /* not CYGWIN */
       /* Now try getting a message or process event */
@@ -1747,42 +1617,84 @@
       else
 	{
 	  int ix = active - WAIT_OBJECT_0;
-	  /* First, try to find which process' output has signaled */
-	  Lisp_Process *p =
-	    get_process_from_usid (HANDLE_TO_USID
-				   (mswindows_waitable_handles[ix]));
-	  if (p != NULL)
-	    {
-	      /* Found a signaled process input handle */
-	      mswindows_enqueue_process_event (p);
-	    }
-	  else
+
+	  /* look for a stream console event; see
+	     emacs_mswindows_select_console below. */
+	  LIST_LOOP_3 (porca_troia, Vconsole_list, vcontail)
 	    {
-	      /* None. This means that the process handle itself has signaled.
-		 Remove the handle from the wait vector, and make status_notify
-		 note the exited process.  First find the process object if 
-	         possible. */
-	      LIST_LOOP_3 (vaffanculo, Vprocess_list, vproctail)
-		if (get_nt_process_handle (XPROCESS (vaffanculo)) ==
-		    mswindows_waitable_handles [ix])
-		  break;
-	      mswindows_waitable_handles [ix] =
-		mswindows_waitable_handles [--mswindows_waitable_count];
-	      kick_status_notify ();
-	      /* We need to return a process event here so that
-		 (1) accept-process-output will return when called on this
-		 process, and (2) status notifications will happen in
-		 accept-process-output, sleep-for, and sit-for. */
-	      if (!NILP (vproctail))
-		  mswindows_enqueue_process_event (XPROCESS (vaffanculo));
+	      struct console *con = XCONSOLE (porca_troia);
+
+	      if (CONSOLE_STREAM_P (con))
+		{
+		  Lisp_Object instr = CONSOLE_STREAM_DATA (con)->instream;
+		  if (!NILP (instr) && !UNBOUNDP (instr) &&
+		      get_ntpipe_input_stream_waitable (XLSTREAM (instr)) ==
+		      mswindows_waitable_handles [ix])
+		    {
+		      Ichar ch = Lstream_get_ichar (XLSTREAM (instr));
+		      if (ch < 0)
+			{
+			  /* deleting the console might not be safe right now
+			     ... */
+			  enqueue_magic_eval_event (io_error_delete_console,
+						    porca_troia);
+			  /* but we definitely need to unselect it to avoid
+			     infinite loops reading EOF's */
+			  Fconsole_disable_input (porca_troia);
+			  mswindows_enqueue_magic_event (NULL, XM_BUMPQUEUE);
+			}
+		      else
+			{
+			  Lisp_Object event = Fmake_event (Qnil, Qnil);
+			  /* Here we really do want to set the
+			     use_console_meta_flag because the char is from the
+			     TTY. */
+			  character_to_event (ch, XEVENT (event), con, 1, 1);
+			  XSET_EVENT_CHANNEL (event, porca_troia);
+			  enqueue_dispatch_event (event);
+			}
+		      break;
+		    }
+		}
+	    }
+
+	  if (NILP (vcontail))
+	    { /* no stream console event, look for process event */
+	      /* First, try to find which process' output has signaled */
+	      Lisp_Process *p =
+		get_process_from_usid (HANDLE_TO_USID
+				       (mswindows_waitable_handles[ix]));
+	      if (p != NULL)
+		/* Found a signaled process input handle */
+		mswindows_enqueue_process_event (p);
 	      else
 		{
-		  /* abort (); */
-		  /* #### FUCKME!  When can this happen?  I hit this abort()
-		     when I tried enabling it. */
-		  /* Have to return something: there may be no accompanying
-		     process event */
-		  mswindows_enqueue_magic_event (NULL, XM_BUMPQUEUE);
+		  /* None. This means that the process handle itself has
+		     signaled.  Remove the handle from the wait vector, and
+		     make status_notify note the exited process.  First
+		     find the process object if possible. */
+		  LIST_LOOP_3 (vaffanculo, Vprocess_list, vproctail)
+		    if (get_nt_process_handle (XPROCESS (vaffanculo)) ==
+			mswindows_waitable_handles [ix])
+		      break;
+		  mswindows_waitable_handles [ix] =
+		    mswindows_waitable_handles [--mswindows_waitable_count];
+		  kick_status_notify ();
+		  /* We need to return a process event here so that (1)
+		     accept-process-output will return when called on this
+		     process, and (2) status notifications will happen in
+		     accept-process-output, sleep-for, and sit-for. */
+		  if (!NILP (vproctail))
+		    mswindows_enqueue_process_event (XPROCESS (vaffanculo));
+		  else
+		    {
+		      /* abort (); */
+		      /* #### FUCKME!  When can this happen?  I hit this
+			 abort() when I tried enabling it. */
+		      /* Have to return something: there may be no
+			 accompanying process event */
+		      mswindows_enqueue_magic_event (NULL, XM_BUMPQUEUE);
+		    }
 		}
 	    }
 	}
@@ -1801,26 +1713,16 @@
 mswindows_wm_timer_callback (HWND hwnd, UINT umsg, UINT id_timer, DWORD dwtime)
 {
   Lisp_Object emacs_event = Fmake_event (Qnil, Qnil);
-  Lisp_Event *event = XEVENT (emacs_event);
 
   if (KillTimer (NULL, id_timer))
     --mswindows_pending_timers_count;
 
-#ifdef USE_KKCC
   XSET_EVENT_CHANNEL (emacs_event, Qnil);
   XSET_EVENT_TIMESTAMP (emacs_event, dwtime);
   XSET_EVENT_TYPE (emacs_event, timeout_event);
-  XSET_TIMEOUT_DATA_INTERVAL_ID (XEVENT_DATA(emacs_event), id_timer);
-  XSET_TIMEOUT_DATA_FUNCTION (XEVENT_DATA(emacs_event), Qnil);
-  XSET_TIMEOUT_DATA_OBJECT (XEVENT_DATA(emacs_event), Qnil);
-#else /* not USE_KKCC */
-  event->channel = Qnil;
-  event->timestamp = dwtime;
-  event->event_type = timeout_event;
-  event->event.timeout.interval_id = id_timer;
-  event->event.timeout.function = Qnil;
-  event->event.timeout.object = Qnil;
-#endif /* not USE_KKCC */
+  XSET_EVENT_TIMEOUT_INTERVAL_ID (emacs_event, id_timer);
+  XSET_EVENT_TIMEOUT_FUNCTION (emacs_event, Qnil);
+  XSET_EVENT_TIMEOUT_OBJECT (emacs_event, Qnil);
 
   mswindows_enqueue_dispatch_event (emacs_event);
 }
@@ -1901,7 +1803,7 @@
   GCPRO1 (obj);
   obj = Feval (XCAR (obj));
 
-  RETURN_UNGCPRO(obj);
+  RETURN_UNGCPRO (obj);
 }
 
 /* Evaluate the supplied string as a sequence of Lisp forms, wrapped in
@@ -1966,7 +1868,7 @@
   Fput(token, QHSZ, make_float ((int)hsz));
   Vdde_advise_items = Fcons (token, Vdde_advise_items);
 
-  RETURN_UNGCPRO(token);
+  RETURN_UNGCPRO (token);
 }
 
 DEFUN("dde-free-advise-item", Fdde_free_advise_item, 1, 1, 0, /*
@@ -2262,29 +2164,17 @@
 	    };
 	  assert (!NILP (event->channel));
 
-#ifdef USE_KKCC
 	  SET_EVENT_TIMESTAMP (event, GetTickCount());
 	  SET_EVENT_TYPE (event, misc_user_event);
-	  XSET_MISC_USER_DATA_BUTTON (EVENT_DATA (event), 1);
-	  XSET_MISC_USER_DATA_MODIFIERS (EVENT_DATA (event), 0);
-	  XSET_MISC_USER_DATA_X (EVENT_DATA (event), -1);
-	  XSET_MISC_USER_DATA_Y (EVENT_DATA (event), -1);
-	  XSET_MISC_USER_DATA_FUNCTION (EVENT_DATA (event),
+	  SET_EVENT_MISC_USER_BUTTON (event, 1);
+	  SET_EVENT_MISC_USER_MODIFIERS (event, 0);
+	  SET_EVENT_MISC_USER_X (event, -1);
+	  SET_EVENT_MISC_USER_Y (event, -1);
+	  SET_EVENT_MISC_USER_FUNCTION (event,
 					Qdragdrop_drop_dispatch);
-	  XSET_MISC_USER_DATA_OBJECT (EVENT_DATA (event),
+	  SET_EVENT_MISC_USER_OBJECT (event,
 				      Fcons (Qdragdrop_URL,
 					    Fcons (l_dndlist, Qnil)));
-#else /* not USE_KKCC */
-	  event->timestamp = GetTickCount();
-	  event->event_type = misc_user_event;
-	  event->event.misc.button = 1;
-	  event->event.misc.modifiers = 0;
-	  event->event.misc.x = -1;
-	  event->event.misc.y = -1;
-	  event->event.misc.function = Qdragdrop_drop_dispatch;
-	  event->event.misc.object = Fcons (Qdragdrop_URL,
-					    Fcons (l_dndlist, Qnil));
-#endif /* not USE_KKCC */
 	  mswindows_enqueue_dispatch_event (emacs_event);
 	  UNGCPRO;
 	  return (HDDEDATA) DDE_FACK;
@@ -2630,8 +2520,9 @@
 LRESULT WINAPI
 mswindows_wnd_proc (HWND hwnd, UINT message_, WPARAM wParam, LPARAM lParam)
 {
-  /* Note: Remember to initialize emacs_event and event before use.
-     This code calls code that can GC. You must GCPRO before calling such code. */
+  /* Note: Remember to initialize emacs_event and event before use.  This
+     code calls code that can GC. You must GCPRO before calling such
+     code. */
   Lisp_Object emacs_event = Qnil;
   Lisp_Object fobj = Qnil;
 
@@ -2801,11 +2692,11 @@
 	  }
 	else	/* Normal keys & modifiers */
 	  {
-	    Ichar quit_ch =
-	      CONSOLE_QUIT_CHAR (XCONSOLE (mswindows_find_console (hwnd)));
 	    POINT pnt = { LOWORD (GetMessagePos()), HIWORD (GetMessagePos()) };
 	    MSG msg, tranmsg;
+#ifdef HAVE_MENUBARS
 	    int potential_accelerator = 0;
+#endif
 	    int got_accelerator = 0;
 	    /* No need to gcpro because the event is already on a
                queue when we retrieve it. */
@@ -2901,27 +2792,9 @@
 		  }
 #endif /* DEBUG_XEMACS */
 
-		/* If a quit char with no modifiers other than control and
-		   shift, then mark it with a fake modifier, which is removed
-		   upon dequeueing the event */
-		if (((quit_ch < ' ' && (mods & XEMACS_MOD_CONTROL)
-		      && DOWNCASE (0, quit_ch + 'a' - 1) ==
-		      DOWNCASE (0, ch))
-		     || (quit_ch >= ' ' && !(mods & XEMACS_MOD_CONTROL)
-			 && DOWNCASE (0, quit_ch) ==
-			 DOWNCASE (0, ch)))
-		    && ((mods_with_shift &
-			 ~(XEMACS_MOD_CONTROL | XEMACS_MOD_SHIFT))
-			== 0))
-		  {
-		    mods_with_quit |= FAKE_MOD_QUIT;
-		    if (mods_with_shift & XEMACS_MOD_SHIFT)
-		      mods_with_quit |= FAKE_MOD_QUIT_CRITICAL;
-		    mswindows_quit_chars_count++;
-		  }
 #ifdef HAVE_MENUBARS
-		else if (potential_accelerator && !got_accelerator &&
-			 mswindows_char_is_accelerator (frame, ch))
+		if (potential_accelerator && !got_accelerator &&
+		    mswindows_char_is_accelerator (frame, ch))
 		  {
 		    got_accelerator = 1;
 		    break;
@@ -3012,8 +2885,8 @@
 				   (received_keys + (tounret - 1) * 2, 2),
 				   C_STRING_ALLOCA, intchar,
 				   Qmswindows_unicode);
-				XEVENT (lastev)->event.key.alt_keychars[i] =
-				  itext_ichar (intchar);
+				XSET_EVENT_KEY_ALT_KEYCHARS
+				  (lastev, i, itext_ichar (intchar));
 			      }
 			  }
 			else
@@ -3050,8 +2923,8 @@
 				   mswindows_locale_to_code_page (lcid),
 				   NULL);
 
-				XEVENT (lastev)->event.key.alt_keychars[i] =
-				  itext_ichar (intchar);
+				XSET_EVENT_KEY_ALT_KEYCHARS
+				  (lastev, i, itext_ichar (intchar));
 				xfree (intchar);
 			      }
 			  }
@@ -3067,7 +2940,7 @@
 			else
 			  altch = 0;
 
-			XEVENT (lastev)->event.key.alt_keychars[i] = altch;
+			XSET_EVENT_KEY_ALT_KEYCHARS (lastev, i, altch);
 		      }
 		  }
 	      }
@@ -3319,23 +3192,13 @@
 	  emacs_event = Fmake_event (Qnil, Qnil);
 	  event = XEVENT(emacs_event);
 
-#ifdef USE_KKCC
 	  XSET_EVENT_CHANNEL (emacs_event,  mswindows_find_frame(hwnd));
 	  XSET_EVENT_TIMESTAMP (emacs_event, GetMessageTime());
 	  XSET_EVENT_TYPE (emacs_event, pointer_motion_event);
-	  XSET_MOTION_DATA_X (XEVENT_DATA (emacs_event),MAKEPOINTS (lParam).x);
-	  XSET_MOTION_DATA_Y (XEVENT_DATA (emacs_event),MAKEPOINTS (lParam).y);
-	  XSET_MOTION_DATA_MODIFIERS (XEVENT_DATA(emacs_event),
+	  XSET_EVENT_MOTION_X (emacs_event,MAKEPOINTS (lParam).x);
+	  XSET_EVENT_MOTION_Y (emacs_event,MAKEPOINTS (lParam).y);
+	  XSET_EVENT_MOTION_MODIFIERS (emacs_event,
                                 mswindows_modifier_state (NULL, wParam, 0));
-#else /* not USE_KKCC */	  
-	  event->channel = mswindows_find_frame (hwnd);
-	  event->timestamp = GetMessageTime ();
-	  event->event_type = pointer_motion_event;
-	  event->event.motion.x = MAKEPOINTS (lParam).x;
-	  event->event.motion.y = MAKEPOINTS (lParam).y;
-	  event->event.motion.modifiers =
-	    mswindows_modifier_state (NULL, wParam, 0);
-#endif /* not USE_KKCC */
 
 	  mswindows_enqueue_dispatch_event (emacs_event);
 	}
@@ -3856,28 +3719,16 @@
 	if (!DragQueryPoint ((HDROP) wParam, &point))
 	  point.x = point.y = -1; /* outside client area */
 
-#ifdef USE_KKCC
 	XSET_EVENT_TYPE (emacs_event, misc_user_event);
 	XSET_EVENT_CHANNEL (emacs_event,  mswindows_find_frame(hwnd));
 	XSET_EVENT_TIMESTAMP (emacs_event, GetMessageTime());
-	XSET_MISC_USER_DATA_BUTTON (XEVENT_DATA (emacs_event), 1);
-	XSET_MISC_USER_DATA_MODIFIERS (XEVENT_DATA (emacs_event),
+	XSET_EVENT_MISC_USER_BUTTON (emacs_event, 1);
+	XSET_EVENT_MISC_USER_MODIFIERS (emacs_event,
               mswindows_modifier_state (NULL, (DWORD) -1, 0));
-	XSET_MISC_USER_DATA_X (XEVENT_DATA (emacs_event), point.x);
-	XSET_MISC_USER_DATA_Y (XEVENT_DATA (emacs_event), point.y);
-	XSET_MISC_USER_DATA_FUNCTION (XEVENT_DATA (emacs_event),
+	XSET_EVENT_MISC_USER_X (emacs_event, point.x);
+	XSET_EVENT_MISC_USER_Y (emacs_event, point.y);
+	XSET_EVENT_MISC_USER_FUNCTION (emacs_event,
 					Qdragdrop_drop_dispatch);
-#else /* not USE_KKCC */
-	event->event_type = misc_user_event;
-	event->channel = mswindows_find_frame (hwnd);
-	event->timestamp = GetMessageTime();
-	event->event.misc.button = 1;		/* #### Should try harder */
-	event->event.misc.modifiers = mswindows_modifier_state (NULL,
-								(DWORD) -1, 0);
-	event->event.misc.x = point.x;
-	event->event.misc.y = point.y;
-	event->event.misc.function = Qdragdrop_drop_dispatch;
-#endif /* not USE_KKCC */
 
 	filecount = qxeDragQueryFile ((HDROP) wParam, 0xffffffff, NULL, 0);
 	for (i = 0; i < filecount; i++)
@@ -4053,12 +3904,8 @@
 
 	DragFinish ((HDROP) wParam);
 
-#ifdef USE_KKCC
-	XSET_MISC_USER_DATA_OBJECT (EVENT_DATA (event),
+	SET_EVENT_MISC_USER_OBJECT (event,
 				    Fcons (Qdragdrop_URL, l_dndlist));
-#else /* not USE_KKCC */
-	event->event.misc.object = Fcons (Qdragdrop_URL, l_dndlist);
-#endif /* not USE_KKCC */
 	mswindows_enqueue_dispatch_event (emacs_event);
 	UNGCPRO;
       }
@@ -4501,27 +4348,30 @@
 		   (TIMERPROC) mswindows_wm_timer_callback);
 }
 
+static int
+remove_timeout_mapper (Lisp_Object ev, void *data)
+{
+  if (XEVENT_TYPE (ev) == timeout_event)
+    {
+      if ((int) data == XEVENT_TIMEOUT_INTERVAL_ID (ev))
+	return 1;
+    }
+
+  return 0;
+}
+
 static void
 emacs_mswindows_remove_timeout (int id)
 {
-  Lisp_Event match_against;
-  Lisp_Object emacs_event;
-
   if (KillTimer (NULL, id))
     --mswindows_pending_timers_count;
 
   /* If there is a dispatch event generated by this
      timeout in the queue, we have to remove it too. */
-#ifdef USE_KKCC
-  SET_EVENT_TYPE(&match_against, timeout_event);
-  XSET_TIMEOUT_DATA_INTERVAL_ID (EVENT_DATA (&match_against), id);
-#else /* not USE_KKCC */
-  match_against.event_type = timeout_event;
-  match_against.event.timeout.interval_id = id;
-#endif /* not USE_KKCC */
-  emacs_event = mswindows_cancel_dispatch_event (&match_against);
-  if (!NILP (emacs_event))
-    Fdeallocate_event(emacs_event);
+  map_event_chain_remove (remove_timeout_mapper,
+			  &mswindows_s_dispatch_event_queue,
+			  &mswindows_s_dispatch_event_queue_tail,
+			  (void *) id, MECR_DEALLOCATE_EVENT);
 }
 
 /* If `user_p' is false, then return whether there are any win32, timeout,
@@ -4537,7 +4387,7 @@
 emacs_mswindows_event_pending_p (int user_p)
 {
   mswindows_need_event (0);
-  return (!NILP (mswindows_u_dispatch_event_queue)
+  return (!NILP (dispatch_event_queue)
 	  || (!user_p && !NILP (mswindows_s_dispatch_event_queue)));
 }
 
@@ -4563,11 +4413,7 @@
 {
 #define FROB(msg) case msg: write_c_string (pstream, "type=" #msg); break
 
-#ifdef USE_KKCC
-  switch (XMAGIC_DATA_MSWINDOWS_EVENT (EVENT_DATA(emacs_event)))
-#else /* not USE_KKCC */
-  switch (EVENT_MSWINDOWS_MAGIC_TYPE (emacs_event))
-#endif /* not USE_KKCC */
+  switch (EVENT_MAGIC_MSWINDOWS_EVENT (emacs_event))
     {
       FROB (XM_BUMPQUEUE);
       FROB (WM_PAINT);
@@ -4590,23 +4436,14 @@
 static int
 emacs_mswindows_compare_magic_event (Lisp_Event *e1, Lisp_Event *e2)
 {
-#ifdef USE_KKCC
-  return (XMAGIC_DATA_MSWINDOWS_EVENT (EVENT_DATA (e1)) ==
-	  XMAGIC_DATA_MSWINDOWS_EVENT (EVENT_DATA (e2)));
-#else /* not USE_KKCC */
-  return (e1->event.magic.underlying_mswindows_event ==
-	  e2->event.magic.underlying_mswindows_event);
-#endif /* not USE_KKCC */
+  return (EVENT_MAGIC_MSWINDOWS_EVENT (e1) ==
+	  EVENT_MAGIC_MSWINDOWS_EVENT (e2));
 }
 
 static Hashcode
 emacs_mswindows_hash_magic_event (Lisp_Event *e)
 {
-#ifdef USE_KKCC
-  return (XMAGIC_DATA_MSWINDOWS_EVENT (EVENT_DATA (e)));
-#else /* not USE_KKCC */
-  return e->event.magic.underlying_mswindows_event;
-#endif /* not USE_KKCC */
+  return (EVENT_MAGIC_MSWINDOWS_EVENT (e));
 }
 
 /*
@@ -4615,11 +4452,7 @@
 static void
 emacs_mswindows_handle_magic_event (Lisp_Event *emacs_event)
 {
-#ifdef USE_KKCC
-  switch (XMAGIC_DATA_MSWINDOWS_EVENT (EVENT_DATA(emacs_event)))
-#else /* not USE_KKCC */
-  switch (EVENT_MSWINDOWS_MAGIC_TYPE (emacs_event))
-#endif /* not USE_KKCC */
+  switch (EVENT_MAGIC_MSWINDOWS_EVENT (emacs_event))
     {
     case XM_BUMPQUEUE:
       break;
@@ -4637,12 +4470,8 @@
       {
 	Lisp_Object frame = EVENT_CHANNEL (emacs_event);
 	struct frame *f = XFRAME (frame);
-#ifdef USE_KKCC
-	int in_p = (XMAGIC_DATA_MSWINDOWS_EVENT (EVENT_DATA(emacs_event))
+	int in_p = (EVENT_MAGIC_MSWINDOWS_EVENT (emacs_event)
 		    == WM_SETFOCUS);
-#else /* not USE_KKCC */
-	int in_p = (EVENT_MSWINDOWS_MAGIC_TYPE (emacs_event) == WM_SETFOCUS);
-#endif /* not USE_KKCC */
 	Lisp_Object conser;
 	struct gcpro gcpro1;
 
@@ -4665,11 +4494,7 @@
     case XM_UNMAPFRAME:
       {
 	Lisp_Object frame = EVENT_CHANNEL (emacs_event);
-#ifdef USE_KKCC
-	va_run_hook_with_args (XMAGIC_DATA_MSWINDOWS_EVENT (EVENT_DATA(emacs_event))
-#else /* not USE_KKCC */
-	va_run_hook_with_args (EVENT_MSWINDOWS_MAGIC_TYPE (emacs_event)
-#endif /* not USE_KKCC */
+	va_run_hook_with_args (EVENT_MAGIC_MSWINDOWS_EVENT (emacs_event)
 			       == XM_MAPFRAME ?
 			       Qmap_frame_hook : Qunmap_frame_hook,
 			       1, frame);
@@ -4806,6 +4631,30 @@
     return; /* mswindows consoles are automatically selected */
 
   event_stream_unixoid_select_console (con);
+#else
+#if 0
+  /* This is an attempt to get `xemacs -batch -l dunnet' to work.
+     Doesn't currently work and fucks other things up. */
+  if (CONSOLE_STREAM_P (con) &&
+      !UNBOUNDP (CONSOLE_STREAM_DATA (con)->instream))
+    {
+      HANDLE h =
+	(HANDLE) _get_osfhandle (fileno (CONSOLE_STREAM_DATA (con)->in));
+      if (PeekNamedPipe (h, 0, 0, 0, 0, 0))
+	{
+	  Lisp_Object lstr = make_ntpipe_input_stream (h, 0);
+	  HANDLE hwait = get_ntpipe_input_stream_waitable (XLSTREAM (lstr));
+
+	  if (!add_waitable_handle (hwait))
+	    invalid_operation ("Too many active processes",
+			       wrap_console (con));
+	  CONSOLE_STREAM_DATA (con)->instream = lstr;
+	}
+      else
+	/* Unable to select on this stream */
+	CONSOLE_STREAM_DATA (con)->instream = Qunbound;
+    }
+#endif /* 0 */
 #endif
 }
 
@@ -4817,56 +4666,21 @@
     return; /* mswindows consoles are automatically selected */
 
   event_stream_unixoid_unselect_console (con);
-#endif
-}
-
-static void
-emacs_mswindows_quit_p (void)
-{
-  /* Quit cannot happen in modal loop: all program
-     input is dedicated to Windows. */
-  if (mswindows_in_modal_loop)
-    return;
-
-  mswindows_quit_chars_count = 0;
-  /* Drain windows queue.  This sets up number of quit characters in
-     the queue. */
-  mswindows_drain_windows_queue ();
-
-  if (mswindows_quit_chars_count > 0)
+#else
+#if 0 /* see above */
+  if (CONSOLE_STREAM_P (con) &&
+      !UNBOUNDP (CONSOLE_STREAM_DATA (con)->instream))
     {
-      /* Yes there's a hidden one... Throw it away */
-      Lisp_Event match_against;
-      Lisp_Object emacs_event;
-      int critical_p = 0;
-
-#ifdef USE_KKCC
-      SET_EVENT_TYPE (&match_against, key_press_event);
-      XSET_KEY_DATA_MODIFIERS (EVENT_DATA (&match_against), FAKE_MOD_QUIT);
-#else /* not USE_KKCC */
-      match_against.event_type = key_press_event;
-      match_against.event.key.modifiers = FAKE_MOD_QUIT;
-#endif /* not USE_KKCC */
-
-      while (mswindows_quit_chars_count > 0)
-	{
-	  emacs_event = mswindows_cancel_dispatch_event (&match_against);
-	  assert (!NILP (emacs_event));
-
-#ifdef USE_KKCC
-	  if (XKEY_DATA_MODIFIERS (XEVENT_DATA(emacs_event)) &
-#else /* not USE_KKCC */
-	  if (XEVENT (emacs_event)->event.key.modifiers &
-#endif /* not USE_KKCC */
-	      FAKE_MOD_QUIT_CRITICAL)
-	    critical_p = 1;
-
-	  Fdeallocate_event (emacs_event);
-	  mswindows_quit_chars_count--;
-	}
-
-      Vquit_flag = critical_p ? Qcritical : Qt;
+      Lisp_Object instr = CONSOLE_STREAM_DATA (con)->instream;
+      HANDLE hwait;
+
+      assert (!NILP (instr));
+      hwait = get_ntpipe_input_stream_waitable (XLSTREAM (instr));
+
+      remove_waitable_handle (hwait);
     }
+#endif /* 0 */
+#endif
 }
 
 static void
@@ -5334,10 +5148,9 @@
   mswindows_in_modal_loop = 0;
   mswindows_pending_timers_count = 0;
 
-  mswindows_event_stream = xnew (struct event_stream);
+  mswindows_event_stream = xnew_and_zero (struct event_stream);
 
   mswindows_event_stream->event_pending_p 	= emacs_mswindows_event_pending_p;
-  mswindows_event_stream->force_event_pending = 0;
   mswindows_event_stream->next_event_cb		= emacs_mswindows_next_event;
   mswindows_event_stream->handle_magic_event_cb = emacs_mswindows_handle_magic_event;
   mswindows_event_stream->format_magic_event_cb = emacs_mswindows_format_magic_event;
@@ -5345,7 +5158,8 @@
   mswindows_event_stream->hash_magic_event_cb   = emacs_mswindows_hash_magic_event;
   mswindows_event_stream->add_timeout_cb 	= emacs_mswindows_add_timeout;
   mswindows_event_stream->remove_timeout_cb 	= emacs_mswindows_remove_timeout;
-  mswindows_event_stream->quit_p_cb		= emacs_mswindows_quit_p;
+  mswindows_event_stream->drain_queue_cb	= emacs_mswindows_drain_queue;
+  mswindows_event_stream->quit_check_disallowed_p_cb = emacs_mswindows_quit_check_disallowed_p;
   mswindows_event_stream->select_console_cb 	= emacs_mswindows_select_console;
   mswindows_event_stream->unselect_console_cb	= emacs_mswindows_unselect_console;
   mswindows_event_stream->select_process_cb 	= emacs_mswindows_select_process;
@@ -5363,15 +5177,10 @@
 {
   reinit_vars_of_event_mswindows ();
 
-  mswindows_u_dispatch_event_queue = Qnil;
-  staticpro (&mswindows_u_dispatch_event_queue);
-  mswindows_u_dispatch_event_queue_tail = Qnil;
-  dump_add_root_object (&mswindows_u_dispatch_event_queue_tail);
-
   mswindows_s_dispatch_event_queue = Qnil;
   staticpro (&mswindows_s_dispatch_event_queue);
   mswindows_s_dispatch_event_queue_tail = Qnil;
-  dump_add_root_object (&mswindows_s_dispatch_event_queue_tail);
+  dump_add_root_lisp_object (&mswindows_s_dispatch_event_queue_tail);
 
   mswindows_error_caught_in_modal_loop = 0;
 
--- a/src/event-stream.c	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/event-stream.c	Sun Jan 12 11:08:22 2003 +0000
@@ -50,7 +50,12 @@
  */
 
 /* TODO:
-   This stuff is way too hard to maintain - needs rework.
+   [This stuff is way too hard to maintain - needs rework.]
+   I don't think it's that bad in the main.  I've done a fair amount of
+   cleanup work over the ages; the only stuff that's probably still somewhat
+   messy is the command-builder handling, which is that way because it's
+   trying to be "compatible" with pseudo-standards established by Emacs
+   v18.
 
    The command builder should deal only with key and button events.
    Other command events should be able to come in the MIDDLE of a key
@@ -274,16 +279,53 @@
 static void echo_key_event (struct command_builder *, Lisp_Object event);
 static void maybe_kbd_translate (Lisp_Object event);
 
-/* This structure is basically a typeahead queue: things like
-   wait-reading-process-output will delay the execution of
-   keyboard and mouse events by pushing them here.
-
-   Chained through event_next()
-   command_event_queue_tail is a pointer to the last-added element.
+/* There are two event queues here -- the command event queue (#### which
+   should be called "deferred event queue" and is in my glyph ws) and the
+   dispatch event queue (#### MS Windows actually has an extra dispatch
+   queue for non-user events and uses the generic one only for user events;
+   we should probably generalize this).
+
+   The dispatch queue (which used to occur duplicated inside of each event
+   implementation) is used for events that have been read from the
+   window-system event queue(s) and not yet process by
+   next_event_internal().  It exists for two reasons: (1) because in many
+   implementations, events often come from the window system by way of
+   callbacks, and need to push the event to be returned onto a queue; (2)
+   in order to handle QUIT in a guaranteed correct fashion without
+   resorting to weird implementation-specific hacks that may or may not
+   work well, we need to drain the window-system event queues and then look
+   through to see if there's an event matching quit-char (usually ^G).  the
+   drained events need to go onto a queue. (There are other, similar cases
+   where we need to drain the pending events so we can look ahead -- for
+   example, checking for pending expose events under X to avoid excessive
+   server activity.)
+
+   The command event queue is used *AFTER* an event has been read from
+   next_event_internal(), when it needs to be pushed back.  This includes,
+   for example, accept-process-output, sleep-for and
+   wait_delaying_user_input().  Eval events and the like, generated by
+   enqueue-eval-event, enqueue_magic_eval_event(), etc. are also pushed
+   onto this queue.  Some events generated by callbacks are also pushed
+   onto this queue, #### although maybe shouldn't be.
+
+   The command queue takes precedence over the dispatch queue.
+
+   #### It is worth investigating to see whether both queues are really
+   needed, and how exactly they should be used.  enqueue-eval-event, for
+   example, could certainly push onto the dispatch queue, and all callbacks
+   maybe should.  wait_delaying_user_input() seems to need both queues,
+   since it can take events from the dispatch queue and push them onto the
+   command queue; but it perhaps could be rewritten to avoid this.  ####
+   In general we need to review the handling of these two queues, figure
+   out exactly what ought to be happening, and document it.
  */
+
 static Lisp_Object command_event_queue;
 static Lisp_Object command_event_queue_tail;
 
+Lisp_Object dispatch_event_queue;
+static Lisp_Object dispatch_event_queue_tail;
+
 /* Nonzero means echo unfinished commands after this many seconds of pause. */
 static Lisp_Object Vecho_keystrokes;
 
@@ -318,26 +360,14 @@
 
 static Lisp_Object Vcommand_builder_free_list;
 
-#ifdef USE_KKCC
-static const struct lrecord_description munging_key_translation_description_1 [] = {
-  { XD_LISP_OBJECT, offsetof (struct munging_key_translation, first_mungeable_event) },
-  { XD_END }
-};
-
-static const struct struct_description munging_key_translation_description = {
-  sizeof (Lisp_Object),
-  munging_key_translation_description_1
-};
-
-static const struct lrecord_description command_builder_description [] = {
+static const struct memory_description command_builder_description [] = {
   { XD_LISP_OBJECT, offsetof (struct command_builder, current_events) },
   { XD_LISP_OBJECT, offsetof (struct command_builder, most_current_event) },
   { XD_LISP_OBJECT, offsetof (struct command_builder, last_non_munged_event) },
   { XD_LISP_OBJECT, offsetof (struct command_builder, console) },
-  { XD_STRUCT_ARRAY, offsetof (struct command_builder, munge_me), 2, &munging_key_translation_description },
+  { XD_LISP_OBJECT_ARRAY, offsetof (struct command_builder, first_mungeable_event), 2 },
   { XD_END }
 };
-#endif /* USE_KKCC */
 
 static Lisp_Object
 mark_command_builder (Lisp_Object obj)
@@ -346,8 +376,8 @@
   mark_object (builder->current_events);
   mark_object (builder->most_current_event);
   mark_object (builder->last_non_munged_event);
-  mark_object (builder->munge_me[0].first_mungeable_event);
-  mark_object (builder->munge_me[1].first_mungeable_event);
+  mark_object (builder->first_mungeable_event[0]);
+  mark_object (builder->first_mungeable_event[1]);
   return builder->console;
 }
 
@@ -365,19 +395,12 @@
     }
 }
 
-#ifdef USE_KKCC
 DEFINE_LRECORD_IMPLEMENTATION ("command-builder", command_builder,
 			       0, /*dumpable-flag*/
                                mark_command_builder, internal_object_printer,
 			       finalize_command_builder, 0, 0, 
 			       command_builder_description,
 			       struct command_builder);
-#else /* not USE_KKCC */
-DEFINE_LRECORD_IMPLEMENTATION ("command-builder", command_builder,
-                               mark_command_builder, internal_object_printer,
-			       finalize_command_builder, 0, 0, 0,
-			       struct command_builder);
-#endif /* not USE_KKCC */
 
 static void
 reset_command_builder_event_chain (struct command_builder *builder)
@@ -385,15 +408,15 @@
   builder->current_events = Qnil;
   builder->most_current_event = Qnil;
   builder->last_non_munged_event = Qnil;
-  builder->munge_me[0].first_mungeable_event = Qnil;
-  builder->munge_me[1].first_mungeable_event = Qnil;
+  builder->first_mungeable_event[0] = Qnil;
+  builder->first_mungeable_event[1] = Qnil;
 }
 
 Lisp_Object
 allocate_command_builder (Lisp_Object console, int with_echo_buf)
 {
   Lisp_Object builder_obj =
-    allocate_managed_lcrecord (Vcommand_builder_free_list);
+    alloc_managed_lcrecord (Vcommand_builder_free_list);
   struct command_builder *builder = XCOMMAND_BUILDER (builder_obj);
 
   builder->console = console;
@@ -442,14 +465,12 @@
     transfer_event_chain_pointer (collapsing->last_non_munged_event,
 				  collapsing->current_events,
 				  new_buildings->current_events);
-  new_buildings->munge_me[0].first_mungeable_event =
-    transfer_event_chain_pointer (collapsing->munge_me[0].
-				  first_mungeable_event,
+  new_buildings->first_mungeable_event[0] =
+    transfer_event_chain_pointer (collapsing->first_mungeable_event[0],
 				  collapsing->current_events,
 				  new_buildings->current_events);
-  new_buildings->munge_me[1].first_mungeable_event =
-    transfer_event_chain_pointer (collapsing->munge_me[1].
-				  first_mungeable_event,
+  new_buildings->first_mungeable_event[1] =
+    transfer_event_chain_pointer (collapsing->first_mungeable_event[1],
 				  collapsing->current_events,
 				  new_buildings->current_events);
 
@@ -481,10 +502,10 @@
     builder->current_events = event;
 
   builder->most_current_event = event;
-  if (NILP (builder->munge_me[0].first_mungeable_event))
-    builder->munge_me[0].first_mungeable_event = event;
-  if (NILP (builder->munge_me[1].first_mungeable_event))
-    builder->munge_me[1].first_mungeable_event = event;
+  if (NILP (builder->first_mungeable_event[0]))
+    builder->first_mungeable_event[0] = event;
+  if (NILP (builder->first_mungeable_event[1]))
+    builder->first_mungeable_event[1] = event;
 }
 
 
@@ -513,14 +534,18 @@
 static int
 event_stream_event_pending_p (int user)
 {
+  /* #### Hmmm ...  There may be some duplication in "drain queue" and
+     "event pending".  Couldn't we just drain the queue and see what's in
+     it, and not maybe need a separate event method for this?  Would this
+     work when USER is 0?  Maybe this would be slow? */
   return event_stream && event_stream->event_pending_p (user);
 }
 
 static void
 event_stream_force_event_pending (struct frame *f)
 {
-  if (event_stream->force_event_pending)
-    event_stream->force_event_pending (f);
+  if (event_stream->force_event_pending_cb)
+    event_stream->force_event_pending_cb (f);
 }
 
 static int
@@ -540,11 +565,9 @@
 
   if (sigint_happened)
     {
-      int ch = CONSOLE_QUIT_CHAR (con);
       sigint_happened = 0;
       Vquit_flag = Qnil;
-      character_to_event (ch, event, con, 1, 1);
-      event->channel = wrap_console (con);
+      Fcopy_event (CONSOLE_QUIT_EVENT (con), wrap_event (event));
       return 1;
     }
   return 0;
@@ -754,11 +777,65 @@
 				      in_usid, err_usid);
 }
 
+static void
+event_stream_drain_queue (void)
+{
+  if (event_stream && event_stream->drain_queue_cb)
+    event_stream->drain_queue_cb ();
+}
+
+struct remove_quit_p_data
+{
+  int critical;
+};
+
+static int
+remove_quit_p_event (Lisp_Object ev, void *the_data)
+{
+  struct remove_quit_p_data *data = (struct remove_quit_p_data *) the_data;
+  struct console *con = event_console_or_selected (ev);
+
+  if (XEVENT_TYPE (ev) == key_press_event)
+    {
+      if (event_matches_key_specifier_p (ev, CONSOLE_QUIT_EVENT (con)))
+	return 1;
+      if (event_matches_key_specifier_p (ev,
+					 CONSOLE_CRITICAL_QUIT_EVENT (con)))
+	{
+	  data->critical = 1;
+	  return 1;
+	}
+    }
+
+  return 0;
+}
+
+static int
+event_stream_quit_check_disallowed_p (void)
+{
+  if (event_stream && event_stream->quit_check_disallowed_p_cb)
+    return event_stream->quit_check_disallowed_p_cb ();
+  else
+    return 0;
+}
+
 void
 event_stream_quit_p (void)
 {
-  if (event_stream)
-    event_stream->quit_p_cb ();
+  struct remove_quit_p_data data;
+
+  if (event_stream_quit_check_disallowed_p ())
+    return;
+
+  /* Drain queue so we can check for pending C-g events. */
+  event_stream_drain_queue ();
+  data.critical = 0;
+
+  if (map_event_chain_remove (remove_quit_p_event,
+			      &dispatch_event_queue,
+			      &dispatch_event_queue_tail,
+			      &data, MECR_DEALLOCATE_EVENT))
+    Vquit_flag = data.critical ? Qcritical : Qt;
 }
 
 static int
@@ -791,11 +868,7 @@
       clear_echo_area (selected_frame (), Qnil, 0);
     }
 
-#ifdef USE_KKCC
   format_event_object (buf, event, 1);
-#else /* not USE_KKCC */
-  format_event_object (buf, XEVENT(event), 1);
-#endif /* not USE_KKCC */
   len = eilen (buf);
 
   if (len + buf_index + 4 > command_builder->echo_buf_length)
@@ -903,96 +976,57 @@
   if (EQ (Fhash_table_count (Vkeyboard_translate_table), Qzero))
     return;
 
-  c = event_to_character (XEVENT (event), 0, 0, 0);
+  c = event_to_character (event, 0, 0, 0);
   if (c != -1)
     {
       Lisp_Object traduit = Fgethash (make_char (c), Vkeyboard_translate_table,
 				      Qnil);
       if (!NILP (traduit) && SYMBOLP (traduit))
 	{
-#ifdef USE_KKCC
-	  XSET_KEY_DATA_KEYSYM (XEVENT_DATA (event), traduit);
-	  XSET_KEY_DATA_MODIFIERS (XEVENT_DATA (event), 0);
-#else /* not USE_KKCC */
-	  XEVENT (event)->event.key.keysym = traduit;
-	  XEVENT (event)->event.key.modifiers = 0;
-#endif /* not USE_KKCC */
+	  XSET_EVENT_KEY_KEYSYM (event, traduit);
+	  XSET_EVENT_KEY_MODIFIERS (event, 0);
 	  did_translate = 1;
 	}
       else if (CHARP (traduit))
 	{
-#ifdef USE_KKCC
-	  Lisp_Object ev2 = Fmake_event(Qnil, Qnil);
-#else /* not USE_KKCC */
-	  Lisp_Event ev2;
-
 	  /* This used to call Fcharacter_to_event() directly into EVENT,
 	     but that can eradicate timestamps and other such stuff.
 	     This way is safer. */
-	  zero_event (&ev2);
-#endif /* not USE_KKCC */
-
-#ifdef USE_KKCC
+	  Lisp_Object ev2 = Fmake_event (Qnil, Qnil);
+
 	  character_to_event (XCHAR (traduit), XEVENT (ev2),
-			      XCONSOLE (XEVENT_CHANNEL (event)), 1, 1);
-	  XSET_KEY_DATA_KEYSYM (XEVENT_DATA (event), XKEY_DATA_KEYSYM (XEVENT_DATA (ev2)));
-	  XSET_KEY_DATA_MODIFIERS (XEVENT_DATA (event), 
-                               XKEY_DATA_MODIFIERS (XEVENT_DATA (ev2)));
-#else /* not USE_KKCC */
-	  character_to_event (XCHAR (traduit), &ev2,
-			      XCONSOLE (EVENT_CHANNEL (XEVENT (event))), 1, 1);
-	  XEVENT (event)->event.key.keysym = ev2.event.key.keysym;
-	  XEVENT (event)->event.key.modifiers = ev2.event.key.modifiers;
-#endif /* not USE_KKCC */
+			      XCONSOLE (XEVENT_CHANNEL (event)), 0, 1);
+	  XSET_EVENT_KEY_KEYSYM (event, XEVENT_KEY_KEYSYM (ev2));
+	  XSET_EVENT_KEY_MODIFIERS (event, XEVENT_KEY_MODIFIERS (ev2));
+	  Fdeallocate_event (ev2);
 	  did_translate = 1;
 	}
     }
 
   if (!did_translate)
     {
-#ifdef USE_KKCC
-      Lisp_Object traduit = Fgethash (XKEY_DATA_KEYSYM (XEVENT_DATA (event)),
+      Lisp_Object traduit = Fgethash (XEVENT_KEY_KEYSYM (event),
 				      Vkeyboard_translate_table, Qnil);
-#else /* not USE_KKCC */
-      Lisp_Object traduit = Fgethash (XEVENT (event)->event.key.keysym,
-				      Vkeyboard_translate_table, Qnil);
-#endif /* not USE_KKCC */
       if (!NILP (traduit) && SYMBOLP (traduit))
 	{
-#ifdef USE_KKCC
-	  XSET_KEY_DATA_KEYSYM (XEVENT_DATA (event), traduit);
-#else /* not USE_KKCC */
-	  XEVENT (event)->event.key.keysym = traduit;
-#endif /* not USE_KKCC */
+	  XSET_EVENT_KEY_KEYSYM (event, traduit);
 	  did_translate = 1;
 	}
       else if (CHARP (traduit))
 	{
-#ifdef USE_KKCC
-	  Lisp_Object ev2 = Fmake_event(Qnil, Qnil);
-#else /* not USE_KKCC */
-	  Lisp_Event ev2;
-
 	  /* This used to call Fcharacter_to_event() directly into EVENT,
 	     but that can eradicate timestamps and other such stuff.
 	     This way is safer. */
-	  zero_event (&ev2);
-#endif /* not USE_KKCC */
-
-#ifdef USE_KKCC
+	  Lisp_Object ev2 = Fmake_event (Qnil, Qnil);
+
 	  character_to_event (XCHAR (traduit), XEVENT (ev2),
-			      XCONSOLE (XEVENT_CHANNEL (event)), 1, 1);
-	  XSET_KEY_DATA_KEYSYM (XEVENT_DATA (event), XKEY_DATA_KEYSYM (XEVENT_DATA (ev2)));
-	  XSET_KEY_DATA_MODIFIERS (XEVENT_DATA (event),
-                               XKEY_DATA_MODIFIERS (XEVENT_DATA (event)) |
-                               XKEY_DATA_MODIFIERS (XEVENT_DATA (ev2)));
-#else /* not USE_KKCC */
-	  character_to_event (XCHAR (traduit), &ev2,
-			      XCONSOLE (EVENT_CHANNEL (XEVENT (event))), 1, 1);
-	  XEVENT (event)->event.key.keysym = ev2.event.key.keysym;
-	  XEVENT (event)->event.key.modifiers |= ev2.event.key.modifiers;
-#endif /* not USE_KKCC */
-
+			      XCONSOLE (XEVENT_CHANNEL (event)), 0, 1);
+	  XSET_EVENT_KEY_KEYSYM (event, XEVENT_KEY_KEYSYM (ev2));
+	  XSET_EVENT_KEY_MODIFIERS (event,
+                               XEVENT_KEY_MODIFIERS (event) |
+                               XEVENT_KEY_MODIFIERS (ev2));
+
+	  Fdeallocate_event (ev2);
 	  did_translate = 1;
 	}
     }
@@ -1086,7 +1120,7 @@
   }
 
   redisplay ();
-  if (event_matches_key_specifier_p (XEVENT (event), make_char (' ')))
+  if (event_matches_key_specifier_p (event, make_char (' ')))
     {
       /* Discard next key if it is a space */
       reset_key_echo (command_builder, 1);
@@ -1303,22 +1337,16 @@
   return tm->object;
 }
 
-static const struct lrecord_description timeout_description[] = {
+static const struct memory_description timeout_description[] = {
   { XD_LISP_OBJECT, offsetof (Lisp_Timeout, function) },
   { XD_LISP_OBJECT, offsetof (Lisp_Timeout, object) },
   { XD_END }
 };
 
-#ifdef USE_KKCC
 DEFINE_LRECORD_IMPLEMENTATION ("timeout", timeout,
 			       1, /*dumpable-flag*/
 			       mark_timeout, internal_object_printer,
 			       0, 0, 0, timeout_description, Lisp_Timeout);
-#else /* not USE_KKCC */
-DEFINE_LRECORD_IMPLEMENTATION ("timeout", timeout,
-			       mark_timeout, internal_object_printer,
-			       0, 0, 0, timeout_description, Lisp_Timeout);
-#endif /* not USE_KKCC */
 
 /* Generate a timeout and return its ID. */
 
@@ -1328,7 +1356,7 @@
 			      Lisp_Object function, Lisp_Object object,
 			      int async_p)
 {
-  Lisp_Object op = allocate_managed_lcrecord (Vtimeout_free_list);
+  Lisp_Object op = alloc_managed_lcrecord (Vtimeout_free_list);
   Lisp_Timeout *timeout = XTIMEOUT (op);
   EMACS_TIME current_time;
   EMACS_TIME interval;
@@ -1698,6 +1726,18 @@
   return dequeue_event (&command_event_queue, &command_event_queue_tail);
 }
 
+void
+enqueue_dispatch_event (Lisp_Object event)
+{
+  enqueue_event (event, &dispatch_event_queue, &dispatch_event_queue_tail);
+}
+
+Lisp_Object
+dequeue_dispatch_event (void)
+{
+  return dequeue_event (&dispatch_event_queue, &dispatch_event_queue_tail);
+}
+
 static void
 enqueue_command_event_1 (Lisp_Object event_to_copy)
 {
@@ -1708,17 +1748,10 @@
 enqueue_magic_eval_event (void (*fun) (Lisp_Object), Lisp_Object object)
 {
   Lisp_Object event = Fmake_event (Qnil, Qnil);
-#ifdef USE_KKCC
   XSET_EVENT_TYPE (event, magic_eval_event);
   /* channel for magic_eval events is nil */
-  XSET_MAGIC_EVAL_DATA_INTERNAL_FUNCTION (XEVENT_DATA (event), fun);
-  XSET_MAGIC_EVAL_DATA_OBJECT (XEVENT_DATA (event), object);
-#else /* not USE_KKCC */
-  XEVENT (event)->event_type = magic_eval_event;
-  /* channel for magic_eval events is nil */
-  XEVENT (event)->event.magic_eval.internal_function = fun;
-  XEVENT (event)->event.magic_eval.object = object;
-#endif /* not USE_KKCC */
+  XSET_EVENT_MAGIC_EVAL_INTERNAL_FUNCTION (event, fun);
+  XSET_EVENT_MAGIC_EVAL_OBJECT (event, object);
   enqueue_command_event (event);
 }
 
@@ -1733,17 +1766,10 @@
 {
   Lisp_Object event = Fmake_event (Qnil, Qnil);
 
-#ifdef USE_KKCC
   XSET_EVENT_TYPE (event, eval_event);
   /* channel for eval events is nil */
-  XSET_EVAL_DATA_FUNCTION (XEVENT_DATA (event), function);
-  XSET_EVAL_DATA_OBJECT (XEVENT_DATA (event), object);
-#else /* not USE_KKCC */
-  XEVENT (event)->event_type = eval_event;
-  /* channel for eval events is nil */
-  XEVENT (event)->event.eval.function = function;
-  XEVENT (event)->event.eval.object = object;
-#endif /* not USE_KKCC */
+  XSET_EVENT_EVAL_FUNCTION (event, function);
+  XSET_EVENT_EVAL_OBJECT (event, object);
   enqueue_command_event (event);
 
   return event;
@@ -1754,25 +1780,14 @@
 			 Lisp_Object object)
 {
   Lisp_Object event = Fmake_event (Qnil, Qnil);
-#ifdef USE_KKCC
   XSET_EVENT_TYPE (event, misc_user_event);
   XSET_EVENT_CHANNEL (event, channel);
-  XSET_MISC_USER_DATA_FUNCTION (XEVENT_DATA (event), function);
-  XSET_MISC_USER_DATA_OBJECT (XEVENT_DATA (event), object);
-  XSET_MISC_USER_DATA_BUTTON (XEVENT_DATA (event), 0);
-  XSET_MISC_USER_DATA_MODIFIERS (XEVENT_DATA (event), 0);
-  XSET_MISC_USER_DATA_X (XEVENT_DATA (event), -1);
-  XSET_MISC_USER_DATA_Y (XEVENT_DATA (event), -1);
-#else /* not USE_KKCC */
-  XEVENT (event)->event_type = misc_user_event;
-  XEVENT (event)->channel = channel;
-  XEVENT (event)->event.misc.function  = function;
-  XEVENT (event)->event.misc.object    = object;
-  XEVENT (event)->event.misc.button    = 0;
-  XEVENT (event)->event.misc.modifiers = 0;
-  XEVENT (event)->event.misc.x         = -1;
-  XEVENT (event)->event.misc.y         = -1;
-#endif /* not USE_KKCC */
+  XSET_EVENT_MISC_USER_FUNCTION (event, function);
+  XSET_EVENT_MISC_USER_OBJECT (event, object);
+  XSET_EVENT_MISC_USER_BUTTON (event, 0);
+  XSET_EVENT_MISC_USER_MODIFIERS (event, 0);
+  XSET_EVENT_MISC_USER_X (event, -1);
+  XSET_EVENT_MISC_USER_Y (event, -1);
   enqueue_command_event (event);
 
   return event;
@@ -1785,25 +1800,14 @@
 {
   Lisp_Object event = Fmake_event (Qnil, Qnil);
 
-#ifdef USE_KKCC
   XSET_EVENT_TYPE (event, misc_user_event);
   XSET_EVENT_CHANNEL (event, channel);
-  XSET_MISC_USER_DATA_FUNCTION (XEVENT_DATA (event), function);
-  XSET_MISC_USER_DATA_OBJECT (XEVENT_DATA (event), object);
-  XSET_MISC_USER_DATA_BUTTON (XEVENT_DATA (event), button);
-  XSET_MISC_USER_DATA_MODIFIERS (XEVENT_DATA (event), modifiers);
-  XSET_MISC_USER_DATA_X (XEVENT_DATA (event), x);
-  XSET_MISC_USER_DATA_Y (XEVENT_DATA (event), y);
-#else /* not USE_KKCC */
-  XEVENT (event)->event_type = misc_user_event;
-  XEVENT (event)->channel = channel;
-  XEVENT (event)->event.misc.function  = function;
-  XEVENT (event)->event.misc.object    = object;
-  XEVENT (event)->event.misc.button    = button;
-  XEVENT (event)->event.misc.modifiers = modifiers;
-  XEVENT (event)->event.misc.x         = x;
-  XEVENT (event)->event.misc.y         = y;
-#endif /* not USE_KKCC */
+  XSET_EVENT_MISC_USER_FUNCTION (event, function);
+  XSET_EVENT_MISC_USER_OBJECT (event, object);
+  XSET_EVENT_MISC_USER_BUTTON (event, button);
+  XSET_EVENT_MISC_USER_MODIFIERS (event, modifiers);
+  XSET_EVENT_MISC_USER_X (event, x);
+  XSET_EVENT_MISC_USER_Y (event, y);
   enqueue_command_event (event);
 
   return event;
@@ -2188,45 +2192,26 @@
       /* If this was a timeout, then we need to extract some data
 	 out of the returned closure and might need to resignal
 	 it. */
-#ifdef USE_KKCC
       if (EVENT_TYPE (e) == timeout_event)
-#else /* not USE_KKCC */
-      if (e->event_type == timeout_event)
-#endif /* not USE_KKCC */
 	{
 	  Lisp_Object tristan, isolde;
 
-#ifdef USE_KKCC
-	  XSET_TIMEOUT_DATA_ID_NUMBER (EVENT_DATA (e), 
-                          event_stream_resignal_wakeup (XTIMEOUT_DATA_INTERVAL_ID (EVENT_DATA (e)), 0, &tristan, &isolde));
-
-          XSET_TIMEOUT_DATA_FUNCTION (EVENT_DATA (e), tristan);
-          XSET_TIMEOUT_DATA_OBJECT (EVENT_DATA (e), isolde);
+	  SET_EVENT_TIMEOUT_ID_NUMBER (e, 
+                          event_stream_resignal_wakeup (EVENT_TIMEOUT_INTERVAL_ID (e), 0, &tristan, &isolde));
+
+          SET_EVENT_TIMEOUT_FUNCTION (e, tristan);
+          SET_EVENT_TIMEOUT_OBJECT (e, isolde);
 	  /* next_event_internal() doesn't print out timeout events
 	     because of the extra info we just set. */
-#else /* not USE_KKCC */
-	  e->event.timeout.id_number =
-	    event_stream_resignal_wakeup (e->event.timeout.interval_id, 0,
-					  &tristan, &isolde);
-
-	  e->event.timeout.function = tristan;
-	  e->event.timeout.object = isolde;
-	  /* next_event_internal() doesn't print out timeout events
-	     because of the extra info we just set. */
-#endif /* not USE_KKCC */
 	  DEBUG_PRINT_EMACS_EVENT ("real, timeout", target_event);
 	}
 
       /* If we read a ^G, then set quit-flag and try to QUIT.
 	 This may be blocked (see above).
        */
-#ifdef USE_KKCC
       if (EVENT_TYPE (e) == key_press_event &&
-#else /* not USE_KKCC */
-      if (e->event_type == key_press_event &&
-#endif /* not USE_KKCC */
 	  event_matches_key_specifier_p
-	  (e, make_char (CONSOLE_QUIT_CHAR (XCONSOLE (EVENT_CHANNEL (e))))))
+	  (target_event, CONSOLE_QUIT_EVENT (XCONSOLE (EVENT_CHANNEL (e)))))
 	{
 	  Vquit_flag = Qt;
 	  QUIT;
@@ -2495,8 +2480,7 @@
      Note that last-input-char will never have its high-bit set, in an
      effort to sidestep the ambiguity between M-x and oslash.
      */
-  Vlast_input_char = Fevent_to_character (Vlast_input_event,
-                                          Qnil, Qnil, Qnil);
+  Vlast_input_char = Fevent_to_character (Vlast_input_event, Qnil, Qnil, Qnil);
   {
     EMACS_TIME t;
     EMACS_GET_TIME (t);
@@ -2538,7 +2522,7 @@
      execute_help_form() calls Fnext_command_event(), which calls this
      function, as well as Fdispatch_event.  */
   if (!NILP (Vhelp_form) &&
-      event_matches_key_specifier_p (XEVENT (event), Vhelp_char))
+      event_matches_key_specifier_p (event, Vhelp_char))
     {
       /* temporarily reenable quit checking here, because we could get stuck */
       Vquit_flag = Qnil; /* see begin_dont_check_for_quit() */
@@ -2873,11 +2857,7 @@
 	case process_event:
 	  {
 	    if (NILP (process) ||
-#ifdef USE_KKCC
-                EQ (XPROCESS_DATA_PROCESS (XEVENT_DATA (event)), process))
-#else /* not USE_KKCC */
-                EQ (XEVENT (event)->event.process.process, process))
-#endif /* not USE_KKCC */
+                EQ (XEVENT_PROCESS_PROCESS (event), process))
 	      {
                 done = 1;
 		/* RMS's version always returns nil when proc is nil,
@@ -3159,25 +3139,15 @@
 
     case eval_event:
       {
-#ifdef USE_KKCC
-	call1 (XEVAL_DATA_FUNCTION (XEVENT_DATA (event)),
-	       XEVAL_DATA_OBJECT (XEVENT_DATA (event)));
-#else /* not USE_KKCC */
-	call1 (XEVENT (event)->event.eval.function,
-	       XEVENT (event)->event.eval.object);
-#endif /* not USE_KKCC */
+	call1 (XEVENT_EVAL_FUNCTION (event),
+	       XEVENT_EVAL_OBJECT (event));
 	return;
       }
 
     case magic_eval_event:
       {
-#ifdef USE_KKCC
-	XMAGIC_EVAL_DATA_INTERNAL_FUNCTION (XEVENT_DATA (event))
-	  XMAGIC_EVAL_DATA_OBJECT (XEVENT_DATA (event));
-#else /* not USE_KKCC */
-	(XEVENT (event)->event.magic_eval.internal_function)
-	  (XEVENT (event)->event.magic_eval.object);
-#endif /* not USE_KKCC */
+	XEVENT_MAGIC_EVAL_INTERNAL_FUNCTION (event)
+	  XEVENT_MAGIC_EVAL_OBJECT (event);
 	return;
       }
 
@@ -3190,11 +3160,7 @@
 
     case process_event:
       {
-#ifdef USE_KKCC
-	Lisp_Object p = XPROCESS_DATA_PROCESS (XEVENT_DATA (event));
-#else /* not USE_KKCC */
-	Lisp_Object p = XEVENT (event)->event.process.process;
-#endif /* not USE_KKCC */
+	Lisp_Object p = XEVENT_PROCESS_PROCESS (event);
 	Charcount readstatus;
 	int iter;
 
@@ -3288,15 +3254,9 @@
       {
 	Lisp_Event *e = XEVENT (event);
 
-#ifdef USE_KKCC
-	if (!NILP (XTIMEOUT_DATA_FUNCTION (EVENT_DATA (e))))
-	  call1 (XTIMEOUT_DATA_FUNCTION (EVENT_DATA (e)),
-                 XTIMEOUT_DATA_OBJECT (EVENT_DATA (e)));
-#else /* not USE_KKCC */
-	if (!NILP (e->event.timeout.function))
-	  call1 (e->event.timeout.function,
-		 e->event.timeout.object);
-#endif /* not USE_KKCC */
+	if (!NILP (EVENT_TIMEOUT_FUNCTION (e)))
+	  call1 (EVENT_TIMEOUT_FUNCTION (e),
+                 EVENT_TIMEOUT_OBJECT (e));
 	return;
       }
     case magic_event:
@@ -3364,7 +3324,7 @@
 {
   Lisp_Object suffix;
 
-  EVENT_CHAIN_LOOP (suffix, builder->munge_me[munge].first_mungeable_event)
+  EVENT_CHAIN_LOOP (suffix, builder->first_mungeable_event[munge])
     {
       Lisp_Object result = munging_key_map_event_binding (suffix, munge);
 
@@ -3408,7 +3368,7 @@
 	  EVENT_CHAIN_LOOP (tempev, suffix)
 	    {
 	      Lisp_Object *mungeable_event =
-		&builder->munge_me[1 - munge].first_mungeable_event;
+		&builder->first_mungeable_event[1 - munge];
 	      if (EQ (tempev, *mungeable_event))
 		{
 		  *mungeable_event = new_chain;
@@ -3419,7 +3379,7 @@
 	  /* Now munge the current event chain in the command builder. */
 	  n = event_chain_count (suffix);
 	  command_builder_replace_suffix (builder, suffix, new_chain);
-	  builder->munge_me[munge].first_mungeable_event = Qnil;
+	  builder->first_mungeable_event[munge] = Qnil;
 
 	  *did_munge = 1;
 
@@ -3452,13 +3412,8 @@
   if (XEVENT_TYPE (evee) == misc_user_event)
     {
       if (allow_misc_user_events_p && (NILP (XEVENT_NEXT (evee))))
-#ifdef USE_KKCC
-	return list2  (XEVAL_DATA_FUNCTION (XEVENT_DATA (evee)),
-		       XEVAL_DATA_OBJECT (XEVENT_DATA (evee)));
-#else /* not USE_KKCC */
-	return list2 (XEVENT (evee)->event.eval.function,
-		      XEVENT (evee)->event.eval.object);
-#endif /* not USE_KKCC */
+	return list2  (XEVENT_EVAL_FUNCTION (evee),
+		       XEVENT_EVAL_OBJECT (evee));
       else
 	return Qnil;
     }
@@ -3514,47 +3469,14 @@
   if (XEVENT_TYPE (builder->most_current_event) == key_press_event
       && !NILP (Vretry_undefined_key_binding_unshifted))
     {
-      Lisp_Object terminal = builder->most_current_event;
-#ifdef USE_KKCC
-      Lisp_Key_Data* key = XKEY_DATA (XEVENT_DATA (terminal));
-#else /* not USE_KKCC */
-      struct key_data *key = &XEVENT (terminal)->event.key;
-#endif /* not USE_KKCC */
-      Ichar c = 0;
-#ifdef USE_KKCC
-      if ((KEY_DATA_MODIFIERS (key) & XEMACS_MOD_SHIFT)
-          || (CHAR_OR_CHAR_INTP (KEY_DATA_KEYSYM(key))
-              && ((c = XCHAR_OR_CHAR_INT (KEY_DATA_KEYSYM(key))),
-		  c >= 'A' && c <= 'Z')))
-#else /* not USE_KKCC */
-      if ((key->modifiers & XEMACS_MOD_SHIFT)
-          || (CHAR_OR_CHAR_INTP (key->keysym)
-              && ((c = XCHAR_OR_CHAR_INT (key->keysym)),
-		  c >= 'A' && c <= 'Z')))
-#endif /* not USE_KKCC */
+      if (event_upshifted_p (builder->most_current_event))
         {
 	  Lisp_Object neubauten = copy_command_builder (builder, 0);
 	  struct command_builder *neub = XCOMMAND_BUILDER (neubauten);
 	  struct gcpro gcpro1;
 
 	  GCPRO1 (neubauten);
-          terminal = event_chain_tail (neub->current_events);
-#ifdef USE_KKCC
-	  key = XKEY_DATA (XEVENT_DATA (terminal));
-
-          if (KEY_DATA_MODIFIERS (key) & XEMACS_MOD_SHIFT)
-            SET_KEY_DATA_MODIFIERS (key, (KEY_DATA_MODIFIERS (key) & (~ XEMACS_MOD_SHIFT)));
-          else
-            SET_KEY_DATA_KEYSYM (key, make_char (c + 'a' - 'A'));
-#else /* not USE_KKCC */
-	  key = &XEVENT (terminal)->event.key;
-
-          if (key->modifiers & XEMACS_MOD_SHIFT)
-            key->modifiers &= (~ XEMACS_MOD_SHIFT);
-          else
-            key->keysym = make_char (c + 'a' - 'A');
-#endif /* not USE_KKCC */
-
+	  downshift_event (event_chain_tail (neub->current_events));
           result =
 	    command_builder_find_leaf_no_mule_processing
 	      (neub, allow_misc_user_events_p, did_munge);
@@ -3573,8 +3495,7 @@
 
   /* help-char is `auto-bound' in every keymap */
   if (!NILP (Vprefix_help_command) &&
-      event_matches_key_specifier_p (XEVENT (builder->most_current_event),
-				     Vhelp_char))
+      event_matches_key_specifier_p (builder->most_current_event, Vhelp_char))
     return Vprefix_help_command;
 
   return Qnil;
@@ -3623,13 +3544,7 @@
   if (XEVENT_TYPE (builder->most_current_event) == key_press_event
       && !NILP (Vcomposed_character_default_binding))
     {
-#ifdef USE_KKCC
-      Lisp_Object keysym =
-	XKEY_DATA_KEYSYM(XEVENT_DATA (builder->most_current_event));
-#else /* not USE_KKCC */
-      Lisp_Object keysym =
-	XEVENT (builder->most_current_event)->event.key.keysym;
-#endif /* not USE_KKCC */
+      Lisp_Object keysym = XEVENT_KEY_KEYSYM (builder->most_current_event);
       if (CHARP (keysym) && !ichar_ascii_p (XCHAR (keysym)))
         return Vcomposed_character_default_binding;
     }
@@ -4040,7 +3955,7 @@
     Lisp_Object recent = command_builder->most_current_event;
 
     if (EVENTP (recent)
-	&& event_matches_key_specifier_p (XEVENT (recent), Vmeta_prefix_char))
+	&& event_matches_key_specifier_p (recent, Vmeta_prefix_char))
       {
 	Lisp_Event *e;
 	/* When we see a sequence like "ESC x", pretend we really saw "M-x".
@@ -4052,23 +3967,13 @@
 	   */
 	Fcopy_event (event, recent);
 	e = XEVENT (recent);
-#ifdef USE_KKCC
 	if (EVENT_TYPE (e) == key_press_event)
-          XSET_KEY_DATA_MODIFIERS (EVENT_DATA (e), 
-                          XKEY_DATA_MODIFIERS (EVENT_DATA (e)) | 
-                          XEMACS_MOD_META);
+          SET_EVENT_KEY_MODIFIERS (e, EVENT_KEY_MODIFIERS (e) | 
+				   XEMACS_MOD_META);
 	else if (EVENT_TYPE (e) == button_press_event
 		 || EVENT_TYPE (e) == button_release_event)
-          XSET_BUTTON_DATA_MODIFIERS (EVENT_DATA (e), 
-                              XBUTTON_DATA_MODIFIERS (EVENT_DATA (e)) |
-                              XEMACS_MOD_META);
-#else /* not USE_KKCC */
-	if (e->event_type == key_press_event)
-	  e->event.key.modifiers |= XEMACS_MOD_META;
-	else if (e->event_type == button_press_event
-		 || e->event_type == button_release_event)
-	  e->event.button.modifiers |= XEMACS_MOD_META;
-#endif /* not USE_KKCC */
+          SET_EVENT_BUTTON_MODIFIERS (e, EVENT_BUTTON_MODIFIERS (e) | 
+				      XEMACS_MOD_META);
 	else
 	  abort ();
 
@@ -4125,16 +4030,11 @@
 #if 0
 	else if (!NILP (Vquit_flag))
 	  {
-	    Lisp_Object quit_event = Fmake_event (Qnil, Qnil);
-	    Lisp_Event *e = XEVENT (quit_event);
 	    /* if quit happened during menu acceleration, pretend we read it */
 	    struct console *con = XCONSOLE (Fselected_console ());
-	    int ch = CONSOLE_QUIT_CHAR (con);
-
-	    character_to_event (ch, e, con, 1, 1);
-	    e->channel = wrap_console (con);
-
-	    enqueue_command_event (quit_event);
+
+	    enqueue_command_event (Fcopy_event (CONSOLE_QUIT_EVENT (con),
+						Qnil));
 	    Vquit_flag = Qnil;
 	  }
 #endif
@@ -4162,15 +4062,9 @@
 #ifdef HAVE_SCROLLBARS
   Lisp_Object fun;
 
-#ifdef USE_KKCC
   if (XEVENT_TYPE (event) != misc_user_event)
     return 0;
-  fun = XMISC_USER_DATA_FUNCTION(XEVENT_DATA (event));
-#else /* not USE_KKCC */
-  if (XEVENT (event)->event_type != misc_user_event)
-    return 0;
-  fun = XEVENT (event)->event.misc.function;
-#endif /* not USE_KKCC */
+  fun = XEVENT_MISC_USER_FUNCTION (event);
 
   return (EQ (fun, Qscrollbar_line_up) ||
 	  EQ (fun, Qscrollbar_line_down) ||
@@ -4307,19 +4201,11 @@
 
     pre_command_hook ();
 
-#ifdef USE_KKCC
     if (XEVENT_TYPE (event) == misc_user_event)
       {
-	call1 (XMISC_USER_DATA_FUNCTION (XEVENT_DATA (event)),
-	       XMISC_USER_DATA_OBJECT (XEVENT_DATA (event)));
+	call1 (XEVENT_MISC_USER_FUNCTION (event),
+	       XEVENT_MISC_USER_OBJECT (event));
       }
-#else /* not USE_KKCC */
-    if (XEVENT (event)->event_type == misc_user_event)
-      {
-	call1 (XEVENT (event)->event.eval.function,
-	       XEVENT (event)->event.eval.object);
-      }
-#endif /* not USE_KKCC */
     else
       {
 	Fcommand_execute (Vthis_command, Qnil, Qnil);
@@ -4503,11 +4389,7 @@
     Fselect_console (console);
 
   command_builder = XCOMMAND_BUILDER (XCONSOLE (console)->command_builder);
-#ifdef USE_KKCC
   switch (XEVENT_TYPE (event))
-#else /* not USE_KKCC */
-  switch (XEVENT (event)->event_type)
-#endif /* not USE_KKCC */
     {
     case button_press_event:
     case button_release_event:
@@ -4669,27 +4551,15 @@
 	   (a lambda expression).  So in the `eval' case I'll just
 	   convert it into a lambda expression.
 	   */
-#ifdef USE_KKCC
-	if (EQ (XMISC_USER_DATA_FUNCTION (XEVENT_DATA (event)), Qcall_interactively)
-	    && SYMBOLP (XMISC_USER_DATA_OBJECT (XEVENT_DATA (event))))
-	  Vthis_command = XMISC_USER_DATA_OBJECT (XEVENT_DATA (event));
-	else if (EQ (XMISC_USER_DATA_FUNCTION (XEVENT_DATA (event)), Qeval))
+	if (EQ (XEVENT_MISC_USER_FUNCTION (event), Qcall_interactively)
+	    && SYMBOLP (XEVENT_MISC_USER_OBJECT (event)))
+	  Vthis_command = XEVENT_MISC_USER_OBJECT (event);
+	else if (EQ (XEVENT_MISC_USER_FUNCTION (event), Qeval))
 	  Vthis_command =
-	    Fcons (Qlambda, Fcons (Qnil, XMISC_USER_DATA_OBJECT (XEVENT_DATA (event))));
-	else if (SYMBOLP (XMISC_USER_DATA_FUNCTION (XEVENT_DATA (event))))
+	    Fcons (Qlambda, Fcons (Qnil, XEVENT_MISC_USER_OBJECT (event)));
+	else if (SYMBOLP (XEVENT_MISC_USER_FUNCTION (event)))
 	  /* A scrollbar command or the like. */
-	  Vthis_command = XMISC_USER_DATA_FUNCTION (XEVENT_DATA (event));
-#else /* not USE_KKCC */
-	if (EQ (XEVENT (event)->event.eval.function, Qcall_interactively)
-	    && SYMBOLP (XEVENT (event)->event.eval.object))
-	  Vthis_command = XEVENT (event)->event.eval.object;
-	else if (EQ (XEVENT (event)->event.eval.function, Qeval))
-	  Vthis_command =
-	    Fcons (Qlambda, Fcons (Qnil, XEVENT (event)->event.eval.object));
-	else if (SYMBOLP (XEVENT (event)->event.eval.function))
-	  /* A scrollbar command or the like. */
-	  Vthis_command = XEVENT (event)->event.eval.function;
-#endif /* not USE_KKCC */
+	  Vthis_command = XEVENT_MISC_USER_FUNCTION (event);
 	else
 	  /* Huh? */
 	  Vthis_command = Qnil;
@@ -4781,11 +4651,7 @@
 	execute_internal_event (event);
       else
 	{
-#ifdef USE_KKCC
 	  if (XEVENT_TYPE (event) == misc_user_event)
-#else /* not USE_KKCC */
-	  if (XEVENT (event)->event_type == misc_user_event)
-#endif /* not USE_KKCC */
 	    reset_current_events (command_builder);
 	  result = lookup_command_event (command_builder, event, 1);
 	  if (!KEYMAPP (result))
@@ -4852,19 +4718,11 @@
   if (NILP (Vdribble_file))
     return;
 
-#ifdef USE_KKCC
   if (XEVENT_TYPE (event) == key_press_event &&
-      !XKEY_DATA_MODIFIERS (XEVENT_DATA (event)))
+      !XEVENT_KEY_MODIFIERS (event))
     {
-      Lisp_Object keysym = XKEY_DATA_KEYSYM (XEVENT_DATA (event));
-      if (CHARP (XKEY_DATA_KEYSYM (XEVENT_DATA (event))))
-#else /* not USE_KKCC */
-  if (XEVENT (event)->event_type == key_press_event &&
-      !XEVENT (event)->event.key.modifiers)
-    {
-      Lisp_Object keysym = XEVENT (event)->event.key.keysym;
-      if (CHARP (XEVENT (event)->event.key.keysym))
-#endif /* not USE_KKCC */
+      Lisp_Object keysym = XEVENT_KEY_KEYSYM (event);
+      if (CHARP (XEVENT_KEY_KEYSYM (event)))
 	{
 	  Ichar ch = XCHAR (keysym);
 	  Ibyte str[MAX_ICHAR_LEN];
@@ -5022,12 +4880,17 @@
   Vthis_command_keys = Qnil;
   staticpro (&Vthis_command_keys);
   Vthis_command_keys_tail = Qnil;
-  dump_add_root_object (&Vthis_command_keys_tail);
+  dump_add_root_lisp_object (&Vthis_command_keys_tail);
 
   command_event_queue = Qnil;
   staticpro (&command_event_queue);
   command_event_queue_tail = Qnil;
-  dump_add_root_object (&command_event_queue_tail);
+  dump_add_root_lisp_object (&command_event_queue_tail);
+
+  dispatch_event_queue = Qnil;
+  staticpro (&dispatch_event_queue);
+  dispatch_event_queue_tail = Qnil;
+  dump_add_root_lisp_object (&dispatch_event_queue_tail);
 
   Vlast_selected_frame = Qnil;
   staticpro (&Vlast_selected_frame);
--- a/src/event-tty.c	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/event-tty.c	Sun Jan 12 11:08:22 2003 +0000
@@ -67,20 +67,11 @@
 tty_timeout_to_emacs_event (Lisp_Event *emacs_event)
 {
   /* timeout events have nil as channel */
-#ifdef USE_KKCC
   SET_EVENT_TYPE (emacs_event, timeout_event);
   SET_EVENT_TIMESTAMP_ZERO (emacs_event); /* #### */
-  XSET_TIMEOUT_DATA_INTERVAL_ID (EVENT_DATA (emacs_event), pop_low_level_timeout (&tty_timer_queue, 0));
-  XSET_TIMEOUT_DATA_FUNCTION (EVENT_DATA (emacs_event), Qnil);
-  XSET_TIMEOUT_DATA_OBJECT (EVENT_DATA (emacs_event), Qnil);
-#else /* not USE_KKCC */
-  emacs_event->event_type = timeout_event;
-  emacs_event->timestamp  = 0; /* #### */
-  emacs_event->event.timeout.interval_id =
-    pop_low_level_timeout (&tty_timer_queue, 0);
-  emacs_event->event.timeout.function = Qnil;
-  emacs_event->event.timeout.object = Qnil;
-#endif /* not USE_KKCC */
+  SET_EVENT_TIMEOUT_INTERVAL_ID (emacs_event, pop_low_level_timeout (&tty_timer_queue, 0));
+  SET_EVENT_TIMEOUT_FUNCTION (emacs_event, Qnil);
+  SET_EVENT_TIMEOUT_OBJECT (emacs_event, Qnil);
 }
 
 
@@ -102,23 +93,6 @@
 			     non_fake_input_wait_mask);
 }
 
-struct console *
-tty_find_console_from_fd (int fd)
-{
-  Lisp_Object concons;
-
-  CONSOLE_LOOP (concons)
-    {
-      struct console *c;
-
-      c = XCONSOLE (XCAR (concons));
-      if (CONSOLE_TTY_P (c) && CONSOLE_TTY_DATA (c)->infd == fd)
-	return c;
-    }
-
-  return 0;
-}
-
 static void
 emacs_tty_next_event (Lisp_Event *emacs_event)
 {
@@ -150,7 +124,7 @@
 		 user events ahead of process events. */
 	      if (FD_ISSET (i, &temp_mask) && FD_ISSET (i, &tty_only_mask))
 		{
-		  struct console *c = tty_find_console_from_fd (i);
+		  struct console *c = find_tty_or_stream_console_from_fd (i);
 
 		  assert (c);
 		  if (read_event_from_tty_or_stream_desc (emacs_event, c))
@@ -168,17 +142,10 @@
 
 		  assert (p);
 		  process = wrap_process (p);
-#ifdef USE_KKCC
 		  set_event_type (emacs_event, process_event);
 		  /* process events have nil as channel */
 		  SET_EVENT_TIMESTAMP_ZERO (emacs_event); /* #### */
-		  XSET_PROCESS_DATA_PROCESS (EVENT_DATA (emacs_event), process);
-#else /* not USE_KKCC */
-		  emacs_event->event_type = process_event;
-		  /* process events have nil as channel */
-		  emacs_event->timestamp  = 0; /* #### */
-		  emacs_event->event.process.process = process;
-#endif /* not USE_KKCC */
+		  SET_EVENT_PROCESS_PROCESS (emacs_event, process);
 		  return;
 		}
 	    }
@@ -187,17 +154,10 @@
 	  /* Return a dummy event, so that a cycle of the command loop will
 	     occur. */
 	  drain_signal_event_pipe ();
-#ifdef USE_KKCC
 	  set_event_type (emacs_event, eval_event);
 	  /* eval events have nil as channel */
-	  XSET_EVAL_DATA_FUNCTION (EVENT_DATA (emacs_event), Qidentity);
-	  XSET_EVAL_DATA_OBJECT (EVENT_DATA (emacs_event), Qnil);
-#else /* not USE_KKCC */
-	  emacs_event->event_type = eval_event;
-	  /* eval events have nil as channel */
-	  emacs_event->event.eval.function = Qidentity;
-	  emacs_event->event.eval.object = Qnil;
-#endif /* not USE_KKCC */
+	  SET_EVENT_EVAL_FUNCTION (emacs_event, Qidentity);
+	  SET_EVENT_EVAL_OBJECT (emacs_event, Qnil);
 	  return;
 	}
       else if (ndesc == 0) /* timeout fired */
@@ -262,10 +222,9 @@
 }
 
 static void
-emacs_tty_quit_p (void)
+emacs_tty_drain_queue (void)
 {
-  /* Nothing to do currently because QUIT is handled through SIGINT.
-     This could change. */
+  drain_tty_devices ();
 }
 
 static void
@@ -301,10 +260,9 @@
 void
 reinit_vars_of_event_tty (void)
 {
-  tty_event_stream = xnew (struct event_stream);
+  tty_event_stream = xnew_and_zero (struct event_stream);
 
   tty_event_stream->event_pending_p 	= emacs_tty_event_pending_p;
-  tty_event_stream->force_event_pending = 0;
   tty_event_stream->next_event_cb	= emacs_tty_next_event;
   tty_event_stream->handle_magic_event_cb = emacs_tty_handle_magic_event;
   tty_event_stream->format_magic_event_cb = emacs_tty_format_magic_event;
@@ -316,7 +274,7 @@
   tty_event_stream->unselect_console_cb = emacs_tty_unselect_console;
   tty_event_stream->select_process_cb 	= emacs_tty_select_process;
   tty_event_stream->unselect_process_cb = emacs_tty_unselect_process;
-  tty_event_stream->quit_p_cb		= emacs_tty_quit_p;
+  tty_event_stream->drain_queue_cb	= emacs_tty_drain_queue;
   tty_event_stream->create_io_streams_cb = emacs_tty_create_io_streams;
   tty_event_stream->delete_io_streams_cb = emacs_tty_delete_io_streams;
 }
--- a/src/event-unixoid.c	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/event-unixoid.c	Sun Jan 12 11:08:22 2003 +0000
@@ -31,7 +31,7 @@
 
 #include "console-stream-impl.h"
 #include "console-tty-impl.h"
-#include "device.h"
+#include "device-impl.h"
 #include "events.h"
 #include "lstream.h"
 #include "process.h"
@@ -70,6 +70,24 @@
 
 int fake_event_occurred;
 
+struct console *
+find_tty_or_stream_console_from_fd (int fd)
+{
+  Lisp_Object concons;
+
+  CONSOLE_LOOP (concons)
+    {
+      struct console *c;
+
+      c = XCONSOLE (XCAR (concons));
+      if ((CONSOLE_TTY_P (c) && CONSOLE_TTY_DATA (c)->infd == fd) ||
+	  (CONSOLE_STREAM_P (c) && fileno (CONSOLE_STREAM_DATA (c)->in) == fd))
+	return c;
+    }
+
+  return 0;
+}
+
 int
 read_event_from_tty_or_stream_desc (Lisp_Event *event, struct console *con)
 {
@@ -98,6 +116,8 @@
     }
   else
     {
+      /* Here we really do want to set the use_console_meta_flag because
+         the char is from the TTY. */
       character_to_event (ch, event, con, 1, 1);
       event->channel = console;
       return 1;
@@ -138,6 +158,48 @@
     ;
 }
 
+void
+drain_tty_devices (void)
+{
+  Lisp_Object devcons, concons;
+  CONSOLE_LOOP (concons)
+    {
+      struct console *con = XCONSOLE (XCAR (concons));
+      if (!con->input_enabled)
+	continue;
+
+      CONSOLE_DEVICE_LOOP (devcons, con)
+	{
+	  struct device *d = XDEVICE (XCAR (devcons));
+	  if (DEVICE_TTY_P (d))
+	    {
+	      SELECT_TYPE temp_mask;
+	      int infd = DEVICE_INFD (d);
+
+	      FD_ZERO (&temp_mask);
+	      FD_SET (infd, &temp_mask);
+
+	      while (1)
+		{
+		  Lisp_Object event;
+
+		  if (!poll_fds_for_input (temp_mask))
+		    break;
+
+		  event = Fmake_event (Qnil, Qnil);
+		  if (!read_event_from_tty_or_stream_desc (XEVENT (event),
+							   con))
+		    /* EOF, or something ... */
+		    break;
+
+		  /* queue the read event to be read for real later. */
+		  enqueue_dispatch_event (event);
+		}
+	    }
+	}
+    }
+}
+
 int
 event_stream_unixoid_select_console (struct console *con)
 {
@@ -277,7 +339,7 @@
       /* else, we got interrupted by a signal, so try again. */
     }
 
-  RETURN_NOT_REACHED (0)
+  RETURN_NOT_REACHED (0);
 }
 
 /****************************************************************************/
--- a/src/events.c	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/events.c	Sun Jan 12 11:08:22 2003 +0000
@@ -41,10 +41,6 @@
 
 #include "console-tty-impl.h" /* for stuff in character_to_event */
 
-#ifdef USE_KKCC
-#include "console-x.h"
-#endif /* USE_KKCC */
-
 /* Where old events go when they are explicitly deallocated.
    The event chain here is cut loose before GC, so these will be freed
    eventually.
@@ -73,653 +69,18 @@
   Vevent_resource = Qnil;
 }
 
-#ifdef USE_KKCC
 /* Make sure we lose quickly if we try to use this event */
 static void
 deinitialize_event (Lisp_Object ev)
 {
   Lisp_Event *event = XEVENT (ev);
 
+  int i;
+  for (i = 0; i < (int) (sizeof (Lisp_Event) / sizeof (int)); i++)
+    ((int *) event) [i] = 0xdeadbeef; /* -559038737 base 10 */
+  set_lheader_implementation (&event->lheader, &lrecord_event);
   set_event_type (event, dead_event);
   SET_EVENT_CHANNEL (event, Qnil);
-  set_lheader_implementation (&event->lheader, &lrecord_event);
-  XSET_EVENT_NEXT (ev, Qnil);
-  XSET_EVENT_DATA (ev, Qnil);
-}
-
-/* Set everything to zero or nil so that it's predictable. */
-void
-zero_event (Lisp_Event *e)
-{
-  SET_EVENT_DATA (e, Qnil);
-  set_event_type (e, empty_event);
-  SET_EVENT_NEXT (e, Qnil);
-  SET_EVENT_CHANNEL (e, Qnil);
-  SET_EVENT_TIMESTAMP_ZERO (e);
-}
-
-static const struct lrecord_description event_description [] = {
-  { XD_LISP_OBJECT, offsetof (struct Lisp_Event, next) },
-  { XD_LISP_OBJECT, offsetof (struct Lisp_Event, channel) },
-  { XD_LISP_OBJECT, offsetof (struct Lisp_Event, event_data) },
-  { XD_END }
-};
-
-static Lisp_Object
-mark_event (Lisp_Object obj)
-{
-  mark_object (XEVENT_DATA(obj));
-  mark_object (XEVENT_CHANNEL(obj));
-  return (XEVENT_NEXT(obj));
-}
-
-
-static const struct lrecord_description key_data_description [] = {
-  { XD_LISP_OBJECT, offsetof (struct Lisp_Key_Data, keysym) },
-  { XD_END }
-};
-
-static Lisp_Object
-mark_key_data (Lisp_Object obj)
-{
-  return (XKEY_DATA_KEYSYM(obj));
-}
-
-
-static const struct lrecord_description button_data_description [] = {
-  { XD_END }
-};
-
-static Lisp_Object
-mark_button_data (Lisp_Object obj)
-{
-  return Qnil;
-}
-
-
-static const struct lrecord_description motion_data_description [] = {
-  { XD_END }
-};
-
-static Lisp_Object
-mark_motion_data (Lisp_Object obj)
-{
-  return Qnil;
-}
-
-
-static const struct lrecord_description process_data_description [] = {
-  { XD_LISP_OBJECT, offsetof (struct Lisp_Process_Data, process) },
-  { XD_END }
-};
-
-static Lisp_Object
-mark_process_data (Lisp_Object obj)
-{
-  return (XPROCESS_DATA_PROCESS(obj));
-}
-
-
-static const struct lrecord_description timeout_data_description [] = {
-  { XD_LISP_OBJECT, offsetof (struct Lisp_Timeout_Data, function) },
-  { XD_LISP_OBJECT, offsetof (struct Lisp_Timeout_Data, object) },
-  { XD_END }
-};
-
-static Lisp_Object
-mark_timeout_data (Lisp_Object obj)
-{
-  mark_object (XTIMEOUT_DATA_FUNCTION(obj));
-  return (XTIMEOUT_DATA_OBJECT(obj));
-}
-
-
-static const struct lrecord_description eval_data_description [] = {
-  { XD_LISP_OBJECT, offsetof (struct Lisp_Eval_Data, function) },
-  { XD_LISP_OBJECT, offsetof (struct Lisp_Eval_Data, object) },
-  { XD_END }
-};
-
-static Lisp_Object
-mark_eval_data (Lisp_Object obj)
-{
-  mark_object (XEVAL_DATA_FUNCTION(obj));
-  return (XEVAL_DATA_OBJECT(obj));
-}
-
-
-static const struct lrecord_description misc_user_data_description [] = {
-  { XD_LISP_OBJECT, offsetof (struct Lisp_Misc_User_Data, function) },
-  { XD_LISP_OBJECT, offsetof (struct Lisp_Misc_User_Data, object) },
-  { XD_END }
-};
-
-static Lisp_Object
-mark_misc_user_data (Lisp_Object obj)
-{
-  mark_object (XMISC_USER_DATA_FUNCTION(obj));
-  return (XMISC_USER_DATA_OBJECT(obj));
-}
-
-
-static const struct lrecord_description magic_eval_data_description [] = {
-  { XD_LISP_OBJECT, offsetof (struct Lisp_Magic_Eval_Data, object) },
-  { XD_END }
-};
-
-static Lisp_Object
-mark_magic_eval_data (Lisp_Object obj)
-{
-  return (XMAGIC_EVAL_DATA_OBJECT(obj));
-}
-
-
-static const struct lrecord_description magic_data_description [] = {
-  { XD_END }
-};
-
-static Lisp_Object
-mark_magic_data (Lisp_Object obj)
-{
-  return Qnil;
-}
-
-
-
-static void
-print_event (Lisp_Object obj, Lisp_Object printcharfun, int escapeflag)
-{
-  if (print_readably)
-    printing_unreadable_object ("#<event>");
-
-  switch (XEVENT_TYPE (obj))
-    {
-    case key_press_event:
-      write_c_string (printcharfun, "#<keypress-event ");
-      break;
-    case button_press_event:
-      write_c_string (printcharfun, "#<buttondown-event ");
-      break;
-    case button_release_event:
-      write_c_string (printcharfun, "#<buttonup-event ");
-      break;
-    case magic_eval_event:
-      write_c_string (printcharfun, "#<magic-eval-event ");
-      break;
-    case magic_event:
-      write_c_string (printcharfun, "#<magic-event ");         
-      break;
-    case pointer_motion_event:
-      write_c_string (printcharfun, "#<motion-event ");
-      break;
-    case process_event:
-	write_c_string (printcharfun, "#<process-event ");
-	break;
-    case timeout_event:
-	write_c_string (printcharfun, "#<timeout-event ");
-	break;
-    case misc_user_event:
-	write_c_string (printcharfun, "#<misc-user-event ");
-	break;
-    case eval_event:
-	write_c_string (printcharfun, "#<eval-event ");
-	break;
-    case empty_event:
-	write_c_string (printcharfun, "#<empty-event>");
-        return;
-    case dead_event:
-	write_c_string (printcharfun, "#<DEALLOCATED-EVENT>");
-        return;
-    default:
-	write_c_string (printcharfun, "#<UNKNOWN-EVENT-TYPE>");
-        return;
-      }
-  
-  print_internal (XEVENT_DATA (obj), printcharfun, 1);
-  write_c_string (printcharfun, ">");
-}
-
-
-static void
-print_key_data (Lisp_Object obj, Lisp_Object printcharfun, int escapeflag)
-{
-  char buf[128];
-  if (print_readably)
-    printing_unreadable_object ("#<key_data>");
-
-  sprintf (buf, "#<key_data ");
-  /*  format_event_data_object (buf + 11, obj, 0);
-  sprintf (buf + strlen (buf), ">");
-  write_c_string (printcharfun, buf);*/
-}
-
-static void
-print_button_data (Lisp_Object obj, Lisp_Object printcharfun, int escapeflag)
-{
-  char buf[128];
-  if (print_readably)
-    printing_unreadable_object ("#<button_data>");
-
-  sprintf (buf, "#<button_data ");
-  /*  format_event_data_object (buf + 14, obj, 0);
-  sprintf (buf + strlen (buf), ">");
-  write_c_string (printcharfun, buf);*/
-}
-
-
-static void
-print_motion_data (Lisp_Object obj, Lisp_Object printcharfun, int escapeflag)
-{
-  char buf[64];
-
-  if (print_readably)
-    printing_unreadable_object ("#<motion_data>");
-
-  sprintf (buf, "#<motion-data %ld, %ld>",
-	   (long) XMOTION_DATA_X (obj),
-	   (long) XMOTION_DATA_Y (obj));
-  write_c_string (printcharfun, buf);
-}
-
-
-static void
-print_process_data (Lisp_Object obj, Lisp_Object printcharfun, int escapeflag)
-{
-  if (print_readably)
-    printing_unreadable_object ("#<process_data>");
-
-  write_c_string (printcharfun, "#<process-data ");
-  print_internal (XPROCESS_DATA_PROCESS (obj), printcharfun, 1);
-  write_c_string (printcharfun, ">");
-}
-
-
-static void
-print_timeout_data (Lisp_Object obj, Lisp_Object printcharfun, int escapeflag)
-{
-  if (print_readably)
-    printing_unreadable_object ("#<timeout_data>");
-
-  write_c_string (printcharfun, "#<timeout-data ");
-  print_internal (XTIMEOUT_DATA_OBJECT (obj), printcharfun, 1);
-  write_c_string (printcharfun, ">");
-}
-
-
-static void
-print_eval_data (Lisp_Object obj, Lisp_Object printcharfun, int escapeflag)
-{
-  if (print_readably)
-    printing_unreadable_object ("#<eval_data>");
-
-  write_c_string (printcharfun, "#<eval-data ");
-  print_internal (XEVAL_DATA_FUNCTION (obj), printcharfun, 1);
-  write_c_string (printcharfun, " ");
-  print_internal (XEVAL_DATA_OBJECT (obj), printcharfun, 1);
-  write_c_string (printcharfun, ">");
-}
-
-
-static void
-print_misc_user_data (Lisp_Object obj, Lisp_Object printcharfun, int escapeflag)
-{
-  if (print_readably)
-    printing_unreadable_object ("#<misc_user_data>");
-
-  write_c_string (printcharfun, "#<misc-user-data ");
-  print_internal (XMISC_USER_DATA_FUNCTION (obj), printcharfun, 1);
-  write_c_string (printcharfun, " ");
-  print_internal (XMISC_USER_DATA_OBJECT (obj), printcharfun, 1);
-  write_c_string (printcharfun, ">");
-}
-
-
-static void
-print_magic_eval_data (Lisp_Object obj, Lisp_Object printcharfun, int escapeflag)
-{
-  //  char buf[128];
-
-  if (print_readably)
-    printing_unreadable_object ("#<magic_eval_data>");
-
-  /*  format_event_data_object (buf + 18, obj, 0);*/
-}
-
-
-static void
-print_magic_data (Lisp_Object obj, Lisp_Object printcharfun, int escapeflag)
-{
-  char buf[128];
-
-  if (print_readably)
-    printing_unreadable_object ("#<magic_data>");
-
-  sprintf (buf, "#<magic-data ");
-  /*  format_event_data_object (buf + 13, obj, 0);
-  sprintf (buf + strlen (buf), ">");
-  write_c_string (print_readably, buf);*/
-}
-
-
-static int
-event_equal (Lisp_Object obj1, Lisp_Object obj2, int depth)
-{
-  Lisp_Event *e1 = XEVENT (obj1);
-  Lisp_Event *e2 = XEVENT (obj2);
-
-  if (EVENT_TYPE (e1) != EVENT_TYPE (e2)) return 0;
-  if (!EQ (EVENT_CHANNEL (e1), EVENT_CHANNEL (e2))) return 0;
-/*  if (EVENT_TIMESTAMP (e1) != EVENT_TIMESTAMP (e2)) return 0; */
-  switch (EVENT_TYPE (e1))
-    {
-    default: abort ();
-
-    case process_event:
-    case timeout_event:
-    case pointer_motion_event:
-    case key_press_event:
-    case button_press_event:
-    case button_release_event:
-    case misc_user_event:
-    case eval_event:
-    case magic_eval_event:
-      return internal_equal (EVENT_DATA (e1), EVENT_DATA (e2), 0);
-
-    case magic_event:
-      {
-	struct console *con = XCONSOLE (CDFW_CONSOLE (EVENT_CHANNEL (e1)));
-
-#ifdef HAVE_X_WINDOWS
-	if (CONSOLE_X_P (con))
-	  return (XMAGIC_DATA_X_EVENT (EVENT_DATA (e1)).xany.serial ==
-		  XMAGIC_DATA_X_EVENT (EVENT_DATA (e2)).xany.serial);
-#endif
-#ifdef HAVE_GTK
-	if (CONSOLE_GTK_P (con))
-	  return (XMAGIC_DATA_GTK_EVENT (EVENT_DATA (e1)) ==
-		  XMAGIC_DATA_GTK_EVENT (EVENT_DATA (e2)));
-#endif
-#ifdef HAVE_MS_WINDOWS
-	if (CONSOLE_MSWINDOWS_P (con))
-	  return (XMAGIC_DATA_MSWINDOWS_EVENT (EVENT_DATA (e1)) ==
-		  XMAGIC_DATA_MSWINDOWS_EVENT (EVENT_DATA (e2)));
-#endif
-	abort ();
-	return 1; /* not reached */
-      }
-
-    case empty_event:      /* Empty and deallocated events are equal. */
-    case dead_event:
-      return 1;
-    }
-}
-
-static int
-key_data_equal (Lisp_Object obj1, Lisp_Object obj2, int depth)
-{
-  return (EQ (XKEY_DATA_KEYSYM (obj1), XKEY_DATA_KEYSYM (obj2)) &&
-          (XKEY_DATA_MODIFIERS (obj1) == XKEY_DATA_MODIFIERS (obj2)));
-}
-
-static int
-button_data_equal (Lisp_Object obj1, Lisp_Object obj2, int depth)
-{
-  return (XBUTTON_DATA_BUTTON (obj1) == XBUTTON_DATA_BUTTON (obj2) &&
-          XBUTTON_DATA_MODIFIERS (obj1) == XBUTTON_DATA_MODIFIERS (obj2));
-}
-
-static int
-motion_data_equal (Lisp_Object obj1, Lisp_Object obj2, int depth)
-{
-  return (XMOTION_DATA_X (obj1) == XMOTION_DATA_X (obj2) &&
-          XMOTION_DATA_Y (obj1) == XMOTION_DATA_Y (obj2));
-}
-
-static int
-process_data_equal (Lisp_Object obj1, Lisp_Object obj2, int depth)
-{
-  return EQ (XPROCESS_DATA_PROCESS (obj1), XPROCESS_DATA_PROCESS (obj2));
-}
-
-static int
-timeout_data_equal (Lisp_Object obj1, Lisp_Object obj2, int depth)
-{
-  return (internal_equal (XTIMEOUT_DATA_FUNCTION (obj1),
-                          XTIMEOUT_DATA_FUNCTION (obj2), 0) &&
-          internal_equal (XTIMEOUT_DATA_OBJECT (obj1),
-                          XTIMEOUT_DATA_OBJECT (obj2), 0));
-}
-
-static int
-eval_data_equal (Lisp_Object obj1, Lisp_Object obj2, int depth)
-{
-  return (internal_equal (XEVAL_DATA_FUNCTION (obj1),
-                          XEVAL_DATA_FUNCTION (obj2), 0) &&
-          internal_equal (XEVAL_DATA_OBJECT (obj1),
-                          XEVAL_DATA_OBJECT (obj2), 0));
-}
-
-static int
-misc_user_data_equal (Lisp_Object obj1, Lisp_Object obj2, int depth)
-{
-  return (internal_equal (XMISC_USER_DATA_FUNCTION (obj1), 
-                          XMISC_USER_DATA_FUNCTION (obj2), 0) &&
-          internal_equal (XMISC_USER_DATA_OBJECT (obj1), 
-                          XMISC_USER_DATA_OBJECT (obj2), 0) &&
-	      /* is this really needed for equality
-	         or is x and y also important? */
-          XMISC_USER_DATA_BUTTON (obj1) == XMISC_USER_DATA_BUTTON (obj2) &&
-          XMISC_USER_DATA_MODIFIERS (obj1) == 
-          XMISC_USER_DATA_MODIFIERS (obj2));
-}
-
-static int
-magic_eval_data_equal (Lisp_Object obj1, Lisp_Object obj2, int depth)
-{
-  return (XMAGIC_EVAL_DATA_INTERNAL_FUNCTION (obj1) ==
-          XMAGIC_EVAL_DATA_INTERNAL_FUNCTION (obj2) &&
-          internal_equal (XMAGIC_EVAL_DATA_OBJECT (obj1),
-                          XMAGIC_EVAL_DATA_OBJECT (obj2), 0));
-}
-
-static int
-magic_data_equal (Lisp_Object obj1, Lisp_Object obj2, int depth)
-{assert (0); return 0;}
-
-static unsigned long
-event_hash (Lisp_Object obj, int depth)
-{
-  Lisp_Event *e = XEVENT (obj);
-  unsigned long hash;
-
-  hash = HASH2 (EVENT_TYPE (e), LISP_HASH (EVENT_CHANNEL (e)));
-  switch (EVENT_TYPE (e))
-    {
-    case process_event:
-    case timeout_event:
-    case key_press_event:
-    case button_press_event:
-    case button_release_event:
-    case pointer_motion_event:
-    case misc_user_event:
-    case eval_event:
-    case magic_eval_event:
-      return HASH2 (hash, internal_hash (EVENT_DATA (e), depth + 1));
-
-    case magic_event:
-      {
-	struct console *con = XCONSOLE (CDFW_CONSOLE (EVENT_CHANNEL (e)));
-#ifdef HAVE_X_WINDOWS
-	if (CONSOLE_X_P (con))
-	  return HASH2 (hash, XMAGIC_DATA_X_EVENT (EVENT_DATA (e)).xany.serial);
-#endif
-#ifdef HAVE_GTK
-	if (CONSOLE_GTK_P (con))
-	  return HASH2 (hash, XMAGIC_DATA_GTK_EVENT (EVENT_DATA (e)));
-#endif
-#ifdef HAVE_MS_WINDOWS
-	if (CONSOLE_MSWINDOWS_P (con))
-	  return HASH2 (hash, XMAGIC_DATA_MSWINDOWS_EVENT (EVENT_DATA (e)));
-#endif
-	abort ();
-	return 0;
-      }
-
-    case empty_event:
-    case dead_event:
-      return hash;
-
-    default:
-      abort ();
-    }
-
-  return 0; /* unreached */
-}
-
-static unsigned long
-key_data_hash (Lisp_Object obj, int depth)
-{
-  return HASH2 (LISP_HASH (XKEY_DATA_KEYSYM (obj)),
-                XKEY_DATA_MODIFIERS (obj));
-}
-
-static unsigned long
-button_data_hash (Lisp_Object obj, int depth)
-{
-  return HASH2 (XBUTTON_DATA_BUTTON (obj), XBUTTON_DATA_MODIFIERS (obj));
-}
-
-static unsigned long
-motion_data_hash (Lisp_Object obj, int depth)
-{
-  return HASH2 (XMOTION_DATA_X (obj), XMOTION_DATA_Y (obj));
-}
-
-static unsigned long
-process_data_hash (Lisp_Object obj, int depth)
-{
-  return LISP_HASH (XPROCESS_DATA_PROCESS (obj));
-}
-
-static unsigned long
-timeout_data_hash (Lisp_Object obj, int depth)
-{
-  return HASH2 (internal_hash (XTIMEOUT_DATA_FUNCTION (obj), depth + 1),
-                internal_hash (XTIMEOUT_DATA_OBJECT (obj), depth + 1));
-}
-
-static unsigned long
-eval_data_hash (Lisp_Object obj, int depth)
-{
-  return HASH2 (internal_hash (XEVAL_DATA_FUNCTION (obj), depth + 1),
-                internal_hash (XEVAL_DATA_OBJECT (obj), depth + 1));
-}
-
-static unsigned long
-misc_user_data_hash (Lisp_Object obj, int depth)
-{
-  return HASH4 (internal_hash (XMISC_USER_DATA_FUNCTION (obj), depth + 1),
-                internal_hash (XMISC_USER_DATA_OBJECT (obj), depth + 1),
-                XMISC_USER_DATA_BUTTON (obj), XMISC_USER_DATA_MODIFIERS (obj));
-}
-
-static unsigned long
-magic_eval_data_hash (Lisp_Object obj, int depth)
-{
-  return HASH2 ((unsigned long) XMAGIC_EVAL_DATA_INTERNAL_FUNCTION (obj),
-                internal_hash (XMAGIC_EVAL_DATA_OBJECT (obj), depth + 1));
-}
-
-static unsigned long
-magic_data_hash (Lisp_Object obj, int depth)
-{assert(0); return 1;}
-
-DEFINE_BASIC_LRECORD_IMPLEMENTATION ("key-data", key_data,
-				     0, /*dumpable-flag*/
-				     mark_key_data, 
-				     print_key_data, 0, 
-				     key_data_equal, key_data_hash,
-				     key_data_description, 
-				     Lisp_Key_Data);
-
-DEFINE_BASIC_LRECORD_IMPLEMENTATION ("button-data", button_data,
-				     0, /*dumpable-flag*/
-				     mark_button_data, print_button_data, 0, 
-				     button_data_equal, button_data_hash,
-				     button_data_description, 
-				     Lisp_Button_Data);
-
-DEFINE_BASIC_LRECORD_IMPLEMENTATION ("motion-data", motion_data,
-				     0, /*dumpable-flag*/
-				     mark_motion_data, print_motion_data, 0, 
-				     motion_data_equal, motion_data_hash,
-				     motion_data_description,
-				     Lisp_Motion_Data);
-
-DEFINE_BASIC_LRECORD_IMPLEMENTATION ("process-data", process_data,
-				     0, /*dumpable-flag*/
-				     mark_process_data, 
-				     print_process_data, 0,
-				     process_data_equal, process_data_hash,
-				     process_data_description,
-				     Lisp_Process_Data);
-
-DEFINE_BASIC_LRECORD_IMPLEMENTATION ("timeout-data", timeout_data,
-				     0, /*dumpable-flag*/
-				     mark_timeout_data,
-				     print_timeout_data, 0,
-				     timeout_data_equal, timeout_data_hash,
-				     timeout_data_description,
-				     Lisp_Timeout_Data);
-
-DEFINE_BASIC_LRECORD_IMPLEMENTATION ("eval-data", eval_data,
-				     0, /*dumpable-flag*/
-				     mark_eval_data, 
-				     print_eval_data, 0, 
-				     eval_data_equal, eval_data_hash,
-				     eval_data_description,
-				     Lisp_Eval_Data);
-
-DEFINE_BASIC_LRECORD_IMPLEMENTATION ("misc-user-data", misc_user_data,
-				     0, /*dumpable-flag*/
-				     mark_misc_user_data,
-				     print_misc_user_data,
-				     0, misc_user_data_equal, 
-				     misc_user_data_hash,
-				     misc_user_data_description, 
-				     Lisp_Misc_User_Data);
-
-DEFINE_BASIC_LRECORD_IMPLEMENTATION ("magic-eval-data", magic_eval_data,
-				     0, /*dumpable-flag*/
-				     mark_magic_eval_data, 
-				     print_magic_eval_data, 0, 
-				     magic_eval_data_equal,
-				     magic_eval_data_hash,
-				     magic_eval_data_description, 
-				     Lisp_Magic_Eval_Data);
-
-DEFINE_BASIC_LRECORD_IMPLEMENTATION ("magic-data", magic_data,
-				     0, /*dumpable-flag*/
-				     mark_magic_data, print_magic_data, 0,
-				     magic_data_equal, magic_data_hash,
-				     magic_data_description,
-				     Lisp_Magic_Data);
-
-
-
-#else /* not USE_KKCC */
-/* Make sure we lose quickly if we try to use this event */
-static void
-deinitialize_event (Lisp_Object ev)
-{
-  int i;
-  Lisp_Event *event = XEVENT (ev);
-
-  for (i = 0; i < (int) (sizeof (Lisp_Event) / sizeof (int)); i++)
-    ((int *) event) [i] = 0xdeadbeef;
-  event->event_type = dead_event;
-  event->channel = Qnil;
-  set_lheader_implementation (&event->lheader, &lrecord_event);
   XSET_EVENT_NEXT (ev, Qnil);
 }
 
@@ -729,11 +90,177 @@
 {
   xzero (*e);
   set_lheader_implementation (&e->lheader, &lrecord_event);
-  e->event_type = empty_event;
-  e->next = Qnil;
-  e->channel = Qnil;
+  set_event_type (e, empty_event);
+  SET_EVENT_CHANNEL (e, Qnil);
+  SET_EVENT_NEXT (e, Qnil);
 }
 
+static const struct memory_description key_data_description_1 [] = {
+  { XD_LISP_OBJECT, offsetof (struct Lisp_Key_Data, keysym) },
+  { XD_END }
+};
+
+static const struct sized_memory_description key_data_description = {
+  sizeof (Lisp_Key_Data), key_data_description_1
+};
+
+static const struct memory_description button_data_description_1 [] = {
+  { XD_END }
+};
+
+static const struct sized_memory_description button_data_description = {
+  sizeof (Lisp_Button_Data), button_data_description_1
+};
+
+static const struct memory_description motion_data_description_1 [] = {
+  { XD_END }
+};
+
+static const struct sized_memory_description motion_data_description = {
+  sizeof (Lisp_Motion_Data), motion_data_description_1
+};
+
+static const struct memory_description process_data_description_1 [] = {
+  { XD_LISP_OBJECT, offsetof (struct Lisp_Process_Data, process) },
+  { XD_END }
+};
+
+static const struct sized_memory_description process_data_description = {
+  sizeof (Lisp_Process_Data), process_data_description_1
+};
+
+static const struct memory_description timeout_data_description_1 [] = {
+  { XD_LISP_OBJECT, offsetof (struct Lisp_Timeout_Data, function) },
+  { XD_LISP_OBJECT, offsetof (struct Lisp_Timeout_Data, object) },
+  { XD_END }
+};
+
+static const struct sized_memory_description timeout_data_description = {
+  sizeof (Lisp_Timeout_Data), timeout_data_description_1
+};
+
+static const struct memory_description eval_data_description_1 [] = {
+  { XD_LISP_OBJECT, offsetof (struct Lisp_Eval_Data, function) },
+  { XD_LISP_OBJECT, offsetof (struct Lisp_Eval_Data, object) },
+  { XD_END }
+};
+
+static const struct sized_memory_description eval_data_description = {
+  sizeof (Lisp_Eval_Data), eval_data_description_1
+};
+
+static const struct memory_description misc_user_data_description_1 [] = {
+  { XD_LISP_OBJECT, offsetof (struct Lisp_Misc_User_Data, function) },
+  { XD_LISP_OBJECT, offsetof (struct Lisp_Misc_User_Data, object) },
+  { XD_END }
+};
+
+static const struct sized_memory_description misc_user_data_description = {
+  sizeof (Lisp_Misc_User_Data), misc_user_data_description_1
+};
+
+static const struct memory_description magic_eval_data_description_1 [] = {
+  { XD_LISP_OBJECT, offsetof (struct Lisp_Magic_Eval_Data, object) },
+  { XD_END }
+};
+
+static const struct sized_memory_description magic_eval_data_description = {
+  sizeof (Lisp_Magic_Eval_Data), magic_eval_data_description_1
+};
+
+static const struct memory_description magic_data_description_1 [] = {
+  { XD_END }
+};
+
+static const struct sized_memory_description magic_data_description = {
+  sizeof (Lisp_Magic_Data), magic_data_description_1
+};
+
+static const struct memory_description event_data_description_1 [] = {
+  { XD_STRUCT_ARRAY, key_press_event, 1, &key_data_description },
+  { XD_STRUCT_ARRAY, button_press_event, 1, &button_data_description },
+  { XD_STRUCT_ARRAY, button_release_event, 1, &button_data_description },
+  { XD_STRUCT_ARRAY, pointer_motion_event, 1, &motion_data_description },
+  { XD_STRUCT_ARRAY, process_event, 1, &process_data_description },
+  { XD_STRUCT_ARRAY, timeout_event, 1, &timeout_data_description },
+  { XD_STRUCT_ARRAY, magic_event, 1, &magic_data_description },
+  { XD_STRUCT_ARRAY, magic_eval_event, 1, &magic_eval_data_description },
+  { XD_STRUCT_ARRAY, eval_event, 1, &eval_data_description },
+  { XD_STRUCT_ARRAY, misc_user_event, 1, &misc_user_data_description },
+  { XD_END }
+};
+
+static const struct sized_memory_description event_data_description = {
+  0, event_data_description_1
+};
+
+static const struct memory_description event_description [] = {
+  { XD_INT, offsetof (struct Lisp_Event, event_type) },
+  { XD_LISP_OBJECT, offsetof (struct Lisp_Event, next) },
+  { XD_LISP_OBJECT, offsetof (struct Lisp_Event, channel) },
+  { XD_UNION, offsetof (struct Lisp_Event, event), 
+    XD_INDIRECT (0, 0), &event_data_description },
+  { XD_END }
+};
+
+#ifdef EVENT_DATA_AS_OBJECTS
+
+DEFINE_BASIC_LRECORD_IMPLEMENTATION ("key-data", key_data,
+				     0, /*dumpable-flag*/
+				     0, 0, 0, 0, 0,
+				     key_data_description, 
+				     Lisp_Key_Data);
+
+DEFINE_BASIC_LRECORD_IMPLEMENTATION ("button-data", button_data,
+				     0, /*dumpable-flag*/
+				     0, 0, 0, 0, 0,
+				     button_data_description, 
+				     Lisp_Button_Data);
+
+DEFINE_BASIC_LRECORD_IMPLEMENTATION ("motion-data", motion_data,
+				     0, /*dumpable-flag*/
+				     0, 0, 0, 0, 0,
+				     motion_data_description,
+				     Lisp_Motion_Data);
+
+DEFINE_BASIC_LRECORD_IMPLEMENTATION ("process-data", process_data,
+				     0, /*dumpable-flag*/
+				     0, 0, 0, 0, 0,
+				     process_data_description,
+				     Lisp_Process_Data);
+
+DEFINE_BASIC_LRECORD_IMPLEMENTATION ("timeout-data", timeout_data,
+				     0, /*dumpable-flag*/
+				     0, 0, 0, 0, 0,
+				     timeout_data_description,
+				     Lisp_Timeout_Data);
+
+DEFINE_BASIC_LRECORD_IMPLEMENTATION ("eval-data", eval_data,
+				     0, /*dumpable-flag*/
+				     0, 0, 0, 0, 0,
+				     eval_data_description,
+				     Lisp_Eval_Data);
+
+DEFINE_BASIC_LRECORD_IMPLEMENTATION ("misc-user-data", misc_user_data,
+				     0, /*dumpable-flag*/
+				     0, 0, 0, 0, 0,
+				     misc_user_data_description, 
+				     Lisp_Misc_User_Data);
+
+DEFINE_BASIC_LRECORD_IMPLEMENTATION ("magic-eval-data", magic_eval_data,
+				     0, /*dumpable-flag*/
+				     0, 0, 0, 0, 0,
+				     magic_eval_data_description, 
+				     Lisp_Magic_Eval_Data);
+
+DEFINE_BASIC_LRECORD_IMPLEMENTATION ("magic-data", magic_data,
+				     0, /*dumpable-flag*/
+				     0, 0, 0, 0, 0,
+				     magic_data_description,
+				     Lisp_Magic_Data);
+
+#endif /* EVENT_DATA_AS_OBJECTS */
+
 static Lisp_Object
 mark_event (Lisp_Object obj)
 {
@@ -742,22 +269,22 @@
   switch (event->event_type)
     {
     case key_press_event:
-      mark_object (event->event.key.keysym);
+      mark_object (EVENT_KEY_KEYSYM (event));
       break;
     case process_event:
-      mark_object (event->event.process.process);
+      mark_object (EVENT_PROCESS_PROCESS (event));
       break;
     case timeout_event:
-      mark_object (event->event.timeout.function);
-      mark_object (event->event.timeout.object);
+      mark_object (EVENT_TIMEOUT_FUNCTION (event));
+      mark_object (EVENT_TIMEOUT_OBJECT (event));
       break;
     case eval_event:
     case misc_user_event:
-      mark_object (event->event.eval.function);
-      mark_object (event->event.eval.object);
+      mark_object (EVENT_EVAL_FUNCTION (event));
+      mark_object (EVENT_EVAL_OBJECT (event));
       break;
     case magic_eval_event:
-      mark_object (event->event.magic_eval.object);
+      mark_object (EVENT_MAGIC_EVAL_OBJECT (event));
       break;
     case button_press_event:
     case button_release_event:
@@ -778,7 +305,7 @@
 {
   DECLARE_EISTRING_MALLOC (ei);
   write_c_string (printcharfun, str);
-  format_event_object (ei, XEVENT (obj), 0);
+  format_event_object (ei, obj, 0);
   write_eistring (printcharfun, ei);
   eifree (ei);
 }
@@ -816,21 +343,27 @@
 	break;
       }
     case process_event:
-	write_fmt_string_lisp (printcharfun, "#<process-event %S", 1, XEVENT (obj)->event.process.process);
+	write_fmt_string_lisp (printcharfun, "#<process-event %S", 1,
+			       XEVENT_PROCESS_PROCESS (obj));
 	break;
     case timeout_event:
-	write_fmt_string_lisp (printcharfun, "#<timeout-event %S", 1, XEVENT (obj)->event.timeout.object);
+	write_fmt_string_lisp (printcharfun, "#<timeout-event %S", 1,
+			       XEVENT_TIMEOUT_OBJECT (obj));
 	break;
     case empty_event:
 	write_c_string (printcharfun, "#<empty-event");
 	break;
     case misc_user_event:
-	write_fmt_string_lisp (printcharfun, "#<misc-user-event (%S", 1, XEVENT (obj)->event.misc.function);
-	write_fmt_string_lisp (printcharfun, " %S)", 1, XEVENT (obj)->event.misc.object);
+	write_fmt_string_lisp (printcharfun, "#<misc-user-event (%S", 1,
+			       XEVENT_MISC_USER_FUNCTION (obj));
+	write_fmt_string_lisp (printcharfun, " %S)", 1,
+			       XEVENT_MISC_USER_OBJECT (obj));
 	break;
     case eval_event:
-	write_fmt_string_lisp (printcharfun, "#<eval-event (%S", 1, XEVENT (obj)->event.eval.function);
-	write_fmt_string_lisp (printcharfun, " %S)", 1, XEVENT (obj)->event.eval.object);
+	write_fmt_string_lisp (printcharfun, "#<eval-event (%S", 1,
+			       XEVENT_EVAL_FUNCTION (obj));
+	write_fmt_string_lisp (printcharfun, " %S)", 1,
+			       XEVENT_EVAL_OBJECT (obj));
 	break;
     case dead_event:
 	write_c_string (printcharfun, "#<DEALLOCATED-EVENT");
@@ -856,48 +389,48 @@
     default: abort ();
 
     case process_event:
-      return EQ (e1->event.process.process, e2->event.process.process);
+      return EQ (EVENT_PROCESS_PROCESS (e1), EVENT_PROCESS_PROCESS (e2));
 
     case timeout_event:
-      return (internal_equal (e1->event.timeout.function,
-			      e2->event.timeout.function, 0) &&
-	      internal_equal (e1->event.timeout.object,
-			      e2->event.timeout.object, 0));
+      return (internal_equal (EVENT_TIMEOUT_FUNCTION (e1),
+			      EVENT_TIMEOUT_FUNCTION (e2), 0) &&
+	      internal_equal (EVENT_TIMEOUT_OBJECT (e1),
+			      EVENT_TIMEOUT_OBJECT (e2), 0));
 
     case key_press_event:
-      return (EQ (e1->event.key.keysym, e2->event.key.keysym) &&
-	      (e1->event.key.modifiers == e2->event.key.modifiers));
+      return (EQ (EVENT_KEY_KEYSYM (e1), EVENT_KEY_KEYSYM (e2)) &&
+	      (EVENT_KEY_MODIFIERS (e1) == EVENT_KEY_MODIFIERS (e2)));
 
     case button_press_event:
     case button_release_event:
-      return (e1->event.button.button    == e2->event.button.button &&
-	      e1->event.button.modifiers == e2->event.button.modifiers);
+      return (EVENT_BUTTON_BUTTON (e1)    == EVENT_BUTTON_BUTTON (e2) &&
+	      EVENT_BUTTON_MODIFIERS (e1) == EVENT_BUTTON_MODIFIERS (e2));
 
     case pointer_motion_event:
-      return (e1->event.motion.x == e2->event.motion.x &&
-	      e1->event.motion.y == e2->event.motion.y);
+      return (EVENT_MOTION_X (e1) == EVENT_MOTION_X (e2) &&
+	      EVENT_MOTION_Y (e1) == EVENT_MOTION_Y (e2));
 
     case misc_user_event:
-      return (internal_equal (e1->event.eval.function,
-			      e2->event.eval.function, 0) &&
-	      internal_equal (e1->event.eval.object,
-			      e2->event.eval.object, 0) &&
-	      /* is this really needed for equality
+      return (internal_equal (EVENT_EVAL_FUNCTION (e1),
+			      EVENT_EVAL_FUNCTION (e2), 0) &&
+	      internal_equal (EVENT_EVAL_OBJECT (e1),
+			      EVENT_EVAL_OBJECT (e2), 0) &&
+	      /* #### is this really needed for equality
 	         or is x and y also important? */
-	      e1->event.misc.button    == e2->event.misc.button &&
-	      e1->event.misc.modifiers == e2->event.misc.modifiers);
+	      EVENT_MISC_USER_BUTTON (e1)    == EVENT_MISC_USER_BUTTON (e2) &&
+	      EVENT_MISC_USER_MODIFIERS (e1) == EVENT_MISC_USER_MODIFIERS (e2));
 
     case eval_event:
-      return (internal_equal (e1->event.eval.function,
-			      e2->event.eval.function, 0) &&
-	      internal_equal (e1->event.eval.object,
-			      e2->event.eval.object, 0));
+      return (internal_equal (EVENT_EVAL_FUNCTION (e1),
+			      EVENT_EVAL_FUNCTION (e2), 0) &&
+	      internal_equal (EVENT_EVAL_OBJECT (e1),
+			      EVENT_EVAL_OBJECT (e2), 0));
 
     case magic_eval_event:
-      return (e1->event.magic_eval.internal_function ==
-	      e2->event.magic_eval.internal_function &&
-	      internal_equal (e1->event.magic_eval.object,
-			      e2->event.magic_eval.object, 0));
+      return (EVENT_MAGIC_EVAL_INTERNAL_FUNCTION (e1) ==
+	      EVENT_MAGIC_EVAL_INTERNAL_FUNCTION (e2) &&
+	      internal_equal (EVENT_MAGIC_EVAL_OBJECT (e1),
+			      EVENT_MAGIC_EVAL_OBJECT (e2), 0));
 
     case magic_event:
       return event_stream_compare_magic_event (e1, e2);
@@ -918,36 +451,38 @@
   switch (e->event_type)
     {
     case process_event:
-      return HASH2 (hash, LISP_HASH (e->event.process.process));
+      return HASH2 (hash, LISP_HASH (EVENT_PROCESS_PROCESS (e)));
 
     case timeout_event:
-      return HASH3 (hash, internal_hash (e->event.timeout.function, depth + 1),
-		    internal_hash (e->event.timeout.object, depth + 1));
+      return HASH3 (hash,
+		    internal_hash (EVENT_TIMEOUT_FUNCTION (e), depth + 1),
+		    internal_hash (EVENT_TIMEOUT_OBJECT (e), depth + 1));
 
     case key_press_event:
-      return HASH3 (hash, LISP_HASH (e->event.key.keysym),
-		    e->event.key.modifiers);
+      return HASH3 (hash, LISP_HASH (EVENT_KEY_KEYSYM (e)),
+		    EVENT_KEY_MODIFIERS (e));
 
     case button_press_event:
     case button_release_event:
-      return HASH3 (hash, e->event.button.button, e->event.button.modifiers);
+      return HASH3 (hash, EVENT_BUTTON_BUTTON (e), EVENT_BUTTON_MODIFIERS (e));
 
     case pointer_motion_event:
-      return HASH3 (hash, e->event.motion.x, e->event.motion.y);
+      return HASH3 (hash, EVENT_MOTION_X (e), EVENT_MOTION_Y (e));
 
     case misc_user_event:
-      return HASH5 (hash, internal_hash (e->event.misc.function, depth + 1),
-		    internal_hash (e->event.misc.object, depth + 1),
-		    e->event.misc.button, e->event.misc.modifiers);
+      return HASH5 (hash,
+		    internal_hash (EVENT_MISC_USER_FUNCTION (e), depth + 1),
+		    internal_hash (EVENT_MISC_USER_OBJECT (e), depth + 1),
+		    EVENT_MISC_USER_BUTTON (e), EVENT_MISC_USER_MODIFIERS (e));
 
     case eval_event:
-      return HASH3 (hash, internal_hash (e->event.eval.function, depth + 1),
-		    internal_hash (e->event.eval.object, depth + 1));
+      return HASH3 (hash, internal_hash (EVENT_EVAL_FUNCTION (e), depth + 1),
+		    internal_hash (EVENT_EVAL_OBJECT (e), depth + 1));
 
     case magic_eval_event:
       return HASH3 (hash,
-		    (Hashcode) e->event.magic_eval.internal_function,
-		    internal_hash (e->event.magic_eval.object, depth + 1));
+		    (Hashcode) EVENT_MAGIC_EVAL_INTERNAL_FUNCTION (e),
+		    internal_hash (EVENT_MAGIC_EVAL_OBJECT (e), depth + 1));
 
     case magic_event:
       return HASH2 (hash, event_stream_hash_magic_event (e));
@@ -962,19 +497,12 @@
 
   return 0; /* unreached */
 }
-#endif /* not USE_KKCC */
 
-
-#ifdef USE_KKCC
 DEFINE_BASIC_LRECORD_IMPLEMENTATION ("event", event,
 				     0, /*dumpable-flag*/
 				     mark_event, print_event, 0, event_equal,
-				     event_hash, 0/*event_description*/, Lisp_Event);
-#else /* not USE_KKCC */
-DEFINE_BASIC_LRECORD_IMPLEMENTATION ("event", event,
-				     mark_event, print_event, 0, event_equal,
-				     event_hash, 0, Lisp_Event);
-#endif /* not USE_KKCC */
+				     event_hash, event_description,
+				     Lisp_Event);
 
 DEFUN ("make-event", Fmake_event, 0, 2, 0, /*
 Return a new event of type TYPE, with properties described by PLIST.
@@ -1052,11 +580,7 @@
          PLIST.  In fact, processing PLIST would be wrong, because the
          sanitizing process would fill in the properties
          (e.g. CHANNEL), which we don't want in empty events.  */
-#ifdef USE_KKCC
       set_event_type (e, empty_event);
-#else /* not USE_KKCC */
-      e->event_type = empty_event;
-#endif /* not USE_KKCC */
       if (!NILP (plist))
 	invalid_operation ("Cannot set properties of empty event", plist);
       UNGCPRO;
@@ -1064,42 +588,20 @@
     }
   else if (EQ (type, Qkey_press))
     {
-#ifdef USE_KKCC
       set_event_type (e, key_press_event);
-      XSET_KEY_DATA_KEYSYM (EVENT_DATA (e), Qunbound);
-#else /* not USE_KKCC */
-      e->event_type = key_press_event;
-      e->event.key.keysym = Qunbound;
-#endif /* not USE_KKCC */
+      SET_EVENT_KEY_KEYSYM (e, Qunbound);
     }
   else if (EQ (type, Qbutton_press))
-#ifdef USE_KKCC
     set_event_type (e, button_press_event);
-#else /* not USE_KKCC */
-    e->event_type = button_press_event;
-#endif /* not USE_KKCC */
   else if (EQ (type, Qbutton_release))
-#ifdef USE_KKCC
     set_event_type (e, button_release_event);
-#else /* not USE_KKCC */
-    e->event_type = button_release_event;
-#endif /* not USE_KKCC */
   else if (EQ (type, Qmotion))
-#ifdef USE_KKCC
     set_event_type (e, pointer_motion_event);
-#else /* not USE_KKCC */
-    e->event_type = pointer_motion_event;
-#endif /* not USE_KKCC */
   else if (EQ (type, Qmisc_user))
     {
-#ifdef USE_KKCC
       set_event_type (e, misc_user_event);
-      XSET_MISC_USER_DATA_FUNCTION (EVENT_DATA (e), Qnil); 
-      XSET_MISC_USER_DATA_OBJECT (EVENT_DATA (e), Qnil);
-#else /* not USE_KKCC */
-      e->event_type = misc_user_event;
-      e->event.eval.function = e->event.eval.object = Qnil;
-#endif /* not USE_KKCC */
+      SET_EVENT_MISC_USER_FUNCTION (e, Qnil); 
+      SET_EVENT_MISC_USER_OBJECT (e, Qnil);
     }
   else
     {
@@ -1120,11 +622,7 @@
       {
 	if (EQ (keyword, Qchannel))
 	  {
-#ifdef USE_KKCC
-	    if (EVENT_TYPE(e) == key_press_event)
-#else /* not USE_KKCC */
-	    if (e->event_type == key_press_event)
-#endif /* not USE_KKCC */
+	    if (EVENT_TYPE (e) == key_press_event)
 	      {
 		if (!CONSOLEP (value))
 		  value = wrong_type_argument (Qconsolep, value);
@@ -1138,20 +636,12 @@
 	  }
 	else if (EQ (keyword, Qkey))
 	  {
-#ifdef USE_KKCC
-	    switch (EVENT_TYPE(e))
-#else /* not USE_KKCC */
-	    switch (e->event_type)
-#endif /* not USE_KKCC */
+	    switch (EVENT_TYPE (e))
 	      {
 	      case key_press_event:
 		if (!SYMBOLP (value) && !CHARP (value))
 		  invalid_argument ("Invalid event key", value);
-#ifdef USE_KKCC
-		XSET_KEY_DATA_KEYSYM (EVENT_DATA(e), value);
-#else /* not USE_KKCC */
-		e->event.key.keysym = value;
-#endif /* not USE_KKCC */
+		SET_EVENT_KEY_KEYSYM (e, value);
 		break;
 	      default:
 		WRONG_EVENT_TYPE_FOR_PROPERTY (type, keyword);
@@ -1163,26 +653,14 @@
 	    CHECK_NATNUM (value);
 	    check_int_range (XINT (value), 0, 7);
 
-#ifdef USE_KKCC
-	    switch (EVENT_TYPE(e))
-#else /* not USE_KKCC */
-	    switch (e->event_type)
-#endif /* not USE_KKCC */
+	    switch (EVENT_TYPE (e))
 	      {
 	      case button_press_event:
 	      case button_release_event:
-#ifdef USE_KKCC
-		XSET_BUTTON_DATA_BUTTON (EVENT_DATA (e), XINT (value));
-#else /* not USE_KKCC */
-		e->event.button.button = XINT (value);
-#endif /* not USE_KKCC */
+		SET_EVENT_BUTTON_BUTTON (e, XINT (value));
 		break;
 	      case misc_user_event:
-#ifdef USE_KKCC
-		XSET_MISC_USER_DATA_BUTTON (EVENT_DATA (e), XINT (value));
-#else /* not USE_KKCC */
-		e->event.misc.button = XINT (value);
-#endif /* not USE_KKCC */
+		SET_EVENT_MISC_USER_BUTTON (e, XINT (value));
 		break;
 	      default:
 		WRONG_EVENT_TYPE_FOR_PROPERTY (type, keyword);
@@ -1211,40 +689,20 @@
 		  invalid_constant ("Invalid key modifier", sym);
 	      }
 
-#ifdef USE_KKCC
-	    switch (EVENT_TYPE(e))
-#else /* not USE_KKCC */
-	    switch (e->event_type)
-#endif /* not USE_KKCC */
+	    switch (EVENT_TYPE (e))
 	      {
 	      case key_press_event:
-#ifdef USE_KKCC
-                XSET_KEY_DATA_MODIFIERS (EVENT_DATA (e), modifiers);
-#else /* not USE_KKCC */
-		e->event.key.modifiers = modifiers;
-#endif /* not USE_KKCC */
+                SET_EVENT_KEY_MODIFIERS (e, modifiers);
 		break;
 	      case button_press_event:
 	      case button_release_event:
-#ifdef USE_KKCC
-                XSET_BUTTON_DATA_MODIFIERS (EVENT_DATA (e), modifiers);
-#else /* not USE_KKCC */
-		e->event.button.modifiers = modifiers;
-#endif /* not USE_KKCC */
+                SET_EVENT_BUTTON_MODIFIERS (e, modifiers);
 		break;
 	      case pointer_motion_event:
-#ifdef USE_KKCC
-                XSET_MOTION_DATA_MODIFIERS (EVENT_DATA (e), modifiers);
-#else /* not USE_KKCC */
-		e->event.motion.modifiers = modifiers;
-#endif /* not USE_KKCC */
+                SET_EVENT_MOTION_MODIFIERS (e, modifiers);
 		break;
 	      case misc_user_event:
-#ifdef USE_KKCC
-                XSET_MISC_USER_DATA_MODIFIERS (EVENT_DATA (e), modifiers);
-#else /* not USE_KKCC */
-		e->event.misc.modifiers = modifiers;
-#endif /* not USE_KKCC */
+                SET_EVENT_MISC_USER_MODIFIERS (e, modifiers);
 		break;
 	      default:
 		WRONG_EVENT_TYPE_FOR_PROPERTY (type, keyword);
@@ -1253,11 +711,7 @@
 	  }
 	else if (EQ (keyword, Qx))
 	  {
-#ifdef USE_KKCC
-	    switch (EVENT_TYPE(e))
-#else /* not USE_KKCC */
-	    switch (e->event_type)
-#endif /* not USE_KKCC */
+	    switch (EVENT_TYPE (e))
 	      {
 	      case pointer_motion_event:
 	      case button_press_event:
@@ -1275,11 +729,7 @@
 	  }
 	else if (EQ (keyword, Qy))
 	  {
-#ifdef USE_KKCC
-	    switch (EVENT_TYPE(e))
-#else /* not USE_KKCC */
-	    switch (e->event_type)
-#endif /* not USE_KKCC */
+	    switch (EVENT_TYPE (e))
 	      {
 	      case pointer_motion_event:
 	      case button_press_event:
@@ -1297,26 +747,14 @@
 	else if (EQ (keyword, Qtimestamp))
 	  {
 	    CHECK_NATNUM (value);
-#ifdef USE_KKCC
 	    SET_EVENT_TIMESTAMP (e, XINT (value));
-#else /* not USE_KKCC */
-	    e->timestamp = XINT (value);
-#endif /* not USE_KKCC */
 	  }
 	else if (EQ (keyword, Qfunction))
 	  {
-#ifdef USE_KKCC
-	    switch (EVENT_TYPE(e))
-#else /* not USE_KKCC */
-	    switch (e->event_type)
-#endif /* not USE_KKCC */
+	    switch (EVENT_TYPE (e))
 	      {
 	      case misc_user_event:
-#ifdef USE_KKCC
-                XSET_MISC_USER_DATA_FUNCTION (EVENT_DATA (e), value);
-#else /* not USE_KKCC */
-		e->event.eval.function = value;
-#endif /* not USE_KKCC */
+                SET_EVENT_MISC_USER_FUNCTION (e, value);
 		break;
 	      default:
 		WRONG_EVENT_TYPE_FOR_PROPERTY (type, keyword);
@@ -1325,18 +763,10 @@
 	  }
 	else if (EQ (keyword, Qobject))
 	  {
-#ifdef USE_KKCC
-	    switch (EVENT_TYPE(e))
-#else /* not USE_KKCC */
-	    switch (e->event_type)
-#endif /* not USE_KKCC */
+	    switch (EVENT_TYPE (e))
 	      {
 	      case misc_user_event:
-#ifdef USE_KKCC
-                XSET_MISC_USER_DATA_OBJECT (EVENT_DATA (e), value);
-#else /* not USE_KKCC */
-		e->event.eval.object = value;
-#endif /* not USE_KKCC */
+                SET_EVENT_MISC_USER_OBJECT (e, value);
 		break;
 	      default:
 		WRONG_EVENT_TYPE_FOR_PROPERTY (type, keyword);
@@ -1351,11 +781,7 @@
   /* Insert the channel, if missing. */
   if (NILP (EVENT_CHANNEL (e)))
     {
-#ifdef USE_KKCC
       if (EVENT_TYPE (e) == key_press_event)
-#else /* not USE_KKCC */
-      if (e->event_type == key_press_event)
-#endif /* not USE_KKCC */
 	EVENT_CHANNEL (e) = Vselected_console;
       else
 	EVENT_CHANNEL (e) = Fselected_frame (Qnil);
@@ -1371,80 +797,45 @@
       switch (e->event_type)
 	{
 	case pointer_motion_event:
-#ifdef USE_KKCC
-	  XSET_MOTION_DATA_X (EVENT_DATA (e), coord_x);
-	  XSET_MOTION_DATA_Y (EVENT_DATA (e), coord_y);
-#else /* not USE_KKCC */
-	  e->event.motion.x = coord_x;
-	  e->event.motion.y = coord_y;
-#endif /* not USE_KKCC */
+	  SET_EVENT_MOTION_X (e, coord_x);
+	  SET_EVENT_MOTION_Y (e, coord_y);
 	  break;
 	case button_press_event:
 	case button_release_event:
-#ifdef USE_KKCC
-	  XSET_BUTTON_DATA_X (EVENT_DATA (e), coord_x);
-	  XSET_BUTTON_DATA_Y (EVENT_DATA (e), coord_y);
-#else /* not USE_KKCC */
-	  e->event.button.x = coord_x;
-	  e->event.button.y = coord_y;
-#endif /* not USE_KKCC */
+	  SET_EVENT_BUTTON_X (e, coord_x);
+	  SET_EVENT_BUTTON_Y (e, coord_y);
 	  break;
 	case misc_user_event:
-#ifdef USE_KKCC
-	  XSET_MISC_USER_DATA_X (EVENT_DATA (e), coord_x);
-	  XSET_MISC_USER_DATA_Y (EVENT_DATA (e), coord_y);
-#else /* not USE_KKCC */
-	  e->event.misc.x = coord_x;
-	  e->event.misc.y = coord_y;
-#endif /* not USE_KKCC */
+	  SET_EVENT_MISC_USER_X (e, coord_x);
+	  SET_EVENT_MISC_USER_Y (e, coord_y);
 	  break;
 	default:
-	  abort();
+	  abort ();
 	}
     }
 
   /* Finally, do some more validation.  */
-#ifdef USE_KKCC
-  switch (EVENT_TYPE(e))
-#else /* not USE_KKCC */
-  switch (e->event_type)
-#endif /* not USE_KKCC */
+  switch (EVENT_TYPE (e))
     {
     case key_press_event:
-#ifdef USE_KKCC
-      if (UNBOUNDP (XKEY_DATA_KEYSYM (EVENT_DATA (e))))
-#else /* not USE_KKCC */
-      if (UNBOUNDP (e->event.key.keysym))
-#endif /* not USE_KKCC */
+      if (UNBOUNDP (EVENT_KEY_KEYSYM (e)))
 	sferror ("A key must be specified to make a keypress event",
 		      plist);
       break;
     case button_press_event:
-#ifdef USE_KKCC
-      if (!XBUTTON_DATA_BUTTON (EVENT_DATA (e)))
-#else /* not USE_KKCC */
-      if (!e->event.button.button)
-#endif /* not USE_KKCC */
+      if (!EVENT_BUTTON_BUTTON (e))
 	sferror
 	  ("A button must be specified to make a button-press event",
 	   plist);
       break;
     case button_release_event:
-#ifdef USE_KKCC
-      if (!XBUTTON_DATA_BUTTON (EVENT_DATA (e)))
-#else /* not USE_KKCC */
-      if (!e->event.button.button)
-#endif /* not USE_KKCC */
+      if (!EVENT_BUTTON_BUTTON (e))
 	sferror
 	  ("A button must be specified to make a button-release event",
 	   plist);
       break;
     case misc_user_event:
-#ifdef USE_KKCC
-      if (NILP (XMISC_USER_DATA_FUNCTION (EVENT_DATA (e))))
-#else /* not USE_KKCC */
-      if (NILP (e->event.misc.function))
-#endif /* not USE_KKCC */
+      if (NILP (EVENT_MISC_USER_FUNCTION (e)))
 	sferror ("A function must be specified to make a misc-user event",
 		      plist);
       break;
@@ -1456,114 +847,6 @@
   return event;
 }
 
-
-#ifdef USE_KKCC
-
-Lisp_Object 
-make_key_data (void)
-{
-  Lisp_Object data = allocate_key_data ();
-
-  XSET_KEY_DATA_KEYSYM (data, Qnil);
-  XSET_KEY_DATA_MODIFIERS (data, 0);  
-  
-  return data;
-}
-
-Lisp_Object
-make_button_data (void)
-{
-  Lisp_Object data = allocate_button_data ();
-
-  XSET_BUTTON_DATA_BUTTON (data, 0);
-  XSET_BUTTON_DATA_MODIFIERS (data, 0);
-  XSET_BUTTON_DATA_X (data, 0);
-  XSET_BUTTON_DATA_Y (data, 0);
-  
-  return data;
-}
-
-Lisp_Object
-make_motion_data (void)
-{
-  Lisp_Object data = allocate_motion_data ();
-
-  XSET_MOTION_DATA_X (data, 0);
-  XSET_MOTION_DATA_Y (data, 0);
-  XSET_MOTION_DATA_MODIFIERS (data, 0);
-  
-  return data;
-}
-
-Lisp_Object
-make_process_data (void)
-{
-  Lisp_Object data = allocate_process_data ();
-
-  XSET_PROCESS_DATA_PROCESS (data, Qnil);
-  
-  return data;
-}
-
-Lisp_Object
-make_timeout_data (void)
-{
-  Lisp_Object data = allocate_timeout_data ();
-
-  XSET_TIMEOUT_DATA_INTERVAL_ID (data, 0);
-  XSET_TIMEOUT_DATA_ID_NUMBER(data, 0);
-  XSET_TIMEOUT_DATA_FUNCTION(data, Qnil);
-  XSET_TIMEOUT_DATA_OBJECT (data, Qnil);
-  
-  return data;
-}
-
-Lisp_Object
-make_magic_eval_data (void)
-{
-  Lisp_Object data = allocate_magic_eval_data ();
-
-  XSET_MAGIC_EVAL_DATA_OBJECT (data, Qnil);
-  XSET_MAGIC_EVAL_DATA_INTERNAL_FUNOBJ (data, 0);
-  
-  return data;
-}
-
-Lisp_Object
-make_eval_data (void)
-{
-  Lisp_Object data = allocate_eval_data ();
-
-  XSET_EVAL_DATA_FUNCTION (data, Qnil);
-  XSET_EVAL_DATA_OBJECT (data, Qnil);  
-
-  return data;
-}
-
-Lisp_Object
-make_magic_data (void)
-{
-  return allocate_magic_data ();
-}
-
-Lisp_Object
-make_misc_user_data (void)
-{
-  Lisp_Object data = allocate_misc_user_data ();
-
-  XSET_MISC_USER_DATA_FUNCTION (data, Qnil);
-  XSET_MISC_USER_DATA_OBJECT (data, Qnil);
-  XSET_MISC_USER_DATA_BUTTON (data, 0);
-  XSET_MISC_USER_DATA_MODIFIERS (data, 0);
-  XSET_MISC_USER_DATA_X (data, 0);
-  XSET_MISC_USER_DATA_Y (data, 0);
-
-  return data;
-}
-#endif /* USE_KKCC */
-
-
-
 DEFUN ("deallocate-event", Fdeallocate_event, 1, 1, 0, /*
 Allow the given event structure to be reused.
 You MUST NOT use this event object after calling this function with it.
@@ -1613,60 +896,6 @@
   return Qnil;
 }
 
-#ifdef USE_KKCC
-void
-copy_event_data (Lisp_Object dest, Lisp_Object src)
-{
-  switch (XRECORD_LHEADER (dest)->type) {
-  case lrecord_type_key_data:
-    XSET_KEY_DATA_KEYSYM (dest, XKEY_DATA_KEYSYM (src));
-    XSET_KEY_DATA_MODIFIERS (dest, XKEY_DATA_MODIFIERS (src));
-    break;
-  case lrecord_type_button_data:
-    XSET_BUTTON_DATA_BUTTON (dest, XBUTTON_DATA_BUTTON (src));
-    XSET_BUTTON_DATA_MODIFIERS (dest, XBUTTON_DATA_MODIFIERS (src));
-    XSET_BUTTON_DATA_X (dest, XBUTTON_DATA_X (src));
-    XSET_BUTTON_DATA_Y (dest, XBUTTON_DATA_Y (src));
-    break;
-  case lrecord_type_motion_data:
-    XSET_MOTION_DATA_X (dest, XMOTION_DATA_X (src));
-    XSET_MOTION_DATA_Y (dest, XMOTION_DATA_Y (src));
-    XSET_MOTION_DATA_MODIFIERS (dest, XMOTION_DATA_MODIFIERS (src));
-    break;
-  case lrecord_type_process_data:
-    XSET_PROCESS_DATA_PROCESS (dest, XPROCESS_DATA_PROCESS (src));
-    break;
-  case lrecord_type_timeout_data:
-    XSET_TIMEOUT_DATA_INTERVAL_ID (dest, XTIMEOUT_DATA_INTERVAL_ID (src));
-    XSET_TIMEOUT_DATA_ID_NUMBER (dest, XTIMEOUT_DATA_ID_NUMBER (src));
-    XSET_TIMEOUT_DATA_FUNCTION (dest, XTIMEOUT_DATA_FUNCTION (src));
-    XSET_TIMEOUT_DATA_OBJECT (dest, XTIMEOUT_DATA_OBJECT (src));
-    break;
-  case lrecord_type_eval_data:
-    XSET_EVAL_DATA_FUNCTION (dest, XEVAL_DATA_FUNCTION (src));
-    XSET_EVAL_DATA_OBJECT (dest, XEVAL_DATA_OBJECT (src));
-    break;    
-  case lrecord_type_misc_user_data:
-    XSET_MISC_USER_DATA_FUNCTION (dest, XMISC_USER_DATA_FUNCTION (src));
-    XSET_MISC_USER_DATA_OBJECT (dest, XMISC_USER_DATA_OBJECT (src));
-    XSET_MISC_USER_DATA_BUTTON (dest, XMISC_USER_DATA_BUTTON (src));
-    XSET_MISC_USER_DATA_MODIFIERS (dest, XMISC_USER_DATA_MODIFIERS (src));
-    XSET_MISC_USER_DATA_X (dest, XMISC_USER_DATA_X (src));
-    XSET_MISC_USER_DATA_Y (dest, XMISC_USER_DATA_Y (src));
-    break;
-  case lrecord_type_magic_eval_data:
-    XSET_MAGIC_EVAL_DATA_INTERNAL_FUNCTION (dest, XMAGIC_EVAL_DATA_INTERNAL_FUNCTION (src));
-    XSET_MAGIC_EVAL_DATA_OBJECT (dest, XMAGIC_EVAL_DATA_OBJECT (src));
-    break;
-  case lrecord_type_magic_data:
-    XSET_MAGIC_DATA_UNDERLYING (dest, XMAGIC_DATA_UNDERLYING (src));
-    break;
-  default:
-    break;
-  }
-}
-#endif /* USE_KKCC */
-
 DEFUN ("copy-event", Fcopy_event, 1, 2, 0, /*
 Make a copy of the event object EVENT1.
 If a second event argument EVENT2 is given, EVENT1 is copied into
@@ -1691,26 +920,16 @@
   assert (XEVENT_TYPE (event1) <= last_event_type);
   assert (XEVENT_TYPE (event2) <= last_event_type);
 
-#ifdef USE_KKCC
   XSET_EVENT_TYPE (event2, XEVENT_TYPE (event1));
   XSET_EVENT_CHANNEL (event2, XEVENT_CHANNEL (event1));
   XSET_EVENT_TIMESTAMP (event2, XEVENT_TIMESTAMP (event1));
-  copy_event_data (XEVENT_DATA (event2), XEVENT_DATA (event1));
-
+  
+#ifdef EVENT_DATA_AS_OBJECTS
+  copy_lisp_object (XEVENT_DATA (event2), XEVENT_DATA (event1));
+#else
+  XEVENT (event2)->event = XEVENT (event1)->event;
+#endif
   return event2;
-#else /* not USE_KKCC */
-  {
-    Lisp_Event *ev2 = XEVENT (event2);
-    Lisp_Event *ev1 = XEVENT (event1);
-
-    ev2->event_type = ev1->event_type;
-    ev2->channel    = ev1->channel;
-    ev2->timestamp  = ev1->timestamp;
-    ev2->event      = ev1->event;
-
-    return event2;
-  }
-#endif /* not USE_KKCC */
 }
 
 
@@ -1806,6 +1025,48 @@
     }
 }
 
+/* Map a function over each event in the chain.  If the function returns
+   non-zero, remove the event just processed.  Return the total number of
+   items removed.
+
+   NOTE:
+
+   If you want a simple mapping over an event chain, with no intention to
+   add or remove items, just use EVENT_CHAIN_LOOP().
+*/
+
+int
+map_event_chain_remove (int (*fn) (Lisp_Object ev, void *user_data),
+			Lisp_Object *head, Lisp_Object *tail,
+			void *user_data, int flags)
+{
+  Lisp_Object event;
+  Lisp_Object previous_event = Qnil;
+  int count = 0;
+
+  EVENT_CHAIN_LOOP (event, *head)
+    {
+      if (fn (event, user_data))
+	{
+	  if (NILP (previous_event))
+	    dequeue_event (head, tail);
+	  else
+	    {
+	      XSET_EVENT_NEXT (previous_event, XEVENT_NEXT (event));
+	      if (EQ (*tail, event))
+		*tail = previous_event;
+	    }
+
+	  if (flags & MECR_DEALLOCATE_EVENT)
+	    Fdeallocate_event (event);
+	  count++;
+	}
+      else
+	previous_event = event;
+    }
+  return count;
+}
+
 /* Return the number of events (possibly 0) on an event chain. */
 
 int
@@ -1914,6 +1175,37 @@
     }
 }
 
+/* USE_CONSOLE_META_FLAG is as in `character-to-event'.
+   DO_BACKSPACE_MAPPING means that if CON is a TTY, and C is a the TTY's
+   backspace character, the event will have keysym `backspace' instead of
+   '(control h).  It is clearly correct to do this conversion is the
+   character was just read from a TTY, clearly incorrect inside of
+   define-key, which must be able to handle all consoles.  #### What about
+   in other circumstances?  #### Should the user have access to this flag? 
+
+
+   #### We need to go through and review all the flags in
+   character_to_event() and event_to_character() and figure out exactly
+   under what circumstances they should or should not be set, then go
+   through and review all callers of character_to_event(),
+   Fcharacter_to_event(), event_to_character(), and Fevent_to_character()
+   and check that they are passing the correct flags in for their varied
+   circumstances.
+
+   #### Some of this garbage, and some of the flags, could go away if we
+   implemented the suggestion, originally from event-Xt.c:
+
+   #### The way that keysym correspondence to characters should work:
+   - a Lisp_Event should contain a keysym AND a character slot.
+   - keybindings are tried with the keysym.  If no binding can be found,
+     and there is a corresponding character, call self-insert-command.
+
+   #### Nuke x-iso8859-1.el.
+   #### Nuke the Qascii_character property.
+   #### Nuke Vcharacter_set_property.
+
+   This would apparently solve a lot of different problems.
+*/
 
 void
 character_to_event (Ichar c, Lisp_Event *event, struct console *con,
@@ -1921,11 +1213,7 @@
 {
   Lisp_Object k = Qnil;
   int m = 0;
-#ifdef USE_KKCC
   if (EVENT_TYPE (event) == dead_event)
-#else /* not USE_KKCC */
-  if (event->event_type == dead_event)
-#endif /* not USE_KKCC */
     invalid_argument ("character-to-event called with a deallocated event!", Qunbound);
 
 #ifndef MULE
@@ -1933,6 +1221,9 @@
 #endif
   if (c > 127 && c <= 255)
     {
+      /* #### What if the user wanted a Latin-1 char?  Perhaps the answer
+	 is what was suggested above.
+      */
       int meta_flag = 1;
       if (use_console_meta_flag && CONSOLE_TTY_P (con))
 	meta_flag = TTY_FLAGS (con).meta_key;
@@ -1959,7 +1250,7 @@
 	case 'M': k = QKreturn;	  m &= ~XEMACS_MOD_CONTROL; break;
 	case '[': k = QKescape;	  m &= ~XEMACS_MOD_CONTROL; break;
 	default:
-#if defined(HAVE_TTY)
+#if defined (HAVE_TTY)
 	  if (do_backspace_mapping &&
 	      CHARP (con->tty_erase_char) &&
 	      c - '@' == XCHAR (con->tty_erase_char))
@@ -1967,34 +1258,26 @@
 	      k = QKbackspace;
 	      m &= ~XEMACS_MOD_CONTROL;
 	    }
-#endif /* defined(HAVE_TTY) && !defined(CYGWIN) */
+#endif /* defined (HAVE_TTY) */
 	  break;
 	}
       if (c >= 'A' && c <= 'Z') c -= 'A'-'a';
     }
-#if defined(HAVE_TTY)
+#if defined (HAVE_TTY)
   else if (do_backspace_mapping &&
 	   CHARP (con->tty_erase_char) && c == XCHAR (con->tty_erase_char))
     k = QKbackspace;
-#endif /* defined(HAVE_TTY) && !defined(CYGWIN) */
+#endif /* defined (HAVE_TTY) */
   else if (c == 127)
     k = QKdelete;
   else if (c == ' ')
     k = QKspace;
 
-#ifdef USE_KKCC
   set_event_type (event, key_press_event);
   SET_EVENT_TIMESTAMP_ZERO (event); /* #### */
   SET_EVENT_CHANNEL (event, wrap_console (con));
-  XSET_KEY_DATA_KEYSYM (EVENT_DATA (event), (!NILP (k) ? k : make_char (c)));
-  XSET_KEY_DATA_MODIFIERS (EVENT_DATA (event), m);
-#else /* not USE_KKCC */
-  event->event_type	     = key_press_event;
-  event->timestamp	     = 0; /* #### */
-  event->channel	     = wrap_console (con);
-  event->event.key.keysym    = (!NILP (k) ? k : make_char (c));
-  event->event.key.modifiers = m;
-#endif /* not USE_KKCC */
+  SET_EVENT_KEY_KEYSYM (event, (!NILP (k) ? k : make_char (c)));
+  SET_EVENT_KEY_MODIFIERS (event, m);
 }
 
 /* This variable controls what character name -> character code mapping
@@ -2008,73 +1291,40 @@
 Lisp_Object Vcharacter_set_property;
 
 Ichar
-event_to_character (Lisp_Event *event,
+event_to_character (Lisp_Object event,
 		    int allow_extra_modifiers,
 		    int allow_meta,
-		    int allow_non_ascii)
+		    int map_device_key_names)
 {
   Ichar c = 0;
   Lisp_Object code;
 
-#ifdef USE_KKCC
-  if (EVENT_TYPE (event) != key_press_event)
-#else /* not USE_KKCC */
-  if (event->event_type != key_press_event)
-#endif /* not USE_KKCC */
+  if (XEVENT_TYPE (event) != key_press_event)
     {
-#ifdef USE_KKCC
-      assert (EVENT_TYPE (event) != dead_event);
-#else /* not USE_KKCC */
-      assert (event->event_type != dead_event);
-#endif /* not USE_KKCC */
+      assert (XEVENT_TYPE (event) != dead_event);
       return -1;
     }
   if (!allow_extra_modifiers &&
-#ifdef USE_KKCC
-      XKEY_DATA_MODIFIERS (EVENT_DATA (event)) & (XEMACS_MOD_SUPER|XEMACS_MOD_HYPER|XEMACS_MOD_ALT))
-#else /* not USE_KKCC */
-      event->event.key.modifiers & (XEMACS_MOD_SUPER|XEMACS_MOD_HYPER|XEMACS_MOD_ALT))
-#endif /* not USE_KKCC */
+      XEVENT_KEY_MODIFIERS (event) & (XEMACS_MOD_SUPER|XEMACS_MOD_HYPER|XEMACS_MOD_ALT))
     return -1;
-#ifdef USE_KKCC
-  if (CHAR_OR_CHAR_INTP (XKEY_DATA_KEYSYM (EVENT_DATA (event))))
-    c = XCHAR_OR_CHAR_INT (XKEY_DATA_KEYSYM (EVENT_DATA (event)));
-  else if (!SYMBOLP (XKEY_DATA_KEYSYM (EVENT_DATA (event))))
-#else /* not USE_KKCC */
-  if (CHAR_OR_CHAR_INTP (event->event.key.keysym))
-    c = XCHAR_OR_CHAR_INT (event->event.key.keysym);
-  else if (!SYMBOLP (event->event.key.keysym))
-#endif /* not USE_KKCC */
+  if (CHAR_OR_CHAR_INTP (XEVENT_KEY_KEYSYM (event)))
+    c = XCHAR_OR_CHAR_INT (XEVENT_KEY_KEYSYM (event));
+  else if (!SYMBOLP (XEVENT_KEY_KEYSYM (event)))
     abort ();
-  else if (allow_non_ascii && !NILP (Vcharacter_set_property)
+  else if (map_device_key_names && !NILP (Vcharacter_set_property)
 	   /* Allow window-system-specific extensibility of
 	      keysym->code mapping */
-#ifdef USE_KKCC
-	   && CHAR_OR_CHAR_INTP (code = Fget (XKEY_DATA_KEYSYM (EVENT_DATA (event)),
-					      Vcharacter_set_property,
-					      Qnil)))
-#else /* not USE_KKCC */
-	   && CHAR_OR_CHAR_INTP (code = Fget (event->event.key.keysym,
+	   && CHAR_OR_CHAR_INTP (code = Fget (XEVENT_KEY_KEYSYM (event),
 					      Vcharacter_set_property,
 					      Qnil)))
-#endif /* not USE_KKCC */
     c = XCHAR_OR_CHAR_INT (code);
-#ifdef USE_KKCC
-  else if (CHAR_OR_CHAR_INTP (code = Fget (XKEY_DATA_KEYSYM (EVENT_DATA (event)),
+  else if (CHAR_OR_CHAR_INTP (code = Fget (XEVENT_KEY_KEYSYM (event),
 					   Qascii_character, Qnil)))
-#else /* not USE_KKCC */
-  else if (CHAR_OR_CHAR_INTP (code = Fget (event->event.key.keysym,
-					   Qascii_character, Qnil)))
-#endif /* not USE_KKCC */
     c = XCHAR_OR_CHAR_INT (code);
   else
     return -1;
 
-#ifdef USE_KKCC
-  if (XKEY_DATA_MODIFIERS (EVENT_DATA (event)) & XEMACS_MOD_CONTROL)
-#else /* not USE_KKCC */
-  if (event->event.key.modifiers & XEMACS_MOD_CONTROL)
-#endif /* not USE_KKCC */
+  if (XEVENT_KEY_MODIFIERS (event) & XEMACS_MOD_CONTROL)
     {
       if (c >= 'a' && c <= 'z')
 	c -= ('a' - 'A');
@@ -2092,17 +1342,10 @@
 	if (! allow_extra_modifiers) return -1;
     }
 
-#ifdef USE_KKCC
-  if (XKEY_DATA_MODIFIERS (EVENT_DATA (event)) & XEMACS_MOD_META)
-#else /* not USE_KKCC */
-  if (event->event.key.modifiers & XEMACS_MOD_META)
-#endif /* not USE_KKCC */
+  if (XEVENT_KEY_MODIFIERS (event) & XEMACS_MOD_META)
     {
       if (! allow_meta) return -1;
-      if (c & 0200) return -1;		/* don't allow M-oslash (overlap) */
-#ifdef MULE
-      if (c >= 256) return -1;
-#endif
+      if (c >= 128) return -1;		/* don't allow M-oslash (overlap) */
       c |= 0200;
     }
   return c;
@@ -2119,21 +1362,24 @@
 If the ALLOW-META argument is non-nil, then the Meta modifier will be
  represented by turning on the high bit of the byte returned; otherwise, nil
  will be returned for events containing the Meta modifier.
-If the ALLOW-NON-ASCII argument is non-nil, then characters which are
- present in the prevailing character set (see the `character-set-property'
- variable) will be returned as their code in that character set, instead of
- the return value being restricted to ASCII.
-Note that specifying both ALLOW-META and ALLOW-NON-ASCII is ambiguous, as
- both use the high bit; `M-x' and `oslash' will be indistinguishable.
+If the MAP-DEVICE-KEY-NAMES argument is non-nil, then named keysyms that
+ represent printable characters will be converted to that character.  This
+ means, for example, that under X, where a circumflexed lowercase o returns
+ a key with the name `ocircumflex' rather than the actual character, this
+ name will be converted to the appropriate character.  See
+ `character-set-property' for some sense of how this works. #### This
+ should not be exposed and may be removed at some point.
+Note that ALLOW-META may cause ambiguity between meta characters and
+ Latin-1 characters.
 */
-     (event, allow_extra_modifiers, allow_meta, allow_non_ascii))
+     (event, allow_extra_modifiers, allow_meta, map_device_key_names))
 {
   Ichar c;
   CHECK_LIVE_EVENT (event);
-  c = event_to_character (XEVENT (event),
+  c = event_to_character (event,
 			  !NILP (allow_extra_modifiers),
 			  !NILP (allow_meta),
-			  !NILP (allow_non_ascii));
+			  !NILP (map_device_key_names));
   return c < 0 ? Qnil : make_char (c);
 }
 
@@ -2155,13 +1401,17 @@
 defaults to the selected console.
 
 If KEY-DESCRIPTION is an integer or character, the high bit may be
-interpreted as the meta key. (This is done for backward compatibility
-in lots of places.)  If USE-CONSOLE-META-FLAG is nil, this will always
-be the case.  If USE-CONSOLE-META-FLAG is non-nil, the `meta' flag for
-CONSOLE affects whether the high bit is interpreted as a meta
-key. (See `set-input-mode'.)  If you don't want this silly meta
-interpretation done, you should pass in a list containing the
-character.
+interpreted as the meta key. (This is done for backward compatibility in
+lots of places -- specifically, because lots of Lisp code uses specs like
+?\M-d and "\M-d" in key code, expecting this to work; yet these are in
+reality converted directly to 8-bit characters by the Lisp reader.)  If
+USE-CONSOLE-META-FLAG is nil or CONSOLE is not a TTY, this will always be
+the case.  If USE-CONSOLE-META-FLAG is non-nil and CONSOLE is a TTY, the
+`meta' flag for CONSOLE affects whether the high bit is interpreted as a
+meta key. (See `set-input-mode'.)  Don't set this flag to non-nil unless
+you know what you're doing (more specifically, only if the character came
+directly from a TTY, not from the user).  If you don't want this silly meta
+interpretation done, you should pass in a list containing the character.
 
 Beware that character-to-event and event-to-character are not strictly
 inverse functions, since events contain much more information than the
@@ -2227,33 +1477,19 @@
 /* Concatenate a string description of EVENT onto the end of BUF.  If
    BRIEF, use short forms for keys, e.g. C- instead of control-. */
 
-#ifdef USE_KKCC
 void
 format_event_object (Eistring *buf, Lisp_Object event, int brief)
-#else /* not USE_KKCC */
-void
-format_event_object (Eistring *buf, Lisp_Event *event, int brief)
-#endif /* not USE_KKCC */
 {
   int mouse_p = 0;
   int mod = 0;
   Lisp_Object key;
 
-#ifdef USE_KKCC
-  switch (EVENT_TYPE (XEVENT(event)))
-#else /* not USE_KKCC */
-  switch (event->event_type)
-#endif /* not USE_KKCC */
+  switch (XEVENT_TYPE (event))
     {
     case key_press_event:
       {
-#ifdef USE_KKCC
-	mod = XKEY_DATA_MODIFIERS (XEVENT_DATA(event));
-	key = XKEY_DATA_KEYSYM (XEVENT_DATA(event));
-#else /* not USE_KKCC */
-        mod = event->event.key.modifiers;
-        key = event->event.key.keysym;
-#endif /* not USE_KKCC */
+	mod = XEVENT_KEY_MODIFIERS (event);
+	key = XEVENT_KEY_KEYSYM (event);
         /* Hack. */
         if (! brief && CHARP (key) &&
             mod & (XEMACS_MOD_CONTROL | XEMACS_MOD_META | XEMACS_MOD_SUPER |
@@ -2273,13 +1509,8 @@
     case button_press_event:
       {
         mouse_p++;
-#ifdef USE_KKCC
-	mod = XBUTTON_DATA_MODIFIERS (XEVENT_DATA(event));
-	key = make_char (XBUTTON_DATA_BUTTON (XEVENT_DATA(event)) + '0');
-#else /* not USE_KKCC */
-        mod = event->event.button.modifiers;
-        key = make_char (event->event.button.button + '0');
-#endif /* not USE_KKCC */
+	mod = XEVENT_BUTTON_MODIFIERS (event);
+	key = make_char (XEVENT_BUTTON_BUTTON (event) + '0');
         break;
       }
     case magic_event:
@@ -2289,11 +1520,7 @@
 	GCPRO1 (stream);
 
 	stream = make_resizing_buffer_output_stream ();
-#ifdef USE_KKCC
-	event_stream_format_magic_event (XEVENT(event), stream);
-#else /* not USE_KKCC */
-	event_stream_format_magic_event (event, stream);
-#endif /* not USE_KKCC */
+	event_stream_format_magic_event (XEVENT (event), stream);
 	Lstream_flush (XLSTREAM (stream));
 	eicat_raw (buf, resizing_buffer_stream_ptr (XLSTREAM (stream)),
 		   Lstream_byte_count (XLSTREAM (stream)));
@@ -2355,6 +1582,51 @@
     eicat_c (buf, "up");
 }
 
+void
+upshift_event (Lisp_Object event)
+{
+  Lisp_Object keysym = XEVENT_KEY_KEYSYM (event);
+  Ichar c = 0;
+
+  if (CHAR_OR_CHAR_INTP (keysym)
+      && ((c = XCHAR_OR_CHAR_INT (keysym)),
+	  c >= 'a' && c <= 'z'))
+    XSET_EVENT_KEY_KEYSYM (event, make_char (c + 'A' - 'a'));
+  else
+    if (!(XEVENT_KEY_MODIFIERS (event) & XEMACS_MOD_SHIFT))
+      XSET_EVENT_KEY_MODIFIERS
+	(event, XEVENT_KEY_MODIFIERS (event) |= XEMACS_MOD_SHIFT);
+}
+
+void
+downshift_event (Lisp_Object event)
+{
+  Lisp_Object keysym = XEVENT_KEY_KEYSYM (event);
+  Ichar c = 0;
+
+  if (XEVENT_KEY_MODIFIERS (event) & XEMACS_MOD_SHIFT)
+    XSET_EVENT_KEY_MODIFIERS
+      (event, XEVENT_KEY_MODIFIERS (event) & ~XEMACS_MOD_SHIFT);
+  else if (CHAR_OR_CHAR_INTP (keysym)
+	   && ((c = XCHAR_OR_CHAR_INT (keysym)),
+	       c >= 'A' && c <= 'Z'))
+    XSET_EVENT_KEY_KEYSYM (event, make_char (c + 'a' - 'A'));
+}
+
+int
+event_upshifted_p (Lisp_Object event)
+{
+  Lisp_Object keysym = XEVENT_KEY_KEYSYM (event);
+  Ichar c = 0;
+
+  if ((XEVENT_KEY_MODIFIERS (event) & XEMACS_MOD_SHIFT)
+      || (CHAR_OR_CHAR_INTP (keysym)
+	  && ((c = XCHAR_OR_CHAR_INT (keysym)),
+	      c >= 'A' && c <= 'Z')))
+    return 1;
+  else
+    return 0;
+}
 
 DEFUN ("eventp", Feventp, 1, 1, 0, /*
 True if OBJECT is an event object.
@@ -2369,13 +1641,8 @@
 */
        (object))
 {
-#ifdef USE_KKCC
   return EVENTP (object) && XEVENT_TYPE (object) != dead_event ?
     Qt : Qnil;
-#else /* not USE_KKCC */
-  return EVENTP (object) && XEVENT (object)->event_type != dead_event ?
-    Qt : Qnil;
-#endif /* not USE_KKCC */
 }
 
 #if 0 /* debugging functions */
@@ -2441,11 +1708,7 @@
        (event))
 {
   CHECK_LIVE_EVENT (event);
-#ifdef USE_KKCC
   switch (XEVENT_TYPE (event))
-#else /* not USE_KKCC */
-  switch (XEVENT (event)->event_type)
-#endif /* not USE_KKCC */
     {
     case key_press_event:	return Qkey_press;
     case button_press_event:	return Qbutton_press;
@@ -2481,13 +1744,8 @@
   /* This junk is so that timestamps don't get to be negative, but contain
      as many bits as this particular emacs will allow.
    */
-#ifdef USE_KKCC
   return make_int (((1L << (VALBITS - 1)) - 1) &
 		      XEVENT_TIMESTAMP (event));
-#else /* not USE_KKCC */
-  return make_int (((1L << (VALBITS - 1)) - 1) &
-		      XEVENT (event)->timestamp);
-#endif /* not USE_KKCC */
 }
 
 #define TIMESTAMP_HALFSPACE (1L << (VALBITS - 2))
@@ -2512,21 +1770,12 @@
     return t1 - t2 < TIMESTAMP_HALFSPACE ? Qnil : Qt;
 }
 
-#ifdef USE_KKCC
 #define CHECK_EVENT_TYPE(e,t1,sym) do {		\
   CHECK_LIVE_EVENT (e);				\
   if (XEVENT_TYPE (e) != (t1))	        	\
     e = wrong_type_argument (sym,e);		\
 } while (0)
-#else /* not USE_KKCC */
-#define CHECK_EVENT_TYPE(e,t1,sym) do {		\
-  CHECK_LIVE_EVENT (e);				\
-  if (XEVENT(e)->event_type != (t1))		\
-    e = wrong_type_argument (sym,e);		\
-} while (0)
-#endif /* not USE_KKCC */
 
-#ifdef USE_KKCC
 #define CHECK_EVENT_TYPE2(e,t1,t2,sym) do {		\
   CHECK_LIVE_EVENT (e);					\
   {							\
@@ -2536,19 +1785,7 @@
       e = wrong_type_argument (sym,e);			\
   }							\
 } while (0)
-#else /* not USE_KKCC */
-#define CHECK_EVENT_TYPE2(e,t1,t2,sym) do {		\
-  CHECK_LIVE_EVENT (e);					\
-  {							\
-    emacs_event_type CET_type = XEVENT (e)->event_type;	\
-    if (CET_type != (t1) &&				\
-	CET_type != (t2))				\
-      e = wrong_type_argument (sym,e);			\
-  }							\
-} while (0)
-#endif /* not USE_KKCC */
 
-#ifdef USE_KKCC
 #define CHECK_EVENT_TYPE3(e,t1,t2,t3,sym) do {		\
   CHECK_LIVE_EVENT (e);					\
   {							\
@@ -2559,18 +1796,6 @@
       e = wrong_type_argument (sym,e);			\
   }							\
 } while (0)
-#else /* not USE_KKCC */
-#define CHECK_EVENT_TYPE3(e,t1,t2,t3,sym) do {		\
-  CHECK_LIVE_EVENT (e);					\
-  {							\
-    emacs_event_type CET_type = XEVENT (e)->event_type;	\
-    if (CET_type != (t1) &&				\
-	CET_type != (t2) &&				\
-	CET_type != (t3))				\
-      e = wrong_type_argument (sym,e);			\
-  }							\
-} while (0)
-#endif /* not USE_KKCC */
 
 DEFUN ("event-key", Fevent_key, 1, 1, 0, /*
 Return the Keysym of the key-press event EVENT.
@@ -2579,11 +1804,7 @@
        (event))
 {
   CHECK_EVENT_TYPE (event, key_press_event, Qkey_press_event_p);
-#ifdef USE_KKCC
-  return XKEY_DATA_KEYSYM (XEVENT_DATA (event));
-#else /* not USE_KKCC */
-  return XEVENT (event)->event.key.keysym;
-#endif /* not USE_KKCC */
+  return XEVENT_KEY_KEYSYM (event);
 }
 
 DEFUN ("event-button", Fevent_button, 1, 1, 0, /*
@@ -2591,25 +1812,16 @@
 */
        (event))
 {
-
   CHECK_EVENT_TYPE3 (event, button_press_event, button_release_event,
 		     misc_user_event, Qbutton_event_p);
 #ifdef HAVE_WINDOW_SYSTEM
-#ifdef USE_KKCC
-  if ( XEVENT_TYPE (event) == misc_user_event)
-    return make_int (XMISC_USER_DATA_BUTTON (XEVENT_DATA (event)));
+  if (XEVENT_TYPE (event) == misc_user_event)
+    return make_int (XEVENT_MISC_USER_BUTTON (event));
   else
-    return make_int (XBUTTON_DATA_BUTTON (XEVENT_DATA (event)));
-#else /* not USE_KKCC */
-  if ( XEVENT (event)->event_type == misc_user_event)
-    return make_int (XEVENT (event)->event.misc.button);
-  else
-    return make_int (XEVENT (event)->event.button.button);
-#endif /* not USE_KKCC */
+    return make_int (XEVENT_BUTTON_BUTTON (event));
 #else /* !HAVE_WINDOW_SYSTEM */
   return Qzero;
 #endif /* !HAVE_WINDOW_SYSTEM */
-
 }
 
 DEFUN ("event-modifier-bits", Fevent_modifier_bits, 1, 1, 0, /*
@@ -2621,39 +1833,21 @@
 {
  again:
   CHECK_LIVE_EVENT (event);
-#ifdef USE_KKCC
   switch (XEVENT_TYPE (event))
     {
     case key_press_event:
-      return make_int (XKEY_DATA_MODIFIERS (XEVENT_DATA (event)));
+      return make_int (XEVENT_KEY_MODIFIERS (event));
     case button_press_event:
     case button_release_event:
-      return make_int (XBUTTON_DATA_MODIFIERS (XEVENT_DATA (event)));
+      return make_int (XEVENT_BUTTON_MODIFIERS (event));
     case pointer_motion_event:
-      return make_int (XMOTION_DATA_MODIFIERS (XEVENT_DATA (event)));
+      return make_int (XEVENT_MOTION_MODIFIERS (event));
     case misc_user_event:
-      return make_int (XMISC_USER_DATA_MODIFIERS (XEVENT_DATA (event)));
+      return make_int (XEVENT_MISC_USER_MODIFIERS (event));
     default:
       event = wrong_type_argument (intern ("key-or-mouse-event-p"), event);
       goto again;
     }
-#else /* not USE_KKCC */
-  switch (XEVENT (event)->event_type)
-    {
-    case key_press_event:
-      return make_int (XEVENT (event)->event.key.modifiers);
-    case button_press_event:
-    case button_release_event:
-      return make_int (XEVENT (event)->event.button.modifiers);
-    case pointer_motion_event:
-      return make_int (XEVENT (event)->event.motion.modifiers);
-    case misc_user_event:
-      return make_int (XEVENT (event)->event.misc.modifiers);
-    default:
-      event = wrong_type_argument (intern ("key-or-mouse-event-p"), event);
-      goto again;
-    }
-#endif /* not USE_KKCC */
 }
 
 DEFUN ("event-modifiers", Fevent_modifiers, 1, 1, 0, /*
@@ -2726,45 +1920,24 @@
   struct window *w;
   struct frame *f;
 
-#ifdef USE_KKCC
   if (XEVENT_TYPE (event) == pointer_motion_event)
     {
-      *x = XMOTION_DATA_X (XEVENT_DATA (event));
-      *y = XMOTION_DATA_Y (XEVENT_DATA (event));
+      *x = XEVENT_MOTION_X (event);
+      *y = XEVENT_MOTION_Y (event);
     }
   else if (XEVENT_TYPE (event) == button_press_event ||
 	   XEVENT_TYPE (event) == button_release_event)
     {
-      *x = XBUTTON_DATA_X (XEVENT_DATA (event));
-      *y = XBUTTON_DATA_Y (XEVENT_DATA (event));
+      *x = XEVENT_BUTTON_X (event);
+      *y = XEVENT_BUTTON_Y (event);
     }
   else if (XEVENT_TYPE (event) == misc_user_event)
     {
-      *x = XMISC_USER_DATA_X (XEVENT_DATA (event));
-      *y = XMISC_USER_DATA_Y (XEVENT_DATA (event));
+      *x = XEVENT_MISC_USER_X (event);
+      *y = XEVENT_MISC_USER_Y (event);
     }
   else
     return 0;
-#else /* not USE_KKCC */
-  if (XEVENT (event)->event_type == pointer_motion_event)
-    {
-      *x = XEVENT (event)->event.motion.x;
-      *y = XEVENT (event)->event.motion.y;
-    }
-  else if (XEVENT (event)->event_type == button_press_event ||
-	   XEVENT (event)->event_type == button_release_event)
-    {
-      *x = XEVENT (event)->event.button.x;
-      *y = XEVENT (event)->event.button.y;
-    }
-  else if (XEVENT (event)->event_type == misc_user_event)
-    {
-      *x = XEVENT (event)->event.misc.x;
-      *y = XEVENT (event)->event.misc.y;
-    }
-  else
-    return 0;
-#endif /* not USE_KKCC */
   f = XFRAME (EVENT_CHANNEL (XEVENT (event)));
 
   if (relative)
@@ -2901,47 +2074,25 @@
   Lisp_Object ret_obj1, ret_obj2;
 
   CHECK_LIVE_EVENT (event);
-#ifdef USE_KKCC
   frame = XEVENT_CHANNEL (event);
   switch (XEVENT_TYPE (event))
     {
     case pointer_motion_event :
-      pix_x = XMOTION_DATA_X (XEVENT_DATA (event));
-      pix_y = XMOTION_DATA_Y (XEVENT_DATA (event));
+      pix_x = XEVENT_MOTION_X (event);
+      pix_y = XEVENT_MOTION_Y (event);
       break;
     case button_press_event :
     case button_release_event :
-      pix_x = XBUTTON_DATA_X (XEVENT_DATA (event));
-      pix_y = XBUTTON_DATA_Y (XEVENT_DATA (event));
+      pix_x = XEVENT_BUTTON_X (event);
+      pix_y = XEVENT_BUTTON_Y (event);
       break;
     case misc_user_event :
-      pix_x = XMISC_USER_DATA_X (XEVENT_DATA (event));
-      pix_y = XMISC_USER_DATA_Y (XEVENT_DATA (event));
+      pix_x = XEVENT_MISC_USER_X (event);
+      pix_y = XEVENT_MISC_USER_Y (event);
       break;
     default:
       dead_wrong_type_argument (Qmouse_event_p, event);
     }
-#else /* not USE_KKCC */
-  frame = XEVENT (event)->channel;
-  switch (XEVENT (event)->event_type)
-    {
-    case pointer_motion_event :
-      pix_x = XEVENT (event)->event.motion.x;
-      pix_y = XEVENT (event)->event.motion.y;
-      break;
-    case button_press_event :
-    case button_release_event :
-      pix_x = XEVENT (event)->event.button.x;
-      pix_y = XEVENT (event)->event.button.y;
-      break;
-    case misc_user_event :
-      pix_x = XEVENT (event)->event.misc.x;
-      pix_y = XEVENT (event)->event.misc.y;
-      break;
-    default:
-      dead_wrong_type_argument (Qmouse_event_p, event);
-    }
-#endif /* not USE_KKCC */
 
   result = pixel_to_glyph_translation (XFRAME (frame), pix_x, pix_y,
 				       &ret_x, &ret_y, &ret_obj_x, &ret_obj_y,
@@ -3250,13 +2401,8 @@
 */
        (event))
 {
-#ifdef USE_KKCC
   CHECK_EVENT_TYPE (event, process_event, Qprocess_event_p);
-  return XPROCESS_DATA_PROCESS (XEVENT_DATA (event));
-#else /* not USE_KKCC */
-  CHECK_EVENT_TYPE (event, process_event, Qprocess_event_p);
-  return XEVENT (event)->event.process.process;
-#endif /* not USE_KKCC */
+  return XEVENT_PROCESS_PROCESS (event);
 }
 
 DEFUN ("event-function", Fevent_function, 1, 1, 0, /*
@@ -3267,33 +2413,18 @@
 {
  again:
   CHECK_LIVE_EVENT (event);
-#ifdef USE_KKCC
   switch (XEVENT_TYPE (event))
     {
     case timeout_event:
-      return XTIMEOUT_DATA_FUNCTION (XEVENT_DATA (event));
+      return XEVENT_TIMEOUT_FUNCTION (event);
     case misc_user_event:
-      return XMISC_USER_DATA_FUNCTION (XEVENT_DATA (event));
+      return XEVENT_MISC_USER_FUNCTION (event);
     case eval_event:
-      return XEVAL_DATA_FUNCTION (XEVENT_DATA (event));
+      return XEVENT_EVAL_FUNCTION (event);
     default:
       event = wrong_type_argument (intern ("timeout-or-eval-event-p"), event);
       goto again;
     }
-#else /* not USE_KKCC */
-  switch (XEVENT (event)->event_type)
-    {
-    case timeout_event:
-      return XEVENT (event)->event.timeout.function;
-    case misc_user_event:
-      return XEVENT (event)->event.misc.function;
-    case eval_event:
-      return XEVENT (event)->event.eval.function;
-    default:
-      event = wrong_type_argument (intern ("timeout-or-eval-event-p"), event);
-      goto again;
-    }
-#endif /* not USE_KKCC */
 }
 
 DEFUN ("event-object", Fevent_object, 1, 1, 0, /*
@@ -3304,33 +2435,18 @@
 {
  again:
   CHECK_LIVE_EVENT (event);
-#ifdef USE_KKCC
   switch (XEVENT_TYPE (event))
     {
     case timeout_event:
-      return XTIMEOUT_DATA_OBJECT (XEVENT_DATA (event));
+      return XEVENT_TIMEOUT_OBJECT (event);
     case misc_user_event:
-      return XMISC_USER_DATA_OBJECT (XEVENT_DATA (event));
+      return XEVENT_MISC_USER_OBJECT (event);
     case eval_event:
-      return XEVAL_DATA_OBJECT (XEVENT_DATA (event));
+      return XEVENT_EVAL_OBJECT (event);
     default:
       event = wrong_type_argument (intern ("timeout-or-eval-event-p"), event);
       goto again;
     }
-#else /* not USE_KKCC */
-  switch (XEVENT (event)->event_type)
-    {
-    case timeout_event:
-      return XEVENT (event)->event.timeout.object;
-    case misc_user_event:
-      return XEVENT (event)->event.misc.object;
-    case eval_event:
-      return XEVENT (event)->event.eval.object;
-    default:
-      event = wrong_type_argument (intern ("timeout-or-eval-event-p"), event);
-      goto again;
-    }
-#endif /* not USE_KKCC */
 }
 
 DEFUN ("event-properties", Fevent_properties, 1, 1, 0, /*
@@ -3349,30 +2465,18 @@
 
   props = cons3 (Qtimestamp, Fevent_timestamp (event), props);
 
-#ifdef USE_KKCC
   switch (EVENT_TYPE (e))
-#else /* not USE_KKCC */
-  switch (e->event_type)
-#endif /* not USE_KKCC */
     {
     default: abort ();
 
     case process_event:
-#ifdef USE_KKCC
-      props = cons3 (Qprocess, XPROCESS_DATA_PROCESS (EVENT_DATA (e)), props);
-#else /* not USE_KKCC */
-      props = cons3 (Qprocess, e->event.process.process, props);
-#endif /* not USE_KKCC */
+      props = cons3 (Qprocess, EVENT_PROCESS_PROCESS (e), props);
       break;
 
     case timeout_event:
       props = cons3 (Qobject,	Fevent_object	(event), props);
       props = cons3 (Qfunction, Fevent_function (event), props);
-#ifdef USE_KKCC
-      props = cons3 (Qid, make_int (XTIMEOUT_DATA_ID_NUMBER (EVENT_DATA (e))), props);
-#else /* not USE_KKCC */
-      props = cons3 (Qid, make_int (e->event.timeout.id_number), props);
-#endif /* not USE_KKCC */
+      props = cons3 (Qid, make_int (EVENT_TIMEOUT_ID_NUMBER (e)), props);
       break;
 
     case key_press_event:
@@ -3432,7 +2536,7 @@
 syms_of_events (void)
 {
   INIT_LRECORD_IMPLEMENTATION (event);
-#ifdef USE_KKCC
+#ifdef EVENT_DATA_AS_OBJECTS
   INIT_LRECORD_IMPLEMENTATION (key_data);
   INIT_LRECORD_IMPLEMENTATION (button_data);
   INIT_LRECORD_IMPLEMENTATION (motion_data);
@@ -3442,7 +2546,7 @@
   INIT_LRECORD_IMPLEMENTATION (misc_user_data);
   INIT_LRECORD_IMPLEMENTATION (magic_eval_data);
   INIT_LRECORD_IMPLEMENTATION (magic_data);
-#endif /* USE_KKCC */  
+#endif /* EVENT_DATA_AS_OBJECTS */  
 
   DEFSUBR (Fcharacter_to_event);
   DEFSUBR (Fevent_to_character);
@@ -3520,12 +2624,15 @@
   reinit_vars_of_events ();
 
   DEFVAR_LISP ("character-set-property", &Vcharacter_set_property /*
-A symbol used to look up the 8-bit character of a keysym.
-To convert a keysym symbol to an 8-bit code, as when that key is
-bound to self-insert-command, we will look up the property that this
-variable names on the property list of the keysym-symbol.  The window-
-system-specific code will set up appropriate properties and set this
-variable.
+This is used to map e.g. `ocircumflex' to the appropriate character under X.
+This value of this variable (a symbol, normally `x-iso8859-1' if not nil)
+if used to look up a property on the keysym in question, which should
+correspond to a character.
+
+#### This is way bogus and will be removed soon.
+
+The conversion between X keysyms and characters is now handled more or less
+automatically using XDisplayKeycodes().
 */ );
   Vcharacter_set_property = Qnil;
 }
--- a/src/events.h	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/events.h	Sun Jan 12 11:08:22 2003 +0000
@@ -27,10 +27,6 @@
 
 #include "systime.h"
 
-#ifdef USE_KKCC
-#include "opaque.h"
-#endif /* USE_KKCC */
-
 /* There is one object called an event_stream.  This object contains
    callback functions for doing the window-system-dependent operations
    that XEmacs requires.
@@ -147,17 +143,17 @@
   Stream pairs description
   ------------------------
 
-  Since there are many possible processes/event loop combinations, the event code
-  is responsible for creating an appropriate lstream type. The process
-  implementation does not care about that implementation.
+  Since there are many possible processes/event loop combinations, the
+  event code is responsible for creating an appropriate lstream type. The
+  process implementation does not care about that implementation.
 
-  The Create stream pair function is passed two void* values, which identify
-  process-dependent 'handles'. The process implementation uses these handles
-  to communicate with child processes. The function must be prepared to receive
-  handle types of any process implementation. Since only one process
-  implementation exists in a particular XEmacs configuration, preprocessing
-  is a means of compiling in the support for the code which deals with particular
-  handle types.
+  The Create stream pair function is passed two void* values, which
+  identify process-dependent 'handles'. The process implementation uses
+  these handles to communicate with child processes. The function must be
+  prepared to receive handle types of any process implementation. Since
+  only one process implementation exists in a particular XEmacs
+  configuration, preprocessing is a means of compiling in the support for
+  the code which deals with particular handle types.
 
   For example, a unixoid type loop, which relies on file descriptors, may be
   asked to create a pair of streams by a unix-style process implementation.
@@ -209,8 +205,9 @@
   void (*unselect_console_cb)	(struct console *);
   void (*select_process_cb)	(Lisp_Process *, int doin, int doerr);
   void (*unselect_process_cb)	(Lisp_Process *, int doin, int doerr);
-  void (*quit_p_cb)		(void);
-  void (*force_event_pending)	(struct frame* f);
+  int (*quit_check_disallowed_p_cb)(void);
+  void (*drain_queue_cb)	(void);
+  void (*force_event_pending_cb)(struct frame* f);
   void (*create_io_streams_cb)  (void* /* inhandle*/, void* /*outhandle*/ ,
 				 void * /* errhandle*/,
 				 Lisp_Object* /* instream */,
@@ -231,17 +228,17 @@
 
 extern struct event_stream *event_stream;
 
-#ifdef USE_KKCC
-Lisp_Object make_key_data (void);
-Lisp_Object make_button_data (void);
-Lisp_Object make_motion_data (void);
-Lisp_Object make_process_data (void);
-Lisp_Object make_timeout_data (void);
-Lisp_Object make_magic_data (void);
-Lisp_Object make_magic_eval_data (void);
-Lisp_Object make_eval_data (void);
-Lisp_Object make_misc_user_data (void);
-#endif /* USE_KKCC */
+#ifdef EVENT_DATA_AS_OBJECTS
+#define EVENT_FOO_BAR_1(extractor, field)  ((extractor)->field)
+#define EVENT_FOO_BAR(e, uptype, downtype, field) EVENT_FOO_BAR_1 (X##uptype##_DATA (EVENT_DATA (e)), field)
+#define SET_EVENT_FOO_BAR_1(extractor, field, val)  \
+do { (extractor)->field = (val); } while (0)
+#define SET_EVENT_FOO_BAR(e, uptype, downtype, field, val) SET_EVENT_FOO_BAR_1 (X##uptype##_DATA (EVENT_DATA (e)), field, val)
+#else
+#define EVENT_FOO_BAR(e, uptype, downtype, field) ((e)->event.downtype.field)
+#define SET_EVENT_FOO_BAR(e, uptype, downtype, field, val) \
+do { (e)->event.downtype.field = (val); } while (0)
+#endif
 
 typedef enum emacs_event_type
 {
@@ -278,15 +275,11 @@
 
 #endif /* MULE */
 
-#ifdef USE_KKCC
 struct Lisp_Key_Data
-#else /* not USE_KKCC */
-struct key_data
-#endif /* not USE_KKCC */
 {
-#ifdef USE_KKCC
+#ifdef EVENT_DATA_AS_OBJECTS
   struct lrecord_header lheader;
-#endif /* USE_KKCC */
+#endif /* EVENT_DATA_AS_OBJECTS */
   /* What keysym this is; a character or a symbol. */
   Lisp_Object keysym;
   /* Modifiers held down when key was pressed: control, meta, etc.
@@ -315,36 +308,49 @@
 #endif /* MULE */
 };
 
-#ifdef USE_KKCC
 typedef struct Lisp_Key_Data Lisp_Key_Data;
 
+#define KEY_DATA_KEYSYM(d) ((d)->keysym)
+#define KEY_DATA_MODIFIERS(d) ((d)->modifiers)
+#define SET_KEY_DATA_KEYSYM(d, k) ((d)->keysym = k)
+#define SET_KEY_DATA_MODIFIERS(d, m) ((d)->modifiers = m)
+
+#ifdef EVENT_DATA_AS_OBJECTS
 DECLARE_LRECORD (key_data, Lisp_Key_Data);
 #define XKEY_DATA(x) XRECORD (x, key_data, Lisp_Key_Data)
 #define wrap_key_data(p) wrap_record (p, key_data)
 #define KEY_DATAP(x) RECORDP (x, key_data)
 #define CHECK_KEY_DATA(x) CHECK_RECORD (x, key_data)
 #define CONCHECK_KEY_DATA(x) CONCHECK_RECORD (x, key_data)
+#endif /* EVENT_DATA_AS_OBJECTS */
 
-#define XKEY_DATA_KEYSYM(d) (XKEY_DATA (d)->keysym)
-#define KEY_DATA_KEYSYM(d) ((d)->keysym)
-#define XKEY_DATA_MODIFIERS(d) (XKEY_DATA (d)->modifiers)
-#define KEY_DATA_MODIFIERS(d) ((d)->modifiers)
+#define EVENT_KEY_KEYSYM(e) EVENT_FOO_BAR (e, KEY, key, keysym)
+#define XEVENT_KEY_KEYSYM(e) EVENT_KEY_KEYSYM (XEVENT (e))
+#define SET_EVENT_KEY_KEYSYM(e, val) \
+  SET_EVENT_FOO_BAR (e, KEY, key, keysym, val)
+#define XSET_EVENT_KEY_KEYSYM(e, val) \
+  SET_EVENT_KEY_KEYSYM (XEVENT (e), val)
 
-#define XSET_KEY_DATA_KEYSYM(d, k) (XKEY_DATA (d)->keysym = (k))
-#define SET_KEY_DATA_KEYSYM(d, k) ((d)->keysym = k)
-#define XSET_KEY_DATA_MODIFIERS(d, m) (XKEY_DATA (d)->modifiers = m)
-#define SET_KEY_DATA_MODIFIERS(d, m) ((d)->modifiers = m)
-#endif /* USE_KKCC */
+#define EVENT_KEY_MODIFIERS(e) EVENT_FOO_BAR (e, KEY, key, modifiers)
+#define XEVENT_KEY_MODIFIERS(e) EVENT_KEY_MODIFIERS (XEVENT (e))
+#define SET_EVENT_KEY_MODIFIERS(e, val) \
+  SET_EVENT_FOO_BAR (e, KEY, key, modifiers, val)
+#define XSET_EVENT_KEY_MODIFIERS(e, val) \
+  SET_EVENT_KEY_MODIFIERS (XEVENT (e), val)
 
-#ifdef USE_KKCC
+#define EVENT_KEY_ALT_KEYCHARS(e, n) \
+  EVENT_FOO_BAR (e, KEY, key, alt_keychars[n])
+#define XEVENT_KEY_ALT_KEYCHARS(e, n) EVENT_KEY_ALT_KEYCHARS (XEVENT (e), n)
+#define SET_EVENT_KEY_ALT_KEYCHARS(e, n, val) \
+  SET_EVENT_FOO_BAR (e, KEY, key, alt_keychars[n], val)
+#define XSET_EVENT_KEY_ALT_KEYCHARS(e, n, val) \
+  SET_EVENT_KEY_ALT_KEYCHARS (XEVENT (e), n, val)
+
 struct Lisp_Button_Data
-#else /* not USE_KKCC */
-struct button_data
-#endif /* not USE_KKCC */
 {
-#ifdef USE_KKCC
+#ifdef EVENT_DATA_AS_OBJECTS
   struct lrecord_header lheader;
-#endif /* USE_KKCC */
+#endif /* EVENT_DATA_AS_OBJECTS */
   /* What button went down or up. */
   int button;
   /* Bucky-bits on that button: shift, control, meta, etc.  Also
@@ -353,91 +359,114 @@
   /*  Where it was at the button-state-change (in pixels). */
   int x, y;
 };
-#ifdef USE_KKCC
 typedef struct Lisp_Button_Data Lisp_Button_Data;
 
+#ifdef EVENT_DATA_AS_OBJECTS
 DECLARE_LRECORD (button_data, Lisp_Button_Data);
 #define XBUTTON_DATA(x) XRECORD (x, button_data, Lisp_Button_Data)
 #define wrap_button_data(p) wrap_record (p, button_data)
 #define BUTTON_DATAP(x) RECORDP (x, button_data)
 #define CHECK_BUTTON_DATA(x) CHECK_RECORD (x, button_data)
 #define CONCHECK_BUTTON_DATA(x) CONCHECK_RECORD (x, button_data)
+#endif /* EVENT_DATA_AS_OBJECTS */
 
-#define XBUTTON_DATA_BUTTON(d) (XBUTTON_DATA (d)->button)
-#define XBUTTON_DATA_MODIFIERS(d) (XBUTTON_DATA (d)->modifiers)
-#define XBUTTON_DATA_X(d) (XBUTTON_DATA (d)->x)
-#define XBUTTON_DATA_Y(d) (XBUTTON_DATA (d)->y)
+#define EVENT_BUTTON_BUTTON(e) EVENT_FOO_BAR (e, BUTTON, button, button)
+#define XEVENT_BUTTON_BUTTON(e) EVENT_BUTTON_BUTTON (XEVENT (e))
+#define SET_EVENT_BUTTON_BUTTON(e, val) \
+  SET_EVENT_FOO_BAR (e, BUTTON, button, button, val)
+#define XSET_EVENT_BUTTON_BUTTON(e, val) \
+  SET_EVENT_BUTTON_BUTTON (XEVENT (e), val)
+
+#define EVENT_BUTTON_MODIFIERS(e) EVENT_FOO_BAR (e, BUTTON, button, modifiers)
+#define XEVENT_BUTTON_MODIFIERS(e) EVENT_BUTTON_MODIFIERS (XEVENT (e))
+#define SET_EVENT_BUTTON_MODIFIERS(e, val) \
+  SET_EVENT_FOO_BAR (e, BUTTON, button, modifiers, val)
+#define XSET_EVENT_BUTTON_MODIFIERS(e, val) \
+  SET_EVENT_BUTTON_MODIFIERS (XEVENT (e), val)
 
-#define XSET_BUTTON_DATA_BUTTON(d, b) (XBUTTON_DATA (d)->button = (b))
-#define XSET_BUTTON_DATA_MODIFIERS(d, m) (XBUTTON_DATA (d)->modifiers = (m))
-#define XSET_BUTTON_DATA_X(d, new_x) (XBUTTON_DATA (d)->x = (new_x))
-#define XSET_BUTTON_DATA_Y(d, new_y) (XBUTTON_DATA (d)->y = (new_y))
-#endif /* USE_KKCC */
+#define EVENT_BUTTON_X(e) EVENT_FOO_BAR (e, BUTTON, button, x)
+#define XEVENT_BUTTON_X(e) EVENT_BUTTON_X (XEVENT (e))
+#define SET_EVENT_BUTTON_X(e, val) \
+  SET_EVENT_FOO_BAR (e, BUTTON, button, x, val)
+#define XSET_EVENT_BUTTON_X(e, val) \
+  SET_EVENT_BUTTON_X (XEVENT (e), val)
 
-#ifdef USE_KKCC
+#define EVENT_BUTTON_Y(e) EVENT_FOO_BAR (e, BUTTON, button, y)
+#define XEVENT_BUTTON_Y(e) EVENT_BUTTON_Y (XEVENT (e))
+#define SET_EVENT_BUTTON_Y(e, val) \
+  SET_EVENT_FOO_BAR (e, BUTTON, button, y, val)
+#define XSET_EVENT_BUTTON_Y(e, val) \
+  SET_EVENT_BUTTON_Y (XEVENT (e), val)
+
 struct Lisp_Motion_Data
-#else /* not USE_KKCC */
-struct motion_data
-#endif /* not USE_KKCC */
 {
-#ifdef USE_KKCC
+#ifdef EVENT_DATA_AS_OBJECTS
   struct lrecord_header lheader;
-#endif /* USE_KKCC */
+#endif /* EVENT_DATA_AS_OBJECTS */
   /* Where it was after it moved (in pixels). */
   int x, y;
   /* Bucky-bits down when the motion was detected. */
   int modifiers;
 };
-#ifdef USE_KKCC
 typedef struct Lisp_Motion_Data Lisp_Motion_Data;
 
+#ifdef EVENT_DATA_AS_OBJECTS
 DECLARE_LRECORD (motion_data, Lisp_Motion_Data);
 #define XMOTION_DATA(x) XRECORD (x, motion_data, Lisp_Motion_Data)
 #define wrap_motion_data(p) wrap_record (p, motion_data)
 #define MOTION_DATAP(x) RECORDP (x, motion_data)
 #define CHECK_MOTION_DATA(x) CHECK_RECORD (x, motion_data)
 #define CONCHECK_MOTION_DATA(x) CONCHECK_RECORD (x, motion_data)
-
-#define XMOTION_DATA_X(d) (XMOTION_DATA (d)->x)
-#define XMOTION_DATA_Y(d) (XMOTION_DATA (d)->y)
-#define XMOTION_DATA_MODIFIERS(d) (XMOTION_DATA (d)->modifiers)
+#endif /* EVENT_DATA_AS_OBJECTS */
 
-#define XSET_MOTION_DATA_X(d, new_x) (XMOTION_DATA (d)->x = (new_x))
-#define XSET_MOTION_DATA_Y(d, new_y) (XMOTION_DATA (d)->y = (new_y))
-#define XSET_MOTION_DATA_MODIFIERS(d, m) (XMOTION_DATA (d)->modifiers = (m))
-#endif /* USE_KKCC */
+#define EVENT_MOTION_X(e) EVENT_FOO_BAR (e, MOTION, motion, x)
+#define XEVENT_MOTION_X(e) EVENT_MOTION_X (XEVENT (e))
+#define SET_EVENT_MOTION_X(e, val) \
+  SET_EVENT_FOO_BAR (e, MOTION, motion, x, val)
+#define XSET_EVENT_MOTION_X(e, val) \
+  SET_EVENT_MOTION_X (XEVENT (e), val)
 
-#ifdef USE_KKCC
+#define EVENT_MOTION_Y(e) EVENT_FOO_BAR (e, MOTION, motion, y)
+#define XEVENT_MOTION_Y(e) EVENT_MOTION_Y (XEVENT (e))
+#define SET_EVENT_MOTION_Y(e, val) \
+  SET_EVENT_FOO_BAR (e, MOTION, motion, y, val)
+#define XSET_EVENT_MOTION_Y(e, val) \
+  SET_EVENT_MOTION_Y (XEVENT (e), val)
+
+#define EVENT_MOTION_MODIFIERS(e) EVENT_FOO_BAR (e, MOTION, motion, modifiers)
+#define XEVENT_MOTION_MODIFIERS(e) EVENT_MOTION_MODIFIERS (XEVENT (e))
+#define SET_EVENT_MOTION_MODIFIERS(e, val) \
+  SET_EVENT_FOO_BAR (e, MOTION, motion, modifiers, val)
+#define XSET_EVENT_MOTION_MODIFIERS(e, val) \
+  SET_EVENT_MOTION_MODIFIERS (XEVENT (e), val)
+
 struct Lisp_Process_Data
-#else /* not USE_KKCC */
-struct process_data
-#endif /* not USE_KKCC */
 {
-#ifdef USE_KKCC
+#ifdef EVENT_DATA_AS_OBJECTS
   struct lrecord_header lheader;
-#endif /* USE_KKCC */
+#endif /* EVENT_DATA_AS_OBJECTS */
   /* the XEmacs "process" object in question */
   Lisp_Object process;
 };
-#ifdef USE_KKCC
 typedef struct Lisp_Process_Data Lisp_Process_Data;
 
+#ifdef EVENT_DATA_AS_OBJECTS
 DECLARE_LRECORD (process_data, Lisp_Process_Data);
 #define XPROCESS_DATA(x) XRECORD (x, process_data, Lisp_Process_Data)
 #define wrap_process_data(p) wrap_record (p, process_data)
 #define PROCESS_DATAP(x) RECORDP (x, process_data)
 #define CHECK_PROCESS_DATA(x) CHECK_RECORD (x, process_data)
 #define CONCHECK_PROCESS_DATA(x) CONCHECK_RECORD (x, process_data)
-
-#define XPROCESS_DATA_PROCESS(d) (XPROCESS_DATA (d)->process)
-#define XSET_PROCESS_DATA_PROCESS(d, p) (XPROCESS_DATA (d)->process = (p))
-#endif /* USE_KKCC */
+#endif /* EVENT_DATA_AS_OBJECTS */
 
-#ifdef USE_KKCC
+#define EVENT_PROCESS_PROCESS(e) EVENT_FOO_BAR (e, PROCESS, process, process)
+#define XEVENT_PROCESS_PROCESS(e) EVENT_PROCESS_PROCESS (XEVENT (e))
+#define SET_EVENT_PROCESS_PROCESS(e, val) \
+  SET_EVENT_FOO_BAR (e, PROCESS, process, process, val)
+#define XSET_EVENT_PROCESS_PROCESS(e, val) \
+  SET_EVENT_PROCESS_PROCESS (XEVENT (e), val)
+
 struct Lisp_Timeout_Data
-#else /* not USE_KKCC */
-struct timeout_data
-#endif /* not USE_KKCC */
 {
 /*
     interval_id		The ID returned when the associated call to
@@ -452,40 +481,54 @@
 			processed.
     object		The object passed to that function.
 */
-#ifdef USE_KKCC
+#ifdef EVENT_DATA_AS_OBJECTS
   struct lrecord_header lheader;
-#endif /* USE_KKCC */
+#endif /* EVENT_DATA_AS_OBJECTS */
   int interval_id;
   int id_number;
   Lisp_Object function;
   Lisp_Object object;
 };
-#ifdef USE_KKCC
 typedef struct Lisp_Timeout_Data Lisp_Timeout_Data;
 
+#ifdef EVENT_DATA_AS_OBJECTS
 DECLARE_LRECORD (timeout_data, Lisp_Timeout_Data);
 #define XTIMEOUT_DATA(x) XRECORD (x, timeout_data, Lisp_Timeout_Data)
 #define wrap_timeout_data(p) wrap_record(p, timeout_data)
 #define TIMEOUT_DATAP(x) RECORDP (x, timeout_data)
 #define CHECK_TIMEOUT_DATA(x) CHECK_RECORD (x, timeout_data)
 #define CONCHECK_TIMEOUT_DATA(x) CONCHECK_RECORD (x, timeout_data)
+#endif /* EVENT_DATA_AS_OBJECTS */
 
-#define XTIMEOUT_DATA_INTERVAL_ID(d) XTIMEOUT_DATA (d)->interval_id
-#define XTIMEOUT_DATA_ID_NUMBER(d) XTIMEOUT_DATA (d)->id_number
-#define XTIMEOUT_DATA_FUNCTION(d) XTIMEOUT_DATA (d)->function
-#define XTIMEOUT_DATA_OBJECT(d) XTIMEOUT_DATA (d)->object
+#define EVENT_TIMEOUT_INTERVAL_ID(e) EVENT_FOO_BAR (e, TIMEOUT, timeout, interval_id)
+#define XEVENT_TIMEOUT_INTERVAL_ID(e) EVENT_TIMEOUT_INTERVAL_ID (XEVENT (e))
+#define SET_EVENT_TIMEOUT_INTERVAL_ID(e, val) \
+  SET_EVENT_FOO_BAR (e, TIMEOUT, timeout, interval_id, val)
+#define XSET_EVENT_TIMEOUT_INTERVAL_ID(e, val) \
+  SET_EVENT_TIMEOUT_INTERVAL_ID (XEVENT (e), val)
+
+#define EVENT_TIMEOUT_ID_NUMBER(e) EVENT_FOO_BAR (e, TIMEOUT, timeout, id_number)
+#define XEVENT_TIMEOUT_ID_NUMBER(e) EVENT_TIMEOUT_ID_NUMBER (XEVENT (e))
+#define SET_EVENT_TIMEOUT_ID_NUMBER(e, val) \
+  SET_EVENT_FOO_BAR (e, TIMEOUT, timeout, id_number, val)
+#define XSET_EVENT_TIMEOUT_ID_NUMBER(e, val) \
+  SET_EVENT_TIMEOUT_ID_NUMBER (XEVENT (e), val)
 
-#define XSET_TIMEOUT_DATA_INTERVAL_ID(d, i) XTIMEOUT_DATA (d)->interval_id = (i)
-#define XSET_TIMEOUT_DATA_ID_NUMBER(d, n) XTIMEOUT_DATA (d)->id_number = (n)
-#define XSET_TIMEOUT_DATA_FUNCTION(d, f) XTIMEOUT_DATA (d)->function = f
-#define XSET_TIMEOUT_DATA_OBJECT(d, o) XTIMEOUT_DATA (d)->object = o
-#endif /* USE_KKCC */
+#define EVENT_TIMEOUT_FUNCTION(e) EVENT_FOO_BAR (e, TIMEOUT, timeout, function)
+#define XEVENT_TIMEOUT_FUNCTION(e) EVENT_TIMEOUT_FUNCTION (XEVENT (e))
+#define SET_EVENT_TIMEOUT_FUNCTION(e, val) \
+  SET_EVENT_FOO_BAR (e, TIMEOUT, timeout, function, val)
+#define XSET_EVENT_TIMEOUT_FUNCTION(e, val) \
+  SET_EVENT_TIMEOUT_FUNCTION (XEVENT (e), val)
 
-#ifdef USE_KKCC
+#define EVENT_TIMEOUT_OBJECT(e) EVENT_FOO_BAR (e, TIMEOUT, timeout, object)
+#define XEVENT_TIMEOUT_OBJECT(e) EVENT_TIMEOUT_OBJECT (XEVENT (e))
+#define SET_EVENT_TIMEOUT_OBJECT(e, val) \
+  SET_EVENT_FOO_BAR (e, TIMEOUT, timeout, object, val)
+#define XSET_EVENT_TIMEOUT_OBJECT(e, val) \
+  SET_EVENT_TIMEOUT_OBJECT (XEVENT (e), val)
+
 struct Lisp_Eval_Data
-#else /* not USE_KKCC */
-struct eval_data
-#endif /* not USE_KKCC */
 {
 /* This kind of event is used internally; sometimes the window system
    interface would like to inform XEmacs of some user action (such as
@@ -497,34 +540,38 @@
     function		An elisp function to call with this event object.
     object		Argument of function.
 */
-#ifdef USE_KKCC
+#ifdef EVENT_DATA_AS_OBJECTS
   struct lrecord_header lheader;
-#endif /* USE_KKCC */
+#endif /* EVENT_DATA_AS_OBJECTS */
   Lisp_Object function;
   Lisp_Object object;
 };
-#ifdef USE_KKCC
 typedef struct Lisp_Eval_Data Lisp_Eval_Data;
 
+#ifdef EVENT_DATA_AS_OBJECTS
 DECLARE_LRECORD (eval_data, Lisp_Eval_Data);
 #define XEVAL_DATA(x) XRECORD (x, eval_data, Lisp_Eval_Data)
 #define wrap_eval_data(p) wrap_record(p, eval_data)
 #define EVAL_DATAP(x) RECORDP (x, eval_data)
 #define CHECK_EVAL_DATA(x) CHECK_RECORD (x, eval_data)
 #define CONCHECK_EVAL_DATA(x) CONCHECK_RECORD (x, eval_data)
+#endif /* EVENT_DATA_AS_OBJECTS */
 
-#define XEVAL_DATA_FUNCTION(d) (XEVAL_DATA (d)->function)
-#define XEVAL_DATA_OBJECT(d) (XEVAL_DATA (d)->object)
+#define EVENT_EVAL_FUNCTION(e) EVENT_FOO_BAR (e, EVAL, eval, function)
+#define XEVENT_EVAL_FUNCTION(e) EVENT_EVAL_FUNCTION (XEVENT (e))
+#define SET_EVENT_EVAL_FUNCTION(e, val) \
+  SET_EVENT_FOO_BAR (e, EVAL, eval, function, val)
+#define XSET_EVENT_EVAL_FUNCTION(e, val) \
+  SET_EVENT_EVAL_FUNCTION (XEVENT (e), val)
 
-#define XSET_EVAL_DATA_FUNCTION(d, f) (XEVAL_DATA (d)->function = f)
-#define XSET_EVAL_DATA_OBJECT(d, o) (XEVAL_DATA (d)->object = o)
-#endif /* USE_KKCC */
+#define EVENT_EVAL_OBJECT(e) EVENT_FOO_BAR (e, EVAL, eval, object)
+#define XEVENT_EVAL_OBJECT(e) EVENT_EVAL_OBJECT (XEVENT (e))
+#define SET_EVENT_EVAL_OBJECT(e, val) \
+  SET_EVENT_FOO_BAR (e, EVAL, eval, object, val)
+#define XSET_EVENT_EVAL_OBJECT(e, val) \
+  SET_EVENT_EVAL_OBJECT (XEVENT (e), val)
 
-#ifdef USE_KKCC
 struct Lisp_Misc_User_Data
-#else /* not USE_KKCC */
-struct misc_user_data
-#endif /* not USE_KKCC */
 {
 /* #### The misc-user type is serious junk.  It should be separated
    out into different events.  There's no reason to create
@@ -546,48 +593,69 @@
 			by the XEmacs Drag'n'Drop system. Don't depend on their
 			values for other types of misc_user_events.
 */
-#ifdef USE_KKCC
+#ifdef EVENT_DATA_AS_OBJECTS
   struct lrecord_header lheader;
-#endif /* USE_KKCC */
+#endif /* EVENT_DATA_AS_OBJECTS */
   Lisp_Object function;
   Lisp_Object object;
   int button;
   int modifiers;
   int x, y;
 };
-#ifdef USE_KKCC
 typedef struct Lisp_Misc_User_Data Lisp_Misc_User_Data;
 
+#ifdef EVENT_DATA_AS_OBJECTS
 DECLARE_LRECORD (misc_user_data, Lisp_Misc_User_Data);
 #define XMISC_USER_DATA(x) XRECORD (x, misc_user_data, Lisp_Misc_User_Data)
 #define wrap_misc_user_data(p) wrap_record(p, misc_user_data)
 #define MISC_USER_DATAP(x) RECORDP (x, misc_user_data)
 #define CHECK_MISC_USER_DATA(x) CHECK_RECORD (x, misc_user_data)
 #define CONCHECK_MISC_USER_DATA(x) CONCHECK_RECORD (x, misc_user_data)
+#endif /* EVENT_DATA_AS_OBJECTS */
 
-#define XMISC_USER_DATA_FUNCTION(d) (XMISC_USER_DATA (d)->function)
-#define XMISC_USER_DATA_OBJECT(d) (XMISC_USER_DATA (d)->object)
-#define XMISC_USER_DATA_BUTTON(d) (XMISC_USER_DATA (d)->button)
-#define XMISC_USER_DATA_MODIFIERS(d) (XMISC_USER_DATA (d)->modifiers)
-#define XMISC_USER_DATA_X(d) (XMISC_USER_DATA (d)->x)
-#define XMISC_USER_DATA_Y(d) (XMISC_USER_DATA (d)->y)
+#define EVENT_MISC_USER_FUNCTION(e) EVENT_FOO_BAR (e, MISC_USER, misc_user, function)
+#define XEVENT_MISC_USER_FUNCTION(e) EVENT_MISC_USER_FUNCTION (XEVENT (e))
+#define SET_EVENT_MISC_USER_FUNCTION(e, val) \
+  SET_EVENT_FOO_BAR (e, MISC_USER, misc_user, function, val)
+#define XSET_EVENT_MISC_USER_FUNCTION(e, val) \
+  SET_EVENT_MISC_USER_FUNCTION (XEVENT (e), val)
+
+#define EVENT_MISC_USER_OBJECT(e) EVENT_FOO_BAR (e, MISC_USER, misc_user, object)
+#define XEVENT_MISC_USER_OBJECT(e) EVENT_MISC_USER_OBJECT (XEVENT (e))
+#define SET_EVENT_MISC_USER_OBJECT(e, val) \
+  SET_EVENT_FOO_BAR (e, MISC_USER, misc_user, object, val)
+#define XSET_EVENT_MISC_USER_OBJECT(e, val) \
+  SET_EVENT_MISC_USER_OBJECT (XEVENT (e), val)
 
-#define XSET_MISC_USER_DATA_FUNCTION(d, f) (XMISC_USER_DATA (d)->function = (f))
-#define XSET_MISC_USER_DATA_OBJECT(d, o) (XMISC_USER_DATA (d)->object = (o))
-#define XSET_MISC_USER_DATA_BUTTON(d, b) (XMISC_USER_DATA (d)->button = (b))
-#define XSET_MISC_USER_DATA_MODIFIERS(d, m) (XMISC_USER_DATA (d)->modifiers = (m))
-#define XSET_MISC_USER_DATA_X(d, new_x) (XMISC_USER_DATA (d)->x = (new_x))
-#define XSET_MISC_USER_DATA_Y(d, new_y) (XMISC_USER_DATA (d)->y = (new_y))
-#endif /* USE_KKCC */
+#define EVENT_MISC_USER_BUTTON(e) EVENT_FOO_BAR (e, MISC_USER, misc_user, button)
+#define XEVENT_MISC_USER_BUTTON(e) EVENT_MISC_USER_BUTTON (XEVENT (e))
+#define SET_EVENT_MISC_USER_BUTTON(e, val) \
+  SET_EVENT_FOO_BAR (e, MISC_USER, misc_user, button, val)
+#define XSET_EVENT_MISC_USER_BUTTON(e, val) \
+  SET_EVENT_MISC_USER_BUTTON (XEVENT (e), val)
+
+#define EVENT_MISC_USER_MODIFIERS(e) EVENT_FOO_BAR (e, MISC_USER, misc_user, modifiers)
+#define XEVENT_MISC_USER_MODIFIERS(e) EVENT_MISC_USER_MODIFIERS (XEVENT (e))
+#define SET_EVENT_MISC_USER_MODIFIERS(e, val) \
+  SET_EVENT_FOO_BAR (e, MISC_USER, misc_user, modifiers, val)
+#define XSET_EVENT_MISC_USER_MODIFIERS(e, val) \
+  SET_EVENT_MISC_USER_MODIFIERS (XEVENT (e), val)
 
-#ifdef USE_KKCC
-typedef void (*lisp_obj_arg_fun) (Lisp_Object);
+#define EVENT_MISC_USER_X(e) EVENT_FOO_BAR (e, MISC_USER, misc_user, x)
+#define XEVENT_MISC_USER_X(e) EVENT_MISC_USER_X (XEVENT (e))
+#define SET_EVENT_MISC_USER_X(e, val) \
+  SET_EVENT_FOO_BAR (e, MISC_USER, misc_user, x, val)
+#define XSET_EVENT_MISC_USER_X(e, val) \
+  SET_EVENT_MISC_USER_X (XEVENT (e), val)
 
+#define EVENT_MISC_USER_Y(e) EVENT_FOO_BAR (e, MISC_USER, misc_user, y)
+#define XEVENT_MISC_USER_Y(e) EVENT_MISC_USER_Y (XEVENT (e))
+#define SET_EVENT_MISC_USER_Y(e, val) \
+  SET_EVENT_FOO_BAR (e, MISC_USER, misc_user, y, val)
+#define XSET_EVENT_MISC_USER_Y(e, val) \
+  SET_EVENT_MISC_USER_Y (XEVENT (e), val)
 
 struct Lisp_Magic_Eval_Data
-#else /* not USE_KKCC */
-struct magic_eval_data
-#endif /* not USE_KKCC */
 {
 /* This is like an eval event but its contents are not
    Lisp-accessible.  This allows for "internal eval events" that call
@@ -602,32 +670,36 @@
     object		Argument of function.
 
 */
-#ifdef USE_KKCC
+#ifdef EVENT_DATA_AS_OBJECTS
   struct lrecord_header lheader;
-#endif /* USE_KKCC */
+#endif /* EVENT_DATA_AS_OBJECTS */
   void (*internal_function) (Lisp_Object);
   Lisp_Object object;
 };
-#ifdef USE_KKCC
 typedef struct Lisp_Magic_Eval_Data Lisp_Magic_Eval_Data;
 
+#ifdef EVENT_DATA_AS_OBJECTS
 DECLARE_LRECORD (magic_eval_data, Lisp_Magic_Eval_Data);
 #define XMAGIC_EVAL_DATA(x) XRECORD (x, magic_eval_data, Lisp_Magic_Eval_Data)
 #define wrap_magic_eval_data(p) wrap_record(p, magic_eval_data)
 #define MAGIC_EVAL_DATAP(x) RECORDP (x, magic_eval_data)
 #define CHECK_MAGIC_EVAL_DATA(x) CHECK_RECORD (x, magic_eval_data)
 #define CONCHECK_MAGIC_EVAL_DATA(x) CONCHECK_RECORD (x, magic_eval_data)
-
-#define XMAGIC_EVAL_DATA_INTERNAL_FUNCTION(d) \
-  XMAGIC_EVAL_DATA (d)->internal_function
-#define XMAGIC_EVAL_DATA_INTERNAL_FUNOBJ(d) (XMAGIC_EVAL_DATA (d)->internal_function)
-#define XMAGIC_EVAL_DATA_OBJECT(d) (XMAGIC_EVAL_DATA (d)->object)
+#endif /* EVENT_DATA_AS_OBJECTS */
 
-#define XSET_MAGIC_EVAL_DATA_INTERNAL_FUNCTION(d, f) \
-  (XMAGIC_EVAL_DATA (d)->internal_function = f)
-#define XSET_MAGIC_EVAL_DATA_INTERNAL_FUNOBJ(d, f) (XMAGIC_EVAL_DATA (d)->internal_function = (f))
-#define XSET_MAGIC_EVAL_DATA_OBJECT(d, o) (XMAGIC_EVAL_DATA (d)->object = (o))
-#endif /* USE_KKCC */
+#define EVENT_MAGIC_EVAL_INTERNAL_FUNCTION(e) EVENT_FOO_BAR (e, MAGIC_EVAL, magic_eval, internal_function)
+#define XEVENT_MAGIC_EVAL_INTERNAL_FUNCTION(e) EVENT_MAGIC_EVAL_INTERNAL_FUNCTION (XEVENT (e))
+#define SET_EVENT_MAGIC_EVAL_INTERNAL_FUNCTION(e, val) \
+  SET_EVENT_FOO_BAR (e, MAGIC_EVAL, magic_eval, internal_function, val)
+#define XSET_EVENT_MAGIC_EVAL_INTERNAL_FUNCTION(e, val) \
+  SET_EVENT_MAGIC_EVAL_INTERNAL_FUNCTION (XEVENT (e), val)
+
+#define EVENT_MAGIC_EVAL_OBJECT(e) EVENT_FOO_BAR (e, MAGIC_EVAL, magic_eval, object)
+#define XEVENT_MAGIC_EVAL_OBJECT(e) EVENT_MAGIC_EVAL_OBJECT (XEVENT (e))
+#define SET_EVENT_MAGIC_EVAL_OBJECT(e, val) \
+  SET_EVENT_FOO_BAR (e, MAGIC_EVAL, magic_eval, object, val)
+#define XSET_EVENT_MAGIC_EVAL_OBJECT(e, val) \
+  SET_EVENT_MAGIC_EVAL_OBJECT (XEVENT (e), val)
 
 #if defined (HAVE_X_WINDOWS) && defined(emacs)
 # include <X11/Xlib.h>
@@ -638,10 +710,25 @@
 #endif
 
 
-#ifdef USE_KKCC
 struct Lisp_Magic_Data
 {
+/* No user-serviceable parts within.  This is for things like
+   KeymapNotify and ExposeRegion events and so on that XEmacs itself
+   doesn't care about, but which it must do something with for proper
+   interaction with the window system.
+
+   Magic_events are handled somewhat asynchronously, just like
+   subprocess filters.  However, occasionally a magic_event needs to
+   be handled synchronously; in that case, the asynchronous handling
+   of the magic_event will push an eval_event back onto the queue,
+   which will be handled synchronously later.  This is one of the
+   reasons why eval_events exist; I'm not entirely happy with this
+   aspect of this event model.
+*/
+
+#ifdef EVENT_DATA_AS_OBJECTS
   struct lrecord_header lheader;
+#endif /* EVENT_DATA_AS_OBJECTS */
 
   union {
 #ifdef HAVE_GTK
@@ -658,61 +745,49 @@
 
 typedef struct Lisp_Magic_Data Lisp_Magic_Data;
 
+#ifdef EVENT_DATA_AS_OBJECTS
 DECLARE_LRECORD (magic_data, Lisp_Magic_Data);
 #define XMAGIC_DATA(x) XRECORD (x, magic_data, Lisp_Magic_Data)
 #define wrap_magic_data(p) wrap_record(p, magic_data)
 #define MAGIC_DATAP(x) RECORDP (x, magic_data)
 #define CHECK_MAGIC_DATA(x) CHECK_RECORD (x, magic_data)
 #define CONCHECK_MAGIC_DATA(x) CONCHECK_RECORD (x, magic_data)
+#endif /* EVENT_DATA_AS_OBJECTS */
 
-#define XMAGIC_DATA_UNDERLYING(d) (XMAGIC_DATA (d)->underlying)
-#define XSET_MAGIC_DATA_UNDERLYING(d, u) (XMAGIC_DATA (d)->underlying = (u))
+#define EVENT_MAGIC_UNDERLYING(e) EVENT_FOO_BAR (e, MAGIC, magic, underlying)
+#define XEVENT_MAGIC_UNDERLYING(e) EVENT_MAGIC_UNDERLYING (XEVENT (e))
+#define SET_EVENT_MAGIC_UNDERLYING(e, val) \
+  SET_EVENT_FOO_BAR (e, MAGIC, magic, underlying, val)
+#define XSET_EVENT_MAGIC_UNDERLYING(e, val) \
+  SET_EVENT_MAGIC_UNDERLYING (XEVENT (e), val)
 
 #ifdef HAVE_GTK
-#define XMAGIC_DATA_GDK_EVENT(d) (XMAGIC_DATA (d)->underlying.gdk_event)
-#define XSET_MAGIC_DATA_GDK_EVENT(d, e) (XMAGIC_DATA (d)->underlying.gdk_event = (e))
-#endif /*HAVE_GTK*/
+#define EVENT_MAGIC_GDK_EVENT(e) EVENT_FOO_BAR (e, MAGIC, magic, underlying.gdk_event)
+#define XEVENT_MAGIC_GDK_EVENT(e) EVENT_MAGIC_GDK_EVENT (XEVENT (e))
+#define SET_EVENT_MAGIC_GDK_EVENT(e, val) \
+  SET_EVENT_FOO_BAR (e, MAGIC, magic, underlying.gdk_event, val)
+#define XSET_EVENT_MAGIC_GDK_EVENT(e, val) \
+  SET_EVENT_MAGIC_GDK_EVENT (XEVENT (e), val)
+#endif
 
 #ifdef HAVE_X_WINDOWS
-#define XMAGIC_DATA_X_EVENT(d) (XMAGIC_DATA (d)->underlying.x_event)
-#define XSET_MAGIC_DATA_X_EVENT(d, e) (XMAGIC_DATA (d)->underlying.x_event = (e))
+#define EVENT_MAGIC_X_EVENT(e) EVENT_FOO_BAR (e, MAGIC, magic, underlying.x_event)
+#define XEVENT_MAGIC_X_EVENT(e) EVENT_MAGIC_X_EVENT (XEVENT (e))
+#define SET_EVENT_MAGIC_X_EVENT(e, val) \
+  SET_EVENT_FOO_BAR (e, MAGIC, magic, underlying.x_event, val)
+#define XSET_EVENT_MAGIC_X_EVENT(e, val) \
+  SET_EVENT_MAGIC_X_EVENT (XEVENT (e), val)
 #endif
 
 #ifdef HAVE_MS_WINDOWS
-#define XMAGIC_DATA_MSWINDOWS_EVENT(d) (XMAGIC_DATA (d)->underlying.mswindows_event)
-#define XSET_MAGIC_DATA_MSWINDOWS_EVENT(d, e) (XMAGIC_DATA (d)->underlying.mswindows_event = (e))
+#define EVENT_MAGIC_MSWINDOWS_EVENT(e) EVENT_FOO_BAR (e, MAGIC, magic, underlying.mswindows_event)
+#define XEVENT_MAGIC_MSWINDOWS_EVENT(e) EVENT_MAGIC_MSWINDOWS_EVENT (XEVENT (e))
+#define SET_EVENT_MAGIC_MSWINDOWS_EVENT(e, val) \
+  SET_EVENT_FOO_BAR (e, MAGIC, magic, underlying.mswindows_event, val)
+#define XSET_EVENT_MAGIC_MSWINDOWS_EVENT(e, val) \
+  SET_EVENT_MAGIC_MSWINDOWS_EVENT (XEVENT (e), val)
 #endif
 
-#else /* not USE_KKCC */
-union magic_data
-{
-/* No user-serviceable parts within.  This is for things like
-   KeymapNotify and ExposeRegion events and so on that XEmacs itself
-   doesn't care about, but which it must do something with for proper
-   interaction with the window system.
-
-   Magic_events are handled somewhat asynchronously, just like
-   subprocess filters.  However, occasionally a magic_event needs to
-   be handled synchronously; in that case, the asynchronous handling
-   of the magic_event will push an eval_event back onto the queue,
-   which will be handled synchronously later.  This is one of the
-   reasons why eval_events exist; I'm not entirely happy with this
-   aspect of this event model.
-*/
-
-#ifdef HAVE_GTK
-  GdkEvent          underlying_gdk_event;
-#endif
-#ifdef HAVE_X_WINDOWS
-  XEvent            underlying_x_event;
-#endif
-#ifdef HAVE_MS_WINDOWS
-  int               underlying_mswindows_event;
-#endif
-};
-#endif /* not USE_KKCC */
-
-
 struct Lisp_Timeout
 {
   struct lcrecord_header header;
@@ -784,22 +859,22 @@
      server time and real process time; yuck.) */
   unsigned int          timestamp;
 
-#ifdef USE_KKCC
+#ifdef EVENT_DATA_AS_OBJECTS
   Lisp_Object event_data;
-#else /* not USE_KKCC */
+#else /* not EVENT_DATA_AS_OBJECTS */
   union
     {
-      struct key_data           key;
-      struct button_data        button;
-      struct motion_data        motion;
-      struct process_data       process;
-      struct timeout_data       timeout;
-      struct eval_data          eval;   /* misc_user_event no longer uses this */
-      struct misc_user_data     misc;   /* because it needs position information */
-      union magic_data          magic;
-      struct magic_eval_data    magic_eval;
+      Lisp_Key_Data           key;
+      Lisp_Button_Data        button;
+      Lisp_Motion_Data        motion;
+      Lisp_Process_Data       process;
+      Lisp_Timeout_Data       timeout;
+      Lisp_Eval_Data          eval;   /* misc_user_event no longer uses this */
+      Lisp_Misc_User_Data     misc_user;/* because it needs position information */
+      Lisp_Magic_Data         magic;
+      Lisp_Magic_Eval_Data    magic_eval;
     } event;
-#endif /* not USE_KKCC */
+#endif /* not EVENT_DATA_AS_OBJECTS */
 };
 
 DECLARE_LRECORD (event, Lisp_Event);
@@ -812,26 +887,29 @@
 DECLARE_LRECORD (command_builder, struct command_builder);
 
 #define EVENT_CHANNEL(a) ((a)->channel)
+#define XEVENT_CHANNEL(ev) (XEVENT (ev)->channel)
 #define EVENT_TYPE(a) ((a)->event_type)
 #define XEVENT_TYPE(a) (XEVENT (a)->event_type)
 #define EVENT_NEXT(a) ((a)->next)
 #define XEVENT_NEXT(e) (XEVENT (e)->next)
+#define EVENT_TIMESTAMP(ev) ((ev)->timestamp)
+#define XEVENT_TIMESTAMP(ev) EVENT_TIMESTAMP (XEVENT (ev))
 
-#ifdef USE_KKCC
+#ifdef EVENT_DATA_AS_OBJECTS
 #define XEVENT_DATA(ev) (XEVENT (ev)->event_data)
 #define EVENT_DATA(ev) ((ev)->event_data)
-#define XEVENT_CHANNEL(ev) (XEVENT (ev)->channel)
-#define EVENT_TIMESTAMP(ev)                                     \
-  ((ev)->timestamp)
-#define XEVENT_TIMESTAMP(ev) EVENT_TIMESTAMP (XEVENT (ev))
+#define SET_EVENT_DATA(ev, d)                           \
+do {                                                    \
+  Lisp_Event *mac_event = (ev);                         \
+  mac_event->event_data = (d);                          \
+} while (0)
+#define XSET_EVENT_DATA(ev, d) SET_EVENT_DATA (XEVENT (ev), d)
+#endif /* EVENT_DATA_AS_OBJECTS */
 
-#define SET_EVENT_TIMESTAMP_ZERO(ev) \
-  ((ev)->timestamp = 0)
-#define SET_EVENT_TIMESTAMP(ev, t)                      \
-  (ev)->timestamp = (t)
+#define SET_EVENT_TIMESTAMP_ZERO(ev) ((ev)->timestamp = 0)
+#define SET_EVENT_TIMESTAMP(ev, t) ((ev)->timestamp = (t))
 #define XSET_EVENT_TIMESTAMP(ev, t) SET_EVENT_TIMESTAMP (XEVENT (ev), t)
 
-
 #define SET_EVENT_CHANNEL(ev, c)                        \
 do {                                                    \
   Lisp_Event *mac_event = (ev);                         \
@@ -839,67 +917,93 @@
 } while (0)
 #define XSET_EVENT_CHANNEL(ev, c) SET_EVENT_CHANNEL (XEVENT (ev), c) 
 
-#define SET_EVENT_DATA(ev, d)                           \
-do {                                                    \
-  Lisp_Event *mac_event = (ev);                         \
-  mac_event->event_data = (d);                          \
-} while (0)
-#define XSET_EVENT_DATA(ev, d) SET_EVENT_DATA (XEVENT (ev), d)
+DECLARE_INLINE_HEADER (
+void
+set_event_type (struct Lisp_Event *event, emacs_event_type t) 
+)
+{
+#ifdef EVENT_DATA_AS_OBJECTS
+  switch (EVENT_TYPE (event))
+    {
+    case key_press_event:
+      free_key_data (event->event_data);
+      break;
+    case button_press_event:
+    case button_release_event:
+      free_button_data (event->event_data);
+      break;
+    case pointer_motion_event:
+      free_motion_data (event->event_data);
+      break;
+    case process_event:
+      free_process_data (event->event_data);
+      break;
+    case timeout_event:
+      free_timeout_data (event->event_data);
+      break;
+    case magic_event:
+      free_magic_data (event->event_data);
+      break;
+    case magic_eval_event:
+      free_magic_eval_data (event->event_data);
+      break;
+    case eval_event:
+      free_eval_data (event->event_data);
+      break;
+    case misc_user_event:
+      free_misc_user_data (event->event_data);
+      break;
+    default:
+      break;
+    }
+#endif /* EVENT_DATA_AS_OBJECTS */
 
-INLINE_HEADER  void set_event_type(struct Lisp_Event *event, emacs_event_type t);
-INLINE_HEADER  void
-set_event_type(struct Lisp_Event *event, emacs_event_type t) 
-{
   event->event_type = t;
 
-  switch (t) {
-  case key_press_event:
-    event->event_data = make_key_data ();
-    break;
-  case button_press_event:
-  case button_release_event:
-    event->event_data = make_button_data ();
-    break;
-  case pointer_motion_event:
-    event->event_data = make_motion_data ();
-    break;
-  case process_event:
-    event->event_data = make_process_data ();
-    break;
-  case timeout_event:
-    event->event_data = make_timeout_data ();
-    break;
-  case magic_event:
-    event->event_data = make_magic_data ();
-    break;
-  case magic_eval_event:
-    event->event_data = make_magic_eval_data ();
-    break;
-  case eval_event:
-    event->event_data = make_eval_data ();
-    break;
-  case misc_user_event:
-    event->event_data = make_misc_user_data ();
-    break;
-  default:
-    break;
-  }
+#ifdef EVENT_DATA_AS_OBJECTS
+  switch (t)
+    {
+    case key_press_event:
+      event->event_data = make_key_data ();
+      break;
+    case button_press_event:
+    case button_release_event:
+      event->event_data = make_button_data ();
+      break;
+    case pointer_motion_event:
+      event->event_data = make_motion_data ();
+      break;
+    case process_event:
+      event->event_data = make_process_data ();
+      break;
+    case timeout_event:
+      event->event_data = make_timeout_data ();
+      break;
+    case magic_event:
+      event->event_data = make_magic_data ();
+      break;
+    case magic_eval_event:
+      event->event_data = make_magic_eval_data ();
+      break;
+    case eval_event:
+      event->event_data = make_eval_data ();
+      break;
+    case misc_user_event:
+      event->event_data = make_misc_user_data ();
+      break;
+    default:
+      event->event_data = Qnil;
+      break;
+    }
+#endif /* EVENT_DATA_AS_OBJECTS */
 }
+
+#define SET_EVENT_NEXT(ev, n) do { (ev)->next = (n); } while (0)
+#define XSET_EVENT_NEXT(ev, n) SET_EVENT_NEXT (XEVENT (ev), n)
+
 #define XSET_EVENT_TYPE(ev, t) set_event_type (XEVENT (ev), t)
 #define SET_EVENT_TYPE(ev, t) set_event_type (ev, t)
 
-
-#define SET_EVENT_NEXT(ev, n)                           \
-do {                                                    \
-  Lisp_Event *mac_event = (ev);                         \
-  mac_event->next = (n);                                \
-} while (0)
-#define XSET_EVENT_NEXT(ev, n) SET_EVENT_NEXT (XEVENT (ev), n)
-
-#else /* NOT USE_KKCC */
-#define XSET_EVENT_NEXT(e, n) do { (XEVENT (e)->next = (n)); } while (0)
-#endif
-
 #define EVENT_CHAIN_LOOP(event, chain) \
   for (event = chain; !NILP (event); event = XEVENT_NEXT (event))
 
@@ -956,45 +1060,52 @@
 #define KEYSYM(x) (intern (x))
 
 /* from events.c */
-#ifdef USE_KKCC
 void format_event_object (Eistring *buf, Lisp_Object event, int brief);
-#else /* not USE_KKCC */
-void format_event_object (Eistring *buf, Lisp_Event *event, int brief);
-#endif /* not USE_KKCC */
-#ifdef USE_KKCC
 //void format_event_data_object (Eistring *buf, Lisp_Object data, int brief);
-void copy_event_data (Lisp_Object dest, Lisp_Object src);
-#endif /* USE_KKCC */
 void character_to_event (Ichar c, Lisp_Event *event,
                          struct console *con,
                          int use_console_meta_flag,
                          int do_backspace_mapping);
 void zero_event (Lisp_Event *e);
+
+#define MECR_DEALLOCATE_EVENT 1
+
 void deallocate_event_chain (Lisp_Object event);
 Lisp_Object event_chain_tail (Lisp_Object event);
 void enqueue_event (Lisp_Object event, Lisp_Object *head, Lisp_Object *tail);
 Lisp_Object dequeue_event (Lisp_Object *head, Lisp_Object *tail);
 void enqueue_event_chain (Lisp_Object event_chain, Lisp_Object *head,
                           Lisp_Object *tail);
+int map_event_chain_remove (int (*fn) (Lisp_Object ev, void *user_data),
+			    Lisp_Object *head, Lisp_Object *tail,
+			    void *user_data, int flags);
 int event_chain_count (Lisp_Object event_chain);
-Lisp_Object transfer_event_chain_pointer (Lisp_Object pointer,
-					  Lisp_Object old_chain,
-					  Lisp_Object new_chain);
-void nth_of_key_sequence_as_event (Lisp_Object seq, int n, Lisp_Object event);
-Lisp_Object key_sequence_to_event_chain (Lisp_Object seq);
 Lisp_Object event_chain_find_previous (Lisp_Object event_chain,
                                        Lisp_Object event);
 Lisp_Object event_chain_nth (Lisp_Object event_chain, int n);
 Lisp_Object copy_event_chain (Lisp_Object event_chain);
+Lisp_Object transfer_event_chain_pointer (Lisp_Object pointer,
+					  Lisp_Object old_chain,
+					  Lisp_Object new_chain);
+
+
+void nth_of_key_sequence_as_event (Lisp_Object seq, int n, Lisp_Object event);
+Lisp_Object key_sequence_to_event_chain (Lisp_Object seq);
 /* True if this is a non-internal event
    (keyboard press, menu, scrollbar, mouse button) */
 int command_event_p (Lisp_Object event);
 void define_self_inserting_symbol (Lisp_Object, Lisp_Object);
-Ichar event_to_character (Lisp_Event *, int, int, int);
+Ichar event_to_character (Lisp_Object, int, int, int);
 struct console *event_console_or_selected (Lisp_Object event);
+void upshift_event (Lisp_Object event);
+void downshift_event (Lisp_Object event);
+int event_upshifted_p (Lisp_Object event);
 
 /* from event-stream.c */
+extern Lisp_Object dispatch_event_queue;
 Lisp_Object allocate_command_builder (Lisp_Object console, int with_echo_buf);
+void enqueue_dispatch_event (Lisp_Object event);
+Lisp_Object dequeue_dispatch_event (void);
 void enqueue_magic_eval_event (void (*fun) (Lisp_Object), Lisp_Object object);
 void event_stream_handle_magic_event (Lisp_Event *event);
 void event_stream_format_magic_event (Lisp_Event *event, Lisp_Object pstream);
@@ -1073,6 +1184,7 @@
 
 void signal_fake_event (void);
 void drain_signal_event_pipe (void);
+void drain_tty_devices (void);
 
 extern int fake_event_occurred;
 
@@ -1082,6 +1194,7 @@
 					  int doerr, int *infd, int *errfd);
 void event_stream_unixoid_unselect_process (Lisp_Process *proc, int doin,
 					    int doerr, int *infd, int *errfd);
+struct console *find_tty_or_stream_console_from_fd (int fd);
 int read_event_from_tty_or_stream_desc (Lisp_Event *event,
 					struct console *con);
 void event_stream_unixoid_create_io_streams (void* inhandle, void* outhandle,
@@ -1098,11 +1211,15 @@
 					     USID* in_usid,
 					     USID* err_usid);
 
+#endif /* HAVE_UNIXOID_EVENT_LOOP */
+
+/* The following is not inside of HAVE_UNIXOID_EVENT_LOOP because of the
+   possibility of combiling XEmacs with no-MSW, no-X, no-TTY --
+   process-unix.c is still compiled.  #### Should we still compile
+   subprocesses with no event loops? */
 /* Beware: this evil macro evaluates its arg many times */
 #define FD_TO_USID(fd) ((fd)==0 ? (USID)999999 : ((fd)<0 ? USID_DONTHASH : (USID)(fd)))
 
-#endif /* HAVE_UNIXOID_EVENT_LOOP */
-
 /* Define this if you want the tty event stream to be used when the
    first console is tty, even if HAVE_X_WINDOWS is defined */
 /* #define DEBUG_TTY_EVENT_STREAM */
@@ -1139,20 +1256,17 @@
      something that's not part of a possible function key, we have to
      unread all of those events. */
   Lisp_Object last_non_munged_event;
-  /* One set of values for function-key-map, one for key-translation-map */
-  struct munging_key_translation
-  {
-    /* First event that can begin a possible function key sequence
-       (to be translated according to function-key-map).  Normally
-       this is the first event in the chain.  However, once we've
-       translated a sequence through function-key-map, this will point
-       to the first event after the translated sequence: we don't ever
-       want to translate any events twice through function-key-map, or
-       things could get really screwed up (e.g. if the user created a
-       translation loop).  If this is nil, then the next-read event is
-       the first that can begin a function key sequence. */
-    Lisp_Object first_mungeable_event;
-  } munge_me[2];
+  /* One value for function-key-map, one for key-translation-map:
+     First event that can begin a possible function key sequence
+     (to be translated according to function-key-map).  Normally
+     this is the first event in the chain.  However, once we've
+     translated a sequence through function-key-map, this will point
+     to the first event after the translated sequence: we don't ever
+     want to translate any events twice through function-key-map, or
+     things could get really screwed up (e.g. if the user created a
+     translation loop).  If this is nil, then the next-read event is
+     the first that can begin a function key sequence. */
+  Lisp_Object first_mungeable_event[2];
   Ibyte *echo_buf;
 
   Bytecount echo_buf_length;          /* size of echo_buf */
--- a/src/extents.c	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/extents.c	Sun Jan 12 11:08:22 2003 +0000
@@ -245,20 +245,47 @@
   struct gap_array_marker *next;
 } Gap_Array_Marker;
 
+
 /* Holds a "gap array", which is an array of elements with a gap located
    in it.  Insertions and deletions with a high degree of locality
    are very fast, essentially in constant time.  Array positions as
    used and returned in the gap array functions are independent of
    the gap. */
 
+/* Layout of gap array:
+
+   <------ gap ------><---- gapsize ----><----- numels - gap ---->
+   <---------------------- numels + gapsize --------------------->
+
+   For marking purposes, we use two extra variables computed from
+   the others -- the offset to the data past the gap, plus the number
+   of elements in that data:
+
+   offset_past_gap = elsize * (gap + gapsize)
+   els_past_gap = numels - gap
+
+   #### The current layout will not cut it for dumping purposes.  If we
+   need to dump one of these structures, we will have to rearrange things
+   so that the gap array data and other fields are in one single memory
+   block.  Either this is a pointer off of struct gap_array, or it is
+   struct gap_array itself; in the latter case, the interface below has to
+   be modified so that functions that change the array pass in a ** rather
+   than a *, since the pointer may move.
+*/
+
+
 typedef struct gap_array
 {
-  char *array;
-  int gap;
-  int gapsize;
-  int numels;
-  int elsize;
+  Elemcount gap;
+  Elemcount gapsize;
+  Elemcount numels;
+  Bytecount elsize;
+  /* Redundant numbers computed from the others, for marking purposes */
+  Bytecount offset_past_gap;
+  Elemcount els_past_gap;
   Gap_Array_Marker *markers;
+  /* this is a stretchy array */
+  char array[1];
 } Gap_Array;
 
 static Gap_Array_Marker *gap_array_marker_freelist;
@@ -487,7 +514,7 @@
 
 static void
 gap_array_adjust_markers (Gap_Array *ga, Memxpos from,
-			  Memxpos to, int amount)
+			  Memxpos to, Elemcount amount)
 {
   Gap_Array_Marker *m;
 
@@ -495,16 +522,22 @@
     m->pos = do_marker_adjustment (m->pos, from, to, amount);
 }
 
+static void
+gap_array_recompute_derived_values (Gap_Array *ga)
+{
+  ga->offset_past_gap = ga->elsize * (ga->gap + ga->gapsize);
+  ga->els_past_gap = ga->numels - ga->gap;
+}
+
 /* Move the gap to array position POS.  Parallel to move_gap() in
    insdel.c but somewhat simplified. */
 
 static void
-gap_array_move_gap (Gap_Array *ga, int pos)
-{
-  int gap = ga->gap;
-  int gapsize = ga->gapsize;
-
-  assert (ga->array);
+gap_array_move_gap (Gap_Array *ga, Elemcount pos)
+{
+  Elemcount gap = ga->gap;
+  Elemcount gapsize = ga->gapsize;
+
   if (pos < gap)
     {
       memmove (GAP_ARRAY_MEMEL_ADDR (ga, pos + gapsize),
@@ -522,27 +555,30 @@
 				(Memxpos) (pos + gapsize), - gapsize);
     }
   ga->gap = pos;
+
+  gap_array_recompute_derived_values (ga);
 }
 
 /* Make the gap INCREMENT characters longer.  Parallel to make_gap() in
-   insdel.c. */
-
-static void
-gap_array_make_gap (Gap_Array *ga, int increment)
-{
-  char *ptr = ga->array;
-  int real_gap_loc;
-  int old_gap_size;
+   insdel.c.  The gap array may be moved, so assign the return value back
+   to the array pointer. */
+
+static Gap_Array *
+gap_array_make_gap (Gap_Array *ga, Elemcount increment)
+{
+  Elemcount real_gap_loc;
+  Elemcount old_gap_size;
 
   /* If we have to get more space, get enough to last a while.  We use
      a geometric progression that saves on realloc space. */
   increment += 100 + ga->numels / 8;
 
-  ptr = (char *) xrealloc (ptr,
-			   (ga->numels + ga->gapsize + increment)*ga->elsize);
-  if (ptr == 0)
+  ga = (Gap_Array *) xrealloc (ga,
+			       offsetof (Gap_Array, array) +
+			       (ga->numels + ga->gapsize + increment) *
+			       ga->elsize);
+  if (ga == 0)
     memory_full ();
-  ga->array = ptr;
 
   real_gap_loc = ga->gap;
   old_gap_size = ga->gapsize;
@@ -558,6 +594,10 @@
   /* Now combine the two into one large gap.  */
   ga->gapsize += old_gap_size;
   ga->gap = real_gap_loc;
+
+  gap_array_recompute_derived_values (ga);
+
+  return ga;
 }
 
 /* ------------------------------- */
@@ -565,14 +605,16 @@
 /* ------------------------------- */
 
 /* Insert NUMELS elements (pointed to by ELPTR) into the specified
-   gap array at POS. */
-
-static void
-gap_array_insert_els (Gap_Array *ga, int pos, void *elptr, int numels)
+   gap array at POS.  The gap array may be moved, so assign the
+   return value back to the array pointer. */
+
+static Gap_Array *
+gap_array_insert_els (Gap_Array *ga, Elemcount pos, void *elptr,
+		      Elemcount numels)
 {
   assert (pos >= 0 && pos <= ga->numels);
   if (ga->gapsize < numels)
-    gap_array_make_gap (ga, numels - ga->gapsize);
+    ga = gap_array_make_gap (ga, numels - ga->gapsize);
   if (pos != ga->gap)
     gap_array_move_gap (ga, pos);
 
@@ -581,21 +623,23 @@
   ga->gapsize -= numels;
   ga->gap += numels;
   ga->numels += numels;
+  gap_array_recompute_derived_values (ga);
   /* This is the equivalent of insert-before-markers.
 
      #### Should only happen if marker is "moves forward at insert" type.
      */
 
   gap_array_adjust_markers (ga, pos - 1, pos, numels);
+  return ga;
 }
 
 /* Delete NUMELS elements from the specified gap array, starting at FROM. */
 
 static void
-gap_array_delete_els (Gap_Array *ga, int from, int numdel)
-{
-  int to = from + numdel;
-  int gapsize = ga->gapsize;
+gap_array_delete_els (Gap_Array *ga, Elemcount from, Elemcount numdel)
+{
+  Elemcount to = from + numdel;
+  Elemcount gapsize = ga->gapsize;
 
   assert (from >= 0);
   assert (numdel >= 0);
@@ -615,10 +659,11 @@
   ga->gapsize += numdel;
   ga->numels -= numdel;
   ga->gap = from;
+  gap_array_recompute_derived_values (ga);
 }
 
 static Gap_Array_Marker *
-gap_array_make_marker (Gap_Array *ga, int pos)
+gap_array_make_marker (Gap_Array *ga, Elemcount pos)
 {
   Gap_Array_Marker *m;
 
@@ -650,7 +695,7 @@
   else
     ga->markers = p->next;
   m->next = gap_array_marker_freelist;
-  m->pos = 0xDEADBEEF; /* -559038737 as an int */
+  m->pos = 0xDEADBEEF; /* -559038737 base 10 */
   gap_array_marker_freelist = m;
 }
 
@@ -669,7 +714,7 @@
 }
 
 static void
-gap_array_move_marker (Gap_Array *ga, Gap_Array_Marker *m, int pos)
+gap_array_move_marker (Gap_Array *ga, Gap_Array_Marker *m, Elemcount pos)
 {
   assert (pos >= 0 && pos <= ga->numels);
   m->pos = GAP_ARRAY_ARRAY_TO_MEMORY_POS (ga, pos);
@@ -679,7 +724,7 @@
   GAP_ARRAY_MEMORY_TO_ARRAY_POS (ga, (m)->pos)
 
 static Gap_Array *
-make_gap_array (int elsize)
+make_gap_array (Elemcount elsize)
 {
   Gap_Array *ga = xnew_and_zero (Gap_Array);
   ga->elsize = elsize;
@@ -689,8 +734,6 @@
 static void
 free_gap_array (Gap_Array *ga)
 {
-  if (ga->array)
-    xfree (ga->array);
   gap_array_delete_all_markers (ga);
   xfree (ga);
 }
@@ -713,7 +756,7 @@
 */
 
 /* Number of elements in an extent list */
-#define extent_list_num_els(el) GAP_ARRAY_NUM_ELS(el->start)
+#define extent_list_num_els(el) GAP_ARRAY_NUM_ELS (el->start)
 
 /* Return the position at which EXTENT is located in the specified extent
    list (in the display order if ENDP is 0, in the e-order otherwise).
@@ -815,10 +858,10 @@
 
   pos = extent_list_locate (el, extent, 0, &foundp);
   assert (!foundp);
-  gap_array_insert_els (el->start, pos, &extent, 1);
+  el->start = gap_array_insert_els (el->start, pos, &extent, 1);
   pos = extent_list_locate (el, extent, 1, &foundp);
   assert (!foundp);
-  gap_array_insert_els (el->end, pos, &extent, 1);
+  el->end = gap_array_insert_els (el->end, pos, &extent, 1);
 }
 
 /* Delete an extent from an extent list. */
@@ -909,8 +952,7 @@
 /*                       Auxiliary extent structure                     */
 /************************************************************************/
 
-#ifdef USE_KKCC
-static const struct lrecord_description extent_auxiliary_description[] ={
+static const struct memory_description extent_auxiliary_description[] ={
   { XD_LISP_OBJECT, offsetof (struct extent_auxiliary, begin_glyph) },
   { XD_LISP_OBJECT, offsetof (struct extent_auxiliary, end_glyph) },
   { XD_LISP_OBJECT, offsetof (struct extent_auxiliary, parent) },
@@ -923,7 +965,6 @@
   { XD_LISP_OBJECT, offsetof (struct extent_auxiliary, after_change_functions) },
   { XD_END }
 };
-#endif /* USE_KKCC */
 static Lisp_Object
 mark_extent_auxiliary (Lisp_Object obj)
 {
@@ -940,16 +981,11 @@
   return data->parent;
 }
 
-#ifdef USE_KKCC
 DEFINE_LRECORD_IMPLEMENTATION ("extent-auxiliary", extent_auxiliary,
 			       0, /*dumpable-flag*/
                                mark_extent_auxiliary, internal_object_printer,
-			       0, 0, 0, extent_auxiliary_description, struct extent_auxiliary);
-#else /* not USE_KKCC */
-DEFINE_LRECORD_IMPLEMENTATION ("extent-auxiliary", extent_auxiliary,
-                               mark_extent_auxiliary, internal_object_printer,
-			       0, 0, 0, 0, struct extent_auxiliary);
-#endif /* not USE_KKCC */
+			       0, 0, 0, extent_auxiliary_description,
+			       struct extent_auxiliary);
 void
 allocate_extent_auxiliary (EXTENT ext)
 {
@@ -993,16 +1029,81 @@
 static void free_soe (struct stack_of_extents *soe);
 static void soe_invalidate (Lisp_Object obj);
 
-#ifdef USE_KKCC
-static const struct struct_description extent_list_description = {
+extern const struct sized_memory_description gap_array_marker_description;
+
+static const struct memory_description gap_array_marker_description_1[] = { 
+  { XD_STRUCT_PTR, offsetof (Gap_Array_Marker, next), 1,
+    &gap_array_marker_description },
+  { XD_END }
+};
+
+const struct sized_memory_description gap_array_marker_description = {
+  sizeof (Gap_Array_Marker),
+  gap_array_marker_description_1
 };
 
-static const struct lrecord_description extent_info_description [] = {
-  { XD_STRUCT_PTR, offsetof (struct extent_info, extents), 
-    XD_INDIRECT (0, 0), &extent_list_description },
+static const struct memory_description lispobj_gap_array_description_1[] = { 
+  { XD_INT, offsetof (Gap_Array, gap) },
+  { XD_INT, offsetof (Gap_Array, offset_past_gap) },
+  { XD_INT, offsetof (Gap_Array, els_past_gap) },
+  { XD_STRUCT_PTR, offsetof (Gap_Array, markers), 1,
+    &gap_array_marker_description, XD_FLAG_NO_KKCC },
+  { XD_STRUCT_ARRAY, offsetof (Gap_Array, array), XD_INDIRECT (0, 0),
+    &lisp_object_description },
+  { XD_STRUCT_ARRAY, XD_INDIRECT (1, offsetof (Gap_Array, array)),
+    XD_INDIRECT (2, 0), &lisp_object_description },
   { XD_END }
 };
-#endif /* USE_KKCC */
+
+static const struct sized_memory_description lispobj_gap_array_description = {
+  sizeof (Gap_Array),
+  lispobj_gap_array_description_1
+};
+
+extern const struct sized_memory_description extent_list_marker_description;
+
+static const struct memory_description extent_list_marker_description_1[] = { 
+  { XD_STRUCT_PTR, offsetof (Extent_List_Marker, m), 1,
+    &gap_array_marker_description },
+  { XD_STRUCT_PTR, offsetof (Extent_List_Marker, next), 1,
+    &extent_list_marker_description },
+  { XD_END }
+};
+
+const struct sized_memory_description extent_list_marker_description = {
+  sizeof (Extent_List_Marker),
+  extent_list_marker_description_1
+};
+
+static const struct memory_description extent_list_description_1[] = { 
+  { XD_STRUCT_PTR, offsetof (Extent_List, start), 1, &lispobj_gap_array_description },
+  { XD_STRUCT_PTR, offsetof (Extent_List, end), 1, &lispobj_gap_array_description, XD_FLAG_NO_KKCC },
+  { XD_STRUCT_PTR, offsetof (Extent_List, markers), 1, &extent_list_marker_description, XD_FLAG_NO_KKCC },
+  { XD_END }
+};
+
+static const struct sized_memory_description extent_list_description = {
+  sizeof (Extent_List),
+  extent_list_description_1
+};
+
+static const struct memory_description stack_of_extents_description_1[] = { 
+  { XD_STRUCT_PTR, offsetof (Stack_Of_Extents, extents), 1, &extent_list_description },
+  { XD_END }
+};
+
+static const struct sized_memory_description stack_of_extents_description = {
+  sizeof (Stack_Of_Extents),
+  stack_of_extents_description_1
+};
+
+static const struct memory_description extent_info_description [] = {
+  { XD_STRUCT_PTR, offsetof (struct extent_info, extents), 1,
+    &extent_list_description },
+  { XD_STRUCT_PTR, offsetof (struct extent_info, soe), 1,
+    &stack_of_extents_description, XD_FLAG_NO_KKCC },
+  { XD_END }
+};
 
 static Lisp_Object
 mark_extent_info (Lisp_Object obj)
@@ -1054,19 +1155,12 @@
     }
 }
 
-#ifdef USE_KKCC
 DEFINE_LRECORD_IMPLEMENTATION ("extent-info", extent_info,
 			       0, /*dumpable-flag*/
                                mark_extent_info, internal_object_printer,
 			       finalize_extent_info, 0, 0, 
-			       0 /*extent_info_description*/,
+			       extent_info_description,
 			       struct extent_info);
-#else /* not USE_KKCC */
-DEFINE_LRECORD_IMPLEMENTATION ("extent-info", extent_info,
-                               mark_extent_info, internal_object_printer,
-			       finalize_extent_info, 0, 0, 0,
-			       struct extent_info);
-#endif /* not USE_KKCC */
 
 static Lisp_Object
 allocate_extent_info (void)
@@ -3180,7 +3274,7 @@
 		internal_hash (extent_object (e), depth + 1));
 }
 
-static const struct lrecord_description extent_description[] = {
+static const struct memory_description extent_description[] = {
   { XD_LISP_OBJECT, offsetof (struct extent, object) },
   { XD_LISP_OBJECT, offsetof (struct extent, flags.face) },
   { XD_LISP_OBJECT, offsetof (struct extent, plist) },
@@ -3218,7 +3312,6 @@
   return Fextent_properties (obj);
 }
 
-#ifdef USE_KKCC
 DEFINE_BASIC_LRECORD_IMPLEMENTATION_WITH_PROPS ("extent", extent,
 						1, /*dumpable-flag*/
 						mark_extent,
@@ -3233,21 +3326,6 @@
 						extent_getprop, extent_putprop,
 						extent_remprop, extent_plist,
 						struct extent);
-#else /* not USE_KKCC */
-DEFINE_BASIC_LRECORD_IMPLEMENTATION_WITH_PROPS ("extent", extent,
-						mark_extent,
-						print_extent,
-						/* NOTE: If you declare a
-						   finalization method here,
-						   it will NOT be called.
-						   Shaft city. */
-						0,
-						extent_equal, extent_hash,
-						extent_description,
-						extent_getprop, extent_putprop,
-						extent_remprop, extent_plist,
-						struct extent);
-#endif /* not USE_KKCC */
 
 /************************************************************************/
 /*			basic extent accessors				*/
@@ -4398,7 +4476,7 @@
   if (EQ (at_flag, Qat))     return EXTENT_AT_AT;
 
   invalid_constant ("Invalid AT-FLAG in `extent-at'", at_flag);
-  RETURN_NOT_REACHED (EXTENT_AT_AFTER)
+  RETURN_NOT_REACHED (EXTENT_AT_AFTER);
 }
 
 static int
@@ -4640,7 +4718,7 @@
   while (1)
     Fsignal (Qbuffer_read_only, (list1 (closure->object)));
 
-  RETURN_NOT_REACHED(0)
+  RETURN_NOT_REACHED(0);
 }
 
 /* Value of Vinhibit_read_only is precomputed and passed in for
@@ -5219,7 +5297,7 @@
   if (EQ (layout_obj, Qtext))		return GL_TEXT;
 
   invalid_constant ("Unknown glyph layout type", layout_obj);
-  RETURN_NOT_REACHED (GL_TEXT)
+  RETURN_NOT_REACHED (GL_TEXT);
 }
 
 static Lisp_Object
@@ -7230,7 +7308,7 @@
   DEFSUBR (Fset_extent_endpoints);
   DEFSUBR (Fnext_extent);
   DEFSUBR (Fprevious_extent);
-#if DEBUG_XEMACS
+#ifdef DEBUG_XEMACS
   DEFSUBR (Fnext_e_extent);
   DEFSUBR (Fprevious_e_extent);
 #endif
--- a/src/faces.c	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/faces.c	Sun Jan 12 11:08:22 2003 +0000
@@ -257,7 +257,7 @@
   return result;
 }
 
-static const struct lrecord_description face_description[] = {
+static const struct memory_description face_description[] = {
   { XD_LISP_OBJECT, offsetof (Lisp_Face, name) },
   { XD_LISP_OBJECT, offsetof (Lisp_Face, doc_string) },
   { XD_LISP_OBJECT, offsetof (Lisp_Face, foreground) },
@@ -276,20 +276,13 @@
   { XD_END }
 };
 
-#ifdef USE_KKCC
 DEFINE_LRECORD_IMPLEMENTATION_WITH_PROPS ("face", face,
 					  1, /*dumpable-flag*/
 					  mark_face, print_face, 0, face_equal,
-					  face_hash, face_description, face_getprop,
+					  face_hash, face_description,
+					  face_getprop,
 					  face_putprop, face_remprop,
 					  face_plist, Lisp_Face);
-#else /* not USE_KKCC */
-DEFINE_LRECORD_IMPLEMENTATION_WITH_PROPS ("face", face,
-					  mark_face, print_face, 0, face_equal,
-					  face_hash, face_description, face_getprop,
-					  face_putprop, face_remprop,
-					  face_plist, Lisp_Face);
-#endif /* not USE_KKCC */
 
 /************************************************************************/
 /*                             face read syntax                         */
--- a/src/file-coding.c	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/file-coding.c	Sun Jan 12 11:08:22 2003 +0000
@@ -361,22 +361,22 @@
 
 static coding_system_type_entry_dynarr *the_coding_system_type_entry_dynarr;
 
-static const struct lrecord_description cste_description_1[] = {
+static const struct memory_description cste_description_1[] = {
   { XD_STRUCT_PTR,  offsetof (coding_system_type_entry, meths), 1, &coding_system_methods_description },
   { XD_END }
 };
 
-static const struct struct_description cste_description = {
+static const struct sized_memory_description cste_description = {
   sizeof (coding_system_type_entry),
   cste_description_1
 };
 
-static const struct lrecord_description csted_description_1[] = {
+static const struct memory_description csted_description_1[] = {
   XD_DYNARR_DESC (coding_system_type_entry_dynarr, &cste_description),
   { XD_END }
 };
 
-static const struct struct_description csted_description = {
+static const struct sized_memory_description csted_description = {
   sizeof (coding_system_type_entry_dynarr),
   csted_description_1
 };
@@ -398,32 +398,32 @@
 
 detector_dynarr *all_coding_detectors;
 
-static const struct lrecord_description struct_detector_category_description_1[]
+static const struct memory_description struct_detector_category_description_1[]
 =
 {
   { XD_LISP_OBJECT, offsetof (struct detector_category, sym) },
   { XD_END }
 };
 
-static const struct struct_description struct_detector_category_description =
+static const struct sized_memory_description struct_detector_category_description =
 {
   sizeof (struct detector_category),
   struct_detector_category_description_1
 };
 
-static const struct lrecord_description detector_category_dynarr_description_1[] =
+static const struct memory_description detector_category_dynarr_description_1[] =
 {
   XD_DYNARR_DESC (detector_category_dynarr,
 		  &struct_detector_category_description),
   { XD_END }
 };
 
-static const struct struct_description detector_category_dynarr_description = {
+static const struct sized_memory_description detector_category_dynarr_description = {
   sizeof (detector_category_dynarr),
   detector_category_dynarr_description_1
 };
 
-static const struct lrecord_description struct_detector_description_1[]
+static const struct memory_description struct_detector_description_1[]
 =
 {
   { XD_STRUCT_PTR, offsetof (struct detector, cats), 1,
@@ -431,19 +431,19 @@
   { XD_END }
 };
 
-static const struct struct_description struct_detector_description =
+static const struct sized_memory_description struct_detector_description =
 {
   sizeof (struct detector),
   struct_detector_description_1
 };
 
-static const struct lrecord_description detector_dynarr_description_1[] =
+static const struct memory_description detector_dynarr_description_1[] =
 {
   XD_DYNARR_DESC (detector_dynarr, &struct_detector_description),
   { XD_END }
 };
 
-static const struct struct_description detector_dynarr_description = {
+static const struct sized_memory_description detector_dynarr_description = {
   sizeof (detector_dynarr),
   detector_dynarr_description_1
 };
@@ -512,20 +512,12 @@
 {
   Lisp_Coding_System *codesys = XCODING_SYSTEM (obj);
 
-  mark_object (CODING_SYSTEM_NAME (codesys));
-  mark_object (CODING_SYSTEM_DESCRIPTION (codesys));
-  mark_object (CODING_SYSTEM_MNEMONIC (codesys));
-  mark_object (CODING_SYSTEM_DOCUMENTATION (codesys));
-  mark_object (CODING_SYSTEM_EOL_LF (codesys));
-  mark_object (CODING_SYSTEM_EOL_CRLF (codesys));
-  mark_object (CODING_SYSTEM_EOL_CR (codesys));
-  mark_object (CODING_SYSTEM_SUBSIDIARY_PARENT (codesys));
-  mark_object (CODING_SYSTEM_CANONICAL (codesys));
+#define MARKED_SLOT(x) mark_object (codesys->x);
+#include "coding-system-slots.h"
 
   MAYBE_CODESYSMETH (codesys, mark, (obj));
 
-  mark_object (CODING_SYSTEM_PRE_WRITE_CONVERSION (codesys));
-  return CODING_SYSTEM_POST_READ_CONVERSION (codesys);
+  return Qnil;
 }
 
 static void
@@ -583,7 +575,7 @@
   return offsetof (Lisp_Coding_System, data) + p->methods->extra_data_size;
 }
 
-static const struct lrecord_description coding_system_methods_description_1[]
+static const struct memory_description coding_system_methods_description_1[]
 = {
   { XD_LISP_OBJECT,
     offsetof (struct coding_system_methods, type) },
@@ -592,36 +584,40 @@
   { XD_END }
 };
 
-const struct struct_description coding_system_methods_description = {
+const struct sized_memory_description coding_system_methods_description = {
   sizeof (struct coding_system_methods),
   coding_system_methods_description_1
 };
 
-const struct lrecord_description coding_system_empty_extra_description[] = {
-  { XD_END }
+static const struct sized_memory_description coding_system_extra_description_map[] =
+{
+  { offsetof (Lisp_Coding_System, methods) },
+  { offsetof (struct coding_system_methods, extra_description) },
+  { -1 },
 };
 
-static const struct lrecord_description coding_system_description[] =
+static const struct memory_description coding_system_description[] =
 {
   { XD_STRUCT_PTR,  offsetof (Lisp_Coding_System, methods), 1,
     &coding_system_methods_description },
-  { XD_LISP_OBJECT, offsetof (Lisp_Coding_System, name) },
-  { XD_LISP_OBJECT, offsetof (Lisp_Coding_System, description) },
-  { XD_LISP_OBJECT, offsetof (Lisp_Coding_System, mnemonic) },
-  { XD_LISP_OBJECT, offsetof (Lisp_Coding_System, documentation) },
-  { XD_LISP_OBJECT, offsetof (Lisp_Coding_System, post_read_conversion) },
-  { XD_LISP_OBJECT, offsetof (Lisp_Coding_System, pre_write_conversion) },
-  { XD_LISP_OBJECT, offsetof (Lisp_Coding_System, text_file_wrapper) },
-  { XD_LISP_OBJECT, offsetof (Lisp_Coding_System, auto_eol_wrapper) },
-  { XD_LISP_OBJECT, offsetof (Lisp_Coding_System, eol[0]) },
-  { XD_LISP_OBJECT, offsetof (Lisp_Coding_System, eol[1]) },
-  { XD_LISP_OBJECT, offsetof (Lisp_Coding_System, eol[2]) },
-  { XD_LISP_OBJECT, offsetof (Lisp_Coding_System, subsidiary_parent) },
-  { XD_LISP_OBJECT, offsetof (Lisp_Coding_System, canonical) },
-  { XD_CODING_SYSTEM_END }
+#define MARKED_SLOT(x) { XD_LISP_OBJECT, offsetof (Lisp_Coding_System, x) },
+#define MARKED_SLOT_ARRAY(slot, size) \
+  { XD_LISP_OBJECT_ARRAY, offsetof (Lisp_Coding_System, slot), size },
+#include "coding-system-slots.h"
+  { XD_STRUCT_ARRAY, offsetof (Lisp_Coding_System, data), 1,
+    coding_system_extra_description_map },
+  { XD_END }
 };
 
-#ifdef USE_KKCC
+static const struct memory_description coding_system_empty_extra_description_1[] =
+{
+  { XD_END }
+};
+
+const struct sized_memory_description coding_system_empty_extra_description = {
+  0, coding_system_empty_extra_description_1
+};
+
 DEFINE_LRECORD_SEQUENCE_IMPLEMENTATION ("coding-system", coding_system,
 					1, /*dumpable-flag*/
 					mark_coding_system,
@@ -630,15 +626,6 @@
 					0, 0, coding_system_description,
 					sizeof_coding_system,
 					Lisp_Coding_System);
-#else /* not USE_KKCC */
-DEFINE_LRECORD_SEQUENCE_IMPLEMENTATION ("coding-system", coding_system,
-					mark_coding_system,
-					print_coding_system,
-					finalize_coding_system,
-					0, 0, coding_system_description,
-					sizeof_coding_system,
-					Lisp_Coding_System);
-#endif /* not USE_KKCC */
 
 /************************************************************************/
 /*                       Creating coding systems                        */
@@ -962,22 +949,14 @@
 {
   Bytecount total_size = offsetof (Lisp_Coding_System, data) + data_size;
   Lisp_Coding_System *codesys =
-    (Lisp_Coding_System *) alloc_lcrecord (total_size, &lrecord_coding_system);
-
-  zero_sized_lcrecord (codesys, total_size);
+    (Lisp_Coding_System *) basic_alloc_lcrecord (total_size,
+						 &lrecord_coding_system);
+
   codesys->methods = codesys_meths;
-  CODING_SYSTEM_PRE_WRITE_CONVERSION (codesys) = Qnil;
-  CODING_SYSTEM_POST_READ_CONVERSION (codesys) = Qnil;
+#define MARKED_SLOT(x) codesys->x = Qnil;
+#include "coding-system-slots.h"
+
   CODING_SYSTEM_EOL_TYPE (codesys) = EOL_LF;
-  CODING_SYSTEM_EOL_CRLF (codesys) = Qnil;
-  CODING_SYSTEM_EOL_CR   (codesys) = Qnil;
-  CODING_SYSTEM_EOL_LF   (codesys) = Qnil;
-  CODING_SYSTEM_SUBSIDIARY_PARENT (codesys) = Qnil;
-  CODING_SYSTEM_CANONICAL (codesys) = Qnil;
-  CODING_SYSTEM_MNEMONIC (codesys) = Qnil;
-  CODING_SYSTEM_DOCUMENTATION (codesys) = Qnil;
-  CODING_SYSTEM_TEXT_FILE_WRAPPER (codesys) = Qnil;
-  CODING_SYSTEM_AUTO_EOL_WRAPPER (codesys) = Qnil;
   CODING_SYSTEM_NAME     (codesys) = name;
 
   MAYBE_CODESYSMETH (codesys, init, (wrap_coding_system (codesys)));
@@ -995,7 +974,7 @@
   if (EQ (symbol, Qcr))   return EOL_CR;
 
   invalid_constant ("Unrecognized eol type", symbol);
-  RETURN_NOT_REACHED (EOL_AUTODETECT)
+  RETURN_NOT_REACHED (EOL_AUTODETECT);
 }
 
 static Lisp_Object
@@ -1708,7 +1687,7 @@
     return aliasee;
   else
     invalid_argument ("Symbol is not a coding system alias", alias);
-  RETURN_NOT_REACHED (Qnil)
+  RETURN_NOT_REACHED (Qnil);
 }
 
 /* A maphash function, for removing dangling coding system aliases. */
@@ -1989,7 +1968,30 @@
    that is at the other end, and data that needs to be persistent across
    the lifetime of the stream. */
 
-DEFINE_LSTREAM_IMPLEMENTATION ("coding", coding);
+extern const struct sized_memory_description chain_coding_stream_description;
+extern const struct sized_memory_description undecided_coding_stream_description;
+
+static const struct memory_description coding_stream_data_description_1 []= {
+  { XD_STRUCT_PTR, chain_coding_system, 1, &chain_coding_stream_description},
+  { XD_STRUCT_PTR, undecided_coding_system, 1, &undecided_coding_stream_description},
+  { XD_END }
+};
+
+static const struct sized_memory_description coding_stream_data_description = {
+  sizeof (void *), coding_stream_data_description_1
+};
+
+static const struct memory_description coding_lstream_description[] = {
+  { XD_INT, offsetof (struct coding_stream, type) },
+  { XD_LISP_OBJECT, offsetof (struct coding_stream, orig_codesys) },
+  { XD_LISP_OBJECT, offsetof (struct coding_stream, codesys) },
+  { XD_LISP_OBJECT, offsetof (struct coding_stream, other_end) },
+  { XD_UNION, offsetof (struct coding_stream, data), 
+    XD_INDIRECT (0, 0), &coding_stream_data_description },
+  { XD_END }
+};
+
+DEFINE_LSTREAM_IMPLEMENTATION_WITH_DATA ("coding", coding);
 
 /* Encoding and decoding are parallel operations, so we create just one
    stream for both. "Decoding" may involve the extra step of autodetection
@@ -2309,9 +2311,12 @@
       str->data = 0;
     }
   if (XCODING_SYSTEM_METHODS (str->codesys)->coding_data_size)
-    str->data =
-      xmalloc_and_zero (XCODING_SYSTEM_METHODS (str->codesys)->
-			coding_data_size);
+    {
+      str->data =
+	xmalloc_and_zero (XCODING_SYSTEM_METHODS (str->codesys)->
+			  coding_data_size);
+      str->type = XCODING_SYSTEM_METHODS (str->codesys)->enumtype;
+    }
   MAYBE_XCODESYSMETH (str->codesys, init_coding_stream, (str));
   /* The new coding system may have different ideas regarding whether its
      ends are characters or bytes. */
@@ -2538,7 +2543,6 @@
 
 /* Chain two or more coding systems together to make a combination coding
    system. */
-DEFINE_CODING_SYSTEM_TYPE (chain);
 
 struct chain_coding_system
 {
@@ -2558,30 +2562,28 @@
   int lstream_count;
 };
 
-static const struct lrecord_description lo_description_1[] = {
-  { XD_LISP_OBJECT, 0 },
+static const struct memory_description chain_coding_system_description[] = {
+  { XD_INT, offsetof (struct chain_coding_system, count) },
+  { XD_STRUCT_PTR, offsetof (struct chain_coding_system, chain),
+    XD_INDIRECT (0, 0), &lisp_object_description },
+  { XD_LISP_OBJECT, offsetof (struct chain_coding_system,
+			      canonicalize_after_coding) },
   { XD_END }
 };
 
-static const struct struct_description lo_description = {
-  sizeof (Lisp_Object),
-  lo_description_1
-};
-
-static const struct lrecord_description chain_coding_system_description[] = {
-  { XD_INT,
-      coding_system_data_offset + offsetof (struct chain_coding_system, 
-					    count) },
-  { XD_STRUCT_PTR,
-      coding_system_data_offset + offsetof (struct chain_coding_system,
-					    chain),
-    XD_INDIRECT (0, 0), &lo_description },
-  { XD_LISP_OBJECT,
-      coding_system_data_offset + offsetof (struct chain_coding_system,
-					    canonicalize_after_coding) },
+static const struct memory_description chain_coding_stream_description_1 [] = {
+  { XD_INT, offsetof (struct chain_coding_stream, lstream_count) },
+  { XD_STRUCT_PTR, offsetof (struct chain_coding_stream, lstreams),
+    XD_INDIRECT (0, 0), &lisp_object_description },
   { XD_END }
 };
 
+const struct sized_memory_description chain_coding_stream_description = {
+  sizeof (struct chain_coding_stream), chain_coding_stream_description_1
+};
+
+DEFINE_CODING_SYSTEM_TYPE_WITH_DATA (chain);
+
 static Lisp_Object
 chain_canonicalize (Lisp_Object codesys)
 {
@@ -3024,8 +3026,6 @@
 There is one parameter: `subtype', either `cr', `lf', `crlf', or nil.
 */
 
-DEFINE_CODING_SYSTEM_TYPE (convert_eol);
-
 struct convert_eol_coding_system
 {
   enum eol_type subtype;
@@ -3041,11 +3041,13 @@
   enum eol_type actual;
 };
 
-static const struct lrecord_description
+static const struct memory_description
   convert_eol_coding_system_description[] = {
   { XD_END }
 };
 
+DEFINE_CODING_SYSTEM_TYPE_WITH_DATA (convert_eol);
+
 static void
 convert_eol_print (Lisp_Object cs, Lisp_Object printcharfun, int escapeflag)
 {
@@ -3283,8 +3285,6 @@
    EOL-processed twice.
    */
 
-DEFINE_CODING_SYSTEM_TYPE (undecided);
-
 struct undecided_coding_system
 {
   int do_eol, do_coding;
@@ -3300,14 +3300,24 @@
   struct detection_state *st;
 };
 
-static const struct lrecord_description
-  undecided_coding_system_description[] = {
-  { XD_LISP_OBJECT,
-    coding_system_data_offset + offsetof (struct undecided_coding_system,
-					  cs) },
+static const struct memory_description undecided_coding_system_description[] = {
+  { XD_LISP_OBJECT, offsetof (struct undecided_coding_system, cs) },
   { XD_END }
 };
 
+static const struct memory_description undecided_coding_stream_description_1 [] = {
+  { XD_LISP_OBJECT, offsetof (struct undecided_coding_stream, actual) },
+  { XD_STRUCT_ARRAY, offsetof (struct undecided_coding_stream, c),
+    1, &chain_coding_stream_description },
+  { XD_END }
+};
+
+const struct sized_memory_description undecided_coding_stream_description = {
+  sizeof (struct undecided_coding_stream), undecided_coding_stream_description_1
+};
+
+DEFINE_CODING_SYSTEM_TYPE_WITH_DATA (undecided);
+
 static void
 undecided_init (Lisp_Object codesys)
 {
@@ -3359,6 +3369,7 @@
 static void
 undecided_mark_coding_stream (struct coding_stream *str)
 {
+  mark_object (CODING_STREAM_TYPE_DATA (str, undecided)->actual);
   chain_mark_coding_stream_1 (&CODING_STREAM_TYPE_DATA (str, undecided)->c);
 }
 
@@ -3449,7 +3460,7 @@
     }
   
   invalid_constant ("Unrecognized coding category", symbol);
-  RETURN_NOT_REACHED (0)
+  RETURN_NOT_REACHED (0);
 }
 
 static Lisp_Object
@@ -4271,8 +4282,6 @@
 /*                             Gzip methods                             */
 /************************************************************************/
 
-DEFINE_CODING_SYSTEM_TYPE (gzip);
-
 struct gzip_coding_system
 {
   int level; /* 0 through 9, or -1 for default */
@@ -4291,11 +4300,13 @@
 		      return LSTREAM_EOF */
 };
 
-static const struct lrecord_description
+static const struct memory_description
   gzip_coding_system_description[] = {
   { XD_END }
 };
 
+DEFINE_CODING_SYSTEM_TYPE_WITH_DATA (gzip);
+
 enum source_sink_type
 gzip_conversion_end_type (Lisp_Object codesys)
 {
@@ -4640,7 +4651,7 @@
   for (i = 0; i < MAX_DETECTOR_CATEGORIES; i++)
     {
       coding_category_system[i] = Qnil;
-      dump_add_root_object (&coding_category_system[i]);
+      dump_add_root_lisp_object (&coding_category_system[i]);
       coding_category_by_priority[i] = i;
     }
 
--- a/src/file-coding.h	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/file-coding.h	Sun Jan 12 11:08:22 2003 +0000
@@ -162,7 +162,7 @@
 struct coding_stream;
 struct detection_state;
 
-extern const struct struct_description coding_system_methods_description;
+extern const struct sized_memory_description coding_system_methods_description;
 
 struct coding_system_methods;
 
@@ -187,73 +187,18 @@
   struct lcrecord_header header;
   struct coding_system_methods *methods;
 
-  /* Name and description of this coding system.  The description
-     should be suitable for a menu entry. */
-  Lisp_Object name;
-  Lisp_Object description;
-
-  /* Mnemonic string displayed in the modeline when this coding
-     system is active for a particular buffer. */
-  Lisp_Object mnemonic;
-
-  /* Long documentation on the coding system. */
-  Lisp_Object documentation;
-  /* Functions to handle additional conversion after reading or before
-     writing. #### This mechanism should be replaced by the ability to
-     simply create new coding system types. */
-  Lisp_Object post_read_conversion;
-  Lisp_Object pre_write_conversion;
-
-  /* If this coding system is not of the correct type for text file
-     conversion (i.e. decodes byte->char), we wrap it with appropriate
-     char<->byte converters.  This is created dynamically, when it's
-     needed, and cached here. */
-  Lisp_Object text_file_wrapper;
-
   /* If true, this is an internal coding system, which will not show up in
      coding-system-list unless a special parameter is given to it. */
   int internal_p;
 
-  /* ------------------------ junk to handle EOL -------------------------
-     I had hoped that we could handle this without lots of special-case
-     code, but it appears not to be the case if we want to maintain
-     compatibility with the existing way.  However, at least with the way
-     we do things now, we avoid EOL junk in most of the coding system
-     methods themselves, or in the decode/encode functions.  The EOL
-     special-case code is limited to coding-system creation and to the
-     convert-eol and undecided coding system types. */
+#define CODING_SYSTEM_SLOT_DECLARATION
+#define MARKED_SLOT(x) Lisp_Object x;
+#include "coding-system-slots.h"
 
-  /* If this coding system wants autodetection of the EOL type, then at the
-     appropriate time we wrap this coding system with
-     convert-eol-autodetect. (We do NOT do this at creation time because
-     then we end up with multiple convert-eols wrapped into the final
-     result -- esp. with autodetection using `undecided' -- leading to a
-     big mess.) We cache the wrapped coding system here. */
-  Lisp_Object auto_eol_wrapper;
-  
-  /* Eol type requested by user. */
+  /* Eol type requested by user.  See comment about EOL junk in
+     coding-system-slots.h. */
   enum eol_type eol_type;
 
-  /* Subsidiary coding systems that specify a particular type of EOL
-     marking, rather than autodetecting it.  These will only be non-nil
-     if (eol_type == EOL_AUTODETECT).  These are chains. */
-  Lisp_Object eol[3];
-  /* If this coding system is a subsidiary, this element points back to its
-     parent. */
-  Lisp_Object subsidiary_parent;
-
-  /* At decoding or encoding time, we use the following coding system, if
-     it exists, in place of the coding system object.  This is how we
-     handle coding systems with EOL types of CRLF or CR.  Formerly, we did
-     the canonicalization at creation time, returning a chain in place of
-     the original coding system; but that interferes with
-     `coding-system-property' and causes other complications.  CANONICAL is
-     used when determining the end types of a coding system.
-     canonicalize-after-coding also consults CANONICAL (it has to, because
-     the data in the lstream is based on CANONICAL, not on the original
-     coding system). */
-  Lisp_Object canonical;
-  
   /* type-specific extra data attached to a coding_system */
   char data[1];
 };
@@ -266,11 +211,34 @@
 #define CHECK_CODING_SYSTEM(x) CHECK_RECORD (x, coding_system)
 #define CONCHECK_CODING_SYSTEM(x) CONCHECK_RECORD (x, coding_system)
 
+enum coding_system_variant
+{
+  no_conversion_coding_system,
+  convert_eol_coding_system,
+  undecided_coding_system,
+  chain_coding_system,
+  text_file_wrapper_coding_system,
+  internal_coding_system,
+  gzip_coding_system,
+  mswindows_multibyte_to_unicode_coding_system,
+  mswindows_multibyte_coding_system,
+  iso2022_coding_system,
+  ccl_coding_system,
+  shift_jis_coding_system,
+  big5_coding_system,
+  unicode_coding_system,
+};
+
 struct coding_system_methods
 {
   Lisp_Object type;
   Lisp_Object predicate_symbol;
 
+  /* Type expressed as an enum, needed for KKCC marking of the
+     type-specific lstream data; copied into the struct coding_stream. */
+
+  enum coding_system_variant enumtype;
+
   /* Implementation specific methods: */
 
   /* Init method: Initialize coding-system data.  Optional. */
@@ -401,9 +369,8 @@
      INITIALIZE_CODING_SYSTEM_TYPE_WITH_DATA. */
 
   /* Description of the extra data (struct foo_coding_system) attached to a
-     coding system, for pdump purposes.  NOTE: All offsets must have
-     coding_system_data_offset added to them! */
-  const struct lrecord_description *extra_description;
+     coding system, for pdump purposes. */
+  const struct sized_memory_description *extra_description;
   /* size of struct foo_coding_system -- extra data associated with
      the coding system */
   int extra_data_size;
@@ -445,8 +412,7 @@
 
 /***** Defining new coding-system types *****/
 
-#define coding_system_data_offset (offsetof (Lisp_Coding_System, data))
-extern const struct lrecord_description coding_system_empty_extra_description[];
+extern const struct sized_memory_description coding_system_empty_extra_description;
 
 #ifdef ERROR_CHECK_TYPES
 #define DECLARE_CODING_SYSTEM_TYPE(type)				\
@@ -492,12 +458,21 @@
 #define DEFINE_CODING_SYSTEM_TYPE(type)					\
 struct coding_system_methods * type##_coding_system_methods
 
+#define DEFINE_CODING_SYSTEM_TYPE_WITH_DATA(type)		\
+struct coding_system_methods * type##_coding_system_methods;	\
+static const struct sized_memory_description			\
+  type##_coding_system_description_0 = {			\
+  sizeof (struct type##_coding_system),				\
+  type##_coding_system_description				\
+}
+
 #define INITIALIZE_CODING_SYSTEM_TYPE(ty, pred_sym) do {		\
   ty##_coding_system_methods =						\
     xnew_and_zero (struct coding_system_methods);			\
   ty##_coding_system_methods->type = Q##ty;				\
   ty##_coding_system_methods->extra_description =			\
-    coding_system_empty_extra_description;				\
+    &coding_system_empty_extra_description;				\
+  ty##_coding_system_methods->enumtype = ty##_coding_system;		\
   defsymbol_nodump (&ty##_coding_system_methods->predicate_symbol,	\
                     pred_sym);						\
   add_entry_to_coding_system_type_list (ty##_coding_system_methods);	\
@@ -514,11 +489,10 @@
    struct foo_coding_system (attached to the coding system)
    struct foo_coding_stream (per coding process, attached to the
                              struct coding_stream)
-   const struct foo_coding_system_description[] (pdump description of
-                                                 struct foo_coding_system)
+   const struct memory_description foo_coding_system_description[]
+   (data description of struct foo_coding_system)
 
-   NOTE: The description must have coding_system_data_offset added to
-   all offsets in it!  For an example of how to do things, see
+   For an example of how to do the description, see
    chain_coding_system_description.
 */
 #define INITIALIZE_CODING_SYSTEM_TYPE_WITH_DATA(type, pred_sym)	\
@@ -527,7 +501,7 @@
   type##_coding_system_methods->extra_data_size =		\
     sizeof (struct type##_coding_system);			\
   type##_coding_system_methods->extra_description =		\
-    type##_coding_system_description;				\
+    &type##_coding_system_description_0;			\
   type##_coding_system_methods->coding_data_size =		\
     sizeof (struct type##_coding_stream);			\
 } while (0)
@@ -868,6 +842,14 @@
 
 struct coding_stream
 {
+  /* Enumerated constant listing which type of console this is (TTY, X,
+     MS-Windows, etc.).  This duplicates the method structure in
+     XCODING_SYSTEM (str->codesys)->methods->type, which formerly was the
+     only way to determine the coding system type.  We need this constant
+     now for KKCC, so that it can be used in an XD_UNION clause to
+     determine the Lisp objects in the type-specific data. */
+  enum coding_system_variant type;
+
   /* Coding system that governs the conversion. */
   Lisp_Object codesys;
   /* Original coding system, pre-canonicalization. */
--- a/src/fileio.c	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/fileio.c	Sun Jan 12 11:08:22 2003 +0000
@@ -1568,7 +1568,7 @@
 	qxestrncpy (target, o, s - o);
 	target[s - o] = 0;
 #ifdef WIN32_NATIVE
-	strupr (target); /* $home == $HOME etc.  */
+	qxestrupr (target); /* $home == $HOME etc.  */
 #endif /* WIN32_NATIVE */
 
 	/* Get variable value */
@@ -1619,7 +1619,7 @@
 	qxestrncpy (target, o, s - o);
 	target[s - o] = 0;
 #ifdef WIN32_NATIVE
-	strupr (target); /* $home == $HOME etc.  */
+	qxestrupr (target); /* $home == $HOME etc.  */
 #endif /* WIN32_NATIVE */
 
 	/* Get variable value */
@@ -1663,7 +1663,7 @@
   syntax_error_2 ("Substituting nonexistent environment variable",
 		  filename, build_intstring (target));
 
-  RETURN_NOT_REACHED (Qnil)
+  RETURN_NOT_REACHED (Qnil);
 }
 
 /* A slightly faster and more convenient way to get
--- a/src/floatfns.c	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/floatfns.c	Sun Jan 12 11:08:22 2003 +0000
@@ -178,22 +178,15 @@
   return (unsigned long) fmod (extract_float (obj), 4e9);
 }
 
-static const struct lrecord_description float_description[] = {
+static const struct memory_description float_description[] = {
   { XD_END }
 };
 
-#ifdef USE_KKCC
 DEFINE_BASIC_LRECORD_IMPLEMENTATION ("float", float,
 				     1, /*dumpable-flag*/
 				     mark_float, print_float, 0, float_equal,
 				     float_hash, float_description,
 				     Lisp_Float);
-#else /* not USE_KKCC */
-DEFINE_BASIC_LRECORD_IMPLEMENTATION ("float", float,
-				     mark_float, print_float, 0, float_equal,
-				     float_hash, float_description,
-				     Lisp_Float);
-#endif /* not USE_KKCC */
 
 /* Extract a Lisp number as a `double', or signal an error.  */
 
--- a/src/fns.c	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/fns.c	Sun Jan 12 11:08:22 2003 +0000
@@ -125,26 +125,20 @@
 				       BIT_VECTOR_LONG_STORAGE (bit_vector_length (v)));
 }
 
-static const struct lrecord_description bit_vector_description[] = {
-  { XD_LISP_OBJECT, offsetof (Lisp_Bit_Vector, next) },
+static const struct memory_description bit_vector_description[] = {
   { XD_END }
 };
 
 
-#ifdef USE_KKCC
-DEFINE_BASIC_LRECORD_SEQUENCE_IMPLEMENTATION ("bit-vector", bit_vector,
-					      1, /*dumpable-flag*/
-					      mark_bit_vector, print_bit_vector, 0,
-					      bit_vector_equal, bit_vector_hash,
-					      bit_vector_description, size_bit_vector,
-					      Lisp_Bit_Vector);
-#else /* not USE_KKCC */
-DEFINE_BASIC_LRECORD_SEQUENCE_IMPLEMENTATION ("bit-vector", bit_vector,
-					      mark_bit_vector, print_bit_vector, 0,
-					      bit_vector_equal, bit_vector_hash,
-					      bit_vector_description, size_bit_vector,
-					      Lisp_Bit_Vector);
-#endif /* not USE_KKCC */
+DEFINE_LRECORD_SEQUENCE_IMPLEMENTATION ("bit-vector", bit_vector,
+					1, /*dumpable-flag*/
+					mark_bit_vector,
+					print_bit_vector, 0,
+					bit_vector_equal,
+					bit_vector_hash,
+					bit_vector_description,
+					size_bit_vector,
+					Lisp_Bit_Vector);
 
 
 DEFUN ("identity", Fidentity, 1, 1, 0, /*
@@ -3670,7 +3664,7 @@
 
   while (1)
     {
-      Ibyte c;
+      Ibyte c = 0;
       if (!ADVANCE_INPUT (c, istream))
 	break;
 
--- a/src/frame-gtk.c	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/frame-gtk.c	Sun Jan 12 11:08:22 2003 +0000
@@ -1,6 +1,6 @@
 /* Functions for the X window system.
    Copyright (C) 1989, 1992-5, 1997 Free Software Foundation, Inc.
-   Copyright (C) 1995, 1996 Ben Wing.
+   Copyright (C) 1995, 1996, 2002 Ben Wing.
 
 This file is part of XEmacs.
 
@@ -94,6 +94,20 @@
 
 #endif
 
+static const struct memory_description gtk_frame_data_description_1 [] = {
+  { XD_LISP_OBJECT, offsetof (struct gtk_frame, icon_pixmap) },
+  { XD_LISP_OBJECT, offsetof (struct gtk_frame, icon_pixmap_mask) },
+  { XD_LISP_OBJECT_ARRAY, offsetof (struct gtk_frame, lisp_visible_widgets),
+    3 },
+  { XD_END }
+};
+
+extern const struct sized_memory_description gtk_frame_data_description;
+
+const struct sized_memory_description gtk_frame_data_description = {
+  sizeof (struct gtk_frame), gtk_frame_data_description_1
+};
+
 
 /************************************************************************/
 /*                          helper functions                            */
@@ -743,7 +757,8 @@
       if (!NILP (dtyp) && STRINGP (dtyp))
 	dnd_typ = gdk_atom_intern (XSTRING_DATA (dtyp), FALSE);
 
-      gtk_drag_begin (wid, tl, GDK_ACTION_COPY, lisp_event->event.button.button, NULL);
+      gtk_drag_begin (wid, tl, GDK_ACTION_COPY,
+		      EVENT_BUTTON_BUTTON (lisp_event), NULL);
 
       Vcurrent_drag_object = data;
 
--- a/src/frame-impl.h	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/frame-impl.h	Sun Jan 12 11:08:22 2003 +0000
@@ -47,6 +47,9 @@
      through frame->device->console, but it's faster this way. */
   struct console_methods *framemeths;
 
+  /* Duplicates framemeths->symbol.  See comment in struct console. */
+  enum console_variant frametype;
+
   /* Size of text only area of this frame, excluding scrollbars,
      toolbars and end of line glyphs. The size can be in characters
      or pixels, depending on units in which window system resizes
@@ -109,12 +112,13 @@
   display_line_dynarr *current_display_lines[4];
   display_line_dynarr *desired_display_lines[4];
 
-  /* A structure of auxiliary data specific to the device type.
-     struct x_frame is for X window frames; defined in console-x-impl.h */
+  /* A structure of auxiliary data specific to the device type.  For
+     example, struct x_frame is for X window frames; defined in
+     console-x-impl.h. */
   void *frame_data;
 
 #define FRAME_SLOT_DECLARATION
-#define MARKED_SLOT(x) Lisp_Object x
+#define MARKED_SLOT(x) Lisp_Object x;
 #include "frameslots.h"
 
     /* Nonzero if frame is currently displayed.
--- a/src/frame-msw.c	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/frame-msw.c	Sun Jan 12 11:08:22 2003 +0000
@@ -81,6 +81,23 @@
    frame Lisp_Object already protected by Fmake_frame */
 Lisp_Object Vmswindows_frame_being_created;
 
+static const struct memory_description mswindows_frame_data_description_1 [] = {
+#ifdef HAVE_TOOLBARS
+  { XD_LISP_OBJECT, offsetof (struct mswindows_frame, toolbar_hash_table) },
+#endif
+  { XD_LISP_OBJECT, offsetof (struct mswindows_frame, menu_hash_table) },
+  { XD_LISP_OBJECT, offsetof (struct mswindows_frame, widget_hash_table1) },
+  { XD_LISP_OBJECT, offsetof (struct mswindows_frame, widget_hash_table2) },
+  { XD_LISP_OBJECT, offsetof (struct mswindows_frame, widget_hash_table3) },
+  { XD_END }
+};
+
+extern const struct sized_memory_description mswindows_frame_data_description;
+
+const struct sized_memory_description mswindows_frame_data_description = {
+  sizeof (struct mswindows_frame), mswindows_frame_data_description_1
+};
+
 /*---------------------------------------------------------------------*/
 /*-----                    DISPLAY FRAME                          -----*/
 /*---------------------------------------------------------------------*/
--- a/src/frame-x.c	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/frame-x.c	Sun Jan 12 11:08:22 2003 +0000
@@ -70,6 +70,18 @@
 Lisp_Object Qwindow_id;
 Lisp_Object Qx_resource_name;
 
+static const struct memory_description x_frame_data_description_1 [] = {
+  { XD_LISP_OBJECT, offsetof (struct x_frame, icon_pixmap) },
+  { XD_LISP_OBJECT, offsetof (struct x_frame, icon_pixmap_mask) },
+  { XD_END }
+};
+
+extern const struct sized_memory_description x_frame_data_description;
+
+const struct sized_memory_description x_frame_data_description = {
+  sizeof (struct x_frame), x_frame_data_description_1
+};
+
 EXFUN (Fx_window_id, 1);
 
 
@@ -1096,31 +1108,31 @@
       x_event.xbutton.root = XRootWindow(x_event.xbutton.display, 0);
       x_event.xbutton.subwindow = 0;
       x_event.xbutton.time = lisp_event->timestamp;
-      x_event.xbutton.x = lisp_event->event.button.x;
-      x_event.xbutton.y = lisp_event->event.button.y;
+      x_event.xbutton.x = EVENT_BUTTON_X (lisp_event);
+      x_event.xbutton.y = EVENT_BUTTON_Y (lisp_event);
       if (Success == XGetWindowAttributes (x_event.xbutton.display,
 					   x_event.xbutton.window,
 					   &win_attrib))
 	{
-	  x_event.xbutton.x_root = win_attrib.x + lisp_event->event.button.x;
-	  x_event.xbutton.y_root = win_attrib.y + lisp_event->event.button.y;
+	  x_event.xbutton.x_root = win_attrib.x + EVENT_BUTTON_X (lisp_event);
+	  x_event.xbutton.y_root = win_attrib.y + EVENT_BUTTON_Y (lisp_event);
 	}
       else
 	{
-	  x_event.xbutton.x_root = lisp_event->event.button.x; /* this is wrong */
-	  x_event.xbutton.y_root = lisp_event->event.button.y;
+	  x_event.xbutton.x_root = EVENT_BUTTON_X (lisp_event); /* this is wrong */
+	  x_event.xbutton.y_root = EVENT_BUTTON_Y (lisp_event);
 	}
-      modifier = lisp_event->event.button.modifiers;
+      modifier = EVENT_BUTTON_MODIFIERS (lisp_event);
       if (modifier & XEMACS_MOD_SHIFT)   state |= ShiftMask;
       if (modifier & XEMACS_MOD_CONTROL) state |= ControlMask;
       if (modifier & XEMACS_MOD_META)    state |= xd->MetaMask;
       if (modifier & XEMACS_MOD_SUPER)   state |= xd->SuperMask;
       if (modifier & XEMACS_MOD_HYPER)   state |= xd->HyperMask;
       if (modifier & XEMACS_MOD_ALT)     state |= xd->AltMask;
-      state |= Button1Mask << (lisp_event->event.button.button-1);
+      state |= Button1Mask << (EVENT_BUTTON_BUTTON (lisp_event)-1);
 
       x_event.xbutton.state = state;
-      x_event.xbutton.button = lisp_event->event.button.button;
+      x_event.xbutton.button = EVENT_BUTTON_BUTTON (lisp_event);
       x_event.xkey.same_screen = True;
 
       /* convert data strings into a big string */
@@ -1361,32 +1373,32 @@
       x_event.xbutton.root = XRootWindow(x_event.xkey.display, 0);
       x_event.xbutton.subwindow = 0;
       x_event.xbutton.time = lisp_event->timestamp;
-      x_event.xbutton.x = lisp_event->event.button.x;
-      x_event.xbutton.y = lisp_event->event.button.y;
+      x_event.xbutton.x = EVENT_BUTTON_X (lisp_event);
+      x_event.xbutton.y = EVENT_BUTTON_Y (lisp_event);
       if (Success == XGetWindowAttributes (x_event.xbutton.display,
 					   x_event.xbutton.window,
 					   &win_attrib))
 	{
-	  x_event.xbutton.x_root = win_attrib.x + lisp_event->event.button.x;
-	  x_event.xbutton.y_root = win_attrib.y + lisp_event->event.button.y;
+	  x_event.xbutton.x_root = win_attrib.x + EVENT_BUTTON_X (lisp_event);
+	  x_event.xbutton.y_root = win_attrib.y + EVENT_BUTTON_Y (lisp_event);
 	}
       else
 	{
-	  x_event.xbutton.x_root = lisp_event->event.button.x; /* this is wrong */
-	  x_event.xbutton.y_root = lisp_event->event.button.y;
+	  x_event.xbutton.x_root = EVENT_BUTTON_X (lisp_event); /* this is wrong */
+	  x_event.xbutton.y_root = EVENT_BUTTON_Y (lisp_event);
 	}
 
-      modifier = lisp_event->event.button.modifiers;
+      modifier = EVENT_BUTTON_MODIFIERS (lisp_event);
       if (modifier & XEMACS_MOD_SHIFT)   state |= ShiftMask;
       if (modifier & XEMACS_MOD_CONTROL) state |= ControlMask;
       if (modifier & XEMACS_MOD_META)    state |= xd->MetaMask;
       if (modifier & XEMACS_MOD_SUPER)   state |= xd->SuperMask;
       if (modifier & XEMACS_MOD_HYPER)   state |= xd->HyperMask;
       if (modifier & XEMACS_MOD_ALT)     state |= xd->AltMask;
-      state |= Button1Mask << (lisp_event->event.button.button-1);
+      state |= Button1Mask << (EVENT_BUTTON_BUTTON (lisp_event)-1);
 
       x_event.xbutton.state = state;
-      x_event.xbutton.button = lisp_event->event.button.button;
+      x_event.xbutton.button = EVENT_BUTTON_BUTTON (lisp_event);
       x_event.xkey.same_screen = True;
 
       DndSetData(dnd_typ, (unsigned char *)dnd_data, dnd_len);
--- a/src/frame.c	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/frame.c	Sun Jan 12 11:08:22 2003 +0000
@@ -137,12 +137,85 @@
 static Ichar_dynarr *title_string_ichar_dynarr;
 
 
+
+extern const struct sized_memory_description gtk_frame_data_description;
+extern const struct sized_memory_description mswindows_frame_data_description;
+extern const struct sized_memory_description x_frame_data_description;
+
+static const struct memory_description frame_data_description_1 []= {
+#ifdef HAVE_GTK
+  { XD_STRUCT_PTR, gtk_console, 1, &gtk_frame_data_description},
+#endif
+#ifdef HAVE_MS_WINDOWS
+  { XD_STRUCT_PTR, mswindows_console, 1, &mswindows_frame_data_description},
+#endif
+#ifdef HAVE_X_WINDOWS
+  { XD_STRUCT_PTR, x_console, 1, &x_frame_data_description},
+#endif
+  { XD_END }
+};
+
+static const struct sized_memory_description frame_data_description = {
+  sizeof (void *), frame_data_description_1
+};
+
+extern const struct sized_memory_description expose_ignore_description;
+
+static const struct memory_description expose_ignore_description_1 [] = {
+  { XD_STRUCT_PTR, offsetof (struct expose_ignore, next),
+    1, &expose_ignore_description },
+  { XD_END }
+};
+
+const struct sized_memory_description expose_ignore_description = {
+  sizeof (struct expose_ignore),
+  expose_ignore_description_1
+};
+
+static const struct memory_description display_line_dynarr_pointer_description_1 []= {
+  { XD_STRUCT_PTR, 0, 1, &display_line_dynarr_description},
+  { XD_END }
+};
+
+static const struct sized_memory_description display_line_dynarr_pointer_description = {
+  sizeof (display_line_dynarr *), display_line_dynarr_pointer_description_1
+};
+
+static const struct memory_description frame_description [] = {
+  { XD_INT, offsetof (struct frame, frametype) },
+#define MARKED_SLOT(x) { XD_LISP_OBJECT, offsetof (struct frame, x) },
+#define MARKED_SLOT_ARRAY(slot, size) \
+  { XD_LISP_OBJECT_ARRAY, offsetof (struct frame, slot), size },
+#include "frameslots.h"
+
+  { XD_STRUCT_PTR, offsetof (struct frame, subwindow_exposures),
+    1, &expose_ignore_description },
+  { XD_STRUCT_PTR, offsetof (struct frame, subwindow_exposures_tail),
+    1, &expose_ignore_description },
+
+#ifdef HAVE_SCROLLBARS
+  { XD_LISP_OBJECT, offsetof (struct frame, sb_vcache) },
+  { XD_LISP_OBJECT, offsetof (struct frame, sb_hcache) },
+#endif /* HAVE_SCROLLBARS */
+
+  { XD_STRUCT_ARRAY, offsetof (struct frame, current_display_lines),
+    4, &display_line_dynarr_pointer_description },
+  { XD_STRUCT_ARRAY, offsetof (struct frame, desired_display_lines),
+    4, &display_line_dynarr_pointer_description },
+
+  { XD_STRUCT_PTR, offsetof (struct frame, framemeths), 1,
+    &console_methods_description },
+  { XD_UNION, offsetof (struct frame, frame_data), 
+    XD_INDIRECT (0, 0), &frame_data_description },
+  { XD_END }
+};
+
 static Lisp_Object
 mark_frame (Lisp_Object obj)
 {
   struct frame *f = XFRAME (obj);
 
-#define MARKED_SLOT(x) mark_object (f->x)
+#define MARKED_SLOT(x) mark_object (f->x);
 #include "frameslots.h"
 
   if (FRAME_LIVE_P (f)) /* device is nil for a dead frame */
@@ -175,23 +248,18 @@
   write_fmt_string (printcharfun, " 0x%x>", frm->header.uid);
 }
 
-#ifdef USE_KKCC
 DEFINE_LRECORD_IMPLEMENTATION ("frame", frame,
 			       0, /*dumpable-flag*/
-                               mark_frame, print_frame, 0, 0, 0, 0,
+                               mark_frame, print_frame, 0, 0, 0,
+			       frame_description,
 			       struct frame);
-#else /* not USE_KKCC */
-DEFINE_LRECORD_IMPLEMENTATION ("frame", frame,
-                               mark_frame, print_frame, 0, 0, 0, 0,
-			       struct frame);
-#endif /* not USE_KKCC */
 
 static void
 nuke_all_frame_slots (struct frame *f)
 {
   zero_lcrecord (f);
 
-#define MARKED_SLOT(x)	f->x = Qnil
+#define MARKED_SLOT(x)	f->x = Qnil;
 #include "frameslots.h"
 }
 
@@ -212,6 +280,7 @@
 
   f->device = device;
   f->framemeths = XDEVICE (device)->devmeths;
+  f->frametype = get_console_variant (XDEVICE_TYPE (device));
   f->buffer_alist = Fcopy_sequence (Vbuffer_alist);
 
   root_window = allocate_window ();
@@ -584,9 +653,9 @@
   if (!UNBOUNDP (symbol_function (XSYMBOL (Qcustom_initialize_frame))))
     call1 (Qcustom_initialize_frame, frame);
 
+  UNGCPRO;
   unbind_to (speccount);
 
-  UNGCPRO;
   return frame;
 }
 
@@ -1749,6 +1818,7 @@
 
   nuke_all_frame_slots (f);
   f->framemeths = dead_console_methods;
+  f->frametype = dead_console;
 
   note_object_deleted (frame);
 
@@ -1919,17 +1989,10 @@
   if (mouse_pixel_position_1 (d, &frame, &intx, &inty))
     {
       Lisp_Object event = Fmake_event (Qnil, Qnil);
-#ifdef USE_KKCC
       XSET_EVENT_TYPE (event, pointer_motion_event);
       XSET_EVENT_CHANNEL (event, frame);
-      XSET_MOTION_DATA_X (XEVENT_DATA (event), intx);
-      XSET_MOTION_DATA_Y (XEVENT_DATA (event), inty);
-#else /* not USE_KKCC */
-      XEVENT (event)->event_type = pointer_motion_event;
-      XEVENT (event)->channel = frame;
-      XEVENT (event)->event.motion.x = intx;
-      XEVENT (event)->event.motion.y = inty;
-#endif /* not USE_KKCC */
+      XSET_EVENT_MOTION_X (event, intx);
+      XSET_EVENT_MOTION_Y (event, inty);
       return event;
     }
   else
--- a/src/frameslots.h	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/frameslots.h	Sun Jan 12 11:08:22 2003 +0000
@@ -1,6 +1,6 @@
 /* Definitions of marked slots in frames
    Copyright (C) 1988, 1992, 1993, 1994 Free Software Foundation, Inc.
-   Copyright (C) 1996 Ben Wing.
+   Copyright (C) 1996, 2002 Ben Wing.
 
 This file is part of XEmacs.
 
@@ -30,6 +30,7 @@
    definition, you also need to define FRAME_SLOT_DECLARATION.  No need to
    undefine either value; that happens automatically.  */
 
+#ifndef MARKED_SLOT_ARRAY
 #ifdef FRAME_SLOT_DECLARATION
 #define MARKED_SLOT_ARRAY(slot, size) MARKED_SLOT(slot[size])
 #else
@@ -37,13 +38,14 @@
     int mslotidx;					\
     for (mslotidx = 0; mslotidx < size; mslotidx++)	\
       {							\
-	MARKED_SLOT (slot[mslotidx]);			\
+	MARKED_SLOT (slot[mslotidx])			\
       }							\
   } while (0);
 #endif
+#endif /* not MARKED_SLOT_ARRAY */
 
   /* device frame belongs to. */
-  MARKED_SLOT (device);
+  MARKED_SLOT (device)
 
   /* Name of this frame: a Lisp string.
      NOT the same as the frame's title, even though FSF bogusly
@@ -51,7 +53,7 @@
      and lookup purposes and is something you can count on having
      a specific value, while the frame's title may vary depending
      on the user's choice of `frame-title-format'. */
-  MARKED_SLOT (name);
+  MARKED_SLOT (name)
 
   /* The frame which should receive keystrokes that occur in this
      frame, or nil if they should go to the frame itself.  This is
@@ -64,108 +66,108 @@
      to shift from one frame to the other, any redirections to the
      original frame are shifted to the newly selected frame; if
      focus_frame is nil, Fselect_frame will leave it alone.  */
-  MARKED_SLOT (focus_frame);
+  MARKED_SLOT (focus_frame)
 
   /* This frame's root window.  Every frame has one.
      If the frame has only a minibuffer window, this is it.
      Otherwise, if the frame has a minibuffer window, this is its sibling.  */
-  MARKED_SLOT (root_window);
+  MARKED_SLOT (root_window)
 
   /* This frame's selected window.
      Each frame has its own window hierarchy
      and one of the windows in it is selected within the frame.
      The selected window of the selected frame is Emacs's selected window.  */
-  MARKED_SLOT (selected_window);
+  MARKED_SLOT (selected_window)
 
   /* This frame's minibuffer window.
      Most frames have their own minibuffer windows,
      but only the selected frame's minibuffer window
      can actually appear to exist.  */
-  MARKED_SLOT (minibuffer_window);
+  MARKED_SLOT (minibuffer_window)
 
   /* The most recently selected nonminibuf window.
      This is used by things like the toolbar code, which doesn't
      want the toolbar to change when moving to the minibuffer.
      This will only be a minibuf window if we are a minibuf-only
      frame. */
-  MARKED_SLOT (last_nonminibuf_window);
+  MARKED_SLOT (last_nonminibuf_window)
 
   /* This frame's root window mirror.  This structure exactly mirrors
      the frame's window structure but contains only pointers to the
      display structures. */
-  MARKED_SLOT (root_mirror);
+  MARKED_SLOT (root_mirror)
 
   /* frame property list */
-  MARKED_SLOT (plist);
+  MARKED_SLOT (plist)
 
   /* buffer_alist at last redisplay. */
-  MARKED_SLOT (old_buffer_alist);
+  MARKED_SLOT (old_buffer_alist)
 
   /* A copy of the global Vbuffer_list, to maintain a per-frame buffer
      ordering.  The Vbuffer_list variable and the buffer_list slot of each
      frame contain exactly the same data, just in different orders.  */
-  MARKED_SLOT (buffer_alist);
+  MARKED_SLOT (buffer_alist)
 
   /* Predicate for selecting buffers for other-buffer.  */
-  MARKED_SLOT (buffer_predicate);
+  MARKED_SLOT (buffer_predicate)
 
   /* The current mouse pointer for the frame.  This is set by calling
      `set-frame-pointer'. */
-  MARKED_SLOT (pointer);
+  MARKED_SLOT (pointer)
 
   /* The current icon for the frame. */
-  MARKED_SLOT (icon);
+  MARKED_SLOT (icon)
 
 #ifdef HAVE_MENUBARS
   /* Vector representing the menubar currently displayed.  See menubar-x.c. */
-  MARKED_SLOT (menubar_data);
+  MARKED_SLOT (menubar_data)
 #endif
 
   /* specifier values cached in the struct frame: */
 
 #ifdef HAVE_MENUBARS
-  MARKED_SLOT (menubar_visible_p);
+  MARKED_SLOT (menubar_visible_p)
 #endif
 
 #ifdef HAVE_SCROLLBARS
   /* Width and height of the scrollbars. */
-  MARKED_SLOT (scrollbar_width);
-  MARKED_SLOT (scrollbar_height);
+  MARKED_SLOT (scrollbar_width)
+  MARKED_SLOT (scrollbar_height)
   /* Whether the scrollbars are visible */
-  MARKED_SLOT (horizontal_scrollbar_visible_p);
-  MARKED_SLOT (vertical_scrollbar_visible_p);
+  MARKED_SLOT (horizontal_scrollbar_visible_p)
+  MARKED_SLOT (vertical_scrollbar_visible_p)
   /* Scrollbars location */
-  MARKED_SLOT (scrollbar_on_left_p);
-  MARKED_SLOT (scrollbar_on_top_p);
+  MARKED_SLOT (scrollbar_on_left_p)
+  MARKED_SLOT (scrollbar_on_top_p)
 #endif
 
 #ifdef HAVE_TOOLBARS
   /* The following three don't really need to be cached except
      that we need to know when they've changed. */
-  MARKED_SLOT (default_toolbar_width);
-  MARKED_SLOT (default_toolbar_height);
-  MARKED_SLOT (default_toolbar_visible_p);
-  MARKED_SLOT (default_toolbar_border_width);
-  MARKED_SLOT (toolbar_shadow_thickness);
+  MARKED_SLOT (default_toolbar_width)
+  MARKED_SLOT (default_toolbar_height)
+  MARKED_SLOT (default_toolbar_visible_p)
+  MARKED_SLOT (default_toolbar_border_width)
+  MARKED_SLOT (toolbar_shadow_thickness)
 
   /* List of toolbar buttons of current toolbars */
-  MARKED_SLOT_ARRAY (toolbar_buttons, 4);
+  MARKED_SLOT_ARRAY (toolbar_buttons, 4)
   /* Size of the toolbars.  The frame-local toolbar space is
      subtracted before the windows are arranged.  Window and buffer
      local toolbars overlay their windows. */
-  MARKED_SLOT_ARRAY (toolbar_size, 4);
+  MARKED_SLOT_ARRAY (toolbar_size, 4)
   /* Visibility of the toolbars.  This acts as a valve for toolbar_size. */
-  MARKED_SLOT_ARRAY (toolbar_visible_p, 4);
+  MARKED_SLOT_ARRAY (toolbar_visible_p, 4)
   /* Thickness of the border around the toolbar. */
-  MARKED_SLOT_ARRAY (toolbar_border_width, 4);
+  MARKED_SLOT_ARRAY (toolbar_border_width, 4)
 #endif
 
 /* Cache of subwindow instances for this frame */
-  MARKED_SLOT (subwindow_instance_cache);
+  MARKED_SLOT (subwindow_instance_cache)
 
   /* Possible frame-local default for outside margin widths. */
-  MARKED_SLOT (left_margin_width);
-  MARKED_SLOT (right_margin_width);
+  MARKED_SLOT (left_margin_width)
+  MARKED_SLOT (right_margin_width)
 
 #undef MARKED_SLOT
 #undef MARKED_SLOT_ARRAY
--- a/src/free-hook.c	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/free-hook.c	Sun Jan 12 11:08:22 2003 +0000
@@ -26,9 +26,9 @@
    * Trying to free a pointer not returned by malloc.
    * Trying to realloc a pointer not returned by malloc.
 
-   In addition, every word of every block freed is set to
-   0xdeadbeef.  This causes many uses of freed storage to be
-   trapped or recognized.
+   In addition, every word of every block freed is set to 0xdeadbeef
+   (-559038737).  This causes many uses of freed storage to be trapped or
+   recognized.
 
    When you use this, the storage used by the last FREE_QUEUE_LIMIT
    calls to free() is not recycled.  When you call free for the Nth
--- a/src/glyphs-msw.c	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/glyphs-msw.c	Sun Jan 12 11:08:22 2003 +0000
@@ -2215,7 +2215,7 @@
 			      Lisp_Object instantiator,
 			      Lisp_Object pointer_fg, Lisp_Object pointer_bg,
 			      int dest_mask, Lisp_Object domain,
-			      const CIbyte *class, int flags, int exflags)
+			      const CIbyte *class_, int flags, int exflags)
 {
   /* this function can call lisp */
   Lisp_Image_Instance *ii = XIMAGE_INSTANCE (image_instance);
@@ -2263,7 +2263,7 @@
     gui_error ("window creation failed with code",
 	       make_int (GetLastError()));
 
-  C_STRING_TO_TSTR (class, classext);
+  C_STRING_TO_TSTR (class_, classext);
 
   if ((wnd = qxeCreateWindowEx (exflags /* | WS_EX_NOPARENTNOTIFY*/,
 				classext,
--- a/src/glyphs-x.c	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/glyphs-x.c	Sun Jan 12 11:08:22 2003 +0000
@@ -24,7 +24,7 @@
 
 /* Synched up with: Not in FSF. */
 
-/* 7-8-00 This file is more or less Mule-ized in my Mule workspace. */
+/* 7-8-00 This file is more or less Mule-ized. */
 
 /* Original author: Jamie Zawinski for 19.8
    font-truename stuff added by Jamie Zawinski for 19.10
@@ -187,14 +187,14 @@
   vis = DEVICE_X_VISUAL (XDEVICE(device));
   depth = DEVICE_X_DEPTH(XDEVICE(device));
 
-  if (vis->class == GrayScale || vis->class == StaticColor ||
-      vis->class == StaticGray)
+  if (vis->X_CLASSFIELD == GrayScale || vis->X_CLASSFIELD == StaticColor ||
+      vis->X_CLASSFIELD == StaticGray)
     {
       /* #### Implement me!!! */
       return NULL;
     }
 
-  if (vis->class == PseudoColor)
+  if (vis->X_CLASSFIELD == PseudoColor)
     {
       /* Quantize the image and get a histogram while we're at it.
 	 Do this first to save memory */
@@ -222,7 +222,7 @@
     }
   outimg->data = (char *) data;
 
-  if (vis->class == PseudoColor)
+  if (vis->X_CLASSFIELD == PseudoColor)
     {
       unsigned long pixarray[256];
       int pixcount, n;
--- a/src/glyphs.c	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/glyphs.c	Sun Jan 12 11:08:22 2003 +0000
@@ -406,7 +406,7 @@
   invalid_argument ("Unable to interpret glyph instantiator",
 		       data);
 
-  RETURN_NOT_REACHED (Qnil)
+  RETURN_NOT_REACHED (Qnil);
 }
 
 Lisp_Object
@@ -834,6 +834,77 @@
 
 Lisp_Object Qimage_instancep;
 
+/* %%#### KKCC: Don't yet handle the equivalent of setting the device field
+   of image instances w/dead devices to nil. */
+
+static const struct memory_description text_image_instance_description_1 [] = {
+  { XD_LISP_OBJECT, offsetof (struct text_image_instance, string) },
+  { XD_END }
+};
+
+static const struct sized_memory_description text_image_instance_description = {
+  sizeof (struct text_image_instance), text_image_instance_description_1
+};
+
+static const struct memory_description pixmap_image_instance_description_1 [] = {
+  { XD_LISP_OBJECT, offsetof (struct pixmap_image_instance, hotspot_x) },
+  { XD_LISP_OBJECT, offsetof (struct pixmap_image_instance, hotspot_x) },
+  { XD_LISP_OBJECT, offsetof (struct pixmap_image_instance, filename) },
+  { XD_LISP_OBJECT, offsetof (struct pixmap_image_instance, mask_filename) },
+  { XD_LISP_OBJECT, offsetof (struct pixmap_image_instance, fg) },
+  { XD_LISP_OBJECT, offsetof (struct pixmap_image_instance, bg) },
+  { XD_LISP_OBJECT, offsetof (struct pixmap_image_instance, auxdata) },
+  { XD_END }
+};
+
+static const struct sized_memory_description pixmap_image_instance_description = {
+  sizeof (struct pixmap_image_instance), pixmap_image_instance_description_1
+};
+
+static const struct memory_description subwindow_image_instance_description_1 [] = {
+  { XD_LISP_OBJECT, offsetof (struct subwindow_image_instance, face) },
+  { XD_LISP_OBJECT, offsetof (struct subwindow_image_instance, type) },
+  { XD_LISP_OBJECT, offsetof (struct subwindow_image_instance, props) },
+  { XD_LISP_OBJECT, offsetof (struct subwindow_image_instance, items) },
+  { XD_LISP_OBJECT, offsetof (struct subwindow_image_instance, pending_items) },
+  { XD_LISP_OBJECT, offsetof (struct subwindow_image_instance, children) },
+  { XD_LISP_OBJECT, offsetof (struct subwindow_image_instance, width) },
+  { XD_LISP_OBJECT, offsetof (struct subwindow_image_instance, height) },
+  { XD_END }
+};
+
+static const struct sized_memory_description subwindow_image_instance_description = {
+  sizeof (struct subwindow_image_instance), subwindow_image_instance_description_1
+};
+
+static const struct memory_description image_instance_data_description_1 [] = {
+  { XD_STRUCT_ARRAY, IMAGE_TEXT,
+    1, &text_image_instance_description },
+  { XD_STRUCT_ARRAY, IMAGE_MONO_PIXMAP,
+    1, &pixmap_image_instance_description },
+  { XD_STRUCT_ARRAY, IMAGE_COLOR_PIXMAP,
+    1, &pixmap_image_instance_description },
+  { XD_STRUCT_ARRAY, IMAGE_WIDGET,
+    1, &subwindow_image_instance_description },
+  { XD_END }
+};
+
+static const struct sized_memory_description image_instance_data_description = {
+  0, image_instance_data_description_1
+};
+
+static const struct memory_description image_instance_description[] = {
+  { XD_INT, offsetof (struct Lisp_Image_Instance, type) },
+  { XD_LISP_OBJECT, offsetof (Lisp_Image_Instance, domain) },
+  { XD_LISP_OBJECT, offsetof (Lisp_Image_Instance, device) },
+  { XD_LISP_OBJECT, offsetof (Lisp_Image_Instance, name) },
+  { XD_LISP_OBJECT, offsetof (Lisp_Image_Instance, parent) },
+  { XD_LISP_OBJECT, offsetof (Lisp_Image_Instance, instantiator) },
+  { XD_UNION, offsetof (struct Lisp_Image_Instance, u), 
+    XD_INDIRECT (0, 0), &image_instance_data_description },
+  { XD_END }
+};
+
 static Lisp_Object
 mark_image_instance (Lisp_Object obj)
 {
@@ -849,7 +920,7 @@
 
   mark_object (i->name);
   mark_object (i->instantiator);
-  /* Is this legal in marking? We may get in the situation where the
+  /* #### Is this legal in marking? We may get in the situation where the
      domain has been deleted - making the instance unusable. It seems
      better to remove the domain so that it can be finalized. */
   if (!DOMAIN_LIVE_P (i->domain))
@@ -1229,20 +1300,13 @@
 		 0));
 }
 
-#ifdef USE_KKCC
 DEFINE_LRECORD_IMPLEMENTATION ("image-instance", image_instance,
 			       0, /*dumpable-flag*/
 			       mark_image_instance, print_image_instance,
 			       finalize_image_instance, image_instance_equal,
-			       image_instance_hash, 0,
+			       image_instance_hash,
+			       image_instance_description,
 			       Lisp_Image_Instance);
-#else /* not USE_KKCC */
-DEFINE_LRECORD_IMPLEMENTATION ("image-instance", image_instance,
-			       mark_image_instance, print_image_instance,
-			       finalize_image_instance, image_instance_equal,
-			       image_instance_hash, 0,
-			       Lisp_Image_Instance);
-#endif /* not USE_KKCC */
 
 static Lisp_Object
 allocate_image_instance (Lisp_Object governing_domain, Lisp_Object parent,
@@ -1252,7 +1316,6 @@
     alloc_lcrecord_type (Lisp_Image_Instance, &lrecord_image_instance);
   Lisp_Object val;
 
-  zero_lcrecord (lp);
   /* It's not possible to simply keep a record of the domain in which
      the instance was instantiated. This is because caching may mean
      that the domain becomes invalid but the instance remains
@@ -1267,16 +1330,12 @@
   lp->device = DOMAIN_DEVICE (governing_domain);
   lp->type = IMAGE_NOTHING;
   lp->name = Qnil;
-  lp->x_offset = 0;
-  lp->y_offset = 0;
   lp->width = IMAGE_UNSPECIFIED_GEOMETRY;
-  lp->margin_width = 0;
   lp->height = IMAGE_UNSPECIFIED_GEOMETRY;
   lp->parent = parent;
   lp->instantiator = instantiator;
   /* So that layouts get done. */
   lp->layout_changed = 1;
-  lp->initialized = 0;
 
   val = wrap_image_instance (lp);
   MARK_GLYPHS_CHANGED;
@@ -3056,7 +3115,13 @@
  *                         Image Specifier Object                           *
  ****************************************************************************/
 
-DEFINE_SPECIFIER_TYPE (image);
+static const struct memory_description image_specifier_description[] = {
+  { XD_LISP_OBJECT, offsetof (struct image_specifier, attachee) },
+  { XD_LISP_OBJECT, offsetof (struct image_specifier, attachee_property) },
+  { XD_END }
+};
+
+DEFINE_SPECIFIER_TYPE_WITH_DATA (image);
 
 static void
 image_create (Lisp_Object obj)
@@ -3683,7 +3748,7 @@
   return result;
 }
 
-static const struct lrecord_description glyph_description[] = {
+static const struct memory_description glyph_description[] = {
   { XD_LISP_OBJECT, offsetof (Lisp_Glyph, image) },
   { XD_LISP_OBJECT, offsetof (Lisp_Glyph, contrib_p) },
   { XD_LISP_OBJECT, offsetof (Lisp_Glyph, baseline) },
@@ -3692,22 +3757,14 @@
   { XD_END }
 };
 
-#ifdef USE_KKCC 
 DEFINE_LRECORD_IMPLEMENTATION_WITH_PROPS ("glyph", glyph,
 					  1, /*dumpable-flag*/
 					  mark_glyph, print_glyph, 0,
-					  glyph_equal, glyph_hash, glyph_description,
+					  glyph_equal, glyph_hash,
+					  glyph_description,
 					  glyph_getprop, glyph_putprop,
 					  glyph_remprop, glyph_plist,
 					  Lisp_Glyph);
-#else /* not USE_KKCC */
-DEFINE_LRECORD_IMPLEMENTATION_WITH_PROPS ("glyph", glyph,
-					  mark_glyph, print_glyph, 0,
-					  glyph_equal, glyph_hash, glyph_description,
-					  glyph_getprop, glyph_putprop,
-					  glyph_remprop, glyph_plist,
-					  Lisp_Glyph);
-#endif /* not USE_KKCC */
 
 Lisp_Object
 allocate_glyph (enum glyph_type type,
@@ -4674,10 +4731,9 @@
 
   ERROR_CHECK_IMAGE_INSTANCE (subwindow);
 
-  if (!image_instance_type_to_mask (IMAGE_INSTANCE_TYPE (ii))
-      & (IMAGE_WIDGET_MASK | IMAGE_SUBWINDOW_MASK)
-      ||
-      !IMAGE_INSTANCE_SUBWINDOW_DISPLAYEDP (ii))
+  if (!(image_instance_type_to_mask (IMAGE_INSTANCE_TYPE (ii))
+	& (IMAGE_WIDGET_MASK | IMAGE_SUBWINDOW_MASK))
+      || !IMAGE_INSTANCE_SUBWINDOW_DISPLAYEDP (ii))
     return;
 
 #ifdef DEBUG_WIDGETS
@@ -4706,8 +4762,8 @@
 
   ERROR_CHECK_IMAGE_INSTANCE (subwindow);
 
-  if (!image_instance_type_to_mask (IMAGE_INSTANCE_TYPE (ii))
-      & (IMAGE_WIDGET_MASK | IMAGE_SUBWINDOW_MASK))
+  if (!(image_instance_type_to_mask (IMAGE_INSTANCE_TYPE (ii))
+	& (IMAGE_WIDGET_MASK | IMAGE_SUBWINDOW_MASK)))
     return;
 
 #ifdef DEBUG_WIDGETS
@@ -5119,7 +5175,7 @@
   /* Qbuffer defined in general.c. */
   /* Qpointer defined above */
 
-  /* Unfortunately, timeout handlers must be lisp functions. This is
+  /* Unfortunately, timeout handlers must be lisp functions.  This is
      for animated glyphs. */
   DEFSYMBOL (Qglyph_animated_timeout_handler);
   DEFSUBR (Fglyph_animated_timeout_handler);
@@ -5128,12 +5184,6 @@
   DEFERROR_STANDARD (Qimage_conversion_error, Qconversion_error);
 }
 
-static const struct lrecord_description image_specifier_description[] = {
-  { XD_LISP_OBJECT, specifier_data_offset + offsetof (struct image_specifier, attachee) },
-  { XD_LISP_OBJECT, specifier_data_offset + offsetof (struct image_specifier, attachee_property) },
-  { XD_END }
-};
-
 void
 specifier_type_create_image (void)
 {
@@ -5157,49 +5207,49 @@
 }
 
 
-static const struct lrecord_description iike_description_1[] = {
+static const struct memory_description iike_description_1[] = {
   { XD_LISP_OBJECT, offsetof (ii_keyword_entry, keyword) },
   { XD_END }
 };
 
-static const struct struct_description iike_description = {
+static const struct sized_memory_description iike_description = {
   sizeof (ii_keyword_entry),
   iike_description_1
 };
 
-static const struct lrecord_description iiked_description_1[] = {
+static const struct memory_description iiked_description_1[] = {
   XD_DYNARR_DESC (ii_keyword_entry_dynarr, &iike_description),
   { XD_END }
 };
 
-static const struct struct_description iiked_description = {
+static const struct sized_memory_description iiked_description = {
   sizeof (ii_keyword_entry_dynarr),
   iiked_description_1
 };
 
-static const struct lrecord_description iife_description_1[] = {
+static const struct memory_description iife_description_1[] = {
   { XD_LISP_OBJECT, offsetof (image_instantiator_format_entry, symbol) },
   { XD_LISP_OBJECT, offsetof (image_instantiator_format_entry, device) },
   { XD_STRUCT_PTR,  offsetof (image_instantiator_format_entry, meths),  1, &iim_description },
   { XD_END }
 };
 
-static const struct struct_description iife_description = {
+static const struct sized_memory_description iife_description = {
   sizeof (image_instantiator_format_entry),
   iife_description_1
 };
 
-static const struct lrecord_description iifed_description_1[] = {
+static const struct memory_description iifed_description_1[] = {
   XD_DYNARR_DESC (image_instantiator_format_entry_dynarr, &iife_description),
   { XD_END }
 };
 
-static const struct struct_description iifed_description = {
+static const struct sized_memory_description iifed_description = {
   sizeof (image_instantiator_format_entry_dynarr),
   iifed_description_1
 };
 
-static const struct lrecord_description iim_description_1[] = {
+static const struct memory_description iim_description_1[] = {
   { XD_LISP_OBJECT, offsetof (struct image_instantiator_methods, symbol) },
   { XD_LISP_OBJECT, offsetof (struct image_instantiator_methods, device) },
   { XD_STRUCT_PTR,  offsetof (struct image_instantiator_methods, keywords), 1, &iiked_description },
@@ -5207,7 +5257,7 @@
   { XD_END }
 };
 
-const struct struct_description iim_description = {
+const struct sized_memory_description iim_description = {
   sizeof (struct image_instantiator_methods),
   iim_description_1
 };
--- a/src/glyphs.h	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/glyphs.h	Sun Jan 12 11:08:22 2003 +0000
@@ -84,7 +84,7 @@
   Dynarr_declare (ii_keyword_entry);
 } ii_keyword_entry_dynarr;
 
-extern const struct struct_description iim_description;
+extern const struct sized_memory_description iim_description;
 
 enum image_instance_geometry
 {
@@ -537,6 +537,61 @@
     x = wrong_type_argument (Qwidget_image_instance_p, (x));	\
 } while (0)
 
+struct text_image_instance
+{
+  int descent;
+  Lisp_Object string;
+};
+
+struct pixmap_image_instance
+{ /* used for pointers as well */
+  int depth;
+  int slice, maxslice, timeout;
+  Lisp_Object hotspot_x, hotspot_y; /* integer or Qnil */
+  Lisp_Object filename;	 /* string or Qnil */
+  Lisp_Object mask_filename; /* string or Qnil */
+  Lisp_Object fg, bg; /* foreground and background colors,
+			 if this is a colorized mono-pixmap
+			 or a pointer */
+  Lisp_Object auxdata;    /* list or Qnil: any additional data
+			     to be seen from lisp */
+  void* mask;		/* mask that can be seen from all windowing systems */
+};
+
+struct subwindow_image_instance
+{
+  void* subwindow;	/* specific devices can use this as necessary */
+  struct
+  {				/* We need these so we can do without
+				   subwindow_cachel */
+    int x, y;
+    int width, height;
+  } display_data;
+  unsigned int being_displayed : 1; /* used to detect when needs
+				       to be unmapped */
+  unsigned int v_resize : 1;	/* Whether the vsize is allowed to change. */
+  unsigned int h_resize : 1;	/* Whether the hsize is allowed to change. */
+  unsigned int orientation : 1; /* Vertical or horizontal. */
+  unsigned int h_justification : 2; /* left, right or center. */
+  unsigned int v_justification : 2; /* top, bottom or center. */
+  /* Face for colors and font. We specify this here because we
+     want people to be able to put :face in the instantiator
+     spec. Using glyph-face is more inconvenient, although more
+     general. */
+  Lisp_Object face;
+  Lisp_Object type;
+  Lisp_Object props;	/* properties or border*/
+  Lisp_Object items;	/* a list of displayed gui_items */
+  Lisp_Object pending_items; /* gui_items that should be displayed */
+  Lisp_Object children;	/* a list of children */
+  Lisp_Object width;	/* dynamic width spec. */
+  Lisp_Object height;	/* dynamic height spec. */
+  /* Change flags to augment dirty. */
+  unsigned int face_changed : 1;
+  unsigned int items_changed : 1;
+  unsigned int action_occurred : 1;
+};
+
 struct Lisp_Image_Instance
 {
   struct lcrecord_header header;
@@ -566,58 +621,9 @@
 
   union
   {
-    struct
-    {
-      int descent;
-      Lisp_Object string;
-    } text;
-    struct
-    {
-      int depth;
-      int slice, maxslice, timeout;
-      Lisp_Object hotspot_x, hotspot_y; /* integer or Qnil */
-      Lisp_Object filename;	 /* string or Qnil */
-      Lisp_Object mask_filename; /* string or Qnil */
-      Lisp_Object fg, bg; /* foreground and background colors,
-			     if this is a colorized mono-pixmap
-			     or a pointer */
-      Lisp_Object auxdata;    /* list or Qnil: any additional data
-				 to be seen from lisp */
-      void* mask;		/* mask that can be seen from all windowing systems */
-    } pixmap; /* used for pointers as well */
-    struct
-    {
-      void* subwindow;		/* specific devices can use this as necessary */
-      struct
-      {				/* We need these so we can do without
-				   subwindow_cachel */
-	int x, y;
-	int width, height;
-      } display_data;
-      unsigned int being_displayed : 1; /* used to detect when needs
-					   to be unmapped */
-      unsigned int v_resize : 1;	/* Whether the vsize is allowed to change. */
-      unsigned int h_resize : 1;	/* Whether the hsize is allowed to change. */
-      unsigned int orientation : 1; /* Vertical or horizontal. */
-      unsigned int h_justification : 2; /* left, right or center. */
-      unsigned int v_justification : 2; /* top, bottom or center. */
-      /* Face for colors and font. We specify this here because we
-	 want people to be able to put :face in the instantiator
-	 spec. Using glyph-face is more inconvenient, although more
-	 general. */
-      Lisp_Object face;
-      Lisp_Object type;
-      Lisp_Object props;	/* properties or border*/
-      Lisp_Object items;	/* a list of displayed gui_items */
-      Lisp_Object pending_items; /* gui_items that should be displayed */
-      Lisp_Object children;	/* a list of children */
-      Lisp_Object width;	/* dynamic width spec. */
-      Lisp_Object height;	/* dynamic height spec. */
-      /* Change flags to augment dirty. */
-      unsigned int face_changed : 1;
-      unsigned int items_changed : 1;
-      unsigned int action_occurred : 1;
-    } subwindow;
+    struct text_image_instance text;
+    struct pixmap_image_instance pixmap; /* used for pointers as well */
+    struct subwindow_image_instance subwindow;
   } u;
 
   /* console-type- and image-type-specific data */
--- a/src/gmalloc.c	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/gmalloc.c	Sun Jan 12 11:08:22 2003 +0000
@@ -78,7 +78,7 @@
 #endif	/* _MALLOC_INTERNAL.  */
 
 
-#ifdef	__cplusplus
+#if defined (__cplusplus) && !defined (emacs)
 extern "C"
 {
 #endif
@@ -276,7 +276,7 @@
 /* Pick up the current statistics. */
 extern struct mstats mstats __P ((void));
 
-#ifdef	__cplusplus
+#if defined (__cplusplus) && !defined (emacs)
 }
 #endif
 
--- a/src/gpmevent.c	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/gpmevent.c	Sun Jan 12 11:08:22 2003 +0000
@@ -158,35 +158,19 @@
     {
     case GPM_DOWN:
     case GPM_UP:
-#ifdef USE_KKCC
       SET_EVENT_TYPE (event,
 	(ev.type & GPM_DOWN) ? button_press_event : button_release_event);
-      XSET_BUTTON_DATA_X (EVENT_DATA (event), ev.x);
-      XSET_BUTTON_DATA_Y (EVENT_DATA (event), ev.y);
-      XSET_BUTTON_DATA_BUTTON (EVENT_DATA (event), button);
-      XSET_BUTTON_DATA_MODIFIERS (EVENT_DATA (event), modifiers);
-#else /* not USE_KKCC */
-      event->event_type =
-	(ev.type & GPM_DOWN) ? button_press_event : button_release_event;
-      event->event.button.x         = ev.x;
-      event->event.button.y         = ev.y;
-      event->event.button.button    = button;
-      event->event.button.modifiers = modifiers;
-#endif /* not USE_KKCC */
+      SET_EVENT_BUTTON_X (event, ev.x);
+      SET_EVENT_BUTTON_Y (event, ev.y);
+      SET_EVENT_BUTTON_BUTTON (event, button);
+      SET_EVENT_BUTTON_MODIFIERS (event, modifiers);
       break;
     case GPM_MOVE:
     case GPM_DRAG:
-#ifdef USE_KKCC
       SET_EVENT_TYPE (event, pointer_motion_event);
-      XSET_MOTION_DATA_X (EVENT_DATA (event), ev.x);
-      XSET_MOTION_DATA_Y (EVENT_DATA (event), ev.y);
-      XSET_MOTION_DATA_MODIFIERS (EVENT_DATA (event), modifiers);
-#else /* not USE_KKCC */
-      event->event_type             = pointer_motion_event;
-      event->event.motion.x         = ev.x;
-      event->event.motion.y         = ev.y;
-      event->event.motion.modifiers = modifiers;
-#endif /* not USE_KKCC */
+      SET_EVENT_MOTION_X (event, ev.x);
+      SET_EVENT_MOTION_Y (event, ev.y);
+      SET_EVENT_MOTION_MODIFIERS (event, modifiers);
     default:
       /* This will never happen */
       break;
@@ -451,11 +435,7 @@
 	   ** William M. Perry - Nov 9, 1999
 	   */
 
-#ifdef USE_KKCC
-	  Gpm_DrawPointer (XMOTION_DATA_X (EVENT_DATA (event)),XMOTION_DATA_Y (EVENT_DATA (event)), fd);
-#else /* not USE_KKCC */
-	  Gpm_DrawPointer (event->event.motion.x,event->event.motion.y, fd);
-#endif /* not USE_KKCC */
+	  Gpm_DrawPointer (EVENT_MOTION_X (event),EVENT_MOTION_Y (event), fd);
 	}
 
       return;
@@ -633,8 +613,8 @@
   gpm_event_queue_tail = Qnil;
   staticpro (&gpm_event_queue);
   staticpro (&gpm_event_queue_tail);
-  dump_add_root_object (&gpm_event_queue);
-  dump_add_root_object (&gpm_event_queue_tail);
+  dump_add_root_lisp_object (&gpm_event_queue);
+  dump_add_root_lisp_object (&gpm_event_queue_tail);
 }
 
 void syms_of_gpmevent (void)
--- a/src/gtk-glue.c	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/gtk-glue.c	Sun Jan 12 11:08:22 2003 +0000
@@ -230,36 +230,38 @@
 static Lisp_Object
 gdk_event_to_emacs_event(GdkEvent *ev)
 {
-  Lisp_Object emacs_event = Qnil;
+  Lisp_Object event = Qnil;
 
   if (ev)
     {
-      emacs_event = Fmake_event (Qnil, Qnil);  
-      if (!gtk_event_to_emacs_event (NULL, ev, XEVENT (emacs_event)))
+      Lisp_Event *emacs_event;
+
+      event = Fmake_event (Qnil, Qnil);
+      emacs_event = XEVENT (event);
+
+      if (!gtk_event_to_emacs_event (NULL, ev, emacs_event))
 	{
 	  /* We need to handle a few more cases than the normal event
 	  ** loop does.  Mainly the double/triple click events.
 	  */
 	  if ((ev->type == GDK_2BUTTON_PRESS) || (ev->type == GDK_3BUTTON_PRESS))
 	    {
-	      struct Lisp_Event *le = XEVENT (emacs_event);
-
-	      le->event_type = misc_user_event;
-	      le->event.misc.button = ev->button.button;
-	      le->event.misc.modifiers = 0;
-	      le->event.misc.x = ev->button.x;
-	      le->event.misc.y = ev->button.y;
+	      set_event_type (emacs_event, misc_user_event);
+	      SET_EVENT_MISC_USER_BUTTON (emacs_event, ev->button.button);
+	      SET_EVENT_MISC_USER_MODIFIERS (emacs_event, 0);
+	      SET_EVENT_MISC_USER_X (emacs_event, ev->button.x);
+	      SET_EVENT_MISC_USER_Y (emacs_event, ev->button.y);
 	      if (ev->type == GDK_2BUTTON_PRESS)
-		le->event.misc.function = intern ("double-click");
+		SET_EVENT_MISC_USER_FUNCTION (emacs_event, intern ("double-click"));
 	      else
-		le->event.misc.function = intern ("triple-click");
+		SET_EVENT_MISC_USER_FUNCTION (emacs_event, intern ("triple-click"));
 	    }
 	  else
 	    {
-	      Fdeallocate_event (emacs_event);
-	      emacs_event = Qnil;
+	      Fdeallocate_event (event);
+	      event = Qnil;
 	    }
 	}
     }
-  return (emacs_event);
+  return (event);
 }
--- a/src/gtk-xemacs.c	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/gtk-xemacs.c	Sun Jan 12 11:08:22 2003 +0000
@@ -65,11 +65,11 @@
 static GtkWidgetClass *parent_class;
 
 static void
-gtk_xemacs_class_init (GtkXEmacsClass *class)
+gtk_xemacs_class_init (GtkXEmacsClass *class_)
 {
   GtkWidgetClass *widget_class;
 
-  widget_class = (GtkWidgetClass*) class;
+  widget_class = (GtkWidgetClass*) class_;
   parent_class = (GtkWidgetClass *) gtk_type_class (gtk_fixed_get_type ());
 
   widget_class->size_allocate = gtk_xemacs_size_allocate;
--- a/src/gui-msw.c	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/gui-msw.c	Sun Jan 12 11:08:22 2003 +0000
@@ -1,5 +1,6 @@
 /* mswindows GUI code. (menubars, scrollbars, toolbars, dialogs)
    Copyright (C) 1998 Andy Piper.
+   Copyright (C) 2002 Ben Wing.
 
 This file is part of XEmacs.
 
@@ -64,21 +65,12 @@
     {
       event = Fmake_event (Qnil, Qnil);
 
-#ifdef USE_KKCC
       XSET_EVENT_TYPE (event, misc_user_event);
       XSET_EVENT_CHANNEL (event, frame);
       XSET_EVENT_TIMESTAMP (event, GetTickCount());
-      XSET_MISC_USER_DATA_FUNCTION (XEVENT_DATA (event), Qeval);
-      XSET_MISC_USER_DATA_OBJECT (XEVENT_DATA (event),
+      XSET_EVENT_MISC_USER_FUNCTION (event, Qeval);
+      XSET_EVENT_MISC_USER_OBJECT (event,
 			     list4 (Qfuncall, callback_ex, image_instance, event));
-#else /* not USE_KKCC */
-      XEVENT (event)->event_type = misc_user_event;
-      XEVENT (event)->channel = frame;
-      XEVENT (event)->timestamp = GetTickCount ();
-      XEVENT (event)->event.eval.function = Qeval;
-      XEVENT (event)->event.eval.object =
-	list4 (Qfuncall, callback_ex, image_instance, event);
-#endif /* not USE_KKCC */
     }
   else if (NILP (callback) || UNBOUNDP (callback))
     return Qnil;
@@ -89,19 +81,11 @@
       event = Fmake_event (Qnil, Qnil);
 
       get_gui_callback (callback, &fn, &arg);
-#ifdef USE_KKCC
       XSET_EVENT_TYPE (event, misc_user_event);
       XSET_EVENT_CHANNEL (event, frame);
       XSET_EVENT_TIMESTAMP (event, GetTickCount());
-      XSET_MISC_USER_DATA_FUNCTION (XEVENT_DATA (event), fn);
-      XSET_MISC_USER_DATA_OBJECT (XEVENT_DATA (event), arg);
-#else /* not USE_KKCC */
-      XEVENT (event)->event_type = misc_user_event;
-      XEVENT (event)->channel = frame;
-      XEVENT (event)->timestamp = GetTickCount ();
-      XEVENT (event)->event.eval.function = fn;
-      XEVENT (event)->event.eval.object = arg;
-#endif /* not USE_KKCC */
+      XSET_EVENT_MISC_USER_FUNCTION (event, fn);
+      XSET_EVENT_MISC_USER_OBJECT (event, arg);
     }
 
   mswindows_enqueue_dispatch_event (event);
@@ -111,6 +95,85 @@
   return Qt;
 }
 
+/*
+ * Translate X accelerator syntax to win32 accelerator syntax.
+ * accel = (Ichar*) to receive the accelerator character
+ *         or NULL to suppress accelerators in the menu or dialog item.
+ *
+ * %% is replaced with %
+ * if accel is NULL:
+ *   %_ is removed.
+ * if accel is non-NULL:
+ *   %_ is replaced with &.
+ *   The accelerator character is passed back in *accel.
+ *   (If there is no accelerator, it will be added on the first character.)
+ *
+ */
+
+Lisp_Object
+mswindows_translate_menu_or_dialog_item (Lisp_Object item, Ichar *accel)
+{
+  Bytecount len = XSTRING_LENGTH (item);
+  Ibyte *it = (Ibyte *) ALLOCA (2 * len + 42), *ptr = it;
+
+  memcpy (ptr, XSTRING_DATA (item), len + 1);
+  if (accel)
+    *accel = '\0';
+
+  /* Escape '&' as '&&' */
+  
+  while ((ptr = (Ibyte *) memchr (ptr, '&', len - (ptr - it))) != NULL)
+    {
+      memmove (ptr + 1, ptr, (len - (ptr - it)) + 1);
+      len++;
+      ptr += 2;
+    }
+
+  /* Replace XEmacs accelerator '%_' with Windows accelerator '&'
+     and `%%' with `%'. */
+  ptr = it;
+  while ((ptr = (Ibyte *) memchr (ptr, '%', len - (ptr - it))) != NULL)
+    {
+      if (*(ptr + 1) == '_')
+	{
+	  if (accel)
+	    {
+	      *ptr = '&';
+	      if (!*accel)
+		*accel = DOWNCASE (0, itext_ichar (ptr + 2));
+	      memmove (ptr + 1, ptr + 2, len - (ptr - it + 2) + 1);
+	      len--;
+	    }
+	  else	/* Skip accelerator */
+	    {
+	      memmove (ptr, ptr + 2, len - (ptr - it + 2) + 1);
+	      len -= 2;
+	    }
+	}
+      else if (*(ptr + 1) == '%')
+	{
+	  memmove (ptr + 1, ptr + 2, len - (ptr - it + 2) + 1);
+	  len--;
+	  ptr++;
+	}
+      else	/* % on its own - shouldn't happen */
+	ptr++;
+    }
+
+  if (accel && !*accel)
+    {
+      /* Force a default accelerator */
+      ptr = it;
+      memmove (ptr + 1, ptr, len + 1);
+      *accel = DOWNCASE (0, itext_ichar (ptr + 1));
+      *ptr = '&';
+
+      len++;
+    }
+
+  return make_string (it, len);
+}
+
 void
 syms_of_gui_mswindows (void)
 {
--- a/src/gui-x.c	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/gui-x.c	Sun Jan 12 11:08:22 2003 +0000
@@ -63,54 +63,48 @@
 }
 
 
-static int
-mark_widget_value_mapper (widget_value *val, void *closure)
-{
-  Lisp_Object markee;
-  if (val->call_data)
-    {
-      markee = VOID_TO_LISP (val->call_data);
-      mark_object (markee);
-    }
-
-  if (val->accel)
-    {
-      markee = VOID_TO_LISP (val->accel);
-      mark_object (markee);
-    }
-  return 0;
-}
-
 static Lisp_Object
 mark_popup_data (Lisp_Object obj)
 {
   struct popup_data *data = (struct popup_data *) XPOPUP_DATA (obj);
 
-  /* Now mark the callbacks and such that are hidden in the lwlib
-     call-data */
-
-  if (data->id)
-    lw_map_widget_values (data->id, mark_widget_value_mapper, 0);
-
-  return data->last_menubar_buffer;
+  mark_object (data->last_menubar_buffer);
+  return data->protect_me;
 }
 
-#ifdef USE_KKCC
+static const struct memory_description popup_data_description [] = {
+  { XD_LISP_OBJECT, offsetof (struct popup_data, last_menubar_buffer) },
+  { XD_LISP_OBJECT, offsetof (struct popup_data, protect_me) },
+  { XD_END }
+};
+
 DEFINE_LRECORD_IMPLEMENTATION ("popup-data", popup_data,
 			       0, /*dumpable-flag*/
                                mark_popup_data, internal_object_printer,
-			       0, 0, 0, 0, struct popup_data);
-#else /* not USE_KKCC */
-DEFINE_LRECORD_IMPLEMENTATION ("popup-data", popup_data,
-                               mark_popup_data, internal_object_printer,
-			       0, 0, 0, 0, struct popup_data);
-#endif /* not USE_KKCC */
+			       0, 0, 0,
+			       popup_data_description,
+			       struct popup_data);
 
 /* This is like FRAME_MENUBAR_DATA (f), but contains an alist of
    (id . popup-data) for GCPRO'ing the callbacks of the popup menus
    and dialog boxes. */
 static Lisp_Object Vpopup_callbacks;
 
+static int
+snarf_widget_value_mapper (widget_value *val, void *closure)
+{
+  struct popup_data *pdata = (struct popup_data *) closure;
+
+  if (val->call_data)
+    pdata->protect_me = Fcons (VOID_TO_LISP (val->call_data),
+			      pdata->protect_me);
+  if (val->accel)
+    pdata->protect_me = Fcons (VOID_TO_LISP (val->accel),
+			      pdata->protect_me);
+
+  return 0;
+}
+
 void
 gcpro_popup_callbacks (LWLIB_ID id)
 {
@@ -122,17 +116,30 @@
   pdata = alloc_lcrecord_type (struct popup_data, &lrecord_popup_data);
   pdata->id = id;
   pdata->last_menubar_buffer = Qnil;
+  pdata->protect_me = Qnil;
   pdata->menubar_contents_up_to_date = 0;
   lpdata = wrap_popup_data (pdata);
+
+  /* Now snarf the callbacks and such that are hidden in the lwlib
+     call-data and accel and stick them into the list for proper
+     marking. */
+
+  if (pdata->id)
+    lw_map_widget_values (pdata->id, snarf_widget_value_mapper, pdata);
+
   Vpopup_callbacks = Fcons (Fcons (lid, lpdata), Vpopup_callbacks);
 }
 
 void
 ungcpro_popup_callbacks (LWLIB_ID id)
 {
+  struct popup_data *pdata;
   Lisp_Object lid = make_int (id);
   Lisp_Object this = assq_no_quit (lid, Vpopup_callbacks);
   assert (!NILP (this));
+  pdata = XPOPUP_DATA (XCDR (this));
+  free_list (pdata->protect_me);
+  pdata->protect_me = Qnil;
   Vpopup_callbacks = delq_no_quit (this, Vpopup_callbacks);
 }
 
@@ -201,7 +208,7 @@
   if (wv->value) xfree (wv->value);
   if (wv->name) xfree (wv->name);
 
-  wv->name = wv->value = wv->key = (char *) 0xDEADBEEF;
+  wv->name = wv->value = wv->key = (char *) 0xDEADBEEF; /* -559038737 base 10*/
 
   if (wv->contents && (wv->contents != (widget_value*)1))
     {
@@ -257,17 +264,10 @@
     {
       event = Fmake_event (Qnil, Qnil);
 
-#ifdef USE_KKCC
       XSET_EVENT_TYPE (event, misc_user_event);
       XSET_EVENT_CHANNEL (event, frame);
-      XSET_MISC_USER_DATA_FUNCTION (XEVENT_DATA (event), Qrun_hooks);
-      XSET_MISC_USER_DATA_OBJECT (XEVENT_DATA (event), Qmenu_no_selection_hook);
-#else /* not USE_KKCC */
-      XEVENT (event)->event_type = misc_user_event;
-      XEVENT (event)->channel = frame;
-      XEVENT (event)->event.eval.function = Qrun_hooks;
-      XEVENT (event)->event.eval.object = Qmenu_no_selection_hook;
-#endif /* not USE_KKCC */
+      XSET_EVENT_MISC_USER_FUNCTION (event, Qrun_hooks);
+      XSET_EVENT_MISC_USER_OBJECT (event, Qmenu_no_selection_hook);
     }
   else
     {
@@ -285,18 +285,10 @@
 	{
 	  event = Fmake_event (Qnil, Qnil);
 
-#ifdef USE_KKCC
 	  XSET_EVENT_TYPE (event, misc_user_event);
 	  XSET_EVENT_CHANNEL (event, frame);
-	  XSET_MISC_USER_DATA_FUNCTION (XEVENT_DATA (event), Qeval);
-	  XSET_MISC_USER_DATA_OBJECT (XEVENT_DATA (event), list4 (Qfuncall, callback_ex, image_instance, event));
-#else /* not USE_KKCC */
-	  XEVENT (event)->event_type = misc_user_event;
-	  XEVENT (event)->channel = frame;
-	  XEVENT (event)->event.eval.function = Qeval;
-	  XEVENT (event)->event.eval.object =
-	    list4 (Qfuncall, callback_ex, image_instance, event);
-#endif /* not USE_KKCC */
+	  XSET_EVENT_MISC_USER_FUNCTION (event, Qeval);
+	  XSET_EVENT_MISC_USER_OBJECT (event, list4 (Qfuncall, callback_ex, image_instance, event));
 	}
       else if (NILP (callback) || UNBOUNDP (callback))
 	event = Qnil;
@@ -307,17 +299,10 @@
 	  event = Fmake_event (Qnil, Qnil);
 
 	  get_gui_callback (callback, &fn, &arg);
-#ifdef USE_KKCC
 	  XSET_EVENT_TYPE (event, misc_user_event);
 	  XSET_EVENT_CHANNEL (event, frame);
-	  XSET_MISC_USER_DATA_FUNCTION (XEVENT_DATA (event), fn);
-	  XSET_MISC_USER_DATA_OBJECT (XEVENT_DATA (event), arg);
-#else /* not USE_KKCC */
-	  XEVENT (event)->event_type = misc_user_event;
-	  XEVENT (event)->channel = frame;
-	  XEVENT (event)->event.eval.function = fn;
-	  XEVENT (event)->event.eval.object = arg;
-#endif /* not USE_KKCC */
+	  XSET_EVENT_MISC_USER_FUNCTION (event, fn);
+	  XSET_EVENT_MISC_USER_OBJECT (event, arg);
 	}
     }
 
@@ -330,7 +315,7 @@
   DEVICE_X_MOUSE_TIMESTAMP (d) = DEVICE_X_GLOBAL_MOUSE_TIMESTAMP (d);
 #endif
   if (!NILP (event))
-    enqueue_Xt_dispatch_event (event);
+    enqueue_dispatch_event (event);
   /* The result of this evaluation could cause other instances to change so
      enqueue an update callback to check this. */
   if (update_subwindows_p && !NILP (event))
--- a/src/gui-x.h	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/gui-x.h	Sun Jan 12 11:08:22 2003 +0000
@@ -1,6 +1,6 @@
 /* General GUI code -- X-specific header file.
    Copyright (C) 1993, 1994 Free Software Foundation, Inc.
-   Copyright (C) 1996, 2000 Ben Wing.
+   Copyright (C) 1996, 2000, 2002 Ben Wing.
 
 This file is part of XEmacs.
 
@@ -42,11 +42,16 @@
 {
   struct lcrecord_header header;
 
-  /* lwlib ID of the tree of widgets corresponding to this popup.
-     We pass this to lw_map_widget_values() to retrieve all of our
-     Lispy call-data values that need to be GCPRO'd. */
+  /* lwlib ID of the tree of widgets corresponding to this popup.  We pass
+     this to lw_map_widget_values() to retrieve all of our Lispy call-data
+     and accel values that need to be GCPRO'd, and store them in the
+     following list. (We used to call lw_map_widget_values() during GC
+     mark, but that isn't compatible with KKCC.) */
   LWLIB_ID id;
 
+  /* List of GC-protected objects. */
+  Lisp_Object protect_me;
+
   /* For the frame popup data, this is the last buffer for which the
      menubar was displayed.  If the buffer has changed, we may have to
      update things. */
--- a/src/gui.c	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/gui.c	Sun Jan 12 11:08:22 2003 +0000
@@ -199,7 +199,6 @@
   Lisp_Gui_Item *lp = alloc_lcrecord_type (Lisp_Gui_Item, &lrecord_gui_item);
   Lisp_Object val;
 
-  zero_lcrecord (lp);
   val = wrap_gui_item (lp);
 
   gui_item_init (val);
@@ -549,8 +548,7 @@
   return Qnil;
 }
 
-#ifdef USE_KKCC
-static const struct lrecord_description gui_item_description [] = {
+static const struct memory_description gui_item_description [] = {
   { XD_LISP_OBJECT, offsetof (struct Lisp_Gui_Item, name) },
   { XD_LISP_OBJECT, offsetof (struct Lisp_Gui_Item, callback) },
   { XD_LISP_OBJECT, offsetof (struct Lisp_Gui_Item, callback_ex) },
@@ -566,7 +564,6 @@
   { XD_LISP_OBJECT, offsetof (struct Lisp_Gui_Item, value) },
   { XD_END }
 };
-#endif /* USE_KKCC */
 
 static Lisp_Object
 mark_gui_item (Lisp_Object obj)
@@ -789,7 +786,6 @@
 {
 }
 
-#ifdef USE_KKCC
 DEFINE_LRECORD_IMPLEMENTATION ("gui-item", gui_item,
 			       0, /*dumpable-flag*/
 			       mark_gui_item, print_gui_item,
@@ -797,14 +793,6 @@
 			       gui_item_hash,
 			       gui_item_description,
 			       Lisp_Gui_Item);
-#else /* not USE_KKCC */
-DEFINE_LRECORD_IMPLEMENTATION ("gui-item", gui_item,
-			       mark_gui_item, print_gui_item,
-			       finalize_gui_item, gui_item_equal,
-			       gui_item_hash,
-			       0,
-			       Lisp_Gui_Item);
-#endif /* not USE_KKCC */
 
 DOESNT_RETURN
 gui_error (const Char_ASCII *reason, Lisp_Object frob)
--- a/src/gutter.c	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/gutter.c	Sun Jan 12 11:08:22 2003 +0000
@@ -734,7 +734,7 @@
   if (EQ (position, Qright))  return RIGHT_GUTTER;
   invalid_constant ("Invalid gutter position", position);
 
-  RETURN_NOT_REACHED (TOP_GUTTER)
+  RETURN_NOT_REACHED (TOP_GUTTER);
 }
 
 DEFUN ("set-default-gutter-position", Fset_default_gutter_position, 1, 1, 0, /*
--- a/src/hash.c	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/hash.c	Sun Jan 12 11:08:22 2003 +0000
@@ -24,7 +24,7 @@
 #include "lisp.h"
 #include "hash.h"
 
-#define NULL_ENTRY ((void *) 0xdeadbeef)
+#define NULL_ENTRY ((void *) 0xdeadbeef) /* -559038737 base 10 */
 
 #define COMFORTABLE_SIZE(size) (21 * (size) / 16)
 
--- a/src/indent.c	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/indent.c	Sun Jan 12 11:08:22 2003 +0000
@@ -684,7 +684,7 @@
       return Dynarr_atp (w->line_start_cache, elt)->start;
     }
 
-  RETURN_NOT_REACHED(0)	/* shut up compiler */
+  RETURN_NOT_REACHED(0);	/* shut up compiler */
 }
 
 /*****************************************************************************
@@ -865,7 +865,7 @@
       needed += (remain*lines + abspix-1)/abspix + 3;
     }
 
-  RETURN_NOT_REACHED(0)	/* shut up compiler */
+  RETURN_NOT_REACHED(0);	/* shut up compiler */
 }
 
 DEFUN ("vertical-motion-pixels", Fvertical_motion_pixels, 1, 3, 0, /*
--- a/src/input-method-motif.c	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/input-method-motif.c	Sun Jan 12 11:08:22 2003 +0000
@@ -73,8 +73,8 @@
      default face, rather than foreground and background resources, or
      that the user can use set-frame-parameters to set xic attributes */
 
-#define res(name, class, representation, field, default_value)	\
-  { name, class, representation, sizeof(xim_resources.field),	\
+#define res(name, class_, representation, field, default_value)	\
+  { name, class_, representation, sizeof(xim_resources.field),	\
     XtOffsetOf(xim_resources_t, field),				\
     XtRString, (XtPointer) (default_value) }
 
--- a/src/input-method-xlib.c	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/input-method-xlib.c	Sun Jan 12 11:08:22 2003 +0000
@@ -164,7 +164,7 @@
 {
   struct device *d = (struct device *)client_data;
   XIM xim;
-  char *name, *class;
+  char *name, *class_;
   XIMCallback ximcallback;
   Lisp_Object tail;
 
@@ -172,8 +172,8 @@
   if ( xim_initted == False )
     {
       xim_initted = True;
-      XtGetApplicationNameAndClass (dpy, &name, &class);
-      DEVICE_X_XIM (d) = xim = XOpenIM (dpy, XtDatabase (dpy), name, class);
+      XtGetApplicationNameAndClass (dpy, &name, &class_);
+      DEVICE_X_XIM (d) = xim = XOpenIM (dpy, XtDatabase (dpy), name, class_);
 
       /* destroy callback for im */
       ximcallback.callback = (XIMProc) IMDestroyCallback;
@@ -216,11 +216,11 @@
   return;
 #else /* pre-X11R6 */
   Display *dpy = DEVICE_X_DISPLAY (d);
-  char *name, *class;
+  char *name, *class_;
   XIM xim;
 
-  XtGetApplicationNameAndClass (dpy, &name, &class);
-  DEVICE_X_XIM (d) = xim = XOpenIM (dpy, XtDatabase (dpy), name, class);
+  XtGetApplicationNameAndClass (dpy, &name, &class_);
+  DEVICE_X_XIM (d) = xim = XOpenIM (dpy, XtDatabase (dpy), name, class_);
   if (xim == NULL)
     {
       warn_when_safe
@@ -283,8 +283,8 @@
   xic_vars_t xic_vars;
   XIC xic;
 
-#define res(name, class, representation, field, default_value) \
-  { name, class, representation, sizeof(xic_vars.field), \
+#define res(name, class_, representation, field, default_value) \
+  { name, class_, representation, sizeof(xic_vars.field), \
      XtOffsetOf(xic_vars_t, field), XtRString, default_value }
 
   static XtResource resources[] =
--- a/src/intl-encap-win32.c	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/intl-encap-win32.c	Sun Jan 12 11:08:22 2003 +0000
@@ -1495,6 +1495,8 @@
     return 0;
 }
 
+#ifdef HAVE_MS_WINDOWS
+
 static int
 is_SHBrowseForFolder (HWND hwnd)
 {
@@ -1506,6 +1508,8 @@
   return 0;
 }
 
+#endif /* HAVE_MS_WINDOWS */
+
 LPITEMIDLIST
 qxeSHBrowseForFolder (LPBROWSEINFOW lpbi)
 {
@@ -1619,7 +1623,8 @@
 
       retval = SHGetDataFromIDListA (psf, pidl, nFormat, &ansidat, cb);
       if (retval == NOERROR)
-	copy_win32_find_dataa_to_win32_find_dataw (&ansidat, pv);
+	copy_win32_find_dataa_to_win32_find_dataw (&ansidat,
+						   (WIN32_FIND_DATAW *) pv);
       return retval;
     }
   else
--- a/src/intl-win32.c	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/intl-win32.c	Sun Jan 12 11:08:22 2003 +0000
@@ -1695,8 +1695,6 @@
 /*                 MS Windows multibyte-to-unicode methods              */
 /************************************************************************/
 
-DEFINE_CODING_SYSTEM_TYPE (mswindows_multibyte_to_unicode);
-
 enum mswindows_multibyte_cp_type
 {
   MULTIBYTE_ANSI,
@@ -1729,11 +1727,13 @@
   int cp;
 };
 
-static const struct lrecord_description
+static const struct memory_description
   mswindows_multibyte_to_unicode_coding_system_description[] = {
   { XD_END }
 };
 
+DEFINE_CODING_SYSTEM_TYPE_WITH_DATA (mswindows_multibyte_to_unicode);
+
 static void
 mswindows_multibyte_to_unicode_init (Lisp_Object codesys)
 {
@@ -2083,8 +2083,6 @@
 /*                       MS Windows Multibyte methods                   */
 /************************************************************************/
 
-DEFINE_CODING_SYSTEM_TYPE (mswindows_multibyte);
-
 struct mswindows_multibyte_coding_system
 {
   Lisp_Object code_page;
@@ -2096,17 +2094,17 @@
   int dummy;
 };
 
-static const struct lrecord_description
+static const struct memory_description
   mswindows_multibyte_coding_system_description[] = {
   { XD_LISP_OBJECT,
-      coding_system_data_offset +
-	offsetof (struct mswindows_multibyte_coding_system, code_page) },
+    offsetof (struct mswindows_multibyte_coding_system, code_page) },
   { XD_LISP_OBJECT,
-      coding_system_data_offset +
-	offsetof (struct mswindows_multibyte_coding_system, locale) },
+    offsetof (struct mswindows_multibyte_coding_system, locale) },
   { XD_END }
 };
 
+DEFINE_CODING_SYSTEM_TYPE_WITH_DATA (mswindows_multibyte);
+
 static Bytecount
 mswindows_multibyte_convert (struct coding_stream *str,
 			     const UExtbyte *src,
--- a/src/keymap.c	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/keymap.c	Sun Jan 12 11:08:22 2003 +0000
@@ -269,7 +269,7 @@
 		    (long) XINT (Fkeymap_fullness (obj)), keymap->header.uid);
 }
 
-static const struct lrecord_description keymap_description[] = {
+static const struct memory_description keymap_description[] = {
   { XD_LISP_OBJECT, offsetof (Lisp_Keymap, parents) },
   { XD_LISP_OBJECT, offsetof (Lisp_Keymap, prompt) },
   { XD_LISP_OBJECT, offsetof (Lisp_Keymap, table) },
@@ -281,18 +281,11 @@
 };
 
 /* No need for keymap_equal #### Why not? */
-#ifdef USE_KKCC
 DEFINE_LRECORD_IMPLEMENTATION ("keymap", keymap,
 			       1, /*dumpable-flag*/
                                mark_keymap, print_keymap, 0, 0, 0,
 			       keymap_description,
 			       Lisp_Keymap);
-#else /* not USE_KKCC */
-DEFINE_LRECORD_IMPLEMENTATION ("keymap", keymap,
-                               mark_keymap, print_keymap, 0, 0, 0,
-			       keymap_description,
-			       Lisp_Keymap);
-#endif /* not USE_KKCC */
 
 /************************************************************************/
 /*                Traversing keymaps and their parents                  */
@@ -436,17 +429,10 @@
 }
 
 static Lisp_Object
-#ifdef USE_KKCC
 make_key_description (const Lisp_Key_Data *key, int prettify)
 {
-  Lisp_Object keysym = KEY_DATA_KEYSYM(key);
+  Lisp_Object keysym = KEY_DATA_KEYSYM (key);
   int modifiers = KEY_DATA_MODIFIERS (key);
-#else /* not USE_KKCC */
-make_key_description (const struct key_data *key, int prettify)
-{
-  Lisp_Object keysym = key->keysym;
-  int modifiers = key->modifiers;
-#endif /* not USE_KKCC */
   if (prettify && CHARP (keysym))
     {
       /* This is a little slow, but (control a) is prettier than (control 65).
@@ -470,11 +456,7 @@
 
 static Lisp_Object
 raw_lookup_key (Lisp_Object keymap,
-#ifdef USE_KKCC
                 const Lisp_Key_Data *raw_keys, int raw_keys_count,
-#else /* not USE_KKCC */
-                const struct key_data *raw_keys, int raw_keys_count,
-#endif /* not USE_KKCC */
                 int keys_so_far, int accept_default);
 
 /* Relies on caller to gc-protect args */
@@ -658,20 +640,11 @@
 
 /* Relies on caller to gc-protect keymap, keysym, value */
 static void
-#ifdef USE_KKCC
 keymap_store (Lisp_Object keymap, const Lisp_Key_Data *key,
-#else /* not USE_KKCC */
-keymap_store (Lisp_Object keymap, const struct key_data *key,
-#endif /* not USE_KKCC */
               Lisp_Object value)
 {
-#ifdef USE_KKCC
   Lisp_Object keysym = KEY_DATA_KEYSYM (key);
   int modifiers = KEY_DATA_MODIFIERS (key);
-#else /* not USE_KKCC */
-  Lisp_Object keysym = key->keysym;
-  int modifiers = key->modifiers;
-#endif /* not USE_KKCC */
   Lisp_Keymap *k = XKEYMAP (keymap);
 
   modifiers &= ~(XEMACS_MOD_BUTTON1 | XEMACS_MOD_BUTTON2 | XEMACS_MOD_BUTTON3
@@ -1079,28 +1052,17 @@
   if (!NILP (map))
     {
       Lisp_Object idx = Fcdr (object);
-#ifdef USE_KKCC
       Lisp_Key_Data indirection;
-#else /* not USE_KKCC */
-      struct key_data indirection;
-#endif /* not USE_KKCC */
       if (CHARP (idx))
 	{
-#ifdef USE_KKCC
 	  Lisp_Object event = Fmake_event (Qnil, Qnil);
           struct gcpro gcpro1;
           GCPRO1 (event);
-	  XSET_EVENT_TYPE (event, empty_event);
 	  character_to_event (XCHAR (idx), XEVENT (event),
 			      XCONSOLE (Vselected_console), 0, 0);
-	  indirection = *XKEY_DATA (XEVENT_DATA (event));
-#else /* not USE_KKCC */
-	  Lisp_Event event;
-	  event.event_type = empty_event;
-	  character_to_event (XCHAR (idx), &event,
-			      XCONSOLE (Vselected_console), 0, 0);
-	  indirection = event.event.key;
-#endif /* not USE_KKCC */
+	  indirection.keysym = XEVENT_KEY_KEYSYM (event);
+	  indirection.modifiers = XEVENT_KEY_MODIFIERS (event);
+	  UNGCPRO;
 	}
       else if (CONSP (idx))
 	{
@@ -1112,11 +1074,7 @@
       else if (SYMBOLP (idx))
 	{
 	  indirection.keysym = idx;
-#ifdef USE_KKCC
 	  SET_KEY_DATA_MODIFIERS (&indirection, XINT (XCDR (idx)));
-#else /* not USE_KKCC */
-	  indirection.modifiers = 0;
-#endif /* not USE_KKCC */
 	}
       else
 	{
@@ -1142,24 +1100,14 @@
 }
 
 static Lisp_Object
-#ifdef USE_KKCC
 keymap_lookup_1 (Lisp_Object keymap, const Lisp_Key_Data *key,
-#else /* not USE_KKCC */
-keymap_lookup_1 (Lisp_Object keymap, const struct key_data *key,
-#endif /* not USE_KKCC */
                  int accept_default)
 {
   /* This function can GC */
-#ifdef USE_KKCC
   return get_keyelt (keymap_lookup_directly (keymap,
 					     KEY_DATA_KEYSYM (key), 
                                              KEY_DATA_MODIFIERS (key)),
 		     accept_default);
-#else /* not USE_KKCC */
-  return get_keyelt (keymap_lookup_directly (keymap,
-					     key->keysym, key->modifiers),
-		     accept_default);
-#endif /* not USE_KKCC */
 }
 
 
@@ -1452,58 +1400,35 @@
  */
 
 static void
-#ifdef USE_KKCC
 define_key_parser (Lisp_Object spec, Lisp_Key_Data *returned_value)
-#else /* not USE_KKCC */
-define_key_parser (Lisp_Object spec, struct key_data *returned_value)
-#endif /* not USE_KKCC */
 {
   if (CHAR_OR_CHAR_INTP (spec))
     {
-#ifdef USE_KKCC
       Lisp_Object event = Fmake_event (Qnil, Qnil);
       struct gcpro gcpro1;
       GCPRO1 (event);
-      XSET_EVENT_TYPE (event, empty_event);
       character_to_event (XCHAR_OR_CHAR_INT (spec), XEVENT (event),
 			  XCONSOLE (Vselected_console), 0, 0);
-      SET_KEY_DATA_KEYSYM (returned_value, XKEY_DATA_KEYSYM (XEVENT_DATA (event)));
+      SET_KEY_DATA_KEYSYM (returned_value, XEVENT_KEY_KEYSYM (event));
       SET_KEY_DATA_MODIFIERS (returned_value, 
-                              XKEY_DATA_MODIFIERS (XEVENT_DATA (event)));
-#else /* not USE_KKCC */
-      Lisp_Event event;
-      event.event_type = empty_event;
-      character_to_event (XCHAR_OR_CHAR_INT (spec), &event,
-			  XCONSOLE (Vselected_console), 0, 0);
-      returned_value->keysym    = event.event.key.keysym;
-      returned_value->modifiers = event.event.key.modifiers;
-#endif /* not USE_KKCC */
+                              XEVENT_KEY_MODIFIERS (event));
+      UNGCPRO;
     }
   else if (EVENTP (spec))
     {
-#ifdef USE_KKCC
       switch (XEVENT_TYPE (spec))
-#else /* not USE_KKCC */
-      switch (XEVENT (spec)->event_type)
-#endif /* not USE_KKCC */
 	{
 	case key_press_event:
           {
-#ifdef USE_KKCC
-            SET_KEY_DATA_KEYSYM (returned_value, XKEY_DATA_KEYSYM (XEVENT_DATA (spec)));
-            SET_KEY_DATA_MODIFIERS (returned_value, XKEY_DATA_MODIFIERS (XEVENT_DATA (spec)));
-#else /* not USE_KKCC */
-            returned_value->keysym    = XEVENT (spec)->event.key.keysym;
-            returned_value->modifiers = XEVENT (spec)->event.key.modifiers;
-#endif /* not USE_KKCC */
+            SET_KEY_DATA_KEYSYM (returned_value, XEVENT_KEY_KEYSYM (spec));
+            SET_KEY_DATA_MODIFIERS (returned_value, XEVENT_KEY_MODIFIERS (spec));
 	    break;
           }
 	case button_press_event:
 	case button_release_event:
 	  {
-#ifdef USE_KKCC
 	    int down = (XEVENT_TYPE (spec) == button_press_event);
-	    switch (XBUTTON_DATA_BUTTON (XEVENT_DATA (spec)))
+	    switch (XEVENT_BUTTON_BUTTON (spec))
 	      {
 	      case 1:
 		SET_KEY_DATA_KEYSYM (returned_value, (down ? Qbutton1 : Qbutton1up)); 
@@ -1530,30 +1455,7 @@
 		SET_KEY_DATA_KEYSYM (returned_value, (down ? Qbutton0 : Qbutton0up)); 
 		break;
 	      }
-	    SET_KEY_DATA_MODIFIERS (returned_value, XBUTTON_DATA_MODIFIERS (XEVENT_DATA (spec)));
-#else /* not USE_KKCC */
-	    int down = (XEVENT (spec)->event_type == button_press_event);
-	    switch (XEVENT (spec)->event.button.button)
-	      {
-	      case 1:
-		returned_value->keysym = (down ? Qbutton1 : Qbutton1up); break;
-	      case 2:
-		returned_value->keysym = (down ? Qbutton2 : Qbutton2up); break;
-	      case 3:
-		returned_value->keysym = (down ? Qbutton3 : Qbutton3up); break;
-	      case 4:
-		returned_value->keysym = (down ? Qbutton4 : Qbutton4up); break;
-	      case 5:
-		returned_value->keysym = (down ? Qbutton5 : Qbutton5up); break;
-	      case 6:
-		returned_value->keysym = (down ? Qbutton6 : Qbutton6up); break;
-	      case 7:
-		returned_value->keysym = (down ? Qbutton7 : Qbutton7up); break;
-	      default:
-		returned_value->keysym = (down ? Qbutton0 : Qbutton0up); break;
-	      }
-	    returned_value->modifiers = XEVENT (spec)->event.button.modifiers;
-#endif /* not USE_KKCC */
+	    SET_KEY_DATA_MODIFIERS (returned_value, XEVENT_BUTTON_MODIFIERS (spec));
 	    break;
 	  }
 	default:
@@ -1566,13 +1468,8 @@
       if (bucky_sym_to_bucky_bit (spec) != 0)
         invalid_argument ("Key is a modifier name", spec);
       define_key_check_and_coerce_keysym (spec, &spec, 0);
-#ifdef USE_KKCC
       SET_KEY_DATA_KEYSYM (returned_value, spec);
       SET_KEY_DATA_MODIFIERS (returned_value, 0);
-#else /* not USE_KKCC */
-      returned_value->keysym = spec;
-      returned_value->modifiers = 0;
-#endif /* not USE_KKCC */
     }
   else if (CONSP (spec))
     {
@@ -1607,13 +1504,8 @@
 			   "List must be nil-terminated", spec);
 
       define_key_check_and_coerce_keysym (spec, &keysym, modifiers);
-#ifdef USE_KKCC
       SET_KEY_DATA_KEYSYM(returned_value, keysym);
       SET_KEY_DATA_MODIFIERS (returned_value, modifiers);
-#else /* not USE_KKCC */
-      returned_value->keysym = keysym;
-      returned_value->modifiers = modifiers;
-#endif /* not USE_KKCC */
     }
   else
     {
@@ -1627,11 +1519,7 @@
 key_desc_list_to_event (Lisp_Object list, Lisp_Object event,
                         int allow_menu_events)
 {
-#ifdef USE_KKCC
   Lisp_Key_Data raw_key;
-#else /* not USE_KKCC */
-  struct key_data raw_key;
-#endif /* not USE_KKCC */
 
   if (allow_menu_events &&
       CONSP (list) &&
@@ -1646,17 +1534,10 @@
 	fn = Qcall_interactively;
       else
 	fn = Qeval;
-#ifdef USE_KKCC
       XSET_EVENT_TYPE (event, misc_user_event);
-      XSET_EVENT_CHANNEL (event, wrap_frame (selected_frame));
-      XSET_MISC_USER_DATA_FUNCTION (XEVENT_DATA (event), fn);
-      XSET_MISC_USER_DATA_OBJECT (XEVENT_DATA (event), arg);
-#else /* not USE_KKCC */
-      XEVENT (event)->channel = wrap_frame (selected_frame ());
-      XEVENT (event)->event_type = misc_user_event;
-      XEVENT (event)->event.eval.function = fn;
-      XEVENT (event)->event.eval.object = arg;
-#endif /* not USE_KKCC */
+      XSET_EVENT_CHANNEL (event, wrap_frame (selected_frame ()));
+      XSET_EVENT_MISC_USER_FUNCTION (event, fn);
+      XSET_EVENT_MISC_USER_OBJECT (event, arg);
       return;
     }
 
@@ -1673,28 +1554,21 @@
     invalid_operation ("Mouse-clicks can't appear in saved keyboard macros",
 		       Qunbound);
 
-#ifdef USE_KKCC
   XSET_EVENT_CHANNEL (event, Vselected_console);
   XSET_EVENT_TYPE (event, key_press_event);
-  XSET_KEY_DATA_KEYSYM (XEVENT_DATA (event), raw_key.keysym);
-  XSET_KEY_DATA_MODIFIERS (XEVENT_DATA (event), KEY_DATA_MODIFIERS (&raw_key));
-#else /* not USE_KKCC */
-  XEVENT (event)->channel = Vselected_console;
-  XEVENT (event)->event_type = key_press_event;
-  XEVENT (event)->event.key.keysym = raw_key.keysym;
-  XEVENT (event)->event.key.modifiers = raw_key.modifiers;
-#endif /* not USE_KKCC */
+  XSET_EVENT_KEY_KEYSYM (event, raw_key.keysym);
+  XSET_EVENT_KEY_MODIFIERS (event, KEY_DATA_MODIFIERS (&raw_key));
 }
 
 
 int
-event_matches_key_specifier_p (Lisp_Event *event, Lisp_Object key_specifier)
+event_matches_key_specifier_p (Lisp_Object event, Lisp_Object key_specifier)
 {
   Lisp_Object event2 = Qnil;
   int retval;
   struct gcpro gcpro1;
 
-  if (event->event_type != key_press_event || NILP (key_specifier) ||
+  if (XEVENT_TYPE (event) != key_press_event || NILP (key_specifier) ||
       (INTP (key_specifier) && !CHAR_INTP (key_specifier)))
     return 0;
 
@@ -1716,25 +1590,22 @@
      to compare the ASCII values. */
 
   GCPRO1 (event2);
-  event2 = Fmake_event (Qnil, Qnil);
-  Fcharacter_to_event (key_specifier, event2, Qnil, Qnil);
+  if (EVENTP (key_specifier))
+    event2 = Fcopy_event (key_specifier, Qnil);
+  else
+    event2 = Fcharacter_to_event (key_specifier, Qnil, Qnil, Qnil);
   if (XEVENT (event2)->event_type != key_press_event)
     retval = 0;
-  else if (CONSOLE_TTY_P (XCONSOLE (EVENT_CHANNEL (event))))
+  else if (CONSOLE_TTY_P (XCONSOLE (XEVENT_CHANNEL (event))))
     {
       int ch1, ch2;
 
       ch1 = event_to_character (event, 0, 0, 0);
-      ch2 = event_to_character (XEVENT (event2), 0, 0, 0);
+      ch2 = event_to_character (event2, 0, 0, 0);
       retval = (ch1 >= 0 && ch2 >= 0 && ch1 == ch2);
     }
-#ifdef USE_KKCC
-  else if (EQ (XKEY_DATA_KEYSYM (EVENT_DATA (event)), XKEY_DATA_KEYSYM (XEVENT_DATA (event2))) &&
-	   XKEY_DATA_MODIFIERS (EVENT_DATA (event)) == XKEY_DATA_MODIFIERS (XEVENT_DATA (event2)))
-#else /* not USE_KKCC */
-  else if (EQ (event->event.key.keysym, XEVENT (event2)->event.key.keysym) &&
-	   event->event.key.modifiers == XEVENT (event2)->event.key.modifiers)
-#endif /* not USE_KKCC */
+  else if (EQ (XEVENT_KEY_KEYSYM (event), XEVENT_KEY_KEYSYM (event2)) &&
+	   XEVENT_KEY_MODIFIERS (event) == XEVENT_KEY_MODIFIERS (event2))
     retval = 1;
   else
     retval = 0;
@@ -1744,87 +1615,75 @@
 }
 
 static int
-#ifdef USE_KKCC
 meta_prefix_char_p (const Lisp_Key_Data *key)
-#else /* not USE_KKCC */
-meta_prefix_char_p (const struct key_data *key)
-#endif /* not USE_KKCC */
 {
-#ifdef USE_KKCC
   Lisp_Object event = Fmake_event (Qnil, Qnil);
   struct gcpro gcpro1;
+  int retval;
+
   GCPRO1 (event);
 
   XSET_EVENT_TYPE (event, key_press_event);
   XSET_EVENT_CHANNEL (event, Vselected_console);
-  XSET_KEY_DATA_KEYSYM (XEVENT_DATA (event), KEY_DATA_KEYSYM (key));
-  XSET_KEY_DATA_MODIFIERS (XEVENT_DATA (event), KEY_DATA_MODIFIERS (key));
-  return event_matches_key_specifier_p (XEVENT(event), Vmeta_prefix_char);
-#else /* not USE_KKCC */
-  Lisp_Event event;
-
-  event.event_type = key_press_event;
-  event.channel = Vselected_console;
-  event.event.key.keysym = key->keysym;
-  event.event.key.modifiers = key->modifiers;
-  return event_matches_key_specifier_p (&event, Vmeta_prefix_char);
-#endif /* not USE_KKCC */
+  XSET_EVENT_KEY_KEYSYM (event, KEY_DATA_KEYSYM (key));
+  XSET_EVENT_KEY_MODIFIERS (event, KEY_DATA_MODIFIERS (key));
+  retval = event_matches_key_specifier_p (event, Vmeta_prefix_char);
+  UNGCPRO;
+  return retval;
 }
 
 DEFUN ("event-matches-key-specifier-p", Fevent_matches_key_specifier_p, 2, 2, 0, /*
 Return non-nil if EVENT matches KEY-SPECIFIER.
 This can be useful, e.g., to determine if the user pressed `help-char' or
 `quit-char'.
+
+KEY-SPECIFIER can be a character, integer, a symbol, a list of modifiers
+and symbols, or an event.
+
+What this actually happens is this:
+
+\(1) Return no, if EVENT is not a key press event or if KEY-SPECIFIER is nil
+    or an integer that cannot be converted to a character.
+
+\(2) If KEY-SPECIFIER is a character or integer,
+    (event-to-character EVENT nil nil nil) is called, and the characters are
+    compared to get the result.  The reason for special-casing this and doing
+    it this way is to ensure that, e.g., a KEY-SPECIFIER of 27 matches both
+    a key-press `escape' and a key-press `control ['. #### Think about META
+    argument to event-to-character.
+
+\(3) If KEY-SPECIFIER is an event, fine; else, convert to an event using
+    \(character-to-event KEY-SPECIFIER nil nil nil).  If EVENT is not on a TTY,
+    we just compare keysyms and modifiers and return yes if both are equal.
+    For TTY, we do character-level comparison by converting both to a character
+    with (event-to-character ... nil nil nil) and comparing the characters.
+
 */
        (event, key_specifier))
 {
   CHECK_LIVE_EVENT (event);
-  return (event_matches_key_specifier_p (XEVENT (event), key_specifier)
-	  ? Qt : Qnil);
+  return (event_matches_key_specifier_p (event, key_specifier) ? Qt : Qnil);
 }
-#ifdef USE_KKCC
-#define MACROLET(k,m) do {		\
-  SET_KEY_DATA_KEYSYM(returned_value, k);		\
-  SET_KEY_DATA_MODIFIERS(returned_value, m);    	\
-  RETURN_SANS_WARNINGS;			\
+#define MACROLET(k, m) do {			\
+  SET_KEY_DATA_KEYSYM (returned_value, k);	\
+  SET_KEY_DATA_MODIFIERS (returned_value, m);	\
+  RETURN_SANS_WARNINGS;				\
 } while (0)
-#else /* not USE_KKCC */
-#define MACROLET(k,m) do {		\
-  returned_value->keysym = (k);		\
-  returned_value->modifiers = (m);	\
-  RETURN_SANS_WARNINGS;			\
-} while (0)
-#endif /* not USE_KKCC */
 /* ASCII grunge.
    Given a keysym, return another keysym/modifier pair which could be
    considered the same key in an ASCII world.  Backspace returns ^H, for
    example.
  */
 static void
-#ifdef USE_KKCC
 define_key_alternate_name (Lisp_Key_Data *key,
                            Lisp_Key_Data *returned_value)
-#else /* not USE_KKCC */
-define_key_alternate_name (struct key_data *key,
-                           struct key_data *returned_value)
-#endif /* not USE_KKCC */
 {
-#ifdef USE_KKCC
   Lisp_Object keysym = KEY_DATA_KEYSYM (key);
   int modifiers = KEY_DATA_MODIFIERS (key);
-#else /* not USE_KKCC */
-  Lisp_Object keysym = key->keysym;
-  int modifiers = key->modifiers;
-#endif /* not USE_KKCC */
   int modifiers_sans_control = (modifiers & (~XEMACS_MOD_CONTROL));
   int modifiers_sans_meta = (modifiers & (~XEMACS_MOD_META));
-#ifdef USE_KKCC
   SET_KEY_DATA_KEYSYM (returned_value, Qnil); /* By default, no "alternate" key */
   SET_KEY_DATA_MODIFIERS (returned_value, 0);
-#else /* not USE_KKCC */
-  returned_value->keysym = Qnil; /* By default, no "alternate" key */
-  returned_value->modifiers = 0;
-#endif /* not USE_KKCC */
   if (modifiers_sans_meta == XEMACS_MOD_CONTROL)
     {
       if (EQ (keysym, QKspace))
@@ -1866,7 +1725,6 @@
 #undef MACROLET
 }
 
-
 static void
 ensure_meta_prefix_char_keymapp (Lisp_Object keys, int indx,
                                  Lisp_Object keymap)
@@ -1875,11 +1733,7 @@
   Lisp_Object new_keys;
   int i;
   Lisp_Object mpc_binding;
-#ifdef USE_KKCC
   Lisp_Key_Data meta_key;
-#else /* not USE_KKCC */
-  struct key_data meta_key;
-#endif /* not USE_KKCC */
   if (NILP (Vmeta_prefix_char) ||
       (INTP (Vmeta_prefix_char) && !CHAR_INTP (Vmeta_prefix_char)))
     return;
@@ -2080,13 +1934,8 @@
   while (1)
     {
       Lisp_Object c;
-#ifdef USE_KKCC
       Lisp_Key_Data raw_key1;
       Lisp_Key_Data raw_key2;
-#else /* not USE_KKCC */
-      struct key_data raw_key1;
-      struct key_data raw_key2;
-#endif /* not USE_KKCC */
       if (STRINGP (keys))
 	c = make_char (string_ichar (keys, idx));
       else
@@ -2200,11 +2049,7 @@
 struct raw_lookup_key_mapper_closure
 {
   int remaining;
-#ifdef USE_KKCC
   const Lisp_Key_Data *raw_keys;
-#else /* not USE_KKCC */
-  const struct key_data *raw_keys;
-#endif /* not USE_KKCC */
   int raw_keys_count;
   int keys_so_far;
   int accept_default;
@@ -2215,11 +2060,7 @@
 /* Caller should gc-protect args (keymaps may autoload) */
 static Lisp_Object
 raw_lookup_key (Lisp_Object keymap,
-#ifdef USE_KKCC
                 const Lisp_Key_Data *raw_keys, int raw_keys_count,
-#else /* not USE_KKCC */
-                const struct key_data *raw_keys, int raw_keys_count,
-#endif /* not USE_KKCC */
                 int keys_so_far, int accept_default)
 {
   /* This function can GC */
@@ -2242,11 +2083,7 @@
   int accept_default = c->accept_default;
   int remaining = c->remaining;
   int keys_so_far = c->keys_so_far;
-#ifdef USE_KKCC
   const Lisp_Key_Data *raw_keys = c->raw_keys;
-#else /* not USE_KKCC */
-  const struct key_data *raw_keys = c->raw_keys;
-#endif /* not USE_KKCC */
   Lisp_Object cmd;
 
   if (! meta_prefix_char_p (&(raw_keys[0])))
@@ -2299,11 +2136,7 @@
 				  keys_so_far + 1, accept_default);
 	  else if ((raw_keys[1].modifiers & XEMACS_MOD_META) == 0)
 	    {
-#ifdef USE_KKCC
 	      Lisp_Key_Data metified;
-#else /* not USE_KKCC */
-	      struct key_data metified;
-#endif /* not USE_KKCC */
 	      metified.keysym = raw_keys[1].keysym;
 	      metified.modifiers = raw_keys[1].modifiers |
 		(unsigned char) XEMACS_MOD_META;
@@ -2335,13 +2168,8 @@
              int accept_default)
 {
   /* This function can GC */
-#ifdef USE_KKCC
   Lisp_Key_Data kkk[20];
   Lisp_Key_Data *raw_keys;
-#else /* not USE_KKCC */
-  struct key_data kkk[20];
-  struct key_data *raw_keys;
-#endif /* not USE_KKCC */
   int i;
 
   if (nkeys == 0)
@@ -2350,11 +2178,7 @@
   if (nkeys < countof (kkk))
     raw_keys = kkk;
   else
-#ifdef USE_KKCC
     raw_keys = alloca_array (Lisp_Key_Data, nkeys);
-#else /* not USE_KKCC */
-    raw_keys = alloca_array (struct key_data, nkeys);
-#endif /* not USE_KKCC */
 
   for (i = 0; i < nkeys; i++)
     {
@@ -2368,19 +2192,11 @@
                int accept_default)
 {
   /* This function can GC */
-#ifdef USE_KKCC
   Lisp_Key_Data kkk[20];
-#else /* not USE_KKCC */
-  struct key_data kkk[20];
-#endif /* not USE_KKCC */
   Lisp_Object event;
 
   int nkeys;
-#ifdef USE_KKCC
   Lisp_Key_Data *raw_keys;
-#else /* not USE_KKCC */
-  struct key_data *raw_keys;
-#endif /* not USE_KKCC */
   Lisp_Object tem = Qnil;
   struct gcpro gcpro1, gcpro2;
   int iii;
@@ -2392,11 +2208,7 @@
   if (nkeys < countof (kkk))
     raw_keys = kkk;
   else
-#ifdef USE_KKCC
     raw_keys = alloca_array (Lisp_Key_Data, nkeys);
-#else /* not USE_KKCC */
-    raw_keys = alloca_array (struct key_data, nkeys);
-#endif /* not USE_KKCC */
 
   nkeys = 0;
   EVENT_CHAIN_LOOP (event, event_head)
@@ -2445,11 +2257,7 @@
     {
       int length = string_char_length (keys);
       int i;
-#ifdef USE_KKCC
       Lisp_Key_Data *raw_keys = alloca_array (Lisp_Key_Data, length);
-#else /* not USE_KKCC */
-      struct key_data *raw_keys = alloca_array (struct key_data, length);
-#endif /* not USE_KKCC */
       if (length == 0)
 	return Qnil;
 
@@ -3028,11 +2836,7 @@
 
 struct map_keymap_unsorted_closure
 {
-#ifdef USE_KKCC
   void (*fn) (const Lisp_Key_Data *, Lisp_Object binding, void *arg);
-#else /* not USE_KKCC */
-  void (*fn) (const struct key_data *, Lisp_Object binding, void *arg);
-#endif /* not USE_KKCC */
   void *arg;
   int modifiers;
 };
@@ -3060,11 +2864,7 @@
     }
   else
     {
-#ifdef USE_KKCC
       Lisp_Key_Data key;
-#else /* not USE_KKCC */
-      struct key_data key;
-#endif /* not USE_KKCC */
       key.keysym = keysym;
       key.modifiers = modifiers;
       ((*closure->fn) (&key, value, closure->arg));
@@ -3171,11 +2971,7 @@
 static void
 map_keymap_sorted (Lisp_Object keymap_table,
                    int modifiers,
-#ifdef USE_KKCC
                    void (*function) (const Lisp_Key_Data *key,
-#else /* not USE_KKCC */
-                   void (*function) (const struct key_data *key,
-#endif /* not USE_KKCC */
                                      Lisp_Object binding,
                                      void *map_keymap_sorted_closure),
                    void *map_keymap_sorted_closure)
@@ -3208,11 +3004,7 @@
 			   map_keymap_sorted_closure);
       else
 	{
-#ifdef USE_KKCC
 	  Lisp_Key_Data k;
-#else /* not USE_KKCC */
-	  struct key_data k;
-#endif /* not USE_KKCC */
 	  k.keysym = keysym;
 	  k.modifiers = modifiers;
 	  ((*function) (&k, binding, map_keymap_sorted_closure));
@@ -3224,11 +3016,7 @@
 
 /* used by Fmap_keymap() */
 static void
-#ifdef USE_KKCC
 map_keymap_mapper (const Lisp_Key_Data *key,
-#else /* not USE_KKCC */
-map_keymap_mapper (const struct key_data *key,
-#endif /* not USE_KKCC */
                    Lisp_Object binding,
                    void *function)
 {
@@ -3241,11 +3029,7 @@
 
 static void
 map_keymap (Lisp_Object keymap_table, int sort_first,
-#ifdef USE_KKCC
             void (*function) (const Lisp_Key_Data *key,
-#else /* not USE_KKCC */
-            void (*function) (const struct key_data *key,
-#endif /* not USE_KKCC */
                               Lisp_Object binding,
                               void *fn_arg),
             void *fn_arg)
@@ -3344,11 +3128,7 @@
       Lisp_Object vec;
       int j;
       int len;
-#ifdef USE_KKCC
       Lisp_Key_Data key;
-#else /* not USE_KKCC */
-      struct key_data key;
-#endif /* not USE_KKCC */
       key.keysym = keysym;
       key.modifiers = modifiers;
 
@@ -3436,11 +3216,7 @@
       NGCPRO1 (p);
       for (iii = 0; iii < len; iii++)
 	{
-#ifdef USE_KKCC
 	  Lisp_Key_Data key;
-#else /* not USE_KKCC */
-	  struct key_data key;
-#endif /* not USE_KKCC */
 	  define_key_parser (Faref (prefix, make_int (iii)), &key);
 	  XVECTOR_DATA (p)[iii] = make_key_description (&key, 1);
 	}
@@ -3536,27 +3312,15 @@
       
       if (!EVENTP (key))
 	{
-#ifdef USE_KKCC
 	  Lisp_Object event = Fmake_event (Qnil, Qnil);
-	  XSET_EVENT_TYPE (event, empty_event);
 	  CHECK_CHAR_COERCE_INT (key);
-	  character_to_event (XCHAR (key), XEVENT(event),
+	  character_to_event (XCHAR (key), XEVENT (event),
 			      XCONSOLE (Vselected_console), 0, 1);
 	  format_event_object (buf, event, 1);
+	  Fdeallocate_event (event);
 	}
       else
 	format_event_object (buf, key, 1);
-#else /* not USE_KKCC */
-	  Lisp_Event event;
-	  event.event_type = empty_event;
-	  CHECK_CHAR_COERCE_INT (key);
-	  character_to_event (XCHAR (key), &event,
-			      XCONSOLE (Vselected_console), 0, 1);
-	  format_event_object (buf, &event, 1);
-	}
-      else
-	format_event_object (buf, XEVENT (key), 1);
-#endif /* not USE_KKCC */
       str = eimake_string (buf);
       eifree (buf);
       return str;
@@ -3780,11 +3544,7 @@
 
 
 static Lisp_Object
-#ifdef USE_KKCC
 raw_keys_to_keys (Lisp_Key_Data *keys, int count)
-#else /* not USE_KKCC */
-raw_keys_to_keys (struct key_data *keys, int count)
-#endif /* not USE_KKCC */
 {
   Lisp_Object result = make_vector (count, Qnil);
   while (count--)
@@ -3794,36 +3554,21 @@
 
 
 static void
-#ifdef USE_KKCC
 format_raw_keys (Lisp_Key_Data *keys, int count, Eistring *buf)
-#else /* not USE_KKCC */
-format_raw_keys (struct key_data *keys, int count, Eistring *buf)
-#endif /* not USE_KKCC */
 {
   int i;
-#ifdef USE_KKCC
   Lisp_Object event = Fmake_event (Qnil, Qnil);
   XSET_EVENT_TYPE (event, key_press_event);
   XSET_EVENT_CHANNEL (event, Vselected_console);
-#else /* not USE_KKCC */
-  Lisp_Event event;
-  event.event_type = key_press_event;
-  event.channel = Vselected_console;
-#endif /* not USE_KKCC */
   for (i = 0; i < count; i++)
     {
-#ifdef USE_KKCC
-      XSET_KEY_DATA_KEYSYM (XEVENT_DATA (event), keys[i].keysym);
-      XSET_KEY_DATA_MODIFIERS (XEVENT_DATA (event), KEY_DATA_MODIFIERS (&keys[i]));
+      XSET_EVENT_KEY_KEYSYM (event, keys[i].keysym);
+      XSET_EVENT_KEY_MODIFIERS (event, KEY_DATA_MODIFIERS (&keys[i]));
       format_event_object (buf, event, 1);
-#else /* not USE_KKCC */
-      event.event.key.keysym    = keys[i].keysym;
-      event.event.key.modifiers = keys[i].modifiers;
-      format_event_object (buf, &event, 1);
-#endif /* not USE_KKCC */
       if (i < count - 1)
 	eicat_c (buf, " ");
     }
+  Fdeallocate_event (event);
 }
 
 
@@ -3856,11 +3601,7 @@
     int keys_count;
     int modifiers_so_far;
     Eistring *target_buffer;
-#ifdef USE_KKCC
     Lisp_Key_Data *keys_so_far;
-#else /* not USE_KKCC */
-    struct key_data *keys_so_far;
-#endif /* not USE_KKCC */
     int keys_so_far_total_size;
     int keys_so_far_malloced;
   };
@@ -3889,11 +3630,7 @@
 	 Verify that these bindings aren't shadowed by other bindings
 	 in the shadow maps.  Either nil or number as value from
 	 raw_lookup_key() means undefined.  */
-#ifdef USE_KKCC
       Lisp_Key_Data *so_far = c->keys_so_far;
-#else /* not USE_KKCC */
-      struct key_data *so_far = c->keys_so_far;
-#endif /* not USE_KKCC */
 
       for (;;) /* loop over all keys that match */
 	{
@@ -3901,11 +3638,7 @@
 	  int i;
 
 	  so_far [keys_count].keysym = k;
-#ifdef USE_KKCC
 	  SET_KEY_DATA_MODIFIERS (&so_far [keys_count], modifiers_so_far);
-#else /* not USE_KKCC */
-	  so_far [keys_count].modifiers = modifiers_so_far;
-#endif /* not USE_KKCC */
 
 	  /* now loop over all shadow maps */
 	  for (i = 0; i < c->shadow_count; i++)
@@ -3986,18 +3719,10 @@
 	lower_modifiers = (modifiers_so_far | bucky);
       else
 	{
-#ifdef USE_KKCC
 	  Lisp_Key_Data *so_far = c->keys_so_far;
-#else /* not USE_KKCC */
-	  struct key_data *so_far = c->keys_so_far;
-#endif /* not USE_KKCC */
 	  lower_modifiers = 0;
 	  so_far [lower_keys_count].keysym = key;
-#ifdef USE_KKCC
 	  SET_KEY_DATA_MODIFIERS (&so_far [lower_keys_count], modifiers_so_far);
-#else /* not USE_KKCC */
-	  so_far [lower_keys_count].modifiers = modifiers_so_far;
-#endif /* not USE_KKCC */
 	  lower_keys_count++;
 	}
 
@@ -4006,24 +3731,12 @@
 	  int size = lower_keys_count + 50;
 	  if (! c->keys_so_far_malloced)
 	    {
-#ifdef USE_KKCC
 	      Lisp_Key_Data *new = xnew_array (Lisp_Key_Data, size);
-#else /* not USE_KKCC */
-	      struct key_data *new = xnew_array (struct key_data, size);
-#endif /* not USE_KKCC */
 	      memcpy ((void *)new, (const void *)c->keys_so_far,
-#ifdef USE_KKCC
 		      c->keys_so_far_total_size * sizeof (Lisp_Key_Data));
-#else /* not USE_KKCC */
-		      c->keys_so_far_total_size * sizeof (struct key_data));
-#endif /* not USE_KKCC */
 	    }
 	  else
-#ifdef USE_KKCC
 	    XREALLOC_ARRAY (c->keys_so_far, Lisp_Key_Data, size);
-#else /* not USE_KKCC */
-	    XREALLOC_ARRAY (c->keys_so_far, struct key_data, size);
-#endif /* not USE_KKCC */
 
 	  c->keys_so_far_total_size = size;
 	  c->keys_so_far_malloced = 1;
@@ -4057,11 +3770,7 @@
   /* This function can GC */
   Lisp_Object result = Qnil;
   int i;
-#ifdef USE_KKCC
   Lisp_Key_Data raw[20];
-#else /* not USE_KKCC */
-  struct key_data raw[20];
-#endif /* not USE_KKCC */
   struct where_is_closure c;
 
   c.definition = definition;
@@ -4276,11 +3985,7 @@
 
 struct describe_map_shadow_closure
   {
-#ifdef USE_KKCC
     const Lisp_Key_Data *raw_key;
-#else /* not USE_KKCC */
-    const struct key_data *raw_key;
-#endif /* not USE_KKCC */
     Lisp_Object self;
   };
 
@@ -4293,15 +3998,9 @@
   if (EQ (map, c->self))
     return Qzero;		/* Not shadowed; terminate search */
 
-#ifdef USE_KKCC
   return !NILP (keymap_lookup_directly (map,
 					KEY_DATA_KEYSYM (c->raw_key),
 					KEY_DATA_MODIFIERS (c->raw_key)))
-#else /* not USE_KKCC */
-  return !NILP (keymap_lookup_directly (map,
-					c->raw_key->keysym,
-					c->raw_key->modifiers))
-#endif /* not USE_KKCC */
     ? Qt : Qnil;
 }
 
@@ -4309,35 +4008,21 @@
 static Lisp_Object
 keymap_lookup_inherited_mapper (Lisp_Object km, void *arg)
 {
-#ifdef USE_KKCC
   Lisp_Key_Data *k = (Lisp_Key_Data *) arg;
   return keymap_lookup_directly (km, KEY_DATA_KEYSYM (k), KEY_DATA_MODIFIERS (k));
-#else /* not USE_KKCC */
-  struct key_data *k = (struct key_data *) arg;
-  return keymap_lookup_directly (km, k->keysym, k->modifiers);
-#endif /* not USE_KKCC */
 }
 
 
 static void
-#ifdef USE_KKCC
 describe_map_mapper (const Lisp_Key_Data *key,
-#else /* not USE_KKCC */
-describe_map_mapper (const struct key_data *key,
-#endif /* not USE_KKCC */
                      Lisp_Object binding,
 		     void *describe_map_closure)
 {
   /* This function can GC */
   struct describe_map_closure *closure =
     (struct describe_map_closure *) describe_map_closure;
-#ifdef USE_KKCC
   Lisp_Object keysym = KEY_DATA_KEYSYM (key);
   int modifiers = KEY_DATA_MODIFIERS (key);
-#else /* not USE_KKCC */
-  Lisp_Object keysym = key->keysym;
-  int modifiers = key->modifiers;
-#endif /* not USE_KKCC */
 
   /* Don't mention suppressed commands.  */
   if (SYMBOLP (binding)
--- a/src/keymap.h	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/keymap.h	Sun Jan 12 11:08:22 2003 +0000
@@ -58,7 +58,7 @@
 void key_desc_list_to_event (Lisp_Object list, Lisp_Object event,
 			     int allow_menu_events);
 
-int event_matches_key_specifier_p (Lisp_Event *event,
+int event_matches_key_specifier_p (Lisp_Object event,
 				   Lisp_Object key_specifier);
 
 #endif /* INCLUDED_keymap_h_ */
--- a/src/lisp.h	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/lisp.h	Sun Jan 12 11:08:22 2003 +0000
@@ -315,7 +315,6 @@
    we don't want by creating overloaded versions of them and declaring them
    private. */
    
-#undef class
 #undef this
 
 class Bytecount;
@@ -742,7 +741,6 @@
   { return Bytecount (x - y); }
 #endif
 
-#define class c_class
 #define this c_this
 
 #endif /* __cplusplus */
@@ -815,7 +813,7 @@
 #ifndef DOESNT_RETURN
 # if defined __GNUC__
 #  if ((__GNUC__ > 2) || (__GNUC__ == 2) && (__GNUC_MINOR__ >= 5))
-#   define RETURN_NOT_REACHED(value)
+#   define RETURN_NOT_REACHED(value) DO_NOTHING
 #   define DOESNT_RETURN void
 #   define DECLARE_DOESNT_RETURN(decl) \
            extern void decl __attribute__ ((noreturn))
@@ -843,11 +841,21 @@
 /* "statement not reached */
 #if defined __SUNPRO_C || defined __USLC__
 #define RETURN_SANS_WARNINGS if (1) return
-#define RETURN_NOT_REACHED(value)
+#define RETURN_NOT_REACHED(value) DO_NOTHING
+#endif
+
+/* More ways to shut up compiler.  This works in Fcommand_loop_1(),
+   where there's an infinite loop in a function returning a Lisp object.
+*/
+#if defined (_MSC_VER) || defined (__SUNPRO_C) || defined (__SUNPRO_CC) || \
+  (defined (DEC_ALPHA) && defined (OSF1))
+#define DO_NOTHING_DISABLING_NO_RETURN_WARNINGS if (0) return Qnil
+#else
+#define DO_NOTHING_DISABLING_NO_RETURN_WARNINGS DO_NOTHING
 #endif
 
 #ifndef RETURN_NOT_REACHED
-#define RETURN_NOT_REACHED(value) return value;
+#define RETURN_NOT_REACHED(value) return (value)
 #endif
 
 #ifndef RETURN_SANS_WARNINGS
@@ -1873,6 +1881,36 @@
 } while (0)
 
 
+/* Macros for looping over internal alists.
+
+   *** ALL OF THESE MACROS MUST BE DECLARED INSIDE BRACES -- SEE ABOVE. ***
+
+   ALIST_LOOP_3 loops over an alist, at each iteration setting CAR and CDR
+   to the car and cdr of the acons.  CAR and CDR are automatically
+   declared.
+
+   ALIST_LOOP_4 is similar to ALIST_LOOP_3 but contains an additional
+   variable ACONS at the beginning for access to the acons itself.All of
+   the variables ACONS, CAR and CDR are automatically declared.
+*/
+
+#define ALIST_LOOP_3(car, cdr, alist)				\
+Lisp_Object _al3_acons_##car, car, cdr, _al3_tail_##car;	\
+  for (_al3_tail_##car = alist;					\
+       NILP (_al3_tail_##car) ?					\
+	 0 : (_al3_acons_##car = XCAR (_al3_tail_##car),	\
+	      car = XCAR (_al3_acons_##car),			\
+	      cdr = XCDR (_al3_acons_##car), 1);		\
+       _al3_tail_##car = XCDR (_al3_tail_##car))
+
+#define ALIST_LOOP_4(acons, car, cdr, list)			\
+Lisp_Object acons, car, cdr, _al4_tail_##car;			\
+  for (_al4_tail_##car = list;					\
+       NILP (_al4_tail_##car) ?					\
+	 0 : (elt = XCAR (_al4_tail_##car), car = XCAR (elt),	\
+	      cdr = XCDR (elt), 1);				\
+       _al4_tail_##car = XCDR (tail))
+
 /* Macros for looping over external alists.
 
    *** ALL OF THESE MACROS MUST BE DECLARED INSIDE BRACES -- SEE ABOVE. ***
@@ -2171,9 +2209,6 @@
 {
   struct lcrecord_header header;
   long size;
-  /* next is now chained through v->contents[size], terminated by Qzero.
-     This means that pure vectors don't need a "next" */
-  /* struct Lisp_Vector *next; */
   Lisp_Object contents[1];
 };
 typedef struct Lisp_Vector Lisp_Vector;
@@ -2209,8 +2244,7 @@
 
 struct Lisp_Bit_Vector
 {
-  struct lrecord_header lheader;
-  Lisp_Object next;
+  struct lcrecord_header lheader;
   Elemcount size;
   unsigned long bits[1];
 };
@@ -2236,7 +2270,6 @@
 } while (0)
 
 #define bit_vector_length(v) ((v)->size)
-#define bit_vector_next(v) ((v)->next)
 
 DECLARE_INLINE_HEADER (
 int
@@ -2669,30 +2702,6 @@
 int finish_marking_weak_lists (void);
 void prune_weak_lists (void);
 
-/*-------------------------- lcrecord-list -----------------------------*/
-
-struct lcrecord_list
-{
-  struct lcrecord_header header;
-  Lisp_Object free;
-  Elemcount size;
-  const struct lrecord_implementation *implementation;
-};
-
-DECLARE_LRECORD (lcrecord_list, struct lcrecord_list);
-#define XLCRECORD_LIST(x) XRECORD (x, lcrecord_list, struct lcrecord_list)
-#define wrap_lcrecord_list(p) wrap_record (p, lcrecord_list)
-#define LCRECORD_LISTP(x) RECORDP (x, lcrecord_list)
-/* #define CHECK_LCRECORD_LIST(x) CHECK_RECORD (x, lcrecord_list)
-   Lcrecord lists should never escape to the Lisp level, so
-   functions should not be doing this. */
-
-Lisp_Object make_lcrecord_list (Elemcount size,
-				const struct lrecord_implementation
-				*implementation);
-Lisp_Object allocate_managed_lcrecord (Lisp_Object lcrecord_list);
-void free_managed_lcrecord (Lisp_Object lcrecord_list, Lisp_Object lcrecord);
-
 
 /************************************************************************/
 /*      Definitions related to the format of text and of characters     */
@@ -3004,11 +3013,53 @@
 
 extern struct gcpro *gcprolist;
 
+/* #### Catching insufficient gcpro:
+
+   The C++ code below catches GCPRO without UNGCPRO or vice-versa.
+   Catching cases where there's no GCPRO or UNGCPRO but should be, however,
+   is much harder, but could be done:
+
+   1. Lisp_Object becomes a real object.  Its creator and destructor need to
+      figure out whether the object is on the stack (by looking at the range
+      that `this' is within), and if so, add the pointer to a list of all
+      stack-based Lisp_Objects.
+
+   2. The assignment method needs to do reference-counting on actual Lisp
+      objects -- in particular, we need to know if there are any references
+      to a Lisp object that are *NOT* from stack-based Lisp_Objects.
+
+   3. When we get to a point in the code where we might garbage collect --
+      i.e. Ffuncall(), Feval(), or Fgarbage_collect() is called -- we look
+      at our list of stack-based Lisp_Objects, and if there are any that
+      point to Lisp objects with no non-stack references, see if there are
+      any gcpros pointing to the object, and if not, set a flag indicating
+      that the object is "destroyed". (Don't abort yet because the function
+      might not use the object any more.)
+
+   4. When we detag a pointer using XFOO(), abort if its "destroyed" flag
+      is set.
+
+   --ben
+*/
+
 struct gcpro
 {
   struct gcpro *next;
   const Lisp_Object *var;	/* Address of first protected variable */
   int nvars;			/* Number of consecutive protected variables */
+#if defined (__cplusplus) && defined (ERROR_CHECK_GC)
+  /* Try to catch GCPRO without UNGCPRO, or vice-versa.  G++ complains (at
+     least with sufficient numbers of warnings enabled, i.e. -Weffc++) if a
+     copy constructor or assignment operator is not defined. */
+  gcpro () : next (0), var (0), nvars (0) { }
+  gcpro (const gcpro& g) : next (g.next), var (g.var), nvars (g.nvars) { }
+  gcpro& operator= (const gcpro& g) { next = g.next; var = g.var;
+				      nvars = g.nvars;
+#undef this
+				      return *this;}
+#define this c_this
+  ~gcpro () { assert (!next); }
+#endif /* defined (__cplusplus) && defined (ERROR_CHECK_GC) */
 };
 
 /* Normally, you declare variables gcpro1, gcpro2, ... and use the
@@ -3135,7 +3186,31 @@
   gcpro3.next = &gcpro2,   gcpro3.var = array3, gcpro3.nvars = n3,	\
   gcprolist = &gcpro3 ))
 
-#define UNGCPRO ((void) (gcprolist = gcpro1.next))
+#if defined (__cplusplus) && defined (ERROR_CHECK_GC)
+/* We need to reset each gcpro to avoid triggering the assert() in
+   ~gcpro().  This happens in UNGCPRO and longjmp(). */
+#define UNWIND_GCPRO_TO(val)						   \
+do									   \
+{									   \
+  struct gcpro *__gcpro_stop = (val);					   \
+  /* Try to catch UNGCPRO without GCPRO.  We arrange for there to be a	   \
+     sentinel at the end of the gcprolist, so it should never be NULL. */  \
+  assert (__gcpro_stop);						   \
+  while (gcprolist != __gcpro_stop)					   \
+    {									   \
+      struct gcpro *__gcpro_next = gcprolist->next;			   \
+      gcprolist->next = 0;						   \
+      gcprolist = __gcpro_next;						   \
+      assert (gcprolist);						   \
+    }									   \
+} while (0)
+#else
+#define UNWIND_GCPRO_TO(val) (gcprolist = (val))
+#endif /* defined (__cplusplus) && defined (ERROR_CHECK_GC) */
+
+#define UNGCPRO_1(gcpro1) UNWIND_GCPRO_TO (gcpro1.next)
+
+#define UNGCPRO UNGCPRO_1 (gcpro1)
 
 #define NGCPRO1(var1) ((void) (						\
   ngcpro1.next = gcprolist, ngcpro1.var = &var1, ngcpro1.nvars = 1,	\
@@ -3182,7 +3257,7 @@
   ngcpro3.next = &ngcpro2,  ngcpro3.var = array3, ngcpro3.nvars = n3,	\
   gcprolist = &ngcpro3 ))
 
-#define NUNGCPRO ((void) (gcprolist = ngcpro1.next))
+#define NUNGCPRO UNGCPRO_1 (ngcpro1)
 
 #define NNGCPRO1(var1) ((void) (					\
   nngcpro1.next = gcprolist, nngcpro1.var = &var1, nngcpro1.nvars = 1,	\
@@ -3229,7 +3304,7 @@
   nngcpro3.next = &nngcpro2,  nngcpro3.var = array3, nngcpro3.nvars = n3, \
   gcprolist = &nngcpro3 ))
 
-#define NNUNGCPRO ((void) (gcprolist = nngcpro1.next))
+#define NNUNGCPRO UNGCPRO_1 (nngcpro1)
 
 #endif /* ! DEBUG_GCPRO */
 
@@ -3299,81 +3374,6 @@
 
 
 /************************************************************************/
-/*		                 Dumping                		*/
-/************************************************************************/
-
-/* dump_add_root_struct_ptr (&var, &desc) dumps the structure pointed to by
-   `var'.  This is for a single relocatable pointer located in the data
-   segment (i.e. the block pointed to is in the heap). */
-#ifdef PDUMP
-void dump_add_root_struct_ptr (void *, const struct struct_description *);
-#else
-#define dump_add_root_struct_ptr(varaddr,descaddr) DO_NOTHING
-#endif
-
-/* dump_add_opaque (&var, size) dumps the opaque static structure `var'.
-   This is for a static block of memory (in the data segment, not the
-   heap), with no relocatable pointers in it. */
-#ifdef PDUMP
-void dump_add_opaque (const void *, Bytecount);
-#else
-#define dump_add_opaque(varaddr,size) DO_NOTHING
-#endif
-
-/* dump_add_root_block (&var, &desc) dumps the static structure located at
-   `var' and described by DESC.  This is for a static block of memory (in
-   the data segment, not the heap), with relocatable pointers in it, as
-   described by DESC. (#### Not yet implemented) */
-#ifdef PDUMP
-void dump_add_root_block (void *ptraddress,
-			  const struct lrecord_description *desc);
-#else
-#define dump_add_root_block(ptraddress,desc) DO_NOTHING
-#endif
-
-/* Call dump_add_opaque_int (&int_var) to dump `int_var', of type `int'. */
-#ifdef PDUMP
-#define dump_add_opaque_int(int_varaddr) do {	\
-  int *dao_ = (int_varaddr); /* type check */	\
-  dump_add_opaque (dao_, sizeof (*dao_));	\
-} while (0)
-#else
-#define dump_add_opaque_int(int_varaddr) DO_NOTHING
-#endif
-
-/* Call dump_add_opaque_fixnum (&fixnum_var) to dump `fixnum_var', of type `Fixnum'. */
-#ifdef PDUMP
-#define dump_add_opaque_fixnum(fixnum_varaddr) do {	\
-  Fixnum *dao_ = (fixnum_varaddr); /* type check */	\
-  dump_add_opaque (dao_, sizeof (*dao_));		\
-} while (0)
-#else
-#define dump_add_opaque_fixnum(fixnum_varaddr) DO_NOTHING
-#endif
-
-/* Call dump_add_root_object (&var) to ensure that var is properly updated after pdump. */
-#ifdef PDUMP
-void dump_add_root_object (Lisp_Object *);
-#else
-#define dump_add_root_object(varaddr) DO_NOTHING
-#endif
-
-/* Call dump_add_root_object (&var) to ensure that var is properly updated after
-   pdump.  var must point to a linked list of objects out of which
-   some may not be dumped */
-#ifdef PDUMP
-void dump_add_weak_object_chain (Lisp_Object *);
-#else
-#define dump_add_weak_object_chain(varaddr) DO_NOTHING
-#endif
-
-/* Nonzero means Emacs has already been initialized.
-   Used during startup to detect startup of dumped Emacs.  */
-extern int initialized;
-
-
-
-/************************************************************************/
 /*		              Misc definitions        	                */
 /************************************************************************/
 
@@ -3435,6 +3435,7 @@
 extern EMACS_INT gc_generation_number[1];
 int c_readonly (Lisp_Object);
 int lisp_readonly (Lisp_Object);
+void copy_lisp_object (Lisp_Object dst, Lisp_Object src);
 Lisp_Object build_intstring (const Ibyte *);
 Lisp_Object build_string (const CIbyte *);
 Lisp_Object build_ext_string (const Extbyte *, Lisp_Object);
@@ -3449,8 +3450,7 @@
 void free_cons (Lisp_Object);
 void free_list (Lisp_Object);
 void free_alist (Lisp_Object);
-void mark_conses_in_list (Lisp_Object);
-void free_marker (Lisp_Marker *);
+void free_marker (Lisp_Object);
 int object_dead_p (Lisp_Object);
 void mark_object (Lisp_Object obj);
 int marked_p (Lisp_Object obj);
@@ -3465,29 +3465,27 @@
 Bytecount malloced_storage_size (void *, Bytecount, struct overhead_stats *);
 Bytecount fixed_type_block_overhead (Bytecount);
 #endif
-#ifdef PDUMP
-void pdump (void);
-int pdump_load (const char *);
-
-extern char *pdump_start, *pdump_end;
-#define DUMPEDP(adr) ((((char *)(adr)) < pdump_end) && (((char *)(adr)) >= pdump_start))
-#else
-#define DUMPEDP(adr) 0
-#endif
-
-
-#ifdef USE_KKCC
-Lisp_Object allocate_event (void);
-Lisp_Object allocate_key_data (void);
-Lisp_Object allocate_button_data (void);
-Lisp_Object allocate_motion_data (void);
-Lisp_Object allocate_process_data (void);
-Lisp_Object allocate_timeout_data (void);
-Lisp_Object allocate_magic_data (void);
-Lisp_Object allocate_magic_eval_data (void);
-Lisp_Object allocate_eval_data (void);
-Lisp_Object allocate_misc_user_data (void);
-#endif /* USE_KKCC */
+
+#ifdef EVENT_DATA_AS_OBJECTS
+Lisp_Object make_key_data (void);
+Lisp_Object make_button_data (void);
+Lisp_Object make_motion_data (void);
+Lisp_Object make_process_data (void);
+Lisp_Object make_timeout_data (void);
+Lisp_Object make_magic_data (void);
+Lisp_Object make_magic_eval_data (void);
+Lisp_Object make_eval_data (void);
+Lisp_Object make_misc_user_data (void);
+void free_key_data (Lisp_Object);
+void free_button_data (Lisp_Object);
+void free_motion_data (Lisp_Object);
+void free_process_data (Lisp_Object);
+void free_timeout_data (Lisp_Object);
+void free_magic_data (Lisp_Object);
+void free_magic_eval_data (Lisp_Object);
+void free_eval_data (Lisp_Object);
+void free_misc_user_data (Lisp_Object);
+#endif /* EVENT_DATA_AS_OBJECTS */
 
 /* Defined in buffer.c */
 Lisp_Object get_truename_buffer (Lisp_Object);
@@ -3505,6 +3503,7 @@
 			      int (*mapfun) (struct buffer *buf,
 					     void *closure),
 			      void *closure);
+void cleanup_buffer_undo_lists (void);
 
 extern struct buffer *current_buffer;
 
@@ -4060,7 +4059,6 @@
 extern int modifier_keys_are_sticky;
 
 /* Defined in event-Xt.c */
-void enqueue_Xt_dispatch_event (Lisp_Object event);
 void signal_special_Xt_user_event (Lisp_Object, Lisp_Object, Lisp_Object);
 
 
@@ -4408,6 +4406,10 @@
 /* Lower-level ways to output data: */
 void print_internal (Lisp_Object, Lisp_Object, int);
 void debug_print (Lisp_Object);
+void debug_p4 (Lisp_Object obj);
+void debug_p3 (Lisp_Object obj);
+void debug_short_backtrace (int);
+void debug_backtrace (void);
 /* NOTE: Do not call this with the data of a Lisp_String.  Use princ.
  * Note: stream should be defaulted before calling
  *  (eg Qnil means stdout, not Vstandard_output, etc) */
@@ -4455,8 +4457,6 @@
 						 Lisp_Object, Lisp_Object);
 void float_to_string (char *, double);
 void internal_object_printer (Lisp_Object, Lisp_Object, int);
-void debug_short_backtrace (int);
-void debug_backtrace (void);
 
 /* Defined in profile.c */
 void mark_profiling_info (void);
@@ -4879,6 +4879,16 @@
   return atoi ((const char *) string);
 }
 
+DECLARE_INLINE_HEADER (Ibyte *qxestrupr (Ibyte *s))
+{
+  return (Ibyte *) strupr ((char *) s);
+}
+
+DECLARE_INLINE_HEADER (Ibyte *qxestrlwr (Ibyte *s))
+{
+  return (Ibyte *) strlwr ((char *) s);
+}
+
 int qxesprintf (Ibyte *buffer, const CIbyte *format, ...)
      PRINTF_ARGS (2, 3);
 
@@ -4922,8 +4932,8 @@
 					Bytebpos byte_end);
 
 /* Defined in unicode.c */
-extern const struct struct_description to_unicode_description[];
-extern const struct struct_description from_unicode_description[];
+extern const struct sized_memory_description to_unicode_description;
+extern const struct sized_memory_description from_unicode_description;
 void init_charset_unicode_tables (Lisp_Object charset);
 void free_charset_unicode_tables (Lisp_Object charset);
 void recalculate_unicode_precedence (void);
--- a/src/lrecord.h	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/lrecord.h	Sun Jan 12 11:08:22 2003 +0000
@@ -40,14 +40,14 @@
    of memory called `frob blocks').  These objects have a `struct
    lrecord_header' at the top, containing only the bits needed to find
    the lrecord_implementation for the object.  There are special
-   routines in alloc.c to deal with each such object type.
+   routines in alloc.c to create an object of each such type.
 
    Lcrecords are used for less common sorts of objects that don't do
    their own allocation.  Each such object is malloc()ed individually,
    and the objects are chained together through a `next' pointer.
    Lcrecords have a `struct lcrecord_header' at the top, which
    contains a `struct lrecord_header' and a `next' pointer, and are
-   allocated using alloc_lcrecord().
+   allocated using alloc_lcrecord_type() or its variants.
 
    Creating a new lcrecord type is fairly easy; just follow the
    lead of some existing type (e.g. hash tables).  Note that you
@@ -55,11 +55,14 @@
    defaults are provided for many of them.  Alternatively, if you're
    just looking for a way of encapsulating data (which possibly
    could contain Lisp_Objects in it), you may well be able to use
-   the opaque type. */
+   the opaque type. --ben
+*/
 
 struct lrecord_header
 {
-  /* index into lrecord_implementations_table[] */
+  /* Index into lrecord_implementations_table[].  Objects that have been
+     explicitly freed using e.g. free_cons() have lrecord_type_free in this
+     field. */
   unsigned int type :8;
 
   /* If `mark' is 0 after the GC mark phase, the object will be freed
@@ -97,7 +100,7 @@
 
   /* The `next' field is normally used to chain all lcrecords together
      so that the GC can find (and free) all of them.
-     `alloc_lcrecord' threads lcrecords together.
+     `basic_alloc_lcrecord' threads lcrecords together.
 
      The `next' field may be used for other purposes as long as some
      other mechanism is provided for letting the GC do its work.
@@ -134,15 +137,15 @@
   /* Symbol value magic types come first to make SYMBOL_VALUE_MAGIC_P fast.
      #### This should be replaced by a symbol_value_magic_p flag
      in the Lisp_Symbol lrecord_header. */
-  lrecord_type_symbol_value_forward,
-  lrecord_type_symbol_value_varalias,
-  lrecord_type_symbol_value_lisp_magic,
-  lrecord_type_symbol_value_buffer_local,
+  lrecord_type_symbol_value_forward, /* 0 */
+  lrecord_type_symbol_value_varalias, /* 1 */
+  lrecord_type_symbol_value_lisp_magic, /* 2 */
+  lrecord_type_symbol_value_buffer_local, /* 3 */
   lrecord_type_max_symbol_value_magic = lrecord_type_symbol_value_buffer_local,
 
-  lrecord_type_symbol,
-  lrecord_type_subr,
-  lrecord_type_cons,
+  lrecord_type_symbol, /* 4 */
+  lrecord_type_subr, /* 5 */
+  lrecord_type_cons, /* 6 */
   lrecord_type_vector,
   lrecord_type_string,
   lrecord_type_lcrecord_list,
@@ -166,7 +169,7 @@
   lrecord_type_extent_auxiliary,
   lrecord_type_marker,
   lrecord_type_event,
-#ifdef USE_KKCC
+#ifdef EVENT_DATA_AS_OBJECTS
   lrecord_type_key_data,
   lrecord_type_button_data,
   lrecord_type_motion_data,
@@ -176,7 +179,7 @@
   lrecord_type_misc_user_data,
   lrecord_type_magic_eval_data,
   lrecord_type_magic_data,
-#endif /* USE_KKCC */
+#endif /* EVENT_DATA_AS_OBJECTS */
   lrecord_type_keymap,
   lrecord_type_command_builder,
   lrecord_type_timeout,
@@ -233,7 +236,12 @@
      because the GC routines will do this).  Doing it this way reduces
      recursion, so the object returned should preferably be the one
      with the deepest level of Lisp_Object pointers.  This function
-     can be NULL, meaning no GC marking is necessary. */
+     can be NULL, meaning no GC marking is necessary.
+
+     NOTE NOTE NOTE: This is not used by KKCC (which uses the data
+     description below instead), unless the data description is missing.
+     Yes, this currently means there is logic duplication.  Eventually the
+     mark methods will be removed. */
   Lisp_Object (*marker) (Lisp_Object);
 
   /* `printer' converts the object to a printed representation.
@@ -263,8 +271,8 @@
      also NULL. */
   unsigned long (*hash) (Lisp_Object, int);
 
-  /* External data layout description */
-  const struct lrecord_description *description;
+  /* Data layout description for your object.  See long comment below. */
+  const struct memory_description *description;
 
   /* These functions allow any object type to have builtin property
      lists that can be manipulated from the lisp level with
@@ -275,7 +283,7 @@
   Lisp_Object (*plist) (Lisp_Object obj);
 
   /* Only one of `static_size' and `size_in_bytes_method' is non-0.
-     If both are 0, this type is not instantiable by alloc_lcrecord(). */
+     If both are 0, this type is not instantiable by basic_alloc_lcrecord(). */
   Bytecount static_size;
   Bytecount (*size_in_bytes_method) (const void *header);
 
@@ -284,9 +292,7 @@
 
   /* A "basic" lrecord is any lrecord that's not an lcrecord, i.e.
      one that does not have an lcrecord_header at the front and which
-     is (usually) allocated in frob blocks.  We only use this flag for
-     some consistency checking, and that only when error-checking is
-     enabled. */
+     is (usually) allocated in frob blocks. */
   unsigned int basic_p :1;
 };
 
@@ -320,37 +326,97 @@
   ((void) ((lheader)->lisp_readonly = 1))
 #define RECORD_MARKER(lheader) lrecord_markers[(lheader)->type]
 
-#ifdef USE_KKCC
 #define RECORD_DUMPABLE(lheader) (lrecord_implementations_table[(lheader)->type])->dumpable
-#endif /* USE_KKCC */
+
+/* Data description stuff
 
-/* External description stuff
+   Data layout descriptions describe blocks of memory (in particular, Lisp
+   objects and other objects on the heap, and global objects with pointers
+   to such heap objects), including their size and a list of the elements
+   that need relocating, marking or other special handling.  They are
+   currently used in two places: by pdump [the new, portable dumper] and
+   KKCC [the new garbage collector].  The two subsystems use the
+   descriptions in different ways, and as a result some of the descriptions
+   are appropriate only for one or the other, when it is known that only
+   that subsystem will use the description. (This is particularly the case
+   with objects that can't be dumped, because pdump needs more info than
+   KKCC.) However, properly written descriptions are appropriate for both,
+   and you should strive to write your descriptions that way, since the
+   dumpable status of an object may change and new uses for the
+   descriptions may be created. (An example that comes to mind is a
+   facility for determining the memory usage of XEmacs data structures --
+   like `buffer-memory-usage', `window-memory-usage', etc. but more
+   general.)
+
+   More specifically:
 
-   PLEASE NOTE: Both lrecord_description and struct_description are
-   badly misnamed.  In reality, an lrecord_description is nothing more
-   than a list of the elements in a block of memory that need
-   relocating or other special handling, and a struct_description is
-   no more than an lrecord_description plus the size of the block of
-   memory. (In fact, a struct_description can now have its size given
-   as zero, i.e. unspecified, meaning that the last element in the
-   structure is noted in the list and the size of the block can
-   therefore be computed from it.) The names stem from the fact
-   lrecord_descriptions are used to describe lrecords (the size of the
-   lrecord is elsewhere in its description, attached to its methods,
-   so it does not need to be given here), while struct_descriptions
-   are used to describe C structs; but both are used in various
-   additional ways.  Much better terms would be memory_description and
-   sized_memory_description.
+   Pdump (the portable dumper) needs to write out all objects in heap
+   space, and later on (in another invocation of XEmacs) load them back
+   into the heap, relocating all pointers to the heap objects in the global
+   data space. ("Heap" means anything malloc()ed, including all Lisp
+   objects, and "global data" means anything declared globally or
+   `static'.) Pdump, then, needs to be told about the location of all
+   global pointers to heap objects, all the description of all such
+   objects, including their size and any pointers to other heap (aka
+   "relocatable") objects. (Pdump assumes that the heap may occur in
+   different places in different invocations -- therefore, it is not enough
+   simply to write out the entire heap and later reload it at the same
+   location -- but that global data is always in the same place, and hence
+   pointers to it do not need to be relocated.  This assumption holds true
+   in general for modern operating systems, but would be broken, for
+   example, in a system without virtual memory, or when dealing with shared
+   libraries.  Also, unlike unexec, pdump does not usually write out or
+   restore objects in the global data space, and thus they need to be
+   initialized every time XEmacs is loaded.  This is the purpose of the
+   reinit_*() functions throughout XEmacs. [It's possible, however, to make
+   pdump restore global data.  This must be done, of course, for heap
+   pointers, but is also done for other values that are not easy to
+   recompute -- in particular, values established by the Lisp code loaded
+   at dump time.]) Note that the data type `Lisp_Object' is basically just
+   a relocatable pointer disguised as a long, and in general pdump treats
+   the Lisp_Object values and pointers to Lisp objects (e.g. Lisp_Object
+   vs. `struct frame *') identically. (NOTE: This equivalence depends
+   crucially on the current "minimal tagbits" implementation of Lisp_Object
+   pointers.)
 
-   An lrecord_description is an array of values. (This is actually
+   Descriptions are used by pdump in three places: (a) descriptions of Lisp
+   objects, referenced in the DEFINE_*LRECORD_*IMPLEMENTATION*() call; (b)
+   descriptions of global objects to be dumped, registered by
+   dump_add_root_block(); (c) descriptions of global pointers to
+   non-Lisp_Object heap objects, registered by dump_add_root_struct_ptr().
+   The descriptions need to tell pdump which elements of your structure are
+   Lisp_Objects or structure pointers, plus the descriptions in turn of the
+   non-Lisp_Object structures pointed to.  If these structures are you own
+   private ones, you will have to write these recursive descriptions
+   yourself; otherwise, you are reusing a structure already in existence
+   elsewhere and there is probably already a description for it.
+
+   Pdump does not care about Lisp objects that cannot be dumped (the
+   dumpable flag to DEFINE_*LRECORD_*IMPLEMENTATION*() is 0).
+
+   KKCC also uses data layout descriptions, but differently.  It cares
+   about all objects, dumpable or not, but specifically only wants to know
+   about Lisp_Objects in your object and in structures pointed to.  Thus,
+   it doesn't care about things like pointers to structures ot other blocks
+   of memory with no Lisp Objects in them, which pdump would care a lot
+   about.
+
+   Technically, then, you could write your description differently
+   depending on whether your object is dumpable -- the full pdump
+   description if so, the abbreviated KKCC description if not.  In fact,
+   some descriptions are written this way.  This is dangerous, though,
+   because another use might come along for the data descriptions, that
+   doesn't care about the dumper flag and makes use of some of the stuff
+   normally omitted from the "abbreviated" description -- see above.
+
+   A memory_description is an array of values. (This is actually
    misnamed, in that it does not just describe lrecords, but any
    blocks of memory.) The first value of each line is a type, the
    second the offset in the lrecord structure.  The third and
    following elements are parameters; their presence, type and number
    is type-dependent.
 
-   The description ends with a "XD_END", "XD_SPECIFIER_END" or
-   "XD_CODING_SYSTEM_END" record.
+   The description ends with an "XD_END" record.
 
    The top-level description of an lrecord or lcrecord does not need
    to describe every element, just the ones that need to be relocated,
@@ -358,14 +424,28 @@
    structures, whenever the structure size is given, rather than being
    defaulted by specifying 0 for the size.)
 
-   A struct_description is used for describing nested "structures".  (Again
-   a misnomer, since it can be used for any blocks of memory, not just
-   structures.) It just contains a size for the memory block, a pointer to
-   an lrecord_description, and (for unions only) a union constant,
-   described below.  The size can be 0, in which case the size will be
-   determined from the largest offset logically referenced (i.e. last
-   offset mentioned + size of that object).  This is useful for stretchy
-   arrays.
+   A sized_memory_description is a memory_description plus the size of the
+   block of memory.  The size field in a sized_memory_description can be
+   given as zero, i.e. unspecified, meaning that the last element in the
+   structure is described in the description and the size of the block can
+   therefore be computed from it. (This is useful for stretchy arrays.)
+
+   memory_descriptions are used to describe lrecords (the size of the
+   lrecord is elsewhere in its description, attached to its methods, so it
+   does not need to be given here) and global objects, where the size is an
+   argument to the call to dump_add_root_block().
+   sized_memory_descriptions are used for pointers and arrays in
+   memory_descriptions and for calls to dump_add_root_struct_ptr(). (####
+   It is not obvious why this is so in the latter case.  Probably, calls to
+   dump_add_root_struct_ptr() should use plain memory_descriptions and have
+   the size be an argument to the call.)
+
+   NOTE: Anywhere that a sized_memory_description occurs inside of a plain
+   memory_description, a "description map" can be substituted.  Rather than
+   being an actual description, this describes how to find the description
+   by looking inside of the object being described.  This is a convenient
+   way to describe Lisp objects with subtypes and corresponding
+   type-specific data.
 
    Some example descriptions :
 
@@ -377,7 +457,7 @@
      Lisp_Object plist;
    };
 
-   static const struct lrecord_description cons_description[] = {
+   static const struct memory_description cons_description[] = {
      { XD_LISP_OBJECT, offsetof (Lisp_Cons, car) },
      { XD_LISP_OBJECT, offsetof (Lisp_Cons, cdr) },
      { XD_END }
@@ -385,9 +465,9 @@
 
    Which means "two lisp objects starting at the 'car' and 'cdr' elements"
 
-   static const struct lrecord_description string_description[] = {
+   static const struct memory_description string_description[] = {
      { XD_BYTECOUNT,       offsetof (Lisp_String, size) },
-     { XD_OPAQUE_DATA_PTR, offsetof (Lisp_String, data), XD_INDIRECT(0, 1) },
+     { XD_OPAQUE_DATA_PTR, offsetof (Lisp_String, data), XD_INDIRECT (0, 1) },
      { XD_LISP_OBJECT,     offsetof (Lisp_String, plist) },
      { XD_END }
    };
@@ -408,32 +488,33 @@
 
    You'd use XD_STRUCT_PTR, something like:
 
-   static const struct lrecord_description lo_description_1[] = {
+   static const struct memory_description foo_description[] = {
+     ...
+     { XD_INT,		offsetof (Lisp_Foo, count) },
+     { XD_STRUCT_PTR,	offsetof (Lisp_Foo, objects),
+       XD_INDIRECT (0, 0), &lisp_object_description },
+     ...
+   };
+
+   lisp_object_description is declared in alloc.c, like this:
+
+   static const struct memory_description lisp_object_description_1[] = {
      { XD_LISP_OBJECT, 0 },
      { XD_END }
    };
 
-   static const struct struct_description lo_description = {
+   const struct sized_memory_description lisp_object_description = {
      sizeof (Lisp_Object),
-     lo_description_1
+     lisp_object_description_1
    };
 
-   static const struct lrecord_description foo_description[] = {
-     ...
-     { XD_INT,		offsetof (Lisp_Foo, count) },
-     { XD_STRUCT_PTR,	offsetof (Lisp_Foo, objects),
-       XD_INDIRECT (0, 0), &lo_description },
-     ...
-   };
-
-
    Another example of XD_STRUCT_PTR:
 
-   typedef struct hentry
+   typedef struct htentry
    {
      Lisp_Object key;
      Lisp_Object value;
-   } hentry;
+   } htentry;
    
    struct Lisp_Hash_Table
    {
@@ -446,27 +527,27 @@
      Elemcount golden_ratio;
      hash_table_hash_function_t hash_function;
      hash_table_test_function_t test_function;
-     hentry *hentries;
+     htentry *hentries;
      enum hash_table_weakness weakness;
      Lisp_Object next_weak;     // Used to chain together all of the weak
    			        // hash tables.  Don't mark through this.
    };
 
-   static const struct lrecord_description hentry_description_1[] = {
-     { XD_LISP_OBJECT, offsetof (hentry, key) },
-     { XD_LISP_OBJECT, offsetof (hentry, value) },
+   static const struct memory_description htentry_description_1[] = {
+     { XD_LISP_OBJECT, offsetof (htentry, key) },
+     { XD_LISP_OBJECT, offsetof (htentry, value) },
      { XD_END }
    };
    
-   static const struct struct_description hentry_description = {
-     sizeof (hentry),
-     hentry_description_1
+   static const struct sized_memory_description htentry_description = {
+     sizeof (htentry),
+     htentry_description_1
    };
    
-   const struct lrecord_description hash_table_description[] = {
+   const struct memory_description hash_table_description[] = {
      { XD_ELEMCOUNT,     offsetof (Lisp_Hash_Table, size) },
-     { XD_STRUCT_PTR, offsetof (Lisp_Hash_Table, hentries), XD_INDIRECT(0, 1),
-	 &hentry_description },
+     { XD_STRUCT_PTR, offsetof (Lisp_Hash_Table, hentries), XD_INDIRECT (0, 1),
+	 &htentry_description },
      { XD_LO_LINK,    offsetof (Lisp_Hash_Table, next_weak) },
      { XD_END }
    };
@@ -475,21 +556,71 @@
    the ones that need to be relocated (Lisp_Objects and structures) or that
    need to be referenced as counts for relocated objects.
 
+   A description map looks like this:
+
+   static const struct sized_memory_description specifier_extra_description_map [] = {
+   { offsetof (Lisp_Specifier, methods) },
+   { offsetof (struct specifier_methods, extra_description) },
+   { -1 }
+   };
+ 
+   const struct memory_description specifier_description[] = {
+     ...
+     { XD_STRUCT_ARRAY, offset (Lisp_Specifier, data), 1,
+       specifier_extra_description_map },
+     ...
+     { XD_END }
+   };
+
+   This would be appropriate for an object that looks like this:
+ 
+   struct specifier_methods
+   {
+     ...
+     const struct sized_memory_description *extra_description;
+     ...
+   };
+
+   struct Lisp_Specifier
+   {
+     struct lcrecord_header header;
+     struct specifier_methods *methods;
+   
+     ...
+     // type-specific extra data attached to a specifier
+     max_align_t data[1];
+   };
+
+   The description map means "retrieve a pointer into the object at offset
+   `offsetof (Lisp_Specifier, methods)' , then in turn retrieve a pointer
+   into that object at offset `offsetof (struct specifier_methods,
+   extra_description)', and that is the sized_memory_description to use." 
+   There can be any number of indirections, which can be either into
+   straight pointers or Lisp_Objects.  The way that description maps are
+   distinguished from normal sized_memory_descriptions is that in the
+   former, the memory_description pointer is NULL.
+
+   --ben
+
 
    The existing types :
 
 
     XD_LISP_OBJECT
-  A Lisp object.  This is also the type to use for pointers to other lrecords.
+
+  A Lisp object.  This is also the type to use for pointers to other lrecords
+  (e.g. struct frame *).
 
     XD_LISP_OBJECT_ARRAY
+
   An array of Lisp objects or (equivalently) pointers to lrecords.
   The parameter (i.e. third element) is the count.  This would be declared
   as Lisp_Object foo[666].  For something declared as Lisp_Object *foo,
-  use XD_STRUCT_PTR, whose description parameter is a struct_description
+  use XD_STRUCT_PTR, whose description parameter is a sized_memory_description
   consisting of only XD_LISP_OBJECT and XD_END.
 
     XD_LO_LINK
+
   Weak link in a linked list of objects of the same type.  This is a
   link that does NOT generate a GC reference.  Thus the pdumper will
   not automatically add the referenced object to the table of all
@@ -501,103 +632,137 @@
   object.
 
     XD_OPAQUE_PTR
+
   Pointer to undumpable data.  Must be NULL when dumping.
 
     XD_STRUCT_PTR
+
   Pointer to block of described memory. (This is misnamed: It is NOT
   necessarily a pointer to a struct foo.) Parameters are number of
-  contiguous blocks and struct_description.
+  contiguous blocks and sized_memory_description.
 
     XD_STRUCT_ARRAY
+
   Array of blocks of described memory.  Parameters are number of
-  structures and struct_description.  This differs from XD_STRUCT_PTR
+  structures and sized_memory_description.  This differs from XD_STRUCT_PTR
   in that the parameter is declared as struct foo[666] instead of
   struct *foo.  In other words, the block of memory holding the
   structures is within the containing structure, rather than being
   elsewhere, with a pointer in the containing structure.
 
+  NOTE NOTE NOTE: Be sure that you understand the difference between
+  XD_STRUCT_PTR and XD_STRUCT_ARRAY:
+    - struct foo bar[666], i.e. 666 inline struct foos
+        --> XD_STRUCT_ARRAY, argument 666, pointing to a description of
+            struct foo
+    - struct foo *bar, i.e. pointer to a block of 666 struct foos
+        --> XD_STRUCT_PTR, argument 666, pointing to a description of
+            struct foo
+    - struct foo *bar[666], i.e. 666 pointers to separate blocks of struct foos
+        --> XD_STRUCT_ARRAY, argument 666, pointing to a description of
+	    a single pointer to struct foo; the description is a single
+	    XD_STRUCT_PTR, argument 1, which in turn points to a description
+	    of struct foo.
+
     XD_OPAQUE_DATA_PTR
+
   Pointer to dumpable opaque data.  Parameter is the size of the data.
   Pointed data must be relocatable without changes.
 
     XD_UNION
-  Union of two or more different types of data.  Parameters are a
-  constant which determines which type the data is (this is usually an
-  XD_INDIRECT, referring to one of the fields in the structure), and
-  an array of struct_descriptions, whose values are used as follows,
-  which is *DIFFERENT* from their usage in XD_STRUCT_PTR: the first
-  field is a constant, which is compared to the first parameter of the
-  XD_UNION descriptor to determine if this description applies to the
-  data at the given offset, and the second is a pointer to a *SINGLE*
-  lrecord_description structure, describing the data being pointed at
-  when the associated constant matches.  You can go ahead and create
-  an array of lrecord_description structures and put an XD_END on it,
-  but only the first one is used.  If the data being pointed at is a
-  structure, you *MAY NOT* substitute an array of lrecord_description
-  structures describing the structure; instead, use a single
-  lrecord_description structure with an XD_STRUCT_PTR in it, and point
-  it in turn to the description of the structure.  See charset.h for a
-  description of how to use XD_UNION. (In other words, if the constant
-  matches, the lrecord_description pointed at will in essence be
-  substituted for the XD_UNION declaration.)
+
+  Union of two or more different types of data.  Parameters are a constant
+  which determines which type the data is (this is usually an XD_INDIRECT,
+  referring to one of the fields in the structure), and a "sizing lobby" (a
+  sized_memory_description, which points to a memory_description and
+  indicates its size).  The size field in the sizing lobby describes the
+  size of the union field in the object, and the memory_description in it
+  is referred to as a "union map" and has a special interpretation: The
+  offset field is replaced by a constant, which is compared to the first
+  parameter of the XD_UNION descriptor to determine if this description
+  applies to the union data, and XD_INDIRECT references refer to the
+  containing object and description.  Note that the description applies
+  "inline" to the union data, like XD_STRUCT_ARRAY and not XD_STRUCT_PTR.
+  If the union data is a pointer to different types of structures, each
+  element in the memory_description should be an XD_STRUCT_PTR.  See
+  unicode.c, redisplay.c and objects.c for examples of XD_UNION.
+
+    XD_UNION_DYNAMIC_SIZE
+
+  Same as XD_UNION except that this is used for objects where the size of
+  the object containing the union varies depending on the particular value
+  of the union constant.  That is, an object with plain XD_UNION typically
+  has the union declared as `union foo' or as `void *', where an object
+  with XD_UNION_DYNAMIC_SIZE typically has the union as the last element,
+  and declared as something like char foo[1].  With plain XD_UNION, the
+  object is (usually) of fixed size and always contains enough space for
+  the data associated with all possible union constants, and thus the union
+  constant can potentially change during the lifetime of the object.  With
+  XD_UNION_DYNAMIC_SIZE, however, the union constant is fixed at the time
+  of creation of the object, and the size of the object is computed
+  dynamically at creation time based on the size of the data associated
+  with the union constant.  Currently, the only difference between XD_UNION
+  and XD_UNION_DYNAMIC_SIZE is how the size of the union data is
+  calculated, when (a) the structure containing the union has no size
+  given; (b) the union occurs as the last element in the structure; and (c)
+  the union has no size given (in the first-level sized_memory_description
+  pointed to).  In this circumstance, the size of XD_UNION comes from the
+  max size of the data associated with all possible union constants,
+  whereas the size of XD_UNION_DYNAMIC_SIZE comes from the size of the data
+  associated with the currently specified (and unchangeable) union
+  constant.
 
     XD_C_STRING
+
   Pointer to a C string.
 
     XD_DOC_STRING
+
   Pointer to a doc string (C string if positive, opaque value if negative)
 
     XD_INT_RESET
+
   An integer which will be reset to a given value in the dump file.
 
+    XD_ELEMCOUNT
 
-    XD_ELEMCOUNT
   Elemcount value.  Used for counts.
 
     XD_BYTECOUNT
+
   Bytecount value.  Used for counts.
 
     XD_HASHCODE
+
   Hashcode value.  Used for the results of hashing functions.
 
     XD_INT
+
   int value.  Used for counts.
 
     XD_LONG
+
   long value.  Used for counts.
 
     XD_BYTECOUNT
+
   bytecount value.  Used for counts.
 
     XD_END
+
   Special type indicating the end of the array.
 
-    XD_SPECIFIER_END
-  Special type indicating the end of the array for a specifier.  Extra
-  description, describing the specifier-type-specific data at the end
-  of the specifier object, is going to be fetched from the specifier
-  methods.  This should occur exactly once, in the description of the
-  specifier object, and the dump code knows how to special-case this
-  by fetching the specifier_methods pointer from the appropriate place
-  in the memory block (which will, of course, be a struct
-  Lisp_Specifier), fetching the description of the
-  specifier-type-specific data from this, and continuing processing
-  the memory block.
-
-    XD_CODING_SYSTEM_END
-  Special type indicating the end of the array for a coding system.
-  Extra description is going to be fetched from the coding system
-  methods.  Works just like XD_SPECIFIER_END.
-
 
   Special macros:
-    XD_INDIRECT(line, delta)
-  Usable where  a "count" or "size"  is requested.  Gives the value of
-  the element which is at line number 'line' in the description (count
-  starts at zero) and adds delta to it.
+
+    XD_INDIRECT (line, delta)
+  Usable where a count, size, offset or union constant is requested.  Gives
+  the value of the element which is at line number 'line' in the
+  description (count starts at zero) and adds delta to it, which must
+  (currently) be positive.
 */
 
-enum lrecord_description_type
+enum memory_description_type
 {
   XD_LISP_OBJECT_ARRAY,
   XD_LISP_OBJECT,
@@ -607,6 +772,7 @@
   XD_STRUCT_ARRAY,
   XD_OPAQUE_DATA_PTR,
   XD_UNION,
+  XD_UNION_DYNAMIC_SIZE,
   XD_C_STRING,
   XD_DOC_STRING,
   XD_INT_RESET,
@@ -615,35 +781,83 @@
   XD_HASHCODE,
   XD_INT,
   XD_LONG,
-  XD_END,
-  XD_SPECIFIER_END,
-  XD_CODING_SYSTEM_END
-};
-
-struct lrecord_description
-{
-  enum lrecord_description_type type;
-  int offset;
-  EMACS_INT data1;
-  const struct struct_description *data2;
+  XD_END
 };
 
-struct struct_description
+enum data_description_entry_flags
 {
-  Bytecount size;
-  const struct lrecord_description *description;
+  /* If set, KKCC does not process this entry.
+
+  (1) One obvious use is with things that pdump saves but which do not get
+  marked normally -- for example the next and prev fields in a marker.  The
+  marker chain is weak, with its entries removed when they are finalized.
+
+  (2) This can be set on structures not containing any Lisp objects, or (more
+  usefully) on structures that contain Lisp objects but where the objects
+  always occur in another structure as well.  For example, the extent lists
+  kept by a buffer keep the extents in two lists, one sorted by the start
+  of the extent and the other by the end.  There's no point in marking
+  both, since each contains the same objects as the other; but when dumping
+  (if we were to dump such a structure), when computing memory size, etc.,
+  it's crucial to tag both sides.
+  */
+  XD_FLAG_NO_KKCC = 1,
+  /* If set, pdump does not process this entry. */
+  XD_FLAG_NO_PDUMP = 2,
+  /* Indicates that this is a "default" entry in a union map. */
+  XD_FLAG_UNION_DEFAULT_ENTRY = 4,
+  /* Indicates that this is a free Lisp object we're marking.
+     Only relevant for ERROR_CHECK_GC.  This occurs when we're marking
+     lcrecord-lists, where the objects have had their type changed to
+     lrecord_type_free and also have had their free bit set, but we mark
+     them as normal. */
+  XD_FLAG_FREE_LISP_OBJECT = 8,
+#if 0
+  /* Suggestions for other possible flags: */
+
+  /* Eliminate XD_UNION_DYNAMIC_SIZE and replace it with a flag, like this. */
+  XD_FLAG_UNION_DYNAMIC_SIZE = 16,
+  /* Require that everyone who uses a description map has to flag it, so
+     that it's easy to tell, when looking through the code, where the
+     description maps are and who's using them.  This might also become
+     necessary if for some reason the format of the description map is
+     expanded and we need to stick a pointer in the second slot (although
+     we could still ensure that the second slot in the first entry was NULL
+     or <0). */
+  XD_FLAG_DESCRIPTION_MAP = 32,
+#endif
 };
 
-#define XD_INDIRECT(val, delta) (-1-((val)|(delta<<8)))
+struct memory_description
+{
+  enum memory_description_type type;
+  Bytecount offset;
+  EMACS_INT data1;
+  const struct sized_memory_description *data2;
+  /* Indicates which subsystems process this entry, plus (potentially) other
+     flags that apply to this entry. */
+  int flags;
+};
 
-#define XD_IS_INDIRECT(code) (code<0)
-#define XD_INDIRECT_VAL(code) ((-1-code) & 255)
-#define XD_INDIRECT_DELTA(code) (((-1-code)>>8) & 255)
+struct sized_memory_description
+{
+  Bytecount size;
+  const struct memory_description *description;
+};
+
+extern const struct sized_memory_description lisp_object_description;
+
+#define XD_INDIRECT(val, delta) (-1 - (Bytecount) ((val) | ((delta) << 8)))
 
-#define XD_DYNARR_DESC(base_type, sub_desc) \
+#define XD_IS_INDIRECT(code) ((code) < 0)
+#define XD_INDIRECT_VAL(code) ((-1 - (code)) & 255)
+#define XD_INDIRECT_DELTA(code) ((-1 - (code)) >> 8)
+
+#define XD_DYNARR_DESC(base_type, sub_desc)				      \
   { XD_STRUCT_PTR, offsetof (base_type, base), XD_INDIRECT(1, 0), sub_desc }, \
-  { XD_INT,        offsetof (base_type, cur) }, \
-  { XD_INT_RESET,  offsetof (base_type, max), XD_INDIRECT(1, 0) }
+  { XD_INT,        offsetof (base_type, cur) },				      \
+  { XD_INT_RESET,  offsetof (base_type, max), XD_INDIRECT(1, 0) }	      \
+
 
 /* DEFINE_LRECORD_IMPLEMENTATION is for objects with constant size.
    DEFINE_LRECORD_SEQUENCE_IMPLEMENTATION is for objects whose size varies.
@@ -656,7 +870,6 @@
 #endif
 
 
-#ifdef USE_KKCC
 #define DEFINE_BASIC_LRECORD_IMPLEMENTATION(name,c_name,dumpable,marker,printer,nuker,equal,hash,desc,structtype) \
 DEFINE_BASIC_LRECORD_IMPLEMENTATION_WITH_PROPS(name,c_name,dumpable,marker,printer,nuker,equal,hash,desc,0,0,0,0,structtype)
 
@@ -679,7 +892,7 @@
 MAKE_LRECORD_IMPLEMENTATION(name,c_name,dumpable,marker,printer,nuker,equal,hash,desc,getprop,putprop,remprop,plist,0,sizer,0,structtype)
 
 #define MAKE_LRECORD_IMPLEMENTATION(name,c_name,dumpable,marker,printer,nuker,equal,hash,desc,getprop,putprop,remprop,plist,size,sizer,basic_p,structtype) \
-DECLARE_ERROR_CHECK_TYPES(c_name, structtype)			\
+DECLARE_ERROR_CHECK_TYPES(c_name, structtype)				\
 const struct lrecord_implementation lrecord_##c_name =			\
   { name, dumpable, marker, printer, nuker, equal, hash, desc,		\
     getprop, putprop, remprop, plist, size, sizer,			\
@@ -698,64 +911,13 @@
 MAKE_EXTERNAL_LRECORD_IMPLEMENTATION(name,c_name,dumpable,marker,printer,nuker,equal,hash,desc,getprop,putprop,remprop,plist,0,sizer,0,structtype)
 
 #define MAKE_EXTERNAL_LRECORD_IMPLEMENTATION(name,c_name,dumpable,marker,printer,nuker,equal,hash,desc,getprop,putprop,remprop,plist,size,sizer,basic_p,structtype) \
-DECLARE_ERROR_CHECK_TYPES(c_name, structtype)			\
+DECLARE_ERROR_CHECK_TYPES(c_name, structtype)				\
 int lrecord_type_##c_name;						\
 struct lrecord_implementation lrecord_##c_name =			\
   { name, dumpable, marker, printer, nuker, equal, hash, desc,		\
     getprop, putprop, remprop, plist, size, sizer,			\
     lrecord_type_last_built_in_type, basic_p }
 
-#else /* not USE_KKCC */
-
-#define DEFINE_BASIC_LRECORD_IMPLEMENTATION(name,c_name,marker,printer,nuker,equal,hash,desc,structtype) \
-DEFINE_BASIC_LRECORD_IMPLEMENTATION_WITH_PROPS(name,c_name,marker,printer,nuker,equal,hash,desc,0,0,0,0,structtype)
-
-#define DEFINE_BASIC_LRECORD_IMPLEMENTATION_WITH_PROPS(name,c_name,marker,printer,nuker,equal,hash,desc,getprop,putprop,remprop,plist,structtype) \
-MAKE_LRECORD_IMPLEMENTATION(name,c_name,0,marker,printer,nuker,equal,hash,desc,getprop,putprop,remprop,plist,sizeof(structtype),0,1,structtype)
-
-#define DEFINE_LRECORD_IMPLEMENTATION(name,c_name,marker,printer,nuker,equal,hash,desc,structtype) \
-DEFINE_LRECORD_IMPLEMENTATION_WITH_PROPS(name,c_name,marker,printer,nuker,equal,hash,desc,0,0,0,0,structtype)
-
-#define DEFINE_LRECORD_IMPLEMENTATION_WITH_PROPS(name,c_name,marker,printer,nuker,equal,hash,desc,getprop,putprop,remprop,plist,structtype) \
-MAKE_LRECORD_IMPLEMENTATION(name,c_name,0,marker,printer,nuker,equal,hash,desc,getprop,putprop,remprop,plist,sizeof (structtype),0,0,structtype)
-
-#define DEFINE_LRECORD_SEQUENCE_IMPLEMENTATION(name,c_name,marker,printer,nuker,equal,hash,desc,sizer,structtype) \
-DEFINE_LRECORD_SEQUENCE_IMPLEMENTATION_WITH_PROPS(name,c_name,marker,printer,nuker,equal,hash,desc,0,0,0,0,sizer,structtype)
-
-#define DEFINE_BASIC_LRECORD_SEQUENCE_IMPLEMENTATION(name,c_name,marker,printer,nuker,equal,hash,desc,sizer,structtype) \
-MAKE_LRECORD_IMPLEMENTATION(name,c_name,0,marker,printer,nuker,equal,hash,desc,0,0,0,0,0,sizer,1,structtype)
-
-#define DEFINE_LRECORD_SEQUENCE_IMPLEMENTATION_WITH_PROPS(name,c_name,marker,printer,nuker,equal,hash,desc,getprop,putprop,remprop,plist,sizer,structtype) \
-MAKE_LRECORD_IMPLEMENTATION(name,c_name,0,marker,printer,nuker,equal,hash,desc,getprop,putprop,remprop,plist,0,sizer,0,structtype)
-
-#define MAKE_LRECORD_IMPLEMENTATION(name,c_name,dumpable,marker,printer,nuker,equal,hash,desc,getprop,putprop,remprop,plist,size,sizer,basic_p,structtype) \
-DECLARE_ERROR_CHECK_TYPES(c_name, structtype)			\
-const struct lrecord_implementation lrecord_##c_name =			\
-  { name, dumpable, marker, printer, nuker, equal, hash, desc,		\
-    getprop, putprop, remprop, plist, size, sizer,			\
-    lrecord_type_##c_name, basic_p }
-
-#define DEFINE_EXTERNAL_LRECORD_IMPLEMENTATION(name,c_name,marker,printer,nuker,equal,hash,desc,structtype) \
-DEFINE_EXTERNAL_LRECORD_IMPLEMENTATION_WITH_PROPS(name,c_name,marker,printer,nuker,equal,hash,desc,0,0,0,0,structtype)
-
-#define DEFINE_EXTERNAL_LRECORD_IMPLEMENTATION_WITH_PROPS(name,c_name,marker,printer,nuker,equal,hash,desc,getprop,putprop,remprop,plist,structtype) \
-MAKE_EXTERNAL_LRECORD_IMPLEMENTATION(name,c_name,0,marker,printer,nuker,equal,hash,desc,getprop,putprop,remprop,plist,sizeof (structtype),0,0,structtype)
-
-#define DEFINE_EXTERNAL_LRECORD_SEQUENCE_IMPLEMENTATION(name,c_name,marker,printer,nuker,equal,hash,desc,sizer,structtype) \
-DEFINE_EXTERNAL_LRECORD_SEQUENCE_IMPLEMENTATION_WITH_PROPS(name,c_name,marker,printer,nuker,equal,hash,desc,0,0,0,0,sizer,structtype)
-
-#define DEFINE_EXTERNAL_LRECORD_SEQUENCE_IMPLEMENTATION_WITH_PROPS(name,c_name,marker,printer,nuker,equal,hash,desc,getprop,putprop,remprop,plist,sizer,structtype) \
-MAKE_EXTERNAL_LRECORD_IMPLEMENTATION(name,c_name,0,marker,printer,nuker,equal,hash,desc,getprop,putprop,remprop,plist,0,sizer,0,structtype)
-
-#define MAKE_EXTERNAL_LRECORD_IMPLEMENTATION(name,c_name,dumpable,marker,printer,nuker,equal,hash,desc,getprop,putprop,remprop,plist,size,sizer,basic_p,structtype) \
-DECLARE_ERROR_CHECK_TYPES(c_name, structtype)			\
-int lrecord_type_##c_name;						\
-struct lrecord_implementation lrecord_##c_name =			\
-  { name, dumpable, marker, printer, nuker, equal, hash, desc,		\
-    getprop, putprop, remprop, plist, size, sizer,			\
-    lrecord_type_last_built_in_type, basic_p }
-#endif /* not USE_KKCC */
-
 extern Lisp_Object (*lrecord_markers[]) (Lisp_Object);
 
 #define INIT_LRECORD_IMPLEMENTATION(type) do {				\
@@ -809,17 +971,24 @@
    4. Create the methods for your object.  Note that technically you don't
    need any, but you will almost always want at least a mark method.
 
-   5. Define your object with DEFINE_LRECORD_IMPLEMENTATION() or some
+   4. Create the data layout description for your object.  See
+   toolbar_button_description below; the comment above in `struct lrecord',
+   describing the purpose of the descriptions; and comments elsewhere in
+   this file describing the exact syntax of the description structures.
+
+   6. Define your object with DEFINE_LRECORD_IMPLEMENTATION() or some
    variant.
 
-   6. Include the header file in the .c file where you defined the object.
+   7. Include the header file in the .c file where you defined the object.
 
-   7. Put a call to INIT_LRECORD_IMPLEMENTATION() for the object in the
+   8. Put a call to INIT_LRECORD_IMPLEMENTATION() for the object in the
    .c file's syms_of_foo() function.
 
-   8. Add a type enum for the object to enum lrecord_type, earlier in this
+   9. Add a type enum for the object to enum lrecord_type, earlier in this
    file.
 
+   --ben
+
 An example:
 
 ------------------------------ in toolbar.h -----------------------------
@@ -870,9 +1039,24 @@
 
 ...
 
+static const struct memory_description toolbar_button_description [] = {
+  { XD_LISP_OBJECT, offsetof (struct toolbar_button, next) },
+  { XD_LISP_OBJECT, offsetof (struct toolbar_button, frame) },
+  { XD_LISP_OBJECT, offsetof (struct toolbar_button, up_glyph) },
+  { XD_LISP_OBJECT, offsetof (struct toolbar_button, down_glyph) },
+  { XD_LISP_OBJECT, offsetof (struct toolbar_button, disabled_glyph) },
+  { XD_LISP_OBJECT, offsetof (struct toolbar_button, cap_up_glyph) },
+  { XD_LISP_OBJECT, offsetof (struct toolbar_button, cap_down_glyph) },
+  { XD_LISP_OBJECT, offsetof (struct toolbar_button, cap_disabled_glyph) },
+  { XD_LISP_OBJECT, offsetof (struct toolbar_button, callback) },
+  { XD_LISP_OBJECT, offsetof (struct toolbar_button, enabled_p) },
+  { XD_LISP_OBJECT, offsetof (struct toolbar_button, help_string) },
+  { XD_END }
+};
+
 static Lisp_Object
 mark_toolbar_button (Lisp_Object obj)
-{
+\{
   struct toolbar_button *data = XTOOLBAR_BUTTON (obj);
   mark_object (data->next);
   mark_object (data->frame);
@@ -891,8 +1075,9 @@
    as internal_object_printer instead of 0. ]]
 
 DEFINE_LRECORD_IMPLEMENTATION ("toolbar-button", toolbar_button,
-			       mark_toolbar_button, 0,
-			       0, 0, 0, 0, struct toolbar_button);
+ 			       0, mark_toolbar_button, 0, 0, 0, 0,
+                               toolbar_button_description,
+ 			       struct toolbar_button);
 
 ...
 
@@ -919,6 +1104,9 @@
   ...
 };
 
+
+--ben
+
 */
 
 /*
@@ -1051,21 +1239,117 @@
    dead_wrong_type_argument (predicate, x);		\
  } while (0)
 
+/*-------------------------- lcrecord-list -----------------------------*/
+
+struct lcrecord_list
+{
+  struct lcrecord_header header;
+  Lisp_Object free;
+  Elemcount size;
+  const struct lrecord_implementation *implementation;
+};
+
+DECLARE_LRECORD (lcrecord_list, struct lcrecord_list);
+#define XLCRECORD_LIST(x) XRECORD (x, lcrecord_list, struct lcrecord_list)
+#define wrap_lcrecord_list(p) wrap_record (p, lcrecord_list)
+#define LCRECORD_LISTP(x) RECORDP (x, lcrecord_list)
+/* #define CHECK_LCRECORD_LIST(x) CHECK_RECORD (x, lcrecord_list)
+   Lcrecord lists should never escape to the Lisp level, so
+   functions should not be doing this. */
+
 /* Various ways of allocating lcrecords.  All bytes (except lcrecord
-   header) are zeroed in returned structure. */
+   header) are zeroed in returned structure.
+
+   See above for a discussion of the difference between plain lrecords and
+   lrecords.  lcrecords themselves are divided into three types: (1)
+   auto-managed, (2) hand-managed, and (3) unmanaged.  "Managed" refers to
+   using a special object called an lcrecord-list to keep track of freed
+   lcrecords, which can freed with free_lcrecord() or the like and later be
+   recycled when a new lcrecord is required, rather than requiring new
+   malloc().  Thus, allocation of lcrecords can be very
+   cheap. (Technically, the lcrecord-list manager could divide up large
+   chunks of memory and allocate out of that, mimicking what happens with
+   lrecords.  At that point, however, we'd want to rethink the whole
+   division between lrecords and lcrecords.) 
+
+   NOTE: There is a fundamental limitation of lcrecord-lists, which is that
+   they only handle blocks of a particular, fixed size.  Thus, objects that
+   can be of varying sizes need to do various tricks.  These considerations
+   in particular dictate the various types of management:
+
+   -- "Auto-managed" means that you just go ahead and allocate the lcrecord
+   whenever you want, using alloc_lcrecord_type(), and the appropriate
+   lcrecord-list manager is automatically created.  To free, you just call
+   "free_lcrecord()" and the appropriate lcrecord-list manager is
+   automatically located and called.  The limitation here of course is that
+   all your objects are of the same size. (#### Eventually we should have a
+   more sophisticated system that tracks the sizes seen and creates one
+   lcrecord list per size, indexed in a hash table.  Usually there are only
+   a limited number of sizes, so this works well.)
 
-void *alloc_lcrecord (Bytecount size,
-		      const struct lrecord_implementation *);
+   -- "Hand-managed" exists because we haven't yet written the more
+   sophisticated scheme for auto-handling different-sized lcrecords, as
+   described in the end of the last paragraph.  In this model, you go ahead
+   and create the lcrecord-list objects yourself for the sizes you will
+   need, using make_lcrecord_list().  Then, create lcrecords using
+   alloc_managed_lcrecord(), passing in the lcrecord-list you created, and
+   free them with free_managed_lcrecord().
+
+   -- "Unmanaged" means you simply allocate lcrecords, period.  No
+   lcrecord-lists, no way to free them.  This may be suitable when the
+   lcrecords are variable-sized and (a) you're too lazy to write the code
+   to hand-manage them, or (b) the objects you create are always or almost
+   always Lisp-visible, and thus there's no point in freeing them (and it
+   wouldn't be safe to do so).  You just create them with
+   basic_alloc_lcrecord(), and that's it.
+
+   --ben
+
+   Here is an in-depth look at the steps required to create a allocate an
+   lcrecord using the hand-managed style.  Since this is the most
+   complicated, you will learn a lot about the other styles as well.  In
+   addition, there is useful general information about what freeing an
+   lcrecord really entails, and what are the precautions:
+
+   1) Create an lcrecord-list object using make_lcrecord_list().  This is
+      often done at initialization.  Remember to staticpro_nodump() this
+      object!  The arguments to make_lcrecord_list() are the same as would be
+      passed to basic_alloc_lcrecord().
 
+   2) Instead of calling basic_alloc_lcrecord(), call alloc_managed_lcrecord()
+      and pass the lcrecord-list earlier created.
+
+   3) When done with the lcrecord, call free_managed_lcrecord().  The
+      standard freeing caveats apply: ** make sure there are no pointers to
+      the object anywhere! **
+
+   4) Calling free_managed_lcrecord() is just like kissing the
+      lcrecord goodbye as if it were garbage-collected.  This means:
+      -- the contents of the freed lcrecord are undefined, and the
+         contents of something produced by alloc_managed_lcrecord()
+	 are undefined, just like for basic_alloc_lcrecord().
+      -- the mark method for the lcrecord's type will *NEVER* be called
+         on freed lcrecords.
+      -- the finalize method for the lcrecord's type will be called
+         at the time that free_managed_lcrecord() is called.
+ */
+
+/* UNMANAGED MODEL: */
+void *basic_alloc_lcrecord (Bytecount size,
+			    const struct lrecord_implementation *);
+
+/* HAND-MANAGED MODEL: */
+Lisp_Object make_lcrecord_list (Elemcount size,
+				const struct lrecord_implementation
+				*implementation);
+Lisp_Object alloc_managed_lcrecord (Lisp_Object lcrecord_list);
+void free_managed_lcrecord (Lisp_Object lcrecord_list, Lisp_Object lcrecord);
+
+/* AUTO-MANAGED MODEL: */
 void *alloc_automanaged_lcrecord (Bytecount size,
 				  const struct lrecord_implementation *);
-
-#define alloc_unmanaged_lcrecord_type(type, lrecord_implementation) \
-  ((type *) alloc_lcrecord (sizeof (type), lrecord_implementation))
-
 #define alloc_lcrecord_type(type, lrecord_implementation) \
   ((type *) alloc_automanaged_lcrecord (sizeof (type), lrecord_implementation))
-
 void free_lcrecord (Lisp_Object rec);
 
 
@@ -1083,6 +1367,192 @@
    memset ((char *) (lcr) + sizeof (struct lcrecord_header), 0,	\
 	   (size) - sizeof (struct lcrecord_header))
 
-#define zero_lcrecord(lcr) zero_sized_lcrecord(lcr, sizeof (*(lcr)))
+#define zero_lcrecord(lcr) zero_sized_lcrecord (lcr, sizeof (*(lcr)))
+
+DECLARE_INLINE_HEADER (
+Bytecount
+detagged_lisp_object_size (const struct lrecord_header *h)
+)
+{
+  const struct lrecord_implementation *imp = LHEADER_IMPLEMENTATION (h);
+
+  return (imp->size_in_bytes_method ?
+	  imp->size_in_bytes_method (h) :
+	  imp->static_size);
+}
+
+DECLARE_INLINE_HEADER (
+Bytecount
+lisp_object_size (Lisp_Object o)
+)
+{
+  return detagged_lisp_object_size (XRECORD_LHEADER (o));
+}
+
+
+/************************************************************************/
+/*		                 Dumping                		*/
+/************************************************************************/
+
+/* dump_add_root_struct_ptr (&var, &desc) dumps the structure pointed to by
+   `var'.  This is for a single relocatable pointer located in the data
+   segment (i.e. the block pointed to is in the heap). */
+#ifdef PDUMP
+void dump_add_root_struct_ptr (void *, const struct sized_memory_description *);
+#else
+#define dump_add_root_struct_ptr(varaddr,descaddr) DO_NOTHING
+#endif
+
+/* dump_add_opaque (&var, size) dumps the opaque static structure `var'.
+   This is for a static block of memory (in the data segment, not the
+   heap), with no relocatable pointers in it. */
+#ifdef PDUMP
+#define dump_add_opaque(varaddr,size) dump_add_root_block (varaddr, size, NULL)
+#else
+#define dump_add_opaque(varaddr,size) DO_NOTHING
+#endif
+
+/* dump_add_root_block (ptr, size, desc) dumps the static structure
+   located at `var' of size SIZE and described by DESC.  This is for a
+   static block of memory (in the data segment, not the heap), with
+   relocatable pointers in it. */
+#ifdef PDUMP
+void dump_add_root_block (const void *ptraddress, Bytecount size,
+			  const struct memory_description *desc);
+#else
+#define dump_add_root_block(ptraddress,desc) DO_NOTHING
+#endif
+
+/* Call dump_add_opaque_int (&int_var) to dump `int_var', of type `int'. */
+#ifdef PDUMP
+#define dump_add_opaque_int(int_varaddr) do {	\
+  int *dao_ = (int_varaddr); /* type check */	\
+  dump_add_opaque (dao_, sizeof (*dao_));	\
+} while (0)
+#else
+#define dump_add_opaque_int(int_varaddr) DO_NOTHING
+#endif
+
+/* Call dump_add_opaque_fixnum (&fixnum_var) to dump `fixnum_var', of type
+   `Fixnum'. */
+#ifdef PDUMP
+#define dump_add_opaque_fixnum(fixnum_varaddr) do {	\
+  Fixnum *dao_ = (fixnum_varaddr); /* type check */	\
+  dump_add_opaque (dao_, sizeof (*dao_));		\
+} while (0)
+#else
+#define dump_add_opaque_fixnum(fixnum_varaddr) DO_NOTHING
+#endif
+
+/* Call dump_add_root_lisp_object (&var) to ensure that var is properly
+   updated after pdump. */
+#ifdef PDUMP
+void dump_add_root_lisp_object (Lisp_Object *);
+#else
+#define dump_add_root_lisp_object(varaddr) DO_NOTHING
+#endif
+
+/* Call dump_add_weak_lisp_object (&var) to ensure that var is properly
+   updated after pdump.  var must point to a linked list of objects out of
+   which some may not be dumped */
+#ifdef PDUMP
+void dump_add_weak_object_chain (Lisp_Object *);
+#else
+#define dump_add_weak_object_chain(varaddr) DO_NOTHING
+#endif
+
+/* Nonzero means Emacs has already been initialized.
+   Used during startup to detect startup of dumped Emacs.  */
+extern int initialized;
+
+#ifdef PDUMP
+
+void pdump_objects_unmark (void);
+void pdump (void);
+int pdump_load (const char *argv0);
+void pdump_backtrace (void);
+extern unsigned int dump_id;
+extern char *pdump_start, *pdump_end;
+
+#define DUMPEDP(adr) ((((char *)(adr)) < pdump_end) && (((char *)(adr)) >= pdump_start))
+
+#else
+#define DUMPEDP(adr) 0
+#endif
+
+/***********************************************************************/
+/*                           data descriptions                         */
+/***********************************************************************/
+
+
+#if defined (USE_KKCC) || defined (PDUMP)
+
+extern int in_pdump;
+
+EMACS_INT lispdesc_indirect_count_1 (EMACS_INT code,
+				     const struct memory_description *idesc,
+				     const void *idata);
+const struct sized_memory_description *lispdesc_indirect_description_1
+ (const void *obj, const struct sized_memory_description *sdesc);
+Bytecount lispdesc_structure_size (const void *obj,
+				   const struct sized_memory_description *
+				   sdesc);
+
+DECLARE_INLINE_HEADER (
+EMACS_INT
+lispdesc_indirect_count (EMACS_INT code,
+			 const struct memory_description *idesc,
+			 const void *idata)
+)
+{
+  if (XD_IS_INDIRECT (code))
+    code = lispdesc_indirect_count_1 (code, idesc, idata);
+  return code;
+}
+
+DECLARE_INLINE_HEADER (
+const struct sized_memory_description *
+lispdesc_indirect_description (const void *obj,
+			       const struct sized_memory_description *sdesc)
+)
+{
+  if (sdesc->description)
+    return sdesc;
+  else
+    return lispdesc_indirect_description_1 (obj, sdesc);
+}
+
+
+/* Do standard XD_UNION processing.  DESC1 is an entry in DESC, which
+   describes the entire data structure.  Returns NULL (do nothing, nothing
+   matched), or a new value for DESC1.  In the latter case, assign to DESC1
+   in your function and goto union_switcheroo. */
+
+DECLARE_INLINE_HEADER (
+const struct memory_description *
+lispdesc_process_xd_union (const struct memory_description *desc1,
+			   const struct memory_description *desc,
+			   const void *data)
+)
+{
+  int count = 0;
+  EMACS_INT variant = lispdesc_indirect_count (desc1->data1, desc,
+					       data);
+  desc1 =
+    lispdesc_indirect_description (data, desc1->data2)->description;
+  
+  for (count = 0; desc1[count].type != XD_END; count++)
+    {
+      if ((desc1[count].flags & XD_FLAG_UNION_DEFAULT_ENTRY) ||
+	  desc1[count].offset == variant)
+	{
+	  return &desc1[count];
+	}
+    }
+
+  return NULL;
+}
+
+#endif /* defined (USE_KKCC) || defined (PDUMP) */
 
 #endif /* INCLUDED_lrecord_h_ */
--- a/src/lstream.c	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/lstream.c	Sun Jan 12 11:08:22 2003 +0000
@@ -114,18 +114,47 @@
   return aligned_sizeof_lstream (((const Lstream *) header)->imp->size);
 }
 
-#ifdef USE_KKCC
+static const struct memory_description lstream_implementation_description_1[]
+= {
+  { XD_END }
+};
+
+const struct sized_memory_description lstream_implementation_description = {
+  sizeof (struct lstream_implementation),
+  lstream_implementation_description_1
+};
+
+static const struct sized_memory_description lstream_extra_description_map[] =
+{
+  { offsetof (Lstream, imp) },
+  { offsetof (struct lstream_implementation, extra_description) },
+  { -1 },
+};
+
+static const struct memory_description lstream_description[] =
+{
+  { XD_STRUCT_PTR, offsetof (Lstream, imp), 1,
+    &lstream_implementation_description },
+  { XD_STRUCT_ARRAY, offsetof (Lstream, data), 1,
+    lstream_extra_description_map },
+  { XD_END }
+};
+
+static const struct memory_description lstream_empty_extra_description_1[] =
+{
+  { XD_END }
+};
+
+const struct sized_memory_description lstream_empty_extra_description = {
+  0, lstream_empty_extra_description_1
+};
+
 DEFINE_LRECORD_SEQUENCE_IMPLEMENTATION ("stream", lstream,
 					0, /*dumpable-flag*/
 					mark_lstream, print_lstream,
-					finalize_lstream, 0, 0, 0,
+					finalize_lstream, 0, 0,
+					lstream_description,
 					sizeof_lstream, Lstream);
-#else /* not USE_KKCC */
-DEFINE_LRECORD_SEQUENCE_IMPLEMENTATION ("stream", lstream,
-					mark_lstream, print_lstream,
-					finalize_lstream, 0, 0, 0,
-					sizeof_lstream, Lstream);
-#endif /* not USE_KKCC */
 
 
 /* Change the buffering of a stream.  See lstream.h.  By default the
@@ -182,7 +211,7 @@
       lstream_type_count++;
     }
 
-  p = XLSTREAM (allocate_managed_lcrecord (Vlstream_free_list[i]));
+  p = XLSTREAM (alloc_managed_lcrecord (Vlstream_free_list[i]));
   /* Zero it out, except the header. */
   memset ((char *) p + sizeof (p->header), '\0',
 	  aligned_sizeof_lstream (imp->size) - sizeof (p->header));
@@ -1299,7 +1328,12 @@
   Bytecount offset, end;
 };
 
-DEFINE_LSTREAM_IMPLEMENTATION ("lisp-string", lisp_string);
+static const struct memory_description lisp_string_lstream_description[] = {
+  { XD_LISP_OBJECT, offsetof (struct lisp_string_stream, obj) },
+  { XD_END }
+};
+
+DEFINE_LSTREAM_IMPLEMENTATION_WITH_DATA ("lisp-string", lisp_string);
 
 Lisp_Object
 make_lisp_string_input_stream (Lisp_Object string, Bytecount offset,
@@ -1590,7 +1624,15 @@
   int flags;
 };
 
-DEFINE_LSTREAM_IMPLEMENTATION ("lisp-buffer", lisp_buffer);
+static const struct memory_description lisp_buffer_lstream_description[] = {
+  { XD_LISP_OBJECT, offsetof (struct lisp_buffer_stream, buffer) },
+  { XD_LISP_OBJECT, offsetof (struct lisp_buffer_stream, orig_start) },
+  { XD_LISP_OBJECT, offsetof (struct lisp_buffer_stream, start) },
+  { XD_LISP_OBJECT, offsetof (struct lisp_buffer_stream, end) },
+  { XD_END }
+};
+
+DEFINE_LSTREAM_IMPLEMENTATION_WITH_DATA ("lisp-buffer", lisp_buffer);
 
 static Lisp_Object
 make_lisp_buffer_stream_1 (struct buffer *buf, Charbpos start, Charbpos end,
@@ -1752,6 +1794,7 @@
   struct lisp_buffer_stream *str =
     LISP_BUFFER_STREAM_DATA (XLSTREAM (stream));
 
+  mark_object (str->orig_start);
   mark_object (str->start);
   mark_object (str->end);
   return str->buffer;
--- a/src/lstream.h	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/lstream.h	Sun Jan 12 11:08:22 2003 +0000
@@ -135,6 +135,11 @@
   Bytecount size; /* Number of additional bytes to be
 		     allocated with this stream.  Access this
 		     data using Lstream_data(). */
+
+  /* Description of the extra data (struct foo_lstream) attached to a
+     coding system. */
+  const struct sized_memory_description *extra_description;
+
   /* Read some data from the stream's end and store it into DATA, which
      can hold SIZE bytes.  Return the number of bytes read.  A return
      value of 0 means no bytes can be read at this time.  This may
@@ -201,7 +206,18 @@
 
 #define DEFINE_LSTREAM_IMPLEMENTATION(name, c_name)	\
   Lstream_implementation lstream_##c_name[1] =		\
-    { { (name), sizeof (struct c_name##_stream) } }
+    { { (name), sizeof (struct c_name##_stream),	\
+      &lstream_empty_extra_description } }
+
+#define DEFINE_LSTREAM_IMPLEMENTATION_WITH_DATA(name, c_name)		      \
+  static const struct sized_memory_description c_name##_lstream_description_0 \
+  = {									      \
+    sizeof (struct c_name##_stream),					      \
+    c_name##_lstream_description					      \
+  };									      \
+  Lstream_implementation lstream_##c_name[1] =				      \
+    { { (name), sizeof (struct c_name##_stream),			      \
+      &c_name##_lstream_description_0 } }
 
 #define DECLARE_LSTREAM(c_name) \
   extern Lstream_implementation lstream_##c_name[]
@@ -244,6 +260,8 @@
   max_align_t data[1];
 };
 
+extern const struct sized_memory_description lstream_empty_extra_description;
+
 #define LSTREAM_TYPE_P(lstr, type) \
   ((lstr)->imp == lstream_##type)
 
--- a/src/marker.c	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/marker.c	Sun Jan 12 11:08:22 2003 +0000
@@ -96,25 +96,18 @@
   return hash;
 }
 
-static const struct lrecord_description marker_description[] = {
-  { XD_LISP_OBJECT, offsetof (Lisp_Marker, next) },
-  { XD_LISP_OBJECT, offsetof (Lisp_Marker, prev) },
+static const struct memory_description marker_description[] = {
+  { XD_LISP_OBJECT, offsetof (Lisp_Marker, next), 0, 0, XD_FLAG_NO_KKCC },
+  { XD_LISP_OBJECT, offsetof (Lisp_Marker, prev), 0, 0, XD_FLAG_NO_KKCC },
   { XD_LISP_OBJECT, offsetof (Lisp_Marker, buffer) },
   { XD_END }
 };
 
-#ifdef USE_KKCC
 DEFINE_BASIC_LRECORD_IMPLEMENTATION ("marker", marker,
 				     1, /*dumpable-flag*/
 				     mark_marker, print_marker, 0,
-				     marker_equal, marker_hash, marker_description,
-				     Lisp_Marker);
-#else /* not USE_KKCC */
-DEFINE_BASIC_LRECORD_IMPLEMENTATION ("marker", marker,
-				     mark_marker, print_marker, 0,
-				     marker_equal, marker_hash, marker_description,
-				     Lisp_Marker);
-#endif /* not USE_KKCC */
+				     marker_equal, marker_hash,
+				     marker_description, Lisp_Marker);
 
 /* Operations on markers. */
 
@@ -403,7 +396,7 @@
 	marker = wrong_type_argument (Qinteger_or_marker_p, marker);
     }
 
-  RETURN_NOT_REACHED (Qnil) /* not reached */
+  RETURN_NOT_REACHED (Qnil); /* not reached */
 }
 
 DEFUN ("copy-marker", Fcopy_marker, 1, 2, 0, /*
--- a/src/mem-limits.h	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/mem-limits.h	Sun Jan 12 11:08:22 2003 +0000
@@ -81,14 +81,7 @@
 #endif /* BSD4_2 */
 
 #ifdef emacs
-/* The important properties of this type are that 1) it's a pointer, and
-   2) arithmetic on it should work as if the size of the object pointed
-   to has a size of 1.  */
-#ifdef __STDC__
 typedef void *POINTER;
-#else
-typedef char *POINTER;
-#endif
 
 #ifndef CYGWIN
 typedef unsigned long SIZE;
--- a/src/menubar-gtk.c	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/menubar-gtk.c	Sun Jan 12 11:08:22 2003 +0000
@@ -1,6 +1,7 @@
 /* Implements an elisp-programmable menubar -- X interface.
    Copyright (C) 1993, 1994 Free Software Foundation, Inc.
    Copyright (C) 1995 Tinker Systems and INS Engineering Corp.
+   Copyright (C) 2002 Ben Wing.
 
 This file is part of XEmacs.
 
@@ -1275,7 +1276,7 @@
 		      GTK_SIGNAL_FUNC (popdown_menu_cb), NULL);
 		      
   gtk_menu_popup (GTK_MENU (menu), NULL, NULL, NULL, NULL,
-		  eev ? eev->event.button.button : 0,
+		  eev ? EVENT_BUTTON_BUTTON (eev) : 0,
 		  eev ? eev->timestamp : GDK_CURRENT_TIME);
 }
 
--- a/src/menubar-msw.c	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/menubar-msw.c	Sun Jan 12 11:08:22 2003 +0000
@@ -120,85 +120,6 @@
 static HMENU top_level_menu;
 
 /*
- * Translate X accelerator syntax to win32 accelerator syntax.
- * accel = (Ichar*) to receive the accelerator character
- *         or NULL to suppress accelerators in the menu or dialog item.
- *
- * %% is replaced with %
- * if accel is NULL:
- *   %_ is removed.
- * if accel is non-NULL:
- *   %_ is replaced with &.
- *   The accelerator character is passed back in *accel.
- *   (If there is no accelerator, it will be added on the first character.)
- *
- */
-
-Lisp_Object
-mswindows_translate_menu_or_dialog_item (Lisp_Object item, Ichar *accel)
-{
-  Bytecount len = XSTRING_LENGTH (item);
-  Ibyte *it = (Ibyte *) ALLOCA (2 * len + 42), *ptr = it;
-
-  memcpy (ptr, XSTRING_DATA (item), len + 1);
-  if (accel)
-    *accel = '\0';
-
-  /* Escape '&' as '&&' */
-  
-  while ((ptr = (Ibyte *) memchr (ptr, '&', len - (ptr - it))) != NULL)
-    {
-      memmove (ptr + 1, ptr, (len - (ptr - it)) + 1);
-      len++;
-      ptr += 2;
-    }
-
-  /* Replace XEmacs accelerator '%_' with Windows accelerator '&'
-     and `%%' with `%'. */
-  ptr = it;
-  while ((ptr = (Ibyte *) memchr (ptr, '%', len - (ptr - it))) != NULL)
-    {
-      if (*(ptr + 1) == '_')
-	{
-	  if (accel)
-	    {
-	      *ptr = '&';
-	      if (!*accel)
-		*accel = DOWNCASE (0, itext_ichar (ptr + 2));
-	      memmove (ptr + 1, ptr + 2, len - (ptr - it + 2) + 1);
-	      len--;
-	    }
-	  else	/* Skip accelerator */
-	    {
-	      memmove (ptr, ptr + 2, len - (ptr - it + 2) + 1);
-	      len -= 2;
-	    }
-	}
-      else if (*(ptr + 1) == '%')
-	{
-	  memmove (ptr + 1, ptr + 2, len - (ptr - it + 2) + 1);
-	  len--;
-	  ptr++;
-	}
-      else	/* % on its own - shouldn't happen */
-	ptr++;
-    }
-
-  if (accel && !*accel)
-    {
-      /* Force a default accelerator */
-      ptr = it;
-      memmove (ptr + 1, ptr, len + 1);
-      *accel = DOWNCASE (0, itext_ichar (ptr + 1));
-      *ptr = '&';
-
-      len++;
-    }
-
-  return make_string (it, len);
-}
-
-/*
  * This returns Windows-style menu item string:
  * "Left Flush\tRight Flush"
  */
@@ -888,13 +809,8 @@
   /* Default is to put the menu at the point (10, 10) in frame */
   if (eev)
     {
-#ifdef USE_KKCC
-      pt.x = XBUTTON_DATA_X (EVENT_DATA (eev));
-      pt.y = XBUTTON_DATA_Y (EVENT_DATA (eev));
-#else /* not USE_KKCC */
-      pt.x = eev->event.button.x;
-      pt.y = eev->event.button.y;
-#endif /* not USE_KKCC */
+      pt.x = EVENT_BUTTON_X (eev);
+      pt.y = EVENT_BUTTON_Y (eev);
       ClientToScreen (FRAME_MSWINDOWS_HANDLE (f), &pt);
     }
   else
--- a/src/menubar-x.c	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/menubar-x.c	Sun Jan 12 11:08:22 2003 +0000
@@ -577,6 +577,7 @@
 
       mdata->id = new_lwlib_id ();
       mdata->last_menubar_buffer = Qnil;
+      mdata->protect_me = Qnil;
       mdata->menubar_contents_up_to_date = 0;
       FRAME_MENUBAR_DATA (f) = wrap_popup_data (mdata);
     }
@@ -672,21 +673,12 @@
     {
       Position shellx, shelly, framex, framey;
       Arg al [2];
-#ifdef USE_KKCC
       btn->time = EVENT_TIMESTAMP (eev);
-      btn->button = XBUTTON_DATA_BUTTON (EVENT_DATA (eev));
+      btn->button = EVENT_BUTTON_BUTTON (eev);
       btn->root = RootWindowOfScreen (XtScreen (daddy));
       btn->subwindow = (Window) NULL;
-      btn->x = XBUTTON_DATA_X (EVENT_DATA (eev));
-      btn->y = XBUTTON_DATA_Y (EVENT_DATA (eev));
-#else /* not USE_KKCC */
-      btn->time = eev->timestamp;
-      btn->button = eev->event.button.button;
-      btn->root = RootWindowOfScreen (XtScreen (daddy));
-      btn->subwindow = (Window) NULL;
-      btn->x = eev->event.button.x;
-      btn->y = eev->event.button.y;
-#endif /* not USE_KKCC */
+      btn->x = EVENT_BUTTON_X (eev);
+      btn->y = EVENT_BUTTON_Y (eev);
       shellx = shelly = 0;
 #ifndef HAVE_WMCOMMAND
       {
@@ -1036,7 +1028,7 @@
       accel = VOID_TO_LISP (entries->accel);
       if (entries->name && !NILP (accel))
 	{
-	  if (event_matches_key_specifier_p (XEVENT (evee), accel))
+	  if (event_matches_key_specifier_p (evee, accel))
 	    {
 	      /* a match! */
 
@@ -1180,7 +1172,7 @@
       t=Vmenu_accelerator_prefix;
       while (!NILP (t)
 	     && !NILP (event0)
-	     && event_matches_key_specifier_p (XEVENT (event0), Fcar (t)))
+	     && event_matches_key_specifier_p (event0, Fcar (t)))
 	{
 	  t = Fcdr (t);
 	  event0 = XEVENT_NEXT (event0);
@@ -1190,7 +1182,7 @@
     }
   else if (NILP (event0))
     return Qnil;
-  else if (event_matches_key_specifier_p (XEVENT (event0), Vmenu_accelerator_prefix))
+  else if (event_matches_key_specifier_p (event0, Vmenu_accelerator_prefix))
     event0 = XEVENT_NEXT (event0);
   else
     return Qnil;
@@ -1200,8 +1192,7 @@
 static Lisp_Object
 menu_accelerator_safe_mod_compare (Lisp_Object cons)
 {
-  return (event_matches_key_specifier_p (XEVENT (XCAR (cons)), XCDR (cons))
-	  ? Qt
+  return (event_matches_key_specifier_p (XCAR (cons), XCDR (cons)) ? Qt
 	  : Qnil);
 }
 
--- a/src/mule-charset.c	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/mule-charset.c	Sun Jan 12 11:08:22 2003 +0000
@@ -66,12 +66,12 @@
 
 struct charset_lookup *chlook;
 
-static const struct lrecord_description charset_lookup_description_1[] = {
+static const struct memory_description charset_lookup_description_1[] = {
   { XD_LISP_OBJECT_ARRAY, offsetof (struct charset_lookup, charset_by_leading_byte), NUM_LEADING_BYTES+4*128*2 },
   { XD_END }
 };
 
-static const struct struct_description charset_lookup_description = {
+static const struct sized_memory_description charset_lookup_description = {
   sizeof (struct charset_lookup),
   charset_lookup_description_1
 };
@@ -175,7 +175,9 @@
     }
 }
 
-static const struct lrecord_description charset_description[] = {
+static const struct memory_description charset_description[] = {
+  { XD_INT, offsetof (Lisp_Charset, dimension) },
+  { XD_INT, offsetof (Lisp_Charset, from_unicode_levels) },
   { XD_LISP_OBJECT, offsetof (Lisp_Charset, name) },
   { XD_LISP_OBJECT, offsetof (Lisp_Charset, doc_string) },
   { XD_LISP_OBJECT, offsetof (Lisp_Charset, registry) },
@@ -183,29 +185,17 @@
   { XD_LISP_OBJECT, offsetof (Lisp_Charset, long_name) },
   { XD_LISP_OBJECT, offsetof (Lisp_Charset, reverse_direction_charset) },
   { XD_LISP_OBJECT, offsetof (Lisp_Charset, ccl_program) },
-#if 0
-  /* #### XD_UNION not yet implemented!  pdump version of XEmacs will
-     not work! */
   { XD_UNION, offsetof (Lisp_Charset, to_unicode_table),
-      XD_INDIRECT (offsetof (Lisp_Charset, dimension), 0),
-      to_unicode_description },
+    XD_INDIRECT (0, 0), &to_unicode_description },
   { XD_UNION, offsetof (Lisp_Charset, from_unicode_table),
-      XD_INDIRECT (offsetof (Lisp_Charset, from_unicode_levels), 0),
-      from_unicode_description },
-#endif
+    XD_INDIRECT (1, 0), &from_unicode_description },
   { XD_END }
 };
 
-#ifdef USE_KKCC
 DEFINE_LRECORD_IMPLEMENTATION ("charset", charset,
 			       1, /* dumpable flag */
                                mark_charset, print_charset, finalize_charset,
 			       0, 0, charset_description, Lisp_Charset);
-#else /* not USE_KKCC */
-DEFINE_LRECORD_IMPLEMENTATION ("charset", charset,
-                               mark_charset, print_charset, finalize_charset,
-			       0, 0, charset_description, Lisp_Charset);
-#endif /* not USE_KKCC */
 /* Make a new charset. */
 /* #### SJT Should generic properties be allowed? */
 static Lisp_Object
@@ -221,7 +211,6 @@
   if (!overwrite)
     {
       cs = alloc_lcrecord_type (Lisp_Charset, &lrecord_charset);
-      zero_lcrecord (cs);
       obj = wrap_charset (cs);
 
       if (final)
@@ -836,7 +825,7 @@
       return CHARSETP (obj) ? XCHARSET_NAME (obj) : obj;
     }
   invalid_constant ("Unrecognized charset property name", prop);
-  RETURN_NOT_REACHED (Qnil)
+  RETURN_NOT_REACHED (Qnil);
 }
 
 DEFUN ("charset-id", Fcharset_id, 1, 1, 0, /*
--- a/src/mule-coding.c	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/mule-coding.c	Sun Jan 12 11:08:22 2003 +0000
@@ -660,7 +660,6 @@
 /* Any ISO-2022-compliant coding system.  Includes JIS, EUC, CTEXT
    (Compound Text, the encoding of selections in X Windows).  See below for
    a complete description of ISO-2022. */
-DEFINE_CODING_SYSTEM_TYPE (iso2022);
 
 /* Flags indicating what we've seen so far when parsing an
    ISO2022 escape sequence. */
@@ -925,46 +924,43 @@
   int current_char_boundary;
 };
 
-static const struct lrecord_description ccs_description_1[] =
+static const struct memory_description ccs_description_1[] =
 {
   { XD_LISP_OBJECT, offsetof (charset_conversion_spec, from_charset) },
   { XD_LISP_OBJECT, offsetof (charset_conversion_spec, to_charset) },
   { XD_END }
 };
 
-static const struct struct_description ccs_description =
+static const struct sized_memory_description ccs_description =
 {
   sizeof (charset_conversion_spec),
   ccs_description_1
 };
 
-static const struct lrecord_description ccsd_description_1[] =
+static const struct memory_description ccsd_description_1[] =
 {
   XD_DYNARR_DESC (charset_conversion_spec_dynarr, &ccs_description),
   { XD_END }
 };
 
-static const struct struct_description ccsd_description =
+static const struct sized_memory_description ccsd_description =
 {
   sizeof (charset_conversion_spec_dynarr),
   ccsd_description_1
 };
 
-static const struct lrecord_description iso2022_coding_system_description[] = {
-  { XD_LISP_OBJECT_ARRAY,
-    coding_system_data_offset + offsetof (struct iso2022_coding_system, 
-					  initial_charset), 4 },
-  { XD_STRUCT_PTR,
-    coding_system_data_offset + offsetof (struct iso2022_coding_system,
-					  input_conv),
+static const struct memory_description iso2022_coding_system_description[] = {
+  { XD_LISP_OBJECT_ARRAY, offsetof (struct iso2022_coding_system, 
+				    initial_charset), 4 },
+  { XD_STRUCT_PTR, offsetof (struct iso2022_coding_system, input_conv),
     1, &ccsd_description },
-  { XD_STRUCT_PTR,
-    coding_system_data_offset + offsetof (struct iso2022_coding_system,
-					  output_conv),
+  { XD_STRUCT_PTR, offsetof (struct iso2022_coding_system, output_conv),
     1, &ccsd_description },
   { XD_END }
 };
 
+DEFINE_CODING_SYSTEM_TYPE_WITH_DATA (iso2022);
+
 /* The following note taken directly from FSF 21.0.103. */
 
 /* The following note describes the coding system ISO2022 briefly.
@@ -2985,7 +2981,6 @@
 /************************************************************************/
 
 /* Converter written in CCL. */
-DEFINE_CODING_SYSTEM_TYPE (ccl);
 
 struct ccl_coding_system
 {
@@ -3010,16 +3005,14 @@
   struct ccl_program ccl;
 };
 
-static const struct lrecord_description ccl_coding_system_description[] = {
-  { XD_LISP_OBJECT,
-    coding_system_data_offset + offsetof (struct ccl_coding_system, 
-					  decode) },
-  { XD_LISP_OBJECT,
-    coding_system_data_offset + offsetof (struct ccl_coding_system, 
-					  encode) },
+static const struct memory_description ccl_coding_system_description[] = {
+  { XD_LISP_OBJECT, offsetof (struct ccl_coding_system, decode) },
+  { XD_LISP_OBJECT, offsetof (struct ccl_coding_system, encode) },
   { XD_END }
 };
 
+DEFINE_CODING_SYSTEM_TYPE_WITH_DATA (ccl);
+
 static void
 ccl_mark (Lisp_Object codesys)
 {
--- a/src/nt.c	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/nt.c	Sun Jan 12 11:08:22 2003 +0000
@@ -122,7 +122,7 @@
   if (!pw)
     return pw;
 
-  if (qxestrcasecmp_i18n (name, pw->pw_name))
+  if (qxestrcasecmp_i18n (name, (Ibyte *) pw->pw_name))
     return NULL;
 
   return pw;
@@ -472,7 +472,7 @@
     for (i = 0; i < countof (env_vars); i++) 
       {
 	if (!egetenv (env_vars[i]) &&
-	    (lpval = nt_get_resource (env_vars[i], &dwType)) != NULL)
+	    (lpval = nt_get_resource ((Ibyte *) env_vars[i], &dwType)) != NULL)
 	  {
 	    if (dwType == REG_EXPAND_SZ)
 	      {
@@ -484,14 +484,14 @@
 		buf = (Extbyte *) ALLOCA (cch * XETCHAR_SIZE);
 		qxeExpandEnvironmentStrings ((Extbyte *) lpval, buf, cch);
 		TSTR_TO_C_STRING (buf, envval);
-		eputenv (env_vars[i], envval);
+		eputenv (env_vars[i], (CIbyte *) envval);
 	      }
 	    else if (dwType == REG_SZ)
 	      {
 		Ibyte *envval;
 
 		TSTR_TO_C_STRING (lpval, envval);
-		eputenv (env_vars[i], envval);
+		eputenv (env_vars[i], (CIbyte *) envval);
 	      }
 
 	    xfree (lpval);
@@ -516,9 +516,9 @@
 	if (STRINGP (str))
 	  {
 	    Ibyte *dat = XSTRING_DATA (str);
-	    if (qxestrncasecmp (dat, "PATH=", 5) == 0)
+	    if (qxestrncasecmp_c (dat, "PATH=", 5) == 0)
 	      memcpy (dat, "PATH=", 5);
-	    else if (qxestrncasecmp (dat, "COMSPEC=", 8) == 0)
+	    else if (qxestrncasecmp_c (dat, "COMSPEC=", 8) == 0)
 	      memcpy (dat, "COMSPEC=", 8);
 	  }
       }
@@ -681,9 +681,12 @@
       DWORD maxcomp;
       DWORD flags;
       Extbyte type[256 * MAX_XETCHAR_SIZE];
+      Extbyte *rootdirext;
+
+      C_STRING_TO_TSTR (root_dir, rootdirext);
 
       /* Info is not cached, or is stale. */
-      if (!qxeGetVolumeInformation (root_dir,
+      if (!qxeGetVolumeInformation (rootdirext,
 				    name, sizeof (name) / XETCHAR_SIZE,
 				    &serialnum,
 				    &maxcomp,
@@ -772,14 +775,14 @@
    removed; it was only for NT 3.1, which we hereby do not support. (NT 3.5
    predates Windows 95!) */
 
-static int
-is_exec (const Ibyte *name)
+int
+mswindows_is_executable (const Ibyte *name)
 {
   Ibyte *p = qxestrrchr (name, '.');
-  return (p != NULL && (qxestrcasecmp (p, ".exe") == 0 ||
-			qxestrcasecmp (p, ".com") == 0 ||
-			qxestrcasecmp (p, ".bat") == 0 ||
-			qxestrcasecmp (p, ".cmd") == 0));
+  return (p != NULL && (qxestrcasecmp_c (p, ".exe") == 0 ||
+			qxestrcasecmp_c (p, ".com") == 0 ||
+			qxestrcasecmp_c (p, ".bat") == 0 ||
+			qxestrcasecmp_c (p, ".cmd") == 0));
 }
 
 /* Emulate the Unix directory procedures opendir, closedir, 
@@ -944,11 +947,11 @@
 static Ibyte *
 read_unc_volume (HANDLE henum)
 {
-  int count;
+  DWORD count;
   int result;
   Extbyte buf[16384];
   Ibyte *ptr;
-  Bytecount bufsize = sizeof (buf);
+  DWORD bufsize = sizeof (buf);
 
   count = 1;
   /* #### we should just be querying the size and then allocating the
@@ -1030,7 +1033,7 @@
 	  return -1;
 	}
     }
-  if ((mode & X_OK) != 0 && !is_exec (path))
+  if ((mode & X_OK) != 0 && !mswindows_is_executable (path))
     {
       errno = EACCES;
       return -1;
@@ -1056,6 +1059,7 @@
 {
   HANDLE fileh;
   int result = -1;
+  Extbyte *oldext;
 
   if (old == NULL || new == NULL)
     {
@@ -1063,8 +1067,8 @@
       return -1;
     }
 
-  C_STRING_TO_TSTR (old, old);
-  fileh = qxeCreateFile (old, 0, 0, NULL, OPEN_EXISTING,
+  C_STRING_TO_TSTR (old, oldext);
+  fileh = qxeCreateFile (oldext, 0, 0, NULL, OPEN_EXISTING,
 			 FILE_FLAG_BACKUP_SEMANTICS, NULL);
   if (fileh != INVALID_HANDLE_VALUE)
     {
@@ -1430,12 +1434,7 @@
   else
     {
 #if 0 /* no way of knowing the filename */
-      Ibyte *p = qxestrrchr (name, '.');
-      if (p != NULL &&
-	  (qxestrcasecmp (p, ".exe") == 0 ||
-	   qxestrcasecmp (p, ".com") == 0 ||
-	   qxestrcasecmp (p, ".bat") == 0 ||
-	   qxestrcasecmp (p, ".cmd") == 0))
+      if (mswindows_is_executable (name))
 	permission |= _S_IEXEC;
 #endif
     }
@@ -1666,7 +1665,7 @@
   
   if (wfd.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY)
     permission |= _S_IEXEC;
-  else if (is_exec (name))
+  else if (mswindows_is_executable (name))
     permission |= _S_IEXEC;
 
   buf->st_mode |= permission | (permission >> 3) | (permission >> 6);
@@ -1880,10 +1879,10 @@
   p = qxestrrchr (filename, '.');
 
   /* We can only identify DOS .com programs from the extension. */
-  if (p && qxestrcasecmp (p, ".com") == 0)
+  if (p && qxestrcasecmp_c (p, ".com") == 0)
     *is_dos_app = TRUE;
-  else if (p && (qxestrcasecmp (p, ".bat") == 0 ||
-		 qxestrcasecmp (p, ".cmd") == 0))
+  else if (p && (qxestrcasecmp_c (p, ".bat") == 0 ||
+		 qxestrcasecmp_c (p, ".cmd") == 0))
     {
       /* A DOS shell script - it appears that CreateProcess is happy to
 	 accept this (somewhat surprisingly); presumably it looks at
@@ -2018,7 +2017,7 @@
   TSTR_TO_C_STRING (shortname, shortint);
   MSWINDOWS_NORMALIZE_FILENAME (shortint);
 
-  return build_string (shortint);
+  return build_intstring (shortint);
 }
 
 
@@ -2041,7 +2040,7 @@
     return Qnil;
 
   canon = mswindows_canonicalize_filename (longname);
-  ret = build_string (canon);
+  ret = build_intstring (canon);
   xfree (canon);
   xfree (longname);
   return ret;
--- a/src/objects-impl.h	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/objects-impl.h	Sun Jan 12 11:08:22 2003 +0000
@@ -103,9 +103,8 @@
   Lisp_Object name;
   Lisp_Object device;
 
-#ifdef USE_KKCC
+  /* See comment in struct console about console variants. */
   enum console_variant color_instance_type;
-#endif /* USE_KKCC */
 
   /* console-type-specific data */
   void *data;
@@ -129,9 +128,8 @@
 			   check this and enforce it as a general policy) */
   Lisp_Object device;
 
-#ifdef USE_KKCC
+  /* See comment in struct console about console variants. */
   enum console_variant font_instance_type;
-#endif /* USE_KKCC */
 
   unsigned short ascent;	/* extracted from `font', or made up */
   unsigned short descent;
--- a/src/objects-tty.c	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/objects-tty.c	Sun Jan 12 11:08:22 2003 +0000
@@ -37,6 +37,24 @@
 Lisp_Object Vtty_dynamic_color_bg;
 #endif
 
+static const struct memory_description tty_color_instance_data_description_1 [] = {
+  { XD_LISP_OBJECT, offsetof (struct tty_color_instance_data, symbol) },
+  { XD_END }
+};
+
+const struct sized_memory_description tty_color_instance_data_description = {
+  sizeof (struct tty_color_instance_data), tty_color_instance_data_description_1
+};
+
+static const struct memory_description tty_font_instance_data_description_1 [] = {
+  { XD_LISP_OBJECT, offsetof (struct tty_font_instance_data, charset) },
+  { XD_END }
+};
+
+const struct sized_memory_description tty_font_instance_data_description = {
+  sizeof (struct tty_font_instance_data), tty_font_instance_data_description_1
+};
+
 DEFUN ("register-tty-color", Fregister_tty_color, 3, 3, 0, /*
 Register COLOR as a recognized TTY color.
 COLOR should be a string.
--- a/src/objects-x.c	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/objects-x.c	Sun Jan 12 11:08:22 2003 +0000
@@ -60,7 +60,7 @@
 {
   int status;
 
-  if (visual->class == DirectColor || visual->class == TrueColor)
+  if (visual->X_CLASSFIELD == DirectColor || visual->X_CLASSFIELD == TrueColor)
     {
       if (XAllocColor (display, colormap, color_def) != 0)
 	{
--- a/src/objects.c	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/objects.c	Sun Jan 12 11:08:22 2003 +0000
@@ -35,9 +35,9 @@
 #include "specifier.h"
 #include "window.h"
 
-#ifdef USE_KKCC
-#include "objects-tty-impl.h"
-#endif /* USE_KKCC */
+#ifdef HAVE_TTY
+#include "console-tty.h"
+#endif
 
 /* Objects that are substituted when an instantiation fails.
    If we leave in the Qunbound value, we will probably get crashes. */
@@ -62,35 +62,25 @@
 
 Lisp_Object Qcolor_instancep;
 
-#ifdef USE_KKCC
-static const struct lrecord_description empty_color_instance_data_description [] = {
-  { XD_END }
-};
-
-static const struct lrecord_description tty_color_instance_data_description [] = {
-  { XD_LISP_OBJECT, offsetof (struct tty_color_instance_data, symbol) },
+static const struct memory_description color_instance_data_description_1 []= {
+#ifdef HAVE_TTY
+  { XD_STRUCT_PTR, tty_console, 1, &tty_color_instance_data_description},
+#endif
   { XD_END }
 };
 
-static const struct struct_description color_instance_data_description []= {
-  { dead_console, empty_color_instance_data_description},
-  { tty_console, tty_color_instance_data_description},
-  { gtk_console, empty_color_instance_data_description},
-  { x_console, empty_color_instance_data_description},
-  { mswindows_console, empty_color_instance_data_description},
-  { stream_console, empty_color_instance_data_description},
-  { XD_END }
+static const struct sized_memory_description color_instance_data_description = {
+  sizeof (void *), color_instance_data_description_1
 };
 
-static const struct lrecord_description color_instance_description[] = {
+static const struct memory_description color_instance_description[] = {
   { XD_INT, offsetof (Lisp_Color_Instance, color_instance_type) },
   { XD_LISP_OBJECT, offsetof (Lisp_Color_Instance, name)},
   { XD_LISP_OBJECT, offsetof (Lisp_Color_Instance, device)},
-  { XD_UNION, offsetof (Lisp_Color_Instance, data), 
-    XD_INDIRECT (0, 0), color_instance_data_description },
+  { XD_UNION, offsetof (Lisp_Color_Instance, data),
+    XD_INDIRECT (0, 0), &color_instance_data_description },
   {XD_END}
 };
-#endif /* USE_KKCC */
 
 static Lisp_Object
 mark_color_instance (Lisp_Object obj)
@@ -156,7 +146,6 @@
 				    LISP_HASH (obj)));
 }
 
-#ifdef USE_KKCC
 DEFINE_LRECORD_IMPLEMENTATION ("color-instance", color_instance,
 			       0, /*dumpable-flag*/
 			       mark_color_instance, print_color_instance,
@@ -164,13 +153,6 @@
 			       color_instance_hash, 
 			       color_instance_description,
 			       Lisp_Color_Instance);
-#else /* not USE_KKCC */
-DEFINE_LRECORD_IMPLEMENTATION ("color-instance", color_instance,
-			       mark_color_instance, print_color_instance,
-			       finalize_color_instance, color_instance_equal,
-			       color_instance_hash, 0,
-			       Lisp_Color_Instance);
-#endif /* not USE_KKCC */
 
 DEFUN ("make-color-instance", Fmake_color_instance, 1, 3, 0, /*
 Return a new `color-instance' object named NAME (a string).
@@ -201,9 +183,7 @@
   c->name = name;
   c->device = device;
   c->data = 0;
-#ifdef USE_KKCC
-  c->color_instance_type = get_console_variant(XDEVICE_TYPE(c->device));
-#endif /* USE_KKCC */
+  c->color_instance_type = get_console_variant (XDEVICE_TYPE (c->device));
 
   retval = MAYBE_INT_DEVMETH (XDEVICE (device), initialize_color_instance,
 			      (c, name, device,
@@ -278,36 +258,27 @@
 
 static Lisp_Object font_instance_truename_internal (Lisp_Object xfont,
 						    Error_Behavior errb);
-#ifdef USE_KKCC
-static const struct lrecord_description empty_font_instance_data_description [] = {
-  { XD_END }
-};
 
-static const struct lrecord_description tty_font_instance_data_description [] = {
-  { XD_LISP_OBJECT, offsetof (struct tty_font_instance_data, charset) },
+static const struct memory_description font_instance_data_description_1 []= {
+#ifdef HAVE_TTY
+  { XD_STRUCT_PTR, tty_console, 1, &tty_font_instance_data_description},
+#endif
   { XD_END }
 };
 
-static const struct struct_description font_instance_data_description []= {
-  { dead_console, empty_font_instance_data_description},
-  { tty_console, tty_font_instance_data_description},
-  { gtk_console, empty_font_instance_data_description},
-  { x_console, empty_font_instance_data_description},
-  { mswindows_console, empty_font_instance_data_description},
-  { stream_console, empty_font_instance_data_description},
-  { XD_END }
+static const struct sized_memory_description font_instance_data_description = {
+  sizeof (void *), font_instance_data_description_1
 };
 
-static const struct lrecord_description font_instance_description[] = {
+static const struct memory_description font_instance_description[] = {
   { XD_INT, offsetof (Lisp_Font_Instance, font_instance_type) },
   { XD_LISP_OBJECT, offsetof (Lisp_Font_Instance, name)},
   { XD_LISP_OBJECT, offsetof (Lisp_Font_Instance, truename)},
   { XD_LISP_OBJECT, offsetof (Lisp_Font_Instance, device)},
   { XD_UNION, offsetof (Lisp_Font_Instance, data), 
-    XD_INDIRECT (0, 0), font_instance_data_description },
-  {XD_END}
+    XD_INDIRECT (0, 0), &font_instance_data_description },
+  { XD_END }
 };
-#endif /* USE_KKCC */
 
 
 static Lisp_Object
@@ -372,18 +343,12 @@
 			depth + 1);
 }
 
-#ifdef USE_KKCC
 DEFINE_LRECORD_IMPLEMENTATION ("font-instance", font_instance,
 			       0, /*dumpable-flag*/
 			       mark_font_instance, print_font_instance,
 			       finalize_font_instance, font_instance_equal,
-			       font_instance_hash, font_instance_description, Lisp_Font_Instance);
-#else /* not USE_KKCC */
-DEFINE_LRECORD_IMPLEMENTATION ("font-instance", font_instance,
-			       mark_font_instance, print_font_instance,
-			       finalize_font_instance, font_instance_equal,
-			       font_instance_hash, 0, Lisp_Font_Instance);
-#endif /* not USE_KKCC */
+			       font_instance_hash, font_instance_description,
+			       Lisp_Font_Instance);
 
 
 DEFUN ("make-font-instance", Fmake_font_instance, 1, 3, 0, /*
@@ -417,9 +382,7 @@
   f->device = device;
 
   f->data = 0;
-#ifdef USE_KKCC
-  f->font_instance_type = get_console_variant(XDEVICE_TYPE(f->device));
-#endif /* USE_KKCC */
+  f->font_instance_type = get_console_variant (XDEVICE_TYPE (f->device));
 
   /* Stick some default values here ... */
   f->ascent = f->height = 1;
@@ -561,7 +524,14 @@
 /****************************************************************************
  Color Object
  ***************************************************************************/
-DEFINE_SPECIFIER_TYPE (color);
+
+static const struct memory_description color_specifier_description[] = {
+  { XD_LISP_OBJECT, offsetof (struct color_specifier, face) },
+  { XD_LISP_OBJECT, offsetof (struct color_specifier, face_property) },
+  { XD_END }
+};
+
+DEFINE_SPECIFIER_TYPE_WITH_DATA (color);
 /* Qcolor defined in general.c */
 
 static void
@@ -731,7 +701,14 @@
 /****************************************************************************
  Font Object
  ***************************************************************************/
-DEFINE_SPECIFIER_TYPE (font);
+
+static const struct memory_description font_specifier_description[] = {
+  { XD_LISP_OBJECT, offsetof (struct font_specifier, face) },
+  { XD_LISP_OBJECT, offsetof (struct font_specifier, face_property) },
+  { XD_END }
+};
+
+DEFINE_SPECIFIER_TYPE_WITH_DATA (font);
 /* Qfont defined in general.c */
 
 static void
@@ -822,9 +799,9 @@
   struct device *d = XDEVICE (device);
   Lisp_Object instance;
   Lisp_Object charset = Qnil;
+#ifdef MULE
   int stage = 0;
 
-#ifdef MULE
   if (!UNBOUNDP (matchspec))
     {
       charset = Fget_charset (XCAR (matchspec));
@@ -842,10 +819,8 @@
 					 Ffont_instance_truename
 					 (instantiator),
 					 0, -1, stage))
+#endif
 	    return instantiator;
-#else
-	  return instantiator;
-#endif
 	}
       instantiator = Ffont_instance_name (instantiator);
     }
@@ -979,7 +954,14 @@
 /*****************************************************************************
  Face Boolean Object
  ****************************************************************************/
-DEFINE_SPECIFIER_TYPE (face_boolean);
+
+static const struct memory_description face_boolean_specifier_description[] = {
+  { XD_LISP_OBJECT, offsetof (struct face_boolean_specifier, face) },
+  { XD_LISP_OBJECT, offsetof (struct face_boolean_specifier, face_property) },
+  { XD_END }
+};
+
+DEFINE_SPECIFIER_TYPE_WITH_DATA (face_boolean);
 Lisp_Object Qface_boolean;
 
 static void
@@ -1154,24 +1136,6 @@
   DEFSYMBOL (Qface_boolean);
 }
 
-static const struct lrecord_description color_specifier_description[] = {
-  { XD_LISP_OBJECT, specifier_data_offset + offsetof (struct color_specifier, face) },
-  { XD_LISP_OBJECT, specifier_data_offset + offsetof (struct color_specifier, face_property) },
-  { XD_END }
-};
-
-static const struct lrecord_description font_specifier_description[] = {
-  { XD_LISP_OBJECT, specifier_data_offset + offsetof (struct font_specifier, face) },
-  { XD_LISP_OBJECT, specifier_data_offset + offsetof (struct font_specifier, face_property) },
-  { XD_END }
-};
-
-static const struct lrecord_description face_boolean_specifier_description[] = {
-  { XD_LISP_OBJECT, specifier_data_offset + offsetof (struct face_boolean_specifier, face) },
-  { XD_LISP_OBJECT, specifier_data_offset + offsetof (struct face_boolean_specifier, face_property) },
-  { XD_END }
-};
-
 void
 specifier_type_create_objects (void)
 {
--- a/src/opaque.c	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/opaque.c	Sun Jan 12 11:08:22 2003 +0000
@@ -72,7 +72,7 @@
 make_opaque (const void *data, Bytecount size)
 {
   Lisp_Opaque *p = (Lisp_Opaque *)
-    alloc_lcrecord (aligned_sizeof_opaque (size), &lrecord_opaque);
+    basic_alloc_lcrecord (aligned_sizeof_opaque (size), &lrecord_opaque);
   p->size = size;
 
   if (data == OPAQUE_CLEAR)
@@ -108,24 +108,16 @@
     return memory_hash (XOPAQUE_DATA (obj), XOPAQUE_SIZE (obj));
 }
 
-static const struct lrecord_description opaque_description[] = {
+static const struct memory_description opaque_description[] = {
   { XD_END }
 };
 
-#ifdef USE_KKCC
 DEFINE_LRECORD_SEQUENCE_IMPLEMENTATION ("opaque", opaque,
 					1, /*dumpable-flag*/
 					0, print_opaque, 0,
 					equal_opaque, hash_opaque,
 					opaque_description,
 					sizeof_opaque, Lisp_Opaque);
-#else /* not USE_KKCC */
-DEFINE_LRECORD_SEQUENCE_IMPLEMENTATION ("opaque", opaque,
-					0, print_opaque, 0,
-					equal_opaque, hash_opaque,
-					opaque_description,
-					sizeof_opaque, Lisp_Opaque);
-#endif /* not USE_KKCC */
 
 /* stuff to handle opaque pointers */
 
@@ -153,23 +145,16 @@
   return (unsigned long) XOPAQUE_PTR (obj)->ptr;
 }
 
-#ifdef USE_KKCC
 DEFINE_LRECORD_IMPLEMENTATION ("opaque-ptr", opaque_ptr,
 			       0, /*dumpable-flag*/
 			       0, print_opaque_ptr, 0,
-			       equal_opaque_ptr, hash_opaque_ptr, 0,
-			       Lisp_Opaque_Ptr);
-#else /* not USE_KKCC */
-DEFINE_LRECORD_IMPLEMENTATION ("opaque-ptr", opaque_ptr,
-			       0, print_opaque_ptr, 0,
-			       equal_opaque_ptr, hash_opaque_ptr, 0,
-			       Lisp_Opaque_Ptr);
-#endif /* not USE_KKCC */
+			       equal_opaque_ptr, hash_opaque_ptr,
+			       0, Lisp_Opaque_Ptr);
 
 Lisp_Object
 make_opaque_ptr (void *val)
 {
-  Lisp_Object res = allocate_managed_lcrecord (Vopaque_ptr_free_list);
+  Lisp_Object res = alloc_managed_lcrecord (Vopaque_ptr_free_list);
   set_opaque_ptr (res, val);
   return res;
 }
@@ -184,7 +169,7 @@
 }
 
 void
-reinit_opaque_once_early (void)
+reinit_opaque_early (void)
 {
   Vopaque_ptr_free_list = make_lcrecord_list (sizeof (Lisp_Opaque_Ptr),
 					      &lrecord_opaque_ptr);
@@ -197,5 +182,5 @@
   INIT_LRECORD_IMPLEMENTATION (opaque);
   INIT_LRECORD_IMPLEMENTATION (opaque_ptr);
 
-  reinit_opaque_once_early ();
+  reinit_opaque_early ();
 }
--- a/src/print.c	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/print.c	Sun Jan 12 11:08:22 2003 +0000
@@ -1376,11 +1376,58 @@
 		    (unsigned long) XPNTR (obj));
 }
 
+enum printing_badness
+{
+  BADNESS_INTEGER_OBJECT,
+  BADNESS_POINTER_OBJECT,
+  BADNESS_NO_TYPE
+};
+
+static void
+printing_major_badness (Lisp_Object printcharfun,
+			Char_ASCII *badness_string, int type, void *val,
+			enum printing_badness badness)
+{
+  Ibyte buf[666];
+
+  switch (badness)
+    {
+    case BADNESS_INTEGER_OBJECT:
+      qxesprintf (buf, "%s %d object %ld", badness_string, type,
+		  (EMACS_INT) val);
+      break;
+
+    case BADNESS_POINTER_OBJECT:
+      qxesprintf (buf, "%s %d object %p", badness_string, type, val);
+      break;
+
+    case BADNESS_NO_TYPE:
+      qxesprintf (buf, "%s object %p", badness_string, val);
+      break;
+    }
+
+  /* Don't abort or signal if called from debug_print() or already
+     crashing */
+  if (!inhibit_non_essential_printing_operations)
+    {
+#ifdef ERROR_CHECK_TYPES
+      abort ();
+#else  /* not ERROR_CHECK_TYPES */
+      if (print_readably)
+	signal_ferror (Qinternal_error, "printing %s", buf);
+#endif /* not ERROR_CHECK_TYPES */
+    }
+  write_fmt_string (printcharfun,
+		    "#<EMACS BUG: %s Save your buffers immediately and "
+		    "please report this bug>", buf);
+}
+
 void
 print_internal (Lisp_Object obj, Lisp_Object printcharfun, int escapeflag)
 {
   /* This function can GC */
   int specdepth;
+  struct gcpro gcpro1, gcpro2;
 
   QUIT;
 
@@ -1389,61 +1436,8 @@
   if (gc_in_progress) return;
 #endif
 
-  /* Try to check for a bogus pointer if we're in a situation where it may
-     be likely.  In such cases, crashing is counterproductive. */
-  if (inhibit_non_essential_printing_operations || print_unbuffered)
-    {
-      if (XTYPE (obj) == Lisp_Type_Record)
-	{
-	  struct lrecord_header *lheader = XRECORD_LHEADER (obj);
-      
-	  if (!debug_can_access_memory (lheader, sizeof (*lheader)))
-	    {
-	      write_fmt_string (printcharfun, "#<EMACS BUG: BAD MEMORY %p>",
-				lheader);
-	      return;
-	    }
-	  else
-	    {
-	      const struct lrecord_implementation *impl;
-
-	      if ((int) lheader->type >= lrecord_type_count)
-		{
-		  write_fmt_string (printcharfun,
-				    "#<EMACS BUG: bad type %d BAD MEMORY %p>",
-				    lheader->type, lheader);
-		  return;
-		}
-
-	      impl = LHEADER_IMPLEMENTATION (lheader);
-	      if (!debug_can_access_memory
-		  (lheader,
-		   (impl->size_in_bytes_method ?
-		    impl->size_in_bytes_method (lheader) :
-		    impl->static_size)))
-		{
-		  write_fmt_string (printcharfun,
-				    "#<EMACS BUG: type %s BAD MEMORY %p>",
-				    impl->name, lheader);
-		  return;
-		}
-
-	      if (STRINGP (obj))
-		{
-		  Lisp_String *l = (Lisp_String *) lheader;
-		  if (!debug_can_access_memory
-		      (l->data_, l->size_))
-		    {
-		      write_fmt_string
-			(printcharfun,
-			 "#<EMACS BUG: %p (BAD STRING DATA %p)>",
-			 lheader, l->data_);
-		      return;
-		    }
-		}
-	    }
-	}
-    }
+  /* Just to be safe ... */
+  GCPRO2 (obj, printcharfun);
 
 #ifdef I18N3
   /* #### Both input and output streams should have a flag associated
@@ -1469,6 +1463,7 @@
 	    *buf = '#';
 	    long_to_string (buf + 1, i);
 	    write_c_string (printcharfun, buf);
+	    UNGCPRO;
 	    return;
 	  }
     }
@@ -1558,51 +1553,101 @@
     case Lisp_Type_Record:
       {
 	struct lrecord_header *lheader = XRECORD_LHEADER (obj);
-	struct gcpro gcpro1, gcpro2;
+
+	/* Try to check for various sorts of bogus pointers if we're in a
+	   situation where it may be likely -- i.e. called from
+	   debug_print() or we're already crashing.  In such cases,
+	   (further) crashing is counterproductive. */
 
-	if (CONSP (obj) || VECTORP(obj))
+	if (inhibit_non_essential_printing_operations &&
+	    !debug_can_access_memory (lheader, sizeof (*lheader)))
+	    {
+	      write_fmt_string (printcharfun, "#<EMACS BUG: BAD MEMORY %p>",
+				lheader);
+	      break;
+	    }
+
+	if (CONSP (obj) || VECTORP (obj))
 	  {
 	    /* If deeper than spec'd depth, print placeholder.  */
 	    if (INTP (Vprint_level)
 		&& print_depth > XINT (Vprint_level))
 	      {
-		GCPRO2 (obj, printcharfun);
 		write_c_string (printcharfun, "...");
-		UNGCPRO;
 		break;
 	      }
 	  }
 
-	GCPRO2 (obj, printcharfun);
+	if (lheader->type == lrecord_type_free)
+	  {
+	    printing_major_badness (printcharfun, "freed lrecord", 0,
+				    lheader, BADNESS_NO_TYPE);
+	    break;
+	  }
+	else if (lheader->type == lrecord_type_undefined)
+	  {
+	    printing_major_badness (printcharfun, "lrecord_type_undefined", 0,
+				    lheader, BADNESS_NO_TYPE);
+	    break;
+	  }
+	else if ((int) (lheader->type) >= lrecord_type_count)
+	  {
+	    printing_major_badness (printcharfun, "illegal lrecord type",
+				    (int) (lheader->type),
+				    lheader, BADNESS_POINTER_OBJECT);
+	    break;
+	  }
+
+	/* Further checks for bad memory in critical situations.  We don't
+	   normally do these because they may be expensive or weird
+	   (e.g. under Unix we typically have to set a SIGSEGV handler and
+	   try to trigger a seg fault). */
+
+	if (inhibit_non_essential_printing_operations)
+	  {
+	    if (!debug_can_access_memory
+		(lheader, detagged_lisp_object_size (lheader)))
+	      {
+		write_fmt_string (printcharfun,
+				  "#<EMACS BUG: type %s BAD MEMORY %p>",
+				  LHEADER_IMPLEMENTATION (lheader)->name,
+				  lheader);
+		break;
+	      }
+
+	    if (STRINGP (obj))
+	      {
+		Lisp_String *l = (Lisp_String *) lheader;
+		if (!debug_can_access_memory (l->data_, l->size_))
+		  {
+		    write_fmt_string
+		      (printcharfun,
+		       "#<EMACS BUG: %p (BAD STRING DATA %p)>",
+		       lheader, l->data_);
+		    break;
+		  }
+	      }
+	  }
+
 	if (LHEADER_IMPLEMENTATION (lheader)->printer)
 	  ((LHEADER_IMPLEMENTATION (lheader)->printer)
 	   (obj, printcharfun, escapeflag));
 	else
 	  default_object_printer (obj, printcharfun, escapeflag);
-	UNGCPRO;
 	break;
       }
 
     default:
       {
-#ifdef ERROR_CHECK_TYPES
-	abort ();
-#else  /* not ERROR_CHECK_TYPES */
 	/* We're in trouble if this happens! */
-	if (print_readably)
-	  signal_error (Qinternal_error, "printing illegal data type #o%03o",
-			make_int (XTYPE (obj)));
-	write_c_string (printcharfun, "#<EMACS BUG: ILLEGAL DATATYPE ");
-	write_fmt_string (printcharfun, "(#o%3o)", (int) XTYPE (obj));
-	write_c_string
-	  (printcharfun,
-	   " Save your buffers immediately and please report this bug>");
-#endif /* not ERROR_CHECK_TYPES */
+	printing_major_badness (printcharfun, "illegal data type", XTYPE (obj),
+				LISP_TO_VOID (obj), BADNESS_INTEGER_OBJECT);
 	break;
       }
     }
 
   unbind_to (specdepth);
+  UNGCPRO;
 }
 
 void
@@ -1906,12 +1951,102 @@
 }
 
 void
+debug_p4 (Lisp_Object obj)
+{
+  inhibit_non_essential_printing_operations = 1;
+  if (STRINGP (obj))
+    debug_out ("\"%s\"", XSTRING_DATA (obj));
+  else if (CONSP (obj))
+    {
+      int first = 1;
+      do {
+	debug_out (first ? "(" : " ");
+	first = 0;
+	debug_p4 (XCAR (obj));
+	obj = XCDR (obj);
+      } while (CONSP (obj));
+      if (NILP (obj))
+	debug_out (")");
+      else
+	{
+	  debug_out (" . ");
+	  debug_p4 (obj);
+	  debug_out (")");
+	}
+    }
+  else if (VECTORP (obj))
+    {
+      int size = XVECTOR_LENGTH (obj);
+      int i;
+      int first = 1;
+
+      for (i = 0; i < size; i++)
+	{
+	  debug_out (first ? "[" : " ");
+	  first = 0;
+	  debug_p4 (XVECTOR_DATA (obj)[i]);
+	  debug_out ("]");
+	}
+    }
+  else if (SYMBOLP (obj))
+    {
+      Lisp_Object name = XSYMBOL_NAME (obj);
+      if (!STRINGP (name))
+	debug_out ("<<bad symbol>>");
+      else
+	debug_out ("%s", XSTRING_DATA (name));
+    }
+  else if (INTP (obj))
+    {
+      debug_out ("%ld", XINT (obj));
+    }
+  else if (FLOATP (obj))
+    {
+      debug_out ("%g", XFLOAT_DATA (obj));
+    }
+  else
+    {
+      struct lrecord_header *header =
+	(struct lrecord_header *) XPNTR (obj);
+
+      if (header->type >= lrecord_type_last_built_in_type)
+	debug_out ("<< bad object type=%d 0x%lx>>", header->type,
+		   (EMACS_INT) header);
+      else
+	debug_out ("#<%s 0x%lx>",
+		   LHEADER_IMPLEMENTATION (header)->name,
+		   LHEADER_IMPLEMENTATION (header)->basic_p ?
+		   (EMACS_INT) header :
+		   ((struct lcrecord_header *) header)->uid);
+    }
+
+  inhibit_non_essential_printing_operations = 0;
+}
+
+void
+debug_p3 (Lisp_Object obj)
+{
+  debug_p4 (obj);
+  inhibit_non_essential_printing_operations = 1;
+  debug_out ("\n");
+  inhibit_non_essential_printing_operations = 0;
+}
+
+void
 debug_print (Lisp_Object debug_print_obj)
 {
   debug_print_no_newline (debug_print_obj);
   debug_out ("\n");
 }
 
+/* Getting tired of typing debug_print() ... */
+void dp (Lisp_Object debug_print_obj);
+void
+dp (Lisp_Object debug_print_obj)
+{
+  debug_print (debug_print_obj);
+}
+
 /* Debugging kludge -- unbuffered */
 /* This function provided for the benefit of the debugger.  */
 void
@@ -1940,6 +2075,14 @@
   unbind_to (specdepth);
 }
 
+/* Getting tired of typing debug_backtrace() ... */
+void db (void);
+void
+db (void)
+{
+  debug_backtrace ();
+}
+
 void
 debug_short_backtrace (int length)
 {
--- a/src/process-nt.c	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/process-nt.c	Sun Jan 12 11:08:22 2003 +0000
@@ -751,12 +751,7 @@
     /* SHGetFileInfo tends to return ERROR_FILE_NOT_FOUND on most
        errors. This leads to bogus error message. */
     DWORD image_type;
-    Ibyte *p = qxestrrchr (XSTRING_DATA (program), '.');
-    if (p != NULL &&
-	(qxestrcasecmp (p, ".exe") == 0 ||
-	 qxestrcasecmp (p, ".com") == 0 ||
-	 qxestrcasecmp (p, ".bat") == 0 ||
-	 qxestrcasecmp (p, ".cmd") == 0))
+    if (mswindows_is_executable (XSTRING_DATA (program)))
       {
 	Extbyte *progext;
 	LISP_STRING_TO_TSTR (program, progext);
@@ -926,11 +921,11 @@
 
       for (i = 0; i < new_length; i++)
 	{
-	  eicat_raw (envout, env[i], strlen (env[i]));
-	  eicat_raw (envout, "\0", 1);
+	  eicat_raw (envout, env[i], qxestrlen (env[i]));
+	  eicat_raw (envout, (Ibyte *) "\0", 1);
 	}
 
-      eicat_raw (envout, "\0", 1);
+      eicat_raw (envout, (Ibyte *) "\0", 1);
       eito_external (envout, Qmswindows_tstr);
       proc_env = eiextdata (envout);
     }
@@ -1210,7 +1205,11 @@
 
   /* First check if HOST is already a numeric address */
   {
-    unsigned long inaddr = inet_addr (XSTRING_DATA (host));
+    Extbyte *hostext;
+    unsigned long inaddr;
+
+    LISP_STRING_TO_EXTERNAL (host, hostext, Qmswindows_host_name_encoding);
+    inaddr = inet_addr (hostext);
     if (inaddr != INADDR_NONE)
       {
 	address->sin_addr.s_addr = inaddr;
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/src/process-slots.h	Sun Jan 12 11:08:22 2003 +0000
@@ -0,0 +1,71 @@
+/* Definitions of marked slots in processes
+   Copyright (C) 1985, 1992, 1993, 1994 Free Software Foundation, Inc.
+   Copyright (C) 2002 Ben Wing.
+
+This file is part of XEmacs.
+
+XEmacs is free software; you can redistribute it and/or modify it
+under the terms of the GNU General Public License as published by the
+Free Software Foundation; either version 2, or (at your option) any
+later version.
+
+XEmacs is distributed in the hope that it will be useful, but WITHOUT
+ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
+FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
+for more details.
+
+You should have received a copy of the GNU General Public License
+along with XEmacs; see the file COPYING.  If not, write to
+the Free Software Foundation, Inc., 59 Temple Place - Suite 330,
+Boston, MA 02111-1307, USA.  */
+
+/* Synched up with: ????.  Split out of procimpl.h. */
+
+/* We define the Lisp_Objects in the process structure in a separate file
+   because there are numerous places we want to iterate over them, such
+   as when defining them in the structure, initializing them, or marking
+   them.
+
+   To use, define MARKED_SLOT before including this file.  No need to
+   undefine; that happens automatically.  */
+
+  /* Name of this process */
+  MARKED_SLOT (name)
+  /* List of command arguments that this process was run with */
+  MARKED_SLOT (command)
+  /* (funcall FILTER PROC STRING)  (if FILTER is non-nil)
+     to dispose of a bunch of chars from the process all at once */
+  MARKED_SLOT (filter)
+  /* (funcall FILTER PROC STRING)  (if FILTER is non-nil)
+     to dispose of a bunch of chars from the stderr of process all at once */
+  MARKED_SLOT (stderr_filter)
+  /* (funcall SENTINEL PROCESS) when process state changes */
+  MARKED_SLOT (sentinel)
+  /* Buffer that output or stderr output is going to */
+  MARKED_SLOT (buffer)
+  MARKED_SLOT (stderr_buffer)
+  /* Marker set to end of last buffer-inserted output from this process */
+  MARKED_SLOT (mark)
+  MARKED_SLOT (stderr_mark)
+  /* Lisp_Int of subprocess' PID, or a cons of
+     service/host if this is really a network connection */
+  MARKED_SLOT (pid)
+
+  /* Symbol indicating status of process.
+     This may be a symbol: run, stop, exit, signal */
+  MARKED_SLOT (status_symbol)
+  /* Low level streams used in input and output, connected to child */
+  MARKED_SLOT (pipe_instream)
+  MARKED_SLOT (pipe_outstream)
+  MARKED_SLOT (pipe_errstream)
+  /* Data end streams, decoding and encoding pipe_* streams */
+  MARKED_SLOT (coding_instream)
+  MARKED_SLOT (coding_outstream)
+  MARKED_SLOT (coding_errstream)
+
+  /* Name of subprocess terminal.  Only needed for Unix but we put it
+     here to avoid complications with KKCC, which needs to know about
+     all of the Lisp objects, including in process-type-specific data. */
+  MARKED_SLOT (tty_name)
+
+#undef MARKED_SLOT
--- a/src/process-unix.c	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/process-unix.c	Sun Jan 12 11:08:22 2003 +0000
@@ -72,7 +72,6 @@
  * Implementation-specific data. Pointed to by Lisp_Process->process_data
  */
 
-#ifndef USE_KKCC
 struct unix_process_data
 {
   /* Non-0 if this is really a ToolTalk channel. */
@@ -85,12 +84,9 @@
   /* Descriptor for the tty which this process is using.
      -1 if we didn't record it (on some systems, there's no need).  */
   int subtty;
-  /* Name of subprocess terminal. */
-  Lisp_Object tty_name;
   /* Non-false if communicating through a pty.  */
   char pty_flag;
 };
-#endif /* not USE_KKCC */
 #define UNIX_DATA(p) ((struct unix_process_data*) ((p)->process_data))
 
 
@@ -475,8 +471,7 @@
       h_errno = 0;
 #endif
 
-      TO_EXTERNAL_FORMAT (LISP_STRING, host, C_STRING_ALLOCA, hostext,
-			  Qnative);
+      LISP_STRING_TO_EXTERNAL (host, hostext, Qunix_host_name_encoding);
 
       /* Some systems can't handle SIGIO/SIGALARM in gethostbyname. */
       slow_down_interrupts ();
@@ -497,12 +492,15 @@
   else
     {
       IN_ADDR numeric_addr;
+      Extbyte *hostext;
+
       /* Attempt to interpret host as numeric inet address */
-      numeric_addr = inet_addr ((char *) XSTRING_DATA (host));
+      LISP_STRING_TO_EXTERNAL (host, hostext, Qunix_host_name_encoding);
+      numeric_addr = inet_addr (hostext);
       if (NUMERIC_ADDR_ERROR)
 	{
 	  maybe_signal_error (Qio_error, "Unknown host", host,
-				   Qprocess, errb);
+			      Qprocess, errb);
 	  return 0;
 	}
 
@@ -810,22 +808,11 @@
 {
   p->process_data = xnew (struct unix_process_data);
 
-  UNIX_DATA(p)->connected_via_filedesc_p = 0;
-  UNIX_DATA(p)->infd   = -1;
-  UNIX_DATA(p)->errfd  = -1;
-  UNIX_DATA(p)->subtty = -1;
-  UNIX_DATA(p)->tty_name = Qnil;
-  UNIX_DATA(p)->pty_flag = 0;
-}
-
-/*
- * Mark any Lisp objects in Lisp_Process->process_data
- */
-
-static void
-unix_mark_process_data (Lisp_Process *proc)
-{
-  mark_object (UNIX_DATA(proc)->tty_name);
+  UNIX_DATA (p)->connected_via_filedesc_p = 0;
+  UNIX_DATA (p)->infd   = -1;
+  UNIX_DATA (p)->errfd  = -1;
+  UNIX_DATA (p)->subtty = -1;
+  UNIX_DATA (p)->pty_flag = 0;
 }
 
 /*
@@ -1324,7 +1311,7 @@
   if (separate_err)
     retry_close (forkerr);
 
-  UNIX_DATA (p)->tty_name = pty_flag ? build_intstring (pty_name) : Qnil;
+  p->tty_name = pty_flag ? build_intstring (pty_name) : Qnil;
 
   /* Notice that SIGCHLD was not blocked. (This is not possible on
      some systems.) No biggie if SIGCHLD occurs right around the
@@ -1341,7 +1328,7 @@
     close_descriptor_pair (forkerr, errchannel);
     errno = save_errno;
     report_process_error ("Opening pty or pipe", Qunbound);
-    RETURN_NOT_REACHED (0)
+    RETURN_NOT_REACHED (0);
   }
 }
 
@@ -1642,12 +1629,13 @@
 
 /* If the subtty field of the process data is not filled in, do so now. */
 static void
-try_to_initialize_subtty (struct unix_process_data *upd)
+try_to_initialize_subtty (Lisp_Process *p)
 {
+  struct unix_process_data *upd = UNIX_DATA (p);
   if (upd->pty_flag
       && (upd->subtty == -1 || ! isatty (upd->subtty))
-      && STRINGP (upd->tty_name))
-    upd->subtty = qxe_open (XSTRING_DATA (upd->tty_name), O_RDWR, 0);
+      && STRINGP (p->tty_name))
+    upd->subtty = qxe_open (XSTRING_DATA (p->tty_name), O_RDWR, 0);
 }
 
 /* Send signal number SIGNO to PROCESS.
@@ -1731,7 +1719,7 @@
   */
   if (current_group)
     {
-      try_to_initialize_subtty (d);
+      try_to_initialize_subtty (p);
 
 #ifdef SIGNALS_VIA_CHARACTERS
       /* If possible, send signals to the entire pgrp
@@ -1803,14 +1791,6 @@
   return kill (pid, sigcode);
 }
 
-/* Return TTY name used to communicate with subprocess. */
-
-static Lisp_Object
-unix_get_tty_name (Lisp_Process *p)
-{
-  return UNIX_DATA (p)->tty_name;
-}
-
 /* Canonicalize host name HOST, and return its canonical form.
    The default implementation just takes HOST for a canonical name. */
 
@@ -2329,7 +2309,6 @@
 process_type_create_unix (void)
 {
   PROCESS_HAS_METHOD (unix, alloc_process_data);
-  PROCESS_HAS_METHOD (unix, mark_process_data);
 #ifdef SIGCHLD
   PROCESS_HAS_METHOD (unix, init_process);
   PROCESS_HAS_METHOD (unix, reap_exited_processes);
@@ -2346,7 +2325,6 @@
   PROCESS_HAS_METHOD (unix, deactivate_process);
   PROCESS_HAS_METHOD (unix, kill_child_process);
   PROCESS_HAS_METHOD (unix, kill_process_by_pid);
-  PROCESS_HAS_METHOD (unix, get_tty_name);
 #ifdef HAVE_SOCKETS
   PROCESS_HAS_METHOD (unix, canonicalize_host_name);
   PROCESS_HAS_METHOD (unix, open_network_stream);
--- a/src/process.c	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/process.c	Sun Jan 12 11:08:22 2003 +0000
@@ -138,69 +138,19 @@
 
 
 
-#ifdef USE_KKCC
-static const struct lrecord_description empty_process_data_description [] = {
-  { XD_END }
-};
-
-static const struct lrecord_description unix_process_data_description [] = {
-  { XD_LISP_OBJECT, offsetof (struct unix_process_data, tty_name) },
-  { XD_END }
-};
-
-static const struct struct_description process_data_description []= {
-  { unix_process, unix_process_data_description},
-  { nt_process, empty_process_data_description},
+static const struct memory_description process_description [] = {
+#define MARKED_SLOT(x) { XD_LISP_OBJECT, offsetof (Lisp_Process, x) },
+#include "process-slots.h"
   { XD_END }
 };
 
-static const struct lrecord_description process_description [] = {
-  { XD_INT, offsetof (Lisp_Process, process_type) },
-  { XD_LISP_OBJECT, offsetof (Lisp_Process, name) },
-  { XD_LISP_OBJECT, offsetof (Lisp_Process, command) },
-  { XD_LISP_OBJECT, offsetof (Lisp_Process, filter) },
-  { XD_LISP_OBJECT, offsetof (Lisp_Process, stderr_filter) },
-  { XD_LISP_OBJECT, offsetof (Lisp_Process, sentinel) },
-  { XD_LISP_OBJECT, offsetof (Lisp_Process, buffer) },
-  { XD_LISP_OBJECT, offsetof (Lisp_Process, mark) },
-  { XD_LISP_OBJECT, offsetof (Lisp_Process, stderr_buffer) },
-  { XD_LISP_OBJECT, offsetof (Lisp_Process, stderr_mark) },
-  { XD_LISP_OBJECT, offsetof (Lisp_Process, pid) },
-  { XD_LISP_OBJECT, offsetof (Lisp_Process, pipe_instream) },
-  { XD_LISP_OBJECT, offsetof (Lisp_Process, pipe_outstream) },
-  { XD_LISP_OBJECT, offsetof (Lisp_Process, pipe_errstream) },
-  { XD_LISP_OBJECT, offsetof (Lisp_Process, coding_instream) },
-  { XD_LISP_OBJECT, offsetof (Lisp_Process, coding_outstream) },
-  { XD_LISP_OBJECT, offsetof (Lisp_Process, coding_errstream) },
-  { XD_LISP_OBJECT, offsetof (Lisp_Process, status_symbol) },
-  { XD_UNION, offsetof (Lisp_Process, process_data), 
-    XD_INDIRECT (0, 0), process_data_description },
-  { XD_END }
-};
-#endif /* USE_KKCC */
-
 static Lisp_Object
 mark_process (Lisp_Object object)
 {
   Lisp_Process *process = XPROCESS (object);
-  MAYBE_PROCMETH (mark_process_data, (process));
-  mark_object (process->name);
-  mark_object (process->command);
-  mark_object (process->filter);
-  mark_object (process->stderr_filter);
-  mark_object (process->sentinel);
-  mark_object (process->buffer);
-  mark_object (process->mark);
-  mark_object (process->stderr_buffer);
-  mark_object (process->stderr_mark);
-  mark_object (process->pid);
-  mark_object (process->pipe_instream);
-  mark_object (process->pipe_outstream);
-  mark_object (process->pipe_errstream);
-  mark_object (process->coding_instream);
-  mark_object (process->coding_outstream);
-  mark_object (process->coding_errstream);
-  return process->status_symbol;
+#define MARKED_SLOT(x) mark_object (process->x);
+#include "process-slots.h"
+  return Qnil;
 }
 
 static void
@@ -255,16 +205,10 @@
     }
 }
 
-#ifdef USE_KKCC
 DEFINE_LRECORD_IMPLEMENTATION ("process", process,
 			       0, /*dumpable-flag*/
                                mark_process, print_process, finalize_process,
                                0, 0, process_description, Lisp_Process);
-#else /* not USE_KKCC */
-DEFINE_LRECORD_IMPLEMENTATION ("process", process,
-                               mark_process, print_process, finalize_process,
-                               0, 0, 0, Lisp_Process);
-#endif /* not USE_KKCC */
 
 /************************************************************************/
 /*                       basic process accessors                        */
@@ -537,6 +481,9 @@
   int i;
   Lisp_Process *p = alloc_lcrecord_type (Lisp_Process, &lrecord_process);
 
+#define MARKED_SLOT(x)	p->x = Qnil;
+#include "process-slots.h"
+
   /* If name is already in use, modify it until it is unused.  */
   name1 = name;
   for (i = 1; ; i++)
@@ -551,43 +498,12 @@
   name = name1;
   p->name = name;
 
-  p->command  = Qnil;
-  p->filter   = Qnil;
-  p->stderr_filter   = Qnil;
-  p->sentinel = Qnil;
-  p->buffer   = Qnil;
   p->mark = Fmake_marker ();
-  p->stderr_buffer   = Qnil;
   p->stderr_mark = Fmake_marker ();
-  p->pid = Qnil;
   p->status_symbol = Qrun;
-  p->exit_code = 0;
-  p->core_dumped = 0;
-  p->filter_does_read = 0;
-  p->kill_without_query = 0;
-  p->separate_stderr = 0;
-  p->in_selected = 0;
-  p->err_selected = 0;
-  p->tick = 0;
-  p->update_tick = 0;
-  p->pipe_instream  = Qnil;
-  p->pipe_outstream = Qnil;
-  p->pipe_errstream = Qnil;
-  p->coding_instream  = Qnil;
-  p->coding_outstream = Qnil;
-  p->coding_errstream = Qnil;
 
-  p->process_data = 0;
   MAYBE_PROCMETH (alloc_process_data, (p));
 
-#ifdef USE_KKCC
-#ifdef HAVE_MS_WINDOWS
-  p->process_type = nt_process;
-#else /*HAVE_MS_WINDOWS*/
-  p->process_type = unix_process;
-#endif /*HAVE_MS_WINDOWS*/
-#endif /* USE_KKCC */
-
   val = wrap_process (p);
 
   Vprocess_list = Fcons (val, Vprocess_list);
@@ -966,8 +882,8 @@
 
   event_stream_select_process (XPROCESS (process), 1, 1);
 
+  NUNGCPRO;
   UNGCPRO;
-  NUNGCPRO;
   return process;
 }
 
@@ -1276,7 +1192,7 @@
        (process))
 {
   CHECK_PROCESS (process);
-  return MAYBE_LISP_PROCMETH (get_tty_name, (XPROCESS (process)));
+  return XPROCESS (process)->tty_name;
 }
 
 DEFUN ("set-process-buffer", Fset_process_buffer, 2, 2, 0, /*
@@ -2010,7 +1926,7 @@
 #undef handle_signal
 
       invalid_constant ("Undefined signal name", signal_);
-      RETURN_NOT_REACHED (0)
+      RETURN_NOT_REACHED (0);
     }
 }
 
--- a/src/process.h	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/process.h	Sun Jan 12 11:08:22 2003 +0000
@@ -39,15 +39,6 @@
 /* struct Lisp_Process is defined in procimpl.h; only process-*.c need
    to know about the guts of it. */
 
-#ifdef USE_KKCC
-enum process_variant
-{
-  unix_process,
-  nt_process
-};
-
-#endif /* USE_KKCC */
-
 DECLARE_LRECORD (process, Lisp_Process);
 #define XPROCESS(x) XRECORD (x, process, Lisp_Process)
 #define wrap_process(p) wrap_record (p, process)
--- a/src/procimpl.h	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/procimpl.h	Sun Jan 12 11:08:22 2003 +0000
@@ -19,6 +19,8 @@
 the Free Software Foundation, Inc., 59 Temple Place - Suite 330,
 Boston, MA 02111-1307, USA.  */
 
+/* Synched up with: ????.  Split out of process.h. */
+
 /* This file must be only included by the process implementation files:
    process-unix.c, process-msw.c etc. The Lisp_Process structure and other
    contents of this file is not exported to the rest of the world */
@@ -36,7 +38,6 @@
 
 struct process_methods
 {
-  void (*mark_process_data) (Lisp_Process *proc);
   void (*print_process_data) (Lisp_Process *proc, Lisp_Object printcharfun);
   void (*finalize_process_data) (Lisp_Process *proc, int for_disksave);
   void (*alloc_process_data) (Lisp_Process *p);
@@ -66,7 +67,6 @@
 			      int current_group, int nomsg);
   int  (*kill_process_by_pid) (int pid, int sigcode);
   int  (*process_send_eof) (Lisp_Object proc);
-  Lisp_Object (*get_tty_name) (Lisp_Process *p);
   void (*deactivate_process) (Lisp_Process *p,
 			      USID* in_usid,
 			      USID* err_usid);
@@ -95,29 +95,6 @@
 struct Lisp_Process
 {
   struct lcrecord_header header;
-  /* Name of this process */
-  Lisp_Object name;
-  /* List of command arguments that this process was run with */
-  Lisp_Object command;
-  /* (funcall FILTER PROC STRING)  (if FILTER is non-nil)
-     to dispose of a bunch of chars from the process all at once */
-  Lisp_Object filter;
-  /* (funcall FILTER PROC STRING)  (if FILTER is non-nil)
-     to dispose of a bunch of chars from the stderr of process all at once */
-  Lisp_Object stderr_filter;
-  /* (funcall SENTINEL PROCESS) when process state changes */
-  Lisp_Object sentinel;
-  /* Buffer that output or stderr output is going to */
-  Lisp_Object buffer, stderr_buffer;
-  /* Marker set to end of last buffer-inserted output from this process */
-  Lisp_Object mark, stderr_mark;
-  /* Lisp_Int of subprocess' PID, or a cons of
-     service/host if this is really a network connection */
-  Lisp_Object pid;
-
-  /* Symbol indicating status of process.
-     This may be a symbol: run, stop, exit, signal */
-  Lisp_Object status_symbol;
 
   /* Exit code if process has terminated,
      signal which stopped/interrupted process
@@ -140,18 +117,9 @@
   int update_tick;
   /* Non-zero if stderr and stdout are separated. */
   char separate_stderr;
-  /* Low level streams used in input and output, connected to child */
-  Lisp_Object pipe_instream;
-  Lisp_Object pipe_outstream;
-  Lisp_Object pipe_errstream;
-  /* Data end streams, decoding and encoding pipe_* streams */
-  Lisp_Object coding_instream;
-  Lisp_Object coding_outstream;
-  Lisp_Object coding_errstream;
 
-#ifdef USE_KKCC
-  enum process_variant process_type;
-#endif /* USE_KKCC */
+#define MARKED_SLOT(x) Lisp_Object x;
+#include "process-slots.h"
 
   /* Implementation dependent data */
   void *process_data;
@@ -186,24 +154,4 @@
 		   const Ibyte *nonrelocatable,
 		   int start, int len);
 
-#ifdef USE_KKCC
-struct unix_process_data
-{
-  /* Non-0 if this is really a ToolTalk channel. */
-  int connected_via_filedesc_p;
-  /* Descriptor by which we read from this process.  -1 for dead process */
-  int infd;
-  /* Descriptor by which we read stderr from this process.  -1 for
-     dead process */
-  int errfd;
-  /* Descriptor for the tty which this process is using.
-     -1 if we didn't record it (on some systems, there's no need).  */
-  int subtty;
-  /* Name of subprocess terminal. */
-  Lisp_Object tty_name;
-  /* Non-false if communicating through a pty.  */
-  char pty_flag;
-};
-#endif /* USE_KKCC */
-
 #endif /* INCLUDED_procimpl_h_ */
--- a/src/rangetab.c	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/rangetab.c	Sun Jan 12 11:08:22 2003 +0000
@@ -131,45 +131,37 @@
   return hash;
 }
 
-static const struct lrecord_description rte_description_1[] = {
+static const struct memory_description rte_description_1[] = {
   { XD_LISP_OBJECT, offsetof (range_table_entry, val) },
   { XD_END }
 };
 
-static const struct struct_description rte_description = {
+static const struct sized_memory_description rte_description = {
   sizeof (range_table_entry),
   rte_description_1
 };
 
-static const struct lrecord_description rted_description_1[] = {
+static const struct memory_description rted_description_1[] = {
   XD_DYNARR_DESC (range_table_entry_dynarr, &rte_description),
   { XD_END }
 };
 
-static const struct struct_description rted_description = {
+static const struct sized_memory_description rted_description = {
   sizeof (range_table_entry_dynarr),
   rted_description_1
 };
 
-static const struct lrecord_description range_table_description[] = {
+static const struct memory_description range_table_description[] = {
   { XD_STRUCT_PTR,  offsetof (Lisp_Range_Table, entries),  1, &rted_description },
   { XD_END }
 };
 
-#ifdef USE_KKCC
 DEFINE_LRECORD_IMPLEMENTATION ("range-table", range_table,
 			       1, /*dumpable-flag*/
                                mark_range_table, print_range_table, 0,
 			       range_table_equal, range_table_hash,
 			       range_table_description,
 			       Lisp_Range_Table);
-#else /* not USE_KKCC */
-DEFINE_LRECORD_IMPLEMENTATION ("range-table", range_table,
-                               mark_range_table, print_range_table, 0,
-			       range_table_equal, range_table_hash,
-			       range_table_description,
-			       Lisp_Range_Table);
-#endif /* not USE_KKCC */
 
 /************************************************************************/
 /*                        Range table operations                        */
--- a/src/realpath.c	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/realpath.c	Sun Jan 12 11:08:22 2003 +0000
@@ -94,7 +94,7 @@
     int err = 0;
     const Ibyte *lastname;
     int count = 0;
-    const Ibyte *tmp;
+    const Ibyte *nn;
     DECLARE_EISTRING (result);
   
     assert (*name);
@@ -118,8 +118,8 @@
 
     /* Count slashes in unc path */
     if (abs_start (name) == 2)
-      for (tmp = name; *tmp; tmp++)
-	if (IS_DIRECTORY_SEP (*tmp))
+      for (nn = name; *nn; nn++)
+	if (IS_DIRECTORY_SEP (*nn))
 	  count++;
 
     if (count >= 2 && count < 4)
--- a/src/redisplay-x.c	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/redisplay-x.c	Sun Jan 12 11:08:22 2003 +0000
@@ -307,7 +307,8 @@
 static void
 x_window_output_end (struct window *w)
 {
-  XFlush (DEVICE_X_DISPLAY (WINDOW_XDEVICE (w)));
+  if (!(check_if_pending_expose_event (WINDOW_XDEVICE (w))))
+    XFlush (DEVICE_X_DISPLAY (WINDOW_XDEVICE (w)));
 }
 
 /*****************************************************************************
@@ -1799,7 +1800,8 @@
     {
       x_redraw_exposed_windows (f->root_window, x, y, width, height);
 
-      XFlush (DEVICE_X_DISPLAY (XDEVICE (f->device)));
+      if (!(check_if_pending_expose_event (FRAME_XDEVICE (f))))
+	XFlush (DEVICE_X_DISPLAY (FRAME_XDEVICE (f)));
     }
   else
     MARK_FRAME_CHANGED (f);
@@ -1970,7 +1972,8 @@
       x_clear_frame_windows (f->root_window);
     }
 
-  XFlush (DEVICE_X_DISPLAY (d));
+  if (!(check_if_pending_expose_event (d)))
+    XFlush (DEVICE_X_DISPLAY (d));
 }
 
 /* briefly swap the foreground and background colors.
--- a/src/redisplay.c	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/redisplay.c	Sun Jan 12 11:08:22 2003 +0000
@@ -308,7 +308,9 @@
                                struct display_block *fixup);
 static void calculate_baseline (pos_data *data);
 
+#ifdef ERROR_CHECK_DISPLAY
 static void sledgehammer_check_redisplay_structs (void);
+#endif /* ERROR_CHECK_DISPLAY */
 
 /* This used to be 10 but 30 seems to give much better performance. */
 #define INIT_MAX_PREEMPTS	30
@@ -515,6 +517,114 @@
 
 #define WINDOW_SCROLLED(w) ((w)->hscroll > 0 || (w)->left_xoffset)
 
+static const struct memory_description rune_dglyph_description_1[] = {
+  { XD_LISP_OBJECT, offsetof (struct rune_dglyph, glyph) },
+  { XD_LISP_OBJECT, offsetof (struct rune_dglyph, extent) },
+  { XD_END }
+};
+
+static const struct sized_memory_description rune_dglyph_description = {
+  sizeof (struct rune_dglyph), rune_dglyph_description_1
+};
+
+static const struct memory_description rune_object_description_1[] = {
+  { XD_STRUCT_ARRAY, RUNE_DGLYPH, 1, &rune_dglyph_description },
+  { XD_END }
+};
+
+static const struct sized_memory_description rune_object_description = {
+  0, rune_object_description_1
+};
+
+static const struct memory_description rune_description_1[] = {
+  { XD_INT, offsetof (rune, type) },
+  { XD_UNION, offsetof (rune, object),
+    XD_INDIRECT (0, 0), &rune_object_description },
+  { XD_END }
+};
+
+static const struct sized_memory_description rune_description = {
+  sizeof (rune),
+  rune_description_1
+};
+
+static const struct memory_description rune_dynarr_description_1[] = {
+  XD_DYNARR_DESC (rune_dynarr, &rune_description),
+  { XD_END }
+};
+
+static const struct sized_memory_description rune_dynarr_description = {
+  sizeof (rune_dynarr),
+  rune_dynarr_description_1
+};
+
+static const struct memory_description display_block_description_1[] = {
+  { XD_STRUCT_PTR, offsetof (display_block, runes),
+    1, &rune_dynarr_description },
+  { XD_END }
+};
+
+static const struct sized_memory_description display_block_description = {
+  sizeof (display_block),
+  display_block_description_1
+};
+
+static const struct memory_description display_block_dynarr_description_1[] = {
+  XD_DYNARR_DESC (display_block_dynarr, &display_block_description),
+  { XD_END }
+};
+
+static const struct sized_memory_description display_block_dynarr_description = {
+  sizeof (display_block_dynarr),
+  display_block_dynarr_description_1
+};
+
+static const struct memory_description glyph_block_description_1[] = {
+  { XD_LISP_OBJECT, offsetof (glyph_block, glyph) },
+  { XD_LISP_OBJECT, offsetof (glyph_block, extent) },
+  { XD_END }
+};
+
+static const struct sized_memory_description glyph_block_description = {
+  sizeof (glyph_block),
+  glyph_block_description_1
+};
+
+static const struct memory_description glyph_block_dynarr_description_1[] = {
+  XD_DYNARR_DESC (glyph_block_dynarr, &glyph_block_description),
+  { XD_END }
+};
+
+static const struct sized_memory_description glyph_block_dynarr_description = {
+  sizeof (glyph_block_dynarr),
+  glyph_block_dynarr_description_1
+};
+
+static const struct memory_description display_line_description_1[] = {
+  { XD_STRUCT_PTR, offsetof (display_line, display_blocks),
+    1, &display_block_dynarr_description },
+  { XD_STRUCT_PTR, offsetof (display_line, left_glyphs),
+    1, &glyph_block_dynarr_description },
+  { XD_STRUCT_PTR, offsetof (display_line, right_glyphs),
+    1, &glyph_block_dynarr_description },
+  { XD_END }
+};
+
+static const struct sized_memory_description display_line_description = {
+  sizeof (display_line),
+  display_line_description_1
+};
+
+static const struct memory_description display_line_dynarr_description_1[] = {
+  XD_DYNARR_DESC (display_line_dynarr, &display_line_description),
+  { XD_END }
+};
+
+const struct sized_memory_description display_line_dynarr_description = {
+  sizeof (display_line_dynarr),
+  display_line_dynarr_description_1
+};
+
 
 /***************************************************************************/
 /*									   */
@@ -9451,6 +9561,8 @@
 
 #endif /* MEMORY_USAGE_STATS */
 
+#ifdef ERROR_CHECK_DISPLAY
+
 static int
 sledgehammer_check_redisplay_structs_1 (struct window *w, void *closure)
 {
@@ -9480,6 +9592,8 @@
   map_windows (0, sledgehammer_check_redisplay_structs_1, NULL);
 }
 
+#endif /* ERROR_CHECK_DISPLAY */
+
 
 /***************************************************************************/
 /*									   */
--- a/src/redisplay.h	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/redisplay.h	Sun Jan 12 11:08:22 2003 +0000
@@ -98,6 +98,23 @@
    #### This should really be made smaller.
 */
 
+struct rune_dglyph
+{
+  Lisp_Object glyph;
+  Lisp_Object extent;	/* extent rune is attached to, if any.
+			   If this is a rune in the modeline
+			   then this might be nil. */
+
+  int ascent;           /* Ascent of this glyph, in pixels. */
+  int descent;          /* Descent of this glyph, in pixels. */
+  int yoffset;          /* Offset from line top to reach glyph top */
+  int xoffset;		/* Number of pixels that need to be
+			   chopped off the left of the glyph.
+			   This has the effect of shifting the
+			   glyph to the left while still clipping
+			   at XPOS. */
+};
+
 typedef struct rune rune;
 struct rune
 {
@@ -139,22 +156,7 @@
     /* #### Glyphs are rare. Is it really necessary to waste 8 bytes on every
        rune for that?! */
     /* DGLYPH */
-    struct
-    {
-      Lisp_Object glyph;
-      Lisp_Object extent;	/* extent rune is attached to, if any.
-                                   If this is a rune in the modeline
-                                   then this might be nil. */
-
-      int ascent;               /* Ascent of this glyph, in pixels. */
-      int descent;              /* Descent of this glyph, in pixels. */
-      int yoffset;              /* Offset from line top to reach glyph top */
-      int xoffset;		/* Number of pixels that need to be
-				   chopped off the left of the glyph.
-				   This has the effect of shifting the
-				   glyph to the left while still clipping
-				   at XPOS. */
-    } dglyph;
+    struct rune_dglyph dglyph;
 
     /* CHAR */
     struct
@@ -649,6 +651,8 @@
 /*                       redisplay global variables                      */
 /*************************************************************************/
 
+extern const struct sized_memory_description display_line_dynarr_description;
+
 /* redisplay structure used by various utility routines. */
 extern display_line_dynarr *cmotion_display_lines;
 
--- a/src/scrollbar.c	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/scrollbar.c	Sun Jan 12 11:08:22 2003 +0000
@@ -77,13 +77,11 @@
 static void update_scrollbar_instance (struct window *w, int vertical,
 				       struct scrollbar_instance *instance);
 
-#ifdef USE_KKCC
-static const struct lrecord_description scrollbar_instance_description [] = {
+static const struct memory_description scrollbar_instance_description [] = {
   { XD_LISP_OBJECT, offsetof (struct scrollbar_instance, mirror) },
   { XD_LISP_OBJECT, offsetof (struct scrollbar_instance, next) },
   { XD_END }
 };
-#endif /* USE_KKCC */
 
 
 static Lisp_Object
@@ -97,19 +95,12 @@
     return Qnil;
 }
 
-#ifdef USE_KKCC
 DEFINE_LRECORD_IMPLEMENTATION ("scrollbar-instance", scrollbar_instance,
 			       0, /*dumpable-flag*/
 			       mark_scrollbar_instance,
 			       internal_object_printer, 0, 0, 0, 
 			       scrollbar_instance_description,
 			       struct scrollbar_instance);
-#else /* not USE_KKCC */
-DEFINE_LRECORD_IMPLEMENTATION ("scrollbar-instance", scrollbar_instance,
-			       mark_scrollbar_instance,
-			       internal_object_printer, 0, 0, 0, 0,
-			       struct scrollbar_instance);
-#endif /* not USE_KKCC */
 
 static void
 free_scrollbar_instance (struct scrollbar_instance *instance,
@@ -210,7 +201,6 @@
     alloc_lcrecord_type (struct scrollbar_instance,
 			 &lrecord_scrollbar_instance);
 
-  zero_lcrecord (instance);
   MAYBE_DEVMETH (d, create_scrollbar_instance, (f, vertical, instance));
 
   return instance;
--- a/src/specifier.c	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/specifier.c	Sun Jan 12 11:08:22 2003 +0000
@@ -63,24 +63,24 @@
 
 static specifier_type_entry_dynarr *the_specifier_type_entry_dynarr;
 
-static const struct lrecord_description ste_description_1[] = {
+static const struct memory_description ste_description_1[] = {
   { XD_LISP_OBJECT, offsetof (specifier_type_entry, symbol) },
   { XD_STRUCT_PTR,  offsetof (specifier_type_entry, meths), 1,
     &specifier_methods_description },
   { XD_END }
 };
 
-static const struct struct_description ste_description = {
+static const struct sized_memory_description ste_description = {
   sizeof (specifier_type_entry),
   ste_description_1
 };
 
-static const struct lrecord_description sted_description_1[] = {
+static const struct memory_description sted_description_1[] = {
   XD_DYNARR_DESC (specifier_type_entry_dynarr, &ste_description),
   { XD_END }
 };
 
-static const struct struct_description sted_description = {
+static const struct sized_memory_description sted_description = {
   sizeof (specifier_type_entry_dynarr),
   sted_description_1
 };
@@ -368,26 +368,33 @@
 				   : p->methods->extra_data_size);
 }
 
-static const struct lrecord_description specifier_methods_description_1[] = {
+static const struct memory_description specifier_methods_description_1[] = {
   { XD_LISP_OBJECT, offsetof (struct specifier_methods, predicate_symbol) },
   { XD_END }
 };
 
-const struct struct_description specifier_methods_description = {
+const struct sized_memory_description specifier_methods_description = {
   sizeof (struct specifier_methods),
   specifier_methods_description_1
 };
 
-static const struct lrecord_description specifier_caching_description_1[] = {
+static const struct memory_description specifier_caching_description_1[] = {
   { XD_END }
 };
 
-static const struct struct_description specifier_caching_description = {
+static const struct sized_memory_description specifier_caching_description = {
   sizeof (struct specifier_caching),
   specifier_caching_description_1
 };
 
-static const struct lrecord_description specifier_description[] = {
+static const struct sized_memory_description specifier_extra_description_map[]
+= {
+  { offsetof (Lisp_Specifier, methods) },
+  { offsetof (struct specifier_methods, extra_description) },
+  { -1 },
+};
+
+const struct memory_description specifier_description[] = {
   { XD_STRUCT_PTR,  offsetof (Lisp_Specifier, methods), 1,
     &specifier_methods_description },
   { XD_LO_LINK,     offsetof (Lisp_Specifier, next_specifier) },
@@ -400,14 +407,20 @@
     &specifier_caching_description },
   { XD_LISP_OBJECT, offsetof (Lisp_Specifier, magic_parent) },
   { XD_LISP_OBJECT, offsetof (Lisp_Specifier, fallback) },
-  { XD_SPECIFIER_END }
-};
-
-const struct lrecord_description specifier_empty_extra_description[] = {
+  { XD_STRUCT_ARRAY, offsetof (Lisp_Specifier, data), 1,
+    specifier_extra_description_map },
   { XD_END }
 };
 
-#ifdef USE_KKCC
+static const struct memory_description specifier_empty_extra_description_1[] =
+{
+  { XD_END }
+};
+
+const struct sized_memory_description specifier_empty_extra_description = {
+  0, specifier_empty_extra_description_1
+};
+
 DEFINE_LRECORD_SEQUENCE_IMPLEMENTATION ("specifier", specifier,
 					1, /*dumpable-flag*/
 					mark_specifier, print_specifier,
@@ -416,15 +429,6 @@
 					specifier_description,
 					sizeof_specifier,
 					Lisp_Specifier);
-#else /* not USE_KKCC */
-DEFINE_LRECORD_SEQUENCE_IMPLEMENTATION ("specifier", specifier,
-					mark_specifier, print_specifier,
-					finalize_specifier,
-					specifier_equal, specifier_hash,
-					specifier_description,
-					sizeof_specifier,
-					Lisp_Specifier);
-#endif /* not USE_KKCC */
 
 /************************************************************************/
 /*                       Creating specifiers                            */
@@ -489,7 +493,8 @@
 {
   Lisp_Object specifier;
   Lisp_Specifier *sp = (Lisp_Specifier *)
-    alloc_lcrecord (aligned_sizeof_specifier (data_size), &lrecord_specifier);
+    basic_alloc_lcrecord (aligned_sizeof_specifier (data_size),
+			  &lrecord_specifier);
 
   sp->methods = spec_meths;
   sp->global_specs = Qnil;
@@ -734,7 +739,7 @@
 
   invalid_argument ("Invalid specifier locale type",
 		     locale_type);
-  RETURN_NOT_REACHED (LOCALE_GLOBAL)
+  RETURN_NOT_REACHED (LOCALE_GLOBAL);
 }
 
 Lisp_Object
@@ -1354,7 +1359,7 @@
 
   invalid_constant ("Invalid `how-to-add' flag", how_to_add);
 
-  RETURN_NOT_REACHED (SPEC_PREPEND)
+  RETURN_NOT_REACHED (SPEC_PREPEND);
 }
 
 /* Given a specifier object SPEC, return bodily specifier if SPEC is a
--- a/src/specifier.h	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/specifier.h	Sun Jan 12 11:08:22 2003 +0000
@@ -83,7 +83,7 @@
      same time.
 */
 
-extern const struct struct_description specifier_methods_description;
+extern const struct sized_memory_description specifier_methods_description;
 
 struct specifier_methods
 {
@@ -194,19 +194,20 @@
   void (*after_change_method) (Lisp_Object specifier,
 			       Lisp_Object locale);
 
-  /* Specifier extra data: Specifier objects can have extra data,
-     specific to the type of specifier, stored at the end of the
-     object.  To have this, a specifier declares a structure of type
-     `struct TYPE_specifier' containing the data and uses
-     INITIALIZE_SPECIFIER_TYPE_WITH_DATA instead of
-     INITIALIZE_SPECIFIER_TYPE.  Then, a pointer to the `struct
-     TYPE_specifier' can be obtained from a specifier object using
-     SPECIFIER_TYPE_DATA. */
+  /* Specifier extra data: Specifier objects can have extra data, specific
+     to the type of specifier, stored at the end of the object.  To have
+     this, a specifier declares a structure of type `struct TYPE_specifier'
+     containing the data and uses DEFINE_SPECIFIER_TYPE_WITH_DATA and
+     INITIALIZE_SPECIFIER_TYPE_WITH_DATA instead of the plain versions.
+     Then, a pointer to the `struct TYPE_specifier' can be obtained from a
+     specifier object using SPECIFIER_TYPE_DATA.
 
-  /* Pdump description of the extra data; required, and must be named
-     TYPE_specifier_description.  Initialized when
+     A data description of the extra data must also be provided, in the
+     form of a memory_description named TYPE_specifier_description. */
+
+  /* Description of extra data structure; initialized when
      INITIALIZE_SPECIFIER_TYPE_WITH_DATA is called. */
-  const struct lrecord_description *extra_description;
+  const struct sized_memory_description *extra_description;
 
   /* Size of extra data structure; initialized when
      INITIALIZE_SPECIFIER_TYPE_WITH_DATA is called. */
@@ -276,8 +277,7 @@
 
 /***** Defining new specifier types *****/
 
-#define specifier_data_offset offsetof (Lisp_Specifier, data)
-extern const struct lrecord_description specifier_empty_extra_description[];
+extern const struct sized_memory_description specifier_empty_extra_description;
 
 #ifdef ERROR_CHECK_TYPES
 #define DECLARE_SPECIFIER_TYPE(type)					\
@@ -315,11 +315,19 @@
 #define DEFINE_SPECIFIER_TYPE(type)					\
 struct specifier_methods * type##_specifier_methods
 
+#define DEFINE_SPECIFIER_TYPE_WITH_DATA(type)				    \
+struct specifier_methods * type##_specifier_methods;			    \
+static const struct sized_memory_description type##_specifier_description_0 \
+  = {									    \
+  sizeof (struct type##_specifier),					    \
+  type##_specifier_description						    \
+}
+
 #define INITIALIZE_SPECIFIER_TYPE(type, obj_name, pred_sym) do {	    \
   type##_specifier_methods = xnew_and_zero (struct specifier_methods);	    \
   type##_specifier_methods->name = obj_name;				    \
   type##_specifier_methods->extra_description =				    \
-    specifier_empty_extra_description;					    \
+    &specifier_empty_extra_description;					    \
   defsymbol_nodump (&type##_specifier_methods->predicate_symbol, pred_sym); \
   add_entry_to_specifier_type_list (Q##type, type##_specifier_methods);	    \
   dump_add_root_struct_ptr (&type##_specifier_methods,			    \
@@ -336,7 +344,7 @@
   type##_specifier_methods->extra_data_size =				\
     sizeof (struct type##_specifier);					\
   type##_specifier_methods->extra_description = 			\
-    type##_specifier_description;					\
+    &type##_specifier_description_0;					\
 } while (0)
 
 /* Declare that specifier-type TYPE has method METH; used in
--- a/src/symbols.c	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/symbols.c	Sun Jan 12 11:08:22 2003 +0000
@@ -105,7 +105,7 @@
   }
 }
 
-static const struct lrecord_description symbol_description[] = {
+static const struct memory_description symbol_description[] = {
   { XD_LISP_OBJECT, offsetof (Lisp_Symbol, next) },
   { XD_LISP_OBJECT, offsetof (Lisp_Symbol, name) },
   { XD_LISP_OBJECT, offsetof (Lisp_Symbol, value) },
@@ -136,7 +136,6 @@
   return external_remprop (&XSYMBOL (symbol)->plist, property, 0, ERROR_ME);
 }
 
-#ifdef USE_KKCC
 DEFINE_BASIC_LRECORD_IMPLEMENTATION_WITH_PROPS ("symbol", symbol,
 						1, /*dumpable-flag*/
 						mark_symbol, print_symbol,
@@ -146,16 +145,6 @@
 						symbol_remprop,
 						Fsymbol_plist,
 						Lisp_Symbol);
-#else /* not USE_KKCC */
-DEFINE_BASIC_LRECORD_IMPLEMENTATION_WITH_PROPS ("symbol", symbol,
-						mark_symbol, print_symbol,
-						0, 0, 0, symbol_description,
-						symbol_getprop,
-						symbol_putprop,
-						symbol_remprop,
-						Fsymbol_plist,
-						Lisp_Symbol);
-#endif /* not USE_KKCC */
 
 /**********************************************************************/
 /*                              Intern				      */
@@ -737,11 +726,6 @@
        (symbol, newplist))
 {
   CHECK_SYMBOL (symbol);
-#if 0 /* Inserted for debugging 6/28/1997 -slb */
-  /* Somebody is setting a property list of integer 0, who? */
-  /* Not this way apparently. */
-  if (EQ(newplist, Qzero)) abort();
-#endif
 
   XSYMBOL (symbol)->plist = newplist;
   return newplist;
@@ -941,6 +925,14 @@
    to call follow_varalias_pointers to get the actual
    symbol to operate on.  */
 
+static const struct memory_description symbol_value_buffer_local_description[] = {
+  { XD_LISP_OBJECT, offsetof (struct symbol_value_buffer_local, default_value) },
+  { XD_LISP_OBJECT, offsetof (struct symbol_value_buffer_local, current_value) },
+  { XD_LISP_OBJECT, offsetof (struct symbol_value_buffer_local, current_buffer) },
+  { XD_LISP_OBJECT, offsetof (struct symbol_value_buffer_local, current_alist_element) },
+  { XD_END }
+};
+
 static Lisp_Object
 mark_symbol_value_buffer_local (Lisp_Object obj)
 {
@@ -958,6 +950,14 @@
   return bfwd->current_alist_element;
 }
 
+
+static const struct memory_description symbol_value_lisp_magic_description[] = {
+  { XD_LISP_OBJECT_ARRAY, offsetof (struct symbol_value_lisp_magic, handler), MAGIC_HANDLER_MAX },
+  { XD_LISP_OBJECT_ARRAY, offsetof (struct symbol_value_lisp_magic, harg), MAGIC_HANDLER_MAX },
+  { XD_LISP_OBJECT, offsetof (struct symbol_value_lisp_magic, shadowed) },
+  { XD_END }
+};
+
 static Lisp_Object
 mark_symbol_value_lisp_magic (Lisp_Object obj)
 {
@@ -975,6 +975,12 @@
   return bfwd->shadowed;
 }
 
+static const struct memory_description symbol_value_varalias_description[] = {
+  { XD_LISP_OBJECT, offsetof (struct symbol_value_varalias, aliasee) },
+  { XD_LISP_OBJECT, offsetof (struct symbol_value_varalias, shadowed) },
+  { XD_END }
+};
+
 static Lisp_Object
 mark_symbol_value_varalias (Lisp_Object obj)
 {
@@ -999,30 +1005,10 @@
 		    (long) XPNTR (obj));
 }
 
-static const struct lrecord_description symbol_value_forward_description[] = {
-  { XD_END }
-};
-
-static const struct lrecord_description symbol_value_buffer_local_description[] = {
-  { XD_LISP_OBJECT, offsetof (struct symbol_value_buffer_local, default_value) },
-  { XD_LISP_OBJECT, offsetof (struct symbol_value_buffer_local, current_value) },
-  { XD_LISP_OBJECT, offsetof (struct symbol_value_buffer_local, current_buffer) },
-  { XD_LISP_OBJECT, offsetof (struct symbol_value_buffer_local, current_alist_element) },
+static const struct memory_description symbol_value_forward_description[] = {
   { XD_END }
 };
 
-static const struct lrecord_description symbol_value_lisp_magic_description[] = {
-  { XD_LISP_OBJECT_ARRAY, offsetof (struct symbol_value_lisp_magic, handler), 2*MAGIC_HANDLER_MAX+1 },
-  { XD_END }
-};
-
-static const struct lrecord_description symbol_value_varalias_description[] = {
-  { XD_LISP_OBJECT, offsetof (struct symbol_value_varalias, aliasee) },
-  { XD_LISP_OBJECT, offsetof (struct symbol_value_varalias, shadowed) },
-  { XD_END }
-};
-
-#ifdef USE_KKCC
 DEFINE_LRECORD_IMPLEMENTATION ("symbol-value-forward",
 			       symbol_value_forward,
 			       1, /*dumpable-flag*/
@@ -1055,36 +1041,6 @@
 			       symbol_value_varalias_description,
 			       struct symbol_value_varalias);
 
-#else /* not USE_KKCC */
-
-DEFINE_LRECORD_IMPLEMENTATION ("symbol-value-forward",
-			       symbol_value_forward,
-			       0,
-			       print_symbol_value_magic, 0, 0, 0,
-			       symbol_value_forward_description,
-			       struct symbol_value_forward);
-
-DEFINE_LRECORD_IMPLEMENTATION ("symbol-value-buffer-local",
-			       symbol_value_buffer_local,
-			       mark_symbol_value_buffer_local,
-			       print_symbol_value_magic, 0, 0, 0,
-			       symbol_value_buffer_local_description,
-			       struct symbol_value_buffer_local);
-
-DEFINE_LRECORD_IMPLEMENTATION ("symbol-value-lisp-magic",
-			       symbol_value_lisp_magic,
-			       mark_symbol_value_lisp_magic,
-			       print_symbol_value_magic, 0, 0, 0,
-			       symbol_value_lisp_magic_description,
-			       struct symbol_value_lisp_magic);
-
-DEFINE_LRECORD_IMPLEMENTATION ("symbol-value-varalias",
-			       symbol_value_varalias,
-			       mark_symbol_value_varalias,
-			       print_symbol_value_magic, 0, 0, 0,
-			       symbol_value_varalias_description,
-			       struct symbol_value_varalias);
-#endif /* not USE_KKCC */
 
 /* Getting and setting values of symbols */
 
@@ -2042,7 +1998,7 @@
 				   XCONSOLE (Vselected_console));
     }
 
-  RETURN_NOT_REACHED (Qnil)	/* suppress compiler warning */
+  RETURN_NOT_REACHED (Qnil);	/* suppress compiler warning */
 }
 
 DEFUN ("default-boundp", Fdefault_boundp, 1, 1, 0, /*
@@ -2236,7 +2192,6 @@
       = alloc_lcrecord_type (struct symbol_value_buffer_local,
 			     &lrecord_symbol_value_buffer_local);
     Lisp_Object foo;
-    zero_lcrecord (&bfwd->magic);
     bfwd->magic.type = SYMVAL_BUFFER_LOCAL;
 
     bfwd->default_value = find_symbol_value (variable);
@@ -2344,7 +2299,6 @@
   /* Make sure variable is set up to hold per-buffer values */
   bfwd = alloc_lcrecord_type (struct symbol_value_buffer_local,
 			      &lrecord_symbol_value_buffer_local);
-  zero_lcrecord (&bfwd->magic);
   bfwd->magic.type = SYMVAL_SOME_BUFFER_LOCAL;
 
   bfwd->current_buffer = Qnil;
@@ -2508,7 +2462,7 @@
     default:
       return variable;
     }
-  RETURN_NOT_REACHED(Qnil)	/* suppress compiler warning */
+  RETURN_NOT_REACHED(Qnil);	/* suppress compiler warning */
 }
 
 
@@ -2928,7 +2882,7 @@
   if (EQ (symbol, Qmake_local))      return MAGIC_HANDLER_MAKE_LOCAL;
 
   invalid_constant ("Unrecognized symbol value handler type", symbol);
-  RETURN_NOT_REACHED (MAGIC_HANDLER_MAX)
+  RETURN_NOT_REACHED (MAGIC_HANDLER_MAX);
 }
 
 static enum lisp_magic_handler
@@ -2962,7 +2916,7 @@
   if (abort_if_not_found)
     abort ();
   invalid_argument ("Unrecognized symbol-value function", funsym);
-  RETURN_NOT_REACHED (MAGIC_HANDLER_MAX)
+  RETURN_NOT_REACHED (MAGIC_HANDLER_MAX);
 }
 
 static int
@@ -3064,7 +3018,6 @@
     {
       bfwd = alloc_lcrecord_type (struct symbol_value_lisp_magic,
 				  &lrecord_symbol_value_lisp_magic);
-      zero_lcrecord (&bfwd->magic);
       bfwd->magic.type = SYMVAL_LISP_MAGIC;
       for (i = 0; i < MAGIC_HANDLER_MAX; i++)
 	{
@@ -3201,7 +3154,6 @@
 
   bfwd = alloc_lcrecord_type (struct symbol_value_varalias,
 			      &lrecord_symbol_value_varalias);
-  zero_lcrecord (&bfwd->magic);
   bfwd->magic.type = SYMVAL_VARALIAS;
   bfwd->aliasee = alias;
   bfwd->shadowed = valcontents;
@@ -3303,7 +3255,7 @@
   INIT_LRECORD_IMPLEMENTATION (symbol_value_lisp_magic);
   INIT_LRECORD_IMPLEMENTATION (symbol_value_varalias);
 
-  reinit_symbols_once_early ();
+  reinit_symbols_early ();
 
   /* Bootstrapping problem: Qnil isn't set when make_string_nocopy is
      called the first time. */
@@ -3335,13 +3287,13 @@
   XSYMBOL (Qt)->value = Qt;	/* Veritas aeterna */
   Vquit_flag = Qnil;
 
-  dump_add_root_object (&Qnil);
-  dump_add_root_object (&Qunbound);
-  dump_add_root_object (&Vquit_flag);
+  dump_add_root_lisp_object (&Qnil);
+  dump_add_root_lisp_object (&Qunbound);
+  dump_add_root_lisp_object (&Vquit_flag);
 }
 
 void
-reinit_symbols_once_early (void)
+reinit_symbols_early (void)
 {
 }
 
--- a/src/symsinit.h	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/symsinit.h	Sun Jan 12 11:08:22 2003 +0000
@@ -43,15 +43,15 @@
    stuff, call the reinit() routine, and do some more stuff.) */
 
 void init_alloc_once_early (void);
-void reinit_alloc_once_early (void);
+void reinit_alloc_early (void);
 void init_symbols_once_early (void);
-void reinit_symbols_once_early (void);
+void reinit_symbols_early (void);
 void init_errors_once_early (void);
-void reinit_opaque_once_early (void);
+void reinit_opaque_early (void);
 void init_opaque_once_early (void);
 void init_elhash_once_early (void);
 void init_eistring_once_early (void);
-void reinit_eistring_once_early (void);
+void reinit_eistring_early (void);
 
 /* Reset the Lisp engine.  Called both at dump-time, run-time and
    run-temacs-time; at dump-time, it's called early, before any of the
--- a/src/sysdep.c	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/sysdep.c	Sun Jan 12 11:08:22 2003 +0000
@@ -596,7 +596,7 @@
   if (sh == 0)
     sh = egetenv ("SHELL");
   if (sh == 0)
-    sh = "sh";
+    sh = (Ibyte *) "sh";
 
   C_STRING_TO_EXTERNAL (sh, shext, Qfile_name);
 
@@ -1539,12 +1539,12 @@
 #endif
   if (CONSOLE_TTY_DATA (con)->controlling_terminal)
     {
-      tty.main.c_cc[VINTR] =
-	CONSOLE_QUIT_CHAR (con); /* C-g (usually) gives SIGINT */
+      tty.main.c_cc[VINTR] = /* C-g (usually) gives SIGINT */
+	event_to_character (CONSOLE_QUIT_EVENT (con), 0, 1, 0);
       /* Set up C-g for both SIGQUIT and SIGINT.
 	 We don't know which we will get, but we handle both alike
 	 so which one it really gives us does not matter.  */
-      tty.main.c_cc[VQUIT] = CONSOLE_QUIT_CHAR (con);
+      tty.main.c_cc[VQUIT] = tty.main.c_cc[VINTR];
     }
   else
     {
@@ -1634,7 +1634,7 @@
   /* Note: if not using CBREAK mode, it makes no difference how we
      set this */
   tty.tchars = new_tchars;
-  tty.tchars.t_intrc = CONSOLE_QUIT_CHAR (con);
+  tty.tchars.t_intrc = event_to_character (CONSOLE_QUIT_EVENT (con), 0, 1, 0);
   if (TTY_FLAGS (con).flow_control)
     {
       tty.tchars.t_startc = '\021';
@@ -3329,6 +3329,7 @@
   return ctime_static;
 }
 
+
 /************************************************************************/
 /*                  Emulation of missing functions from wchar.h         */
 /************************************************************************/
@@ -3345,6 +3346,34 @@
   return p - s;
 }
 #endif
+
+/************************************************************************/
+/*                  Emulation of missing functions from string.h        */
+/************************************************************************/
+
+#ifndef HAVE_STRLWR
+char *
+strlwr (char *s)
+{
+  while (*s)
+    {
+      *s = tolower (*s);
+      ++s;
+    }
+}
+#endif
+
+#ifndef HAVE_STRLWR
+char *
+strupr (char *s)
+{
+  while (*s)
+    {
+      *s = toupper (*s);
+      ++s;
+    }
+}
+#endif
 
 
 /************************************************************************/
--- a/src/syswindows.h	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/syswindows.h	Sun Jan 12 11:08:22 2003 +0000
@@ -1070,8 +1070,9 @@
 int mswindows_stat (const Ibyte *path, struct stat *buf);
 int mswindows_fstat (int desc, struct stat *buf);
 time_t mswindows_convert_time (FILETIME ft);
-void mswindows_executable_type (const Ibyte * filename, int * is_dos_app,
-				int * is_cygnus_app);
+int mswindows_is_executable (const Ibyte *filename);
+void mswindows_executable_type (const Ibyte *filename, int *is_dos_app,
+				int *is_cygnus_app);
 Ibyte *mswindows_getdcwd (int drivelet);
 
 /* in process-nt.c */
--- a/src/text.c	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/text.c	Sun Jan 12 11:08:22 2003 +0000
@@ -2867,7 +2867,6 @@
       int delete_count;
       Lisp_Object instream, outstream;
       Lstream *reader, *writer;
-      struct gcpro gcpro1, gcpro2;
 
 #ifdef HAVE_WIN32_CODING_SYSTEMS
     the_hard_way:
@@ -2903,30 +2902,33 @@
       reader = XLSTREAM (instream);
       writer = XLSTREAM (outstream);
       /* decoding_stream will gc-protect outstream */
-      GCPRO2 (instream, outstream);
-
-      while (1)
-        {
-          Bytecount size_in_bytes;
-	  char tempbuf[1024]; /* some random amount */
-
-	  size_in_bytes = Lstream_read (reader, tempbuf, sizeof (tempbuf));
-
-          if (size_in_bytes == 0)
-            break;
-	  else if (size_in_bytes < 0)
-	    signal_error (Qtext_conversion_error,
-			  "Error converting to external format", Qunbound);
-
-	  if (Lstream_write (writer, tempbuf, size_in_bytes) < 0)
-	    signal_error (Qtext_conversion_error,
-			  "Error converting to external format", Qunbound);
-        }
-
-      /* Closing writer will close any stream at the other end of writer. */
-      Lstream_close (writer);
-      Lstream_close (reader);
-      UNGCPRO;
+      {
+	struct gcpro gcpro1, gcpro2;
+	GCPRO2 (instream, outstream);
+
+	while (1)
+	  {
+	    Bytecount size_in_bytes;
+	    char tempbuf[1024]; /* some random amount */
+
+	    size_in_bytes = Lstream_read (reader, tempbuf, sizeof (tempbuf));
+
+	    if (size_in_bytes == 0)
+	      break;
+	    else if (size_in_bytes < 0)
+	      signal_error (Qtext_conversion_error,
+			    "Error converting to external format", Qunbound);
+
+	    if (Lstream_write (writer, tempbuf, size_in_bytes) < 0)
+	      signal_error (Qtext_conversion_error,
+			    "Error converting to external format", Qunbound);
+	  }
+
+	/* Closing writer will close any stream at the other end of writer. */
+	Lstream_close (writer);
+	Lstream_close (reader);
+	UNGCPRO;
+      }
 
       /* The idea is that this function will create no garbage. */
       while (delete_count)
@@ -3056,7 +3058,6 @@
       int delete_count;
       Lisp_Object instream, outstream;
       Lstream *reader, *writer;
-      struct gcpro gcpro1, gcpro2;
 
 #ifdef HAVE_WIN32_CODING_SYSTEMS
     the_hard_way:
@@ -3087,31 +3088,34 @@
 
       reader = XLSTREAM (instream);
       writer = XLSTREAM (outstream);
-      /* outstream will gc-protect its sink stream, if necessary */
-      GCPRO2 (instream, outstream);
-
-      while (1)
-        {
-          Bytecount size_in_bytes;
-	  char tempbuf[1024]; /* some random amount */
-
-	  size_in_bytes = Lstream_read (reader, tempbuf, sizeof (tempbuf));
-
-          if (size_in_bytes == 0)
-            break;
-	  else if (size_in_bytes < 0)
-	    signal_error (Qtext_conversion_error,
-			  "Error converting to internal format", Qunbound);
-
-	  if (Lstream_write (writer, tempbuf, size_in_bytes) < 0)
-	    signal_error (Qtext_conversion_error,
-			  "Error converting to internal format", Qunbound);
-        }
-
-      /* Closing writer will close any stream at the other end of writer. */
-      Lstream_close (writer);
-      Lstream_close (reader);
-      UNGCPRO;
+      {
+	struct gcpro gcpro1, gcpro2;
+	/* outstream will gc-protect its sink stream, if necessary */
+	GCPRO2 (instream, outstream);
+
+	while (1)
+	  {
+	    Bytecount size_in_bytes;
+	    char tempbuf[1024]; /* some random amount */
+
+	    size_in_bytes = Lstream_read (reader, tempbuf, sizeof (tempbuf));
+
+	    if (size_in_bytes == 0)
+	      break;
+	    else if (size_in_bytes < 0)
+	      signal_error (Qtext_conversion_error,
+			    "Error converting to internal format", Qunbound);
+
+	    if (Lstream_write (writer, tempbuf, size_in_bytes) < 0)
+	      signal_error (Qtext_conversion_error,
+			    "Error converting to internal format", Qunbound);
+	  }
+
+	/* Closing writer will close any stream at the other end of writer. */
+	Lstream_close (writer);
+	Lstream_close (reader);
+	UNGCPRO;
+      }
 
       /* The idea is that this function will create no garbage. */
       while (delete_count)
@@ -3616,7 +3620,7 @@
 /************************************************************************/
 
 void
-reinit_eistring_once_early (void)
+reinit_eistring_early (void)
 {
   the_eistring_malloc_zero_init = the_eistring_zero_init;
   the_eistring_malloc_zero_init.mallocp_ = 1;
@@ -3625,7 +3629,7 @@
 void
 init_eistring_once_early (void)
 {
-  reinit_eistring_once_early ();
+  reinit_eistring_early ();
 }
 
 void
@@ -3655,7 +3659,6 @@
   conversion_out_dynarr_list = Dynarr_new2 (Extbyte_dynarr_dynarr,
 					    Extbyte_dynarr *);
 
-  /* #### Olivier, why does this need to be reinitted? */
   for (i = 0; i <= MAX_BYTEBPOS_GAP_SIZE_3; i++)
     three_to_one_table[i] = i / 3;
 }
--- a/src/text.h	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/text.h	Sun Jan 12 11:08:22 2003 +0000
@@ -38,6 +38,12 @@
 #else
 size_t wcslen(const wchar_t *);
 #endif
+#ifndef HAVE_STRLWR
+char *strlwr(char *);
+#endif
+#ifndef HAVE_STRUPR
+char *strupr(char *);
+#endif
 
 /* ---------------------------------------------------------------------- */
 /*                     Super-basic character properties                   */
@@ -453,27 +459,27 @@
   (ptr) += rep_bytes_by_first_byte (* (ptr));	\
 } while (0)
 
-#define INC_IBYTEPTR_FMT(ptr, fmt)					\
-do {									\
-  Internal_Format __icf_fmt = (fmt);					\
-  switch (__icf_fmt)							\
-    {									\
-    case FORMAT_DEFAULT:						\
-      INC_IBYTEPTR (ptr);						\
-      break;								\
-    case FORMAT_16_BIT_FIXED:						\
-      text_checking_assert (ptr == ALIGN_PTR (ptr, UINT_16_BIT));	\
-      (ptr) += 2;							\
-      break;								\
-    case FORMAT_32_BIT_FIXED:						\
-      text_checking_assert (ptr == ALIGN_PTR (ptr, UINT_32_BIT));	\
-      (ptr) += 4;							\
-      break;								\
-    default:								\
-      text_checking_assert (fmt == FORMAT_8_BIT_FIXED);			\
-      (ptr)++;								\
-      break;								\
-    }									\
+#define INC_IBYTEPTR_FMT(ptr, fmt)					   \
+do {									   \
+  Internal_Format __icf_fmt = (fmt);					   \
+  switch (__icf_fmt)							   \
+    {									   \
+    case FORMAT_DEFAULT:						   \
+      INC_IBYTEPTR (ptr);						   \
+      break;								   \
+    case FORMAT_16_BIT_FIXED:						   \
+      text_checking_assert ((void *) ptr == ALIGN_PTR (ptr, UINT_16_BIT)); \
+      (ptr) += 2;							   \
+      break;								   \
+    case FORMAT_32_BIT_FIXED:						   \
+      text_checking_assert ((void *) ptr == ALIGN_PTR (ptr, UINT_32_BIT)); \
+      (ptr) += 4;							   \
+      break;								   \
+    default:								   \
+      text_checking_assert (fmt == FORMAT_8_BIT_FIXED);			   \
+      (ptr)++;								   \
+      break;								   \
+    }									   \
 } while (0)
 
 /* Given a itext (assumed to point at the beginning of a character or at
@@ -498,27 +504,27 @@
 } while (0)
 #endif /* ERROR_CHECK_TEXT */
 
-#define DEC_IBYTEPTR_FMT(ptr, fmt)					\
-do {									\
-  Internal_Format __icf_fmt = (fmt);					\
-  switch (__icf_fmt)							\
-    {									\
-    case FORMAT_DEFAULT:						\
-      DEC_IBYTEPTR (ptr);						\
-      break;								\
-    case FORMAT_16_BIT_FIXED:						\
-      text_checking_assert (ptr == ALIGN_PTR (ptr, UINT_16_BIT));	\
-      (ptr) -= 2;							\
-      break;								\
-    case FORMAT_32_BIT_FIXED:						\
-      text_checking_assert (ptr == ALIGN_PTR (ptr, UINT_32_BIT));	\
-      (ptr) -= 4;							\
-      break;								\
-    default:								\
-      text_checking_assert (fmt == FORMAT_8_BIT_FIXED);			\
-      (ptr)--;								\
-      break;								\
-    }									\
+#define DEC_IBYTEPTR_FMT(ptr, fmt)					   \
+do {									   \
+  Internal_Format __icf_fmt = (fmt);					   \
+  switch (__icf_fmt)							   \
+    {									   \
+    case FORMAT_DEFAULT:						   \
+      DEC_IBYTEPTR (ptr);						   \
+      break;								   \
+    case FORMAT_16_BIT_FIXED:						   \
+      text_checking_assert ((void *) ptr == ALIGN_PTR (ptr, UINT_16_BIT)); \
+      (ptr) -= 2;							   \
+      break;								   \
+    case FORMAT_32_BIT_FIXED:						   \
+      text_checking_assert ((void *) ptr == ALIGN_PTR (ptr, UINT_32_BIT)); \
+      (ptr) -= 4;							   \
+      break;								   \
+    default:								   \
+      text_checking_assert (fmt == FORMAT_8_BIT_FIXED);			   \
+      (ptr)--;								   \
+      break;								   \
+    }									   \
 } while (0)
 
 #ifdef MULE
@@ -675,10 +681,10 @@
     case FORMAT_DEFAULT:
       return bytecount_to_charcount (ptr, len);
     case FORMAT_16_BIT_FIXED:
-      text_checking_assert (ptr == ALIGN_PTR (ptr, UINT_16_BIT));
+      text_checking_assert ((void *) ptr == ALIGN_PTR (ptr, UINT_16_BIT));
       return (Charcount) (len << 1);
     case FORMAT_32_BIT_FIXED:
-      text_checking_assert (ptr == ALIGN_PTR (ptr, UINT_32_BIT));
+      text_checking_assert ((void *) ptr == ALIGN_PTR (ptr, UINT_32_BIT));
       return (Charcount) (len << 2);
     default:
       text_checking_assert (fmt == FORMAT_8_BIT_FIXED);
@@ -701,12 +707,12 @@
     case FORMAT_DEFAULT:
       return charcount_to_bytecount (ptr, len);
     case FORMAT_16_BIT_FIXED:
-      text_checking_assert (ptr == ALIGN_PTR (ptr, UINT_16_BIT));
+      text_checking_assert ((void *) ptr == ALIGN_PTR (ptr, UINT_16_BIT));
       text_checking_assert (!(len & 1));
       return (Bytecount) (len >> 1);
     case FORMAT_32_BIT_FIXED:
       text_checking_assert (!(len & 3));
-      text_checking_assert (ptr == ALIGN_PTR (ptr, UINT_32_BIT));
+      text_checking_assert ((void *) ptr == ALIGN_PTR (ptr, UINT_32_BIT));
       return (Bytecount) (len >> 2);
     default:
       text_checking_assert (fmt == FORMAT_8_BIT_FIXED);
@@ -749,10 +755,10 @@
     case FORMAT_DEFAULT:
       return itext_ichar_len (ptr);
     case FORMAT_16_BIT_FIXED:
-      text_checking_assert (ptr == ALIGN_PTR (ptr, UINT_16_BIT));
+      text_checking_assert ((void *) ptr == ALIGN_PTR (ptr, UINT_16_BIT));
       return 2;
     case FORMAT_32_BIT_FIXED:
-      text_checking_assert (ptr == ALIGN_PTR (ptr, UINT_32_BIT));
+      text_checking_assert ((void *) ptr == ALIGN_PTR (ptr, UINT_32_BIT));
       return 4;
     default:
       text_checking_assert (fmt == FORMAT_8_BIT_FIXED);
@@ -831,10 +837,10 @@
     case FORMAT_DEFAULT:
       return itext_ichar (ptr);
     case FORMAT_16_BIT_FIXED:
-      text_checking_assert (ptr == ALIGN_PTR (ptr, UINT_16_BIT));
+      text_checking_assert ((void *) ptr == ALIGN_PTR (ptr, UINT_16_BIT));
       return raw_16_bit_fixed_to_ichar (* (UINT_16_BIT *) ptr, object);
     case FORMAT_32_BIT_FIXED:
-      text_checking_assert (ptr == ALIGN_PTR (ptr, UINT_32_BIT));
+      text_checking_assert ((void *) ptr == ALIGN_PTR (ptr, UINT_32_BIT));
       return raw_32_bit_fixed_to_ichar (* (UINT_32_BIT *) ptr, object);
     default:
       text_checking_assert (fmt == FORMAT_8_BIT_FIXED);
@@ -861,10 +867,10 @@
     case FORMAT_DEFAULT:
       return (Ichar) *ptr;
     case FORMAT_16_BIT_FIXED:
-      text_checking_assert (ptr == ALIGN_PTR (ptr, UINT_16_BIT));
+      text_checking_assert ((void *) ptr == ALIGN_PTR (ptr, UINT_16_BIT));
       return raw_16_bit_fixed_to_ichar (* (UINT_16_BIT *) ptr, object);
     case FORMAT_32_BIT_FIXED:
-      text_checking_assert (ptr == ALIGN_PTR (ptr, UINT_32_BIT));
+      text_checking_assert ((void *) ptr == ALIGN_PTR (ptr, UINT_32_BIT));
       return raw_32_bit_fixed_to_ichar (* (UINT_32_BIT *) ptr, object);
     default:
       text_checking_assert (fmt == FORMAT_8_BIT_FIXED);
@@ -886,10 +892,10 @@
     case FORMAT_DEFAULT:
       return (Raw_Ichar) itext_ichar (ptr);
     case FORMAT_16_BIT_FIXED:
-      text_checking_assert (ptr == ALIGN_PTR (ptr, UINT_16_BIT));
+      text_checking_assert ((void *) ptr == ALIGN_PTR (ptr, UINT_16_BIT));
       return (Raw_Ichar) (* (UINT_16_BIT *) ptr);
     case FORMAT_32_BIT_FIXED:
-      text_checking_assert (ptr == ALIGN_PTR (ptr, UINT_32_BIT));
+      text_checking_assert ((void *) ptr == ALIGN_PTR (ptr, UINT_32_BIT));
       return (Raw_Ichar) (* (UINT_32_BIT *) ptr);
     default:
       text_checking_assert (fmt == FORMAT_8_BIT_FIXED);
@@ -928,11 +934,11 @@
       return set_itext_ichar (ptr, x);
     case FORMAT_16_BIT_FIXED:
       text_checking_assert (ichar_16_bit_fixed_p (x, object));
-      text_checking_assert (ptr == ALIGN_PTR (ptr, UINT_16_BIT));
+      text_checking_assert ((void *) ptr == ALIGN_PTR (ptr, UINT_16_BIT));
       * (UINT_16_BIT *) ptr = ichar_to_raw_16_bit_fixed (x, object);
       return 2;
     case FORMAT_32_BIT_FIXED:
-      text_checking_assert (ptr == ALIGN_PTR (ptr, UINT_32_BIT));
+      text_checking_assert ((void *) ptr == ALIGN_PTR (ptr, UINT_32_BIT));
       * (UINT_32_BIT *) ptr = ichar_to_raw_32_bit_fixed (x, object);
       return 4;
     default:
--- a/src/toolbar.c	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/toolbar.c	Sun Jan 12 11:08:22 2003 +0000
@@ -56,8 +56,7 @@
 
 Lisp_Object Qinit_toolbar_from_resources;
 
-#ifdef USE_KKCC
-static const struct lrecord_description toolbar_button_description [] = {
+static const struct memory_description toolbar_button_description [] = {
   { XD_LISP_OBJECT, offsetof (struct toolbar_button, next) },
   { XD_LISP_OBJECT, offsetof (struct toolbar_button, frame) },
   { XD_LISP_OBJECT, offsetof (struct toolbar_button, up_glyph) },
@@ -71,7 +70,6 @@
   { XD_LISP_OBJECT, offsetof (struct toolbar_button, help_string) },
   { XD_END }
 };
-#endif /* USE_KKCC */
 
 static Lisp_Object
 mark_toolbar_button (Lisp_Object obj)
@@ -90,17 +88,11 @@
   return data->help_string;
 }
 
-#ifdef USE_KKCC
 DEFINE_LRECORD_IMPLEMENTATION ("toolbar-button", toolbar_button,
 			       0, /*dumpable-flag*/
 			       mark_toolbar_button, 0, 0, 0, 0, 
 			       toolbar_button_description,
 			       struct toolbar_button);
-#else /* not USE_KKCC */
-DEFINE_LRECORD_IMPLEMENTATION ("toolbar-button", toolbar_button,
-			       mark_toolbar_button, 0, 0, 0, 0, 0,
-			       struct toolbar_button);
-#endif /* not USE_KKCC */
 
 DEFUN ("toolbar-button-p", Ftoolbar_button_p, 1, 1, 0, /*
 Return non-nil if OBJECT is a toolbar button.
@@ -247,7 +239,7 @@
   if (EQ (position, Qright))  return RIGHT_TOOLBAR;
   invalid_constant ("Invalid toolbar position", position);
 
-  RETURN_NOT_REACHED (TOP_TOOLBAR)
+  RETURN_NOT_REACHED (TOP_TOOLBAR);
 }
 
 DEFUN ("set-default-toolbar-position", Fset_default_toolbar_position, 1, 1, 0, /*
--- a/src/tooltalk.c	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/tooltalk.c	Sun Jan 12 11:08:22 2003 +0000
@@ -152,13 +152,11 @@
   Tt_message m;
 };
 
-#ifdef USE_KKCC
-static const struct lrecord_description tooltalk_message_description [] = {
+static const struct memory_description tooltalk_message_description [] = {
   { XD_LISP_OBJECT, offsetof (struct Lisp_Tooltalk_Message, callback) },
   { XD_LISP_OBJECT, offsetof (struct Lisp_Tooltalk_Message, plist_sym) },
   { XD_END }
 };
-#endif /* USE_KKCC */
 
 static Lisp_Object
 mark_tooltalk_message (Lisp_Object obj)
@@ -181,19 +179,12 @@
 		    (long) (p->m), p->header.uid);
 }
 
-#ifdef USE_KKCC
 DEFINE_LRECORD_IMPLEMENTATION ("tooltalk-message", tooltalk_message,
 			       0, /*dumpable-flag*/
                                mark_tooltalk_message, print_tooltalk_message,
                                0, 0, 0, 
 			       tooltalk_message_description,
 			       Lisp_Tooltalk_Message);
-#else /* not USE_KKCC */
-DEFINE_LRECORD_IMPLEMENTATION ("tooltalk-message", tooltalk_message,
-                               mark_tooltalk_message, print_tooltalk_message,
-                               0, 0, 0, 0,
-			       Lisp_Tooltalk_Message);
-#endif /* not USE_KKCC */
 
 static Lisp_Object
 make_tooltalk_message (Tt_message m)
@@ -239,13 +230,11 @@
   Tt_pattern p;
 };
 
-#ifdef USE_KKCC
-static const struct lrecord_description tooltalk_pattern_description [] = {
+static const struct memory_description tooltalk_pattern_description [] = {
   { XD_LISP_OBJECT, offsetof (struct Lisp_Tooltalk_Pattern, callback) },
   { XD_LISP_OBJECT, offsetof (struct Lisp_Tooltalk_Pattern, plist_sym) },
   { XD_END }
 };
-#endif /* USE_KKCC */
 
 static Lisp_Object
 mark_tooltalk_pattern (Lisp_Object obj)
@@ -268,19 +257,12 @@
 		    (long) (p->p), p->header.uid);
 }
 
-#ifdef USE_KKCC
 DEFINE_LRECORD_IMPLEMENTATION ("tooltalk-pattern", tooltalk_pattern,
 			       0, /*dumpable-flag*/
                                mark_tooltalk_pattern, print_tooltalk_pattern,
                                0, 0, 0, 
 			       tooltalk_pattern_description,
 			       Lisp_Tooltalk_Pattern);
-#else /* not USE_KKCC */
-DEFINE_LRECORD_IMPLEMENTATION ("tooltalk-pattern", tooltalk_pattern,
-                               mark_tooltalk_pattern, print_tooltalk_pattern,
-                               0, 0, 0, 0,
-			       Lisp_Tooltalk_Pattern);
-#endif /* not USE_KKCC */
 
 static Lisp_Object
 make_tooltalk_pattern (Tt_pattern p)
--- a/src/ui-gtk.c	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/ui-gtk.c	Sun Jan 12 11:08:22 2003 +0000
@@ -298,12 +298,10 @@
   return (data);
 }
 
-#ifdef USE_KKCC
-static const struct lrecord_description ffi_data_description [] = {
-  { XD_LISP_OBJECT, offsetof (struct emacs_ffi_data, function_name) }, 
+static const struct memory_description ffi_data_description [] = {
+  { XD_LISP_OBJECT, offsetof (emacs_ffi_data, function_name) }, 
   { XD_END }
 };
-#endif /* USE_KKCC */
 
 static Lisp_Object
 mark_ffi_data (Lisp_Object obj)
@@ -326,17 +324,11 @@
   write_fmt_string (printcharfun, " %p>", (void *)XFFI (obj)->function_ptr);
 }
 
-#ifdef USE_KKCC
 DEFINE_LRECORD_IMPLEMENTATION ("ffi", emacs_ffi,
 			       0, /*dumpable-flag*/
 			       mark_ffi_data, ffi_object_printer,
 			       0, 0, 0, 
 			       ffi_data_description, emacs_ffi_data);
-#else /* not USE_KKCC */
-DEFINE_LRECORD_IMPLEMENTATION ("ffi", emacs_ffi,
-			       mark_ffi_data, ffi_object_printer,
-			       0, 0, 0, NULL, emacs_ffi_data);
-#endif /* not USE_KKCC */
 
 typedef GtkObject * (*__OBJECT_fn) ();
 typedef gint (*__INT_fn) ();
@@ -900,12 +892,10 @@
   return (1);
 }
 
-#ifdef USE_KKCC
-static const struct lrecord_description gtk_object_data_description [] = {
-  { XD_LISP_OBJECT, offsetof (struct emacs_gtk_object_data, plist) }, 
+static const struct memory_description gtk_object_data_description [] = {
+  { XD_LISP_OBJECT, offsetof (emacs_gtk_object_data, plist) }, 
   { XD_END }
 };
-#endif /* USE_KKCC */
 
 static Lisp_Object
 mark_gtk_object_data (Lisp_Object obj)
@@ -933,34 +923,19 @@
     }
 }
 
-#ifdef USE_KKCC
 DEFINE_LRECORD_IMPLEMENTATION_WITH_PROPS ("GtkObject", emacs_gtk_object,
 					  0, /*dumpable-flag*/
-					  mark_gtk_object_data, /* marker function */
-					  emacs_gtk_object_printer, /* print function */
-					  emacs_gtk_object_finalizer, /* finalizer */
+					  mark_gtk_object_data,
+					  emacs_gtk_object_printer,
+					  emacs_gtk_object_finalizer,
 					  0, /* equality */
 					  0, /* hash */
-					  gtk_object_data_description, /* desc */
-					  object_getprop, /* get prop */
-					  object_putprop, /* put prop */
+					  gtk_object_data_description,
+					  object_getprop,
+					  object_putprop,
 					  0, /* rem prop */
 					  0, /* plist */
 					  emacs_gtk_object_data);
-#else /* not USE_KKCC */
-DEFINE_LRECORD_IMPLEMENTATION_WITH_PROPS ("GtkObject", emacs_gtk_object,
-					  mark_gtk_object_data, /* marker function */
-					  emacs_gtk_object_printer, /* print function */
-					  emacs_gtk_object_finalizer, /* finalizer */
-					  0, /* equality */
-					  0, /* hash */
-					  NULL, /* desc */
-					  object_getprop, /* get prop */
-					  object_putprop, /* put prop */
-					  0, /* rem prop */
-					  0, /* plist */
-					  emacs_gtk_object_data);
-#endif /* not USE_KKCC */
 
 static emacs_gtk_object_data *
 allocate_emacs_gtk_object_data (void)
@@ -1111,11 +1086,9 @@
 
 
 /* GTK_TYPE_BOXED wrapper for Emacs lisp */
-#ifdef USE_KKCC
-static const struct lrecord_description emacs_gtk_boxed_description [] = {
+static const struct memory_description emacs_gtk_boxed_description [] = {
   { XD_END }
 };
-#endif /* USE_KKCC */
 
 static void
 emacs_gtk_boxed_printer (Lisp_Object obj, Lisp_Object printcharfun, int escapeflag)
@@ -1145,34 +1118,19 @@
   return (HASH2 ((unsigned long)data->object, data->object_type));
 }
 
-#ifdef USE_KKCC
 DEFINE_LRECORD_IMPLEMENTATION_WITH_PROPS ("GtkBoxed", emacs_gtk_boxed,
 					  0, /*dumpable-flag*/
 					  0, /* marker function */
-					  emacs_gtk_boxed_printer, /* print function */
+					  emacs_gtk_boxed_printer,
 					  0, /* nuker */
-					  emacs_gtk_boxed_equality, /* equality */
-					  emacs_gtk_boxed_hash, /* hash */
-					  emacs_gtk_boxed_description, /* desc */
+					  emacs_gtk_boxed_equality,
+					  emacs_gtk_boxed_hash,
+					  emacs_gtk_boxed_description,
 					  0, /* get prop */
 					  0, /* put prop */
 					  0, /* rem prop */
 					  0, /* plist */
 					  emacs_gtk_boxed_data);
-#else /* not USE_KKCC */
-DEFINE_LRECORD_IMPLEMENTATION_WITH_PROPS ("GtkBoxed", emacs_gtk_boxed,
-					  0, /* marker function */
-					  emacs_gtk_boxed_printer, /* print function */
-					  0, /* nuker */
-					  emacs_gtk_boxed_equality, /* equality */
-					  emacs_gtk_boxed_hash, /* hash */
-					  NULL, /* desc */
-					  0, /* get prop */
-					  0, /* put prop */
-					  0, /* rem prop */
-					  0, /* plist */
-					  emacs_gtk_boxed_data);
-#endif /* not USE_KKCC */
 /* Currently defined GTK_TYPE_BOXED structures are:
 
    GtkAccelGroup -
--- a/src/unexnt.c	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/unexnt.c	Sun Jan 12 11:08:22 2003 +0000
@@ -1,5 +1,6 @@
 /* unexec for XEmacs on Windows NT.
    Copyright (C) 1994 Free Software Foundation, Inc.
+   Copyright (C) 2002 Ben Wing.
 
 This file is part of XEmacs.
 
@@ -108,16 +109,18 @@
    code supplied by NT (primarily because that code relies upon malloc ()).  */
 
 /* **********************
-   Hackers please remember, this _start() thingy is *not* called neither
-   when dumping portably, nor when running from temacs! Do not put
+   Hackers please remember, this _start() thingy is *not* called either
+   when dumping portably, or when running from temacs! Do not put
    significant XEmacs initialization here!
    ********************** */
 
-void
+EXTERN_C void mainCRTStartup (void);
+
+EXTERN_C int _start (void);
+
+int
 _start (void)
 {
-  extern void mainCRTStartup (void);
-
   /* Cache system info, e.g., the NT page size.  */
   cache_system_info ();
   /* Set OS type, so that tchar stuff below works */
@@ -172,6 +175,7 @@
 #endif
 
   mainCRTStartup ();
+  return 0; /* not reached? */
 }
 
 /* Dump out .data and .bss sections into a new executable.  */
@@ -190,13 +194,13 @@
      ".exe" extension...patch them up if they don't.  */
   qxestrcpy (in_filename, old_name);
   ptr = in_filename + qxestrlen (in_filename) - 4;
-  if (qxestrcmp (ptr, ".exe"))
-    qxestrcat (in_filename, ".exe");
+  if (qxestrcmp_c (ptr, ".exe"))
+    qxestrcat_c (in_filename, ".exe");
 
   qxestrcpy (out_filename, new_name);
   ptr = out_filename + qxestrlen (out_filename) - 4;
-  if (qxestrcmp (ptr, ".exe"))
-    qxestrcat (out_filename, ".exe");
+  if (qxestrcmp_c (ptr, ".exe"))
+    qxestrcat_c (out_filename, ".exe");
 
   stdout_out ("Dumping from %s\n", in_filename);
   stdout_out ("          to %s\n", out_filename);
@@ -359,7 +363,7 @@
   for (i = 0; i < nt_header->FileHeader.NumberOfSections; i++) 
     {
 #ifndef DUMP_SEPARATE_SECTION
-      if (!strcmp (section->Name, ".bss")) 
+      if (!qxestrcmp_c (section->Name, ".bss")) 
 	{
 	  extern int my_ebss;		/* From lastfile.c  */
 
@@ -369,9 +373,9 @@
 	  bss_size = (UChar_Binary*) &my_ebss - (UChar_Binary*) bss_start;
 	}
 
-      if (!strcmp (section->Name, ".data")) 
+      if (!qxestrcmp_c (section->Name, ".data")) 
 #else
-      if (!strcmp (section->Name, "xdata"))
+      if (!qxestrcmp_c (section->Name, "xdata"))
 #endif
 	{
 	  extern Char_Binary my_edata[];	/* From lastfile.c  */
--- a/src/unicode.c	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/unicode.c	Sun Jan 12 11:08:22 2003 +0000
@@ -173,108 +173,92 @@
 static short ***from_unicode_blank_3;
 static short ****from_unicode_blank_4;
 
-#if 0
-
-static const struct lrecord_description to_unicode_level_0_desc[] = {
+static const struct memory_description to_unicode_level_0_desc_1[] = {
   { XD_END }
 };
 
-static const struct struct_description to_unicode_level_0_ptr_desc = {
-  sizeof (int), to_unicode_level_0_desc
+static const struct sized_memory_description to_unicode_level_0_desc = {
+  sizeof (int), to_unicode_level_0_desc_1
 };
 
-static const struct lrecord_description to_unicode_level_1_desc[] = {
-  { XD_STRUCT_PTR, 0, 96, &to_unicode_level_0_ptr_desc },
+static const struct memory_description to_unicode_level_1_desc_1[] = {
+  { XD_STRUCT_PTR, 0, 96, &to_unicode_level_0_desc },
   { XD_END }
 };
 
-static const struct struct_description to_unicode_level_1_ptr_desc = {
-  0, to_unicode_level_1_desc
+static const struct sized_memory_description to_unicode_level_1_desc = {
+  sizeof (void *), to_unicode_level_1_desc_1
 };
 
-static const struct lrecord_description to_unicode_level_2_desc[] = {
-  { XD_STRUCT_PTR, 0, 96, &to_unicode_level_1_ptr_desc },
+static const struct memory_description to_unicode_description_1[] = {
+  { XD_STRUCT_PTR, 1, 96, &to_unicode_level_0_desc },
+  { XD_STRUCT_PTR, 2, 96, &to_unicode_level_1_desc },
   { XD_END }
 };
 
 /* Not static because each charset has a set of to and from tables and
    needs to describe them to pdump. */
-const struct struct_description to_unicode_description[] = {
-  { 1, to_unicode_level_1_desc },
-  { 2, to_unicode_level_2_desc },
+const struct sized_memory_description to_unicode_description = {
+  sizeof (void *), to_unicode_description_1
+};
+
+static const struct memory_description from_unicode_level_0_desc_1[] = {
   { XD_END }
 };
 
-static const struct lrecord_description from_unicode_level_0_desc[] = {
-  { XD_END }
+static const struct sized_memory_description from_unicode_level_0_desc = {
+   sizeof (short), from_unicode_level_0_desc_1
 };
 
-static const struct struct_description from_unicode_level_0_ptr_desc = {
-   sizeof (short), from_unicode_level_0_desc
-};
-
-static const struct lrecord_description from_unicode_level_1_desc[] = {
-  { XD_STRUCT_PTR, 0, 256, &from_unicode_level_0_ptr_desc },
+static const struct memory_description from_unicode_level_1_desc_1[] = {
+  { XD_STRUCT_PTR, 0, 256, &from_unicode_level_0_desc },
   { XD_END }
 };
 
-static const struct struct_description from_unicode_level_1_ptr_desc = {
-  0, from_unicode_level_1_desc
+static const struct sized_memory_description from_unicode_level_1_desc = {
+  sizeof (void *), from_unicode_level_1_desc_1
 };
 
-static const struct lrecord_description from_unicode_level_2_desc[] = {
-  { XD_STRUCT_PTR, 0, 256, &from_unicode_level_1_ptr_desc },
+static const struct memory_description from_unicode_level_2_desc_1[] = {
+  { XD_STRUCT_PTR, 0, 256, &from_unicode_level_1_desc },
   { XD_END }
 };
 
-static const struct struct_description from_unicode_level_2_ptr_desc = {
-  0, from_unicode_level_2_desc
+static const struct sized_memory_description from_unicode_level_2_desc = {
+  sizeof (void *), from_unicode_level_2_desc_1
 };
 
-static const struct lrecord_description from_unicode_level_3_desc[] = {
-  { XD_STRUCT_PTR, 0, 256, &from_unicode_level_2_ptr_desc },
+static const struct memory_description from_unicode_level_3_desc_1[] = {
+  { XD_STRUCT_PTR, 0, 256, &from_unicode_level_2_desc },
   { XD_END }
 };
 
-static const struct struct_description from_unicode_level_3_ptr_desc = {
-  0, from_unicode_level_3_desc
+static const struct sized_memory_description from_unicode_level_3_desc = {
+  sizeof (void *), from_unicode_level_3_desc_1
 };
 
-static const struct lrecord_description from_unicode_level_4_desc[] = {
-  { XD_STRUCT_PTR, 0, 256, &from_unicode_level_3_ptr_desc },
+static const struct memory_description from_unicode_description_1[] = {
+  { XD_STRUCT_PTR, 1, 256, &from_unicode_level_0_desc },
+  { XD_STRUCT_PTR, 2, 256, &from_unicode_level_1_desc },
+  { XD_STRUCT_PTR, 3, 256, &from_unicode_level_2_desc },
+  { XD_STRUCT_PTR, 4, 256, &from_unicode_level_3_desc },
   { XD_END }
 };
 
 /* Not static because each charset has a set of to and from tables and
    needs to describe them to pdump. */
-const struct struct_description from_unicode_description[] = {
-  { 1, from_unicode_level_1_desc },
-  { 2, from_unicode_level_2_desc },
-  { 3, from_unicode_level_3_desc },
-  { 4, from_unicode_level_4_desc },
-  { XD_END }
+const struct sized_memory_description from_unicode_description = {
+  sizeof (void *), from_unicode_description_1
 };
 
-#endif /* 0 */
-
 static Lisp_Object_dynarr *unicode_precedence_dynarr;
 
-static const struct lrecord_description lo_description_1[] = {
-  { XD_LISP_OBJECT, 0 },
+static const struct memory_description lod_description_1[] = {
+  XD_DYNARR_DESC (Lisp_Object_dynarr, &lisp_object_description),
   { XD_END }
 };
 
-static const struct struct_description lo_description = {
-  sizeof (Lisp_Object),
-  lo_description_1
-};
-
-static const struct lrecord_description lod_description_1[] = {
-  XD_DYNARR_DESC (Lisp_Object_dynarr, &lo_description),
-  { XD_END }
-};
-
-static const struct struct_description lisp_object_dynarr_description = {
+static const struct sized_memory_description lisp_object_dynarr_description = {
   sizeof (Lisp_Object_dynarr),
   lod_description_1
 };
@@ -1463,7 +1447,6 @@
 /************************************************************************/
 
 /* ISO 10646 UTF-16, UCS-4, UTF-8, UTF-7, etc. */
-DEFINE_CODING_SYSTEM_TYPE (unicode);
 
 enum unicode_type
 {
@@ -1504,10 +1487,12 @@
   int wrote_bom;
 };
 
-static const struct lrecord_description unicode_coding_system_description[] = {
+static const struct memory_description unicode_coding_system_description[] = {
   { XD_END }
 };
 
+DEFINE_CODING_SYSTEM_TYPE_WITH_DATA (unicode);
+
 /* Decode a UCS-2 or UCS-4 character into a buffer.  If the lookup fails, use
    <GETA MARK> (U+3013) of JIS X 0208, which means correct character
    is not found, instead.
@@ -2351,15 +2336,5 @@
   unicode_precedence_dynarr = Dynarr_new (Lisp_Object);
   dump_add_root_struct_ptr (&unicode_precedence_dynarr,
 			    &lisp_object_dynarr_description);
-#if 0
-  dump_add_root_thing (&to_unicode_blank_1, to_unicode_level_1_desc);
-  dump_add_root_thing (&to_unicode_blank_2, to_unicode_level_2_desc);
-
-  dump_add_root_thing (&from_unicode_blank_1, from_unicode_level_1_desc);
-  dump_add_root_thing (&from_unicode_blank_2, from_unicode_level_2_desc);
-  dump_add_root_thing (&from_unicode_blank_3, from_unicode_level_3_desc);
-  dump_add_root_thing (&from_unicode_blank_4, from_unicode_level_4_desc);
-#endif
-
 #endif /* MULE */
 }
--- a/src/window.c	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/window.c	Sun Jan 12 11:08:22 2003 +0000
@@ -147,10 +147,106 @@
 } while (0)
 
 
-#define MARK_DISP_VARIABLE(field)		\
-  mark_object (window->field[CURRENT_DISP]);	\
-  mark_object (window->field[DESIRED_DISP]);	\
-  mark_object (window->field[CMOTION_DISP]);
+
+static const struct memory_description int_description_1[] = {
+  { XD_END }
+};
+
+static const struct sized_memory_description int_description = {
+  sizeof (int),
+  int_description_1
+};
+
+static const struct memory_description int_dynarr_description_1[] = {
+  XD_DYNARR_DESC (int_dynarr, &int_description),
+  { XD_END }
+};
+
+static const struct sized_memory_description int_dynarr_description = {
+  sizeof (int_dynarr),
+  int_dynarr_description_1
+};
+
+static const struct memory_description face_cachel_description_1[] = {
+  { XD_STRUCT_PTR, offsetof (face_cachel, merged_faces),
+    1, &int_dynarr_description },
+  { XD_LISP_OBJECT, offsetof (face_cachel, face) },
+  { XD_LISP_OBJECT, offsetof (face_cachel, foreground) },
+  { XD_LISP_OBJECT, offsetof (face_cachel, background) },
+  { XD_LISP_OBJECT_ARRAY, offsetof (face_cachel, font), NUM_LEADING_BYTES },
+  { XD_LISP_OBJECT, offsetof (face_cachel, display_table) },
+  { XD_LISP_OBJECT, offsetof (face_cachel, background_pixmap) },
+  { XD_END }
+};
+
+static const struct sized_memory_description face_cachel_description = {
+  sizeof (face_cachel),
+  face_cachel_description_1
+};
+
+static const struct memory_description face_cachel_dynarr_description_1[] = {
+  XD_DYNARR_DESC (face_cachel_dynarr, &face_cachel_description),
+  { XD_END }
+};
+
+static const struct sized_memory_description face_cachel_dynarr_description = {
+  sizeof (face_cachel_dynarr),
+  face_cachel_dynarr_description_1
+};
+
+static const struct memory_description glyph_cachel_description_1[] = {
+  { XD_LISP_OBJECT, offsetof (glyph_cachel, glyph) },
+  { XD_END }
+};
+
+static const struct sized_memory_description glyph_cachel_description = {
+  sizeof (glyph_cachel),
+  glyph_cachel_description_1
+};
+
+static const struct memory_description glyph_cachel_dynarr_description_1[] = {
+  XD_DYNARR_DESC (glyph_cachel_dynarr, &glyph_cachel_description),
+  { XD_END }
+};
+
+static const struct sized_memory_description glyph_cachel_dynarr_description = {
+  sizeof (glyph_cachel_dynarr),
+  glyph_cachel_dynarr_description_1
+};
+
+static const struct memory_description line_start_cache_description_1[] = {
+  { XD_END }
+};
+
+static const struct sized_memory_description line_start_cache_description = {
+  sizeof (line_start_cache),
+  line_start_cache_description_1
+};
+
+static const struct memory_description line_start_cache_dynarr_description_1[] = {
+  XD_DYNARR_DESC (line_start_cache_dynarr, &line_start_cache_description),
+  { XD_END }
+};
+
+static const struct sized_memory_description line_start_cache_dynarr_description = {
+  sizeof (line_start_cache_dynarr),
+  line_start_cache_dynarr_description_1
+};
+
+static const struct memory_description window_description [] = {
+#define WINDOW_SLOT(slot) { XD_LISP_OBJECT, offsetof (struct window, slot) },
+#define WINDOW_SLOT_ARRAY(slot, size) \
+  { XD_LISP_OBJECT_ARRAY, offsetof (struct window, slot), size },
+#include "winslots.h"
+
+  { XD_STRUCT_PTR, offsetof (struct window, face_cachels),
+    1, &face_cachel_dynarr_description },
+  { XD_STRUCT_PTR, offsetof (struct window, glyph_cachels),
+    1, &glyph_cachel_dynarr_description },
+  { XD_STRUCT_PTR, offsetof (struct window, line_start_cache),
+    1, &line_start_cache_dynarr_description },
+  { XD_END }
+};
 
 static Lisp_Object
 mark_window (Lisp_Object obj)
@@ -225,16 +321,10 @@
   return make_lisp_hash_table (20, HASH_TABLE_KEY_WEAK, HASH_TABLE_EQ);
 }
 
-#ifdef USE_KKCC
 DEFINE_LRECORD_IMPLEMENTATION ("window", window,
 			       0, /*dumpable-flag*/
                                mark_window, print_window, finalize_window,
-			       0, 0, 0, struct window);
-#else /* not USE_KKCC */
-DEFINE_LRECORD_IMPLEMENTATION ("window", window,
-                               mark_window, print_window, finalize_window,
-			       0, 0, 0, struct window);
-#endif /* not USE_KKCC */
+			       0, 0, window_description, struct window);
 
 #define INIT_DISP_VARIABLE(field, initialization)	\
   p->field[CURRENT_DISP] = initialization;		\
@@ -253,13 +343,8 @@
 Lisp_Object
 allocate_window (void)
 {
-  Lisp_Object val;
   struct window *p = alloc_lcrecord_type (struct window, &lrecord_window);
-
-  zero_lcrecord (p);
-  val = wrap_window (p);
-
-  p->dead = 0;
+  Lisp_Object val = wrap_window (p);
 
 #define WINDOW_SLOT(slot) p->slot = Qnil;
 #include "winslots.h"
@@ -280,14 +365,6 @@
   p->subwindow_instance_cache = make_image_instance_cache_hash_table ();
 
   p->line_cache_last_updated = Qzero;
-  INIT_DISP_VARIABLE (last_point_x, 0);
-  INIT_DISP_VARIABLE (last_point_y, 0);
-  INIT_DISP_VARIABLE (window_end_pos, 0);
-
-  p->gutter_extent_modiff[0] = 0;
-  p->gutter_extent_modiff[1] = 0;
-  p->gutter_extent_modiff[2] = 0;
-  p->gutter_extent_modiff[3] = 0;
 
   p->windows_changed = 1;
   p->shadow_thickness_changed = 1;
@@ -336,11 +413,28 @@
    because neither structure is created very often (only when windows are
    created or deleted). --ben */
 
-#ifdef USE_KKCC
-static const struct lrecord_description window_mirror_description [] = {
+static const struct memory_description window_mirror_description [] = {
+  { XD_LISP_OBJECT, offsetof (struct window_mirror, frame) },
+  { XD_LISP_OBJECT, offsetof (struct window_mirror, next) },
+  { XD_LISP_OBJECT, offsetof (struct window_mirror, hchild) },
+  { XD_LISP_OBJECT, offsetof (struct window_mirror, vchild) },
+
+  { XD_STRUCT_PTR, offsetof (struct window_mirror, current_display_lines),
+    1, &display_line_dynarr_description },
+  { XD_STRUCT_PTR, offsetof (struct window_mirror, desired_display_lines),
+    1, &display_line_dynarr_description },
+
+  { XD_LISP_OBJECT, offsetof (struct window_mirror, buffer) },
+
+#ifdef HAVE_SCROLLBARS
+  { XD_LISP_OBJECT, offsetof (struct window_mirror,
+			      scrollbar_vertical_instance) },
+  { XD_LISP_OBJECT, offsetof (struct window_mirror,
+			      scrollbar_horizontal_instance) },
+#endif /* HAVE_SCROLLBARS */
+
   { XD_END }
 };
-#endif /* USE_KKCC */
 
 static Lisp_Object
 mark_window_mirror (Lisp_Object obj)
@@ -374,16 +468,11 @@
     return Qnil;
 }
 
-#ifdef USE_KKCC
 DEFINE_LRECORD_IMPLEMENTATION ("window-mirror", window_mirror,
 			       0, /*dumpable-flag*/
                                mark_window_mirror, internal_object_printer,
-			       0, 0, 0, 0/*window_mirror_description*/, struct window_mirror);
-#else /* not USE_KKCC */
-DEFINE_LRECORD_IMPLEMENTATION ("window-mirror", window_mirror,
-                               mark_window_mirror, internal_object_printer,
-			       0, 0, 0, 0, struct window_mirror);
-#endif /* not USE_KKCC */
+			       0, 0, 0, window_mirror_description,
+			       struct window_mirror);
 
 /* Create a new window mirror structure and associated redisplay
    structs. */
@@ -392,7 +481,6 @@
 {
   struct window_mirror *t =
     alloc_lcrecord_type (struct window_mirror, &lrecord_window_mirror);
-  zero_lcrecord (t);
 
   t->frame = f;
   t->current_display_lines = Dynarr_new (display_line);
@@ -420,7 +508,7 @@
     if (!mir)
       mir = new_window_mirror (XFRAME (XWINDOW (win)->frame));
 
-  mir->next   = update_mirror_internal (XWINDOW (win)->next,   mir->next);
+  mir->next   = update_mirror_internal (XWINDOW (win)->next, mir->next);
   mir->hchild = update_mirror_internal (XWINDOW (win)->hchild, mir->hchild);
   mir->vchild = update_mirror_internal (XWINDOW (win)->vchild, mir->vchild);
 
@@ -1540,7 +1628,7 @@
     needed += ((hlimit - height)*(nelt - elt) + height-1)/height + 3;
   }
 
-  RETURN_NOT_REACHED(make_int (0))        /* shut up compiler */
+  RETURN_NOT_REACHED(make_int (0));        /* shut up compiler */
 }
 
 DEFUN ("window-width", Fwindow_width, 0, 1, 0, /*
--- a/src/winslots.h	Sun Jan 12 11:04:30 2003 +0000
+++ b/src/winslots.h	Sun Jan 12 11:08:22 2003 +0000
@@ -42,34 +42,45 @@
      either EQ or EQUAL_WRAPPED (i.e. Feq() or Fequal()).
    WINDOW_SAVED_SLOT_ARRAY is the same for an array of Lisp_Objects.
 
-   Callers should define WINDOW_SLOT (with a terminating semicolon if
-   not blank), and WINDOW_SAVED_SLOT if different; otherwise the
-   latter will be defined using WINDOW_SLOT.  Callers do not define
-   the _ARRAY versions.  Instead, they either do or do not define
-   WINDOW_SLOT_DECLARATION.  It should be defined in the definition of
-   a struct and not elsewhere.
+   Callers should define WINDOW_SLOT (with a terminating semicolon if not
+   blank), and WINDOW_SAVED_SLOT if different; otherwise the latter will be
+   defined using WINDOW_SLOT.  Callers should also either (a) do nothing
+   else (which defines WINDOW_SLOT_ARRAY using a for() loop, appropriate
+   for normal code), define WINDOW_SLOT_DECLARATION (which defines
+   WINDOW_SLOT_ARRAY using WINDOW_SLOT (slot[size]), appropriate for a
+   struct definition), or define WINDOW_SLOT_ARRAY themselves.  In the
+   first two cases, WINDOW_SAVED_SLOT_ARRAY will be defined in the same
+   fashion, using WINDOW_SAVED_SLOT.  In the last case, if
+   WINDOW_SAVED_SLOT is defined, the caller must provide an appropriate
+   definition of WINDOW_SAVED_SLOT_ARRAY; otherwise, it will be defined
+   using WINDOW_SLOT_ARRAY.
 
    Callers do not need to undefine these definitions; it is done
    automatically.
 */
 
-#ifndef WINDOW_SAVED_SLOT
-#define WINDOW_SAVED_SLOT(slot, compare) WINDOW_SLOT (slot)
-#endif
-
-#ifdef WINDOW_SLOT_DECLARATION
-#define WINDOW_SLOT_ARRAY(slot, size) WINDOW_SLOT (slot[size])
-#define WINDOW_SAVED_SLOT_ARRAY(slot, size, compare) \
+#ifdef WINDOW_SLOT_ARRAY
+# ifndef WINDOW_SAVED_SLOT_ARRAY
+#  ifdef WINDOW_SAVED_SLOT
+#   error must define WINDOW_SAVED_SLOT_ARRAY if WINDOW_SAVED_SLOT and WINDOW_SLOT_ARRAY are defined
+#  else
+#   define WINDOW_SAVED_SLOT_ARRAY(slot, size, compare) \
+     WINDOW_SLOT_ARRAY (slot, size)
+#  endif /* WINDOW_SAVED_SLOT */
+# endif /* not WINDOW_SAVED_SLOT_ARRAY */
+#elif defined (WINDOW_SLOT_DECLARATION) /* not WINDOW_SLOT_ARRAY */
+# define WINDOW_SLOT_ARRAY(slot, size) WINDOW_SLOT (slot[size])
+# define WINDOW_SAVED_SLOT_ARRAY(slot, size, compare) \
   WINDOW_SAVED_SLOT (slot[size], compare)
-#else
-#define WINDOW_SLOT_ARRAY(slot, size) do {	\
+#else /* not WINDOW_SLOT_DECLARATION, not WINDOW_SLOT_ARRAY */
+# define WINDOW_SLOT_ARRAY(slot, size) do {	\
   int wsaidx;					\
   for (wsaidx = 0; wsaidx < size; wsaidx++)	\
     {						\
       WINDOW_SLOT (slot[wsaidx]);		\
     }						\
 } while (0);
-#define WINDOW_SAVED_SLOT_ARRAY(slot, size, compare) do {	\
+# define WINDOW_SAVED_SLOT_ARRAY(slot, size, compare) do {	\
   int wsaidx;							\
   for (wsaidx = 0; wsaidx < size; wsaidx++)			\
     {								\
@@ -78,6 +89,10 @@
 } while (0);
 #endif /* WINDOW_SLOT_DECLARATION */
 
+#ifndef WINDOW_SAVED_SLOT
+#define WINDOW_SAVED_SLOT(slot, compare) WINDOW_SLOT (slot)
+#endif
+
 #define EQUAL_WRAPPED(x,y) internal_equal ((x), (y), 0)